@unerr-ai/unerr 0.2.1 → 0.2.2
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/README.md +6 -0
- package/dist/cli.js +37236 -35793
- package/package.json +1 -1
- package/dist/behaviors/agent-llm-bridge.js +0 -166
- package/dist/behaviors/architecture-guard.js +0 -256
- package/dist/behaviors/auto-doc.js +0 -247
- package/dist/behaviors/cascade-guard.js +0 -289
- package/dist/behaviors/change-narrative.js +0 -270
- package/dist/behaviors/convention-drift.js +0 -290
- package/dist/behaviors/framework.js +0 -235
- package/dist/behaviors/guard-formatter.js +0 -44
- package/dist/behaviors/incomplete-work.js +0 -270
- package/dist/behaviors/loop-breaker.js +0 -300
- package/dist/behaviors/session-continuity.js +0 -208
- package/dist/commands/branches.js +0 -97
- package/dist/commands/check-commit.js +0 -225
- package/dist/commands/compress-output.js +0 -64
- package/dist/commands/config-verify.js +0 -243
- package/dist/commands/daemon.js +0 -905
- package/dist/commands/dashboard.js +0 -52
- package/dist/commands/debug.js +0 -200
- package/dist/commands/enrich.js +0 -184
- package/dist/commands/exec.js +0 -233
- package/dist/commands/gain.js +0 -156
- package/dist/commands/hook.js +0 -88
- package/dist/commands/index.js +0 -88
- package/dist/commands/init.js +0 -74
- package/dist/commands/install.js +0 -505
- package/dist/commands/learn.js +0 -116
- package/dist/commands/manifest.js +0 -193
- package/dist/commands/rewind.js +0 -103
- package/dist/commands/serve.js +0 -19
- package/dist/commands/setup-wizard.js +0 -414
- package/dist/commands/skills.js +0 -64
- package/dist/commands/stats.js +0 -20
- package/dist/commands/status.js +0 -654
- package/dist/commands/timeline.js +0 -139
- package/dist/commands/uninstall.js +0 -230
- package/dist/components/App.js +0 -109
- package/dist/components/Banner.js +0 -12
- package/dist/components/ConfirmPrompt.js +0 -25
- package/dist/components/DriftSummary.js +0 -23
- package/dist/components/GradeBadge.js +0 -15
- package/dist/components/HealthCard.js +0 -18
- package/dist/components/InkSpinner.js +0 -22
- package/dist/components/InputBox.js +0 -17
- package/dist/components/KeyValue.js +0 -13
- package/dist/components/MessageList.js +0 -14
- package/dist/components/ProgressBar.js +0 -26
- package/dist/components/Section.js +0 -16
- package/dist/components/SessionSummaryCard.js +0 -73
- package/dist/components/StartupDisplay.js +0 -24
- package/dist/components/StatusDashboard.js +0 -57
- package/dist/components/StatusLine.js +0 -8
- package/dist/components/StepLine.js +0 -22
- package/dist/components/Theme.js +0 -20
- package/dist/components/ToolProgress.js +0 -8
- package/dist/components/ViolationList.js +0 -21
- package/dist/components/render.js +0 -13
- package/dist/config/agent-registry.js +0 -237
- package/dist/config/claude-settings-hooks.js +0 -304
- package/dist/config/hook-installer.js +0 -65
- package/dist/config/instruction-writer.js +0 -388
- package/dist/config/mcp-config-writer.js +0 -266
- package/dist/config/settings.js +0 -174
- package/dist/config/tool-detector.js +0 -42
- package/dist/config/value-surfacing.js +0 -119
- package/dist/core/context-assembly.js +0 -108
- package/dist/core/conversation.js +0 -33
- package/dist/core/local-chat-provider.js +0 -475
- package/dist/core/provider-factory.js +0 -55
- package/dist/core/providers.js +0 -90
- package/dist/core/query-engine.js +0 -174
- package/dist/daemon/api.js +0 -312
- package/dist/daemon/autostart.js +0 -119
- package/dist/daemon/bootstrap.js +0 -39
- package/dist/daemon/client.js +0 -164
- package/dist/daemon/detect-ci.js +0 -81
- package/dist/daemon/platform-linux.js +0 -146
- package/dist/daemon/platform-macos.js +0 -134
- package/dist/daemon/platform-windows.js +0 -116
- package/dist/daemon/process-manager.js +0 -299
- package/dist/daemon/protocol.js +0 -23
- package/dist/daemon/registry.js +0 -270
- package/dist/daemon/settings-schema.js +0 -72
- package/dist/daemon/system-health.js +0 -134
- package/dist/daemon/version-checker.js +0 -262
- package/dist/daemon/warm-start.js +0 -223
- package/dist/entrypoints/cli.js +0 -1043
- package/dist/entrypoints/daemon.js +0 -380
- package/dist/entrypoints/repl.js +0 -147
- package/dist/hooks/adapters/claude-code.js +0 -90
- package/dist/hooks/adapters/cline.js +0 -100
- package/dist/hooks/adapters/cursor.js +0 -98
- package/dist/hooks/hook-dedup.js +0 -79
- package/dist/hooks/hook-runner.js +0 -113
- package/dist/hooks/navigation-hooks.js +0 -175
- package/dist/hooks/prompt-hooks.js +0 -63
- package/dist/hooks/shell-hooks.js +0 -47
- package/dist/ignore.js +0 -111
- package/dist/intelligence/approach-suggester.js +0 -61
- package/dist/intelligence/ast-extractor.js +0 -2615
- package/dist/intelligence/ast-worker.js +0 -34
- package/dist/intelligence/background-indexer.js +0 -121
- package/dist/intelligence/blast-radius.js +0 -200
- package/dist/intelligence/community-detection.js +0 -691
- package/dist/intelligence/community-detector.js +0 -184
- package/dist/intelligence/computation-scheduler.js +0 -75
- package/dist/intelligence/confidence-propagation.js +0 -47
- package/dist/intelligence/convention-detector.js +0 -242
- package/dist/intelligence/convention-learner.js +0 -205
- package/dist/intelligence/convention-matcher.js +0 -205
- package/dist/intelligence/cozo-schema.js +0 -376
- package/dist/intelligence/decision-point-detector.js +0 -90
- package/dist/intelligence/deep-dive-tools.js +0 -586
- package/dist/intelligence/durability-scorer.js +0 -84
- package/dist/intelligence/exploration-cost.js +0 -204
- package/dist/intelligence/exploration-pattern-tracker.js +0 -61
- package/dist/intelligence/fact-generator.js +0 -322
- package/dist/intelligence/facts-schema.js +0 -90
- package/dist/intelligence/file-intelligence.js +0 -59
- package/dist/intelligence/graph-holder.js +0 -220
- package/dist/intelligence/graph-temporal-joiner.js +0 -238
- package/dist/intelligence/health-grade.js +0 -423
- package/dist/intelligence/health-grader.js +0 -200
- package/dist/intelligence/health-map-data.js +0 -259
- package/dist/intelligence/import-symbols.js +0 -136
- package/dist/intelligence/incremental-indexer.js +0 -658
- package/dist/intelligence/indexer/centrality.js +0 -62
- package/dist/intelligence/indexer/cfg-context.js +0 -95
- package/dist/intelligence/indexer/confidence.js +0 -34
- package/dist/intelligence/indexer/cross-file-resolver.js +0 -104
- package/dist/intelligence/indexer/edge-repair.js +0 -89
- package/dist/intelligence/indexer/entity-key.js +0 -17
- package/dist/intelligence/indexer/export-map.js +0 -132
- package/dist/intelligence/indexer/git-cochange.js +0 -128
- package/dist/intelligence/indexer/graph-patch.js +0 -147
- package/dist/intelligence/indexer/incremental.js +0 -78
- package/dist/intelligence/indexer/ingest.js +0 -160
- package/dist/intelligence/indexer/language-detect.js +0 -226
- package/dist/intelligence/indexer/metadata.js +0 -63
- package/dist/intelligence/indexer/mutation-tracker.js +0 -79
- package/dist/intelligence/indexer/orchestrator.js +0 -155
- package/dist/intelligence/indexer/plugin-interface.js +0 -31
- package/dist/intelligence/indexer/plugins/csharp.js +0 -440
- package/dist/intelligence/indexer/plugins/go.js +0 -335
- package/dist/intelligence/indexer/plugins/java.js +0 -370
- package/dist/intelligence/indexer/plugins/python.js +0 -358
- package/dist/intelligence/indexer/plugins/regex-fallback.js +0 -82
- package/dist/intelligence/indexer/plugins/ruby.js +0 -290
- package/dist/intelligence/indexer/plugins/rust.js +0 -484
- package/dist/intelligence/indexer/plugins/tier2-generic.js +0 -310
- package/dist/intelligence/indexer/plugins/typescript.js +0 -456
- package/dist/intelligence/indexer/resource-monitor.js +0 -93
- package/dist/intelligence/indexer/scip/decoder.js +0 -253
- package/dist/intelligence/indexer/scip/detector.js +0 -232
- package/dist/intelligence/indexer/scip/downloader.js +0 -427
- package/dist/intelligence/indexer/scip/fallback.js +0 -34
- package/dist/intelligence/indexer/scip/merger.js +0 -109
- package/dist/intelligence/indexer/scip/orchestrator.js +0 -433
- package/dist/intelligence/indexer/scip/runner.js +0 -98
- package/dist/intelligence/indexer/snapshot.js +0 -66
- package/dist/intelligence/indexer/test-detector.js +0 -196
- package/dist/intelligence/indexer/watch-integration.js +0 -61
- package/dist/intelligence/indexer/worker.js +0 -85
- package/dist/intelligence/local-convention-detector.js +0 -437
- package/dist/intelligence/local-embeddings.js +0 -190
- package/dist/intelligence/local-graph.js +0 -1946
- package/dist/intelligence/local-indexer.js +0 -1575
- package/dist/intelligence/local-llm.js +0 -163
- package/dist/intelligence/local-rule-generator.js +0 -154
- package/dist/intelligence/local-snapshot.js +0 -213
- package/dist/intelligence/negative-knowledge.js +0 -103
- package/dist/intelligence/persistent-db.js +0 -85
- package/dist/intelligence/query-router.js +0 -2556
- package/dist/intelligence/risk-classifier.js +0 -116
- package/dist/intelligence/rule-evaluator.js +0 -380
- package/dist/intelligence/rule-generator.js +0 -49
- package/dist/intelligence/search-index.js +0 -173
- package/dist/intelligence/semantic/docstring-extractor.js +0 -67
- package/dist/intelligence/semantic/embedding-store.js +0 -52
- package/dist/intelligence/semantic/enrichment-orchestrator.js +0 -48
- package/dist/intelligence/semantic/git-message-miner.js +0 -114
- package/dist/intelligence/semantic/identifier-tokenizer.js +0 -51
- package/dist/intelligence/semantic/node2vec-embeddings.js +0 -71
- package/dist/intelligence/semantic/node2vec-walks.js +0 -103
- package/dist/intelligence/semantic/path-domain-inference.js +0 -112
- package/dist/intelligence/semantic/similarity-engine.js +0 -60
- package/dist/intelligence/semantic/tfidf-vectors.js +0 -88
- package/dist/intelligence/session-brief-builder.js +0 -159
- package/dist/intelligence/session-context.js +0 -221
- package/dist/intelligence/session-health-monitor.js +0 -211
- package/dist/intelligence/session-narrative.js +0 -197
- package/dist/intelligence/session-pattern-analyzer.js +0 -218
- package/dist/intelligence/signal-scorer.js +0 -390
- package/dist/intelligence/signal-show-store.js +0 -182
- package/dist/intelligence/smart-truncate.js +0 -158
- package/dist/intelligence/subgraph-cache.js +0 -88
- package/dist/intelligence/temporal-facts.js +0 -494
- package/dist/intelligence/token-estimator.js +0 -100
- package/dist/intelligence/tool-injector.js +0 -87
- package/dist/intelligence/tree-sitter-loader.js +0 -71
- package/dist/intelligence/worker-pool.js +0 -116
- package/dist/proxy/arg-validator.js +0 -79
- package/dist/proxy/auto-bootstrap.js +0 -167
- package/dist/proxy/bridge.js +0 -147
- package/dist/proxy/budget-enforcer.js +0 -70
- package/dist/proxy/compression-quality-monitor.js +0 -160
- package/dist/proxy/compression-stats.js +0 -51
- package/dist/proxy/context-rot-detector.js +0 -137
- package/dist/proxy/drift-detector.js +0 -139
- package/dist/proxy/efficiency-tracker.js +0 -79
- package/dist/proxy/fact-ranking.js +0 -154
- package/dist/proxy/format-encoder.js +0 -266
- package/dist/proxy/http-transport.js +0 -90
- package/dist/proxy/lifecycle-actor.js +0 -55
- package/dist/proxy/lifecycle-machine.js +0 -187
- package/dist/proxy/log-tailer.js +0 -265
- package/dist/proxy/model-pricing.js +0 -98
- package/dist/proxy/network-firewall.js +0 -141
- package/dist/proxy/nudge-state.js +0 -93
- package/dist/proxy/output-compressor.js +0 -185
- package/dist/proxy/pid-lock.js +0 -291
- package/dist/proxy/proxy-context.js +0 -11
- package/dist/proxy/proxy.js +0 -2633
- package/dist/proxy/response-enrichment.js +0 -32
- package/dist/proxy/response-envelope.js +0 -313
- package/dist/proxy/session-dedup.js +0 -82
- package/dist/proxy/session-legend.js +0 -30
- package/dist/proxy/session-persistence.js +0 -210
- package/dist/proxy/session-resume.js +0 -94
- package/dist/proxy/session-stats.js +0 -513
- package/dist/proxy/shell-classifier.js +0 -1346
- package/dist/proxy/shell-compression-log.js +0 -93
- package/dist/proxy/shell-compressor.js +0 -390
- package/dist/proxy/shell-graph-boost.js +0 -202
- package/dist/proxy/shell-monitor-map.js +0 -18
- package/dist/proxy/shell-stats.js +0 -54
- package/dist/proxy/shell-strategies/cloud.js +0 -215
- package/dist/proxy/shell-strategies/diff.js +0 -159
- package/dist/proxy/shell-strategies/error-diagnostic.js +0 -796
- package/dist/proxy/shell-strategies/filter-dsl.js +0 -358
- package/dist/proxy/shell-strategies/git-status.js +0 -177
- package/dist/proxy/shell-strategies/key-value.js +0 -193
- package/dist/proxy/shell-strategies/log-text.js +0 -154
- package/dist/proxy/shell-strategies/omni.js +0 -188
- package/dist/proxy/shell-strategies/progress.js +0 -55
- package/dist/proxy/shell-strategies/redact.js +0 -76
- package/dist/proxy/shell-strategies/structured.js +0 -241
- package/dist/proxy/shell-strategies/tabular.js +0 -243
- package/dist/proxy/shell-strategies/test-results-types.js +0 -13
- package/dist/proxy/shell-strategies/test-results.js +0 -784
- package/dist/proxy/shell-strategies/tree-paths.js +0 -144
- package/dist/proxy/shell-strategies/yaml.js +0 -182
- package/dist/proxy/shell-tee.js +0 -111
- package/dist/proxy/signal-dedup.js +0 -171
- package/dist/proxy/startup-renderer.js +0 -158
- package/dist/proxy/task-token-display.js +0 -38
- package/dist/proxy/token-counter.js +0 -61
- package/dist/proxy/tool-clusters.js +0 -273
- package/dist/proxy/tool-definitions.js +0 -525
- package/dist/proxy/transport-mux.js +0 -229
- package/dist/proxy/wire-cap.js +0 -268
- package/dist/rules/developer.mozilla.org.json +0 -9
- package/dist/rules/github.com.json +0 -21
- package/dist/schemas/api/skills.js +0 -19
- package/dist/schemas/common/errors.js +0 -7
- package/dist/schemas/common/headers.js +0 -5
- package/dist/schemas/entities/edge.js +0 -25
- package/dist/schemas/entities/entity.js +0 -22
- package/dist/schemas/entities/rule.js +0 -18
- package/dist/schemas/index.js +0 -14
- package/dist/server/event-bus.js +0 -59
- package/dist/server/http.js +0 -156
- package/dist/server/middleware.js +0 -70
- package/dist/server/routes/drift.js +0 -97
- package/dist/server/routes/intelligence.js +0 -1217
- package/dist/server/routes/reasoning-quality.js +0 -444
- package/dist/server/routes/session.js +0 -86
- package/dist/server/routes/stream.js +0 -120
- package/dist/server/routes/system.js +0 -73
- package/dist/server/routes/temporal.js +0 -170
- package/dist/server/routes/timeline.js +0 -232
- package/dist/server/routes/token-flow.js +0 -403
- package/dist/skills/effectiveness-tracker.js +0 -93
- package/dist/skills/local-pack.js +0 -380
- package/dist/skills/resolver.js +0 -495
- package/dist/state-detector.js +0 -83
- package/dist/timeline/intent-detector.js +0 -263
- package/dist/timeline/loop-miner.js +0 -140
- package/dist/timeline/open-threads.js +0 -49
- package/dist/timeline/signal-reinforcer.js +0 -62
- package/dist/timeline/timeline-bootstrap.js +0 -151
- package/dist/timeline/timeline-store.js +0 -618
- package/dist/tools/coding/bash.js +0 -49
- package/dist/tools/coding/file-edit.js +0 -72
- package/dist/tools/coding/file-outline.js +0 -227
- package/dist/tools/coding/file-read-protocol.js +0 -425
- package/dist/tools/coding/file-read.js +0 -35
- package/dist/tools/coding/file-write.js +0 -43
- package/dist/tools/coding/glob-tool.js +0 -109
- package/dist/tools/coding/grep.js +0 -162
- package/dist/tools/coding/index.js +0 -27
- package/dist/tools/intelligence/index.js +0 -269
- package/dist/tools/intelligence/record-fact.js +0 -48
- package/dist/tools/intelligence/timeline-markers.js +0 -130
- package/dist/tools/registry.js +0 -47
- package/dist/tools/types.js +0 -8
- package/dist/tracking/auto-snapshot-triggers.js +0 -246
- package/dist/tracking/branch-context.js +0 -115
- package/dist/tracking/branch-snapshot.js +0 -217
- package/dist/tracking/causal-bridge.js +0 -317
- package/dist/tracking/circuit-breaker.js +0 -147
- package/dist/tracking/commit-watcher.js +0 -114
- package/dist/tracking/context-ledger.js +0 -119
- package/dist/tracking/correction-detector.js +0 -324
- package/dist/tracking/drift-tracker.js +0 -874
- package/dist/tracking/durability-tracker.js +0 -94
- package/dist/tracking/entity-rewind.js +0 -200
- package/dist/tracking/file-hash-state.js +0 -114
- package/dist/tracking/git-attribution.js +0 -132
- package/dist/tracking/git-trailers.js +0 -171
- package/dist/tracking/intelligence-counter.js +0 -46
- package/dist/tracking/intent-correlator.js +0 -202
- package/dist/tracking/intent-encoder.js +0 -52
- package/dist/tracking/intent-token-tracker.js +0 -159
- package/dist/tracking/ledger-archiver.js +0 -94
- package/dist/tracking/ledger-chains.js +0 -245
- package/dist/tracking/metrics-store.js +0 -361
- package/dist/tracking/native-watcher.js +0 -131
- package/dist/tracking/offline-rewind.js +0 -295
- package/dist/tracking/pending-violations.js +0 -74
- package/dist/tracking/persistence-effectiveness.js +0 -167
- package/dist/tracking/prompt-durability.js +0 -202
- package/dist/tracking/quality-signals.js +0 -213
- package/dist/tracking/redactor.js +0 -73
- package/dist/tracking/rewind-engine.js +0 -161
- package/dist/tracking/session-history.js +0 -128
- package/dist/tracking/session-receipt.js +0 -88
- package/dist/tracking/session-summary-writer.js +0 -157
- package/dist/tracking/shadow-ledger.js +0 -321
- package/dist/tracking/stash-manager.js +0 -258
- package/dist/tracking/timeline-fork.js +0 -213
- package/dist/tracking/timeline.js +0 -69
- package/dist/tracking/token-flow.js +0 -276
- package/dist/tracking/turn-segmenter.js +0 -122
- package/dist/tracking/weekly-accumulator.js +0 -179
- package/dist/tracking/working-snapshots.js +0 -188
- package/dist/tracking/workspace-manifest.js +0 -176
- package/dist/transport/http.js +0 -102
- package/dist/utils/counterfactual.js +0 -65
- package/dist/utils/deep-link.js +0 -34
- package/dist/utils/detect.js +0 -193
- package/dist/utils/exec.js +0 -73
- package/dist/utils/file-logger.js +0 -87
- package/dist/utils/format-error.js +0 -29
- package/dist/utils/git.js +0 -181
- package/dist/utils/log.js +0 -57
- package/dist/utils/logger.js +0 -35
- package/dist/utils/mcp-content-json.js +0 -8
- package/dist/utils/session-logger.js +0 -154
- package/dist/utils/startup-log.js +0 -512
- package/dist/utils/ui.js +0 -56
|
@@ -1,131 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Native file watcher — @parcel/watcher with debounced event emission.
|
|
3
|
-
*
|
|
4
|
-
* Uses OS-level file watching (FSEvents on macOS, inotify on Linux, ReadDirectoryChanges on Windows)
|
|
5
|
-
* for millisecond-scale change detection. Events are debounced into batches (default 50ms window)
|
|
6
|
-
* and deduplicated before emission.
|
|
7
|
-
*
|
|
8
|
-
* Respects .gitignore patterns and always excludes .unerr/, node_modules/, .git/.
|
|
9
|
-
*/
|
|
10
|
-
import fs from "node:fs";
|
|
11
|
-
import path from "node:path";
|
|
12
|
-
import { createModuleLogger } from "../utils/logger.js";
|
|
13
|
-
const log = createModuleLogger("native-watcher");
|
|
14
|
-
const ALWAYS_IGNORE_DIRS = [".unerr", "node_modules", ".git"];
|
|
15
|
-
function loadGitignorePatterns(projectRoot) {
|
|
16
|
-
const gitignorePath = path.join(projectRoot, ".gitignore");
|
|
17
|
-
try {
|
|
18
|
-
if (!fs.existsSync(gitignorePath))
|
|
19
|
-
return [];
|
|
20
|
-
const content = fs.readFileSync(gitignorePath, "utf-8");
|
|
21
|
-
return content
|
|
22
|
-
.split("\n")
|
|
23
|
-
.map((line) => line.trim())
|
|
24
|
-
.filter((line) => line && !line.startsWith("#"));
|
|
25
|
-
}
|
|
26
|
-
catch {
|
|
27
|
-
return [];
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
/**
|
|
31
|
-
* Build ignore list for @parcel/watcher. Uses relative directory names
|
|
32
|
-
* so the wrapper resolves them against the (real) watched directory,
|
|
33
|
-
* avoiding symlink mismatches (e.g. /tmp → /private/tmp on macOS).
|
|
34
|
-
*/
|
|
35
|
-
function buildIgnoreList(projectRoot, extra = []) {
|
|
36
|
-
const patterns = [...ALWAYS_IGNORE_DIRS];
|
|
37
|
-
for (const pattern of loadGitignorePatterns(projectRoot)) {
|
|
38
|
-
patterns.push(pattern.startsWith("/") ? pattern.slice(1) : pattern);
|
|
39
|
-
}
|
|
40
|
-
for (const pattern of extra) {
|
|
41
|
-
patterns.push(pattern);
|
|
42
|
-
}
|
|
43
|
-
return patterns;
|
|
44
|
-
}
|
|
45
|
-
export function createNativeWatcher(opts) {
|
|
46
|
-
const { projectRoot, debounceMs = 50, ignorePatterns = [], onEvents } = opts;
|
|
47
|
-
let subscription = null;
|
|
48
|
-
let running = false;
|
|
49
|
-
let debounceTimer = null;
|
|
50
|
-
let pendingEvents = [];
|
|
51
|
-
function flushEvents() {
|
|
52
|
-
if (pendingEvents.length === 0)
|
|
53
|
-
return;
|
|
54
|
-
const deduped = new Map();
|
|
55
|
-
for (const evt of pendingEvents) {
|
|
56
|
-
deduped.set(evt.path, evt);
|
|
57
|
-
}
|
|
58
|
-
const batch = Array.from(deduped.values());
|
|
59
|
-
pendingEvents = [];
|
|
60
|
-
try {
|
|
61
|
-
onEvents(batch);
|
|
62
|
-
}
|
|
63
|
-
catch (err) {
|
|
64
|
-
log.error("Error in onEvents callback:", err);
|
|
65
|
-
}
|
|
66
|
-
}
|
|
67
|
-
function scheduleFlush() {
|
|
68
|
-
if (debounceTimer !== null)
|
|
69
|
-
return;
|
|
70
|
-
debounceTimer = setTimeout(() => {
|
|
71
|
-
debounceTimer = null;
|
|
72
|
-
flushEvents();
|
|
73
|
-
}, debounceMs);
|
|
74
|
-
debounceTimer.unref();
|
|
75
|
-
}
|
|
76
|
-
async function start() {
|
|
77
|
-
if (running)
|
|
78
|
-
return;
|
|
79
|
-
let subscribeFn;
|
|
80
|
-
try {
|
|
81
|
-
const mod = await import("@parcel/watcher");
|
|
82
|
-
subscribeFn = mod.subscribe;
|
|
83
|
-
}
|
|
84
|
-
catch (err) {
|
|
85
|
-
log.warn("@parcel/watcher native bindings unavailable — file watching disabled.", err instanceof Error ? err.message : String(err));
|
|
86
|
-
return;
|
|
87
|
-
}
|
|
88
|
-
const resolvedRoot = fs.realpathSync(projectRoot);
|
|
89
|
-
const ignore = buildIgnoreList(resolvedRoot, ignorePatterns);
|
|
90
|
-
log.debug("Starting native watcher", { projectRoot: resolvedRoot, ignore });
|
|
91
|
-
subscription = await subscribeFn(resolvedRoot, (err, events) => {
|
|
92
|
-
if (err) {
|
|
93
|
-
log.error("Watcher error:", err);
|
|
94
|
-
return;
|
|
95
|
-
}
|
|
96
|
-
for (const evt of events) {
|
|
97
|
-
const rel = path.relative(resolvedRoot, evt.path);
|
|
98
|
-
const segs = rel.split(path.sep);
|
|
99
|
-
if (segs.some((s) => ALWAYS_IGNORE_DIRS.includes(s)))
|
|
100
|
-
continue;
|
|
101
|
-
pendingEvents.push({
|
|
102
|
-
type: evt.type,
|
|
103
|
-
path: evt.path,
|
|
104
|
-
});
|
|
105
|
-
}
|
|
106
|
-
if (pendingEvents.length > 0)
|
|
107
|
-
scheduleFlush();
|
|
108
|
-
}, { ignore });
|
|
109
|
-
running = true;
|
|
110
|
-
log.debug("Native watcher started");
|
|
111
|
-
}
|
|
112
|
-
async function stop() {
|
|
113
|
-
if (!running)
|
|
114
|
-
return;
|
|
115
|
-
if (debounceTimer !== null) {
|
|
116
|
-
clearTimeout(debounceTimer);
|
|
117
|
-
debounceTimer = null;
|
|
118
|
-
flushEvents();
|
|
119
|
-
}
|
|
120
|
-
if (subscription) {
|
|
121
|
-
await subscription.unsubscribe();
|
|
122
|
-
subscription = null;
|
|
123
|
-
}
|
|
124
|
-
running = false;
|
|
125
|
-
log.debug("Native watcher stopped");
|
|
126
|
-
}
|
|
127
|
-
function isRunning() {
|
|
128
|
-
return running;
|
|
129
|
-
}
|
|
130
|
-
return { start, stop, isRunning };
|
|
131
|
-
}
|
|
@@ -1,295 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Offline Rewind — Zero-Network rewind using CozoDB + local git.
|
|
3
|
-
* Phase 5.5 §1.3.4 (P5.5-CB-03)
|
|
4
|
-
*
|
|
5
|
-
* When the user is offline, this module performs a filesystem-level rewind
|
|
6
|
-
* using the local CozoDB graph for blast radius computation and git for
|
|
7
|
-
* file restoration.
|
|
8
|
-
*
|
|
9
|
-
* Integration Points:
|
|
10
|
-
* • P10 Bridge: Reads shadow ledger entries from .unerr/ledger/shadow.jsonl
|
|
11
|
-
* • P13 Bridge: Uses `git checkout {ref} -- {files}` for atomic file restore
|
|
12
|
-
* • P5.5 Bridge: On connectivity restore, flushes simulated rewind via sync-intent
|
|
13
|
-
*
|
|
14
|
-
* Invariant: Offline rewind produces identical file state as online rewind.
|
|
15
|
-
* Difference: Anti-pattern rule synthesis deferred until connectivity (requires LLM).
|
|
16
|
-
*
|
|
17
|
-
* Blast radius resolution target: <200ms (CozoDB is in-process, no network).
|
|
18
|
-
*/
|
|
19
|
-
import { randomBytes } from "node:crypto";
|
|
20
|
-
import { appendFileSync, existsSync, mkdirSync, readFileSync, writeFileSync, } from "node:fs";
|
|
21
|
-
import { join } from "node:path";
|
|
22
|
-
import { checkoutFile } from "../utils/git.js";
|
|
23
|
-
import { createModuleLogger } from "../utils/logger.js";
|
|
24
|
-
const log = createModuleLogger("offline-rewind");
|
|
25
|
-
// ── Core Logic ─────────────────────────────────────────────────────────────
|
|
26
|
-
/**
|
|
27
|
-
* Perform an offline rewind using local resources only.
|
|
28
|
-
*
|
|
29
|
-
* Steps (per §1.3.4):
|
|
30
|
-
* 1. Read local ledger entries from shadow.jsonl
|
|
31
|
-
* 2. Identify the target entry and entries to revert
|
|
32
|
-
* 3. Compute blast radius using CozoDB (entity lookup + 1-hop callers)
|
|
33
|
-
* 4. If not dry run: apply file restores via `git checkout`
|
|
34
|
-
* 5. Update local state: shadow.jsonl, branch_context.json, drift overlay
|
|
35
|
-
*/
|
|
36
|
-
export async function offlineRewind(input) {
|
|
37
|
-
const startTime = Date.now();
|
|
38
|
-
const { targetEntryId, cwd, unerrDir, graph, ledger, dryRun } = input;
|
|
39
|
-
try {
|
|
40
|
-
// ── Step 1: Read local ledger entries ─────────────────────────────
|
|
41
|
-
const allEntries = ledger.readAllEntries();
|
|
42
|
-
const targetEntry = allEntries.find((e) => e.id === targetEntryId);
|
|
43
|
-
if (!targetEntry) {
|
|
44
|
-
return errorResult(`Target entry ${targetEntryId} not found in local ledger`);
|
|
45
|
-
}
|
|
46
|
-
// ── Step 2: Identify entries to revert ────────────────────────────
|
|
47
|
-
// All entries created AFTER the target, on the same branch, not already reverted
|
|
48
|
-
const entriesToRevert = allEntries.filter((e) => e.branch === targetEntry.branch &&
|
|
49
|
-
e.ts > targetEntry.ts &&
|
|
50
|
-
e.id !== targetEntryId &&
|
|
51
|
-
!e.result_summary?.rewind_status);
|
|
52
|
-
// Collect all files changed in entries to revert
|
|
53
|
-
const filesToRevert = new Set();
|
|
54
|
-
for (const entry of entriesToRevert) {
|
|
55
|
-
const files = extractFilesFromEntry(entry);
|
|
56
|
-
for (const f of files)
|
|
57
|
-
filesToRevert.add(f);
|
|
58
|
-
}
|
|
59
|
-
// Files in the target entry
|
|
60
|
-
const targetFiles = new Set(extractFilesFromEntry(targetEntry));
|
|
61
|
-
// ── Step 3: Compute blast radius using CozoDB (<200ms) ────────────
|
|
62
|
-
const blastRadius = await computeLocalBlastRadius(graph, Array.from(filesToRevert), targetFiles, startTime);
|
|
63
|
-
if (dryRun) {
|
|
64
|
-
return {
|
|
65
|
-
status: "dry_run",
|
|
66
|
-
rewindEntryId: null,
|
|
67
|
-
timelineBranch: readBranchCounter(unerrDir),
|
|
68
|
-
entriesReverted: entriesToRevert.length,
|
|
69
|
-
blastRadius,
|
|
70
|
-
filesRestored: [],
|
|
71
|
-
};
|
|
72
|
-
}
|
|
73
|
-
// ── Step 4: Apply file restores via git checkout ──────────────────
|
|
74
|
-
const filesRestored = [];
|
|
75
|
-
const safeToRestore = blastRadius.safeFiles;
|
|
76
|
-
if (safeToRestore.length > 0) {
|
|
77
|
-
// Find the HEAD SHA to restore from (target entry's head_sha)
|
|
78
|
-
const restoreRef = targetEntry.head_sha;
|
|
79
|
-
if (restoreRef) {
|
|
80
|
-
for (const filePath of safeToRestore) {
|
|
81
|
-
try {
|
|
82
|
-
await checkoutFile(cwd, restoreRef, filePath);
|
|
83
|
-
filesRestored.push(filePath);
|
|
84
|
-
}
|
|
85
|
-
catch {
|
|
86
|
-
log.warn(`Could not restore ${filePath}`);
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
// ── Step 5: Update local state ────────────────────────────────────
|
|
92
|
-
// 5a. Increment timeline branch
|
|
93
|
-
const newBranch = incrementBranchCounter(unerrDir);
|
|
94
|
-
// 5b. Create rewind entry in shadow.jsonl
|
|
95
|
-
const rewindEntryId = generateId();
|
|
96
|
-
const rewindEntry = {
|
|
97
|
-
id: rewindEntryId,
|
|
98
|
-
ts: new Date().toISOString(),
|
|
99
|
-
tool: "revert_to_working_state",
|
|
100
|
-
args_summary: { target_entry_id: targetEntryId, offline: true },
|
|
101
|
-
result_summary: {
|
|
102
|
-
rewind_status: "simulated",
|
|
103
|
-
entries_reverted: entriesToRevert.length,
|
|
104
|
-
files_restored: filesRestored.length,
|
|
105
|
-
timeline_branch: newBranch,
|
|
106
|
-
rewind_target_id: targetEntryId,
|
|
107
|
-
blast_radius: {
|
|
108
|
-
safeFiles: blastRadius.safeFiles,
|
|
109
|
-
conflictedFiles: blastRadius.conflictedFiles.map((f) => f.filePath),
|
|
110
|
-
},
|
|
111
|
-
},
|
|
112
|
-
branch: targetEntry.branch,
|
|
113
|
-
head_sha: targetEntry.head_sha,
|
|
114
|
-
session_id: ledger.getSessionId(),
|
|
115
|
-
correlation_id: null,
|
|
116
|
-
};
|
|
117
|
-
// Append to shadow.jsonl
|
|
118
|
-
const ledgerPath = join(unerrDir, "ledger", "shadow.jsonl");
|
|
119
|
-
appendFileSync(ledgerPath, `${JSON.stringify(rewindEntry)}\n`, "utf-8");
|
|
120
|
-
// 5c. Mark intermediate entries as reverted locally
|
|
121
|
-
markEntriesReverted(unerrDir, entriesToRevert.map((e) => e.id));
|
|
122
|
-
// 5d. Clear drift overlay for reverted files (they're now at target state)
|
|
123
|
-
for (const _filePath of filesRestored) {
|
|
124
|
-
await graph.clearDriftOverlay();
|
|
125
|
-
}
|
|
126
|
-
return {
|
|
127
|
-
status: "simulated",
|
|
128
|
-
rewindEntryId,
|
|
129
|
-
timelineBranch: newBranch,
|
|
130
|
-
entriesReverted: entriesToRevert.length,
|
|
131
|
-
blastRadius,
|
|
132
|
-
filesRestored,
|
|
133
|
-
};
|
|
134
|
-
}
|
|
135
|
-
catch (error) {
|
|
136
|
-
const message = error instanceof Error ? error.message : String(error);
|
|
137
|
-
return errorResult(message);
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
|
-
// ── Blast Radius Computation (CozoDB, <200ms) ─────────────────────────────
|
|
141
|
-
async function computeLocalBlastRadius(graph, revertFiles, targetFiles, startTime) {
|
|
142
|
-
const safeFiles = [];
|
|
143
|
-
const conflictedFiles = [];
|
|
144
|
-
const affectedEntities = [];
|
|
145
|
-
const callerKeys = new Set();
|
|
146
|
-
for (const filePath of revertFiles) {
|
|
147
|
-
// Conflict: file exists in both target and reverted entries
|
|
148
|
-
if (targetFiles.has(filePath)) {
|
|
149
|
-
conflictedFiles.push({
|
|
150
|
-
filePath,
|
|
151
|
-
reason: "File modified in both target and reverted entries",
|
|
152
|
-
});
|
|
153
|
-
}
|
|
154
|
-
else {
|
|
155
|
-
safeFiles.push(filePath);
|
|
156
|
-
}
|
|
157
|
-
// Entity lookup via CozoDB file_index (in-process, <10ms)
|
|
158
|
-
const entities = await graph.getEntitiesByFile(filePath);
|
|
159
|
-
for (const entity of entities) {
|
|
160
|
-
affectedEntities.push({
|
|
161
|
-
key: entity.key,
|
|
162
|
-
name: entity.name,
|
|
163
|
-
filePath: entity.file_path,
|
|
164
|
-
riskLevel: entity.risk_level,
|
|
165
|
-
});
|
|
166
|
-
// 1-hop callers via CozoDB edges (pre-indexed, <50ms)
|
|
167
|
-
const callers = await graph.getCallersOf(entity.key);
|
|
168
|
-
for (const caller of callers) {
|
|
169
|
-
callerKeys.add(caller.key);
|
|
170
|
-
}
|
|
171
|
-
}
|
|
172
|
-
}
|
|
173
|
-
return {
|
|
174
|
-
safeFiles,
|
|
175
|
-
conflictedFiles,
|
|
176
|
-
affectedEntities,
|
|
177
|
-
affectedCallers: callerKeys.size,
|
|
178
|
-
resolvedInMs: Date.now() - startTime,
|
|
179
|
-
};
|
|
180
|
-
}
|
|
181
|
-
// ── Local State Helpers ────────────────────────────────────────────────────
|
|
182
|
-
/**
|
|
183
|
-
* Read the current timeline branch counter from branch_context.json.
|
|
184
|
-
*/
|
|
185
|
-
function readBranchCounter(unerrDir) {
|
|
186
|
-
const contextPath = join(unerrDir, "ledger", "branch_context.json");
|
|
187
|
-
if (!existsSync(contextPath))
|
|
188
|
-
return 1;
|
|
189
|
-
try {
|
|
190
|
-
const data = JSON.parse(readFileSync(contextPath, "utf-8"));
|
|
191
|
-
return data.timeline_branch ?? 1;
|
|
192
|
-
}
|
|
193
|
-
catch {
|
|
194
|
-
return 1;
|
|
195
|
-
}
|
|
196
|
-
}
|
|
197
|
-
/**
|
|
198
|
-
* Increment the timeline branch counter and return the new value.
|
|
199
|
-
*/
|
|
200
|
-
function incrementBranchCounter(unerrDir) {
|
|
201
|
-
const contextPath = join(unerrDir, "ledger", "branch_context.json");
|
|
202
|
-
const ledgerDir = join(unerrDir, "ledger");
|
|
203
|
-
if (!existsSync(ledgerDir)) {
|
|
204
|
-
mkdirSync(ledgerDir, { recursive: true });
|
|
205
|
-
}
|
|
206
|
-
let data = {};
|
|
207
|
-
if (existsSync(contextPath)) {
|
|
208
|
-
try {
|
|
209
|
-
data = JSON.parse(readFileSync(contextPath, "utf-8"));
|
|
210
|
-
}
|
|
211
|
-
catch {
|
|
212
|
-
// Start fresh
|
|
213
|
-
}
|
|
214
|
-
}
|
|
215
|
-
const current = typeof data.timeline_branch === "number" ? data.timeline_branch : 1;
|
|
216
|
-
const next = current + 1;
|
|
217
|
-
data.timeline_branch = next;
|
|
218
|
-
data.last_rewind_at = new Date().toISOString();
|
|
219
|
-
writeFileSync(contextPath, JSON.stringify(data, null, 2), "utf-8");
|
|
220
|
-
return next;
|
|
221
|
-
}
|
|
222
|
-
/**
|
|
223
|
-
* Mark entries as reverted in the pending_correlations.json file.
|
|
224
|
-
* These entries are tracked locally as reverted.
|
|
225
|
-
*/
|
|
226
|
-
function markEntriesReverted(unerrDir, entryIds) {
|
|
227
|
-
if (entryIds.length === 0)
|
|
228
|
-
return;
|
|
229
|
-
const revertedPath = join(unerrDir, "ledger", "reverted_entries.json");
|
|
230
|
-
const ledgerDir = join(unerrDir, "ledger");
|
|
231
|
-
if (!existsSync(ledgerDir)) {
|
|
232
|
-
mkdirSync(ledgerDir, { recursive: true });
|
|
233
|
-
}
|
|
234
|
-
let existing = [];
|
|
235
|
-
if (existsSync(revertedPath)) {
|
|
236
|
-
try {
|
|
237
|
-
existing = JSON.parse(readFileSync(revertedPath, "utf-8"));
|
|
238
|
-
}
|
|
239
|
-
catch {
|
|
240
|
-
// Start fresh
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
const merged = Array.from(new Set([...existing, ...entryIds]));
|
|
244
|
-
writeFileSync(revertedPath, JSON.stringify(merged, null, 2), "utf-8");
|
|
245
|
-
}
|
|
246
|
-
/**
|
|
247
|
-
* Extract file paths from a shadow ledger entry's args_summary.
|
|
248
|
-
*/
|
|
249
|
-
function extractFilesFromEntry(entry) {
|
|
250
|
-
const files = [];
|
|
251
|
-
// Check args_summary for file references
|
|
252
|
-
const args = entry.args_summary;
|
|
253
|
-
if (Array.isArray(args.files)) {
|
|
254
|
-
for (const f of args.files) {
|
|
255
|
-
if (typeof f === "string")
|
|
256
|
-
files.push(f);
|
|
257
|
-
else if (f &&
|
|
258
|
-
typeof f.file_path === "string") {
|
|
259
|
-
files.push(f.file_path);
|
|
260
|
-
}
|
|
261
|
-
}
|
|
262
|
-
}
|
|
263
|
-
if (typeof args.file_path === "string")
|
|
264
|
-
files.push(args.file_path);
|
|
265
|
-
// Check result_summary for affected files
|
|
266
|
-
const result = entry.result_summary;
|
|
267
|
-
if (Array.isArray(result.files_changed)) {
|
|
268
|
-
for (const f of result.files_changed) {
|
|
269
|
-
if (typeof f === "string")
|
|
270
|
-
files.push(f);
|
|
271
|
-
}
|
|
272
|
-
}
|
|
273
|
-
return files;
|
|
274
|
-
}
|
|
275
|
-
// ── Utilities ──────────────────────────────────────────────────────────────
|
|
276
|
-
function generateId() {
|
|
277
|
-
return randomBytes(6).toString("hex");
|
|
278
|
-
}
|
|
279
|
-
function errorResult(message) {
|
|
280
|
-
return {
|
|
281
|
-
status: "error",
|
|
282
|
-
rewindEntryId: null,
|
|
283
|
-
timelineBranch: 0,
|
|
284
|
-
entriesReverted: 0,
|
|
285
|
-
blastRadius: {
|
|
286
|
-
safeFiles: [],
|
|
287
|
-
conflictedFiles: [],
|
|
288
|
-
affectedEntities: [],
|
|
289
|
-
affectedCallers: 0,
|
|
290
|
-
resolvedInMs: 0,
|
|
291
|
-
},
|
|
292
|
-
filesRestored: [],
|
|
293
|
-
errorMessage: message,
|
|
294
|
-
};
|
|
295
|
-
}
|
|
@@ -1,74 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Sprint 7.3: Pending Violation Store — push-based rule enforcement safety net.
|
|
3
|
-
*
|
|
4
|
-
* Stores rule violations detected by file-watcher-triggered rule evaluation.
|
|
5
|
-
* Violations attach to the internal `context` carrier on the next MCP response
|
|
6
|
-
* and surface as `ur|wrn` prefix lines (then cleared).
|
|
7
|
-
*
|
|
8
|
-
* Flow:
|
|
9
|
-
* File change → DriftTracker → evaluateRules → store violations here
|
|
10
|
-
* Next MCP tool call → QueryRouter drains pending violations → `ur|wrn` prefix lines
|
|
11
|
-
*/
|
|
12
|
-
export class PendingViolationStore {
|
|
13
|
-
violations = new Map();
|
|
14
|
-
/**
|
|
15
|
-
* Add violations from a file change. Replaces any existing violations for that file.
|
|
16
|
-
*/
|
|
17
|
-
addViolations(filePath, violations) {
|
|
18
|
-
if (violations.length === 0) {
|
|
19
|
-
this.violations.delete(filePath);
|
|
20
|
-
return;
|
|
21
|
-
}
|
|
22
|
-
this.violations.set(filePath, violations.map((v) => ({
|
|
23
|
-
file: v.filePath,
|
|
24
|
-
rule: v.ruleName,
|
|
25
|
-
message: v.message,
|
|
26
|
-
line: v.line,
|
|
27
|
-
})));
|
|
28
|
-
}
|
|
29
|
-
/**
|
|
30
|
-
* Drain all pending violations — returns them and clears the store.
|
|
31
|
-
* Returns undefined if no violations pending.
|
|
32
|
-
*/
|
|
33
|
-
drain() {
|
|
34
|
-
if (this.violations.size === 0)
|
|
35
|
-
return undefined;
|
|
36
|
-
const all = [];
|
|
37
|
-
for (const [, vs] of this.violations) {
|
|
38
|
-
all.push(...vs);
|
|
39
|
-
}
|
|
40
|
-
this.violations.clear();
|
|
41
|
-
return all;
|
|
42
|
-
}
|
|
43
|
-
/**
|
|
44
|
-
* Drain violations for specific files only.
|
|
45
|
-
* Returns undefined if no violations pending for those files.
|
|
46
|
-
*/
|
|
47
|
-
drainForFiles(filePaths) {
|
|
48
|
-
const result = [];
|
|
49
|
-
for (const fp of filePaths) {
|
|
50
|
-
const vs = this.violations.get(fp);
|
|
51
|
-
if (vs) {
|
|
52
|
-
result.push(...vs);
|
|
53
|
-
this.violations.delete(fp);
|
|
54
|
-
}
|
|
55
|
-
}
|
|
56
|
-
return result.length > 0 ? result : undefined;
|
|
57
|
-
}
|
|
58
|
-
/** Check if any violations are pending. */
|
|
59
|
-
get hasPending() {
|
|
60
|
-
return this.violations.size > 0;
|
|
61
|
-
}
|
|
62
|
-
/** Total number of pending violations across all files. */
|
|
63
|
-
get count() {
|
|
64
|
-
let total = 0;
|
|
65
|
-
for (const [, vs] of this.violations) {
|
|
66
|
-
total += vs.length;
|
|
67
|
-
}
|
|
68
|
-
return total;
|
|
69
|
-
}
|
|
70
|
-
/** Clear all pending violations. */
|
|
71
|
-
clear() {
|
|
72
|
-
this.violations.clear();
|
|
73
|
-
}
|
|
74
|
-
}
|
|
@@ -1,167 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Persistent Memory Effectiveness Tracker.
|
|
3
|
-
*
|
|
4
|
-
* Surfaces the "was this signal load-bearing?" question for the /reasoning
|
|
5
|
-
* dashboard. Tracks fact/convention/resume/negative-warning injections as
|
|
6
|
-
* open windows, then evaluates each window's verdict on turn close.
|
|
7
|
-
*
|
|
8
|
-
* Verdicts (emitted as TokenFlow events, mechanism: "persistent_memory"):
|
|
9
|
-
* reinforced – same signal re-surfaced, no correction observed
|
|
10
|
-
* acted_on – entity edited within window, no rollback / circuit-breaker
|
|
11
|
-
* ignored – window closed without any observable downstream activity
|
|
12
|
-
* corrected – circuit-breaker / drift / blast-radius fired for same entity
|
|
13
|
-
* caught – (negative-fact only) anti-pattern warned, no recurrence
|
|
14
|
-
*
|
|
15
|
-
* Fired events also emit (with verdict:"fired") so the dashboard can count
|
|
16
|
-
* raw surface volume separately from outcome.
|
|
17
|
-
*/
|
|
18
|
-
export class PersistenceEffectivenessTracker {
|
|
19
|
-
tokenFlow;
|
|
20
|
-
windowTurns;
|
|
21
|
-
tokensPerSignal;
|
|
22
|
-
open = new Map();
|
|
23
|
-
constructor(tokenFlow, options = {}) {
|
|
24
|
-
this.tokenFlow = tokenFlow;
|
|
25
|
-
this.windowTurns = options.windowTurns ?? 5;
|
|
26
|
-
this.tokensPerSignal = options.tokensPerSignal ?? 60;
|
|
27
|
-
}
|
|
28
|
-
/**
|
|
29
|
-
* Record that a persistent-memory signal was injected/surfaced.
|
|
30
|
-
* Emits a "fired" verdict event immediately AND opens a window for later
|
|
31
|
-
* verdict resolution. Re-firing the same (kind, signal_id) within an open
|
|
32
|
-
* window counts as a reinforcement, not a new fired event.
|
|
33
|
-
*/
|
|
34
|
-
recordSignalFired(p) {
|
|
35
|
-
const sessionId = p.session_id ?? this.tokenFlow.sessionId;
|
|
36
|
-
const key = `${p.kind}:${p.signal_id}`;
|
|
37
|
-
const existing = this.open.get(key);
|
|
38
|
-
if (existing) {
|
|
39
|
-
existing.reinforcements++;
|
|
40
|
-
return;
|
|
41
|
-
}
|
|
42
|
-
this.open.set(key, {
|
|
43
|
-
kind: p.kind,
|
|
44
|
-
signal_id: p.signal_id,
|
|
45
|
-
entity_key: p.entity_key ?? null,
|
|
46
|
-
session_id: sessionId,
|
|
47
|
-
fired_turn: p.turn,
|
|
48
|
-
fired_at_ms: Date.now(),
|
|
49
|
-
reinforcements: 0,
|
|
50
|
-
corrections: 0,
|
|
51
|
-
edits: 0,
|
|
52
|
-
});
|
|
53
|
-
this.emit({
|
|
54
|
-
verdict: "fired",
|
|
55
|
-
kind: p.kind,
|
|
56
|
-
signal_id: p.signal_id,
|
|
57
|
-
entity_key: p.entity_key ?? null,
|
|
58
|
-
session_id: sessionId,
|
|
59
|
-
turn: p.turn,
|
|
60
|
-
reinforcements: 0,
|
|
61
|
-
corrections: 0,
|
|
62
|
-
edits: 0,
|
|
63
|
-
});
|
|
64
|
-
}
|
|
65
|
-
/**
|
|
66
|
-
* Observed correction for an entity (circuit-breaker, drift, blast-radius
|
|
67
|
-
* warning, convention-violation auto-fire). Counts against any open signal
|
|
68
|
-
* scoped to the same entity_key.
|
|
69
|
-
*/
|
|
70
|
-
recordCorrection(entityKey, type) {
|
|
71
|
-
if (!entityKey)
|
|
72
|
-
return;
|
|
73
|
-
for (const s of this.open.values()) {
|
|
74
|
-
if (s.entity_key === entityKey) {
|
|
75
|
-
s.corrections++;
|
|
76
|
-
}
|
|
77
|
-
}
|
|
78
|
-
// type captured for future detail enrichment
|
|
79
|
-
void type;
|
|
80
|
-
}
|
|
81
|
-
/**
|
|
82
|
-
* Observed Edit/Write on an entity. Counts toward "acted_on" verdict for
|
|
83
|
-
* any open signal scoped to the same entity_key.
|
|
84
|
-
*/
|
|
85
|
-
recordEdit(entityKey) {
|
|
86
|
-
if (!entityKey)
|
|
87
|
-
return;
|
|
88
|
-
for (const s of this.open.values()) {
|
|
89
|
-
if (s.entity_key === entityKey) {
|
|
90
|
-
s.edits++;
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
/**
|
|
95
|
-
* Evaluate all open signals whose window has closed (current_turn - fired_turn >= windowTurns).
|
|
96
|
-
* Emits one verdict event per closed signal.
|
|
97
|
-
*/
|
|
98
|
-
closeWindow(currentTurn) {
|
|
99
|
-
for (const [key, s] of this.open) {
|
|
100
|
-
if (currentTurn - s.fired_turn < this.windowTurns)
|
|
101
|
-
continue;
|
|
102
|
-
this.emitVerdict(s);
|
|
103
|
-
this.open.delete(key);
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
|
-
/**
|
|
107
|
-
* Close every open window — called at session end. Forces a verdict for
|
|
108
|
-
* everything still open regardless of turn distance.
|
|
109
|
-
*/
|
|
110
|
-
closeAll(currentTurn) {
|
|
111
|
-
for (const [key, s] of this.open) {
|
|
112
|
-
this.emitVerdict(s, currentTurn);
|
|
113
|
-
this.open.delete(key);
|
|
114
|
-
}
|
|
115
|
-
}
|
|
116
|
-
/** Number of signals still open (for testing / introspection). */
|
|
117
|
-
openCount() {
|
|
118
|
-
return this.open.size;
|
|
119
|
-
}
|
|
120
|
-
emitVerdict(s, _currentTurn) {
|
|
121
|
-
const verdict = this.classify(s);
|
|
122
|
-
this.emit({
|
|
123
|
-
verdict,
|
|
124
|
-
kind: s.kind,
|
|
125
|
-
signal_id: s.signal_id,
|
|
126
|
-
entity_key: s.entity_key,
|
|
127
|
-
session_id: s.session_id,
|
|
128
|
-
turn: s.fired_turn,
|
|
129
|
-
reinforcements: s.reinforcements,
|
|
130
|
-
corrections: s.corrections,
|
|
131
|
-
edits: s.edits,
|
|
132
|
-
});
|
|
133
|
-
}
|
|
134
|
-
classify(s) {
|
|
135
|
-
if (s.corrections > 0)
|
|
136
|
-
return "corrected";
|
|
137
|
-
if (s.kind === "negative_warned") {
|
|
138
|
-
// For negative facts the "no recurrence" case is the win — caught.
|
|
139
|
-
return "caught";
|
|
140
|
-
}
|
|
141
|
-
if (s.edits > 0)
|
|
142
|
-
return "acted_on";
|
|
143
|
-
if (s.reinforcements > 0)
|
|
144
|
-
return "reinforced";
|
|
145
|
-
return "ignored";
|
|
146
|
-
}
|
|
147
|
-
emit(p) {
|
|
148
|
-
this.tokenFlow.record({
|
|
149
|
-
session_id: p.session_id,
|
|
150
|
-
turn: p.turn,
|
|
151
|
-
mechanism: "persistent_memory",
|
|
152
|
-
tool: null,
|
|
153
|
-
tokens_without: 0,
|
|
154
|
-
tokens_with: this.tokensPerSignal,
|
|
155
|
-
tokens_saved: 0,
|
|
156
|
-
detail: {
|
|
157
|
-
kind: p.kind,
|
|
158
|
-
verdict: p.verdict,
|
|
159
|
-
signal_id: p.signal_id,
|
|
160
|
-
entity_key: p.entity_key,
|
|
161
|
-
reinforcements: p.reinforcements,
|
|
162
|
-
corrections: p.corrections,
|
|
163
|
-
edits: p.edits,
|
|
164
|
-
},
|
|
165
|
-
});
|
|
166
|
-
}
|
|
167
|
-
}
|