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.
- package/LICENSE +202 -0
- package/README.md +94 -0
- package/lib/aspect/docker-function-hook.d.ts +18 -0
- package/lib/aspect/docker-function-hook.js +31 -0
- package/lib/aspect/live-lambda-aspect.d.ts +85 -0
- package/lib/aspect/live-lambda-aspect.js +277 -0
- package/lib/aspect/live-lambda-bootstrap.d.ts +17 -0
- package/lib/aspect/live-lambda-bootstrap.js +260 -0
- package/lib/aspect/nodejs-function-hook.d.ts +20 -0
- package/lib/aspect/nodejs-function-hook.js +27 -0
- package/lib/bootstrap-stack/bootstrap-stack.d.ts +60 -0
- package/lib/bootstrap-stack/bootstrap-stack.js +338 -0
- package/lib/cli/appsync/client.d.ts +30 -0
- package/lib/cli/appsync/client.js +227 -0
- package/lib/cli/cdk-app.d.ts +7 -0
- package/lib/cli/cdk-app.js +25 -0
- package/lib/cli/commands/bootstrap.d.ts +9 -0
- package/lib/cli/commands/bootstrap.js +50 -0
- package/lib/cli/commands/local.d.ts +40 -0
- package/lib/cli/commands/local.js +1172 -0
- package/lib/cli/daemon.d.ts +22 -0
- package/lib/cli/daemon.js +18 -0
- package/lib/cli/docker/container.d.ts +116 -0
- package/lib/cli/docker/container.js +414 -0
- package/lib/cli/docker/types.d.ts +71 -0
- package/lib/cli/docker/types.js +5 -0
- package/lib/cli/docker/watcher.d.ts +44 -0
- package/lib/cli/docker/watcher.js +115 -0
- package/lib/cli/index.d.ts +9 -0
- package/lib/cli/index.js +26 -0
- package/lib/cli/runtime-api/server.d.ts +102 -0
- package/lib/cli/runtime-api/server.js +396 -0
- package/lib/cli/runtime-api/types.d.ts +149 -0
- package/lib/cli/runtime-api/types.js +10 -0
- package/lib/cli/runtime-wrapper/nodejs-runtime.d.ts +16 -0
- package/lib/cli/runtime-wrapper/nodejs-runtime.js +248 -0
- package/lib/cli/watcher/file-watcher.d.ts +32 -0
- package/lib/cli/watcher/file-watcher.js +57 -0
- package/lib/functions/bridge/appsync-client.d.ts +73 -0
- package/lib/functions/bridge/appsync-client.js +345 -0
- package/lib/functions/bridge/handler.d.ts +17 -0
- package/lib/functions/bridge/handler.js +79 -0
- package/lib/functions/bridge/ssm-config.d.ts +19 -0
- package/lib/functions/bridge/ssm-config.js +45 -0
- package/lib/functions/bridge-builder/handler.d.ts +12 -0
- package/lib/functions/bridge-builder/handler.js +181 -0
- package/lib/functions/bridge-docker/runtime.d.ts +9 -0
- package/lib/functions/bridge-docker/runtime.js +127 -0
- package/lib/index.d.ts +24 -0
- package/lib/index.js +28 -0
- package/lib/shared/types.d.ts +102 -0
- package/lib/shared/types.js +125 -0
- package/package.json +111 -0
|
@@ -0,0 +1,1172 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Local command for running Lambda functions locally using Docker.
|
|
3
|
+
*
|
|
4
|
+
* This command:
|
|
5
|
+
* 1. Starts CDK watch with CDK_LIVE=true and hotswap
|
|
6
|
+
* 2. Discovers Lambda functions with live-lambda:handler tag
|
|
7
|
+
* 3. Connects to AppSync Events
|
|
8
|
+
* 4. Subscribes to invocation channels
|
|
9
|
+
* 5. For each function, maintains ONE Docker container that handles all invocations
|
|
10
|
+
* 6. Sends responses back via AppSync
|
|
11
|
+
* 7. Re-discovers functions after each CDK deploy
|
|
12
|
+
*/
|
|
13
|
+
import { spawn } from "node:child_process";
|
|
14
|
+
import * as path from "node:path";
|
|
15
|
+
import { fileURLToPath } from "node:url";
|
|
16
|
+
import { LambdaClient, ListFunctionsCommand, ListTagsCommand, } from "@aws-sdk/client-lambda";
|
|
17
|
+
import { GetParameterCommand, SSMClient } from "@aws-sdk/client-ssm";
|
|
18
|
+
import { Command, Options } from "@effect/cli";
|
|
19
|
+
import { Command as PlatformCommand, } from "@effect/platform";
|
|
20
|
+
import { BunContext } from "@effect/platform-bun";
|
|
21
|
+
import { Duration, Effect, Exit, Fiber, Logger, LogLevel, Queue, Ref, Schedule, Scope, Stream, } from "effect";
|
|
22
|
+
import { BOOTSTRAP_STACK_NAME, BOOTSTRAP_VERSION, buildChannelName, LIVE_LAMBDA_DOCKER_TAG, LIVE_LAMBDA_TAG, SSM_BASE_PATH, } from "../../shared/types.js";
|
|
23
|
+
import { makeAppSyncClient } from "../appsync/client.js";
|
|
24
|
+
import { buildExtensionWrapperCommand, Docker, DockerLive, makeLambdaContainerConfig, } from "../docker/container.js";
|
|
25
|
+
import { watchDockerContexts, } from "../docker/watcher.js";
|
|
26
|
+
import { notifyExtensionsInvoke, queueInvocation, startRuntimeApiServer, waitForResponse, } from "../runtime-api/server.js";
|
|
27
|
+
/**
|
|
28
|
+
* Default idle timeout before proactively stopping containers.
|
|
29
|
+
* Set slightly below the poll timeout (240s) to stop container before
|
|
30
|
+
* the Lambda RIC gets a 503 error.
|
|
31
|
+
*/
|
|
32
|
+
const DEFAULT_IDLE_TIMEOUT_MS = 230_000; // 230 seconds (~4 minutes)
|
|
33
|
+
/**
|
|
34
|
+
* Check if bootstrap stack version matches the expected version.
|
|
35
|
+
* Returns true if version matches, false if missing or mismatched.
|
|
36
|
+
*/
|
|
37
|
+
const checkBootstrapVersion = (qualifier) => Effect.gen(function* () {
|
|
38
|
+
const ssmClient = new SSMClient({});
|
|
39
|
+
const basePath = `${SSM_BASE_PATH}/${qualifier}`;
|
|
40
|
+
const result = yield* Effect.tryPromise({
|
|
41
|
+
try: async () => {
|
|
42
|
+
const response = await ssmClient.send(new GetParameterCommand({ Name: `${basePath}/version` }));
|
|
43
|
+
return response.Parameter?.Value;
|
|
44
|
+
},
|
|
45
|
+
catch: () => null,
|
|
46
|
+
}).pipe(Effect.catchAll(() => Effect.succeed(null)));
|
|
47
|
+
if (result === null) {
|
|
48
|
+
yield* Effect.logInfo("Bootstrap stack version parameter not found");
|
|
49
|
+
return false;
|
|
50
|
+
}
|
|
51
|
+
if (result !== BOOTSTRAP_VERSION) {
|
|
52
|
+
yield* Effect.logInfo(`Bootstrap stack version mismatch: found ${result}, expected ${BOOTSTRAP_VERSION}`);
|
|
53
|
+
return false;
|
|
54
|
+
}
|
|
55
|
+
return true;
|
|
56
|
+
});
|
|
57
|
+
/**
|
|
58
|
+
* Run the bootstrap stack deployment.
|
|
59
|
+
*/
|
|
60
|
+
const runBootstrap = (options) => Effect.gen(function* () {
|
|
61
|
+
yield* Effect.logInfo("Running bootstrap stack deployment...");
|
|
62
|
+
// Resolve the CDK app path relative to this module
|
|
63
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
64
|
+
const __dirname = path.dirname(__filename);
|
|
65
|
+
const cdkAppPath = path.resolve(__dirname, "..", "cdk-app.js");
|
|
66
|
+
const args = [
|
|
67
|
+
"cdk",
|
|
68
|
+
"deploy",
|
|
69
|
+
BOOTSTRAP_STACK_NAME,
|
|
70
|
+
"--require-approval",
|
|
71
|
+
"never",
|
|
72
|
+
"--app",
|
|
73
|
+
`bun ${cdkAppPath}`,
|
|
74
|
+
];
|
|
75
|
+
if (options.profile) {
|
|
76
|
+
args.push("--profile", options.profile);
|
|
77
|
+
}
|
|
78
|
+
const env = {
|
|
79
|
+
...process.env,
|
|
80
|
+
};
|
|
81
|
+
if (options.region) {
|
|
82
|
+
env.AWS_REGION = options.region;
|
|
83
|
+
env.CDK_DEFAULT_REGION = options.region;
|
|
84
|
+
}
|
|
85
|
+
yield* Effect.logInfo(`Running: npx ${args.join(" ")}`);
|
|
86
|
+
const command = PlatformCommand.make("npx", ...args).pipe(PlatformCommand.env(env), PlatformCommand.stdin("inherit"));
|
|
87
|
+
const proc = yield* PlatformCommand.start(command);
|
|
88
|
+
// Stream output to console
|
|
89
|
+
yield* Stream.merge(proc.stdout, proc.stderr).pipe(Stream.decodeText(), Stream.splitLines, Stream.runForEach((line) => Effect.sync(() => {
|
|
90
|
+
process.stdout.write(`${line}\n`);
|
|
91
|
+
})));
|
|
92
|
+
const exitCode = yield* proc.exitCode;
|
|
93
|
+
if (exitCode !== 0) {
|
|
94
|
+
return yield* Effect.fail(new Error(`Bootstrap deployment failed with exit code ${exitCode}`));
|
|
95
|
+
}
|
|
96
|
+
yield* Effect.logInfo("Bootstrap stack deployed successfully!");
|
|
97
|
+
}).pipe(Effect.scoped);
|
|
98
|
+
/**
|
|
99
|
+
* Ensure bootstrap stack is deployed with correct version.
|
|
100
|
+
* Automatically deploys if missing or outdated.
|
|
101
|
+
*/
|
|
102
|
+
const ensureBootstrap = (options) => Effect.gen(function* () {
|
|
103
|
+
yield* Effect.logDebug("[Local] Checking bootstrap stack version...");
|
|
104
|
+
const versionOk = yield* checkBootstrapVersion(options.qualifier);
|
|
105
|
+
if (!versionOk) {
|
|
106
|
+
yield* Effect.logInfo("[Local] Bootstrap stack needs to be deployed or updated.");
|
|
107
|
+
yield* runBootstrap({ profile: options.profile, region: options.region });
|
|
108
|
+
}
|
|
109
|
+
else {
|
|
110
|
+
yield* Effect.logDebug("[Local] Bootstrap stack version OK.");
|
|
111
|
+
}
|
|
112
|
+
});
|
|
113
|
+
/**
|
|
114
|
+
* Read AppSync endpoints from SSM.
|
|
115
|
+
*/
|
|
116
|
+
const getAppSyncEndpoints = (qualifier) => Effect.gen(function* () {
|
|
117
|
+
const ssmClient = new SSMClient({});
|
|
118
|
+
const basePath = `${SSM_BASE_PATH}/${qualifier}`;
|
|
119
|
+
const getParam = (name) => Effect.tryPromise({
|
|
120
|
+
try: async () => {
|
|
121
|
+
const result = await ssmClient.send(new GetParameterCommand({ Name: `${basePath}/${name}` }));
|
|
122
|
+
return result.Parameter?.Value ?? "";
|
|
123
|
+
},
|
|
124
|
+
catch: (error) => new Error(`Failed to get SSM parameter ${name}: ${String(error)}`),
|
|
125
|
+
});
|
|
126
|
+
const httpEndpoint = yield* getParam("http-endpoint");
|
|
127
|
+
const realtimeEndpoint = yield* getParam("realtime-endpoint");
|
|
128
|
+
return { httpEndpoint, realtimeEndpoint };
|
|
129
|
+
});
|
|
130
|
+
/**
|
|
131
|
+
* CloudFormation stack name tag (set automatically by CDK)
|
|
132
|
+
*/
|
|
133
|
+
const CFN_STACK_NAME_TAG = "aws:cloudformation:stack-name";
|
|
134
|
+
/**
|
|
135
|
+
* Discover Lambda functions with live-lambda tags.
|
|
136
|
+
* @param stackFilter - Optional list of stack names to filter by
|
|
137
|
+
*/
|
|
138
|
+
const discoverFunctions = (stackFilter) => Effect.gen(function* () {
|
|
139
|
+
const lambdaClient = new LambdaClient({});
|
|
140
|
+
const functions = [];
|
|
141
|
+
// List all functions
|
|
142
|
+
let nextMarker;
|
|
143
|
+
do {
|
|
144
|
+
const listResponse = yield* Effect.tryPromise({
|
|
145
|
+
try: () => lambdaClient.send(new ListFunctionsCommand({ Marker: nextMarker })),
|
|
146
|
+
catch: (error) => new Error(`Failed to list functions: ${String(error)}`),
|
|
147
|
+
});
|
|
148
|
+
for (const fn of listResponse.Functions ?? []) {
|
|
149
|
+
// Get tags for each function
|
|
150
|
+
const tagsResult = yield* Effect.tryPromise({
|
|
151
|
+
try: () => lambdaClient.send(new ListTagsCommand({ Resource: fn.FunctionArn })),
|
|
152
|
+
catch: () => new Error(`Failed to get tags for ${fn.FunctionName}`),
|
|
153
|
+
}).pipe(Effect.catchAll(() => Effect.succeed({ Tags: {} })));
|
|
154
|
+
const tags = tagsResult.Tags ?? {};
|
|
155
|
+
// Check for live-lambda tag (handler) or docker-context tag
|
|
156
|
+
if (tags[LIVE_LAMBDA_TAG] || tags[LIVE_LAMBDA_DOCKER_TAG]) {
|
|
157
|
+
// Filter by stack name if specified
|
|
158
|
+
if (stackFilter && stackFilter.length > 0) {
|
|
159
|
+
const stackName = tags[CFN_STACK_NAME_TAG];
|
|
160
|
+
if (!stackName || !stackFilter.includes(stackName)) {
|
|
161
|
+
continue;
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
// Determine architecture from Lambda config
|
|
165
|
+
const architectures = fn.Architectures ?? ["x86_64"];
|
|
166
|
+
const architecture = architectures.includes("arm64")
|
|
167
|
+
? "arm64"
|
|
168
|
+
: "x86_64";
|
|
169
|
+
const discovered = {
|
|
170
|
+
functionName: fn.FunctionName,
|
|
171
|
+
functionArn: fn.FunctionArn,
|
|
172
|
+
localHandler: tags[LIVE_LAMBDA_TAG] ?? "",
|
|
173
|
+
dockerContextPath: tags[LIVE_LAMBDA_DOCKER_TAG],
|
|
174
|
+
memoryMB: fn.MemorySize ?? 128,
|
|
175
|
+
architecture,
|
|
176
|
+
};
|
|
177
|
+
functions.push(discovered);
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
nextMarker = listResponse.NextMarker;
|
|
181
|
+
} while (nextMarker);
|
|
182
|
+
return functions;
|
|
183
|
+
});
|
|
184
|
+
/**
|
|
185
|
+
* Start a long-running Docker container for a function.
|
|
186
|
+
* Builds the image from the local Docker context path, then runs it.
|
|
187
|
+
* The container continuously polls our Runtime API for invocations.
|
|
188
|
+
* Uses Effect.forkDaemon to ensure the container runs independently with context.
|
|
189
|
+
*/
|
|
190
|
+
const startFunctionContainer = (fn, port, projectRoot, additionalEnv, invocationContexts) => Effect.gen(function* () {
|
|
191
|
+
if (!fn.dockerContextPath) {
|
|
192
|
+
return yield* Effect.fail(new Error(`Function ${fn.functionName} has no Docker context path`));
|
|
193
|
+
}
|
|
194
|
+
const docker = yield* Docker;
|
|
195
|
+
const dockerRuntime = yield* docker.getRuntimeInfo();
|
|
196
|
+
const runtimeApiHost = dockerRuntime.isDockerDesktop
|
|
197
|
+
? "host.docker.internal"
|
|
198
|
+
: "runtime.api";
|
|
199
|
+
// Generate a local image name from function name
|
|
200
|
+
const imageName = `live-lambda-${fn.functionName.toLowerCase().replace(/[^a-z0-9-]/g, "-")}`;
|
|
201
|
+
// Resolve the context path relative to project root
|
|
202
|
+
const contextPath = fn.dockerContextPath.startsWith("/")
|
|
203
|
+
? fn.dockerContextPath
|
|
204
|
+
: `${projectRoot}/${fn.dockerContextPath}`;
|
|
205
|
+
// Determine platform from architecture
|
|
206
|
+
const platform = fn.architecture === "arm64" ? "linux/arm64" : "linux/amd64";
|
|
207
|
+
// Build the Docker image from local context
|
|
208
|
+
yield* docker
|
|
209
|
+
.build({
|
|
210
|
+
contextPath,
|
|
211
|
+
imageName,
|
|
212
|
+
platform,
|
|
213
|
+
})
|
|
214
|
+
.pipe(Effect.scoped);
|
|
215
|
+
// Inspect the image to get original entrypoint/cmd for extension wrapper
|
|
216
|
+
const imageConfig = yield* docker.inspect(imageName).pipe(Effect.scoped);
|
|
217
|
+
const extensionWrapper = buildExtensionWrapperCommand(imageConfig.entrypoint, imageConfig.cmd);
|
|
218
|
+
const containerConfig = makeLambdaContainerConfig({
|
|
219
|
+
imageUri: imageName,
|
|
220
|
+
runtimeApiHost,
|
|
221
|
+
runtimeApiPort: port,
|
|
222
|
+
functionName: fn.functionName,
|
|
223
|
+
functionVersion: "$LATEST",
|
|
224
|
+
memoryMB: fn.memoryMB,
|
|
225
|
+
timeoutSeconds: 3600, // Long timeout - container stays running
|
|
226
|
+
platform,
|
|
227
|
+
additionalEnv,
|
|
228
|
+
invocationContexts,
|
|
229
|
+
});
|
|
230
|
+
// Apply extension wrapper to start extensions before the app
|
|
231
|
+
containerConfig.entrypoint = extensionWrapper.entrypoint;
|
|
232
|
+
containerConfig.command = extensionWrapper.command;
|
|
233
|
+
yield* Effect.logInfo(`Starting container for ${fn.functionName} on port ${port}`);
|
|
234
|
+
// Use Effect.forkDaemon to run the container independently with context preserved
|
|
235
|
+
// Effect.scoped provides the scope needed by docker.run
|
|
236
|
+
const fiber = yield* docker.run(containerConfig).pipe(Effect.scoped, Effect.map(() => undefined), Effect.forkDaemon);
|
|
237
|
+
return fiber;
|
|
238
|
+
}).pipe(Effect.provide(DockerLive));
|
|
239
|
+
// Type guards for response types
|
|
240
|
+
const isLambdaResponse = (r) => "body" in r && !("errorType" in r);
|
|
241
|
+
const isLambdaError = (r) => "requestId" in r && "errorType" in r;
|
|
242
|
+
const isLambdaInitError = (r) => !("requestId" in r) && "errorType" in r;
|
|
243
|
+
/**
|
|
244
|
+
* Reset the idle timer for a container.
|
|
245
|
+
* After the timeout expires with no new responses, the container is stopped.
|
|
246
|
+
* This prevents the confusing 503 error from the Lambda RIC when the poll times out.
|
|
247
|
+
*/
|
|
248
|
+
const resetIdleTimer = (container, containersRef, idleTimeoutMs) => Effect.gen(function* () {
|
|
249
|
+
// Cancel existing timer if any
|
|
250
|
+
if (container.idleTimerFiber) {
|
|
251
|
+
yield* Fiber.interrupt(container.idleTimerFiber).pipe(Effect.catchAll(() => Effect.void));
|
|
252
|
+
container.idleTimerFiber = null;
|
|
253
|
+
}
|
|
254
|
+
// Start new timer - use never for error type since we catch all errors
|
|
255
|
+
const timerFiber = yield* Effect.gen(function* () {
|
|
256
|
+
yield* Effect.sleep(Duration.millis(idleTimeoutMs));
|
|
257
|
+
// Timer expired - stop container proactively
|
|
258
|
+
yield* Effect.logInfo(`[Local] Stopping idle container ${container.fn.functionName} (will restart on next invocation)`);
|
|
259
|
+
// Stop the container using Docker with fast timeout (1s)
|
|
260
|
+
// The container may be blocked on long-polling, so we need to force kill
|
|
261
|
+
const docker = yield* Docker;
|
|
262
|
+
yield* docker.stop(container.containerName, 1).pipe(Effect.scoped, Effect.catchAll(() => Effect.void));
|
|
263
|
+
// Remove from map so next invocation creates fresh container
|
|
264
|
+
const currentContainers = yield* Ref.get(containersRef);
|
|
265
|
+
currentContainers.delete(container.fn.functionName);
|
|
266
|
+
yield* Ref.set(containersRef, currentContainers);
|
|
267
|
+
}).pipe(Effect.provide(DockerLive), Effect.catchAll(() => Effect.void), Effect.forkDaemon);
|
|
268
|
+
container.idleTimerFiber = timerFiber;
|
|
269
|
+
});
|
|
270
|
+
/**
|
|
271
|
+
* Process responses from a container and dispatch to waiting callers.
|
|
272
|
+
* After each response, resets the idle timer to proactively stop the container
|
|
273
|
+
* before the poll timeout causes confusing Lambda RIC errors.
|
|
274
|
+
*/
|
|
275
|
+
const processContainerResponses = (container, containersRef, idleTimeoutMs, client, invocationContexts) => Effect.gen(function* () {
|
|
276
|
+
const responseChannel = buildChannelName.response(container.fn.functionName);
|
|
277
|
+
// Continuously process responses from the container
|
|
278
|
+
while (true) {
|
|
279
|
+
const result = yield* waitForResponse(container.runtimeState);
|
|
280
|
+
let response;
|
|
281
|
+
if (isLambdaResponse(result)) {
|
|
282
|
+
response = {
|
|
283
|
+
type: "response",
|
|
284
|
+
requestId: result.requestId,
|
|
285
|
+
result: result.body,
|
|
286
|
+
};
|
|
287
|
+
}
|
|
288
|
+
else if (isLambdaError(result)) {
|
|
289
|
+
response = {
|
|
290
|
+
type: "response",
|
|
291
|
+
requestId: result.requestId,
|
|
292
|
+
error: {
|
|
293
|
+
errorType: result.errorType,
|
|
294
|
+
errorMessage: result.errorMessage,
|
|
295
|
+
stackTrace: result.stackTrace,
|
|
296
|
+
},
|
|
297
|
+
};
|
|
298
|
+
}
|
|
299
|
+
else if (isLambdaInitError(result)) {
|
|
300
|
+
yield* Effect.logError(`[Local] Init error: ${result.errorType}: ${result.errorMessage}`);
|
|
301
|
+
continue;
|
|
302
|
+
}
|
|
303
|
+
else {
|
|
304
|
+
continue;
|
|
305
|
+
}
|
|
306
|
+
// Send response back via AppSync
|
|
307
|
+
yield* client.publishResponse(responseChannel, response);
|
|
308
|
+
yield* Effect.logDebug(`[Local] Sent response for ${response.requestId}`);
|
|
309
|
+
// Print end marker with timing
|
|
310
|
+
const ctx = invocationContexts.get(response.requestId);
|
|
311
|
+
if (ctx) {
|
|
312
|
+
const durationMs = Date.now() - ctx.start;
|
|
313
|
+
if (response.error) {
|
|
314
|
+
console.log(`[${ctx.num}] \u2514\u2500\u2500 \u2717 ${response.error.errorType}: ${response.error.errorMessage} (${durationMs}ms) \u2500\u2500`);
|
|
315
|
+
}
|
|
316
|
+
else {
|
|
317
|
+
console.log(`[${ctx.num}] \u2514\u2500\u2500 \u2713 Done (${durationMs}ms) \u2500\u2500`);
|
|
318
|
+
}
|
|
319
|
+
invocationContexts.delete(response.requestId);
|
|
320
|
+
}
|
|
321
|
+
// Reset idle timer - container will be stopped if no new invocations arrive
|
|
322
|
+
yield* resetIdleTimer(container, containersRef, idleTimeoutMs);
|
|
323
|
+
}
|
|
324
|
+
});
|
|
325
|
+
/**
|
|
326
|
+
* Start a Node.js worker process for a function.
|
|
327
|
+
* Spawns Bun to run the runtime wrapper with the handler path.
|
|
328
|
+
* The worker continuously polls our Runtime API for invocations.
|
|
329
|
+
*/
|
|
330
|
+
const startNodejsWorker = (fn, port, projectRoot, env, invocationContexts) => Effect.gen(function* () {
|
|
331
|
+
if (!fn.localHandler) {
|
|
332
|
+
return yield* Effect.fail(new Error(`Function ${fn.functionName} has no local handler path`));
|
|
333
|
+
}
|
|
334
|
+
// Resolve the runtime wrapper path relative to this module
|
|
335
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
336
|
+
const __dirname = path.dirname(__filename);
|
|
337
|
+
const runtimeWrapperPath = path.resolve(__dirname, "..", "runtime-wrapper", "nodejs-runtime.js");
|
|
338
|
+
// Get absolute path to bun for pure env isolation (no PATH dependency)
|
|
339
|
+
const bunPath = Bun.which("bun");
|
|
340
|
+
if (!bunPath) {
|
|
341
|
+
return yield* Effect.fail(new Error("Could not find 'bun' executable in PATH"));
|
|
342
|
+
}
|
|
343
|
+
// Build environment for the worker process
|
|
344
|
+
// Pure isolation: only env from bridge + local overrides, no local PATH/HOME
|
|
345
|
+
const workerEnv = {
|
|
346
|
+
// Start with env vars from the bridge Lambda (AWS credentials, user-defined vars)
|
|
347
|
+
...env,
|
|
348
|
+
// Local overrides (these are set by the daemon, not from bridge)
|
|
349
|
+
AWS_LAMBDA_RUNTIME_API: `localhost:${port}`,
|
|
350
|
+
_HANDLER: fn.localHandler,
|
|
351
|
+
LAMBDA_TASK_ROOT: projectRoot,
|
|
352
|
+
// Memory limit for context object
|
|
353
|
+
AWS_LAMBDA_FUNCTION_MEMORY_SIZE: String(fn.memoryMB),
|
|
354
|
+
};
|
|
355
|
+
yield* Effect.logDebug(`[Local] Starting Node.js worker for ${fn.functionName} on port ${port}`);
|
|
356
|
+
yield* Effect.logDebug(`[Local] Handler: ${fn.localHandler}`);
|
|
357
|
+
// Spawn Bun with --watch to automatically restart when handler files change
|
|
358
|
+
// This enables hot-reload without needing to restart the daemon
|
|
359
|
+
// Use absolute bun path for pure env isolation (no PATH dependency)
|
|
360
|
+
const workerProcess = spawn(bunPath, ["--watch", runtimeWrapperPath], {
|
|
361
|
+
cwd: projectRoot,
|
|
362
|
+
env: workerEnv,
|
|
363
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
364
|
+
});
|
|
365
|
+
// Pattern to parse Lambda log format: TIMESTAMP\tREQUEST_ID\tLEVEL\tMESSAGE
|
|
366
|
+
// Lambda uses tabs between fields. Captures: [1] = request ID, [2] = level + message
|
|
367
|
+
const lambdaLogPattern = /^(\d{4}-\d{2}-\d{2}T[\d:.]+Z)[\t\s]+([0-9a-f-]{36})[\t\s]+(.*)$/i;
|
|
368
|
+
// Helper to format log line with invocation prefix
|
|
369
|
+
const formatLine = (rawLine) => {
|
|
370
|
+
// Strip carriage returns that can cause terminal corruption
|
|
371
|
+
const line = rawLine.replace(/\r/g, "");
|
|
372
|
+
const match = lambdaLogPattern.exec(line);
|
|
373
|
+
if (match) {
|
|
374
|
+
const requestId = match[2];
|
|
375
|
+
const ctx = invocationContexts.get(requestId);
|
|
376
|
+
if (ctx) {
|
|
377
|
+
// Strip timestamp and request ID, keep just LEVEL MESSAGE
|
|
378
|
+
return { prefix: `[${ctx.num}]`, content: match[3] };
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
return { prefix: "[Worker]", content: line };
|
|
382
|
+
};
|
|
383
|
+
// Forward stdout/stderr with invocation number prefix
|
|
384
|
+
workerProcess.stdout?.on("data", (data) => {
|
|
385
|
+
const lines = data.toString().trim().split("\n");
|
|
386
|
+
for (const rawLine of lines) {
|
|
387
|
+
const { prefix, content } = formatLine(rawLine);
|
|
388
|
+
console.log(`${prefix} ${content}`);
|
|
389
|
+
}
|
|
390
|
+
});
|
|
391
|
+
workerProcess.stderr?.on("data", (data) => {
|
|
392
|
+
const lines = data.toString().trim().split("\n");
|
|
393
|
+
for (const rawLine of lines) {
|
|
394
|
+
const { prefix, content } = formatLine(rawLine);
|
|
395
|
+
console.error(`${prefix} ${content}`);
|
|
396
|
+
}
|
|
397
|
+
});
|
|
398
|
+
workerProcess.on("error", (err) => {
|
|
399
|
+
Effect.runSync(Effect.logError(`Worker error: ${err.message}`).pipe(Effect.annotateLogs("function", fn.functionName)));
|
|
400
|
+
});
|
|
401
|
+
workerProcess.on("close", (code) => {
|
|
402
|
+
Effect.runSync(Effect.logInfo(`Worker exited with code ${code}`).pipe(Effect.annotateLogs("function", fn.functionName)));
|
|
403
|
+
});
|
|
404
|
+
return workerProcess;
|
|
405
|
+
});
|
|
406
|
+
/**
|
|
407
|
+
* Process responses from a Node.js worker and dispatch to waiting callers.
|
|
408
|
+
*/
|
|
409
|
+
const processWorkerResponses = (worker, client, invocationContexts) => Effect.gen(function* () {
|
|
410
|
+
const responseChannel = buildChannelName.response(worker.fn.functionName);
|
|
411
|
+
// Continuously process responses from the worker
|
|
412
|
+
while (true) {
|
|
413
|
+
const result = yield* waitForResponse(worker.runtimeState);
|
|
414
|
+
let response;
|
|
415
|
+
if (isLambdaResponse(result)) {
|
|
416
|
+
response = {
|
|
417
|
+
type: "response",
|
|
418
|
+
requestId: result.requestId,
|
|
419
|
+
result: result.body,
|
|
420
|
+
};
|
|
421
|
+
}
|
|
422
|
+
else if (isLambdaError(result)) {
|
|
423
|
+
response = {
|
|
424
|
+
type: "response",
|
|
425
|
+
requestId: result.requestId,
|
|
426
|
+
error: {
|
|
427
|
+
errorType: result.errorType,
|
|
428
|
+
errorMessage: result.errorMessage,
|
|
429
|
+
stackTrace: result.stackTrace,
|
|
430
|
+
},
|
|
431
|
+
};
|
|
432
|
+
}
|
|
433
|
+
else if (isLambdaInitError(result)) {
|
|
434
|
+
yield* Effect.logError(`[Local] Worker init error: ${result.errorType}: ${result.errorMessage}`);
|
|
435
|
+
continue;
|
|
436
|
+
}
|
|
437
|
+
else {
|
|
438
|
+
continue;
|
|
439
|
+
}
|
|
440
|
+
// Send response back via AppSync
|
|
441
|
+
yield* client.publishResponse(responseChannel, response);
|
|
442
|
+
yield* Effect.logDebug(`[Local] Sent response for ${response.requestId}`);
|
|
443
|
+
// Print end marker with timing
|
|
444
|
+
const ctx = invocationContexts.get(response.requestId);
|
|
445
|
+
if (ctx) {
|
|
446
|
+
const durationMs = Date.now() - ctx.start;
|
|
447
|
+
if (response.error) {
|
|
448
|
+
console.log(`[${ctx.num}] \u2514\u2500\u2500 \u2717 ${response.error.errorType}: ${response.error.errorMessage} (${durationMs}ms) \u2500\u2500`);
|
|
449
|
+
}
|
|
450
|
+
else {
|
|
451
|
+
console.log(`[${ctx.num}] \u2514\u2500\u2500 \u2713 Done (${durationMs}ms) \u2500\u2500`);
|
|
452
|
+
}
|
|
453
|
+
invocationContexts.delete(response.requestId);
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
});
|
|
457
|
+
/**
|
|
458
|
+
* Environment variables to completely filter out from invocation env.
|
|
459
|
+
* These are not relevant for local execution.
|
|
460
|
+
*/
|
|
461
|
+
const ENV_VARS_TO_FILTER = new Set(["AWS_LAMBDA_LOG_STREAM_NAME"]);
|
|
462
|
+
/**
|
|
463
|
+
* Environment variables to ignore when calculating if env has changed.
|
|
464
|
+
* These change frequently but don't require a container restart.
|
|
465
|
+
*
|
|
466
|
+
* TODO: AWS credentials (ACCESS_KEY_ID, SECRET_ACCESS_KEY, SESSION_TOKEN)
|
|
467
|
+
* do expire and will need refreshing. For now we ignore them to avoid
|
|
468
|
+
* unnecessary restarts, but we should implement credential refresh logic
|
|
469
|
+
* that updates the container's credentials without a full restart.
|
|
470
|
+
*/
|
|
471
|
+
const ENV_VARS_TO_IGNORE_IN_DIFF = new Set([
|
|
472
|
+
"AWS_ACCESS_KEY_ID",
|
|
473
|
+
"AWS_SECRET_ACCESS_KEY",
|
|
474
|
+
"AWS_SESSION_TOKEN",
|
|
475
|
+
]);
|
|
476
|
+
/**
|
|
477
|
+
* Filter out irrelevant env vars from invocation environment.
|
|
478
|
+
*/
|
|
479
|
+
const sanitizeInvocationEnv = (env) => {
|
|
480
|
+
const result = {};
|
|
481
|
+
for (const [key, value] of Object.entries(env)) {
|
|
482
|
+
if (!ENV_VARS_TO_FILTER.has(key)) {
|
|
483
|
+
result[key] = value;
|
|
484
|
+
}
|
|
485
|
+
}
|
|
486
|
+
return result;
|
|
487
|
+
};
|
|
488
|
+
/**
|
|
489
|
+
* Compute which environment variables have changed between two env objects.
|
|
490
|
+
* Returns a human-readable summary of the changes.
|
|
491
|
+
* Ignores env vars in ENV_VARS_TO_IGNORE_IN_DIFF.
|
|
492
|
+
*/
|
|
493
|
+
const getEnvDiff = (oldEnv, newEnv) => {
|
|
494
|
+
const changes = [];
|
|
495
|
+
// Check for added or modified keys
|
|
496
|
+
for (const key of Object.keys(newEnv)) {
|
|
497
|
+
if (ENV_VARS_TO_IGNORE_IN_DIFF.has(key))
|
|
498
|
+
continue;
|
|
499
|
+
if (!(key in oldEnv)) {
|
|
500
|
+
changes.push(`+${key}`);
|
|
501
|
+
}
|
|
502
|
+
else if (oldEnv[key] !== newEnv[key]) {
|
|
503
|
+
changes.push(`~${key}`);
|
|
504
|
+
}
|
|
505
|
+
}
|
|
506
|
+
// Check for removed keys
|
|
507
|
+
for (const key of Object.keys(oldEnv)) {
|
|
508
|
+
if (ENV_VARS_TO_IGNORE_IN_DIFF.has(key))
|
|
509
|
+
continue;
|
|
510
|
+
if (!(key in newEnv)) {
|
|
511
|
+
changes.push(`-${key}`);
|
|
512
|
+
}
|
|
513
|
+
}
|
|
514
|
+
return changes.join(", ");
|
|
515
|
+
};
|
|
516
|
+
/**
|
|
517
|
+
* Ensure a Node.js worker is started for a function.
|
|
518
|
+
* If the worker already exists, return it.
|
|
519
|
+
* Otherwise, create the Runtime API server, add to workers map, and start the worker.
|
|
520
|
+
*/
|
|
521
|
+
const ensureWorkerStarted = (fn, invocationEnv, workersRef, serverScope, projectRoot, appSyncClient, invocationContexts) => Effect.gen(function* () {
|
|
522
|
+
// Check if worker already exists
|
|
523
|
+
const currentWorkers = yield* Ref.get(workersRef);
|
|
524
|
+
const existing = currentWorkers.get(fn.functionName);
|
|
525
|
+
if (existing) {
|
|
526
|
+
// Check if env vars have changed (e.g., after CDK redeploy)
|
|
527
|
+
const envDiff = getEnvDiff(existing.env, invocationEnv);
|
|
528
|
+
if (envDiff) {
|
|
529
|
+
yield* Effect.logInfo(`[Local] Environment changed for ${fn.functionName}, restarting worker... (${envDiff})`);
|
|
530
|
+
// Kill the old worker
|
|
531
|
+
yield* Effect.try(() => existing.workerProcess.kill("SIGTERM")).pipe(Effect.catchAll(() => Effect.void));
|
|
532
|
+
// Remove from map so we create a new one below
|
|
533
|
+
currentWorkers.delete(fn.functionName);
|
|
534
|
+
yield* Ref.set(workersRef, currentWorkers);
|
|
535
|
+
}
|
|
536
|
+
else {
|
|
537
|
+
return existing;
|
|
538
|
+
}
|
|
539
|
+
}
|
|
540
|
+
// Worker doesn't exist - start it lazily
|
|
541
|
+
yield* Effect.logDebug(`[Local] Starting Node.js worker for first invocation of ${fn.functionName}...`);
|
|
542
|
+
// Create Runtime API server on ephemeral port
|
|
543
|
+
const { port, state: runtimeState } = yield* startRuntimeApiServer({
|
|
544
|
+
functionMetadata: {
|
|
545
|
+
functionName: fn.functionName,
|
|
546
|
+
functionVersion: "$LATEST",
|
|
547
|
+
handler: fn.localHandler,
|
|
548
|
+
},
|
|
549
|
+
}).pipe(Effect.provideService(Scope.Scope, serverScope));
|
|
550
|
+
// Create worker object (without process initially - will be set after start)
|
|
551
|
+
// We add to map BEFORE starting worker to handle concurrent invocations
|
|
552
|
+
const worker = {
|
|
553
|
+
fn,
|
|
554
|
+
runtimeState,
|
|
555
|
+
port,
|
|
556
|
+
workerProcess: undefined, // Will be set shortly
|
|
557
|
+
env: invocationEnv,
|
|
558
|
+
};
|
|
559
|
+
// Add to map immediately to prevent race conditions
|
|
560
|
+
currentWorkers.set(fn.functionName, worker);
|
|
561
|
+
yield* Ref.set(workersRef, currentWorkers);
|
|
562
|
+
// Start the worker process - remove from map on failure
|
|
563
|
+
const workerProcess = yield* startNodejsWorker(fn, port, projectRoot, invocationEnv, invocationContexts).pipe(Effect.catchAll((error) => Effect.gen(function* () {
|
|
564
|
+
// Remove broken worker from map on failure
|
|
565
|
+
yield* Effect.logError(`[Local] Failed to start worker for ${fn.functionName}: ${error}`);
|
|
566
|
+
const current = yield* Ref.get(workersRef);
|
|
567
|
+
current.delete(fn.functionName);
|
|
568
|
+
yield* Ref.set(workersRef, current);
|
|
569
|
+
return yield* Effect.fail(error);
|
|
570
|
+
})));
|
|
571
|
+
// Update worker with the process
|
|
572
|
+
worker.workerProcess = workerProcess;
|
|
573
|
+
// Start processing responses in the background
|
|
574
|
+
Effect.runFork(processWorkerResponses(worker, appSyncClient, invocationContexts));
|
|
575
|
+
return worker;
|
|
576
|
+
});
|
|
577
|
+
/**
|
|
578
|
+
* Ensure a container is started for a function.
|
|
579
|
+
* If the container already exists, return it.
|
|
580
|
+
* If the env vars have changed, restart the container.
|
|
581
|
+
* Otherwise, create the Runtime API server, add to containers map, and start the container.
|
|
582
|
+
*/
|
|
583
|
+
const ensureContainerStarted = (fn, invocationEnv, containersRef, serverScope, projectRoot, idleTimeoutMs, pollTimeoutMs, appSyncClient, invocationContexts) => Effect.gen(function* () {
|
|
584
|
+
// Check if container already exists
|
|
585
|
+
const currentContainers = yield* Ref.get(containersRef);
|
|
586
|
+
const existing = currentContainers.get(fn.functionName);
|
|
587
|
+
if (existing) {
|
|
588
|
+
// Check if env vars have changed (e.g., after CDK redeploy)
|
|
589
|
+
const envDiff = getEnvDiff(existing.env, invocationEnv);
|
|
590
|
+
if (envDiff) {
|
|
591
|
+
// IMPORTANT: Update env immediately (before any yields) to prevent
|
|
592
|
+
// other concurrent invocations from also detecting the change and
|
|
593
|
+
// triggering duplicate restarts
|
|
594
|
+
existing.env = invocationEnv;
|
|
595
|
+
// If already restarting for env change, just return the existing container
|
|
596
|
+
// The invocation will be queued and picked up by the new container
|
|
597
|
+
if (existing.isRebuilding) {
|
|
598
|
+
yield* Effect.logDebug(`[Local] Environment change restart already in progress for ${fn.functionName}, queueing invocation`);
|
|
599
|
+
return existing;
|
|
600
|
+
}
|
|
601
|
+
yield* Effect.logInfo(`[Local] Environment changed for ${fn.functionName}, restarting container... (${envDiff})`);
|
|
602
|
+
existing.isRebuilding = true;
|
|
603
|
+
// Stop the Docker container forcefully (timeout=1s)
|
|
604
|
+
// This sends SIGTERM and then SIGKILL after 1 second
|
|
605
|
+
// We must do this BEFORE interrupting the fiber, because the fiber is
|
|
606
|
+
// blocked waiting for the docker process to exit
|
|
607
|
+
yield* Effect.gen(function* () {
|
|
608
|
+
const docker = yield* Docker;
|
|
609
|
+
yield* docker.stop(existing.containerName, 1).pipe(Effect.scoped);
|
|
610
|
+
}).pipe(Effect.provide(DockerLive), Effect.catchAll((error) => Effect.logDebug(`[Local] Error stopping container (may already be stopped): ${error}`)));
|
|
611
|
+
// Now interrupt the fiber (it should complete quickly since container stopped)
|
|
612
|
+
yield* Fiber.interrupt(existing.containerFiber).pipe(Effect.catchAll(() => Effect.void));
|
|
613
|
+
// Restart the container with new environment, reusing existing RuntimeAPI
|
|
614
|
+
const newFiber = yield* startFunctionContainer(fn, existing.port, projectRoot, invocationEnv, invocationContexts).pipe(Effect.catchAll((error) => Effect.gen(function* () {
|
|
615
|
+
yield* Effect.logError(`[Local] Failed to restart container for ${fn.functionName}: ${error}`);
|
|
616
|
+
existing.isRebuilding = false;
|
|
617
|
+
return yield* Effect.fail(error);
|
|
618
|
+
})));
|
|
619
|
+
// Update container with new fiber
|
|
620
|
+
existing.containerFiber = newFiber;
|
|
621
|
+
existing.isRebuilding = false;
|
|
622
|
+
return existing;
|
|
623
|
+
}
|
|
624
|
+
else {
|
|
625
|
+
return existing;
|
|
626
|
+
}
|
|
627
|
+
}
|
|
628
|
+
// Container doesn't exist - start it lazily
|
|
629
|
+
yield* Effect.logInfo(`[Local] Starting container for first invocation of ${fn.functionName}...`);
|
|
630
|
+
// Create Runtime API server on ephemeral port
|
|
631
|
+
// Use poll timeout that's shorter than idle timeout so container exits naturally
|
|
632
|
+
const { port, state: runtimeState } = yield* startRuntimeApiServer({
|
|
633
|
+
pollTimeoutMs,
|
|
634
|
+
functionMetadata: {
|
|
635
|
+
functionName: fn.functionName,
|
|
636
|
+
functionVersion: "$LATEST",
|
|
637
|
+
handler: "index.handler",
|
|
638
|
+
},
|
|
639
|
+
}).pipe(Effect.provideService(Scope.Scope, serverScope));
|
|
640
|
+
// Generate container name and image name
|
|
641
|
+
const containerName = `lambda-${fn.functionName.replace(/[^a-zA-Z0-9]/g, "-")}`;
|
|
642
|
+
const imageName = `live-lambda-${fn.functionName.toLowerCase().replace(/[^a-z0-9-]/g, "-")}`;
|
|
643
|
+
// Create container object (without fiber initially - will be set after start)
|
|
644
|
+
// We add to map BEFORE starting container to handle concurrent invocations
|
|
645
|
+
const container = {
|
|
646
|
+
fn,
|
|
647
|
+
runtimeState,
|
|
648
|
+
port,
|
|
649
|
+
containerFiber: undefined, // Will be set shortly
|
|
650
|
+
containerName,
|
|
651
|
+
imageName,
|
|
652
|
+
isRebuilding: false,
|
|
653
|
+
pendingResponses: new Map(),
|
|
654
|
+
idleTimerFiber: null,
|
|
655
|
+
env: invocationEnv,
|
|
656
|
+
};
|
|
657
|
+
// Add to map immediately to prevent race conditions
|
|
658
|
+
currentContainers.set(fn.functionName, container);
|
|
659
|
+
yield* Ref.set(containersRef, currentContainers);
|
|
660
|
+
// Build and start the container - remove from map on failure
|
|
661
|
+
const containerFiber = yield* startFunctionContainer(fn, port, projectRoot, invocationEnv, invocationContexts).pipe(Effect.catchAll((error) => Effect.gen(function* () {
|
|
662
|
+
// Remove broken container from map on failure
|
|
663
|
+
yield* Effect.logError(`[Local] Failed to start container for ${fn.functionName}: ${error}`);
|
|
664
|
+
const current = yield* Ref.get(containersRef);
|
|
665
|
+
current.delete(fn.functionName);
|
|
666
|
+
yield* Ref.set(containersRef, current);
|
|
667
|
+
return yield* Effect.fail(error);
|
|
668
|
+
})));
|
|
669
|
+
// Update container with the fiber
|
|
670
|
+
container.containerFiber = containerFiber;
|
|
671
|
+
// Start processing responses in the background using forkDaemon
|
|
672
|
+
yield* processContainerResponses(container, containersRef, idleTimeoutMs, appSyncClient, invocationContexts).pipe(Effect.forkDaemon);
|
|
673
|
+
return container;
|
|
674
|
+
});
|
|
675
|
+
/**
|
|
676
|
+
* Rebuild a Docker container after file changes.
|
|
677
|
+
* Invocations arriving during rebuild will be queued and picked up by the new container.
|
|
678
|
+
*/
|
|
679
|
+
const rebuildDockerContainer = (functionId, containersRef, projectRoot) => Effect.gen(function* () {
|
|
680
|
+
const containers = yield* Ref.get(containersRef);
|
|
681
|
+
const container = containers.get(functionId);
|
|
682
|
+
if (!container) {
|
|
683
|
+
yield* Effect.logInfo(`[Local] Cannot rebuild ${functionId} - container not found`);
|
|
684
|
+
return;
|
|
685
|
+
}
|
|
686
|
+
// Mark as rebuilding - invocations will still queue but we log it
|
|
687
|
+
container.isRebuilding = true;
|
|
688
|
+
yield* Effect.logDebug(`[Local] Rebuilding container for ${functionId}...`);
|
|
689
|
+
const docker = yield* Docker;
|
|
690
|
+
// Stop the existing container using Docker service
|
|
691
|
+
const containerId = container.containerName;
|
|
692
|
+
yield* Effect.logInfo(`[Local] Stopping container with name prefix: ${containerId}`);
|
|
693
|
+
const stopCount = yield* docker.stop(containerId).pipe(Effect.scoped, Effect.catchAll((error) => Effect.gen(function* () {
|
|
694
|
+
yield* Effect.logInfo(`Note: Container stop had issue: ${error}`);
|
|
695
|
+
return 0;
|
|
696
|
+
})));
|
|
697
|
+
yield* Effect.logDebug(`[Local] Stopped ${stopCount} container(s)`);
|
|
698
|
+
// Resolve the context path
|
|
699
|
+
const fn = container.fn;
|
|
700
|
+
const contextPath = fn.dockerContextPath?.startsWith("/")
|
|
701
|
+
? fn.dockerContextPath
|
|
702
|
+
: `${projectRoot}/${fn.dockerContextPath}`;
|
|
703
|
+
// Determine platform from architecture
|
|
704
|
+
const platform = fn.architecture === "arm64" ? "linux/arm64" : "linux/amd64";
|
|
705
|
+
// Rebuild the Docker image
|
|
706
|
+
yield* docker
|
|
707
|
+
.build({
|
|
708
|
+
contextPath,
|
|
709
|
+
imageName: container.imageName,
|
|
710
|
+
platform,
|
|
711
|
+
})
|
|
712
|
+
.pipe(Effect.scoped);
|
|
713
|
+
// Inspect the image to get original entrypoint/cmd for extension wrapper
|
|
714
|
+
const imageConfig = yield* docker
|
|
715
|
+
.inspect(container.imageName)
|
|
716
|
+
.pipe(Effect.scoped);
|
|
717
|
+
const extensionWrapper = buildExtensionWrapperCommand(imageConfig.entrypoint, imageConfig.cmd);
|
|
718
|
+
// Restart the container by triggering container startup
|
|
719
|
+
// The existing fiber will have exited when we stopped the container
|
|
720
|
+
// We need to start a new one
|
|
721
|
+
const dockerRuntime = yield* docker.getRuntimeInfo();
|
|
722
|
+
const runtimeApiHost = dockerRuntime.isDockerDesktop
|
|
723
|
+
? "host.docker.internal"
|
|
724
|
+
: "runtime.api";
|
|
725
|
+
yield* Effect.logInfo(`[Local] New container will connect to Runtime API at ${runtimeApiHost}:${container.port}`);
|
|
726
|
+
const containerConfig = makeLambdaContainerConfig({
|
|
727
|
+
imageUri: container.imageName,
|
|
728
|
+
runtimeApiHost,
|
|
729
|
+
runtimeApiPort: container.port,
|
|
730
|
+
functionName: fn.functionName,
|
|
731
|
+
functionVersion: "$LATEST",
|
|
732
|
+
memoryMB: fn.memoryMB,
|
|
733
|
+
timeoutSeconds: 3600,
|
|
734
|
+
platform,
|
|
735
|
+
});
|
|
736
|
+
// Apply extension wrapper to start extensions before the app
|
|
737
|
+
containerConfig.entrypoint = extensionWrapper.entrypoint;
|
|
738
|
+
containerConfig.command = extensionWrapper.command;
|
|
739
|
+
// Start the new container
|
|
740
|
+
yield* Effect.logDebug(`[Local] Starting new container for ${functionId}...`);
|
|
741
|
+
// Use Effect.forkDaemon to run the container independently
|
|
742
|
+
const newFiber = yield* docker.run(containerConfig).pipe(Effect.scoped, Effect.tap((result) => Effect.gen(function* () {
|
|
743
|
+
// Only suppress errors for expected exit codes from docker stop:
|
|
744
|
+
// 0: clean, 143: SIGTERM, 137: SIGKILL
|
|
745
|
+
const code = result.exitCode;
|
|
746
|
+
if (code !== 0 && code !== 143 && code !== 137) {
|
|
747
|
+
yield* Effect.logError(`Container for ${functionId} exited with code ${code}`);
|
|
748
|
+
yield* Effect.logError(`stderr: ${result.stderr}`);
|
|
749
|
+
}
|
|
750
|
+
})), Effect.map(() => undefined), Effect.catchAll((error) => Effect.logError(`Container error for ${functionId}: ${error}`)), Effect.forkDaemon);
|
|
751
|
+
// Update the container state with the new fiber
|
|
752
|
+
container.containerFiber = newFiber;
|
|
753
|
+
// Wait a moment for the container to start and begin polling
|
|
754
|
+
yield* Effect.sleep("2 seconds");
|
|
755
|
+
container.isRebuilding = false;
|
|
756
|
+
yield* Effect.logDebug(`[Local] Container rebuilt for ${functionId}`);
|
|
757
|
+
}).pipe(Effect.provide(DockerLive));
|
|
758
|
+
/**
|
|
759
|
+
* Handle incoming invocations for a Docker container function by queueing them.
|
|
760
|
+
* Starts the container lazily if not already running.
|
|
761
|
+
* Invocations are queued even if a rebuild is in progress - the new container will pick them up.
|
|
762
|
+
*/
|
|
763
|
+
const handleDockerInvocation = (fn, invocation, containersRef, serverScope, projectRoot, idleTimeoutMs, pollTimeoutMs, appSyncClient, invocationCounter, invocationContexts) => Effect.gen(function* () {
|
|
764
|
+
// Ensure container is started (lazy startup on first invocation)
|
|
765
|
+
// Use env from invocation (captured from bridge Lambda on first invoke)
|
|
766
|
+
const container = yield* ensureContainerStarted(fn, sanitizeInvocationEnv(invocation.env ?? {}), containersRef, serverScope, projectRoot, idleTimeoutMs, pollTimeoutMs, appSyncClient, invocationContexts);
|
|
767
|
+
// Assign invocation number and track context for logging
|
|
768
|
+
const invocationNum = yield* Ref.updateAndGet(invocationCounter, (n) => n + 1);
|
|
769
|
+
const startTime = Date.now();
|
|
770
|
+
invocationContexts.set(invocation.requestId, {
|
|
771
|
+
num: invocationNum,
|
|
772
|
+
start: startTime,
|
|
773
|
+
fn: fn.functionName,
|
|
774
|
+
isDocker: true,
|
|
775
|
+
});
|
|
776
|
+
// Print start marker
|
|
777
|
+
console.log(`\n[${invocationNum}] \u250c\u2500\u2500 [Docker] ${fn.functionName} \u2500\u2500`);
|
|
778
|
+
// Log if queuing during a rebuild
|
|
779
|
+
if (container.isRebuilding) {
|
|
780
|
+
yield* Effect.logDebug(`[Local] Queueing invocation ${invocation.requestId} for ${fn.functionName} (rebuild in progress, will be picked up by new container)`);
|
|
781
|
+
}
|
|
782
|
+
else {
|
|
783
|
+
yield* Effect.logDebug(`[Local] Queueing invocation ${invocation.requestId} for ${fn.functionName}`);
|
|
784
|
+
}
|
|
785
|
+
const lambdaInvocation = {
|
|
786
|
+
requestId: invocation.requestId,
|
|
787
|
+
event: invocation.event,
|
|
788
|
+
invokedFunctionArn: invocation.context.invokedFunctionArn,
|
|
789
|
+
deadlineMs: Date.now() + invocation.context.getRemainingTimeInMillis,
|
|
790
|
+
functionName: fn.functionName,
|
|
791
|
+
functionVersion: invocation.context.functionVersion,
|
|
792
|
+
memoryLimitMB: fn.memoryMB,
|
|
793
|
+
logGroupName: invocation.context.logGroupName,
|
|
794
|
+
logStreamName: invocation.context.logStreamName,
|
|
795
|
+
};
|
|
796
|
+
yield* Effect.logDebug(`[Local] Queueing to Runtime API on port ${container.port}`);
|
|
797
|
+
// Notify extensions about the invocation (for Lambda Web Adapter support)
|
|
798
|
+
yield* notifyExtensionsInvoke(container.runtimeState, lambdaInvocation);
|
|
799
|
+
yield* queueInvocation(container.runtimeState, lambdaInvocation);
|
|
800
|
+
yield* Effect.logDebug(`[Local] Invocation queued successfully`);
|
|
801
|
+
});
|
|
802
|
+
/**
|
|
803
|
+
* Handle incoming invocations for a Node.js function by queueing them.
|
|
804
|
+
* Starts the worker lazily if not already running.
|
|
805
|
+
*/
|
|
806
|
+
const handleNodejsInvocation = (fn, invocation, workersRef, serverScope, projectRoot, appSyncClient, invocationCounter, invocationContexts) => Effect.gen(function* () {
|
|
807
|
+
// Get env vars from invocation (forwarded from bridge Lambda)
|
|
808
|
+
const invocationEnv = sanitizeInvocationEnv(invocation.env ?? {});
|
|
809
|
+
// Ensure worker is started (lazy startup on first invocation)
|
|
810
|
+
const worker = yield* ensureWorkerStarted(fn, invocationEnv, workersRef, serverScope, projectRoot, appSyncClient, invocationContexts);
|
|
811
|
+
// Assign invocation number and track context for logging
|
|
812
|
+
const invocationNum = yield* Ref.updateAndGet(invocationCounter, (n) => n + 1);
|
|
813
|
+
const startTime = Date.now();
|
|
814
|
+
invocationContexts.set(invocation.requestId, {
|
|
815
|
+
num: invocationNum,
|
|
816
|
+
start: startTime,
|
|
817
|
+
fn: fn.functionName,
|
|
818
|
+
isDocker: false,
|
|
819
|
+
});
|
|
820
|
+
// Print start marker
|
|
821
|
+
console.log(`\n[${invocationNum}] \u250c\u2500\u2500 ${fn.functionName} \u2500\u2500`);
|
|
822
|
+
yield* Effect.logDebug(`[Local] Queueing invocation ${invocation.requestId} for ${fn.functionName}`);
|
|
823
|
+
const lambdaInvocation = {
|
|
824
|
+
requestId: invocation.requestId,
|
|
825
|
+
event: invocation.event,
|
|
826
|
+
invokedFunctionArn: invocation.context.invokedFunctionArn,
|
|
827
|
+
deadlineMs: Date.now() + invocation.context.getRemainingTimeInMillis,
|
|
828
|
+
functionName: fn.functionName,
|
|
829
|
+
functionVersion: invocation.context.functionVersion,
|
|
830
|
+
memoryLimitMB: fn.memoryMB,
|
|
831
|
+
logGroupName: invocation.context.logGroupName,
|
|
832
|
+
logStreamName: invocation.context.logStreamName,
|
|
833
|
+
};
|
|
834
|
+
yield* queueInvocation(worker.runtimeState, lambdaInvocation);
|
|
835
|
+
});
|
|
836
|
+
/**
|
|
837
|
+
* Start CDK watch process with CDK_LIVE=true using Effect's Command.
|
|
838
|
+
* Returns the process and a queue of events.
|
|
839
|
+
*/
|
|
840
|
+
const startCdkWatch = (options, scope) => Effect.gen(function* () {
|
|
841
|
+
const args = [
|
|
842
|
+
"cdk",
|
|
843
|
+
"watch",
|
|
844
|
+
"--hotswap-fallback",
|
|
845
|
+
"--no-logs",
|
|
846
|
+
"--method=direct",
|
|
847
|
+
];
|
|
848
|
+
if (options.stacks && options.stacks.length > 0) {
|
|
849
|
+
args.push(...options.stacks);
|
|
850
|
+
}
|
|
851
|
+
else if (options.all) {
|
|
852
|
+
args.push("--all");
|
|
853
|
+
}
|
|
854
|
+
if (options.profile) {
|
|
855
|
+
args.push("--profile", options.profile);
|
|
856
|
+
}
|
|
857
|
+
const env = {
|
|
858
|
+
...process.env,
|
|
859
|
+
CDK_LIVE: "true",
|
|
860
|
+
};
|
|
861
|
+
if (options.region) {
|
|
862
|
+
env.AWS_REGION = options.region;
|
|
863
|
+
env.CDK_DEFAULT_REGION = options.region;
|
|
864
|
+
}
|
|
865
|
+
yield* Effect.logDebug(`Starting: npx ${args.join(" ")}`);
|
|
866
|
+
const command = PlatformCommand.make("npx", ...args).pipe(PlatformCommand.env(env), PlatformCommand.runInShell(true), PlatformCommand.stdin("inherit"));
|
|
867
|
+
// Extend the process resource lifetime to the provided scope
|
|
868
|
+
const proc = yield* PlatformCommand.start(command).pipe(Scope.extend(scope));
|
|
869
|
+
// Event queue for callers to subscribe to
|
|
870
|
+
const events = yield* Queue.unbounded();
|
|
871
|
+
// State tracking for deploy status messages
|
|
872
|
+
let isDeploying = false;
|
|
873
|
+
let isFirstDeploy = true;
|
|
874
|
+
let outputBuffer = "";
|
|
875
|
+
const discoveredStacks = new Set();
|
|
876
|
+
// Patterns to detect CDK watch behavior
|
|
877
|
+
const deployCompletePattern = /✅\s+\S+|Deployment time:/;
|
|
878
|
+
const deployStartPattern = /Deploying|hotswap|Hotswapping|Bundling/i;
|
|
879
|
+
const noChangesPattern = /no changes|identical|up to date/i;
|
|
880
|
+
const errorPattern = /error|failed|Error|Failed|ERR!/i;
|
|
881
|
+
const stackNamePattern = /^(\S+):\s*deploying|✅\s+(\S+)/;
|
|
882
|
+
// Merge stdout and stderr, decode to text, split into lines
|
|
883
|
+
const outputStream = Stream.merge(proc.stdout, proc.stderr).pipe(Stream.decodeText(), Stream.splitLines);
|
|
884
|
+
// Fork stream processing in background
|
|
885
|
+
yield* outputStream.pipe(Stream.runForEach((line) => Effect.gen(function* () {
|
|
886
|
+
outputBuffer += line + "\n";
|
|
887
|
+
// Log all output at debug level
|
|
888
|
+
yield* Effect.logDebug(`[CDK] ${line}`);
|
|
889
|
+
// Try to extract stack name
|
|
890
|
+
const match = stackNamePattern.exec(line.trim());
|
|
891
|
+
if (match) {
|
|
892
|
+
const stackName = match[1] || match[2];
|
|
893
|
+
if (stackName && !discoveredStacks.has(stackName)) {
|
|
894
|
+
discoveredStacks.add(stackName);
|
|
895
|
+
yield* Queue.offer(events, {
|
|
896
|
+
_tag: "StackDiscovered",
|
|
897
|
+
stackName,
|
|
898
|
+
});
|
|
899
|
+
}
|
|
900
|
+
}
|
|
901
|
+
// Detect deploy start
|
|
902
|
+
if (!isDeploying && deployStartPattern.test(line)) {
|
|
903
|
+
isDeploying = true;
|
|
904
|
+
if (!isFirstDeploy) {
|
|
905
|
+
yield* Effect.logInfo("[CDK] Deploying...");
|
|
906
|
+
}
|
|
907
|
+
}
|
|
908
|
+
// Detect errors - output immediately
|
|
909
|
+
if (errorPattern.test(line)) {
|
|
910
|
+
yield* Effect.sync(() => process.stderr.write(line + "\n"));
|
|
911
|
+
}
|
|
912
|
+
// Check for deploy completion (only trigger once per deploy cycle)
|
|
913
|
+
if (isDeploying && deployCompletePattern.test(line)) {
|
|
914
|
+
isDeploying = false;
|
|
915
|
+
isFirstDeploy = false;
|
|
916
|
+
outputBuffer = "";
|
|
917
|
+
yield* Effect.logInfo("[CDK] Deploy complete");
|
|
918
|
+
// Small delay to ensure AWS has propagated the changes
|
|
919
|
+
yield* Effect.sleep("1 second");
|
|
920
|
+
yield* Queue.offer(events, { _tag: "DeployComplete" });
|
|
921
|
+
}
|
|
922
|
+
// Check for no changes
|
|
923
|
+
if (noChangesPattern.test(line)) {
|
|
924
|
+
isDeploying = false;
|
|
925
|
+
outputBuffer = "";
|
|
926
|
+
}
|
|
927
|
+
})),
|
|
928
|
+
// Log errors and exit code when stream ends
|
|
929
|
+
Effect.tapError((error) => Effect.logError(`[CDK] CDK watch error: ${error}`)), Effect.ensuring(proc.exitCode.pipe(Effect.flatMap((code) => code !== 0
|
|
930
|
+
? Effect.logError(`[CDK] CDK watch exited with code ${code}`)
|
|
931
|
+
: Effect.logDebug(`[CDK] CDK watch exited with code ${code}`)), Effect.catchAll(() => Effect.void))), Effect.fork);
|
|
932
|
+
return { process: proc, events };
|
|
933
|
+
});
|
|
934
|
+
/**
|
|
935
|
+
* Common CLI options.
|
|
936
|
+
*/
|
|
937
|
+
const profileOption = Options.text("profile").pipe(Options.optional, Options.withDescription("AWS profile to use"));
|
|
938
|
+
const regionOption = Options.text("region").pipe(Options.optional, Options.withDescription("AWS region"));
|
|
939
|
+
const qualifierOption = Options.text("qualifier").pipe(Options.withDefault("hnb659fds"), Options.withDescription("CDK bootstrap qualifier"));
|
|
940
|
+
const stacksOption = Options.text("stacks").pipe(Options.optional, Options.withDescription("Stack names to deploy (comma-separated, default: all)"));
|
|
941
|
+
const allStacksOption = Options.boolean("all").pipe(Options.withDefault(false), Options.withDescription("Deploy all stacks (like cdk deploy --all)"));
|
|
942
|
+
const debugOption = Options.boolean("debug").pipe(Options.withDefault(false), Options.withDescription("Enable debug logging"));
|
|
943
|
+
const idleTimeoutOption = Options.integer("idle-timeout").pipe(Options.withDefault(DEFAULT_IDLE_TIMEOUT_MS), Options.withDescription("Idle timeout in ms before stopping containers (default: 230000)"));
|
|
944
|
+
const pollTimeoutOption = Options.integer("poll-timeout").pipe(Options.optional, Options.withDescription("Poll timeout in ms for Runtime API (default: idle-timeout - 10s). Must be shorter than idle-timeout."));
|
|
945
|
+
/**
|
|
946
|
+
* Local command definition.
|
|
947
|
+
*/
|
|
948
|
+
export const localCommand = Command.make("local", {
|
|
949
|
+
profile: profileOption,
|
|
950
|
+
region: regionOption,
|
|
951
|
+
qualifier: qualifierOption,
|
|
952
|
+
stacks: stacksOption,
|
|
953
|
+
all: allStacksOption,
|
|
954
|
+
debug: debugOption,
|
|
955
|
+
idleTimeout: idleTimeoutOption,
|
|
956
|
+
pollTimeout: pollTimeoutOption,
|
|
957
|
+
}, ({ profile, region, qualifier, stacks, all, debug, idleTimeout, pollTimeout, }) => {
|
|
958
|
+
const logLevel = debug ? LogLevel.Debug : LogLevel.Info;
|
|
959
|
+
return Effect.gen(function* () {
|
|
960
|
+
yield* Effect.logInfo("[Local] Starting local Lambda development...");
|
|
961
|
+
const profileValue = profile._tag === "Some" ? profile.value : undefined;
|
|
962
|
+
const regionValue = region._tag === "Some" ? region.value : undefined;
|
|
963
|
+
const stacksFromOption = stacks._tag === "Some"
|
|
964
|
+
? stacks.value.split(",").map((s) => s.trim())
|
|
965
|
+
: undefined;
|
|
966
|
+
// Poll timeout should be shorter than idle timeout so containers exit naturally
|
|
967
|
+
// before we try to stop them. Default: idle timeout - 10 seconds.
|
|
968
|
+
const effectivePollTimeout = pollTimeout._tag === "Some"
|
|
969
|
+
? pollTimeout.value
|
|
970
|
+
: Math.max(idleTimeout - 10_000, 5_000); // At least 5 seconds
|
|
971
|
+
if (profileValue) {
|
|
972
|
+
process.env.AWS_PROFILE = profileValue;
|
|
973
|
+
}
|
|
974
|
+
if (regionValue) {
|
|
975
|
+
process.env.AWS_REGION = regionValue;
|
|
976
|
+
}
|
|
977
|
+
// Bootstrap check must complete first
|
|
978
|
+
yield* ensureBootstrap({
|
|
979
|
+
qualifier,
|
|
980
|
+
profile: profileValue,
|
|
981
|
+
region: regionValue,
|
|
982
|
+
});
|
|
983
|
+
// Stack filter - populated from CDK watch output, used to filter Lambda discovery
|
|
984
|
+
// Using object so closure in startOrUpdateDaemon sees updates
|
|
985
|
+
const filterState = { stacks: stacksFromOption ?? [] };
|
|
986
|
+
// Create a long-lived scope for all Runtime API servers and CDK watch process
|
|
987
|
+
// Resources will run until this scope is closed (when the program ends)
|
|
988
|
+
const serverScope = yield* Scope.make();
|
|
989
|
+
// Start CDK watch immediately after bootstrap
|
|
990
|
+
// Stack names are discovered from CDK watch output (no need for separate cdk ls)
|
|
991
|
+
yield* Effect.logInfo("[CDK] Deploying...");
|
|
992
|
+
const { process: cdkWatchProcess, events: cdkEvents } = yield* startCdkWatch({
|
|
993
|
+
profile: profileValue,
|
|
994
|
+
region: regionValue,
|
|
995
|
+
stacks: stacksFromOption,
|
|
996
|
+
all,
|
|
997
|
+
}, serverScope);
|
|
998
|
+
// Track running Docker containers by function name
|
|
999
|
+
const containers = yield* Ref.make(new Map());
|
|
1000
|
+
// Track running Node.js workers by function name
|
|
1001
|
+
const workers = yield* Ref.make(new Map());
|
|
1002
|
+
// Track registered functions (for lazy container/worker startup)
|
|
1003
|
+
const registeredFunctions = yield* Ref.make(new Map());
|
|
1004
|
+
// Track Docker functions with active file watchers
|
|
1005
|
+
const watchedDockerFunctions = new Set();
|
|
1006
|
+
// Invocation tracking for improved logging output
|
|
1007
|
+
const invocationCounter = yield* Ref.make(0);
|
|
1008
|
+
// Use a plain Map for invocation contexts so it can be accessed from stream callbacks
|
|
1009
|
+
const invocationContexts = new Map();
|
|
1010
|
+
// Project root is the current working directory (where CDK app lives)
|
|
1011
|
+
const projectRoot = process.cwd();
|
|
1012
|
+
let appSyncClient = null;
|
|
1013
|
+
// Track if we've logged the "Watching" message (only log once)
|
|
1014
|
+
const logState = { hasLoggedWatching: false, hasDiscoveredOnce: false };
|
|
1015
|
+
// Function to start/update the daemon with discovered functions
|
|
1016
|
+
const startOrUpdateDaemon = Effect.gen(function* () {
|
|
1017
|
+
// Only show "Discovering functions..." on first run
|
|
1018
|
+
if (!logState.hasDiscoveredOnce) {
|
|
1019
|
+
yield* Effect.logInfo("[Local] Discovering functions...");
|
|
1020
|
+
}
|
|
1021
|
+
// Get AppSync endpoints (may need to wait for first deploy)
|
|
1022
|
+
if (!appSyncClient) {
|
|
1023
|
+
yield* Effect.logDebug("[Local] Reading AppSync endpoints from SSM...");
|
|
1024
|
+
const endpoints = yield* getAppSyncEndpoints(qualifier).pipe(Effect.retry({ times: 10, schedule: Schedule.spaced("3 seconds") }));
|
|
1025
|
+
yield* Effect.logDebug(`[Local] HTTP endpoint: ${endpoints.httpEndpoint}`);
|
|
1026
|
+
appSyncClient = makeAppSyncClient(endpoints);
|
|
1027
|
+
}
|
|
1028
|
+
// Discover functions (filtered to stacks in this CDK project)
|
|
1029
|
+
yield* Effect.logDebug("[Local] Discovering Lambda functions...");
|
|
1030
|
+
const functions = yield* discoverFunctions(filterState.stacks);
|
|
1031
|
+
if (functions.length === 0) {
|
|
1032
|
+
yield* Effect.logInfo("[Local] No functions found with live-lambda tags yet. Have you patched your CDK project (bootstrap) and added the LiveLambdaAspect?");
|
|
1033
|
+
return;
|
|
1034
|
+
}
|
|
1035
|
+
const currentRegistered = yield* Ref.get(registeredFunctions);
|
|
1036
|
+
// Collect new functions to register
|
|
1037
|
+
const newFunctions = [];
|
|
1038
|
+
for (const fn of functions) {
|
|
1039
|
+
const isDocker = Boolean(fn.dockerContextPath);
|
|
1040
|
+
const isNodejs = !isDocker && Boolean(fn.localHandler);
|
|
1041
|
+
if (!isDocker && !isNodejs) {
|
|
1042
|
+
yield* Effect.logDebug(`[Local] Skipping ${fn.functionName} - no Docker context or local handler`);
|
|
1043
|
+
continue;
|
|
1044
|
+
}
|
|
1045
|
+
if (currentRegistered.has(fn.functionName)) {
|
|
1046
|
+
yield* Effect.logDebug(`[Local] Already watching ${fn.functionName}`);
|
|
1047
|
+
continue;
|
|
1048
|
+
}
|
|
1049
|
+
newFunctions.push({ fn, isDocker });
|
|
1050
|
+
}
|
|
1051
|
+
// Print summary of discovered functions
|
|
1052
|
+
if (newFunctions.length > 0) {
|
|
1053
|
+
const summary = newFunctions
|
|
1054
|
+
.map(({ fn, isDocker }) => {
|
|
1055
|
+
const mode = isDocker ? "docker" : "node";
|
|
1056
|
+
const shortName = fn.functionName.includes("-")
|
|
1057
|
+
? fn.functionName.split("-").slice(-2, -1)[0] || fn.functionName
|
|
1058
|
+
: fn.functionName;
|
|
1059
|
+
return `${shortName} (${mode})`;
|
|
1060
|
+
})
|
|
1061
|
+
.join(", ");
|
|
1062
|
+
// Use different message for first discovery vs subsequent
|
|
1063
|
+
const prefix = logState.hasDiscoveredOnce
|
|
1064
|
+
? "[Local] Functions added:"
|
|
1065
|
+
: "[Local] Functions:";
|
|
1066
|
+
yield* Effect.logInfo(`${prefix} ${summary}`);
|
|
1067
|
+
}
|
|
1068
|
+
// Register functions and set up subscriptions
|
|
1069
|
+
for (const { fn, isDocker } of newFunctions) {
|
|
1070
|
+
yield* Effect.logDebug(`[Local] Registering ${fn.functionName} (${isDocker ? "Docker" : "Node.js"})`);
|
|
1071
|
+
// Register the function
|
|
1072
|
+
currentRegistered.set(fn.functionName, fn);
|
|
1073
|
+
// Subscribe to invocations for this function
|
|
1074
|
+
const invocationChannel = buildChannelName.invocation(fn.functionName);
|
|
1075
|
+
yield* Effect.logDebug(`[Local] Subscribing to invocations for ${fn.functionName}`);
|
|
1076
|
+
// Subscribe using forkDaemon to run independently with context preserved
|
|
1077
|
+
// Route to Docker or Node.js handler based on function type
|
|
1078
|
+
if (isDocker) {
|
|
1079
|
+
yield* appSyncClient
|
|
1080
|
+
.subscribeToInvocations(invocationChannel)
|
|
1081
|
+
.pipe(Stream.runForEach((invocation) => handleDockerInvocation(fn, invocation, containers, serverScope, projectRoot, idleTimeout, effectivePollTimeout, appSyncClient, invocationCounter, invocationContexts).pipe(Effect.catchAll((error) => Effect.logError(`[Local] Docker invocation error: ${error}`)))), Effect.forkDaemon);
|
|
1082
|
+
}
|
|
1083
|
+
else {
|
|
1084
|
+
yield* appSyncClient
|
|
1085
|
+
.subscribeToInvocations(invocationChannel)
|
|
1086
|
+
.pipe(Stream.runForEach((invocation) => handleNodejsInvocation(fn, invocation, workers, serverScope, projectRoot, appSyncClient, invocationCounter, invocationContexts).pipe(Effect.catchAll((error) => Effect.logError(`[Local] Node.js invocation error: ${error}`)))), Effect.forkDaemon);
|
|
1087
|
+
}
|
|
1088
|
+
}
|
|
1089
|
+
yield* Ref.set(registeredFunctions, currentRegistered);
|
|
1090
|
+
// Start file watchers for new Docker functions
|
|
1091
|
+
const newDockerFunctions = [];
|
|
1092
|
+
for (const fn of functions) {
|
|
1093
|
+
if (fn.dockerContextPath &&
|
|
1094
|
+
!watchedDockerFunctions.has(fn.functionName)) {
|
|
1095
|
+
// Resolve the context path
|
|
1096
|
+
const contextPath = fn.dockerContextPath.startsWith("/")
|
|
1097
|
+
? fn.dockerContextPath
|
|
1098
|
+
: `${projectRoot}/${fn.dockerContextPath}`;
|
|
1099
|
+
newDockerFunctions.push({
|
|
1100
|
+
functionId: fn.functionName,
|
|
1101
|
+
dockerContextPath: contextPath,
|
|
1102
|
+
});
|
|
1103
|
+
watchedDockerFunctions.add(fn.functionName);
|
|
1104
|
+
}
|
|
1105
|
+
}
|
|
1106
|
+
// Start watching new Docker contexts
|
|
1107
|
+
if (newDockerFunctions.length > 0) {
|
|
1108
|
+
yield* Effect.logDebug(`[Local] Starting file watchers for ${newDockerFunctions.length} Docker function(s)...`);
|
|
1109
|
+
// Fork a daemon fiber to handle file change events with context preserved
|
|
1110
|
+
yield* watchDockerContexts(newDockerFunctions, 500).pipe(Stream.runForEach((event) => Effect.gen(function* () {
|
|
1111
|
+
yield* Effect.logDebug(`[Local] File changed in ${event.functionId}: ${event.filePath}`);
|
|
1112
|
+
yield* rebuildDockerContainer(event.functionId, containers, projectRoot).pipe(Effect.catchAll((error) => Effect.logError(`[Local] Rebuild failed for ${event.functionId}: ${error}`)));
|
|
1113
|
+
})), Effect.forkDaemon);
|
|
1114
|
+
}
|
|
1115
|
+
if (!logState.hasLoggedWatching) {
|
|
1116
|
+
logState.hasLoggedWatching = true;
|
|
1117
|
+
yield* Effect.logInfo("[Local] Ready for invocations");
|
|
1118
|
+
}
|
|
1119
|
+
// Mark that we've completed first discovery
|
|
1120
|
+
logState.hasDiscoveredOnce = true;
|
|
1121
|
+
});
|
|
1122
|
+
// Handle CDK watch events (stack discovery and deploy completion)
|
|
1123
|
+
yield* Queue.take(cdkEvents).pipe(Effect.flatMap((event) => Effect.gen(function* () {
|
|
1124
|
+
switch (event._tag) {
|
|
1125
|
+
case "StackDiscovered":
|
|
1126
|
+
// Add discovered stack to filter (if not using --stacks)
|
|
1127
|
+
if (!stacksFromOption &&
|
|
1128
|
+
!filterState.stacks.includes(event.stackName)) {
|
|
1129
|
+
filterState.stacks.push(event.stackName);
|
|
1130
|
+
yield* Effect.logDebug(`[Local] Discovered stack: ${event.stackName}`);
|
|
1131
|
+
}
|
|
1132
|
+
break;
|
|
1133
|
+
case "DeployComplete":
|
|
1134
|
+
// Run daemon update on deploy completion
|
|
1135
|
+
yield* startOrUpdateDaemon.pipe(Effect.catchAll((error) => Effect.logError(`Failed to update daemon: ${error}`)));
|
|
1136
|
+
break;
|
|
1137
|
+
}
|
|
1138
|
+
})), Effect.forever, Effect.fork);
|
|
1139
|
+
// Handle cleanup on exit
|
|
1140
|
+
const cleanup = async () => {
|
|
1141
|
+
await Effect.runPromise(Effect.gen(function* () {
|
|
1142
|
+
yield* Effect.logInfo("\nShutting down...");
|
|
1143
|
+
// Stop CDK watch
|
|
1144
|
+
yield* cdkWatchProcess
|
|
1145
|
+
.kill("SIGTERM")
|
|
1146
|
+
.pipe(Effect.catchAll(() => Effect.void));
|
|
1147
|
+
// Stop all Docker containers using the Docker service
|
|
1148
|
+
const docker = yield* Docker;
|
|
1149
|
+
const currentContainers = yield* Ref.get(containers);
|
|
1150
|
+
for (const [name, container] of currentContainers) {
|
|
1151
|
+
yield* Effect.logInfo(`Stopping container: ${name}`);
|
|
1152
|
+
yield* docker.stop(container.containerName).pipe(Effect.scoped, Effect.catchAll(() => Effect.void));
|
|
1153
|
+
}
|
|
1154
|
+
// Stop all Node.js workers
|
|
1155
|
+
const currentWorkers = yield* Ref.get(workers);
|
|
1156
|
+
for (const [name, worker] of currentWorkers) {
|
|
1157
|
+
yield* Effect.logInfo(`Stopping worker: ${name}`);
|
|
1158
|
+
yield* Effect.try(() => worker.workerProcess.kill("SIGTERM")).pipe(Effect.catchAll(() => Effect.void));
|
|
1159
|
+
}
|
|
1160
|
+
// Close the server scope to clean up Runtime API servers
|
|
1161
|
+
yield* Scope.close(serverScope, Exit.void);
|
|
1162
|
+
}).pipe(Effect.provide(DockerLive), Effect.provide(BunContext.layer), Effect.provide(Logger.pretty), Effect.provide(Logger.minimumLogLevel(logLevel))));
|
|
1163
|
+
process.exit(0);
|
|
1164
|
+
};
|
|
1165
|
+
process.on("SIGINT", cleanup);
|
|
1166
|
+
process.on("SIGTERM", cleanup);
|
|
1167
|
+
yield* Effect.logInfo("[Local] Press Ctrl+C to stop");
|
|
1168
|
+
// Keep the process running
|
|
1169
|
+
yield* Effect.never;
|
|
1170
|
+
}).pipe(Effect.provide(Logger.pretty), Effect.provide(Logger.minimumLogLevel(logLevel)));
|
|
1171
|
+
}).pipe(Command.withDescription("Start local Lambda development with CDK watch and Docker containers"));
|
|
1172
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"local.js","sourceRoot":"","sources":["../../../src/cli/commands/local.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Local command for running Lambda functions locally using Docker.\n *\n * This command:\n * 1. Starts CDK watch with CDK_LIVE=true and hotswap\n * 2. Discovers Lambda functions with live-lambda:handler tag\n * 3. Connects to AppSync Events\n * 4. Subscribes to invocation channels\n * 5. For each function, maintains ONE Docker container that handles all invocations\n * 6. Sends responses back via AppSync\n * 7. Re-discovers functions after each CDK deploy\n */\n\nimport { type ChildProcess, spawn } from \"node:child_process\"\nimport * as path from \"node:path\"\nimport { fileURLToPath } from \"node:url\"\nimport {\n  LambdaClient,\n  ListFunctionsCommand,\n  ListTagsCommand,\n} from \"@aws-sdk/client-lambda\"\nimport { GetParameterCommand, SSMClient } from \"@aws-sdk/client-ssm\"\nimport { Command, Options } from \"@effect/cli\"\nimport {\n  type CommandExecutor,\n  Command as PlatformCommand,\n} from \"@effect/platform\"\nimport type { Process as EffectProcess } from \"@effect/platform/CommandExecutor\"\nimport { BunContext } from \"@effect/platform-bun\"\nimport {\n  Duration,\n  Effect,\n  Exit,\n  Fiber,\n  Logger,\n  LogLevel,\n  Queue,\n  Ref,\n  Schedule,\n  Scope,\n  Stream,\n} from \"effect\"\nimport {\n  BOOTSTRAP_STACK_NAME,\n  BOOTSTRAP_VERSION,\n  buildChannelName,\n  type InvocationMessage,\n  LIVE_LAMBDA_DOCKER_TAG,\n  LIVE_LAMBDA_TAG,\n  type ResponseMessage,\n  SSM_BASE_PATH,\n} from \"../../shared/types.js\"\nimport { makeAppSyncClient } from \"../appsync/client.js\"\nimport {\n  buildExtensionWrapperCommand,\n  Docker,\n  DockerLive,\n  makeLambdaContainerConfig,\n} from \"../docker/container.js\"\nimport {\n  type WatchedDockerFunction,\n  watchDockerContexts,\n} from \"../docker/watcher.js\"\nimport {\n  notifyExtensionsInvoke,\n  queueInvocation,\n  type RuntimeApiState,\n  startRuntimeApiServer,\n  waitForResponse,\n} from \"../runtime-api/server.js\"\nimport type {\n  LambdaError,\n  LambdaInitError,\n  LambdaInvocation,\n  LambdaResponse,\n} from \"../runtime-api/types.js\"\n\n/**\n * Discovered Lambda function info.\n */\ninterface DiscoveredFunction {\n  functionName: string\n  functionArn: string\n  localHandler: string\n  /** Local Docker context path for DockerImageFunction */\n  dockerContextPath?: string\n  memoryMB: number\n  architecture?: \"arm64\" | \"x86_64\"\n}\n\n/**\n * Default idle timeout before proactively stopping containers.\n * Set slightly below the poll timeout (240s) to stop container before\n * the Lambda RIC gets a 503 error.\n */\nconst DEFAULT_IDLE_TIMEOUT_MS = 230_000 // 230 seconds (~4 minutes)\n\n/**\n * State for a running function container.\n */\ninterface FunctionContainer {\n  fn: DiscoveredFunction\n  runtimeState: RuntimeApiState\n  /** The port the Runtime API server is listening on */\n  port: number\n  containerFiber: Fiber.RuntimeFiber<void, Error>\n  containerName: string\n  /** Docker image name for rebuilds */\n  imageName: string\n  /** Whether the container is currently being rebuilt (invocations will queue) */\n  isRebuilding: boolean\n  /** Map of requestId -> response resolver */\n  pendingResponses: Map<\n    string,\n    {\n      resolve: (response: ResponseMessage) => void\n    }\n  >\n  /** Fiber for the idle shutdown timer, cancelled when new responses arrive */\n  idleTimerFiber: Fiber.RuntimeFiber<void, never> | null\n  /** Environment variables captured from first invocation */\n  env: Record<string, string>\n}\n\n/**\n * State for a running Node.js worker process.\n */\ninterface NodejsWorker {\n  fn: DiscoveredFunction\n  runtimeState: RuntimeApiState\n  /** The port the Runtime API server is listening on */\n  port: number\n  /** The spawned Bun process */\n  workerProcess: ChildProcess\n  /** Environment variables captured from first invocation */\n  env: Record<string, string>\n}\n\n/**\n * Invocation context for tracking and logging.\n */\nexport interface InvocationContext {\n  /** Sequential invocation number */\n  num: number\n  /** Start timestamp in milliseconds */\n  start: number\n  /** Function name */\n  fn: string\n  /** Whether this is a Docker container invocation */\n  isDocker: boolean\n}\n\n/**\n * Check if bootstrap stack version matches the expected version.\n * Returns true if version matches, false if missing or mismatched.\n */\nconst checkBootstrapVersion = (qualifier: string) =>\n  Effect.gen(function* () {\n    const ssmClient = new SSMClient({})\n    const basePath = `${SSM_BASE_PATH}/${qualifier}`\n\n    const result = yield* Effect.tryPromise({\n      try: async () => {\n        const response = await ssmClient.send(\n          new GetParameterCommand({ Name: `${basePath}/version` }),\n        )\n        return response.Parameter?.Value\n      },\n      catch: () => null,\n    }).pipe(Effect.catchAll(() => Effect.succeed(null)))\n\n    if (result === null) {\n      yield* Effect.logInfo(\"Bootstrap stack version parameter not found\")\n      return false\n    }\n\n    if (result !== BOOTSTRAP_VERSION) {\n      yield* Effect.logInfo(\n        `Bootstrap stack version mismatch: found ${result}, expected ${BOOTSTRAP_VERSION}`,\n      )\n      return false\n    }\n\n    return true\n  })\n\n/**\n * Run the bootstrap stack deployment.\n */\nconst runBootstrap = (options: { profile?: string; region?: string }) =>\n  Effect.gen(function* () {\n    yield* Effect.logInfo(\"Running bootstrap stack deployment...\")\n\n    // Resolve the CDK app path relative to this module\n    const __filename = fileURLToPath(import.meta.url)\n    const __dirname = path.dirname(__filename)\n    const cdkAppPath = path.resolve(__dirname, \"..\", \"cdk-app.js\")\n\n    const args = [\n      \"cdk\",\n      \"deploy\",\n      BOOTSTRAP_STACK_NAME,\n      \"--require-approval\",\n      \"never\",\n      \"--app\",\n      `bun ${cdkAppPath}`,\n    ]\n\n    if (options.profile) {\n      args.push(\"--profile\", options.profile)\n    }\n\n    const env: Record<string, string> = {\n      ...process.env,\n    } as Record<string, string>\n    if (options.region) {\n      env.AWS_REGION = options.region\n      env.CDK_DEFAULT_REGION = options.region\n    }\n\n    yield* Effect.logInfo(`Running: npx ${args.join(\" \")}`)\n\n    const command = PlatformCommand.make(\"npx\", ...args).pipe(\n      PlatformCommand.env(env),\n      PlatformCommand.stdin(\"inherit\"),\n    )\n\n    const proc = yield* PlatformCommand.start(command)\n\n    // Stream output to console\n    yield* Stream.merge(proc.stdout, proc.stderr).pipe(\n      Stream.decodeText(),\n      Stream.splitLines,\n      Stream.runForEach((line) =>\n        Effect.sync(() => {\n          process.stdout.write(`${line}\\n`)\n        }),\n      ),\n    )\n\n    const exitCode = yield* proc.exitCode\n    if (exitCode !== 0) {\n      return yield* Effect.fail(\n        new Error(`Bootstrap deployment failed with exit code ${exitCode}`),\n      )\n    }\n\n    yield* Effect.logInfo(\"Bootstrap stack deployed successfully!\")\n  }).pipe(Effect.scoped)\n\n/**\n * Ensure bootstrap stack is deployed with correct version.\n * Automatically deploys if missing or outdated.\n */\nconst ensureBootstrap = (options: {\n  qualifier: string\n  profile?: string\n  region?: string\n}) =>\n  Effect.gen(function* () {\n    yield* Effect.logDebug(\"[Local] Checking bootstrap stack version...\")\n\n    const versionOk = yield* checkBootstrapVersion(options.qualifier)\n\n    if (!versionOk) {\n      yield* Effect.logInfo(\n        \"[Local] Bootstrap stack needs to be deployed or updated.\",\n      )\n      yield* runBootstrap({ profile: options.profile, region: options.region })\n    } else {\n      yield* Effect.logDebug(\"[Local] Bootstrap stack version OK.\")\n    }\n  })\n\n/**\n * Read AppSync endpoints from SSM.\n */\nconst getAppSyncEndpoints = (qualifier: string) =>\n  Effect.gen(function* () {\n    const ssmClient = new SSMClient({})\n    const basePath = `${SSM_BASE_PATH}/${qualifier}`\n\n    const getParam = (name: string) =>\n      Effect.tryPromise({\n        try: async () => {\n          const result = await ssmClient.send(\n            new GetParameterCommand({ Name: `${basePath}/${name}` }),\n          )\n          return result.Parameter?.Value ?? \"\"\n        },\n        catch: (error) =>\n          new Error(`Failed to get SSM parameter ${name}: ${String(error)}`),\n      })\n\n    const httpEndpoint = yield* getParam(\"http-endpoint\")\n    const realtimeEndpoint = yield* getParam(\"realtime-endpoint\")\n\n    return { httpEndpoint, realtimeEndpoint }\n  })\n\n/**\n * CloudFormation stack name tag (set automatically by CDK)\n */\nconst CFN_STACK_NAME_TAG = \"aws:cloudformation:stack-name\"\n\n/**\n * Discover Lambda functions with live-lambda tags.\n * @param stackFilter - Optional list of stack names to filter by\n */\nconst discoverFunctions = (stackFilter?: string[]) =>\n  Effect.gen(function* () {\n    const lambdaClient = new LambdaClient({})\n    const functions: DiscoveredFunction[] = []\n\n    // List all functions\n    let nextMarker: string | undefined\n    do {\n      const listResponse = yield* Effect.tryPromise({\n        try: () =>\n          lambdaClient.send(new ListFunctionsCommand({ Marker: nextMarker })),\n        catch: (error) =>\n          new Error(`Failed to list functions: ${String(error)}`),\n      })\n\n      for (const fn of listResponse.Functions ?? []) {\n        // Get tags for each function\n        const tagsResult = yield* Effect.tryPromise({\n          try: () =>\n            lambdaClient.send(\n              new ListTagsCommand({ Resource: fn.FunctionArn }),\n            ),\n          catch: () => new Error(`Failed to get tags for ${fn.FunctionName}`),\n        }).pipe(\n          Effect.catchAll(() =>\n            Effect.succeed({ Tags: {} as Record<string, string> }),\n          ),\n        )\n\n        const tags = tagsResult.Tags ?? {}\n\n        // Check for live-lambda tag (handler) or docker-context tag\n        if (tags[LIVE_LAMBDA_TAG] || tags[LIVE_LAMBDA_DOCKER_TAG]) {\n          // Filter by stack name if specified\n          if (stackFilter && stackFilter.length > 0) {\n            const stackName = tags[CFN_STACK_NAME_TAG]\n            if (!stackName || !stackFilter.includes(stackName)) {\n              continue\n            }\n          }\n\n          // Determine architecture from Lambda config\n          const architectures = fn.Architectures ?? [\"x86_64\"]\n          const architecture = architectures.includes(\"arm64\")\n            ? (\"arm64\" as const)\n            : (\"x86_64\" as const)\n\n          const discovered: DiscoveredFunction = {\n            functionName: fn.FunctionName!,\n            functionArn: fn.FunctionArn!,\n            localHandler: tags[LIVE_LAMBDA_TAG] ?? \"\",\n            dockerContextPath: tags[LIVE_LAMBDA_DOCKER_TAG],\n            memoryMB: fn.MemorySize ?? 128,\n            architecture,\n          }\n          functions.push(discovered)\n        }\n      }\n\n      nextMarker = listResponse.NextMarker\n    } while (nextMarker)\n\n    return functions\n  })\n\n/**\n * Start a long-running Docker container for a function.\n * Builds the image from the local Docker context path, then runs it.\n * The container continuously polls our Runtime API for invocations.\n * Uses Effect.forkDaemon to ensure the container runs independently with context.\n */\nconst startFunctionContainer = (\n  fn: DiscoveredFunction,\n  port: number,\n  projectRoot: string,\n  additionalEnv: Record<string, string>,\n  invocationContexts?: Map<string, { num: number }>,\n): Effect.Effect<\n  Fiber.RuntimeFiber<void, Error>,\n  Error,\n  CommandExecutor.CommandExecutor\n> =>\n  Effect.gen(function* () {\n    if (!fn.dockerContextPath) {\n      return yield* Effect.fail(\n        new Error(`Function ${fn.functionName} has no Docker context path`),\n      )\n    }\n\n    const docker = yield* Docker\n\n    const dockerRuntime = yield* docker.getRuntimeInfo()\n    const runtimeApiHost = dockerRuntime.isDockerDesktop\n      ? \"host.docker.internal\"\n      : \"runtime.api\"\n\n    // Generate a local image name from function name\n    const imageName = `live-lambda-${fn.functionName.toLowerCase().replace(/[^a-z0-9-]/g, \"-\")}`\n\n    // Resolve the context path relative to project root\n    const contextPath = fn.dockerContextPath.startsWith(\"/\")\n      ? fn.dockerContextPath\n      : `${projectRoot}/${fn.dockerContextPath}`\n\n    // Determine platform from architecture\n    const platform = fn.architecture === \"arm64\" ? \"linux/arm64\" : \"linux/amd64\"\n\n    // Build the Docker image from local context\n    yield* docker\n      .build({\n        contextPath,\n        imageName,\n        platform,\n      })\n      .pipe(Effect.scoped)\n\n    // Inspect the image to get original entrypoint/cmd for extension wrapper\n    const imageConfig = yield* docker.inspect(imageName).pipe(Effect.scoped)\n    const extensionWrapper = buildExtensionWrapperCommand(\n      imageConfig.entrypoint,\n      imageConfig.cmd,\n    )\n\n    const containerConfig = makeLambdaContainerConfig({\n      imageUri: imageName,\n      runtimeApiHost,\n      runtimeApiPort: port,\n      functionName: fn.functionName,\n      functionVersion: \"$LATEST\",\n      memoryMB: fn.memoryMB,\n      timeoutSeconds: 3600, // Long timeout - container stays running\n      platform,\n      additionalEnv,\n      invocationContexts,\n    })\n\n    // Apply extension wrapper to start extensions before the app\n    containerConfig.entrypoint = extensionWrapper.entrypoint\n    containerConfig.command = extensionWrapper.command\n\n    yield* Effect.logInfo(\n      `Starting container for ${fn.functionName} on port ${port}`,\n    )\n\n    // Use Effect.forkDaemon to run the container independently with context preserved\n    // Effect.scoped provides the scope needed by docker.run\n    const fiber = yield* docker.run(containerConfig).pipe(\n      Effect.scoped,\n      Effect.map(() => undefined as void),\n      Effect.forkDaemon,\n    )\n\n    return fiber\n  }).pipe(Effect.provide(DockerLive))\n\n// Type guards for response types\nconst isLambdaResponse = (\n  r: LambdaResponse | LambdaError | LambdaInitError,\n): r is LambdaResponse => \"body\" in r && !(\"errorType\" in r)\n\nconst isLambdaError = (\n  r: LambdaResponse | LambdaError | LambdaInitError,\n): r is LambdaError => \"requestId\" in r && \"errorType\" in r\n\nconst isLambdaInitError = (\n  r: LambdaResponse | LambdaError | LambdaInitError,\n): r is LambdaInitError => !(\"requestId\" in r) && \"errorType\" in r\n\n/**\n * Reset the idle timer for a container.\n * After the timeout expires with no new responses, the container is stopped.\n * This prevents the confusing 503 error from the Lambda RIC when the poll times out.\n */\nconst resetIdleTimer = (\n  container: FunctionContainer,\n  containersRef: Ref.Ref<Map<string, FunctionContainer>>,\n  idleTimeoutMs: number,\n): Effect.Effect<void, never, CommandExecutor.CommandExecutor> =>\n  Effect.gen(function* () {\n    // Cancel existing timer if any\n    if (container.idleTimerFiber) {\n      yield* Fiber.interrupt(container.idleTimerFiber).pipe(\n        Effect.catchAll(() => Effect.void),\n      )\n      container.idleTimerFiber = null\n    }\n\n    // Start new timer - use never for error type since we catch all errors\n    const timerFiber: Fiber.RuntimeFiber<void, never> = yield* Effect.gen(\n      function* () {\n        yield* Effect.sleep(Duration.millis(idleTimeoutMs))\n\n        // Timer expired - stop container proactively\n        yield* Effect.logInfo(\n          `[Local] Stopping idle container ${container.fn.functionName} (will restart on next invocation)`,\n        )\n\n        // Stop the container using Docker with fast timeout (1s)\n        // The container may be blocked on long-polling, so we need to force kill\n        const docker = yield* Docker\n        yield* docker.stop(container.containerName, 1).pipe(\n          Effect.scoped,\n          Effect.catchAll(() => Effect.void),\n        )\n\n        // Remove from map so next invocation creates fresh container\n        const currentContainers = yield* Ref.get(containersRef)\n        currentContainers.delete(container.fn.functionName)\n        yield* Ref.set(containersRef, currentContainers)\n      },\n    ).pipe(\n      Effect.provide(DockerLive),\n      Effect.catchAll(() => Effect.void),\n      Effect.forkDaemon,\n    )\n\n    container.idleTimerFiber = timerFiber\n  })\n\n/**\n * Process responses from a container and dispatch to waiting callers.\n * After each response, resets the idle timer to proactively stop the container\n * before the poll timeout causes confusing Lambda RIC errors.\n */\nconst processContainerResponses = (\n  container: FunctionContainer,\n  containersRef: Ref.Ref<Map<string, FunctionContainer>>,\n  idleTimeoutMs: number,\n  client: ReturnType<typeof makeAppSyncClient>,\n  invocationContexts: Map<string, InvocationContext>,\n): Effect.Effect<void, Error, CommandExecutor.CommandExecutor> =>\n  Effect.gen(function* () {\n    const responseChannel = buildChannelName.response(container.fn.functionName)\n\n    // Continuously process responses from the container\n    while (true) {\n      const result = yield* waitForResponse(container.runtimeState)\n\n      let response: ResponseMessage\n      if (isLambdaResponse(result)) {\n        response = {\n          type: \"response\",\n          requestId: result.requestId,\n          result: result.body,\n        }\n      } else if (isLambdaError(result)) {\n        response = {\n          type: \"response\",\n          requestId: result.requestId,\n          error: {\n            errorType: result.errorType,\n            errorMessage: result.errorMessage,\n            stackTrace: result.stackTrace,\n          },\n        }\n      } else if (isLambdaInitError(result)) {\n        yield* Effect.logError(\n          `[Local] Init error: ${result.errorType}: ${result.errorMessage}`,\n        )\n        continue\n      } else {\n        continue\n      }\n\n      // Send response back via AppSync\n      yield* client.publishResponse(responseChannel, response)\n      yield* Effect.logDebug(`[Local] Sent response for ${response.requestId}`)\n\n      // Print end marker with timing\n      const ctx = invocationContexts.get(response.requestId)\n      if (ctx) {\n        const durationMs = Date.now() - ctx.start\n        if (response.error) {\n          console.log(\n            `[${ctx.num}] \\u2514\\u2500\\u2500 \\u2717 ${response.error.errorType}: ${response.error.errorMessage} (${durationMs}ms) \\u2500\\u2500`,\n          )\n        } else {\n          console.log(\n            `[${ctx.num}] \\u2514\\u2500\\u2500 \\u2713 Done (${durationMs}ms) \\u2500\\u2500`,\n          )\n        }\n        invocationContexts.delete(response.requestId)\n      }\n\n      // Reset idle timer - container will be stopped if no new invocations arrive\n      yield* resetIdleTimer(container, containersRef, idleTimeoutMs)\n    }\n  })\n\n/**\n * Start a Node.js worker process for a function.\n * Spawns Bun to run the runtime wrapper with the handler path.\n * The worker continuously polls our Runtime API for invocations.\n */\nconst startNodejsWorker = (\n  fn: DiscoveredFunction,\n  port: number,\n  projectRoot: string,\n  env: Record<string, string>,\n  invocationContexts: Map<string, InvocationContext>,\n): Effect.Effect<ChildProcess, Error> =>\n  Effect.gen(function* () {\n    if (!fn.localHandler) {\n      return yield* Effect.fail(\n        new Error(`Function ${fn.functionName} has no local handler path`),\n      )\n    }\n\n    // Resolve the runtime wrapper path relative to this module\n    const __filename = fileURLToPath(import.meta.url)\n    const __dirname = path.dirname(__filename)\n    const runtimeWrapperPath = path.resolve(\n      __dirname,\n      \"..\",\n      \"runtime-wrapper\",\n      \"nodejs-runtime.js\",\n    )\n\n    // Get absolute path to bun for pure env isolation (no PATH dependency)\n    const bunPath = Bun.which(\"bun\")\n    if (!bunPath) {\n      return yield* Effect.fail(\n        new Error(\"Could not find 'bun' executable in PATH\"),\n      )\n    }\n\n    // Build environment for the worker process\n    // Pure isolation: only env from bridge + local overrides, no local PATH/HOME\n    const workerEnv: NodeJS.ProcessEnv = {\n      // Start with env vars from the bridge Lambda (AWS credentials, user-defined vars)\n      ...env,\n      // Local overrides (these are set by the daemon, not from bridge)\n      AWS_LAMBDA_RUNTIME_API: `localhost:${port}`,\n      _HANDLER: fn.localHandler,\n      LAMBDA_TASK_ROOT: projectRoot,\n      // Memory limit for context object\n      AWS_LAMBDA_FUNCTION_MEMORY_SIZE: String(fn.memoryMB),\n    }\n\n    yield* Effect.logDebug(\n      `[Local] Starting Node.js worker for ${fn.functionName} on port ${port}`,\n    )\n    yield* Effect.logDebug(`[Local] Handler: ${fn.localHandler}`)\n\n    // Spawn Bun with --watch to automatically restart when handler files change\n    // This enables hot-reload without needing to restart the daemon\n    // Use absolute bun path for pure env isolation (no PATH dependency)\n    const workerProcess = spawn(bunPath, [\"--watch\", runtimeWrapperPath], {\n      cwd: projectRoot,\n      env: workerEnv,\n      stdio: [\"ignore\", \"pipe\", \"pipe\"],\n    })\n\n    // Pattern to parse Lambda log format: TIMESTAMP\\tREQUEST_ID\\tLEVEL\\tMESSAGE\n    // Lambda uses tabs between fields. Captures: [1] = request ID, [2] = level + message\n    const lambdaLogPattern =\n      /^(\\d{4}-\\d{2}-\\d{2}T[\\d:.]+Z)[\\t\\s]+([0-9a-f-]{36})[\\t\\s]+(.*)$/i\n\n    // Helper to format log line with invocation prefix\n    const formatLine = (\n      rawLine: string,\n    ): { prefix: string; content: string } => {\n      // Strip carriage returns that can cause terminal corruption\n      const line = rawLine.replace(/\\r/g, \"\")\n      const match = lambdaLogPattern.exec(line)\n      if (match) {\n        const requestId = match[2]\n        const ctx = invocationContexts.get(requestId)\n        if (ctx) {\n          // Strip timestamp and request ID, keep just LEVEL MESSAGE\n          return { prefix: `[${ctx.num}]`, content: match[3] }\n        }\n      }\n      return { prefix: \"[Worker]\", content: line }\n    }\n\n    // Forward stdout/stderr with invocation number prefix\n    workerProcess.stdout?.on(\"data\", (data: Buffer) => {\n      const lines = data.toString().trim().split(\"\\n\")\n      for (const rawLine of lines) {\n        const { prefix, content } = formatLine(rawLine)\n        console.log(`${prefix} ${content}`)\n      }\n    })\n\n    workerProcess.stderr?.on(\"data\", (data: Buffer) => {\n      const lines = data.toString().trim().split(\"\\n\")\n      for (const rawLine of lines) {\n        const { prefix, content } = formatLine(rawLine)\n        console.error(`${prefix} ${content}`)\n      }\n    })\n\n    workerProcess.on(\"error\", (err) => {\n      Effect.runSync(\n        Effect.logError(`Worker error: ${err.message}`).pipe(\n          Effect.annotateLogs(\"function\", fn.functionName),\n        ),\n      )\n    })\n\n    workerProcess.on(\"close\", (code) => {\n      Effect.runSync(\n        Effect.logInfo(`Worker exited with code ${code}`).pipe(\n          Effect.annotateLogs(\"function\", fn.functionName),\n        ),\n      )\n    })\n\n    return workerProcess\n  })\n\n/**\n * Process responses from a Node.js worker and dispatch to waiting callers.\n */\nconst processWorkerResponses = (\n  worker: NodejsWorker,\n  client: ReturnType<typeof makeAppSyncClient>,\n  invocationContexts: Map<string, InvocationContext>,\n) =>\n  Effect.gen(function* () {\n    const responseChannel = buildChannelName.response(worker.fn.functionName)\n\n    // Continuously process responses from the worker\n    while (true) {\n      const result = yield* waitForResponse(worker.runtimeState)\n\n      let response: ResponseMessage\n      if (isLambdaResponse(result)) {\n        response = {\n          type: \"response\",\n          requestId: result.requestId,\n          result: result.body,\n        }\n      } else if (isLambdaError(result)) {\n        response = {\n          type: \"response\",\n          requestId: result.requestId,\n          error: {\n            errorType: result.errorType,\n            errorMessage: result.errorMessage,\n            stackTrace: result.stackTrace,\n          },\n        }\n      } else if (isLambdaInitError(result)) {\n        yield* Effect.logError(\n          `[Local] Worker init error: ${result.errorType}: ${result.errorMessage}`,\n        )\n        continue\n      } else {\n        continue\n      }\n\n      // Send response back via AppSync\n      yield* client.publishResponse(responseChannel, response)\n      yield* Effect.logDebug(`[Local] Sent response for ${response.requestId}`)\n\n      // Print end marker with timing\n      const ctx = invocationContexts.get(response.requestId)\n      if (ctx) {\n        const durationMs = Date.now() - ctx.start\n        if (response.error) {\n          console.log(\n            `[${ctx.num}] \\u2514\\u2500\\u2500 \\u2717 ${response.error.errorType}: ${response.error.errorMessage} (${durationMs}ms) \\u2500\\u2500`,\n          )\n        } else {\n          console.log(\n            `[${ctx.num}] \\u2514\\u2500\\u2500 \\u2713 Done (${durationMs}ms) \\u2500\\u2500`,\n          )\n        }\n        invocationContexts.delete(response.requestId)\n      }\n    }\n  })\n\n/**\n * Environment variables to completely filter out from invocation env.\n * These are not relevant for local execution.\n */\nconst ENV_VARS_TO_FILTER = new Set([\"AWS_LAMBDA_LOG_STREAM_NAME\"])\n\n/**\n * Environment variables to ignore when calculating if env has changed.\n * These change frequently but don't require a container restart.\n *\n * TODO: AWS credentials (ACCESS_KEY_ID, SECRET_ACCESS_KEY, SESSION_TOKEN)\n * do expire and will need refreshing. For now we ignore them to avoid\n * unnecessary restarts, but we should implement credential refresh logic\n * that updates the container's credentials without a full restart.\n */\nconst ENV_VARS_TO_IGNORE_IN_DIFF = new Set([\n  \"AWS_ACCESS_KEY_ID\",\n  \"AWS_SECRET_ACCESS_KEY\",\n  \"AWS_SESSION_TOKEN\",\n])\n\n/**\n * Filter out irrelevant env vars from invocation environment.\n */\nconst sanitizeInvocationEnv = (\n  env: Record<string, string>,\n): Record<string, string> => {\n  const result: Record<string, string> = {}\n  for (const [key, value] of Object.entries(env)) {\n    if (!ENV_VARS_TO_FILTER.has(key)) {\n      result[key] = value\n    }\n  }\n  return result\n}\n\n/**\n * Compute which environment variables have changed between two env objects.\n * Returns a human-readable summary of the changes.\n * Ignores env vars in ENV_VARS_TO_IGNORE_IN_DIFF.\n */\nconst getEnvDiff = (\n  oldEnv: Record<string, string>,\n  newEnv: Record<string, string>,\n): string => {\n  const changes: string[] = []\n\n  // Check for added or modified keys\n  for (const key of Object.keys(newEnv)) {\n    if (ENV_VARS_TO_IGNORE_IN_DIFF.has(key)) continue\n    if (!(key in oldEnv)) {\n      changes.push(`+${key}`)\n    } else if (oldEnv[key] !== newEnv[key]) {\n      changes.push(`~${key}`)\n    }\n  }\n\n  // Check for removed keys\n  for (const key of Object.keys(oldEnv)) {\n    if (ENV_VARS_TO_IGNORE_IN_DIFF.has(key)) continue\n    if (!(key in newEnv)) {\n      changes.push(`-${key}`)\n    }\n  }\n\n  return changes.join(\", \")\n}\n\n/**\n * Ensure a Node.js worker is started for a function.\n * If the worker already exists, return it.\n * Otherwise, create the Runtime API server, add to workers map, and start the worker.\n */\nconst ensureWorkerStarted = (\n  fn: DiscoveredFunction,\n  invocationEnv: Record<string, string>,\n  workersRef: Ref.Ref<Map<string, NodejsWorker>>,\n  serverScope: Scope.Scope,\n  projectRoot: string,\n  appSyncClient: ReturnType<typeof makeAppSyncClient>,\n  invocationContexts: Map<string, InvocationContext>,\n): Effect.Effect<NodejsWorker, Error> =>\n  Effect.gen(function* () {\n    // Check if worker already exists\n    const currentWorkers = yield* Ref.get(workersRef)\n    const existing = currentWorkers.get(fn.functionName)\n    if (existing) {\n      // Check if env vars have changed (e.g., after CDK redeploy)\n      const envDiff = getEnvDiff(existing.env, invocationEnv)\n      if (envDiff) {\n        yield* Effect.logInfo(\n          `[Local] Environment changed for ${fn.functionName}, restarting worker... (${envDiff})`,\n        )\n        // Kill the old worker\n        yield* Effect.try(() => existing.workerProcess.kill(\"SIGTERM\")).pipe(\n          Effect.catchAll(() => Effect.void),\n        )\n        // Remove from map so we create a new one below\n        currentWorkers.delete(fn.functionName)\n        yield* Ref.set(workersRef, currentWorkers)\n      } else {\n        return existing\n      }\n    }\n\n    // Worker doesn't exist - start it lazily\n    yield* Effect.logDebug(\n      `[Local] Starting Node.js worker for first invocation of ${fn.functionName}...`,\n    )\n\n    // Create Runtime API server on ephemeral port\n    const { port, state: runtimeState } = yield* startRuntimeApiServer({\n      functionMetadata: {\n        functionName: fn.functionName,\n        functionVersion: \"$LATEST\",\n        handler: fn.localHandler,\n      },\n    }).pipe(Effect.provideService(Scope.Scope, serverScope))\n\n    // Create worker object (without process initially - will be set after start)\n    // We add to map BEFORE starting worker to handle concurrent invocations\n    const worker: NodejsWorker = {\n      fn,\n      runtimeState,\n      port,\n      workerProcess: undefined as unknown as ChildProcess, // Will be set shortly\n      env: invocationEnv,\n    }\n\n    // Add to map immediately to prevent race conditions\n    currentWorkers.set(fn.functionName, worker)\n    yield* Ref.set(workersRef, currentWorkers)\n\n    // Start the worker process - remove from map on failure\n    const workerProcess = yield* startNodejsWorker(\n      fn,\n      port,\n      projectRoot,\n      invocationEnv,\n      invocationContexts,\n    ).pipe(\n      Effect.catchAll((error) =>\n        Effect.gen(function* () {\n          // Remove broken worker from map on failure\n          yield* Effect.logError(\n            `[Local] Failed to start worker for ${fn.functionName}: ${error}`,\n          )\n          const current = yield* Ref.get(workersRef)\n          current.delete(fn.functionName)\n          yield* Ref.set(workersRef, current)\n          return yield* Effect.fail(error)\n        }),\n      ),\n    )\n\n    // Update worker with the process\n    worker.workerProcess = workerProcess\n\n    // Start processing responses in the background\n    Effect.runFork(\n      processWorkerResponses(worker, appSyncClient, invocationContexts),\n    )\n\n    return worker\n  })\n\n/**\n * Ensure a container is started for a function.\n * If the container already exists, return it.\n * If the env vars have changed, restart the container.\n * Otherwise, create the Runtime API server, add to containers map, and start the container.\n */\nconst ensureContainerStarted = (\n  fn: DiscoveredFunction,\n  invocationEnv: Record<string, string>,\n  containersRef: Ref.Ref<Map<string, FunctionContainer>>,\n  serverScope: Scope.Scope,\n  projectRoot: string,\n  idleTimeoutMs: number,\n  pollTimeoutMs: number,\n  appSyncClient: ReturnType<typeof makeAppSyncClient>,\n  invocationContexts: Map<string, InvocationContext>,\n): Effect.Effect<FunctionContainer, Error, CommandExecutor.CommandExecutor> =>\n  Effect.gen(function* () {\n    // Check if container already exists\n    const currentContainers = yield* Ref.get(containersRef)\n    const existing = currentContainers.get(fn.functionName)\n    if (existing) {\n      // Check if env vars have changed (e.g., after CDK redeploy)\n      const envDiff = getEnvDiff(existing.env, invocationEnv)\n      if (envDiff) {\n        // IMPORTANT: Update env immediately (before any yields) to prevent\n        // other concurrent invocations from also detecting the change and\n        // triggering duplicate restarts\n        existing.env = invocationEnv\n\n        // If already restarting for env change, just return the existing container\n        // The invocation will be queued and picked up by the new container\n        if (existing.isRebuilding) {\n          yield* Effect.logDebug(\n            `[Local] Environment change restart already in progress for ${fn.functionName}, queueing invocation`,\n          )\n          return existing\n        }\n\n        yield* Effect.logInfo(\n          `[Local] Environment changed for ${fn.functionName}, restarting container... (${envDiff})`,\n        )\n        existing.isRebuilding = true\n\n        // Stop the Docker container forcefully (timeout=1s)\n        // This sends SIGTERM and then SIGKILL after 1 second\n        // We must do this BEFORE interrupting the fiber, because the fiber is\n        // blocked waiting for the docker process to exit\n        yield* Effect.gen(function* () {\n          const docker = yield* Docker\n          yield* docker.stop(existing.containerName, 1).pipe(Effect.scoped)\n        }).pipe(\n          Effect.provide(DockerLive),\n          Effect.catchAll((error) =>\n            Effect.logDebug(\n              `[Local] Error stopping container (may already be stopped): ${error}`,\n            ),\n          ),\n        )\n\n        // Now interrupt the fiber (it should complete quickly since container stopped)\n        yield* Fiber.interrupt(existing.containerFiber).pipe(\n          Effect.catchAll(() => Effect.void),\n        )\n\n        // Restart the container with new environment, reusing existing RuntimeAPI\n        const newFiber = yield* startFunctionContainer(\n          fn,\n          existing.port,\n          projectRoot,\n          invocationEnv,\n          invocationContexts,\n        ).pipe(\n          Effect.catchAll((error) =>\n            Effect.gen(function* () {\n              yield* Effect.logError(\n                `[Local] Failed to restart container for ${fn.functionName}: ${error}`,\n              )\n              existing.isRebuilding = false\n              return yield* Effect.fail(error)\n            }),\n          ),\n        )\n\n        // Update container with new fiber\n        existing.containerFiber = newFiber\n        existing.isRebuilding = false\n\n        return existing\n      } else {\n        return existing\n      }\n    }\n\n    // Container doesn't exist - start it lazily\n    yield* Effect.logInfo(\n      `[Local] Starting container for first invocation of ${fn.functionName}...`,\n    )\n\n    // Create Runtime API server on ephemeral port\n    // Use poll timeout that's shorter than idle timeout so container exits naturally\n    const { port, state: runtimeState } = yield* startRuntimeApiServer({\n      pollTimeoutMs,\n      functionMetadata: {\n        functionName: fn.functionName,\n        functionVersion: \"$LATEST\",\n        handler: \"index.handler\",\n      },\n    }).pipe(Effect.provideService(Scope.Scope, serverScope))\n\n    // Generate container name and image name\n    const containerName = `lambda-${fn.functionName.replace(/[^a-zA-Z0-9]/g, \"-\")}`\n    const imageName = `live-lambda-${fn.functionName.toLowerCase().replace(/[^a-z0-9-]/g, \"-\")}`\n\n    // Create container object (without fiber initially - will be set after start)\n    // We add to map BEFORE starting container to handle concurrent invocations\n    const container: FunctionContainer = {\n      fn,\n      runtimeState,\n      port,\n      containerFiber: undefined as unknown as Fiber.RuntimeFiber<void, Error>, // Will be set shortly\n      containerName,\n      imageName,\n      isRebuilding: false,\n      pendingResponses: new Map(),\n      idleTimerFiber: null,\n      env: invocationEnv,\n    }\n\n    // Add to map immediately to prevent race conditions\n    currentContainers.set(fn.functionName, container)\n    yield* Ref.set(containersRef, currentContainers)\n\n    // Build and start the container - remove from map on failure\n    const containerFiber = yield* startFunctionContainer(\n      fn,\n      port,\n      projectRoot,\n      invocationEnv,\n      invocationContexts,\n    ).pipe(\n      Effect.catchAll((error) =>\n        Effect.gen(function* () {\n          // Remove broken container from map on failure\n          yield* Effect.logError(\n            `[Local] Failed to start container for ${fn.functionName}: ${error}`,\n          )\n          const current = yield* Ref.get(containersRef)\n          current.delete(fn.functionName)\n          yield* Ref.set(containersRef, current)\n          return yield* Effect.fail(error)\n        }),\n      ),\n    )\n\n    // Update container with the fiber\n    container.containerFiber = containerFiber\n\n    // Start processing responses in the background using forkDaemon\n    yield* processContainerResponses(\n      container,\n      containersRef,\n      idleTimeoutMs,\n      appSyncClient,\n      invocationContexts,\n    ).pipe(Effect.forkDaemon)\n\n    return container\n  })\n\n/**\n * Rebuild a Docker container after file changes.\n * Invocations arriving during rebuild will be queued and picked up by the new container.\n */\nconst rebuildDockerContainer = (\n  functionId: string,\n  containersRef: Ref.Ref<Map<string, FunctionContainer>>,\n  projectRoot: string,\n): Effect.Effect<void, Error, CommandExecutor.CommandExecutor> =>\n  Effect.gen(function* () {\n    const containers = yield* Ref.get(containersRef)\n    const container = containers.get(functionId)\n\n    if (!container) {\n      yield* Effect.logInfo(\n        `[Local] Cannot rebuild ${functionId} - container not found`,\n      )\n      return\n    }\n\n    // Mark as rebuilding - invocations will still queue but we log it\n    container.isRebuilding = true\n\n    yield* Effect.logDebug(`[Local] Rebuilding container for ${functionId}...`)\n\n    const docker = yield* Docker\n\n    // Stop the existing container using Docker service\n    const containerId = container.containerName\n    yield* Effect.logInfo(\n      `[Local] Stopping container with name prefix: ${containerId}`,\n    )\n\n    const stopCount = yield* docker.stop(containerId).pipe(\n      Effect.scoped,\n      Effect.catchAll((error) =>\n        Effect.gen(function* () {\n          yield* Effect.logInfo(`Note: Container stop had issue: ${error}`)\n          return 0\n        }),\n      ),\n    )\n    yield* Effect.logDebug(`[Local] Stopped ${stopCount} container(s)`)\n\n    // Resolve the context path\n    const fn = container.fn\n    const contextPath = fn.dockerContextPath?.startsWith(\"/\")\n      ? fn.dockerContextPath\n      : `${projectRoot}/${fn.dockerContextPath}`\n\n    // Determine platform from architecture\n    const platform = fn.architecture === \"arm64\" ? \"linux/arm64\" : \"linux/amd64\"\n\n    // Rebuild the Docker image\n    yield* docker\n      .build({\n        contextPath,\n        imageName: container.imageName,\n        platform,\n      })\n      .pipe(Effect.scoped)\n\n    // Inspect the image to get original entrypoint/cmd for extension wrapper\n    const imageConfig = yield* docker\n      .inspect(container.imageName)\n      .pipe(Effect.scoped)\n    const extensionWrapper = buildExtensionWrapperCommand(\n      imageConfig.entrypoint,\n      imageConfig.cmd,\n    )\n\n    // Restart the container by triggering container startup\n    // The existing fiber will have exited when we stopped the container\n    // We need to start a new one\n    const dockerRuntime = yield* docker.getRuntimeInfo()\n    const runtimeApiHost = dockerRuntime.isDockerDesktop\n      ? \"host.docker.internal\"\n      : \"runtime.api\"\n\n    yield* Effect.logInfo(\n      `[Local] New container will connect to Runtime API at ${runtimeApiHost}:${container.port}`,\n    )\n\n    const containerConfig = makeLambdaContainerConfig({\n      imageUri: container.imageName,\n      runtimeApiHost,\n      runtimeApiPort: container.port,\n      functionName: fn.functionName,\n      functionVersion: \"$LATEST\",\n      memoryMB: fn.memoryMB,\n      timeoutSeconds: 3600,\n      platform,\n    })\n\n    // Apply extension wrapper to start extensions before the app\n    containerConfig.entrypoint = extensionWrapper.entrypoint\n    containerConfig.command = extensionWrapper.command\n\n    // Start the new container\n    yield* Effect.logDebug(\n      `[Local] Starting new container for ${functionId}...`,\n    )\n\n    // Use Effect.forkDaemon to run the container independently\n    const newFiber = yield* docker.run(containerConfig).pipe(\n      Effect.scoped,\n      Effect.tap((result) =>\n        Effect.gen(function* () {\n          // Only suppress errors for expected exit codes from docker stop:\n          // 0: clean, 143: SIGTERM, 137: SIGKILL\n          const code = result.exitCode\n          if (code !== 0 && code !== 143 && code !== 137) {\n            yield* Effect.logError(\n              `Container for ${functionId} exited with code ${code}`,\n            )\n            yield* Effect.logError(`stderr: ${result.stderr}`)\n          }\n        }),\n      ),\n      Effect.map(() => undefined as void),\n      Effect.catchAll((error) =>\n        Effect.logError(`Container error for ${functionId}: ${error}`),\n      ),\n      Effect.forkDaemon,\n    )\n\n    // Update the container state with the new fiber\n    container.containerFiber = newFiber\n\n    // Wait a moment for the container to start and begin polling\n    yield* Effect.sleep(\"2 seconds\")\n\n    container.isRebuilding = false\n    yield* Effect.logDebug(`[Local] Container rebuilt for ${functionId}`)\n  }).pipe(Effect.provide(DockerLive))\n\n/**\n * Handle incoming invocations for a Docker container function by queueing them.\n * Starts the container lazily if not already running.\n * Invocations are queued even if a rebuild is in progress - the new container will pick them up.\n */\nconst handleDockerInvocation = (\n  fn: DiscoveredFunction,\n  invocation: InvocationMessage,\n  containersRef: Ref.Ref<Map<string, FunctionContainer>>,\n  serverScope: Scope.Scope,\n  projectRoot: string,\n  idleTimeoutMs: number,\n  pollTimeoutMs: number,\n  appSyncClient: ReturnType<typeof makeAppSyncClient>,\n  invocationCounter: Ref.Ref<number>,\n  invocationContexts: Map<string, InvocationContext>,\n): Effect.Effect<void, Error, CommandExecutor.CommandExecutor> =>\n  Effect.gen(function* () {\n    // Ensure container is started (lazy startup on first invocation)\n    // Use env from invocation (captured from bridge Lambda on first invoke)\n    const container = yield* ensureContainerStarted(\n      fn,\n      sanitizeInvocationEnv(invocation.env ?? {}),\n      containersRef,\n      serverScope,\n      projectRoot,\n      idleTimeoutMs,\n      pollTimeoutMs,\n      appSyncClient,\n      invocationContexts,\n    )\n\n    // Assign invocation number and track context for logging\n    const invocationNum = yield* Ref.updateAndGet(\n      invocationCounter,\n      (n) => n + 1,\n    )\n    const startTime = Date.now()\n    invocationContexts.set(invocation.requestId, {\n      num: invocationNum,\n      start: startTime,\n      fn: fn.functionName,\n      isDocker: true,\n    })\n\n    // Print start marker\n    console.log(\n      `\\n[${invocationNum}] \\u250c\\u2500\\u2500 [Docker] ${fn.functionName} \\u2500\\u2500`,\n    )\n\n    // Log if queuing during a rebuild\n    if (container.isRebuilding) {\n      yield* Effect.logDebug(\n        `[Local] Queueing invocation ${invocation.requestId} for ${fn.functionName} (rebuild in progress, will be picked up by new container)`,\n      )\n    } else {\n      yield* Effect.logDebug(\n        `[Local] Queueing invocation ${invocation.requestId} for ${fn.functionName}`,\n      )\n    }\n\n    const lambdaInvocation: LambdaInvocation = {\n      requestId: invocation.requestId,\n      event: invocation.event,\n      invokedFunctionArn: invocation.context.invokedFunctionArn,\n      deadlineMs: Date.now() + invocation.context.getRemainingTimeInMillis,\n      functionName: fn.functionName,\n      functionVersion: invocation.context.functionVersion,\n      memoryLimitMB: fn.memoryMB,\n      logGroupName: invocation.context.logGroupName,\n      logStreamName: invocation.context.logStreamName,\n    }\n\n    yield* Effect.logDebug(\n      `[Local] Queueing to Runtime API on port ${container.port}`,\n    )\n    // Notify extensions about the invocation (for Lambda Web Adapter support)\n    yield* notifyExtensionsInvoke(container.runtimeState, lambdaInvocation)\n    yield* queueInvocation(container.runtimeState, lambdaInvocation)\n    yield* Effect.logDebug(`[Local] Invocation queued successfully`)\n  })\n\n/**\n * Handle incoming invocations for a Node.js function by queueing them.\n * Starts the worker lazily if not already running.\n */\nconst handleNodejsInvocation = (\n  fn: DiscoveredFunction,\n  invocation: InvocationMessage,\n  workersRef: Ref.Ref<Map<string, NodejsWorker>>,\n  serverScope: Scope.Scope,\n  projectRoot: string,\n  appSyncClient: ReturnType<typeof makeAppSyncClient>,\n  invocationCounter: Ref.Ref<number>,\n  invocationContexts: Map<string, InvocationContext>,\n): Effect.Effect<void, Error> =>\n  Effect.gen(function* () {\n    // Get env vars from invocation (forwarded from bridge Lambda)\n    const invocationEnv = sanitizeInvocationEnv(invocation.env ?? {})\n\n    // Ensure worker is started (lazy startup on first invocation)\n    const worker = yield* ensureWorkerStarted(\n      fn,\n      invocationEnv,\n      workersRef,\n      serverScope,\n      projectRoot,\n      appSyncClient,\n      invocationContexts,\n    )\n\n    // Assign invocation number and track context for logging\n    const invocationNum = yield* Ref.updateAndGet(\n      invocationCounter,\n      (n) => n + 1,\n    )\n    const startTime = Date.now()\n    invocationContexts.set(invocation.requestId, {\n      num: invocationNum,\n      start: startTime,\n      fn: fn.functionName,\n      isDocker: false,\n    })\n\n    // Print start marker\n    console.log(\n      `\\n[${invocationNum}] \\u250c\\u2500\\u2500 ${fn.functionName} \\u2500\\u2500`,\n    )\n\n    yield* Effect.logDebug(\n      `[Local] Queueing invocation ${invocation.requestId} for ${fn.functionName}`,\n    )\n\n    const lambdaInvocation: LambdaInvocation = {\n      requestId: invocation.requestId,\n      event: invocation.event,\n      invokedFunctionArn: invocation.context.invokedFunctionArn,\n      deadlineMs: Date.now() + invocation.context.getRemainingTimeInMillis,\n      functionName: fn.functionName,\n      functionVersion: invocation.context.functionVersion,\n      memoryLimitMB: fn.memoryMB,\n      logGroupName: invocation.context.logGroupName,\n      logStreamName: invocation.context.logStreamName,\n    }\n\n    yield* queueInvocation(worker.runtimeState, lambdaInvocation)\n  })\n\n/**\n * CDK watch event types.\n */\ntype CdkWatchEvent =\n  | { readonly _tag: \"StackDiscovered\"; readonly stackName: string }\n  | { readonly _tag: \"DeployComplete\" }\n\n/**\n * Start CDK watch process with CDK_LIVE=true using Effect's Command.\n * Returns the process and a queue of events.\n */\nconst startCdkWatch = (\n  options: {\n    profile?: string\n    region?: string\n    stacks?: string[]\n    all?: boolean\n  },\n  scope: Scope.Scope,\n): Effect.Effect<\n  { process: EffectProcess; events: Queue.Queue<CdkWatchEvent> },\n  Error,\n  CommandExecutor.CommandExecutor\n> =>\n  Effect.gen(function* () {\n    const args = [\n      \"cdk\",\n      \"watch\",\n      \"--hotswap-fallback\",\n      \"--no-logs\",\n      \"--method=direct\",\n    ]\n\n    if (options.stacks && options.stacks.length > 0) {\n      args.push(...options.stacks)\n    } else if (options.all) {\n      args.push(\"--all\")\n    }\n\n    if (options.profile) {\n      args.push(\"--profile\", options.profile)\n    }\n\n    const env: Record<string, string> = {\n      ...process.env,\n      CDK_LIVE: \"true\",\n    } as Record<string, string>\n\n    if (options.region) {\n      env.AWS_REGION = options.region\n      env.CDK_DEFAULT_REGION = options.region\n    }\n\n    yield* Effect.logDebug(`Starting: npx ${args.join(\" \")}`)\n\n    const command = PlatformCommand.make(\"npx\", ...args).pipe(\n      PlatformCommand.env(env),\n      PlatformCommand.runInShell(true),\n      PlatformCommand.stdin(\"inherit\"),\n    )\n\n    // Extend the process resource lifetime to the provided scope\n    const proc = yield* PlatformCommand.start(command).pipe(Scope.extend(scope))\n\n    // Event queue for callers to subscribe to\n    const events = yield* Queue.unbounded<CdkWatchEvent>()\n\n    // State tracking for deploy status messages\n    let isDeploying = false\n    let isFirstDeploy = true\n    let outputBuffer = \"\"\n    const discoveredStacks = new Set<string>()\n\n    // Patterns to detect CDK watch behavior\n    const deployCompletePattern = /✅\\s+\\S+|Deployment time:/\n    const deployStartPattern = /Deploying|hotswap|Hotswapping|Bundling/i\n    const noChangesPattern = /no changes|identical|up to date/i\n    const errorPattern = /error|failed|Error|Failed|ERR!/i\n    const stackNamePattern = /^(\\S+):\\s*deploying|✅\\s+(\\S+)/\n\n    // Merge stdout and stderr, decode to text, split into lines\n    const outputStream = Stream.merge(proc.stdout, proc.stderr).pipe(\n      Stream.decodeText(),\n      Stream.splitLines,\n    )\n\n    // Fork stream processing in background\n    yield* outputStream.pipe(\n      Stream.runForEach((line) =>\n        Effect.gen(function* () {\n          outputBuffer += line + \"\\n\"\n\n          // Log all output at debug level\n          yield* Effect.logDebug(`[CDK] ${line}`)\n\n          // Try to extract stack name\n          const match = stackNamePattern.exec(line.trim())\n          if (match) {\n            const stackName = match[1] || match[2]\n            if (stackName && !discoveredStacks.has(stackName)) {\n              discoveredStacks.add(stackName)\n              yield* Queue.offer(events, {\n                _tag: \"StackDiscovered\",\n                stackName,\n              })\n            }\n          }\n\n          // Detect deploy start\n          if (!isDeploying && deployStartPattern.test(line)) {\n            isDeploying = true\n            if (!isFirstDeploy) {\n              yield* Effect.logInfo(\"[CDK] Deploying...\")\n            }\n          }\n\n          // Detect errors - output immediately\n          if (errorPattern.test(line)) {\n            yield* Effect.sync(() => process.stderr.write(line + \"\\n\"))\n          }\n\n          // Check for deploy completion (only trigger once per deploy cycle)\n          if (isDeploying && deployCompletePattern.test(line)) {\n            isDeploying = false\n            isFirstDeploy = false\n            outputBuffer = \"\"\n            yield* Effect.logInfo(\"[CDK] Deploy complete\")\n            // Small delay to ensure AWS has propagated the changes\n            yield* Effect.sleep(\"1 second\")\n            yield* Queue.offer(events, { _tag: \"DeployComplete\" })\n          }\n\n          // Check for no changes\n          if (noChangesPattern.test(line)) {\n            isDeploying = false\n            outputBuffer = \"\"\n          }\n        }),\n      ),\n      // Log errors and exit code when stream ends\n      Effect.tapError((error) =>\n        Effect.logError(`[CDK] CDK watch error: ${error}`),\n      ),\n      Effect.ensuring(\n        proc.exitCode.pipe(\n          Effect.flatMap((code) =>\n            code !== 0\n              ? Effect.logError(`[CDK] CDK watch exited with code ${code}`)\n              : Effect.logDebug(`[CDK] CDK watch exited with code ${code}`),\n          ),\n          Effect.catchAll(() => Effect.void),\n        ),\n      ),\n      Effect.fork,\n    )\n\n    return { process: proc, events }\n  })\n\n/**\n * Common CLI options.\n */\nconst profileOption = Options.text(\"profile\").pipe(\n  Options.optional,\n  Options.withDescription(\"AWS profile to use\"),\n)\n\nconst regionOption = Options.text(\"region\").pipe(\n  Options.optional,\n  Options.withDescription(\"AWS region\"),\n)\n\nconst qualifierOption = Options.text(\"qualifier\").pipe(\n  Options.withDefault(\"hnb659fds\"),\n  Options.withDescription(\"CDK bootstrap qualifier\"),\n)\n\nconst stacksOption = Options.text(\"stacks\").pipe(\n  Options.optional,\n  Options.withDescription(\n    \"Stack names to deploy (comma-separated, default: all)\",\n  ),\n)\n\nconst allStacksOption = Options.boolean(\"all\").pipe(\n  Options.withDefault(false),\n  Options.withDescription(\"Deploy all stacks (like cdk deploy --all)\"),\n)\n\nconst debugOption = Options.boolean(\"debug\").pipe(\n  Options.withDefault(false),\n  Options.withDescription(\"Enable debug logging\"),\n)\n\nconst idleTimeoutOption = Options.integer(\"idle-timeout\").pipe(\n  Options.withDefault(DEFAULT_IDLE_TIMEOUT_MS),\n  Options.withDescription(\n    \"Idle timeout in ms before stopping containers (default: 230000)\",\n  ),\n)\n\nconst pollTimeoutOption = Options.integer(\"poll-timeout\").pipe(\n  Options.optional,\n  Options.withDescription(\n    \"Poll timeout in ms for Runtime API (default: idle-timeout - 10s). Must be shorter than idle-timeout.\",\n  ),\n)\n\n/**\n * Local command definition.\n */\nexport const localCommand = Command.make(\n  \"local\",\n  {\n    profile: profileOption,\n    region: regionOption,\n    qualifier: qualifierOption,\n    stacks: stacksOption,\n    all: allStacksOption,\n    debug: debugOption,\n    idleTimeout: idleTimeoutOption,\n    pollTimeout: pollTimeoutOption,\n  },\n  ({\n    profile,\n    region,\n    qualifier,\n    stacks,\n    all,\n    debug,\n    idleTimeout,\n    pollTimeout,\n  }) => {\n    const logLevel = debug ? LogLevel.Debug : LogLevel.Info\n    return Effect.gen(function* () {\n      yield* Effect.logInfo(\"[Local] Starting local Lambda development...\")\n\n      const profileValue = profile._tag === \"Some\" ? profile.value : undefined\n      const regionValue = region._tag === \"Some\" ? region.value : undefined\n      const stacksFromOption =\n        stacks._tag === \"Some\"\n          ? stacks.value.split(\",\").map((s) => s.trim())\n          : undefined\n\n      // Poll timeout should be shorter than idle timeout so containers exit naturally\n      // before we try to stop them. Default: idle timeout - 10 seconds.\n      const effectivePollTimeout =\n        pollTimeout._tag === \"Some\"\n          ? pollTimeout.value\n          : Math.max(idleTimeout - 10_000, 5_000) // At least 5 seconds\n\n      if (profileValue) {\n        process.env.AWS_PROFILE = profileValue\n      }\n      if (regionValue) {\n        process.env.AWS_REGION = regionValue\n      }\n\n      // Bootstrap check must complete first\n      yield* ensureBootstrap({\n        qualifier,\n        profile: profileValue,\n        region: regionValue,\n      })\n\n      // Stack filter - populated from CDK watch output, used to filter Lambda discovery\n      // Using object so closure in startOrUpdateDaemon sees updates\n      const filterState = { stacks: stacksFromOption ?? ([] as string[]) }\n\n      // Create a long-lived scope for all Runtime API servers and CDK watch process\n      // Resources will run until this scope is closed (when the program ends)\n      const serverScope = yield* Scope.make()\n\n      // Start CDK watch immediately after bootstrap\n      // Stack names are discovered from CDK watch output (no need for separate cdk ls)\n      yield* Effect.logInfo(\"[CDK] Deploying...\")\n\n      const { process: cdkWatchProcess, events: cdkEvents } =\n        yield* startCdkWatch(\n          {\n            profile: profileValue,\n            region: regionValue,\n            stacks: stacksFromOption,\n            all,\n          },\n          serverScope,\n        )\n\n      // Track running Docker containers by function name\n      const containers = yield* Ref.make<Map<string, FunctionContainer>>(\n        new Map(),\n      )\n\n      // Track running Node.js workers by function name\n      const workers = yield* Ref.make<Map<string, NodejsWorker>>(new Map())\n\n      // Track registered functions (for lazy container/worker startup)\n      const registeredFunctions = yield* Ref.make<\n        Map<string, DiscoveredFunction>\n      >(new Map())\n\n      // Track Docker functions with active file watchers\n      const watchedDockerFunctions = new Set<string>()\n\n      // Invocation tracking for improved logging output\n      const invocationCounter = yield* Ref.make<number>(0)\n      // Use a plain Map for invocation contexts so it can be accessed from stream callbacks\n      const invocationContexts = new Map<string, InvocationContext>()\n\n      // Project root is the current working directory (where CDK app lives)\n      const projectRoot = process.cwd()\n\n      let appSyncClient: ReturnType<typeof makeAppSyncClient> | null = null\n\n      // Track if we've logged the \"Watching\" message (only log once)\n      const logState = { hasLoggedWatching: false, hasDiscoveredOnce: false }\n\n      // Function to start/update the daemon with discovered functions\n      const startOrUpdateDaemon = Effect.gen(function* () {\n        // Only show \"Discovering functions...\" on first run\n        if (!logState.hasDiscoveredOnce) {\n          yield* Effect.logInfo(\"[Local] Discovering functions...\")\n        }\n\n        // Get AppSync endpoints (may need to wait for first deploy)\n        if (!appSyncClient) {\n          yield* Effect.logDebug(\n            \"[Local] Reading AppSync endpoints from SSM...\",\n          )\n          const endpoints = yield* getAppSyncEndpoints(qualifier).pipe(\n            Effect.retry({ times: 10, schedule: Schedule.spaced(\"3 seconds\") }),\n          )\n          yield* Effect.logDebug(\n            `[Local] HTTP endpoint: ${endpoints.httpEndpoint}`,\n          )\n          appSyncClient = makeAppSyncClient(endpoints)\n        }\n\n        // Discover functions (filtered to stacks in this CDK project)\n        yield* Effect.logDebug(\"[Local] Discovering Lambda functions...\")\n        const functions = yield* discoverFunctions(filterState.stacks)\n\n        if (functions.length === 0) {\n          yield* Effect.logInfo(\n            \"[Local] No functions found with live-lambda tags yet. Have you patched your CDK project (bootstrap) and added the LiveLambdaAspect?\",\n          )\n          return\n        }\n\n        const currentRegistered = yield* Ref.get(registeredFunctions)\n\n        // Collect new functions to register\n        const newFunctions: Array<{\n          fn: DiscoveredFunction\n          isDocker: boolean\n        }> = []\n\n        for (const fn of functions) {\n          const isDocker = Boolean(fn.dockerContextPath)\n          const isNodejs = !isDocker && Boolean(fn.localHandler)\n\n          if (!isDocker && !isNodejs) {\n            yield* Effect.logDebug(\n              `[Local] Skipping ${fn.functionName} - no Docker context or local handler`,\n            )\n            continue\n          }\n\n          if (currentRegistered.has(fn.functionName)) {\n            yield* Effect.logDebug(\n              `[Local] Already watching ${fn.functionName}`,\n            )\n            continue\n          }\n\n          newFunctions.push({ fn, isDocker })\n        }\n\n        // Print summary of discovered functions\n        if (newFunctions.length > 0) {\n          const summary = newFunctions\n            .map(({ fn, isDocker }) => {\n              const mode = isDocker ? \"docker\" : \"node\"\n              const shortName = fn.functionName.includes(\"-\")\n                ? fn.functionName.split(\"-\").slice(-2, -1)[0] || fn.functionName\n                : fn.functionName\n              return `${shortName} (${mode})`\n            })\n            .join(\", \")\n          // Use different message for first discovery vs subsequent\n          const prefix = logState.hasDiscoveredOnce\n            ? \"[Local] Functions added:\"\n            : \"[Local] Functions:\"\n          yield* Effect.logInfo(`${prefix} ${summary}`)\n        }\n\n        // Register functions and set up subscriptions\n        for (const { fn, isDocker } of newFunctions) {\n          yield* Effect.logDebug(\n            `[Local] Registering ${fn.functionName} (${isDocker ? \"Docker\" : \"Node.js\"})`,\n          )\n\n          // Register the function\n          currentRegistered.set(fn.functionName, fn)\n\n          // Subscribe to invocations for this function\n          const invocationChannel = buildChannelName.invocation(fn.functionName)\n          yield* Effect.logDebug(\n            `[Local] Subscribing to invocations for ${fn.functionName}`,\n          )\n\n          // Subscribe using forkDaemon to run independently with context preserved\n          // Route to Docker or Node.js handler based on function type\n          if (isDocker) {\n            yield* appSyncClient!\n              .subscribeToInvocations(invocationChannel)\n              .pipe(\n                Stream.runForEach((invocation) =>\n                  handleDockerInvocation(\n                    fn,\n                    invocation,\n                    containers,\n                    serverScope,\n                    projectRoot,\n                    idleTimeout,\n                    effectivePollTimeout,\n                    appSyncClient!,\n                    invocationCounter,\n                    invocationContexts,\n                  ).pipe(\n                    Effect.catchAll((error) =>\n                      Effect.logError(\n                        `[Local] Docker invocation error: ${error}`,\n                      ),\n                    ),\n                  ),\n                ),\n                Effect.forkDaemon,\n              )\n          } else {\n            yield* appSyncClient!\n              .subscribeToInvocations(invocationChannel)\n              .pipe(\n                Stream.runForEach((invocation) =>\n                  handleNodejsInvocation(\n                    fn,\n                    invocation,\n                    workers,\n                    serverScope,\n                    projectRoot,\n                    appSyncClient!,\n                    invocationCounter,\n                    invocationContexts,\n                  ).pipe(\n                    Effect.catchAll((error) =>\n                      Effect.logError(\n                        `[Local] Node.js invocation error: ${error}`,\n                      ),\n                    ),\n                  ),\n                ),\n                Effect.forkDaemon,\n              )\n          }\n        }\n\n        yield* Ref.set(registeredFunctions, currentRegistered)\n\n        // Start file watchers for new Docker functions\n        const newDockerFunctions: WatchedDockerFunction[] = []\n        for (const fn of functions) {\n          if (\n            fn.dockerContextPath &&\n            !watchedDockerFunctions.has(fn.functionName)\n          ) {\n            // Resolve the context path\n            const contextPath = fn.dockerContextPath.startsWith(\"/\")\n              ? fn.dockerContextPath\n              : `${projectRoot}/${fn.dockerContextPath}`\n\n            newDockerFunctions.push({\n              functionId: fn.functionName,\n              dockerContextPath: contextPath,\n            })\n            watchedDockerFunctions.add(fn.functionName)\n          }\n        }\n\n        // Start watching new Docker contexts\n        if (newDockerFunctions.length > 0) {\n          yield* Effect.logDebug(\n            `[Local] Starting file watchers for ${newDockerFunctions.length} Docker function(s)...`,\n          )\n\n          // Fork a daemon fiber to handle file change events with context preserved\n          yield* watchDockerContexts(newDockerFunctions, 500).pipe(\n            Stream.runForEach((event) =>\n              Effect.gen(function* () {\n                yield* Effect.logDebug(\n                  `[Local] File changed in ${event.functionId}: ${event.filePath}`,\n                )\n                yield* rebuildDockerContainer(\n                  event.functionId,\n                  containers,\n                  projectRoot,\n                ).pipe(\n                  Effect.catchAll((error) =>\n                    Effect.logError(\n                      `[Local] Rebuild failed for ${event.functionId}: ${error}`,\n                    ),\n                  ),\n                )\n              }),\n            ),\n            Effect.forkDaemon,\n          )\n        }\n\n        if (!logState.hasLoggedWatching) {\n          logState.hasLoggedWatching = true\n          yield* Effect.logInfo(\"[Local] Ready for invocations\")\n        }\n\n        // Mark that we've completed first discovery\n        logState.hasDiscoveredOnce = true\n      })\n\n      // Handle CDK watch events (stack discovery and deploy completion)\n      yield* Queue.take(cdkEvents).pipe(\n        Effect.flatMap((event) =>\n          Effect.gen(function* () {\n            switch (event._tag) {\n              case \"StackDiscovered\":\n                // Add discovered stack to filter (if not using --stacks)\n                if (\n                  !stacksFromOption &&\n                  !filterState.stacks.includes(event.stackName)\n                ) {\n                  filterState.stacks.push(event.stackName)\n                  yield* Effect.logDebug(\n                    `[Local] Discovered stack: ${event.stackName}`,\n                  )\n                }\n                break\n              case \"DeployComplete\":\n                // Run daemon update on deploy completion\n                yield* startOrUpdateDaemon.pipe(\n                  Effect.catchAll((error) =>\n                    Effect.logError(`Failed to update daemon: ${error}`),\n                  ),\n                )\n                break\n            }\n          }),\n        ),\n        Effect.forever,\n        Effect.fork,\n      )\n\n      // Handle cleanup on exit\n      const cleanup = async () => {\n        await Effect.runPromise(\n          Effect.gen(function* () {\n            yield* Effect.logInfo(\"\\nShutting down...\")\n\n            // Stop CDK watch\n            yield* cdkWatchProcess\n              .kill(\"SIGTERM\")\n              .pipe(Effect.catchAll(() => Effect.void))\n\n            // Stop all Docker containers using the Docker service\n            const docker = yield* Docker\n            const currentContainers = yield* Ref.get(containers)\n            for (const [name, container] of currentContainers) {\n              yield* Effect.logInfo(`Stopping container: ${name}`)\n              yield* docker.stop(container.containerName).pipe(\n                Effect.scoped,\n                Effect.catchAll(() => Effect.void),\n              )\n            }\n\n            // Stop all Node.js workers\n            const currentWorkers = yield* Ref.get(workers)\n            for (const [name, worker] of currentWorkers) {\n              yield* Effect.logInfo(`Stopping worker: ${name}`)\n              yield* Effect.try(() =>\n                worker.workerProcess.kill(\"SIGTERM\"),\n              ).pipe(Effect.catchAll(() => Effect.void))\n            }\n\n            // Close the server scope to clean up Runtime API servers\n            yield* Scope.close(serverScope, Exit.void)\n          }).pipe(\n            Effect.provide(DockerLive),\n            Effect.provide(BunContext.layer),\n            Effect.provide(Logger.pretty),\n            Effect.provide(Logger.minimumLogLevel(logLevel)),\n          ),\n        )\n\n        process.exit(0)\n      }\n\n      process.on(\"SIGINT\", cleanup)\n      process.on(\"SIGTERM\", cleanup)\n\n      yield* Effect.logInfo(\"[Local] Press Ctrl+C to stop\")\n\n      // Keep the process running\n      yield* Effect.never\n    }).pipe(\n      Effect.provide(Logger.pretty),\n      Effect.provide(Logger.minimumLogLevel(logLevel)),\n    )\n  },\n).pipe(\n  Command.withDescription(\n    \"Start local Lambda development with CDK watch and Docker containers\",\n  ),\n)\n"]}
|