UNPKG

@scloud/lambda-local

Version:

Run typical Lambda handlers locally.

105 lines 16.8 kB
"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.webappRoutesLocal = webappRoutesLocal; const express_1 = __importDefault(require("express")); function webappRoutesLocal(cloudfrontPathMappings, staticContent, debug = false) { 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: '*/*' })); Object.keys(cloudfrontPathMappings).forEach((route) => { app.all(route, 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 = { body: typeof req.body === 'string' ? req.body : JSON.stringify(req.body), headers, multiValueHeaders, httpMethod: req.method, path: req.path, queryStringParameters, multiValueQueryStringParameters, requestContext: { httpMethod: req.method, path: req.path, protocol: req.protocol, }, }; try { // Print out the event that will be sent to the handler if (debug) { console.log('Event:'); console.log(event.httpMethod, event.path); console.log(JSON.stringify(event, null, 2)); } const paths = Object.keys(cloudfrontPathMappings); // Use the handler for this Express route const handler = cloudfrontPathMappings[route]; if (!handler) console.log(`Unmatched path: ${event.path}`); // Invoke the function handler: const result = handler ? await handler(event, {}) : { statusCode: 404, body: `Path not matched: ${event.path} (${paths})` }; // Print out the response if successful if (debug) { console.log('Result:'); console.log(event.httpMethod, event.path, result.statusCode); console.log(JSON.stringify(result, null, 2)); } // 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]}`); }); } // Body 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}`); } }); }); if (staticContent) app.use(express_1.default.static(staticContent)); app.listen(port, () => { console.log(`Lambda handler can be invoked at http://localhost:${port}`); }); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webappRoutesLocal.js","sourceRoot":"","sources":["../../src/webappRoutesLocal.ts"],"names":[],"mappings":"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","sourcesContent":["import 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\nexport function webappRoutesLocal(cloudfrontPathMappings: CloudfrontPathMappings, staticContent?: string, debug = false) {\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  Object.keys(cloudfrontPathMappings).forEach((route) => {\n    app.all(route, 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: APIGatewayProxyEvent = {\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          httpMethod: req.method,\n          path: req.path,\n          protocol: req.protocol,\n        } as unknown as APIGatewayProxyEvent['requestContext'],\n      } as unknown as APIGatewayProxyEvent;\n\n      try {\n        // Print out the event that will be sent to the handler\n        if (debug) {\n          console.log('Event:');\n          console.log(event.httpMethod, event.path);\n          console.log(JSON.stringify(event, null, 2));\n        }\n\n        const paths = Object.keys(cloudfrontPathMappings);\n\n        // Use the handler for this Express route\n        const handler = cloudfrontPathMappings[route];\n        if (!handler) console.log(`Unmatched path: ${event.path}`);\n\n        // Invoke the function handler:\n        const result = handler ? await handler(event, {} as Context) : { statusCode: 404, body: `Path not matched: ${event.path} (${paths})` };\n\n        // Print out the response if successful\n        if (debug) {\n          console.log('Result:');\n          console.log(event.httpMethod, event.path, result.statusCode);\n          console.log(JSON.stringify(result, null, 2));\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\n        // Body\n        res.send(result.body);\n\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\n  if (staticContent) app.use(express.static(staticContent));\n\n  app.listen(port, () => {\n    console.log(`Lambda handler can be invoked at http://localhost:${port}`);\n  });\n}\n"]}