@remnic/core 1.1.1 → 1.1.3
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/dist/abort-error.js +1 -0
- package/dist/abstraction-nodes.js +1 -0
- package/dist/access-audit.js +1 -0
- package/dist/access-cli.js +76 -51
- package/dist/access-cli.js.map +1 -1
- package/dist/access-http.d.ts +50 -5
- package/dist/access-http.js +38 -16
- package/dist/access-idempotency.js +1 -0
- package/dist/access-mcp.d.ts +10 -5
- package/dist/access-mcp.js +37 -14
- package/dist/access-schema.d.ts +133 -13
- package/dist/access-schema.js +20 -1
- package/dist/access-service-_AEUMVyX.d.ts +1981 -0
- package/dist/access-service.d.ts +11 -6
- package/dist/access-service.js +39 -14
- package/dist/active-memory-bridge.js +1 -0
- package/dist/active-recall.js +1 -0
- package/dist/active-recall.js.map +1 -1
- package/dist/behavior-learner.js +1 -0
- package/dist/behavior-learner.js.map +1 -1
- package/dist/behavior-signals.js +1 -0
- package/dist/bootstrap.d.ts +6 -4
- package/dist/bootstrap.js +1 -0
- package/dist/boxes.js +1 -0
- package/dist/briefing.d.ts +9 -5
- package/dist/briefing.js +10 -6
- package/dist/buffer-surprise-report.js +1 -0
- package/dist/buffer-surprise.js +1 -0
- package/dist/buffer.d.ts +1 -1
- package/dist/buffer.js +1 -0
- package/dist/calibration.d.ts +8 -1
- package/dist/calibration.js +10 -2
- package/dist/calibration.js.map +1 -1
- package/dist/capsule-cli.d.ts +137 -0
- package/dist/capsule-cli.js +34 -0
- package/dist/capsule-crypto-5CYAGVC5.js +18 -0
- package/dist/capsule-export-NZQPOTQ4.js +17 -0
- package/dist/capsule-export-NZQPOTQ4.js.map +1 -0
- package/dist/capsule-import-SDCUXLEV.js +16 -0
- package/dist/capsule-import-SDCUXLEV.js.map +1 -0
- package/dist/capsule-merge-DI7PNQ2H.js +189 -0
- package/dist/capsule-merge-DI7PNQ2H.js.map +1 -0
- package/dist/causal-behavior.js +1 -0
- package/dist/causal-behavior.js.map +1 -1
- package/dist/causal-chain.js +1 -0
- package/dist/causal-consolidation.js +12 -9
- package/dist/causal-consolidation.js.map +1 -1
- package/dist/causal-retrieval.js +2 -1
- package/dist/causal-retrieval.js.map +1 -1
- package/dist/causal-trajectory-graph.js +4 -1
- package/dist/causal-trajectory-graph.js.map +1 -1
- package/dist/causal-trajectory.js +2 -1
- package/dist/chunk-2LSZVONP.js +67 -0
- package/dist/chunk-2LSZVONP.js.map +1 -0
- package/dist/chunk-32KD5IHZ.js +245 -0
- package/dist/chunk-32KD5IHZ.js.map +1 -0
- package/dist/{chunk-VDX363PS.js → chunk-34F3PLWZ.js} +10 -3
- package/dist/chunk-34F3PLWZ.js.map +1 -0
- package/dist/chunk-3KIS4VGT.js +228 -0
- package/dist/chunk-3KIS4VGT.js.map +1 -0
- package/dist/chunk-3LCWFNVS.js +350 -0
- package/dist/chunk-3LCWFNVS.js.map +1 -0
- package/dist/chunk-43EKP2UK.js +26 -0
- package/dist/chunk-43EKP2UK.js.map +1 -0
- package/dist/chunk-457A4P3L.js +119 -0
- package/dist/chunk-457A4P3L.js.map +1 -0
- package/dist/{chunk-KUB6JU6H.js → chunk-47WOM4YW.js} +2 -2
- package/dist/{chunk-HK3FGIEW.js → chunk-4PLGJRBV.js} +656 -20
- package/dist/chunk-4PLGJRBV.js.map +1 -0
- package/dist/{chunk-BGJGXLZ7.js → chunk-55FXRRSJ.js} +11 -8
- package/dist/chunk-55FXRRSJ.js.map +1 -0
- package/dist/{chunk-ULYOGL6R.js → chunk-5HRY2WRF.js} +7 -3
- package/dist/chunk-5HRY2WRF.js.map +1 -0
- package/dist/chunk-6TBWYBJ3.js +236 -0
- package/dist/chunk-6TBWYBJ3.js.map +1 -0
- package/dist/chunk-74EMIVE4.js +329 -0
- package/dist/chunk-74EMIVE4.js.map +1 -0
- package/dist/chunk-74WWN7ZW.js +82 -0
- package/dist/chunk-74WWN7ZW.js.map +1 -0
- package/dist/{chunk-B5WXLVDY.js → chunk-7GCMLT7J.js} +245 -25
- package/dist/chunk-7GCMLT7J.js.map +1 -0
- package/dist/chunk-A6XUJE5D.js +126 -0
- package/dist/chunk-A6XUJE5D.js.map +1 -0
- package/dist/chunk-AJA46VX5.js +393 -0
- package/dist/chunk-AJA46VX5.js.map +1 -0
- package/dist/{chunk-DFTTJYSO.js → chunk-AKUCB2OG.js} +525 -24
- package/dist/chunk-AKUCB2OG.js.map +1 -0
- package/dist/chunk-ASIQZXYO.js +277 -0
- package/dist/chunk-ASIQZXYO.js.map +1 -0
- package/dist/{chunk-ZEM3OK2K.js → chunk-B2TL6GA2.js} +3 -3
- package/dist/chunk-BJMBJZ2Y.js +290 -0
- package/dist/chunk-BJMBJZ2Y.js.map +1 -0
- package/dist/chunk-BT7NVCML.js +79 -0
- package/dist/chunk-BT7NVCML.js.map +1 -0
- package/dist/chunk-CK5NTM2S.js +454 -0
- package/dist/chunk-CK5NTM2S.js.map +1 -0
- package/dist/{chunk-3GXCSUXR.js → chunk-CRU27Q4J.js} +2 -2
- package/dist/{chunk-F5VP6YCB.js → chunk-DCE6SQLA.js} +572 -155
- package/dist/chunk-DCE6SQLA.js.map +1 -0
- package/dist/{chunk-CUPFXL3J.js → chunk-DHRQHX36.js} +4 -4
- package/dist/chunk-DHRQHX36.js.map +1 -0
- package/dist/{chunk-GKFXUTJ2.js → chunk-DR7MCMPS.js} +981 -61
- package/dist/chunk-DR7MCMPS.js.map +1 -0
- package/dist/chunk-FP2373TW.js +149 -0
- package/dist/chunk-FP2373TW.js.map +1 -0
- package/dist/{chunk-RBBWYEFJ.js → chunk-G2WADRQ3.js} +1 -1
- package/dist/chunk-G7D6GZ5J.js +48 -0
- package/dist/chunk-G7D6GZ5J.js.map +1 -0
- package/dist/chunk-H7XKCNR6.js +60 -0
- package/dist/chunk-H7XKCNR6.js.map +1 -0
- package/dist/{chunk-VYM3VWOF.js → chunk-IM3JSE73.js} +966 -329
- package/dist/chunk-IM3JSE73.js.map +1 -0
- package/dist/chunk-IXEJRKCZ.js +18 -0
- package/dist/chunk-IXEJRKCZ.js.map +1 -0
- package/dist/chunk-IYY4MCPG.js +275 -0
- package/dist/chunk-IYY4MCPG.js.map +1 -0
- package/dist/{chunk-BK2EFTE2.js → chunk-JWSENLQI.js} +508 -28
- package/dist/chunk-JWSENLQI.js.map +1 -0
- package/dist/chunk-KNKUID7G.js +183 -0
- package/dist/chunk-KNKUID7G.js.map +1 -0
- package/dist/chunk-L2IO2QPY.js +2036 -0
- package/dist/chunk-L2IO2QPY.js.map +1 -0
- package/dist/{chunk-SPI27QT6.js → chunk-L5IIGA5V.js} +9 -4
- package/dist/chunk-L5IIGA5V.js.map +1 -0
- package/dist/{chunk-RGLL5SPU.js → chunk-LVYGDT5V.js} +56 -82
- package/dist/chunk-LVYGDT5V.js.map +1 -0
- package/dist/{chunk-ZAIM4TUE.js → chunk-LW2NMHDW.js} +46 -1
- package/dist/chunk-LW2NMHDW.js.map +1 -0
- package/dist/{chunk-3OGMS3PE.js → chunk-LZRYQK6L.js} +3 -2
- package/dist/chunk-LZRYQK6L.js.map +1 -0
- package/dist/chunk-MDYG7VI7.js +48 -0
- package/dist/chunk-MDYG7VI7.js.map +1 -0
- package/dist/chunk-MXC3AP5I.js +74 -0
- package/dist/chunk-MXC3AP5I.js.map +1 -0
- package/dist/{chunk-S3EEFKNY.js → chunk-N7X62G74.js} +26 -11
- package/dist/chunk-N7X62G74.js.map +1 -0
- package/dist/chunk-NN3TS5BM.js +147 -0
- package/dist/chunk-NN3TS5BM.js.map +1 -0
- package/dist/chunk-OA3L7BFR.js +183 -0
- package/dist/chunk-OA3L7BFR.js.map +1 -0
- package/dist/{chunk-LK6SGL53.js → chunk-OR64ZGRZ.js} +3 -2
- package/dist/chunk-OR64ZGRZ.js.map +1 -0
- package/dist/chunk-OZHRDTDX.js +240 -0
- package/dist/chunk-OZHRDTDX.js.map +1 -0
- package/dist/chunk-PCUKNJAZ.js +165 -0
- package/dist/chunk-PCUKNJAZ.js.map +1 -0
- package/dist/{chunk-6PFRXT4K.js → chunk-PFV5C235.js} +11 -6
- package/dist/chunk-PFV5C235.js.map +1 -0
- package/dist/chunk-PZ5AY32C.js +10 -0
- package/dist/chunk-PZ5AY32C.js.map +1 -0
- package/dist/{chunk-XZ2TIKGC.js → chunk-Q7FJ5ZHM.js} +30 -10
- package/dist/chunk-Q7FJ5ZHM.js.map +1 -0
- package/dist/{chunk-7I7FKFZH.js → chunk-R2L7SUX2.js} +6 -6
- package/dist/{chunk-JL2PU6AI.js → chunk-R2XRID2N.js} +2 -2
- package/dist/{chunk-WCLICCGB.js → chunk-RILIVK4O.js} +91 -4
- package/dist/chunk-RILIVK4O.js.map +1 -0
- package/dist/{chunk-C2EFFULQ.js → chunk-RK2Y4XOM.js} +163 -20
- package/dist/chunk-RK2Y4XOM.js.map +1 -0
- package/dist/{chunk-TP4FZJIZ.js → chunk-RULE4VG5.js} +5 -1
- package/dist/chunk-RULE4VG5.js.map +1 -0
- package/dist/{chunk-PVPWZSSI.js → chunk-SMA4IMHV.js} +19 -3
- package/dist/chunk-SMA4IMHV.js.map +1 -0
- package/dist/{chunk-WVVA7F5A.js → chunk-SS253RXF.js} +30 -16
- package/dist/chunk-SS253RXF.js.map +1 -0
- package/dist/chunk-TUFG6VXY.js +875 -0
- package/dist/chunk-TUFG6VXY.js.map +1 -0
- package/dist/chunk-TYEOAFH3.js +251 -0
- package/dist/chunk-TYEOAFH3.js.map +1 -0
- package/dist/chunk-UKJAGEXH.js +260 -0
- package/dist/chunk-UKJAGEXH.js.map +1 -0
- package/dist/{chunk-KVBLZUKV.js → chunk-USFPPRAF.js} +93 -3
- package/dist/chunk-USFPPRAF.js.map +1 -0
- package/dist/{chunk-EPQJM2GC.js → chunk-VTJVUHRK.js} +22 -36
- package/dist/chunk-VTJVUHRK.js.map +1 -0
- package/dist/{chunk-O5ETUNBT.js → chunk-VTU2B4VF.js} +7 -3
- package/dist/chunk-VTU2B4VF.js.map +1 -0
- package/dist/chunk-WIICJPET.js +45 -0
- package/dist/chunk-WIICJPET.js.map +1 -0
- package/dist/{chunk-VBVG2M5G.js → chunk-WPGJYVUH.js} +6 -2
- package/dist/chunk-WPGJYVUH.js.map +1 -0
- package/dist/{chunk-YNQKWQT4.js → chunk-WSZIHQBK.js} +31 -11
- package/dist/{chunk-YNQKWQT4.js.map → chunk-WSZIHQBK.js.map} +1 -1
- package/dist/{chunk-NZLQTHS5.js → chunk-WW3QQF4H.js} +4 -1
- package/dist/chunk-WW3QQF4H.js.map +1 -0
- package/dist/{chunk-FVA6TGI3.js → chunk-Y3WQ4ZWK.js} +42 -2
- package/dist/chunk-Y3WQ4ZWK.js.map +1 -0
- package/dist/chunk-YNJHCGDT.js +309 -0
- package/dist/chunk-YNJHCGDT.js.map +1 -0
- package/dist/{chunk-ALXMCZEU.js → chunk-Z2E7VW55.js} +6 -3
- package/dist/chunk-Z2E7VW55.js.map +1 -0
- package/dist/{chunk-INXV5JBT.js → chunk-ZGXSCMQN.js} +1992 -410
- package/dist/chunk-ZGXSCMQN.js.map +1 -0
- package/dist/{chunk-W6SL7OFG.js → chunk-ZTSE2ZJ6.js} +12 -2
- package/dist/{chunk-W6SL7OFG.js.map → chunk-ZTSE2ZJ6.js.map} +1 -1
- package/dist/chunking.js +1 -0
- package/dist/cipher-GVE2GQ5H.js +28 -0
- package/dist/cipher-GVE2GQ5H.js.map +1 -0
- package/dist/citations.js +1 -0
- package/dist/{cli-BkeRaYfk.d.ts → cli-x2APT9a6.d.ts} +26 -7
- package/dist/cli.d.ts +11 -6
- package/dist/cli.js +68 -34
- package/dist/codex-thread-key.js +1 -0
- package/dist/commitment-ledger.js +1 -0
- package/dist/compression-optimizer.js +1 -0
- package/dist/config.d.ts +2 -1
- package/dist/config.js +5 -2
- package/dist/connectors-cli-DFGtY2DB.d.ts +257 -0
- package/dist/connectors-cli.d.ts +2 -0
- package/dist/connectors-cli.js +22 -0
- package/dist/connectors-cli.js.map +1 -0
- package/dist/consolidation-operator.d.ts +65 -5
- package/dist/consolidation-operator.js +6 -1
- package/dist/consolidation-provenance-check.d.ts +1 -1
- package/dist/consolidation-provenance-check.js +3 -2
- package/dist/consolidation-undo.d.ts +1 -1
- package/dist/consolidation-undo.js +1 -0
- package/dist/consolidation-undo.js.map +1 -1
- package/dist/{contradiction-review-WIUBAR52.js → contradiction-review-5LTTVDQV.js} +2 -1
- package/dist/contradiction-review-5LTTVDQV.js.map +1 -0
- package/dist/{contradiction-scan-E3GJTI4F.js → contradiction-scan-3Z6YW7YA.js} +2 -1
- package/dist/{contradiction-scan-E3GJTI4F.js.map → contradiction-scan-3Z6YW7YA.js.map} +1 -1
- package/dist/cross-namespace-budget.js +1 -0
- package/dist/cue-anchors.js +1 -0
- package/dist/dashboard-runtime.js +1 -0
- package/dist/day-summary.js +1 -0
- package/dist/delinearize.js +1 -0
- package/dist/direct-answer-wiring.js +1 -0
- package/dist/direct-answer.js +1 -0
- package/dist/dreams-ledger-LR2NBAZE.js +286 -0
- package/dist/dreams-ledger-LR2NBAZE.js.map +1 -0
- package/dist/embedding-fallback.js +3 -1
- package/dist/{engine-F3GOXGE5.js → engine-ICC2DSQF.js} +10 -7
- package/dist/engine-ICC2DSQF.js.map +1 -0
- package/dist/entity-retrieval.d.ts +1 -1
- package/dist/entity-retrieval.js +9 -6
- package/dist/entity-schema.js +1 -0
- package/dist/evals.js +1 -0
- package/dist/evidence-pack.d.ts +16 -0
- package/dist/evidence-pack.js +8 -0
- package/dist/evidence-pack.js.map +1 -0
- package/dist/explicit-capture.d.ts +6 -4
- package/dist/explicit-capture.js +1 -0
- package/dist/extraction-judge-telemetry.js +1 -0
- package/dist/extraction-judge-training.js +1 -0
- package/dist/extraction-judge.js +1 -0
- package/dist/extraction.js +9 -8
- package/dist/fallback-llm.js +3 -2
- package/dist/first-start-migration-4MHQEOSD.js +263 -0
- package/dist/first-start-migration-4MHQEOSD.js.map +1 -0
- package/dist/forget-PLR6J5DN.js +69 -0
- package/dist/forget-PLR6J5DN.js.map +1 -0
- package/dist/framework-CyHYDcri.d.ts +153 -0
- package/dist/fs-utils-IRVUFB6G.js +30 -0
- package/dist/fs-utils-IRVUFB6G.js.map +1 -0
- package/dist/graph-dashboard-diff.js +1 -0
- package/dist/graph-dashboard-key.js +1 -0
- package/dist/graph-dashboard-parser.js +1 -0
- package/dist/graph-edge-decay-PWB63GRE.js +207 -0
- package/dist/graph-edge-decay-PWB63GRE.js.map +1 -0
- package/dist/graph-edge-reinforcement.d.ts +81 -0
- package/dist/graph-edge-reinforcement.js +24 -0
- package/dist/graph-edge-reinforcement.js.map +1 -0
- package/dist/graph-events.d.ts +87 -0
- package/dist/graph-events.js +14 -0
- package/dist/graph-events.js.map +1 -0
- package/dist/graph-recall.js +1 -0
- package/dist/graph-retrieval.js +1 -0
- package/dist/graph-snapshot.d.ts +112 -0
- package/dist/graph-snapshot.js +19 -0
- package/dist/graph-snapshot.js.map +1 -0
- package/dist/graph.d.ts +105 -7
- package/dist/graph.js +20 -3
- package/dist/harmonic-retrieval.js +1 -0
- package/dist/himem.js +1 -0
- package/dist/hygiene.js +1 -0
- package/dist/identity-continuity.js +1 -0
- package/dist/importance.js +1 -0
- package/dist/index.d.ts +562 -13
- package/dist/index.js +365 -96
- package/dist/index.js.map +1 -1
- package/dist/intent.js +1 -0
- package/dist/json-extract.js +1 -0
- package/dist/json-store.js +1 -0
- package/dist/kdf-7S6RWKLZ.js +26 -0
- package/dist/kdf-7S6RWKLZ.js.map +1 -0
- package/dist/legacy-hook-compat.js +1 -0
- package/dist/legacy-hook-compat.js.map +1 -1
- package/dist/lifecycle.js +1 -0
- package/dist/live-connectors-runner.d.ts +48 -0
- package/dist/live-connectors-runner.js +17 -0
- package/dist/live-connectors-runner.js.map +1 -0
- package/dist/local-llm.js +3 -2
- package/dist/logger.js +1 -0
- package/dist/memory-action-policy.js +1 -0
- package/dist/memory-cache.d.ts +2 -1
- package/dist/memory-cache.js +4 -1
- package/dist/memory-governance-KG52RITE.js +37 -0
- package/dist/memory-governance-KG52RITE.js.map +1 -0
- package/dist/memory-lifecycle-ledger-utils.d.ts +2 -1
- package/dist/memory-lifecycle-ledger-utils.js +4 -1
- package/dist/memory-projection-format.js +1 -0
- package/dist/{memory-projection-store-DeSXPh1j.d.ts → memory-projection-store-D3vBHS4J.d.ts} +1 -0
- package/dist/memory-projection-store.d.ts +1 -1
- package/dist/memory-projection-store.js +1 -0
- package/dist/memory-worth-bench.js +1 -0
- package/dist/memory-worth-bench.js.map +1 -1
- package/dist/memory-worth-filter.js +1 -0
- package/dist/memory-worth-outcomes.d.ts +1 -1
- package/dist/memory-worth-outcomes.js +1 -0
- package/dist/memory-worth.js +1 -0
- package/dist/metadata-FC3XPDRQ.js +21 -0
- package/dist/metadata-FC3XPDRQ.js.map +1 -0
- package/dist/migrate-from-identity-anchor-TTEDEJGX.js +8 -0
- package/dist/migrate-from-identity-anchor-TTEDEJGX.js.map +1 -0
- package/dist/model-registry.js +1 -0
- package/dist/models-json.js +1 -0
- package/dist/native-knowledge.js +1 -0
- package/dist/negative.js +1 -0
- package/dist/objective-state-writers.js +1 -0
- package/dist/objective-state-writers.js.map +1 -1
- package/dist/objective-state.js +1 -0
- package/dist/openai-chat-compat.js +1 -0
- package/dist/operator-toolkit.d.ts +46 -2
- package/dist/operator-toolkit.js +29 -17
- package/dist/opik-exporter.js +1 -0
- package/dist/opik-exporter.js.map +1 -1
- package/dist/{orchestrator-CmJ-NTdJ.d.ts → orchestrator-ChkesB8U.d.ts} +177 -13
- package/dist/orchestrator.d.ts +6 -4
- package/dist/orchestrator.js +58 -42
- package/dist/page-versioning.js +1 -0
- package/dist/path-RMTY5Y5A.js +9 -0
- package/dist/path-RMTY5Y5A.js.map +1 -0
- package/dist/patterns-cli.d.ts +160 -0
- package/dist/patterns-cli.js +29 -0
- package/dist/patterns-cli.js.map +1 -0
- package/dist/peers-6OSQ3NK6.js +44 -0
- package/dist/peers-6OSQ3NK6.js.map +1 -0
- package/dist/plugin-id.js +1 -0
- package/dist/policy-runtime.js +1 -0
- package/dist/{port-BADbLZU5.d.ts → port-hqGnoStS.d.ts} +6 -0
- package/dist/profiling.js +1 -0
- package/dist/purge-6ATBGT77.js +205 -0
- package/dist/purge-6ATBGT77.js.map +1 -0
- package/dist/qmd-recall-cache.d.ts +1 -1
- package/dist/qmd-recall-cache.js +1 -0
- package/dist/qmd.d.ts +2 -1
- package/dist/qmd.js +4 -3
- package/dist/reasoning-trace-recall.js +1 -0
- package/dist/reasoning-trace-types.js +1 -0
- package/dist/recall-audit-anomaly.js +1 -0
- package/dist/recall-audit.js +1 -0
- package/dist/recall-disclosure-escalation.d.ts +84 -0
- package/dist/recall-disclosure-escalation.js +14 -0
- package/dist/recall-disclosure-escalation.js.map +1 -0
- package/dist/recall-explain-renderer.js +4 -1
- package/dist/recall-mmr.js +1 -0
- package/dist/recall-qos.js +1 -0
- package/dist/recall-query-policy.js +1 -0
- package/dist/recall-state.d.ts +7 -0
- package/dist/recall-state.js +2 -1
- package/dist/recall-tag-filter.d.ts +56 -0
- package/dist/recall-tag-filter.js +14 -0
- package/dist/recall-tag-filter.js.map +1 -0
- package/dist/recall-tokenization.js +1 -0
- package/dist/recall-xray-cli.d.ts +9 -2
- package/dist/recall-xray-cli.js +9 -4
- package/dist/recall-xray-renderer.js +4 -1
- package/dist/recall-xray.d.ts +116 -2
- package/dist/recall-xray.js +9 -3
- package/dist/reconstruct.js +1 -0
- package/dist/release-changelog.js +2 -0
- package/dist/release-changelog.js.map +1 -1
- package/dist/relevance.js +1 -0
- package/dist/rerank.js +1 -0
- package/dist/{resolution-QBTDHTG7.js → resolution-YGIBORXI.js} +2 -1
- package/dist/{resolution-QBTDHTG7.js.map → resolution-YGIBORXI.js.map} +1 -1
- package/dist/resolve-auth-token.d.ts +51 -0
- package/dist/resolve-auth-token.js +12 -0
- package/dist/resolve-auth-token.js.map +1 -0
- package/dist/resolve-provider-secret.d.ts +13 -1
- package/dist/resolve-provider-secret.js +6 -1
- package/dist/resume-bundles.js +5 -4
- package/dist/retrieval-agents.d.ts +1 -1
- package/dist/retrieval-agents.js +1 -0
- package/dist/retrieval-tiers.js +1 -0
- package/dist/retrieval.js +1 -0
- package/dist/sanitize.js +1 -0
- package/dist/schemas.d.ts +15 -2
- package/dist/schemas.js +2 -1
- package/dist/sdk-compat.js +1 -0
- package/dist/sdk-compat.js.map +1 -1
- package/dist/secure-store-4R2GSO7S.js +156 -0
- package/dist/secure-store-4R2GSO7S.js.map +1 -0
- package/dist/semantic-chunking.js +1 -0
- package/dist/{semantic-consolidation-CxJU6MJk.d.ts → semantic-consolidation-ByBXb-sf.d.ts} +3 -3
- package/dist/semantic-consolidation.d.ts +2 -2
- package/dist/semantic-consolidation.js +12 -6
- package/dist/semantic-rule-promotion.d.ts +1 -1
- package/dist/semantic-rule-promotion.js +9 -6
- package/dist/semantic-rule-verifier.d.ts +1 -1
- package/dist/semantic-rule-verifier.js +9 -6
- package/dist/session-integrity.js +1 -0
- package/dist/session-observer-bands.js +1 -0
- package/dist/session-observer-state.js +1 -0
- package/dist/session-toggles.js +2 -0
- package/dist/session-toggles.js.map +1 -1
- package/dist/signal.js +1 -0
- package/dist/skills-registry.js +2 -0
- package/dist/skills-registry.js.map +1 -1
- package/dist/source-attribution.js +1 -0
- package/dist/state-NCHQ4TRG.js +8 -0
- package/dist/state-NCHQ4TRG.js.map +1 -0
- package/dist/state-store-3EH7HYIN.js +16 -0
- package/dist/state-store-3EH7HYIN.js.map +1 -0
- package/dist/storage.d.ts +76 -2
- package/dist/storage.js +8 -5
- package/dist/store-contract.js +1 -0
- package/dist/summarizer.js +6 -5
- package/dist/summary-snapshot.js +1 -0
- package/dist/temporal-index.js +1 -0
- package/dist/temporal-supersession.d.ts +1 -1
- package/dist/temporal-supersession.js +2 -1
- package/dist/temporal-validity.d.ts +52 -0
- package/dist/temporal-validity.js +14 -0
- package/dist/temporal-validity.js.map +1 -0
- package/dist/threading.js +1 -0
- package/dist/tier-migration.d.ts +2 -2
- package/dist/tier-migration.js +1 -0
- package/dist/tier-routing.js +1 -0
- package/dist/tier-stats-62ZVDFKS.js +152 -0
- package/dist/tier-stats-62ZVDFKS.js.map +1 -0
- package/dist/tmt.js +1 -0
- package/dist/tokens.js +3 -1
- package/dist/topics.js +1 -0
- package/dist/trace-C5ETWBEF.js +290 -0
- package/dist/trace-C5ETWBEF.js.map +1 -0
- package/dist/transcript.js +1 -0
- package/dist/trust-zones.js +1 -0
- package/dist/tui-RI7P6PBS.js +13 -0
- package/dist/tui-RI7P6PBS.js.map +1 -0
- package/dist/types-V3FJ26TF.js +30 -0
- package/dist/types-V3FJ26TF.js.map +1 -0
- package/dist/types.d.ts +634 -9
- package/dist/types.js +10 -3
- package/dist/utility-learner.js +1 -0
- package/dist/utility-runtime.js +1 -0
- package/dist/utility-telemetry.js +1 -0
- package/dist/verified-recall.js +9 -6
- package/dist/version-utils.js +1 -0
- package/dist/whitespace.js +1 -0
- package/dist/work-product-ledger.js +1 -0
- package/package.json +2 -1
- package/dist/access-service-Br8ZydTK.d.ts +0 -827
- package/dist/chunk-3OGMS3PE.js.map +0 -1
- package/dist/chunk-6PFRXT4K.js.map +0 -1
- package/dist/chunk-ALXMCZEU.js.map +0 -1
- package/dist/chunk-B5WXLVDY.js.map +0 -1
- package/dist/chunk-BGJGXLZ7.js.map +0 -1
- package/dist/chunk-BK2EFTE2.js.map +0 -1
- package/dist/chunk-C2EFFULQ.js.map +0 -1
- package/dist/chunk-CUPFXL3J.js.map +0 -1
- package/dist/chunk-DFTTJYSO.js.map +0 -1
- package/dist/chunk-EPQJM2GC.js.map +0 -1
- package/dist/chunk-F5VP6YCB.js.map +0 -1
- package/dist/chunk-FVA6TGI3.js.map +0 -1
- package/dist/chunk-GKFXUTJ2.js.map +0 -1
- package/dist/chunk-HK3FGIEW.js.map +0 -1
- package/dist/chunk-INXV5JBT.js.map +0 -1
- package/dist/chunk-KVBLZUKV.js.map +0 -1
- package/dist/chunk-LK6SGL53.js.map +0 -1
- package/dist/chunk-LTCGGW2D.js +0 -14
- package/dist/chunk-LTCGGW2D.js.map +0 -1
- package/dist/chunk-NZLQTHS5.js.map +0 -1
- package/dist/chunk-O5ETUNBT.js.map +0 -1
- package/dist/chunk-PVPWZSSI.js.map +0 -1
- package/dist/chunk-RGLL5SPU.js.map +0 -1
- package/dist/chunk-S3EEFKNY.js.map +0 -1
- package/dist/chunk-SPI27QT6.js.map +0 -1
- package/dist/chunk-TP4FZJIZ.js.map +0 -1
- package/dist/chunk-ULYOGL6R.js.map +0 -1
- package/dist/chunk-VBVG2M5G.js.map +0 -1
- package/dist/chunk-VDX363PS.js.map +0 -1
- package/dist/chunk-VYM3VWOF.js.map +0 -1
- package/dist/chunk-WCLICCGB.js.map +0 -1
- package/dist/chunk-WVVA7F5A.js.map +0 -1
- package/dist/chunk-X6GF3FX2.js +0 -26
- package/dist/chunk-X6GF3FX2.js.map +0 -1
- package/dist/chunk-XZ2TIKGC.js.map +0 -1
- package/dist/chunk-ZAIM4TUE.js.map +0 -1
- /package/dist/{contradiction-review-WIUBAR52.js.map → capsule-cli.js.map} +0 -0
- /package/dist/{engine-F3GOXGE5.js.map → capsule-crypto-5CYAGVC5.js.map} +0 -0
- /package/dist/{chunk-KUB6JU6H.js.map → chunk-47WOM4YW.js.map} +0 -0
- /package/dist/{chunk-ZEM3OK2K.js.map → chunk-B2TL6GA2.js.map} +0 -0
- /package/dist/{chunk-3GXCSUXR.js.map → chunk-CRU27Q4J.js.map} +0 -0
- /package/dist/{chunk-RBBWYEFJ.js.map → chunk-G2WADRQ3.js.map} +0 -0
- /package/dist/{chunk-7I7FKFZH.js.map → chunk-R2L7SUX2.js.map} +0 -0
- /package/dist/{chunk-JL2PU6AI.js.map → chunk-R2XRID2N.js.map} +0 -0
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/objective-state-writers.ts"],"sourcesContent":["import crypto from \"node:crypto\";\nimport {\n type ObjectiveStateChangeKind,\n type ObjectiveStateOutcome,\n type ObjectiveStateSnapshot,\n recordObjectiveStateSnapshot,\n} from \"./objective-state.js\";\n\ninterface ToolCallContext {\n toolName?: string;\n toolCallId?: string;\n args?: Record<string, unknown>;\n}\n\ninterface DerivedObjectiveStateResult {\n snapshots: ObjectiveStateSnapshot[];\n filePaths: string[];\n}\n\nfunction hashSha256(value: string): string {\n return crypto.createHash(\"sha256\").update(value).digest(\"hex\");\n}\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction optionalString(value: unknown): string | undefined {\n if (typeof value !== \"string\") return undefined;\n const trimmed = value.trim();\n return trimmed.length > 0 ? trimmed : undefined;\n}\n\nfunction toolNameTokens(toolName: string | undefined): string[] {\n if (!toolName) return [];\n return toolName\n .replace(/([a-z0-9])([A-Z])/g, \"$1_$2\")\n .toLowerCase()\n .split(/[^a-z0-9]+/)\n .filter((token) => token.length > 0);\n}\n\nfunction normalizedToolName(toolName: string | undefined): string {\n return toolNameTokens(toolName).join(\"_\");\n}\n\nfunction parseToolArguments(value: unknown): Record<string, unknown> | undefined {\n if (isRecord(value)) return value;\n if (typeof value !== \"string\") return undefined;\n try {\n const parsed = JSON.parse(value) as unknown;\n return isRecord(parsed) ? parsed : undefined;\n } catch {\n return undefined;\n }\n}\n\nfunction extractTextContent(value: unknown): string {\n if (typeof value === \"string\") return value.trim();\n if (Array.isArray(value)) {\n return value\n .map((block) => {\n if (typeof block === \"string\") return block.trim();\n if (isRecord(block) && block.type === \"text\" && typeof block.text === \"string\") {\n return block.text.trim();\n }\n return \"\";\n })\n .filter((item) => item.length > 0)\n .join(\"\\n\");\n }\n if (isRecord(value)) {\n return JSON.stringify(value);\n }\n return \"\";\n}\n\nfunction parseToolResultPayload(content: unknown): unknown {\n const text = extractTextContent(content);\n if (text.length === 0) return undefined;\n try {\n return JSON.parse(text) as unknown;\n } catch {\n return text;\n }\n}\n\nfunction resultHash(value: unknown): string | undefined {\n if (value === undefined) return undefined;\n const canonical =\n typeof value === \"string\" ? value : JSON.stringify(value);\n if (!canonical || canonical.length === 0) return undefined;\n return `sha256:${hashSha256(canonical)}`;\n}\n\nfunction getToolCallContexts(messages: Array<Record<string, unknown>>): Map<string, ToolCallContext> {\n const contexts = new Map<string, ToolCallContext>();\n for (const message of messages) {\n if (message.role !== \"assistant\") continue;\n const toolCalls = message.tool_calls ?? message.toolCalls;\n if (!Array.isArray(toolCalls)) continue;\n for (const call of toolCalls) {\n if (!isRecord(call)) continue;\n const toolCallId = optionalString(call.id) ?? optionalString(call.toolCallId);\n if (!toolCallId) continue;\n const fn = isRecord(call.function) ? call.function : undefined;\n const toolName =\n optionalString(fn?.name) ??\n optionalString(call.name);\n const args =\n parseToolArguments(fn?.arguments) ??\n parseToolArguments(call.arguments) ??\n parseToolArguments(call.args) ??\n parseToolArguments(call.input);\n contexts.set(toolCallId, { toolCallId, toolName, args });\n }\n }\n return contexts;\n}\n\nfunction toolCallIdForMessage(message: Record<string, unknown>): string | undefined {\n return (\n optionalString(message.tool_call_id) ??\n optionalString(message.toolCallId) ??\n optionalString(message.tool_use_id) ??\n optionalString(message.toolUseId)\n );\n}\n\nfunction toolNameForMessage(message: Record<string, unknown>, context?: ToolCallContext): string | undefined {\n return (\n optionalString(message.name) ??\n optionalString(message.toolName) ??\n optionalString(message.tool) ??\n context?.toolName\n );\n}\n\nfunction pickString(args: Record<string, unknown> | undefined, keys: string[]): string | undefined {\n if (!args) return undefined;\n for (const key of keys) {\n const value = optionalString(args[key]);\n if (value) return value;\n }\n return undefined;\n}\n\nfunction pickFirstStringArrayValue(args: Record<string, unknown> | undefined, key: string): string | undefined {\n const value = args?.[key];\n if (!Array.isArray(value)) return undefined;\n for (const item of value) {\n const candidate = optionalString(item);\n if (candidate) return candidate;\n }\n return undefined;\n}\n\nfunction fileScopeFromArgs(args: Record<string, unknown> | undefined): {\n scope?: string;\n sourcePath?: string;\n destinationPath?: string;\n} {\n const destinationPath =\n pickString(args, [\"destination\", \"dest\", \"targetPath\", \"target\", \"to\"]) ??\n pickString(args, [\"path\", \"filePath\", \"workspacePath\", \"projectPath\"]) ??\n pickFirstStringArrayValue(args, \"paths\");\n const sourcePath =\n pickString(args, [\"source\", \"src\", \"from\", \"oldPath\"]);\n const scope = destinationPath ?? sourcePath;\n return { scope, sourcePath, destinationPath };\n}\n\nfunction fileContentHash(args: Record<string, unknown> | undefined): string | undefined {\n const content =\n pickString(args, [\"content\", \"patch\", \"diff\", \"text\", \"value\"]) ??\n args?.updates;\n return resultHash(content);\n}\n\nfunction inferOutcome(message: Record<string, unknown>, parsedPayload: unknown): ObjectiveStateOutcome {\n if (message.isError === true) return \"failure\";\n if (isRecord(parsedPayload)) {\n if (parsedPayload.partial === true || parsedPayload.status === \"partial\") return \"partial\";\n if (parsedPayload.success === false || parsedPayload.ok === false) return \"failure\";\n if (parsedPayload.success === true || parsedPayload.ok === true) return \"success\";\n if (typeof parsedPayload.exitCode === \"number\") {\n return parsedPayload.exitCode === 0 ? \"success\" : \"failure\";\n }\n if (optionalString(parsedPayload.error)) return \"failure\";\n if (parsedPayload.status === \"error\" || parsedPayload.status === \"failed\") return \"failure\";\n if (parsedPayload.status === \"ok\" || parsedPayload.status === \"success\") return \"success\";\n }\n if (typeof parsedPayload === \"string\") {\n const lowered = parsedPayload.toLowerCase();\n const loweredForFailure = lowered\n .replace(/\\b(?:previously\\s+)?failed tests?\\s+now\\s+pass(?:ed|es)?\\b/g, \"\");\n const hasZeroCountMarker = /\\b(?:0|no)\\s+(?:errors?|failures?|exceptions?|timeouts?)\\b/.test(lowered);\n const loweredForFailureCounts = loweredForFailure\n .replace(/\\b(?:0|no)\\s+errors?\\b/g, \"\")\n .replace(/\\b(?:0|no)\\s+failures?\\b/g, \"\")\n .replace(/\\b(?:0|no)\\s+exceptions?\\b/g, \"\")\n .replace(/\\b(?:0|no)\\s+timeouts?\\b/g, \"\");\n const hasNonZeroErrorCounts = /\\b[1-9]\\d*\\s+errors?\\b/.test(loweredForFailureCounts);\n const hasNegatedSuccessMarkers =\n /\\b(?:not|did not|didn't|doesn't|isn't|aren't|wasn't|weren't|won't|can't|couldn't|shouldn't|wouldn't)\\s+(?:ok|pass|passed|passes|succeeded|success)\\b/.test(loweredForFailure);\n const hasSuccessMarkers =\n /\\b(success|succeeded|pass|passes|passed|ok)\\b/.test(lowered) ||\n hasZeroCountMarker;\n const hasFailureMarkers =\n hasNegatedSuccessMarkers ||\n /\\b(exceptions?|failed|failures?|fatal|timeouts?|timed out)\\b/.test(loweredForFailureCounts) ||\n hasNonZeroErrorCounts ||\n /\\berrors?\\b/.test(loweredForFailureCounts) ||\n /\\b[a-z]+error\\b/.test(loweredForFailureCounts) ||\n /\\b[a-z]+exception\\b/.test(loweredForFailureCounts);\n\n if (hasFailureMarkers) return \"failure\";\n if (hasSuccessMarkers) return \"success\";\n }\n return \"unknown\";\n}\n\nfunction isProcessTool(toolName: string | undefined, args: Record<string, unknown> | undefined): boolean {\n const tokens = toolNameTokens(toolName);\n const normalizedName = normalizedToolName(toolName);\n if (pickString(args, [\"cmd\", \"command\", \"script\"])) return true;\n return [\"exec\", \"shell\", \"bash\", \"terminal\", \"run_command\", \"exec_command\"].some((token) =>\n token.includes(\"_\") ? normalizedName === token : tokens.includes(token),\n );\n}\n\nfunction isFileTool(toolName: string | undefined, args: Record<string, unknown> | undefined): boolean {\n const tokens = toolNameTokens(toolName);\n const fileScope = fileScopeFromArgs(args);\n if (fileScope.scope) return true;\n return [\"file\", \"path\", \"patch\", \"directory\", \"mkdir\", \"rename\", \"move\"].some((token) =>\n tokens.includes(token),\n );\n}\n\nfunction inferFileChangeKind(toolName: string | undefined, outcome: ObjectiveStateOutcome): ObjectiveStateChangeKind {\n if (outcome === \"failure\") return \"failed\";\n const tokens = toolNameTokens(toolName);\n if ([\"delete\", \"remove\", \"unlink\"].some((token) => tokens.includes(token))) return \"deleted\";\n if ([\"create\", \"mkdir\", \"new\"].some((token) => tokens.includes(token))) return \"created\";\n if ([\"write\", \"edit\", \"patch\", \"update\", \"append\", \"move\", \"rename\"].some((token) => tokens.includes(token))) {\n return \"updated\";\n }\n return \"observed\";\n}\n\nfunction buildFileValueRefs(\n args: Record<string, unknown> | undefined,\n changeKind: ObjectiveStateChangeKind,\n): Pick<ObjectiveStateSnapshot, \"before\" | \"after\"> {\n const { sourcePath, destinationPath, scope } = fileScopeFromArgs(args);\n const contentHash = fileContentHash(args);\n\n if (changeKind === \"failed\") {\n if (sourcePath && destinationPath && sourcePath !== destinationPath) {\n return {\n before: { ref: sourcePath },\n after: { ref: destinationPath },\n };\n }\n return {\n before: sourcePath ? { ref: sourcePath } : undefined,\n after: scope ? { ref: scope } : undefined,\n };\n }\n\n if (changeKind === \"deleted\") {\n return {\n before: scope ? { exists: true, ref: scope } : undefined,\n after: { exists: false },\n };\n }\n\n if (changeKind === \"created\") {\n return {\n after: {\n exists: true,\n ref: scope,\n valueHash: contentHash,\n },\n };\n }\n\n if (sourcePath && destinationPath && sourcePath !== destinationPath) {\n return {\n before: { exists: true, ref: sourcePath },\n after: {\n exists: true,\n ref: destinationPath,\n },\n };\n }\n\n return {\n after: {\n exists: true,\n ref: scope,\n valueHash: contentHash,\n },\n };\n}\n\nfunction summarizeSnapshot(\n kind: ObjectiveStateSnapshot[\"kind\"],\n changeKind: ObjectiveStateChangeKind,\n toolName: string,\n scope: string,\n): string {\n const action =\n changeKind === \"executed\"\n ? \"Executed\"\n : changeKind === \"failed\"\n ? \"Failed\"\n : changeKind === \"created\"\n ? \"Created\"\n : changeKind === \"deleted\"\n ? \"Deleted\"\n : changeKind === \"updated\"\n ? \"Updated\"\n : \"Observed\";\n if (kind === \"process\") return `${action} process via ${toolName}: ${scope}`;\n if (kind === \"file\") return `${action} file via ${toolName}: ${scope}`;\n return `${action} tool result from ${toolName}: ${scope}`;\n}\n\nfunction buildGenericToolAfterRef(outcome: ObjectiveStateOutcome, parsedPayload: unknown): ObjectiveStateSnapshot[\"after\"] {\n const valueHash = resultHash(parsedPayload);\n return valueHash ? { valueHash } : { exists: outcome !== \"failure\" };\n}\n\nfunction snapshotIdFor(\n sessionKey: string,\n recordedAt: string,\n index: number,\n toolName: string,\n scope: string,\n): string {\n const digest = crypto\n .createHash(\"sha256\")\n .update(`${sessionKey}|${recordedAt}|${index}|${toolName}|${scope}`)\n .digest(\"hex\")\n .slice(0, 12);\n return `obj-${digest}`;\n}\n\nexport function deriveObjectiveStateSnapshotsFromAgentMessages(options: {\n sessionKey: string;\n recordedAt: string;\n messages: Array<Record<string, unknown>>;\n}): ObjectiveStateSnapshot[] {\n const toolCallsById = getToolCallContexts(options.messages);\n const snapshots: ObjectiveStateSnapshot[] = [];\n\n for (const message of options.messages) {\n if (message.role !== \"tool\") continue;\n const toolCallId = toolCallIdForMessage(message);\n const context = toolCallId ? toolCallsById.get(toolCallId) : undefined;\n const toolName = toolNameForMessage(message, context);\n if (!toolName) continue;\n\n const parsedPayload = parseToolResultPayload(message.content);\n const outcome = inferOutcome(message, parsedPayload);\n const args = context?.args;\n const command = pickString(args, [\"cmd\", \"command\", \"script\"]);\n\n let kind: ObjectiveStateSnapshot[\"kind\"] = \"tool\";\n let changeKind: ObjectiveStateChangeKind = outcome === \"failure\" ? \"failed\" : \"observed\";\n let scope = toolName;\n let before: ObjectiveStateSnapshot[\"before\"];\n let after: ObjectiveStateSnapshot[\"after\"];\n\n if (isProcessTool(toolName, args)) {\n kind = \"process\";\n changeKind = outcome === \"failure\" ? \"failed\" : \"executed\";\n scope = command ?? toolName;\n after = { exists: outcome !== \"failure\", valueHash: resultHash(parsedPayload) };\n } else if (isFileTool(toolName, args)) {\n kind = \"file\";\n changeKind = inferFileChangeKind(toolName, outcome);\n const fileScope = fileScopeFromArgs(args);\n scope = fileScope.scope ?? toolName;\n const refs = buildFileValueRefs(args, changeKind);\n before = refs.before;\n after = refs.after;\n } else {\n after = buildGenericToolAfterRef(outcome, parsedPayload);\n }\n\n snapshots.push({\n schemaVersion: 1,\n snapshotId: snapshotIdFor(options.sessionKey, options.recordedAt, snapshots.length, toolName, scope),\n recordedAt: options.recordedAt,\n sessionKey: options.sessionKey,\n source: \"tool_result\",\n kind,\n changeKind,\n scope,\n summary: summarizeSnapshot(kind, changeKind, toolName, scope),\n toolName,\n command,\n outcome,\n before,\n after,\n tags: [\"agent-end\", `tool:${toolName}`],\n metadata: toolCallId ? { toolCallId } : undefined,\n });\n }\n\n return snapshots;\n}\n\nexport async function recordObjectiveStateSnapshotsFromAgentMessages(options: {\n memoryDir: string;\n objectiveStateStoreDir?: string;\n objectiveStateMemoryEnabled: boolean;\n objectiveStateSnapshotWritesEnabled: boolean;\n sessionKey: string;\n recordedAt: string;\n messages: Array<Record<string, unknown>>;\n}): Promise<DerivedObjectiveStateResult> {\n if (!options.objectiveStateMemoryEnabled || !options.objectiveStateSnapshotWritesEnabled) {\n return { snapshots: [], filePaths: [] };\n }\n\n const snapshots = deriveObjectiveStateSnapshotsFromAgentMessages({\n sessionKey: options.sessionKey,\n recordedAt: options.recordedAt,\n messages: options.messages,\n });\n\n const filePaths: string[] = [];\n for (const snapshot of snapshots) {\n filePaths.push(\n await recordObjectiveStateSnapshot({\n memoryDir: options.memoryDir,\n objectiveStateStoreDir: options.objectiveStateStoreDir,\n snapshot,\n }),\n );\n }\n\n return { snapshots, filePaths };\n}\n"],"mappings":"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","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/objective-state-writers.ts"],"sourcesContent":["import crypto from \"node:crypto\";\nimport {\n type ObjectiveStateChangeKind,\n type ObjectiveStateOutcome,\n type ObjectiveStateSnapshot,\n recordObjectiveStateSnapshot,\n} from \"./objective-state.js\";\n\ninterface ToolCallContext {\n toolName?: string;\n toolCallId?: string;\n args?: Record<string, unknown>;\n}\n\ninterface DerivedObjectiveStateResult {\n snapshots: ObjectiveStateSnapshot[];\n filePaths: string[];\n}\n\nfunction hashSha256(value: string): string {\n return crypto.createHash(\"sha256\").update(value).digest(\"hex\");\n}\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction optionalString(value: unknown): string | undefined {\n if (typeof value !== \"string\") return undefined;\n const trimmed = value.trim();\n return trimmed.length > 0 ? trimmed : undefined;\n}\n\nfunction toolNameTokens(toolName: string | undefined): string[] {\n if (!toolName) return [];\n return toolName\n .replace(/([a-z0-9])([A-Z])/g, \"$1_$2\")\n .toLowerCase()\n .split(/[^a-z0-9]+/)\n .filter((token) => token.length > 0);\n}\n\nfunction normalizedToolName(toolName: string | undefined): string {\n return toolNameTokens(toolName).join(\"_\");\n}\n\nfunction parseToolArguments(value: unknown): Record<string, unknown> | undefined {\n if (isRecord(value)) return value;\n if (typeof value !== \"string\") return undefined;\n try {\n const parsed = JSON.parse(value) as unknown;\n return isRecord(parsed) ? parsed : undefined;\n } catch {\n return undefined;\n }\n}\n\nfunction extractTextContent(value: unknown): string {\n if (typeof value === \"string\") return value.trim();\n if (Array.isArray(value)) {\n return value\n .map((block) => {\n if (typeof block === \"string\") return block.trim();\n if (isRecord(block) && block.type === \"text\" && typeof block.text === \"string\") {\n return block.text.trim();\n }\n return \"\";\n })\n .filter((item) => item.length > 0)\n .join(\"\\n\");\n }\n if (isRecord(value)) {\n return JSON.stringify(value);\n }\n return \"\";\n}\n\nfunction parseToolResultPayload(content: unknown): unknown {\n const text = extractTextContent(content);\n if (text.length === 0) return undefined;\n try {\n return JSON.parse(text) as unknown;\n } catch {\n return text;\n }\n}\n\nfunction resultHash(value: unknown): string | undefined {\n if (value === undefined) return undefined;\n const canonical =\n typeof value === \"string\" ? value : JSON.stringify(value);\n if (!canonical || canonical.length === 0) return undefined;\n return `sha256:${hashSha256(canonical)}`;\n}\n\nfunction getToolCallContexts(messages: Array<Record<string, unknown>>): Map<string, ToolCallContext> {\n const contexts = new Map<string, ToolCallContext>();\n for (const message of messages) {\n if (message.role !== \"assistant\") continue;\n const toolCalls = message.tool_calls ?? message.toolCalls;\n if (!Array.isArray(toolCalls)) continue;\n for (const call of toolCalls) {\n if (!isRecord(call)) continue;\n const toolCallId = optionalString(call.id) ?? optionalString(call.toolCallId);\n if (!toolCallId) continue;\n const fn = isRecord(call.function) ? call.function : undefined;\n const toolName =\n optionalString(fn?.name) ??\n optionalString(call.name);\n const args =\n parseToolArguments(fn?.arguments) ??\n parseToolArguments(call.arguments) ??\n parseToolArguments(call.args) ??\n parseToolArguments(call.input);\n contexts.set(toolCallId, { toolCallId, toolName, args });\n }\n }\n return contexts;\n}\n\nfunction toolCallIdForMessage(message: Record<string, unknown>): string | undefined {\n return (\n optionalString(message.tool_call_id) ??\n optionalString(message.toolCallId) ??\n optionalString(message.tool_use_id) ??\n optionalString(message.toolUseId)\n );\n}\n\nfunction toolNameForMessage(message: Record<string, unknown>, context?: ToolCallContext): string | undefined {\n return (\n optionalString(message.name) ??\n optionalString(message.toolName) ??\n optionalString(message.tool) ??\n context?.toolName\n );\n}\n\nfunction pickString(args: Record<string, unknown> | undefined, keys: string[]): string | undefined {\n if (!args) return undefined;\n for (const key of keys) {\n const value = optionalString(args[key]);\n if (value) return value;\n }\n return undefined;\n}\n\nfunction pickFirstStringArrayValue(args: Record<string, unknown> | undefined, key: string): string | undefined {\n const value = args?.[key];\n if (!Array.isArray(value)) return undefined;\n for (const item of value) {\n const candidate = optionalString(item);\n if (candidate) return candidate;\n }\n return undefined;\n}\n\nfunction fileScopeFromArgs(args: Record<string, unknown> | undefined): {\n scope?: string;\n sourcePath?: string;\n destinationPath?: string;\n} {\n const destinationPath =\n pickString(args, [\"destination\", \"dest\", \"targetPath\", \"target\", \"to\"]) ??\n pickString(args, [\"path\", \"filePath\", \"workspacePath\", \"projectPath\"]) ??\n pickFirstStringArrayValue(args, \"paths\");\n const sourcePath =\n pickString(args, [\"source\", \"src\", \"from\", \"oldPath\"]);\n const scope = destinationPath ?? sourcePath;\n return { scope, sourcePath, destinationPath };\n}\n\nfunction fileContentHash(args: Record<string, unknown> | undefined): string | undefined {\n const content =\n pickString(args, [\"content\", \"patch\", \"diff\", \"text\", \"value\"]) ??\n args?.updates;\n return resultHash(content);\n}\n\nfunction inferOutcome(message: Record<string, unknown>, parsedPayload: unknown): ObjectiveStateOutcome {\n if (message.isError === true) return \"failure\";\n if (isRecord(parsedPayload)) {\n if (parsedPayload.partial === true || parsedPayload.status === \"partial\") return \"partial\";\n if (parsedPayload.success === false || parsedPayload.ok === false) return \"failure\";\n if (parsedPayload.success === true || parsedPayload.ok === true) return \"success\";\n if (typeof parsedPayload.exitCode === \"number\") {\n return parsedPayload.exitCode === 0 ? \"success\" : \"failure\";\n }\n if (optionalString(parsedPayload.error)) return \"failure\";\n if (parsedPayload.status === \"error\" || parsedPayload.status === \"failed\") return \"failure\";\n if (parsedPayload.status === \"ok\" || parsedPayload.status === \"success\") return \"success\";\n }\n if (typeof parsedPayload === \"string\") {\n const lowered = parsedPayload.toLowerCase();\n const loweredForFailure = lowered\n .replace(/\\b(?:previously\\s+)?failed tests?\\s+now\\s+pass(?:ed|es)?\\b/g, \"\");\n const hasZeroCountMarker = /\\b(?:0|no)\\s+(?:errors?|failures?|exceptions?|timeouts?)\\b/.test(lowered);\n const loweredForFailureCounts = loweredForFailure\n .replace(/\\b(?:0|no)\\s+errors?\\b/g, \"\")\n .replace(/\\b(?:0|no)\\s+failures?\\b/g, \"\")\n .replace(/\\b(?:0|no)\\s+exceptions?\\b/g, \"\")\n .replace(/\\b(?:0|no)\\s+timeouts?\\b/g, \"\");\n const hasNonZeroErrorCounts = /\\b[1-9]\\d*\\s+errors?\\b/.test(loweredForFailureCounts);\n const hasNegatedSuccessMarkers =\n /\\b(?:not|did not|didn't|doesn't|isn't|aren't|wasn't|weren't|won't|can't|couldn't|shouldn't|wouldn't)\\s+(?:ok|pass|passed|passes|succeeded|success)\\b/.test(loweredForFailure);\n const hasSuccessMarkers =\n /\\b(success|succeeded|pass|passes|passed|ok)\\b/.test(lowered) ||\n hasZeroCountMarker;\n const hasFailureMarkers =\n hasNegatedSuccessMarkers ||\n /\\b(exceptions?|failed|failures?|fatal|timeouts?|timed out)\\b/.test(loweredForFailureCounts) ||\n hasNonZeroErrorCounts ||\n /\\berrors?\\b/.test(loweredForFailureCounts) ||\n /\\b[a-z]+error\\b/.test(loweredForFailureCounts) ||\n /\\b[a-z]+exception\\b/.test(loweredForFailureCounts);\n\n if (hasFailureMarkers) return \"failure\";\n if (hasSuccessMarkers) return \"success\";\n }\n return \"unknown\";\n}\n\nfunction isProcessTool(toolName: string | undefined, args: Record<string, unknown> | undefined): boolean {\n const tokens = toolNameTokens(toolName);\n const normalizedName = normalizedToolName(toolName);\n if (pickString(args, [\"cmd\", \"command\", \"script\"])) return true;\n return [\"exec\", \"shell\", \"bash\", \"terminal\", \"run_command\", \"exec_command\"].some((token) =>\n token.includes(\"_\") ? normalizedName === token : tokens.includes(token),\n );\n}\n\nfunction isFileTool(toolName: string | undefined, args: Record<string, unknown> | undefined): boolean {\n const tokens = toolNameTokens(toolName);\n const fileScope = fileScopeFromArgs(args);\n if (fileScope.scope) return true;\n return [\"file\", \"path\", \"patch\", \"directory\", \"mkdir\", \"rename\", \"move\"].some((token) =>\n tokens.includes(token),\n );\n}\n\nfunction inferFileChangeKind(toolName: string | undefined, outcome: ObjectiveStateOutcome): ObjectiveStateChangeKind {\n if (outcome === \"failure\") return \"failed\";\n const tokens = toolNameTokens(toolName);\n if ([\"delete\", \"remove\", \"unlink\"].some((token) => tokens.includes(token))) return \"deleted\";\n if ([\"create\", \"mkdir\", \"new\"].some((token) => tokens.includes(token))) return \"created\";\n if ([\"write\", \"edit\", \"patch\", \"update\", \"append\", \"move\", \"rename\"].some((token) => tokens.includes(token))) {\n return \"updated\";\n }\n return \"observed\";\n}\n\nfunction buildFileValueRefs(\n args: Record<string, unknown> | undefined,\n changeKind: ObjectiveStateChangeKind,\n): Pick<ObjectiveStateSnapshot, \"before\" | \"after\"> {\n const { sourcePath, destinationPath, scope } = fileScopeFromArgs(args);\n const contentHash = fileContentHash(args);\n\n if (changeKind === \"failed\") {\n if (sourcePath && destinationPath && sourcePath !== destinationPath) {\n return {\n before: { ref: sourcePath },\n after: { ref: destinationPath },\n };\n }\n return {\n before: sourcePath ? { ref: sourcePath } : undefined,\n after: scope ? { ref: scope } : undefined,\n };\n }\n\n if (changeKind === \"deleted\") {\n return {\n before: scope ? { exists: true, ref: scope } : undefined,\n after: { exists: false },\n };\n }\n\n if (changeKind === \"created\") {\n return {\n after: {\n exists: true,\n ref: scope,\n valueHash: contentHash,\n },\n };\n }\n\n if (sourcePath && destinationPath && sourcePath !== destinationPath) {\n return {\n before: { exists: true, ref: sourcePath },\n after: {\n exists: true,\n ref: destinationPath,\n },\n };\n }\n\n return {\n after: {\n exists: true,\n ref: scope,\n valueHash: contentHash,\n },\n };\n}\n\nfunction summarizeSnapshot(\n kind: ObjectiveStateSnapshot[\"kind\"],\n changeKind: ObjectiveStateChangeKind,\n toolName: string,\n scope: string,\n): string {\n const action =\n changeKind === \"executed\"\n ? \"Executed\"\n : changeKind === \"failed\"\n ? \"Failed\"\n : changeKind === \"created\"\n ? \"Created\"\n : changeKind === \"deleted\"\n ? \"Deleted\"\n : changeKind === \"updated\"\n ? \"Updated\"\n : \"Observed\";\n if (kind === \"process\") return `${action} process via ${toolName}: ${scope}`;\n if (kind === \"file\") return `${action} file via ${toolName}: ${scope}`;\n return `${action} tool result from ${toolName}: ${scope}`;\n}\n\nfunction buildGenericToolAfterRef(outcome: ObjectiveStateOutcome, parsedPayload: unknown): ObjectiveStateSnapshot[\"after\"] {\n const valueHash = resultHash(parsedPayload);\n return valueHash ? { valueHash } : { exists: outcome !== \"failure\" };\n}\n\nfunction snapshotIdFor(\n sessionKey: string,\n recordedAt: string,\n index: number,\n toolName: string,\n scope: string,\n): string {\n const digest = crypto\n .createHash(\"sha256\")\n .update(`${sessionKey}|${recordedAt}|${index}|${toolName}|${scope}`)\n .digest(\"hex\")\n .slice(0, 12);\n return `obj-${digest}`;\n}\n\nexport function deriveObjectiveStateSnapshotsFromAgentMessages(options: {\n sessionKey: string;\n recordedAt: string;\n messages: Array<Record<string, unknown>>;\n}): ObjectiveStateSnapshot[] {\n const toolCallsById = getToolCallContexts(options.messages);\n const snapshots: ObjectiveStateSnapshot[] = [];\n\n for (const message of options.messages) {\n if (message.role !== \"tool\") continue;\n const toolCallId = toolCallIdForMessage(message);\n const context = toolCallId ? toolCallsById.get(toolCallId) : undefined;\n const toolName = toolNameForMessage(message, context);\n if (!toolName) continue;\n\n const parsedPayload = parseToolResultPayload(message.content);\n const outcome = inferOutcome(message, parsedPayload);\n const args = context?.args;\n const command = pickString(args, [\"cmd\", \"command\", \"script\"]);\n\n let kind: ObjectiveStateSnapshot[\"kind\"] = \"tool\";\n let changeKind: ObjectiveStateChangeKind = outcome === \"failure\" ? \"failed\" : \"observed\";\n let scope = toolName;\n let before: ObjectiveStateSnapshot[\"before\"];\n let after: ObjectiveStateSnapshot[\"after\"];\n\n if (isProcessTool(toolName, args)) {\n kind = \"process\";\n changeKind = outcome === \"failure\" ? \"failed\" : \"executed\";\n scope = command ?? toolName;\n after = { exists: outcome !== \"failure\", valueHash: resultHash(parsedPayload) };\n } else if (isFileTool(toolName, args)) {\n kind = \"file\";\n changeKind = inferFileChangeKind(toolName, outcome);\n const fileScope = fileScopeFromArgs(args);\n scope = fileScope.scope ?? toolName;\n const refs = buildFileValueRefs(args, changeKind);\n before = refs.before;\n after = refs.after;\n } else {\n after = buildGenericToolAfterRef(outcome, parsedPayload);\n }\n\n snapshots.push({\n schemaVersion: 1,\n snapshotId: snapshotIdFor(options.sessionKey, options.recordedAt, snapshots.length, toolName, scope),\n recordedAt: options.recordedAt,\n sessionKey: options.sessionKey,\n source: \"tool_result\",\n kind,\n changeKind,\n scope,\n summary: summarizeSnapshot(kind, changeKind, toolName, scope),\n toolName,\n command,\n outcome,\n before,\n after,\n tags: [\"agent-end\", `tool:${toolName}`],\n metadata: toolCallId ? { toolCallId } : undefined,\n });\n }\n\n return snapshots;\n}\n\nexport async function recordObjectiveStateSnapshotsFromAgentMessages(options: {\n memoryDir: string;\n objectiveStateStoreDir?: string;\n objectiveStateMemoryEnabled: boolean;\n objectiveStateSnapshotWritesEnabled: boolean;\n sessionKey: string;\n recordedAt: string;\n messages: Array<Record<string, unknown>>;\n}): Promise<DerivedObjectiveStateResult> {\n if (!options.objectiveStateMemoryEnabled || !options.objectiveStateSnapshotWritesEnabled) {\n return { snapshots: [], filePaths: [] };\n }\n\n const snapshots = deriveObjectiveStateSnapshotsFromAgentMessages({\n sessionKey: options.sessionKey,\n recordedAt: options.recordedAt,\n messages: options.messages,\n });\n\n const filePaths: string[] = [];\n for (const snapshot of snapshots) {\n filePaths.push(\n await recordObjectiveStateSnapshot({\n memoryDir: options.memoryDir,\n objectiveStateStoreDir: options.objectiveStateStoreDir,\n snapshot,\n }),\n );\n }\n\n return { snapshots, filePaths };\n}\n"],"mappings":"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","names":[]}
|
package/dist/objective-state.js
CHANGED
|
@@ -5,7 +5,7 @@ import { SessionIntegrityReport, SessionRepairPlan, SessionRepairApplyResult } f
|
|
|
5
5
|
import { PluginConfig } from './types.js';
|
|
6
6
|
import './page-versioning.js';
|
|
7
7
|
import './consolidation-operator.js';
|
|
8
|
-
import './memory-projection-store-
|
|
8
|
+
import './memory-projection-store-D3vBHS4J.js';
|
|
9
9
|
import 'better-sqlite3';
|
|
10
10
|
|
|
11
11
|
interface QmdRuntimeLike {
|
|
@@ -50,6 +50,7 @@ interface ConversationIndexLike {
|
|
|
50
50
|
}
|
|
51
51
|
interface OperatorToolkitOrchestrator extends ConversationIndexLike {
|
|
52
52
|
config: PluginConfig;
|
|
53
|
+
storage: StorageManager;
|
|
53
54
|
qmd: QmdRuntimeLike;
|
|
54
55
|
}
|
|
55
56
|
interface OperatorConfigLoadResult {
|
|
@@ -333,6 +334,49 @@ declare function summarizeMemoryWorthLegacyCounters(storage: StorageManager): Pr
|
|
|
333
334
|
* orchestrator.
|
|
334
335
|
*/
|
|
335
336
|
declare function summarizeBufferSurpriseDistribution(storage: StorageManager, config: PluginConfig): Promise<OperatorDoctorCheck>;
|
|
337
|
+
/**
|
|
338
|
+
* Summarize hot/cold tier distribution for the Doctor report
|
|
339
|
+
* (issue #686 retention-completion).
|
|
340
|
+
*
|
|
341
|
+
* Shows:
|
|
342
|
+
* - hot vs cold memory counts
|
|
343
|
+
* - forgotten-memory count
|
|
344
|
+
* - top per-status breakdown
|
|
345
|
+
* - recent migrations in the last 7 days (from the tier-migration journal)
|
|
346
|
+
* - top demotion reasons from the journal
|
|
347
|
+
*
|
|
348
|
+
* Always returns an `ok` check — this section is informational, never
|
|
349
|
+
* a gate. A missing journal or unreadable corpus is surfaced as a note,
|
|
350
|
+
* not an error.
|
|
351
|
+
*
|
|
352
|
+
* Exported so tests can exercise the summarizer without a full orchestrator.
|
|
353
|
+
*/
|
|
354
|
+
declare function summarizeTierDistribution(storage: StorageManager): Promise<OperatorDoctorCheck>;
|
|
355
|
+
/**
|
|
356
|
+
* Dreams phases doctor check (issue #678 PR 2/4).
|
|
357
|
+
*
|
|
358
|
+
* Reports per-phase: enabled status, cadence, threshold values, and last-run
|
|
359
|
+
* timestamp sourced from `meta.json` (the existing maintenance ledger).
|
|
360
|
+
*
|
|
361
|
+
* Last-run mapping (best available in PR 2; PR 3/4 will emit per-phase events):
|
|
362
|
+
* light sleep → `meta.lastExtractionAt` (lifecycle pass runs with extraction)
|
|
363
|
+
* REM → `meta.lastConsolidationAt` (semantic consolidation)
|
|
364
|
+
* deep sleep → latest governance run manifest when present, otherwise null
|
|
365
|
+
*/
|
|
366
|
+
declare function summarizeDreamsPhases(config: Pick<PluginConfig, "memoryDir" | "dreamsPhases">, storage?: StorageManager): Promise<OperatorDoctorCheck>;
|
|
367
|
+
/**
|
|
368
|
+
* Observation throughput check for `remnic doctor` (issue #685).
|
|
369
|
+
*
|
|
370
|
+
* Reads the extraction-judge verdict ledger to surface:
|
|
371
|
+
* - total observations (verdicts) recorded
|
|
372
|
+
* - accept / reject / defer breakdown
|
|
373
|
+
* - most-recent `observedAt` timestamp
|
|
374
|
+
*
|
|
375
|
+
* The check is purely informational (always `ok`): a missing or empty
|
|
376
|
+
* ledger is the expected state on a fresh install and is never an error.
|
|
377
|
+
* This closes the `*(future)*` item deferred in `docs/trace-to-primitive.md`.
|
|
378
|
+
*/
|
|
379
|
+
declare function summarizeObservationThroughput(memoryDir: string): Promise<OperatorDoctorCheck>;
|
|
336
380
|
declare function summarizeConsolidationProvenance(storage: StorageManager, config: Pick<PluginConfig, "memoryDir"> & {
|
|
337
381
|
versioningSidecarDir?: string;
|
|
338
382
|
}): Promise<OperatorDoctorCheck>;
|
|
@@ -340,4 +384,4 @@ declare function runOperatorInventory(options: OperatorInventoryOptions): Promis
|
|
|
340
384
|
declare function runBenchmarkRecall(options: BenchmarkRecallOptions): Promise<BenchmarkRecallReport>;
|
|
341
385
|
declare function runOperatorRepair(options: OperatorRepairOptions): Promise<OperatorRepairReport>;
|
|
342
386
|
|
|
343
|
-
export { type BenchmarkRecallOptions, type BenchmarkRecallReport, type OperatorConfigLoadResult, type OperatorConfigReviewFinding, type OperatorConfigReviewOptions, type OperatorConfigReviewReport, type OperatorDoctorCheck, type OperatorDoctorOptions, type OperatorDoctorReport, type OperatorInventoryNamespaceSummary, type OperatorInventoryOptions, type OperatorInventoryReport, type OperatorRepairOptions, type OperatorRepairReport, type OperatorSetupOptions, type OperatorSetupReport, type OperatorToolkitOrchestrator, runBenchmarkRecall, runOperatorConfigReview, runOperatorDoctor, runOperatorInventory, runOperatorRepair, runOperatorSetup, summarizeBufferSurpriseDistribution, summarizeConsolidationProvenance, summarizeMemoryWorthLegacyCounters };
|
|
387
|
+
export { type BenchmarkRecallOptions, type BenchmarkRecallReport, type OperatorConfigLoadResult, type OperatorConfigReviewFinding, type OperatorConfigReviewOptions, type OperatorConfigReviewReport, type OperatorDoctorCheck, type OperatorDoctorOptions, type OperatorDoctorReport, type OperatorInventoryNamespaceSummary, type OperatorInventoryOptions, type OperatorInventoryReport, type OperatorRepairOptions, type OperatorRepairReport, type OperatorSetupOptions, type OperatorSetupReport, type OperatorToolkitOrchestrator, runBenchmarkRecall, runOperatorConfigReview, runOperatorDoctor, runOperatorInventory, runOperatorRepair, runOperatorSetup, summarizeBufferSurpriseDistribution, summarizeConsolidationProvenance, summarizeDreamsPhases, summarizeMemoryWorthLegacyCounters, summarizeObservationThroughput, summarizeTierDistribution };
|
package/dist/operator-toolkit.js
CHANGED
|
@@ -7,40 +7,49 @@ import {
|
|
|
7
7
|
runOperatorSetup,
|
|
8
8
|
summarizeBufferSurpriseDistribution,
|
|
9
9
|
summarizeConsolidationProvenance,
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
10
|
+
summarizeDreamsPhases,
|
|
11
|
+
summarizeMemoryWorthLegacyCounters,
|
|
12
|
+
summarizeObservationThroughput,
|
|
13
|
+
summarizeTierDistribution
|
|
14
|
+
} from "./chunk-7GCMLT7J.js";
|
|
15
|
+
import "./chunk-5HRY2WRF.js";
|
|
13
16
|
import "./chunk-ASAITVLA.js";
|
|
14
|
-
import "./chunk-
|
|
17
|
+
import "./chunk-R2L7SUX2.js";
|
|
15
18
|
import "./chunk-YRMVARQP.js";
|
|
16
|
-
import "./chunk-
|
|
19
|
+
import "./chunk-WSZIHQBK.js";
|
|
17
20
|
import "./chunk-U66YHYC7.js";
|
|
18
|
-
import "./chunk-LK6SGL53.js";
|
|
19
|
-
import "./chunk-7SEAZFFB.js";
|
|
20
21
|
import "./chunk-K6WK37A6.js";
|
|
21
|
-
import "./chunk-
|
|
22
|
+
import "./chunk-AJU4PJGY.js";
|
|
23
|
+
import "./chunk-7SEAZFFB.js";
|
|
24
|
+
import "./chunk-JWSENLQI.js";
|
|
22
25
|
import "./chunk-Z5LAYHGJ.js";
|
|
23
|
-
import "./chunk-C2EFFULQ.js";
|
|
24
26
|
import "./chunk-PVGDJXVK.js";
|
|
25
|
-
import "./chunk-
|
|
26
|
-
import "./chunk-3OGMS3PE.js";
|
|
27
|
+
import "./chunk-LZRYQK6L.js";
|
|
27
28
|
import "./chunk-2LGMW3DJ.js";
|
|
29
|
+
import "./chunk-OR64ZGRZ.js";
|
|
30
|
+
import "./chunk-RK2Y4XOM.js";
|
|
31
|
+
import "./chunk-2LSZVONP.js";
|
|
32
|
+
import "./chunk-WIICJPET.js";
|
|
28
33
|
import "./chunk-TBBDFYXW.js";
|
|
29
|
-
import "./chunk-
|
|
30
|
-
import "./chunk-
|
|
34
|
+
import "./chunk-DCE6SQLA.js";
|
|
35
|
+
import "./chunk-YNJHCGDT.js";
|
|
31
36
|
import "./chunk-4KAN3GZ3.js";
|
|
32
|
-
import "./chunk-
|
|
33
|
-
import "./chunk-
|
|
37
|
+
import "./chunk-43EKP2UK.js";
|
|
38
|
+
import "./chunk-PFV5C235.js";
|
|
39
|
+
import "./chunk-RULE4VG5.js";
|
|
34
40
|
import "./chunk-SCU65EZI.js";
|
|
35
41
|
import "./chunk-BOUYNNYD.js";
|
|
36
42
|
import "./chunk-DM2T26WE.js";
|
|
37
43
|
import "./chunk-QSVPYQPG.js";
|
|
38
44
|
import "./chunk-M62O4P4T.js";
|
|
39
45
|
import "./chunk-4DJQYKMN.js";
|
|
40
|
-
import "./chunk-
|
|
46
|
+
import "./chunk-G7D6GZ5J.js";
|
|
41
47
|
import "./chunk-FAAFWE4G.js";
|
|
48
|
+
import "./chunk-MARWOCVP.js";
|
|
42
49
|
import "./chunk-2ODBA7MQ.js";
|
|
43
50
|
import "./chunk-LPSF4OQH.js";
|
|
51
|
+
import "./chunk-A6XUJE5D.js";
|
|
52
|
+
import "./chunk-PZ5AY32C.js";
|
|
44
53
|
export {
|
|
45
54
|
runBenchmarkRecall,
|
|
46
55
|
runOperatorConfigReview,
|
|
@@ -50,6 +59,9 @@ export {
|
|
|
50
59
|
runOperatorSetup,
|
|
51
60
|
summarizeBufferSurpriseDistribution,
|
|
52
61
|
summarizeConsolidationProvenance,
|
|
53
|
-
|
|
62
|
+
summarizeDreamsPhases,
|
|
63
|
+
summarizeMemoryWorthLegacyCounters,
|
|
64
|
+
summarizeObservationThroughput,
|
|
65
|
+
summarizeTierDistribution
|
|
54
66
|
};
|
|
55
67
|
//# sourceMappingURL=operator-toolkit.js.map
|
package/dist/opik-exporter.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/opik-exporter.ts"],"sourcesContent":["import { createHash, randomBytes } from \"node:crypto\";\n/**\n * opik-exporter.ts — Engram-native Opik trace exporter\n *\n * Subscribes to the globalThis.__openclawEngramTrace slot that Engram already\n * emits on, and forwards recall + LLM events to a self-hosted (or cloud) Opik\n * instance via its REST API. No extra npm dependencies — uses Node's built-in\n * fetch.\n *\n * Auto-detects apiUrl / projectName from the opik-openclaw plugin config so\n * traces land in the same project and are grouped by session thread.\n */\n\nimport { readFileSync } from \"node:fs\";\nimport path from \"node:path\";\n\nimport type { LoggerBackend } from \"./logger.js\";\nimport { readEnvVar, resolveHomeDir } from \"./runtime/env.js\";\n\n// GlobalThis slot that tracks the active OpikExporter instance.\n// Used to make subscribe() idempotent across hot-reload / stop-start cycles.\nconst OPIK_EXPORTER_SLOT = \"__openclawOpikExporter\";\n\n// ---------------------------------------------------------------------------\n// Engram event types (mirrors types.ts without importing from it)\n// ---------------------------------------------------------------------------\n\ntype EngramLlmTraceEvent = {\n kind: \"llm_start\" | \"llm_end\" | \"llm_error\";\n traceId: string;\n model: string;\n operation: \"extraction\" | \"consolidation\" | \"profile_consolidation\" | \"identity_consolidation\" | \"day_summary\";\n input?: string;\n output?: string;\n durationMs?: number;\n error?: string;\n tokenUsage?: { input?: number; output?: number; total?: number };\n};\n\ntype EngramRecallTraceEvent = {\n kind: \"recall_summary\";\n traceId: string;\n operation: \"recall\";\n sessionKey?: string;\n promptLength: number;\n retrievalQueryLength: number;\n recallMode: string;\n recallResultLimit: number;\n qmdEnabled: boolean;\n qmdAvailable: boolean;\n recallNamespaces: string[];\n source: string;\n recalledMemoryCount: number;\n injected: boolean;\n contextChars: number;\n identityInjectionMode?: string;\n identityInjectedChars?: number;\n durationMs: number;\n timings?: Record<string, string>;\n recalledContent?: string;\n};\n\ntype EngramTraceEvent = EngramLlmTraceEvent | EngramRecallTraceEvent;\n\n// ---------------------------------------------------------------------------\n// Config\n// ---------------------------------------------------------------------------\n\nexport interface OpikExporterConfig {\n enabled: boolean;\n /** Base URL of the Opik API, e.g. http://192.168.3.147:5173/api */\n apiUrl: string;\n projectName: string;\n workspaceName: string;\n /** Optional API key (not required for self-hosted deployments) */\n apiKey?: string;\n /** Include recalled memory text in spans (default false) */\n traceRecallContent: boolean;\n}\n\n// ---------------------------------------------------------------------------\n// Auto-detect from opik-openclaw plugin config\n// ---------------------------------------------------------------------------\n\nfunction readOpikOpenclawConfig(log?: LoggerBackend): Partial<OpikExporterConfig> {\n try {\n const configPath =\n readEnvVar(\"OPENCLAW_ENGRAM_CONFIG_PATH\") ||\n readEnvVar(\"OPENCLAW_CONFIG_PATH\") ||\n path.join(resolveHomeDir(), \".openclaw\", \"openclaw.json\");\n const raw = JSON.parse(readFileSync(configPath, \"utf-8\"));\n const entry = raw?.plugins?.entries?.[\"opik-openclaw\"];\n if (!entry?.enabled || !entry?.config) return {};\n const c = entry.config as Record<string, unknown>;\n return {\n apiUrl: typeof c.apiUrl === \"string\" && c.apiUrl.length > 0 ? c.apiUrl : undefined,\n projectName: typeof c.projectName === \"string\" ? c.projectName : undefined,\n workspaceName: typeof c.workspaceName === \"string\" ? c.workspaceName : undefined,\n apiKey: typeof c.apiKey === \"string\" && c.apiKey.length > 0 ? c.apiKey : undefined,\n };\n } catch (err) {\n log?.debug?.(`[opik-exporter] could not read opik-openclaw config: ${err}`);\n return {};\n }\n}\n\n// ---------------------------------------------------------------------------\n// UUID v7 generator (Opik requires v7, not v4)\n// ---------------------------------------------------------------------------\n\nfunction uuidV7(): string {\n const now = Date.now();\n const bytes = randomBytes(16);\n // Timestamp: 48-bit ms since epoch in bytes 0-5\n bytes[0] = (now / 2 ** 40) & 0xff;\n bytes[1] = (now / 2 ** 32) & 0xff;\n bytes[2] = (now / 2 ** 24) & 0xff;\n bytes[3] = (now / 2 ** 16) & 0xff;\n bytes[4] = (now / 2 ** 8) & 0xff;\n bytes[5] = now & 0xff;\n // Version 7\n bytes[6] = (bytes[6] & 0x0f) | 0x70;\n // Variant 10xx\n bytes[8] = (bytes[8] & 0x3f) | 0x80;\n const hex = bytes.toString(\"hex\");\n return `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`;\n}\n\n// ---------------------------------------------------------------------------\n// REST helpers\n// ---------------------------------------------------------------------------\n\nfunction buildHeaders(cfg: OpikExporterConfig): Record<string, string> {\n const headers: Record<string, string> = { \"Content-Type\": \"application/json\" };\n if (cfg.workspaceName) headers[\"Comet-Workspace\"] = cfg.workspaceName;\n // Opik expects the raw API key in the authorization header without a Bearer prefix.\n if (cfg.apiKey) headers[\"authorization\"] = cfg.apiKey;\n return headers;\n}\n\nasync function postSpanBatch(\n cfg: OpikExporterConfig,\n spans: unknown[],\n log: LoggerBackend,\n): Promise<void> {\n const url = `${cfg.apiUrl.replace(/\\/$/, \"\")}/v1/private/spans/batch`;\n try {\n const res = await fetch(url, {\n method: \"POST\",\n headers: buildHeaders(cfg),\n body: JSON.stringify({ spans }),\n });\n if (!res.ok) {\n const text = await res.text().catch(() => \"\");\n log.debug?.(`[opik-exporter] span batch failed ${res.status}: ${text}`);\n }\n } catch (err) {\n log.debug?.(`[opik-exporter] span batch error: ${err}`);\n }\n}\n\nasync function postTraceBatch(\n cfg: OpikExporterConfig,\n traces: unknown[],\n log: LoggerBackend,\n): Promise<boolean> {\n const url = `${cfg.apiUrl.replace(/\\/$/, \"\")}/v1/private/traces/batch`;\n try {\n const res = await fetch(url, {\n method: \"POST\",\n headers: buildHeaders(cfg),\n body: JSON.stringify({ traces }),\n });\n if (!res.ok) {\n const text = await res.text().catch(() => \"\");\n log.debug?.(`[opik-exporter] trace batch failed ${res.status}: ${text}`);\n return false;\n }\n return true;\n } catch (err) {\n log.debug?.(`[opik-exporter] trace batch error: ${err}`);\n return false;\n }\n}\n\n// ---------------------------------------------------------------------------\n// In-flight LLM span tracking\n// ---------------------------------------------------------------------------\n\ntype InFlightLlm = {\n startedAt: number;\n startTime: string;\n model: string;\n operation: string;\n input?: string;\n spanId: string;\n traceId: string;\n};\n\n// ---------------------------------------------------------------------------\n// Main exporter class\n// ---------------------------------------------------------------------------\n\nexport class OpikExporter {\n private readonly cfg: OpikExporterConfig;\n private readonly log: LoggerBackend;\n private _handler: ((e: EngramTraceEvent) => void) | undefined;\n private readonly inFlight = new Map<string, InFlightLlm>();\n /** Track which trace_ids we have already created parent trace objects for. */\n private readonly createdTraces = new Set<string>();\n /** In-flight ensureTrace promises keyed by traceId — concurrent callers await the same promise. */\n private readonly pendingTraces = new Map<string, Promise<void>>();\n /** TTL for in-flight LLM entries (ms). Entries older than this are discarded. */\n private static readonly IN_FLIGHT_TTL_MS = 5 * 60 * 1000; // 5 minutes\n\n constructor(cfg: OpikExporterConfig, log: LoggerBackend) {\n this.cfg = cfg;\n this.log = log;\n }\n\n /**\n * Subscribe to Engram's global trace slot. Safe to call multiple times —\n * chains with existing subscribers (e.g. Langfuse) rather than replacing.\n */\n subscribe(): void {\n if (!this.cfg.enabled) return;\n\n // Idempotency guard: if this exact instance is already registered on globalThis,\n // skip re-wrapping to avoid stacking callbacks on hot-reload / stop-start cycles.\n const g = globalThis as Record<string, unknown>;\n const active = g[OPIK_EXPORTER_SLOT] as OpikExporter | undefined;\n if (active === this) {\n this.log.debug?.(\"[opik-exporter] already subscribed — skipping duplicate\");\n return;\n }\n // If a different (stale) exporter instance is registered, evict it first.\n if (active) active._detach();\n\n g[OPIK_EXPORTER_SLOT] = this;\n\n const existing = g.__openclawEngramTrace as\n | ((e: EngramTraceEvent) => void)\n | undefined;\n\n const handler = (event: EngramTraceEvent) => {\n try {\n this.handleEvent(event);\n } catch (err) {\n this.log.debug?.(`[opik-exporter] handler error: ${err}`);\n }\n };\n // Store handler ref so _detach() can remove it from the chain.\n this._handler = handler;\n\n // Chain: call existing subscribers first, then ours.\n // NOTE: when _detach() is later called, it marks this.cfg.enabled = false\n // rather than restoring the chain, so that any subscriber that chained\n // *after* us is not dropped. The cost is one disabled no-op wrapper\n // remaining in the chain per stop/start cycle. In practice, Engram\n // restarts Opik only a handful of times per process lifetime, so the\n // overhead is negligible (one boolean check per leftover closure).\n g.__openclawEngramTrace =\n typeof existing === \"function\"\n ? (event: EngramTraceEvent) => {\n try { existing(event); } catch { /* prior subscriber threw; continue */ }\n handler(event);\n }\n : handler;\n\n this.log.info(\n `[opik-exporter] subscribed — apiUrl=${this.cfg.apiUrl} project=${this.cfg.projectName}`,\n );\n }\n\n /** Deactivate this exporter instance.\n *\n * We intentionally do NOT restore __openclawEngramTrace to a prior handler\n * because any subscriber that chained *after* us would be silently dropped\n * by such a restoration. Instead we mark the instance as disabled so that\n * handleEvent() becomes a no-op while the callback chain itself is left\n * untouched. The next subscribe() call from a fresh instance will re-enter\n * the chain on top of whatever is there at that point.\n */\n _detach(): void {\n const g = globalThis as Record<string, unknown>;\n // Only clear the global slot if we are still the active exporter.\n // A stale instance must not evict a newer exporter that already took over.\n if (g[OPIK_EXPORTER_SLOT] === this) delete g[OPIK_EXPORTER_SLOT];\n // Disable event processing; the wrapper closure in the chain becomes a no-op.\n (this.cfg as OpikExporterConfig & { enabled: boolean }).enabled = false;\n this._handler = undefined;\n this.log.debug?.(\"[opik-exporter] detached — events silenced, chain preserved\");\n }\n\n unsubscribe(): void {\n this._detach();\n }\n\n private handleEvent(event: EngramTraceEvent): void {\n if (!this.cfg.enabled) return; // guard for after _detach()\n if (event.kind === \"recall_summary\") {\n void this.onRecall(event);\n } else if (event.kind === \"llm_start\") {\n this.onLlmStart(event);\n } else if (event.kind === \"llm_end\" || event.kind === \"llm_error\") {\n void this.onLlmEnd(event);\n }\n }\n\n // -------------------------------------------------------------------------\n // Recall events → general span\n // -------------------------------------------------------------------------\n\n private async onRecall(evt: EngramRecallTraceEvent): Promise<void> {\n const now = new Date();\n const startTime = new Date(now.getTime() - evt.durationMs).toISOString();\n const endTime = now.toISOString();\n\n const input: Record<string, unknown> = {\n recallMode: evt.recallMode,\n recalledMemoryCount: evt.recalledMemoryCount,\n recallNamespaces: evt.recallNamespaces,\n source: evt.source,\n qmdEnabled: evt.qmdEnabled,\n qmdAvailable: evt.qmdAvailable,\n retrievalQueryLength: evt.retrievalQueryLength,\n recallResultLimit: evt.recallResultLimit,\n promptLength: evt.promptLength,\n };\n\n if (this.cfg.traceRecallContent && evt.recalledContent) {\n input.recalledContent = evt.recalledContent;\n }\n\n const metadata: Record<string, unknown> = {\n source: \"engram\",\n injected: evt.injected,\n contextChars: evt.contextChars,\n durationMs: evt.durationMs,\n };\n\n if (evt.identityInjectionMode) {\n metadata.identityInjectionMode = evt.identityInjectionMode;\n metadata.identityInjectedChars = evt.identityInjectedChars;\n }\n\n if (evt.timings) metadata.timings = evt.timings;\n\n const traceId = evt.sessionKey ? this.sessionToTraceId(evt.sessionKey) : uuidV7();\n\n // Ensure parent trace exists so the span is not orphaned in Opik.\n await this.ensureTrace(traceId, evt.sessionKey ?? \"engram:recall\", startTime, endTime);\n\n const output: Record<string, unknown> = {\n injected: evt.injected,\n recalledMemoryCount: evt.recalledMemoryCount,\n contextChars: evt.contextChars,\n durationMs: evt.durationMs,\n };\n if (this.cfg.traceRecallContent && evt.recalledContent) {\n output.recalledContent = evt.recalledContent;\n }\n\n const span = {\n id: uuidV7(),\n trace_id: traceId,\n project_name: this.cfg.projectName,\n name: \"engram:recall\",\n type: \"general\",\n start_time: startTime,\n end_time: endTime,\n input,\n output,\n metadata,\n tags: [\"engram\", \"recall\"],\n };\n\n await postSpanBatch(this.cfg, [span], this.log);\n }\n\n // -------------------------------------------------------------------------\n // Engram internal LLM events (extraction / consolidation) → llm span\n // -------------------------------------------------------------------------\n\n private onLlmStart(evt: EngramLlmTraceEvent): void {\n // Use evt.traceId (Engram's correlation key) as the Opik spanId so spans\n // are deterministic and stable across retries. Derive a per-call Opik\n // trace_id from the same key so each LLM call maps to its own trace.\n this.inFlight.set(evt.traceId, {\n startedAt: Date.now(),\n startTime: new Date().toISOString(),\n model: evt.model,\n operation: evt.operation,\n input: evt.input,\n spanId: this.sessionToTraceId(evt.traceId + \":span\"),\n traceId: this.sessionToTraceId(evt.traceId),\n });\n }\n\n private sweepStaleInFlight(): void {\n const cutoff = Date.now() - OpikExporter.IN_FLIGHT_TTL_MS;\n for (const [id, entry] of this.inFlight) {\n if (entry.startedAt < cutoff) {\n this.inFlight.delete(id);\n }\n }\n }\n\n private async onLlmEnd(evt: EngramLlmTraceEvent): Promise<void> {\n // Opportunistically sweep stale entries to prevent unbounded map growth.\n if (this.inFlight.size > 20) this.sweepStaleInFlight();\n const state = this.inFlight.get(evt.traceId);\n this.inFlight.delete(evt.traceId);\n\n const startTime = state?.startTime ?? new Date().toISOString();\n const endTime = new Date().toISOString();\n\n const usage: Record<string, number> = {};\n if (evt.tokenUsage?.input != null) usage.prompt_tokens = evt.tokenUsage.input;\n if (evt.tokenUsage?.output != null) usage.completion_tokens = evt.tokenUsage.output;\n if (evt.tokenUsage?.total != null) usage.total_tokens = evt.tokenUsage.total;\n\n const traceId = state?.traceId ?? uuidV7();\n\n // Ensure parent trace exists so the span is not orphaned in Opik.\n await this.ensureTrace(traceId, `engram:${evt.operation}`, startTime, endTime);\n\n const span: Record<string, unknown> = {\n id: state?.spanId ?? uuidV7(),\n trace_id: traceId,\n project_name: this.cfg.projectName,\n name: `engram:${evt.operation}`,\n type: \"llm\",\n model: evt.model,\n start_time: startTime,\n end_time: endTime,\n input: state?.input != null ? { prompt: state.input } : undefined,\n output:\n evt.kind === \"llm_end\" && evt.output != null\n ? { completion: evt.output }\n : evt.kind === \"llm_error\"\n ? { error: evt.error }\n : undefined,\n metadata: {\n source: \"engram\",\n operation: evt.operation,\n durationMs: evt.durationMs ?? (state ? Date.now() - state.startedAt : undefined),\n },\n tags: [\"engram\", evt.operation],\n usage: Object.keys(usage).length > 0 ? usage : undefined,\n };\n\n await postSpanBatch(this.cfg, [span], this.log);\n }\n\n // -------------------------------------------------------------------------\n // Helpers\n // -------------------------------------------------------------------------\n\n /**\n * Create a parent trace object in Opik if we haven't already.\n * Opik does NOT auto-create traces from spans, so without this\n * the spans are orphaned and don't show up in the trace list UI.\n */\n private async ensureTrace(\n traceId: string,\n name: string,\n startTime: string,\n endTime: string,\n ): Promise<void> {\n if (this.createdTraces.has(traceId)) return;\n\n // If another call is already creating this trace, wait for it and then\n // re-check — if the first attempt failed we'll retry below.\n const existing = this.pendingTraces.get(traceId);\n if (existing) {\n await existing;\n if (this.createdTraces.has(traceId)) return;\n // First attempt failed — check if another waiter is already retrying.\n const retrying = this.pendingTraces.get(traceId);\n if (retrying) {\n await retrying;\n // Only return if the retry actually succeeded.\n if (this.createdTraces.has(traceId)) return;\n }\n // No one else retrying; fall through to create.\n }\n\n const work = (async () => {\n const trace = {\n id: traceId,\n project_name: this.cfg.projectName,\n name,\n start_time: startTime,\n end_time: endTime,\n tags: [\"engram\"],\n };\n\n const ok = await postTraceBatch(this.cfg, [trace], this.log);\n if (!ok) {\n this.pendingTraces.delete(traceId);\n return; // transient failure — allow retry on next span\n }\n\n this.createdTraces.add(traceId);\n this.pendingTraces.delete(traceId);\n\n // Cap the set size to prevent unbounded growth in long-running processes.\n if (this.createdTraces.size > 10_000) {\n const first = this.createdTraces.values().next().value;\n if (first) this.createdTraces.delete(first);\n }\n })();\n\n this.pendingTraces.set(traceId, work);\n await work;\n }\n\n /**\n * Convert a sessionKey to a stable, deterministic UUID v7-shaped ID so that\n * spans for the same session share a trace_id and are threaded in Opik.\n * Uses SHA-256 for collision resistance. The timestamp bytes (0-5) come from\n * the hash itself — they don't reflect real time, but Opik only validates\n * the version/variant bits, not timestamp ordering.\n */\n private sessionToTraceId(sessionKey: string): string {\n const digest = createHash(\"sha256\").update(sessionKey).digest();\n // Version 7 (bytes 0-5 are hash-derived, not real timestamps)\n digest[6] = (digest[6] & 0x0f) | 0x70;\n // Variant 10xx\n digest[8] = (digest[8] & 0x3f) | 0x80;\n const hex = digest.slice(0, 16).toString(\"hex\");\n return `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`;\n }\n}\n\n// ---------------------------------------------------------------------------\n// Factory: build from Engram resolved config + auto-detect from opik-openclaw\n// ---------------------------------------------------------------------------\n\nexport interface OpikExporterRawConfig {\n opikTraceEnabled?: boolean;\n opikApiUrl?: string;\n opikProjectName?: string;\n opikWorkspaceName?: string;\n opikApiKey?: string;\n opikTraceRecallContent?: boolean;\n}\n\nexport function createOpikExporter(\n raw: OpikExporterRawConfig,\n log: LoggerBackend,\n): OpikExporter | null {\n // Explicit opt-out: if set to false, never enable.\n if (raw.opikTraceEnabled === false) return null;\n\n // Auto-detect from opik-openclaw plugin config.\n // If opikTraceEnabled is not set (undefined), we enable automatically\n // whenever opik-openclaw is configured — avoids needing extra fields in\n // Engram's openclaw.json config section (which would fail schema validation).\n const detected = readOpikOpenclawConfig(log);\n\n const apiUrl = raw.opikApiUrl ?? detected.apiUrl;\n if (!apiUrl) {\n // Only warn when the user explicitly opted in but forgot to set the URL.\n // When auto-detecting (opikTraceEnabled not set), return silently so non-Opik\n // users see no log noise on every startup.\n if (raw.opikTraceEnabled === true) {\n log.warn(\n \"[opik-exporter] opikTraceEnabled=true but no apiUrl found — \" +\n \"set opikApiUrl in Engram config or ensure opik-openclaw plugin is configured\",\n );\n }\n return null;\n }\n\n const cfg: OpikExporterConfig = {\n enabled: true,\n apiUrl,\n projectName: raw.opikProjectName ?? detected.projectName ?? \"openclaw\",\n workspaceName: raw.opikWorkspaceName ?? detected.workspaceName ?? \"default\",\n apiKey: raw.opikApiKey ?? detected.apiKey,\n traceRecallContent: raw.opikTraceRecallContent === true,\n };\n\n return new OpikExporter(cfg, log);\n}\n"],"mappings":";;;;;;AAAA,SAAS,YAAY,mBAAmB;AAaxC,SAAS,oBAAoB;AAC7B,OAAO,UAAU;AAOjB,IAAM,qBAAqB;AA+D3B,SAAS,uBAAuB,KAAkD;AAChF,MAAI;AACF,UAAM,aACJ,WAAW,6BAA6B,KACxC,WAAW,sBAAsB,KACjC,KAAK,KAAK,eAAe,GAAG,aAAa,eAAe;AAC1D,UAAM,MAAM,KAAK,MAAM,aAAa,YAAY,OAAO,CAAC;AACxD,UAAM,QAAQ,KAAK,SAAS,UAAU,eAAe;AACrD,QAAI,CAAC,OAAO,WAAW,CAAC,OAAO,OAAQ,QAAO,CAAC;AAC/C,UAAM,IAAI,MAAM;AAChB,WAAO;AAAA,MACL,QAAQ,OAAO,EAAE,WAAW,YAAY,EAAE,OAAO,SAAS,IAAI,EAAE,SAAS;AAAA,MACzE,aAAa,OAAO,EAAE,gBAAgB,WAAW,EAAE,cAAc;AAAA,MACjE,eAAe,OAAO,EAAE,kBAAkB,WAAW,EAAE,gBAAgB;AAAA,MACvE,QAAQ,OAAO,EAAE,WAAW,YAAY,EAAE,OAAO,SAAS,IAAI,EAAE,SAAS;AAAA,IAC3E;AAAA,EACF,SAAS,KAAK;AACZ,SAAK,QAAQ,wDAAwD,GAAG,EAAE;AAC1E,WAAO,CAAC;AAAA,EACV;AACF;AAMA,SAAS,SAAiB;AACxB,QAAM,MAAM,KAAK,IAAI;AACrB,QAAM,QAAQ,YAAY,EAAE;AAE5B,QAAM,CAAC,IAAK,MAAM,KAAK,KAAM;AAC7B,QAAM,CAAC,IAAK,MAAM,KAAK,KAAM;AAC7B,QAAM,CAAC,IAAK,MAAM,KAAK,KAAM;AAC7B,QAAM,CAAC,IAAK,MAAM,KAAK,KAAM;AAC7B,QAAM,CAAC,IAAK,MAAM,KAAK,IAAK;AAC5B,QAAM,CAAC,IAAI,MAAM;AAEjB,QAAM,CAAC,IAAK,MAAM,CAAC,IAAI,KAAQ;AAE/B,QAAM,CAAC,IAAK,MAAM,CAAC,IAAI,KAAQ;AAC/B,QAAM,MAAM,MAAM,SAAS,KAAK;AAChC,SAAO,GAAG,IAAI,MAAM,GAAG,CAAC,CAAC,IAAI,IAAI,MAAM,GAAG,EAAE,CAAC,IAAI,IAAI,MAAM,IAAI,EAAE,CAAC,IAAI,IAAI,MAAM,IAAI,EAAE,CAAC,IAAI,IAAI,MAAM,EAAE,CAAC;AAC1G;AAMA,SAAS,aAAa,KAAiD;AACrE,QAAM,UAAkC,EAAE,gBAAgB,mBAAmB;AAC7E,MAAI,IAAI,cAAe,SAAQ,iBAAiB,IAAI,IAAI;AAExD,MAAI,IAAI,OAAQ,SAAQ,eAAe,IAAI,IAAI;AAC/C,SAAO;AACT;AAEA,eAAe,cACb,KACA,OACA,KACe;AACf,QAAM,MAAM,GAAG,IAAI,OAAO,QAAQ,OAAO,EAAE,CAAC;AAC5C,MAAI;AACF,UAAM,MAAM,MAAM,MAAM,KAAK;AAAA,MAC3B,QAAQ;AAAA,MACR,SAAS,aAAa,GAAG;AAAA,MACzB,MAAM,KAAK,UAAU,EAAE,MAAM,CAAC;AAAA,IAChC,CAAC;AACD,QAAI,CAAC,IAAI,IAAI;AACX,YAAM,OAAO,MAAM,IAAI,KAAK,EAAE,MAAM,MAAM,EAAE;AAC5C,UAAI,QAAQ,qCAAqC,IAAI,MAAM,KAAK,IAAI,EAAE;AAAA,IACxE;AAAA,EACF,SAAS,KAAK;AACZ,QAAI,QAAQ,qCAAqC,GAAG,EAAE;AAAA,EACxD;AACF;AAEA,eAAe,eACb,KACA,QACA,KACkB;AAClB,QAAM,MAAM,GAAG,IAAI,OAAO,QAAQ,OAAO,EAAE,CAAC;AAC5C,MAAI;AACF,UAAM,MAAM,MAAM,MAAM,KAAK;AAAA,MAC3B,QAAQ;AAAA,MACR,SAAS,aAAa,GAAG;AAAA,MACzB,MAAM,KAAK,UAAU,EAAE,OAAO,CAAC;AAAA,IACjC,CAAC;AACD,QAAI,CAAC,IAAI,IAAI;AACX,YAAM,OAAO,MAAM,IAAI,KAAK,EAAE,MAAM,MAAM,EAAE;AAC5C,UAAI,QAAQ,sCAAsC,IAAI,MAAM,KAAK,IAAI,EAAE;AACvE,aAAO;AAAA,IACT;AACA,WAAO;AAAA,EACT,SAAS,KAAK;AACZ,QAAI,QAAQ,sCAAsC,GAAG,EAAE;AACvD,WAAO;AAAA,EACT;AACF;AAoBO,IAAM,eAAN,MAAM,cAAa;AAAA,EACP;AAAA,EACA;AAAA,EACT;AAAA,EACS,WAAW,oBAAI,IAAyB;AAAA;AAAA,EAExC,gBAAgB,oBAAI,IAAY;AAAA;AAAA,EAEhC,gBAAgB,oBAAI,IAA2B;AAAA;AAAA,EAEhE,OAAwB,mBAAmB,IAAI,KAAK;AAAA;AAAA,EAEpD,YAAY,KAAyB,KAAoB;AACvD,SAAK,MAAM;AACX,SAAK,MAAM;AAAA,EACb;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,YAAkB;AAChB,QAAI,CAAC,KAAK,IAAI,QAAS;AAIvB,UAAM,IAAI;AACV,UAAM,SAAS,EAAE,kBAAkB;AACnC,QAAI,WAAW,MAAM;AACnB,WAAK,IAAI,QAAQ,8DAAyD;AAC1E;AAAA,IACF;AAEA,QAAI,OAAQ,QAAO,QAAQ;AAE3B,MAAE,kBAAkB,IAAI;AAExB,UAAM,WAAW,EAAE;AAInB,UAAM,UAAU,CAAC,UAA4B;AAC3C,UAAI;AACF,aAAK,YAAY,KAAK;AAAA,MACxB,SAAS,KAAK;AACZ,aAAK,IAAI,QAAQ,kCAAkC,GAAG,EAAE;AAAA,MAC1D;AAAA,IACF;AAEA,SAAK,WAAW;AAShB,MAAE,wBACA,OAAO,aAAa,aAChB,CAAC,UAA4B;AAC3B,UAAI;AAAE,iBAAS,KAAK;AAAA,MAAG,QAAQ;AAAA,MAAyC;AACxE,cAAQ,KAAK;AAAA,IACf,IACA;AAEN,SAAK,IAAI;AAAA,MACP,4CAAuC,KAAK,IAAI,MAAM,YAAY,KAAK,IAAI,WAAW;AAAA,IACxF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAWA,UAAgB;AACd,UAAM,IAAI;AAGV,QAAI,EAAE,kBAAkB,MAAM,KAAM,QAAO,EAAE,kBAAkB;AAE/D,IAAC,KAAK,IAAkD,UAAU;AAClE,SAAK,WAAW;AAChB,SAAK,IAAI,QAAQ,kEAA6D;AAAA,EAChF;AAAA,EAEA,cAAoB;AAClB,SAAK,QAAQ;AAAA,EACf;AAAA,EAEQ,YAAY,OAA+B;AACjD,QAAI,CAAC,KAAK,IAAI,QAAS;AACvB,QAAI,MAAM,SAAS,kBAAkB;AACnC,WAAK,KAAK,SAAS,KAAK;AAAA,IAC1B,WAAW,MAAM,SAAS,aAAa;AACrC,WAAK,WAAW,KAAK;AAAA,IACvB,WAAW,MAAM,SAAS,aAAa,MAAM,SAAS,aAAa;AACjE,WAAK,KAAK,SAAS,KAAK;AAAA,IAC1B;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAMA,MAAc,SAAS,KAA4C;AACjE,UAAM,MAAM,oBAAI,KAAK;AACrB,UAAM,YAAY,IAAI,KAAK,IAAI,QAAQ,IAAI,IAAI,UAAU,EAAE,YAAY;AACvE,UAAM,UAAU,IAAI,YAAY;AAEhC,UAAM,QAAiC;AAAA,MACrC,YAAY,IAAI;AAAA,MAChB,qBAAqB,IAAI;AAAA,MACzB,kBAAkB,IAAI;AAAA,MACtB,QAAQ,IAAI;AAAA,MACZ,YAAY,IAAI;AAAA,MAChB,cAAc,IAAI;AAAA,MAClB,sBAAsB,IAAI;AAAA,MAC1B,mBAAmB,IAAI;AAAA,MACvB,cAAc,IAAI;AAAA,IACpB;AAEA,QAAI,KAAK,IAAI,sBAAsB,IAAI,iBAAiB;AACtD,YAAM,kBAAkB,IAAI;AAAA,IAC9B;AAEA,UAAM,WAAoC;AAAA,MACxC,QAAQ;AAAA,MACR,UAAU,IAAI;AAAA,MACd,cAAc,IAAI;AAAA,MAClB,YAAY,IAAI;AAAA,IAClB;AAEA,QAAI,IAAI,uBAAuB;AAC7B,eAAS,wBAAwB,IAAI;AACrC,eAAS,wBAAwB,IAAI;AAAA,IACvC;AAEA,QAAI,IAAI,QAAS,UAAS,UAAU,IAAI;AAExC,UAAM,UAAU,IAAI,aAAa,KAAK,iBAAiB,IAAI,UAAU,IAAI,OAAO;AAGhF,UAAM,KAAK,YAAY,SAAS,IAAI,cAAc,iBAAiB,WAAW,OAAO;AAErF,UAAM,SAAkC;AAAA,MACtC,UAAU,IAAI;AAAA,MACd,qBAAqB,IAAI;AAAA,MACzB,cAAc,IAAI;AAAA,MAClB,YAAY,IAAI;AAAA,IAClB;AACA,QAAI,KAAK,IAAI,sBAAsB,IAAI,iBAAiB;AACtD,aAAO,kBAAkB,IAAI;AAAA,IAC/B;AAEA,UAAM,OAAO;AAAA,MACX,IAAI,OAAO;AAAA,MACX,UAAU;AAAA,MACV,cAAc,KAAK,IAAI;AAAA,MACvB,MAAM;AAAA,MACN,MAAM;AAAA,MACN,YAAY;AAAA,MACZ,UAAU;AAAA,MACV;AAAA,MACA;AAAA,MACA;AAAA,MACA,MAAM,CAAC,UAAU,QAAQ;AAAA,IAC3B;AAEA,UAAM,cAAc,KAAK,KAAK,CAAC,IAAI,GAAG,KAAK,GAAG;AAAA,EAChD;AAAA;AAAA;AAAA;AAAA,EAMQ,WAAW,KAAgC;AAIjD,SAAK,SAAS,IAAI,IAAI,SAAS;AAAA,MAC7B,WAAW,KAAK,IAAI;AAAA,MACpB,YAAW,oBAAI,KAAK,GAAE,YAAY;AAAA,MAClC,OAAO,IAAI;AAAA,MACX,WAAW,IAAI;AAAA,MACf,OAAO,IAAI;AAAA,MACX,QAAQ,KAAK,iBAAiB,IAAI,UAAU,OAAO;AAAA,MACnD,SAAS,KAAK,iBAAiB,IAAI,OAAO;AAAA,IAC5C,CAAC;AAAA,EACH;AAAA,EAEQ,qBAA2B;AACjC,UAAM,SAAS,KAAK,IAAI,IAAI,cAAa;AACzC,eAAW,CAAC,IAAI,KAAK,KAAK,KAAK,UAAU;AACvC,UAAI,MAAM,YAAY,QAAQ;AAC5B,aAAK,SAAS,OAAO,EAAE;AAAA,MACzB;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAc,SAAS,KAAyC;AAE9D,QAAI,KAAK,SAAS,OAAO,GAAI,MAAK,mBAAmB;AACrD,UAAM,QAAQ,KAAK,SAAS,IAAI,IAAI,OAAO;AAC3C,SAAK,SAAS,OAAO,IAAI,OAAO;AAEhC,UAAM,YAAY,OAAO,cAAa,oBAAI,KAAK,GAAE,YAAY;AAC7D,UAAM,WAAU,oBAAI,KAAK,GAAE,YAAY;AAEvC,UAAM,QAAgC,CAAC;AACvC,QAAI,IAAI,YAAY,SAAS,KAAM,OAAM,gBAAgB,IAAI,WAAW;AACxE,QAAI,IAAI,YAAY,UAAU,KAAM,OAAM,oBAAoB,IAAI,WAAW;AAC7E,QAAI,IAAI,YAAY,SAAS,KAAM,OAAM,eAAe,IAAI,WAAW;AAEvE,UAAM,UAAU,OAAO,WAAW,OAAO;AAGzC,UAAM,KAAK,YAAY,SAAS,UAAU,IAAI,SAAS,IAAI,WAAW,OAAO;AAE7E,UAAM,OAAgC;AAAA,MACpC,IAAI,OAAO,UAAU,OAAO;AAAA,MAC5B,UAAU;AAAA,MACV,cAAc,KAAK,IAAI;AAAA,MACvB,MAAM,UAAU,IAAI,SAAS;AAAA,MAC7B,MAAM;AAAA,MACN,OAAO,IAAI;AAAA,MACX,YAAY;AAAA,MACZ,UAAU;AAAA,MACV,OAAO,OAAO,SAAS,OAAO,EAAE,QAAQ,MAAM,MAAM,IAAI;AAAA,MACxD,QACE,IAAI,SAAS,aAAa,IAAI,UAAU,OACpC,EAAE,YAAY,IAAI,OAAO,IACzB,IAAI,SAAS,cACX,EAAE,OAAO,IAAI,MAAM,IACnB;AAAA,MACR,UAAU;AAAA,QACR,QAAQ;AAAA,QACR,WAAW,IAAI;AAAA,QACf,YAAY,IAAI,eAAe,QAAQ,KAAK,IAAI,IAAI,MAAM,YAAY;AAAA,MACxE;AAAA,MACA,MAAM,CAAC,UAAU,IAAI,SAAS;AAAA,MAC9B,OAAO,OAAO,KAAK,KAAK,EAAE,SAAS,IAAI,QAAQ;AAAA,IACjD;AAEA,UAAM,cAAc,KAAK,KAAK,CAAC,IAAI,GAAG,KAAK,GAAG;AAAA,EAChD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAWA,MAAc,YACZ,SACA,MACA,WACA,SACe;AACf,QAAI,KAAK,cAAc,IAAI,OAAO,EAAG;AAIrC,UAAM,WAAW,KAAK,cAAc,IAAI,OAAO;AAC/C,QAAI,UAAU;AACZ,YAAM;AACN,UAAI,KAAK,cAAc,IAAI,OAAO,EAAG;AAErC,YAAM,WAAW,KAAK,cAAc,IAAI,OAAO;AAC/C,UAAI,UAAU;AACZ,cAAM;AAEN,YAAI,KAAK,cAAc,IAAI,OAAO,EAAG;AAAA,MACvC;AAAA,IAEF;AAEA,UAAM,QAAQ,YAAY;AACxB,YAAM,QAAQ;AAAA,QACZ,IAAI;AAAA,QACJ,cAAc,KAAK,IAAI;AAAA,QACvB;AAAA,QACA,YAAY;AAAA,QACZ,UAAU;AAAA,QACV,MAAM,CAAC,QAAQ;AAAA,MACjB;AAEA,YAAM,KAAK,MAAM,eAAe,KAAK,KAAK,CAAC,KAAK,GAAG,KAAK,GAAG;AAC3D,UAAI,CAAC,IAAI;AACP,aAAK,cAAc,OAAO,OAAO;AACjC;AAAA,MACF;AAEA,WAAK,cAAc,IAAI,OAAO;AAC9B,WAAK,cAAc,OAAO,OAAO;AAGjC,UAAI,KAAK,cAAc,OAAO,KAAQ;AACpC,cAAM,QAAQ,KAAK,cAAc,OAAO,EAAE,KAAK,EAAE;AACjD,YAAI,MAAO,MAAK,cAAc,OAAO,KAAK;AAAA,MAC5C;AAAA,IACF,GAAG;AAEH,SAAK,cAAc,IAAI,SAAS,IAAI;AACpC,UAAM;AAAA,EACR;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASQ,iBAAiB,YAA4B;AACnD,UAAM,SAAS,WAAW,QAAQ,EAAE,OAAO,UAAU,EAAE,OAAO;AAE9D,WAAO,CAAC,IAAK,OAAO,CAAC,IAAI,KAAQ;AAEjC,WAAO,CAAC,IAAK,OAAO,CAAC,IAAI,KAAQ;AACjC,UAAM,MAAM,OAAO,MAAM,GAAG,EAAE,EAAE,SAAS,KAAK;AAC9C,WAAO,GAAG,IAAI,MAAM,GAAG,CAAC,CAAC,IAAI,IAAI,MAAM,GAAG,EAAE,CAAC,IAAI,IAAI,MAAM,IAAI,EAAE,CAAC,IAAI,IAAI,MAAM,IAAI,EAAE,CAAC,IAAI,IAAI,MAAM,EAAE,CAAC;AAAA,EAC1G;AACF;AAeO,SAAS,mBACd,KACA,KACqB;AAErB,MAAI,IAAI,qBAAqB,MAAO,QAAO;AAM3C,QAAM,WAAW,uBAAuB,GAAG;AAE3C,QAAM,SAAS,IAAI,cAAc,SAAS;AAC1C,MAAI,CAAC,QAAQ;AAIX,QAAI,IAAI,qBAAqB,MAAM;AACjC,UAAI;AAAA,QACF;AAAA,MAEF;AAAA,IACF;AACA,WAAO;AAAA,EACT;AAEA,QAAM,MAA0B;AAAA,IAC9B,SAAS;AAAA,IACT;AAAA,IACA,aAAa,IAAI,mBAAmB,SAAS,eAAe;AAAA,IAC5D,eAAe,IAAI,qBAAqB,SAAS,iBAAiB;AAAA,IAClE,QAAQ,IAAI,cAAc,SAAS;AAAA,IACnC,oBAAoB,IAAI,2BAA2B;AAAA,EACrD;AAEA,SAAO,IAAI,aAAa,KAAK,GAAG;AAClC;","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/opik-exporter.ts"],"sourcesContent":["import { createHash, randomBytes } from \"node:crypto\";\n/**\n * opik-exporter.ts — Engram-native Opik trace exporter\n *\n * Subscribes to the globalThis.__openclawEngramTrace slot that Engram already\n * emits on, and forwards recall + LLM events to a self-hosted (or cloud) Opik\n * instance via its REST API. No extra npm dependencies — uses Node's built-in\n * fetch.\n *\n * Auto-detects apiUrl / projectName from the opik-openclaw plugin config so\n * traces land in the same project and are grouped by session thread.\n */\n\nimport { readFileSync } from \"node:fs\";\nimport path from \"node:path\";\n\nimport type { LoggerBackend } from \"./logger.js\";\nimport { readEnvVar, resolveHomeDir } from \"./runtime/env.js\";\n\n// GlobalThis slot that tracks the active OpikExporter instance.\n// Used to make subscribe() idempotent across hot-reload / stop-start cycles.\nconst OPIK_EXPORTER_SLOT = \"__openclawOpikExporter\";\n\n// ---------------------------------------------------------------------------\n// Engram event types (mirrors types.ts without importing from it)\n// ---------------------------------------------------------------------------\n\ntype EngramLlmTraceEvent = {\n kind: \"llm_start\" | \"llm_end\" | \"llm_error\";\n traceId: string;\n model: string;\n operation: \"extraction\" | \"consolidation\" | \"profile_consolidation\" | \"identity_consolidation\" | \"day_summary\";\n input?: string;\n output?: string;\n durationMs?: number;\n error?: string;\n tokenUsage?: { input?: number; output?: number; total?: number };\n};\n\ntype EngramRecallTraceEvent = {\n kind: \"recall_summary\";\n traceId: string;\n operation: \"recall\";\n sessionKey?: string;\n promptLength: number;\n retrievalQueryLength: number;\n recallMode: string;\n recallResultLimit: number;\n qmdEnabled: boolean;\n qmdAvailable: boolean;\n recallNamespaces: string[];\n source: string;\n recalledMemoryCount: number;\n injected: boolean;\n contextChars: number;\n identityInjectionMode?: string;\n identityInjectedChars?: number;\n durationMs: number;\n timings?: Record<string, string>;\n recalledContent?: string;\n};\n\ntype EngramTraceEvent = EngramLlmTraceEvent | EngramRecallTraceEvent;\n\n// ---------------------------------------------------------------------------\n// Config\n// ---------------------------------------------------------------------------\n\nexport interface OpikExporterConfig {\n enabled: boolean;\n /** Base URL of the Opik API, e.g. http://192.168.3.147:5173/api */\n apiUrl: string;\n projectName: string;\n workspaceName: string;\n /** Optional API key (not required for self-hosted deployments) */\n apiKey?: string;\n /** Include recalled memory text in spans (default false) */\n traceRecallContent: boolean;\n}\n\n// ---------------------------------------------------------------------------\n// Auto-detect from opik-openclaw plugin config\n// ---------------------------------------------------------------------------\n\nfunction readOpikOpenclawConfig(log?: LoggerBackend): Partial<OpikExporterConfig> {\n try {\n const configPath =\n readEnvVar(\"OPENCLAW_ENGRAM_CONFIG_PATH\") ||\n readEnvVar(\"OPENCLAW_CONFIG_PATH\") ||\n path.join(resolveHomeDir(), \".openclaw\", \"openclaw.json\");\n const raw = JSON.parse(readFileSync(configPath, \"utf-8\"));\n const entry = raw?.plugins?.entries?.[\"opik-openclaw\"];\n if (!entry?.enabled || !entry?.config) return {};\n const c = entry.config as Record<string, unknown>;\n return {\n apiUrl: typeof c.apiUrl === \"string\" && c.apiUrl.length > 0 ? c.apiUrl : undefined,\n projectName: typeof c.projectName === \"string\" ? c.projectName : undefined,\n workspaceName: typeof c.workspaceName === \"string\" ? c.workspaceName : undefined,\n apiKey: typeof c.apiKey === \"string\" && c.apiKey.length > 0 ? c.apiKey : undefined,\n };\n } catch (err) {\n log?.debug?.(`[opik-exporter] could not read opik-openclaw config: ${err}`);\n return {};\n }\n}\n\n// ---------------------------------------------------------------------------\n// UUID v7 generator (Opik requires v7, not v4)\n// ---------------------------------------------------------------------------\n\nfunction uuidV7(): string {\n const now = Date.now();\n const bytes = randomBytes(16);\n // Timestamp: 48-bit ms since epoch in bytes 0-5\n bytes[0] = (now / 2 ** 40) & 0xff;\n bytes[1] = (now / 2 ** 32) & 0xff;\n bytes[2] = (now / 2 ** 24) & 0xff;\n bytes[3] = (now / 2 ** 16) & 0xff;\n bytes[4] = (now / 2 ** 8) & 0xff;\n bytes[5] = now & 0xff;\n // Version 7\n bytes[6] = (bytes[6] & 0x0f) | 0x70;\n // Variant 10xx\n bytes[8] = (bytes[8] & 0x3f) | 0x80;\n const hex = bytes.toString(\"hex\");\n return `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`;\n}\n\n// ---------------------------------------------------------------------------\n// REST helpers\n// ---------------------------------------------------------------------------\n\nfunction buildHeaders(cfg: OpikExporterConfig): Record<string, string> {\n const headers: Record<string, string> = { \"Content-Type\": \"application/json\" };\n if (cfg.workspaceName) headers[\"Comet-Workspace\"] = cfg.workspaceName;\n // Opik expects the raw API key in the authorization header without a Bearer prefix.\n if (cfg.apiKey) headers[\"authorization\"] = cfg.apiKey;\n return headers;\n}\n\nasync function postSpanBatch(\n cfg: OpikExporterConfig,\n spans: unknown[],\n log: LoggerBackend,\n): Promise<void> {\n const url = `${cfg.apiUrl.replace(/\\/$/, \"\")}/v1/private/spans/batch`;\n try {\n const res = await fetch(url, {\n method: \"POST\",\n headers: buildHeaders(cfg),\n body: JSON.stringify({ spans }),\n });\n if (!res.ok) {\n const text = await res.text().catch(() => \"\");\n log.debug?.(`[opik-exporter] span batch failed ${res.status}: ${text}`);\n }\n } catch (err) {\n log.debug?.(`[opik-exporter] span batch error: ${err}`);\n }\n}\n\nasync function postTraceBatch(\n cfg: OpikExporterConfig,\n traces: unknown[],\n log: LoggerBackend,\n): Promise<boolean> {\n const url = `${cfg.apiUrl.replace(/\\/$/, \"\")}/v1/private/traces/batch`;\n try {\n const res = await fetch(url, {\n method: \"POST\",\n headers: buildHeaders(cfg),\n body: JSON.stringify({ traces }),\n });\n if (!res.ok) {\n const text = await res.text().catch(() => \"\");\n log.debug?.(`[opik-exporter] trace batch failed ${res.status}: ${text}`);\n return false;\n }\n return true;\n } catch (err) {\n log.debug?.(`[opik-exporter] trace batch error: ${err}`);\n return false;\n }\n}\n\n// ---------------------------------------------------------------------------\n// In-flight LLM span tracking\n// ---------------------------------------------------------------------------\n\ntype InFlightLlm = {\n startedAt: number;\n startTime: string;\n model: string;\n operation: string;\n input?: string;\n spanId: string;\n traceId: string;\n};\n\n// ---------------------------------------------------------------------------\n// Main exporter class\n// ---------------------------------------------------------------------------\n\nexport class OpikExporter {\n private readonly cfg: OpikExporterConfig;\n private readonly log: LoggerBackend;\n private _handler: ((e: EngramTraceEvent) => void) | undefined;\n private readonly inFlight = new Map<string, InFlightLlm>();\n /** Track which trace_ids we have already created parent trace objects for. */\n private readonly createdTraces = new Set<string>();\n /** In-flight ensureTrace promises keyed by traceId — concurrent callers await the same promise. */\n private readonly pendingTraces = new Map<string, Promise<void>>();\n /** TTL for in-flight LLM entries (ms). Entries older than this are discarded. */\n private static readonly IN_FLIGHT_TTL_MS = 5 * 60 * 1000; // 5 minutes\n\n constructor(cfg: OpikExporterConfig, log: LoggerBackend) {\n this.cfg = cfg;\n this.log = log;\n }\n\n /**\n * Subscribe to Engram's global trace slot. Safe to call multiple times —\n * chains with existing subscribers (e.g. Langfuse) rather than replacing.\n */\n subscribe(): void {\n if (!this.cfg.enabled) return;\n\n // Idempotency guard: if this exact instance is already registered on globalThis,\n // skip re-wrapping to avoid stacking callbacks on hot-reload / stop-start cycles.\n const g = globalThis as Record<string, unknown>;\n const active = g[OPIK_EXPORTER_SLOT] as OpikExporter | undefined;\n if (active === this) {\n this.log.debug?.(\"[opik-exporter] already subscribed — skipping duplicate\");\n return;\n }\n // If a different (stale) exporter instance is registered, evict it first.\n if (active) active._detach();\n\n g[OPIK_EXPORTER_SLOT] = this;\n\n const existing = g.__openclawEngramTrace as\n | ((e: EngramTraceEvent) => void)\n | undefined;\n\n const handler = (event: EngramTraceEvent) => {\n try {\n this.handleEvent(event);\n } catch (err) {\n this.log.debug?.(`[opik-exporter] handler error: ${err}`);\n }\n };\n // Store handler ref so _detach() can remove it from the chain.\n this._handler = handler;\n\n // Chain: call existing subscribers first, then ours.\n // NOTE: when _detach() is later called, it marks this.cfg.enabled = false\n // rather than restoring the chain, so that any subscriber that chained\n // *after* us is not dropped. The cost is one disabled no-op wrapper\n // remaining in the chain per stop/start cycle. In practice, Engram\n // restarts Opik only a handful of times per process lifetime, so the\n // overhead is negligible (one boolean check per leftover closure).\n g.__openclawEngramTrace =\n typeof existing === \"function\"\n ? (event: EngramTraceEvent) => {\n try { existing(event); } catch { /* prior subscriber threw; continue */ }\n handler(event);\n }\n : handler;\n\n this.log.info(\n `[opik-exporter] subscribed — apiUrl=${this.cfg.apiUrl} project=${this.cfg.projectName}`,\n );\n }\n\n /** Deactivate this exporter instance.\n *\n * We intentionally do NOT restore __openclawEngramTrace to a prior handler\n * because any subscriber that chained *after* us would be silently dropped\n * by such a restoration. Instead we mark the instance as disabled so that\n * handleEvent() becomes a no-op while the callback chain itself is left\n * untouched. The next subscribe() call from a fresh instance will re-enter\n * the chain on top of whatever is there at that point.\n */\n _detach(): void {\n const g = globalThis as Record<string, unknown>;\n // Only clear the global slot if we are still the active exporter.\n // A stale instance must not evict a newer exporter that already took over.\n if (g[OPIK_EXPORTER_SLOT] === this) delete g[OPIK_EXPORTER_SLOT];\n // Disable event processing; the wrapper closure in the chain becomes a no-op.\n (this.cfg as OpikExporterConfig & { enabled: boolean }).enabled = false;\n this._handler = undefined;\n this.log.debug?.(\"[opik-exporter] detached — events silenced, chain preserved\");\n }\n\n unsubscribe(): void {\n this._detach();\n }\n\n private handleEvent(event: EngramTraceEvent): void {\n if (!this.cfg.enabled) return; // guard for after _detach()\n if (event.kind === \"recall_summary\") {\n void this.onRecall(event);\n } else if (event.kind === \"llm_start\") {\n this.onLlmStart(event);\n } else if (event.kind === \"llm_end\" || event.kind === \"llm_error\") {\n void this.onLlmEnd(event);\n }\n }\n\n // -------------------------------------------------------------------------\n // Recall events → general span\n // -------------------------------------------------------------------------\n\n private async onRecall(evt: EngramRecallTraceEvent): Promise<void> {\n const now = new Date();\n const startTime = new Date(now.getTime() - evt.durationMs).toISOString();\n const endTime = now.toISOString();\n\n const input: Record<string, unknown> = {\n recallMode: evt.recallMode,\n recalledMemoryCount: evt.recalledMemoryCount,\n recallNamespaces: evt.recallNamespaces,\n source: evt.source,\n qmdEnabled: evt.qmdEnabled,\n qmdAvailable: evt.qmdAvailable,\n retrievalQueryLength: evt.retrievalQueryLength,\n recallResultLimit: evt.recallResultLimit,\n promptLength: evt.promptLength,\n };\n\n if (this.cfg.traceRecallContent && evt.recalledContent) {\n input.recalledContent = evt.recalledContent;\n }\n\n const metadata: Record<string, unknown> = {\n source: \"engram\",\n injected: evt.injected,\n contextChars: evt.contextChars,\n durationMs: evt.durationMs,\n };\n\n if (evt.identityInjectionMode) {\n metadata.identityInjectionMode = evt.identityInjectionMode;\n metadata.identityInjectedChars = evt.identityInjectedChars;\n }\n\n if (evt.timings) metadata.timings = evt.timings;\n\n const traceId = evt.sessionKey ? this.sessionToTraceId(evt.sessionKey) : uuidV7();\n\n // Ensure parent trace exists so the span is not orphaned in Opik.\n await this.ensureTrace(traceId, evt.sessionKey ?? \"engram:recall\", startTime, endTime);\n\n const output: Record<string, unknown> = {\n injected: evt.injected,\n recalledMemoryCount: evt.recalledMemoryCount,\n contextChars: evt.contextChars,\n durationMs: evt.durationMs,\n };\n if (this.cfg.traceRecallContent && evt.recalledContent) {\n output.recalledContent = evt.recalledContent;\n }\n\n const span = {\n id: uuidV7(),\n trace_id: traceId,\n project_name: this.cfg.projectName,\n name: \"engram:recall\",\n type: \"general\",\n start_time: startTime,\n end_time: endTime,\n input,\n output,\n metadata,\n tags: [\"engram\", \"recall\"],\n };\n\n await postSpanBatch(this.cfg, [span], this.log);\n }\n\n // -------------------------------------------------------------------------\n // Engram internal LLM events (extraction / consolidation) → llm span\n // -------------------------------------------------------------------------\n\n private onLlmStart(evt: EngramLlmTraceEvent): void {\n // Use evt.traceId (Engram's correlation key) as the Opik spanId so spans\n // are deterministic and stable across retries. Derive a per-call Opik\n // trace_id from the same key so each LLM call maps to its own trace.\n this.inFlight.set(evt.traceId, {\n startedAt: Date.now(),\n startTime: new Date().toISOString(),\n model: evt.model,\n operation: evt.operation,\n input: evt.input,\n spanId: this.sessionToTraceId(evt.traceId + \":span\"),\n traceId: this.sessionToTraceId(evt.traceId),\n });\n }\n\n private sweepStaleInFlight(): void {\n const cutoff = Date.now() - OpikExporter.IN_FLIGHT_TTL_MS;\n for (const [id, entry] of this.inFlight) {\n if (entry.startedAt < cutoff) {\n this.inFlight.delete(id);\n }\n }\n }\n\n private async onLlmEnd(evt: EngramLlmTraceEvent): Promise<void> {\n // Opportunistically sweep stale entries to prevent unbounded map growth.\n if (this.inFlight.size > 20) this.sweepStaleInFlight();\n const state = this.inFlight.get(evt.traceId);\n this.inFlight.delete(evt.traceId);\n\n const startTime = state?.startTime ?? new Date().toISOString();\n const endTime = new Date().toISOString();\n\n const usage: Record<string, number> = {};\n if (evt.tokenUsage?.input != null) usage.prompt_tokens = evt.tokenUsage.input;\n if (evt.tokenUsage?.output != null) usage.completion_tokens = evt.tokenUsage.output;\n if (evt.tokenUsage?.total != null) usage.total_tokens = evt.tokenUsage.total;\n\n const traceId = state?.traceId ?? uuidV7();\n\n // Ensure parent trace exists so the span is not orphaned in Opik.\n await this.ensureTrace(traceId, `engram:${evt.operation}`, startTime, endTime);\n\n const span: Record<string, unknown> = {\n id: state?.spanId ?? uuidV7(),\n trace_id: traceId,\n project_name: this.cfg.projectName,\n name: `engram:${evt.operation}`,\n type: \"llm\",\n model: evt.model,\n start_time: startTime,\n end_time: endTime,\n input: state?.input != null ? { prompt: state.input } : undefined,\n output:\n evt.kind === \"llm_end\" && evt.output != null\n ? { completion: evt.output }\n : evt.kind === \"llm_error\"\n ? { error: evt.error }\n : undefined,\n metadata: {\n source: \"engram\",\n operation: evt.operation,\n durationMs: evt.durationMs ?? (state ? Date.now() - state.startedAt : undefined),\n },\n tags: [\"engram\", evt.operation],\n usage: Object.keys(usage).length > 0 ? usage : undefined,\n };\n\n await postSpanBatch(this.cfg, [span], this.log);\n }\n\n // -------------------------------------------------------------------------\n // Helpers\n // -------------------------------------------------------------------------\n\n /**\n * Create a parent trace object in Opik if we haven't already.\n * Opik does NOT auto-create traces from spans, so without this\n * the spans are orphaned and don't show up in the trace list UI.\n */\n private async ensureTrace(\n traceId: string,\n name: string,\n startTime: string,\n endTime: string,\n ): Promise<void> {\n if (this.createdTraces.has(traceId)) return;\n\n // If another call is already creating this trace, wait for it and then\n // re-check — if the first attempt failed we'll retry below.\n const existing = this.pendingTraces.get(traceId);\n if (existing) {\n await existing;\n if (this.createdTraces.has(traceId)) return;\n // First attempt failed — check if another waiter is already retrying.\n const retrying = this.pendingTraces.get(traceId);\n if (retrying) {\n await retrying;\n // Only return if the retry actually succeeded.\n if (this.createdTraces.has(traceId)) return;\n }\n // No one else retrying; fall through to create.\n }\n\n const work = (async () => {\n const trace = {\n id: traceId,\n project_name: this.cfg.projectName,\n name,\n start_time: startTime,\n end_time: endTime,\n tags: [\"engram\"],\n };\n\n const ok = await postTraceBatch(this.cfg, [trace], this.log);\n if (!ok) {\n this.pendingTraces.delete(traceId);\n return; // transient failure — allow retry on next span\n }\n\n this.createdTraces.add(traceId);\n this.pendingTraces.delete(traceId);\n\n // Cap the set size to prevent unbounded growth in long-running processes.\n if (this.createdTraces.size > 10_000) {\n const first = this.createdTraces.values().next().value;\n if (first) this.createdTraces.delete(first);\n }\n })();\n\n this.pendingTraces.set(traceId, work);\n await work;\n }\n\n /**\n * Convert a sessionKey to a stable, deterministic UUID v7-shaped ID so that\n * spans for the same session share a trace_id and are threaded in Opik.\n * Uses SHA-256 for collision resistance. The timestamp bytes (0-5) come from\n * the hash itself — they don't reflect real time, but Opik only validates\n * the version/variant bits, not timestamp ordering.\n */\n private sessionToTraceId(sessionKey: string): string {\n const digest = createHash(\"sha256\").update(sessionKey).digest();\n // Version 7 (bytes 0-5 are hash-derived, not real timestamps)\n digest[6] = (digest[6] & 0x0f) | 0x70;\n // Variant 10xx\n digest[8] = (digest[8] & 0x3f) | 0x80;\n const hex = digest.slice(0, 16).toString(\"hex\");\n return `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`;\n }\n}\n\n// ---------------------------------------------------------------------------\n// Factory: build from Engram resolved config + auto-detect from opik-openclaw\n// ---------------------------------------------------------------------------\n\nexport interface OpikExporterRawConfig {\n opikTraceEnabled?: boolean;\n opikApiUrl?: string;\n opikProjectName?: string;\n opikWorkspaceName?: string;\n opikApiKey?: string;\n opikTraceRecallContent?: boolean;\n}\n\nexport function createOpikExporter(\n raw: OpikExporterRawConfig,\n log: LoggerBackend,\n): OpikExporter | null {\n // Explicit opt-out: if set to false, never enable.\n if (raw.opikTraceEnabled === false) return null;\n\n // Auto-detect from opik-openclaw plugin config.\n // If opikTraceEnabled is not set (undefined), we enable automatically\n // whenever opik-openclaw is configured — avoids needing extra fields in\n // Engram's openclaw.json config section (which would fail schema validation).\n const detected = readOpikOpenclawConfig(log);\n\n const apiUrl = raw.opikApiUrl ?? detected.apiUrl;\n if (!apiUrl) {\n // Only warn when the user explicitly opted in but forgot to set the URL.\n // When auto-detecting (opikTraceEnabled not set), return silently so non-Opik\n // users see no log noise on every startup.\n if (raw.opikTraceEnabled === true) {\n log.warn(\n \"[opik-exporter] opikTraceEnabled=true but no apiUrl found — \" +\n \"set opikApiUrl in Engram config or ensure opik-openclaw plugin is configured\",\n );\n }\n return null;\n }\n\n const cfg: OpikExporterConfig = {\n enabled: true,\n apiUrl,\n projectName: raw.opikProjectName ?? detected.projectName ?? \"openclaw\",\n workspaceName: raw.opikWorkspaceName ?? detected.workspaceName ?? \"default\",\n apiKey: raw.opikApiKey ?? detected.apiKey,\n traceRecallContent: raw.opikTraceRecallContent === true,\n };\n\n return new OpikExporter(cfg, log);\n}\n"],"mappings":"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","names":[]}
|