@unerr-ai/unerr 0.2.0 → 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 +6 -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,171 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Sprint 10.5: Git Trailer Injection — prepare-commit-msg hook + UDS endpoint.
|
|
3
|
-
*
|
|
4
|
-
* Injects `Unerr-Ledger-Id`, `Unerr-Session`, `Unerr-Timeline-Branch` trailers
|
|
5
|
-
* into commit messages via prepare-commit-msg hook.
|
|
6
|
-
*
|
|
7
|
-
* Two integration modes:
|
|
8
|
-
* 1. Hook mode: Shell script calls UDS `/commit-context` endpoint
|
|
9
|
-
* 2. Direct mode: CommitWatcher calls `getTrailers()` after commit detection
|
|
10
|
-
*
|
|
11
|
-
* Design authority: Phase 5.5 §1.4.1 (Git Trailer Injection)
|
|
12
|
-
*/
|
|
13
|
-
import { existsSync, readFileSync, writeFileSync } from "node:fs";
|
|
14
|
-
import { join } from "node:path";
|
|
15
|
-
/** stderr logger */
|
|
16
|
-
const _log = {
|
|
17
|
-
info: (msg) => process.stderr.write(`[unerr:trailers] ${msg}\n`),
|
|
18
|
-
};
|
|
19
|
-
/**
|
|
20
|
-
* Get trailers for the current commit context.
|
|
21
|
-
* Called by the UDS `/commit-context` endpoint or directly by CommitWatcher.
|
|
22
|
-
*/
|
|
23
|
-
export function getCommitTrailers(ledger, timelineBranch, branch) {
|
|
24
|
-
const recent = ledger.getRecentEntries(1);
|
|
25
|
-
if (recent.length === 0)
|
|
26
|
-
return null;
|
|
27
|
-
const latest = recent[0];
|
|
28
|
-
return {
|
|
29
|
-
ledgerId: latest.id,
|
|
30
|
-
sessionId: ledger.getSessionId(),
|
|
31
|
-
timelineBranch,
|
|
32
|
-
branch,
|
|
33
|
-
};
|
|
34
|
-
}
|
|
35
|
-
/**
|
|
36
|
-
* Format trailers for git commit message injection.
|
|
37
|
-
*/
|
|
38
|
-
export function formatTrailers(trailers) {
|
|
39
|
-
return [
|
|
40
|
-
`Unerr-Ledger-Id: ${trailers.ledgerId}`,
|
|
41
|
-
`Unerr-Session: ${trailers.sessionId}`,
|
|
42
|
-
`Unerr-Timeline-Branch: ${trailers.timelineBranch}`,
|
|
43
|
-
].join("\n");
|
|
44
|
-
}
|
|
45
|
-
/**
|
|
46
|
-
* Install the prepare-commit-msg hook that calls the UDS endpoint.
|
|
47
|
-
* Only installs if:
|
|
48
|
-
* 1. The hook doesn't already exist or doesn't contain our marker
|
|
49
|
-
* 2. The .git/hooks directory exists
|
|
50
|
-
*/
|
|
51
|
-
export function installPrepareCommitMsgHook(projectRoot) {
|
|
52
|
-
const hooksDir = join(projectRoot, ".git", "hooks");
|
|
53
|
-
if (!existsSync(hooksDir)) {
|
|
54
|
-
// Not a git repo or hooks dir missing
|
|
55
|
-
return false;
|
|
56
|
-
}
|
|
57
|
-
const hookPath = join(hooksDir, "prepare-commit-msg");
|
|
58
|
-
const marker = "# unerr-trailer-injection";
|
|
59
|
-
// Check if hook already exists with our marker
|
|
60
|
-
if (existsSync(hookPath)) {
|
|
61
|
-
try {
|
|
62
|
-
const existing = readFileSync(hookPath, "utf-8");
|
|
63
|
-
if (existing.includes(marker)) {
|
|
64
|
-
return true; // Already installed
|
|
65
|
-
}
|
|
66
|
-
// Hook exists but isn't ours — append our section
|
|
67
|
-
const appendSection = `\n\n${marker}\n${generateHookScript()}`;
|
|
68
|
-
writeFileSync(hookPath, existing + appendSection, { mode: 0o755 });
|
|
69
|
-
_log.info("Appended trailer injection to existing prepare-commit-msg hook");
|
|
70
|
-
return true;
|
|
71
|
-
}
|
|
72
|
-
catch {
|
|
73
|
-
return false;
|
|
74
|
-
}
|
|
75
|
-
}
|
|
76
|
-
// Create new hook
|
|
77
|
-
const hookContent = `#!/bin/sh\n${marker}\n${generateHookScript()}`;
|
|
78
|
-
try {
|
|
79
|
-
writeFileSync(hookPath, hookContent, { mode: 0o755 });
|
|
80
|
-
_log.info("Installed prepare-commit-msg hook for trailer injection");
|
|
81
|
-
return true;
|
|
82
|
-
}
|
|
83
|
-
catch {
|
|
84
|
-
return false;
|
|
85
|
-
}
|
|
86
|
-
}
|
|
87
|
-
/**
|
|
88
|
-
* Remove the prepare-commit-msg hook (or our section from it).
|
|
89
|
-
*/
|
|
90
|
-
export function uninstallPrepareCommitMsgHook(projectRoot) {
|
|
91
|
-
const hookPath = join(projectRoot, ".git", "hooks", "prepare-commit-msg");
|
|
92
|
-
if (!existsSync(hookPath))
|
|
93
|
-
return true;
|
|
94
|
-
try {
|
|
95
|
-
const content = readFileSync(hookPath, "utf-8");
|
|
96
|
-
const marker = "# unerr-trailer-injection";
|
|
97
|
-
if (!content.includes(marker))
|
|
98
|
-
return true; // Not our hook
|
|
99
|
-
// If the entire hook is ours, remove the file
|
|
100
|
-
const lines = content.split("\n");
|
|
101
|
-
const nonUnerrLines = lines.filter((line) => !line.includes(marker) &&
|
|
102
|
-
!line.includes("unerr") &&
|
|
103
|
-
!line.includes("Unerr-"));
|
|
104
|
-
// If only shebang remains, remove the file
|
|
105
|
-
if (nonUnerrLines.length <= 1 && nonUnerrLines[0]?.startsWith("#!/")) {
|
|
106
|
-
const { unlinkSync } = require("node:fs");
|
|
107
|
-
unlinkSync(hookPath);
|
|
108
|
-
}
|
|
109
|
-
else {
|
|
110
|
-
// Remove our section
|
|
111
|
-
const markerIdx = content.indexOf(marker);
|
|
112
|
-
if (markerIdx > 0) {
|
|
113
|
-
writeFileSync(hookPath, `${content.slice(0, markerIdx).trimEnd()}\n`, {
|
|
114
|
-
mode: 0o755,
|
|
115
|
-
});
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
return true;
|
|
119
|
-
}
|
|
120
|
-
catch {
|
|
121
|
-
return false;
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
/**
|
|
125
|
-
* Generate the hook script body that queries the UDS endpoint.
|
|
126
|
-
*/
|
|
127
|
-
function generateHookScript() {
|
|
128
|
-
return `# Inject unerr ledger trailers into commit message
|
|
129
|
-
COMMIT_MSG_FILE="$1"
|
|
130
|
-
SOCK=".unerr/state/proxy.sock"
|
|
131
|
-
|
|
132
|
-
if [ -S "$SOCK" ]; then
|
|
133
|
-
TRAILERS=$(curl -s --unix-socket "$SOCK" http://localhost/commit-context 2>/dev/null)
|
|
134
|
-
if [ -n "$TRAILERS" ] && [ "$TRAILERS" != "null" ]; then
|
|
135
|
-
LEDGER_ID=$(echo "$TRAILERS" | grep -o '"ledgerId":"[^"]*"' | cut -d'"' -f4)
|
|
136
|
-
SESSION_ID=$(echo "$TRAILERS" | grep -o '"sessionId":"[^"]*"' | cut -d'"' -f4)
|
|
137
|
-
TIMELINE=$(echo "$TRAILERS" | grep -o '"timelineBranch":[0-9]*' | cut -d: -f2)
|
|
138
|
-
if [ -n "$LEDGER_ID" ]; then
|
|
139
|
-
echo "" >> "$COMMIT_MSG_FILE"
|
|
140
|
-
echo "Unerr-Ledger-Id: $LEDGER_ID" >> "$COMMIT_MSG_FILE"
|
|
141
|
-
echo "Unerr-Session: $SESSION_ID" >> "$COMMIT_MSG_FILE"
|
|
142
|
-
echo "Unerr-Timeline-Branch: $TIMELINE" >> "$COMMIT_MSG_FILE"
|
|
143
|
-
fi
|
|
144
|
-
fi
|
|
145
|
-
fi`;
|
|
146
|
-
}
|
|
147
|
-
/**
|
|
148
|
-
* Parse trailers from a commit message. Returns null if no unerr trailers found.
|
|
149
|
-
*/
|
|
150
|
-
export function parseTrailersFromMessage(commitMessage) {
|
|
151
|
-
const lines = commitMessage.split("\n");
|
|
152
|
-
let ledgerId = null;
|
|
153
|
-
let sessionId = null;
|
|
154
|
-
let timelineBranch = 0;
|
|
155
|
-
const branch = "unknown";
|
|
156
|
-
for (const line of lines) {
|
|
157
|
-
const trimmed = line.trim();
|
|
158
|
-
if (trimmed.startsWith("Unerr-Ledger-Id:")) {
|
|
159
|
-
ledgerId = trimmed.slice("Unerr-Ledger-Id:".length).trim();
|
|
160
|
-
}
|
|
161
|
-
else if (trimmed.startsWith("Unerr-Session:")) {
|
|
162
|
-
sessionId = trimmed.slice("Unerr-Session:".length).trim();
|
|
163
|
-
}
|
|
164
|
-
else if (trimmed.startsWith("Unerr-Timeline-Branch:")) {
|
|
165
|
-
timelineBranch = Number.parseInt(trimmed.slice("Unerr-Timeline-Branch:".length).trim(), 10);
|
|
166
|
-
}
|
|
167
|
-
}
|
|
168
|
-
if (!ledgerId || !sessionId)
|
|
169
|
-
return null;
|
|
170
|
-
return { ledgerId, sessionId, timelineBranch, branch };
|
|
171
|
-
}
|
|
@@ -1,46 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Compound Intelligence Counter — tracks accumulated knowledge.
|
|
3
|
-
*
|
|
4
|
-
* VS.7: Displays in status + weekly: entity count, edge count,
|
|
5
|
-
* conventions learned, rules generated, corrections applied.
|
|
6
|
-
*/
|
|
7
|
-
export function createIntelligenceCounter() {
|
|
8
|
-
const metrics = {
|
|
9
|
-
entityCount: 0,
|
|
10
|
-
edgeCount: 0,
|
|
11
|
-
conventionsLearned: 0,
|
|
12
|
-
rulesGenerated: 0,
|
|
13
|
-
correctionsApplied: 0,
|
|
14
|
-
communitiesDetected: 0,
|
|
15
|
-
sessionsIndexed: 0,
|
|
16
|
-
};
|
|
17
|
-
function record(field, value) {
|
|
18
|
-
metrics[field] = value;
|
|
19
|
-
}
|
|
20
|
-
function increment(field) {
|
|
21
|
-
metrics[field]++;
|
|
22
|
-
}
|
|
23
|
-
function getMetrics() {
|
|
24
|
-
return { ...metrics };
|
|
25
|
-
}
|
|
26
|
-
function formatSummary() {
|
|
27
|
-
const parts = [
|
|
28
|
-
`${metrics.entityCount.toLocaleString()} entities`,
|
|
29
|
-
`${metrics.edgeCount.toLocaleString()} edges`,
|
|
30
|
-
`${metrics.communitiesDetected} communities`,
|
|
31
|
-
];
|
|
32
|
-
if (metrics.conventionsLearned > 0)
|
|
33
|
-
parts.push(`${metrics.conventionsLearned} conventions`);
|
|
34
|
-
if (metrics.rulesGenerated > 0)
|
|
35
|
-
parts.push(`${metrics.rulesGenerated} rules`);
|
|
36
|
-
if (metrics.correctionsApplied > 0)
|
|
37
|
-
parts.push(`${metrics.correctionsApplied} corrections`);
|
|
38
|
-
return `Intelligence: ${parts.join(", ")}`;
|
|
39
|
-
}
|
|
40
|
-
function reset() {
|
|
41
|
-
for (const k of Object.keys(metrics)) {
|
|
42
|
-
metrics[k] = 0;
|
|
43
|
-
}
|
|
44
|
-
}
|
|
45
|
-
return { record, increment, getMetrics, formatSummary, reset };
|
|
46
|
-
}
|
|
@@ -1,202 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Intent Correlator — builds intent chains linking related tool calls.
|
|
3
|
-
*
|
|
4
|
-
* Maintains `pending_correlations.json` — intents that produced file changes
|
|
5
|
-
* (via `sync_local_diff`) but haven't been committed yet. Each pending
|
|
6
|
-
* correlation records the prompt, changed files, affected entities, and the
|
|
7
|
-
* full tool chain that led to the change.
|
|
8
|
-
*
|
|
9
|
-
* Lifecycle:
|
|
10
|
-
* 1. Tool calls flow through ShadowLedger → each gets an entry with correlation_id
|
|
11
|
-
* 2. When sync_local_diff is called → IntentCorrelator creates a pending correlation
|
|
12
|
-
* 3. On git commit (Sprint 5) → pending correlations are associated with commit SHA
|
|
13
|
-
* 4. On push (Sprint 5) → committed entries flushed to local ledger
|
|
14
|
-
*
|
|
15
|
-
* All logging to stderr.
|
|
16
|
-
*/
|
|
17
|
-
import { existsSync, mkdirSync, readFileSync, renameSync, writeFileSync, } from "node:fs";
|
|
18
|
-
import { join } from "node:path";
|
|
19
|
-
export class IntentCorrelator {
|
|
20
|
-
ledgerDir;
|
|
21
|
-
pendingPath;
|
|
22
|
-
pending = [];
|
|
23
|
-
constructor(unerrDir) {
|
|
24
|
-
this.ledgerDir = join(unerrDir, "ledger");
|
|
25
|
-
this.pendingPath = join(this.ledgerDir, "pending_correlations.json");
|
|
26
|
-
if (!existsSync(this.ledgerDir)) {
|
|
27
|
-
mkdirSync(this.ledgerDir, { recursive: true });
|
|
28
|
-
}
|
|
29
|
-
this.load();
|
|
30
|
-
}
|
|
31
|
-
/**
|
|
32
|
-
* Called when sync_local_diff is invoked. Creates a pending correlation
|
|
33
|
-
* from the active tool chain in the ledger.
|
|
34
|
-
*/
|
|
35
|
-
onSyncLocalDiff(ledger, args) {
|
|
36
|
-
const rootId = ledger.getCurrentRootId();
|
|
37
|
-
if (!rootId)
|
|
38
|
-
return null;
|
|
39
|
-
// Extract data from sync_local_diff args
|
|
40
|
-
const prompt = args.prompt ?? args.message ?? "";
|
|
41
|
-
const filesChanged = extractFiles(args);
|
|
42
|
-
const entitiesAffected = extractEntities(args);
|
|
43
|
-
// Build the tool chain from ledger buffer
|
|
44
|
-
const recentEntries = ledger.getRecentEntries(50);
|
|
45
|
-
const toolChain = buildToolChain(recentEntries, rootId);
|
|
46
|
-
const correlation = {
|
|
47
|
-
rootIntentId: rootId,
|
|
48
|
-
prompt,
|
|
49
|
-
files: filesChanged,
|
|
50
|
-
entities: entitiesAffected,
|
|
51
|
-
toolChain,
|
|
52
|
-
createdAt: new Date().toISOString(),
|
|
53
|
-
};
|
|
54
|
-
this.pending.push(correlation);
|
|
55
|
-
this.save();
|
|
56
|
-
return correlation;
|
|
57
|
-
}
|
|
58
|
-
/**
|
|
59
|
-
* Get all pending correlations (not yet committed).
|
|
60
|
-
*/
|
|
61
|
-
getPending() {
|
|
62
|
-
return this.pending.filter((c) => !c.commitSha);
|
|
63
|
-
}
|
|
64
|
-
/**
|
|
65
|
-
* Get all pending correlations (including committed but not flushed).
|
|
66
|
-
*/
|
|
67
|
-
getAll() {
|
|
68
|
-
return [...this.pending];
|
|
69
|
-
}
|
|
70
|
-
/**
|
|
71
|
-
* Get the count of pending correlations.
|
|
72
|
-
*/
|
|
73
|
-
getPendingCount() {
|
|
74
|
-
return this.pending.filter((c) => !c.commitSha).length;
|
|
75
|
-
}
|
|
76
|
-
/**
|
|
77
|
-
* Associate pending correlations with a commit SHA.
|
|
78
|
-
* Called by the commit watcher when HEAD changes.
|
|
79
|
-
* Only associates correlations whose files appear in the commit diff.
|
|
80
|
-
*/
|
|
81
|
-
associateCommit(commitSha, committedFiles) {
|
|
82
|
-
let associated = 0;
|
|
83
|
-
const committedSet = new Set(committedFiles);
|
|
84
|
-
for (const correlation of this.pending) {
|
|
85
|
-
if (correlation.commitSha)
|
|
86
|
-
continue; // Already committed
|
|
87
|
-
// Check if any of the correlation's files are in the commit
|
|
88
|
-
const hasOverlap = correlation.files.some((f) => committedSet.has(f));
|
|
89
|
-
if (hasOverlap) {
|
|
90
|
-
correlation.commitSha = commitSha;
|
|
91
|
-
associated++;
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
if (associated > 0) {
|
|
95
|
-
this.save();
|
|
96
|
-
}
|
|
97
|
-
return associated;
|
|
98
|
-
}
|
|
99
|
-
/**
|
|
100
|
-
* Get committed but not-yet-flushed correlations (for Sprint 5 push).
|
|
101
|
-
*/
|
|
102
|
-
getCommittedUnflushed() {
|
|
103
|
-
return this.pending.filter((c) => c.commitSha != null);
|
|
104
|
-
}
|
|
105
|
-
/**
|
|
106
|
-
* Remove correlations that have been flushed.
|
|
107
|
-
* Called after successful flush in Sprint 5.
|
|
108
|
-
*/
|
|
109
|
-
removeFlushed(rootIntentIds) {
|
|
110
|
-
const flushedSet = new Set(rootIntentIds);
|
|
111
|
-
this.pending = this.pending.filter((c) => !flushedSet.has(c.rootIntentId));
|
|
112
|
-
this.save();
|
|
113
|
-
}
|
|
114
|
-
/**
|
|
115
|
-
* Clear all pending correlations (e.g., on branch switch).
|
|
116
|
-
*/
|
|
117
|
-
clear() {
|
|
118
|
-
this.pending = [];
|
|
119
|
-
this.save();
|
|
120
|
-
}
|
|
121
|
-
// ── Persistence ─────────────────────────────────────────────────
|
|
122
|
-
load() {
|
|
123
|
-
if (!existsSync(this.pendingPath))
|
|
124
|
-
return;
|
|
125
|
-
try {
|
|
126
|
-
const raw = readFileSync(this.pendingPath, "utf-8");
|
|
127
|
-
const parsed = JSON.parse(raw);
|
|
128
|
-
if (Array.isArray(parsed)) {
|
|
129
|
-
this.pending = parsed;
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
catch {
|
|
133
|
-
// Corrupted file — start fresh
|
|
134
|
-
this.pending = [];
|
|
135
|
-
}
|
|
136
|
-
}
|
|
137
|
-
save() {
|
|
138
|
-
try {
|
|
139
|
-
const tmpPath = `${this.pendingPath}.tmp`;
|
|
140
|
-
writeFileSync(tmpPath, JSON.stringify(this.pending, null, 2), "utf-8");
|
|
141
|
-
renameSync(tmpPath, this.pendingPath);
|
|
142
|
-
}
|
|
143
|
-
catch (err) {
|
|
144
|
-
process.stderr.write(`[unerr:correlator] WARN: Failed to save pending correlations: ${err instanceof Error ? err.message : String(err)}\n`);
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
}
|
|
148
|
-
/**
|
|
149
|
-
* Extract changed file paths from sync_local_diff args.
|
|
150
|
-
*/
|
|
151
|
-
function extractFiles(args) {
|
|
152
|
-
// Structured format: { files: [{ path, content }] }
|
|
153
|
-
const files = args.files;
|
|
154
|
-
if (files && Array.isArray(files)) {
|
|
155
|
-
return files.map((f) => f.path).filter(Boolean);
|
|
156
|
-
}
|
|
157
|
-
// Diff format: parse git diff for file paths
|
|
158
|
-
const diff = args.diff;
|
|
159
|
-
if (diff) {
|
|
160
|
-
const paths = new Set();
|
|
161
|
-
const matches = diff.matchAll(/^(?:---|\+\+\+)\s+[ab]\/(.+)$/gm);
|
|
162
|
-
for (const m of matches) {
|
|
163
|
-
if (m[1])
|
|
164
|
-
paths.add(m[1]);
|
|
165
|
-
}
|
|
166
|
-
return Array.from(paths);
|
|
167
|
-
}
|
|
168
|
-
// filesChanged from args directly
|
|
169
|
-
const filesChanged = args.filesChanged;
|
|
170
|
-
if (filesChanged && Array.isArray(filesChanged)) {
|
|
171
|
-
return filesChanged;
|
|
172
|
-
}
|
|
173
|
-
return [];
|
|
174
|
-
}
|
|
175
|
-
/**
|
|
176
|
-
* Extract affected entity keys from sync_local_diff args.
|
|
177
|
-
*/
|
|
178
|
-
function extractEntities(args) {
|
|
179
|
-
const entities = args.entitiesAffected;
|
|
180
|
-
if (entities && Array.isArray(entities)) {
|
|
181
|
-
return entities;
|
|
182
|
-
}
|
|
183
|
-
return [];
|
|
184
|
-
}
|
|
185
|
-
/**
|
|
186
|
-
* Build the tool chain for a correlation by following the intent chain
|
|
187
|
-
* from root to the current sync_local_diff call.
|
|
188
|
-
*/
|
|
189
|
-
function buildToolChain(entries, rootId) {
|
|
190
|
-
const chain = [];
|
|
191
|
-
for (const entry of entries) {
|
|
192
|
-
// Include the root entry itself
|
|
193
|
-
if (entry.id === rootId) {
|
|
194
|
-
chain.push(entry.tool);
|
|
195
|
-
}
|
|
196
|
-
// Include correlated entries
|
|
197
|
-
else if (entry.correlation_id === rootId) {
|
|
198
|
-
chain.push(entry.tool);
|
|
199
|
-
}
|
|
200
|
-
}
|
|
201
|
-
return chain;
|
|
202
|
-
}
|
|
@@ -1,52 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Intent Encoder — writes correlated intents as git notes (Layer 2 telemetry).
|
|
3
|
-
*
|
|
4
|
-
* When CommitWatcher detects a commit with associated intents, this encoder
|
|
5
|
-
* writes a compact JSON note to `refs/notes/unerr`. Notes travel with commits
|
|
6
|
-
* and are invisible in `git log` (only visible with `--notes=unerr`).
|
|
7
|
-
*
|
|
8
|
-
* This is insurance — as backup, notes carry intent data with pushes.
|
|
9
|
-
* Fire-and-forget: failures are logged as warnings, never block the proxy.
|
|
10
|
-
*/
|
|
11
|
-
import { writeNote } from "../utils/git.js";
|
|
12
|
-
import { createModuleLogger } from "../utils/logger.js";
|
|
13
|
-
const log = createModuleLogger("notes");
|
|
14
|
-
/**
|
|
15
|
-
* Encode correlated intents as a git note on the given commit.
|
|
16
|
-
* Fire-and-forget — logs warning on failure, never throws.
|
|
17
|
-
*/
|
|
18
|
-
export async function encodeIntentAsNote(commitSha, correlations, sessionId, branchContext, driftSummary, cwd) {
|
|
19
|
-
if (correlations.length === 0)
|
|
20
|
-
return false;
|
|
21
|
-
const note = {
|
|
22
|
-
v: 1,
|
|
23
|
-
intents: correlations.map((c) => ({
|
|
24
|
-
id: c.rootIntentId,
|
|
25
|
-
prompt: (c.prompt ?? "").slice(0, 200),
|
|
26
|
-
tools: c.toolChain,
|
|
27
|
-
entities: c.entities,
|
|
28
|
-
files: c.files,
|
|
29
|
-
risk: "l",
|
|
30
|
-
})),
|
|
31
|
-
drift: {
|
|
32
|
-
a: driftSummary.added,
|
|
33
|
-
m: driftSummary.modified,
|
|
34
|
-
d: driftSummary.deleted,
|
|
35
|
-
},
|
|
36
|
-
sid: sessionId.slice(0, 12),
|
|
37
|
-
br: branchContext?.currentBranch ?? "unknown",
|
|
38
|
-
};
|
|
39
|
-
const noteJson = JSON.stringify(note);
|
|
40
|
-
if (noteJson.length > 2048) {
|
|
41
|
-
log.warn(`Note payload exceeds 2KB budget (${noteJson.length} bytes) — writing anyway`);
|
|
42
|
-
}
|
|
43
|
-
try {
|
|
44
|
-
await writeNote(cwd ?? process.cwd(), "unerr", commitSha, noteJson);
|
|
45
|
-
log.info(`Note written for ${commitSha.slice(0, 8)} (${correlations.length} intent(s), ${noteJson.length}B)`);
|
|
46
|
-
return true;
|
|
47
|
-
}
|
|
48
|
-
catch (err) {
|
|
49
|
-
log.warn(`Failed to write git note for ${commitSha.slice(0, 8)}: ${err instanceof Error ? err.message : String(err)}`);
|
|
50
|
-
return false;
|
|
51
|
-
}
|
|
52
|
-
}
|
|
@@ -1,159 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Intent Token Tracker — groups tool calls by intent and tracks tokens per task.
|
|
3
|
-
*
|
|
4
|
-
* Each "intent" represents a coherent unit of work (e.g., "add error handling
|
|
5
|
-
* to auth module"). Tool calls within an intent are aggregated for token
|
|
6
|
-
* consumption, token savings, entity modifications, and duration tracking.
|
|
7
|
-
*
|
|
8
|
-
* Intent groups start when a new intentId is first seen and end when
|
|
9
|
-
* explicitly marked with an outcome. Abandoned intents are detected via
|
|
10
|
-
* inactivity timeout (30 minutes).
|
|
11
|
-
*/
|
|
12
|
-
import { createModuleLogger } from "../utils/logger.js";
|
|
13
|
-
const log = createModuleLogger("intent-tokens");
|
|
14
|
-
const ABANDON_TIMEOUT_MS = 30 * 60 * 1000;
|
|
15
|
-
/**
|
|
16
|
-
* Creates an intent-scoped token tracker. Each intent groups related tool calls
|
|
17
|
-
* and accumulates token consumption/savings metrics.
|
|
18
|
-
*/
|
|
19
|
-
export function createIntentTokenTracker(options) {
|
|
20
|
-
const dashboardSink = options?.dashboardSink;
|
|
21
|
-
const intents = new Map();
|
|
22
|
-
let activeIntentId = null;
|
|
23
|
-
function startIntent(intentId, prompt) {
|
|
24
|
-
if (intents.has(intentId))
|
|
25
|
-
return;
|
|
26
|
-
const now = Date.now();
|
|
27
|
-
intents.set(intentId, {
|
|
28
|
-
intentId,
|
|
29
|
-
prompt,
|
|
30
|
-
toolCalls: 0,
|
|
31
|
-
tokensConsumed: 0,
|
|
32
|
-
tokensSaved: 0,
|
|
33
|
-
entities: new Set(),
|
|
34
|
-
startedAt: now,
|
|
35
|
-
lastActivity: now,
|
|
36
|
-
outcome: "in_progress",
|
|
37
|
-
});
|
|
38
|
-
activeIntentId = intentId;
|
|
39
|
-
dashboardSink?.({
|
|
40
|
-
phase: "started",
|
|
41
|
-
intent_id: intentId,
|
|
42
|
-
prompt,
|
|
43
|
-
});
|
|
44
|
-
}
|
|
45
|
-
function recordToolCall(intentId, tokens, saved, entity) {
|
|
46
|
-
let state = intents.get(intentId);
|
|
47
|
-
if (!state) {
|
|
48
|
-
startIntent(intentId);
|
|
49
|
-
state = intents.get(intentId);
|
|
50
|
-
}
|
|
51
|
-
state.toolCalls += 1;
|
|
52
|
-
state.tokensConsumed += tokens;
|
|
53
|
-
state.tokensSaved += saved;
|
|
54
|
-
state.lastActivity = Date.now();
|
|
55
|
-
if (entity) {
|
|
56
|
-
state.entities.add(entity);
|
|
57
|
-
}
|
|
58
|
-
activeIntentId = intentId;
|
|
59
|
-
}
|
|
60
|
-
function getGroup(intentId) {
|
|
61
|
-
const state = intents.get(intentId);
|
|
62
|
-
if (!state)
|
|
63
|
-
return null;
|
|
64
|
-
return stateToGroup(state);
|
|
65
|
-
}
|
|
66
|
-
function getAllGroups() {
|
|
67
|
-
checkAbandoned();
|
|
68
|
-
const groups = [];
|
|
69
|
-
for (const state of intents.values()) {
|
|
70
|
-
groups.push(stateToGroup(state));
|
|
71
|
-
}
|
|
72
|
-
return groups.sort((a, b) => b.durationMs - a.durationMs);
|
|
73
|
-
}
|
|
74
|
-
function markOutcome(intentId, outcome) {
|
|
75
|
-
const state = intents.get(intentId);
|
|
76
|
-
if (!state)
|
|
77
|
-
return;
|
|
78
|
-
state.outcome = outcome;
|
|
79
|
-
state.lastActivity = Date.now();
|
|
80
|
-
dashboardSink?.({
|
|
81
|
-
phase: "outcome",
|
|
82
|
-
intent_id: intentId,
|
|
83
|
-
outcome,
|
|
84
|
-
});
|
|
85
|
-
if (outcome !== "in_progress" && activeIntentId === intentId) {
|
|
86
|
-
activeIntentId = null;
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
function getActiveIntentId() {
|
|
90
|
-
if (!activeIntentId)
|
|
91
|
-
return null;
|
|
92
|
-
const state = intents.get(activeIntentId);
|
|
93
|
-
if (!state || state.outcome !== "in_progress") {
|
|
94
|
-
activeIntentId = null;
|
|
95
|
-
return null;
|
|
96
|
-
}
|
|
97
|
-
return activeIntentId;
|
|
98
|
-
}
|
|
99
|
-
function getTotalTokens() {
|
|
100
|
-
let consumed = 0;
|
|
101
|
-
let saved = 0;
|
|
102
|
-
for (const state of intents.values()) {
|
|
103
|
-
consumed += state.tokensConsumed;
|
|
104
|
-
saved += state.tokensSaved;
|
|
105
|
-
}
|
|
106
|
-
const total = consumed + saved;
|
|
107
|
-
const ratio = total > 0 ? saved / total : 0;
|
|
108
|
-
return {
|
|
109
|
-
consumed,
|
|
110
|
-
saved,
|
|
111
|
-
ratio: Math.round(ratio * 1000) / 1000,
|
|
112
|
-
};
|
|
113
|
-
}
|
|
114
|
-
function pruneAbandoned() {
|
|
115
|
-
return checkAbandoned();
|
|
116
|
-
}
|
|
117
|
-
function checkAbandoned() {
|
|
118
|
-
const now = Date.now();
|
|
119
|
-
let count = 0;
|
|
120
|
-
for (const state of intents.values()) {
|
|
121
|
-
if (state.outcome !== "in_progress")
|
|
122
|
-
continue;
|
|
123
|
-
if (now - state.lastActivity > ABANDON_TIMEOUT_MS) {
|
|
124
|
-
state.outcome = "abandoned";
|
|
125
|
-
count += 1;
|
|
126
|
-
dashboardSink?.({
|
|
127
|
-
phase: "abandoned",
|
|
128
|
-
intent_id: state.intentId,
|
|
129
|
-
});
|
|
130
|
-
log.info(`Intent ${state.intentId} marked abandoned (inactive ${Math.round((now - state.lastActivity) / 60_000)}min)`);
|
|
131
|
-
}
|
|
132
|
-
}
|
|
133
|
-
return count;
|
|
134
|
-
}
|
|
135
|
-
function stateToGroup(state) {
|
|
136
|
-
const now = Date.now();
|
|
137
|
-
const endTime = state.outcome === "in_progress" ? now : state.lastActivity;
|
|
138
|
-
return {
|
|
139
|
-
intentId: state.intentId,
|
|
140
|
-
prompt: state.prompt,
|
|
141
|
-
toolCalls: state.toolCalls,
|
|
142
|
-
tokensConsumed: state.tokensConsumed,
|
|
143
|
-
tokensSaved: state.tokensSaved,
|
|
144
|
-
entitiesModified: Array.from(state.entities),
|
|
145
|
-
durationMs: endTime - state.startedAt,
|
|
146
|
-
outcome: state.outcome,
|
|
147
|
-
};
|
|
148
|
-
}
|
|
149
|
-
return {
|
|
150
|
-
startIntent,
|
|
151
|
-
recordToolCall,
|
|
152
|
-
getGroup,
|
|
153
|
-
getAllGroups,
|
|
154
|
-
markOutcome,
|
|
155
|
-
getActiveIntentId,
|
|
156
|
-
getTotalTokens,
|
|
157
|
-
pruneAbandoned,
|
|
158
|
-
};
|
|
159
|
-
}
|