Skip to content

Commit

Permalink
Merge pull request #290 from bcgov/upload
Browse files Browse the repository at this point in the history
Sync versions object during object create
  • Loading branch information
TimCsaky authored Dec 9, 2024
2 parents 30833d7 + 2af2fab commit 4b8f276
Show file tree
Hide file tree
Showing 2 changed files with 60 additions and 55 deletions.
105 changes: 51 additions & 54 deletions app/src/controllers/object.js
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ const {
metadataService,
objectService,
storageService,
syncService,
tagService,
userService,
versionService,
Expand Down Expand Up @@ -297,20 +298,31 @@ const controller = {
bucketId: bucketId
});

// sync existing file versions and get COMS objectId
const objectId =
await syncService.syncJob(joinPath(bucketKey, req.currentUpload.filename), bucketId, true, userId);

// Hard short circuit skip file as the object already exists on bucket
throw new Problem(409, {
throw new Problem(409, 'Bucket already contains object', req.originalUrl, {
detail: 'Bucket already contains object',
instance: req.originalUrl
existingObjectId: objectId,
});

} catch (err) {
if (err instanceof Problem) throw err; // Rethrow Problem type errors

// Object is soft deleted from the bucket
if (err.$response?.headers['x-amz-delete-marker']) {
throw new Problem(409, {

// sync existing file versions and get COMS objectId
const objectId =
await syncService.syncJob(joinPath(bucketKey, req.currentUpload.filename), bucketId, true, userId);

throw new Problem(409, 'Bucket already contains object', req.originalUrl, {
detail: 'Bucket already contains object',
instance: req.originalUrl
existingObjectId: objectId,
});

}

// Skip upload in the unlikely event we get an unexpected error from headObject
Expand Down Expand Up @@ -1032,67 +1044,52 @@ const controller = {

let s3Response;
try {
// Preflight S3 Object check
const headResponse = await storageService.headObject({
// Preflight check for object in bucket
const vs = await storageService.listObjectVersion({
filePath: joinPath(bucketKey, filename),
bucketId: bucketId
});

// Skip upload in the unlikely event we get an unexpected response from headObject
if (headResponse.$metadata?.httpStatusCode !== 200) {
throw new Problem(502, {
detail: 'Bucket communication error',
instance: req.originalUrl
});
}

// Object exists on bucket
if (req.currentUpload.contentLength < MAXCOPYOBJECTLENGTH) {
log.debug('Uploading with putObject', {
contentLength: req.currentUpload.contentLength,
function: 'createObject',
uploadMethod: 'putObject'
});
s3Response = await storageService.putObject({ ...data, stream: req });
} else if (req.currentUpload.contentLength < MAXFILEOBJECTLENGTH) {
log.debug('Uploading with lib-storage', {
contentLength: req.currentUpload.contentLength,
function: 'createObject',
uploadMethod: 'lib-storage'
});
s3Response = await storageService.upload({ ...data, stream: req });
} else {
throw new Problem(413, {
detail: 'File exceeds maximum 50GB limit',
instance: req.originalUrl
});
}
} catch (err) {
if (err instanceof Problem) throw err; // Rethrow Problem type errors
else if (err.$metadata?.httpStatusCode !== 404) {
// An unexpected response from headObject
throw new Problem(502, {
detail: 'Bucket communication error',
instance: req.originalUrl
});
} else {
if (err.$response?.headers['x-amz-delete-marker']) {
// Object is soft deleted from the bucket
throw new Problem(409, {
detail: 'Unable to update soft deleted object',
instance: req.originalUrl
// if object exists
if (Array.isArray(vs.Versions)) {
// if smaller file use putObject
if (req.currentUpload.contentLength < MAXCOPYOBJECTLENGTH) {
log.debug('Uploading with putObject', {
contentLength: req.currentUpload.contentLength,
function: 'createObject',
uploadMethod: 'putObject'
});
s3Response = await storageService.putObject({ ...data, stream: req });
}
// else use upload command
else if (req.currentUpload.contentLength < MAXFILEOBJECTLENGTH) {
log.debug('Uploading with lib-storage', {
contentLength: req.currentUpload.contentLength,
function: 'createObject',
uploadMethod: 'lib-storage'
});
s3Response = await storageService.upload({ ...data, stream: req });
} else {
// Bucket is missing the existing object
throw new Problem(409, {
detail: 'Bucket does not contain existing object',
throw new Problem(413, {
detail: 'File exceeds maximum 50GB limit',
instance: req.originalUrl
});
}
// TODO: Add in sync operation to update object record in COMS DB?
}
// else no file in bucket
else {
throw new Problem(409, {
detail: 'Bucket does not contain existing object',
instance: req.originalUrl
});
}
} catch (error) {
throw new Problem(502, {
detail: 'Bucket communication error',
instance: req.originalUrl
});
}

// do COMS database updates:
const s3Head = await storageService.headObject({
filePath: joinPath(bucketKey, req.currentUpload.filename),
bucketId: bucketId
Expand Down
10 changes: 9 additions & 1 deletion app/src/docs/v1.api-spec.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -374,7 +374,15 @@ paths:
content:
application/json:
schema:
$ref: "#/components/schemas/Response-Conflict"
allOf:
- $ref: "#/components/schemas/Response-Conflict"
- type: object
properties:
existingObjectId:
type: string
format: uuid
description: The ID of the existing object
example: ac246e31-c807-496c-bc93-cd8bc2f1b2b4
"411":
$ref: "#/components/responses/LengthRequired"
"413":
Expand Down

0 comments on commit 4b8f276

Please sign in to comment.