agent-relay 2.0.23 → 2.0.25
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/src/cli/index.js +160 -17
- package/package.json +18 -52
- package/packages/api-types/package.json +1 -1
- package/packages/bridge/package.json +8 -8
- package/packages/cli-tester/package.json +1 -1
- package/packages/config/package.json +2 -2
- package/packages/continuity/package.json +1 -1
- package/packages/daemon/package.json +12 -12
- package/packages/hooks/package.json +4 -4
- package/packages/mcp/package.json +2 -2
- package/packages/memory/package.json +2 -2
- package/packages/policy/package.json +2 -2
- package/packages/protocol/package.json +1 -1
- package/packages/resiliency/package.json +1 -1
- package/packages/sdk/package.json +2 -2
- package/packages/spawner/package.json +1 -1
- package/packages/state/package.json +1 -1
- package/packages/storage/package.json +2 -2
- package/packages/telemetry/package.json +1 -1
- package/packages/trajectory/package.json +2 -2
- package/packages/user-directory/package.json +2 -2
- package/packages/utils/package.json +1 -1
- package/packages/wrapper/package.json +6 -6
- package/deploy/init-db.sql +0 -5
- package/deploy/scripts/setup-fly-workspaces.sh +0 -69
- package/deploy/scripts/setup-railway.sh +0 -75
- package/dist/src/cloud/index.d.ts +0 -8
- package/dist/src/cloud/index.js +0 -8
- package/packages/cloud/dist/api/admin.d.ts +0 -8
- package/packages/cloud/dist/api/admin.js +0 -225
- package/packages/cloud/dist/api/auth.d.ts +0 -20
- package/packages/cloud/dist/api/auth.js +0 -138
- package/packages/cloud/dist/api/billing.d.ts +0 -7
- package/packages/cloud/dist/api/billing.js +0 -564
- package/packages/cloud/dist/api/cli-pty-runner.d.ts +0 -53
- package/packages/cloud/dist/api/cli-pty-runner.js +0 -175
- package/packages/cloud/dist/api/codex-auth-helper.d.ts +0 -21
- package/packages/cloud/dist/api/codex-auth-helper.js +0 -327
- package/packages/cloud/dist/api/consensus.d.ts +0 -13
- package/packages/cloud/dist/api/consensus.js +0 -261
- package/packages/cloud/dist/api/coordinators.d.ts +0 -8
- package/packages/cloud/dist/api/coordinators.js +0 -750
- package/packages/cloud/dist/api/daemons.d.ts +0 -12
- package/packages/cloud/dist/api/daemons.js +0 -535
- package/packages/cloud/dist/api/email-auth.d.ts +0 -11
- package/packages/cloud/dist/api/email-auth.js +0 -347
- package/packages/cloud/dist/api/generic-webhooks.d.ts +0 -8
- package/packages/cloud/dist/api/generic-webhooks.js +0 -129
- package/packages/cloud/dist/api/git.d.ts +0 -8
- package/packages/cloud/dist/api/git.js +0 -269
- package/packages/cloud/dist/api/github-app.d.ts +0 -11
- package/packages/cloud/dist/api/github-app.js +0 -223
- package/packages/cloud/dist/api/middleware/planLimits.d.ts +0 -43
- package/packages/cloud/dist/api/middleware/planLimits.js +0 -202
- package/packages/cloud/dist/api/monitoring.d.ts +0 -11
- package/packages/cloud/dist/api/monitoring.js +0 -578
- package/packages/cloud/dist/api/nango-auth.d.ts +0 -9
- package/packages/cloud/dist/api/nango-auth.js +0 -741
- package/packages/cloud/dist/api/onboarding.d.ts +0 -15
- package/packages/cloud/dist/api/onboarding.js +0 -679
- package/packages/cloud/dist/api/policy.d.ts +0 -8
- package/packages/cloud/dist/api/policy.js +0 -229
- package/packages/cloud/dist/api/provider-env.d.ts +0 -26
- package/packages/cloud/dist/api/provider-env.js +0 -141
- package/packages/cloud/dist/api/providers.d.ts +0 -7
- package/packages/cloud/dist/api/providers.js +0 -574
- package/packages/cloud/dist/api/repos.d.ts +0 -8
- package/packages/cloud/dist/api/repos.js +0 -577
- package/packages/cloud/dist/api/sessions.d.ts +0 -11
- package/packages/cloud/dist/api/sessions.js +0 -302
- package/packages/cloud/dist/api/teams.d.ts +0 -7
- package/packages/cloud/dist/api/teams.js +0 -281
- package/packages/cloud/dist/api/test-helpers.d.ts +0 -10
- package/packages/cloud/dist/api/test-helpers.js +0 -745
- package/packages/cloud/dist/api/usage.d.ts +0 -7
- package/packages/cloud/dist/api/usage.js +0 -111
- package/packages/cloud/dist/api/webhooks.d.ts +0 -8
- package/packages/cloud/dist/api/webhooks.js +0 -645
- package/packages/cloud/dist/api/workspaces.d.ts +0 -25
- package/packages/cloud/dist/api/workspaces.js +0 -1799
- package/packages/cloud/dist/billing/index.d.ts +0 -9
- package/packages/cloud/dist/billing/index.js +0 -9
- package/packages/cloud/dist/billing/plans.d.ts +0 -39
- package/packages/cloud/dist/billing/plans.js +0 -245
- package/packages/cloud/dist/billing/service.d.ts +0 -80
- package/packages/cloud/dist/billing/service.js +0 -388
- package/packages/cloud/dist/billing/types.d.ts +0 -141
- package/packages/cloud/dist/billing/types.js +0 -7
- package/packages/cloud/dist/config.d.ts +0 -5
- package/packages/cloud/dist/config.js +0 -5
- package/packages/cloud/dist/db/bulk-ingest.d.ts +0 -89
- package/packages/cloud/dist/db/bulk-ingest.js +0 -268
- package/packages/cloud/dist/db/drizzle.d.ts +0 -290
- package/packages/cloud/dist/db/drizzle.js +0 -1422
- package/packages/cloud/dist/db/index.d.ts +0 -56
- package/packages/cloud/dist/db/index.js +0 -70
- package/packages/cloud/dist/db/schema.d.ts +0 -5117
- package/packages/cloud/dist/db/schema.js +0 -656
- package/packages/cloud/dist/index.d.ts +0 -11
- package/packages/cloud/dist/index.js +0 -38
- package/packages/cloud/dist/provisioner/index.d.ts +0 -207
- package/packages/cloud/dist/provisioner/index.js +0 -2118
- package/packages/cloud/dist/server.d.ts +0 -17
- package/packages/cloud/dist/server.js +0 -2055
- package/packages/cloud/dist/services/auto-scaler.d.ts +0 -152
- package/packages/cloud/dist/services/auto-scaler.js +0 -439
- package/packages/cloud/dist/services/capacity-manager.d.ts +0 -148
- package/packages/cloud/dist/services/capacity-manager.js +0 -449
- package/packages/cloud/dist/services/ci-agent-spawner.d.ts +0 -49
- package/packages/cloud/dist/services/ci-agent-spawner.js +0 -373
- package/packages/cloud/dist/services/cloud-message-bus.d.ts +0 -28
- package/packages/cloud/dist/services/cloud-message-bus.js +0 -19
- package/packages/cloud/dist/services/compute-enforcement.d.ts +0 -57
- package/packages/cloud/dist/services/compute-enforcement.js +0 -175
- package/packages/cloud/dist/services/coordinator.d.ts +0 -62
- package/packages/cloud/dist/services/coordinator.js +0 -389
- package/packages/cloud/dist/services/index.d.ts +0 -17
- package/packages/cloud/dist/services/index.js +0 -25
- package/packages/cloud/dist/services/intro-expiration.d.ts +0 -60
- package/packages/cloud/dist/services/intro-expiration.js +0 -252
- package/packages/cloud/dist/services/mention-handler.d.ts +0 -65
- package/packages/cloud/dist/services/mention-handler.js +0 -405
- package/packages/cloud/dist/services/nango.d.ts +0 -219
- package/packages/cloud/dist/services/nango.js +0 -424
- package/packages/cloud/dist/services/persistence.d.ts +0 -131
- package/packages/cloud/dist/services/persistence.js +0 -200
- package/packages/cloud/dist/services/planLimits.d.ts +0 -147
- package/packages/cloud/dist/services/planLimits.js +0 -335
- package/packages/cloud/dist/services/presence-registry.d.ts +0 -56
- package/packages/cloud/dist/services/presence-registry.js +0 -91
- package/packages/cloud/dist/services/scaling-orchestrator.d.ts +0 -159
- package/packages/cloud/dist/services/scaling-orchestrator.js +0 -502
- package/packages/cloud/dist/services/scaling-policy.d.ts +0 -121
- package/packages/cloud/dist/services/scaling-policy.js +0 -415
- package/packages/cloud/dist/services/ssh-security.d.ts +0 -31
- package/packages/cloud/dist/services/ssh-security.js +0 -63
- package/packages/cloud/dist/services/workspace-keepalive.d.ts +0 -76
- package/packages/cloud/dist/services/workspace-keepalive.js +0 -234
- package/packages/cloud/dist/shims/consensus.d.ts +0 -23
- package/packages/cloud/dist/shims/consensus.js +0 -5
- package/packages/cloud/dist/webhooks/index.d.ts +0 -24
- package/packages/cloud/dist/webhooks/index.js +0 -29
- package/packages/cloud/dist/webhooks/parsers/github.d.ts +0 -8
- package/packages/cloud/dist/webhooks/parsers/github.js +0 -234
- package/packages/cloud/dist/webhooks/parsers/index.d.ts +0 -23
- package/packages/cloud/dist/webhooks/parsers/index.js +0 -30
- package/packages/cloud/dist/webhooks/parsers/linear.d.ts +0 -9
- package/packages/cloud/dist/webhooks/parsers/linear.js +0 -258
- package/packages/cloud/dist/webhooks/parsers/slack.d.ts +0 -9
- package/packages/cloud/dist/webhooks/parsers/slack.js +0 -214
- package/packages/cloud/dist/webhooks/responders/github.d.ts +0 -8
- package/packages/cloud/dist/webhooks/responders/github.js +0 -73
- package/packages/cloud/dist/webhooks/responders/index.d.ts +0 -23
- package/packages/cloud/dist/webhooks/responders/index.js +0 -30
- package/packages/cloud/dist/webhooks/responders/linear.d.ts +0 -9
- package/packages/cloud/dist/webhooks/responders/linear.js +0 -149
- package/packages/cloud/dist/webhooks/responders/slack.d.ts +0 -20
- package/packages/cloud/dist/webhooks/responders/slack.js +0 -178
- package/packages/cloud/dist/webhooks/router.d.ts +0 -25
- package/packages/cloud/dist/webhooks/router.js +0 -504
- package/packages/cloud/dist/webhooks/rules-engine.d.ts +0 -24
- package/packages/cloud/dist/webhooks/rules-engine.js +0 -287
- package/packages/cloud/dist/webhooks/types.d.ts +0 -186
- package/packages/cloud/dist/webhooks/types.js +0 -8
- package/packages/cloud/package.json +0 -60
- package/scripts/run-migrations.js +0 -43
- package/scripts/setup-stripe-products.ts +0 -312
- package/scripts/verify-schema.js +0 -134
|
@@ -1,234 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Workspace Keepalive Service
|
|
3
|
-
*
|
|
4
|
-
* Prevents Fly.io from idling workspace machines that have active agents running.
|
|
5
|
-
*
|
|
6
|
-
* Problem: Fly.io uses request-based concurrency tracking to determine when to
|
|
7
|
-
* idle a machine. If a Claude agent is running but no HTTP requests are coming
|
|
8
|
-
* in (e.g., no one has the dashboard open), Fly.io may idle the machine.
|
|
9
|
-
*
|
|
10
|
-
* Solution: The cloud server periodically pings workspace machines that have
|
|
11
|
-
* active agents. This inbound HTTP request counts as activity for Fly.io's
|
|
12
|
-
* idle detection, keeping the machine awake.
|
|
13
|
-
*
|
|
14
|
-
* Flow:
|
|
15
|
-
* 1. Daemons report their running agents via heartbeat
|
|
16
|
-
* 2. This service queries for workspaces with active agents
|
|
17
|
-
* 3. Pings each workspace's /keep-alive endpoint
|
|
18
|
-
* 4. Workspace stays awake as long as agents are active
|
|
19
|
-
*/
|
|
20
|
-
import { EventEmitter } from 'events';
|
|
21
|
-
import { db } from '../db/index.js';
|
|
22
|
-
const DEFAULT_CONFIG = {
|
|
23
|
-
pingIntervalMs: 60_000, // 1 minute (well under Fly's ~5-10 min idle timeout)
|
|
24
|
-
requestTimeoutMs: 5_000, // 5 seconds
|
|
25
|
-
staleThresholdMs: 2 * 60 * 1000, // 2 minutes
|
|
26
|
-
verbose: false,
|
|
27
|
-
};
|
|
28
|
-
export class WorkspaceKeepaliveService extends EventEmitter {
|
|
29
|
-
config;
|
|
30
|
-
pingTimer = null;
|
|
31
|
-
stats = {
|
|
32
|
-
lastRun: null,
|
|
33
|
-
totalPings: 0,
|
|
34
|
-
successfulPings: 0,
|
|
35
|
-
failedPings: 0,
|
|
36
|
-
activeWorkspaces: 0,
|
|
37
|
-
};
|
|
38
|
-
constructor(config = {}) {
|
|
39
|
-
super();
|
|
40
|
-
this.config = { ...DEFAULT_CONFIG, ...config };
|
|
41
|
-
}
|
|
42
|
-
/**
|
|
43
|
-
* Start the keepalive service
|
|
44
|
-
*/
|
|
45
|
-
start() {
|
|
46
|
-
if (this.pingTimer) {
|
|
47
|
-
return; // Already running
|
|
48
|
-
}
|
|
49
|
-
console.log('[keepalive] Starting workspace keepalive service', {
|
|
50
|
-
intervalMs: this.config.pingIntervalMs,
|
|
51
|
-
});
|
|
52
|
-
// Initial ping
|
|
53
|
-
this.pingActiveWorkspaces().catch((err) => {
|
|
54
|
-
console.error('[keepalive] Initial ping failed:', err);
|
|
55
|
-
});
|
|
56
|
-
// Start periodic pings
|
|
57
|
-
this.pingTimer = setInterval(() => {
|
|
58
|
-
this.pingActiveWorkspaces().catch((err) => {
|
|
59
|
-
console.error('[keepalive] Periodic ping failed:', err);
|
|
60
|
-
});
|
|
61
|
-
}, this.config.pingIntervalMs);
|
|
62
|
-
}
|
|
63
|
-
/**
|
|
64
|
-
* Stop the keepalive service
|
|
65
|
-
*/
|
|
66
|
-
stop() {
|
|
67
|
-
if (this.pingTimer) {
|
|
68
|
-
clearInterval(this.pingTimer);
|
|
69
|
-
this.pingTimer = null;
|
|
70
|
-
console.log('[keepalive] Stopped workspace keepalive service');
|
|
71
|
-
}
|
|
72
|
-
}
|
|
73
|
-
/**
|
|
74
|
-
* Get current statistics
|
|
75
|
-
*/
|
|
76
|
-
getStats() {
|
|
77
|
-
return { ...this.stats };
|
|
78
|
-
}
|
|
79
|
-
/**
|
|
80
|
-
* Find workspaces with active agents and ping them
|
|
81
|
-
*/
|
|
82
|
-
async pingActiveWorkspaces() {
|
|
83
|
-
const startTime = Date.now();
|
|
84
|
-
try {
|
|
85
|
-
// Find workspaces with active agents
|
|
86
|
-
const activeWorkspaces = await this.findWorkspacesWithActiveAgents();
|
|
87
|
-
this.stats.activeWorkspaces = activeWorkspaces.length;
|
|
88
|
-
this.stats.lastRun = new Date();
|
|
89
|
-
if (activeWorkspaces.length === 0) {
|
|
90
|
-
if (this.config.verbose) {
|
|
91
|
-
console.log('[keepalive] No active workspaces to ping');
|
|
92
|
-
}
|
|
93
|
-
return;
|
|
94
|
-
}
|
|
95
|
-
if (this.config.verbose) {
|
|
96
|
-
console.log(`[keepalive] Pinging ${activeWorkspaces.length} active workspace(s)`);
|
|
97
|
-
}
|
|
98
|
-
// Ping each workspace in parallel
|
|
99
|
-
const results = await Promise.allSettled(activeWorkspaces.map((ws) => this.pingWorkspace(ws)));
|
|
100
|
-
// Update stats
|
|
101
|
-
for (const result of results) {
|
|
102
|
-
this.stats.totalPings++;
|
|
103
|
-
if (result.status === 'fulfilled' && result.value) {
|
|
104
|
-
this.stats.successfulPings++;
|
|
105
|
-
}
|
|
106
|
-
else {
|
|
107
|
-
this.stats.failedPings++;
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
const duration = Date.now() - startTime;
|
|
111
|
-
if (this.config.verbose) {
|
|
112
|
-
console.log(`[keepalive] Ping cycle complete`, {
|
|
113
|
-
workspaces: activeWorkspaces.length,
|
|
114
|
-
durationMs: duration,
|
|
115
|
-
});
|
|
116
|
-
}
|
|
117
|
-
this.emit('ping-cycle', {
|
|
118
|
-
workspaces: activeWorkspaces.length,
|
|
119
|
-
duration,
|
|
120
|
-
results: results.map((r) => r.status === 'fulfilled' && r.value),
|
|
121
|
-
});
|
|
122
|
-
}
|
|
123
|
-
catch (err) {
|
|
124
|
-
console.error('[keepalive] Error in ping cycle:', err);
|
|
125
|
-
this.emit('error', err);
|
|
126
|
-
}
|
|
127
|
-
}
|
|
128
|
-
/**
|
|
129
|
-
* Find all workspaces that have daemons with active agents
|
|
130
|
-
*/
|
|
131
|
-
async findWorkspacesWithActiveAgents() {
|
|
132
|
-
const staleThreshold = new Date(Date.now() - this.config.staleThresholdMs);
|
|
133
|
-
// Get all workspaces and check each for active agents
|
|
134
|
-
const allWorkspaces = await db.workspaces.findAll();
|
|
135
|
-
const activeWorkspaces = [];
|
|
136
|
-
for (const workspace of allWorkspaces) {
|
|
137
|
-
// Skip workspaces that aren't running or don't have a URL
|
|
138
|
-
if (workspace.status !== 'running' || !workspace.publicUrl) {
|
|
139
|
-
continue;
|
|
140
|
-
}
|
|
141
|
-
// Get daemons for this workspace
|
|
142
|
-
const daemons = await db.linkedDaemons.findByWorkspaceId(workspace.id);
|
|
143
|
-
for (const daemon of daemons) {
|
|
144
|
-
// Skip offline daemons or those with stale heartbeats
|
|
145
|
-
if (daemon.status !== 'online')
|
|
146
|
-
continue;
|
|
147
|
-
if (daemon.lastSeenAt && daemon.lastSeenAt < staleThreshold)
|
|
148
|
-
continue;
|
|
149
|
-
// Check if daemon has any active agents
|
|
150
|
-
const metadata = daemon.metadata;
|
|
151
|
-
const agents = metadata?.agents || [];
|
|
152
|
-
// Count agents that appear to be active (not offline/disconnected)
|
|
153
|
-
const activeAgents = agents.filter((a) => a.status === 'online' || a.status === 'running' || a.status === 'active');
|
|
154
|
-
if (activeAgents.length > 0) {
|
|
155
|
-
activeWorkspaces.push({
|
|
156
|
-
workspaceId: workspace.id,
|
|
157
|
-
publicUrl: workspace.publicUrl,
|
|
158
|
-
daemonId: daemon.id,
|
|
159
|
-
daemonName: daemon.name,
|
|
160
|
-
agentCount: activeAgents.length,
|
|
161
|
-
});
|
|
162
|
-
// Only need one daemon per workspace to keep it alive
|
|
163
|
-
break;
|
|
164
|
-
}
|
|
165
|
-
}
|
|
166
|
-
}
|
|
167
|
-
return activeWorkspaces;
|
|
168
|
-
}
|
|
169
|
-
/**
|
|
170
|
-
* Ping a single workspace's keep-alive endpoint
|
|
171
|
-
*/
|
|
172
|
-
async pingWorkspace(workspace) {
|
|
173
|
-
const url = `${workspace.publicUrl.replace(/\/$/, '')}/keep-alive`;
|
|
174
|
-
try {
|
|
175
|
-
const controller = new AbortController();
|
|
176
|
-
const timeout = setTimeout(() => controller.abort(), this.config.requestTimeoutMs);
|
|
177
|
-
const response = await fetch(url, {
|
|
178
|
-
method: 'GET',
|
|
179
|
-
signal: controller.signal,
|
|
180
|
-
headers: {
|
|
181
|
-
'User-Agent': 'AgentRelay-Keepalive/1.0',
|
|
182
|
-
},
|
|
183
|
-
});
|
|
184
|
-
clearTimeout(timeout);
|
|
185
|
-
if (response.ok) {
|
|
186
|
-
const data = await response.json();
|
|
187
|
-
if (this.config.verbose) {
|
|
188
|
-
console.log(`[keepalive] Pinged ${workspace.daemonName}`, {
|
|
189
|
-
workspaceId: workspace.workspaceId,
|
|
190
|
-
activeAgents: data.activeAgents,
|
|
191
|
-
});
|
|
192
|
-
}
|
|
193
|
-
return true;
|
|
194
|
-
}
|
|
195
|
-
else {
|
|
196
|
-
console.warn(`[keepalive] Ping failed for ${workspace.daemonName}:`, {
|
|
197
|
-
status: response.status,
|
|
198
|
-
url,
|
|
199
|
-
});
|
|
200
|
-
return false;
|
|
201
|
-
}
|
|
202
|
-
}
|
|
203
|
-
catch (err) {
|
|
204
|
-
// Don't log aborted requests as errors (timeout is expected for stopped machines)
|
|
205
|
-
if (err instanceof Error && err.name === 'AbortError') {
|
|
206
|
-
if (this.config.verbose) {
|
|
207
|
-
console.log(`[keepalive] Ping timeout for ${workspace.daemonName} (machine may be starting)`);
|
|
208
|
-
}
|
|
209
|
-
}
|
|
210
|
-
else {
|
|
211
|
-
console.warn(`[keepalive] Ping error for ${workspace.daemonName}:`, err);
|
|
212
|
-
}
|
|
213
|
-
return false;
|
|
214
|
-
}
|
|
215
|
-
}
|
|
216
|
-
}
|
|
217
|
-
// Singleton instance
|
|
218
|
-
let _keepaliveService = null;
|
|
219
|
-
/**
|
|
220
|
-
* Get or create the keepalive service singleton
|
|
221
|
-
*/
|
|
222
|
-
export function getWorkspaceKeepaliveService(config) {
|
|
223
|
-
if (!_keepaliveService) {
|
|
224
|
-
_keepaliveService = new WorkspaceKeepaliveService(config);
|
|
225
|
-
}
|
|
226
|
-
return _keepaliveService;
|
|
227
|
-
}
|
|
228
|
-
/**
|
|
229
|
-
* Create a new keepalive service (for testing)
|
|
230
|
-
*/
|
|
231
|
-
export function createWorkspaceKeepaliveService(config) {
|
|
232
|
-
return new WorkspaceKeepaliveService(config);
|
|
233
|
-
}
|
|
234
|
-
//# sourceMappingURL=workspace-keepalive.js.map
|
|
@@ -1,23 +0,0 @@
|
|
|
1
|
-
export interface Vote {
|
|
2
|
-
agent: string;
|
|
3
|
-
vote: string;
|
|
4
|
-
comment?: string;
|
|
5
|
-
timestamp?: number;
|
|
6
|
-
}
|
|
7
|
-
export interface Proposal {
|
|
8
|
-
id: string;
|
|
9
|
-
title: string;
|
|
10
|
-
description?: string;
|
|
11
|
-
status: string;
|
|
12
|
-
proposer?: string;
|
|
13
|
-
participants: string[];
|
|
14
|
-
votes: Vote[];
|
|
15
|
-
createdAt: number;
|
|
16
|
-
updatedAt?: number;
|
|
17
|
-
}
|
|
18
|
-
export interface ConsensusState {
|
|
19
|
-
proposals: Proposal[];
|
|
20
|
-
stats?: Record<string, unknown>;
|
|
21
|
-
}
|
|
22
|
-
export declare function getConsensus(_workspaceId?: string): Promise<ConsensusState>;
|
|
23
|
-
//# sourceMappingURL=consensus.d.ts.map
|
|
@@ -1,24 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Generic Webhook System
|
|
3
|
-
*
|
|
4
|
-
* A configurable webhook system that can handle events from any source
|
|
5
|
-
* (GitHub, GitLab, Linear, Slack, Jira, etc.) and route them to agents.
|
|
6
|
-
*
|
|
7
|
-
* Components:
|
|
8
|
-
* - Parsers: Transform source-specific payloads to normalized events
|
|
9
|
-
* - Responders: Send responses back to source systems
|
|
10
|
-
* - Rules Engine: Match events to actions based on configuration
|
|
11
|
-
* - Router: Orchestrates the full webhook processing pipeline
|
|
12
|
-
*/
|
|
13
|
-
export * from './types.js';
|
|
14
|
-
export { getParser, registerParser, parsers } from './parsers/index.js';
|
|
15
|
-
export { githubParser } from './parsers/github.js';
|
|
16
|
-
export { linearParser } from './parsers/linear.js';
|
|
17
|
-
export { slackParser } from './parsers/slack.js';
|
|
18
|
-
export { getResponder, registerResponder, responders } from './responders/index.js';
|
|
19
|
-
export { githubResponder } from './responders/github.js';
|
|
20
|
-
export { linearResponder } from './responders/linear.js';
|
|
21
|
-
export { slackResponder, formatSlackBlocks } from './responders/slack.js';
|
|
22
|
-
export { matchesRule, findMatchingRules, resolveActionTemplate, defaultRules, } from './rules-engine.js';
|
|
23
|
-
export { processWebhook, getWebhookConfig, defaultSources, } from './router.js';
|
|
24
|
-
//# sourceMappingURL=index.d.ts.map
|
|
@@ -1,29 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Generic Webhook System
|
|
3
|
-
*
|
|
4
|
-
* A configurable webhook system that can handle events from any source
|
|
5
|
-
* (GitHub, GitLab, Linear, Slack, Jira, etc.) and route them to agents.
|
|
6
|
-
*
|
|
7
|
-
* Components:
|
|
8
|
-
* - Parsers: Transform source-specific payloads to normalized events
|
|
9
|
-
* - Responders: Send responses back to source systems
|
|
10
|
-
* - Rules Engine: Match events to actions based on configuration
|
|
11
|
-
* - Router: Orchestrates the full webhook processing pipeline
|
|
12
|
-
*/
|
|
13
|
-
// Types
|
|
14
|
-
export * from './types.js';
|
|
15
|
-
// Parsers
|
|
16
|
-
export { getParser, registerParser, parsers } from './parsers/index.js';
|
|
17
|
-
export { githubParser } from './parsers/github.js';
|
|
18
|
-
export { linearParser } from './parsers/linear.js';
|
|
19
|
-
export { slackParser } from './parsers/slack.js';
|
|
20
|
-
// Responders
|
|
21
|
-
export { getResponder, registerResponder, responders } from './responders/index.js';
|
|
22
|
-
export { githubResponder } from './responders/github.js';
|
|
23
|
-
export { linearResponder } from './responders/linear.js';
|
|
24
|
-
export { slackResponder, formatSlackBlocks } from './responders/slack.js';
|
|
25
|
-
// Rules Engine
|
|
26
|
-
export { matchesRule, findMatchingRules, resolveActionTemplate, defaultRules, } from './rules-engine.js';
|
|
27
|
-
// Router
|
|
28
|
-
export { processWebhook, getWebhookConfig, defaultSources, } from './router.js';
|
|
29
|
-
//# sourceMappingURL=index.js.map
|
|
@@ -1,234 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* GitHub Webhook Parser
|
|
3
|
-
*
|
|
4
|
-
* Transforms GitHub webhook payloads into normalized events.
|
|
5
|
-
*/
|
|
6
|
-
/**
|
|
7
|
-
* Extract @mentions from text
|
|
8
|
-
*/
|
|
9
|
-
function extractMentions(text) {
|
|
10
|
-
if (!text)
|
|
11
|
-
return [];
|
|
12
|
-
const mentionPattern = /@([a-zA-Z][a-zA-Z0-9_-]*)/g;
|
|
13
|
-
const mentions = [];
|
|
14
|
-
let match;
|
|
15
|
-
while ((match = mentionPattern.exec(text)) !== null) {
|
|
16
|
-
mentions.push(match[1].toLowerCase());
|
|
17
|
-
}
|
|
18
|
-
return [...new Set(mentions)];
|
|
19
|
-
}
|
|
20
|
-
/**
|
|
21
|
-
* Map GitHub priority labels to normalized priority
|
|
22
|
-
*/
|
|
23
|
-
function extractPriority(labels) {
|
|
24
|
-
const labelNames = labels.map(l => l.name.toLowerCase());
|
|
25
|
-
if (labelNames.includes('critical') || labelNames.includes('p0'))
|
|
26
|
-
return 'critical';
|
|
27
|
-
if (labelNames.includes('high') || labelNames.includes('p1'))
|
|
28
|
-
return 'high';
|
|
29
|
-
if (labelNames.includes('medium') || labelNames.includes('p2'))
|
|
30
|
-
return 'medium';
|
|
31
|
-
if (labelNames.includes('low') || labelNames.includes('p3'))
|
|
32
|
-
return 'low';
|
|
33
|
-
return undefined;
|
|
34
|
-
}
|
|
35
|
-
export const githubParser = {
|
|
36
|
-
id: 'github',
|
|
37
|
-
parse(payload, headers) {
|
|
38
|
-
const eventType = headers['x-github-event'];
|
|
39
|
-
const deliveryId = headers['x-github-delivery'];
|
|
40
|
-
const data = payload;
|
|
41
|
-
const events = [];
|
|
42
|
-
const repository = data.repository;
|
|
43
|
-
const sender = data.sender;
|
|
44
|
-
const baseEvent = {
|
|
45
|
-
id: deliveryId || `github-${Date.now()}`,
|
|
46
|
-
source: 'github',
|
|
47
|
-
timestamp: new Date(),
|
|
48
|
-
actor: {
|
|
49
|
-
id: String(sender?.id || 'unknown'),
|
|
50
|
-
name: String(sender?.login || 'unknown'),
|
|
51
|
-
},
|
|
52
|
-
context: {
|
|
53
|
-
name: String(repository?.full_name || 'unknown'),
|
|
54
|
-
url: String(repository?.html_url || ''),
|
|
55
|
-
},
|
|
56
|
-
labels: [],
|
|
57
|
-
mentions: [],
|
|
58
|
-
metadata: {},
|
|
59
|
-
rawPayload: payload,
|
|
60
|
-
};
|
|
61
|
-
switch (eventType) {
|
|
62
|
-
case 'check_run': {
|
|
63
|
-
const checkRun = data.check_run;
|
|
64
|
-
const action = data.action;
|
|
65
|
-
const conclusion = checkRun?.conclusion;
|
|
66
|
-
const pullRequests = checkRun?.pull_requests;
|
|
67
|
-
if (action === 'completed' && conclusion === 'failure' && pullRequests?.length) {
|
|
68
|
-
const pr = pullRequests[0];
|
|
69
|
-
const output = checkRun.output;
|
|
70
|
-
const annotations = output?.annotations;
|
|
71
|
-
events.push({
|
|
72
|
-
...baseEvent,
|
|
73
|
-
type: 'ci_failure',
|
|
74
|
-
item: {
|
|
75
|
-
type: 'check',
|
|
76
|
-
id: String(checkRun.id),
|
|
77
|
-
number: pr.number,
|
|
78
|
-
title: String(checkRun.name),
|
|
79
|
-
body: String(output?.summary || ''),
|
|
80
|
-
url: String(checkRun.html_url || ''),
|
|
81
|
-
state: 'failure',
|
|
82
|
-
},
|
|
83
|
-
metadata: {
|
|
84
|
-
checkName: checkRun.name,
|
|
85
|
-
conclusion,
|
|
86
|
-
branch: pr.head?.ref,
|
|
87
|
-
commitSha: pr.head?.sha,
|
|
88
|
-
failureTitle: output?.title,
|
|
89
|
-
failureSummary: output?.summary,
|
|
90
|
-
failureDetails: output?.text,
|
|
91
|
-
annotations: annotations?.map(a => ({
|
|
92
|
-
path: a.path,
|
|
93
|
-
startLine: a.start_line,
|
|
94
|
-
endLine: a.end_line,
|
|
95
|
-
level: a.annotation_level,
|
|
96
|
-
message: a.message,
|
|
97
|
-
})),
|
|
98
|
-
},
|
|
99
|
-
});
|
|
100
|
-
}
|
|
101
|
-
break;
|
|
102
|
-
}
|
|
103
|
-
case 'issues': {
|
|
104
|
-
const issue = data.issue;
|
|
105
|
-
const action = data.action;
|
|
106
|
-
const labels = (issue?.labels || []);
|
|
107
|
-
if (action === 'opened' || action === 'labeled') {
|
|
108
|
-
events.push({
|
|
109
|
-
...baseEvent,
|
|
110
|
-
type: 'issue_created',
|
|
111
|
-
item: {
|
|
112
|
-
type: 'issue',
|
|
113
|
-
id: String(issue.id),
|
|
114
|
-
number: issue.number,
|
|
115
|
-
title: String(issue.title),
|
|
116
|
-
body: String(issue.body || ''),
|
|
117
|
-
url: String(issue.html_url),
|
|
118
|
-
state: String(issue.state),
|
|
119
|
-
},
|
|
120
|
-
labels: labels.map(l => l.name),
|
|
121
|
-
priority: extractPriority(labels),
|
|
122
|
-
mentions: extractMentions(issue.body),
|
|
123
|
-
metadata: {
|
|
124
|
-
action,
|
|
125
|
-
assignees: (issue.assignees || []).map(a => a.login),
|
|
126
|
-
},
|
|
127
|
-
});
|
|
128
|
-
}
|
|
129
|
-
break;
|
|
130
|
-
}
|
|
131
|
-
case 'issue_comment': {
|
|
132
|
-
const issue = data.issue;
|
|
133
|
-
const comment = data.comment;
|
|
134
|
-
const action = data.action;
|
|
135
|
-
const isPR = !!(issue?.pull_request);
|
|
136
|
-
if (action === 'created') {
|
|
137
|
-
const mentions = extractMentions(comment.body);
|
|
138
|
-
if (mentions.length > 0) {
|
|
139
|
-
events.push({
|
|
140
|
-
...baseEvent,
|
|
141
|
-
type: 'mention',
|
|
142
|
-
item: {
|
|
143
|
-
type: isPR ? 'pull_request' : 'issue',
|
|
144
|
-
id: String(comment.id),
|
|
145
|
-
number: issue.number,
|
|
146
|
-
title: String(issue.title),
|
|
147
|
-
body: String(comment.body),
|
|
148
|
-
url: String(comment.html_url),
|
|
149
|
-
},
|
|
150
|
-
mentions,
|
|
151
|
-
metadata: {
|
|
152
|
-
commentId: comment.id,
|
|
153
|
-
commentUrl: comment.html_url,
|
|
154
|
-
isPR,
|
|
155
|
-
},
|
|
156
|
-
});
|
|
157
|
-
}
|
|
158
|
-
}
|
|
159
|
-
break;
|
|
160
|
-
}
|
|
161
|
-
case 'pull_request_review_comment': {
|
|
162
|
-
const pr = data.pull_request;
|
|
163
|
-
const comment = data.comment;
|
|
164
|
-
const action = data.action;
|
|
165
|
-
if (action === 'created') {
|
|
166
|
-
const mentions = extractMentions(comment.body);
|
|
167
|
-
if (mentions.length > 0) {
|
|
168
|
-
events.push({
|
|
169
|
-
...baseEvent,
|
|
170
|
-
type: 'mention',
|
|
171
|
-
item: {
|
|
172
|
-
type: 'pull_request',
|
|
173
|
-
id: String(comment.id),
|
|
174
|
-
number: pr.number,
|
|
175
|
-
title: String(pr.title),
|
|
176
|
-
body: String(comment.body),
|
|
177
|
-
url: String(comment.html_url),
|
|
178
|
-
},
|
|
179
|
-
mentions,
|
|
180
|
-
metadata: {
|
|
181
|
-
commentId: comment.id,
|
|
182
|
-
commentUrl: comment.html_url,
|
|
183
|
-
filePath: comment.path,
|
|
184
|
-
line: comment.line,
|
|
185
|
-
isPR: true,
|
|
186
|
-
isReviewComment: true,
|
|
187
|
-
},
|
|
188
|
-
});
|
|
189
|
-
}
|
|
190
|
-
}
|
|
191
|
-
break;
|
|
192
|
-
}
|
|
193
|
-
case 'pull_request': {
|
|
194
|
-
const pr = data.pull_request;
|
|
195
|
-
const action = data.action;
|
|
196
|
-
const labels = (pr?.labels || []);
|
|
197
|
-
if (action === 'opened') {
|
|
198
|
-
events.push({
|
|
199
|
-
...baseEvent,
|
|
200
|
-
type: 'pr_opened',
|
|
201
|
-
item: {
|
|
202
|
-
type: 'pull_request',
|
|
203
|
-
id: String(pr.id),
|
|
204
|
-
number: pr.number,
|
|
205
|
-
title: String(pr.title),
|
|
206
|
-
body: String(pr.body || ''),
|
|
207
|
-
url: String(pr.html_url),
|
|
208
|
-
state: String(pr.state),
|
|
209
|
-
},
|
|
210
|
-
labels: labels.map(l => l.name),
|
|
211
|
-
priority: extractPriority(labels),
|
|
212
|
-
mentions: extractMentions(pr.body),
|
|
213
|
-
metadata: {
|
|
214
|
-
action,
|
|
215
|
-
head: pr.head?.ref,
|
|
216
|
-
base: pr.base?.ref,
|
|
217
|
-
draft: pr.draft,
|
|
218
|
-
},
|
|
219
|
-
});
|
|
220
|
-
}
|
|
221
|
-
break;
|
|
222
|
-
}
|
|
223
|
-
default:
|
|
224
|
-
// Unknown event type - create a generic event
|
|
225
|
-
events.push({
|
|
226
|
-
...baseEvent,
|
|
227
|
-
type: `github.${eventType}`,
|
|
228
|
-
metadata: { action: data.action },
|
|
229
|
-
});
|
|
230
|
-
}
|
|
231
|
-
return events;
|
|
232
|
-
},
|
|
233
|
-
};
|
|
234
|
-
//# sourceMappingURL=github.js.map
|
|
@@ -1,23 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Webhook Parsers Index
|
|
3
|
-
*
|
|
4
|
-
* Registry of all available parsers.
|
|
5
|
-
*/
|
|
6
|
-
import type { WebhookParser } from '../types.js';
|
|
7
|
-
import { githubParser } from './github.js';
|
|
8
|
-
import { linearParser } from './linear.js';
|
|
9
|
-
import { slackParser } from './slack.js';
|
|
10
|
-
/**
|
|
11
|
-
* Registry of all available parsers
|
|
12
|
-
*/
|
|
13
|
-
export declare const parsers: Record<string, WebhookParser>;
|
|
14
|
-
/**
|
|
15
|
-
* Get a parser by ID
|
|
16
|
-
*/
|
|
17
|
-
export declare function getParser(id: string): WebhookParser | undefined;
|
|
18
|
-
/**
|
|
19
|
-
* Register a custom parser
|
|
20
|
-
*/
|
|
21
|
-
export declare function registerParser(parser: WebhookParser): void;
|
|
22
|
-
export { githubParser, linearParser, slackParser };
|
|
23
|
-
//# sourceMappingURL=index.d.ts.map
|
|
@@ -1,30 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Webhook Parsers Index
|
|
3
|
-
*
|
|
4
|
-
* Registry of all available parsers.
|
|
5
|
-
*/
|
|
6
|
-
import { githubParser } from './github.js';
|
|
7
|
-
import { linearParser } from './linear.js';
|
|
8
|
-
import { slackParser } from './slack.js';
|
|
9
|
-
/**
|
|
10
|
-
* Registry of all available parsers
|
|
11
|
-
*/
|
|
12
|
-
export const parsers = {
|
|
13
|
-
github: githubParser,
|
|
14
|
-
linear: linearParser,
|
|
15
|
-
slack: slackParser,
|
|
16
|
-
};
|
|
17
|
-
/**
|
|
18
|
-
* Get a parser by ID
|
|
19
|
-
*/
|
|
20
|
-
export function getParser(id) {
|
|
21
|
-
return parsers[id];
|
|
22
|
-
}
|
|
23
|
-
/**
|
|
24
|
-
* Register a custom parser
|
|
25
|
-
*/
|
|
26
|
-
export function registerParser(parser) {
|
|
27
|
-
parsers[parser.id] = parser;
|
|
28
|
-
}
|
|
29
|
-
export { githubParser, linearParser, slackParser };
|
|
30
|
-
//# sourceMappingURL=index.js.map
|
|
@@ -1,9 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Linear Webhook Parser
|
|
3
|
-
*
|
|
4
|
-
* Transforms Linear webhook payloads into normalized events.
|
|
5
|
-
* Linear webhooks: https://developers.linear.app/docs/graphql/webhooks
|
|
6
|
-
*/
|
|
7
|
-
import type { WebhookParser } from '../types.js';
|
|
8
|
-
export declare const linearParser: WebhookParser;
|
|
9
|
-
//# sourceMappingURL=linear.d.ts.map
|