@renxqoo/renx-code 0.0.4 → 0.0.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 +82 -51
- package/bin/renx.cjs +16 -0
- package/package.json +2 -45
- package/src/agent/runtime/runtime.context-usage.test.ts +4 -5
- package/src/agent/runtime/runtime.error-handling.test.ts +4 -5
- package/src/agent/runtime/runtime.test.ts +7 -4
- package/src/agent/runtime/runtime.ts +3 -9
- package/src/agent/runtime/runtime.usage-forwarding.test.ts +4 -5
- package/src/agent/runtime/source-modules.test.ts +16 -35
- package/src/agent/runtime/source-modules.ts +17 -0
- package/vendor/agent-root/src/agent/ENTERPRISE_ACCEPTANCE_CHECKLIST.md +95 -0
- package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.html +1345 -0
- package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.md +1353 -0
- package/vendor/agent-root/src/agent/ERROR_CONTRACT.md +60 -0
- package/vendor/agent-root/src/agent/TEST_COVERAGE_ANALYSIS.md +278 -0
- package/vendor/agent-root/src/agent/__test__/error-contract.test.ts +72 -0
- package/vendor/agent-root/src/agent/__test__/types.test.ts +137 -0
- package/vendor/agent-root/src/agent/agent/__test__/abort-runtime.test.ts +83 -0
- package/vendor/agent-root/src/agent/agent/__test__/callback-safety.test.ts +34 -0
- package/vendor/agent-root/src/agent/agent/__test__/compaction.test.ts +323 -0
- package/vendor/agent-root/src/agent/agent/__test__/concurrency.test.ts +290 -0
- package/vendor/agent-root/src/agent/agent/__test__/error-normalizer.test.ts +377 -0
- package/vendor/agent-root/src/agent/agent/__test__/error.test.ts +212 -0
- package/vendor/agent-root/src/agent/agent/__test__/fault-injection.test.ts +295 -0
- package/vendor/agent-root/src/agent/agent/__test__/index.test.ts +3607 -0
- package/vendor/agent-root/src/agent/agent/__test__/logger.test.ts +35 -0
- package/vendor/agent-root/src/agent/agent/__test__/message-utils.test.ts +517 -0
- package/vendor/agent-root/src/agent/agent/__test__/telemetry.test.ts +97 -0
- package/vendor/agent-root/src/agent/agent/__test__/timeout-budget.test.ts +479 -0
- package/vendor/agent-root/src/agent/agent/__test__/tool-call-merge.test.ts +80 -0
- package/vendor/agent-root/src/agent/agent/__test__/tool-execution-ledger.test.ts +76 -0
- package/vendor/agent-root/src/agent/agent/__test__/write-buffer.test.ts +173 -0
- package/vendor/agent-root/src/agent/agent/__test__/write-file-session.test.ts +109 -0
- package/vendor/agent-root/src/agent/agent/abort-runtime.ts +71 -0
- package/vendor/agent-root/src/agent/agent/callback-safety.ts +33 -0
- package/vendor/agent-root/src/agent/agent/compaction.ts +291 -0
- package/vendor/agent-root/src/agent/agent/concurrency.ts +103 -0
- package/vendor/agent-root/src/agent/agent/error-normalizer.ts +190 -0
- package/vendor/agent-root/src/agent/agent/error.ts +198 -0
- package/vendor/agent-root/src/agent/agent/index.ts +1772 -0
- package/vendor/agent-root/src/agent/agent/logger.ts +65 -0
- package/vendor/agent-root/src/agent/agent/message-utils.ts +101 -0
- package/vendor/agent-root/src/agent/agent/stream-events.ts +61 -0
- package/vendor/agent-root/src/agent/agent/telemetry.ts +123 -0
- package/vendor/agent-root/src/agent/agent/timeout-budget.ts +227 -0
- package/vendor/agent-root/src/agent/agent/tool-call-merge.ts +111 -0
- package/vendor/agent-root/src/agent/agent/tool-execution-ledger.ts +164 -0
- package/vendor/agent-root/src/agent/agent/write-buffer.ts +188 -0
- package/vendor/agent-root/src/agent/agent/write-file-session.ts +238 -0
- package/vendor/agent-root/src/agent/app/__test__/agent-app-service.test.ts +1053 -0
- package/vendor/agent-root/src/agent/app/__test__/minimal-agent-application.test.ts +158 -0
- package/vendor/agent-root/src/agent/app/__test__/sqlite-agent-app-store.test.ts +437 -0
- package/vendor/agent-root/src/agent/app/agent-app-service.ts +748 -0
- package/vendor/agent-root/src/agent/app/contracts.ts +109 -0
- package/vendor/agent-root/src/agent/app/index.ts +5 -0
- package/vendor/agent-root/src/agent/app/minimal-agent-application.ts +151 -0
- package/vendor/agent-root/src/agent/app/ports.ts +72 -0
- package/vendor/agent-root/src/agent/app/sqlite-agent-app-store.ts +1182 -0
- package/vendor/agent-root/src/agent/app/sqlite-client.ts +177 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/00-README.md +36 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/01-scope-and-goals.md +33 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/02-architecture-overview.md +40 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/03-domain-model-and-contracts.md +91 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/04-ports-and-interfaces.md +116 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/05-run-orchestration-and-state-machine.md +52 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/06-cli-commands-and-ux.md +53 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/07-storage-design-local.md +52 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/08-error-and-observability.md +40 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/09-security-and-policy-boundary.md +19 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/10-test-plan-and-acceptance.md +28 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/11-implementation-phases.md +26 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/12-open-questions-and-risks.md +30 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/13-sqlite-schema-fields-and-rationale.md +567 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/14-project-flow-mermaid.md +583 -0
- package/vendor/agent-root/src/agent/docs/cli-app-layer/15-openclaw-style-project-blueprint.md +972 -0
- package/vendor/agent-root/src/agent/error-contract.ts +154 -0
- package/vendor/agent-root/src/agent/prompts/system.ts +246 -0
- package/vendor/agent-root/src/agent/prompts/system1.ts +208 -0
- package/vendor/agent-root/src/agent/storage/__test__/file-history-store.test.ts +98 -0
- package/vendor/agent-root/src/agent/storage/file-history-store.ts +313 -0
- package/vendor/agent-root/src/agent/storage/file-storage-config.ts +94 -0
- package/vendor/agent-root/src/agent/storage/file-system.ts +31 -0
- package/vendor/agent-root/src/agent/storage/file-write-service.ts +21 -0
- package/vendor/agent-root/src/agent/tool/__test__/base-tool.test.ts +413 -0
- package/vendor/agent-root/src/agent/tool/__test__/bash-policy.test.ts +356 -0
- package/vendor/agent-root/src/agent/tool/__test__/bash.mocked-coverage.test.ts +375 -0
- package/vendor/agent-root/src/agent/tool/__test__/bash.test.ts +372 -0
- package/vendor/agent-root/src/agent/tool/__test__/error.test.ts +108 -0
- package/vendor/agent-root/src/agent/tool/__test__/file-edit-tool.test.ts +258 -0
- package/vendor/agent-root/src/agent/tool/__test__/file-history-tools.test.ts +121 -0
- package/vendor/agent-root/src/agent/tool/__test__/file-read-tool.test.ts +210 -0
- package/vendor/agent-root/src/agent/tool/__test__/glob.test.ts +139 -0
- package/vendor/agent-root/src/agent/tool/__test__/grep.mocked-coverage.test.ts +456 -0
- package/vendor/agent-root/src/agent/tool/__test__/grep.test.ts +192 -0
- package/vendor/agent-root/src/agent/tool/__test__/lsp.test.ts +300 -0
- package/vendor/agent-root/src/agent/tool/__test__/outside-workspace-confirmation.test.ts +214 -0
- package/vendor/agent-root/src/agent/tool/__test__/path-security.test.ts +336 -0
- package/vendor/agent-root/src/agent/tool/__test__/skill-loader.test.ts +494 -0
- package/vendor/agent-root/src/agent/tool/__test__/skill-parser.test.ts +543 -0
- package/vendor/agent-root/src/agent/tool/__test__/skill-tool.test.ts +172 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-concurrency-and-version.test.ts +116 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-create-get-list-update.test.ts +267 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-create.test.ts +519 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-errors.test.ts +225 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-output-blocking.test.ts +223 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-output.test.ts +184 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-parent-abort.test.ts +287 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-real-runner-adapter.test.ts +190 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-run-lifecycle.test.ts +352 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-store-runner-branches.test.ts +395 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-store.test.ts +391 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config-integration.test.ts +176 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config.test.ts +68 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-tools-core-edges.test.ts +630 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-tools-runtime-edges.test.ts +732 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-types.test.ts +494 -0
- package/vendor/agent-root/src/agent/tool/__test__/task-utils-branches.test.ts +175 -0
- package/vendor/agent-root/src/agent/tool/__test__/tool-manager.test.ts +505 -0
- package/vendor/agent-root/src/agent/tool/__test__/types.test.ts +55 -0
- package/vendor/agent-root/src/agent/tool/__test__/web-fetch.test.ts +244 -0
- package/vendor/agent-root/src/agent/tool/__test__/web-search.test.ts +290 -0
- package/vendor/agent-root/src/agent/tool/__test__/write-file.test.ts +368 -0
- package/vendor/agent-root/src/agent/tool/base-tool.ts +345 -0
- package/vendor/agent-root/src/agent/tool/bash-policy.ts +636 -0
- package/vendor/agent-root/src/agent/tool/bash.ts +688 -0
- package/vendor/agent-root/src/agent/tool/error.ts +131 -0
- package/vendor/agent-root/src/agent/tool/file-edit-tool.ts +264 -0
- package/vendor/agent-root/src/agent/tool/file-history-list.ts +103 -0
- package/vendor/agent-root/src/agent/tool/file-history-restore.ts +149 -0
- package/vendor/agent-root/src/agent/tool/file-read-tool.ts +211 -0
- package/vendor/agent-root/src/agent/tool/glob.ts +171 -0
- package/vendor/agent-root/src/agent/tool/grep.ts +496 -0
- package/vendor/agent-root/src/agent/tool/lsp.ts +481 -0
- package/vendor/agent-root/src/agent/tool/path-security.ts +117 -0
- package/vendor/agent-root/src/agent/tool/search/common.ts +153 -0
- package/vendor/agent-root/src/agent/tool/skill/index.ts +13 -0
- package/vendor/agent-root/src/agent/tool/skill/loader.ts +229 -0
- package/vendor/agent-root/src/agent/tool/skill/parser.ts +124 -0
- package/vendor/agent-root/src/agent/tool/skill/types.ts +27 -0
- package/vendor/agent-root/src/agent/tool/skill-tool.ts +143 -0
- package/vendor/agent-root/src/agent/tool/task-create.ts +186 -0
- package/vendor/agent-root/src/agent/tool/task-errors.ts +42 -0
- package/vendor/agent-root/src/agent/tool/task-get.ts +116 -0
- package/vendor/agent-root/src/agent/tool/task-graph.ts +78 -0
- package/vendor/agent-root/src/agent/tool/task-list.ts +141 -0
- package/vendor/agent-root/src/agent/tool/task-mock-runner-adapter.ts +232 -0
- package/vendor/agent-root/src/agent/tool/task-output.ts +223 -0
- package/vendor/agent-root/src/agent/tool/task-parent-abort.ts +115 -0
- package/vendor/agent-root/src/agent/tool/task-real-runner-adapter.ts +336 -0
- package/vendor/agent-root/src/agent/tool/task-runner-adapter.ts +55 -0
- package/vendor/agent-root/src/agent/tool/task-stop.ts +187 -0
- package/vendor/agent-root/src/agent/tool/task-store.ts +217 -0
- package/vendor/agent-root/src/agent/tool/task-subagent-config.ts +149 -0
- package/vendor/agent-root/src/agent/tool/task-types.ts +264 -0
- package/vendor/agent-root/src/agent/tool/task-update.ts +315 -0
- package/vendor/agent-root/src/agent/tool/task.ts +209 -0
- package/vendor/agent-root/src/agent/tool/tool-manager.ts +362 -0
- package/vendor/agent-root/src/agent/tool/tool-prompts.ts +242 -0
- package/vendor/agent-root/src/agent/tool/types.ts +116 -0
- package/vendor/agent-root/src/agent/tool/web-fetch.ts +227 -0
- package/vendor/agent-root/src/agent/tool/web-search.ts +208 -0
- package/vendor/agent-root/src/agent/tool/write-file.ts +497 -0
- package/vendor/agent-root/src/agent/types.ts +232 -0
- package/vendor/agent-root/src/agent/utils/__tests__/index.test.ts +18 -0
- package/vendor/agent-root/src/agent/utils/__tests__/message-utils.test.ts +610 -0
- package/vendor/agent-root/src/agent/utils/__tests__/message.test.ts +223 -0
- package/vendor/agent-root/src/agent/utils/__tests__/token.test.ts +42 -0
- package/vendor/agent-root/src/agent/utils/index.ts +16 -0
- package/vendor/agent-root/src/agent/utils/message.ts +171 -0
- package/vendor/agent-root/src/agent/utils/token.ts +28 -0
- package/vendor/agent-root/src/config/__tests__/load-config-to-env.test.ts +129 -0
- package/vendor/agent-root/src/config/__tests__/loader.test.ts +247 -0
- package/vendor/agent-root/src/config/__tests__/runtime.test.ts +88 -0
- package/vendor/agent-root/src/config/index.ts +54 -0
- package/vendor/agent-root/src/config/loader.ts +431 -0
- package/vendor/agent-root/src/config/paths.ts +30 -0
- package/vendor/agent-root/src/config/runtime.ts +163 -0
- package/vendor/agent-root/src/config/types.ts +70 -0
- package/vendor/agent-root/src/logger/index.ts +57 -0
- package/vendor/agent-root/src/logger/logger.ts +819 -0
- package/vendor/agent-root/src/logger/types.ts +150 -0
- package/vendor/agent-root/src/providers/__tests__/errors.test.ts +441 -0
- package/vendor/agent-root/src/providers/__tests__/index.test.ts +16 -0
- package/vendor/agent-root/src/providers/__tests__/openai-compatible.options.test.ts +318 -0
- package/vendor/agent-root/src/providers/__tests__/openai-compatible.test.ts +600 -0
- package/vendor/agent-root/src/providers/__tests__/registry.test.ts +449 -0
- package/vendor/agent-root/src/providers/__tests__/responses-adapter.test.ts +298 -0
- package/vendor/agent-root/src/providers/adapters/__tests__/anthropic.test.ts +354 -0
- package/vendor/agent-root/src/providers/adapters/__tests__/kimi.test.ts +58 -0
- package/vendor/agent-root/src/providers/adapters/__tests__/standard.test.ts +261 -0
- package/vendor/agent-root/src/providers/adapters/anthropic.ts +572 -0
- package/vendor/agent-root/src/providers/adapters/base.ts +131 -0
- package/vendor/agent-root/src/providers/adapters/kimi.ts +48 -0
- package/vendor/agent-root/src/providers/adapters/responses.ts +732 -0
- package/vendor/agent-root/src/providers/adapters/standard.ts +120 -0
- package/vendor/agent-root/src/providers/http/__tests__/client.timeout.test.ts +313 -0
- package/vendor/agent-root/src/providers/http/client.ts +289 -0
- package/vendor/agent-root/src/providers/http/stream-parser.ts +109 -0
- package/vendor/agent-root/src/providers/index.ts +76 -0
- package/vendor/agent-root/src/providers/kimi-headers.ts +177 -0
- package/vendor/agent-root/src/providers/openai-compatible.ts +387 -0
- package/vendor/agent-root/src/providers/registry/model-config.ts +230 -0
- package/vendor/agent-root/src/providers/registry/provider-factory.ts +123 -0
- package/vendor/agent-root/src/providers/registry.ts +135 -0
- package/vendor/agent-root/src/providers/types/api.ts +284 -0
- package/vendor/agent-root/src/providers/types/config.ts +58 -0
- package/vendor/agent-root/src/providers/types/errors.ts +323 -0
- package/vendor/agent-root/src/providers/types/index.ts +72 -0
- package/vendor/agent-root/src/providers/types/provider.ts +45 -0
- package/vendor/agent-root/src/providers/types/registry.ts +88 -0
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { promises as fs } from 'node:fs';
|
|
3
|
+
import * as os from 'node:os';
|
|
4
|
+
import * as path from 'node:path';
|
|
5
|
+
import { GlobTool } from '../glob';
|
|
6
|
+
|
|
7
|
+
describe('GlobTool', () => {
|
|
8
|
+
let rootDir: string;
|
|
9
|
+
let tool: GlobTool;
|
|
10
|
+
|
|
11
|
+
beforeEach(async () => {
|
|
12
|
+
rootDir = await fs.mkdtemp(path.join(os.tmpdir(), 'renx-glob-tool-'));
|
|
13
|
+
tool = new GlobTool({ allowedDirectories: [rootDir] });
|
|
14
|
+
});
|
|
15
|
+
|
|
16
|
+
afterEach(async () => {
|
|
17
|
+
await fs.rm(rootDir, { recursive: true, force: true });
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
it('finds files by pattern and returns metadata', async () => {
|
|
21
|
+
await fs.mkdir(path.join(rootDir, 'src'), { recursive: true });
|
|
22
|
+
await fs.writeFile(path.join(rootDir, 'src', 'a.ts'), 'export const a = 1;', 'utf8');
|
|
23
|
+
await fs.writeFile(path.join(rootDir, 'src', 'b.ts'), 'export const b = 2;', 'utf8');
|
|
24
|
+
await fs.writeFile(path.join(rootDir, 'src', 'c.js'), 'module.exports = {};', 'utf8');
|
|
25
|
+
|
|
26
|
+
const result = await tool.execute({
|
|
27
|
+
pattern: '**/*.ts',
|
|
28
|
+
path: rootDir,
|
|
29
|
+
include_hidden: false,
|
|
30
|
+
max_results: 200,
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
expect(result.success).toBe(true);
|
|
34
|
+
expect(result.output).toContain('Found 2 file(s)');
|
|
35
|
+
|
|
36
|
+
const metadata = result.metadata as {
|
|
37
|
+
total: number;
|
|
38
|
+
files: string[];
|
|
39
|
+
relative_files: string[];
|
|
40
|
+
truncated: boolean;
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
expect(metadata.total).toBe(2);
|
|
44
|
+
expect(metadata.files.every((file) => file.endsWith('.ts'))).toBe(true);
|
|
45
|
+
expect(metadata.relative_files).toEqual(['src/a.ts', 'src/b.ts']);
|
|
46
|
+
expect(metadata.truncated).toBe(false);
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
it('supports include_hidden flag', async () => {
|
|
50
|
+
await fs.writeFile(path.join(rootDir, '.hidden.ts'), 'export const hidden = true;', 'utf8');
|
|
51
|
+
|
|
52
|
+
const hiddenOff = await tool.execute({
|
|
53
|
+
pattern: '**/*.ts',
|
|
54
|
+
path: rootDir,
|
|
55
|
+
include_hidden: false,
|
|
56
|
+
max_results: 200,
|
|
57
|
+
});
|
|
58
|
+
expect(hiddenOff.success).toBe(true);
|
|
59
|
+
expect((hiddenOff.metadata as { total: number }).total).toBe(0);
|
|
60
|
+
|
|
61
|
+
const hiddenOn = await tool.execute({
|
|
62
|
+
pattern: '**/*.ts',
|
|
63
|
+
path: rootDir,
|
|
64
|
+
include_hidden: true,
|
|
65
|
+
max_results: 200,
|
|
66
|
+
});
|
|
67
|
+
expect(hiddenOn.success).toBe(true);
|
|
68
|
+
expect((hiddenOn.metadata as { total: number }).total).toBe(1);
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
it('returns failure for path outside allowed directories', async () => {
|
|
72
|
+
const result = await tool.execute({
|
|
73
|
+
pattern: '**/*.ts',
|
|
74
|
+
path: path.resolve(rootDir, '..'),
|
|
75
|
+
include_hidden: false,
|
|
76
|
+
max_results: 200,
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
expect(result.success).toBe(false);
|
|
80
|
+
expect(result.output).toContain('SEARCH_PATH_NOT_ALLOWED');
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
it('exposes parallel-safe concurrency policy and stable lock key', () => {
|
|
84
|
+
const mode = tool.getConcurrencyMode();
|
|
85
|
+
const lockKey = tool.getConcurrencyLockKey({
|
|
86
|
+
pattern: '**/*.ts',
|
|
87
|
+
path: rootDir,
|
|
88
|
+
include_hidden: false,
|
|
89
|
+
max_results: 10,
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
expect(mode).toBe('parallel-safe');
|
|
93
|
+
expect(lockKey).toBe(`glob:${rootDir}`);
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
it('uses cwd fallback in lock key when path is omitted', () => {
|
|
97
|
+
const lockKey = tool.getConcurrencyLockKey({
|
|
98
|
+
pattern: '**/*.ts',
|
|
99
|
+
include_hidden: false,
|
|
100
|
+
max_results: 10,
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
expect(lockKey).toBe(`glob:${process.cwd()}`);
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
it('falls back to default error code for non-prefixed messages', () => {
|
|
107
|
+
const errorCode = (
|
|
108
|
+
tool as unknown as { extractErrorCode: (msg: string) => string }
|
|
109
|
+
).extractErrorCode('plain error text');
|
|
110
|
+
expect(errorCode).toBe('GLOB_OPERATION_FAILED');
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
it('stringifies non-Error throws from internals', async () => {
|
|
114
|
+
vi.resetModules();
|
|
115
|
+
vi.doMock('../search/common', async () => {
|
|
116
|
+
const actual = await vi.importActual<typeof import('../search/common')>('../search/common');
|
|
117
|
+
return {
|
|
118
|
+
...actual,
|
|
119
|
+
resolveSearchRoot: vi.fn(async () => {
|
|
120
|
+
throw 'plain string thrown';
|
|
121
|
+
}),
|
|
122
|
+
};
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
const { GlobTool: MockedGlobTool } = await import('../glob');
|
|
126
|
+
const mockedTool = new MockedGlobTool({ allowedDirectories: [rootDir] });
|
|
127
|
+
const result = await mockedTool.execute({
|
|
128
|
+
pattern: '**/*.ts',
|
|
129
|
+
path: rootDir,
|
|
130
|
+
include_hidden: false,
|
|
131
|
+
max_results: 10,
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
expect(result.success).toBe(false);
|
|
135
|
+
expect(result.output).toBe('plain string thrown');
|
|
136
|
+
expect((result.metadata as { error: string }).error).toBe('GLOB_OPERATION_FAILED');
|
|
137
|
+
vi.doUnmock('../search/common');
|
|
138
|
+
});
|
|
139
|
+
});
|
|
@@ -0,0 +1,456 @@
|
|
|
1
|
+
import { EventEmitter } from 'node:events';
|
|
2
|
+
import { promises as fs } from 'node:fs';
|
|
3
|
+
import * as os from 'node:os';
|
|
4
|
+
import * as path from 'node:path';
|
|
5
|
+
import { PassThrough } from 'node:stream';
|
|
6
|
+
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
|
7
|
+
import { GrepTool } from '../grep';
|
|
8
|
+
|
|
9
|
+
const { spawnMock, spawnSyncMock } = vi.hoisted(() => ({
|
|
10
|
+
spawnMock: vi.fn(),
|
|
11
|
+
spawnSyncMock: vi.fn(),
|
|
12
|
+
}));
|
|
13
|
+
|
|
14
|
+
vi.mock('node:child_process', () => ({
|
|
15
|
+
spawn: spawnMock,
|
|
16
|
+
spawnSync: spawnSyncMock,
|
|
17
|
+
}));
|
|
18
|
+
|
|
19
|
+
vi.mock('@vscode/ripgrep', () => ({
|
|
20
|
+
rgPath: '/mock/vscode-rg',
|
|
21
|
+
}));
|
|
22
|
+
|
|
23
|
+
function createChild(options: { withStdout?: boolean; withStderr?: boolean } = {}) {
|
|
24
|
+
const child = new EventEmitter() as EventEmitter & {
|
|
25
|
+
stdout?: PassThrough;
|
|
26
|
+
stderr?: PassThrough;
|
|
27
|
+
killed: boolean;
|
|
28
|
+
kill: (signal?: string) => boolean;
|
|
29
|
+
};
|
|
30
|
+
if (options.withStdout !== false) {
|
|
31
|
+
child.stdout = new PassThrough();
|
|
32
|
+
}
|
|
33
|
+
if (options.withStderr !== false) {
|
|
34
|
+
child.stderr = new PassThrough();
|
|
35
|
+
}
|
|
36
|
+
child.killed = false;
|
|
37
|
+
child.kill = vi.fn((): boolean => {
|
|
38
|
+
child.killed = true;
|
|
39
|
+
child.emit('close', 0, null);
|
|
40
|
+
return true;
|
|
41
|
+
}) as unknown as (signal?: string) => boolean;
|
|
42
|
+
return child;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
describe('GrepTool mocked branch coverage', () => {
|
|
46
|
+
let rootDir: string;
|
|
47
|
+
|
|
48
|
+
beforeEach(async () => {
|
|
49
|
+
rootDir = await fs.mkdtemp(path.join(os.tmpdir(), 'renx-grep-mocked-'));
|
|
50
|
+
spawnMock.mockReset();
|
|
51
|
+
spawnSyncMock.mockReset();
|
|
52
|
+
spawnSyncMock.mockReturnValue({ status: 0, error: undefined });
|
|
53
|
+
delete process.env.RIPGREP_PATH;
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
afterEach(async () => {
|
|
57
|
+
vi.restoreAllMocks();
|
|
58
|
+
delete process.env.RIPGREP_PATH;
|
|
59
|
+
await fs.rm(rootDir, { recursive: true, force: true });
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it('returns RIPGREP_NOT_FOUND when all candidate probes fail', async () => {
|
|
63
|
+
process.env.RIPGREP_PATH = 'env-rg';
|
|
64
|
+
spawnSyncMock.mockReturnValue({
|
|
65
|
+
status: 1,
|
|
66
|
+
error: new Error('missing'),
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
const tool = new GrepTool({
|
|
70
|
+
allowedDirectories: [rootDir],
|
|
71
|
+
rgPath: 'custom-rg',
|
|
72
|
+
});
|
|
73
|
+
const result = await tool.execute({
|
|
74
|
+
pattern: 'hello',
|
|
75
|
+
path: rootDir,
|
|
76
|
+
timeout_ms: 1000,
|
|
77
|
+
max_results: 10,
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
expect(result.success).toBe(false);
|
|
81
|
+
expect(result.output).toContain('RIPGREP_NOT_FOUND');
|
|
82
|
+
expect((result.metadata as { error: string }).error).toBe('RIPGREP_NOT_FOUND');
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
it('maps non-Error execute failures using String(error)', async () => {
|
|
86
|
+
const tool = new GrepTool({ allowedDirectories: [rootDir], rgPath: 'custom-rg' });
|
|
87
|
+
const helper = tool as unknown as {
|
|
88
|
+
runRipgrep: () => Promise<never>;
|
|
89
|
+
};
|
|
90
|
+
vi.spyOn(helper, 'runRipgrep').mockRejectedValueOnce('plain failure');
|
|
91
|
+
|
|
92
|
+
const result = await tool.execute({
|
|
93
|
+
pattern: 'hello',
|
|
94
|
+
path: rootDir,
|
|
95
|
+
timeout_ms: 1000,
|
|
96
|
+
max_results: 10,
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
expect(result.success).toBe(false);
|
|
100
|
+
expect(result.output).toBe('plain failure');
|
|
101
|
+
expect((result.metadata as { error: string }).error).toBe('GREP_EXECUTION_ERROR');
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
it('covers parser fallbacks, stderr chunk handling, and max-results truncation', async () => {
|
|
105
|
+
const tool = new GrepTool({ allowedDirectories: [rootDir] });
|
|
106
|
+
const helper = tool as unknown as {
|
|
107
|
+
runRipgrep: (
|
|
108
|
+
ripgrepBinary: string,
|
|
109
|
+
commandArgs: string[],
|
|
110
|
+
rootPath: string,
|
|
111
|
+
args: { pattern: string; timeout_ms: number; max_results: number },
|
|
112
|
+
context?: { onChunk?: (chunk: unknown) => Promise<void> | void }
|
|
113
|
+
) => Promise<{ data: Record<string, unknown>; output: string }>;
|
|
114
|
+
};
|
|
115
|
+
|
|
116
|
+
const child = createChild();
|
|
117
|
+
spawnMock.mockReturnValueOnce(child);
|
|
118
|
+
|
|
119
|
+
const chunks: unknown[] = [];
|
|
120
|
+
const run = helper.runRipgrep(
|
|
121
|
+
'rg',
|
|
122
|
+
['--json'],
|
|
123
|
+
rootDir,
|
|
124
|
+
{ pattern: 'hello', timeout_ms: 2000, max_results: 1 },
|
|
125
|
+
{
|
|
126
|
+
onChunk: async (chunk) => {
|
|
127
|
+
chunks.push(chunk);
|
|
128
|
+
},
|
|
129
|
+
}
|
|
130
|
+
);
|
|
131
|
+
|
|
132
|
+
child.stderr?.emit('data', 'stderr-string');
|
|
133
|
+
child.stderr?.write(Buffer.from('stderr-line'));
|
|
134
|
+
child.stdout?.write('\n');
|
|
135
|
+
child.stdout?.write('123\n');
|
|
136
|
+
child.stdout?.write('{"type":"match","data":null}\n');
|
|
137
|
+
child.stdout?.write(
|
|
138
|
+
'{"type":"match","data":{"path":123,"lines":{"text":"skip\\n"},"line_number":3}}\n'
|
|
139
|
+
);
|
|
140
|
+
child.stdout?.write(
|
|
141
|
+
'{"type":"match","data":{"path":{},"lines":{"text":"skip\\n"},"line_number":4}}\n'
|
|
142
|
+
);
|
|
143
|
+
child.stdout?.write(
|
|
144
|
+
'{"type":"match","data":{"path":{"bytes":"ZmlsZS50cw=="},"lines":{"bytes":"aGVsbG8K"},"line_number":"x","submatches":"oops"}}\n'
|
|
145
|
+
);
|
|
146
|
+
child.stdout?.end();
|
|
147
|
+
|
|
148
|
+
const result = await run;
|
|
149
|
+
expect((result.data as { truncated: boolean }).truncated).toBe(true);
|
|
150
|
+
expect(result.output).toContain('(truncated)');
|
|
151
|
+
expect(chunks.length).toBeGreaterThan(0);
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
it('covers stderr Buffer chunk decoding branch', async () => {
|
|
155
|
+
const tool = new GrepTool({ allowedDirectories: [rootDir] });
|
|
156
|
+
const helper = tool as unknown as {
|
|
157
|
+
runRipgrep: (
|
|
158
|
+
ripgrepBinary: string,
|
|
159
|
+
commandArgs: string[],
|
|
160
|
+
rootPath: string,
|
|
161
|
+
args: { pattern: string; timeout_ms: number; max_results: number }
|
|
162
|
+
) => Promise<{ data: Record<string, unknown>; output: string }>;
|
|
163
|
+
};
|
|
164
|
+
|
|
165
|
+
const child = createChild();
|
|
166
|
+
const stderrEmitter = new EventEmitter() as EventEmitter & {
|
|
167
|
+
setEncoding: (encoding: string) => void;
|
|
168
|
+
};
|
|
169
|
+
stderrEmitter.setEncoding = vi.fn();
|
|
170
|
+
child.stderr = stderrEmitter as unknown as PassThrough;
|
|
171
|
+
spawnMock.mockReturnValueOnce(child);
|
|
172
|
+
|
|
173
|
+
const run = helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
174
|
+
pattern: 'x',
|
|
175
|
+
timeout_ms: 1000,
|
|
176
|
+
max_results: 10,
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
stderrEmitter.emit('data', Buffer.from('stderr-buffer'));
|
|
180
|
+
child.stdout?.end();
|
|
181
|
+
child.emit('close', 1, null);
|
|
182
|
+
|
|
183
|
+
const result = await run;
|
|
184
|
+
expect(result.output).toBe('No matches found');
|
|
185
|
+
});
|
|
186
|
+
|
|
187
|
+
it('covers timeout branch and killChild catch branch', async () => {
|
|
188
|
+
const tool = new GrepTool({ allowedDirectories: [rootDir] });
|
|
189
|
+
const helper = tool as unknown as {
|
|
190
|
+
runRipgrep: (
|
|
191
|
+
ripgrepBinary: string,
|
|
192
|
+
commandArgs: string[],
|
|
193
|
+
rootPath: string,
|
|
194
|
+
args: { pattern: string; timeout_ms: number; max_results: number }
|
|
195
|
+
) => Promise<{ data: Record<string, unknown>; output: string }>;
|
|
196
|
+
};
|
|
197
|
+
|
|
198
|
+
const child = createChild();
|
|
199
|
+
child.kill = vi.fn((): boolean => {
|
|
200
|
+
throw new Error('kill failed');
|
|
201
|
+
}) as unknown as (signal?: string) => boolean;
|
|
202
|
+
spawnMock.mockReturnValueOnce(child);
|
|
203
|
+
|
|
204
|
+
const run = helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
205
|
+
pattern: 'slow',
|
|
206
|
+
timeout_ms: 10,
|
|
207
|
+
max_results: 100,
|
|
208
|
+
});
|
|
209
|
+
setTimeout(() => {
|
|
210
|
+
child.stdout?.end();
|
|
211
|
+
child.emit('close', null, 'SIGTERM');
|
|
212
|
+
}, 30);
|
|
213
|
+
|
|
214
|
+
const result = await run;
|
|
215
|
+
expect(result.output).toContain('Search timed out after 10ms');
|
|
216
|
+
expect((result.data as { timed_out: boolean }).timed_out).toBe(true);
|
|
217
|
+
});
|
|
218
|
+
|
|
219
|
+
it('throws RIPGREP_ERROR when stdout stream cannot be captured', async () => {
|
|
220
|
+
const tool = new GrepTool({ allowedDirectories: [rootDir] });
|
|
221
|
+
const helper = tool as unknown as {
|
|
222
|
+
runRipgrep: (
|
|
223
|
+
ripgrepBinary: string,
|
|
224
|
+
commandArgs: string[],
|
|
225
|
+
rootPath: string,
|
|
226
|
+
args: { pattern: string; timeout_ms: number; max_results: number }
|
|
227
|
+
) => Promise<{ data: Record<string, unknown>; output: string }>;
|
|
228
|
+
};
|
|
229
|
+
|
|
230
|
+
const child = createChild({ withStdout: false });
|
|
231
|
+
spawnMock.mockReturnValueOnce(child);
|
|
232
|
+
|
|
233
|
+
await expect(
|
|
234
|
+
helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
235
|
+
pattern: 'x',
|
|
236
|
+
timeout_ms: 1000,
|
|
237
|
+
max_results: 10,
|
|
238
|
+
})
|
|
239
|
+
).rejects.toThrow('RIPGREP_ERROR: failed to capture ripgrep stdout stream');
|
|
240
|
+
});
|
|
241
|
+
|
|
242
|
+
it('throws RIPGREP_STREAM_ERROR on stdout stream failures', async () => {
|
|
243
|
+
const tool = new GrepTool({ allowedDirectories: [rootDir] });
|
|
244
|
+
const helper = tool as unknown as {
|
|
245
|
+
runRipgrep: (
|
|
246
|
+
ripgrepBinary: string,
|
|
247
|
+
commandArgs: string[],
|
|
248
|
+
rootPath: string,
|
|
249
|
+
args: { pattern: string; timeout_ms: number; max_results: number }
|
|
250
|
+
) => Promise<{ data: Record<string, unknown>; output: string }>;
|
|
251
|
+
};
|
|
252
|
+
|
|
253
|
+
const child = createChild();
|
|
254
|
+
spawnMock.mockReturnValueOnce(child);
|
|
255
|
+
|
|
256
|
+
const run = helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
257
|
+
pattern: 'x',
|
|
258
|
+
timeout_ms: 1000,
|
|
259
|
+
max_results: 10,
|
|
260
|
+
});
|
|
261
|
+
|
|
262
|
+
child.stdout?.emit('error', new Error('stream exploded'));
|
|
263
|
+
child.emit('close', 0, null);
|
|
264
|
+
await expect(run).rejects.toThrow('RIPGREP_STREAM_ERROR: stream exploded');
|
|
265
|
+
|
|
266
|
+
const secondChild = createChild();
|
|
267
|
+
spawnMock.mockReturnValueOnce(secondChild);
|
|
268
|
+
const secondRun = helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
269
|
+
pattern: 'x',
|
|
270
|
+
timeout_ms: 1000,
|
|
271
|
+
max_results: 10,
|
|
272
|
+
});
|
|
273
|
+
secondChild.stdout?.emit('error', 'string-stream-error');
|
|
274
|
+
secondChild.emit('close', 0, null);
|
|
275
|
+
await expect(secondRun).rejects.toThrow('RIPGREP_STREAM_ERROR: string-stream-error');
|
|
276
|
+
});
|
|
277
|
+
|
|
278
|
+
it('throws signal and exit-code errors from ripgrep process', async () => {
|
|
279
|
+
const tool = new GrepTool({ allowedDirectories: [rootDir] });
|
|
280
|
+
const helper = tool as unknown as {
|
|
281
|
+
runRipgrep: (
|
|
282
|
+
ripgrepBinary: string,
|
|
283
|
+
commandArgs: string[],
|
|
284
|
+
rootPath: string,
|
|
285
|
+
args: { pattern: string; timeout_ms: number; max_results: number }
|
|
286
|
+
) => Promise<{ data: Record<string, unknown>; output: string }>;
|
|
287
|
+
};
|
|
288
|
+
|
|
289
|
+
const signalChild = createChild();
|
|
290
|
+
spawnMock.mockReturnValueOnce(signalChild);
|
|
291
|
+
const signalRun = helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
292
|
+
pattern: 'x',
|
|
293
|
+
timeout_ms: 1000,
|
|
294
|
+
max_results: 10,
|
|
295
|
+
});
|
|
296
|
+
signalChild.stdout?.end();
|
|
297
|
+
signalChild.emit('close', null, 'SIGTERM');
|
|
298
|
+
await expect(signalRun).rejects.toThrow('RIPGREP_ERROR: ripgrep terminated by signal SIGTERM');
|
|
299
|
+
|
|
300
|
+
const code2Child = createChild();
|
|
301
|
+
spawnMock.mockReturnValueOnce(code2Child);
|
|
302
|
+
const code2Run = helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
303
|
+
pattern: 'x',
|
|
304
|
+
timeout_ms: 1000,
|
|
305
|
+
max_results: 10,
|
|
306
|
+
});
|
|
307
|
+
code2Child.stderr?.write('bad regex');
|
|
308
|
+
code2Child.stdout?.end();
|
|
309
|
+
code2Child.emit('close', 2, null);
|
|
310
|
+
await expect(code2Run).rejects.toThrow('RIPGREP_ERROR: bad regex');
|
|
311
|
+
|
|
312
|
+
const code2FallbackChild = createChild();
|
|
313
|
+
spawnMock.mockReturnValueOnce(code2FallbackChild);
|
|
314
|
+
const code2FallbackRun = helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
315
|
+
pattern: 'x',
|
|
316
|
+
timeout_ms: 1000,
|
|
317
|
+
max_results: 10,
|
|
318
|
+
});
|
|
319
|
+
code2FallbackChild.stdout?.end();
|
|
320
|
+
code2FallbackChild.emit('close', 2, null);
|
|
321
|
+
await expect(code2FallbackRun).rejects.toThrow('RIPGREP_ERROR: ripgrep execution failed');
|
|
322
|
+
|
|
323
|
+
const code5Child = createChild();
|
|
324
|
+
spawnMock.mockReturnValueOnce(code5Child);
|
|
325
|
+
const code5Run = helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
326
|
+
pattern: 'x',
|
|
327
|
+
timeout_ms: 1000,
|
|
328
|
+
max_results: 10,
|
|
329
|
+
});
|
|
330
|
+
code5Child.stdout?.end();
|
|
331
|
+
code5Child.emit('close', 5, null);
|
|
332
|
+
await expect(code5Run).rejects.toThrow('RIPGREP_ERROR: ripgrep exited with code 5');
|
|
333
|
+
});
|
|
334
|
+
|
|
335
|
+
it('covers decodeTextField base64 catch path with mocked Buffer.from failure', async () => {
|
|
336
|
+
const tool = new GrepTool({ allowedDirectories: [rootDir] });
|
|
337
|
+
const helper = tool as unknown as {
|
|
338
|
+
runRipgrep: (
|
|
339
|
+
ripgrepBinary: string,
|
|
340
|
+
commandArgs: string[],
|
|
341
|
+
rootPath: string,
|
|
342
|
+
args: { pattern: string; timeout_ms: number; max_results: number }
|
|
343
|
+
) => Promise<{ data: Record<string, unknown>; output: string }>;
|
|
344
|
+
};
|
|
345
|
+
|
|
346
|
+
const originalBufferFrom = Buffer.from.bind(Buffer);
|
|
347
|
+
const originalBufferFromString = originalBufferFrom as unknown as (
|
|
348
|
+
value: string,
|
|
349
|
+
encoding?: BufferEncoding
|
|
350
|
+
) => Buffer;
|
|
351
|
+
const originalBufferFromBytes = originalBufferFrom as unknown as (
|
|
352
|
+
value: ArrayLike<number>
|
|
353
|
+
) => Buffer;
|
|
354
|
+
const bufferSpy = vi.spyOn(Buffer, 'from').mockImplementation(((
|
|
355
|
+
value: string | ArrayBuffer | SharedArrayBuffer | ArrayBufferView,
|
|
356
|
+
encodingOrOffset?: BufferEncoding | number,
|
|
357
|
+
length?: number
|
|
358
|
+
) => {
|
|
359
|
+
if (encodingOrOffset === 'base64') {
|
|
360
|
+
throw new Error('bad-base64');
|
|
361
|
+
}
|
|
362
|
+
if (typeof value === 'string') {
|
|
363
|
+
return originalBufferFromString(value, encodingOrOffset as BufferEncoding | undefined);
|
|
364
|
+
}
|
|
365
|
+
if (typeof encodingOrOffset === 'number') {
|
|
366
|
+
return originalBufferFromBytes(
|
|
367
|
+
new Uint8Array(value as ArrayBuffer | SharedArrayBuffer).subarray(
|
|
368
|
+
encodingOrOffset,
|
|
369
|
+
typeof length === 'number' ? encodingOrOffset + length : undefined
|
|
370
|
+
)
|
|
371
|
+
);
|
|
372
|
+
}
|
|
373
|
+
if (ArrayBuffer.isView(value)) {
|
|
374
|
+
return originalBufferFromBytes(
|
|
375
|
+
new Uint8Array(value.buffer, value.byteOffset, value.byteLength)
|
|
376
|
+
);
|
|
377
|
+
}
|
|
378
|
+
return originalBufferFromBytes(new Uint8Array(value as ArrayBuffer | SharedArrayBuffer));
|
|
379
|
+
}) as typeof Buffer.from);
|
|
380
|
+
|
|
381
|
+
const child = createChild();
|
|
382
|
+
spawnMock.mockReturnValueOnce(child);
|
|
383
|
+
const run = helper.runRipgrep('rg', ['--json'], rootDir, {
|
|
384
|
+
pattern: 'x',
|
|
385
|
+
timeout_ms: 1000,
|
|
386
|
+
max_results: 10,
|
|
387
|
+
});
|
|
388
|
+
|
|
389
|
+
child.stdout?.write(
|
|
390
|
+
'{"type":"match","data":{"path":{"bytes":"bm90LXVzZWQ="},"lines":{"text":"line\\n"},"line_number":1,"submatches":[{"start":0}]}}\n'
|
|
391
|
+
);
|
|
392
|
+
child.stdout?.end();
|
|
393
|
+
child.emit('close', 1, null);
|
|
394
|
+
|
|
395
|
+
const result = await run;
|
|
396
|
+
expect(result.output).toBe('No matches found');
|
|
397
|
+
bufferSpy.mockRestore();
|
|
398
|
+
});
|
|
399
|
+
|
|
400
|
+
it('covers line label fallback and lock-key cwd fallback', () => {
|
|
401
|
+
const tool = new GrepTool({ allowedDirectories: [rootDir] });
|
|
402
|
+
const helper = tool as unknown as {
|
|
403
|
+
buildSuccessResult: (
|
|
404
|
+
fileMap: Map<
|
|
405
|
+
string,
|
|
406
|
+
{
|
|
407
|
+
file: string;
|
|
408
|
+
matchCount: number;
|
|
409
|
+
matches: Array<{ line: number | null; column: number | null; text: string }>;
|
|
410
|
+
}
|
|
411
|
+
>,
|
|
412
|
+
totalMatches: number,
|
|
413
|
+
options: {
|
|
414
|
+
pattern: string;
|
|
415
|
+
rootPath: string;
|
|
416
|
+
truncated: boolean;
|
|
417
|
+
timedOut: boolean;
|
|
418
|
+
message: string;
|
|
419
|
+
}
|
|
420
|
+
) => { data: Record<string, unknown>; output: string };
|
|
421
|
+
};
|
|
422
|
+
const map = new Map<
|
|
423
|
+
string,
|
|
424
|
+
{
|
|
425
|
+
file: string;
|
|
426
|
+
matchCount: number;
|
|
427
|
+
matches: Array<{ line: number | null; column: number | null; text: string }>;
|
|
428
|
+
}
|
|
429
|
+
>([
|
|
430
|
+
[
|
|
431
|
+
'a.ts',
|
|
432
|
+
{
|
|
433
|
+
file: 'a.ts',
|
|
434
|
+
matchCount: 1,
|
|
435
|
+
matches: [{ line: null, column: null, text: 'hello' }],
|
|
436
|
+
},
|
|
437
|
+
],
|
|
438
|
+
]);
|
|
439
|
+
|
|
440
|
+
const built = helper.buildSuccessResult(map, 1, {
|
|
441
|
+
pattern: 'hello',
|
|
442
|
+
rootPath: rootDir,
|
|
443
|
+
truncated: false,
|
|
444
|
+
timedOut: false,
|
|
445
|
+
message: 'Search completed',
|
|
446
|
+
});
|
|
447
|
+
expect(built.output).toContain('Line ?');
|
|
448
|
+
|
|
449
|
+
const lockKey = tool.getConcurrencyLockKey({
|
|
450
|
+
pattern: 'hello',
|
|
451
|
+
timeout_ms: 1000,
|
|
452
|
+
max_results: 10,
|
|
453
|
+
});
|
|
454
|
+
expect(lockKey).toBe(`grep:${process.cwd()}:hello`);
|
|
455
|
+
});
|
|
456
|
+
});
|