@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,604 @@
1
+ import { describe, expect, test } from "vitest";
2
+ import type { Rule, RuleOrigin, Ruleset } from "#src/rule";
3
+ import {
4
+ evaluate,
5
+ evaluateAnyValue,
6
+ evaluateFirst,
7
+ evaluateMostRestrictive,
8
+ } from "#src/rule";
9
+
10
+ describe("evaluate", () => {
11
+ const allowBashGit: Rule = {
12
+ surface: "bash",
13
+ pattern: "git *",
14
+ action: "allow",
15
+ origin: "global",
16
+ };
17
+ const denyBashGitPush: Rule = {
18
+ surface: "bash",
19
+ pattern: "git push *",
20
+ action: "deny",
21
+ origin: "global",
22
+ };
23
+ const allowRead: Rule = {
24
+ surface: "read",
25
+ pattern: "*",
26
+ action: "allow",
27
+ origin: "global",
28
+ };
29
+ const askMcp: Rule = {
30
+ surface: "mcp",
31
+ pattern: "*",
32
+ action: "ask",
33
+ origin: "global",
34
+ };
35
+ const allowSkillLibrarian: Rule = {
36
+ surface: "skill",
37
+ pattern: "librarian",
38
+ action: "allow",
39
+ origin: "global",
40
+ };
41
+ const askSpecialExtDir: Rule = {
42
+ surface: "special",
43
+ pattern: "external_directory",
44
+ action: "ask",
45
+ origin: "global",
46
+ };
47
+
48
+ test("returns matching rule when a rule matches", () => {
49
+ const ruleset: Ruleset = [allowBashGit];
50
+ const result = evaluate("bash", "git status", ruleset);
51
+ expect(result).toEqual(allowBashGit);
52
+ });
53
+
54
+ test("returns synthetic rule with 'ask' when no rules match and no defaultAction", () => {
55
+ const result = evaluate("bash", "npm install", [allowBashGit]);
56
+ expect(result.surface).toBe("bash");
57
+ expect(result.pattern).toBe("npm install");
58
+ expect(result.action).toBe("ask");
59
+ });
60
+
61
+ test("returns synthetic rule with custom defaultAction when no rules match", () => {
62
+ const result = evaluate("bash", "npm install", [allowBashGit], "deny");
63
+ expect(result.surface).toBe("bash");
64
+ expect(result.pattern).toBe("npm install");
65
+ expect(result.action).toBe("deny");
66
+ });
67
+
68
+ test("defaultAction does not affect matched rules", () => {
69
+ const result = evaluate("bash", "git status", [allowBashGit], "deny");
70
+ expect(result).toEqual(allowBashGit);
71
+ });
72
+
73
+ test("returns synthetic rule for empty ruleset", () => {
74
+ const result = evaluate("mcp", "exa_search", []);
75
+ expect(result.surface).toBe("mcp");
76
+ expect(result.pattern).toBe("exa_search");
77
+ expect(result.action).toBe("ask");
78
+ });
79
+
80
+ test("matches rules for all permission surfaces", () => {
81
+ expect(evaluate("read", "src/foo.ts", [allowRead]).action).toBe("allow");
82
+ expect(evaluate("mcp", "exa_search", [askMcp]).action).toBe("ask");
83
+ expect(evaluate("skill", "librarian", [allowSkillLibrarian]).action).toBe(
84
+ "allow",
85
+ );
86
+ expect(
87
+ evaluate("special", "external_directory", [askSpecialExtDir]).action,
88
+ ).toBe("ask");
89
+ });
90
+
91
+ test("last-match-wins: later conflicting rule overrides earlier", () => {
92
+ const ruleset: Ruleset = [allowBashGit, denyBashGitPush];
93
+ const result = evaluate("bash", "git push origin main", ruleset);
94
+ expect(result).toEqual(denyBashGitPush);
95
+ });
96
+
97
+ test("last-match-wins: broad deny followed by specific allow", () => {
98
+ const denyAll: Rule = {
99
+ surface: "bash",
100
+ pattern: "*",
101
+ action: "deny",
102
+ origin: "global",
103
+ };
104
+ const allowStatus: Rule = {
105
+ surface: "bash",
106
+ pattern: "git status",
107
+ action: "allow",
108
+ origin: "global",
109
+ };
110
+ const result = evaluate("bash", "git status", [denyAll, allowStatus]);
111
+ expect(result).toEqual(allowStatus);
112
+ });
113
+
114
+ test("wildcard surface in rule matches any surface value", () => {
115
+ const universalAllow: Rule = {
116
+ surface: "*",
117
+ pattern: "*",
118
+ action: "allow",
119
+ origin: "global",
120
+ };
121
+ expect(evaluate("bash", "anything", [universalAllow]).action).toBe("allow");
122
+ expect(evaluate("mcp", "something", [universalAllow]).action).toBe("allow");
123
+ expect(evaluate("skill", "librarian", [universalAllow]).action).toBe(
124
+ "allow",
125
+ );
126
+ });
127
+
128
+ test("specific surface rule does not match a different surface", () => {
129
+ const ruleset: Ruleset = [allowBashGit];
130
+ // bash rule should not match mcp surface
131
+ const result = evaluate("mcp", "git status", ruleset);
132
+ expect(result.action).toBe("ask"); // falls back to default
133
+ });
134
+
135
+ test("merged rulesets: rules from later scope take priority", () => {
136
+ const globalRules: Ruleset = [
137
+ { surface: "bash", pattern: "git *", action: "ask", origin: "global" },
138
+ ];
139
+ const agentRules: Ruleset = [
140
+ { surface: "bash", pattern: "git *", action: "allow", origin: "agent" },
141
+ ];
142
+ const merged = [...globalRules, ...agentRules];
143
+ const result = evaluate("bash", "git status", merged);
144
+ expect(result.action).toBe("allow"); // agent rule wins
145
+ });
146
+
147
+ test("merged rulesets: earlier scope used when later scope has no match", () => {
148
+ const globalRules: Ruleset = [
149
+ { surface: "bash", pattern: "git *", action: "allow", origin: "global" },
150
+ ];
151
+ const agentRules: Ruleset = [
152
+ { surface: "bash", pattern: "npm *", action: "deny", origin: "agent" },
153
+ ];
154
+ // git status matches global but not agent rule
155
+ const merged = [...globalRules, ...agentRules];
156
+ const result = evaluate("bash", "git status", merged);
157
+ expect(result.action).toBe("allow"); // global rule is the last match for this pattern
158
+ });
159
+
160
+ test("empty ruleset returns synthetic default", () => {
161
+ const result = evaluate("bash", "git status", []);
162
+ expect(result.surface).toBe("bash");
163
+ expect(result.pattern).toBe("git status");
164
+ expect(result.action).toBe("ask");
165
+ });
166
+
167
+ test("rule.layer is ignored by evaluate() — matching is identical with or without it", () => {
168
+ const withLayer: Rule = {
169
+ surface: "bash",
170
+ pattern: "git *",
171
+ action: "allow",
172
+ layer: "config",
173
+ origin: "global",
174
+ };
175
+ const withoutLayer: Rule = {
176
+ surface: "bash",
177
+ pattern: "git *",
178
+ action: "allow",
179
+ origin: "global",
180
+ };
181
+ const withDefault: Rule = {
182
+ surface: "bash",
183
+ pattern: "*",
184
+ action: "ask",
185
+ layer: "default",
186
+ origin: "builtin",
187
+ };
188
+ // Both rules with and without layer field produce the same match.
189
+ expect(evaluate("bash", "git status", [withLayer]).action).toBe("allow");
190
+ expect(evaluate("bash", "git status", [withoutLayer]).action).toBe("allow");
191
+ // Layer metadata does not affect last-match-wins ordering.
192
+ const ruleset: Rule[] = [withDefault, withLayer];
193
+ expect(evaluate("bash", "git status", ruleset)).toEqual(withLayer);
194
+ // A rule with layer: "default" still wins if it is last in the array.
195
+ const reversedRuleset: Rule[] = [withLayer, withDefault];
196
+ expect(evaluate("bash", "git status", reversedRuleset)).toEqual(
197
+ withDefault,
198
+ );
199
+ });
200
+
201
+ test("evaluate() preserves origin on a matched rule", () => {
202
+ const origin: RuleOrigin = "project";
203
+ const rule: Rule = {
204
+ surface: "bash",
205
+ pattern: "git *",
206
+ action: "allow",
207
+ layer: "config",
208
+ origin,
209
+ };
210
+ const result = evaluate("bash", "git status", [rule]);
211
+ expect(result.origin).toBe("project");
212
+ });
213
+
214
+ test("evaluate() synthetic fallback rule has origin 'builtin'", () => {
215
+ const result = evaluate("bash", "npm install", []);
216
+ expect(result.origin).toBe("builtin");
217
+ });
218
+
219
+ test("evaluate() propagates reason from the matched deny rule", () => {
220
+ const rule: Rule = {
221
+ surface: "bash",
222
+ pattern: "npm *",
223
+ action: "deny",
224
+ reason: "Use pnpm instead",
225
+ layer: "config",
226
+ origin: "global",
227
+ };
228
+ const result = evaluate("bash", "npm install", [rule]);
229
+ expect(result.action).toBe("deny");
230
+ expect(result.reason).toBe("Use pnpm instead");
231
+ });
232
+
233
+ test("evaluate() carries reason through last-match-wins when deny wins", () => {
234
+ const allowAll: Rule = {
235
+ surface: "bash",
236
+ pattern: "*",
237
+ action: "allow",
238
+ layer: "config",
239
+ origin: "global",
240
+ };
241
+ const denyNpm: Rule = {
242
+ surface: "bash",
243
+ pattern: "npm *",
244
+ action: "deny",
245
+ reason: "Use pnpm",
246
+ layer: "config",
247
+ origin: "global",
248
+ };
249
+ const result = evaluate("bash", "npm install", [allowAll, denyNpm]);
250
+ expect(result.action).toBe("deny");
251
+ expect(result.reason).toBe("Use pnpm");
252
+ });
253
+
254
+ test("evaluate() drops reason when a later allow overrides the deny", () => {
255
+ const denyNpm: Rule = {
256
+ surface: "bash",
257
+ pattern: "npm *",
258
+ action: "deny",
259
+ reason: "Use pnpm",
260
+ layer: "config",
261
+ origin: "global",
262
+ };
263
+ const allowInstall: Rule = {
264
+ surface: "bash",
265
+ pattern: "npm install",
266
+ action: "allow",
267
+ layer: "config",
268
+ origin: "global",
269
+ };
270
+ const result = evaluate("bash", "npm install", [denyNpm, allowInstall]);
271
+ expect(result.action).toBe("allow");
272
+ expect(result.reason).toBeUndefined();
273
+ });
274
+
275
+ test("evaluate() synthetic fallback rule has no reason", () => {
276
+ const result = evaluate("bash", "npm install", []);
277
+ expect(result.reason).toBeUndefined();
278
+ });
279
+
280
+ test("RuleOrigin covers all seven provenance values", () => {
281
+ const origins: RuleOrigin[] = [
282
+ "global",
283
+ "project",
284
+ "agent",
285
+ "project-agent",
286
+ "builtin",
287
+ "baseline",
288
+ "session",
289
+ ];
290
+ for (const origin of origins) {
291
+ const rule: Rule = {
292
+ surface: "read",
293
+ pattern: "*",
294
+ action: "allow",
295
+ layer: "config",
296
+ origin,
297
+ };
298
+ expect(evaluate("read", "*", [rule]).origin).toBe(origin);
299
+ }
300
+ });
301
+
302
+ // ── Windows: path-surface patterns fold case (last-match-wins) ──────────
303
+
304
+ const denyExternalAll: Rule = {
305
+ surface: "external_directory",
306
+ pattern: "*",
307
+ action: "deny",
308
+ layer: "config",
309
+ origin: "global",
310
+ };
311
+ const allowExternalPi: Rule = {
312
+ surface: "external_directory",
313
+ pattern: "C:\\Users\\Foo\\pi\\*",
314
+ action: "allow",
315
+ layer: "config",
316
+ origin: "global",
317
+ };
318
+
319
+ test("win32: external_directory allow override matches a lowercased path over a preceding deny", () => {
320
+ const result = evaluate(
321
+ "external_directory",
322
+ "c:\\users\\foo\\pi\\docs\\readme.md",
323
+ [denyExternalAll, allowExternalPi],
324
+ undefined,
325
+ "win32",
326
+ );
327
+ expect(result.action).toBe("allow");
328
+ });
329
+
330
+ test("posix: the same mixed-case override stays case-sensitive (falls through to deny)", () => {
331
+ const result = evaluate(
332
+ "external_directory",
333
+ "c:\\users\\foo\\pi\\docs\\readme.md",
334
+ [denyExternalAll, allowExternalPi],
335
+ undefined,
336
+ "linux",
337
+ );
338
+ expect(result.action).toBe("deny");
339
+ });
340
+
341
+ test("win32: a forward-slash external_directory pattern matches a backslash value", () => {
342
+ const allowForwardSlash: Rule = {
343
+ surface: "external_directory",
344
+ pattern: "C:/Users/Foo/pi/*",
345
+ action: "allow",
346
+ layer: "config",
347
+ origin: "global",
348
+ };
349
+ const result = evaluate(
350
+ "external_directory",
351
+ "c:\\users\\foo\\pi\\docs\\readme.md",
352
+ [denyExternalAll, allowForwardSlash],
353
+ undefined,
354
+ "win32",
355
+ );
356
+ expect(result.action).toBe("allow");
357
+ });
358
+
359
+ test("win32: bash surface stays case-sensitive (not a path surface)", () => {
360
+ const result = evaluate(
361
+ "bash",
362
+ "GIT push",
363
+ [
364
+ {
365
+ surface: "bash",
366
+ pattern: "git *",
367
+ action: "allow",
368
+ origin: "global",
369
+ },
370
+ ],
371
+ undefined,
372
+ "win32",
373
+ );
374
+ expect(result.action).toBe("ask");
375
+ });
376
+ });
377
+
378
+ describe("evaluateFirst", () => {
379
+ const defaultRule: Rule = {
380
+ surface: "*",
381
+ pattern: "*",
382
+ action: "ask",
383
+ layer: "default",
384
+ origin: "builtin",
385
+ };
386
+ const allowBash: Rule = {
387
+ surface: "bash",
388
+ pattern: "git *",
389
+ action: "allow",
390
+ layer: "config",
391
+ origin: "global",
392
+ };
393
+ const denyMcp: Rule = {
394
+ surface: "mcp",
395
+ pattern: "exa_search",
396
+ action: "deny",
397
+ layer: "config",
398
+ origin: "global",
399
+ };
400
+
401
+ test("returns the first candidate that matches a non-default rule", () => {
402
+ const rules: Ruleset = [defaultRule, allowBash];
403
+ const result = evaluateFirst("bash", ["git status", "*"], rules);
404
+ expect(result.rule).toEqual(allowBash);
405
+ expect(result.value).toBe("git status");
406
+ });
407
+
408
+ test("skips candidates that only match the default rule", () => {
409
+ // "npm install" matches only the default; "*" also matches only the
410
+ // default — falls back to first candidate.
411
+ const rules: Ruleset = [defaultRule];
412
+ const result = evaluateFirst("bash", ["npm install", "*"], rules);
413
+ expect(result.rule.layer).toBe("default");
414
+ expect(result.value).toBe("npm install");
415
+ });
416
+
417
+ test("falls back to first candidate when all candidates match only the default", () => {
418
+ const rules: Ruleset = [defaultRule];
419
+ const result = evaluateFirst("bash", ["a", "b", "c"], rules);
420
+ expect(result.value).toBe("a");
421
+ });
422
+
423
+ test("stops at first non-default match, does not continue to remaining candidates", () => {
424
+ // "exa_search" matches denyMcp (non-default). The loop stops there;
425
+ // "mcp" is never evaluated even though it would match a different rule.
426
+ const allowMcpCatchAll: Rule = {
427
+ surface: "mcp",
428
+ pattern: "mcp",
429
+ action: "allow",
430
+ layer: "config",
431
+ origin: "global",
432
+ };
433
+ const rules: Ruleset = [defaultRule, denyMcp, allowMcpCatchAll];
434
+ const result = evaluateFirst("mcp", ["exa_search", "mcp"], rules);
435
+ expect(result.rule).toEqual(denyMcp);
436
+ expect(result.value).toBe("exa_search");
437
+ });
438
+
439
+ test("skips candidates that match only the default and continues to next", () => {
440
+ // "unknown_tool" matches only the universal default;
441
+ // "exa_search" matches denyMcp (non-default) — that is the result.
442
+ const rules: Ruleset = [defaultRule, denyMcp];
443
+ const result = evaluateFirst("mcp", ["unknown_tool", "exa_search"], rules);
444
+ expect(result.rule).toEqual(denyMcp);
445
+ expect(result.value).toBe("exa_search");
446
+ });
447
+
448
+ test("single-candidate array behaves like evaluate()", () => {
449
+ const rules: Ruleset = [defaultRule, allowBash];
450
+ const result = evaluateFirst("bash", ["git status"], rules);
451
+ expect(result.rule).toEqual(allowBash);
452
+ expect(result.value).toBe("git status");
453
+ });
454
+
455
+ test("uses '*' as fallback value when values array is empty", () => {
456
+ const rules: Ruleset = [defaultRule];
457
+ const result = evaluateFirst("bash", [], rules);
458
+ expect(result.value).toBe("*");
459
+ });
460
+ });
461
+
462
+ describe("evaluateAnyValue", () => {
463
+ const catchAllAllow: Rule = {
464
+ surface: "path",
465
+ pattern: "*",
466
+ action: "allow",
467
+ layer: "config",
468
+ origin: "global",
469
+ };
470
+ const catchAllAsk: Rule = {
471
+ surface: "path",
472
+ pattern: "*",
473
+ action: "ask",
474
+ layer: "config",
475
+ origin: "global",
476
+ };
477
+ const relativeDeny: Rule = {
478
+ surface: "path",
479
+ pattern: "src/*",
480
+ action: "deny",
481
+ layer: "config",
482
+ origin: "global",
483
+ };
484
+ const absoluteAllow: Rule = {
485
+ surface: "path",
486
+ pattern: "/proj/*",
487
+ action: "allow",
488
+ layer: "config",
489
+ origin: "global",
490
+ };
491
+
492
+ test("a later relative rule wins over a catch-all matched by another alias", () => {
493
+ const rules: Ruleset = [catchAllAllow, relativeDeny];
494
+ const result = evaluateAnyValue(
495
+ "path",
496
+ ["/proj/src/foo.ts", "src/foo.ts"],
497
+ rules,
498
+ );
499
+ expect(result.rule).toEqual(relativeDeny);
500
+ expect(result.value).toBe("src/foo.ts");
501
+ });
502
+
503
+ test("uses an absolute alias when no later relative rule matches", () => {
504
+ const rules: Ruleset = [catchAllAsk, absoluteAllow];
505
+ const result = evaluateAnyValue(
506
+ "path",
507
+ ["/proj/src/foo.ts", "src/foo.ts"],
508
+ rules,
509
+ );
510
+ expect(result.rule).toEqual(absoluteAllow);
511
+ expect(result.value).toBe("/proj/src/foo.ts");
512
+ });
513
+
514
+ test("falls back to the first value's default when no rule matches", () => {
515
+ const result = evaluateAnyValue(
516
+ "path",
517
+ ["/proj/src/foo.ts", "src/foo.ts"],
518
+ [],
519
+ );
520
+ expect(result.rule.action).toBe("ask");
521
+ expect(result.value).toBe("/proj/src/foo.ts");
522
+ });
523
+
524
+ test("uses '*' as fallback value when values array is empty", () => {
525
+ const result = evaluateAnyValue("path", [], []);
526
+ expect(result.value).toBe("*");
527
+ });
528
+ });
529
+
530
+ describe("evaluateMostRestrictive", () => {
531
+ const denyEnv: Rule = {
532
+ surface: "path",
533
+ pattern: "*.env",
534
+ action: "deny",
535
+ layer: "config",
536
+ origin: "global",
537
+ };
538
+ const askSsh: Rule = {
539
+ surface: "path",
540
+ pattern: "/home/user/.ssh/*",
541
+ action: "ask",
542
+ layer: "config",
543
+ origin: "global",
544
+ };
545
+ const allowAll: Rule = {
546
+ surface: "path",
547
+ pattern: "*",
548
+ action: "allow",
549
+ layer: "config",
550
+ origin: "global",
551
+ };
552
+
553
+ test("deny short-circuits: returns immediately without evaluating remaining values", () => {
554
+ const rules: Ruleset = [allowAll, denyEnv];
555
+ const result = evaluateMostRestrictive(
556
+ "path",
557
+ [".env", "README.md"],
558
+ rules,
559
+ );
560
+ expect(result).not.toBeNull();
561
+ expect(result!.rule.action).toBe("deny");
562
+ expect(result!.value).toBe(".env");
563
+ });
564
+
565
+ test("ask accumulates: returns first ask when no deny found", () => {
566
+ const rules: Ruleset = [allowAll, askSsh];
567
+ const result = evaluateMostRestrictive(
568
+ "path",
569
+ ["/home/user/.ssh/id_rsa", "README.md"],
570
+ rules,
571
+ );
572
+ expect(result).not.toBeNull();
573
+ expect(result!.rule.action).toBe("ask");
574
+ expect(result!.value).toBe("/home/user/.ssh/id_rsa");
575
+ });
576
+
577
+ test("all allow: returns null", () => {
578
+ const rules: Ruleset = [allowAll];
579
+ const result = evaluateMostRestrictive(
580
+ "path",
581
+ ["README.md", "src/index.ts"],
582
+ rules,
583
+ );
584
+ expect(result).toBeNull();
585
+ });
586
+
587
+ test("empty values: returns null", () => {
588
+ const rules: Ruleset = [allowAll, denyEnv];
589
+ const result = evaluateMostRestrictive("path", [], rules);
590
+ expect(result).toBeNull();
591
+ });
592
+
593
+ test("deny wins over ask", () => {
594
+ const rules: Ruleset = [allowAll, askSsh, denyEnv];
595
+ const result = evaluateMostRestrictive(
596
+ "path",
597
+ ["/home/user/.ssh/id_rsa", ".env"],
598
+ rules,
599
+ );
600
+ expect(result).not.toBeNull();
601
+ expect(result!.rule.action).toBe("deny");
602
+ expect(result!.value).toBe(".env");
603
+ });
604
+ });