expxagents 0.12.2 → 0.13.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 (72) hide show
  1. package/dist/dashboard/assets/{BufferResource-DTKCcSRK.js → BufferResource-BMmxDv0X.js} +1 -1
  2. package/dist/dashboard/assets/{CanvasRenderer-RC-R9HHB.js → CanvasRenderer-C8lgY4Ra.js} +1 -1
  3. package/dist/dashboard/assets/{JarvisView-BxEaO5CE.js → JarvisView-CRecMJxY.js} +1 -1
  4. package/dist/dashboard/assets/{RenderTargetSystem-BNs6mZy_.js → RenderTargetSystem-DrrJs7jI.js} +1 -1
  5. package/dist/dashboard/assets/{WebGLRenderer-DZqhYwUo.js → WebGLRenderer-DzCm1r8W.js} +1 -1
  6. package/dist/dashboard/assets/{WebGPURenderer-DKvNGGnd.js → WebGPURenderer-DFAfTUvS.js} +1 -1
  7. package/dist/dashboard/assets/{browserAll-DpQDqd5Z.js → browserAll-D5H10uru.js} +1 -1
  8. package/dist/dashboard/assets/index-DHu3wE8e.css +1 -0
  9. package/dist/dashboard/assets/index-Dbrb96eX.js +400 -0
  10. package/dist/dashboard/assets/{webworkerAll-DKgN-y6e.js → webworkerAll-BuXYXspl.js} +1 -1
  11. package/dist/dashboard/index.html +2 -2
  12. package/dist/data/opensquad.db-shm +0 -0
  13. package/dist/data/opensquad.db-wal +0 -0
  14. package/dist/server/api/__tests__/graph-routes.test.d.ts +2 -0
  15. package/dist/server/api/__tests__/graph-routes.test.d.ts.map +1 -0
  16. package/dist/server/api/__tests__/graph-routes.test.js +105 -0
  17. package/dist/server/api/__tests__/graph-routes.test.js.map +1 -0
  18. package/dist/server/api/__tests__/integration-routes.test.d.ts +2 -0
  19. package/dist/server/api/__tests__/integration-routes.test.d.ts.map +1 -0
  20. package/dist/server/api/__tests__/integration-routes.test.js +243 -0
  21. package/dist/server/api/__tests__/integration-routes.test.js.map +1 -0
  22. package/dist/server/api/__tests__/team-routes.test.d.ts +2 -0
  23. package/dist/server/api/__tests__/team-routes.test.d.ts.map +1 -0
  24. package/dist/server/api/__tests__/team-routes.test.js +116 -0
  25. package/dist/server/api/__tests__/team-routes.test.js.map +1 -0
  26. package/dist/server/api/graph-routes.d.ts +24 -0
  27. package/dist/server/api/graph-routes.d.ts.map +1 -0
  28. package/dist/server/api/graph-routes.js +189 -0
  29. package/dist/server/api/graph-routes.js.map +1 -0
  30. package/dist/server/api/integration-routes.d.ts +23 -0
  31. package/dist/server/api/integration-routes.d.ts.map +1 -0
  32. package/dist/server/api/integration-routes.js +321 -0
  33. package/dist/server/api/integration-routes.js.map +1 -0
  34. package/dist/server/api/squads-routes.d.ts.map +1 -1
  35. package/dist/server/api/squads-routes.js +12 -1
  36. package/dist/server/api/squads-routes.js.map +1 -1
  37. package/dist/server/api/team-routes.d.ts +8 -0
  38. package/dist/server/api/team-routes.d.ts.map +1 -0
  39. package/dist/server/api/team-routes.js +55 -0
  40. package/dist/server/api/team-routes.js.map +1 -0
  41. package/dist/server/app.d.ts.map +1 -1
  42. package/dist/server/app.js +6 -0
  43. package/dist/server/app.js.map +1 -1
  44. package/dist/server/bridge/__tests__/claude-bridge.test.js +79 -11
  45. package/dist/server/bridge/__tests__/claude-bridge.test.js.map +1 -1
  46. package/dist/server/bridge/chat-handler.d.ts +6 -1
  47. package/dist/server/bridge/chat-handler.d.ts.map +1 -1
  48. package/dist/server/bridge/chat-handler.js +106 -5
  49. package/dist/server/bridge/chat-handler.js.map +1 -1
  50. package/dist/server/bridge/claude-bridge.d.ts +6 -0
  51. package/dist/server/bridge/claude-bridge.d.ts.map +1 -1
  52. package/dist/server/bridge/claude-bridge.js +90 -10
  53. package/dist/server/bridge/claude-bridge.js.map +1 -1
  54. package/dist/server/bridge/conversation.d.ts +1 -0
  55. package/dist/server/bridge/conversation.d.ts.map +1 -1
  56. package/dist/server/bridge/conversation.js +5 -0
  57. package/dist/server/bridge/conversation.js.map +1 -1
  58. package/dist/server/db/schema.d.ts +1 -1
  59. package/dist/server/db/schema.d.ts.map +1 -1
  60. package/dist/server/db/schema.js +18 -0
  61. package/dist/server/db/schema.js.map +1 -1
  62. package/dist/server/routes/__tests__/conversations.test.js +18 -1
  63. package/dist/server/routes/__tests__/conversations.test.js.map +1 -1
  64. package/dist/server/routes/conversations.d.ts.map +1 -1
  65. package/dist/server/routes/conversations.js +12 -1
  66. package/dist/server/routes/conversations.js.map +1 -1
  67. package/dist/server/ws/ws-handler.d.ts.map +1 -1
  68. package/dist/server/ws/ws-handler.js +17 -3
  69. package/dist/server/ws/ws-handler.js.map +1 -1
  70. package/package.json +1 -1
  71. package/dist/dashboard/assets/index-B-8_BLE5.css +0 -1
  72. package/dist/dashboard/assets/index-C_HJOTiO.js +0 -344
@@ -1,4 +1,4 @@
1
- import{a0 as G,a2 as I,a3 as B,k as _,M as k,V as O,L as A,a8 as m,T as v,ar as C,R as E,w as z,a7 as U,t as w}from"./index-C_HJOTiO.js";var M=`in vec2 aPosition;
1
+ import{a0 as G,a2 as I,a3 as B,k as _,M as k,V as O,L as A,a8 as m,T as v,ar as C,R as E,w as z,a7 as U,t as w}from"./index-Dbrb96eX.js";var M=`in vec2 aPosition;
2
2
  out vec2 vTextureCoord;
3
3
 
4
4
  uniform vec4 uInputSize;
@@ -4,8 +4,8 @@
4
4
  <meta charset="UTF-8" />
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
6
  <title>ExpxAgents — Mission Control</title>
7
- <script type="module" crossorigin src="/assets/index-C_HJOTiO.js"></script>
8
- <link rel="stylesheet" crossorigin href="/assets/index-B-8_BLE5.css">
7
+ <script type="module" crossorigin src="/assets/index-Dbrb96eX.js"></script>
8
+ <link rel="stylesheet" crossorigin href="/assets/index-DHu3wE8e.css">
9
9
  </head>
10
10
  <body>
11
11
  <div id="root"></div>
Binary file
Binary file
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=graph-routes.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graph-routes.test.d.ts","sourceRoot":"","sources":["../../../src/api/__tests__/graph-routes.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,105 @@
1
+ import { describe, it, expect, beforeAll, afterAll } from 'vitest';
2
+ import { buildApp } from '../../app.js';
3
+ describe('graph routes', () => {
4
+ let app;
5
+ let cookie;
6
+ beforeAll(async () => {
7
+ app = await buildApp({ config: { jwtSecret: 'test-secret-graph' } });
8
+ await app.ready();
9
+ const loginRes = await app.inject({
10
+ method: 'POST',
11
+ url: '/api/auth/login',
12
+ payload: { username: 'admin', password: 'admin' },
13
+ });
14
+ const cookies = loginRes.cookies;
15
+ const access = cookies.find((c) => c.name === 'access_token');
16
+ const refresh = cookies.find((c) => c.name === 'refresh_token');
17
+ cookie = `access_token=${access.value}; refresh_token=${refresh.value}`;
18
+ });
19
+ afterAll(async () => {
20
+ await app.close();
21
+ });
22
+ it('GET /api/graph — returns graph with nodes and edges', async () => {
23
+ const res = await app.inject({
24
+ method: 'GET',
25
+ url: '/api/graph',
26
+ headers: { cookie },
27
+ });
28
+ expect(res.statusCode).toBe(200);
29
+ const body = res.json();
30
+ expect(body.nodes).toBeInstanceOf(Array);
31
+ expect(body.edges).toBeInstanceOf(Array);
32
+ // Should have at least the sample-squad
33
+ expect(body.nodes.length).toBeGreaterThanOrEqual(1);
34
+ });
35
+ it('GET /api/graph — includes squad nodes', async () => {
36
+ const res = await app.inject({
37
+ method: 'GET',
38
+ url: '/api/graph',
39
+ headers: { cookie },
40
+ });
41
+ const body = res.json();
42
+ const squads = body.nodes.filter((n) => n.type === 'squad');
43
+ expect(squads.length).toBeGreaterThanOrEqual(1);
44
+ const sampleSquad = squads.find((n) => n.id === 'squad:sample-squad');
45
+ expect(sampleSquad).toBeDefined();
46
+ expect(sampleSquad.label).toBe('Sample Squad');
47
+ });
48
+ it('GET /api/graph — includes agent nodes with squad ref', async () => {
49
+ const res = await app.inject({
50
+ method: 'GET',
51
+ url: '/api/graph',
52
+ headers: { cookie },
53
+ });
54
+ const body = res.json();
55
+ const agents = body.nodes.filter((n) => n.type === 'agent');
56
+ expect(agents.length).toBeGreaterThanOrEqual(1);
57
+ const researcher = agents.find((n) => n.id === 'agent:sample-squad/researcher');
58
+ expect(researcher).toBeDefined();
59
+ expect(researcher.squad).toBe('sample-squad');
60
+ });
61
+ it('GET /api/graph — includes contains edges', async () => {
62
+ const res = await app.inject({
63
+ method: 'GET',
64
+ url: '/api/graph',
65
+ headers: { cookie },
66
+ });
67
+ const body = res.json();
68
+ const containsEdges = body.edges.filter((e) => e.type === 'contains');
69
+ expect(containsEdges.length).toBeGreaterThanOrEqual(1);
70
+ const squadToAgent = containsEdges.find((e) => e.source === 'squad:sample-squad' && e.target === 'agent:sample-squad/researcher');
71
+ expect(squadToAgent).toBeDefined();
72
+ });
73
+ it('GET /api/graph — includes pipeline edges for deliverFrom', async () => {
74
+ const res = await app.inject({
75
+ method: 'GET',
76
+ url: '/api/graph',
77
+ headers: { cookie },
78
+ });
79
+ const body = res.json();
80
+ const delivers = body.edges.filter((e) => e.type === 'delivers');
81
+ // sample-squad has deliverFrom: researcher in writer step
82
+ const delivery = delivers.find((e) => e.source === 'agent:sample-squad/researcher' && e.target === 'agent:sample-squad/writer');
83
+ expect(delivery).toBeDefined();
84
+ });
85
+ it('GET /api/graph — includes skill nodes', async () => {
86
+ const res = await app.inject({
87
+ method: 'GET',
88
+ url: '/api/graph',
89
+ headers: { cookie },
90
+ });
91
+ const body = res.json();
92
+ const skills = body.nodes.filter((n) => n.type === 'skill');
93
+ expect(skills.length).toBeGreaterThanOrEqual(1);
94
+ const webSearch = skills.find((n) => n.id === 'skill:web_search');
95
+ expect(webSearch).toBeDefined();
96
+ });
97
+ it('rejects unauthenticated requests', async () => {
98
+ const res = await app.inject({
99
+ method: 'GET',
100
+ url: '/api/graph',
101
+ });
102
+ expect(res.statusCode).toBe(401);
103
+ });
104
+ });
105
+ //# sourceMappingURL=graph-routes.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graph-routes.test.js","sourceRoot":"","sources":["../../../src/api/__tests__/graph-routes.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=integration-routes.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"integration-routes.test.d.ts","sourceRoot":"","sources":["../../../src/api/__tests__/integration-routes.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,243 @@
1
+ import { describe, it, expect, beforeAll, afterAll } from 'vitest';
2
+ import { buildApp } from '../../app.js';
3
+ import { parseEnvFile } from '../settings-routes.js';
4
+ describe('integration routes', () => {
5
+ let app;
6
+ let cookie;
7
+ beforeAll(async () => {
8
+ app = await buildApp({ config: { jwtSecret: 'test-secret-integrations' } });
9
+ await app.ready();
10
+ const loginRes = await app.inject({
11
+ method: 'POST',
12
+ url: '/api/auth/login',
13
+ payload: { username: 'admin', password: 'admin' },
14
+ });
15
+ const cookies = loginRes.cookies;
16
+ const access = cookies.find((c) => c.name === 'access_token');
17
+ const refresh = cookies.find((c) => c.name === 'refresh_token');
18
+ cookie = `access_token=${access.value}; refresh_token=${refresh.value}`;
19
+ // Clean table from previous runs
20
+ app.db.exec('DELETE FROM integrations');
21
+ });
22
+ afterAll(async () => {
23
+ // Clean up: remove all test integrations so env keys don't persist
24
+ app.db.exec('DELETE FROM integrations');
25
+ // Clean env keys written by tests
26
+ for (const key of Object.keys(process.env)) {
27
+ if (/^(GITHUB|TELEGRAM|BREVO|DISCORD)_/.test(key)) {
28
+ delete process.env[key];
29
+ }
30
+ }
31
+ // Rewrite .env without test keys
32
+ const envPath = process.env.DOTENV_PATH ?? new URL('../../../../.env', import.meta.url).pathname;
33
+ const entries = parseEnvFile(envPath);
34
+ for (const k of [...entries.keys()]) {
35
+ if (/^(GITHUB|TELEGRAM|BREVO|DISCORD)_/.test(k)) {
36
+ entries.delete(k);
37
+ }
38
+ }
39
+ const { writeEnvFile } = await import('../settings-routes.js');
40
+ writeEnvFile(envPath, entries);
41
+ await app.close();
42
+ });
43
+ it('GET /api/integrations/catalog — returns provider catalog', async () => {
44
+ const res = await app.inject({
45
+ method: 'GET',
46
+ url: '/api/integrations/catalog',
47
+ headers: { cookie },
48
+ });
49
+ expect(res.statusCode).toBe(200);
50
+ const body = res.json();
51
+ expect(body.providers).toBeInstanceOf(Array);
52
+ expect(body.providers.length).toBeGreaterThanOrEqual(30);
53
+ const github = body.providers.find((p) => p.key === 'github');
54
+ expect(github).toBeDefined();
55
+ expect(github.name).toBe('GitHub');
56
+ expect(github.category).toBe('dev');
57
+ expect(github.fields.length).toBeGreaterThanOrEqual(1);
58
+ });
59
+ it('GET /api/integrations — returns empty list initially', async () => {
60
+ const res = await app.inject({
61
+ method: 'GET',
62
+ url: '/api/integrations',
63
+ headers: { cookie },
64
+ });
65
+ expect(res.statusCode).toBe(200);
66
+ const body = res.json();
67
+ expect(body.integrations).toEqual([]);
68
+ });
69
+ it('PATCH /api/integrations/:provider — creates a new integration', async () => {
70
+ const res = await app.inject({
71
+ method: 'PATCH',
72
+ url: '/api/integrations/github',
73
+ headers: { cookie },
74
+ payload: {
75
+ enabled: true,
76
+ config: { token: 'ghp_abc123def456ghi789', org: 'my-org' },
77
+ },
78
+ });
79
+ expect(res.statusCode).toBe(200);
80
+ const body = res.json();
81
+ expect(body.integration.provider).toBe('github');
82
+ expect(body.integration.enabled).toBe(true);
83
+ // Token should be masked
84
+ expect(body.integration.config.token).toContain('••••');
85
+ expect(body.integration.config.token).not.toBe('ghp_abc123def456ghi789');
86
+ // Org is plain text, not masked
87
+ expect(body.integration.config.org).toBe('my-org');
88
+ });
89
+ it('PATCH /api/integrations/:provider — updates existing integration', async () => {
90
+ // First create
91
+ await app.inject({
92
+ method: 'PATCH',
93
+ url: '/api/integrations/telegram',
94
+ headers: { cookie },
95
+ payload: {
96
+ enabled: true,
97
+ config: { bot_token: 'secret-bot-token-123', default_chat_id: '-100123' },
98
+ },
99
+ });
100
+ // Update only chat_id, send masked token back
101
+ const res = await app.inject({
102
+ method: 'PATCH',
103
+ url: '/api/integrations/telegram',
104
+ headers: { cookie },
105
+ payload: {
106
+ config: { bot_token: 'secr••••23', default_chat_id: '-100456' },
107
+ },
108
+ });
109
+ expect(res.statusCode).toBe(200);
110
+ const body = res.json();
111
+ expect(body.integration.config.default_chat_id).toBe('-100456');
112
+ // Token should still be masked (preserved original)
113
+ expect(body.integration.config.bot_token).toContain('••••');
114
+ });
115
+ it('PATCH /api/integrations/:provider — rejects unknown provider', async () => {
116
+ const res = await app.inject({
117
+ method: 'PATCH',
118
+ url: '/api/integrations/unknown_service',
119
+ headers: { cookie },
120
+ payload: { enabled: true, config: {} },
121
+ });
122
+ expect(res.statusCode).toBe(400);
123
+ expect(res.json().error).toContain('Unknown provider');
124
+ });
125
+ it('PATCH /api/integrations/:provider — toggles enabled', async () => {
126
+ await app.inject({
127
+ method: 'PATCH',
128
+ url: '/api/integrations/brevo',
129
+ headers: { cookie },
130
+ payload: { enabled: true, config: { api_key: 'xkeysib-test123' } },
131
+ });
132
+ const res = await app.inject({
133
+ method: 'PATCH',
134
+ url: '/api/integrations/brevo',
135
+ headers: { cookie },
136
+ payload: { enabled: false },
137
+ });
138
+ expect(res.statusCode).toBe(200);
139
+ expect(res.json().integration.enabled).toBe(false);
140
+ });
141
+ it('GET /api/integrations — lists configured integrations', async () => {
142
+ const res = await app.inject({
143
+ method: 'GET',
144
+ url: '/api/integrations',
145
+ headers: { cookie },
146
+ });
147
+ expect(res.statusCode).toBe(200);
148
+ const body = res.json();
149
+ expect(body.integrations.length).toBeGreaterThanOrEqual(1);
150
+ // All secrets should be masked
151
+ for (const integ of body.integrations) {
152
+ expect(integ.provider).toBeDefined();
153
+ expect(typeof integ.enabled).toBe('boolean');
154
+ }
155
+ });
156
+ it('DELETE /api/integrations/:provider — deletes an integration', async () => {
157
+ await app.inject({
158
+ method: 'PATCH',
159
+ url: '/api/integrations/discord',
160
+ headers: { cookie },
161
+ payload: { enabled: true, config: { bot_token: 'tok123' } },
162
+ });
163
+ const res = await app.inject({
164
+ method: 'DELETE',
165
+ url: '/api/integrations/discord',
166
+ headers: { cookie },
167
+ });
168
+ expect(res.statusCode).toBe(200);
169
+ expect(res.json().ok).toBe(true);
170
+ // Verify deleted
171
+ const delRes = await app.inject({
172
+ method: 'DELETE',
173
+ url: '/api/integrations/discord',
174
+ headers: { cookie },
175
+ });
176
+ expect(delRes.statusCode).toBe(404);
177
+ });
178
+ it('PATCH syncs enabled integration config to .env', async () => {
179
+ await app.inject({
180
+ method: 'PATCH',
181
+ url: '/api/integrations/github',
182
+ headers: { cookie },
183
+ payload: {
184
+ enabled: true,
185
+ config: { token: 'ghp_testtoken123', org: 'test-org' },
186
+ },
187
+ });
188
+ // Check process.env was updated
189
+ expect(process.env.GITHUB_TOKEN).toBe('ghp_testtoken123');
190
+ expect(process.env.GITHUB_ORG).toBe('test-org');
191
+ // Check .env file was updated
192
+ const envPath = process.env.DOTENV_PATH ?? new URL('../../../../.env', import.meta.url).pathname;
193
+ const entries = parseEnvFile(envPath);
194
+ expect(entries.get('GITHUB_TOKEN')).toBe('ghp_testtoken123');
195
+ expect(entries.get('GITHUB_ORG')).toBe('test-org');
196
+ });
197
+ it('DELETE removes integration keys from .env', async () => {
198
+ // Create first
199
+ await app.inject({
200
+ method: 'PATCH',
201
+ url: '/api/integrations/discord',
202
+ headers: { cookie },
203
+ payload: { enabled: true, config: { bot_token: 'test-discord-tok' } },
204
+ });
205
+ expect(process.env.DISCORD_BOT_TOKEN).toBe('test-discord-tok');
206
+ // Delete
207
+ await app.inject({
208
+ method: 'DELETE',
209
+ url: '/api/integrations/discord',
210
+ headers: { cookie },
211
+ });
212
+ expect(process.env.DISCORD_BOT_TOKEN).toBeUndefined();
213
+ const envPath = process.env.DOTENV_PATH ?? new URL('../../../../.env', import.meta.url).pathname;
214
+ const entries = parseEnvFile(envPath);
215
+ expect(entries.has('DISCORD_BOT_TOKEN')).toBe(false);
216
+ });
217
+ it('disabled integration removes keys from .env', async () => {
218
+ // Create enabled
219
+ await app.inject({
220
+ method: 'PATCH',
221
+ url: '/api/integrations/brevo',
222
+ headers: { cookie },
223
+ payload: { enabled: true, config: { api_key: 'xkeysib-test' } },
224
+ });
225
+ expect(process.env.BREVO_API_KEY).toBe('xkeysib-test');
226
+ // Disable
227
+ await app.inject({
228
+ method: 'PATCH',
229
+ url: '/api/integrations/brevo',
230
+ headers: { cookie },
231
+ payload: { enabled: false },
232
+ });
233
+ expect(process.env.BREVO_API_KEY).toBeUndefined();
234
+ });
235
+ it('rejects unauthenticated requests', async () => {
236
+ const res = await app.inject({
237
+ method: 'GET',
238
+ url: '/api/integrations',
239
+ });
240
+ expect(res.statusCode).toBe(401);
241
+ });
242
+ });
243
+ //# sourceMappingURL=integration-routes.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"integration-routes.test.js","sourceRoot":"","sources":["../../../src/api/__tests__/integration-routes.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=team-routes.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"team-routes.test.d.ts","sourceRoot":"","sources":["../../../src/api/__tests__/team-routes.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,116 @@
1
+ import { describe, it, expect, beforeAll, afterAll } from 'vitest';
2
+ import { buildApp } from '../../app.js';
3
+ describe('team routes', () => {
4
+ let app;
5
+ let cookie;
6
+ beforeAll(async () => {
7
+ app = await buildApp({ config: { jwtSecret: 'test-secret-team' } });
8
+ await app.ready();
9
+ // Login as default admin
10
+ const loginRes = await app.inject({
11
+ method: 'POST',
12
+ url: '/api/auth/login',
13
+ payload: { username: 'admin', password: 'admin' },
14
+ });
15
+ const cookies = loginRes.cookies;
16
+ const access = cookies.find((c) => c.name === 'access_token');
17
+ const refresh = cookies.find((c) => c.name === 'refresh_token');
18
+ cookie = `access_token=${access.value}; refresh_token=${refresh.value}`;
19
+ // Clean table from previous runs
20
+ app.db.exec('DELETE FROM team_members');
21
+ });
22
+ afterAll(async () => {
23
+ await app.close();
24
+ });
25
+ it('GET /api/team — returns empty list initially', async () => {
26
+ const res = await app.inject({
27
+ method: 'GET',
28
+ url: '/api/team',
29
+ headers: { cookie },
30
+ });
31
+ expect(res.statusCode).toBe(200);
32
+ const body = res.json();
33
+ expect(body.members).toEqual([]);
34
+ });
35
+ it('POST /api/team — creates a team member', async () => {
36
+ const res = await app.inject({
37
+ method: 'POST',
38
+ url: '/api/team',
39
+ headers: { cookie },
40
+ payload: {
41
+ name: 'Maria Silva',
42
+ role: 'Designer',
43
+ email: 'maria@example.com',
44
+ phone: '+55 11 99999-0000',
45
+ },
46
+ });
47
+ expect(res.statusCode).toBe(201);
48
+ const body = res.json();
49
+ expect(body.member.name).toBe('Maria Silva');
50
+ expect(body.member.role).toBe('Designer');
51
+ expect(body.member.email).toBe('maria@example.com');
52
+ expect(body.member.phone).toBe('+55 11 99999-0000');
53
+ expect(body.member.id).toMatch(/^tm-/);
54
+ });
55
+ it('POST /api/team — rejects missing name', async () => {
56
+ const res = await app.inject({
57
+ method: 'POST',
58
+ url: '/api/team',
59
+ headers: { cookie },
60
+ payload: { role: 'Dev' },
61
+ });
62
+ expect(res.statusCode).toBe(400);
63
+ });
64
+ it('PATCH /api/team/:id — updates a team member', async () => {
65
+ // Create first
66
+ const createRes = await app.inject({
67
+ method: 'POST',
68
+ url: '/api/team',
69
+ headers: { cookie },
70
+ payload: { name: 'Joao', role: 'Dev' },
71
+ });
72
+ const { member } = createRes.json();
73
+ const res = await app.inject({
74
+ method: 'PATCH',
75
+ url: `/api/team/${member.id}`,
76
+ headers: { cookie },
77
+ payload: { role: 'Senior Dev', email: 'joao@test.com' },
78
+ });
79
+ expect(res.statusCode).toBe(200);
80
+ const updated = res.json().member;
81
+ expect(updated.name).toBe('Joao');
82
+ expect(updated.role).toBe('Senior Dev');
83
+ expect(updated.email).toBe('joao@test.com');
84
+ });
85
+ it('DELETE /api/team/:id — deletes a team member', async () => {
86
+ const createRes = await app.inject({
87
+ method: 'POST',
88
+ url: '/api/team',
89
+ headers: { cookie },
90
+ payload: { name: 'Temp User', role: 'Intern' },
91
+ });
92
+ const { member } = createRes.json();
93
+ const res = await app.inject({
94
+ method: 'DELETE',
95
+ url: `/api/team/${member.id}`,
96
+ headers: { cookie },
97
+ });
98
+ expect(res.statusCode).toBe(200);
99
+ expect(res.json().ok).toBe(true);
100
+ // Verify deleted
101
+ const delRes = await app.inject({
102
+ method: 'DELETE',
103
+ url: `/api/team/${member.id}`,
104
+ headers: { cookie },
105
+ });
106
+ expect(delRes.statusCode).toBe(404);
107
+ });
108
+ it('rejects unauthenticated requests', async () => {
109
+ const res = await app.inject({
110
+ method: 'GET',
111
+ url: '/api/team',
112
+ });
113
+ expect(res.statusCode).toBe(401);
114
+ });
115
+ });
116
+ //# sourceMappingURL=team-routes.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"team-routes.test.js","sourceRoot":"","sources":["../../../src/api/__tests__/team-routes.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,24 @@
1
+ import type { FastifyInstance, FastifyPluginOptions } from 'fastify';
2
+ interface GraphRoutesOptions extends FastifyPluginOptions {
3
+ squadsDir: string;
4
+ }
5
+ interface GraphNode {
6
+ id: string;
7
+ label: string;
8
+ type: 'squad' | 'agent' | 'memory' | 'skill';
9
+ squad?: string;
10
+ }
11
+ interface GraphEdge {
12
+ source: string;
13
+ target: string;
14
+ type: 'contains' | 'pipeline' | 'delivers' | 'uses_skill' | 'has_memory' | 'shared_skill' | 'shared_memory';
15
+ label?: string;
16
+ }
17
+ interface GraphData {
18
+ nodes: GraphNode[];
19
+ edges: GraphEdge[];
20
+ }
21
+ declare function buildGraph(squadsDir: string): GraphData;
22
+ export declare function graphRoutes(app: FastifyInstance, opts: GraphRoutesOptions): Promise<void>;
23
+ export { buildGraph };
24
+ //# sourceMappingURL=graph-routes.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graph-routes.d.ts","sourceRoot":"","sources":["../../src/api/graph-routes.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,oBAAoB,EAAE,MAAM,SAAS,CAAC;AAKrE,UAAU,kBAAmB,SAAQ,oBAAoB;IACvD,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,UAAU,SAAS;IACjB,EAAE,EAAE,MAAM,CAAC;IACX,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,OAAO,GAAG,OAAO,GAAG,QAAQ,GAAG,OAAO,CAAC;IAC7C,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,UAAU,SAAS;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,EAAE,MAAM,CAAC;IACf,IAAI,EAAE,UAAU,GAAG,UAAU,GAAG,UAAU,GAAG,YAAY,GAAG,YAAY,GAAG,cAAc,GAAG,eAAe,CAAC;IAC5G,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,UAAU,SAAS;IACjB,KAAK,EAAE,SAAS,EAAE,CAAC;IACnB,KAAK,EAAE,SAAS,EAAE,CAAC;CACpB;AAsBD,iBAAS,UAAU,CAAC,SAAS,EAAE,MAAM,GAAG,SAAS,CAqKhD;AAED,wBAAsB,WAAW,CAC/B,GAAG,EAAE,eAAe,EACpB,IAAI,EAAE,kBAAkB,GACvB,OAAO,CAAC,IAAI,CAAC,CAMf;AAED,OAAO,EAAE,UAAU,EAAE,CAAC"}