@agentuity/runtime 2.0.11 → 3.0.0-alpha.1
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/index.d.ts +37 -65
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +59 -61
- package/dist/index.js.map +1 -1
- package/package.json +9 -38
- package/src/index.ts +58 -259
- package/AGENTS.md +0 -116
- package/dist/_config.d.ts +0 -100
- package/dist/_config.d.ts.map +0 -1
- package/dist/_config.js +0 -147
- package/dist/_config.js.map +0 -1
- package/dist/_context.d.ts +0 -80
- package/dist/_context.d.ts.map +0 -1
- package/dist/_context.js +0 -160
- package/dist/_context.js.map +0 -1
- package/dist/_events.d.ts +0 -64
- package/dist/_events.d.ts.map +0 -1
- package/dist/_events.js +0 -92
- package/dist/_events.js.map +0 -1
- package/dist/_globals.d.ts +0 -58
- package/dist/_globals.d.ts.map +0 -1
- package/dist/_globals.js +0 -71
- package/dist/_globals.js.map +0 -1
- package/dist/_idle.d.ts +0 -7
- package/dist/_idle.d.ts.map +0 -1
- package/dist/_idle.js +0 -10
- package/dist/_idle.js.map +0 -1
- package/dist/_metadata.d.ts +0 -117
- package/dist/_metadata.d.ts.map +0 -1
- package/dist/_metadata.js +0 -268
- package/dist/_metadata.js.map +0 -1
- package/dist/_process-protection.d.ts +0 -27
- package/dist/_process-protection.d.ts.map +0 -1
- package/dist/_process-protection.js +0 -56
- package/dist/_process-protection.js.map +0 -1
- package/dist/_server.d.ts +0 -50
- package/dist/_server.d.ts.map +0 -1
- package/dist/_server.js +0 -89
- package/dist/_server.js.map +0 -1
- package/dist/_services.d.ts +0 -25
- package/dist/_services.d.ts.map +0 -1
- package/dist/_services.js +0 -286
- package/dist/_services.js.map +0 -1
- package/dist/_standalone.d.ts +0 -212
- package/dist/_standalone.d.ts.map +0 -1
- package/dist/_standalone.js +0 -556
- package/dist/_standalone.js.map +0 -1
- package/dist/_tokens.d.ts +0 -12
- package/dist/_tokens.d.ts.map +0 -1
- package/dist/_tokens.js +0 -97
- package/dist/_tokens.js.map +0 -1
- package/dist/_util.d.ts +0 -16
- package/dist/_util.d.ts.map +0 -1
- package/dist/_util.js +0 -54
- package/dist/_util.js.map +0 -1
- package/dist/_validation.d.ts +0 -89
- package/dist/_validation.d.ts.map +0 -1
- package/dist/_validation.js +0 -29
- package/dist/_validation.js.map +0 -1
- package/dist/_waituntil.d.ts +0 -32
- package/dist/_waituntil.d.ts.map +0 -1
- package/dist/_waituntil.js +0 -156
- package/dist/_waituntil.js.map +0 -1
- package/dist/agent.d.ts +0 -1262
- package/dist/agent.d.ts.map +0 -1
- package/dist/agent.js +0 -981
- package/dist/agent.js.map +0 -1
- package/dist/app.d.ts +0 -514
- package/dist/app.d.ts.map +0 -1
- package/dist/app.js +0 -228
- package/dist/app.js.map +0 -1
- package/dist/bootstrap.d.ts +0 -44
- package/dist/bootstrap.d.ts.map +0 -1
- package/dist/bootstrap.js +0 -259
- package/dist/bootstrap.js.map +0 -1
- package/dist/bun-s3-patch.d.ts +0 -37
- package/dist/bun-s3-patch.d.ts.map +0 -1
- package/dist/bun-s3-patch.js +0 -142
- package/dist/bun-s3-patch.js.map +0 -1
- package/dist/cors.d.ts +0 -42
- package/dist/cors.d.ts.map +0 -1
- package/dist/cors.js +0 -117
- package/dist/cors.js.map +0 -1
- package/dist/dev-patches/aisdk.d.ts +0 -17
- package/dist/dev-patches/aisdk.d.ts.map +0 -1
- package/dist/dev-patches/aisdk.js +0 -160
- package/dist/dev-patches/aisdk.js.map +0 -1
- package/dist/dev-patches/gateway.d.ts +0 -16
- package/dist/dev-patches/gateway.d.ts.map +0 -1
- package/dist/dev-patches/gateway.js +0 -54
- package/dist/dev-patches/gateway.js.map +0 -1
- package/dist/dev-patches/index.d.ts +0 -21
- package/dist/dev-patches/index.d.ts.map +0 -1
- package/dist/dev-patches/index.js +0 -33
- package/dist/dev-patches/index.js.map +0 -1
- package/dist/dev-patches/otel-llm.d.ts +0 -12
- package/dist/dev-patches/otel-llm.d.ts.map +0 -1
- package/dist/dev-patches/otel-llm.js +0 -352
- package/dist/dev-patches/otel-llm.js.map +0 -1
- package/dist/devmode.d.ts +0 -3
- package/dist/devmode.d.ts.map +0 -1
- package/dist/devmode.js +0 -167
- package/dist/devmode.js.map +0 -1
- package/dist/eval.d.ts +0 -91
- package/dist/eval.d.ts.map +0 -1
- package/dist/eval.js +0 -16
- package/dist/eval.js.map +0 -1
- package/dist/handlers/_route-meta.d.ts +0 -22
- package/dist/handlers/_route-meta.d.ts.map +0 -1
- package/dist/handlers/_route-meta.js +0 -25
- package/dist/handlers/_route-meta.js.map +0 -1
- package/dist/handlers/cron.d.ts +0 -73
- package/dist/handlers/cron.d.ts.map +0 -1
- package/dist/handlers/cron.js +0 -43
- package/dist/handlers/cron.js.map +0 -1
- package/dist/handlers/index.d.ts +0 -6
- package/dist/handlers/index.d.ts.map +0 -1
- package/dist/handlers/index.js +0 -6
- package/dist/handlers/index.js.map +0 -1
- package/dist/handlers/sse.d.ts +0 -163
- package/dist/handlers/sse.d.ts.map +0 -1
- package/dist/handlers/sse.js +0 -175
- package/dist/handlers/sse.js.map +0 -1
- package/dist/handlers/stream.d.ts +0 -52
- package/dist/handlers/stream.d.ts.map +0 -1
- package/dist/handlers/stream.js +0 -108
- package/dist/handlers/stream.js.map +0 -1
- package/dist/handlers/webrtc.d.ts +0 -49
- package/dist/handlers/webrtc.d.ts.map +0 -1
- package/dist/handlers/webrtc.js +0 -109
- package/dist/handlers/webrtc.js.map +0 -1
- package/dist/handlers/websocket.d.ts +0 -88
- package/dist/handlers/websocket.d.ts.map +0 -1
- package/dist/handlers/websocket.js +0 -161
- package/dist/handlers/websocket.js.map +0 -1
- package/dist/logger/console.d.ts +0 -70
- package/dist/logger/console.d.ts.map +0 -1
- package/dist/logger/console.js +0 -278
- package/dist/logger/console.js.map +0 -1
- package/dist/logger/index.d.ts +0 -3
- package/dist/logger/index.d.ts.map +0 -1
- package/dist/logger/index.js +0 -3
- package/dist/logger/index.js.map +0 -1
- package/dist/logger/internal.d.ts +0 -79
- package/dist/logger/internal.d.ts.map +0 -1
- package/dist/logger/internal.js +0 -133
- package/dist/logger/internal.js.map +0 -1
- package/dist/logger/logger.d.ts +0 -41
- package/dist/logger/logger.d.ts.map +0 -1
- package/dist/logger/logger.js +0 -2
- package/dist/logger/logger.js.map +0 -1
- package/dist/logger/user.d.ts +0 -8
- package/dist/logger/user.d.ts.map +0 -1
- package/dist/logger/user.js +0 -7
- package/dist/logger/user.js.map +0 -1
- package/dist/logger/util.d.ts +0 -11
- package/dist/logger/util.d.ts.map +0 -1
- package/dist/logger/util.js +0 -77
- package/dist/logger/util.js.map +0 -1
- package/dist/middleware.d.ts +0 -105
- package/dist/middleware.d.ts.map +0 -1
- package/dist/middleware.js +0 -763
- package/dist/middleware.js.map +0 -1
- package/dist/otel/config.d.ts +0 -19
- package/dist/otel/config.d.ts.map +0 -1
- package/dist/otel/config.js +0 -26
- package/dist/otel/config.js.map +0 -1
- package/dist/otel/console.d.ts +0 -33
- package/dist/otel/console.d.ts.map +0 -1
- package/dist/otel/console.js +0 -86
- package/dist/otel/console.js.map +0 -1
- package/dist/otel/exporters/index.d.ts +0 -4
- package/dist/otel/exporters/index.d.ts.map +0 -1
- package/dist/otel/exporters/index.js +0 -4
- package/dist/otel/exporters/index.js.map +0 -1
- package/dist/otel/exporters/jsonl-log-exporter.d.ts +0 -36
- package/dist/otel/exporters/jsonl-log-exporter.d.ts.map +0 -1
- package/dist/otel/exporters/jsonl-log-exporter.js +0 -103
- package/dist/otel/exporters/jsonl-log-exporter.js.map +0 -1
- package/dist/otel/exporters/jsonl-metric-exporter.d.ts +0 -40
- package/dist/otel/exporters/jsonl-metric-exporter.d.ts.map +0 -1
- package/dist/otel/exporters/jsonl-metric-exporter.js +0 -104
- package/dist/otel/exporters/jsonl-metric-exporter.js.map +0 -1
- package/dist/otel/exporters/jsonl-trace-exporter.d.ts +0 -36
- package/dist/otel/exporters/jsonl-trace-exporter.d.ts.map +0 -1
- package/dist/otel/exporters/jsonl-trace-exporter.js +0 -111
- package/dist/otel/exporters/jsonl-trace-exporter.js.map +0 -1
- package/dist/otel/fetch.d.ts +0 -12
- package/dist/otel/fetch.d.ts.map +0 -1
- package/dist/otel/fetch.js +0 -82
- package/dist/otel/fetch.js.map +0 -1
- package/dist/otel/http.d.ts +0 -16
- package/dist/otel/http.d.ts.map +0 -1
- package/dist/otel/http.js +0 -44
- package/dist/otel/http.js.map +0 -1
- package/dist/otel/logger.d.ts +0 -37
- package/dist/otel/logger.d.ts.map +0 -1
- package/dist/otel/logger.js +0 -265
- package/dist/otel/logger.js.map +0 -1
- package/dist/otel/otel.d.ts +0 -68
- package/dist/otel/otel.d.ts.map +0 -1
- package/dist/otel/otel.js +0 -245
- package/dist/otel/otel.js.map +0 -1
- package/dist/otel/tracestate.d.ts +0 -44
- package/dist/otel/tracestate.d.ts.map +0 -1
- package/dist/otel/tracestate.js +0 -84
- package/dist/otel/tracestate.js.map +0 -1
- package/dist/router.d.ts +0 -66
- package/dist/router.d.ts.map +0 -1
- package/dist/router.js +0 -44
- package/dist/router.js.map +0 -1
- package/dist/services/evalrun/composite.d.ts +0 -21
- package/dist/services/evalrun/composite.d.ts.map +0 -1
- package/dist/services/evalrun/composite.js +0 -26
- package/dist/services/evalrun/composite.js.map +0 -1
- package/dist/services/evalrun/http.d.ts +0 -24
- package/dist/services/evalrun/http.d.ts.map +0 -1
- package/dist/services/evalrun/http.js +0 -115
- package/dist/services/evalrun/http.js.map +0 -1
- package/dist/services/evalrun/index.d.ts +0 -5
- package/dist/services/evalrun/index.d.ts.map +0 -1
- package/dist/services/evalrun/index.js +0 -5
- package/dist/services/evalrun/index.js.map +0 -1
- package/dist/services/evalrun/json.d.ts +0 -21
- package/dist/services/evalrun/json.d.ts.map +0 -1
- package/dist/services/evalrun/json.js +0 -38
- package/dist/services/evalrun/json.js.map +0 -1
- package/dist/services/evalrun/local.d.ts +0 -19
- package/dist/services/evalrun/local.d.ts.map +0 -1
- package/dist/services/evalrun/local.js +0 -22
- package/dist/services/evalrun/local.js.map +0 -1
- package/dist/services/local/_db.d.ts +0 -4
- package/dist/services/local/_db.d.ts.map +0 -1
- package/dist/services/local/_db.js +0 -281
- package/dist/services/local/_db.js.map +0 -1
- package/dist/services/local/_router.d.ts +0 -3
- package/dist/services/local/_router.d.ts.map +0 -1
- package/dist/services/local/_router.js +0 -28
- package/dist/services/local/_router.js.map +0 -1
- package/dist/services/local/_util.d.ts +0 -18
- package/dist/services/local/_util.d.ts.map +0 -1
- package/dist/services/local/_util.js +0 -44
- package/dist/services/local/_util.js.map +0 -1
- package/dist/services/local/email.d.ts +0 -24
- package/dist/services/local/email.d.ts.map +0 -1
- package/dist/services/local/email.js +0 -58
- package/dist/services/local/email.js.map +0 -1
- package/dist/services/local/index.d.ts +0 -10
- package/dist/services/local/index.d.ts.map +0 -1
- package/dist/services/local/index.js +0 -10
- package/dist/services/local/index.js.map +0 -1
- package/dist/services/local/keyvalue.d.ts +0 -17
- package/dist/services/local/keyvalue.d.ts.map +0 -1
- package/dist/services/local/keyvalue.js +0 -133
- package/dist/services/local/keyvalue.js.map +0 -1
- package/dist/services/local/queue.d.ts +0 -10
- package/dist/services/local/queue.d.ts.map +0 -1
- package/dist/services/local/queue.js +0 -96
- package/dist/services/local/queue.js.map +0 -1
- package/dist/services/local/stream.d.ts +0 -12
- package/dist/services/local/stream.d.ts.map +0 -1
- package/dist/services/local/stream.js +0 -266
- package/dist/services/local/stream.js.map +0 -1
- package/dist/services/local/task.d.ts +0 -55
- package/dist/services/local/task.d.ts.map +0 -1
- package/dist/services/local/task.js +0 -1248
- package/dist/services/local/task.js.map +0 -1
- package/dist/services/local/vector.d.ts +0 -17
- package/dist/services/local/vector.d.ts.map +0 -1
- package/dist/services/local/vector.js +0 -303
- package/dist/services/local/vector.js.map +0 -1
- package/dist/services/sandbox/http.d.ts +0 -23
- package/dist/services/sandbox/http.d.ts.map +0 -1
- package/dist/services/sandbox/http.js +0 -327
- package/dist/services/sandbox/http.js.map +0 -1
- package/dist/services/sandbox/index.d.ts +0 -2
- package/dist/services/sandbox/index.d.ts.map +0 -1
- package/dist/services/sandbox/index.js +0 -2
- package/dist/services/sandbox/index.js.map +0 -1
- package/dist/services/session/composite.d.ts +0 -21
- package/dist/services/session/composite.d.ts.map +0 -1
- package/dist/services/session/composite.js +0 -26
- package/dist/services/session/composite.js.map +0 -1
- package/dist/services/session/http.d.ts +0 -34
- package/dist/services/session/http.d.ts.map +0 -1
- package/dist/services/session/http.js +0 -124
- package/dist/services/session/http.js.map +0 -1
- package/dist/services/session/index.d.ts +0 -5
- package/dist/services/session/index.d.ts.map +0 -1
- package/dist/services/session/index.js +0 -5
- package/dist/services/session/index.js.map +0 -1
- package/dist/services/session/json.d.ts +0 -22
- package/dist/services/session/json.d.ts.map +0 -1
- package/dist/services/session/json.js +0 -35
- package/dist/services/session/json.js.map +0 -1
- package/dist/services/session/local.d.ts +0 -19
- package/dist/services/session/local.d.ts.map +0 -1
- package/dist/services/session/local.js +0 -23
- package/dist/services/session/local.js.map +0 -1
- package/dist/services/thread/local.d.ts +0 -20
- package/dist/services/thread/local.d.ts.map +0 -1
- package/dist/services/thread/local.js +0 -158
- package/dist/services/thread/local.js.map +0 -1
- package/dist/session.d.ts +0 -734
- package/dist/session.d.ts.map +0 -1
- package/dist/session.js +0 -1140
- package/dist/session.js.map +0 -1
- package/dist/signature.d.ts +0 -22
- package/dist/signature.d.ts.map +0 -1
- package/dist/signature.js +0 -63
- package/dist/signature.js.map +0 -1
- package/dist/validator.d.ts +0 -142
- package/dist/validator.d.ts.map +0 -1
- package/dist/validator.js +0 -149
- package/dist/validator.js.map +0 -1
- package/dist/version-check.d.ts +0 -20
- package/dist/version-check.d.ts.map +0 -1
- package/dist/version-check.js +0 -157
- package/dist/version-check.js.map +0 -1
- package/dist/web.d.ts +0 -8
- package/dist/web.d.ts.map +0 -1
- package/dist/web.js +0 -67
- package/dist/web.js.map +0 -1
- package/dist/webrtc-signaling.d.ts +0 -80
- package/dist/webrtc-signaling.d.ts.map +0 -1
- package/dist/webrtc-signaling.js +0 -237
- package/dist/webrtc-signaling.js.map +0 -1
- package/dist/workbench.d.ts +0 -17
- package/dist/workbench.d.ts.map +0 -1
- package/dist/workbench.js +0 -605
- package/dist/workbench.js.map +0 -1
- package/src/_config.ts +0 -163
- package/src/_context.ts +0 -240
- package/src/_events.ts +0 -142
- package/src/_globals.ts +0 -92
- package/src/_idle.ts +0 -10
- package/src/_metadata.ts +0 -407
- package/src/_process-protection.ts +0 -71
- package/src/_server.ts +0 -109
- package/src/_services.ts +0 -379
- package/src/_standalone.ts +0 -710
- package/src/_tokens.ts +0 -114
- package/src/_util.ts +0 -62
- package/src/_validation.ts +0 -119
- package/src/_waituntil.ts +0 -188
- package/src/agent.ts +0 -2739
- package/src/app.ts +0 -769
- package/src/bootstrap.ts +0 -321
- package/src/bun-s3-patch.ts +0 -224
- package/src/cors.ts +0 -137
- package/src/dev-patches/aisdk.ts +0 -169
- package/src/dev-patches/gateway.ts +0 -68
- package/src/dev-patches/index.ts +0 -37
- package/src/dev-patches/otel-llm.ts +0 -405
- package/src/devmode.ts +0 -171
- package/src/eval.ts +0 -109
- package/src/globals.d.ts +0 -28
- package/src/handlers/_route-meta.ts +0 -33
- package/src/handlers/cron.ts +0 -141
- package/src/handlers/index.ts +0 -18
- package/src/handlers/sse.ts +0 -358
- package/src/handlers/stream.ts +0 -121
- package/src/handlers/webrtc.ts +0 -125
- package/src/handlers/websocket.ts +0 -203
- package/src/logger/console.ts +0 -323
- package/src/logger/index.ts +0 -2
- package/src/logger/internal.ts +0 -165
- package/src/logger/logger.ts +0 -44
- package/src/logger/user.ts +0 -15
- package/src/logger/util.ts +0 -80
- package/src/middleware.ts +0 -1095
- package/src/otel/config.ts +0 -47
- package/src/otel/console.ts +0 -91
- package/src/otel/exporters/README.md +0 -217
- package/src/otel/exporters/index.ts +0 -3
- package/src/otel/exporters/jsonl-log-exporter.ts +0 -113
- package/src/otel/exporters/jsonl-metric-exporter.ts +0 -120
- package/src/otel/exporters/jsonl-trace-exporter.ts +0 -121
- package/src/otel/fetch.ts +0 -105
- package/src/otel/http.ts +0 -53
- package/src/otel/logger.ts +0 -293
- package/src/otel/otel.ts +0 -354
- package/src/otel/tracestate.ts +0 -108
- package/src/router.ts +0 -75
- package/src/services/evalrun/composite.ts +0 -34
- package/src/services/evalrun/http.ts +0 -167
- package/src/services/evalrun/index.ts +0 -4
- package/src/services/evalrun/json.ts +0 -46
- package/src/services/evalrun/local.ts +0 -28
- package/src/services/local/README.md +0 -1576
- package/src/services/local/_db.ts +0 -353
- package/src/services/local/_router.ts +0 -40
- package/src/services/local/_util.ts +0 -55
- package/src/services/local/email.ts +0 -91
- package/src/services/local/index.ts +0 -9
- package/src/services/local/keyvalue.ts +0 -174
- package/src/services/local/queue.ts +0 -145
- package/src/services/local/stream.ts +0 -358
- package/src/services/local/task.ts +0 -1711
- package/src/services/local/vector.ts +0 -438
- package/src/services/sandbox/http.ts +0 -522
- package/src/services/sandbox/index.ts +0 -1
- package/src/services/session/composite.ts +0 -33
- package/src/services/session/http.ts +0 -167
- package/src/services/session/index.ts +0 -4
- package/src/services/session/json.ts +0 -42
- package/src/services/session/local.ts +0 -33
- package/src/services/thread/local.ts +0 -199
- package/src/session.ts +0 -1960
- package/src/signature.ts +0 -82
- package/src/validator.ts +0 -283
- package/src/version-check.ts +0 -184
- package/src/web.ts +0 -76
- package/src/webrtc-signaling.ts +0 -288
- package/src/workbench.ts +0 -725
package/src/workbench.ts
DELETED
|
@@ -1,725 +0,0 @@
|
|
|
1
|
-
import type { Context, Handler, MiddlewareHandler } from 'hono';
|
|
2
|
-
import { toJSONSchema } from '@agentuity/server';
|
|
3
|
-
import type { JSONSchema } from '@agentuity/schema';
|
|
4
|
-
import { getAgents, createAgentMiddleware } from './agent';
|
|
5
|
-
import { createRouter } from './router';
|
|
6
|
-
import { websocket, type WebSocketConnection } from './handlers/websocket';
|
|
7
|
-
import { privateContext } from './_server';
|
|
8
|
-
import { getThreadProvider } from './_services';
|
|
9
|
-
import {
|
|
10
|
-
loadBuildMetadata,
|
|
11
|
-
getAgentMetadataByAgentId,
|
|
12
|
-
hasMetadata,
|
|
13
|
-
ensureAgentsImported,
|
|
14
|
-
} from './_metadata';
|
|
15
|
-
import { TOKENS_HEADER, DURATION_HEADER } from './_tokens';
|
|
16
|
-
import { verifySignature } from './signature';
|
|
17
|
-
import { isProduction } from './_config';
|
|
18
|
-
import { createCorsMiddleware } from './middleware';
|
|
19
|
-
|
|
20
|
-
/**
|
|
21
|
-
* Trusted Agentuity domain suffixes for workbench CORS.
|
|
22
|
-
* Any origin matching https://*.{suffix} is allowed.
|
|
23
|
-
* In development, any origin is allowed.
|
|
24
|
-
*/
|
|
25
|
-
const TRUSTED_WORKBENCH_DOMAIN_SUFFIXES = ['.agentuity.com', '.agentuity.dev', '.agentuity.io'];
|
|
26
|
-
|
|
27
|
-
/**
|
|
28
|
-
* Check if an origin is a trusted Agentuity app origin.
|
|
29
|
-
* Matches any HTTPS subdomain of the trusted domain suffixes.
|
|
30
|
-
*/
|
|
31
|
-
function isTrustedWorkbenchOrigin(origin: string): boolean {
|
|
32
|
-
try {
|
|
33
|
-
const url = new URL(origin);
|
|
34
|
-
if (url.protocol !== 'https:') return false;
|
|
35
|
-
return TRUSTED_WORKBENCH_DOMAIN_SUFFIXES.some((suffix) => url.hostname.endsWith(suffix));
|
|
36
|
-
} catch {
|
|
37
|
-
return false;
|
|
38
|
-
}
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
/**
|
|
42
|
-
* Middleware that verifies workbench request signatures in production.
|
|
43
|
-
* In development mode, all requests are allowed.
|
|
44
|
-
* Supports both header-based auth (for HTTP) and query param auth (for WebSocket).
|
|
45
|
-
*/
|
|
46
|
-
const createWorkbenchAuthMiddleware = (): MiddlewareHandler => {
|
|
47
|
-
return async (c, next) => {
|
|
48
|
-
// Allow CORS preflight requests through (they don't have auth headers)
|
|
49
|
-
if (c.req.method === 'OPTIONS') {
|
|
50
|
-
return next();
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
// Skip auth in dev mode
|
|
54
|
-
if (!isProduction()) {
|
|
55
|
-
return next();
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
// Check signature from headers or query params (for WebSocket)
|
|
59
|
-
const signature = c.req.header('X-Agentuity-Workbench-Signature') || c.req.query('signature');
|
|
60
|
-
const timestamp = c.req.header('X-Agentuity-Workbench-Timestamp') || c.req.query('timestamp');
|
|
61
|
-
|
|
62
|
-
// For non-POST requests, body is empty
|
|
63
|
-
let body = '';
|
|
64
|
-
if (c.req.method === 'POST') {
|
|
65
|
-
const clonedReq = c.req.raw.clone();
|
|
66
|
-
body = await clonedReq.text();
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
const isValid = await verifySignature(signature, timestamp, body);
|
|
70
|
-
if (!isValid) {
|
|
71
|
-
return c.json({ error: 'Unauthorized' }, 401);
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
return next();
|
|
75
|
-
};
|
|
76
|
-
};
|
|
77
|
-
|
|
78
|
-
/**
|
|
79
|
-
* Middleware that captures execution metadata (tokens, duration, sessionId) after the handler completes
|
|
80
|
-
* and saves it to thread state. Applied only to the /execute route.
|
|
81
|
-
*/
|
|
82
|
-
const createWorkbenchExecutionMetadataMiddleware = (): MiddlewareHandler => {
|
|
83
|
-
return async (ctx, next) => {
|
|
84
|
-
const started = performance.now();
|
|
85
|
-
|
|
86
|
-
await next();
|
|
87
|
-
|
|
88
|
-
// After handler completes, tokens and duration headers are available
|
|
89
|
-
const thread = ctx.var.thread;
|
|
90
|
-
if (!thread) {
|
|
91
|
-
return;
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
// Get execution context set by the handler
|
|
95
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
96
|
-
const executionCtx = (ctx as any).var.workbenchExecution as
|
|
97
|
-
| { agentId: string; input: unknown; result: unknown }
|
|
98
|
-
| undefined;
|
|
99
|
-
if (!executionCtx) {
|
|
100
|
-
return;
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
const { agentId, input, result } = executionCtx;
|
|
104
|
-
const agentMessagesKey = `messages_${agentId}`;
|
|
105
|
-
const maxMessages = 50;
|
|
106
|
-
|
|
107
|
-
// Read tokens and duration from response headers
|
|
108
|
-
const tokens = ctx.res.headers.get(TOKENS_HEADER) ?? undefined;
|
|
109
|
-
const duration =
|
|
110
|
-
ctx.res.headers.get(DURATION_HEADER) ??
|
|
111
|
-
`${((performance.now() - started) / 1000).toFixed(1)}s`;
|
|
112
|
-
const sessionId = ctx.var.sessionId;
|
|
113
|
-
|
|
114
|
-
// Store input with metadata
|
|
115
|
-
await thread.state.push(
|
|
116
|
-
agentMessagesKey,
|
|
117
|
-
{
|
|
118
|
-
type: 'input',
|
|
119
|
-
data: input,
|
|
120
|
-
sessionId,
|
|
121
|
-
timestamp: Date.now(),
|
|
122
|
-
},
|
|
123
|
-
maxMessages
|
|
124
|
-
);
|
|
125
|
-
|
|
126
|
-
// Store output with metadata (tokens, duration)
|
|
127
|
-
if (result !== undefined && result !== null) {
|
|
128
|
-
await thread.state.push(
|
|
129
|
-
agentMessagesKey,
|
|
130
|
-
{
|
|
131
|
-
type: 'output',
|
|
132
|
-
data: result,
|
|
133
|
-
sessionId,
|
|
134
|
-
tokens,
|
|
135
|
-
duration,
|
|
136
|
-
timestamp: Date.now(),
|
|
137
|
-
},
|
|
138
|
-
maxMessages
|
|
139
|
-
);
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
// Save thread state
|
|
143
|
-
try {
|
|
144
|
-
const threadProvider = getThreadProvider();
|
|
145
|
-
await threadProvider.save(thread);
|
|
146
|
-
} catch {
|
|
147
|
-
ctx.var.logger?.warn('Failed to save thread state');
|
|
148
|
-
}
|
|
149
|
-
};
|
|
150
|
-
};
|
|
151
|
-
|
|
152
|
-
export const createWorkbenchExecutionRoute = (): Handler => {
|
|
153
|
-
return async (ctx: Context) => {
|
|
154
|
-
// Content-type validation
|
|
155
|
-
const contentType = ctx.req.header('Content-Type');
|
|
156
|
-
if (!contentType || !contentType.includes('application/json')) {
|
|
157
|
-
return ctx.json({ error: 'Content-Type must be application/json' }, { status: 400 });
|
|
158
|
-
}
|
|
159
|
-
|
|
160
|
-
try {
|
|
161
|
-
let agentId: string;
|
|
162
|
-
let input: unknown;
|
|
163
|
-
try {
|
|
164
|
-
const requestData = await ctx.req.json();
|
|
165
|
-
agentId = requestData.agentId;
|
|
166
|
-
input = requestData.input;
|
|
167
|
-
} catch (_error) {
|
|
168
|
-
return ctx.json({ error: 'Invalid JSON in request body' }, { status: 400 });
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
// Read metadata to find agent name by agentId
|
|
172
|
-
const agentMeta = getAgentMetadataByAgentId(agentId);
|
|
173
|
-
if (!agentMeta) {
|
|
174
|
-
if (!hasMetadata()) {
|
|
175
|
-
return ctx.json({ error: 'Metadata file not found' }, { status: 500 });
|
|
176
|
-
}
|
|
177
|
-
return ctx.text('Agent not found', { status: 404 });
|
|
178
|
-
}
|
|
179
|
-
|
|
180
|
-
// Get runtime agent by name
|
|
181
|
-
const allAgents = getAgents();
|
|
182
|
-
const agentName = agentMeta.name;
|
|
183
|
-
const agentObj = allAgents.get(agentName);
|
|
184
|
-
|
|
185
|
-
if (!agentObj || !agentName) {
|
|
186
|
-
return ctx.text('Agent not found', { status: 404 });
|
|
187
|
-
}
|
|
188
|
-
|
|
189
|
-
// Track agent ID for telemetry (otelMiddleware sets up agentIds)
|
|
190
|
-
const _ctx = privateContext(ctx);
|
|
191
|
-
if (agentObj.metadata?.id) {
|
|
192
|
-
_ctx.var.agentIds.add(agentObj.metadata.id);
|
|
193
|
-
}
|
|
194
|
-
if (agentObj.metadata?.agentId) {
|
|
195
|
-
_ctx.var.agentIds.add(agentObj.metadata.agentId);
|
|
196
|
-
}
|
|
197
|
-
|
|
198
|
-
// Execute the agent handler directly
|
|
199
|
-
// The agentMiddleware has already set up the AsyncLocalStorage context
|
|
200
|
-
// so the handler can access it via getAgentContext()
|
|
201
|
-
let result: unknown;
|
|
202
|
-
if (agentObj.inputSchema) {
|
|
203
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
204
|
-
result = await (agentObj as any).handler(input);
|
|
205
|
-
} else {
|
|
206
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
207
|
-
result = await (agentObj as any).handler();
|
|
208
|
-
}
|
|
209
|
-
|
|
210
|
-
// Store execution context for the metadata middleware to save with tokens/duration
|
|
211
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
212
|
-
(ctx as any).set('workbenchExecution', { agentId, input, result });
|
|
213
|
-
|
|
214
|
-
return ctx.json({ success: true, data: result ?? null });
|
|
215
|
-
} catch (error) {
|
|
216
|
-
const err = error instanceof Error ? error : new Error(String(error));
|
|
217
|
-
// Return 200 with wrapped error so UI can display it properly
|
|
218
|
-
return ctx.json({
|
|
219
|
-
success: false,
|
|
220
|
-
error: {
|
|
221
|
-
message: err.message,
|
|
222
|
-
stack: err.stack,
|
|
223
|
-
code: 'code' in err && typeof err.code === 'string' ? err.code : 'EXECUTION_ERROR',
|
|
224
|
-
cause: err.cause,
|
|
225
|
-
},
|
|
226
|
-
});
|
|
227
|
-
}
|
|
228
|
-
};
|
|
229
|
-
};
|
|
230
|
-
|
|
231
|
-
export const createWorkbenchClearStateRoute = (): Handler => {
|
|
232
|
-
return async (ctx: Context) => {
|
|
233
|
-
const agentId = ctx.req.query('agentId');
|
|
234
|
-
|
|
235
|
-
if (!agentId) {
|
|
236
|
-
return ctx.json({ error: 'agentId query parameter is required' }, { status: 400 });
|
|
237
|
-
}
|
|
238
|
-
|
|
239
|
-
if (!ctx.var.thread) {
|
|
240
|
-
return ctx.json({ error: 'Thread not available' }, { status: 404 });
|
|
241
|
-
}
|
|
242
|
-
|
|
243
|
-
// Clear state associated with this specific agent:
|
|
244
|
-
// 1. messages_${agentId} - workbench message history
|
|
245
|
-
// 2. Any keys starting with ${agentId}_ - agent-specific state
|
|
246
|
-
const allKeys = await ctx.var.thread.state.keys();
|
|
247
|
-
const agentPrefix = `${agentId}_`;
|
|
248
|
-
const messagesKey = `messages_${agentId}`;
|
|
249
|
-
|
|
250
|
-
for (const key of allKeys) {
|
|
251
|
-
if (key === messagesKey || key.startsWith(agentPrefix)) {
|
|
252
|
-
await ctx.var.thread.state.delete(key);
|
|
253
|
-
}
|
|
254
|
-
}
|
|
255
|
-
|
|
256
|
-
// Save the thread to persist the cleared state
|
|
257
|
-
try {
|
|
258
|
-
const threadProvider = getThreadProvider();
|
|
259
|
-
await threadProvider.save(ctx.var.thread);
|
|
260
|
-
} catch {
|
|
261
|
-
return ctx.json({ error: 'Failed to save thread state' }, { status: 500 });
|
|
262
|
-
}
|
|
263
|
-
|
|
264
|
-
return ctx.json({ success: true, message: `State cleared for agent ${agentId}` });
|
|
265
|
-
};
|
|
266
|
-
};
|
|
267
|
-
|
|
268
|
-
export const createWorkbenchStateRoute = (): Handler => {
|
|
269
|
-
return async (ctx: Context) => {
|
|
270
|
-
const agentId = ctx.req.query('agentId');
|
|
271
|
-
if (!agentId) {
|
|
272
|
-
return ctx.json({ error: 'agentId query parameter is required' }, { status: 400 });
|
|
273
|
-
}
|
|
274
|
-
|
|
275
|
-
if (!ctx.var.thread) {
|
|
276
|
-
return ctx.json({ error: 'Thread not available' }, { status: 404 });
|
|
277
|
-
}
|
|
278
|
-
|
|
279
|
-
const agentMessagesKey = `messages_${agentId}`;
|
|
280
|
-
const messages = await ctx.var.thread.state.get(agentMessagesKey);
|
|
281
|
-
|
|
282
|
-
return ctx.json({
|
|
283
|
-
threadId: ctx.var.thread.id,
|
|
284
|
-
agentId,
|
|
285
|
-
messages: Array.isArray(messages) ? messages : [],
|
|
286
|
-
});
|
|
287
|
-
};
|
|
288
|
-
};
|
|
289
|
-
|
|
290
|
-
/**
|
|
291
|
-
* Creates a workbench router with proper agent middleware for execution routes
|
|
292
|
-
*/
|
|
293
|
-
export const createWorkbenchRouter = () => {
|
|
294
|
-
const router = createRouter();
|
|
295
|
-
|
|
296
|
-
// Apply CORS middleware first so that even error responses get CORS headers
|
|
297
|
-
// In production, restrict origins to known Agentuity app domains + same-origin
|
|
298
|
-
// In development, allow any origin for local testing flexibility
|
|
299
|
-
router.use(
|
|
300
|
-
'/_agentuity/workbench/*',
|
|
301
|
-
createCorsMiddleware({
|
|
302
|
-
origin: (origin: string, c) => {
|
|
303
|
-
// In dev mode, allow any origin for local testing flexibility
|
|
304
|
-
if (!isProduction()) {
|
|
305
|
-
return origin;
|
|
306
|
-
}
|
|
307
|
-
// In production, allow any *.agentuity.{com,dev,io} origin
|
|
308
|
-
if (isTrustedWorkbenchOrigin(origin)) {
|
|
309
|
-
return origin;
|
|
310
|
-
}
|
|
311
|
-
// Allow same-origin requests (agent calling its own workbench)
|
|
312
|
-
try {
|
|
313
|
-
const requestOrigin = new URL(c.req.url).origin;
|
|
314
|
-
if (origin === requestOrigin) {
|
|
315
|
-
return origin;
|
|
316
|
-
}
|
|
317
|
-
} catch {
|
|
318
|
-
// Invalid URL, reject
|
|
319
|
-
}
|
|
320
|
-
// Reject unknown origins — no Access-Control-Allow-Origin header
|
|
321
|
-
return undefined;
|
|
322
|
-
},
|
|
323
|
-
allowHeaders: [
|
|
324
|
-
'Content-Type',
|
|
325
|
-
'Authorization',
|
|
326
|
-
'Accept',
|
|
327
|
-
'Origin',
|
|
328
|
-
'X-Requested-With',
|
|
329
|
-
'X-Agentuity-Workbench-Signature',
|
|
330
|
-
'X-Agentuity-Workbench-Timestamp',
|
|
331
|
-
'x-thread-id',
|
|
332
|
-
],
|
|
333
|
-
exposeHeaders: [
|
|
334
|
-
'x-thread-id',
|
|
335
|
-
'x-session-id',
|
|
336
|
-
'x-agentuity-tokens',
|
|
337
|
-
'x-agentuity-duration',
|
|
338
|
-
],
|
|
339
|
-
})
|
|
340
|
-
);
|
|
341
|
-
|
|
342
|
-
// Apply auth middleware (signature verification in production)
|
|
343
|
-
router.use('/_agentuity/workbench/*', createWorkbenchAuthMiddleware());
|
|
344
|
-
|
|
345
|
-
// Apply agent middleware to ensure proper context is available
|
|
346
|
-
router.use('/_agentuity/workbench/*', createAgentMiddleware(''));
|
|
347
|
-
|
|
348
|
-
// Add workbench routes
|
|
349
|
-
router.get('/_agentuity/workbench/ws', websocket(createWorkbenchWebsocketHandler()));
|
|
350
|
-
router.get('/_agentuity/workbench/metadata.json', createWorkbenchMetadataRoute());
|
|
351
|
-
router.get('/_agentuity/workbench/sample', createWorkbenchSampleRoute());
|
|
352
|
-
router.get('/_agentuity/workbench/state', createWorkbenchStateRoute());
|
|
353
|
-
router.delete('/_agentuity/workbench/state', createWorkbenchClearStateRoute());
|
|
354
|
-
router.post(
|
|
355
|
-
'/_agentuity/workbench/execute',
|
|
356
|
-
createWorkbenchExecutionMetadataMiddleware(),
|
|
357
|
-
createWorkbenchExecutionRoute()
|
|
358
|
-
);
|
|
359
|
-
return router;
|
|
360
|
-
};
|
|
361
|
-
|
|
362
|
-
export const createWorkbenchSampleRoute = (): Handler => {
|
|
363
|
-
return async (ctx: Context) => {
|
|
364
|
-
try {
|
|
365
|
-
const agentId = ctx.req.query('agentId');
|
|
366
|
-
if (!agentId) {
|
|
367
|
-
return ctx.json({ error: 'Missing agentId query parameter' }, { status: 400 });
|
|
368
|
-
}
|
|
369
|
-
|
|
370
|
-
// Read metadata to find agent name by agentId
|
|
371
|
-
const agentMeta = getAgentMetadataByAgentId(agentId);
|
|
372
|
-
if (!agentMeta) {
|
|
373
|
-
if (!hasMetadata()) {
|
|
374
|
-
return ctx.json({ error: 'Metadata file not found' }, { status: 500 });
|
|
375
|
-
}
|
|
376
|
-
return ctx.text('Agent not found', { status: 404 });
|
|
377
|
-
}
|
|
378
|
-
|
|
379
|
-
// Get runtime agent by name
|
|
380
|
-
const allAgents = getAgents();
|
|
381
|
-
const agentObj = allAgents.get(agentMeta.name);
|
|
382
|
-
|
|
383
|
-
if (!agentObj) {
|
|
384
|
-
return ctx.text('Agent not found', { status: 404 });
|
|
385
|
-
}
|
|
386
|
-
|
|
387
|
-
// Check if agent has input schema
|
|
388
|
-
if (!agentObj.inputSchema) {
|
|
389
|
-
return ctx.json({ error: 'Agent has no input schema' }, { status: 400 });
|
|
390
|
-
}
|
|
391
|
-
|
|
392
|
-
// Convert schema to JSON Schema
|
|
393
|
-
const jsonSchema = toJSONSchema(agentObj.inputSchema);
|
|
394
|
-
|
|
395
|
-
// Get Agentuity SDK key and gateway URL
|
|
396
|
-
const sdkKey = process.env.AGENTUITY_SDK_KEY;
|
|
397
|
-
const gatewayUrl =
|
|
398
|
-
process.env.AGENTUITY_AIGATEWAY_URL ||
|
|
399
|
-
process.env.AGENTUITY_TRANSPORT_URL ||
|
|
400
|
-
(sdkKey ? 'https://catalyst.agentuity.cloud' : '');
|
|
401
|
-
|
|
402
|
-
if (!sdkKey || !gatewayUrl) {
|
|
403
|
-
return ctx.json(
|
|
404
|
-
{
|
|
405
|
-
error: 'AGENTUITY_SDK_KEY and gateway URL must be configured',
|
|
406
|
-
message:
|
|
407
|
-
'Set AGENTUITY_SDK_KEY and either AGENTUITY_AIGATEWAY_URL or AGENTUITY_TRANSPORT_URL',
|
|
408
|
-
},
|
|
409
|
-
{ status: 500 }
|
|
410
|
-
);
|
|
411
|
-
}
|
|
412
|
-
|
|
413
|
-
// Generate sample using Groq via Agentuity Gateway
|
|
414
|
-
const prompt = `Generate a realistic sample data object that matches this JSON schema. Return only valid JSON, no markdown code blocks or explanations.
|
|
415
|
-
|
|
416
|
-
JSON Schema:
|
|
417
|
-
${JSON.stringify(jsonSchema, null, 2)}
|
|
418
|
-
|
|
419
|
-
Return a JSON object that matches this schema with realistic values.`;
|
|
420
|
-
|
|
421
|
-
const gatewayEndpoint = `${gatewayUrl}/gateway/groq/openai/v1/chat/completions`;
|
|
422
|
-
const groqResponse = await fetch(gatewayEndpoint, {
|
|
423
|
-
method: 'POST',
|
|
424
|
-
headers: {
|
|
425
|
-
Authorization: `Bearer ${sdkKey}`,
|
|
426
|
-
'Content-Type': 'application/json',
|
|
427
|
-
},
|
|
428
|
-
body: JSON.stringify({
|
|
429
|
-
model: 'llama-3.3-70b-versatile',
|
|
430
|
-
messages: [
|
|
431
|
-
{
|
|
432
|
-
role: 'user',
|
|
433
|
-
content: prompt,
|
|
434
|
-
},
|
|
435
|
-
],
|
|
436
|
-
}),
|
|
437
|
-
});
|
|
438
|
-
|
|
439
|
-
if (!groqResponse.ok) {
|
|
440
|
-
const errorText = await groqResponse.text();
|
|
441
|
-
return ctx.json(
|
|
442
|
-
{
|
|
443
|
-
error: 'Groq API request failed',
|
|
444
|
-
message: `Status ${groqResponse.status}: ${errorText}`,
|
|
445
|
-
},
|
|
446
|
-
{ status: 500 }
|
|
447
|
-
);
|
|
448
|
-
}
|
|
449
|
-
|
|
450
|
-
const groqData = (await groqResponse.json()) as {
|
|
451
|
-
choices?: Array<{ message?: { content?: string } }>;
|
|
452
|
-
};
|
|
453
|
-
const text = groqData.choices?.[0]?.message?.content;
|
|
454
|
-
if (!text) {
|
|
455
|
-
return ctx.json(
|
|
456
|
-
{ error: 'Invalid response from Groq API', response: groqData },
|
|
457
|
-
{ status: 500 }
|
|
458
|
-
);
|
|
459
|
-
}
|
|
460
|
-
|
|
461
|
-
// Parse the JSON response
|
|
462
|
-
let sample: unknown;
|
|
463
|
-
try {
|
|
464
|
-
// Remove markdown code blocks if present
|
|
465
|
-
const cleanedText = text
|
|
466
|
-
.trim()
|
|
467
|
-
.replace(/^```json\s*|\s*```$/g, '')
|
|
468
|
-
.replace(/^```\s*|\s*```$/g, '');
|
|
469
|
-
sample = JSON.parse(cleanedText);
|
|
470
|
-
} catch (parseError) {
|
|
471
|
-
return ctx.json(
|
|
472
|
-
{
|
|
473
|
-
error: 'Failed to parse generated JSON',
|
|
474
|
-
message: parseError instanceof Error ? parseError.message : String(parseError),
|
|
475
|
-
generatedText: text,
|
|
476
|
-
},
|
|
477
|
-
{ status: 500 }
|
|
478
|
-
);
|
|
479
|
-
}
|
|
480
|
-
|
|
481
|
-
return ctx.json(sample);
|
|
482
|
-
} catch (error) {
|
|
483
|
-
return ctx.json(
|
|
484
|
-
{
|
|
485
|
-
error: 'Internal server error',
|
|
486
|
-
message: error instanceof Error ? error.message : String(error),
|
|
487
|
-
},
|
|
488
|
-
{ status: 500 }
|
|
489
|
-
);
|
|
490
|
-
}
|
|
491
|
-
};
|
|
492
|
-
};
|
|
493
|
-
|
|
494
|
-
export const createWorkbenchMetadataRoute = (): Handler => {
|
|
495
|
-
return async (ctx) => {
|
|
496
|
-
// Read metadata from agentuity.metadata.json file
|
|
497
|
-
const metadata = loadBuildMetadata();
|
|
498
|
-
if (!metadata) {
|
|
499
|
-
return ctx.json(
|
|
500
|
-
{ error: 'Metadata file not found. Run build to generate metadata.' },
|
|
501
|
-
{ status: 500 }
|
|
502
|
-
);
|
|
503
|
-
}
|
|
504
|
-
|
|
505
|
-
try {
|
|
506
|
-
// Ensure all agents are imported so their schemas are available
|
|
507
|
-
await ensureAgentsImported();
|
|
508
|
-
|
|
509
|
-
// Get runtime agents for JSON schema generation
|
|
510
|
-
const agents = getAgents();
|
|
511
|
-
const agentsByName = new Map();
|
|
512
|
-
for (const [name, agent] of agents) {
|
|
513
|
-
agentsByName.set(name, agent);
|
|
514
|
-
}
|
|
515
|
-
|
|
516
|
-
// Transform metadata structure to workbench format
|
|
517
|
-
const schemas: { agents: Record<string, unknown> } = { agents: {} };
|
|
518
|
-
|
|
519
|
-
for (const agent of metadata.agents || []) {
|
|
520
|
-
// Try to find runtime agent by name to get JSON schemas
|
|
521
|
-
const runtimeAgent = agentsByName.get(agent.name);
|
|
522
|
-
|
|
523
|
-
const inputJsonSchema = runtimeAgent?.inputSchema
|
|
524
|
-
? toJSONSchema(runtimeAgent.inputSchema)
|
|
525
|
-
: undefined;
|
|
526
|
-
const outputJsonSchema = runtimeAgent?.outputSchema
|
|
527
|
-
? toJSONSchema(runtimeAgent.outputSchema)
|
|
528
|
-
: undefined;
|
|
529
|
-
|
|
530
|
-
schemas.agents[agent.id] = {
|
|
531
|
-
schema: {
|
|
532
|
-
input: inputJsonSchema
|
|
533
|
-
? {
|
|
534
|
-
code: jsonSchemaToTypeScript(inputJsonSchema),
|
|
535
|
-
json: inputJsonSchema,
|
|
536
|
-
}
|
|
537
|
-
: agent.schema?.input
|
|
538
|
-
? { code: agent.schema.input, json: undefined }
|
|
539
|
-
: undefined,
|
|
540
|
-
output: outputJsonSchema
|
|
541
|
-
? {
|
|
542
|
-
code: jsonSchemaToTypeScript(outputJsonSchema),
|
|
543
|
-
json: outputJsonSchema,
|
|
544
|
-
}
|
|
545
|
-
: agent.schema?.output
|
|
546
|
-
? { code: agent.schema.output, json: undefined }
|
|
547
|
-
: undefined,
|
|
548
|
-
},
|
|
549
|
-
metadata: {
|
|
550
|
-
id: agent.id,
|
|
551
|
-
agentId: agent.agentId,
|
|
552
|
-
name: agent.name,
|
|
553
|
-
description: agent.description,
|
|
554
|
-
filename: agent.filename,
|
|
555
|
-
version: agent.version,
|
|
556
|
-
},
|
|
557
|
-
};
|
|
558
|
-
}
|
|
559
|
-
|
|
560
|
-
return ctx.json(schemas);
|
|
561
|
-
} catch (error) {
|
|
562
|
-
return ctx.json(
|
|
563
|
-
{
|
|
564
|
-
error: 'Failed to read metadata file',
|
|
565
|
-
message: error instanceof Error ? error.message : String(error),
|
|
566
|
-
},
|
|
567
|
-
{ status: 500 }
|
|
568
|
-
);
|
|
569
|
-
}
|
|
570
|
-
};
|
|
571
|
-
};
|
|
572
|
-
|
|
573
|
-
// Store WebSocket connections to notify them on app restart
|
|
574
|
-
const workbenchWebSockets = new Set<WebSocketConnection>();
|
|
575
|
-
|
|
576
|
-
export const createWorkbenchWebsocketHandler = () => {
|
|
577
|
-
return (_c: Context, ws: WebSocketConnection) => {
|
|
578
|
-
ws.onOpen(() => {
|
|
579
|
-
workbenchWebSockets.add(ws);
|
|
580
|
-
ws.send('alive');
|
|
581
|
-
});
|
|
582
|
-
|
|
583
|
-
ws.onMessage((event) => {
|
|
584
|
-
const message = (event as MessageEvent).data;
|
|
585
|
-
|
|
586
|
-
// If a client sends a message (CLI), broadcast to all other clients
|
|
587
|
-
if (message === 'restarting' || message === 'alive') {
|
|
588
|
-
// Broadcast the message to all other clients (excluding this CLI connection)
|
|
589
|
-
for (const clientWs of workbenchWebSockets) {
|
|
590
|
-
if (clientWs !== ws) {
|
|
591
|
-
try {
|
|
592
|
-
clientWs.send(message);
|
|
593
|
-
} catch (_error) {
|
|
594
|
-
workbenchWebSockets.delete(clientWs);
|
|
595
|
-
}
|
|
596
|
-
}
|
|
597
|
-
}
|
|
598
|
-
}
|
|
599
|
-
});
|
|
600
|
-
|
|
601
|
-
ws.onClose(() => {
|
|
602
|
-
workbenchWebSockets.delete(ws);
|
|
603
|
-
});
|
|
604
|
-
};
|
|
605
|
-
};
|
|
606
|
-
|
|
607
|
-
/**
|
|
608
|
-
* @deprecated Use createWorkbenchWebsocketHandler instead
|
|
609
|
-
*/
|
|
610
|
-
export const createWorkbenchWebsocketRoute = createWorkbenchWebsocketHandler;
|
|
611
|
-
|
|
612
|
-
/**
|
|
613
|
-
* Convert a JSON Schema object to a TypeScript type string for display.
|
|
614
|
-
* Produces clean, readable type notation for the workbench schema panel.
|
|
615
|
-
*
|
|
616
|
-
* @example
|
|
617
|
-
* ```typescript
|
|
618
|
-
* jsonSchemaToTypeScript({
|
|
619
|
-
* type: 'object',
|
|
620
|
-
* properties: { name: { type: 'string' }, age: { type: 'number' } },
|
|
621
|
-
* required: ['name', 'age'],
|
|
622
|
-
* });
|
|
623
|
-
* // "{\n name: string;\n age: number;\n}"
|
|
624
|
-
* ```
|
|
625
|
-
*/
|
|
626
|
-
/** Escape a string for use inside a double-quoted TypeScript string literal. */
|
|
627
|
-
function escapeString(s: string): string {
|
|
628
|
-
return s.replace(/\\/g, '\\\\').replace(/"/g, '\\"').replace(/\n/g, '\\n').replace(/\r/g, '\\r');
|
|
629
|
-
}
|
|
630
|
-
|
|
631
|
-
/** Check if a property key is a valid unquoted TypeScript identifier. */
|
|
632
|
-
function isValidIdentifier(key: string): boolean {
|
|
633
|
-
return /^[a-zA-Z_$][a-zA-Z0-9_$]*$/.test(key);
|
|
634
|
-
}
|
|
635
|
-
|
|
636
|
-
function jsonSchemaToTypeScript(schema: JSONSchema, indent = 0): string {
|
|
637
|
-
const pad = ' '.repeat(indent);
|
|
638
|
-
const inner = ' '.repeat(indent + 1);
|
|
639
|
-
|
|
640
|
-
// Handle const (literal type)
|
|
641
|
-
if (schema.const !== undefined) {
|
|
642
|
-
return typeof schema.const === 'string'
|
|
643
|
-
? `"${escapeString(schema.const)}"`
|
|
644
|
-
: String(schema.const);
|
|
645
|
-
}
|
|
646
|
-
|
|
647
|
-
// Handle enum (union of literals)
|
|
648
|
-
if (schema.enum) {
|
|
649
|
-
return schema.enum
|
|
650
|
-
.map((v) => (typeof v === 'string' ? `"${escapeString(String(v))}"` : String(v)))
|
|
651
|
-
.join(' | ');
|
|
652
|
-
}
|
|
653
|
-
|
|
654
|
-
// Handle anyOf / oneOf (union types)
|
|
655
|
-
const unionSchemas = schema.anyOf ?? schema.oneOf;
|
|
656
|
-
if (unionSchemas) {
|
|
657
|
-
// Nullable pattern: anyOf with one type and one null
|
|
658
|
-
if (unionSchemas.length === 2) {
|
|
659
|
-
const nullIdx = unionSchemas.findIndex((s) => s.type === 'null');
|
|
660
|
-
if (nullIdx !== -1) {
|
|
661
|
-
const other = unionSchemas[nullIdx === 0 ? 1 : 0];
|
|
662
|
-
if (other) {
|
|
663
|
-
return `${jsonSchemaToTypeScript(other, indent)} | null`;
|
|
664
|
-
}
|
|
665
|
-
}
|
|
666
|
-
}
|
|
667
|
-
return unionSchemas.map((s) => jsonSchemaToTypeScript(s, indent)).join(' | ');
|
|
668
|
-
}
|
|
669
|
-
|
|
670
|
-
// Handle allOf (intersection types)
|
|
671
|
-
if (schema.allOf) {
|
|
672
|
-
return schema.allOf.map((s) => jsonSchemaToTypeScript(s, indent)).join(' & ');
|
|
673
|
-
}
|
|
674
|
-
|
|
675
|
-
switch (schema.type) {
|
|
676
|
-
case 'string':
|
|
677
|
-
return 'string';
|
|
678
|
-
case 'number':
|
|
679
|
-
case 'integer':
|
|
680
|
-
return 'number';
|
|
681
|
-
case 'boolean':
|
|
682
|
-
return 'boolean';
|
|
683
|
-
case 'null':
|
|
684
|
-
return 'null';
|
|
685
|
-
|
|
686
|
-
case 'array': {
|
|
687
|
-
if (!schema.items) return 'unknown[]';
|
|
688
|
-
const itemType = jsonSchemaToTypeScript(schema.items, indent);
|
|
689
|
-
// Wrap union types in parens: (A | B)[]
|
|
690
|
-
return itemType.includes(' | ') ? `(${itemType})[]` : `${itemType}[]`;
|
|
691
|
-
}
|
|
692
|
-
|
|
693
|
-
case 'object': {
|
|
694
|
-
if (!schema.properties || Object.keys(schema.properties).length === 0) {
|
|
695
|
-
if (schema.additionalProperties && typeof schema.additionalProperties === 'object') {
|
|
696
|
-
return `Record<string, ${jsonSchemaToTypeScript(schema.additionalProperties, indent)}>`;
|
|
697
|
-
}
|
|
698
|
-
return 'Record<string, unknown>';
|
|
699
|
-
}
|
|
700
|
-
|
|
701
|
-
const required = new Set(schema.required ?? []);
|
|
702
|
-
const lines: string[] = ['{'];
|
|
703
|
-
|
|
704
|
-
for (const [key, propSchema] of Object.entries(schema.properties)) {
|
|
705
|
-
const optional = !required.has(key);
|
|
706
|
-
const propType = jsonSchemaToTypeScript(propSchema, indent + 1);
|
|
707
|
-
const desc = propSchema.description ? ` // ${propSchema.description}` : '';
|
|
708
|
-
const quotedKey = isValidIdentifier(key) ? key : `"${escapeString(key)}"`;
|
|
709
|
-
lines.push(`${inner}${quotedKey}${optional ? '?' : ''}: ${propType};${desc}`);
|
|
710
|
-
}
|
|
711
|
-
|
|
712
|
-
lines.push(`${pad}}`);
|
|
713
|
-
return lines.join('\n');
|
|
714
|
-
}
|
|
715
|
-
|
|
716
|
-
default:
|
|
717
|
-
if (schema.properties) {
|
|
718
|
-
return jsonSchemaToTypeScript({ ...schema, type: 'object' }, indent);
|
|
719
|
-
}
|
|
720
|
-
if (schema.items) {
|
|
721
|
-
return jsonSchemaToTypeScript({ ...schema, type: 'array' }, indent);
|
|
722
|
-
}
|
|
723
|
-
return 'unknown';
|
|
724
|
-
}
|
|
725
|
-
}
|