@venturekit/runtime 0.0.0-dev.20260307234057
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 +191 -0
- package/dist/context.d.ts +86 -0
- package/dist/context.js +76 -0
- package/dist/errors.d.ts +80 -0
- package/dist/errors.js +134 -0
- package/dist/handler.d.ts +71 -0
- package/dist/handler.js +176 -0
- package/dist/index.d.ts +18 -0
- package/dist/index.js +53 -0
- package/dist/logger.d.ts +72 -0
- package/dist/logger.js +105 -0
- package/dist/middleware.d.ts +46 -0
- package/dist/middleware.js +147 -0
- package/dist/response.d.ts +75 -0
- package/dist/response.js +107 -0
- package/dist/ws.d.ts +138 -0
- package/dist/ws.js +277 -0
- package/package.json +61 -0
package/dist/ws.js
ADDED
|
@@ -0,0 +1,277 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* WebSocket Connection Store
|
|
4
|
+
*
|
|
5
|
+
* DynamoDB-backed connection management for API Gateway WebSocket APIs.
|
|
6
|
+
*
|
|
7
|
+
* Authentication:
|
|
8
|
+
* Two-phase auth — the client connects without credentials, then sends
|
|
9
|
+
* an { action: "auth", token: "<jwt>" } message over the encrypted channel.
|
|
10
|
+
* The connection is stored as unauthenticated on $connect and upgraded
|
|
11
|
+
* to authenticated via connectionStore.authenticate().
|
|
12
|
+
* This avoids leaking JWTs in query strings, server logs, and access logs.
|
|
13
|
+
*
|
|
14
|
+
* Multi-tenancy:
|
|
15
|
+
* If enabled, each connection stores a tenantId alongside userId.
|
|
16
|
+
* Use sendToTenant() to broadcast within a tenant boundary.
|
|
17
|
+
*
|
|
18
|
+
* API:
|
|
19
|
+
* connectionStore.save(connectionId) — $connect (unauthenticated)
|
|
20
|
+
* connectionStore.authenticate(connectionId, metadata) — after JWT verification
|
|
21
|
+
* connectionStore.remove(connectionId) — $disconnect
|
|
22
|
+
* connectionStore.get(connectionId) — single connection
|
|
23
|
+
* connectionStore.getAll() — all connections
|
|
24
|
+
* connectionStore.getByUser(userId) — by user (GSI)
|
|
25
|
+
* connectionStore.getByTenant(tenantId) — by tenant (GSI)
|
|
26
|
+
* connectionStore.postToConnection(domain, stage, id, data) — send to one
|
|
27
|
+
* connectionStore.sendToUser(domain, stage, userId, data) — send to all user sessions
|
|
28
|
+
* connectionStore.sendToTenant(domain, stage, tenantId, data) — broadcast within tenant
|
|
29
|
+
* connectionStore.broadcast(domain, stage, data) — broadcast to all
|
|
30
|
+
*
|
|
31
|
+
* Environment variables:
|
|
32
|
+
* CONNECTIONS_TABLE — DynamoDB table name (required)
|
|
33
|
+
*
|
|
34
|
+
* DynamoDB table schema:
|
|
35
|
+
* Partition key: connectionId (String)
|
|
36
|
+
* TTL attribute: ttl
|
|
37
|
+
* GSI: userId-index (partition key: userId)
|
|
38
|
+
* GSI: tenantId-index (partition key: tenantId) — only if multi-tenancy enabled
|
|
39
|
+
*/
|
|
40
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
41
|
+
exports.connectionStore = void 0;
|
|
42
|
+
const client_dynamodb_1 = require("@aws-sdk/client-dynamodb");
|
|
43
|
+
const lib_dynamodb_1 = require("@aws-sdk/lib-dynamodb");
|
|
44
|
+
const client_apigatewaymanagementapi_1 = require("@aws-sdk/client-apigatewaymanagementapi");
|
|
45
|
+
const DEFAULT_TTL_SECONDS = 7200; // 2 hours
|
|
46
|
+
const AUTH_GRACE_PERIOD_SECONDS = 30; // unauthenticated connections expire quickly
|
|
47
|
+
function getTableName() {
|
|
48
|
+
const table = process.env.CONNECTIONS_TABLE;
|
|
49
|
+
if (!table) {
|
|
50
|
+
throw new Error('CONNECTIONS_TABLE environment variable is not set');
|
|
51
|
+
}
|
|
52
|
+
return table;
|
|
53
|
+
}
|
|
54
|
+
let _ddb = null;
|
|
55
|
+
function ddb() {
|
|
56
|
+
if (!_ddb) {
|
|
57
|
+
_ddb = lib_dynamodb_1.DynamoDBDocumentClient.from(new client_dynamodb_1.DynamoDBClient({}));
|
|
58
|
+
}
|
|
59
|
+
return _ddb;
|
|
60
|
+
}
|
|
61
|
+
function encode(data) {
|
|
62
|
+
return new TextEncoder().encode(JSON.stringify(data));
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Deliver data to a list of connections in parallel.
|
|
66
|
+
* Removes stale connections automatically.
|
|
67
|
+
* Returns count of successful deliveries.
|
|
68
|
+
*/
|
|
69
|
+
async function deliverToMany(store, domainName, stage, connections, data) {
|
|
70
|
+
let delivered = 0;
|
|
71
|
+
const results = await Promise.allSettled(connections
|
|
72
|
+
.filter((c) => c.authenticated)
|
|
73
|
+
.map((c) => store.postToConnection(domainName, stage, c.connectionId, data)));
|
|
74
|
+
for (const result of results) {
|
|
75
|
+
if (result.status === 'fulfilled' && result.value)
|
|
76
|
+
delivered++;
|
|
77
|
+
}
|
|
78
|
+
return delivered;
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* WebSocket connection store backed by DynamoDB.
|
|
82
|
+
*
|
|
83
|
+
* Two-phase authentication flow:
|
|
84
|
+
*
|
|
85
|
+
* 1. $connect handler:
|
|
86
|
+
* ```typescript
|
|
87
|
+
* await connectionStore.save(connectionId);
|
|
88
|
+
* // Connection is unauthenticated, TTL = 30s
|
|
89
|
+
* ```
|
|
90
|
+
*
|
|
91
|
+
* 2. Client sends auth message over the encrypted WebSocket:
|
|
92
|
+
* ```json
|
|
93
|
+
* { "action": "auth", "token": "<jwt>" }
|
|
94
|
+
* ```
|
|
95
|
+
*
|
|
96
|
+
* 3. $default handler verifies JWT, then:
|
|
97
|
+
* ```typescript
|
|
98
|
+
* await connectionStore.authenticate(connectionId, { userId, tenantId, email });
|
|
99
|
+
* // Connection is now authenticated, TTL extended to 2h
|
|
100
|
+
* ```
|
|
101
|
+
*/
|
|
102
|
+
exports.connectionStore = {
|
|
103
|
+
/**
|
|
104
|
+
* Save a new connection on $connect.
|
|
105
|
+
* The connection is unauthenticated with a short TTL.
|
|
106
|
+
* The client must send an auth message to upgrade.
|
|
107
|
+
*/
|
|
108
|
+
async save(connectionId) {
|
|
109
|
+
const now = Date.now();
|
|
110
|
+
await ddb().send(new lib_dynamodb_1.PutCommand({
|
|
111
|
+
TableName: getTableName(),
|
|
112
|
+
Item: {
|
|
113
|
+
connectionId,
|
|
114
|
+
authenticated: false,
|
|
115
|
+
connectedAt: now,
|
|
116
|
+
ttl: Math.floor(now / 1000) + AUTH_GRACE_PERIOD_SECONDS,
|
|
117
|
+
},
|
|
118
|
+
}));
|
|
119
|
+
},
|
|
120
|
+
/**
|
|
121
|
+
* Authenticate a connection after JWT verification.
|
|
122
|
+
* Updates the record with user metadata and extends the TTL.
|
|
123
|
+
*
|
|
124
|
+
* @param connectionId — the connection to authenticate
|
|
125
|
+
* @param metadata — must include userId, optionally tenantId, email, etc.
|
|
126
|
+
*/
|
|
127
|
+
async authenticate(connectionId, metadata) {
|
|
128
|
+
const now = Date.now();
|
|
129
|
+
const { userId, tenantId, email, ...extra } = metadata;
|
|
130
|
+
const updateParts = [
|
|
131
|
+
'authenticated = :auth',
|
|
132
|
+
'userId = :uid',
|
|
133
|
+
'authenticatedAt = :authAt',
|
|
134
|
+
'ttl = :ttl',
|
|
135
|
+
];
|
|
136
|
+
const values = {
|
|
137
|
+
':auth': true,
|
|
138
|
+
':uid': userId,
|
|
139
|
+
':authAt': now,
|
|
140
|
+
':ttl': Math.floor(now / 1000) + DEFAULT_TTL_SECONDS,
|
|
141
|
+
};
|
|
142
|
+
if (tenantId) {
|
|
143
|
+
updateParts.push('tenantId = :tid');
|
|
144
|
+
values[':tid'] = tenantId;
|
|
145
|
+
}
|
|
146
|
+
if (email) {
|
|
147
|
+
updateParts.push('email = :email');
|
|
148
|
+
values[':email'] = email;
|
|
149
|
+
}
|
|
150
|
+
// Store any extra metadata fields
|
|
151
|
+
let extraIdx = 0;
|
|
152
|
+
for (const [key, val] of Object.entries(extra)) {
|
|
153
|
+
const placeholder = `:extra${extraIdx}`;
|
|
154
|
+
updateParts.push(`#extra${extraIdx} = ${placeholder}`);
|
|
155
|
+
values[placeholder] = val;
|
|
156
|
+
extraIdx++;
|
|
157
|
+
}
|
|
158
|
+
const names = {};
|
|
159
|
+
for (let i = 0; i < extraIdx; i++) {
|
|
160
|
+
names[`#extra${i}`] = Object.keys(extra)[i];
|
|
161
|
+
}
|
|
162
|
+
await ddb().send(new lib_dynamodb_1.UpdateCommand({
|
|
163
|
+
TableName: getTableName(),
|
|
164
|
+
Key: { connectionId },
|
|
165
|
+
UpdateExpression: `SET ${updateParts.join(', ')}`,
|
|
166
|
+
ExpressionAttributeValues: values,
|
|
167
|
+
...(extraIdx > 0 ? { ExpressionAttributeNames: names } : {}),
|
|
168
|
+
}));
|
|
169
|
+
},
|
|
170
|
+
/**
|
|
171
|
+
* Remove a connection on $disconnect.
|
|
172
|
+
*/
|
|
173
|
+
async remove(connectionId) {
|
|
174
|
+
await ddb().send(new lib_dynamodb_1.DeleteCommand({
|
|
175
|
+
TableName: getTableName(),
|
|
176
|
+
Key: { connectionId },
|
|
177
|
+
}));
|
|
178
|
+
},
|
|
179
|
+
/**
|
|
180
|
+
* Get a single connection record.
|
|
181
|
+
*/
|
|
182
|
+
async get(connectionId) {
|
|
183
|
+
const result = await ddb().send(new lib_dynamodb_1.GetCommand({
|
|
184
|
+
TableName: getTableName(),
|
|
185
|
+
Key: { connectionId },
|
|
186
|
+
}));
|
|
187
|
+
return result.Item ?? null;
|
|
188
|
+
},
|
|
189
|
+
/**
|
|
190
|
+
* Get all authenticated connections.
|
|
191
|
+
* For large-scale apps, prefer getByUser() or getByTenant().
|
|
192
|
+
*/
|
|
193
|
+
async getAll() {
|
|
194
|
+
const result = await ddb().send(new lib_dynamodb_1.ScanCommand({
|
|
195
|
+
TableName: getTableName(),
|
|
196
|
+
FilterExpression: 'authenticated = :auth',
|
|
197
|
+
ExpressionAttributeValues: { ':auth': true },
|
|
198
|
+
}));
|
|
199
|
+
return (result.Items ?? []);
|
|
200
|
+
},
|
|
201
|
+
/**
|
|
202
|
+
* Get all connections for a specific user.
|
|
203
|
+
* A user can have multiple active connections (e.g. multiple tabs/devices).
|
|
204
|
+
* Requires GSI: userId-index (partition key: userId).
|
|
205
|
+
*/
|
|
206
|
+
async getByUser(userId) {
|
|
207
|
+
const result = await ddb().send(new lib_dynamodb_1.QueryCommand({
|
|
208
|
+
TableName: getTableName(),
|
|
209
|
+
IndexName: 'userId-index',
|
|
210
|
+
KeyConditionExpression: 'userId = :uid',
|
|
211
|
+
ExpressionAttributeValues: { ':uid': userId },
|
|
212
|
+
}));
|
|
213
|
+
return (result.Items ?? []);
|
|
214
|
+
},
|
|
215
|
+
/**
|
|
216
|
+
* Get all connections for a specific tenant.
|
|
217
|
+
* Requires GSI: tenantId-index (partition key: tenantId).
|
|
218
|
+
*/
|
|
219
|
+
async getByTenant(tenantId) {
|
|
220
|
+
const result = await ddb().send(new lib_dynamodb_1.QueryCommand({
|
|
221
|
+
TableName: getTableName(),
|
|
222
|
+
IndexName: 'tenantId-index',
|
|
223
|
+
KeyConditionExpression: 'tenantId = :tid',
|
|
224
|
+
ExpressionAttributeValues: { ':tid': tenantId },
|
|
225
|
+
}));
|
|
226
|
+
return (result.Items ?? []);
|
|
227
|
+
},
|
|
228
|
+
/**
|
|
229
|
+
* Send data to a specific connection via API Gateway Management API.
|
|
230
|
+
* Automatically cleans up stale connections (GoneException).
|
|
231
|
+
* Returns true if sent, false if the connection was stale and removed.
|
|
232
|
+
*/
|
|
233
|
+
async postToConnection(domainName, stage, connectionId, data) {
|
|
234
|
+
const apigw = new client_apigatewaymanagementapi_1.ApiGatewayManagementApiClient({
|
|
235
|
+
endpoint: `https://${domainName}/${stage}`,
|
|
236
|
+
});
|
|
237
|
+
try {
|
|
238
|
+
await apigw.send(new client_apigatewaymanagementapi_1.PostToConnectionCommand({
|
|
239
|
+
ConnectionId: connectionId,
|
|
240
|
+
Data: encode(data),
|
|
241
|
+
}));
|
|
242
|
+
return true;
|
|
243
|
+
}
|
|
244
|
+
catch (err) {
|
|
245
|
+
if (err instanceof client_apigatewaymanagementapi_1.GoneException) {
|
|
246
|
+
await this.remove(connectionId);
|
|
247
|
+
return false;
|
|
248
|
+
}
|
|
249
|
+
throw err;
|
|
250
|
+
}
|
|
251
|
+
},
|
|
252
|
+
/**
|
|
253
|
+
* Send data to all connections belonging to a specific user.
|
|
254
|
+
* Returns the number of connections that received the message.
|
|
255
|
+
*/
|
|
256
|
+
async sendToUser(domainName, stage, userId, data) {
|
|
257
|
+
const connections = await this.getByUser(userId);
|
|
258
|
+
return deliverToMany(this, domainName, stage, connections, data);
|
|
259
|
+
},
|
|
260
|
+
/**
|
|
261
|
+
* Send data to all connections within a specific tenant.
|
|
262
|
+
* Returns the number of connections that received the message.
|
|
263
|
+
*/
|
|
264
|
+
async sendToTenant(domainName, stage, tenantId, data) {
|
|
265
|
+
const connections = await this.getByTenant(tenantId);
|
|
266
|
+
return deliverToMany(this, domainName, stage, connections, data);
|
|
267
|
+
},
|
|
268
|
+
/**
|
|
269
|
+
* Broadcast data to all authenticated connections.
|
|
270
|
+
* Returns the number of clients that received the message.
|
|
271
|
+
*/
|
|
272
|
+
async broadcast(domainName, stage, data) {
|
|
273
|
+
const connections = await this.getAll();
|
|
274
|
+
return deliverToMany(this, domainName, stage, connections, data);
|
|
275
|
+
},
|
|
276
|
+
};
|
|
277
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ws.js","sourceRoot":"","sources":["../src/ws.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqCG;;;AAEH,8DAA0D;AAC1D,wDAQ+B;AAC/B,4FAIiD;AAqBjD,MAAM,mBAAmB,GAAG,IAAI,CAAC,CAAC,UAAU;AAC5C,MAAM,yBAAyB,GAAG,EAAE,CAAC,CAAC,6CAA6C;AAEnF,SAAS,YAAY;IACnB,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC;IAC5C,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,MAAM,IAAI,KAAK,CAAC,mDAAmD,CAAC,CAAC;IACvE,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,IAAI,IAAI,GAAkC,IAAI,CAAC;AAE/C,SAAS,GAAG;IACV,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,IAAI,GAAG,qCAAsB,CAAC,IAAI,CAAC,IAAI,gCAAc,CAAC,EAAE,CAAC,CAAC,CAAC;IAC7D,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED,SAAS,MAAM,CAAC,IAAa;IAC3B,OAAO,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;AACxD,CAAC;AAED;;;;GAIG;AACH,KAAK,UAAU,aAAa,CAC1B,KAA6B,EAC7B,UAAkB,EAClB,KAAa,EACb,WAA+B,EAC/B,IAAa;IAEb,IAAI,SAAS,GAAG,CAAC,CAAC;IAClB,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,UAAU,CACtC,WAAW;SACR,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,aAAa,CAAC;SAC9B,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,gBAAgB,CAAC,UAAU,EAAE,KAAK,EAAE,CAAC,CAAC,YAAY,EAAE,IAAI,CAAC,CAAC,CAC/E,CAAC;IACF,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;QAC7B,IAAI,MAAM,CAAC,MAAM,KAAK,WAAW,IAAI,MAAM,CAAC,KAAK;YAAE,SAAS,EAAE,CAAC;IACjE,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;GAqBG;AACU,QAAA,eAAe,GAAG;IAC7B;;;;OAIG;IACH,KAAK,CAAC,IAAI,CAAC,YAAoB;QAC7B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,MAAM,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,yBAAU,CAAC;YAC9B,SAAS,EAAE,YAAY,EAAE;YACzB,IAAI,EAAE;gBACJ,YAAY;gBACZ,aAAa,EAAE,KAAK;gBACpB,WAAW,EAAE,GAAG;gBAChB,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,IAAI,CAAC,GAAG,yBAAyB;aACxD;SACF,CAAC,CAAC,CAAC;IACN,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,YAAY,CAAC,YAAoB,EAAE,QAA4B;QACnE,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,GAAG,KAAK,EAAE,GAAG,QAAQ,CAAC;QAEvD,MAAM,WAAW,GAAG;YAClB,uBAAuB;YACvB,eAAe;YACf,2BAA2B;YAC3B,YAAY;SACb,CAAC;QACF,MAAM,MAAM,GAA4B;YACtC,OAAO,EAAE,IAAI;YACb,MAAM,EAAE,MAAM;YACd,SAAS,EAAE,GAAG;YACd,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,IAAI,CAAC,GAAG,mBAAmB;SACrD,CAAC;QAEF,IAAI,QAAQ,EAAE,CAAC;YACb,WAAW,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YACpC,MAAM,CAAC,MAAM,CAAC,GAAG,QAAQ,CAAC;QAC5B,CAAC;QAED,IAAI,KAAK,EAAE,CAAC;YACV,WAAW,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YACnC,MAAM,CAAC,QAAQ,CAAC,GAAG,KAAK,CAAC;QAC3B,CAAC;QAED,kCAAkC;QAClC,IAAI,QAAQ,GAAG,CAAC,CAAC;QACjB,KAAK,MAAM,CAAC,GAAG,EAAE,GAAG,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;YAC/C,MAAM,WAAW,GAAG,SAAS,QAAQ,EAAE,CAAC;YACxC,WAAW,CAAC,IAAI,CAAC,SAAS,QAAQ,MAAM,WAAW,EAAE,CAAC,CAAC;YACvD,MAAM,CAAC,WAAW,CAAC,GAAG,GAAG,CAAC;YAC1B,QAAQ,EAAE,CAAC;QACb,CAAC;QAED,MAAM,KAAK,GAA2B,EAAE,CAAC;QACzC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,QAAQ,EAAE,CAAC,EAAE,EAAE,CAAC;YAClC,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QAC9C,CAAC;QAED,MAAM,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,4BAAa,CAAC;YACjC,SAAS,EAAE,YAAY,EAAE;YACzB,GAAG,EAAE,EAAE,YAAY,EAAE;YACrB,gBAAgB,EAAE,OAAO,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;YACjD,yBAAyB,EAAE,MAAM;YACjC,GAAG,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,wBAAwB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;SAC7D,CAAC,CAAC,CAAC;IACN,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,YAAoB;QAC/B,MAAM,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,4BAAa,CAAC;YACjC,SAAS,EAAE,YAAY,EAAE;YACzB,GAAG,EAAE,EAAE,YAAY,EAAE;SACtB,CAAC,CAAC,CAAC;IACN,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,GAAG,CAAC,YAAoB;QAC5B,MAAM,MAAM,GAAG,MAAM,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,yBAAU,CAAC;YAC7C,SAAS,EAAE,YAAY,EAAE;YACzB,GAAG,EAAE,EAAE,YAAY,EAAE;SACtB,CAAC,CAAC,CAAC;QACJ,OAAQ,MAAM,CAAC,IAAyB,IAAI,IAAI,CAAC;IACnD,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,MAAM;QACV,MAAM,MAAM,GAAG,MAAM,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,0BAAW,CAAC;YAC9C,SAAS,EAAE,YAAY,EAAE;YACzB,gBAAgB,EAAE,uBAAuB;YACzC,yBAAyB,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE;SAC7C,CAAC,CAAC,CAAC;QACJ,OAAO,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE,CAAuB,CAAC;IACpD,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,SAAS,CAAC,MAAc;QAC5B,MAAM,MAAM,GAAG,MAAM,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,2BAAY,CAAC;YAC/C,SAAS,EAAE,YAAY,EAAE;YACzB,SAAS,EAAE,cAAc;YACzB,sBAAsB,EAAE,eAAe;YACvC,yBAAyB,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE;SAC9C,CAAC,CAAC,CAAC;QACJ,OAAO,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE,CAAuB,CAAC;IACpD,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,WAAW,CAAC,QAAgB;QAChC,MAAM,MAAM,GAAG,MAAM,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,2BAAY,CAAC;YAC/C,SAAS,EAAE,YAAY,EAAE;YACzB,SAAS,EAAE,gBAAgB;YAC3B,sBAAsB,EAAE,iBAAiB;YACzC,yBAAyB,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE;SAChD,CAAC,CAAC,CAAC;QACJ,OAAO,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE,CAAuB,CAAC;IACpD,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,gBAAgB,CACpB,UAAkB,EAClB,KAAa,EACb,YAAoB,EACpB,IAAa;QAEb,MAAM,KAAK,GAAG,IAAI,8DAA6B,CAAC;YAC9C,QAAQ,EAAE,WAAW,UAAU,IAAI,KAAK,EAAE;SAC3C,CAAC,CAAC;QAEH,IAAI,CAAC;YACH,MAAM,KAAK,CAAC,IAAI,CAAC,IAAI,wDAAuB,CAAC;gBAC3C,YAAY,EAAE,YAAY;gBAC1B,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC;aACnB,CAAC,CAAC,CAAC;YACJ,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,GAAG,YAAY,8CAAa,EAAE,CAAC;gBACjC,MAAM,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;gBAChC,OAAO,KAAK,CAAC;YACf,CAAC;YACD,MAAM,GAAG,CAAC;QACZ,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,UAAU,CACd,UAAkB,EAClB,KAAa,EACb,MAAc,EACd,IAAa;QAEb,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;QACjD,OAAO,aAAa,CAAC,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE,WAAW,EAAE,IAAI,CAAC,CAAC;IACnE,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,YAAY,CAChB,UAAkB,EAClB,KAAa,EACb,QAAgB,EAChB,IAAa;QAEb,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;QACrD,OAAO,aAAa,CAAC,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE,WAAW,EAAE,IAAI,CAAC,CAAC;IACnE,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,SAAS,CAAC,UAAkB,EAAE,KAAa,EAAE,IAAa;QAC9D,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,MAAM,EAAE,CAAC;QACxC,OAAO,aAAa,CAAC,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE,WAAW,EAAE,IAAI,CAAC,CAAC;IACnE,CAAC;CACF,CAAC","sourcesContent":["/**\n * WebSocket Connection Store\n * \n * DynamoDB-backed connection management for API Gateway WebSocket APIs.\n * \n * Authentication:\n *   Two-phase auth — the client connects without credentials, then sends\n *   an { action: \"auth\", token: \"<jwt>\" } message over the encrypted channel.\n *   The connection is stored as unauthenticated on $connect and upgraded\n *   to authenticated via connectionStore.authenticate().\n *   This avoids leaking JWTs in query strings, server logs, and access logs.\n * \n * Multi-tenancy:\n *   If enabled, each connection stores a tenantId alongside userId.\n *   Use sendToTenant() to broadcast within a tenant boundary.\n * \n * API:\n *   connectionStore.save(connectionId)                        — $connect (unauthenticated)\n *   connectionStore.authenticate(connectionId, metadata)      — after JWT verification\n *   connectionStore.remove(connectionId)                      — $disconnect\n *   connectionStore.get(connectionId)                         — single connection\n *   connectionStore.getAll()                                  — all connections\n *   connectionStore.getByUser(userId)                         — by user (GSI)\n *   connectionStore.getByTenant(tenantId)                     — by tenant (GSI)\n *   connectionStore.postToConnection(domain, stage, id, data) — send to one\n *   connectionStore.sendToUser(domain, stage, userId, data)   — send to all user sessions\n *   connectionStore.sendToTenant(domain, stage, tenantId, data) — broadcast within tenant\n *   connectionStore.broadcast(domain, stage, data)            — broadcast to all\n * \n * Environment variables:\n *   CONNECTIONS_TABLE — DynamoDB table name (required)\n * \n * DynamoDB table schema:\n *   Partition key: connectionId (String)\n *   TTL attribute: ttl\n *   GSI: userId-index  (partition key: userId)\n *   GSI: tenantId-index (partition key: tenantId) — only if multi-tenancy enabled\n */\n\nimport { DynamoDBClient } from '@aws-sdk/client-dynamodb';\nimport {\n  DynamoDBDocumentClient,\n  PutCommand,\n  GetCommand,\n  UpdateCommand,\n  DeleteCommand,\n  ScanCommand,\n  QueryCommand,\n} from '@aws-sdk/lib-dynamodb';\nimport {\n  ApiGatewayManagementApiClient,\n  PostToConnectionCommand,\n  GoneException,\n} from '@aws-sdk/client-apigatewaymanagementapi';\n\nexport interface ConnectionRecord {\n  connectionId: string;\n  authenticated: boolean;\n  userId?: string;\n  tenantId?: string;\n  email?: string;\n  connectedAt: number;\n  authenticatedAt?: number;\n  ttl: number;\n  [key: string]: unknown;\n}\n\nexport interface ConnectionMetadata {\n  userId: string;\n  tenantId?: string;\n  email?: string;\n  [key: string]: unknown;\n}\n\nconst DEFAULT_TTL_SECONDS = 7200; // 2 hours\nconst AUTH_GRACE_PERIOD_SECONDS = 30; // unauthenticated connections expire quickly\n\nfunction getTableName(): string {\n  const table = process.env.CONNECTIONS_TABLE;\n  if (!table) {\n    throw new Error('CONNECTIONS_TABLE environment variable is not set');\n  }\n  return table;\n}\n\nlet _ddb: DynamoDBDocumentClient | null = null;\n\nfunction ddb(): DynamoDBDocumentClient {\n  if (!_ddb) {\n    _ddb = DynamoDBDocumentClient.from(new DynamoDBClient({}));\n  }\n  return _ddb;\n}\n\nfunction encode(data: unknown): Uint8Array {\n  return new TextEncoder().encode(JSON.stringify(data));\n}\n\n/**\n * Deliver data to a list of connections in parallel.\n * Removes stale connections automatically.\n * Returns count of successful deliveries.\n */\nasync function deliverToMany(\n  store: typeof connectionStore,\n  domainName: string,\n  stage: string,\n  connections: ConnectionRecord[],\n  data: unknown,\n): Promise<number> {\n  let delivered = 0;\n  const results = await Promise.allSettled(\n    connections\n      .filter((c) => c.authenticated)\n      .map((c) => store.postToConnection(domainName, stage, c.connectionId, data)),\n  );\n  for (const result of results) {\n    if (result.status === 'fulfilled' && result.value) delivered++;\n  }\n  return delivered;\n}\n\n/**\n * WebSocket connection store backed by DynamoDB.\n * \n * Two-phase authentication flow:\n * \n * 1. $connect handler:\n * ```typescript\n * await connectionStore.save(connectionId);\n * // Connection is unauthenticated, TTL = 30s\n * ```\n * \n * 2. Client sends auth message over the encrypted WebSocket:\n * ```json\n * { \"action\": \"auth\", \"token\": \"<jwt>\" }\n * ```\n * \n * 3. $default handler verifies JWT, then:\n * ```typescript\n * await connectionStore.authenticate(connectionId, { userId, tenantId, email });\n * // Connection is now authenticated, TTL extended to 2h\n * ```\n */\nexport const connectionStore = {\n  /**\n   * Save a new connection on $connect.\n   * The connection is unauthenticated with a short TTL.\n   * The client must send an auth message to upgrade.\n   */\n  async save(connectionId: string): Promise<void> {\n    const now = Date.now();\n    await ddb().send(new PutCommand({\n      TableName: getTableName(),\n      Item: {\n        connectionId,\n        authenticated: false,\n        connectedAt: now,\n        ttl: Math.floor(now / 1000) + AUTH_GRACE_PERIOD_SECONDS,\n      },\n    }));\n  },\n\n  /**\n   * Authenticate a connection after JWT verification.\n   * Updates the record with user metadata and extends the TTL.\n   * \n   * @param connectionId — the connection to authenticate\n   * @param metadata — must include userId, optionally tenantId, email, etc.\n   */\n  async authenticate(connectionId: string, metadata: ConnectionMetadata): Promise<void> {\n    const now = Date.now();\n    const { userId, tenantId, email, ...extra } = metadata;\n\n    const updateParts = [\n      'authenticated = :auth',\n      'userId = :uid',\n      'authenticatedAt = :authAt',\n      'ttl = :ttl',\n    ];\n    const values: Record<string, unknown> = {\n      ':auth': true,\n      ':uid': userId,\n      ':authAt': now,\n      ':ttl': Math.floor(now / 1000) + DEFAULT_TTL_SECONDS,\n    };\n\n    if (tenantId) {\n      updateParts.push('tenantId = :tid');\n      values[':tid'] = tenantId;\n    }\n\n    if (email) {\n      updateParts.push('email = :email');\n      values[':email'] = email;\n    }\n\n    // Store any extra metadata fields\n    let extraIdx = 0;\n    for (const [key, val] of Object.entries(extra)) {\n      const placeholder = `:extra${extraIdx}`;\n      updateParts.push(`#extra${extraIdx} = ${placeholder}`);\n      values[placeholder] = val;\n      extraIdx++;\n    }\n\n    const names: Record<string, string> = {};\n    for (let i = 0; i < extraIdx; i++) {\n      names[`#extra${i}`] = Object.keys(extra)[i];\n    }\n\n    await ddb().send(new UpdateCommand({\n      TableName: getTableName(),\n      Key: { connectionId },\n      UpdateExpression: `SET ${updateParts.join(', ')}`,\n      ExpressionAttributeValues: values,\n      ...(extraIdx > 0 ? { ExpressionAttributeNames: names } : {}),\n    }));\n  },\n\n  /**\n   * Remove a connection on $disconnect.\n   */\n  async remove(connectionId: string): Promise<void> {\n    await ddb().send(new DeleteCommand({\n      TableName: getTableName(),\n      Key: { connectionId },\n    }));\n  },\n\n  /**\n   * Get a single connection record.\n   */\n  async get(connectionId: string): Promise<ConnectionRecord | null> {\n    const result = await ddb().send(new GetCommand({\n      TableName: getTableName(),\n      Key: { connectionId },\n    }));\n    return (result.Item as ConnectionRecord) ?? null;\n  },\n\n  /**\n   * Get all authenticated connections.\n   * For large-scale apps, prefer getByUser() or getByTenant().\n   */\n  async getAll(): Promise<ConnectionRecord[]> {\n    const result = await ddb().send(new ScanCommand({\n      TableName: getTableName(),\n      FilterExpression: 'authenticated = :auth',\n      ExpressionAttributeValues: { ':auth': true },\n    }));\n    return (result.Items ?? []) as ConnectionRecord[];\n  },\n\n  /**\n   * Get all connections for a specific user.\n   * A user can have multiple active connections (e.g. multiple tabs/devices).\n   * Requires GSI: userId-index (partition key: userId).\n   */\n  async getByUser(userId: string): Promise<ConnectionRecord[]> {\n    const result = await ddb().send(new QueryCommand({\n      TableName: getTableName(),\n      IndexName: 'userId-index',\n      KeyConditionExpression: 'userId = :uid',\n      ExpressionAttributeValues: { ':uid': userId },\n    }));\n    return (result.Items ?? []) as ConnectionRecord[];\n  },\n\n  /**\n   * Get all connections for a specific tenant.\n   * Requires GSI: tenantId-index (partition key: tenantId).\n   */\n  async getByTenant(tenantId: string): Promise<ConnectionRecord[]> {\n    const result = await ddb().send(new QueryCommand({\n      TableName: getTableName(),\n      IndexName: 'tenantId-index',\n      KeyConditionExpression: 'tenantId = :tid',\n      ExpressionAttributeValues: { ':tid': tenantId },\n    }));\n    return (result.Items ?? []) as ConnectionRecord[];\n  },\n\n  /**\n   * Send data to a specific connection via API Gateway Management API.\n   * Automatically cleans up stale connections (GoneException).\n   * Returns true if sent, false if the connection was stale and removed.\n   */\n  async postToConnection(\n    domainName: string,\n    stage: string,\n    connectionId: string,\n    data: unknown,\n  ): Promise<boolean> {\n    const apigw = new ApiGatewayManagementApiClient({\n      endpoint: `https://${domainName}/${stage}`,\n    });\n\n    try {\n      await apigw.send(new PostToConnectionCommand({\n        ConnectionId: connectionId,\n        Data: encode(data),\n      }));\n      return true;\n    } catch (err) {\n      if (err instanceof GoneException) {\n        await this.remove(connectionId);\n        return false;\n      }\n      throw err;\n    }\n  },\n\n  /**\n   * Send data to all connections belonging to a specific user.\n   * Returns the number of connections that received the message.\n   */\n  async sendToUser(\n    domainName: string,\n    stage: string,\n    userId: string,\n    data: unknown,\n  ): Promise<number> {\n    const connections = await this.getByUser(userId);\n    return deliverToMany(this, domainName, stage, connections, data);\n  },\n\n  /**\n   * Send data to all connections within a specific tenant.\n   * Returns the number of connections that received the message.\n   */\n  async sendToTenant(\n    domainName: string,\n    stage: string,\n    tenantId: string,\n    data: unknown,\n  ): Promise<number> {\n    const connections = await this.getByTenant(tenantId);\n    return deliverToMany(this, domainName, stage, connections, data);\n  },\n\n  /**\n   * Broadcast data to all authenticated connections.\n   * Returns the number of clients that received the message.\n   */\n  async broadcast(domainName: string, stage: string, data: unknown): Promise<number> {\n    const connections = await this.getAll();\n    return deliverToMany(this, domainName, stage, connections, data);\n  },\n};\n"]}
|
package/package.json
ADDED
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@venturekit/runtime",
|
|
3
|
+
"version": "0.0.0-dev.20260307234057",
|
|
4
|
+
"description": "VentureKit runtime utilities - handlers, context, middleware, logging",
|
|
5
|
+
"main": "dist/index.js",
|
|
6
|
+
"types": "dist/index.d.ts",
|
|
7
|
+
"files": [
|
|
8
|
+
"dist"
|
|
9
|
+
],
|
|
10
|
+
"repository": {
|
|
11
|
+
"type": "git",
|
|
12
|
+
"url": "https://github.com/venturekit-dev/venturekit.private.git",
|
|
13
|
+
"directory": "packages/runtime"
|
|
14
|
+
},
|
|
15
|
+
"publishConfig": {
|
|
16
|
+
"registry": "https://registry.npmjs.org",
|
|
17
|
+
"access": "public"
|
|
18
|
+
},
|
|
19
|
+
"keywords": [
|
|
20
|
+
"venturekit",
|
|
21
|
+
"saas",
|
|
22
|
+
"lambda",
|
|
23
|
+
"runtime"
|
|
24
|
+
],
|
|
25
|
+
"license": "Apache-2.0",
|
|
26
|
+
"dependencies": {
|
|
27
|
+
"@venturekit/core": "0.0.0-dev.20260307234057",
|
|
28
|
+
"pino": "^9.0.0"
|
|
29
|
+
},
|
|
30
|
+
"peerDependencies": {
|
|
31
|
+
"@venturekit/data": "0.0.0-dev.20260307234057",
|
|
32
|
+
"@aws-sdk/client-dynamodb": "^3.500.0",
|
|
33
|
+
"@aws-sdk/lib-dynamodb": "^3.500.0",
|
|
34
|
+
"@aws-sdk/client-apigatewaymanagementapi": "^3.500.0"
|
|
35
|
+
},
|
|
36
|
+
"peerDependenciesMeta": {
|
|
37
|
+
"@venturekit/data": {
|
|
38
|
+
"optional": true
|
|
39
|
+
},
|
|
40
|
+
"@aws-sdk/client-dynamodb": {
|
|
41
|
+
"optional": true
|
|
42
|
+
},
|
|
43
|
+
"@aws-sdk/lib-dynamodb": {
|
|
44
|
+
"optional": true
|
|
45
|
+
},
|
|
46
|
+
"@aws-sdk/client-apigatewaymanagementapi": {
|
|
47
|
+
"optional": true
|
|
48
|
+
}
|
|
49
|
+
},
|
|
50
|
+
"devDependencies": {
|
|
51
|
+
"@venturekit/data": "0.0.0-dev.20260307234057",
|
|
52
|
+
"@types/aws-lambda": "^8.10.131",
|
|
53
|
+
"@types/node": "^20.10.0",
|
|
54
|
+
"typescript": "^5.3.0"
|
|
55
|
+
},
|
|
56
|
+
"scripts": {
|
|
57
|
+
"build": "tsc",
|
|
58
|
+
"dev": "tsc --watch",
|
|
59
|
+
"clean": "rm -rf dist"
|
|
60
|
+
}
|
|
61
|
+
}
|