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.
Files changed (144) hide show
  1. package/README.md +451 -0
  2. package/bin/run.js +5 -0
  3. package/dist/agent/context.d.ts +25 -0
  4. package/dist/agent/context.d.ts.map +1 -0
  5. package/dist/agent/context.js +366 -0
  6. package/dist/agent/context.js.map +1 -0
  7. package/dist/agent/fix-loop.d.ts +20 -0
  8. package/dist/agent/fix-loop.d.ts.map +1 -0
  9. package/dist/agent/fix-loop.js +466 -0
  10. package/dist/agent/fix-loop.js.map +1 -0
  11. package/dist/agent/generator.d.ts +35 -0
  12. package/dist/agent/generator.d.ts.map +1 -0
  13. package/dist/agent/generator.js +220 -0
  14. package/dist/agent/generator.js.map +1 -0
  15. package/dist/agent/loop.d.ts +23 -0
  16. package/dist/agent/loop.d.ts.map +1 -0
  17. package/dist/agent/loop.js +394 -0
  18. package/dist/agent/loop.js.map +1 -0
  19. package/dist/agent/project-memory.d.ts +10 -0
  20. package/dist/agent/project-memory.d.ts.map +1 -0
  21. package/dist/agent/project-memory.js +57 -0
  22. package/dist/agent/project-memory.js.map +1 -0
  23. package/dist/agent/prompts.d.ts +44 -0
  24. package/dist/agent/prompts.d.ts.map +1 -0
  25. package/dist/agent/prompts.js +377 -0
  26. package/dist/agent/prompts.js.map +1 -0
  27. package/dist/ci/comment.d.ts +2 -0
  28. package/dist/ci/comment.d.ts.map +1 -0
  29. package/dist/ci/comment.js +97 -0
  30. package/dist/ci/comment.js.map +1 -0
  31. package/dist/ci/parse-outputs.d.ts +2 -0
  32. package/dist/ci/parse-outputs.d.ts.map +1 -0
  33. package/dist/ci/parse-outputs.js +30 -0
  34. package/dist/ci/parse-outputs.js.map +1 -0
  35. package/dist/commands/analyze.d.ts +13 -0
  36. package/dist/commands/analyze.d.ts.map +1 -0
  37. package/dist/commands/analyze.js +151 -0
  38. package/dist/commands/analyze.js.map +1 -0
  39. package/dist/commands/fix.d.ts +15 -0
  40. package/dist/commands/fix.d.ts.map +1 -0
  41. package/dist/commands/fix.js +106 -0
  42. package/dist/commands/fix.js.map +1 -0
  43. package/dist/commands/generate.d.ts +18 -0
  44. package/dist/commands/generate.d.ts.map +1 -0
  45. package/dist/commands/generate.js +129 -0
  46. package/dist/commands/generate.js.map +1 -0
  47. package/dist/commands/init.d.ts +7 -0
  48. package/dist/commands/init.d.ts.map +1 -0
  49. package/dist/commands/init.js +131 -0
  50. package/dist/commands/init.js.map +1 -0
  51. package/dist/commands/run.d.ts +10 -0
  52. package/dist/commands/run.d.ts.map +1 -0
  53. package/dist/commands/run.js +45 -0
  54. package/dist/commands/run.js.map +1 -0
  55. package/dist/lib/config.d.ts +58 -0
  56. package/dist/lib/config.d.ts.map +1 -0
  57. package/dist/lib/config.js +68 -0
  58. package/dist/lib/config.js.map +1 -0
  59. package/dist/lib/coverage/gaps.d.ts +12 -0
  60. package/dist/lib/coverage/gaps.d.ts.map +1 -0
  61. package/dist/lib/coverage/gaps.js +186 -0
  62. package/dist/lib/coverage/gaps.js.map +1 -0
  63. package/dist/lib/coverage/index.d.ts +7 -0
  64. package/dist/lib/coverage/index.d.ts.map +1 -0
  65. package/dist/lib/coverage/index.js +24 -0
  66. package/dist/lib/coverage/index.js.map +1 -0
  67. package/dist/lib/coverage/json.d.ts +3 -0
  68. package/dist/lib/coverage/json.d.ts.map +1 -0
  69. package/dist/lib/coverage/json.js +24 -0
  70. package/dist/lib/coverage/json.js.map +1 -0
  71. package/dist/lib/coverage/lcov.d.ts +3 -0
  72. package/dist/lib/coverage/lcov.d.ts.map +1 -0
  73. package/dist/lib/coverage/lcov.js +58 -0
  74. package/dist/lib/coverage/lcov.js.map +1 -0
  75. package/dist/lib/coverage/types.d.ts +27 -0
  76. package/dist/lib/coverage/types.d.ts.map +1 -0
  77. package/dist/lib/coverage/types.js +2 -0
  78. package/dist/lib/coverage/types.js.map +1 -0
  79. package/dist/lib/coverage-spinner.d.ts +6 -0
  80. package/dist/lib/coverage-spinner.d.ts.map +1 -0
  81. package/dist/lib/coverage-spinner.js +101 -0
  82. package/dist/lib/coverage-spinner.js.map +1 -0
  83. package/dist/lib/detector.d.ts +13 -0
  84. package/dist/lib/detector.d.ts.map +1 -0
  85. package/dist/lib/detector.js +106 -0
  86. package/dist/lib/detector.js.map +1 -0
  87. package/dist/lib/extract-error.d.ts +2 -0
  88. package/dist/lib/extract-error.d.ts.map +1 -0
  89. package/dist/lib/extract-error.js +116 -0
  90. package/dist/lib/extract-error.js.map +1 -0
  91. package/dist/lib/providers/anthropic.d.ts +8 -0
  92. package/dist/lib/providers/anthropic.d.ts.map +1 -0
  93. package/dist/lib/providers/anthropic.js +38 -0
  94. package/dist/lib/providers/anthropic.js.map +1 -0
  95. package/dist/lib/providers/index.d.ts +6 -0
  96. package/dist/lib/providers/index.d.ts.map +1 -0
  97. package/dist/lib/providers/index.js +27 -0
  98. package/dist/lib/providers/index.js.map +1 -0
  99. package/dist/lib/providers/openai-compatible.d.ts +11 -0
  100. package/dist/lib/providers/openai-compatible.d.ts.map +1 -0
  101. package/dist/lib/providers/openai-compatible.js +93 -0
  102. package/dist/lib/providers/openai-compatible.js.map +1 -0
  103. package/dist/lib/providers/types.d.ts +17 -0
  104. package/dist/lib/providers/types.d.ts.map +1 -0
  105. package/dist/lib/providers/types.js +97 -0
  106. package/dist/lib/providers/types.js.map +1 -0
  107. package/dist/lib/report-upload.d.ts +3 -0
  108. package/dist/lib/report-upload.d.ts.map +1 -0
  109. package/dist/lib/report-upload.js +15 -0
  110. package/dist/lib/report-upload.js.map +1 -0
  111. package/dist/lib/reporter.d.ts +51 -0
  112. package/dist/lib/reporter.d.ts.map +1 -0
  113. package/dist/lib/reporter.js +172 -0
  114. package/dist/lib/reporter.js.map +1 -0
  115. package/dist/lib/runner.d.ts +9 -0
  116. package/dist/lib/runner.d.ts.map +1 -0
  117. package/dist/lib/runner.js +50 -0
  118. package/dist/lib/runner.js.map +1 -0
  119. package/dist/lib/skeleton.d.ts +8 -0
  120. package/dist/lib/skeleton.d.ts.map +1 -0
  121. package/dist/lib/skeleton.js +122 -0
  122. package/dist/lib/skeleton.js.map +1 -0
  123. package/dist/lib/streaming-viewer.d.ts +14 -0
  124. package/dist/lib/streaming-viewer.d.ts.map +1 -0
  125. package/dist/lib/streaming-viewer.js +80 -0
  126. package/dist/lib/streaming-viewer.js.map +1 -0
  127. package/dist/lib/tips.d.ts +16 -0
  128. package/dist/lib/tips.d.ts.map +1 -0
  129. package/dist/lib/tips.js +76 -0
  130. package/dist/lib/tips.js.map +1 -0
  131. package/dist/lib/typecheck.d.ts +3 -0
  132. package/dist/lib/typecheck.d.ts.map +1 -0
  133. package/dist/lib/typecheck.js +28 -0
  134. package/dist/lib/typecheck.js.map +1 -0
  135. package/dist/lib/validate.d.ts +7 -0
  136. package/dist/lib/validate.d.ts.map +1 -0
  137. package/dist/lib/validate.js +82 -0
  138. package/dist/lib/validate.js.map +1 -0
  139. package/dist/lib/worker-display.d.ts +45 -0
  140. package/dist/lib/worker-display.d.ts.map +1 -0
  141. package/dist/lib/worker-display.js +168 -0
  142. package/dist/lib/worker-display.js.map +1 -0
  143. package/oclif.manifest.json +295 -0
  144. 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,3 @@
1
+ import type { CoverageReport } from './types.js';
2
+ export declare function parseLcov(coverageDir: string, cwd?: string): Promise<CoverageReport>;
3
+ //# sourceMappingURL=lcov.d.ts.map
@@ -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,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/lib/coverage/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,6 @@
1
+ export interface CoverageSpinner {
2
+ onLine: (line: string) => void;
3
+ stop: () => void;
4
+ }
5
+ export declare function startCoverageSpinner(label: string, runner?: string): CoverageSpinner;
6
+ //# sourceMappingURL=coverage-spinner.d.ts.map
@@ -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,2 @@
1
+ export declare function extractTestFailure(rawOutput: string): string;
2
+ //# sourceMappingURL=extract-error.d.ts.map
@@ -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"}