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 { zodToJsonSchema } from '../contract/schema.js';
8
+ import { normalizeSchema } from '../contract/schema.js';
9
9
  import { RunContext } from './RunContext.js';
10
- // Caches for describe outputs
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
- const cached = openAPICache.get(chart);
31
- if (cached)
32
- return cached;
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
- const path = (_e = options === null || options === void 0 ? void 0 : options.path) !== null && _e !== void 0 ? _e : `/${((_g = (_f = builderChart.root) === null || _f === void 0 ? void 0 : _f.name) !== null && _g !== void 0 ? _g : 'execute').toLowerCase().replace(/\s+/g, '-')}`;
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
- openAPICache.set(chart, spec);
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
- const name = ((_b = (_a = builderChart.root) === null || _a === void 0 ? void 0 : _a.name) !== null && _b !== void 0 ? _b : 'execute').toLowerCase().replace(/\s+/g, '_');
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
- const tool = {
76
- name,
77
- description,
78
- ...(builderChart.inputSchema ? { inputSchema: normalizeSchema(builderChart.inputSchema) } : {}),
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
- /** Normalize a Zod schema or plain JSON Schema to JSON Schema object. */
86
- function normalizeSchema(schema) {
87
- if (!schema)
88
- return schema;
89
- // If it's a Zod schema with ._def, convert to JSON Schema
90
- if (typeof schema === 'object' &&
91
- schema !== null &&
92
- typeof schema._def !== 'undefined') {
93
- try {
94
- return zodToJsonSchema(schema);
95
- }
96
- catch (_a) {
97
- // If conversion fails (e.g. unsupported Zod type), return as-is
98
- return schema;
99
- }
100
- }
101
- return schema;
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
- // Caches for describe outputs
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
- const cached = openAPICache.get(chart);
34
- if (cached)
35
- return cached;
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
- const path = (_e = options === null || options === void 0 ? void 0 : options.path) !== null && _e !== void 0 ? _e : `/${((_g = (_f = builderChart.root) === null || _f === void 0 ? void 0 : _f.name) !== null && _g !== void 0 ? _g : 'execute').toLowerCase().replace(/\s+/g, '-')}`;
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
- openAPICache.set(chart, spec);
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
- const name = ((_b = (_a = builderChart.root) === null || _a === void 0 ? void 0 : _a.name) !== null && _b !== void 0 ? _b : 'execute').toLowerCase().replace(/\s+/g, '_');
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
- const tool = {
79
- name,
80
- description,
81
- ...(builderChart.inputSchema ? { inputSchema: normalizeSchema(builderChart.inputSchema) } : {}),
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
- /** Normalize a Zod schema or plain JSON Schema to JSON Schema object. */
90
- function normalizeSchema(schema) {
91
- if (!schema)
92
- return schema;
93
- // If it's a Zod schema with ._def, convert to JSON Schema
94
- if (typeof schema === 'object' &&
95
- schema !== null &&
96
- typeof schema._def !== 'undefined') {
97
- try {
98
- return (0, schema_js_1.zodToJsonSchema)(schema);
99
- }
100
- catch (_a) {
101
- // If conversion fails (e.g. unsupported Zod type), return as-is
102
- return schema;
103
- }
104
- }
105
- return schema;
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":";AAAA;;;;;;GAMG;;;AAGH,qDAAwD;AAIxD,mDAA6D;AAqC7D,8BAA8B;AAC9B,MAAM,YAAY,GAAG,IAAI,OAAO,EAAqB,CAAC;AACtD,MAAM,QAAQ,GAAG,IAAI,OAAO,EAAiC,CAAC;AAE9D;;;GAGG;AACH,SAAgB,YAAY,CAAe,KAA8B;IACvE,MAAM,QAAQ,GAAG,KAAwC,CAAC;IAE1D,QAAQ,CAAC,QAAQ,GAAG,UAAU,CAA0B;QACtD,OAAO,IAAI,0BAAU,CAAC,KAAK,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;IAC3C,CAAC,CAAC;IAEF,QAAQ,CAAC,MAAM,GAAG,UAAU,MAAuB;QACjD,OAAO,IAAI,0BAAU,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IAC9C,CAAC,CAAC;IAEF,QAAQ,CAAC,GAAG,GAAG,UAAU,OAAoB;QAC3C,OAAO,IAAI,0BAAU,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAC5C,CAAC,CAAC;IAEF,QAAQ,CAAC,SAAS,GAAG,UAAU,OAA6B;;QAC1D,MAAM,MAAM,GAAG,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACvC,IAAI,MAAM;YAAE,OAAO,MAAM,CAAC;QAE1B,MAAM,YAAY,GAAG,KAAY,CAAC;QAClC,MAAM,KAAK,GAAG,MAAA,MAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,KAAK,mCAAI,MAAA,YAAY,CAAC,WAAW,0CAAE,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC,mCAAI,KAAK,CAAC;QAClF,MAAM,OAAO,GAAG,MAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,OAAO,mCAAI,OAAO,CAAC;QAC5C,MAAM,IAAI,GAAG,MAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,IAAI,mCAAI,IAAI,CAAC,MAAA,MAAA,YAAY,CAAC,IAAI,0CAAE,IAAI,mCAAI,SAAS,CAAC,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,EAAE,CAAC;QAE9G,MAAM,IAAI,GAA4B;YACpC,OAAO,EAAE,OAAO;YAChB,IAAI,EAAE,EAAE,KAAK,EAAE,OAAO,EAAE,WAAW,EAAE,MAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,WAAW,mCAAI,YAAY,CAAC,WAAW,EAAE;YACvF,KAAK,EAAE;gBACL,CAAC,IAAI,CAAC,EAAE;oBACN,IAAI,EAAE;wBACJ,OAAO,EAAE,KAAK;wBACd,WAAW,EAAE,YAAY,CAAC,WAAW;wBACrC,GAAG,CAAC,YAAY,CAAC,WAAW;4BAC1B,CAAC,CAAC;gCACE,WAAW,EAAE;oCACX,OAAO,EAAE,EAAE,kBAAkB,EAAE,EAAE,MAAM,EAAE,eAAe,CAAC,YAAY,CAAC,WAAW,CAAC,EAAE,EAAE;iCACvF;6BACF;4BACH,CAAC,CAAC,EAAE,CAAC;wBACP,SAAS,EAAE;4BACT,KAAK,EAAE;gCACL,WAAW,EAAE,SAAS;gCACtB,GAAG,CAAC,YAAY,CAAC,YAAY;oCAC3B,CAAC,CAAC,EAAE,OAAO,EAAE,EAAE,kBAAkB,EAAE,EAAE,MAAM,EAAE,eAAe,CAAC,YAAY,CAAC,YAAY,CAAC,EAAE,EAAE,EAAE;oCAC7F,CAAC,CAAC,EAAE,CAAC;6BACR;yBACF;qBACF;iBACF;aACF;SACF,CAAC;QAEF,YAAY,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;QAC9B,OAAO,IAAI,CAAC;IACd,CAAC,CAAC;IAEF,QAAQ,CAAC,SAAS,GAAG;;QACnB,MAAM,MAAM,GAAG,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACnC,IAAI,MAAM;YAAE,OAAO,MAAM,CAAC;QAE1B,MAAM,YAAY,GAAG,KAAY,CAAC;QAClC,MAAM,IAAI,GAAG,CAAC,MAAA,MAAA,YAAY,CAAC,IAAI,0CAAE,IAAI,mCAAI,SAAS,CAAC,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;QACvF,MAAM,WAAW,GAAG,YAAY,CAAC,WAAW,IAAI,eAAe,IAAI,YAAY,CAAC;QAEhF,MAAM,IAAI,GAAuB;YAC/B,IAAI;YACJ,WAAW;YACX,GAAG,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC,CAAC,EAAE,WAAW,EAAE,eAAe,CAAC,YAAY,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;SAChG,CAAC;QAEF,QAAQ,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;QAC1B,OAAO,IAAI,CAAC;IACd,CAAC,CAAC;IAEF,OAAO,QAAQ,CAAC;AAClB,CAAC;AA3ED,oCA2EC;AAED,yEAAyE;AACzE,SAAS,eAAe,CAAC,MAAe;IACtC,IAAI,CAAC,MAAM;QAAE,OAAO,MAAM,CAAC;IAC3B,0DAA0D;IAC1D,IACE,OAAO,MAAM,KAAK,QAAQ;QAC1B,MAAM,KAAK,IAAI;QACf,OAAQ,MAAkC,CAAC,IAAI,KAAK,WAAW,EAC/D,CAAC;QACD,IAAI,CAAC;YACH,OAAO,IAAA,2BAAe,EAAC,MAAiC,CAAC,CAAC;QAC5D,CAAC;QAAC,WAAM,CAAC;YACP,gEAAgE;YAChE,OAAO,MAAM,CAAC;QAChB,CAAC;IACH,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC","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
- inputSchema?: unknown;
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
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "footprintjs",
3
- "version": "3.0.16",
3
+ "version": "3.0.17",
4
4
  "description": "Turn your whiteboard flowchart into running code — with automatic causal traces for LLM reasoning",
5
5
  "license": "MIT",
6
6
  "author": "Sanjay Krishna Anbalagan",