footprintjs 3.0.16 → 3.0.17
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.
|
@@ -5,9 +5,10 @@
|
|
|
5
5
|
* Called by FlowChartBuilder.build() to enrich the compiled chart with
|
|
6
6
|
* d3-style chainable run methods and self-describing outputs.
|
|
7
7
|
*/
|
|
8
|
-
import {
|
|
8
|
+
import { normalizeSchema } from '../contract/schema.js';
|
|
9
9
|
import { RunContext } from './RunContext.js';
|
|
10
|
-
//
|
|
10
|
+
// Cache for no-options toOpenAPI() calls only — parameterized calls are not cached
|
|
11
|
+
// because different options produce different output.
|
|
11
12
|
const openAPICache = new WeakMap();
|
|
12
13
|
const mcpCache = new WeakMap();
|
|
13
14
|
/**
|
|
@@ -27,13 +28,18 @@ export function makeRunnable(chart) {
|
|
|
27
28
|
};
|
|
28
29
|
runnable.toOpenAPI = function (options) {
|
|
29
30
|
var _a, _b, _c, _d, _e, _f, _g, _h;
|
|
30
|
-
|
|
31
|
-
if (
|
|
32
|
-
|
|
31
|
+
// Only cache no-options calls — parameterized calls vary by options
|
|
32
|
+
if (!options) {
|
|
33
|
+
const cached = openAPICache.get(chart);
|
|
34
|
+
if (cached)
|
|
35
|
+
return cached;
|
|
36
|
+
}
|
|
33
37
|
const builderChart = chart;
|
|
34
38
|
const title = (_c = (_a = options === null || options === void 0 ? void 0 : options.title) !== null && _a !== void 0 ? _a : (_b = builderChart.description) === null || _b === void 0 ? void 0 : _b.split('\n')[0]) !== null && _c !== void 0 ? _c : 'API';
|
|
35
39
|
const version = (_d = options === null || options === void 0 ? void 0 : options.version) !== null && _d !== void 0 ? _d : '1.0.0';
|
|
36
|
-
|
|
40
|
+
// Use root.id (the explicit machine-readable id) as the path segment, sanitized
|
|
41
|
+
const rawId = (_f = (_e = builderChart.root) === null || _e === void 0 ? void 0 : _e.id) !== null && _f !== void 0 ? _f : 'execute';
|
|
42
|
+
const path = (_g = options === null || options === void 0 ? void 0 : options.path) !== null && _g !== void 0 ? _g : `/${slugify(rawId)}`;
|
|
37
43
|
const spec = {
|
|
38
44
|
openapi: '3.1.0',
|
|
39
45
|
info: { title, version, description: (_h = options === null || options === void 0 ? void 0 : options.description) !== null && _h !== void 0 ? _h : builderChart.description },
|
|
@@ -61,7 +67,8 @@ export function makeRunnable(chart) {
|
|
|
61
67
|
},
|
|
62
68
|
},
|
|
63
69
|
};
|
|
64
|
-
|
|
70
|
+
if (!options)
|
|
71
|
+
openAPICache.set(chart, spec);
|
|
65
72
|
return spec;
|
|
66
73
|
};
|
|
67
74
|
runnable.toMCPTool = function () {
|
|
@@ -70,34 +77,36 @@ export function makeRunnable(chart) {
|
|
|
70
77
|
if (cached)
|
|
71
78
|
return cached;
|
|
72
79
|
const builderChart = chart;
|
|
73
|
-
|
|
80
|
+
// Use root.id (explicit machine-readable id), sanitized to MCP name character allowlist
|
|
81
|
+
const name = sanitizeMCPName((_b = (_a = builderChart.root) === null || _a === void 0 ? void 0 : _a.id) !== null && _b !== void 0 ? _b : 'execute');
|
|
74
82
|
const description = builderChart.description || `Execute the ${name} flowchart`;
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
83
|
+
// MCP spec requires inputSchema to always be present.
|
|
84
|
+
// When no contract is defined, use the recommended no-parameter form.
|
|
85
|
+
const inputSchema = builderChart.inputSchema
|
|
86
|
+
? normalizeSchema(builderChart.inputSchema)
|
|
87
|
+
: { type: 'object', properties: {}, additionalProperties: false };
|
|
88
|
+
const tool = { name, description, inputSchema };
|
|
80
89
|
mcpCache.set(chart, tool);
|
|
81
90
|
return tool;
|
|
82
91
|
};
|
|
83
92
|
return runnable;
|
|
84
93
|
}
|
|
85
|
-
/**
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
94
|
+
/**
|
|
95
|
+
* Sanitize a string to conform to the MCP tool name character allowlist:
|
|
96
|
+
* [A-Za-z0-9_\-.] — any other character is replaced with underscore.
|
|
97
|
+
* Trims leading/trailing underscores introduced by replacement.
|
|
98
|
+
*/
|
|
99
|
+
function sanitizeMCPName(id) {
|
|
100
|
+
return id.replace(/[^A-Za-z0-9_\-.]/g, '_').replace(/^_+|_+$/g, '') || 'execute';
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Slugify a string for use as an OpenAPI path segment.
|
|
104
|
+
* Lowercases and replaces non-alphanumeric runs with hyphens.
|
|
105
|
+
*/
|
|
106
|
+
function slugify(id) {
|
|
107
|
+
return (id
|
|
108
|
+
.toLowerCase()
|
|
109
|
+
.replace(/[^a-z0-9]+/g, '-')
|
|
110
|
+
.replace(/^-|-$/g, '') || 'execute');
|
|
102
111
|
}
|
|
103
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RunnableChart.js","sourceRoot":"","sources":["../../../../src/lib/runner/RunnableChart.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * RunnableChart -- Adds .recorder(), .redact(), .run(), .toOpenAPI(), .toMCPTool()\n * to a FlowChart object.\n *\n * Called by FlowChartBuilder.build() to enrich the compiled chart with\n * d3-style chainable run methods and self-describing outputs.\n */\n\nimport type { FlowChart } from '../builder/types.js';\nimport { zodToJsonSchema } from '../contract/schema.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport type { RunOptions } from '../engine/types.js';\nimport type { Recorder, RedactionPolicy } from '../scope/types.js';\nimport { type RunResult, RunContext } from './RunContext.js';\n\n/** OpenAPI generation options. */\nexport interface ChartOpenAPIOptions {\n  title?: string;\n  version?: string;\n  description?: string;\n  path?: string;\n}\n\n/** MCP tool description. */\nexport interface MCPToolDescription {\n  name: string;\n  description: string;\n  inputSchema?: unknown;\n}\n\n/**\n * FlowChart enriched with d3-style run methods and self-describing outputs.\n *\n * Extends builder.FlowChart (which already carries buildTimeStructure, description,\n * stageDescriptions, inputSchema, outputSchema, outputMapper) and adds the runtime\n * methods attached by makeRunnable().\n */\nexport interface RunnableFlowChart<TOut = any, TScope = any> extends FlowChart<TOut, TScope> {\n  /** Attach a recorder for the next run. Returns a chainable RunContext. */\n  recorder(r: Recorder | FlowRecorder): RunContext<TOut, TScope>;\n  /** Set redaction policy for the next run. Returns a chainable RunContext. */\n  redact(policy: RedactionPolicy): RunContext<TOut, TScope>;\n  /** Execute the chart directly (bare run, no recorders). */\n  run(options?: RunOptions): Promise<RunResult>;\n  /** Generate OpenAPI 3.1 spec from chart metadata + contract. Cached. */\n  toOpenAPI(options?: ChartOpenAPIOptions): object;\n  /** Generate MCP tool description from chart metadata. Cached. */\n  toMCPTool(): MCPToolDescription;\n}\n\n// Caches for describe outputs\nconst openAPICache = new WeakMap<FlowChart, object>();\nconst mcpCache = new WeakMap<FlowChart, MCPToolDescription>();\n\n/**\n * Enrich a FlowChart with run + describe methods.\n * Called by FlowChartBuilder.build().\n */\nexport function makeRunnable<TOut, TScope>(chart: FlowChart<TOut, TScope>): RunnableFlowChart<TOut, TScope> {\n  const runnable = chart as RunnableFlowChart<TOut, TScope>;\n\n  runnable.recorder = function (r: Recorder | FlowRecorder): RunContext<TOut, TScope> {\n    return new RunContext(chart).recorder(r);\n  };\n\n  runnable.redact = function (policy: RedactionPolicy): RunContext<TOut, TScope> {\n    return new RunContext(chart).redact(policy);\n  };\n\n  runnable.run = function (options?: RunOptions): Promise<RunResult> {\n    return new RunContext(chart).run(options);\n  };\n\n  runnable.toOpenAPI = function (options?: ChartOpenAPIOptions): object {\n    const cached = openAPICache.get(chart);\n    if (cached) return cached;\n\n    const builderChart = chart as any;\n    const title = options?.title ?? builderChart.description?.split('\\n')[0] ?? 'API';\n    const version = options?.version ?? '1.0.0';\n    const path = options?.path ?? `/${(builderChart.root?.name ?? 'execute').toLowerCase().replace(/\\s+/g, '-')}`;\n\n    const spec: Record<string, unknown> = {\n      openapi: '3.1.0',\n      info: { title, version, description: options?.description ?? builderChart.description },\n      paths: {\n        [path]: {\n          post: {\n            summary: title,\n            description: builderChart.description,\n            ...(builderChart.inputSchema\n              ? {\n                  requestBody: {\n                    content: { 'application/json': { schema: normalizeSchema(builderChart.inputSchema) } },\n                  },\n                }\n              : {}),\n            responses: {\n              '200': {\n                description: 'Success',\n                ...(builderChart.outputSchema\n                  ? { content: { 'application/json': { schema: normalizeSchema(builderChart.outputSchema) } } }\n                  : {}),\n              },\n            },\n          },\n        },\n      },\n    };\n\n    openAPICache.set(chart, spec);\n    return spec;\n  };\n\n  runnable.toMCPTool = function (): MCPToolDescription {\n    const cached = mcpCache.get(chart);\n    if (cached) return cached;\n\n    const builderChart = chart as any;\n    const name = (builderChart.root?.name ?? 'execute').toLowerCase().replace(/\\s+/g, '_');\n    const description = builderChart.description || `Execute the ${name} flowchart`;\n\n    const tool: MCPToolDescription = {\n      name,\n      description,\n      ...(builderChart.inputSchema ? { inputSchema: normalizeSchema(builderChart.inputSchema) } : {}),\n    };\n\n    mcpCache.set(chart, tool);\n    return tool;\n  };\n\n  return runnable;\n}\n\n/** Normalize a Zod schema or plain JSON Schema to JSON Schema object. */\nfunction normalizeSchema(schema: unknown): unknown {\n  if (!schema) return schema;\n  // If it's a Zod schema with ._def, convert to JSON Schema\n  if (\n    typeof schema === 'object' &&\n    schema !== null &&\n    typeof (schema as Record<string, unknown>)._def !== 'undefined'\n  ) {\n    try {\n      return zodToJsonSchema(schema as Record<string, unknown>);\n    } catch {\n      // If conversion fails (e.g. unsupported Zod type), return as-is\n      return schema;\n    }\n  }\n  return schema;\n}\n"]}
|
|
112
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RunnableChart.js","sourceRoot":"","sources":["../../../../src/lib/runner/RunnableChart.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * RunnableChart -- Adds .recorder(), .redact(), .run(), .toOpenAPI(), .toMCPTool()\n * to a FlowChart object.\n *\n * Called by FlowChartBuilder.build() to enrich the compiled chart with\n * d3-style chainable run methods and self-describing outputs.\n */\n\nimport type { FlowChart } from '../builder/types.js';\nimport { normalizeSchema } from '../contract/schema.js';\nimport type { JsonSchema } from '../contract/types.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport type { RunOptions } from '../engine/types.js';\nimport type { Recorder, RedactionPolicy } from '../scope/types.js';\nimport { type RunResult, RunContext } from './RunContext.js';\n\n/** OpenAPI generation options. */\nexport interface ChartOpenAPIOptions {\n  title?: string;\n  version?: string;\n  description?: string;\n  path?: string;\n}\n\n/** MCP tool description — shape required by the Model Context Protocol spec. */\nexport interface MCPToolDescription {\n  name: string;\n  description: string;\n  /**\n   * JSON Schema object describing the tool's input.\n   * Always present — the MCP spec requires inputSchema even for tools with no parameters.\n   * Defaults to `{ type: 'object', properties: {}, additionalProperties: false }`.\n   */\n  inputSchema: JsonSchema;\n}\n\n/**\n * FlowChart enriched with d3-style run methods and self-describing outputs.\n *\n * Extends builder.FlowChart (which already carries buildTimeStructure, description,\n * stageDescriptions, inputSchema, outputSchema, outputMapper) and adds the runtime\n * methods attached by makeRunnable().\n */\nexport interface RunnableFlowChart<TOut = any, TScope = any> extends FlowChart<TOut, TScope> {\n  /** Attach a recorder for the next run. Returns a chainable RunContext. */\n  recorder(r: Recorder | FlowRecorder): RunContext<TOut, TScope>;\n  /** Set redaction policy for the next run. Returns a chainable RunContext. */\n  redact(policy: RedactionPolicy): RunContext<TOut, TScope>;\n  /** Execute the chart directly (bare run, no recorders). */\n  run(options?: RunOptions): Promise<RunResult>;\n  /** Generate OpenAPI 3.1 spec from chart metadata + contract. Cached for no-options calls. */\n  toOpenAPI(options?: ChartOpenAPIOptions): object;\n  /** Generate MCP tool description from chart metadata. Cached. */\n  toMCPTool(): MCPToolDescription;\n}\n\n// Cache for no-options toOpenAPI() calls only — parameterized calls are not cached\n// because different options produce different output.\nconst openAPICache = new WeakMap<FlowChart, object>();\nconst mcpCache = new WeakMap<FlowChart, MCPToolDescription>();\n\n/**\n * Enrich a FlowChart with run + describe methods.\n * Called by FlowChartBuilder.build().\n */\nexport function makeRunnable<TOut, TScope>(chart: FlowChart<TOut, TScope>): RunnableFlowChart<TOut, TScope> {\n  const runnable = chart as RunnableFlowChart<TOut, TScope>;\n\n  runnable.recorder = function (r: Recorder | FlowRecorder): RunContext<TOut, TScope> {\n    return new RunContext(chart).recorder(r);\n  };\n\n  runnable.redact = function (policy: RedactionPolicy): RunContext<TOut, TScope> {\n    return new RunContext(chart).redact(policy);\n  };\n\n  runnable.run = function (options?: RunOptions): Promise<RunResult> {\n    return new RunContext(chart).run(options);\n  };\n\n  runnable.toOpenAPI = function (options?: ChartOpenAPIOptions): object {\n    // Only cache no-options calls — parameterized calls vary by options\n    if (!options) {\n      const cached = openAPICache.get(chart);\n      if (cached) return cached;\n    }\n\n    const builderChart = chart as any;\n    const title = options?.title ?? builderChart.description?.split('\\n')[0] ?? 'API';\n    const version = options?.version ?? '1.0.0';\n    // Use root.id (the explicit machine-readable id) as the path segment, sanitized\n    const rawId = builderChart.root?.id ?? 'execute';\n    const path = options?.path ?? `/${slugify(rawId)}`;\n\n    const spec: Record<string, unknown> = {\n      openapi: '3.1.0',\n      info: { title, version, description: options?.description ?? builderChart.description },\n      paths: {\n        [path]: {\n          post: {\n            summary: title,\n            description: builderChart.description,\n            ...(builderChart.inputSchema\n              ? {\n                  requestBody: {\n                    content: { 'application/json': { schema: normalizeSchema(builderChart.inputSchema) } },\n                  },\n                }\n              : {}),\n            responses: {\n              '200': {\n                description: 'Success',\n                ...(builderChart.outputSchema\n                  ? { content: { 'application/json': { schema: normalizeSchema(builderChart.outputSchema) } } }\n                  : {}),\n              },\n            },\n          },\n        },\n      },\n    };\n\n    if (!options) openAPICache.set(chart, spec);\n    return spec;\n  };\n\n  runnable.toMCPTool = function (): MCPToolDescription {\n    const cached = mcpCache.get(chart);\n    if (cached) return cached;\n\n    const builderChart = chart as any;\n    // Use root.id (explicit machine-readable id), sanitized to MCP name character allowlist\n    const name = sanitizeMCPName(builderChart.root?.id ?? 'execute');\n    const description = builderChart.description || `Execute the ${name} flowchart`;\n\n    // MCP spec requires inputSchema to always be present.\n    // When no contract is defined, use the recommended no-parameter form.\n    const inputSchema: JsonSchema = builderChart.inputSchema\n      ? normalizeSchema(builderChart.inputSchema)\n      : { type: 'object', properties: {}, additionalProperties: false };\n\n    const tool: MCPToolDescription = { name, description, inputSchema };\n\n    mcpCache.set(chart, tool);\n    return tool;\n  };\n\n  return runnable;\n}\n\n/**\n * Sanitize a string to conform to the MCP tool name character allowlist:\n * [A-Za-z0-9_\\-.] — any other character is replaced with underscore.\n * Trims leading/trailing underscores introduced by replacement.\n */\nfunction sanitizeMCPName(id: string): string {\n  return id.replace(/[^A-Za-z0-9_\\-.]/g, '_').replace(/^_+|_+$/g, '') || 'execute';\n}\n\n/**\n * Slugify a string for use as an OpenAPI path segment.\n * Lowercases and replaces non-alphanumeric runs with hyphens.\n */\nfunction slugify(id: string): string {\n  return (\n    id\n      .toLowerCase()\n      .replace(/[^a-z0-9]+/g, '-')\n      .replace(/^-|-$/g, '') || 'execute'\n  );\n}\n"]}
|
|
@@ -10,7 +10,8 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
10
10
|
exports.makeRunnable = void 0;
|
|
11
11
|
const schema_js_1 = require("../contract/schema.js");
|
|
12
12
|
const RunContext_js_1 = require("./RunContext.js");
|
|
13
|
-
//
|
|
13
|
+
// Cache for no-options toOpenAPI() calls only — parameterized calls are not cached
|
|
14
|
+
// because different options produce different output.
|
|
14
15
|
const openAPICache = new WeakMap();
|
|
15
16
|
const mcpCache = new WeakMap();
|
|
16
17
|
/**
|
|
@@ -30,13 +31,18 @@ function makeRunnable(chart) {
|
|
|
30
31
|
};
|
|
31
32
|
runnable.toOpenAPI = function (options) {
|
|
32
33
|
var _a, _b, _c, _d, _e, _f, _g, _h;
|
|
33
|
-
|
|
34
|
-
if (
|
|
35
|
-
|
|
34
|
+
// Only cache no-options calls — parameterized calls vary by options
|
|
35
|
+
if (!options) {
|
|
36
|
+
const cached = openAPICache.get(chart);
|
|
37
|
+
if (cached)
|
|
38
|
+
return cached;
|
|
39
|
+
}
|
|
36
40
|
const builderChart = chart;
|
|
37
41
|
const title = (_c = (_a = options === null || options === void 0 ? void 0 : options.title) !== null && _a !== void 0 ? _a : (_b = builderChart.description) === null || _b === void 0 ? void 0 : _b.split('\n')[0]) !== null && _c !== void 0 ? _c : 'API';
|
|
38
42
|
const version = (_d = options === null || options === void 0 ? void 0 : options.version) !== null && _d !== void 0 ? _d : '1.0.0';
|
|
39
|
-
|
|
43
|
+
// Use root.id (the explicit machine-readable id) as the path segment, sanitized
|
|
44
|
+
const rawId = (_f = (_e = builderChart.root) === null || _e === void 0 ? void 0 : _e.id) !== null && _f !== void 0 ? _f : 'execute';
|
|
45
|
+
const path = (_g = options === null || options === void 0 ? void 0 : options.path) !== null && _g !== void 0 ? _g : `/${slugify(rawId)}`;
|
|
40
46
|
const spec = {
|
|
41
47
|
openapi: '3.1.0',
|
|
42
48
|
info: { title, version, description: (_h = options === null || options === void 0 ? void 0 : options.description) !== null && _h !== void 0 ? _h : builderChart.description },
|
|
@@ -48,7 +54,7 @@ function makeRunnable(chart) {
|
|
|
48
54
|
...(builderChart.inputSchema
|
|
49
55
|
? {
|
|
50
56
|
requestBody: {
|
|
51
|
-
content: { 'application/json': { schema: normalizeSchema(builderChart.inputSchema) } },
|
|
57
|
+
content: { 'application/json': { schema: (0, schema_js_1.normalizeSchema)(builderChart.inputSchema) } },
|
|
52
58
|
},
|
|
53
59
|
}
|
|
54
60
|
: {}),
|
|
@@ -56,7 +62,7 @@ function makeRunnable(chart) {
|
|
|
56
62
|
'200': {
|
|
57
63
|
description: 'Success',
|
|
58
64
|
...(builderChart.outputSchema
|
|
59
|
-
? { content: { 'application/json': { schema: normalizeSchema(builderChart.outputSchema) } } }
|
|
65
|
+
? { content: { 'application/json': { schema: (0, schema_js_1.normalizeSchema)(builderChart.outputSchema) } } }
|
|
60
66
|
: {}),
|
|
61
67
|
},
|
|
62
68
|
},
|
|
@@ -64,7 +70,8 @@ function makeRunnable(chart) {
|
|
|
64
70
|
},
|
|
65
71
|
},
|
|
66
72
|
};
|
|
67
|
-
|
|
73
|
+
if (!options)
|
|
74
|
+
openAPICache.set(chart, spec);
|
|
68
75
|
return spec;
|
|
69
76
|
};
|
|
70
77
|
runnable.toMCPTool = function () {
|
|
@@ -73,35 +80,37 @@ function makeRunnable(chart) {
|
|
|
73
80
|
if (cached)
|
|
74
81
|
return cached;
|
|
75
82
|
const builderChart = chart;
|
|
76
|
-
|
|
83
|
+
// Use root.id (explicit machine-readable id), sanitized to MCP name character allowlist
|
|
84
|
+
const name = sanitizeMCPName((_b = (_a = builderChart.root) === null || _a === void 0 ? void 0 : _a.id) !== null && _b !== void 0 ? _b : 'execute');
|
|
77
85
|
const description = builderChart.description || `Execute the ${name} flowchart`;
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
86
|
+
// MCP spec requires inputSchema to always be present.
|
|
87
|
+
// When no contract is defined, use the recommended no-parameter form.
|
|
88
|
+
const inputSchema = builderChart.inputSchema
|
|
89
|
+
? (0, schema_js_1.normalizeSchema)(builderChart.inputSchema)
|
|
90
|
+
: { type: 'object', properties: {}, additionalProperties: false };
|
|
91
|
+
const tool = { name, description, inputSchema };
|
|
83
92
|
mcpCache.set(chart, tool);
|
|
84
93
|
return tool;
|
|
85
94
|
};
|
|
86
95
|
return runnable;
|
|
87
96
|
}
|
|
88
97
|
exports.makeRunnable = makeRunnable;
|
|
89
|
-
/**
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
98
|
+
/**
|
|
99
|
+
* Sanitize a string to conform to the MCP tool name character allowlist:
|
|
100
|
+
* [A-Za-z0-9_\-.] — any other character is replaced with underscore.
|
|
101
|
+
* Trims leading/trailing underscores introduced by replacement.
|
|
102
|
+
*/
|
|
103
|
+
function sanitizeMCPName(id) {
|
|
104
|
+
return id.replace(/[^A-Za-z0-9_\-.]/g, '_').replace(/^_+|_+$/g, '') || 'execute';
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* Slugify a string for use as an OpenAPI path segment.
|
|
108
|
+
* Lowercases and replaces non-alphanumeric runs with hyphens.
|
|
109
|
+
*/
|
|
110
|
+
function slugify(id) {
|
|
111
|
+
return (id
|
|
112
|
+
.toLowerCase()
|
|
113
|
+
.replace(/[^a-z0-9]+/g, '-')
|
|
114
|
+
.replace(/^-|-$/g, '') || 'execute');
|
|
106
115
|
}
|
|
107
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RunnableChart.js","sourceRoot":"","sources":["../../../src/lib/runner/RunnableChart.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * RunnableChart -- Adds .recorder(), .redact(), .run(), .toOpenAPI(), .toMCPTool()\n * to a FlowChart object.\n *\n * Called by FlowChartBuilder.build() to enrich the compiled chart with\n * d3-style chainable run methods and self-describing outputs.\n */\n\nimport type { FlowChart } from '../builder/types.js';\nimport { zodToJsonSchema } from '../contract/schema.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport type { RunOptions } from '../engine/types.js';\nimport type { Recorder, RedactionPolicy } from '../scope/types.js';\nimport { type RunResult, RunContext } from './RunContext.js';\n\n/** OpenAPI generation options. */\nexport interface ChartOpenAPIOptions {\n  title?: string;\n  version?: string;\n  description?: string;\n  path?: string;\n}\n\n/** MCP tool description. */\nexport interface MCPToolDescription {\n  name: string;\n  description: string;\n  inputSchema?: unknown;\n}\n\n/**\n * FlowChart enriched with d3-style run methods and self-describing outputs.\n *\n * Extends builder.FlowChart (which already carries buildTimeStructure, description,\n * stageDescriptions, inputSchema, outputSchema, outputMapper) and adds the runtime\n * methods attached by makeRunnable().\n */\nexport interface RunnableFlowChart<TOut = any, TScope = any> extends FlowChart<TOut, TScope> {\n  /** Attach a recorder for the next run. Returns a chainable RunContext. */\n  recorder(r: Recorder | FlowRecorder): RunContext<TOut, TScope>;\n  /** Set redaction policy for the next run. Returns a chainable RunContext. */\n  redact(policy: RedactionPolicy): RunContext<TOut, TScope>;\n  /** Execute the chart directly (bare run, no recorders). */\n  run(options?: RunOptions): Promise<RunResult>;\n  /** Generate OpenAPI 3.1 spec from chart metadata + contract. Cached. */\n  toOpenAPI(options?: ChartOpenAPIOptions): object;\n  /** Generate MCP tool description from chart metadata. Cached. */\n  toMCPTool(): MCPToolDescription;\n}\n\n// Caches for describe outputs\nconst openAPICache = new WeakMap<FlowChart, object>();\nconst mcpCache = new WeakMap<FlowChart, MCPToolDescription>();\n\n/**\n * Enrich a FlowChart with run + describe methods.\n * Called by FlowChartBuilder.build().\n */\nexport function makeRunnable<TOut, TScope>(chart: FlowChart<TOut, TScope>): RunnableFlowChart<TOut, TScope> {\n  const runnable = chart as RunnableFlowChart<TOut, TScope>;\n\n  runnable.recorder = function (r: Recorder | FlowRecorder): RunContext<TOut, TScope> {\n    return new RunContext(chart).recorder(r);\n  };\n\n  runnable.redact = function (policy: RedactionPolicy): RunContext<TOut, TScope> {\n    return new RunContext(chart).redact(policy);\n  };\n\n  runnable.run = function (options?: RunOptions): Promise<RunResult> {\n    return new RunContext(chart).run(options);\n  };\n\n  runnable.toOpenAPI = function (options?: ChartOpenAPIOptions): object {\n    const cached = openAPICache.get(chart);\n    if (cached) return cached;\n\n    const builderChart = chart as any;\n    const title = options?.title ?? builderChart.description?.split('\\n')[0] ?? 'API';\n    const version = options?.version ?? '1.0.0';\n    const path = options?.path ?? `/${(builderChart.root?.name ?? 'execute').toLowerCase().replace(/\\s+/g, '-')}`;\n\n    const spec: Record<string, unknown> = {\n      openapi: '3.1.0',\n      info: { title, version, description: options?.description ?? builderChart.description },\n      paths: {\n        [path]: {\n          post: {\n            summary: title,\n            description: builderChart.description,\n            ...(builderChart.inputSchema\n              ? {\n                  requestBody: {\n                    content: { 'application/json': { schema: normalizeSchema(builderChart.inputSchema) } },\n                  },\n                }\n              : {}),\n            responses: {\n              '200': {\n                description: 'Success',\n                ...(builderChart.outputSchema\n                  ? { content: { 'application/json': { schema: normalizeSchema(builderChart.outputSchema) } } }\n                  : {}),\n              },\n            },\n          },\n        },\n      },\n    };\n\n    openAPICache.set(chart, spec);\n    return spec;\n  };\n\n  runnable.toMCPTool = function (): MCPToolDescription {\n    const cached = mcpCache.get(chart);\n    if (cached) return cached;\n\n    const builderChart = chart as any;\n    const name = (builderChart.root?.name ?? 'execute').toLowerCase().replace(/\\s+/g, '_');\n    const description = builderChart.description || `Execute the ${name} flowchart`;\n\n    const tool: MCPToolDescription = {\n      name,\n      description,\n      ...(builderChart.inputSchema ? { inputSchema: normalizeSchema(builderChart.inputSchema) } : {}),\n    };\n\n    mcpCache.set(chart, tool);\n    return tool;\n  };\n\n  return runnable;\n}\n\n/** Normalize a Zod schema or plain JSON Schema to JSON Schema object. */\nfunction normalizeSchema(schema: unknown): unknown {\n  if (!schema) return schema;\n  // If it's a Zod schema with ._def, convert to JSON Schema\n  if (\n    typeof schema === 'object' &&\n    schema !== null &&\n    typeof (schema as Record<string, unknown>)._def !== 'undefined'\n  ) {\n    try {\n      return zodToJsonSchema(schema as Record<string, unknown>);\n    } catch {\n      // If conversion fails (e.g. unsupported Zod type), return as-is\n      return schema;\n    }\n  }\n  return schema;\n}\n"]}
|
|
116
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RunnableChart.js","sourceRoot":"","sources":["../../../src/lib/runner/RunnableChart.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * RunnableChart -- Adds .recorder(), .redact(), .run(), .toOpenAPI(), .toMCPTool()\n * to a FlowChart object.\n *\n * Called by FlowChartBuilder.build() to enrich the compiled chart with\n * d3-style chainable run methods and self-describing outputs.\n */\n\nimport type { FlowChart } from '../builder/types.js';\nimport { normalizeSchema } from '../contract/schema.js';\nimport type { JsonSchema } from '../contract/types.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport type { RunOptions } from '../engine/types.js';\nimport type { Recorder, RedactionPolicy } from '../scope/types.js';\nimport { type RunResult, RunContext } from './RunContext.js';\n\n/** OpenAPI generation options. */\nexport interface ChartOpenAPIOptions {\n  title?: string;\n  version?: string;\n  description?: string;\n  path?: string;\n}\n\n/** MCP tool description — shape required by the Model Context Protocol spec. */\nexport interface MCPToolDescription {\n  name: string;\n  description: string;\n  /**\n   * JSON Schema object describing the tool's input.\n   * Always present — the MCP spec requires inputSchema even for tools with no parameters.\n   * Defaults to `{ type: 'object', properties: {}, additionalProperties: false }`.\n   */\n  inputSchema: JsonSchema;\n}\n\n/**\n * FlowChart enriched with d3-style run methods and self-describing outputs.\n *\n * Extends builder.FlowChart (which already carries buildTimeStructure, description,\n * stageDescriptions, inputSchema, outputSchema, outputMapper) and adds the runtime\n * methods attached by makeRunnable().\n */\nexport interface RunnableFlowChart<TOut = any, TScope = any> extends FlowChart<TOut, TScope> {\n  /** Attach a recorder for the next run. Returns a chainable RunContext. */\n  recorder(r: Recorder | FlowRecorder): RunContext<TOut, TScope>;\n  /** Set redaction policy for the next run. Returns a chainable RunContext. */\n  redact(policy: RedactionPolicy): RunContext<TOut, TScope>;\n  /** Execute the chart directly (bare run, no recorders). */\n  run(options?: RunOptions): Promise<RunResult>;\n  /** Generate OpenAPI 3.1 spec from chart metadata + contract. Cached for no-options calls. */\n  toOpenAPI(options?: ChartOpenAPIOptions): object;\n  /** Generate MCP tool description from chart metadata. Cached. */\n  toMCPTool(): MCPToolDescription;\n}\n\n// Cache for no-options toOpenAPI() calls only — parameterized calls are not cached\n// because different options produce different output.\nconst openAPICache = new WeakMap<FlowChart, object>();\nconst mcpCache = new WeakMap<FlowChart, MCPToolDescription>();\n\n/**\n * Enrich a FlowChart with run + describe methods.\n * Called by FlowChartBuilder.build().\n */\nexport function makeRunnable<TOut, TScope>(chart: FlowChart<TOut, TScope>): RunnableFlowChart<TOut, TScope> {\n  const runnable = chart as RunnableFlowChart<TOut, TScope>;\n\n  runnable.recorder = function (r: Recorder | FlowRecorder): RunContext<TOut, TScope> {\n    return new RunContext(chart).recorder(r);\n  };\n\n  runnable.redact = function (policy: RedactionPolicy): RunContext<TOut, TScope> {\n    return new RunContext(chart).redact(policy);\n  };\n\n  runnable.run = function (options?: RunOptions): Promise<RunResult> {\n    return new RunContext(chart).run(options);\n  };\n\n  runnable.toOpenAPI = function (options?: ChartOpenAPIOptions): object {\n    // Only cache no-options calls — parameterized calls vary by options\n    if (!options) {\n      const cached = openAPICache.get(chart);\n      if (cached) return cached;\n    }\n\n    const builderChart = chart as any;\n    const title = options?.title ?? builderChart.description?.split('\\n')[0] ?? 'API';\n    const version = options?.version ?? '1.0.0';\n    // Use root.id (the explicit machine-readable id) as the path segment, sanitized\n    const rawId = builderChart.root?.id ?? 'execute';\n    const path = options?.path ?? `/${slugify(rawId)}`;\n\n    const spec: Record<string, unknown> = {\n      openapi: '3.1.0',\n      info: { title, version, description: options?.description ?? builderChart.description },\n      paths: {\n        [path]: {\n          post: {\n            summary: title,\n            description: builderChart.description,\n            ...(builderChart.inputSchema\n              ? {\n                  requestBody: {\n                    content: { 'application/json': { schema: normalizeSchema(builderChart.inputSchema) } },\n                  },\n                }\n              : {}),\n            responses: {\n              '200': {\n                description: 'Success',\n                ...(builderChart.outputSchema\n                  ? { content: { 'application/json': { schema: normalizeSchema(builderChart.outputSchema) } } }\n                  : {}),\n              },\n            },\n          },\n        },\n      },\n    };\n\n    if (!options) openAPICache.set(chart, spec);\n    return spec;\n  };\n\n  runnable.toMCPTool = function (): MCPToolDescription {\n    const cached = mcpCache.get(chart);\n    if (cached) return cached;\n\n    const builderChart = chart as any;\n    // Use root.id (explicit machine-readable id), sanitized to MCP name character allowlist\n    const name = sanitizeMCPName(builderChart.root?.id ?? 'execute');\n    const description = builderChart.description || `Execute the ${name} flowchart`;\n\n    // MCP spec requires inputSchema to always be present.\n    // When no contract is defined, use the recommended no-parameter form.\n    const inputSchema: JsonSchema = builderChart.inputSchema\n      ? normalizeSchema(builderChart.inputSchema)\n      : { type: 'object', properties: {}, additionalProperties: false };\n\n    const tool: MCPToolDescription = { name, description, inputSchema };\n\n    mcpCache.set(chart, tool);\n    return tool;\n  };\n\n  return runnable;\n}\n\n/**\n * Sanitize a string to conform to the MCP tool name character allowlist:\n * [A-Za-z0-9_\\-.] — any other character is replaced with underscore.\n * Trims leading/trailing underscores introduced by replacement.\n */\nfunction sanitizeMCPName(id: string): string {\n  return id.replace(/[^A-Za-z0-9_\\-.]/g, '_').replace(/^_+|_+$/g, '') || 'execute';\n}\n\n/**\n * Slugify a string for use as an OpenAPI path segment.\n * Lowercases and replaces non-alphanumeric runs with hyphens.\n */\nfunction slugify(id: string): string {\n  return (\n    id\n      .toLowerCase()\n      .replace(/[^a-z0-9]+/g, '-')\n      .replace(/^-|-$/g, '') || 'execute'\n  );\n}\n"]}
|
|
@@ -6,6 +6,7 @@
|
|
|
6
6
|
* d3-style chainable run methods and self-describing outputs.
|
|
7
7
|
*/
|
|
8
8
|
import type { FlowChart } from '../builder/types.js';
|
|
9
|
+
import type { JsonSchema } from '../contract/types.js';
|
|
9
10
|
import type { FlowRecorder } from '../engine/narrative/types.js';
|
|
10
11
|
import type { RunOptions } from '../engine/types.js';
|
|
11
12
|
import type { Recorder, RedactionPolicy } from '../scope/types.js';
|
|
@@ -17,11 +18,16 @@ export interface ChartOpenAPIOptions {
|
|
|
17
18
|
description?: string;
|
|
18
19
|
path?: string;
|
|
19
20
|
}
|
|
20
|
-
/** MCP tool description. */
|
|
21
|
+
/** MCP tool description — shape required by the Model Context Protocol spec. */
|
|
21
22
|
export interface MCPToolDescription {
|
|
22
23
|
name: string;
|
|
23
24
|
description: string;
|
|
24
|
-
|
|
25
|
+
/**
|
|
26
|
+
* JSON Schema object describing the tool's input.
|
|
27
|
+
* Always present — the MCP spec requires inputSchema even for tools with no parameters.
|
|
28
|
+
* Defaults to `{ type: 'object', properties: {}, additionalProperties: false }`.
|
|
29
|
+
*/
|
|
30
|
+
inputSchema: JsonSchema;
|
|
25
31
|
}
|
|
26
32
|
/**
|
|
27
33
|
* FlowChart enriched with d3-style run methods and self-describing outputs.
|
|
@@ -37,7 +43,7 @@ export interface RunnableFlowChart<TOut = any, TScope = any> extends FlowChart<T
|
|
|
37
43
|
redact(policy: RedactionPolicy): RunContext<TOut, TScope>;
|
|
38
44
|
/** Execute the chart directly (bare run, no recorders). */
|
|
39
45
|
run(options?: RunOptions): Promise<RunResult>;
|
|
40
|
-
/** Generate OpenAPI 3.1 spec from chart metadata + contract. Cached. */
|
|
46
|
+
/** Generate OpenAPI 3.1 spec from chart metadata + contract. Cached for no-options calls. */
|
|
41
47
|
toOpenAPI(options?: ChartOpenAPIOptions): object;
|
|
42
48
|
/** Generate MCP tool description from chart metadata. Cached. */
|
|
43
49
|
toMCPTool(): MCPToolDescription;
|
package/package.json
CHANGED