xgrep 0.0.1 → 0.1.6

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 CHANGED
@@ -1,5 +1,119 @@
1
1
  # xgrep
2
2
 
3
- Ultra-fast indexed code search engine. Node.js bindings coming soon.
3
+ Ultra-fast indexed code search engine for Node.js. Powered by [trigram inverted index](https://swtch.com/~rsc/regexp/regexp4.html) in Rust via [napi-rs](https://napi.rs/).
4
4
 
5
- See https://github.com/momokun7/xgrep
5
+ 27-59x faster than ripgrep on repeated searches.
6
+
7
+ ## Install
8
+
9
+ ```bash
10
+ npm install xgrep
11
+ ```
12
+
13
+ Pre-built binaries are available for:
14
+ - Linux (x86_64, ARM64)
15
+ - macOS (x86_64, ARM64)
16
+ - Windows (x86_64)
17
+
18
+ ## Usage
19
+
20
+ ```typescript
21
+ import { Xgrep } from 'xgrep';
22
+
23
+ // Open a directory and build the index
24
+ const xg = Xgrep.open('/path/to/your/repo');
25
+ xg.buildIndex();
26
+
27
+ // Search for a pattern
28
+ const results = xg.search('fn main');
29
+ for (const r of results) {
30
+ console.log(`${r.file}:${r.lineNumber}: ${r.line}`);
31
+ }
32
+
33
+ // Search with options
34
+ const filtered = xg.search('TODO', {
35
+ fileType: 'ts', // Filter by file type
36
+ caseInsensitive: true, // Case-insensitive search
37
+ maxCount: 10, // Limit results
38
+ pathPattern: 'src/', // Filter by path
39
+ });
40
+
41
+ // Regex search
42
+ const regexResults = xg.search('fn\\s+\\w+', { regex: true });
43
+
44
+ // Index status
45
+ console.log(xg.indexStatus());
46
+ ```
47
+
48
+ ## API
49
+
50
+ ### `Xgrep.open(root: string): Xgrep`
51
+
52
+ Open a directory for searching. Index location is auto-resolved to `~/.cache/xgrep/`.
53
+
54
+ ### `Xgrep.openLocal(root: string): Xgrep`
55
+
56
+ Open with local index storage (`.xgrep/` in the project root).
57
+
58
+ ### `xg.buildIndex(): void`
59
+
60
+ Build or rebuild the search index.
61
+
62
+ ### `xg.search(pattern: string, opts?: SearchOptions): SearchResult[]`
63
+
64
+ Search for a pattern in the indexed codebase.
65
+
66
+ **SearchOptions:**
67
+
68
+ | Option | Type | Default | Description |
69
+ |--------|------|---------|-------------|
70
+ | `caseInsensitive` | `boolean` | `false` | Case-insensitive search (ASCII-only) |
71
+ | `regex` | `boolean` | `false` | Treat pattern as regex |
72
+ | `fileType` | `string` | - | Filter by file type (e.g., `"rs"`, `"py"`, `"js"`) |
73
+ | `maxCount` | `number` | - | Maximum number of results |
74
+ | `changedOnly` | `boolean` | `false` | Only search git-changed files |
75
+ | `since` | `string` | - | Files changed within duration (`"1h"`, `"2d"`, `"3.commits"`) |
76
+ | `pathPattern` | `string` | - | Filter by path substring |
77
+ | `fresh` | `boolean` | `false` | Check index freshness before searching |
78
+
79
+ **SearchResult:**
80
+
81
+ | Field | Type | Description |
82
+ |-------|------|-------------|
83
+ | `file` | `string` | File path relative to root |
84
+ | `lineNumber` | `number` | Line number (1-based) |
85
+ | `line` | `string` | The matching line content |
86
+
87
+ ### `xg.indexStatus(): string`
88
+
89
+ Get the current index status.
90
+
91
+ ### `xg.root: string` (getter)
92
+
93
+ Get the root directory path.
94
+
95
+ ### `xg.indexPath: string` (getter)
96
+
97
+ Get the index file path.
98
+
99
+ ## How It Works
100
+
101
+ xgrep builds a trigram inverted index of your codebase. On the first search, the index is built automatically. Subsequent searches use the index to narrow down candidate files before scanning, making repeated searches extremely fast.
102
+
103
+ | Benchmark | xgrep | ripgrep | Speedup |
104
+ |-----------|-------|---------|---------|
105
+ | Linux kernel (92K files) | 38ms | 2,236ms | **59x** |
106
+ | ripgrep source (248 files) | 2.5ms | 7.9ms | **3.1x** |
107
+
108
+ ## CLI
109
+
110
+ xgrep also provides a CLI tool. Install via Rust:
111
+
112
+ ```bash
113
+ cargo install xgrep-search
114
+ xg "pattern" --type rs
115
+ ```
116
+
117
+ ## License
118
+
119
+ MIT
package/index.d.ts ADDED
@@ -0,0 +1,50 @@
1
+ /* tslint:disable */
2
+ /* eslint-disable */
3
+
4
+ /* auto-generated by NAPI-RS */
5
+
6
+ /** Search result returned from xgrep. */
7
+ export interface SearchResult {
8
+ /** File path relative to root. */
9
+ file: string
10
+ /** Line number (1-based). */
11
+ lineNumber: number
12
+ /** The matching line content. */
13
+ line: string
14
+ }
15
+ /** Search options for xgrep. */
16
+ export interface SearchOptions {
17
+ /** Case-insensitive search (ASCII-only). */
18
+ caseInsensitive?: boolean
19
+ /** Treat pattern as regex. */
20
+ regex?: boolean
21
+ /** Filter by file type (e.g., "rs", "py", "js"). */
22
+ fileType?: string
23
+ /** Maximum number of results. */
24
+ maxCount?: number
25
+ /** Only search git-changed files. */
26
+ changedOnly?: boolean
27
+ /** Search files changed within duration (e.g., "1h", "2d"). */
28
+ since?: string
29
+ /** Filter by path substring. */
30
+ pathPattern?: string
31
+ /** Check index freshness before searching. */
32
+ fresh?: boolean
33
+ }
34
+ /** Ultra-fast indexed code search engine. */
35
+ export declare class Xgrep {
36
+ /** Open a directory for searching. Index location is auto-resolved. */
37
+ static open(root: string): Xgrep
38
+ /** Open a directory with local index storage (.xgrep/ in the project root). */
39
+ static openLocal(root: string): Xgrep
40
+ /** Build or rebuild the search index. */
41
+ buildIndex(): void
42
+ /** Search for a pattern in the indexed codebase. */
43
+ search(pattern: string, opts?: SearchOptions | undefined | null): Array<SearchResult>
44
+ /** Get the current index status. */
45
+ indexStatus(): string
46
+ /** Get the root directory path. */
47
+ get root(): string
48
+ /** Get the index file path. */
49
+ get indexPath(): string
50
+ }
package/index.js CHANGED
@@ -1 +1,315 @@
1
- module.exports = {};
1
+ /* tslint:disable */
2
+ /* eslint-disable */
3
+ /* prettier-ignore */
4
+
5
+ /* auto-generated by NAPI-RS */
6
+
7
+ const { existsSync, readFileSync } = require('fs')
8
+ const { join } = require('path')
9
+
10
+ const { platform, arch } = process
11
+
12
+ let nativeBinding = null
13
+ let localFileExisted = false
14
+ let loadError = null
15
+
16
+ function isMusl() {
17
+ // For Node 10
18
+ if (!process.report || typeof process.report.getReport !== 'function') {
19
+ try {
20
+ const lddPath = require('child_process').execSync('which ldd').toString().trim()
21
+ return readFileSync(lddPath, 'utf8').includes('musl')
22
+ } catch (e) {
23
+ return true
24
+ }
25
+ } else {
26
+ const { glibcVersionRuntime } = process.report.getReport().header
27
+ return !glibcVersionRuntime
28
+ }
29
+ }
30
+
31
+ switch (platform) {
32
+ case 'android':
33
+ switch (arch) {
34
+ case 'arm64':
35
+ localFileExisted = existsSync(join(__dirname, 'xgrep.android-arm64.node'))
36
+ try {
37
+ if (localFileExisted) {
38
+ nativeBinding = require('./xgrep.android-arm64.node')
39
+ } else {
40
+ nativeBinding = require('xgrep-android-arm64')
41
+ }
42
+ } catch (e) {
43
+ loadError = e
44
+ }
45
+ break
46
+ case 'arm':
47
+ localFileExisted = existsSync(join(__dirname, 'xgrep.android-arm-eabi.node'))
48
+ try {
49
+ if (localFileExisted) {
50
+ nativeBinding = require('./xgrep.android-arm-eabi.node')
51
+ } else {
52
+ nativeBinding = require('xgrep-android-arm-eabi')
53
+ }
54
+ } catch (e) {
55
+ loadError = e
56
+ }
57
+ break
58
+ default:
59
+ throw new Error(`Unsupported architecture on Android ${arch}`)
60
+ }
61
+ break
62
+ case 'win32':
63
+ switch (arch) {
64
+ case 'x64':
65
+ localFileExisted = existsSync(
66
+ join(__dirname, 'xgrep.win32-x64-msvc.node')
67
+ )
68
+ try {
69
+ if (localFileExisted) {
70
+ nativeBinding = require('./xgrep.win32-x64-msvc.node')
71
+ } else {
72
+ nativeBinding = require('xgrep-win32-x64-msvc')
73
+ }
74
+ } catch (e) {
75
+ loadError = e
76
+ }
77
+ break
78
+ case 'ia32':
79
+ localFileExisted = existsSync(
80
+ join(__dirname, 'xgrep.win32-ia32-msvc.node')
81
+ )
82
+ try {
83
+ if (localFileExisted) {
84
+ nativeBinding = require('./xgrep.win32-ia32-msvc.node')
85
+ } else {
86
+ nativeBinding = require('xgrep-win32-ia32-msvc')
87
+ }
88
+ } catch (e) {
89
+ loadError = e
90
+ }
91
+ break
92
+ case 'arm64':
93
+ localFileExisted = existsSync(
94
+ join(__dirname, 'xgrep.win32-arm64-msvc.node')
95
+ )
96
+ try {
97
+ if (localFileExisted) {
98
+ nativeBinding = require('./xgrep.win32-arm64-msvc.node')
99
+ } else {
100
+ nativeBinding = require('xgrep-win32-arm64-msvc')
101
+ }
102
+ } catch (e) {
103
+ loadError = e
104
+ }
105
+ break
106
+ default:
107
+ throw new Error(`Unsupported architecture on Windows: ${arch}`)
108
+ }
109
+ break
110
+ case 'darwin':
111
+ localFileExisted = existsSync(join(__dirname, 'xgrep.darwin-universal.node'))
112
+ try {
113
+ if (localFileExisted) {
114
+ nativeBinding = require('./xgrep.darwin-universal.node')
115
+ } else {
116
+ nativeBinding = require('xgrep-darwin-universal')
117
+ }
118
+ break
119
+ } catch {}
120
+ switch (arch) {
121
+ case 'x64':
122
+ localFileExisted = existsSync(join(__dirname, 'xgrep.darwin-x64.node'))
123
+ try {
124
+ if (localFileExisted) {
125
+ nativeBinding = require('./xgrep.darwin-x64.node')
126
+ } else {
127
+ nativeBinding = require('xgrep-darwin-x64')
128
+ }
129
+ } catch (e) {
130
+ loadError = e
131
+ }
132
+ break
133
+ case 'arm64':
134
+ localFileExisted = existsSync(
135
+ join(__dirname, 'xgrep.darwin-arm64.node')
136
+ )
137
+ try {
138
+ if (localFileExisted) {
139
+ nativeBinding = require('./xgrep.darwin-arm64.node')
140
+ } else {
141
+ nativeBinding = require('xgrep-darwin-arm64')
142
+ }
143
+ } catch (e) {
144
+ loadError = e
145
+ }
146
+ break
147
+ default:
148
+ throw new Error(`Unsupported architecture on macOS: ${arch}`)
149
+ }
150
+ break
151
+ case 'freebsd':
152
+ if (arch !== 'x64') {
153
+ throw new Error(`Unsupported architecture on FreeBSD: ${arch}`)
154
+ }
155
+ localFileExisted = existsSync(join(__dirname, 'xgrep.freebsd-x64.node'))
156
+ try {
157
+ if (localFileExisted) {
158
+ nativeBinding = require('./xgrep.freebsd-x64.node')
159
+ } else {
160
+ nativeBinding = require('xgrep-freebsd-x64')
161
+ }
162
+ } catch (e) {
163
+ loadError = e
164
+ }
165
+ break
166
+ case 'linux':
167
+ switch (arch) {
168
+ case 'x64':
169
+ if (isMusl()) {
170
+ localFileExisted = existsSync(
171
+ join(__dirname, 'xgrep.linux-x64-musl.node')
172
+ )
173
+ try {
174
+ if (localFileExisted) {
175
+ nativeBinding = require('./xgrep.linux-x64-musl.node')
176
+ } else {
177
+ nativeBinding = require('xgrep-linux-x64-musl')
178
+ }
179
+ } catch (e) {
180
+ loadError = e
181
+ }
182
+ } else {
183
+ localFileExisted = existsSync(
184
+ join(__dirname, 'xgrep.linux-x64-gnu.node')
185
+ )
186
+ try {
187
+ if (localFileExisted) {
188
+ nativeBinding = require('./xgrep.linux-x64-gnu.node')
189
+ } else {
190
+ nativeBinding = require('xgrep-linux-x64-gnu')
191
+ }
192
+ } catch (e) {
193
+ loadError = e
194
+ }
195
+ }
196
+ break
197
+ case 'arm64':
198
+ if (isMusl()) {
199
+ localFileExisted = existsSync(
200
+ join(__dirname, 'xgrep.linux-arm64-musl.node')
201
+ )
202
+ try {
203
+ if (localFileExisted) {
204
+ nativeBinding = require('./xgrep.linux-arm64-musl.node')
205
+ } else {
206
+ nativeBinding = require('xgrep-linux-arm64-musl')
207
+ }
208
+ } catch (e) {
209
+ loadError = e
210
+ }
211
+ } else {
212
+ localFileExisted = existsSync(
213
+ join(__dirname, 'xgrep.linux-arm64-gnu.node')
214
+ )
215
+ try {
216
+ if (localFileExisted) {
217
+ nativeBinding = require('./xgrep.linux-arm64-gnu.node')
218
+ } else {
219
+ nativeBinding = require('xgrep-linux-arm64-gnu')
220
+ }
221
+ } catch (e) {
222
+ loadError = e
223
+ }
224
+ }
225
+ break
226
+ case 'arm':
227
+ if (isMusl()) {
228
+ localFileExisted = existsSync(
229
+ join(__dirname, 'xgrep.linux-arm-musleabihf.node')
230
+ )
231
+ try {
232
+ if (localFileExisted) {
233
+ nativeBinding = require('./xgrep.linux-arm-musleabihf.node')
234
+ } else {
235
+ nativeBinding = require('xgrep-linux-arm-musleabihf')
236
+ }
237
+ } catch (e) {
238
+ loadError = e
239
+ }
240
+ } else {
241
+ localFileExisted = existsSync(
242
+ join(__dirname, 'xgrep.linux-arm-gnueabihf.node')
243
+ )
244
+ try {
245
+ if (localFileExisted) {
246
+ nativeBinding = require('./xgrep.linux-arm-gnueabihf.node')
247
+ } else {
248
+ nativeBinding = require('xgrep-linux-arm-gnueabihf')
249
+ }
250
+ } catch (e) {
251
+ loadError = e
252
+ }
253
+ }
254
+ break
255
+ case 'riscv64':
256
+ if (isMusl()) {
257
+ localFileExisted = existsSync(
258
+ join(__dirname, 'xgrep.linux-riscv64-musl.node')
259
+ )
260
+ try {
261
+ if (localFileExisted) {
262
+ nativeBinding = require('./xgrep.linux-riscv64-musl.node')
263
+ } else {
264
+ nativeBinding = require('xgrep-linux-riscv64-musl')
265
+ }
266
+ } catch (e) {
267
+ loadError = e
268
+ }
269
+ } else {
270
+ localFileExisted = existsSync(
271
+ join(__dirname, 'xgrep.linux-riscv64-gnu.node')
272
+ )
273
+ try {
274
+ if (localFileExisted) {
275
+ nativeBinding = require('./xgrep.linux-riscv64-gnu.node')
276
+ } else {
277
+ nativeBinding = require('xgrep-linux-riscv64-gnu')
278
+ }
279
+ } catch (e) {
280
+ loadError = e
281
+ }
282
+ }
283
+ break
284
+ case 's390x':
285
+ localFileExisted = existsSync(
286
+ join(__dirname, 'xgrep.linux-s390x-gnu.node')
287
+ )
288
+ try {
289
+ if (localFileExisted) {
290
+ nativeBinding = require('./xgrep.linux-s390x-gnu.node')
291
+ } else {
292
+ nativeBinding = require('xgrep-linux-s390x-gnu')
293
+ }
294
+ } catch (e) {
295
+ loadError = e
296
+ }
297
+ break
298
+ default:
299
+ throw new Error(`Unsupported architecture on Linux: ${arch}`)
300
+ }
301
+ break
302
+ default:
303
+ throw new Error(`Unsupported OS: ${platform}, architecture: ${arch}`)
304
+ }
305
+
306
+ if (!nativeBinding) {
307
+ if (loadError) {
308
+ throw loadError
309
+ }
310
+ throw new Error(`Failed to load native binding`)
311
+ }
312
+
313
+ const { Xgrep } = nativeBinding
314
+
315
+ module.exports.Xgrep = Xgrep
package/package.json CHANGED
@@ -1,14 +1,52 @@
1
1
  {
2
2
  "name": "xgrep",
3
- "version": "0.0.1",
3
+ "version": "0.1.6",
4
4
  "description": "Ultra-fast indexed code search engine powered by trigram index. Native Rust bindings for Node.js.",
5
5
  "main": "index.js",
6
+ "types": "index.d.ts",
6
7
  "repository": {
7
8
  "type": "git",
8
9
  "url": "https://github.com/momokun7/xgrep"
9
10
  },
10
- "keywords": ["search", "grep", "trigram", "index", "code-search", "rust", "napi"],
11
+ "keywords": [
12
+ "search",
13
+ "grep",
14
+ "trigram",
15
+ "index",
16
+ "code-search",
17
+ "rust",
18
+ "napi"
19
+ ],
11
20
  "author": "nakui",
12
21
  "license": "MIT",
13
- "homepage": "https://github.com/momokun7/xgrep"
22
+ "homepage": "https://github.com/momokun7/xgrep",
23
+ "engines": {
24
+ "node": ">= 14"
25
+ },
26
+ "scripts": {
27
+ "build": "napi build --release --platform",
28
+ "build:debug": "napi build",
29
+ "test": "node __test__/index.mjs"
30
+ },
31
+ "devDependencies": {
32
+ "@napi-rs/cli": "^2"
33
+ },
34
+ "napi": {
35
+ "name": "xgrep",
36
+ "triples": {
37
+ "defaults": true,
38
+ "additional": [
39
+ "aarch64-apple-darwin",
40
+ "x86_64-apple-darwin",
41
+ "x86_64-unknown-linux-gnu",
42
+ "aarch64-unknown-linux-gnu"
43
+ ]
44
+ }
45
+ },
46
+ "files": [
47
+ "index.js",
48
+ "index.d.ts",
49
+ "README.md",
50
+ "xgrep.*.node"
51
+ ]
14
52
  }
Binary file
Binary file
Binary file
Binary file
Binary file