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.
- package/dist/__tests__/cli.test.d.ts +1 -0
- package/dist/__tests__/cli.test.js +23 -0
- package/dist/cli/src/commands/init.js +12 -2
- package/dist/cli/src/commands/virtual-office.js +50 -5
- package/dist/cli/src/utils/config.d.ts +5 -0
- package/dist/cli/src/utils/config.js +1 -1
- package/dist/commands/create.d.ts +1 -0
- package/dist/commands/create.js +34 -0
- package/dist/commands/init.d.ts +1 -0
- package/dist/commands/init.js +58 -0
- package/dist/commands/install.d.ts +1 -0
- package/dist/commands/install.js +16 -0
- package/dist/commands/list.d.ts +1 -0
- package/dist/commands/list.js +58 -0
- package/dist/commands/run.d.ts +1 -0
- package/dist/commands/run.js +173 -0
- package/dist/commands/server.d.ts +1 -0
- package/dist/commands/server.js +22 -0
- package/dist/commands/stop.d.ts +1 -0
- package/dist/commands/stop.js +23 -0
- package/dist/commands/uninstall.d.ts +1 -0
- package/dist/commands/uninstall.js +12 -0
- package/dist/dashboard/assets/BufferResource-Cf2Uo4_f.js +185 -0
- package/dist/dashboard/assets/CanvasRenderer-DFdMBORe.js +1 -0
- package/dist/dashboard/assets/RenderTargetSystem-CMh8XRf_.js +172 -0
- package/dist/dashboard/assets/WebGLRenderer-B5huw0RY.js +156 -0
- package/dist/dashboard/assets/WebGPURenderer-BdIKurkV.js +41 -0
- package/dist/dashboard/assets/browserAll-BjVJrv1L.js +14 -0
- package/dist/dashboard/assets/index-DwTFo09S.js +344 -0
- package/dist/dashboard/assets/webworkerAll-DMtK63GZ.js +83 -0
- package/dist/dashboard/index.html +16 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +47 -0
- package/dist/server/api/health-routes.d.ts +3 -0
- package/dist/server/api/health-routes.d.ts.map +1 -0
- package/dist/server/api/health-routes.js +6 -0
- package/dist/server/api/health-routes.js.map +1 -0
- package/dist/server/api/squads-routes.d.ts +7 -0
- package/dist/server/api/squads-routes.d.ts.map +1 -0
- package/dist/server/api/squads-routes.js +91 -0
- package/dist/server/api/squads-routes.js.map +1 -0
- package/dist/server/api/users-routes.d.ts +8 -0
- package/dist/server/api/users-routes.d.ts.map +1 -0
- package/dist/server/api/users-routes.js +45 -0
- package/dist/server/api/users-routes.js.map +1 -0
- package/dist/server/app.d.ts +7 -0
- package/dist/server/app.d.ts.map +1 -0
- package/dist/server/app.js +109 -0
- package/dist/server/app.js.map +1 -0
- package/dist/server/auth/auth-middleware.d.ts +13 -0
- package/dist/server/auth/auth-middleware.d.ts.map +1 -0
- package/dist/server/auth/auth-middleware.js +25 -0
- package/dist/server/auth/auth-middleware.js.map +1 -0
- package/dist/server/auth/auth-routes.d.ts +9 -0
- package/dist/server/auth/auth-routes.d.ts.map +1 -0
- package/dist/server/auth/auth-routes.js +149 -0
- package/dist/server/auth/auth-routes.js.map +1 -0
- package/dist/server/auth/jwt.d.ts +14 -0
- package/dist/server/auth/jwt.d.ts.map +1 -0
- package/dist/server/auth/jwt.js +16 -0
- package/dist/server/auth/jwt.js.map +1 -0
- package/dist/server/auth/password.d.ts +3 -0
- package/dist/server/auth/password.d.ts.map +1 -0
- package/dist/server/auth/password.js +9 -0
- package/dist/server/auth/password.js.map +1 -0
- package/dist/server/bridge/__tests__/chat-handler.test.d.ts +2 -0
- package/dist/server/bridge/__tests__/chat-handler.test.d.ts.map +1 -0
- package/dist/server/bridge/__tests__/chat-handler.test.js +132 -0
- package/dist/server/bridge/__tests__/chat-handler.test.js.map +1 -0
- package/dist/server/bridge/__tests__/chat-integration.test.d.ts +2 -0
- package/dist/server/bridge/__tests__/chat-integration.test.d.ts.map +1 -0
- package/dist/server/bridge/__tests__/chat-integration.test.js +141 -0
- package/dist/server/bridge/__tests__/chat-integration.test.js.map +1 -0
- package/dist/server/bridge/__tests__/claude-bridge.test.d.ts +2 -0
- package/dist/server/bridge/__tests__/claude-bridge.test.d.ts.map +1 -0
- package/dist/server/bridge/__tests__/claude-bridge.test.js +223 -0
- package/dist/server/bridge/__tests__/claude-bridge.test.js.map +1 -0
- package/dist/server/bridge/__tests__/conversation.test.d.ts +2 -0
- package/dist/server/bridge/__tests__/conversation.test.d.ts.map +1 -0
- package/dist/server/bridge/__tests__/conversation.test.js +168 -0
- package/dist/server/bridge/__tests__/conversation.test.js.map +1 -0
- package/dist/server/bridge/__tests__/stream-parser.test.d.ts +2 -0
- package/dist/server/bridge/__tests__/stream-parser.test.d.ts.map +1 -0
- package/dist/server/bridge/__tests__/stream-parser.test.js +66 -0
- package/dist/server/bridge/__tests__/stream-parser.test.js.map +1 -0
- package/dist/server/bridge/chat-handler.d.ts +19 -0
- package/dist/server/bridge/chat-handler.d.ts.map +1 -0
- package/dist/server/bridge/chat-handler.js +104 -0
- package/dist/server/bridge/chat-handler.js.map +1 -0
- package/dist/server/bridge/claude-bridge.d.ts +27 -0
- package/dist/server/bridge/claude-bridge.d.ts.map +1 -0
- package/dist/server/bridge/claude-bridge.js +91 -0
- package/dist/server/bridge/claude-bridge.js.map +1 -0
- package/dist/server/bridge/conversation.d.ts +44 -0
- package/dist/server/bridge/conversation.d.ts.map +1 -0
- package/dist/server/bridge/conversation.js +73 -0
- package/dist/server/bridge/conversation.js.map +1 -0
- package/dist/server/bridge/stream-parser.d.ts +14 -0
- package/dist/server/bridge/stream-parser.d.ts.map +1 -0
- package/dist/server/bridge/stream-parser.js +26 -0
- package/dist/server/bridge/stream-parser.js.map +1 -0
- package/dist/server/config.d.ts +10 -0
- package/dist/server/config.d.ts.map +1 -0
- package/dist/server/config.js +22 -0
- package/dist/server/config.js.map +1 -0
- package/dist/server/db/__tests__/chat-tables.test.d.ts +2 -0
- package/dist/server/db/__tests__/chat-tables.test.d.ts.map +1 -0
- package/dist/server/db/__tests__/chat-tables.test.js +82 -0
- package/dist/server/db/__tests__/chat-tables.test.js.map +1 -0
- package/dist/server/db/connection.d.ts +4 -0
- package/dist/server/db/connection.d.ts.map +1 -0
- package/dist/server/db/connection.js +21 -0
- package/dist/server/db/connection.js.map +1 -0
- package/dist/server/db/migrations.d.ts +4 -0
- package/dist/server/db/migrations.d.ts.map +1 -0
- package/dist/server/db/migrations.js +17 -0
- package/dist/server/db/migrations.js.map +1 -0
- package/dist/server/db/schema.d.ts +2 -0
- package/dist/server/db/schema.d.ts.map +1 -0
- package/dist/server/db/schema.js +44 -0
- package/dist/server/db/schema.js.map +1 -0
- package/dist/server/index.d.ts +2 -0
- package/dist/server/index.d.ts.map +1 -0
- package/dist/server/index.js +18 -0
- package/dist/server/index.js.map +1 -0
- package/dist/server/routes/__tests__/conversations.test.d.ts +2 -0
- package/dist/server/routes/__tests__/conversations.test.d.ts.map +1 -0
- package/dist/server/routes/__tests__/conversations.test.js +94 -0
- package/dist/server/routes/__tests__/conversations.test.js.map +1 -0
- package/dist/server/routes/conversations.d.ts +8 -0
- package/dist/server/routes/conversations.d.ts.map +1 -0
- package/dist/server/routes/conversations.js +25 -0
- package/dist/server/routes/conversations.js.map +1 -0
- package/dist/server/watcher/file-watcher.d.ts +12 -0
- package/dist/server/watcher/file-watcher.d.ts.map +1 -0
- package/dist/server/watcher/file-watcher.js +68 -0
- package/dist/server/watcher/file-watcher.js.map +1 -0
- package/dist/server/watcher/state-parser.d.ts +77 -0
- package/dist/server/watcher/state-parser.d.ts.map +1 -0
- package/dist/server/watcher/state-parser.js +74 -0
- package/dist/server/watcher/state-parser.js.map +1 -0
- package/dist/server/ws/ws-auth.d.ts +4 -0
- package/dist/server/ws/ws-auth.d.ts.map +1 -0
- package/dist/server/ws/ws-auth.js +42 -0
- package/dist/server/ws/ws-auth.js.map +1 -0
- package/dist/server/ws/ws-handler.d.ts +11 -0
- package/dist/server/ws/ws-handler.d.ts.map +1 -0
- package/dist/server/ws/ws-handler.js +107 -0
- package/dist/server/ws/ws-handler.js.map +1 -0
- package/dist/server/ws/ws-rooms.d.ts +12 -0
- package/dist/server/ws/ws-rooms.d.ts.map +1 -0
- package/dist/server/ws/ws-rooms.js +52 -0
- package/dist/server/ws/ws-rooms.js.map +1 -0
- package/dist/utils/config.d.ts +15 -0
- package/dist/utils/config.js +23 -0
- 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":"AA8CA,MAAM,UAAU,SAAS,CACvB,SAAiB,EACjB,IAAuB,EACvB,IAAgB;IAEhB,MAAM,MAAM,GAAiB,EAAE,CAAC;IAEhC,sCAAsC;IACtC,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,aAAa,EAAE,SAAS,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;QAC7D,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,sBAAsB;IACtB,MAAM,YAAY,GAChB,IAAI,CAAC,MAAM,KAAK,IAAI,CAAC,MAAM;QAC3B,IAAI,CAAC,IAAI,CAAC,OAAO,KAAK,IAAI,CAAC,IAAI,CAAC,OAAO;QACvC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAEhE,IAAI,YAAY,EAAE,CAAC;QACjB,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,aAAa,EAAE,SAAS,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IAC/D,CAAC;IAED,uEAAuE;IACvE,IAAI,IAAI,CAAC,MAAM,KAAK,YAAY,IAAI,IAAI,CAAC,MAAM,KAAK,YAAY,EAAE,CAAC;QACjE,MAAM,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,YAAY,CAAC,CAAC;QAC3E,IAAI,eAAe,EAAE,CAAC;YACpB,MAAM,CAAC,IAAI,CAAC;gBACV,IAAI,EAAE,kBAAkB;gBACxB,SAAS;gBACT,OAAO,EAAE,eAAe,CAAC,EAAE;gBAC3B,SAAS,EAAE,eAAe,CAAC,IAAI;gBAC/B,QAAQ,EAAE,eAAe,CAAC,OAAO,IAAI,EAAE;aACxC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,sBAAsB;IACtB,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;IAEhE,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;QAChC,MAAM,SAAS,GAAG,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAE7C,IAAI,CAAC,SAAS,IAAI,SAAS,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM,EAAE,CAAC;YACpD,MAAM,CAAC,IAAI,CAAC;gBACV,IAAI,EAAE,cAAc;gBACpB,OAAO,EAAE,KAAK,CAAC,EAAE;gBACjB,SAAS,EAAE,KAAK,CAAC,IAAI;gBACrB,MAAM,EAAE,KAAK,CAAC,MAAM;aACrB,CAAC,CAAC;QACL,CAAC;QAED,wBAAwB;QACxB,IACE,SAAS;YACT,KAAK,CAAC,SAAS,KAAK,SAAS;YAC7B,SAAS,CAAC,SAAS,KAAK,SAAS;YACjC,SAAS,CAAC,SAAS,KAAK,KAAK,CAAC,SAAS,EACvC,CAAC;YACD,MAAM,CAAC,IAAI,CAAC;gBACV,IAAI,EAAE,qBAAqB;gBAC3B,SAAS;gBACT,OAAO,EAAE,KAAK,CAAC,EAAE;gBACjB,SAAS,EAAE,KAAK,CAAC,IAAI;gBACrB,SAAS,EAAE,SAAS,CAAC,SAAS;gBAC9B,SAAS,EAAE,SAAS,CAAC,SAAS,IAAI,EAAE;aACrC,CAAC,CAAC;YACH,MAAM,CAAC,IAAI,CAAC;gBACV,IAAI,EAAE,kBAAkB;gBACxB,SAAS;gBACT,OAAO,EAAE,KAAK,CAAC,EAAE;gBACjB,SAAS,EAAE,KAAK,CAAC,IAAI;gBACrB,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,SAAS,EAAE,KAAK,CAAC,SAAS,IAAI,EAAE;aACjC,CAAC,CAAC;QACL,CAAC;QAED,0DAA0D;QAC1D,IAAI,KAAK,CAAC,OAAO,IAAI,CAAC,CAAC,SAAS,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,EAAE,CAAC;YACxD,MAAM,CAAC,IAAI,CAAC;gBACV,IAAI,EAAE,eAAe;gBACrB,OAAO,EAAE,KAAK,CAAC,EAAE;gBACjB,SAAS,EAAE,KAAK,CAAC,IAAI;gBACrB,OAAO,EAAE,KAAK,CAAC,OAAO;aACvB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC"}
|
|
@@ -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"}
|