cdk-local-lambda 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (53) hide show
  1. package/LICENSE +202 -0
  2. package/README.md +94 -0
  3. package/lib/aspect/docker-function-hook.d.ts +18 -0
  4. package/lib/aspect/docker-function-hook.js +31 -0
  5. package/lib/aspect/live-lambda-aspect.d.ts +85 -0
  6. package/lib/aspect/live-lambda-aspect.js +277 -0
  7. package/lib/aspect/live-lambda-bootstrap.d.ts +17 -0
  8. package/lib/aspect/live-lambda-bootstrap.js +260 -0
  9. package/lib/aspect/nodejs-function-hook.d.ts +20 -0
  10. package/lib/aspect/nodejs-function-hook.js +27 -0
  11. package/lib/bootstrap-stack/bootstrap-stack.d.ts +60 -0
  12. package/lib/bootstrap-stack/bootstrap-stack.js +338 -0
  13. package/lib/cli/appsync/client.d.ts +30 -0
  14. package/lib/cli/appsync/client.js +227 -0
  15. package/lib/cli/cdk-app.d.ts +7 -0
  16. package/lib/cli/cdk-app.js +25 -0
  17. package/lib/cli/commands/bootstrap.d.ts +9 -0
  18. package/lib/cli/commands/bootstrap.js +50 -0
  19. package/lib/cli/commands/local.d.ts +40 -0
  20. package/lib/cli/commands/local.js +1172 -0
  21. package/lib/cli/daemon.d.ts +22 -0
  22. package/lib/cli/daemon.js +18 -0
  23. package/lib/cli/docker/container.d.ts +116 -0
  24. package/lib/cli/docker/container.js +414 -0
  25. package/lib/cli/docker/types.d.ts +71 -0
  26. package/lib/cli/docker/types.js +5 -0
  27. package/lib/cli/docker/watcher.d.ts +44 -0
  28. package/lib/cli/docker/watcher.js +115 -0
  29. package/lib/cli/index.d.ts +9 -0
  30. package/lib/cli/index.js +26 -0
  31. package/lib/cli/runtime-api/server.d.ts +102 -0
  32. package/lib/cli/runtime-api/server.js +396 -0
  33. package/lib/cli/runtime-api/types.d.ts +149 -0
  34. package/lib/cli/runtime-api/types.js +10 -0
  35. package/lib/cli/runtime-wrapper/nodejs-runtime.d.ts +16 -0
  36. package/lib/cli/runtime-wrapper/nodejs-runtime.js +248 -0
  37. package/lib/cli/watcher/file-watcher.d.ts +32 -0
  38. package/lib/cli/watcher/file-watcher.js +57 -0
  39. package/lib/functions/bridge/appsync-client.d.ts +73 -0
  40. package/lib/functions/bridge/appsync-client.js +345 -0
  41. package/lib/functions/bridge/handler.d.ts +17 -0
  42. package/lib/functions/bridge/handler.js +79 -0
  43. package/lib/functions/bridge/ssm-config.d.ts +19 -0
  44. package/lib/functions/bridge/ssm-config.js +45 -0
  45. package/lib/functions/bridge-builder/handler.d.ts +12 -0
  46. package/lib/functions/bridge-builder/handler.js +181 -0
  47. package/lib/functions/bridge-docker/runtime.d.ts +9 -0
  48. package/lib/functions/bridge-docker/runtime.js +127 -0
  49. package/lib/index.d.ts +24 -0
  50. package/lib/index.js +28 -0
  51. package/lib/shared/types.d.ts +102 -0
  52. package/lib/shared/types.js +125 -0
  53. package/package.json +111 -0
@@ -0,0 +1,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"]}