lacuna-cli 0.1.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 +451 -0
- package/bin/run.js +5 -0
- package/dist/agent/context.d.ts +25 -0
- package/dist/agent/context.d.ts.map +1 -0
- package/dist/agent/context.js +366 -0
- package/dist/agent/context.js.map +1 -0
- package/dist/agent/fix-loop.d.ts +20 -0
- package/dist/agent/fix-loop.d.ts.map +1 -0
- package/dist/agent/fix-loop.js +466 -0
- package/dist/agent/fix-loop.js.map +1 -0
- package/dist/agent/generator.d.ts +35 -0
- package/dist/agent/generator.d.ts.map +1 -0
- package/dist/agent/generator.js +220 -0
- package/dist/agent/generator.js.map +1 -0
- package/dist/agent/loop.d.ts +23 -0
- package/dist/agent/loop.d.ts.map +1 -0
- package/dist/agent/loop.js +394 -0
- package/dist/agent/loop.js.map +1 -0
- package/dist/agent/project-memory.d.ts +10 -0
- package/dist/agent/project-memory.d.ts.map +1 -0
- package/dist/agent/project-memory.js +57 -0
- package/dist/agent/project-memory.js.map +1 -0
- package/dist/agent/prompts.d.ts +44 -0
- package/dist/agent/prompts.d.ts.map +1 -0
- package/dist/agent/prompts.js +377 -0
- package/dist/agent/prompts.js.map +1 -0
- package/dist/ci/comment.d.ts +2 -0
- package/dist/ci/comment.d.ts.map +1 -0
- package/dist/ci/comment.js +97 -0
- package/dist/ci/comment.js.map +1 -0
- package/dist/ci/parse-outputs.d.ts +2 -0
- package/dist/ci/parse-outputs.d.ts.map +1 -0
- package/dist/ci/parse-outputs.js +30 -0
- package/dist/ci/parse-outputs.js.map +1 -0
- package/dist/commands/analyze.d.ts +13 -0
- package/dist/commands/analyze.d.ts.map +1 -0
- package/dist/commands/analyze.js +151 -0
- package/dist/commands/analyze.js.map +1 -0
- package/dist/commands/fix.d.ts +15 -0
- package/dist/commands/fix.d.ts.map +1 -0
- package/dist/commands/fix.js +106 -0
- package/dist/commands/fix.js.map +1 -0
- package/dist/commands/generate.d.ts +18 -0
- package/dist/commands/generate.d.ts.map +1 -0
- package/dist/commands/generate.js +129 -0
- package/dist/commands/generate.js.map +1 -0
- package/dist/commands/init.d.ts +7 -0
- package/dist/commands/init.d.ts.map +1 -0
- package/dist/commands/init.js +131 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/run.d.ts +10 -0
- package/dist/commands/run.d.ts.map +1 -0
- package/dist/commands/run.js +45 -0
- package/dist/commands/run.js.map +1 -0
- package/dist/lib/config.d.ts +58 -0
- package/dist/lib/config.d.ts.map +1 -0
- package/dist/lib/config.js +68 -0
- package/dist/lib/config.js.map +1 -0
- package/dist/lib/coverage/gaps.d.ts +12 -0
- package/dist/lib/coverage/gaps.d.ts.map +1 -0
- package/dist/lib/coverage/gaps.js +186 -0
- package/dist/lib/coverage/gaps.js.map +1 -0
- package/dist/lib/coverage/index.d.ts +7 -0
- package/dist/lib/coverage/index.d.ts.map +1 -0
- package/dist/lib/coverage/index.js +24 -0
- package/dist/lib/coverage/index.js.map +1 -0
- package/dist/lib/coverage/json.d.ts +3 -0
- package/dist/lib/coverage/json.d.ts.map +1 -0
- package/dist/lib/coverage/json.js +24 -0
- package/dist/lib/coverage/json.js.map +1 -0
- package/dist/lib/coverage/lcov.d.ts +3 -0
- package/dist/lib/coverage/lcov.d.ts.map +1 -0
- package/dist/lib/coverage/lcov.js +58 -0
- package/dist/lib/coverage/lcov.js.map +1 -0
- package/dist/lib/coverage/types.d.ts +27 -0
- package/dist/lib/coverage/types.d.ts.map +1 -0
- package/dist/lib/coverage/types.js +2 -0
- package/dist/lib/coverage/types.js.map +1 -0
- package/dist/lib/coverage-spinner.d.ts +6 -0
- package/dist/lib/coverage-spinner.d.ts.map +1 -0
- package/dist/lib/coverage-spinner.js +101 -0
- package/dist/lib/coverage-spinner.js.map +1 -0
- package/dist/lib/detector.d.ts +13 -0
- package/dist/lib/detector.d.ts.map +1 -0
- package/dist/lib/detector.js +106 -0
- package/dist/lib/detector.js.map +1 -0
- package/dist/lib/extract-error.d.ts +2 -0
- package/dist/lib/extract-error.d.ts.map +1 -0
- package/dist/lib/extract-error.js +116 -0
- package/dist/lib/extract-error.js.map +1 -0
- package/dist/lib/providers/anthropic.d.ts +8 -0
- package/dist/lib/providers/anthropic.d.ts.map +1 -0
- package/dist/lib/providers/anthropic.js +38 -0
- package/dist/lib/providers/anthropic.js.map +1 -0
- package/dist/lib/providers/index.d.ts +6 -0
- package/dist/lib/providers/index.d.ts.map +1 -0
- package/dist/lib/providers/index.js +27 -0
- package/dist/lib/providers/index.js.map +1 -0
- package/dist/lib/providers/openai-compatible.d.ts +11 -0
- package/dist/lib/providers/openai-compatible.d.ts.map +1 -0
- package/dist/lib/providers/openai-compatible.js +93 -0
- package/dist/lib/providers/openai-compatible.js.map +1 -0
- package/dist/lib/providers/types.d.ts +17 -0
- package/dist/lib/providers/types.d.ts.map +1 -0
- package/dist/lib/providers/types.js +97 -0
- package/dist/lib/providers/types.js.map +1 -0
- package/dist/lib/report-upload.d.ts +3 -0
- package/dist/lib/report-upload.d.ts.map +1 -0
- package/dist/lib/report-upload.js +15 -0
- package/dist/lib/report-upload.js.map +1 -0
- package/dist/lib/reporter.d.ts +51 -0
- package/dist/lib/reporter.d.ts.map +1 -0
- package/dist/lib/reporter.js +172 -0
- package/dist/lib/reporter.js.map +1 -0
- package/dist/lib/runner.d.ts +9 -0
- package/dist/lib/runner.d.ts.map +1 -0
- package/dist/lib/runner.js +50 -0
- package/dist/lib/runner.js.map +1 -0
- package/dist/lib/skeleton.d.ts +8 -0
- package/dist/lib/skeleton.d.ts.map +1 -0
- package/dist/lib/skeleton.js +122 -0
- package/dist/lib/skeleton.js.map +1 -0
- package/dist/lib/streaming-viewer.d.ts +14 -0
- package/dist/lib/streaming-viewer.d.ts.map +1 -0
- package/dist/lib/streaming-viewer.js +80 -0
- package/dist/lib/streaming-viewer.js.map +1 -0
- package/dist/lib/tips.d.ts +16 -0
- package/dist/lib/tips.d.ts.map +1 -0
- package/dist/lib/tips.js +76 -0
- package/dist/lib/tips.js.map +1 -0
- package/dist/lib/typecheck.d.ts +3 -0
- package/dist/lib/typecheck.d.ts.map +1 -0
- package/dist/lib/typecheck.js +28 -0
- package/dist/lib/typecheck.js.map +1 -0
- package/dist/lib/validate.d.ts +7 -0
- package/dist/lib/validate.d.ts.map +1 -0
- package/dist/lib/validate.js +82 -0
- package/dist/lib/validate.js.map +1 -0
- package/dist/lib/worker-display.d.ts +45 -0
- package/dist/lib/worker-display.d.ts.map +1 -0
- package/dist/lib/worker-display.js +168 -0
- package/dist/lib/worker-display.js.map +1 -0
- package/oclif.manifest.json +295 -0
- package/package.json +62 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"json.js","sourceRoot":"","sources":["../../../src/lib/coverage/json.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAA;AACtC,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAA;AAY3B,SAAS,mBAAmB,CAAC,IAAY,EAAE,KAAuB;IAChE,OAAO;QACL,IAAI;QACJ,KAAK,EAAE,EAAE;QACT,SAAS,EAAE,EAAE;QACb,QAAQ,EAAE,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,OAAO,GAAG,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACzE,YAAY,EAAE,KAAK,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,OAAO,GAAG,KAAK,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;KAC1F,CAAA;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,WAAmB,EAAE,MAAc,OAAO,CAAC,GAAG,EAAE;IACrF,MAAM,WAAW,GAAG,IAAI,CAAC,GAAG,EAAE,WAAW,EAAE,uBAAuB,CAAC,CAAA;IACnE,MAAM,GAAG,GAAG,MAAM,QAAQ,CAAC,WAAW,EAAE,OAAO,CAAC,CAAA;IAChD,MAAM,OAAO,GAAgB,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAA;IAE5C,MAAM,KAAK,GAAmB,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC;SAClD,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,EAAE,CAAC,IAAI,KAAK,OAAO,CAAC;SACpC,GAAG,CAAC,CAAC,CAAC,IAAI,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,mBAAmB,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAA;IAE3D,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,CAAA;IAC9B,MAAM,aAAa,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,IAAI,CAAC,CAAC,CAAA;IACrH,MAAM,iBAAiB,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,YAAY,EAAE,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,IAAI,CAAC,CAAC,CAAA;IAEjI,OAAO,EAAE,KAAK,EAAE,aAAa,EAAE,iBAAiB,EAAE,CAAA;AACpD,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"lcov.d.ts","sourceRoot":"","sources":["../../../src/lib/coverage/lcov.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,cAAc,EAAgD,MAAM,YAAY,CAAA;AAiD9F,wBAAsB,SAAS,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,GAAE,MAAsB,GAAG,OAAO,CAAC,cAAc,CAAC,CAgBzG"}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import { readFile } from 'fs/promises';
|
|
2
|
+
import { join } from 'path';
|
|
3
|
+
function parseLcovText(text) {
|
|
4
|
+
const entries = [];
|
|
5
|
+
let current = null;
|
|
6
|
+
const fnNames = {};
|
|
7
|
+
for (const raw of text.split('\n')) {
|
|
8
|
+
const line = raw.trim();
|
|
9
|
+
if (line.startsWith('SF:')) {
|
|
10
|
+
current = { file: line.slice(3), lines: [], functions: [] };
|
|
11
|
+
Object.keys(fnNames).forEach((k) => delete fnNames[k]);
|
|
12
|
+
}
|
|
13
|
+
else if (line.startsWith('FN:') && current) {
|
|
14
|
+
const [lineNo, name] = line.slice(3).split(',');
|
|
15
|
+
fnNames[name] = parseInt(lineNo, 10);
|
|
16
|
+
}
|
|
17
|
+
else if (line.startsWith('FNDA:') && current) {
|
|
18
|
+
const [hitStr, name] = line.slice(5).split(',');
|
|
19
|
+
current.functions.push({ name, line: fnNames[name] ?? 0, hit: parseInt(hitStr, 10) });
|
|
20
|
+
}
|
|
21
|
+
else if (line.startsWith('DA:') && current) {
|
|
22
|
+
const [lineNo, hitStr] = line.slice(3).split(',');
|
|
23
|
+
current.lines.push({ line: parseInt(lineNo, 10), hit: parseInt(hitStr, 10) });
|
|
24
|
+
}
|
|
25
|
+
else if (line === 'end_of_record' && current) {
|
|
26
|
+
entries.push(current);
|
|
27
|
+
current = null;
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
return entries;
|
|
31
|
+
}
|
|
32
|
+
function toFileCoverage(entry) {
|
|
33
|
+
const coveredLines = entry.lines.filter((l) => l.hit > 0).length;
|
|
34
|
+
const coveredFns = entry.functions.filter((f) => f.hit > 0).length;
|
|
35
|
+
return {
|
|
36
|
+
path: entry.file,
|
|
37
|
+
lines: entry.lines,
|
|
38
|
+
functions: entry.functions,
|
|
39
|
+
lineRate: entry.lines.length ? coveredLines / entry.lines.length : 1,
|
|
40
|
+
functionRate: entry.functions.length ? coveredFns / entry.functions.length : 1,
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
export async function parseLcov(coverageDir, cwd = process.cwd()) {
|
|
44
|
+
const lcovPath = join(cwd, coverageDir, 'lcov.info');
|
|
45
|
+
const text = await readFile(lcovPath, 'utf-8');
|
|
46
|
+
const entries = parseLcovText(text);
|
|
47
|
+
const files = entries.map(toFileCoverage);
|
|
48
|
+
const totalLines = files.reduce((sum, f) => sum + f.lines.length, 0);
|
|
49
|
+
const coveredLines = files.reduce((sum, f) => sum + f.lines.filter((l) => l.hit > 0).length, 0);
|
|
50
|
+
const totalFns = files.reduce((sum, f) => sum + f.functions.length, 0);
|
|
51
|
+
const coveredFns = files.reduce((sum, f) => sum + f.functions.filter((fn) => fn.hit > 0).length, 0);
|
|
52
|
+
return {
|
|
53
|
+
files,
|
|
54
|
+
totalLineRate: totalLines ? coveredLines / totalLines : 1,
|
|
55
|
+
totalFunctionRate: totalFns ? coveredFns / totalFns : 1,
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
//# sourceMappingURL=lcov.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"lcov.js","sourceRoot":"","sources":["../../../src/lib/coverage/lcov.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
export interface LineCoverage {
|
|
2
|
+
line: number;
|
|
3
|
+
hit: number;
|
|
4
|
+
}
|
|
5
|
+
export interface FunctionCoverage {
|
|
6
|
+
name: string;
|
|
7
|
+
line: number;
|
|
8
|
+
hit: number;
|
|
9
|
+
}
|
|
10
|
+
export interface FileCoverage {
|
|
11
|
+
path: string;
|
|
12
|
+
lines: LineCoverage[];
|
|
13
|
+
functions: FunctionCoverage[];
|
|
14
|
+
lineRate: number;
|
|
15
|
+
functionRate: number;
|
|
16
|
+
}
|
|
17
|
+
export interface CoverageReport {
|
|
18
|
+
files: FileCoverage[];
|
|
19
|
+
totalLineRate: number;
|
|
20
|
+
totalFunctionRate: number;
|
|
21
|
+
}
|
|
22
|
+
export interface CoverageGap {
|
|
23
|
+
filePath: string;
|
|
24
|
+
uncoveredLines: number[];
|
|
25
|
+
uncoveredFunctions: string[];
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../src/lib/coverage/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAA;IACZ,GAAG,EAAE,MAAM,CAAA;CACZ;AAED,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,MAAM,CAAA;IACZ,IAAI,EAAE,MAAM,CAAA;IACZ,GAAG,EAAE,MAAM,CAAA;CACZ;AAED,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAA;IACZ,KAAK,EAAE,YAAY,EAAE,CAAA;IACrB,SAAS,EAAE,gBAAgB,EAAE,CAAA;IAC7B,QAAQ,EAAE,MAAM,CAAA;IAChB,YAAY,EAAE,MAAM,CAAA;CACrB;AAED,MAAM,WAAW,cAAc;IAC7B,KAAK,EAAE,YAAY,EAAE,CAAA;IACrB,aAAa,EAAE,MAAM,CAAA;IACrB,iBAAiB,EAAE,MAAM,CAAA;CAC1B;AAED,MAAM,WAAW,WAAW;IAC1B,QAAQ,EAAE,MAAM,CAAA;IAChB,cAAc,EAAE,MAAM,EAAE,CAAA;IACxB,kBAAkB,EAAE,MAAM,EAAE,CAAA;CAC7B"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/lib/coverage/types.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"coverage-spinner.d.ts","sourceRoot":"","sources":["../../src/lib/coverage-spinner.ts"],"names":[],"mappings":"AAyCA,MAAM,WAAW,eAAe;IAC9B,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAA;IAC9B,IAAI,EAAE,MAAM,IAAI,CAAA;CACjB;AAED,wBAAgB,oBAAoB,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,SAAY,GAAG,eAAe,CAqEvF"}
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
import chalk from 'chalk';
|
|
2
|
+
const FRAMES = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];
|
|
3
|
+
const ANSI_RE = /\x1B\[[0-9;]*m/g;
|
|
4
|
+
const MAX_VISIBLE = 7;
|
|
5
|
+
function stripAnsi(s) {
|
|
6
|
+
return s.replace(ANSI_RE, '').trim();
|
|
7
|
+
}
|
|
8
|
+
function parseFileLine(line, runner) {
|
|
9
|
+
const clean = stripAnsi(line);
|
|
10
|
+
if (runner === 'vitest' || runner === 'unknown') {
|
|
11
|
+
// ✓ src/foo.test.ts (3 tests) 23ms
|
|
12
|
+
// × src/foo.test.ts (2 tests | 1 failed) 45ms
|
|
13
|
+
const m = clean.match(/^([✓✗×↓])\s+([\w./\\-]+\.(?:test|spec)\.[a-z]+)(?:.*?(\d+ms))?/);
|
|
14
|
+
if (m)
|
|
15
|
+
return { file: m[2], passed: m[1] === '✓', duration: m[3] };
|
|
16
|
+
}
|
|
17
|
+
if (runner === 'jest') {
|
|
18
|
+
const pass = clean.match(/^PASS\s+(.+?)(?:\s+\([\d.]+\s*s\))?$/);
|
|
19
|
+
if (pass)
|
|
20
|
+
return { file: pass[1].trim(), passed: true };
|
|
21
|
+
const fail = clean.match(/^FAIL\s+(.+?)(?:\s+\([\d.]+\s*s\))?$/);
|
|
22
|
+
if (fail)
|
|
23
|
+
return { file: fail[1].trim(), passed: false };
|
|
24
|
+
}
|
|
25
|
+
if (runner === 'pytest') {
|
|
26
|
+
const m = clean.match(/^(PASSED|FAILED)\s+(.+?)\s*$/);
|
|
27
|
+
if (m)
|
|
28
|
+
return { file: m[2], passed: m[1] === 'PASSED' };
|
|
29
|
+
}
|
|
30
|
+
return null;
|
|
31
|
+
}
|
|
32
|
+
export function startCoverageSpinner(label, runner = 'unknown') {
|
|
33
|
+
const isTTY = Boolean(process.stdout.isTTY);
|
|
34
|
+
const start = Date.now();
|
|
35
|
+
let tick = 0;
|
|
36
|
+
let rendered = 0;
|
|
37
|
+
const files = [];
|
|
38
|
+
if (!isTTY) {
|
|
39
|
+
process.stdout.write(label + '\n');
|
|
40
|
+
return {
|
|
41
|
+
onLine: (line) => {
|
|
42
|
+
const entry = parseFileLine(line, runner);
|
|
43
|
+
if (entry) {
|
|
44
|
+
process.stdout.write(` ${entry.passed ? '✓' : '✗'} ${entry.file}\n`);
|
|
45
|
+
}
|
|
46
|
+
},
|
|
47
|
+
stop: () => { },
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
function render() {
|
|
51
|
+
if (rendered > 0)
|
|
52
|
+
process.stdout.write(`\x1B[${rendered}A\x1B[0J`);
|
|
53
|
+
const secs = Math.floor((Date.now() - start) / 1000);
|
|
54
|
+
const frame = chalk.cyan(FRAMES[tick % FRAMES.length]);
|
|
55
|
+
const cols = Math.max(60, process.stdout.columns ?? 80);
|
|
56
|
+
const lines = [];
|
|
57
|
+
lines.push(`${label} ${frame} ${chalk.dim(secs + 's')}`);
|
|
58
|
+
if (files.length > 0) {
|
|
59
|
+
lines.push('');
|
|
60
|
+
const recent = files.slice(-MAX_VISIBLE);
|
|
61
|
+
for (const f of recent) {
|
|
62
|
+
const icon = f.passed ? chalk.green('✓') : chalk.red('✗');
|
|
63
|
+
const maxLen = cols - 10;
|
|
64
|
+
const short = f.file.length > maxLen ? '…' + f.file.slice(-(maxLen - 1)) : f.file;
|
|
65
|
+
const dur = f.duration ? chalk.dim(` ${f.duration}`) : '';
|
|
66
|
+
lines.push(` ${icon} ${chalk.dim(short)}${dur}`);
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
lines.push('');
|
|
70
|
+
const out = lines.join('\n');
|
|
71
|
+
process.stdout.write(out);
|
|
72
|
+
rendered = (out.match(/\n/g) ?? []).length;
|
|
73
|
+
}
|
|
74
|
+
const timer = setInterval(() => { tick++; render(); }, 100);
|
|
75
|
+
render();
|
|
76
|
+
return {
|
|
77
|
+
onLine: (line) => {
|
|
78
|
+
const entry = parseFileLine(line, runner);
|
|
79
|
+
if (entry) {
|
|
80
|
+
files.push(entry);
|
|
81
|
+
render();
|
|
82
|
+
}
|
|
83
|
+
},
|
|
84
|
+
stop: () => {
|
|
85
|
+
clearInterval(timer);
|
|
86
|
+
if (rendered > 0) {
|
|
87
|
+
process.stdout.write(`\x1B[${rendered}A\x1B[0J`);
|
|
88
|
+
rendered = 0;
|
|
89
|
+
}
|
|
90
|
+
const secs = Math.floor((Date.now() - start) / 1000);
|
|
91
|
+
const total = files.length;
|
|
92
|
+
const passed = files.filter(f => f.passed).length;
|
|
93
|
+
const failed = total - passed;
|
|
94
|
+
const summary = total > 0
|
|
95
|
+
? ` ${chalk.dim(`${passed} passed${failed > 0 ? `, ${failed} failed` : ''}, ${total} files — ${secs}s`)}`
|
|
96
|
+
: chalk.dim(` done in ${secs}s`);
|
|
97
|
+
process.stdout.write(`${label}${summary}\n`);
|
|
98
|
+
},
|
|
99
|
+
};
|
|
100
|
+
}
|
|
101
|
+
//# sourceMappingURL=coverage-spinner.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"coverage-spinner.js","sourceRoot":"","sources":["../../src/lib/coverage-spinner.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
export type TestRunner = 'jest' | 'vitest' | 'pytest' | 'mocha' | 'go-test' | 'unknown';
|
|
2
|
+
export type Language = 'typescript' | 'javascript' | 'python' | 'go' | 'unknown';
|
|
3
|
+
export interface DetectedEnvironment {
|
|
4
|
+
testRunner: TestRunner;
|
|
5
|
+
language: Language;
|
|
6
|
+
testFilePattern: string;
|
|
7
|
+
coverageCommand: string;
|
|
8
|
+
testCommand: string;
|
|
9
|
+
}
|
|
10
|
+
export declare function envForRunner(runner: string): DetectedEnvironment;
|
|
11
|
+
export declare function fileTestCommand(env: DetectedEnvironment, testFilePath: string): string;
|
|
12
|
+
export declare function detectEnvironment(cwd?: string, configRunner?: string): Promise<DetectedEnvironment>;
|
|
13
|
+
//# sourceMappingURL=detector.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"detector.d.ts","sourceRoot":"","sources":["../../src/lib/detector.ts"],"names":[],"mappings":"AAGA,MAAM,MAAM,UAAU,GAAG,MAAM,GAAG,QAAQ,GAAG,QAAQ,GAAG,OAAO,GAAG,SAAS,GAAG,SAAS,CAAA;AACvF,MAAM,MAAM,QAAQ,GAAG,YAAY,GAAG,YAAY,GAAG,QAAQ,GAAG,IAAI,GAAG,SAAS,CAAA;AAEhF,MAAM,WAAW,mBAAmB;IAClC,UAAU,EAAE,UAAU,CAAA;IACtB,QAAQ,EAAE,QAAQ,CAAA;IAClB,eAAe,EAAE,MAAM,CAAA;IACvB,eAAe,EAAE,MAAM,CAAA;IACvB,WAAW,EAAE,MAAM,CAAA;CACpB;AAwCD,wBAAgB,YAAY,CAAC,MAAM,EAAE,MAAM,GAAG,mBAAmB,CAUhE;AAED,wBAAgB,eAAe,CAAC,GAAG,EAAE,mBAAmB,EAAE,YAAY,EAAE,MAAM,GAAG,MAAM,CAYtF;AAED,wBAAsB,iBAAiB,CACrC,GAAG,GAAE,MAAsB,EAC3B,YAAY,CAAC,EAAE,MAAM,GACpB,OAAO,CAAC,mBAAmB,CAAC,CA2C9B"}
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import { readFile } from 'fs/promises';
|
|
2
|
+
import { join } from 'path';
|
|
3
|
+
const RUNNER_DEFAULTS = {
|
|
4
|
+
vitest: {
|
|
5
|
+
testRunner: 'vitest',
|
|
6
|
+
language: 'typescript',
|
|
7
|
+
testFilePattern: '**/*.{test,spec}.{ts,tsx,js,jsx}',
|
|
8
|
+
coverageCommand: 'npx vitest run --coverage',
|
|
9
|
+
testCommand: 'npx vitest run',
|
|
10
|
+
},
|
|
11
|
+
jest: {
|
|
12
|
+
testRunner: 'jest',
|
|
13
|
+
language: 'typescript',
|
|
14
|
+
testFilePattern: '**/*.{test,spec}.{ts,tsx,js,jsx}',
|
|
15
|
+
coverageCommand: 'npx jest --coverage',
|
|
16
|
+
testCommand: 'npx jest',
|
|
17
|
+
},
|
|
18
|
+
mocha: {
|
|
19
|
+
testRunner: 'mocha',
|
|
20
|
+
language: 'javascript',
|
|
21
|
+
testFilePattern: '**/*.{test,spec}.{js,mjs}',
|
|
22
|
+
coverageCommand: 'npx nyc mocha',
|
|
23
|
+
testCommand: 'npx mocha',
|
|
24
|
+
},
|
|
25
|
+
pytest: {
|
|
26
|
+
testRunner: 'pytest',
|
|
27
|
+
language: 'python',
|
|
28
|
+
testFilePattern: 'test_*.py',
|
|
29
|
+
coverageCommand: 'python -m pytest --cov --cov-report=lcov',
|
|
30
|
+
testCommand: 'python -m pytest',
|
|
31
|
+
},
|
|
32
|
+
'go-test': {
|
|
33
|
+
testRunner: 'go-test',
|
|
34
|
+
language: 'go',
|
|
35
|
+
testFilePattern: '*_test.go',
|
|
36
|
+
coverageCommand: 'go test ./... -coverprofile=coverage/lcov.info',
|
|
37
|
+
testCommand: 'go test ./...',
|
|
38
|
+
},
|
|
39
|
+
};
|
|
40
|
+
export function envForRunner(runner) {
|
|
41
|
+
return (RUNNER_DEFAULTS[runner] ?? {
|
|
42
|
+
testRunner: 'unknown',
|
|
43
|
+
language: 'unknown',
|
|
44
|
+
testFilePattern: '**/*.test.*',
|
|
45
|
+
coverageCommand: '',
|
|
46
|
+
testCommand: '',
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
export function fileTestCommand(env, testFilePath) {
|
|
50
|
+
switch (env.testRunner) {
|
|
51
|
+
case 'vitest': return `npx vitest run ${testFilePath}`;
|
|
52
|
+
case 'jest': return `npx jest --testPathPattern=${testFilePath}`;
|
|
53
|
+
case 'mocha': return `npx mocha ${testFilePath}`;
|
|
54
|
+
case 'pytest': return `python -m pytest ${testFilePath} -v`;
|
|
55
|
+
case 'go-test': {
|
|
56
|
+
const dir = testFilePath.includes('/') ? testFilePath.replace(/\/[^/]+$/, '') : '.';
|
|
57
|
+
return `go test ./${dir}/...`;
|
|
58
|
+
}
|
|
59
|
+
default: return `${env.testCommand} ${testFilePath}`;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
export async function detectEnvironment(cwd = process.cwd(), configRunner) {
|
|
63
|
+
// config always wins over auto-detection
|
|
64
|
+
if (configRunner && configRunner !== 'unknown') {
|
|
65
|
+
return envForRunner(configRunner);
|
|
66
|
+
}
|
|
67
|
+
let pkg = {};
|
|
68
|
+
try {
|
|
69
|
+
const raw = await readFile(join(cwd, 'package.json'), 'utf-8');
|
|
70
|
+
pkg = JSON.parse(raw);
|
|
71
|
+
}
|
|
72
|
+
catch { /* not a Node project */ }
|
|
73
|
+
const deps = {
|
|
74
|
+
...(pkg.dependencies ?? {}),
|
|
75
|
+
...(pkg.devDependencies ?? {}),
|
|
76
|
+
};
|
|
77
|
+
if ('vitest' in deps)
|
|
78
|
+
return RUNNER_DEFAULTS.vitest;
|
|
79
|
+
if ('jest' in deps || '@jest/core' in deps)
|
|
80
|
+
return RUNNER_DEFAULTS.jest;
|
|
81
|
+
if ('mocha' in deps)
|
|
82
|
+
return RUNNER_DEFAULTS.mocha;
|
|
83
|
+
try {
|
|
84
|
+
await readFile(join(cwd, 'pytest.ini'), 'utf-8');
|
|
85
|
+
return RUNNER_DEFAULTS.pytest;
|
|
86
|
+
}
|
|
87
|
+
catch { /* not pytest */ }
|
|
88
|
+
try {
|
|
89
|
+
await readFile(join(cwd, 'pyproject.toml'), 'utf-8');
|
|
90
|
+
return RUNNER_DEFAULTS.pytest;
|
|
91
|
+
}
|
|
92
|
+
catch { /* not Python */ }
|
|
93
|
+
try {
|
|
94
|
+
await readFile(join(cwd, 'go.mod'), 'utf-8');
|
|
95
|
+
return RUNNER_DEFAULTS['go-test'];
|
|
96
|
+
}
|
|
97
|
+
catch { /* not Go */ }
|
|
98
|
+
return {
|
|
99
|
+
testRunner: 'unknown',
|
|
100
|
+
language: 'unknown',
|
|
101
|
+
testFilePattern: '**/*.test.*',
|
|
102
|
+
coverageCommand: '',
|
|
103
|
+
testCommand: '',
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
//# sourceMappingURL=detector.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"detector.js","sourceRoot":"","sources":["../../src/lib/detector.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"extract-error.d.ts","sourceRoot":"","sources":["../../src/lib/extract-error.ts"],"names":[],"mappings":"AA4CA,wBAAgB,kBAAkB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,CA8E5D"}
|
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
// Extracts the signal from a test runner's stdout+stderr, stripping passing-test
|
|
2
|
+
// noise so the AI receives only what it needs to fix the failure.
|
|
3
|
+
//
|
|
4
|
+
// Raw runner output contains: passing-test lines, timing headers, summary footers,
|
|
5
|
+
// and the actual failure details. Only the failure details are useful on a retry.
|
|
6
|
+
const NOISE_PATTERNS = [
|
|
7
|
+
/^✓\s+/, // vitest passing file
|
|
8
|
+
/^✔\s+/, // alternate check mark
|
|
9
|
+
/^PASS\s+/, // jest passing file
|
|
10
|
+
/^RUN\s+v\d/, // vitest run header
|
|
11
|
+
/^>\s+\S+@[\d.]+\s+test/, // npm test header
|
|
12
|
+
/^Start\s+at\s+\d/, // timing footer
|
|
13
|
+
/^Duration\s+[\d.]/, // timing footer
|
|
14
|
+
];
|
|
15
|
+
const SIGNAL_PATTERNS = [
|
|
16
|
+
/error/i,
|
|
17
|
+
/fail/i,
|
|
18
|
+
/×\s+/, // vitest failing file marker
|
|
19
|
+
/✗\s+/,
|
|
20
|
+
/Expected/,
|
|
21
|
+
/Received/,
|
|
22
|
+
/AssertionError/,
|
|
23
|
+
/TypeError/,
|
|
24
|
+
/ReferenceError/,
|
|
25
|
+
/SyntaxError/,
|
|
26
|
+
/Cannot find module/,
|
|
27
|
+
/error TS\d+/, // TypeScript errors
|
|
28
|
+
];
|
|
29
|
+
// Stack frames from project files — not node_modules, not runner internals
|
|
30
|
+
function isProjectFrame(line) {
|
|
31
|
+
return (/^\s+at\s+/.test(line) &&
|
|
32
|
+
!line.includes('node_modules') &&
|
|
33
|
+
!line.includes('node:internal') &&
|
|
34
|
+
!line.includes('@vitest/runner'));
|
|
35
|
+
}
|
|
36
|
+
const CONTEXT_BEFORE = 2;
|
|
37
|
+
const CONTEXT_AFTER = 3;
|
|
38
|
+
export function extractTestFailure(rawOutput) {
|
|
39
|
+
if (!rawOutput.trim())
|
|
40
|
+
return rawOutput;
|
|
41
|
+
const stripped = rawOutput.replace(/\x1B\[[0-9;]*m/g, '');
|
|
42
|
+
const lines = stripped.split('\n');
|
|
43
|
+
// Pass 1 — mark signal lines
|
|
44
|
+
const isSignalLine = lines.map((line) => {
|
|
45
|
+
const trimmed = line.trim();
|
|
46
|
+
if (!trimmed)
|
|
47
|
+
return false;
|
|
48
|
+
if (NOISE_PATTERNS.some((p) => p.test(trimmed)))
|
|
49
|
+
return false;
|
|
50
|
+
return SIGNAL_PATTERNS.some((p) => p.test(trimmed)) || isProjectFrame(line);
|
|
51
|
+
});
|
|
52
|
+
// Pass 2 — expand each signal line into a context window
|
|
53
|
+
const include = new Set();
|
|
54
|
+
for (let i = 0; i < lines.length; i++) {
|
|
55
|
+
if (!isSignalLine[i])
|
|
56
|
+
continue;
|
|
57
|
+
for (let j = Math.max(0, i - CONTEXT_BEFORE); j <= Math.min(lines.length - 1, i + CONTEXT_AFTER); j++) {
|
|
58
|
+
include.add(j);
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
// Pass 2.5 — extend any window that ends inside a diff block (+ / - lines)
|
|
62
|
+
// Vitest/Jest can output 20-30 line JSON diffs that would otherwise be clipped.
|
|
63
|
+
for (let i = 0; i < lines.length; i++) {
|
|
64
|
+
if (!include.has(i))
|
|
65
|
+
continue;
|
|
66
|
+
// If the next line is a diff line not yet included, extend until the diff ends
|
|
67
|
+
let j = i + 1;
|
|
68
|
+
while (j < lines.length &&
|
|
69
|
+
!include.has(j) &&
|
|
70
|
+
j < i + 40 // safety cap — never extend more than 40 lines
|
|
71
|
+
) {
|
|
72
|
+
const trimmed = lines[j].trimStart();
|
|
73
|
+
// Diff content lines start with +/- (with space after) or are indented context lines
|
|
74
|
+
if (/^[+-] /.test(trimmed) || /^\s{2,}/.test(lines[j])) {
|
|
75
|
+
include.add(j);
|
|
76
|
+
j++;
|
|
77
|
+
}
|
|
78
|
+
else {
|
|
79
|
+
break;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
// Pass 3 — build output, skipping pure noise outside the windows and deduping blanks
|
|
84
|
+
const kept = [];
|
|
85
|
+
let lastWasBlank = false;
|
|
86
|
+
let lastIncluded = -1;
|
|
87
|
+
for (let i = 0; i < lines.length; i++) {
|
|
88
|
+
if (!include.has(i))
|
|
89
|
+
continue;
|
|
90
|
+
// Insert separator when there's a gap in included lines
|
|
91
|
+
if (lastIncluded >= 0 && i > lastIncluded + 1) {
|
|
92
|
+
if (!lastWasBlank)
|
|
93
|
+
kept.push('');
|
|
94
|
+
}
|
|
95
|
+
lastIncluded = i;
|
|
96
|
+
const trimmed = lines[i].trim();
|
|
97
|
+
if (!trimmed) {
|
|
98
|
+
if (lastWasBlank)
|
|
99
|
+
continue;
|
|
100
|
+
lastWasBlank = true;
|
|
101
|
+
}
|
|
102
|
+
else {
|
|
103
|
+
lastWasBlank = false;
|
|
104
|
+
}
|
|
105
|
+
kept.push(lines[i]);
|
|
106
|
+
}
|
|
107
|
+
// Trim leading/trailing blank lines
|
|
108
|
+
while (kept.length && !kept[0].trim())
|
|
109
|
+
kept.shift();
|
|
110
|
+
while (kept.length && !kept[kept.length - 1].trim())
|
|
111
|
+
kept.pop();
|
|
112
|
+
const result = kept.join('\n');
|
|
113
|
+
// Safety: if we stripped too aggressively and lost everything, fall back
|
|
114
|
+
return result.trim() || rawOutput.slice(0, 2000);
|
|
115
|
+
}
|
|
116
|
+
//# sourceMappingURL=extract-error.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"extract-error.js","sourceRoot":"","sources":["../../src/lib/extract-error.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import type { ModelProvider, ChatMessage } from './types.js';
|
|
2
|
+
export declare class AnthropicProvider implements ModelProvider {
|
|
3
|
+
private client;
|
|
4
|
+
private model;
|
|
5
|
+
constructor(model: string, apiKey: string);
|
|
6
|
+
generate(messages: ChatMessage[], system: string, onToken?: (token: string) => void, maxTokens?: number, temperature?: number): Promise<string>;
|
|
7
|
+
}
|
|
8
|
+
//# sourceMappingURL=anthropic.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"anthropic.d.ts","sourceRoot":"","sources":["../../../src/lib/providers/anthropic.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,aAAa,EAAE,WAAW,EAAE,MAAM,YAAY,CAAA;AAE5D,qBAAa,iBAAkB,YAAW,aAAa;IACrD,OAAO,CAAC,MAAM,CAAW;IACzB,OAAO,CAAC,KAAK,CAAQ;gBAET,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM;IAKnC,QAAQ,CACZ,QAAQ,EAAE,WAAW,EAAE,EACvB,MAAM,EAAE,MAAM,EACd,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,EACjC,SAAS,SAAQ,EACjB,WAAW,CAAC,EAAE,MAAM,GACnB,OAAO,CAAC,MAAM,CAAC;CAgCnB"}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import Anthropic from '@anthropic-ai/sdk';
|
|
2
|
+
export class AnthropicProvider {
|
|
3
|
+
client;
|
|
4
|
+
model;
|
|
5
|
+
constructor(model, apiKey) {
|
|
6
|
+
this.client = new Anthropic({ apiKey });
|
|
7
|
+
this.model = model;
|
|
8
|
+
}
|
|
9
|
+
async generate(messages, system, onToken, maxTokens = 16000, temperature) {
|
|
10
|
+
let content = '';
|
|
11
|
+
try {
|
|
12
|
+
const stream = this.client.messages.stream({
|
|
13
|
+
model: this.model,
|
|
14
|
+
max_tokens: maxTokens,
|
|
15
|
+
stop_sequences: ['</code_output>'],
|
|
16
|
+
...(temperature !== undefined ? { temperature } : {}),
|
|
17
|
+
system,
|
|
18
|
+
messages,
|
|
19
|
+
});
|
|
20
|
+
for await (const event of stream) {
|
|
21
|
+
if (event.type === 'content_block_delta' && event.delta.type === 'text_delta') {
|
|
22
|
+
content += event.delta.text;
|
|
23
|
+
onToken?.(event.delta.text);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
catch (err) {
|
|
28
|
+
if (err instanceof Error && /prompt is too long|max_tokens.*exceed|too many tokens/i.test(err.message)) {
|
|
29
|
+
throw new Error(`${this.model} rejected the request — prompt too large.\n` +
|
|
30
|
+
`The assembled context (source file + test file + type definitions + mocks) exceeds the model's input limit.\n` +
|
|
31
|
+
`Try: lower maxTokens in .lacuna.json, or use --file to target a smaller source file.`);
|
|
32
|
+
}
|
|
33
|
+
throw err;
|
|
34
|
+
}
|
|
35
|
+
return content.trim();
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
//# sourceMappingURL=anthropic.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"anthropic.js","sourceRoot":"","sources":["../../../src/lib/providers/anthropic.ts"],"names":[],"mappings":"AAAA,OAAO,SAAS,MAAM,mBAAmB,CAAA;AAGzC,MAAM,OAAO,iBAAiB;IACpB,MAAM,CAAW;IACjB,KAAK,CAAQ;IAErB,YAAY,KAAa,EAAE,MAAc;QACvC,IAAI,CAAC,MAAM,GAAG,IAAI,SAAS,CAAC,EAAE,MAAM,EAAE,CAAC,CAAA;QACvC,IAAI,CAAC,KAAK,GAAG,KAAK,CAAA;IACpB,CAAC;IAED,KAAK,CAAC,QAAQ,CACZ,QAAuB,EACvB,MAAc,EACd,OAAiC,EACjC,SAAS,GAAG,KAAK,EACjB,WAAoB;QAEpB,IAAI,OAAO,GAAG,EAAE,CAAA;QAEhB,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC;gBACzC,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,UAAU,EAAE,SAAS;gBACrB,cAAc,EAAE,CAAC,gBAAgB,CAAC;gBAClC,GAAG,CAAC,WAAW,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE,WAAW,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;gBACrD,MAAM;gBACN,QAAQ;aACT,CAAC,CAAA;YAEF,IAAI,KAAK,EAAE,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;gBACjC,IAAI,KAAK,CAAC,IAAI,KAAK,qBAAqB,IAAI,KAAK,CAAC,KAAK,CAAC,IAAI,KAAK,YAAY,EAAE,CAAC;oBAC9E,OAAO,IAAI,KAAK,CAAC,KAAK,CAAC,IAAI,CAAA;oBAC3B,OAAO,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,CAAA;gBAC7B,CAAC;YACH,CAAC;QACH,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,GAAG,YAAY,KAAK,IAAI,wDAAwD,CAAC,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC;gBACvG,MAAM,IAAI,KAAK,CACb,GAAG,IAAI,CAAC,KAAK,6CAA6C;oBAC1D,+GAA+G;oBAC/G,sFAAsF,CACvF,CAAA;YACH,CAAC;YACD,MAAM,GAAG,CAAA;QACX,CAAC;QAED,OAAO,OAAO,CAAC,IAAI,EAAE,CAAA;IACvB,CAAC;CACF"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { ModelProvider } from './types.js';
|
|
2
|
+
import type { LacunaConfig } from '../config.js';
|
|
3
|
+
export { PRESETS } from './types.js';
|
|
4
|
+
export type { ModelProvider, ChatMessage, ProviderPreset } from './types.js';
|
|
5
|
+
export declare function createProvider(config: LacunaConfig): ModelProvider;
|
|
6
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/lib/providers/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,YAAY,CAAA;AAG/C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,cAAc,CAAA;AAEhD,OAAO,EAAE,OAAO,EAAE,MAAM,YAAY,CAAA;AACpC,YAAY,EAAE,aAAa,EAAE,WAAW,EAAE,cAAc,EAAE,MAAM,YAAY,CAAA;AAE5E,wBAAgB,cAAc,CAAC,MAAM,EAAE,YAAY,GAAG,aAAa,CA6BlE"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { AnthropicProvider } from './anthropic.js';
|
|
2
|
+
import { OpenAICompatibleProvider } from './openai-compatible.js';
|
|
3
|
+
export { PRESETS } from './types.js';
|
|
4
|
+
export function createProvider(config) {
|
|
5
|
+
const apiKey = config.apiKeyEnv ? (process.env[config.apiKeyEnv] ?? '') : '';
|
|
6
|
+
if (config.provider === 'anthropic') {
|
|
7
|
+
if (!apiKey) {
|
|
8
|
+
throw new Error(`${config.apiKeyEnv} environment variable is not set.\nGet your key at https://console.anthropic.com`);
|
|
9
|
+
}
|
|
10
|
+
return new AnthropicProvider(config.model, apiKey);
|
|
11
|
+
}
|
|
12
|
+
if (config.provider === 'openai-compatible') {
|
|
13
|
+
if (!config.baseURL) {
|
|
14
|
+
throw new Error('baseURL is required for openai-compatible provider. Check your .lacuna.json');
|
|
15
|
+
}
|
|
16
|
+
const isLocal = config.baseURL.includes('localhost') || config.baseURL.includes('127.0.0.1');
|
|
17
|
+
if (!isLocal && !apiKey) {
|
|
18
|
+
throw new Error(`${config.apiKeyEnv} environment variable is not set.`);
|
|
19
|
+
}
|
|
20
|
+
return new OpenAICompatibleProvider(config.model, {
|
|
21
|
+
baseURL: config.baseURL,
|
|
22
|
+
apiKey: apiKey || undefined,
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
throw new Error(`Unknown provider: ${config.provider}`);
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/lib/providers/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAA;AAClD,OAAO,EAAE,wBAAwB,EAAE,MAAM,wBAAwB,CAAA;AAGjE,OAAO,EAAE,OAAO,EAAE,MAAM,YAAY,CAAA;AAGpC,MAAM,UAAU,cAAc,CAAC,MAAoB;IACjD,MAAM,MAAM,GAAG,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAA;IAE5E,IAAI,MAAM,CAAC,QAAQ,KAAK,WAAW,EAAE,CAAC;QACpC,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,MAAM,IAAI,KAAK,CACb,GAAG,MAAM,CAAC,SAAS,kFAAkF,CACtG,CAAA;QACH,CAAC;QACD,OAAO,IAAI,iBAAiB,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,CAAA;IACpD,CAAC;IAED,IAAI,MAAM,CAAC,QAAQ,KAAK,mBAAmB,EAAE,CAAC;QAC5C,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpB,MAAM,IAAI,KAAK,CAAC,6EAA6E,CAAC,CAAA;QAChG,CAAC;QACD,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAA;QAC5F,IAAI,CAAC,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CACb,GAAG,MAAM,CAAC,SAAS,mCAAmC,CACvD,CAAA;QACH,CAAC;QACD,OAAO,IAAI,wBAAwB,CAAC,MAAM,CAAC,KAAK,EAAE;YAChD,OAAO,EAAE,MAAM,CAAC,OAAO;YACvB,MAAM,EAAE,MAAM,IAAI,SAAS;SAC5B,CAAC,CAAA;IACJ,CAAC;IAED,MAAM,IAAI,KAAK,CAAC,qBAAqB,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAA;AACzD,CAAC"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import type { ModelProvider, ChatMessage } from './types.js';
|
|
2
|
+
export declare class OpenAICompatibleProvider implements ModelProvider {
|
|
3
|
+
private client;
|
|
4
|
+
private model;
|
|
5
|
+
constructor(model: string, options: {
|
|
6
|
+
baseURL: string;
|
|
7
|
+
apiKey?: string;
|
|
8
|
+
});
|
|
9
|
+
generate(messages: ChatMessage[], system: string, onToken?: (token: string) => void, maxTokens?: number, temperature?: number): Promise<string>;
|
|
10
|
+
}
|
|
11
|
+
//# sourceMappingURL=openai-compatible.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"openai-compatible.d.ts","sourceRoot":"","sources":["../../../src/lib/providers/openai-compatible.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,aAAa,EAAE,WAAW,EAAE,MAAM,YAAY,CAAA;AAE5D,qBAAa,wBAAyB,YAAW,aAAa;IAC5D,OAAO,CAAC,MAAM,CAAQ;IACtB,OAAO,CAAC,KAAK,CAAQ;gBAET,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE;QAAE,OAAO,EAAE,MAAM,CAAC;QAAC,MAAM,CAAC,EAAE,MAAM,CAAA;KAAE;IAqDlE,QAAQ,CACZ,QAAQ,EAAE,WAAW,EAAE,EACvB,MAAM,EAAE,MAAM,EACd,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,EACjC,SAAS,SAAQ,EACjB,WAAW,CAAC,EAAE,MAAM,GACnB,OAAO,CAAC,MAAM,CAAC;CAoCnB"}
|