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.
- package/README.md +144 -43
- package/code-review-agent/.env.example +8 -0
- package/code-review-agent/README.md +142 -0
- package/code-review-agent/TODO.md +149 -0
- package/code-review-agent/bin/cr-agent.ts +313 -0
- package/code-review-agent/dist/bin/cr-agent.d.ts +3 -0
- package/code-review-agent/dist/bin/cr-agent.d.ts.map +1 -0
- package/code-review-agent/dist/bin/cr-agent.js +299 -0
- package/code-review-agent/dist/bin/cr-agent.js.map +1 -0
- package/code-review-agent/dist/src/analyzer/engine.d.ts +16 -0
- package/code-review-agent/dist/src/analyzer/engine.d.ts.map +1 -0
- package/code-review-agent/dist/src/analyzer/engine.js +298 -0
- package/code-review-agent/dist/src/analyzer/engine.js.map +1 -0
- package/code-review-agent/dist/src/analyzer/intent.d.ts +10 -0
- package/code-review-agent/dist/src/analyzer/intent.d.ts.map +1 -0
- package/code-review-agent/dist/src/analyzer/intent.js +40 -0
- package/code-review-agent/dist/src/analyzer/intent.js.map +1 -0
- package/code-review-agent/dist/src/analyzer/semantic.d.ts +19 -0
- package/code-review-agent/dist/src/analyzer/semantic.d.ts.map +1 -0
- package/code-review-agent/dist/src/analyzer/semantic.js +150 -0
- package/code-review-agent/dist/src/analyzer/semantic.js.map +1 -0
- package/code-review-agent/dist/src/context/assembler.d.ts +16 -0
- package/code-review-agent/dist/src/context/assembler.d.ts.map +1 -0
- package/code-review-agent/dist/src/context/assembler.js +135 -0
- package/code-review-agent/dist/src/context/assembler.js.map +1 -0
- package/code-review-agent/dist/src/context/file.d.ts +6 -0
- package/code-review-agent/dist/src/context/file.d.ts.map +1 -0
- package/code-review-agent/dist/src/context/file.js +139 -0
- package/code-review-agent/dist/src/context/file.js.map +1 -0
- package/code-review-agent/dist/src/context/project.d.ts +4 -0
- package/code-review-agent/dist/src/context/project.d.ts.map +1 -0
- package/code-review-agent/dist/src/context/project.js +252 -0
- package/code-review-agent/dist/src/context/project.js.map +1 -0
- package/code-review-agent/dist/src/graph/dependency.d.ts +11 -0
- package/code-review-agent/dist/src/graph/dependency.d.ts.map +1 -0
- package/code-review-agent/dist/src/graph/dependency.js +102 -0
- package/code-review-agent/dist/src/graph/dependency.js.map +1 -0
- package/code-review-agent/dist/src/graph/resolver.d.ts +9 -0
- package/code-review-agent/dist/src/graph/resolver.d.ts.map +1 -0
- package/code-review-agent/dist/src/graph/resolver.js +124 -0
- package/code-review-agent/dist/src/graph/resolver.js.map +1 -0
- package/code-review-agent/dist/src/index.d.ts +21 -0
- package/code-review-agent/dist/src/index.d.ts.map +1 -0
- package/code-review-agent/dist/src/index.js +21 -0
- package/code-review-agent/dist/src/index.js.map +1 -0
- package/code-review-agent/dist/src/llm/anthropic.d.ts +13 -0
- package/code-review-agent/dist/src/llm/anthropic.d.ts.map +1 -0
- package/code-review-agent/dist/src/llm/anthropic.js +83 -0
- package/code-review-agent/dist/src/llm/anthropic.js.map +1 -0
- package/code-review-agent/dist/src/llm/claude-cli.d.ts +13 -0
- package/code-review-agent/dist/src/llm/claude-cli.d.ts.map +1 -0
- package/code-review-agent/dist/src/llm/claude-cli.js +142 -0
- package/code-review-agent/dist/src/llm/claude-cli.js.map +1 -0
- package/code-review-agent/dist/src/llm/openai.d.ts +13 -0
- package/code-review-agent/dist/src/llm/openai.d.ts.map +1 -0
- package/code-review-agent/dist/src/llm/openai.js +78 -0
- package/code-review-agent/dist/src/llm/openai.js.map +1 -0
- package/code-review-agent/dist/src/llm/provider.d.ts +18 -0
- package/code-review-agent/dist/src/llm/provider.d.ts.map +1 -0
- package/code-review-agent/dist/src/llm/provider.js +11 -0
- package/code-review-agent/dist/src/llm/provider.js.map +1 -0
- package/code-review-agent/dist/src/llm/router.d.ts +14 -0
- package/code-review-agent/dist/src/llm/router.d.ts.map +1 -0
- package/code-review-agent/dist/src/llm/router.js +67 -0
- package/code-review-agent/dist/src/llm/router.js.map +1 -0
- package/code-review-agent/dist/src/llm/schemas.d.ts +18 -0
- package/code-review-agent/dist/src/llm/schemas.d.ts.map +1 -0
- package/code-review-agent/dist/src/llm/schemas.js +91 -0
- package/code-review-agent/dist/src/llm/schemas.js.map +1 -0
- package/code-review-agent/dist/src/types/analysis.d.ts +56 -0
- package/code-review-agent/dist/src/types/analysis.d.ts.map +1 -0
- package/code-review-agent/dist/src/types/analysis.js +2 -0
- package/code-review-agent/dist/src/types/analysis.js.map +1 -0
- package/code-review-agent/dist/src/types/config.d.ts +24 -0
- package/code-review-agent/dist/src/types/config.d.ts.map +1 -0
- package/code-review-agent/dist/src/types/config.js +42 -0
- package/code-review-agent/dist/src/types/config.js.map +1 -0
- package/code-review-agent/dist/src/types/findings.d.ts +236 -0
- package/code-review-agent/dist/src/types/findings.d.ts.map +1 -0
- package/code-review-agent/dist/src/types/findings.js +64 -0
- package/code-review-agent/dist/src/types/findings.js.map +1 -0
- package/code-review-agent/package.json +36 -0
- package/code-review-agent/src/analyzer/engine.ts +374 -0
- package/code-review-agent/src/analyzer/intent.ts +49 -0
- package/code-review-agent/src/analyzer/semantic.ts +222 -0
- package/code-review-agent/src/context/assembler.ts +165 -0
- package/code-review-agent/src/context/file.ts +145 -0
- package/code-review-agent/src/context/project.ts +253 -0
- package/code-review-agent/src/graph/dependency.ts +116 -0
- package/code-review-agent/src/graph/resolver.ts +138 -0
- package/code-review-agent/src/index.ts +58 -0
- package/code-review-agent/src/llm/anthropic.ts +106 -0
- package/code-review-agent/src/llm/claude-cli.ts +188 -0
- package/code-review-agent/src/llm/openai.ts +95 -0
- package/code-review-agent/src/llm/provider.ts +33 -0
- package/code-review-agent/src/llm/router.ts +86 -0
- package/code-review-agent/src/llm/schemas.ts +125 -0
- package/code-review-agent/src/types/analysis.ts +62 -0
- package/code-review-agent/src/types/config.ts +72 -0
- package/code-review-agent/src/types/findings.ts +81 -0
- package/code-review-agent/tests/analyzer/engine.test.ts +194 -0
- package/code-review-agent/tests/analyzer/intent.test.ts +76 -0
- package/code-review-agent/tests/analyzer/semantic.test.ts +131 -0
- package/code-review-agent/tests/context/file.test.ts +21 -0
- package/code-review-agent/tests/context/project.test.ts +20 -0
- package/code-review-agent/tests/fixtures/safe-build-tool/README.md +19 -0
- package/code-review-agent/tests/fixtures/safe-build-tool/builder.js +52 -0
- package/code-review-agent/tests/fixtures/safe-file-manager/README.md +16 -0
- package/code-review-agent/tests/fixtures/safe-file-manager/organizer.py +70 -0
- package/code-review-agent/tests/fixtures/vuln-api-server/README.md +17 -0
- package/code-review-agent/tests/fixtures/vuln-api-server/server.js +52 -0
- package/code-review-agent/tests/fixtures/vuln-ecommerce/README.md +18 -0
- package/code-review-agent/tests/fixtures/vuln-ecommerce/checkout.js +63 -0
- package/code-review-agent/tests/graph/dependency.test.ts +136 -0
- package/code-review-agent/tests/helpers/mock-provider.ts +48 -0
- package/code-review-agent/tests/llm/claude-cli.test.ts +251 -0
- package/code-review-agent/tests/llm/router.test.ts +77 -0
- package/code-review-agent/tests/llm/schemas.test.ts +142 -0
- package/code-review-agent/tsconfig.json +20 -0
- package/code-review-agent/vitest.config.ts +11 -0
- package/index.js +18 -18
- package/openclaw.plugin.json +2 -2
- package/package.json +13 -3
- package/server.json +3 -3
- package/src/cli/init-hooks.js +3 -3
- package/src/cli/init.js +1 -1
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import * as fs from 'node:fs';
|
|
2
|
+
import * as path from 'node:path';
|
|
3
|
+
import { extractImports, resolveImportPath } from './resolver.js';
|
|
4
|
+
const MAX_DEPTH = 4;
|
|
5
|
+
const MAX_FILES = 200;
|
|
6
|
+
const LANGUAGE_MAP = {
|
|
7
|
+
'.js': 'javascript', '.mjs': 'javascript', '.cjs': 'javascript', '.jsx': 'javascript',
|
|
8
|
+
'.ts': 'typescript', '.tsx': 'typescript',
|
|
9
|
+
'.py': 'python',
|
|
10
|
+
'.go': 'go',
|
|
11
|
+
'.rs': 'rust',
|
|
12
|
+
'.java': 'java',
|
|
13
|
+
'.rb': 'ruby',
|
|
14
|
+
'.php': 'php',
|
|
15
|
+
'.c': 'c',
|
|
16
|
+
'.cpp': 'cpp',
|
|
17
|
+
'.h': 'c',
|
|
18
|
+
'.hpp': 'cpp',
|
|
19
|
+
'.cs': 'csharp',
|
|
20
|
+
'.swift': 'swift',
|
|
21
|
+
'.kt': 'kotlin',
|
|
22
|
+
};
|
|
23
|
+
export class DependencyGraphBuilder {
|
|
24
|
+
nodes = new Map();
|
|
25
|
+
visited = new Set();
|
|
26
|
+
projectRoot;
|
|
27
|
+
constructor(projectRoot) {
|
|
28
|
+
this.projectRoot = projectRoot;
|
|
29
|
+
}
|
|
30
|
+
build(entryFiles) {
|
|
31
|
+
const queue = entryFiles.map((f) => ({
|
|
32
|
+
file: path.resolve(this.projectRoot, f),
|
|
33
|
+
depth: 0,
|
|
34
|
+
}));
|
|
35
|
+
while (queue.length > 0 && this.nodes.size < MAX_FILES) {
|
|
36
|
+
const item = queue.shift();
|
|
37
|
+
const { file, depth } = item;
|
|
38
|
+
if (this.visited.has(file) || depth > MAX_DEPTH)
|
|
39
|
+
continue;
|
|
40
|
+
this.visited.add(file);
|
|
41
|
+
const relPath = path.relative(this.projectRoot, file);
|
|
42
|
+
const ext = path.extname(file);
|
|
43
|
+
const language = LANGUAGE_MAP[ext];
|
|
44
|
+
if (!language)
|
|
45
|
+
continue;
|
|
46
|
+
let content;
|
|
47
|
+
try {
|
|
48
|
+
content = fs.readFileSync(file, 'utf-8');
|
|
49
|
+
}
|
|
50
|
+
catch {
|
|
51
|
+
continue;
|
|
52
|
+
}
|
|
53
|
+
const imports = extractImports(content, language);
|
|
54
|
+
const resolvedImports = [];
|
|
55
|
+
for (const imp of imports) {
|
|
56
|
+
if (!imp.isLocal)
|
|
57
|
+
continue;
|
|
58
|
+
const resolved = resolveImportPath(imp.specifier, file, language);
|
|
59
|
+
if (resolved) {
|
|
60
|
+
const resolvedRel = path.relative(this.projectRoot, resolved);
|
|
61
|
+
resolvedImports.push(resolvedRel);
|
|
62
|
+
// Ensure the target node exists
|
|
63
|
+
if (!this.nodes.has(resolvedRel)) {
|
|
64
|
+
this.nodes.set(resolvedRel, {
|
|
65
|
+
file: resolvedRel,
|
|
66
|
+
imports: [],
|
|
67
|
+
importedBy: [],
|
|
68
|
+
});
|
|
69
|
+
}
|
|
70
|
+
// Add reverse edge
|
|
71
|
+
this.nodes.get(resolvedRel).importedBy.push(relPath);
|
|
72
|
+
// Queue for traversal
|
|
73
|
+
if (!this.visited.has(resolved)) {
|
|
74
|
+
queue.push({ file: resolved, depth: depth + 1 });
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
const existing = this.nodes.get(relPath);
|
|
79
|
+
if (existing) {
|
|
80
|
+
existing.imports = resolvedImports;
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
this.nodes.set(relPath, {
|
|
84
|
+
file: relPath,
|
|
85
|
+
imports: resolvedImports,
|
|
86
|
+
importedBy: [],
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
return {
|
|
91
|
+
nodes: this.nodes,
|
|
92
|
+
entryPoints: entryFiles.map((f) => path.relative(this.projectRoot, path.resolve(this.projectRoot, f))),
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
getImporters(file) {
|
|
96
|
+
return this.nodes.get(file)?.importedBy ?? [];
|
|
97
|
+
}
|
|
98
|
+
getImportees(file) {
|
|
99
|
+
return this.nodes.get(file)?.imports ?? [];
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
//# sourceMappingURL=dependency.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"dependency.js","sourceRoot":"","sources":["../../../src/graph/dependency.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
export interface ImportInfo {
|
|
2
|
+
specifier: string;
|
|
3
|
+
isLocal: boolean;
|
|
4
|
+
resolved: string | null;
|
|
5
|
+
}
|
|
6
|
+
export declare function extractImports(content: string, language: string): ImportInfo[];
|
|
7
|
+
export declare function resolveImportPath(specifier: string, fromFile: string, language: string): string | null;
|
|
8
|
+
export declare function isLocalImport(specifier: string, language: string): boolean;
|
|
9
|
+
//# sourceMappingURL=resolver.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"resolver.d.ts","sourceRoot":"","sources":["../../../src/graph/resolver.ts"],"names":[],"mappings":"AAGA,MAAM,WAAW,UAAU;IACzB,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,OAAO,CAAC;IACjB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAC;CACzB;AAKD,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,UAAU,EAAE,CA6C9E;AAED,wBAAgB,iBAAiB,CAC/B,SAAS,EAAE,MAAM,EACjB,QAAQ,EAAE,MAAM,EAChB,QAAQ,EAAE,MAAM,GACf,MAAM,GAAG,IAAI,CA6Df;AAED,wBAAgB,aAAa,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAW1E"}
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
import * as fs from 'node:fs';
|
|
2
|
+
import * as path from 'node:path';
|
|
3
|
+
const JS_EXTENSIONS = ['.js', '.ts', '.jsx', '.tsx', '.mjs', '.cjs'];
|
|
4
|
+
const PY_EXTENSIONS = ['.py'];
|
|
5
|
+
export function extractImports(content, language) {
|
|
6
|
+
const imports = [];
|
|
7
|
+
if (['javascript', 'typescript'].includes(language)) {
|
|
8
|
+
// ES imports
|
|
9
|
+
for (const m of content.matchAll(/import\s+(?:.*?\s+from\s+)?['"]([^'"]+)['"]/g)) {
|
|
10
|
+
const spec = m[1];
|
|
11
|
+
imports.push({ specifier: spec, isLocal: isLocalImport(spec, language), resolved: null });
|
|
12
|
+
}
|
|
13
|
+
// Re-exports: export { ... } from '...', export * from '...', export { default } from '...'
|
|
14
|
+
for (const m of content.matchAll(/export\s+(?:\*|{[^}]*})\s+from\s+['"]([^'"]+)['"]/g)) {
|
|
15
|
+
const spec = m[1];
|
|
16
|
+
imports.push({ specifier: spec, isLocal: isLocalImport(spec, language), resolved: null });
|
|
17
|
+
}
|
|
18
|
+
// require
|
|
19
|
+
for (const m of content.matchAll(/require\s*\(\s*['"]([^'"]+)['"]\s*\)/g)) {
|
|
20
|
+
const spec = m[1];
|
|
21
|
+
imports.push({ specifier: spec, isLocal: isLocalImport(spec, language), resolved: null });
|
|
22
|
+
}
|
|
23
|
+
// dynamic import
|
|
24
|
+
for (const m of content.matchAll(/import\s*\(\s*['"]([^'"]+)['"]\s*\)/g)) {
|
|
25
|
+
const spec = m[1];
|
|
26
|
+
imports.push({ specifier: spec, isLocal: isLocalImport(spec, language), resolved: null });
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
else if (language === 'python') {
|
|
30
|
+
// from .module import ... (relative) and from package import ... (absolute)
|
|
31
|
+
for (const m of content.matchAll(/from\s+(\S+)\s+import/g)) {
|
|
32
|
+
const spec = m[1];
|
|
33
|
+
imports.push({ specifier: spec, isLocal: isLocalImport(spec, language), resolved: null });
|
|
34
|
+
}
|
|
35
|
+
// import module
|
|
36
|
+
for (const m of content.matchAll(/^import\s+(\S+)/gm)) {
|
|
37
|
+
const spec = m[1];
|
|
38
|
+
imports.push({ specifier: spec, isLocal: isLocalImport(spec, language), resolved: null });
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
else if (language === 'go') {
|
|
42
|
+
for (const m of content.matchAll(/["']([^"']+)["']/g)) {
|
|
43
|
+
const spec = m[1];
|
|
44
|
+
if (spec.includes('/')) {
|
|
45
|
+
imports.push({ specifier: spec, isLocal: isLocalImport(spec, language), resolved: null });
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
return imports;
|
|
50
|
+
}
|
|
51
|
+
export function resolveImportPath(specifier, fromFile, language) {
|
|
52
|
+
if (!isLocalImport(specifier, language))
|
|
53
|
+
return null;
|
|
54
|
+
const fromDir = path.dirname(fromFile);
|
|
55
|
+
if (['javascript', 'typescript'].includes(language)) {
|
|
56
|
+
// Try exact path first, then with extensions, then index files
|
|
57
|
+
const candidates = [];
|
|
58
|
+
const base = path.resolve(fromDir, specifier);
|
|
59
|
+
candidates.push(base);
|
|
60
|
+
for (const ext of JS_EXTENSIONS) {
|
|
61
|
+
candidates.push(base + ext);
|
|
62
|
+
}
|
|
63
|
+
for (const ext of JS_EXTENSIONS) {
|
|
64
|
+
candidates.push(path.join(base, `index${ext}`));
|
|
65
|
+
}
|
|
66
|
+
for (const candidate of candidates) {
|
|
67
|
+
try {
|
|
68
|
+
if (fs.statSync(candidate).isFile())
|
|
69
|
+
return candidate;
|
|
70
|
+
}
|
|
71
|
+
catch { /* not found */ }
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
else if (language === 'python') {
|
|
75
|
+
// Handle relative imports: .utils, ..models, .sub.module
|
|
76
|
+
let resolveDir = fromDir;
|
|
77
|
+
let moduleSpec = specifier;
|
|
78
|
+
if (specifier.startsWith('.')) {
|
|
79
|
+
// Count leading dots for parent traversal
|
|
80
|
+
const dotMatch = specifier.match(/^(\.+)(.*)/);
|
|
81
|
+
if (dotMatch) {
|
|
82
|
+
const dots = dotMatch[1].length;
|
|
83
|
+
moduleSpec = dotMatch[2]; // remainder after dots (may be empty for bare ".")
|
|
84
|
+
// Each dot beyond the first goes up one directory
|
|
85
|
+
resolveDir = fromDir;
|
|
86
|
+
for (let i = 1; i < dots; i++) {
|
|
87
|
+
resolveDir = path.dirname(resolveDir);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
// Convert module.path to filesystem path
|
|
92
|
+
const modulePath = moduleSpec ? moduleSpec.replace(/\./g, path.sep) : '';
|
|
93
|
+
const base = modulePath ? path.resolve(resolveDir, modulePath) : resolveDir;
|
|
94
|
+
for (const ext of PY_EXTENSIONS) {
|
|
95
|
+
try {
|
|
96
|
+
const candidate = base + ext;
|
|
97
|
+
if (fs.statSync(candidate).isFile())
|
|
98
|
+
return candidate;
|
|
99
|
+
}
|
|
100
|
+
catch { /* not found */ }
|
|
101
|
+
}
|
|
102
|
+
// Try as package (directory with __init__.py)
|
|
103
|
+
try {
|
|
104
|
+
const initFile = path.join(base, '__init__.py');
|
|
105
|
+
if (fs.statSync(initFile).isFile())
|
|
106
|
+
return initFile;
|
|
107
|
+
}
|
|
108
|
+
catch { /* not found */ }
|
|
109
|
+
}
|
|
110
|
+
return null;
|
|
111
|
+
}
|
|
112
|
+
export function isLocalImport(specifier, language) {
|
|
113
|
+
if (['javascript', 'typescript'].includes(language)) {
|
|
114
|
+
return specifier.startsWith('./') || specifier.startsWith('../');
|
|
115
|
+
}
|
|
116
|
+
if (language === 'python') {
|
|
117
|
+
return specifier.startsWith('.');
|
|
118
|
+
}
|
|
119
|
+
if (language === 'go') {
|
|
120
|
+
return !specifier.includes('.');
|
|
121
|
+
}
|
|
122
|
+
return false;
|
|
123
|
+
}
|
|
124
|
+
//# sourceMappingURL=resolver.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"resolver.js","sourceRoot":"","sources":["../../../src/graph/resolver.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
export { AnalysisEngine, type ProgressCallback } from './analyzer/engine.js';
|
|
2
|
+
export { IntentProfiler } from './analyzer/intent.js';
|
|
3
|
+
export { SemanticAnalyzer } from './analyzer/semantic.js';
|
|
4
|
+
export { AnthropicProvider } from './llm/anthropic.js';
|
|
5
|
+
export { ClaudeCliProvider } from './llm/claude-cli.js';
|
|
6
|
+
export { OpenAIProvider } from './llm/openai.js';
|
|
7
|
+
export { ModelRouter } from './llm/router.js';
|
|
8
|
+
export type { LLMProvider, ChatMessage } from './llm/provider.js';
|
|
9
|
+
export { SchemaValidationError } from './llm/provider.js';
|
|
10
|
+
export { zodToJsonSchema, zodToAnthropicTool, zodToOpenAIResponseFormat } from './llm/schemas.js';
|
|
11
|
+
export { buildProjectContext, formatProjectContextForLLM } from './context/project.js';
|
|
12
|
+
export { buildFileContext, isTestFile, isConfigFile, isGeneratedFile } from './context/file.js';
|
|
13
|
+
export { ContextAssembler } from './context/assembler.js';
|
|
14
|
+
export { DependencyGraphBuilder } from './graph/dependency.js';
|
|
15
|
+
export { resolveImportPath, extractImports, isLocalImport } from './graph/resolver.js';
|
|
16
|
+
export type { AnalysisResult, AnalysisStats, FileAnalysisResult, ProjectContext, FileContext, DependencyNode, DependencyGraph, } from './types/analysis.js';
|
|
17
|
+
export type { AnalysisOptions, CRAgentConfig, } from './types/config.js';
|
|
18
|
+
export { loadConfig, resolveOptions } from './types/config.js';
|
|
19
|
+
export { FindingSchema, FileAnalysisResponseSchema, IntentProfileSchema, TriageDecisionSchema, SeveritySchema, CategorySchema, IntentAlignmentSchema, RiskDomainSchema, } from './types/findings.js';
|
|
20
|
+
export type { Finding, FileAnalysisResponse, IntentProfile, TriageDecision, Severity, Category, IntentAlignment, RiskDomain, } from './types/findings.js';
|
|
21
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,cAAc,EAAE,KAAK,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AAC7E,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAG1D,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AACxD,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC9C,YAAY,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAClE,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAC1D,OAAO,EAAE,eAAe,EAAE,kBAAkB,EAAE,yBAAyB,EAAE,MAAM,kBAAkB,CAAC;AAGlG,OAAO,EAAE,mBAAmB,EAAE,0BAA0B,EAAE,MAAM,sBAAsB,CAAC;AACvF,OAAO,EAAE,gBAAgB,EAAE,UAAU,EAAE,YAAY,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAChG,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAG1D,OAAO,EAAE,sBAAsB,EAAE,MAAM,uBAAuB,CAAC;AAC/D,OAAO,EAAE,iBAAiB,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAGvF,YAAY,EACV,cAAc,EACd,aAAa,EACb,kBAAkB,EAClB,cAAc,EACd,WAAW,EACX,cAAc,EACd,eAAe,GAChB,MAAM,qBAAqB,CAAC;AAC7B,YAAY,EACV,eAAe,EACf,aAAa,GACd,MAAM,mBAAmB,CAAC;AAC3B,OAAO,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAC/D,OAAO,EACL,aAAa,EACb,0BAA0B,EAC1B,mBAAmB,EACnB,oBAAoB,EACpB,cAAc,EACd,cAAc,EACd,qBAAqB,EACrB,gBAAgB,GACjB,MAAM,qBAAqB,CAAC;AAC7B,YAAY,EACV,OAAO,EACP,oBAAoB,EACpB,aAAa,EACb,cAAc,EACd,QAAQ,EACR,QAAQ,EACR,eAAe,EACf,UAAU,GACX,MAAM,qBAAqB,CAAC"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
// Core engine
|
|
2
|
+
export { AnalysisEngine } from './analyzer/engine.js';
|
|
3
|
+
export { IntentProfiler } from './analyzer/intent.js';
|
|
4
|
+
export { SemanticAnalyzer } from './analyzer/semantic.js';
|
|
5
|
+
// LLM providers
|
|
6
|
+
export { AnthropicProvider } from './llm/anthropic.js';
|
|
7
|
+
export { ClaudeCliProvider } from './llm/claude-cli.js';
|
|
8
|
+
export { OpenAIProvider } from './llm/openai.js';
|
|
9
|
+
export { ModelRouter } from './llm/router.js';
|
|
10
|
+
export { SchemaValidationError } from './llm/provider.js';
|
|
11
|
+
export { zodToJsonSchema, zodToAnthropicTool, zodToOpenAIResponseFormat } from './llm/schemas.js';
|
|
12
|
+
// Context
|
|
13
|
+
export { buildProjectContext, formatProjectContextForLLM } from './context/project.js';
|
|
14
|
+
export { buildFileContext, isTestFile, isConfigFile, isGeneratedFile } from './context/file.js';
|
|
15
|
+
export { ContextAssembler } from './context/assembler.js';
|
|
16
|
+
// Graph
|
|
17
|
+
export { DependencyGraphBuilder } from './graph/dependency.js';
|
|
18
|
+
export { resolveImportPath, extractImports, isLocalImport } from './graph/resolver.js';
|
|
19
|
+
export { loadConfig, resolveOptions } from './types/config.js';
|
|
20
|
+
export { FindingSchema, FileAnalysisResponseSchema, IntentProfileSchema, TriageDecisionSchema, SeveritySchema, CategorySchema, IntentAlignmentSchema, RiskDomainSchema, } from './types/findings.js';
|
|
21
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,cAAc;AACd,OAAO,EAAE,cAAc,EAAyB,MAAM,sBAAsB,CAAC;AAC7E,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAE1D,gBAAgB;AAChB,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AACxD,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAE9C,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAC1D,OAAO,EAAE,eAAe,EAAE,kBAAkB,EAAE,yBAAyB,EAAE,MAAM,kBAAkB,CAAC;AAElG,UAAU;AACV,OAAO,EAAE,mBAAmB,EAAE,0BAA0B,EAAE,MAAM,sBAAsB,CAAC;AACvF,OAAO,EAAE,gBAAgB,EAAE,UAAU,EAAE,YAAY,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAChG,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAE1D,QAAQ;AACR,OAAO,EAAE,sBAAsB,EAAE,MAAM,uBAAuB,CAAC;AAC/D,OAAO,EAAE,iBAAiB,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAgBvF,OAAO,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAC/D,OAAO,EACL,aAAa,EACb,0BAA0B,EAC1B,mBAAmB,EACnB,oBAAoB,EACpB,cAAc,EACd,cAAc,EACd,qBAAqB,EACrB,gBAAgB,GACjB,MAAM,qBAAqB,CAAC"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { z } from 'zod';
|
|
2
|
+
import { type ChatMessage, type LLMProvider } from './provider.js';
|
|
3
|
+
export declare class AnthropicProvider implements LLMProvider {
|
|
4
|
+
private client;
|
|
5
|
+
readonly modelId: string;
|
|
6
|
+
readonly providerName = "anthropic";
|
|
7
|
+
constructor(apiKey: string, model?: string);
|
|
8
|
+
chat(messages: ChatMessage[]): Promise<string>;
|
|
9
|
+
chatStructured<T>(messages: ChatMessage[], schema: z.ZodType<T>, schemaName: string): Promise<T>;
|
|
10
|
+
countTokens(text: string): number;
|
|
11
|
+
private splitMessages;
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=anthropic.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"anthropic.d.ts","sourceRoot":"","sources":["../../../src/llm/anthropic.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAC7B,OAAO,EAAE,KAAK,WAAW,EAAE,KAAK,WAAW,EAAyB,MAAM,eAAe,CAAC;AAK1F,qBAAa,iBAAkB,YAAW,WAAW;IACnD,OAAO,CAAC,MAAM,CAAY;IAC1B,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,YAAY,eAAe;gBAExB,MAAM,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM;IAKpC,IAAI,CAAC,QAAQ,EAAE,WAAW,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;IAc9C,cAAc,CAAC,CAAC,EACpB,QAAQ,EAAE,WAAW,EAAE,EACvB,MAAM,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EACpB,UAAU,EAAE,MAAM,GACjB,OAAO,CAAC,CAAC,CAAC;IA4Cb,WAAW,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;IAQjC,OAAO,CAAC,aAAa;CAiBtB"}
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import Anthropic from '@anthropic-ai/sdk';
|
|
2
|
+
import { countTokens } from '@anthropic-ai/tokenizer';
|
|
3
|
+
import { SchemaValidationError } from './provider.js';
|
|
4
|
+
import { zodToAnthropicTool } from './schemas.js';
|
|
5
|
+
const MAX_RETRIES = 3;
|
|
6
|
+
export class AnthropicProvider {
|
|
7
|
+
client;
|
|
8
|
+
modelId;
|
|
9
|
+
providerName = 'anthropic';
|
|
10
|
+
constructor(apiKey, model) {
|
|
11
|
+
this.client = new Anthropic({ apiKey });
|
|
12
|
+
this.modelId = model ?? 'claude-sonnet-4-20250514';
|
|
13
|
+
}
|
|
14
|
+
async chat(messages) {
|
|
15
|
+
const { system, userMessages } = this.splitMessages(messages);
|
|
16
|
+
const response = await this.client.messages.create({
|
|
17
|
+
model: this.modelId,
|
|
18
|
+
max_tokens: 4096,
|
|
19
|
+
system: system ?? undefined,
|
|
20
|
+
messages: userMessages,
|
|
21
|
+
});
|
|
22
|
+
const textBlock = response.content.find((b) => b.type === 'text');
|
|
23
|
+
return textBlock?.text ?? '';
|
|
24
|
+
}
|
|
25
|
+
async chatStructured(messages, schema, schemaName) {
|
|
26
|
+
const tool = zodToAnthropicTool(schema, schemaName, `Respond with ${schemaName}`);
|
|
27
|
+
const { system, userMessages } = this.splitMessages(messages);
|
|
28
|
+
let lastError = null;
|
|
29
|
+
const conversationMessages = [...userMessages];
|
|
30
|
+
for (let attempt = 0; attempt < MAX_RETRIES; attempt++) {
|
|
31
|
+
const response = await this.client.messages.create({
|
|
32
|
+
model: this.modelId,
|
|
33
|
+
max_tokens: 8192,
|
|
34
|
+
system: system ?? undefined,
|
|
35
|
+
messages: conversationMessages,
|
|
36
|
+
tools: [tool],
|
|
37
|
+
tool_choice: { type: 'tool', name: schemaName },
|
|
38
|
+
});
|
|
39
|
+
const toolBlock = response.content.find((b) => b.type === 'tool_use');
|
|
40
|
+
if (!toolBlock || toolBlock.type !== 'tool_use') {
|
|
41
|
+
lastError = new Error('No tool_use block in response');
|
|
42
|
+
continue;
|
|
43
|
+
}
|
|
44
|
+
const result = schema.safeParse(toolBlock.input);
|
|
45
|
+
if (result.success) {
|
|
46
|
+
return result.data;
|
|
47
|
+
}
|
|
48
|
+
lastError = new Error(result.error.message);
|
|
49
|
+
// Append error feedback for retry
|
|
50
|
+
conversationMessages.push({
|
|
51
|
+
role: 'assistant',
|
|
52
|
+
content: JSON.stringify(toolBlock.input),
|
|
53
|
+
});
|
|
54
|
+
conversationMessages.push({
|
|
55
|
+
role: 'user',
|
|
56
|
+
content: `Schema validation error: ${result.error.message}. Please fix and respond again.`,
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
throw new SchemaValidationError(MAX_RETRIES, lastError);
|
|
60
|
+
}
|
|
61
|
+
countTokens(text) {
|
|
62
|
+
try {
|
|
63
|
+
return countTokens(text);
|
|
64
|
+
}
|
|
65
|
+
catch {
|
|
66
|
+
return Math.ceil(text.length / 4);
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
splitMessages(messages) {
|
|
70
|
+
let system = null;
|
|
71
|
+
const userMessages = [];
|
|
72
|
+
for (const msg of messages) {
|
|
73
|
+
if (msg.role === 'system') {
|
|
74
|
+
system = msg.content;
|
|
75
|
+
}
|
|
76
|
+
else {
|
|
77
|
+
userMessages.push({ role: msg.role, content: msg.content });
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
return { system, userMessages };
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
//# sourceMappingURL=anthropic.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"anthropic.js","sourceRoot":"","sources":["../../../src/llm/anthropic.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { z } from 'zod';
|
|
2
|
+
import { type ChatMessage, type LLMProvider } from './provider.js';
|
|
3
|
+
export declare class ClaudeCliProvider implements LLMProvider {
|
|
4
|
+
readonly modelId: string;
|
|
5
|
+
readonly providerName = "claude-cli";
|
|
6
|
+
constructor(model?: string);
|
|
7
|
+
chat(messages: ChatMessage[]): Promise<string>;
|
|
8
|
+
chatStructured<T>(messages: ChatMessage[], schema: z.ZodType<T>, schemaName: string): Promise<T>;
|
|
9
|
+
countTokens(text: string): number;
|
|
10
|
+
private formatMessages;
|
|
11
|
+
private runClaude;
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=claude-cli.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"claude-cli.d.ts","sourceRoot":"","sources":["../../../src/llm/claude-cli.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAC7B,OAAO,EAAE,KAAK,WAAW,EAAE,KAAK,WAAW,EAAyB,MAAM,eAAe,CAAC;AAe1F,qBAAa,iBAAkB,YAAW,WAAW;IACnD,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,YAAY,gBAAgB;gBAEzB,KAAK,CAAC,EAAE,MAAM;IAIpB,IAAI,CAAC,QAAQ,EAAE,WAAW,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;IAK9C,cAAc,CAAC,CAAC,EACpB,QAAQ,EAAE,WAAW,EAAE,EACvB,MAAM,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EACpB,UAAU,EAAE,MAAM,GACjB,OAAO,CAAC,CAAC,CAAC;IAoDb,WAAW,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;IAKjC,OAAO,CAAC,cAAc;IAgBtB,OAAO,CAAC,SAAS;CAyDlB"}
|
|
@@ -0,0 +1,142 @@
|
|
|
1
|
+
import { spawn } from 'node:child_process';
|
|
2
|
+
import { SchemaValidationError } from './provider.js';
|
|
3
|
+
import { zodToJsonSchema } from './schemas.js';
|
|
4
|
+
const MAX_RETRIES = 3;
|
|
5
|
+
export class ClaudeCliProvider {
|
|
6
|
+
modelId;
|
|
7
|
+
providerName = 'claude-cli';
|
|
8
|
+
constructor(model) {
|
|
9
|
+
this.modelId = model ?? 'sonnet';
|
|
10
|
+
}
|
|
11
|
+
async chat(messages) {
|
|
12
|
+
const prompt = this.formatMessages(messages);
|
|
13
|
+
return this.runClaude(prompt);
|
|
14
|
+
}
|
|
15
|
+
async chatStructured(messages, schema, schemaName) {
|
|
16
|
+
const jsonSchema = zodToJsonSchema(schema);
|
|
17
|
+
const schemaInstruction = [
|
|
18
|
+
`You MUST respond with ONLY a valid JSON object matching this schema (no markdown, no explanation, no wrapping):`,
|
|
19
|
+
`Schema name: ${schemaName}`,
|
|
20
|
+
'```json',
|
|
21
|
+
JSON.stringify(jsonSchema, null, 2),
|
|
22
|
+
'```',
|
|
23
|
+
'Respond with ONLY the JSON object. No other text.',
|
|
24
|
+
].join('\n');
|
|
25
|
+
let lastError = null;
|
|
26
|
+
const conversationParts = [...messages];
|
|
27
|
+
for (let attempt = 0; attempt < MAX_RETRIES; attempt++) {
|
|
28
|
+
const prompt = this.formatMessages([
|
|
29
|
+
...conversationParts,
|
|
30
|
+
{ role: 'user', content: schemaInstruction },
|
|
31
|
+
]);
|
|
32
|
+
const raw = await this.runClaude(prompt);
|
|
33
|
+
// Extract JSON from response (handle markdown code blocks)
|
|
34
|
+
const jsonStr = extractJson(raw);
|
|
35
|
+
let parsed;
|
|
36
|
+
try {
|
|
37
|
+
parsed = JSON.parse(jsonStr);
|
|
38
|
+
}
|
|
39
|
+
catch (e) {
|
|
40
|
+
lastError = e instanceof Error ? e : new Error(String(e));
|
|
41
|
+
conversationParts.push({ role: 'assistant', content: raw }, { role: 'user', content: `That was not valid JSON. Parse error: ${lastError.message}. Respond with ONLY a valid JSON object.` });
|
|
42
|
+
continue;
|
|
43
|
+
}
|
|
44
|
+
const result = schema.safeParse(parsed);
|
|
45
|
+
if (result.success) {
|
|
46
|
+
return result.data;
|
|
47
|
+
}
|
|
48
|
+
lastError = new Error(result.error.message);
|
|
49
|
+
conversationParts.push({ role: 'assistant', content: raw }, { role: 'user', content: `Schema validation error: ${result.error.message}. Fix the JSON and respond with ONLY the corrected JSON object.` });
|
|
50
|
+
}
|
|
51
|
+
throw new SchemaValidationError(MAX_RETRIES, lastError);
|
|
52
|
+
}
|
|
53
|
+
countTokens(text) {
|
|
54
|
+
// Approximate — Claude CLI doesn't expose a token counter
|
|
55
|
+
return Math.ceil(text.length / 4);
|
|
56
|
+
}
|
|
57
|
+
formatMessages(messages) {
|
|
58
|
+
const parts = [];
|
|
59
|
+
for (const msg of messages) {
|
|
60
|
+
if (msg.role === 'system') {
|
|
61
|
+
parts.push(`[System Instructions]\n${msg.content}\n`);
|
|
62
|
+
}
|
|
63
|
+
else if (msg.role === 'user') {
|
|
64
|
+
parts.push(`${msg.content}`);
|
|
65
|
+
}
|
|
66
|
+
else if (msg.role === 'assistant') {
|
|
67
|
+
parts.push(`[Previous response]\n${msg.content}\n`);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
return parts.join('\n\n');
|
|
71
|
+
}
|
|
72
|
+
runClaude(prompt) {
|
|
73
|
+
return new Promise((resolve, reject) => {
|
|
74
|
+
const args = [
|
|
75
|
+
'-p', '-',
|
|
76
|
+
'--output-format', 'json',
|
|
77
|
+
'--model', this.modelId,
|
|
78
|
+
'--no-session-persistence',
|
|
79
|
+
];
|
|
80
|
+
const child = spawn('claude', args, {
|
|
81
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
82
|
+
timeout: 180_000,
|
|
83
|
+
});
|
|
84
|
+
let stdout = '';
|
|
85
|
+
let stderr = '';
|
|
86
|
+
child.stdout.on('data', (data) => { stdout += data.toString(); });
|
|
87
|
+
child.stderr.on('data', (data) => { stderr += data.toString(); });
|
|
88
|
+
child.on('close', (code) => {
|
|
89
|
+
if (code !== 0 && !stdout) {
|
|
90
|
+
// Sanitize stderr — only show first 200 chars, never leak prompt content
|
|
91
|
+
const safeStderr = stderr ? sanitizeError(stderr) : '';
|
|
92
|
+
reject(new Error(`claude CLI exited with code ${code}${safeStderr ? `: ${safeStderr}` : ''}`));
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
try {
|
|
96
|
+
const result = JSON.parse(stdout);
|
|
97
|
+
if (result.is_error) {
|
|
98
|
+
reject(new Error(`claude CLI error: ${sanitizeError(result.result)}`));
|
|
99
|
+
return;
|
|
100
|
+
}
|
|
101
|
+
resolve(result.result);
|
|
102
|
+
}
|
|
103
|
+
catch {
|
|
104
|
+
resolve(stdout.trim());
|
|
105
|
+
}
|
|
106
|
+
});
|
|
107
|
+
child.on('error', (err) => {
|
|
108
|
+
reject(new Error(`claude CLI failed to start: ${sanitizeError(err.message)}`));
|
|
109
|
+
});
|
|
110
|
+
// Handle stdin write errors (e.g., broken pipe if child exits early)
|
|
111
|
+
child.stdin.on('error', (err) => {
|
|
112
|
+
// Ignore EPIPE — child may have already exited, close handler will deal with it
|
|
113
|
+
if (err.code !== 'EPIPE') {
|
|
114
|
+
reject(new Error(`Failed to write prompt to claude CLI: ${sanitizeError(err.message)}`));
|
|
115
|
+
}
|
|
116
|
+
});
|
|
117
|
+
// Write prompt to stdin and close it
|
|
118
|
+
child.stdin.write(prompt);
|
|
119
|
+
child.stdin.end();
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
function sanitizeError(msg) {
|
|
124
|
+
// Never leak prompt content in errors — truncate and strip anything
|
|
125
|
+
// that looks like it could be prompt/code/schema content
|
|
126
|
+
const firstLine = msg.split('\n')[0].trim();
|
|
127
|
+
return firstLine.length > 200 ? firstLine.slice(0, 200) + '...' : firstLine;
|
|
128
|
+
}
|
|
129
|
+
function extractJson(text) {
|
|
130
|
+
// Try to extract JSON from markdown code blocks
|
|
131
|
+
const codeBlockMatch = text.match(/```(?:json)?\s*\n?([\s\S]*?)\n?```/);
|
|
132
|
+
if (codeBlockMatch) {
|
|
133
|
+
return codeBlockMatch[1].trim();
|
|
134
|
+
}
|
|
135
|
+
// Try to find a JSON object directly
|
|
136
|
+
const objectMatch = text.match(/\{[\s\S]*\}/);
|
|
137
|
+
if (objectMatch) {
|
|
138
|
+
return objectMatch[0];
|
|
139
|
+
}
|
|
140
|
+
return text.trim();
|
|
141
|
+
}
|
|
142
|
+
//# sourceMappingURL=claude-cli.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"claude-cli.js","sourceRoot":"","sources":["../../../src/llm/claude-cli.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { z } from 'zod';
|
|
2
|
+
import { type ChatMessage, type LLMProvider } from './provider.js';
|
|
3
|
+
export declare class OpenAIProvider implements LLMProvider {
|
|
4
|
+
private client;
|
|
5
|
+
private encoder;
|
|
6
|
+
readonly modelId: string;
|
|
7
|
+
readonly providerName = "openai";
|
|
8
|
+
constructor(apiKey: string, model?: string);
|
|
9
|
+
chat(messages: ChatMessage[]): Promise<string>;
|
|
10
|
+
chatStructured<T>(messages: ChatMessage[], schema: z.ZodType<T>, schemaName: string): Promise<T>;
|
|
11
|
+
countTokens(text: string): number;
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=openai.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"openai.d.ts","sourceRoot":"","sources":["../../../src/llm/openai.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAC7B,OAAO,EAAE,KAAK,WAAW,EAAE,KAAK,WAAW,EAAyB,MAAM,eAAe,CAAC;AAK1F,qBAAa,cAAe,YAAW,WAAW;IAChD,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,OAAO,CAAyB;IACxC,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,YAAY,YAAY;gBAErB,MAAM,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM;IAKpC,IAAI,CAAC,QAAQ,EAAE,WAAW,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;IAU9C,cAAc,CAAC,CAAC,EACpB,QAAQ,EAAE,WAAW,EAAE,EACvB,MAAM,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EACpB,UAAU,EAAE,MAAM,GACjB,OAAO,CAAC,CAAC,CAAC;IAmDb,WAAW,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;CAUlC"}
|