cdk-local-lambda 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (53) hide show
  1. package/LICENSE +202 -0
  2. package/README.md +94 -0
  3. package/lib/aspect/docker-function-hook.d.ts +18 -0
  4. package/lib/aspect/docker-function-hook.js +31 -0
  5. package/lib/aspect/live-lambda-aspect.d.ts +85 -0
  6. package/lib/aspect/live-lambda-aspect.js +277 -0
  7. package/lib/aspect/live-lambda-bootstrap.d.ts +17 -0
  8. package/lib/aspect/live-lambda-bootstrap.js +260 -0
  9. package/lib/aspect/nodejs-function-hook.d.ts +20 -0
  10. package/lib/aspect/nodejs-function-hook.js +27 -0
  11. package/lib/bootstrap-stack/bootstrap-stack.d.ts +60 -0
  12. package/lib/bootstrap-stack/bootstrap-stack.js +338 -0
  13. package/lib/cli/appsync/client.d.ts +30 -0
  14. package/lib/cli/appsync/client.js +227 -0
  15. package/lib/cli/cdk-app.d.ts +7 -0
  16. package/lib/cli/cdk-app.js +25 -0
  17. package/lib/cli/commands/bootstrap.d.ts +9 -0
  18. package/lib/cli/commands/bootstrap.js +50 -0
  19. package/lib/cli/commands/local.d.ts +40 -0
  20. package/lib/cli/commands/local.js +1172 -0
  21. package/lib/cli/daemon.d.ts +22 -0
  22. package/lib/cli/daemon.js +18 -0
  23. package/lib/cli/docker/container.d.ts +116 -0
  24. package/lib/cli/docker/container.js +414 -0
  25. package/lib/cli/docker/types.d.ts +71 -0
  26. package/lib/cli/docker/types.js +5 -0
  27. package/lib/cli/docker/watcher.d.ts +44 -0
  28. package/lib/cli/docker/watcher.js +115 -0
  29. package/lib/cli/index.d.ts +9 -0
  30. package/lib/cli/index.js +26 -0
  31. package/lib/cli/runtime-api/server.d.ts +102 -0
  32. package/lib/cli/runtime-api/server.js +396 -0
  33. package/lib/cli/runtime-api/types.d.ts +149 -0
  34. package/lib/cli/runtime-api/types.js +10 -0
  35. package/lib/cli/runtime-wrapper/nodejs-runtime.d.ts +16 -0
  36. package/lib/cli/runtime-wrapper/nodejs-runtime.js +248 -0
  37. package/lib/cli/watcher/file-watcher.d.ts +32 -0
  38. package/lib/cli/watcher/file-watcher.js +57 -0
  39. package/lib/functions/bridge/appsync-client.d.ts +73 -0
  40. package/lib/functions/bridge/appsync-client.js +345 -0
  41. package/lib/functions/bridge/handler.d.ts +17 -0
  42. package/lib/functions/bridge/handler.js +79 -0
  43. package/lib/functions/bridge/ssm-config.d.ts +19 -0
  44. package/lib/functions/bridge/ssm-config.js +45 -0
  45. package/lib/functions/bridge-builder/handler.d.ts +12 -0
  46. package/lib/functions/bridge-builder/handler.js +181 -0
  47. package/lib/functions/bridge-docker/runtime.d.ts +9 -0
  48. package/lib/functions/bridge-docker/runtime.js +127 -0
  49. package/lib/index.d.ts +24 -0
  50. package/lib/index.js +28 -0
  51. package/lib/shared/types.d.ts +102 -0
  52. package/lib/shared/types.js +125 -0
  53. package/package.json +111 -0
@@ -0,0 +1,181 @@
1
+ /**
2
+ * Custom resource handler that builds and uploads the bridge Lambda code.
3
+ *
4
+ * This handler:
5
+ * 1. Receives the bridge source code (base64 encoded) and AppSync endpoints
6
+ * 2. Replaces endpoint placeholders with actual values
7
+ * 3. Creates a zip file with the code
8
+ * 4. Uploads the zip to S3
9
+ * 5. Returns the S3 location for Lambda functions to use
10
+ */
11
+ import * as crypto from "node:crypto";
12
+ import * as zlib from "node:zlib";
13
+ import { DeleteObjectCommand, PutObjectCommand, S3Client, } from "@aws-sdk/client-s3";
14
+ const s3 = new S3Client({});
15
+ /**
16
+ * Create a minimal zip file containing a single file.
17
+ * This is a simple implementation that creates a valid zip without external dependencies.
18
+ */
19
+ function createZip(filename, content) {
20
+ const contentBuffer = Buffer.from(content, "utf-8");
21
+ // Compress the content using deflate
22
+ const compressed = zlib.deflateRawSync(contentBuffer);
23
+ const crc32 = crc32buf(contentBuffer);
24
+ const now = new Date();
25
+ // DOS time and date
26
+ const dosTime = ((now.getHours() << 11) |
27
+ (now.getMinutes() << 5) |
28
+ Math.floor(now.getSeconds() / 2)) &
29
+ 0xffff;
30
+ const dosDate = (((now.getFullYear() - 1980) << 9) |
31
+ ((now.getMonth() + 1) << 5) |
32
+ now.getDate()) &
33
+ 0xffff;
34
+ const filenameBuffer = Buffer.from(filename, "utf-8");
35
+ // Local file header
36
+ const localHeader = Buffer.alloc(30);
37
+ localHeader.writeUInt32LE(0x04034b50, 0); // Local file header signature
38
+ localHeader.writeUInt16LE(20, 4); // Version needed to extract (2.0)
39
+ localHeader.writeUInt16LE(0, 6); // General purpose bit flag
40
+ localHeader.writeUInt16LE(8, 8); // Compression method (deflate)
41
+ localHeader.writeUInt16LE(dosTime, 10); // Last mod file time
42
+ localHeader.writeUInt16LE(dosDate, 12); // Last mod file date
43
+ localHeader.writeUInt32LE(crc32, 14); // CRC-32
44
+ localHeader.writeUInt32LE(compressed.length, 18); // Compressed size
45
+ localHeader.writeUInt32LE(contentBuffer.length, 22); // Uncompressed size
46
+ localHeader.writeUInt16LE(filenameBuffer.length, 26); // Filename length
47
+ localHeader.writeUInt16LE(0, 28); // Extra field length
48
+ // Central directory header
49
+ const centralHeader = Buffer.alloc(46);
50
+ centralHeader.writeUInt32LE(0x02014b50, 0); // Central directory signature
51
+ centralHeader.writeUInt16LE(20, 4); // Version made by
52
+ centralHeader.writeUInt16LE(20, 6); // Version needed to extract
53
+ centralHeader.writeUInt16LE(0, 8); // General purpose bit flag
54
+ centralHeader.writeUInt16LE(8, 10); // Compression method
55
+ centralHeader.writeUInt16LE(dosTime, 12); // Last mod file time
56
+ centralHeader.writeUInt16LE(dosDate, 14); // Last mod file date
57
+ centralHeader.writeUInt32LE(crc32, 16); // CRC-32
58
+ centralHeader.writeUInt32LE(compressed.length, 20); // Compressed size
59
+ centralHeader.writeUInt32LE(contentBuffer.length, 24); // Uncompressed size
60
+ centralHeader.writeUInt16LE(filenameBuffer.length, 28); // Filename length
61
+ centralHeader.writeUInt16LE(0, 30); // Extra field length
62
+ centralHeader.writeUInt16LE(0, 32); // File comment length
63
+ centralHeader.writeUInt16LE(0, 34); // Disk number start
64
+ centralHeader.writeUInt16LE(0, 36); // Internal file attributes
65
+ centralHeader.writeUInt32LE(0, 38); // External file attributes
66
+ centralHeader.writeUInt32LE(0, 42); // Relative offset of local header
67
+ // End of central directory
68
+ const localFileDataSize = localHeader.length + filenameBuffer.length + compressed.length;
69
+ const centralDirSize = centralHeader.length + filenameBuffer.length;
70
+ const endOfCentralDir = Buffer.alloc(22);
71
+ endOfCentralDir.writeUInt32LE(0x06054b50, 0); // End of central dir signature
72
+ endOfCentralDir.writeUInt16LE(0, 4); // Disk number
73
+ endOfCentralDir.writeUInt16LE(0, 6); // Disk number with central dir
74
+ endOfCentralDir.writeUInt16LE(1, 8); // Number of entries on this disk
75
+ endOfCentralDir.writeUInt16LE(1, 10); // Total number of entries
76
+ endOfCentralDir.writeUInt32LE(centralDirSize, 12); // Size of central directory
77
+ endOfCentralDir.writeUInt32LE(localFileDataSize, 16); // Offset of central directory
78
+ endOfCentralDir.writeUInt16LE(0, 20); // Comment length
79
+ return Buffer.concat([
80
+ localHeader,
81
+ filenameBuffer,
82
+ compressed,
83
+ centralHeader,
84
+ filenameBuffer,
85
+ endOfCentralDir,
86
+ ]);
87
+ }
88
+ /**
89
+ * Calculate CRC-32 of a buffer.
90
+ */
91
+ function crc32buf(buf) {
92
+ let crc = 0xffffffff;
93
+ for (let i = 0; i < buf.length; i++) {
94
+ crc ^= buf[i];
95
+ for (let j = 0; j < 8; j++) {
96
+ crc = crc & 1 ? (crc >>> 1) ^ 0xedb88320 : crc >>> 1;
97
+ }
98
+ }
99
+ return (crc ^ 0xffffffff) >>> 0;
100
+ }
101
+ export async function handler(event) {
102
+ console.log("Received event:", JSON.stringify(event, null, 2));
103
+ const props = event.ResourceProperties;
104
+ const { HttpEndpoint, RealtimeEndpoint, BucketName, BridgeSource } = props;
105
+ // Generate a unique key for this version
106
+ const sourceHash = crypto
107
+ .createHash("md5")
108
+ .update(BridgeSource)
109
+ .digest("hex")
110
+ .substring(0, 8);
111
+ const s3Key = `bridge/handler-${sourceHash}.zip`;
112
+ try {
113
+ if (event.RequestType === "Delete") {
114
+ // Clean up: delete the S3 object
115
+ // Use the physical resource ID from the event, not the computed s3Key
116
+ // This handles the case where the key format changed between versions
117
+ const keyToDelete = event.PhysicalResourceId || s3Key;
118
+ console.log(`Deleting S3 object: s3://${BucketName}/${keyToDelete}`);
119
+ try {
120
+ await s3.send(new DeleteObjectCommand({
121
+ Bucket: BucketName,
122
+ Key: keyToDelete,
123
+ }));
124
+ }
125
+ catch (deleteError) {
126
+ // Ignore delete errors - the object might not exist
127
+ console.log(`Note: Could not delete ${keyToDelete}: ${deleteError}`);
128
+ }
129
+ return {
130
+ Status: "SUCCESS",
131
+ PhysicalResourceId: event.PhysicalResourceId || s3Key,
132
+ StackId: event.StackId,
133
+ RequestId: event.RequestId,
134
+ LogicalResourceId: event.LogicalResourceId,
135
+ Data: {},
136
+ };
137
+ }
138
+ // Create or Update
139
+ // Decode the base64 source
140
+ const bridgeCode = Buffer.from(BridgeSource, "base64").toString("utf-8");
141
+ // Replace placeholders with actual endpoints
142
+ const finalCode = bridgeCode
143
+ .replace(/__APPSYNC_HTTP_ENDPOINT__/g, HttpEndpoint)
144
+ .replace(/__APPSYNC_REALTIME_ENDPOINT__/g, RealtimeEndpoint);
145
+ // Create a zip file containing the handler code
146
+ // Lambda expects the handler file to be named 'index.js' when handler is 'index.handler'
147
+ const zipBuffer = createZip("index.js", finalCode);
148
+ // Upload to S3
149
+ await s3.send(new PutObjectCommand({
150
+ Bucket: BucketName,
151
+ Key: s3Key,
152
+ Body: zipBuffer,
153
+ ContentType: "application/zip",
154
+ }));
155
+ console.log(`Uploaded bridge code to s3://${BucketName}/${s3Key}`);
156
+ return {
157
+ Status: "SUCCESS",
158
+ PhysicalResourceId: s3Key,
159
+ StackId: event.StackId,
160
+ RequestId: event.RequestId,
161
+ LogicalResourceId: event.LogicalResourceId,
162
+ Data: {
163
+ BucketName,
164
+ S3Key: s3Key,
165
+ },
166
+ };
167
+ }
168
+ catch (error) {
169
+ console.error("Error:", error);
170
+ return {
171
+ Status: "FAILED",
172
+ Reason: error instanceof Error ? error.message : "Unknown error",
173
+ PhysicalResourceId: s3Key,
174
+ StackId: event.StackId,
175
+ RequestId: event.RequestId,
176
+ LogicalResourceId: event.LogicalResourceId,
177
+ Data: {},
178
+ };
179
+ }
180
+ }
181
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../../src/functions/bridge-builder/handler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Custom resource handler that builds and uploads the bridge Lambda code.\n *\n * This handler:\n * 1. Receives the bridge source code (base64 encoded) and AppSync endpoints\n * 2. Replaces endpoint placeholders with actual values\n * 3. Creates a zip file with the code\n * 4. Uploads the zip to S3\n * 5. Returns the S3 location for Lambda functions to use\n */\n\nimport * as crypto from \"node:crypto\"\nimport * as zlib from \"node:zlib\"\nimport {\n  DeleteObjectCommand,\n  PutObjectCommand,\n  S3Client,\n} from \"@aws-sdk/client-s3\"\nimport type {\n  CloudFormationCustomResourceEvent,\n  CloudFormationCustomResourceResponse,\n} from \"aws-lambda\"\n\nconst s3 = new S3Client({})\n\n/**\n * Create a minimal zip file containing a single file.\n * This is a simple implementation that creates a valid zip without external dependencies.\n */\nfunction createZip(filename: string, content: string): Buffer {\n  const contentBuffer = Buffer.from(content, \"utf-8\")\n\n  // Compress the content using deflate\n  const compressed = zlib.deflateRawSync(contentBuffer)\n\n  const crc32 = crc32buf(contentBuffer)\n  const now = new Date()\n\n  // DOS time and date\n  const dosTime =\n    ((now.getHours() << 11) |\n      (now.getMinutes() << 5) |\n      Math.floor(now.getSeconds() / 2)) &\n    0xffff\n  const dosDate =\n    (((now.getFullYear() - 1980) << 9) |\n      ((now.getMonth() + 1) << 5) |\n      now.getDate()) &\n    0xffff\n\n  const filenameBuffer = Buffer.from(filename, \"utf-8\")\n\n  // Local file header\n  const localHeader = Buffer.alloc(30)\n  localHeader.writeUInt32LE(0x04034b50, 0) // Local file header signature\n  localHeader.writeUInt16LE(20, 4) // Version needed to extract (2.0)\n  localHeader.writeUInt16LE(0, 6) // General purpose bit flag\n  localHeader.writeUInt16LE(8, 8) // Compression method (deflate)\n  localHeader.writeUInt16LE(dosTime, 10) // Last mod file time\n  localHeader.writeUInt16LE(dosDate, 12) // Last mod file date\n  localHeader.writeUInt32LE(crc32, 14) // CRC-32\n  localHeader.writeUInt32LE(compressed.length, 18) // Compressed size\n  localHeader.writeUInt32LE(contentBuffer.length, 22) // Uncompressed size\n  localHeader.writeUInt16LE(filenameBuffer.length, 26) // Filename length\n  localHeader.writeUInt16LE(0, 28) // Extra field length\n\n  // Central directory header\n  const centralHeader = Buffer.alloc(46)\n  centralHeader.writeUInt32LE(0x02014b50, 0) // Central directory signature\n  centralHeader.writeUInt16LE(20, 4) // Version made by\n  centralHeader.writeUInt16LE(20, 6) // Version needed to extract\n  centralHeader.writeUInt16LE(0, 8) // General purpose bit flag\n  centralHeader.writeUInt16LE(8, 10) // Compression method\n  centralHeader.writeUInt16LE(dosTime, 12) // Last mod file time\n  centralHeader.writeUInt16LE(dosDate, 14) // Last mod file date\n  centralHeader.writeUInt32LE(crc32, 16) // CRC-32\n  centralHeader.writeUInt32LE(compressed.length, 20) // Compressed size\n  centralHeader.writeUInt32LE(contentBuffer.length, 24) // Uncompressed size\n  centralHeader.writeUInt16LE(filenameBuffer.length, 28) // Filename length\n  centralHeader.writeUInt16LE(0, 30) // Extra field length\n  centralHeader.writeUInt16LE(0, 32) // File comment length\n  centralHeader.writeUInt16LE(0, 34) // Disk number start\n  centralHeader.writeUInt16LE(0, 36) // Internal file attributes\n  centralHeader.writeUInt32LE(0, 38) // External file attributes\n  centralHeader.writeUInt32LE(0, 42) // Relative offset of local header\n\n  // End of central directory\n  const localFileDataSize =\n    localHeader.length + filenameBuffer.length + compressed.length\n  const centralDirSize = centralHeader.length + filenameBuffer.length\n\n  const endOfCentralDir = Buffer.alloc(22)\n  endOfCentralDir.writeUInt32LE(0x06054b50, 0) // End of central dir signature\n  endOfCentralDir.writeUInt16LE(0, 4) // Disk number\n  endOfCentralDir.writeUInt16LE(0, 6) // Disk number with central dir\n  endOfCentralDir.writeUInt16LE(1, 8) // Number of entries on this disk\n  endOfCentralDir.writeUInt16LE(1, 10) // Total number of entries\n  endOfCentralDir.writeUInt32LE(centralDirSize, 12) // Size of central directory\n  endOfCentralDir.writeUInt32LE(localFileDataSize, 16) // Offset of central directory\n  endOfCentralDir.writeUInt16LE(0, 20) // Comment length\n\n  return Buffer.concat([\n    localHeader,\n    filenameBuffer,\n    compressed,\n    centralHeader,\n    filenameBuffer,\n    endOfCentralDir,\n  ])\n}\n\n/**\n * Calculate CRC-32 of a buffer.\n */\nfunction crc32buf(buf: Buffer): number {\n  let crc = 0xffffffff\n  for (let i = 0; i < buf.length; i++) {\n    crc ^= buf[i]\n    for (let j = 0; j < 8; j++) {\n      crc = crc & 1 ? (crc >>> 1) ^ 0xedb88320 : crc >>> 1\n    }\n  }\n  return (crc ^ 0xffffffff) >>> 0\n}\n\ninterface ResourceProperties {\n  HttpEndpoint: string\n  RealtimeEndpoint: string\n  BucketName: string\n  BridgeSource: string // Base64 encoded\n  SourceHash?: string\n}\n\nexport async function handler(\n  event: CloudFormationCustomResourceEvent,\n): Promise<CloudFormationCustomResourceResponse> {\n  console.log(\"Received event:\", JSON.stringify(event, null, 2))\n\n  const props = event.ResourceProperties as unknown as ResourceProperties\n  const { HttpEndpoint, RealtimeEndpoint, BucketName, BridgeSource } = props\n\n  // Generate a unique key for this version\n  const sourceHash = crypto\n    .createHash(\"md5\")\n    .update(BridgeSource)\n    .digest(\"hex\")\n    .substring(0, 8)\n  const s3Key = `bridge/handler-${sourceHash}.zip`\n\n  try {\n    if (event.RequestType === \"Delete\") {\n      // Clean up: delete the S3 object\n      // Use the physical resource ID from the event, not the computed s3Key\n      // This handles the case where the key format changed between versions\n      const keyToDelete = event.PhysicalResourceId || s3Key\n      console.log(`Deleting S3 object: s3://${BucketName}/${keyToDelete}`)\n\n      try {\n        await s3.send(\n          new DeleteObjectCommand({\n            Bucket: BucketName,\n            Key: keyToDelete,\n          }),\n        )\n      } catch (deleteError) {\n        // Ignore delete errors - the object might not exist\n        console.log(`Note: Could not delete ${keyToDelete}: ${deleteError}`)\n      }\n\n      return {\n        Status: \"SUCCESS\",\n        PhysicalResourceId: event.PhysicalResourceId || s3Key,\n        StackId: event.StackId,\n        RequestId: event.RequestId,\n        LogicalResourceId: event.LogicalResourceId,\n        Data: {},\n      }\n    }\n\n    // Create or Update\n    // Decode the base64 source\n    const bridgeCode = Buffer.from(BridgeSource, \"base64\").toString(\"utf-8\")\n\n    // Replace placeholders with actual endpoints\n    const finalCode = bridgeCode\n      .replace(/__APPSYNC_HTTP_ENDPOINT__/g, HttpEndpoint)\n      .replace(/__APPSYNC_REALTIME_ENDPOINT__/g, RealtimeEndpoint)\n\n    // Create a zip file containing the handler code\n    // Lambda expects the handler file to be named 'index.js' when handler is 'index.handler'\n    const zipBuffer = createZip(\"index.js\", finalCode)\n\n    // Upload to S3\n    await s3.send(\n      new PutObjectCommand({\n        Bucket: BucketName,\n        Key: s3Key,\n        Body: zipBuffer,\n        ContentType: \"application/zip\",\n      }),\n    )\n\n    console.log(`Uploaded bridge code to s3://${BucketName}/${s3Key}`)\n\n    return {\n      Status: \"SUCCESS\",\n      PhysicalResourceId: s3Key,\n      StackId: event.StackId,\n      RequestId: event.RequestId,\n      LogicalResourceId: event.LogicalResourceId,\n      Data: {\n        BucketName,\n        S3Key: s3Key,\n      },\n    }\n  } catch (error) {\n    console.error(\"Error:\", error)\n\n    return {\n      Status: \"FAILED\",\n      Reason: error instanceof Error ? error.message : \"Unknown error\",\n      PhysicalResourceId: s3Key,\n      StackId: event.StackId,\n      RequestId: event.RequestId,\n      LogicalResourceId: event.LogicalResourceId,\n      Data: {},\n    }\n  }\n}\n"]}
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Lambda Custom Runtime implementation for the Docker bridge.
3
+ *
4
+ * This implements the Lambda Runtime API to:
5
+ * 1. Fetch invocations from the runtime API
6
+ * 2. Call the bridge handler
7
+ * 3. Post responses back to the runtime API
8
+ */
9
+ export {};
@@ -0,0 +1,127 @@
1
+ /**
2
+ * Lambda Custom Runtime implementation for the Docker bridge.
3
+ *
4
+ * This implements the Lambda Runtime API to:
5
+ * 1. Fetch invocations from the runtime API
6
+ * 2. Call the bridge handler
7
+ * 3. Post responses back to the runtime API
8
+ */
9
+ import { handler as bridgeHandler } from "../bridge/handler.js";
10
+ const RUNTIME_API = process.env.AWS_LAMBDA_RUNTIME_API;
11
+ const HANDLER = process.env._HANDLER || "index.handler";
12
+ if (!RUNTIME_API) {
13
+ console.error("[Runtime] AWS_LAMBDA_RUNTIME_API not set");
14
+ process.exit(1);
15
+ }
16
+ const RUNTIME_BASE = `http://${RUNTIME_API}/2018-06-01/runtime`;
17
+ async function getNextInvocation() {
18
+ const response = await fetch(`${RUNTIME_BASE}/invocation/next`);
19
+ if (!response.ok) {
20
+ throw new Error(`Failed to get next invocation: ${response.status}`);
21
+ }
22
+ const headers = Object.fromEntries(response.headers.entries());
23
+ const event = await response.json();
24
+ const requestId = headers["lambda-runtime-aws-request-id"];
25
+ const deadlineMs = parseInt(headers["lambda-runtime-deadline-ms"], 10);
26
+ const invokedFunctionArn = headers["lambda-runtime-invoked-function-arn"];
27
+ const traceId = headers["lambda-runtime-trace-id"];
28
+ // Set trace ID environment variable for X-Ray
29
+ if (traceId) {
30
+ process.env._X_AMZN_TRACE_ID = traceId;
31
+ }
32
+ const context = {
33
+ functionName: process.env.AWS_LAMBDA_FUNCTION_NAME || "unknown",
34
+ functionVersion: process.env.AWS_LAMBDA_FUNCTION_VERSION || "$LATEST",
35
+ invokedFunctionArn,
36
+ memoryLimitInMB: process.env.AWS_LAMBDA_FUNCTION_MEMORY_SIZE || "256",
37
+ awsRequestId: requestId,
38
+ logGroupName: process.env.AWS_LAMBDA_LOG_GROUP_NAME || "/aws/lambda/unknown",
39
+ logStreamName: process.env.AWS_LAMBDA_LOG_STREAM_NAME || "unknown",
40
+ getRemainingTimeInMillis: () => deadlineMs - Date.now(),
41
+ callbackWaitsForEmptyEventLoop: true,
42
+ done: () => { },
43
+ fail: () => { },
44
+ succeed: () => { },
45
+ };
46
+ return { event, context };
47
+ }
48
+ async function postResponse(requestId, response) {
49
+ const res = await fetch(`${RUNTIME_BASE}/invocation/${requestId}/response`, {
50
+ method: "POST",
51
+ headers: { "Content-Type": "application/json" },
52
+ body: JSON.stringify(response),
53
+ });
54
+ if (!res.ok) {
55
+ throw new Error(`Failed to post response: ${res.status}`);
56
+ }
57
+ }
58
+ async function postError(requestId, error) {
59
+ const errorPayload = {
60
+ errorType: error.name || "Error",
61
+ errorMessage: error.message,
62
+ stackTrace: error.stack?.split("\n") || [],
63
+ };
64
+ const res = await fetch(`${RUNTIME_BASE}/invocation/${requestId}/error`, {
65
+ method: "POST",
66
+ headers: {
67
+ "Content-Type": "application/json",
68
+ "Lambda-Runtime-Function-Error-Type": "Unhandled",
69
+ },
70
+ body: JSON.stringify(errorPayload),
71
+ });
72
+ if (!res.ok) {
73
+ console.error(`[Runtime] Failed to post error: ${res.status}`);
74
+ }
75
+ }
76
+ async function postInitError(error) {
77
+ const errorPayload = {
78
+ errorType: error.name || "Error",
79
+ errorMessage: error.message,
80
+ stackTrace: error.stack?.split("\n") || [],
81
+ };
82
+ const res = await fetch(`${RUNTIME_BASE}/init/error`, {
83
+ method: "POST",
84
+ headers: {
85
+ "Content-Type": "application/json",
86
+ "Lambda-Runtime-Function-Error-Type": "Unhandled",
87
+ },
88
+ body: JSON.stringify(errorPayload),
89
+ });
90
+ if (!res.ok) {
91
+ console.error(`[Runtime] Failed to post init error: ${res.status}`);
92
+ }
93
+ }
94
+ async function main() {
95
+ console.log("[Runtime] Starting Lambda custom runtime");
96
+ console.log(`[Runtime] Handler: ${HANDLER}`);
97
+ console.log(`[Runtime] Function: ${process.env.AWS_LAMBDA_FUNCTION_NAME}`);
98
+ // Process invocations in a loop
99
+ while (true) {
100
+ let requestId;
101
+ try {
102
+ // Get next invocation
103
+ const { event, context } = await getNextInvocation();
104
+ requestId = context.awsRequestId;
105
+ console.log(`[Runtime] Processing invocation ${requestId}`);
106
+ // Call the bridge handler
107
+ const result = await bridgeHandler(event, context);
108
+ // Post response
109
+ await postResponse(requestId, result);
110
+ console.log(`[Runtime] Response sent for ${requestId}`);
111
+ }
112
+ catch (error) {
113
+ console.error(`[Runtime] Error:`, error);
114
+ if (requestId) {
115
+ await postError(requestId, error instanceof Error ? error : new Error(String(error)));
116
+ }
117
+ else {
118
+ await postInitError(error instanceof Error ? error : new Error(String(error)));
119
+ }
120
+ }
121
+ }
122
+ }
123
+ main().catch((error) => {
124
+ console.error("[Runtime] Fatal error:", error);
125
+ postInitError(error instanceof Error ? error : new Error(String(error))).finally(() => process.exit(1));
126
+ });
127
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runtime.js","sourceRoot":"","sources":["../../../src/functions/bridge-docker/runtime.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Lambda Custom Runtime implementation for the Docker bridge.\n *\n * This implements the Lambda Runtime API to:\n * 1. Fetch invocations from the runtime API\n * 2. Call the bridge handler\n * 3. Post responses back to the runtime API\n */\n\nimport type { Context } from \"aws-lambda\"\nimport { handler as bridgeHandler } from \"../bridge/handler.js\"\n\nconst RUNTIME_API = process.env.AWS_LAMBDA_RUNTIME_API\nconst HANDLER = process.env._HANDLER || \"index.handler\"\n\nif (!RUNTIME_API) {\n  console.error(\"[Runtime] AWS_LAMBDA_RUNTIME_API not set\")\n  process.exit(1)\n}\n\nconst RUNTIME_BASE = `http://${RUNTIME_API}/2018-06-01/runtime`\n\ninterface RuntimeHeaders {\n  \"lambda-runtime-aws-request-id\": string\n  \"lambda-runtime-deadline-ms\": string\n  \"lambda-runtime-invoked-function-arn\": string\n  \"lambda-runtime-trace-id\"?: string\n  \"lambda-runtime-client-context\"?: string\n  \"lambda-runtime-cognito-identity\"?: string\n}\n\nasync function getNextInvocation(): Promise<{\n  event: unknown\n  context: Context\n}> {\n  const response = await fetch(`${RUNTIME_BASE}/invocation/next`)\n\n  if (!response.ok) {\n    throw new Error(`Failed to get next invocation: ${response.status}`)\n  }\n\n  const headers = Object.fromEntries(\n    response.headers.entries(),\n  ) as unknown as RuntimeHeaders\n  const event = await response.json()\n\n  const requestId = headers[\"lambda-runtime-aws-request-id\"]\n  const deadlineMs = parseInt(headers[\"lambda-runtime-deadline-ms\"], 10)\n  const invokedFunctionArn = headers[\"lambda-runtime-invoked-function-arn\"]\n  const traceId = headers[\"lambda-runtime-trace-id\"]\n\n  // Set trace ID environment variable for X-Ray\n  if (traceId) {\n    process.env._X_AMZN_TRACE_ID = traceId\n  }\n\n  const context: Context = {\n    functionName: process.env.AWS_LAMBDA_FUNCTION_NAME || \"unknown\",\n    functionVersion: process.env.AWS_LAMBDA_FUNCTION_VERSION || \"$LATEST\",\n    invokedFunctionArn,\n    memoryLimitInMB: process.env.AWS_LAMBDA_FUNCTION_MEMORY_SIZE || \"256\",\n    awsRequestId: requestId,\n    logGroupName:\n      process.env.AWS_LAMBDA_LOG_GROUP_NAME || \"/aws/lambda/unknown\",\n    logStreamName: process.env.AWS_LAMBDA_LOG_STREAM_NAME || \"unknown\",\n    getRemainingTimeInMillis: () => deadlineMs - Date.now(),\n    callbackWaitsForEmptyEventLoop: true,\n    done: () => {},\n    fail: () => {},\n    succeed: () => {},\n  }\n\n  return { event, context }\n}\n\nasync function postResponse(\n  requestId: string,\n  response: unknown,\n): Promise<void> {\n  const res = await fetch(`${RUNTIME_BASE}/invocation/${requestId}/response`, {\n    method: \"POST\",\n    headers: { \"Content-Type\": \"application/json\" },\n    body: JSON.stringify(response),\n  })\n\n  if (!res.ok) {\n    throw new Error(`Failed to post response: ${res.status}`)\n  }\n}\n\nasync function postError(requestId: string, error: Error): Promise<void> {\n  const errorPayload = {\n    errorType: error.name || \"Error\",\n    errorMessage: error.message,\n    stackTrace: error.stack?.split(\"\\n\") || [],\n  }\n\n  const res = await fetch(`${RUNTIME_BASE}/invocation/${requestId}/error`, {\n    method: \"POST\",\n    headers: {\n      \"Content-Type\": \"application/json\",\n      \"Lambda-Runtime-Function-Error-Type\": \"Unhandled\",\n    },\n    body: JSON.stringify(errorPayload),\n  })\n\n  if (!res.ok) {\n    console.error(`[Runtime] Failed to post error: ${res.status}`)\n  }\n}\n\nasync function postInitError(error: Error): Promise<void> {\n  const errorPayload = {\n    errorType: error.name || \"Error\",\n    errorMessage: error.message,\n    stackTrace: error.stack?.split(\"\\n\") || [],\n  }\n\n  const res = await fetch(`${RUNTIME_BASE}/init/error`, {\n    method: \"POST\",\n    headers: {\n      \"Content-Type\": \"application/json\",\n      \"Lambda-Runtime-Function-Error-Type\": \"Unhandled\",\n    },\n    body: JSON.stringify(errorPayload),\n  })\n\n  if (!res.ok) {\n    console.error(`[Runtime] Failed to post init error: ${res.status}`)\n  }\n}\n\nasync function main() {\n  console.log(\"[Runtime] Starting Lambda custom runtime\")\n  console.log(`[Runtime] Handler: ${HANDLER}`)\n  console.log(`[Runtime] Function: ${process.env.AWS_LAMBDA_FUNCTION_NAME}`)\n\n  // Process invocations in a loop\n  while (true) {\n    let requestId: string | undefined\n\n    try {\n      // Get next invocation\n      const { event, context } = await getNextInvocation()\n      requestId = context.awsRequestId\n\n      console.log(`[Runtime] Processing invocation ${requestId}`)\n\n      // Call the bridge handler\n      const result = await bridgeHandler(event, context)\n\n      // Post response\n      await postResponse(requestId, result)\n      console.log(`[Runtime] Response sent for ${requestId}`)\n    } catch (error) {\n      console.error(`[Runtime] Error:`, error)\n\n      if (requestId) {\n        await postError(\n          requestId,\n          error instanceof Error ? error : new Error(String(error)),\n        )\n      } else {\n        await postInitError(\n          error instanceof Error ? error : new Error(String(error)),\n        )\n      }\n    }\n  }\n}\n\nmain().catch((error) => {\n  console.error(\"[Runtime] Fatal error:\", error)\n  postInitError(\n    error instanceof Error ? error : new Error(String(error)),\n  ).finally(() => process.exit(1))\n})\n"]}
package/lib/index.d.ts ADDED
@@ -0,0 +1,24 @@
1
+ /**
2
+ * Main entry point for cdk-local-lambda.
3
+ *
4
+ * Exports:
5
+ * - LiveLambdaAspect: CDK Aspect to transform Lambda functions
6
+ * - applyLiveLambdaAspect: Helper to apply the aspect if CDK_LIVE=true
7
+ * - isLiveModeEnabled: Check if live mode is enabled
8
+ * - CdkLocalLambdaBootstrapStack: The bootstrap stack for shared infrastructure
9
+ *
10
+ * For Docker function support, you must install the bootstrap BEFORE any CDK
11
+ * imports in your app entry point:
12
+ *
13
+ * ```typescript
14
+ * import "cdk-local-lambda/bootstrap"
15
+ * import * as cdk from "aws-cdk-lib"
16
+ * ```
17
+ *
18
+ * Node.js: this is sufficient.
19
+ * Bun: use `bun --preload cdk-local-lambda/bootstrap` (static ESM imports are
20
+ * linked before this module runs).
21
+ */
22
+ export { applyLiveLambdaAspect, isLiveModeEnabled, LiveLambdaAspect, type LiveLambdaAspectProps, } from "./aspect/live-lambda-aspect.js";
23
+ export { CdkLocalLambdaBootstrapStack, type CdkLocalLambdaBootstrapStackProps, } from "./bootstrap-stack/bootstrap-stack.js";
24
+ export { buildChannelName, type ErrorPayload, hashFunctionName, type InvocationMessage, type LambdaContext, LIVE_LAMBDA_DOCKER_TAG, LIVE_LAMBDA_TAG, type ResponseMessage, SSM_BASE_PATH, } from "./shared/types.js";
package/lib/index.js ADDED
@@ -0,0 +1,28 @@
1
+ /**
2
+ * Main entry point for cdk-local-lambda.
3
+ *
4
+ * Exports:
5
+ * - LiveLambdaAspect: CDK Aspect to transform Lambda functions
6
+ * - applyLiveLambdaAspect: Helper to apply the aspect if CDK_LIVE=true
7
+ * - isLiveModeEnabled: Check if live mode is enabled
8
+ * - CdkLocalLambdaBootstrapStack: The bootstrap stack for shared infrastructure
9
+ *
10
+ * For Docker function support, you must install the bootstrap BEFORE any CDK
11
+ * imports in your app entry point:
12
+ *
13
+ * ```typescript
14
+ * import "cdk-local-lambda/bootstrap"
15
+ * import * as cdk from "aws-cdk-lib"
16
+ * ```
17
+ *
18
+ * Node.js: this is sufficient.
19
+ * Bun: use `bun --preload cdk-local-lambda/bootstrap` (static ESM imports are
20
+ * linked before this module runs).
21
+ */
22
+ // Re-export the aspect and helpers
23
+ export { applyLiveLambdaAspect, isLiveModeEnabled, LiveLambdaAspect, } from "./aspect/live-lambda-aspect.js";
24
+ // Re-export the bootstrap stack
25
+ export { CdkLocalLambdaBootstrapStack, } from "./bootstrap-stack/bootstrap-stack.js";
26
+ // Re-export shared types
27
+ export { buildChannelName, hashFunctionName, LIVE_LAMBDA_DOCKER_TAG, LIVE_LAMBDA_TAG, SSM_BASE_PATH, } from "./shared/types.js";
28
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0JHO0FBRUgsbUNBQW1DO0FBQ25DLE9BQU8sRUFDTCxxQkFBcUIsRUFDckIsaUJBQWlCLEVBQ2pCLGdCQUFnQixHQUVqQixNQUFNLGdDQUFnQyxDQUFBO0FBRXZDLGdDQUFnQztBQUNoQyxPQUFPLEVBQ0wsNEJBQTRCLEdBRTdCLE1BQU0sc0NBQXNDLENBQUE7QUFFN0MseUJBQXlCO0FBQ3pCLE9BQU8sRUFDTCxnQkFBZ0IsRUFFaEIsZ0JBQWdCLEVBR2hCLHNCQUFzQixFQUN0QixlQUFlLEVBRWYsYUFBYSxHQUNkLE1BQU0sbUJBQW1CLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE1haW4gZW50cnkgcG9pbnQgZm9yIGNkay1sb2NhbC1sYW1iZGEuXG4gKlxuICogRXhwb3J0czpcbiAqIC0gTGl2ZUxhbWJkYUFzcGVjdDogQ0RLIEFzcGVjdCB0byB0cmFuc2Zvcm0gTGFtYmRhIGZ1bmN0aW9uc1xuICogLSBhcHBseUxpdmVMYW1iZGFBc3BlY3Q6IEhlbHBlciB0byBhcHBseSB0aGUgYXNwZWN0IGlmIENES19MSVZFPXRydWVcbiAqIC0gaXNMaXZlTW9kZUVuYWJsZWQ6IENoZWNrIGlmIGxpdmUgbW9kZSBpcyBlbmFibGVkXG4gKiAtIENka0xvY2FsTGFtYmRhQm9vdHN0cmFwU3RhY2s6IFRoZSBib290c3RyYXAgc3RhY2sgZm9yIHNoYXJlZCBpbmZyYXN0cnVjdHVyZVxuICpcbiAqIEZvciBEb2NrZXIgZnVuY3Rpb24gc3VwcG9ydCwgeW91IG11c3QgaW5zdGFsbCB0aGUgYm9vdHN0cmFwIEJFRk9SRSBhbnkgQ0RLXG4gKiBpbXBvcnRzIGluIHlvdXIgYXBwIGVudHJ5IHBvaW50OlxuICpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCBcImNkay1sb2NhbC1sYW1iZGEvYm9vdHN0cmFwXCJcbiAqIGltcG9ydCAqIGFzIGNkayBmcm9tIFwiYXdzLWNkay1saWJcIlxuICogYGBgXG4gKlxuICogTm9kZS5qczogdGhpcyBpcyBzdWZmaWNpZW50LlxuICogQnVuOiB1c2UgYGJ1biAtLXByZWxvYWQgY2RrLWxvY2FsLWxhbWJkYS9ib290c3RyYXBgIChzdGF0aWMgRVNNIGltcG9ydHMgYXJlXG4gKiBsaW5rZWQgYmVmb3JlIHRoaXMgbW9kdWxlIHJ1bnMpLlxuICovXG5cbi8vIFJlLWV4cG9ydCB0aGUgYXNwZWN0IGFuZCBoZWxwZXJzXG5leHBvcnQge1xuICBhcHBseUxpdmVMYW1iZGFBc3BlY3QsXG4gIGlzTGl2ZU1vZGVFbmFibGVkLFxuICBMaXZlTGFtYmRhQXNwZWN0LFxuICB0eXBlIExpdmVMYW1iZGFBc3BlY3RQcm9wcyxcbn0gZnJvbSBcIi4vYXNwZWN0L2xpdmUtbGFtYmRhLWFzcGVjdC5qc1wiXG5cbi8vIFJlLWV4cG9ydCB0aGUgYm9vdHN0cmFwIHN0YWNrXG5leHBvcnQge1xuICBDZGtMb2NhbExhbWJkYUJvb3RzdHJhcFN0YWNrLFxuICB0eXBlIENka0xvY2FsTGFtYmRhQm9vdHN0cmFwU3RhY2tQcm9wcyxcbn0gZnJvbSBcIi4vYm9vdHN0cmFwLXN0YWNrL2Jvb3RzdHJhcC1zdGFjay5qc1wiXG5cbi8vIFJlLWV4cG9ydCBzaGFyZWQgdHlwZXNcbmV4cG9ydCB7XG4gIGJ1aWxkQ2hhbm5lbE5hbWUsXG4gIHR5cGUgRXJyb3JQYXlsb2FkLFxuICBoYXNoRnVuY3Rpb25OYW1lLFxuICB0eXBlIEludm9jYXRpb25NZXNzYWdlLFxuICB0eXBlIExhbWJkYUNvbnRleHQsXG4gIExJVkVfTEFNQkRBX0RPQ0tFUl9UQUcsXG4gIExJVkVfTEFNQkRBX1RBRyxcbiAgdHlwZSBSZXNwb25zZU1lc3NhZ2UsXG4gIFNTTV9CQVNFX1BBVEgsXG59IGZyb20gXCIuL3NoYXJlZC90eXBlcy5qc1wiXG4iXX0=
@@ -0,0 +1,102 @@
1
+ /**
2
+ * Shared constants and types used by bootstrap stack, aspect, and CLI.
3
+ */
4
+ /**
5
+ * SSM parameter paths for Live Lambda infrastructure
6
+ * Base path for all Live Lambda SSM parameters
7
+ */
8
+ export declare const SSM_BASE_PATH = "/cdk-local-lambda";
9
+ /**
10
+ * Tag key used to store the local handler path on Lambda functions
11
+ */
12
+ export declare const LIVE_LAMBDA_TAG = "live-lambda:handler";
13
+ /**
14
+ * Tag key used to store the local Docker context path on DockerImageFunction.
15
+ * The daemon uses this to build and run the container locally.
16
+ */
17
+ export declare const LIVE_LAMBDA_DOCKER_TAG = "live-lambda:docker-context";
18
+ /**
19
+ * Name of the CDK bootstrap stack
20
+ */
21
+ export declare const BOOTSTRAP_STACK_NAME = "CdkLocalLambdaBootstrapStack";
22
+ /**
23
+ * Current version of the bootstrap stack.
24
+ * Increment this when making breaking changes to the bootstrap infrastructure.
25
+ */
26
+ export declare const BOOTSTRAP_VERSION = "1";
27
+ /**
28
+ * Environment variable names used by the bridge handler
29
+ */
30
+ export declare const ENV_VARS: {
31
+ readonly HTTP_ENDPOINT: "APPSYNC_HTTP_ENDPOINT";
32
+ readonly REALTIME_ENDPOINT: "APPSYNC_REALTIME_ENDPOINT";
33
+ };
34
+ /**
35
+ * Message types for AppSync Events communication
36
+ */
37
+ export interface InvocationMessage {
38
+ type: "invocation";
39
+ requestId: string;
40
+ event: unknown;
41
+ context: LambdaContext;
42
+ /** Lambda environment variables (forwarded from bridge on first invocation) */
43
+ env?: Record<string, string>;
44
+ }
45
+ /**
46
+ * Environment variables that are set locally by the daemon.
47
+ * These override any values from the bridge Lambda and are excluded from forwarding.
48
+ * Note: AWS_LAMBDA_FUNCTION_MEMORY_SIZE is also set locally but isn't in EXCLUDED_ENV_VARS
49
+ * because it doesn't come from the bridge Lambda's process.env (it's in the context).
50
+ */
51
+ export declare const LOCAL_OVERRIDE_ENV_VARS: Set<string>;
52
+ /**
53
+ * Environment variables to exclude when forwarding from Lambda.
54
+ * These are either Lambda internals or should use local values instead.
55
+ */
56
+ export declare const EXCLUDED_ENV_VARS: Set<string>;
57
+ /**
58
+ * Filter environment variables for forwarding to local execution.
59
+ * Removes Lambda internals and system variables that should use local values.
60
+ */
61
+ export declare function filterEnvVars(env: NodeJS.ProcessEnv): Record<string, string>;
62
+ export interface ResponseMessage {
63
+ type: "response";
64
+ requestId: string;
65
+ result?: unknown;
66
+ error?: ErrorPayload;
67
+ }
68
+ export interface ErrorPayload {
69
+ errorType: string;
70
+ errorMessage: string;
71
+ stackTrace?: string[];
72
+ }
73
+ export interface LambdaContext {
74
+ functionName: string;
75
+ functionVersion: string;
76
+ invokedFunctionArn: string;
77
+ memoryLimitInMB: string;
78
+ awsRequestId: string;
79
+ logGroupName: string;
80
+ logStreamName: string;
81
+ getRemainingTimeInMillis: number;
82
+ }
83
+ /**
84
+ * Hash a function name for use in channel paths.
85
+ * Uses first 16 hex chars of SHA256 to stay within AppSync limits.
86
+ */
87
+ export declare function hashFunctionName(functionName: string): string;
88
+ /**
89
+ * Channel name builders for AppSync Events
90
+ */
91
+ export declare const buildChannelName: {
92
+ /**
93
+ * Channel for sending invocations to the daemon
94
+ * Bridge -> Daemon
95
+ */
96
+ readonly invocation: (functionName: string) => string;
97
+ /**
98
+ * Channel for receiving responses from the daemon
99
+ * Daemon -> Bridge
100
+ */
101
+ readonly response: (functionName: string) => string;
102
+ };