expxagents 0.6.0 → 0.7.1

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 (156) hide show
  1. package/dist/__tests__/cli.test.d.ts +1 -0
  2. package/dist/__tests__/cli.test.js +23 -0
  3. package/dist/cli/src/commands/init.js +12 -2
  4. package/dist/cli/src/commands/virtual-office.js +50 -5
  5. package/dist/cli/src/utils/config.d.ts +5 -0
  6. package/dist/cli/src/utils/config.js +1 -1
  7. package/dist/commands/create.d.ts +1 -0
  8. package/dist/commands/create.js +34 -0
  9. package/dist/commands/init.d.ts +1 -0
  10. package/dist/commands/init.js +58 -0
  11. package/dist/commands/install.d.ts +1 -0
  12. package/dist/commands/install.js +16 -0
  13. package/dist/commands/list.d.ts +1 -0
  14. package/dist/commands/list.js +58 -0
  15. package/dist/commands/run.d.ts +1 -0
  16. package/dist/commands/run.js +173 -0
  17. package/dist/commands/server.d.ts +1 -0
  18. package/dist/commands/server.js +22 -0
  19. package/dist/commands/stop.d.ts +1 -0
  20. package/dist/commands/stop.js +23 -0
  21. package/dist/commands/uninstall.d.ts +1 -0
  22. package/dist/commands/uninstall.js +12 -0
  23. package/dist/dashboard/assets/BufferResource-Cf2Uo4_f.js +185 -0
  24. package/dist/dashboard/assets/CanvasRenderer-DFdMBORe.js +1 -0
  25. package/dist/dashboard/assets/RenderTargetSystem-CMh8XRf_.js +172 -0
  26. package/dist/dashboard/assets/WebGLRenderer-B5huw0RY.js +156 -0
  27. package/dist/dashboard/assets/WebGPURenderer-BdIKurkV.js +41 -0
  28. package/dist/dashboard/assets/browserAll-BjVJrv1L.js +14 -0
  29. package/dist/dashboard/assets/index-DwTFo09S.js +344 -0
  30. package/dist/dashboard/assets/webworkerAll-DMtK63GZ.js +83 -0
  31. package/dist/dashboard/index.html +16 -0
  32. package/dist/index.d.ts +1 -0
  33. package/dist/index.js +47 -0
  34. package/dist/server/api/health-routes.d.ts +3 -0
  35. package/dist/server/api/health-routes.d.ts.map +1 -0
  36. package/dist/server/api/health-routes.js +6 -0
  37. package/dist/server/api/health-routes.js.map +1 -0
  38. package/dist/server/api/squads-routes.d.ts +7 -0
  39. package/dist/server/api/squads-routes.d.ts.map +1 -0
  40. package/dist/server/api/squads-routes.js +91 -0
  41. package/dist/server/api/squads-routes.js.map +1 -0
  42. package/dist/server/api/users-routes.d.ts +8 -0
  43. package/dist/server/api/users-routes.d.ts.map +1 -0
  44. package/dist/server/api/users-routes.js +45 -0
  45. package/dist/server/api/users-routes.js.map +1 -0
  46. package/dist/server/app.d.ts +7 -0
  47. package/dist/server/app.d.ts.map +1 -0
  48. package/dist/server/app.js +109 -0
  49. package/dist/server/app.js.map +1 -0
  50. package/dist/server/auth/auth-middleware.d.ts +13 -0
  51. package/dist/server/auth/auth-middleware.d.ts.map +1 -0
  52. package/dist/server/auth/auth-middleware.js +25 -0
  53. package/dist/server/auth/auth-middleware.js.map +1 -0
  54. package/dist/server/auth/auth-routes.d.ts +9 -0
  55. package/dist/server/auth/auth-routes.d.ts.map +1 -0
  56. package/dist/server/auth/auth-routes.js +149 -0
  57. package/dist/server/auth/auth-routes.js.map +1 -0
  58. package/dist/server/auth/jwt.d.ts +14 -0
  59. package/dist/server/auth/jwt.d.ts.map +1 -0
  60. package/dist/server/auth/jwt.js +16 -0
  61. package/dist/server/auth/jwt.js.map +1 -0
  62. package/dist/server/auth/password.d.ts +3 -0
  63. package/dist/server/auth/password.d.ts.map +1 -0
  64. package/dist/server/auth/password.js +9 -0
  65. package/dist/server/auth/password.js.map +1 -0
  66. package/dist/server/bridge/__tests__/chat-handler.test.d.ts +2 -0
  67. package/dist/server/bridge/__tests__/chat-handler.test.d.ts.map +1 -0
  68. package/dist/server/bridge/__tests__/chat-handler.test.js +132 -0
  69. package/dist/server/bridge/__tests__/chat-handler.test.js.map +1 -0
  70. package/dist/server/bridge/__tests__/chat-integration.test.d.ts +2 -0
  71. package/dist/server/bridge/__tests__/chat-integration.test.d.ts.map +1 -0
  72. package/dist/server/bridge/__tests__/chat-integration.test.js +141 -0
  73. package/dist/server/bridge/__tests__/chat-integration.test.js.map +1 -0
  74. package/dist/server/bridge/__tests__/claude-bridge.test.d.ts +2 -0
  75. package/dist/server/bridge/__tests__/claude-bridge.test.d.ts.map +1 -0
  76. package/dist/server/bridge/__tests__/claude-bridge.test.js +223 -0
  77. package/dist/server/bridge/__tests__/claude-bridge.test.js.map +1 -0
  78. package/dist/server/bridge/__tests__/conversation.test.d.ts +2 -0
  79. package/dist/server/bridge/__tests__/conversation.test.d.ts.map +1 -0
  80. package/dist/server/bridge/__tests__/conversation.test.js +168 -0
  81. package/dist/server/bridge/__tests__/conversation.test.js.map +1 -0
  82. package/dist/server/bridge/__tests__/stream-parser.test.d.ts +2 -0
  83. package/dist/server/bridge/__tests__/stream-parser.test.d.ts.map +1 -0
  84. package/dist/server/bridge/__tests__/stream-parser.test.js +66 -0
  85. package/dist/server/bridge/__tests__/stream-parser.test.js.map +1 -0
  86. package/dist/server/bridge/chat-handler.d.ts +19 -0
  87. package/dist/server/bridge/chat-handler.d.ts.map +1 -0
  88. package/dist/server/bridge/chat-handler.js +104 -0
  89. package/dist/server/bridge/chat-handler.js.map +1 -0
  90. package/dist/server/bridge/claude-bridge.d.ts +27 -0
  91. package/dist/server/bridge/claude-bridge.d.ts.map +1 -0
  92. package/dist/server/bridge/claude-bridge.js +91 -0
  93. package/dist/server/bridge/claude-bridge.js.map +1 -0
  94. package/dist/server/bridge/conversation.d.ts +44 -0
  95. package/dist/server/bridge/conversation.d.ts.map +1 -0
  96. package/dist/server/bridge/conversation.js +73 -0
  97. package/dist/server/bridge/conversation.js.map +1 -0
  98. package/dist/server/bridge/stream-parser.d.ts +14 -0
  99. package/dist/server/bridge/stream-parser.d.ts.map +1 -0
  100. package/dist/server/bridge/stream-parser.js +26 -0
  101. package/dist/server/bridge/stream-parser.js.map +1 -0
  102. package/dist/server/config.d.ts +10 -0
  103. package/dist/server/config.d.ts.map +1 -0
  104. package/dist/server/config.js +22 -0
  105. package/dist/server/config.js.map +1 -0
  106. package/dist/server/db/__tests__/chat-tables.test.d.ts +2 -0
  107. package/dist/server/db/__tests__/chat-tables.test.d.ts.map +1 -0
  108. package/dist/server/db/__tests__/chat-tables.test.js +82 -0
  109. package/dist/server/db/__tests__/chat-tables.test.js.map +1 -0
  110. package/dist/server/db/connection.d.ts +4 -0
  111. package/dist/server/db/connection.d.ts.map +1 -0
  112. package/dist/server/db/connection.js +21 -0
  113. package/dist/server/db/connection.js.map +1 -0
  114. package/dist/server/db/migrations.d.ts +4 -0
  115. package/dist/server/db/migrations.d.ts.map +1 -0
  116. package/dist/server/db/migrations.js +17 -0
  117. package/dist/server/db/migrations.js.map +1 -0
  118. package/dist/server/db/schema.d.ts +2 -0
  119. package/dist/server/db/schema.d.ts.map +1 -0
  120. package/dist/server/db/schema.js +44 -0
  121. package/dist/server/db/schema.js.map +1 -0
  122. package/dist/server/index.d.ts +2 -0
  123. package/dist/server/index.d.ts.map +1 -0
  124. package/dist/server/index.js +18 -0
  125. package/dist/server/index.js.map +1 -0
  126. package/dist/server/routes/__tests__/conversations.test.d.ts +2 -0
  127. package/dist/server/routes/__tests__/conversations.test.d.ts.map +1 -0
  128. package/dist/server/routes/__tests__/conversations.test.js +94 -0
  129. package/dist/server/routes/__tests__/conversations.test.js.map +1 -0
  130. package/dist/server/routes/conversations.d.ts +8 -0
  131. package/dist/server/routes/conversations.d.ts.map +1 -0
  132. package/dist/server/routes/conversations.js +25 -0
  133. package/dist/server/routes/conversations.js.map +1 -0
  134. package/dist/server/watcher/file-watcher.d.ts +12 -0
  135. package/dist/server/watcher/file-watcher.d.ts.map +1 -0
  136. package/dist/server/watcher/file-watcher.js +68 -0
  137. package/dist/server/watcher/file-watcher.js.map +1 -0
  138. package/dist/server/watcher/state-parser.d.ts +77 -0
  139. package/dist/server/watcher/state-parser.d.ts.map +1 -0
  140. package/dist/server/watcher/state-parser.js +74 -0
  141. package/dist/server/watcher/state-parser.js.map +1 -0
  142. package/dist/server/ws/ws-auth.d.ts +4 -0
  143. package/dist/server/ws/ws-auth.d.ts.map +1 -0
  144. package/dist/server/ws/ws-auth.js +42 -0
  145. package/dist/server/ws/ws-auth.js.map +1 -0
  146. package/dist/server/ws/ws-handler.d.ts +11 -0
  147. package/dist/server/ws/ws-handler.d.ts.map +1 -0
  148. package/dist/server/ws/ws-handler.js +107 -0
  149. package/dist/server/ws/ws-handler.js.map +1 -0
  150. package/dist/server/ws/ws-rooms.d.ts +12 -0
  151. package/dist/server/ws/ws-rooms.d.ts.map +1 -0
  152. package/dist/server/ws/ws-rooms.js +52 -0
  153. package/dist/server/ws/ws-rooms.js.map +1 -0
  154. package/dist/utils/config.d.ts +15 -0
  155. package/dist/utils/config.js +23 -0
  156. package/package.json +16 -5
@@ -0,0 +1,94 @@
1
+ import { describe, it, expect, beforeEach } from 'vitest';
2
+ import Database from 'better-sqlite3';
3
+ import { createConversation, getConversations, getMessages, } from '../../bridge/conversation.js';
4
+ function createTestDb() {
5
+ const db = new Database(':memory:');
6
+ db.pragma('journal_mode = WAL');
7
+ db.exec(`
8
+ CREATE TABLE IF NOT EXISTS conversations (
9
+ id TEXT PRIMARY KEY,
10
+ squad_name TEXT NOT NULL,
11
+ created_at TEXT NOT NULL
12
+ );
13
+ CREATE TABLE IF NOT EXISTS chat_messages (
14
+ id TEXT PRIMARY KEY,
15
+ conversation_id TEXT NOT NULL REFERENCES conversations(id),
16
+ role TEXT NOT NULL,
17
+ content TEXT NOT NULL,
18
+ agent_id TEXT,
19
+ agent_name TEXT,
20
+ created_at TEXT NOT NULL
21
+ );
22
+ `);
23
+ return db;
24
+ }
25
+ // Simulates the handler logic
26
+ function handleGetConversations(db, squadName, cursor, limit = 20) {
27
+ return getConversations(db, squadName, { limit, cursor });
28
+ }
29
+ function handleGetMessages(db, conversationId, cursor, limit = 50) {
30
+ return getMessages(db, conversationId, { limit, cursor });
31
+ }
32
+ describe('conversation REST handlers', () => {
33
+ let db;
34
+ beforeEach(() => {
35
+ db = createTestDb();
36
+ });
37
+ describe('GET /api/squads/:squadName/conversations', () => {
38
+ it('should return empty list when no conversations', () => {
39
+ const result = handleGetConversations(db, 'content-team');
40
+ expect(result.data).toEqual([]);
41
+ expect(result.nextCursor).toBeNull();
42
+ });
43
+ it('should return conversations for the specified squad', () => {
44
+ createConversation(db, 'content-team');
45
+ createConversation(db, 'other-squad');
46
+ const result = handleGetConversations(db, 'content-team');
47
+ expect(result.data).toHaveLength(1);
48
+ expect(result.data[0].squad_name).toBe('content-team');
49
+ });
50
+ it('should paginate conversations', () => {
51
+ for (let i = 0; i < 5; i++) {
52
+ const ts = `2026-03-13T10:0${i}:00.000Z`;
53
+ db.prepare('INSERT INTO conversations (id, squad_name, created_at) VALUES (?, ?, ?)').run(`${ts}-c${i}`, 'content-team', ts);
54
+ }
55
+ const page1 = handleGetConversations(db, 'content-team', undefined, 2);
56
+ expect(page1.data).toHaveLength(2);
57
+ expect(page1.nextCursor).not.toBeNull();
58
+ const page2 = handleGetConversations(db, 'content-team', page1.nextCursor, 2);
59
+ expect(page2.data).toHaveLength(2);
60
+ });
61
+ });
62
+ describe('GET /api/conversations/:id/messages', () => {
63
+ it('should return empty list for conversation with no messages', () => {
64
+ const conv = createConversation(db, 'content-team');
65
+ const result = handleGetMessages(db, conv.id);
66
+ expect(result.data).toEqual([]);
67
+ expect(result.nextCursor).toBeNull();
68
+ });
69
+ it('should return messages oldest-first', () => {
70
+ const conv = createConversation(db, 'content-team');
71
+ // Use explicit IDs to guarantee ordering (same-millisecond generateId can randomize order)
72
+ db.prepare('INSERT INTO chat_messages (id, conversation_id, role, content, created_at) VALUES (?, ?, ?, ?, ?)').run('2026-03-13T10:00:01.000Z-aaa', conv.id, 'user', 'First', '2026-03-13T10:00:01.000Z');
73
+ db.prepare('INSERT INTO chat_messages (id, conversation_id, role, content, created_at) VALUES (?, ?, ?, ?, ?)').run('2026-03-13T10:00:02.000Z-bbb', conv.id, 'assistant', 'Second', '2026-03-13T10:00:02.000Z');
74
+ const result = handleGetMessages(db, conv.id);
75
+ expect(result.data).toHaveLength(2);
76
+ expect(result.data[0].content).toBe('First');
77
+ expect(result.data[1].content).toBe('Second');
78
+ });
79
+ it('should paginate messages', () => {
80
+ const conv = createConversation(db, 'content-team');
81
+ for (let i = 0; i < 5; i++) {
82
+ const ts = `2026-03-13T10:00:0${i}.000Z`;
83
+ db.prepare('INSERT INTO chat_messages (id, conversation_id, role, content, created_at) VALUES (?, ?, ?, ?, ?)').run(`${ts}-m${i}`, conv.id, 'user', `msg-${i}`, ts);
84
+ }
85
+ const page1 = handleGetMessages(db, conv.id, undefined, 2);
86
+ expect(page1.data).toHaveLength(2);
87
+ expect(page1.nextCursor).not.toBeNull();
88
+ const page2 = handleGetMessages(db, conv.id, page1.nextCursor, 2);
89
+ expect(page2.data).toHaveLength(2);
90
+ expect(page2.data[0].content).toBe('msg-2');
91
+ });
92
+ });
93
+ });
94
+ //# sourceMappingURL=conversations.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"conversations.test.js","sourceRoot":"","sources":["../../../src/routes/__tests__/conversations.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ import type { FastifyInstance, FastifyPluginOptions } from 'fastify';
2
+ import type Database from 'better-sqlite3';
3
+ interface ConversationRoutesOptions extends FastifyPluginOptions {
4
+ db: Database.Database;
5
+ }
6
+ export declare function conversationRoutes(app: FastifyInstance, opts: ConversationRoutesOptions): Promise<void>;
7
+ export {};
8
+ //# sourceMappingURL=conversations.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"conversations.d.ts","sourceRoot":"","sources":["../../src/routes/conversations.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,oBAAoB,EAAE,MAAM,SAAS,CAAC;AAErE,OAAO,KAAK,QAAQ,MAAM,gBAAgB,CAAC;AAE3C,UAAU,yBAA0B,SAAQ,oBAAoB;IAC9D,EAAE,EAAE,QAAQ,CAAC,QAAQ,CAAC;CACvB;AAED,wBAAsB,kBAAkB,CACtC,GAAG,EAAE,eAAe,EACpB,IAAI,EAAE,yBAAyB,GAC9B,OAAO,CAAC,IAAI,CAAC,CAgCf"}
@@ -0,0 +1,25 @@
1
+ import { getConversations, getMessages } from '../bridge/conversation.js';
2
+ export async function conversationRoutes(app, opts) {
3
+ const { db } = opts;
4
+ // GET /api/squads/:squadName/conversations
5
+ app.get('/api/squads/:squadName/conversations', {
6
+ preHandler: [app.requireAuth],
7
+ }, async (request, reply) => {
8
+ const { squadName } = request.params;
9
+ const cursor = request.query.cursor;
10
+ const limit = Math.min(parseInt(request.query.limit ?? '20', 10) || 20, 100);
11
+ const result = getConversations(db, squadName, { limit, cursor });
12
+ return reply.send(result);
13
+ });
14
+ // GET /api/conversations/:id/messages
15
+ app.get('/api/conversations/:id/messages', {
16
+ preHandler: [app.requireAuth],
17
+ }, async (request, reply) => {
18
+ const { id } = request.params;
19
+ const cursor = request.query.cursor;
20
+ const limit = Math.min(parseInt(request.query.limit ?? '50', 10) || 50, 200);
21
+ const result = getMessages(db, id, { limit, cursor });
22
+ return reply.send(result);
23
+ });
24
+ }
25
+ //# sourceMappingURL=conversations.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"conversations.js","sourceRoot":"","sources":["../../src/routes/conversations.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM,2BAA2B,CAAC;AAO1E,MAAM,CAAC,KAAK,UAAU,kBAAkB,CACtC,GAAoB,EACpB,IAA+B;IAE/B,MAAM,EAAE,EAAE,EAAE,GAAG,IAAI,CAAC;IAEpB,2CAA2C;IAC3C,GAAG,CAAC,GAAG,CAGJ,sCAAsC,EAAE;QACzC,UAAU,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC;KAC9B,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QAC1B,MAAM,EAAE,SAAS,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;QACrC,MAAM,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC;QACpC,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,IAAI,IAAI,EAAE,EAAE,CAAC,IAAI,EAAE,EAAE,GAAG,CAAC,CAAC;QAE7E,MAAM,MAAM,GAAG,gBAAgB,CAAC,EAAE,EAAE,SAAS,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;QAClE,OAAO,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC5B,CAAC,CAAC,CAAC;IAEH,sCAAsC;IACtC,GAAG,CAAC,GAAG,CAGJ,iCAAiC,EAAE;QACpC,UAAU,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC;KAC9B,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QAC1B,MAAM,EAAE,EAAE,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;QAC9B,MAAM,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC;QACpC,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,IAAI,IAAI,EAAE,EAAE,CAAC,IAAI,EAAE,EAAE,GAAG,CAAC,CAAC;QAE7E,MAAM,MAAM,GAAG,WAAW,CAAC,EAAE,EAAE,EAAE,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;QACtD,OAAO,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC5B,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,12 @@
1
+ import { type StateEvent } from './state-parser.js';
2
+ export type FileWatcherEvents = StateEvent | {
3
+ type: 'squad:inactive';
4
+ squadName: string;
5
+ };
6
+ type EventCallback = (event: FileWatcherEvents) => void;
7
+ export declare function createFileWatcher(squadsDir: string, onEvent: EventCallback): {
8
+ ready: Promise<void>;
9
+ stop: () => void;
10
+ };
11
+ export {};
12
+ //# sourceMappingURL=file-watcher.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-watcher.d.ts","sourceRoot":"","sources":["../../src/watcher/file-watcher.ts"],"names":[],"mappings":"AAGA,OAAO,EAA8B,KAAK,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAEhF,MAAM,MAAM,iBAAiB,GACzB,UAAU,GACV;IAAE,IAAI,EAAE,gBAAgB,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,CAAC;AAElD,KAAK,aAAa,GAAG,CAAC,KAAK,EAAE,iBAAiB,KAAK,IAAI,CAAC;AAExD,wBAAgB,iBAAiB,CAC/B,SAAS,EAAE,MAAM,EACjB,OAAO,EAAE,aAAa,GACrB;IAAE,KAAK,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC;IAAC,IAAI,EAAE,MAAM,IAAI,CAAA;CAAE,CA2E5C"}
@@ -0,0 +1,68 @@
1
+ import chokidar from 'chokidar';
2
+ import fs from 'node:fs';
3
+ import path from 'node:path';
4
+ import { diffState } from './state-parser.js';
5
+ export function createFileWatcher(squadsDir, onEvent) {
6
+ const lastStates = new Map();
7
+ const debounceTimers = new Map();
8
+ // Watch entire squadsDir (chokidar v4 on macOS doesn't support glob patterns well)
9
+ const watcher = chokidar.watch(squadsDir, {
10
+ ignoreInitial: true,
11
+ depth: 2,
12
+ });
13
+ function handleStateChange(filePath) {
14
+ const squadName = path.basename(path.dirname(filePath));
15
+ // Debounce per squad
16
+ clearTimeout(debounceTimers.get(squadName));
17
+ debounceTimers.set(squadName, setTimeout(() => {
18
+ if (!fs.existsSync(filePath)) {
19
+ lastStates.delete(squadName);
20
+ onEvent({ type: 'squad:inactive', squadName });
21
+ return;
22
+ }
23
+ try {
24
+ const raw = fs.readFileSync(filePath, 'utf-8');
25
+ const state = JSON.parse(raw);
26
+ const prev = lastStates.get(squadName) || null;
27
+ const events = diffState(squadName, prev, state);
28
+ lastStates.set(squadName, state);
29
+ for (const event of events) {
30
+ onEvent(event);
31
+ }
32
+ }
33
+ catch {
34
+ // Invalid JSON — skip, will retry on next write
35
+ }
36
+ }, 50));
37
+ }
38
+ function isStateJson(filePath) {
39
+ return path.basename(filePath) === 'state.json';
40
+ }
41
+ watcher.on('change', (filePath) => {
42
+ if (isStateJson(filePath)) {
43
+ handleStateChange(filePath);
44
+ }
45
+ });
46
+ watcher.on('add', (filePath) => {
47
+ if (isStateJson(filePath)) {
48
+ handleStateChange(filePath);
49
+ }
50
+ });
51
+ watcher.on('unlink', (filePath) => {
52
+ if (isStateJson(filePath)) {
53
+ const squadName = path.basename(path.dirname(filePath));
54
+ lastStates.delete(squadName);
55
+ onEvent({ type: 'squad:inactive', squadName });
56
+ }
57
+ });
58
+ const ready = new Promise((resolve) => {
59
+ watcher.on('ready', resolve);
60
+ });
61
+ const stop = () => {
62
+ for (const timer of debounceTimers.values())
63
+ clearTimeout(timer);
64
+ watcher.close();
65
+ };
66
+ return { ready, stop };
67
+ }
68
+ //# sourceMappingURL=file-watcher.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-watcher.js","sourceRoot":"","sources":["../../src/watcher/file-watcher.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,77 @@
1
+ export interface AgentDesk {
2
+ col: number;
3
+ row: number;
4
+ }
5
+ export type AgentStatus = 'idle' | 'working' | 'delivering' | 'done' | 'checkpoint';
6
+ export interface Agent {
7
+ id: string;
8
+ name: string;
9
+ icon: string;
10
+ status: AgentStatus;
11
+ deliverTo: string | null;
12
+ desk: AgentDesk;
13
+ message?: string;
14
+ stepIndex?: number;
15
+ stepLabel?: string;
16
+ }
17
+ export interface Handoff {
18
+ from: string;
19
+ to: string;
20
+ message: string;
21
+ completedAt: string;
22
+ }
23
+ export type SquadStatus = 'idle' | 'running' | 'completed' | 'checkpoint';
24
+ export interface SquadState {
25
+ squad: string;
26
+ status: SquadStatus;
27
+ step: {
28
+ current: number;
29
+ total: number;
30
+ label: string;
31
+ };
32
+ agents: Agent[];
33
+ handoff: Handoff | null;
34
+ startedAt: string | null;
35
+ updatedAt: string;
36
+ }
37
+ export type StateEvent = {
38
+ type: 'squad:state';
39
+ squadName: string;
40
+ state: SquadState;
41
+ } | {
42
+ type: 'agent:status';
43
+ agentId: string;
44
+ agentName: string;
45
+ status: string;
46
+ position?: {
47
+ x: number;
48
+ y: number;
49
+ };
50
+ } | {
51
+ type: 'chat:agent_dm';
52
+ agentId: string;
53
+ agentName: string;
54
+ message: string;
55
+ } | {
56
+ type: 'agent:step_start';
57
+ squadName: string;
58
+ agentId: string;
59
+ agentName: string;
60
+ stepIndex: number;
61
+ stepLabel: string;
62
+ } | {
63
+ type: 'agent:step_complete';
64
+ squadName: string;
65
+ agentId: string;
66
+ agentName: string;
67
+ stepIndex: number;
68
+ stepLabel: string;
69
+ } | {
70
+ type: 'squad:checkpoint';
71
+ squadName: string;
72
+ agentId: string;
73
+ agentName: string;
74
+ question: string;
75
+ };
76
+ export declare function diffState(squadName: string, prev: SquadState | null, next: SquadState): StateEvent[];
77
+ //# sourceMappingURL=state-parser.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"state-parser.d.ts","sourceRoot":"","sources":["../../src/watcher/state-parser.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,SAAS;IACxB,GAAG,EAAE,MAAM,CAAC;IACZ,GAAG,EAAE,MAAM,CAAC;CACb;AAED,MAAM,MAAM,WAAW,GAAG,MAAM,GAAG,SAAS,GAAG,YAAY,GAAG,MAAM,GAAG,YAAY,CAAC;AAEpF,MAAM,WAAW,KAAK;IACpB,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,WAAW,CAAC;IACpB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,IAAI,EAAE,SAAS,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,OAAO;IACtB,IAAI,EAAE,MAAM,CAAC;IACb,EAAE,EAAE,MAAM,CAAC;IACX,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,MAAM,WAAW,GAAG,MAAM,GAAG,SAAS,GAAG,WAAW,GAAG,YAAY,CAAC;AAE1E,MAAM,WAAW,UAAU;IACzB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,WAAW,CAAC;IACpB,IAAI,EAAE;QAAE,OAAO,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC;IACxD,MAAM,EAAE,KAAK,EAAE,CAAC;IAChB,OAAO,EAAE,OAAO,GAAG,IAAI,CAAC;IACxB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,MAAM,UAAU,GAClB;IAAE,IAAI,EAAE,aAAa,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,KAAK,EAAE,UAAU,CAAA;CAAE,GAC7D;IAAE,IAAI,EAAE,cAAc,CAAC;IAAC,OAAO,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC;IAAC,QAAQ,CAAC,EAAE;QAAE,CAAC,EAAE,MAAM,CAAC;QAAC,CAAC,EAAE,MAAM,CAAA;KAAE,CAAA;CAAE,GACjH;IAAE,IAAI,EAAE,eAAe,CAAC;IAAC,OAAO,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,GAC9E;IAAE,IAAI,EAAE,kBAAkB,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,GACzH;IAAE,IAAI,EAAE,qBAAqB,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,GAC5H;IAAE,IAAI,EAAE,kBAAkB,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAA;CAAE,CAAC;AAE1G,wBAAgB,SAAS,CACvB,SAAS,EAAE,MAAM,EACjB,IAAI,EAAE,UAAU,GAAG,IAAI,EACvB,IAAI,EAAE,UAAU,GACf,UAAU,EAAE,CAqFd"}
@@ -0,0 +1,74 @@
1
+ export function diffState(squadName, prev, next) {
2
+ const events = [];
3
+ // No previous state — emit full state
4
+ if (!prev) {
5
+ events.push({ type: 'squad:state', squadName, state: next });
6
+ return events;
7
+ }
8
+ // Squad-level changes
9
+ const squadChanged = prev.status !== next.status ||
10
+ prev.step.current !== next.step.current ||
11
+ JSON.stringify(prev.handoff) !== JSON.stringify(next.handoff);
12
+ if (squadChanged) {
13
+ events.push({ type: 'squad:state', squadName, state: next });
14
+ }
15
+ // Checkpoint detection — when squad status transitions to 'checkpoint'
16
+ if (prev.status !== 'checkpoint' && next.status === 'checkpoint') {
17
+ const checkpointAgent = next.agents.find((a) => a.status === 'checkpoint');
18
+ if (checkpointAgent) {
19
+ events.push({
20
+ type: 'squad:checkpoint',
21
+ squadName,
22
+ agentId: checkpointAgent.id,
23
+ agentName: checkpointAgent.name,
24
+ question: checkpointAgent.message ?? '',
25
+ });
26
+ }
27
+ }
28
+ // Agent-level changes
29
+ const prevAgentMap = new Map(prev.agents.map((a) => [a.id, a]));
30
+ for (const agent of next.agents) {
31
+ const prevAgent = prevAgentMap.get(agent.id);
32
+ if (!prevAgent || prevAgent.status !== agent.status) {
33
+ events.push({
34
+ type: 'agent:status',
35
+ agentId: agent.id,
36
+ agentName: agent.name,
37
+ status: agent.status,
38
+ });
39
+ }
40
+ // Step change detection
41
+ if (prevAgent &&
42
+ agent.stepIndex !== undefined &&
43
+ prevAgent.stepIndex !== undefined &&
44
+ prevAgent.stepIndex !== agent.stepIndex) {
45
+ events.push({
46
+ type: 'agent:step_complete',
47
+ squadName,
48
+ agentId: agent.id,
49
+ agentName: agent.name,
50
+ stepIndex: prevAgent.stepIndex,
51
+ stepLabel: prevAgent.stepLabel ?? '',
52
+ });
53
+ events.push({
54
+ type: 'agent:step_start',
55
+ squadName,
56
+ agentId: agent.id,
57
+ agentName: agent.name,
58
+ stepIndex: agent.stepIndex,
59
+ stepLabel: agent.stepLabel ?? '',
60
+ });
61
+ }
62
+ // Agent DM: message field populated when it wasn't before
63
+ if (agent.message && (!prevAgent || !prevAgent.message)) {
64
+ events.push({
65
+ type: 'chat:agent_dm',
66
+ agentId: agent.id,
67
+ agentName: agent.name,
68
+ message: agent.message,
69
+ });
70
+ }
71
+ }
72
+ return events;
73
+ }
74
+ //# sourceMappingURL=state-parser.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"state-parser.js","sourceRoot":"","sources":["../../src/watcher/state-parser.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import { type AccessTokenPayload } from '../auth/jwt.js';
2
+ export declare function extractTokenFromUrl(url: string): string | null;
3
+ export declare function authenticateWs(url: string, cookieHeader: string | undefined, jwtSecret: string): AccessTokenPayload | null;
4
+ //# sourceMappingURL=ws-auth.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-auth.d.ts","sourceRoot":"","sources":["../../src/ws/ws-auth.ts"],"names":[],"mappings":"AAAA,OAAO,EAAqB,KAAK,kBAAkB,EAAE,MAAM,gBAAgB,CAAC;AAE5E,wBAAgB,mBAAmB,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAQ9D;AAED,wBAAgB,cAAc,CAC5B,GAAG,EAAE,MAAM,EACX,YAAY,EAAE,MAAM,GAAG,SAAS,EAChC,SAAS,EAAE,MAAM,GAChB,kBAAkB,GAAG,IAAI,CA8B3B"}
@@ -0,0 +1,42 @@
1
+ import { verifyAccessToken } from '../auth/jwt.js';
2
+ export function extractTokenFromUrl(url) {
3
+ if (!url)
4
+ return null;
5
+ try {
6
+ const searchParams = new URLSearchParams(url.split('?')[1] || '');
7
+ return searchParams.get('token') || null;
8
+ }
9
+ catch {
10
+ return null;
11
+ }
12
+ }
13
+ export function authenticateWs(url, cookieHeader, jwtSecret) {
14
+ // Try query param first (for WebSocket connections from browser)
15
+ const queryToken = extractTokenFromUrl(url);
16
+ if (queryToken) {
17
+ try {
18
+ return verifyAccessToken(queryToken, jwtSecret);
19
+ }
20
+ catch {
21
+ return null;
22
+ }
23
+ }
24
+ // Try cookie
25
+ if (cookieHeader) {
26
+ const cookies = Object.fromEntries(cookieHeader.split(';').map((c) => {
27
+ const [key, ...vals] = c.trim().split('=');
28
+ return [key, vals.join('=')];
29
+ }));
30
+ const token = cookies['access_token'];
31
+ if (token) {
32
+ try {
33
+ return verifyAccessToken(token, jwtSecret);
34
+ }
35
+ catch {
36
+ return null;
37
+ }
38
+ }
39
+ }
40
+ return null;
41
+ }
42
+ //# sourceMappingURL=ws-auth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-auth.js","sourceRoot":"","sources":["../../src/ws/ws-auth.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,iBAAiB,EAA2B,MAAM,gBAAgB,CAAC;AAE5E,MAAM,UAAU,mBAAmB,CAAC,GAAW;IAC7C,IAAI,CAAC,GAAG;QAAE,OAAO,IAAI,CAAC;IACtB,IAAI,CAAC;QACH,MAAM,YAAY,GAAG,IAAI,eAAe,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;QAClE,OAAO,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC;IAC3C,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,MAAM,UAAU,cAAc,CAC5B,GAAW,EACX,YAAgC,EAChC,SAAiB;IAEjB,iEAAiE;IACjE,MAAM,UAAU,GAAG,mBAAmB,CAAC,GAAG,CAAC,CAAC;IAC5C,IAAI,UAAU,EAAE,CAAC;QACf,IAAI,CAAC;YACH,OAAO,iBAAiB,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;QAClD,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED,aAAa;IACb,IAAI,YAAY,EAAE,CAAC;QACjB,MAAM,OAAO,GAAG,MAAM,CAAC,WAAW,CAChC,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;YAChC,MAAM,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAC3C,OAAO,CAAC,GAAG,EAAE,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;QAC/B,CAAC,CAAC,CACH,CAAC;QACF,MAAM,KAAK,GAAG,OAAO,CAAC,cAAc,CAAC,CAAC;QACtC,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,CAAC;gBACH,OAAO,iBAAiB,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;YAC7C,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,IAAI,CAAC;YACd,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC"}
@@ -0,0 +1,11 @@
1
+ import type { FastifyInstance } from 'fastify';
2
+ import { WsRooms } from './ws-rooms.js';
3
+ import type { ChatHandler } from '../bridge/chat-handler.js';
4
+ export interface WsContext {
5
+ rooms: WsRooms;
6
+ broadcastToSquad: (squadName: string, data: object) => void;
7
+ broadcastToAll: (data: object) => void;
8
+ setChatHandler: (handler: ChatHandler) => void;
9
+ }
10
+ export declare function registerWebSocket(app: FastifyInstance, jwtSecret: string, squadsDir?: string): Promise<WsContext>;
11
+ //# sourceMappingURL=ws-handler.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-handler.d.ts","sourceRoot":"","sources":["../../src/ws/ws-handler.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAM/C,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,CAAC;AAExC,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,2BAA2B,CAAC;AAO7D,MAAM,WAAW,SAAS;IACxB,KAAK,EAAE,OAAO,CAAC;IACf,gBAAgB,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;IAC5D,cAAc,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;IACvC,cAAc,EAAE,CAAC,OAAO,EAAE,WAAW,KAAK,IAAI,CAAC;CAChD;AAED,wBAAsB,iBAAiB,CACrC,GAAG,EAAE,eAAe,EACpB,SAAS,EAAE,MAAM,EACjB,SAAS,GAAE,MAAW,GACrB,OAAO,CAAC,SAAS,CAAC,CAgDpB"}
@@ -0,0 +1,107 @@
1
+ import websocket from '@fastify/websocket';
2
+ import fs from 'fs';
3
+ import path from 'path';
4
+ import { authenticateWs } from './ws-auth.js';
5
+ import { WsRooms } from './ws-rooms.js';
6
+ export async function registerWebSocket(app, jwtSecret, squadsDir = '') {
7
+ const rooms = new WsRooms();
8
+ let chatHandler = null;
9
+ await app.register(websocket);
10
+ app.get('/ws', { websocket: true }, (socket, request) => {
11
+ const user = authenticateWs(request.url, request.headers.cookie, jwtSecret);
12
+ if (!user) {
13
+ socket.close(4001, 'Authentication required');
14
+ return;
15
+ }
16
+ // Attach user info to socket for reference
17
+ socket.user = user;
18
+ socket.on('message', (raw) => {
19
+ try {
20
+ const msg = JSON.parse(raw.toString());
21
+ handleMessage(socket, user, msg, rooms, chatHandler, squadsDir);
22
+ }
23
+ catch {
24
+ // Silently ignore malformed messages
25
+ }
26
+ });
27
+ socket.on('close', () => {
28
+ rooms.removeClient(socket);
29
+ });
30
+ });
31
+ const broadcastToSquad = (squadName, data) => {
32
+ rooms.broadcast(`squad:${squadName}`, JSON.stringify(data));
33
+ };
34
+ const broadcastToAll = (data) => {
35
+ rooms.broadcastAll(JSON.stringify(data));
36
+ };
37
+ const setChatHandler = (handler) => {
38
+ chatHandler = handler;
39
+ };
40
+ return { rooms, broadcastToSquad, broadcastToAll, setChatHandler };
41
+ }
42
+ function handleMessage(ws, user, msg, rooms, chatHandler, squadsDir = '') {
43
+ switch (msg.type) {
44
+ case 'squad:subscribe':
45
+ if (typeof msg.squadName === 'string') {
46
+ rooms.subscribe(ws, `squad:${msg.squadName}`);
47
+ }
48
+ break;
49
+ case 'squad:unsubscribe':
50
+ if (typeof msg.squadName === 'string') {
51
+ rooms.unsubscribe(ws, `squad:${msg.squadName}`);
52
+ }
53
+ break;
54
+ case 'chat:send':
55
+ if (typeof msg.squadName === 'string' && typeof msg.message === 'string' && chatHandler) {
56
+ if (user.role === 'viewer') {
57
+ ws.send(JSON.stringify({
58
+ type: 'chat:error',
59
+ squadName: msg.squadName,
60
+ error: 'Insufficient permissions',
61
+ }));
62
+ return;
63
+ }
64
+ chatHandler.handleChatSend(msg.squadName, msg.message);
65
+ }
66
+ break;
67
+ case 'chat:new_conversation':
68
+ if (typeof msg.squadName === 'string' && chatHandler) {
69
+ if (user.role === 'viewer') {
70
+ ws.send(JSON.stringify({
71
+ type: 'chat:error',
72
+ squadName: msg.squadName,
73
+ error: 'Insufficient permissions',
74
+ }));
75
+ return;
76
+ }
77
+ chatHandler.handleNewConversation(msg.squadName);
78
+ }
79
+ break;
80
+ case 'squad:checkpoint_response':
81
+ if (typeof msg.squadName === 'string' && typeof msg.answer === 'string') {
82
+ if (user.role === 'viewer') {
83
+ ws.send(JSON.stringify({
84
+ type: 'squad:error',
85
+ squadName: msg.squadName,
86
+ error: 'Insufficient permissions',
87
+ }));
88
+ return;
89
+ }
90
+ try {
91
+ const squadDir = path.join(squadsDir, msg.squadName);
92
+ const filePath = path.join(squadDir, 'checkpoint-response.json');
93
+ fs.writeFileSync(filePath, JSON.stringify({
94
+ answer: msg.answer,
95
+ timestamp: new Date().toISOString(),
96
+ }));
97
+ }
98
+ catch {
99
+ // Silently handle missing squad directories
100
+ }
101
+ }
102
+ break;
103
+ default:
104
+ break;
105
+ }
106
+ }
107
+ //# sourceMappingURL=ws-handler.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-handler.js","sourceRoot":"","sources":["../../src/ws/ws-handler.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,12 @@
1
+ import type { WebSocket } from 'ws';
2
+ export declare class WsRooms {
3
+ private rooms;
4
+ private clientRooms;
5
+ subscribe(ws: WebSocket, room: string): void;
6
+ unsubscribe(ws: WebSocket, room: string): void;
7
+ removeClient(ws: WebSocket): void;
8
+ broadcast(room: string, data: string): void;
9
+ broadcastAll(data: string): void;
10
+ getSubscribers(room: string): Set<WebSocket>;
11
+ }
12
+ //# sourceMappingURL=ws-rooms.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-rooms.d.ts","sourceRoot":"","sources":["../../src/ws/ws-rooms.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AAEpC,qBAAa,OAAO;IAClB,OAAO,CAAC,KAAK,CAAqC;IAClD,OAAO,CAAC,WAAW,CAAqC;IAExD,SAAS,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,GAAG,IAAI;IAY5C,WAAW,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,GAAG,IAAI;IAK9C,YAAY,CAAC,EAAE,EAAE,SAAS,GAAG,IAAI;IAUjC,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,IAAI;IAU3C,YAAY,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI;IAYhC,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,GAAG,CAAC,SAAS,CAAC;CAG7C"}