agent-relay 1.1.0 → 1.2.3

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 (567) hide show
  1. package/.gitattributes +3 -0
  2. package/.nvmrc +1 -0
  3. package/.trajectories/completed/2026-01/traj_1dviorhnkcb5.json +65 -0
  4. package/.trajectories/completed/2026-01/traj_1dviorhnkcb5.md +37 -0
  5. package/.trajectories/completed/2026-01/traj_1k5if5snst2e.json +65 -0
  6. package/.trajectories/completed/2026-01/traj_1k5if5snst2e.md +37 -0
  7. package/.trajectories/completed/2026-01/traj_1rp3rges5811.json +49 -0
  8. package/.trajectories/completed/2026-01/traj_1rp3rges5811.md +31 -0
  9. package/.trajectories/completed/2026-01/traj_22bhyulruouw.json +113 -0
  10. package/.trajectories/completed/2026-01/traj_22bhyulruouw.md +57 -0
  11. package/.trajectories/completed/2026-01/traj_2dao7ddgnta0.json +53 -0
  12. package/.trajectories/completed/2026-01/traj_2dao7ddgnta0.md +32 -0
  13. package/.trajectories/completed/2026-01/traj_3t0440mjeunc.json +26 -0
  14. package/.trajectories/completed/2026-01/traj_3t0440mjeunc.md +6 -0
  15. package/.trajectories/completed/2026-01/traj_45x9494d9xnr.json +47 -0
  16. package/.trajectories/completed/2026-01/traj_45x9494d9xnr.md +32 -0
  17. package/.trajectories/completed/2026-01/traj_4aa0bb77s4nh.json +53 -0
  18. package/.trajectories/completed/2026-01/traj_4aa0bb77s4nh.md +32 -0
  19. package/.trajectories/completed/2026-01/traj_5lhmzq8rxpqv.json +59 -0
  20. package/.trajectories/completed/2026-01/traj_5lhmzq8rxpqv.md +33 -0
  21. package/.trajectories/completed/2026-01/traj_5vr4e9erb1fs.json +53 -0
  22. package/.trajectories/completed/2026-01/traj_5vr4e9erb1fs.md +32 -0
  23. package/.trajectories/completed/2026-01/traj_6fgiwdoklvym.json +48 -0
  24. package/.trajectories/completed/2026-01/traj_6fgiwdoklvym.md +24 -0
  25. package/.trajectories/completed/2026-01/traj_7ludwvz45veh.json +209 -0
  26. package/.trajectories/completed/2026-01/traj_7ludwvz45veh.md +97 -0
  27. package/.trajectories/completed/2026-01/traj_9921cuhel0pj.json +48 -0
  28. package/.trajectories/completed/2026-01/traj_9921cuhel0pj.md +24 -0
  29. package/.trajectories/completed/2026-01/traj_ajs7zqfux4wc.json +49 -0
  30. package/.trajectories/completed/2026-01/traj_ajs7zqfux4wc.md +23 -0
  31. package/.trajectories/completed/2026-01/traj_cvtqhlwcq9s0.json +53 -0
  32. package/.trajectories/completed/2026-01/traj_cvtqhlwcq9s0.md +32 -0
  33. package/.trajectories/completed/2026-01/traj_cxofprm2m2en.json +49 -0
  34. package/.trajectories/completed/2026-01/traj_cxofprm2m2en.md +31 -0
  35. package/.trajectories/completed/2026-01/traj_d2hhz3k0vrhn.json +26 -0
  36. package/.trajectories/completed/2026-01/traj_d2hhz3k0vrhn.md +6 -0
  37. package/.trajectories/completed/2026-01/traj_dfuvww9pege5.json +59 -0
  38. package/.trajectories/completed/2026-01/traj_dfuvww9pege5.md +37 -0
  39. package/.trajectories/completed/2026-01/traj_g0fisy9h51mf.json +77 -0
  40. package/.trajectories/completed/2026-01/traj_g0fisy9h51mf.md +42 -0
  41. package/.trajectories/completed/2026-01/traj_gjdre5voouod.json +53 -0
  42. package/.trajectories/completed/2026-01/traj_gjdre5voouod.md +32 -0
  43. package/.trajectories/completed/2026-01/traj_gtlyqtta3x8l.json +25 -0
  44. package/.trajectories/completed/2026-01/traj_gtlyqtta3x8l.md +15 -0
  45. package/.trajectories/completed/2026-01/traj_h4xijiuip3w4.json +101 -0
  46. package/.trajectories/completed/2026-01/traj_h4xijiuip3w4.md +44 -0
  47. package/.trajectories/completed/2026-01/traj_hhxte7w4gjjx.json +22 -0
  48. package/.trajectories/completed/2026-01/traj_hhxte7w4gjjx.md +5 -0
  49. package/.trajectories/completed/2026-01/traj_hpungyhoj6v5.json +53 -0
  50. package/.trajectories/completed/2026-01/traj_hpungyhoj6v5.md +32 -0
  51. package/.trajectories/completed/2026-01/traj_m2xkjv0w2sq7.json +25 -0
  52. package/.trajectories/completed/2026-01/traj_m2xkjv0w2sq7.md +15 -0
  53. package/.trajectories/completed/2026-01/traj_noq5zbvnrdvz.json +53 -0
  54. package/.trajectories/completed/2026-01/traj_noq5zbvnrdvz.md +32 -0
  55. package/.trajectories/completed/2026-01/traj_ntbs6ppopf46.json +53 -0
  56. package/.trajectories/completed/2026-01/traj_ntbs6ppopf46.md +32 -0
  57. package/.trajectories/completed/2026-01/traj_ozd98si6a7ns.json +48 -0
  58. package/.trajectories/completed/2026-01/traj_ozd98si6a7ns.md +24 -0
  59. package/.trajectories/completed/2026-01/traj_prdza7a5cxp5.json +53 -0
  60. package/.trajectories/completed/2026-01/traj_prdza7a5cxp5.md +32 -0
  61. package/.trajectories/completed/2026-01/traj_qb3twvvywfwi.json +77 -0
  62. package/.trajectories/completed/2026-01/traj_qb3twvvywfwi.md +42 -0
  63. package/.trajectories/completed/2026-01/traj_qft54mi7nfor.json +53 -0
  64. package/.trajectories/completed/2026-01/traj_qft54mi7nfor.md +32 -0
  65. package/.trajectories/completed/2026-01/traj_qx9uhf8whhxo.json +83 -0
  66. package/.trajectories/completed/2026-01/traj_qx9uhf8whhxo.md +47 -0
  67. package/.trajectories/completed/2026-01/traj_rd9toccj18a0.json +59 -0
  68. package/.trajectories/completed/2026-01/traj_rd9toccj18a0.md +37 -0
  69. package/.trajectories/completed/2026-01/traj_rt4fiw3ecp50.json +48 -0
  70. package/.trajectories/completed/2026-01/traj_rt4fiw3ecp50.md +16 -0
  71. package/.trajectories/completed/2026-01/traj_st8j35b0hrlc.json +59 -0
  72. package/.trajectories/completed/2026-01/traj_st8j35b0hrlc.md +37 -0
  73. package/.trajectories/completed/2026-01/traj_t1yy8m7hbuxp.json +53 -0
  74. package/.trajectories/completed/2026-01/traj_t1yy8m7hbuxp.md +32 -0
  75. package/.trajectories/completed/2026-01/traj_tmux_orchestrator_analysis.json +84 -0
  76. package/.trajectories/completed/2026-01/traj_tmux_orchestrator_analysis.md +109 -0
  77. package/.trajectories/completed/2026-01/traj_u9n9eqasw16k.json +53 -0
  78. package/.trajectories/completed/2026-01/traj_u9n9eqasw16k.md +32 -0
  79. package/.trajectories/completed/2026-01/traj_v87hypnongqx.json +71 -0
  80. package/.trajectories/completed/2026-01/traj_v87hypnongqx.md +42 -0
  81. package/.trajectories/completed/2026-01/traj_wkp2fgzdyinb.json +53 -0
  82. package/.trajectories/completed/2026-01/traj_wkp2fgzdyinb.md +32 -0
  83. package/.trajectories/completed/2026-01/traj_x14t8w8rn7xg.json +20 -0
  84. package/.trajectories/completed/2026-01/traj_x14t8w8rn7xg.md +6 -0
  85. package/.trajectories/completed/2026-01/traj_xnwbznkvv8ua.json +175 -0
  86. package/.trajectories/completed/2026-01/traj_xnwbznkvv8ua.md +82 -0
  87. package/.trajectories/completed/2026-01/traj_ysjc8zaeqtd3.json +47 -0
  88. package/.trajectories/completed/2026-01/traj_ysjc8zaeqtd3.md +32 -0
  89. package/.trajectories/completed/2026-01/traj_yvdadtvdgnz3.json +59 -0
  90. package/.trajectories/completed/2026-01/traj_yvdadtvdgnz3.md +37 -0
  91. package/.trajectories/completed/2026-01/traj_z0vcw1wrzide.json +53 -0
  92. package/.trajectories/completed/2026-01/traj_z0vcw1wrzide.md +32 -0
  93. package/.trajectories/index.json +314 -0
  94. package/ARCHITECTURE.md +1245 -0
  95. package/README.md +1 -1
  96. package/TESTING.md +278 -0
  97. package/deploy/init-db.sql +5 -0
  98. package/deploy/scripts/setup-fly-workspaces.sh +69 -0
  99. package/deploy/scripts/setup-railway.sh +75 -0
  100. package/deploy/workspace/entrypoint-browser.sh +118 -0
  101. package/deploy/workspace/entrypoint.sh +348 -0
  102. package/deploy/workspace/git-credential-relay +111 -0
  103. package/dist/bridge/spawner.d.ts +53 -0
  104. package/dist/bridge/spawner.js +203 -19
  105. package/dist/bridge/types.d.ts +12 -0
  106. package/dist/cli/index.js +618 -5
  107. package/dist/cloud/api/auth.d.ts +3 -2
  108. package/dist/cloud/api/auth.js +10 -98
  109. package/dist/cloud/api/billing.js +30 -9
  110. package/dist/cloud/api/cli-pty-runner.d.ts +54 -0
  111. package/dist/cloud/api/cli-pty-runner.js +119 -0
  112. package/dist/cloud/api/codex-auth-helper.d.ts +15 -0
  113. package/dist/cloud/api/codex-auth-helper.js +100 -0
  114. package/dist/cloud/api/generic-webhooks.d.ts +8 -0
  115. package/dist/cloud/api/generic-webhooks.js +129 -0
  116. package/dist/cloud/api/git.d.ts +8 -0
  117. package/dist/cloud/api/git.js +152 -0
  118. package/dist/cloud/api/github-app.d.ts +11 -0
  119. package/dist/cloud/api/github-app.js +189 -0
  120. package/dist/cloud/api/middleware/planLimits.d.ts +7 -0
  121. package/dist/cloud/api/middleware/planLimits.js +39 -1
  122. package/dist/cloud/api/monitoring.d.ts +11 -0
  123. package/dist/cloud/api/monitoring.js +578 -0
  124. package/dist/cloud/api/nango-auth.d.ts +9 -0
  125. package/dist/cloud/api/nango-auth.js +377 -0
  126. package/dist/cloud/api/onboarding.d.ts +8 -1
  127. package/dist/cloud/api/onboarding.js +313 -119
  128. package/dist/cloud/api/policy.d.ts +8 -0
  129. package/dist/cloud/api/policy.js +229 -0
  130. package/dist/cloud/api/providers.js +114 -42
  131. package/dist/cloud/api/repos.d.ts +1 -0
  132. package/dist/cloud/api/repos.js +186 -0
  133. package/dist/cloud/api/test-helpers.d.ts +10 -0
  134. package/dist/cloud/api/test-helpers.js +575 -0
  135. package/dist/cloud/api/webhooks.d.ts +8 -0
  136. package/dist/cloud/api/webhooks.js +645 -0
  137. package/dist/cloud/api/workspaces.js +320 -12
  138. package/dist/cloud/billing/plans.js +32 -19
  139. package/dist/cloud/billing/types.d.ts +9 -3
  140. package/dist/cloud/config.d.ts +9 -2
  141. package/dist/cloud/config.js +13 -4
  142. package/dist/cloud/db/drizzle.d.ts +84 -1
  143. package/dist/cloud/db/drizzle.js +470 -0
  144. package/dist/cloud/db/index.d.ts +9 -4
  145. package/dist/cloud/db/index.js +11 -3
  146. package/dist/cloud/db/schema.d.ts +3283 -556
  147. package/dist/cloud/db/schema.js +314 -1
  148. package/dist/cloud/index.d.ts +1 -0
  149. package/dist/cloud/index.js +2 -0
  150. package/dist/cloud/provisioner/index.d.ts +56 -0
  151. package/dist/cloud/provisioner/index.js +676 -34
  152. package/dist/cloud/server.d.ts +1 -0
  153. package/dist/cloud/server.js +362 -13
  154. package/dist/cloud/services/auto-scaler.d.ts +152 -0
  155. package/dist/cloud/services/auto-scaler.js +439 -0
  156. package/dist/cloud/services/capacity-manager.d.ts +148 -0
  157. package/dist/cloud/services/capacity-manager.js +449 -0
  158. package/dist/cloud/services/ci-agent-spawner.d.ts +49 -0
  159. package/dist/cloud/services/ci-agent-spawner.js +373 -0
  160. package/dist/cloud/services/index.d.ts +12 -0
  161. package/dist/cloud/services/index.js +15 -0
  162. package/dist/cloud/services/mention-handler.d.ts +65 -0
  163. package/dist/cloud/services/mention-handler.js +405 -0
  164. package/dist/cloud/services/nango.d.ts +186 -0
  165. package/dist/cloud/services/nango.js +344 -0
  166. package/dist/cloud/services/persistence.d.ts +131 -0
  167. package/dist/cloud/services/persistence.js +200 -0
  168. package/dist/cloud/services/planLimits.d.ts +37 -0
  169. package/dist/cloud/services/planLimits.js +86 -5
  170. package/dist/cloud/services/scaling-orchestrator.d.ts +159 -0
  171. package/dist/cloud/services/scaling-orchestrator.js +502 -0
  172. package/dist/cloud/services/scaling-policy.d.ts +121 -0
  173. package/dist/cloud/services/scaling-policy.js +415 -0
  174. package/dist/cloud/vault/index.js +1 -1
  175. package/dist/cloud/webhooks/index.d.ts +24 -0
  176. package/dist/cloud/webhooks/index.js +29 -0
  177. package/dist/cloud/webhooks/parsers/github.d.ts +8 -0
  178. package/dist/cloud/webhooks/parsers/github.js +234 -0
  179. package/dist/cloud/webhooks/parsers/index.d.ts +23 -0
  180. package/dist/cloud/webhooks/parsers/index.js +30 -0
  181. package/dist/cloud/webhooks/parsers/linear.d.ts +9 -0
  182. package/dist/cloud/webhooks/parsers/linear.js +258 -0
  183. package/dist/cloud/webhooks/parsers/slack.d.ts +9 -0
  184. package/dist/cloud/webhooks/parsers/slack.js +214 -0
  185. package/dist/cloud/webhooks/responders/github.d.ts +8 -0
  186. package/dist/cloud/webhooks/responders/github.js +73 -0
  187. package/dist/cloud/webhooks/responders/index.d.ts +23 -0
  188. package/dist/cloud/webhooks/responders/index.js +30 -0
  189. package/dist/cloud/webhooks/responders/linear.d.ts +9 -0
  190. package/dist/cloud/webhooks/responders/linear.js +149 -0
  191. package/dist/cloud/webhooks/responders/slack.d.ts +20 -0
  192. package/dist/cloud/webhooks/responders/slack.js +178 -0
  193. package/dist/cloud/webhooks/router.d.ts +25 -0
  194. package/dist/cloud/webhooks/router.js +504 -0
  195. package/dist/cloud/webhooks/rules-engine.d.ts +24 -0
  196. package/dist/cloud/webhooks/rules-engine.js +287 -0
  197. package/dist/cloud/webhooks/types.d.ts +186 -0
  198. package/dist/cloud/webhooks/types.js +8 -0
  199. package/dist/continuity/formatter.d.ts +51 -0
  200. package/dist/continuity/formatter.js +313 -0
  201. package/dist/continuity/handoff-store.d.ts +67 -0
  202. package/dist/continuity/handoff-store.js +472 -0
  203. package/dist/continuity/index.d.ts +45 -0
  204. package/dist/continuity/index.js +48 -0
  205. package/dist/continuity/ledger-store.d.ts +110 -0
  206. package/dist/continuity/ledger-store.js +500 -0
  207. package/dist/continuity/manager.d.ts +178 -0
  208. package/dist/continuity/manager.js +562 -0
  209. package/dist/continuity/parser.d.ts +76 -0
  210. package/dist/continuity/parser.js +579 -0
  211. package/dist/continuity/types.d.ts +180 -0
  212. package/dist/continuity/types.js +9 -0
  213. package/dist/daemon/agent-manager.d.ts +27 -0
  214. package/dist/daemon/agent-manager.js +107 -6
  215. package/dist/daemon/agent-registry.d.ts +32 -0
  216. package/dist/daemon/agent-registry.js +42 -2
  217. package/dist/daemon/api.d.ts +12 -0
  218. package/dist/daemon/api.js +131 -2
  219. package/dist/daemon/cli-auth.d.ts +67 -0
  220. package/dist/daemon/cli-auth.js +537 -0
  221. package/dist/daemon/cloud-sync.js +9 -7
  222. package/dist/daemon/orchestrator.js +30 -0
  223. package/dist/daemon/router.d.ts +5 -0
  224. package/dist/daemon/router.js +78 -26
  225. package/dist/daemon/server.d.ts +5 -0
  226. package/dist/daemon/server.js +9 -1
  227. package/dist/daemon/services/browser-testing.d.ts +88 -0
  228. package/dist/daemon/services/browser-testing.js +244 -0
  229. package/dist/daemon/services/container-spawner.d.ts +135 -0
  230. package/dist/daemon/services/container-spawner.js +313 -0
  231. package/dist/daemon/types.d.ts +5 -1
  232. package/dist/dashboard/out/404.html +1 -1
  233. package/dist/dashboard/out/_next/static/chunks/116-2502180def231162.js +1 -0
  234. package/dist/dashboard/out/_next/static/chunks/282-980c2eb8fff20123.js +1 -0
  235. package/dist/dashboard/out/_next/static/chunks/699-3b1cd6618a45d259.js +1 -0
  236. package/dist/dashboard/out/_next/static/chunks/724-2dae7627550ab88f.js +9 -0
  237. package/dist/dashboard/out/_next/static/chunks/766-1f2dd8cb7f766b0b.js +1 -0
  238. package/dist/dashboard/out/_next/static/chunks/app/app/onboarding/page-3fdfa60e53f2810d.js +1 -0
  239. package/dist/dashboard/out/_next/static/chunks/app/app/page-e6381e5a6e1fbcfd.js +1 -0
  240. package/dist/dashboard/out/_next/static/chunks/app/connect-repos/page-3538dfe0ffe984b8.js +1 -0
  241. package/dist/dashboard/out/_next/static/chunks/app/history/{page-b6edd4dde8d08194.js → page-abb9ab2d329f56e9.js} +1 -1
  242. package/dist/dashboard/out/_next/static/chunks/app/layout-c0d118c0f92d969c.js +1 -0
  243. package/dist/dashboard/out/_next/static/chunks/app/login/page-c22d080201cbd9fb.js +1 -0
  244. package/dist/dashboard/out/_next/static/chunks/app/metrics/page-67a3e98d9a43a6ed.js +1 -0
  245. package/dist/dashboard/out/_next/static/chunks/app/page-77e9c65420a06cfb.js +1 -0
  246. package/dist/dashboard/out/_next/static/chunks/app/pricing/page-b08ed1c34d14434a.js +1 -0
  247. package/dist/dashboard/out/_next/static/chunks/app/providers/page-e88bc117ef7671c3.js +1 -0
  248. package/dist/dashboard/out/_next/static/chunks/app/signup/page-68d34f50baa8ab6b.js +1 -0
  249. package/dist/dashboard/out/_next/static/chunks/e868780c-48e5f147c90a3a41.js +18 -0
  250. package/dist/dashboard/out/_next/static/chunks/{main-app-5d692157a8eb1fd9.js → main-app-6e8e8d3ef4e0192a.js} +1 -1
  251. package/dist/dashboard/out/_next/static/chunks/{main-c2f423b9c9f4591b.js → main-ed4e1fb6f29c34cf.js} +1 -1
  252. package/dist/dashboard/out/_next/static/chunks/webpack-1cdd8ed57114d5e1.js +1 -0
  253. package/dist/dashboard/out/_next/static/css/29852f26181969a0.css +1 -0
  254. package/dist/dashboard/out/_next/static/css/7c3ae9e8617d42a5.css +1 -0
  255. package/dist/dashboard/out/app/onboarding.html +1 -0
  256. package/dist/dashboard/out/app/onboarding.txt +7 -0
  257. package/dist/dashboard/out/app.html +1 -14
  258. package/dist/dashboard/out/app.txt +2 -2
  259. package/dist/dashboard/out/connect-repos.html +1 -0
  260. package/dist/dashboard/out/connect-repos.txt +7 -0
  261. package/dist/dashboard/out/history.html +1 -1
  262. package/dist/dashboard/out/history.txt +2 -2
  263. package/dist/dashboard/out/index.html +1 -1
  264. package/dist/dashboard/out/index.txt +2 -2
  265. package/dist/dashboard/out/login.html +6 -0
  266. package/dist/dashboard/out/login.txt +7 -0
  267. package/dist/dashboard/out/metrics.html +1 -1
  268. package/dist/dashboard/out/metrics.txt +2 -2
  269. package/dist/dashboard/out/pricing.html +3 -3
  270. package/dist/dashboard/out/pricing.txt +2 -2
  271. package/dist/dashboard/out/providers.html +1 -0
  272. package/dist/dashboard/out/providers.txt +7 -0
  273. package/dist/dashboard/out/signup.html +6 -0
  274. package/dist/dashboard/out/signup.txt +7 -0
  275. package/dist/dashboard-server/server.js +1308 -8
  276. package/dist/hooks/emitter.d.ts +40 -0
  277. package/dist/hooks/emitter.js +63 -0
  278. package/dist/hooks/index.d.ts +3 -0
  279. package/dist/hooks/index.js +3 -0
  280. package/dist/hooks/registry.d.ts +173 -0
  281. package/dist/hooks/registry.js +476 -0
  282. package/dist/hooks/trajectory-hooks.d.ts +52 -0
  283. package/dist/hooks/trajectory-hooks.js +183 -0
  284. package/dist/hooks/types.d.ts +141 -0
  285. package/dist/index.d.ts +2 -0
  286. package/dist/index.js +3 -0
  287. package/dist/memory/adapters/index.d.ts +8 -0
  288. package/dist/memory/adapters/index.js +8 -0
  289. package/dist/memory/adapters/inmemory.d.ts +59 -0
  290. package/dist/memory/adapters/inmemory.js +195 -0
  291. package/dist/memory/adapters/supermemory.d.ts +71 -0
  292. package/dist/memory/adapters/supermemory.js +338 -0
  293. package/dist/memory/factory.d.ts +48 -0
  294. package/dist/memory/factory.js +143 -0
  295. package/dist/memory/index.d.ts +32 -0
  296. package/dist/memory/index.js +32 -0
  297. package/dist/memory/memory-hooks.d.ts +60 -0
  298. package/dist/memory/memory-hooks.js +313 -0
  299. package/dist/memory/service.d.ts +49 -0
  300. package/dist/memory/service.js +146 -0
  301. package/dist/memory/types.d.ts +195 -0
  302. package/dist/memory/types.js +8 -0
  303. package/dist/policy/agent-policy.d.ts +225 -0
  304. package/dist/policy/agent-policy.js +665 -0
  305. package/dist/policy/cloud-policy-fetcher.d.ts +12 -0
  306. package/dist/policy/cloud-policy-fetcher.js +64 -0
  307. package/dist/resiliency/crash-insights.d.ts +156 -0
  308. package/dist/resiliency/crash-insights.js +492 -0
  309. package/dist/resiliency/gossip-health.d.ts +137 -0
  310. package/dist/resiliency/gossip-health.js +241 -0
  311. package/dist/resiliency/index.d.ts +5 -0
  312. package/dist/resiliency/index.js +5 -0
  313. package/dist/resiliency/leader-watchdog.d.ts +109 -0
  314. package/dist/resiliency/leader-watchdog.js +189 -0
  315. package/dist/resiliency/memory-monitor.d.ts +172 -0
  316. package/dist/resiliency/memory-monitor.js +593 -0
  317. package/dist/resiliency/stateless-lead.d.ts +149 -0
  318. package/dist/resiliency/stateless-lead.js +308 -0
  319. package/dist/resiliency/supervisor.d.ts +38 -0
  320. package/dist/resiliency/supervisor.js +122 -0
  321. package/dist/shared/cli-auth-config.d.ts +91 -0
  322. package/dist/shared/cli-auth-config.js +264 -0
  323. package/dist/storage/adapter.d.ts +1 -1
  324. package/dist/trajectory/config.d.ts +84 -0
  325. package/dist/trajectory/config.js +163 -0
  326. package/dist/trajectory/index.d.ts +8 -0
  327. package/dist/trajectory/index.js +8 -0
  328. package/dist/trajectory/integration.d.ts +292 -0
  329. package/dist/trajectory/integration.js +834 -0
  330. package/dist/utils/logger.js +1 -1
  331. package/dist/utils/project-namespace.d.ts +24 -0
  332. package/dist/utils/project-namespace.js +84 -0
  333. package/dist/wrapper/parser.d.ts +10 -0
  334. package/dist/wrapper/parser.js +100 -33
  335. package/dist/wrapper/pty-wrapper.d.ts +197 -16
  336. package/dist/wrapper/pty-wrapper.js +943 -106
  337. package/dist/wrapper/shared.d.ts +165 -0
  338. package/dist/wrapper/shared.js +270 -0
  339. package/dist/wrapper/tmux-wrapper.d.ts +73 -11
  340. package/dist/wrapper/tmux-wrapper.js +541 -120
  341. package/package.json +16 -16
  342. package/scripts/postinstall.js +60 -0
  343. package/test-push.txt +1 -0
  344. package/bin/tmux +0 -0
  345. package/dist/bridge/config.d.ts.map +0 -1
  346. package/dist/bridge/config.js.map +0 -1
  347. package/dist/bridge/index.d.ts.map +0 -1
  348. package/dist/bridge/index.js.map +0 -1
  349. package/dist/bridge/multi-project-client.d.ts.map +0 -1
  350. package/dist/bridge/multi-project-client.js.map +0 -1
  351. package/dist/bridge/shadow-cli.d.ts.map +0 -1
  352. package/dist/bridge/shadow-cli.js.map +0 -1
  353. package/dist/bridge/shadow-config.d.ts.map +0 -1
  354. package/dist/bridge/shadow-config.js.map +0 -1
  355. package/dist/bridge/spawner.d.ts.map +0 -1
  356. package/dist/bridge/spawner.js.map +0 -1
  357. package/dist/bridge/teams-config.d.ts.map +0 -1
  358. package/dist/bridge/teams-config.js.map +0 -1
  359. package/dist/bridge/types.d.ts.map +0 -1
  360. package/dist/bridge/types.js.map +0 -1
  361. package/dist/bridge/utils.d.ts.map +0 -1
  362. package/dist/bridge/utils.js.map +0 -1
  363. package/dist/cli/index.d.ts.map +0 -1
  364. package/dist/cli/index.js.map +0 -1
  365. package/dist/cloud/api/auth.d.ts.map +0 -1
  366. package/dist/cloud/api/auth.js.map +0 -1
  367. package/dist/cloud/api/billing.d.ts.map +0 -1
  368. package/dist/cloud/api/billing.js.map +0 -1
  369. package/dist/cloud/api/coordinators.d.ts.map +0 -1
  370. package/dist/cloud/api/coordinators.js.map +0 -1
  371. package/dist/cloud/api/daemons.d.ts.map +0 -1
  372. package/dist/cloud/api/daemons.js.map +0 -1
  373. package/dist/cloud/api/middleware/planLimits.d.ts.map +0 -1
  374. package/dist/cloud/api/middleware/planLimits.js.map +0 -1
  375. package/dist/cloud/api/onboarding.d.ts.map +0 -1
  376. package/dist/cloud/api/onboarding.js.map +0 -1
  377. package/dist/cloud/api/providers.d.ts.map +0 -1
  378. package/dist/cloud/api/providers.js.map +0 -1
  379. package/dist/cloud/api/repos.d.ts.map +0 -1
  380. package/dist/cloud/api/repos.js.map +0 -1
  381. package/dist/cloud/api/teams.d.ts.map +0 -1
  382. package/dist/cloud/api/teams.js.map +0 -1
  383. package/dist/cloud/api/usage.d.ts.map +0 -1
  384. package/dist/cloud/api/usage.js.map +0 -1
  385. package/dist/cloud/api/workspaces.d.ts.map +0 -1
  386. package/dist/cloud/api/workspaces.js.map +0 -1
  387. package/dist/cloud/billing/index.d.ts.map +0 -1
  388. package/dist/cloud/billing/index.js.map +0 -1
  389. package/dist/cloud/billing/plans.d.ts.map +0 -1
  390. package/dist/cloud/billing/plans.js.map +0 -1
  391. package/dist/cloud/billing/service.d.ts.map +0 -1
  392. package/dist/cloud/billing/service.js.map +0 -1
  393. package/dist/cloud/billing/types.d.ts.map +0 -1
  394. package/dist/cloud/billing/types.js.map +0 -1
  395. package/dist/cloud/config.d.ts.map +0 -1
  396. package/dist/cloud/config.js.map +0 -1
  397. package/dist/cloud/db/drizzle.d.ts.map +0 -1
  398. package/dist/cloud/db/drizzle.js.map +0 -1
  399. package/dist/cloud/db/index.d.ts.map +0 -1
  400. package/dist/cloud/db/index.js.map +0 -1
  401. package/dist/cloud/db/schema.d.ts.map +0 -1
  402. package/dist/cloud/db/schema.js.map +0 -1
  403. package/dist/cloud/index.d.ts.map +0 -1
  404. package/dist/cloud/index.js.map +0 -1
  405. package/dist/cloud/provisioner/index.d.ts.map +0 -1
  406. package/dist/cloud/provisioner/index.js.map +0 -1
  407. package/dist/cloud/server.d.ts.map +0 -1
  408. package/dist/cloud/server.js.map +0 -1
  409. package/dist/cloud/services/coordinator.d.ts.map +0 -1
  410. package/dist/cloud/services/coordinator.js.map +0 -1
  411. package/dist/cloud/services/planLimits.d.ts.map +0 -1
  412. package/dist/cloud/services/planLimits.js.map +0 -1
  413. package/dist/cloud/vault/index.d.ts.map +0 -1
  414. package/dist/cloud/vault/index.js.map +0 -1
  415. package/dist/daemon/agent-manager.d.ts.map +0 -1
  416. package/dist/daemon/agent-manager.js.map +0 -1
  417. package/dist/daemon/agent-registry.d.ts.map +0 -1
  418. package/dist/daemon/agent-registry.js.map +0 -1
  419. package/dist/daemon/api.d.ts.map +0 -1
  420. package/dist/daemon/api.js.map +0 -1
  421. package/dist/daemon/auth.d.ts.map +0 -1
  422. package/dist/daemon/auth.js.map +0 -1
  423. package/dist/daemon/cloud-sync.d.ts.map +0 -1
  424. package/dist/daemon/cloud-sync.js.map +0 -1
  425. package/dist/daemon/connection.d.ts.map +0 -1
  426. package/dist/daemon/connection.js.map +0 -1
  427. package/dist/daemon/index.d.ts.map +0 -1
  428. package/dist/daemon/index.js.map +0 -1
  429. package/dist/daemon/orchestrator.d.ts.map +0 -1
  430. package/dist/daemon/orchestrator.js.map +0 -1
  431. package/dist/daemon/registry.d.ts.map +0 -1
  432. package/dist/daemon/registry.js.map +0 -1
  433. package/dist/daemon/router.d.ts.map +0 -1
  434. package/dist/daemon/router.js.map +0 -1
  435. package/dist/daemon/server.d.ts.map +0 -1
  436. package/dist/daemon/server.js.map +0 -1
  437. package/dist/daemon/types.d.ts.map +0 -1
  438. package/dist/daemon/types.js.map +0 -1
  439. package/dist/daemon/workspace-manager.d.ts.map +0 -1
  440. package/dist/daemon/workspace-manager.js.map +0 -1
  441. package/dist/dashboard/out/_next/static/chunks/693-7b3301d8f6bc5014.js +0 -1
  442. package/dist/dashboard/out/_next/static/chunks/713-f78477eb185f1f4d.js +0 -1
  443. package/dist/dashboard/out/_next/static/chunks/766-e53e1cfe39b0b5b5.js +0 -1
  444. package/dist/dashboard/out/_next/static/chunks/900-037c64bfd797fb2a.js +0 -1
  445. package/dist/dashboard/out/_next/static/chunks/app/app/page-e3d9e1f4466b9bae.js +0 -1
  446. package/dist/dashboard/out/_next/static/chunks/app/layout-2433bb48965f4333.js +0 -1
  447. package/dist/dashboard/out/_next/static/chunks/app/metrics/page-e68825a81db67ba1.js +0 -1
  448. package/dist/dashboard/out/_next/static/chunks/app/page-cc108bf68c8a657f.js +0 -1
  449. package/dist/dashboard/out/_next/static/chunks/app/pricing/page-d80e03a5297f95b6.js +0 -1
  450. package/dist/dashboard/out/_next/static/chunks/webpack-a5acc2831d094776.js +0 -1
  451. package/dist/dashboard/out/_next/static/css/79b80143647a07d7.css +0 -1
  452. package/dist/dashboard/out/_next/static/css/8cf277370ad48cfe.css +0 -1
  453. package/dist/dashboard-server/metrics.d.ts.map +0 -1
  454. package/dist/dashboard-server/metrics.js.map +0 -1
  455. package/dist/dashboard-server/needs-attention.d.ts.map +0 -1
  456. package/dist/dashboard-server/needs-attention.js.map +0 -1
  457. package/dist/dashboard-server/server.d.ts.map +0 -1
  458. package/dist/dashboard-server/server.js.map +0 -1
  459. package/dist/dashboard-server/start.d.ts.map +0 -1
  460. package/dist/dashboard-server/start.js.map +0 -1
  461. package/dist/hooks/inbox-check/hook.d.ts.map +0 -1
  462. package/dist/hooks/inbox-check/hook.js.map +0 -1
  463. package/dist/hooks/inbox-check/index.d.ts.map +0 -1
  464. package/dist/hooks/inbox-check/index.js.map +0 -1
  465. package/dist/hooks/inbox-check/types.d.ts.map +0 -1
  466. package/dist/hooks/inbox-check/types.js.map +0 -1
  467. package/dist/hooks/inbox-check/utils.d.ts.map +0 -1
  468. package/dist/hooks/inbox-check/utils.js.map +0 -1
  469. package/dist/hooks/index.d.ts.map +0 -1
  470. package/dist/hooks/index.js.map +0 -1
  471. package/dist/hooks/types.d.ts.map +0 -1
  472. package/dist/hooks/types.js.map +0 -1
  473. package/dist/index.d.ts.map +0 -1
  474. package/dist/index.js.map +0 -1
  475. package/dist/protocol/framing.d.ts.map +0 -1
  476. package/dist/protocol/framing.js.map +0 -1
  477. package/dist/protocol/index.d.ts.map +0 -1
  478. package/dist/protocol/index.js.map +0 -1
  479. package/dist/protocol/types.d.ts.map +0 -1
  480. package/dist/protocol/types.js.map +0 -1
  481. package/dist/resiliency/context-persistence.d.ts.map +0 -1
  482. package/dist/resiliency/context-persistence.js.map +0 -1
  483. package/dist/resiliency/health-monitor.d.ts.map +0 -1
  484. package/dist/resiliency/health-monitor.js.map +0 -1
  485. package/dist/resiliency/index.d.ts.map +0 -1
  486. package/dist/resiliency/index.js.map +0 -1
  487. package/dist/resiliency/logger.d.ts.map +0 -1
  488. package/dist/resiliency/logger.js.map +0 -1
  489. package/dist/resiliency/metrics.d.ts.map +0 -1
  490. package/dist/resiliency/metrics.js.map +0 -1
  491. package/dist/resiliency/provider-context.d.ts.map +0 -1
  492. package/dist/resiliency/provider-context.js.map +0 -1
  493. package/dist/resiliency/supervisor.d.ts.map +0 -1
  494. package/dist/resiliency/supervisor.js.map +0 -1
  495. package/dist/state/agent-state.d.ts.map +0 -1
  496. package/dist/state/agent-state.js.map +0 -1
  497. package/dist/storage/adapter.d.ts.map +0 -1
  498. package/dist/storage/adapter.js.map +0 -1
  499. package/dist/storage/sqlite-adapter.d.ts.map +0 -1
  500. package/dist/storage/sqlite-adapter.js.map +0 -1
  501. package/dist/utils/agent-config.d.ts.map +0 -1
  502. package/dist/utils/agent-config.js.map +0 -1
  503. package/dist/utils/command-resolver.d.ts.map +0 -1
  504. package/dist/utils/command-resolver.js.map +0 -1
  505. package/dist/utils/index.d.ts.map +0 -1
  506. package/dist/utils/index.js.map +0 -1
  507. package/dist/utils/logger.d.ts.map +0 -1
  508. package/dist/utils/logger.js.map +0 -1
  509. package/dist/utils/name-generator.d.ts.map +0 -1
  510. package/dist/utils/name-generator.js.map +0 -1
  511. package/dist/utils/project-namespace.d.ts.map +0 -1
  512. package/dist/utils/project-namespace.js.map +0 -1
  513. package/dist/utils/tmux-resolver.d.ts.map +0 -1
  514. package/dist/utils/tmux-resolver.js.map +0 -1
  515. package/dist/utils/update-checker.d.ts.map +0 -1
  516. package/dist/utils/update-checker.js.map +0 -1
  517. package/dist/wrapper/client.d.ts.map +0 -1
  518. package/dist/wrapper/client.js.map +0 -1
  519. package/dist/wrapper/inbox.d.ts.map +0 -1
  520. package/dist/wrapper/inbox.js.map +0 -1
  521. package/dist/wrapper/index.d.ts.map +0 -1
  522. package/dist/wrapper/index.js.map +0 -1
  523. package/dist/wrapper/parser.d.ts.map +0 -1
  524. package/dist/wrapper/parser.js.map +0 -1
  525. package/dist/wrapper/pty-wrapper.d.ts.map +0 -1
  526. package/dist/wrapper/pty-wrapper.js.map +0 -1
  527. package/dist/wrapper/tmux-wrapper.d.ts.map +0 -1
  528. package/dist/wrapper/tmux-wrapper.js.map +0 -1
  529. package/docs/AGENTS.md +0 -513
  530. package/docs/ARCHITECTURE_DECISIONS.md +0 -175
  531. package/docs/CHANGELOG.md +0 -11
  532. package/docs/CLI-SIMPLIFICATION-COMPLETE.md +0 -48
  533. package/docs/CLOUD-ARCHITECTURE.md +0 -652
  534. package/docs/CLOUD-ONBOARDING-DESIGN.md +0 -1983
  535. package/docs/COMPETITIVE_ANALYSIS.md +0 -897
  536. package/docs/CONTRIBUTING.md +0 -151
  537. package/docs/DESIGN_BRIDGE_STAFFING.md +0 -878
  538. package/docs/DESIGN_V2.md +0 -1079
  539. package/docs/INTEGRATION-GUIDE.md +0 -926
  540. package/docs/MONETIZATION.md +0 -1679
  541. package/docs/PROPOSAL-trajectories.md +0 -1582
  542. package/docs/PROTOCOL.md +0 -325
  543. package/docs/SCALING_ANALYSIS.md +0 -280
  544. package/docs/TESTING_PRESENCE_FEATURES.md +0 -327
  545. package/docs/TMUX_IMPLEMENTATION_NOTES.md +0 -364
  546. package/docs/TMUX_IMPROVEMENTS.md +0 -968
  547. package/docs/agent-relay-snippet.md +0 -168
  548. package/docs/competitive-analysis-mcp-agent-mail.md +0 -389
  549. package/docs/dashboard-v2-plan.md +0 -179
  550. package/docs/guides/CLOUD.md +0 -236
  551. package/docs/guides/LOCAL.md +0 -535
  552. package/docs/guides/SELF-HOSTED.md +0 -494
  553. package/docs/proposals/shadow-as-subagent.md +0 -765
  554. package/docs/proposals/slack-bot-integration.md +0 -1457
  555. package/docs/removable-code-analysis.md +0 -24
  556. package/scripts/dev/PUBLIC_RELEASE_PLAN.md +0 -88
  557. package/scripts/dev/dev-team-setup.sh +0 -431
  558. package/scripts/e2e-test.sh +0 -119
  559. package/scripts/games/game-protocol.md +0 -79
  560. package/scripts/games/hearts-setup.sh +0 -264
  561. package/scripts/tictactoe-setup.sh +0 -181
  562. /package/dist/dashboard/out/_next/static/chunks/{117-b2cd8d6485aacf2b.js → 117-f7b8ab0809342e77.js} +0 -0
  563. /package/dist/dashboard/out/_next/static/chunks/{648-8f3f26864ce515e5.js → 648-5cc6e1921389a58a.js} +0 -0
  564. /package/dist/dashboard/out/_next/static/chunks/app/_not-found/{page-0b990dbb71d72a98.js → page-53b8a69f76db17d0.js} +0 -0
  565. /package/dist/dashboard/out/_next/static/chunks/{fd9d1056-bf46c09eb57e019c.js → fd9d1056-609918ca7b6280bb.js} +0 -0
  566. /package/dist/dashboard/out/_next/static/{6HHWb2ZmnJ4OSm0zUP7h4 → wPgKJtcOmTFLpUncDg16A}/_buildManifest.js +0 -0
  567. /package/dist/dashboard/out/_next/static/{6HHWb2ZmnJ4OSm0zUP7h4 → wPgKJtcOmTFLpUncDg16A}/_ssgManifest.js +0 -0
@@ -0,0 +1,645 @@
1
+ /**
2
+ * Webhook API Routes
3
+ *
4
+ * Handles GitHub App webhooks for installation events.
5
+ * Also provides workspace webhook forwarding for external integrations.
6
+ */
7
+ import { Router } from 'express';
8
+ import crypto from 'crypto';
9
+ import { getConfig } from '../config.js';
10
+ import { db } from '../db/index.js';
11
+ export const webhooksRouter = Router();
12
+ // ============================================================================
13
+ // Workspace Webhook Forwarding
14
+ // ============================================================================
15
+ /**
16
+ * Convert workspace public URL to internal Fly.io URL
17
+ */
18
+ function getWorkspaceInternalUrl(publicUrl) {
19
+ const isOnFly = !!process.env.FLY_APP_NAME;
20
+ let url = publicUrl.replace(/\/$/, '');
21
+ if (isOnFly && url.includes('.fly.dev')) {
22
+ // Use Fly.io internal networking
23
+ // ar-583f273b.fly.dev -> http://ar-583f273b.internal:3888
24
+ // .internal uses IPv6 and works by default for apps in the same org
25
+ const appName = url.match(/https?:\/\/([^.]+)\.fly\.dev/)?.[1];
26
+ if (appName) {
27
+ url = `http://${appName}.internal:3888`;
28
+ }
29
+ }
30
+ return url;
31
+ }
32
+ /**
33
+ * Wake a suspended Fly.io workspace machine
34
+ */
35
+ async function wakeWorkspaceMachine(workspaceId) {
36
+ const workspace = await db.workspaces.findById(workspaceId);
37
+ if (!workspace?.computeId)
38
+ return false;
39
+ const appName = `ar-${workspaceId.substring(0, 8)}`;
40
+ const apiToken = process.env.FLY_API_TOKEN;
41
+ if (!apiToken) {
42
+ console.warn('[webhooks] FLY_API_TOKEN not set, cannot wake machine');
43
+ return false;
44
+ }
45
+ try {
46
+ const response = await fetch(`https://api.machines.dev/v1/apps/${appName}/machines/${workspace.computeId}/start`, {
47
+ method: 'POST',
48
+ headers: { Authorization: `Bearer ${apiToken}` },
49
+ });
50
+ if (response.ok) {
51
+ console.log(`[webhooks] Started workspace machine ${workspace.computeId}`);
52
+ // Wait a bit for machine to start
53
+ await new Promise(resolve => setTimeout(resolve, 5000));
54
+ return true;
55
+ }
56
+ // 200 OK means already running
57
+ if (response.status === 200) {
58
+ return true;
59
+ }
60
+ console.warn(`[webhooks] Failed to start machine: ${response.status}`);
61
+ return false;
62
+ }
63
+ catch (error) {
64
+ console.error('[webhooks] Error waking machine:', error);
65
+ return false;
66
+ }
67
+ }
68
+ /**
69
+ * POST /api/webhooks/workspace/:workspaceId/*
70
+ * Forward webhooks to a specific workspace
71
+ *
72
+ * External services can send webhooks to:
73
+ * https://agent-relay.com/api/webhooks/workspace/{workspaceId}/your/path
74
+ *
75
+ * This will be forwarded to:
76
+ * http://{workspace-internal}/webhooks/your/path
77
+ */
78
+ // Handler for workspace webhook forwarding - matches any path under /workspace/:workspaceId
79
+ async function handleWorkspaceWebhook(req, res) {
80
+ // Extract workspaceId from URL path
81
+ const pathMatch = req.originalUrl.match(/\/workspace\/([^/]+)\/?(.*)$/);
82
+ if (!pathMatch) {
83
+ res.status(400).json({ error: 'Invalid workspace webhook URL' });
84
+ return;
85
+ }
86
+ const workspaceId = pathMatch[1];
87
+ const forwardPath = pathMatch[2] || '';
88
+ console.log(`[webhooks] Forwarding to workspace ${workspaceId}: ${req.method} /${forwardPath}`);
89
+ try {
90
+ // Find the workspace
91
+ const workspace = await db.workspaces.findById(workspaceId);
92
+ if (!workspace) {
93
+ res.status(404).json({ error: 'Workspace not found' });
94
+ return;
95
+ }
96
+ if (!workspace.publicUrl) {
97
+ res.status(400).json({ error: 'Workspace has no public URL' });
98
+ return;
99
+ }
100
+ // Try to wake the machine if it might be suspended
101
+ if (workspace.status === 'running' || workspace.status === 'suspended') {
102
+ await wakeWorkspaceMachine(workspaceId);
103
+ }
104
+ // Get internal URL for server-to-server communication
105
+ const internalUrl = getWorkspaceInternalUrl(workspace.publicUrl);
106
+ const targetUrl = `${internalUrl}/webhooks/${forwardPath}`;
107
+ console.log(`[webhooks] Forwarding to: ${targetUrl}`);
108
+ // Forward the request with original headers and body
109
+ const forwardHeaders = {};
110
+ // Copy relevant headers
111
+ const headersToForward = [
112
+ 'content-type',
113
+ 'x-hub-signature-256',
114
+ 'x-hub-signature',
115
+ 'x-github-event',
116
+ 'x-github-delivery',
117
+ 'x-gitlab-token',
118
+ 'x-gitlab-event',
119
+ 'user-agent',
120
+ ];
121
+ for (const header of headersToForward) {
122
+ const value = req.get(header);
123
+ if (value) {
124
+ forwardHeaders[header] = value;
125
+ }
126
+ }
127
+ // Add workspace context header
128
+ forwardHeaders['x-forwarded-for-workspace'] = workspaceId;
129
+ forwardHeaders['x-original-host'] = req.get('host') || '';
130
+ // Get raw body if available, otherwise use JSON stringified body
131
+ const rawBody = req.rawBody || JSON.stringify(req.body);
132
+ const response = await fetch(targetUrl, {
133
+ method: req.method,
134
+ headers: forwardHeaders,
135
+ body: ['GET', 'HEAD'].includes(req.method) ? undefined : rawBody,
136
+ });
137
+ // Forward response back
138
+ const responseBody = await response.text();
139
+ res.status(response.status);
140
+ // Copy response headers
141
+ response.headers.forEach((value, key) => {
142
+ if (!['content-encoding', 'transfer-encoding', 'content-length'].includes(key.toLowerCase())) {
143
+ res.set(key, value);
144
+ }
145
+ });
146
+ res.send(responseBody);
147
+ }
148
+ catch (error) {
149
+ console.error(`[webhooks] Error forwarding to workspace ${workspaceId}:`, error);
150
+ res.status(502).json({
151
+ error: 'Failed to forward webhook to workspace',
152
+ details: error.message,
153
+ });
154
+ }
155
+ }
156
+ // TODO: Re-enable workspace webhook forwarding once route pattern issue is resolved
157
+ // The newer path-to-regexp version doesn't support wildcard patterns like /workspace/*
158
+ // See: https://git.new/pathToRegexpError
159
+ // webhooksRouter.all('/workspace/*', handleWorkspaceWebhook);
160
+ // GitHub webhook signature verification
161
+ function verifyGitHubSignature(payload, signature) {
162
+ if (!signature)
163
+ return false;
164
+ const config = getConfig();
165
+ const secret = config.github.webhookSecret || config.github.clientSecret;
166
+ const expectedSignature = `sha256=${crypto
167
+ .createHmac('sha256', secret)
168
+ .update(payload)
169
+ .digest('hex')}`;
170
+ return crypto.timingSafeEqual(Buffer.from(signature), Buffer.from(expectedSignature));
171
+ }
172
+ /**
173
+ * POST /api/webhooks/github
174
+ * Handle GitHub App webhook events
175
+ */
176
+ webhooksRouter.post('/github', async (req, res) => {
177
+ const signature = req.get('x-hub-signature-256');
178
+ const event = req.get('x-github-event');
179
+ const deliveryId = req.get('x-github-delivery');
180
+ // Get raw body for signature verification
181
+ // Note: This requires raw body middleware to be set up
182
+ const rawBody = JSON.stringify(req.body);
183
+ // Verify signature
184
+ if (!verifyGitHubSignature(rawBody, signature)) {
185
+ console.error(`[webhook] Invalid signature for delivery ${deliveryId}`);
186
+ return res.status(401).json({ error: 'Invalid signature' });
187
+ }
188
+ console.log(`[webhook] Received ${event} event (delivery: ${deliveryId})`);
189
+ try {
190
+ switch (event) {
191
+ case 'installation':
192
+ await handleInstallationEvent(req.body);
193
+ break;
194
+ case 'installation_repositories':
195
+ await handleInstallationRepositoriesEvent(req.body);
196
+ break;
197
+ case 'push':
198
+ // Future: trigger sync for push events
199
+ console.log(`[webhook] Push to ${req.body.repository?.full_name}`);
200
+ break;
201
+ case 'pull_request':
202
+ // Future: handle PR events
203
+ console.log(`[webhook] PR ${req.body.action} on ${req.body.repository?.full_name}`);
204
+ break;
205
+ case 'issues':
206
+ await handleIssueEvent(req.body);
207
+ break;
208
+ case 'issue_comment':
209
+ await handleIssueCommentEvent(req.body);
210
+ break;
211
+ case 'pull_request_review_comment':
212
+ await handlePRReviewCommentEvent(req.body);
213
+ break;
214
+ case 'check_run':
215
+ await handleCheckRunEvent(req.body);
216
+ break;
217
+ case 'workflow_run':
218
+ await handleWorkflowRunEvent(req.body);
219
+ break;
220
+ default:
221
+ console.log(`[webhook] Unhandled event: ${event}`);
222
+ }
223
+ res.json({ received: true });
224
+ }
225
+ catch (error) {
226
+ console.error(`[webhook] Error processing ${event}:`, error);
227
+ res.status(500).json({ error: 'Failed to process webhook' });
228
+ }
229
+ });
230
+ /**
231
+ * Handle installation events (created, deleted, suspended, etc.)
232
+ */
233
+ async function handleInstallationEvent(payload) {
234
+ const { action, installation, sender, repositories } = payload;
235
+ const installationId = String(installation.id);
236
+ console.log(`[webhook] Installation ${action}: ${installation.account.login} (${installationId})`);
237
+ switch (action) {
238
+ case 'created': {
239
+ // Find the user by their GitHub ID (the sender who installed the app)
240
+ const user = await db.users.findByGithubId(String(sender.id));
241
+ // Create/update the installation record
242
+ await db.githubInstallations.upsert({
243
+ installationId,
244
+ accountType: installation.account.type.toLowerCase(),
245
+ accountLogin: installation.account.login,
246
+ accountId: String(installation.account.id),
247
+ installedById: user?.id ?? null,
248
+ permissions: installation.permissions,
249
+ events: installation.events,
250
+ });
251
+ // If repositories were included, sync them
252
+ if (repositories && user) {
253
+ for (const repo of repositories) {
254
+ const dbInstallation = await db.githubInstallations.findByInstallationId(installationId);
255
+ if (dbInstallation) {
256
+ await db.repositories.upsert({
257
+ userId: user.id,
258
+ githubFullName: repo.full_name,
259
+ githubId: repo.id,
260
+ isPrivate: repo.private,
261
+ installationId: dbInstallation.id,
262
+ syncStatus: 'synced',
263
+ lastSyncedAt: new Date(),
264
+ });
265
+ }
266
+ }
267
+ }
268
+ console.log(`[webhook] Created installation for ${installation.account.login}`);
269
+ break;
270
+ }
271
+ case 'deleted': {
272
+ // Remove the installation
273
+ await db.githubInstallations.delete(installationId);
274
+ console.log(`[webhook] Deleted installation for ${installation.account.login}`);
275
+ break;
276
+ }
277
+ case 'suspend': {
278
+ await db.githubInstallations.suspend(installationId, installation.suspended_by?.login || 'unknown');
279
+ console.log(`[webhook] Suspended installation for ${installation.account.login}`);
280
+ break;
281
+ }
282
+ case 'unsuspend': {
283
+ await db.githubInstallations.unsuspend(installationId);
284
+ console.log(`[webhook] Unsuspended installation for ${installation.account.login}`);
285
+ break;
286
+ }
287
+ case 'new_permissions_accepted': {
288
+ // Update permissions
289
+ await db.githubInstallations.updatePermissions(installationId, installation.permissions, installation.events);
290
+ console.log(`[webhook] Updated permissions for ${installation.account.login}`);
291
+ break;
292
+ }
293
+ default:
294
+ console.log(`[webhook] Unhandled installation action: ${action}`);
295
+ }
296
+ }
297
+ /**
298
+ * Handle installation_repositories events (added/removed repos)
299
+ */
300
+ async function handleInstallationRepositoriesEvent(payload) {
301
+ const { action, installation, repositories_added, repositories_removed, sender } = payload;
302
+ const installationId = String(installation.id);
303
+ console.log(`[webhook] Repositories ${action} for ${installation.account.login}`);
304
+ // Find the installation in our database
305
+ const dbInstallation = await db.githubInstallations.findByInstallationId(installationId);
306
+ if (!dbInstallation) {
307
+ console.error(`[webhook] Installation ${installationId} not found in database`);
308
+ return;
309
+ }
310
+ // Get the user who triggered this (should be the installedBy user)
311
+ const user = await db.users.findByGithubId(String(sender.id));
312
+ if (!user) {
313
+ console.error(`[webhook] User ${sender.login} not found in database`);
314
+ return;
315
+ }
316
+ if (action === 'added' && repositories_added) {
317
+ for (const repo of repositories_added) {
318
+ await db.repositories.upsert({
319
+ userId: user.id,
320
+ githubFullName: repo.full_name,
321
+ githubId: repo.id,
322
+ isPrivate: repo.private,
323
+ installationId: dbInstallation.id,
324
+ syncStatus: 'synced',
325
+ lastSyncedAt: new Date(),
326
+ });
327
+ }
328
+ console.log(`[webhook] Added ${repositories_added.length} repositories`);
329
+ }
330
+ if (action === 'removed' && repositories_removed) {
331
+ // We don't delete repos, just remove the installation link
332
+ // This preserves any user config while showing the repo is no longer accessible
333
+ for (const repo of repositories_removed) {
334
+ // Find the repo and clear its installation reference
335
+ const repos = await db.repositories.findByUserId(user.id);
336
+ const existingRepo = repos.find(r => r.githubFullName === repo.full_name);
337
+ if (existingRepo) {
338
+ // Update sync status to indicate repo access was removed
339
+ await db.repositories.updateSyncStatus(existingRepo.id, 'access_removed');
340
+ }
341
+ }
342
+ console.log(`[webhook] Removed access to ${repositories_removed.length} repositories`);
343
+ }
344
+ }
345
+ /**
346
+ * Handle check_run webhook events
347
+ *
348
+ * When a CI check fails on a PR, we:
349
+ * 1. Record the failure in our database
350
+ * 2. Check if an agent is already working on the PR
351
+ * 3. Either message the existing agent or spawn a new one
352
+ */
353
+ async function handleCheckRunEvent(payload) {
354
+ const { action, check_run, repository } = payload;
355
+ // Only handle completed checks
356
+ if (action !== 'completed') {
357
+ console.log(`[webhook] Ignoring check_run action: ${action}`);
358
+ return;
359
+ }
360
+ // Only handle failures
361
+ if (check_run.conclusion !== 'failure') {
362
+ console.log(`[webhook] Check ${check_run.name} conclusion: ${check_run.conclusion} (not a failure)`);
363
+ return;
364
+ }
365
+ // Only handle checks on PRs
366
+ if (check_run.pull_requests.length === 0) {
367
+ console.log(`[webhook] Check ${check_run.name} failed but not on a PR, skipping`);
368
+ return;
369
+ }
370
+ const pr = check_run.pull_requests[0];
371
+ console.log(`[webhook] CI failure: ${check_run.name} on ${repository.full_name}#${pr.number}`);
372
+ // Build failure context
373
+ const failureContext = {
374
+ repository: repository.full_name,
375
+ prNumber: pr.number,
376
+ branch: pr.head.ref,
377
+ commitSha: pr.head.sha,
378
+ checkName: check_run.name,
379
+ checkId: check_run.id,
380
+ conclusion: check_run.conclusion,
381
+ failureTitle: check_run.output.title,
382
+ failureSummary: check_run.output.summary,
383
+ failureDetails: check_run.output.text,
384
+ annotations: (check_run.output.annotations || []).map(a => ({
385
+ path: a.path,
386
+ startLine: a.start_line,
387
+ endLine: a.end_line,
388
+ annotationLevel: a.annotation_level,
389
+ message: a.message,
390
+ })),
391
+ };
392
+ // Record the failure in the database
393
+ try {
394
+ const failureEvent = await db.ciFailureEvents.create({
395
+ repository: failureContext.repository,
396
+ prNumber: failureContext.prNumber,
397
+ branch: failureContext.branch,
398
+ commitSha: failureContext.commitSha,
399
+ checkName: failureContext.checkName,
400
+ checkId: failureContext.checkId,
401
+ conclusion: failureContext.conclusion,
402
+ failureTitle: failureContext.failureTitle,
403
+ failureSummary: failureContext.failureSummary,
404
+ failureDetails: failureContext.failureDetails,
405
+ annotations: failureContext.annotations,
406
+ });
407
+ console.log(`[webhook] Recorded CI failure event: ${failureEvent.id}`);
408
+ // Check for existing active fix attempts on this repo
409
+ const activeAttempts = await db.ciFixAttempts.findActiveByRepository(repository.full_name);
410
+ if (activeAttempts.length > 0) {
411
+ console.log(`[webhook] ${activeAttempts.length} active fix attempt(s) already exist, skipping spawn`);
412
+ await db.ciFailureEvents.markProcessed(failureEvent.id, false);
413
+ return;
414
+ }
415
+ // Import and call the CI agent spawner (lazy import to avoid circular deps)
416
+ const { spawnCIFixAgent } = await import('../services/ci-agent-spawner.js');
417
+ await spawnCIFixAgent(failureEvent);
418
+ // Mark as processed with agent spawned
419
+ await db.ciFailureEvents.markProcessed(failureEvent.id, true);
420
+ console.log(`[webhook] Agent spawned for CI failure: ${failureEvent.id}`);
421
+ }
422
+ catch (error) {
423
+ console.error(`[webhook] Failed to handle CI failure:`, error);
424
+ // Don't re-throw - we still want to return 200 to GitHub
425
+ }
426
+ }
427
+ /**
428
+ * Handle workflow_run webhook events
429
+ *
430
+ * This handles the entire workflow completion. Useful for:
431
+ * - Waiting for all checks to complete before acting
432
+ * - Getting workflow-level context
433
+ */
434
+ async function handleWorkflowRunEvent(payload) {
435
+ const { action, workflow_run, repository } = payload;
436
+ // Only handle completed workflows
437
+ if (action !== 'completed') {
438
+ console.log(`[webhook] Ignoring workflow_run action: ${action}`);
439
+ return;
440
+ }
441
+ // Only handle failures
442
+ if (workflow_run.conclusion !== 'failure') {
443
+ console.log(`[webhook] Workflow ${workflow_run.name} conclusion: ${workflow_run.conclusion}`);
444
+ return;
445
+ }
446
+ // Log for now - we primarily handle individual check_runs
447
+ // but workflow_run events can be used for aggregate failure handling
448
+ console.log(`[webhook] Workflow failed: ${workflow_run.name} on ${repository.full_name} ` +
449
+ `(branch: ${workflow_run.head_branch}, PRs: ${workflow_run.pull_requests.map(p => p.number).join(', ')})`);
450
+ // Future: Could use this to trigger workflow-level actions
451
+ // For now, individual check_run events handle the actual failure processing
452
+ }
453
+ /**
454
+ * Extract @mentions from comment text
455
+ * Returns list of mentioned agent names (without @ prefix)
456
+ */
457
+ function extractMentions(text) {
458
+ // Match @agent-name patterns (alphanumeric, hyphens, underscores)
459
+ const mentionPattern = /@([a-zA-Z][a-zA-Z0-9_-]*)/g;
460
+ const mentions = [];
461
+ let match;
462
+ while ((match = mentionPattern.exec(text)) !== null) {
463
+ mentions.push(match[1].toLowerCase());
464
+ }
465
+ return [...new Set(mentions)]; // Remove duplicates
466
+ }
467
+ /**
468
+ * Get context around a mention (for prompt building)
469
+ */
470
+ function getMentionContext(text, mention, contextLength = 200) {
471
+ const mentionIndex = text.toLowerCase().indexOf(`@${mention.toLowerCase()}`);
472
+ if (mentionIndex === -1)
473
+ return text.slice(0, contextLength);
474
+ const start = Math.max(0, mentionIndex - contextLength / 2);
475
+ const end = Math.min(text.length, mentionIndex + mention.length + 1 + contextLength / 2);
476
+ let context = text.slice(start, end);
477
+ if (start > 0)
478
+ context = '...' + context;
479
+ if (end < text.length)
480
+ context = context + '...';
481
+ return context;
482
+ }
483
+ /**
484
+ * Handle issues webhook events
485
+ *
486
+ * When a new issue is opened or labeled, we can:
487
+ * 1. Auto-assign an agent based on labels
488
+ * 2. Record the issue for later assignment
489
+ */
490
+ async function handleIssueEvent(payload) {
491
+ const { action, issue, repository } = payload;
492
+ console.log(`[webhook] Issue ${action}: #${issue.number} on ${repository.full_name}`);
493
+ // Only handle opened issues for now
494
+ if (action !== 'opened' && action !== 'labeled') {
495
+ return;
496
+ }
497
+ try {
498
+ // Check if we already have an assignment for this issue
499
+ const existing = await db.issueAssignments.findByIssue(repository.full_name, issue.number);
500
+ if (existing) {
501
+ console.log(`[webhook] Issue #${issue.number} already has an assignment`);
502
+ return;
503
+ }
504
+ // Determine priority based on labels
505
+ const labels = issue.labels.map(l => l.name.toLowerCase());
506
+ let priority;
507
+ if (labels.includes('critical') || labels.includes('p0'))
508
+ priority = 'critical';
509
+ else if (labels.includes('high') || labels.includes('p1'))
510
+ priority = 'high';
511
+ else if (labels.includes('medium') || labels.includes('p2'))
512
+ priority = 'medium';
513
+ else if (labels.includes('low') || labels.includes('p3'))
514
+ priority = 'low';
515
+ // Create issue assignment record
516
+ const assignment = await db.issueAssignments.create({
517
+ repository: repository.full_name,
518
+ issueNumber: issue.number,
519
+ issueTitle: issue.title,
520
+ issueBody: issue.body,
521
+ issueUrl: issue.html_url,
522
+ status: 'pending',
523
+ labels: issue.labels.map(l => l.name),
524
+ priority,
525
+ });
526
+ console.log(`[webhook] Created issue assignment: ${assignment.id}`);
527
+ // Check if we should auto-assign an agent
528
+ // TODO: Load repo configuration for auto-assign settings
529
+ // For now, issues remain in 'pending' status for manual assignment
530
+ }
531
+ catch (error) {
532
+ console.error(`[webhook] Failed to handle issue event:`, error);
533
+ }
534
+ }
535
+ /**
536
+ * Handle issue_comment webhook events
537
+ *
538
+ * When someone @mentions an agent in a comment:
539
+ * 1. Detect the mention
540
+ * 2. Record it for agent processing
541
+ * 3. Route to appropriate agent
542
+ */
543
+ async function handleIssueCommentEvent(payload) {
544
+ const { action, issue, comment, repository, sender } = payload;
545
+ // Only handle new comments
546
+ if (action !== 'created') {
547
+ return;
548
+ }
549
+ const isPR = !!issue.pull_request;
550
+ const sourceType = isPR ? 'pr_comment' : 'issue_comment';
551
+ console.log(`[webhook] ${sourceType} on ${repository.full_name}#${issue.number} by @${sender.login}`);
552
+ // Extract @mentions from comment
553
+ const mentions = extractMentions(comment.body);
554
+ if (mentions.length === 0) {
555
+ return; // No mentions to process
556
+ }
557
+ console.log(`[webhook] Found mentions: ${mentions.join(', ')}`);
558
+ try {
559
+ for (const mention of mentions) {
560
+ // Check if this is a known agent mention
561
+ // TODO: Load configured agents from repo/workspace settings
562
+ // For now, we accept any mention that looks like an agent name
563
+ const context = getMentionContext(comment.body, mention);
564
+ // Create mention record
565
+ const mentionRecord = await db.commentMentions.create({
566
+ repository: repository.full_name,
567
+ sourceType,
568
+ sourceId: comment.id,
569
+ issueOrPrNumber: issue.number,
570
+ commentBody: comment.body,
571
+ commentUrl: comment.html_url,
572
+ authorLogin: sender.login,
573
+ authorId: sender.id,
574
+ mentionedAgent: mention,
575
+ mentionContext: context,
576
+ status: 'pending',
577
+ });
578
+ console.log(`[webhook] Created mention record for @${mention}: ${mentionRecord.id}`);
579
+ // Import and call the mention handler (lazy import)
580
+ try {
581
+ const { handleMention } = await import('../services/mention-handler.js');
582
+ await handleMention(mentionRecord);
583
+ }
584
+ catch (_importError) {
585
+ // Handler not implemented yet - mentions will be processed later
586
+ console.log(`[webhook] Mention handler not available, mention queued for later processing`);
587
+ }
588
+ }
589
+ }
590
+ catch (error) {
591
+ console.error(`[webhook] Failed to handle comment mentions:`, error);
592
+ }
593
+ }
594
+ /**
595
+ * Handle pull_request_review_comment webhook events
596
+ *
597
+ * Similar to issue_comment, but for PR review comments (inline code comments)
598
+ */
599
+ async function handlePRReviewCommentEvent(payload) {
600
+ const { action, pull_request, comment, repository, sender } = payload;
601
+ // Only handle new comments
602
+ if (action !== 'created') {
603
+ return;
604
+ }
605
+ console.log(`[webhook] PR review comment on ${repository.full_name}#${pull_request.number} ` +
606
+ `(${comment.path}:${comment.line}) by @${sender.login}`);
607
+ // Extract @mentions from comment
608
+ const mentions = extractMentions(comment.body);
609
+ if (mentions.length === 0) {
610
+ return; // No mentions to process
611
+ }
612
+ console.log(`[webhook] Found mentions in review comment: ${mentions.join(', ')}`);
613
+ try {
614
+ for (const mention of mentions) {
615
+ const context = getMentionContext(comment.body, mention);
616
+ // Create mention record
617
+ const mentionRecord = await db.commentMentions.create({
618
+ repository: repository.full_name,
619
+ sourceType: 'pr_review',
620
+ sourceId: comment.id,
621
+ issueOrPrNumber: pull_request.number,
622
+ commentBody: comment.body,
623
+ commentUrl: comment.html_url,
624
+ authorLogin: sender.login,
625
+ authorId: sender.id,
626
+ mentionedAgent: mention,
627
+ mentionContext: `${comment.path}:${comment.line || '?'}\n\n${context}`,
628
+ status: 'pending',
629
+ });
630
+ console.log(`[webhook] Created review mention for @${mention}: ${mentionRecord.id}`);
631
+ // Try to handle mention immediately
632
+ try {
633
+ const { handleMention } = await import('../services/mention-handler.js');
634
+ await handleMention(mentionRecord);
635
+ }
636
+ catch {
637
+ console.log(`[webhook] Mention handler not available, mention queued for later processing`);
638
+ }
639
+ }
640
+ }
641
+ catch (error) {
642
+ console.error(`[webhook] Failed to handle PR review comment mentions:`, error);
643
+ }
644
+ }
645
+ //# sourceMappingURL=webhooks.js.map