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,93 @@
|
|
|
1
|
+
import { gunzipSync } from 'node:zlib';
|
|
2
|
+
import OpenAI from 'openai';
|
|
3
|
+
export class OpenAICompatibleProvider {
|
|
4
|
+
client;
|
|
5
|
+
model;
|
|
6
|
+
constructor(model, options) {
|
|
7
|
+
this.client = new OpenAI({
|
|
8
|
+
apiKey: options.apiKey || 'no-key-required',
|
|
9
|
+
baseURL: options.baseURL,
|
|
10
|
+
// For error responses, manually decompress gzip and strip the
|
|
11
|
+
// content-encoding header so the SDK always receives a plain-text body.
|
|
12
|
+
// Some providers (e.g. DeepSeek on GCP) return gzip-encoded 4xx bodies
|
|
13
|
+
// but ignore Accept-Encoding: identity, making error messages unreadable.
|
|
14
|
+
fetch: async (url, init) => {
|
|
15
|
+
const response = await globalThis.fetch(url, init);
|
|
16
|
+
if (response.ok)
|
|
17
|
+
return response;
|
|
18
|
+
// Decode error response body regardless of compression.
|
|
19
|
+
// Some providers (e.g. Gemini) set content-encoding: gzip but don't actually
|
|
20
|
+
// gzip the body; we try gunzip and fall back to raw UTF-8 if it fails.
|
|
21
|
+
const encoding = response.headers.get('content-encoding') ?? '';
|
|
22
|
+
const raw = Buffer.from(await response.arrayBuffer());
|
|
23
|
+
let bodyText;
|
|
24
|
+
try {
|
|
25
|
+
bodyText = encoding === 'gzip' ? gunzipSync(raw).toString('utf-8') : raw.toString('utf-8');
|
|
26
|
+
}
|
|
27
|
+
catch {
|
|
28
|
+
bodyText = raw.toString('utf-8');
|
|
29
|
+
}
|
|
30
|
+
// Normalize non-OpenAI error shapes to {error:{message,type,code}} so the
|
|
31
|
+
// SDK can extract the message. Google's format is [{error:{code,message,status}}].
|
|
32
|
+
let normalized = bodyText;
|
|
33
|
+
try {
|
|
34
|
+
const parsed = JSON.parse(bodyText);
|
|
35
|
+
const obj = Array.isArray(parsed) ? parsed[0] : parsed;
|
|
36
|
+
const err = obj?.['error'];
|
|
37
|
+
if (err) {
|
|
38
|
+
normalized = JSON.stringify({
|
|
39
|
+
error: {
|
|
40
|
+
message: String(err['message'] ?? err['code'] ?? 'unknown error'),
|
|
41
|
+
type: String(err['status'] ?? err['type'] ?? 'api_error'),
|
|
42
|
+
code: String(err['code'] ?? err['reason'] ?? ''),
|
|
43
|
+
},
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
catch {
|
|
48
|
+
// not JSON — leave bodyText as-is; SDK will surface it in e.message
|
|
49
|
+
}
|
|
50
|
+
const newHeaders = new Headers(response.headers);
|
|
51
|
+
newHeaders.delete('content-encoding');
|
|
52
|
+
newHeaders.set('content-length', String(Buffer.byteLength(normalized, 'utf-8')));
|
|
53
|
+
return new Response(normalized, { status: response.status, statusText: response.statusText, headers: newHeaders });
|
|
54
|
+
},
|
|
55
|
+
});
|
|
56
|
+
this.model = model;
|
|
57
|
+
}
|
|
58
|
+
async generate(messages, system, onToken, maxTokens = 16000, temperature) {
|
|
59
|
+
let content = '';
|
|
60
|
+
try {
|
|
61
|
+
const stream = await this.client.chat.completions.create({
|
|
62
|
+
model: this.model,
|
|
63
|
+
max_tokens: maxTokens,
|
|
64
|
+
stop: ['</code_output>'],
|
|
65
|
+
...(temperature !== undefined ? { temperature } : {}),
|
|
66
|
+
stream: true,
|
|
67
|
+
messages: [
|
|
68
|
+
{ role: 'system', content: system },
|
|
69
|
+
...messages.map((m) => ({ role: m.role, content: m.content })),
|
|
70
|
+
],
|
|
71
|
+
});
|
|
72
|
+
for await (const chunk of stream) {
|
|
73
|
+
const token = chunk.choices[0]?.delta?.content ?? '';
|
|
74
|
+
if (token) {
|
|
75
|
+
content += token;
|
|
76
|
+
onToken?.(token);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
catch (err) {
|
|
81
|
+
if (err != null && typeof err === 'object' && 'status' in err) {
|
|
82
|
+
const e = err;
|
|
83
|
+
const body = e.error?.message
|
|
84
|
+
? `${e.error.message}${e.error.type ? ` (type: ${e.error.type})` : ''}${e.error.code ? ` [${e.error.code}]` : ''}`
|
|
85
|
+
: (e.message ?? 'no message');
|
|
86
|
+
throw new Error(`${this.model} API error (HTTP ${e.status ?? '?'}): ${body}`);
|
|
87
|
+
}
|
|
88
|
+
throw err;
|
|
89
|
+
}
|
|
90
|
+
return content.trim();
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
//# sourceMappingURL=openai-compatible.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"openai-compatible.js","sourceRoot":"","sources":["../../../src/lib/providers/openai-compatible.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
export interface ChatMessage {
|
|
2
|
+
role: 'user' | 'assistant';
|
|
3
|
+
content: string;
|
|
4
|
+
}
|
|
5
|
+
export interface ModelProvider {
|
|
6
|
+
generate(messages: ChatMessage[], system: string, onToken?: (token: string) => void, maxTokens?: number, temperature?: number): Promise<string>;
|
|
7
|
+
}
|
|
8
|
+
export interface ProviderPreset {
|
|
9
|
+
label: string;
|
|
10
|
+
provider: 'anthropic' | 'openai-compatible';
|
|
11
|
+
model: string;
|
|
12
|
+
baseURL?: string;
|
|
13
|
+
apiKeyEnv: string;
|
|
14
|
+
apiKeyHint: string;
|
|
15
|
+
}
|
|
16
|
+
export declare const PRESETS: Record<string, ProviderPreset>;
|
|
17
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../src/lib/providers/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,GAAG,WAAW,CAAA;IAC1B,OAAO,EAAE,MAAM,CAAA;CAChB;AAED,MAAM,WAAW,aAAa;IAC5B,QAAQ,CACN,QAAQ,EAAE,WAAW,EAAE,EACvB,MAAM,EAAE,MAAM,EACd,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,EACjC,SAAS,CAAC,EAAE,MAAM,EAClB,WAAW,CAAC,EAAE,MAAM,GACnB,OAAO,CAAC,MAAM,CAAC,CAAA;CACnB;AAED,MAAM,WAAW,cAAc;IAC7B,KAAK,EAAE,MAAM,CAAA;IACb,QAAQ,EAAE,WAAW,GAAG,mBAAmB,CAAA;IAC3C,KAAK,EAAE,MAAM,CAAA;IACb,OAAO,CAAC,EAAE,MAAM,CAAA;IAChB,SAAS,EAAE,MAAM,CAAA;IACjB,UAAU,EAAE,MAAM,CAAA;CACnB;AAED,eAAO,MAAM,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,cAAc,CA+FlD,CAAA"}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
export const PRESETS = {
|
|
2
|
+
claude: {
|
|
3
|
+
label: 'Claude (Anthropic) — claude-sonnet-4-6',
|
|
4
|
+
provider: 'anthropic',
|
|
5
|
+
model: 'claude-sonnet-4-6',
|
|
6
|
+
apiKeyEnv: 'ANTHROPIC_API_KEY',
|
|
7
|
+
apiKeyHint: 'https://console.anthropic.com',
|
|
8
|
+
},
|
|
9
|
+
'claude-opus': {
|
|
10
|
+
label: 'Claude Opus (Anthropic) — claude-opus-4-7',
|
|
11
|
+
provider: 'anthropic',
|
|
12
|
+
model: 'claude-opus-4-7',
|
|
13
|
+
apiKeyEnv: 'ANTHROPIC_API_KEY',
|
|
14
|
+
apiKeyHint: 'https://console.anthropic.com',
|
|
15
|
+
},
|
|
16
|
+
deepseek: {
|
|
17
|
+
label: 'DeepSeek — deepseek-chat',
|
|
18
|
+
provider: 'openai-compatible',
|
|
19
|
+
model: 'deepseek-chat',
|
|
20
|
+
baseURL: 'https://api.deepseek.com/v1',
|
|
21
|
+
apiKeyEnv: 'DEEPSEEK_API_KEY',
|
|
22
|
+
apiKeyHint: 'https://platform.deepseek.com',
|
|
23
|
+
},
|
|
24
|
+
'deepseek-r1': {
|
|
25
|
+
label: 'DeepSeek R1 (reasoning) — deepseek-reasoner',
|
|
26
|
+
provider: 'openai-compatible',
|
|
27
|
+
model: 'deepseek-reasoner',
|
|
28
|
+
baseURL: 'https://api.deepseek.com/v1',
|
|
29
|
+
apiKeyEnv: 'DEEPSEEK_API_KEY',
|
|
30
|
+
apiKeyHint: 'https://platform.deepseek.com',
|
|
31
|
+
},
|
|
32
|
+
'gpt-4o': {
|
|
33
|
+
label: 'GPT-4o (OpenAI)',
|
|
34
|
+
provider: 'openai-compatible',
|
|
35
|
+
model: 'gpt-4o',
|
|
36
|
+
baseURL: 'https://api.openai.com/v1',
|
|
37
|
+
apiKeyEnv: 'OPENAI_API_KEY',
|
|
38
|
+
apiKeyHint: 'https://platform.openai.com/api-keys',
|
|
39
|
+
},
|
|
40
|
+
groq: {
|
|
41
|
+
label: 'Groq — Llama 3.3 70B (fast & free tier)',
|
|
42
|
+
provider: 'openai-compatible',
|
|
43
|
+
model: 'llama-3.3-70b-versatile',
|
|
44
|
+
baseURL: 'https://api.groq.com/openai/v1',
|
|
45
|
+
apiKeyEnv: 'GROQ_API_KEY',
|
|
46
|
+
apiKeyHint: 'https://console.groq.com',
|
|
47
|
+
},
|
|
48
|
+
openrouter: {
|
|
49
|
+
label: 'OpenRouter — any model, one API key',
|
|
50
|
+
provider: 'openai-compatible',
|
|
51
|
+
model: 'anthropic/claude-sonnet-4-6',
|
|
52
|
+
baseURL: 'https://openrouter.ai/api/v1',
|
|
53
|
+
apiKeyEnv: 'OPENROUTER_API_KEY',
|
|
54
|
+
apiKeyHint: 'https://openrouter.ai/keys',
|
|
55
|
+
},
|
|
56
|
+
ollama: {
|
|
57
|
+
label: 'Ollama — local models (no API key needed)',
|
|
58
|
+
provider: 'openai-compatible',
|
|
59
|
+
model: 'llama3.2',
|
|
60
|
+
baseURL: 'http://localhost:11434/v1',
|
|
61
|
+
apiKeyEnv: '',
|
|
62
|
+
apiKeyHint: 'Run: ollama pull llama3.2',
|
|
63
|
+
},
|
|
64
|
+
'lm-studio': {
|
|
65
|
+
label: 'LM Studio — local models (no API key needed)',
|
|
66
|
+
provider: 'openai-compatible',
|
|
67
|
+
model: 'local-model',
|
|
68
|
+
baseURL: 'http://localhost:1234/v1',
|
|
69
|
+
apiKeyEnv: '',
|
|
70
|
+
apiKeyHint: 'Start LM Studio server on port 1234',
|
|
71
|
+
},
|
|
72
|
+
gemini: {
|
|
73
|
+
label: 'Gemini 2.5 Pro (Google)',
|
|
74
|
+
provider: 'openai-compatible',
|
|
75
|
+
model: 'gemini-2.5-pro',
|
|
76
|
+
baseURL: 'https://generativelanguage.googleapis.com/v1beta/openai/',
|
|
77
|
+
apiKeyEnv: 'GEMINI_API_KEY',
|
|
78
|
+
apiKeyHint: 'https://aistudio.google.com/apikey',
|
|
79
|
+
},
|
|
80
|
+
'gemini-flash': {
|
|
81
|
+
label: 'Gemini 2.5 Flash (Google) — fast & cheap',
|
|
82
|
+
provider: 'openai-compatible',
|
|
83
|
+
model: 'gemini-2.5-flash',
|
|
84
|
+
baseURL: 'https://generativelanguage.googleapis.com/v1beta/openai/',
|
|
85
|
+
apiKeyEnv: 'GEMINI_API_KEY',
|
|
86
|
+
apiKeyHint: 'https://aistudio.google.com/apikey',
|
|
87
|
+
},
|
|
88
|
+
custom: {
|
|
89
|
+
label: 'Custom — any OpenAI-compatible endpoint',
|
|
90
|
+
provider: 'openai-compatible',
|
|
91
|
+
model: '',
|
|
92
|
+
baseURL: '',
|
|
93
|
+
apiKeyEnv: 'LLM_API_KEY',
|
|
94
|
+
apiKeyHint: 'Your provider docs',
|
|
95
|
+
},
|
|
96
|
+
};
|
|
97
|
+
//# sourceMappingURL=types.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/lib/providers/types.ts"],"names":[],"mappings":"AAwBA,MAAM,CAAC,MAAM,OAAO,GAAmC;IACrD,MAAM,EAAE;QACN,KAAK,EAAE,wCAAwC;QAC/C,QAAQ,EAAE,WAAW;QACrB,KAAK,EAAE,mBAAmB;QAC1B,SAAS,EAAE,mBAAmB;QAC9B,UAAU,EAAE,+BAA+B;KAC5C;IACD,aAAa,EAAE;QACb,KAAK,EAAE,2CAA2C;QAClD,QAAQ,EAAE,WAAW;QACrB,KAAK,EAAE,iBAAiB;QACxB,SAAS,EAAE,mBAAmB;QAC9B,UAAU,EAAE,+BAA+B;KAC5C;IACD,QAAQ,EAAE;QACR,KAAK,EAAE,0BAA0B;QACjC,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,eAAe;QACtB,OAAO,EAAE,6BAA6B;QACtC,SAAS,EAAE,kBAAkB;QAC7B,UAAU,EAAE,+BAA+B;KAC5C;IACD,aAAa,EAAE;QACb,KAAK,EAAE,6CAA6C;QACpD,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,mBAAmB;QAC1B,OAAO,EAAE,6BAA6B;QACtC,SAAS,EAAE,kBAAkB;QAC7B,UAAU,EAAE,+BAA+B;KAC5C;IACD,QAAQ,EAAE;QACR,KAAK,EAAE,iBAAiB;QACxB,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,QAAQ;QACf,OAAO,EAAE,2BAA2B;QACpC,SAAS,EAAE,gBAAgB;QAC3B,UAAU,EAAE,sCAAsC;KACnD;IACD,IAAI,EAAE;QACJ,KAAK,EAAE,yCAAyC;QAChD,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,yBAAyB;QAChC,OAAO,EAAE,gCAAgC;QACzC,SAAS,EAAE,cAAc;QACzB,UAAU,EAAE,0BAA0B;KACvC;IACD,UAAU,EAAE;QACV,KAAK,EAAE,qCAAqC;QAC5C,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,6BAA6B;QACpC,OAAO,EAAE,8BAA8B;QACvC,SAAS,EAAE,oBAAoB;QAC/B,UAAU,EAAE,4BAA4B;KACzC;IACD,MAAM,EAAE;QACN,KAAK,EAAE,2CAA2C;QAClD,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,UAAU;QACjB,OAAO,EAAE,2BAA2B;QACpC,SAAS,EAAE,EAAE;QACb,UAAU,EAAE,2BAA2B;KACxC;IACD,WAAW,EAAE;QACX,KAAK,EAAE,8CAA8C;QACrD,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,aAAa;QACpB,OAAO,EAAE,0BAA0B;QACnC,SAAS,EAAE,EAAE;QACb,UAAU,EAAE,qCAAqC;KAClD;IACD,MAAM,EAAE;QACN,KAAK,EAAE,yBAAyB;QAChC,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,gBAAgB;QACvB,OAAO,EAAE,0DAA0D;QACnE,SAAS,EAAE,gBAAgB;QAC3B,UAAU,EAAE,oCAAoC;KACjD;IACD,cAAc,EAAE;QACd,KAAK,EAAE,0CAA0C;QACjD,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,kBAAkB;QACzB,OAAO,EAAE,0DAA0D;QACnE,SAAS,EAAE,gBAAgB;QAC3B,UAAU,EAAE,oCAAoC;KACjD;IACD,MAAM,EAAE;QACN,KAAK,EAAE,yCAAyC;QAChD,QAAQ,EAAE,mBAAmB;QAC7B,KAAK,EAAE,EAAE;QACT,OAAO,EAAE,EAAE;QACX,SAAS,EAAE,aAAa;QACxB,UAAU,EAAE,oBAAoB;KACjC;CACF,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"report-upload.d.ts","sourceRoot":"","sources":["../../src/lib/report-upload.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,eAAe,CAAA;AAE/C,wBAAsB,YAAY,CAAC,SAAS,EAAE,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAcvG"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
export async function uploadReport(serverUrl, report, apiKey) {
|
|
2
|
+
const res = await fetch(`${serverUrl}/api/runs`, {
|
|
3
|
+
method: 'POST',
|
|
4
|
+
headers: {
|
|
5
|
+
'Content-Type': 'application/json',
|
|
6
|
+
Authorization: `Bearer ${apiKey}`,
|
|
7
|
+
},
|
|
8
|
+
body: JSON.stringify(report),
|
|
9
|
+
});
|
|
10
|
+
if (!res.ok) {
|
|
11
|
+
const text = await res.text().catch(() => res.statusText);
|
|
12
|
+
throw new Error(`Failed to upload report (${res.status}): ${text}`);
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
//# sourceMappingURL=report-upload.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"report-upload.js","sourceRoot":"","sources":["../../src/lib/report-upload.ts"],"names":[],"mappings":"AAEA,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,SAAiB,EAAE,MAAkB,EAAE,MAAc;IACtF,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,GAAG,SAAS,WAAW,EAAE;QAC/C,MAAM,EAAE,MAAM;QACd,OAAO,EAAE;YACP,cAAc,EAAE,kBAAkB;YAClC,aAAa,EAAE,UAAU,MAAM,EAAE;SAClC;QACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC;KAC7B,CAAC,CAAA;IAEF,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;QACZ,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC,UAAU,CAAC,CAAA;QACzD,MAAM,IAAI,KAAK,CAAC,4BAA4B,GAAG,CAAC,MAAM,MAAM,IAAI,EAAE,CAAC,CAAA;IACrE,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import type { LoopResult } from '../agent/loop.js';
|
|
2
|
+
import type { CoverageGap } from './coverage/types.js';
|
|
3
|
+
export interface AnalyzeResult {
|
|
4
|
+
testRunner: string;
|
|
5
|
+
language: string;
|
|
6
|
+
threshold: number;
|
|
7
|
+
coveragePct: number;
|
|
8
|
+
functionCoveragePct: number;
|
|
9
|
+
gaps: CoverageGap[];
|
|
10
|
+
untouchedCount: number;
|
|
11
|
+
passed: boolean;
|
|
12
|
+
}
|
|
13
|
+
export interface ReportInput {
|
|
14
|
+
type: 'analyze' | 'generate';
|
|
15
|
+
threshold: number;
|
|
16
|
+
analyze?: AnalyzeResult;
|
|
17
|
+
generate?: LoopResult;
|
|
18
|
+
timestamp?: string;
|
|
19
|
+
untouchedCount: number;
|
|
20
|
+
}
|
|
21
|
+
export declare function reportTerminal(input: ReportInput): void;
|
|
22
|
+
export interface JsonReport {
|
|
23
|
+
lacuna: string;
|
|
24
|
+
timestamp: string;
|
|
25
|
+
type: 'analyze' | 'generate';
|
|
26
|
+
threshold: number;
|
|
27
|
+
passed: boolean;
|
|
28
|
+
coverage: {
|
|
29
|
+
before?: number;
|
|
30
|
+
after?: number;
|
|
31
|
+
lines?: number;
|
|
32
|
+
functions?: number;
|
|
33
|
+
};
|
|
34
|
+
filesProcessed?: number;
|
|
35
|
+
testsWritten?: number;
|
|
36
|
+
gaps?: Array<{
|
|
37
|
+
file: string;
|
|
38
|
+
uncoveredFunctions: string[];
|
|
39
|
+
uncoveredLines: number[];
|
|
40
|
+
}>;
|
|
41
|
+
errors?: string[];
|
|
42
|
+
}
|
|
43
|
+
export declare function buildJsonReport(input: ReportInput): JsonReport;
|
|
44
|
+
export declare function buildMarkdownReport(input: ReportInput): string;
|
|
45
|
+
export declare const EXIT: {
|
|
46
|
+
readonly OK: 0;
|
|
47
|
+
readonly BELOW_THRESHOLD: 1;
|
|
48
|
+
readonly ERROR: 2;
|
|
49
|
+
};
|
|
50
|
+
export declare function getExitCode(input: ReportInput): number;
|
|
51
|
+
//# sourceMappingURL=reporter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"reporter.d.ts","sourceRoot":"","sources":["../../src/lib/reporter.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAA;AAClD,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAA;AAEtD,MAAM,WAAW,aAAa;IAC5B,UAAU,EAAE,MAAM,CAAA;IAClB,QAAQ,EAAE,MAAM,CAAA;IAChB,SAAS,EAAE,MAAM,CAAA;IACjB,WAAW,EAAE,MAAM,CAAA;IACnB,mBAAmB,EAAE,MAAM,CAAA;IAC3B,IAAI,EAAE,WAAW,EAAE,CAAA;IACnB,cAAc,EAAE,MAAM,CAAA;IACtB,MAAM,EAAE,OAAO,CAAA;CAChB;AAED,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,SAAS,GAAG,UAAU,CAAA;IAC5B,SAAS,EAAE,MAAM,CAAA;IACjB,OAAO,CAAC,EAAE,aAAa,CAAA;IACvB,QAAQ,CAAC,EAAE,UAAU,CAAA;IACrB,SAAS,CAAC,EAAE,MAAM,CAAA;IAClB,cAAc,EAAE,MAAM,CAAA;CACvB;AAID,wBAAgB,cAAc,CAAC,KAAK,EAAE,WAAW,GAAG,IAAI,CAuEvD;AAID,MAAM,WAAW,UAAU;IACzB,MAAM,EAAE,MAAM,CAAA;IACd,SAAS,EAAE,MAAM,CAAA;IACjB,IAAI,EAAE,SAAS,GAAG,UAAU,CAAA;IAC5B,SAAS,EAAE,MAAM,CAAA;IACjB,MAAM,EAAE,OAAO,CAAA;IACf,QAAQ,EAAE;QACR,MAAM,CAAC,EAAE,MAAM,CAAA;QACf,KAAK,CAAC,EAAE,MAAM,CAAA;QACd,KAAK,CAAC,EAAE,MAAM,CAAA;QACd,SAAS,CAAC,EAAE,MAAM,CAAA;KACnB,CAAA;IACD,cAAc,CAAC,EAAE,MAAM,CAAA;IACvB,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,IAAI,CAAC,EAAE,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,kBAAkB,EAAE,MAAM,EAAE,CAAC;QAAC,cAAc,EAAE,MAAM,EAAE,CAAA;KAAE,CAAC,CAAA;IACtF,MAAM,CAAC,EAAE,MAAM,EAAE,CAAA;CAClB;AAED,wBAAgB,eAAe,CAAC,KAAK,EAAE,WAAW,GAAG,UAAU,CAoC9D;AAID,wBAAgB,mBAAmB,CAAC,KAAK,EAAE,WAAW,GAAG,MAAM,CAyD9D;AAID,eAAO,MAAM,IAAI;;;;CAIP,CAAA;AAEV,wBAAgB,WAAW,CAAC,KAAK,EAAE,WAAW,GAAG,MAAM,CActD"}
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
import chalk from 'chalk';
|
|
2
|
+
// ─── Terminal ────────────────────────────────────────────────────────────────
|
|
3
|
+
export function reportTerminal(input) {
|
|
4
|
+
const { threshold } = input;
|
|
5
|
+
if (input.type === 'analyze' && input.analyze) {
|
|
6
|
+
const r = input.analyze;
|
|
7
|
+
const lineColor = r.passed ? chalk.green : chalk.red;
|
|
8
|
+
const status = r.passed ? chalk.green('PASS') : chalk.red('FAIL');
|
|
9
|
+
console.log(chalk.bold('\nCoverage Summary'));
|
|
10
|
+
console.log(` Lines: ${lineColor(r.coveragePct.toFixed(1) + '%')} Functions: ${lineColor(r.functionCoveragePct.toFixed(1) + '%')}`);
|
|
11
|
+
console.log(` Threshold: ${threshold}% Status: ${status}\n`);
|
|
12
|
+
if (r.gaps.length === 0) {
|
|
13
|
+
console.log(chalk.green(' All files meet the threshold.'));
|
|
14
|
+
return;
|
|
15
|
+
}
|
|
16
|
+
const belowThreshold = r.gaps.length - r.untouchedCount;
|
|
17
|
+
const parts = [];
|
|
18
|
+
if (belowThreshold > 0)
|
|
19
|
+
parts.push(`${belowThreshold} below ${r.threshold}%`);
|
|
20
|
+
if (r.untouchedCount > 0)
|
|
21
|
+
parts.push(`${r.untouchedCount} with no tests yet`);
|
|
22
|
+
console.log(chalk.yellow(` ${r.gaps.length} testable file(s) need attention — ${parts.join(', ')}:\n`));
|
|
23
|
+
for (const gap of r.gaps) {
|
|
24
|
+
const short = gap.filePath.replace(process.cwd() + '/', '');
|
|
25
|
+
console.log(` ${chalk.cyan(short)}`);
|
|
26
|
+
if (gap.uncoveredFunctions.length > 0) {
|
|
27
|
+
console.log(chalk.dim(` functions: ${gap.uncoveredFunctions.join(', ')}`));
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
console.log(`\n Run ${chalk.cyan('lacuna generate')} to write tests for ${r.gaps.length} file(s).\n`);
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
if (input.type === 'generate' && input.generate) {
|
|
34
|
+
const r = input.generate;
|
|
35
|
+
console.log(chalk.bold('\n─── Results ───────────────────────────────'));
|
|
36
|
+
console.log(` Files processed : ${r.filesProcessed}`);
|
|
37
|
+
console.log(` Tests written : ${r.testsWritten}`);
|
|
38
|
+
if (r.hasCoverage) {
|
|
39
|
+
const delta = r.coverageAfter - r.coverageBefore;
|
|
40
|
+
const deltaStr = (delta >= 0 ? '+' : '') + delta.toFixed(1) + '%';
|
|
41
|
+
const afterColor = r.coverageAfter >= threshold ? chalk.green : chalk.yellow;
|
|
42
|
+
console.log(` Coverage : ${chalk.dim(r.coverageBefore.toFixed(1) + '%')} → ${afterColor(r.coverageAfter.toFixed(1) + '%')} (${delta >= 0 ? chalk.green(deltaStr) : chalk.red(deltaStr)})`);
|
|
43
|
+
console.log(` Threshold : ${threshold}% ${r.coverageAfter >= threshold ? chalk.green('PASS') : chalk.red('FAIL')}`);
|
|
44
|
+
}
|
|
45
|
+
else {
|
|
46
|
+
const allPassed = r.testsWritten === r.filesProcessed && r.errors.length === 0;
|
|
47
|
+
console.log(` Coverage : ${chalk.dim('n/a')} (single-file mode — no suite run)`);
|
|
48
|
+
console.log(` Status : ${allPassed ? chalk.green('PASS') : chalk.red('FAIL')}`);
|
|
49
|
+
}
|
|
50
|
+
if (r.errors.length > 0) {
|
|
51
|
+
console.log(chalk.red(`\n ${r.errors.length} file(s) could not be fixed:`));
|
|
52
|
+
for (const err of r.errors) {
|
|
53
|
+
const lines = err.split('\n').filter(Boolean).slice(0, 8);
|
|
54
|
+
for (const line of lines) {
|
|
55
|
+
console.log(chalk.dim(` ${line}`));
|
|
56
|
+
}
|
|
57
|
+
console.log('');
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
console.log('');
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
export function buildJsonReport(input) {
|
|
64
|
+
const timestamp = input.timestamp ?? new Date().toISOString();
|
|
65
|
+
if (input.type === 'analyze' && input.analyze) {
|
|
66
|
+
const r = input.analyze;
|
|
67
|
+
return {
|
|
68
|
+
lacuna: '0.1.0',
|
|
69
|
+
timestamp,
|
|
70
|
+
type: 'analyze',
|
|
71
|
+
threshold: input.threshold,
|
|
72
|
+
passed: r.passed,
|
|
73
|
+
coverage: { lines: r.coveragePct, functions: r.functionCoveragePct },
|
|
74
|
+
gaps: r.gaps.map((g) => ({
|
|
75
|
+
file: g.filePath,
|
|
76
|
+
uncoveredFunctions: g.uncoveredFunctions,
|
|
77
|
+
uncoveredLines: g.uncoveredLines,
|
|
78
|
+
})),
|
|
79
|
+
errors: [],
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
const r = input.generate;
|
|
83
|
+
const passed = r.hasCoverage
|
|
84
|
+
? r.coverageAfter >= input.threshold
|
|
85
|
+
: r.testsWritten === r.filesProcessed && r.errors.length === 0;
|
|
86
|
+
return {
|
|
87
|
+
lacuna: '0.1.0',
|
|
88
|
+
timestamp,
|
|
89
|
+
type: 'generate',
|
|
90
|
+
threshold: input.threshold,
|
|
91
|
+
passed,
|
|
92
|
+
coverage: r.hasCoverage ? { before: r.coverageBefore, after: r.coverageAfter } : {},
|
|
93
|
+
filesProcessed: r.filesProcessed,
|
|
94
|
+
testsWritten: r.testsWritten,
|
|
95
|
+
errors: r.errors,
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
// ─── Markdown ────────────────────────────────────────────────────────────────
|
|
99
|
+
export function buildMarkdownReport(input) {
|
|
100
|
+
const lines = [];
|
|
101
|
+
const { threshold } = input;
|
|
102
|
+
lines.push('## lacuna Coverage Report');
|
|
103
|
+
lines.push('');
|
|
104
|
+
if (input.type === 'analyze' && input.analyze) {
|
|
105
|
+
const r = input.analyze;
|
|
106
|
+
const status = r.passed ? '✅ Pass' : '❌ Fail';
|
|
107
|
+
lines.push(`| Metric | Value |`);
|
|
108
|
+
lines.push(`|--------|-------|`);
|
|
109
|
+
lines.push(`| Line coverage | ${r.coveragePct.toFixed(1)}% |`);
|
|
110
|
+
lines.push(`| Function coverage | ${r.functionCoveragePct.toFixed(1)}% |`);
|
|
111
|
+
lines.push(`| Threshold | ${threshold}% |`);
|
|
112
|
+
lines.push(`| Status | ${status} |`);
|
|
113
|
+
if (r.gaps.length > 0) {
|
|
114
|
+
lines.push('');
|
|
115
|
+
lines.push('### Files below threshold');
|
|
116
|
+
for (const gap of r.gaps) {
|
|
117
|
+
const short = gap.filePath.replace(process.cwd() + '/', '');
|
|
118
|
+
lines.push(`- \`${short}\` — uncovered: ${gap.uncoveredFunctions.join(', ') || `${gap.uncoveredLines.length} lines`}`);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
if (input.type === 'generate' && input.generate) {
|
|
123
|
+
const r = input.generate;
|
|
124
|
+
const delta = r.coverageAfter - r.coverageBefore;
|
|
125
|
+
const deltaStr = (delta >= 0 ? '+' : '') + delta.toFixed(1) + '%';
|
|
126
|
+
const status = r.coverageAfter >= threshold ? '✅ Pass' : '❌ Below threshold';
|
|
127
|
+
lines.push(`| Metric | Value |`);
|
|
128
|
+
lines.push(`|--------|-------|`);
|
|
129
|
+
lines.push(`| Coverage before | ${r.coverageBefore.toFixed(1)}% |`);
|
|
130
|
+
lines.push(`| Coverage after | ${r.coverageAfter.toFixed(1)}% |`);
|
|
131
|
+
lines.push(`| Delta | ${deltaStr} |`);
|
|
132
|
+
lines.push(`| Threshold | ${threshold}% |`);
|
|
133
|
+
lines.push(`| Files processed | ${r.filesProcessed} |`);
|
|
134
|
+
lines.push(`| Tests written | ${r.testsWritten} |`);
|
|
135
|
+
lines.push(`| Status | ${status} |`);
|
|
136
|
+
if (r.errors.length > 0) {
|
|
137
|
+
lines.push('');
|
|
138
|
+
lines.push('### Errors');
|
|
139
|
+
for (const err of r.errors) {
|
|
140
|
+
const summary = err.split('\n').filter(Boolean).slice(0, 3).join(' | ');
|
|
141
|
+
lines.push(`- ${summary}`);
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
lines.push('');
|
|
146
|
+
lines.push(`> Generated by [lacuna](https://github.com/lacuna-dev/lacuna)`);
|
|
147
|
+
return lines.join('\n');
|
|
148
|
+
}
|
|
149
|
+
// ─── Exit codes ──────────────────────────────────────────────────────────────
|
|
150
|
+
export const EXIT = {
|
|
151
|
+
OK: 0,
|
|
152
|
+
BELOW_THRESHOLD: 1,
|
|
153
|
+
ERROR: 2,
|
|
154
|
+
};
|
|
155
|
+
export function getExitCode(input) {
|
|
156
|
+
if (input.type === 'analyze') {
|
|
157
|
+
return input.analyze?.passed ? EXIT.OK : EXIT.BELOW_THRESHOLD;
|
|
158
|
+
}
|
|
159
|
+
if (input.type === 'generate') {
|
|
160
|
+
const r = input.generate;
|
|
161
|
+
if (!r)
|
|
162
|
+
return EXIT.ERROR;
|
|
163
|
+
if (r.errors.length > 0)
|
|
164
|
+
return EXIT.ERROR;
|
|
165
|
+
if (!r.hasCoverage) {
|
|
166
|
+
return r.testsWritten === r.filesProcessed ? EXIT.OK : EXIT.BELOW_THRESHOLD;
|
|
167
|
+
}
|
|
168
|
+
return r.coverageAfter >= input.threshold ? EXIT.OK : EXIT.BELOW_THRESHOLD;
|
|
169
|
+
}
|
|
170
|
+
return EXIT.ERROR;
|
|
171
|
+
}
|
|
172
|
+
//# sourceMappingURL=reporter.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"reporter.js","sourceRoot":"","sources":["../../src/lib/reporter.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
export interface RunResult {
|
|
2
|
+
stdout: string;
|
|
3
|
+
stderr: string;
|
|
4
|
+
exitCode: number;
|
|
5
|
+
success: boolean;
|
|
6
|
+
timedOut?: boolean;
|
|
7
|
+
}
|
|
8
|
+
export declare function runCommand(command: string, cwd?: string, timeoutMs?: number, onLine?: (line: string) => void): Promise<RunResult>;
|
|
9
|
+
//# sourceMappingURL=runner.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runner.d.ts","sourceRoot":"","sources":["../../src/lib/runner.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,SAAS;IACxB,MAAM,EAAE,MAAM,CAAA;IACd,MAAM,EAAE,MAAM,CAAA;IACd,QAAQ,EAAE,MAAM,CAAA;IAChB,OAAO,EAAE,OAAO,CAAA;IAChB,QAAQ,CAAC,EAAE,OAAO,CAAA;CACnB;AAED,wBAAsB,UAAU,CAC9B,OAAO,EAAE,MAAM,EACf,GAAG,GAAE,MAAsB,EAC3B,SAAS,SAAU,EACnB,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,GAC9B,OAAO,CAAC,SAAS,CAAC,CA0CpB"}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { spawn } from 'child_process';
|
|
2
|
+
export async function runCommand(command, cwd = process.cwd(), timeoutMs = 300_000, onLine) {
|
|
3
|
+
return new Promise((resolve) => {
|
|
4
|
+
const proc = spawn(command, { cwd, shell: true, detached: false });
|
|
5
|
+
let stdout = '';
|
|
6
|
+
let stderr = '';
|
|
7
|
+
let settled = false;
|
|
8
|
+
const timer = setTimeout(() => {
|
|
9
|
+
if (settled)
|
|
10
|
+
return;
|
|
11
|
+
settled = true;
|
|
12
|
+
try {
|
|
13
|
+
process.kill(-proc.pid, 'SIGKILL');
|
|
14
|
+
}
|
|
15
|
+
catch {
|
|
16
|
+
proc.kill('SIGKILL');
|
|
17
|
+
}
|
|
18
|
+
resolve({ stdout, stderr, exitCode: 1, success: false, timedOut: true });
|
|
19
|
+
}, timeoutMs);
|
|
20
|
+
function handleChunk(str, dest) {
|
|
21
|
+
if (dest === 'stdout')
|
|
22
|
+
stdout += str;
|
|
23
|
+
else
|
|
24
|
+
stderr += str;
|
|
25
|
+
if (onLine) {
|
|
26
|
+
for (const line of str.split('\n')) {
|
|
27
|
+
if (line.trim())
|
|
28
|
+
onLine(line);
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
proc.stdout.on('data', (chunk) => handleChunk(chunk.toString(), 'stdout'));
|
|
33
|
+
proc.stderr.on('data', (chunk) => handleChunk(chunk.toString(), 'stderr'));
|
|
34
|
+
proc.on('close', (code) => {
|
|
35
|
+
if (settled)
|
|
36
|
+
return;
|
|
37
|
+
settled = true;
|
|
38
|
+
clearTimeout(timer);
|
|
39
|
+
resolve({ stdout, stderr, exitCode: code ?? 1, success: (code ?? 1) === 0 });
|
|
40
|
+
});
|
|
41
|
+
proc.on('error', (err) => {
|
|
42
|
+
if (settled)
|
|
43
|
+
return;
|
|
44
|
+
settled = true;
|
|
45
|
+
clearTimeout(timer);
|
|
46
|
+
resolve({ stdout, stderr: err.message, exitCode: 1, success: false });
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
//# sourceMappingURL=runner.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runner.js","sourceRoot":"","sources":["../../src/lib/runner.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,MAAM,eAAe,CAAA;AAUrC,MAAM,CAAC,KAAK,UAAU,UAAU,CAC9B,OAAe,EACf,MAAc,OAAO,CAAC,GAAG,EAAE,EAC3B,SAAS,GAAG,OAAO,EACnB,MAA+B;IAE/B,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC7B,MAAM,IAAI,GAAG,KAAK,CAAC,OAAO,EAAE,EAAE,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC,CAAA;QAElE,IAAI,MAAM,GAAG,EAAE,CAAA;QACf,IAAI,MAAM,GAAG,EAAE,CAAA;QACf,IAAI,OAAO,GAAG,KAAK,CAAA;QAEnB,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,EAAE;YAC5B,IAAI,OAAO;gBAAE,OAAM;YACnB,OAAO,GAAG,IAAI,CAAA;YACd,IAAI,CAAC;gBAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,GAAI,EAAE,SAAS,CAAC,CAAA;YAAC,CAAC;YAAC,MAAM,CAAC;gBAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAA;YAAC,CAAC;YAC1E,OAAO,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAA;QAC1E,CAAC,EAAE,SAAS,CAAC,CAAA;QAEb,SAAS,WAAW,CAAC,GAAW,EAAE,IAAyB;YACzD,IAAI,IAAI,KAAK,QAAQ;gBAAE,MAAM,IAAI,GAAG,CAAA;;gBAC/B,MAAM,IAAI,GAAG,CAAA;YAClB,IAAI,MAAM,EAAE,CAAC;gBACX,KAAK,MAAM,IAAI,IAAI,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;oBACnC,IAAI,IAAI,CAAC,IAAI,EAAE;wBAAE,MAAM,CAAC,IAAI,CAAC,CAAA;gBAC/B,CAAC;YACH,CAAC;QACH,CAAC;QAED,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,QAAQ,EAAE,EAAE,QAAQ,CAAC,CAAC,CAAA;QAC1E,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,QAAQ,EAAE,EAAE,QAAQ,CAAC,CAAC,CAAA;QAE1E,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,EAAE;YACxB,IAAI,OAAO;gBAAE,OAAM;YACnB,OAAO,GAAG,IAAI,CAAA;YACd,YAAY,CAAC,KAAK,CAAC,CAAA;YACnB,OAAO,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,IAAI,IAAI,CAAC,EAAE,OAAO,EAAE,CAAC,IAAI,IAAI,CAAC,CAAC,KAAK,CAAC,EAAE,CAAC,CAAA;QAC9E,CAAC,CAAC,CAAA;QAEF,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;YACvB,IAAI,OAAO;gBAAE,OAAM;YACnB,OAAO,GAAG,IAAI,CAAA;YACd,YAAY,CAAC,KAAK,CAAC,CAAA;YACnB,OAAO,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,CAAC,OAAO,EAAE,QAAQ,EAAE,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAA;QACvE,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;AACJ,CAAC"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
export declare function shouldUseSkeleton(code: string): boolean;
|
|
2
|
+
/**
|
|
3
|
+
* Returns a skeletonized version of sourceCode.
|
|
4
|
+
* expandFunctions: names of functions whose full body must be included (the uncovered ones).
|
|
5
|
+
* If the file is short enough, returns the original code unchanged.
|
|
6
|
+
*/
|
|
7
|
+
export declare function buildSourceSkeleton(sourceCode: string, expandFunctions?: string[]): string;
|
|
8
|
+
//# sourceMappingURL=skeleton.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"skeleton.d.ts","sourceRoot":"","sources":["../../src/lib/skeleton.ts"],"names":[],"mappings":"AAoEA,wBAAgB,iBAAiB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAEvD;AAED;;;;GAIG;AACH,wBAAgB,mBAAmB,CAAC,UAAU,EAAE,MAAM,EAAE,eAAe,GAAE,MAAM,EAAO,GAAG,MAAM,CAwD9F"}
|