@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,202 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Prompt Durability Profiler — analyzes which prompt types produce durable vs fragile code.
|
|
3
|
-
*
|
|
4
|
-
* Classifies ledger entries by action type (add/modify/refactor/fix/delete),
|
|
5
|
-
* target risk level, and file scope. Computes durability per profile bucket
|
|
6
|
-
* and generates recommendations for prompt strategies that historically produce
|
|
7
|
-
* more stable code.
|
|
8
|
-
*
|
|
9
|
-
* Data flow: shadow ledger entries → classification → aggregation → profiles.
|
|
10
|
-
*/
|
|
11
|
-
import { createModuleLogger } from "../utils/logger.js";
|
|
12
|
-
const log = createModuleLogger("prompt-durability");
|
|
13
|
-
const ADD_PATTERNS = [
|
|
14
|
-
/\badd(?:ing|ed|s)?\b/i,
|
|
15
|
-
/\bcreate(?:d|s|ing)?\b/i,
|
|
16
|
-
/\bnew\b/i,
|
|
17
|
-
/\bimplement(?:ed|s|ing)?\b/i,
|
|
18
|
-
/\bintroduc(?:e|ed|ing)\b/i,
|
|
19
|
-
/\bbuild(?:ing|s)?\b/i,
|
|
20
|
-
/\bset\s*up\b/i,
|
|
21
|
-
];
|
|
22
|
-
const MODIFY_PATTERNS = [
|
|
23
|
-
/\bmodif(?:y|ied|ying|ies)\b/i,
|
|
24
|
-
/\bupdate(?:d|s|ing)?\b/i,
|
|
25
|
-
/\bchange(?:d|s|ing)?\b/i,
|
|
26
|
-
/\bedit(?:ed|s|ing)?\b/i,
|
|
27
|
-
/\badjust(?:ed|s|ing)?\b/i,
|
|
28
|
-
/\btweak(?:ed|s|ing)?\b/i,
|
|
29
|
-
/\balter(?:ed|s|ing)?\b/i,
|
|
30
|
-
];
|
|
31
|
-
const REFACTOR_PATTERNS = [
|
|
32
|
-
/\brefactor(?:ed|s|ing)?\b/i,
|
|
33
|
-
/\brestructur(?:e|ed|ing)\b/i,
|
|
34
|
-
/\breorganiz(?:e|ed|ing)\b/i,
|
|
35
|
-
/\bclean\s*up\b/i,
|
|
36
|
-
/\bextract(?:ed|s|ing)?\b/i,
|
|
37
|
-
/\bmove(?:d|s|ing)?\b/i,
|
|
38
|
-
/\brename(?:d|s|ing)?\b/i,
|
|
39
|
-
/\bsimplif(?:y|ied|ying|ies)\b/i,
|
|
40
|
-
];
|
|
41
|
-
const FIX_PATTERNS = [
|
|
42
|
-
/\bfix(?:ed|es|ing)?\b/i,
|
|
43
|
-
/\bbug\s*fix\b/i,
|
|
44
|
-
/\bresolv(?:e|ed|ing)\b/i,
|
|
45
|
-
/\brepair(?:ed|s|ing)?\b/i,
|
|
46
|
-
/\bpatch(?:ed|es|ing)?\b/i,
|
|
47
|
-
/\bcorrect(?:ed|s|ing)?\b/i,
|
|
48
|
-
/\bhandle\b.*\berror\b/i,
|
|
49
|
-
/\bworkaround\b/i,
|
|
50
|
-
];
|
|
51
|
-
const DELETE_PATTERNS = [
|
|
52
|
-
/\bdelete(?:d|s|ing)?\b/i,
|
|
53
|
-
/\bremove(?:d|s|ing)?\b/i,
|
|
54
|
-
/\bdrop(?:ped|s|ping)?\b/i,
|
|
55
|
-
/\bstrip(?:ped|s|ping)?\b/i,
|
|
56
|
-
/\bdeprecate(?:d|s|ing)?\b/i,
|
|
57
|
-
/\beliminate(?:d|s|ing)?\b/i,
|
|
58
|
-
];
|
|
59
|
-
/**
|
|
60
|
-
* Classifies a prompt string into an action type based on keyword patterns.
|
|
61
|
-
*
|
|
62
|
-
* Priority order: fix > refactor > delete > add > modify > other.
|
|
63
|
-
* Fix is highest priority because fix prompts often also contain "modify" or "change".
|
|
64
|
-
*/
|
|
65
|
-
export function extractActionType(prompt) {
|
|
66
|
-
if (!prompt || prompt.trim().length === 0)
|
|
67
|
-
return "other";
|
|
68
|
-
if (FIX_PATTERNS.some((p) => p.test(prompt)))
|
|
69
|
-
return "fix";
|
|
70
|
-
if (REFACTOR_PATTERNS.some((p) => p.test(prompt)))
|
|
71
|
-
return "refactor";
|
|
72
|
-
if (DELETE_PATTERNS.some((p) => p.test(prompt)))
|
|
73
|
-
return "delete";
|
|
74
|
-
if (ADD_PATTERNS.some((p) => p.test(prompt)))
|
|
75
|
-
return "add";
|
|
76
|
-
if (MODIFY_PATTERNS.some((p) => p.test(prompt)))
|
|
77
|
-
return "modify";
|
|
78
|
-
return "other";
|
|
79
|
-
}
|
|
80
|
-
/**
|
|
81
|
-
* Computes durability profiles from ledger entries.
|
|
82
|
-
*
|
|
83
|
-
* Groups entries by (actionType, targetRisk, scope), computes per-bucket
|
|
84
|
-
* durability as survived/total, and attaches recommendations for low-durability buckets.
|
|
85
|
-
*/
|
|
86
|
-
export function computePromptDurabilityProfiles(ledgerEntries) {
|
|
87
|
-
const buckets = new Map();
|
|
88
|
-
for (const entry of ledgerEntries) {
|
|
89
|
-
const actionType = extractActionType(entry.prompt ?? "");
|
|
90
|
-
const targetRisk = normalizeRisk(entry.riskLevel);
|
|
91
|
-
const scope = classifyScope(entry.files);
|
|
92
|
-
const bucketId = `${actionType}:${targetRisk}:${scope}`;
|
|
93
|
-
const existing = buckets.get(bucketId) ?? {
|
|
94
|
-
survived: 0,
|
|
95
|
-
total: 0,
|
|
96
|
-
key: { actionType, targetRisk, scope },
|
|
97
|
-
};
|
|
98
|
-
existing.total += 1;
|
|
99
|
-
if (entry.survived) {
|
|
100
|
-
existing.survived += 1;
|
|
101
|
-
}
|
|
102
|
-
buckets.set(bucketId, existing);
|
|
103
|
-
}
|
|
104
|
-
const profiles = [];
|
|
105
|
-
for (const bucket of buckets.values()) {
|
|
106
|
-
const durability = bucket.total > 0 ? bucket.survived / bucket.total : 0;
|
|
107
|
-
const profile = {
|
|
108
|
-
actionType: bucket.key.actionType,
|
|
109
|
-
targetRisk: bucket.key.targetRisk,
|
|
110
|
-
scope: bucket.key.scope,
|
|
111
|
-
durability: Math.round(durability * 1000) / 1000,
|
|
112
|
-
sampleCount: bucket.total,
|
|
113
|
-
};
|
|
114
|
-
const recommendation = generateRecommendation(profile);
|
|
115
|
-
if (recommendation) {
|
|
116
|
-
profile.recommendation = recommendation;
|
|
117
|
-
}
|
|
118
|
-
profiles.push(profile);
|
|
119
|
-
}
|
|
120
|
-
profiles.sort((a, b) => {
|
|
121
|
-
if (a.durability !== b.durability)
|
|
122
|
-
return a.durability - b.durability;
|
|
123
|
-
return b.sampleCount - a.sampleCount;
|
|
124
|
-
});
|
|
125
|
-
return profiles;
|
|
126
|
-
}
|
|
127
|
-
/**
|
|
128
|
-
* Returns the overall durability across all profiles, weighted by sample count.
|
|
129
|
-
*/
|
|
130
|
-
export function computeOverallDurability(profiles) {
|
|
131
|
-
let totalWeighted = 0;
|
|
132
|
-
let totalSamples = 0;
|
|
133
|
-
for (const p of profiles) {
|
|
134
|
-
totalWeighted += p.durability * p.sampleCount;
|
|
135
|
-
totalSamples += p.sampleCount;
|
|
136
|
-
}
|
|
137
|
-
return totalSamples > 0
|
|
138
|
-
? Math.round((totalWeighted / totalSamples) * 1000) / 1000
|
|
139
|
-
: 0;
|
|
140
|
-
}
|
|
141
|
-
/**
|
|
142
|
-
* Returns the top N most fragile profiles (lowest durability with sufficient samples).
|
|
143
|
-
*/
|
|
144
|
-
export function getMostFragile(profiles, limit = 5, minSamples = 3) {
|
|
145
|
-
return profiles
|
|
146
|
-
.filter((p) => p.sampleCount >= minSamples)
|
|
147
|
-
.sort((a, b) => a.durability - b.durability)
|
|
148
|
-
.slice(0, limit);
|
|
149
|
-
}
|
|
150
|
-
/**
|
|
151
|
-
* Returns the top N most durable profiles.
|
|
152
|
-
*/
|
|
153
|
-
export function getMostDurable(profiles, limit = 5, minSamples = 3) {
|
|
154
|
-
return profiles
|
|
155
|
-
.filter((p) => p.sampleCount >= minSamples)
|
|
156
|
-
.sort((a, b) => b.durability - a.durability)
|
|
157
|
-
.slice(0, limit);
|
|
158
|
-
}
|
|
159
|
-
function normalizeRisk(risk) {
|
|
160
|
-
if (!risk)
|
|
161
|
-
return "low";
|
|
162
|
-
const lower = risk.toLowerCase();
|
|
163
|
-
if (lower === "critical")
|
|
164
|
-
return "critical";
|
|
165
|
-
if (lower === "high")
|
|
166
|
-
return "high";
|
|
167
|
-
if (lower === "medium")
|
|
168
|
-
return "medium";
|
|
169
|
-
return "low";
|
|
170
|
-
}
|
|
171
|
-
function classifyScope(files) {
|
|
172
|
-
if (!files || files.length <= 1)
|
|
173
|
-
return "single_file";
|
|
174
|
-
return "multi_file";
|
|
175
|
-
}
|
|
176
|
-
function generateRecommendation(profile) {
|
|
177
|
-
if (profile.sampleCount < 3)
|
|
178
|
-
return undefined;
|
|
179
|
-
if (profile.durability < 0.3) {
|
|
180
|
-
if (profile.targetRisk === "critical" && profile.scope === "multi_file") {
|
|
181
|
-
return "Multi-file changes to critical entities have very low durability. Break into smaller, single-file changes with tests between each.";
|
|
182
|
-
}
|
|
183
|
-
if (profile.actionType === "refactor" &&
|
|
184
|
-
profile.targetRisk === "critical") {
|
|
185
|
-
return "Refactoring critical entities is high-risk. Consider adding tests first, then refactoring in small verified steps.";
|
|
186
|
-
}
|
|
187
|
-
if (profile.actionType === "modify" && profile.scope === "multi_file") {
|
|
188
|
-
return "Multi-file modifications have low survival. Consider narrowing scope or creating a working snapshot before proceeding.";
|
|
189
|
-
}
|
|
190
|
-
return `${profile.actionType} actions on ${profile.targetRisk}-risk entities have low durability (${Math.round(profile.durability * 100)}%). Consider smaller increments with verification.`;
|
|
191
|
-
}
|
|
192
|
-
if (profile.durability < 0.5) {
|
|
193
|
-
if (profile.actionType === "fix") {
|
|
194
|
-
return "Fix attempts have moderate durability. Ensure root cause analysis before applying fixes.";
|
|
195
|
-
}
|
|
196
|
-
return `${profile.actionType} actions show moderate durability. Consider verifying with tests after each change.`;
|
|
197
|
-
}
|
|
198
|
-
if (profile.durability >= 0.9 && profile.sampleCount >= 5) {
|
|
199
|
-
return `${profile.actionType} actions on ${profile.targetRisk}-risk entities are highly durable — this is a strong pattern.`;
|
|
200
|
-
}
|
|
201
|
-
return undefined;
|
|
202
|
-
}
|
|
@@ -1,213 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Sprint 10.7: Code Quality Signals — Durability Score + Developer Correction Tracking.
|
|
3
|
-
*
|
|
4
|
-
* Two signals:
|
|
5
|
-
* 1. Code Durability Score: how long AI-generated code survives before modification.
|
|
6
|
-
* 2. Developer Corrections: detects when humans edit AI-generated lines.
|
|
7
|
-
*
|
|
8
|
-
* Computed on drift events. Stored locally, stored locally.
|
|
9
|
-
*
|
|
10
|
-
* Design authority: Phase 5.5 §1.6.1 (Code Durability Score), §1.6.2 (Developer Correction Tracking)
|
|
11
|
-
*/
|
|
12
|
-
import { existsSync, readFileSync, writeFileSync } from "node:fs";
|
|
13
|
-
import { join } from "node:path";
|
|
14
|
-
/** stderr logger */
|
|
15
|
-
const _log = {
|
|
16
|
-
info: (msg) => process.stderr.write(`[unerr:quality] ${msg}\n`),
|
|
17
|
-
};
|
|
18
|
-
/** Durability score thresholds (hours). */
|
|
19
|
-
const FRAGILE_THRESHOLD_MS = 24 * 60 * 60 * 1000; // <24h = fragile
|
|
20
|
-
const MODERATE_THRESHOLD_MS = 7 * 24 * 60 * 60 * 1000; // 7d = moderate
|
|
21
|
-
const DURABLE_THRESHOLD_MS = 30 * 24 * 60 * 60 * 1000; // 30d = durable
|
|
22
|
-
/** Human origin threshold: >60s after last sync = human timing. */
|
|
23
|
-
const HUMAN_TIMING_MS = 60_000;
|
|
24
|
-
/**
|
|
25
|
-
* Compute durability score from survival time.
|
|
26
|
-
*/
|
|
27
|
-
function computeDurabilityFromAge(survivalMs) {
|
|
28
|
-
if (survivalMs < FRAGILE_THRESHOLD_MS) {
|
|
29
|
-
// 0.0 - 0.3 range
|
|
30
|
-
return 0.1 + (survivalMs / FRAGILE_THRESHOLD_MS) * 0.2;
|
|
31
|
-
}
|
|
32
|
-
if (survivalMs < MODERATE_THRESHOLD_MS) {
|
|
33
|
-
// 0.3 - 0.7 range
|
|
34
|
-
return (0.3 +
|
|
35
|
-
((survivalMs - FRAGILE_THRESHOLD_MS) /
|
|
36
|
-
(MODERATE_THRESHOLD_MS - FRAGILE_THRESHOLD_MS)) *
|
|
37
|
-
0.4);
|
|
38
|
-
}
|
|
39
|
-
if (survivalMs < DURABLE_THRESHOLD_MS) {
|
|
40
|
-
// 0.7 - 0.9 range
|
|
41
|
-
return (0.7 +
|
|
42
|
-
((survivalMs - MODERATE_THRESHOLD_MS) /
|
|
43
|
-
(DURABLE_THRESHOLD_MS - MODERATE_THRESHOLD_MS)) *
|
|
44
|
-
0.2);
|
|
45
|
-
}
|
|
46
|
-
return 0.9;
|
|
47
|
-
}
|
|
48
|
-
export class QualitySignalTracker {
|
|
49
|
-
signalsPath;
|
|
50
|
-
signals;
|
|
51
|
-
/** Maximum corrections to retain in memory/disk. */
|
|
52
|
-
static MAX_CORRECTIONS = 200;
|
|
53
|
-
constructor(unerrDir) {
|
|
54
|
-
this.signalsPath = join(unerrDir, "state", "quality_signals.json");
|
|
55
|
-
this.signals = this.load();
|
|
56
|
-
}
|
|
57
|
-
/**
|
|
58
|
-
* Record a drift event for an entity.
|
|
59
|
-
* Called by DriftTracker when an entity is modified.
|
|
60
|
-
*
|
|
61
|
-
* @param entityKey Entity key
|
|
62
|
-
* @param entityName Entity name
|
|
63
|
-
* @param origin "ai" | "human" | "mixed" from DriftTracker attribution
|
|
64
|
-
* @param lastSyncTimestamp Timestamp of last sync_local_diff
|
|
65
|
-
* @param ledgerEntries Recent ledger entries for AI creation lookup
|
|
66
|
-
*/
|
|
67
|
-
onEntityModified(entityKey, entityName, origin, lastSyncTimestamp, ledgerEntries) {
|
|
68
|
-
const now = Date.now();
|
|
69
|
-
// Check if this entity was previously created/modified by AI
|
|
70
|
-
const aiEntry = this.findAiCreationEntry(entityKey, entityName, ledgerEntries);
|
|
71
|
-
if (aiEntry) {
|
|
72
|
-
const aiCreatedAt = new Date(aiEntry.ts).getTime();
|
|
73
|
-
const survivalMs = now - aiCreatedAt;
|
|
74
|
-
// Update durability score
|
|
75
|
-
this.signals.durabilityScores[entityKey] = {
|
|
76
|
-
entityKey,
|
|
77
|
-
entityName,
|
|
78
|
-
score: computeDurabilityFromAge(survivalMs),
|
|
79
|
-
survivalMs,
|
|
80
|
-
aiCreatedAt: aiEntry.ts,
|
|
81
|
-
computedAt: new Date().toISOString(),
|
|
82
|
-
};
|
|
83
|
-
// Check for developer correction (human modifying AI code)
|
|
84
|
-
if (origin === "human" && lastSyncTimestamp > 0) {
|
|
85
|
-
const timeSinceSync = now - lastSyncTimestamp;
|
|
86
|
-
if (timeSinceSync > HUMAN_TIMING_MS) {
|
|
87
|
-
this.signals.corrections.push({
|
|
88
|
-
entityKey,
|
|
89
|
-
entityName,
|
|
90
|
-
correctionLatencyMs: timeSinceSync,
|
|
91
|
-
originalPrompt: this.extractPrompt(aiEntry),
|
|
92
|
-
linesChanged: 0, // Updated by caller if available
|
|
93
|
-
correctedAt: new Date().toISOString(),
|
|
94
|
-
});
|
|
95
|
-
// Cap corrections list
|
|
96
|
-
if (this.signals.corrections.length >
|
|
97
|
-
QualitySignalTracker.MAX_CORRECTIONS) {
|
|
98
|
-
this.signals.corrections = this.signals.corrections.slice(-QualitySignalTracker.MAX_CORRECTIONS);
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
}
|
|
103
|
-
this.signals.updatedAt = new Date().toISOString();
|
|
104
|
-
}
|
|
105
|
-
/**
|
|
106
|
-
* Get current quality signals for ledger flush.
|
|
107
|
-
*/
|
|
108
|
-
getSignals() {
|
|
109
|
-
return { ...this.signals };
|
|
110
|
-
}
|
|
111
|
-
/**
|
|
112
|
-
* Get durability scores for specific entities.
|
|
113
|
-
*/
|
|
114
|
-
getDurabilityScores(entityKeys) {
|
|
115
|
-
const result = {};
|
|
116
|
-
for (const key of entityKeys) {
|
|
117
|
-
const score = this.signals.durabilityScores[key];
|
|
118
|
-
if (score) {
|
|
119
|
-
result[key] = score.score;
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
return result;
|
|
123
|
-
}
|
|
124
|
-
/**
|
|
125
|
-
* Get average durability score across all tracked entities.
|
|
126
|
-
*/
|
|
127
|
-
getAverageDurability() {
|
|
128
|
-
const scores = Object.values(this.signals.durabilityScores);
|
|
129
|
-
if (scores.length === 0)
|
|
130
|
-
return 0;
|
|
131
|
-
const sum = scores.reduce((acc, s) => acc + s.score, 0);
|
|
132
|
-
return sum / scores.length;
|
|
133
|
-
}
|
|
134
|
-
/**
|
|
135
|
-
* Get recent corrections (for status display).
|
|
136
|
-
*/
|
|
137
|
-
getRecentCorrections(limit = 10) {
|
|
138
|
-
return this.signals.corrections.slice(-limit);
|
|
139
|
-
}
|
|
140
|
-
/**
|
|
141
|
-
* Persist signals to disk.
|
|
142
|
-
*/
|
|
143
|
-
save() {
|
|
144
|
-
try {
|
|
145
|
-
const dir = join(this.signalsPath, "..");
|
|
146
|
-
if (!existsSync(dir)) {
|
|
147
|
-
const { mkdirSync } = require("node:fs");
|
|
148
|
-
mkdirSync(dir, { recursive: true });
|
|
149
|
-
}
|
|
150
|
-
writeFileSync(this.signalsPath, JSON.stringify(this.signals, null, 2), "utf-8");
|
|
151
|
-
}
|
|
152
|
-
catch {
|
|
153
|
-
// Non-critical — signals persist best-effort
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
/**
|
|
157
|
-
* Find the ledger entry where AI created/modified this entity.
|
|
158
|
-
*/
|
|
159
|
-
findAiCreationEntry(entityKey, entityName, entries) {
|
|
160
|
-
// Look for sync_local_diff entries that mention this entity
|
|
161
|
-
for (let i = entries.length - 1; i >= 0; i--) {
|
|
162
|
-
const entry = entries[i];
|
|
163
|
-
if (!entry || entry.tool !== "sync_local_diff")
|
|
164
|
-
continue;
|
|
165
|
-
const args = entry.args_summary;
|
|
166
|
-
// Check if the diff mentions this entity name
|
|
167
|
-
const diff = args.diff ?? "";
|
|
168
|
-
if (diff.includes(entityName) || diff.includes(entityKey)) {
|
|
169
|
-
return entry;
|
|
170
|
-
}
|
|
171
|
-
// Check structured files for entity reference
|
|
172
|
-
const files = args.files;
|
|
173
|
-
if (files) {
|
|
174
|
-
for (const file of files) {
|
|
175
|
-
if (file.entities?.includes(entityName) ||
|
|
176
|
-
file.entities?.includes(entityKey)) {
|
|
177
|
-
return entry;
|
|
178
|
-
}
|
|
179
|
-
}
|
|
180
|
-
}
|
|
181
|
-
}
|
|
182
|
-
return null;
|
|
183
|
-
}
|
|
184
|
-
/**
|
|
185
|
-
* Extract the prompt text from a ledger entry.
|
|
186
|
-
*/
|
|
187
|
-
extractPrompt(entry) {
|
|
188
|
-
const prompt = entry.args_summary.prompt ?? "";
|
|
189
|
-
return prompt.slice(0, 200); // Truncate for storage
|
|
190
|
-
}
|
|
191
|
-
/**
|
|
192
|
-
* Load signals from disk.
|
|
193
|
-
*/
|
|
194
|
-
load() {
|
|
195
|
-
if (!existsSync(this.signalsPath)) {
|
|
196
|
-
return {
|
|
197
|
-
durabilityScores: {},
|
|
198
|
-
corrections: [],
|
|
199
|
-
updatedAt: new Date().toISOString(),
|
|
200
|
-
};
|
|
201
|
-
}
|
|
202
|
-
try {
|
|
203
|
-
return JSON.parse(readFileSync(this.signalsPath, "utf-8"));
|
|
204
|
-
}
|
|
205
|
-
catch {
|
|
206
|
-
return {
|
|
207
|
-
durabilityScores: {},
|
|
208
|
-
corrections: [],
|
|
209
|
-
updatedAt: new Date().toISOString(),
|
|
210
|
-
};
|
|
211
|
-
}
|
|
212
|
-
}
|
|
213
|
-
}
|
|
@@ -1,73 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Shadow Ledger Redactor (ST-6).
|
|
3
|
-
*
|
|
4
|
-
* Strips known secret patterns from any `args_summary` payload before it lands
|
|
5
|
-
* in `.unerr/ledger/shadow.jsonl`. Best-effort defence — patterns are tuned
|
|
6
|
-
* for the obvious cases (API tokens, bearer headers, env-style assignments)
|
|
7
|
-
* and DO NOT replace a real secret scanner. We deliberately stay regex-only
|
|
8
|
-
* to keep the hot path under a millisecond.
|
|
9
|
-
*
|
|
10
|
-
* Replacement: `<redacted>` (visible in logs and the timeline UI so users know
|
|
11
|
-
* a value was scrubbed).
|
|
12
|
-
*/
|
|
13
|
-
const REDACTION_PATTERNS = [
|
|
14
|
-
// Anthropic / OpenAI-style secret prefixes
|
|
15
|
-
/sk-[a-zA-Z0-9_-]{16,}/g,
|
|
16
|
-
/sk_(?:live|test)_[a-zA-Z0-9_-]{16,}/g,
|
|
17
|
-
// GitHub personal access tokens (ghp_, gho_, ghu_, ghs_, ghr_)
|
|
18
|
-
/gh[opsur]_[A-Za-z0-9_]{20,}/g,
|
|
19
|
-
// Slack tokens (xox*-...)
|
|
20
|
-
/xox[abprs]-[A-Za-z0-9-]{10,}/g,
|
|
21
|
-
// AWS access key id
|
|
22
|
-
/AKIA[0-9A-Z]{16}/g,
|
|
23
|
-
// Google API keys
|
|
24
|
-
/AIza[0-9A-Za-z_-]{30,}/g,
|
|
25
|
-
// JWT-ish three-segment base64url
|
|
26
|
-
/eyJ[A-Za-z0-9_-]{10,}\.[A-Za-z0-9_-]{10,}\.[A-Za-z0-9_-]{10,}/g,
|
|
27
|
-
// Authorization headers (Bearer / Basic / Token)
|
|
28
|
-
/(?:Bearer|Basic|Token)\s+[A-Za-z0-9._=/+-]{8,}/gi,
|
|
29
|
-
// password|secret|token|api_key = value | : value
|
|
30
|
-
/\b(?:password|passwd|secret|token|api[-_]?key)\s*[:=]\s*['"]?[^\s'"&]+/gi,
|
|
31
|
-
// VAR=value style env assignments where VAR looks secret-like
|
|
32
|
-
/\b[A-Z][A-Z0-9_]*(?:KEY|TOKEN|SECRET|PASS|PASSWORD|CRED|CREDENTIAL)\s*=\s*[^\s]+/g,
|
|
33
|
-
// .env-style export FOO=secret
|
|
34
|
-
/export\s+[A-Z][A-Z0-9_]*(?:KEY|TOKEN|SECRET|PASS|PASSWORD|CRED|CREDENTIAL)=[^\s]+/g,
|
|
35
|
-
];
|
|
36
|
-
const REDACTED = "<redacted>";
|
|
37
|
-
export function redactString(input) {
|
|
38
|
-
if (typeof input !== "string" || input.length === 0)
|
|
39
|
-
return input;
|
|
40
|
-
let out = input;
|
|
41
|
-
for (const re of REDACTION_PATTERNS) {
|
|
42
|
-
out = out.replace(re, REDACTED);
|
|
43
|
-
}
|
|
44
|
-
return out;
|
|
45
|
-
}
|
|
46
|
-
/**
|
|
47
|
-
* Walk an arg payload depth-first and redact every string value in place.
|
|
48
|
-
* Arrays/objects are recursed; non-string scalars are returned unchanged.
|
|
49
|
-
* The input is treated as immutable — a new object is returned for any
|
|
50
|
-
* branch that contained a redaction (and the original for branches that
|
|
51
|
-
* didn't, which keeps GC pressure low on clean inputs).
|
|
52
|
-
*/
|
|
53
|
-
export function redactArgs(args) {
|
|
54
|
-
const result = {};
|
|
55
|
-
for (const [k, v] of Object.entries(args)) {
|
|
56
|
-
result[k] = redactValue(v);
|
|
57
|
-
}
|
|
58
|
-
return result;
|
|
59
|
-
}
|
|
60
|
-
function redactValue(v) {
|
|
61
|
-
if (typeof v === "string")
|
|
62
|
-
return redactString(v);
|
|
63
|
-
if (Array.isArray(v))
|
|
64
|
-
return v.map(redactValue);
|
|
65
|
-
if (v && typeof v === "object") {
|
|
66
|
-
const out = {};
|
|
67
|
-
for (const [k, vv] of Object.entries(v)) {
|
|
68
|
-
out[k] = redactValue(vv);
|
|
69
|
-
}
|
|
70
|
-
return out;
|
|
71
|
-
}
|
|
72
|
-
return v;
|
|
73
|
-
}
|
|
@@ -1,161 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Sprint 10.2: Deterministic Rewind — `unerr_revert_to_working_state` MCP tool.
|
|
3
|
-
*
|
|
4
|
-
* Wraps offline-rewind.ts with working snapshot targeting.
|
|
5
|
-
* The MCP tool accepts either a snapshot ID or "latest" to rewind to.
|
|
6
|
-
*
|
|
7
|
-
* Design authority: Phase 5.5 §1.3 (Deterministic Rewind)
|
|
8
|
-
*/
|
|
9
|
-
import { checkoutFile, getChangedFiles } from "../utils/git.js";
|
|
10
|
-
import { createModuleLogger } from "../utils/logger.js";
|
|
11
|
-
import { offlineRewind } from "./offline-rewind.js";
|
|
12
|
-
const log = createModuleLogger("rewind");
|
|
13
|
-
/**
|
|
14
|
-
* Execute a deterministic rewind to a working snapshot.
|
|
15
|
-
*
|
|
16
|
-
* Steps:
|
|
17
|
-
* 1. Resolve snapshot (by ID or "latest")
|
|
18
|
-
* 2. Find the ledger entry closest to the snapshot timestamp
|
|
19
|
-
* 3. Delegate to offlineRewind() for file restoration
|
|
20
|
-
* 4. Return result with blast radius and restored files
|
|
21
|
-
*/
|
|
22
|
-
export async function revertToWorkingState(request) {
|
|
23
|
-
const { snapshotId, cwd, unerrDir, graph, ledger, snapshotStore, dryRun } = request;
|
|
24
|
-
// Step 1: Resolve snapshot
|
|
25
|
-
let snapshot;
|
|
26
|
-
if (snapshotId === "latest") {
|
|
27
|
-
snapshot = snapshotStore.getLatest();
|
|
28
|
-
}
|
|
29
|
-
else {
|
|
30
|
-
snapshot = snapshotStore.get(snapshotId);
|
|
31
|
-
}
|
|
32
|
-
if (!snapshot) {
|
|
33
|
-
return {
|
|
34
|
-
success: false,
|
|
35
|
-
snapshot: null,
|
|
36
|
-
rewindResult: null,
|
|
37
|
-
error: snapshotId === "latest"
|
|
38
|
-
? "No working snapshots found. Use unerr_mark_working to create one."
|
|
39
|
-
: `Snapshot ${snapshotId} not found.`,
|
|
40
|
-
};
|
|
41
|
-
}
|
|
42
|
-
log.info(`Reverting to snapshot ${snapshot.id} at ${snapshot.commitSha.slice(0, 8)} (${snapshot.reason})`);
|
|
43
|
-
// Step 2: Find the closest ledger entry to the snapshot
|
|
44
|
-
const allEntries = ledger.readAllEntries();
|
|
45
|
-
const snapshotTime = new Date(snapshot.timestamp).getTime();
|
|
46
|
-
// Find the entry closest to (but not after) the snapshot timestamp
|
|
47
|
-
let targetEntry = null;
|
|
48
|
-
let closestDelta = Number.POSITIVE_INFINITY;
|
|
49
|
-
for (const entry of allEntries) {
|
|
50
|
-
const entryTime = new Date(entry.ts).getTime();
|
|
51
|
-
const delta = snapshotTime - entryTime;
|
|
52
|
-
// Entry must be at or before snapshot time
|
|
53
|
-
if (delta >= 0 && delta < closestDelta) {
|
|
54
|
-
closestDelta = delta;
|
|
55
|
-
targetEntry = entry;
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
if (!targetEntry) {
|
|
59
|
-
// No ledger entry found — fall back to git-only rewind
|
|
60
|
-
log.info("No matching ledger entry found. Performing git-only rewind.");
|
|
61
|
-
return await performGitOnlyRewind(snapshot, cwd, dryRun);
|
|
62
|
-
}
|
|
63
|
-
const result = await offlineRewind({
|
|
64
|
-
targetEntryId: targetEntry.id,
|
|
65
|
-
cwd,
|
|
66
|
-
unerrDir,
|
|
67
|
-
graph,
|
|
68
|
-
ledger,
|
|
69
|
-
dryRun,
|
|
70
|
-
});
|
|
71
|
-
return {
|
|
72
|
-
success: result.status !== "error",
|
|
73
|
-
snapshot,
|
|
74
|
-
rewindResult: result,
|
|
75
|
-
};
|
|
76
|
-
}
|
|
77
|
-
/**
|
|
78
|
-
* Git-only rewind when no ledger entry matches.
|
|
79
|
-
* Uses checkoutFile to restore files from the snapshot commit.
|
|
80
|
-
*/
|
|
81
|
-
async function performGitOnlyRewind(snapshot, cwd, dryRun) {
|
|
82
|
-
if (dryRun) {
|
|
83
|
-
return {
|
|
84
|
-
success: true,
|
|
85
|
-
snapshot,
|
|
86
|
-
rewindResult: {
|
|
87
|
-
status: "dry_run",
|
|
88
|
-
rewindEntryId: null,
|
|
89
|
-
timelineBranch: snapshot.timelineBranch,
|
|
90
|
-
entriesReverted: 0,
|
|
91
|
-
blastRadius: {
|
|
92
|
-
safeFiles: [],
|
|
93
|
-
conflictedFiles: [],
|
|
94
|
-
affectedEntities: [],
|
|
95
|
-
affectedCallers: 0,
|
|
96
|
-
resolvedInMs: 0,
|
|
97
|
-
},
|
|
98
|
-
filesRestored: [],
|
|
99
|
-
},
|
|
100
|
-
};
|
|
101
|
-
}
|
|
102
|
-
try {
|
|
103
|
-
const changedFiles = await getChangedFiles(cwd, snapshot.commitSha, "HEAD");
|
|
104
|
-
if (changedFiles.length === 0) {
|
|
105
|
-
return {
|
|
106
|
-
success: true,
|
|
107
|
-
snapshot,
|
|
108
|
-
rewindResult: {
|
|
109
|
-
status: "simulated",
|
|
110
|
-
rewindEntryId: null,
|
|
111
|
-
timelineBranch: snapshot.timelineBranch,
|
|
112
|
-
entriesReverted: 0,
|
|
113
|
-
blastRadius: {
|
|
114
|
-
safeFiles: [],
|
|
115
|
-
conflictedFiles: [],
|
|
116
|
-
affectedEntities: [],
|
|
117
|
-
affectedCallers: 0,
|
|
118
|
-
resolvedInMs: 0,
|
|
119
|
-
},
|
|
120
|
-
filesRestored: [],
|
|
121
|
-
},
|
|
122
|
-
};
|
|
123
|
-
}
|
|
124
|
-
const filesRestored = [];
|
|
125
|
-
for (const file of changedFiles) {
|
|
126
|
-
try {
|
|
127
|
-
await checkoutFile(cwd, snapshot.commitSha, file);
|
|
128
|
-
filesRestored.push(file);
|
|
129
|
-
}
|
|
130
|
-
catch {
|
|
131
|
-
// File may not exist at snapshot commit
|
|
132
|
-
}
|
|
133
|
-
}
|
|
134
|
-
return {
|
|
135
|
-
success: true,
|
|
136
|
-
snapshot,
|
|
137
|
-
rewindResult: {
|
|
138
|
-
status: "simulated",
|
|
139
|
-
rewindEntryId: null,
|
|
140
|
-
timelineBranch: snapshot.timelineBranch,
|
|
141
|
-
entriesReverted: 0,
|
|
142
|
-
blastRadius: {
|
|
143
|
-
safeFiles: filesRestored,
|
|
144
|
-
conflictedFiles: [],
|
|
145
|
-
affectedEntities: [],
|
|
146
|
-
affectedCallers: 0,
|
|
147
|
-
resolvedInMs: 0,
|
|
148
|
-
},
|
|
149
|
-
filesRestored,
|
|
150
|
-
},
|
|
151
|
-
};
|
|
152
|
-
}
|
|
153
|
-
catch (err) {
|
|
154
|
-
return {
|
|
155
|
-
success: false,
|
|
156
|
-
snapshot,
|
|
157
|
-
rewindResult: null,
|
|
158
|
-
error: `Git rewind failed: ${err instanceof Error ? err.message : String(err)}`,
|
|
159
|
-
};
|
|
160
|
-
}
|
|
161
|
-
}
|