@nanocollective/nanocoder 1.14.3 → 1.15.1

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 (192) hide show
  1. package/README.md +31 -1
  2. package/dist/ai-sdk-client.d.ts +22 -0
  3. package/dist/ai-sdk-client.d.ts.map +1 -0
  4. package/dist/ai-sdk-client.js +389 -0
  5. package/dist/ai-sdk-client.js.map +1 -0
  6. package/dist/app.d.ts.map +1 -1
  7. package/dist/app.js +1 -1
  8. package/dist/app.js.map +1 -1
  9. package/dist/client-factory.d.ts.map +1 -1
  10. package/dist/client-factory.js +5 -5
  11. package/dist/client-factory.js.map +1 -1
  12. package/dist/commands/index.d.ts +1 -0
  13. package/dist/commands/index.d.ts.map +1 -1
  14. package/dist/commands/index.js +1 -0
  15. package/dist/commands/index.js.map +1 -1
  16. package/dist/commands/recommendations.js +1 -1
  17. package/dist/commands/recommendations.js.map +1 -1
  18. package/dist/commands/streaming.d.ts +3 -0
  19. package/dist/commands/streaming.d.ts.map +1 -0
  20. package/dist/commands/streaming.js +23 -0
  21. package/dist/commands/streaming.js.map +1 -0
  22. package/dist/commands.d.ts.map +1 -1
  23. package/dist/commands.js +17 -3
  24. package/dist/commands.js.map +1 -1
  25. package/dist/components/assistant-message.d.ts +0 -1
  26. package/dist/components/assistant-message.d.ts.map +1 -1
  27. package/dist/components/assistant-message.js +70 -122
  28. package/dist/components/assistant-message.js.map +1 -1
  29. package/dist/components/thinking-indicator.js +1 -1
  30. package/dist/components/thinking-indicator.js.map +1 -1
  31. package/dist/components/user-input.d.ts.map +1 -1
  32. package/dist/components/user-input.js +135 -27
  33. package/dist/components/user-input.js.map +1 -1
  34. package/dist/components/user-message.d.ts.map +1 -1
  35. package/dist/components/user-message.js +34 -1
  36. package/dist/components/user-message.js.map +1 -1
  37. package/dist/config/index.d.ts.map +1 -1
  38. package/dist/config/index.js +2 -1
  39. package/dist/config/index.js.map +1 -1
  40. package/dist/config/index.spec.d.ts +2 -0
  41. package/dist/config/index.spec.d.ts.map +1 -0
  42. package/dist/config/index.spec.js +99 -0
  43. package/dist/config/index.spec.js.map +1 -0
  44. package/dist/config/preferences.d.ts +2 -0
  45. package/dist/config/preferences.d.ts.map +1 -1
  46. package/dist/config/preferences.js +10 -0
  47. package/dist/config/preferences.js.map +1 -1
  48. package/dist/hooks/useAppInitialization.d.ts.map +1 -1
  49. package/dist/hooks/useAppInitialization.js +2 -1
  50. package/dist/hooks/useAppInitialization.js.map +1 -1
  51. package/dist/hooks/useChatHandler.d.ts +2 -0
  52. package/dist/hooks/useChatHandler.d.ts.map +1 -1
  53. package/dist/hooks/useChatHandler.js +59 -9
  54. package/dist/hooks/useChatHandler.js.map +1 -1
  55. package/dist/mcp/mcp-client.d.ts +20 -1
  56. package/dist/mcp/mcp-client.d.ts.map +1 -1
  57. package/dist/mcp/mcp-client.js +57 -0
  58. package/dist/mcp/mcp-client.js.map +1 -1
  59. package/dist/message-handler.d.ts.map +1 -1
  60. package/dist/message-handler.js +7 -2
  61. package/dist/message-handler.js.map +1 -1
  62. package/dist/tool-calling/index.d.ts +1 -1
  63. package/dist/tool-calling/index.d.ts.map +1 -1
  64. package/dist/tool-calling/index.js +1 -1
  65. package/dist/tool-calling/index.js.map +1 -1
  66. package/dist/tool-calling/json-parser.d.ts +19 -2
  67. package/dist/tool-calling/json-parser.d.ts.map +1 -1
  68. package/dist/tool-calling/json-parser.js +65 -28
  69. package/dist/tool-calling/json-parser.js.map +1 -1
  70. package/dist/tool-calling/json-parser.spec.d.ts +2 -0
  71. package/dist/tool-calling/json-parser.spec.d.ts.map +1 -0
  72. package/dist/tool-calling/json-parser.spec.js +518 -0
  73. package/dist/tool-calling/json-parser.spec.js.map +1 -0
  74. package/dist/tool-calling/tool-parser.d.ts +20 -0
  75. package/dist/tool-calling/tool-parser.d.ts.map +1 -0
  76. package/dist/tool-calling/tool-parser.js +58 -0
  77. package/dist/tool-calling/tool-parser.js.map +1 -0
  78. package/dist/tool-calling/tool-parser.spec.d.ts +2 -0
  79. package/dist/tool-calling/tool-parser.spec.d.ts.map +1 -0
  80. package/dist/tool-calling/tool-parser.spec.js +250 -0
  81. package/dist/tool-calling/tool-parser.spec.js.map +1 -0
  82. package/dist/tool-calling/xml-parser.d.ts +17 -0
  83. package/dist/tool-calling/xml-parser.d.ts.map +1 -1
  84. package/dist/tool-calling/xml-parser.js +101 -13
  85. package/dist/tool-calling/xml-parser.js.map +1 -1
  86. package/dist/tool-calling/xml-parser.spec.d.ts +2 -0
  87. package/dist/tool-calling/xml-parser.spec.d.ts.map +1 -0
  88. package/dist/tool-calling/xml-parser.spec.js +437 -0
  89. package/dist/tool-calling/xml-parser.spec.js.map +1 -0
  90. package/dist/tools/create-file.d.ts.map +1 -1
  91. package/dist/tools/create-file.js +26 -23
  92. package/dist/tools/create-file.js.map +1 -1
  93. package/dist/tools/delete-lines.d.ts.map +1 -1
  94. package/dist/tools/delete-lines.js +29 -27
  95. package/dist/tools/delete-lines.js.map +1 -1
  96. package/dist/tools/execute-bash.d.ts.map +1 -1
  97. package/dist/tools/execute-bash.js +21 -19
  98. package/dist/tools/execute-bash.js.map +1 -1
  99. package/dist/tools/fetch-url.d.ts.map +1 -1
  100. package/dist/tools/fetch-url.js +21 -19
  101. package/dist/tools/fetch-url.js.map +1 -1
  102. package/dist/tools/index.d.ts +2 -2
  103. package/dist/tools/index.d.ts.map +1 -1
  104. package/dist/tools/index.js +8 -6
  105. package/dist/tools/index.js.map +1 -1
  106. package/dist/tools/insert-lines.d.ts.map +1 -1
  107. package/dist/tools/insert-lines.js +29 -27
  108. package/dist/tools/insert-lines.js.map +1 -1
  109. package/dist/tools/read-file.d.ts.map +1 -1
  110. package/dist/tools/read-file.js +23 -19
  111. package/dist/tools/read-file.js.map +1 -1
  112. package/dist/tools/read-many-files.d.ts.map +1 -1
  113. package/dist/tools/read-many-files.js +24 -20
  114. package/dist/tools/read-many-files.js.map +1 -1
  115. package/dist/tools/replace-lines.d.ts.map +1 -1
  116. package/dist/tools/replace-lines.js +33 -31
  117. package/dist/tools/replace-lines.js.map +1 -1
  118. package/dist/tools/search-files.d.ts.map +1 -1
  119. package/dist/tools/search-files.js +33 -31
  120. package/dist/tools/search-files.js.map +1 -1
  121. package/dist/tools/tool-manager.d.ts +35 -19
  122. package/dist/tools/tool-manager.d.ts.map +1 -1
  123. package/dist/tools/tool-manager.js +63 -33
  124. package/dist/tools/tool-manager.js.map +1 -1
  125. package/dist/tools/tool-registry.d.ts +121 -0
  126. package/dist/tools/tool-registry.d.ts.map +1 -0
  127. package/dist/tools/tool-registry.js +195 -0
  128. package/dist/tools/tool-registry.js.map +1 -0
  129. package/dist/tools/web-search.d.ts.map +1 -1
  130. package/dist/tools/web-search.js +25 -23
  131. package/dist/tools/web-search.js.map +1 -1
  132. package/dist/types/config.d.ts +2 -1
  133. package/dist/types/config.d.ts.map +1 -1
  134. package/dist/types/core.d.ts +58 -3
  135. package/dist/types/core.d.ts.map +1 -1
  136. package/dist/types/core.js +2 -0
  137. package/dist/types/core.js.map +1 -1
  138. package/dist/utils/file-autocomplete.d.ts +31 -0
  139. package/dist/utils/file-autocomplete.d.ts.map +1 -0
  140. package/dist/utils/file-autocomplete.js +156 -0
  141. package/dist/utils/file-autocomplete.js.map +1 -0
  142. package/dist/utils/file-autocomplete.spec.d.ts +2 -0
  143. package/dist/utils/file-autocomplete.spec.d.ts.map +1 -0
  144. package/dist/utils/file-autocomplete.spec.js +142 -0
  145. package/dist/utils/file-autocomplete.spec.js.map +1 -0
  146. package/dist/utils/file-content-loader.d.ts +31 -0
  147. package/dist/utils/file-content-loader.d.ts.map +1 -0
  148. package/dist/utils/file-content-loader.js +142 -0
  149. package/dist/utils/file-content-loader.js.map +1 -0
  150. package/dist/utils/file-content-loader.spec.d.ts +2 -0
  151. package/dist/utils/file-content-loader.spec.d.ts.map +1 -0
  152. package/dist/utils/file-content-loader.spec.js +140 -0
  153. package/dist/utils/file-content-loader.spec.js.map +1 -0
  154. package/dist/utils/file-mention-handler.d.ts +21 -0
  155. package/dist/utils/file-mention-handler.d.ts.map +1 -0
  156. package/dist/utils/file-mention-handler.js +59 -0
  157. package/dist/utils/file-mention-handler.js.map +1 -0
  158. package/dist/utils/file-mention-handler.spec.d.ts +2 -0
  159. package/dist/utils/file-mention-handler.spec.d.ts.map +1 -0
  160. package/dist/utils/file-mention-handler.spec.js +147 -0
  161. package/dist/utils/file-mention-handler.spec.js.map +1 -0
  162. package/dist/utils/file-mention-parser.d.ts +40 -0
  163. package/dist/utils/file-mention-parser.d.ts.map +1 -0
  164. package/dist/utils/file-mention-parser.js +122 -0
  165. package/dist/utils/file-mention-parser.js.map +1 -0
  166. package/dist/utils/file-mention-parser.spec.d.ts +2 -0
  167. package/dist/utils/file-mention-parser.spec.d.ts.map +1 -0
  168. package/dist/utils/file-mention-parser.spec.js +149 -0
  169. package/dist/utils/file-mention-parser.spec.js.map +1 -0
  170. package/dist/utils/fuzzy-matching.d.ts +13 -0
  171. package/dist/utils/fuzzy-matching.d.ts.map +1 -0
  172. package/dist/utils/fuzzy-matching.js +127 -0
  173. package/dist/utils/fuzzy-matching.js.map +1 -0
  174. package/dist/utils/fuzzy-matching.spec.d.ts +2 -0
  175. package/dist/utils/fuzzy-matching.spec.d.ts.map +1 -0
  176. package/dist/utils/fuzzy-matching.spec.js +123 -0
  177. package/dist/utils/fuzzy-matching.spec.js.map +1 -0
  178. package/dist/utils/prompt-assembly.spec.js +100 -2
  179. package/dist/utils/prompt-assembly.spec.js.map +1 -1
  180. package/dist/utils/prompt-processor.d.ts +2 -2
  181. package/dist/utils/prompt-processor.d.ts.map +1 -1
  182. package/dist/utils/prompt-processor.js +24 -41
  183. package/dist/utils/prompt-processor.js.map +1 -1
  184. package/package.json +8 -5
  185. package/dist/langgraph-client.d.ts +0 -19
  186. package/dist/langgraph-client.d.ts.map +0 -1
  187. package/dist/langgraph-client.js +0 -295
  188. package/dist/langgraph-client.js.map +0 -1
  189. package/dist/mcp/mcp-tool-adapter.d.ts +0 -22
  190. package/dist/mcp/mcp-tool-adapter.d.ts.map +0 -1
  191. package/dist/mcp/mcp-tool-adapter.js +0 -50
  192. package/dist/mcp/mcp-tool-adapter.js.map +0 -1
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-content-loader.js","sourceRoot":"","sources":["../../source/utils/file-content-loader.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=file-content-loader.spec.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-content-loader.spec.d.ts","sourceRoot":"","sources":["../../source/utils/file-content-loader.spec.ts"],"names":[],"mappings":""}
@@ -0,0 +1,140 @@
1
+ import test from 'ava';
2
+ import { loadFileContent, formatFileForContext } from './file-content-loader.js';
3
+ import { writeFile, mkdir, rm } from 'node:fs/promises';
4
+ import { join } from 'node:path';
5
+ import { tmpdir } from 'node:os';
6
+ console.log(`\nfile-content-loader.spec.ts`);
7
+ // Test setup - create temp directory for test files
8
+ let testDir;
9
+ test.before(async () => {
10
+ // Create a unique temp directory for our tests
11
+ testDir = join(tmpdir(), `nanocoder-test-${Date.now()}`);
12
+ await mkdir(testDir, { recursive: true });
13
+ // Create test files
14
+ await writeFile(join(testDir, 'test.txt'), 'Line 1\nLine 2\nLine 3\nLine 4\nLine 5', 'utf-8');
15
+ await writeFile(join(testDir, 'app.tsx'), 'import React from "react";\n\nexport function App() {\n return <div>Hello</div>;\n}', 'utf-8');
16
+ });
17
+ test.after.always(async () => {
18
+ // Clean up temp directory
19
+ try {
20
+ await rm(testDir, { recursive: true, force: true });
21
+ }
22
+ catch {
23
+ // Ignore cleanup errors
24
+ }
25
+ });
26
+ // Tests for loadFileContent()
27
+ test('loads file content successfully', async (t) => {
28
+ const result = await loadFileContent(join(testDir, 'test.txt'));
29
+ t.true(result.success);
30
+ t.truthy(result.content);
31
+ t.is(result.metadata.lineCount, 5);
32
+ t.true(result.metadata.size > 0);
33
+ });
34
+ test('loads file with line range', async (t) => {
35
+ const result = await loadFileContent(join(testDir, 'test.txt'), {
36
+ start: 2,
37
+ end: 4,
38
+ });
39
+ t.true(result.success);
40
+ t.truthy(result.content);
41
+ t.is(result.metadata.lineCount, 3); // Lines 2, 3, 4
42
+ t.deepEqual(result.metadata.lineRange, { start: 2, end: 4 });
43
+ });
44
+ test('loads single line', async (t) => {
45
+ const result = await loadFileContent(join(testDir, 'test.txt'), {
46
+ start: 3,
47
+ });
48
+ t.true(result.success);
49
+ t.is(result.metadata.lineCount, 1);
50
+ t.deepEqual(result.metadata.lineRange, { start: 3, end: undefined });
51
+ });
52
+ test('handles line range beyond file length', async (t) => {
53
+ const result = await loadFileContent(join(testDir, 'test.txt'), {
54
+ start: 3,
55
+ end: 100,
56
+ });
57
+ t.true(result.success);
58
+ t.is(result.metadata.lineCount, 3); // Lines 3, 4, 5 (only 3 lines available)
59
+ });
60
+ test('handles line start beyond file length', async (t) => {
61
+ const result = await loadFileContent(join(testDir, 'test.txt'), {
62
+ start: 100,
63
+ end: 200,
64
+ });
65
+ t.true(result.success);
66
+ t.is(result.metadata.lineCount, 0); // No lines available
67
+ });
68
+ test('handles non-existent file', async (t) => {
69
+ const result = await loadFileContent(join(testDir, 'nonexistent.txt'));
70
+ t.false(result.success);
71
+ t.truthy(result.error);
72
+ t.is(result.metadata.size, 0);
73
+ t.is(result.metadata.lineCount, 0);
74
+ });
75
+ test('formats content with line numbers', async (t) => {
76
+ const result = await loadFileContent(join(testDir, 'test.txt'), {
77
+ start: 2,
78
+ end: 3,
79
+ });
80
+ t.true(result.success);
81
+ t.truthy(result.content);
82
+ // Should have line numbers like " 2: Line 2"
83
+ t.true(result.content.includes(' 2:'));
84
+ t.true(result.content.includes(' 3:'));
85
+ });
86
+ test('calculates token estimate', async (t) => {
87
+ const result = await loadFileContent(join(testDir, 'test.txt'));
88
+ t.true(result.success);
89
+ t.true(result.metadata.tokens > 0);
90
+ // Rough token estimate should be ~1/4 of character count
91
+ t.true(result.metadata.tokens <= result.metadata.size);
92
+ });
93
+ test('stores absolute path in metadata', async (t) => {
94
+ const relativePath = join(testDir, 'test.txt');
95
+ const result = await loadFileContent(relativePath);
96
+ t.true(result.success);
97
+ t.is(result.metadata.path, relativePath);
98
+ t.truthy(result.metadata.absolutePath);
99
+ // Absolute path should be longer or equal (handles relative paths)
100
+ t.true(result.metadata.absolutePath.length >= relativePath.length);
101
+ });
102
+ // Tests for formatFileForContext()
103
+ test('formats file for LLM context with header and footer', async (t) => {
104
+ const result = await loadFileContent(join(testDir, 'app.tsx'));
105
+ const formatted = formatFileForContext(result);
106
+ // The header includes the full path, not just filename
107
+ t.true(formatted.includes('=== File:'));
108
+ t.true(formatted.includes('app.tsx'));
109
+ t.true(formatted.includes('==='));
110
+ t.true(result.content ? formatted.includes(result.content) : false);
111
+ });
112
+ test('formats file with line range info', async (t) => {
113
+ const result = await loadFileContent(join(testDir, 'test.txt'), {
114
+ start: 2,
115
+ end: 4,
116
+ });
117
+ const formatted = formatFileForContext(result);
118
+ t.true(formatted.includes('Lines 2-4'));
119
+ t.true(formatted.includes('(3 lines'));
120
+ });
121
+ test('formats file with single line info', async (t) => {
122
+ const result = await loadFileContent(join(testDir, 'test.txt'), {
123
+ start: 3,
124
+ });
125
+ const formatted = formatFileForContext(result);
126
+ t.true(formatted.includes('Lines 3'));
127
+ t.false(formatted.includes('Lines 3-')); // Should not have a range
128
+ });
129
+ test('handles error in formatFileForContext', async (t) => {
130
+ const result = await loadFileContent(join(testDir, 'nonexistent.txt'));
131
+ const formatted = formatFileForContext(result);
132
+ t.true(formatted.includes('Error loading file'));
133
+ });
134
+ test('includes token estimate in formatted output', async (t) => {
135
+ const result = await loadFileContent(join(testDir, 'test.txt'));
136
+ const formatted = formatFileForContext(result);
137
+ t.true(formatted.includes('tokens'));
138
+ t.true(/~\d+ tokens/.test(formatted)); // Should have "~123 tokens" format
139
+ });
140
+ //# sourceMappingURL=file-content-loader.spec.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-content-loader.spec.js","sourceRoot":"","sources":["../../source/utils/file-content-loader.spec.ts"],"names":[],"mappings":"AAAA,OAAO,IAAI,MAAM,KAAK,CAAC;AACvB,OAAO,EAAC,eAAe,EAAE,oBAAoB,EAAC,MAAM,0BAA0B,CAAC;AAC/E,OAAO,EAAC,SAAS,EAAE,KAAK,EAAE,EAAE,EAAC,MAAM,kBAAkB,CAAC;AACtD,OAAO,EAAC,IAAI,EAAC,MAAM,WAAW,CAAC;AAC/B,OAAO,EAAC,MAAM,EAAC,MAAM,SAAS,CAAC;AAE/B,OAAO,CAAC,GAAG,CAAC,+BAA+B,CAAC,CAAC;AAE7C,oDAAoD;AACpD,IAAI,OAAe,CAAC;AAEpB,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE;IACtB,+CAA+C;IAC/C,OAAO,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,kBAAkB,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IACzD,MAAM,KAAK,CAAC,OAAO,EAAE,EAAC,SAAS,EAAE,IAAI,EAAC,CAAC,CAAC;IAExC,oBAAoB;IACpB,MAAM,SAAS,CACd,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,EACzB,wCAAwC,EACxC,OAAO,CACP,CAAC;IAEF,MAAM,SAAS,CACd,IAAI,CAAC,OAAO,EAAE,SAAS,CAAC,EACxB,sFAAsF,EACtF,OAAO,CACP,CAAC;AACH,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE;IAC5B,0BAA0B;IAC1B,IAAI,CAAC;QACJ,MAAM,EAAE,CAAC,OAAO,EAAE,EAAC,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAC,CAAC,CAAC;IACnD,CAAC;IAAC,MAAM,CAAC;QACR,wBAAwB;IACzB,CAAC;AACF,CAAC,CAAC,CAAC;AAEH,8BAA8B;AAC9B,IAAI,CAAC,iCAAiC,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACjD,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC,CAAC;IAEhE,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACvB,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACzB,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;IACnC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC;AAClC,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,4BAA4B,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IAC5C,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,EAAE;QAC/D,KAAK,EAAE,CAAC;QACR,GAAG,EAAE,CAAC;KACN,CAAC,CAAC;IAEH,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACvB,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACzB,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC,gBAAgB;IACpD,CAAC,CAAC,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,EAAC,KAAK,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAC,CAAC,CAAC;AAC5D,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,mBAAmB,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACnC,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,EAAE;QAC/D,KAAK,EAAE,CAAC;KACR,CAAC,CAAC;IAEH,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACvB,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;IACnC,CAAC,CAAC,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,EAAC,KAAK,EAAE,CAAC,EAAE,GAAG,EAAE,SAAS,EAAC,CAAC,CAAC;AACpE,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,uCAAuC,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACvD,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,EAAE;QAC/D,KAAK,EAAE,CAAC;QACR,GAAG,EAAE,GAAG;KACR,CAAC,CAAC;IAEH,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACvB,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC,yCAAyC;AAC9E,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,uCAAuC,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACvD,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,EAAE;QAC/D,KAAK,EAAE,GAAG;QACV,GAAG,EAAE,GAAG;KACR,CAAC,CAAC;IAEH,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACvB,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC,qBAAqB;AAC1D,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,2BAA2B,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IAC3C,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC,CAAC;IAEvE,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACxB,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IACvB,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;IAC9B,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;AACpC,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,mCAAmC,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACnD,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,EAAE;QAC/D,KAAK,EAAE,CAAC;QACR,GAAG,EAAE,CAAC;KACN,CAAC,CAAC;IAEH,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACvB,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACzB,+CAA+C;IAC/C,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;IAC1C,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;AAC3C,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,2BAA2B,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IAC3C,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC,CAAC;IAEhE,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACvB,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IACnC,yDAAyD;IACzD,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;AACxD,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,kCAAkC,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IAClD,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;IAC/C,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,YAAY,CAAC,CAAC;IAEnD,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACvB,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,YAAY,CAAC,CAAC;IACzC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;IACvC,mEAAmE;IACnE,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,YAAY,CAAC,MAAM,IAAI,YAAY,CAAC,MAAM,CAAC,CAAC;AACpE,CAAC,CAAC,CAAC;AAEH,mCAAmC;AACnC,IAAI,CAAC,qDAAqD,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACrE,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC,CAAC;IAC/D,MAAM,SAAS,GAAG,oBAAoB,CAAC,MAAM,CAAC,CAAC;IAE/C,uDAAuD;IACvD,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC;IACxC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC;IACtC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;IAClC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;AACrE,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,mCAAmC,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACnD,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,EAAE;QAC/D,KAAK,EAAE,CAAC;QACR,GAAG,EAAE,CAAC;KACN,CAAC,CAAC;IACH,MAAM,SAAS,GAAG,oBAAoB,CAAC,MAAM,CAAC,CAAC;IAE/C,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC;IACxC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC;AACxC,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,oCAAoC,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACpD,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,EAAE;QAC/D,KAAK,EAAE,CAAC;KACR,CAAC,CAAC;IACH,MAAM,SAAS,GAAG,oBAAoB,CAAC,MAAM,CAAC,CAAC;IAE/C,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC;IACtC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,0BAA0B;AACpE,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,uCAAuC,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACvD,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC,CAAC;IACvE,MAAM,SAAS,GAAG,oBAAoB,CAAC,MAAM,CAAC,CAAC;IAE/C,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,oBAAoB,CAAC,CAAC,CAAC;AAClD,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,6CAA6C,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IAC7D,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC,CAAC;IAChE,MAAM,SAAS,GAAG,oBAAoB,CAAC,MAAM,CAAC,CAAC;IAE/C,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,CAAC;IACrC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,mCAAmC;AAC3E,CAAC,CAAC,CAAC"}
@@ -0,0 +1,21 @@
1
+ import { InputState, PlaceholderContent } from '../types/hooks.js';
2
+ /**
3
+ * Handle @file mention by creating a placeholder
4
+ * Called when file is selected from autocomplete or on message submit
5
+ *
6
+ * Returns null if file doesn't exist (silent failure per spec)
7
+ */
8
+ export declare function handleFileMention(filePath: string, currentDisplayValue: string, currentPlaceholderContent: Record<string, PlaceholderContent>, mentionText: string, // The original "@src/app.tsx:10-20" text to replace
9
+ lineRange?: {
10
+ start: number;
11
+ end?: number;
12
+ }): Promise<InputState | null>;
13
+ /**
14
+ * Parse line range from mention text if present
15
+ * e.g., "@app.tsx:10-20" -> {start: 10, end: 20}
16
+ */
17
+ export declare function parseLineRangeFromMention(mentionText: string): {
18
+ start: number;
19
+ end?: number;
20
+ } | undefined;
21
+ //# sourceMappingURL=file-mention-handler.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mention-handler.d.ts","sourceRoot":"","sources":["../../source/utils/file-mention-handler.ts"],"names":[],"mappings":"AAAA,OAAO,EACN,UAAU,EACV,kBAAkB,EAElB,MAAM,mBAAmB,CAAC;AAG3B;;;;;GAKG;AACH,wBAAsB,iBAAiB,CACtC,QAAQ,EAAE,MAAM,EAChB,mBAAmB,EAAE,MAAM,EAC3B,yBAAyB,EAAE,MAAM,CAAC,MAAM,EAAE,kBAAkB,CAAC,EAC7D,WAAW,EAAE,MAAM,EAAE,oDAAoD;AACzE,SAAS,CAAC,EAAE;IAAC,KAAK,EAAE,MAAM,CAAC;IAAC,GAAG,CAAC,EAAE,MAAM,CAAA;CAAC,GACvC,OAAO,CAAC,UAAU,GAAG,IAAI,CAAC,CA8C5B;AAED;;;GAGG;AACH,wBAAgB,yBAAyB,CACxC,WAAW,EAAE,MAAM,GACjB;IAAC,KAAK,EAAE,MAAM,CAAC;IAAC,GAAG,CAAC,EAAE,MAAM,CAAA;CAAC,GAAG,SAAS,CAU3C"}
@@ -0,0 +1,59 @@
1
+ import { PlaceholderType, } from '../types/hooks.js';
2
+ import { loadFileContent } from './file-content-loader.js';
3
+ /**
4
+ * Handle @file mention by creating a placeholder
5
+ * Called when file is selected from autocomplete or on message submit
6
+ *
7
+ * Returns null if file doesn't exist (silent failure per spec)
8
+ */
9
+ export async function handleFileMention(filePath, currentDisplayValue, currentPlaceholderContent, mentionText, // The original "@src/app.tsx:10-20" text to replace
10
+ lineRange) {
11
+ // Load file content
12
+ const fileResult = await loadFileContent(filePath, lineRange);
13
+ // If file doesn't exist or failed to load, return null (silently skip per spec)
14
+ if (!fileResult.success) {
15
+ return null;
16
+ }
17
+ // Generate unique ID for this file placeholder
18
+ const existingFileCount = Object.values(currentPlaceholderContent).filter(content => content.type === PlaceholderType.FILE).length;
19
+ const fileId = `file_${existingFileCount + 1}`;
20
+ // Create compact placeholder for display
21
+ const placeholder = lineRange
22
+ ? `[@${filePath}:${lineRange.start}${lineRange.end ? `-${lineRange.end}` : ''}]`
23
+ : `[@${filePath}]`;
24
+ // Create file placeholder content
25
+ // Use the existing FilePlaceholderContent interface
26
+ const fileContent = {
27
+ type: PlaceholderType.FILE,
28
+ displayText: placeholder,
29
+ filePath: fileResult.metadata.absolutePath,
30
+ content: fileResult.content || '',
31
+ lastModified: Date.now(),
32
+ encoding: 'utf-8',
33
+ fileSize: fileResult.metadata.size,
34
+ };
35
+ const newPlaceholderContent = {
36
+ ...currentPlaceholderContent,
37
+ [fileId]: fileContent,
38
+ };
39
+ // Replace the @mention text with placeholder in display
40
+ const newDisplayValue = currentDisplayValue.replace(mentionText, placeholder);
41
+ return {
42
+ displayValue: newDisplayValue,
43
+ placeholderContent: newPlaceholderContent,
44
+ };
45
+ }
46
+ /**
47
+ * Parse line range from mention text if present
48
+ * e.g., "@app.tsx:10-20" -> {start: 10, end: 20}
49
+ */
50
+ export function parseLineRangeFromMention(mentionText) {
51
+ const match = mentionText.match(/:(\d+)(?:-(\d+))?$/);
52
+ if (!match) {
53
+ return undefined;
54
+ }
55
+ const start = parseInt(match[1], 10);
56
+ const end = match[2] ? parseInt(match[2], 10) : undefined;
57
+ return { start, end };
58
+ }
59
+ //# sourceMappingURL=file-mention-handler.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mention-handler.js","sourceRoot":"","sources":["../../source/utils/file-mention-handler.ts"],"names":[],"mappings":"AAAA,OAAO,EAGN,eAAe,GACf,MAAM,mBAAmB,CAAC;AAC3B,OAAO,EAAC,eAAe,EAAC,MAAM,0BAA0B,CAAC;AAEzD;;;;;GAKG;AACH,MAAM,CAAC,KAAK,UAAU,iBAAiB,CACtC,QAAgB,EAChB,mBAA2B,EAC3B,yBAA6D,EAC7D,WAAmB,EAAE,oDAAoD;AACzE,SAAyC;IAEzC,oBAAoB;IACpB,MAAM,UAAU,GAAG,MAAM,eAAe,CAAC,QAAQ,EAAE,SAAS,CAAC,CAAC;IAE9D,gFAAgF;IAChF,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;QACzB,OAAO,IAAI,CAAC;IACb,CAAC;IAED,+CAA+C;IAC/C,MAAM,iBAAiB,GAAG,MAAM,CAAC,MAAM,CAAC,yBAAyB,CAAC,CAAC,MAAM,CACxE,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,IAAI,KAAK,eAAe,CAAC,IAAI,CAChD,CAAC,MAAM,CAAC;IACT,MAAM,MAAM,GAAG,QAAQ,iBAAiB,GAAG,CAAC,EAAE,CAAC;IAE/C,yCAAyC;IACzC,MAAM,WAAW,GAAG,SAAS;QAC5B,CAAC,CAAC,KAAK,QAAQ,IAAI,SAAS,CAAC,KAAK,GAChC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,SAAS,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,EACtC,GAAG;QACL,CAAC,CAAC,KAAK,QAAQ,GAAG,CAAC;IAEpB,kCAAkC;IAClC,oDAAoD;IACpD,MAAM,WAAW,GAAuB;QACvC,IAAI,EAAE,eAAe,CAAC,IAAI;QAC1B,WAAW,EAAE,WAAW;QACxB,QAAQ,EAAE,UAAU,CAAC,QAAQ,CAAC,YAAY;QAC1C,OAAO,EAAE,UAAU,CAAC,OAAO,IAAI,EAAE;QACjC,YAAY,EAAE,IAAI,CAAC,GAAG,EAAE;QACxB,QAAQ,EAAE,OAAO;QACjB,QAAQ,EAAE,UAAU,CAAC,QAAQ,CAAC,IAAI;KAClC,CAAC;IAEF,MAAM,qBAAqB,GAAG;QAC7B,GAAG,yBAAyB;QAC5B,CAAC,MAAM,CAAC,EAAE,WAAW;KACrB,CAAC;IAEF,wDAAwD;IACxD,MAAM,eAAe,GAAG,mBAAmB,CAAC,OAAO,CAAC,WAAW,EAAE,WAAW,CAAC,CAAC;IAE9E,OAAO;QACN,YAAY,EAAE,eAAe;QAC7B,kBAAkB,EAAE,qBAAqB;KACzC,CAAC;AACH,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,yBAAyB,CACxC,WAAmB;IAEnB,MAAM,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;IACtD,IAAI,CAAC,KAAK,EAAE,CAAC;QACZ,OAAO,SAAS,CAAC;IAClB,CAAC;IAED,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IACrC,MAAM,GAAG,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IAE1D,OAAO,EAAC,KAAK,EAAE,GAAG,EAAC,CAAC;AACrB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=file-mention-handler.spec.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mention-handler.spec.d.ts","sourceRoot":"","sources":["../../source/utils/file-mention-handler.spec.ts"],"names":[],"mappings":""}
@@ -0,0 +1,147 @@
1
+ import test from 'ava';
2
+ import { handleFileMention, parseLineRangeFromMention, } from './file-mention-handler.js';
3
+ import { PlaceholderType } from '../types/hooks.js';
4
+ import { writeFile, mkdir, rm } from 'node:fs/promises';
5
+ import { join } from 'node:path';
6
+ import { tmpdir } from 'node:os';
7
+ console.log(`\nfile-mention-handler.spec.ts`);
8
+ // Test setup - create temp directory for test files
9
+ let testDir;
10
+ test.before(async () => {
11
+ // Create a unique temp directory for our tests
12
+ testDir = join(tmpdir(), `nanocoder-handler-test-${Date.now()}`);
13
+ await mkdir(testDir, { recursive: true });
14
+ // Create test files
15
+ await writeFile(join(testDir, 'test.txt'), 'Line 1\nLine 2\nLine 3\nLine 4\nLine 5', 'utf-8');
16
+ await writeFile(join(testDir, 'app.tsx'), 'import React from "react";\n\nexport function App() {\n return <div>Hello</div>;\n}', 'utf-8');
17
+ });
18
+ test.after.always(async () => {
19
+ // Clean up temp directory
20
+ try {
21
+ await rm(testDir, { recursive: true, force: true });
22
+ }
23
+ catch {
24
+ // Ignore cleanup errors
25
+ }
26
+ });
27
+ // Tests for handleFileMention()
28
+ test('creates placeholder for existing file', async (t) => {
29
+ const filePath = join(testDir, 'test.txt');
30
+ const result = await handleFileMention(filePath, 'Check @test.txt', {}, '@test.txt');
31
+ t.truthy(result);
32
+ // Handler uses the filePath passed in for the placeholder display
33
+ t.is(result.displayValue, `Check [@${filePath}]`);
34
+ t.is(Object.keys(result.placeholderContent).length, 1);
35
+ const placeholder = Object.values(result.placeholderContent)[0];
36
+ t.is(placeholder.type, PlaceholderType.FILE);
37
+ t.true(placeholder.content.includes('Line 1'));
38
+ });
39
+ test('creates placeholder with line range', async (t) => {
40
+ const filePath = join(testDir, 'test.txt');
41
+ const result = await handleFileMention(filePath, 'Check @test.txt:2-4', {}, '@test.txt:2-4', { start: 2, end: 4 });
42
+ t.truthy(result);
43
+ t.is(result.displayValue, `Check [@${filePath}:2-4]`);
44
+ const placeholder = Object.values(result.placeholderContent)[0];
45
+ t.true(placeholder.content.includes('Line 2'));
46
+ t.true(placeholder.content.includes('Line 4'));
47
+ t.false(placeholder.content.includes('Line 1'));
48
+ });
49
+ test('creates placeholder with single line', async (t) => {
50
+ const filePath = join(testDir, 'test.txt');
51
+ const result = await handleFileMention(filePath, 'Check @test.txt:3', {}, '@test.txt:3', { start: 3 });
52
+ t.truthy(result);
53
+ t.is(result.displayValue, `Check [@${filePath}:3]`);
54
+ const placeholder = Object.values(result.placeholderContent)[0];
55
+ t.true(placeholder.content.includes('Line 3'));
56
+ });
57
+ test('returns null for non-existent file', async (t) => {
58
+ const result = await handleFileMention(join(testDir, 'nonexistent.txt'), 'Check @nonexistent.txt', {}, '@nonexistent.txt');
59
+ t.is(result, null);
60
+ });
61
+ test('generates unique file IDs', async (t) => {
62
+ // First file mention
63
+ const result1 = await handleFileMention(join(testDir, 'test.txt'), 'Check @test.txt', {}, '@test.txt');
64
+ t.truthy(result1);
65
+ t.true(Object.keys(result1.placeholderContent).includes('file_1'));
66
+ // Second file mention (with existing placeholder)
67
+ const result2 = await handleFileMention(join(testDir, 'app.tsx'), 'Check @test.txt and @app.tsx', result1.placeholderContent, '@app.tsx');
68
+ t.truthy(result2);
69
+ t.true(Object.keys(result2.placeholderContent).includes('file_1'));
70
+ t.true(Object.keys(result2.placeholderContent).includes('file_2'));
71
+ });
72
+ test('preserves existing placeholders', async (t) => {
73
+ const existingPlaceholder = {
74
+ file_1: {
75
+ type: PlaceholderType.FILE,
76
+ displayText: '[@existing.txt]',
77
+ filePath: '/test/existing.txt',
78
+ content: 'existing content',
79
+ fileSize: 100,
80
+ },
81
+ };
82
+ const result = await handleFileMention(join(testDir, 'test.txt'), 'Check @test.txt', existingPlaceholder, '@test.txt');
83
+ t.truthy(result);
84
+ t.is(Object.keys(result.placeholderContent).length, 2);
85
+ t.truthy(result.placeholderContent.file_1);
86
+ t.truthy(result.placeholderContent.file_2);
87
+ });
88
+ test('replaces mention text in display value', async (t) => {
89
+ const filePath = join(testDir, 'app.tsx');
90
+ const result = await handleFileMention(filePath, 'Please review the code in @app.tsx carefully', {}, '@app.tsx');
91
+ t.truthy(result);
92
+ t.is(result.displayValue, `Please review the code in [@${filePath}] carefully`);
93
+ });
94
+ test('handles multiple mentions in display value', async (t) => {
95
+ const testPath = join(testDir, 'test.txt');
96
+ const appPath = join(testDir, 'app.tsx');
97
+ // First mention
98
+ const result1 = await handleFileMention(testPath, 'Compare @test.txt and @app.tsx', {}, '@test.txt');
99
+ t.truthy(result1);
100
+ // Second mention
101
+ const result2 = await handleFileMention(appPath, result1.displayValue, result1.placeholderContent, '@app.tsx');
102
+ t.truthy(result2);
103
+ t.is(result2.displayValue, `Compare [@${testPath}] and [@${appPath}]`);
104
+ });
105
+ test('stores absolute path in placeholder', async (t) => {
106
+ const relativePath = join(testDir, 'test.txt');
107
+ const result = await handleFileMention(relativePath, 'Check @test.txt', {}, '@test.txt');
108
+ t.truthy(result);
109
+ const placeholder = Object.values(result.placeholderContent)[0];
110
+ // Absolute path should be stored
111
+ t.is(placeholder.type, PlaceholderType.FILE);
112
+ if (placeholder.type === PlaceholderType.FILE) {
113
+ t.true(placeholder.filePath.startsWith('/'));
114
+ t.true(placeholder.filePath.includes('test.txt'));
115
+ }
116
+ });
117
+ test('includes file content in placeholder', async (t) => {
118
+ const result = await handleFileMention(join(testDir, 'app.tsx'), 'Check @app.tsx', {}, '@app.tsx');
119
+ t.truthy(result);
120
+ const placeholder = Object.values(result.placeholderContent)[0];
121
+ t.truthy(placeholder.content);
122
+ t.true(placeholder.content.length > 0);
123
+ t.true(placeholder.content.includes('React'));
124
+ });
125
+ // Tests for parseLineRangeFromMention()
126
+ test('parseLineRangeFromMention parses range', t => {
127
+ const result = parseLineRangeFromMention('@file.ts:10-20');
128
+ t.deepEqual(result, { start: 10, end: 20 });
129
+ });
130
+ test('parseLineRangeFromMention parses single line', t => {
131
+ const result = parseLineRangeFromMention('@file.ts:10');
132
+ t.deepEqual(result, { start: 10, end: undefined });
133
+ });
134
+ test('parseLineRangeFromMention returns undefined for no range', t => {
135
+ const result = parseLineRangeFromMention('@file.ts');
136
+ t.is(result, undefined);
137
+ });
138
+ test('parseLineRangeFromMention handles invalid range', t => {
139
+ const result = parseLineRangeFromMention('@file.ts:abc');
140
+ t.is(result, undefined);
141
+ });
142
+ test('parseLineRangeFromMention handles path with colons', t => {
143
+ // Should only parse the last colon as line range
144
+ const result = parseLineRangeFromMention('@src/path:to:file.ts:10-20');
145
+ t.deepEqual(result, { start: 10, end: 20 });
146
+ });
147
+ //# sourceMappingURL=file-mention-handler.spec.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mention-handler.spec.js","sourceRoot":"","sources":["../../source/utils/file-mention-handler.spec.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,40 @@
1
+ /**
2
+ * Represents a parsed file mention from user input
3
+ * Supports:
4
+ * - @filename.ts
5
+ * - @src/components/Button.tsx
6
+ * - @file.ts:10-20 (line ranges)
7
+ * - @file.ts:10 (single line)
8
+ */
9
+ interface FileMention {
10
+ rawText: string;
11
+ filePath: string;
12
+ lineRange?: {
13
+ start: number;
14
+ end?: number;
15
+ };
16
+ startIndex: number;
17
+ endIndex: number;
18
+ }
19
+ /**
20
+ * Parse all @mentions from user input
21
+ */
22
+ export declare function parseFileMentions(input: string): FileMention[];
23
+ /**
24
+ * Validate file path to prevent directory traversal attacks
25
+ * and ensure it's within the project directory
26
+ */
27
+ export declare function isValidFilePath(filePath: string): boolean;
28
+ /**
29
+ * Resolve a relative file path to an absolute path within the project
30
+ */
31
+ export declare function resolveFilePath(filePath: string, cwd: string): string;
32
+ /**
33
+ * Parse line range from a string like "10-20" or "10"
34
+ */
35
+ export declare function parseLineRange(rangeStr: string): {
36
+ start: number;
37
+ end?: number;
38
+ } | null;
39
+ export {};
40
+ //# sourceMappingURL=file-mention-parser.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mention-parser.d.ts","sourceRoot":"","sources":["../../source/utils/file-mention-parser.ts"],"names":[],"mappings":"AAEA;;;;;;;GAOG;AACH,UAAU,WAAW;IACpB,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,CAAC,EAAE;QACX,KAAK,EAAE,MAAM,CAAC;QACd,GAAG,CAAC,EAAE,MAAM,CAAC;KACb,CAAC;IACF,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CACjB;AAYD;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,KAAK,EAAE,MAAM,GAAG,WAAW,EAAE,CAwC9D;AAED;;;GAGG;AACH,wBAAgB,eAAe,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAgCzD;AAED;;GAEG;AACH,wBAAgB,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,MAAM,CAkBrE;AAED;;GAEG;AACH,wBAAgB,cAAc,CAC7B,QAAQ,EAAE,MAAM,GACd;IAAC,KAAK,EAAE,MAAM,CAAC;IAAC,GAAG,CAAC,EAAE,MAAM,CAAA;CAAC,GAAG,IAAI,CA2BtC"}
@@ -0,0 +1,122 @@
1
+ import path from 'node:path';
2
+ /**
3
+ * Regex pattern to match @file mentions
4
+ * Matches:
5
+ * - @file.ts
6
+ * - @src/path/to/file.tsx
7
+ * - @file.ts:10
8
+ * - @file.ts:10-20
9
+ */
10
+ const FILE_MENTION_REGEX = /@([^\s:]+)(?::(\d+)(?:-(\d+))?)?/g;
11
+ /**
12
+ * Parse all @mentions from user input
13
+ */
14
+ export function parseFileMentions(input) {
15
+ const mentions = [];
16
+ let match;
17
+ // Reset regex state
18
+ FILE_MENTION_REGEX.lastIndex = 0;
19
+ while ((match = FILE_MENTION_REGEX.exec(input)) !== null) {
20
+ const rawText = match[0]; // Full match: "@src/app.tsx:10-20"
21
+ const filePath = match[1]; // Captured group 1: "src/app.tsx"
22
+ const lineStart = match[2]; // Captured group 2: "10"
23
+ const lineEnd = match[3]; // Captured group 3: "20"
24
+ // Skip if the file path is empty or invalid
25
+ if (!filePath || !isValidFilePath(filePath)) {
26
+ continue;
27
+ }
28
+ const mention = {
29
+ rawText,
30
+ filePath,
31
+ startIndex: match.index,
32
+ endIndex: match.index + rawText.length,
33
+ };
34
+ // Parse line range if present
35
+ if (lineStart) {
36
+ const start = parseInt(lineStart, 10);
37
+ const end = lineEnd ? parseInt(lineEnd, 10) : undefined;
38
+ // Validate line numbers
39
+ if (start > 0 && (!end || end >= start)) {
40
+ mention.lineRange = { start, end };
41
+ }
42
+ }
43
+ mentions.push(mention);
44
+ }
45
+ return mentions;
46
+ }
47
+ /**
48
+ * Validate file path to prevent directory traversal attacks
49
+ * and ensure it's within the project directory
50
+ */
51
+ export function isValidFilePath(filePath) {
52
+ // Reject empty paths
53
+ if (!filePath || filePath.trim().length === 0) {
54
+ return false;
55
+ }
56
+ // Reject paths that try to escape parent directories
57
+ if (filePath.includes('..')) {
58
+ return false;
59
+ }
60
+ // Reject absolute paths (outside project)
61
+ if (path.isAbsolute(filePath)) {
62
+ return false;
63
+ }
64
+ // Reject Windows absolute paths (C:\, D:\, etc.) even on Unix systems
65
+ if (/^[A-Za-z]:[/\\]/.test(filePath)) {
66
+ return false;
67
+ }
68
+ // Reject paths with null bytes (security)
69
+ if (filePath.includes('\0')) {
70
+ return false;
71
+ }
72
+ // Reject paths that start with special characters that could be problematic
73
+ if (filePath.startsWith('/') || filePath.startsWith('\\')) {
74
+ return false;
75
+ }
76
+ return true;
77
+ }
78
+ /**
79
+ * Resolve a relative file path to an absolute path within the project
80
+ */
81
+ export function resolveFilePath(filePath, cwd) {
82
+ // Validate first
83
+ if (!isValidFilePath(filePath)) {
84
+ throw new Error(`Invalid file path: ${filePath}`);
85
+ }
86
+ // Resolve to absolute path
87
+ const absolutePath = path.resolve(cwd, filePath);
88
+ // Ensure the resolved path is still within the project directory
89
+ const normalizedCwd = path.resolve(cwd);
90
+ if (!absolutePath.startsWith(normalizedCwd)) {
91
+ throw new Error(`File path escapes project directory: ${filePath} -> ${absolutePath}`);
92
+ }
93
+ return absolutePath;
94
+ }
95
+ /**
96
+ * Parse line range from a string like "10-20" or "10"
97
+ */
98
+ export function parseLineRange(rangeStr) {
99
+ if (!rangeStr) {
100
+ return null;
101
+ }
102
+ const parts = rangeStr.split('-');
103
+ if (parts.length === 1) {
104
+ // Single line: "10"
105
+ const line = parseInt(parts[0], 10);
106
+ if (isNaN(line) || line <= 0) {
107
+ return null;
108
+ }
109
+ return { start: line, end: undefined };
110
+ }
111
+ else if (parts.length === 2) {
112
+ // Range: "10-20"
113
+ const start = parseInt(parts[0], 10);
114
+ const end = parseInt(parts[1], 10);
115
+ if (isNaN(start) || isNaN(end) || start <= 0 || end < start) {
116
+ return null;
117
+ }
118
+ return { start, end };
119
+ }
120
+ return null;
121
+ }
122
+ //# sourceMappingURL=file-mention-parser.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mention-parser.js","sourceRoot":"","sources":["../../source/utils/file-mention-parser.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=file-mention-parser.spec.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mention-parser.spec.d.ts","sourceRoot":"","sources":["../../source/utils/file-mention-parser.spec.ts"],"names":[],"mappings":""}