heyio 1.13.0 → 3.0.0
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/api/middleware/auth.d.ts +14 -0
- package/dist/api/middleware/auth.d.ts.map +1 -0
- package/dist/api/middleware/auth.js +66 -0
- package/dist/api/middleware/auth.js.map +1 -0
- package/dist/api/notifications.d.ts +14 -0
- package/dist/api/notifications.d.ts.map +1 -0
- package/dist/api/notifications.js +112 -0
- package/dist/api/notifications.js.map +1 -0
- package/dist/api/routes/activity.d.ts +3 -0
- package/dist/api/routes/activity.d.ts.map +1 -0
- package/dist/api/routes/activity.js +28 -0
- package/dist/api/routes/activity.js.map +1 -0
- package/dist/api/routes/attachments.d.ts +3 -0
- package/dist/api/routes/attachments.d.ts.map +1 -0
- package/dist/api/routes/attachments.js +83 -0
- package/dist/api/routes/attachments.js.map +1 -0
- package/dist/api/routes/config.d.ts +3 -0
- package/dist/api/routes/config.d.ts.map +1 -0
- package/dist/api/routes/config.js +106 -0
- package/dist/api/routes/config.js.map +1 -0
- package/dist/api/routes/conversations.d.ts +3 -0
- package/dist/api/routes/conversations.d.ts.map +1 -0
- package/dist/api/routes/conversations.js +69 -0
- package/dist/api/routes/conversations.js.map +1 -0
- package/dist/api/routes/health.d.ts +3 -0
- package/dist/api/routes/health.d.ts.map +1 -0
- package/dist/api/routes/health.js +16 -0
- package/dist/api/routes/health.js.map +1 -0
- package/dist/api/routes/inbox.d.ts +3 -0
- package/dist/api/routes/inbox.d.ts.map +1 -0
- package/dist/api/routes/inbox.js +88 -0
- package/dist/api/routes/inbox.js.map +1 -0
- package/dist/api/routes/schedules.d.ts +3 -0
- package/dist/api/routes/schedules.d.ts.map +1 -0
- package/dist/api/routes/schedules.js +96 -0
- package/dist/api/routes/schedules.js.map +1 -0
- package/dist/api/routes/skills.d.ts +2 -0
- package/dist/api/routes/skills.d.ts.map +1 -0
- package/dist/api/routes/skills.js +85 -0
- package/dist/api/routes/skills.js.map +1 -0
- package/dist/api/routes/squads.d.ts +3 -0
- package/dist/api/routes/squads.d.ts.map +1 -0
- package/dist/api/routes/squads.js +129 -0
- package/dist/api/routes/squads.js.map +1 -0
- package/dist/api/routes/usage.d.ts +3 -0
- package/dist/api/routes/usage.d.ts.map +1 -0
- package/dist/api/routes/usage.js +55 -0
- package/dist/api/routes/usage.js.map +1 -0
- package/dist/api/routes/wiki.d.ts +2 -0
- package/dist/api/routes/wiki.d.ts.map +1 -0
- package/dist/api/routes/wiki.js +43 -0
- package/dist/api/routes/wiki.js.map +1 -0
- package/dist/api/server.d.ts +7 -0
- package/dist/api/server.d.ts.map +1 -0
- package/dist/api/server.js +136 -634
- package/dist/api/server.js.map +1 -0
- package/dist/config.d.ts +3 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +2 -91
- package/dist/config.js.map +1 -0
- package/dist/copilot/client.d.ts +5 -0
- package/dist/copilot/client.d.ts.map +1 -0
- package/dist/copilot/client.js +19 -11
- package/dist/copilot/client.js.map +1 -0
- package/dist/copilot/health-monitor.d.ts +14 -0
- package/dist/copilot/health-monitor.d.ts.map +1 -0
- package/dist/copilot/health-monitor.js +70 -0
- package/dist/copilot/health-monitor.js.map +1 -0
- package/dist/copilot/orchestrator.d.ts +5 -0
- package/dist/copilot/orchestrator.d.ts.map +1 -0
- package/dist/copilot/orchestrator.js +127 -123
- package/dist/copilot/orchestrator.js.map +1 -0
- package/dist/copilot/tools.d.ts +49 -0
- package/dist/copilot/tools.d.ts.map +1 -0
- package/dist/copilot/tools.js +545 -321
- package/dist/copilot/tools.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +82 -26
- package/dist/index.js.map +1 -0
- package/dist/logging/logger.d.ts +6 -0
- package/dist/logging/logger.d.ts.map +1 -0
- package/dist/logging/logger.js +21 -0
- package/dist/logging/logger.js.map +1 -0
- package/dist/models/index.d.ts +6 -0
- package/dist/models/index.d.ts.map +1 -0
- package/dist/models/index.js +4 -0
- package/dist/models/index.js.map +1 -0
- package/dist/models/pricing.d.ts +25 -0
- package/dist/models/pricing.d.ts.map +1 -0
- package/dist/models/pricing.js +96 -0
- package/dist/models/pricing.js.map +1 -0
- package/dist/models/registry.d.ts +34 -0
- package/dist/models/registry.d.ts.map +1 -0
- package/dist/models/registry.js +109 -0
- package/dist/models/registry.js.map +1 -0
- package/dist/models/token-tracker.d.ts +40 -0
- package/dist/models/token-tracker.d.ts.map +1 -0
- package/dist/models/token-tracker.js +102 -0
- package/dist/models/token-tracker.js.map +1 -0
- package/dist/scheduler/engine.d.ts +9 -0
- package/dist/scheduler/engine.d.ts.map +1 -0
- package/dist/scheduler/engine.js +127 -0
- package/dist/scheduler/engine.js.map +1 -0
- package/dist/skills/index.d.ts +3 -0
- package/dist/skills/index.d.ts.map +1 -0
- package/dist/skills/index.js +2 -0
- package/dist/skills/index.js.map +1 -0
- package/dist/skills/store.d.ts +52 -0
- package/dist/skills/store.d.ts.map +1 -0
- package/dist/skills/store.js +148 -0
- package/dist/skills/store.js.map +1 -0
- package/dist/squad/agent.d.ts +46 -0
- package/dist/squad/agent.d.ts.map +1 -0
- package/dist/squad/agent.js +261 -0
- package/dist/squad/agent.js.map +1 -0
- package/dist/squad/autonomy.d.ts +16 -0
- package/dist/squad/autonomy.d.ts.map +1 -0
- package/dist/squad/autonomy.js +63 -0
- package/dist/squad/autonomy.js.map +1 -0
- package/dist/squad/event-bus.d.ts +22 -0
- package/dist/squad/event-bus.d.ts.map +1 -0
- package/dist/squad/event-bus.js +56 -0
- package/dist/squad/event-bus.js.map +1 -0
- package/dist/squad/execution/index.d.ts +12 -0
- package/dist/squad/execution/index.d.ts.map +1 -0
- package/dist/squad/execution/index.js +7 -0
- package/dist/squad/execution/index.js.map +1 -0
- package/dist/squad/execution/instance.d.ts +40 -0
- package/dist/squad/execution/instance.d.ts.map +1 -0
- package/dist/squad/execution/instance.js +138 -0
- package/dist/squad/execution/instance.js.map +1 -0
- package/dist/squad/execution/meeting.d.ts +25 -0
- package/dist/squad/execution/meeting.d.ts.map +1 -0
- package/dist/squad/execution/meeting.js +140 -0
- package/dist/squad/execution/meeting.js.map +1 -0
- package/dist/squad/execution/pr.d.ts +15 -0
- package/dist/squad/execution/pr.d.ts.map +1 -0
- package/dist/squad/execution/pr.js +93 -0
- package/dist/squad/execution/pr.js.map +1 -0
- package/dist/squad/execution/runner.d.ts +22 -0
- package/dist/squad/execution/runner.d.ts.map +1 -0
- package/dist/squad/execution/runner.js +68 -0
- package/dist/squad/execution/runner.js.map +1 -0
- package/dist/squad/execution/tasks.d.ts +11 -0
- package/dist/squad/execution/tasks.d.ts.map +1 -0
- package/dist/squad/execution/tasks.js +85 -0
- package/dist/squad/execution/tasks.js.map +1 -0
- package/dist/squad/execution/worktree.d.ts +26 -0
- package/dist/squad/execution/worktree.d.ts.map +1 -0
- package/dist/squad/execution/worktree.js +111 -0
- package/dist/squad/execution/worktree.js.map +1 -0
- package/dist/squad/hiring.d.ts +32 -0
- package/dist/squad/hiring.d.ts.map +1 -0
- package/dist/squad/hiring.js +200 -0
- package/dist/squad/hiring.js.map +1 -0
- package/dist/squad/index.d.ts +8 -0
- package/dist/squad/index.d.ts.map +1 -0
- package/dist/squad/index.js +6 -0
- package/dist/squad/index.js.map +1 -0
- package/dist/squad/manager.d.ts +48 -0
- package/dist/squad/manager.d.ts.map +1 -0
- package/dist/squad/manager.js +274 -0
- package/dist/squad/manager.js.map +1 -0
- package/dist/squad/name-generator.d.ts +16 -0
- package/dist/squad/name-generator.d.ts.map +1 -0
- package/dist/squad/name-generator.js +113 -0
- package/dist/squad/name-generator.js.map +1 -0
- package/dist/squad/roles/templates.d.ts +5 -0
- package/dist/squad/roles/templates.d.ts.map +1 -0
- package/dist/squad/roles/templates.js +102 -0
- package/dist/squad/roles/templates.js.map +1 -0
- package/dist/squad/skill-parser.d.ts +36 -0
- package/dist/squad/skill-parser.d.ts.map +1 -0
- package/dist/squad/skill-parser.js +83 -0
- package/dist/squad/skill-parser.js.map +1 -0
- package/dist/squad/source-resolver.d.ts +20 -0
- package/dist/squad/source-resolver.d.ts.map +1 -0
- package/dist/squad/source-resolver.js +52 -0
- package/dist/squad/source-resolver.js.map +1 -0
- package/dist/store/activity.d.ts +43 -0
- package/dist/store/activity.d.ts.map +1 -0
- package/dist/store/activity.js +131 -0
- package/dist/store/activity.js.map +1 -0
- package/dist/store/db.d.ts +5 -0
- package/dist/store/db.d.ts.map +1 -0
- package/dist/store/db.js +209 -248
- package/dist/store/db.js.map +1 -0
- package/dist/store/inbox.d.ts +53 -0
- package/dist/store/inbox.d.ts.map +1 -0
- package/dist/store/inbox.js +151 -0
- package/dist/store/inbox.js.map +1 -0
- package/dist/store/schedules.d.ts +53 -0
- package/dist/store/schedules.d.ts.map +1 -0
- package/dist/store/schedules.js +149 -54
- package/dist/store/schedules.js.map +1 -0
- package/dist/wiki/index.d.ts +3 -0
- package/dist/wiki/index.d.ts.map +1 -0
- package/dist/wiki/index.js +2 -0
- package/dist/wiki/index.js.map +1 -0
- package/dist/wiki/store.d.ts +49 -0
- package/dist/wiki/store.d.ts.map +1 -0
- package/dist/wiki/store.js +115 -0
- package/dist/wiki/store.js.map +1 -0
- package/package.json +52 -59
- package/src/api/middleware/auth.ts +76 -0
- package/src/api/notifications.ts +122 -0
- package/src/api/routes/activity.ts +29 -0
- package/src/api/routes/attachments.ts +93 -0
- package/src/api/routes/config.ts +115 -0
- package/src/api/routes/conversations.ts +87 -0
- package/src/api/routes/health.ts +18 -0
- package/src/api/routes/inbox.ts +98 -0
- package/src/api/routes/schedules.ts +121 -0
- package/src/api/routes/skills.ts +105 -0
- package/src/api/routes/squads.ts +145 -0
- package/src/api/routes/usage.ts +57 -0
- package/src/api/routes/wiki.ts +49 -0
- package/src/api/server.ts +186 -0
- package/src/config.ts +3 -0
- package/src/copilot/client.ts +42 -0
- package/src/copilot/health-monitor.ts +85 -0
- package/src/copilot/orchestrator.ts +222 -0
- package/src/copilot/tools.ts +707 -0
- package/src/index.ts +112 -0
- package/src/logging/logger.ts +26 -0
- package/src/models/index.ts +11 -0
- package/src/models/pricing.ts +121 -0
- package/src/models/registry.ts +131 -0
- package/src/models/token-tracker.ts +151 -0
- package/src/scheduler/engine.ts +146 -0
- package/src/skills/index.ts +13 -0
- package/src/skills/store.ts +188 -0
- package/src/squad/agent.ts +326 -0
- package/src/squad/autonomy.ts +78 -0
- package/src/squad/event-bus.ts +71 -0
- package/src/squad/execution/index.ts +17 -0
- package/src/squad/execution/instance.ts +186 -0
- package/src/squad/execution/meeting.ts +191 -0
- package/src/squad/execution/pr.ts +127 -0
- package/src/squad/execution/runner.ts +97 -0
- package/src/squad/execution/tasks.ts +111 -0
- package/src/squad/execution/worktree.ts +138 -0
- package/src/squad/hiring.ts +222 -0
- package/src/squad/index.ts +17 -0
- package/src/squad/manager.ts +337 -0
- package/src/squad/name-generator.ts +135 -0
- package/src/squad/roles/templates.ts +104 -0
- package/src/squad/skill-parser.ts +120 -0
- package/src/squad/source-resolver.ts +57 -0
- package/src/store/activity.ts +176 -0
- package/src/store/db.ts +237 -0
- package/src/store/inbox.ts +199 -0
- package/src/store/schedules.ts +199 -0
- package/src/wiki/index.ts +12 -0
- package/src/wiki/store.ts +139 -0
- package/tsconfig.json +9 -0
- package/LICENSE +0 -21
- package/README.md +0 -333
- package/dist/api/auth.js +0 -46
- package/dist/chat/attachments.js +0 -112
- package/dist/copilot/agents.js +0 -309
- package/dist/copilot/ceremonies.js +0 -174
- package/dist/copilot/gh-token.js +0 -64
- package/dist/copilot/io-scheduler.js +0 -79
- package/dist/copilot/model-router.js +0 -114
- package/dist/copilot/scheduler.js +0 -88
- package/dist/copilot/skills.js +0 -252
- package/dist/copilot/specialist-runner.js +0 -191
- package/dist/copilot/squad-tools.js +0 -258
- package/dist/copilot/system-message.js +0 -86
- package/dist/copilot/token-tracker.js +0 -98
- package/dist/copilot/trigger-schedule.js +0 -33
- package/dist/daemon.js +0 -67
- package/dist/logging.js +0 -27
- package/dist/mcp/config.js +0 -29
- package/dist/mcp/index.js +0 -3
- package/dist/mcp/registry.js +0 -42
- package/dist/notify.js +0 -25
- package/dist/paths.js +0 -17
- package/dist/setup.js +0 -35
- package/dist/store/agent-events.js +0 -19
- package/dist/store/audit-log.js +0 -71
- package/dist/store/conversations.js +0 -164
- package/dist/store/feed.js +0 -44
- package/dist/store/instances.js +0 -75
- package/dist/store/squad-colors.js +0 -23
- package/dist/store/squads.js +0 -60
- package/dist/store/tasks.js +0 -78
- package/dist/store/token-usage.js +0 -94
- package/dist/telegram/bot.js +0 -41
- package/dist/telegram/handlers.js +0 -42
- package/dist/watchdog.js +0 -37
- package/dist/wiki/backlinks.js +0 -51
- package/dist/wiki/fs.js +0 -108
- package/dist/wiki/search.js +0 -47
- package/web-dist/assets/AuditLogView-BzfjNXBT.js +0 -6
- package/web-dist/assets/ChatView-BdMukPKG.js +0 -1
- package/web-dist/assets/FeedView-BfPIabGr.js +0 -6
- package/web-dist/assets/HistoryView-BmEEk3Rs.js +0 -1
- package/web-dist/assets/LoginView-D7LrkeX7.js +0 -1
- package/web-dist/assets/McpView-BAP_ah3T.js +0 -1
- package/web-dist/assets/SchedulesView-CAtsUPCZ.js +0 -6
- package/web-dist/assets/SettingsView-BovjWZDa.js +0 -1
- package/web-dist/assets/SkillsView-DQSMM5LN.js +0 -16
- package/web-dist/assets/SquadDetailView-DBscu0m2.js +0 -26
- package/web-dist/assets/SquadHealthView-D686BuQo.js +0 -11
- package/web-dist/assets/SquadsView-AzMht2NJ.js +0 -6
- package/web-dist/assets/ToggleSwitch.vue_vue_type_script_setup_true_lang-DtShZAjW.js +0 -1
- package/web-dist/assets/UsageView-DiVn97aI.js +0 -16
- package/web-dist/assets/WikiView-Cn7KipkZ.js +0 -26
- package/web-dist/assets/api-D4mHJ3u0.js +0 -1
- package/web-dist/assets/arrow-left-D_qUNUWW.js +0 -6
- package/web-dist/assets/git-branch-DOM-orcl.js +0 -6
- package/web-dist/assets/index-Bo83B1LR.css +0 -1
- package/web-dist/assets/index-ELvnkQjd.js +0 -273
- package/web-dist/assets/pencil-CFsi7ufI.js +0 -6
- package/web-dist/assets/plus-BAzlGFd_.js +0 -6
- package/web-dist/assets/save-BmgCYJ1g.js +0 -6
- package/web-dist/assets/search-CS9zSIeW.js +0 -6
- package/web-dist/assets/squad-colors-B8B_Y-lz.js +0 -1
- package/web-dist/assets/trash-2-DLveUEsd.js +0 -6
- package/web-dist/assets/triangle-alert-lj4I30rL.js +0 -6
- package/web-dist/assets/x-CjXR97Fa.js +0 -6
- package/web-dist/favicon.svg +0 -10
- package/web-dist/index.html +0 -14
- package/web-dist/logo.svg +0 -10
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import type { NextFunction, Request, Response } from 'express';
|
|
2
|
+
import type { IOConfig } from '../../config.js';
|
|
3
|
+
/**
|
|
4
|
+
* Express middleware that verifies Supabase JWT tokens.
|
|
5
|
+
* If Supabase is not configured (no jwtSecret), all requests pass through.
|
|
6
|
+
* Exempt routes always pass through regardless of config.
|
|
7
|
+
*/
|
|
8
|
+
export declare function authMiddleware(config: IOConfig): (req: Request, res: Response, next: NextFunction) => void;
|
|
9
|
+
/**
|
|
10
|
+
* Verify a JWT token for WebSocket connections.
|
|
11
|
+
* Returns true if valid or if auth is not configured.
|
|
12
|
+
*/
|
|
13
|
+
export declare function verifyWsToken(config: IOConfig, token: string | null): boolean;
|
|
14
|
+
//# sourceMappingURL=auth.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"auth.d.ts","sourceRoot":"","sources":["../../../src/api/middleware/auth.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAE/D,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAehD;;;;GAIG;AACH,wBAAgB,cAAc,CAAC,MAAM,EAAE,QAAQ,IACtC,KAAK,OAAO,EAAE,KAAK,QAAQ,EAAE,MAAM,YAAY,KAAG,IAAI,CA+B9D;AAED;;;GAGG;AACH,wBAAgB,aAAa,CAAC,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,GAAG,IAAI,GAAG,OAAO,CAe7E"}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import jwt from 'jsonwebtoken';
|
|
2
|
+
import { createChildLogger } from '../../logging/logger.js';
|
|
3
|
+
const logger = () => createChildLogger('auth');
|
|
4
|
+
// Routes that don't require authentication
|
|
5
|
+
const EXEMPT_ROUTES = [
|
|
6
|
+
{ method: 'GET', path: '/api/health' },
|
|
7
|
+
{ method: 'GET', path: '/api/config' },
|
|
8
|
+
];
|
|
9
|
+
function isExempt(method, path) {
|
|
10
|
+
return EXEMPT_ROUTES.some((r) => r.method === method.toUpperCase() && path.startsWith(r.path));
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* Express middleware that verifies Supabase JWT tokens.
|
|
14
|
+
* If Supabase is not configured (no jwtSecret), all requests pass through.
|
|
15
|
+
* Exempt routes always pass through regardless of config.
|
|
16
|
+
*/
|
|
17
|
+
export function authMiddleware(config) {
|
|
18
|
+
return (req, res, next) => {
|
|
19
|
+
// If no Supabase JWT secret configured, skip auth entirely (local-only mode)
|
|
20
|
+
if (!config.supabase.jwtSecret) {
|
|
21
|
+
next();
|
|
22
|
+
return;
|
|
23
|
+
}
|
|
24
|
+
// Exempt routes don't require auth
|
|
25
|
+
if (isExempt(req.method, req.path)) {
|
|
26
|
+
next();
|
|
27
|
+
return;
|
|
28
|
+
}
|
|
29
|
+
const authHeader = req.headers.authorization;
|
|
30
|
+
if (!authHeader?.startsWith('Bearer ')) {
|
|
31
|
+
res.status(401).json({ error: 'Missing or invalid Authorization header' });
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
const token = authHeader.slice(7);
|
|
35
|
+
try {
|
|
36
|
+
jwt.verify(token, config.supabase.jwtSecret, {
|
|
37
|
+
algorithms: ['HS256'],
|
|
38
|
+
});
|
|
39
|
+
next();
|
|
40
|
+
}
|
|
41
|
+
catch (err) {
|
|
42
|
+
logger().debug({ err }, 'JWT verification failed');
|
|
43
|
+
res.status(401).json({ error: 'Invalid or expired token' });
|
|
44
|
+
}
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Verify a JWT token for WebSocket connections.
|
|
49
|
+
* Returns true if valid or if auth is not configured.
|
|
50
|
+
*/
|
|
51
|
+
export function verifyWsToken(config, token) {
|
|
52
|
+
if (!config.supabase.jwtSecret) {
|
|
53
|
+
return true; // No auth configured — allow
|
|
54
|
+
}
|
|
55
|
+
if (!token) {
|
|
56
|
+
return false;
|
|
57
|
+
}
|
|
58
|
+
try {
|
|
59
|
+
jwt.verify(token, config.supabase.jwtSecret, { algorithms: ['HS256'] });
|
|
60
|
+
return true;
|
|
61
|
+
}
|
|
62
|
+
catch {
|
|
63
|
+
return false;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
//# sourceMappingURL=auth.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../src/api/middleware/auth.ts"],"names":[],"mappings":"AACA,OAAO,GAAG,MAAM,cAAc,CAAC;AAE/B,OAAO,EAAE,iBAAiB,EAAE,MAAM,yBAAyB,CAAC;AAE5D,MAAM,MAAM,GAAG,GAAG,EAAE,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;AAE/C,2CAA2C;AAC3C,MAAM,aAAa,GAA4C;IAC9D,EAAE,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,aAAa,EAAE;IACtC,EAAE,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,aAAa,EAAE;CACtC,CAAC;AAEF,SAAS,QAAQ,CAAC,MAAc,EAAE,IAAY;IAC7C,OAAO,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,MAAM,CAAC,WAAW,EAAE,IAAI,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;AAChG,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,cAAc,CAAC,MAAgB;IAC9C,OAAO,CAAC,GAAY,EAAE,GAAa,EAAE,IAAkB,EAAQ,EAAE;QAChE,6EAA6E;QAC7E,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC;YAChC,IAAI,EAAE,CAAC;YACP,OAAO;QACR,CAAC;QAED,mCAAmC;QACnC,IAAI,QAAQ,CAAC,GAAG,CAAC,MAAM,EAAE,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC;YACpC,IAAI,EAAE,CAAC;YACP,OAAO;QACR,CAAC;QAED,MAAM,UAAU,GAAG,GAAG,CAAC,OAAO,CAAC,aAAa,CAAC;QAC7C,IAAI,CAAC,UAAU,EAAE,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;YACxC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,yCAAyC,EAAE,CAAC,CAAC;YAC3E,OAAO;QACR,CAAC;QAED,MAAM,KAAK,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAElC,IAAI,CAAC;YACJ,GAAG,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE;gBAC5C,UAAU,EAAE,CAAC,OAAO,CAAC;aACrB,CAAC,CAAC;YACH,IAAI,EAAE,CAAC;QACR,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACd,MAAM,EAAE,CAAC,KAAK,CAAC,EAAE,GAAG,EAAE,EAAE,yBAAyB,CAAC,CAAC;YACnD,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,0BAA0B,EAAE,CAAC,CAAC;QAC7D,CAAC;IACF,CAAC,CAAC;AACH,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,aAAa,CAAC,MAAgB,EAAE,KAAoB;IACnE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC;QAChC,OAAO,IAAI,CAAC,CAAC,6BAA6B;IAC3C,CAAC;IAED,IAAI,CAAC,KAAK,EAAE,CAAC;QACZ,OAAO,KAAK,CAAC;IACd,CAAC;IAED,IAAI,CAAC;QACJ,GAAG,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,EAAE,UAAU,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QACxE,OAAO,IAAI,CAAC;IACb,CAAC;IAAC,MAAM,CAAC;QACR,OAAO,KAAK,CAAC;IACd,CAAC;AACF,CAAC"}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import type { WebSocket } from 'ws';
|
|
2
|
+
/**
|
|
3
|
+
* Register a WebSocket client for event notifications.
|
|
4
|
+
*/
|
|
5
|
+
export declare function subscribeClient(connectionId: string, ws: WebSocket): void;
|
|
6
|
+
/**
|
|
7
|
+
* Unregister a WebSocket client.
|
|
8
|
+
*/
|
|
9
|
+
export declare function unsubscribeClient(connectionId: string): void;
|
|
10
|
+
/**
|
|
11
|
+
* Initialize the notification system — subscribes to event bus and broadcasts to clients.
|
|
12
|
+
*/
|
|
13
|
+
export declare function initNotifications(): void;
|
|
14
|
+
//# sourceMappingURL=notifications.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"notifications.d.ts","sourceRoot":"","sources":["../../src/api/notifications.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AASpC;;GAEG;AACH,wBAAgB,eAAe,CAAC,YAAY,EAAE,MAAM,EAAE,EAAE,EAAE,SAAS,GAAG,IAAI,CAEzE;AAED;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,YAAY,EAAE,MAAM,GAAG,IAAI,CAE5D;AA8DD;;GAEG;AACH,wBAAgB,iBAAiB,IAAI,IAAI,CAkCxC"}
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import { createChildLogger } from '../logging/logger.js';
|
|
2
|
+
import { getEventBus } from '../squad/event-bus.js';
|
|
3
|
+
const logger = () => createChildLogger('notifications');
|
|
4
|
+
// Connected clients that want event notifications
|
|
5
|
+
const subscribers = new Map();
|
|
6
|
+
/**
|
|
7
|
+
* Register a WebSocket client for event notifications.
|
|
8
|
+
*/
|
|
9
|
+
export function subscribeClient(connectionId, ws) {
|
|
10
|
+
subscribers.set(connectionId, ws);
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* Unregister a WebSocket client.
|
|
14
|
+
*/
|
|
15
|
+
export function unsubscribeClient(connectionId) {
|
|
16
|
+
subscribers.delete(connectionId);
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Render an event to a human-readable notification string.
|
|
20
|
+
*/
|
|
21
|
+
function renderNotification(event) {
|
|
22
|
+
switch (event.type) {
|
|
23
|
+
case 'squad:created':
|
|
24
|
+
return `🆕 Squad "${event.squadName}" has been hired`;
|
|
25
|
+
case 'squad:disbanded':
|
|
26
|
+
return `🗑️ Squad "${event.squadName}" has been disbanded`;
|
|
27
|
+
case 'squad:member_added':
|
|
28
|
+
return `👋 New member added to "${event.squadName}"`;
|
|
29
|
+
case 'squad:member_retired':
|
|
30
|
+
return `👤 Member retired from "${event.squadName}"`;
|
|
31
|
+
case 'instance:created':
|
|
32
|
+
return '🚀 New work instance started';
|
|
33
|
+
case 'instance:meeting_started':
|
|
34
|
+
return '🤝 Round-table meeting in progress';
|
|
35
|
+
case 'instance:meeting_complete':
|
|
36
|
+
return '✅ Meeting complete — consensus reached';
|
|
37
|
+
case 'instance:work_started':
|
|
38
|
+
return '⚡ Squad is working on tasks';
|
|
39
|
+
case 'instance:pr_created':
|
|
40
|
+
return `📬 PR created: ${event.data?.prUrl ?? ''}`;
|
|
41
|
+
case 'instance:complete':
|
|
42
|
+
return '🎉 Work instance completed successfully';
|
|
43
|
+
case 'instance:failed':
|
|
44
|
+
return '❌ Work instance failed';
|
|
45
|
+
case 'agent:task_started':
|
|
46
|
+
return `🔧 ${event.agentRole} started a task`;
|
|
47
|
+
case 'agent:task_completed':
|
|
48
|
+
return `✔️ ${event.agentRole} completed a task`;
|
|
49
|
+
case 'agent:error':
|
|
50
|
+
return `⚠️ ${event.agentRole} encountered an error`;
|
|
51
|
+
case 'agent:permission_denied':
|
|
52
|
+
return `🚫 ${event.agentRole} was denied permission`;
|
|
53
|
+
case 'agent:tool_call':
|
|
54
|
+
return `🛠️ ${event.agentRole} used a tool`;
|
|
55
|
+
case 'meeting:contribution':
|
|
56
|
+
return `💬 ${event.agentRole}: "${event.content.slice(0, 80)}"`;
|
|
57
|
+
case 'meeting:consensus_reached':
|
|
58
|
+
return '🤝 Consensus reached in meeting';
|
|
59
|
+
case 'meeting:veto':
|
|
60
|
+
return `🛑 ${event.agentRole} vetoed the proposal`;
|
|
61
|
+
case 'inbox:new':
|
|
62
|
+
return event.kind === 'question'
|
|
63
|
+
? `❓ Squad has a question: "${event.title}"`
|
|
64
|
+
: `📋 Squad delivered: "${event.title}"`;
|
|
65
|
+
case 'inbox:resolved':
|
|
66
|
+
return `✅ Inbox item resolved: "${event.title}"`;
|
|
67
|
+
case 'schedule:fired':
|
|
68
|
+
return `⏰ Schedule fired: "${event.data?.name ?? 'unknown'}"`;
|
|
69
|
+
case 'schedule:completed':
|
|
70
|
+
return `✅ Schedule completed: "${event.data?.name ?? 'unknown'}"`;
|
|
71
|
+
case 'schedule:failed':
|
|
72
|
+
return `❌ Schedule failed: "${event.data?.name ?? 'unknown'}"`;
|
|
73
|
+
default:
|
|
74
|
+
return `📣 Event: ${event.type}`;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Initialize the notification system — subscribes to event bus and broadcasts to clients.
|
|
79
|
+
*/
|
|
80
|
+
export function initNotifications() {
|
|
81
|
+
const log = logger();
|
|
82
|
+
getEventBus().onAny((event) => {
|
|
83
|
+
const notification = renderNotification(event);
|
|
84
|
+
const payload = JSON.stringify({
|
|
85
|
+
type: 'event',
|
|
86
|
+
notification,
|
|
87
|
+
event: {
|
|
88
|
+
id: event.id,
|
|
89
|
+
type: event.type,
|
|
90
|
+
timestamp: event.timestamp.toISOString(),
|
|
91
|
+
squadId: event.squadId,
|
|
92
|
+
instanceId: event.instanceId,
|
|
93
|
+
data: 'data' in event ? event.data : undefined,
|
|
94
|
+
},
|
|
95
|
+
});
|
|
96
|
+
let delivered = 0;
|
|
97
|
+
for (const [connId, ws] of subscribers) {
|
|
98
|
+
if (ws.readyState === ws.OPEN) {
|
|
99
|
+
ws.send(payload);
|
|
100
|
+
delivered++;
|
|
101
|
+
}
|
|
102
|
+
else {
|
|
103
|
+
subscribers.delete(connId);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
if (delivered > 0) {
|
|
107
|
+
log.debug({ eventType: event.type, delivered }, 'Event broadcast');
|
|
108
|
+
}
|
|
109
|
+
});
|
|
110
|
+
log.info('Notification system initialized');
|
|
111
|
+
}
|
|
112
|
+
//# sourceMappingURL=notifications.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"notifications.js","sourceRoot":"","sources":["../../src/api/notifications.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"activity.d.ts","sourceRoot":"","sources":["../../../src/api/routes/activity.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAGjC,wBAAgB,cAAc,IAAI,MAAM,CAyBvC"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { Router } from 'express';
|
|
2
|
+
import { queryActivity } from '../../store/activity.js';
|
|
3
|
+
export function activityRouter() {
|
|
4
|
+
const router = Router();
|
|
5
|
+
/**
|
|
6
|
+
* GET /api/activity
|
|
7
|
+
* Query agent activity log with optional filters.
|
|
8
|
+
* Query params: squad, instance, agent, type, limit, offset
|
|
9
|
+
*/
|
|
10
|
+
router.get('/activity', async (req, res) => {
|
|
11
|
+
try {
|
|
12
|
+
const entries = await queryActivity({
|
|
13
|
+
squadId: req.query.squad,
|
|
14
|
+
instanceId: req.query.instance,
|
|
15
|
+
agentRole: req.query.agent,
|
|
16
|
+
activityType: req.query.type,
|
|
17
|
+
limit: req.query.limit ? Number.parseInt(req.query.limit, 10) : undefined,
|
|
18
|
+
offset: req.query.offset ? Number.parseInt(req.query.offset, 10) : undefined,
|
|
19
|
+
});
|
|
20
|
+
res.json({ entries });
|
|
21
|
+
}
|
|
22
|
+
catch (err) {
|
|
23
|
+
res.status(500).json({ error: 'Failed to query activity' });
|
|
24
|
+
}
|
|
25
|
+
});
|
|
26
|
+
return router;
|
|
27
|
+
}
|
|
28
|
+
//# sourceMappingURL=activity.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"activity.js","sourceRoot":"","sources":["../../../src/api/routes/activity.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAqB,aAAa,EAAE,MAAM,yBAAyB,CAAC;AAE3E,MAAM,UAAU,cAAc;IAC7B,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC;IAExB;;;;OAIG;IACH,MAAM,CAAC,GAAG,CAAC,WAAW,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;QAC1C,IAAI,CAAC;YACJ,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC;gBACnC,OAAO,EAAE,GAAG,CAAC,KAAK,CAAC,KAA2B;gBAC9C,UAAU,EAAE,GAAG,CAAC,KAAK,CAAC,QAA8B;gBACpD,SAAS,EAAE,GAAG,CAAC,KAAK,CAAC,KAA2B;gBAChD,YAAY,EAAE,GAAG,CAAC,KAAK,CAAC,IAAgC;gBACxD,KAAK,EAAE,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,KAAe,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS;gBACnF,MAAM,EAAE,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,MAAgB,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS;aACtF,CAAC,CAAC;YACH,GAAG,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC;QACvB,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACd,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,0BAA0B,EAAE,CAAC,CAAC;QAC7D,CAAC;IACF,CAAC,CAAC,CAAC;IAEH,OAAO,MAAM,CAAC;AACf,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"attachments.d.ts","sourceRoot":"","sources":["../../../src/api/routes/attachments.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAIjC,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,CAqFzD"}
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import { createReadStream, existsSync, mkdirSync } from 'node:fs';
|
|
2
|
+
import { writeFile } from 'node:fs/promises';
|
|
3
|
+
import { join } from 'node:path';
|
|
4
|
+
import { Router } from 'express';
|
|
5
|
+
import multer from 'multer';
|
|
6
|
+
import { getDatabase } from '../../store/db.js';
|
|
7
|
+
export function attachmentsRouter(dataDir) {
|
|
8
|
+
const router = Router();
|
|
9
|
+
const attachmentsDir = join(dataDir, 'attachments');
|
|
10
|
+
mkdirSync(attachmentsDir, { recursive: true });
|
|
11
|
+
// Use memory storage — we'll write to disk ourselves with proper naming
|
|
12
|
+
const upload = multer({
|
|
13
|
+
storage: multer.memoryStorage(),
|
|
14
|
+
limits: { fileSize: 50 * 1024 * 1024 },
|
|
15
|
+
});
|
|
16
|
+
/**
|
|
17
|
+
* POST /api/attachments
|
|
18
|
+
* Upload a file attachment. Returns the attachment metadata.
|
|
19
|
+
*/
|
|
20
|
+
router.post('/attachments', upload.single('file'), async (req, res) => {
|
|
21
|
+
try {
|
|
22
|
+
const file = req.file;
|
|
23
|
+
if (!file) {
|
|
24
|
+
res.status(400).json({ error: 'No file uploaded' });
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
const id = crypto.randomUUID();
|
|
28
|
+
const messageId = req.body?.messageId ?? null;
|
|
29
|
+
const fileDir = join(attachmentsDir, id);
|
|
30
|
+
mkdirSync(fileDir, { recursive: true });
|
|
31
|
+
const diskPath = join(fileDir, file.originalname);
|
|
32
|
+
await writeFile(diskPath, file.buffer);
|
|
33
|
+
const db = getDatabase();
|
|
34
|
+
await db.execute({
|
|
35
|
+
sql: `INSERT INTO attachments (id, message_id, filename, mime_type, size_bytes, disk_path)
|
|
36
|
+
VALUES (?, ?, ?, ?, ?, ?)`,
|
|
37
|
+
args: [id, messageId, file.originalname, file.mimetype, file.size, diskPath],
|
|
38
|
+
});
|
|
39
|
+
res.status(201).json({
|
|
40
|
+
id,
|
|
41
|
+
filename: file.originalname,
|
|
42
|
+
mimeType: file.mimetype,
|
|
43
|
+
sizeBytes: file.size,
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
catch (err) {
|
|
47
|
+
res.status(500).json({ error: 'Failed to upload attachment' });
|
|
48
|
+
}
|
|
49
|
+
});
|
|
50
|
+
/**
|
|
51
|
+
* GET /api/attachments/:id
|
|
52
|
+
* Download an attachment by ID.
|
|
53
|
+
*/
|
|
54
|
+
router.get('/attachments/:id', async (req, res) => {
|
|
55
|
+
try {
|
|
56
|
+
const db = getDatabase();
|
|
57
|
+
const result = await db.execute({
|
|
58
|
+
sql: 'SELECT filename, mime_type, disk_path FROM attachments WHERE id = ?',
|
|
59
|
+
args: [req.params.id],
|
|
60
|
+
});
|
|
61
|
+
if (result.rows.length === 0) {
|
|
62
|
+
res.status(404).json({ error: 'Attachment not found' });
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
const row = result.rows[0];
|
|
66
|
+
const diskPath = row.disk_path;
|
|
67
|
+
const filename = row.filename;
|
|
68
|
+
const mimeType = row.mime_type ?? 'application/octet-stream';
|
|
69
|
+
if (!existsSync(diskPath)) {
|
|
70
|
+
res.status(404).json({ error: 'Attachment file missing from disk' });
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
res.setHeader('Content-Type', mimeType);
|
|
74
|
+
res.setHeader('Content-Disposition', `attachment; filename="${filename}"`);
|
|
75
|
+
createReadStream(diskPath).pipe(res);
|
|
76
|
+
}
|
|
77
|
+
catch (err) {
|
|
78
|
+
res.status(500).json({ error: 'Failed to retrieve attachment' });
|
|
79
|
+
}
|
|
80
|
+
});
|
|
81
|
+
return router;
|
|
82
|
+
}
|
|
83
|
+
//# sourceMappingURL=attachments.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"attachments.js","sourceRoot":"","sources":["../../../src/api/routes/attachments.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../../src/api/routes/config.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAGjC,wBAAgB,YAAY,IAAI,MAAM,CA4GrC"}
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import { existsSync, readFileSync, writeFileSync } from 'node:fs';
|
|
2
|
+
import { homedir } from 'node:os';
|
|
3
|
+
import { join } from 'node:path';
|
|
4
|
+
import { Router } from 'express';
|
|
5
|
+
import { loadConfig } from '../../config.js';
|
|
6
|
+
export function configRouter() {
|
|
7
|
+
const router = Router();
|
|
8
|
+
function getConfigPath() {
|
|
9
|
+
const dataDir = process.env.IO_DATA_DIR ?? join(homedir(), '.io');
|
|
10
|
+
const resolved = dataDir.startsWith('~') ? join(homedir(), dataDir.slice(1)) : dataDir;
|
|
11
|
+
return join(resolved, 'config.json');
|
|
12
|
+
}
|
|
13
|
+
/**
|
|
14
|
+
* GET /api/config
|
|
15
|
+
* Return current config with sensitive fields redacted.
|
|
16
|
+
*/
|
|
17
|
+
router.get('/config', (_req, res) => {
|
|
18
|
+
try {
|
|
19
|
+
const config = loadConfig();
|
|
20
|
+
// Redact sensitive fields
|
|
21
|
+
const redacted = {
|
|
22
|
+
apiPort: config.apiPort,
|
|
23
|
+
logLevel: config.logLevel,
|
|
24
|
+
defaultModel: config.defaultModel,
|
|
25
|
+
maxInstancesPerSquad: config.maxInstancesPerSquad,
|
|
26
|
+
dataDir: config.dataDir,
|
|
27
|
+
pricing: config.pricing,
|
|
28
|
+
telegram: {
|
|
29
|
+
botToken: config.telegram.botToken ? '••••••••' : null,
|
|
30
|
+
allowedChatIds: config.telegram.allowedChatIds,
|
|
31
|
+
},
|
|
32
|
+
supabase: {
|
|
33
|
+
projectUrl: config.supabase.projectUrl,
|
|
34
|
+
anonKey: config.supabase.anonKey,
|
|
35
|
+
jwtSecret: config.supabase.jwtSecret ? '••••••••' : null,
|
|
36
|
+
},
|
|
37
|
+
};
|
|
38
|
+
res.json({ config: redacted });
|
|
39
|
+
}
|
|
40
|
+
catch (err) {
|
|
41
|
+
res.status(500).json({ error: 'Failed to load config' });
|
|
42
|
+
}
|
|
43
|
+
});
|
|
44
|
+
/**
|
|
45
|
+
* PATCH /api/config
|
|
46
|
+
* Merge partial config into config.json on disk.
|
|
47
|
+
* Does NOT accept dataDir changes (immutable).
|
|
48
|
+
* Body: partial config object
|
|
49
|
+
*/
|
|
50
|
+
router.patch('/config', (req, res) => {
|
|
51
|
+
try {
|
|
52
|
+
const updates = req.body;
|
|
53
|
+
if (!updates || typeof updates !== 'object') {
|
|
54
|
+
res.status(400).json({ error: 'Body must be a JSON object' });
|
|
55
|
+
return;
|
|
56
|
+
}
|
|
57
|
+
// Disallow changing dataDir (would break running daemon)
|
|
58
|
+
delete updates.dataDir;
|
|
59
|
+
const configPath = getConfigPath();
|
|
60
|
+
let existing = {};
|
|
61
|
+
if (existsSync(configPath)) {
|
|
62
|
+
existing = JSON.parse(readFileSync(configPath, 'utf-8'));
|
|
63
|
+
}
|
|
64
|
+
// Deep merge one level (telegram, pricing)
|
|
65
|
+
const merged = { ...existing };
|
|
66
|
+
for (const [key, value] of Object.entries(updates)) {
|
|
67
|
+
if (value !== null &&
|
|
68
|
+
typeof value === 'object' &&
|
|
69
|
+
!Array.isArray(value) &&
|
|
70
|
+
typeof merged[key] === 'object') {
|
|
71
|
+
merged[key] = { ...merged[key], ...value };
|
|
72
|
+
}
|
|
73
|
+
else {
|
|
74
|
+
merged[key] = value;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
writeFileSync(configPath, JSON.stringify(merged, null, 2), 'utf-8');
|
|
78
|
+
// Reload and return redacted
|
|
79
|
+
const config = loadConfig();
|
|
80
|
+
res.json({
|
|
81
|
+
config: {
|
|
82
|
+
apiPort: config.apiPort,
|
|
83
|
+
logLevel: config.logLevel,
|
|
84
|
+
defaultModel: config.defaultModel,
|
|
85
|
+
maxInstancesPerSquad: config.maxInstancesPerSquad,
|
|
86
|
+
dataDir: config.dataDir,
|
|
87
|
+
pricing: config.pricing,
|
|
88
|
+
telegram: {
|
|
89
|
+
botToken: config.telegram.botToken ? '••••••••' : null,
|
|
90
|
+
allowedChatIds: config.telegram.allowedChatIds,
|
|
91
|
+
},
|
|
92
|
+
supabase: {
|
|
93
|
+
projectUrl: config.supabase.projectUrl,
|
|
94
|
+
anonKey: config.supabase.anonKey,
|
|
95
|
+
jwtSecret: config.supabase.jwtSecret ? '••••••••' : null,
|
|
96
|
+
},
|
|
97
|
+
},
|
|
98
|
+
});
|
|
99
|
+
}
|
|
100
|
+
catch (err) {
|
|
101
|
+
res.status(500).json({ error: 'Failed to update config' });
|
|
102
|
+
}
|
|
103
|
+
});
|
|
104
|
+
return router;
|
|
105
|
+
}
|
|
106
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../../../src/api/routes/config.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conversations.d.ts","sourceRoot":"","sources":["../../../src/api/routes/conversations.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAGjC,wBAAgB,mBAAmB,IAAI,MAAM,CAmF5C"}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { Router } from 'express';
|
|
2
|
+
import { getDatabase } from '../../store/db.js';
|
|
3
|
+
export function conversationsRouter() {
|
|
4
|
+
const router = Router();
|
|
5
|
+
/**
|
|
6
|
+
* GET /api/conversations
|
|
7
|
+
* Load chat history with cursor-based pagination.
|
|
8
|
+
* Query params: limit (default 50), before (message id for pagination)
|
|
9
|
+
*/
|
|
10
|
+
router.get('/conversations', async (req, res) => {
|
|
11
|
+
try {
|
|
12
|
+
const limit = Math.min(Number.parseInt(req.query.limit, 10) || 50, 200);
|
|
13
|
+
const before = req.query.before;
|
|
14
|
+
const db = getDatabase();
|
|
15
|
+
let rows;
|
|
16
|
+
if (before) {
|
|
17
|
+
// Get the timestamp of the cursor message
|
|
18
|
+
const cursorResult = await db.execute({
|
|
19
|
+
sql: 'SELECT created_at FROM conversations WHERE id = ?',
|
|
20
|
+
args: [before],
|
|
21
|
+
});
|
|
22
|
+
if (cursorResult.rows.length === 0) {
|
|
23
|
+
res.status(400).json({ error: 'Invalid cursor: message not found' });
|
|
24
|
+
return;
|
|
25
|
+
}
|
|
26
|
+
const cursorTime = cursorResult.rows[0].created_at;
|
|
27
|
+
const result = await db.execute({
|
|
28
|
+
sql: `SELECT id, role, content, source, attachments, created_at
|
|
29
|
+
FROM conversations
|
|
30
|
+
WHERE created_at < ? OR (created_at = ? AND id < ?)
|
|
31
|
+
ORDER BY created_at DESC, id DESC
|
|
32
|
+
LIMIT ?`,
|
|
33
|
+
args: [cursorTime, cursorTime, before, limit],
|
|
34
|
+
});
|
|
35
|
+
rows = result.rows;
|
|
36
|
+
}
|
|
37
|
+
else {
|
|
38
|
+
const result = await db.execute({
|
|
39
|
+
sql: `SELECT id, role, content, source, attachments, created_at
|
|
40
|
+
FROM conversations
|
|
41
|
+
ORDER BY created_at DESC, id DESC
|
|
42
|
+
LIMIT ?`,
|
|
43
|
+
args: [limit],
|
|
44
|
+
});
|
|
45
|
+
rows = result.rows;
|
|
46
|
+
}
|
|
47
|
+
// Reverse so messages are in chronological order
|
|
48
|
+
const messages = rows.reverse().map((row) => ({
|
|
49
|
+
id: row.id,
|
|
50
|
+
role: row.role,
|
|
51
|
+
content: row.content,
|
|
52
|
+
source: row.source,
|
|
53
|
+
attachments: row.attachments ? JSON.parse(row.attachments) : null,
|
|
54
|
+
timestamp: row.created_at,
|
|
55
|
+
}));
|
|
56
|
+
const cursor = rows.length > 0 ? rows[0].id : null;
|
|
57
|
+
res.json({
|
|
58
|
+
messages,
|
|
59
|
+
cursor,
|
|
60
|
+
hasMore: rows.length === limit,
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
catch (err) {
|
|
64
|
+
res.status(500).json({ error: 'Failed to load conversations' });
|
|
65
|
+
}
|
|
66
|
+
});
|
|
67
|
+
return router;
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=conversations.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conversations.js","sourceRoot":"","sources":["../../../src/api/routes/conversations.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,UAAU,mBAAmB;IAClC,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC;IAExB;;;;OAIG;IACH,MAAM,CAAC,GAAG,CAAC,gBAAgB,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;QAC/C,IAAI,CAAC;YACJ,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,KAAe,EAAE,EAAE,CAAC,IAAI,EAAE,EAAE,GAAG,CAAC,CAAC;YAClF,MAAM,MAAM,GAAG,GAAG,CAAC,KAAK,CAAC,MAA4B,CAAC;YAEtD,MAAM,EAAE,GAAG,WAAW,EAAE,CAAC;YAEzB,IAAI,IAOF,CAAC;YAEH,IAAI,MAAM,EAAE,CAAC;gBACZ,0CAA0C;gBAC1C,MAAM,YAAY,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC;oBACrC,GAAG,EAAE,mDAAmD;oBACxD,IAAI,EAAE,CAAC,MAAM,CAAC;iBACd,CAAC,CAAC;gBAEH,IAAI,YAAY,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBACpC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mCAAmC,EAAE,CAAC,CAAC;oBACrE,OAAO;gBACR,CAAC;gBAED,MAAM,UAAU,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,UAAoB,CAAC;gBAE7D,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC;oBAC/B,GAAG,EAAE;;;;cAII;oBACT,IAAI,EAAE,CAAC,UAAU,EAAE,UAAU,EAAE,MAAM,EAAE,KAAK,CAAC;iBAC7C,CAAC,CAAC;gBAEH,IAAI,GAAG,MAAM,CAAC,IAAW,CAAC;YAC3B,CAAC;iBAAM,CAAC;gBACP,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC;oBAC/B,GAAG,EAAE;;;cAGI;oBACT,IAAI,EAAE,CAAC,KAAK,CAAC;iBACb,CAAC,CAAC;gBAEH,IAAI,GAAG,MAAM,CAAC,IAAW,CAAC;YAC3B,CAAC;YAED,iDAAiD;YACjD,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;gBAC7C,EAAE,EAAE,GAAG,CAAC,EAAE;gBACV,IAAI,EAAE,GAAG,CAAC,IAAI;gBACd,OAAO,EAAE,GAAG,CAAC,OAAO;gBACpB,MAAM,EAAE,GAAG,CAAC,MAAM;gBAClB,WAAW,EAAE,GAAG,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,IAAI;gBACjE,SAAS,EAAE,GAAG,CAAC,UAAU;aACzB,CAAC,CAAC,CAAC;YAEJ,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;YAEnD,GAAG,CAAC,IAAI,CAAC;gBACR,QAAQ;gBACR,MAAM;gBACN,OAAO,EAAE,IAAI,CAAC,MAAM,KAAK,KAAK;aAC9B,CAAC,CAAC;QACJ,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACd,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,8BAA8B,EAAE,CAAC,CAAC;QACjE,CAAC;IACF,CAAC,CAAC,CAAC;IAEH,OAAO,MAAM,CAAC;AACf,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"health.d.ts","sourceRoot":"","sources":["../../../src/api/routes/health.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAGjC,wBAAgB,YAAY,IAAI,MAAM,CAcrC"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { Router } from 'express';
|
|
2
|
+
import { getHealthStatus } from '../../copilot/health-monitor.js';
|
|
3
|
+
export function healthRouter() {
|
|
4
|
+
const router = Router();
|
|
5
|
+
router.get('/health', (_req, res) => {
|
|
6
|
+
const health = getHealthStatus();
|
|
7
|
+
res.json({
|
|
8
|
+
status: health.status,
|
|
9
|
+
uptime: health.uptime,
|
|
10
|
+
copilotConnected: health.copilotConnected,
|
|
11
|
+
lastCheck: health.lastCheck.toISOString(),
|
|
12
|
+
});
|
|
13
|
+
});
|
|
14
|
+
return router;
|
|
15
|
+
}
|
|
16
|
+
//# sourceMappingURL=health.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"health.js","sourceRoot":"","sources":["../../../src/api/routes/health.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,eAAe,EAAE,MAAM,iCAAiC,CAAC;AAElE,MAAM,UAAU,YAAY;IAC3B,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC;IAExB,MAAM,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,IAAI,EAAE,GAAG,EAAE,EAAE;QACnC,MAAM,MAAM,GAAG,eAAe,EAAE,CAAC;QACjC,GAAG,CAAC,IAAI,CAAC;YACR,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,gBAAgB,EAAE,MAAM,CAAC,gBAAgB;YACzC,SAAS,EAAE,MAAM,CAAC,SAAS,CAAC,WAAW,EAAE;SACzC,CAAC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,OAAO,MAAM,CAAC;AACf,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"inbox.d.ts","sourceRoot":"","sources":["../../../src/api/routes/inbox.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAWjC,wBAAgB,WAAW,IAAI,MAAM,CAsFpC"}
|