@rebasepro/server-postgresql 0.0.1-canary.4d4fb3e

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 (147) hide show
  1. package/LICENSE +6 -0
  2. package/README.md +106 -0
  3. package/build-errors.txt +37 -0
  4. package/dist/common/src/collections/CollectionRegistry.d.ts +48 -0
  5. package/dist/common/src/collections/index.d.ts +1 -0
  6. package/dist/common/src/data/buildRebaseData.d.ts +14 -0
  7. package/dist/common/src/index.d.ts +3 -0
  8. package/dist/common/src/util/builders.d.ts +57 -0
  9. package/dist/common/src/util/callbacks.d.ts +6 -0
  10. package/dist/common/src/util/collections.d.ts +11 -0
  11. package/dist/common/src/util/common.d.ts +2 -0
  12. package/dist/common/src/util/conditions.d.ts +26 -0
  13. package/dist/common/src/util/entities.d.ts +36 -0
  14. package/dist/common/src/util/enums.d.ts +3 -0
  15. package/dist/common/src/util/index.d.ts +16 -0
  16. package/dist/common/src/util/navigation_from_path.d.ts +34 -0
  17. package/dist/common/src/util/navigation_utils.d.ts +20 -0
  18. package/dist/common/src/util/parent_references_from_path.d.ts +6 -0
  19. package/dist/common/src/util/paths.d.ts +14 -0
  20. package/dist/common/src/util/permissions.d.ts +5 -0
  21. package/dist/common/src/util/references.d.ts +2 -0
  22. package/dist/common/src/util/relations.d.ts +12 -0
  23. package/dist/common/src/util/resolutions.d.ts +72 -0
  24. package/dist/common/src/util/storage.d.ts +24 -0
  25. package/dist/index.es.js +10635 -0
  26. package/dist/index.es.js.map +1 -0
  27. package/dist/index.umd.js +10643 -0
  28. package/dist/index.umd.js.map +1 -0
  29. package/dist/server-postgresql/src/PostgresBackendDriver.d.ts +112 -0
  30. package/dist/server-postgresql/src/PostgresBootstrapper.d.ts +40 -0
  31. package/dist/server-postgresql/src/auth/ensure-tables.d.ts +6 -0
  32. package/dist/server-postgresql/src/auth/services.d.ts +188 -0
  33. package/dist/server-postgresql/src/cli.d.ts +1 -0
  34. package/dist/server-postgresql/src/collections/PostgresCollectionRegistry.d.ts +43 -0
  35. package/dist/server-postgresql/src/connection.d.ts +7 -0
  36. package/dist/server-postgresql/src/data-transformer.d.ts +36 -0
  37. package/dist/server-postgresql/src/databasePoolManager.d.ts +20 -0
  38. package/dist/server-postgresql/src/history/HistoryService.d.ts +71 -0
  39. package/dist/server-postgresql/src/history/ensure-history-table.d.ts +7 -0
  40. package/dist/server-postgresql/src/index.d.ts +13 -0
  41. package/dist/server-postgresql/src/interfaces.d.ts +18 -0
  42. package/dist/server-postgresql/src/schema/auth-schema.d.ts +767 -0
  43. package/dist/server-postgresql/src/schema/generate-drizzle-schema-logic.d.ts +2 -0
  44. package/dist/server-postgresql/src/schema/generate-drizzle-schema.d.ts +1 -0
  45. package/dist/server-postgresql/src/services/BranchService.d.ts +47 -0
  46. package/dist/server-postgresql/src/services/EntityFetchService.d.ts +195 -0
  47. package/dist/server-postgresql/src/services/EntityPersistService.d.ts +41 -0
  48. package/dist/server-postgresql/src/services/RelationService.d.ts +92 -0
  49. package/dist/server-postgresql/src/services/entity-helpers.d.ts +24 -0
  50. package/dist/server-postgresql/src/services/entityService.d.ts +102 -0
  51. package/dist/server-postgresql/src/services/index.d.ts +4 -0
  52. package/dist/server-postgresql/src/services/realtimeService.d.ts +186 -0
  53. package/dist/server-postgresql/src/utils/drizzle-conditions.d.ts +116 -0
  54. package/dist/server-postgresql/src/websocket.d.ts +5 -0
  55. package/dist/types/src/controllers/analytics_controller.d.ts +7 -0
  56. package/dist/types/src/controllers/auth.d.ts +117 -0
  57. package/dist/types/src/controllers/client.d.ts +58 -0
  58. package/dist/types/src/controllers/collection_registry.d.ts +44 -0
  59. package/dist/types/src/controllers/customization_controller.d.ts +54 -0
  60. package/dist/types/src/controllers/data.d.ts +141 -0
  61. package/dist/types/src/controllers/data_driver.d.ts +168 -0
  62. package/dist/types/src/controllers/database_admin.d.ts +11 -0
  63. package/dist/types/src/controllers/dialogs_controller.d.ts +36 -0
  64. package/dist/types/src/controllers/effective_role.d.ts +4 -0
  65. package/dist/types/src/controllers/index.d.ts +17 -0
  66. package/dist/types/src/controllers/local_config_persistence.d.ts +20 -0
  67. package/dist/types/src/controllers/navigation.d.ts +213 -0
  68. package/dist/types/src/controllers/registry.d.ts +51 -0
  69. package/dist/types/src/controllers/side_dialogs_controller.d.ts +67 -0
  70. package/dist/types/src/controllers/side_entity_controller.d.ts +89 -0
  71. package/dist/types/src/controllers/snackbar.d.ts +24 -0
  72. package/dist/types/src/controllers/storage.d.ts +173 -0
  73. package/dist/types/src/index.d.ts +4 -0
  74. package/dist/types/src/rebase_context.d.ts +101 -0
  75. package/dist/types/src/types/backend.d.ts +533 -0
  76. package/dist/types/src/types/builders.d.ts +14 -0
  77. package/dist/types/src/types/chips.d.ts +5 -0
  78. package/dist/types/src/types/collections.d.ts +812 -0
  79. package/dist/types/src/types/data_source.d.ts +64 -0
  80. package/dist/types/src/types/entities.d.ts +145 -0
  81. package/dist/types/src/types/entity_actions.d.ts +98 -0
  82. package/dist/types/src/types/entity_callbacks.d.ts +173 -0
  83. package/dist/types/src/types/entity_link_builder.d.ts +7 -0
  84. package/dist/types/src/types/entity_overrides.d.ts +9 -0
  85. package/dist/types/src/types/entity_views.d.ts +61 -0
  86. package/dist/types/src/types/export_import.d.ts +21 -0
  87. package/dist/types/src/types/index.d.ts +22 -0
  88. package/dist/types/src/types/locales.d.ts +4 -0
  89. package/dist/types/src/types/modify_collections.d.ts +5 -0
  90. package/dist/types/src/types/plugins.d.ts +225 -0
  91. package/dist/types/src/types/properties.d.ts +1091 -0
  92. package/dist/types/src/types/property_config.d.ts +70 -0
  93. package/dist/types/src/types/relations.d.ts +336 -0
  94. package/dist/types/src/types/slots.d.ts +228 -0
  95. package/dist/types/src/types/translations.d.ts +826 -0
  96. package/dist/types/src/types/user_management_delegate.d.ts +120 -0
  97. package/dist/types/src/types/websockets.d.ts +78 -0
  98. package/dist/types/src/users/index.d.ts +2 -0
  99. package/dist/types/src/users/roles.d.ts +22 -0
  100. package/dist/types/src/users/user.d.ts +46 -0
  101. package/jest-all.log +3128 -0
  102. package/jest.log +49 -0
  103. package/package.json +93 -0
  104. package/src/PostgresBackendDriver.ts +1024 -0
  105. package/src/PostgresBootstrapper.ts +232 -0
  106. package/src/auth/ensure-tables.ts +309 -0
  107. package/src/auth/services.ts +740 -0
  108. package/src/cli.ts +347 -0
  109. package/src/collections/PostgresCollectionRegistry.ts +96 -0
  110. package/src/connection.ts +62 -0
  111. package/src/data-transformer.ts +569 -0
  112. package/src/databasePoolManager.ts +84 -0
  113. package/src/history/HistoryService.ts +257 -0
  114. package/src/history/ensure-history-table.ts +45 -0
  115. package/src/index.ts +13 -0
  116. package/src/interfaces.ts +60 -0
  117. package/src/schema/auth-schema.ts +146 -0
  118. package/src/schema/generate-drizzle-schema-logic.ts +618 -0
  119. package/src/schema/generate-drizzle-schema.ts +151 -0
  120. package/src/services/BranchService.ts +237 -0
  121. package/src/services/EntityFetchService.ts +1447 -0
  122. package/src/services/EntityPersistService.ts +351 -0
  123. package/src/services/RelationService.ts +1012 -0
  124. package/src/services/entity-helpers.ts +121 -0
  125. package/src/services/entityService.ts +209 -0
  126. package/src/services/index.ts +13 -0
  127. package/src/services/realtimeService.ts +1005 -0
  128. package/src/utils/drizzle-conditions.ts +999 -0
  129. package/src/websocket.ts +487 -0
  130. package/test/auth-services.test.ts +569 -0
  131. package/test/branchService.test.ts +357 -0
  132. package/test/drizzle-conditions.test.ts +895 -0
  133. package/test/entityService.errors.test.ts +352 -0
  134. package/test/entityService.relations.test.ts +912 -0
  135. package/test/entityService.subcollection-search.test.ts +516 -0
  136. package/test/entityService.test.ts +977 -0
  137. package/test/generate-drizzle-schema.test.ts +795 -0
  138. package/test/historyService.test.ts +126 -0
  139. package/test/postgresDataDriver.test.ts +556 -0
  140. package/test/realtimeService.test.ts +276 -0
  141. package/test/relations.test.ts +662 -0
  142. package/test_drizzle_mock.js +3 -0
  143. package/test_find_changed.mjs +30 -0
  144. package/test_output.txt +3145 -0
  145. package/tsconfig.json +49 -0
  146. package/tsconfig.prod.json +20 -0
  147. package/vite.config.ts +82 -0
@@ -0,0 +1,487 @@
1
+ import { RealtimeService } from "./services/realtimeService";
2
+ import { PostgresBackendDriver } from "./PostgresBackendDriver";
3
+ import { DataDriver, DeleteEntityProps, FetchCollectionProps, FetchEntityProps, SaveEntityProps, TableMetadata, BranchInfo } from "@rebasepro/types";
4
+ import { WebSocketServer, WebSocket } from "ws";
5
+ import { Server } from "http";
6
+ import { inspect } from "util";
7
+ // @ts-ignore
8
+ import { extractUserFromToken, AccessTokenPayload } from "@rebasepro/server-core";
9
+ // @ts-ignore
10
+ import { AuthConfig } from "@rebasepro/server-core";
11
+
12
+ interface ClientSession {
13
+ ws: WebSocket;
14
+ user?: AccessTokenPayload;
15
+ authenticated: boolean;
16
+ /** Sliding window message counter for rate limiting */
17
+ messageCount: number;
18
+ messageWindowStart: number;
19
+ }
20
+
21
+ const clientSessions = new Map<string, ClientSession>();
22
+
23
+ /** Maximum messages per client per window */
24
+ const WS_RATE_LIMIT = 200;
25
+ /** Rate limit window in milliseconds (60 seconds) */
26
+ const WS_RATE_WINDOW_MS = 60_000;
27
+
28
+ /** Admin-only WebSocket message types */
29
+ const ADMIN_ONLY_TYPES = new Set([
30
+ "EXECUTE_SQL",
31
+ "FETCH_DATABASES",
32
+ "FETCH_ROLES",
33
+ "FETCH_UNMAPPED_TABLES",
34
+ "FETCH_TABLE_METADATA",
35
+ "FETCH_CURRENT_DATABASE",
36
+ "CREATE_BRANCH",
37
+ "DELETE_BRANCH",
38
+ "LIST_BRANCHES"
39
+ ]);
40
+
41
+ /**
42
+ * Check if the current session belongs to an admin user.
43
+ */
44
+ function isAdminSession(session: ClientSession | undefined): boolean {
45
+ if (!session?.user?.roles) return false;
46
+ return session.user.roles.some((r: unknown) => {
47
+ if (typeof r === "string") return r === "admin";
48
+ if (r && typeof r === "object" && "isAdmin" in r) return (r as { isAdmin: boolean }).isAdmin;
49
+ if (r && typeof r === "object" && "id" in r) return (r as { id: string }).id === "admin";
50
+ return false;
51
+ });
52
+ }
53
+
54
+ export function createPostgresWebSocket(
55
+ server: Server,
56
+ realtimeService: RealtimeService,
57
+ driver: PostgresBackendDriver,
58
+ authConfig?: AuthConfig
59
+ ) {
60
+ const isProduction = process.env.NODE_ENV === "production";
61
+ /** Debug logger that is suppressed in production to prevent PII/data leaks */
62
+ const wsDebug = (...args: unknown[]) => { if (!isProduction) console.debug(...args); };
63
+ const wss = new WebSocketServer({ server });
64
+ const requireAuth = authConfig?.requireAuth !== false && authConfig?.jwtSecret;
65
+
66
+ wss.on("connection", (ws) => {
67
+ const clientId = `client_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`;
68
+ wsDebug(`WebSocket client connected: ${clientId}`);
69
+
70
+ // Initialize client session
71
+ clientSessions.set(clientId, { ws, authenticated: !requireAuth, messageCount: 0, messageWindowStart: Date.now() });
72
+ realtimeService.addClient(clientId, ws);
73
+
74
+ ws.on("close", () => {
75
+ wsDebug(`WebSocket client disconnected: ${clientId}`);
76
+ clientSessions.delete(clientId);
77
+ });
78
+
79
+ // Route all messages through RealtimeService for unified handling
80
+ ws.on("message", async (message) => {
81
+ let requestId: string | undefined;
82
+ try {
83
+ const {
84
+ type,
85
+ payload,
86
+ requestId: reqId
87
+ } = JSON.parse(message.toString());
88
+ requestId = reqId; // Capture requestId for use in catch block
89
+
90
+ wsDebug(`[WS] ${clientId} → ${type}`, requestId ? `(${requestId})` : "");
91
+
92
+ // Handle authentication first
93
+ // Helper: send a canonical error frame
94
+ const sendError = (errType: "ERROR" | "AUTH_ERROR", code: string, msg: string) => {
95
+ ws.send(JSON.stringify({
96
+ type: errType,
97
+ requestId,
98
+ payload: { error: { message: msg, code } }
99
+ }));
100
+ };
101
+
102
+ if (type === "AUTHENTICATE") {
103
+ const { token } = payload || {};
104
+ if (!token) {
105
+ sendError("AUTH_ERROR", "INVALID_INPUT", "Token is required");
106
+ return;
107
+ }
108
+
109
+ const user = extractUserFromToken(token);
110
+ if (user) {
111
+ const session = clientSessions.get(clientId);
112
+ if (session) {
113
+ session.user = user;
114
+ session.authenticated = true;
115
+ }
116
+ wsDebug(`[WS] replying AUTH_SUCCESS for requestId ${requestId}`);
117
+ ws.send(JSON.stringify({
118
+ type: "AUTH_SUCCESS",
119
+ requestId,
120
+ payload: { userId: user.userId, roles: user.roles }
121
+ }));
122
+ wsDebug(`🔐 [WebSocket Server] Client ${clientId} authenticated as ${user.userId}`);
123
+ } else {
124
+ wsDebug(`[WS] replying AUTH_ERROR for requestId ${requestId} (invalid token)`);
125
+ sendError("AUTH_ERROR", "INVALID_TOKEN", "Invalid or expired token");
126
+ }
127
+ return;
128
+ }
129
+
130
+ // Check authentication for protected operations
131
+ if (requireAuth) {
132
+ const session = clientSessions.get(clientId);
133
+ if (!session?.authenticated) {
134
+ sendError("ERROR", "UNAUTHORIZED", "Authentication required");
135
+ return;
136
+ }
137
+ }
138
+
139
+ // Rate limiting: reject if client exceeds message limit
140
+ {
141
+ const session = clientSessions.get(clientId);
142
+ if (session) {
143
+ const now = Date.now();
144
+ if (now - session.messageWindowStart > WS_RATE_WINDOW_MS) {
145
+ session.messageCount = 0;
146
+ session.messageWindowStart = now;
147
+ }
148
+ session.messageCount++;
149
+ if (session.messageCount > WS_RATE_LIMIT) {
150
+ sendError("ERROR", "RATE_LIMITED", "Too many requests. Please slow down.");
151
+ return;
152
+ }
153
+ }
154
+ }
155
+
156
+ // Admin-only operations require admin role
157
+ if (ADMIN_ONLY_TYPES.has(type)) {
158
+ const session = clientSessions.get(clientId);
159
+ if (!isAdminSession(session)) {
160
+ sendError("ERROR", "FORBIDDEN", "Admin access required for this operation");
161
+ return;
162
+ }
163
+ }
164
+
165
+ // Helper to get correctly scoped delegate for the current request
166
+ const getScopedDelegate = async () => {
167
+ const session = clientSessions.get(clientId);
168
+ if (session?.user && "withAuth" in driver && typeof (driver as unknown as Record<string, unknown>).withAuth === "function") {
169
+ try {
170
+ // Map AccessTokenPayload back to User interface for withAuth (roles are already string IDs from JWT)
171
+ const userForAuth: Record<string, unknown> = {
172
+ uid: session.user.userId,
173
+ roles: session.user.roles ?? []
174
+ };
175
+ return await (driver as unknown as { withAuth: (user: Record<string, unknown>) => Promise<DataDriver> }).withAuth(userForAuth);
176
+ } catch (e) {
177
+ console.error("Failed to create authenticated delegate for WS request", e);
178
+ return driver;
179
+ }
180
+ }
181
+ return driver;
182
+ };
183
+
184
+ switch (type) {
185
+ case "FETCH_COLLECTION": {
186
+ wsDebug("📋 [WebSocket Server] Processing FETCH_COLLECTION request");
187
+ const request: FetchCollectionProps = payload;
188
+ const delegate = await getScopedDelegate();
189
+ const entities = await delegate.fetchCollection(request);
190
+ wsDebug("📋 [WebSocket Server] FETCH_COLLECTION result - entities count:", entities.length);
191
+ const response = {
192
+ type: "FETCH_COLLECTION_SUCCESS",
193
+ payload: { entities },
194
+ requestId
195
+ };
196
+ wsDebug("📋 [WebSocket Server] Sending FETCH_COLLECTION_SUCCESS response");
197
+ ws.send(JSON.stringify(response));
198
+ }
199
+ break;
200
+
201
+ case "FETCH_ENTITY": {
202
+ wsDebug("📄 [WebSocket Server] Processing FETCH_ENTITY request");
203
+ const request: FetchEntityProps = payload;
204
+ const delegate = await getScopedDelegate();
205
+ const entity = await delegate.fetchEntity(request);
206
+ wsDebug("📄 [WebSocket Server] FETCH_ENTITY result:", entity);
207
+ const response = {
208
+ type: "FETCH_ENTITY_SUCCESS",
209
+ payload: { entity },
210
+ requestId
211
+ };
212
+ wsDebug("📄 [WebSocket Server] Sending FETCH_ENTITY_SUCCESS response");
213
+ ws.send(JSON.stringify(response));
214
+ }
215
+ break;
216
+
217
+ case "SAVE_ENTITY": {
218
+ wsDebug("💾 [WebSocket Server] Processing SAVE_ENTITY request");
219
+ const request: SaveEntityProps = payload;
220
+ wsDebug("💾 [WebSocket Server] Saving entity with request:", inspect(request, { depth: null, colors: true }));
221
+ const delegate = await getScopedDelegate();
222
+ const entity = await delegate.saveEntity(request);
223
+ wsDebug("💾 [WebSocket Server] SAVE_ENTITY result:", inspect(entity, { depth: null, colors: true }));
224
+ const response = {
225
+ type: "SAVE_ENTITY_SUCCESS",
226
+ payload: { entity },
227
+ requestId
228
+ };
229
+ wsDebug("💾 [WebSocket Server] Sending SAVE_ENTITY_SUCCESS response");
230
+ ws.send(JSON.stringify(response));
231
+ }
232
+ break;
233
+
234
+ case "DELETE_ENTITY": {
235
+ wsDebug("🗑️ [WebSocket Server] Processing DELETE_ENTITY request");
236
+ const request: DeleteEntityProps = payload;
237
+ wsDebug("🗑️ [WebSocket Server] Deleting entity:", request.entity);
238
+ const delegate = await getScopedDelegate();
239
+ await delegate.deleteEntity(request);
240
+ wsDebug("🗑️ [WebSocket Server] DELETE_ENTITY completed successfully");
241
+ const response = {
242
+ type: "DELETE_ENTITY_SUCCESS",
243
+ payload: { success: true },
244
+ requestId
245
+ };
246
+ wsDebug("🗑️ [WebSocket Server] Sending DELETE_ENTITY_SUCCESS response");
247
+ ws.send(JSON.stringify(response));
248
+ }
249
+ break;
250
+
251
+ case "CHECK_UNIQUE_FIELD": {
252
+ wsDebug("🔍 [WebSocket Server] Processing CHECK_UNIQUE_FIELD request");
253
+ const {
254
+ path,
255
+ name,
256
+ value,
257
+ entityId,
258
+ collection
259
+ } = payload;
260
+ const delegate = await getScopedDelegate();
261
+ const isUnique = await delegate.checkUniqueField(path, name, value, entityId, collection);
262
+ wsDebug("🔍 [WebSocket Server] CHECK_UNIQUE_FIELD result:", isUnique);
263
+ const response = {
264
+ type: "CHECK_UNIQUE_FIELD_SUCCESS",
265
+ payload: { isUnique },
266
+ requestId
267
+ };
268
+ wsDebug("🔍 [WebSocket Server] Sending CHECK_UNIQUE_FIELD_SUCCESS response");
269
+ ws.send(JSON.stringify(response));
270
+ }
271
+ break;
272
+
273
+
274
+ case "COUNT_ENTITIES": {
275
+ const request: FetchCollectionProps = payload;
276
+ const delegate = await getScopedDelegate();
277
+ const count = await delegate.countEntities!(request);
278
+ const response = {
279
+ type: "COUNT_ENTITIES_SUCCESS",
280
+ payload: { count },
281
+ requestId
282
+ };
283
+ ws.send(JSON.stringify(response));
284
+ }
285
+ break;
286
+
287
+ case "EXECUTE_SQL": {
288
+ const { sql, options } = payload;
289
+ const delegate = await getScopedDelegate();
290
+ const result = await (delegate as unknown as { executeSql: (sql: string, options?: { database?: string, role?: string }) => Promise<Record<string, unknown>[]> }).executeSql(sql, options);
291
+ if (process.env.NODE_ENV !== "production") {
292
+ wsDebug(`⚡ [WebSocket Server] SQL executed. Returned ${Array.isArray(result) ? result.length : 'non-array'} rows.`);
293
+ }
294
+ const response = {
295
+ type: "EXECUTE_SQL_SUCCESS",
296
+ payload: { result },
297
+ requestId
298
+ };
299
+ ws.send(JSON.stringify(response));
300
+ }
301
+ break;
302
+
303
+ case "FETCH_DATABASES": {
304
+ wsDebug("📚 [WebSocket Server] Processing FETCH_DATABASES request");
305
+ const delegate = await getScopedDelegate();
306
+ let databases: string[] = [];
307
+ if (delegate.fetchAvailableDatabases) {
308
+ databases = await delegate.fetchAvailableDatabases();
309
+ }
310
+ wsDebug(`📚 [WebSocket Server] Fetched ${databases.length} databases.`);
311
+ const response = {
312
+ type: "FETCH_DATABASES_SUCCESS",
313
+ payload: { databases },
314
+ requestId
315
+ };
316
+ ws.send(JSON.stringify(response));
317
+ }
318
+ break;
319
+
320
+ case "FETCH_ROLES": {
321
+ wsDebug("👤 [WebSocket Server] Processing FETCH_ROLES request");
322
+ const delegate = await getScopedDelegate();
323
+ let roles: string[] = [];
324
+ if (delegate.fetchAvailableRoles) {
325
+ roles = await delegate.fetchAvailableRoles();
326
+ }
327
+ wsDebug(`👤 [WebSocket Server] Fetched ${roles.length} roles.`);
328
+ const response = {
329
+ type: "FETCH_ROLES_SUCCESS",
330
+ payload: { roles },
331
+ requestId
332
+ };
333
+ ws.send(JSON.stringify(response));
334
+ }
335
+ break;
336
+
337
+ case "FETCH_CURRENT_DATABASE": {
338
+ wsDebug("📚 [WebSocket Server] Processing FETCH_CURRENT_DATABASE request");
339
+ const delegate = await getScopedDelegate();
340
+ let database: string | undefined = undefined;
341
+ if (delegate.fetchCurrentDatabase) {
342
+ database = await delegate.fetchCurrentDatabase();
343
+ }
344
+ const response = {
345
+ type: "FETCH_CURRENT_DATABASE_SUCCESS",
346
+ payload: { database },
347
+ requestId
348
+ };
349
+ ws.send(JSON.stringify(response));
350
+ }
351
+ break;
352
+
353
+ case "FETCH_UNMAPPED_TABLES": {
354
+ wsDebug("📋 [WebSocket Server] Processing FETCH_UNMAPPED_TABLES request");
355
+ const delegate = await getScopedDelegate();
356
+ let tables: string[] = [];
357
+ if (delegate.fetchUnmappedTables) {
358
+ tables = await delegate.fetchUnmappedTables(payload?.mappedPaths);
359
+ }
360
+ wsDebug(`📋 [WebSocket Server] Fetched ${tables.length} unmapped tables.`);
361
+ const response = {
362
+ type: "FETCH_UNMAPPED_TABLES_SUCCESS",
363
+ payload: { tables },
364
+ requestId
365
+ };
366
+ ws.send(JSON.stringify(response));
367
+ }
368
+ break;
369
+
370
+ case "FETCH_TABLE_METADATA": {
371
+ wsDebug("📋 [WebSocket Server] Processing FETCH_TABLE_METADATA request");
372
+ const { tableName } = payload;
373
+ const delegate = await getScopedDelegate();
374
+ let metadata: TableMetadata | undefined;
375
+ if (delegate.fetchTableMetadata) {
376
+ metadata = await delegate.fetchTableMetadata(tableName);
377
+ }
378
+ wsDebug(`📋 [WebSocket Server] Fetched metadata for table '${tableName}'. (${metadata?.columns?.length ?? 0} columns)`);
379
+ const response = {
380
+ type: "FETCH_TABLE_METADATA_SUCCESS",
381
+ payload: { metadata },
382
+ requestId
383
+ };
384
+ ws.send(JSON.stringify(response));
385
+ }
386
+ break;
387
+
388
+ case "CREATE_BRANCH": {
389
+ wsDebug("🌿 [WebSocket Server] Processing CREATE_BRANCH request");
390
+ const { name, options } = payload;
391
+ const delegate = await getScopedDelegate();
392
+ if (!delegate.admin?.createBranch) {
393
+ sendError("ERROR", "NOT_SUPPORTED", "Database branching is not available. Configure adminConnectionString.");
394
+ break;
395
+ }
396
+ const branch: BranchInfo = await delegate.admin.createBranch(name, options);
397
+ wsDebug(`🌿 [WebSocket Server] Branch created: ${branch.name}`);
398
+ const response = {
399
+ type: "CREATE_BRANCH_SUCCESS",
400
+ payload: { branch },
401
+ requestId
402
+ };
403
+ ws.send(JSON.stringify(response));
404
+ }
405
+ break;
406
+
407
+ case "DELETE_BRANCH": {
408
+ wsDebug("🗑️ [WebSocket Server] Processing DELETE_BRANCH request");
409
+ const { name: branchName } = payload;
410
+ const delegate = await getScopedDelegate();
411
+ if (!delegate.admin?.deleteBranch) {
412
+ sendError("ERROR", "NOT_SUPPORTED", "Database branching is not available.");
413
+ break;
414
+ }
415
+ await delegate.admin.deleteBranch(branchName);
416
+ wsDebug(`🗑️ [WebSocket Server] Branch deleted: ${branchName}`);
417
+ const response = {
418
+ type: "DELETE_BRANCH_SUCCESS",
419
+ payload: { success: true },
420
+ requestId
421
+ };
422
+ ws.send(JSON.stringify(response));
423
+ }
424
+ break;
425
+
426
+ case "LIST_BRANCHES": {
427
+ wsDebug("🌿 [WebSocket Server] Processing LIST_BRANCHES request");
428
+ const delegate = await getScopedDelegate();
429
+ let branches: BranchInfo[] = [];
430
+ if (delegate.admin?.listBranches) {
431
+ branches = await delegate.admin.listBranches();
432
+ }
433
+ wsDebug(`🌿 [WebSocket Server] Listed ${branches.length} branches.`);
434
+ const response = {
435
+ type: "LIST_BRANCHES_SUCCESS",
436
+ payload: { branches },
437
+ requestId
438
+ };
439
+ ws.send(JSON.stringify(response));
440
+ }
441
+ break;
442
+
443
+ // Route subscription messages to RealtimeService
444
+ case "subscribe_collection":
445
+ case "subscribe_entity":
446
+ case "unsubscribe": {
447
+ wsDebug("🔄 [WebSocket Server] Routing subscription message to RealtimeService:", type);
448
+ // Attach auth context from the WS session so RLS-aware refetches work
449
+ const session = clientSessions.get(clientId);
450
+ const authContext = session?.user
451
+ ? { userId: session.user.userId, roles: session.user.roles ?? [] }
452
+ : undefined;
453
+ // Let RealtimeService handle these messages
454
+ await realtimeService.handleClientMessage(clientId, {
455
+ type,
456
+ payload,
457
+ subscriptionId: payload?.subscriptionId
458
+ }, authContext);
459
+ break;
460
+ }
461
+
462
+ default:
463
+ console.error("❌ [WebSocket Server] Unknown message type:", type);
464
+ }
465
+ } catch (error: unknown) {
466
+ console.error("💥 [WebSocket Server] Error handling message:", error);
467
+ if (error instanceof Error) {
468
+ console.error("Stack trace:", error.stack);
469
+ }
470
+ const errorMessage = process.env.NODE_ENV === "production"
471
+ ? "An unexpected error occurred"
472
+ : (error instanceof Error ? error.message : "An unexpected error occurred");
473
+ const errorResponse = {
474
+ type: "ERROR",
475
+ requestId,
476
+ payload: {
477
+ error: {
478
+ message: errorMessage,
479
+ code: "INTERNAL_ERROR"
480
+ }
481
+ }
482
+ };
483
+ ws.send(JSON.stringify(errorResponse));
484
+ }
485
+ });
486
+ });
487
+ }