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.
Files changed (327) hide show
  1. package/dist/api/middleware/auth.d.ts +14 -0
  2. package/dist/api/middleware/auth.d.ts.map +1 -0
  3. package/dist/api/middleware/auth.js +66 -0
  4. package/dist/api/middleware/auth.js.map +1 -0
  5. package/dist/api/notifications.d.ts +14 -0
  6. package/dist/api/notifications.d.ts.map +1 -0
  7. package/dist/api/notifications.js +112 -0
  8. package/dist/api/notifications.js.map +1 -0
  9. package/dist/api/routes/activity.d.ts +3 -0
  10. package/dist/api/routes/activity.d.ts.map +1 -0
  11. package/dist/api/routes/activity.js +28 -0
  12. package/dist/api/routes/activity.js.map +1 -0
  13. package/dist/api/routes/attachments.d.ts +3 -0
  14. package/dist/api/routes/attachments.d.ts.map +1 -0
  15. package/dist/api/routes/attachments.js +83 -0
  16. package/dist/api/routes/attachments.js.map +1 -0
  17. package/dist/api/routes/config.d.ts +3 -0
  18. package/dist/api/routes/config.d.ts.map +1 -0
  19. package/dist/api/routes/config.js +106 -0
  20. package/dist/api/routes/config.js.map +1 -0
  21. package/dist/api/routes/conversations.d.ts +3 -0
  22. package/dist/api/routes/conversations.d.ts.map +1 -0
  23. package/dist/api/routes/conversations.js +69 -0
  24. package/dist/api/routes/conversations.js.map +1 -0
  25. package/dist/api/routes/health.d.ts +3 -0
  26. package/dist/api/routes/health.d.ts.map +1 -0
  27. package/dist/api/routes/health.js +16 -0
  28. package/dist/api/routes/health.js.map +1 -0
  29. package/dist/api/routes/inbox.d.ts +3 -0
  30. package/dist/api/routes/inbox.d.ts.map +1 -0
  31. package/dist/api/routes/inbox.js +88 -0
  32. package/dist/api/routes/inbox.js.map +1 -0
  33. package/dist/api/routes/schedules.d.ts +3 -0
  34. package/dist/api/routes/schedules.d.ts.map +1 -0
  35. package/dist/api/routes/schedules.js +96 -0
  36. package/dist/api/routes/schedules.js.map +1 -0
  37. package/dist/api/routes/skills.d.ts +2 -0
  38. package/dist/api/routes/skills.d.ts.map +1 -0
  39. package/dist/api/routes/skills.js +85 -0
  40. package/dist/api/routes/skills.js.map +1 -0
  41. package/dist/api/routes/squads.d.ts +3 -0
  42. package/dist/api/routes/squads.d.ts.map +1 -0
  43. package/dist/api/routes/squads.js +129 -0
  44. package/dist/api/routes/squads.js.map +1 -0
  45. package/dist/api/routes/usage.d.ts +3 -0
  46. package/dist/api/routes/usage.d.ts.map +1 -0
  47. package/dist/api/routes/usage.js +55 -0
  48. package/dist/api/routes/usage.js.map +1 -0
  49. package/dist/api/routes/wiki.d.ts +2 -0
  50. package/dist/api/routes/wiki.d.ts.map +1 -0
  51. package/dist/api/routes/wiki.js +43 -0
  52. package/dist/api/routes/wiki.js.map +1 -0
  53. package/dist/api/server.d.ts +7 -0
  54. package/dist/api/server.d.ts.map +1 -0
  55. package/dist/api/server.js +136 -634
  56. package/dist/api/server.js.map +1 -0
  57. package/dist/config.d.ts +3 -0
  58. package/dist/config.d.ts.map +1 -0
  59. package/dist/config.js +2 -91
  60. package/dist/config.js.map +1 -0
  61. package/dist/copilot/client.d.ts +5 -0
  62. package/dist/copilot/client.d.ts.map +1 -0
  63. package/dist/copilot/client.js +19 -11
  64. package/dist/copilot/client.js.map +1 -0
  65. package/dist/copilot/health-monitor.d.ts +14 -0
  66. package/dist/copilot/health-monitor.d.ts.map +1 -0
  67. package/dist/copilot/health-monitor.js +70 -0
  68. package/dist/copilot/health-monitor.js.map +1 -0
  69. package/dist/copilot/orchestrator.d.ts +5 -0
  70. package/dist/copilot/orchestrator.d.ts.map +1 -0
  71. package/dist/copilot/orchestrator.js +127 -123
  72. package/dist/copilot/orchestrator.js.map +1 -0
  73. package/dist/copilot/tools.d.ts +49 -0
  74. package/dist/copilot/tools.d.ts.map +1 -0
  75. package/dist/copilot/tools.js +545 -321
  76. package/dist/copilot/tools.js.map +1 -0
  77. package/dist/index.d.ts +3 -0
  78. package/dist/index.d.ts.map +1 -0
  79. package/dist/index.js +82 -26
  80. package/dist/index.js.map +1 -0
  81. package/dist/logging/logger.d.ts +6 -0
  82. package/dist/logging/logger.d.ts.map +1 -0
  83. package/dist/logging/logger.js +21 -0
  84. package/dist/logging/logger.js.map +1 -0
  85. package/dist/models/index.d.ts +6 -0
  86. package/dist/models/index.d.ts.map +1 -0
  87. package/dist/models/index.js +4 -0
  88. package/dist/models/index.js.map +1 -0
  89. package/dist/models/pricing.d.ts +25 -0
  90. package/dist/models/pricing.d.ts.map +1 -0
  91. package/dist/models/pricing.js +96 -0
  92. package/dist/models/pricing.js.map +1 -0
  93. package/dist/models/registry.d.ts +34 -0
  94. package/dist/models/registry.d.ts.map +1 -0
  95. package/dist/models/registry.js +109 -0
  96. package/dist/models/registry.js.map +1 -0
  97. package/dist/models/token-tracker.d.ts +40 -0
  98. package/dist/models/token-tracker.d.ts.map +1 -0
  99. package/dist/models/token-tracker.js +102 -0
  100. package/dist/models/token-tracker.js.map +1 -0
  101. package/dist/scheduler/engine.d.ts +9 -0
  102. package/dist/scheduler/engine.d.ts.map +1 -0
  103. package/dist/scheduler/engine.js +127 -0
  104. package/dist/scheduler/engine.js.map +1 -0
  105. package/dist/skills/index.d.ts +3 -0
  106. package/dist/skills/index.d.ts.map +1 -0
  107. package/dist/skills/index.js +2 -0
  108. package/dist/skills/index.js.map +1 -0
  109. package/dist/skills/store.d.ts +52 -0
  110. package/dist/skills/store.d.ts.map +1 -0
  111. package/dist/skills/store.js +148 -0
  112. package/dist/skills/store.js.map +1 -0
  113. package/dist/squad/agent.d.ts +46 -0
  114. package/dist/squad/agent.d.ts.map +1 -0
  115. package/dist/squad/agent.js +261 -0
  116. package/dist/squad/agent.js.map +1 -0
  117. package/dist/squad/autonomy.d.ts +16 -0
  118. package/dist/squad/autonomy.d.ts.map +1 -0
  119. package/dist/squad/autonomy.js +63 -0
  120. package/dist/squad/autonomy.js.map +1 -0
  121. package/dist/squad/event-bus.d.ts +22 -0
  122. package/dist/squad/event-bus.d.ts.map +1 -0
  123. package/dist/squad/event-bus.js +56 -0
  124. package/dist/squad/event-bus.js.map +1 -0
  125. package/dist/squad/execution/index.d.ts +12 -0
  126. package/dist/squad/execution/index.d.ts.map +1 -0
  127. package/dist/squad/execution/index.js +7 -0
  128. package/dist/squad/execution/index.js.map +1 -0
  129. package/dist/squad/execution/instance.d.ts +40 -0
  130. package/dist/squad/execution/instance.d.ts.map +1 -0
  131. package/dist/squad/execution/instance.js +138 -0
  132. package/dist/squad/execution/instance.js.map +1 -0
  133. package/dist/squad/execution/meeting.d.ts +25 -0
  134. package/dist/squad/execution/meeting.d.ts.map +1 -0
  135. package/dist/squad/execution/meeting.js +140 -0
  136. package/dist/squad/execution/meeting.js.map +1 -0
  137. package/dist/squad/execution/pr.d.ts +15 -0
  138. package/dist/squad/execution/pr.d.ts.map +1 -0
  139. package/dist/squad/execution/pr.js +93 -0
  140. package/dist/squad/execution/pr.js.map +1 -0
  141. package/dist/squad/execution/runner.d.ts +22 -0
  142. package/dist/squad/execution/runner.d.ts.map +1 -0
  143. package/dist/squad/execution/runner.js +68 -0
  144. package/dist/squad/execution/runner.js.map +1 -0
  145. package/dist/squad/execution/tasks.d.ts +11 -0
  146. package/dist/squad/execution/tasks.d.ts.map +1 -0
  147. package/dist/squad/execution/tasks.js +85 -0
  148. package/dist/squad/execution/tasks.js.map +1 -0
  149. package/dist/squad/execution/worktree.d.ts +26 -0
  150. package/dist/squad/execution/worktree.d.ts.map +1 -0
  151. package/dist/squad/execution/worktree.js +111 -0
  152. package/dist/squad/execution/worktree.js.map +1 -0
  153. package/dist/squad/hiring.d.ts +32 -0
  154. package/dist/squad/hiring.d.ts.map +1 -0
  155. package/dist/squad/hiring.js +200 -0
  156. package/dist/squad/hiring.js.map +1 -0
  157. package/dist/squad/index.d.ts +8 -0
  158. package/dist/squad/index.d.ts.map +1 -0
  159. package/dist/squad/index.js +6 -0
  160. package/dist/squad/index.js.map +1 -0
  161. package/dist/squad/manager.d.ts +48 -0
  162. package/dist/squad/manager.d.ts.map +1 -0
  163. package/dist/squad/manager.js +274 -0
  164. package/dist/squad/manager.js.map +1 -0
  165. package/dist/squad/name-generator.d.ts +16 -0
  166. package/dist/squad/name-generator.d.ts.map +1 -0
  167. package/dist/squad/name-generator.js +113 -0
  168. package/dist/squad/name-generator.js.map +1 -0
  169. package/dist/squad/roles/templates.d.ts +5 -0
  170. package/dist/squad/roles/templates.d.ts.map +1 -0
  171. package/dist/squad/roles/templates.js +102 -0
  172. package/dist/squad/roles/templates.js.map +1 -0
  173. package/dist/squad/skill-parser.d.ts +36 -0
  174. package/dist/squad/skill-parser.d.ts.map +1 -0
  175. package/dist/squad/skill-parser.js +83 -0
  176. package/dist/squad/skill-parser.js.map +1 -0
  177. package/dist/squad/source-resolver.d.ts +20 -0
  178. package/dist/squad/source-resolver.d.ts.map +1 -0
  179. package/dist/squad/source-resolver.js +52 -0
  180. package/dist/squad/source-resolver.js.map +1 -0
  181. package/dist/store/activity.d.ts +43 -0
  182. package/dist/store/activity.d.ts.map +1 -0
  183. package/dist/store/activity.js +131 -0
  184. package/dist/store/activity.js.map +1 -0
  185. package/dist/store/db.d.ts +5 -0
  186. package/dist/store/db.d.ts.map +1 -0
  187. package/dist/store/db.js +209 -248
  188. package/dist/store/db.js.map +1 -0
  189. package/dist/store/inbox.d.ts +53 -0
  190. package/dist/store/inbox.d.ts.map +1 -0
  191. package/dist/store/inbox.js +151 -0
  192. package/dist/store/inbox.js.map +1 -0
  193. package/dist/store/schedules.d.ts +53 -0
  194. package/dist/store/schedules.d.ts.map +1 -0
  195. package/dist/store/schedules.js +149 -54
  196. package/dist/store/schedules.js.map +1 -0
  197. package/dist/wiki/index.d.ts +3 -0
  198. package/dist/wiki/index.d.ts.map +1 -0
  199. package/dist/wiki/index.js +2 -0
  200. package/dist/wiki/index.js.map +1 -0
  201. package/dist/wiki/store.d.ts +49 -0
  202. package/dist/wiki/store.d.ts.map +1 -0
  203. package/dist/wiki/store.js +115 -0
  204. package/dist/wiki/store.js.map +1 -0
  205. package/package.json +52 -59
  206. package/src/api/middleware/auth.ts +76 -0
  207. package/src/api/notifications.ts +122 -0
  208. package/src/api/routes/activity.ts +29 -0
  209. package/src/api/routes/attachments.ts +93 -0
  210. package/src/api/routes/config.ts +115 -0
  211. package/src/api/routes/conversations.ts +87 -0
  212. package/src/api/routes/health.ts +18 -0
  213. package/src/api/routes/inbox.ts +98 -0
  214. package/src/api/routes/schedules.ts +121 -0
  215. package/src/api/routes/skills.ts +105 -0
  216. package/src/api/routes/squads.ts +145 -0
  217. package/src/api/routes/usage.ts +57 -0
  218. package/src/api/routes/wiki.ts +49 -0
  219. package/src/api/server.ts +186 -0
  220. package/src/config.ts +3 -0
  221. package/src/copilot/client.ts +42 -0
  222. package/src/copilot/health-monitor.ts +85 -0
  223. package/src/copilot/orchestrator.ts +222 -0
  224. package/src/copilot/tools.ts +707 -0
  225. package/src/index.ts +112 -0
  226. package/src/logging/logger.ts +26 -0
  227. package/src/models/index.ts +11 -0
  228. package/src/models/pricing.ts +121 -0
  229. package/src/models/registry.ts +131 -0
  230. package/src/models/token-tracker.ts +151 -0
  231. package/src/scheduler/engine.ts +146 -0
  232. package/src/skills/index.ts +13 -0
  233. package/src/skills/store.ts +188 -0
  234. package/src/squad/agent.ts +326 -0
  235. package/src/squad/autonomy.ts +78 -0
  236. package/src/squad/event-bus.ts +71 -0
  237. package/src/squad/execution/index.ts +17 -0
  238. package/src/squad/execution/instance.ts +186 -0
  239. package/src/squad/execution/meeting.ts +191 -0
  240. package/src/squad/execution/pr.ts +127 -0
  241. package/src/squad/execution/runner.ts +97 -0
  242. package/src/squad/execution/tasks.ts +111 -0
  243. package/src/squad/execution/worktree.ts +138 -0
  244. package/src/squad/hiring.ts +222 -0
  245. package/src/squad/index.ts +17 -0
  246. package/src/squad/manager.ts +337 -0
  247. package/src/squad/name-generator.ts +135 -0
  248. package/src/squad/roles/templates.ts +104 -0
  249. package/src/squad/skill-parser.ts +120 -0
  250. package/src/squad/source-resolver.ts +57 -0
  251. package/src/store/activity.ts +176 -0
  252. package/src/store/db.ts +237 -0
  253. package/src/store/inbox.ts +199 -0
  254. package/src/store/schedules.ts +199 -0
  255. package/src/wiki/index.ts +12 -0
  256. package/src/wiki/store.ts +139 -0
  257. package/tsconfig.json +9 -0
  258. package/LICENSE +0 -21
  259. package/README.md +0 -333
  260. package/dist/api/auth.js +0 -46
  261. package/dist/chat/attachments.js +0 -112
  262. package/dist/copilot/agents.js +0 -309
  263. package/dist/copilot/ceremonies.js +0 -174
  264. package/dist/copilot/gh-token.js +0 -64
  265. package/dist/copilot/io-scheduler.js +0 -79
  266. package/dist/copilot/model-router.js +0 -114
  267. package/dist/copilot/scheduler.js +0 -88
  268. package/dist/copilot/skills.js +0 -252
  269. package/dist/copilot/specialist-runner.js +0 -191
  270. package/dist/copilot/squad-tools.js +0 -258
  271. package/dist/copilot/system-message.js +0 -86
  272. package/dist/copilot/token-tracker.js +0 -98
  273. package/dist/copilot/trigger-schedule.js +0 -33
  274. package/dist/daemon.js +0 -67
  275. package/dist/logging.js +0 -27
  276. package/dist/mcp/config.js +0 -29
  277. package/dist/mcp/index.js +0 -3
  278. package/dist/mcp/registry.js +0 -42
  279. package/dist/notify.js +0 -25
  280. package/dist/paths.js +0 -17
  281. package/dist/setup.js +0 -35
  282. package/dist/store/agent-events.js +0 -19
  283. package/dist/store/audit-log.js +0 -71
  284. package/dist/store/conversations.js +0 -164
  285. package/dist/store/feed.js +0 -44
  286. package/dist/store/instances.js +0 -75
  287. package/dist/store/squad-colors.js +0 -23
  288. package/dist/store/squads.js +0 -60
  289. package/dist/store/tasks.js +0 -78
  290. package/dist/store/token-usage.js +0 -94
  291. package/dist/telegram/bot.js +0 -41
  292. package/dist/telegram/handlers.js +0 -42
  293. package/dist/watchdog.js +0 -37
  294. package/dist/wiki/backlinks.js +0 -51
  295. package/dist/wiki/fs.js +0 -108
  296. package/dist/wiki/search.js +0 -47
  297. package/web-dist/assets/AuditLogView-BzfjNXBT.js +0 -6
  298. package/web-dist/assets/ChatView-BdMukPKG.js +0 -1
  299. package/web-dist/assets/FeedView-BfPIabGr.js +0 -6
  300. package/web-dist/assets/HistoryView-BmEEk3Rs.js +0 -1
  301. package/web-dist/assets/LoginView-D7LrkeX7.js +0 -1
  302. package/web-dist/assets/McpView-BAP_ah3T.js +0 -1
  303. package/web-dist/assets/SchedulesView-CAtsUPCZ.js +0 -6
  304. package/web-dist/assets/SettingsView-BovjWZDa.js +0 -1
  305. package/web-dist/assets/SkillsView-DQSMM5LN.js +0 -16
  306. package/web-dist/assets/SquadDetailView-DBscu0m2.js +0 -26
  307. package/web-dist/assets/SquadHealthView-D686BuQo.js +0 -11
  308. package/web-dist/assets/SquadsView-AzMht2NJ.js +0 -6
  309. package/web-dist/assets/ToggleSwitch.vue_vue_type_script_setup_true_lang-DtShZAjW.js +0 -1
  310. package/web-dist/assets/UsageView-DiVn97aI.js +0 -16
  311. package/web-dist/assets/WikiView-Cn7KipkZ.js +0 -26
  312. package/web-dist/assets/api-D4mHJ3u0.js +0 -1
  313. package/web-dist/assets/arrow-left-D_qUNUWW.js +0 -6
  314. package/web-dist/assets/git-branch-DOM-orcl.js +0 -6
  315. package/web-dist/assets/index-Bo83B1LR.css +0 -1
  316. package/web-dist/assets/index-ELvnkQjd.js +0 -273
  317. package/web-dist/assets/pencil-CFsi7ufI.js +0 -6
  318. package/web-dist/assets/plus-BAzlGFd_.js +0 -6
  319. package/web-dist/assets/save-BmgCYJ1g.js +0 -6
  320. package/web-dist/assets/search-CS9zSIeW.js +0 -6
  321. package/web-dist/assets/squad-colors-B8B_Y-lz.js +0 -1
  322. package/web-dist/assets/trash-2-DLveUEsd.js +0 -6
  323. package/web-dist/assets/triangle-alert-lj4I30rL.js +0 -6
  324. package/web-dist/assets/x-CjXR97Fa.js +0 -6
  325. package/web-dist/favicon.svg +0 -10
  326. package/web-dist/index.html +0 -14
  327. 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,3 @@
1
+ import { Router } from 'express';
2
+ export declare function activityRouter(): Router;
3
+ //# sourceMappingURL=activity.d.ts.map
@@ -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,3 @@
1
+ import { Router } from 'express';
2
+ export declare function attachmentsRouter(dataDir: string): Router;
3
+ //# sourceMappingURL=attachments.d.ts.map
@@ -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,3 @@
1
+ import { Router } from 'express';
2
+ export declare function configRouter(): Router;
3
+ //# sourceMappingURL=config.d.ts.map
@@ -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,3 @@
1
+ import { Router } from 'express';
2
+ export declare function conversationsRouter(): Router;
3
+ //# sourceMappingURL=conversations.d.ts.map
@@ -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":"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"}
@@ -0,0 +1,3 @@
1
+ import { Router } from 'express';
2
+ export declare function healthRouter(): Router;
3
+ //# sourceMappingURL=health.d.ts.map
@@ -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,3 @@
1
+ import { Router } from 'express';
2
+ export declare function inboxRouter(): Router;
3
+ //# sourceMappingURL=inbox.d.ts.map
@@ -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"}