@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,433 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* SCIP Inline Orchestrator — runs SCIP enrichment as part of the indexing pipeline.
|
|
3
|
-
*
|
|
4
|
-
* NOT a background process. Runs inline after tree-sitter extraction completes.
|
|
5
|
-
* For TypeScript projects (~375 files), completes in ~5-15 seconds.
|
|
6
|
-
*
|
|
7
|
-
* Pipeline:
|
|
8
|
-
* 1. Detect primary language from discovered files
|
|
9
|
-
* 2. Resolve SCIP binary (bundled for TS, cached/PATH for others)
|
|
10
|
-
* 3. If not available, auto-download from GitHub releases
|
|
11
|
-
* 4. Run SCIP indexer → protobuf output
|
|
12
|
-
* 5. Decode protobuf → symbol occurrences
|
|
13
|
-
* 6. Merge with tree-sitter edges → compiler-verified graph
|
|
14
|
-
*/
|
|
15
|
-
import { existsSync, mkdirSync, readFileSync, statSync, writeFileSync, } from "node:fs";
|
|
16
|
-
import { join } from "node:path";
|
|
17
|
-
import { exec } from "../../../utils/exec.js";
|
|
18
|
-
import { createModuleLogger } from "../../../utils/logger.js";
|
|
19
|
-
import { decodeScipOutput } from "./decoder.js";
|
|
20
|
-
import { detectProjectLanguages, detectScipBinary, } from "./detector.js";
|
|
21
|
-
import { downloadScipBinary, getManualInstallInstructions, isAutoDownloadSupported, } from "./downloader.js";
|
|
22
|
-
import { mergeScipResults, } from "./merger.js";
|
|
23
|
-
import { runScipIndexer } from "./runner.js";
|
|
24
|
-
const log = createModuleLogger("scip-orchestrator");
|
|
25
|
-
export async function enrichWithScip(files, projectRoot, existingEdges, entities, options) {
|
|
26
|
-
// Step 1: Detect ALL languages in the project
|
|
27
|
-
const languages = detectProjectLanguages(files);
|
|
28
|
-
if (languages.length === 0) {
|
|
29
|
-
return skipResult(existingEdges, "No supported language detected");
|
|
30
|
-
}
|
|
31
|
-
let currentEdges = markAsStructural(existingEdges);
|
|
32
|
-
let lastRunResult = null;
|
|
33
|
-
let lastDecodeResult = null;
|
|
34
|
-
let lastMergeResult = null;
|
|
35
|
-
let anySucceeded = false;
|
|
36
|
-
const processedLanguages = [];
|
|
37
|
-
for (const { language, fileCount } of languages) {
|
|
38
|
-
// Step 2: Resolve SCIP binary
|
|
39
|
-
let binaryInfo = await detectScipBinary(language);
|
|
40
|
-
// Step 3: Auto-install if not available
|
|
41
|
-
if (!binaryInfo.available) {
|
|
42
|
-
if (language === "csharp") {
|
|
43
|
-
// C# uses dotnet tool install instead of binary download
|
|
44
|
-
const installed = await installScipDotnet();
|
|
45
|
-
if (installed) {
|
|
46
|
-
binaryInfo = await detectScipBinary(language);
|
|
47
|
-
}
|
|
48
|
-
else {
|
|
49
|
-
continue;
|
|
50
|
-
}
|
|
51
|
-
}
|
|
52
|
-
else if (isAutoDownloadSupported(language)) {
|
|
53
|
-
log.info(`SCIP binary not found for ${language}, downloading...`);
|
|
54
|
-
const downloadResult = await downloadScipBinary(language, (msg) => {
|
|
55
|
-
log.info(msg);
|
|
56
|
-
});
|
|
57
|
-
if (downloadResult.success && downloadResult.binaryPath) {
|
|
58
|
-
binaryInfo = await detectScipBinary(language);
|
|
59
|
-
}
|
|
60
|
-
else {
|
|
61
|
-
const manualInstr = getManualInstallInstructions(language);
|
|
62
|
-
log.warn(`SCIP auto-download failed for ${language}: ${downloadResult.error}${manualInstr ? `\n Manual install: ${manualInstr}` : ""}`);
|
|
63
|
-
continue;
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
}
|
|
67
|
-
if (!binaryInfo.available) {
|
|
68
|
-
log.info(`SCIP binary not available for ${language}, skipping`);
|
|
69
|
-
continue;
|
|
70
|
-
}
|
|
71
|
-
// Step 4: Resolve language-specific options. Pre-flight (build-tool /
|
|
72
|
-
// compdb / tsconfig checks) may prompt the user — defer the
|
|
73
|
-
// "processing"/"enrichment" announce logs until AFTER pre-flight so the
|
|
74
|
-
// prompt zone isn't bracketed by SCIP log lines.
|
|
75
|
-
const outputDir = join(projectRoot, ".unerr", "scip");
|
|
76
|
-
const binaryPath = binaryInfo.path ?? binaryInfo.binaryName;
|
|
77
|
-
let extraArgs;
|
|
78
|
-
if (language === "typescript") {
|
|
79
|
-
// scip-typescript requires a tsconfig.json (or jsconfig.json) to resolve the project
|
|
80
|
-
if (!existsSync(join(projectRoot, "tsconfig.json")) &&
|
|
81
|
-
!existsSync(join(projectRoot, "jsconfig.json"))) {
|
|
82
|
-
log.info("SCIP skipping TypeScript: no tsconfig.json or jsconfig.json found in project root. Create one to enable SCIP indexing.");
|
|
83
|
-
continue;
|
|
84
|
-
}
|
|
85
|
-
}
|
|
86
|
-
if (language === "java") {
|
|
87
|
-
const buildToolResult = await resolveJavaBuildTool(projectRoot, options);
|
|
88
|
-
if (buildToolResult) {
|
|
89
|
-
extraArgs = buildToolResult.extraArgs;
|
|
90
|
-
}
|
|
91
|
-
}
|
|
92
|
-
if (language === "cpp") {
|
|
93
|
-
const compdbPath = resolveCompileCommandsJson(projectRoot);
|
|
94
|
-
if (!compdbPath) {
|
|
95
|
-
log.info("SCIP skipping C/C++: no compile_commands.json found. Generate one with CMake (-DCMAKE_EXPORT_COMPILE_COMMANDS=ON), Bear, or your build system.");
|
|
96
|
-
continue;
|
|
97
|
-
}
|
|
98
|
-
extraArgs = [`--compdb-path=${compdbPath}`];
|
|
99
|
-
}
|
|
100
|
-
// Pre-flight is done — safe to announce now that any interactive prompts
|
|
101
|
-
// have been resolved and won't be visually wedged between these logs.
|
|
102
|
-
log.info(`SCIP: processing ${language} (${fileCount} files)`);
|
|
103
|
-
log.info(`SCIP enrichment: ${language} (${binaryInfo.bundled ? "bundled" : "external"}: ${binaryInfo.binaryName})`);
|
|
104
|
-
const runResult = await runScipIndexer({
|
|
105
|
-
language,
|
|
106
|
-
binaryPath,
|
|
107
|
-
projectRoot,
|
|
108
|
-
outputDir,
|
|
109
|
-
timeoutMs: 30_000,
|
|
110
|
-
extraArgs,
|
|
111
|
-
});
|
|
112
|
-
lastRunResult = runResult;
|
|
113
|
-
if (!runResult.success || !runResult.outputPath) {
|
|
114
|
-
log.warn(`SCIP indexer failed for ${language}: ${runResult.error}`);
|
|
115
|
-
continue;
|
|
116
|
-
}
|
|
117
|
-
// Step 5: Decode protobuf output
|
|
118
|
-
const decodeResult = await decodeScipOutput(runResult.outputPath);
|
|
119
|
-
lastDecodeResult = decodeResult;
|
|
120
|
-
// Step 6: Merge with current edges (accumulates across languages)
|
|
121
|
-
const indexedEdges = currentEdges.map((e) => ({
|
|
122
|
-
from_key: e.from_key,
|
|
123
|
-
to_key: e.to_key,
|
|
124
|
-
type: e.type,
|
|
125
|
-
file_path: e.file_path,
|
|
126
|
-
line: e.line,
|
|
127
|
-
}));
|
|
128
|
-
const { edges: enrichedEdges, result: mergeResult } = mergeScipResults(indexedEdges, decodeResult, entities);
|
|
129
|
-
currentEdges = enrichedEdges;
|
|
130
|
-
lastMergeResult = mergeResult;
|
|
131
|
-
anySucceeded = true;
|
|
132
|
-
processedLanguages.push(language);
|
|
133
|
-
log.info(`SCIP ${language}: ${mergeResult.edgesUpgraded} edges upgraded to compiler-verified (${Math.round(runResult.durationMs)}ms)`);
|
|
134
|
-
}
|
|
135
|
-
if (!anySucceeded) {
|
|
136
|
-
return skipResult(existingEdges, `No SCIP binary available for any detected language (${languages.map((l) => l.language).join(", ")})`);
|
|
137
|
-
}
|
|
138
|
-
return {
|
|
139
|
-
language: processedLanguages.join("+"),
|
|
140
|
-
binaryAvailable: true,
|
|
141
|
-
bundled: false,
|
|
142
|
-
runResult: lastRunResult,
|
|
143
|
-
decodeResult: lastDecodeResult,
|
|
144
|
-
mergeResult: lastMergeResult,
|
|
145
|
-
enrichedEdges: currentEdges,
|
|
146
|
-
skipped: false,
|
|
147
|
-
skipReason: null,
|
|
148
|
-
};
|
|
149
|
-
}
|
|
150
|
-
import { writeNeedsInput } from "../../../daemon/registry.js";
|
|
151
|
-
const BUILD_TOOL_FILES = {
|
|
152
|
-
Maven: ["pom.xml"],
|
|
153
|
-
Gradle: [
|
|
154
|
-
"build.gradle",
|
|
155
|
-
"build.gradle.kts",
|
|
156
|
-
"settings.gradle",
|
|
157
|
-
"settings.gradle.kts",
|
|
158
|
-
],
|
|
159
|
-
Bazel: ["BUILD", "BUILD.bazel", "WORKSPACE", "WORKSPACE.bazel"],
|
|
160
|
-
Sbt: ["build.sbt"],
|
|
161
|
-
};
|
|
162
|
-
const WRAPPER_FILES = {
|
|
163
|
-
gradlew: "Gradle",
|
|
164
|
-
"gradlew.bat": "Gradle",
|
|
165
|
-
mvnw: "Maven",
|
|
166
|
-
"mvnw.cmd": "Maven",
|
|
167
|
-
};
|
|
168
|
-
/** Detect Java build tools present in the project root. */
|
|
169
|
-
export function detectJavaBuildTools(projectRoot) {
|
|
170
|
-
const detected = [];
|
|
171
|
-
for (const [tool, files] of Object.entries(BUILD_TOOL_FILES)) {
|
|
172
|
-
if (files.some((f) => existsSync(join(projectRoot, f)))) {
|
|
173
|
-
detected.push(tool);
|
|
174
|
-
}
|
|
175
|
-
}
|
|
176
|
-
return detected;
|
|
177
|
-
}
|
|
178
|
-
/** Read stored Java build tool preference from .unerr/config.json. */
|
|
179
|
-
function getStoredBuildTool(projectRoot) {
|
|
180
|
-
try {
|
|
181
|
-
const configPath = join(projectRoot, ".unerr", "config.json");
|
|
182
|
-
const config = JSON.parse(readFileSync(configPath, "utf-8"));
|
|
183
|
-
return config.javaBuildTool ?? null;
|
|
184
|
-
}
|
|
185
|
-
catch {
|
|
186
|
-
return null;
|
|
187
|
-
}
|
|
188
|
-
}
|
|
189
|
-
/** Store Java build tool preference in .unerr/config.json. */
|
|
190
|
-
function storeBuildTool(projectRoot, tool) {
|
|
191
|
-
const configPath = join(projectRoot, ".unerr", "config.json");
|
|
192
|
-
let config = {};
|
|
193
|
-
try {
|
|
194
|
-
config = JSON.parse(readFileSync(configPath, "utf-8"));
|
|
195
|
-
}
|
|
196
|
-
catch {
|
|
197
|
-
/* new config */
|
|
198
|
-
}
|
|
199
|
-
config.javaBuildTool = tool;
|
|
200
|
-
const dir = join(projectRoot, ".unerr");
|
|
201
|
-
if (!existsSync(dir))
|
|
202
|
-
mkdirSync(dir, { recursive: true });
|
|
203
|
-
writeFileSync(configPath, `${JSON.stringify(config, null, 2)}\n`);
|
|
204
|
-
}
|
|
205
|
-
/**
|
|
206
|
-
* Pure-function build tool chooser — deterministic, no TTY interaction.
|
|
207
|
-
*
|
|
208
|
-
* Priority chain (when multiple tools detected):
|
|
209
|
-
* (a) Explicit config → use it
|
|
210
|
-
* (b) Exactly one tool → use it
|
|
211
|
-
* (c) Any tool + Bazel → Bazel (polyglot/monorepo signal)
|
|
212
|
-
* (d) Any tool + Sbt → Sbt (Scala-first signal)
|
|
213
|
-
* (e) Maven + Gradle → prefer the one with a wrapper (gradlew/mvnw)
|
|
214
|
-
* (f) Tie: prefer the tool whose build files have more recent mtime
|
|
215
|
-
*/
|
|
216
|
-
export function chooseBuildTool(detected, projectRoot) {
|
|
217
|
-
if (detected.length === 0)
|
|
218
|
-
return null;
|
|
219
|
-
if (detected.length === 1) {
|
|
220
|
-
return {
|
|
221
|
-
tool: detected[0],
|
|
222
|
-
reason: "sole build tool detected",
|
|
223
|
-
alternatives: [],
|
|
224
|
-
ambiguous: false,
|
|
225
|
-
};
|
|
226
|
-
}
|
|
227
|
-
// (c) Bazel takes precedence over everything
|
|
228
|
-
if (detected.includes("Bazel")) {
|
|
229
|
-
return {
|
|
230
|
-
tool: "Bazel",
|
|
231
|
-
reason: "Bazel present (polyglot/monorepo signal)",
|
|
232
|
-
alternatives: detected.filter((t) => t !== "Bazel"),
|
|
233
|
-
ambiguous: true,
|
|
234
|
-
};
|
|
235
|
-
}
|
|
236
|
-
// (d) Sbt takes precedence (Scala-first)
|
|
237
|
-
if (detected.includes("Sbt")) {
|
|
238
|
-
return {
|
|
239
|
-
tool: "Sbt",
|
|
240
|
-
reason: "Sbt present (Scala-first project)",
|
|
241
|
-
alternatives: detected.filter((t) => t !== "Sbt"),
|
|
242
|
-
ambiguous: true,
|
|
243
|
-
};
|
|
244
|
-
}
|
|
245
|
-
// (e) Maven + Gradle: check for wrappers
|
|
246
|
-
if (detected.includes("Maven") && detected.includes("Gradle")) {
|
|
247
|
-
const hasGradlew = existsSync(join(projectRoot, "gradlew")) ||
|
|
248
|
-
existsSync(join(projectRoot, "gradlew.bat"));
|
|
249
|
-
const hasMvnw = existsSync(join(projectRoot, "mvnw")) ||
|
|
250
|
-
existsSync(join(projectRoot, "mvnw.cmd"));
|
|
251
|
-
if (hasGradlew && !hasMvnw) {
|
|
252
|
-
return {
|
|
253
|
-
tool: "Gradle",
|
|
254
|
-
reason: "gradlew wrapper present",
|
|
255
|
-
alternatives: ["Maven"],
|
|
256
|
-
ambiguous: true,
|
|
257
|
-
};
|
|
258
|
-
}
|
|
259
|
-
if (hasMvnw && !hasGradlew) {
|
|
260
|
-
return {
|
|
261
|
-
tool: "Maven",
|
|
262
|
-
reason: "mvnw wrapper present",
|
|
263
|
-
alternatives: ["Gradle"],
|
|
264
|
-
ambiguous: true,
|
|
265
|
-
};
|
|
266
|
-
}
|
|
267
|
-
// Both or neither wrapper — fall through to mtime tiebreaker
|
|
268
|
-
}
|
|
269
|
-
// (f) Tiebreak by most recent build-file mtime
|
|
270
|
-
return tiebreakByMtime(detected, projectRoot);
|
|
271
|
-
}
|
|
272
|
-
/** Pick the tool whose build files have the most recent mtime. */
|
|
273
|
-
function tiebreakByMtime(tools, projectRoot) {
|
|
274
|
-
let best = tools[0];
|
|
275
|
-
let bestMtime = 0;
|
|
276
|
-
for (const tool of tools) {
|
|
277
|
-
const files = BUILD_TOOL_FILES[tool];
|
|
278
|
-
for (const f of files) {
|
|
279
|
-
try {
|
|
280
|
-
const mt = statSync(join(projectRoot, f)).mtimeMs;
|
|
281
|
-
if (mt > bestMtime) {
|
|
282
|
-
bestMtime = mt;
|
|
283
|
-
best = tool;
|
|
284
|
-
}
|
|
285
|
-
}
|
|
286
|
-
catch {
|
|
287
|
-
// file doesn't exist for this tool
|
|
288
|
-
}
|
|
289
|
-
}
|
|
290
|
-
}
|
|
291
|
-
return {
|
|
292
|
-
tool: best,
|
|
293
|
-
reason: `most recently modified build file (${best})`,
|
|
294
|
-
alternatives: tools.filter((t) => t !== best),
|
|
295
|
-
ambiguous: true,
|
|
296
|
-
};
|
|
297
|
-
}
|
|
298
|
-
/**
|
|
299
|
-
* Resolve the Java build tool for scip-java --build-tool flag.
|
|
300
|
-
*
|
|
301
|
-
* Deterministic, pure-function resolution — no interactive prompts,
|
|
302
|
-
* no process.stdin.isTTY checks. Works identically under TTY and headless.
|
|
303
|
-
*
|
|
304
|
-
* Priority:
|
|
305
|
-
* 1. Explicit config in .unerr/config.json → use it
|
|
306
|
-
* 2. Exactly one detected → use it
|
|
307
|
-
* 3. Multiple → chooseBuildTool heuristic
|
|
308
|
-
* 4. Cache choice + emit needs_input signal for ambiguous picks
|
|
309
|
-
*/
|
|
310
|
-
async function resolveJavaBuildTool(projectRoot, _options) {
|
|
311
|
-
const detected = detectJavaBuildTools(projectRoot);
|
|
312
|
-
if (detected.length === 0)
|
|
313
|
-
return null;
|
|
314
|
-
// Explicit config takes absolute precedence
|
|
315
|
-
const stored = getStoredBuildTool(projectRoot);
|
|
316
|
-
if (stored && detected.includes(stored)) {
|
|
317
|
-
log.info(`Java build tool: ${stored} (from config)`);
|
|
318
|
-
return { tool: stored, extraArgs: [`--build-tool=${stored}`] };
|
|
319
|
-
}
|
|
320
|
-
const choice = chooseBuildTool(detected, projectRoot);
|
|
321
|
-
if (!choice)
|
|
322
|
-
return null;
|
|
323
|
-
// Cache the deterministic choice for O(1) subsequent boots
|
|
324
|
-
storeBuildTool(projectRoot, choice.tool);
|
|
325
|
-
// Emit needs_input signal when the choice was ambiguous
|
|
326
|
-
if (choice.ambiguous) {
|
|
327
|
-
const signal = {
|
|
328
|
-
type: "needs_input",
|
|
329
|
-
key: "javaBuildTool",
|
|
330
|
-
auto: choice.tool,
|
|
331
|
-
alternatives: choice.alternatives,
|
|
332
|
-
reason: choice.reason,
|
|
333
|
-
};
|
|
334
|
-
writeNeedsInput(projectRoot, [signal]);
|
|
335
|
-
log.info(`Java build tool: ${choice.tool} (auto: ${choice.reason}). Override: unerr daemon config . --java-build-tool=<tool>`);
|
|
336
|
-
}
|
|
337
|
-
else {
|
|
338
|
-
log.info(`Java build tool: ${choice.tool} (${choice.reason})`);
|
|
339
|
-
}
|
|
340
|
-
return {
|
|
341
|
-
tool: choice.tool,
|
|
342
|
-
extraArgs: [`--build-tool=${choice.tool}`],
|
|
343
|
-
};
|
|
344
|
-
}
|
|
345
|
-
// ── C/C++ Compile Commands Detection ─────────────────────────────
|
|
346
|
-
/** Common locations for compile_commands.json in C/C++ projects. */
|
|
347
|
-
const COMPDB_SEARCH_PATHS = [
|
|
348
|
-
"compile_commands.json",
|
|
349
|
-
"build/compile_commands.json",
|
|
350
|
-
"cmake-build-debug/compile_commands.json",
|
|
351
|
-
"cmake-build-release/compile_commands.json",
|
|
352
|
-
"out/compile_commands.json",
|
|
353
|
-
];
|
|
354
|
-
/**
|
|
355
|
-
* Find compile_commands.json in the project.
|
|
356
|
-
* scip-clang requires a compilation database to index C/C++ code.
|
|
357
|
-
* Returns the absolute path if found, null otherwise.
|
|
358
|
-
*/
|
|
359
|
-
function resolveCompileCommandsJson(projectRoot) {
|
|
360
|
-
for (const relPath of COMPDB_SEARCH_PATHS) {
|
|
361
|
-
const absPath = join(projectRoot, relPath);
|
|
362
|
-
if (existsSync(absPath)) {
|
|
363
|
-
log.info(`Found compile_commands.json at ${relPath}`);
|
|
364
|
-
return absPath;
|
|
365
|
-
}
|
|
366
|
-
}
|
|
367
|
-
return null;
|
|
368
|
-
}
|
|
369
|
-
// ── C# scip-dotnet Installation ──────────────────────────────────
|
|
370
|
-
/**
|
|
371
|
-
* Install scip-dotnet via `dotnet tool install`.
|
|
372
|
-
* Requires the .NET SDK (dotnet CLI) to be on PATH.
|
|
373
|
-
* Returns true if installation succeeded, false otherwise.
|
|
374
|
-
*/
|
|
375
|
-
async function installScipDotnet() {
|
|
376
|
-
// Check if dotnet is available
|
|
377
|
-
try {
|
|
378
|
-
const dotnetCheck = await exec("which", ["dotnet"]);
|
|
379
|
-
if (dotnetCheck.exitCode !== 0) {
|
|
380
|
-
log.warn("SCIP skipping C#: dotnet CLI not found on PATH. Install the .NET SDK to enable SCIP indexing for C#.");
|
|
381
|
-
return false;
|
|
382
|
-
}
|
|
383
|
-
}
|
|
384
|
-
catch {
|
|
385
|
-
log.warn("SCIP skipping C#: dotnet CLI not found on PATH. Install the .NET SDK to enable SCIP indexing for C#.");
|
|
386
|
-
return false;
|
|
387
|
-
}
|
|
388
|
-
log.info("Installing scip-dotnet via dotnet tool install...");
|
|
389
|
-
try {
|
|
390
|
-
const result = await exec("dotnet", [
|
|
391
|
-
"tool",
|
|
392
|
-
"install",
|
|
393
|
-
"--global",
|
|
394
|
-
"scip-dotnet",
|
|
395
|
-
]);
|
|
396
|
-
if (result.exitCode === 0) {
|
|
397
|
-
log.info("scip-dotnet installed successfully");
|
|
398
|
-
return true;
|
|
399
|
-
}
|
|
400
|
-
// Exit code 1 with "already installed" is OK
|
|
401
|
-
if (result.stderr.includes("already installed")) {
|
|
402
|
-
log.info("scip-dotnet already installed");
|
|
403
|
-
return true;
|
|
404
|
-
}
|
|
405
|
-
log.warn(`scip-dotnet install failed: ${result.stderr.slice(0, 300)}`);
|
|
406
|
-
return false;
|
|
407
|
-
}
|
|
408
|
-
catch (err) {
|
|
409
|
-
log.warn(`scip-dotnet install failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
410
|
-
return false;
|
|
411
|
-
}
|
|
412
|
-
}
|
|
413
|
-
function skipResult(existingEdges, reason) {
|
|
414
|
-
log.info(`SCIP skipped: ${reason}`);
|
|
415
|
-
return {
|
|
416
|
-
language: null,
|
|
417
|
-
binaryAvailable: false,
|
|
418
|
-
bundled: false,
|
|
419
|
-
runResult: null,
|
|
420
|
-
decodeResult: null,
|
|
421
|
-
mergeResult: null,
|
|
422
|
-
enrichedEdges: markAsStructural(existingEdges),
|
|
423
|
-
skipped: true,
|
|
424
|
-
skipReason: reason,
|
|
425
|
-
};
|
|
426
|
-
}
|
|
427
|
-
function markAsStructural(edges) {
|
|
428
|
-
return edges.map((e) => ({
|
|
429
|
-
...e,
|
|
430
|
-
confidence: "structural",
|
|
431
|
-
scipVerified: false,
|
|
432
|
-
}));
|
|
433
|
-
}
|
|
@@ -1,98 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* SCIP Invocation Runner — spawns SCIP binary, captures protobuf output.
|
|
3
|
-
*
|
|
4
|
-
* For bundled binaries (TypeScript), uses the resolved path directly from
|
|
5
|
-
* node_modules/.bin. For external binaries, uses the PATH-resolved name.
|
|
6
|
-
*
|
|
7
|
-
* Resource-safe: respects timeout (30s default for inline indexing).
|
|
8
|
-
*/
|
|
9
|
-
import { existsSync, mkdirSync } from "node:fs";
|
|
10
|
-
import { join } from "node:path";
|
|
11
|
-
import { exec } from "../../../utils/exec.js";
|
|
12
|
-
import { createModuleLogger } from "../../../utils/logger.js";
|
|
13
|
-
const log = createModuleLogger("scip-runner");
|
|
14
|
-
/**
|
|
15
|
-
* Run a SCIP indexer and capture the output.
|
|
16
|
-
* Default timeout is 30s (inline indexing — not background).
|
|
17
|
-
*/
|
|
18
|
-
export async function runScipIndexer(options) {
|
|
19
|
-
const start = performance.now();
|
|
20
|
-
const { language, binaryPath, projectRoot, outputDir, timeoutMs = 30_000, } = options;
|
|
21
|
-
if (!existsSync(outputDir)) {
|
|
22
|
-
mkdirSync(outputDir, { recursive: true });
|
|
23
|
-
}
|
|
24
|
-
const outputPath = join(outputDir, "index.scip");
|
|
25
|
-
const args = buildScipArgs(language, binaryPath, projectRoot, outputPath);
|
|
26
|
-
if (options.extraArgs?.length) {
|
|
27
|
-
args.push(...options.extraArgs);
|
|
28
|
-
}
|
|
29
|
-
try {
|
|
30
|
-
log.info(`Running SCIP indexer: ${args[0]} for ${language}`);
|
|
31
|
-
const result = await exec(args[0], args.slice(1), {
|
|
32
|
-
cwd: projectRoot,
|
|
33
|
-
timeout: timeoutMs,
|
|
34
|
-
});
|
|
35
|
-
const durationMs = performance.now() - start;
|
|
36
|
-
if (result.exitCode !== 0) {
|
|
37
|
-
// Some SCIP indexers (e.g. scip-java via Maven) exit non-zero due to
|
|
38
|
-
// JVM deprecation warnings (sun.misc.Unsafe) even when indexing succeeds.
|
|
39
|
-
// If the output file was created despite the non-zero exit, treat as success.
|
|
40
|
-
if (existsSync(outputPath)) {
|
|
41
|
-
log.warn(`SCIP indexer exited with code ${result.exitCode} but output file was created — treating as success`);
|
|
42
|
-
return { success: true, outputPath, durationMs, error: null };
|
|
43
|
-
}
|
|
44
|
-
log.warn(`SCIP indexer failed (exit ${result.exitCode}): ${result.stderr.slice(0, 500)}`);
|
|
45
|
-
return {
|
|
46
|
-
success: false,
|
|
47
|
-
outputPath: null,
|
|
48
|
-
durationMs,
|
|
49
|
-
error: result.stderr.slice(0, 500),
|
|
50
|
-
};
|
|
51
|
-
}
|
|
52
|
-
if (!existsSync(outputPath)) {
|
|
53
|
-
return {
|
|
54
|
-
success: false,
|
|
55
|
-
outputPath: null,
|
|
56
|
-
durationMs,
|
|
57
|
-
error: "SCIP output file not created",
|
|
58
|
-
};
|
|
59
|
-
}
|
|
60
|
-
log.info(`SCIP indexer completed in ${Math.round(durationMs)}ms`);
|
|
61
|
-
return { success: true, outputPath, durationMs, error: null };
|
|
62
|
-
}
|
|
63
|
-
catch (err) {
|
|
64
|
-
const durationMs = performance.now() - start;
|
|
65
|
-
const message = err instanceof Error ? err.message : String(err);
|
|
66
|
-
log.warn(`SCIP indexer error: ${message}`);
|
|
67
|
-
return { success: false, outputPath: null, durationMs, error: message };
|
|
68
|
-
}
|
|
69
|
-
}
|
|
70
|
-
function buildScipArgs(language, binaryPath, projectRoot, outputPath) {
|
|
71
|
-
switch (language) {
|
|
72
|
-
case "typescript":
|
|
73
|
-
return [binaryPath, "index", "--output", outputPath];
|
|
74
|
-
case "python":
|
|
75
|
-
return [
|
|
76
|
-
binaryPath,
|
|
77
|
-
"index",
|
|
78
|
-
"--cwd",
|
|
79
|
-
projectRoot,
|
|
80
|
-
"--output",
|
|
81
|
-
outputPath,
|
|
82
|
-
];
|
|
83
|
-
case "go":
|
|
84
|
-
return [binaryPath, "-o", outputPath, "./..."];
|
|
85
|
-
case "java":
|
|
86
|
-
return [binaryPath, "index", "--output", outputPath];
|
|
87
|
-
case "rust":
|
|
88
|
-
return [binaryPath, "scip", projectRoot, "--output", outputPath];
|
|
89
|
-
case "ruby":
|
|
90
|
-
return [binaryPath, "--output", outputPath];
|
|
91
|
-
case "cpp":
|
|
92
|
-
return [binaryPath, "--output", outputPath];
|
|
93
|
-
case "csharp":
|
|
94
|
-
return [binaryPath, "index", "--output", outputPath];
|
|
95
|
-
default:
|
|
96
|
-
return [binaryPath, "--output", outputPath];
|
|
97
|
-
}
|
|
98
|
-
}
|
|
@@ -1,66 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Index Snapshot — save/load the full index state as msgpack for cold start.
|
|
3
|
-
*
|
|
4
|
-
* When the proxy restarts, instead of re-indexing the entire project,
|
|
5
|
-
* it loads the snapshot and only incrementally indexes changed files.
|
|
6
|
-
*
|
|
7
|
-
* Snapshot format: { entities, edges, metadata, version }
|
|
8
|
-
*/
|
|
9
|
-
import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
|
|
10
|
-
import { join } from "node:path";
|
|
11
|
-
import { createModuleLogger } from "../../utils/logger.js";
|
|
12
|
-
const log = createModuleLogger("index-snapshot");
|
|
13
|
-
const SNAPSHOT_VERSION = 1;
|
|
14
|
-
/**
|
|
15
|
-
* Save index state as a snapshot.
|
|
16
|
-
*/
|
|
17
|
-
export function saveSnapshot(unerrDir, entities, edges, metadata) {
|
|
18
|
-
const indexDir = join(unerrDir, "index");
|
|
19
|
-
if (!existsSync(indexDir)) {
|
|
20
|
-
mkdirSync(indexDir, { recursive: true });
|
|
21
|
-
}
|
|
22
|
-
const snapshot = {
|
|
23
|
-
version: SNAPSHOT_VERSION,
|
|
24
|
-
entities,
|
|
25
|
-
edges,
|
|
26
|
-
metadata,
|
|
27
|
-
savedAt: new Date().toISOString(),
|
|
28
|
-
};
|
|
29
|
-
const snapshotPath = join(indexDir, "snapshot.json");
|
|
30
|
-
try {
|
|
31
|
-
writeFileSync(snapshotPath, JSON.stringify(snapshot), "utf-8");
|
|
32
|
-
log.info(`Snapshot saved: ${entities.length} entities, ${edges.length} edges`);
|
|
33
|
-
}
|
|
34
|
-
catch (err) {
|
|
35
|
-
log.warn(`Failed to save snapshot: ${err instanceof Error ? err.message : String(err)}`);
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
/**
|
|
39
|
-
* Load index snapshot from disk.
|
|
40
|
-
* Returns null if no valid snapshot exists.
|
|
41
|
-
*/
|
|
42
|
-
export function loadSnapshot(unerrDir) {
|
|
43
|
-
const snapshotPath = join(unerrDir, "index", "snapshot.json");
|
|
44
|
-
if (!existsSync(snapshotPath))
|
|
45
|
-
return null;
|
|
46
|
-
try {
|
|
47
|
-
const raw = readFileSync(snapshotPath, "utf-8");
|
|
48
|
-
const parsed = JSON.parse(raw);
|
|
49
|
-
if (parsed.version !== SNAPSHOT_VERSION) {
|
|
50
|
-
log.info("Snapshot version mismatch — full re-index needed");
|
|
51
|
-
return null;
|
|
52
|
-
}
|
|
53
|
-
log.info(`Snapshot loaded: ${parsed.entities.length} entities, ${parsed.edges.length} edges (saved ${parsed.savedAt})`);
|
|
54
|
-
return parsed;
|
|
55
|
-
}
|
|
56
|
-
catch (err) {
|
|
57
|
-
log.warn(`Failed to load snapshot: ${err instanceof Error ? err.message : String(err)}`);
|
|
58
|
-
return null;
|
|
59
|
-
}
|
|
60
|
-
}
|
|
61
|
-
/**
|
|
62
|
-
* Check if a valid snapshot exists.
|
|
63
|
-
*/
|
|
64
|
-
export function hasSnapshot(unerrDir) {
|
|
65
|
-
return existsSync(join(unerrDir, "index", "snapshot.json"));
|
|
66
|
-
}
|