@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,77 @@
1
+ import { describe, expect, test } from "vitest";
2
+
3
+ import {
4
+ type ToolAccessExtractor,
5
+ ToolAccessExtractorRegistry,
6
+ } from "#src/tool-access-extractor-registry";
7
+
8
+ const noopExtractor: ToolAccessExtractor = () => "/tmp/x";
9
+
10
+ describe("ToolAccessExtractorRegistry", () => {
11
+ describe("register", () => {
12
+ test("stores an extractor so get() returns it", () => {
13
+ const registry = new ToolAccessExtractorRegistry();
14
+ registry.register("my-tool", noopExtractor);
15
+ expect(registry.get("my-tool")).toBe(noopExtractor);
16
+ });
17
+
18
+ test("returns a disposer that removes the extractor", () => {
19
+ const registry = new ToolAccessExtractorRegistry();
20
+ const dispose = registry.register("my-tool", noopExtractor);
21
+ dispose();
22
+ expect(registry.get("my-tool")).toBeUndefined();
23
+ });
24
+
25
+ test("throws when an extractor is already registered for the same tool name", () => {
26
+ const registry = new ToolAccessExtractorRegistry();
27
+ registry.register("my-tool", noopExtractor);
28
+ expect(() => registry.register("my-tool", () => undefined)).toThrow(
29
+ "my-tool",
30
+ );
31
+ });
32
+
33
+ test("allows registering different tool names independently", () => {
34
+ const registry = new ToolAccessExtractorRegistry();
35
+ const extractorA: ToolAccessExtractor = () => "/a";
36
+ const extractorB: ToolAccessExtractor = () => "/b";
37
+ registry.register("tool-a", extractorA);
38
+ registry.register("tool-b", extractorB);
39
+ expect(registry.get("tool-a")).toBe(extractorA);
40
+ expect(registry.get("tool-b")).toBe(extractorB);
41
+ });
42
+ });
43
+
44
+ describe("disposer identity guard", () => {
45
+ test("stale disposer does not evict a later registration", () => {
46
+ const registry = new ToolAccessExtractorRegistry();
47
+ const first: ToolAccessExtractor = () => "/first";
48
+ const second: ToolAccessExtractor = () => "/second";
49
+
50
+ const disposeFirst = registry.register("my-tool", first);
51
+ disposeFirst(); // removes first
52
+
53
+ registry.register("my-tool", second); // second registration is now valid
54
+ disposeFirst(); // calling stale disposer again — must not remove second
55
+
56
+ expect(registry.get("my-tool")).toBe(second);
57
+ });
58
+ });
59
+
60
+ describe("get", () => {
61
+ test("returns undefined for an unregistered tool name", () => {
62
+ const registry = new ToolAccessExtractorRegistry();
63
+ expect(registry.get("unknown")).toBeUndefined();
64
+ });
65
+
66
+ test("the registered extractor is callable and returns its path", () => {
67
+ const registry = new ToolAccessExtractorRegistry();
68
+ const extractor: ToolAccessExtractor = (input) =>
69
+ typeof input.target === "string" ? input.target : undefined;
70
+ registry.register("ffgrep", extractor);
71
+ expect(registry.get("ffgrep")?.({ target: "/etc/hosts" })).toBe(
72
+ "/etc/hosts",
73
+ );
74
+ expect(registry.get("ffgrep")?.({ other: true })).toBeUndefined();
75
+ });
76
+ });
77
+ });
@@ -0,0 +1,75 @@
1
+ import { describe, expect, test } from "vitest";
2
+
3
+ import {
4
+ type ToolInputFormatter,
5
+ ToolInputFormatterRegistry,
6
+ } from "#src/tool-input-formatter-registry";
7
+
8
+ const noopFormatter: ToolInputFormatter = () => "preview";
9
+
10
+ describe("ToolInputFormatterRegistry", () => {
11
+ describe("register", () => {
12
+ test("stores a formatter so get() returns it", () => {
13
+ const registry = new ToolInputFormatterRegistry();
14
+ registry.register("my-tool", noopFormatter);
15
+ expect(registry.get("my-tool")).toBe(noopFormatter);
16
+ });
17
+
18
+ test("returns a disposer that removes the formatter", () => {
19
+ const registry = new ToolInputFormatterRegistry();
20
+ const dispose = registry.register("my-tool", noopFormatter);
21
+ dispose();
22
+ expect(registry.get("my-tool")).toBeUndefined();
23
+ });
24
+
25
+ test("throws when a formatter is already registered for the same tool name", () => {
26
+ const registry = new ToolInputFormatterRegistry();
27
+ registry.register("my-tool", noopFormatter);
28
+ expect(() => registry.register("my-tool", () => undefined)).toThrow(
29
+ "my-tool",
30
+ );
31
+ });
32
+
33
+ test("allows registering different tool names independently", () => {
34
+ const registry = new ToolInputFormatterRegistry();
35
+ const formatterA: ToolInputFormatter = () => "a";
36
+ const formatterB: ToolInputFormatter = () => "b";
37
+ registry.register("tool-a", formatterA);
38
+ registry.register("tool-b", formatterB);
39
+ expect(registry.get("tool-a")).toBe(formatterA);
40
+ expect(registry.get("tool-b")).toBe(formatterB);
41
+ });
42
+ });
43
+
44
+ describe("disposer identity guard", () => {
45
+ test("stale disposer does not evict a later registration", () => {
46
+ const registry = new ToolInputFormatterRegistry();
47
+ const first: ToolInputFormatter = () => "first";
48
+ const second: ToolInputFormatter = () => "second";
49
+
50
+ const disposeFirst = registry.register("my-tool", first);
51
+ disposeFirst(); // removes first
52
+
53
+ registry.register("my-tool", second); // second registration is now valid
54
+ disposeFirst(); // calling stale disposer again — must not remove second
55
+
56
+ expect(registry.get("my-tool")).toBe(second);
57
+ });
58
+ });
59
+
60
+ describe("get", () => {
61
+ test("returns undefined for an unregistered tool name", () => {
62
+ const registry = new ToolInputFormatterRegistry();
63
+ expect(registry.get("unknown")).toBeUndefined();
64
+ });
65
+
66
+ test("the registered formatter is callable and returns its result", () => {
67
+ const registry = new ToolInputFormatterRegistry();
68
+ const fmt: ToolInputFormatter = (input) =>
69
+ typeof input.cmd === "string" ? `runs ${input.cmd}` : undefined;
70
+ registry.register("run", fmt);
71
+ expect(registry.get("run")?.({ cmd: "ls" })).toBe("runs ls");
72
+ expect(registry.get("run")?.({ other: true })).toBeUndefined();
73
+ });
74
+ });
75
+ });
@@ -0,0 +1,129 @@
1
+ import { afterEach, beforeEach, describe, expect, test, vi } from "vitest";
2
+
3
+ // Mock logging collaborator before importing the module under test.
4
+ vi.mock("../src/logging.js", () => ({
5
+ safeJsonStringify: vi.fn((value: unknown) => JSON.stringify(value)),
6
+ }));
7
+
8
+ import { safeJsonStringify } from "#src/logging";
9
+ import {
10
+ countTextLines,
11
+ formatCount,
12
+ serializeToolInputPreview,
13
+ TOOL_INPUT_LOG_PREVIEW_MAX_LENGTH,
14
+ TOOL_INPUT_PREVIEW_MAX_LENGTH,
15
+ TOOL_TEXT_SUMMARY_MAX_LENGTH,
16
+ truncateInlineText,
17
+ } from "#src/tool-input-preview";
18
+
19
+ const mockedStringify = vi.mocked(safeJsonStringify);
20
+
21
+ beforeEach(() => {
22
+ mockedStringify.mockReset();
23
+ });
24
+
25
+ afterEach(() => {
26
+ vi.restoreAllMocks();
27
+ });
28
+
29
+ describe("constants", () => {
30
+ test("TOOL_INPUT_PREVIEW_MAX_LENGTH is 200", () => {
31
+ expect(TOOL_INPUT_PREVIEW_MAX_LENGTH).toBe(200);
32
+ });
33
+
34
+ test("TOOL_INPUT_LOG_PREVIEW_MAX_LENGTH is 1000", () => {
35
+ expect(TOOL_INPUT_LOG_PREVIEW_MAX_LENGTH).toBe(1000);
36
+ });
37
+
38
+ test("TOOL_TEXT_SUMMARY_MAX_LENGTH is 80", () => {
39
+ expect(TOOL_TEXT_SUMMARY_MAX_LENGTH).toBe(80);
40
+ });
41
+ });
42
+
43
+ describe("truncateInlineText", () => {
44
+ test("returns text unchanged when within maxLength", () => {
45
+ expect(truncateInlineText("hello", 10)).toBe("hello");
46
+ });
47
+
48
+ test("does not truncate when length equals maxLength", () => {
49
+ const text = "a".repeat(200);
50
+ expect(truncateInlineText(text, 200)).toBe(text);
51
+ });
52
+
53
+ test("truncates and appends ellipsis when length exceeds maxLength", () => {
54
+ const text = "a".repeat(201);
55
+ const result = truncateInlineText(text, 200);
56
+ expect(result).toBe(`${"a".repeat(200)}…`);
57
+ });
58
+
59
+ test("truncates long text and appends ellipsis", () => {
60
+ const result = truncateInlineText("abcdef", 3);
61
+ expect(result).toBe("abc…");
62
+ });
63
+ });
64
+
65
+ describe("countTextLines", () => {
66
+ test("returns 0 for empty string", () => {
67
+ expect(countTextLines("")).toBe(0);
68
+ });
69
+
70
+ test("returns 1 for a single line with no newline", () => {
71
+ expect(countTextLines("hello")).toBe(1);
72
+ });
73
+
74
+ test("counts LF-separated lines", () => {
75
+ expect(countTextLines("line1\nline2\nline3")).toBe(3);
76
+ });
77
+
78
+ test("counts CRLF-separated lines", () => {
79
+ expect(countTextLines("line1\r\nline2")).toBe(2);
80
+ });
81
+
82
+ test("counts CR-separated lines", () => {
83
+ expect(countTextLines("line1\rline2")).toBe(2);
84
+ });
85
+ });
86
+
87
+ describe("formatCount", () => {
88
+ test("uses singular form for 1", () => {
89
+ expect(formatCount(1, "line", "lines")).toBe("1 line");
90
+ });
91
+
92
+ test("uses plural form for 0", () => {
93
+ expect(formatCount(0, "line", "lines")).toBe("0 lines");
94
+ });
95
+
96
+ test("uses plural form for 2+", () => {
97
+ expect(formatCount(3, "line", "lines")).toBe("3 lines");
98
+ });
99
+ });
100
+
101
+ describe("serializeToolInputPreview", () => {
102
+ test("delegates serialization to safeJsonStringify", () => {
103
+ mockedStringify.mockReturnValue('{"key":"value"}');
104
+ const result = serializeToolInputPreview({ key: "value" });
105
+ expect(mockedStringify).toHaveBeenCalledWith({ key: "value" });
106
+ expect(result).toBe('{"key":"value"}');
107
+ });
108
+
109
+ test("returns empty string when safeJsonStringify returns undefined", () => {
110
+ mockedStringify.mockReturnValue(undefined);
111
+ expect(serializeToolInputPreview({})).toBe("");
112
+ });
113
+
114
+ test("returns empty string when serialized value is '{}'", () => {
115
+ mockedStringify.mockReturnValue("{}");
116
+ expect(serializeToolInputPreview({})).toBe("");
117
+ });
118
+
119
+ test("returns empty string when serialized value is 'null'", () => {
120
+ mockedStringify.mockReturnValue("null");
121
+ expect(serializeToolInputPreview(null)).toBe("");
122
+ });
123
+
124
+ test("collapses whitespace in serialized output", () => {
125
+ mockedStringify.mockReturnValue('{\n "key": "val"\n}');
126
+ const result = serializeToolInputPreview({});
127
+ expect(result).toBe('{ "key": "val" }');
128
+ });
129
+ });
@@ -0,0 +1,115 @@
1
+ import { describe, expect, test } from "vitest";
2
+
3
+ import {
4
+ formatEditInputForPrompt,
5
+ formatReadInputForPrompt,
6
+ formatWriteInputForPrompt,
7
+ getPromptPath,
8
+ } from "#src/tool-input-prompt-formatters";
9
+
10
+ describe("getPromptPath", () => {
11
+ test("returns path from 'path' key", () => {
12
+ expect(getPromptPath({ path: "/foo/bar" })).toBe("/foo/bar");
13
+ });
14
+
15
+ test("falls back to 'file_path' key", () => {
16
+ expect(getPromptPath({ file_path: "/baz" })).toBe("/baz");
17
+ });
18
+
19
+ test("returns null when neither key is present", () => {
20
+ expect(getPromptPath({})).toBeNull();
21
+ });
22
+
23
+ test("returns null when path is empty string", () => {
24
+ expect(getPromptPath({ path: "" })).toBeNull();
25
+ });
26
+ });
27
+
28
+ describe("formatEditInputForPrompt", () => {
29
+ test("returns path-only description when no edits provided", () => {
30
+ const result = formatEditInputForPrompt({ path: "/foo.ts" });
31
+ expect(result).toBe("for '/foo.ts' with edit input");
32
+ });
33
+
34
+ test("formats single replacement with line counts", () => {
35
+ const result = formatEditInputForPrompt({
36
+ path: "/foo.ts",
37
+ edits: [{ oldText: "line1\nline2", newText: "replaced" }],
38
+ });
39
+ expect(result).toContain("for '/foo.ts'");
40
+ expect(result).toContain("1 replacement");
41
+ expect(result).toContain("2 lines");
42
+ expect(result).toContain("1 line");
43
+ });
44
+
45
+ test("formats multiple replacements mentioning additional edits", () => {
46
+ const result = formatEditInputForPrompt({
47
+ path: "/foo.ts",
48
+ edits: [
49
+ { oldText: "a", newText: "b" },
50
+ { oldText: "c", newText: "d" },
51
+ { oldText: "e", newText: "f" },
52
+ ],
53
+ });
54
+ expect(result).toContain("3 replacements");
55
+ expect(result).toContain("2 additional edits");
56
+ });
57
+
58
+ test("falls back to oldText/newText when no edits array", () => {
59
+ const result = formatEditInputForPrompt({
60
+ path: "/bar.ts",
61
+ oldText: "old",
62
+ newText: "new",
63
+ });
64
+ expect(result).toContain("for '/bar.ts'");
65
+ expect(result).toContain("1 replacement");
66
+ });
67
+
68
+ test("works without a path", () => {
69
+ const result = formatEditInputForPrompt({
70
+ edits: [{ oldText: "x", newText: "y" }],
71
+ });
72
+ expect(result).not.toContain("for '");
73
+ expect(result).toContain("1 replacement");
74
+ });
75
+ });
76
+
77
+ describe("formatWriteInputForPrompt", () => {
78
+ test("includes path, line count, and character count", () => {
79
+ const result = formatWriteInputForPrompt({
80
+ path: "/out.ts",
81
+ content: "line1\nline2",
82
+ });
83
+ expect(result).toContain("for '/out.ts'");
84
+ expect(result).toContain("2 lines");
85
+ expect(result).toContain("11 characters");
86
+ });
87
+
88
+ test("handles missing content as empty", () => {
89
+ const result = formatWriteInputForPrompt({ path: "/out.ts" });
90
+ expect(result).toContain("0 lines");
91
+ expect(result).toContain("0 characters");
92
+ });
93
+ });
94
+
95
+ describe("formatReadInputForPrompt", () => {
96
+ test("includes path", () => {
97
+ expect(formatReadInputForPrompt({ path: "/src/foo.ts" })).toBe(
98
+ "for path '/src/foo.ts'",
99
+ );
100
+ });
101
+
102
+ test("includes offset and limit when present", () => {
103
+ const result = formatReadInputForPrompt({
104
+ path: "/x",
105
+ offset: 10,
106
+ limit: 50,
107
+ });
108
+ expect(result).toContain("offset 10");
109
+ expect(result).toContain("limit 50");
110
+ });
111
+
112
+ test("returns empty string when no path and no options", () => {
113
+ expect(formatReadInputForPrompt({})).toBe("");
114
+ });
115
+ });