UNPKG

@cdkx/web-application

Version:

Static web application hosting related constructs

154 lines 19.8 kB
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; (function (factory) { if (typeof module === "object" && typeof module.exports === "object") { var v = factory(require, exports); if (v !== undefined) module.exports = v; } else if (typeof define === "function" && define.amd) { define("@cdkx/web-application/handlers/assets-uploader", ["require", "exports", "busboy", "@cdkx/web-application/handlers/base-handler", "change-case"], factory); } })(function (require, exports) { "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.AssetsUploader = void 0; const busboy_1 = __importDefault(require("busboy")); const base_handler_1 = require("@cdkx/web-application/handlers/base-handler"); const change_case_1 = require("change-case"); /** * Upload files to s3 or return upload Url * @params returnSignedUrlOnly: boolean @default false */ class AssetsUploader extends base_handler_1.BaseHandler { constructor(s3) { super(); this.s3 = s3; } run(event) { var _a, _b, _c; return __awaiter(this, void 0, void 0, function* () { const bucketName = process.env.BUCKET_NAME; if (!bucketName) { return this.encodedResponse({ statusCode: 500, body: { message: 'Missing required environment variable: BUCKET_NAME' }, }); } if (!event.body) { return this.encodedResponse({ statusCode: 400, body: { message: 'Cannot upload process request with empty body' }, }); } try { const { parsedBody } = yield this.parseFormData(event); if (!parsedBody) { throw new Error('No body after parsing.'); } if (!((_a = parsedBody.fields) === null || _a === void 0 ? void 0 : _a.s3Prefix)) { return this.encodedResponse({ statusCode: 400, body: { message: `Missing required property: s3Prefix` }, }); } const s3UploadPromises = []; for (const fileKey in parsedBody === null || parsedBody === void 0 ? void 0 : parsedBody.files) { const currentFile = parsedBody === null || parsedBody === void 0 ? void 0 : parsedBody.files[fileKey]; if (!(currentFile === null || currentFile === void 0 ? void 0 : currentFile.data)) { return this.encodedResponse({ statusCode: 400, body: `Unable to parse binary ${currentFile.fileName}`, }); } const filePublicName = (_b = currentFile.fileName) !== null && _b !== void 0 ? _b : currentFile.fieldName; console.log('Uploading File: ', filePublicName); const [name, ext] = filePublicName.split('.'); // upload each file to s3 s3UploadPromises.push(this.s3 .upload({ Bucket: bucketName, Key: `${parsedBody.fields.s3Prefix}/${[change_case_1.snakeCase(name), ext].join('.')}`, ContentType: currentFile.mimeType, Body: currentFile.data, ContentEncoding: currentFile.encoding, }) .promise()); } const response = yield Promise.all(s3UploadPromises); const responseToReturn = response.map((returnedItem) => ({ location: returnedItem.Location, key: returnedItem.Key, bucket: returnedItem.Bucket, eTag: returnedItem.ETag, })); return this.encodedResponse({ statusCode: 200, body: responseToReturn.length > 1 ? responseToReturn : responseToReturn[0], }); } catch (err) { console.log('Error processing request: ', err); return this.encodedResponse({ statusCode: 500, body: { message: (_c = err.message) !== null && _c !== void 0 ? _c : 'Something went wrong!' }, }); } }); } parseFormData(event) { const busboy = new busboy_1.default({ headers: { 'content-type': event.headers['Content-Type'] || event.headers['content-type'], }, }); const result = { files: {}, fields: {}, }; return new Promise((resolve, reject) => { busboy .on('file', (fieldName, file, fileName, encoding, mimeType) => { let fileData; file.on('data', (data) => { fileData = data; }); file.on('end', () => { result.files[fieldName] = { data: fileData, fileName, fieldName, encoding, mimeType, isBinary: true, }; }); }) .on('field', (fieldName, value) => { result.fields[fieldName] = value; }) .on('error', (err) => { console.error(err); reject(err); }) .on('finish', () => { event.parsedBody = result; resolve(event); }); busboy.write(event.body || '', event.isBase64Encoded ? 'base64' : 'binary'); busboy.end(); }); } } exports.AssetsUploader = AssetsUploader; }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assets-uploader.js","sourceRoot":"","sources":["../../../../../../packages/web-application/handlers/assets-uploader.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;IAEA,oDAA4B;IAC5B,+EAA6C;IAC7C,6CAAwC;IASxC;;;OAGG;IACH,MAAa,cAAe,SAAQ,0BAAW;QAC7C,YAAoB,EAAM;YACxB,KAAK,EAAE,CAAC;YADU,OAAE,GAAF,EAAE,CAAI;QAE1B,CAAC;QACK,GAAG,CACP,KAAsB;;;gBAOtB,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC,WAAW,CAAC;gBAE3C,IAAI,CAAC,UAAU,EAAE;oBACf,OAAO,IAAI,CAAC,eAAe,CAAC;wBAC1B,UAAU,EAAE,GAAG;wBACf,IAAI,EAAE,EAAE,OAAO,EAAE,oDAAoD,EAAE;qBACxE,CAAC,CAAC;iBACJ;gBAED,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE;oBACf,OAAO,IAAI,CAAC,eAAe,CAAC;wBAC1B,UAAU,EAAE,GAAG;wBACf,IAAI,EAAE,EAAE,OAAO,EAAE,+CAA+C,EAAE;qBACnE,CAAC,CAAC;iBACJ;gBAED,IAAI;oBACF,MAAM,EAAE,UAAU,EAAE,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;oBAEvD,IAAI,CAAC,UAAU,EAAE;wBACf,MAAM,IAAI,KAAK,CAAC,wBAAwB,CAAC,CAAC;qBAC3C;oBAED,IAAI,CAAC,CAAA,MAAA,UAAU,CAAC,MAAM,0CAAE,QAAQ,CAAA,EAAE;wBAChC,OAAO,IAAI,CAAC,eAAe,CAAC;4BAC1B,UAAU,EAAE,GAAG;4BACf,IAAI,EAAE,EAAE,OAAO,EAAE,qCAAqC,EAAE;yBACzD,CAAC,CAAC;qBACJ;oBAED,MAAM,gBAAgB,GAAG,EAA0C,CAAC;oBAEpE,KAAK,MAAM,OAAO,IAAI,UAAU,aAAV,UAAU,uBAAV,UAAU,CAAE,KAAK,EAAE;wBACvC,MAAM,WAAW,GAAG,UAAU,aAAV,UAAU,uBAAV,UAAU,CAAE,KAAK,CAAC,OAAO,CAAC,CAAC;wBAC/C,IAAI,CAAC,CAAA,WAAW,aAAX,WAAW,uBAAX,WAAW,CAAE,IAAI,CAAA,EAAE;4BACtB,OAAO,IAAI,CAAC,eAAe,CAAC;gCAC1B,UAAU,EAAE,GAAG;gCACf,IAAI,EAAE,0BAA0B,WAAW,CAAC,QAAQ,EAAE;6BACvD,CAAC,CAAC;yBACJ;wBAED,MAAM,cAAc,GAAG,MAAA,WAAW,CAAC,QAAQ,mCAAI,WAAW,CAAC,SAAS,CAAC;wBACrE,OAAO,CAAC,GAAG,CAAC,kBAAkB,EAAE,cAAc,CAAC,CAAC;wBAChD,MAAM,CAAC,IAAI,EAAE,GAAG,CAAC,GAAG,cAAc,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;wBAE9C,yBAAyB;wBACzB,gBAAgB,CAAC,IAAI,CACnB,IAAI,CAAC,EAAE;6BACJ,MAAM,CAAC;4BACN,MAAM,EAAE,UAAU;4BAClB,GAAG,EAAE,GAAG,UAAU,CAAC,MAAM,CAAC,QAAQ,IAAI,CAAC,uBAAS,CAAC,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,IAAI,CAC/D,GAAG,CACJ,EAAE;4BACH,WAAW,EAAE,WAAW,CAAC,QAAQ;4BACjC,IAAI,EAAE,WAAW,CAAC,IAAI;4BACtB,eAAe,EAAE,WAAW,CAAC,QAAQ;yBACtC,CAAC;6BACD,OAAO,EAAE,CACb,CAAC;qBACH;oBACD,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;oBAErD,MAAM,gBAAgB,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC;wBACvD,QAAQ,EAAE,YAAY,CAAC,QAAQ;wBAC/B,GAAG,EAAE,YAAY,CAAC,GAAG;wBACrB,MAAM,EAAE,YAAY,CAAC,MAAM;wBAC3B,IAAI,EAAE,YAAY,CAAC,IAAI;qBACxB,CAAC,CAAyB,CAAC;oBAE5B,OAAO,IAAI,CAAC,eAAe,CAAC;wBAC1B,UAAU,EAAE,GAAG;wBACf,IAAI,EACF,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC;qBACvE,CAAC,CAAC;iBACJ;gBAAC,OAAO,GAAG,EAAE;oBACZ,OAAO,CAAC,GAAG,CAAC,4BAA4B,EAAE,GAAG,CAAC,CAAC;oBAC/C,OAAO,IAAI,CAAC,eAAe,CAAC;wBAC1B,UAAU,EAAE,GAAG;wBACf,IAAI,EAAE,EAAE,OAAO,EAAE,MAAA,GAAG,CAAC,OAAO,mCAAI,uBAAuB,EAAE;qBAC1D,CAAC,CAAC;iBACJ;;SACF;QAED,aAAa,CACX,KAcC;YAED,MAAM,MAAM,GAAG,IAAI,gBAAM,CAAC;gBACxB,OAAO,EAAE;oBACP,cAAc,EACZ,KAAK,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,KAAK,CAAC,OAAO,CAAC,cAAc,CAAC;iBACjE;aACF,CAAC,CAAC;YAEH,MAAM,MAAM,GAAG;gBACb,KAAK,EAAE,EAAE;gBACT,MAAM,EAAE,EAAE;aAgBX,CAAC;YAEF,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;gBACrC,MAAM;qBACH,EAAE,CAAC,MAAM,EAAE,CAAC,SAAS,EAAE,IAAI,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,EAAE;oBAC5D,IAAI,QAAgB,CAAC;oBACrB,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,IAAY,EAAE,EAAE;wBAC/B,QAAQ,GAAG,IAAI,CAAC;oBAClB,CAAC,CAAC,CAAC;oBAEH,IAAI,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE;wBAClB,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG;4BACxB,IAAI,EAAE,QAAQ;4BACd,QAAQ;4BACR,SAAS;4BACT,QAAQ;4BACR,QAAQ;4BACR,QAAQ,EAAE,IAAI;yBACf,CAAC;oBACJ,CAAC,CAAC,CAAC;gBACL,CAAC,CAAC;qBACD,EAAE,CAAC,OAAO,EAAE,CAAC,SAAS,EAAE,KAAK,EAAE,EAAE;oBAChC,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,GAAG,KAAK,CAAC;gBACnC,CAAC,CAAC;qBACD,EAAE,CAAC,OAAO,EAAE,CAAC,GAAY,EAAE,EAAE;oBAC5B,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;oBACnB,MAAM,CAAC,GAAG,CAAC,CAAC;gBACd,CAAC,CAAC;qBAED,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE;oBACjB,KAAK,CAAC,UAAU,GAAG,MAAM,CAAC;oBAC1B,OAAO,CAAC,KAAK,CAAC,CAAC;gBACjB,CAAC,CAAC,CAAC;gBAEL,MAAM,CAAC,KAAK,CACV,KAAK,CAAC,IAAI,IAAI,EAAE,EAChB,KAAK,CAAC,eAAe,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAC5C,CAAC;gBACF,MAAM,CAAC,GAAG,EAAE,CAAC;YACf,CAAC,CAAC,CAAC;QACL,CAAC;KACF;IAlLD,wCAkLC","sourcesContent":["import { APIGatewayEvent } from 'aws-lambda';\nimport { S3 } from 'aws-sdk';\nimport Busboy from 'busboy';\nimport { BaseHandler } from './base-handler';\nimport { snakeCase } from 'change-case';\n\nexport interface IUploadSuccessBody {\n  location: string;\n  key: string;\n  bucket: string;\n  eTag: string;\n}\n\n/**\n * Upload files to s3 or return upload Url\n * @params returnSignedUrlOnly: boolean @default false\n */\nexport class AssetsUploader extends BaseHandler {\n  constructor(private s3: S3) {\n    super();\n  }\n  async run(\n    event: APIGatewayEvent\n  ): Promise<{\n    statusCode: number;\n    body: string;\n    headers?: unknown;\n    isBase64Encoded: boolean;\n  }> {\n    const bucketName = process.env.BUCKET_NAME;\n\n    if (!bucketName) {\n      return this.encodedResponse({\n        statusCode: 500,\n        body: { message: 'Missing required environment variable: BUCKET_NAME' },\n      });\n    }\n\n    if (!event.body) {\n      return this.encodedResponse({\n        statusCode: 400,\n        body: { message: 'Cannot upload process request with empty body' },\n      });\n    }\n\n    try {\n      const { parsedBody } = await this.parseFormData(event);\n\n      if (!parsedBody) {\n        throw new Error('No body after parsing.');\n      }\n\n      if (!parsedBody.fields?.s3Prefix) {\n        return this.encodedResponse({\n          statusCode: 400,\n          body: { message: `Missing required property: s3Prefix` },\n        });\n      }\n\n      const s3UploadPromises = [] as Promise<S3.ManagedUpload.SendData>[];\n\n      for (const fileKey in parsedBody?.files) {\n        const currentFile = parsedBody?.files[fileKey];\n        if (!currentFile?.data) {\n          return this.encodedResponse({\n            statusCode: 400,\n            body: `Unable to parse binary ${currentFile.fileName}`,\n          });\n        }\n\n        const filePublicName = currentFile.fileName ?? currentFile.fieldName;\n        console.log('Uploading File: ', filePublicName);\n        const [name, ext] = filePublicName.split('.');\n\n        // upload each file to s3\n        s3UploadPromises.push(\n          this.s3\n            .upload({\n              Bucket: bucketName,\n              Key: `${parsedBody.fields.s3Prefix}/${[snakeCase(name), ext].join(\n                '.'\n              )}`,\n              ContentType: currentFile.mimeType,\n              Body: currentFile.data,\n              ContentEncoding: currentFile.encoding,\n            })\n            .promise()\n        );\n      }\n      const response = await Promise.all(s3UploadPromises);\n\n      const responseToReturn = response.map((returnedItem) => ({\n        location: returnedItem.Location,\n        key: returnedItem.Key,\n        bucket: returnedItem.Bucket,\n        eTag: returnedItem.ETag,\n      })) as IUploadSuccessBody[];\n\n      return this.encodedResponse({\n        statusCode: 200,\n        body:\n          responseToReturn.length > 1 ? responseToReturn : responseToReturn[0],\n      });\n    } catch (err) {\n      console.log('Error processing request: ', err);\n      return this.encodedResponse({\n        statusCode: 500,\n        body: { message: err.message ?? 'Something went wrong!' },\n      });\n    }\n  }\n\n  parseFormData(\n    event: APIGatewayEvent & {\n      parsedBody?: {\n        files: {\n          [key: string]: {\n            data?: Buffer;\n            fileName?: string;\n            fieldName: string;\n            encoding: string;\n            mimeType: string;\n            value?: unknown;\n          };\n        };\n        fields: { [fieldName: string]: unknown };\n      };\n    }\n  ): Promise<typeof event> {\n    const busboy = new Busboy({\n      headers: {\n        'content-type':\n          event.headers['Content-Type'] || event.headers['content-type'],\n      },\n    });\n\n    const result = {\n      files: {},\n      fields: {},\n    } as {\n      files: {\n        [fileName: string]: {\n          data?: Buffer;\n          fileName?: string;\n          fieldName: string;\n          encoding: string;\n          mimeType: string;\n          value?: unknown;\n          isBinary?: boolean;\n        };\n      };\n      fields: {\n        [fieldName: string]: unknown;\n      };\n    };\n\n    return new Promise((resolve, reject) => {\n      busboy\n        .on('file', (fieldName, file, fileName, encoding, mimeType) => {\n          let fileData: Buffer;\n          file.on('data', (data: Buffer) => {\n            fileData = data;\n          });\n\n          file.on('end', () => {\n            result.files[fieldName] = {\n              data: fileData,\n              fileName,\n              fieldName,\n              encoding,\n              mimeType,\n              isBinary: true,\n            };\n          });\n        })\n        .on('field', (fieldName, value) => {\n          result.fields[fieldName] = value;\n        })\n        .on('error', (err: unknown) => {\n          console.error(err);\n          reject(err);\n        })\n\n        .on('finish', () => {\n          event.parsedBody = result;\n          resolve(event);\n        });\n\n      busboy.write(\n        event.body || '',\n        event.isBase64Encoded ? 'base64' : 'binary'\n      );\n      busboy.end();\n    });\n  }\n}\n"]}