footprintjs 4.0.3 → 4.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CLAUDE.md
CHANGED
|
@@ -46,7 +46,10 @@ const chart = flowChart<LoanState>('Intake', async (scope) => {
|
|
|
46
46
|
scope.creditTier = 'A'; // typed write
|
|
47
47
|
scope.amount = 50000; // typed write
|
|
48
48
|
scope.customer.address.zip = '90210'; // deep write (updateValue)
|
|
49
|
-
scope.tags.push('vip'); // array copy-on-write
|
|
49
|
+
scope.tags.push('vip'); // array copy-on-write (single push)
|
|
50
|
+
scope.$batchArray('tags', (arr) => { // O(1) batch: 1 clone + 1 commit
|
|
51
|
+
arr.push('vip', 'premium', 'verified');
|
|
52
|
+
});
|
|
50
53
|
scope.approved = true; // optional field
|
|
51
54
|
|
|
52
55
|
// $-prefixed escape hatches
|
|
@@ -50,4 +50,4 @@ export const EXECUTOR_INTERNAL_METHODS = new Set([
|
|
|
50
50
|
'useSharedRedactedKeys', // FlowChartExecutor.createTraverser() — redaction wrapping
|
|
51
51
|
'useRedactionPolicy', // FlowChartExecutor.createTraverser() — redaction wrapping
|
|
52
52
|
]);
|
|
53
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../src/lib/reactive/types.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAmHH,+EAA+E;AAC/E,uEAAuE;AAEvE,MAAM,CAAC,MAAM,kBAAkB,GAAG,IAAI,GAAG,CAAS;IAChD,WAAW;IACX,WAAW;IACX,SAAS;IACT,SAAS;IACT,OAAO;IACP,UAAU;IACV,SAAS;IACT,QAAQ;IACR,MAAM;IACN,QAAQ;IACR,SAAS;IACT,OAAO;IACP,iBAAiB;IACjB,iBAAiB;IACjB,eAAe;IACf,aAAa;IACb,QAAQ;IACR,QAAQ;CACT,CAAC,CAAC;AAEH,+EAA+E;AAC/E,gEAAgE;AAChE,qEAAqE;AAErE,MAAM,CAAC,MAAM,YAAY,GAAG,MAAM,CAAC,6BAA6B,CAAC,CAAC;AAElE,+EAA+E;AAC/E,6EAA6E;AAC7E,+CAA+C;AAE/C,MAAM,CAAC,MAAM,cAAc,GAAG,MAAM,CAAC,iCAAiC,CAAC,CAAC;AAExE,+EAA+E;AAC/E,gFAAgF;AAChF,0EAA0E;AAC1E,0EAA0E;AAE1E,MAAM,CAAC,MAAM,yBAAyB,GAAG,IAAI,GAAG,CAAC;IAC/C,kBAAkB,EAAE,4BAA4B;IAChD,gBAAgB,EAAE,4BAA4B;IAC9C,gBAAgB,EAAE,mEAAmE;IACrF,gBAAgB,EAAE,qCAAqC;IACvD,cAAc,EAAE,mCAAmC;IACnD,uBAAuB,EAAE,2DAA2D;IACpF,oBAAoB,EAAE,2DAA2D;CAClF,CAAC,CAAC","sourcesContent":["/**\n * reactive/types -- Type definitions for the TypedScope<T> reactive proxy system.\n *\n * TypedScope<T> wraps a ReactiveTarget (ScopeFacade) in a Proxy that provides\n * typed property access. All scope infrastructure methods are $-prefixed to\n * avoid collisions with user state keys.\n *\n * Dependency: type-only imports from engine/ and scope/ (zero runtime cost).\n */\n\nimport type { ExecutionEnv } from '../engine/types.js';\nimport type { Recorder } from '../scope/types.js';\n\n// -- ReactiveTarget ----------------------------------------------------------\n// Minimum protocol required by TypedScope -- a curated subset of ScopeFacade's\n// public API. Not a full mirror: excludes internal/redaction methods that are\n// handled at the executor level (useRedactionPolicy, useSharedRedactedKeys, etc.)\n// and infrastructure methods (getPipelineId, setGlobal, getGlobal, etc.).\n\nexport interface ReactiveTarget {\n  // State access (tracked by recorders)\n  getValue(key?: string): unknown;\n  setValue(key: string, value: unknown, shouldRedact?: boolean, description?: string): void;\n  updateValue(key: string, value: unknown, description?: string): void;\n  deleteValue(key: string, description?: string): void;\n\n  // Non-tracking state inspection (for proxy internals — no recorder dispatch)\n  /** Returns all state keys without firing onRead. Used by ownKeys/has traps. */\n  getStateKeys?(): string[];\n  /** Check key existence without firing onRead. Used by has trap. */\n  hasKey?(key: string): boolean;\n\n  // Input & environment (readonly, NOT tracked)\n  getArgs<T = Record<string, unknown>>(): T;\n  getEnv(): Readonly<ExecutionEnv>;\n\n  // Recorder management\n  attachRecorder(recorder: Recorder): void;\n  detachRecorder(recorderId: string): void;\n  getRecorders(): Recorder[];\n\n  // Diagnostics\n  addDebugInfo(key: string, value: unknown): void;\n  addDebugMessage(value: unknown): void;\n  addErrorInfo(key: string, value: unknown): void;\n  addMetric(name: string, value: unknown): void;\n  addEval(name: string, value: unknown): void;\n}\n\n// -- ScopeMethods ------------------------------------------------------------\n// $-prefixed escape hatches. Non-enumerable on the proxy -- don't appear in\n// Object.keys(), destructuring, or for...in. Only state keys are visible.\n\nexport interface ScopeMethods {\n  // State (untyped escape hatch -- for dynamic keys, redaction, description)\n  $getValue(key: string): unknown;\n  $setValue(key: string, value: unknown, shouldRedact?: boolean, description?: string): void;\n  $update(key: string, value: unknown, description?: string): void;\n  $delete(key: string, description?: string): void;\n  /** Proxy-synthesized: calls getValue(rootKey) then lodash.get for nested path. Not a direct delegation. */\n  $read(dotPath: string): unknown;\n\n  // Input & environment (readonly)\n  $getArgs<T = Record<string, unknown>>(): T;\n  $getEnv(): Readonly<ExecutionEnv>;\n\n  // Observability\n  $debug(key: string, value: unknown): void;\n  $log(value: unknown): void;\n  $error(key: string, value: unknown): void;\n  $metric(name: string, value: unknown): void;\n  $eval(name: string, value: unknown): void;\n\n  // Recorder management\n  $attachRecorder(recorder: Recorder): void;\n  $detachRecorder(recorderId: string): void;\n  $getRecorders(): Recorder[];\n\n  /**\n   * Batch-mutate an array key in a single clone+write cycle.\n   *\n   * Every `scope.items.push(x)` clones the entire array and commits it — O(N) per call.\n   * For N mutations on an M-length array that is O(N×M). Use `$batchArray` to clone once,\n   * apply all mutations inside `fn`, then commit once — O(M) total.\n   *\n   * ```typescript\n   * // Before: 1000 clones × growing array = O(N²)\n   * for (let i = 0; i < 1000; i++) scope.items.push(i);\n   *\n   * // After: 1 clone + 1 commit = O(N)\n   * scope.$batchArray('items', (arr) => {\n   *   for (let i = 0; i < 1000; i++) arr.push(i);\n   * });\n   * ```\n   *\n   * `fn` receives a plain (non-proxy) mutable copy of the current array. Mutations\n   * inside `fn` are NOT tracked individually — only the final committed array appears\n   * in the narrative as a single write. If the key does not exist or is not an array,\n   * `fn` receives an empty array and the result is committed as the new value.\n   */\n  $batchArray(key: string, fn: (arr: unknown[]) => void): void;\n\n  // Pipeline control\n  $break(): void;\n\n  // Escape hatch -- unwrap to underlying ReactiveTarget\n  $toRaw(): ReactiveTarget;\n}\n\n// -- TypedScope<T> -----------------------------------------------------------\n// The consumer-facing type. T is the user's state interface.\n// Property access is typed; $-methods provide escape hatches.\n\nexport type TypedScope<T extends object = Record<string, unknown>> = T & ScopeMethods;\n\n// -- ReactiveOptions ---------------------------------------------------------\n// Configuration passed to createTypedScope.\n\nexport interface ReactiveOptions {\n  /** Pipeline break function -- injected by StageRunner after scope creation. */\n  breakPipeline?: () => void;\n}\n\n// -- Internal: $-method name set ---------------------------------------------\n// Used by the Proxy get trap to distinguish $-methods from state keys.\n\nexport const SCOPE_METHOD_NAMES = new Set<string>([\n  '$getValue',\n  '$setValue',\n  '$update',\n  '$delete',\n  '$read',\n  '$getArgs',\n  '$getEnv',\n  '$debug',\n  '$log',\n  '$error',\n  '$metric',\n  '$eval',\n  '$attachRecorder',\n  '$detachRecorder',\n  '$getRecorders',\n  '$batchArray',\n  '$break',\n  '$toRaw',\n]);\n\n// -- Internal: Symbol for deferred break injection ---------------------------\n// StageRunner sets this after scope creation so $break() works.\n// Private Symbol (not Symbol.for) to prevent cross-module tampering.\n\nexport const BREAK_SETTER = Symbol('footprint:reactive:setBreak');\n\n// -- Internal: Symbol for TypedScope detection -------------------------------\n// Used by StageRunner to skip createProtectedScope for TypedScope instances.\n// Private Symbol prevents string-tag spoofing.\n\nexport const IS_TYPED_SCOPE = Symbol('footprint:reactive:isTypedScope');\n\n// -- Internal: executor method allowlist -------------------------------------\n// ScopeFacade methods called by FlowChartExecutor wrapping code and StageRunner\n// THROUGH a TypedScope proxy. Only add methods with confirmed call sites.\n// Shared between createTypedScope.ts and StageRunner.ts to prevent drift.\n\nexport const EXECUTOR_INTERNAL_METHODS = new Set([\n  'notifyStageStart', // StageRunner.run() line 59\n  'notifyStageEnd', // StageRunner.run() line 79\n  'attachRecorder', // FlowChartExecutor.createTraverser() — narrative + user recorders\n  'detachRecorder', // FlowChartExecutor.detachRecorder()\n  'getRecorders', // FlowChartExecutor.getRecorders()\n  'useSharedRedactedKeys', // FlowChartExecutor.createTraverser() — redaction wrapping\n  'useRedactionPolicy', // FlowChartExecutor.createTraverser() — redaction wrapping\n]);\n"]}
|
|
53
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../src/lib/reactive/types.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AA+HH,+EAA+E;AAC/E,uEAAuE;AAEvE,MAAM,CAAC,MAAM,kBAAkB,GAAG,IAAI,GAAG,CAAS;IAChD,WAAW;IACX,WAAW;IACX,SAAS;IACT,SAAS;IACT,OAAO;IACP,UAAU;IACV,SAAS;IACT,QAAQ;IACR,MAAM;IACN,QAAQ;IACR,SAAS;IACT,OAAO;IACP,iBAAiB;IACjB,iBAAiB;IACjB,eAAe;IACf,aAAa;IACb,QAAQ;IACR,QAAQ;CACT,CAAC,CAAC;AAEH,+EAA+E;AAC/E,gEAAgE;AAChE,qEAAqE;AAErE,MAAM,CAAC,MAAM,YAAY,GAAG,MAAM,CAAC,6BAA6B,CAAC,CAAC;AAElE,+EAA+E;AAC/E,6EAA6E;AAC7E,+CAA+C;AAE/C,MAAM,CAAC,MAAM,cAAc,GAAG,MAAM,CAAC,iCAAiC,CAAC,CAAC;AAExE,+EAA+E;AAC/E,gFAAgF;AAChF,0EAA0E;AAC1E,0EAA0E;AAE1E,MAAM,CAAC,MAAM,yBAAyB,GAAG,IAAI,GAAG,CAAC;IAC/C,kBAAkB,EAAE,4BAA4B;IAChD,gBAAgB,EAAE,4BAA4B;IAC9C,gBAAgB,EAAE,mEAAmE;IACrF,gBAAgB,EAAE,qCAAqC;IACvD,cAAc,EAAE,mCAAmC;IACnD,uBAAuB,EAAE,2DAA2D;IACpF,oBAAoB,EAAE,2DAA2D;CAClF,CAAC,CAAC","sourcesContent":["/**\n * reactive/types -- Type definitions for the TypedScope<T> reactive proxy system.\n *\n * TypedScope<T> wraps a ReactiveTarget (ScopeFacade) in a Proxy that provides\n * typed property access. All scope infrastructure methods are $-prefixed to\n * avoid collisions with user state keys.\n *\n * Dependency: type-only imports from engine/ and scope/ (zero runtime cost).\n */\n\nimport type { ExecutionEnv } from '../engine/types.js';\nimport type { Recorder } from '../scope/types.js';\n\n// -- ReactiveTarget ----------------------------------------------------------\n// Minimum protocol required by TypedScope -- a curated subset of ScopeFacade's\n// public API. Not a full mirror: excludes internal/redaction methods that are\n// handled at the executor level (useRedactionPolicy, useSharedRedactedKeys, etc.)\n// and infrastructure methods (getPipelineId, setGlobal, getGlobal, etc.).\n\nexport interface ReactiveTarget {\n  // State access (tracked by recorders)\n  getValue(key?: string): unknown;\n  setValue(key: string, value: unknown, shouldRedact?: boolean, description?: string): void;\n  updateValue(key: string, value: unknown, description?: string): void;\n  deleteValue(key: string, description?: string): void;\n\n  // Non-tracking state inspection (for proxy internals — no recorder dispatch)\n  /** Returns all state keys without firing onRead. Used by ownKeys/has traps. */\n  getStateKeys?(): string[];\n  /** Check key existence without firing onRead. Used by has trap. */\n  hasKey?(key: string): boolean;\n\n  // Input & environment (readonly, NOT tracked)\n  getArgs<T = Record<string, unknown>>(): T;\n  getEnv(): Readonly<ExecutionEnv>;\n\n  // Recorder management\n  attachRecorder(recorder: Recorder): void;\n  detachRecorder(recorderId: string): void;\n  getRecorders(): Recorder[];\n\n  // Diagnostics\n  addDebugInfo(key: string, value: unknown): void;\n  addDebugMessage(value: unknown): void;\n  addErrorInfo(key: string, value: unknown): void;\n  addMetric(name: string, value: unknown): void;\n  addEval(name: string, value: unknown): void;\n}\n\n// -- ScopeMethods ------------------------------------------------------------\n// $-prefixed escape hatches. Non-enumerable on the proxy -- don't appear in\n// Object.keys(), destructuring, or for...in. Only state keys are visible.\n\nexport interface ScopeMethods {\n  // State (untyped escape hatch -- for dynamic keys, redaction, description)\n  $getValue(key: string): unknown;\n  $setValue(key: string, value: unknown, shouldRedact?: boolean, description?: string): void;\n  $update(key: string, value: unknown, description?: string): void;\n  $delete(key: string, description?: string): void;\n  /** Proxy-synthesized: calls getValue(rootKey) then lodash.get for nested path. Not a direct delegation. */\n  $read(dotPath: string): unknown;\n\n  // Input & environment (readonly)\n  $getArgs<T = Record<string, unknown>>(): T;\n  $getEnv(): Readonly<ExecutionEnv>;\n\n  // Observability\n  $debug(key: string, value: unknown): void;\n  $log(value: unknown): void;\n  $error(key: string, value: unknown): void;\n  $metric(name: string, value: unknown): void;\n  $eval(name: string, value: unknown): void;\n\n  // Recorder management\n  $attachRecorder(recorder: Recorder): void;\n  $detachRecorder(recorderId: string): void;\n  $getRecorders(): Recorder[];\n\n  /**\n   * Batch-mutate an array key in a single clone+write cycle.\n   *\n   * Every `scope.items.push(x)` clones the entire array and commits it — O(N) per call.\n   * For N mutations on an M-length array that is O(N×M). Use `$batchArray` to clone once,\n   * apply all mutations inside `fn`, then commit once — O(M) total.\n   *\n   * ```typescript\n   * // Before: 1000 clones × growing array = O(N²)\n   * for (let i = 0; i < 1000; i++) scope.items.push(i);\n   *\n   * // After: 1 clone + 1 commit = O(N)\n   * scope.$batchArray('items', (arr) => {\n   *   for (let i = 0; i < 1000; i++) arr.push(i);\n   * });\n   * ```\n   *\n   * `fn` receives a plain (non-proxy) mutable **shallow copy** of the current array.\n   * The array itself is a new instance, but object references inside it are shared with\n   * the original state — mutations to nested objects inside `fn` affect those originals.\n   * Only push/pop/sort/splice and other operations that change the array's own slots are\n   * safely isolated.\n   *\n   * Mutations inside `fn` are NOT tracked individually — only the final committed array\n   * appears in the narrative as a single write. If the key does not exist or is not an\n   * array, `fn` receives an empty array and the result is committed as the new value.\n   *\n   * If `fn` throws, `setValue` is never called and state remains unchanged (atomic on\n   * error). The exception propagates to the caller.\n   *\n   * `key` is untyped (`string`) — TypeScript will not catch typos. `arr` is typed as\n   * `unknown[]` because `ScopeMethods` is not parameterized by `T`; cast inside `fn`\n   * when element types are known: `(arr as string[]).push(x)`.\n   */\n  $batchArray(key: string, fn: (arr: unknown[]) => void): void;\n\n  // Pipeline control\n  $break(): void;\n\n  // Escape hatch -- unwrap to underlying ReactiveTarget\n  $toRaw(): ReactiveTarget;\n}\n\n// -- TypedScope<T> -----------------------------------------------------------\n// The consumer-facing type. T is the user's state interface.\n// Property access is typed; $-methods provide escape hatches.\n\nexport type TypedScope<T extends object = Record<string, unknown>> = T & ScopeMethods;\n\n// -- ReactiveOptions ---------------------------------------------------------\n// Configuration passed to createTypedScope.\n\nexport interface ReactiveOptions {\n  /** Pipeline break function -- injected by StageRunner after scope creation. */\n  breakPipeline?: () => void;\n}\n\n// -- Internal: $-method name set ---------------------------------------------\n// Used by the Proxy get trap to distinguish $-methods from state keys.\n\nexport const SCOPE_METHOD_NAMES = new Set<string>([\n  '$getValue',\n  '$setValue',\n  '$update',\n  '$delete',\n  '$read',\n  '$getArgs',\n  '$getEnv',\n  '$debug',\n  '$log',\n  '$error',\n  '$metric',\n  '$eval',\n  '$attachRecorder',\n  '$detachRecorder',\n  '$getRecorders',\n  '$batchArray',\n  '$break',\n  '$toRaw',\n]);\n\n// -- Internal: Symbol for deferred break injection ---------------------------\n// StageRunner sets this after scope creation so $break() works.\n// Private Symbol (not Symbol.for) to prevent cross-module tampering.\n\nexport const BREAK_SETTER = Symbol('footprint:reactive:setBreak');\n\n// -- Internal: Symbol for TypedScope detection -------------------------------\n// Used by StageRunner to skip createProtectedScope for TypedScope instances.\n// Private Symbol prevents string-tag spoofing.\n\nexport const IS_TYPED_SCOPE = Symbol('footprint:reactive:isTypedScope');\n\n// -- Internal: executor method allowlist -------------------------------------\n// ScopeFacade methods called by FlowChartExecutor wrapping code and StageRunner\n// THROUGH a TypedScope proxy. Only add methods with confirmed call sites.\n// Shared between createTypedScope.ts and StageRunner.ts to prevent drift.\n\nexport const EXECUTOR_INTERNAL_METHODS = new Set([\n  'notifyStageStart', // StageRunner.run() line 59\n  'notifyStageEnd', // StageRunner.run() line 79\n  'attachRecorder', // FlowChartExecutor.createTraverser() — narrative + user recorders\n  'detachRecorder', // FlowChartExecutor.detachRecorder()\n  'getRecorders', // FlowChartExecutor.getRecorders()\n  'useSharedRedactedKeys', // FlowChartExecutor.createTraverser() — redaction wrapping\n  'useRedactionPolicy', // FlowChartExecutor.createTraverser() — redaction wrapping\n]);\n"]}
|
|
@@ -53,4 +53,4 @@ exports.EXECUTOR_INTERNAL_METHODS = new Set([
|
|
|
53
53
|
'useSharedRedactedKeys', // FlowChartExecutor.createTraverser() — redaction wrapping
|
|
54
54
|
'useRedactionPolicy', // FlowChartExecutor.createTraverser() — redaction wrapping
|
|
55
55
|
]);
|
|
56
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/lib/reactive/types.ts"],"names":[],"mappings":";AAAA;;;;;;;;GAQG;;;AAmHH,+EAA+E;AAC/E,uEAAuE;AAE1D,QAAA,kBAAkB,GAAG,IAAI,GAAG,CAAS;IAChD,WAAW;IACX,WAAW;IACX,SAAS;IACT,SAAS;IACT,OAAO;IACP,UAAU;IACV,SAAS;IACT,QAAQ;IACR,MAAM;IACN,QAAQ;IACR,SAAS;IACT,OAAO;IACP,iBAAiB;IACjB,iBAAiB;IACjB,eAAe;IACf,aAAa;IACb,QAAQ;IACR,QAAQ;CACT,CAAC,CAAC;AAEH,+EAA+E;AAC/E,gEAAgE;AAChE,qEAAqE;AAExD,QAAA,YAAY,GAAG,MAAM,CAAC,6BAA6B,CAAC,CAAC;AAElE,+EAA+E;AAC/E,6EAA6E;AAC7E,+CAA+C;AAElC,QAAA,cAAc,GAAG,MAAM,CAAC,iCAAiC,CAAC,CAAC;AAExE,+EAA+E;AAC/E,gFAAgF;AAChF,0EAA0E;AAC1E,0EAA0E;AAE7D,QAAA,yBAAyB,GAAG,IAAI,GAAG,CAAC;IAC/C,kBAAkB,EAAE,4BAA4B;IAChD,gBAAgB,EAAE,4BAA4B;IAC9C,gBAAgB,EAAE,mEAAmE;IACrF,gBAAgB,EAAE,qCAAqC;IACvD,cAAc,EAAE,mCAAmC;IACnD,uBAAuB,EAAE,2DAA2D;IACpF,oBAAoB,EAAE,2DAA2D;CAClF,CAAC,CAAC","sourcesContent":["/**\n * reactive/types -- Type definitions for the TypedScope<T> reactive proxy system.\n *\n * TypedScope<T> wraps a ReactiveTarget (ScopeFacade) in a Proxy that provides\n * typed property access. All scope infrastructure methods are $-prefixed to\n * avoid collisions with user state keys.\n *\n * Dependency: type-only imports from engine/ and scope/ (zero runtime cost).\n */\n\nimport type { ExecutionEnv } from '../engine/types.js';\nimport type { Recorder } from '../scope/types.js';\n\n// -- ReactiveTarget ----------------------------------------------------------\n// Minimum protocol required by TypedScope -- a curated subset of ScopeFacade's\n// public API. Not a full mirror: excludes internal/redaction methods that are\n// handled at the executor level (useRedactionPolicy, useSharedRedactedKeys, etc.)\n// and infrastructure methods (getPipelineId, setGlobal, getGlobal, etc.).\n\nexport interface ReactiveTarget {\n  // State access (tracked by recorders)\n  getValue(key?: string): unknown;\n  setValue(key: string, value: unknown, shouldRedact?: boolean, description?: string): void;\n  updateValue(key: string, value: unknown, description?: string): void;\n  deleteValue(key: string, description?: string): void;\n\n  // Non-tracking state inspection (for proxy internals — no recorder dispatch)\n  /** Returns all state keys without firing onRead. Used by ownKeys/has traps. */\n  getStateKeys?(): string[];\n  /** Check key existence without firing onRead. Used by has trap. */\n  hasKey?(key: string): boolean;\n\n  // Input & environment (readonly, NOT tracked)\n  getArgs<T = Record<string, unknown>>(): T;\n  getEnv(): Readonly<ExecutionEnv>;\n\n  // Recorder management\n  attachRecorder(recorder: Recorder): void;\n  detachRecorder(recorderId: string): void;\n  getRecorders(): Recorder[];\n\n  // Diagnostics\n  addDebugInfo(key: string, value: unknown): void;\n  addDebugMessage(value: unknown): void;\n  addErrorInfo(key: string, value: unknown): void;\n  addMetric(name: string, value: unknown): void;\n  addEval(name: string, value: unknown): void;\n}\n\n// -- ScopeMethods ------------------------------------------------------------\n// $-prefixed escape hatches. Non-enumerable on the proxy -- don't appear in\n// Object.keys(), destructuring, or for...in. Only state keys are visible.\n\nexport interface ScopeMethods {\n  // State (untyped escape hatch -- for dynamic keys, redaction, description)\n  $getValue(key: string): unknown;\n  $setValue(key: string, value: unknown, shouldRedact?: boolean, description?: string): void;\n  $update(key: string, value: unknown, description?: string): void;\n  $delete(key: string, description?: string): void;\n  /** Proxy-synthesized: calls getValue(rootKey) then lodash.get for nested path. Not a direct delegation. */\n  $read(dotPath: string): unknown;\n\n  // Input & environment (readonly)\n  $getArgs<T = Record<string, unknown>>(): T;\n  $getEnv(): Readonly<ExecutionEnv>;\n\n  // Observability\n  $debug(key: string, value: unknown): void;\n  $log(value: unknown): void;\n  $error(key: string, value: unknown): void;\n  $metric(name: string, value: unknown): void;\n  $eval(name: string, value: unknown): void;\n\n  // Recorder management\n  $attachRecorder(recorder: Recorder): void;\n  $detachRecorder(recorderId: string): void;\n  $getRecorders(): Recorder[];\n\n  /**\n   * Batch-mutate an array key in a single clone+write cycle.\n   *\n   * Every `scope.items.push(x)` clones the entire array and commits it — O(N) per call.\n   * For N mutations on an M-length array that is O(N×M). Use `$batchArray` to clone once,\n   * apply all mutations inside `fn`, then commit once — O(M) total.\n   *\n   * ```typescript\n   * // Before: 1000 clones × growing array = O(N²)\n   * for (let i = 0; i < 1000; i++) scope.items.push(i);\n   *\n   * // After: 1 clone + 1 commit = O(N)\n   * scope.$batchArray('items', (arr) => {\n   *   for (let i = 0; i < 1000; i++) arr.push(i);\n   * });\n   * ```\n   *\n   * `fn` receives a plain (non-proxy) mutable copy of the current array. Mutations\n   * inside `fn` are NOT tracked individually — only the final committed array appears\n   * in the narrative as a single write. If the key does not exist or is not an array,\n   * `fn` receives an empty array and the result is committed as the new value.\n   */\n  $batchArray(key: string, fn: (arr: unknown[]) => void): void;\n\n  // Pipeline control\n  $break(): void;\n\n  // Escape hatch -- unwrap to underlying ReactiveTarget\n  $toRaw(): ReactiveTarget;\n}\n\n// -- TypedScope<T> -----------------------------------------------------------\n// The consumer-facing type. T is the user's state interface.\n// Property access is typed; $-methods provide escape hatches.\n\nexport type TypedScope<T extends object = Record<string, unknown>> = T & ScopeMethods;\n\n// -- ReactiveOptions ---------------------------------------------------------\n// Configuration passed to createTypedScope.\n\nexport interface ReactiveOptions {\n  /** Pipeline break function -- injected by StageRunner after scope creation. */\n  breakPipeline?: () => void;\n}\n\n// -- Internal: $-method name set ---------------------------------------------\n// Used by the Proxy get trap to distinguish $-methods from state keys.\n\nexport const SCOPE_METHOD_NAMES = new Set<string>([\n  '$getValue',\n  '$setValue',\n  '$update',\n  '$delete',\n  '$read',\n  '$getArgs',\n  '$getEnv',\n  '$debug',\n  '$log',\n  '$error',\n  '$metric',\n  '$eval',\n  '$attachRecorder',\n  '$detachRecorder',\n  '$getRecorders',\n  '$batchArray',\n  '$break',\n  '$toRaw',\n]);\n\n// -- Internal: Symbol for deferred break injection ---------------------------\n// StageRunner sets this after scope creation so $break() works.\n// Private Symbol (not Symbol.for) to prevent cross-module tampering.\n\nexport const BREAK_SETTER = Symbol('footprint:reactive:setBreak');\n\n// -- Internal: Symbol for TypedScope detection -------------------------------\n// Used by StageRunner to skip createProtectedScope for TypedScope instances.\n// Private Symbol prevents string-tag spoofing.\n\nexport const IS_TYPED_SCOPE = Symbol('footprint:reactive:isTypedScope');\n\n// -- Internal: executor method allowlist -------------------------------------\n// ScopeFacade methods called by FlowChartExecutor wrapping code and StageRunner\n// THROUGH a TypedScope proxy. Only add methods with confirmed call sites.\n// Shared between createTypedScope.ts and StageRunner.ts to prevent drift.\n\nexport const EXECUTOR_INTERNAL_METHODS = new Set([\n  'notifyStageStart', // StageRunner.run() line 59\n  'notifyStageEnd', // StageRunner.run() line 79\n  'attachRecorder', // FlowChartExecutor.createTraverser() — narrative + user recorders\n  'detachRecorder', // FlowChartExecutor.detachRecorder()\n  'getRecorders', // FlowChartExecutor.getRecorders()\n  'useSharedRedactedKeys', // FlowChartExecutor.createTraverser() — redaction wrapping\n  'useRedactionPolicy', // FlowChartExecutor.createTraverser() — redaction wrapping\n]);\n"]}
|
|
56
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/lib/reactive/types.ts"],"names":[],"mappings":";AAAA;;;;;;;;GAQG;;;AA+HH,+EAA+E;AAC/E,uEAAuE;AAE1D,QAAA,kBAAkB,GAAG,IAAI,GAAG,CAAS;IAChD,WAAW;IACX,WAAW;IACX,SAAS;IACT,SAAS;IACT,OAAO;IACP,UAAU;IACV,SAAS;IACT,QAAQ;IACR,MAAM;IACN,QAAQ;IACR,SAAS;IACT,OAAO;IACP,iBAAiB;IACjB,iBAAiB;IACjB,eAAe;IACf,aAAa;IACb,QAAQ;IACR,QAAQ;CACT,CAAC,CAAC;AAEH,+EAA+E;AAC/E,gEAAgE;AAChE,qEAAqE;AAExD,QAAA,YAAY,GAAG,MAAM,CAAC,6BAA6B,CAAC,CAAC;AAElE,+EAA+E;AAC/E,6EAA6E;AAC7E,+CAA+C;AAElC,QAAA,cAAc,GAAG,MAAM,CAAC,iCAAiC,CAAC,CAAC;AAExE,+EAA+E;AAC/E,gFAAgF;AAChF,0EAA0E;AAC1E,0EAA0E;AAE7D,QAAA,yBAAyB,GAAG,IAAI,GAAG,CAAC;IAC/C,kBAAkB,EAAE,4BAA4B;IAChD,gBAAgB,EAAE,4BAA4B;IAC9C,gBAAgB,EAAE,mEAAmE;IACrF,gBAAgB,EAAE,qCAAqC;IACvD,cAAc,EAAE,mCAAmC;IACnD,uBAAuB,EAAE,2DAA2D;IACpF,oBAAoB,EAAE,2DAA2D;CAClF,CAAC,CAAC","sourcesContent":["/**\n * reactive/types -- Type definitions for the TypedScope<T> reactive proxy system.\n *\n * TypedScope<T> wraps a ReactiveTarget (ScopeFacade) in a Proxy that provides\n * typed property access. All scope infrastructure methods are $-prefixed to\n * avoid collisions with user state keys.\n *\n * Dependency: type-only imports from engine/ and scope/ (zero runtime cost).\n */\n\nimport type { ExecutionEnv } from '../engine/types.js';\nimport type { Recorder } from '../scope/types.js';\n\n// -- ReactiveTarget ----------------------------------------------------------\n// Minimum protocol required by TypedScope -- a curated subset of ScopeFacade's\n// public API. Not a full mirror: excludes internal/redaction methods that are\n// handled at the executor level (useRedactionPolicy, useSharedRedactedKeys, etc.)\n// and infrastructure methods (getPipelineId, setGlobal, getGlobal, etc.).\n\nexport interface ReactiveTarget {\n  // State access (tracked by recorders)\n  getValue(key?: string): unknown;\n  setValue(key: string, value: unknown, shouldRedact?: boolean, description?: string): void;\n  updateValue(key: string, value: unknown, description?: string): void;\n  deleteValue(key: string, description?: string): void;\n\n  // Non-tracking state inspection (for proxy internals — no recorder dispatch)\n  /** Returns all state keys without firing onRead. Used by ownKeys/has traps. */\n  getStateKeys?(): string[];\n  /** Check key existence without firing onRead. Used by has trap. */\n  hasKey?(key: string): boolean;\n\n  // Input & environment (readonly, NOT tracked)\n  getArgs<T = Record<string, unknown>>(): T;\n  getEnv(): Readonly<ExecutionEnv>;\n\n  // Recorder management\n  attachRecorder(recorder: Recorder): void;\n  detachRecorder(recorderId: string): void;\n  getRecorders(): Recorder[];\n\n  // Diagnostics\n  addDebugInfo(key: string, value: unknown): void;\n  addDebugMessage(value: unknown): void;\n  addErrorInfo(key: string, value: unknown): void;\n  addMetric(name: string, value: unknown): void;\n  addEval(name: string, value: unknown): void;\n}\n\n// -- ScopeMethods ------------------------------------------------------------\n// $-prefixed escape hatches. Non-enumerable on the proxy -- don't appear in\n// Object.keys(), destructuring, or for...in. Only state keys are visible.\n\nexport interface ScopeMethods {\n  // State (untyped escape hatch -- for dynamic keys, redaction, description)\n  $getValue(key: string): unknown;\n  $setValue(key: string, value: unknown, shouldRedact?: boolean, description?: string): void;\n  $update(key: string, value: unknown, description?: string): void;\n  $delete(key: string, description?: string): void;\n  /** Proxy-synthesized: calls getValue(rootKey) then lodash.get for nested path. Not a direct delegation. */\n  $read(dotPath: string): unknown;\n\n  // Input & environment (readonly)\n  $getArgs<T = Record<string, unknown>>(): T;\n  $getEnv(): Readonly<ExecutionEnv>;\n\n  // Observability\n  $debug(key: string, value: unknown): void;\n  $log(value: unknown): void;\n  $error(key: string, value: unknown): void;\n  $metric(name: string, value: unknown): void;\n  $eval(name: string, value: unknown): void;\n\n  // Recorder management\n  $attachRecorder(recorder: Recorder): void;\n  $detachRecorder(recorderId: string): void;\n  $getRecorders(): Recorder[];\n\n  /**\n   * Batch-mutate an array key in a single clone+write cycle.\n   *\n   * Every `scope.items.push(x)` clones the entire array and commits it — O(N) per call.\n   * For N mutations on an M-length array that is O(N×M). Use `$batchArray` to clone once,\n   * apply all mutations inside `fn`, then commit once — O(M) total.\n   *\n   * ```typescript\n   * // Before: 1000 clones × growing array = O(N²)\n   * for (let i = 0; i < 1000; i++) scope.items.push(i);\n   *\n   * // After: 1 clone + 1 commit = O(N)\n   * scope.$batchArray('items', (arr) => {\n   *   for (let i = 0; i < 1000; i++) arr.push(i);\n   * });\n   * ```\n   *\n   * `fn` receives a plain (non-proxy) mutable **shallow copy** of the current array.\n   * The array itself is a new instance, but object references inside it are shared with\n   * the original state — mutations to nested objects inside `fn` affect those originals.\n   * Only push/pop/sort/splice and other operations that change the array's own slots are\n   * safely isolated.\n   *\n   * Mutations inside `fn` are NOT tracked individually — only the final committed array\n   * appears in the narrative as a single write. If the key does not exist or is not an\n   * array, `fn` receives an empty array and the result is committed as the new value.\n   *\n   * If `fn` throws, `setValue` is never called and state remains unchanged (atomic on\n   * error). The exception propagates to the caller.\n   *\n   * `key` is untyped (`string`) — TypeScript will not catch typos. `arr` is typed as\n   * `unknown[]` because `ScopeMethods` is not parameterized by `T`; cast inside `fn`\n   * when element types are known: `(arr as string[]).push(x)`.\n   */\n  $batchArray(key: string, fn: (arr: unknown[]) => void): void;\n\n  // Pipeline control\n  $break(): void;\n\n  // Escape hatch -- unwrap to underlying ReactiveTarget\n  $toRaw(): ReactiveTarget;\n}\n\n// -- TypedScope<T> -----------------------------------------------------------\n// The consumer-facing type. T is the user's state interface.\n// Property access is typed; $-methods provide escape hatches.\n\nexport type TypedScope<T extends object = Record<string, unknown>> = T & ScopeMethods;\n\n// -- ReactiveOptions ---------------------------------------------------------\n// Configuration passed to createTypedScope.\n\nexport interface ReactiveOptions {\n  /** Pipeline break function -- injected by StageRunner after scope creation. */\n  breakPipeline?: () => void;\n}\n\n// -- Internal: $-method name set ---------------------------------------------\n// Used by the Proxy get trap to distinguish $-methods from state keys.\n\nexport const SCOPE_METHOD_NAMES = new Set<string>([\n  '$getValue',\n  '$setValue',\n  '$update',\n  '$delete',\n  '$read',\n  '$getArgs',\n  '$getEnv',\n  '$debug',\n  '$log',\n  '$error',\n  '$metric',\n  '$eval',\n  '$attachRecorder',\n  '$detachRecorder',\n  '$getRecorders',\n  '$batchArray',\n  '$break',\n  '$toRaw',\n]);\n\n// -- Internal: Symbol for deferred break injection ---------------------------\n// StageRunner sets this after scope creation so $break() works.\n// Private Symbol (not Symbol.for) to prevent cross-module tampering.\n\nexport const BREAK_SETTER = Symbol('footprint:reactive:setBreak');\n\n// -- Internal: Symbol for TypedScope detection -------------------------------\n// Used by StageRunner to skip createProtectedScope for TypedScope instances.\n// Private Symbol prevents string-tag spoofing.\n\nexport const IS_TYPED_SCOPE = Symbol('footprint:reactive:isTypedScope');\n\n// -- Internal: executor method allowlist -------------------------------------\n// ScopeFacade methods called by FlowChartExecutor wrapping code and StageRunner\n// THROUGH a TypedScope proxy. Only add methods with confirmed call sites.\n// Shared between createTypedScope.ts and StageRunner.ts to prevent drift.\n\nexport const EXECUTOR_INTERNAL_METHODS = new Set([\n  'notifyStageStart', // StageRunner.run() line 59\n  'notifyStageEnd', // StageRunner.run() line 79\n  'attachRecorder', // FlowChartExecutor.createTraverser() — narrative + user recorders\n  'detachRecorder', // FlowChartExecutor.detachRecorder()\n  'getRecorders', // FlowChartExecutor.getRecorders()\n  'useSharedRedactedKeys', // FlowChartExecutor.createTraverser() — redaction wrapping\n  'useRedactionPolicy', // FlowChartExecutor.createTraverser() — redaction wrapping\n]);\n"]}
|
|
@@ -63,10 +63,22 @@ export interface ScopeMethods {
|
|
|
63
63
|
* });
|
|
64
64
|
* ```
|
|
65
65
|
*
|
|
66
|
-
* `fn` receives a plain (non-proxy) mutable copy of the current array.
|
|
67
|
-
*
|
|
68
|
-
*
|
|
69
|
-
*
|
|
66
|
+
* `fn` receives a plain (non-proxy) mutable **shallow copy** of the current array.
|
|
67
|
+
* The array itself is a new instance, but object references inside it are shared with
|
|
68
|
+
* the original state — mutations to nested objects inside `fn` affect those originals.
|
|
69
|
+
* Only push/pop/sort/splice and other operations that change the array's own slots are
|
|
70
|
+
* safely isolated.
|
|
71
|
+
*
|
|
72
|
+
* Mutations inside `fn` are NOT tracked individually — only the final committed array
|
|
73
|
+
* appears in the narrative as a single write. If the key does not exist or is not an
|
|
74
|
+
* array, `fn` receives an empty array and the result is committed as the new value.
|
|
75
|
+
*
|
|
76
|
+
* If `fn` throws, `setValue` is never called and state remains unchanged (atomic on
|
|
77
|
+
* error). The exception propagates to the caller.
|
|
78
|
+
*
|
|
79
|
+
* `key` is untyped (`string`) — TypeScript will not catch typos. `arr` is typed as
|
|
80
|
+
* `unknown[]` because `ScopeMethods` is not parameterized by `T`; cast inside `fn`
|
|
81
|
+
* when element types are known: `(arr as string[]).push(x)`.
|
|
70
82
|
*/
|
|
71
83
|
$batchArray(key: string, fn: (arr: unknown[]) => void): void;
|
|
72
84
|
$break(): void;
|
package/package.json
CHANGED