footprintjs 4.0.2 → 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
@@ -59,6 +59,16 @@ const METHOD_ROUTES = {
59
59
  $attachRecorder: (t) => t.attachRecorder.bind(t),
60
60
  $detachRecorder: (t) => t.detachRecorder.bind(t),
61
61
  $getRecorders: (t) => t.getRecorders.bind(t),
62
+ $batchArray: (t) => (key, fn) => {
63
+ // One getValue — fires onRead once
64
+ const current = t.getValue(key);
65
+ // Clone once (or start empty if missing/non-array)
66
+ const clone = Array.isArray(current) ? [...current] : [];
67
+ // User applies all mutations to the plain clone — no Proxy, no per-mutation commit
68
+ fn(clone);
69
+ // One setValue — fires onWrite once with the final array
70
+ t.setValue(key, clone);
71
+ },
62
72
  $break: (_t, opts) => () => {
63
73
  if (!opts.breakFn)
64
74
  throw new Error('$break() is not available outside stage execution');
@@ -325,4 +335,4 @@ export function createTypedScope(target, options) {
325
335
  });
326
336
  return proxy;
327
337
  }
328
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createTypedScope.js","sourceRoot":"","sources":["../../../../src/lib/reactive/createTypedScope.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * reactive/createTypedScope -- Core Proxy factory for TypedScope<T>.\n *\n * Wraps a ReactiveTarget (ScopeFacade) in a Proxy that provides:\n * - Typed property access: scope.creditTier (read), scope.creditTier = 'A' (write)\n * - Deep write interception: scope.customer.address.zip = '90210'\n * - Array mutation interception: scope.items.push('new')\n * - $-prefixed escape hatches: $getValue, $setValue, $read, $getArgs, etc.\n *\n * Read semantics: top-level get calls getValue() (fires onRead ONCE).\n *   Nested get traps navigate in-memory -- no additional onRead.\n *\n * Write semantics: top-level set calls setValue(). Nested set calls\n *   updateValue() with a partial object built from the accumulated path.\n */\n\nimport { nativeGet as lodashGet } from '../memory/pathOps.js';\nimport { shouldWrapWithProxy } from './allowlist.js';\nimport { createArrayProxy } from './arrayTraps.js';\nimport { buildNestedPatch } from './pathBuilder.js';\nimport type { ReactiveOptions, ReactiveTarget, TypedScope } from './types.js';\nimport { BREAK_SETTER, EXECUTOR_INTERNAL_METHODS, IS_TYPED_SCOPE, SCOPE_METHOD_NAMES } from './types.js';\n\n// -- Proxy unwrapping --------------------------------------------------------\n// structuredClone in TransactionBuffer cannot clone Proxy objects.\n// When a user does `scope.backup = scope.customer`, the value is a Proxy.\n// Unwrap to a plain object before storing.\n\nfunction unwrapProxy(value: unknown): unknown {\n  if (value === null || value === undefined) return value;\n  if (typeof value !== 'object') return value;\n  // Fast path: plain objects and arrays don't need unwrapping\n  try {\n    // JSON round-trip strips Proxies. Safe because state values must be JSON-serializable.\n    return JSON.parse(JSON.stringify(value));\n  } catch {\n    // Non-serializable (functions, symbols, etc.) — return as-is\n    return value;\n  }\n}\n\n// -- $-method routing --------------------------------------------------------\n\ntype MethodRouter = (target: ReactiveTarget, opts: ReactiveState) => unknown;\n\nconst METHOD_ROUTES: Record<string, MethodRouter> = {\n  $getValue: (t) => t.getValue.bind(t),\n  $setValue: (t) => t.setValue.bind(t),\n  $update: (t) => t.updateValue.bind(t),\n  $delete: (t) => t.deleteValue.bind(t),\n  $read: (t) => (dotPath: string) => {\n    const rootKey = dotPath.split('.')[0];\n    const value = t.getValue(rootKey);\n    if (!dotPath.includes('.')) return value;\n    return lodashGet(value, dotPath.slice(rootKey.length + 1));\n  },\n  $getArgs: (t) => t.getArgs.bind(t),\n  $getEnv: (t) => t.getEnv.bind(t),\n  $debug: (t) => t.addDebugInfo.bind(t),\n  $log: (t) => t.addDebugMessage.bind(t),\n  $error: (t) => t.addErrorInfo.bind(t),\n  $metric: (t) => t.addMetric.bind(t),\n  $eval: (t) => t.addEval.bind(t),\n  $attachRecorder: (t) => t.attachRecorder.bind(t),\n  $detachRecorder: (t) => t.detachRecorder.bind(t),\n  $getRecorders: (t) => t.getRecorders.bind(t),\n  $break: (_t, opts) => () => {\n    if (!opts.breakFn) throw new Error('$break() is not available outside stage execution');\n    opts.breakFn();\n  },\n  $toRaw: (t) => () => t,\n};\n\n// -- Guard properties --------------------------------------------------------\n// These must be handled to prevent Proxy from being treated as a Promise,\n// breaking instanceof checks, or confusing test matchers.\n\nconst GUARD_PROPS: Record<string | symbol, unknown> = {\n  then: undefined, // prevent Promise detection\n  asymmetricMatch: undefined, // prevent vitest/jest matcher confusion\n  constructor: Object, // safe prototype\n  [Symbol.toStringTag]: 'TypedScope',\n};\n\n// -- Mutable state per proxy instance ----------------------------------------\n\ninterface ReactiveState {\n  breakFn?: () => void;\n  /** Cache: top-level key -> { raw object ref, child proxy } */\n  childCache: Map<string, { ref: object; proxy: object }>;\n}\n\n// -- Nested child proxy (for deep write interception) ------------------------\n//\n// Cycle safety: an immutable Set<object> of ancestor objects is passed down\n// each access chain. Each branch gets its own copy (new Set(parent)) so\n// scope.x.friend and scope.x.coworker don't pollute each other's tracking.\n// When a child value is already in the ancestor set, we've hit a cycle.\n// At the cycle break: return a terminal proxy that tracks writes (set trap\n// still builds path + calls updateValue) but doesn't recurse reads further.\n\nfunction createTerminalProxy(\n  obj: Record<string, unknown>,\n  rootKey: string,\n  segments: string[],\n  target: ReactiveTarget,\n  state: ReactiveState,\n  visited: Set<object> = new Set(),\n): unknown {\n  visited.add(obj);\n\n  return new Proxy(obj, {\n    get(raw, prop) {\n      if (typeof prop === 'symbol') return (raw as any)[prop];\n      if (prop === 'then') return undefined;\n      if (prop === 'asymmetricMatch') return undefined;\n      if (prop === 'constructor') return Object;\n      if (prop === 'toJSON')\n        return () => {\n          // Strip object-typed values to prevent circular JSON errors\n          const safe: Record<string, unknown> = {};\n          for (const k of Object.keys(raw)) {\n            const v = (raw as any)[k];\n            if (v === null || typeof v !== 'object') safe[k] = v;\n          }\n          return safe;\n        };\n\n      const value = (raw as any)[prop];\n\n      // Continue tracking writes at deeper levels via chained terminal proxies.\n      // Use visited set to prevent re-entering the same object (cycle in terminal chain).\n      if (shouldWrapWithProxy(value) && !Array.isArray(value) && !visited.has(value as object)) {\n        return createTerminalProxy(\n          value as Record<string, unknown>,\n          rootKey,\n          [...segments, prop as string],\n          target,\n          state,\n          visited,\n        );\n      }\n\n      return value;\n    },\n    set(raw, prop, value) {\n      if (typeof prop !== 'string') return true;\n      const childSegments = [...segments, prop];\n      const patch = buildNestedPatch(childSegments, unwrapProxy(value));\n      target.updateValue(rootKey, patch);\n      state.childCache.delete(rootKey);\n      return true;\n    },\n  });\n}\n\nfunction createNestedProxy(\n  obj: Record<string, unknown>,\n  rootKey: string,\n  segments: string[],\n  target: ReactiveTarget,\n  state: ReactiveState,\n  ancestors: Set<object> = new Set(),\n): unknown {\n  return new Proxy(obj, {\n    get(raw, prop) {\n      if (typeof prop === 'symbol') return (raw as any)[prop];\n\n      // Guard properties\n      if (prop === 'then') return undefined;\n      if (prop === 'asymmetricMatch') return undefined;\n      if (prop === 'constructor') return Object;\n      if (prop === 'toJSON')\n        return () => {\n          // Strip object-typed values to prevent circular JSON errors\n          const safe: Record<string, unknown> = {};\n          for (const k of Object.keys(raw)) {\n            const v = (raw as any)[k];\n            if (v === null || typeof v !== 'object') safe[k] = v;\n          }\n          return safe;\n        };\n\n      const value = (raw as any)[prop];\n\n      // Primitive or non-wrappable -- return as-is (no deeper proxy)\n      if (!shouldWrapWithProxy(value)) return value;\n\n      const childSegments = [...segments, prop as string];\n\n      // Array -- return array proxy\n      if (Array.isArray(value)) {\n        return createArrayProxy(\n          () => {\n            const current = target.getValue(rootKey) as any;\n            return lodashGet(current, childSegments.join('.')) ?? [];\n          },\n          (newArr) => {\n            const patch = buildNestedPatch(childSegments, unwrapProxy(newArr));\n            target.updateValue(rootKey, patch);\n            state.childCache.delete(rootKey);\n          },\n        );\n      }\n\n      // Cycle detection: if this value is an ancestor in the current access\n      // chain, return a terminal proxy (tracks writes, stops recursing reads).\n      if (ancestors.has(value as object)) {\n        return createTerminalProxy(value as Record<string, unknown>, rootKey, childSegments, target, state);\n      }\n\n      // Build new ancestor set for this branch (immutable -- no cross-branch pollution)\n      const childAncestors = new Set(ancestors);\n      childAncestors.add(value as object);\n\n      return createNestedProxy(value as Record<string, unknown>, rootKey, childSegments, target, state, childAncestors);\n    },\n\n    set(raw, prop, value) {\n      if (typeof prop !== 'string') return true;\n\n      const childSegments = [...segments, prop];\n      const patch = buildNestedPatch(childSegments, unwrapProxy(value));\n      target.updateValue(rootKey, patch);\n      state.childCache.delete(rootKey);\n      return true;\n    },\n  });\n}\n\n// -- Top-level proxy (the main TypedScope) -----------------------------------\n\n/**\n * Creates a TypedScope<T> proxy wrapping a ReactiveTarget.\n *\n * @param target - The underlying scope (ScopeFacade or any ReactiveTarget)\n * @param options - Optional configuration (breakPipeline injection)\n * @returns A Proxy with typed property access and $-prefixed methods\n */\nexport function createTypedScope<T extends object>(target: ReactiveTarget, options?: ReactiveOptions): TypedScope<T> {\n  const state: ReactiveState = {\n    breakFn: options?.breakPipeline,\n    childCache: new Map(),\n  };\n\n  const proxy = new Proxy(target as unknown as TypedScope<T>, {\n    get(_proxyTarget, prop, _receiver) {\n      // 1. Internal symbols (check before other symbols)\n      if (prop === IS_TYPED_SCOPE) return true;\n      if (prop === BREAK_SETTER) {\n        return (fn: () => void) => {\n          state.breakFn = fn;\n        };\n      }\n\n      // 2. Symbol properties (guard + inspection)\n      if (typeof prop === 'symbol') {\n        if (Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop)) return GUARD_PROPS[prop];\n        // Node.js util.inspect — show state snapshot, not proxy internals\n        if (prop === Symbol.for('nodejs.util.inspect.custom')) {\n          return () => target.getValue();\n        }\n        return undefined;\n      }\n\n      // 3. String guard properties\n      if (Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop)) return GUARD_PROPS[prop];\n\n      // 4. $-prefixed methods -- route to facade\n      if (SCOPE_METHOD_NAMES.has(prop)) {\n        const router = METHOD_ROUTES[prop];\n        if (router) return router(target, state);\n        return undefined;\n      }\n\n      // 5. Executor-internal method pass-through (explicit allowlist)\n      //    FlowChartExecutor wrapping calls attachRecorder, notifyStageStart, etc.\n      //    directly on the scope. Forward only allowlisted methods.\n      if (EXECUTOR_INTERNAL_METHODS.has(prop) && typeof (target as any)[prop] === 'function') {\n        return (target as any)[prop].bind(target);\n      }\n\n      // 6. State key -- call getValue (fires onRead ONCE)\n      const value = target.getValue(prop);\n\n      // Primitive or null/undefined -- return as-is\n      if (value === null || value === undefined || typeof value !== 'object') {\n        return value;\n      }\n\n      // Non-wrappable (Date, Map, class instance, etc.) -- return unwrapped\n      if (!shouldWrapWithProxy(value)) return value;\n\n      // Array -- return array proxy (cached for identity equality)\n      if (Array.isArray(value)) {\n        const cached = state.childCache.get(prop);\n        if (cached && cached.ref === value) return cached.proxy;\n\n        const arrProxy = createArrayProxy(\n          () => (target.getValue(prop) as unknown[]) ?? [],\n          (newArr) => {\n            target.setValue(prop, unwrapProxy(newArr));\n            state.childCache.delete(prop);\n          },\n        );\n        state.childCache.set(prop, { ref: value as object, proxy: arrProxy as unknown as object });\n        return arrProxy;\n      }\n\n      // Plain object -- return nested proxy (cached for identity equality)\n      const cached = state.childCache.get(prop);\n      if (cached && cached.ref === value) return cached.proxy;\n\n      const nested = createNestedProxy(\n        value as Record<string, unknown>,\n        prop,\n        [],\n        target,\n        state,\n        new Set<object>([value as object]), // seed ancestor set with root object\n      );\n      state.childCache.set(prop, { ref: value as object, proxy: nested as object });\n      return nested;\n    },\n\n    set(_proxyTarget, prop, value) {\n      if (typeof prop !== 'string') return true;\n      if (SCOPE_METHOD_NAMES.has(prop)) {\n        throw new Error(\n          `Cannot set state key \"${prop}\" -- it conflicts with a reserved TypedScope method. Rename the state key to avoid $-prefixed names.`,\n        );\n      }\n      // Unwrap Proxy values before storing — structuredClone in TransactionBuffer\n      // cannot clone Proxy objects. This handles: scope.backup = scope.customer\n      const unwrapped = unwrapProxy(value);\n      target.setValue(prop, unwrapped);\n      state.childCache.delete(prop); // invalidate cache\n      return true;\n    },\n\n    deleteProperty(_proxyTarget, prop) {\n      if (typeof prop !== 'string') return true;\n      target.deleteValue(prop);\n      state.childCache.delete(prop);\n      return true;\n    },\n\n    has(_proxyTarget, prop) {\n      if (typeof prop === 'symbol') return Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop);\n      if (SCOPE_METHOD_NAMES.has(prop)) return true;\n      // Use non-tracking hasKey if available, else fallback to getStateKeys\n      if (target.hasKey) return target.hasKey(prop);\n      if (target.getStateKeys) return target.getStateKeys().includes(prop);\n      // Fallback: getValue fires onRead (acceptable degradation)\n      return target.getValue(prop) !== undefined;\n    },\n\n    ownKeys() {\n      // Use non-tracking getStateKeys if available, else fallback\n      if (target.getStateKeys) return target.getStateKeys();\n      const snapshot = target.getValue() as Record<string, unknown> | undefined;\n      if (!snapshot || typeof snapshot !== 'object') return [];\n      return Object.keys(snapshot);\n    },\n\n    getOwnPropertyDescriptor(_proxyTarget, prop) {\n      if (typeof prop !== 'string') return undefined;\n      if (SCOPE_METHOD_NAMES.has(prop)) return undefined; // $-methods are non-enumerable\n      // Check existence without firing onRead — no getValue call here\n      const exists = target.hasKey\n        ? target.hasKey(prop)\n        : target.getStateKeys\n        ? target.getStateKeys().includes(prop)\n        : target.getValue(prop) !== undefined; // fallback only\n      if (!exists) return undefined;\n      // Return a minimal descriptor — actual value is fetched via the get trap\n      return { configurable: true, enumerable: true, writable: true };\n    },\n  });\n\n  return proxy;\n}\n"]}
338
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createTypedScope.js","sourceRoot":"","sources":["../../../../src/lib/reactive/createTypedScope.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * reactive/createTypedScope -- Core Proxy factory for TypedScope<T>.\n *\n * Wraps a ReactiveTarget (ScopeFacade) in a Proxy that provides:\n * - Typed property access: scope.creditTier (read), scope.creditTier = 'A' (write)\n * - Deep write interception: scope.customer.address.zip = '90210'\n * - Array mutation interception: scope.items.push('new')\n * - $-prefixed escape hatches: $getValue, $setValue, $read, $getArgs, etc.\n *\n * Read semantics: top-level get calls getValue() (fires onRead ONCE).\n *   Nested get traps navigate in-memory -- no additional onRead.\n *\n * Write semantics: top-level set calls setValue(). Nested set calls\n *   updateValue() with a partial object built from the accumulated path.\n */\n\nimport { nativeGet as lodashGet } from '../memory/pathOps.js';\nimport { shouldWrapWithProxy } from './allowlist.js';\nimport { createArrayProxy } from './arrayTraps.js';\nimport { buildNestedPatch } from './pathBuilder.js';\nimport type { ReactiveOptions, ReactiveTarget, TypedScope } from './types.js';\nimport { BREAK_SETTER, EXECUTOR_INTERNAL_METHODS, IS_TYPED_SCOPE, SCOPE_METHOD_NAMES } from './types.js';\n\n// -- Proxy unwrapping --------------------------------------------------------\n// structuredClone in TransactionBuffer cannot clone Proxy objects.\n// When a user does `scope.backup = scope.customer`, the value is a Proxy.\n// Unwrap to a plain object before storing.\n\nfunction unwrapProxy(value: unknown): unknown {\n  if (value === null || value === undefined) return value;\n  if (typeof value !== 'object') return value;\n  // Fast path: plain objects and arrays don't need unwrapping\n  try {\n    // JSON round-trip strips Proxies. Safe because state values must be JSON-serializable.\n    return JSON.parse(JSON.stringify(value));\n  } catch {\n    // Non-serializable (functions, symbols, etc.) — return as-is\n    return value;\n  }\n}\n\n// -- $-method routing --------------------------------------------------------\n\ntype MethodRouter = (target: ReactiveTarget, opts: ReactiveState) => unknown;\n\nconst METHOD_ROUTES: Record<string, MethodRouter> = {\n  $getValue: (t) => t.getValue.bind(t),\n  $setValue: (t) => t.setValue.bind(t),\n  $update: (t) => t.updateValue.bind(t),\n  $delete: (t) => t.deleteValue.bind(t),\n  $read: (t) => (dotPath: string) => {\n    const rootKey = dotPath.split('.')[0];\n    const value = t.getValue(rootKey);\n    if (!dotPath.includes('.')) return value;\n    return lodashGet(value, dotPath.slice(rootKey.length + 1));\n  },\n  $getArgs: (t) => t.getArgs.bind(t),\n  $getEnv: (t) => t.getEnv.bind(t),\n  $debug: (t) => t.addDebugInfo.bind(t),\n  $log: (t) => t.addDebugMessage.bind(t),\n  $error: (t) => t.addErrorInfo.bind(t),\n  $metric: (t) => t.addMetric.bind(t),\n  $eval: (t) => t.addEval.bind(t),\n  $attachRecorder: (t) => t.attachRecorder.bind(t),\n  $detachRecorder: (t) => t.detachRecorder.bind(t),\n  $getRecorders: (t) => t.getRecorders.bind(t),\n  $batchArray: (t) => (key: string, fn: (arr: unknown[]) => void) => {\n    // One getValue — fires onRead once\n    const current = t.getValue(key);\n    // Clone once (or start empty if missing/non-array)\n    const clone: unknown[] = Array.isArray(current) ? [...current] : [];\n    // User applies all mutations to the plain clone — no Proxy, no per-mutation commit\n    fn(clone);\n    // One setValue — fires onWrite once with the final array\n    t.setValue(key, clone);\n  },\n  $break: (_t, opts) => () => {\n    if (!opts.breakFn) throw new Error('$break() is not available outside stage execution');\n    opts.breakFn();\n  },\n  $toRaw: (t) => () => t,\n};\n\n// -- Guard properties --------------------------------------------------------\n// These must be handled to prevent Proxy from being treated as a Promise,\n// breaking instanceof checks, or confusing test matchers.\n\nconst GUARD_PROPS: Record<string | symbol, unknown> = {\n  then: undefined, // prevent Promise detection\n  asymmetricMatch: undefined, // prevent vitest/jest matcher confusion\n  constructor: Object, // safe prototype\n  [Symbol.toStringTag]: 'TypedScope',\n};\n\n// -- Mutable state per proxy instance ----------------------------------------\n\ninterface ReactiveState {\n  breakFn?: () => void;\n  /** Cache: top-level key -> { raw object ref, child proxy } */\n  childCache: Map<string, { ref: object; proxy: object }>;\n}\n\n// -- Nested child proxy (for deep write interception) ------------------------\n//\n// Cycle safety: an immutable Set<object> of ancestor objects is passed down\n// each access chain. Each branch gets its own copy (new Set(parent)) so\n// scope.x.friend and scope.x.coworker don't pollute each other's tracking.\n// When a child value is already in the ancestor set, we've hit a cycle.\n// At the cycle break: return a terminal proxy that tracks writes (set trap\n// still builds path + calls updateValue) but doesn't recurse reads further.\n\nfunction createTerminalProxy(\n  obj: Record<string, unknown>,\n  rootKey: string,\n  segments: string[],\n  target: ReactiveTarget,\n  state: ReactiveState,\n  visited: Set<object> = new Set(),\n): unknown {\n  visited.add(obj);\n\n  return new Proxy(obj, {\n    get(raw, prop) {\n      if (typeof prop === 'symbol') return (raw as any)[prop];\n      if (prop === 'then') return undefined;\n      if (prop === 'asymmetricMatch') return undefined;\n      if (prop === 'constructor') return Object;\n      if (prop === 'toJSON')\n        return () => {\n          // Strip object-typed values to prevent circular JSON errors\n          const safe: Record<string, unknown> = {};\n          for (const k of Object.keys(raw)) {\n            const v = (raw as any)[k];\n            if (v === null || typeof v !== 'object') safe[k] = v;\n          }\n          return safe;\n        };\n\n      const value = (raw as any)[prop];\n\n      // Continue tracking writes at deeper levels via chained terminal proxies.\n      // Use visited set to prevent re-entering the same object (cycle in terminal chain).\n      if (shouldWrapWithProxy(value) && !Array.isArray(value) && !visited.has(value as object)) {\n        return createTerminalProxy(\n          value as Record<string, unknown>,\n          rootKey,\n          [...segments, prop as string],\n          target,\n          state,\n          visited,\n        );\n      }\n\n      return value;\n    },\n    set(raw, prop, value) {\n      if (typeof prop !== 'string') return true;\n      const childSegments = [...segments, prop];\n      const patch = buildNestedPatch(childSegments, unwrapProxy(value));\n      target.updateValue(rootKey, patch);\n      state.childCache.delete(rootKey);\n      return true;\n    },\n  });\n}\n\nfunction createNestedProxy(\n  obj: Record<string, unknown>,\n  rootKey: string,\n  segments: string[],\n  target: ReactiveTarget,\n  state: ReactiveState,\n  ancestors: Set<object> = new Set(),\n): unknown {\n  return new Proxy(obj, {\n    get(raw, prop) {\n      if (typeof prop === 'symbol') return (raw as any)[prop];\n\n      // Guard properties\n      if (prop === 'then') return undefined;\n      if (prop === 'asymmetricMatch') return undefined;\n      if (prop === 'constructor') return Object;\n      if (prop === 'toJSON')\n        return () => {\n          // Strip object-typed values to prevent circular JSON errors\n          const safe: Record<string, unknown> = {};\n          for (const k of Object.keys(raw)) {\n            const v = (raw as any)[k];\n            if (v === null || typeof v !== 'object') safe[k] = v;\n          }\n          return safe;\n        };\n\n      const value = (raw as any)[prop];\n\n      // Primitive or non-wrappable -- return as-is (no deeper proxy)\n      if (!shouldWrapWithProxy(value)) return value;\n\n      const childSegments = [...segments, prop as string];\n\n      // Array -- return array proxy\n      if (Array.isArray(value)) {\n        return createArrayProxy(\n          () => {\n            const current = target.getValue(rootKey) as any;\n            return lodashGet(current, childSegments.join('.')) ?? [];\n          },\n          (newArr) => {\n            const patch = buildNestedPatch(childSegments, unwrapProxy(newArr));\n            target.updateValue(rootKey, patch);\n            state.childCache.delete(rootKey);\n          },\n        );\n      }\n\n      // Cycle detection: if this value is an ancestor in the current access\n      // chain, return a terminal proxy (tracks writes, stops recursing reads).\n      if (ancestors.has(value as object)) {\n        return createTerminalProxy(value as Record<string, unknown>, rootKey, childSegments, target, state);\n      }\n\n      // Build new ancestor set for this branch (immutable -- no cross-branch pollution)\n      const childAncestors = new Set(ancestors);\n      childAncestors.add(value as object);\n\n      return createNestedProxy(value as Record<string, unknown>, rootKey, childSegments, target, state, childAncestors);\n    },\n\n    set(raw, prop, value) {\n      if (typeof prop !== 'string') return true;\n\n      const childSegments = [...segments, prop];\n      const patch = buildNestedPatch(childSegments, unwrapProxy(value));\n      target.updateValue(rootKey, patch);\n      state.childCache.delete(rootKey);\n      return true;\n    },\n  });\n}\n\n// -- Top-level proxy (the main TypedScope) -----------------------------------\n\n/**\n * Creates a TypedScope<T> proxy wrapping a ReactiveTarget.\n *\n * @param target - The underlying scope (ScopeFacade or any ReactiveTarget)\n * @param options - Optional configuration (breakPipeline injection)\n * @returns A Proxy with typed property access and $-prefixed methods\n */\nexport function createTypedScope<T extends object>(target: ReactiveTarget, options?: ReactiveOptions): TypedScope<T> {\n  const state: ReactiveState = {\n    breakFn: options?.breakPipeline,\n    childCache: new Map(),\n  };\n\n  const proxy = new Proxy(target as unknown as TypedScope<T>, {\n    get(_proxyTarget, prop, _receiver) {\n      // 1. Internal symbols (check before other symbols)\n      if (prop === IS_TYPED_SCOPE) return true;\n      if (prop === BREAK_SETTER) {\n        return (fn: () => void) => {\n          state.breakFn = fn;\n        };\n      }\n\n      // 2. Symbol properties (guard + inspection)\n      if (typeof prop === 'symbol') {\n        if (Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop)) return GUARD_PROPS[prop];\n        // Node.js util.inspect — show state snapshot, not proxy internals\n        if (prop === Symbol.for('nodejs.util.inspect.custom')) {\n          return () => target.getValue();\n        }\n        return undefined;\n      }\n\n      // 3. String guard properties\n      if (Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop)) return GUARD_PROPS[prop];\n\n      // 4. $-prefixed methods -- route to facade\n      if (SCOPE_METHOD_NAMES.has(prop)) {\n        const router = METHOD_ROUTES[prop];\n        if (router) return router(target, state);\n        return undefined;\n      }\n\n      // 5. Executor-internal method pass-through (explicit allowlist)\n      //    FlowChartExecutor wrapping calls attachRecorder, notifyStageStart, etc.\n      //    directly on the scope. Forward only allowlisted methods.\n      if (EXECUTOR_INTERNAL_METHODS.has(prop) && typeof (target as any)[prop] === 'function') {\n        return (target as any)[prop].bind(target);\n      }\n\n      // 6. State key -- call getValue (fires onRead ONCE)\n      const value = target.getValue(prop);\n\n      // Primitive or null/undefined -- return as-is\n      if (value === null || value === undefined || typeof value !== 'object') {\n        return value;\n      }\n\n      // Non-wrappable (Date, Map, class instance, etc.) -- return unwrapped\n      if (!shouldWrapWithProxy(value)) return value;\n\n      // Array -- return array proxy (cached for identity equality)\n      if (Array.isArray(value)) {\n        const cached = state.childCache.get(prop);\n        if (cached && cached.ref === value) return cached.proxy;\n\n        const arrProxy = createArrayProxy(\n          () => (target.getValue(prop) as unknown[]) ?? [],\n          (newArr) => {\n            target.setValue(prop, unwrapProxy(newArr));\n            state.childCache.delete(prop);\n          },\n        );\n        state.childCache.set(prop, { ref: value as object, proxy: arrProxy as unknown as object });\n        return arrProxy;\n      }\n\n      // Plain object -- return nested proxy (cached for identity equality)\n      const cached = state.childCache.get(prop);\n      if (cached && cached.ref === value) return cached.proxy;\n\n      const nested = createNestedProxy(\n        value as Record<string, unknown>,\n        prop,\n        [],\n        target,\n        state,\n        new Set<object>([value as object]), // seed ancestor set with root object\n      );\n      state.childCache.set(prop, { ref: value as object, proxy: nested as object });\n      return nested;\n    },\n\n    set(_proxyTarget, prop, value) {\n      if (typeof prop !== 'string') return true;\n      if (SCOPE_METHOD_NAMES.has(prop)) {\n        throw new Error(\n          `Cannot set state key \"${prop}\" -- it conflicts with a reserved TypedScope method. Rename the state key to avoid $-prefixed names.`,\n        );\n      }\n      // Unwrap Proxy values before storing — structuredClone in TransactionBuffer\n      // cannot clone Proxy objects. This handles: scope.backup = scope.customer\n      const unwrapped = unwrapProxy(value);\n      target.setValue(prop, unwrapped);\n      state.childCache.delete(prop); // invalidate cache\n      return true;\n    },\n\n    deleteProperty(_proxyTarget, prop) {\n      if (typeof prop !== 'string') return true;\n      target.deleteValue(prop);\n      state.childCache.delete(prop);\n      return true;\n    },\n\n    has(_proxyTarget, prop) {\n      if (typeof prop === 'symbol') return Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop);\n      if (SCOPE_METHOD_NAMES.has(prop)) return true;\n      // Use non-tracking hasKey if available, else fallback to getStateKeys\n      if (target.hasKey) return target.hasKey(prop);\n      if (target.getStateKeys) return target.getStateKeys().includes(prop);\n      // Fallback: getValue fires onRead (acceptable degradation)\n      return target.getValue(prop) !== undefined;\n    },\n\n    ownKeys() {\n      // Use non-tracking getStateKeys if available, else fallback\n      if (target.getStateKeys) return target.getStateKeys();\n      const snapshot = target.getValue() as Record<string, unknown> | undefined;\n      if (!snapshot || typeof snapshot !== 'object') return [];\n      return Object.keys(snapshot);\n    },\n\n    getOwnPropertyDescriptor(_proxyTarget, prop) {\n      if (typeof prop !== 'string') return undefined;\n      if (SCOPE_METHOD_NAMES.has(prop)) return undefined; // $-methods are non-enumerable\n      // Check existence without firing onRead — no getValue call here\n      const exists = target.hasKey\n        ? target.hasKey(prop)\n        : target.getStateKeys\n        ? target.getStateKeys().includes(prop)\n        : target.getValue(prop) !== undefined; // fallback only\n      if (!exists) return undefined;\n      // Return a minimal descriptor — actual value is fetched via the get trap\n      return { configurable: true, enumerable: true, writable: true };\n    },\n  });\n\n  return proxy;\n}\n"]}
@@ -25,6 +25,7 @@ export const SCOPE_METHOD_NAMES = new Set([
25
25
  '$attachRecorder',
26
26
  '$detachRecorder',
27
27
  '$getRecorders',
28
+ '$batchArray',
28
29
  '$break',
29
30
  '$toRaw',
30
31
  ]);
@@ -49,4 +50,4 @@ export const EXECUTOR_INTERNAL_METHODS = new Set([
49
50
  'useSharedRedactedKeys', // FlowChartExecutor.createTraverser() — redaction wrapping
50
51
  'useRedactionPolicy', // FlowChartExecutor.createTraverser() — redaction wrapping
51
52
  ]);
52
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../src/lib/reactive/types.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AA2FH,+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,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  // 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  '$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"]}
@@ -62,6 +62,16 @@ const METHOD_ROUTES = {
62
62
  $attachRecorder: (t) => t.attachRecorder.bind(t),
63
63
  $detachRecorder: (t) => t.detachRecorder.bind(t),
64
64
  $getRecorders: (t) => t.getRecorders.bind(t),
65
+ $batchArray: (t) => (key, fn) => {
66
+ // One getValue — fires onRead once
67
+ const current = t.getValue(key);
68
+ // Clone once (or start empty if missing/non-array)
69
+ const clone = Array.isArray(current) ? [...current] : [];
70
+ // User applies all mutations to the plain clone — no Proxy, no per-mutation commit
71
+ fn(clone);
72
+ // One setValue — fires onWrite once with the final array
73
+ t.setValue(key, clone);
74
+ },
65
75
  $break: (_t, opts) => () => {
66
76
  if (!opts.breakFn)
67
77
  throw new Error('$break() is not available outside stage execution');
@@ -329,4 +339,4 @@ function createTypedScope(target, options) {
329
339
  return proxy;
330
340
  }
331
341
  exports.createTypedScope = createTypedScope;
332
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createTypedScope.js","sourceRoot":"","sources":["../../../src/lib/reactive/createTypedScope.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * reactive/createTypedScope -- Core Proxy factory for TypedScope<T>.\n *\n * Wraps a ReactiveTarget (ScopeFacade) in a Proxy that provides:\n * - Typed property access: scope.creditTier (read), scope.creditTier = 'A' (write)\n * - Deep write interception: scope.customer.address.zip = '90210'\n * - Array mutation interception: scope.items.push('new')\n * - $-prefixed escape hatches: $getValue, $setValue, $read, $getArgs, etc.\n *\n * Read semantics: top-level get calls getValue() (fires onRead ONCE).\n *   Nested get traps navigate in-memory -- no additional onRead.\n *\n * Write semantics: top-level set calls setValue(). Nested set calls\n *   updateValue() with a partial object built from the accumulated path.\n */\n\nimport { nativeGet as lodashGet } from '../memory/pathOps.js';\nimport { shouldWrapWithProxy } from './allowlist.js';\nimport { createArrayProxy } from './arrayTraps.js';\nimport { buildNestedPatch } from './pathBuilder.js';\nimport type { ReactiveOptions, ReactiveTarget, TypedScope } from './types.js';\nimport { BREAK_SETTER, EXECUTOR_INTERNAL_METHODS, IS_TYPED_SCOPE, SCOPE_METHOD_NAMES } from './types.js';\n\n// -- Proxy unwrapping --------------------------------------------------------\n// structuredClone in TransactionBuffer cannot clone Proxy objects.\n// When a user does `scope.backup = scope.customer`, the value is a Proxy.\n// Unwrap to a plain object before storing.\n\nfunction unwrapProxy(value: unknown): unknown {\n  if (value === null || value === undefined) return value;\n  if (typeof value !== 'object') return value;\n  // Fast path: plain objects and arrays don't need unwrapping\n  try {\n    // JSON round-trip strips Proxies. Safe because state values must be JSON-serializable.\n    return JSON.parse(JSON.stringify(value));\n  } catch {\n    // Non-serializable (functions, symbols, etc.) — return as-is\n    return value;\n  }\n}\n\n// -- $-method routing --------------------------------------------------------\n\ntype MethodRouter = (target: ReactiveTarget, opts: ReactiveState) => unknown;\n\nconst METHOD_ROUTES: Record<string, MethodRouter> = {\n  $getValue: (t) => t.getValue.bind(t),\n  $setValue: (t) => t.setValue.bind(t),\n  $update: (t) => t.updateValue.bind(t),\n  $delete: (t) => t.deleteValue.bind(t),\n  $read: (t) => (dotPath: string) => {\n    const rootKey = dotPath.split('.')[0];\n    const value = t.getValue(rootKey);\n    if (!dotPath.includes('.')) return value;\n    return lodashGet(value, dotPath.slice(rootKey.length + 1));\n  },\n  $getArgs: (t) => t.getArgs.bind(t),\n  $getEnv: (t) => t.getEnv.bind(t),\n  $debug: (t) => t.addDebugInfo.bind(t),\n  $log: (t) => t.addDebugMessage.bind(t),\n  $error: (t) => t.addErrorInfo.bind(t),\n  $metric: (t) => t.addMetric.bind(t),\n  $eval: (t) => t.addEval.bind(t),\n  $attachRecorder: (t) => t.attachRecorder.bind(t),\n  $detachRecorder: (t) => t.detachRecorder.bind(t),\n  $getRecorders: (t) => t.getRecorders.bind(t),\n  $break: (_t, opts) => () => {\n    if (!opts.breakFn) throw new Error('$break() is not available outside stage execution');\n    opts.breakFn();\n  },\n  $toRaw: (t) => () => t,\n};\n\n// -- Guard properties --------------------------------------------------------\n// These must be handled to prevent Proxy from being treated as a Promise,\n// breaking instanceof checks, or confusing test matchers.\n\nconst GUARD_PROPS: Record<string | symbol, unknown> = {\n  then: undefined, // prevent Promise detection\n  asymmetricMatch: undefined, // prevent vitest/jest matcher confusion\n  constructor: Object, // safe prototype\n  [Symbol.toStringTag]: 'TypedScope',\n};\n\n// -- Mutable state per proxy instance ----------------------------------------\n\ninterface ReactiveState {\n  breakFn?: () => void;\n  /** Cache: top-level key -> { raw object ref, child proxy } */\n  childCache: Map<string, { ref: object; proxy: object }>;\n}\n\n// -- Nested child proxy (for deep write interception) ------------------------\n//\n// Cycle safety: an immutable Set<object> of ancestor objects is passed down\n// each access chain. Each branch gets its own copy (new Set(parent)) so\n// scope.x.friend and scope.x.coworker don't pollute each other's tracking.\n// When a child value is already in the ancestor set, we've hit a cycle.\n// At the cycle break: return a terminal proxy that tracks writes (set trap\n// still builds path + calls updateValue) but doesn't recurse reads further.\n\nfunction createTerminalProxy(\n  obj: Record<string, unknown>,\n  rootKey: string,\n  segments: string[],\n  target: ReactiveTarget,\n  state: ReactiveState,\n  visited: Set<object> = new Set(),\n): unknown {\n  visited.add(obj);\n\n  return new Proxy(obj, {\n    get(raw, prop) {\n      if (typeof prop === 'symbol') return (raw as any)[prop];\n      if (prop === 'then') return undefined;\n      if (prop === 'asymmetricMatch') return undefined;\n      if (prop === 'constructor') return Object;\n      if (prop === 'toJSON')\n        return () => {\n          // Strip object-typed values to prevent circular JSON errors\n          const safe: Record<string, unknown> = {};\n          for (const k of Object.keys(raw)) {\n            const v = (raw as any)[k];\n            if (v === null || typeof v !== 'object') safe[k] = v;\n          }\n          return safe;\n        };\n\n      const value = (raw as any)[prop];\n\n      // Continue tracking writes at deeper levels via chained terminal proxies.\n      // Use visited set to prevent re-entering the same object (cycle in terminal chain).\n      if (shouldWrapWithProxy(value) && !Array.isArray(value) && !visited.has(value as object)) {\n        return createTerminalProxy(\n          value as Record<string, unknown>,\n          rootKey,\n          [...segments, prop as string],\n          target,\n          state,\n          visited,\n        );\n      }\n\n      return value;\n    },\n    set(raw, prop, value) {\n      if (typeof prop !== 'string') return true;\n      const childSegments = [...segments, prop];\n      const patch = buildNestedPatch(childSegments, unwrapProxy(value));\n      target.updateValue(rootKey, patch);\n      state.childCache.delete(rootKey);\n      return true;\n    },\n  });\n}\n\nfunction createNestedProxy(\n  obj: Record<string, unknown>,\n  rootKey: string,\n  segments: string[],\n  target: ReactiveTarget,\n  state: ReactiveState,\n  ancestors: Set<object> = new Set(),\n): unknown {\n  return new Proxy(obj, {\n    get(raw, prop) {\n      if (typeof prop === 'symbol') return (raw as any)[prop];\n\n      // Guard properties\n      if (prop === 'then') return undefined;\n      if (prop === 'asymmetricMatch') return undefined;\n      if (prop === 'constructor') return Object;\n      if (prop === 'toJSON')\n        return () => {\n          // Strip object-typed values to prevent circular JSON errors\n          const safe: Record<string, unknown> = {};\n          for (const k of Object.keys(raw)) {\n            const v = (raw as any)[k];\n            if (v === null || typeof v !== 'object') safe[k] = v;\n          }\n          return safe;\n        };\n\n      const value = (raw as any)[prop];\n\n      // Primitive or non-wrappable -- return as-is (no deeper proxy)\n      if (!shouldWrapWithProxy(value)) return value;\n\n      const childSegments = [...segments, prop as string];\n\n      // Array -- return array proxy\n      if (Array.isArray(value)) {\n        return createArrayProxy(\n          () => {\n            const current = target.getValue(rootKey) as any;\n            return lodashGet(current, childSegments.join('.')) ?? [];\n          },\n          (newArr) => {\n            const patch = buildNestedPatch(childSegments, unwrapProxy(newArr));\n            target.updateValue(rootKey, patch);\n            state.childCache.delete(rootKey);\n          },\n        );\n      }\n\n      // Cycle detection: if this value is an ancestor in the current access\n      // chain, return a terminal proxy (tracks writes, stops recursing reads).\n      if (ancestors.has(value as object)) {\n        return createTerminalProxy(value as Record<string, unknown>, rootKey, childSegments, target, state);\n      }\n\n      // Build new ancestor set for this branch (immutable -- no cross-branch pollution)\n      const childAncestors = new Set(ancestors);\n      childAncestors.add(value as object);\n\n      return createNestedProxy(value as Record<string, unknown>, rootKey, childSegments, target, state, childAncestors);\n    },\n\n    set(raw, prop, value) {\n      if (typeof prop !== 'string') return true;\n\n      const childSegments = [...segments, prop];\n      const patch = buildNestedPatch(childSegments, unwrapProxy(value));\n      target.updateValue(rootKey, patch);\n      state.childCache.delete(rootKey);\n      return true;\n    },\n  });\n}\n\n// -- Top-level proxy (the main TypedScope) -----------------------------------\n\n/**\n * Creates a TypedScope<T> proxy wrapping a ReactiveTarget.\n *\n * @param target - The underlying scope (ScopeFacade or any ReactiveTarget)\n * @param options - Optional configuration (breakPipeline injection)\n * @returns A Proxy with typed property access and $-prefixed methods\n */\nexport function createTypedScope<T extends object>(target: ReactiveTarget, options?: ReactiveOptions): TypedScope<T> {\n  const state: ReactiveState = {\n    breakFn: options?.breakPipeline,\n    childCache: new Map(),\n  };\n\n  const proxy = new Proxy(target as unknown as TypedScope<T>, {\n    get(_proxyTarget, prop, _receiver) {\n      // 1. Internal symbols (check before other symbols)\n      if (prop === IS_TYPED_SCOPE) return true;\n      if (prop === BREAK_SETTER) {\n        return (fn: () => void) => {\n          state.breakFn = fn;\n        };\n      }\n\n      // 2. Symbol properties (guard + inspection)\n      if (typeof prop === 'symbol') {\n        if (Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop)) return GUARD_PROPS[prop];\n        // Node.js util.inspect — show state snapshot, not proxy internals\n        if (prop === Symbol.for('nodejs.util.inspect.custom')) {\n          return () => target.getValue();\n        }\n        return undefined;\n      }\n\n      // 3. String guard properties\n      if (Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop)) return GUARD_PROPS[prop];\n\n      // 4. $-prefixed methods -- route to facade\n      if (SCOPE_METHOD_NAMES.has(prop)) {\n        const router = METHOD_ROUTES[prop];\n        if (router) return router(target, state);\n        return undefined;\n      }\n\n      // 5. Executor-internal method pass-through (explicit allowlist)\n      //    FlowChartExecutor wrapping calls attachRecorder, notifyStageStart, etc.\n      //    directly on the scope. Forward only allowlisted methods.\n      if (EXECUTOR_INTERNAL_METHODS.has(prop) && typeof (target as any)[prop] === 'function') {\n        return (target as any)[prop].bind(target);\n      }\n\n      // 6. State key -- call getValue (fires onRead ONCE)\n      const value = target.getValue(prop);\n\n      // Primitive or null/undefined -- return as-is\n      if (value === null || value === undefined || typeof value !== 'object') {\n        return value;\n      }\n\n      // Non-wrappable (Date, Map, class instance, etc.) -- return unwrapped\n      if (!shouldWrapWithProxy(value)) return value;\n\n      // Array -- return array proxy (cached for identity equality)\n      if (Array.isArray(value)) {\n        const cached = state.childCache.get(prop);\n        if (cached && cached.ref === value) return cached.proxy;\n\n        const arrProxy = createArrayProxy(\n          () => (target.getValue(prop) as unknown[]) ?? [],\n          (newArr) => {\n            target.setValue(prop, unwrapProxy(newArr));\n            state.childCache.delete(prop);\n          },\n        );\n        state.childCache.set(prop, { ref: value as object, proxy: arrProxy as unknown as object });\n        return arrProxy;\n      }\n\n      // Plain object -- return nested proxy (cached for identity equality)\n      const cached = state.childCache.get(prop);\n      if (cached && cached.ref === value) return cached.proxy;\n\n      const nested = createNestedProxy(\n        value as Record<string, unknown>,\n        prop,\n        [],\n        target,\n        state,\n        new Set<object>([value as object]), // seed ancestor set with root object\n      );\n      state.childCache.set(prop, { ref: value as object, proxy: nested as object });\n      return nested;\n    },\n\n    set(_proxyTarget, prop, value) {\n      if (typeof prop !== 'string') return true;\n      if (SCOPE_METHOD_NAMES.has(prop)) {\n        throw new Error(\n          `Cannot set state key \"${prop}\" -- it conflicts with a reserved TypedScope method. Rename the state key to avoid $-prefixed names.`,\n        );\n      }\n      // Unwrap Proxy values before storing — structuredClone in TransactionBuffer\n      // cannot clone Proxy objects. This handles: scope.backup = scope.customer\n      const unwrapped = unwrapProxy(value);\n      target.setValue(prop, unwrapped);\n      state.childCache.delete(prop); // invalidate cache\n      return true;\n    },\n\n    deleteProperty(_proxyTarget, prop) {\n      if (typeof prop !== 'string') return true;\n      target.deleteValue(prop);\n      state.childCache.delete(prop);\n      return true;\n    },\n\n    has(_proxyTarget, prop) {\n      if (typeof prop === 'symbol') return Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop);\n      if (SCOPE_METHOD_NAMES.has(prop)) return true;\n      // Use non-tracking hasKey if available, else fallback to getStateKeys\n      if (target.hasKey) return target.hasKey(prop);\n      if (target.getStateKeys) return target.getStateKeys().includes(prop);\n      // Fallback: getValue fires onRead (acceptable degradation)\n      return target.getValue(prop) !== undefined;\n    },\n\n    ownKeys() {\n      // Use non-tracking getStateKeys if available, else fallback\n      if (target.getStateKeys) return target.getStateKeys();\n      const snapshot = target.getValue() as Record<string, unknown> | undefined;\n      if (!snapshot || typeof snapshot !== 'object') return [];\n      return Object.keys(snapshot);\n    },\n\n    getOwnPropertyDescriptor(_proxyTarget, prop) {\n      if (typeof prop !== 'string') return undefined;\n      if (SCOPE_METHOD_NAMES.has(prop)) return undefined; // $-methods are non-enumerable\n      // Check existence without firing onRead — no getValue call here\n      const exists = target.hasKey\n        ? target.hasKey(prop)\n        : target.getStateKeys\n        ? target.getStateKeys().includes(prop)\n        : target.getValue(prop) !== undefined; // fallback only\n      if (!exists) return undefined;\n      // Return a minimal descriptor — actual value is fetched via the get trap\n      return { configurable: true, enumerable: true, writable: true };\n    },\n  });\n\n  return proxy;\n}\n"]}
342
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createTypedScope.js","sourceRoot":"","sources":["../../../src/lib/reactive/createTypedScope.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * reactive/createTypedScope -- Core Proxy factory for TypedScope<T>.\n *\n * Wraps a ReactiveTarget (ScopeFacade) in a Proxy that provides:\n * - Typed property access: scope.creditTier (read), scope.creditTier = 'A' (write)\n * - Deep write interception: scope.customer.address.zip = '90210'\n * - Array mutation interception: scope.items.push('new')\n * - $-prefixed escape hatches: $getValue, $setValue, $read, $getArgs, etc.\n *\n * Read semantics: top-level get calls getValue() (fires onRead ONCE).\n *   Nested get traps navigate in-memory -- no additional onRead.\n *\n * Write semantics: top-level set calls setValue(). Nested set calls\n *   updateValue() with a partial object built from the accumulated path.\n */\n\nimport { nativeGet as lodashGet } from '../memory/pathOps.js';\nimport { shouldWrapWithProxy } from './allowlist.js';\nimport { createArrayProxy } from './arrayTraps.js';\nimport { buildNestedPatch } from './pathBuilder.js';\nimport type { ReactiveOptions, ReactiveTarget, TypedScope } from './types.js';\nimport { BREAK_SETTER, EXECUTOR_INTERNAL_METHODS, IS_TYPED_SCOPE, SCOPE_METHOD_NAMES } from './types.js';\n\n// -- Proxy unwrapping --------------------------------------------------------\n// structuredClone in TransactionBuffer cannot clone Proxy objects.\n// When a user does `scope.backup = scope.customer`, the value is a Proxy.\n// Unwrap to a plain object before storing.\n\nfunction unwrapProxy(value: unknown): unknown {\n  if (value === null || value === undefined) return value;\n  if (typeof value !== 'object') return value;\n  // Fast path: plain objects and arrays don't need unwrapping\n  try {\n    // JSON round-trip strips Proxies. Safe because state values must be JSON-serializable.\n    return JSON.parse(JSON.stringify(value));\n  } catch {\n    // Non-serializable (functions, symbols, etc.) — return as-is\n    return value;\n  }\n}\n\n// -- $-method routing --------------------------------------------------------\n\ntype MethodRouter = (target: ReactiveTarget, opts: ReactiveState) => unknown;\n\nconst METHOD_ROUTES: Record<string, MethodRouter> = {\n  $getValue: (t) => t.getValue.bind(t),\n  $setValue: (t) => t.setValue.bind(t),\n  $update: (t) => t.updateValue.bind(t),\n  $delete: (t) => t.deleteValue.bind(t),\n  $read: (t) => (dotPath: string) => {\n    const rootKey = dotPath.split('.')[0];\n    const value = t.getValue(rootKey);\n    if (!dotPath.includes('.')) return value;\n    return lodashGet(value, dotPath.slice(rootKey.length + 1));\n  },\n  $getArgs: (t) => t.getArgs.bind(t),\n  $getEnv: (t) => t.getEnv.bind(t),\n  $debug: (t) => t.addDebugInfo.bind(t),\n  $log: (t) => t.addDebugMessage.bind(t),\n  $error: (t) => t.addErrorInfo.bind(t),\n  $metric: (t) => t.addMetric.bind(t),\n  $eval: (t) => t.addEval.bind(t),\n  $attachRecorder: (t) => t.attachRecorder.bind(t),\n  $detachRecorder: (t) => t.detachRecorder.bind(t),\n  $getRecorders: (t) => t.getRecorders.bind(t),\n  $batchArray: (t) => (key: string, fn: (arr: unknown[]) => void) => {\n    // One getValue — fires onRead once\n    const current = t.getValue(key);\n    // Clone once (or start empty if missing/non-array)\n    const clone: unknown[] = Array.isArray(current) ? [...current] : [];\n    // User applies all mutations to the plain clone — no Proxy, no per-mutation commit\n    fn(clone);\n    // One setValue — fires onWrite once with the final array\n    t.setValue(key, clone);\n  },\n  $break: (_t, opts) => () => {\n    if (!opts.breakFn) throw new Error('$break() is not available outside stage execution');\n    opts.breakFn();\n  },\n  $toRaw: (t) => () => t,\n};\n\n// -- Guard properties --------------------------------------------------------\n// These must be handled to prevent Proxy from being treated as a Promise,\n// breaking instanceof checks, or confusing test matchers.\n\nconst GUARD_PROPS: Record<string | symbol, unknown> = {\n  then: undefined, // prevent Promise detection\n  asymmetricMatch: undefined, // prevent vitest/jest matcher confusion\n  constructor: Object, // safe prototype\n  [Symbol.toStringTag]: 'TypedScope',\n};\n\n// -- Mutable state per proxy instance ----------------------------------------\n\ninterface ReactiveState {\n  breakFn?: () => void;\n  /** Cache: top-level key -> { raw object ref, child proxy } */\n  childCache: Map<string, { ref: object; proxy: object }>;\n}\n\n// -- Nested child proxy (for deep write interception) ------------------------\n//\n// Cycle safety: an immutable Set<object> of ancestor objects is passed down\n// each access chain. Each branch gets its own copy (new Set(parent)) so\n// scope.x.friend and scope.x.coworker don't pollute each other's tracking.\n// When a child value is already in the ancestor set, we've hit a cycle.\n// At the cycle break: return a terminal proxy that tracks writes (set trap\n// still builds path + calls updateValue) but doesn't recurse reads further.\n\nfunction createTerminalProxy(\n  obj: Record<string, unknown>,\n  rootKey: string,\n  segments: string[],\n  target: ReactiveTarget,\n  state: ReactiveState,\n  visited: Set<object> = new Set(),\n): unknown {\n  visited.add(obj);\n\n  return new Proxy(obj, {\n    get(raw, prop) {\n      if (typeof prop === 'symbol') return (raw as any)[prop];\n      if (prop === 'then') return undefined;\n      if (prop === 'asymmetricMatch') return undefined;\n      if (prop === 'constructor') return Object;\n      if (prop === 'toJSON')\n        return () => {\n          // Strip object-typed values to prevent circular JSON errors\n          const safe: Record<string, unknown> = {};\n          for (const k of Object.keys(raw)) {\n            const v = (raw as any)[k];\n            if (v === null || typeof v !== 'object') safe[k] = v;\n          }\n          return safe;\n        };\n\n      const value = (raw as any)[prop];\n\n      // Continue tracking writes at deeper levels via chained terminal proxies.\n      // Use visited set to prevent re-entering the same object (cycle in terminal chain).\n      if (shouldWrapWithProxy(value) && !Array.isArray(value) && !visited.has(value as object)) {\n        return createTerminalProxy(\n          value as Record<string, unknown>,\n          rootKey,\n          [...segments, prop as string],\n          target,\n          state,\n          visited,\n        );\n      }\n\n      return value;\n    },\n    set(raw, prop, value) {\n      if (typeof prop !== 'string') return true;\n      const childSegments = [...segments, prop];\n      const patch = buildNestedPatch(childSegments, unwrapProxy(value));\n      target.updateValue(rootKey, patch);\n      state.childCache.delete(rootKey);\n      return true;\n    },\n  });\n}\n\nfunction createNestedProxy(\n  obj: Record<string, unknown>,\n  rootKey: string,\n  segments: string[],\n  target: ReactiveTarget,\n  state: ReactiveState,\n  ancestors: Set<object> = new Set(),\n): unknown {\n  return new Proxy(obj, {\n    get(raw, prop) {\n      if (typeof prop === 'symbol') return (raw as any)[prop];\n\n      // Guard properties\n      if (prop === 'then') return undefined;\n      if (prop === 'asymmetricMatch') return undefined;\n      if (prop === 'constructor') return Object;\n      if (prop === 'toJSON')\n        return () => {\n          // Strip object-typed values to prevent circular JSON errors\n          const safe: Record<string, unknown> = {};\n          for (const k of Object.keys(raw)) {\n            const v = (raw as any)[k];\n            if (v === null || typeof v !== 'object') safe[k] = v;\n          }\n          return safe;\n        };\n\n      const value = (raw as any)[prop];\n\n      // Primitive or non-wrappable -- return as-is (no deeper proxy)\n      if (!shouldWrapWithProxy(value)) return value;\n\n      const childSegments = [...segments, prop as string];\n\n      // Array -- return array proxy\n      if (Array.isArray(value)) {\n        return createArrayProxy(\n          () => {\n            const current = target.getValue(rootKey) as any;\n            return lodashGet(current, childSegments.join('.')) ?? [];\n          },\n          (newArr) => {\n            const patch = buildNestedPatch(childSegments, unwrapProxy(newArr));\n            target.updateValue(rootKey, patch);\n            state.childCache.delete(rootKey);\n          },\n        );\n      }\n\n      // Cycle detection: if this value is an ancestor in the current access\n      // chain, return a terminal proxy (tracks writes, stops recursing reads).\n      if (ancestors.has(value as object)) {\n        return createTerminalProxy(value as Record<string, unknown>, rootKey, childSegments, target, state);\n      }\n\n      // Build new ancestor set for this branch (immutable -- no cross-branch pollution)\n      const childAncestors = new Set(ancestors);\n      childAncestors.add(value as object);\n\n      return createNestedProxy(value as Record<string, unknown>, rootKey, childSegments, target, state, childAncestors);\n    },\n\n    set(raw, prop, value) {\n      if (typeof prop !== 'string') return true;\n\n      const childSegments = [...segments, prop];\n      const patch = buildNestedPatch(childSegments, unwrapProxy(value));\n      target.updateValue(rootKey, patch);\n      state.childCache.delete(rootKey);\n      return true;\n    },\n  });\n}\n\n// -- Top-level proxy (the main TypedScope) -----------------------------------\n\n/**\n * Creates a TypedScope<T> proxy wrapping a ReactiveTarget.\n *\n * @param target - The underlying scope (ScopeFacade or any ReactiveTarget)\n * @param options - Optional configuration (breakPipeline injection)\n * @returns A Proxy with typed property access and $-prefixed methods\n */\nexport function createTypedScope<T extends object>(target: ReactiveTarget, options?: ReactiveOptions): TypedScope<T> {\n  const state: ReactiveState = {\n    breakFn: options?.breakPipeline,\n    childCache: new Map(),\n  };\n\n  const proxy = new Proxy(target as unknown as TypedScope<T>, {\n    get(_proxyTarget, prop, _receiver) {\n      // 1. Internal symbols (check before other symbols)\n      if (prop === IS_TYPED_SCOPE) return true;\n      if (prop === BREAK_SETTER) {\n        return (fn: () => void) => {\n          state.breakFn = fn;\n        };\n      }\n\n      // 2. Symbol properties (guard + inspection)\n      if (typeof prop === 'symbol') {\n        if (Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop)) return GUARD_PROPS[prop];\n        // Node.js util.inspect — show state snapshot, not proxy internals\n        if (prop === Symbol.for('nodejs.util.inspect.custom')) {\n          return () => target.getValue();\n        }\n        return undefined;\n      }\n\n      // 3. String guard properties\n      if (Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop)) return GUARD_PROPS[prop];\n\n      // 4. $-prefixed methods -- route to facade\n      if (SCOPE_METHOD_NAMES.has(prop)) {\n        const router = METHOD_ROUTES[prop];\n        if (router) return router(target, state);\n        return undefined;\n      }\n\n      // 5. Executor-internal method pass-through (explicit allowlist)\n      //    FlowChartExecutor wrapping calls attachRecorder, notifyStageStart, etc.\n      //    directly on the scope. Forward only allowlisted methods.\n      if (EXECUTOR_INTERNAL_METHODS.has(prop) && typeof (target as any)[prop] === 'function') {\n        return (target as any)[prop].bind(target);\n      }\n\n      // 6. State key -- call getValue (fires onRead ONCE)\n      const value = target.getValue(prop);\n\n      // Primitive or null/undefined -- return as-is\n      if (value === null || value === undefined || typeof value !== 'object') {\n        return value;\n      }\n\n      // Non-wrappable (Date, Map, class instance, etc.) -- return unwrapped\n      if (!shouldWrapWithProxy(value)) return value;\n\n      // Array -- return array proxy (cached for identity equality)\n      if (Array.isArray(value)) {\n        const cached = state.childCache.get(prop);\n        if (cached && cached.ref === value) return cached.proxy;\n\n        const arrProxy = createArrayProxy(\n          () => (target.getValue(prop) as unknown[]) ?? [],\n          (newArr) => {\n            target.setValue(prop, unwrapProxy(newArr));\n            state.childCache.delete(prop);\n          },\n        );\n        state.childCache.set(prop, { ref: value as object, proxy: arrProxy as unknown as object });\n        return arrProxy;\n      }\n\n      // Plain object -- return nested proxy (cached for identity equality)\n      const cached = state.childCache.get(prop);\n      if (cached && cached.ref === value) return cached.proxy;\n\n      const nested = createNestedProxy(\n        value as Record<string, unknown>,\n        prop,\n        [],\n        target,\n        state,\n        new Set<object>([value as object]), // seed ancestor set with root object\n      );\n      state.childCache.set(prop, { ref: value as object, proxy: nested as object });\n      return nested;\n    },\n\n    set(_proxyTarget, prop, value) {\n      if (typeof prop !== 'string') return true;\n      if (SCOPE_METHOD_NAMES.has(prop)) {\n        throw new Error(\n          `Cannot set state key \"${prop}\" -- it conflicts with a reserved TypedScope method. Rename the state key to avoid $-prefixed names.`,\n        );\n      }\n      // Unwrap Proxy values before storing — structuredClone in TransactionBuffer\n      // cannot clone Proxy objects. This handles: scope.backup = scope.customer\n      const unwrapped = unwrapProxy(value);\n      target.setValue(prop, unwrapped);\n      state.childCache.delete(prop); // invalidate cache\n      return true;\n    },\n\n    deleteProperty(_proxyTarget, prop) {\n      if (typeof prop !== 'string') return true;\n      target.deleteValue(prop);\n      state.childCache.delete(prop);\n      return true;\n    },\n\n    has(_proxyTarget, prop) {\n      if (typeof prop === 'symbol') return Object.prototype.hasOwnProperty.call(GUARD_PROPS, prop);\n      if (SCOPE_METHOD_NAMES.has(prop)) return true;\n      // Use non-tracking hasKey if available, else fallback to getStateKeys\n      if (target.hasKey) return target.hasKey(prop);\n      if (target.getStateKeys) return target.getStateKeys().includes(prop);\n      // Fallback: getValue fires onRead (acceptable degradation)\n      return target.getValue(prop) !== undefined;\n    },\n\n    ownKeys() {\n      // Use non-tracking getStateKeys if available, else fallback\n      if (target.getStateKeys) return target.getStateKeys();\n      const snapshot = target.getValue() as Record<string, unknown> | undefined;\n      if (!snapshot || typeof snapshot !== 'object') return [];\n      return Object.keys(snapshot);\n    },\n\n    getOwnPropertyDescriptor(_proxyTarget, prop) {\n      if (typeof prop !== 'string') return undefined;\n      if (SCOPE_METHOD_NAMES.has(prop)) return undefined; // $-methods are non-enumerable\n      // Check existence without firing onRead — no getValue call here\n      const exists = target.hasKey\n        ? target.hasKey(prop)\n        : target.getStateKeys\n        ? target.getStateKeys().includes(prop)\n        : target.getValue(prop) !== undefined; // fallback only\n      if (!exists) return undefined;\n      // Return a minimal descriptor — actual value is fetched via the get trap\n      return { configurable: true, enumerable: true, writable: true };\n    },\n  });\n\n  return proxy;\n}\n"]}
@@ -28,6 +28,7 @@ exports.SCOPE_METHOD_NAMES = new Set([
28
28
  '$attachRecorder',
29
29
  '$detachRecorder',
30
30
  '$getRecorders',
31
+ '$batchArray',
31
32
  '$break',
32
33
  '$toRaw',
33
34
  ]);
@@ -52,4 +53,4 @@ exports.EXECUTOR_INTERNAL_METHODS = new Set([
52
53
  'useSharedRedactedKeys', // FlowChartExecutor.createTraverser() — redaction wrapping
53
54
  'useRedactionPolicy', // FlowChartExecutor.createTraverser() — redaction wrapping
54
55
  ]);
55
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/lib/reactive/types.ts"],"names":[],"mappings":";AAAA;;;;;;;;GAQG;;;AA2FH,+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,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  // 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  '$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"]}
@@ -46,6 +46,41 @@ export interface ScopeMethods {
46
46
  $attachRecorder(recorder: Recorder): void;
47
47
  $detachRecorder(recorderId: string): void;
48
48
  $getRecorders(): Recorder[];
49
+ /**
50
+ * Batch-mutate an array key in a single clone+write cycle.
51
+ *
52
+ * Every `scope.items.push(x)` clones the entire array and commits it — O(N) per call.
53
+ * For N mutations on an M-length array that is O(N×M). Use `$batchArray` to clone once,
54
+ * apply all mutations inside `fn`, then commit once — O(M) total.
55
+ *
56
+ * ```typescript
57
+ * // Before: 1000 clones × growing array = O(N²)
58
+ * for (let i = 0; i < 1000; i++) scope.items.push(i);
59
+ *
60
+ * // After: 1 clone + 1 commit = O(N)
61
+ * scope.$batchArray('items', (arr) => {
62
+ * for (let i = 0; i < 1000; i++) arr.push(i);
63
+ * });
64
+ * ```
65
+ *
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)`.
82
+ */
83
+ $batchArray(key: string, fn: (arr: unknown[]) => void): void;
49
84
  $break(): void;
50
85
  $toRaw(): ReactiveTarget;
51
86
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "footprintjs",
3
- "version": "4.0.2",
3
+ "version": "4.0.4",
4
4
  "description": "Explainable backend flows — automatic causal traces, decision evidence, and MCP tool generation for AI agents",
5
5
  "license": "MIT",
6
6
  "author": "Sanjay Krishna Anbalagan",