@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.
- package/README.md +106 -29
- package/dist/__tests__/lib/errors.test.js +3 -23
- package/dist/__tests__/lib/errors.test.js.map +1 -1
- package/dist/__tests__/lib/file-operations.test.js +34 -0
- package/dist/__tests__/lib/file-operations.test.js.map +1 -1
- package/dist/__tests__/lib/path-validation.test.js +8 -0
- package/dist/__tests__/lib/path-validation.test.js.map +1 -1
- package/dist/__tests__/schemas/validators.test.js +101 -122
- package/dist/__tests__/schemas/validators.test.js.map +1 -1
- package/dist/__tests__/security/filesystem-boundary.test.js +1 -1
- package/dist/__tests__/security/filesystem-boundary.test.js.map +1 -1
- package/dist/config/types.d.ts +1 -85
- package/dist/config/types.d.ts.map +1 -1
- package/dist/config/types.js +0 -2
- package/dist/config/types.js.map +1 -1
- package/dist/index.js +3 -3
- package/dist/index.js.map +1 -1
- package/dist/instructions.md +1 -2
- package/dist/lib/constants.d.ts +3 -1
- package/dist/lib/constants.d.ts.map +1 -1
- package/dist/lib/constants.js +73 -259
- package/dist/lib/constants.js.map +1 -1
- package/dist/lib/errors.d.ts +25 -3
- package/dist/lib/errors.d.ts.map +1 -1
- package/dist/lib/errors.js +4 -94
- package/dist/lib/errors.js.map +1 -1
- package/dist/lib/file-operations/analyze-directory.d.ts +8 -0
- package/dist/lib/file-operations/analyze-directory.d.ts.map +1 -0
- package/dist/lib/file-operations/analyze-directory.js +117 -0
- package/dist/lib/file-operations/analyze-directory.js.map +1 -0
- package/dist/lib/file-operations/directory-helpers.d.ts +35 -0
- package/dist/lib/file-operations/directory-helpers.d.ts.map +1 -0
- package/dist/lib/file-operations/directory-helpers.js +136 -0
- package/dist/lib/file-operations/directory-helpers.js.map +1 -0
- package/dist/lib/file-operations/directory-items.d.ts +20 -0
- package/dist/lib/file-operations/directory-items.d.ts.map +1 -0
- package/dist/lib/file-operations/directory-items.js +85 -0
- package/dist/lib/file-operations/directory-items.js.map +1 -0
- package/dist/lib/file-operations/directory-iteration.d.ts +17 -0
- package/dist/lib/file-operations/directory-iteration.d.ts.map +1 -0
- package/dist/lib/file-operations/directory-iteration.js +55 -0
- package/dist/lib/file-operations/directory-iteration.js.map +1 -0
- package/dist/lib/file-operations/directory-tree.d.ts +9 -0
- package/dist/lib/file-operations/directory-tree.d.ts.map +1 -0
- package/dist/lib/file-operations/directory-tree.js +175 -0
- package/dist/lib/file-operations/directory-tree.js.map +1 -0
- package/dist/lib/file-operations/file-info.d.ts +3 -0
- package/dist/lib/file-operations/file-info.d.ts.map +1 -0
- package/dist/lib/file-operations/file-info.js +56 -0
- package/dist/lib/file-operations/file-info.js.map +1 -0
- package/dist/lib/file-operations/list-directory.d.ts +10 -0
- package/dist/lib/file-operations/list-directory.d.ts.map +1 -0
- package/dist/lib/file-operations/list-directory.js +189 -0
- package/dist/lib/file-operations/list-directory.js.map +1 -0
- package/dist/lib/file-operations/pattern-complexity.d.ts +16 -0
- package/dist/lib/file-operations/pattern-complexity.d.ts.map +1 -0
- package/dist/lib/file-operations/pattern-complexity.js +86 -0
- package/dist/lib/file-operations/pattern-complexity.js.map +1 -0
- package/dist/lib/file-operations/pattern-validator.d.ts +15 -0
- package/dist/lib/file-operations/pattern-validator.d.ts.map +1 -0
- package/dist/lib/file-operations/pattern-validator.js +69 -0
- package/dist/lib/file-operations/pattern-validator.js.map +1 -0
- package/dist/lib/file-operations/read-media-file.d.ts +5 -0
- package/dist/lib/file-operations/read-media-file.d.ts.map +1 -0
- package/dist/lib/file-operations/read-media-file.js +31 -0
- package/dist/lib/file-operations/read-media-file.js.map +1 -0
- package/dist/lib/file-operations/read-multiple-files.d.ts +16 -0
- package/dist/lib/file-operations/read-multiple-files.d.ts.map +1 -0
- package/dist/lib/file-operations/read-multiple-files.js +100 -0
- package/dist/lib/file-operations/read-multiple-files.js.map +1 -0
- package/dist/lib/file-operations/search-content.d.ts +18 -0
- package/dist/lib/file-operations/search-content.d.ts.map +1 -0
- package/dist/lib/file-operations/search-content.js +436 -0
- package/dist/lib/file-operations/search-content.js.map +1 -0
- package/dist/lib/file-operations/search-files.d.ts +11 -0
- package/dist/lib/file-operations/search-files.d.ts.map +1 -0
- package/dist/lib/file-operations/search-files.js +143 -0
- package/dist/lib/file-operations/search-files.js.map +1 -0
- package/dist/lib/file-operations/sorting.d.ts +12 -0
- package/dist/lib/file-operations/sorting.d.ts.map +1 -0
- package/dist/lib/file-operations/sorting.js +24 -0
- package/dist/lib/file-operations/sorting.js.map +1 -0
- package/dist/lib/file-operations.d.ts +9 -57
- package/dist/lib/file-operations.d.ts.map +1 -1
- package/dist/lib/file-operations.js +9 -773
- package/dist/lib/file-operations.js.map +1 -1
- package/dist/lib/fs-helpers/binary-detect.d.ts +3 -0
- package/dist/lib/fs-helpers/binary-detect.d.ts.map +1 -0
- package/dist/lib/fs-helpers/binary-detect.js +54 -0
- package/dist/lib/fs-helpers/binary-detect.js.map +1 -0
- package/dist/lib/fs-helpers/concurrency.d.ts +11 -0
- package/dist/lib/fs-helpers/concurrency.d.ts.map +1 -0
- package/dist/lib/fs-helpers/concurrency.js +95 -0
- package/dist/lib/fs-helpers/concurrency.js.map +1 -0
- package/dist/lib/fs-helpers/fs-utils.d.ts +5 -0
- package/dist/lib/fs-helpers/fs-utils.d.ts.map +1 -0
- package/dist/lib/fs-helpers/fs-utils.js +13 -0
- package/dist/lib/fs-helpers/fs-utils.js.map +1 -0
- package/dist/lib/fs-helpers/readers/head-file.d.ts +2 -0
- package/dist/lib/fs-helpers/readers/head-file.d.ts.map +1 -0
- package/dist/lib/fs-helpers/readers/head-file.js +73 -0
- package/dist/lib/fs-helpers/readers/head-file.js.map +1 -0
- package/dist/lib/fs-helpers/readers/line-range.d.ts +7 -0
- package/dist/lib/fs-helpers/readers/line-range.d.ts.map +1 -0
- package/dist/lib/fs-helpers/readers/line-range.js +46 -0
- package/dist/lib/fs-helpers/readers/line-range.js.map +1 -0
- package/dist/lib/fs-helpers/readers/read-file.d.ts +16 -0
- package/dist/lib/fs-helpers/readers/read-file.d.ts.map +1 -0
- package/dist/lib/fs-helpers/readers/read-file.js +87 -0
- package/dist/lib/fs-helpers/readers/read-file.js.map +1 -0
- package/dist/lib/fs-helpers/readers/stream-lines.d.ts +7 -0
- package/dist/lib/fs-helpers/readers/stream-lines.d.ts.map +1 -0
- package/dist/lib/fs-helpers/readers/stream-lines.js +61 -0
- package/dist/lib/fs-helpers/readers/stream-lines.js.map +1 -0
- package/dist/lib/fs-helpers/readers/tail-file.d.ts +2 -0
- package/dist/lib/fs-helpers/readers/tail-file.d.ts.map +1 -0
- package/dist/lib/fs-helpers/readers/tail-file.js +98 -0
- package/dist/lib/fs-helpers/readers/tail-file.js.map +1 -0
- package/dist/lib/fs-helpers/readers/utf8.d.ts +3 -0
- package/dist/lib/fs-helpers/readers/utf8.d.ts.map +1 -0
- package/dist/lib/fs-helpers/readers/utf8.js +22 -0
- package/dist/lib/fs-helpers/readers/utf8.js.map +1 -0
- package/dist/lib/fs-helpers/readers.d.ts +4 -0
- package/dist/lib/fs-helpers/readers.d.ts.map +1 -0
- package/dist/lib/fs-helpers/readers.js +4 -0
- package/dist/lib/fs-helpers/readers.js.map +1 -0
- package/dist/lib/fs-helpers.d.ts +4 -25
- package/dist/lib/fs-helpers.d.ts.map +1 -1
- package/dist/lib/fs-helpers.js +4 -327
- package/dist/lib/fs-helpers.js.map +1 -1
- package/dist/lib/path-validation/allowed-directories.d.ts +9 -0
- package/dist/lib/path-validation/allowed-directories.d.ts.map +1 -0
- package/dist/lib/path-validation/allowed-directories.js +94 -0
- package/dist/lib/path-validation/allowed-directories.js.map +1 -0
- package/dist/lib/path-validation/errors.d.ts +5 -0
- package/dist/lib/path-validation/errors.d.ts.map +1 -0
- package/dist/lib/path-validation/errors.js +33 -0
- package/dist/lib/path-validation/errors.js.map +1 -0
- package/dist/lib/path-validation/roots.d.ts +3 -0
- package/dist/lib/path-validation/roots.d.ts.map +1 -0
- package/dist/lib/path-validation/roots.js +49 -0
- package/dist/lib/path-validation/roots.js.map +1 -0
- package/dist/lib/path-validation/validators.d.ts +9 -0
- package/dist/lib/path-validation/validators.d.ts.map +1 -0
- package/dist/lib/path-validation/validators.js +70 -0
- package/dist/lib/path-validation/validators.js.map +1 -0
- package/dist/lib/path-validation.d.ts +3 -7
- package/dist/lib/path-validation.d.ts.map +1 -1
- package/dist/lib/path-validation.js +3 -141
- package/dist/lib/path-validation.js.map +1 -1
- package/dist/lib/performance/monitor.d.ts +25 -0
- package/dist/lib/performance/monitor.d.ts.map +1 -0
- package/dist/lib/performance/monitor.js +84 -0
- package/dist/lib/performance/monitor.js.map +1 -0
- package/dist/schemas/input-helpers.d.ts +27 -0
- package/dist/schemas/input-helpers.d.ts.map +1 -0
- package/dist/schemas/input-helpers.js +176 -0
- package/dist/schemas/input-helpers.js.map +1 -0
- package/dist/schemas/inputs.d.ts +14 -7
- package/dist/schemas/inputs.d.ts.map +1 -1
- package/dist/schemas/inputs.js +56 -228
- package/dist/schemas/inputs.js.map +1 -1
- package/dist/schemas/output-helpers.d.ts +24 -0
- package/dist/schemas/output-helpers.d.ts.map +1 -0
- package/dist/schemas/output-helpers.js +13 -0
- package/dist/schemas/output-helpers.js.map +1 -0
- package/dist/schemas/outputs.d.ts +476 -42
- package/dist/schemas/outputs.d.ts.map +1 -1
- package/dist/schemas/outputs.js +26 -41
- package/dist/schemas/outputs.js.map +1 -1
- package/dist/server.d.ts +9 -1
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +29 -45
- package/dist/server.js.map +1 -1
- package/dist/tools/analyze-directory.d.ts.map +1 -1
- package/dist/tools/analyze-directory.js +115 -53
- package/dist/tools/analyze-directory.js.map +1 -1
- package/dist/tools/directory-tree.d.ts.map +1 -1
- package/dist/tools/directory-tree.js +86 -49
- package/dist/tools/directory-tree.js.map +1 -1
- package/dist/tools/get-file-info.d.ts.map +1 -1
- package/dist/tools/get-file-info.js +71 -37
- package/dist/tools/get-file-info.js.map +1 -1
- package/dist/tools/list-allowed-dirs.d.ts.map +1 -1
- package/dist/tools/list-allowed-dirs.js +48 -35
- package/dist/tools/list-allowed-dirs.js.map +1 -1
- package/dist/tools/list-directory.d.ts.map +1 -1
- package/dist/tools/list-directory.js +129 -58
- package/dist/tools/list-directory.js.map +1 -1
- package/dist/tools/read-file.d.ts.map +1 -1
- package/dist/tools/read-file.js +70 -56
- package/dist/tools/read-file.js.map +1 -1
- package/dist/tools/read-media-file.d.ts.map +1 -1
- package/dist/tools/read-media-file.js +39 -41
- package/dist/tools/read-media-file.js.map +1 -1
- package/dist/tools/read-multiple-files.d.ts.map +1 -1
- package/dist/tools/read-multiple-files.js +58 -50
- package/dist/tools/read-multiple-files.js.map +1 -1
- package/dist/tools/search-content.d.ts.map +1 -1
- package/dist/tools/search-content.js +148 -89
- package/dist/tools/search-content.js.map +1 -1
- package/dist/tools/search-files.d.ts.map +1 -1
- package/dist/tools/search-files.js +123 -50
- package/dist/tools/search-files.js.map +1 -1
- package/dist/tools/tool-response.d.ts +9 -0
- package/dist/tools/tool-response.d.ts.map +1 -0
- package/dist/tools/tool-response.js +7 -0
- package/dist/tools/tool-response.js.map +1 -0
- package/package.json +2 -1
- package/dist/__tests__/errors.test.d.ts +0 -2
- package/dist/__tests__/errors.test.d.ts.map +0 -1
- package/dist/__tests__/errors.test.js +0 -88
- package/dist/__tests__/errors.test.js.map +0 -1
- package/dist/__tests__/file-operations.test.d.ts +0 -2
- package/dist/__tests__/file-operations.test.d.ts.map +0 -1
- package/dist/__tests__/file-operations.test.js +0 -230
- package/dist/__tests__/file-operations.test.js.map +0 -1
- package/dist/__tests__/lib/formatters.test.d.ts +0 -2
- package/dist/__tests__/lib/formatters.test.d.ts.map +0 -1
- package/dist/__tests__/lib/formatters.test.js +0 -248
- package/dist/__tests__/lib/formatters.test.js.map +0 -1
- package/dist/__tests__/lib/image-parsing.test.d.ts +0 -2
- package/dist/__tests__/lib/image-parsing.test.d.ts.map +0 -1
- package/dist/__tests__/lib/image-parsing.test.js +0 -262
- package/dist/__tests__/lib/image-parsing.test.js.map +0 -1
- package/dist/__tests__/path-validation.test.d.ts +0 -2
- package/dist/__tests__/path-validation.test.d.ts.map +0 -1
- package/dist/__tests__/path-validation.test.js +0 -92
- package/dist/__tests__/path-validation.test.js.map +0 -1
- package/dist/lib/directory-helpers.d.ts +0 -4
- package/dist/lib/directory-helpers.d.ts.map +0 -1
- package/dist/lib/directory-helpers.js +0 -36
- package/dist/lib/directory-helpers.js.map +0 -1
- package/dist/lib/formatters.d.ts +0 -10
- package/dist/lib/formatters.d.ts.map +0 -1
- package/dist/lib/formatters.js +0 -202
- package/dist/lib/formatters.js.map +0 -1
- package/dist/lib/image-parsing.d.ts +0 -4
- package/dist/lib/image-parsing.d.ts.map +0 -1
- package/dist/lib/image-parsing.js +0 -124
- package/dist/lib/image-parsing.js.map +0 -1
- package/dist/lib/mcp-logger.d.ts +0 -11
- package/dist/lib/mcp-logger.d.ts.map +0 -1
- package/dist/lib/mcp-logger.js +0 -49
- package/dist/lib/mcp-logger.js.map +0 -1
- package/dist/lib/roots-utils.d.ts +0 -7
- package/dist/lib/roots-utils.d.ts.map +0 -1
- package/dist/lib/roots-utils.js +0 -39
- package/dist/lib/roots-utils.js.map +0 -1
- package/dist/lib/search-helpers.d.ts +0 -16
- package/dist/lib/search-helpers.d.ts.map +0 -1
- package/dist/lib/search-helpers.js +0 -169
- package/dist/lib/search-helpers.js.map +0 -1
- package/dist/lib/sorting.d.ts +0 -12
- package/dist/lib/sorting.d.ts.map +0 -1
- package/dist/lib/sorting.js +0 -41
- package/dist/lib/sorting.js.map +0 -1
- package/dist/lib/types.d.ts +0 -6
- package/dist/lib/types.d.ts.map +0 -1
- package/dist/lib/types.js +0 -2
- package/dist/lib/types.js.map +0 -1
- package/dist/prompts/analyze-codebase.d.ts +0 -3
- package/dist/prompts/analyze-codebase.d.ts.map +0 -1
- package/dist/prompts/analyze-codebase.js +0 -144
- package/dist/prompts/analyze-codebase.js.map +0 -1
- package/dist/prompts/filesystem-query.d.ts +0 -3
- package/dist/prompts/filesystem-query.d.ts.map +0 -1
- package/dist/prompts/filesystem-query.js +0 -168
- package/dist/prompts/filesystem-query.js.map +0 -1
- package/dist/prompts/find-duplicates.d.ts +0 -3
- package/dist/prompts/find-duplicates.d.ts.map +0 -1
- package/dist/prompts/find-duplicates.js +0 -77
- package/dist/prompts/find-duplicates.js.map +0 -1
- package/dist/prompts/index.d.ts +0 -3
- package/dist/prompts/index.d.ts.map +0 -1
- package/dist/prompts/index.js +0 -13
- package/dist/prompts/index.js.map +0 -1
- package/dist/prompts/project-overview.d.ts +0 -3
- package/dist/prompts/project-overview.d.ts.map +0 -1
- package/dist/prompts/project-overview.js +0 -122
- package/dist/prompts/project-overview.js.map +0 -1
- package/dist/prompts/search-and-replace.d.ts +0 -3
- package/dist/prompts/search-and-replace.d.ts.map +0 -1
- package/dist/prompts/search-and-replace.js +0 -130
- package/dist/prompts/search-and-replace.js.map +0 -1
- package/dist/prompts/shared.d.ts +0 -11
- package/dist/prompts/shared.d.ts.map +0 -1
- package/dist/prompts/shared.js +0 -32
- package/dist/prompts/shared.js.map +0 -1
- package/dist/resources/index.d.ts +0 -3
- package/dist/resources/index.d.ts.map +0 -1
- package/dist/resources/index.js +0 -54
- package/dist/resources/index.js.map +0 -1
- package/dist/schemas/validators.d.ts +0 -12
- package/dist/schemas/validators.d.ts.map +0 -1
- package/dist/schemas/validators.js +0 -35
- package/dist/schemas/validators.js.map +0 -1
- package/dist/utils/index.d.ts +0 -2
- package/dist/utils/index.d.ts.map +0 -1
- package/dist/utils/index.js +0 -2
- package/dist/utils/index.js.map +0 -1
- package/dist/utils/response-helpers.d.ts +0 -22
- package/dist/utils/response-helpers.d.ts.map +0 -1
- package/dist/utils/response-helpers.js +0 -24
- 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 @@
|
|
|
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"}
|