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.
- package/.gitattributes +3 -0
- package/.nvmrc +1 -0
- package/.trajectories/completed/2026-01/traj_1dviorhnkcb5.json +65 -0
- package/.trajectories/completed/2026-01/traj_1dviorhnkcb5.md +37 -0
- package/.trajectories/completed/2026-01/traj_1k5if5snst2e.json +65 -0
- package/.trajectories/completed/2026-01/traj_1k5if5snst2e.md +37 -0
- package/.trajectories/completed/2026-01/traj_1rp3rges5811.json +49 -0
- package/.trajectories/completed/2026-01/traj_1rp3rges5811.md +31 -0
- package/.trajectories/completed/2026-01/traj_22bhyulruouw.json +113 -0
- package/.trajectories/completed/2026-01/traj_22bhyulruouw.md +57 -0
- package/.trajectories/completed/2026-01/traj_2dao7ddgnta0.json +53 -0
- package/.trajectories/completed/2026-01/traj_2dao7ddgnta0.md +32 -0
- package/.trajectories/completed/2026-01/traj_3t0440mjeunc.json +26 -0
- package/.trajectories/completed/2026-01/traj_3t0440mjeunc.md +6 -0
- package/.trajectories/completed/2026-01/traj_45x9494d9xnr.json +47 -0
- package/.trajectories/completed/2026-01/traj_45x9494d9xnr.md +32 -0
- package/.trajectories/completed/2026-01/traj_4aa0bb77s4nh.json +53 -0
- package/.trajectories/completed/2026-01/traj_4aa0bb77s4nh.md +32 -0
- package/.trajectories/completed/2026-01/traj_5lhmzq8rxpqv.json +59 -0
- package/.trajectories/completed/2026-01/traj_5lhmzq8rxpqv.md +33 -0
- package/.trajectories/completed/2026-01/traj_5vr4e9erb1fs.json +53 -0
- package/.trajectories/completed/2026-01/traj_5vr4e9erb1fs.md +32 -0
- package/.trajectories/completed/2026-01/traj_6fgiwdoklvym.json +48 -0
- package/.trajectories/completed/2026-01/traj_6fgiwdoklvym.md +24 -0
- package/.trajectories/completed/2026-01/traj_7ludwvz45veh.json +209 -0
- package/.trajectories/completed/2026-01/traj_7ludwvz45veh.md +97 -0
- package/.trajectories/completed/2026-01/traj_9921cuhel0pj.json +48 -0
- package/.trajectories/completed/2026-01/traj_9921cuhel0pj.md +24 -0
- package/.trajectories/completed/2026-01/traj_ajs7zqfux4wc.json +49 -0
- package/.trajectories/completed/2026-01/traj_ajs7zqfux4wc.md +23 -0
- package/.trajectories/completed/2026-01/traj_cvtqhlwcq9s0.json +53 -0
- package/.trajectories/completed/2026-01/traj_cvtqhlwcq9s0.md +32 -0
- package/.trajectories/completed/2026-01/traj_cxofprm2m2en.json +49 -0
- package/.trajectories/completed/2026-01/traj_cxofprm2m2en.md +31 -0
- package/.trajectories/completed/2026-01/traj_d2hhz3k0vrhn.json +26 -0
- package/.trajectories/completed/2026-01/traj_d2hhz3k0vrhn.md +6 -0
- package/.trajectories/completed/2026-01/traj_dfuvww9pege5.json +59 -0
- package/.trajectories/completed/2026-01/traj_dfuvww9pege5.md +37 -0
- package/.trajectories/completed/2026-01/traj_g0fisy9h51mf.json +77 -0
- package/.trajectories/completed/2026-01/traj_g0fisy9h51mf.md +42 -0
- package/.trajectories/completed/2026-01/traj_gjdre5voouod.json +53 -0
- package/.trajectories/completed/2026-01/traj_gjdre5voouod.md +32 -0
- package/.trajectories/completed/2026-01/traj_gtlyqtta3x8l.json +25 -0
- package/.trajectories/completed/2026-01/traj_gtlyqtta3x8l.md +15 -0
- package/.trajectories/completed/2026-01/traj_h4xijiuip3w4.json +101 -0
- package/.trajectories/completed/2026-01/traj_h4xijiuip3w4.md +44 -0
- package/.trajectories/completed/2026-01/traj_hhxte7w4gjjx.json +22 -0
- package/.trajectories/completed/2026-01/traj_hhxte7w4gjjx.md +5 -0
- package/.trajectories/completed/2026-01/traj_hpungyhoj6v5.json +53 -0
- package/.trajectories/completed/2026-01/traj_hpungyhoj6v5.md +32 -0
- package/.trajectories/completed/2026-01/traj_m2xkjv0w2sq7.json +25 -0
- package/.trajectories/completed/2026-01/traj_m2xkjv0w2sq7.md +15 -0
- package/.trajectories/completed/2026-01/traj_noq5zbvnrdvz.json +53 -0
- package/.trajectories/completed/2026-01/traj_noq5zbvnrdvz.md +32 -0
- package/.trajectories/completed/2026-01/traj_ntbs6ppopf46.json +53 -0
- package/.trajectories/completed/2026-01/traj_ntbs6ppopf46.md +32 -0
- package/.trajectories/completed/2026-01/traj_ozd98si6a7ns.json +48 -0
- package/.trajectories/completed/2026-01/traj_ozd98si6a7ns.md +24 -0
- package/.trajectories/completed/2026-01/traj_prdza7a5cxp5.json +53 -0
- package/.trajectories/completed/2026-01/traj_prdza7a5cxp5.md +32 -0
- package/.trajectories/completed/2026-01/traj_qb3twvvywfwi.json +77 -0
- package/.trajectories/completed/2026-01/traj_qb3twvvywfwi.md +42 -0
- package/.trajectories/completed/2026-01/traj_qft54mi7nfor.json +53 -0
- package/.trajectories/completed/2026-01/traj_qft54mi7nfor.md +32 -0
- package/.trajectories/completed/2026-01/traj_qx9uhf8whhxo.json +83 -0
- package/.trajectories/completed/2026-01/traj_qx9uhf8whhxo.md +47 -0
- package/.trajectories/completed/2026-01/traj_rd9toccj18a0.json +59 -0
- package/.trajectories/completed/2026-01/traj_rd9toccj18a0.md +37 -0
- package/.trajectories/completed/2026-01/traj_rt4fiw3ecp50.json +48 -0
- package/.trajectories/completed/2026-01/traj_rt4fiw3ecp50.md +16 -0
- package/.trajectories/completed/2026-01/traj_st8j35b0hrlc.json +59 -0
- package/.trajectories/completed/2026-01/traj_st8j35b0hrlc.md +37 -0
- package/.trajectories/completed/2026-01/traj_t1yy8m7hbuxp.json +53 -0
- package/.trajectories/completed/2026-01/traj_t1yy8m7hbuxp.md +32 -0
- package/.trajectories/completed/2026-01/traj_tmux_orchestrator_analysis.json +84 -0
- package/.trajectories/completed/2026-01/traj_tmux_orchestrator_analysis.md +109 -0
- package/.trajectories/completed/2026-01/traj_u9n9eqasw16k.json +53 -0
- package/.trajectories/completed/2026-01/traj_u9n9eqasw16k.md +32 -0
- package/.trajectories/completed/2026-01/traj_v87hypnongqx.json +71 -0
- package/.trajectories/completed/2026-01/traj_v87hypnongqx.md +42 -0
- package/.trajectories/completed/2026-01/traj_wkp2fgzdyinb.json +53 -0
- package/.trajectories/completed/2026-01/traj_wkp2fgzdyinb.md +32 -0
- package/.trajectories/completed/2026-01/traj_x14t8w8rn7xg.json +20 -0
- package/.trajectories/completed/2026-01/traj_x14t8w8rn7xg.md +6 -0
- package/.trajectories/completed/2026-01/traj_xnwbznkvv8ua.json +175 -0
- package/.trajectories/completed/2026-01/traj_xnwbznkvv8ua.md +82 -0
- package/.trajectories/completed/2026-01/traj_ysjc8zaeqtd3.json +47 -0
- package/.trajectories/completed/2026-01/traj_ysjc8zaeqtd3.md +32 -0
- package/.trajectories/completed/2026-01/traj_yvdadtvdgnz3.json +59 -0
- package/.trajectories/completed/2026-01/traj_yvdadtvdgnz3.md +37 -0
- package/.trajectories/completed/2026-01/traj_z0vcw1wrzide.json +53 -0
- package/.trajectories/completed/2026-01/traj_z0vcw1wrzide.md +32 -0
- package/.trajectories/index.json +314 -0
- package/ARCHITECTURE.md +1245 -0
- package/README.md +1 -1
- package/TESTING.md +278 -0
- package/deploy/init-db.sql +5 -0
- package/deploy/scripts/setup-fly-workspaces.sh +69 -0
- package/deploy/scripts/setup-railway.sh +75 -0
- package/deploy/workspace/entrypoint-browser.sh +118 -0
- package/deploy/workspace/entrypoint.sh +348 -0
- package/deploy/workspace/git-credential-relay +111 -0
- package/dist/bridge/spawner.d.ts +53 -0
- package/dist/bridge/spawner.js +203 -19
- package/dist/bridge/types.d.ts +12 -0
- package/dist/cli/index.js +618 -5
- package/dist/cloud/api/auth.d.ts +3 -2
- package/dist/cloud/api/auth.js +10 -98
- package/dist/cloud/api/billing.js +30 -9
- package/dist/cloud/api/cli-pty-runner.d.ts +54 -0
- package/dist/cloud/api/cli-pty-runner.js +119 -0
- package/dist/cloud/api/codex-auth-helper.d.ts +15 -0
- package/dist/cloud/api/codex-auth-helper.js +100 -0
- package/dist/cloud/api/generic-webhooks.d.ts +8 -0
- package/dist/cloud/api/generic-webhooks.js +129 -0
- package/dist/cloud/api/git.d.ts +8 -0
- package/dist/cloud/api/git.js +152 -0
- package/dist/cloud/api/github-app.d.ts +11 -0
- package/dist/cloud/api/github-app.js +189 -0
- package/dist/cloud/api/middleware/planLimits.d.ts +7 -0
- package/dist/cloud/api/middleware/planLimits.js +39 -1
- package/dist/cloud/api/monitoring.d.ts +11 -0
- package/dist/cloud/api/monitoring.js +578 -0
- package/dist/cloud/api/nango-auth.d.ts +9 -0
- package/dist/cloud/api/nango-auth.js +377 -0
- package/dist/cloud/api/onboarding.d.ts +8 -1
- package/dist/cloud/api/onboarding.js +313 -119
- package/dist/cloud/api/policy.d.ts +8 -0
- package/dist/cloud/api/policy.js +229 -0
- package/dist/cloud/api/providers.js +114 -42
- package/dist/cloud/api/repos.d.ts +1 -0
- package/dist/cloud/api/repos.js +186 -0
- package/dist/cloud/api/test-helpers.d.ts +10 -0
- package/dist/cloud/api/test-helpers.js +575 -0
- package/dist/cloud/api/webhooks.d.ts +8 -0
- package/dist/cloud/api/webhooks.js +645 -0
- package/dist/cloud/api/workspaces.js +320 -12
- package/dist/cloud/billing/plans.js +32 -19
- package/dist/cloud/billing/types.d.ts +9 -3
- package/dist/cloud/config.d.ts +9 -2
- package/dist/cloud/config.js +13 -4
- package/dist/cloud/db/drizzle.d.ts +84 -1
- package/dist/cloud/db/drizzle.js +470 -0
- package/dist/cloud/db/index.d.ts +9 -4
- package/dist/cloud/db/index.js +11 -3
- package/dist/cloud/db/schema.d.ts +3283 -556
- package/dist/cloud/db/schema.js +314 -1
- package/dist/cloud/index.d.ts +1 -0
- package/dist/cloud/index.js +2 -0
- package/dist/cloud/provisioner/index.d.ts +56 -0
- package/dist/cloud/provisioner/index.js +676 -34
- package/dist/cloud/server.d.ts +1 -0
- package/dist/cloud/server.js +362 -13
- package/dist/cloud/services/auto-scaler.d.ts +152 -0
- package/dist/cloud/services/auto-scaler.js +439 -0
- package/dist/cloud/services/capacity-manager.d.ts +148 -0
- package/dist/cloud/services/capacity-manager.js +449 -0
- package/dist/cloud/services/ci-agent-spawner.d.ts +49 -0
- package/dist/cloud/services/ci-agent-spawner.js +373 -0
- package/dist/cloud/services/index.d.ts +12 -0
- package/dist/cloud/services/index.js +15 -0
- package/dist/cloud/services/mention-handler.d.ts +65 -0
- package/dist/cloud/services/mention-handler.js +405 -0
- package/dist/cloud/services/nango.d.ts +186 -0
- package/dist/cloud/services/nango.js +344 -0
- package/dist/cloud/services/persistence.d.ts +131 -0
- package/dist/cloud/services/persistence.js +200 -0
- package/dist/cloud/services/planLimits.d.ts +37 -0
- package/dist/cloud/services/planLimits.js +86 -5
- package/dist/cloud/services/scaling-orchestrator.d.ts +159 -0
- package/dist/cloud/services/scaling-orchestrator.js +502 -0
- package/dist/cloud/services/scaling-policy.d.ts +121 -0
- package/dist/cloud/services/scaling-policy.js +415 -0
- package/dist/cloud/vault/index.js +1 -1
- package/dist/cloud/webhooks/index.d.ts +24 -0
- package/dist/cloud/webhooks/index.js +29 -0
- package/dist/cloud/webhooks/parsers/github.d.ts +8 -0
- package/dist/cloud/webhooks/parsers/github.js +234 -0
- package/dist/cloud/webhooks/parsers/index.d.ts +23 -0
- package/dist/cloud/webhooks/parsers/index.js +30 -0
- package/dist/cloud/webhooks/parsers/linear.d.ts +9 -0
- package/dist/cloud/webhooks/parsers/linear.js +258 -0
- package/dist/cloud/webhooks/parsers/slack.d.ts +9 -0
- package/dist/cloud/webhooks/parsers/slack.js +214 -0
- package/dist/cloud/webhooks/responders/github.d.ts +8 -0
- package/dist/cloud/webhooks/responders/github.js +73 -0
- package/dist/cloud/webhooks/responders/index.d.ts +23 -0
- package/dist/cloud/webhooks/responders/index.js +30 -0
- package/dist/cloud/webhooks/responders/linear.d.ts +9 -0
- package/dist/cloud/webhooks/responders/linear.js +149 -0
- package/dist/cloud/webhooks/responders/slack.d.ts +20 -0
- package/dist/cloud/webhooks/responders/slack.js +178 -0
- package/dist/cloud/webhooks/router.d.ts +25 -0
- package/dist/cloud/webhooks/router.js +504 -0
- package/dist/cloud/webhooks/rules-engine.d.ts +24 -0
- package/dist/cloud/webhooks/rules-engine.js +287 -0
- package/dist/cloud/webhooks/types.d.ts +186 -0
- package/dist/cloud/webhooks/types.js +8 -0
- package/dist/continuity/formatter.d.ts +51 -0
- package/dist/continuity/formatter.js +313 -0
- package/dist/continuity/handoff-store.d.ts +67 -0
- package/dist/continuity/handoff-store.js +472 -0
- package/dist/continuity/index.d.ts +45 -0
- package/dist/continuity/index.js +48 -0
- package/dist/continuity/ledger-store.d.ts +110 -0
- package/dist/continuity/ledger-store.js +500 -0
- package/dist/continuity/manager.d.ts +178 -0
- package/dist/continuity/manager.js +562 -0
- package/dist/continuity/parser.d.ts +76 -0
- package/dist/continuity/parser.js +579 -0
- package/dist/continuity/types.d.ts +180 -0
- package/dist/continuity/types.js +9 -0
- package/dist/daemon/agent-manager.d.ts +27 -0
- package/dist/daemon/agent-manager.js +107 -6
- package/dist/daemon/agent-registry.d.ts +32 -0
- package/dist/daemon/agent-registry.js +42 -2
- package/dist/daemon/api.d.ts +12 -0
- package/dist/daemon/api.js +131 -2
- package/dist/daemon/cli-auth.d.ts +67 -0
- package/dist/daemon/cli-auth.js +537 -0
- package/dist/daemon/cloud-sync.js +9 -7
- package/dist/daemon/orchestrator.js +30 -0
- package/dist/daemon/router.d.ts +5 -0
- package/dist/daemon/router.js +78 -26
- package/dist/daemon/server.d.ts +5 -0
- package/dist/daemon/server.js +9 -1
- package/dist/daemon/services/browser-testing.d.ts +88 -0
- package/dist/daemon/services/browser-testing.js +244 -0
- package/dist/daemon/services/container-spawner.d.ts +135 -0
- package/dist/daemon/services/container-spawner.js +313 -0
- package/dist/daemon/types.d.ts +5 -1
- package/dist/dashboard/out/404.html +1 -1
- package/dist/dashboard/out/_next/static/chunks/116-2502180def231162.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/282-980c2eb8fff20123.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/699-3b1cd6618a45d259.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/724-2dae7627550ab88f.js +9 -0
- package/dist/dashboard/out/_next/static/chunks/766-1f2dd8cb7f766b0b.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/app/onboarding/page-3fdfa60e53f2810d.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/app/page-e6381e5a6e1fbcfd.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/connect-repos/page-3538dfe0ffe984b8.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/history/{page-b6edd4dde8d08194.js → page-abb9ab2d329f56e9.js} +1 -1
- package/dist/dashboard/out/_next/static/chunks/app/layout-c0d118c0f92d969c.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/login/page-c22d080201cbd9fb.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/metrics/page-67a3e98d9a43a6ed.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/page-77e9c65420a06cfb.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/pricing/page-b08ed1c34d14434a.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/providers/page-e88bc117ef7671c3.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/signup/page-68d34f50baa8ab6b.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/e868780c-48e5f147c90a3a41.js +18 -0
- package/dist/dashboard/out/_next/static/chunks/{main-app-5d692157a8eb1fd9.js → main-app-6e8e8d3ef4e0192a.js} +1 -1
- package/dist/dashboard/out/_next/static/chunks/{main-c2f423b9c9f4591b.js → main-ed4e1fb6f29c34cf.js} +1 -1
- package/dist/dashboard/out/_next/static/chunks/webpack-1cdd8ed57114d5e1.js +1 -0
- package/dist/dashboard/out/_next/static/css/29852f26181969a0.css +1 -0
- package/dist/dashboard/out/_next/static/css/7c3ae9e8617d42a5.css +1 -0
- package/dist/dashboard/out/app/onboarding.html +1 -0
- package/dist/dashboard/out/app/onboarding.txt +7 -0
- package/dist/dashboard/out/app.html +1 -14
- package/dist/dashboard/out/app.txt +2 -2
- package/dist/dashboard/out/connect-repos.html +1 -0
- package/dist/dashboard/out/connect-repos.txt +7 -0
- package/dist/dashboard/out/history.html +1 -1
- package/dist/dashboard/out/history.txt +2 -2
- package/dist/dashboard/out/index.html +1 -1
- package/dist/dashboard/out/index.txt +2 -2
- package/dist/dashboard/out/login.html +6 -0
- package/dist/dashboard/out/login.txt +7 -0
- package/dist/dashboard/out/metrics.html +1 -1
- package/dist/dashboard/out/metrics.txt +2 -2
- package/dist/dashboard/out/pricing.html +3 -3
- package/dist/dashboard/out/pricing.txt +2 -2
- package/dist/dashboard/out/providers.html +1 -0
- package/dist/dashboard/out/providers.txt +7 -0
- package/dist/dashboard/out/signup.html +6 -0
- package/dist/dashboard/out/signup.txt +7 -0
- package/dist/dashboard-server/server.js +1308 -8
- package/dist/hooks/emitter.d.ts +40 -0
- package/dist/hooks/emitter.js +63 -0
- package/dist/hooks/index.d.ts +3 -0
- package/dist/hooks/index.js +3 -0
- package/dist/hooks/registry.d.ts +173 -0
- package/dist/hooks/registry.js +476 -0
- package/dist/hooks/trajectory-hooks.d.ts +52 -0
- package/dist/hooks/trajectory-hooks.js +183 -0
- package/dist/hooks/types.d.ts +141 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +3 -0
- package/dist/memory/adapters/index.d.ts +8 -0
- package/dist/memory/adapters/index.js +8 -0
- package/dist/memory/adapters/inmemory.d.ts +59 -0
- package/dist/memory/adapters/inmemory.js +195 -0
- package/dist/memory/adapters/supermemory.d.ts +71 -0
- package/dist/memory/adapters/supermemory.js +338 -0
- package/dist/memory/factory.d.ts +48 -0
- package/dist/memory/factory.js +143 -0
- package/dist/memory/index.d.ts +32 -0
- package/dist/memory/index.js +32 -0
- package/dist/memory/memory-hooks.d.ts +60 -0
- package/dist/memory/memory-hooks.js +313 -0
- package/dist/memory/service.d.ts +49 -0
- package/dist/memory/service.js +146 -0
- package/dist/memory/types.d.ts +195 -0
- package/dist/memory/types.js +8 -0
- package/dist/policy/agent-policy.d.ts +225 -0
- package/dist/policy/agent-policy.js +665 -0
- package/dist/policy/cloud-policy-fetcher.d.ts +12 -0
- package/dist/policy/cloud-policy-fetcher.js +64 -0
- package/dist/resiliency/crash-insights.d.ts +156 -0
- package/dist/resiliency/crash-insights.js +492 -0
- package/dist/resiliency/gossip-health.d.ts +137 -0
- package/dist/resiliency/gossip-health.js +241 -0
- package/dist/resiliency/index.d.ts +5 -0
- package/dist/resiliency/index.js +5 -0
- package/dist/resiliency/leader-watchdog.d.ts +109 -0
- package/dist/resiliency/leader-watchdog.js +189 -0
- package/dist/resiliency/memory-monitor.d.ts +172 -0
- package/dist/resiliency/memory-monitor.js +593 -0
- package/dist/resiliency/stateless-lead.d.ts +149 -0
- package/dist/resiliency/stateless-lead.js +308 -0
- package/dist/resiliency/supervisor.d.ts +38 -0
- package/dist/resiliency/supervisor.js +122 -0
- package/dist/shared/cli-auth-config.d.ts +91 -0
- package/dist/shared/cli-auth-config.js +264 -0
- package/dist/storage/adapter.d.ts +1 -1
- package/dist/trajectory/config.d.ts +84 -0
- package/dist/trajectory/config.js +163 -0
- package/dist/trajectory/index.d.ts +8 -0
- package/dist/trajectory/index.js +8 -0
- package/dist/trajectory/integration.d.ts +292 -0
- package/dist/trajectory/integration.js +834 -0
- package/dist/utils/logger.js +1 -1
- package/dist/utils/project-namespace.d.ts +24 -0
- package/dist/utils/project-namespace.js +84 -0
- package/dist/wrapper/parser.d.ts +10 -0
- package/dist/wrapper/parser.js +100 -33
- package/dist/wrapper/pty-wrapper.d.ts +197 -16
- package/dist/wrapper/pty-wrapper.js +943 -106
- package/dist/wrapper/shared.d.ts +165 -0
- package/dist/wrapper/shared.js +270 -0
- package/dist/wrapper/tmux-wrapper.d.ts +73 -11
- package/dist/wrapper/tmux-wrapper.js +541 -120
- package/package.json +16 -16
- package/scripts/postinstall.js +60 -0
- package/test-push.txt +1 -0
- package/bin/tmux +0 -0
- package/dist/bridge/config.d.ts.map +0 -1
- package/dist/bridge/config.js.map +0 -1
- package/dist/bridge/index.d.ts.map +0 -1
- package/dist/bridge/index.js.map +0 -1
- package/dist/bridge/multi-project-client.d.ts.map +0 -1
- package/dist/bridge/multi-project-client.js.map +0 -1
- package/dist/bridge/shadow-cli.d.ts.map +0 -1
- package/dist/bridge/shadow-cli.js.map +0 -1
- package/dist/bridge/shadow-config.d.ts.map +0 -1
- package/dist/bridge/shadow-config.js.map +0 -1
- package/dist/bridge/spawner.d.ts.map +0 -1
- package/dist/bridge/spawner.js.map +0 -1
- package/dist/bridge/teams-config.d.ts.map +0 -1
- package/dist/bridge/teams-config.js.map +0 -1
- package/dist/bridge/types.d.ts.map +0 -1
- package/dist/bridge/types.js.map +0 -1
- package/dist/bridge/utils.d.ts.map +0 -1
- package/dist/bridge/utils.js.map +0 -1
- package/dist/cli/index.d.ts.map +0 -1
- package/dist/cli/index.js.map +0 -1
- package/dist/cloud/api/auth.d.ts.map +0 -1
- package/dist/cloud/api/auth.js.map +0 -1
- package/dist/cloud/api/billing.d.ts.map +0 -1
- package/dist/cloud/api/billing.js.map +0 -1
- package/dist/cloud/api/coordinators.d.ts.map +0 -1
- package/dist/cloud/api/coordinators.js.map +0 -1
- package/dist/cloud/api/daemons.d.ts.map +0 -1
- package/dist/cloud/api/daemons.js.map +0 -1
- package/dist/cloud/api/middleware/planLimits.d.ts.map +0 -1
- package/dist/cloud/api/middleware/planLimits.js.map +0 -1
- package/dist/cloud/api/onboarding.d.ts.map +0 -1
- package/dist/cloud/api/onboarding.js.map +0 -1
- package/dist/cloud/api/providers.d.ts.map +0 -1
- package/dist/cloud/api/providers.js.map +0 -1
- package/dist/cloud/api/repos.d.ts.map +0 -1
- package/dist/cloud/api/repos.js.map +0 -1
- package/dist/cloud/api/teams.d.ts.map +0 -1
- package/dist/cloud/api/teams.js.map +0 -1
- package/dist/cloud/api/usage.d.ts.map +0 -1
- package/dist/cloud/api/usage.js.map +0 -1
- package/dist/cloud/api/workspaces.d.ts.map +0 -1
- package/dist/cloud/api/workspaces.js.map +0 -1
- package/dist/cloud/billing/index.d.ts.map +0 -1
- package/dist/cloud/billing/index.js.map +0 -1
- package/dist/cloud/billing/plans.d.ts.map +0 -1
- package/dist/cloud/billing/plans.js.map +0 -1
- package/dist/cloud/billing/service.d.ts.map +0 -1
- package/dist/cloud/billing/service.js.map +0 -1
- package/dist/cloud/billing/types.d.ts.map +0 -1
- package/dist/cloud/billing/types.js.map +0 -1
- package/dist/cloud/config.d.ts.map +0 -1
- package/dist/cloud/config.js.map +0 -1
- package/dist/cloud/db/drizzle.d.ts.map +0 -1
- package/dist/cloud/db/drizzle.js.map +0 -1
- package/dist/cloud/db/index.d.ts.map +0 -1
- package/dist/cloud/db/index.js.map +0 -1
- package/dist/cloud/db/schema.d.ts.map +0 -1
- package/dist/cloud/db/schema.js.map +0 -1
- package/dist/cloud/index.d.ts.map +0 -1
- package/dist/cloud/index.js.map +0 -1
- package/dist/cloud/provisioner/index.d.ts.map +0 -1
- package/dist/cloud/provisioner/index.js.map +0 -1
- package/dist/cloud/server.d.ts.map +0 -1
- package/dist/cloud/server.js.map +0 -1
- package/dist/cloud/services/coordinator.d.ts.map +0 -1
- package/dist/cloud/services/coordinator.js.map +0 -1
- package/dist/cloud/services/planLimits.d.ts.map +0 -1
- package/dist/cloud/services/planLimits.js.map +0 -1
- package/dist/cloud/vault/index.d.ts.map +0 -1
- package/dist/cloud/vault/index.js.map +0 -1
- package/dist/daemon/agent-manager.d.ts.map +0 -1
- package/dist/daemon/agent-manager.js.map +0 -1
- package/dist/daemon/agent-registry.d.ts.map +0 -1
- package/dist/daemon/agent-registry.js.map +0 -1
- package/dist/daemon/api.d.ts.map +0 -1
- package/dist/daemon/api.js.map +0 -1
- package/dist/daemon/auth.d.ts.map +0 -1
- package/dist/daemon/auth.js.map +0 -1
- package/dist/daemon/cloud-sync.d.ts.map +0 -1
- package/dist/daemon/cloud-sync.js.map +0 -1
- package/dist/daemon/connection.d.ts.map +0 -1
- package/dist/daemon/connection.js.map +0 -1
- package/dist/daemon/index.d.ts.map +0 -1
- package/dist/daemon/index.js.map +0 -1
- package/dist/daemon/orchestrator.d.ts.map +0 -1
- package/dist/daemon/orchestrator.js.map +0 -1
- package/dist/daemon/registry.d.ts.map +0 -1
- package/dist/daemon/registry.js.map +0 -1
- package/dist/daemon/router.d.ts.map +0 -1
- package/dist/daemon/router.js.map +0 -1
- package/dist/daemon/server.d.ts.map +0 -1
- package/dist/daemon/server.js.map +0 -1
- package/dist/daemon/types.d.ts.map +0 -1
- package/dist/daemon/types.js.map +0 -1
- package/dist/daemon/workspace-manager.d.ts.map +0 -1
- package/dist/daemon/workspace-manager.js.map +0 -1
- package/dist/dashboard/out/_next/static/chunks/693-7b3301d8f6bc5014.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/713-f78477eb185f1f4d.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/766-e53e1cfe39b0b5b5.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/900-037c64bfd797fb2a.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/app/page-e3d9e1f4466b9bae.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/layout-2433bb48965f4333.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/metrics/page-e68825a81db67ba1.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/page-cc108bf68c8a657f.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/pricing/page-d80e03a5297f95b6.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/webpack-a5acc2831d094776.js +0 -1
- package/dist/dashboard/out/_next/static/css/79b80143647a07d7.css +0 -1
- package/dist/dashboard/out/_next/static/css/8cf277370ad48cfe.css +0 -1
- package/dist/dashboard-server/metrics.d.ts.map +0 -1
- package/dist/dashboard-server/metrics.js.map +0 -1
- package/dist/dashboard-server/needs-attention.d.ts.map +0 -1
- package/dist/dashboard-server/needs-attention.js.map +0 -1
- package/dist/dashboard-server/server.d.ts.map +0 -1
- package/dist/dashboard-server/server.js.map +0 -1
- package/dist/dashboard-server/start.d.ts.map +0 -1
- package/dist/dashboard-server/start.js.map +0 -1
- package/dist/hooks/inbox-check/hook.d.ts.map +0 -1
- package/dist/hooks/inbox-check/hook.js.map +0 -1
- package/dist/hooks/inbox-check/index.d.ts.map +0 -1
- package/dist/hooks/inbox-check/index.js.map +0 -1
- package/dist/hooks/inbox-check/types.d.ts.map +0 -1
- package/dist/hooks/inbox-check/types.js.map +0 -1
- package/dist/hooks/inbox-check/utils.d.ts.map +0 -1
- package/dist/hooks/inbox-check/utils.js.map +0 -1
- package/dist/hooks/index.d.ts.map +0 -1
- package/dist/hooks/index.js.map +0 -1
- package/dist/hooks/types.d.ts.map +0 -1
- package/dist/hooks/types.js.map +0 -1
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/protocol/framing.d.ts.map +0 -1
- package/dist/protocol/framing.js.map +0 -1
- package/dist/protocol/index.d.ts.map +0 -1
- package/dist/protocol/index.js.map +0 -1
- package/dist/protocol/types.d.ts.map +0 -1
- package/dist/protocol/types.js.map +0 -1
- package/dist/resiliency/context-persistence.d.ts.map +0 -1
- package/dist/resiliency/context-persistence.js.map +0 -1
- package/dist/resiliency/health-monitor.d.ts.map +0 -1
- package/dist/resiliency/health-monitor.js.map +0 -1
- package/dist/resiliency/index.d.ts.map +0 -1
- package/dist/resiliency/index.js.map +0 -1
- package/dist/resiliency/logger.d.ts.map +0 -1
- package/dist/resiliency/logger.js.map +0 -1
- package/dist/resiliency/metrics.d.ts.map +0 -1
- package/dist/resiliency/metrics.js.map +0 -1
- package/dist/resiliency/provider-context.d.ts.map +0 -1
- package/dist/resiliency/provider-context.js.map +0 -1
- package/dist/resiliency/supervisor.d.ts.map +0 -1
- package/dist/resiliency/supervisor.js.map +0 -1
- package/dist/state/agent-state.d.ts.map +0 -1
- package/dist/state/agent-state.js.map +0 -1
- package/dist/storage/adapter.d.ts.map +0 -1
- package/dist/storage/adapter.js.map +0 -1
- package/dist/storage/sqlite-adapter.d.ts.map +0 -1
- package/dist/storage/sqlite-adapter.js.map +0 -1
- package/dist/utils/agent-config.d.ts.map +0 -1
- package/dist/utils/agent-config.js.map +0 -1
- package/dist/utils/command-resolver.d.ts.map +0 -1
- package/dist/utils/command-resolver.js.map +0 -1
- package/dist/utils/index.d.ts.map +0 -1
- package/dist/utils/index.js.map +0 -1
- package/dist/utils/logger.d.ts.map +0 -1
- package/dist/utils/logger.js.map +0 -1
- package/dist/utils/name-generator.d.ts.map +0 -1
- package/dist/utils/name-generator.js.map +0 -1
- package/dist/utils/project-namespace.d.ts.map +0 -1
- package/dist/utils/project-namespace.js.map +0 -1
- package/dist/utils/tmux-resolver.d.ts.map +0 -1
- package/dist/utils/tmux-resolver.js.map +0 -1
- package/dist/utils/update-checker.d.ts.map +0 -1
- package/dist/utils/update-checker.js.map +0 -1
- package/dist/wrapper/client.d.ts.map +0 -1
- package/dist/wrapper/client.js.map +0 -1
- package/dist/wrapper/inbox.d.ts.map +0 -1
- package/dist/wrapper/inbox.js.map +0 -1
- package/dist/wrapper/index.d.ts.map +0 -1
- package/dist/wrapper/index.js.map +0 -1
- package/dist/wrapper/parser.d.ts.map +0 -1
- package/dist/wrapper/parser.js.map +0 -1
- package/dist/wrapper/pty-wrapper.d.ts.map +0 -1
- package/dist/wrapper/pty-wrapper.js.map +0 -1
- package/dist/wrapper/tmux-wrapper.d.ts.map +0 -1
- package/dist/wrapper/tmux-wrapper.js.map +0 -1
- package/docs/AGENTS.md +0 -513
- package/docs/ARCHITECTURE_DECISIONS.md +0 -175
- package/docs/CHANGELOG.md +0 -11
- package/docs/CLI-SIMPLIFICATION-COMPLETE.md +0 -48
- package/docs/CLOUD-ARCHITECTURE.md +0 -652
- package/docs/CLOUD-ONBOARDING-DESIGN.md +0 -1983
- package/docs/COMPETITIVE_ANALYSIS.md +0 -897
- package/docs/CONTRIBUTING.md +0 -151
- package/docs/DESIGN_BRIDGE_STAFFING.md +0 -878
- package/docs/DESIGN_V2.md +0 -1079
- package/docs/INTEGRATION-GUIDE.md +0 -926
- package/docs/MONETIZATION.md +0 -1679
- package/docs/PROPOSAL-trajectories.md +0 -1582
- package/docs/PROTOCOL.md +0 -325
- package/docs/SCALING_ANALYSIS.md +0 -280
- package/docs/TESTING_PRESENCE_FEATURES.md +0 -327
- package/docs/TMUX_IMPLEMENTATION_NOTES.md +0 -364
- package/docs/TMUX_IMPROVEMENTS.md +0 -968
- package/docs/agent-relay-snippet.md +0 -168
- package/docs/competitive-analysis-mcp-agent-mail.md +0 -389
- package/docs/dashboard-v2-plan.md +0 -179
- package/docs/guides/CLOUD.md +0 -236
- package/docs/guides/LOCAL.md +0 -535
- package/docs/guides/SELF-HOSTED.md +0 -494
- package/docs/proposals/shadow-as-subagent.md +0 -765
- package/docs/proposals/slack-bot-integration.md +0 -1457
- package/docs/removable-code-analysis.md +0 -24
- package/scripts/dev/PUBLIC_RELEASE_PLAN.md +0 -88
- package/scripts/dev/dev-team-setup.sh +0 -431
- package/scripts/e2e-test.sh +0 -119
- package/scripts/games/game-protocol.md +0 -79
- package/scripts/games/hearts-setup.sh +0 -264
- package/scripts/tictactoe-setup.sh +0 -181
- /package/dist/dashboard/out/_next/static/chunks/{117-b2cd8d6485aacf2b.js → 117-f7b8ab0809342e77.js} +0 -0
- /package/dist/dashboard/out/_next/static/chunks/{648-8f3f26864ce515e5.js → 648-5cc6e1921389a58a.js} +0 -0
- /package/dist/dashboard/out/_next/static/chunks/app/_not-found/{page-0b990dbb71d72a98.js → page-53b8a69f76db17d0.js} +0 -0
- /package/dist/dashboard/out/_next/static/chunks/{fd9d1056-bf46c09eb57e019c.js → fd9d1056-609918ca7b6280bb.js} +0 -0
- /package/dist/dashboard/out/_next/static/{6HHWb2ZmnJ4OSm0zUP7h4 → wPgKJtcOmTFLpUncDg16A}/_buildManifest.js +0 -0
- /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
|