agent-relay 1.0.21 → 1.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/bridge/shadow-cli.d.ts +17 -0
- package/dist/bridge/shadow-cli.d.ts.map +1 -0
- package/dist/bridge/shadow-cli.js +75 -0
- package/dist/bridge/shadow-cli.js.map +1 -0
- package/dist/bridge/shadow-config.d.ts +87 -0
- package/dist/bridge/shadow-config.d.ts.map +1 -0
- package/dist/bridge/shadow-config.js +134 -0
- package/dist/bridge/shadow-config.js.map +1 -0
- package/dist/bridge/spawner.d.ts +15 -1
- package/dist/bridge/spawner.d.ts.map +1 -1
- package/dist/bridge/spawner.js +164 -4
- package/dist/bridge/spawner.js.map +1 -1
- package/dist/bridge/types.d.ts +55 -0
- package/dist/bridge/types.d.ts.map +1 -1
- package/dist/cli/index.js +796 -11
- package/dist/cli/index.js.map +1 -1
- package/dist/cloud/api/auth.d.ts +19 -0
- package/dist/cloud/api/auth.d.ts.map +1 -0
- package/dist/cloud/api/auth.js +216 -0
- package/dist/cloud/api/auth.js.map +1 -0
- package/dist/cloud/api/billing.d.ts +17 -0
- package/dist/cloud/api/billing.d.ts.map +1 -0
- package/dist/cloud/api/billing.js +353 -0
- package/dist/cloud/api/billing.js.map +1 -0
- package/dist/cloud/api/coordinators.d.ts +8 -0
- package/dist/cloud/api/coordinators.d.ts.map +1 -0
- package/dist/cloud/api/coordinators.js +347 -0
- package/dist/cloud/api/coordinators.js.map +1 -0
- package/dist/cloud/api/daemons.d.ts +12 -0
- package/dist/cloud/api/daemons.d.ts.map +1 -0
- package/dist/cloud/api/daemons.js +320 -0
- package/dist/cloud/api/daemons.js.map +1 -0
- package/dist/cloud/api/middleware/planLimits.d.ts +36 -0
- package/dist/cloud/api/middleware/planLimits.d.ts.map +1 -0
- package/dist/cloud/api/middleware/planLimits.js +164 -0
- package/dist/cloud/api/middleware/planLimits.js.map +1 -0
- package/dist/cloud/api/onboarding.d.ts +8 -0
- package/dist/cloud/api/onboarding.d.ts.map +1 -0
- package/dist/cloud/api/onboarding.js +407 -0
- package/dist/cloud/api/onboarding.js.map +1 -0
- package/dist/cloud/api/providers.d.ts +7 -0
- package/dist/cloud/api/providers.d.ts.map +1 -0
- package/dist/cloud/api/providers.js +435 -0
- package/dist/cloud/api/providers.js.map +1 -0
- package/dist/cloud/api/repos.d.ts +7 -0
- package/dist/cloud/api/repos.d.ts.map +1 -0
- package/dist/cloud/api/repos.js +314 -0
- package/dist/cloud/api/repos.js.map +1 -0
- package/dist/cloud/api/teams.d.ts +7 -0
- package/dist/cloud/api/teams.d.ts.map +1 -0
- package/dist/cloud/api/teams.js +279 -0
- package/dist/cloud/api/teams.js.map +1 -0
- package/dist/cloud/api/usage.d.ts +7 -0
- package/dist/cloud/api/usage.d.ts.map +1 -0
- package/dist/cloud/api/usage.js +98 -0
- package/dist/cloud/api/usage.js.map +1 -0
- package/dist/cloud/api/workspaces.d.ts +7 -0
- package/dist/cloud/api/workspaces.d.ts.map +1 -0
- package/dist/cloud/api/workspaces.js +510 -0
- package/dist/cloud/api/workspaces.js.map +1 -0
- package/dist/cloud/billing/index.d.ts +9 -0
- package/dist/cloud/billing/index.d.ts.map +1 -0
- package/dist/cloud/billing/index.js +9 -0
- package/dist/cloud/billing/index.js.map +1 -0
- package/dist/cloud/billing/plans.d.ts +39 -0
- package/dist/cloud/billing/plans.d.ts.map +1 -0
- package/dist/cloud/billing/plans.js +232 -0
- package/dist/cloud/billing/plans.js.map +1 -0
- package/dist/cloud/billing/service.d.ts +80 -0
- package/dist/cloud/billing/service.d.ts.map +1 -0
- package/dist/cloud/billing/service.js +388 -0
- package/dist/cloud/billing/service.js.map +1 -0
- package/dist/cloud/billing/types.d.ts +135 -0
- package/dist/cloud/billing/types.d.ts.map +1 -0
- package/dist/cloud/billing/types.js +7 -0
- package/dist/cloud/billing/types.js.map +1 -0
- package/dist/cloud/config.d.ts +59 -0
- package/dist/cloud/config.d.ts.map +1 -0
- package/dist/cloud/config.js +83 -0
- package/dist/cloud/config.js.map +1 -0
- package/dist/cloud/db/drizzle.d.ts +132 -0
- package/dist/cloud/db/drizzle.d.ts.map +1 -0
- package/dist/cloud/db/drizzle.js +613 -0
- package/dist/cloud/db/drizzle.js.map +1 -0
- package/dist/cloud/db/index.d.ts +30 -0
- package/dist/cloud/db/index.d.ts.map +1 -0
- package/dist/cloud/db/index.js +44 -0
- package/dist/cloud/db/index.js.map +1 -0
- package/dist/cloud/db/schema.d.ts +1792 -0
- package/dist/cloud/db/schema.d.ts.map +1 -0
- package/dist/cloud/db/schema.js +234 -0
- package/dist/cloud/db/schema.js.map +1 -0
- package/dist/cloud/index.d.ts +11 -0
- package/dist/cloud/index.d.ts.map +1 -0
- package/dist/cloud/index.js +37 -0
- package/dist/cloud/index.js.map +1 -0
- package/dist/cloud/provisioner/index.d.ts +51 -0
- package/dist/cloud/provisioner/index.d.ts.map +1 -0
- package/dist/cloud/provisioner/index.js +676 -0
- package/dist/cloud/provisioner/index.js.map +1 -0
- package/dist/cloud/server.d.ts +16 -0
- package/dist/cloud/server.d.ts.map +1 -0
- package/dist/cloud/server.js +190 -0
- package/dist/cloud/server.js.map +1 -0
- package/dist/cloud/services/coordinator.d.ts +62 -0
- package/dist/cloud/services/coordinator.d.ts.map +1 -0
- package/dist/cloud/services/coordinator.js +389 -0
- package/dist/cloud/services/coordinator.js.map +1 -0
- package/dist/cloud/services/planLimits.d.ts +110 -0
- package/dist/cloud/services/planLimits.d.ts.map +1 -0
- package/dist/cloud/services/planLimits.js +254 -0
- package/dist/cloud/services/planLimits.js.map +1 -0
- package/dist/cloud/vault/index.d.ts +76 -0
- package/dist/cloud/vault/index.d.ts.map +1 -0
- package/dist/cloud/vault/index.js +219 -0
- package/dist/cloud/vault/index.js.map +1 -0
- package/dist/daemon/agent-manager.d.ts +87 -0
- package/dist/daemon/agent-manager.d.ts.map +1 -0
- package/dist/daemon/agent-manager.js +412 -0
- package/dist/daemon/agent-manager.js.map +1 -0
- package/dist/daemon/agent-registry.d.ts +2 -0
- package/dist/daemon/agent-registry.d.ts.map +1 -1
- package/dist/daemon/agent-registry.js +3 -0
- package/dist/daemon/agent-registry.js.map +1 -1
- package/dist/daemon/api.d.ts +69 -0
- package/dist/daemon/api.d.ts.map +1 -0
- package/dist/daemon/api.js +425 -0
- package/dist/daemon/api.js.map +1 -0
- package/dist/daemon/cloud-sync.d.ts +101 -0
- package/dist/daemon/cloud-sync.d.ts.map +1 -0
- package/dist/daemon/cloud-sync.js +261 -0
- package/dist/daemon/cloud-sync.js.map +1 -0
- package/dist/daemon/index.d.ts +4 -0
- package/dist/daemon/index.d.ts.map +1 -1
- package/dist/daemon/index.js +6 -0
- package/dist/daemon/index.js.map +1 -1
- package/dist/daemon/orchestrator.d.ts +155 -0
- package/dist/daemon/orchestrator.d.ts.map +1 -0
- package/dist/daemon/orchestrator.js +736 -0
- package/dist/daemon/orchestrator.js.map +1 -0
- package/dist/daemon/router.d.ts +24 -0
- package/dist/daemon/router.d.ts.map +1 -1
- package/dist/daemon/router.js +71 -1
- package/dist/daemon/router.js.map +1 -1
- package/dist/daemon/server.d.ts +37 -0
- package/dist/daemon/server.d.ts.map +1 -1
- package/dist/daemon/server.js +191 -16
- package/dist/daemon/server.js.map +1 -1
- package/dist/daemon/types.d.ts +127 -0
- package/dist/daemon/types.d.ts.map +1 -0
- package/dist/daemon/types.js +6 -0
- package/dist/daemon/types.js.map +1 -0
- package/dist/daemon/workspace-manager.d.ts +75 -0
- package/dist/daemon/workspace-manager.d.ts.map +1 -0
- package/dist/daemon/workspace-manager.js +289 -0
- package/dist/daemon/workspace-manager.js.map +1 -0
- package/dist/dashboard/out/404.html +1 -1
- package/dist/dashboard/out/_next/static/chunks/693-7b3301d8f6bc5014.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/713-f78477eb185f1f4d.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/766-e53e1cfe39b0b5b5.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/900-037c64bfd797fb2a.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/app/page-e3d9e1f4466b9bae.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/history/page-b6edd4dde8d08194.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/layout-2433bb48965f4333.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/metrics/page-e68825a81db67ba1.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/page-cc108bf68c8a657f.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/app/pricing/page-d80e03a5297f95b6.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/main-app-5d692157a8eb1fd9.js +1 -0
- package/dist/dashboard/out/_next/static/chunks/{main-e0a1f53fe0617a63.js → main-c2f423b9c9f4591b.js} +1 -1
- package/dist/dashboard/out/_next/static/chunks/{webpack-c81f7fd28659d64f.js → webpack-a5acc2831d094776.js} +1 -1
- package/dist/dashboard/out/_next/static/css/79b80143647a07d7.css +1 -0
- package/dist/dashboard/out/_next/static/css/8cf277370ad48cfe.css +1 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo-128.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo-256.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo-32.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo-512.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo-64.png +0 -0
- package/dist/dashboard/out/alt-logos/agent-relay-logo.svg +45 -0
- package/dist/dashboard/out/alt-logos/logo.svg +38 -0
- package/dist/dashboard/out/alt-logos/monogram-logo-128.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo-256.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo-32.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo-512.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo-64.png +0 -0
- package/dist/dashboard/out/alt-logos/monogram-logo.svg +38 -0
- package/dist/dashboard/out/app.html +14 -0
- package/dist/dashboard/out/app.txt +7 -0
- package/dist/dashboard/out/history.html +1 -0
- package/dist/dashboard/out/history.txt +7 -0
- package/dist/dashboard/out/index.html +1 -1
- package/dist/dashboard/out/index.txt +2 -2
- package/dist/dashboard/out/metrics.html +1 -515
- package/dist/dashboard/out/metrics.txt +2 -2
- package/dist/dashboard/out/pricing.html +13 -0
- package/dist/dashboard/out/pricing.txt +7 -0
- package/dist/dashboard-server/metrics.d.ts.map +1 -1
- package/dist/dashboard-server/metrics.js +3 -2
- package/dist/dashboard-server/metrics.js.map +1 -1
- package/dist/dashboard-server/server.d.ts.map +1 -1
- package/dist/dashboard-server/server.js +1279 -56
- package/dist/dashboard-server/server.js.map +1 -1
- package/dist/protocol/types.d.ts +10 -1
- package/dist/protocol/types.d.ts.map +1 -1
- package/dist/resiliency/context-persistence.d.ts +140 -0
- package/dist/resiliency/context-persistence.d.ts.map +1 -0
- package/dist/resiliency/context-persistence.js +397 -0
- package/dist/resiliency/context-persistence.js.map +1 -0
- package/dist/resiliency/health-monitor.d.ts +97 -0
- package/dist/resiliency/health-monitor.d.ts.map +1 -0
- package/dist/resiliency/health-monitor.js +291 -0
- package/dist/resiliency/health-monitor.js.map +1 -0
- package/dist/resiliency/index.d.ts +63 -0
- package/dist/resiliency/index.d.ts.map +1 -0
- package/dist/resiliency/index.js +63 -0
- package/dist/resiliency/index.js.map +1 -0
- package/dist/resiliency/logger.d.ts +114 -0
- package/dist/resiliency/logger.d.ts.map +1 -0
- package/dist/resiliency/logger.js +250 -0
- package/dist/resiliency/logger.js.map +1 -0
- package/dist/resiliency/metrics.d.ts +115 -0
- package/dist/resiliency/metrics.d.ts.map +1 -0
- package/dist/resiliency/metrics.js +239 -0
- package/dist/resiliency/metrics.js.map +1 -0
- package/dist/resiliency/provider-context.d.ts +100 -0
- package/dist/resiliency/provider-context.d.ts.map +1 -0
- package/dist/resiliency/provider-context.js +360 -0
- package/dist/resiliency/provider-context.js.map +1 -0
- package/dist/resiliency/supervisor.d.ts +109 -0
- package/dist/resiliency/supervisor.d.ts.map +1 -0
- package/dist/resiliency/supervisor.js +337 -0
- package/dist/resiliency/supervisor.js.map +1 -0
- package/dist/storage/adapter.d.ts +2 -0
- package/dist/storage/adapter.d.ts.map +1 -1
- package/dist/storage/adapter.js +12 -2
- package/dist/storage/adapter.js.map +1 -1
- package/dist/storage/sqlite-adapter.d.ts.map +1 -1
- package/dist/storage/sqlite-adapter.js +18 -14
- package/dist/storage/sqlite-adapter.js.map +1 -1
- package/dist/utils/index.d.ts +1 -0
- package/dist/utils/index.d.ts.map +1 -1
- package/dist/utils/index.js +1 -0
- package/dist/utils/index.js.map +1 -1
- package/dist/utils/logger.d.ts +40 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +84 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/wrapper/client.d.ts +16 -1
- package/dist/wrapper/client.d.ts.map +1 -1
- package/dist/wrapper/client.js +32 -1
- package/dist/wrapper/client.js.map +1 -1
- package/dist/wrapper/parser.d.ts +3 -0
- package/dist/wrapper/parser.d.ts.map +1 -1
- package/dist/wrapper/parser.js +121 -18
- package/dist/wrapper/parser.js.map +1 -1
- package/dist/wrapper/pty-wrapper.d.ts +28 -1
- package/dist/wrapper/pty-wrapper.d.ts.map +1 -1
- package/dist/wrapper/pty-wrapper.js +166 -30
- package/dist/wrapper/pty-wrapper.js.map +1 -1
- package/dist/wrapper/tmux-wrapper.d.ts +5 -0
- package/dist/wrapper/tmux-wrapper.d.ts.map +1 -1
- package/dist/wrapper/tmux-wrapper.js +58 -18
- package/dist/wrapper/tmux-wrapper.js.map +1 -1
- package/docs/CLOUD-ARCHITECTURE.md +652 -0
- package/docs/CLOUD-ONBOARDING-DESIGN.md +1983 -0
- package/docs/TESTING_PRESENCE_FEATURES.md +327 -0
- package/docs/agent-relay-snippet.md +107 -4
- package/docs/guides/CLOUD.md +236 -0
- package/docs/guides/LOCAL.md +535 -0
- package/docs/guides/SELF-HOSTED.md +494 -0
- package/docs/proposals/shadow-as-subagent.md +765 -0
- package/docs/proposals/slack-bot-integration.md +1457 -0
- package/package.json +33 -4
- package/dist/dashboard/out/_next/static/chunks/app/layout-c9d8c5d95e48c6bf.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/metrics/page-8aa9936bc6c771ab.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/app/page-49055e5d2b5e34ec.js +0 -1
- package/dist/dashboard/out/_next/static/chunks/main-app-bae2e535de00de50.js +0 -1
- package/dist/dashboard/out/_next/static/css/50ed6996e3df7bdd.css +0 -1
- /package/dist/dashboard/out/_next/static/{gZXwjIKGDKJ0hiTH-HMeJ → 6HHWb2ZmnJ4OSm0zUP7h4}/_buildManifest.js +0 -0
- /package/dist/dashboard/out/_next/static/{gZXwjIKGDKJ0hiTH-HMeJ → 6HHWb2ZmnJ4OSm0zUP7h4}/_ssgManifest.js +0 -0
- /package/dist/dashboard/out/_next/static/chunks/{117-3bef7b19f3e60751.js → 117-b2cd8d6485aacf2b.js} +0 -0
- /package/dist/dashboard/out/_next/static/chunks/{648-6cf686106c891ad3.js → 648-8f3f26864ce515e5.js} +0 -0
- /package/dist/dashboard/out/_next/static/chunks/app/_not-found/{page-8ff6572bc7c9bc61.js → page-0b990dbb71d72a98.js} +0 -0
- /package/dist/dashboard/out/_next/static/chunks/{fd9d1056-26bd8d656b496dba.js → fd9d1056-bf46c09eb57e019c.js} +0 -0
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Usage API Routes
|
|
3
|
+
*
|
|
4
|
+
* Track and report user resource usage against plan limits.
|
|
5
|
+
*/
|
|
6
|
+
import { Router } from 'express';
|
|
7
|
+
import { requireAuth } from './auth.js';
|
|
8
|
+
import { getRemainingQuota, getUserUsage, getPlanLimits } from '../services/planLimits.js';
|
|
9
|
+
import { db } from '../db/index.js';
|
|
10
|
+
export const usageRouter = Router();
|
|
11
|
+
// All routes require authentication
|
|
12
|
+
usageRouter.use(requireAuth);
|
|
13
|
+
/**
|
|
14
|
+
* GET /api/usage
|
|
15
|
+
* Get current usage vs limits for the authenticated user
|
|
16
|
+
*/
|
|
17
|
+
usageRouter.get('/', async (req, res) => {
|
|
18
|
+
const userId = req.session.userId;
|
|
19
|
+
try {
|
|
20
|
+
const user = await db.users.findById(userId);
|
|
21
|
+
if (!user) {
|
|
22
|
+
return res.status(404).json({ error: 'User not found' });
|
|
23
|
+
}
|
|
24
|
+
const plan = user.plan || 'free';
|
|
25
|
+
const quota = await getRemainingQuota(userId);
|
|
26
|
+
const calcPercent = (current, limit) => limit === Infinity ? 0 : Math.round((current / limit) * 100);
|
|
27
|
+
res.json({
|
|
28
|
+
plan,
|
|
29
|
+
limits: {
|
|
30
|
+
workspaces: quota.limits.maxWorkspaces,
|
|
31
|
+
repos: quota.limits.maxRepos,
|
|
32
|
+
concurrentAgents: quota.limits.maxConcurrentAgents,
|
|
33
|
+
computeHoursPerMonth: quota.limits.maxComputeHoursPerMonth,
|
|
34
|
+
coordinatorsEnabled: quota.limits.coordinatorsEnabled,
|
|
35
|
+
},
|
|
36
|
+
usage: {
|
|
37
|
+
workspaces: quota.usage.workspaceCount,
|
|
38
|
+
repos: quota.usage.repoCount,
|
|
39
|
+
concurrentAgents: quota.usage.concurrentAgents,
|
|
40
|
+
computeHoursThisMonth: quota.usage.computeHoursThisMonth,
|
|
41
|
+
},
|
|
42
|
+
remaining: {
|
|
43
|
+
workspaces: quota.remaining.workspaces,
|
|
44
|
+
repos: quota.remaining.repos,
|
|
45
|
+
concurrentAgents: quota.remaining.concurrentAgents,
|
|
46
|
+
computeHours: quota.remaining.computeHours,
|
|
47
|
+
},
|
|
48
|
+
percentUsed: {
|
|
49
|
+
workspaces: calcPercent(quota.usage.workspaceCount, quota.limits.maxWorkspaces),
|
|
50
|
+
repos: calcPercent(quota.usage.repoCount, quota.limits.maxRepos),
|
|
51
|
+
concurrentAgents: calcPercent(quota.usage.concurrentAgents, quota.limits.maxConcurrentAgents),
|
|
52
|
+
computeHours: calcPercent(quota.usage.computeHoursThisMonth, quota.limits.maxComputeHoursPerMonth),
|
|
53
|
+
},
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
catch (error) {
|
|
57
|
+
console.error('Error getting usage:', error);
|
|
58
|
+
res.status(500).json({ error: 'Failed to get usage' });
|
|
59
|
+
}
|
|
60
|
+
});
|
|
61
|
+
/**
|
|
62
|
+
* GET /api/usage/summary
|
|
63
|
+
* Get a quick summary of plan and usage status
|
|
64
|
+
*/
|
|
65
|
+
usageRouter.get('/summary', async (req, res) => {
|
|
66
|
+
const userId = req.session.userId;
|
|
67
|
+
try {
|
|
68
|
+
const user = await db.users.findById(userId);
|
|
69
|
+
if (!user) {
|
|
70
|
+
return res.status(404).json({ error: 'User not found' });
|
|
71
|
+
}
|
|
72
|
+
const plan = user.plan || 'free';
|
|
73
|
+
const limits = getPlanLimits(plan);
|
|
74
|
+
const usage = await getUserUsage(userId);
|
|
75
|
+
// Calculate warnings (when at 80%+ of limit)
|
|
76
|
+
const warnings = [];
|
|
77
|
+
const checkLimit = (resource, message, current, limit) => {
|
|
78
|
+
if (limit !== Infinity && current >= limit * 0.8) {
|
|
79
|
+
warnings.push({ resource, message, current, limit });
|
|
80
|
+
}
|
|
81
|
+
};
|
|
82
|
+
checkLimit('workspaces', 'Approaching workspace limit', usage.workspaceCount, limits.maxWorkspaces);
|
|
83
|
+
checkLimit('repos', 'Approaching repository limit', usage.repoCount, limits.maxRepos);
|
|
84
|
+
checkLimit('concurrentAgents', 'Approaching concurrent agent limit', usage.concurrentAgents, limits.maxConcurrentAgents);
|
|
85
|
+
checkLimit('computeHours', 'Approaching compute hours limit', usage.computeHoursThisMonth, limits.maxComputeHoursPerMonth);
|
|
86
|
+
res.json({
|
|
87
|
+
plan,
|
|
88
|
+
coordinatorsEnabled: limits.coordinatorsEnabled,
|
|
89
|
+
status: warnings.length > 0 ? 'warning' : 'healthy',
|
|
90
|
+
warnings,
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
catch (error) {
|
|
94
|
+
console.error('Error getting usage summary:', error);
|
|
95
|
+
res.status(500).json({ error: 'Failed to get usage summary' });
|
|
96
|
+
}
|
|
97
|
+
});
|
|
98
|
+
//# sourceMappingURL=usage.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"usage.js","sourceRoot":"","sources":["../../../src/cloud/api/usage.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,MAAM,EAAqB,MAAM,SAAS,CAAC;AACpD,OAAO,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AACxC,OAAO,EAAE,iBAAiB,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAC3F,OAAO,EAAE,EAAE,EAAY,MAAM,gBAAgB,CAAC;AAE9C,MAAM,CAAC,MAAM,WAAW,GAAG,MAAM,EAAE,CAAC;AAEpC,oCAAoC;AACpC,WAAW,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;AAE7B;;;GAGG;AACH,WAAW,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,EAAE,GAAY,EAAE,GAAa,EAAE,EAAE;IACzD,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,CAAC,MAAO,CAAC;IAEnC,IAAI,CAAC;QACH,MAAM,IAAI,GAAG,MAAM,EAAE,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAC7C,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,OAAO,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,gBAAgB,EAAE,CAAC,CAAC;QAC3D,CAAC;QAED,MAAM,IAAI,GAAI,IAAI,CAAC,IAAiB,IAAI,MAAM,CAAC;QAC/C,MAAM,KAAK,GAAG,MAAM,iBAAiB,CAAC,MAAM,CAAC,CAAC;QAE9C,MAAM,WAAW,GAAG,CAAC,OAAe,EAAE,KAAa,EAAE,EAAE,CACrD,KAAK,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,GAAG,KAAK,CAAC,GAAG,GAAG,CAAC,CAAC;QAE/D,GAAG,CAAC,IAAI,CAAC;YACP,IAAI;YACJ,MAAM,EAAE;gBACN,UAAU,EAAE,KAAK,CAAC,MAAM,CAAC,aAAa;gBACtC,KAAK,EAAE,KAAK,CAAC,MAAM,CAAC,QAAQ;gBAC5B,gBAAgB,EAAE,KAAK,CAAC,MAAM,CAAC,mBAAmB;gBAClD,oBAAoB,EAAE,KAAK,CAAC,MAAM,CAAC,uBAAuB;gBAC1D,mBAAmB,EAAE,KAAK,CAAC,MAAM,CAAC,mBAAmB;aACtD;YACD,KAAK,EAAE;gBACL,UAAU,EAAE,KAAK,CAAC,KAAK,CAAC,cAAc;gBACtC,KAAK,EAAE,KAAK,CAAC,KAAK,CAAC,SAAS;gBAC5B,gBAAgB,EAAE,KAAK,CAAC,KAAK,CAAC,gBAAgB;gBAC9C,qBAAqB,EAAE,KAAK,CAAC,KAAK,CAAC,qBAAqB;aACzD;YACD,SAAS,EAAE;gBACT,UAAU,EAAE,KAAK,CAAC,SAAS,CAAC,UAAU;gBACtC,KAAK,EAAE,KAAK,CAAC,SAAS,CAAC,KAAK;gBAC5B,gBAAgB,EAAE,KAAK,CAAC,SAAS,CAAC,gBAAgB;gBAClD,YAAY,EAAE,KAAK,CAAC,SAAS,CAAC,YAAY;aAC3C;YACD,WAAW,EAAE;gBACX,UAAU,EAAE,WAAW,CAAC,KAAK,CAAC,KAAK,CAAC,cAAc,EAAE,KAAK,CAAC,MAAM,CAAC,aAAa,CAAC;gBAC/E,KAAK,EAAE,WAAW,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,MAAM,CAAC,QAAQ,CAAC;gBAChE,gBAAgB,EAAE,WAAW,CAAC,KAAK,CAAC,KAAK,CAAC,gBAAgB,EAAE,KAAK,CAAC,MAAM,CAAC,mBAAmB,CAAC;gBAC7F,YAAY,EAAE,WAAW,CAAC,KAAK,CAAC,KAAK,CAAC,qBAAqB,EAAE,KAAK,CAAC,MAAM,CAAC,uBAAuB,CAAC;aACnG;SACF,CAAC,CAAC;IACL,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,sBAAsB,EAAE,KAAK,CAAC,CAAC;QAC7C,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,qBAAqB,EAAE,CAAC,CAAC;IACzD,CAAC;AACH,CAAC,CAAC,CAAC;AAEH;;;GAGG;AACH,WAAW,CAAC,GAAG,CAAC,UAAU,EAAE,KAAK,EAAE,GAAY,EAAE,GAAa,EAAE,EAAE;IAChE,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,CAAC,MAAO,CAAC;IAEnC,IAAI,CAAC;QACH,MAAM,IAAI,GAAG,MAAM,EAAE,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAC7C,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,OAAO,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,gBAAgB,EAAE,CAAC,CAAC;QAC3D,CAAC;QAED,MAAM,IAAI,GAAI,IAAI,CAAC,IAAiB,IAAI,MAAM,CAAC;QAC/C,MAAM,MAAM,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC;QACnC,MAAM,KAAK,GAAG,MAAM,YAAY,CAAC,MAAM,CAAC,CAAC;QAEzC,6CAA6C;QAC7C,MAAM,QAAQ,GAAiF,EAAE,CAAC;QAElG,MAAM,UAAU,GAAG,CACjB,QAAgB,EAChB,OAAe,EACf,OAAe,EACf,KAAa,EACb,EAAE;YACF,IAAI,KAAK,KAAK,QAAQ,IAAI,OAAO,IAAI,KAAK,GAAG,GAAG,EAAE,CAAC;gBACjD,QAAQ,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;YACvD,CAAC;QACH,CAAC,CAAC;QAEF,UAAU,CAAC,YAAY,EAAE,6BAA6B,EAAE,KAAK,CAAC,cAAc,EAAE,MAAM,CAAC,aAAa,CAAC,CAAC;QACpG,UAAU,CAAC,OAAO,EAAE,8BAA8B,EAAE,KAAK,CAAC,SAAS,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;QACtF,UAAU,CAAC,kBAAkB,EAAE,oCAAoC,EAAE,KAAK,CAAC,gBAAgB,EAAE,MAAM,CAAC,mBAAmB,CAAC,CAAC;QACzH,UAAU,CAAC,cAAc,EAAE,iCAAiC,EAAE,KAAK,CAAC,qBAAqB,EAAE,MAAM,CAAC,uBAAuB,CAAC,CAAC;QAE3H,GAAG,CAAC,IAAI,CAAC;YACP,IAAI;YACJ,mBAAmB,EAAE,MAAM,CAAC,mBAAmB;YAC/C,MAAM,EAAE,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS;YACnD,QAAQ;SACT,CAAC,CAAC;IACL,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;QACrD,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,6BAA6B,EAAE,CAAC,CAAC;IACjE,CAAC;AACH,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"workspaces.d.ts","sourceRoot":"","sources":["../../../src/cloud/api/workspaces.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAQH,eAAO,MAAM,gBAAgB,4CAAW,CAAC"}
|
|
@@ -0,0 +1,510 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Workspaces API Routes
|
|
3
|
+
*
|
|
4
|
+
* One-click workspace provisioning and management.
|
|
5
|
+
*/
|
|
6
|
+
import { Router } from 'express';
|
|
7
|
+
import { requireAuth } from './auth.js';
|
|
8
|
+
import { db } from '../db/index.js';
|
|
9
|
+
import { getProvisioner } from '../provisioner/index.js';
|
|
10
|
+
import { checkWorkspaceLimit } from './middleware/planLimits.js';
|
|
11
|
+
export const workspacesRouter = Router();
|
|
12
|
+
// All routes require authentication
|
|
13
|
+
workspacesRouter.use(requireAuth);
|
|
14
|
+
/**
|
|
15
|
+
* GET /api/workspaces
|
|
16
|
+
* List user's workspaces
|
|
17
|
+
*/
|
|
18
|
+
workspacesRouter.get('/', async (req, res) => {
|
|
19
|
+
const userId = req.session.userId;
|
|
20
|
+
try {
|
|
21
|
+
const workspaces = await db.workspaces.findByUserId(userId);
|
|
22
|
+
res.json({
|
|
23
|
+
workspaces: workspaces.map((w) => ({
|
|
24
|
+
id: w.id,
|
|
25
|
+
name: w.name,
|
|
26
|
+
status: w.status,
|
|
27
|
+
publicUrl: w.publicUrl,
|
|
28
|
+
providers: w.config.providers,
|
|
29
|
+
repositories: w.config.repositories,
|
|
30
|
+
createdAt: w.createdAt,
|
|
31
|
+
})),
|
|
32
|
+
});
|
|
33
|
+
}
|
|
34
|
+
catch (error) {
|
|
35
|
+
console.error('Error listing workspaces:', error);
|
|
36
|
+
res.status(500).json({ error: 'Failed to list workspaces' });
|
|
37
|
+
}
|
|
38
|
+
});
|
|
39
|
+
/**
|
|
40
|
+
* POST /api/workspaces
|
|
41
|
+
* Create (provision) a new workspace
|
|
42
|
+
*/
|
|
43
|
+
workspacesRouter.post('/', checkWorkspaceLimit, async (req, res) => {
|
|
44
|
+
const userId = req.session.userId;
|
|
45
|
+
const { name, providers, repositories, supervisorEnabled, maxAgents } = req.body;
|
|
46
|
+
// Validation
|
|
47
|
+
if (!name || typeof name !== 'string') {
|
|
48
|
+
return res.status(400).json({ error: 'Name is required' });
|
|
49
|
+
}
|
|
50
|
+
if (!providers || !Array.isArray(providers) || providers.length === 0) {
|
|
51
|
+
return res.status(400).json({ error: 'At least one provider is required' });
|
|
52
|
+
}
|
|
53
|
+
if (!repositories || !Array.isArray(repositories)) {
|
|
54
|
+
return res.status(400).json({ error: 'Repositories array is required' });
|
|
55
|
+
}
|
|
56
|
+
// Verify user has credentials for all providers
|
|
57
|
+
const credentials = await db.credentials.findByUserId(userId);
|
|
58
|
+
const connectedProviders = new Set(credentials.map((c) => c.provider));
|
|
59
|
+
for (const provider of providers) {
|
|
60
|
+
if (!connectedProviders.has(provider)) {
|
|
61
|
+
return res.status(400).json({
|
|
62
|
+
error: `Provider ${provider} not connected. Please connect it first.`,
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
try {
|
|
67
|
+
const provisioner = getProvisioner();
|
|
68
|
+
const result = await provisioner.provision({
|
|
69
|
+
userId,
|
|
70
|
+
name,
|
|
71
|
+
providers,
|
|
72
|
+
repositories,
|
|
73
|
+
supervisorEnabled,
|
|
74
|
+
maxAgents,
|
|
75
|
+
});
|
|
76
|
+
if (result.status === 'error') {
|
|
77
|
+
return res.status(500).json({
|
|
78
|
+
error: 'Failed to provision workspace',
|
|
79
|
+
details: result.error,
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
res.status(201).json({
|
|
83
|
+
workspaceId: result.workspaceId,
|
|
84
|
+
status: result.status,
|
|
85
|
+
publicUrl: result.publicUrl,
|
|
86
|
+
});
|
|
87
|
+
}
|
|
88
|
+
catch (error) {
|
|
89
|
+
console.error('Error creating workspace:', error);
|
|
90
|
+
res.status(500).json({ error: 'Failed to create workspace' });
|
|
91
|
+
}
|
|
92
|
+
});
|
|
93
|
+
/**
|
|
94
|
+
* GET /api/workspaces/:id
|
|
95
|
+
* Get workspace details
|
|
96
|
+
*/
|
|
97
|
+
workspacesRouter.get('/:id', async (req, res) => {
|
|
98
|
+
const userId = req.session.userId;
|
|
99
|
+
const { id } = req.params;
|
|
100
|
+
try {
|
|
101
|
+
const workspace = await db.workspaces.findById(id);
|
|
102
|
+
if (!workspace) {
|
|
103
|
+
return res.status(404).json({ error: 'Workspace not found' });
|
|
104
|
+
}
|
|
105
|
+
if (workspace.userId !== userId) {
|
|
106
|
+
return res.status(403).json({ error: 'Unauthorized' });
|
|
107
|
+
}
|
|
108
|
+
// Get repositories assigned to this workspace
|
|
109
|
+
const repositories = await db.repositories.findByWorkspaceId(id);
|
|
110
|
+
res.json({
|
|
111
|
+
id: workspace.id,
|
|
112
|
+
name: workspace.name,
|
|
113
|
+
status: workspace.status,
|
|
114
|
+
publicUrl: workspace.publicUrl,
|
|
115
|
+
computeProvider: workspace.computeProvider,
|
|
116
|
+
config: workspace.config,
|
|
117
|
+
errorMessage: workspace.errorMessage,
|
|
118
|
+
repositories: repositories.map((r) => ({
|
|
119
|
+
id: r.id,
|
|
120
|
+
fullName: r.githubFullName,
|
|
121
|
+
syncStatus: r.syncStatus,
|
|
122
|
+
lastSyncedAt: r.lastSyncedAt,
|
|
123
|
+
})),
|
|
124
|
+
createdAt: workspace.createdAt,
|
|
125
|
+
updatedAt: workspace.updatedAt,
|
|
126
|
+
});
|
|
127
|
+
}
|
|
128
|
+
catch (error) {
|
|
129
|
+
console.error('Error getting workspace:', error);
|
|
130
|
+
res.status(500).json({ error: 'Failed to get workspace' });
|
|
131
|
+
}
|
|
132
|
+
});
|
|
133
|
+
/**
|
|
134
|
+
* GET /api/workspaces/:id/status
|
|
135
|
+
* Get current workspace status (polls compute provider)
|
|
136
|
+
*/
|
|
137
|
+
workspacesRouter.get('/:id/status', async (req, res) => {
|
|
138
|
+
const userId = req.session.userId;
|
|
139
|
+
const { id } = req.params;
|
|
140
|
+
try {
|
|
141
|
+
const workspace = await db.workspaces.findById(id);
|
|
142
|
+
if (!workspace) {
|
|
143
|
+
return res.status(404).json({ error: 'Workspace not found' });
|
|
144
|
+
}
|
|
145
|
+
if (workspace.userId !== userId) {
|
|
146
|
+
return res.status(403).json({ error: 'Unauthorized' });
|
|
147
|
+
}
|
|
148
|
+
const provisioner = getProvisioner();
|
|
149
|
+
const status = await provisioner.getStatus(id);
|
|
150
|
+
res.json({ status });
|
|
151
|
+
}
|
|
152
|
+
catch (error) {
|
|
153
|
+
console.error('Error getting workspace status:', error);
|
|
154
|
+
res.status(500).json({ error: 'Failed to get status' });
|
|
155
|
+
}
|
|
156
|
+
});
|
|
157
|
+
/**
|
|
158
|
+
* POST /api/workspaces/:id/restart
|
|
159
|
+
* Restart a workspace
|
|
160
|
+
*/
|
|
161
|
+
workspacesRouter.post('/:id/restart', async (req, res) => {
|
|
162
|
+
const userId = req.session.userId;
|
|
163
|
+
const { id } = req.params;
|
|
164
|
+
try {
|
|
165
|
+
const workspace = await db.workspaces.findById(id);
|
|
166
|
+
if (!workspace) {
|
|
167
|
+
return res.status(404).json({ error: 'Workspace not found' });
|
|
168
|
+
}
|
|
169
|
+
if (workspace.userId !== userId) {
|
|
170
|
+
return res.status(403).json({ error: 'Unauthorized' });
|
|
171
|
+
}
|
|
172
|
+
const provisioner = getProvisioner();
|
|
173
|
+
await provisioner.restart(id);
|
|
174
|
+
res.json({ success: true, message: 'Workspace restarting' });
|
|
175
|
+
}
|
|
176
|
+
catch (error) {
|
|
177
|
+
console.error('Error restarting workspace:', error);
|
|
178
|
+
res.status(500).json({ error: 'Failed to restart workspace' });
|
|
179
|
+
}
|
|
180
|
+
});
|
|
181
|
+
/**
|
|
182
|
+
* POST /api/workspaces/:id/stop
|
|
183
|
+
* Stop a workspace
|
|
184
|
+
*/
|
|
185
|
+
workspacesRouter.post('/:id/stop', async (req, res) => {
|
|
186
|
+
const userId = req.session.userId;
|
|
187
|
+
const { id } = req.params;
|
|
188
|
+
try {
|
|
189
|
+
const workspace = await db.workspaces.findById(id);
|
|
190
|
+
if (!workspace) {
|
|
191
|
+
return res.status(404).json({ error: 'Workspace not found' });
|
|
192
|
+
}
|
|
193
|
+
if (workspace.userId !== userId) {
|
|
194
|
+
return res.status(403).json({ error: 'Unauthorized' });
|
|
195
|
+
}
|
|
196
|
+
const provisioner = getProvisioner();
|
|
197
|
+
await provisioner.stop(id);
|
|
198
|
+
res.json({ success: true, message: 'Workspace stopped' });
|
|
199
|
+
}
|
|
200
|
+
catch (error) {
|
|
201
|
+
console.error('Error stopping workspace:', error);
|
|
202
|
+
res.status(500).json({ error: 'Failed to stop workspace' });
|
|
203
|
+
}
|
|
204
|
+
});
|
|
205
|
+
/**
|
|
206
|
+
* DELETE /api/workspaces/:id
|
|
207
|
+
* Delete (deprovision) a workspace
|
|
208
|
+
*/
|
|
209
|
+
workspacesRouter.delete('/:id', async (req, res) => {
|
|
210
|
+
const userId = req.session.userId;
|
|
211
|
+
const { id } = req.params;
|
|
212
|
+
try {
|
|
213
|
+
const workspace = await db.workspaces.findById(id);
|
|
214
|
+
if (!workspace) {
|
|
215
|
+
return res.status(404).json({ error: 'Workspace not found' });
|
|
216
|
+
}
|
|
217
|
+
if (workspace.userId !== userId) {
|
|
218
|
+
return res.status(403).json({ error: 'Unauthorized' });
|
|
219
|
+
}
|
|
220
|
+
const provisioner = getProvisioner();
|
|
221
|
+
await provisioner.deprovision(id);
|
|
222
|
+
res.json({ success: true, message: 'Workspace deleted' });
|
|
223
|
+
}
|
|
224
|
+
catch (error) {
|
|
225
|
+
console.error('Error deleting workspace:', error);
|
|
226
|
+
res.status(500).json({ error: 'Failed to delete workspace' });
|
|
227
|
+
}
|
|
228
|
+
});
|
|
229
|
+
/**
|
|
230
|
+
* POST /api/workspaces/:id/repos
|
|
231
|
+
* Add repositories to a workspace
|
|
232
|
+
*/
|
|
233
|
+
workspacesRouter.post('/:id/repos', async (req, res) => {
|
|
234
|
+
const userId = req.session.userId;
|
|
235
|
+
const { id } = req.params;
|
|
236
|
+
const { repositoryIds } = req.body;
|
|
237
|
+
if (!repositoryIds || !Array.isArray(repositoryIds)) {
|
|
238
|
+
return res.status(400).json({ error: 'repositoryIds array is required' });
|
|
239
|
+
}
|
|
240
|
+
try {
|
|
241
|
+
const workspace = await db.workspaces.findById(id);
|
|
242
|
+
if (!workspace) {
|
|
243
|
+
return res.status(404).json({ error: 'Workspace not found' });
|
|
244
|
+
}
|
|
245
|
+
if (workspace.userId !== userId) {
|
|
246
|
+
return res.status(403).json({ error: 'Unauthorized' });
|
|
247
|
+
}
|
|
248
|
+
// Assign repositories to workspace
|
|
249
|
+
for (const repoId of repositoryIds) {
|
|
250
|
+
await db.repositories.assignToWorkspace(repoId, id);
|
|
251
|
+
}
|
|
252
|
+
res.json({ success: true, message: 'Repositories added' });
|
|
253
|
+
}
|
|
254
|
+
catch (error) {
|
|
255
|
+
console.error('Error adding repos to workspace:', error);
|
|
256
|
+
res.status(500).json({ error: 'Failed to add repositories' });
|
|
257
|
+
}
|
|
258
|
+
});
|
|
259
|
+
/**
|
|
260
|
+
* POST /api/workspaces/:id/domain
|
|
261
|
+
* Add or update custom domain (Premium feature - Team/Enterprise only)
|
|
262
|
+
*/
|
|
263
|
+
workspacesRouter.post('/:id/domain', async (req, res) => {
|
|
264
|
+
const userId = req.session.userId;
|
|
265
|
+
const { id } = req.params;
|
|
266
|
+
const { domain } = req.body;
|
|
267
|
+
if (!domain || typeof domain !== 'string') {
|
|
268
|
+
return res.status(400).json({ error: 'Domain is required' });
|
|
269
|
+
}
|
|
270
|
+
// Basic domain validation
|
|
271
|
+
const domainRegex = /^(?:[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?\.)+[a-zA-Z]{2,}$/;
|
|
272
|
+
if (!domainRegex.test(domain)) {
|
|
273
|
+
return res.status(400).json({ error: 'Invalid domain format' });
|
|
274
|
+
}
|
|
275
|
+
try {
|
|
276
|
+
const workspace = await db.workspaces.findById(id);
|
|
277
|
+
if (!workspace) {
|
|
278
|
+
return res.status(404).json({ error: 'Workspace not found' });
|
|
279
|
+
}
|
|
280
|
+
if (workspace.userId !== userId) {
|
|
281
|
+
return res.status(403).json({ error: 'Unauthorized' });
|
|
282
|
+
}
|
|
283
|
+
// Check if user has premium plan (Team/Enterprise)
|
|
284
|
+
const user = await db.users.findById(userId);
|
|
285
|
+
const hasPremium = user?.plan === 'team' || user?.plan === 'enterprise';
|
|
286
|
+
if (!hasPremium) {
|
|
287
|
+
return res.status(402).json({
|
|
288
|
+
error: 'Custom domains require Team or Enterprise plan',
|
|
289
|
+
upgrade: '/settings/billing',
|
|
290
|
+
});
|
|
291
|
+
}
|
|
292
|
+
// Check if domain is already in use
|
|
293
|
+
const existing = await db.workspaces.findByCustomDomain(domain);
|
|
294
|
+
if (existing && existing.id !== id) {
|
|
295
|
+
return res.status(409).json({ error: 'Domain already in use' });
|
|
296
|
+
}
|
|
297
|
+
// Set the custom domain (pending verification)
|
|
298
|
+
await db.workspaces.setCustomDomain(id, domain, 'pending');
|
|
299
|
+
// Return DNS instructions
|
|
300
|
+
res.json({
|
|
301
|
+
success: true,
|
|
302
|
+
domain,
|
|
303
|
+
status: 'pending',
|
|
304
|
+
instructions: {
|
|
305
|
+
type: 'CNAME',
|
|
306
|
+
name: domain,
|
|
307
|
+
value: workspace.publicUrl?.replace('https://', '') || `${id}.agentrelay.dev`,
|
|
308
|
+
ttl: 300,
|
|
309
|
+
},
|
|
310
|
+
verifyEndpoint: `/api/workspaces/${id}/domain/verify`,
|
|
311
|
+
message: 'Add the CNAME record to your DNS, then call the verify endpoint',
|
|
312
|
+
});
|
|
313
|
+
}
|
|
314
|
+
catch (error) {
|
|
315
|
+
console.error('Error setting custom domain:', error);
|
|
316
|
+
res.status(500).json({ error: 'Failed to set custom domain' });
|
|
317
|
+
}
|
|
318
|
+
});
|
|
319
|
+
/**
|
|
320
|
+
* POST /api/workspaces/:id/domain/verify
|
|
321
|
+
* Verify custom domain DNS is configured correctly
|
|
322
|
+
*/
|
|
323
|
+
workspacesRouter.post('/:id/domain/verify', async (req, res) => {
|
|
324
|
+
const userId = req.session.userId;
|
|
325
|
+
const { id } = req.params;
|
|
326
|
+
try {
|
|
327
|
+
const workspace = await db.workspaces.findById(id);
|
|
328
|
+
if (!workspace) {
|
|
329
|
+
return res.status(404).json({ error: 'Workspace not found' });
|
|
330
|
+
}
|
|
331
|
+
if (workspace.userId !== userId) {
|
|
332
|
+
return res.status(403).json({ error: 'Unauthorized' });
|
|
333
|
+
}
|
|
334
|
+
if (!workspace.customDomain) {
|
|
335
|
+
return res.status(400).json({ error: 'No custom domain configured' });
|
|
336
|
+
}
|
|
337
|
+
// Verify DNS resolution
|
|
338
|
+
const dns = await import('dns').then(m => m.promises);
|
|
339
|
+
try {
|
|
340
|
+
const records = await dns.resolveCname(workspace.customDomain);
|
|
341
|
+
const expectedTarget = workspace.publicUrl?.replace('https://', '') || `${id}.agentrelay.dev`;
|
|
342
|
+
if (records.some(r => r.includes(expectedTarget) || r.includes('agentrelay'))) {
|
|
343
|
+
// DNS is configured, now provision SSL cert
|
|
344
|
+
await db.workspaces.updateCustomDomainStatus(id, 'verifying');
|
|
345
|
+
// Trigger SSL cert provisioning on compute provider
|
|
346
|
+
// For Railway/Fly, this is automatic once domain is added
|
|
347
|
+
await provisionDomainSSL(workspace);
|
|
348
|
+
await db.workspaces.updateCustomDomainStatus(id, 'active');
|
|
349
|
+
res.json({
|
|
350
|
+
success: true,
|
|
351
|
+
status: 'active',
|
|
352
|
+
domain: workspace.customDomain,
|
|
353
|
+
message: 'Custom domain verified and SSL certificate provisioned',
|
|
354
|
+
});
|
|
355
|
+
}
|
|
356
|
+
else {
|
|
357
|
+
res.status(400).json({
|
|
358
|
+
success: false,
|
|
359
|
+
status: 'pending',
|
|
360
|
+
error: 'DNS not configured correctly',
|
|
361
|
+
expected: expectedTarget,
|
|
362
|
+
found: records,
|
|
363
|
+
});
|
|
364
|
+
}
|
|
365
|
+
}
|
|
366
|
+
catch (_dnsError) {
|
|
367
|
+
res.status(400).json({
|
|
368
|
+
success: false,
|
|
369
|
+
status: 'pending',
|
|
370
|
+
error: 'Could not resolve domain. DNS may not be configured yet.',
|
|
371
|
+
});
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
catch (error) {
|
|
375
|
+
console.error('Error verifying domain:', error);
|
|
376
|
+
res.status(500).json({ error: 'Failed to verify domain' });
|
|
377
|
+
}
|
|
378
|
+
});
|
|
379
|
+
/**
|
|
380
|
+
* DELETE /api/workspaces/:id/domain
|
|
381
|
+
* Remove custom domain
|
|
382
|
+
*/
|
|
383
|
+
workspacesRouter.delete('/:id/domain', async (req, res) => {
|
|
384
|
+
const userId = req.session.userId;
|
|
385
|
+
const { id } = req.params;
|
|
386
|
+
try {
|
|
387
|
+
const workspace = await db.workspaces.findById(id);
|
|
388
|
+
if (!workspace) {
|
|
389
|
+
return res.status(404).json({ error: 'Workspace not found' });
|
|
390
|
+
}
|
|
391
|
+
if (workspace.userId !== userId) {
|
|
392
|
+
return res.status(403).json({ error: 'Unauthorized' });
|
|
393
|
+
}
|
|
394
|
+
// Remove from compute provider
|
|
395
|
+
if (workspace.customDomain) {
|
|
396
|
+
await removeDomainFromCompute(workspace);
|
|
397
|
+
}
|
|
398
|
+
await db.workspaces.removeCustomDomain(id);
|
|
399
|
+
res.json({ success: true, message: 'Custom domain removed' });
|
|
400
|
+
}
|
|
401
|
+
catch (error) {
|
|
402
|
+
console.error('Error removing domain:', error);
|
|
403
|
+
res.status(500).json({ error: 'Failed to remove domain' });
|
|
404
|
+
}
|
|
405
|
+
});
|
|
406
|
+
/**
|
|
407
|
+
* Helper: Provision SSL for custom domain on compute provider
|
|
408
|
+
*/
|
|
409
|
+
async function provisionDomainSSL(workspace) {
|
|
410
|
+
const config = (await import('../config.js')).getConfig();
|
|
411
|
+
if (workspace.computeProvider === 'fly' && config.compute.fly) {
|
|
412
|
+
// Fly.io: Add certificate
|
|
413
|
+
await fetch(`https://api.machines.dev/v1/apps/ar-${workspace.id.substring(0, 8)}/certificates`, {
|
|
414
|
+
method: 'POST',
|
|
415
|
+
headers: {
|
|
416
|
+
Authorization: `Bearer ${config.compute.fly.apiToken}`,
|
|
417
|
+
'Content-Type': 'application/json',
|
|
418
|
+
},
|
|
419
|
+
body: JSON.stringify({ hostname: workspace.customDomain }),
|
|
420
|
+
});
|
|
421
|
+
}
|
|
422
|
+
else if (workspace.computeProvider === 'railway' && config.compute.railway) {
|
|
423
|
+
// Railway: Add custom domain via GraphQL
|
|
424
|
+
await fetch('https://backboard.railway.app/graphql/v2', {
|
|
425
|
+
method: 'POST',
|
|
426
|
+
headers: {
|
|
427
|
+
Authorization: `Bearer ${config.compute.railway.apiToken}`,
|
|
428
|
+
'Content-Type': 'application/json',
|
|
429
|
+
},
|
|
430
|
+
body: JSON.stringify({
|
|
431
|
+
query: `
|
|
432
|
+
mutation AddCustomDomain($input: CustomDomainCreateInput!) {
|
|
433
|
+
customDomainCreate(input: $input) { id }
|
|
434
|
+
}
|
|
435
|
+
`,
|
|
436
|
+
variables: {
|
|
437
|
+
input: {
|
|
438
|
+
projectId: workspace.computeId,
|
|
439
|
+
domain: workspace.customDomain,
|
|
440
|
+
},
|
|
441
|
+
},
|
|
442
|
+
}),
|
|
443
|
+
});
|
|
444
|
+
}
|
|
445
|
+
// Docker: Would need reverse proxy config (Caddy/nginx)
|
|
446
|
+
}
|
|
447
|
+
/**
|
|
448
|
+
* Helper: Remove custom domain from compute provider
|
|
449
|
+
*/
|
|
450
|
+
async function removeDomainFromCompute(workspace) {
|
|
451
|
+
const config = (await import('../config.js')).getConfig();
|
|
452
|
+
if (workspace.computeProvider === 'fly' && config.compute.fly) {
|
|
453
|
+
await fetch(`https://api.machines.dev/v1/apps/ar-${workspace.id.substring(0, 8)}/certificates/${workspace.customDomain}`, {
|
|
454
|
+
method: 'DELETE',
|
|
455
|
+
headers: { Authorization: `Bearer ${config.compute.fly.apiToken}` },
|
|
456
|
+
});
|
|
457
|
+
}
|
|
458
|
+
// Railway and Docker: similar cleanup
|
|
459
|
+
}
|
|
460
|
+
/**
|
|
461
|
+
* POST /api/workspaces/quick
|
|
462
|
+
* Quick provision: one-click with defaults
|
|
463
|
+
*/
|
|
464
|
+
workspacesRouter.post('/quick', checkWorkspaceLimit, async (req, res) => {
|
|
465
|
+
const userId = req.session.userId;
|
|
466
|
+
const { name, repositoryFullName } = req.body;
|
|
467
|
+
if (!repositoryFullName) {
|
|
468
|
+
return res.status(400).json({ error: 'Repository name is required' });
|
|
469
|
+
}
|
|
470
|
+
try {
|
|
471
|
+
// Get user's connected providers
|
|
472
|
+
const credentials = await db.credentials.findByUserId(userId);
|
|
473
|
+
const providers = credentials
|
|
474
|
+
.filter((c) => c.provider !== 'github')
|
|
475
|
+
.map((c) => c.provider);
|
|
476
|
+
if (providers.length === 0) {
|
|
477
|
+
return res.status(400).json({
|
|
478
|
+
error: 'No AI providers connected. Please connect at least one provider.',
|
|
479
|
+
});
|
|
480
|
+
}
|
|
481
|
+
// Create workspace with defaults
|
|
482
|
+
const provisioner = getProvisioner();
|
|
483
|
+
const workspaceName = name || `Workspace for ${repositoryFullName}`;
|
|
484
|
+
const result = await provisioner.provision({
|
|
485
|
+
userId,
|
|
486
|
+
name: workspaceName,
|
|
487
|
+
providers,
|
|
488
|
+
repositories: [repositoryFullName],
|
|
489
|
+
supervisorEnabled: true,
|
|
490
|
+
maxAgents: 10,
|
|
491
|
+
});
|
|
492
|
+
if (result.status === 'error') {
|
|
493
|
+
return res.status(500).json({
|
|
494
|
+
error: 'Failed to provision workspace',
|
|
495
|
+
details: result.error,
|
|
496
|
+
});
|
|
497
|
+
}
|
|
498
|
+
res.status(201).json({
|
|
499
|
+
workspaceId: result.workspaceId,
|
|
500
|
+
status: result.status,
|
|
501
|
+
publicUrl: result.publicUrl,
|
|
502
|
+
message: 'Workspace provisioned successfully!',
|
|
503
|
+
});
|
|
504
|
+
}
|
|
505
|
+
catch (error) {
|
|
506
|
+
console.error('Error quick provisioning:', error);
|
|
507
|
+
res.status(500).json({ error: 'Failed to provision workspace' });
|
|
508
|
+
}
|
|
509
|
+
});
|
|
510
|
+
//# sourceMappingURL=workspaces.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"workspaces.js","sourceRoot":"","sources":["../../../src/cloud/api/workspaces.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Agent Relay Cloud - Billing Module
|
|
3
|
+
*
|
|
4
|
+
* Stripe-based subscription management and billing.
|
|
5
|
+
*/
|
|
6
|
+
export * from './types.js';
|
|
7
|
+
export * from './plans.js';
|
|
8
|
+
export { BillingService, getBillingService } from './service.js';
|
|
9
|
+
//# sourceMappingURL=index.d.ts.map
|