@gotgenes/pi-permission-system 4.2.0 → 4.4.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/CHANGELOG.md +36 -0
- package/README.md +18 -9
- package/package.json +1 -1
- package/src/forwarded-permissions/polling.ts +7 -1
- package/src/handlers/tool-call.ts +36 -0
- package/src/handlers/types.ts +2 -0
- package/src/index.ts +10 -3
- package/src/pattern-suggest.ts +91 -0
- package/src/permission-dialog.ts +16 -2
- package/src/permission-gate.ts +11 -1
- package/src/permission-manager.ts +59 -0
- package/src/permission-prompter.ts +146 -0
- package/src/runtime.ts +0 -75
- package/tests/handlers/tool-call.test.ts +212 -0
- package/tests/pattern-suggest.test.ts +139 -0
- package/tests/permission-dialog.test.ts +39 -0
- package/tests/permission-gate.test.ts +68 -0
- package/tests/permission-prompter.test.ts +398 -0
- package/tests/permission-system.test.ts +181 -0
- package/tests/runtime.test.ts +0 -3
|
@@ -0,0 +1,398 @@
|
|
|
1
|
+
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
2
|
+
|
|
3
|
+
// ── Module mocks ────────────────────────────────────────────────────────────
|
|
4
|
+
|
|
5
|
+
const { mockConfirmPermission } = vi.hoisted(() => ({
|
|
6
|
+
mockConfirmPermission: vi.fn(),
|
|
7
|
+
}));
|
|
8
|
+
|
|
9
|
+
vi.mock("../src/forwarded-permissions/polling", () => ({
|
|
10
|
+
confirmPermission: mockConfirmPermission,
|
|
11
|
+
processForwardedPermissionRequests: vi.fn().mockResolvedValue(undefined),
|
|
12
|
+
}));
|
|
13
|
+
|
|
14
|
+
// ── Imports (after mocks) ───────────────────────────────────────────────────
|
|
15
|
+
|
|
16
|
+
import type { ExtensionContext } from "@mariozechner/pi-coding-agent";
|
|
17
|
+
import { DEFAULT_EXTENSION_CONFIG } from "../src/extension-config";
|
|
18
|
+
import type { PromptPermissionDetails } from "../src/handlers/types";
|
|
19
|
+
import type { PermissionPromptDecision } from "../src/permission-dialog";
|
|
20
|
+
import {
|
|
21
|
+
PermissionPrompter,
|
|
22
|
+
type PermissionPrompterDeps,
|
|
23
|
+
} from "../src/permission-prompter";
|
|
24
|
+
|
|
25
|
+
// ── Helpers ─────────────────────────────────────────────────────────────────
|
|
26
|
+
|
|
27
|
+
function makeCtx(hasUI: boolean): ExtensionContext {
|
|
28
|
+
return {
|
|
29
|
+
hasUI,
|
|
30
|
+
ui: { select: vi.fn(), input: vi.fn() },
|
|
31
|
+
sessionManager: { getSessionDir: vi.fn().mockReturnValue(null) },
|
|
32
|
+
} as unknown as ExtensionContext;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
function makeDetails(
|
|
36
|
+
overrides?: Partial<PromptPermissionDetails>,
|
|
37
|
+
): PromptPermissionDetails {
|
|
38
|
+
return {
|
|
39
|
+
requestId: "req-123",
|
|
40
|
+
source: "tool_call",
|
|
41
|
+
agentName: "test-agent",
|
|
42
|
+
message: "Allow read?",
|
|
43
|
+
toolName: "read",
|
|
44
|
+
...overrides,
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
function makeDeps(
|
|
49
|
+
overrides?: Partial<PermissionPrompterDeps>,
|
|
50
|
+
): PermissionPrompterDeps {
|
|
51
|
+
return {
|
|
52
|
+
getConfig: () => ({ ...DEFAULT_EXTENSION_CONFIG, yoloMode: false }),
|
|
53
|
+
writeReviewLog: vi.fn(),
|
|
54
|
+
subagentSessionsDir: "/sessions/subagents",
|
|
55
|
+
forwardingDir: "/sessions/permission-forwarding",
|
|
56
|
+
requestPermissionDecisionFromUi: vi
|
|
57
|
+
.fn()
|
|
58
|
+
.mockResolvedValue({ approved: true, state: "approved" }),
|
|
59
|
+
...overrides,
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// ── Tests ────────────────────────────────────────────────────────────────────
|
|
64
|
+
|
|
65
|
+
describe("PermissionPrompter", () => {
|
|
66
|
+
beforeEach(() => {
|
|
67
|
+
mockConfirmPermission.mockReset();
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
// ── Yolo-mode auto-approve ───────────────────────────────────────────────
|
|
71
|
+
|
|
72
|
+
describe("yolo-mode auto-approve", () => {
|
|
73
|
+
it("returns approved without calling confirmPermission when yoloMode is true", async () => {
|
|
74
|
+
const deps = makeDeps({
|
|
75
|
+
getConfig: () => ({ ...DEFAULT_EXTENSION_CONFIG, yoloMode: true }),
|
|
76
|
+
});
|
|
77
|
+
const prompter = new PermissionPrompter(deps);
|
|
78
|
+
|
|
79
|
+
const decision = await prompter.prompt(makeCtx(false), makeDetails());
|
|
80
|
+
|
|
81
|
+
expect(decision).toEqual({ approved: true, state: "approved" });
|
|
82
|
+
expect(mockConfirmPermission).not.toHaveBeenCalled();
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
it("logs permission_request.auto_approved in yolo mode", async () => {
|
|
86
|
+
const writeReviewLog = vi.fn();
|
|
87
|
+
const deps = makeDeps({
|
|
88
|
+
getConfig: () => ({ ...DEFAULT_EXTENSION_CONFIG, yoloMode: true }),
|
|
89
|
+
writeReviewLog,
|
|
90
|
+
});
|
|
91
|
+
const prompter = new PermissionPrompter(deps);
|
|
92
|
+
|
|
93
|
+
await prompter.prompt(makeCtx(false), makeDetails());
|
|
94
|
+
|
|
95
|
+
expect(writeReviewLog).toHaveBeenCalledWith(
|
|
96
|
+
"permission_request.auto_approved",
|
|
97
|
+
expect.objectContaining({ requestId: "req-123" }),
|
|
98
|
+
);
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
it("does not log permission_request.waiting in yolo mode", async () => {
|
|
102
|
+
const writeReviewLog = vi.fn();
|
|
103
|
+
const deps = makeDeps({
|
|
104
|
+
getConfig: () => ({ ...DEFAULT_EXTENSION_CONFIG, yoloMode: true }),
|
|
105
|
+
writeReviewLog,
|
|
106
|
+
});
|
|
107
|
+
const prompter = new PermissionPrompter(deps);
|
|
108
|
+
|
|
109
|
+
await prompter.prompt(makeCtx(false), makeDetails());
|
|
110
|
+
|
|
111
|
+
expect(writeReviewLog).not.toHaveBeenCalledWith(
|
|
112
|
+
"permission_request.waiting",
|
|
113
|
+
expect.anything(),
|
|
114
|
+
);
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
it("does not call confirmPermission with yoloMode even when ctx has UI", async () => {
|
|
118
|
+
const deps = makeDeps({
|
|
119
|
+
getConfig: () => ({ ...DEFAULT_EXTENSION_CONFIG, yoloMode: true }),
|
|
120
|
+
});
|
|
121
|
+
const prompter = new PermissionPrompter(deps);
|
|
122
|
+
|
|
123
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
124
|
+
|
|
125
|
+
expect(mockConfirmPermission).not.toHaveBeenCalled();
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
// ── Non-yolo path ────────────────────────────────────────────────────────
|
|
130
|
+
|
|
131
|
+
describe("non-yolo path (UI present)", () => {
|
|
132
|
+
it("logs permission_request.waiting before calling confirmPermission", async () => {
|
|
133
|
+
const writeReviewLog = vi.fn();
|
|
134
|
+
const approved: PermissionPromptDecision = {
|
|
135
|
+
approved: true,
|
|
136
|
+
state: "approved",
|
|
137
|
+
};
|
|
138
|
+
mockConfirmPermission.mockResolvedValue(approved);
|
|
139
|
+
const deps = makeDeps({ writeReviewLog });
|
|
140
|
+
const prompter = new PermissionPrompter(deps);
|
|
141
|
+
|
|
142
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
143
|
+
|
|
144
|
+
const calls = writeReviewLog.mock.calls.map((c) => c[0] as string);
|
|
145
|
+
expect(
|
|
146
|
+
calls.indexOf("permission_request.waiting"),
|
|
147
|
+
).toBeGreaterThanOrEqual(0);
|
|
148
|
+
expect(calls.indexOf("permission_request.waiting")).toBeLessThan(
|
|
149
|
+
calls.indexOf("permission_request.approved"),
|
|
150
|
+
);
|
|
151
|
+
});
|
|
152
|
+
|
|
153
|
+
it("logs permission_request.approved when confirmPermission returns approved", async () => {
|
|
154
|
+
const writeReviewLog = vi.fn();
|
|
155
|
+
mockConfirmPermission.mockResolvedValue({
|
|
156
|
+
approved: true,
|
|
157
|
+
state: "approved",
|
|
158
|
+
});
|
|
159
|
+
const deps = makeDeps({ writeReviewLog });
|
|
160
|
+
const prompter = new PermissionPrompter(deps);
|
|
161
|
+
|
|
162
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
163
|
+
|
|
164
|
+
expect(writeReviewLog).toHaveBeenCalledWith(
|
|
165
|
+
"permission_request.approved",
|
|
166
|
+
expect.objectContaining({
|
|
167
|
+
requestId: "req-123",
|
|
168
|
+
resolution: "approved",
|
|
169
|
+
}),
|
|
170
|
+
);
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
it("logs permission_request.denied when confirmPermission returns denied", async () => {
|
|
174
|
+
const writeReviewLog = vi.fn();
|
|
175
|
+
mockConfirmPermission.mockResolvedValue({
|
|
176
|
+
approved: false,
|
|
177
|
+
state: "denied",
|
|
178
|
+
});
|
|
179
|
+
const deps = makeDeps({ writeReviewLog });
|
|
180
|
+
const prompter = new PermissionPrompter(deps);
|
|
181
|
+
|
|
182
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
183
|
+
|
|
184
|
+
expect(writeReviewLog).toHaveBeenCalledWith(
|
|
185
|
+
"permission_request.denied",
|
|
186
|
+
expect.objectContaining({
|
|
187
|
+
requestId: "req-123",
|
|
188
|
+
resolution: "denied",
|
|
189
|
+
}),
|
|
190
|
+
);
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
it("logs permission_request.denied with denialReason when present", async () => {
|
|
194
|
+
const writeReviewLog = vi.fn();
|
|
195
|
+
mockConfirmPermission.mockResolvedValue({
|
|
196
|
+
approved: false,
|
|
197
|
+
state: "denied_with_reason",
|
|
198
|
+
denialReason: "too sensitive",
|
|
199
|
+
});
|
|
200
|
+
const deps = makeDeps({ writeReviewLog });
|
|
201
|
+
const prompter = new PermissionPrompter(deps);
|
|
202
|
+
|
|
203
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
204
|
+
|
|
205
|
+
expect(writeReviewLog).toHaveBeenCalledWith(
|
|
206
|
+
"permission_request.denied",
|
|
207
|
+
expect.objectContaining({
|
|
208
|
+
denialReason: "too sensitive",
|
|
209
|
+
}),
|
|
210
|
+
);
|
|
211
|
+
});
|
|
212
|
+
|
|
213
|
+
it("returns the decision from confirmPermission", async () => {
|
|
214
|
+
const decision: PermissionPromptDecision = {
|
|
215
|
+
approved: false,
|
|
216
|
+
state: "denied_with_reason",
|
|
217
|
+
denialReason: "sensitive",
|
|
218
|
+
};
|
|
219
|
+
mockConfirmPermission.mockResolvedValue(decision);
|
|
220
|
+
const deps = makeDeps();
|
|
221
|
+
const prompter = new PermissionPrompter(deps);
|
|
222
|
+
|
|
223
|
+
const result = await prompter.prompt(makeCtx(true), makeDetails());
|
|
224
|
+
|
|
225
|
+
expect(result).toEqual(decision);
|
|
226
|
+
});
|
|
227
|
+
|
|
228
|
+
it("passes sessionLabel option to confirmPermission when present", async () => {
|
|
229
|
+
mockConfirmPermission.mockResolvedValue({
|
|
230
|
+
approved: true,
|
|
231
|
+
state: "approved",
|
|
232
|
+
});
|
|
233
|
+
const deps = makeDeps();
|
|
234
|
+
const prompter = new PermissionPrompter(deps);
|
|
235
|
+
const details = makeDetails({ sessionLabel: "Yes, for 'read' tool" });
|
|
236
|
+
|
|
237
|
+
await prompter.prompt(makeCtx(true), details);
|
|
238
|
+
|
|
239
|
+
expect(mockConfirmPermission).toHaveBeenCalledWith(
|
|
240
|
+
expect.anything(),
|
|
241
|
+
expect.any(String),
|
|
242
|
+
expect.anything(),
|
|
243
|
+
{ sessionLabel: "Yes, for 'read' tool" },
|
|
244
|
+
);
|
|
245
|
+
});
|
|
246
|
+
|
|
247
|
+
it("passes undefined options to confirmPermission when sessionLabel is absent", async () => {
|
|
248
|
+
mockConfirmPermission.mockResolvedValue({
|
|
249
|
+
approved: true,
|
|
250
|
+
state: "approved",
|
|
251
|
+
});
|
|
252
|
+
const deps = makeDeps();
|
|
253
|
+
const prompter = new PermissionPrompter(deps);
|
|
254
|
+
|
|
255
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
256
|
+
|
|
257
|
+
expect(mockConfirmPermission).toHaveBeenCalledWith(
|
|
258
|
+
expect.anything(),
|
|
259
|
+
expect.any(String),
|
|
260
|
+
expect.anything(),
|
|
261
|
+
undefined,
|
|
262
|
+
);
|
|
263
|
+
});
|
|
264
|
+
|
|
265
|
+
it("passes the message from details to confirmPermission", async () => {
|
|
266
|
+
mockConfirmPermission.mockResolvedValue({
|
|
267
|
+
approved: true,
|
|
268
|
+
state: "approved",
|
|
269
|
+
});
|
|
270
|
+
const deps = makeDeps();
|
|
271
|
+
const prompter = new PermissionPrompter(deps);
|
|
272
|
+
const details = makeDetails({ message: "Allow bash: git status?" });
|
|
273
|
+
|
|
274
|
+
await prompter.prompt(makeCtx(true), details);
|
|
275
|
+
|
|
276
|
+
expect(mockConfirmPermission).toHaveBeenCalledWith(
|
|
277
|
+
expect.anything(),
|
|
278
|
+
"Allow bash: git status?",
|
|
279
|
+
expect.anything(),
|
|
280
|
+
undefined,
|
|
281
|
+
);
|
|
282
|
+
});
|
|
283
|
+
});
|
|
284
|
+
|
|
285
|
+
// ── Review log field coverage ────────────────────────────────────────────
|
|
286
|
+
|
|
287
|
+
describe("review log fields", () => {
|
|
288
|
+
it("includes all standard fields in the waiting log entry", async () => {
|
|
289
|
+
const writeReviewLog = vi.fn();
|
|
290
|
+
mockConfirmPermission.mockResolvedValue({
|
|
291
|
+
approved: true,
|
|
292
|
+
state: "approved",
|
|
293
|
+
});
|
|
294
|
+
const deps = makeDeps({ writeReviewLog });
|
|
295
|
+
const prompter = new PermissionPrompter(deps);
|
|
296
|
+
const details = makeDetails({
|
|
297
|
+
toolCallId: "tc-1",
|
|
298
|
+
skillName: "librarian",
|
|
299
|
+
path: "/src/foo.ts",
|
|
300
|
+
command: "git status",
|
|
301
|
+
target: "server:tool",
|
|
302
|
+
toolInputPreview: "{ path: '...' }",
|
|
303
|
+
});
|
|
304
|
+
|
|
305
|
+
await prompter.prompt(makeCtx(true), details);
|
|
306
|
+
|
|
307
|
+
expect(writeReviewLog).toHaveBeenCalledWith(
|
|
308
|
+
"permission_request.waiting",
|
|
309
|
+
expect.objectContaining({
|
|
310
|
+
requestId: "req-123",
|
|
311
|
+
source: "tool_call",
|
|
312
|
+
agentName: "test-agent",
|
|
313
|
+
message: "Allow read?",
|
|
314
|
+
toolCallId: "tc-1",
|
|
315
|
+
toolName: "read",
|
|
316
|
+
skillName: "librarian",
|
|
317
|
+
path: "/src/foo.ts",
|
|
318
|
+
command: "git status",
|
|
319
|
+
target: "server:tool",
|
|
320
|
+
toolInputPreview: "{ path: '...' }",
|
|
321
|
+
}),
|
|
322
|
+
);
|
|
323
|
+
});
|
|
324
|
+
|
|
325
|
+
it("uses null for optional fields not present in details", async () => {
|
|
326
|
+
const writeReviewLog = vi.fn();
|
|
327
|
+
mockConfirmPermission.mockResolvedValue({
|
|
328
|
+
approved: true,
|
|
329
|
+
state: "approved",
|
|
330
|
+
});
|
|
331
|
+
const deps = makeDeps({ writeReviewLog });
|
|
332
|
+
const prompter = new PermissionPrompter(deps);
|
|
333
|
+
|
|
334
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
335
|
+
|
|
336
|
+
expect(writeReviewLog).toHaveBeenCalledWith(
|
|
337
|
+
"permission_request.waiting",
|
|
338
|
+
expect.objectContaining({
|
|
339
|
+
toolCallId: null,
|
|
340
|
+
skillName: null,
|
|
341
|
+
path: null,
|
|
342
|
+
command: null,
|
|
343
|
+
target: null,
|
|
344
|
+
toolInputPreview: null,
|
|
345
|
+
}),
|
|
346
|
+
);
|
|
347
|
+
});
|
|
348
|
+
});
|
|
349
|
+
|
|
350
|
+
// ── Subagent forwarding path ─────────────────────────────────────────────
|
|
351
|
+
|
|
352
|
+
describe("subagent forwarding path", () => {
|
|
353
|
+
it("calls confirmPermission even when ctx has no UI", async () => {
|
|
354
|
+
const forwarded: PermissionPromptDecision = {
|
|
355
|
+
approved: true,
|
|
356
|
+
state: "approved",
|
|
357
|
+
};
|
|
358
|
+
mockConfirmPermission.mockResolvedValue(forwarded);
|
|
359
|
+
const deps = makeDeps();
|
|
360
|
+
const prompter = new PermissionPrompter(deps);
|
|
361
|
+
|
|
362
|
+
await prompter.prompt(makeCtx(false), makeDetails());
|
|
363
|
+
|
|
364
|
+
expect(mockConfirmPermission).toHaveBeenCalled();
|
|
365
|
+
});
|
|
366
|
+
|
|
367
|
+
it("returns the decision from confirmPermission in the subagent path", async () => {
|
|
368
|
+
const forwarded: PermissionPromptDecision = {
|
|
369
|
+
approved: false,
|
|
370
|
+
state: "denied",
|
|
371
|
+
};
|
|
372
|
+
mockConfirmPermission.mockResolvedValue(forwarded);
|
|
373
|
+
const deps = makeDeps();
|
|
374
|
+
const prompter = new PermissionPrompter(deps);
|
|
375
|
+
|
|
376
|
+
const result = await prompter.prompt(makeCtx(false), makeDetails());
|
|
377
|
+
|
|
378
|
+
expect(result).toEqual(forwarded);
|
|
379
|
+
});
|
|
380
|
+
|
|
381
|
+
it("logs the outcome when confirmPermission resolves via forwarding", async () => {
|
|
382
|
+
const writeReviewLog = vi.fn();
|
|
383
|
+
mockConfirmPermission.mockResolvedValue({
|
|
384
|
+
approved: true,
|
|
385
|
+
state: "approved",
|
|
386
|
+
});
|
|
387
|
+
const deps = makeDeps({ writeReviewLog });
|
|
388
|
+
const prompter = new PermissionPrompter(deps);
|
|
389
|
+
|
|
390
|
+
await prompter.prompt(makeCtx(false), makeDetails());
|
|
391
|
+
|
|
392
|
+
expect(writeReviewLog).toHaveBeenCalledWith(
|
|
393
|
+
"permission_request.approved",
|
|
394
|
+
expect.objectContaining({ requestId: "req-123" }),
|
|
395
|
+
);
|
|
396
|
+
});
|
|
397
|
+
});
|
|
398
|
+
});
|
|
@@ -2620,6 +2620,187 @@ test("session rules override config deny for external_directory", () => {
|
|
|
2620
2620
|
}
|
|
2621
2621
|
});
|
|
2622
2622
|
|
|
2623
|
+
// ── Session rule evaluation for all surfaces ─────────────────────────────
|
|
2624
|
+
|
|
2625
|
+
test("checkPermission returns source 'session' for bash when session rules match", () => {
|
|
2626
|
+
const { manager, cleanup } = createManager({ permission: {} });
|
|
2627
|
+
|
|
2628
|
+
try {
|
|
2629
|
+
const sessionRules = [
|
|
2630
|
+
{
|
|
2631
|
+
surface: "bash",
|
|
2632
|
+
pattern: "git *",
|
|
2633
|
+
action: "allow" as const,
|
|
2634
|
+
layer: "session" as const,
|
|
2635
|
+
},
|
|
2636
|
+
];
|
|
2637
|
+
|
|
2638
|
+
const result = manager.checkPermission(
|
|
2639
|
+
"bash",
|
|
2640
|
+
{ command: "git status --short" },
|
|
2641
|
+
undefined,
|
|
2642
|
+
sessionRules,
|
|
2643
|
+
);
|
|
2644
|
+
assert.equal(result.state, "allow");
|
|
2645
|
+
assert.equal(result.source, "session");
|
|
2646
|
+
assert.equal(result.matchedPattern, "git *");
|
|
2647
|
+
} finally {
|
|
2648
|
+
cleanup();
|
|
2649
|
+
}
|
|
2650
|
+
});
|
|
2651
|
+
|
|
2652
|
+
test("checkPermission returns source 'session' for bash when session rule is exact match", () => {
|
|
2653
|
+
const { manager, cleanup } = createManager({ permission: {} });
|
|
2654
|
+
|
|
2655
|
+
try {
|
|
2656
|
+
const sessionRules = [
|
|
2657
|
+
{
|
|
2658
|
+
surface: "bash",
|
|
2659
|
+
pattern: "ls",
|
|
2660
|
+
action: "allow" as const,
|
|
2661
|
+
layer: "session" as const,
|
|
2662
|
+
},
|
|
2663
|
+
];
|
|
2664
|
+
|
|
2665
|
+
const result = manager.checkPermission(
|
|
2666
|
+
"bash",
|
|
2667
|
+
{ command: "ls" },
|
|
2668
|
+
undefined,
|
|
2669
|
+
sessionRules,
|
|
2670
|
+
);
|
|
2671
|
+
assert.equal(result.state, "allow");
|
|
2672
|
+
assert.equal(result.source, "session");
|
|
2673
|
+
} finally {
|
|
2674
|
+
cleanup();
|
|
2675
|
+
}
|
|
2676
|
+
});
|
|
2677
|
+
|
|
2678
|
+
test("checkPermission falls back to config for bash when session rules do not match the command", () => {
|
|
2679
|
+
const { manager, cleanup } = createManager({ permission: { bash: "deny" } });
|
|
2680
|
+
|
|
2681
|
+
try {
|
|
2682
|
+
const sessionRules = [
|
|
2683
|
+
{
|
|
2684
|
+
surface: "bash",
|
|
2685
|
+
pattern: "git *",
|
|
2686
|
+
action: "allow" as const,
|
|
2687
|
+
layer: "session" as const,
|
|
2688
|
+
},
|
|
2689
|
+
];
|
|
2690
|
+
|
|
2691
|
+
const result = manager.checkPermission(
|
|
2692
|
+
"bash",
|
|
2693
|
+
{ command: "npm run build" },
|
|
2694
|
+
undefined,
|
|
2695
|
+
sessionRules,
|
|
2696
|
+
);
|
|
2697
|
+
assert.equal(result.state, "deny");
|
|
2698
|
+
assert.equal(result.source, "bash");
|
|
2699
|
+
} finally {
|
|
2700
|
+
cleanup();
|
|
2701
|
+
}
|
|
2702
|
+
});
|
|
2703
|
+
|
|
2704
|
+
test("checkPermission returns source 'session' for mcp when session rules match the target", () => {
|
|
2705
|
+
const { manager, cleanup } = createManager({ permission: {} });
|
|
2706
|
+
|
|
2707
|
+
try {
|
|
2708
|
+
const sessionRules = [
|
|
2709
|
+
{
|
|
2710
|
+
surface: "mcp",
|
|
2711
|
+
pattern: "exa:*",
|
|
2712
|
+
action: "allow" as const,
|
|
2713
|
+
layer: "session" as const,
|
|
2714
|
+
},
|
|
2715
|
+
];
|
|
2716
|
+
|
|
2717
|
+
const result = manager.checkPermission(
|
|
2718
|
+
"mcp",
|
|
2719
|
+
{ tool: "exa:search" },
|
|
2720
|
+
undefined,
|
|
2721
|
+
sessionRules,
|
|
2722
|
+
);
|
|
2723
|
+
assert.equal(result.state, "allow");
|
|
2724
|
+
assert.equal(result.source, "session");
|
|
2725
|
+
} finally {
|
|
2726
|
+
cleanup();
|
|
2727
|
+
}
|
|
2728
|
+
});
|
|
2729
|
+
|
|
2730
|
+
test("checkPermission returns source 'session' for skill when session rules match", () => {
|
|
2731
|
+
const { manager, cleanup } = createManager({ permission: {} });
|
|
2732
|
+
|
|
2733
|
+
try {
|
|
2734
|
+
const sessionRules = [
|
|
2735
|
+
{
|
|
2736
|
+
surface: "skill",
|
|
2737
|
+
pattern: "librarian",
|
|
2738
|
+
action: "allow" as const,
|
|
2739
|
+
layer: "session" as const,
|
|
2740
|
+
},
|
|
2741
|
+
];
|
|
2742
|
+
|
|
2743
|
+
const result = manager.checkPermission(
|
|
2744
|
+
"skill",
|
|
2745
|
+
{ name: "librarian" },
|
|
2746
|
+
undefined,
|
|
2747
|
+
sessionRules,
|
|
2748
|
+
);
|
|
2749
|
+
assert.equal(result.state, "allow");
|
|
2750
|
+
assert.equal(result.source, "session");
|
|
2751
|
+
assert.equal(result.matchedPattern, "librarian");
|
|
2752
|
+
} finally {
|
|
2753
|
+
cleanup();
|
|
2754
|
+
}
|
|
2755
|
+
});
|
|
2756
|
+
|
|
2757
|
+
test("checkPermission returns source 'session' for tool surface when session rules match", () => {
|
|
2758
|
+
const { manager, cleanup } = createManager({ permission: {} });
|
|
2759
|
+
|
|
2760
|
+
try {
|
|
2761
|
+
const sessionRules = [
|
|
2762
|
+
{
|
|
2763
|
+
surface: "read",
|
|
2764
|
+
pattern: "*",
|
|
2765
|
+
action: "allow" as const,
|
|
2766
|
+
layer: "session" as const,
|
|
2767
|
+
},
|
|
2768
|
+
];
|
|
2769
|
+
|
|
2770
|
+
const result = manager.checkPermission("read", {}, undefined, sessionRules);
|
|
2771
|
+
assert.equal(result.state, "allow");
|
|
2772
|
+
assert.equal(result.source, "session");
|
|
2773
|
+
} finally {
|
|
2774
|
+
cleanup();
|
|
2775
|
+
}
|
|
2776
|
+
});
|
|
2777
|
+
|
|
2778
|
+
test("bash session rules do not bleed into mcp checks", () => {
|
|
2779
|
+
const { manager, cleanup } = createManager({ permission: {} });
|
|
2780
|
+
|
|
2781
|
+
try {
|
|
2782
|
+
const sessionRules = [
|
|
2783
|
+
{
|
|
2784
|
+
surface: "bash",
|
|
2785
|
+
pattern: "git *",
|
|
2786
|
+
action: "allow" as const,
|
|
2787
|
+
layer: "session" as const,
|
|
2788
|
+
},
|
|
2789
|
+
];
|
|
2790
|
+
|
|
2791
|
+
const result = manager.checkPermission(
|
|
2792
|
+
"mcp",
|
|
2793
|
+
{ tool: "exa:search" },
|
|
2794
|
+
undefined,
|
|
2795
|
+
sessionRules,
|
|
2796
|
+
);
|
|
2797
|
+
// bash session rule must not affect mcp surface
|
|
2798
|
+
assert.notEqual(result.source, "session");
|
|
2799
|
+
} finally {
|
|
2800
|
+
cleanup();
|
|
2801
|
+
}
|
|
2802
|
+
});
|
|
2803
|
+
|
|
2623
2804
|
// Suppress unused import warning — PermissionState used in type annotations
|
|
2624
2805
|
const _unused: PermissionState = "ask";
|
|
2625
2806
|
void _unused;
|
package/tests/runtime.test.ts
CHANGED
|
@@ -59,9 +59,6 @@ vi.mock("../src/config-reporter", () => ({
|
|
|
59
59
|
|
|
60
60
|
vi.mock("../src/forwarded-permissions/polling", () => ({
|
|
61
61
|
processForwardedPermissionRequests: vi.fn().mockResolvedValue(undefined),
|
|
62
|
-
confirmPermission: vi
|
|
63
|
-
.fn()
|
|
64
|
-
.mockResolvedValue({ approved: true, state: "approved" }),
|
|
65
62
|
}));
|
|
66
63
|
|
|
67
64
|
vi.mock("../src/subagent-context", () => ({
|