tracelattice 1.2.5
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 +24 -0
- package/README.md +112 -0
- package/dist/ServerConfig.d.ts +229 -0
- package/dist/ServerConfig.d.ts.map +1 -0
- package/dist/ServerConfig.js +121 -0
- package/dist/ServerConfig.js.map +1 -0
- package/dist/__tests__/base-registry.test.d.ts +2 -0
- package/dist/__tests__/base-registry.test.d.ts.map +1 -0
- package/dist/__tests__/base-transport-cov.test.d.ts +2 -0
- package/dist/__tests__/base-transport-cov.test.d.ts.map +1 -0
- package/dist/__tests__/base-transport.test.d.ts +2 -0
- package/dist/__tests__/base-transport.test.d.ts.map +1 -0
- package/dist/__tests__/config-loader.test.d.ts +2 -0
- package/dist/__tests__/config-loader.test.d.ts.map +1 -0
- package/dist/__tests__/connection-pool-cov.test.d.ts +2 -0
- package/dist/__tests__/connection-pool-cov.test.d.ts.map +1 -0
- package/dist/__tests__/connection-pool.test.d.ts +2 -0
- package/dist/__tests__/connection-pool.test.d.ts.map +1 -0
- package/dist/__tests__/container.test.d.ts +2 -0
- package/dist/__tests__/container.test.d.ts.map +1 -0
- package/dist/__tests__/crud.test.d.ts +2 -0
- package/dist/__tests__/crud.test.d.ts.map +1 -0
- package/dist/__tests__/discovery-cache.test.d.ts +2 -0
- package/dist/__tests__/discovery-cache.test.d.ts.map +1 -0
- package/dist/__tests__/errors.test.d.ts +2 -0
- package/dist/__tests__/errors.test.d.ts.map +1 -0
- package/dist/__tests__/factories.test.d.ts +2 -0
- package/dist/__tests__/factories.test.d.ts.map +1 -0
- package/dist/__tests__/health-checker-cov.test.d.ts +2 -0
- package/dist/__tests__/health-checker-cov.test.d.ts.map +1 -0
- package/dist/__tests__/health-checker.test.d.ts +2 -0
- package/dist/__tests__/health-checker.test.d.ts.map +1 -0
- package/dist/__tests__/helpers/factories.d.ts +36 -0
- package/dist/__tests__/helpers/factories.d.ts.map +1 -0
- package/dist/__tests__/helpers/index.d.ts +3 -0
- package/dist/__tests__/helpers/index.d.ts.map +1 -0
- package/dist/__tests__/helpers/timers.d.ts +4 -0
- package/dist/__tests__/helpers/timers.d.ts.map +1 -0
- package/dist/__tests__/history-manager.test.d.ts +2 -0
- package/dist/__tests__/history-manager.test.d.ts.map +1 -0
- package/dist/__tests__/http-helpers-cov.test.d.ts +2 -0
- package/dist/__tests__/http-helpers-cov.test.d.ts.map +1 -0
- package/dist/__tests__/http-transport-cov.test.d.ts +2 -0
- package/dist/__tests__/http-transport-cov.test.d.ts.map +1 -0
- package/dist/__tests__/http-transport.test.d.ts +2 -0
- package/dist/__tests__/http-transport.test.d.ts.map +1 -0
- package/dist/__tests__/input-normalizer.test.d.ts +8 -0
- package/dist/__tests__/input-normalizer.test.d.ts.map +1 -0
- package/dist/__tests__/integration.test.d.ts +2 -0
- package/dist/__tests__/integration.test.d.ts.map +1 -0
- package/dist/__tests__/lib-server.test.d.ts +2 -0
- package/dist/__tests__/lib-server.test.d.ts.map +1 -0
- package/dist/__tests__/memory-persistence.test.d.ts +2 -0
- package/dist/__tests__/memory-persistence.test.d.ts.map +1 -0
- package/dist/__tests__/metrics-integration.test.d.ts +2 -0
- package/dist/__tests__/metrics-integration.test.d.ts.map +1 -0
- package/dist/__tests__/persistence.test.d.ts +2 -0
- package/dist/__tests__/persistence.test.d.ts.map +1 -0
- package/dist/__tests__/reasoning-integration.test.d.ts +11 -0
- package/dist/__tests__/reasoning-integration.test.d.ts.map +1 -0
- package/dist/__tests__/reasoning-types.test.d.ts +2 -0
- package/dist/__tests__/reasoning-types.test.d.ts.map +1 -0
- package/dist/__tests__/request-context.test.d.ts +2 -0
- package/dist/__tests__/request-context.test.d.ts.map +1 -0
- package/dist/__tests__/sanitize.test.d.ts +2 -0
- package/dist/__tests__/sanitize.test.d.ts.map +1 -0
- package/dist/__tests__/schema.test.d.ts +2 -0
- package/dist/__tests__/schema.test.d.ts.map +1 -0
- package/dist/__tests__/sequentialthinking-tools.test.d.ts +2 -0
- package/dist/__tests__/sequentialthinking-tools.test.d.ts.map +1 -0
- package/dist/__tests__/server-config.test.d.ts +2 -0
- package/dist/__tests__/server-config.test.d.ts.map +1 -0
- package/dist/__tests__/skill-discovery.test.d.ts +2 -0
- package/dist/__tests__/skill-discovery.test.d.ts.map +1 -0
- package/dist/__tests__/skill-registry.test.d.ts +2 -0
- package/dist/__tests__/skill-registry.test.d.ts.map +1 -0
- package/dist/__tests__/skill-watcher.test.d.ts +2 -0
- package/dist/__tests__/skill-watcher.test.d.ts.map +1 -0
- package/dist/__tests__/sqlite-persistence.test.d.ts +2 -0
- package/dist/__tests__/sqlite-persistence.test.d.ts.map +1 -0
- package/dist/__tests__/sse-transport-cov.test.d.ts +2 -0
- package/dist/__tests__/sse-transport-cov.test.d.ts.map +1 -0
- package/dist/__tests__/sse-transport.test.d.ts +2 -0
- package/dist/__tests__/sse-transport.test.d.ts.map +1 -0
- package/dist/__tests__/streamable-http-cov.test.d.ts +2 -0
- package/dist/__tests__/streamable-http-cov.test.d.ts.map +1 -0
- package/dist/__tests__/streamable-http-transport.test.d.ts +2 -0
- package/dist/__tests__/streamable-http-transport.test.d.ts.map +1 -0
- package/dist/__tests__/structured-logger.test.d.ts +2 -0
- package/dist/__tests__/structured-logger.test.d.ts.map +1 -0
- package/dist/__tests__/thought-evaluator.test.d.ts +2 -0
- package/dist/__tests__/thought-evaluator.test.d.ts.map +1 -0
- package/dist/__tests__/thought-formatter.test.d.ts +2 -0
- package/dist/__tests__/thought-formatter.test.d.ts.map +1 -0
- package/dist/__tests__/thought-processor.test.d.ts +8 -0
- package/dist/__tests__/thought-processor.test.d.ts.map +1 -0
- package/dist/__tests__/tool-registry-cov.test.d.ts +2 -0
- package/dist/__tests__/tool-registry-cov.test.d.ts.map +1 -0
- package/dist/__tests__/tool-registry.test.d.ts +2 -0
- package/dist/__tests__/tool-registry.test.d.ts.map +1 -0
- package/dist/__tests__/tool-watcher.test.d.ts +2 -0
- package/dist/__tests__/tool-watcher.test.d.ts.map +1 -0
- package/dist/__tests__/worker-manager-cov.test.d.ts +2 -0
- package/dist/__tests__/worker-manager-cov.test.d.ts.map +1 -0
- package/dist/__tests__/worker-manager.test.d.ts +2 -0
- package/dist/__tests__/worker-manager.test.d.ts.map +1 -0
- package/dist/cache/DiscoveryCache.d.ts +269 -0
- package/dist/cache/DiscoveryCache.d.ts.map +1 -0
- package/dist/cache/DiscoveryCache.js +100 -0
- package/dist/cache/DiscoveryCache.js.map +1 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +114 -0
- package/dist/cli.js.map +1 -0
- package/dist/cluster/WorkerManager.d.ts +166 -0
- package/dist/cluster/WorkerManager.d.ts.map +1 -0
- package/dist/cluster/WorkerManager.js +202 -0
- package/dist/cluster/WorkerManager.js.map +1 -0
- package/dist/cluster/worker.d.ts +11 -0
- package/dist/cluster/worker.d.ts.map +1 -0
- package/dist/cluster/worker.js +36 -0
- package/dist/cluster/worker.js.map +1 -0
- package/dist/config/ConfigLoader.d.ts +224 -0
- package/dist/config/ConfigLoader.d.ts.map +1 -0
- package/dist/config/ConfigLoader.js +85 -0
- package/dist/config/ConfigLoader.js.map +1 -0
- package/dist/context/RequestContext.d.ts +61 -0
- package/dist/context/RequestContext.d.ts.map +1 -0
- package/dist/context/RequestContext.js +17 -0
- package/dist/context/RequestContext.js.map +1 -0
- package/dist/contracts/index.d.ts +10 -0
- package/dist/contracts/index.d.ts.map +1 -0
- package/dist/contracts/index.js +1 -0
- package/dist/contracts/interfaces.d.ts +107 -0
- package/dist/contracts/interfaces.d.ts.map +1 -0
- package/dist/contracts/interfaces.js +1 -0
- package/dist/core/HistoryManager.d.ts +514 -0
- package/dist/core/HistoryManager.d.ts.map +1 -0
- package/dist/core/HistoryManager.js +331 -0
- package/dist/core/HistoryManager.js.map +1 -0
- package/dist/core/IHistoryManager.d.ts +100 -0
- package/dist/core/IHistoryManager.d.ts.map +1 -0
- package/dist/core/IHistoryManager.js +1 -0
- package/dist/core/InputNormalizer.d.ts +139 -0
- package/dist/core/InputNormalizer.d.ts.map +1 -0
- package/dist/core/InputNormalizer.js +101 -0
- package/dist/core/InputNormalizer.js.map +1 -0
- package/dist/core/ThoughtEvaluator.d.ts +127 -0
- package/dist/core/ThoughtEvaluator.d.ts.map +1 -0
- package/dist/core/ThoughtEvaluator.js +346 -0
- package/dist/core/ThoughtEvaluator.js.map +1 -0
- package/dist/core/ThoughtFormatter.d.ts +133 -0
- package/dist/core/ThoughtFormatter.d.ts.map +1 -0
- package/dist/core/ThoughtFormatter.js +70 -0
- package/dist/core/ThoughtFormatter.js.map +1 -0
- package/dist/core/ThoughtProcessor.d.ts +218 -0
- package/dist/core/ThoughtProcessor.d.ts.map +1 -0
- package/dist/core/ThoughtProcessor.js +205 -0
- package/dist/core/ThoughtProcessor.js.map +1 -0
- package/dist/core/reasoning.d.ts +169 -0
- package/dist/core/reasoning.d.ts.map +1 -0
- package/dist/core/reasoning.js +1 -0
- package/dist/core/step.d.ts +45 -0
- package/dist/core/step.d.ts.map +1 -0
- package/dist/core/step.js +1 -0
- package/dist/core/thought.d.ts +190 -0
- package/dist/core/thought.d.ts.map +1 -0
- package/dist/core/thought.js +1 -0
- package/dist/di/Container.d.ts +226 -0
- package/dist/di/Container.d.ts.map +1 -0
- package/dist/di/Container.js +96 -0
- package/dist/di/Container.js.map +1 -0
- package/dist/di/ServiceRegistry.d.ts +32 -0
- package/dist/di/ServiceRegistry.d.ts.map +1 -0
- package/dist/di/ServiceRegistry.js +1 -0
- package/dist/errors.d.ts +482 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +108 -0
- package/dist/errors.js.map +1 -0
- package/dist/health/HealthChecker.d.ts +73 -0
- package/dist/health/HealthChecker.d.ts.map +1 -0
- package/dist/health/HealthChecker.js +69 -0
- package/dist/health/HealthChecker.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +1 -0
- package/dist/lib.d.ts +205 -0
- package/dist/lib.d.ts.map +1 -0
- package/dist/lib.js +219 -0
- package/dist/lib.js.map +1 -0
- package/dist/logger/NullLogger.d.ts +154 -0
- package/dist/logger/NullLogger.d.ts.map +1 -0
- package/dist/logger/NullLogger.js +24 -0
- package/dist/logger/NullLogger.js.map +1 -0
- package/dist/logger/StructuredLogger.d.ts +327 -0
- package/dist/logger/StructuredLogger.d.ts.map +1 -0
- package/dist/logger/StructuredLogger.js +72 -0
- package/dist/logger/StructuredLogger.js.map +1 -0
- package/dist/metrics/__tests__/metrics.test.d.ts +2 -0
- package/dist/metrics/__tests__/metrics.test.d.ts.map +1 -0
- package/dist/metrics/metrics.impl.d.ts +252 -0
- package/dist/metrics/metrics.impl.d.ts.map +1 -0
- package/dist/metrics/metrics.impl.js +197 -0
- package/dist/metrics/metrics.impl.js.map +1 -0
- package/dist/persistence/FilePersistence.d.ts +66 -0
- package/dist/persistence/FilePersistence.d.ts.map +1 -0
- package/dist/persistence/FilePersistence.js +132 -0
- package/dist/persistence/FilePersistence.js.map +1 -0
- package/dist/persistence/MemoryPersistence.d.ts +68 -0
- package/dist/persistence/MemoryPersistence.d.ts.map +1 -0
- package/dist/persistence/MemoryPersistence.js +51 -0
- package/dist/persistence/MemoryPersistence.js.map +1 -0
- package/dist/persistence/PersistenceBackend.d.ts +69 -0
- package/dist/persistence/PersistenceBackend.d.ts.map +1 -0
- package/dist/persistence/PersistenceBackend.js +1 -0
- package/dist/persistence/PersistenceFactory.d.ts +21 -0
- package/dist/persistence/PersistenceFactory.d.ts.map +1 -0
- package/dist/persistence/PersistenceFactory.js +25 -0
- package/dist/persistence/PersistenceFactory.js.map +1 -0
- package/dist/persistence/SqlitePersistence.d.ts +60 -0
- package/dist/persistence/SqlitePersistence.d.ts.map +1 -0
- package/dist/persistence/SqlitePersistence.js +136 -0
- package/dist/persistence/SqlitePersistence.js.map +1 -0
- package/dist/pool/ConnectionPool.d.ts +215 -0
- package/dist/pool/ConnectionPool.d.ts.map +1 -0
- package/dist/pool/ConnectionPool.js +187 -0
- package/dist/pool/ConnectionPool.js.map +1 -0
- package/dist/registry/BaseRegistry.d.ts +203 -0
- package/dist/registry/BaseRegistry.d.ts.map +1 -0
- package/dist/registry/BaseRegistry.js +165 -0
- package/dist/registry/BaseRegistry.js.map +1 -0
- package/dist/registry/SkillRegistry.d.ts +69 -0
- package/dist/registry/SkillRegistry.d.ts.map +1 -0
- package/dist/registry/SkillRegistry.js +88 -0
- package/dist/registry/SkillRegistry.js.map +1 -0
- package/dist/registry/ToolRegistry.d.ts +69 -0
- package/dist/registry/ToolRegistry.d.ts.map +1 -0
- package/dist/registry/ToolRegistry.js +93 -0
- package/dist/registry/ToolRegistry.js.map +1 -0
- package/dist/sanitize.d.ts +63 -0
- package/dist/sanitize.d.ts.map +1 -0
- package/dist/sanitize.js +14 -0
- package/dist/sanitize.js.map +1 -0
- package/dist/schema.d.ts +531 -0
- package/dist/schema.d.ts.map +1 -0
- package/dist/schema.js +204 -0
- package/dist/schema.js.map +1 -0
- package/dist/telemetry/Telemetry.d.ts +36 -0
- package/dist/telemetry/Telemetry.d.ts.map +1 -0
- package/dist/telemetry/Telemetry.js +68 -0
- package/dist/telemetry/Telemetry.js.map +1 -0
- package/dist/telemetry/__tests__/Telemetry.test.d.ts +2 -0
- package/dist/telemetry/__tests__/Telemetry.test.d.ts.map +1 -0
- package/dist/transport/BaseTransport.d.ts +184 -0
- package/dist/transport/BaseTransport.d.ts.map +1 -0
- package/dist/transport/BaseTransport.js +200 -0
- package/dist/transport/BaseTransport.js.map +1 -0
- package/dist/transport/HttpHelpers.d.ts +60 -0
- package/dist/transport/HttpHelpers.d.ts.map +1 -0
- package/dist/transport/HttpHelpers.js +50 -0
- package/dist/transport/HttpHelpers.js.map +1 -0
- package/dist/transport/HttpTransport.d.ts +134 -0
- package/dist/transport/HttpTransport.d.ts.map +1 -0
- package/dist/transport/HttpTransport.js +175 -0
- package/dist/transport/HttpTransport.js.map +1 -0
- package/dist/transport/SseTransport.d.ts +133 -0
- package/dist/transport/SseTransport.d.ts.map +1 -0
- package/dist/transport/SseTransport.js +318 -0
- package/dist/transport/SseTransport.js.map +1 -0
- package/dist/transport/StreamableHttpTransport.d.ts +224 -0
- package/dist/transport/StreamableHttpTransport.d.ts.map +1 -0
- package/dist/transport/StreamableHttpTransport.js +407 -0
- package/dist/transport/StreamableHttpTransport.js.map +1 -0
- package/dist/types/disposable.d.ts +22 -0
- package/dist/types/disposable.d.ts.map +1 -0
- package/dist/types/disposable.js +1 -0
- package/dist/types/server-config.d.ts +32 -0
- package/dist/types/server-config.d.ts.map +1 -0
- package/dist/types/server-config.js +1 -0
- package/dist/types/skill.d.ts +69 -0
- package/dist/types/skill.d.ts.map +1 -0
- package/dist/types/skill.js +1 -0
- package/dist/types/tool.d.ts +68 -0
- package/dist/types/tool.d.ts.map +1 -0
- package/dist/types/tool.js +1 -0
- package/dist/watchers/SkillWatcher.d.ts +132 -0
- package/dist/watchers/SkillWatcher.d.ts.map +1 -0
- package/dist/watchers/SkillWatcher.js +73 -0
- package/dist/watchers/SkillWatcher.js.map +1 -0
- package/dist/watchers/ToolWatcher.d.ts +109 -0
- package/dist/watchers/ToolWatcher.d.ts.map +1 -0
- package/dist/watchers/ToolWatcher.js +71 -0
- package/dist/watchers/ToolWatcher.js.map +1 -0
- package/package.json +95 -0
|
@@ -0,0 +1,407 @@
|
|
|
1
|
+
import { randomUUID } from "node:crypto";
|
|
2
|
+
import { createServer } from "node:http";
|
|
3
|
+
import { safeParse } from "valibot";
|
|
4
|
+
import { JsonRpcRequestSchema } from "../schema.js";
|
|
5
|
+
import { BaseTransport } from "./BaseTransport.js";
|
|
6
|
+
class StreamableHttpTransport extends BaseTransport {
|
|
7
|
+
_server = null;
|
|
8
|
+
_mcpServer = null;
|
|
9
|
+
_path;
|
|
10
|
+
_stateful;
|
|
11
|
+
_sessionIdGenerator;
|
|
12
|
+
_sessions = new Map();
|
|
13
|
+
_requestCount = 0;
|
|
14
|
+
_activeRequests = 0;
|
|
15
|
+
_bodySizeLimitEnabled;
|
|
16
|
+
_maxBodySize;
|
|
17
|
+
_requestTimeout;
|
|
18
|
+
_metrics;
|
|
19
|
+
_metricsProvider;
|
|
20
|
+
constructor(options = {}){
|
|
21
|
+
super(options);
|
|
22
|
+
this._path = options.path ?? '/mcp';
|
|
23
|
+
this._stateful = options.stateful ?? true;
|
|
24
|
+
this._sessionIdGenerator = options.sessionIdGenerator ?? (()=>randomUUID());
|
|
25
|
+
this._bodySizeLimitEnabled = options.enableBodySizeLimit ?? true;
|
|
26
|
+
this._maxBodySize = options.maxBodySize ?? 10485760;
|
|
27
|
+
this._requestTimeout = options.requestTimeout ?? 30000;
|
|
28
|
+
this._metrics = options.metrics;
|
|
29
|
+
this._metricsProvider = options.metricsProvider ?? null;
|
|
30
|
+
}
|
|
31
|
+
get clientCount() {
|
|
32
|
+
return this._stateful ? this._sessions.size : this._activeRequests;
|
|
33
|
+
}
|
|
34
|
+
get requestCount() {
|
|
35
|
+
return this._requestCount;
|
|
36
|
+
}
|
|
37
|
+
async connect(mcpServer) {
|
|
38
|
+
this._mcpServer = mcpServer;
|
|
39
|
+
this._server = createServer((req, res)=>this._handleRequest(req, res));
|
|
40
|
+
return new Promise((resolve)=>{
|
|
41
|
+
this._server.listen(this._port, this._host, ()=>{
|
|
42
|
+
this.log('info', `Streamable HTTP transport listening on http://${this._host}:${this._port}`);
|
|
43
|
+
resolve();
|
|
44
|
+
});
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
async _handleRequest(req, res) {
|
|
48
|
+
const startTime = Date.now();
|
|
49
|
+
this._metrics?.counter('streamable_http_requests_total', 1, {}, 'Total Streamable HTTP transport requests');
|
|
50
|
+
res.once('finish', ()=>{
|
|
51
|
+
const durationSeconds = (Date.now() - startTime) / 1000;
|
|
52
|
+
this._metrics?.histogram('streamable_http_request_duration_seconds', durationSeconds, {});
|
|
53
|
+
});
|
|
54
|
+
if (!this.validateHostHeader(req)) return void this._sendJsonRpcError(res, 403, -32000, 'Forbidden - invalid host header');
|
|
55
|
+
if (this.isShuttingDown()) return void this._sendJsonRpcError(res, 503, -32603, 'Server is shutting down');
|
|
56
|
+
const clientIp = this.getClientIp(req);
|
|
57
|
+
if (this.checkRateLimit(clientIp)) {
|
|
58
|
+
res.setHeader('Retry-After', '60');
|
|
59
|
+
this._sendJsonRpcError(res, 429, -32000, 'Too many requests');
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
if (!this.validateCorsOrigin(req)) return void this._sendJsonRpcError(res, 403, -32000, 'Forbidden - invalid origin');
|
|
63
|
+
this.setCorsHeaders(res);
|
|
64
|
+
if ('OPTIONS' === req.method) {
|
|
65
|
+
res.setHeader('Access-Control-Allow-Headers', 'Content-Type, Mcp-Session-Id');
|
|
66
|
+
res.writeHead(204);
|
|
67
|
+
res.end();
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
const urlPath = req.url?.split('?')[0] ?? '/';
|
|
71
|
+
if ('GET' === req.method && '/metrics' === urlPath) return void this._handleMetrics(res);
|
|
72
|
+
if ('GET' === req.method && '/health' === urlPath) return void this._handleHealthCheck(res);
|
|
73
|
+
if ('GET' === req.method && '/ready' === urlPath) return void await this._handleReadinessCheck(res);
|
|
74
|
+
if (urlPath === this._path) {
|
|
75
|
+
if ('POST' === req.method) return void await this._handleMcpPost(req, res);
|
|
76
|
+
if ('GET' === req.method) return void this._handleMcpGet(req, res);
|
|
77
|
+
res.writeHead(405, {
|
|
78
|
+
'Content-Type': 'application/json',
|
|
79
|
+
Allow: 'GET, POST'
|
|
80
|
+
});
|
|
81
|
+
res.end(JSON.stringify({
|
|
82
|
+
jsonrpc: '2.0',
|
|
83
|
+
id: null,
|
|
84
|
+
error: {
|
|
85
|
+
code: -32601,
|
|
86
|
+
message: 'Method not allowed'
|
|
87
|
+
}
|
|
88
|
+
}));
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
this._sendJsonRpcError(res, 404, -32601, 'Not Found');
|
|
92
|
+
}
|
|
93
|
+
_sendJsonRpcError(res, statusCode, code, message, id = null, extra) {
|
|
94
|
+
const error = {
|
|
95
|
+
code,
|
|
96
|
+
message
|
|
97
|
+
};
|
|
98
|
+
if (extra) Object.assign(error, extra);
|
|
99
|
+
res.writeHead(statusCode, {
|
|
100
|
+
'Content-Type': 'application/json'
|
|
101
|
+
});
|
|
102
|
+
res.end(JSON.stringify({
|
|
103
|
+
jsonrpc: '2.0',
|
|
104
|
+
id,
|
|
105
|
+
error
|
|
106
|
+
}));
|
|
107
|
+
}
|
|
108
|
+
async _handleMcpPost(req, res) {
|
|
109
|
+
this._requestCount++;
|
|
110
|
+
this._activeRequests++;
|
|
111
|
+
const timeout = setTimeout(()=>{
|
|
112
|
+
this._activeRequests--;
|
|
113
|
+
this._sendJsonRpcError(res, 500, -32603, 'Request timeout');
|
|
114
|
+
}, this._requestTimeout);
|
|
115
|
+
try {
|
|
116
|
+
const body = await this._readRequestBody(req);
|
|
117
|
+
if (null === body) {
|
|
118
|
+
clearTimeout(timeout);
|
|
119
|
+
this._activeRequests--;
|
|
120
|
+
this._sendJsonRpcError(res, 413, -32000, 'Request body too large');
|
|
121
|
+
return;
|
|
122
|
+
}
|
|
123
|
+
let jsonRpcRequest;
|
|
124
|
+
try {
|
|
125
|
+
jsonRpcRequest = JSON.parse(body);
|
|
126
|
+
} catch {
|
|
127
|
+
clearTimeout(timeout);
|
|
128
|
+
this._activeRequests--;
|
|
129
|
+
this._sendJsonRpcError(res, 200, -32700, 'Parse error');
|
|
130
|
+
return;
|
|
131
|
+
}
|
|
132
|
+
const parseResult = safeParse(JsonRpcRequestSchema, jsonRpcRequest);
|
|
133
|
+
if (!parseResult.success) {
|
|
134
|
+
clearTimeout(timeout);
|
|
135
|
+
this._activeRequests--;
|
|
136
|
+
this._sendJsonRpcError(res, 200, -32600, 'Invalid Request', jsonRpcRequest?.id ?? null, {
|
|
137
|
+
data: parseResult.issues
|
|
138
|
+
});
|
|
139
|
+
return;
|
|
140
|
+
}
|
|
141
|
+
let sessionId;
|
|
142
|
+
if (this._stateful) {
|
|
143
|
+
const sessionResult = this._resolveSession(req, res);
|
|
144
|
+
if (false === sessionResult) {
|
|
145
|
+
clearTimeout(timeout);
|
|
146
|
+
this._activeRequests--;
|
|
147
|
+
return;
|
|
148
|
+
}
|
|
149
|
+
sessionId = sessionResult;
|
|
150
|
+
}
|
|
151
|
+
if (!this._mcpServer) {
|
|
152
|
+
clearTimeout(timeout);
|
|
153
|
+
this._activeRequests--;
|
|
154
|
+
this._sendJsonRpcError(res, 503, -32603, 'Server not ready', jsonRpcRequest?.id ?? null);
|
|
155
|
+
return;
|
|
156
|
+
}
|
|
157
|
+
const response = await this._mcpServer.receive(jsonRpcRequest, {
|
|
158
|
+
sessionInfo: {}
|
|
159
|
+
});
|
|
160
|
+
clearTimeout(timeout);
|
|
161
|
+
this._activeRequests--;
|
|
162
|
+
const responseHeaders = {
|
|
163
|
+
'Content-Type': 'application/json'
|
|
164
|
+
};
|
|
165
|
+
if (sessionId) responseHeaders['Mcp-Session-Id'] = sessionId;
|
|
166
|
+
if (response) {
|
|
167
|
+
res.writeHead(200, responseHeaders);
|
|
168
|
+
res.end(JSON.stringify(response));
|
|
169
|
+
} else {
|
|
170
|
+
if (sessionId) res.setHeader('Mcp-Session-Id', sessionId);
|
|
171
|
+
res.writeHead(202);
|
|
172
|
+
res.end();
|
|
173
|
+
}
|
|
174
|
+
} catch (error) {
|
|
175
|
+
clearTimeout(timeout);
|
|
176
|
+
this._activeRequests--;
|
|
177
|
+
this._sendJsonRpcError(res, 200, -32603, 'Internal error', null, {
|
|
178
|
+
data: error instanceof Error ? error.message : String(error)
|
|
179
|
+
});
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
_handleMcpGet(req, res) {
|
|
183
|
+
if (!this._stateful) {
|
|
184
|
+
res.writeHead(405, {
|
|
185
|
+
'Content-Type': 'application/json',
|
|
186
|
+
Allow: 'POST'
|
|
187
|
+
});
|
|
188
|
+
res.end(JSON.stringify({
|
|
189
|
+
jsonrpc: '2.0',
|
|
190
|
+
id: null,
|
|
191
|
+
error: {
|
|
192
|
+
code: -32601,
|
|
193
|
+
message: 'GET not supported in stateless mode'
|
|
194
|
+
}
|
|
195
|
+
}));
|
|
196
|
+
return;
|
|
197
|
+
}
|
|
198
|
+
const sessionId = this._getSessionIdFromHeader(req);
|
|
199
|
+
if (!sessionId) {
|
|
200
|
+
res.writeHead(400, {
|
|
201
|
+
'Content-Type': 'application/json'
|
|
202
|
+
});
|
|
203
|
+
res.end(JSON.stringify({
|
|
204
|
+
jsonrpc: '2.0',
|
|
205
|
+
id: null,
|
|
206
|
+
error: {
|
|
207
|
+
code: -32600,
|
|
208
|
+
message: 'Missing Mcp-Session-Id header'
|
|
209
|
+
}
|
|
210
|
+
}));
|
|
211
|
+
return;
|
|
212
|
+
}
|
|
213
|
+
const session = this._sessions.get(sessionId);
|
|
214
|
+
if (!session) {
|
|
215
|
+
res.writeHead(404, {
|
|
216
|
+
'Content-Type': 'application/json'
|
|
217
|
+
});
|
|
218
|
+
res.end(JSON.stringify({
|
|
219
|
+
jsonrpc: '2.0',
|
|
220
|
+
id: null,
|
|
221
|
+
error: {
|
|
222
|
+
code: -32001,
|
|
223
|
+
message: 'Session not found'
|
|
224
|
+
}
|
|
225
|
+
}));
|
|
226
|
+
return;
|
|
227
|
+
}
|
|
228
|
+
res.writeHead(200, {
|
|
229
|
+
'Content-Type': 'text/event-stream',
|
|
230
|
+
'Cache-Control': 'no-cache',
|
|
231
|
+
Connection: 'keep-alive',
|
|
232
|
+
'Mcp-Session-Id': sessionId
|
|
233
|
+
});
|
|
234
|
+
this._sendSseEvent(res, 'connected', {
|
|
235
|
+
sessionId,
|
|
236
|
+
timestamp: Date.now()
|
|
237
|
+
});
|
|
238
|
+
session.notificationStreams.add(res);
|
|
239
|
+
session.lastActivityAt = Date.now();
|
|
240
|
+
this._updateSessionMetrics();
|
|
241
|
+
req.on('close', ()=>{
|
|
242
|
+
session.notificationStreams.delete(res);
|
|
243
|
+
this._updateSessionMetrics();
|
|
244
|
+
});
|
|
245
|
+
}
|
|
246
|
+
_resolveSession(req, res) {
|
|
247
|
+
const headerSessionId = this._getSessionIdFromHeader(req);
|
|
248
|
+
if (headerSessionId) {
|
|
249
|
+
if (!this.validateSessionId(headerSessionId)) {
|
|
250
|
+
res.writeHead(400, {
|
|
251
|
+
'Content-Type': 'application/json'
|
|
252
|
+
});
|
|
253
|
+
res.end(JSON.stringify({
|
|
254
|
+
jsonrpc: '2.0',
|
|
255
|
+
id: null,
|
|
256
|
+
error: {
|
|
257
|
+
code: -32600,
|
|
258
|
+
message: 'Invalid Mcp-Session-Id format'
|
|
259
|
+
}
|
|
260
|
+
}));
|
|
261
|
+
return false;
|
|
262
|
+
}
|
|
263
|
+
const session = this._sessions.get(headerSessionId);
|
|
264
|
+
if (session) {
|
|
265
|
+
session.lastActivityAt = Date.now();
|
|
266
|
+
return headerSessionId;
|
|
267
|
+
}
|
|
268
|
+
res.writeHead(404, {
|
|
269
|
+
'Content-Type': 'application/json'
|
|
270
|
+
});
|
|
271
|
+
res.end(JSON.stringify({
|
|
272
|
+
jsonrpc: '2.0',
|
|
273
|
+
id: null,
|
|
274
|
+
error: {
|
|
275
|
+
code: -32001,
|
|
276
|
+
message: 'Session not found'
|
|
277
|
+
}
|
|
278
|
+
}));
|
|
279
|
+
return false;
|
|
280
|
+
}
|
|
281
|
+
const newSessionId = this._sessionIdGenerator();
|
|
282
|
+
const sessionState = {
|
|
283
|
+
id: newSessionId,
|
|
284
|
+
createdAt: Date.now(),
|
|
285
|
+
lastActivityAt: Date.now(),
|
|
286
|
+
notificationStreams: new Set()
|
|
287
|
+
};
|
|
288
|
+
this._sessions.set(newSessionId, sessionState);
|
|
289
|
+
this.log('info', `New session created: ${newSessionId}`);
|
|
290
|
+
this._updateSessionMetrics();
|
|
291
|
+
return newSessionId;
|
|
292
|
+
}
|
|
293
|
+
_getSessionIdFromHeader(req) {
|
|
294
|
+
const value = req.headers['mcp-session-id'];
|
|
295
|
+
if ('string' == typeof value && value.length > 0) return value;
|
|
296
|
+
}
|
|
297
|
+
async _readRequestBody(req) {
|
|
298
|
+
let body = '';
|
|
299
|
+
let bodySize = 0;
|
|
300
|
+
for await (const chunk of req){
|
|
301
|
+
const chunkStr = 'string' == typeof chunk ? chunk : chunk.toString();
|
|
302
|
+
bodySize += chunkStr.length;
|
|
303
|
+
if (this._bodySizeLimitEnabled && bodySize > this._maxBodySize) return null;
|
|
304
|
+
body += chunkStr;
|
|
305
|
+
}
|
|
306
|
+
return body;
|
|
307
|
+
}
|
|
308
|
+
_sendSseEvent(res, event, data) {
|
|
309
|
+
try {
|
|
310
|
+
res.write(`event: ${event}\n`);
|
|
311
|
+
res.write(`data: ${JSON.stringify(data)}\n\n`);
|
|
312
|
+
} catch {}
|
|
313
|
+
}
|
|
314
|
+
broadcastToSession(sessionId, event, data) {
|
|
315
|
+
const session = this._sessions.get(sessionId);
|
|
316
|
+
if (!session) return;
|
|
317
|
+
for (const stream of session.notificationStreams)this._sendSseEvent(stream, event, data);
|
|
318
|
+
}
|
|
319
|
+
_handleMetrics(res) {
|
|
320
|
+
if (!this._metricsProvider) {
|
|
321
|
+
res.writeHead(404, {
|
|
322
|
+
'Content-Type': 'text/plain'
|
|
323
|
+
});
|
|
324
|
+
res.end('Not Found');
|
|
325
|
+
return;
|
|
326
|
+
}
|
|
327
|
+
res.writeHead(200, {
|
|
328
|
+
'Content-Type': 'text/plain; version=0.0.4; charset=utf-8'
|
|
329
|
+
});
|
|
330
|
+
res.end(this._metricsProvider());
|
|
331
|
+
}
|
|
332
|
+
_handleHealthCheck(res) {
|
|
333
|
+
const healthData = {
|
|
334
|
+
status: 'healthy',
|
|
335
|
+
requests: this._requestCount,
|
|
336
|
+
sessions: this._sessions.size,
|
|
337
|
+
transport: 'streamable-http'
|
|
338
|
+
};
|
|
339
|
+
if (this._healthChecker) {
|
|
340
|
+
const liveness = this._healthChecker.checkLiveness();
|
|
341
|
+
healthData.liveness = liveness;
|
|
342
|
+
}
|
|
343
|
+
res.writeHead(200, {
|
|
344
|
+
'Content-Type': 'application/json'
|
|
345
|
+
});
|
|
346
|
+
res.end(JSON.stringify(healthData));
|
|
347
|
+
}
|
|
348
|
+
async _handleReadinessCheck(res) {
|
|
349
|
+
if (this._healthChecker) {
|
|
350
|
+
const readiness = await this._healthChecker.checkReadiness();
|
|
351
|
+
const statusCode = 'ok' === readiness.status ? 200 : 503;
|
|
352
|
+
res.writeHead(statusCode, {
|
|
353
|
+
'Content-Type': 'application/json'
|
|
354
|
+
});
|
|
355
|
+
res.end(JSON.stringify(readiness));
|
|
356
|
+
} else {
|
|
357
|
+
res.writeHead(200, {
|
|
358
|
+
'Content-Type': 'application/json'
|
|
359
|
+
});
|
|
360
|
+
res.end(JSON.stringify({
|
|
361
|
+
status: 'ok',
|
|
362
|
+
timestamp: new Date().toISOString(),
|
|
363
|
+
components: {}
|
|
364
|
+
}));
|
|
365
|
+
}
|
|
366
|
+
}
|
|
367
|
+
_updateSessionMetrics() {
|
|
368
|
+
this._metrics?.gauge('streamable_http_active_sessions', this._sessions.size, {}, 'Active Streamable HTTP sessions');
|
|
369
|
+
let totalStreams = 0;
|
|
370
|
+
for (const session of this._sessions.values())totalStreams += session.notificationStreams.size;
|
|
371
|
+
this._metrics?.gauge('streamable_http_notification_streams', totalStreams, {}, 'Active SSE notification streams');
|
|
372
|
+
}
|
|
373
|
+
async stop(timeout) {
|
|
374
|
+
this._isShuttingDown = true;
|
|
375
|
+
this._stopRateLimitCleanup();
|
|
376
|
+
const shutdownTimeout = timeout ?? 30000;
|
|
377
|
+
for (const session of this._sessions.values()){
|
|
378
|
+
for (const stream of session.notificationStreams)try {
|
|
379
|
+
stream.end();
|
|
380
|
+
} catch {}
|
|
381
|
+
session.notificationStreams.clear();
|
|
382
|
+
}
|
|
383
|
+
this._sessions.clear();
|
|
384
|
+
return new Promise((resolve)=>{
|
|
385
|
+
if (!this._server) {
|
|
386
|
+
this.log('info', 'Streamable HTTP transport stopped (no server)');
|
|
387
|
+
resolve();
|
|
388
|
+
return;
|
|
389
|
+
}
|
|
390
|
+
const forceClose = setTimeout(()=>{
|
|
391
|
+
this.log('warn', 'Streamable HTTP transport force-closing after timeout');
|
|
392
|
+
resolve();
|
|
393
|
+
}, shutdownTimeout);
|
|
394
|
+
this._server.close(()=>{
|
|
395
|
+
clearTimeout(forceClose);
|
|
396
|
+
this.log('info', 'Streamable HTTP transport stopped');
|
|
397
|
+
resolve();
|
|
398
|
+
});
|
|
399
|
+
});
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
function createStreamableHttpTransport(options = {}) {
|
|
403
|
+
return new StreamableHttpTransport(options);
|
|
404
|
+
}
|
|
405
|
+
export { StreamableHttpTransport, createStreamableHttpTransport };
|
|
406
|
+
|
|
407
|
+
//# sourceMappingURL=StreamableHttpTransport.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transport/StreamableHttpTransport.js","sources":["../../src/transport/StreamableHttpTransport.ts"],"sourcesContent":["/**\n * Streamable HTTP Transport implementation (MCP spec recommended transport).\n *\n * This transport implements the MCP Streamable HTTP specification, which replaces\n * the deprecated SSE transport as the recommended HTTP-based transport since March 2025.\n *\n * Key features:\n * - POST /mcp for JSON-RPC requests (main MCP endpoint)\n * - GET /mcp for optional SSE server-to-client notifications\n * - Session management via Mcp-Session-Id header\n * - Supports both stateful (session-based) and stateless (per-request) modes\n * - Health endpoints (/health, /ready)\n *\n * @example\n * ```typescript\n * const transport = new StreamableHttpTransport({\n * port: 3000,\n * host: 'localhost',\n * stateful: true,\n * });\n * await transport.connect(server);\n * ```\n */\n\nimport { randomUUID } from 'node:crypto';\nimport { createServer, type IncomingMessage, type Server, type ServerResponse } from 'node:http';\nimport type { McpServer } from 'tmcp';\nimport { safeParse } from 'valibot';\nimport type { IMetrics } from '../contracts/index.js';\nimport { JsonRpcRequestSchema } from '../schema.js';\nimport { BaseTransport, type TransportOptions } from './BaseTransport.js';\n\n/**\n * MCP Streamable HTTP transport options extending base TransportOptions.\n */\nexport interface StreamableHttpTransportOptions extends TransportOptions {\n\t/**\n\t * Path for the MCP endpoint\n\t * @default '/mcp'\n\t */\n\tpath?: string;\n\n\t/**\n\t * Metrics collector for transport telemetry\n\t */\n\tmetrics?: IMetrics;\n\n\t/**\n\t * Prometheus metrics provider function\n\t */\n\tmetricsProvider?: () => string;\n\n\t/**\n\t * Enable stateful session-based mode.\n\t * When true, sessions are tracked via Mcp-Session-Id header.\n\t * When false, each request is processed independently (stateless).\n\t * @default true\n\t */\n\tstateful?: boolean;\n\n\t/**\n\t * Custom session ID generator function.\n\t * @default () => randomUUID()\n\t */\n\tsessionIdGenerator?: () => string;\n\n\t/**\n\t * Enable request body size limit\n\t * @default true\n\t */\n\tenableBodySizeLimit?: boolean;\n\n\t/**\n\t * Maximum request body size in bytes\n\t * @default 10485760 (10MB)\n\t */\n\tmaxBodySize?: number;\n\n\t/**\n\t * Request timeout in milliseconds\n\t * @default 30000 (30 seconds)\n\t */\n\trequestTimeout?: number;\n}\n\n/**\n * Internal session state for stateful mode.\n */\ninterface SessionState {\n\t/** Unique session identifier */\n\tid: string;\n\t/** Timestamp when the session was created */\n\tcreatedAt: number;\n\t/** Timestamp of the last activity */\n\tlastActivityAt: number;\n\t/** Active SSE notification streams for this session */\n\tnotificationStreams: Set<ServerResponse>;\n}\n\n/**\n * Streamable HTTP Transport for MCP server.\n *\n * This transport implements the MCP Streamable HTTP specification,\n * providing JSON-RPC over HTTP with optional session management\n * and server-to-client SSE notification streams.\n *\n * @remarks\n * **Security Features (inherited from BaseTransport):**\n * - Session ID validation (alphanumeric, max 64 chars)\n * - Query parameter sanitization (whitelist allowed keys)\n * - Rate limiting per IP (configurable, default 100 req/min)\n * - CORS origin validation\n * - Host header validation\n *\n * **MCP Streamable HTTP Spec Compliance:**\n * - POST /mcp — JSON-RPC method calls\n * - GET /mcp — SSE notification stream (server-to-client)\n * - Mcp-Session-Id header for session management\n * - Content-Type: application/json for JSON-RPC responses\n * - Content-Type: text/event-stream for SSE notification streams\n *\n * **HTTP Status Code Mapping:**\n * - 200: Success (JSON-RPC response or SSE stream)\n * - 202: Accepted (JSON-RPC notification, no response body)\n * - 204: CORS Preflight (empty body)\n * - 400: Bad Request (invalid JSON, invalid session ID)\n * - 403: Forbidden (invalid CORS, invalid host)\n * - 404: Not Found\n * - 405: Method Not Allowed\n * - 413: Payload Too Large\n * - 429: Too Many Requests\n * - 500: Internal Server Error\n * - 503: Server Not Ready / Shutting Down\n */\nexport class StreamableHttpTransport extends BaseTransport {\n\tprivate _server: Server | null = null;\n\tprivate _mcpServer: McpServer | null = null;\n\tprivate _path: string;\n\tprivate _stateful: boolean;\n\tprivate _sessionIdGenerator: () => string;\n\tprivate _sessions: Map<string, SessionState> = new Map();\n\tprivate _requestCount: number = 0;\n\tprivate _activeRequests: number = 0;\n\tprivate _bodySizeLimitEnabled: boolean;\n\tprivate _maxBodySize: number;\n\tprivate _requestTimeout: number;\n\tprivate _metrics?: IMetrics;\n\tprivate _metricsProvider: (() => string) | null;\n\n\tconstructor(options: StreamableHttpTransportOptions = {}) {\n\t\tsuper(options);\n\n\t\tthis._path = options.path ?? '/mcp';\n\t\tthis._stateful = options.stateful ?? true;\n\t\tthis._sessionIdGenerator = options.sessionIdGenerator ?? (() => randomUUID());\n\t\tthis._bodySizeLimitEnabled = options.enableBodySizeLimit ?? true;\n\t\tthis._maxBodySize = options.maxBodySize ?? 10 * 1024 * 1024;\n\t\tthis._requestTimeout = options.requestTimeout ?? 30000;\n\t\tthis._metrics = options.metrics;\n\t\tthis._metricsProvider = options.metricsProvider ?? null;\n\t}\n\n\t/**\n\t * Get number of active sessions (stateful) or active requests (stateless).\n\t */\n\tget clientCount(): number {\n\t\treturn this._stateful ? this._sessions.size : this._activeRequests;\n\t}\n\n\t/**\n\t * Get the total number of requests handled.\n\t */\n\tget requestCount(): number {\n\t\treturn this._requestCount;\n\t}\n\n\t/**\n\t * Connects MCP server to this transport and starts listening.\n\t */\n\tasync connect(mcpServer: McpServer): Promise<void> {\n\t\tthis._mcpServer = mcpServer;\n\t\tthis._server = createServer((req, res) => this._handleRequest(req, res));\n\n\t\treturn new Promise((resolve) => {\n\t\t\tthis._server!.listen(this._port, this._host, () => {\n\t\t\t\tthis.log(\n\t\t\t\t\t'info',\n\t\t\t\t\t`Streamable HTTP transport listening on http://${this._host}:${this._port}`\n\t\t\t\t);\n\t\t\t\tresolve();\n\t\t\t});\n\t\t});\n\t}\n\n\t/**\n\t * Route and handle incoming HTTP requests.\n\t */\n\tprivate async _handleRequest(req: IncomingMessage, res: ServerResponse): Promise<void> {\n\t\tconst startTime = Date.now();\n\t\tthis._metrics?.counter('streamable_http_requests_total', 1, {}, 'Total Streamable HTTP transport requests');\n\t\tres.once('finish', () => {\n\t\t\tconst durationSeconds = (Date.now() - startTime) / 1000;\n\t\t\tthis._metrics?.histogram('streamable_http_request_duration_seconds', durationSeconds, {});\n\t\t});\n\n\t\t// Host validation\n\t\tif (!this.validateHostHeader(req)) {\n\t\t\tthis._sendJsonRpcError(res, 403, -32000, 'Forbidden - invalid host header');\n\t\t\treturn;\n\t\t}\n\n\t\t// Shutdown check\n\t\tif (this.isShuttingDown()) {\n\t\t\tthis._sendJsonRpcError(res, 503, -32603, 'Server is shutting down');\n\t\t\treturn;\n\t\t}\n\n\t\t// Rate limiting\n\t\tconst clientIp = this.getClientIp(req);\n\t\tif (this.checkRateLimit(clientIp)) {\n\t\t\tres.setHeader('Retry-After', '60');\n\t\t\tthis._sendJsonRpcError(res, 429, -32000, 'Too many requests');\n\t\t\treturn;\n\t\t}\n\n\t\t// CORS validation\n\t\tif (!this.validateCorsOrigin(req)) {\n\t\t\tthis._sendJsonRpcError(res, 403, -32000, 'Forbidden - invalid origin');\n\t\t\treturn;\n\t\t}\n\n\t\tthis.setCorsHeaders(res);\n\n\t\t// CORS preflight\n\t\tif (req.method === 'OPTIONS') {\n\t\t\tres.setHeader('Access-Control-Allow-Headers', 'Content-Type, Mcp-Session-Id');\n\t\t\tres.writeHead(204);\n\t\t\tres.end();\n\t\t\treturn;\n\t\t}\n\n\t\t// Parse URL path\n\t\tconst urlPath = req.url?.split('?')[0] ?? '/';\n\n\t\t// Metrics endpoint\n\t\tif (req.method === 'GET' && urlPath === '/metrics') {\n\t\t\tthis._handleMetrics(res);\n\t\t\treturn;\n\t\t}\n\n\t\t// Health check (liveness)\n\t\tif (req.method === 'GET' && urlPath === '/health') {\n\t\t\tthis._handleHealthCheck(res);\n\t\t\treturn;\n\t\t}\n\n\t\t// Readiness check\n\t\tif (req.method === 'GET' && urlPath === '/ready') {\n\t\t\tawait this._handleReadinessCheck(res);\n\t\t\treturn;\n\t\t}\n\n\t\t// MCP endpoint routing\n\t\tif (urlPath === this._path) {\n\t\t\tif (req.method === 'POST') {\n\t\t\t\tawait this._handleMcpPost(req, res);\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tif (req.method === 'GET') {\n\t\t\t\tthis._handleMcpGet(req, res);\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tres.writeHead(405, { 'Content-Type': 'application/json', Allow: 'GET, POST' });\n\t\t\tres.end(JSON.stringify({ jsonrpc: '2.0', id: null, error: { code: -32601, message: 'Method not allowed' } }));\n\t\t\treturn;\n\t\t}\n\n\t\t// 404 for unknown paths\n\t\tthis._sendJsonRpcError(res, 404, -32601, 'Not Found');\n\t}\n\n\t/**\n\t * Send a JSON-RPC error response.\n\t */\n\tprivate _sendJsonRpcError(res: ServerResponse, statusCode: number, code: number, message: string, id: unknown = null, extra?: Record<string, unknown>): void {\n\t\tconst error: Record<string, unknown> = { code, message };\n\t\tif (extra) {\n\t\t\tObject.assign(error, extra);\n\t\t}\n\t\tres.writeHead(statusCode, { 'Content-Type': 'application/json' });\n\t\tres.end(JSON.stringify({ jsonrpc: '2.0', id, error }));\n\t}\n\n\t/**\n\t * Handle POST /mcp — JSON-RPC method calls.\n\t *\n\t * Per the MCP Streamable HTTP spec:\n\t * - Accepts JSON-RPC request bodies\n\t * - Returns Mcp-Session-Id header for new sessions (stateful mode)\n\t * - Validates Mcp-Session-Id for existing sessions (stateful mode)\n\t * - Content-Type: application/json for responses\n\t */\n\tprivate async _handleMcpPost(req: IncomingMessage, res: ServerResponse): Promise<void> {\n\t\tthis._requestCount++;\n\t\tthis._activeRequests++;\n\n\t\tconst timeout = setTimeout(() => {\n\t\t\tthis._activeRequests--;\n\t\t\tthis._sendJsonRpcError(res, 500, -32603, 'Request timeout');\n\t\t}, this._requestTimeout);\n\n\t\ttry {\n\t\t\t// Read request body with size limit\n\t\t\tconst body = await this._readRequestBody(req);\n\t\t\tif (body === null) {\n\t\t\t\tclearTimeout(timeout);\n\t\t\t\tthis._activeRequests--;\n\t\t\t\tthis._sendJsonRpcError(res, 413, -32000, 'Request body too large');\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Parse JSON\n\t\t\tlet jsonRpcRequest;\n\t\t\ttry {\n\t\t\t\tjsonRpcRequest = JSON.parse(body);\n\t\t\t} catch {\n\t\t\t\tclearTimeout(timeout);\n\t\t\t\tthis._activeRequests--;\n\t\t\t\tthis._sendJsonRpcError(res, 200, -32700, 'Parse error');\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Validate JSON-RPC schema\n\t\t\tconst parseResult = safeParse(JsonRpcRequestSchema, jsonRpcRequest);\n\t\t\tif (!parseResult.success) {\n\t\t\t\tclearTimeout(timeout);\n\t\t\t\tthis._activeRequests--;\n\t\t\t\tthis._sendJsonRpcError(res, 200, -32600, 'Invalid Request', jsonRpcRequest?.id ?? null, { data: parseResult.issues });\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Session management (stateful mode)\n\t\t\tlet sessionId: string | undefined;\n\t\t\tif (this._stateful) {\n\t\t\t\tconst sessionResult = this._resolveSession(req, res);\n\t\t\t\tif (sessionResult === false) {\n\t\t\t\t\tclearTimeout(timeout);\n\t\t\t\t\tthis._activeRequests--;\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t\tsessionId = sessionResult;\n\t\t\t}\n\n\t\t\t// Check if MCP server is ready\n\t\t\tif (!this._mcpServer) {\n\t\t\t\tclearTimeout(timeout);\n\t\t\t\tthis._activeRequests--;\n\t\t\t\tthis._sendJsonRpcError(res, 503, -32603, 'Server not ready', jsonRpcRequest?.id ?? null);\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Process JSON-RPC request through MCP server\n\t\t\tconst response = await this._mcpServer.receive(jsonRpcRequest, { sessionInfo: {} });\n\t\t\tclearTimeout(timeout);\n\t\t\tthis._activeRequests--;\n\n\t\t\t// Send response with session header if applicable\n\t\t\tconst responseHeaders: Record<string, string> = { 'Content-Type': 'application/json' };\n\t\t\tif (sessionId) responseHeaders['Mcp-Session-Id'] = sessionId;\n\n\t\t\tif (response) {\n\t\t\t\tres.writeHead(200, responseHeaders);\n\t\t\t\tres.end(JSON.stringify(response));\n\t\t\t} else {\n\t\t\t\tif (sessionId) res.setHeader('Mcp-Session-Id', sessionId);\n\t\t\t\tres.writeHead(202);\n\t\t\t\tres.end();\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tclearTimeout(timeout);\n\t\t\tthis._activeRequests--;\n\t\t\tthis._sendJsonRpcError(res, 200, -32603, 'Internal error', null, {\n\t\t\t\tdata: error instanceof Error ? error.message : String(error),\n\t\t\t});\n\t\t}\n\t}\n\n\n\t/**\n\t * Handle GET /mcp — Optional SSE notification stream.\n\t *\n\t * Per the MCP Streamable HTTP spec, clients may open a GET request\n\t * to receive server-initiated notifications as SSE events.\n\t * Requires a valid Mcp-Session-Id in stateful mode.\n\t */\n\tprivate _handleMcpGet(req: IncomingMessage, res: ServerResponse): void {\n\t\tif (!this._stateful) {\n\t\t\t// SSE notification streams require stateful mode\n\t\t\tres.writeHead(405, {\n\t\t\t\t'Content-Type': 'application/json',\n\t\t\t\tAllow: 'POST',\n\t\t\t});\n\t\t\tres.end(\n\t\t\t\tJSON.stringify({\n\t\t\t\t\tjsonrpc: '2.0',\n\t\t\t\t\tid: null,\n\t\t\t\t\terror: { code: -32601, message: 'GET not supported in stateless mode' },\n\t\t\t\t})\n\t\t\t);\n\t\t\treturn;\n\t\t}\n\n\t\t// Require Mcp-Session-Id for GET requests\n\t\tconst sessionId = this._getSessionIdFromHeader(req);\n\t\tif (!sessionId) {\n\t\t\tres.writeHead(400, { 'Content-Type': 'application/json' });\n\t\t\tres.end(\n\t\t\t\tJSON.stringify({\n\t\t\t\t\tjsonrpc: '2.0',\n\t\t\t\t\tid: null,\n\t\t\t\t\terror: { code: -32600, message: 'Missing Mcp-Session-Id header' },\n\t\t\t\t})\n\t\t\t);\n\t\t\treturn;\n\t\t}\n\n\t\tconst session = this._sessions.get(sessionId);\n\t\tif (!session) {\n\t\t\tres.writeHead(404, { 'Content-Type': 'application/json' });\n\t\t\tres.end(\n\t\t\t\tJSON.stringify({\n\t\t\t\t\tjsonrpc: '2.0',\n\t\t\t\t\tid: null,\n\t\t\t\t\terror: { code: -32001, message: 'Session not found' },\n\t\t\t\t})\n\t\t\t);\n\t\t\treturn;\n\t\t}\n\n\t\t// Set SSE headers\n\t\tres.writeHead(200, {\n\t\t\t'Content-Type': 'text/event-stream',\n\t\t\t'Cache-Control': 'no-cache',\n\t\t\tConnection: 'keep-alive',\n\t\t\t'Mcp-Session-Id': sessionId,\n\t\t});\n\n\t\t// Send initial connected event\n\t\tthis._sendSseEvent(res, 'connected', {\n\t\t\tsessionId,\n\t\t\ttimestamp: Date.now(),\n\t\t});\n\n\t\t// Track this notification stream\n\t\tsession.notificationStreams.add(res);\n\t\tsession.lastActivityAt = Date.now();\n\t\tthis._updateSessionMetrics();\n\n\t\t// Handle client disconnect\n\t\treq.on('close', () => {\n\t\t\tsession.notificationStreams.delete(res);\n\t\t\tthis._updateSessionMetrics();\n\t\t});\n\t}\n\n\t/**\n\t * Resolve or create a session for a stateful request.\n\t *\n\t * @returns Session ID string on success, or `false` if the response was already sent (error).\n\t */\n\tprivate _resolveSession(req: IncomingMessage, res: ServerResponse): string | false {\n\t\tconst headerSessionId = this._getSessionIdFromHeader(req);\n\n\t\tif (headerSessionId) {\n\t\t\t// Validate format\n\t\t\tif (!this.validateSessionId(headerSessionId)) {\n\t\t\t\tres.writeHead(400, { 'Content-Type': 'application/json' });\n\t\t\t\tres.end(\n\t\t\t\t\tJSON.stringify({\n\t\t\t\t\t\tjsonrpc: '2.0',\n\t\t\t\t\t\tid: null,\n\t\t\t\t\t\terror: { code: -32600, message: 'Invalid Mcp-Session-Id format' },\n\t\t\t\t\t})\n\t\t\t\t);\n\t\t\t\treturn false;\n\t\t\t}\n\n\t\t\t// Check if session exists\n\t\t\tconst session = this._sessions.get(headerSessionId);\n\t\t\tif (session) {\n\t\t\t\tsession.lastActivityAt = Date.now();\n\t\t\t\treturn headerSessionId;\n\t\t\t}\n\n\t\t\t// Unknown session ID — per spec, return 404\n\t\t\tres.writeHead(404, { 'Content-Type': 'application/json' });\n\t\t\tres.end(\n\t\t\t\tJSON.stringify({\n\t\t\t\t\tjsonrpc: '2.0',\n\t\t\t\t\tid: null,\n\t\t\t\t\terror: { code: -32001, message: 'Session not found' },\n\t\t\t\t})\n\t\t\t);\n\t\t\treturn false;\n\t\t}\n\n\t\t// No session header — create new session\n\t\tconst newSessionId = this._sessionIdGenerator();\n\t\tconst sessionState: SessionState = {\n\t\t\tid: newSessionId,\n\t\t\tcreatedAt: Date.now(),\n\t\t\tlastActivityAt: Date.now(),\n\t\t\tnotificationStreams: new Set(),\n\t\t};\n\t\tthis._sessions.set(newSessionId, sessionState);\n\t\tthis.log('info', `New session created: ${newSessionId}`);\n\t\tthis._updateSessionMetrics();\n\n\t\treturn newSessionId;\n\t}\n\n\t/**\n\t * Extract Mcp-Session-Id from request headers.\n\t */\n\tprivate _getSessionIdFromHeader(req: IncomingMessage): string | undefined {\n\t\tconst value = req.headers['mcp-session-id'];\n\t\tif (typeof value === 'string' && value.length > 0) {\n\t\t\treturn value;\n\t\t}\n\t\treturn undefined;\n\t}\n\n\t/**\n\t * Read the request body with optional size limit enforcement.\n\t *\n\t * @returns Body string, or null if the body exceeds the size limit.\n\t */\n\tprivate async _readRequestBody(req: IncomingMessage): Promise<string | null> {\n\t\tlet body = '';\n\t\tlet bodySize = 0;\n\n\t\tfor await (const chunk of req) {\n\t\t\tconst chunkStr = typeof chunk === 'string' ? chunk : (chunk as Buffer).toString();\n\t\t\tbodySize += chunkStr.length;\n\n\t\t\tif (this._bodySizeLimitEnabled && bodySize > this._maxBodySize) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tbody += chunkStr;\n\t\t}\n\n\t\treturn body;\n\t}\n\n\t/**\n\t * Send an SSE event to a specific client response stream.\n\t */\n\tprivate _sendSseEvent(res: ServerResponse, event: string, data: unknown): void {\n\t\ttry {\n\t\t\tres.write(`event: ${event}\\n`);\n\t\t\tres.write(`data: ${JSON.stringify(data)}\\n\\n`);\n\t\t} catch {\n\t\t\t// Client disconnected — ignore\n\t\t}\n\t}\n\n\t/**\n\t * Broadcast a notification to all SSE streams in a given session.\n\t *\n\t * @param sessionId - Target session ID\n\t * @param event - SSE event name\n\t * @param data - Event payload\n\t */\n\tbroadcastToSession(sessionId: string, event: string, data: unknown): void {\n\t\tconst session = this._sessions.get(sessionId);\n\t\tif (!session) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor (const stream of session.notificationStreams) {\n\t\t\tthis._sendSseEvent(stream, event, data);\n\t\t}\n\t}\n\n\t/**\n\t * Handle GET /metrics endpoint.\n\t */\n\tprivate _handleMetrics(res: ServerResponse): void {\n\t\tif (!this._metricsProvider) {\n\t\t\tres.writeHead(404, { 'Content-Type': 'text/plain' });\n\t\t\tres.end('Not Found');\n\t\t\treturn;\n\t\t}\n\n\t\tres.writeHead(200, { 'Content-Type': 'text/plain; version=0.0.4; charset=utf-8' });\n\t\tres.end(this._metricsProvider());\n\t}\n\n\t/**\n\t * Handle GET /health — Liveness check.\n\t */\n\tprivate _handleHealthCheck(res: ServerResponse): void {\n\t\tconst healthData: Record<string, unknown> = {\n\t\t\tstatus: 'healthy',\n\t\t\trequests: this._requestCount,\n\t\t\tsessions: this._sessions.size,\n\t\t\ttransport: 'streamable-http',\n\t\t};\n\t\tif (this._healthChecker) {\n\t\t\tconst liveness = this._healthChecker.checkLiveness();\n\t\t\thealthData.liveness = liveness;\n\t\t}\n\t\tres.writeHead(200, { 'Content-Type': 'application/json' });\n\t\tres.end(JSON.stringify(healthData));\n\t}\n\n\t/**\n\t * Handle GET /ready — Readiness check.\n\t */\n\tprivate async _handleReadinessCheck(res: ServerResponse): Promise<void> {\n\t\tif (this._healthChecker) {\n\t\t\tconst readiness = await this._healthChecker.checkReadiness();\n\t\t\tconst statusCode = readiness.status === 'ok' ? 200 : 503;\n\t\t\tres.writeHead(statusCode, { 'Content-Type': 'application/json' });\n\t\t\tres.end(JSON.stringify(readiness));\n\t\t} else {\n\t\t\tres.writeHead(200, { 'Content-Type': 'application/json' });\n\t\t\tres.end(\n\t\t\t\tJSON.stringify({\n\t\t\t\t\tstatus: 'ok',\n\t\t\t\t\ttimestamp: new Date().toISOString(),\n\t\t\t\t\tcomponents: {},\n\t\t\t\t})\n\t\t\t);\n\t\t}\n\t}\n\n\t/**\n\t * Update session-related metrics.\n\t */\n\tprivate _updateSessionMetrics(): void {\n\t\tthis._metrics?.gauge(\n\t\t\t'streamable_http_active_sessions',\n\t\t\tthis._sessions.size,\n\t\t\t{},\n\t\t\t'Active Streamable HTTP sessions'\n\t\t);\n\n\t\tlet totalStreams = 0;\n\t\tfor (const session of this._sessions.values()) {\n\t\t\ttotalStreams += session.notificationStreams.size;\n\t\t}\n\t\tthis._metrics?.gauge(\n\t\t\t'streamable_http_notification_streams',\n\t\t\ttotalStreams,\n\t\t\t{},\n\t\t\t'Active SSE notification streams'\n\t\t);\n\t}\n\n\t/**\n\t * Stop transport server with graceful shutdown.\n\t *\n\t * @param timeout - Maximum time to wait for in-flight requests (default: 30s)\n\t */\n\tasync stop(timeout?: number): Promise<void> {\n\t\tthis._isShuttingDown = true;\n\t\tthis._stopRateLimitCleanup();\n\n\t\tconst shutdownTimeout = timeout ?? 30000;\n\n\t\t// Close all SSE notification streams\n\t\tfor (const session of this._sessions.values()) {\n\t\t\tfor (const stream of session.notificationStreams) {\n\t\t\t\ttry {\n\t\t\t\t\tstream.end();\n\t\t\t\t} catch {\n\t\t\t\t\t// Ignore errors\n\t\t\t\t}\n\t\t\t}\n\t\t\tsession.notificationStreams.clear();\n\t\t}\n\t\tthis._sessions.clear();\n\n\t\treturn new Promise((resolve) => {\n\t\t\tif (!this._server) {\n\t\t\t\tthis.log('info', 'Streamable HTTP transport stopped (no server)');\n\t\t\t\tresolve();\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Force close after timeout\n\t\t\tconst forceClose = setTimeout(() => {\n\t\t\t\tthis.log('warn', 'Streamable HTTP transport force-closing after timeout');\n\t\t\t\tresolve();\n\t\t\t}, shutdownTimeout);\n\n\t\t\tthis._server.close(() => {\n\t\t\t\tclearTimeout(forceClose);\n\t\t\t\tthis.log('info', 'Streamable HTTP transport stopped');\n\t\t\t\tresolve();\n\t\t\t});\n\t\t});\n\t}\n}\n\n/**\n * Create a Streamable HTTP transport with given options.\n *\n * @param options - Transport configuration\n * @returns A configured Streamable HTTP transport\n *\n * @example\n * ```typescript\n * const transport = createStreamableHttpTransport({ port: 3000, stateful: true });\n * await transport.connect(server);\n * ```\n */\nexport function createStreamableHttpTransport(\n\toptions: StreamableHttpTransportOptions = {}\n): StreamableHttpTransport {\n\treturn new StreamableHttpTransport(options);\n}\n"],"names":["StreamableHttpTransport","BaseTransport","Map","options","randomUUID","mcpServer","createServer","req","res","Promise","resolve","startTime","Date","durationSeconds","clientIp","urlPath","JSON","statusCode","code","message","id","extra","error","Object","timeout","setTimeout","body","clearTimeout","jsonRpcRequest","parseResult","safeParse","JsonRpcRequestSchema","sessionId","sessionResult","response","responseHeaders","Error","String","session","headerSessionId","newSessionId","sessionState","Set","value","bodySize","chunk","chunkStr","event","data","stream","healthData","liveness","readiness","totalStreams","shutdownTimeout","forceClose","createStreamableHttpTransport"],"mappings":"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"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Interface for services that need resource cleanup.
|
|
3
|
+
*
|
|
4
|
+
* Implement this on any service that holds resources (connections, file handles, timers, etc.)
|
|
5
|
+
*
|
|
6
|
+
* @module types/disposable
|
|
7
|
+
*
|
|
8
|
+
* @example
|
|
9
|
+
* ```typescript
|
|
10
|
+
* class MyService implements IDisposable {
|
|
11
|
+
* async dispose(): Promise<void> {
|
|
12
|
+
* await this.closeConnections();
|
|
13
|
+
* this.clearTimers();
|
|
14
|
+
* }
|
|
15
|
+
* }
|
|
16
|
+
* ```
|
|
17
|
+
*/
|
|
18
|
+
export interface IDisposable {
|
|
19
|
+
/** Release all resources held by this service. */
|
|
20
|
+
dispose(): Promise<void>;
|
|
21
|
+
}
|
|
22
|
+
//# sourceMappingURL=disposable.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"disposable.d.ts","sourceRoot":"","sources":["../../src/types/disposable.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;GAgBG;AACH,MAAM,WAAW,WAAW;IAC3B,kDAAkD;IAClD,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC;CACzB"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { };
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Runtime server configuration type.
|
|
3
|
+
*
|
|
4
|
+
* @module types/server-config
|
|
5
|
+
*/
|
|
6
|
+
import type { Skill } from './skill.js';
|
|
7
|
+
import type { Tool } from './tool.js';
|
|
8
|
+
/**
|
|
9
|
+
* Runtime server configuration containing available tools and skills.
|
|
10
|
+
*
|
|
11
|
+
* This interface represents the active configuration at runtime,
|
|
12
|
+
* providing access to the registry of available tools and skills.
|
|
13
|
+
*
|
|
14
|
+
* @example
|
|
15
|
+
* ```typescript
|
|
16
|
+
* const config: ServerConfig = {
|
|
17
|
+
* available_tools: new Map([
|
|
18
|
+
* ['Read', { name: 'Read', description: '...', inputSchema: {...} }]
|
|
19
|
+
* ]),
|
|
20
|
+
* available_skills: new Map([
|
|
21
|
+
* ['commit', { name: 'commit', description: '...', user_invocable: true }]
|
|
22
|
+
* ])
|
|
23
|
+
* };
|
|
24
|
+
* ```
|
|
25
|
+
*/
|
|
26
|
+
export interface ServerConfig {
|
|
27
|
+
/** Map of registered tools indexed by their unique names. */
|
|
28
|
+
available_tools: Map<string, Tool>;
|
|
29
|
+
/** Map of registered skills indexed by their unique names. */
|
|
30
|
+
available_skills: Map<string, Skill>;
|
|
31
|
+
}
|
|
32
|
+
//# sourceMappingURL=server-config.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server-config.d.ts","sourceRoot":"","sources":["../../src/types/server-config.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AACxC,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEtC;;;;;;;;;;;;;;;;;GAiBG;AACH,MAAM,WAAW,YAAY;IAC5B,6DAA6D;IAC7D,eAAe,EAAE,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;IAEnC,8DAA8D;IAC9D,gBAAgB,EAAE,GAAG,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;CACrC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { };
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Skill types for MCP skill recommendations and definitions.
|
|
3
|
+
*
|
|
4
|
+
* @module types/skill
|
|
5
|
+
*/
|
|
6
|
+
/**
|
|
7
|
+
* Represents a recommended skill with associated metadata.
|
|
8
|
+
*
|
|
9
|
+
* Skills are higher-level workflows that coordinate multiple tools and operations.
|
|
10
|
+
* Skill recommendations include information about which tools the skill is allowed
|
|
11
|
+
* to use and whether the skill can be invoked directly by users.
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* ```typescript
|
|
15
|
+
* const recommendation: SkillRecommendation = {
|
|
16
|
+
* skill_name: 'commit',
|
|
17
|
+
* confidence: 0.95,
|
|
18
|
+
* rationale: 'Handles the complete git commit workflow',
|
|
19
|
+
* priority: 1,
|
|
20
|
+
* alternatives: ['review-pr'],
|
|
21
|
+
* allowed_tools: ['Bash', 'Read', 'Grep'],
|
|
22
|
+
* user_invocable: true
|
|
23
|
+
* };
|
|
24
|
+
* ```
|
|
25
|
+
*/
|
|
26
|
+
export interface SkillRecommendation {
|
|
27
|
+
/** The unique name/identifier of the recommended skill. */
|
|
28
|
+
skill_name: string;
|
|
29
|
+
/** Confidence score from 0-1 indicating how appropriate this skill is for the current task. */
|
|
30
|
+
confidence: number;
|
|
31
|
+
/** Explanation of why this skill is recommended and how it addresses the current need. */
|
|
32
|
+
rationale: string;
|
|
33
|
+
/** Order in the recommendation sequence (lower numbers = higher priority). */
|
|
34
|
+
priority: number;
|
|
35
|
+
/** Alternative skills that could be used if the primary recommendation is not available. */
|
|
36
|
+
alternatives?: string[];
|
|
37
|
+
/** List of tool names this skill is allowed to use during execution. */
|
|
38
|
+
allowed_tools?: string[];
|
|
39
|
+
/** Whether this skill can be directly invoked by users (via slash commands). */
|
|
40
|
+
user_invocable?: boolean;
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Defines a skill that coordinates multiple tools and operations.
|
|
44
|
+
*
|
|
45
|
+
* Skills are higher-level workflows that can be invoked directly
|
|
46
|
+
* by users (via slash commands) or recommended by the sequential
|
|
47
|
+
* thinking process.
|
|
48
|
+
*
|
|
49
|
+
* @example
|
|
50
|
+
* ```typescript
|
|
51
|
+
* const skill: Skill = {
|
|
52
|
+
* name: 'commit',
|
|
53
|
+
* description: 'Create a git commit with proper message formatting',
|
|
54
|
+
* user_invocable: true,
|
|
55
|
+
* allowed_tools: ['Bash', 'Read', 'Grep']
|
|
56
|
+
* };
|
|
57
|
+
* ```
|
|
58
|
+
*/
|
|
59
|
+
export interface Skill {
|
|
60
|
+
/** Unique identifier for the skill. */
|
|
61
|
+
name: string;
|
|
62
|
+
/** Human-readable description of what the skill does. */
|
|
63
|
+
description: string;
|
|
64
|
+
/** Whether users can invoke this skill directly via slash commands. */
|
|
65
|
+
user_invocable?: boolean;
|
|
66
|
+
/** List of tool names this skill is allowed to use. */
|
|
67
|
+
allowed_tools?: string[];
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=skill.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"skill.d.ts","sourceRoot":"","sources":["../../src/types/skill.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAM,WAAW,mBAAmB;IACnC,2DAA2D;IAC3D,UAAU,EAAE,MAAM,CAAC;IAEnB,+FAA+F;IAC/F,UAAU,EAAE,MAAM,CAAC;IAEnB,0FAA0F;IAC1F,SAAS,EAAE,MAAM,CAAC;IAElB,8EAA8E;IAC9E,QAAQ,EAAE,MAAM,CAAC;IAEjB,4FAA4F;IAC5F,YAAY,CAAC,EAAE,MAAM,EAAE,CAAC;IAExB,wEAAwE;IACxE,aAAa,CAAC,EAAE,MAAM,EAAE,CAAC;IAEzB,gFAAgF;IAChF,cAAc,CAAC,EAAE,OAAO,CAAC;CACzB;AAED;;;;;;;;;;;;;;;;GAgBG;AACH,MAAM,WAAW,KAAK;IACrB,uCAAuC;IACvC,IAAI,EAAE,MAAM,CAAC;IAEb,yDAAyD;IACzD,WAAW,EAAE,MAAM,CAAC;IAEpB,uEAAuE;IACvE,cAAc,CAAC,EAAE,OAAO,CAAC;IAEzB,uDAAuD;IACvD,aAAa,CAAC,EAAE,MAAM,EAAE,CAAC;CACzB"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { };
|