-
Notifications
You must be signed in to change notification settings - Fork 10
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add webresource vocab & begin/commitUpload actions
Change-type: minor
- Loading branch information
1 parent
a23172f
commit 4e33c2c
Showing
5 changed files
with
573 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,275 @@ | ||
import type { WebResourceType as WebResource } from '@balena/sbvr-types'; | ||
import { randomUUID } from 'node:crypto'; | ||
import type { AnyObject } from 'pinejs-client-core'; | ||
import type { | ||
BeginMultipartUploadPayload, | ||
UploadPart, | ||
WebResourceHandler, | ||
} from '.'; | ||
import { getWebResourceFields } from '.'; | ||
import type { PinejsClient } from '../sbvr-api/sbvr-utils'; | ||
import { api } from '../sbvr-api/sbvr-utils'; | ||
import type { ODataRequest } from '../sbvr-api/uri-parser'; | ||
import { errors, sbvrUtils } from '../server-glue/module'; | ||
|
||
type BeginUploadDbCheck = BeginMultipartUploadPayload & WebResource; | ||
|
||
export interface PendingUpload extends BeginMultipartUploadPayload { | ||
fieldName: string; | ||
fileKey: string; | ||
uploadId: string; | ||
} | ||
|
||
export interface BeginUploadResponse { | ||
[fieldName: string]: { | ||
uuid: string; | ||
uploadParts: UploadPart[]; | ||
}; | ||
} | ||
|
||
const MB = 1024 * 1024; | ||
|
||
export const multipartUploadHooks = ( | ||
webResourceHandler: WebResourceHandler, | ||
): sbvrUtils.Hooks => { | ||
return { | ||
POSTPARSE: async ({ req, request, tx, api: applicationApi }) => { | ||
if (request.odataQuery.property?.resource === 'beginUpload') { | ||
const uploadParams = await validateBeginUpload(request, applicationApi); | ||
|
||
// This transaction is necessary because beginUpload requests | ||
// will rollback the transaction (in order to first validate) | ||
// The metadata requested. If we don't pass any transaction | ||
// It will use the default transaction handler which will error out | ||
// on any rollback. | ||
tx = await sbvrUtils.db.transaction(); | ||
req.tx = tx; | ||
request.tx = tx; | ||
|
||
request.method = 'PATCH'; | ||
request.values = uploadParams; | ||
request.odataQuery.resource = request.resourceName; | ||
delete request.odataQuery.property; | ||
request.custom.isAction = 'beginUpload'; | ||
} else if (request.odataQuery.property?.resource === 'commitUpload') { | ||
const commitPayload = await validateCommitUpload( | ||
request, | ||
applicationApi, | ||
); | ||
|
||
const webresource = await webResourceHandler.commitMultipartUpload({ | ||
fileKey: commitPayload.metadata.fileKey, | ||
uploadId: commitPayload.metadata.uploadId, | ||
filename: commitPayload.metadata.filename, | ||
providerCommitData: commitPayload.providerCommitData, | ||
}); | ||
|
||
await api.webresource.patch({ | ||
resource: 'multipart_upload', | ||
body: { | ||
status: 'completed', | ||
}, | ||
options: { | ||
$filter: { | ||
uuid: commitPayload.uuid, | ||
}, | ||
}, | ||
passthrough: { | ||
tx: tx, | ||
}, | ||
}); | ||
|
||
request.method = 'PATCH'; | ||
request.values = { | ||
[commitPayload.metadata.fieldName]: webresource, | ||
}; | ||
request.odataQuery.resource = request.resourceName; | ||
delete request.odataQuery.property; | ||
request.custom.isAction = 'commitUpload'; | ||
request.custom.commitUploadPayload = webresource; | ||
} | ||
}, | ||
PRERESPOND: async ({ req, request, response, tx }) => { | ||
if (request.custom.isAction === 'beginUpload') { | ||
// In the case where the transaction has failed because it had invalid payload | ||
// such as breaking a db constraint, this hook wouldn't have been called | ||
// and would rather throw with the rule it failed to validate | ||
// We rollback here as the patch was just a way to validate the upload payload | ||
await tx.rollback(); | ||
|
||
response.statusCode = 200; | ||
response.body = await beginUpload( | ||
webResourceHandler, | ||
request, | ||
req.user?.actor, | ||
); | ||
} else if (request.custom.isAction === 'commitUpload') { | ||
response.body = await webResourceHandler.onPreRespond( | ||
request.custom.commitUploadPayload, | ||
); | ||
} | ||
}, | ||
}; | ||
}; | ||
|
||
export const beginUpload = async ( | ||
webResourceHandler: WebResourceHandler, | ||
odataRequest: ODataRequest, | ||
actorId?: number, | ||
): Promise<BeginUploadResponse> => { | ||
const payload = odataRequest.values as { | ||
[x: string]: BeginMultipartUploadPayload; | ||
}; | ||
const fieldName = Object.keys(payload)[0]; | ||
const metadata = payload[fieldName]; | ||
|
||
const { fileKey, uploadId, uploadParts } = | ||
await webResourceHandler.beginMultipartUpload(fieldName, metadata); | ||
const uuid = randomUUID(); | ||
|
||
try { | ||
await api.webresource.post({ | ||
resource: 'multipart_upload', | ||
body: { | ||
uuid, | ||
resource_name: odataRequest.resourceName, | ||
field_name: fieldName, | ||
resource_id: odataRequest.affectedIds?.[0], | ||
upload_id: uploadId, | ||
file_key: fileKey, | ||
status: 'pending', | ||
filename: metadata.filename, | ||
content_type: metadata.content_type, | ||
size: metadata.size, | ||
chunk_size: metadata.chunk_size, | ||
expiry_date: Date.now() + 7 * 24 * 60 * 60 * 1000, // 7 days in ms | ||
is_created_by__actor: actorId, | ||
}, | ||
}); | ||
} catch (err) { | ||
console.error('failed to start multipart upload', err); | ||
throw new errors.BadRequestError('Failed to start multipart upload'); | ||
} | ||
|
||
return { [fieldName]: { uuid, uploadParts } }; | ||
}; | ||
|
||
const validateBeginUpload = async ( | ||
request: ODataRequest, | ||
applicationApi: PinejsClient, | ||
) => { | ||
if (request.odataQuery.key == null) { | ||
throw new errors.BadRequestError(); | ||
} | ||
|
||
await applicationApi.post({ | ||
url: request.url.substring(1).replace('beginUpload', 'canAccess'), | ||
body: { method: 'PATCH' }, | ||
}); | ||
|
||
const fieldNames = Object.keys(request.values); | ||
if (fieldNames.length !== 1) { | ||
throw new errors.BadRequestError( | ||
'You can only get upload url for one field at a time', | ||
); | ||
} | ||
|
||
const [fieldName] = fieldNames; | ||
const webResourceFields = getWebResourceFields(request, false); | ||
if (!webResourceFields.includes(fieldName)) { | ||
throw new errors.BadRequestError( | ||
`You must provide a valid webresource field from: ${JSON.stringify(webResourceFields)}`, | ||
); | ||
} | ||
|
||
const beginUploadPayload = parseBeginUploadPayload(request.values[fieldName]); | ||
if (beginUploadPayload == null) { | ||
throw new errors.BadRequestError('Invalid file metadata'); | ||
} | ||
|
||
const uploadMetadataCheck: BeginUploadDbCheck = { | ||
...beginUploadPayload, | ||
href: 'metadata_check', | ||
}; | ||
|
||
return { [fieldName]: uploadMetadataCheck }; | ||
}; | ||
|
||
const parseBeginUploadPayload = ( | ||
payload: AnyObject, | ||
): BeginMultipartUploadPayload | null => { | ||
if (typeof payload !== 'object') { | ||
return null; | ||
} | ||
|
||
let { filename, content_type, size, chunk_size } = payload; | ||
if ( | ||
typeof filename !== 'string' || | ||
typeof content_type !== 'string' || | ||
typeof size !== 'number' || | ||
(chunk_size != null && typeof chunk_size !== 'number') || | ||
(chunk_size != null && chunk_size < 5 * MB) | ||
) { | ||
return null; | ||
} | ||
|
||
if (chunk_size == null) { | ||
chunk_size = 5 * MB; | ||
} | ||
return { filename, content_type, size, chunk_size }; | ||
}; | ||
|
||
const validateCommitUpload = async ( | ||
request: ODataRequest, | ||
applicationApi: PinejsClient, | ||
) => { | ||
if (request.odataQuery.key == null) { | ||
throw new errors.BadRequestError(); | ||
} | ||
|
||
await applicationApi.post({ | ||
url: request.url.substring(1).replace('commitUpload', 'canAccess'), | ||
body: { method: 'PATCH' }, | ||
}); | ||
|
||
const { uuid, providerCommitData } = request.values; | ||
if (typeof uuid !== 'string') { | ||
throw new errors.BadRequestError('Invalid uuid type'); | ||
} | ||
|
||
const [multipartUpload] = (await api.webresource.get({ | ||
resource: 'multipart_upload', | ||
options: { | ||
$select: ['id', 'file_key', 'upload_id', 'field_name', 'filename'], | ||
$filter: { | ||
uuid, | ||
status: 'pending', | ||
expiry_date: { $gt: { $now: {} } }, | ||
}, | ||
}, | ||
passthrough: { | ||
tx: request.tx, | ||
}, | ||
})) as [ | ||
{ | ||
id: number; | ||
file_key: string; | ||
upload_id: string; | ||
field_name: string; | ||
filename: string; | ||
}?, | ||
]; | ||
|
||
if (multipartUpload == null) { | ||
throw new errors.BadRequestError(`Invalid upload for uuid ${uuid}`); | ||
} | ||
|
||
const metadata = { | ||
fileKey: multipartUpload.file_key, | ||
uploadId: multipartUpload.upload_id, | ||
filename: multipartUpload.filename, | ||
fieldName: multipartUpload.field_name, | ||
}; | ||
|
||
return { uuid, providerCommitData, metadata }; | ||
}; |
Oops, something went wrong.