sverklo 0.1.0
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/LICENSE +21 -0
- package/README.md +133 -0
- package/dist/bin/sverklo.d.ts +2 -0
- package/dist/bin/sverklo.js +41 -0
- package/dist/bin/sverklo.js.map +1 -0
- package/dist/src/index.d.ts +3 -0
- package/dist/src/index.js +3 -0
- package/dist/src/index.js.map +1 -0
- package/dist/src/indexer/describer.d.ts +2 -0
- package/dist/src/indexer/describer.js +112 -0
- package/dist/src/indexer/describer.js.map +1 -0
- package/dist/src/indexer/embedder.d.ts +4 -0
- package/dist/src/indexer/embedder.js +249 -0
- package/dist/src/indexer/embedder.js.map +1 -0
- package/dist/src/indexer/file-discovery.d.ts +9 -0
- package/dist/src/indexer/file-discovery.js +52 -0
- package/dist/src/indexer/file-discovery.js.map +1 -0
- package/dist/src/indexer/graph-builder.d.ts +5 -0
- package/dist/src/indexer/graph-builder.js +77 -0
- package/dist/src/indexer/graph-builder.js.map +1 -0
- package/dist/src/indexer/indexer.d.ts +26 -0
- package/dist/src/indexer/indexer.js +180 -0
- package/dist/src/indexer/indexer.js.map +1 -0
- package/dist/src/indexer/parser.d.ts +2 -0
- package/dist/src/indexer/parser.js +467 -0
- package/dist/src/indexer/parser.js.map +1 -0
- package/dist/src/indexer/setup.d.ts +1 -0
- package/dist/src/indexer/setup.js +38 -0
- package/dist/src/indexer/setup.js.map +1 -0
- package/dist/src/indexer/watcher.d.ts +2 -0
- package/dist/src/indexer/watcher.js +51 -0
- package/dist/src/indexer/watcher.js.map +1 -0
- package/dist/src/memory/git-state.d.ts +4 -0
- package/dist/src/memory/git-state.js +12 -0
- package/dist/src/memory/git-state.js.map +1 -0
- package/dist/src/memory/staleness.d.ts +4 -0
- package/dist/src/memory/staleness.js +19 -0
- package/dist/src/memory/staleness.js.map +1 -0
- package/dist/src/search/hybrid-search.d.ts +13 -0
- package/dist/src/search/hybrid-search.js +97 -0
- package/dist/src/search/hybrid-search.js.map +1 -0
- package/dist/src/search/pagerank.d.ts +4 -0
- package/dist/src/search/pagerank.js +64 -0
- package/dist/src/search/pagerank.js.map +1 -0
- package/dist/src/search/token-budget.d.ts +10 -0
- package/dist/src/search/token-budget.js +67 -0
- package/dist/src/search/token-budget.js.map +1 -0
- package/dist/src/server/mcp-server.d.ts +1 -0
- package/dist/src/server/mcp-server.js +120 -0
- package/dist/src/server/mcp-server.js.map +1 -0
- package/dist/src/server/tools/dependencies.d.ts +29 -0
- package/dist/src/server/tools/dependencies.js +118 -0
- package/dist/src/server/tools/dependencies.js.map +1 -0
- package/dist/src/server/tools/find-references.d.ts +20 -0
- package/dist/src/server/tools/find-references.js +80 -0
- package/dist/src/server/tools/find-references.js.map +1 -0
- package/dist/src/server/tools/forget.d.ts +16 -0
- package/dist/src/server/tools/forget.js +24 -0
- package/dist/src/server/tools/forget.js.map +1 -0
- package/dist/src/server/tools/index-status.d.ts +10 -0
- package/dist/src/server/tools/index-status.js +21 -0
- package/dist/src/server/tools/index-status.js.map +1 -0
- package/dist/src/server/tools/lookup.d.ts +25 -0
- package/dist/src/server/tools/lookup.js +54 -0
- package/dist/src/server/tools/lookup.js.map +1 -0
- package/dist/src/server/tools/memories.d.ts +24 -0
- package/dist/src/server/tools/memories.js +61 -0
- package/dist/src/server/tools/memories.js.map +1 -0
- package/dist/src/server/tools/overview.d.ts +19 -0
- package/dist/src/server/tools/overview.js +32 -0
- package/dist/src/server/tools/overview.js.map +1 -0
- package/dist/src/server/tools/recall.d.ts +29 -0
- package/dist/src/server/tools/recall.js +113 -0
- package/dist/src/server/tools/recall.js.map +1 -0
- package/dist/src/server/tools/remember.d.ts +39 -0
- package/dist/src/server/tools/remember.js +56 -0
- package/dist/src/server/tools/remember.js.map +1 -0
- package/dist/src/server/tools/search.d.ts +33 -0
- package/dist/src/server/tools/search.js +43 -0
- package/dist/src/server/tools/search.js.map +1 -0
- package/dist/src/storage/chunk-store.d.ts +25 -0
- package/dist/src/storage/chunk-store.js +74 -0
- package/dist/src/storage/chunk-store.js.map +1 -0
- package/dist/src/storage/database.d.ts +2 -0
- package/dist/src/storage/database.js +140 -0
- package/dist/src/storage/database.js.map +1 -0
- package/dist/src/storage/embedding-store.d.ts +14 -0
- package/dist/src/storage/embedding-store.js +42 -0
- package/dist/src/storage/embedding-store.js.map +1 -0
- package/dist/src/storage/file-store.d.ts +19 -0
- package/dist/src/storage/file-store.js +44 -0
- package/dist/src/storage/file-store.js.map +1 -0
- package/dist/src/storage/graph-store.d.ts +16 -0
- package/dist/src/storage/graph-store.js +37 -0
- package/dist/src/storage/graph-store.js.map +1 -0
- package/dist/src/storage/memory-embedding-store.d.ts +13 -0
- package/dist/src/storage/memory-embedding-store.js +38 -0
- package/dist/src/storage/memory-embedding-store.js.map +1 -0
- package/dist/src/storage/memory-store.d.ts +29 -0
- package/dist/src/storage/memory-store.js +86 -0
- package/dist/src/storage/memory-store.js.map +1 -0
- package/dist/src/types/index.d.ts +87 -0
- package/dist/src/types/index.js +21 -0
- package/dist/src/types/index.js.map +1 -0
- package/dist/src/utils/config.d.ts +2 -0
- package/dist/src/utils/config.js +18 -0
- package/dist/src/utils/config.js.map +1 -0
- package/dist/src/utils/ignore.d.ts +2 -0
- package/dist/src/utils/ignore.js +72 -0
- package/dist/src/utils/ignore.js.map +1 -0
- package/dist/src/utils/logger.d.ts +2 -0
- package/dist/src/utils/logger.js +16 -0
- package/dist/src/utils/logger.js.map +1 -0
- package/dist/src/utils/tokens.d.ts +2 -0
- package/dist/src/utils/tokens.js +12 -0
- package/dist/src/utils/tokens.js.map +1 -0
- package/package.json +59 -0
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import { readdirSync, statSync } from "node:fs";
|
|
2
|
+
import { join, relative } from "node:path";
|
|
3
|
+
import { detectLanguage } from "../types/index.js";
|
|
4
|
+
const MAX_FILE_SIZE = 1_000_000; // 1MB - skip huge files
|
|
5
|
+
export function discoverFiles(rootPath, ignoreFilter) {
|
|
6
|
+
const files = [];
|
|
7
|
+
function walk(dir) {
|
|
8
|
+
let entries;
|
|
9
|
+
try {
|
|
10
|
+
entries = readdirSync(dir, { withFileTypes: true });
|
|
11
|
+
}
|
|
12
|
+
catch {
|
|
13
|
+
return;
|
|
14
|
+
}
|
|
15
|
+
for (const entry of entries) {
|
|
16
|
+
const absPath = join(dir, entry.name);
|
|
17
|
+
const relPath = relative(rootPath, absPath);
|
|
18
|
+
if (ignoreFilter.ignores(relPath))
|
|
19
|
+
continue;
|
|
20
|
+
if (entry.isDirectory()) {
|
|
21
|
+
// Also check directory with trailing slash
|
|
22
|
+
if (!ignoreFilter.ignores(relPath + "/")) {
|
|
23
|
+
walk(absPath);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
else if (entry.isFile()) {
|
|
27
|
+
const lang = detectLanguage(entry.name);
|
|
28
|
+
if (!lang)
|
|
29
|
+
continue;
|
|
30
|
+
let stat;
|
|
31
|
+
try {
|
|
32
|
+
stat = statSync(absPath);
|
|
33
|
+
}
|
|
34
|
+
catch {
|
|
35
|
+
continue;
|
|
36
|
+
}
|
|
37
|
+
if (stat.size > MAX_FILE_SIZE)
|
|
38
|
+
continue;
|
|
39
|
+
files.push({
|
|
40
|
+
absolutePath: absPath,
|
|
41
|
+
relativePath: relPath,
|
|
42
|
+
language: lang,
|
|
43
|
+
lastModified: stat.mtimeMs,
|
|
44
|
+
sizeBytes: stat.size,
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
walk(rootPath);
|
|
50
|
+
return files;
|
|
51
|
+
}
|
|
52
|
+
//# sourceMappingURL=file-discovery.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"file-discovery.js","sourceRoot":"","sources":["../../../src/indexer/file-discovery.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAChD,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AAE3C,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAUnD,MAAM,aAAa,GAAG,SAAS,CAAC,CAAC,wBAAwB;AAEzD,MAAM,UAAU,aAAa,CAC3B,QAAgB,EAChB,YAAoB;IAEpB,MAAM,KAAK,GAAqB,EAAE,CAAC;IAEnC,SAAS,IAAI,CAAC,GAAW;QACvB,IAAI,OAAO,CAAC;QACZ,IAAI,CAAC;YACH,OAAO,GAAG,WAAW,CAAC,GAAG,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;QACtD,CAAC;QAAC,MAAM,CAAC;YACP,OAAO;QACT,CAAC;QAED,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;YAC5B,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;YACtC,MAAM,OAAO,GAAG,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YAE5C,IAAI,YAAY,CAAC,OAAO,CAAC,OAAO,CAAC;gBAAE,SAAS;YAE5C,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC;gBACxB,2CAA2C;gBAC3C,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,OAAO,GAAG,GAAG,CAAC,EAAE,CAAC;oBACzC,IAAI,CAAC,OAAO,CAAC,CAAC;gBAChB,CAAC;YACH,CAAC;iBAAM,IAAI,KAAK,CAAC,MAAM,EAAE,EAAE,CAAC;gBAC1B,MAAM,IAAI,GAAG,cAAc,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;gBACxC,IAAI,CAAC,IAAI;oBAAE,SAAS;gBAEpB,IAAI,IAAI,CAAC;gBACT,IAAI,CAAC;oBACH,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAC3B,CAAC;gBAAC,MAAM,CAAC;oBACP,SAAS;gBACX,CAAC;gBAED,IAAI,IAAI,CAAC,IAAI,GAAG,aAAa;oBAAE,SAAS;gBAExC,KAAK,CAAC,IAAI,CAAC;oBACT,YAAY,EAAE,OAAO;oBACrB,YAAY,EAAE,OAAO;oBACrB,QAAQ,EAAE,IAAI;oBACd,YAAY,EAAE,IAAI,CAAC,OAAO;oBAC1B,SAAS,EAAE,IAAI,CAAC,IAAI;iBACrB,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;IAED,IAAI,CAAC,QAAQ,CAAC,CAAC;IACf,OAAO,KAAK,CAAC;AACf,CAAC"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import type { ImportRef } from "../types/index.js";
|
|
2
|
+
import type { FileStore } from "../storage/file-store.js";
|
|
3
|
+
import type { GraphStore } from "../storage/graph-store.js";
|
|
4
|
+
export declare function buildGraph(fileImports: Map<string, ImportRef[]>, // relativePath -> imports
|
|
5
|
+
fileStore: FileStore, graphStore: GraphStore, rootPath: string): void;
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
import { dirname, join } from "node:path";
|
|
2
|
+
import { computePageRank } from "../search/pagerank.js";
|
|
3
|
+
import { log } from "../utils/logger.js";
|
|
4
|
+
// File extension resolution for relative imports
|
|
5
|
+
const EXTENSIONS = [
|
|
6
|
+
"",
|
|
7
|
+
".ts",
|
|
8
|
+
".tsx",
|
|
9
|
+
".js",
|
|
10
|
+
".jsx",
|
|
11
|
+
".mts",
|
|
12
|
+
".cts",
|
|
13
|
+
".mjs",
|
|
14
|
+
".cjs",
|
|
15
|
+
"/index.ts",
|
|
16
|
+
"/index.tsx",
|
|
17
|
+
"/index.js",
|
|
18
|
+
"/index.jsx",
|
|
19
|
+
".py",
|
|
20
|
+
".go",
|
|
21
|
+
".rs",
|
|
22
|
+
".java",
|
|
23
|
+
".rb",
|
|
24
|
+
".php",
|
|
25
|
+
];
|
|
26
|
+
export function buildGraph(fileImports, // relativePath -> imports
|
|
27
|
+
fileStore, graphStore, rootPath) {
|
|
28
|
+
const allFiles = fileStore.getAll();
|
|
29
|
+
const pathToId = new Map();
|
|
30
|
+
for (const f of allFiles) {
|
|
31
|
+
pathToId.set(f.path, f.id);
|
|
32
|
+
}
|
|
33
|
+
// Build edges
|
|
34
|
+
const edges = [];
|
|
35
|
+
for (const [filePath, imports] of fileImports) {
|
|
36
|
+
const sourceId = pathToId.get(filePath);
|
|
37
|
+
if (!sourceId)
|
|
38
|
+
continue;
|
|
39
|
+
const fileDir = dirname(filePath);
|
|
40
|
+
for (const imp of imports) {
|
|
41
|
+
if (!imp.isRelative)
|
|
42
|
+
continue;
|
|
43
|
+
// Try to resolve the import to a file in the index
|
|
44
|
+
const resolved = resolveImport(imp.source, fileDir, pathToId);
|
|
45
|
+
if (resolved !== undefined) {
|
|
46
|
+
graphStore.upsert(sourceId, resolved, imp.names.length || 1);
|
|
47
|
+
edges.push({ source: sourceId, target: resolved });
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
// Compute PageRank
|
|
52
|
+
const fileIds = allFiles.map((f) => f.id);
|
|
53
|
+
const ranks = computePageRank(fileIds, edges);
|
|
54
|
+
// Write ranks back to file store
|
|
55
|
+
for (const [id, rank] of ranks) {
|
|
56
|
+
fileStore.updatePagerank(id, rank);
|
|
57
|
+
}
|
|
58
|
+
log(`Graph built: ${edges.length} edges, PageRank computed for ${fileIds.length} files`);
|
|
59
|
+
}
|
|
60
|
+
function resolveImport(importPath, fromDir, pathToId) {
|
|
61
|
+
const resolved = join(fromDir, importPath);
|
|
62
|
+
// Strip JS-family extensions so we can re-resolve to .ts, .tsx, etc.
|
|
63
|
+
const stripped = resolved.replace(/\.(m?jsx?|cjs)$/, "");
|
|
64
|
+
const bases = stripped !== resolved ? [stripped, resolved] : [resolved];
|
|
65
|
+
for (const base of bases) {
|
|
66
|
+
for (const ext of EXTENSIONS) {
|
|
67
|
+
const candidate = base + ext;
|
|
68
|
+
// Normalize path separators
|
|
69
|
+
const normalized = candidate.replace(/\\/g, "/");
|
|
70
|
+
const id = pathToId.get(normalized);
|
|
71
|
+
if (id !== undefined)
|
|
72
|
+
return id;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
return undefined;
|
|
76
|
+
}
|
|
77
|
+
//# sourceMappingURL=graph-builder.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"graph-builder.js","sourceRoot":"","sources":["../../../src/indexer/graph-builder.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { FileStore } from "../storage/file-store.js";
|
|
2
|
+
import { ChunkStore } from "../storage/chunk-store.js";
|
|
3
|
+
import { EmbeddingStore } from "../storage/embedding-store.js";
|
|
4
|
+
import { GraphStore } from "../storage/graph-store.js";
|
|
5
|
+
import { MemoryStore } from "../storage/memory-store.js";
|
|
6
|
+
import { MemoryEmbeddingStore } from "../storage/memory-embedding-store.js";
|
|
7
|
+
import type { ProjectConfig, IndexStatus } from "../types/index.js";
|
|
8
|
+
export declare class Indexer {
|
|
9
|
+
private config;
|
|
10
|
+
private db;
|
|
11
|
+
fileStore: FileStore;
|
|
12
|
+
chunkStore: ChunkStore;
|
|
13
|
+
embeddingStore: EmbeddingStore;
|
|
14
|
+
graphStore: GraphStore;
|
|
15
|
+
memoryStore: MemoryStore;
|
|
16
|
+
memoryEmbeddingStore: MemoryEmbeddingStore;
|
|
17
|
+
private indexing;
|
|
18
|
+
private progress;
|
|
19
|
+
constructor(config: ProjectConfig);
|
|
20
|
+
get rootPath(): string;
|
|
21
|
+
index(): Promise<void>;
|
|
22
|
+
reindexFile(relativePath: string, absolutePath: string, language: string): Promise<void>;
|
|
23
|
+
removeFile(relativePath: string): void;
|
|
24
|
+
getStatus(): IndexStatus;
|
|
25
|
+
close(): void;
|
|
26
|
+
}
|
|
@@ -0,0 +1,180 @@
|
|
|
1
|
+
import { readFileSync } from "node:fs";
|
|
2
|
+
import { createHash } from "node:crypto";
|
|
3
|
+
import { createDatabase } from "../storage/database.js";
|
|
4
|
+
import { FileStore } from "../storage/file-store.js";
|
|
5
|
+
import { ChunkStore } from "../storage/chunk-store.js";
|
|
6
|
+
import { EmbeddingStore } from "../storage/embedding-store.js";
|
|
7
|
+
import { GraphStore } from "../storage/graph-store.js";
|
|
8
|
+
import { MemoryStore } from "../storage/memory-store.js";
|
|
9
|
+
import { MemoryEmbeddingStore } from "../storage/memory-embedding-store.js";
|
|
10
|
+
import { discoverFiles } from "./file-discovery.js";
|
|
11
|
+
import { parseFile } from "./parser.js";
|
|
12
|
+
import { describeChunk } from "./describer.js";
|
|
13
|
+
import { embed, initEmbedder } from "./embedder.js";
|
|
14
|
+
import { buildGraph } from "./graph-builder.js";
|
|
15
|
+
import { createIgnoreFilter } from "../utils/ignore.js";
|
|
16
|
+
import { estimateTokens } from "../utils/tokens.js";
|
|
17
|
+
import { log, logError } from "../utils/logger.js";
|
|
18
|
+
export class Indexer {
|
|
19
|
+
config;
|
|
20
|
+
db;
|
|
21
|
+
fileStore;
|
|
22
|
+
chunkStore;
|
|
23
|
+
embeddingStore;
|
|
24
|
+
graphStore;
|
|
25
|
+
memoryStore;
|
|
26
|
+
memoryEmbeddingStore;
|
|
27
|
+
indexing = false;
|
|
28
|
+
progress = { done: 0, total: 0 };
|
|
29
|
+
constructor(config) {
|
|
30
|
+
this.config = config;
|
|
31
|
+
this.db = createDatabase(config.dbPath);
|
|
32
|
+
this.fileStore = new FileStore(this.db);
|
|
33
|
+
this.chunkStore = new ChunkStore(this.db);
|
|
34
|
+
this.embeddingStore = new EmbeddingStore(this.db);
|
|
35
|
+
this.graphStore = new GraphStore(this.db);
|
|
36
|
+
this.memoryStore = new MemoryStore(this.db);
|
|
37
|
+
this.memoryEmbeddingStore = new MemoryEmbeddingStore(this.db);
|
|
38
|
+
}
|
|
39
|
+
get rootPath() {
|
|
40
|
+
return this.config.rootPath;
|
|
41
|
+
}
|
|
42
|
+
async index() {
|
|
43
|
+
if (this.indexing)
|
|
44
|
+
return;
|
|
45
|
+
this.indexing = true;
|
|
46
|
+
try {
|
|
47
|
+
log(`Indexing ${this.config.rootPath}...`);
|
|
48
|
+
const startTime = Date.now();
|
|
49
|
+
await initEmbedder();
|
|
50
|
+
// 1. Discover files
|
|
51
|
+
const ignoreFilter = createIgnoreFilter(this.config.rootPath);
|
|
52
|
+
const files = discoverFiles(this.config.rootPath, ignoreFilter);
|
|
53
|
+
this.progress = { done: 0, total: files.length };
|
|
54
|
+
log(`Discovered ${files.length} files`);
|
|
55
|
+
// 2. Determine which files need (re)indexing
|
|
56
|
+
const toIndex = files.filter((f) => {
|
|
57
|
+
const existing = this.fileStore.getByPath(f.relativePath);
|
|
58
|
+
if (!existing)
|
|
59
|
+
return true;
|
|
60
|
+
const contentHash = hashFile(f.absolutePath);
|
|
61
|
+
return existing.hash !== contentHash;
|
|
62
|
+
});
|
|
63
|
+
// 3. Remove files that no longer exist
|
|
64
|
+
const currentPaths = new Set(files.map((f) => f.relativePath));
|
|
65
|
+
for (const existing of this.fileStore.getAll()) {
|
|
66
|
+
if (!currentPaths.has(existing.path)) {
|
|
67
|
+
this.fileStore.delete(existing.path);
|
|
68
|
+
log(`Removed deleted file: ${existing.path}`);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
if (toIndex.length === 0) {
|
|
72
|
+
log("Index is up to date");
|
|
73
|
+
this.indexing = false;
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
log(`Indexing ${toIndex.length} files (${files.length - toIndex.length} cached)`);
|
|
77
|
+
// 4. Parse, chunk, describe, embed
|
|
78
|
+
const fileImports = new Map();
|
|
79
|
+
const BATCH_SIZE = 32;
|
|
80
|
+
const embeddingBatch = [];
|
|
81
|
+
// Use a transaction for bulk inserts
|
|
82
|
+
const transaction = this.db.transaction(() => {
|
|
83
|
+
for (const file of toIndex) {
|
|
84
|
+
try {
|
|
85
|
+
const content = readFileSync(file.absolutePath, "utf-8");
|
|
86
|
+
const contentHash = hashFile(file.absolutePath);
|
|
87
|
+
// Upsert file record
|
|
88
|
+
const fileId = this.fileStore.upsert(file.relativePath, file.language, contentHash, file.lastModified, file.sizeBytes);
|
|
89
|
+
// Clear old chunks for this file
|
|
90
|
+
this.chunkStore.deleteByFile(fileId);
|
|
91
|
+
// Parse
|
|
92
|
+
const result = parseFile(content, file.language);
|
|
93
|
+
fileImports.set(file.relativePath, result.imports);
|
|
94
|
+
// Store chunks
|
|
95
|
+
for (const chunk of result.chunks) {
|
|
96
|
+
const description = describeChunk(chunk, file.relativePath, file.language);
|
|
97
|
+
const tokenCount = estimateTokens(chunk.content);
|
|
98
|
+
const chunkId = this.chunkStore.insert(fileId, chunk.type, chunk.name, chunk.signature, chunk.startLine, chunk.endLine, chunk.content, description, tokenCount);
|
|
99
|
+
// Queue for embedding
|
|
100
|
+
const embText = description + "\n" + chunk.content.slice(0, 512);
|
|
101
|
+
embeddingBatch.push({ chunkId, text: embText });
|
|
102
|
+
}
|
|
103
|
+
this.progress.done++;
|
|
104
|
+
}
|
|
105
|
+
catch (err) {
|
|
106
|
+
logError(`Failed to index ${file.relativePath}`, err);
|
|
107
|
+
this.progress.done++;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
});
|
|
111
|
+
transaction();
|
|
112
|
+
// 5. Generate embeddings in batches
|
|
113
|
+
log(`Generating embeddings for ${embeddingBatch.length} chunks...`);
|
|
114
|
+
for (let i = 0; i < embeddingBatch.length; i += BATCH_SIZE) {
|
|
115
|
+
const batch = embeddingBatch.slice(i, i + BATCH_SIZE);
|
|
116
|
+
const texts = batch.map((b) => b.text);
|
|
117
|
+
const vectors = await embed(texts);
|
|
118
|
+
for (let j = 0; j < batch.length; j++) {
|
|
119
|
+
this.embeddingStore.insert(batch[j].chunkId, vectors[j]);
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
// 6. Rebuild FTS index (ensures sync with content table)
|
|
123
|
+
this.db.exec("INSERT INTO chunks_fts(chunks_fts) VALUES('rebuild')");
|
|
124
|
+
// 7. Build dependency graph and compute PageRank
|
|
125
|
+
log("Building dependency graph...");
|
|
126
|
+
buildGraph(fileImports, this.fileStore, this.graphStore, this.config.rootPath);
|
|
127
|
+
// 8. Update project metadata
|
|
128
|
+
const elapsed = Date.now() - startTime;
|
|
129
|
+
log(`Indexing complete: ${this.fileStore.count()} files, ` +
|
|
130
|
+
`${this.chunkStore.count()} chunks in ${elapsed}ms`);
|
|
131
|
+
}
|
|
132
|
+
finally {
|
|
133
|
+
this.indexing = false;
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
async reindexFile(relativePath, absolutePath, language) {
|
|
137
|
+
try {
|
|
138
|
+
const content = readFileSync(absolutePath, "utf-8");
|
|
139
|
+
const contentHash = hashFile(absolutePath);
|
|
140
|
+
const stat = require("node:fs").statSync(absolutePath);
|
|
141
|
+
const fileId = this.fileStore.upsert(relativePath, language, contentHash, stat.mtimeMs, stat.size);
|
|
142
|
+
this.chunkStore.deleteByFile(fileId);
|
|
143
|
+
const result = parseFile(content, language);
|
|
144
|
+
for (const chunk of result.chunks) {
|
|
145
|
+
const description = describeChunk(chunk, relativePath, language);
|
|
146
|
+
const tokenCount = estimateTokens(chunk.content);
|
|
147
|
+
const chunkId = this.chunkStore.insert(fileId, chunk.type, chunk.name, chunk.signature, chunk.startLine, chunk.endLine, chunk.content, description, tokenCount);
|
|
148
|
+
const embText = description + "\n" + chunk.content.slice(0, 512);
|
|
149
|
+
const [vector] = await embed([embText]);
|
|
150
|
+
this.embeddingStore.insert(chunkId, vector);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
catch (err) {
|
|
154
|
+
logError(`Failed to reindex ${relativePath}`, err);
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
removeFile(relativePath) {
|
|
158
|
+
this.fileStore.delete(relativePath);
|
|
159
|
+
}
|
|
160
|
+
getStatus() {
|
|
161
|
+
return {
|
|
162
|
+
projectName: this.config.name,
|
|
163
|
+
rootPath: this.config.rootPath,
|
|
164
|
+
fileCount: this.fileStore.count(),
|
|
165
|
+
chunkCount: this.chunkStore.count(),
|
|
166
|
+
languages: this.fileStore.getLanguages(),
|
|
167
|
+
lastIndexedAt: Date.now(),
|
|
168
|
+
indexing: this.indexing,
|
|
169
|
+
progress: this.indexing ? this.progress : undefined,
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
close() {
|
|
173
|
+
this.db.close();
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
function hashFile(filePath) {
|
|
177
|
+
const content = readFileSync(filePath, "utf-8");
|
|
178
|
+
return createHash("sha256").update(content).digest("hex").slice(0, 16);
|
|
179
|
+
}
|
|
180
|
+
//# sourceMappingURL=indexer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"indexer.js","sourceRoot":"","sources":["../../../src/indexer/indexer.ts"],"names":[],"mappings":"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"}
|