@komarspn/pi-permission-system 16.0.2
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 +2234 -0
- package/LICENSE +21 -0
- package/README.md +158 -0
- package/config/config.example.json +39 -0
- package/package.json +82 -0
- package/schemas/permissions.schema.json +158 -0
- package/src/active-agent.ts +72 -0
- package/src/async-cache.ts +21 -0
- package/src/bash-arity.ts +210 -0
- package/src/builtin-tool-input-formatters.ts +82 -0
- package/src/canonicalize-path.ts +30 -0
- package/src/common.ts +121 -0
- package/src/config-loader.ts +432 -0
- package/src/config-modal.ts +259 -0
- package/src/config-paths.ts +47 -0
- package/src/config-reporter.ts +34 -0
- package/src/config-store.ts +222 -0
- package/src/decision-audit.ts +75 -0
- package/src/decision-reporter.ts +41 -0
- package/src/denial-messages.ts +232 -0
- package/src/expand-home.ts +28 -0
- package/src/extension-config.ts +79 -0
- package/src/extension-paths.ts +66 -0
- package/src/forwarded-permissions/io.ts +404 -0
- package/src/forwarded-permissions/permission-forwarder.ts +580 -0
- package/src/forwarding-manager.ts +74 -0
- package/src/gate-prompter.ts +12 -0
- package/src/handlers/before-agent-start.ts +94 -0
- package/src/handlers/gates/bash-command.ts +75 -0
- package/src/handlers/gates/bash-external-directory.ts +127 -0
- package/src/handlers/gates/bash-path-extractor.ts +15 -0
- package/src/handlers/gates/bash-path.ts +152 -0
- package/src/handlers/gates/bash-program.ts +1143 -0
- package/src/handlers/gates/bash-token-classification.ts +105 -0
- package/src/handlers/gates/candidate-check.ts +32 -0
- package/src/handlers/gates/descriptor.ts +81 -0
- package/src/handlers/gates/external-directory-messages.ts +20 -0
- package/src/handlers/gates/external-directory.ts +133 -0
- package/src/handlers/gates/helpers.ts +76 -0
- package/src/handlers/gates/path.ts +91 -0
- package/src/handlers/gates/runner.ts +186 -0
- package/src/handlers/gates/skill-input-gate-pipeline.ts +104 -0
- package/src/handlers/gates/skill-input.ts +46 -0
- package/src/handlers/gates/skill-read.ts +87 -0
- package/src/handlers/gates/tool-call-gate-pipeline.ts +129 -0
- package/src/handlers/gates/tool.ts +102 -0
- package/src/handlers/gates/types.ts +13 -0
- package/src/handlers/index.ts +3 -0
- package/src/handlers/lifecycle.ts +95 -0
- package/src/handlers/permission-gate-handler.ts +190 -0
- package/src/handlers/tool-call-boundary.ts +91 -0
- package/src/index.ts +225 -0
- package/src/input-normalizer.ts +157 -0
- package/src/logging.ts +113 -0
- package/src/mcp-targets.ts +170 -0
- package/src/node-modules-discovery.ts +76 -0
- package/src/normalize.ts +43 -0
- package/src/path-utils.ts +355 -0
- package/src/pattern-suggest.ts +132 -0
- package/src/permission-dialog.ts +138 -0
- package/src/permission-event-rpc.ts +223 -0
- package/src/permission-events.ts +266 -0
- package/src/permission-forwarding.ts +188 -0
- package/src/permission-gate.ts +94 -0
- package/src/permission-manager.ts +392 -0
- package/src/permission-merge.ts +32 -0
- package/src/permission-prompter.ts +142 -0
- package/src/permission-prompts.ts +93 -0
- package/src/permission-resolver.ts +109 -0
- package/src/permission-session.ts +189 -0
- package/src/permission-ui-prompt.ts +127 -0
- package/src/permissions-service.ts +63 -0
- package/src/persistent-approval-recorder.ts +139 -0
- package/src/policy-loader.ts +350 -0
- package/src/prompting-gateway.ts +104 -0
- package/src/rule.ts +188 -0
- package/src/scope-merge.ts +72 -0
- package/src/service-lifecycle.ts +49 -0
- package/src/service.ts +163 -0
- package/src/session-approval-recorder.ts +6 -0
- package/src/session-approval.ts +43 -0
- package/src/session-logger.ts +91 -0
- package/src/session-rules.ts +79 -0
- package/src/skill-prompt-sanitizer.ts +292 -0
- package/src/status.ts +35 -0
- package/src/subagent-context.ts +104 -0
- package/src/subagent-lifecycle-events.ts +72 -0
- package/src/subagent-registry.ts +105 -0
- package/src/synthesize.ts +92 -0
- package/src/system-prompt-sanitizer.ts +274 -0
- package/src/tool-access-extractor-registry.ts +68 -0
- package/src/tool-input-formatter-registry.ts +67 -0
- package/src/tool-input-preview.ts +34 -0
- package/src/tool-input-prompt-formatters.ts +63 -0
- package/src/tool-preview-formatter.ts +207 -0
- package/src/tool-registry.ts +148 -0
- package/src/types.ts +64 -0
- package/src/wildcard-matcher.ts +120 -0
- package/src/yolo-mode.ts +30 -0
- package/test/active-agent.test.ts +155 -0
- package/test/async-cache.test.ts +48 -0
- package/test/bash-arity.test.ts +144 -0
- package/test/bash-external-directory.test.ts +956 -0
- package/test/builtin-tool-input-formatters.test.ts +109 -0
- package/test/canonicalize-path.test.ts +93 -0
- package/test/common.test.ts +287 -0
- package/test/composition-root.test.ts +603 -0
- package/test/config-loader.test.ts +740 -0
- package/test/config-modal.test.ts +320 -0
- package/test/config-paths.test.ts +83 -0
- package/test/config-pipeline.test.ts +90 -0
- package/test/config-reporter.test.ts +147 -0
- package/test/config-store.test.ts +466 -0
- package/test/decision-audit.test.ts +72 -0
- package/test/decision-reporter.test.ts +112 -0
- package/test/denial-messages.test.ts +656 -0
- package/test/detect-permissive-bash-fallback.test.ts +56 -0
- package/test/expand-home.test.ts +93 -0
- package/test/extension-config.test.ts +129 -0
- package/test/extension-paths.test.ts +108 -0
- package/test/forwarded-permissions/io.test.ts +251 -0
- package/test/forwarding-manager.test.ts +194 -0
- package/test/handlers/before-agent-start.test.ts +317 -0
- package/test/handlers/external-directory-integration.test.ts +623 -0
- package/test/handlers/external-directory-session-dedup.test.ts +430 -0
- package/test/handlers/external-directory-symlink-acceptance.test.ts +149 -0
- package/test/handlers/gates/bash-command-metamorphic.test.ts +83 -0
- package/test/handlers/gates/bash-command.test.ts +191 -0
- package/test/handlers/gates/bash-external-directory.test.ts +269 -0
- package/test/handlers/gates/bash-path.test.ts +337 -0
- package/test/handlers/gates/bash-program.test.ts +410 -0
- package/test/handlers/gates/bash-token-classification.test.ts +241 -0
- package/test/handlers/gates/candidate-check.test.ts +52 -0
- package/test/handlers/gates/external-directory-messages.test.ts +61 -0
- package/test/handlers/gates/external-directory.test.ts +259 -0
- package/test/handlers/gates/helpers.test.ts +177 -0
- package/test/handlers/gates/path.test.ts +294 -0
- package/test/handlers/gates/runner.test.ts +447 -0
- package/test/handlers/gates/skill-input-gate-pipeline.test.ts +176 -0
- package/test/handlers/gates/skill-input.test.ts +131 -0
- package/test/handlers/gates/skill-read.test.ts +158 -0
- package/test/handlers/gates/tool-call-gate-pipeline.test.ts +252 -0
- package/test/handlers/gates/tool.test.ts +223 -0
- package/test/handlers/input-events.test.ts +168 -0
- package/test/handlers/input.test.ts +199 -0
- package/test/handlers/lifecycle.test.ts +221 -0
- package/test/handlers/tool-call-boundary.test.ts +145 -0
- package/test/handlers/tool-call-events.test.ts +277 -0
- package/test/handlers/tool-call.test.ts +395 -0
- package/test/handlers/validate-requested-tool.test.ts +92 -0
- package/test/helpers/gate-fixtures.ts +323 -0
- package/test/helpers/handler-fixtures.ts +335 -0
- package/test/helpers/make-fake-pi.ts +100 -0
- package/test/helpers/manager-harness.ts +112 -0
- package/test/helpers/session-fixtures.ts +204 -0
- package/test/input-normalizer.test.ts +367 -0
- package/test/logging.test.ts +51 -0
- package/test/mcp-targets.test.ts +233 -0
- package/test/node-modules-discovery.test.ts +97 -0
- package/test/normalize.test.ts +247 -0
- package/test/path-utils.test.ts +650 -0
- package/test/pattern-suggest.test.ts +248 -0
- package/test/permission-dialog.test.ts +241 -0
- package/test/permission-event-rpc.test.ts +541 -0
- package/test/permission-events.test.ts +402 -0
- package/test/permission-forwarder.test.ts +369 -0
- package/test/permission-forwarding.test.ts +315 -0
- package/test/permission-gate.test.ts +305 -0
- package/test/permission-manager-unified.test.ts +3368 -0
- package/test/permission-merge.test.ts +61 -0
- package/test/permission-prompter.test.ts +518 -0
- package/test/permission-prompts.test.ts +363 -0
- package/test/permission-resolver.test.ts +265 -0
- package/test/permission-session.test.ts +363 -0
- package/test/permission-ui-prompt.test.ts +146 -0
- package/test/permissions-service.test.ts +177 -0
- package/test/persistent-approval-recorder.test.ts +133 -0
- package/test/pi-infrastructure-read.test.ts +369 -0
- package/test/policy-loader.test.ts +561 -0
- package/test/prompting-gateway.test.ts +230 -0
- package/test/rule.test.ts +604 -0
- package/test/scope-merge.test.ts +116 -0
- package/test/service-lifecycle.test.ts +163 -0
- package/test/service.test.ts +308 -0
- package/test/session-approval.test.ts +75 -0
- package/test/session-logger.test.ts +200 -0
- package/test/session-rules.test.ts +304 -0
- package/test/session-start.test.ts +112 -0
- package/test/skill-prompt-sanitizer.test.ts +374 -0
- package/test/status.test.ts +10 -0
- package/test/subagent-context.test.ts +326 -0
- package/test/subagent-lifecycle-events.test.ts +132 -0
- package/test/subagent-registry.test.ts +145 -0
- package/test/synthesize.test.ts +300 -0
- package/test/system-prompt-sanitizer.test.ts +382 -0
- package/test/tool-access-extractor-registry.test.ts +77 -0
- package/test/tool-input-formatter-registry.test.ts +75 -0
- package/test/tool-input-preview.test.ts +129 -0
- package/test/tool-input-prompt-formatters.test.ts +115 -0
- package/test/tool-preview-formatter.test.ts +458 -0
- package/test/tool-registry.test.ts +197 -0
- package/test/wildcard-matcher.test.ts +424 -0
- package/test/yolo-mode.test.ts +188 -0
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { describe, expect, test } from "vitest";
|
|
2
|
+
|
|
3
|
+
import { mergeFlatPermissions } from "#src/permission-merge";
|
|
4
|
+
|
|
5
|
+
describe("mergeFlatPermissions", () => {
|
|
6
|
+
test("string replaces string", () => {
|
|
7
|
+
const result = mergeFlatPermissions({ tools: "ask" }, { tools: "allow" });
|
|
8
|
+
expect(result).toEqual({ tools: "allow" });
|
|
9
|
+
});
|
|
10
|
+
|
|
11
|
+
test("both objects → shallow-merge pattern maps", () => {
|
|
12
|
+
const result = mergeFlatPermissions(
|
|
13
|
+
{ bash: { "rm *": "deny", "git *": "ask" } },
|
|
14
|
+
{ bash: { "rm *": "allow", "npm *": "allow" } },
|
|
15
|
+
);
|
|
16
|
+
expect(result).toEqual({
|
|
17
|
+
bash: { "rm *": "allow", "git *": "ask", "npm *": "allow" },
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
|
|
21
|
+
test("object replaces string", () => {
|
|
22
|
+
const result = mergeFlatPermissions(
|
|
23
|
+
{ tools: "ask" },
|
|
24
|
+
{ tools: { Write: "deny" } },
|
|
25
|
+
);
|
|
26
|
+
expect(result).toEqual({ tools: { Write: "deny" } });
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
test("string replaces object", () => {
|
|
30
|
+
const result = mergeFlatPermissions(
|
|
31
|
+
{ tools: { Write: "deny" } },
|
|
32
|
+
{ tools: "allow" },
|
|
33
|
+
);
|
|
34
|
+
expect(result).toEqual({ tools: "allow" });
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
test("empty override returns base unchanged", () => {
|
|
38
|
+
const base = { tools: "ask" as const, bash: { "rm *": "deny" as const } };
|
|
39
|
+
const result = mergeFlatPermissions(base, {});
|
|
40
|
+
expect(result).toEqual(base);
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
test("empty base returns override", () => {
|
|
44
|
+
const override = { tools: "allow" as const };
|
|
45
|
+
const result = mergeFlatPermissions({}, override);
|
|
46
|
+
expect(result).toEqual(override);
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
test("preserves keys only in base", () => {
|
|
50
|
+
const result = mergeFlatPermissions(
|
|
51
|
+
{ tools: "ask", bash: "deny" },
|
|
52
|
+
{ tools: "allow" },
|
|
53
|
+
);
|
|
54
|
+
expect(result).toEqual({ tools: "allow", bash: "deny" });
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
test("adds keys only in override", () => {
|
|
58
|
+
const result = mergeFlatPermissions({ tools: "ask" }, { bash: "allow" });
|
|
59
|
+
expect(result).toEqual({ tools: "ask", bash: "allow" });
|
|
60
|
+
});
|
|
61
|
+
});
|
|
@@ -0,0 +1,518 @@
|
|
|
1
|
+
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
2
|
+
|
|
3
|
+
// ── Injected mock ───────────────────────────────────────────────────────────
|
|
4
|
+
|
|
5
|
+
const mockRequestApproval = vi.fn();
|
|
6
|
+
|
|
7
|
+
// ── Imports ─────────────────────────────────────────────────────────────────
|
|
8
|
+
|
|
9
|
+
import type { ExtensionContext } from "@earendil-works/pi-coding-agent";
|
|
10
|
+
import type { ConfigReader } from "#src/config-store";
|
|
11
|
+
import { DEFAULT_EXTENSION_CONFIG } from "#src/extension-config";
|
|
12
|
+
import type { PermissionPromptDecision } from "#src/permission-dialog";
|
|
13
|
+
import type { PromptPermissionDetails } from "#src/permission-prompter";
|
|
14
|
+
import {
|
|
15
|
+
PermissionPrompter,
|
|
16
|
+
type PermissionPrompterDeps,
|
|
17
|
+
} from "#src/permission-prompter";
|
|
18
|
+
|
|
19
|
+
// ── Helpers ─────────────────────────────────────────────────────────────────
|
|
20
|
+
|
|
21
|
+
function makeCtx(hasUI: boolean): ExtensionContext {
|
|
22
|
+
return {
|
|
23
|
+
hasUI,
|
|
24
|
+
ui: { select: vi.fn(), input: vi.fn() },
|
|
25
|
+
sessionManager: { getSessionDir: vi.fn().mockReturnValue(null) },
|
|
26
|
+
} as unknown as ExtensionContext;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
function makeDetails(
|
|
30
|
+
overrides?: Partial<PromptPermissionDetails>,
|
|
31
|
+
): PromptPermissionDetails {
|
|
32
|
+
return {
|
|
33
|
+
requestId: "req-123",
|
|
34
|
+
source: "tool_call",
|
|
35
|
+
agentName: "test-agent",
|
|
36
|
+
message: "Allow read?",
|
|
37
|
+
toolName: "read",
|
|
38
|
+
...overrides,
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function makeConfigReader(
|
|
43
|
+
config: Partial<typeof DEFAULT_EXTENSION_CONFIG> = {},
|
|
44
|
+
): ConfigReader {
|
|
45
|
+
return { current: () => ({ ...DEFAULT_EXTENSION_CONFIG, ...config }) };
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
function makeDeps(
|
|
49
|
+
overrides?: Partial<PermissionPrompterDeps>,
|
|
50
|
+
): PermissionPrompterDeps {
|
|
51
|
+
return {
|
|
52
|
+
config: makeConfigReader(),
|
|
53
|
+
logger: { review: vi.fn() },
|
|
54
|
+
events: { emit: vi.fn(), on: vi.fn().mockReturnValue(() => undefined) },
|
|
55
|
+
forwarder: { requestApproval: mockRequestApproval },
|
|
56
|
+
...overrides,
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// ── Tests ────────────────────────────────────────────────────────────────────
|
|
61
|
+
|
|
62
|
+
describe("PermissionPrompter", () => {
|
|
63
|
+
beforeEach(() => {
|
|
64
|
+
mockRequestApproval.mockReset();
|
|
65
|
+
mockRequestApproval.mockResolvedValue({
|
|
66
|
+
approved: true,
|
|
67
|
+
state: "approved",
|
|
68
|
+
});
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
// ── Yolo-mode auto-approve ───────────────────────────────────────────────
|
|
72
|
+
|
|
73
|
+
describe("yolo-mode auto-approve", () => {
|
|
74
|
+
it("returns approved without calling confirmPermission when yoloMode is true", async () => {
|
|
75
|
+
const events = {
|
|
76
|
+
emit: vi.fn(),
|
|
77
|
+
on: vi.fn().mockReturnValue(() => undefined),
|
|
78
|
+
};
|
|
79
|
+
const deps = makeDeps({
|
|
80
|
+
config: makeConfigReader({ yoloMode: true }),
|
|
81
|
+
events,
|
|
82
|
+
});
|
|
83
|
+
const prompter = new PermissionPrompter(deps);
|
|
84
|
+
|
|
85
|
+
const decision = await prompter.prompt(makeCtx(false), makeDetails());
|
|
86
|
+
|
|
87
|
+
expect(decision).toEqual({
|
|
88
|
+
approved: true,
|
|
89
|
+
state: "approved",
|
|
90
|
+
autoApproved: true,
|
|
91
|
+
});
|
|
92
|
+
expect(mockRequestApproval).not.toHaveBeenCalled();
|
|
93
|
+
expect(events.emit).not.toHaveBeenCalledWith(
|
|
94
|
+
"permissions:ui_prompt",
|
|
95
|
+
expect.anything(),
|
|
96
|
+
);
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
it("logs permission_request.auto_approved in yolo mode", async () => {
|
|
100
|
+
const logger = { review: vi.fn() };
|
|
101
|
+
const deps = makeDeps({
|
|
102
|
+
config: makeConfigReader({ yoloMode: true }),
|
|
103
|
+
logger,
|
|
104
|
+
});
|
|
105
|
+
const prompter = new PermissionPrompter(deps);
|
|
106
|
+
|
|
107
|
+
await prompter.prompt(makeCtx(false), makeDetails());
|
|
108
|
+
|
|
109
|
+
expect(logger.review).toHaveBeenCalledWith(
|
|
110
|
+
"permission_request.auto_approved",
|
|
111
|
+
expect.objectContaining({ requestId: "req-123" }),
|
|
112
|
+
);
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
it("does not log permission_request.waiting in yolo mode", async () => {
|
|
116
|
+
const logger = { review: vi.fn() };
|
|
117
|
+
const deps = makeDeps({
|
|
118
|
+
config: makeConfigReader({ yoloMode: true }),
|
|
119
|
+
logger,
|
|
120
|
+
});
|
|
121
|
+
const prompter = new PermissionPrompter(deps);
|
|
122
|
+
|
|
123
|
+
await prompter.prompt(makeCtx(false), makeDetails());
|
|
124
|
+
|
|
125
|
+
expect(logger.review).not.toHaveBeenCalledWith(
|
|
126
|
+
"permission_request.waiting",
|
|
127
|
+
expect.anything(),
|
|
128
|
+
);
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
it("does not call confirmPermission with yoloMode even when ctx has UI", async () => {
|
|
132
|
+
const deps = makeDeps({
|
|
133
|
+
config: makeConfigReader({ yoloMode: true }),
|
|
134
|
+
});
|
|
135
|
+
const prompter = new PermissionPrompter(deps);
|
|
136
|
+
|
|
137
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
138
|
+
|
|
139
|
+
expect(mockRequestApproval).not.toHaveBeenCalled();
|
|
140
|
+
});
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
// ── Non-yolo path ────────────────────────────────────────────────────────
|
|
144
|
+
|
|
145
|
+
describe("non-yolo path (UI present)", () => {
|
|
146
|
+
it("logs permission_request.waiting before calling confirmPermission", async () => {
|
|
147
|
+
const logger = { review: vi.fn() };
|
|
148
|
+
const approved: PermissionPromptDecision = {
|
|
149
|
+
approved: true,
|
|
150
|
+
state: "approved",
|
|
151
|
+
};
|
|
152
|
+
mockRequestApproval.mockResolvedValue(approved);
|
|
153
|
+
const deps = makeDeps({ logger });
|
|
154
|
+
const prompter = new PermissionPrompter(deps);
|
|
155
|
+
|
|
156
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
157
|
+
|
|
158
|
+
const calls = logger.review.mock.calls.map((c) => c[0] as string);
|
|
159
|
+
expect(
|
|
160
|
+
calls.indexOf("permission_request.waiting"),
|
|
161
|
+
).toBeGreaterThanOrEqual(0);
|
|
162
|
+
expect(calls.indexOf("permission_request.waiting")).toBeLessThan(
|
|
163
|
+
calls.indexOf("permission_request.approved"),
|
|
164
|
+
);
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
it("emits a UI prompt event with normalized surface and value when the session has UI", async () => {
|
|
168
|
+
const events = {
|
|
169
|
+
emit: vi.fn(),
|
|
170
|
+
on: vi.fn().mockReturnValue(() => undefined),
|
|
171
|
+
};
|
|
172
|
+
mockRequestApproval.mockResolvedValue({
|
|
173
|
+
approved: true,
|
|
174
|
+
state: "approved",
|
|
175
|
+
});
|
|
176
|
+
const deps = makeDeps({ events });
|
|
177
|
+
const prompter = new PermissionPrompter(deps);
|
|
178
|
+
|
|
179
|
+
await prompter.prompt(
|
|
180
|
+
makeCtx(true),
|
|
181
|
+
makeDetails({
|
|
182
|
+
toolName: "bash",
|
|
183
|
+
command: "git push",
|
|
184
|
+
toolInputPreview: "git push",
|
|
185
|
+
}),
|
|
186
|
+
);
|
|
187
|
+
|
|
188
|
+
expect(events.emit).toHaveBeenCalledWith("permissions:ui_prompt", {
|
|
189
|
+
requestId: "req-123",
|
|
190
|
+
source: "tool_call",
|
|
191
|
+
surface: "bash",
|
|
192
|
+
value: "git push",
|
|
193
|
+
agentName: "test-agent",
|
|
194
|
+
message: "Allow read?",
|
|
195
|
+
forwarding: null,
|
|
196
|
+
});
|
|
197
|
+
});
|
|
198
|
+
|
|
199
|
+
it("normalizes skill UI prompt events to the skill surface", async () => {
|
|
200
|
+
const events = {
|
|
201
|
+
emit: vi.fn(),
|
|
202
|
+
on: vi.fn().mockReturnValue(() => undefined),
|
|
203
|
+
};
|
|
204
|
+
mockRequestApproval.mockResolvedValue({
|
|
205
|
+
approved: true,
|
|
206
|
+
state: "approved",
|
|
207
|
+
});
|
|
208
|
+
const deps = makeDeps({ events });
|
|
209
|
+
const prompter = new PermissionPrompter(deps);
|
|
210
|
+
|
|
211
|
+
await prompter.prompt(
|
|
212
|
+
makeCtx(true),
|
|
213
|
+
makeDetails({
|
|
214
|
+
source: "skill_input",
|
|
215
|
+
toolName: undefined,
|
|
216
|
+
skillName: "deploy-helper",
|
|
217
|
+
}),
|
|
218
|
+
);
|
|
219
|
+
|
|
220
|
+
expect(events.emit).toHaveBeenCalledWith("permissions:ui_prompt", {
|
|
221
|
+
requestId: "req-123",
|
|
222
|
+
source: "skill_input",
|
|
223
|
+
surface: "skill",
|
|
224
|
+
value: "deploy-helper",
|
|
225
|
+
agentName: "test-agent",
|
|
226
|
+
message: "Allow read?",
|
|
227
|
+
forwarding: null,
|
|
228
|
+
});
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
it("does not emit a UI prompt event when the session has no UI", async () => {
|
|
232
|
+
const events = {
|
|
233
|
+
emit: vi.fn(),
|
|
234
|
+
on: vi.fn().mockReturnValue(() => undefined),
|
|
235
|
+
};
|
|
236
|
+
mockRequestApproval.mockResolvedValue({
|
|
237
|
+
approved: true,
|
|
238
|
+
state: "approved",
|
|
239
|
+
});
|
|
240
|
+
const deps = makeDeps({ events });
|
|
241
|
+
const prompter = new PermissionPrompter(deps);
|
|
242
|
+
|
|
243
|
+
await prompter.prompt(makeCtx(false), makeDetails());
|
|
244
|
+
|
|
245
|
+
expect(events.emit).not.toHaveBeenCalledWith(
|
|
246
|
+
"permissions:ui_prompt",
|
|
247
|
+
expect.anything(),
|
|
248
|
+
);
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
it("logs permission_request.approved when confirmPermission returns approved", async () => {
|
|
252
|
+
const logger = { review: vi.fn() };
|
|
253
|
+
mockRequestApproval.mockResolvedValue({
|
|
254
|
+
approved: true,
|
|
255
|
+
state: "approved",
|
|
256
|
+
});
|
|
257
|
+
const deps = makeDeps({ logger });
|
|
258
|
+
const prompter = new PermissionPrompter(deps);
|
|
259
|
+
|
|
260
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
261
|
+
|
|
262
|
+
expect(logger.review).toHaveBeenCalledWith(
|
|
263
|
+
"permission_request.approved",
|
|
264
|
+
expect.objectContaining({
|
|
265
|
+
requestId: "req-123",
|
|
266
|
+
resolution: "approved",
|
|
267
|
+
}),
|
|
268
|
+
);
|
|
269
|
+
});
|
|
270
|
+
|
|
271
|
+
it("logs permission_request.denied when confirmPermission returns denied", async () => {
|
|
272
|
+
const logger = { review: vi.fn() };
|
|
273
|
+
mockRequestApproval.mockResolvedValue({
|
|
274
|
+
approved: false,
|
|
275
|
+
state: "denied",
|
|
276
|
+
});
|
|
277
|
+
const deps = makeDeps({ logger });
|
|
278
|
+
const prompter = new PermissionPrompter(deps);
|
|
279
|
+
|
|
280
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
281
|
+
|
|
282
|
+
expect(logger.review).toHaveBeenCalledWith(
|
|
283
|
+
"permission_request.denied",
|
|
284
|
+
expect.objectContaining({
|
|
285
|
+
requestId: "req-123",
|
|
286
|
+
resolution: "denied",
|
|
287
|
+
}),
|
|
288
|
+
);
|
|
289
|
+
});
|
|
290
|
+
|
|
291
|
+
it("logs permission_request.denied with denialReason when present", async () => {
|
|
292
|
+
const logger = { review: vi.fn() };
|
|
293
|
+
mockRequestApproval.mockResolvedValue({
|
|
294
|
+
approved: false,
|
|
295
|
+
state: "denied_with_reason",
|
|
296
|
+
denialReason: "too sensitive",
|
|
297
|
+
});
|
|
298
|
+
const deps = makeDeps({ logger });
|
|
299
|
+
const prompter = new PermissionPrompter(deps);
|
|
300
|
+
|
|
301
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
302
|
+
|
|
303
|
+
expect(logger.review).toHaveBeenCalledWith(
|
|
304
|
+
"permission_request.denied",
|
|
305
|
+
expect.objectContaining({
|
|
306
|
+
denialReason: "too sensitive",
|
|
307
|
+
}),
|
|
308
|
+
);
|
|
309
|
+
});
|
|
310
|
+
|
|
311
|
+
it("returns the decision from confirmPermission", async () => {
|
|
312
|
+
const decision: PermissionPromptDecision = {
|
|
313
|
+
approved: false,
|
|
314
|
+
state: "denied_with_reason",
|
|
315
|
+
denialReason: "sensitive",
|
|
316
|
+
};
|
|
317
|
+
mockRequestApproval.mockResolvedValue(decision);
|
|
318
|
+
const deps = makeDeps();
|
|
319
|
+
const prompter = new PermissionPrompter(deps);
|
|
320
|
+
|
|
321
|
+
const result = await prompter.prompt(makeCtx(true), makeDetails());
|
|
322
|
+
|
|
323
|
+
expect(result).toEqual(decision);
|
|
324
|
+
});
|
|
325
|
+
|
|
326
|
+
it("passes sessionLabel option to confirmPermission when present", async () => {
|
|
327
|
+
mockRequestApproval.mockResolvedValue({
|
|
328
|
+
approved: true,
|
|
329
|
+
state: "approved",
|
|
330
|
+
});
|
|
331
|
+
const deps = makeDeps();
|
|
332
|
+
const prompter = new PermissionPrompter(deps);
|
|
333
|
+
const details = makeDetails({ sessionLabel: "Yes, for 'read' tool" });
|
|
334
|
+
|
|
335
|
+
await prompter.prompt(makeCtx(true), details);
|
|
336
|
+
|
|
337
|
+
expect(mockRequestApproval).toHaveBeenCalledWith(
|
|
338
|
+
expect.anything(),
|
|
339
|
+
expect.any(String),
|
|
340
|
+
{ sessionLabel: "Yes, for 'read' tool" },
|
|
341
|
+
{ source: "tool_call", surface: "read", value: "read" },
|
|
342
|
+
);
|
|
343
|
+
});
|
|
344
|
+
|
|
345
|
+
it("passes the display fields (source/surface/value) to confirmPermission", async () => {
|
|
346
|
+
mockRequestApproval.mockResolvedValue({
|
|
347
|
+
approved: true,
|
|
348
|
+
state: "approved",
|
|
349
|
+
});
|
|
350
|
+
const deps = makeDeps();
|
|
351
|
+
const prompter = new PermissionPrompter(deps);
|
|
352
|
+
const details = makeDetails({
|
|
353
|
+
toolName: "bash",
|
|
354
|
+
command: "git push",
|
|
355
|
+
});
|
|
356
|
+
|
|
357
|
+
await prompter.prompt(makeCtx(false), details);
|
|
358
|
+
|
|
359
|
+
expect(mockRequestApproval).toHaveBeenCalledWith(
|
|
360
|
+
expect.anything(),
|
|
361
|
+
expect.any(String),
|
|
362
|
+
undefined,
|
|
363
|
+
{ source: "tool_call", surface: "bash", value: "git push" },
|
|
364
|
+
);
|
|
365
|
+
});
|
|
366
|
+
|
|
367
|
+
it("passes undefined options to confirmPermission when sessionLabel is absent", async () => {
|
|
368
|
+
mockRequestApproval.mockResolvedValue({
|
|
369
|
+
approved: true,
|
|
370
|
+
state: "approved",
|
|
371
|
+
});
|
|
372
|
+
const deps = makeDeps();
|
|
373
|
+
const prompter = new PermissionPrompter(deps);
|
|
374
|
+
|
|
375
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
376
|
+
|
|
377
|
+
expect(mockRequestApproval).toHaveBeenCalledWith(
|
|
378
|
+
expect.anything(),
|
|
379
|
+
expect.any(String),
|
|
380
|
+
undefined,
|
|
381
|
+
{ source: "tool_call", surface: "read", value: "read" },
|
|
382
|
+
);
|
|
383
|
+
});
|
|
384
|
+
|
|
385
|
+
it("passes the message from details to confirmPermission", async () => {
|
|
386
|
+
mockRequestApproval.mockResolvedValue({
|
|
387
|
+
approved: true,
|
|
388
|
+
state: "approved",
|
|
389
|
+
});
|
|
390
|
+
const deps = makeDeps();
|
|
391
|
+
const prompter = new PermissionPrompter(deps);
|
|
392
|
+
const details = makeDetails({ message: "Allow bash: git status?" });
|
|
393
|
+
|
|
394
|
+
await prompter.prompt(makeCtx(true), details);
|
|
395
|
+
|
|
396
|
+
expect(mockRequestApproval).toHaveBeenCalledWith(
|
|
397
|
+
expect.anything(),
|
|
398
|
+
"Allow bash: git status?",
|
|
399
|
+
undefined,
|
|
400
|
+
{ source: "tool_call", surface: "read", value: "read" },
|
|
401
|
+
);
|
|
402
|
+
});
|
|
403
|
+
});
|
|
404
|
+
|
|
405
|
+
// ── Review log field coverage ────────────────────────────────────────────
|
|
406
|
+
|
|
407
|
+
describe("review log fields", () => {
|
|
408
|
+
it("includes all standard fields in the waiting log entry", async () => {
|
|
409
|
+
const logger = { review: vi.fn() };
|
|
410
|
+
mockRequestApproval.mockResolvedValue({
|
|
411
|
+
approved: true,
|
|
412
|
+
state: "approved",
|
|
413
|
+
});
|
|
414
|
+
const deps = makeDeps({ logger });
|
|
415
|
+
const prompter = new PermissionPrompter(deps);
|
|
416
|
+
const details = makeDetails({
|
|
417
|
+
toolCallId: "tc-1",
|
|
418
|
+
skillName: "librarian",
|
|
419
|
+
path: "/src/foo.ts",
|
|
420
|
+
command: "git status",
|
|
421
|
+
target: "server:tool",
|
|
422
|
+
toolInputPreview: "{ path: '...' }",
|
|
423
|
+
});
|
|
424
|
+
|
|
425
|
+
await prompter.prompt(makeCtx(true), details);
|
|
426
|
+
|
|
427
|
+
expect(logger.review).toHaveBeenCalledWith(
|
|
428
|
+
"permission_request.waiting",
|
|
429
|
+
expect.objectContaining({
|
|
430
|
+
requestId: "req-123",
|
|
431
|
+
source: "tool_call",
|
|
432
|
+
agentName: "test-agent",
|
|
433
|
+
message: "Allow read?",
|
|
434
|
+
toolCallId: "tc-1",
|
|
435
|
+
toolName: "read",
|
|
436
|
+
skillName: "librarian",
|
|
437
|
+
path: "/src/foo.ts",
|
|
438
|
+
command: "git status",
|
|
439
|
+
target: "server:tool",
|
|
440
|
+
toolInputPreview: "{ path: '...' }",
|
|
441
|
+
}),
|
|
442
|
+
);
|
|
443
|
+
});
|
|
444
|
+
|
|
445
|
+
it("uses null for optional fields not present in details", async () => {
|
|
446
|
+
const logger = { review: vi.fn() };
|
|
447
|
+
mockRequestApproval.mockResolvedValue({
|
|
448
|
+
approved: true,
|
|
449
|
+
state: "approved",
|
|
450
|
+
});
|
|
451
|
+
const deps = makeDeps({ logger });
|
|
452
|
+
const prompter = new PermissionPrompter(deps);
|
|
453
|
+
|
|
454
|
+
await prompter.prompt(makeCtx(true), makeDetails());
|
|
455
|
+
|
|
456
|
+
expect(logger.review).toHaveBeenCalledWith(
|
|
457
|
+
"permission_request.waiting",
|
|
458
|
+
expect.objectContaining({
|
|
459
|
+
toolCallId: null,
|
|
460
|
+
skillName: null,
|
|
461
|
+
path: null,
|
|
462
|
+
command: null,
|
|
463
|
+
target: null,
|
|
464
|
+
toolInputPreview: null,
|
|
465
|
+
}),
|
|
466
|
+
);
|
|
467
|
+
});
|
|
468
|
+
});
|
|
469
|
+
|
|
470
|
+
// ── Subagent forwarding path ─────────────────────────────────────────────
|
|
471
|
+
|
|
472
|
+
describe("subagent forwarding path", () => {
|
|
473
|
+
it("calls confirmPermission even when ctx has no UI", async () => {
|
|
474
|
+
const forwarded: PermissionPromptDecision = {
|
|
475
|
+
approved: true,
|
|
476
|
+
state: "approved",
|
|
477
|
+
};
|
|
478
|
+
mockRequestApproval.mockResolvedValue(forwarded);
|
|
479
|
+
const deps = makeDeps();
|
|
480
|
+
const prompter = new PermissionPrompter(deps);
|
|
481
|
+
|
|
482
|
+
await prompter.prompt(makeCtx(false), makeDetails());
|
|
483
|
+
|
|
484
|
+
expect(mockRequestApproval).toHaveBeenCalled();
|
|
485
|
+
});
|
|
486
|
+
|
|
487
|
+
it("returns the decision from confirmPermission in the subagent path", async () => {
|
|
488
|
+
const forwarded: PermissionPromptDecision = {
|
|
489
|
+
approved: false,
|
|
490
|
+
state: "denied",
|
|
491
|
+
};
|
|
492
|
+
mockRequestApproval.mockResolvedValue(forwarded);
|
|
493
|
+
const deps = makeDeps();
|
|
494
|
+
const prompter = new PermissionPrompter(deps);
|
|
495
|
+
|
|
496
|
+
const result = await prompter.prompt(makeCtx(false), makeDetails());
|
|
497
|
+
|
|
498
|
+
expect(result).toEqual(forwarded);
|
|
499
|
+
});
|
|
500
|
+
|
|
501
|
+
it("logs the outcome when confirmPermission resolves via forwarding", async () => {
|
|
502
|
+
const logger = { review: vi.fn() };
|
|
503
|
+
mockRequestApproval.mockResolvedValue({
|
|
504
|
+
approved: true,
|
|
505
|
+
state: "approved",
|
|
506
|
+
});
|
|
507
|
+
const deps = makeDeps({ logger });
|
|
508
|
+
const prompter = new PermissionPrompter(deps);
|
|
509
|
+
|
|
510
|
+
await prompter.prompt(makeCtx(false), makeDetails());
|
|
511
|
+
|
|
512
|
+
expect(logger.review).toHaveBeenCalledWith(
|
|
513
|
+
"permission_request.approved",
|
|
514
|
+
expect.objectContaining({ requestId: "req-123" }),
|
|
515
|
+
);
|
|
516
|
+
});
|
|
517
|
+
});
|
|
518
|
+
});
|