footprintjs 3.0.21 → 3.1.0

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.
Files changed (63) hide show
  1. package/CLAUDE.md +2 -0
  2. package/dist/esm/lib/builder/FlowChartBuilder.js +3 -3
  3. package/dist/esm/lib/builder/typedFlowChart.js +16 -5
  4. package/dist/esm/lib/builder/types.js +1 -1
  5. package/dist/esm/lib/contract/defineContract.js +33 -7
  6. package/dist/esm/lib/contract/openapi.js +12 -33
  7. package/dist/esm/lib/contract/types.js +1 -1
  8. package/dist/esm/lib/decide/decide.js +33 -3
  9. package/dist/esm/lib/decide/types.js +1 -1
  10. package/dist/esm/lib/engine/handlers/RuntimeStructureManager.js +19 -7
  11. package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +4 -2
  12. package/dist/esm/lib/engine/narrative/ControlFlowNarrativeGenerator.js +4 -2
  13. package/dist/esm/lib/engine/traversal/FlowchartTraverser.js +325 -266
  14. package/dist/esm/lib/engine/types.js +1 -1
  15. package/dist/esm/lib/memory/StageContext.js +15 -3
  16. package/dist/esm/lib/memory/pathOps.js +31 -3
  17. package/dist/esm/lib/memory/utils.js +30 -10
  18. package/dist/esm/lib/runner/FlowChartExecutor.js +79 -6
  19. package/dist/esm/lib/runner/index.js +1 -1
  20. package/dist/esm/lib/scope/ScopeFacade.js +36 -7
  21. package/dist/esm/lib/scope/providers/baseStateCompatible.js +2 -2
  22. package/dist/esm/lib/scope/providers/types.js +1 -1
  23. package/dist/esm/lib/scope/types.js +1 -1
  24. package/dist/lib/builder/FlowChartBuilder.js +3 -3
  25. package/dist/lib/builder/typedFlowChart.js +16 -5
  26. package/dist/lib/builder/types.js +1 -1
  27. package/dist/lib/contract/defineContract.js +33 -7
  28. package/dist/lib/contract/openapi.js +12 -33
  29. package/dist/lib/contract/types.js +1 -1
  30. package/dist/lib/decide/decide.js +33 -3
  31. package/dist/lib/decide/types.js +1 -1
  32. package/dist/lib/engine/handlers/RuntimeStructureManager.js +19 -7
  33. package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +4 -2
  34. package/dist/lib/engine/narrative/ControlFlowNarrativeGenerator.js +4 -2
  35. package/dist/lib/engine/traversal/FlowchartTraverser.js +325 -266
  36. package/dist/lib/engine/types.js +1 -1
  37. package/dist/lib/memory/StageContext.js +15 -3
  38. package/dist/lib/memory/pathOps.js +31 -3
  39. package/dist/lib/memory/utils.js +30 -10
  40. package/dist/lib/runner/FlowChartExecutor.js +79 -6
  41. package/dist/lib/runner/index.js +1 -1
  42. package/dist/lib/scope/ScopeFacade.js +36 -7
  43. package/dist/lib/scope/providers/baseStateCompatible.js +2 -2
  44. package/dist/lib/scope/providers/types.js +1 -1
  45. package/dist/lib/scope/types.js +1 -1
  46. package/dist/types/lib/builder/typedFlowChart.d.ts +15 -4
  47. package/dist/types/lib/builder/types.d.ts +10 -1
  48. package/dist/types/lib/contract/openapi.d.ts +9 -3
  49. package/dist/types/lib/contract/types.d.ts +13 -2
  50. package/dist/types/lib/decide/decide.d.ts +10 -0
  51. package/dist/types/lib/decide/types.d.ts +20 -0
  52. package/dist/types/lib/engine/handlers/RuntimeStructureManager.d.ts +2 -1
  53. package/dist/types/lib/engine/traversal/FlowchartTraverser.d.ts +47 -0
  54. package/dist/types/lib/engine/types.d.ts +9 -2
  55. package/dist/types/lib/memory/StageContext.d.ts +12 -0
  56. package/dist/types/lib/memory/pathOps.d.ts +26 -2
  57. package/dist/types/lib/memory/utils.d.ts +16 -3
  58. package/dist/types/lib/runner/FlowChartExecutor.d.ts +98 -3
  59. package/dist/types/lib/runner/index.d.ts +1 -0
  60. package/dist/types/lib/scope/ScopeFacade.d.ts +18 -1
  61. package/dist/types/lib/scope/providers/types.d.ts +6 -0
  62. package/dist/types/lib/scope/types.d.ts +7 -1
  63. package/package.json +1 -1
@@ -10,4 +10,4 @@
10
10
  * The builder remains standalone at runtime.
11
11
  */
12
12
  Object.defineProperty(exports, "__esModule", { value: true });
13
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/lib/builder/types.ts"],"names":[],"mappings":";AAAA;;;;;;;;;GASG","sourcesContent":["/**\n * builder/types.ts — All types used by the builder library.\n *\n * Shared types (StageNode, StageFunction, etc.) are imported from the engine.\n * Builder-specific types (FlowChartSpec, FlowChart, SerializedPipelineStructure)\n * are defined locally — they carry builder-only fields (description, outputMapper, etc.).\n *\n * NOTE: All engine imports are `import type` — zero runtime dependency.\n * The builder remains standalone at runtime.\n */\n\nimport type { StageNode } from '../engine/graph/StageNode.js';\nimport type { ILogger, ScopeFactory, StageFunction, TraversalExtractor } from '../engine/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Re-exports from engine (canonical definitions)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type { StageNode } from '../engine/graph/StageNode.js';\nexport type {\n  ILogger,\n  StageFunction,\n  StreamCallback,\n  StreamHandlers,\n  StreamLifecycleHandler,\n  StreamTokenHandler,\n  SubflowMountOptions,\n} from '../engine/types.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Aliases\n// ─────────────────────────────────────────────────────────────────────────────\n\n/** Relaxed-generic alias for builder ergonomics. */\nexport type StageFn = StageFunction<any, any>;\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Scope Protection — canonical definition in scope/protection/types.ts\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type { ScopeProtectionMode };\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Serialized Pipeline Structure (JSON-safe, for visualization)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport interface SerializedPipelineStructure {\n  name: string;\n  id?: string;\n  type: 'stage' | 'decider' | 'fork' | 'streaming';\n  /** Semantic icon hint for visualization (e.g., \"llm\", \"tool\", \"rag\", \"agent\", \"start\") */\n  icon?: string;\n  description?: string;\n  children?: SerializedPipelineStructure[];\n  next?: SerializedPipelineStructure;\n  hasDecider?: boolean;\n  hasSelector?: boolean;\n  branchIds?: string[];\n  loopTarget?: string;\n  isStreaming?: boolean;\n  streamId?: string;\n  isParallelChild?: boolean;\n  parallelGroupId?: string;\n  isSubflowRoot?: boolean;\n  subflowId?: string;\n  subflowName?: string;\n  subflowStructure?: SerializedPipelineStructure;\n  iterationCount?: number;\n  /** True when this subflow uses lazy resolution (deferred until execution). */\n  isLazy?: boolean;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// FlowChartSpec (pure JSON, no functions — for FE→BE transport)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport interface FlowChartSpec {\n  name: string;\n  id?: string;\n  /** Semantic icon hint for visualization (e.g., \"llm\", \"tool\", \"rag\", \"agent\", \"start\") */\n  icon?: string;\n  description?: string;\n  children?: FlowChartSpec[];\n  next?: FlowChartSpec;\n  hasDecider?: boolean;\n  hasSelector?: boolean;\n  branchIds?: string[];\n  loopTarget?: string;\n  isStreaming?: boolean;\n  streamId?: string;\n  isParallelChild?: boolean;\n  parallelGroupId?: string;\n  isSubflowRoot?: boolean;\n  subflowId?: string;\n  subflowName?: string;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Build-Time Extractor\n// ─────────────────────────────────────────────────────────────────────────────\n\n/** Metadata provided to the build-time extractor for each node. */\nexport type BuildTimeNodeMetadata = FlowChartSpec;\n\nexport type BuildTimeExtractor<TResult = FlowChartSpec> = (metadata: BuildTimeNodeMetadata) => TResult;\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Traversal Extractor (runtime) — canonical definition lives in engine/types.\n// Re-exported here so callers don't need two import paths.\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type { TraversalExtractor } from '../engine/types.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// FlowChart — compiled output of build()\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type FlowChart<TOut = any, TScope = any> = {\n  root: StageNode<TOut, TScope>;\n  stageMap: Map<string, StageFunction<TOut, TScope>>;\n  extractor?: TraversalExtractor;\n  subflows?: Record<string, { root: StageNode<TOut, TScope> }>;\n  buildTimeStructure: SerializedPipelineStructure;\n  enableNarrative?: boolean;\n  logger?: ILogger;\n  description: string;\n  stageDescriptions: Map<string, string>;\n  /** Input schema (Zod or JSON Schema) — declared via setInputSchema() or .contract(). */\n  inputSchema?: unknown;\n  /** Output schema (Zod or JSON Schema) — declared via setOutputSchema() or .contract(). */\n  outputSchema?: unknown;\n  /** Output mapper — extracts response from final scope. */\n  outputMapper?: (finalScope: Record<string, unknown>) => unknown;\n  /** Scope factory — auto-embedded by flowChart<T>(). Executor reads this if no factory param. */\n  scopeFactory?: ScopeFactory<TScope>;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Simplified Parallel Spec (for addListOfFunction)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type SimplifiedParallelSpec<TOut = any, TScope = any> = {\n  id: string;\n  name: string;\n  fn?: StageFunction<TOut, TScope>;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n// ExecOptions (for execute() convenience — used by runner layer)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type ExecOptions = {\n  defaults?: unknown;\n  initial?: unknown;\n  readOnly?: unknown;\n  throttlingErrorChecker?: (e: unknown) => boolean;\n  scopeProtectionMode?: ScopeProtectionMode;\n  enableNarrative?: boolean;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n// SubflowRef\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport interface SubflowRef {\n  $ref: string;\n  mountId: string;\n}\n"]}
13
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/lib/builder/types.ts"],"names":[],"mappings":";AAAA;;;;;;;;;GASG","sourcesContent":["/**\n * builder/types.ts — All types used by the builder library.\n *\n * Shared types (StageNode, StageFunction, etc.) are imported from the engine.\n * Builder-specific types (FlowChartSpec, FlowChart, SerializedPipelineStructure)\n * are defined locally — they carry builder-only fields (description, outputMapper, etc.).\n *\n * NOTE: All engine imports are `import type` — zero runtime dependency.\n * The builder remains standalone at runtime.\n */\n\nimport type { StageNode } from '../engine/graph/StageNode.js';\nimport type { ILogger, ScopeFactory, StageFunction, TraversalExtractor } from '../engine/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Re-exports from engine (canonical definitions)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type { StageNode } from '../engine/graph/StageNode.js';\nexport type {\n  ILogger,\n  StageFunction,\n  StreamCallback,\n  StreamHandlers,\n  StreamLifecycleHandler,\n  StreamTokenHandler,\n  SubflowMountOptions,\n} from '../engine/types.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Aliases\n// ─────────────────────────────────────────────────────────────────────────────\n\n/** Relaxed-generic alias for builder ergonomics. */\nexport type StageFn = StageFunction<any, any>;\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Scope Protection — canonical definition in scope/protection/types.ts\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type { ScopeProtectionMode };\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Serialized Pipeline Structure (JSON-safe, for visualization)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport interface SerializedPipelineStructure {\n  name: string;\n  id?: string;\n  type: 'stage' | 'decider' | 'selector' | 'fork' | 'streaming';\n  /** Semantic icon hint for visualization (e.g., \"llm\", \"tool\", \"rag\", \"agent\", \"start\") */\n  icon?: string;\n  description?: string;\n  children?: SerializedPipelineStructure[];\n  next?: SerializedPipelineStructure;\n  hasDecider?: boolean;\n  hasSelector?: boolean;\n  branchIds?: string[];\n  loopTarget?: string;\n  isStreaming?: boolean;\n  streamId?: string;\n  isParallelChild?: boolean;\n  parallelGroupId?: string;\n  isSubflowRoot?: boolean;\n  subflowId?: string;\n  subflowName?: string;\n  /**\n   * Nested pipeline structure for a subflow node.\n   * WARNING: Any future walker that traverses this field recursively must apply its own\n   * depth guard (see MAX_WALK_DEPTH in contract/openapi.ts). The current `buildDescription`\n   * walk in openapi.ts does NOT traverse subflowStructure — if it ever does, the depth\n   * guard must cover both the `next` chain and this nested structure.\n   */\n  subflowStructure?: SerializedPipelineStructure;\n  iterationCount?: number;\n  /** True when this subflow uses lazy resolution (deferred until execution). */\n  isLazy?: boolean;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// FlowChartSpec (pure JSON, no functions — for FE→BE transport)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport interface FlowChartSpec {\n  name: string;\n  id?: string;\n  /** Node type — matches `SerializedPipelineStructure.type` for visualization alignment. */\n  type?: 'stage' | 'decider' | 'selector' | 'fork' | 'streaming';\n  /** Semantic icon hint for visualization (e.g., \"llm\", \"tool\", \"rag\", \"agent\", \"start\") */\n  icon?: string;\n  description?: string;\n  children?: FlowChartSpec[];\n  next?: FlowChartSpec;\n  hasDecider?: boolean;\n  hasSelector?: boolean;\n  branchIds?: string[];\n  loopTarget?: string;\n  isStreaming?: boolean;\n  streamId?: string;\n  isParallelChild?: boolean;\n  parallelGroupId?: string;\n  isSubflowRoot?: boolean;\n  subflowId?: string;\n  subflowName?: string;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Build-Time Extractor\n// ─────────────────────────────────────────────────────────────────────────────\n\n/** Metadata provided to the build-time extractor for each node. */\nexport type BuildTimeNodeMetadata = FlowChartSpec;\n\nexport type BuildTimeExtractor<TResult = FlowChartSpec> = (metadata: BuildTimeNodeMetadata) => TResult;\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Traversal Extractor (runtime) — canonical definition lives in engine/types.\n// Re-exported here so callers don't need two import paths.\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type { TraversalExtractor } from '../engine/types.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// FlowChart — compiled output of build()\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type FlowChart<TOut = any, TScope = any> = {\n  root: StageNode<TOut, TScope>;\n  stageMap: Map<string, StageFunction<TOut, TScope>>;\n  extractor?: TraversalExtractor;\n  subflows?: Record<string, { root: StageNode<TOut, TScope> }>;\n  buildTimeStructure: SerializedPipelineStructure;\n  enableNarrative?: boolean;\n  logger?: ILogger;\n  description: string;\n  stageDescriptions: Map<string, string>;\n  /** Input schema (Zod or JSON Schema) — declared via setInputSchema() or .contract(). */\n  inputSchema?: unknown;\n  /** Output schema (Zod or JSON Schema) — declared via setOutputSchema() or .contract(). */\n  outputSchema?: unknown;\n  /** Output mapper — extracts response from final scope. */\n  outputMapper?: (finalScope: Record<string, unknown>) => unknown;\n  /** Scope factory — auto-embedded by flowChart<T>(). Executor reads this if no factory param. */\n  scopeFactory?: ScopeFactory<TScope>;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Simplified Parallel Spec (for addListOfFunction)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type SimplifiedParallelSpec<TOut = any, TScope = any> = {\n  id: string;\n  name: string;\n  fn?: StageFunction<TOut, TScope>;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n// ExecOptions (for execute() convenience — used by runner layer)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport type ExecOptions = {\n  defaults?: unknown;\n  initial?: unknown;\n  readOnly?: unknown;\n  throttlingErrorChecker?: (e: unknown) => boolean;\n  scopeProtectionMode?: ScopeProtectionMode;\n  enableNarrative?: boolean;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n// SubflowRef\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport interface SubflowRef {\n  $ref: string;\n  mountId: string;\n}\n"]}
@@ -21,14 +21,40 @@ const schema_js_1 = require("./schema.js");
21
21
  function defineContract(chart, options) {
22
22
  const inputSchema = options.inputSchema ? (0, schema_js_1.normalizeSchema)(options.inputSchema) : undefined;
23
23
  const outputSchema = options.outputSchema ? (0, schema_js_1.normalizeSchema)(options.outputSchema) : undefined;
24
- // Propagate original schema to chart for runtime validation (if chart doesn't have one).
25
- // Contract schemas are normalized to JSON Schema for OpenAPI, but the chart needs the
26
- // original Zod schema to call .safeParse() at runtime in FlowChartExecutor.run().
27
- if (options.inputSchema && !chart.inputSchema) {
28
- chart.inputSchema = options.inputSchema;
24
+ // Build a lightweight chart view for the contract.
25
+ //
26
+ // We must NOT mutate the original `chart` object because charts are compiled
27
+ // artifacts meant to be shared across multiple concurrent executors. Mutating
28
+ // any schema field after build would be visible to all holders of that chart
29
+ // reference, causing cross-contract contamination.
30
+ //
31
+ // Instead, use Object.create(chart) to create a prototype-linked view:
32
+ // - All properties (root, stageMap, subflows, methods…) are inherited via
33
+ // the prototype chain — zero extra copying.
34
+ // - Setting schema fields on the view creates OWN properties that shadow
35
+ // the prototype's value, leaving the original chart untouched.
36
+ // - FlowChartExecutor reads chartView.inputSchema which resolves to the
37
+ // own-property (contract schema) before the prototype (builder schema).
38
+ // - RunContext reads chartView.outputMapper to apply the output transform —
39
+ // that must also be shadowed here so the contract's mapper wins.
40
+ //
41
+ // Limitation: Object.keys(chartView) returns only own properties (the schema
42
+ // fields that were shadowed). Do NOT use Object.keys(), spread ({...chartView}),
43
+ // or JSON.stringify(chartView) on this view — use named property access or
44
+ // chart.toSpec() instead.
45
+ const chartView = Object.create(chart);
46
+ const view = chartView;
47
+ if (options.inputSchema) {
48
+ view.inputSchema = options.inputSchema;
49
+ }
50
+ if (options.outputSchema) {
51
+ view.outputSchema = options.outputSchema;
52
+ }
53
+ if (options.outputMapper) {
54
+ view.outputMapper = options.outputMapper;
29
55
  }
30
56
  const contract = {
31
- chart,
57
+ chart: chartView,
32
58
  inputSchema,
33
59
  outputSchema,
34
60
  outputMapper: options.outputMapper,
@@ -39,4 +65,4 @@ function defineContract(chart, options) {
39
65
  return contract;
40
66
  }
41
67
  exports.defineContract = defineContract;
42
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVmaW5lQ29udHJhY3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL2NvbnRyYWN0L2RlZmluZUNvbnRyYWN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7Ozs7Ozs7Ozs7R0FjRzs7O0FBR0gsNkNBQStDO0FBQy9DLDJDQUE4QztBQUc5QyxTQUFnQixjQUFjLENBQzVCLEtBQWdCLEVBQ2hCLE9BQWtEO0lBRWxELE1BQU0sV0FBVyxHQUFHLE9BQU8sQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDLElBQUEsMkJBQWUsRUFBQyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQztJQUMzRixNQUFNLFlBQVksR0FBRyxPQUFPLENBQUMsWUFBWSxDQUFDLENBQUMsQ0FBQyxJQUFBLDJCQUFlLEVBQUMsT0FBTyxDQUFDLFlBQVksQ0FBQyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUM7SUFFOUYseUZBQXlGO0lBQ3pGLHNGQUFzRjtJQUN0RixrRkFBa0Y7SUFDbEYsSUFBSSxPQUFPLENBQUMsV0FBVyxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzdDLEtBQW1DLENBQUMsV0FBVyxHQUFHLE9BQU8sQ0FBQyxXQUFXLENBQUM7SUFDekUsQ0FBQztJQUVELE1BQU0sUUFBUSxHQUF1QztRQUNuRCxLQUFLO1FBQ0wsV0FBVztRQUNYLFlBQVk7UUFDWixZQUFZLEVBQUUsT0FBTyxDQUFDLFlBQVk7UUFDbEMsU0FBUyxDQUFDLFVBQTJCO1lBQ25DLE9BQU8sSUFBQSw0QkFBZSxFQUFDLFFBQVEsRUFBRSxVQUFVLENBQUMsQ0FBQztRQUMvQyxDQUFDO0tBQ0YsQ0FBQztJQUVGLE9BQU8sUUFBUSxDQUFDO0FBQ2xCLENBQUM7QUF6QkQsd0NBeUJDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBjb250cmFjdC9kZWZpbmVDb250cmFjdC50cyDigJQgRmFjdG9yeSBmb3IgY3JlYXRpbmcgYSBGbG93Q2hhcnRDb250cmFjdC5cbiAqXG4gKiBXcmFwcyBhIGNvbXBpbGVkIEZsb3dDaGFydCB3aXRoIEkvTyBzY2hlbWFzIGFuZCBhbiBvdXRwdXQgbWFwcGVyLFxuICogdXNpbmcgdGhlIHNhbWUgcGF0dGVybiBhcyBTdWJmbG93TW91bnRPcHRpb25zIChpbnB1dE1hcHBlci9vdXRwdXRNYXBwZXIpLlxuICpcbiAqIFVzYWdlOlxuICogICBjb25zdCBjb250cmFjdCA9IGRlZmluZUNvbnRyYWN0KGNoYXJ0LCB7XG4gKiAgICAgaW5wdXRTY2hlbWE6IHoub2JqZWN0KHsgbmFtZTogei5zdHJpbmcoKSB9KSxcbiAqICAgICBvdXRwdXRTY2hlbWE6IHoub2JqZWN0KHsgZ3JlZXRpbmc6IHouc3RyaW5nKCkgfSksXG4gKiAgICAgb3V0cHV0TWFwcGVyOiAoc2NvcGUpID0+ICh7IGdyZWV0aW5nOiBzY29wZS5tZXNzYWdlIGFzIHN0cmluZyB9KSxcbiAqICAgfSk7XG4gKlxuICogICBjb25zdCBvcGVuYXBpID0gY29udHJhY3QudG9PcGVuQVBJKCk7XG4gKi9cblxuaW1wb3J0IHR5cGUgeyBGbG93Q2hhcnQgfSBmcm9tICcuLi9idWlsZGVyL3R5cGVzLmpzJztcbmltcG9ydCB7IGdlbmVyYXRlT3BlbkFQSSB9IGZyb20gJy4vb3BlbmFwaS5qcyc7XG5pbXBvcnQgeyBub3JtYWxpemVTY2hlbWEgfSBmcm9tICcuL3NjaGVtYS5qcyc7XG5pbXBvcnQgdHlwZSB7IEZsb3dDaGFydENvbnRyYWN0LCBGbG93Q2hhcnRDb250cmFjdE9wdGlvbnMsIE9wZW5BUElPcHRpb25zLCBPcGVuQVBJU3BlYyB9IGZyb20gJy4vdHlwZXMuanMnO1xuXG5leHBvcnQgZnVuY3Rpb24gZGVmaW5lQ29udHJhY3Q8VElucHV0ID0gdW5rbm93biwgVE91dHB1dCA9IHVua25vd24+KFxuICBjaGFydDogRmxvd0NoYXJ0LFxuICBvcHRpb25zOiBGbG93Q2hhcnRDb250cmFjdE9wdGlvbnM8VElucHV0LCBUT3V0cHV0Pixcbik6IEZsb3dDaGFydENvbnRyYWN0PFRJbnB1dCwgVE91dHB1dD4ge1xuICBjb25zdCBpbnB1dFNjaGVtYSA9IG9wdGlvbnMuaW5wdXRTY2hlbWEgPyBub3JtYWxpemVTY2hlbWEob3B0aW9ucy5pbnB1dFNjaGVtYSkgOiB1bmRlZmluZWQ7XG4gIGNvbnN0IG91dHB1dFNjaGVtYSA9IG9wdGlvbnMub3V0cHV0U2NoZW1hID8gbm9ybWFsaXplU2NoZW1hKG9wdGlvbnMub3V0cHV0U2NoZW1hKSA6IHVuZGVmaW5lZDtcblxuICAvLyBQcm9wYWdhdGUgb3JpZ2luYWwgc2NoZW1hIHRvIGNoYXJ0IGZvciBydW50aW1lIHZhbGlkYXRpb24gKGlmIGNoYXJ0IGRvZXNuJ3QgaGF2ZSBvbmUpLlxuICAvLyBDb250cmFjdCBzY2hlbWFzIGFyZSBub3JtYWxpemVkIHRvIEpTT04gU2NoZW1hIGZvciBPcGVuQVBJLCBidXQgdGhlIGNoYXJ0IG5lZWRzIHRoZVxuICAvLyBvcmlnaW5hbCBab2Qgc2NoZW1hIHRvIGNhbGwgLnNhZmVQYXJzZSgpIGF0IHJ1bnRpbWUgaW4gRmxvd0NoYXJ0RXhlY3V0b3IucnVuKCkuXG4gIGlmIChvcHRpb25zLmlucHV0U2NoZW1hICYmICFjaGFydC5pbnB1dFNjaGVtYSkge1xuICAgIChjaGFydCBhcyB7IGlucHV0U2NoZW1hPzogdW5rbm93biB9KS5pbnB1dFNjaGVtYSA9IG9wdGlvbnMuaW5wdXRTY2hlbWE7XG4gIH1cblxuICBjb25zdCBjb250cmFjdDogRmxvd0NoYXJ0Q29udHJhY3Q8VElucHV0LCBUT3V0cHV0PiA9IHtcbiAgICBjaGFydCxcbiAgICBpbnB1dFNjaGVtYSxcbiAgICBvdXRwdXRTY2hlbWEsXG4gICAgb3V0cHV0TWFwcGVyOiBvcHRpb25zLm91dHB1dE1hcHBlcixcbiAgICB0b09wZW5BUEkoYXBpT3B0aW9ucz86IE9wZW5BUElPcHRpb25zKTogT3BlbkFQSVNwZWMge1xuICAgICAgcmV0dXJuIGdlbmVyYXRlT3BlbkFQSShjb250cmFjdCwgYXBpT3B0aW9ucyk7XG4gICAgfSxcbiAgfTtcblxuICByZXR1cm4gY29udHJhY3Q7XG59XG4iXX0=
68
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVmaW5lQ29udHJhY3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL2NvbnRyYWN0L2RlZmluZUNvbnRyYWN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7Ozs7Ozs7Ozs7R0FjRzs7O0FBR0gsNkNBQStDO0FBQy9DLDJDQUE4QztBQUc5QyxTQUFnQixjQUFjLENBQzVCLEtBQWdCLEVBQ2hCLE9BQWtEO0lBRWxELE1BQU0sV0FBVyxHQUFHLE9BQU8sQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDLElBQUEsMkJBQWUsRUFBQyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQztJQUMzRixNQUFNLFlBQVksR0FBRyxPQUFPLENBQUMsWUFBWSxDQUFDLENBQUMsQ0FBQyxJQUFBLDJCQUFlLEVBQUMsT0FBTyxDQUFDLFlBQVksQ0FBQyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUM7SUFFOUYsbURBQW1EO0lBQ25ELEVBQUU7SUFDRiw2RUFBNkU7SUFDN0UsOEVBQThFO0lBQzlFLDZFQUE2RTtJQUM3RSxtREFBbUQ7SUFDbkQsRUFBRTtJQUNGLHVFQUF1RTtJQUN2RSw0RUFBNEU7SUFDNUUsZ0RBQWdEO0lBQ2hELDJFQUEyRTtJQUMzRSxtRUFBbUU7SUFDbkUsMEVBQTBFO0lBQzFFLDRFQUE0RTtJQUM1RSw4RUFBOEU7SUFDOUUscUVBQXFFO0lBQ3JFLEVBQUU7SUFDRiw2RUFBNkU7SUFDN0UsaUZBQWlGO0lBQ2pGLDJFQUEyRTtJQUMzRSwwQkFBMEI7SUFDMUIsTUFBTSxTQUFTLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQWMsQ0FBQztJQUNwRCxNQUFNLElBQUksR0FBRyxTQUErQixDQUFDO0lBQzdDLElBQUksT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQ3hCLElBQUksQ0FBQyxXQUFXLEdBQUcsT0FBTyxDQUFDLFdBQVcsQ0FBQztJQUN6QyxDQUFDO0lBQ0QsSUFBSSxPQUFPLENBQUMsWUFBWSxFQUFFLENBQUM7UUFDekIsSUFBSSxDQUFDLFlBQVksR0FBRyxPQUFPLENBQUMsWUFBWSxDQUFDO0lBQzNDLENBQUM7SUFDRCxJQUFJLE9BQU8sQ0FBQyxZQUFZLEVBQUUsQ0FBQztRQUN6QixJQUFJLENBQUMsWUFBWSxHQUFHLE9BQU8sQ0FBQyxZQUFxRSxDQUFDO0lBQ3BHLENBQUM7SUFFRCxNQUFNLFFBQVEsR0FBdUM7UUFDbkQsS0FBSyxFQUFFLFNBQVM7UUFDaEIsV0FBVztRQUNYLFlBQVk7UUFDWixZQUFZLEVBQUUsT0FBTyxDQUFDLFlBQVk7UUFDbEMsU0FBUyxDQUFDLFVBQTJCO1lBQ25DLE9BQU8sSUFBQSw0QkFBZSxFQUFDLFFBQVEsRUFBRSxVQUFVLENBQUMsQ0FBQztRQUMvQyxDQUFDO0tBQ0YsQ0FBQztJQUVGLE9BQU8sUUFBUSxDQUFDO0FBQ2xCLENBQUM7QUFuREQsd0NBbURDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBjb250cmFjdC9kZWZpbmVDb250cmFjdC50cyDigJQgRmFjdG9yeSBmb3IgY3JlYXRpbmcgYSBGbG93Q2hhcnRDb250cmFjdC5cbiAqXG4gKiBXcmFwcyBhIGNvbXBpbGVkIEZsb3dDaGFydCB3aXRoIEkvTyBzY2hlbWFzIGFuZCBhbiBvdXRwdXQgbWFwcGVyLFxuICogdXNpbmcgdGhlIHNhbWUgcGF0dGVybiBhcyBTdWJmbG93TW91bnRPcHRpb25zIChpbnB1dE1hcHBlci9vdXRwdXRNYXBwZXIpLlxuICpcbiAqIFVzYWdlOlxuICogICBjb25zdCBjb250cmFjdCA9IGRlZmluZUNvbnRyYWN0KGNoYXJ0LCB7XG4gKiAgICAgaW5wdXRTY2hlbWE6IHoub2JqZWN0KHsgbmFtZTogei5zdHJpbmcoKSB9KSxcbiAqICAgICBvdXRwdXRTY2hlbWE6IHoub2JqZWN0KHsgZ3JlZXRpbmc6IHouc3RyaW5nKCkgfSksXG4gKiAgICAgb3V0cHV0TWFwcGVyOiAoc2NvcGUpID0+ICh7IGdyZWV0aW5nOiBzY29wZS5tZXNzYWdlIGFzIHN0cmluZyB9KSxcbiAqICAgfSk7XG4gKlxuICogICBjb25zdCBvcGVuYXBpID0gY29udHJhY3QudG9PcGVuQVBJKCk7XG4gKi9cblxuaW1wb3J0IHR5cGUgeyBGbG93Q2hhcnQgfSBmcm9tICcuLi9idWlsZGVyL3R5cGVzLmpzJztcbmltcG9ydCB7IGdlbmVyYXRlT3BlbkFQSSB9IGZyb20gJy4vb3BlbmFwaS5qcyc7XG5pbXBvcnQgeyBub3JtYWxpemVTY2hlbWEgfSBmcm9tICcuL3NjaGVtYS5qcyc7XG5pbXBvcnQgdHlwZSB7IEZsb3dDaGFydENvbnRyYWN0LCBGbG93Q2hhcnRDb250cmFjdE9wdGlvbnMsIE9wZW5BUElPcHRpb25zLCBPcGVuQVBJU3BlYyB9IGZyb20gJy4vdHlwZXMuanMnO1xuXG5leHBvcnQgZnVuY3Rpb24gZGVmaW5lQ29udHJhY3Q8VElucHV0ID0gdW5rbm93biwgVE91dHB1dCA9IHVua25vd24+KFxuICBjaGFydDogRmxvd0NoYXJ0LFxuICBvcHRpb25zOiBGbG93Q2hhcnRDb250cmFjdE9wdGlvbnM8VElucHV0LCBUT3V0cHV0Pixcbik6IEZsb3dDaGFydENvbnRyYWN0PFRJbnB1dCwgVE91dHB1dD4ge1xuICBjb25zdCBpbnB1dFNjaGVtYSA9IG9wdGlvbnMuaW5wdXRTY2hlbWEgPyBub3JtYWxpemVTY2hlbWEob3B0aW9ucy5pbnB1dFNjaGVtYSkgOiB1bmRlZmluZWQ7XG4gIGNvbnN0IG91dHB1dFNjaGVtYSA9IG9wdGlvbnMub3V0cHV0U2NoZW1hID8gbm9ybWFsaXplU2NoZW1hKG9wdGlvbnMub3V0cHV0U2NoZW1hKSA6IHVuZGVmaW5lZDtcblxuICAvLyBCdWlsZCBhIGxpZ2h0d2VpZ2h0IGNoYXJ0IHZpZXcgZm9yIHRoZSBjb250cmFjdC5cbiAgLy9cbiAgLy8gV2UgbXVzdCBOT1QgbXV0YXRlIHRoZSBvcmlnaW5hbCBgY2hhcnRgIG9iamVjdCBiZWNhdXNlIGNoYXJ0cyBhcmUgY29tcGlsZWRcbiAgLy8gYXJ0aWZhY3RzIG1lYW50IHRvIGJlIHNoYXJlZCBhY3Jvc3MgbXVsdGlwbGUgY29uY3VycmVudCBleGVjdXRvcnMuIE11dGF0aW5nXG4gIC8vIGFueSBzY2hlbWEgZmllbGQgYWZ0ZXIgYnVpbGQgd291bGQgYmUgdmlzaWJsZSB0byBhbGwgaG9sZGVycyBvZiB0aGF0IGNoYXJ0XG4gIC8vIHJlZmVyZW5jZSwgY2F1c2luZyBjcm9zcy1jb250cmFjdCBjb250YW1pbmF0aW9uLlxuICAvL1xuICAvLyBJbnN0ZWFkLCB1c2UgT2JqZWN0LmNyZWF0ZShjaGFydCkgdG8gY3JlYXRlIGEgcHJvdG90eXBlLWxpbmtlZCB2aWV3OlxuICAvLyAgIC0gQWxsIHByb3BlcnRpZXMgKHJvb3QsIHN0YWdlTWFwLCBzdWJmbG93cywgbWV0aG9kc+KApikgYXJlIGluaGVyaXRlZCB2aWFcbiAgLy8gICAgIHRoZSBwcm90b3R5cGUgY2hhaW4g4oCUIHplcm8gZXh0cmEgY29weWluZy5cbiAgLy8gICAtIFNldHRpbmcgc2NoZW1hIGZpZWxkcyBvbiB0aGUgdmlldyBjcmVhdGVzIE9XTiBwcm9wZXJ0aWVzIHRoYXQgc2hhZG93XG4gIC8vICAgICB0aGUgcHJvdG90eXBlJ3MgdmFsdWUsIGxlYXZpbmcgdGhlIG9yaWdpbmFsIGNoYXJ0IHVudG91Y2hlZC5cbiAgLy8gICAtIEZsb3dDaGFydEV4ZWN1dG9yIHJlYWRzIGNoYXJ0Vmlldy5pbnB1dFNjaGVtYSB3aGljaCByZXNvbHZlcyB0byB0aGVcbiAgLy8gICAgIG93bi1wcm9wZXJ0eSAoY29udHJhY3Qgc2NoZW1hKSBiZWZvcmUgdGhlIHByb3RvdHlwZSAoYnVpbGRlciBzY2hlbWEpLlxuICAvLyAgIC0gUnVuQ29udGV4dCByZWFkcyBjaGFydFZpZXcub3V0cHV0TWFwcGVyIHRvIGFwcGx5IHRoZSBvdXRwdXQgdHJhbnNmb3JtIOKAlFxuICAvLyAgICAgdGhhdCBtdXN0IGFsc28gYmUgc2hhZG93ZWQgaGVyZSBzbyB0aGUgY29udHJhY3QncyBtYXBwZXIgd2lucy5cbiAgLy9cbiAgLy8gTGltaXRhdGlvbjogT2JqZWN0LmtleXMoY2hhcnRWaWV3KSByZXR1cm5zIG9ubHkgb3duIHByb3BlcnRpZXMgKHRoZSBzY2hlbWFcbiAgLy8gZmllbGRzIHRoYXQgd2VyZSBzaGFkb3dlZCkuIERvIE5PVCB1c2UgT2JqZWN0LmtleXMoKSwgc3ByZWFkICh7Li4uY2hhcnRWaWV3fSksXG4gIC8vIG9yIEpTT04uc3RyaW5naWZ5KGNoYXJ0Vmlldykgb24gdGhpcyB2aWV3IOKAlCB1c2UgbmFtZWQgcHJvcGVydHkgYWNjZXNzIG9yXG4gIC8vIGNoYXJ0LnRvU3BlYygpIGluc3RlYWQuXG4gIGNvbnN0IGNoYXJ0VmlldyA9IE9iamVjdC5jcmVhdGUoY2hhcnQpIGFzIEZsb3dDaGFydDtcbiAgY29uc3QgdmlldyA9IGNoYXJ0VmlldyBhcyBQYXJ0aWFsPEZsb3dDaGFydD47XG4gIGlmIChvcHRpb25zLmlucHV0U2NoZW1hKSB7XG4gICAgdmlldy5pbnB1dFNjaGVtYSA9IG9wdGlvbnMuaW5wdXRTY2hlbWE7XG4gIH1cbiAgaWYgKG9wdGlvbnMub3V0cHV0U2NoZW1hKSB7XG4gICAgdmlldy5vdXRwdXRTY2hlbWEgPSBvcHRpb25zLm91dHB1dFNjaGVtYTtcbiAgfVxuICBpZiAob3B0aW9ucy5vdXRwdXRNYXBwZXIpIHtcbiAgICB2aWV3Lm91dHB1dE1hcHBlciA9IG9wdGlvbnMub3V0cHV0TWFwcGVyIGFzICgoczogUmVjb3JkPHN0cmluZywgdW5rbm93bj4pID0+IHVua25vd24pIHwgdW5kZWZpbmVkO1xuICB9XG5cbiAgY29uc3QgY29udHJhY3Q6IEZsb3dDaGFydENvbnRyYWN0PFRJbnB1dCwgVE91dHB1dD4gPSB7XG4gICAgY2hhcnQ6IGNoYXJ0VmlldyxcbiAgICBpbnB1dFNjaGVtYSxcbiAgICBvdXRwdXRTY2hlbWEsXG4gICAgb3V0cHV0TWFwcGVyOiBvcHRpb25zLm91dHB1dE1hcHBlcixcbiAgICB0b09wZW5BUEkoYXBpT3B0aW9ucz86IE9wZW5BUElPcHRpb25zKTogT3BlbkFQSVNwZWMge1xuICAgICAgcmV0dXJuIGdlbmVyYXRlT3BlbkFQSShjb250cmFjdCwgYXBpT3B0aW9ucyk7XG4gICAgfSxcbiAgfTtcblxuICByZXR1cm4gY29udHJhY3Q7XG59XG4iXX0=
@@ -3,11 +3,12 @@
3
3
  * contract/openapi.ts — OpenAPI 3.1 spec generator.
4
4
  *
5
5
  * Generates an OpenAPI spec from a FlowChartContract by combining:
6
- * - chart.description → operation description
7
- * - chart.stageDescriptions → step-by-step detail
6
+ * - chart.description → operation description (built incrementally during FlowChartBuilder.build())
8
7
  * - inputSchema → requestBody
9
8
  * - outputSchema → response
10
- * - chart.buildTimeStructure → operation metadata (branches, forks, etc.)
9
+ *
10
+ * chart.description is assembled by FlowChartBuilder as each stage is added —
11
+ * no post-processing walk of buildTimeStructure is needed or performed here.
11
12
  */
12
13
  Object.defineProperty(exports, "__esModule", { value: true });
13
14
  exports.generateOpenAPI = void 0;
@@ -20,37 +21,14 @@ function slugify(name) {
20
21
  .replace(/[^a-z0-9]+/g, '-')
21
22
  .replace(/^-|-$/g, '');
22
23
  }
23
- function buildDescription(chart) {
24
- // Walk buildTimeStructure to produce a detailed step-by-step description
25
- // that includes decider branches, parallel forks, etc.
26
- const lines = [];
27
- let step = 0;
28
- const walk = (node) => {
29
- step++;
30
- const desc = node.description ? ` — ${node.description}` : '';
31
- if (node.hasDecider && node.branchIds) {
32
- lines.push(`${step}. ${node.name}${desc} — Decides between: ${node.branchIds.join(', ')}`);
33
- }
34
- else if (node.children && node.children.length > 0 && !node.hasDecider) {
35
- const childNames = node.children.map((c) => c.name).join(', ');
36
- lines.push(`${step}. ${node.name}${desc} (parallel: ${childNames})`);
37
- }
38
- else {
39
- lines.push(`${step}. ${node.name}${desc}`);
40
- }
41
- if (node.children) {
42
- for (const child of node.children)
43
- walk(child);
44
- }
45
- if (node.next)
46
- walk(node.next);
47
- };
48
- walk(chart.buildTimeStructure);
49
- return `FlowChart: ${chart.root.name}\nSteps:\n${lines.join('\n')}`;
50
- }
51
24
  // ─────────────────────────────────────────────────────────────────────────────
52
25
  // Public API
53
26
  // ─────────────────────────────────────────────────────────────────────────────
27
+ /**
28
+ * Generates an OpenAPI 3.1 spec from a FlowChartContract.
29
+ * Uses `chart.description` which FlowChartBuilder assembles at build time —
30
+ * no post-processing walk of buildTimeStructure is performed here.
31
+ */
54
32
  function generateOpenAPI(contract, options) {
55
33
  var _a, _b, _c;
56
34
  const { chart, inputSchema, outputSchema } = contract;
@@ -60,7 +38,8 @@ function generateOpenAPI(contract, options) {
60
38
  const rootName = chart.root.name;
61
39
  const operationId = slugify(rootName);
62
40
  const path = `${basePath === '/' ? '' : basePath}/${operationId}`;
63
- const fullDescription = buildDescription(chart);
41
+ // Description was built incrementally during FlowChartBuilder.build() — read it directly.
42
+ const fullDescription = chart.description;
64
43
  // Build schemas for components
65
44
  const schemas = {};
66
45
  const inputRef = `${rootName}Input`;
@@ -121,4 +100,4 @@ function generateOpenAPI(contract, options) {
121
100
  return spec;
122
101
  }
123
102
  exports.generateOpenAPI = generateOpenAPI;
124
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"openapi.js","sourceRoot":"","sources":["../../../src/lib/contract/openapi.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * contract/openapi.ts — OpenAPI 3.1 spec generator.\n *\n * Generates an OpenAPI spec from a FlowChartContract by combining:\n * - chart.description → operation description\n * - chart.stageDescriptions → step-by-step detail\n * - inputSchema → requestBody\n * - outputSchema → response\n * - chart.buildTimeStructure → operation metadata (branches, forks, etc.)\n */\n\nimport type { FlowChart, SerializedPipelineStructure } from '../builder/types.js';\nimport type { FlowChartContract, JsonSchema, OpenAPIOperation, OpenAPIOptions, OpenAPISpec } from './types.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Internal helpers\n// ─────────────────────────────────────────────────────────────────────────────\n\nfunction slugify(name: string): string {\n  return name\n    .toLowerCase()\n    .replace(/[^a-z0-9]+/g, '-')\n    .replace(/^-|-$/g, '');\n}\n\nfunction buildDescription(chart: FlowChart): string {\n  // Walk buildTimeStructure to produce a detailed step-by-step description\n  // that includes decider branches, parallel forks, etc.\n  const lines: string[] = [];\n  let step = 0;\n\n  const walk = (node: SerializedPipelineStructure) => {\n    step++;\n    const desc = node.description ? ` — ${node.description}` : '';\n\n    if (node.hasDecider && node.branchIds) {\n      lines.push(`${step}. ${node.name}${desc} — Decides between: ${node.branchIds.join(', ')}`);\n    } else if (node.children && node.children.length > 0 && !node.hasDecider) {\n      const childNames = node.children.map((c) => c.name).join(', ');\n      lines.push(`${step}. ${node.name}${desc} (parallel: ${childNames})`);\n    } else {\n      lines.push(`${step}. ${node.name}${desc}`);\n    }\n\n    if (node.children) {\n      for (const child of node.children) walk(child);\n    }\n    if (node.next) walk(node.next);\n  };\n\n  walk(chart.buildTimeStructure);\n  return `FlowChart: ${chart.root.name}\\nSteps:\\n${lines.join('\\n')}`;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Public API\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport function generateOpenAPI(contract: FlowChartContract, options?: OpenAPIOptions): OpenAPISpec {\n  const { chart, inputSchema, outputSchema } = contract;\n  const version = options?.version ?? '1.0.0';\n  const basePath = options?.basePath ?? '/';\n  const method = options?.method ?? 'post';\n\n  const rootName = chart.root.name;\n  const operationId = slugify(rootName);\n  const path = `${basePath === '/' ? '' : basePath}/${operationId}`;\n\n  const fullDescription = buildDescription(chart);\n\n  // Build schemas for components\n  const schemas: Record<string, JsonSchema> = {};\n  const inputRef = `${rootName}Input`;\n  const outputRef = `${rootName}Output`;\n\n  if (inputSchema) schemas[inputRef] = inputSchema;\n  if (outputSchema) schemas[outputRef] = outputSchema;\n\n  // Build operation\n  const operation: OpenAPIOperation = {\n    operationId,\n    summary: rootName,\n    description: fullDescription,\n    ...(inputSchema\n      ? {\n          requestBody: {\n            required: true,\n            content: {\n              'application/json': {\n                schema: { $ref: `#/components/schemas/${inputRef}` },\n              },\n            },\n          },\n        }\n      : {}),\n    responses: {\n      '200': {\n        description: 'Successful execution',\n        ...(outputSchema\n          ? {\n              content: {\n                'application/json': {\n                  schema: { $ref: `#/components/schemas/${outputRef}` },\n                },\n              },\n            }\n          : {}),\n      },\n      '500': {\n        description: 'Pipeline execution error',\n      },\n    },\n  };\n\n  const spec: OpenAPISpec = {\n    openapi: '3.1.0',\n    info: {\n      title: rootName,\n      description: fullDescription,\n      version,\n    },\n    paths: {\n      [path]: {\n        [method]: operation,\n      },\n    },\n    ...(Object.keys(schemas).length > 0 ? { components: { schemas } } : {}),\n  };\n\n  return spec;\n}\n"]}
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"openapi.js","sourceRoot":"","sources":["../../../src/lib/contract/openapi.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * contract/openapi.ts — OpenAPI 3.1 spec generator.\n *\n * Generates an OpenAPI spec from a FlowChartContract by combining:\n * - chart.description → operation description (built incrementally during FlowChartBuilder.build())\n * - inputSchema → requestBody\n * - outputSchema → response\n *\n * chart.description is assembled by FlowChartBuilder as each stage is added —\n * no post-processing walk of buildTimeStructure is needed or performed here.\n */\n\nimport type { FlowChart } from '../builder/types.js';\nimport type { FlowChartContract, JsonSchema, OpenAPIOperation, OpenAPIOptions, OpenAPISpec } from './types.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Internal helpers\n// ─────────────────────────────────────────────────────────────────────────────\n\nfunction slugify(name: string): string {\n  return name\n    .toLowerCase()\n    .replace(/[^a-z0-9]+/g, '-')\n    .replace(/^-|-$/g, '');\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Public API\n// ─────────────────────────────────────────────────────────────────────────────\n\n/**\n * Generates an OpenAPI 3.1 spec from a FlowChartContract.\n * Uses `chart.description` which FlowChartBuilder assembles at build time —\n * no post-processing walk of buildTimeStructure is performed here.\n */\nexport function generateOpenAPI(contract: FlowChartContract, options?: OpenAPIOptions): OpenAPISpec {\n  const { chart, inputSchema, outputSchema } = contract;\n  const version = options?.version ?? '1.0.0';\n  const basePath = options?.basePath ?? '/';\n  const method = options?.method ?? 'post';\n\n  const rootName = chart.root.name;\n  const operationId = slugify(rootName);\n  const path = `${basePath === '/' ? '' : basePath}/${operationId}`;\n\n  // Description was built incrementally during FlowChartBuilder.build() — read it directly.\n  const fullDescription = chart.description;\n\n  // Build schemas for components\n  const schemas: Record<string, JsonSchema> = {};\n  const inputRef = `${rootName}Input`;\n  const outputRef = `${rootName}Output`;\n\n  if (inputSchema) schemas[inputRef] = inputSchema;\n  if (outputSchema) schemas[outputRef] = outputSchema;\n\n  // Build operation\n  const operation: OpenAPIOperation = {\n    operationId,\n    summary: rootName,\n    description: fullDescription,\n    ...(inputSchema\n      ? {\n          requestBody: {\n            required: true,\n            content: {\n              'application/json': {\n                schema: { $ref: `#/components/schemas/${inputRef}` },\n              },\n            },\n          },\n        }\n      : {}),\n    responses: {\n      '200': {\n        description: 'Successful execution',\n        ...(outputSchema\n          ? {\n              content: {\n                'application/json': {\n                  schema: { $ref: `#/components/schemas/${outputRef}` },\n                },\n              },\n            }\n          : {}),\n      },\n      '500': {\n        description: 'Pipeline execution error',\n      },\n    },\n  };\n\n  const spec: OpenAPISpec = {\n    openapi: '3.1.0',\n    info: {\n      title: rootName,\n      description: fullDescription,\n      version,\n    },\n    paths: {\n      [path]: {\n        [method]: operation,\n      },\n    },\n    ...(Object.keys(schemas).length > 0 ? { components: { schemas } } : {}),\n  };\n\n  return spec;\n}\n"]}
@@ -7,4 +7,4 @@
7
7
  * (inputMapper/outputMapper) but at the top-level flowchart boundary.
8
8
  */
9
9
  Object.defineProperty(exports, "__esModule", { value: true });
10
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL2NvbnRyYWN0L3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7O0dBTUciLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIGNvbnRyYWN0L3R5cGVzLnRzIOKAlCBUeXBlcyBmb3IgdGhlIEZsb3dDaGFydCBjb250cmFjdCBsYXllci5cbiAqXG4gKiBEZWZpbmVzIHRoZSBJL08gYm91bmRhcnkgZm9yIGEgZmxvd2NoYXJ0OiBpbnB1dCBzY2hlbWEsIG91dHB1dCBzY2hlbWEsXG4gKiBhbmQgb3V0cHV0IG1hcHBlci4gVXNlcyB0aGUgc2FtZSBwYXR0ZXJuIGFzIFN1YmZsb3dNb3VudE9wdGlvbnNcbiAqIChpbnB1dE1hcHBlci9vdXRwdXRNYXBwZXIpIGJ1dCBhdCB0aGUgdG9wLWxldmVsIGZsb3djaGFydCBib3VuZGFyeS5cbiAqL1xuXG5pbXBvcnQgdHlwZSB7IEZsb3dDaGFydCB9IGZyb20gJy4uL2J1aWxkZXIvdHlwZXMuanMnO1xuXG4vLyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcbi8vIEpTT04gU2NoZW1hIChzdWJzZXQgb2YgSlNPTiBTY2hlbWEgRHJhZnQgMjAyMC0xMiAvIE9wZW5BUEkgMy4xKVxuLy8g4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG5cbmV4cG9ydCB0eXBlIEpzb25TY2hlbWEgPSB7XG4gIHR5cGU/OiBzdHJpbmcgfCBzdHJpbmdbXTtcbiAgcHJvcGVydGllcz86IFJlY29yZDxzdHJpbmcsIEpzb25TY2hlbWE+O1xuICByZXF1aXJlZD86IHN0cmluZ1tdO1xuICBpdGVtcz86IEpzb25TY2hlbWE7XG4gIGVudW0/OiB1bmtub3duW107XG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuICBkZWZhdWx0PzogdW5rbm93bjtcbiAgZm9ybWF0Pzogc3RyaW5nO1xuICBhZGRpdGlvbmFsUHJvcGVydGllcz86IGJvb2xlYW4gfCBKc29uU2NoZW1hO1xuICBvbmVPZj86IEpzb25TY2hlbWFbXTtcbiAgYW55T2Y/OiBKc29uU2NoZW1hW107XG4gIGFsbE9mPzogSnNvblNjaGVtYVtdO1xuICAkcmVmPzogc3RyaW5nO1xuICBba2V5OiBzdHJpbmddOiB1bmtub3duO1xufTtcblxuLy8g4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG4vLyBTY2hlbWEgSW5wdXQg4oCUIGFjY2VwdHMgZWl0aGVyIFpvZCBzY2hlbWEgb3IgcmF3IEpTT04gU2NoZW1hXG4vLyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcblxuLyoqIEFueXRoaW5nIHdpdGggYSBgZGVmYCAoWm9kIHY0KSBvciBgX2RlZmAgKFpvZCB2MykgcHJvcGVydHkgaXMgdHJlYXRlZCBhcyBhIFpvZCBzY2hlbWEuICovXG5leHBvcnQgdHlwZSBTY2hlbWFJbnB1dCA9IEpzb25TY2hlbWEgfCB7IGRlZjogdW5rbm93bjsgW2tleTogc3RyaW5nXTogdW5rbm93biB9O1xuXG4vLyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcbi8vIEZsb3dDaGFydCBDb250cmFjdCDigJQgSS9PIGJvdW5kYXJ5IGRlZmluaXRpb25cbi8vIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG5leHBvcnQgaW50ZXJmYWNlIEZsb3dDaGFydENvbnRyYWN0T3B0aW9uczxfVElucHV0ID0gdW5rbm93biwgVE91dHB1dCA9IHVua25vd24+IHtcbiAgLyoqIFNjaGVtYSBkZXNjcmliaW5nIHRoZSBpbnB1dCAocmVhZE9ubHlDb250ZXh0KSBzaGFwZS4gWm9kIG9yIEpTT04gU2NoZW1hLiAqL1xuICBpbnB1dFNjaGVtYT86IFNjaGVtYUlucHV0O1xuICAvKiogU2NoZW1hIGRlc2NyaWJpbmcgdGhlIG91dHB1dCBzaGFwZS4gWm9kIG9yIEpTT04gU2NoZW1hLiAqL1xuICBvdXRwdXRTY2hlbWE/OiBTY2hlbWFJbnB1dDtcbiAgLyoqIE1hcHMgdGhlIGZpbmFsIHNjb3BlIHN0YXRlIGludG8gdGhlIHJlc3BvbnNlIHNoYXBlLiAqL1xuICBvdXRwdXRNYXBwZXI/OiAoZmluYWxTY29wZTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4pID0+IFRPdXRwdXQ7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRmxvd0NoYXJ0Q29udHJhY3Q8X1RJbnB1dCA9IHVua25vd24sIFRPdXRwdXQgPSB1bmtub3duPiB7XG4gIC8qKiBUaGUgY29tcGlsZWQgZmxvd2NoYXJ0LiAqL1xuICBjaGFydDogRmxvd0NoYXJ0O1xuICAvKiogSlNPTiBTY2hlbWEgZm9yIHRoZSBpbnB1dCAobm9ybWFsaXplZCBmcm9tIFpvZCBvciByYXcpLiAqL1xuICBpbnB1dFNjaGVtYT86IEpzb25TY2hlbWE7XG4gIC8qKiBKU09OIFNjaGVtYSBmb3IgdGhlIG91dHB1dCAobm9ybWFsaXplZCBmcm9tIFpvZCBvciByYXcpLiAqL1xuICBvdXRwdXRTY2hlbWE/OiBKc29uU2NoZW1hO1xuICAvKiogTWFwcyB0aGUgZmluYWwgc2NvcGUgc3RhdGUgaW50byB0aGUgcmVzcG9uc2Ugc2hhcGUuICovXG4gIG91dHB1dE1hcHBlcj86IChmaW5hbFNjb3BlOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPikgPT4gVE91dHB1dDtcbiAgLyoqIEF1dG8tZ2VuZXJhdGVkIE9wZW5BUEkgc3BlYy4gKi9cbiAgdG9PcGVuQVBJKG9wdGlvbnM/OiBPcGVuQVBJT3B0aW9ucyk6IE9wZW5BUElTcGVjO1xufVxuXG4vLyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcbi8vIE9wZW5BUEkgVHlwZXMgKG1pbmltYWwgc3Vic2V0IG9mIE9wZW5BUEkgMy4xKVxuLy8g4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG5cbmV4cG9ydCBpbnRlcmZhY2UgT3BlbkFQSU9wdGlvbnMge1xuICAvKiogQVBJIHZlcnNpb24gc3RyaW5nIChkZWZhdWx0OiBcIjEuMC4wXCIpLiAqL1xuICB2ZXJzaW9uPzogc3RyaW5nO1xuICAvKiogQmFzZSBwYXRoIHByZWZpeCAoZGVmYXVsdDogXCIvXCIpLiAqL1xuICBiYXNlUGF0aD86IHN0cmluZztcbiAgLyoqIEhUVFAgbWV0aG9kIGZvciB0aGUgZXhlY3V0ZSBlbmRwb2ludCAoZGVmYXVsdDogXCJwb3N0XCIpLiAqL1xuICBtZXRob2Q/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgT3BlbkFQSVNwZWMge1xuICBvcGVuYXBpOiAnMy4xLjAnO1xuICBpbmZvOiB7XG4gICAgdGl0bGU6IHN0cmluZztcbiAgICBkZXNjcmlwdGlvbjogc3RyaW5nO1xuICAgIHZlcnNpb246IHN0cmluZztcbiAgfTtcbiAgcGF0aHM6IFJlY29yZDxzdHJpbmcsIFJlY29yZDxzdHJpbmcsIE9wZW5BUElPcGVyYXRpb24+PjtcbiAgY29tcG9uZW50cz86IHtcbiAgICBzY2hlbWFzPzogUmVjb3JkPHN0cmluZywgSnNvblNjaGVtYT47XG4gIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgT3BlbkFQSU9wZXJhdGlvbiB7XG4gIG9wZXJhdGlvbklkOiBzdHJpbmc7XG4gIHN1bW1hcnk6IHN0cmluZztcbiAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgcmVxdWVzdEJvZHk/OiB7XG4gICAgcmVxdWlyZWQ6IGJvb2xlYW47XG4gICAgY29udGVudDoge1xuICAgICAgJ2FwcGxpY2F0aW9uL2pzb24nOiB7XG4gICAgICAgIHNjaGVtYTogSnNvblNjaGVtYSB8IHsgJHJlZjogc3RyaW5nIH07XG4gICAgICB9O1xuICAgIH07XG4gIH07XG4gIHJlc3BvbnNlczogUmVjb3JkPFxuICAgIHN0cmluZyxcbiAgICB7XG4gICAgICBkZXNjcmlwdGlvbjogc3RyaW5nO1xuICAgICAgY29udGVudD86IHtcbiAgICAgICAgJ2FwcGxpY2F0aW9uL2pzb24nOiB7XG4gICAgICAgICAgc2NoZW1hOiBKc29uU2NoZW1hIHwgeyAkcmVmOiBzdHJpbmcgfTtcbiAgICAgICAgfTtcbiAgICAgIH07XG4gICAgfVxuICA+O1xufVxuIl19
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL2NvbnRyYWN0L3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7O0dBTUciLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIGNvbnRyYWN0L3R5cGVzLnRzIOKAlCBUeXBlcyBmb3IgdGhlIEZsb3dDaGFydCBjb250cmFjdCBsYXllci5cbiAqXG4gKiBEZWZpbmVzIHRoZSBJL08gYm91bmRhcnkgZm9yIGEgZmxvd2NoYXJ0OiBpbnB1dCBzY2hlbWEsIG91dHB1dCBzY2hlbWEsXG4gKiBhbmQgb3V0cHV0IG1hcHBlci4gVXNlcyB0aGUgc2FtZSBwYXR0ZXJuIGFzIFN1YmZsb3dNb3VudE9wdGlvbnNcbiAqIChpbnB1dE1hcHBlci9vdXRwdXRNYXBwZXIpIGJ1dCBhdCB0aGUgdG9wLWxldmVsIGZsb3djaGFydCBib3VuZGFyeS5cbiAqL1xuXG5pbXBvcnQgdHlwZSB7IEZsb3dDaGFydCB9IGZyb20gJy4uL2J1aWxkZXIvdHlwZXMuanMnO1xuXG4vLyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcbi8vIEpTT04gU2NoZW1hIChzdWJzZXQgb2YgSlNPTiBTY2hlbWEgRHJhZnQgMjAyMC0xMiAvIE9wZW5BUEkgMy4xKVxuLy8g4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG5cbmV4cG9ydCB0eXBlIEpzb25TY2hlbWEgPSB7XG4gIHR5cGU/OiBzdHJpbmcgfCBzdHJpbmdbXTtcbiAgcHJvcGVydGllcz86IFJlY29yZDxzdHJpbmcsIEpzb25TY2hlbWE+O1xuICByZXF1aXJlZD86IHN0cmluZ1tdO1xuICBpdGVtcz86IEpzb25TY2hlbWE7XG4gIGVudW0/OiB1bmtub3duW107XG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuICBkZWZhdWx0PzogdW5rbm93bjtcbiAgZm9ybWF0Pzogc3RyaW5nO1xuICBhZGRpdGlvbmFsUHJvcGVydGllcz86IGJvb2xlYW4gfCBKc29uU2NoZW1hO1xuICBvbmVPZj86IEpzb25TY2hlbWFbXTtcbiAgYW55T2Y/OiBKc29uU2NoZW1hW107XG4gIGFsbE9mPzogSnNvblNjaGVtYVtdO1xuICAkcmVmPzogc3RyaW5nO1xuICBba2V5OiBzdHJpbmddOiB1bmtub3duO1xufTtcblxuLy8g4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG4vLyBTY2hlbWEgSW5wdXQg4oCUIGFjY2VwdHMgZWl0aGVyIFpvZCBzY2hlbWEgb3IgcmF3IEpTT04gU2NoZW1hXG4vLyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcblxuLyoqIEFueXRoaW5nIHdpdGggYSBgZGVmYCAoWm9kIHY0KSBvciBgX2RlZmAgKFpvZCB2MykgcHJvcGVydHkgaXMgdHJlYXRlZCBhcyBhIFpvZCBzY2hlbWEuICovXG5leHBvcnQgdHlwZSBTY2hlbWFJbnB1dCA9IEpzb25TY2hlbWEgfCB7IGRlZjogdW5rbm93bjsgW2tleTogc3RyaW5nXTogdW5rbm93biB9O1xuXG4vLyDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcbi8vIEZsb3dDaGFydCBDb250cmFjdCDigJQgSS9PIGJvdW5kYXJ5IGRlZmluaXRpb25cbi8vIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG5leHBvcnQgaW50ZXJmYWNlIEZsb3dDaGFydENvbnRyYWN0T3B0aW9uczxfVElucHV0ID0gdW5rbm93biwgVE91dHB1dCA9IHVua25vd24+IHtcbiAgLyoqIFNjaGVtYSBkZXNjcmliaW5nIHRoZSBpbnB1dCAocmVhZE9ubHlDb250ZXh0KSBzaGFwZS4gWm9kIG9yIEpTT04gU2NoZW1hLiAqL1xuICBpbnB1dFNjaGVtYT86IFNjaGVtYUlucHV0O1xuICAvKiogU2NoZW1hIGRlc2NyaWJpbmcgdGhlIG91dHB1dCBzaGFwZS4gWm9kIG9yIEpTT04gU2NoZW1hLiAqL1xuICBvdXRwdXRTY2hlbWE/OiBTY2hlbWFJbnB1dDtcbiAgLyoqIE1hcHMgdGhlIGZpbmFsIHNjb3BlIHN0YXRlIGludG8gdGhlIHJlc3BvbnNlIHNoYXBlLiAqL1xuICBvdXRwdXRNYXBwZXI/OiAoZmluYWxTY29wZTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4pID0+IFRPdXRwdXQ7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRmxvd0NoYXJ0Q29udHJhY3Q8X1RJbnB1dCA9IHVua25vd24sIFRPdXRwdXQgPSB1bmtub3duPiB7XG4gIC8qKlxuICAgKiBQcm90b3R5cGUtbGlua2VkIHZpZXcgb2YgdGhlIG9yaWdpbmFsIGNvbXBpbGVkIEZsb3dDaGFydC5cbiAgICpcbiAgICogT3duIHByb3BlcnRpZXMgKGlucHV0U2NoZW1hLCBvdXRwdXRTY2hlbWEsIG91dHB1dE1hcHBlcikgYXJlIHNoYWRvd2VkIGhlcmVcbiAgICogc28gdGhhdCB0aGUgY29udHJhY3QncyB2YWx1ZXMgdGFrZSBwcmVjZWRlbmNlIG92ZXIgYnVpbGRlci1zZXQgdmFsdWVzLlxuICAgKiBBbGwgb3RoZXIgZmllbGRzIChyb290LCBzdGFnZU1hcCwgbWV0aG9kcykgYXJlIGluaGVyaXRlZCB2aWEgdGhlIHByb3RvdHlwZVxuICAgKiBjaGFpbiB3aXRoIHplcm8gY29weWluZy4gVGhlIG9yaWdpbmFsIGNoYXJ0IGlzIG5ldmVyIG11dGF0ZWQuXG4gICAqXG4gICAqIOKaoO+4jyBEbyBOT1QgdXNlIE9iamVjdC5rZXlzKCksIHNwcmVhZCAoey4uLmNoYXJ0fSksIG9yIEpTT04uc3RyaW5naWZ5KCkgb25cbiAgICogdGhpcyBvYmplY3Qg4oCUIGl0IHdpbGwgb25seSBzZWUgb3duIChzaGFkb3dlZCkgcHJvcGVydGllcy4gVXNlIG5hbWVkXG4gICAqIHByb3BlcnR5IGFjY2VzcyBvciBjaGFydC50b1NwZWMoKSBpbnN0ZWFkLlxuICAgKi9cbiAgY2hhcnQ6IEZsb3dDaGFydDtcbiAgLyoqIEpTT04gU2NoZW1hIGZvciB0aGUgaW5wdXQgKG5vcm1hbGl6ZWQgZnJvbSBab2Qgb3IgcmF3KS4gKi9cbiAgaW5wdXRTY2hlbWE/OiBKc29uU2NoZW1hO1xuICAvKiogSlNPTiBTY2hlbWEgZm9yIHRoZSBvdXRwdXQgKG5vcm1hbGl6ZWQgZnJvbSBab2Qgb3IgcmF3KS4gKi9cbiAgb3V0cHV0U2NoZW1hPzogSnNvblNjaGVtYTtcbiAgLyoqIE1hcHMgdGhlIGZpbmFsIHNjb3BlIHN0YXRlIGludG8gdGhlIHJlc3BvbnNlIHNoYXBlLiAqL1xuICBvdXRwdXRNYXBwZXI/OiAoZmluYWxTY29wZTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4pID0+IFRPdXRwdXQ7XG4gIC8qKiBBdXRvLWdlbmVyYXRlZCBPcGVuQVBJIHNwZWMuIERlc2NyaXB0aW9uIGlzIHJlYWQgZnJvbSBgY2hhcnQuZGVzY3JpcHRpb25gIChwcmUtYnVpbHQgYXQgYnVpbGQgdGltZSkuICovXG4gIHRvT3BlbkFQSShvcHRpb25zPzogT3BlbkFQSU9wdGlvbnMpOiBPcGVuQVBJU3BlYztcbn1cblxuLy8g4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG4vLyBPcGVuQVBJIFR5cGVzIChtaW5pbWFsIHN1YnNldCBvZiBPcGVuQVBJIDMuMSlcbi8vIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG5leHBvcnQgaW50ZXJmYWNlIE9wZW5BUElPcHRpb25zIHtcbiAgLyoqIEFQSSB2ZXJzaW9uIHN0cmluZyAoZGVmYXVsdDogXCIxLjAuMFwiKS4gKi9cbiAgdmVyc2lvbj86IHN0cmluZztcbiAgLyoqIEJhc2UgcGF0aCBwcmVmaXggKGRlZmF1bHQ6IFwiL1wiKS4gKi9cbiAgYmFzZVBhdGg/OiBzdHJpbmc7XG4gIC8qKiBIVFRQIG1ldGhvZCBmb3IgdGhlIGV4ZWN1dGUgZW5kcG9pbnQgKGRlZmF1bHQ6IFwicG9zdFwiKS4gKi9cbiAgbWV0aG9kPzogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE9wZW5BUElTcGVjIHtcbiAgb3BlbmFwaTogJzMuMS4wJztcbiAgaW5mbzoge1xuICAgIHRpdGxlOiBzdHJpbmc7XG4gICAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgICB2ZXJzaW9uOiBzdHJpbmc7XG4gIH07XG4gIHBhdGhzOiBSZWNvcmQ8c3RyaW5nLCBSZWNvcmQ8c3RyaW5nLCBPcGVuQVBJT3BlcmF0aW9uPj47XG4gIGNvbXBvbmVudHM/OiB7XG4gICAgc2NoZW1hcz86IFJlY29yZDxzdHJpbmcsIEpzb25TY2hlbWE+O1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE9wZW5BUElPcGVyYXRpb24ge1xuICBvcGVyYXRpb25JZDogc3RyaW5nO1xuICBzdW1tYXJ5OiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIHJlcXVlc3RCb2R5Pzoge1xuICAgIHJlcXVpcmVkOiBib29sZWFuO1xuICAgIGNvbnRlbnQ6IHtcbiAgICAgICdhcHBsaWNhdGlvbi9qc29uJzoge1xuICAgICAgICBzY2hlbWE6IEpzb25TY2hlbWEgfCB7ICRyZWY6IHN0cmluZyB9O1xuICAgICAgfTtcbiAgICB9O1xuICB9O1xuICByZXNwb25zZXM6IFJlY29yZDxcbiAgICBzdHJpbmcsXG4gICAge1xuICAgICAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgICAgIGNvbnRlbnQ/OiB7XG4gICAgICAgICdhcHBsaWNhdGlvbi9qc29uJzoge1xuICAgICAgICAgIHNjaGVtYTogSnNvblNjaGVtYSB8IHsgJHJlZjogc3RyaW5nIH07XG4gICAgICAgIH07XG4gICAgICB9O1xuICAgIH1cbiAgPjtcbn1cbiJdfQ==
@@ -11,6 +11,7 @@
11
11
  */
12
12
  Object.defineProperty(exports, "__esModule", { value: true });
13
13
  exports.select = exports.decide = void 0;
14
+ const detectCircular_js_1 = require("../scope/detectCircular.js");
14
15
  const evaluator_js_1 = require("./evaluator.js");
15
16
  const evidence_js_1 = require("./evidence.js");
16
17
  const types_js_1 = require("./types.js");
@@ -63,11 +64,19 @@ function evaluateRule(scope, rule, index, attachFn, detachFn, valueFn, redactedF
63
64
  if (collector && attachFn)
64
65
  attachFn(collector);
65
66
  let matched;
67
+ let matchError;
66
68
  try {
67
69
  matched = rule.when(scope);
68
70
  }
69
- catch (_b) {
71
+ catch (e) {
70
72
  matched = false;
73
+ // Capture the error for debugging — surface it in evidence instead of swallowing silently
74
+ matchError = e instanceof Error ? e.message : String(e);
75
+ if ((0, detectCircular_js_1.isDevMode)()) {
76
+ const label = rule.label ? ` ('${rule.label}')` : '';
77
+ // eslint-disable-next-line no-console
78
+ console.warn(`[footprint] decide() rule ${index}${label} threw during evaluation: ${matchError}`);
79
+ }
71
80
  }
72
81
  finally {
73
82
  if (collector && detachFn)
@@ -79,7 +88,9 @@ function evaluateRule(scope, rule, index, attachFn, detachFn, valueFn, redactedF
79
88
  branch: rule.then,
80
89
  matched,
81
90
  label: rule.label,
91
+ // Partial reads: if rule threw after some getValue() calls, collector holds reads up to the throw point
82
92
  inputs: (_a = collector === null || collector === void 0 ? void 0 : collector.getInputs()) !== null && _a !== void 0 ? _a : [],
93
+ ...(matchError !== undefined && { matchError }),
83
94
  };
84
95
  return evidence;
85
96
  }
@@ -89,14 +100,22 @@ function evaluateRule(scope, rule, index, attachFn, detachFn, valueFn, redactedF
89
100
  const resolvedRedactedFn = redactedFn !== null && redactedFn !== void 0 ? redactedFn : (() => false);
90
101
  let filterMatched = false;
91
102
  let filterConditions = [];
103
+ let matchError;
92
104
  try {
93
105
  const result = (0, evaluator_js_1.evaluateFilter)(resolvedValueFn, resolvedRedactedFn, rule.when);
94
106
  filterMatched = result.matched;
95
107
  filterConditions = result.conditions;
96
108
  }
97
- catch (_c) {
109
+ catch (e) {
98
110
  filterMatched = false;
99
111
  filterConditions = [];
112
+ // Capture the error for debugging — surface it in evidence instead of swallowing silently
113
+ matchError = e instanceof Error ? e.message : String(e);
114
+ if ((0, detectCircular_js_1.isDevMode)()) {
115
+ const label = rule.label ? ` ('${rule.label}')` : '';
116
+ // eslint-disable-next-line no-console
117
+ console.warn(`[footprint] decide() filter rule ${index}${label} threw during evaluation: ${matchError}`);
118
+ }
100
119
  }
101
120
  const evidence = {
102
121
  type: 'filter',
@@ -105,6 +124,7 @@ function evaluateRule(scope, rule, index, attachFn, detachFn, valueFn, redactedF
105
124
  matched: filterMatched,
106
125
  label: rule.label,
107
126
  conditions: filterConditions,
127
+ ...(matchError !== undefined && { matchError }),
108
128
  };
109
129
  return evidence;
110
130
  }
@@ -116,6 +136,11 @@ function evaluateRule(scope, rule, index, attachFn, detachFn, valueFn, redactedF
116
136
  * @param scope - TypedScope or ScopeFacade
117
137
  * @param rules - Array of DecideRule (function or filter when clauses)
118
138
  * @param defaultBranch - Branch ID if no rule matches
139
+ *
140
+ * **Error behavior:** If a `when` function throws during evaluation, the rule is
141
+ * treated as non-matching (`matched: false`) and the error message is captured in
142
+ * `matchError` on that rule's `RuleEvidence` entry. Execution continues with
143
+ * subsequent rules; errors do not propagate to the caller.
119
144
  */
120
145
  function decide(scope, rules, defaultBranch) {
121
146
  const attachFn = getAttachFn(scope);
@@ -150,6 +175,11 @@ exports.decide = decide;
150
175
  *
151
176
  * @param scope - TypedScope or ScopeFacade
152
177
  * @param rules - Array of DecideRule (function or filter when clauses)
178
+ *
179
+ * **Error behavior:** If a `when` function throws during evaluation, the rule is
180
+ * treated as non-matching (`matched: false`) and the error message is captured in
181
+ * `matchError` on that rule's `RuleEvidence` entry. Evaluation continues with
182
+ * remaining rules; errors do not propagate to the caller.
153
183
  */
154
184
  function select(scope, rules) {
155
185
  const attachFn = getAttachFn(scope);
@@ -172,4 +202,4 @@ function select(scope, rules) {
172
202
  return { branches: selectedBranches, [types_js_1.DECISION_RESULT]: true, evidence };
173
203
  }
174
204
  exports.select = select;
175
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decide.js","sourceRoot":"","sources":["../../../src/lib/decide/decide.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * decide/decide -- Core decide() and select() helper functions.\n *\n * decide() evaluates rules in order (first-match) and returns a DecisionResult.\n * select() evaluates ALL rules and returns a SelectionResult with all matches.\n *\n * Each rule's `when` can be:\n * - A function: (s) => s.creditScore > 700  (auto-captures reads via temp recorder)\n * - A filter:   { creditScore: { gt: 700 } } (captures reads + operators + thresholds)\n */\n\nimport type { Recorder } from '../scope/types.js';\nimport { evaluateFilter } from './evaluator.js';\nimport { EvidenceCollector } from './evidence.js';\nimport type {\n  DecideRule,\n  DecisionEvidence,\n  DecisionResult,\n  FilterCondition,\n  FilterRuleEvidence,\n  FunctionRuleEvidence,\n  RuleEvidence,\n  SelectionEvidence,\n  SelectionResult,\n  WhereFilter,\n} from './types.js';\nimport { DECISION_RESULT } from './types.js';\n\n// -- Scope accessor helpers --------------------------------------------------\n\nfunction getAttachFn(scope: unknown): ((r: Recorder) => void) | undefined {\n  const s = scope as Record<string, unknown>;\n  if (typeof s.$attachRecorder === 'function') return s.$attachRecorder.bind(s);\n  if (typeof s.attachRecorder === 'function') return s.attachRecorder.bind(s);\n  return undefined;\n}\n\nfunction getDetachFn(scope: unknown): ((id: string) => void) | undefined {\n  const s = scope as Record<string, unknown>;\n  if (typeof s.$detachRecorder === 'function') return s.$detachRecorder.bind(s);\n  if (typeof s.detachRecorder === 'function') return s.detachRecorder.bind(s);\n  return undefined;\n}\n\nfunction getValueFn(scope: unknown): (key: string) => unknown {\n  const s = scope as Record<string, unknown>;\n  // Check $getValue first: on TypedScope, accessing .getValue triggers a spurious\n  // onRead for key \"getValue\" via the Proxy get trap. $getValue routes through\n  // SCOPE_METHOD_NAMES and avoids the state-read path.\n  if (typeof s.$getValue === 'function') return s.$getValue.bind(s);\n  if (typeof s.getValue === 'function') return s.getValue.bind(s);\n  return () => undefined;\n}\n\nfunction getRedactedFn(scope: unknown): (key: string) => boolean {\n  const s = scope as Record<string, unknown>;\n  // Try $toRaw() first (TypedScope), then direct\n  const raw = typeof s.$toRaw === 'function' ? s.$toRaw() : s;\n  const r = raw as Record<string, unknown>;\n  if (typeof r.getRedactedKeys === 'function') {\n    const keys = r.getRedactedKeys() as Set<string>;\n    return (key: string) => keys.has(key);\n  }\n  return () => false;\n}\n\n// -- evaluate a single rule --------------------------------------------------\n\nfunction evaluateRule<S extends object>(\n  scope: S,\n  rule: DecideRule<S>,\n  index: number,\n  attachFn?: (r: Recorder) => void,\n  detachFn?: (id: string) => void,\n  valueFn?: (key: string) => unknown,\n  redactedFn?: (key: string) => boolean,\n): RuleEvidence {\n  if (typeof rule.when === 'function') {\n    // FUNCTION PATH: temp recorder captures reads (lazy — skip if no recorder support)\n    const hasRecorderSupport = Boolean(attachFn);\n    const collector = hasRecorderSupport ? new EvidenceCollector() : undefined;\n    if (collector && attachFn) attachFn(collector);\n\n    let matched: boolean;\n    try {\n      matched = rule.when(scope);\n    } catch {\n      matched = false;\n    } finally {\n      if (collector && detachFn) detachFn(collector.id);\n    }\n\n    const evidence: FunctionRuleEvidence = {\n      type: 'function',\n      ruleIndex: index,\n      branch: rule.then,\n      matched,\n      label: rule.label,\n      inputs: collector?.getInputs() ?? [],\n    };\n    return evidence;\n  } else {\n    // FILTER PATH: reads values directly via callbacks (no recorder); exceptions treated as non-match\n    const resolvedValueFn = valueFn ?? (() => undefined);\n    const resolvedRedactedFn = redactedFn ?? (() => false);\n    let filterMatched = false;\n    let filterConditions: FilterCondition[] = [];\n    try {\n      const result = evaluateFilter(resolvedValueFn, resolvedRedactedFn, rule.when as WhereFilter<S>);\n      filterMatched = result.matched;\n      filterConditions = result.conditions;\n    } catch {\n      filterMatched = false;\n      filterConditions = [];\n    }\n\n    const evidence: FilterRuleEvidence = {\n      type: 'filter',\n      ruleIndex: index,\n      branch: rule.then,\n      matched: filterMatched,\n      label: rule.label,\n      conditions: filterConditions,\n    };\n    return evidence;\n  }\n}\n\n// -- decide() ----------------------------------------------------------------\n\n/**\n * Evaluates rules in order (first-match). Returns a branded DecisionResult.\n *\n * @param scope - TypedScope or ScopeFacade\n * @param rules - Array of DecideRule (function or filter when clauses)\n * @param defaultBranch - Branch ID if no rule matches\n */\nexport function decide<S extends object>(scope: S, rules: DecideRule<S>[], defaultBranch: string): DecisionResult {\n  const attachFn = getAttachFn(scope);\n  const detachFn = getDetachFn(scope);\n  const valueFn = getValueFn(scope);\n  const redactedFn = getRedactedFn(scope);\n\n  const evaluatedRules: RuleEvidence[] = [];\n\n  for (const [index, rule] of rules.entries()) {\n    const ruleEvidence = evaluateRule(scope, rule, index, attachFn, detachFn, valueFn, redactedFn);\n    evaluatedRules.push(ruleEvidence);\n\n    if (ruleEvidence.matched) {\n      const evidence: DecisionEvidence = {\n        rules: evaluatedRules,\n        chosen: rule.then,\n        default: defaultBranch,\n      };\n      return { branch: rule.then, [DECISION_RESULT]: true, evidence };\n    }\n  }\n\n  // Default: no rule matched\n  const evidence: DecisionEvidence = {\n    rules: evaluatedRules,\n    chosen: defaultBranch,\n    default: defaultBranch,\n  };\n  return { branch: defaultBranch, [DECISION_RESULT]: true, evidence };\n}\n\n// -- select() ----------------------------------------------------------------\n\n/**\n * Evaluates ALL rules (not first-match). Returns a branded SelectionResult.\n *\n * @param scope - TypedScope or ScopeFacade\n * @param rules - Array of DecideRule (function or filter when clauses)\n */\nexport function select<S extends object>(scope: S, rules: DecideRule<S>[]): SelectionResult {\n  const attachFn = getAttachFn(scope);\n  const detachFn = getDetachFn(scope);\n  const valueFn = getValueFn(scope);\n  const redactedFn = getRedactedFn(scope);\n\n  const evaluatedRules: RuleEvidence[] = [];\n  const selectedBranches: string[] = [];\n\n  for (const [index, rule] of rules.entries()) {\n    const ruleEvidence = evaluateRule(scope, rule, index, attachFn, detachFn, valueFn, redactedFn);\n    evaluatedRules.push(ruleEvidence);\n\n    if (ruleEvidence.matched) {\n      selectedBranches.push(rule.then);\n    }\n  }\n\n  const evidence: SelectionEvidence = {\n    rules: evaluatedRules,\n    selected: selectedBranches,\n  };\n  return { branches: selectedBranches, [DECISION_RESULT]: true, evidence };\n}\n"]}
205
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decide.js","sourceRoot":"","sources":["../../../src/lib/decide/decide.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * decide/decide -- Core decide() and select() helper functions.\n *\n * decide() evaluates rules in order (first-match) and returns a DecisionResult.\n * select() evaluates ALL rules and returns a SelectionResult with all matches.\n *\n * Each rule's `when` can be:\n * - A function: (s) => s.creditScore > 700  (auto-captures reads via temp recorder)\n * - A filter:   { creditScore: { gt: 700 } } (captures reads + operators + thresholds)\n */\n\nimport { isDevMode } from '../scope/detectCircular.js';\nimport type { Recorder } from '../scope/types.js';\nimport { evaluateFilter } from './evaluator.js';\nimport { EvidenceCollector } from './evidence.js';\nimport type {\n  DecideRule,\n  DecisionEvidence,\n  DecisionResult,\n  FilterCondition,\n  FilterRuleEvidence,\n  FunctionRuleEvidence,\n  RuleEvidence,\n  SelectionEvidence,\n  SelectionResult,\n  WhereFilter,\n} from './types.js';\nimport { DECISION_RESULT } from './types.js';\n\n// -- Scope accessor helpers --------------------------------------------------\n\nfunction getAttachFn(scope: unknown): ((r: Recorder) => void) | undefined {\n  const s = scope as Record<string, unknown>;\n  if (typeof s.$attachRecorder === 'function') return s.$attachRecorder.bind(s);\n  if (typeof s.attachRecorder === 'function') return s.attachRecorder.bind(s);\n  return undefined;\n}\n\nfunction getDetachFn(scope: unknown): ((id: string) => void) | undefined {\n  const s = scope as Record<string, unknown>;\n  if (typeof s.$detachRecorder === 'function') return s.$detachRecorder.bind(s);\n  if (typeof s.detachRecorder === 'function') return s.detachRecorder.bind(s);\n  return undefined;\n}\n\nfunction getValueFn(scope: unknown): (key: string) => unknown {\n  const s = scope as Record<string, unknown>;\n  // Check $getValue first: on TypedScope, accessing .getValue triggers a spurious\n  // onRead for key \"getValue\" via the Proxy get trap. $getValue routes through\n  // SCOPE_METHOD_NAMES and avoids the state-read path.\n  if (typeof s.$getValue === 'function') return s.$getValue.bind(s);\n  if (typeof s.getValue === 'function') return s.getValue.bind(s);\n  return () => undefined;\n}\n\nfunction getRedactedFn(scope: unknown): (key: string) => boolean {\n  const s = scope as Record<string, unknown>;\n  // Try $toRaw() first (TypedScope), then direct\n  const raw = typeof s.$toRaw === 'function' ? s.$toRaw() : s;\n  const r = raw as Record<string, unknown>;\n  if (typeof r.getRedactedKeys === 'function') {\n    const keys = r.getRedactedKeys() as Set<string>;\n    return (key: string) => keys.has(key);\n  }\n  return () => false;\n}\n\n// -- evaluate a single rule --------------------------------------------------\n\nfunction evaluateRule<S extends object>(\n  scope: S,\n  rule: DecideRule<S>,\n  index: number,\n  attachFn?: (r: Recorder) => void,\n  detachFn?: (id: string) => void,\n  valueFn?: (key: string) => unknown,\n  redactedFn?: (key: string) => boolean,\n): RuleEvidence {\n  if (typeof rule.when === 'function') {\n    // FUNCTION PATH: temp recorder captures reads (lazy — skip if no recorder support)\n    const hasRecorderSupport = Boolean(attachFn);\n    const collector = hasRecorderSupport ? new EvidenceCollector() : undefined;\n    if (collector && attachFn) attachFn(collector);\n\n    let matched: boolean;\n    let matchError: string | undefined;\n    try {\n      matched = rule.when(scope);\n    } catch (e) {\n      matched = false;\n      // Capture the error for debugging — surface it in evidence instead of swallowing silently\n      matchError = e instanceof Error ? e.message : String(e);\n      if (isDevMode()) {\n        const label = rule.label ? ` ('${rule.label}')` : '';\n        // eslint-disable-next-line no-console\n        console.warn(`[footprint] decide() rule ${index}${label} threw during evaluation: ${matchError}`);\n      }\n    } finally {\n      if (collector && detachFn) detachFn(collector.id);\n    }\n\n    const evidence: FunctionRuleEvidence = {\n      type: 'function',\n      ruleIndex: index,\n      branch: rule.then,\n      matched,\n      label: rule.label,\n      // Partial reads: if rule threw after some getValue() calls, collector holds reads up to the throw point\n      inputs: collector?.getInputs() ?? [],\n      ...(matchError !== undefined && { matchError }),\n    };\n    return evidence;\n  } else {\n    // FILTER PATH: reads values directly via callbacks (no recorder); exceptions treated as non-match\n    const resolvedValueFn = valueFn ?? (() => undefined);\n    const resolvedRedactedFn = redactedFn ?? (() => false);\n    let filterMatched = false;\n    let filterConditions: FilterCondition[] = [];\n    let matchError: string | undefined;\n    try {\n      const result = evaluateFilter(resolvedValueFn, resolvedRedactedFn, rule.when as WhereFilter<S>);\n      filterMatched = result.matched;\n      filterConditions = result.conditions;\n    } catch (e) {\n      filterMatched = false;\n      filterConditions = [];\n      // Capture the error for debugging — surface it in evidence instead of swallowing silently\n      matchError = e instanceof Error ? e.message : String(e);\n      if (isDevMode()) {\n        const label = rule.label ? ` ('${rule.label}')` : '';\n        // eslint-disable-next-line no-console\n        console.warn(`[footprint] decide() filter rule ${index}${label} threw during evaluation: ${matchError}`);\n      }\n    }\n\n    const evidence: FilterRuleEvidence = {\n      type: 'filter',\n      ruleIndex: index,\n      branch: rule.then,\n      matched: filterMatched,\n      label: rule.label,\n      conditions: filterConditions,\n      ...(matchError !== undefined && { matchError }),\n    };\n    return evidence;\n  }\n}\n\n// -- decide() ----------------------------------------------------------------\n\n/**\n * Evaluates rules in order (first-match). Returns a branded DecisionResult.\n *\n * @param scope - TypedScope or ScopeFacade\n * @param rules - Array of DecideRule (function or filter when clauses)\n * @param defaultBranch - Branch ID if no rule matches\n *\n * **Error behavior:** If a `when` function throws during evaluation, the rule is\n * treated as non-matching (`matched: false`) and the error message is captured in\n * `matchError` on that rule's `RuleEvidence` entry. Execution continues with\n * subsequent rules; errors do not propagate to the caller.\n */\nexport function decide<S extends object>(scope: S, rules: DecideRule<S>[], defaultBranch: string): DecisionResult {\n  const attachFn = getAttachFn(scope);\n  const detachFn = getDetachFn(scope);\n  const valueFn = getValueFn(scope);\n  const redactedFn = getRedactedFn(scope);\n\n  const evaluatedRules: RuleEvidence[] = [];\n\n  for (const [index, rule] of rules.entries()) {\n    const ruleEvidence = evaluateRule(scope, rule, index, attachFn, detachFn, valueFn, redactedFn);\n    evaluatedRules.push(ruleEvidence);\n\n    if (ruleEvidence.matched) {\n      const evidence: DecisionEvidence = {\n        rules: evaluatedRules,\n        chosen: rule.then,\n        default: defaultBranch,\n      };\n      return { branch: rule.then, [DECISION_RESULT]: true, evidence };\n    }\n  }\n\n  // Default: no rule matched\n  const evidence: DecisionEvidence = {\n    rules: evaluatedRules,\n    chosen: defaultBranch,\n    default: defaultBranch,\n  };\n  return { branch: defaultBranch, [DECISION_RESULT]: true, evidence };\n}\n\n// -- select() ----------------------------------------------------------------\n\n/**\n * Evaluates ALL rules (not first-match). Returns a branded SelectionResult.\n *\n * @param scope - TypedScope or ScopeFacade\n * @param rules - Array of DecideRule (function or filter when clauses)\n *\n * **Error behavior:** If a `when` function throws during evaluation, the rule is\n * treated as non-matching (`matched: false`) and the error message is captured in\n * `matchError` on that rule's `RuleEvidence` entry. Evaluation continues with\n * remaining rules; errors do not propagate to the caller.\n */\nexport function select<S extends object>(scope: S, rules: DecideRule<S>[]): SelectionResult {\n  const attachFn = getAttachFn(scope);\n  const detachFn = getDetachFn(scope);\n  const valueFn = getValueFn(scope);\n  const redactedFn = getRedactedFn(scope);\n\n  const evaluatedRules: RuleEvidence[] = [];\n  const selectedBranches: string[] = [];\n\n  for (const [index, rule] of rules.entries()) {\n    const ruleEvidence = evaluateRule(scope, rule, index, attachFn, detachFn, valueFn, redactedFn);\n    evaluatedRules.push(ruleEvidence);\n\n    if (ruleEvidence.matched) {\n      selectedBranches.push(rule.then);\n    }\n  }\n\n  const evidence: SelectionEvidence = {\n    rules: evaluatedRules,\n    selected: selectedBranches,\n  };\n  return { branches: selectedBranches, [DECISION_RESULT]: true, evidence };\n}\n"]}
@@ -10,4 +10,4 @@ Object.defineProperty(exports, "__esModule", { value: true });
10
10
  exports.DECISION_RESULT = void 0;
11
11
  // -- Symbol Brand (duck-type safety) -----------------------------------------
12
12
  exports.DECISION_RESULT = Symbol('footprint:decide:result');
13
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL2RlY2lkZS90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7OztHQU1HOzs7QUF5Q0gsK0VBQStFO0FBRWxFLFFBQUEsZUFBZSxHQUFHLE1BQU0sQ0FBQyx5QkFBeUIsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBkZWNpZGUvdHlwZXMgLS0gVHlwZSBkZWZpbml0aW9ucyBmb3IgdGhlIGRlY2lkZSgpL3NlbGVjdCgpIGRlY2lzaW9uIHJlYXNvbmluZyBzeXN0ZW0uXG4gKlxuICogVHdvIGB3aGVuYCBmb3JtYXRzIGluIG9uZSBBUEk6XG4gKiAtIEZ1bmN0aW9uOiAocykgPT4gcy5jcmVkaXRTY29yZSA+IDcwMCAgIChhdXRvLWNhcHR1cmVzIHJlYWRzIHZpYSB0ZW1wIHJlY29yZGVyKVxuICogLSBGaWx0ZXI6ICAgeyBjcmVkaXRTY29yZTogeyBndDogNzAwIH0gfSAoY2FwdHVyZXMgcmVhZHMgKyBvcGVyYXRvcnMgKyB0aHJlc2hvbGRzKVxuICovXG5cbi8vIC0tIEZpbHRlciBPcGVyYXRvcnMgKFByaXNtYSBuYW1pbmcsIDggb3BlcmF0b3JzKSAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuLyoqIE11bHRpcGxlIG9wZXJhdG9ycyBvbiB0aGUgc2FtZSBrZXkgYXJlIEFORGVkIChlLmcuLCB7IGd0OiA1LCBsdDogMTAgfSBtZWFucyA1IDwgdmFsdWUgPCAxMCkuICovXG5leHBvcnQgdHlwZSBGaWx0ZXJPcHM8Vj4gPSB7XG4gIC8qKiBFcXVhbDogdmFsdWUgPT09IHRocmVzaG9sZCAqL1xuICBlcT86IFY7XG4gIC8qKiBOb3QgZXF1YWw6IHZhbHVlICE9PSB0aHJlc2hvbGQgKi9cbiAgbmU/OiBWO1xuICAvKiogR3JlYXRlciB0aGFuOiB2YWx1ZSA+IHRocmVzaG9sZCAqL1xuICBndD86IFY7XG4gIC8qKiBHcmVhdGVyIHRoYW4gb3IgZXF1YWw6IHZhbHVlID49IHRocmVzaG9sZCAqL1xuICBndGU/OiBWO1xuICAvKiogTGVzcyB0aGFuOiB2YWx1ZSA8IHRocmVzaG9sZCAqL1xuICBsdD86IFY7XG4gIC8qKiBMZXNzIHRoYW4gb3IgZXF1YWw6IHZhbHVlIDw9IHRocmVzaG9sZCAqL1xuICBsdGU/OiBWO1xuICAvKiogVmFsdWUgaXMgaW4gYXJyYXkgKi9cbiAgaW4/OiBWW107XG4gIC8qKiBWYWx1ZSBpcyBOT1QgaW4gYXJyYXkgKi9cbiAgbm90SW4/OiBWW107XG59O1xuXG4vLyAtLSBXaGVyZUZpbHRlciAoZmxhdCBrZXlzIG9ubHksIG5vIG5lc3RlZCB2MSkgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cbmV4cG9ydCB0eXBlIFdoZXJlRmlsdGVyPFQgZXh0ZW5kcyBvYmplY3QgPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4gPSB7XG4gIFtLIGluIGtleW9mIFRdPzogRmlsdGVyT3BzPFRbS10+O1xufTtcblxuLy8gLS0gUnVsZSBEZWZpbml0aW9uIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuXG5leHBvcnQgdHlwZSBXaGVuQ2xhdXNlPFQgZXh0ZW5kcyBvYmplY3QgPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4gPSAoKHM6IFQpID0+IGJvb2xlYW4pIHwgV2hlcmVGaWx0ZXI8VD47XG5cbmV4cG9ydCBpbnRlcmZhY2UgRGVjaWRlUnVsZTxUIGV4dGVuZHMgb2JqZWN0ID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj4+IHtcbiAgd2hlbjogV2hlbkNsYXVzZTxUPjtcbiAgdGhlbjogc3RyaW5nO1xuICAvKiogSHVtYW4tcmVhZGFibGUgcnVsZSBuYW1lIGZvciBuYXJyYXRpdmU6IFwiR29vZCBjcmVkaXRcIiAqL1xuICBsYWJlbD86IHN0cmluZztcbn1cblxuLy8gLS0gU3ltYm9sIEJyYW5kIChkdWNrLXR5cGUgc2FmZXR5KSAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuXG5leHBvcnQgY29uc3QgREVDSVNJT05fUkVTVUxUID0gU3ltYm9sKCdmb290cHJpbnQ6ZGVjaWRlOnJlc3VsdCcpO1xuXG4vLyAtLSBEZWNpc2lvbiBSZXN1bHQgKGZyb20gZGVjaWRlKCkpIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuZXhwb3J0IGludGVyZmFjZSBEZWNpc2lvblJlc3VsdCB7XG4gIGJyYW5jaDogc3RyaW5nO1xuICBbREVDSVNJT05fUkVTVUxUXTogdHJ1ZTtcbiAgZXZpZGVuY2U6IERlY2lzaW9uRXZpZGVuY2U7XG59XG5cbi8vIC0tIFNlbGVjdGlvbiBSZXN1bHQgKGZyb20gc2VsZWN0KCkpIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuZXhwb3J0IGludGVyZmFjZSBTZWxlY3Rpb25SZXN1bHQge1xuICBicmFuY2hlczogc3RyaW5nW107XG4gIFtERUNJU0lPTl9SRVNVTFRdOiB0cnVlO1xuICBldmlkZW5jZTogU2VsZWN0aW9uRXZpZGVuY2U7XG59XG5cbi8vIC0tIEV2aWRlbmNlIFR5cGVzIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuZXhwb3J0IGludGVyZmFjZSBGdW5jdGlvblJ1bGVFdmlkZW5jZSB7XG4gIHR5cGU6ICdmdW5jdGlvbic7XG4gIHJ1bGVJbmRleDogbnVtYmVyO1xuICAvKiogVGhlIGJyYW5jaCBJRCB0aGlzIHJ1bGUgbWFwcyB0by4gU2VsZi1kZXNjcmliaW5nIOKAlCBubyBpbmRleCBjb3JyZWxhdGlvbiBuZWVkZWQuICovXG4gIGJyYW5jaDogc3RyaW5nO1xuICBtYXRjaGVkOiBib29sZWFuO1xuICBsYWJlbD86IHN0cmluZztcbiAgaW5wdXRzOiBSZWFkSW5wdXRbXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBGaWx0ZXJSdWxlRXZpZGVuY2Uge1xuICB0eXBlOiAnZmlsdGVyJztcbiAgcnVsZUluZGV4OiBudW1iZXI7XG4gIC8qKiBUaGUgYnJhbmNoIElEIHRoaXMgcnVsZSBtYXBzIHRvLiBTZWxmLWRlc2NyaWJpbmcg4oCUIG5vIGluZGV4IGNvcnJlbGF0aW9uIG5lZWRlZC4gKi9cbiAgYnJhbmNoOiBzdHJpbmc7XG4gIG1hdGNoZWQ6IGJvb2xlYW47XG4gIGxhYmVsPzogc3RyaW5nO1xuICBjb25kaXRpb25zOiBGaWx0ZXJDb25kaXRpb25bXTtcbn1cblxuZXhwb3J0IHR5cGUgUnVsZUV2aWRlbmNlID0gRnVuY3Rpb25SdWxlRXZpZGVuY2UgfCBGaWx0ZXJSdWxlRXZpZGVuY2U7XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVhZElucHV0IHtcbiAga2V5OiBzdHJpbmc7XG4gIHZhbHVlU3VtbWFyeTogc3RyaW5nO1xuICByZWRhY3RlZDogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBGaWx0ZXJDb25kaXRpb24ge1xuICBrZXk6IHN0cmluZztcbiAgb3A6IHN0cmluZztcbiAgLyoqIEtlcHQgcmF3IGZvciBhdWRpdCBhY2N1cmFjeTsgZW5naW5lL3NlcmlhbGl6ZXIgbXVzdCBoYW5kbGUgc2FmZWx5LiAqL1xuICB0aHJlc2hvbGQ6IHVua25vd247XG4gIGFjdHVhbFN1bW1hcnk6IHN0cmluZztcbiAgcmVzdWx0OiBib29sZWFuO1xuICByZWRhY3RlZDogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBEZWNpc2lvbkV2aWRlbmNlIHtcbiAgcnVsZXM6IFJ1bGVFdmlkZW5jZVtdO1xuICAvKiogVGhlIGJyYW5jaCBzZWxlY3RlZC4gRXF1YWxzIGBkZWZhdWx0YCB3aGVuIG5vIHJ1bGUgbWF0Y2hlZC4gKi9cbiAgY2hvc2VuOiBzdHJpbmc7XG4gIC8qKiBUaGUgZmFsbGJhY2sgYnJhbmNoIHBhc3NlZCBhcyBkZWZhdWx0QnJhbmNoLiBBbHdheXMgc2V0LiAqL1xuICBkZWZhdWx0OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VsZWN0aW9uRXZpZGVuY2Uge1xuICBydWxlczogUnVsZUV2aWRlbmNlW107XG4gIHNlbGVjdGVkOiBzdHJpbmdbXTtcbn1cbiJdfQ==
13
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL2RlY2lkZS90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7OztHQU1HOzs7QUF5Q0gsK0VBQStFO0FBRWxFLFFBQUEsZUFBZSxHQUFHLE1BQU0sQ0FBQyx5QkFBeUIsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBkZWNpZGUvdHlwZXMgLS0gVHlwZSBkZWZpbml0aW9ucyBmb3IgdGhlIGRlY2lkZSgpL3NlbGVjdCgpIGRlY2lzaW9uIHJlYXNvbmluZyBzeXN0ZW0uXG4gKlxuICogVHdvIGB3aGVuYCBmb3JtYXRzIGluIG9uZSBBUEk6XG4gKiAtIEZ1bmN0aW9uOiAocykgPT4gcy5jcmVkaXRTY29yZSA+IDcwMCAgIChhdXRvLWNhcHR1cmVzIHJlYWRzIHZpYSB0ZW1wIHJlY29yZGVyKVxuICogLSBGaWx0ZXI6ICAgeyBjcmVkaXRTY29yZTogeyBndDogNzAwIH0gfSAoY2FwdHVyZXMgcmVhZHMgKyBvcGVyYXRvcnMgKyB0aHJlc2hvbGRzKVxuICovXG5cbi8vIC0tIEZpbHRlciBPcGVyYXRvcnMgKFByaXNtYSBuYW1pbmcsIDggb3BlcmF0b3JzKSAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuLyoqIE11bHRpcGxlIG9wZXJhdG9ycyBvbiB0aGUgc2FtZSBrZXkgYXJlIEFORGVkIChlLmcuLCB7IGd0OiA1LCBsdDogMTAgfSBtZWFucyA1IDwgdmFsdWUgPCAxMCkuICovXG5leHBvcnQgdHlwZSBGaWx0ZXJPcHM8Vj4gPSB7XG4gIC8qKiBFcXVhbDogdmFsdWUgPT09IHRocmVzaG9sZCAqL1xuICBlcT86IFY7XG4gIC8qKiBOb3QgZXF1YWw6IHZhbHVlICE9PSB0aHJlc2hvbGQgKi9cbiAgbmU/OiBWO1xuICAvKiogR3JlYXRlciB0aGFuOiB2YWx1ZSA+IHRocmVzaG9sZCAqL1xuICBndD86IFY7XG4gIC8qKiBHcmVhdGVyIHRoYW4gb3IgZXF1YWw6IHZhbHVlID49IHRocmVzaG9sZCAqL1xuICBndGU/OiBWO1xuICAvKiogTGVzcyB0aGFuOiB2YWx1ZSA8IHRocmVzaG9sZCAqL1xuICBsdD86IFY7XG4gIC8qKiBMZXNzIHRoYW4gb3IgZXF1YWw6IHZhbHVlIDw9IHRocmVzaG9sZCAqL1xuICBsdGU/OiBWO1xuICAvKiogVmFsdWUgaXMgaW4gYXJyYXkgKi9cbiAgaW4/OiBWW107XG4gIC8qKiBWYWx1ZSBpcyBOT1QgaW4gYXJyYXkgKi9cbiAgbm90SW4/OiBWW107XG59O1xuXG4vLyAtLSBXaGVyZUZpbHRlciAoZmxhdCBrZXlzIG9ubHksIG5vIG5lc3RlZCB2MSkgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cbmV4cG9ydCB0eXBlIFdoZXJlRmlsdGVyPFQgZXh0ZW5kcyBvYmplY3QgPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4gPSB7XG4gIFtLIGluIGtleW9mIFRdPzogRmlsdGVyT3BzPFRbS10+O1xufTtcblxuLy8gLS0gUnVsZSBEZWZpbml0aW9uIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuXG5leHBvcnQgdHlwZSBXaGVuQ2xhdXNlPFQgZXh0ZW5kcyBvYmplY3QgPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4gPSAoKHM6IFQpID0+IGJvb2xlYW4pIHwgV2hlcmVGaWx0ZXI8VD47XG5cbmV4cG9ydCBpbnRlcmZhY2UgRGVjaWRlUnVsZTxUIGV4dGVuZHMgb2JqZWN0ID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj4+IHtcbiAgd2hlbjogV2hlbkNsYXVzZTxUPjtcbiAgdGhlbjogc3RyaW5nO1xuICAvKiogSHVtYW4tcmVhZGFibGUgcnVsZSBuYW1lIGZvciBuYXJyYXRpdmU6IFwiR29vZCBjcmVkaXRcIiAqL1xuICBsYWJlbD86IHN0cmluZztcbn1cblxuLy8gLS0gU3ltYm9sIEJyYW5kIChkdWNrLXR5cGUgc2FmZXR5KSAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuXG5leHBvcnQgY29uc3QgREVDSVNJT05fUkVTVUxUID0gU3ltYm9sKCdmb290cHJpbnQ6ZGVjaWRlOnJlc3VsdCcpO1xuXG4vLyAtLSBEZWNpc2lvbiBSZXN1bHQgKGZyb20gZGVjaWRlKCkpIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuZXhwb3J0IGludGVyZmFjZSBEZWNpc2lvblJlc3VsdCB7XG4gIGJyYW5jaDogc3RyaW5nO1xuICBbREVDSVNJT05fUkVTVUxUXTogdHJ1ZTtcbiAgZXZpZGVuY2U6IERlY2lzaW9uRXZpZGVuY2U7XG59XG5cbi8vIC0tIFNlbGVjdGlvbiBSZXN1bHQgKGZyb20gc2VsZWN0KCkpIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuZXhwb3J0IGludGVyZmFjZSBTZWxlY3Rpb25SZXN1bHQge1xuICBicmFuY2hlczogc3RyaW5nW107XG4gIFtERUNJU0lPTl9SRVNVTFRdOiB0cnVlO1xuICBldmlkZW5jZTogU2VsZWN0aW9uRXZpZGVuY2U7XG59XG5cbi8vIC0tIEV2aWRlbmNlIFR5cGVzIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuZXhwb3J0IGludGVyZmFjZSBGdW5jdGlvblJ1bGVFdmlkZW5jZSB7XG4gIHR5cGU6ICdmdW5jdGlvbic7XG4gIHJ1bGVJbmRleDogbnVtYmVyO1xuICAvKiogVGhlIGJyYW5jaCBJRCB0aGlzIHJ1bGUgbWFwcyB0by4gU2VsZi1kZXNjcmliaW5nIOKAlCBubyBpbmRleCBjb3JyZWxhdGlvbiBuZWVkZWQuICovXG4gIGJyYW5jaDogc3RyaW5nO1xuICBtYXRjaGVkOiBib29sZWFuO1xuICBsYWJlbD86IHN0cmluZztcbiAgaW5wdXRzOiBSZWFkSW5wdXRbXTtcbiAgLyoqXG4gICAqIEVycm9yIG1lc3NhZ2UgaWYgdGhlIGB3aGVuYCBmdW5jdGlvbiB0aHJldyBkdXJpbmcgZXZhbHVhdGlvbi5cbiAgICogUHJlc2VudCBvbmx5IHdoZW4gYW4gZXhjZXB0aW9uIG9jY3VycmVkOyBgbWF0Y2hlZGAgaXMgYGZhbHNlYCBpbiB0aGF0IGNhc2UuXG4gICAqIFN1cmZhY2VzIHRoZSBlcnJvciBmb3IgZGVidWdnaW5nIHJhdGhlciB0aGFuIHN3YWxsb3dpbmcgaXQgc2lsZW50bHkuXG4gICAqXG4gICAqICoqU2VjdXJpdHkgbm90ZToqKiBFcnJvciBtZXNzYWdlcyBmcm9tIHVzZXItcHJvdmlkZWQgYHdoZW5gIGZ1bmN0aW9ucyBhcmUgY2FwdHVyZWRcbiAgICogYXMtaXMgYW5kIGFyZSBOT1QgZmlsdGVyZWQgdGhyb3VnaCB0aGUgcmVkYWN0aW9uIHBvbGljeS4gQXZvaWQgaW5jbHVkaW5nIHNlbnNpdGl2ZVxuICAgKiBzY29wZSB2YWx1ZXMgaW4gdGhyb3duIGVycm9yIG1lc3NhZ2VzLlxuICAgKi9cbiAgbWF0Y2hFcnJvcj86IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBGaWx0ZXJSdWxlRXZpZGVuY2Uge1xuICB0eXBlOiAnZmlsdGVyJztcbiAgcnVsZUluZGV4OiBudW1iZXI7XG4gIC8qKiBUaGUgYnJhbmNoIElEIHRoaXMgcnVsZSBtYXBzIHRvLiBTZWxmLWRlc2NyaWJpbmcg4oCUIG5vIGluZGV4IGNvcnJlbGF0aW9uIG5lZWRlZC4gKi9cbiAgYnJhbmNoOiBzdHJpbmc7XG4gIG1hdGNoZWQ6IGJvb2xlYW47XG4gIGxhYmVsPzogc3RyaW5nO1xuICBjb25kaXRpb25zOiBGaWx0ZXJDb25kaXRpb25bXTtcbiAgLyoqXG4gICAqIEVycm9yIG1lc3NhZ2UgaWYgdGhlIGZpbHRlciBldmFsdWF0b3IgdGhyZXcgZHVyaW5nIGV2YWx1YXRpb24uXG4gICAqIFByZXNlbnQgb25seSB3aGVuIGFuIGV4Y2VwdGlvbiBvY2N1cnJlZDsgYG1hdGNoZWRgIGlzIGBmYWxzZWAgaW4gdGhhdCBjYXNlLlxuICAgKiBTdXJmYWNlcyB0aGUgZXJyb3IgZm9yIGRlYnVnZ2luZyByYXRoZXIgdGhhbiBzd2FsbG93aW5nIGl0IHNpbGVudGx5LlxuICAgKlxuICAgKiAqKlNlY3VyaXR5IG5vdGU6KiogRXJyb3IgbWVzc2FnZXMgZnJvbSB1c2VyLXByb3ZpZGVkIGB3aGVuYCBmdW5jdGlvbnMgYXJlIGNhcHR1cmVkXG4gICAqIGFzLWlzIGFuZCBhcmUgTk9UIGZpbHRlcmVkIHRocm91Z2ggdGhlIHJlZGFjdGlvbiBwb2xpY3kuIEF2b2lkIGluY2x1ZGluZyBzZW5zaXRpdmVcbiAgICogc2NvcGUgdmFsdWVzIGluIHRocm93biBlcnJvciBtZXNzYWdlcy5cbiAgICovXG4gIG1hdGNoRXJyb3I/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCB0eXBlIFJ1bGVFdmlkZW5jZSA9IEZ1bmN0aW9uUnVsZUV2aWRlbmNlIHwgRmlsdGVyUnVsZUV2aWRlbmNlO1xuXG5leHBvcnQgaW50ZXJmYWNlIFJlYWRJbnB1dCB7XG4gIGtleTogc3RyaW5nO1xuICB2YWx1ZVN1bW1hcnk6IHN0cmluZztcbiAgcmVkYWN0ZWQ6IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRmlsdGVyQ29uZGl0aW9uIHtcbiAga2V5OiBzdHJpbmc7XG4gIG9wOiBzdHJpbmc7XG4gIC8qKiBLZXB0IHJhdyBmb3IgYXVkaXQgYWNjdXJhY3k7IGVuZ2luZS9zZXJpYWxpemVyIG11c3QgaGFuZGxlIHNhZmVseS4gKi9cbiAgdGhyZXNob2xkOiB1bmtub3duO1xuICBhY3R1YWxTdW1tYXJ5OiBzdHJpbmc7XG4gIHJlc3VsdDogYm9vbGVhbjtcbiAgcmVkYWN0ZWQ6IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRGVjaXNpb25FdmlkZW5jZSB7XG4gIHJ1bGVzOiBSdWxlRXZpZGVuY2VbXTtcbiAgLyoqIFRoZSBicmFuY2ggc2VsZWN0ZWQuIEVxdWFscyBgZGVmYXVsdGAgd2hlbiBubyBydWxlIG1hdGNoZWQuICovXG4gIGNob3Nlbjogc3RyaW5nO1xuICAvKiogVGhlIGZhbGxiYWNrIGJyYW5jaCBwYXNzZWQgYXMgZGVmYXVsdEJyYW5jaC4gQWx3YXlzIHNldC4gKi9cbiAgZGVmYXVsdDogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNlbGVjdGlvbkV2aWRlbmNlIHtcbiAgcnVsZXM6IFJ1bGVFdmlkZW5jZVtdO1xuICBzZWxlY3RlZDogc3RyaW5nW107XG59XG4iXX0=