airlock-bot 0.0.1 → 0.2.2
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/LICENSE +21 -0
- package/README.md +337 -0
- package/airlock.service +27 -0
- package/dist/allowlist/engine.d.ts +9 -0
- package/dist/allowlist/engine.d.ts.map +1 -0
- package/dist/allowlist/engine.js +24 -0
- package/dist/allowlist/engine.js.map +1 -0
- package/dist/allowlist/pattern.d.ts +13 -0
- package/dist/allowlist/pattern.d.ts.map +1 -0
- package/dist/allowlist/pattern.js +33 -0
- package/dist/allowlist/pattern.js.map +1 -0
- package/dist/audit/api.d.ts +7 -0
- package/dist/audit/api.d.ts.map +1 -0
- package/dist/audit/api.js +31 -0
- package/dist/audit/api.js.map +1 -0
- package/dist/audit/db.d.ts +44 -0
- package/dist/audit/db.d.ts.map +1 -0
- package/dist/audit/db.js +121 -0
- package/dist/audit/db.js.map +1 -0
- package/dist/audit/logger.d.ts +25 -0
- package/dist/audit/logger.d.ts.map +1 -0
- package/dist/audit/logger.js +58 -0
- package/dist/audit/logger.js.map +1 -0
- package/dist/audit/redactor.d.ts +5 -0
- package/dist/audit/redactor.d.ts.map +1 -0
- package/dist/audit/redactor.js +27 -0
- package/dist/audit/redactor.js.map +1 -0
- package/dist/backend/cli/adapter.d.ts +23 -0
- package/dist/backend/cli/adapter.d.ts.map +1 -0
- package/dist/backend/cli/adapter.js +176 -0
- package/dist/backend/cli/adapter.js.map +1 -0
- package/dist/backend/cli/builder.d.ts +3 -0
- package/dist/backend/cli/builder.d.ts.map +1 -0
- package/dist/backend/cli/builder.js +52 -0
- package/dist/backend/cli/builder.js.map +1 -0
- package/dist/backend/cli/escaper.d.ts +2 -0
- package/dist/backend/cli/escaper.d.ts.map +1 -0
- package/dist/backend/cli/escaper.js +8 -0
- package/dist/backend/cli/escaper.js.map +1 -0
- package/dist/backend/exec-adapter.d.ts +13 -0
- package/dist/backend/exec-adapter.d.ts.map +1 -0
- package/dist/backend/exec-adapter.js +39 -0
- package/dist/backend/exec-adapter.js.map +1 -0
- package/dist/backend/factory.d.ts +9 -0
- package/dist/backend/factory.d.ts.map +1 -0
- package/dist/backend/factory.js +35 -0
- package/dist/backend/factory.js.map +1 -0
- package/dist/backend/http-adapter.d.ts +15 -0
- package/dist/backend/http-adapter.d.ts.map +1 -0
- package/dist/backend/http-adapter.js +39 -0
- package/dist/backend/http-adapter.js.map +1 -0
- package/dist/backend/mcp-adapter.d.ts +14 -0
- package/dist/backend/mcp-adapter.d.ts.map +1 -0
- package/dist/backend/mcp-adapter.js +38 -0
- package/dist/backend/mcp-adapter.js.map +1 -0
- package/dist/backend/openapi/adapter.d.ts +17 -0
- package/dist/backend/openapi/adapter.d.ts.map +1 -0
- package/dist/backend/openapi/adapter.js +144 -0
- package/dist/backend/openapi/adapter.js.map +1 -0
- package/dist/backend/openapi/parser.d.ts +21 -0
- package/dist/backend/openapi/parser.d.ts.map +1 -0
- package/dist/backend/openapi/parser.js +145 -0
- package/dist/backend/openapi/parser.js.map +1 -0
- package/dist/backend/types.d.ts +9 -0
- package/dist/backend/types.d.ts.map +1 -0
- package/dist/backend/types.js +2 -0
- package/dist/backend/types.js.map +1 -0
- package/dist/config/loader.d.ts +12 -0
- package/dist/config/loader.d.ts.map +1 -0
- package/dist/config/loader.js +178 -0
- package/dist/config/loader.js.map +1 -0
- package/dist/config/profiles.d.ts +12 -0
- package/dist/config/profiles.d.ts.map +1 -0
- package/dist/config/profiles.js +34 -0
- package/dist/config/profiles.js.map +1 -0
- package/dist/config/schema.d.ts +2034 -0
- package/dist/config/schema.d.ts.map +1 -0
- package/dist/config/schema.js +257 -0
- package/dist/config/schema.js.map +1 -0
- package/dist/config/watcher.d.ts +11 -0
- package/dist/config/watcher.d.ts.map +1 -0
- package/dist/config/watcher.js +39 -0
- package/dist/config/watcher.js.map +1 -0
- package/dist/configure-agent/cli.d.ts +2 -0
- package/dist/configure-agent/cli.d.ts.map +1 -0
- package/dist/configure-agent/cli.js +390 -0
- package/dist/configure-agent/cli.js.map +1 -0
- package/dist/discover/cli.d.ts +2 -0
- package/dist/discover/cli.d.ts.map +1 -0
- package/dist/discover/cli.js +97 -0
- package/dist/discover/cli.js.map +1 -0
- package/dist/discover/index.d.ts +19 -0
- package/dist/discover/index.d.ts.map +1 -0
- package/dist/discover/index.js +70 -0
- package/dist/discover/index.js.map +1 -0
- package/dist/discover/openapi.d.ts +9 -0
- package/dist/discover/openapi.d.ts.map +1 -0
- package/dist/discover/openapi.js +47 -0
- package/dist/discover/openapi.js.map +1 -0
- package/dist/discover/strategies/fig.d.ts +29 -0
- package/dist/discover/strategies/fig.d.ts.map +1 -0
- package/dist/discover/strategies/fig.js +82 -0
- package/dist/discover/strategies/fig.js.map +1 -0
- package/dist/discover/strategies/help-parser.d.ts +21 -0
- package/dist/discover/strategies/help-parser.d.ts.map +1 -0
- package/dist/discover/strategies/help-parser.js +121 -0
- package/dist/discover/strategies/help-parser.js.map +1 -0
- package/dist/discover/writer.d.ts +5 -0
- package/dist/discover/writer.d.ts.map +1 -0
- package/dist/discover/writer.js +14 -0
- package/dist/discover/writer.js.map +1 -0
- package/dist/gateway.d.ts +20 -0
- package/dist/gateway.d.ts.map +1 -0
- package/dist/gateway.js +125 -0
- package/dist/gateway.js.map +1 -0
- package/dist/hitl/api.d.ts +7 -0
- package/dist/hitl/api.d.ts.map +1 -0
- package/dist/hitl/api.js +35 -0
- package/dist/hitl/api.js.map +1 -0
- package/dist/hitl/batcher.d.ts +11 -0
- package/dist/hitl/batcher.d.ts.map +1 -0
- package/dist/hitl/batcher.js +37 -0
- package/dist/hitl/batcher.js.map +1 -0
- package/dist/hitl/engine.d.ts +36 -0
- package/dist/hitl/engine.d.ts.map +1 -0
- package/dist/hitl/engine.js +150 -0
- package/dist/hitl/engine.js.map +1 -0
- package/dist/hitl/formatter.d.ts +4 -0
- package/dist/hitl/formatter.d.ts.map +1 -0
- package/dist/hitl/formatter.js +31 -0
- package/dist/hitl/formatter.js.map +1 -0
- package/dist/hitl/parser.d.ts +7 -0
- package/dist/hitl/parser.d.ts.map +1 -0
- package/dist/hitl/parser.js +17 -0
- package/dist/hitl/parser.js.map +1 -0
- package/dist/hitl/provider-factory.d.ts +4 -0
- package/dist/hitl/provider-factory.d.ts.map +1 -0
- package/dist/hitl/provider-factory.js +42 -0
- package/dist/hitl/provider-factory.js.map +1 -0
- package/dist/hitl/providers/composite.d.ts +9 -0
- package/dist/hitl/providers/composite.d.ts.map +1 -0
- package/dist/hitl/providers/composite.js +23 -0
- package/dist/hitl/providers/composite.js.map +1 -0
- package/dist/hitl/providers/dashboard.d.ts +17 -0
- package/dist/hitl/providers/dashboard.d.ts.map +1 -0
- package/dist/hitl/providers/dashboard.js +210 -0
- package/dist/hitl/providers/dashboard.js.map +1 -0
- package/dist/hitl/providers/macos.d.ts +10 -0
- package/dist/hitl/providers/macos.d.ts.map +1 -0
- package/dist/hitl/providers/macos.js +65 -0
- package/dist/hitl/providers/macos.js.map +1 -0
- package/dist/hitl/providers/openclaw.d.ts +21 -0
- package/dist/hitl/providers/openclaw.d.ts.map +1 -0
- package/dist/hitl/providers/openclaw.js +106 -0
- package/dist/hitl/providers/openclaw.js.map +1 -0
- package/dist/hitl/providers/slack.d.ts +12 -0
- package/dist/hitl/providers/slack.d.ts.map +1 -0
- package/dist/hitl/providers/slack.js +24 -0
- package/dist/hitl/providers/slack.js.map +1 -0
- package/dist/hitl/providers/stdio.d.ts +12 -0
- package/dist/hitl/providers/stdio.d.ts.map +1 -0
- package/dist/hitl/providers/stdio.js +41 -0
- package/dist/hitl/providers/stdio.js.map +1 -0
- package/dist/hitl/providers/telegram.d.ts +22 -0
- package/dist/hitl/providers/telegram.d.ts.map +1 -0
- package/dist/hitl/providers/telegram.js +87 -0
- package/dist/hitl/providers/telegram.js.map +1 -0
- package/dist/hitl/providers/tui.d.ts +16 -0
- package/dist/hitl/providers/tui.d.ts.map +1 -0
- package/dist/hitl/providers/tui.js +169 -0
- package/dist/hitl/providers/tui.js.map +1 -0
- package/dist/hitl/providers/types.d.ts +18 -0
- package/dist/hitl/providers/types.d.ts.map +1 -0
- package/dist/hitl/providers/types.js +2 -0
- package/dist/hitl/providers/types.js.map +1 -0
- package/dist/hitl/providers/webhook.d.ts +13 -0
- package/dist/hitl/providers/webhook.d.ts.map +1 -0
- package/dist/hitl/providers/webhook.js +27 -0
- package/dist/hitl/providers/webhook.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +115 -0
- package/dist/index.js.map +1 -0
- package/dist/middleware/chain-builder.d.ts +16 -0
- package/dist/middleware/chain-builder.d.ts.map +1 -0
- package/dist/middleware/chain-builder.js +139 -0
- package/dist/middleware/chain-builder.js.map +1 -0
- package/dist/middleware/compose.d.ts +3 -0
- package/dist/middleware/compose.d.ts.map +1 -0
- package/dist/middleware/compose.js +15 -0
- package/dist/middleware/compose.js.map +1 -0
- package/dist/middleware/core/allowlist.d.ts +3 -0
- package/dist/middleware/core/allowlist.d.ts.map +1 -0
- package/dist/middleware/core/allowlist.js +23 -0
- package/dist/middleware/core/allowlist.js.map +1 -0
- package/dist/middleware/core/exec-policy.d.ts +3 -0
- package/dist/middleware/core/exec-policy.d.ts.map +1 -0
- package/dist/middleware/core/exec-policy.js +30 -0
- package/dist/middleware/core/exec-policy.js.map +1 -0
- package/dist/middleware/core/execute.d.ts +3 -0
- package/dist/middleware/core/execute.d.ts.map +1 -0
- package/dist/middleware/core/execute.js +35 -0
- package/dist/middleware/core/execute.js.map +1 -0
- package/dist/middleware/core/hitl-gate.d.ts +3 -0
- package/dist/middleware/core/hitl-gate.d.ts.map +1 -0
- package/dist/middleware/core/hitl-gate.js +38 -0
- package/dist/middleware/core/hitl-gate.js.map +1 -0
- package/dist/middleware/core/rate-limiter.d.ts +10 -0
- package/dist/middleware/core/rate-limiter.d.ts.map +1 -0
- package/dist/middleware/core/rate-limiter.js +32 -0
- package/dist/middleware/core/rate-limiter.js.map +1 -0
- package/dist/middleware/core/schema-validator.d.ts +3 -0
- package/dist/middleware/core/schema-validator.d.ts.map +1 -0
- package/dist/middleware/core/schema-validator.js +31 -0
- package/dist/middleware/core/schema-validator.js.map +1 -0
- package/dist/middleware/detectors/injection-detector.d.ts +12 -0
- package/dist/middleware/detectors/injection-detector.d.ts.map +1 -0
- package/dist/middleware/detectors/injection-detector.js +129 -0
- package/dist/middleware/detectors/injection-detector.js.map +1 -0
- package/dist/middleware/detectors/sensitivity-classifier.d.ts +12 -0
- package/dist/middleware/detectors/sensitivity-classifier.d.ts.map +1 -0
- package/dist/middleware/detectors/sensitivity-classifier.js +125 -0
- package/dist/middleware/detectors/sensitivity-classifier.js.map +1 -0
- package/dist/middleware/post/canary-token-injector.d.ts +10 -0
- package/dist/middleware/post/canary-token-injector.d.ts.map +1 -0
- package/dist/middleware/post/canary-token-injector.js +53 -0
- package/dist/middleware/post/canary-token-injector.js.map +1 -0
- package/dist/middleware/post/output-injection-detector.d.ts +7 -0
- package/dist/middleware/post/output-injection-detector.d.ts.map +1 -0
- package/dist/middleware/post/output-injection-detector.js +46 -0
- package/dist/middleware/post/output-injection-detector.js.map +1 -0
- package/dist/middleware/post/output-size-limiter.d.ts +7 -0
- package/dist/middleware/post/output-size-limiter.d.ts.map +1 -0
- package/dist/middleware/post/output-size-limiter.js +47 -0
- package/dist/middleware/post/output-size-limiter.js.map +1 -0
- package/dist/middleware/post/output-summarizer.d.ts +15 -0
- package/dist/middleware/post/output-summarizer.d.ts.map +1 -0
- package/dist/middleware/post/output-summarizer.js +38 -0
- package/dist/middleware/post/output-summarizer.js.map +1 -0
- package/dist/middleware/post/strip-query-params.d.ts +3 -0
- package/dist/middleware/post/strip-query-params.d.ts.map +1 -0
- package/dist/middleware/post/strip-query-params.js +22 -0
- package/dist/middleware/post/strip-query-params.js.map +1 -0
- package/dist/middleware/post/untrusted-envelope.d.ts +3 -0
- package/dist/middleware/post/untrusted-envelope.d.ts.map +1 -0
- package/dist/middleware/post/untrusted-envelope.js +10 -0
- package/dist/middleware/post/untrusted-envelope.js.map +1 -0
- package/dist/middleware/types.d.ts +32 -0
- package/dist/middleware/types.d.ts.map +1 -0
- package/dist/middleware/types.js +2 -0
- package/dist/middleware/types.js.map +1 -0
- package/dist/pool/http-client.d.ts +26 -0
- package/dist/pool/http-client.d.ts.map +1 -0
- package/dist/pool/http-client.js +109 -0
- package/dist/pool/http-client.js.map +1 -0
- package/dist/pool/oauth-provider.d.ts +34 -0
- package/dist/pool/oauth-provider.d.ts.map +1 -0
- package/dist/pool/oauth-provider.js +135 -0
- package/dist/pool/oauth-provider.js.map +1 -0
- package/dist/pool/pool.d.ts +30 -0
- package/dist/pool/pool.d.ts.map +1 -0
- package/dist/pool/pool.js +119 -0
- package/dist/pool/pool.js.map +1 -0
- package/dist/pool/required-mcps.d.ts +7 -0
- package/dist/pool/required-mcps.d.ts.map +1 -0
- package/dist/pool/required-mcps.js +18 -0
- package/dist/pool/required-mcps.js.map +1 -0
- package/dist/pool/sse-client.d.ts +22 -0
- package/dist/pool/sse-client.d.ts.map +1 -0
- package/dist/pool/sse-client.js +70 -0
- package/dist/pool/sse-client.js.map +1 -0
- package/dist/pool/stdio-client.d.ts +24 -0
- package/dist/pool/stdio-client.d.ts.map +1 -0
- package/dist/pool/stdio-client.js +77 -0
- package/dist/pool/stdio-client.js.map +1 -0
- package/dist/registry/registry.d.ts +19 -0
- package/dist/registry/registry.d.ts.map +1 -0
- package/dist/registry/registry.js +85 -0
- package/dist/registry/registry.js.map +1 -0
- package/dist/registry/sanitizer.d.ts +2 -0
- package/dist/registry/sanitizer.d.ts.map +1 -0
- package/dist/registry/sanitizer.js +31 -0
- package/dist/registry/sanitizer.js.map +1 -0
- package/dist/security/blocked-hosts.d.ts +6 -0
- package/dist/security/blocked-hosts.d.ts.map +1 -0
- package/dist/security/blocked-hosts.js +26 -0
- package/dist/security/blocked-hosts.js.map +1 -0
- package/dist/security/domain-allowlist.d.ts +7 -0
- package/dist/security/domain-allowlist.d.ts.map +1 -0
- package/dist/security/domain-allowlist.js +19 -0
- package/dist/security/domain-allowlist.js.map +1 -0
- package/dist/stdio-mode.d.ts +3 -0
- package/dist/stdio-mode.d.ts.map +1 -0
- package/dist/stdio-mode.js +130 -0
- package/dist/stdio-mode.js.map +1 -0
- package/dist/tools/exec.d.ts +20 -0
- package/dist/tools/exec.d.ts.map +1 -0
- package/dist/tools/exec.js +105 -0
- package/dist/tools/exec.js.map +1 -0
- package/dist/tools/http.d.ts +13 -0
- package/dist/tools/http.d.ts.map +1 -0
- package/dist/tools/http.js +99 -0
- package/dist/tools/http.js.map +1 -0
- package/dist/transport/agent-server.d.ts +26 -0
- package/dist/transport/agent-server.d.ts.map +1 -0
- package/dist/transport/agent-server.js +55 -0
- package/dist/transport/agent-server.js.map +1 -0
- package/dist/transport/mcp-normalizer.d.ts +9 -0
- package/dist/transport/mcp-normalizer.d.ts.map +1 -0
- package/dist/transport/mcp-normalizer.js +12 -0
- package/dist/transport/mcp-normalizer.js.map +1 -0
- package/dist/transport/sse-server.d.ts +7 -0
- package/dist/transport/sse-server.d.ts.map +1 -0
- package/dist/transport/sse-server.js +94 -0
- package/dist/transport/sse-server.js.map +1 -0
- package/dist/transport/stdio-server.d.ts +3 -0
- package/dist/transport/stdio-server.d.ts.map +1 -0
- package/dist/transport/stdio-server.js +12 -0
- package/dist/transport/stdio-server.js.map +1 -0
- package/dist/types.d.ts +15 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/dist/util/id.d.ts +5 -0
- package/dist/util/id.d.ts.map +1 -0
- package/dist/util/id.js +16 -0
- package/dist/util/id.js.map +1 -0
- package/dist/util/logger.d.ts +4 -0
- package/dist/util/logger.d.ts.map +1 -0
- package/dist/util/logger.js +24 -0
- package/dist/util/logger.js.map +1 -0
- package/dist/version.d.ts +2 -0
- package/dist/version.d.ts.map +1 -0
- package/dist/version.js +4 -0
- package/dist/version.js.map +1 -0
- package/examples/claude-code-setup.md +77 -0
- package/examples/gateway.yaml +118 -0
- package/examples/local-dev.yaml +41 -0
- package/examples/openclaw-setup.md +52 -0
- package/examples/profiles.yaml +103 -0
- package/package.json +80 -3
- package/schema.json +943 -0
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { McpError, ErrorCode } from '@modelcontextprotocol/sdk/types.js';
|
|
2
|
+
const windows = new Map();
|
|
3
|
+
function getKey(agentId, toolName, per) {
|
|
4
|
+
return per === 'tool' ? `${agentId}:${toolName}` : agentId;
|
|
5
|
+
}
|
|
6
|
+
function pruneWindow(timestamps, windowMs, now) {
|
|
7
|
+
const cutoff = now - windowMs;
|
|
8
|
+
const idx = timestamps.findIndex((t) => t > cutoff);
|
|
9
|
+
return idx === -1 ? [] : timestamps.slice(idx);
|
|
10
|
+
}
|
|
11
|
+
export function rateLimiterMiddleware(opts) {
|
|
12
|
+
const { max_requests, window_ms, per = 'agent' } = opts;
|
|
13
|
+
return async (ctx, next) => {
|
|
14
|
+
const key = getKey(ctx.agentId, ctx.toolName, per);
|
|
15
|
+
const now = Date.now();
|
|
16
|
+
let timestamps = windows.get(key) ?? [];
|
|
17
|
+
timestamps = pruneWindow(timestamps, window_ms, now);
|
|
18
|
+
if (timestamps.length >= max_requests) {
|
|
19
|
+
const retryAfterMs = timestamps[0] + window_ms - now;
|
|
20
|
+
throw new McpError(ErrorCode.InvalidRequest, `Rate limit exceeded (${max_requests}/${window_ms}ms). Retry after ${Math.ceil(retryAfterMs)}ms.`);
|
|
21
|
+
}
|
|
22
|
+
const response = await next();
|
|
23
|
+
timestamps.push(now);
|
|
24
|
+
windows.set(key, timestamps);
|
|
25
|
+
return response;
|
|
26
|
+
};
|
|
27
|
+
}
|
|
28
|
+
/** For testing */
|
|
29
|
+
export function resetRateLimiterState() {
|
|
30
|
+
windows.clear();
|
|
31
|
+
}
|
|
32
|
+
//# sourceMappingURL=rate-limiter.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"rate-limiter.js","sourceRoot":"","sources":["../../../src/middleware/core/rate-limiter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,oCAAoC,CAAC;AASzE,MAAM,OAAO,GAAG,IAAI,GAAG,EAAoB,CAAC;AAE5C,SAAS,MAAM,CAAC,OAAe,EAAE,QAAgB,EAAE,GAAqB;IACtE,OAAO,GAAG,KAAK,MAAM,CAAC,CAAC,CAAC,GAAG,OAAO,IAAI,QAAQ,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC;AAC7D,CAAC;AAED,SAAS,WAAW,CAAC,UAAoB,EAAE,QAAgB,EAAE,GAAW;IACtE,MAAM,MAAM,GAAG,GAAG,GAAG,QAAQ,CAAC;IAC9B,MAAM,GAAG,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC;IACpD,OAAO,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;AACjD,CAAC;AAED,MAAM,UAAU,qBAAqB,CAAC,IAAwB;IAC5D,MAAM,EAAE,YAAY,EAAE,SAAS,EAAE,GAAG,GAAG,OAAO,EAAE,GAAG,IAAI,CAAC;IAExD,OAAO,KAAK,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE;QACzB,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC;QACnD,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,IAAI,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;QACxC,UAAU,GAAG,WAAW,CAAC,UAAU,EAAE,SAAS,EAAE,GAAG,CAAC,CAAC;QAErD,IAAI,UAAU,CAAC,MAAM,IAAI,YAAY,EAAE,CAAC;YACtC,MAAM,YAAY,GAAG,UAAU,CAAC,CAAC,CAAC,GAAG,SAAS,GAAG,GAAG,CAAC;YACrD,MAAM,IAAI,QAAQ,CAChB,SAAS,CAAC,cAAc,EACxB,wBAAwB,YAAY,IAAI,SAAS,oBAAoB,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAClG,CAAC;QACJ,CAAC;QAED,MAAM,QAAQ,GAAG,MAAM,IAAI,EAAE,CAAC;QAC9B,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACrB,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,UAAU,CAAC,CAAC;QAC7B,OAAO,QAAQ,CAAC;IAClB,CAAC,CAAC;AACJ,CAAC;AAED,kBAAkB;AAClB,MAAM,UAAU,qBAAqB;IACnC,OAAO,CAAC,KAAK,EAAE,CAAC;AAClB,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"schema-validator.d.ts","sourceRoot":"","sources":["../../../src/middleware/core/schema-validator.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAU9C,wBAAgB,yBAAyB,IAAI,UAAU,CAuBtD"}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import AjvModule from 'ajv';
|
|
2
|
+
import { McpError, ErrorCode } from '@modelcontextprotocol/sdk/types.js';
|
|
3
|
+
/* eslint-disable @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access */
|
|
4
|
+
// Handle both ESM default and CJS exports — Ajv uses CJS which can appear as { default: Ajv } in ESM
|
|
5
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
6
|
+
const Ajv = AjvModule.default ?? AjvModule;
|
|
7
|
+
const ajv = new Ajv({ allErrors: true, strict: false });
|
|
8
|
+
const validatorCache = new Map();
|
|
9
|
+
/* eslint-enable @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access */
|
|
10
|
+
export function schemaValidatorMiddleware() {
|
|
11
|
+
return async (ctx, next) => {
|
|
12
|
+
const tools = ctx.deps.registry.getAllTools();
|
|
13
|
+
const tool = tools.find((t) => t.name === ctx.toolName);
|
|
14
|
+
if (!tool?.inputSchema)
|
|
15
|
+
return next();
|
|
16
|
+
let validate = validatorCache.get(ctx.toolName);
|
|
17
|
+
if (!validate) {
|
|
18
|
+
// eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access
|
|
19
|
+
validate = ajv.compile(tool.inputSchema);
|
|
20
|
+
validatorCache.set(ctx.toolName, validate);
|
|
21
|
+
}
|
|
22
|
+
if (!validate(ctx.args)) {
|
|
23
|
+
const errors = (validate.errors ?? [])
|
|
24
|
+
.map((e) => `${e.instancePath || '/'}: ${e.message ?? 'unknown error'}`)
|
|
25
|
+
.join('; ') || 'Unknown validation error';
|
|
26
|
+
throw new McpError(ErrorCode.InvalidParams, `Invalid arguments: ${errors}`);
|
|
27
|
+
}
|
|
28
|
+
return next();
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
//# sourceMappingURL=schema-validator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"schema-validator.js","sourceRoot":"","sources":["../../../src/middleware/core/schema-validator.ts"],"names":[],"mappings":"AAAA,OAAO,SAAS,MAAM,KAAK,CAAC;AAE5B,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,oCAAoC,CAAC;AAGzE,2IAA2I;AAC3I,qGAAqG;AACrG,8DAA8D;AAC9D,MAAM,GAAG,GAAI,SAAiB,CAAC,OAAO,IAAI,SAAS,CAAC;AACpD,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,CAAC;AACxD,MAAM,cAAc,GAAG,IAAI,GAAG,EAA4B,CAAC;AAC3D,0IAA0I;AAE1I,MAAM,UAAU,yBAAyB;IACvC,OAAO,KAAK,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE;QACzB,MAAM,KAAK,GAAG,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,CAAC;QAC9C,MAAM,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,GAAG,CAAC,QAAQ,CAAC,CAAC;QACxD,IAAI,CAAC,IAAI,EAAE,WAAW;YAAE,OAAO,IAAI,EAAE,CAAC;QAEtC,IAAI,QAAQ,GAAG,cAAc,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAChD,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,yGAAyG;YACzG,QAAQ,GAAG,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAqB,CAAC;YAC7D,cAAc,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;QAC7C,CAAC;QAED,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC;YACxB,MAAM,MAAM,GACV,CAAC,QAAQ,CAAC,MAAM,IAAI,EAAE,CAAC;iBACpB,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC,YAAY,IAAI,GAAG,KAAK,CAAC,CAAC,OAAO,IAAI,eAAe,EAAE,CAAC;iBACvE,IAAI,CAAC,IAAI,CAAC,IAAI,0BAA0B,CAAC;YAC9C,MAAM,IAAI,QAAQ,CAAC,SAAS,CAAC,aAAa,EAAE,sBAAsB,MAAM,EAAE,CAAC,CAAC;QAC9E,CAAC;QAED,OAAO,IAAI,EAAE,CAAC;IAChB,CAAC,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import type { Middleware } from '../types.js';
|
|
2
|
+
export type InjectionDetectorMode = 'detect' | 'mangle' | 'escalate';
|
|
3
|
+
export interface InjectionDetectorOptions {
|
|
4
|
+
backend?: 'regex' | 'deberta';
|
|
5
|
+
mode?: InjectionDetectorMode;
|
|
6
|
+
/** URL of DeBERTa inference server (e.g. http://localhost:8000/predict) */
|
|
7
|
+
inference_url?: string;
|
|
8
|
+
/** Confidence threshold for DeBERTa (0-1, default 0.8) */
|
|
9
|
+
threshold?: number;
|
|
10
|
+
}
|
|
11
|
+
export declare function injectionDetectorMiddleware(opts?: InjectionDetectorOptions): Middleware;
|
|
12
|
+
//# sourceMappingURL=injection-detector.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"injection-detector.d.ts","sourceRoot":"","sources":["../../../src/middleware/detectors/injection-detector.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAyB9C,MAAM,MAAM,qBAAqB,GAAG,QAAQ,GAAG,QAAQ,GAAG,UAAU,CAAC;AAErE,MAAM,WAAW,wBAAwB;IACvC,OAAO,CAAC,EAAE,OAAO,GAAG,SAAS,CAAC;IAC9B,IAAI,CAAC,EAAE,qBAAqB,CAAC;IAC7B,2EAA2E;IAC3E,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,0DAA0D;IAC1D,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AA6CD,wBAAgB,2BAA2B,CAAC,IAAI,GAAE,wBAA6B,GAAG,UAAU,CA2F3F"}
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
import { childLogger } from '../../util/logger.js';
|
|
2
|
+
const log = childLogger('mw:injection-detector');
|
|
3
|
+
const INJECTION_PATTERNS = [
|
|
4
|
+
/ignore\s+(all\s+)?previous\s+instructions/i,
|
|
5
|
+
/ignore\s+(all\s+)?prior\s+instructions/i,
|
|
6
|
+
/disregard\s+(all\s+)?previous/i,
|
|
7
|
+
/you\s+are\s+now\s+(a|an)\s/i,
|
|
8
|
+
/new\s+instructions?\s*:/i,
|
|
9
|
+
/<\s*system\s*>/i,
|
|
10
|
+
/\[INST\]/i,
|
|
11
|
+
/\[\/INST\]/i,
|
|
12
|
+
/<<\s*SYS\s*>>/i,
|
|
13
|
+
/do\s+not\s+follow\s+(any\s+)?previous/i,
|
|
14
|
+
/forget\s+(all\s+)?(your\s+)?instructions/i,
|
|
15
|
+
/override\s+(all\s+)?instructions/i,
|
|
16
|
+
/\bact\s+as\s+(a|an|if)\b/i,
|
|
17
|
+
/\bpretend\s+(you\s+are|to\s+be)\b/i,
|
|
18
|
+
/\brole\s*play\b/i,
|
|
19
|
+
/\bjailbreak\b/i,
|
|
20
|
+
/\bDAN\s+mode\b/i,
|
|
21
|
+
];
|
|
22
|
+
async function classifyWithDeberta(text, inferenceUrl, threshold) {
|
|
23
|
+
const response = await fetch(inferenceUrl, {
|
|
24
|
+
method: 'POST',
|
|
25
|
+
headers: { 'Content-Type': 'application/json' },
|
|
26
|
+
body: JSON.stringify({ text }),
|
|
27
|
+
signal: AbortSignal.timeout(5000),
|
|
28
|
+
});
|
|
29
|
+
if (!response.ok) {
|
|
30
|
+
throw new Error(`DeBERTa inference failed: ${response.status}`);
|
|
31
|
+
}
|
|
32
|
+
const result = (await response.json());
|
|
33
|
+
const score = result.score ?? result.probability ?? 0;
|
|
34
|
+
return { isInjection: score >= threshold, score };
|
|
35
|
+
}
|
|
36
|
+
function classifyWithRegex(text) {
|
|
37
|
+
const matched = [];
|
|
38
|
+
for (const pattern of INJECTION_PATTERNS) {
|
|
39
|
+
if (pattern.test(text)) {
|
|
40
|
+
matched.push(pattern.source);
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
return { isInjection: matched.length > 0, matchedPatterns: matched };
|
|
44
|
+
}
|
|
45
|
+
function mangleText(text) {
|
|
46
|
+
let result = text;
|
|
47
|
+
for (const pattern of INJECTION_PATTERNS) {
|
|
48
|
+
result = result.replace(new RegExp(pattern, 'gi'), '[REDACTED: suspected injection]');
|
|
49
|
+
}
|
|
50
|
+
return result;
|
|
51
|
+
}
|
|
52
|
+
export function injectionDetectorMiddleware(opts = {}) {
|
|
53
|
+
const { backend = 'regex', mode = 'detect', inference_url, threshold = 0.8 } = opts;
|
|
54
|
+
return async (ctx, next) => {
|
|
55
|
+
// Pre-execution: scan args
|
|
56
|
+
const argsText = JSON.stringify(ctx.args);
|
|
57
|
+
let preInjection;
|
|
58
|
+
if (backend === 'deberta' && inference_url) {
|
|
59
|
+
try {
|
|
60
|
+
const result = await classifyWithDeberta(argsText, inference_url, threshold);
|
|
61
|
+
preInjection = result.isInjection;
|
|
62
|
+
if (preInjection) {
|
|
63
|
+
log.warn({ agentId: ctx.agentId, tool: ctx.toolName, score: result.score }, 'DeBERTa: injection detected in args');
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
catch (err) {
|
|
67
|
+
log.warn({ err }, 'DeBERTa inference failed for args, falling back to regex');
|
|
68
|
+
preInjection = classifyWithRegex(argsText).isInjection;
|
|
69
|
+
if (mode === 'escalate') {
|
|
70
|
+
ctx.meta.needsApproval = true;
|
|
71
|
+
ctx.meta.approvalReason = 'DeBERTa inference unavailable — escalating as precaution';
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
else {
|
|
76
|
+
const result = classifyWithRegex(argsText);
|
|
77
|
+
preInjection = result.isInjection;
|
|
78
|
+
if (preInjection) {
|
|
79
|
+
log.warn({ agentId: ctx.agentId, tool: ctx.toolName, patterns: result.matchedPatterns }, 'Regex: injection detected in args');
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
if (preInjection) {
|
|
83
|
+
ctx.deps.auditLogger.log({
|
|
84
|
+
agent_id: ctx.agentId,
|
|
85
|
+
tool: ctx.toolName,
|
|
86
|
+
args: JSON.stringify(ctx.args),
|
|
87
|
+
result: 'injection_detected_args',
|
|
88
|
+
});
|
|
89
|
+
if (mode === 'escalate') {
|
|
90
|
+
ctx.meta.needsApproval = true;
|
|
91
|
+
ctx.meta.approvalReason = 'Prompt injection detected in tool arguments';
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
const response = await next();
|
|
95
|
+
// Post-execution: scan response
|
|
96
|
+
let postInjection;
|
|
97
|
+
if (backend === 'deberta' && inference_url) {
|
|
98
|
+
try {
|
|
99
|
+
const result = await classifyWithDeberta(response.text, inference_url, threshold);
|
|
100
|
+
postInjection = result.isInjection;
|
|
101
|
+
}
|
|
102
|
+
catch (err) {
|
|
103
|
+
log.warn({ err }, 'DeBERTa inference failed for response, falling back to regex');
|
|
104
|
+
postInjection = classifyWithRegex(response.text).isInjection;
|
|
105
|
+
if (mode === 'escalate') {
|
|
106
|
+
ctx.meta.needsApproval = true;
|
|
107
|
+
ctx.meta.approvalReason = 'DeBERTa inference unavailable — escalating as precaution';
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
else {
|
|
112
|
+
postInjection = classifyWithRegex(response.text).isInjection;
|
|
113
|
+
}
|
|
114
|
+
if (postInjection) {
|
|
115
|
+
log.warn({ agentId: ctx.agentId, tool: ctx.toolName }, 'Injection pattern detected in response');
|
|
116
|
+
ctx.deps.auditLogger.log({
|
|
117
|
+
agent_id: ctx.agentId,
|
|
118
|
+
tool: ctx.toolName,
|
|
119
|
+
args: JSON.stringify(ctx.args),
|
|
120
|
+
result: 'injection_detected_response',
|
|
121
|
+
});
|
|
122
|
+
if (mode === 'mangle') {
|
|
123
|
+
response.text = mangleText(response.text);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
return response;
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
//# sourceMappingURL=injection-detector.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"injection-detector.js","sourceRoot":"","sources":["../../../src/middleware/detectors/injection-detector.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import type { Middleware } from '../types.js';
|
|
2
|
+
export type SensitivityMode = 'detect' | 'escalate';
|
|
3
|
+
export type SensitivityBackend = 'heuristic' | 'llm';
|
|
4
|
+
export interface SensitivityClassifierOptions {
|
|
5
|
+
mode?: SensitivityMode;
|
|
6
|
+
threshold?: number;
|
|
7
|
+
backend?: SensitivityBackend;
|
|
8
|
+
/** Model ID for LLM backend (Vercel AI SDK model reference) */
|
|
9
|
+
model?: string;
|
|
10
|
+
}
|
|
11
|
+
export declare function sensitivityClassifierMiddleware(opts?: SensitivityClassifierOptions): Middleware;
|
|
12
|
+
//# sourceMappingURL=sensitivity-classifier.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sensitivity-classifier.d.ts","sourceRoot":"","sources":["../../../src/middleware/detectors/sensitivity-classifier.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAK9C,MAAM,MAAM,eAAe,GAAG,QAAQ,GAAG,UAAU,CAAC;AACpD,MAAM,MAAM,kBAAkB,GAAG,WAAW,GAAG,KAAK,CAAC;AAErD,MAAM,WAAW,4BAA4B;IAC3C,IAAI,CAAC,EAAE,eAAe,CAAC;IACvB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,kBAAkB,CAAC;IAC7B,+DAA+D;IAC/D,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAsFD,wBAAgB,+BAA+B,CAC7C,IAAI,GAAE,4BAAiC,GACtC,UAAU,CAeZ"}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
import { childLogger } from '../../util/logger.js';
|
|
2
|
+
const log = childLogger('mw:sensitivity-classifier');
|
|
3
|
+
// PII and sensitive data patterns
|
|
4
|
+
const SENSITIVITY_PATTERNS = [
|
|
5
|
+
// SSN
|
|
6
|
+
{ pattern: /\b\d{3}-\d{2}-\d{4}\b/, weight: 0.9, label: 'SSN' },
|
|
7
|
+
// Credit card (Luhn-like)
|
|
8
|
+
{
|
|
9
|
+
pattern: /\b(?:4[0-9]{12}(?:[0-9]{3})?|5[1-5][0-9]{14}|3[47][0-9]{13}|6(?:011|5[0-9]{2})[0-9]{12})\b/,
|
|
10
|
+
weight: 0.9,
|
|
11
|
+
label: 'credit_card',
|
|
12
|
+
},
|
|
13
|
+
// Email
|
|
14
|
+
{ pattern: /\b[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\.[A-Z|a-z]{2,}\b/, weight: 0.3, label: 'email' },
|
|
15
|
+
// Phone
|
|
16
|
+
{
|
|
17
|
+
pattern: /\b(?:\+?1[-.\s]?)?\(?\d{3}\)?[-.\s]?\d{3}[-.\s]?\d{4}\b/,
|
|
18
|
+
weight: 0.3,
|
|
19
|
+
label: 'phone',
|
|
20
|
+
},
|
|
21
|
+
// API keys / tokens (generic long hex/base64 strings)
|
|
22
|
+
{
|
|
23
|
+
pattern: /\b(?:sk|pk|api|token|key|secret|bearer)[-_]?[A-Za-z0-9_-]{20,}\b/i,
|
|
24
|
+
weight: 0.7,
|
|
25
|
+
label: 'api_key',
|
|
26
|
+
},
|
|
27
|
+
// AWS keys
|
|
28
|
+
{ pattern: /\bAKIA[0-9A-Z]{16}\b/, weight: 0.95, label: 'aws_access_key' },
|
|
29
|
+
// Private keys
|
|
30
|
+
{ pattern: /-----BEGIN\s(?:RSA\s)?PRIVATE\sKEY-----/, weight: 0.95, label: 'private_key' },
|
|
31
|
+
// JWT
|
|
32
|
+
{
|
|
33
|
+
pattern: /\beyJ[A-Za-z0-9_-]+\.eyJ[A-Za-z0-9_-]+\.[A-Za-z0-9_-]+\b/,
|
|
34
|
+
weight: 0.6,
|
|
35
|
+
label: 'jwt',
|
|
36
|
+
},
|
|
37
|
+
// Internal IPs
|
|
38
|
+
{
|
|
39
|
+
pattern: /\b(?:10\.\d{1,3}\.\d{1,3}\.\d{1,3}|172\.(?:1[6-9]|2\d|3[01])\.\d{1,3}\.\d{1,3}|192\.168\.\d{1,3}\.\d{1,3})\b/,
|
|
40
|
+
weight: 0.4,
|
|
41
|
+
label: 'internal_ip',
|
|
42
|
+
},
|
|
43
|
+
// Password-like fields
|
|
44
|
+
{ pattern: /["']?password["']?\s*[:=]\s*["'][^"']+["']/i, weight: 0.8, label: 'password' },
|
|
45
|
+
];
|
|
46
|
+
function heuristicScore(text) {
|
|
47
|
+
let maxWeight = 0;
|
|
48
|
+
const labels = [];
|
|
49
|
+
for (const { pattern, weight, label } of SENSITIVITY_PATTERNS) {
|
|
50
|
+
if (pattern.test(text)) {
|
|
51
|
+
labels.push(label);
|
|
52
|
+
maxWeight = Math.max(maxWeight, weight);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
return { score: maxWeight, labels };
|
|
56
|
+
}
|
|
57
|
+
async function llmScore(text, model) {
|
|
58
|
+
const { generateText } = await import('ai');
|
|
59
|
+
const { text: result } = await generateText({
|
|
60
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-assignment
|
|
61
|
+
model: model,
|
|
62
|
+
system: 'You are a data sensitivity classifier. Rate the sensitivity of the following data on a scale from 0.0 to 1.0, where 0 is public information and 1.0 is highly sensitive (PII, credentials, medical records, financial data). Respond with ONLY a JSON object: {"score": <number>, "reasoning": "<brief explanation>"}',
|
|
63
|
+
prompt: text.slice(0, 20_000),
|
|
64
|
+
maxOutputTokens: 256,
|
|
65
|
+
});
|
|
66
|
+
try {
|
|
67
|
+
return JSON.parse(result);
|
|
68
|
+
}
|
|
69
|
+
catch {
|
|
70
|
+
log.warn('Failed to parse LLM sensitivity response, falling back to high score');
|
|
71
|
+
return {
|
|
72
|
+
score: 1.0,
|
|
73
|
+
reasoning: 'Failed to parse LLM response — defaulting to high sensitivity',
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
export function sensitivityClassifierMiddleware(opts = {}) {
|
|
78
|
+
const { mode = 'detect', threshold = 0.7, backend = 'heuristic', model } = opts;
|
|
79
|
+
return async (ctx, next) => {
|
|
80
|
+
// Pre-execution: scan args
|
|
81
|
+
const argsText = JSON.stringify(ctx.args);
|
|
82
|
+
await checkSensitivity(argsText, 'args', ctx, backend, model, threshold, mode);
|
|
83
|
+
const response = await next();
|
|
84
|
+
// Post-execution: scan response
|
|
85
|
+
await checkSensitivity(response.text, 'response', ctx, backend, model, threshold, mode);
|
|
86
|
+
return response;
|
|
87
|
+
};
|
|
88
|
+
}
|
|
89
|
+
async function checkSensitivity(text, phase, ctx, backend, model, threshold, mode) {
|
|
90
|
+
let score;
|
|
91
|
+
let details;
|
|
92
|
+
if (backend === 'llm' && model) {
|
|
93
|
+
try {
|
|
94
|
+
const result = await llmScore(text, model);
|
|
95
|
+
score = result.score;
|
|
96
|
+
details = result.reasoning;
|
|
97
|
+
}
|
|
98
|
+
catch (err) {
|
|
99
|
+
log.warn({ err }, 'LLM sensitivity classification failed, falling back to heuristic');
|
|
100
|
+
const h = heuristicScore(text);
|
|
101
|
+
score = h.score;
|
|
102
|
+
details = h.labels.join(', ');
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
else {
|
|
106
|
+
const h = heuristicScore(text);
|
|
107
|
+
score = h.score;
|
|
108
|
+
details = h.labels.join(', ');
|
|
109
|
+
}
|
|
110
|
+
if (score >= threshold) {
|
|
111
|
+
log.warn({ agentId: ctx.agentId, tool: ctx.toolName, score, phase, details }, 'High sensitivity data detected');
|
|
112
|
+
ctx.deps.auditLogger.log({
|
|
113
|
+
agent_id: ctx.agentId,
|
|
114
|
+
tool: ctx.toolName,
|
|
115
|
+
args: JSON.stringify(ctx.args),
|
|
116
|
+
result: `sensitivity_${phase}`,
|
|
117
|
+
error: `score=${score}, labels=${details}`,
|
|
118
|
+
});
|
|
119
|
+
if (mode === 'escalate' && phase === 'args' && !ctx.meta.needsApproval) {
|
|
120
|
+
ctx.meta.needsApproval = true;
|
|
121
|
+
ctx.meta.approvalReason = `High sensitivity data detected (score: ${score}, ${details})`;
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
//# sourceMappingURL=sensitivity-classifier.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sensitivity-classifier.js","sourceRoot":"","sources":["../../../src/middleware/detectors/sensitivity-classifier.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import type { Middleware } from '../types.js';
|
|
2
|
+
export declare function canaryTokenInjectorMiddleware(): Middleware;
|
|
3
|
+
/** For testing */
|
|
4
|
+
export declare function getActiveCanaryTokens(): Map<string, {
|
|
5
|
+
agentId: string;
|
|
6
|
+
tool: string;
|
|
7
|
+
createdAt: number;
|
|
8
|
+
}>;
|
|
9
|
+
export declare function resetCanaryTokens(): void;
|
|
10
|
+
//# sourceMappingURL=canary-token-injector.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"canary-token-injector.d.ts","sourceRoot":"","sources":["../../../src/middleware/post/canary-token-injector.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAqB9C,wBAAgB,6BAA6B,IAAI,UAAU,CAkC1D;AAED,kBAAkB;AAClB,wBAAgB,qBAAqB,IAAI,GAAG,CAC1C,MAAM,EACN;IAAE,OAAO,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,CACrD,CAEA;AAED,wBAAgB,iBAAiB,IAAI,IAAI,CAExC"}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { randomBytes } from 'crypto';
|
|
2
|
+
import { childLogger } from '../../util/logger.js';
|
|
3
|
+
const log = childLogger('mw:canary-token');
|
|
4
|
+
const activeTokens = new Map();
|
|
5
|
+
const TOKEN_TTL_MS = 600_000; // 10 minutes
|
|
6
|
+
function generateCanary() {
|
|
7
|
+
return `CANARY-${randomBytes(8).toString('hex').toUpperCase()}`;
|
|
8
|
+
}
|
|
9
|
+
function pruneExpired() {
|
|
10
|
+
const now = Date.now();
|
|
11
|
+
for (const [token, meta] of activeTokens) {
|
|
12
|
+
if (now - meta.createdAt > TOKEN_TTL_MS) {
|
|
13
|
+
activeTokens.delete(token);
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
export function canaryTokenInjectorMiddleware() {
|
|
18
|
+
return async (ctx, next) => {
|
|
19
|
+
// Pre-execution: scan inbound args for leaked canary tokens
|
|
20
|
+
const argsStr = JSON.stringify(ctx.args);
|
|
21
|
+
for (const [token, meta] of activeTokens) {
|
|
22
|
+
if (argsStr.includes(token)) {
|
|
23
|
+
log.error({ agentId: ctx.agentId, tool: ctx.toolName, leakedFrom: meta.tool, token }, 'Canary token detected in tool arguments — possible data exfiltration');
|
|
24
|
+
ctx.deps.auditLogger.log({
|
|
25
|
+
agent_id: ctx.agentId,
|
|
26
|
+
tool: ctx.toolName,
|
|
27
|
+
args: JSON.stringify(ctx.args),
|
|
28
|
+
result: 'canary_leaked',
|
|
29
|
+
error: `Canary token ${token} from ${meta.tool} found in args`,
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
const response = await next();
|
|
34
|
+
// Post-execution: inject a canary token into the response
|
|
35
|
+
pruneExpired();
|
|
36
|
+
const canary = generateCanary();
|
|
37
|
+
activeTokens.set(canary, {
|
|
38
|
+
agentId: ctx.agentId,
|
|
39
|
+
tool: ctx.toolName,
|
|
40
|
+
createdAt: Date.now(),
|
|
41
|
+
});
|
|
42
|
+
response.text += `\n<!-- ${canary} -->`;
|
|
43
|
+
return response;
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
/** For testing */
|
|
47
|
+
export function getActiveCanaryTokens() {
|
|
48
|
+
return activeTokens;
|
|
49
|
+
}
|
|
50
|
+
export function resetCanaryTokens() {
|
|
51
|
+
activeTokens.clear();
|
|
52
|
+
}
|
|
53
|
+
//# sourceMappingURL=canary-token-injector.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"canary-token-injector.js","sourceRoot":"","sources":["../../../src/middleware/post/canary-token-injector.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,QAAQ,CAAC;AAErC,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAEnD,MAAM,GAAG,GAAG,WAAW,CAAC,iBAAiB,CAAC,CAAC;AAE3C,MAAM,YAAY,GAAG,IAAI,GAAG,EAAgE,CAAC;AAC7F,MAAM,YAAY,GAAG,OAAO,CAAC,CAAC,aAAa;AAE3C,SAAS,cAAc;IACrB,OAAO,UAAU,WAAW,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,WAAW,EAAE,EAAE,CAAC;AAClE,CAAC;AAED,SAAS,YAAY;IACnB,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IACvB,KAAK,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,IAAI,YAAY,EAAE,CAAC;QACzC,IAAI,GAAG,GAAG,IAAI,CAAC,SAAS,GAAG,YAAY,EAAE,CAAC;YACxC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC;AACH,CAAC;AAED,MAAM,UAAU,6BAA6B;IAC3C,OAAO,KAAK,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE;QACzB,4DAA4D;QAC5D,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACzC,KAAK,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,IAAI,YAAY,EAAE,CAAC;YACzC,IAAI,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC5B,GAAG,CAAC,KAAK,CACP,EAAE,OAAO,EAAE,GAAG,CAAC,OAAO,EAAE,IAAI,EAAE,GAAG,CAAC,QAAQ,EAAE,UAAU,EAAE,IAAI,CAAC,IAAI,EAAE,KAAK,EAAE,EAC1E,sEAAsE,CACvE,CAAC;gBACF,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC;oBACvB,QAAQ,EAAE,GAAG,CAAC,OAAO;oBACrB,IAAI,EAAE,GAAG,CAAC,QAAQ;oBAClB,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC;oBAC9B,MAAM,EAAE,eAAe;oBACvB,KAAK,EAAE,gBAAgB,KAAK,SAAS,IAAI,CAAC,IAAI,gBAAgB;iBAC/D,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,MAAM,QAAQ,GAAG,MAAM,IAAI,EAAE,CAAC;QAE9B,0DAA0D;QAC1D,YAAY,EAAE,CAAC;QACf,MAAM,MAAM,GAAG,cAAc,EAAE,CAAC;QAChC,YAAY,CAAC,GAAG,CAAC,MAAM,EAAE;YACvB,OAAO,EAAE,GAAG,CAAC,OAAO;YACpB,IAAI,EAAE,GAAG,CAAC,QAAQ;YAClB,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;SACtB,CAAC,CAAC;QAEH,QAAQ,CAAC,IAAI,IAAI,UAAU,MAAM,MAAM,CAAC;QACxC,OAAO,QAAQ,CAAC;IAClB,CAAC,CAAC;AACJ,CAAC;AAED,kBAAkB;AAClB,MAAM,UAAU,qBAAqB;IAInC,OAAO,YAAY,CAAC;AACtB,CAAC;AAED,MAAM,UAAU,iBAAiB;IAC/B,YAAY,CAAC,KAAK,EAAE,CAAC;AACvB,CAAC"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import type { Middleware } from '../types.js';
|
|
2
|
+
export type InjectionDetectorMode = 'detect' | 'mangle';
|
|
3
|
+
export interface OutputInjectionDetectorOptions {
|
|
4
|
+
mode?: InjectionDetectorMode;
|
|
5
|
+
}
|
|
6
|
+
export declare function outputInjectionDetectorMiddleware(opts?: OutputInjectionDetectorOptions): Middleware;
|
|
7
|
+
//# sourceMappingURL=output-injection-detector.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"output-injection-detector.d.ts","sourceRoot":"","sources":["../../../src/middleware/post/output-injection-detector.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAwB9C,MAAM,MAAM,qBAAqB,GAAG,QAAQ,GAAG,QAAQ,CAAC;AAExD,MAAM,WAAW,8BAA8B;IAC7C,IAAI,CAAC,EAAE,qBAAqB,CAAC;CAC9B;AAED,wBAAgB,iCAAiC,CAC/C,IAAI,GAAE,8BAAmC,GACxC,UAAU,CAkCZ"}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { childLogger } from '../../util/logger.js';
|
|
2
|
+
const log = childLogger('mw:output-injection');
|
|
3
|
+
const INJECTION_PATTERNS = [
|
|
4
|
+
/ignore\s+(all\s+)?previous\s+instructions/i,
|
|
5
|
+
/ignore\s+(all\s+)?prior\s+instructions/i,
|
|
6
|
+
/disregard\s+(all\s+)?previous/i,
|
|
7
|
+
/you\s+are\s+now\s+(a|an)\s/i,
|
|
8
|
+
/new\s+instructions?\s*:/i,
|
|
9
|
+
/system\s*:\s/i,
|
|
10
|
+
/<\s*system\s*>/i,
|
|
11
|
+
/\[INST\]/i,
|
|
12
|
+
/\[\/INST\]/i,
|
|
13
|
+
/<<\s*SYS\s*>>/i,
|
|
14
|
+
/\bAssistant:\s/i,
|
|
15
|
+
/\bHuman:\s/i,
|
|
16
|
+
/\bUser:\s/i,
|
|
17
|
+
/do\s+not\s+follow\s+(any\s+)?previous/i,
|
|
18
|
+
/forget\s+(all\s+)?(your\s+)?instructions/i,
|
|
19
|
+
/override\s+(all\s+)?instructions/i,
|
|
20
|
+
];
|
|
21
|
+
export function outputInjectionDetectorMiddleware(opts = {}) {
|
|
22
|
+
const mode = opts.mode ?? 'detect';
|
|
23
|
+
return async (ctx, next) => {
|
|
24
|
+
const response = await next();
|
|
25
|
+
let flagged = false;
|
|
26
|
+
for (const pattern of INJECTION_PATTERNS) {
|
|
27
|
+
if (pattern.test(response.text)) {
|
|
28
|
+
flagged = true;
|
|
29
|
+
if (mode === 'mangle') {
|
|
30
|
+
response.text = response.text.replace(new RegExp(pattern.source, pattern.flags + (pattern.flags.includes('g') ? '' : 'g')), '[REDACTED: suspected injection]');
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
if (flagged) {
|
|
35
|
+
log.warn({ agentId: ctx.agentId, tool: ctx.toolName, mode }, 'Injection pattern detected in output');
|
|
36
|
+
ctx.deps.auditLogger.log({
|
|
37
|
+
agent_id: ctx.agentId,
|
|
38
|
+
tool: ctx.toolName,
|
|
39
|
+
args: JSON.stringify(ctx.args),
|
|
40
|
+
result: 'injection_detected',
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
return response;
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
//# sourceMappingURL=output-injection-detector.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"output-injection-detector.js","sourceRoot":"","sources":["../../../src/middleware/post/output-injection-detector.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAEnD,MAAM,GAAG,GAAG,WAAW,CAAC,qBAAqB,CAAC,CAAC;AAE/C,MAAM,kBAAkB,GAAG;IACzB,4CAA4C;IAC5C,yCAAyC;IACzC,gCAAgC;IAChC,6BAA6B;IAC7B,0BAA0B;IAC1B,eAAe;IACf,iBAAiB;IACjB,WAAW;IACX,aAAa;IACb,gBAAgB;IAChB,iBAAiB;IACjB,aAAa;IACb,YAAY;IACZ,wCAAwC;IACxC,2CAA2C;IAC3C,mCAAmC;CACpC,CAAC;AAQF,MAAM,UAAU,iCAAiC,CAC/C,OAAuC,EAAE;IAEzC,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,IAAI,QAAQ,CAAC;IAEnC,OAAO,KAAK,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE;QACzB,MAAM,QAAQ,GAAG,MAAM,IAAI,EAAE,CAAC;QAE9B,IAAI,OAAO,GAAG,KAAK,CAAC;QACpB,KAAK,MAAM,OAAO,IAAI,kBAAkB,EAAE,CAAC;YACzC,IAAI,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;gBAChC,OAAO,GAAG,IAAI,CAAC;gBACf,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;oBACtB,QAAQ,CAAC,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC,OAAO,CACnC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,GAAG,CAAC,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,EACpF,iCAAiC,CAClC,CAAC;gBACJ,CAAC;YACH,CAAC;QACH,CAAC;QAED,IAAI,OAAO,EAAE,CAAC;YACZ,GAAG,CAAC,IAAI,CACN,EAAE,OAAO,EAAE,GAAG,CAAC,OAAO,EAAE,IAAI,EAAE,GAAG,CAAC,QAAQ,EAAE,IAAI,EAAE,EAClD,sCAAsC,CACvC,CAAC;YACF,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC;gBACvB,QAAQ,EAAE,GAAG,CAAC,OAAO;gBACrB,IAAI,EAAE,GAAG,CAAC,QAAQ;gBAClB,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC;gBAC9B,MAAM,EAAE,oBAAoB;aAC7B,CAAC,CAAC;QACL,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import type { Middleware } from '../types.js';
|
|
2
|
+
export interface OutputSizeLimiterOptions {
|
|
3
|
+
max_lines?: number;
|
|
4
|
+
max_chars?: number;
|
|
5
|
+
}
|
|
6
|
+
export declare function outputSizeLimiterMiddleware(opts?: OutputSizeLimiterOptions): Middleware;
|
|
7
|
+
//# sourceMappingURL=output-size-limiter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"output-size-limiter.d.ts","sourceRoot":"","sources":["../../../src/middleware/post/output-size-limiter.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAK9C,MAAM,WAAW,wBAAwB;IACvC,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAKD,wBAAgB,2BAA2B,CAAC,IAAI,GAAE,wBAA6B,GAAG,UAAU,CA2C3F"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import { writeFile, mkdir } from 'fs/promises';
|
|
2
|
+
import { join } from 'path';
|
|
3
|
+
import { tmpdir } from 'os';
|
|
4
|
+
import { childLogger } from '../../util/logger.js';
|
|
5
|
+
const log = childLogger('mw:output-size-limiter');
|
|
6
|
+
const DEFAULT_MAX_LINES = 200;
|
|
7
|
+
const DEFAULT_MAX_CHARS = 30_000;
|
|
8
|
+
export function outputSizeLimiterMiddleware(opts = {}) {
|
|
9
|
+
const maxLines = opts.max_lines ?? DEFAULT_MAX_LINES;
|
|
10
|
+
const maxChars = opts.max_chars ?? DEFAULT_MAX_CHARS;
|
|
11
|
+
return async (ctx, next) => {
|
|
12
|
+
const response = await next();
|
|
13
|
+
const text = response.text;
|
|
14
|
+
const lines = text.split('\n');
|
|
15
|
+
const needsLineTruncation = lines.length > maxLines;
|
|
16
|
+
const needsCharTruncation = text.length > maxChars;
|
|
17
|
+
if (!needsLineTruncation && !needsCharTruncation)
|
|
18
|
+
return response;
|
|
19
|
+
// Write full output to file
|
|
20
|
+
const safeAgentId = ctx.agentId.replace(/[^a-zA-Z0-9_-]/g, '_');
|
|
21
|
+
const dir = join(tmpdir(), 'airlock', safeAgentId);
|
|
22
|
+
const filePath = join(dir, `${ctx.callId}.txt`);
|
|
23
|
+
try {
|
|
24
|
+
await mkdir(dir, { recursive: true });
|
|
25
|
+
await writeFile(filePath, text, 'utf-8');
|
|
26
|
+
response.fullOutputPath = filePath;
|
|
27
|
+
}
|
|
28
|
+
catch (err) {
|
|
29
|
+
log.warn({ err, filePath }, 'Failed to write full output to file');
|
|
30
|
+
}
|
|
31
|
+
// Truncate
|
|
32
|
+
let truncated;
|
|
33
|
+
if (needsLineTruncation) {
|
|
34
|
+
truncated = lines.slice(0, maxLines).join('\n');
|
|
35
|
+
}
|
|
36
|
+
else {
|
|
37
|
+
truncated = text.slice(0, maxChars);
|
|
38
|
+
}
|
|
39
|
+
const totalLines = lines.length;
|
|
40
|
+
const totalChars = text.length;
|
|
41
|
+
truncated += `\n\n[Truncated: ${totalLines} lines / ${totalChars} chars total. Full output: ${filePath}. Use exec/run with cat/grep to read.]`;
|
|
42
|
+
response.text = truncated;
|
|
43
|
+
response.truncated = true;
|
|
44
|
+
return response;
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
//# sourceMappingURL=output-size-limiter.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"output-size-limiter.js","sourceRoot":"","sources":["../../../src/middleware/post/output-size-limiter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,aAAa,CAAC;AAC/C,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,MAAM,EAAE,MAAM,IAAI,CAAC;AAE5B,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAEnD,MAAM,GAAG,GAAG,WAAW,CAAC,wBAAwB,CAAC,CAAC;AAOlD,MAAM,iBAAiB,GAAG,GAAG,CAAC;AAC9B,MAAM,iBAAiB,GAAG,MAAM,CAAC;AAEjC,MAAM,UAAU,2BAA2B,CAAC,OAAiC,EAAE;IAC7E,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,IAAI,iBAAiB,CAAC;IACrD,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,IAAI,iBAAiB,CAAC;IAErD,OAAO,KAAK,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE;QACzB,MAAM,QAAQ,GAAG,MAAM,IAAI,EAAE,CAAC;QAC9B,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC;QAE3B,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAC/B,MAAM,mBAAmB,GAAG,KAAK,CAAC,MAAM,GAAG,QAAQ,CAAC;QACpD,MAAM,mBAAmB,GAAG,IAAI,CAAC,MAAM,GAAG,QAAQ,CAAC;QAEnD,IAAI,CAAC,mBAAmB,IAAI,CAAC,mBAAmB;YAAE,OAAO,QAAQ,CAAC;QAElE,4BAA4B;QAC5B,MAAM,WAAW,GAAG,GAAG,CAAC,OAAO,CAAC,OAAO,CAAC,iBAAiB,EAAE,GAAG,CAAC,CAAC;QAChE,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,SAAS,EAAE,WAAW,CAAC,CAAC;QACnD,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,GAAG,CAAC,MAAM,MAAM,CAAC,CAAC;QAEhD,IAAI,CAAC;YACH,MAAM,KAAK,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;YACtC,MAAM,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,OAAO,CAAC,CAAC;YACzC,QAAQ,CAAC,cAAc,GAAG,QAAQ,CAAC;QACrC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,GAAG,CAAC,IAAI,CAAC,EAAE,GAAG,EAAE,QAAQ,EAAE,EAAE,qCAAqC,CAAC,CAAC;QACrE,CAAC;QAED,WAAW;QACX,IAAI,SAAiB,CAAC;QACtB,IAAI,mBAAmB,EAAE,CAAC;YACxB,SAAS,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAClD,CAAC;aAAM,CAAC;YACN,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC;QACtC,CAAC;QAED,MAAM,UAAU,GAAG,KAAK,CAAC,MAAM,CAAC;QAChC,MAAM,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC;QAC/B,SAAS,IAAI,mBAAmB,UAAU,YAAY,UAAU,8BAA8B,QAAQ,wCAAwC,CAAC;QAE/I,QAAQ,CAAC,IAAI,GAAG,SAAS,CAAC;QAC1B,QAAQ,CAAC,SAAS,GAAG,IAAI,CAAC;QAC1B,OAAO,QAAQ,CAAC;IAClB,CAAC,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type { Middleware } from '../types.js';
|
|
2
|
+
export interface OutputSummarizerOptions {
|
|
3
|
+
/** Minimum character length before summarization kicks in */
|
|
4
|
+
threshold_chars?: number;
|
|
5
|
+
/** Model ID to use (provider-specific, e.g. 'claude-haiku-4-5-20251001', 'gpt-4o-mini') */
|
|
6
|
+
model: string;
|
|
7
|
+
}
|
|
8
|
+
/**
|
|
9
|
+
* Summarizes large tool outputs using the Vercel AI SDK.
|
|
10
|
+
* Requires `ai` package and a provider to be configured.
|
|
11
|
+
* The caller must pass a model string that the AI SDK can resolve.
|
|
12
|
+
* Falls back gracefully if the SDK or model is unavailable.
|
|
13
|
+
*/
|
|
14
|
+
export declare function outputSummarizerMiddleware(opts: OutputSummarizerOptions): Middleware;
|
|
15
|
+
//# sourceMappingURL=output-summarizer.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"output-summarizer.d.ts","sourceRoot":"","sources":["../../../src/middleware/post/output-summarizer.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAK9C,MAAM,WAAW,uBAAuB;IACtC,6DAA6D;IAC7D,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,2FAA2F;IAC3F,KAAK,EAAE,MAAM,CAAC;CACf;AAID;;;;;GAKG;AACH,wBAAgB,0BAA0B,CAAC,IAAI,EAAE,uBAAuB,GAAG,UAAU,CAgCpF"}
|