UNPKG

@scloud/lambda-local

Version:

Run typical Lambda handlers locally.

179 lines 23.2 kB
"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.cloudfrontLocal = void 0; const express_1 = __importDefault(require("express")); const eventTemplate = { body: '', headers: {}, multiValueHeaders: {}, httpMethod: '', isBase64Encoded: false, path: '', pathParameters: null, queryStringParameters: {}, multiValueQueryStringParameters: {}, stageVariables: null, resource: '?', requestContext: { accountId: '', apiId: '', authorizer: null, httpMethod: '', identity: { accessKey: null, accountId: null, apiKey: null, apiKeyId: null, caller: null, clientCert: null, cognitoAuthenticationProvider: null, cognitoAuthenticationType: null, cognitoIdentityId: null, cognitoIdentityPoolId: null, principalOrgId: null, sourceIp: '', user: null, userAgent: null, userArn: null, }, path: '', protocol: '', requestId: '', requestTimeEpoch: 0, resourceId: '', resourcePath: '', stage: '', }, }; const contextTemplate = { awsRequestId: '', callbackWaitsForEmptyEventLoop: false, functionName: '', functionVersion: '', invokedFunctionArn: '', logGroupName: '', logStreamName: '', memoryLimitInMB: '', getRemainingTimeInMillis: () => 0, done: () => { }, fail: () => { }, succeed: () => { }, }; // eslint-disable-next-line no-unused-vars function cloudfrontLocal(cloudfrontPathMappings) { const port = +(process.env.port || '3000'); const app = (0, express_1.default)(); // https://stackoverflow.com/questions/12345166/how-to-force-parse-request-body-as-plain-text-instead-of-json-in-express app.use(express_1.default.text({ type: '*/*' })); app.all('/*', async (req, res) => { // const url = new URL(req.originalUrl, 'https://example.com'); // Headers - NB it seems that in Lambda multiValueHeaders always contains the values from headers const headers = {}; const multiValueHeaders = {}; Object.keys(req.headers).forEach((header) => { if (req.headers[header] === undefined) { headers[header] = undefined; multiValueHeaders[header] = undefined; } if (typeof req.headers[header] === 'string') { headers[header] = req.headers[header]; multiValueHeaders[header] = [req.headers[header]]; } if (Array.isArray(req.headers[header])) { multiValueHeaders[header] = req.headers[header]; } }); // Query string - basic translation const queryStringParameters = {}; const multiValueQueryStringParameters = {}; Object.keys(req.query).forEach((parameter) => { queryStringParameters[parameter] = undefined; if (typeof req.query[parameter] === 'string') queryStringParameters[parameter] = req.query[parameter]; if (Array.isArray(req.query[parameter])) multiValueQueryStringParameters[parameter] = req.query[parameter]; }); const event = { ...eventTemplate, body: typeof req.body === 'string' ? req.body : JSON.stringify(req.body), headers, multiValueHeaders, httpMethod: req.method, path: req.path, queryStringParameters, multiValueQueryStringParameters, requestContext: { ...eventTemplate.requestContext, httpMethod: req.method, path: req.path, protocol: req.protocol, }, }; try { // Print out the event that will be sent to the handler console.log('Event:'); Object.keys(event).forEach((key) => { console.log(` - ${key}: ${JSON.stringify(event[key])}`); }); const paths = Object.keys(cloudfrontPathMappings); // Try a simple mapping let handler = cloudfrontPathMappings[event.path]; // Fall back to a '*' match: paths.forEach((path) => { let partialMatch = path; // Strip leading slash: if (partialMatch.startsWith('/')) { partialMatch = path.slice(1); } // Remove trailing '*' wildcard: if (partialMatch.endsWith('*')) { partialMatch = path.slice(0, -1); } // Get the first match: const candidate = event.path.startsWith(partialMatch) ? cloudfrontPathMappings[path] : undefined; handler = handler || candidate; }); // Invoke the function handler: const result = handler ? await handler(event, contextTemplate) : { statusCode: 404, body: `Path not matched: ${event.path} (${paths})` }; // Print out the response if successful if (result) { if (result.statusCode === 404) { console.log(`404: ${event.path}`); } else { console.log('Result:'); Object.keys(result).forEach((key) => { console.log(` - ${key}: ${JSON.stringify(result[key]).slice(0, 100)}`); }); } } // Send the response res.status(result.statusCode); if (result.multiValueHeaders) { Object.keys(result.multiValueHeaders).forEach((key) => { res.set(key, result.multiValueHeaders[key].map((value) => `${value}`)); }); } if (result.headers) { Object.keys(result.headers).forEach((key) => { res.set(key, `${result.headers[key]}`); }); } res.send(result.body); } catch (e) { // Log the error and send a 500 response console.log(e); console.log(e.stack); res.status(500).send(`${e}`); } }); app.listen(port, () => { console.log(`Lambda handler can be invoked at http://localhost:${port}`); }); } exports.cloudfrontLocal = cloudfrontLocal; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudfrontLocal.js","sourceRoot":"","sources":["../../src/cloudfrontLocal.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-unused-vars */\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { request } from 'http';\nimport express, { Request, Response } from 'express';\nimport { APIGatewayProxyEvent, APIGatewayProxyResult, Context } from 'aws-lambda';\n\nexport interface CloudfrontPathMappings {\n  [key: string]: (event: APIGatewayProxyEvent, context: Context) => Promise<APIGatewayProxyResult>;\n}\n\nconst eventTemplate: APIGatewayProxyEvent = {\n  body: '',\n  headers: {},\n  multiValueHeaders: {},\n  httpMethod: '',\n  isBase64Encoded: false,\n  path: '',\n  pathParameters: null,\n  queryStringParameters: {},\n  multiValueQueryStringParameters: {},\n  stageVariables: null,\n  resource: '?',\n  requestContext: {\n    accountId: '',\n    apiId: '',\n    authorizer: null,\n    httpMethod: '',\n    identity: {\n      accessKey: null,\n      accountId: null,\n      apiKey: null,\n      apiKeyId: null,\n      caller: null,\n      clientCert: null,\n      cognitoAuthenticationProvider: null,\n      cognitoAuthenticationType: null,\n      cognitoIdentityId: null,\n      cognitoIdentityPoolId: null,\n      principalOrgId: null,\n      sourceIp: '',\n      user: null,\n      userAgent: null,\n      userArn: null,\n    },\n    path: '',\n    protocol: '',\n    requestId: '',\n    requestTimeEpoch: 0,\n    resourceId: '',\n    resourcePath: '',\n    stage: '',\n  },\n};\nconst contextTemplate: Context = {\n  awsRequestId: '',\n  callbackWaitsForEmptyEventLoop: false,\n  functionName: '',\n  functionVersion: '',\n  invokedFunctionArn: '',\n  logGroupName: '',\n  logStreamName: '',\n  memoryLimitInMB: '',\n  getRemainingTimeInMillis: () => 0,\n  done: () => { },\n  fail: () => { },\n  succeed: () => { },\n};\n\n// eslint-disable-next-line no-unused-vars\nexport function cloudfrontLocal(cloudfrontPathMappings: CloudfrontPathMappings) {\n  const port = +(process.env.port || '3000');\n  const app = express();\n\n  // https://stackoverflow.com/questions/12345166/how-to-force-parse-request-body-as-plain-text-instead-of-json-in-express\n  app.use(express.text({ type: '*/*' }));\n\n  app.all('/*', async (req: Request, res: Response) => {\n    // const url = new URL(req.originalUrl, 'https://example.com');\n    // Headers - NB it seems that in Lambda multiValueHeaders always contains the values from headers\n    const headers: Record<string, string | undefined> = {};\n    const multiValueHeaders: Record<string, string[] | undefined> = {};\n    Object.keys(req.headers).forEach((header) => {\n      if (req.headers[header] === undefined) {\n        headers[header] = undefined;\n        multiValueHeaders[header] = undefined;\n      }\n      if (typeof req.headers[header] === 'string') {\n        headers[header] = req.headers[header] as string;\n        multiValueHeaders[header] = [req.headers[header] as string];\n      }\n      if (Array.isArray(req.headers[header])) {\n        multiValueHeaders[header] = req.headers[header] as string[];\n      }\n    });\n\n    // Query string - basic translation\n    const queryStringParameters: Record<string, string | undefined> = {};\n    const multiValueQueryStringParameters: Record<string, string[] | undefined> = {};\n    Object.keys(req.query).forEach((parameter) => {\n      queryStringParameters[parameter] = undefined;\n      if (typeof req.query[parameter] === 'string') queryStringParameters[parameter] = req.query[parameter] as string;\n      if (Array.isArray(req.query[parameter])) multiValueQueryStringParameters[parameter] = req.query[parameter] as string[];\n    });\n\n    const event = {\n      ...eventTemplate,\n      body: typeof req.body === 'string' ? req.body : JSON.stringify(req.body),\n      headers,\n      multiValueHeaders,\n      httpMethod: req.method,\n      path: req.path,\n      queryStringParameters,\n      multiValueQueryStringParameters,\n      requestContext: {\n        ...eventTemplate.requestContext,\n        httpMethod: req.method,\n        path: req.path,\n        protocol: req.protocol,\n      },\n    };\n\n    try {\n      // Print out the event that will be sent to the handler\n      console.log('Event:');\n      Object.keys(event).forEach((key) => {\n        console.log(` - ${key}: ${JSON.stringify(event[key as keyof APIGatewayProxyEvent])}`);\n      });\n\n      const paths = Object.keys(cloudfrontPathMappings);\n\n      // Try a simple mapping\n      let handler = cloudfrontPathMappings[event.path];\n\n      // Fall back to a '*' match:\n      paths.forEach((path) => {\n        let partialMatch = path;\n        // Strip leading slash:\n        if (partialMatch.startsWith('/')) {\n          partialMatch = path.slice(1);\n        }\n        // Remove trailing '*' wildcard:\n        if (partialMatch.endsWith('*')) {\n          partialMatch = path.slice(0, -1);\n        }\n        // Get the first match:\n        const candidate = event.path.startsWith(partialMatch) ? cloudfrontPathMappings[path] : undefined;\n        handler = handler || candidate;\n      });\n\n      // Invoke the function handler:\n      const result: APIGatewayProxyResult = handler ? await handler(event, contextTemplate) : { statusCode: 404, body: `Path not matched: ${event.path} (${paths})` };\n\n      // Print out the response if successful\n      if (result) {\n        if (result.statusCode === 404) {\n          console.log(`404: ${event.path}`);\n        } else {\n          console.log('Result:');\n          Object.keys(result).forEach((key) => {\n            console.log(` - ${key}: ${JSON.stringify(result[key as keyof APIGatewayProxyResult]).slice(0, 100)}`);\n          });\n        }\n      }\n\n      // Send the response\n      res.status(result.statusCode);\n      if (result.multiValueHeaders) {\n        Object.keys(result.multiValueHeaders).forEach((key) => {\n          res.set(key, result.multiValueHeaders![key].map((value) => `${value}`));\n        });\n      }\n      if (result.headers) {\n        Object.keys(result.headers).forEach((key) => {\n          res.set(key, `${result.headers![key]}`);\n        });\n      }\n      res.send(result.body);\n    } catch (e) {\n      // Log the error and send a 500 response\n      console.log(e);\n      console.log((e as Error).stack);\n      res.status(500).send(`${e}`);\n    }\n  });\n\n  app.listen(port, () => {\n    console.log(`Lambda handler can be invoked at http://localhost:${port}`);\n  });\n}\n"]}