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,344 @@
|
|
|
1
|
+
import { Nango } from '@nangohq/node';
|
|
2
|
+
import crypto from 'node:crypto';
|
|
3
|
+
import { getConfig } from '../config.js';
|
|
4
|
+
export const NANGO_INTEGRATIONS = {
|
|
5
|
+
GITHUB_USER: 'github',
|
|
6
|
+
GITHUB_APP: 'github-app-oauth',
|
|
7
|
+
};
|
|
8
|
+
class NangoService {
|
|
9
|
+
client;
|
|
10
|
+
secret;
|
|
11
|
+
constructor() {
|
|
12
|
+
const config = getConfig();
|
|
13
|
+
this.secret = config.nango.secretKey;
|
|
14
|
+
this.client = new Nango({
|
|
15
|
+
secretKey: config.nango.secretKey,
|
|
16
|
+
...(config.nango.host ? { host: config.nango.host } : {}),
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Create a Nango connect session restricted to specific integrations.
|
|
21
|
+
*/
|
|
22
|
+
async createConnectSession(allowedIntegrations, endUser) {
|
|
23
|
+
const { data } = await this.client.createConnectSession({
|
|
24
|
+
allowed_integrations: allowedIntegrations,
|
|
25
|
+
end_user: {
|
|
26
|
+
id: endUser.id,
|
|
27
|
+
email: endUser.email,
|
|
28
|
+
},
|
|
29
|
+
});
|
|
30
|
+
return data;
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Fetch GitHub user profile via Nango proxy.
|
|
34
|
+
*/
|
|
35
|
+
async getGithubUser(connectionId) {
|
|
36
|
+
const response = await this.client.get({
|
|
37
|
+
connectionId,
|
|
38
|
+
providerConfigKey: NANGO_INTEGRATIONS.GITHUB_USER,
|
|
39
|
+
endpoint: '/user',
|
|
40
|
+
});
|
|
41
|
+
return response.data;
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Retrieve an installation access token from a GitHub App connection.
|
|
45
|
+
* Use this ONLY when you need the raw token (e.g., for git clone URLs).
|
|
46
|
+
* For API calls, use the proxy methods instead.
|
|
47
|
+
*/
|
|
48
|
+
async getGithubAppToken(connectionId) {
|
|
49
|
+
const token = await this.client.getToken(NANGO_INTEGRATIONS.GITHUB_APP, connectionId, false, true);
|
|
50
|
+
// Handle different return formats from Nango
|
|
51
|
+
if (typeof token === 'string') {
|
|
52
|
+
return token;
|
|
53
|
+
}
|
|
54
|
+
// Nango may return an object with access_token
|
|
55
|
+
if (token && typeof token === 'object') {
|
|
56
|
+
const tokenObj = token;
|
|
57
|
+
if (tokenObj.access_token) {
|
|
58
|
+
return tokenObj.access_token;
|
|
59
|
+
}
|
|
60
|
+
if (tokenObj.token) {
|
|
61
|
+
return tokenObj.token;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
console.error('[nango] Unexpected token format:', typeof token, token);
|
|
65
|
+
throw new Error('Expected GitHub App token to be a string');
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Retrieve the user's OAuth access token from a GitHub App OAuth connection.
|
|
69
|
+
* This is the user-level token (not the installation token).
|
|
70
|
+
* Use this for operations that require user context (e.g., gh CLI).
|
|
71
|
+
*
|
|
72
|
+
* The user token can be found in:
|
|
73
|
+
* 1. getToken() without installation flag
|
|
74
|
+
* 2. connection_config.access_token in github-app-oauth
|
|
75
|
+
* 3. Separate 'github' user connection
|
|
76
|
+
*/
|
|
77
|
+
async getGithubUserOAuthToken(connectionId) {
|
|
78
|
+
// First try: Get token from github-app-oauth connection credentials
|
|
79
|
+
try {
|
|
80
|
+
const token = await this.client.getToken(NANGO_INTEGRATIONS.GITHUB_APP, connectionId);
|
|
81
|
+
if (typeof token === 'string' && token.length > 0) {
|
|
82
|
+
return token;
|
|
83
|
+
}
|
|
84
|
+
if (token && typeof token === 'object') {
|
|
85
|
+
const tokenObj = token;
|
|
86
|
+
if (tokenObj.access_token) {
|
|
87
|
+
return tokenObj.access_token;
|
|
88
|
+
}
|
|
89
|
+
if (tokenObj.token) {
|
|
90
|
+
return tokenObj.token;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
catch (err) {
|
|
95
|
+
console.log('[nango] getToken for user OAuth failed, trying connection_config:', err);
|
|
96
|
+
}
|
|
97
|
+
// Second try: Check connection_config for user token
|
|
98
|
+
try {
|
|
99
|
+
const connection = await this.client.getConnection(NANGO_INTEGRATIONS.GITHUB_APP, connectionId);
|
|
100
|
+
const connConfig = connection.connection_config;
|
|
101
|
+
if (connConfig?.access_token && typeof connConfig.access_token === 'string') {
|
|
102
|
+
return connConfig.access_token;
|
|
103
|
+
}
|
|
104
|
+
// Also check credentials object
|
|
105
|
+
const credentials = connection.credentials;
|
|
106
|
+
if (credentials?.access_token) {
|
|
107
|
+
return credentials.access_token;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
catch (err) {
|
|
111
|
+
console.log('[nango] connection_config check failed:', err);
|
|
112
|
+
}
|
|
113
|
+
throw new Error('Could not retrieve GitHub user OAuth token');
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Retrieve the user's OAuth token from a 'github' user connection.
|
|
117
|
+
* This is for the separate GitHub OAuth login (not the App connection).
|
|
118
|
+
*/
|
|
119
|
+
async getGithubUserToken(connectionId) {
|
|
120
|
+
const token = await this.client.getToken(NANGO_INTEGRATIONS.GITHUB_USER, connectionId);
|
|
121
|
+
if (typeof token === 'string') {
|
|
122
|
+
return token;
|
|
123
|
+
}
|
|
124
|
+
if (token && typeof token === 'object') {
|
|
125
|
+
const tokenObj = token;
|
|
126
|
+
if (tokenObj.access_token) {
|
|
127
|
+
return tokenObj.access_token;
|
|
128
|
+
}
|
|
129
|
+
if (tokenObj.token) {
|
|
130
|
+
return tokenObj.token;
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
throw new Error('Could not retrieve GitHub user token');
|
|
134
|
+
}
|
|
135
|
+
/**
|
|
136
|
+
* List repositories available to a GitHub App installation using the Nango Proxy.
|
|
137
|
+
* The proxy automatically handles token injection and refresh.
|
|
138
|
+
* @see https://nango.dev/docs/implementation-guides/requests-proxy/implement-requests-proxy
|
|
139
|
+
*/
|
|
140
|
+
async listGithubAppRepos(connectionId) {
|
|
141
|
+
const response = await this.client.get({
|
|
142
|
+
connectionId,
|
|
143
|
+
providerConfigKey: NANGO_INTEGRATIONS.GITHUB_APP,
|
|
144
|
+
endpoint: '/installation/repositories',
|
|
145
|
+
params: { per_page: '100' },
|
|
146
|
+
});
|
|
147
|
+
return response.data;
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Get the GitHub App installation ID from a connection.
|
|
151
|
+
* The installation ID is stored in connection_config.installation_id
|
|
152
|
+
*/
|
|
153
|
+
async getGithubAppInstallationId(connectionId) {
|
|
154
|
+
try {
|
|
155
|
+
const connection = await this.client.getConnection(NANGO_INTEGRATIONS.GITHUB_APP, connectionId);
|
|
156
|
+
// Extract installation_id from connection_config (where Nango stores it for GitHub App OAuth)
|
|
157
|
+
const connectionConfig = connection.connection_config;
|
|
158
|
+
if (connectionConfig?.installation_id) {
|
|
159
|
+
return Number(connectionConfig.installation_id);
|
|
160
|
+
}
|
|
161
|
+
console.warn('[nango] No installation_id in connection_config');
|
|
162
|
+
return null;
|
|
163
|
+
}
|
|
164
|
+
catch (err) {
|
|
165
|
+
console.error('[nango] Failed to get installation ID:', err);
|
|
166
|
+
return null;
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Create an issue via Nango Proxy.
|
|
171
|
+
*/
|
|
172
|
+
async createGithubIssue(connectionId, owner, repo, data) {
|
|
173
|
+
const response = await this.client.post({
|
|
174
|
+
connectionId,
|
|
175
|
+
providerConfigKey: NANGO_INTEGRATIONS.GITHUB_APP,
|
|
176
|
+
endpoint: `/repos/${owner}/${repo}/issues`,
|
|
177
|
+
data,
|
|
178
|
+
});
|
|
179
|
+
return response.data;
|
|
180
|
+
}
|
|
181
|
+
/**
|
|
182
|
+
* Create a pull request via Nango Proxy.
|
|
183
|
+
*/
|
|
184
|
+
async createGithubPullRequest(connectionId, owner, repo, data) {
|
|
185
|
+
const response = await this.client.post({
|
|
186
|
+
connectionId,
|
|
187
|
+
providerConfigKey: NANGO_INTEGRATIONS.GITHUB_APP,
|
|
188
|
+
endpoint: `/repos/${owner}/${repo}/pulls`,
|
|
189
|
+
data,
|
|
190
|
+
});
|
|
191
|
+
return response.data;
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Add a comment to an issue via Nango Proxy.
|
|
195
|
+
*/
|
|
196
|
+
async addGithubIssueComment(connectionId, owner, repo, issueNumber, body) {
|
|
197
|
+
const response = await this.client.post({
|
|
198
|
+
connectionId,
|
|
199
|
+
providerConfigKey: NANGO_INTEGRATIONS.GITHUB_APP,
|
|
200
|
+
endpoint: `/repos/${owner}/${repo}/issues/${issueNumber}/comments`,
|
|
201
|
+
data: { body },
|
|
202
|
+
});
|
|
203
|
+
return response.data;
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Update connection end user metadata (e.g., after creating a user record).
|
|
207
|
+
*/
|
|
208
|
+
async updateEndUser(connectionId, providerConfigKey, endUser) {
|
|
209
|
+
await this.client.patchConnection({ connectionId, provider_config_key: providerConfigKey }, { end_user: endUser });
|
|
210
|
+
}
|
|
211
|
+
/**
|
|
212
|
+
* Delete a connection from Nango.
|
|
213
|
+
*
|
|
214
|
+
* Used to remove temporary session connections for returning users
|
|
215
|
+
* to prevent duplicate connections in Nango. In the two-connection pattern,
|
|
216
|
+
* new users get a permanent connection but returning users authenticate
|
|
217
|
+
* with a temporary one that gets deleted.
|
|
218
|
+
*
|
|
219
|
+
* @param connectionId - Nango connection ID to delete
|
|
220
|
+
* @param providerConfigKey - The integration key (e.g., 'github')
|
|
221
|
+
*/
|
|
222
|
+
async deleteConnection(connectionId, providerConfigKey) {
|
|
223
|
+
await this.client.deleteConnection(providerConfigKey, connectionId);
|
|
224
|
+
}
|
|
225
|
+
/**
|
|
226
|
+
* Get connection metadata including end_user info.
|
|
227
|
+
* Useful when webhook doesn't include end_user data.
|
|
228
|
+
*/
|
|
229
|
+
async getConnection(connectionId, providerConfigKey) {
|
|
230
|
+
const connection = await this.client.getConnection(providerConfigKey, connectionId);
|
|
231
|
+
return connection;
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Check if user has access to a specific GitHub repository.
|
|
235
|
+
* Uses the user's OAuth connection to query GitHub API.
|
|
236
|
+
* @param connectionId - User's Nango connection ID (github user OAuth)
|
|
237
|
+
* @param owner - Repository owner
|
|
238
|
+
* @param repo - Repository name
|
|
239
|
+
* @returns Access details or null if no access
|
|
240
|
+
*/
|
|
241
|
+
async checkUserRepoAccess(connectionId, owner, repo) {
|
|
242
|
+
try {
|
|
243
|
+
const response = await this.client.get({
|
|
244
|
+
connectionId,
|
|
245
|
+
providerConfigKey: NANGO_INTEGRATIONS.GITHUB_USER,
|
|
246
|
+
endpoint: `/repos/${owner}/${repo}`,
|
|
247
|
+
});
|
|
248
|
+
const data = response.data;
|
|
249
|
+
let permission = 'none';
|
|
250
|
+
if (data.permissions) {
|
|
251
|
+
if (data.permissions.admin) {
|
|
252
|
+
permission = 'admin';
|
|
253
|
+
}
|
|
254
|
+
else if (data.permissions.push) {
|
|
255
|
+
permission = 'write';
|
|
256
|
+
}
|
|
257
|
+
else if (data.permissions.pull) {
|
|
258
|
+
permission = 'read';
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
return {
|
|
262
|
+
hasAccess: true,
|
|
263
|
+
permission,
|
|
264
|
+
repository: {
|
|
265
|
+
id: data.id,
|
|
266
|
+
fullName: data.full_name,
|
|
267
|
+
isPrivate: data.private,
|
|
268
|
+
defaultBranch: data.default_branch,
|
|
269
|
+
},
|
|
270
|
+
};
|
|
271
|
+
}
|
|
272
|
+
catch (err) {
|
|
273
|
+
// 404 = no access or repo doesn't exist
|
|
274
|
+
const error = err;
|
|
275
|
+
if (error.response?.status === 404 || error.response?.status === 403) {
|
|
276
|
+
return { hasAccess: false };
|
|
277
|
+
}
|
|
278
|
+
console.error('[nango] checkUserRepoAccess error:', err);
|
|
279
|
+
throw err;
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
/**
|
|
283
|
+
* List all repositories the user has access to via their OAuth connection.
|
|
284
|
+
* Uses the user's personal OAuth token (not the GitHub App).
|
|
285
|
+
* @param connectionId - User's Nango connection ID (github user OAuth)
|
|
286
|
+
* @param options - Pagination and filter options
|
|
287
|
+
* @returns List of accessible repositories
|
|
288
|
+
*/
|
|
289
|
+
async listUserAccessibleRepos(connectionId, options) {
|
|
290
|
+
const page = options?.page ?? 1;
|
|
291
|
+
const perPage = options?.perPage ?? 100;
|
|
292
|
+
const type = options?.type ?? 'all';
|
|
293
|
+
const sort = options?.sort ?? 'updated';
|
|
294
|
+
const response = await this.client.get({
|
|
295
|
+
connectionId,
|
|
296
|
+
providerConfigKey: NANGO_INTEGRATIONS.GITHUB_USER,
|
|
297
|
+
endpoint: '/user/repos',
|
|
298
|
+
params: {
|
|
299
|
+
page: String(page),
|
|
300
|
+
per_page: String(perPage),
|
|
301
|
+
type,
|
|
302
|
+
sort,
|
|
303
|
+
direction: 'desc',
|
|
304
|
+
},
|
|
305
|
+
});
|
|
306
|
+
const repos = response.data || [];
|
|
307
|
+
return {
|
|
308
|
+
repositories: repos.map(r => ({
|
|
309
|
+
id: r.id,
|
|
310
|
+
fullName: r.full_name,
|
|
311
|
+
isPrivate: r.private,
|
|
312
|
+
defaultBranch: r.default_branch,
|
|
313
|
+
permissions: r.permissions || { admin: false, push: false, pull: false },
|
|
314
|
+
})),
|
|
315
|
+
hasMore: repos.length === perPage,
|
|
316
|
+
};
|
|
317
|
+
}
|
|
318
|
+
/**
|
|
319
|
+
* Verify webhook signature sent by Nango.
|
|
320
|
+
* Uses the new verifyIncomingWebhookRequest method.
|
|
321
|
+
* @see https://nango.dev/docs/reference/sdks/node#verify-webhook-signature
|
|
322
|
+
*/
|
|
323
|
+
verifyWebhookSignature(rawBody, headers) {
|
|
324
|
+
try {
|
|
325
|
+
// Use the new method: verifyIncomingWebhookRequest(body, headers)
|
|
326
|
+
return this.client.verifyIncomingWebhookRequest(rawBody, headers);
|
|
327
|
+
}
|
|
328
|
+
catch (err) {
|
|
329
|
+
console.error('[nango] verifyIncomingWebhookRequest error:', err);
|
|
330
|
+
// Fall back to manual HMAC verification using the secret key
|
|
331
|
+
const signature = headers['x-nango-signature'];
|
|
332
|
+
const hmacSha256 = headers['x-nango-hmac-sha256'];
|
|
333
|
+
if (!signature && !hmacSha256)
|
|
334
|
+
return false;
|
|
335
|
+
const expectedSignature = crypto
|
|
336
|
+
.createHmac('sha256', this.secret)
|
|
337
|
+
.update(rawBody)
|
|
338
|
+
.digest('hex');
|
|
339
|
+
return signature === expectedSignature || hmacSha256 === expectedSignature;
|
|
340
|
+
}
|
|
341
|
+
}
|
|
342
|
+
}
|
|
343
|
+
export const nangoService = new NangoService();
|
|
344
|
+
//# sourceMappingURL=nango.js.map
|
|
@@ -0,0 +1,131 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cloud Persistence Service
|
|
3
|
+
*
|
|
4
|
+
* Handles durable persistence of agent session data for cloud deployments.
|
|
5
|
+
* Subscribes to PtyWrapper events ('summary', 'session-end') and persists
|
|
6
|
+
* to PostgreSQL via Drizzle ORM.
|
|
7
|
+
*
|
|
8
|
+
* This decouples PtyWrapper from storage concerns - the wrapper emits events,
|
|
9
|
+
* this service handles persistence. Different storage backends can be swapped
|
|
10
|
+
* by implementing alternative persistence services.
|
|
11
|
+
*
|
|
12
|
+
* @see PtyWrapperEvents in src/wrapper/pty-wrapper.ts for event definitions
|
|
13
|
+
*/
|
|
14
|
+
import type { PtyWrapper } from '../../wrapper/pty-wrapper.js';
|
|
15
|
+
/**
|
|
16
|
+
* Configuration for CloudPersistenceService
|
|
17
|
+
*/
|
|
18
|
+
export interface CloudPersistenceConfig {
|
|
19
|
+
/** Workspace ID for scoping sessions */
|
|
20
|
+
workspaceId: string;
|
|
21
|
+
/** Optional callback when summary is persisted */
|
|
22
|
+
onSummaryPersisted?: (agentName: string, summaryId: string) => void;
|
|
23
|
+
/** Optional callback when session ends */
|
|
24
|
+
onSessionEnded?: (agentName: string, sessionId: string) => void;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* CloudPersistenceService manages durable storage for agent sessions.
|
|
28
|
+
*
|
|
29
|
+
* Usage:
|
|
30
|
+
* ```typescript
|
|
31
|
+
* const persistence = new CloudPersistenceService({
|
|
32
|
+
* workspaceId: 'workspace-123',
|
|
33
|
+
* });
|
|
34
|
+
*
|
|
35
|
+
* // Bind to a PtyWrapper instance
|
|
36
|
+
* const pty = new PtyWrapper(config);
|
|
37
|
+
* const sessionId = await persistence.bindToPtyWrapper(pty);
|
|
38
|
+
*
|
|
39
|
+
* // When done, unbind to clean up listeners
|
|
40
|
+
* persistence.unbindFromPtyWrapper(pty);
|
|
41
|
+
* ```
|
|
42
|
+
*/
|
|
43
|
+
export declare class CloudPersistenceService {
|
|
44
|
+
private config;
|
|
45
|
+
private boundWrappers;
|
|
46
|
+
constructor(config: CloudPersistenceConfig);
|
|
47
|
+
/**
|
|
48
|
+
* Bind to a PtyWrapper instance and start persisting its events.
|
|
49
|
+
* Creates a new agent session record and returns the session ID.
|
|
50
|
+
*
|
|
51
|
+
* @param wrapper The PtyWrapper to bind to
|
|
52
|
+
* @returns The session ID for this agent session
|
|
53
|
+
*/
|
|
54
|
+
bindToPtyWrapper(wrapper: PtyWrapper): Promise<string>;
|
|
55
|
+
/**
|
|
56
|
+
* Unbind from a PtyWrapper and clean up event listeners.
|
|
57
|
+
*
|
|
58
|
+
* @param wrapper The PtyWrapper to unbind from
|
|
59
|
+
*/
|
|
60
|
+
unbindFromPtyWrapper(wrapper: PtyWrapper): void;
|
|
61
|
+
/**
|
|
62
|
+
* Handle a summary event - persist to agent_summaries table.
|
|
63
|
+
*/
|
|
64
|
+
private handleSummary;
|
|
65
|
+
/**
|
|
66
|
+
* Handle a session-end event - update agent_sessions with end marker.
|
|
67
|
+
*/
|
|
68
|
+
private handleSessionEnd;
|
|
69
|
+
/**
|
|
70
|
+
* Get the session ID for a bound wrapper.
|
|
71
|
+
*/
|
|
72
|
+
getSessionId(wrapper: PtyWrapper): string | undefined;
|
|
73
|
+
/**
|
|
74
|
+
* Get all summaries for a session.
|
|
75
|
+
*/
|
|
76
|
+
getSessionSummaries(sessionId: string): Promise<{
|
|
77
|
+
id: string;
|
|
78
|
+
sessionId: string;
|
|
79
|
+
agentName: string;
|
|
80
|
+
summary: {
|
|
81
|
+
currentTask?: string;
|
|
82
|
+
completedTasks?: string[];
|
|
83
|
+
decisions?: string[];
|
|
84
|
+
context?: string;
|
|
85
|
+
files?: string[];
|
|
86
|
+
};
|
|
87
|
+
createdAt: Date;
|
|
88
|
+
}[]>;
|
|
89
|
+
/**
|
|
90
|
+
* Get the latest summary for an agent in THIS workspace.
|
|
91
|
+
* Joins through agent_sessions to ensure workspace scoping.
|
|
92
|
+
*/
|
|
93
|
+
getLatestSummary(agentName: string): Promise<{
|
|
94
|
+
id: string;
|
|
95
|
+
sessionId: string;
|
|
96
|
+
agentName: string;
|
|
97
|
+
summary: {
|
|
98
|
+
currentTask?: string;
|
|
99
|
+
completedTasks?: string[];
|
|
100
|
+
decisions?: string[];
|
|
101
|
+
context?: string;
|
|
102
|
+
files?: string[];
|
|
103
|
+
};
|
|
104
|
+
createdAt: Date;
|
|
105
|
+
}>;
|
|
106
|
+
/**
|
|
107
|
+
* Get active sessions for a workspace.
|
|
108
|
+
*/
|
|
109
|
+
getActiveSessions(): Promise<{
|
|
110
|
+
id: string;
|
|
111
|
+
workspaceId: string;
|
|
112
|
+
agentName: string;
|
|
113
|
+
status: string;
|
|
114
|
+
startedAt: Date;
|
|
115
|
+
endedAt: Date | null;
|
|
116
|
+
endMarker: {
|
|
117
|
+
summary?: string;
|
|
118
|
+
completedTasks?: string[];
|
|
119
|
+
} | null;
|
|
120
|
+
metadata: unknown;
|
|
121
|
+
}[]>;
|
|
122
|
+
/**
|
|
123
|
+
* Clean up all bindings.
|
|
124
|
+
*/
|
|
125
|
+
destroy(): void;
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Factory function to create a persistence service for a workspace.
|
|
129
|
+
*/
|
|
130
|
+
export declare function createPersistenceService(workspaceId: string): CloudPersistenceService;
|
|
131
|
+
//# sourceMappingURL=persistence.d.ts.map
|
|
@@ -0,0 +1,200 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cloud Persistence Service
|
|
3
|
+
*
|
|
4
|
+
* Handles durable persistence of agent session data for cloud deployments.
|
|
5
|
+
* Subscribes to PtyWrapper events ('summary', 'session-end') and persists
|
|
6
|
+
* to PostgreSQL via Drizzle ORM.
|
|
7
|
+
*
|
|
8
|
+
* This decouples PtyWrapper from storage concerns - the wrapper emits events,
|
|
9
|
+
* this service handles persistence. Different storage backends can be swapped
|
|
10
|
+
* by implementing alternative persistence services.
|
|
11
|
+
*
|
|
12
|
+
* @see PtyWrapperEvents in src/wrapper/pty-wrapper.ts for event definitions
|
|
13
|
+
*/
|
|
14
|
+
import { eq, and, desc } from 'drizzle-orm';
|
|
15
|
+
import { getDb } from '../db/drizzle.js';
|
|
16
|
+
import { agentSessions, agentSummaries } from '../db/schema.js';
|
|
17
|
+
/**
|
|
18
|
+
* CloudPersistenceService manages durable storage for agent sessions.
|
|
19
|
+
*
|
|
20
|
+
* Usage:
|
|
21
|
+
* ```typescript
|
|
22
|
+
* const persistence = new CloudPersistenceService({
|
|
23
|
+
* workspaceId: 'workspace-123',
|
|
24
|
+
* });
|
|
25
|
+
*
|
|
26
|
+
* // Bind to a PtyWrapper instance
|
|
27
|
+
* const pty = new PtyWrapper(config);
|
|
28
|
+
* const sessionId = await persistence.bindToPtyWrapper(pty);
|
|
29
|
+
*
|
|
30
|
+
* // When done, unbind to clean up listeners
|
|
31
|
+
* persistence.unbindFromPtyWrapper(pty);
|
|
32
|
+
* ```
|
|
33
|
+
*/
|
|
34
|
+
export class CloudPersistenceService {
|
|
35
|
+
config;
|
|
36
|
+
boundWrappers = new Map();
|
|
37
|
+
constructor(config) {
|
|
38
|
+
this.config = config;
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Bind to a PtyWrapper instance and start persisting its events.
|
|
42
|
+
* Creates a new agent session record and returns the session ID.
|
|
43
|
+
*
|
|
44
|
+
* @param wrapper The PtyWrapper to bind to
|
|
45
|
+
* @returns The session ID for this agent session
|
|
46
|
+
*/
|
|
47
|
+
async bindToPtyWrapper(wrapper) {
|
|
48
|
+
const db = getDb();
|
|
49
|
+
const agentName = wrapper.name;
|
|
50
|
+
// Create session record
|
|
51
|
+
const result = await db.insert(agentSessions).values({
|
|
52
|
+
workspaceId: this.config.workspaceId,
|
|
53
|
+
agentName,
|
|
54
|
+
status: 'active',
|
|
55
|
+
startedAt: new Date(),
|
|
56
|
+
}).returning();
|
|
57
|
+
const session = result[0];
|
|
58
|
+
if (!session) {
|
|
59
|
+
throw new Error(`Failed to create session for agent ${agentName}`);
|
|
60
|
+
}
|
|
61
|
+
const sessionId = session.id;
|
|
62
|
+
// Create event handlers
|
|
63
|
+
const summaryHandler = async (event) => {
|
|
64
|
+
await this.handleSummary(sessionId, event);
|
|
65
|
+
};
|
|
66
|
+
const sessionEndHandler = async (event) => {
|
|
67
|
+
await this.handleSessionEnd(sessionId, event);
|
|
68
|
+
};
|
|
69
|
+
// Bind handlers
|
|
70
|
+
wrapper.on('summary', summaryHandler);
|
|
71
|
+
wrapper.on('session-end', sessionEndHandler);
|
|
72
|
+
// Track binding for cleanup
|
|
73
|
+
this.boundWrappers.set(wrapper, {
|
|
74
|
+
sessionId,
|
|
75
|
+
summaryHandler,
|
|
76
|
+
sessionEndHandler,
|
|
77
|
+
});
|
|
78
|
+
console.log(`[persistence] Bound to ${agentName}, session=${sessionId}`);
|
|
79
|
+
return sessionId;
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Unbind from a PtyWrapper and clean up event listeners.
|
|
83
|
+
*
|
|
84
|
+
* @param wrapper The PtyWrapper to unbind from
|
|
85
|
+
*/
|
|
86
|
+
unbindFromPtyWrapper(wrapper) {
|
|
87
|
+
const binding = this.boundWrappers.get(wrapper);
|
|
88
|
+
if (!binding)
|
|
89
|
+
return;
|
|
90
|
+
wrapper.off('summary', binding.summaryHandler);
|
|
91
|
+
wrapper.off('session-end', binding.sessionEndHandler);
|
|
92
|
+
this.boundWrappers.delete(wrapper);
|
|
93
|
+
console.log(`[persistence] Unbound from ${wrapper.name}`);
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Handle a summary event - persist to agent_summaries table.
|
|
97
|
+
*/
|
|
98
|
+
async handleSummary(sessionId, event) {
|
|
99
|
+
try {
|
|
100
|
+
const db = getDb();
|
|
101
|
+
const result = await db.insert(agentSummaries).values({
|
|
102
|
+
sessionId,
|
|
103
|
+
agentName: event.agentName,
|
|
104
|
+
summary: event.summary,
|
|
105
|
+
createdAt: new Date(),
|
|
106
|
+
}).returning();
|
|
107
|
+
const summaryRecord = result[0];
|
|
108
|
+
if (!summaryRecord) {
|
|
109
|
+
console.error(`[persistence] Insert returned no record for ${event.agentName}`);
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
console.log(`[persistence] Saved summary for ${event.agentName}: ${event.summary.currentTask || 'no task'}`);
|
|
113
|
+
this.config.onSummaryPersisted?.(event.agentName, summaryRecord.id);
|
|
114
|
+
}
|
|
115
|
+
catch (err) {
|
|
116
|
+
console.error(`[persistence] Failed to save summary for ${event.agentName}:`, err);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
/**
|
|
120
|
+
* Handle a session-end event - update agent_sessions with end marker.
|
|
121
|
+
*/
|
|
122
|
+
async handleSessionEnd(sessionId, event) {
|
|
123
|
+
try {
|
|
124
|
+
const db = getDb();
|
|
125
|
+
await db.update(agentSessions)
|
|
126
|
+
.set({
|
|
127
|
+
status: 'ended',
|
|
128
|
+
endedAt: new Date(),
|
|
129
|
+
endMarker: event.marker,
|
|
130
|
+
})
|
|
131
|
+
.where(eq(agentSessions.id, sessionId));
|
|
132
|
+
console.log(`[persistence] Session ended for ${event.agentName}: ${event.marker.summary || 'no summary'}`);
|
|
133
|
+
this.config.onSessionEnded?.(event.agentName, sessionId);
|
|
134
|
+
}
|
|
135
|
+
catch (err) {
|
|
136
|
+
console.error(`[persistence] Failed to end session for ${event.agentName}:`, err);
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* Get the session ID for a bound wrapper.
|
|
141
|
+
*/
|
|
142
|
+
getSessionId(wrapper) {
|
|
143
|
+
return this.boundWrappers.get(wrapper)?.sessionId;
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Get all summaries for a session.
|
|
147
|
+
*/
|
|
148
|
+
async getSessionSummaries(sessionId) {
|
|
149
|
+
const db = getDb();
|
|
150
|
+
return db.select()
|
|
151
|
+
.from(agentSummaries)
|
|
152
|
+
.where(eq(agentSummaries.sessionId, sessionId))
|
|
153
|
+
.orderBy(agentSummaries.createdAt);
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Get the latest summary for an agent in THIS workspace.
|
|
157
|
+
* Joins through agent_sessions to ensure workspace scoping.
|
|
158
|
+
*/
|
|
159
|
+
async getLatestSummary(agentName) {
|
|
160
|
+
const db = getDb();
|
|
161
|
+
// Join with sessions to ensure we only get summaries from this workspace
|
|
162
|
+
const results = await db.select({
|
|
163
|
+
id: agentSummaries.id,
|
|
164
|
+
sessionId: agentSummaries.sessionId,
|
|
165
|
+
agentName: agentSummaries.agentName,
|
|
166
|
+
summary: agentSummaries.summary,
|
|
167
|
+
createdAt: agentSummaries.createdAt,
|
|
168
|
+
})
|
|
169
|
+
.from(agentSummaries)
|
|
170
|
+
.innerJoin(agentSessions, eq(agentSummaries.sessionId, agentSessions.id))
|
|
171
|
+
.where(and(eq(agentSummaries.agentName, agentName), eq(agentSessions.workspaceId, this.config.workspaceId)))
|
|
172
|
+
.orderBy(desc(agentSummaries.createdAt))
|
|
173
|
+
.limit(1);
|
|
174
|
+
return results[0] || null;
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* Get active sessions for a workspace.
|
|
178
|
+
*/
|
|
179
|
+
async getActiveSessions() {
|
|
180
|
+
const db = getDb();
|
|
181
|
+
return db.select()
|
|
182
|
+
.from(agentSessions)
|
|
183
|
+
.where(and(eq(agentSessions.workspaceId, this.config.workspaceId), eq(agentSessions.status, 'active')));
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* Clean up all bindings.
|
|
187
|
+
*/
|
|
188
|
+
destroy() {
|
|
189
|
+
for (const wrapper of this.boundWrappers.keys()) {
|
|
190
|
+
this.unbindFromPtyWrapper(wrapper);
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
/**
|
|
195
|
+
* Factory function to create a persistence service for a workspace.
|
|
196
|
+
*/
|
|
197
|
+
export function createPersistenceService(workspaceId) {
|
|
198
|
+
return new CloudPersistenceService({ workspaceId });
|
|
199
|
+
}
|
|
200
|
+
//# sourceMappingURL=persistence.js.map
|