forked from ultraworkers/claw-code
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathindex.ts
More file actions
370 lines (330 loc) · 11.7 KB
/
Copy pathindex.ts
File metadata and controls
370 lines (330 loc) · 11.7 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
/**
* Pure-TypeScript port of vendor/file-index-src (Rust NAPI module).
*
* The native module wraps nucleo (https://github.com/helix-editor/nucleo) for
* high-performance fuzzy file searching. This port reimplements the same API
* and scoring behavior without native dependencies.
*
* Key API:
* new FileIndex()
* .loadFromFileList(fileList: string[]): void — dedupe + index paths
* .search(query: string, limit: number): SearchResult[]
*
* Score semantics: lower = better. Score is position-in-results / result-count,
* so the best match is 0.0. Paths containing "test" get a 1.05× penalty (capped
* at 1.0) so non-test files rank slightly higher.
*/
export type SearchResult = {
path: string
score: number
}
// nucleo-style scoring constants (approximating fzf-v2 / nucleo bonuses)
const SCORE_MATCH = 16
const BONUS_BOUNDARY = 8
const BONUS_CAMEL = 6
const BONUS_CONSECUTIVE = 4
const BONUS_FIRST_CHAR = 8
const PENALTY_GAP_START = 3
const PENALTY_GAP_EXTENSION = 1
const TOP_LEVEL_CACHE_LIMIT = 100
const MAX_QUERY_LEN = 64
// Yield to event loop after this many ms of sync work. Chunk sizes are
// time-based (not count-based) so slow machines get smaller chunks and
// stay responsive — 5k paths is ~2ms on M-series but could be 15ms+ on
// older Windows hardware.
const CHUNK_MS = 4
// Reusable buffer: records where each needle char matched during the indexOf scan
const posBuf = new Int32Array(MAX_QUERY_LEN)
export class FileIndex {
private paths: string[] = []
private lowerPaths: string[] = []
private charBits: Int32Array = new Int32Array(0)
private pathLens: Uint16Array = new Uint16Array(0)
private topLevelCache: SearchResult[] | null = null
// During async build, tracks how many paths have bitmap/lowerPath filled.
// search() uses this to search the ready prefix while build continues.
private readyCount = 0
/**
* Load paths from an array of strings.
* This is the main way to populate the index — ripgrep collects files, we just search them.
* Automatically deduplicates paths.
*/
loadFromFileList(fileList: string[]): void {
// Deduplicate and filter empty strings (matches Rust HashSet behavior)
const seen = new Set<string>()
const paths: string[] = []
for (const line of fileList) {
if (line.length > 0 && !seen.has(line)) {
seen.add(line)
paths.push(line)
}
}
this.buildIndex(paths)
}
/**
* Async variant: yields to the event loop every ~8–12k paths so large
* indexes (270k+ files) don't block the main thread for >10ms at a time.
* Identical result to loadFromFileList.
*
* Returns { queryable, done }:
* - queryable: resolves as soon as the first chunk is indexed (search
* returns partial results). For a 270k-path list this is ~5–10ms of
* sync work after the paths array is available.
* - done: resolves when the entire index is built.
*/
loadFromFileListAsync(fileList: string[]): {
queryable: Promise<void>
done: Promise<void>
} {
let markQueryable: () => void = () => {}
const queryable = new Promise<void>(resolve => {
markQueryable = resolve
})
const done = this.buildAsync(fileList, markQueryable)
return { queryable, done }
}
private async buildAsync(
fileList: string[],
markQueryable: () => void,
): Promise<void> {
const seen = new Set<string>()
const paths: string[] = []
let chunkStart = performance.now()
for (let i = 0; i < fileList.length; i++) {
const line = fileList[i]!
if (line.length > 0 && !seen.has(line)) {
seen.add(line)
paths.push(line)
}
// Check every 256 iterations to amortize performance.now() overhead
if ((i & 0xff) === 0xff && performance.now() - chunkStart > CHUNK_MS) {
await yieldToEventLoop()
chunkStart = performance.now()
}
}
this.resetArrays(paths)
chunkStart = performance.now()
let firstChunk = true
for (let i = 0; i < paths.length; i++) {
this.indexPath(i)
if ((i & 0xff) === 0xff && performance.now() - chunkStart > CHUNK_MS) {
this.readyCount = i + 1
if (firstChunk) {
markQueryable()
firstChunk = false
}
await yieldToEventLoop()
chunkStart = performance.now()
}
}
this.readyCount = paths.length
markQueryable()
}
private buildIndex(paths: string[]): void {
this.resetArrays(paths)
for (let i = 0; i < paths.length; i++) {
this.indexPath(i)
}
this.readyCount = paths.length
}
private resetArrays(paths: string[]): void {
const n = paths.length
this.paths = paths
this.lowerPaths = new Array(n)
this.charBits = new Int32Array(n)
this.pathLens = new Uint16Array(n)
this.readyCount = 0
this.topLevelCache = computeTopLevelEntries(paths, TOP_LEVEL_CACHE_LIMIT)
}
// Precompute: lowercase, a–z bitmap, length. Bitmap gives O(1) rejection
// of paths missing any needle letter (89% survival for broad queries like
// "test" → still a 10%+ free win; 90%+ rejection for rare chars).
private indexPath(i: number): void {
const lp = this.paths[i]!.toLowerCase()
this.lowerPaths[i] = lp
const len = lp.length
this.pathLens[i] = len
let bits = 0
for (let j = 0; j < len; j++) {
const c = lp.charCodeAt(j)
if (c >= 97 && c <= 122) bits |= 1 << (c - 97)
}
this.charBits[i] = bits
}
/**
* Search for files matching the query using fuzzy matching.
* Returns top N results sorted by match score.
*/
search(query: string, limit: number): SearchResult[] {
if (limit <= 0) return []
if (query.length === 0) {
if (this.topLevelCache) {
return this.topLevelCache.slice(0, limit)
}
return []
}
// Smart case: lowercase query → case-insensitive; any uppercase → case-sensitive
const caseSensitive = query !== query.toLowerCase()
const needle = caseSensitive ? query : query.toLowerCase()
const nLen = Math.min(needle.length, MAX_QUERY_LEN)
const needleChars: string[] = new Array(nLen)
let needleBitmap = 0
for (let j = 0; j < nLen; j++) {
const ch = needle.charAt(j)
needleChars[j] = ch
const cc = ch.charCodeAt(0)
if (cc >= 97 && cc <= 122) needleBitmap |= 1 << (cc - 97)
}
// Upper bound on score assuming every match gets the max boundary bonus.
// Used to reject paths whose gap penalties alone make them unable to beat
// the current top-k threshold, before the charCodeAt-heavy boundary pass.
const scoreCeiling =
nLen * (SCORE_MATCH + BONUS_BOUNDARY) + BONUS_FIRST_CHAR + 32
// Top-k: maintain a sorted-ascending array of the best `limit` matches.
// Avoids O(n log n) sort of all matches when we only need `limit` of them.
const topK: { path: string; fuzzScore: number }[] = []
let threshold = -Infinity
const { paths, lowerPaths, charBits, pathLens, readyCount } = this
outer: for (let i = 0; i < readyCount; i++) {
// O(1) bitmap reject: path must contain every letter in the needle
if ((charBits[i]! & needleBitmap) !== needleBitmap) continue
const haystack = caseSensitive ? paths[i]! : lowerPaths[i]!
// Fused indexOf scan: find positions (SIMD-accelerated in JSC/V8) AND
// accumulate gap/consecutive terms inline. The greedy-earliest positions
// found here are identical to what the charCodeAt scorer would find, so
// we score directly from them — no second scan.
let pos = haystack.indexOf(needleChars[0]!)
if (pos === -1) continue
posBuf[0] = pos
let gapPenalty = 0
let consecBonus = 0
let prev = pos
for (let j = 1; j < nLen; j++) {
pos = haystack.indexOf(needleChars[j]!, prev + 1)
if (pos === -1) continue outer
posBuf[j] = pos
const gap = pos - prev - 1
if (gap === 0) consecBonus += BONUS_CONSECUTIVE
else gapPenalty += PENALTY_GAP_START + gap * PENALTY_GAP_EXTENSION
prev = pos
}
// Gap-bound reject: if the best-case score (all boundary bonuses) minus
// known gap penalties can't beat threshold, skip the boundary pass.
if (
topK.length === limit &&
scoreCeiling + consecBonus - gapPenalty <= threshold
) {
continue
}
// Boundary/camelCase scoring: check the char before each match position.
const path = paths[i]!
const hLen = pathLens[i]!
let score = nLen * SCORE_MATCH + consecBonus - gapPenalty
score += scoreBonusAt(path, posBuf[0]!, true)
for (let j = 1; j < nLen; j++) {
score += scoreBonusAt(path, posBuf[j]!, false)
}
score += Math.max(0, 32 - (hLen >> 2))
if (topK.length < limit) {
topK.push({ path, fuzzScore: score })
if (topK.length === limit) {
topK.sort((a, b) => a.fuzzScore - b.fuzzScore)
threshold = topK[0]!.fuzzScore
}
} else if (score > threshold) {
let lo = 0
let hi = topK.length
while (lo < hi) {
const mid = (lo + hi) >> 1
if (topK[mid]!.fuzzScore < score) lo = mid + 1
else hi = mid
}
topK.splice(lo, 0, { path, fuzzScore: score })
topK.shift()
threshold = topK[0]!.fuzzScore
}
}
// topK is ascending; reverse to descending (best first)
topK.sort((a, b) => b.fuzzScore - a.fuzzScore)
const matchCount = topK.length
const denom = Math.max(matchCount, 1)
const results: SearchResult[] = new Array(matchCount)
for (let i = 0; i < matchCount; i++) {
const path = topK[i]!.path
const positionScore = i / denom
const finalScore = path.includes('test')
? Math.min(positionScore * 1.05, 1.0)
: positionScore
results[i] = { path, score: finalScore }
}
return results
}
}
/**
* Boundary/camelCase bonus for a match at position `pos` in the original-case
* path. `first` enables the start-of-string bonus (only for needle[0]).
*/
function scoreBonusAt(path: string, pos: number, first: boolean): number {
if (pos === 0) return first ? BONUS_FIRST_CHAR : 0
const prevCh = path.charCodeAt(pos - 1)
if (isBoundary(prevCh)) return BONUS_BOUNDARY
if (isLower(prevCh) && isUpper(path.charCodeAt(pos))) return BONUS_CAMEL
return 0
}
function isBoundary(code: number): boolean {
// / \ - _ . space
return (
code === 47 || // /
code === 92 || // \
code === 45 || // -
code === 95 || // _
code === 46 || // .
code === 32 // space
)
}
function isLower(code: number): boolean {
return code >= 97 && code <= 122
}
function isUpper(code: number): boolean {
return code >= 65 && code <= 90
}
export function yieldToEventLoop(): Promise<void> {
return new Promise(resolve => setImmediate(resolve))
}
export { CHUNK_MS }
/**
* Extract unique top-level path segments, sorted by (length asc, then alpha asc).
* Handles both Unix (/) and Windows (\) path separators.
* Mirrors FileIndex::compute_top_level_entries in lib.rs.
*/
function computeTopLevelEntries(
paths: string[],
limit: number,
): SearchResult[] {
const topLevel = new Set<string>()
for (const p of paths) {
// Split on first / or \ separator
let end = p.length
for (let i = 0; i < p.length; i++) {
const c = p.charCodeAt(i)
if (c === 47 || c === 92) {
end = i
break
}
}
const segment = p.slice(0, end)
if (segment.length > 0) {
topLevel.add(segment)
if (topLevel.size >= limit) break
}
}
const sorted = Array.from(topLevel)
sorted.sort((a, b) => {
const lenDiff = a.length - b.length
if (lenDiff !== 0) return lenDiff
return a < b ? -1 : a > b ? 1 : 0
})
return sorted.slice(0, limit).map(path => ({ path, score: 0.0 }))
}
export default FileIndex
export type { FileIndex as FileIndexType }