agents 0.0.0-c8f53b8 → 0.0.0-ca5cdf0
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/ai-chat-agent.d.ts +31 -5
- package/dist/ai-chat-agent.js +119 -115
- package/dist/ai-chat-agent.js.map +1 -1
- package/dist/ai-react.d.ts +17 -4
- package/dist/ai-react.js +28 -29
- package/dist/ai-react.js.map +1 -1
- package/dist/{chunk-HD4VEHBA.js → chunk-4CIGD73X.js} +341 -158
- package/dist/chunk-4CIGD73X.js.map +1 -0
- package/dist/chunk-767EASBA.js +106 -0
- package/dist/chunk-767EASBA.js.map +1 -0
- package/dist/{chunk-Q5ZBHY4Z.js → chunk-E3LCYPCB.js} +49 -36
- package/dist/chunk-E3LCYPCB.js.map +1 -0
- package/dist/chunk-NKZZ66QY.js +116 -0
- package/dist/chunk-NKZZ66QY.js.map +1 -0
- package/dist/client.d.ts +15 -1
- package/dist/client.js +6 -126
- package/dist/client.js.map +1 -1
- package/dist/index.d.ts +110 -12
- package/dist/index.js +4 -3
- package/dist/mcp/client.d.ts +301 -23
- package/dist/mcp/client.js +1 -2
- package/dist/mcp/do-oauth-client-provider.d.ts +3 -3
- package/dist/mcp/do-oauth-client-provider.js +3 -103
- package/dist/mcp/do-oauth-client-provider.js.map +1 -1
- package/dist/mcp/index.d.ts +17 -7
- package/dist/mcp/index.js +147 -173
- package/dist/mcp/index.js.map +1 -1
- package/dist/react.d.ts +85 -5
- package/dist/react.js +20 -8
- package/dist/react.js.map +1 -1
- package/dist/schedule.d.ts +2 -2
- package/dist/schedule.js +4 -6
- package/dist/schedule.js.map +1 -1
- package/dist/serializable.d.ts +32 -0
- package/dist/serializable.js +1 -0
- package/package.json +70 -68
- package/src/index.ts +395 -80
- package/dist/chunk-HD4VEHBA.js.map +0 -1
- package/dist/chunk-HMLY7DHA.js +0 -16
- package/dist/chunk-Q5ZBHY4Z.js.map +0 -1
- /package/dist/{chunk-HMLY7DHA.js.map → serializable.js.map} +0 -0
|
@@ -1,22 +1,22 @@
|
|
|
1
1
|
import {
|
|
2
2
|
MCPClientManager
|
|
3
|
-
} from "./chunk-
|
|
3
|
+
} from "./chunk-E3LCYPCB.js";
|
|
4
4
|
import {
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
} from "./chunk-
|
|
5
|
+
DurableObjectOAuthClientProvider
|
|
6
|
+
} from "./chunk-767EASBA.js";
|
|
7
|
+
import {
|
|
8
|
+
camelCaseToKebabCase
|
|
9
|
+
} from "./chunk-NKZZ66QY.js";
|
|
10
10
|
|
|
11
11
|
// src/index.ts
|
|
12
|
+
import { AsyncLocalStorage } from "async_hooks";
|
|
13
|
+
import { parseCronExpression } from "cron-schedule";
|
|
14
|
+
import { nanoid } from "nanoid";
|
|
12
15
|
import {
|
|
13
|
-
|
|
16
|
+
getServerByName,
|
|
14
17
|
routePartykitRequest,
|
|
15
|
-
|
|
18
|
+
Server
|
|
16
19
|
} from "partyserver";
|
|
17
|
-
import { parseCronExpression } from "cron-schedule";
|
|
18
|
-
import { nanoid } from "nanoid";
|
|
19
|
-
import { AsyncLocalStorage } from "node:async_hooks";
|
|
20
20
|
function isRPCRequest(msg) {
|
|
21
21
|
return typeof msg === "object" && msg !== null && "type" in msg && msg.type === "rpc" && "id" in msg && typeof msg.id === "string" && "method" in msg && typeof msg.method === "string" && "args" in msg && Array.isArray(msg.args);
|
|
22
22
|
}
|
|
@@ -51,19 +51,60 @@ function getCurrentAgent() {
|
|
|
51
51
|
}
|
|
52
52
|
return store;
|
|
53
53
|
}
|
|
54
|
-
var _state, _ParentClass, _Agent_instances, setStateInternal_fn, tryCatch_fn, scheduleNextAlarm_fn, isCallable_fn;
|
|
55
54
|
var Agent = class extends Server {
|
|
56
55
|
constructor(ctx, env) {
|
|
57
56
|
super(ctx, env);
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
this.mcp = new MCPClientManager(__privateGet(this, _ParentClass).name, "0.0.1");
|
|
57
|
+
this._state = DEFAULT_STATE;
|
|
58
|
+
this._ParentClass = Object.getPrototypeOf(this).constructor;
|
|
59
|
+
this.mcp = new MCPClientManager(this._ParentClass.name, "0.0.1");
|
|
62
60
|
/**
|
|
63
61
|
* Initial state for the Agent
|
|
64
62
|
* Override to provide default state values
|
|
65
63
|
*/
|
|
66
64
|
this.initialState = DEFAULT_STATE;
|
|
65
|
+
/**
|
|
66
|
+
* Method called when an alarm fires.
|
|
67
|
+
* Executes any scheduled tasks that are due.
|
|
68
|
+
*
|
|
69
|
+
* @remarks
|
|
70
|
+
* To schedule a task, please use the `this.schedule` method instead.
|
|
71
|
+
* See {@link https://developers.cloudflare.com/agents/api-reference/schedule-tasks/}
|
|
72
|
+
*/
|
|
73
|
+
this.alarm = async () => {
|
|
74
|
+
const now = Math.floor(Date.now() / 1e3);
|
|
75
|
+
const result = this.sql`
|
|
76
|
+
SELECT * FROM cf_agents_schedules WHERE time <= ${now}
|
|
77
|
+
`;
|
|
78
|
+
for (const row of result || []) {
|
|
79
|
+
const callback = this[row.callback];
|
|
80
|
+
if (!callback) {
|
|
81
|
+
console.error(`callback ${row.callback} not found`);
|
|
82
|
+
continue;
|
|
83
|
+
}
|
|
84
|
+
await agentContext.run(
|
|
85
|
+
{ agent: this, connection: void 0, request: void 0 },
|
|
86
|
+
async () => {
|
|
87
|
+
try {
|
|
88
|
+
await callback.bind(this)(JSON.parse(row.payload), row);
|
|
89
|
+
} catch (e) {
|
|
90
|
+
console.error(`error executing callback "${row.callback}"`, e);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
);
|
|
94
|
+
if (row.type === "cron") {
|
|
95
|
+
const nextExecutionTime = getNextCronTime(row.cron);
|
|
96
|
+
const nextTimestamp = Math.floor(nextExecutionTime.getTime() / 1e3);
|
|
97
|
+
this.sql`
|
|
98
|
+
UPDATE cf_agents_schedules SET time = ${nextTimestamp} WHERE id = ${row.id}
|
|
99
|
+
`;
|
|
100
|
+
} else {
|
|
101
|
+
this.sql`
|
|
102
|
+
DELETE FROM cf_agents_schedules WHERE id = ${row.id}
|
|
103
|
+
`;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
await this._scheduleNextAlarm();
|
|
107
|
+
};
|
|
67
108
|
this.sql`
|
|
68
109
|
CREATE TABLE IF NOT EXISTS cf_agents_state (
|
|
69
110
|
id TEXT PRIMARY KEY NOT NULL,
|
|
@@ -71,7 +112,7 @@ var Agent = class extends Server {
|
|
|
71
112
|
)
|
|
72
113
|
`;
|
|
73
114
|
void this.ctx.blockConcurrencyWhile(async () => {
|
|
74
|
-
return
|
|
115
|
+
return this._tryCatch(async () => {
|
|
75
116
|
this.sql`
|
|
76
117
|
CREATE TABLE IF NOT EXISTS cf_agents_schedules (
|
|
77
118
|
id TEXT PRIMARY KEY NOT NULL DEFAULT (randomblob(9)),
|
|
@@ -87,22 +128,55 @@ var Agent = class extends Server {
|
|
|
87
128
|
await this.alarm();
|
|
88
129
|
});
|
|
89
130
|
});
|
|
131
|
+
this.sql`
|
|
132
|
+
CREATE TABLE IF NOT EXISTS cf_agents_mcp_servers (
|
|
133
|
+
id TEXT PRIMARY KEY NOT NULL,
|
|
134
|
+
name TEXT NOT NULL,
|
|
135
|
+
server_url TEXT NOT NULL,
|
|
136
|
+
callback_url TEXT NOT NULL,
|
|
137
|
+
client_id TEXT,
|
|
138
|
+
auth_url TEXT,
|
|
139
|
+
server_options TEXT
|
|
140
|
+
)
|
|
141
|
+
`;
|
|
142
|
+
const _onRequest = this.onRequest.bind(this);
|
|
143
|
+
this.onRequest = (request) => {
|
|
144
|
+
return agentContext.run(
|
|
145
|
+
{ agent: this, connection: void 0, request },
|
|
146
|
+
async () => {
|
|
147
|
+
if (this.mcp.isCallbackRequest(request)) {
|
|
148
|
+
await this.mcp.handleCallbackRequest(request);
|
|
149
|
+
this.broadcast(
|
|
150
|
+
JSON.stringify({
|
|
151
|
+
mcp: this.getMcpServers(),
|
|
152
|
+
type: "cf_agent_mcp_servers"
|
|
153
|
+
})
|
|
154
|
+
);
|
|
155
|
+
return new Response("<script>window.close();</script>", {
|
|
156
|
+
headers: { "content-type": "text/html" },
|
|
157
|
+
status: 200
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
return this._tryCatch(() => _onRequest(request));
|
|
161
|
+
}
|
|
162
|
+
);
|
|
163
|
+
};
|
|
90
164
|
const _onMessage = this.onMessage.bind(this);
|
|
91
165
|
this.onMessage = async (connection, message) => {
|
|
92
166
|
return agentContext.run(
|
|
93
167
|
{ agent: this, connection, request: void 0 },
|
|
94
168
|
async () => {
|
|
95
169
|
if (typeof message !== "string") {
|
|
96
|
-
return
|
|
170
|
+
return this._tryCatch(() => _onMessage(connection, message));
|
|
97
171
|
}
|
|
98
172
|
let parsed;
|
|
99
173
|
try {
|
|
100
174
|
parsed = JSON.parse(message);
|
|
101
|
-
} catch (
|
|
102
|
-
return
|
|
175
|
+
} catch (_e) {
|
|
176
|
+
return this._tryCatch(() => _onMessage(connection, message));
|
|
103
177
|
}
|
|
104
178
|
if (isStateUpdateMessage(parsed)) {
|
|
105
|
-
|
|
179
|
+
this._setStateInternal(parsed.state, connection);
|
|
106
180
|
return;
|
|
107
181
|
}
|
|
108
182
|
if (isRPCRequest(parsed)) {
|
|
@@ -112,7 +186,7 @@ var Agent = class extends Server {
|
|
|
112
186
|
if (typeof methodFn !== "function") {
|
|
113
187
|
throw new Error(`Method ${method} does not exist`);
|
|
114
188
|
}
|
|
115
|
-
if (!
|
|
189
|
+
if (!this._isCallable(method)) {
|
|
116
190
|
throw new Error(`Method ${method} is not callable`);
|
|
117
191
|
}
|
|
118
192
|
const metadata = callableMetadata.get(methodFn);
|
|
@@ -123,26 +197,26 @@ var Agent = class extends Server {
|
|
|
123
197
|
}
|
|
124
198
|
const result = await methodFn.apply(this, args);
|
|
125
199
|
const response = {
|
|
126
|
-
|
|
200
|
+
done: true,
|
|
127
201
|
id,
|
|
128
|
-
success: true,
|
|
129
202
|
result,
|
|
130
|
-
|
|
203
|
+
success: true,
|
|
204
|
+
type: "rpc"
|
|
131
205
|
};
|
|
132
206
|
connection.send(JSON.stringify(response));
|
|
133
207
|
} catch (e) {
|
|
134
208
|
const response = {
|
|
135
|
-
|
|
209
|
+
error: e instanceof Error ? e.message : "Unknown error occurred",
|
|
136
210
|
id: parsed.id,
|
|
137
211
|
success: false,
|
|
138
|
-
|
|
212
|
+
type: "rpc"
|
|
139
213
|
};
|
|
140
214
|
connection.send(JSON.stringify(response));
|
|
141
215
|
console.error("RPC error:", e);
|
|
142
216
|
}
|
|
143
217
|
return;
|
|
144
218
|
}
|
|
145
|
-
return
|
|
219
|
+
return this._tryCatch(() => _onMessage(connection, message));
|
|
146
220
|
}
|
|
147
221
|
);
|
|
148
222
|
};
|
|
@@ -155,23 +229,62 @@ var Agent = class extends Server {
|
|
|
155
229
|
if (this.state) {
|
|
156
230
|
connection.send(
|
|
157
231
|
JSON.stringify({
|
|
158
|
-
|
|
159
|
-
|
|
232
|
+
state: this.state,
|
|
233
|
+
type: "cf_agent_state"
|
|
160
234
|
})
|
|
161
235
|
);
|
|
162
236
|
}
|
|
163
|
-
|
|
237
|
+
connection.send(
|
|
238
|
+
JSON.stringify({
|
|
239
|
+
mcp: this.getMcpServers(),
|
|
240
|
+
type: "cf_agent_mcp_servers"
|
|
241
|
+
})
|
|
242
|
+
);
|
|
243
|
+
return this._tryCatch(() => _onConnect(connection, ctx2));
|
|
164
244
|
}, 20);
|
|
165
245
|
}
|
|
166
246
|
);
|
|
167
247
|
};
|
|
248
|
+
const _onStart = this.onStart.bind(this);
|
|
249
|
+
this.onStart = async () => {
|
|
250
|
+
return agentContext.run(
|
|
251
|
+
{ agent: this, connection: void 0, request: void 0 },
|
|
252
|
+
async () => {
|
|
253
|
+
const servers = this.sql`
|
|
254
|
+
SELECT id, name, server_url, client_id, auth_url, callback_url, server_options FROM cf_agents_mcp_servers;
|
|
255
|
+
`;
|
|
256
|
+
Promise.allSettled(
|
|
257
|
+
servers.map((server) => {
|
|
258
|
+
return this._connectToMcpServerInternal(
|
|
259
|
+
server.name,
|
|
260
|
+
server.server_url,
|
|
261
|
+
server.callback_url,
|
|
262
|
+
server.server_options ? JSON.parse(server.server_options) : void 0,
|
|
263
|
+
{
|
|
264
|
+
id: server.id,
|
|
265
|
+
oauthClientId: server.client_id ?? void 0
|
|
266
|
+
}
|
|
267
|
+
);
|
|
268
|
+
})
|
|
269
|
+
).then((_results) => {
|
|
270
|
+
this.broadcast(
|
|
271
|
+
JSON.stringify({
|
|
272
|
+
mcp: this.getMcpServers(),
|
|
273
|
+
type: "cf_agent_mcp_servers"
|
|
274
|
+
})
|
|
275
|
+
);
|
|
276
|
+
});
|
|
277
|
+
await this._tryCatch(() => _onStart());
|
|
278
|
+
}
|
|
279
|
+
);
|
|
280
|
+
};
|
|
168
281
|
}
|
|
169
282
|
/**
|
|
170
283
|
* Current state of the Agent
|
|
171
284
|
*/
|
|
172
285
|
get state() {
|
|
173
|
-
if (
|
|
174
|
-
return
|
|
286
|
+
if (this._state !== DEFAULT_STATE) {
|
|
287
|
+
return this._state;
|
|
175
288
|
}
|
|
176
289
|
const wasChanged = this.sql`
|
|
177
290
|
SELECT state FROM cf_agents_state WHERE id = ${STATE_WAS_CHANGED}
|
|
@@ -182,8 +295,8 @@ var Agent = class extends Server {
|
|
|
182
295
|
if (wasChanged[0]?.state === "true" || // we do this check for people who updated their code before we shipped wasChanged
|
|
183
296
|
result[0]?.state) {
|
|
184
297
|
const state = result[0]?.state;
|
|
185
|
-
|
|
186
|
-
return
|
|
298
|
+
this._state = JSON.parse(state);
|
|
299
|
+
return this._state;
|
|
187
300
|
}
|
|
188
301
|
if (this.initialState === DEFAULT_STATE) {
|
|
189
302
|
return void 0;
|
|
@@ -211,24 +324,53 @@ var Agent = class extends Server {
|
|
|
211
324
|
throw this.onError(e);
|
|
212
325
|
}
|
|
213
326
|
}
|
|
327
|
+
_setStateInternal(state, source = "server") {
|
|
328
|
+
this._state = state;
|
|
329
|
+
this.sql`
|
|
330
|
+
INSERT OR REPLACE INTO cf_agents_state (id, state)
|
|
331
|
+
VALUES (${STATE_ROW_ID}, ${JSON.stringify(state)})
|
|
332
|
+
`;
|
|
333
|
+
this.sql`
|
|
334
|
+
INSERT OR REPLACE INTO cf_agents_state (id, state)
|
|
335
|
+
VALUES (${STATE_WAS_CHANGED}, ${JSON.stringify(true)})
|
|
336
|
+
`;
|
|
337
|
+
this.broadcast(
|
|
338
|
+
JSON.stringify({
|
|
339
|
+
state,
|
|
340
|
+
type: "cf_agent_state"
|
|
341
|
+
}),
|
|
342
|
+
source !== "server" ? [source.id] : []
|
|
343
|
+
);
|
|
344
|
+
return this._tryCatch(() => {
|
|
345
|
+
const { connection, request } = agentContext.getStore() || {};
|
|
346
|
+
return agentContext.run(
|
|
347
|
+
{ agent: this, connection, request },
|
|
348
|
+
async () => {
|
|
349
|
+
return this.onStateUpdate(state, source);
|
|
350
|
+
}
|
|
351
|
+
);
|
|
352
|
+
});
|
|
353
|
+
}
|
|
214
354
|
/**
|
|
215
355
|
* Update the Agent's state
|
|
216
356
|
* @param state New state to set
|
|
217
357
|
*/
|
|
218
358
|
setState(state) {
|
|
219
|
-
|
|
359
|
+
this._setStateInternal(state, "server");
|
|
220
360
|
}
|
|
221
361
|
/**
|
|
222
362
|
* Called when the Agent's state is updated
|
|
223
363
|
* @param state Updated state
|
|
224
364
|
* @param source Source of the state update ("server" or a client connection)
|
|
225
365
|
*/
|
|
366
|
+
// biome-ignore lint/correctness/noUnusedFunctionParameters: overridden later
|
|
226
367
|
onStateUpdate(state, source) {
|
|
227
368
|
}
|
|
228
369
|
/**
|
|
229
370
|
* Called when the Agent receives an email
|
|
230
371
|
* @param email Email message to process
|
|
231
372
|
*/
|
|
373
|
+
// biome-ignore lint/correctness/noUnusedFunctionParameters: overridden later
|
|
232
374
|
onEmail(email) {
|
|
233
375
|
return agentContext.run(
|
|
234
376
|
{ agent: this, connection: void 0, request: void 0 },
|
|
@@ -237,6 +379,13 @@ var Agent = class extends Server {
|
|
|
237
379
|
}
|
|
238
380
|
);
|
|
239
381
|
}
|
|
382
|
+
async _tryCatch(fn) {
|
|
383
|
+
try {
|
|
384
|
+
return await fn();
|
|
385
|
+
} catch (e) {
|
|
386
|
+
throw this.onError(e);
|
|
387
|
+
}
|
|
388
|
+
}
|
|
240
389
|
onError(connectionOrError, error) {
|
|
241
390
|
let theError;
|
|
242
391
|
if (connectionOrError && error) {
|
|
@@ -286,10 +435,10 @@ var Agent = class extends Server {
|
|
|
286
435
|
payload
|
|
287
436
|
)}, 'scheduled', ${timestamp})
|
|
288
437
|
`;
|
|
289
|
-
await
|
|
438
|
+
await this._scheduleNextAlarm();
|
|
290
439
|
return {
|
|
291
|
-
id,
|
|
292
440
|
callback,
|
|
441
|
+
id,
|
|
293
442
|
payload,
|
|
294
443
|
time: timestamp,
|
|
295
444
|
type: "scheduled"
|
|
@@ -304,12 +453,12 @@ var Agent = class extends Server {
|
|
|
304
453
|
payload
|
|
305
454
|
)}, 'delayed', ${when}, ${timestamp})
|
|
306
455
|
`;
|
|
307
|
-
await
|
|
456
|
+
await this._scheduleNextAlarm();
|
|
308
457
|
return {
|
|
309
|
-
id,
|
|
310
458
|
callback,
|
|
311
|
-
payload,
|
|
312
459
|
delayInSeconds: when,
|
|
460
|
+
id,
|
|
461
|
+
payload,
|
|
313
462
|
time: timestamp,
|
|
314
463
|
type: "delayed"
|
|
315
464
|
};
|
|
@@ -323,12 +472,12 @@ var Agent = class extends Server {
|
|
|
323
472
|
payload
|
|
324
473
|
)}, 'cron', ${when}, ${timestamp})
|
|
325
474
|
`;
|
|
326
|
-
await
|
|
475
|
+
await this._scheduleNextAlarm();
|
|
327
476
|
return {
|
|
328
|
-
id,
|
|
329
477
|
callback,
|
|
330
|
-
payload,
|
|
331
478
|
cron: when,
|
|
479
|
+
id,
|
|
480
|
+
payload,
|
|
332
481
|
time: timestamp,
|
|
333
482
|
type: "cron"
|
|
334
483
|
};
|
|
@@ -390,47 +539,21 @@ var Agent = class extends Server {
|
|
|
390
539
|
*/
|
|
391
540
|
async cancelSchedule(id) {
|
|
392
541
|
this.sql`DELETE FROM cf_agents_schedules WHERE id = ${id}`;
|
|
393
|
-
await
|
|
542
|
+
await this._scheduleNextAlarm();
|
|
394
543
|
return true;
|
|
395
544
|
}
|
|
396
|
-
|
|
397
|
-
* Method called when an alarm fires
|
|
398
|
-
* Executes any scheduled tasks that are due
|
|
399
|
-
*/
|
|
400
|
-
async alarm() {
|
|
401
|
-
const now = Math.floor(Date.now() / 1e3);
|
|
545
|
+
async _scheduleNextAlarm() {
|
|
402
546
|
const result = this.sql`
|
|
403
|
-
SELECT
|
|
547
|
+
SELECT time FROM cf_agents_schedules
|
|
548
|
+
WHERE time > ${Math.floor(Date.now() / 1e3)}
|
|
549
|
+
ORDER BY time ASC
|
|
550
|
+
LIMIT 1
|
|
404
551
|
`;
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
continue;
|
|
410
|
-
}
|
|
411
|
-
await agentContext.run(
|
|
412
|
-
{ agent: this, connection: void 0, request: void 0 },
|
|
413
|
-
async () => {
|
|
414
|
-
try {
|
|
415
|
-
await callback.bind(this)(JSON.parse(row.payload), row);
|
|
416
|
-
} catch (e) {
|
|
417
|
-
console.error(`error executing callback "${row.callback}"`, e);
|
|
418
|
-
}
|
|
419
|
-
}
|
|
420
|
-
);
|
|
421
|
-
if (row.type === "cron") {
|
|
422
|
-
const nextExecutionTime = getNextCronTime(row.cron);
|
|
423
|
-
const nextTimestamp = Math.floor(nextExecutionTime.getTime() / 1e3);
|
|
424
|
-
this.sql`
|
|
425
|
-
UPDATE cf_agents_schedules SET time = ${nextTimestamp} WHERE id = ${row.id}
|
|
426
|
-
`;
|
|
427
|
-
} else {
|
|
428
|
-
this.sql`
|
|
429
|
-
DELETE FROM cf_agents_schedules WHERE id = ${row.id}
|
|
430
|
-
`;
|
|
431
|
-
}
|
|
552
|
+
if (!result) return;
|
|
553
|
+
if (result.length > 0 && "time" in result[0]) {
|
|
554
|
+
const nextTime = result[0].time * 1e3;
|
|
555
|
+
await this.ctx.storage.setAlarm(nextTime);
|
|
432
556
|
}
|
|
433
|
-
await __privateMethod(this, _Agent_instances, scheduleNextAlarm_fn).call(this);
|
|
434
557
|
}
|
|
435
558
|
/**
|
|
436
559
|
* Destroy the Agent, removing all state and scheduled tasks
|
|
@@ -438,67 +561,133 @@ var Agent = class extends Server {
|
|
|
438
561
|
async destroy() {
|
|
439
562
|
this.sql`DROP TABLE IF EXISTS cf_agents_state`;
|
|
440
563
|
this.sql`DROP TABLE IF EXISTS cf_agents_schedules`;
|
|
564
|
+
this.sql`DROP TABLE IF EXISTS cf_agents_mcp_servers`;
|
|
441
565
|
await this.ctx.storage.deleteAlarm();
|
|
442
566
|
await this.ctx.storage.deleteAll();
|
|
567
|
+
this.ctx.abort("destroyed");
|
|
443
568
|
}
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
569
|
+
/**
|
|
570
|
+
* Get all methods marked as callable on this Agent
|
|
571
|
+
* @returns A map of method names to their metadata
|
|
572
|
+
*/
|
|
573
|
+
_isCallable(method) {
|
|
574
|
+
return callableMetadata.has(this[method]);
|
|
575
|
+
}
|
|
576
|
+
/**
|
|
577
|
+
* Connect to a new MCP Server
|
|
578
|
+
*
|
|
579
|
+
* @param url MCP Server SSE URL
|
|
580
|
+
* @param callbackHost Base host for the agent, used for the redirect URI.
|
|
581
|
+
* @param agentsPrefix agents routing prefix if not using `agents`
|
|
582
|
+
* @param options MCP client and transport (header) options
|
|
583
|
+
* @returns authUrl
|
|
584
|
+
*/
|
|
585
|
+
async addMcpServer(serverName, url, callbackHost, agentsPrefix = "agents", options) {
|
|
586
|
+
const callbackUrl = `${callbackHost}/${agentsPrefix}/${camelCaseToKebabCase(this._ParentClass.name)}/${this.name}/callback`;
|
|
587
|
+
const result = await this._connectToMcpServerInternal(
|
|
588
|
+
serverName,
|
|
589
|
+
url,
|
|
590
|
+
callbackUrl,
|
|
591
|
+
options
|
|
592
|
+
);
|
|
593
|
+
this.sql`
|
|
594
|
+
INSERT
|
|
595
|
+
OR REPLACE INTO cf_agents_mcp_servers (id, name, server_url, client_id, auth_url, callback_url, server_options)
|
|
596
|
+
VALUES (
|
|
597
|
+
${result.id},
|
|
598
|
+
${serverName},
|
|
599
|
+
${url},
|
|
600
|
+
${result.clientId ?? null},
|
|
601
|
+
${result.authUrl ?? null},
|
|
602
|
+
${callbackUrl},
|
|
603
|
+
${options ? JSON.stringify(options) : null}
|
|
604
|
+
);
|
|
605
|
+
`;
|
|
606
|
+
this.broadcast(
|
|
607
|
+
JSON.stringify({
|
|
608
|
+
mcp: this.getMcpServers(),
|
|
609
|
+
type: "cf_agent_mcp_servers"
|
|
610
|
+
})
|
|
611
|
+
);
|
|
612
|
+
return result;
|
|
613
|
+
}
|
|
614
|
+
async _connectToMcpServerInternal(_serverName, url, callbackUrl, options, reconnect) {
|
|
615
|
+
const authProvider = new DurableObjectOAuthClientProvider(
|
|
616
|
+
this.ctx.storage,
|
|
617
|
+
this.name,
|
|
618
|
+
callbackUrl
|
|
619
|
+
);
|
|
620
|
+
if (reconnect) {
|
|
621
|
+
authProvider.serverId = reconnect.id;
|
|
622
|
+
if (reconnect.oauthClientId) {
|
|
623
|
+
authProvider.clientId = reconnect.oauthClientId;
|
|
624
|
+
}
|
|
625
|
+
}
|
|
626
|
+
let headerTransportOpts = {};
|
|
627
|
+
if (options?.transport?.headers) {
|
|
628
|
+
headerTransportOpts = {
|
|
629
|
+
eventSourceInit: {
|
|
630
|
+
fetch: (url2, init) => fetch(url2, {
|
|
631
|
+
...init,
|
|
632
|
+
headers: options?.transport?.headers
|
|
633
|
+
})
|
|
634
|
+
},
|
|
635
|
+
requestInit: {
|
|
636
|
+
headers: options?.transport?.headers
|
|
637
|
+
}
|
|
638
|
+
};
|
|
639
|
+
}
|
|
640
|
+
const { id, authUrl, clientId } = await this.mcp.connect(url, {
|
|
641
|
+
client: options?.client,
|
|
642
|
+
reconnect,
|
|
643
|
+
transport: {
|
|
644
|
+
...headerTransportOpts,
|
|
645
|
+
authProvider
|
|
471
646
|
}
|
|
647
|
+
});
|
|
648
|
+
return {
|
|
649
|
+
authUrl,
|
|
650
|
+
clientId,
|
|
651
|
+
id
|
|
652
|
+
};
|
|
653
|
+
}
|
|
654
|
+
async removeMcpServer(id) {
|
|
655
|
+
this.mcp.closeConnection(id);
|
|
656
|
+
this.sql`
|
|
657
|
+
DELETE FROM cf_agents_mcp_servers WHERE id = ${id};
|
|
658
|
+
`;
|
|
659
|
+
this.broadcast(
|
|
660
|
+
JSON.stringify({
|
|
661
|
+
mcp: this.getMcpServers(),
|
|
662
|
+
type: "cf_agent_mcp_servers"
|
|
663
|
+
})
|
|
472
664
|
);
|
|
473
|
-
});
|
|
474
|
-
};
|
|
475
|
-
tryCatch_fn = async function(fn) {
|
|
476
|
-
try {
|
|
477
|
-
return await fn();
|
|
478
|
-
} catch (e) {
|
|
479
|
-
throw this.onError(e);
|
|
480
665
|
}
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
666
|
+
getMcpServers() {
|
|
667
|
+
const mcpState = {
|
|
668
|
+
prompts: this.mcp.listPrompts(),
|
|
669
|
+
resources: this.mcp.listResources(),
|
|
670
|
+
servers: {},
|
|
671
|
+
tools: this.mcp.listTools()
|
|
672
|
+
};
|
|
673
|
+
const servers = this.sql`
|
|
674
|
+
SELECT id, name, server_url, client_id, auth_url, callback_url, server_options FROM cf_agents_mcp_servers;
|
|
488
675
|
`;
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
676
|
+
for (const server of servers) {
|
|
677
|
+
const serverConn = this.mcp.mcpConnections[server.id];
|
|
678
|
+
mcpState.servers[server.id] = {
|
|
679
|
+
auth_url: server.auth_url,
|
|
680
|
+
capabilities: serverConn?.serverCapabilities ?? null,
|
|
681
|
+
instructions: serverConn?.instructions ?? null,
|
|
682
|
+
name: server.name,
|
|
683
|
+
server_url: server.server_url,
|
|
684
|
+
// mark as "authenticating" because the server isn't automatically connected, so it's pending authenticating
|
|
685
|
+
state: serverConn?.connectionState ?? "authenticating"
|
|
686
|
+
};
|
|
687
|
+
}
|
|
688
|
+
return mcpState;
|
|
493
689
|
}
|
|
494
690
|
};
|
|
495
|
-
/**
|
|
496
|
-
* Get all methods marked as callable on this Agent
|
|
497
|
-
* @returns A map of method names to their metadata
|
|
498
|
-
*/
|
|
499
|
-
isCallable_fn = function(method) {
|
|
500
|
-
return callableMetadata.has(this[method]);
|
|
501
|
-
};
|
|
502
691
|
/**
|
|
503
692
|
* Agent configuration options
|
|
504
693
|
*/
|
|
@@ -509,9 +698,9 @@ Agent.options = {
|
|
|
509
698
|
};
|
|
510
699
|
async function routeAgentRequest(request, env, options) {
|
|
511
700
|
const corsHeaders = options?.cors === true ? {
|
|
512
|
-
"Access-Control-Allow-Origin": "*",
|
|
513
|
-
"Access-Control-Allow-Methods": "GET, POST, HEAD, OPTIONS",
|
|
514
701
|
"Access-Control-Allow-Credentials": "true",
|
|
702
|
+
"Access-Control-Allow-Methods": "GET, POST, HEAD, OPTIONS",
|
|
703
|
+
"Access-Control-Allow-Origin": "*",
|
|
515
704
|
"Access-Control-Max-Age": "86400"
|
|
516
705
|
} : options?.cors;
|
|
517
706
|
if (request.method === "OPTIONS") {
|
|
@@ -542,59 +731,53 @@ async function routeAgentRequest(request, env, options) {
|
|
|
542
731
|
}
|
|
543
732
|
return response;
|
|
544
733
|
}
|
|
545
|
-
async function routeAgentEmail(
|
|
734
|
+
async function routeAgentEmail(_email, _env, _options) {
|
|
546
735
|
}
|
|
547
736
|
async function getAgentByName(namespace, name, options) {
|
|
548
737
|
return getServerByName(namespace, name, options);
|
|
549
738
|
}
|
|
550
|
-
var _connection, _id, _closed;
|
|
551
739
|
var StreamingResponse = class {
|
|
552
740
|
constructor(connection, id) {
|
|
553
|
-
|
|
554
|
-
|
|
555
|
-
|
|
556
|
-
__privateSet(this, _connection, connection);
|
|
557
|
-
__privateSet(this, _id, id);
|
|
741
|
+
this._closed = false;
|
|
742
|
+
this._connection = connection;
|
|
743
|
+
this._id = id;
|
|
558
744
|
}
|
|
559
745
|
/**
|
|
560
746
|
* Send a chunk of data to the client
|
|
561
747
|
* @param chunk The data to send
|
|
562
748
|
*/
|
|
563
749
|
send(chunk) {
|
|
564
|
-
if (
|
|
750
|
+
if (this._closed) {
|
|
565
751
|
throw new Error("StreamingResponse is already closed");
|
|
566
752
|
}
|
|
567
753
|
const response = {
|
|
568
|
-
|
|
569
|
-
id:
|
|
570
|
-
success: true,
|
|
754
|
+
done: false,
|
|
755
|
+
id: this._id,
|
|
571
756
|
result: chunk,
|
|
572
|
-
|
|
757
|
+
success: true,
|
|
758
|
+
type: "rpc"
|
|
573
759
|
};
|
|
574
|
-
|
|
760
|
+
this._connection.send(JSON.stringify(response));
|
|
575
761
|
}
|
|
576
762
|
/**
|
|
577
763
|
* End the stream and send the final chunk (if any)
|
|
578
764
|
* @param finalChunk Optional final chunk of data to send
|
|
579
765
|
*/
|
|
580
766
|
end(finalChunk) {
|
|
581
|
-
if (
|
|
767
|
+
if (this._closed) {
|
|
582
768
|
throw new Error("StreamingResponse is already closed");
|
|
583
769
|
}
|
|
584
|
-
|
|
770
|
+
this._closed = true;
|
|
585
771
|
const response = {
|
|
586
|
-
|
|
587
|
-
id:
|
|
588
|
-
success: true,
|
|
772
|
+
done: true,
|
|
773
|
+
id: this._id,
|
|
589
774
|
result: finalChunk,
|
|
590
|
-
|
|
775
|
+
success: true,
|
|
776
|
+
type: "rpc"
|
|
591
777
|
};
|
|
592
|
-
|
|
778
|
+
this._connection.send(JSON.stringify(response));
|
|
593
779
|
}
|
|
594
780
|
};
|
|
595
|
-
_connection = new WeakMap();
|
|
596
|
-
_id = new WeakMap();
|
|
597
|
-
_closed = new WeakMap();
|
|
598
781
|
|
|
599
782
|
export {
|
|
600
783
|
unstable_callable,
|
|
@@ -605,4 +788,4 @@ export {
|
|
|
605
788
|
getAgentByName,
|
|
606
789
|
StreamingResponse
|
|
607
790
|
};
|
|
608
|
-
//# sourceMappingURL=chunk-
|
|
791
|
+
//# sourceMappingURL=chunk-4CIGD73X.js.map
|