cdk-local-lambda 0.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +202 -0
- package/README.md +94 -0
- package/lib/aspect/docker-function-hook.d.ts +18 -0
- package/lib/aspect/docker-function-hook.js +31 -0
- package/lib/aspect/live-lambda-aspect.d.ts +85 -0
- package/lib/aspect/live-lambda-aspect.js +277 -0
- package/lib/aspect/live-lambda-bootstrap.d.ts +17 -0
- package/lib/aspect/live-lambda-bootstrap.js +260 -0
- package/lib/aspect/nodejs-function-hook.d.ts +20 -0
- package/lib/aspect/nodejs-function-hook.js +27 -0
- package/lib/bootstrap-stack/bootstrap-stack.d.ts +60 -0
- package/lib/bootstrap-stack/bootstrap-stack.js +338 -0
- package/lib/cli/appsync/client.d.ts +30 -0
- package/lib/cli/appsync/client.js +227 -0
- package/lib/cli/cdk-app.d.ts +7 -0
- package/lib/cli/cdk-app.js +25 -0
- package/lib/cli/commands/bootstrap.d.ts +9 -0
- package/lib/cli/commands/bootstrap.js +50 -0
- package/lib/cli/commands/local.d.ts +40 -0
- package/lib/cli/commands/local.js +1172 -0
- package/lib/cli/daemon.d.ts +22 -0
- package/lib/cli/daemon.js +18 -0
- package/lib/cli/docker/container.d.ts +116 -0
- package/lib/cli/docker/container.js +414 -0
- package/lib/cli/docker/types.d.ts +71 -0
- package/lib/cli/docker/types.js +5 -0
- package/lib/cli/docker/watcher.d.ts +44 -0
- package/lib/cli/docker/watcher.js +115 -0
- package/lib/cli/index.d.ts +9 -0
- package/lib/cli/index.js +26 -0
- package/lib/cli/runtime-api/server.d.ts +102 -0
- package/lib/cli/runtime-api/server.js +396 -0
- package/lib/cli/runtime-api/types.d.ts +149 -0
- package/lib/cli/runtime-api/types.js +10 -0
- package/lib/cli/runtime-wrapper/nodejs-runtime.d.ts +16 -0
- package/lib/cli/runtime-wrapper/nodejs-runtime.js +248 -0
- package/lib/cli/watcher/file-watcher.d.ts +32 -0
- package/lib/cli/watcher/file-watcher.js +57 -0
- package/lib/functions/bridge/appsync-client.d.ts +73 -0
- package/lib/functions/bridge/appsync-client.js +345 -0
- package/lib/functions/bridge/handler.d.ts +17 -0
- package/lib/functions/bridge/handler.js +79 -0
- package/lib/functions/bridge/ssm-config.d.ts +19 -0
- package/lib/functions/bridge/ssm-config.js +45 -0
- package/lib/functions/bridge-builder/handler.d.ts +12 -0
- package/lib/functions/bridge-builder/handler.js +181 -0
- package/lib/functions/bridge-docker/runtime.d.ts +9 -0
- package/lib/functions/bridge-docker/runtime.js +127 -0
- package/lib/index.d.ts +24 -0
- package/lib/index.js +28 -0
- package/lib/shared/types.d.ts +102 -0
- package/lib/shared/types.js +125 -0
- package/package.json +111 -0
|
@@ -0,0 +1,345 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AppSync Events client with AWS SigV4 signing.
|
|
3
|
+
*
|
|
4
|
+
* This client handles:
|
|
5
|
+
* - HTTP requests to publish events (with IAM signing)
|
|
6
|
+
* - WebSocket connections for real-time subscriptions (with IAM signing)
|
|
7
|
+
*/
|
|
8
|
+
import { Sha256 } from "@aws-crypto/sha256-js";
|
|
9
|
+
import { defaultProvider } from "@aws-sdk/credential-provider-node";
|
|
10
|
+
import { HttpRequest } from "@aws-sdk/protocol-http";
|
|
11
|
+
import { SignatureV4 } from "@aws-sdk/signature-v4";
|
|
12
|
+
import WebSocket from "ws";
|
|
13
|
+
/**
|
|
14
|
+
* Client for AppSync Events API
|
|
15
|
+
*/
|
|
16
|
+
export class AppSyncEventsClient {
|
|
17
|
+
httpEndpoint;
|
|
18
|
+
realtimeEndpoint;
|
|
19
|
+
region;
|
|
20
|
+
ws = null;
|
|
21
|
+
constructor(config) {
|
|
22
|
+
this.httpEndpoint = config.httpEndpoint;
|
|
23
|
+
this.realtimeEndpoint = config.realtimeEndpoint;
|
|
24
|
+
// Extract region from endpoint URL
|
|
25
|
+
// Format: https://{api-id}.appsync-api.{region}.amazonaws.com/event
|
|
26
|
+
const match = config.httpEndpoint.match(/\.([a-z0-9-]+)\.amazonaws\.com/);
|
|
27
|
+
this.region =
|
|
28
|
+
config.region ?? match?.[1] ?? process.env.AWS_REGION ?? "us-east-1";
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Publish a message to a channel and wait for a response on another channel.
|
|
32
|
+
*/
|
|
33
|
+
async publishAndWaitForResponse(options) {
|
|
34
|
+
const { publishChannel, subscribeChannel, message, timeoutMs, matchResponse, } = options;
|
|
35
|
+
return new Promise((resolve, reject) => {
|
|
36
|
+
const timeoutId = setTimeout(() => {
|
|
37
|
+
this.close();
|
|
38
|
+
reject(new Error(`Timeout waiting for response after ${timeoutMs}ms`));
|
|
39
|
+
}, timeoutMs);
|
|
40
|
+
// Set up WebSocket connection first
|
|
41
|
+
this.connectAndSubscribe(subscribeChannel, (receivedMessage) => {
|
|
42
|
+
if (matchResponse(receivedMessage)) {
|
|
43
|
+
clearTimeout(timeoutId);
|
|
44
|
+
resolve(receivedMessage);
|
|
45
|
+
return true; // Signal to stop listening
|
|
46
|
+
}
|
|
47
|
+
return false;
|
|
48
|
+
})
|
|
49
|
+
.then(() => {
|
|
50
|
+
// Once subscribed, publish the message
|
|
51
|
+
return this.publish(publishChannel, message);
|
|
52
|
+
})
|
|
53
|
+
.catch((err) => {
|
|
54
|
+
clearTimeout(timeoutId);
|
|
55
|
+
reject(err);
|
|
56
|
+
});
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Publish a message to a channel via HTTP
|
|
61
|
+
*/
|
|
62
|
+
async publish(channel, message) {
|
|
63
|
+
const url = new URL(this.httpEndpoint);
|
|
64
|
+
const body = JSON.stringify({
|
|
65
|
+
channel,
|
|
66
|
+
events: [JSON.stringify(message)],
|
|
67
|
+
});
|
|
68
|
+
// AppSync Events requires /event path for publishing
|
|
69
|
+
const path = url.pathname.endsWith("/event") ? url.pathname : "/event";
|
|
70
|
+
const request = new HttpRequest({
|
|
71
|
+
method: "POST",
|
|
72
|
+
protocol: url.protocol,
|
|
73
|
+
hostname: url.hostname,
|
|
74
|
+
path,
|
|
75
|
+
headers: {
|
|
76
|
+
"Content-Type": "application/json",
|
|
77
|
+
host: url.hostname,
|
|
78
|
+
},
|
|
79
|
+
body,
|
|
80
|
+
});
|
|
81
|
+
// Sign the request
|
|
82
|
+
const signer = new SignatureV4({
|
|
83
|
+
credentials: defaultProvider(),
|
|
84
|
+
region: this.region,
|
|
85
|
+
service: "appsync",
|
|
86
|
+
sha256: Sha256,
|
|
87
|
+
});
|
|
88
|
+
const signedRequest = await signer.sign(request);
|
|
89
|
+
// Make the HTTP request - use the correct endpoint with /event path
|
|
90
|
+
const publishUrl = `${url.protocol}//${url.hostname}${path}`;
|
|
91
|
+
const response = await fetch(publishUrl, {
|
|
92
|
+
method: "POST",
|
|
93
|
+
headers: signedRequest.headers,
|
|
94
|
+
body,
|
|
95
|
+
});
|
|
96
|
+
if (!response.ok) {
|
|
97
|
+
const text = await response.text();
|
|
98
|
+
throw new Error(`Failed to publish event: ${response.status} ${text}`);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Subscribe to a channel and receive messages continuously.
|
|
103
|
+
* Returns an unsubscribe function.
|
|
104
|
+
*/
|
|
105
|
+
async subscribe(options) {
|
|
106
|
+
const { channel, onMessage, onError } = options;
|
|
107
|
+
await this.connectAndSubscribeContinuous(channel, onMessage, onError);
|
|
108
|
+
return () => {
|
|
109
|
+
this.close();
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* Check if WebSocket is connected
|
|
114
|
+
*/
|
|
115
|
+
isConnected() {
|
|
116
|
+
return this.ws !== null && this.ws.readyState === WebSocket.OPEN;
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Connect to WebSocket and subscribe to a channel
|
|
120
|
+
*/
|
|
121
|
+
async connectAndSubscribe(channel, onMessage) {
|
|
122
|
+
// Build signed WebSocket connection
|
|
123
|
+
const [wsUrl, subprotocols] = await this.buildSignedWebSocketConnection();
|
|
124
|
+
return new Promise((resolve, reject) => {
|
|
125
|
+
this.ws = new WebSocket(wsUrl, subprotocols);
|
|
126
|
+
this.ws.on("open", () => {
|
|
127
|
+
console.log("[Bridge] WebSocket connected");
|
|
128
|
+
// Send connection init
|
|
129
|
+
this.ws?.send(JSON.stringify({ type: "connection_init" }));
|
|
130
|
+
});
|
|
131
|
+
this.ws.on("message", (data) => {
|
|
132
|
+
const message = JSON.parse(data.toString());
|
|
133
|
+
if (message.type === "connection_ack") {
|
|
134
|
+
console.log("[Bridge] Connection acknowledged");
|
|
135
|
+
// Subscribe to channel with authorization
|
|
136
|
+
this.createSubscribeAuthorization(channel).then((auth) => {
|
|
137
|
+
this.ws?.send(JSON.stringify({
|
|
138
|
+
type: "subscribe",
|
|
139
|
+
id: "sub-1",
|
|
140
|
+
channel,
|
|
141
|
+
authorization: auth,
|
|
142
|
+
}));
|
|
143
|
+
});
|
|
144
|
+
}
|
|
145
|
+
else if (message.type === "subscribe_success") {
|
|
146
|
+
console.log(`[Bridge] Subscribed to ${channel}`);
|
|
147
|
+
resolve();
|
|
148
|
+
}
|
|
149
|
+
else if (message.type === "data" && message.id === "sub-1") {
|
|
150
|
+
// Parse the event data
|
|
151
|
+
try {
|
|
152
|
+
const eventData = JSON.parse(message.event);
|
|
153
|
+
const shouldStop = onMessage(eventData);
|
|
154
|
+
if (shouldStop) {
|
|
155
|
+
this.close();
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
catch (err) {
|
|
159
|
+
console.error("[Bridge] Failed to parse event data:", err);
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
else if (message.type === "error") {
|
|
163
|
+
console.error("[Bridge] WebSocket error:", message);
|
|
164
|
+
reject(new Error(message.errors
|
|
165
|
+
?.map((e) => e.message)
|
|
166
|
+
.join(", ") ?? "Unknown error"));
|
|
167
|
+
}
|
|
168
|
+
});
|
|
169
|
+
this.ws.on("error", (err) => {
|
|
170
|
+
console.error("[Bridge] WebSocket error:", err);
|
|
171
|
+
reject(err);
|
|
172
|
+
});
|
|
173
|
+
this.ws.on("close", () => {
|
|
174
|
+
console.log("[Bridge] WebSocket closed");
|
|
175
|
+
});
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
/**
|
|
179
|
+
* Connect to WebSocket and subscribe to a channel with continuous message handling.
|
|
180
|
+
* Unlike connectAndSubscribe, this doesn't close after receiving a message.
|
|
181
|
+
*/
|
|
182
|
+
async connectAndSubscribeContinuous(channel, onMessage, onError) {
|
|
183
|
+
// Build signed WebSocket connection
|
|
184
|
+
const [wsUrl, subprotocols] = await this.buildSignedWebSocketConnection();
|
|
185
|
+
return new Promise((resolve, reject) => {
|
|
186
|
+
this.ws = new WebSocket(wsUrl, subprotocols);
|
|
187
|
+
this.ws.on("open", () => {
|
|
188
|
+
console.log("[AppSync] WebSocket connected");
|
|
189
|
+
// Send connection init
|
|
190
|
+
this.ws?.send(JSON.stringify({ type: "connection_init" }));
|
|
191
|
+
});
|
|
192
|
+
this.ws.on("message", async (data) => {
|
|
193
|
+
const message = JSON.parse(data.toString());
|
|
194
|
+
if (message.type === "connection_ack") {
|
|
195
|
+
console.log("[AppSync] Connection acknowledged");
|
|
196
|
+
// Subscribe to channel with authorization
|
|
197
|
+
this.createSubscribeAuthorization(channel).then((auth) => {
|
|
198
|
+
this.ws?.send(JSON.stringify({
|
|
199
|
+
type: "subscribe",
|
|
200
|
+
id: "sub-1",
|
|
201
|
+
channel,
|
|
202
|
+
authorization: auth,
|
|
203
|
+
}));
|
|
204
|
+
});
|
|
205
|
+
}
|
|
206
|
+
else if (message.type === "subscribe_success") {
|
|
207
|
+
console.log(`[AppSync] Subscribed to ${channel}`);
|
|
208
|
+
resolve();
|
|
209
|
+
}
|
|
210
|
+
else if (message.type === "data" && message.id === "sub-1") {
|
|
211
|
+
// Parse the event data
|
|
212
|
+
try {
|
|
213
|
+
const eventData = JSON.parse(message.event);
|
|
214
|
+
await onMessage(eventData);
|
|
215
|
+
}
|
|
216
|
+
catch (err) {
|
|
217
|
+
console.error("[AppSync] Failed to parse event data:", err);
|
|
218
|
+
onError?.(err instanceof Error ? err : new Error(String(err)));
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
else if (message.type === "error") {
|
|
222
|
+
console.error("[AppSync] WebSocket error:", message);
|
|
223
|
+
const error = new Error(message.errors
|
|
224
|
+
?.map((e) => e.message)
|
|
225
|
+
.join(", ") ?? "Unknown error");
|
|
226
|
+
onError?.(error);
|
|
227
|
+
reject(error);
|
|
228
|
+
}
|
|
229
|
+
});
|
|
230
|
+
this.ws.on("error", (err) => {
|
|
231
|
+
console.error("[AppSync] WebSocket error:", err);
|
|
232
|
+
onError?.(err);
|
|
233
|
+
reject(err);
|
|
234
|
+
});
|
|
235
|
+
this.ws.on("close", () => {
|
|
236
|
+
console.log("[AppSync] WebSocket closed");
|
|
237
|
+
});
|
|
238
|
+
});
|
|
239
|
+
}
|
|
240
|
+
/**
|
|
241
|
+
* Create authorization headers for subscribe operation
|
|
242
|
+
*/
|
|
243
|
+
async createSubscribeAuthorization(channel) {
|
|
244
|
+
const httpUrl = new URL(this.httpEndpoint);
|
|
245
|
+
const payload = JSON.stringify({ channel });
|
|
246
|
+
const request = new HttpRequest({
|
|
247
|
+
method: "POST",
|
|
248
|
+
protocol: "https:",
|
|
249
|
+
hostname: httpUrl.hostname,
|
|
250
|
+
path: "/event",
|
|
251
|
+
headers: {
|
|
252
|
+
accept: "application/json, text/javascript",
|
|
253
|
+
"content-encoding": "amz-1.0",
|
|
254
|
+
"content-type": "application/json; charset=UTF-8",
|
|
255
|
+
host: httpUrl.hostname,
|
|
256
|
+
},
|
|
257
|
+
body: payload,
|
|
258
|
+
});
|
|
259
|
+
const signer = new SignatureV4({
|
|
260
|
+
credentials: defaultProvider(),
|
|
261
|
+
region: this.region,
|
|
262
|
+
service: "appsync",
|
|
263
|
+
sha256: Sha256,
|
|
264
|
+
});
|
|
265
|
+
const signedRequest = await signer.sign(request);
|
|
266
|
+
const auth = {
|
|
267
|
+
accept: "application/json, text/javascript",
|
|
268
|
+
"content-encoding": "amz-1.0",
|
|
269
|
+
"content-type": "application/json; charset=UTF-8",
|
|
270
|
+
host: httpUrl.hostname,
|
|
271
|
+
"x-amz-date": signedRequest.headers["x-amz-date"],
|
|
272
|
+
"x-amz-content-sha256": signedRequest.headers["x-amz-content-sha256"],
|
|
273
|
+
Authorization: signedRequest.headers["authorization"],
|
|
274
|
+
};
|
|
275
|
+
if (signedRequest.headers["x-amz-security-token"]) {
|
|
276
|
+
auth["x-amz-security-token"] =
|
|
277
|
+
signedRequest.headers["x-amz-security-token"];
|
|
278
|
+
}
|
|
279
|
+
return auth;
|
|
280
|
+
}
|
|
281
|
+
/**
|
|
282
|
+
* Build signed WebSocket connection info for AppSync Events
|
|
283
|
+
* Returns [url, subprotocols] for WebSocket constructor
|
|
284
|
+
*/
|
|
285
|
+
async buildSignedWebSocketConnection() {
|
|
286
|
+
const realtimeUrl = new URL(this.realtimeEndpoint);
|
|
287
|
+
realtimeUrl.pathname = "/event/realtime";
|
|
288
|
+
const httpUrl = new URL(this.httpEndpoint);
|
|
289
|
+
// For IAM auth, sign a POST request to the HTTP endpoint
|
|
290
|
+
const request = new HttpRequest({
|
|
291
|
+
method: "POST",
|
|
292
|
+
protocol: "https:",
|
|
293
|
+
hostname: httpUrl.hostname,
|
|
294
|
+
path: "/event",
|
|
295
|
+
headers: {
|
|
296
|
+
accept: "application/json, text/javascript",
|
|
297
|
+
"content-encoding": "amz-1.0",
|
|
298
|
+
"content-type": "application/json; charset=UTF-8",
|
|
299
|
+
host: httpUrl.hostname,
|
|
300
|
+
},
|
|
301
|
+
body: "{}",
|
|
302
|
+
});
|
|
303
|
+
const signer = new SignatureV4({
|
|
304
|
+
credentials: defaultProvider(),
|
|
305
|
+
region: this.region,
|
|
306
|
+
service: "appsync",
|
|
307
|
+
sha256: Sha256,
|
|
308
|
+
});
|
|
309
|
+
const signedRequest = await signer.sign(request);
|
|
310
|
+
// Build header payload for subprotocol - include all signed headers
|
|
311
|
+
const headerPayload = {
|
|
312
|
+
accept: "application/json, text/javascript",
|
|
313
|
+
"content-encoding": "amz-1.0",
|
|
314
|
+
"content-type": "application/json; charset=UTF-8",
|
|
315
|
+
host: httpUrl.hostname,
|
|
316
|
+
"x-amz-date": signedRequest.headers["x-amz-date"],
|
|
317
|
+
"x-amz-content-sha256": signedRequest.headers["x-amz-content-sha256"],
|
|
318
|
+
Authorization: signedRequest.headers["authorization"],
|
|
319
|
+
};
|
|
320
|
+
// Add session token if present
|
|
321
|
+
if (signedRequest.headers["x-amz-security-token"]) {
|
|
322
|
+
headerPayload["x-amz-security-token"] =
|
|
323
|
+
signedRequest.headers["x-amz-security-token"];
|
|
324
|
+
}
|
|
325
|
+
// Base64url encode (no padding, URL-safe)
|
|
326
|
+
const encodedHeader = Buffer.from(JSON.stringify(headerPayload))
|
|
327
|
+
.toString("base64")
|
|
328
|
+
.replace(/\+/g, "-")
|
|
329
|
+
.replace(/\//g, "_")
|
|
330
|
+
.replace(/=+$/, "");
|
|
331
|
+
// Auth is passed as header-{base64url} subprotocol
|
|
332
|
+
const authSubprotocol = `header-${encodedHeader}`;
|
|
333
|
+
return [realtimeUrl.toString(), ["aws-appsync-event-ws", authSubprotocol]];
|
|
334
|
+
}
|
|
335
|
+
/**
|
|
336
|
+
* Close the WebSocket connection
|
|
337
|
+
*/
|
|
338
|
+
async close() {
|
|
339
|
+
if (this.ws) {
|
|
340
|
+
this.ws.close();
|
|
341
|
+
this.ws = null;
|
|
342
|
+
}
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"appsync-client.js","sourceRoot":"","sources":["../../../src/functions/bridge/appsync-client.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * AppSync Events client with AWS SigV4 signing.\n *\n * This client handles:\n * - HTTP requests to publish events (with IAM signing)\n * - WebSocket connections for real-time subscriptions (with IAM signing)\n */\n\nimport { Sha256 } from \"@aws-crypto/sha256-js\"\nimport { defaultProvider } from \"@aws-sdk/credential-provider-node\"\nimport { HttpRequest } from \"@aws-sdk/protocol-http\"\nimport { SignatureV4 } from \"@aws-sdk/signature-v4\"\nimport WebSocket from \"ws\"\n\nexport interface AppSyncEventsClientConfig {\n  httpEndpoint: string\n  realtimeEndpoint: string\n  region?: string\n}\n\nexport interface PublishAndWaitOptions<T> {\n  publishChannel: string\n  subscribeChannel: string\n  message: unknown\n  timeoutMs: number\n  matchResponse: (message: T) => boolean\n}\n\nexport interface SubscriptionOptions<T> {\n  channel: string\n  onMessage: (message: T) => void | Promise<void>\n  onError?: (error: Error) => void\n}\n\n/**\n * Client for AppSync Events API\n */\nexport class AppSyncEventsClient {\n  private readonly httpEndpoint: string\n  private readonly realtimeEndpoint: string\n  private readonly region: string\n  private ws: WebSocket | null = null\n\n  constructor(config: AppSyncEventsClientConfig) {\n    this.httpEndpoint = config.httpEndpoint\n    this.realtimeEndpoint = config.realtimeEndpoint\n    // Extract region from endpoint URL\n    // Format: https://{api-id}.appsync-api.{region}.amazonaws.com/event\n    const match = config.httpEndpoint.match(/\\.([a-z0-9-]+)\\.amazonaws\\.com/)\n    this.region =\n      config.region ?? match?.[1] ?? process.env.AWS_REGION ?? \"us-east-1\"\n  }\n\n  /**\n   * Publish a message to a channel and wait for a response on another channel.\n   */\n  async publishAndWaitForResponse<T>(\n    options: PublishAndWaitOptions<T>,\n  ): Promise<T> {\n    const {\n      publishChannel,\n      subscribeChannel,\n      message,\n      timeoutMs,\n      matchResponse,\n    } = options\n\n    return new Promise<T>((resolve, reject) => {\n      const timeoutId = setTimeout(() => {\n        this.close()\n        reject(new Error(`Timeout waiting for response after ${timeoutMs}ms`))\n      }, timeoutMs)\n\n      // Set up WebSocket connection first\n      this.connectAndSubscribe<T>(subscribeChannel, (receivedMessage) => {\n        if (matchResponse(receivedMessage)) {\n          clearTimeout(timeoutId)\n          resolve(receivedMessage)\n          return true // Signal to stop listening\n        }\n        return false\n      })\n        .then(() => {\n          // Once subscribed, publish the message\n          return this.publish(publishChannel, message)\n        })\n        .catch((err) => {\n          clearTimeout(timeoutId)\n          reject(err)\n        })\n    })\n  }\n\n  /**\n   * Publish a message to a channel via HTTP\n   */\n  async publish(channel: string, message: unknown): Promise<void> {\n    const url = new URL(this.httpEndpoint)\n    const body = JSON.stringify({\n      channel,\n      events: [JSON.stringify(message)],\n    })\n\n    // AppSync Events requires /event path for publishing\n    const path = url.pathname.endsWith(\"/event\") ? url.pathname : \"/event\"\n\n    const request = new HttpRequest({\n      method: \"POST\",\n      protocol: url.protocol,\n      hostname: url.hostname,\n      path,\n      headers: {\n        \"Content-Type\": \"application/json\",\n        host: url.hostname,\n      },\n      body,\n    })\n\n    // Sign the request\n    const signer = new SignatureV4({\n      credentials: defaultProvider(),\n      region: this.region,\n      service: \"appsync\",\n      sha256: Sha256,\n    })\n\n    const signedRequest = await signer.sign(request)\n\n    // Make the HTTP request - use the correct endpoint with /event path\n    const publishUrl = `${url.protocol}//${url.hostname}${path}`\n    const response = await fetch(publishUrl, {\n      method: \"POST\",\n      headers: signedRequest.headers as Record<string, string>,\n      body,\n    })\n\n    if (!response.ok) {\n      const text = await response.text()\n      throw new Error(`Failed to publish event: ${response.status} ${text}`)\n    }\n  }\n\n  /**\n   * Subscribe to a channel and receive messages continuously.\n   * Returns an unsubscribe function.\n   */\n  async subscribe<T>(options: SubscriptionOptions<T>): Promise<() => void> {\n    const { channel, onMessage, onError } = options\n\n    await this.connectAndSubscribeContinuous<T>(channel, onMessage, onError)\n\n    return () => {\n      this.close()\n    }\n  }\n\n  /**\n   * Check if WebSocket is connected\n   */\n  isConnected(): boolean {\n    return this.ws !== null && this.ws.readyState === WebSocket.OPEN\n  }\n\n  /**\n   * Connect to WebSocket and subscribe to a channel\n   */\n  private async connectAndSubscribe<T>(\n    channel: string,\n    onMessage: (message: T) => boolean,\n  ): Promise<void> {\n    // Build signed WebSocket connection\n    const [wsUrl, subprotocols] = await this.buildSignedWebSocketConnection()\n\n    return new Promise((resolve, reject) => {\n      this.ws = new WebSocket(wsUrl, subprotocols)\n\n      this.ws.on(\"open\", () => {\n        console.log(\"[Bridge] WebSocket connected\")\n        // Send connection init\n        this.ws?.send(JSON.stringify({ type: \"connection_init\" }))\n      })\n\n      this.ws.on(\"message\", (data) => {\n        const message = JSON.parse(data.toString())\n\n        if (message.type === \"connection_ack\") {\n          console.log(\"[Bridge] Connection acknowledged\")\n          // Subscribe to channel with authorization\n          this.createSubscribeAuthorization(channel).then((auth) => {\n            this.ws?.send(\n              JSON.stringify({\n                type: \"subscribe\",\n                id: \"sub-1\",\n                channel,\n                authorization: auth,\n              }),\n            )\n          })\n        } else if (message.type === \"subscribe_success\") {\n          console.log(`[Bridge] Subscribed to ${channel}`)\n          resolve()\n        } else if (message.type === \"data\" && message.id === \"sub-1\") {\n          // Parse the event data\n          try {\n            const eventData = JSON.parse(message.event) as T\n            const shouldStop = onMessage(eventData)\n            if (shouldStop) {\n              this.close()\n            }\n          } catch (err) {\n            console.error(\"[Bridge] Failed to parse event data:\", err)\n          }\n        } else if (message.type === \"error\") {\n          console.error(\"[Bridge] WebSocket error:\", message)\n          reject(\n            new Error(\n              message.errors\n                ?.map((e: { message: string }) => e.message)\n                .join(\", \") ?? \"Unknown error\",\n            ),\n          )\n        }\n      })\n\n      this.ws.on(\"error\", (err) => {\n        console.error(\"[Bridge] WebSocket error:\", err)\n        reject(err)\n      })\n\n      this.ws.on(\"close\", () => {\n        console.log(\"[Bridge] WebSocket closed\")\n      })\n    })\n  }\n\n  /**\n   * Connect to WebSocket and subscribe to a channel with continuous message handling.\n   * Unlike connectAndSubscribe, this doesn't close after receiving a message.\n   */\n  private async connectAndSubscribeContinuous<T>(\n    channel: string,\n    onMessage: (message: T) => void | Promise<void>,\n    onError?: (error: Error) => void,\n  ): Promise<void> {\n    // Build signed WebSocket connection\n    const [wsUrl, subprotocols] = await this.buildSignedWebSocketConnection()\n\n    return new Promise((resolve, reject) => {\n      this.ws = new WebSocket(wsUrl, subprotocols)\n\n      this.ws.on(\"open\", () => {\n        console.log(\"[AppSync] WebSocket connected\")\n        // Send connection init\n        this.ws?.send(JSON.stringify({ type: \"connection_init\" }))\n      })\n\n      this.ws.on(\"message\", async (data) => {\n        const message = JSON.parse(data.toString())\n\n        if (message.type === \"connection_ack\") {\n          console.log(\"[AppSync] Connection acknowledged\")\n          // Subscribe to channel with authorization\n          this.createSubscribeAuthorization(channel).then((auth) => {\n            this.ws?.send(\n              JSON.stringify({\n                type: \"subscribe\",\n                id: \"sub-1\",\n                channel,\n                authorization: auth,\n              }),\n            )\n          })\n        } else if (message.type === \"subscribe_success\") {\n          console.log(`[AppSync] Subscribed to ${channel}`)\n          resolve()\n        } else if (message.type === \"data\" && message.id === \"sub-1\") {\n          // Parse the event data\n          try {\n            const eventData = JSON.parse(message.event) as T\n            await onMessage(eventData)\n          } catch (err) {\n            console.error(\"[AppSync] Failed to parse event data:\", err)\n            onError?.(err instanceof Error ? err : new Error(String(err)))\n          }\n        } else if (message.type === \"error\") {\n          console.error(\"[AppSync] WebSocket error:\", message)\n          const error = new Error(\n            message.errors\n              ?.map((e: { message: string }) => e.message)\n              .join(\", \") ?? \"Unknown error\",\n          )\n          onError?.(error)\n          reject(error)\n        }\n      })\n\n      this.ws.on(\"error\", (err) => {\n        console.error(\"[AppSync] WebSocket error:\", err)\n        onError?.(err)\n        reject(err)\n      })\n\n      this.ws.on(\"close\", () => {\n        console.log(\"[AppSync] WebSocket closed\")\n      })\n    })\n  }\n\n  /**\n   * Create authorization headers for subscribe operation\n   */\n  private async createSubscribeAuthorization(\n    channel: string,\n  ): Promise<Record<string, string>> {\n    const httpUrl = new URL(this.httpEndpoint)\n    const payload = JSON.stringify({ channel })\n\n    const request = new HttpRequest({\n      method: \"POST\",\n      protocol: \"https:\",\n      hostname: httpUrl.hostname,\n      path: \"/event\",\n      headers: {\n        accept: \"application/json, text/javascript\",\n        \"content-encoding\": \"amz-1.0\",\n        \"content-type\": \"application/json; charset=UTF-8\",\n        host: httpUrl.hostname,\n      },\n      body: payload,\n    })\n\n    const signer = new SignatureV4({\n      credentials: defaultProvider(),\n      region: this.region,\n      service: \"appsync\",\n      sha256: Sha256,\n    })\n\n    const signedRequest = await signer.sign(request)\n\n    const auth: Record<string, string> = {\n      accept: \"application/json, text/javascript\",\n      \"content-encoding\": \"amz-1.0\",\n      \"content-type\": \"application/json; charset=UTF-8\",\n      host: httpUrl.hostname,\n      \"x-amz-date\": signedRequest.headers[\"x-amz-date\"],\n      \"x-amz-content-sha256\": signedRequest.headers[\"x-amz-content-sha256\"],\n      Authorization: signedRequest.headers[\"authorization\"],\n    }\n\n    if (signedRequest.headers[\"x-amz-security-token\"]) {\n      auth[\"x-amz-security-token\"] =\n        signedRequest.headers[\"x-amz-security-token\"]\n    }\n\n    return auth\n  }\n\n  /**\n   * Build signed WebSocket connection info for AppSync Events\n   * Returns [url, subprotocols] for WebSocket constructor\n   */\n  private async buildSignedWebSocketConnection(): Promise<[string, string[]]> {\n    const realtimeUrl = new URL(this.realtimeEndpoint)\n    realtimeUrl.pathname = \"/event/realtime\"\n    const httpUrl = new URL(this.httpEndpoint)\n\n    // For IAM auth, sign a POST request to the HTTP endpoint\n    const request = new HttpRequest({\n      method: \"POST\",\n      protocol: \"https:\",\n      hostname: httpUrl.hostname,\n      path: \"/event\",\n      headers: {\n        accept: \"application/json, text/javascript\",\n        \"content-encoding\": \"amz-1.0\",\n        \"content-type\": \"application/json; charset=UTF-8\",\n        host: httpUrl.hostname,\n      },\n      body: \"{}\",\n    })\n\n    const signer = new SignatureV4({\n      credentials: defaultProvider(),\n      region: this.region,\n      service: \"appsync\",\n      sha256: Sha256,\n    })\n\n    const signedRequest = await signer.sign(request)\n\n    // Build header payload for subprotocol - include all signed headers\n    const headerPayload: Record<string, string> = {\n      accept: \"application/json, text/javascript\",\n      \"content-encoding\": \"amz-1.0\",\n      \"content-type\": \"application/json; charset=UTF-8\",\n      host: httpUrl.hostname,\n      \"x-amz-date\": signedRequest.headers[\"x-amz-date\"],\n      \"x-amz-content-sha256\": signedRequest.headers[\"x-amz-content-sha256\"],\n      Authorization: signedRequest.headers[\"authorization\"],\n    }\n\n    // Add session token if present\n    if (signedRequest.headers[\"x-amz-security-token\"]) {\n      headerPayload[\"x-amz-security-token\"] =\n        signedRequest.headers[\"x-amz-security-token\"]\n    }\n\n    // Base64url encode (no padding, URL-safe)\n    const encodedHeader = Buffer.from(JSON.stringify(headerPayload))\n      .toString(\"base64\")\n      .replace(/\\+/g, \"-\")\n      .replace(/\\//g, \"_\")\n      .replace(/=+$/, \"\")\n\n    // Auth is passed as header-{base64url} subprotocol\n    const authSubprotocol = `header-${encodedHeader}`\n\n    return [realtimeUrl.toString(), [\"aws-appsync-event-ws\", authSubprotocol]]\n  }\n\n  /**\n   * Close the WebSocket connection\n   */\n  async close(): Promise<void> {\n    if (this.ws) {\n      this.ws.close()\n      this.ws = null\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Bridge Lambda handler that forwards invocations to the local daemon via AppSync Events.
|
|
3
|
+
*
|
|
4
|
+
* This handler:
|
|
5
|
+
* 1. Receives Lambda invocations from AWS
|
|
6
|
+
* 2. Publishes the invocation to AppSync Events channel
|
|
7
|
+
* 3. Subscribes to response channel
|
|
8
|
+
* 4. Waits for the daemon to process and respond
|
|
9
|
+
* 5. Returns the response or throws an error
|
|
10
|
+
*
|
|
11
|
+
* AppSync endpoints are read from SSM Parameter Store at runtime.
|
|
12
|
+
*/
|
|
13
|
+
import type { Context } from "aws-lambda";
|
|
14
|
+
/**
|
|
15
|
+
* Main Lambda handler - forwards all invocations to the local daemon
|
|
16
|
+
*/
|
|
17
|
+
export declare function handler(event: unknown, context: Context): Promise<unknown>;
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Bridge Lambda handler that forwards invocations to the local daemon via AppSync Events.
|
|
3
|
+
*
|
|
4
|
+
* This handler:
|
|
5
|
+
* 1. Receives Lambda invocations from AWS
|
|
6
|
+
* 2. Publishes the invocation to AppSync Events channel
|
|
7
|
+
* 3. Subscribes to response channel
|
|
8
|
+
* 4. Waits for the daemon to process and respond
|
|
9
|
+
* 5. Returns the response or throws an error
|
|
10
|
+
*
|
|
11
|
+
* AppSync endpoints are read from SSM Parameter Store at runtime.
|
|
12
|
+
*/
|
|
13
|
+
import { buildChannelName, filterEnvVars, } from "../../shared/types.js";
|
|
14
|
+
import { AppSyncEventsClient } from "./appsync-client.js";
|
|
15
|
+
import { getAppSyncEndpoints } from "./ssm-config.js";
|
|
16
|
+
// Timeout waiting for daemon response (ms)
|
|
17
|
+
const RESPONSE_TIMEOUT_MS = 290_000; // 4:50 to leave buffer for 5 min Lambda timeout
|
|
18
|
+
/**
|
|
19
|
+
* Main Lambda handler - forwards all invocations to the local daemon
|
|
20
|
+
*/
|
|
21
|
+
export async function handler(event, context) {
|
|
22
|
+
const functionName = context.functionName;
|
|
23
|
+
const requestId = context.awsRequestId;
|
|
24
|
+
console.log(`[Bridge] Processing invocation for ${functionName}`);
|
|
25
|
+
console.log(`[Bridge] Request ID: ${requestId}`);
|
|
26
|
+
// Get AppSync endpoints from SSM
|
|
27
|
+
const { httpEndpoint, realtimeEndpoint } = await getAppSyncEndpoints();
|
|
28
|
+
// Create AppSync client
|
|
29
|
+
const client = new AppSyncEventsClient({
|
|
30
|
+
httpEndpoint,
|
|
31
|
+
realtimeEndpoint,
|
|
32
|
+
});
|
|
33
|
+
// Build channel names
|
|
34
|
+
const invocationChannel = buildChannelName.invocation(functionName);
|
|
35
|
+
const responseChannel = buildChannelName.response(functionName);
|
|
36
|
+
try {
|
|
37
|
+
// Create the invocation message with filtered env vars
|
|
38
|
+
const invocationMessage = {
|
|
39
|
+
type: "invocation",
|
|
40
|
+
requestId,
|
|
41
|
+
event,
|
|
42
|
+
context: {
|
|
43
|
+
functionName: context.functionName,
|
|
44
|
+
functionVersion: context.functionVersion,
|
|
45
|
+
invokedFunctionArn: context.invokedFunctionArn,
|
|
46
|
+
memoryLimitInMB: context.memoryLimitInMB,
|
|
47
|
+
awsRequestId: context.awsRequestId,
|
|
48
|
+
logGroupName: context.logGroupName,
|
|
49
|
+
logStreamName: context.logStreamName,
|
|
50
|
+
getRemainingTimeInMillis: context.getRemainingTimeInMillis(),
|
|
51
|
+
},
|
|
52
|
+
env: filterEnvVars(process.env),
|
|
53
|
+
};
|
|
54
|
+
// Subscribe to response channel first, then publish invocation
|
|
55
|
+
const response = await client.publishAndWaitForResponse({
|
|
56
|
+
publishChannel: invocationChannel,
|
|
57
|
+
subscribeChannel: responseChannel,
|
|
58
|
+
message: invocationMessage,
|
|
59
|
+
timeoutMs: RESPONSE_TIMEOUT_MS,
|
|
60
|
+
matchResponse: (msg) => msg.requestId === requestId,
|
|
61
|
+
});
|
|
62
|
+
console.log(`[Bridge] Received response for ${requestId}`);
|
|
63
|
+
// Check for error response
|
|
64
|
+
if (response.error) {
|
|
65
|
+
const error = new Error(response.error.errorMessage);
|
|
66
|
+
error.name = response.error.errorType;
|
|
67
|
+
if (response.error.stackTrace) {
|
|
68
|
+
error.stack = response.error.stackTrace.join("\n");
|
|
69
|
+
}
|
|
70
|
+
throw error;
|
|
71
|
+
}
|
|
72
|
+
return response.result;
|
|
73
|
+
}
|
|
74
|
+
finally {
|
|
75
|
+
// Clean up client connection
|
|
76
|
+
await client.close();
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9mdW5jdGlvbnMvYnJpZGdlL2hhbmRsZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7O0dBV0c7QUFHSCxPQUFPLEVBQ0wsZ0JBQWdCLEVBQ2hCLGFBQWEsR0FHZCxNQUFNLHVCQUF1QixDQUFBO0FBQzlCLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLHFCQUFxQixDQUFBO0FBQ3pELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGlCQUFpQixDQUFBO0FBRXJELDJDQUEyQztBQUMzQyxNQUFNLG1CQUFtQixHQUFHLE9BQU8sQ0FBQSxDQUFDLGdEQUFnRDtBQUVwRjs7R0FFRztBQUNILE1BQU0sQ0FBQyxLQUFLLFVBQVUsT0FBTyxDQUMzQixLQUFjLEVBQ2QsT0FBZ0I7SUFFaEIsTUFBTSxZQUFZLEdBQUcsT0FBTyxDQUFDLFlBQVksQ0FBQTtJQUN6QyxNQUFNLFNBQVMsR0FBRyxPQUFPLENBQUMsWUFBWSxDQUFBO0lBRXRDLE9BQU8sQ0FBQyxHQUFHLENBQUMsc0NBQXNDLFlBQVksRUFBRSxDQUFDLENBQUE7SUFDakUsT0FBTyxDQUFDLEdBQUcsQ0FBQyx3QkFBd0IsU0FBUyxFQUFFLENBQUMsQ0FBQTtJQUVoRCxpQ0FBaUM7SUFDakMsTUFBTSxFQUFFLFlBQVksRUFBRSxnQkFBZ0IsRUFBRSxHQUFHLE1BQU0sbUJBQW1CLEVBQUUsQ0FBQTtJQUV0RSx3QkFBd0I7SUFDeEIsTUFBTSxNQUFNLEdBQUcsSUFBSSxtQkFBbUIsQ0FBQztRQUNyQyxZQUFZO1FBQ1osZ0JBQWdCO0tBQ2pCLENBQUMsQ0FBQTtJQUVGLHNCQUFzQjtJQUN0QixNQUFNLGlCQUFpQixHQUFHLGdCQUFnQixDQUFDLFVBQVUsQ0FBQyxZQUFZLENBQUMsQ0FBQTtJQUNuRSxNQUFNLGVBQWUsR0FBRyxnQkFBZ0IsQ0FBQyxRQUFRLENBQUMsWUFBWSxDQUFDLENBQUE7SUFFL0QsSUFBSSxDQUFDO1FBQ0gsdURBQXVEO1FBQ3ZELE1BQU0saUJBQWlCLEdBQXNCO1lBQzNDLElBQUksRUFBRSxZQUFZO1lBQ2xCLFNBQVM7WUFDVCxLQUFLO1lBQ0wsT0FBTyxFQUFFO2dCQUNQLFlBQVksRUFBRSxPQUFPLENBQUMsWUFBWTtnQkFDbEMsZUFBZSxFQUFFLE9BQU8sQ0FBQyxlQUFlO2dCQUN4QyxrQkFBa0IsRUFBRSxPQUFPLENBQUMsa0JBQWtCO2dCQUM5QyxlQUFlLEVBQUUsT0FBTyxDQUFDLGVBQWU7Z0JBQ3hDLFlBQVksRUFBRSxPQUFPLENBQUMsWUFBWTtnQkFDbEMsWUFBWSxFQUFFLE9BQU8sQ0FBQyxZQUFZO2dCQUNsQyxhQUFhLEVBQUUsT0FBTyxDQUFDLGFBQWE7Z0JBQ3BDLHdCQUF3QixFQUFFLE9BQU8sQ0FBQyx3QkFBd0IsRUFBRTthQUM3RDtZQUNELEdBQUcsRUFBRSxhQUFhLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQztTQUNoQyxDQUFBO1FBRUQsK0RBQStEO1FBQy9ELE1BQU0sUUFBUSxHQUFHLE1BQU0sTUFBTSxDQUFDLHlCQUF5QixDQUFrQjtZQUN2RSxjQUFjLEVBQUUsaUJBQWlCO1lBQ2pDLGdCQUFnQixFQUFFLGVBQWU7WUFDakMsT0FBTyxFQUFFLGlCQUFpQjtZQUMxQixTQUFTLEVBQUUsbUJBQW1CO1lBQzlCLGFBQWEsRUFBRSxDQUFDLEdBQUcsRUFBRSxFQUFFLENBQUMsR0FBRyxDQUFDLFNBQVMsS0FBSyxTQUFTO1NBQ3BELENBQUMsQ0FBQTtRQUVGLE9BQU8sQ0FBQyxHQUFHLENBQUMsa0NBQWtDLFNBQVMsRUFBRSxDQUFDLENBQUE7UUFFMUQsMkJBQTJCO1FBQzNCLElBQUksUUFBUSxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ25CLE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsWUFBWSxDQUFDLENBQUE7WUFDcEQsS0FBSyxDQUFDLElBQUksR0FBRyxRQUFRLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQTtZQUNyQyxJQUFJLFFBQVEsQ0FBQyxLQUFLLENBQUMsVUFBVSxFQUFFLENBQUM7Z0JBQzlCLEtBQUssQ0FBQyxLQUFLLEdBQUcsUUFBUSxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFBO1lBQ3BELENBQUM7WUFDRCxNQUFNLEtBQUssQ0FBQTtRQUNiLENBQUM7UUFFRCxPQUFPLFFBQVEsQ0FBQyxNQUFNLENBQUE7SUFDeEIsQ0FBQztZQUFTLENBQUM7UUFDVCw2QkFBNkI7UUFDN0IsTUFBTSxNQUFNLENBQUMsS0FBSyxFQUFFLENBQUE7SUFDdEIsQ0FBQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEJyaWRnZSBMYW1iZGEgaGFuZGxlciB0aGF0IGZvcndhcmRzIGludm9jYXRpb25zIHRvIHRoZSBsb2NhbCBkYWVtb24gdmlhIEFwcFN5bmMgRXZlbnRzLlxuICpcbiAqIFRoaXMgaGFuZGxlcjpcbiAqIDEuIFJlY2VpdmVzIExhbWJkYSBpbnZvY2F0aW9ucyBmcm9tIEFXU1xuICogMi4gUHVibGlzaGVzIHRoZSBpbnZvY2F0aW9uIHRvIEFwcFN5bmMgRXZlbnRzIGNoYW5uZWxcbiAqIDMuIFN1YnNjcmliZXMgdG8gcmVzcG9uc2UgY2hhbm5lbFxuICogNC4gV2FpdHMgZm9yIHRoZSBkYWVtb24gdG8gcHJvY2VzcyBhbmQgcmVzcG9uZFxuICogNS4gUmV0dXJucyB0aGUgcmVzcG9uc2Ugb3IgdGhyb3dzIGFuIGVycm9yXG4gKlxuICogQXBwU3luYyBlbmRwb2ludHMgYXJlIHJlYWQgZnJvbSBTU00gUGFyYW1ldGVyIFN0b3JlIGF0IHJ1bnRpbWUuXG4gKi9cblxuaW1wb3J0IHR5cGUgeyBDb250ZXh0IH0gZnJvbSBcImF3cy1sYW1iZGFcIlxuaW1wb3J0IHtcbiAgYnVpbGRDaGFubmVsTmFtZSxcbiAgZmlsdGVyRW52VmFycyxcbiAgdHlwZSBJbnZvY2F0aW9uTWVzc2FnZSxcbiAgdHlwZSBSZXNwb25zZU1lc3NhZ2UsXG59IGZyb20gXCIuLi8uLi9zaGFyZWQvdHlwZXMuanNcIlxuaW1wb3J0IHsgQXBwU3luY0V2ZW50c0NsaWVudCB9IGZyb20gXCIuL2FwcHN5bmMtY2xpZW50LmpzXCJcbmltcG9ydCB7IGdldEFwcFN5bmNFbmRwb2ludHMgfSBmcm9tIFwiLi9zc20tY29uZmlnLmpzXCJcblxuLy8gVGltZW91dCB3YWl0aW5nIGZvciBkYWVtb24gcmVzcG9uc2UgKG1zKVxuY29uc3QgUkVTUE9OU0VfVElNRU9VVF9NUyA9IDI5MF8wMDAgLy8gNDo1MCB0byBsZWF2ZSBidWZmZXIgZm9yIDUgbWluIExhbWJkYSB0aW1lb3V0XG5cbi8qKlxuICogTWFpbiBMYW1iZGEgaGFuZGxlciAtIGZvcndhcmRzIGFsbCBpbnZvY2F0aW9ucyB0byB0aGUgbG9jYWwgZGFlbW9uXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBoYW5kbGVyKFxuICBldmVudDogdW5rbm93bixcbiAgY29udGV4dDogQ29udGV4dCxcbik6IFByb21pc2U8dW5rbm93bj4ge1xuICBjb25zdCBmdW5jdGlvbk5hbWUgPSBjb250ZXh0LmZ1bmN0aW9uTmFtZVxuICBjb25zdCByZXF1ZXN0SWQgPSBjb250ZXh0LmF3c1JlcXVlc3RJZFxuXG4gIGNvbnNvbGUubG9nKGBbQnJpZGdlXSBQcm9jZXNzaW5nIGludm9jYXRpb24gZm9yICR7ZnVuY3Rpb25OYW1lfWApXG4gIGNvbnNvbGUubG9nKGBbQnJpZGdlXSBSZXF1ZXN0IElEOiAke3JlcXVlc3RJZH1gKVxuXG4gIC8vIEdldCBBcHBTeW5jIGVuZHBvaW50cyBmcm9tIFNTTVxuICBjb25zdCB7IGh0dHBFbmRwb2ludCwgcmVhbHRpbWVFbmRwb2ludCB9ID0gYXdhaXQgZ2V0QXBwU3luY0VuZHBvaW50cygpXG5cbiAgLy8gQ3JlYXRlIEFwcFN5bmMgY2xpZW50XG4gIGNvbnN0IGNsaWVudCA9IG5ldyBBcHBTeW5jRXZlbnRzQ2xpZW50KHtcbiAgICBodHRwRW5kcG9pbnQsXG4gICAgcmVhbHRpbWVFbmRwb2ludCxcbiAgfSlcblxuICAvLyBCdWlsZCBjaGFubmVsIG5hbWVzXG4gIGNvbnN0IGludm9jYXRpb25DaGFubmVsID0gYnVpbGRDaGFubmVsTmFtZS5pbnZvY2F0aW9uKGZ1bmN0aW9uTmFtZSlcbiAgY29uc3QgcmVzcG9uc2VDaGFubmVsID0gYnVpbGRDaGFubmVsTmFtZS5yZXNwb25zZShmdW5jdGlvbk5hbWUpXG5cbiAgdHJ5IHtcbiAgICAvLyBDcmVhdGUgdGhlIGludm9jYXRpb24gbWVzc2FnZSB3aXRoIGZpbHRlcmVkIGVudiB2YXJzXG4gICAgY29uc3QgaW52b2NhdGlvbk1lc3NhZ2U6IEludm9jYXRpb25NZXNzYWdlID0ge1xuICAgICAgdHlwZTogXCJpbnZvY2F0aW9uXCIsXG4gICAgICByZXF1ZXN0SWQsXG4gICAgICBldmVudCxcbiAgICAgIGNvbnRleHQ6IHtcbiAgICAgICAgZnVuY3Rpb25OYW1lOiBjb250ZXh0LmZ1bmN0aW9uTmFtZSxcbiAgICAgICAgZnVuY3Rpb25WZXJzaW9uOiBjb250ZXh0LmZ1bmN0aW9uVmVyc2lvbixcbiAgICAgICAgaW52b2tlZEZ1bmN0aW9uQXJuOiBjb250ZXh0Lmludm9rZWRGdW5jdGlvbkFybixcbiAgICAgICAgbWVtb3J5TGltaXRJbk1COiBjb250ZXh0Lm1lbW9yeUxpbWl0SW5NQixcbiAgICAgICAgYXdzUmVxdWVzdElkOiBjb250ZXh0LmF3c1JlcXVlc3RJZCxcbiAgICAgICAgbG9nR3JvdXBOYW1lOiBjb250ZXh0LmxvZ0dyb3VwTmFtZSxcbiAgICAgICAgbG9nU3RyZWFtTmFtZTogY29udGV4dC5sb2dTdHJlYW1OYW1lLFxuICAgICAgICBnZXRSZW1haW5pbmdUaW1lSW5NaWxsaXM6IGNvbnRleHQuZ2V0UmVtYWluaW5nVGltZUluTWlsbGlzKCksXG4gICAgICB9LFxuICAgICAgZW52OiBmaWx0ZXJFbnZWYXJzKHByb2Nlc3MuZW52KSxcbiAgICB9XG5cbiAgICAvLyBTdWJzY3JpYmUgdG8gcmVzcG9uc2UgY2hhbm5lbCBmaXJzdCwgdGhlbiBwdWJsaXNoIGludm9jYXRpb25cbiAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IGNsaWVudC5wdWJsaXNoQW5kV2FpdEZvclJlc3BvbnNlPFJlc3BvbnNlTWVzc2FnZT4oe1xuICAgICAgcHVibGlzaENoYW5uZWw6IGludm9jYXRpb25DaGFubmVsLFxuICAgICAgc3Vic2NyaWJlQ2hhbm5lbDogcmVzcG9uc2VDaGFubmVsLFxuICAgICAgbWVzc2FnZTogaW52b2NhdGlvbk1lc3NhZ2UsXG4gICAgICB0aW1lb3V0TXM6IFJFU1BPTlNFX1RJTUVPVVRfTVMsXG4gICAgICBtYXRjaFJlc3BvbnNlOiAobXNnKSA9PiBtc2cucmVxdWVzdElkID09PSByZXF1ZXN0SWQsXG4gICAgfSlcblxuICAgIGNvbnNvbGUubG9nKGBbQnJpZGdlXSBSZWNlaXZlZCByZXNwb25zZSBmb3IgJHtyZXF1ZXN0SWR9YClcblxuICAgIC8vIENoZWNrIGZvciBlcnJvciByZXNwb25zZVxuICAgIGlmIChyZXNwb25zZS5lcnJvcikge1xuICAgICAgY29uc3QgZXJyb3IgPSBuZXcgRXJyb3IocmVzcG9uc2UuZXJyb3IuZXJyb3JNZXNzYWdlKVxuICAgICAgZXJyb3IubmFtZSA9IHJlc3BvbnNlLmVycm9yLmVycm9yVHlwZVxuICAgICAgaWYgKHJlc3BvbnNlLmVycm9yLnN0YWNrVHJhY2UpIHtcbiAgICAgICAgZXJyb3Iuc3RhY2sgPSByZXNwb25zZS5lcnJvci5zdGFja1RyYWNlLmpvaW4oXCJcXG5cIilcbiAgICAgIH1cbiAgICAgIHRocm93IGVycm9yXG4gICAgfVxuXG4gICAgcmV0dXJuIHJlc3BvbnNlLnJlc3VsdFxuICB9IGZpbmFsbHkge1xuICAgIC8vIENsZWFuIHVwIGNsaWVudCBjb25uZWN0aW9uXG4gICAgYXdhaXQgY2xpZW50LmNsb3NlKClcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SSM parameter reader for AppSync endpoints.
|
|
3
|
+
*
|
|
4
|
+
* Reads the AppSync HTTP and realtime endpoints from SSM Parameter Store
|
|
5
|
+
* and caches them to avoid repeated SSM calls during Lambda warm starts.
|
|
6
|
+
*/
|
|
7
|
+
export interface AppSyncEndpoints {
|
|
8
|
+
httpEndpoint: string;
|
|
9
|
+
realtimeEndpoint: string;
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* Get AppSync endpoints from SSM Parameter Store.
|
|
13
|
+
* Results are cached after the first call.
|
|
14
|
+
*/
|
|
15
|
+
export declare function getAppSyncEndpoints(): Promise<AppSyncEndpoints>;
|
|
16
|
+
/**
|
|
17
|
+
* Clear the cached endpoints. Useful for testing.
|
|
18
|
+
*/
|
|
19
|
+
export declare function clearEndpointCache(): void;
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SSM parameter reader for AppSync endpoints.
|
|
3
|
+
*
|
|
4
|
+
* Reads the AppSync HTTP and realtime endpoints from SSM Parameter Store
|
|
5
|
+
* and caches them to avoid repeated SSM calls during Lambda warm starts.
|
|
6
|
+
*/
|
|
7
|
+
import { GetParameterCommand, SSMClient } from "@aws-sdk/client-ssm";
|
|
8
|
+
import { SSM_BASE_PATH } from "../../shared/types.js";
|
|
9
|
+
// Cache endpoints to avoid repeated SSM calls
|
|
10
|
+
let cachedEndpoints = null;
|
|
11
|
+
/**
|
|
12
|
+
* Get AppSync endpoints from SSM Parameter Store.
|
|
13
|
+
* Results are cached after the first call.
|
|
14
|
+
*/
|
|
15
|
+
export async function getAppSyncEndpoints() {
|
|
16
|
+
if (cachedEndpoints)
|
|
17
|
+
return cachedEndpoints;
|
|
18
|
+
const client = new SSMClient({});
|
|
19
|
+
const qualifier = process.env.CDK_BOOTSTRAP_QUALIFIER ?? "hnb659fds";
|
|
20
|
+
const basePath = `${SSM_BASE_PATH}/${qualifier}`;
|
|
21
|
+
const [httpResponse, realtimeResponse] = await Promise.all([
|
|
22
|
+
client.send(new GetParameterCommand({
|
|
23
|
+
Name: `${basePath}/http-endpoint`,
|
|
24
|
+
})),
|
|
25
|
+
client.send(new GetParameterCommand({
|
|
26
|
+
Name: `${basePath}/realtime-endpoint`,
|
|
27
|
+
})),
|
|
28
|
+
]);
|
|
29
|
+
if (!httpResponse.Parameter?.Value || !realtimeResponse.Parameter?.Value) {
|
|
30
|
+
throw new Error(`[LiveLambda] SSM parameters not found at ${basePath}. ` +
|
|
31
|
+
"Ensure the bootstrap stack is deployed.");
|
|
32
|
+
}
|
|
33
|
+
cachedEndpoints = {
|
|
34
|
+
httpEndpoint: httpResponse.Parameter.Value,
|
|
35
|
+
realtimeEndpoint: realtimeResponse.Parameter.Value,
|
|
36
|
+
};
|
|
37
|
+
return cachedEndpoints;
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Clear the cached endpoints. Useful for testing.
|
|
41
|
+
*/
|
|
42
|
+
export function clearEndpointCache() {
|
|
43
|
+
cachedEndpoints = null;
|
|
44
|
+
}
|
|
45
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3NtLWNvbmZpZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9mdW5jdGlvbnMvYnJpZGdlL3NzbS1jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7O0dBS0c7QUFFSCxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsU0FBUyxFQUFFLE1BQU0scUJBQXFCLENBQUE7QUFDcEUsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLHVCQUF1QixDQUFBO0FBT3JELDhDQUE4QztBQUM5QyxJQUFJLGVBQWUsR0FBNEIsSUFBSSxDQUFBO0FBRW5EOzs7R0FHRztBQUNILE1BQU0sQ0FBQyxLQUFLLFVBQVUsbUJBQW1CO0lBQ3ZDLElBQUksZUFBZTtRQUFFLE9BQU8sZUFBZSxDQUFBO0lBRTNDLE1BQU0sTUFBTSxHQUFHLElBQUksU0FBUyxDQUFDLEVBQUUsQ0FBQyxDQUFBO0lBQ2hDLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxHQUFHLENBQUMsdUJBQXVCLElBQUksV0FBVyxDQUFBO0lBQ3BFLE1BQU0sUUFBUSxHQUFHLEdBQUcsYUFBYSxJQUFJLFNBQVMsRUFBRSxDQUFBO0lBRWhELE1BQU0sQ0FBQyxZQUFZLEVBQUUsZ0JBQWdCLENBQUMsR0FBRyxNQUFNLE9BQU8sQ0FBQyxHQUFHLENBQUM7UUFDekQsTUFBTSxDQUFDLElBQUksQ0FDVCxJQUFJLG1CQUFtQixDQUFDO1lBQ3RCLElBQUksRUFBRSxHQUFHLFFBQVEsZ0JBQWdCO1NBQ2xDLENBQUMsQ0FDSDtRQUNELE1BQU0sQ0FBQyxJQUFJLENBQ1QsSUFBSSxtQkFBbUIsQ0FBQztZQUN0QixJQUFJLEVBQUUsR0FBRyxRQUFRLG9CQUFvQjtTQUN0QyxDQUFDLENBQ0g7S0FDRixDQUFDLENBQUE7SUFFRixJQUFJLENBQUMsWUFBWSxDQUFDLFNBQVMsRUFBRSxLQUFLLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxTQUFTLEVBQUUsS0FBSyxFQUFFLENBQUM7UUFDekUsTUFBTSxJQUFJLEtBQUssQ0FDYiw0Q0FBNEMsUUFBUSxJQUFJO1lBQ3RELHlDQUF5QyxDQUM1QyxDQUFBO0lBQ0gsQ0FBQztJQUVELGVBQWUsR0FBRztRQUNoQixZQUFZLEVBQUUsWUFBWSxDQUFDLFNBQVMsQ0FBQyxLQUFLO1FBQzFDLGdCQUFnQixFQUFFLGdCQUFnQixDQUFDLFNBQVMsQ0FBQyxLQUFLO0tBQ25ELENBQUE7SUFFRCxPQUFPLGVBQWUsQ0FBQTtBQUN4QixDQUFDO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLFVBQVUsa0JBQWtCO0lBQ2hDLGVBQWUsR0FBRyxJQUFJLENBQUE7QUFDeEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogU1NNIHBhcmFtZXRlciByZWFkZXIgZm9yIEFwcFN5bmMgZW5kcG9pbnRzLlxuICpcbiAqIFJlYWRzIHRoZSBBcHBTeW5jIEhUVFAgYW5kIHJlYWx0aW1lIGVuZHBvaW50cyBmcm9tIFNTTSBQYXJhbWV0ZXIgU3RvcmVcbiAqIGFuZCBjYWNoZXMgdGhlbSB0byBhdm9pZCByZXBlYXRlZCBTU00gY2FsbHMgZHVyaW5nIExhbWJkYSB3YXJtIHN0YXJ0cy5cbiAqL1xuXG5pbXBvcnQgeyBHZXRQYXJhbWV0ZXJDb21tYW5kLCBTU01DbGllbnQgfSBmcm9tIFwiQGF3cy1zZGsvY2xpZW50LXNzbVwiXG5pbXBvcnQgeyBTU01fQkFTRV9QQVRIIH0gZnJvbSBcIi4uLy4uL3NoYXJlZC90eXBlcy5qc1wiXG5cbmV4cG9ydCBpbnRlcmZhY2UgQXBwU3luY0VuZHBvaW50cyB7XG4gIGh0dHBFbmRwb2ludDogc3RyaW5nXG4gIHJlYWx0aW1lRW5kcG9pbnQ6IHN0cmluZ1xufVxuXG4vLyBDYWNoZSBlbmRwb2ludHMgdG8gYXZvaWQgcmVwZWF0ZWQgU1NNIGNhbGxzXG5sZXQgY2FjaGVkRW5kcG9pbnRzOiBBcHBTeW5jRW5kcG9pbnRzIHwgbnVsbCA9IG51bGxcblxuLyoqXG4gKiBHZXQgQXBwU3luYyBlbmRwb2ludHMgZnJvbSBTU00gUGFyYW1ldGVyIFN0b3JlLlxuICogUmVzdWx0cyBhcmUgY2FjaGVkIGFmdGVyIHRoZSBmaXJzdCBjYWxsLlxuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gZ2V0QXBwU3luY0VuZHBvaW50cygpOiBQcm9taXNlPEFwcFN5bmNFbmRwb2ludHM+IHtcbiAgaWYgKGNhY2hlZEVuZHBvaW50cykgcmV0dXJuIGNhY2hlZEVuZHBvaW50c1xuXG4gIGNvbnN0IGNsaWVudCA9IG5ldyBTU01DbGllbnQoe30pXG4gIGNvbnN0IHF1YWxpZmllciA9IHByb2Nlc3MuZW52LkNES19CT09UU1RSQVBfUVVBTElGSUVSID8/IFwiaG5iNjU5ZmRzXCJcbiAgY29uc3QgYmFzZVBhdGggPSBgJHtTU01fQkFTRV9QQVRIfS8ke3F1YWxpZmllcn1gXG5cbiAgY29uc3QgW2h0dHBSZXNwb25zZSwgcmVhbHRpbWVSZXNwb25zZV0gPSBhd2FpdCBQcm9taXNlLmFsbChbXG4gICAgY2xpZW50LnNlbmQoXG4gICAgICBuZXcgR2V0UGFyYW1ldGVyQ29tbWFuZCh7XG4gICAgICAgIE5hbWU6IGAke2Jhc2VQYXRofS9odHRwLWVuZHBvaW50YCxcbiAgICAgIH0pLFxuICAgICksXG4gICAgY2xpZW50LnNlbmQoXG4gICAgICBuZXcgR2V0UGFyYW1ldGVyQ29tbWFuZCh7XG4gICAgICAgIE5hbWU6IGAke2Jhc2VQYXRofS9yZWFsdGltZS1lbmRwb2ludGAsXG4gICAgICB9KSxcbiAgICApLFxuICBdKVxuXG4gIGlmICghaHR0cFJlc3BvbnNlLlBhcmFtZXRlcj8uVmFsdWUgfHwgIXJlYWx0aW1lUmVzcG9uc2UuUGFyYW1ldGVyPy5WYWx1ZSkge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgIGBbTGl2ZUxhbWJkYV0gU1NNIHBhcmFtZXRlcnMgbm90IGZvdW5kIGF0ICR7YmFzZVBhdGh9LiBgICtcbiAgICAgICAgXCJFbnN1cmUgdGhlIGJvb3RzdHJhcCBzdGFjayBpcyBkZXBsb3llZC5cIixcbiAgICApXG4gIH1cblxuICBjYWNoZWRFbmRwb2ludHMgPSB7XG4gICAgaHR0cEVuZHBvaW50OiBodHRwUmVzcG9uc2UuUGFyYW1ldGVyLlZhbHVlLFxuICAgIHJlYWx0aW1lRW5kcG9pbnQ6IHJlYWx0aW1lUmVzcG9uc2UuUGFyYW1ldGVyLlZhbHVlLFxuICB9XG5cbiAgcmV0dXJuIGNhY2hlZEVuZHBvaW50c1xufVxuXG4vKipcbiAqIENsZWFyIHRoZSBjYWNoZWQgZW5kcG9pbnRzLiBVc2VmdWwgZm9yIHRlc3RpbmcuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjbGVhckVuZHBvaW50Q2FjaGUoKTogdm9pZCB7XG4gIGNhY2hlZEVuZHBvaW50cyA9IG51bGxcbn1cbiJdfQ==
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Custom resource handler that builds and uploads the bridge Lambda code.
|
|
3
|
+
*
|
|
4
|
+
* This handler:
|
|
5
|
+
* 1. Receives the bridge source code (base64 encoded) and AppSync endpoints
|
|
6
|
+
* 2. Replaces endpoint placeholders with actual values
|
|
7
|
+
* 3. Creates a zip file with the code
|
|
8
|
+
* 4. Uploads the zip to S3
|
|
9
|
+
* 5. Returns the S3 location for Lambda functions to use
|
|
10
|
+
*/
|
|
11
|
+
import type { CloudFormationCustomResourceEvent, CloudFormationCustomResourceResponse } from "aws-lambda";
|
|
12
|
+
export declare function handler(event: CloudFormationCustomResourceEvent): Promise<CloudFormationCustomResourceResponse>;
|