@diff-review-system/drs 2.2.1 → 3.0.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/README.md +228 -92
- package/dist/ci/runner.d.ts.map +1 -1
- package/dist/ci/runner.js +19 -22
- package/dist/ci/runner.js.map +1 -1
- package/dist/cli/describe-mr.d.ts.map +1 -1
- package/dist/cli/describe-mr.js +39 -20
- package/dist/cli/describe-mr.js.map +1 -1
- package/dist/cli/describe-pr.d.ts.map +1 -1
- package/dist/cli/describe-pr.js +39 -20
- package/dist/cli/describe-pr.js.map +1 -1
- package/dist/cli/index.js +11 -7
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/init.d.ts.map +1 -1
- package/dist/cli/init.js +30 -2
- package/dist/cli/init.js.map +1 -1
- package/dist/cli/post-comments.d.ts.map +1 -1
- package/dist/cli/post-comments.js +5 -5
- package/dist/cli/post-comments.js.map +1 -1
- package/dist/cli/review-local.d.ts.map +1 -1
- package/dist/cli/review-local.integration.test.d.ts +2 -0
- package/dist/cli/review-local.integration.test.d.ts.map +1 -0
- package/dist/cli/review-local.integration.test.js +343 -0
- package/dist/cli/review-local.integration.test.js.map +1 -0
- package/dist/cli/review-local.js +5 -4
- package/dist/cli/review-local.js.map +1 -1
- package/dist/cli/review-local.live.e2e.test.d.ts +2 -0
- package/dist/cli/review-local.live.e2e.test.d.ts.map +1 -0
- package/dist/cli/review-local.live.e2e.test.js +154 -0
- package/dist/cli/review-local.live.e2e.test.js.map +1 -0
- package/dist/cli/review-local.test.d.ts +2 -0
- package/dist/cli/review-local.test.d.ts.map +1 -0
- package/dist/cli/review-local.test.js +164 -0
- package/dist/cli/review-local.test.js.map +1 -0
- package/dist/cli/review-mr.d.ts +1 -1
- package/dist/cli/review-mr.d.ts.map +1 -1
- package/dist/cli/review-mr.js +92 -17
- package/dist/cli/review-mr.js.map +1 -1
- package/dist/cli/review-mr.test.d.ts +2 -0
- package/dist/cli/review-mr.test.d.ts.map +1 -0
- package/dist/cli/review-mr.test.js +142 -0
- package/dist/cli/review-mr.test.js.map +1 -0
- package/dist/cli/review-pr.d.ts +1 -1
- package/dist/cli/review-pr.d.ts.map +1 -1
- package/dist/cli/review-pr.js +96 -13
- package/dist/cli/review-pr.js.map +1 -1
- package/dist/cli/review-pr.test.d.ts +2 -0
- package/dist/cli/review-pr.test.d.ts.map +1 -0
- package/dist/cli/review-pr.test.js +137 -0
- package/dist/cli/review-pr.test.js.map +1 -0
- package/dist/cli/show-changes.js +4 -4
- package/dist/github/platform-adapter.js +2 -2
- package/dist/gitlab/client.js +1 -1
- package/dist/lib/code-quality-report.js +1 -1
- package/dist/lib/comment-formatter.d.ts +2 -1
- package/dist/lib/comment-formatter.d.ts.map +1 -1
- package/dist/lib/comment-formatter.js +33 -1
- package/dist/lib/comment-formatter.js.map +1 -1
- package/dist/lib/comment-formatter.test.js +43 -0
- package/dist/lib/comment-formatter.test.js.map +1 -1
- package/dist/lib/comment-manager.d.ts.map +1 -1
- package/dist/lib/comment-manager.js +4 -3
- package/dist/lib/comment-manager.js.map +1 -1
- package/dist/lib/comment-poster.d.ts +2 -1
- package/dist/lib/comment-poster.d.ts.map +1 -1
- package/dist/lib/comment-poster.js +2 -2
- package/dist/lib/comment-poster.js.map +1 -1
- package/dist/lib/comment-poster.test.js +27 -11
- package/dist/lib/comment-poster.test.js.map +1 -1
- package/dist/lib/config-model-overrides.test.d.ts +1 -1
- package/dist/lib/config-model-overrides.test.js +2 -2
- package/dist/lib/config-model-overrides.test.js.map +1 -1
- package/dist/lib/config.d.ts +34 -7
- package/dist/lib/config.d.ts.map +1 -1
- package/dist/lib/config.js +35 -13
- package/dist/lib/config.js.map +1 -1
- package/dist/lib/config.test.js +16 -0
- package/dist/lib/config.test.js.map +1 -1
- package/dist/lib/context-compression.d.ts +27 -1
- package/dist/lib/context-compression.d.ts.map +1 -1
- package/dist/lib/context-compression.js +106 -4
- package/dist/lib/context-compression.js.map +1 -1
- package/dist/lib/context-compression.test.js +305 -1
- package/dist/lib/context-compression.test.js.map +1 -1
- package/dist/lib/context-loader.d.ts +3 -2
- package/dist/lib/context-loader.d.ts.map +1 -1
- package/dist/lib/context-loader.js +11 -11
- package/dist/lib/context-loader.js.map +1 -1
- package/dist/lib/description-executor.d.ts +19 -2
- package/dist/lib/description-executor.d.ts.map +1 -1
- package/dist/lib/description-executor.js +52 -21
- package/dist/lib/description-executor.js.map +1 -1
- package/dist/lib/description-executor.test.d.ts +2 -0
- package/dist/lib/description-executor.test.d.ts.map +1 -0
- package/dist/lib/description-executor.test.js +120 -0
- package/dist/lib/description-executor.test.js.map +1 -0
- package/dist/lib/description-formatter.d.ts +8 -3
- package/dist/lib/description-formatter.d.ts.map +1 -1
- package/dist/lib/description-formatter.js +88 -13
- package/dist/lib/description-formatter.js.map +1 -1
- package/dist/lib/description-formatter.test.d.ts +2 -0
- package/dist/lib/description-formatter.test.d.ts.map +1 -0
- package/dist/lib/description-formatter.test.js +57 -0
- package/dist/lib/description-formatter.test.js.map +1 -0
- package/dist/lib/diff-parser.test.d.ts +2 -0
- package/dist/lib/diff-parser.test.d.ts.map +1 -0
- package/dist/lib/diff-parser.test.js +335 -0
- package/dist/lib/diff-parser.test.js.map +1 -0
- package/dist/lib/exit.d.ts +35 -0
- package/dist/lib/exit.d.ts.map +1 -0
- package/dist/lib/exit.js +53 -0
- package/dist/lib/exit.js.map +1 -0
- package/dist/lib/exit.test.d.ts +2 -0
- package/dist/lib/exit.test.d.ts.map +1 -0
- package/dist/lib/exit.test.js +120 -0
- package/dist/lib/exit.test.js.map +1 -0
- package/dist/lib/format-utils.d.ts +3 -0
- package/dist/lib/format-utils.d.ts.map +1 -0
- package/dist/lib/format-utils.js +7 -0
- package/dist/lib/format-utils.js.map +1 -0
- package/dist/lib/json-output.d.ts +4 -1
- package/dist/lib/json-output.d.ts.map +1 -1
- package/dist/lib/json-output.js +2 -1
- package/dist/lib/json-output.js.map +1 -1
- package/dist/lib/json-output.test.d.ts +2 -0
- package/dist/lib/json-output.test.d.ts.map +1 -0
- package/dist/lib/json-output.test.js +135 -0
- package/dist/lib/json-output.test.js.map +1 -0
- package/dist/lib/logger.d.ts +10 -2
- package/dist/lib/logger.d.ts.map +1 -1
- package/dist/lib/logger.js +22 -4
- package/dist/lib/logger.js.map +1 -1
- package/dist/lib/logger.test.d.ts +2 -0
- package/dist/lib/logger.test.d.ts.map +1 -0
- package/dist/lib/logger.test.js +324 -0
- package/dist/lib/logger.test.js.map +1 -0
- package/dist/lib/position-validator.test.d.ts +2 -0
- package/dist/lib/position-validator.test.d.ts.map +1 -0
- package/dist/lib/position-validator.test.js +128 -0
- package/dist/lib/position-validator.test.js.map +1 -0
- package/dist/lib/repository-validator.js +1 -1
- package/dist/lib/review-core.d.ts +9 -4
- package/dist/lib/review-core.d.ts.map +1 -1
- package/dist/lib/review-core.js +207 -112
- package/dist/lib/review-core.js.map +1 -1
- package/dist/lib/review-core.test.js +76 -30
- package/dist/lib/review-core.test.js.map +1 -1
- package/dist/lib/review-orchestrator.d.ts +12 -7
- package/dist/lib/review-orchestrator.d.ts.map +1 -1
- package/dist/lib/review-orchestrator.js +78 -22
- package/dist/lib/review-orchestrator.js.map +1 -1
- package/dist/lib/review-orchestrator.test.js +160 -42
- package/dist/lib/review-orchestrator.test.js.map +1 -1
- package/dist/lib/review-parser.test.d.ts +2 -0
- package/dist/lib/review-parser.test.d.ts.map +1 -0
- package/dist/lib/review-parser.test.js +130 -0
- package/dist/lib/review-parser.test.js.map +1 -0
- package/dist/lib/review-usage.d.ts +32 -0
- package/dist/lib/review-usage.d.ts.map +1 -0
- package/dist/lib/review-usage.js +72 -0
- package/dist/lib/review-usage.js.map +1 -0
- package/dist/lib/review-usage.test.d.ts +2 -0
- package/dist/lib/review-usage.test.d.ts.map +1 -0
- package/dist/lib/review-usage.test.js +83 -0
- package/dist/lib/review-usage.test.js.map +1 -0
- package/dist/lib/unified-review-executor.d.ts +6 -2
- package/dist/lib/unified-review-executor.d.ts.map +1 -1
- package/dist/lib/unified-review-executor.js +54 -28
- package/dist/lib/unified-review-executor.js.map +1 -1
- package/dist/lib/unified-review-executor.test.js +138 -16
- package/dist/lib/unified-review-executor.test.js.map +1 -1
- package/dist/lib/write-json-output.test.d.ts +2 -0
- package/dist/lib/write-json-output.test.d.ts.map +1 -0
- package/dist/lib/write-json-output.test.js +259 -0
- package/dist/lib/write-json-output.test.js.map +1 -0
- package/dist/pi/sdk.d.ts +94 -0
- package/dist/pi/sdk.d.ts.map +1 -0
- package/dist/pi/sdk.js +486 -0
- package/dist/pi/sdk.js.map +1 -0
- package/dist/pi/sdk.test.d.ts +2 -0
- package/dist/pi/sdk.test.d.ts.map +1 -0
- package/dist/pi/sdk.test.js +331 -0
- package/dist/pi/sdk.test.js.map +1 -0
- package/dist/{opencode → runtime}/agent-loader.d.ts +7 -5
- package/dist/runtime/agent-loader.d.ts.map +1 -0
- package/dist/{opencode → runtime}/agent-loader.js +24 -19
- package/dist/runtime/agent-loader.js.map +1 -0
- package/dist/runtime/agent-loader.test.d.ts +2 -0
- package/dist/runtime/agent-loader.test.d.ts.map +1 -0
- package/dist/runtime/agent-loader.test.js +280 -0
- package/dist/runtime/agent-loader.test.js.map +1 -0
- package/dist/runtime/built-in-paths.d.ts +2 -0
- package/dist/runtime/built-in-paths.d.ts.map +1 -0
- package/dist/runtime/built-in-paths.js +14 -0
- package/dist/runtime/built-in-paths.js.map +1 -0
- package/dist/{opencode → runtime}/client.d.ts +35 -18
- package/dist/runtime/client.d.ts.map +1 -0
- package/dist/runtime/client.js +486 -0
- package/dist/runtime/client.js.map +1 -0
- package/dist/{opencode → runtime}/client.test.d.ts.map +1 -1
- package/dist/runtime/client.test.js +392 -0
- package/dist/runtime/client.test.js.map +1 -0
- package/dist/runtime/path-config.d.ts +8 -0
- package/dist/runtime/path-config.d.ts.map +1 -0
- package/dist/runtime/path-config.js +68 -0
- package/dist/runtime/path-config.js.map +1 -0
- package/dist/runtime/path-config.test.d.ts +2 -0
- package/dist/runtime/path-config.test.d.ts.map +1 -0
- package/dist/runtime/path-config.test.js +103 -0
- package/dist/runtime/path-config.test.js.map +1 -0
- package/package.json +5 -5
- package/.opencode/opencode.jsonc +0 -15
- package/.opencode/tool/write_json_output.ts +0 -24
- package/.opencode/tools/drs_skill.ts +0 -67
- package/dist/lib/skills-prompt.d.ts +0 -3
- package/dist/lib/skills-prompt.d.ts.map +0 -1
- package/dist/lib/skills-prompt.js +0 -70
- package/dist/lib/skills-prompt.js.map +0 -1
- package/dist/opencode/agent-loader.d.ts.map +0 -1
- package/dist/opencode/agent-loader.js.map +0 -1
- package/dist/opencode/client.d.ts.map +0 -1
- package/dist/opencode/client.js +0 -456
- package/dist/opencode/client.js.map +0 -1
- package/dist/opencode/client.test.js +0 -317
- package/dist/opencode/client.test.js.map +0 -1
- package/dist/opencode/opencode-paths.d.ts +0 -2
- package/dist/opencode/opencode-paths.d.ts.map +0 -1
- package/dist/opencode/opencode-paths.js +0 -7
- package/dist/opencode/opencode-paths.js.map +0 -1
- package/dist/opencode/skill-loader.d.ts +0 -15
- package/dist/opencode/skill-loader.d.ts.map +0 -1
- package/dist/opencode/skill-loader.js +0 -88
- package/dist/opencode/skill-loader.js.map +0 -1
- /package/{.opencode/agent → .pi/agents}/describe/pr-describer.md +0 -0
- /package/{.opencode/agent → .pi/agents}/review/documentation.md +0 -0
- /package/{.opencode/agent → .pi/agents}/review/performance.md +0 -0
- /package/{.opencode/agent → .pi/agents}/review/quality.md +0 -0
- /package/{.opencode/agent → .pi/agents}/review/security.md +0 -0
- /package/{.opencode/agent → .pi/agents}/review/style.md +0 -0
- /package/{.opencode/agent → .pi/agents}/review/unified-reviewer.md +0 -0
- /package/dist/{opencode → runtime}/client.test.d.ts +0 -0
|
@@ -0,0 +1,392 @@
|
|
|
1
|
+
import { beforeEach, describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { RuntimeClient, createRuntimeClient, createRuntimeClientInstance } from './client.js';
|
|
3
|
+
function createRuntime(sessionOverrides = {}) {
|
|
4
|
+
return {
|
|
5
|
+
server: {
|
|
6
|
+
url: 'pi://in-process',
|
|
7
|
+
close: vi.fn(),
|
|
8
|
+
},
|
|
9
|
+
client: {
|
|
10
|
+
session: {
|
|
11
|
+
create: vi.fn(async () => ({ data: { id: 'session-123' } })),
|
|
12
|
+
prompt: vi.fn(async () => { }),
|
|
13
|
+
messages: vi.fn(async () => ({ data: [] })),
|
|
14
|
+
delete: vi.fn(async () => { }),
|
|
15
|
+
...sessionOverrides,
|
|
16
|
+
},
|
|
17
|
+
},
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
const mocks = vi.hoisted(() => ({
|
|
21
|
+
createPiInProcessServer: vi.fn(),
|
|
22
|
+
loadReviewAgents: vi.fn(() => []),
|
|
23
|
+
}));
|
|
24
|
+
vi.mock('../pi/sdk.js', () => ({
|
|
25
|
+
createPiInProcessServer: mocks.createPiInProcessServer,
|
|
26
|
+
}));
|
|
27
|
+
vi.mock('./agent-loader.js', () => ({
|
|
28
|
+
loadReviewAgents: mocks.loadReviewAgents,
|
|
29
|
+
}));
|
|
30
|
+
describe('RuntimeClient', () => {
|
|
31
|
+
beforeEach(() => {
|
|
32
|
+
vi.clearAllMocks();
|
|
33
|
+
vi.spyOn(console, 'log').mockImplementation(() => { });
|
|
34
|
+
vi.spyOn(console, 'warn').mockImplementation(() => { });
|
|
35
|
+
mocks.createPiInProcessServer.mockResolvedValue(createRuntime());
|
|
36
|
+
mocks.loadReviewAgents.mockReturnValue([]);
|
|
37
|
+
});
|
|
38
|
+
describe('constructor', () => {
|
|
39
|
+
it('creates an instance with minimal config', () => {
|
|
40
|
+
const client = new RuntimeClient({
|
|
41
|
+
directory: '/test/dir',
|
|
42
|
+
});
|
|
43
|
+
expect(client).toBeInstanceOf(RuntimeClient);
|
|
44
|
+
});
|
|
45
|
+
it('supports optional model overrides and provider config', () => {
|
|
46
|
+
const client = new RuntimeClient({
|
|
47
|
+
modelOverrides: {
|
|
48
|
+
'review/security': 'anthropic/claude-opus-4-5-20251101',
|
|
49
|
+
},
|
|
50
|
+
provider: {
|
|
51
|
+
custom: {
|
|
52
|
+
npm: '@custom/provider',
|
|
53
|
+
name: 'custom',
|
|
54
|
+
models: { model: { name: 'model' } },
|
|
55
|
+
options: {
|
|
56
|
+
baseURL: 'https://api.custom.example',
|
|
57
|
+
apiKey: '{env:CUSTOM_API_KEY}',
|
|
58
|
+
},
|
|
59
|
+
},
|
|
60
|
+
},
|
|
61
|
+
});
|
|
62
|
+
expect(client).toBeInstanceOf(RuntimeClient);
|
|
63
|
+
});
|
|
64
|
+
});
|
|
65
|
+
describe('initialize', () => {
|
|
66
|
+
it('wires Pi runtime agent prompts and model overrides', async () => {
|
|
67
|
+
mocks.loadReviewAgents.mockReturnValue([
|
|
68
|
+
{
|
|
69
|
+
name: 'review/security',
|
|
70
|
+
path: '/tmp/security.md',
|
|
71
|
+
description: 'Security specialist',
|
|
72
|
+
prompt: 'Security prompt',
|
|
73
|
+
tools: { Read: true },
|
|
74
|
+
},
|
|
75
|
+
{
|
|
76
|
+
name: 'review/quality',
|
|
77
|
+
path: '/tmp/quality.md',
|
|
78
|
+
description: 'Quality specialist',
|
|
79
|
+
prompt: 'Quality prompt',
|
|
80
|
+
},
|
|
81
|
+
]);
|
|
82
|
+
const client = await createRuntimeClientInstance({
|
|
83
|
+
directory: process.cwd(),
|
|
84
|
+
config: {
|
|
85
|
+
review: {
|
|
86
|
+
agents: ['security', 'quality'],
|
|
87
|
+
default: {
|
|
88
|
+
skills: [],
|
|
89
|
+
},
|
|
90
|
+
},
|
|
91
|
+
},
|
|
92
|
+
modelOverrides: {
|
|
93
|
+
'review/security': 'anthropic/claude-security',
|
|
94
|
+
},
|
|
95
|
+
});
|
|
96
|
+
expect(mocks.createPiInProcessServer).toHaveBeenCalledWith(expect.objectContaining({
|
|
97
|
+
config: expect.objectContaining({
|
|
98
|
+
agent: expect.objectContaining({
|
|
99
|
+
'review/security': expect.objectContaining({
|
|
100
|
+
prompt: 'Security prompt',
|
|
101
|
+
model: 'anthropic/claude-security',
|
|
102
|
+
tools: { Read: true },
|
|
103
|
+
}),
|
|
104
|
+
'review/quality': expect.objectContaining({
|
|
105
|
+
prompt: 'Quality prompt',
|
|
106
|
+
}),
|
|
107
|
+
}),
|
|
108
|
+
}),
|
|
109
|
+
}));
|
|
110
|
+
await client.shutdown();
|
|
111
|
+
});
|
|
112
|
+
it('passes skill search paths and agent skill configuration to Pi runtime', async () => {
|
|
113
|
+
const projectRoot = process.cwd();
|
|
114
|
+
const config = {
|
|
115
|
+
review: {
|
|
116
|
+
agents: [
|
|
117
|
+
{
|
|
118
|
+
name: 'security',
|
|
119
|
+
skills: ['security-audit'],
|
|
120
|
+
},
|
|
121
|
+
],
|
|
122
|
+
default: {
|
|
123
|
+
skills: ['baseline-review'],
|
|
124
|
+
},
|
|
125
|
+
},
|
|
126
|
+
};
|
|
127
|
+
const client = await createRuntimeClientInstance({
|
|
128
|
+
directory: projectRoot,
|
|
129
|
+
config,
|
|
130
|
+
});
|
|
131
|
+
expect(mocks.createPiInProcessServer).toHaveBeenCalledWith(expect.objectContaining({
|
|
132
|
+
config: expect.objectContaining({
|
|
133
|
+
skillSearchPaths: expect.any(Array),
|
|
134
|
+
agentSkills: {
|
|
135
|
+
'review/security': ['baseline-review', 'security-audit'],
|
|
136
|
+
},
|
|
137
|
+
}),
|
|
138
|
+
}));
|
|
139
|
+
await client.shutdown();
|
|
140
|
+
});
|
|
141
|
+
it('passes per-agent tool overrides to Pi runtime config', async () => {
|
|
142
|
+
mocks.loadReviewAgents.mockReturnValue([
|
|
143
|
+
{
|
|
144
|
+
name: 'review/security',
|
|
145
|
+
path: '/tmp/security.md',
|
|
146
|
+
description: 'Security agent',
|
|
147
|
+
prompt: 'Security prompt',
|
|
148
|
+
tools: { Read: true, Bash: false, Edit: true },
|
|
149
|
+
},
|
|
150
|
+
{
|
|
151
|
+
name: 'review/quality',
|
|
152
|
+
path: '/tmp/quality.md',
|
|
153
|
+
description: 'Quality agent',
|
|
154
|
+
prompt: 'Quality prompt',
|
|
155
|
+
// No tools override — uses global defaults
|
|
156
|
+
},
|
|
157
|
+
]);
|
|
158
|
+
const client = await createRuntimeClientInstance({
|
|
159
|
+
directory: process.cwd(),
|
|
160
|
+
config: {
|
|
161
|
+
review: {
|
|
162
|
+
agents: ['security', 'quality'],
|
|
163
|
+
default: { skills: [] },
|
|
164
|
+
},
|
|
165
|
+
},
|
|
166
|
+
});
|
|
167
|
+
expect(mocks.createPiInProcessServer).toHaveBeenCalledWith(expect.objectContaining({
|
|
168
|
+
config: expect.objectContaining({
|
|
169
|
+
agent: expect.objectContaining({
|
|
170
|
+
'review/security': expect.objectContaining({
|
|
171
|
+
tools: { Read: true, Bash: false, Edit: true },
|
|
172
|
+
}),
|
|
173
|
+
'review/quality': expect.not.objectContaining({
|
|
174
|
+
tools: expect.anything(),
|
|
175
|
+
}),
|
|
176
|
+
}),
|
|
177
|
+
}),
|
|
178
|
+
}));
|
|
179
|
+
await client.shutdown();
|
|
180
|
+
});
|
|
181
|
+
});
|
|
182
|
+
describe('createSession', () => {
|
|
183
|
+
it('throws error if not initialized', async () => {
|
|
184
|
+
const client = new RuntimeClient({});
|
|
185
|
+
await expect(client.createSession({
|
|
186
|
+
agent: 'review/security',
|
|
187
|
+
message: 'Review this code',
|
|
188
|
+
})).rejects.toThrow('Runtime client not initialized');
|
|
189
|
+
});
|
|
190
|
+
it('maps authentication errors to actionable messages', async () => {
|
|
191
|
+
mocks.createPiInProcessServer.mockResolvedValueOnce(createRuntime({
|
|
192
|
+
create: vi.fn(async () => {
|
|
193
|
+
throw new Error('401 Unauthorized');
|
|
194
|
+
}),
|
|
195
|
+
}));
|
|
196
|
+
const client = await createRuntimeClientInstance({
|
|
197
|
+
directory: process.cwd(),
|
|
198
|
+
});
|
|
199
|
+
await expect(client.createSession({
|
|
200
|
+
agent: 'review/security',
|
|
201
|
+
message: 'Review this code',
|
|
202
|
+
})).rejects.toThrow('Authentication failed with the configured model provider');
|
|
203
|
+
});
|
|
204
|
+
it('maps model resolution errors to actionable messages', async () => {
|
|
205
|
+
mocks.createPiInProcessServer.mockResolvedValueOnce(createRuntime({
|
|
206
|
+
create: vi.fn(async () => {
|
|
207
|
+
throw new Error('Failed to resolve model "anthropic/does-not-exist"');
|
|
208
|
+
}),
|
|
209
|
+
}));
|
|
210
|
+
const client = await createRuntimeClientInstance({
|
|
211
|
+
directory: process.cwd(),
|
|
212
|
+
});
|
|
213
|
+
await expect(client.createSession({
|
|
214
|
+
agent: 'review/security',
|
|
215
|
+
message: 'Review this code',
|
|
216
|
+
})).rejects.toThrow('Model configuration is invalid or unavailable');
|
|
217
|
+
});
|
|
218
|
+
it('includes local-runtime hint when connectivity errors occur', async () => {
|
|
219
|
+
mocks.createPiInProcessServer.mockResolvedValueOnce(createRuntime({
|
|
220
|
+
create: vi.fn(async () => {
|
|
221
|
+
throw new Error('fetch failed');
|
|
222
|
+
}),
|
|
223
|
+
}));
|
|
224
|
+
const client = await createRuntimeClientInstance({
|
|
225
|
+
directory: process.cwd(),
|
|
226
|
+
});
|
|
227
|
+
await expect(client.createSession({
|
|
228
|
+
agent: 'review/security',
|
|
229
|
+
message: 'Review this code',
|
|
230
|
+
})).rejects.toThrow('Verify local Pi runtime setup and model provider connectivity');
|
|
231
|
+
});
|
|
232
|
+
});
|
|
233
|
+
describe('lifecycle and helper methods', () => {
|
|
234
|
+
it('streamMessages throws if not initialized', async () => {
|
|
235
|
+
const client = new RuntimeClient({});
|
|
236
|
+
const generator = client.streamMessages('session-123');
|
|
237
|
+
await expect(generator.next()).rejects.toThrow('Runtime client not initialized');
|
|
238
|
+
});
|
|
239
|
+
it('applies configured model pricing when runtime cost is missing or zero', async () => {
|
|
240
|
+
const runtimeMessages = [
|
|
241
|
+
{
|
|
242
|
+
info: {
|
|
243
|
+
id: 'msg-1',
|
|
244
|
+
role: 'assistant',
|
|
245
|
+
time: { completed: Date.now() },
|
|
246
|
+
provider: 'opencode',
|
|
247
|
+
model: 'glm-5-free',
|
|
248
|
+
usage: {
|
|
249
|
+
input: 1000,
|
|
250
|
+
output: 100,
|
|
251
|
+
cacheRead: 0,
|
|
252
|
+
cacheWrite: 0,
|
|
253
|
+
totalTokens: 1100,
|
|
254
|
+
},
|
|
255
|
+
},
|
|
256
|
+
parts: [{ text: 'done' }],
|
|
257
|
+
},
|
|
258
|
+
];
|
|
259
|
+
mocks.createPiInProcessServer.mockResolvedValueOnce(createRuntime({
|
|
260
|
+
messages: vi.fn(async () => ({ data: runtimeMessages })),
|
|
261
|
+
}));
|
|
262
|
+
const client = await createRuntimeClientInstance({
|
|
263
|
+
directory: process.cwd(),
|
|
264
|
+
config: {
|
|
265
|
+
review: {
|
|
266
|
+
agents: [],
|
|
267
|
+
default: {
|
|
268
|
+
skills: [],
|
|
269
|
+
},
|
|
270
|
+
},
|
|
271
|
+
pricing: {
|
|
272
|
+
models: {
|
|
273
|
+
'opencode/glm-5-free': {
|
|
274
|
+
input: 2,
|
|
275
|
+
output: 8,
|
|
276
|
+
},
|
|
277
|
+
},
|
|
278
|
+
},
|
|
279
|
+
},
|
|
280
|
+
});
|
|
281
|
+
const collected = [];
|
|
282
|
+
for await (const message of client.streamMessages('session-123')) {
|
|
283
|
+
collected.push(message);
|
|
284
|
+
}
|
|
285
|
+
expect(collected).toHaveLength(1);
|
|
286
|
+
expect(collected[0].usage?.cost).toBeCloseTo(0.0028, 10);
|
|
287
|
+
await client.shutdown();
|
|
288
|
+
});
|
|
289
|
+
it('closeSession throws if not initialized', async () => {
|
|
290
|
+
const client = new RuntimeClient({});
|
|
291
|
+
await expect(client.closeSession('session-123')).rejects.toThrow('Runtime client not initialized');
|
|
292
|
+
});
|
|
293
|
+
it('getServerUrl throws when server is not initialized', () => {
|
|
294
|
+
const client = new RuntimeClient({});
|
|
295
|
+
expect(() => client.getServerUrl()).toThrow('Server not initialized');
|
|
296
|
+
});
|
|
297
|
+
it('shutdown does not throw when no runtime is active', async () => {
|
|
298
|
+
const client = new RuntimeClient({});
|
|
299
|
+
await expect(client.shutdown()).resolves.toBeUndefined();
|
|
300
|
+
});
|
|
301
|
+
});
|
|
302
|
+
describe('environment variable resolution', () => {
|
|
303
|
+
it('resolves provider apiKey env references before runtime init', async () => {
|
|
304
|
+
const originalEnv = process.env.TEST_API_KEY;
|
|
305
|
+
process.env.TEST_API_KEY = 'test-key-123';
|
|
306
|
+
try {
|
|
307
|
+
const client = await createRuntimeClientInstance({
|
|
308
|
+
directory: process.cwd(),
|
|
309
|
+
provider: {
|
|
310
|
+
'test-provider': {
|
|
311
|
+
npm: '@test/provider',
|
|
312
|
+
name: 'test-provider',
|
|
313
|
+
models: { 'test-model': { name: 'test-model' } },
|
|
314
|
+
options: {
|
|
315
|
+
baseURL: 'https://api.test.com',
|
|
316
|
+
apiKey: '{env:TEST_API_KEY}',
|
|
317
|
+
},
|
|
318
|
+
},
|
|
319
|
+
},
|
|
320
|
+
});
|
|
321
|
+
expect(mocks.createPiInProcessServer).toHaveBeenCalledWith(expect.objectContaining({
|
|
322
|
+
config: expect.objectContaining({
|
|
323
|
+
provider: expect.objectContaining({
|
|
324
|
+
'test-provider': expect.objectContaining({
|
|
325
|
+
options: expect.objectContaining({
|
|
326
|
+
apiKey: 'test-key-123',
|
|
327
|
+
}),
|
|
328
|
+
}),
|
|
329
|
+
}),
|
|
330
|
+
}),
|
|
331
|
+
}));
|
|
332
|
+
await client.shutdown();
|
|
333
|
+
}
|
|
334
|
+
finally {
|
|
335
|
+
if (originalEnv === undefined) {
|
|
336
|
+
delete process.env.TEST_API_KEY;
|
|
337
|
+
}
|
|
338
|
+
else {
|
|
339
|
+
process.env.TEST_API_KEY = originalEnv;
|
|
340
|
+
}
|
|
341
|
+
}
|
|
342
|
+
});
|
|
343
|
+
it('warns when referenced environment variables are missing', async () => {
|
|
344
|
+
const logSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
|
|
345
|
+
const originalEnv = process.env.NONEXISTENT_VAR;
|
|
346
|
+
delete process.env.NONEXISTENT_VAR;
|
|
347
|
+
try {
|
|
348
|
+
const client = await createRuntimeClientInstance({
|
|
349
|
+
directory: process.cwd(),
|
|
350
|
+
provider: {
|
|
351
|
+
'test-provider': {
|
|
352
|
+
npm: '@test/provider',
|
|
353
|
+
name: 'test-provider',
|
|
354
|
+
models: { 'test-model': { name: 'test-model' } },
|
|
355
|
+
options: {
|
|
356
|
+
baseURL: 'https://api.test.com',
|
|
357
|
+
apiKey: '{env:NONEXISTENT_VAR}',
|
|
358
|
+
},
|
|
359
|
+
},
|
|
360
|
+
},
|
|
361
|
+
});
|
|
362
|
+
// Logger outputs warning via console.log (human format)
|
|
363
|
+
const allOutput = logSpy.mock.calls.map((c) => String(c[0])).join('\n');
|
|
364
|
+
expect(allOutput).toContain('NONEXISTENT_VAR is not set');
|
|
365
|
+
await client.shutdown();
|
|
366
|
+
}
|
|
367
|
+
finally {
|
|
368
|
+
logSpy.mockRestore();
|
|
369
|
+
if (originalEnv !== undefined) {
|
|
370
|
+
process.env.NONEXISTENT_VAR = originalEnv;
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
});
|
|
374
|
+
});
|
|
375
|
+
describe('factory functions', () => {
|
|
376
|
+
it('createRuntimeClient returns an uninitialized client instance', () => {
|
|
377
|
+
const client = createRuntimeClient({
|
|
378
|
+
directory: process.cwd(),
|
|
379
|
+
});
|
|
380
|
+
expect(client).toBeInstanceOf(RuntimeClient);
|
|
381
|
+
});
|
|
382
|
+
it('createRuntimeClientInstance initializes in-process runtime', async () => {
|
|
383
|
+
const client = await createRuntimeClientInstance({
|
|
384
|
+
directory: process.cwd(),
|
|
385
|
+
});
|
|
386
|
+
expect(client).toBeInstanceOf(RuntimeClient);
|
|
387
|
+
expect(mocks.createPiInProcessServer).toHaveBeenCalled();
|
|
388
|
+
await client.shutdown();
|
|
389
|
+
});
|
|
390
|
+
});
|
|
391
|
+
});
|
|
392
|
+
//# sourceMappingURL=client.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"client.test.js","sourceRoot":"","sources":["../../src/runtime/client.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import type { DRSConfig } from '../lib/config.js';
|
|
2
|
+
export interface ResolvedReviewPaths {
|
|
3
|
+
agentsPath: string;
|
|
4
|
+
skillsPath: string;
|
|
5
|
+
skillSearchPaths: string[];
|
|
6
|
+
}
|
|
7
|
+
export declare function resolveReviewPaths(projectPath: string, config?: DRSConfig): ResolvedReviewPaths;
|
|
8
|
+
//# sourceMappingURL=path-config.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"path-config.d.ts","sourceRoot":"","sources":["../../src/runtime/path-config.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AAQlD,MAAM,WAAW,mBAAmB;IAClC,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,MAAM,CAAC;IACnB,gBAAgB,EAAE,MAAM,EAAE,CAAC;CAC5B;AAoFD,wBAAgB,kBAAkB,CAAC,WAAW,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,SAAS,GAAG,mBAAmB,CAc/F"}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { existsSync, statSync } from 'fs';
|
|
2
|
+
import { isAbsolute, relative, resolve } from 'path';
|
|
3
|
+
const DEFAULT_AGENT_PATH = '.drs/agents';
|
|
4
|
+
const DEFAULT_SKILL_PATH = '.drs/skills';
|
|
5
|
+
const PI_DEFAULT_SKILL_PATH = '.pi/skills';
|
|
6
|
+
function isOutsideProjectRoot(projectRoot, targetPath) {
|
|
7
|
+
const relativePath = relative(projectRoot, targetPath);
|
|
8
|
+
const firstSegment = relativePath.split(/[\\/]/).filter(Boolean)[0];
|
|
9
|
+
return firstSegment === '..';
|
|
10
|
+
}
|
|
11
|
+
function resolveConfiguredPath(projectRoot, configuredPath, fallbackPath, pathType) {
|
|
12
|
+
if (configuredPath === undefined || configuredPath === null) {
|
|
13
|
+
return resolve(projectRoot, fallbackPath);
|
|
14
|
+
}
|
|
15
|
+
if (typeof configuredPath !== 'string') {
|
|
16
|
+
throw new Error(`Invalid review.paths.${pathType}: expected a string path. Use a repo-relative path like "${fallbackPath}" or an absolute path.`);
|
|
17
|
+
}
|
|
18
|
+
const trimmed = configuredPath.trim();
|
|
19
|
+
if (!trimmed) {
|
|
20
|
+
throw new Error(`Invalid review.paths.${pathType}: path cannot be empty. Use a repo-relative path like "${fallbackPath}" or an absolute path.`);
|
|
21
|
+
}
|
|
22
|
+
const resolvedPath = isAbsolute(trimmed) ? resolve(trimmed) : resolve(projectRoot, trimmed);
|
|
23
|
+
if (!isAbsolute(trimmed) && isOutsideProjectRoot(projectRoot, resolvedPath)) {
|
|
24
|
+
throw new Error(`Invalid review.paths.${pathType}: "${configuredPath}" resolves outside repository root (${projectRoot}). Use a repo-relative path within the repository or an absolute path.`);
|
|
25
|
+
}
|
|
26
|
+
if (!existsSync(resolvedPath)) {
|
|
27
|
+
throw new Error(`Invalid review.paths.${pathType}: "${configuredPath}" resolved to "${resolvedPath}", but the directory does not exist. Create the directory or remove review.paths.${pathType} to use "${fallbackPath}".`);
|
|
28
|
+
}
|
|
29
|
+
if (!statSync(resolvedPath).isDirectory()) {
|
|
30
|
+
throw new Error(`Invalid review.paths.${pathType}: "${configuredPath}" resolved to "${resolvedPath}", but it is not a directory. Point review.paths.${pathType} to a directory or remove it to use "${fallbackPath}".`);
|
|
31
|
+
}
|
|
32
|
+
return resolvedPath;
|
|
33
|
+
}
|
|
34
|
+
function resolveDefaultSkillSearchPaths(projectRoot) {
|
|
35
|
+
const candidates = [
|
|
36
|
+
resolve(projectRoot, DEFAULT_SKILL_PATH),
|
|
37
|
+
resolve(projectRoot, PI_DEFAULT_SKILL_PATH),
|
|
38
|
+
];
|
|
39
|
+
const existing = candidates.filter((candidate) => {
|
|
40
|
+
if (!existsSync(candidate)) {
|
|
41
|
+
return false;
|
|
42
|
+
}
|
|
43
|
+
try {
|
|
44
|
+
return statSync(candidate).isDirectory();
|
|
45
|
+
}
|
|
46
|
+
catch {
|
|
47
|
+
return false;
|
|
48
|
+
}
|
|
49
|
+
});
|
|
50
|
+
return existing.length > 0 ? existing : [candidates[0]];
|
|
51
|
+
}
|
|
52
|
+
function resolveSkillSearchPaths(projectRoot, config) {
|
|
53
|
+
const configuredSkillsPath = config?.review?.paths?.skills;
|
|
54
|
+
if (configuredSkillsPath === undefined || configuredSkillsPath === null) {
|
|
55
|
+
return resolveDefaultSkillSearchPaths(projectRoot);
|
|
56
|
+
}
|
|
57
|
+
return [resolveConfiguredPath(projectRoot, configuredSkillsPath, DEFAULT_SKILL_PATH, 'skills')];
|
|
58
|
+
}
|
|
59
|
+
export function resolveReviewPaths(projectPath, config) {
|
|
60
|
+
const projectRoot = resolve(projectPath);
|
|
61
|
+
const skillSearchPaths = resolveSkillSearchPaths(projectRoot, config);
|
|
62
|
+
return {
|
|
63
|
+
agentsPath: resolveConfiguredPath(projectRoot, config?.review?.paths?.agents, DEFAULT_AGENT_PATH, 'agents'),
|
|
64
|
+
skillsPath: skillSearchPaths[0],
|
|
65
|
+
skillSearchPaths,
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
//# sourceMappingURL=path-config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"path-config.js","sourceRoot":"","sources":["../../src/runtime/path-config.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,IAAI,CAAC;AAC1C,OAAO,EAAE,UAAU,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAGrD,MAAM,kBAAkB,GAAG,aAAa,CAAC;AACzC,MAAM,kBAAkB,GAAG,aAAa,CAAC;AACzC,MAAM,qBAAqB,GAAG,YAAY,CAAC;AAU3C,SAAS,oBAAoB,CAAC,WAAmB,EAAE,UAAkB;IACnE,MAAM,YAAY,GAAG,QAAQ,CAAC,WAAW,EAAE,UAAU,CAAC,CAAC;IACvD,MAAM,YAAY,GAAG,YAAY,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;IACpE,OAAO,YAAY,KAAK,IAAI,CAAC;AAC/B,CAAC;AAED,SAAS,qBAAqB,CAC5B,WAAmB,EACnB,cAAuB,EACvB,YAAoB,EACpB,QAAwB;IAExB,IAAI,cAAc,KAAK,SAAS,IAAI,cAAc,KAAK,IAAI,EAAE,CAAC;QAC5D,OAAO,OAAO,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;IAC5C,CAAC;IAED,IAAI,OAAO,cAAc,KAAK,QAAQ,EAAE,CAAC;QACvC,MAAM,IAAI,KAAK,CACb,wBAAwB,QAAQ,4DAA4D,YAAY,wBAAwB,CACjI,CAAC;IACJ,CAAC;IAED,MAAM,OAAO,GAAG,cAAc,CAAC,IAAI,EAAE,CAAC;IACtC,IAAI,CAAC,OAAO,EAAE,CAAC;QACb,MAAM,IAAI,KAAK,CACb,wBAAwB,QAAQ,0DAA0D,YAAY,wBAAwB,CAC/H,CAAC;IACJ,CAAC;IAED,MAAM,YAAY,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;IAE5F,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,oBAAoB,CAAC,WAAW,EAAE,YAAY,CAAC,EAAE,CAAC;QAC5E,MAAM,IAAI,KAAK,CACb,wBAAwB,QAAQ,MAAM,cAAc,uCAAuC,WAAW,wEAAwE,CAC/K,CAAC;IACJ,CAAC;IAED,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,EAAE,CAAC;QAC9B,MAAM,IAAI,KAAK,CACb,wBAAwB,QAAQ,MAAM,cAAc,kBAAkB,YAAY,oFAAoF,QAAQ,YAAY,YAAY,IAAI,CAC3M,CAAC;IACJ,CAAC;IAED,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC,WAAW,EAAE,EAAE,CAAC;QAC1C,MAAM,IAAI,KAAK,CACb,wBAAwB,QAAQ,MAAM,cAAc,kBAAkB,YAAY,oDAAoD,QAAQ,wCAAwC,YAAY,IAAI,CACvM,CAAC;IACJ,CAAC;IAED,OAAO,YAAY,CAAC;AACtB,CAAC;AAED,SAAS,8BAA8B,CAAC,WAAmB;IACzD,MAAM,UAAU,GAAG;QACjB,OAAO,CAAC,WAAW,EAAE,kBAAkB,CAAC;QACxC,OAAO,CAAC,WAAW,EAAE,qBAAqB,CAAC;KAC5C,CAAC;IAEF,MAAM,QAAQ,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,SAAS,EAAE,EAAE;QAC/C,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;YAC3B,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,CAAC;YACH,OAAO,QAAQ,CAAC,SAAS,CAAC,CAAC,WAAW,EAAE,CAAC;QAC3C,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,OAAO,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC;AAC1D,CAAC;AAED,SAAS,uBAAuB,CAAC,WAAmB,EAAE,MAAkB;IACtE,MAAM,oBAAoB,GAAG,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,CAAC;IAC3D,IAAI,oBAAoB,KAAK,SAAS,IAAI,oBAAoB,KAAK,IAAI,EAAE,CAAC;QACxE,OAAO,8BAA8B,CAAC,WAAW,CAAC,CAAC;IACrD,CAAC;IAED,OAAO,CAAC,qBAAqB,CAAC,WAAW,EAAE,oBAAoB,EAAE,kBAAkB,EAAE,QAAQ,CAAC,CAAC,CAAC;AAClG,CAAC;AAED,MAAM,UAAU,kBAAkB,CAAC,WAAmB,EAAE,MAAkB;IACxE,MAAM,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC;IACzC,MAAM,gBAAgB,GAAG,uBAAuB,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;IAEtE,OAAO;QACL,UAAU,EAAE,qBAAqB,CAC/B,WAAW,EACX,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAC7B,kBAAkB,EAClB,QAAQ,CACT;QACD,UAAU,EAAE,gBAAgB,CAAC,CAAC,CAAC;QAC/B,gBAAgB;KACjB,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"path-config.test.d.ts","sourceRoot":"","sources":["../../src/runtime/path-config.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from 'fs';
|
|
2
|
+
import { tmpdir } from 'os';
|
|
3
|
+
import { join, resolve } from 'path';
|
|
4
|
+
import { afterEach, describe, expect, it } from 'vitest';
|
|
5
|
+
import { resolveReviewPaths } from './path-config.js';
|
|
6
|
+
function createConfig(paths) {
|
|
7
|
+
return {
|
|
8
|
+
review: {
|
|
9
|
+
paths: {
|
|
10
|
+
agents: paths.agents,
|
|
11
|
+
skills: paths.skills,
|
|
12
|
+
},
|
|
13
|
+
},
|
|
14
|
+
};
|
|
15
|
+
}
|
|
16
|
+
describe('resolveReviewPaths', () => {
|
|
17
|
+
const tempDirs = [];
|
|
18
|
+
function createTempDir(prefix) {
|
|
19
|
+
const dir = mkdtempSync(join(tmpdir(), prefix));
|
|
20
|
+
tempDirs.push(dir);
|
|
21
|
+
return dir;
|
|
22
|
+
}
|
|
23
|
+
afterEach(() => {
|
|
24
|
+
for (const dir of tempDirs.splice(0, tempDirs.length)) {
|
|
25
|
+
rmSync(dir, { recursive: true, force: true });
|
|
26
|
+
}
|
|
27
|
+
});
|
|
28
|
+
it('resolves default paths from project root deterministically', () => {
|
|
29
|
+
const projectRoot = createTempDir('drs-path-default-');
|
|
30
|
+
const canonical = resolveReviewPaths(projectRoot);
|
|
31
|
+
const nonCanonical = resolveReviewPaths(join(projectRoot, 'nested', '..'));
|
|
32
|
+
expect(canonical.agentsPath).toBe(resolve(projectRoot, '.drs/agents'));
|
|
33
|
+
expect(canonical.skillsPath).toBe(resolve(projectRoot, '.drs/skills'));
|
|
34
|
+
expect(canonical.skillSearchPaths).toEqual([resolve(projectRoot, '.drs/skills')]);
|
|
35
|
+
expect(nonCanonical).toEqual(canonical);
|
|
36
|
+
});
|
|
37
|
+
it('auto-discovers both .drs/skills and .pi/skills when present', () => {
|
|
38
|
+
const projectRoot = createTempDir('drs-path-skill-discovery-');
|
|
39
|
+
mkdirSync(join(projectRoot, '.drs', 'skills'), { recursive: true });
|
|
40
|
+
mkdirSync(join(projectRoot, '.pi', 'skills'), { recursive: true });
|
|
41
|
+
const result = resolveReviewPaths(projectRoot);
|
|
42
|
+
expect(result.skillSearchPaths).toEqual([
|
|
43
|
+
resolve(projectRoot, '.drs/skills'),
|
|
44
|
+
resolve(projectRoot, '.pi/skills'),
|
|
45
|
+
]);
|
|
46
|
+
expect(result.skillsPath).toBe(resolve(projectRoot, '.drs/skills'));
|
|
47
|
+
});
|
|
48
|
+
it('uses .pi/skills when .drs/skills is missing', () => {
|
|
49
|
+
const projectRoot = createTempDir('drs-path-pi-skills-');
|
|
50
|
+
mkdirSync(join(projectRoot, '.pi', 'skills'), { recursive: true });
|
|
51
|
+
const result = resolveReviewPaths(projectRoot);
|
|
52
|
+
expect(result.skillSearchPaths).toEqual([resolve(projectRoot, '.pi/skills')]);
|
|
53
|
+
expect(result.skillsPath).toBe(resolve(projectRoot, '.pi/skills'));
|
|
54
|
+
});
|
|
55
|
+
it('resolves repo-relative configured paths', () => {
|
|
56
|
+
const projectRoot = createTempDir('drs-path-relative-');
|
|
57
|
+
mkdirSync(join(projectRoot, 'config', 'agents'), { recursive: true });
|
|
58
|
+
mkdirSync(join(projectRoot, 'config', 'skills'), { recursive: true });
|
|
59
|
+
const result = resolveReviewPaths(projectRoot, createConfig({
|
|
60
|
+
agents: 'config/agents',
|
|
61
|
+
skills: 'config/skills',
|
|
62
|
+
}));
|
|
63
|
+
expect(result.agentsPath).toBe(resolve(projectRoot, 'config/agents'));
|
|
64
|
+
expect(result.skillsPath).toBe(resolve(projectRoot, 'config/skills'));
|
|
65
|
+
expect(result.skillSearchPaths).toEqual([resolve(projectRoot, 'config/skills')]);
|
|
66
|
+
});
|
|
67
|
+
it('resolves absolute configured paths', () => {
|
|
68
|
+
const projectRoot = createTempDir('drs-path-project-');
|
|
69
|
+
const absoluteRoot = createTempDir('drs-path-absolute-');
|
|
70
|
+
const agentsPath = join(absoluteRoot, 'agents');
|
|
71
|
+
const skillsPath = join(absoluteRoot, 'skills');
|
|
72
|
+
mkdirSync(agentsPath, { recursive: true });
|
|
73
|
+
mkdirSync(skillsPath, { recursive: true });
|
|
74
|
+
const result = resolveReviewPaths(projectRoot, createConfig({
|
|
75
|
+
agents: agentsPath,
|
|
76
|
+
skills: skillsPath,
|
|
77
|
+
}));
|
|
78
|
+
expect(result.agentsPath).toBe(resolve(agentsPath));
|
|
79
|
+
expect(result.skillsPath).toBe(resolve(skillsPath));
|
|
80
|
+
expect(result.skillSearchPaths).toEqual([resolve(skillsPath)]);
|
|
81
|
+
});
|
|
82
|
+
it('throws actionable error when repo-relative path escapes project root', () => {
|
|
83
|
+
const projectRoot = createTempDir('drs-path-escape-');
|
|
84
|
+
expect(() => resolveReviewPaths(projectRoot, createConfig({ agents: '../shared-agents' }))).toThrow('resolves outside repository root');
|
|
85
|
+
});
|
|
86
|
+
it('throws actionable error when configured path does not exist', () => {
|
|
87
|
+
const projectRoot = createTempDir('drs-path-missing-');
|
|
88
|
+
expect(() => resolveReviewPaths(projectRoot, createConfig({ skills: 'missing/skills' }))).toThrow('directory does not exist');
|
|
89
|
+
});
|
|
90
|
+
it('throws actionable error when configured path is not a directory', () => {
|
|
91
|
+
const projectRoot = createTempDir('drs-path-file-');
|
|
92
|
+
const filePath = join(projectRoot, 'agents-file.md');
|
|
93
|
+
writeFileSync(filePath, '# not a directory\n');
|
|
94
|
+
expect(() => resolveReviewPaths(projectRoot, createConfig({ agents: filePath }))).toThrow('not a directory');
|
|
95
|
+
});
|
|
96
|
+
it('throws actionable error when configured path is not a string', () => {
|
|
97
|
+
const projectRoot = createTempDir('drs-path-invalid-type-');
|
|
98
|
+
expect(() => resolveReviewPaths(projectRoot, createConfig({
|
|
99
|
+
skills: 123,
|
|
100
|
+
}))).toThrow('expected a string path');
|
|
101
|
+
});
|
|
102
|
+
});
|
|
103
|
+
//# sourceMappingURL=path-config.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"path-config.test.js","sourceRoot":"","sources":["../../src/runtime/path-config.test.ts"],"names":[],"mappings":"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"}
|