agent-security-scanner-mcp 3.20.0 → 4.0.0

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 (126) hide show
  1. package/README.md +144 -43
  2. package/code-review-agent/.env.example +8 -0
  3. package/code-review-agent/README.md +142 -0
  4. package/code-review-agent/TODO.md +149 -0
  5. package/code-review-agent/bin/cr-agent.ts +313 -0
  6. package/code-review-agent/dist/bin/cr-agent.d.ts +3 -0
  7. package/code-review-agent/dist/bin/cr-agent.d.ts.map +1 -0
  8. package/code-review-agent/dist/bin/cr-agent.js +299 -0
  9. package/code-review-agent/dist/bin/cr-agent.js.map +1 -0
  10. package/code-review-agent/dist/src/analyzer/engine.d.ts +16 -0
  11. package/code-review-agent/dist/src/analyzer/engine.d.ts.map +1 -0
  12. package/code-review-agent/dist/src/analyzer/engine.js +298 -0
  13. package/code-review-agent/dist/src/analyzer/engine.js.map +1 -0
  14. package/code-review-agent/dist/src/analyzer/intent.d.ts +10 -0
  15. package/code-review-agent/dist/src/analyzer/intent.d.ts.map +1 -0
  16. package/code-review-agent/dist/src/analyzer/intent.js +40 -0
  17. package/code-review-agent/dist/src/analyzer/intent.js.map +1 -0
  18. package/code-review-agent/dist/src/analyzer/semantic.d.ts +19 -0
  19. package/code-review-agent/dist/src/analyzer/semantic.d.ts.map +1 -0
  20. package/code-review-agent/dist/src/analyzer/semantic.js +150 -0
  21. package/code-review-agent/dist/src/analyzer/semantic.js.map +1 -0
  22. package/code-review-agent/dist/src/context/assembler.d.ts +16 -0
  23. package/code-review-agent/dist/src/context/assembler.d.ts.map +1 -0
  24. package/code-review-agent/dist/src/context/assembler.js +135 -0
  25. package/code-review-agent/dist/src/context/assembler.js.map +1 -0
  26. package/code-review-agent/dist/src/context/file.d.ts +6 -0
  27. package/code-review-agent/dist/src/context/file.d.ts.map +1 -0
  28. package/code-review-agent/dist/src/context/file.js +139 -0
  29. package/code-review-agent/dist/src/context/file.js.map +1 -0
  30. package/code-review-agent/dist/src/context/project.d.ts +4 -0
  31. package/code-review-agent/dist/src/context/project.d.ts.map +1 -0
  32. package/code-review-agent/dist/src/context/project.js +252 -0
  33. package/code-review-agent/dist/src/context/project.js.map +1 -0
  34. package/code-review-agent/dist/src/graph/dependency.d.ts +11 -0
  35. package/code-review-agent/dist/src/graph/dependency.d.ts.map +1 -0
  36. package/code-review-agent/dist/src/graph/dependency.js +102 -0
  37. package/code-review-agent/dist/src/graph/dependency.js.map +1 -0
  38. package/code-review-agent/dist/src/graph/resolver.d.ts +9 -0
  39. package/code-review-agent/dist/src/graph/resolver.d.ts.map +1 -0
  40. package/code-review-agent/dist/src/graph/resolver.js +124 -0
  41. package/code-review-agent/dist/src/graph/resolver.js.map +1 -0
  42. package/code-review-agent/dist/src/index.d.ts +21 -0
  43. package/code-review-agent/dist/src/index.d.ts.map +1 -0
  44. package/code-review-agent/dist/src/index.js +21 -0
  45. package/code-review-agent/dist/src/index.js.map +1 -0
  46. package/code-review-agent/dist/src/llm/anthropic.d.ts +13 -0
  47. package/code-review-agent/dist/src/llm/anthropic.d.ts.map +1 -0
  48. package/code-review-agent/dist/src/llm/anthropic.js +83 -0
  49. package/code-review-agent/dist/src/llm/anthropic.js.map +1 -0
  50. package/code-review-agent/dist/src/llm/claude-cli.d.ts +13 -0
  51. package/code-review-agent/dist/src/llm/claude-cli.d.ts.map +1 -0
  52. package/code-review-agent/dist/src/llm/claude-cli.js +142 -0
  53. package/code-review-agent/dist/src/llm/claude-cli.js.map +1 -0
  54. package/code-review-agent/dist/src/llm/openai.d.ts +13 -0
  55. package/code-review-agent/dist/src/llm/openai.d.ts.map +1 -0
  56. package/code-review-agent/dist/src/llm/openai.js +78 -0
  57. package/code-review-agent/dist/src/llm/openai.js.map +1 -0
  58. package/code-review-agent/dist/src/llm/provider.d.ts +18 -0
  59. package/code-review-agent/dist/src/llm/provider.d.ts.map +1 -0
  60. package/code-review-agent/dist/src/llm/provider.js +11 -0
  61. package/code-review-agent/dist/src/llm/provider.js.map +1 -0
  62. package/code-review-agent/dist/src/llm/router.d.ts +14 -0
  63. package/code-review-agent/dist/src/llm/router.d.ts.map +1 -0
  64. package/code-review-agent/dist/src/llm/router.js +67 -0
  65. package/code-review-agent/dist/src/llm/router.js.map +1 -0
  66. package/code-review-agent/dist/src/llm/schemas.d.ts +18 -0
  67. package/code-review-agent/dist/src/llm/schemas.d.ts.map +1 -0
  68. package/code-review-agent/dist/src/llm/schemas.js +91 -0
  69. package/code-review-agent/dist/src/llm/schemas.js.map +1 -0
  70. package/code-review-agent/dist/src/types/analysis.d.ts +56 -0
  71. package/code-review-agent/dist/src/types/analysis.d.ts.map +1 -0
  72. package/code-review-agent/dist/src/types/analysis.js +2 -0
  73. package/code-review-agent/dist/src/types/analysis.js.map +1 -0
  74. package/code-review-agent/dist/src/types/config.d.ts +24 -0
  75. package/code-review-agent/dist/src/types/config.d.ts.map +1 -0
  76. package/code-review-agent/dist/src/types/config.js +42 -0
  77. package/code-review-agent/dist/src/types/config.js.map +1 -0
  78. package/code-review-agent/dist/src/types/findings.d.ts +236 -0
  79. package/code-review-agent/dist/src/types/findings.d.ts.map +1 -0
  80. package/code-review-agent/dist/src/types/findings.js +64 -0
  81. package/code-review-agent/dist/src/types/findings.js.map +1 -0
  82. package/code-review-agent/package.json +36 -0
  83. package/code-review-agent/src/analyzer/engine.ts +374 -0
  84. package/code-review-agent/src/analyzer/intent.ts +49 -0
  85. package/code-review-agent/src/analyzer/semantic.ts +222 -0
  86. package/code-review-agent/src/context/assembler.ts +165 -0
  87. package/code-review-agent/src/context/file.ts +145 -0
  88. package/code-review-agent/src/context/project.ts +253 -0
  89. package/code-review-agent/src/graph/dependency.ts +116 -0
  90. package/code-review-agent/src/graph/resolver.ts +138 -0
  91. package/code-review-agent/src/index.ts +58 -0
  92. package/code-review-agent/src/llm/anthropic.ts +106 -0
  93. package/code-review-agent/src/llm/claude-cli.ts +188 -0
  94. package/code-review-agent/src/llm/openai.ts +95 -0
  95. package/code-review-agent/src/llm/provider.ts +33 -0
  96. package/code-review-agent/src/llm/router.ts +86 -0
  97. package/code-review-agent/src/llm/schemas.ts +125 -0
  98. package/code-review-agent/src/types/analysis.ts +62 -0
  99. package/code-review-agent/src/types/config.ts +72 -0
  100. package/code-review-agent/src/types/findings.ts +81 -0
  101. package/code-review-agent/tests/analyzer/engine.test.ts +194 -0
  102. package/code-review-agent/tests/analyzer/intent.test.ts +76 -0
  103. package/code-review-agent/tests/analyzer/semantic.test.ts +131 -0
  104. package/code-review-agent/tests/context/file.test.ts +21 -0
  105. package/code-review-agent/tests/context/project.test.ts +20 -0
  106. package/code-review-agent/tests/fixtures/safe-build-tool/README.md +19 -0
  107. package/code-review-agent/tests/fixtures/safe-build-tool/builder.js +52 -0
  108. package/code-review-agent/tests/fixtures/safe-file-manager/README.md +16 -0
  109. package/code-review-agent/tests/fixtures/safe-file-manager/organizer.py +70 -0
  110. package/code-review-agent/tests/fixtures/vuln-api-server/README.md +17 -0
  111. package/code-review-agent/tests/fixtures/vuln-api-server/server.js +52 -0
  112. package/code-review-agent/tests/fixtures/vuln-ecommerce/README.md +18 -0
  113. package/code-review-agent/tests/fixtures/vuln-ecommerce/checkout.js +63 -0
  114. package/code-review-agent/tests/graph/dependency.test.ts +136 -0
  115. package/code-review-agent/tests/helpers/mock-provider.ts +48 -0
  116. package/code-review-agent/tests/llm/claude-cli.test.ts +251 -0
  117. package/code-review-agent/tests/llm/router.test.ts +77 -0
  118. package/code-review-agent/tests/llm/schemas.test.ts +142 -0
  119. package/code-review-agent/tsconfig.json +20 -0
  120. package/code-review-agent/vitest.config.ts +11 -0
  121. package/index.js +18 -18
  122. package/openclaw.plugin.json +2 -2
  123. package/package.json +13 -3
  124. package/server.json +3 -3
  125. package/src/cli/init-hooks.js +3 -3
  126. package/src/cli/init.js +1 -1
@@ -0,0 +1,135 @@
1
+ import { formatProjectContextForLLM } from './project.js';
2
+ const TOKEN_BUDGETS = {
3
+ anthropic: 100_000,
4
+ openai: 60_000,
5
+ 'claude-cli': 100_000,
6
+ };
7
+ const TRUNCATION_MARKER = '\n[TRUNCATED — file too large for context window]\n';
8
+ // Reserve 20% of budget for LLM output tokens
9
+ const OUTPUT_RESERVE = 0.2;
10
+ export class ContextAssembler {
11
+ provider;
12
+ constructor(provider) {
13
+ this.provider = provider;
14
+ }
15
+ /**
16
+ * Calculate how many lines of source code fit in the remaining
17
+ * token budget after system prompt, intent, project context, and
18
+ * metadata are accounted for.
19
+ */
20
+ calculateMaxLines(intent, project, file, systemPrompt) {
21
+ const budget = TOKEN_BUDGETS[this.provider.providerName] ?? 60_000;
22
+ const usableBudget = budget * (1 - OUTPUT_RESERVE);
23
+ // Measure fixed overhead
24
+ const overheadParts = [
25
+ systemPrompt,
26
+ formatIntent(intent),
27
+ formatProjectContextForLLM(project),
28
+ formatFileMetadata(file),
29
+ // Framing text around file content
30
+ `\n## File Content\nFile: ${file.filePath} (${file.language})\n\`\`\`\n\`\`\`\n`,
31
+ ];
32
+ const overheadTokens = this.provider.countTokens(overheadParts.join('\n'));
33
+ const remainingTokens = usableBudget - overheadTokens;
34
+ if (remainingTokens <= 0)
35
+ return 100; // absolute minimum
36
+ // Estimate chars per line from actual file content (avg line length)
37
+ const lines = file.content.split('\n');
38
+ const avgCharsPerLine = lines.length > 0
39
+ ? file.content.length / lines.length
40
+ : 40;
41
+ // ~4 chars per token + line number prefix ("1234: ")
42
+ const charsPerToken = 4;
43
+ const lineNumberOverhead = 6;
44
+ const tokensPerLine = (avgCharsPerLine + lineNumberOverhead) / charsPerToken;
45
+ const maxLines = Math.floor(remainingTokens / tokensPerLine);
46
+ return Math.max(maxLines, 100); // never go below 100 lines
47
+ }
48
+ assembleAnalysisContext(intent, project, file) {
49
+ const budget = TOKEN_BUDGETS[this.provider.providerName] ?? 60_000;
50
+ // Priority order: intent > file content > project context
51
+ const sections = [
52
+ {
53
+ label: 'Intent Profile',
54
+ content: formatIntent(intent),
55
+ priority: 1,
56
+ },
57
+ {
58
+ label: 'File Content',
59
+ content: formatFileContent(file),
60
+ priority: 2,
61
+ },
62
+ {
63
+ label: 'Project Context',
64
+ content: formatProjectContextForLLM(project),
65
+ priority: 3,
66
+ },
67
+ {
68
+ label: 'File Metadata',
69
+ content: formatFileMetadata(file),
70
+ priority: 4,
71
+ },
72
+ ];
73
+ // Sort by priority and assemble within budget
74
+ sections.sort((a, b) => a.priority - b.priority);
75
+ let assembled = '';
76
+ let usedTokens = 0;
77
+ for (const section of sections) {
78
+ const sectionText = `\n## ${section.label}\n${section.content}\n`;
79
+ const sectionTokens = this.provider.countTokens(sectionText);
80
+ if (usedTokens + sectionTokens > budget * 0.8) {
81
+ // Truncate this section to fit
82
+ const remainingBudget = Math.floor((budget * 0.8 - usedTokens) * 4); // rough chars
83
+ if (remainingBudget > 200) {
84
+ assembled += `\n## ${section.label}\n${section.content.slice(0, remainingBudget)}${TRUNCATION_MARKER}`;
85
+ }
86
+ break;
87
+ }
88
+ assembled += sectionText;
89
+ usedTokens += sectionTokens;
90
+ }
91
+ return assembled;
92
+ }
93
+ assembleTriageContext(project, file) {
94
+ // Triage needs less context — just file overview + project summary
95
+ const sections = [
96
+ `## File: ${file.filePath}`,
97
+ `Language: ${file.language} | Lines: ${file.lineCount}`,
98
+ `Test: ${file.isTestFile} | Config: ${file.isConfigFile} | Generated: ${file.isGenerated}`,
99
+ `Imports: ${file.imports.slice(0, 10).join(', ')}`,
100
+ '',
101
+ `## Project`,
102
+ `Language: ${project.language} | Framework: ${project.framework}`,
103
+ project.readme ? `README excerpt: ${project.readme.slice(0, 500)}` : 'No README',
104
+ ];
105
+ // Include first 100 lines of file content for triage
106
+ const preview = file.content.split('\n').slice(0, 100).join('\n');
107
+ sections.push('', '## File Preview (first 100 lines)', '```', preview, '```');
108
+ return sections.join('\n');
109
+ }
110
+ }
111
+ function formatIntent(intent) {
112
+ return [
113
+ `Purpose: ${intent.purpose}`,
114
+ `Risk Domain: ${intent.riskDomain}`,
115
+ `Framework: ${intent.framework}`,
116
+ `Expected Behaviors: ${intent.expectedBehaviors.join('; ')}`,
117
+ `Unexpected Behaviors: ${intent.unexpectedBehaviors.join('; ')}`,
118
+ ].join('\n');
119
+ }
120
+ function formatFileContent(file) {
121
+ const numbered = file.content
122
+ .split('\n')
123
+ .map((line, i) => `${i + 1}: ${line}`)
124
+ .join('\n');
125
+ return `File: ${file.filePath} (${file.language})\n\`\`\`\n${numbered}\n\`\`\``;
126
+ }
127
+ function formatFileMetadata(file) {
128
+ const parts = [
129
+ `Imports: ${file.imports.join(', ') || 'none'}`,
130
+ `Imported by: ${file.importedBy.join(', ') || 'none'}`,
131
+ `Siblings: ${file.siblingFiles.join(', ') || 'none'}`,
132
+ ];
133
+ return parts.join('\n');
134
+ }
135
+ //# sourceMappingURL=assembler.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"assembler.js","sourceRoot":"","sources":["../../../src/context/assembler.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ import type { DependencyGraph, FileContext } from '../types/analysis.js';
2
+ export declare function buildFileContext(filePath: string, projectRoot: string, graph?: DependencyGraph): FileContext;
3
+ export declare function isTestFile(filePath: string): boolean;
4
+ export declare function isConfigFile(filePath: string): boolean;
5
+ export declare function isGeneratedFile(content: string): boolean;
6
+ //# sourceMappingURL=file.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file.d.ts","sourceRoot":"","sources":["../../../src/context/file.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,eAAe,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAqDzE,wBAAgB,gBAAgB,CAC9B,QAAQ,EAAE,MAAM,EAChB,WAAW,EAAE,MAAM,EACnB,KAAK,CAAC,EAAE,eAAe,GACtB,WAAW,CA2Cb;AAED,wBAAgB,UAAU,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAMpD;AAED,wBAAgB,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAGtD;AAED,wBAAgB,eAAe,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAGxD"}
@@ -0,0 +1,139 @@
1
+ import * as fs from 'node:fs';
2
+ import * as path from 'node:path';
3
+ const LANGUAGE_MAP = {
4
+ '.js': 'javascript',
5
+ '.mjs': 'javascript',
6
+ '.cjs': 'javascript',
7
+ '.jsx': 'javascript',
8
+ '.ts': 'typescript',
9
+ '.tsx': 'typescript',
10
+ '.py': 'python',
11
+ '.go': 'go',
12
+ '.rs': 'rust',
13
+ '.java': 'java',
14
+ '.rb': 'ruby',
15
+ '.php': 'php',
16
+ '.c': 'c',
17
+ '.cpp': 'cpp',
18
+ '.cs': 'csharp',
19
+ '.swift': 'swift',
20
+ '.kt': 'kotlin',
21
+ };
22
+ const TEST_PATTERNS = [
23
+ /\.test\.[jt]sx?$/,
24
+ /\.spec\.[jt]sx?$/,
25
+ /_test\.go$/,
26
+ /test_.*\.py$/,
27
+ /.*_test\.py$/,
28
+ /\.test\.py$/,
29
+ /Test\.java$/,
30
+ /\.test\.rb$/,
31
+ ];
32
+ const CONFIG_PATTERNS = [
33
+ /\.config\.[jt]s$/,
34
+ /\.rc$/,
35
+ /\.json$/,
36
+ /\.ya?ml$/,
37
+ /\.toml$/,
38
+ /\.ini$/,
39
+ /\.env/,
40
+ /Makefile$/,
41
+ /Dockerfile$/,
42
+ ];
43
+ const GENERATED_MARKERS = [
44
+ '// Code generated',
45
+ '# Generated by',
46
+ '// AUTO-GENERATED',
47
+ '/* eslint-disable */',
48
+ '// @generated',
49
+ ];
50
+ export function buildFileContext(filePath, projectRoot, graph) {
51
+ const content = fs.readFileSync(filePath, 'utf-8');
52
+ const ext = path.extname(filePath);
53
+ const language = LANGUAGE_MAP[ext] ?? 'unknown';
54
+ const lines = content.split('\n');
55
+ const relativePath = path.relative(projectRoot, filePath);
56
+ const dirName = path.dirname(filePath);
57
+ let siblingFiles = [];
58
+ try {
59
+ siblingFiles = fs
60
+ .readdirSync(dirName)
61
+ .filter((f) => {
62
+ const full = path.join(dirName, f);
63
+ try {
64
+ return fs.statSync(full).isFile();
65
+ }
66
+ catch {
67
+ return false;
68
+ }
69
+ })
70
+ .filter((f) => f !== path.basename(filePath))
71
+ .slice(0, 20);
72
+ }
73
+ catch { /* dir read error */ }
74
+ const imports = extractImports(content, language);
75
+ let importedBy = [];
76
+ if (graph) {
77
+ const node = graph.nodes.get(relativePath) ?? graph.nodes.get(filePath);
78
+ if (node) {
79
+ importedBy = node.importedBy;
80
+ }
81
+ }
82
+ return {
83
+ filePath: relativePath,
84
+ content,
85
+ language,
86
+ lineCount: lines.length,
87
+ imports,
88
+ importedBy,
89
+ siblingFiles,
90
+ isTestFile: isTestFile(relativePath),
91
+ isConfigFile: isConfigFile(relativePath),
92
+ isGenerated: isGeneratedFile(content),
93
+ };
94
+ }
95
+ export function isTestFile(filePath) {
96
+ const name = path.basename(filePath);
97
+ // Normalize separators for cross-platform matching
98
+ const normalized = filePath.replace(/\\/g, '/');
99
+ return TEST_PATTERNS.some((p) => p.test(name)) ||
100
+ /(^|\/)(test|tests|__tests__)\//.test(normalized);
101
+ }
102
+ export function isConfigFile(filePath) {
103
+ const name = path.basename(filePath);
104
+ return CONFIG_PATTERNS.some((p) => p.test(name));
105
+ }
106
+ export function isGeneratedFile(content) {
107
+ const header = content.slice(0, 500);
108
+ return GENERATED_MARKERS.some((m) => header.includes(m));
109
+ }
110
+ function extractImports(content, language) {
111
+ const imports = [];
112
+ if (['javascript', 'typescript'].includes(language)) {
113
+ // ES imports
114
+ const esImports = content.matchAll(/import\s+(?:.*?\s+from\s+)?['"]([^'"]+)['"]/g);
115
+ for (const m of esImports)
116
+ imports.push(m[1]);
117
+ // require
118
+ const requires = content.matchAll(/require\s*\(\s*['"]([^'"]+)['"]\s*\)/g);
119
+ for (const m of requires)
120
+ imports.push(m[1]);
121
+ }
122
+ else if (language === 'python') {
123
+ const pyImports = content.matchAll(/(?:from\s+(\S+)\s+import|import\s+(\S+))/g);
124
+ for (const m of pyImports)
125
+ imports.push(m[1] ?? m[2]);
126
+ }
127
+ else if (language === 'go') {
128
+ const goImports = content.matchAll(/import\s+(?:\(\s*)?["']([^"']+)["']/g);
129
+ for (const m of goImports)
130
+ imports.push(m[1]);
131
+ }
132
+ else if (language === 'java') {
133
+ const javaImports = content.matchAll(/import\s+([\w.]+);/g);
134
+ for (const m of javaImports)
135
+ imports.push(m[1]);
136
+ }
137
+ return [...new Set(imports)];
138
+ }
139
+ //# sourceMappingURL=file.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file.js","sourceRoot":"","sources":["../../../src/context/file.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { ProjectContext } from '../types/analysis.js';
2
+ export declare function buildProjectContext(projectRoot: string): ProjectContext;
3
+ export declare function formatProjectContextForLLM(ctx: ProjectContext): string;
4
+ //# sourceMappingURL=project.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"project.d.ts","sourceRoot":"","sources":["../../../src/context/project.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AAM3D,wBAAgB,mBAAmB,CAAC,WAAW,EAAE,MAAM,GAAG,cAAc,CAcvE;AAED,wBAAgB,0BAA0B,CAAC,GAAG,EAAE,cAAc,GAAG,MAAM,CAmCtE"}
@@ -0,0 +1,252 @@
1
+ import * as fs from 'node:fs';
2
+ import * as path from 'node:path';
3
+ const README_MAX_CHARS = 2000;
4
+ const TREE_MAX_DEPTH = 2;
5
+ const LANGUAGE_CENSUS_MAX_FILES = 200;
6
+ export function buildProjectContext(projectRoot) {
7
+ return {
8
+ readme: readReadme(projectRoot),
9
+ packageMeta: readPackageMeta(projectRoot),
10
+ directoryTree: buildDirectoryTree(projectRoot, TREE_MAX_DEPTH),
11
+ envVars: readEnvVarNames(projectRoot),
12
+ hasDockerfile: fileExists(projectRoot, 'Dockerfile'),
13
+ hasCI: dirExists(projectRoot, '.github/workflows') ||
14
+ fileExists(projectRoot, '.gitlab-ci.yml') ||
15
+ fileExists(projectRoot, 'Jenkinsfile'),
16
+ language: detectLanguage(projectRoot),
17
+ framework: detectFramework(projectRoot),
18
+ };
19
+ }
20
+ export function formatProjectContextForLLM(ctx) {
21
+ const sections = [];
22
+ if (ctx.readme) {
23
+ sections.push(`## Project README\n${ctx.readme}`);
24
+ }
25
+ if (ctx.packageMeta) {
26
+ const meta = ctx.packageMeta;
27
+ const parts = [];
28
+ if (meta.name)
29
+ parts.push(`Name: ${meta.name}`);
30
+ if (meta.description)
31
+ parts.push(`Description: ${meta.description}`);
32
+ if (meta.dependencies) {
33
+ const deps = Object.keys(meta.dependencies);
34
+ parts.push(`Dependencies: ${deps.join(', ')}`);
35
+ }
36
+ sections.push(`## Package Metadata\n${parts.join('\n')}`);
37
+ }
38
+ sections.push(`## Directory Structure\n\`\`\`\n${ctx.directoryTree}\n\`\`\``);
39
+ if (ctx.envVars.length > 0) {
40
+ sections.push(`## Environment Variables\n${ctx.envVars.join(', ')}`);
41
+ }
42
+ const infra = [];
43
+ if (ctx.hasDockerfile)
44
+ infra.push('Dockerfile');
45
+ if (ctx.hasCI)
46
+ infra.push('CI/CD');
47
+ if (infra.length > 0) {
48
+ sections.push(`## Infrastructure\n${infra.join(', ')}`);
49
+ }
50
+ sections.push(`## Language: ${ctx.language}\n## Framework: ${ctx.framework}`);
51
+ return sections.join('\n\n');
52
+ }
53
+ function readReadme(root) {
54
+ for (const name of ['README.md', 'README.txt', 'README', 'readme.md']) {
55
+ const filePath = path.join(root, name);
56
+ try {
57
+ const content = fs.readFileSync(filePath, 'utf-8');
58
+ return content.slice(0, README_MAX_CHARS);
59
+ }
60
+ catch {
61
+ continue;
62
+ }
63
+ }
64
+ return '';
65
+ }
66
+ function readPackageMeta(root) {
67
+ for (const name of ['package.json', 'pyproject.toml', 'go.mod', 'Cargo.toml']) {
68
+ const filePath = path.join(root, name);
69
+ try {
70
+ const content = fs.readFileSync(filePath, 'utf-8');
71
+ if (name === 'package.json') {
72
+ return JSON.parse(content);
73
+ }
74
+ return { _raw: content.slice(0, 500), _type: name };
75
+ }
76
+ catch {
77
+ continue;
78
+ }
79
+ }
80
+ return null;
81
+ }
82
+ function readEnvVarNames(root) {
83
+ const envPath = path.join(root, '.env.example');
84
+ try {
85
+ const content = fs.readFileSync(envPath, 'utf-8');
86
+ return content
87
+ .split('\n')
88
+ .map((line) => line.trim())
89
+ .filter((line) => line && !line.startsWith('#'))
90
+ .map((line) => line.split('=')[0].trim())
91
+ .filter(Boolean);
92
+ }
93
+ catch {
94
+ return [];
95
+ }
96
+ }
97
+ function buildDirectoryTree(root, maxDepth, prefix = '', depth = 0) {
98
+ if (depth >= maxDepth)
99
+ return '';
100
+ const EXCLUDE = new Set([
101
+ 'node_modules', 'dist', 'build', '.git', 'vendor', '__pycache__',
102
+ '.venv', '.next', 'coverage', '.nyc_output',
103
+ ]);
104
+ let entries;
105
+ try {
106
+ entries = fs.readdirSync(root, { withFileTypes: true });
107
+ }
108
+ catch {
109
+ return '';
110
+ }
111
+ const filtered = entries
112
+ .filter((e) => !EXCLUDE.has(e.name) && !e.name.startsWith('.'))
113
+ .sort((a, b) => {
114
+ if (a.isDirectory() && !b.isDirectory())
115
+ return -1;
116
+ if (!a.isDirectory() && b.isDirectory())
117
+ return 1;
118
+ return a.name.localeCompare(b.name);
119
+ });
120
+ const lines = [];
121
+ for (let i = 0; i < filtered.length; i++) {
122
+ const entry = filtered[i];
123
+ const isLast = i === filtered.length - 1;
124
+ const connector = isLast ? '└── ' : '├── ';
125
+ const childPrefix = isLast ? ' ' : '│ ';
126
+ lines.push(`${prefix}${connector}${entry.name}${entry.isDirectory() ? '/' : ''}`);
127
+ if (entry.isDirectory()) {
128
+ const subtree = buildDirectoryTree(path.join(root, entry.name), maxDepth, `${prefix}${childPrefix}`, depth + 1);
129
+ if (subtree)
130
+ lines.push(subtree);
131
+ }
132
+ }
133
+ return lines.join('\n');
134
+ }
135
+ function detectLanguage(root) {
136
+ // Check manifest files first
137
+ if (fileExists(root, 'package.json') || fileExists(root, 'tsconfig.json'))
138
+ return 'javascript/typescript';
139
+ if (fileExists(root, 'requirements.txt') || fileExists(root, 'setup.py') || fileExists(root, 'pyproject.toml'))
140
+ return 'python';
141
+ if (fileExists(root, 'go.mod'))
142
+ return 'go';
143
+ if (fileExists(root, 'Cargo.toml'))
144
+ return 'rust';
145
+ if (fileExists(root, 'pom.xml') || fileExists(root, 'build.gradle'))
146
+ return 'java';
147
+ // Fallback: recursive census of file extensions across the project
148
+ try {
149
+ const extCounts = {};
150
+ let countedFiles = 0;
151
+ const exclude = new Set([
152
+ 'node_modules', 'dist', 'build', '.git', 'vendor', '__pycache__',
153
+ '.venv', 'venv', 'env', '.next', 'coverage', '.nyc_output',
154
+ ]);
155
+ const walk = (dir) => {
156
+ if (countedFiles >= LANGUAGE_CENSUS_MAX_FILES)
157
+ return;
158
+ let entries;
159
+ try {
160
+ entries = fs.readdirSync(dir, { withFileTypes: true });
161
+ }
162
+ catch {
163
+ return;
164
+ }
165
+ for (const entry of entries) {
166
+ if (countedFiles >= LANGUAGE_CENSUS_MAX_FILES)
167
+ return;
168
+ if (exclude.has(entry.name) || entry.name.startsWith('.'))
169
+ continue;
170
+ const fullPath = path.join(dir, entry.name);
171
+ if (entry.isDirectory()) {
172
+ walk(fullPath);
173
+ continue;
174
+ }
175
+ if (!entry.isFile())
176
+ continue;
177
+ const ext = path.extname(entry.name);
178
+ if (!ext)
179
+ continue;
180
+ extCounts[ext] = (extCounts[ext] ?? 0) + 1;
181
+ countedFiles++;
182
+ }
183
+ };
184
+ walk(root);
185
+ if (countedFiles === 0) {
186
+ return 'unknown';
187
+ }
188
+ const jsExts = ['.js', '.mjs', '.cjs', '.jsx', '.ts', '.tsx'];
189
+ const pyExts = ['.py'];
190
+ const jsCount = jsExts.reduce((n, e) => n + (extCounts[e] ?? 0), 0);
191
+ const pyCount = pyExts.reduce((n, e) => n + (extCounts[e] ?? 0), 0);
192
+ if (jsCount > 0 && jsCount >= pyCount)
193
+ return 'javascript/typescript';
194
+ if (pyCount > 0)
195
+ return 'python';
196
+ if (extCounts['.go'])
197
+ return 'go';
198
+ if (extCounts['.rs'])
199
+ return 'rust';
200
+ if (extCounts['.java'])
201
+ return 'java';
202
+ }
203
+ catch { /* can't read directory */ }
204
+ return 'unknown';
205
+ }
206
+ function detectFramework(root) {
207
+ try {
208
+ const pkg = JSON.parse(fs.readFileSync(path.join(root, 'package.json'), 'utf-8'));
209
+ const deps = { ...pkg.dependencies, ...pkg.devDependencies };
210
+ if (deps.next)
211
+ return 'next.js';
212
+ if (deps.express)
213
+ return 'express';
214
+ if (deps.fastify)
215
+ return 'fastify';
216
+ if (deps.react)
217
+ return 'react';
218
+ if (deps.vue)
219
+ return 'vue';
220
+ if (deps.hono)
221
+ return 'hono';
222
+ }
223
+ catch { /* no package.json */ }
224
+ try {
225
+ const req = fs.readFileSync(path.join(root, 'requirements.txt'), 'utf-8');
226
+ if (req.includes('django'))
227
+ return 'django';
228
+ if (req.includes('flask'))
229
+ return 'flask';
230
+ if (req.includes('fastapi'))
231
+ return 'fastapi';
232
+ }
233
+ catch { /* no requirements.txt */ }
234
+ return 'none';
235
+ }
236
+ function fileExists(root, name) {
237
+ try {
238
+ return fs.statSync(path.join(root, name)).isFile();
239
+ }
240
+ catch {
241
+ return false;
242
+ }
243
+ }
244
+ function dirExists(root, name) {
245
+ try {
246
+ return fs.statSync(path.join(root, name)).isDirectory();
247
+ }
248
+ catch {
249
+ return false;
250
+ }
251
+ }
252
+ //# sourceMappingURL=project.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"project.js","sourceRoot":"","sources":["../../../src/context/project.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,SAAS,CAAC;AAC9B,OAAO,KAAK,IAAI,MAAM,WAAW,CAAC;AAGlC,MAAM,gBAAgB,GAAG,IAAI,CAAC;AAC9B,MAAM,cAAc,GAAG,CAAC,CAAC;AACzB,MAAM,yBAAyB,GAAG,GAAG,CAAC;AAEtC,MAAM,UAAU,mBAAmB,CAAC,WAAmB;IACrD,OAAO;QACL,MAAM,EAAE,UAAU,CAAC,WAAW,CAAC;QAC/B,WAAW,EAAE,eAAe,CAAC,WAAW,CAAC;QACzC,aAAa,EAAE,kBAAkB,CAAC,WAAW,EAAE,cAAc,CAAC;QAC9D,OAAO,EAAE,eAAe,CAAC,WAAW,CAAC;QACrC,aAAa,EAAE,UAAU,CAAC,WAAW,EAAE,YAAY,CAAC;QACpD,KAAK,EACH,SAAS,CAAC,WAAW,EAAE,mBAAmB,CAAC;YAC3C,UAAU,CAAC,WAAW,EAAE,gBAAgB,CAAC;YACzC,UAAU,CAAC,WAAW,EAAE,aAAa,CAAC;QACxC,QAAQ,EAAE,cAAc,CAAC,WAAW,CAAC;QACrC,SAAS,EAAE,eAAe,CAAC,WAAW,CAAC;KACxC,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,0BAA0B,CAAC,GAAmB;IAC5D,MAAM,QAAQ,GAAa,EAAE,CAAC;IAE9B,IAAI,GAAG,CAAC,MAAM,EAAE,CAAC;QACf,QAAQ,CAAC,IAAI,CAAC,sBAAsB,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;IACpD,CAAC;IAED,IAAI,GAAG,CAAC,WAAW,EAAE,CAAC;QACpB,MAAM,IAAI,GAAG,GAAG,CAAC,WAAW,CAAC;QAC7B,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,IAAI,IAAI,CAAC,IAAI;YAAE,KAAK,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QAChD,IAAI,IAAI,CAAC,WAAW;YAAE,KAAK,CAAC,IAAI,CAAC,gBAAgB,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;QACrE,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;YACtB,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,YAAsC,CAAC,CAAC;YACtE,KAAK,CAAC,IAAI,CAAC,iBAAiB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjD,CAAC;QACD,QAAQ,CAAC,IAAI,CAAC,wBAAwB,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAC5D,CAAC;IAED,QAAQ,CAAC,IAAI,CAAC,mCAAmC,GAAG,CAAC,aAAa,UAAU,CAAC,CAAC;IAE9E,IAAI,GAAG,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC3B,QAAQ,CAAC,IAAI,CAAC,6BAA6B,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACvE,CAAC;IAED,MAAM,KAAK,GAAa,EAAE,CAAC;IAC3B,IAAI,GAAG,CAAC,aAAa;QAAE,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAChD,IAAI,GAAG,CAAC,KAAK;QAAE,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACnC,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACrB,QAAQ,CAAC,IAAI,CAAC,sBAAsB,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAC1D,CAAC;IAED,QAAQ,CAAC,IAAI,CAAC,gBAAgB,GAAG,CAAC,QAAQ,mBAAmB,GAAG,CAAC,SAAS,EAAE,CAAC,CAAC;IAE9E,OAAO,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;AAC/B,CAAC;AAED,SAAS,UAAU,CAAC,IAAY;IAC9B,KAAK,MAAM,IAAI,IAAI,CAAC,WAAW,EAAE,YAAY,EAAE,QAAQ,EAAE,WAAW,CAAC,EAAE,CAAC;QACtE,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QACvC,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YACnD,OAAO,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,gBAAgB,CAAC,CAAC;QAC5C,CAAC;QAAC,MAAM,CAAC;YACP,SAAS;QACX,CAAC;IACH,CAAC;IACD,OAAO,EAAE,CAAC;AACZ,CAAC;AAED,SAAS,eAAe,CAAC,IAAY;IACnC,KAAK,MAAM,IAAI,IAAI,CAAC,cAAc,EAAE,gBAAgB,EAAE,QAAQ,EAAE,YAAY,CAAC,EAAE,CAAC;QAC9E,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QACvC,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YACnD,IAAI,IAAI,KAAK,cAAc,EAAE,CAAC;gBAC5B,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YAC7B,CAAC;YACD,OAAO,EAAE,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;QACtD,CAAC;QAAC,MAAM,CAAC;YACP,SAAS;QACX,CAAC;IACH,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED,SAAS,eAAe,CAAC,IAAY;IACnC,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;IAChD,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAClD,OAAO,OAAO;aACX,KAAK,CAAC,IAAI,CAAC;aACX,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;aAC1B,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;aAC/C,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;aACxC,MAAM,CAAC,OAAO,CAAC,CAAC;IACrB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC;AAED,SAAS,kBAAkB,CAAC,IAAY,EAAE,QAAgB,EAAE,MAAM,GAAG,EAAE,EAAE,KAAK,GAAG,CAAC;IAChF,IAAI,KAAK,IAAI,QAAQ;QAAE,OAAO,EAAE,CAAC;IAEjC,MAAM,OAAO,GAAG,IAAI,GAAG,CAAC;QACtB,cAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,aAAa;QAChE,OAAO,EAAE,OAAO,EAAE,UAAU,EAAE,aAAa;KAC5C,CAAC,CAAC;IAEH,IAAI,OAAoB,CAAC;IACzB,IAAI,CAAC;QACH,OAAO,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;IAC1D,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,MAAM,QAAQ,GAAG,OAAO;SACrB,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;SAC9D,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;QACb,IAAI,CAAC,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,CAAC,WAAW,EAAE;YAAE,OAAO,CAAC,CAAC,CAAC;QACnD,IAAI,CAAC,CAAC,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,WAAW,EAAE;YAAE,OAAO,CAAC,CAAC;QAClD,OAAO,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;IACtC,CAAC,CAAC,CAAC;IAEL,MAAM,KAAK,GAAa,EAAE,CAAC;IAC3B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,QAAQ,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACzC,MAAM,KAAK,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC;QAC1B,MAAM,MAAM,GAAG,CAAC,KAAK,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC;QACzC,MAAM,SAAS,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC;QAC3C,MAAM,WAAW,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC;QAE7C,KAAK,CAAC,IAAI,CAAC,GAAG,MAAM,GAAG,SAAS,GAAG,KAAK,CAAC,IAAI,GAAG,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAElF,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC;YACxB,MAAM,OAAO,GAAG,kBAAkB,CAChC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,CAAC,EAC3B,QAAQ,EACR,GAAG,MAAM,GAAG,WAAW,EAAE,EACzB,KAAK,GAAG,CAAC,CACV,CAAC;YACF,IAAI,OAAO;gBAAE,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACnC,CAAC;IACH,CAAC;IAED,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AAC1B,CAAC;AAED,SAAS,cAAc,CAAC,IAAY;IAClC,6BAA6B;IAC7B,IAAI,UAAU,CAAC,IAAI,EAAE,cAAc,CAAC,IAAI,UAAU,CAAC,IAAI,EAAE,eAAe,CAAC;QAAE,OAAO,uBAAuB,CAAC;IAC1G,IAAI,UAAU,CAAC,IAAI,EAAE,kBAAkB,CAAC,IAAI,UAAU,CAAC,IAAI,EAAE,UAAU,CAAC,IAAI,UAAU,CAAC,IAAI,EAAE,gBAAgB,CAAC;QAAE,OAAO,QAAQ,CAAC;IAChI,IAAI,UAAU,CAAC,IAAI,EAAE,QAAQ,CAAC;QAAE,OAAO,IAAI,CAAC;IAC5C,IAAI,UAAU,CAAC,IAAI,EAAE,YAAY,CAAC;QAAE,OAAO,MAAM,CAAC;IAClD,IAAI,UAAU,CAAC,IAAI,EAAE,SAAS,CAAC,IAAI,UAAU,CAAC,IAAI,EAAE,cAAc,CAAC;QAAE,OAAO,MAAM,CAAC;IAEnF,mEAAmE;IACnE,IAAI,CAAC;QACH,MAAM,SAAS,GAA2B,EAAE,CAAC;QAC7C,IAAI,YAAY,GAAG,CAAC,CAAC;QACrB,MAAM,OAAO,GAAG,IAAI,GAAG,CAAC;YACtB,cAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,aAAa;YAChE,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,UAAU,EAAE,aAAa;SAC3D,CAAC,CAAC;QAEH,MAAM,IAAI,GAAG,CAAC,GAAW,EAAQ,EAAE;YACjC,IAAI,YAAY,IAAI,yBAAyB;gBAAE,OAAO;YAEtD,IAAI,OAAoB,CAAC;YACzB,IAAI,CAAC;gBACH,OAAO,GAAG,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;YACzD,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO;YACT,CAAC;YAED,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;gBAC5B,IAAI,YAAY,IAAI,yBAAyB;oBAAE,OAAO;gBACtD,IAAI,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;oBAAE,SAAS;gBAEpE,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;gBAC5C,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC;oBACxB,IAAI,CAAC,QAAQ,CAAC,CAAC;oBACf,SAAS;gBACX,CAAC;gBAED,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE;oBAAE,SAAS;gBAC9B,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;gBACrC,IAAI,CAAC,GAAG;oBAAE,SAAS;gBACnB,SAAS,CAAC,GAAG,CAAC,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;gBAC3C,YAAY,EAAE,CAAC;YACjB,CAAC;QACH,CAAC,CAAC;QAEF,IAAI,CAAC,IAAI,CAAC,CAAC;QAEX,IAAI,YAAY,KAAK,CAAC,EAAE,CAAC;YACvB,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,MAAM,MAAM,GAAG,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;QAC9D,MAAM,MAAM,GAAG,CAAC,KAAK,CAAC,CAAC;QACvB,MAAM,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACpE,MAAM,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACpE,IAAI,OAAO,GAAG,CAAC,IAAI,OAAO,IAAI,OAAO;YAAE,OAAO,uBAAuB,CAAC;QACtE,IAAI,OAAO,GAAG,CAAC;YAAE,OAAO,QAAQ,CAAC;QACjC,IAAI,SAAS,CAAC,KAAK,CAAC;YAAE,OAAO,IAAI,CAAC;QAClC,IAAI,SAAS,CAAC,KAAK,CAAC;YAAE,OAAO,MAAM,CAAC;QACpC,IAAI,SAAS,CAAC,OAAO,CAAC;YAAE,OAAO,MAAM,CAAC;IACxC,CAAC;IAAC,MAAM,CAAC,CAAC,0BAA0B,CAAC,CAAC;IAEtC,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,SAAS,eAAe,CAAC,IAAY;IACnC,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,cAAc,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;QAClF,MAAM,IAAI,GAAG,EAAE,GAAG,GAAG,CAAC,YAAY,EAAE,GAAG,GAAG,CAAC,eAAe,EAAE,CAAC;QAC7D,IAAI,IAAI,CAAC,IAAI;YAAE,OAAO,SAAS,CAAC;QAChC,IAAI,IAAI,CAAC,OAAO;YAAE,OAAO,SAAS,CAAC;QACnC,IAAI,IAAI,CAAC,OAAO;YAAE,OAAO,SAAS,CAAC;QACnC,IAAI,IAAI,CAAC,KAAK;YAAE,OAAO,OAAO,CAAC;QAC/B,IAAI,IAAI,CAAC,GAAG;YAAE,OAAO,KAAK,CAAC;QAC3B,IAAI,IAAI,CAAC,IAAI;YAAE,OAAO,MAAM,CAAC;IAC/B,CAAC;IAAC,MAAM,CAAC,CAAC,qBAAqB,CAAC,CAAC;IAEjC,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,kBAAkB,CAAC,EAAE,OAAO,CAAC,CAAC;QAC1E,IAAI,GAAG,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAAE,OAAO,QAAQ,CAAC;QAC5C,IAAI,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC;YAAE,OAAO,OAAO,CAAC;QAC1C,IAAI,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC;YAAE,OAAO,SAAS,CAAC;IAChD,CAAC;IAAC,MAAM,CAAC,CAAC,yBAAyB,CAAC,CAAC;IAErC,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,SAAS,UAAU,CAAC,IAAY,EAAE,IAAY;IAC5C,IAAI,CAAC;QACH,OAAO,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;IACrD,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAED,SAAS,SAAS,CAAC,IAAY,EAAE,IAAY;IAC3C,IAAI,CAAC;QACH,OAAO,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;IAC1D,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC"}
@@ -0,0 +1,11 @@
1
+ import type { DependencyGraph } from '../types/analysis.js';
2
+ export declare class DependencyGraphBuilder {
3
+ private nodes;
4
+ private visited;
5
+ private projectRoot;
6
+ constructor(projectRoot: string);
7
+ build(entryFiles: string[]): DependencyGraph;
8
+ getImporters(file: string): string[];
9
+ getImportees(file: string): string[];
10
+ }
11
+ //# sourceMappingURL=dependency.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dependency.d.ts","sourceRoot":"","sources":["../../../src/graph/dependency.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,eAAe,EAAkB,MAAM,sBAAsB,CAAC;AAwB5E,qBAAa,sBAAsB;IACjC,OAAO,CAAC,KAAK,CAAqC;IAClD,OAAO,CAAC,OAAO,CAAqB;IACpC,OAAO,CAAC,WAAW,CAAS;gBAEhB,WAAW,EAAE,MAAM;IAI/B,KAAK,CAAC,UAAU,EAAE,MAAM,EAAE,GAAG,eAAe;IAyE5C,YAAY,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,EAAE;IAIpC,YAAY,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,EAAE;CAGrC"}