@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,369 @@
|
|
|
1
|
+
import { mkdirSync, mkdtempSync, rmSync, writeFileSync } from "node:fs";
|
|
2
|
+
import { tmpdir } from "node:os";
|
|
3
|
+
import { join } from "node:path";
|
|
4
|
+
import { afterEach, describe, expect, test, vi } from "vitest";
|
|
5
|
+
import { DEFAULT_EXTENSION_CONFIG } from "#src/extension-config";
|
|
6
|
+
import {
|
|
7
|
+
type ForwarderContext,
|
|
8
|
+
PermissionForwarder,
|
|
9
|
+
type PermissionForwarderDeps,
|
|
10
|
+
} from "#src/forwarded-permissions/permission-forwarder";
|
|
11
|
+
import { createPermissionForwardingLocation } from "#src/permission-forwarding";
|
|
12
|
+
|
|
13
|
+
// ── Helpers ───────────────────────────────────────────────────────────────
|
|
14
|
+
|
|
15
|
+
function makeDeps(
|
|
16
|
+
overrides: Partial<PermissionForwarderDeps> = {},
|
|
17
|
+
): PermissionForwarderDeps {
|
|
18
|
+
return {
|
|
19
|
+
forwardingDir: "/tmp/forwarding",
|
|
20
|
+
subagentSessionsDir: "/tmp/subagents",
|
|
21
|
+
logger: { review: vi.fn(), debug: vi.fn() },
|
|
22
|
+
requestPermissionDecisionFromUi: vi
|
|
23
|
+
.fn()
|
|
24
|
+
.mockResolvedValue({ approved: true, state: "approved" as const }),
|
|
25
|
+
config: { current: () => ({ ...DEFAULT_EXTENSION_CONFIG }) },
|
|
26
|
+
...overrides,
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
function makeCtx(
|
|
31
|
+
overrides: {
|
|
32
|
+
hasUI?: boolean;
|
|
33
|
+
ui?: ForwarderContext["ui"];
|
|
34
|
+
sessionManager?: Partial<ForwarderContext["sessionManager"]>;
|
|
35
|
+
} = {},
|
|
36
|
+
): ForwarderContext {
|
|
37
|
+
return {
|
|
38
|
+
hasUI: overrides.hasUI ?? false,
|
|
39
|
+
ui: overrides.ui ?? { select: vi.fn(), input: vi.fn() },
|
|
40
|
+
sessionManager: {
|
|
41
|
+
getSessionId: vi.fn(() => ""),
|
|
42
|
+
getSessionDir: vi.fn(() => ""),
|
|
43
|
+
getEntries: vi.fn(() => []),
|
|
44
|
+
...overrides.sessionManager,
|
|
45
|
+
},
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
afterEach(() => {
|
|
50
|
+
vi.unstubAllEnvs();
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
// ── requestApproval ───────────────────────────────────────────────────────
|
|
54
|
+
|
|
55
|
+
describe("requestApproval — UI fast path", () => {
|
|
56
|
+
test("calls requestPermissionDecisionFromUi but does not emit a UI prompt event (the prompter does)", async () => {
|
|
57
|
+
const events = {
|
|
58
|
+
emit: vi.fn(),
|
|
59
|
+
on: vi.fn().mockReturnValue(() => undefined),
|
|
60
|
+
};
|
|
61
|
+
const requestPermissionDecisionFromUi = vi
|
|
62
|
+
.fn()
|
|
63
|
+
.mockResolvedValue({ approved: true, state: "approved" as const });
|
|
64
|
+
|
|
65
|
+
const forwarder = new PermissionForwarder(
|
|
66
|
+
makeDeps({ events, requestPermissionDecisionFromUi }),
|
|
67
|
+
);
|
|
68
|
+
|
|
69
|
+
await forwarder.requestApproval(
|
|
70
|
+
makeCtx({ hasUI: true }),
|
|
71
|
+
"Allow git push?",
|
|
72
|
+
);
|
|
73
|
+
|
|
74
|
+
expect(requestPermissionDecisionFromUi).toHaveBeenCalled();
|
|
75
|
+
expect(events.emit).not.toHaveBeenCalledWith(
|
|
76
|
+
"permissions:ui_prompt",
|
|
77
|
+
expect.anything(),
|
|
78
|
+
);
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
describe("requestApproval — non-UI, non-subagent path", () => {
|
|
83
|
+
test("returns denied without showing a dialog or emitting when there is no active UI", async () => {
|
|
84
|
+
const events = {
|
|
85
|
+
emit: vi.fn(),
|
|
86
|
+
on: vi.fn().mockReturnValue(() => undefined),
|
|
87
|
+
};
|
|
88
|
+
const requestPermissionDecisionFromUi = vi.fn();
|
|
89
|
+
|
|
90
|
+
const forwarder = new PermissionForwarder(
|
|
91
|
+
makeDeps({ events, requestPermissionDecisionFromUi }),
|
|
92
|
+
);
|
|
93
|
+
|
|
94
|
+
const result = await forwarder.requestApproval(
|
|
95
|
+
makeCtx({ hasUI: false }),
|
|
96
|
+
"Allow git push?",
|
|
97
|
+
);
|
|
98
|
+
|
|
99
|
+
expect(result).toEqual({ approved: false, state: "denied" });
|
|
100
|
+
expect(events.emit).not.toHaveBeenCalledWith(
|
|
101
|
+
"permissions:ui_prompt",
|
|
102
|
+
expect.anything(),
|
|
103
|
+
);
|
|
104
|
+
expect(requestPermissionDecisionFromUi).not.toHaveBeenCalled();
|
|
105
|
+
});
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
// ── processInbox ──────────────────────────────────────────────────────────
|
|
109
|
+
|
|
110
|
+
describe("processInbox", () => {
|
|
111
|
+
test("emits a UI prompt event before showing a forwarded permission dialog", async () => {
|
|
112
|
+
const root = mkdtempSync(join(tmpdir(), "permission-forwarding-"));
|
|
113
|
+
try {
|
|
114
|
+
const forwardingDir = join(root, "forwarding");
|
|
115
|
+
const location = createPermissionForwardingLocation(
|
|
116
|
+
forwardingDir,
|
|
117
|
+
"parent-session",
|
|
118
|
+
);
|
|
119
|
+
mkdirSync(location.requestsDir, { recursive: true });
|
|
120
|
+
mkdirSync(location.responsesDir, { recursive: true });
|
|
121
|
+
writeFileSync(
|
|
122
|
+
join(location.requestsDir, "req-forwarded.json"),
|
|
123
|
+
JSON.stringify({
|
|
124
|
+
id: "req-forwarded",
|
|
125
|
+
createdAt: Date.now(),
|
|
126
|
+
requesterSessionId: "child-session",
|
|
127
|
+
targetSessionId: "parent-session",
|
|
128
|
+
requesterAgentName: "Explore",
|
|
129
|
+
message: "Allow git push?",
|
|
130
|
+
}),
|
|
131
|
+
"utf-8",
|
|
132
|
+
);
|
|
133
|
+
|
|
134
|
+
const events = {
|
|
135
|
+
emit: vi.fn(),
|
|
136
|
+
on: vi.fn().mockReturnValue(() => undefined),
|
|
137
|
+
};
|
|
138
|
+
const requestPermissionDecisionFromUi = vi
|
|
139
|
+
.fn()
|
|
140
|
+
.mockResolvedValue({ approved: true, state: "approved" as const });
|
|
141
|
+
|
|
142
|
+
const forwarder = new PermissionForwarder(
|
|
143
|
+
makeDeps({
|
|
144
|
+
forwardingDir,
|
|
145
|
+
events,
|
|
146
|
+
requestPermissionDecisionFromUi,
|
|
147
|
+
}),
|
|
148
|
+
);
|
|
149
|
+
|
|
150
|
+
await forwarder.processInbox(
|
|
151
|
+
makeCtx({
|
|
152
|
+
hasUI: true,
|
|
153
|
+
sessionManager: {
|
|
154
|
+
getSessionId: vi.fn(() => "parent-session"),
|
|
155
|
+
},
|
|
156
|
+
}),
|
|
157
|
+
);
|
|
158
|
+
|
|
159
|
+
expect(events.emit).toHaveBeenCalledWith(
|
|
160
|
+
"permissions:ui_prompt",
|
|
161
|
+
expect.objectContaining({
|
|
162
|
+
requestId: "req-forwarded",
|
|
163
|
+
source: "tool_call",
|
|
164
|
+
surface: null,
|
|
165
|
+
value: null,
|
|
166
|
+
agentName: "Explore",
|
|
167
|
+
message: expect.stringContaining("Allow git push?"),
|
|
168
|
+
forwarding: {
|
|
169
|
+
requesterAgentName: "Explore",
|
|
170
|
+
requesterSessionId: "child-session",
|
|
171
|
+
},
|
|
172
|
+
}),
|
|
173
|
+
);
|
|
174
|
+
expect(requestPermissionDecisionFromUi).toHaveBeenCalled();
|
|
175
|
+
} finally {
|
|
176
|
+
rmSync(root, { recursive: true, force: true });
|
|
177
|
+
}
|
|
178
|
+
});
|
|
179
|
+
|
|
180
|
+
test("emits a non-degraded UI prompt event when the request carries display fields", async () => {
|
|
181
|
+
const root = mkdtempSync(join(tmpdir(), "permission-forwarding-"));
|
|
182
|
+
try {
|
|
183
|
+
const forwardingDir = join(root, "forwarding");
|
|
184
|
+
const location = createPermissionForwardingLocation(
|
|
185
|
+
forwardingDir,
|
|
186
|
+
"parent-session",
|
|
187
|
+
);
|
|
188
|
+
mkdirSync(location.requestsDir, { recursive: true });
|
|
189
|
+
mkdirSync(location.responsesDir, { recursive: true });
|
|
190
|
+
writeFileSync(
|
|
191
|
+
join(location.requestsDir, "req-forwarded-rich.json"),
|
|
192
|
+
JSON.stringify({
|
|
193
|
+
id: "req-forwarded-rich",
|
|
194
|
+
createdAt: Date.now(),
|
|
195
|
+
requesterSessionId: "child-session",
|
|
196
|
+
targetSessionId: "parent-session",
|
|
197
|
+
requesterAgentName: "Explore",
|
|
198
|
+
message: "Allow git push?",
|
|
199
|
+
source: "tool_call",
|
|
200
|
+
surface: "bash",
|
|
201
|
+
value: "git push",
|
|
202
|
+
}),
|
|
203
|
+
"utf-8",
|
|
204
|
+
);
|
|
205
|
+
|
|
206
|
+
const events = {
|
|
207
|
+
emit: vi.fn(),
|
|
208
|
+
on: vi.fn().mockReturnValue(() => undefined),
|
|
209
|
+
};
|
|
210
|
+
const requestPermissionDecisionFromUi = vi
|
|
211
|
+
.fn()
|
|
212
|
+
.mockResolvedValue({ approved: true, state: "approved" as const });
|
|
213
|
+
|
|
214
|
+
const forwarder = new PermissionForwarder(
|
|
215
|
+
makeDeps({
|
|
216
|
+
forwardingDir,
|
|
217
|
+
events,
|
|
218
|
+
requestPermissionDecisionFromUi,
|
|
219
|
+
}),
|
|
220
|
+
);
|
|
221
|
+
|
|
222
|
+
await forwarder.processInbox(
|
|
223
|
+
makeCtx({
|
|
224
|
+
hasUI: true,
|
|
225
|
+
sessionManager: {
|
|
226
|
+
getSessionId: vi.fn(() => "parent-session"),
|
|
227
|
+
},
|
|
228
|
+
}),
|
|
229
|
+
);
|
|
230
|
+
|
|
231
|
+
expect(events.emit).toHaveBeenCalledWith(
|
|
232
|
+
"permissions:ui_prompt",
|
|
233
|
+
expect.objectContaining({
|
|
234
|
+
requestId: "req-forwarded-rich",
|
|
235
|
+
source: "tool_call",
|
|
236
|
+
surface: "bash",
|
|
237
|
+
value: "git push",
|
|
238
|
+
agentName: "Explore",
|
|
239
|
+
message: expect.stringContaining("Allow git push?"),
|
|
240
|
+
forwarding: {
|
|
241
|
+
requesterAgentName: "Explore",
|
|
242
|
+
requesterSessionId: "child-session",
|
|
243
|
+
},
|
|
244
|
+
}),
|
|
245
|
+
);
|
|
246
|
+
expect(requestPermissionDecisionFromUi).toHaveBeenCalled();
|
|
247
|
+
} finally {
|
|
248
|
+
rmSync(root, { recursive: true, force: true });
|
|
249
|
+
}
|
|
250
|
+
});
|
|
251
|
+
|
|
252
|
+
test("does not emit a UI prompt event when forwarded permission auto-approves", async () => {
|
|
253
|
+
const root = mkdtempSync(join(tmpdir(), "permission-forwarding-"));
|
|
254
|
+
try {
|
|
255
|
+
const forwardingDir = join(root, "forwarding");
|
|
256
|
+
const location = createPermissionForwardingLocation(
|
|
257
|
+
forwardingDir,
|
|
258
|
+
"parent-session",
|
|
259
|
+
);
|
|
260
|
+
mkdirSync(location.requestsDir, { recursive: true });
|
|
261
|
+
mkdirSync(location.responsesDir, { recursive: true });
|
|
262
|
+
writeFileSync(
|
|
263
|
+
join(location.requestsDir, "req-forwarded-auto.json"),
|
|
264
|
+
JSON.stringify({
|
|
265
|
+
id: "req-forwarded-auto",
|
|
266
|
+
createdAt: Date.now(),
|
|
267
|
+
requesterSessionId: "child-session",
|
|
268
|
+
targetSessionId: "parent-session",
|
|
269
|
+
requesterAgentName: "Explore",
|
|
270
|
+
message: "Allow git push?",
|
|
271
|
+
}),
|
|
272
|
+
"utf-8",
|
|
273
|
+
);
|
|
274
|
+
|
|
275
|
+
const events = {
|
|
276
|
+
emit: vi.fn(),
|
|
277
|
+
on: vi.fn().mockReturnValue(() => undefined),
|
|
278
|
+
};
|
|
279
|
+
const requestPermissionDecisionFromUi = vi.fn();
|
|
280
|
+
|
|
281
|
+
const forwarder = new PermissionForwarder(
|
|
282
|
+
makeDeps({
|
|
283
|
+
forwardingDir,
|
|
284
|
+
events,
|
|
285
|
+
requestPermissionDecisionFromUi,
|
|
286
|
+
config: {
|
|
287
|
+
current: () => ({ ...DEFAULT_EXTENSION_CONFIG, yoloMode: true }),
|
|
288
|
+
},
|
|
289
|
+
}),
|
|
290
|
+
);
|
|
291
|
+
|
|
292
|
+
await forwarder.processInbox(
|
|
293
|
+
makeCtx({
|
|
294
|
+
hasUI: true,
|
|
295
|
+
sessionManager: {
|
|
296
|
+
getSessionId: vi.fn(() => "parent-session"),
|
|
297
|
+
},
|
|
298
|
+
}),
|
|
299
|
+
);
|
|
300
|
+
|
|
301
|
+
expect(events.emit).not.toHaveBeenCalledWith(
|
|
302
|
+
"permissions:ui_prompt",
|
|
303
|
+
expect.anything(),
|
|
304
|
+
);
|
|
305
|
+
expect(requestPermissionDecisionFromUi).not.toHaveBeenCalled();
|
|
306
|
+
} finally {
|
|
307
|
+
rmSync(root, { recursive: true, force: true });
|
|
308
|
+
}
|
|
309
|
+
});
|
|
310
|
+
|
|
311
|
+
test("recreates a missing responses/ directory and still writes the response", async () => {
|
|
312
|
+
const root = mkdtempSync(join(tmpdir(), "permission-forwarding-"));
|
|
313
|
+
try {
|
|
314
|
+
const forwardingDir = join(root, "forwarding");
|
|
315
|
+
const location = createPermissionForwardingLocation(
|
|
316
|
+
forwardingDir,
|
|
317
|
+
"parent-session",
|
|
318
|
+
);
|
|
319
|
+
// Simulate the race: requests/ exists with a pending file, but
|
|
320
|
+
// responses/ was removed by a concurrent cleanup pass.
|
|
321
|
+
mkdirSync(location.requestsDir, { recursive: true });
|
|
322
|
+
// Deliberately do NOT create location.responsesDir.
|
|
323
|
+
writeFileSync(
|
|
324
|
+
join(location.requestsDir, "req-race.json"),
|
|
325
|
+
JSON.stringify({
|
|
326
|
+
id: "req-race",
|
|
327
|
+
createdAt: Date.now(),
|
|
328
|
+
requesterSessionId: "child-session",
|
|
329
|
+
targetSessionId: "parent-session",
|
|
330
|
+
requesterAgentName: "Explore",
|
|
331
|
+
message: "Allow read?",
|
|
332
|
+
}),
|
|
333
|
+
"utf-8",
|
|
334
|
+
);
|
|
335
|
+
|
|
336
|
+
const logger = { review: vi.fn(), debug: vi.fn() };
|
|
337
|
+
const requestPermissionDecisionFromUi = vi
|
|
338
|
+
.fn()
|
|
339
|
+
.mockResolvedValue({ approved: true, state: "approved" as const });
|
|
340
|
+
|
|
341
|
+
const forwarder = new PermissionForwarder(
|
|
342
|
+
makeDeps({
|
|
343
|
+
forwardingDir,
|
|
344
|
+
logger,
|
|
345
|
+
requestPermissionDecisionFromUi,
|
|
346
|
+
}),
|
|
347
|
+
);
|
|
348
|
+
|
|
349
|
+
await forwarder.processInbox(
|
|
350
|
+
makeCtx({
|
|
351
|
+
hasUI: true,
|
|
352
|
+
sessionManager: {
|
|
353
|
+
getSessionId: vi.fn(() => "parent-session"),
|
|
354
|
+
},
|
|
355
|
+
}),
|
|
356
|
+
);
|
|
357
|
+
|
|
358
|
+
// processInbox must have recreated responses/ and written a response
|
|
359
|
+
// file — no permission_forwarding.error should have been logged.
|
|
360
|
+
expect(logger.review).not.toHaveBeenCalledWith(
|
|
361
|
+
"permission_forwarding.error",
|
|
362
|
+
expect.anything(),
|
|
363
|
+
);
|
|
364
|
+
expect(requestPermissionDecisionFromUi).toHaveBeenCalled();
|
|
365
|
+
} finally {
|
|
366
|
+
rmSync(root, { recursive: true, force: true });
|
|
367
|
+
}
|
|
368
|
+
});
|
|
369
|
+
});
|
|
@@ -0,0 +1,315 @@
|
|
|
1
|
+
import { tmpdir } from "node:os";
|
|
2
|
+
import { join } from "node:path";
|
|
3
|
+
import { afterEach, describe, expect, test, vi } from "vitest";
|
|
4
|
+
import {
|
|
5
|
+
createPermissionForwardingLocation,
|
|
6
|
+
isForwardedPermissionRequestForSession,
|
|
7
|
+
resolvePermissionForwardingTargetSessionId,
|
|
8
|
+
SUBAGENT_PARENT_SESSION_ENV_CANDIDATES,
|
|
9
|
+
SUBAGENT_PARENT_SESSION_ENV_KEY,
|
|
10
|
+
} from "#src/permission-forwarding";
|
|
11
|
+
import { SubagentSessionRegistry } from "#src/subagent-registry";
|
|
12
|
+
|
|
13
|
+
afterEach(() => {
|
|
14
|
+
vi.unstubAllEnvs();
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
describe("SUBAGENT_PARENT_SESSION_ENV_CANDIDATES", () => {
|
|
18
|
+
test("is an array containing PI_AGENT_ROUTER_PARENT_SESSION_ID", () => {
|
|
19
|
+
expect(Array.isArray(SUBAGENT_PARENT_SESSION_ENV_CANDIDATES)).toBe(true);
|
|
20
|
+
expect(SUBAGENT_PARENT_SESSION_ENV_CANDIDATES).toContain(
|
|
21
|
+
"PI_AGENT_ROUTER_PARENT_SESSION_ID",
|
|
22
|
+
);
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
test("contains PI_SUBAGENT_PARENT_SESSION for CLI-based subagent extensions", () => {
|
|
26
|
+
expect(SUBAGENT_PARENT_SESSION_ENV_CANDIDATES).toContain(
|
|
27
|
+
"PI_SUBAGENT_PARENT_SESSION",
|
|
28
|
+
);
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
test("deprecated SUBAGENT_PARENT_SESSION_ENV_KEY equals the first candidate", () => {
|
|
32
|
+
// eslint-disable-next-line @typescript-eslint/no-deprecated -- test verifying the deprecated alias
|
|
33
|
+
expect(SUBAGENT_PARENT_SESSION_ENV_KEY).toBe(
|
|
34
|
+
SUBAGENT_PARENT_SESSION_ENV_CANDIDATES[0],
|
|
35
|
+
);
|
|
36
|
+
});
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
describe("resolvePermissionForwardingTargetSessionId", () => {
|
|
40
|
+
test("hasUI=true returns the current session ID (UI host owns forwarding)", () => {
|
|
41
|
+
expect(
|
|
42
|
+
resolvePermissionForwardingTargetSessionId({
|
|
43
|
+
hasUI: true,
|
|
44
|
+
isSubagent: false,
|
|
45
|
+
currentSessionId: "parent-session-abc",
|
|
46
|
+
env: {},
|
|
47
|
+
}),
|
|
48
|
+
).toBe("parent-session-abc");
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
test("hasUI=true with isSubagent=true still returns current session ID", () => {
|
|
52
|
+
expect(
|
|
53
|
+
resolvePermissionForwardingTargetSessionId({
|
|
54
|
+
hasUI: true,
|
|
55
|
+
isSubagent: true,
|
|
56
|
+
currentSessionId: "session-xyz",
|
|
57
|
+
env: { PI_AGENT_ROUTER_PARENT_SESSION_ID: "other" },
|
|
58
|
+
}),
|
|
59
|
+
).toBe("session-xyz");
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
test("hasUI=false, isSubagent=false returns null", () => {
|
|
63
|
+
expect(
|
|
64
|
+
resolvePermissionForwardingTargetSessionId({
|
|
65
|
+
hasUI: false,
|
|
66
|
+
isSubagent: false,
|
|
67
|
+
currentSessionId: "session-xyz",
|
|
68
|
+
env: { PI_AGENT_ROUTER_PARENT_SESSION_ID: "parent-session-abc" },
|
|
69
|
+
}),
|
|
70
|
+
).toBeNull();
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
test("isSubagent=true, no candidates set returns null", () => {
|
|
74
|
+
expect(
|
|
75
|
+
resolvePermissionForwardingTargetSessionId({
|
|
76
|
+
hasUI: false,
|
|
77
|
+
isSubagent: true,
|
|
78
|
+
currentSessionId: "session-xyz",
|
|
79
|
+
env: {},
|
|
80
|
+
}),
|
|
81
|
+
).toBeNull();
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
test("isSubagent=true, PI_AGENT_ROUTER_PARENT_SESSION_ID set returns its value", () => {
|
|
85
|
+
expect(
|
|
86
|
+
resolvePermissionForwardingTargetSessionId({
|
|
87
|
+
hasUI: false,
|
|
88
|
+
isSubagent: true,
|
|
89
|
+
currentSessionId: "session-xyz",
|
|
90
|
+
env: { PI_AGENT_ROUTER_PARENT_SESSION_ID: "parent-session-abc" },
|
|
91
|
+
}),
|
|
92
|
+
).toBe("parent-session-abc");
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
test("isSubagent=true, PI_SUBAGENT_PARENT_SESSION resolves when PI_AGENT_ROUTER_PARENT_SESSION_ID is absent", () => {
|
|
96
|
+
expect(
|
|
97
|
+
resolvePermissionForwardingTargetSessionId({
|
|
98
|
+
hasUI: false,
|
|
99
|
+
isSubagent: true,
|
|
100
|
+
currentSessionId: "session-xyz",
|
|
101
|
+
env: {
|
|
102
|
+
PI_SUBAGENT_PARENT_SESSION: "parent-from-convention",
|
|
103
|
+
},
|
|
104
|
+
}),
|
|
105
|
+
).toBe("parent-from-convention");
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
test("isSubagent=true, PI_AGENT_ROUTER_PARENT_SESSION_ID takes precedence over PI_SUBAGENT_PARENT_SESSION", () => {
|
|
109
|
+
expect(
|
|
110
|
+
resolvePermissionForwardingTargetSessionId({
|
|
111
|
+
hasUI: false,
|
|
112
|
+
isSubagent: true,
|
|
113
|
+
currentSessionId: "session-xyz",
|
|
114
|
+
env: {
|
|
115
|
+
PI_AGENT_ROUTER_PARENT_SESSION_ID: "parent-from-router",
|
|
116
|
+
PI_SUBAGENT_PARENT_SESSION: "parent-from-convention",
|
|
117
|
+
},
|
|
118
|
+
}),
|
|
119
|
+
).toBe("parent-from-router");
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
test("isSubagent=true, candidate value is empty string returns null", () => {
|
|
123
|
+
expect(
|
|
124
|
+
resolvePermissionForwardingTargetSessionId({
|
|
125
|
+
hasUI: false,
|
|
126
|
+
isSubagent: true,
|
|
127
|
+
currentSessionId: "session-xyz",
|
|
128
|
+
env: { PI_AGENT_ROUTER_PARENT_SESSION_ID: "" },
|
|
129
|
+
}),
|
|
130
|
+
).toBeNull();
|
|
131
|
+
});
|
|
132
|
+
|
|
133
|
+
test("isSubagent=true, candidate value is 'unknown' returns null", () => {
|
|
134
|
+
expect(
|
|
135
|
+
resolvePermissionForwardingTargetSessionId({
|
|
136
|
+
hasUI: false,
|
|
137
|
+
isSubagent: true,
|
|
138
|
+
currentSessionId: "session-xyz",
|
|
139
|
+
env: { PI_AGENT_ROUTER_PARENT_SESSION_ID: "unknown" },
|
|
140
|
+
}),
|
|
141
|
+
).toBeNull();
|
|
142
|
+
});
|
|
143
|
+
|
|
144
|
+
test("env defaults to process.env when omitted", () => {
|
|
145
|
+
vi.stubEnv("PI_AGENT_ROUTER_PARENT_SESSION_ID", "env-session-abc");
|
|
146
|
+
expect(
|
|
147
|
+
resolvePermissionForwardingTargetSessionId({
|
|
148
|
+
hasUI: false,
|
|
149
|
+
isSubagent: true,
|
|
150
|
+
}),
|
|
151
|
+
).toBe("env-session-abc");
|
|
152
|
+
});
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
describe("resolvePermissionForwardingTargetSessionId — registry resolution", () => {
|
|
156
|
+
const childSessionId = "child-session-abc";
|
|
157
|
+
|
|
158
|
+
test("returns parentSessionId from registry when env vars are absent", () => {
|
|
159
|
+
const registry = new SubagentSessionRegistry();
|
|
160
|
+
registry.register(childSessionId, {
|
|
161
|
+
parentSessionId: "parent-from-registry",
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
expect(
|
|
165
|
+
resolvePermissionForwardingTargetSessionId({
|
|
166
|
+
hasUI: false,
|
|
167
|
+
isSubagent: true,
|
|
168
|
+
sessionId: childSessionId,
|
|
169
|
+
registry,
|
|
170
|
+
env: {},
|
|
171
|
+
}),
|
|
172
|
+
).toBe("parent-from-registry");
|
|
173
|
+
});
|
|
174
|
+
|
|
175
|
+
test("registry takes priority over env vars", () => {
|
|
176
|
+
const registry = new SubagentSessionRegistry();
|
|
177
|
+
registry.register(childSessionId, {
|
|
178
|
+
parentSessionId: "parent-from-registry",
|
|
179
|
+
});
|
|
180
|
+
|
|
181
|
+
expect(
|
|
182
|
+
resolvePermissionForwardingTargetSessionId({
|
|
183
|
+
hasUI: false,
|
|
184
|
+
isSubagent: true,
|
|
185
|
+
sessionId: childSessionId,
|
|
186
|
+
registry,
|
|
187
|
+
env: { PI_AGENT_ROUTER_PARENT_SESSION_ID: "parent-from-env" },
|
|
188
|
+
}),
|
|
189
|
+
).toBe("parent-from-registry");
|
|
190
|
+
});
|
|
191
|
+
|
|
192
|
+
test("falls through to env vars when registry entry has no parentSessionId", () => {
|
|
193
|
+
const registry = new SubagentSessionRegistry();
|
|
194
|
+
registry.register(childSessionId, {}); // no parentSessionId
|
|
195
|
+
|
|
196
|
+
expect(
|
|
197
|
+
resolvePermissionForwardingTargetSessionId({
|
|
198
|
+
hasUI: false,
|
|
199
|
+
isSubagent: true,
|
|
200
|
+
sessionId: childSessionId,
|
|
201
|
+
registry,
|
|
202
|
+
env: { PI_AGENT_ROUTER_PARENT_SESSION_ID: "parent-from-env" },
|
|
203
|
+
}),
|
|
204
|
+
).toBe("parent-from-env");
|
|
205
|
+
});
|
|
206
|
+
|
|
207
|
+
test("falls through to env vars when sessionId is not in registry", () => {
|
|
208
|
+
const registry = new SubagentSessionRegistry(); // empty
|
|
209
|
+
|
|
210
|
+
expect(
|
|
211
|
+
resolvePermissionForwardingTargetSessionId({
|
|
212
|
+
hasUI: false,
|
|
213
|
+
isSubagent: true,
|
|
214
|
+
sessionId: childSessionId,
|
|
215
|
+
registry,
|
|
216
|
+
env: { PI_AGENT_ROUTER_PARENT_SESSION_ID: "parent-from-env" },
|
|
217
|
+
}),
|
|
218
|
+
).toBe("parent-from-env");
|
|
219
|
+
});
|
|
220
|
+
|
|
221
|
+
test("returns null when registry entry has no parentSessionId and no env vars set", () => {
|
|
222
|
+
const registry = new SubagentSessionRegistry();
|
|
223
|
+
registry.register(childSessionId, {}); // no parentSessionId
|
|
224
|
+
|
|
225
|
+
expect(
|
|
226
|
+
resolvePermissionForwardingTargetSessionId({
|
|
227
|
+
hasUI: false,
|
|
228
|
+
isSubagent: true,
|
|
229
|
+
sessionId: childSessionId,
|
|
230
|
+
registry,
|
|
231
|
+
env: {},
|
|
232
|
+
}),
|
|
233
|
+
).toBeNull();
|
|
234
|
+
});
|
|
235
|
+
|
|
236
|
+
test("omitting registry preserves existing behaviour", () => {
|
|
237
|
+
expect(
|
|
238
|
+
resolvePermissionForwardingTargetSessionId({
|
|
239
|
+
hasUI: false,
|
|
240
|
+
isSubagent: true,
|
|
241
|
+
sessionId: childSessionId,
|
|
242
|
+
env: { PI_AGENT_ROUTER_PARENT_SESSION_ID: "parent-from-env" },
|
|
243
|
+
}),
|
|
244
|
+
).toBe("parent-from-env");
|
|
245
|
+
});
|
|
246
|
+
});
|
|
247
|
+
|
|
248
|
+
// ---------------------------------------------------------------------------
|
|
249
|
+
// Moved from permission-system.test.ts catch-all (#342)
|
|
250
|
+
// ---------------------------------------------------------------------------
|
|
251
|
+
|
|
252
|
+
test("Permission forwarding resolves the parent interactive session from subagent runtime env", () => {
|
|
253
|
+
const targetSessionId = resolvePermissionForwardingTargetSessionId({
|
|
254
|
+
hasUI: false,
|
|
255
|
+
isSubagent: true,
|
|
256
|
+
currentSessionId: "child-session",
|
|
257
|
+
env: {
|
|
258
|
+
PI_AGENT_ROUTER_PARENT_SESSION_ID: "parent-session",
|
|
259
|
+
},
|
|
260
|
+
});
|
|
261
|
+
|
|
262
|
+
expect(targetSessionId).toBe("parent-session");
|
|
263
|
+
});
|
|
264
|
+
|
|
265
|
+
test("Permission forwarding does not guess a target session when subagent runtime env is missing", () => {
|
|
266
|
+
const targetSessionId = resolvePermissionForwardingTargetSessionId({
|
|
267
|
+
hasUI: false,
|
|
268
|
+
isSubagent: true,
|
|
269
|
+
currentSessionId: "child-session",
|
|
270
|
+
env: {},
|
|
271
|
+
});
|
|
272
|
+
|
|
273
|
+
expect(targetSessionId).toBe(null);
|
|
274
|
+
});
|
|
275
|
+
|
|
276
|
+
test("Permission forwarding uses session-scoped directories per interactive session", () => {
|
|
277
|
+
const forwardingRoot = join(tmpdir(), "pi-permission-system-forwarding-root");
|
|
278
|
+
const sessionA = createPermissionForwardingLocation(
|
|
279
|
+
forwardingRoot,
|
|
280
|
+
"session-a",
|
|
281
|
+
);
|
|
282
|
+
const sessionB = createPermissionForwardingLocation(
|
|
283
|
+
forwardingRoot,
|
|
284
|
+
"session-b",
|
|
285
|
+
);
|
|
286
|
+
|
|
287
|
+
expect(sessionA.sessionRootDir).not.toBe(sessionB.sessionRootDir);
|
|
288
|
+
expect(sessionA.requestsDir).not.toBe(sessionB.requestsDir);
|
|
289
|
+
expect(sessionA.responsesDir).not.toBe(sessionB.responsesDir);
|
|
290
|
+
});
|
|
291
|
+
|
|
292
|
+
test("Permission forwarding request routing only matches the intended UI session", () => {
|
|
293
|
+
expect(
|
|
294
|
+
isForwardedPermissionRequestForSession(
|
|
295
|
+
{ targetSessionId: "session-a" },
|
|
296
|
+
"session-a",
|
|
297
|
+
),
|
|
298
|
+
).toBe(true);
|
|
299
|
+
expect(
|
|
300
|
+
isForwardedPermissionRequestForSession(
|
|
301
|
+
{ targetSessionId: "session-a" },
|
|
302
|
+
"session-b",
|
|
303
|
+
),
|
|
304
|
+
).toBe(false);
|
|
305
|
+
});
|
|
306
|
+
|
|
307
|
+
test("Permission forwarding rejects unresolved sentinel session ids", () => {
|
|
308
|
+
const targetSessionId = resolvePermissionForwardingTargetSessionId({
|
|
309
|
+
hasUI: true,
|
|
310
|
+
isSubagent: false,
|
|
311
|
+
currentSessionId: "unknown",
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
expect(targetSessionId).toBe(null);
|
|
315
|
+
});
|