@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.
Files changed (203) hide show
  1. package/CHANGELOG.md +2234 -0
  2. package/LICENSE +21 -0
  3. package/README.md +158 -0
  4. package/config/config.example.json +39 -0
  5. package/package.json +82 -0
  6. package/schemas/permissions.schema.json +158 -0
  7. package/src/active-agent.ts +72 -0
  8. package/src/async-cache.ts +21 -0
  9. package/src/bash-arity.ts +210 -0
  10. package/src/builtin-tool-input-formatters.ts +82 -0
  11. package/src/canonicalize-path.ts +30 -0
  12. package/src/common.ts +121 -0
  13. package/src/config-loader.ts +432 -0
  14. package/src/config-modal.ts +259 -0
  15. package/src/config-paths.ts +47 -0
  16. package/src/config-reporter.ts +34 -0
  17. package/src/config-store.ts +222 -0
  18. package/src/decision-audit.ts +75 -0
  19. package/src/decision-reporter.ts +41 -0
  20. package/src/denial-messages.ts +232 -0
  21. package/src/expand-home.ts +28 -0
  22. package/src/extension-config.ts +79 -0
  23. package/src/extension-paths.ts +66 -0
  24. package/src/forwarded-permissions/io.ts +404 -0
  25. package/src/forwarded-permissions/permission-forwarder.ts +580 -0
  26. package/src/forwarding-manager.ts +74 -0
  27. package/src/gate-prompter.ts +12 -0
  28. package/src/handlers/before-agent-start.ts +94 -0
  29. package/src/handlers/gates/bash-command.ts +75 -0
  30. package/src/handlers/gates/bash-external-directory.ts +127 -0
  31. package/src/handlers/gates/bash-path-extractor.ts +15 -0
  32. package/src/handlers/gates/bash-path.ts +152 -0
  33. package/src/handlers/gates/bash-program.ts +1143 -0
  34. package/src/handlers/gates/bash-token-classification.ts +105 -0
  35. package/src/handlers/gates/candidate-check.ts +32 -0
  36. package/src/handlers/gates/descriptor.ts +81 -0
  37. package/src/handlers/gates/external-directory-messages.ts +20 -0
  38. package/src/handlers/gates/external-directory.ts +133 -0
  39. package/src/handlers/gates/helpers.ts +76 -0
  40. package/src/handlers/gates/path.ts +91 -0
  41. package/src/handlers/gates/runner.ts +186 -0
  42. package/src/handlers/gates/skill-input-gate-pipeline.ts +104 -0
  43. package/src/handlers/gates/skill-input.ts +46 -0
  44. package/src/handlers/gates/skill-read.ts +87 -0
  45. package/src/handlers/gates/tool-call-gate-pipeline.ts +129 -0
  46. package/src/handlers/gates/tool.ts +102 -0
  47. package/src/handlers/gates/types.ts +13 -0
  48. package/src/handlers/index.ts +3 -0
  49. package/src/handlers/lifecycle.ts +95 -0
  50. package/src/handlers/permission-gate-handler.ts +190 -0
  51. package/src/handlers/tool-call-boundary.ts +91 -0
  52. package/src/index.ts +225 -0
  53. package/src/input-normalizer.ts +157 -0
  54. package/src/logging.ts +113 -0
  55. package/src/mcp-targets.ts +170 -0
  56. package/src/node-modules-discovery.ts +76 -0
  57. package/src/normalize.ts +43 -0
  58. package/src/path-utils.ts +355 -0
  59. package/src/pattern-suggest.ts +132 -0
  60. package/src/permission-dialog.ts +138 -0
  61. package/src/permission-event-rpc.ts +223 -0
  62. package/src/permission-events.ts +266 -0
  63. package/src/permission-forwarding.ts +188 -0
  64. package/src/permission-gate.ts +94 -0
  65. package/src/permission-manager.ts +392 -0
  66. package/src/permission-merge.ts +32 -0
  67. package/src/permission-prompter.ts +142 -0
  68. package/src/permission-prompts.ts +93 -0
  69. package/src/permission-resolver.ts +109 -0
  70. package/src/permission-session.ts +189 -0
  71. package/src/permission-ui-prompt.ts +127 -0
  72. package/src/permissions-service.ts +63 -0
  73. package/src/persistent-approval-recorder.ts +139 -0
  74. package/src/policy-loader.ts +350 -0
  75. package/src/prompting-gateway.ts +104 -0
  76. package/src/rule.ts +188 -0
  77. package/src/scope-merge.ts +72 -0
  78. package/src/service-lifecycle.ts +49 -0
  79. package/src/service.ts +163 -0
  80. package/src/session-approval-recorder.ts +6 -0
  81. package/src/session-approval.ts +43 -0
  82. package/src/session-logger.ts +91 -0
  83. package/src/session-rules.ts +79 -0
  84. package/src/skill-prompt-sanitizer.ts +292 -0
  85. package/src/status.ts +35 -0
  86. package/src/subagent-context.ts +104 -0
  87. package/src/subagent-lifecycle-events.ts +72 -0
  88. package/src/subagent-registry.ts +105 -0
  89. package/src/synthesize.ts +92 -0
  90. package/src/system-prompt-sanitizer.ts +274 -0
  91. package/src/tool-access-extractor-registry.ts +68 -0
  92. package/src/tool-input-formatter-registry.ts +67 -0
  93. package/src/tool-input-preview.ts +34 -0
  94. package/src/tool-input-prompt-formatters.ts +63 -0
  95. package/src/tool-preview-formatter.ts +207 -0
  96. package/src/tool-registry.ts +148 -0
  97. package/src/types.ts +64 -0
  98. package/src/wildcard-matcher.ts +120 -0
  99. package/src/yolo-mode.ts +30 -0
  100. package/test/active-agent.test.ts +155 -0
  101. package/test/async-cache.test.ts +48 -0
  102. package/test/bash-arity.test.ts +144 -0
  103. package/test/bash-external-directory.test.ts +956 -0
  104. package/test/builtin-tool-input-formatters.test.ts +109 -0
  105. package/test/canonicalize-path.test.ts +93 -0
  106. package/test/common.test.ts +287 -0
  107. package/test/composition-root.test.ts +603 -0
  108. package/test/config-loader.test.ts +740 -0
  109. package/test/config-modal.test.ts +320 -0
  110. package/test/config-paths.test.ts +83 -0
  111. package/test/config-pipeline.test.ts +90 -0
  112. package/test/config-reporter.test.ts +147 -0
  113. package/test/config-store.test.ts +466 -0
  114. package/test/decision-audit.test.ts +72 -0
  115. package/test/decision-reporter.test.ts +112 -0
  116. package/test/denial-messages.test.ts +656 -0
  117. package/test/detect-permissive-bash-fallback.test.ts +56 -0
  118. package/test/expand-home.test.ts +93 -0
  119. package/test/extension-config.test.ts +129 -0
  120. package/test/extension-paths.test.ts +108 -0
  121. package/test/forwarded-permissions/io.test.ts +251 -0
  122. package/test/forwarding-manager.test.ts +194 -0
  123. package/test/handlers/before-agent-start.test.ts +317 -0
  124. package/test/handlers/external-directory-integration.test.ts +623 -0
  125. package/test/handlers/external-directory-session-dedup.test.ts +430 -0
  126. package/test/handlers/external-directory-symlink-acceptance.test.ts +149 -0
  127. package/test/handlers/gates/bash-command-metamorphic.test.ts +83 -0
  128. package/test/handlers/gates/bash-command.test.ts +191 -0
  129. package/test/handlers/gates/bash-external-directory.test.ts +269 -0
  130. package/test/handlers/gates/bash-path.test.ts +337 -0
  131. package/test/handlers/gates/bash-program.test.ts +410 -0
  132. package/test/handlers/gates/bash-token-classification.test.ts +241 -0
  133. package/test/handlers/gates/candidate-check.test.ts +52 -0
  134. package/test/handlers/gates/external-directory-messages.test.ts +61 -0
  135. package/test/handlers/gates/external-directory.test.ts +259 -0
  136. package/test/handlers/gates/helpers.test.ts +177 -0
  137. package/test/handlers/gates/path.test.ts +294 -0
  138. package/test/handlers/gates/runner.test.ts +447 -0
  139. package/test/handlers/gates/skill-input-gate-pipeline.test.ts +176 -0
  140. package/test/handlers/gates/skill-input.test.ts +131 -0
  141. package/test/handlers/gates/skill-read.test.ts +158 -0
  142. package/test/handlers/gates/tool-call-gate-pipeline.test.ts +252 -0
  143. package/test/handlers/gates/tool.test.ts +223 -0
  144. package/test/handlers/input-events.test.ts +168 -0
  145. package/test/handlers/input.test.ts +199 -0
  146. package/test/handlers/lifecycle.test.ts +221 -0
  147. package/test/handlers/tool-call-boundary.test.ts +145 -0
  148. package/test/handlers/tool-call-events.test.ts +277 -0
  149. package/test/handlers/tool-call.test.ts +395 -0
  150. package/test/handlers/validate-requested-tool.test.ts +92 -0
  151. package/test/helpers/gate-fixtures.ts +323 -0
  152. package/test/helpers/handler-fixtures.ts +335 -0
  153. package/test/helpers/make-fake-pi.ts +100 -0
  154. package/test/helpers/manager-harness.ts +112 -0
  155. package/test/helpers/session-fixtures.ts +204 -0
  156. package/test/input-normalizer.test.ts +367 -0
  157. package/test/logging.test.ts +51 -0
  158. package/test/mcp-targets.test.ts +233 -0
  159. package/test/node-modules-discovery.test.ts +97 -0
  160. package/test/normalize.test.ts +247 -0
  161. package/test/path-utils.test.ts +650 -0
  162. package/test/pattern-suggest.test.ts +248 -0
  163. package/test/permission-dialog.test.ts +241 -0
  164. package/test/permission-event-rpc.test.ts +541 -0
  165. package/test/permission-events.test.ts +402 -0
  166. package/test/permission-forwarder.test.ts +369 -0
  167. package/test/permission-forwarding.test.ts +315 -0
  168. package/test/permission-gate.test.ts +305 -0
  169. package/test/permission-manager-unified.test.ts +3368 -0
  170. package/test/permission-merge.test.ts +61 -0
  171. package/test/permission-prompter.test.ts +518 -0
  172. package/test/permission-prompts.test.ts +363 -0
  173. package/test/permission-resolver.test.ts +265 -0
  174. package/test/permission-session.test.ts +363 -0
  175. package/test/permission-ui-prompt.test.ts +146 -0
  176. package/test/permissions-service.test.ts +177 -0
  177. package/test/persistent-approval-recorder.test.ts +133 -0
  178. package/test/pi-infrastructure-read.test.ts +369 -0
  179. package/test/policy-loader.test.ts +561 -0
  180. package/test/prompting-gateway.test.ts +230 -0
  181. package/test/rule.test.ts +604 -0
  182. package/test/scope-merge.test.ts +116 -0
  183. package/test/service-lifecycle.test.ts +163 -0
  184. package/test/service.test.ts +308 -0
  185. package/test/session-approval.test.ts +75 -0
  186. package/test/session-logger.test.ts +200 -0
  187. package/test/session-rules.test.ts +304 -0
  188. package/test/session-start.test.ts +112 -0
  189. package/test/skill-prompt-sanitizer.test.ts +374 -0
  190. package/test/status.test.ts +10 -0
  191. package/test/subagent-context.test.ts +326 -0
  192. package/test/subagent-lifecycle-events.test.ts +132 -0
  193. package/test/subagent-registry.test.ts +145 -0
  194. package/test/synthesize.test.ts +300 -0
  195. package/test/system-prompt-sanitizer.test.ts +382 -0
  196. package/test/tool-access-extractor-registry.test.ts +77 -0
  197. package/test/tool-input-formatter-registry.test.ts +75 -0
  198. package/test/tool-input-preview.test.ts +129 -0
  199. package/test/tool-input-prompt-formatters.test.ts +115 -0
  200. package/test/tool-preview-formatter.test.ts +458 -0
  201. package/test/tool-registry.test.ts +197 -0
  202. package/test/wildcard-matcher.test.ts +424 -0
  203. package/test/yolo-mode.test.ts +188 -0
@@ -0,0 +1,447 @@
1
+ import { describe, expect, it, vi } from "vitest";
2
+
3
+ import type { DenialContext } from "#src/denial-messages";
4
+ import { EXTENSION_TAG } from "#src/denial-messages";
5
+ import type { GateBypass } from "#src/handlers/gates/descriptor";
6
+ import { SessionApproval } from "#src/session-approval";
7
+ import {
8
+ makeDenialDescriptor,
9
+ makeDescriptor,
10
+ makeGateRunner,
11
+ } from "#test/helpers/gate-fixtures";
12
+ import { makeCheckResult } from "#test/helpers/handler-fixtures";
13
+
14
+ // ── GateRunner — descriptor path ───────────────────────────────────────────
15
+
16
+ describe("GateRunner — descriptor path", () => {
17
+ it("returns allow and emits policy_allow when policy is allow", async () => {
18
+ const { runner, deps } = makeGateRunner();
19
+ const result = await runner.run(makeDescriptor(), null, "tc-1");
20
+ expect(result).toEqual({ action: "allow" });
21
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
22
+ expect.objectContaining({
23
+ surface: "read",
24
+ value: "read",
25
+ result: "allow",
26
+ resolution: "policy_allow",
27
+ }),
28
+ );
29
+ });
30
+
31
+ it("returns block and emits policy_deny when policy is deny", async () => {
32
+ const { runner, deps } = makeGateRunner({
33
+ resolveResult: makeCheckResult({ state: "deny", matchedPattern: "*" }),
34
+ });
35
+ const result = await runner.run(makeDescriptor(), null, "tc-1");
36
+ expect(result).toMatchObject({ action: "block" });
37
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
38
+ expect.objectContaining({
39
+ result: "deny",
40
+ resolution: "policy_deny",
41
+ }),
42
+ );
43
+ expect(deps.reporter.writeReviewLog).toHaveBeenCalledWith(
44
+ "permission_request.blocked",
45
+ expect.objectContaining({ resolution: "policy_denied" }),
46
+ );
47
+ });
48
+
49
+ it("returns allow and emits session_approved on session hit", async () => {
50
+ const { runner, deps } = makeGateRunner({
51
+ resolveResult: makeCheckResult({
52
+ source: "session",
53
+ matchedPattern: "git *",
54
+ }),
55
+ });
56
+ const result = await runner.run(
57
+ makeDescriptor({
58
+ surface: "bash",
59
+ input: { command: "git status" },
60
+ decision: { surface: "bash", value: "git status" },
61
+ }),
62
+ null,
63
+ "tc-1",
64
+ );
65
+ expect(result).toEqual({ action: "allow" });
66
+ expect(deps.reporter.writeReviewLog).toHaveBeenCalledWith(
67
+ "permission_request.session_approved",
68
+ expect.objectContaining({
69
+ resolution: "session_approved",
70
+ sessionApprovalPattern: "git *",
71
+ }),
72
+ );
73
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
74
+ expect.objectContaining({
75
+ resolution: "session_approved",
76
+ matchedPattern: "git *",
77
+ }),
78
+ );
79
+ });
80
+
81
+ it("returns allow and emits user_approved when ask + user approves", async () => {
82
+ const { runner, deps } = makeGateRunner({
83
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
84
+ prompt: vi.fn().mockResolvedValue({ approved: true, state: "approved" }),
85
+ });
86
+ const result = await runner.run(makeDescriptor(), null, "tc-1");
87
+ expect(result).toEqual({ action: "allow" });
88
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
89
+ expect.objectContaining({
90
+ result: "allow",
91
+ resolution: "user_approved",
92
+ }),
93
+ );
94
+ });
95
+
96
+ it("returns allow, emits user_approved_for_session, and records session rule on approved_for_session", async () => {
97
+ const { runner, deps } = makeGateRunner({
98
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
99
+ prompt: vi
100
+ .fn()
101
+ .mockResolvedValue({ approved: true, state: "approved_for_session" }),
102
+ });
103
+ const descriptor = makeDescriptor({
104
+ sessionApproval: SessionApproval.single("read", "*"),
105
+ });
106
+ const result = await runner.run(descriptor, null, "tc-1");
107
+ expect(result).toEqual({ action: "allow" });
108
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
109
+ expect.objectContaining({
110
+ resolution: "user_approved_for_session",
111
+ }),
112
+ );
113
+ expect(deps.recordSessionApproval).toHaveBeenCalledWith(
114
+ SessionApproval.single("read", "*"),
115
+ );
116
+ });
117
+
118
+ it("emits user_approved_for_project and records project rule on approved_for_project", async () => {
119
+ const { runner, deps } = makeGateRunner({
120
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
121
+ prompt: vi
122
+ .fn()
123
+ .mockResolvedValue({ approved: true, state: "approved_for_project" }),
124
+ });
125
+ const approval = SessionApproval.single("read", "*");
126
+ const descriptor = makeDescriptor({ sessionApproval: approval });
127
+ const result = await runner.run(descriptor, null, "tc-1");
128
+ expect(result).toEqual({ action: "allow" });
129
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
130
+ expect.objectContaining({
131
+ resolution: "user_approved_for_project",
132
+ }),
133
+ );
134
+ expect(deps.recordPersistentApproval).toHaveBeenCalledWith(
135
+ "project",
136
+ approval,
137
+ );
138
+ expect(deps.recordSessionApproval).not.toHaveBeenCalled();
139
+ });
140
+
141
+ it("emits user_approved_globally and records global rule on approved_globally", async () => {
142
+ const { runner, deps } = makeGateRunner({
143
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
144
+ prompt: vi
145
+ .fn()
146
+ .mockResolvedValue({ approved: true, state: "approved_globally" }),
147
+ });
148
+ const approval = SessionApproval.single("read", "*");
149
+ const descriptor = makeDescriptor({ sessionApproval: approval });
150
+ const result = await runner.run(descriptor, null, "tc-1");
151
+ expect(result).toEqual({ action: "allow" });
152
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
153
+ expect.objectContaining({
154
+ resolution: "user_approved_globally",
155
+ }),
156
+ );
157
+ expect(deps.recordPersistentApproval).toHaveBeenCalledWith("global", approval);
158
+ expect(deps.recordSessionApproval).not.toHaveBeenCalled();
159
+ });
160
+
161
+ it("calls recordSessionApproval once with the full SessionApproval when sessionApproval has multiple patterns", async () => {
162
+ const { runner, deps } = makeGateRunner({
163
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
164
+ prompt: vi
165
+ .fn()
166
+ .mockResolvedValue({ approved: true, state: "approved_for_session" }),
167
+ });
168
+ const approval = SessionApproval.multiple("external_directory", [
169
+ "/outside/a/*",
170
+ "/outside/b/*",
171
+ ]);
172
+ const descriptor = makeDescriptor({ sessionApproval: approval });
173
+ const result = await runner.run(descriptor, null, "tc-1");
174
+ expect(result).toEqual({ action: "allow" });
175
+ expect(deps.recordSessionApproval).toHaveBeenCalledTimes(1);
176
+ expect(deps.recordSessionApproval).toHaveBeenCalledWith(approval);
177
+ });
178
+
179
+ it("returns block and emits user_denied when ask + user denies", async () => {
180
+ const { runner, deps } = makeGateRunner({
181
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
182
+ prompt: vi.fn().mockResolvedValue({ approved: false, state: "denied" }),
183
+ });
184
+ const result = await runner.run(makeDescriptor(), null, "tc-1");
185
+ expect(result).toMatchObject({ action: "block" });
186
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
187
+ expect.objectContaining({
188
+ result: "deny",
189
+ resolution: "user_denied",
190
+ }),
191
+ );
192
+ });
193
+
194
+ it("returns block and emits confirmation_unavailable when ask + no UI", async () => {
195
+ const { runner, deps } = makeGateRunner({
196
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
197
+ canConfirm: vi.fn().mockReturnValue(false),
198
+ });
199
+ const result = await runner.run(makeDescriptor(), null, "tc-1");
200
+ expect(result).toMatchObject({ action: "block" });
201
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
202
+ expect.objectContaining({
203
+ result: "deny",
204
+ resolution: "confirmation_unavailable",
205
+ }),
206
+ );
207
+ });
208
+
209
+ it("emits auto_approved resolution when decision has autoApproved flag", async () => {
210
+ const { runner, deps } = makeGateRunner({
211
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
212
+ prompt: vi.fn().mockResolvedValue({
213
+ approved: true,
214
+ state: "approved",
215
+ autoApproved: true,
216
+ }),
217
+ });
218
+ const result = await runner.run(makeDescriptor(), null, "tc-1");
219
+ expect(result).toEqual({ action: "allow" });
220
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
221
+ expect.objectContaining({
222
+ resolution: "auto_approved",
223
+ }),
224
+ );
225
+ });
226
+
227
+ it("uses preResolved.state instead of calling resolve", async () => {
228
+ const { runner, deps } = makeGateRunner();
229
+ const descriptor = makeDescriptor({
230
+ preResolved: { state: "deny" },
231
+ });
232
+ const result = await runner.run(descriptor, null, "tc-1");
233
+ expect(result).toMatchObject({ action: "block" });
234
+ expect(deps.resolve).not.toHaveBeenCalled();
235
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
236
+ expect.objectContaining({
237
+ resolution: "policy_deny",
238
+ }),
239
+ );
240
+ });
241
+
242
+ it("uses preResolved.state allow without calling resolve", async () => {
243
+ const { runner, deps } = makeGateRunner();
244
+ const descriptor = makeDescriptor({
245
+ preResolved: { state: "allow" },
246
+ });
247
+ const result = await runner.run(descriptor, null, "tc-1");
248
+ expect(result).toEqual({ action: "allow" });
249
+ expect(deps.resolve).not.toHaveBeenCalled();
250
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
251
+ expect.objectContaining({
252
+ resolution: "policy_allow",
253
+ }),
254
+ );
255
+ });
256
+
257
+ it("passes agentName to resolve and decision event", async () => {
258
+ const { runner, deps } = makeGateRunner();
259
+ const result = await runner.run(makeDescriptor(), "test-agent", "tc-1");
260
+ expect(result).toEqual({ action: "allow" });
261
+ expect(deps.resolve).toHaveBeenCalledWith("read", {}, "test-agent");
262
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
263
+ expect.objectContaining({
264
+ agentName: "test-agent",
265
+ }),
266
+ );
267
+ });
268
+
269
+ it("passes requestId from toolCallId to prompt", async () => {
270
+ const { runner, deps } = makeGateRunner({
271
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
272
+ });
273
+ await runner.run(makeDescriptor(), null, "tc-42");
274
+ expect(deps.prompt).toHaveBeenCalledWith(
275
+ expect.objectContaining({ requestId: "tc-42" }),
276
+ );
277
+ });
278
+
279
+ it("does not call recordSessionApproval or recordPersistentApproval when user approves once (no sessionApproval)", async () => {
280
+ const { runner, deps } = makeGateRunner({
281
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
282
+ prompt: vi.fn().mockResolvedValue({ approved: true, state: "approved" }),
283
+ });
284
+ await runner.run(makeDescriptor(), null, "tc-1");
285
+ expect(deps.recordSessionApproval).not.toHaveBeenCalled();
286
+ expect(deps.recordPersistentApproval).not.toHaveBeenCalled();
287
+ });
288
+
289
+ it("uses preCheck result directly instead of calling resolve", async () => {
290
+ const { runner, deps } = makeGateRunner();
291
+ const descriptor = makeDescriptor({
292
+ preCheck: makeCheckResult({
293
+ state: "deny",
294
+ origin: "global",
295
+ matchedPattern: "rm *",
296
+ }),
297
+ });
298
+ const result = await runner.run(descriptor, null, "tc-1");
299
+ expect(result).toMatchObject({ action: "block" });
300
+ expect(deps.resolve).not.toHaveBeenCalled();
301
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(
302
+ expect.objectContaining({
303
+ resolution: "policy_deny",
304
+ origin: "global",
305
+ matchedPattern: "rm *",
306
+ }),
307
+ );
308
+ });
309
+
310
+ it("does not call recordSessionApproval when user approves for session but no sessionApproval on descriptor", async () => {
311
+ const { runner, deps } = makeGateRunner({
312
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
313
+ prompt: vi
314
+ .fn()
315
+ .mockResolvedValue({ approved: true, state: "approved_for_session" }),
316
+ });
317
+ // No sessionApproval on descriptor
318
+ await runner.run(makeDescriptor(), null, "tc-1");
319
+ expect(deps.recordSessionApproval).not.toHaveBeenCalled();
320
+ });
321
+
322
+ describe("denialContext formatting", () => {
323
+ it("uses denialContext to format denyReason with extension tag", async () => {
324
+ const { runner } = makeGateRunner({
325
+ resolveResult: makeCheckResult({ state: "deny", matchedPattern: "*" }),
326
+ });
327
+ const ctx: DenialContext = {
328
+ kind: "tool",
329
+ check: makeCheckResult({ state: "deny", matchedPattern: "*" }),
330
+ agentName: "test-agent",
331
+ };
332
+ const result = await runner.run(
333
+ makeDenialDescriptor(ctx),
334
+ "test-agent",
335
+ "tc-1",
336
+ );
337
+ expect(result.action).toBe("block");
338
+ if (result.action === "block") {
339
+ expect(result.reason).toContain(EXTENSION_TAG);
340
+ expect(result.reason).not.toContain("Hard stop");
341
+ }
342
+ });
343
+
344
+ it("uses denialContext to format unavailableReason with extension tag", async () => {
345
+ const { runner } = makeGateRunner({
346
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
347
+ canConfirm: vi.fn().mockReturnValue(false),
348
+ });
349
+ const ctx: DenialContext = {
350
+ kind: "tool",
351
+ check: makeCheckResult({ state: "ask", matchedPattern: "*" }),
352
+ };
353
+ const result = await runner.run(makeDenialDescriptor(ctx), null, "tc-1");
354
+ expect(result.action).toBe("block");
355
+ if (result.action === "block") {
356
+ expect(result.reason).toContain(EXTENSION_TAG);
357
+ expect(result.reason).toContain("no interactive UI");
358
+ }
359
+ });
360
+
361
+ it("uses denialContext to format userDeniedReason with extension tag", async () => {
362
+ const { runner } = makeGateRunner({
363
+ resolveResult: makeCheckResult({ state: "ask", matchedPattern: "*" }),
364
+ prompt: vi.fn().mockResolvedValue({
365
+ approved: false,
366
+ state: "denied",
367
+ denialReason: "too risky",
368
+ }),
369
+ });
370
+ const ctx: DenialContext = {
371
+ kind: "tool",
372
+ check: makeCheckResult({ state: "ask", matchedPattern: "*" }),
373
+ };
374
+ const result = await runner.run(makeDenialDescriptor(ctx), null, "tc-1");
375
+ expect(result.action).toBe("block");
376
+ if (result.action === "block") {
377
+ expect(result.reason).toContain(EXTENSION_TAG);
378
+ expect(result.reason).toContain("too risky");
379
+ }
380
+ });
381
+ });
382
+ });
383
+
384
+ // ── GateRunner.run — null and bypass dispatch ──────────────────────────────
385
+
386
+ describe("GateRunner.run — null and bypass dispatch", () => {
387
+ it("returns allow for a null gate", async () => {
388
+ const { runner, deps } = makeGateRunner();
389
+ const result = await runner.run(null, null, "tc-1");
390
+ expect(result).toEqual({ action: "allow" });
391
+ expect(deps.reporter.writeReviewLog).not.toHaveBeenCalled();
392
+ expect(deps.reporter.emitDecision).not.toHaveBeenCalled();
393
+ });
394
+
395
+ it("returns allow for a bypass with no log or decision", async () => {
396
+ const { runner, deps } = makeGateRunner();
397
+ const bypass: GateBypass = { action: "allow" };
398
+ const result = await runner.run(bypass, null, "tc-1");
399
+ expect(result).toEqual({ action: "allow" });
400
+ expect(deps.reporter.writeReviewLog).not.toHaveBeenCalled();
401
+ expect(deps.reporter.emitDecision).not.toHaveBeenCalled();
402
+ });
403
+
404
+ it("fires writeReviewLog for a bypass with a log entry", async () => {
405
+ const { runner, deps } = makeGateRunner();
406
+ const bypass: GateBypass = {
407
+ action: "allow",
408
+ log: { event: "infra.bypass", details: { path: "/x" } },
409
+ };
410
+ await runner.run(bypass, null, "tc-1");
411
+ expect(deps.reporter.writeReviewLog).toHaveBeenCalledWith("infra.bypass", {
412
+ path: "/x",
413
+ });
414
+ expect(deps.reporter.emitDecision).not.toHaveBeenCalled();
415
+ });
416
+
417
+ it("fires emitDecision for a bypass with a decision", async () => {
418
+ const { runner, deps } = makeGateRunner();
419
+ const decision = {
420
+ surface: "path",
421
+ value: "/x",
422
+ result: "allow" as const,
423
+ resolution: "policy_allow" as const,
424
+ origin: null,
425
+ agentName: null,
426
+ matchedPattern: null,
427
+ };
428
+ const bypass: GateBypass = { action: "allow", decision };
429
+ await runner.run(bypass, null, "tc-1");
430
+ expect(deps.reporter.emitDecision).toHaveBeenCalledWith(decision);
431
+ expect(deps.reporter.writeReviewLog).not.toHaveBeenCalled();
432
+ });
433
+
434
+ it("routes a descriptor to the gate check logic and returns allow", async () => {
435
+ const { runner } = makeGateRunner();
436
+ const result = await runner.run(makeDescriptor(), null, "tc-1");
437
+ expect(result).toEqual({ action: "allow" });
438
+ });
439
+
440
+ it("routes a descriptor to the gate check logic and returns block", async () => {
441
+ const { runner } = makeGateRunner({
442
+ resolveResult: makeCheckResult({ state: "deny", matchedPattern: "*" }),
443
+ });
444
+ const result = await runner.run(makeDescriptor(), null, "tc-1");
445
+ expect(result).toMatchObject({ action: "block" });
446
+ });
447
+ });
@@ -0,0 +1,176 @@
1
+ import { describe, expect, it } from "vitest";
2
+
3
+ import {
4
+ createSkillInputRequestId,
5
+ formatSkillDenyNotice,
6
+ SkillInputGatePipeline,
7
+ } from "#src/handlers/gates/skill-input-gate-pipeline";
8
+
9
+ import {
10
+ makeGateRunner,
11
+ makeNotifier,
12
+ makeSkillInputInputs,
13
+ } from "#test/helpers/gate-fixtures";
14
+ import { makeCheckResult } from "#test/helpers/handler-fixtures";
15
+
16
+ // ── createSkillInputRequestId ─────────────────────────────────────────────
17
+
18
+ describe("createSkillInputRequestId", () => {
19
+ it("starts with 'skill-input-'", () => {
20
+ expect(createSkillInputRequestId().startsWith("skill-input-")).toBe(true);
21
+ });
22
+
23
+ it("returns a unique id on each call", () => {
24
+ const id1 = createSkillInputRequestId();
25
+ const id2 = createSkillInputRequestId();
26
+ expect(id1).not.toBe(id2);
27
+ });
28
+ });
29
+
30
+ // ── formatSkillDenyNotice ─────────────────────────────────────────────────
31
+
32
+ describe("formatSkillDenyNotice", () => {
33
+ it("includes the skill name in the message (no agent)", () => {
34
+ const msg = formatSkillDenyNotice("librarian", null);
35
+ expect(msg).toContain("librarian");
36
+ });
37
+
38
+ it("includes the skill name and agent name when agent is present", () => {
39
+ const msg = formatSkillDenyNotice("librarian", "code-agent");
40
+ expect(msg).toContain("librarian");
41
+ expect(msg).toContain("code-agent");
42
+ });
43
+ });
44
+
45
+ // ── SkillInputGatePipeline.evaluate ───────────────────────────────────────
46
+
47
+ describe("SkillInputGatePipeline.evaluate", () => {
48
+ // ── notifier behaviour ──────────────────────────────────────────────────
49
+
50
+ it("calls notifier.warn when the skill is denied", async () => {
51
+ const inputs = makeSkillInputInputs({
52
+ checkPermission: () => makeCheckResult({ state: "deny" }),
53
+ });
54
+ const notifier = makeNotifier();
55
+ const { runner } = makeGateRunner();
56
+ const pipeline = new SkillInputGatePipeline(inputs);
57
+
58
+ await pipeline.evaluate("librarian", null, notifier, runner);
59
+
60
+ expect(notifier.warn).toHaveBeenCalledOnce();
61
+ expect(notifier.warn).toHaveBeenCalledWith(
62
+ expect.stringContaining("librarian"),
63
+ );
64
+ });
65
+
66
+ it("does not call notifier.warn when the skill is allowed", async () => {
67
+ const inputs = makeSkillInputInputs({
68
+ checkPermission: () => makeCheckResult({ state: "allow" }),
69
+ });
70
+ const notifier = makeNotifier();
71
+ const { runner } = makeGateRunner();
72
+ const pipeline = new SkillInputGatePipeline(inputs);
73
+
74
+ await pipeline.evaluate("librarian", null, notifier, runner);
75
+
76
+ expect(notifier.warn).not.toHaveBeenCalled();
77
+ });
78
+
79
+ it("does not call notifier.warn when the skill requires approval (ask)", async () => {
80
+ const inputs = makeSkillInputInputs({
81
+ checkPermission: () => makeCheckResult({ state: "ask" }),
82
+ });
83
+ const notifier = makeNotifier();
84
+ const { runner } = makeGateRunner();
85
+ const pipeline = new SkillInputGatePipeline(inputs);
86
+
87
+ await pipeline.evaluate("librarian", null, notifier, runner);
88
+
89
+ expect(notifier.warn).not.toHaveBeenCalled();
90
+ });
91
+
92
+ it("includes agent name in the deny notice when agent is present", async () => {
93
+ const inputs = makeSkillInputInputs({
94
+ checkPermission: () => makeCheckResult({ state: "deny" }),
95
+ });
96
+ const notifier = makeNotifier();
97
+ const { runner } = makeGateRunner();
98
+ const pipeline = new SkillInputGatePipeline(inputs);
99
+
100
+ await pipeline.evaluate("librarian", "code-agent", notifier, runner);
101
+
102
+ expect(notifier.warn).toHaveBeenCalledWith(
103
+ expect.stringContaining("code-agent"),
104
+ );
105
+ });
106
+
107
+ // ── outcome mapping ─────────────────────────────────────────────────────
108
+
109
+ it("returns allow when the gate passes", async () => {
110
+ const inputs = makeSkillInputInputs({
111
+ checkPermission: () => makeCheckResult({ state: "allow" }),
112
+ });
113
+ const { runner } = makeGateRunner();
114
+ const pipeline = new SkillInputGatePipeline(inputs);
115
+
116
+ const result = await pipeline.evaluate(
117
+ "librarian",
118
+ null,
119
+ makeNotifier(),
120
+ runner,
121
+ );
122
+
123
+ expect(result).toEqual({ action: "allow" });
124
+ });
125
+
126
+ it("returns block when the gate denies", async () => {
127
+ const inputs = makeSkillInputInputs({
128
+ checkPermission: () =>
129
+ makeCheckResult({ state: "deny", matchedPattern: "*" }),
130
+ });
131
+ const { runner } = makeGateRunner();
132
+ const pipeline = new SkillInputGatePipeline(inputs);
133
+
134
+ const result = await pipeline.evaluate(
135
+ "librarian",
136
+ null,
137
+ makeNotifier(),
138
+ runner,
139
+ );
140
+
141
+ expect(result).toEqual({
142
+ action: "block",
143
+ reason: expect.stringContaining("librarian"),
144
+ });
145
+ });
146
+
147
+ // ── checkPermission call ────────────────────────────────────────────────
148
+
149
+ it("calls checkPermission with the skill surface, skill name, and agent name", async () => {
150
+ const inputs = makeSkillInputInputs();
151
+ const { runner } = makeGateRunner();
152
+ const pipeline = new SkillInputGatePipeline(inputs);
153
+
154
+ await pipeline.evaluate("explorer", "code-agent", makeNotifier(), runner);
155
+
156
+ expect(inputs.checkPermission).toHaveBeenCalledWith(
157
+ "skill",
158
+ { name: "explorer" },
159
+ "code-agent",
160
+ );
161
+ });
162
+
163
+ it("calls checkPermission with undefined agentName when agentName is null", async () => {
164
+ const inputs = makeSkillInputInputs();
165
+ const { runner } = makeGateRunner();
166
+ const pipeline = new SkillInputGatePipeline(inputs);
167
+
168
+ await pipeline.evaluate("explorer", null, makeNotifier(), runner);
169
+
170
+ expect(inputs.checkPermission).toHaveBeenCalledWith(
171
+ "skill",
172
+ { name: "explorer" },
173
+ undefined,
174
+ );
175
+ });
176
+ });