@j0hanz/filesystem-context-mcp 1.1.0 → 1.2.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 (305) hide show
  1. package/README.md +106 -29
  2. package/dist/__tests__/lib/errors.test.js +3 -23
  3. package/dist/__tests__/lib/errors.test.js.map +1 -1
  4. package/dist/__tests__/lib/file-operations.test.js +34 -0
  5. package/dist/__tests__/lib/file-operations.test.js.map +1 -1
  6. package/dist/__tests__/lib/path-validation.test.js +8 -0
  7. package/dist/__tests__/lib/path-validation.test.js.map +1 -1
  8. package/dist/__tests__/schemas/validators.test.js +101 -122
  9. package/dist/__tests__/schemas/validators.test.js.map +1 -1
  10. package/dist/__tests__/security/filesystem-boundary.test.js +1 -1
  11. package/dist/__tests__/security/filesystem-boundary.test.js.map +1 -1
  12. package/dist/config/types.d.ts +1 -85
  13. package/dist/config/types.d.ts.map +1 -1
  14. package/dist/config/types.js +0 -2
  15. package/dist/config/types.js.map +1 -1
  16. package/dist/index.js +3 -3
  17. package/dist/index.js.map +1 -1
  18. package/dist/instructions.md +1 -2
  19. package/dist/lib/constants.d.ts +3 -1
  20. package/dist/lib/constants.d.ts.map +1 -1
  21. package/dist/lib/constants.js +73 -259
  22. package/dist/lib/constants.js.map +1 -1
  23. package/dist/lib/errors.d.ts +25 -3
  24. package/dist/lib/errors.d.ts.map +1 -1
  25. package/dist/lib/errors.js +4 -94
  26. package/dist/lib/errors.js.map +1 -1
  27. package/dist/lib/file-operations/analyze-directory.d.ts +8 -0
  28. package/dist/lib/file-operations/analyze-directory.d.ts.map +1 -0
  29. package/dist/lib/file-operations/analyze-directory.js +117 -0
  30. package/dist/lib/file-operations/analyze-directory.js.map +1 -0
  31. package/dist/lib/file-operations/directory-helpers.d.ts +35 -0
  32. package/dist/lib/file-operations/directory-helpers.d.ts.map +1 -0
  33. package/dist/lib/file-operations/directory-helpers.js +136 -0
  34. package/dist/lib/file-operations/directory-helpers.js.map +1 -0
  35. package/dist/lib/file-operations/directory-items.d.ts +20 -0
  36. package/dist/lib/file-operations/directory-items.d.ts.map +1 -0
  37. package/dist/lib/file-operations/directory-items.js +85 -0
  38. package/dist/lib/file-operations/directory-items.js.map +1 -0
  39. package/dist/lib/file-operations/directory-iteration.d.ts +17 -0
  40. package/dist/lib/file-operations/directory-iteration.d.ts.map +1 -0
  41. package/dist/lib/file-operations/directory-iteration.js +55 -0
  42. package/dist/lib/file-operations/directory-iteration.js.map +1 -0
  43. package/dist/lib/file-operations/directory-tree.d.ts +9 -0
  44. package/dist/lib/file-operations/directory-tree.d.ts.map +1 -0
  45. package/dist/lib/file-operations/directory-tree.js +175 -0
  46. package/dist/lib/file-operations/directory-tree.js.map +1 -0
  47. package/dist/lib/file-operations/file-info.d.ts +3 -0
  48. package/dist/lib/file-operations/file-info.d.ts.map +1 -0
  49. package/dist/lib/file-operations/file-info.js +56 -0
  50. package/dist/lib/file-operations/file-info.js.map +1 -0
  51. package/dist/lib/file-operations/list-directory.d.ts +10 -0
  52. package/dist/lib/file-operations/list-directory.d.ts.map +1 -0
  53. package/dist/lib/file-operations/list-directory.js +189 -0
  54. package/dist/lib/file-operations/list-directory.js.map +1 -0
  55. package/dist/lib/file-operations/pattern-complexity.d.ts +16 -0
  56. package/dist/lib/file-operations/pattern-complexity.d.ts.map +1 -0
  57. package/dist/lib/file-operations/pattern-complexity.js +86 -0
  58. package/dist/lib/file-operations/pattern-complexity.js.map +1 -0
  59. package/dist/lib/file-operations/pattern-validator.d.ts +15 -0
  60. package/dist/lib/file-operations/pattern-validator.d.ts.map +1 -0
  61. package/dist/lib/file-operations/pattern-validator.js +69 -0
  62. package/dist/lib/file-operations/pattern-validator.js.map +1 -0
  63. package/dist/lib/file-operations/read-media-file.d.ts +5 -0
  64. package/dist/lib/file-operations/read-media-file.d.ts.map +1 -0
  65. package/dist/lib/file-operations/read-media-file.js +31 -0
  66. package/dist/lib/file-operations/read-media-file.js.map +1 -0
  67. package/dist/lib/file-operations/read-multiple-files.d.ts +16 -0
  68. package/dist/lib/file-operations/read-multiple-files.d.ts.map +1 -0
  69. package/dist/lib/file-operations/read-multiple-files.js +100 -0
  70. package/dist/lib/file-operations/read-multiple-files.js.map +1 -0
  71. package/dist/lib/file-operations/search-content.d.ts +18 -0
  72. package/dist/lib/file-operations/search-content.d.ts.map +1 -0
  73. package/dist/lib/file-operations/search-content.js +436 -0
  74. package/dist/lib/file-operations/search-content.js.map +1 -0
  75. package/dist/lib/file-operations/search-files.d.ts +11 -0
  76. package/dist/lib/file-operations/search-files.d.ts.map +1 -0
  77. package/dist/lib/file-operations/search-files.js +143 -0
  78. package/dist/lib/file-operations/search-files.js.map +1 -0
  79. package/dist/lib/file-operations/sorting.d.ts +12 -0
  80. package/dist/lib/file-operations/sorting.d.ts.map +1 -0
  81. package/dist/lib/file-operations/sorting.js +24 -0
  82. package/dist/lib/file-operations/sorting.js.map +1 -0
  83. package/dist/lib/file-operations.d.ts +9 -57
  84. package/dist/lib/file-operations.d.ts.map +1 -1
  85. package/dist/lib/file-operations.js +9 -773
  86. package/dist/lib/file-operations.js.map +1 -1
  87. package/dist/lib/fs-helpers/binary-detect.d.ts +3 -0
  88. package/dist/lib/fs-helpers/binary-detect.d.ts.map +1 -0
  89. package/dist/lib/fs-helpers/binary-detect.js +54 -0
  90. package/dist/lib/fs-helpers/binary-detect.js.map +1 -0
  91. package/dist/lib/fs-helpers/concurrency.d.ts +11 -0
  92. package/dist/lib/fs-helpers/concurrency.d.ts.map +1 -0
  93. package/dist/lib/fs-helpers/concurrency.js +95 -0
  94. package/dist/lib/fs-helpers/concurrency.js.map +1 -0
  95. package/dist/lib/fs-helpers/fs-utils.d.ts +5 -0
  96. package/dist/lib/fs-helpers/fs-utils.d.ts.map +1 -0
  97. package/dist/lib/fs-helpers/fs-utils.js +13 -0
  98. package/dist/lib/fs-helpers/fs-utils.js.map +1 -0
  99. package/dist/lib/fs-helpers/readers/head-file.d.ts +2 -0
  100. package/dist/lib/fs-helpers/readers/head-file.d.ts.map +1 -0
  101. package/dist/lib/fs-helpers/readers/head-file.js +73 -0
  102. package/dist/lib/fs-helpers/readers/head-file.js.map +1 -0
  103. package/dist/lib/fs-helpers/readers/line-range.d.ts +7 -0
  104. package/dist/lib/fs-helpers/readers/line-range.d.ts.map +1 -0
  105. package/dist/lib/fs-helpers/readers/line-range.js +46 -0
  106. package/dist/lib/fs-helpers/readers/line-range.js.map +1 -0
  107. package/dist/lib/fs-helpers/readers/read-file.d.ts +16 -0
  108. package/dist/lib/fs-helpers/readers/read-file.d.ts.map +1 -0
  109. package/dist/lib/fs-helpers/readers/read-file.js +87 -0
  110. package/dist/lib/fs-helpers/readers/read-file.js.map +1 -0
  111. package/dist/lib/fs-helpers/readers/stream-lines.d.ts +7 -0
  112. package/dist/lib/fs-helpers/readers/stream-lines.d.ts.map +1 -0
  113. package/dist/lib/fs-helpers/readers/stream-lines.js +61 -0
  114. package/dist/lib/fs-helpers/readers/stream-lines.js.map +1 -0
  115. package/dist/lib/fs-helpers/readers/tail-file.d.ts +2 -0
  116. package/dist/lib/fs-helpers/readers/tail-file.d.ts.map +1 -0
  117. package/dist/lib/fs-helpers/readers/tail-file.js +98 -0
  118. package/dist/lib/fs-helpers/readers/tail-file.js.map +1 -0
  119. package/dist/lib/fs-helpers/readers/utf8.d.ts +3 -0
  120. package/dist/lib/fs-helpers/readers/utf8.d.ts.map +1 -0
  121. package/dist/lib/fs-helpers/readers/utf8.js +22 -0
  122. package/dist/lib/fs-helpers/readers/utf8.js.map +1 -0
  123. package/dist/lib/fs-helpers/readers.d.ts +4 -0
  124. package/dist/lib/fs-helpers/readers.d.ts.map +1 -0
  125. package/dist/lib/fs-helpers/readers.js +4 -0
  126. package/dist/lib/fs-helpers/readers.js.map +1 -0
  127. package/dist/lib/fs-helpers.d.ts +4 -25
  128. package/dist/lib/fs-helpers.d.ts.map +1 -1
  129. package/dist/lib/fs-helpers.js +4 -327
  130. package/dist/lib/fs-helpers.js.map +1 -1
  131. package/dist/lib/path-validation/allowed-directories.d.ts +9 -0
  132. package/dist/lib/path-validation/allowed-directories.d.ts.map +1 -0
  133. package/dist/lib/path-validation/allowed-directories.js +94 -0
  134. package/dist/lib/path-validation/allowed-directories.js.map +1 -0
  135. package/dist/lib/path-validation/errors.d.ts +5 -0
  136. package/dist/lib/path-validation/errors.d.ts.map +1 -0
  137. package/dist/lib/path-validation/errors.js +33 -0
  138. package/dist/lib/path-validation/errors.js.map +1 -0
  139. package/dist/lib/path-validation/roots.d.ts +3 -0
  140. package/dist/lib/path-validation/roots.d.ts.map +1 -0
  141. package/dist/lib/path-validation/roots.js +49 -0
  142. package/dist/lib/path-validation/roots.js.map +1 -0
  143. package/dist/lib/path-validation/validators.d.ts +9 -0
  144. package/dist/lib/path-validation/validators.d.ts.map +1 -0
  145. package/dist/lib/path-validation/validators.js +70 -0
  146. package/dist/lib/path-validation/validators.js.map +1 -0
  147. package/dist/lib/path-validation.d.ts +3 -7
  148. package/dist/lib/path-validation.d.ts.map +1 -1
  149. package/dist/lib/path-validation.js +3 -141
  150. package/dist/lib/path-validation.js.map +1 -1
  151. package/dist/lib/performance/monitor.d.ts +25 -0
  152. package/dist/lib/performance/monitor.d.ts.map +1 -0
  153. package/dist/lib/performance/monitor.js +84 -0
  154. package/dist/lib/performance/monitor.js.map +1 -0
  155. package/dist/schemas/input-helpers.d.ts +27 -0
  156. package/dist/schemas/input-helpers.d.ts.map +1 -0
  157. package/dist/schemas/input-helpers.js +176 -0
  158. package/dist/schemas/input-helpers.js.map +1 -0
  159. package/dist/schemas/inputs.d.ts +14 -7
  160. package/dist/schemas/inputs.d.ts.map +1 -1
  161. package/dist/schemas/inputs.js +56 -228
  162. package/dist/schemas/inputs.js.map +1 -1
  163. package/dist/schemas/output-helpers.d.ts +24 -0
  164. package/dist/schemas/output-helpers.d.ts.map +1 -0
  165. package/dist/schemas/output-helpers.js +13 -0
  166. package/dist/schemas/output-helpers.js.map +1 -0
  167. package/dist/schemas/outputs.d.ts +476 -42
  168. package/dist/schemas/outputs.d.ts.map +1 -1
  169. package/dist/schemas/outputs.js +26 -41
  170. package/dist/schemas/outputs.js.map +1 -1
  171. package/dist/server.d.ts +9 -1
  172. package/dist/server.d.ts.map +1 -1
  173. package/dist/server.js +29 -45
  174. package/dist/server.js.map +1 -1
  175. package/dist/tools/analyze-directory.d.ts.map +1 -1
  176. package/dist/tools/analyze-directory.js +115 -53
  177. package/dist/tools/analyze-directory.js.map +1 -1
  178. package/dist/tools/directory-tree.d.ts.map +1 -1
  179. package/dist/tools/directory-tree.js +86 -49
  180. package/dist/tools/directory-tree.js.map +1 -1
  181. package/dist/tools/get-file-info.d.ts.map +1 -1
  182. package/dist/tools/get-file-info.js +71 -37
  183. package/dist/tools/get-file-info.js.map +1 -1
  184. package/dist/tools/list-allowed-dirs.d.ts.map +1 -1
  185. package/dist/tools/list-allowed-dirs.js +48 -35
  186. package/dist/tools/list-allowed-dirs.js.map +1 -1
  187. package/dist/tools/list-directory.d.ts.map +1 -1
  188. package/dist/tools/list-directory.js +129 -58
  189. package/dist/tools/list-directory.js.map +1 -1
  190. package/dist/tools/read-file.d.ts.map +1 -1
  191. package/dist/tools/read-file.js +70 -56
  192. package/dist/tools/read-file.js.map +1 -1
  193. package/dist/tools/read-media-file.d.ts.map +1 -1
  194. package/dist/tools/read-media-file.js +39 -41
  195. package/dist/tools/read-media-file.js.map +1 -1
  196. package/dist/tools/read-multiple-files.d.ts.map +1 -1
  197. package/dist/tools/read-multiple-files.js +58 -50
  198. package/dist/tools/read-multiple-files.js.map +1 -1
  199. package/dist/tools/search-content.d.ts.map +1 -1
  200. package/dist/tools/search-content.js +148 -89
  201. package/dist/tools/search-content.js.map +1 -1
  202. package/dist/tools/search-files.d.ts.map +1 -1
  203. package/dist/tools/search-files.js +123 -50
  204. package/dist/tools/search-files.js.map +1 -1
  205. package/dist/tools/tool-response.d.ts +9 -0
  206. package/dist/tools/tool-response.d.ts.map +1 -0
  207. package/dist/tools/tool-response.js +7 -0
  208. package/dist/tools/tool-response.js.map +1 -0
  209. package/package.json +2 -1
  210. package/dist/__tests__/errors.test.d.ts +0 -2
  211. package/dist/__tests__/errors.test.d.ts.map +0 -1
  212. package/dist/__tests__/errors.test.js +0 -88
  213. package/dist/__tests__/errors.test.js.map +0 -1
  214. package/dist/__tests__/file-operations.test.d.ts +0 -2
  215. package/dist/__tests__/file-operations.test.d.ts.map +0 -1
  216. package/dist/__tests__/file-operations.test.js +0 -230
  217. package/dist/__tests__/file-operations.test.js.map +0 -1
  218. package/dist/__tests__/lib/formatters.test.d.ts +0 -2
  219. package/dist/__tests__/lib/formatters.test.d.ts.map +0 -1
  220. package/dist/__tests__/lib/formatters.test.js +0 -248
  221. package/dist/__tests__/lib/formatters.test.js.map +0 -1
  222. package/dist/__tests__/lib/image-parsing.test.d.ts +0 -2
  223. package/dist/__tests__/lib/image-parsing.test.d.ts.map +0 -1
  224. package/dist/__tests__/lib/image-parsing.test.js +0 -262
  225. package/dist/__tests__/lib/image-parsing.test.js.map +0 -1
  226. package/dist/__tests__/path-validation.test.d.ts +0 -2
  227. package/dist/__tests__/path-validation.test.d.ts.map +0 -1
  228. package/dist/__tests__/path-validation.test.js +0 -92
  229. package/dist/__tests__/path-validation.test.js.map +0 -1
  230. package/dist/lib/directory-helpers.d.ts +0 -4
  231. package/dist/lib/directory-helpers.d.ts.map +0 -1
  232. package/dist/lib/directory-helpers.js +0 -36
  233. package/dist/lib/directory-helpers.js.map +0 -1
  234. package/dist/lib/formatters.d.ts +0 -10
  235. package/dist/lib/formatters.d.ts.map +0 -1
  236. package/dist/lib/formatters.js +0 -202
  237. package/dist/lib/formatters.js.map +0 -1
  238. package/dist/lib/image-parsing.d.ts +0 -4
  239. package/dist/lib/image-parsing.d.ts.map +0 -1
  240. package/dist/lib/image-parsing.js +0 -124
  241. package/dist/lib/image-parsing.js.map +0 -1
  242. package/dist/lib/mcp-logger.d.ts +0 -11
  243. package/dist/lib/mcp-logger.d.ts.map +0 -1
  244. package/dist/lib/mcp-logger.js +0 -49
  245. package/dist/lib/mcp-logger.js.map +0 -1
  246. package/dist/lib/roots-utils.d.ts +0 -7
  247. package/dist/lib/roots-utils.d.ts.map +0 -1
  248. package/dist/lib/roots-utils.js +0 -39
  249. package/dist/lib/roots-utils.js.map +0 -1
  250. package/dist/lib/search-helpers.d.ts +0 -16
  251. package/dist/lib/search-helpers.d.ts.map +0 -1
  252. package/dist/lib/search-helpers.js +0 -169
  253. package/dist/lib/search-helpers.js.map +0 -1
  254. package/dist/lib/sorting.d.ts +0 -12
  255. package/dist/lib/sorting.d.ts.map +0 -1
  256. package/dist/lib/sorting.js +0 -41
  257. package/dist/lib/sorting.js.map +0 -1
  258. package/dist/lib/types.d.ts +0 -6
  259. package/dist/lib/types.d.ts.map +0 -1
  260. package/dist/lib/types.js +0 -2
  261. package/dist/lib/types.js.map +0 -1
  262. package/dist/prompts/analyze-codebase.d.ts +0 -3
  263. package/dist/prompts/analyze-codebase.d.ts.map +0 -1
  264. package/dist/prompts/analyze-codebase.js +0 -144
  265. package/dist/prompts/analyze-codebase.js.map +0 -1
  266. package/dist/prompts/filesystem-query.d.ts +0 -3
  267. package/dist/prompts/filesystem-query.d.ts.map +0 -1
  268. package/dist/prompts/filesystem-query.js +0 -168
  269. package/dist/prompts/filesystem-query.js.map +0 -1
  270. package/dist/prompts/find-duplicates.d.ts +0 -3
  271. package/dist/prompts/find-duplicates.d.ts.map +0 -1
  272. package/dist/prompts/find-duplicates.js +0 -77
  273. package/dist/prompts/find-duplicates.js.map +0 -1
  274. package/dist/prompts/index.d.ts +0 -3
  275. package/dist/prompts/index.d.ts.map +0 -1
  276. package/dist/prompts/index.js +0 -13
  277. package/dist/prompts/index.js.map +0 -1
  278. package/dist/prompts/project-overview.d.ts +0 -3
  279. package/dist/prompts/project-overview.d.ts.map +0 -1
  280. package/dist/prompts/project-overview.js +0 -122
  281. package/dist/prompts/project-overview.js.map +0 -1
  282. package/dist/prompts/search-and-replace.d.ts +0 -3
  283. package/dist/prompts/search-and-replace.d.ts.map +0 -1
  284. package/dist/prompts/search-and-replace.js +0 -130
  285. package/dist/prompts/search-and-replace.js.map +0 -1
  286. package/dist/prompts/shared.d.ts +0 -11
  287. package/dist/prompts/shared.d.ts.map +0 -1
  288. package/dist/prompts/shared.js +0 -32
  289. package/dist/prompts/shared.js.map +0 -1
  290. package/dist/resources/index.d.ts +0 -3
  291. package/dist/resources/index.d.ts.map +0 -1
  292. package/dist/resources/index.js +0 -54
  293. package/dist/resources/index.js.map +0 -1
  294. package/dist/schemas/validators.d.ts +0 -12
  295. package/dist/schemas/validators.d.ts.map +0 -1
  296. package/dist/schemas/validators.js +0 -35
  297. package/dist/schemas/validators.js.map +0 -1
  298. package/dist/utils/index.d.ts +0 -2
  299. package/dist/utils/index.d.ts.map +0 -1
  300. package/dist/utils/index.js +0 -2
  301. package/dist/utils/index.js.map +0 -1
  302. package/dist/utils/response-helpers.d.ts +0 -22
  303. package/dist/utils/response-helpers.d.ts.map +0 -1
  304. package/dist/utils/response-helpers.js +0 -24
  305. package/dist/utils/response-helpers.js.map +0 -1
@@ -0,0 +1,175 @@
1
+ import * as fs from 'node:fs/promises';
2
+ import * as path from 'node:path';
3
+ import { DEFAULT_TREE_DEPTH, DEFAULT_TREE_MAX_FILES, DIR_TRAVERSAL_CONCURRENCY, } from '../constants.js';
4
+ import { ErrorCode, McpError } from '../errors.js';
5
+ import { runWorkQueue } from '../fs-helpers.js';
6
+ import { validateExistingPath, validateExistingPathDetailed, } from '../path-validation.js';
7
+ import { classifyAccessError, createExcludeMatcher, forEachDirectoryEntry, } from './directory-helpers.js';
8
+ function initTreeState(basePath) {
9
+ return {
10
+ totalFiles: 0,
11
+ totalDirectories: 0,
12
+ maxDepthReached: 0,
13
+ skippedInaccessible: 0,
14
+ symlinksNotFollowed: 0,
15
+ truncated: false,
16
+ collectedEntries: [],
17
+ directoriesFound: new Set([basePath]),
18
+ };
19
+ }
20
+ function hitMaxFiles(state, maxFiles) {
21
+ if (state.totalFiles < maxFiles)
22
+ return false;
23
+ state.truncated = true;
24
+ return true;
25
+ }
26
+ function markTruncated(state) {
27
+ state.truncated = true;
28
+ }
29
+ function addFileEntry(state, params, name, size) {
30
+ state.totalFiles++;
31
+ state.collectedEntries.push({
32
+ parentPath: params.currentPath,
33
+ name,
34
+ type: 'file',
35
+ size,
36
+ depth: params.depth,
37
+ });
38
+ }
39
+ function addDirectoryEntry(state, params, name, resolvedPath, enqueue, maxDepth) {
40
+ state.totalDirectories++;
41
+ state.directoriesFound.add(resolvedPath);
42
+ state.collectedEntries.push({
43
+ parentPath: params.currentPath,
44
+ name,
45
+ type: 'directory',
46
+ depth: params.depth,
47
+ });
48
+ if (params.depth + 1 <= maxDepth) {
49
+ enqueue({ currentPath: resolvedPath, depth: params.depth + 1 });
50
+ }
51
+ else {
52
+ markTruncated(state);
53
+ }
54
+ }
55
+ async function handleTreeNode(params, enqueue, state, options) {
56
+ if (hitMaxFiles(state, options.maxFiles))
57
+ return;
58
+ if (params.depth > options.maxDepth) {
59
+ markTruncated(state);
60
+ return;
61
+ }
62
+ state.maxDepthReached = Math.max(state.maxDepthReached, params.depth);
63
+ await forEachDirectoryEntry(params.currentPath, options.basePath, {
64
+ includeHidden: options.includeHidden,
65
+ shouldExclude: options.shouldExclude,
66
+ onInaccessible: () => {
67
+ state.skippedInaccessible++;
68
+ },
69
+ shouldStop: () => hitMaxFiles(state, options.maxFiles),
70
+ }, async ({ item, name, fullPath }) => {
71
+ if (item.isSymbolicLink()) {
72
+ state.symlinksNotFollowed++;
73
+ return;
74
+ }
75
+ try {
76
+ const { resolvedPath, isSymlink } = await validateExistingPathDetailed(fullPath);
77
+ if (isSymlink) {
78
+ state.symlinksNotFollowed++;
79
+ return;
80
+ }
81
+ const stats = await fs.stat(resolvedPath);
82
+ if (stats.isFile()) {
83
+ addFileEntry(state, params, name, options.includeSize ? stats.size : undefined);
84
+ return;
85
+ }
86
+ if (stats.isDirectory()) {
87
+ addDirectoryEntry(state, params, name, resolvedPath, enqueue, options.maxDepth);
88
+ }
89
+ }
90
+ catch (error) {
91
+ if (classifyAccessError(error) === 'symlink') {
92
+ state.symlinksNotFollowed++;
93
+ }
94
+ else {
95
+ state.skippedInaccessible++;
96
+ }
97
+ }
98
+ });
99
+ }
100
+ function buildChildrenByParent(directoriesFound, collectedEntries) {
101
+ const childrenByParent = new Map();
102
+ for (const dirPath of directoriesFound) {
103
+ childrenByParent.set(dirPath, []);
104
+ }
105
+ for (const entry of collectedEntries) {
106
+ const treeEntry = {
107
+ name: entry.name,
108
+ type: entry.type,
109
+ };
110
+ if (entry.type === 'file' && entry.size !== undefined) {
111
+ treeEntry.size = entry.size;
112
+ }
113
+ if (entry.type === 'directory') {
114
+ const fullPath = path.join(entry.parentPath, entry.name);
115
+ treeEntry.children = childrenByParent.get(fullPath) ?? [];
116
+ }
117
+ const siblings = childrenByParent.get(entry.parentPath);
118
+ if (siblings) {
119
+ siblings.push(treeEntry);
120
+ }
121
+ }
122
+ return childrenByParent;
123
+ }
124
+ function sortTreeChildren(childrenByParent) {
125
+ for (const children of childrenByParent.values()) {
126
+ children.sort((a, b) => {
127
+ if (a.type !== b.type) {
128
+ return a.type === 'directory' ? -1 : 1;
129
+ }
130
+ return a.name.localeCompare(b.name);
131
+ });
132
+ }
133
+ }
134
+ function buildTree(rootPath, childrenByParent) {
135
+ const rootName = path.basename(rootPath);
136
+ return {
137
+ name: rootName || rootPath,
138
+ type: 'directory',
139
+ children: childrenByParent.get(rootPath) ?? [],
140
+ };
141
+ }
142
+ export async function getDirectoryTree(dirPath, options = {}) {
143
+ const { maxDepth = DEFAULT_TREE_DEPTH, excludePatterns = [], includeHidden = false, includeSize = false, maxFiles, } = options;
144
+ const basePath = await validateExistingPath(dirPath);
145
+ const rootStats = await fs.stat(basePath);
146
+ if (!rootStats.isDirectory()) {
147
+ throw new McpError(ErrorCode.E_NOT_DIRECTORY, `Not a directory: ${dirPath}`, dirPath);
148
+ }
149
+ const state = initTreeState(basePath);
150
+ const shouldExclude = createExcludeMatcher(excludePatterns);
151
+ const effectiveMaxFiles = maxFiles ?? DEFAULT_TREE_MAX_FILES;
152
+ await runWorkQueue([{ currentPath: basePath, depth: 0 }], async (params, enqueue) => handleTreeNode(params, enqueue, state, {
153
+ basePath,
154
+ maxDepth,
155
+ includeHidden,
156
+ includeSize,
157
+ maxFiles: effectiveMaxFiles,
158
+ shouldExclude,
159
+ }), DIR_TRAVERSAL_CONCURRENCY);
160
+ const childrenByParent = buildChildrenByParent(state.directoriesFound, state.collectedEntries);
161
+ sortTreeChildren(childrenByParent);
162
+ const tree = buildTree(basePath, childrenByParent);
163
+ return {
164
+ tree,
165
+ summary: {
166
+ totalFiles: state.totalFiles,
167
+ totalDirectories: state.totalDirectories,
168
+ maxDepthReached: state.maxDepthReached,
169
+ truncated: state.truncated,
170
+ skippedInaccessible: state.skippedInaccessible,
171
+ symlinksNotFollowed: state.symlinksNotFollowed,
172
+ },
173
+ };
174
+ }
175
+ //# sourceMappingURL=directory-tree.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"directory-tree.js","sourceRoot":"","sources":["../../../src/lib/file-operations/directory-tree.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { FileInfo } from '../../config/types.js';
2
+ export declare function getFileInfo(filePath: string): Promise<FileInfo>;
3
+ //# sourceMappingURL=file-info.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-info.d.ts","sourceRoot":"","sources":["../../../src/lib/file-operations/file-info.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AAqCtD,wBAAsB,WAAW,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC,CA0BrE"}
@@ -0,0 +1,56 @@
1
+ import * as fs from 'node:fs/promises';
2
+ import * as path from 'node:path';
3
+ import { getMimeType } from '../constants.js';
4
+ import { getFileType, isHidden } from '../fs-helpers.js';
5
+ import { validateExistingPathDetailed } from '../path-validation.js';
6
+ const PERM_STRINGS = [
7
+ '---',
8
+ '--x',
9
+ '-w-',
10
+ '-wx',
11
+ 'r--',
12
+ 'r-x',
13
+ 'rw-',
14
+ 'rwx',
15
+ ];
16
+ function getPermissions(mode) {
17
+ const ownerIndex = (mode >> 6) & 0b111;
18
+ const groupIndex = (mode >> 3) & 0b111;
19
+ const otherIndex = mode & 0b111;
20
+ const owner = PERM_STRINGS[ownerIndex] ?? '---';
21
+ const group = PERM_STRINGS[groupIndex] ?? '---';
22
+ const other = PERM_STRINGS[otherIndex] ?? '---';
23
+ return `${owner}${group}${other}`;
24
+ }
25
+ async function getSymlinkTarget(pathToRead) {
26
+ try {
27
+ return await fs.readlink(pathToRead);
28
+ }
29
+ catch {
30
+ return undefined;
31
+ }
32
+ }
33
+ export async function getFileInfo(filePath) {
34
+ const { requestedPath, resolvedPath, isSymlink } = await validateExistingPathDetailed(filePath);
35
+ const name = path.basename(requestedPath);
36
+ const ext = path.extname(name).toLowerCase();
37
+ const mimeType = ext ? getMimeType(ext) : undefined;
38
+ const symlinkTarget = isSymlink
39
+ ? await getSymlinkTarget(requestedPath)
40
+ : undefined;
41
+ const stats = await fs.stat(resolvedPath);
42
+ return {
43
+ name,
44
+ path: requestedPath,
45
+ type: isSymlink ? 'symlink' : getFileType(stats),
46
+ size: stats.size,
47
+ created: stats.birthtime,
48
+ modified: stats.mtime,
49
+ accessed: stats.atime,
50
+ permissions: getPermissions(stats.mode),
51
+ isHidden: isHidden(name),
52
+ mimeType,
53
+ symlinkTarget,
54
+ };
55
+ }
56
+ //# sourceMappingURL=file-info.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-info.js","sourceRoot":"","sources":["../../../src/lib/file-operations/file-info.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,kBAAkB,CAAC;AACvC,OAAO,KAAK,IAAI,MAAM,WAAW,CAAC;AAGlC,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC9C,OAAO,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AACzD,OAAO,EAAE,4BAA4B,EAAE,MAAM,uBAAuB,CAAC;AAErE,MAAM,YAAY,GAAG;IACnB,KAAK;IACL,KAAK;IACL,KAAK;IACL,KAAK;IACL,KAAK;IACL,KAAK;IACL,KAAK;IACL,KAAK;CAC+B,CAAC;AAEvC,SAAS,cAAc,CAAC,IAAY;IAClC,MAAM,UAAU,GAAG,CAAC,IAAI,IAAI,CAAC,CAAC,GAAG,KAAK,CAAC;IACvC,MAAM,UAAU,GAAG,CAAC,IAAI,IAAI,CAAC,CAAC,GAAG,KAAK,CAAC;IACvC,MAAM,UAAU,GAAG,IAAI,GAAG,KAAK,CAAC;IAChC,MAAM,KAAK,GAAG,YAAY,CAAC,UAAU,CAAC,IAAI,KAAK,CAAC;IAChD,MAAM,KAAK,GAAG,YAAY,CAAC,UAAU,CAAC,IAAI,KAAK,CAAC;IAChD,MAAM,KAAK,GAAG,YAAY,CAAC,UAAU,CAAC,IAAI,KAAK,CAAC;IAEhD,OAAO,GAAG,KAAK,GAAG,KAAK,GAAG,KAAK,EAAE,CAAC;AACpC,CAAC;AAED,KAAK,UAAU,gBAAgB,CAC7B,UAAkB;IAElB,IAAI,CAAC;QACH,OAAO,MAAM,EAAE,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;IACvC,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,SAAS,CAAC;IACnB,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,WAAW,CAAC,QAAgB;IAChD,MAAM,EAAE,aAAa,EAAE,YAAY,EAAE,SAAS,EAAE,GAC9C,MAAM,4BAA4B,CAAC,QAAQ,CAAC,CAAC;IAE/C,MAAM,IAAI,GAAG,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC;IAC1C,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,WAAW,EAAE,CAAC;IAC7C,MAAM,QAAQ,GAAG,GAAG,CAAC,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IACpD,MAAM,aAAa,GAAG,SAAS;QAC7B,CAAC,CAAC,MAAM,gBAAgB,CAAC,aAAa,CAAC;QACvC,CAAC,CAAC,SAAS,CAAC;IAEd,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAE1C,OAAO;QACL,IAAI;QACJ,IAAI,EAAE,aAAa;QACnB,IAAI,EAAE,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC;QAChD,IAAI,EAAE,KAAK,CAAC,IAAI;QAChB,OAAO,EAAE,KAAK,CAAC,SAAS;QACxB,QAAQ,EAAE,KAAK,CAAC,KAAK;QACrB,QAAQ,EAAE,KAAK,CAAC,KAAK;QACrB,WAAW,EAAE,cAAc,CAAC,KAAK,CAAC,IAAI,CAAC;QACvC,QAAQ,EAAE,QAAQ,CAAC,IAAI,CAAC;QACxB,QAAQ;QACR,aAAa;KACd,CAAC;AACJ,CAAC"}
@@ -0,0 +1,10 @@
1
+ import type { ListDirectoryResult } from '../../config/types.js';
2
+ export declare function listDirectory(dirPath: string, options?: {
3
+ recursive?: boolean;
4
+ includeHidden?: boolean;
5
+ maxDepth?: number;
6
+ maxEntries?: number;
7
+ sortBy?: 'name' | 'size' | 'modified' | 'type';
8
+ includeSymlinkTargets?: boolean;
9
+ }): Promise<ListDirectoryResult>;
10
+ //# sourceMappingURL=list-directory.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"list-directory.d.ts","sourceRoot":"","sources":["../../../src/lib/file-operations/list-directory.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAEV,mBAAmB,EACpB,MAAM,uBAAuB,CAAC;AA0R/B,wBAAsB,aAAa,CACjC,OAAO,EAAE,MAAM,EACf,OAAO,GAAE;IACP,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,aAAa,CAAC,EAAE,OAAO,CAAC;IACxB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,MAAM,CAAC,EAAE,MAAM,GAAG,MAAM,GAAG,UAAU,GAAG,MAAM,CAAC;IAC/C,qBAAqB,CAAC,EAAE,OAAO,CAAC;CAC5B,GACL,OAAO,CAAC,mBAAmB,CAAC,CA4C9B"}
@@ -0,0 +1,189 @@
1
+ import * as fs from 'node:fs/promises';
2
+ import * as path from 'node:path';
3
+ import { DEFAULT_MAX_DEPTH, DIR_TRAVERSAL_CONCURRENCY, PARALLEL_CONCURRENCY, } from '../constants.js';
4
+ import { isHidden, processInParallel, runWorkQueue } from '../fs-helpers.js';
5
+ import { validateExistingPath } from '../path-validation.js';
6
+ import { sortByField } from './sorting.js';
7
+ function initListState() {
8
+ return {
9
+ entries: [],
10
+ totalFiles: 0,
11
+ totalDirectories: 0,
12
+ maxDepthReached: 0,
13
+ truncated: false,
14
+ skippedInaccessible: 0,
15
+ symlinksNotFollowed: 0,
16
+ };
17
+ }
18
+ function buildEntryBase(item, fullPath, relativePath, type) {
19
+ return {
20
+ name: item.name,
21
+ path: fullPath,
22
+ relativePath,
23
+ type,
24
+ };
25
+ }
26
+ function resolveEntryType(item, stats) {
27
+ if (item.isDirectory())
28
+ return 'directory';
29
+ if (item.isFile())
30
+ return 'file';
31
+ return stats.isSymbolicLink() ? 'symlink' : 'other';
32
+ }
33
+ async function buildSymlinkResult(item, fullPath, relativePath, includeSymlinkTargets) {
34
+ const stats = await fs.lstat(fullPath);
35
+ let symlinkTarget;
36
+ if (includeSymlinkTargets) {
37
+ try {
38
+ symlinkTarget = await fs.readlink(fullPath);
39
+ }
40
+ catch {
41
+ symlinkTarget = undefined;
42
+ }
43
+ }
44
+ const entry = {
45
+ name: item.name,
46
+ path: fullPath,
47
+ relativePath,
48
+ type: 'symlink',
49
+ size: stats.size,
50
+ modified: stats.mtime,
51
+ symlinkTarget,
52
+ };
53
+ return { entry, symlinkNotFollowed: true };
54
+ }
55
+ async function buildEnqueueDir(fullPath, depth, maxDepth, recursive) {
56
+ if (!recursive || depth + 1 > maxDepth)
57
+ return undefined;
58
+ return {
59
+ currentPath: await validateExistingPath(fullPath),
60
+ depth: depth + 1,
61
+ };
62
+ }
63
+ async function buildRegularResult(item, fullPath, relativePath, options) {
64
+ const stats = await fs.stat(fullPath);
65
+ const type = resolveEntryType(item, stats);
66
+ const entry = {
67
+ ...buildEntryBase(item, fullPath, relativePath, type),
68
+ size: type === 'file' ? stats.size : undefined,
69
+ modified: stats.mtime,
70
+ };
71
+ const enqueueDir = await buildEnqueueDir(fullPath, options.depth, options.maxDepth, options.recursive);
72
+ return { entry, enqueueDir };
73
+ }
74
+ function buildFallbackEntry(item, fullPath, relativePath) {
75
+ const type = item.isDirectory()
76
+ ? 'directory'
77
+ : item.isFile()
78
+ ? 'file'
79
+ : 'other';
80
+ return {
81
+ entry: buildEntryBase(item, fullPath, relativePath, type),
82
+ skippedInaccessible: true,
83
+ };
84
+ }
85
+ async function buildDirectoryItemResult(item, currentPath, basePath, options) {
86
+ const fullPath = path.join(currentPath, item.name);
87
+ const relativePath = path.relative(basePath, fullPath) || item.name;
88
+ try {
89
+ if (item.isSymbolicLink()) {
90
+ return await buildSymlinkResult(item, fullPath, relativePath, options.includeSymlinkTargets);
91
+ }
92
+ return await buildRegularResult(item, fullPath, relativePath, {
93
+ ...options,
94
+ basePath,
95
+ });
96
+ }
97
+ catch {
98
+ return buildFallbackEntry(item, fullPath, relativePath);
99
+ }
100
+ }
101
+ function createStopChecker(maxEntries, state) {
102
+ return () => {
103
+ if (maxEntries !== undefined && state.entries.length >= maxEntries) {
104
+ state.truncated = true;
105
+ return true;
106
+ }
107
+ return false;
108
+ };
109
+ }
110
+ async function readVisibleItems(currentPath, includeHidden) {
111
+ try {
112
+ const items = await fs.readdir(currentPath, { withFileTypes: true });
113
+ return includeHidden ? items : items.filter((item) => !isHidden(item.name));
114
+ }
115
+ catch {
116
+ return null;
117
+ }
118
+ }
119
+ function applyDirectoryItemResult(result, state, enqueue) {
120
+ const { entry, enqueueDir, skippedInaccessible, symlinkNotFollowed } = result;
121
+ state.entries.push(entry);
122
+ if (entry.type === 'directory')
123
+ state.totalDirectories++;
124
+ if (entry.type === 'file')
125
+ state.totalFiles++;
126
+ if (enqueueDir)
127
+ enqueue(enqueueDir);
128
+ if (skippedInaccessible)
129
+ state.skippedInaccessible++;
130
+ if (symlinkNotFollowed)
131
+ state.symlinksNotFollowed++;
132
+ }
133
+ function processResults(results, state, enqueue, shouldStop) {
134
+ for (const result of results) {
135
+ if (shouldStop())
136
+ break;
137
+ applyDirectoryItemResult(result, state, enqueue);
138
+ }
139
+ }
140
+ async function handleDirectory(params, enqueue, config, state, shouldStop) {
141
+ if (params.depth > config.maxDepth)
142
+ return;
143
+ if (shouldStop())
144
+ return;
145
+ state.maxDepthReached = Math.max(state.maxDepthReached, params.depth);
146
+ const items = await readVisibleItems(params.currentPath, config.includeHidden);
147
+ if (!items) {
148
+ state.skippedInaccessible++;
149
+ return;
150
+ }
151
+ const { results, errors } = await processInParallel(items, async (item) => buildDirectoryItemResult(item, params.currentPath, config.basePath, {
152
+ includeSymlinkTargets: config.includeSymlinkTargets,
153
+ recursive: config.recursive,
154
+ depth: params.depth,
155
+ maxDepth: config.maxDepth,
156
+ }), PARALLEL_CONCURRENCY);
157
+ state.skippedInaccessible += errors.length;
158
+ processResults(results, state, enqueue, shouldStop);
159
+ }
160
+ export async function listDirectory(dirPath, options = {}) {
161
+ const { recursive = false, includeHidden = false, maxDepth = DEFAULT_MAX_DEPTH, maxEntries, sortBy = 'name', includeSymlinkTargets = false, } = options;
162
+ const basePath = await validateExistingPath(dirPath);
163
+ const state = initListState();
164
+ const shouldStop = createStopChecker(maxEntries, state);
165
+ const config = {
166
+ basePath,
167
+ recursive,
168
+ includeHidden,
169
+ maxDepth,
170
+ maxEntries,
171
+ includeSymlinkTargets,
172
+ };
173
+ await runWorkQueue([{ currentPath: basePath, depth: 0 }], async (params, enqueue) => handleDirectory(params, enqueue, config, state, shouldStop), DIR_TRAVERSAL_CONCURRENCY);
174
+ sortByField(state.entries, sortBy);
175
+ return {
176
+ path: basePath,
177
+ entries: state.entries,
178
+ summary: {
179
+ totalEntries: state.entries.length,
180
+ totalFiles: state.totalFiles,
181
+ totalDirectories: state.totalDirectories,
182
+ maxDepthReached: state.maxDepthReached,
183
+ truncated: state.truncated,
184
+ skippedInaccessible: state.skippedInaccessible,
185
+ symlinksNotFollowed: state.symlinksNotFollowed,
186
+ },
187
+ };
188
+ }
189
+ //# sourceMappingURL=list-directory.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"list-directory.js","sourceRoot":"","sources":["../../../src/lib/file-operations/list-directory.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,16 @@
1
+ /**
2
+ * Estimates the complexity/performance impact of a glob pattern
3
+ * Helps users understand which patterns might be slow
4
+ */
5
+ export interface PatternComplexityAnalysis {
6
+ complexity: 'fast' | 'moderate' | 'slow';
7
+ score: number;
8
+ factors: string[];
9
+ recommendations: string[];
10
+ }
11
+ export declare function analyzePatternComplexity(pattern: string): PatternComplexityAnalysis;
12
+ /**
13
+ * Provides user-friendly complexity warning
14
+ */
15
+ export declare function getComplexityWarning(analysis: PatternComplexityAnalysis): string | null;
16
+ //# sourceMappingURL=pattern-complexity.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pattern-complexity.d.ts","sourceRoot":"","sources":["../../../src/lib/file-operations/pattern-complexity.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,MAAM,WAAW,yBAAyB;IACxC,UAAU,EAAE,MAAM,GAAG,UAAU,GAAG,MAAM,CAAC;IACzC,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,eAAe,EAAE,MAAM,EAAE,CAAC;CAC3B;AAWD,wBAAgB,wBAAwB,CACtC,OAAO,EAAE,MAAM,GACd,yBAAyB,CAwE3B;AAED;;GAEG;AACH,wBAAgB,oBAAoB,CAClC,QAAQ,EAAE,yBAAyB,GAClC,MAAM,GAAG,IAAI,CAQf"}
@@ -0,0 +1,86 @@
1
+ /**
2
+ * Estimates the complexity/performance impact of a glob pattern
3
+ * Helps users understand which patterns might be slow
4
+ */
5
+ const COMPLEXITY_WEIGHTS = {
6
+ leadingGlobstar: 15, // **/ at start
7
+ extglob: 20, // +(pattern), !(pattern), etc.
8
+ largeBraceExpansion: 25, // {a,b,c,...}
9
+ nestedQuantifiers: 30, // Complex regex
10
+ multiplePatternsInBrace: 10, // {a,b} = 10 points per item
11
+ leadingRecursion: 20, // Starting with **/
12
+ };
13
+ export function analyzePatternComplexity(pattern) {
14
+ let score = 0;
15
+ const factors = [];
16
+ const recommendations = [];
17
+ // Check for leading globstar
18
+ if (pattern.startsWith('**/')) {
19
+ score += COMPLEXITY_WEIGHTS.leadingGlobstar;
20
+ factors.push('Pattern starts with **/');
21
+ recommendations.push('Consider using a base directory if available (e.g., src/**/*.ts instead of **/src/**/*.ts)');
22
+ }
23
+ // Check for extglob patterns: +(), !(), ?(), *()
24
+ const extglobMatches = pattern.match(/[+!?*]\(\S+\)/g) ?? [];
25
+ if (extglobMatches.length > 0) {
26
+ score += extglobMatches.length * COMPLEXITY_WEIGHTS.extglob;
27
+ factors.push(`Extglob patterns: ${extglobMatches.join(', ')}`);
28
+ recommendations.push('Extglobs are slower. Use simpler patterns if possible (e.g., *.{js,ts} instead of +(js|ts))');
29
+ }
30
+ // Check for large brace expansions
31
+ const braceMatches = pattern.match(/\{[^}]*\}/g) ?? [];
32
+ for (const brace of braceMatches) {
33
+ const items = brace.slice(1, -1).split(',');
34
+ if (items.length > 20) {
35
+ score += COMPLEXITY_WEIGHTS.largeBraceExpansion;
36
+ factors.push(`Large brace expansion: ${items.length} items in {${items.slice(0, 3).join(',')}...}`);
37
+ recommendations.push('Use character classes or split into multiple simpler patterns');
38
+ }
39
+ else if (items.length > 5) {
40
+ score += items.length * COMPLEXITY_WEIGHTS.multiplePatternsInBrace;
41
+ }
42
+ }
43
+ // Check for nested quantifiers (regex complexity)
44
+ if (/([+*?}])\s*\)\s*([+*?{])/.test(pattern)) {
45
+ score += COMPLEXITY_WEIGHTS.nestedQuantifiers;
46
+ factors.push('Nested quantifiers detected');
47
+ recommendations.push('Simplify nested patterns to reduce regex complexity');
48
+ }
49
+ // Check for multiple ** segments
50
+ const globstarCount = (pattern.match(/\*\*/g) ?? []).length;
51
+ if (globstarCount > 2) {
52
+ score += 15;
53
+ factors.push(`Multiple globstars: ${globstarCount}`);
54
+ recommendations.push('Limit use of ** to narrow down search space');
55
+ }
56
+ // Determine complexity level
57
+ let complexity;
58
+ if (score < 20) {
59
+ complexity = 'fast';
60
+ }
61
+ else if (score < 50) {
62
+ complexity = 'moderate';
63
+ }
64
+ else {
65
+ complexity = 'slow';
66
+ }
67
+ return {
68
+ complexity,
69
+ score: Math.min(score, 100),
70
+ factors,
71
+ recommendations,
72
+ };
73
+ }
74
+ /**
75
+ * Provides user-friendly complexity warning
76
+ */
77
+ export function getComplexityWarning(analysis) {
78
+ if (analysis.complexity === 'slow') {
79
+ return `⚠️ Pattern complexity is HIGH (score: ${analysis.score}/100). This search may take longer than expected. ${analysis.recommendations[0] ?? ''}`;
80
+ }
81
+ if (analysis.complexity === 'moderate' && analysis.score > 40) {
82
+ return `ℹ️ Pattern complexity is MODERATE (score: ${analysis.score}/100). Consider optimizing: ${analysis.recommendations[0] ?? ''}`;
83
+ }
84
+ return null;
85
+ }
86
+ //# sourceMappingURL=pattern-complexity.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pattern-complexity.js","sourceRoot":"","sources":["../../../src/lib/file-operations/pattern-complexity.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,15 @@
1
+ export interface PatternValidationResult {
2
+ isValid: boolean;
3
+ error?: string;
4
+ suggestion?: string;
5
+ }
6
+ /**
7
+ * Validates glob patterns for safety and common mistakes
8
+ * Prevents backslashes, absolute paths, and overly complex patterns
9
+ */
10
+ export declare function validateGlobPattern(pattern: string): PatternValidationResult;
11
+ /**
12
+ * Throws McpError if pattern is invalid
13
+ */
14
+ export declare function validateGlobPatternOrThrow(pattern: string, basePath: string): void;
15
+ //# sourceMappingURL=pattern-validator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pattern-validator.d.ts","sourceRoot":"","sources":["../../../src/lib/file-operations/pattern-validator.ts"],"names":[],"mappings":"AAOA,MAAM,WAAW,uBAAuB;IACtC,OAAO,EAAE,OAAO,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED;;;GAGG;AACH,wBAAgB,mBAAmB,CAAC,OAAO,EAAE,MAAM,GAAG,uBAAuB,CAsD5E;AAED;;GAEG;AACH,wBAAgB,0BAA0B,CACxC,OAAO,EAAE,MAAM,EACf,QAAQ,EAAE,MAAM,GACf,IAAI,CAaN"}