codeprobe 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +406 -0
- package/dist/cli.d.ts +7 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +104 -0
- package/dist/cli.js.map +1 -0
- package/dist/commands/ab.d.ts +7 -0
- package/dist/commands/ab.d.ts.map +1 -0
- package/dist/commands/ab.js +230 -0
- package/dist/commands/ab.js.map +1 -0
- package/dist/commands/agents.d.ts +10 -0
- package/dist/commands/agents.d.ts.map +1 -0
- package/dist/commands/agents.js +326 -0
- package/dist/commands/agents.js.map +1 -0
- package/dist/commands/autotest.d.ts +10 -0
- package/dist/commands/autotest.d.ts.map +1 -0
- package/dist/commands/autotest.js +408 -0
- package/dist/commands/autotest.js.map +1 -0
- package/dist/commands/benchmark.d.ts +6 -0
- package/dist/commands/benchmark.d.ts.map +1 -0
- package/dist/commands/benchmark.js +215 -0
- package/dist/commands/benchmark.js.map +1 -0
- package/dist/commands/check.d.ts +10 -0
- package/dist/commands/check.d.ts.map +1 -0
- package/dist/commands/check.js +333 -0
- package/dist/commands/check.js.map +1 -0
- package/dist/commands/context.d.ts +16 -0
- package/dist/commands/context.d.ts.map +1 -0
- package/dist/commands/context.js +219 -0
- package/dist/commands/context.js.map +1 -0
- package/dist/commands/cost.d.ts +9 -0
- package/dist/commands/cost.d.ts.map +1 -0
- package/dist/commands/cost.js +142 -0
- package/dist/commands/cost.js.map +1 -0
- package/dist/commands/dashboard.d.ts +10 -0
- package/dist/commands/dashboard.d.ts.map +1 -0
- package/dist/commands/dashboard.js +462 -0
- package/dist/commands/dashboard.js.map +1 -0
- package/dist/commands/diff.d.ts +6 -0
- package/dist/commands/diff.d.ts.map +1 -0
- package/dist/commands/diff.js +118 -0
- package/dist/commands/diff.js.map +1 -0
- package/dist/commands/doctor.d.ts +12 -0
- package/dist/commands/doctor.d.ts.map +1 -0
- package/dist/commands/doctor.js +203 -0
- package/dist/commands/doctor.js.map +1 -0
- package/dist/commands/explain.d.ts +7 -0
- package/dist/commands/explain.d.ts.map +1 -0
- package/dist/commands/explain.js +164 -0
- package/dist/commands/explain.js.map +1 -0
- package/dist/commands/flaky.d.ts +10 -0
- package/dist/commands/flaky.d.ts.map +1 -0
- package/dist/commands/flaky.js +141 -0
- package/dist/commands/flaky.js.map +1 -0
- package/dist/commands/generateClaudeMd.d.ts +11 -0
- package/dist/commands/generateClaudeMd.d.ts.map +1 -0
- package/dist/commands/generateClaudeMd.js +278 -0
- package/dist/commands/generateClaudeMd.js.map +1 -0
- package/dist/commands/generateRules.d.ts +11 -0
- package/dist/commands/generateRules.d.ts.map +1 -0
- package/dist/commands/generateRules.js +413 -0
- package/dist/commands/generateRules.js.map +1 -0
- package/dist/commands/heatmap.d.ts +7 -0
- package/dist/commands/heatmap.d.ts.map +1 -0
- package/dist/commands/heatmap.js +117 -0
- package/dist/commands/heatmap.js.map +1 -0
- package/dist/commands/history.d.ts +13 -0
- package/dist/commands/history.d.ts.map +1 -0
- package/dist/commands/history.js +113 -0
- package/dist/commands/history.js.map +1 -0
- package/dist/commands/hooks.d.ts +9 -0
- package/dist/commands/hooks.d.ts.map +1 -0
- package/dist/commands/hooks.js +199 -0
- package/dist/commands/hooks.js.map +1 -0
- package/dist/commands/improve.d.ts +7 -0
- package/dist/commands/improve.d.ts.map +1 -0
- package/dist/commands/improve.js +192 -0
- package/dist/commands/improve.js.map +1 -0
- package/dist/commands/init.d.ts +9 -0
- package/dist/commands/init.d.ts.map +1 -0
- package/dist/commands/init.js +270 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/installHook.d.ts +9 -0
- package/dist/commands/installHook.d.ts.map +1 -0
- package/dist/commands/installHook.js +78 -0
- package/dist/commands/installHook.js.map +1 -0
- package/dist/commands/lint.d.ts +6 -0
- package/dist/commands/lint.d.ts.map +1 -0
- package/dist/commands/lint.js +237 -0
- package/dist/commands/lint.js.map +1 -0
- package/dist/commands/map.d.ts +9 -0
- package/dist/commands/map.d.ts.map +1 -0
- package/dist/commands/map.js +114 -0
- package/dist/commands/map.js.map +1 -0
- package/dist/commands/mcp.d.ts +6 -0
- package/dist/commands/mcp.d.ts.map +1 -0
- package/dist/commands/mcp.js +151 -0
- package/dist/commands/mcp.js.map +1 -0
- package/dist/commands/models.d.ts +9 -0
- package/dist/commands/models.d.ts.map +1 -0
- package/dist/commands/models.js +89 -0
- package/dist/commands/models.js.map +1 -0
- package/dist/commands/pack.d.ts +10 -0
- package/dist/commands/pack.d.ts.map +1 -0
- package/dist/commands/pack.js +248 -0
- package/dist/commands/pack.js.map +1 -0
- package/dist/commands/recommend.d.ts +10 -0
- package/dist/commands/recommend.d.ts.map +1 -0
- package/dist/commands/recommend.js +472 -0
- package/dist/commands/recommend.js.map +1 -0
- package/dist/commands/regression.d.ts +10 -0
- package/dist/commands/regression.d.ts.map +1 -0
- package/dist/commands/regression.js +212 -0
- package/dist/commands/regression.js.map +1 -0
- package/dist/commands/repl.d.ts +9 -0
- package/dist/commands/repl.d.ts.map +1 -0
- package/dist/commands/repl.js +245 -0
- package/dist/commands/repl.js.map +1 -0
- package/dist/commands/scan.d.ts +10 -0
- package/dist/commands/scan.d.ts.map +1 -0
- package/dist/commands/scan.js +352 -0
- package/dist/commands/scan.js.map +1 -0
- package/dist/commands/score.d.ts +10 -0
- package/dist/commands/score.d.ts.map +1 -0
- package/dist/commands/score.js +192 -0
- package/dist/commands/score.js.map +1 -0
- package/dist/commands/security.d.ts +10 -0
- package/dist/commands/security.d.ts.map +1 -0
- package/dist/commands/security.js +211 -0
- package/dist/commands/security.js.map +1 -0
- package/dist/commands/simulate.d.ts +7 -0
- package/dist/commands/simulate.d.ts.map +1 -0
- package/dist/commands/simulate.js +149 -0
- package/dist/commands/simulate.js.map +1 -0
- package/dist/commands/summary.d.ts +9 -0
- package/dist/commands/summary.d.ts.map +1 -0
- package/dist/commands/summary.js +271 -0
- package/dist/commands/summary.js.map +1 -0
- package/dist/commands/test.d.ts +9 -0
- package/dist/commands/test.d.ts.map +1 -0
- package/dist/commands/test.js +219 -0
- package/dist/commands/test.js.map +1 -0
- package/dist/commands/ui.d.ts +8 -0
- package/dist/commands/ui.d.ts.map +1 -0
- package/dist/commands/ui.js +222 -0
- package/dist/commands/ui.js.map +1 -0
- package/dist/commands/validate.d.ts +7 -0
- package/dist/commands/validate.d.ts.map +1 -0
- package/dist/commands/validate.js +254 -0
- package/dist/commands/validate.js.map +1 -0
- package/dist/commands/workflow.d.ts +39 -0
- package/dist/commands/workflow.d.ts.map +1 -0
- package/dist/commands/workflow.js +309 -0
- package/dist/commands/workflow.js.map +1 -0
- package/dist/core/__tests__/contextAnalyzer.test.d.ts +2 -0
- package/dist/core/__tests__/contextAnalyzer.test.d.ts.map +1 -0
- package/dist/core/__tests__/contextAnalyzer.test.js +48 -0
- package/dist/core/__tests__/contextAnalyzer.test.js.map +1 -0
- package/dist/core/__tests__/promptLinter.test.d.ts +2 -0
- package/dist/core/__tests__/promptLinter.test.d.ts.map +1 -0
- package/dist/core/__tests__/promptLinter.test.js +74 -0
- package/dist/core/__tests__/promptLinter.test.js.map +1 -0
- package/dist/core/__tests__/promptRunner.test.d.ts +2 -0
- package/dist/core/__tests__/promptRunner.test.d.ts.map +1 -0
- package/dist/core/__tests__/promptRunner.test.js +84 -0
- package/dist/core/__tests__/promptRunner.test.js.map +1 -0
- package/dist/core/__tests__/securityScanner.test.d.ts +2 -0
- package/dist/core/__tests__/securityScanner.test.d.ts.map +1 -0
- package/dist/core/__tests__/securityScanner.test.js +39 -0
- package/dist/core/__tests__/securityScanner.test.js.map +1 -0
- package/dist/core/agentTracer.d.ts +21 -0
- package/dist/core/agentTracer.d.ts.map +1 -0
- package/dist/core/agentTracer.js +355 -0
- package/dist/core/agentTracer.js.map +1 -0
- package/dist/core/anthropicClient.d.ts +26 -0
- package/dist/core/anthropicClient.d.ts.map +1 -0
- package/dist/core/anthropicClient.js +62 -0
- package/dist/core/anthropicClient.js.map +1 -0
- package/dist/core/benchmarkRunner.d.ts +25 -0
- package/dist/core/benchmarkRunner.d.ts.map +1 -0
- package/dist/core/benchmarkRunner.js +182 -0
- package/dist/core/benchmarkRunner.js.map +1 -0
- package/dist/core/contextAnalyzer.d.ts +19 -0
- package/dist/core/contextAnalyzer.d.ts.map +1 -0
- package/dist/core/contextAnalyzer.js +221 -0
- package/dist/core/contextAnalyzer.js.map +1 -0
- package/dist/core/contextPacker.d.ts +26 -0
- package/dist/core/contextPacker.d.ts.map +1 -0
- package/dist/core/contextPacker.js +358 -0
- package/dist/core/contextPacker.js.map +1 -0
- package/dist/core/datasetRunner.d.ts +10 -0
- package/dist/core/datasetRunner.d.ts.map +1 -0
- package/dist/core/datasetRunner.js +130 -0
- package/dist/core/datasetRunner.js.map +1 -0
- package/dist/core/doctorRunner.d.ts +24 -0
- package/dist/core/doctorRunner.d.ts.map +1 -0
- package/dist/core/doctorRunner.js +278 -0
- package/dist/core/doctorRunner.js.map +1 -0
- package/dist/core/hookScanner.d.ts +24 -0
- package/dist/core/hookScanner.d.ts.map +1 -0
- package/dist/core/hookScanner.js +226 -0
- package/dist/core/hookScanner.js.map +1 -0
- package/dist/core/mcpScanner.d.ts +22 -0
- package/dist/core/mcpScanner.d.ts.map +1 -0
- package/dist/core/mcpScanner.js +290 -0
- package/dist/core/mcpScanner.js.map +1 -0
- package/dist/core/modelRegistry.d.ts +35 -0
- package/dist/core/modelRegistry.d.ts.map +1 -0
- package/dist/core/modelRegistry.js +97 -0
- package/dist/core/modelRegistry.js.map +1 -0
- package/dist/core/promptDiff.d.ts +25 -0
- package/dist/core/promptDiff.d.ts.map +1 -0
- package/dist/core/promptDiff.js +130 -0
- package/dist/core/promptDiff.js.map +1 -0
- package/dist/core/promptExplainer.d.ts +17 -0
- package/dist/core/promptExplainer.d.ts.map +1 -0
- package/dist/core/promptExplainer.js +334 -0
- package/dist/core/promptExplainer.js.map +1 -0
- package/dist/core/promptImprover.d.ts +19 -0
- package/dist/core/promptImprover.d.ts.map +1 -0
- package/dist/core/promptImprover.js +260 -0
- package/dist/core/promptImprover.js.map +1 -0
- package/dist/core/promptLinter.d.ts +24 -0
- package/dist/core/promptLinter.d.ts.map +1 -0
- package/dist/core/promptLinter.js +319 -0
- package/dist/core/promptLinter.js.map +1 -0
- package/dist/core/promptRunner.d.ts +31 -0
- package/dist/core/promptRunner.d.ts.map +1 -0
- package/dist/core/promptRunner.js +427 -0
- package/dist/core/promptRunner.js.map +1 -0
- package/dist/core/providers/anthropic.d.ts +10 -0
- package/dist/core/providers/anthropic.d.ts.map +1 -0
- package/dist/core/providers/anthropic.js +26 -0
- package/dist/core/providers/anthropic.js.map +1 -0
- package/dist/core/providers/base.d.ts +22 -0
- package/dist/core/providers/base.d.ts.map +1 -0
- package/dist/core/providers/base.js +2 -0
- package/dist/core/providers/base.js.map +1 -0
- package/dist/core/providers/factory.d.ts +7 -0
- package/dist/core/providers/factory.d.ts.map +1 -0
- package/dist/core/providers/factory.js +42 -0
- package/dist/core/providers/factory.js.map +1 -0
- package/dist/core/providers/google.d.ts +10 -0
- package/dist/core/providers/google.d.ts.map +1 -0
- package/dist/core/providers/google.js +47 -0
- package/dist/core/providers/google.js.map +1 -0
- package/dist/core/providers/openai.d.ts +19 -0
- package/dist/core/providers/openai.d.ts.map +1 -0
- package/dist/core/providers/openai.js +54 -0
- package/dist/core/providers/openai.js.map +1 -0
- package/dist/core/regressionRunner.d.ts +11 -0
- package/dist/core/regressionRunner.d.ts.map +1 -0
- package/dist/core/regressionRunner.js +116 -0
- package/dist/core/regressionRunner.js.map +1 -0
- package/dist/core/repositorySimulator.d.ts +17 -0
- package/dist/core/repositorySimulator.d.ts.map +1 -0
- package/dist/core/repositorySimulator.js +104 -0
- package/dist/core/repositorySimulator.js.map +1 -0
- package/dist/core/scorer.d.ts +30 -0
- package/dist/core/scorer.d.ts.map +1 -0
- package/dist/core/scorer.js +317 -0
- package/dist/core/scorer.js.map +1 -0
- package/dist/core/securityScanner.d.ts +23 -0
- package/dist/core/securityScanner.d.ts.map +1 -0
- package/dist/core/securityScanner.js +216 -0
- package/dist/core/securityScanner.js.map +1 -0
- package/dist/core/skillValidator.d.ts +41 -0
- package/dist/core/skillValidator.d.ts.map +1 -0
- package/dist/core/skillValidator.js +235 -0
- package/dist/core/skillValidator.js.map +1 -0
- package/dist/core/testHistory.d.ts +44 -0
- package/dist/core/testHistory.d.ts.map +1 -0
- package/dist/core/testHistory.js +91 -0
- package/dist/core/testHistory.js.map +1 -0
- package/dist/tokenizers/claudeTokenizer.d.ts +26 -0
- package/dist/tokenizers/claudeTokenizer.d.ts.map +1 -0
- package/dist/tokenizers/claudeTokenizer.js +83 -0
- package/dist/tokenizers/claudeTokenizer.js.map +1 -0
- package/dist/types/agent.d.ts +26 -0
- package/dist/types/agent.d.ts.map +1 -0
- package/dist/types/agent.js +5 -0
- package/dist/types/agent.js.map +1 -0
- package/dist/types/config.d.ts +30 -0
- package/dist/types/config.d.ts.map +1 -0
- package/dist/types/config.js +5 -0
- package/dist/types/config.js.map +1 -0
- package/dist/types/context.d.ts +77 -0
- package/dist/types/context.d.ts.map +1 -0
- package/dist/types/context.js +5 -0
- package/dist/types/context.js.map +1 -0
- package/dist/types/dataset.d.ts +26 -0
- package/dist/types/dataset.d.ts.map +1 -0
- package/dist/types/dataset.js +5 -0
- package/dist/types/dataset.js.map +1 -0
- package/dist/types/diagnostics.d.ts +31 -0
- package/dist/types/diagnostics.d.ts.map +1 -0
- package/dist/types/diagnostics.js +5 -0
- package/dist/types/diagnostics.js.map +1 -0
- package/dist/types/prompt.d.ts +53 -0
- package/dist/types/prompt.d.ts.map +1 -0
- package/dist/types/prompt.js +5 -0
- package/dist/types/prompt.js.map +1 -0
- package/dist/types/results.d.ts +42 -0
- package/dist/types/results.d.ts.map +1 -0
- package/dist/types/results.js +5 -0
- package/dist/types/results.js.map +1 -0
- package/dist/ui/dashboard.d.ts +57 -0
- package/dist/ui/dashboard.d.ts.map +1 -0
- package/dist/ui/dashboard.js +644 -0
- package/dist/ui/dashboard.js.map +1 -0
- package/dist/utils/__tests__/hashing.test.d.ts +2 -0
- package/dist/utils/__tests__/hashing.test.d.ts.map +1 -0
- package/dist/utils/__tests__/hashing.test.js +28 -0
- package/dist/utils/__tests__/hashing.test.js.map +1 -0
- package/dist/utils/__tests__/output.test.d.ts +2 -0
- package/dist/utils/__tests__/output.test.d.ts.map +1 -0
- package/dist/utils/__tests__/output.test.js +62 -0
- package/dist/utils/__tests__/output.test.js.map +1 -0
- package/dist/utils/cache.d.ts +29 -0
- package/dist/utils/cache.d.ts.map +1 -0
- package/dist/utils/cache.js +87 -0
- package/dist/utils/cache.js.map +1 -0
- package/dist/utils/config.d.ts +15 -0
- package/dist/utils/config.d.ts.map +1 -0
- package/dist/utils/config.js +61 -0
- package/dist/utils/config.js.map +1 -0
- package/dist/utils/errors.d.ts +43 -0
- package/dist/utils/errors.d.ts.map +1 -0
- package/dist/utils/errors.js +83 -0
- package/dist/utils/errors.js.map +1 -0
- package/dist/utils/fs.d.ts +44 -0
- package/dist/utils/fs.d.ts.map +1 -0
- package/dist/utils/fs.js +119 -0
- package/dist/utils/fs.js.map +1 -0
- package/dist/utils/hashing.d.ts +13 -0
- package/dist/utils/hashing.d.ts.map +1 -0
- package/dist/utils/hashing.js +18 -0
- package/dist/utils/hashing.js.map +1 -0
- package/dist/utils/logger.d.ts +32 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +76 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/output.d.ts +34 -0
- package/dist/utils/output.d.ts.map +1 -0
- package/dist/utils/output.js +99 -0
- package/dist/utils/output.js.map +1 -0
- package/dist/utils/paths.d.ts +33 -0
- package/dist/utils/paths.d.ts.map +1 -0
- package/dist/utils/paths.js +51 -0
- package/dist/utils/paths.js.map +1 -0
- package/dist/utils/spinner.d.ts +23 -0
- package/dist/utils/spinner.d.ts.map +1 -0
- package/dist/utils/spinner.js +79 -0
- package/dist/utils/spinner.js.map +1 -0
- package/package.json +60 -0
|
@@ -0,0 +1,427 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Prompt test runner — parses prompt specs, runs tests, and evaluates assertions.
|
|
3
|
+
*/
|
|
4
|
+
import { readFile } from 'node:fs/promises';
|
|
5
|
+
import { createHash } from 'node:crypto';
|
|
6
|
+
import { resolve, dirname } from 'node:path';
|
|
7
|
+
import yaml from 'js-yaml';
|
|
8
|
+
import { z } from 'zod';
|
|
9
|
+
import _Ajv from 'ajv';
|
|
10
|
+
// With module:ES2022 + moduleResolution:node16, the default import of Ajv
|
|
11
|
+
// resolves to the module namespace rather than the class constructor.
|
|
12
|
+
// Cast through unknown to obtain a callable constructor.
|
|
13
|
+
const Ajv = _Ajv;
|
|
14
|
+
import { cacheKey, getCached, setCached } from '../utils/cache.js';
|
|
15
|
+
import { logger } from '../utils/logger.js';
|
|
16
|
+
import { readTextFile } from '../utils/fs.js';
|
|
17
|
+
// ---------------------------------------------------------------------------
|
|
18
|
+
// Zod schema for validating prompt spec YAML files
|
|
19
|
+
// ---------------------------------------------------------------------------
|
|
20
|
+
const TestExpectationSchema = z.object({
|
|
21
|
+
contains: z.array(z.string()).optional(),
|
|
22
|
+
notContains: z.array(z.string()).optional(),
|
|
23
|
+
regex: z.array(z.string()).optional(),
|
|
24
|
+
equals: z.string().optional(),
|
|
25
|
+
jsonSchema: z.record(z.unknown()).optional(),
|
|
26
|
+
minLength: z.number().optional(),
|
|
27
|
+
maxLength: z.number().optional(),
|
|
28
|
+
lineCount: z.number().optional(),
|
|
29
|
+
wordCount: z.object({ min: z.number().optional(), max: z.number().optional() }).optional(),
|
|
30
|
+
startsWith: z.string().optional(),
|
|
31
|
+
endsWith: z.string().optional(),
|
|
32
|
+
isSorted: z.boolean().optional(),
|
|
33
|
+
custom: z.string().optional(),
|
|
34
|
+
});
|
|
35
|
+
const PromptTestSchema = z.object({
|
|
36
|
+
name: z.string(),
|
|
37
|
+
input: z.string().optional(),
|
|
38
|
+
inputFile: z.string().optional(),
|
|
39
|
+
expect: TestExpectationSchema.optional(),
|
|
40
|
+
});
|
|
41
|
+
const PromptSpecSchema = z.object({
|
|
42
|
+
name: z.string(),
|
|
43
|
+
description: z.string().optional(),
|
|
44
|
+
model: z.string().optional(),
|
|
45
|
+
system: z.string().optional(),
|
|
46
|
+
prompt: z.string(),
|
|
47
|
+
tests: z.array(PromptTestSchema).optional(),
|
|
48
|
+
});
|
|
49
|
+
// ---------------------------------------------------------------------------
|
|
50
|
+
// Parse
|
|
51
|
+
// ---------------------------------------------------------------------------
|
|
52
|
+
/**
|
|
53
|
+
* Parse and validate a prompt spec YAML file.
|
|
54
|
+
*/
|
|
55
|
+
export async function parsePromptSpec(filePath) {
|
|
56
|
+
const absolutePath = resolve(filePath);
|
|
57
|
+
const content = await readFile(absolutePath, 'utf-8');
|
|
58
|
+
const raw = yaml.load(content);
|
|
59
|
+
const parsed = PromptSpecSchema.parse(raw);
|
|
60
|
+
// Resolve any inputFile references relative to the spec file directory
|
|
61
|
+
const specDir = dirname(absolutePath);
|
|
62
|
+
if (parsed.tests) {
|
|
63
|
+
for (const test of parsed.tests) {
|
|
64
|
+
if (test.inputFile && !test.input) {
|
|
65
|
+
const inputPath = resolve(specDir, test.inputFile);
|
|
66
|
+
const inputContent = await readTextFile(inputPath);
|
|
67
|
+
if (inputContent !== null) {
|
|
68
|
+
test.input = inputContent;
|
|
69
|
+
}
|
|
70
|
+
else {
|
|
71
|
+
throw new Error(`Could not read inputFile "${test.inputFile}" referenced in test "${test.name}"`);
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
return parsed;
|
|
77
|
+
}
|
|
78
|
+
// ---------------------------------------------------------------------------
|
|
79
|
+
// Mock generation
|
|
80
|
+
// ---------------------------------------------------------------------------
|
|
81
|
+
/**
|
|
82
|
+
* Generate a deterministic mock response by hashing inputs and
|
|
83
|
+
* incorporating words from the input to produce a plausible response.
|
|
84
|
+
*
|
|
85
|
+
* The mock tries to respect format hints from the system/prompt:
|
|
86
|
+
* - If "bullet" or "- " patterns are detected, output bullet-point format
|
|
87
|
+
* - Otherwise, use prose format
|
|
88
|
+
*/
|
|
89
|
+
function generateMockOutput(prompt, system, input) {
|
|
90
|
+
const hashSource = [prompt, system ?? '', input ?? ''].join('|');
|
|
91
|
+
const hash = createHash('sha256').update(hashSource).digest('hex');
|
|
92
|
+
// Extract meaningful words from the input to make the mock feel relevant
|
|
93
|
+
// Keep words >= 2 chars to preserve acronyms like "MCP", "AI", etc.
|
|
94
|
+
const sourceText = input ?? prompt;
|
|
95
|
+
const words = sourceText
|
|
96
|
+
.split(/\s+/)
|
|
97
|
+
.map((w) => w.replace(/[^a-zA-Z0-9-]/g, ''))
|
|
98
|
+
.filter((w) => w.length >= 2)
|
|
99
|
+
.slice(0, 15);
|
|
100
|
+
const combined = [prompt, system ?? ''].join(' ').toLowerCase();
|
|
101
|
+
const wantsBullets = combined.includes('bullet') ||
|
|
102
|
+
combined.includes('- ') ||
|
|
103
|
+
combined.includes('list') ||
|
|
104
|
+
combined.includes('points');
|
|
105
|
+
const seed = parseInt(hash.slice(0, 8), 16);
|
|
106
|
+
if (wantsBullets) {
|
|
107
|
+
// Generate bullet-point output incorporating input words
|
|
108
|
+
const bullets = [];
|
|
109
|
+
const groupSize = Math.max(1, Math.floor(words.length / 3));
|
|
110
|
+
for (let i = 0; i < 3; i++) {
|
|
111
|
+
const chunk = words.slice(i * groupSize, (i + 1) * groupSize);
|
|
112
|
+
if (chunk.length > 0) {
|
|
113
|
+
bullets.push(`- ${chunk.join(' ')}: key insight from the analysis`);
|
|
114
|
+
}
|
|
115
|
+
else {
|
|
116
|
+
bullets.push(`- Point ${i + 1}: additional consideration based on context`);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
return bullets.join('\n');
|
|
120
|
+
}
|
|
121
|
+
const templates = [
|
|
122
|
+
`Based on the provided context, here is the analysis: ${words.join(', ')}. The key considerations include ${words.slice(0, 3).join(' and ')}.`,
|
|
123
|
+
`After reviewing the input about ${words.slice(0, 2).join(' and ')}, the focus should be on ${words.slice(2, 5).join(', ')}. This approach ensures comprehensive coverage.`,
|
|
124
|
+
`The response addresses: ${words.join(', ')}. Each element has been carefully evaluated to provide actionable insights.`,
|
|
125
|
+
`Regarding ${words.slice(0, 3).join(', ')}: the analysis suggests that ${words.slice(3, 6).join(' and ')} are the primary factors to consider.`,
|
|
126
|
+
];
|
|
127
|
+
const templateIndex = seed % templates.length;
|
|
128
|
+
return `${templates[templateIndex]}`; // No [MOCK] prefix — cleaner for assertion testing
|
|
129
|
+
}
|
|
130
|
+
// ---------------------------------------------------------------------------
|
|
131
|
+
// Assertion evaluation
|
|
132
|
+
// ---------------------------------------------------------------------------
|
|
133
|
+
const ajvInstance = new Ajv({ allErrors: true, strict: false });
|
|
134
|
+
/**
|
|
135
|
+
* Evaluate all assertions for a test expectation against actual output.
|
|
136
|
+
*/
|
|
137
|
+
export function evaluateAssertions(output, expect) {
|
|
138
|
+
const results = [];
|
|
139
|
+
// contains checks
|
|
140
|
+
if (expect.contains) {
|
|
141
|
+
for (const substring of expect.contains) {
|
|
142
|
+
results.push({
|
|
143
|
+
type: 'contains',
|
|
144
|
+
expected: substring,
|
|
145
|
+
actual: output.includes(substring) ? substring : undefined,
|
|
146
|
+
passed: output.includes(substring),
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
// notContains checks
|
|
151
|
+
if (expect.notContains) {
|
|
152
|
+
for (const substring of expect.notContains) {
|
|
153
|
+
const found = output.includes(substring);
|
|
154
|
+
results.push({
|
|
155
|
+
type: 'notContains',
|
|
156
|
+
expected: substring,
|
|
157
|
+
actual: found ? substring : undefined,
|
|
158
|
+
passed: !found,
|
|
159
|
+
});
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
// regex checks (multiline by default so ^ and $ match line boundaries)
|
|
163
|
+
if (expect.regex) {
|
|
164
|
+
for (const pattern of expect.regex) {
|
|
165
|
+
let passed = false;
|
|
166
|
+
try {
|
|
167
|
+
const re = new RegExp(pattern, 'm');
|
|
168
|
+
passed = re.test(output);
|
|
169
|
+
}
|
|
170
|
+
catch {
|
|
171
|
+
// Invalid regex always fails
|
|
172
|
+
}
|
|
173
|
+
results.push({
|
|
174
|
+
type: 'regex',
|
|
175
|
+
expected: pattern,
|
|
176
|
+
actual: passed ? 'matched' : 'no match',
|
|
177
|
+
passed,
|
|
178
|
+
});
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
// equals check
|
|
182
|
+
if (expect.equals !== undefined) {
|
|
183
|
+
const trimmedOutput = output.trim();
|
|
184
|
+
const trimmedExpected = expect.equals.trim();
|
|
185
|
+
results.push({
|
|
186
|
+
type: 'equals',
|
|
187
|
+
expected: expect.equals,
|
|
188
|
+
actual: output,
|
|
189
|
+
passed: trimmedOutput === trimmedExpected,
|
|
190
|
+
});
|
|
191
|
+
}
|
|
192
|
+
// minLength check
|
|
193
|
+
if (expect.minLength !== undefined) {
|
|
194
|
+
const passed = output.length >= expect.minLength;
|
|
195
|
+
results.push({
|
|
196
|
+
type: 'minLength',
|
|
197
|
+
expected: `>= ${expect.minLength} chars`,
|
|
198
|
+
actual: `${output.length} chars`,
|
|
199
|
+
passed,
|
|
200
|
+
});
|
|
201
|
+
}
|
|
202
|
+
// maxLength check
|
|
203
|
+
if (expect.maxLength !== undefined) {
|
|
204
|
+
const passed = output.length <= expect.maxLength;
|
|
205
|
+
results.push({
|
|
206
|
+
type: 'maxLength',
|
|
207
|
+
expected: `<= ${expect.maxLength} chars`,
|
|
208
|
+
actual: `${output.length} chars`,
|
|
209
|
+
passed,
|
|
210
|
+
});
|
|
211
|
+
}
|
|
212
|
+
// lineCount check
|
|
213
|
+
if (expect.lineCount !== undefined) {
|
|
214
|
+
const lines = output.trim().split('\n').length;
|
|
215
|
+
const passed = lines === expect.lineCount;
|
|
216
|
+
results.push({
|
|
217
|
+
type: 'lineCount',
|
|
218
|
+
expected: `${expect.lineCount} lines`,
|
|
219
|
+
actual: `${lines} lines`,
|
|
220
|
+
passed,
|
|
221
|
+
});
|
|
222
|
+
}
|
|
223
|
+
// wordCount check
|
|
224
|
+
if (expect.wordCount) {
|
|
225
|
+
const words = output.trim().split(/\s+/).length;
|
|
226
|
+
const minOk = expect.wordCount.min === undefined || words >= expect.wordCount.min;
|
|
227
|
+
const maxOk = expect.wordCount.max === undefined || words <= expect.wordCount.max;
|
|
228
|
+
const passed = minOk && maxOk;
|
|
229
|
+
const expectedStr = [
|
|
230
|
+
expect.wordCount.min !== undefined ? `>= ${expect.wordCount.min}` : '',
|
|
231
|
+
expect.wordCount.max !== undefined ? `<= ${expect.wordCount.max}` : '',
|
|
232
|
+
].filter(Boolean).join(', ');
|
|
233
|
+
results.push({
|
|
234
|
+
type: 'wordCount',
|
|
235
|
+
expected: expectedStr + ' words',
|
|
236
|
+
actual: `${words} words`,
|
|
237
|
+
passed,
|
|
238
|
+
});
|
|
239
|
+
}
|
|
240
|
+
// startsWith check
|
|
241
|
+
if (expect.startsWith !== undefined) {
|
|
242
|
+
const passed = output.trimStart().startsWith(expect.startsWith);
|
|
243
|
+
results.push({
|
|
244
|
+
type: 'startsWith',
|
|
245
|
+
expected: expect.startsWith,
|
|
246
|
+
actual: output.slice(0, expect.startsWith.length + 10),
|
|
247
|
+
passed,
|
|
248
|
+
});
|
|
249
|
+
}
|
|
250
|
+
// endsWith check
|
|
251
|
+
if (expect.endsWith !== undefined) {
|
|
252
|
+
const passed = output.trimEnd().endsWith(expect.endsWith);
|
|
253
|
+
results.push({
|
|
254
|
+
type: 'endsWith',
|
|
255
|
+
expected: expect.endsWith,
|
|
256
|
+
actual: output.slice(-expect.endsWith.length - 10),
|
|
257
|
+
passed,
|
|
258
|
+
});
|
|
259
|
+
}
|
|
260
|
+
// custom function check
|
|
261
|
+
if (expect.custom) {
|
|
262
|
+
try {
|
|
263
|
+
const fn = new Function('output', `return (${expect.custom})(output);`);
|
|
264
|
+
const result = fn(output);
|
|
265
|
+
const passed = Boolean(result);
|
|
266
|
+
results.push({
|
|
267
|
+
type: 'custom',
|
|
268
|
+
expected: 'custom function returns truthy',
|
|
269
|
+
actual: passed ? 'passed' : 'failed',
|
|
270
|
+
passed,
|
|
271
|
+
});
|
|
272
|
+
}
|
|
273
|
+
catch (err) {
|
|
274
|
+
results.push({
|
|
275
|
+
type: 'custom',
|
|
276
|
+
expected: 'custom function',
|
|
277
|
+
actual: `error: ${err instanceof Error ? err.message : String(err)}`,
|
|
278
|
+
passed: false,
|
|
279
|
+
});
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
// jsonSchema check
|
|
283
|
+
if (expect.jsonSchema) {
|
|
284
|
+
let parsed;
|
|
285
|
+
let parseOk = false;
|
|
286
|
+
try {
|
|
287
|
+
parsed = JSON.parse(output);
|
|
288
|
+
parseOk = true;
|
|
289
|
+
}
|
|
290
|
+
catch {
|
|
291
|
+
// output is not valid JSON
|
|
292
|
+
}
|
|
293
|
+
if (parseOk) {
|
|
294
|
+
try {
|
|
295
|
+
const validate = ajvInstance.compile(expect.jsonSchema);
|
|
296
|
+
const valid = validate(parsed);
|
|
297
|
+
results.push({
|
|
298
|
+
type: 'jsonSchema',
|
|
299
|
+
expected: JSON.stringify(expect.jsonSchema),
|
|
300
|
+
actual: valid ? 'valid' : JSON.stringify(validate.errors),
|
|
301
|
+
passed: Boolean(valid),
|
|
302
|
+
});
|
|
303
|
+
}
|
|
304
|
+
catch (err) {
|
|
305
|
+
results.push({
|
|
306
|
+
type: 'jsonSchema',
|
|
307
|
+
expected: JSON.stringify(expect.jsonSchema),
|
|
308
|
+
actual: `schema compilation error: ${err instanceof Error ? err.message : String(err)}`,
|
|
309
|
+
passed: false,
|
|
310
|
+
});
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
else {
|
|
314
|
+
results.push({
|
|
315
|
+
type: 'jsonSchema',
|
|
316
|
+
expected: JSON.stringify(expect.jsonSchema),
|
|
317
|
+
actual: 'output is not valid JSON',
|
|
318
|
+
passed: false,
|
|
319
|
+
});
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
return results;
|
|
323
|
+
}
|
|
324
|
+
// ---------------------------------------------------------------------------
|
|
325
|
+
// Run single test
|
|
326
|
+
// ---------------------------------------------------------------------------
|
|
327
|
+
/**
|
|
328
|
+
* Run a single prompt test and return the result.
|
|
329
|
+
*/
|
|
330
|
+
export async function runSingleTest(spec, test, options) {
|
|
331
|
+
const startTime = Date.now();
|
|
332
|
+
// Check cache
|
|
333
|
+
if (options.cache) {
|
|
334
|
+
const key = cacheKey(spec.prompt, spec.system ?? '', test.input ?? '', options.mode);
|
|
335
|
+
const cached = await getCached(key);
|
|
336
|
+
if (cached !== null) {
|
|
337
|
+
logger.debug(`Cache hit for test "${test.name}"`);
|
|
338
|
+
const assertions = test.expect
|
|
339
|
+
? evaluateAssertions(cached, test.expect)
|
|
340
|
+
: [];
|
|
341
|
+
const allPassed = assertions.length === 0 || assertions.every((a) => a.passed);
|
|
342
|
+
return {
|
|
343
|
+
testName: test.name,
|
|
344
|
+
promptName: spec.name,
|
|
345
|
+
passed: allPassed,
|
|
346
|
+
output: cached,
|
|
347
|
+
duration: Date.now() - startTime,
|
|
348
|
+
assertions,
|
|
349
|
+
cached: true,
|
|
350
|
+
};
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
let output;
|
|
354
|
+
try {
|
|
355
|
+
if (options.mode === 'mock') {
|
|
356
|
+
output = generateMockOutput(spec.prompt, spec.system, test.input);
|
|
357
|
+
}
|
|
358
|
+
else {
|
|
359
|
+
// Live mode: use provider factory for multi-provider support
|
|
360
|
+
const { createProvider } = await import('./providers/factory.js');
|
|
361
|
+
const model = options.modelOverride ?? spec.model ?? 'claude-sonnet-4-6';
|
|
362
|
+
const fullPrompt = spec.prompt.replace('{{input}}', test.input ?? '');
|
|
363
|
+
const provider = createProvider(model);
|
|
364
|
+
const response = await provider.call({
|
|
365
|
+
model,
|
|
366
|
+
system: spec.system,
|
|
367
|
+
messages: [{ role: 'user', content: fullPrompt }],
|
|
368
|
+
});
|
|
369
|
+
output = response.content;
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
catch (err) {
|
|
373
|
+
const errorMessage = err instanceof Error ? err.message : String(err);
|
|
374
|
+
return {
|
|
375
|
+
testName: test.name,
|
|
376
|
+
promptName: spec.name,
|
|
377
|
+
passed: false,
|
|
378
|
+
output: '',
|
|
379
|
+
duration: Date.now() - startTime,
|
|
380
|
+
assertions: [],
|
|
381
|
+
cached: false,
|
|
382
|
+
error: errorMessage,
|
|
383
|
+
};
|
|
384
|
+
}
|
|
385
|
+
// Cache the result
|
|
386
|
+
if (options.cache) {
|
|
387
|
+
const key = cacheKey(spec.prompt, spec.system ?? '', test.input ?? '', options.mode);
|
|
388
|
+
await setCached(key, output);
|
|
389
|
+
}
|
|
390
|
+
// Evaluate assertions
|
|
391
|
+
const assertions = test.expect
|
|
392
|
+
? evaluateAssertions(output, test.expect)
|
|
393
|
+
: [];
|
|
394
|
+
const allPassed = assertions.length === 0 || assertions.every((a) => a.passed);
|
|
395
|
+
if (options.verbose) {
|
|
396
|
+
logger.debug(`Test "${test.name}" output (${output.length} chars): ${output.slice(0, 200)}`);
|
|
397
|
+
}
|
|
398
|
+
return {
|
|
399
|
+
testName: test.name,
|
|
400
|
+
promptName: spec.name,
|
|
401
|
+
passed: allPassed,
|
|
402
|
+
output,
|
|
403
|
+
duration: Date.now() - startTime,
|
|
404
|
+
assertions,
|
|
405
|
+
cached: false,
|
|
406
|
+
};
|
|
407
|
+
}
|
|
408
|
+
// ---------------------------------------------------------------------------
|
|
409
|
+
// Run all tests in a spec
|
|
410
|
+
// ---------------------------------------------------------------------------
|
|
411
|
+
/**
|
|
412
|
+
* Run all tests defined in a prompt spec file.
|
|
413
|
+
*/
|
|
414
|
+
export async function runPromptTests(specPath, options) {
|
|
415
|
+
const spec = await parsePromptSpec(specPath);
|
|
416
|
+
const tests = spec.tests ?? [];
|
|
417
|
+
if (tests.length === 0) {
|
|
418
|
+
return [];
|
|
419
|
+
}
|
|
420
|
+
const results = [];
|
|
421
|
+
for (const test of tests) {
|
|
422
|
+
const result = await runSingleTest(spec, test, options);
|
|
423
|
+
results.push(result);
|
|
424
|
+
}
|
|
425
|
+
return results;
|
|
426
|
+
}
|
|
427
|
+
//# sourceMappingURL=promptRunner.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"promptRunner.js","sourceRoot":"","sources":["../../src/core/promptRunner.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Anthropic provider — wraps the existing anthropicClient.ts
|
|
3
|
+
* into the ProviderClient interface.
|
|
4
|
+
*/
|
|
5
|
+
import type { ProviderClient, ProviderRequest, ProviderResponse } from './base.js';
|
|
6
|
+
export declare class AnthropicProvider implements ProviderClient {
|
|
7
|
+
isAvailable(): Promise<boolean>;
|
|
8
|
+
call(request: ProviderRequest): Promise<ProviderResponse>;
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=anthropic.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"anthropic.d.ts","sourceRoot":"","sources":["../../../src/core/providers/anthropic.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,eAAe,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAGnF,qBAAa,iBAAkB,YAAW,cAAc;IAChD,WAAW,IAAI,OAAO,CAAC,OAAO,CAAC;IAI/B,IAAI,CAAC,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,gBAAgB,CAAC;CAgBhE"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Anthropic provider — wraps the existing anthropicClient.ts
|
|
3
|
+
* into the ProviderClient interface.
|
|
4
|
+
*/
|
|
5
|
+
import { callAnthropic, isAnthropicAvailable } from '../anthropicClient.js';
|
|
6
|
+
export class AnthropicProvider {
|
|
7
|
+
async isAvailable() {
|
|
8
|
+
return isAnthropicAvailable();
|
|
9
|
+
}
|
|
10
|
+
async call(request) {
|
|
11
|
+
const result = await callAnthropic({
|
|
12
|
+
model: request.model,
|
|
13
|
+
system: request.system,
|
|
14
|
+
messages: request.messages,
|
|
15
|
+
maxTokens: request.maxTokens,
|
|
16
|
+
});
|
|
17
|
+
return {
|
|
18
|
+
content: result.content,
|
|
19
|
+
inputTokens: result.inputTokens,
|
|
20
|
+
outputTokens: result.outputTokens,
|
|
21
|
+
model: result.model,
|
|
22
|
+
stopReason: result.stopReason,
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
//# sourceMappingURL=anthropic.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"anthropic.js","sourceRoot":"","sources":["../../../src/core/providers/anthropic.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAGH,OAAO,EAAE,aAAa,EAAE,oBAAoB,EAAE,MAAM,uBAAuB,CAAC;AAE5E,MAAM,OAAO,iBAAiB;IAC5B,KAAK,CAAC,WAAW;QACf,OAAO,oBAAoB,EAAE,CAAC;IAChC,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,OAAwB;QACjC,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC;YACjC,KAAK,EAAE,OAAO,CAAC,KAAK;YACpB,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,SAAS,EAAE,OAAO,CAAC,SAAS;SAC7B,CAAC,CAAC;QAEH,OAAO;YACL,OAAO,EAAE,MAAM,CAAC,OAAO;YACvB,WAAW,EAAE,MAAM,CAAC,WAAW;YAC/B,YAAY,EAAE,MAAM,CAAC,YAAY;YACjC,KAAK,EAAE,MAAM,CAAC,KAAK;YACnB,UAAU,EAAE,MAAM,CAAC,UAAU;SAC9B,CAAC;IACJ,CAAC;CACF"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
export interface ProviderClient {
|
|
2
|
+
call(request: ProviderRequest): Promise<ProviderResponse>;
|
|
3
|
+
isAvailable(): Promise<boolean>;
|
|
4
|
+
}
|
|
5
|
+
export interface ProviderRequest {
|
|
6
|
+
model: string;
|
|
7
|
+
system?: string;
|
|
8
|
+
messages: Array<{
|
|
9
|
+
role: 'user' | 'assistant';
|
|
10
|
+
content: string;
|
|
11
|
+
}>;
|
|
12
|
+
maxTokens?: number;
|
|
13
|
+
}
|
|
14
|
+
export interface ProviderResponse {
|
|
15
|
+
content: string;
|
|
16
|
+
inputTokens: number;
|
|
17
|
+
outputTokens: number;
|
|
18
|
+
model: string;
|
|
19
|
+
stopReason: string;
|
|
20
|
+
}
|
|
21
|
+
export type ProviderName = 'anthropic' | 'openai' | 'google' | 'deepseek' | 'qwen' | 'local';
|
|
22
|
+
//# sourceMappingURL=base.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../../src/core/providers/base.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,cAAc;IAC7B,IAAI,CAAC,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAAC;IAC1D,WAAW,IAAI,OAAO,CAAC,OAAO,CAAC,CAAC;CACjC;AAED,MAAM,WAAW,eAAe;IAC9B,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,GAAG,WAAW,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IACjE,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,gBAAgB;IAC/B,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,MAAM,YAAY,GAAG,WAAW,GAAG,QAAQ,GAAG,QAAQ,GAAG,UAAU,GAAG,MAAM,GAAG,OAAO,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"base.js","sourceRoot":"","sources":["../../../src/core/providers/base.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Provider factory — creates the right ProviderClient from a model name.
|
|
3
|
+
*/
|
|
4
|
+
import type { ProviderClient } from './base.js';
|
|
5
|
+
export declare function createProvider(modelId: string): ProviderClient;
|
|
6
|
+
export declare function getRequiredEnvVar(modelId: string): string;
|
|
7
|
+
//# sourceMappingURL=factory.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"factory.d.ts","sourceRoot":"","sources":["../../../src/core/providers/factory.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,WAAW,CAAC;AAKhD,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,cAAc,CAY9D;AAED,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,CAQzD"}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Provider factory — creates the right ProviderClient from a model name.
|
|
3
|
+
*/
|
|
4
|
+
import { AnthropicProvider } from './anthropic.js';
|
|
5
|
+
import { OpenAIProvider } from './openai.js';
|
|
6
|
+
import { GoogleProvider } from './google.js';
|
|
7
|
+
export function createProvider(modelId) {
|
|
8
|
+
if (modelId.startsWith('claude-'))
|
|
9
|
+
return new AnthropicProvider();
|
|
10
|
+
if (modelId.startsWith('gpt-') || modelId.startsWith('o3') || modelId.startsWith('o4'))
|
|
11
|
+
return new OpenAIProvider();
|
|
12
|
+
if (modelId.startsWith('gemini-'))
|
|
13
|
+
return new GoogleProvider();
|
|
14
|
+
if (modelId.startsWith('deepseek-'))
|
|
15
|
+
return new OpenAIProvider({ baseURL: 'https://api.deepseek.com/v1', apiKeyEnv: 'DEEPSEEK_API_KEY' });
|
|
16
|
+
if (modelId.startsWith('qwen-'))
|
|
17
|
+
return new OpenAIProvider({ baseURL: 'https://dashscope.aliyuncs.com/compatible-mode/v1', apiKeyEnv: 'DASHSCOPE_API_KEY' });
|
|
18
|
+
if (modelId.startsWith('llama-'))
|
|
19
|
+
return new OpenAIProvider(); // Via OpenAI-compatible endpoints
|
|
20
|
+
if (modelId.startsWith('codestral') || modelId.startsWith('mistral'))
|
|
21
|
+
return new OpenAIProvider({ baseURL: 'https://api.mistral.ai/v1', apiKeyEnv: 'MISTRAL_API_KEY' });
|
|
22
|
+
if (modelId === 'local')
|
|
23
|
+
return new OpenAIProvider({ baseURL: 'http://localhost:11434/v1', apiKeyEnv: 'OLLAMA_API_KEY' });
|
|
24
|
+
// Default: try OpenAI-compatible
|
|
25
|
+
return new OpenAIProvider();
|
|
26
|
+
}
|
|
27
|
+
export function getRequiredEnvVar(modelId) {
|
|
28
|
+
if (modelId.startsWith('claude-'))
|
|
29
|
+
return 'ANTHROPIC_API_KEY';
|
|
30
|
+
if (modelId.startsWith('gpt-') || modelId.startsWith('o3') || modelId.startsWith('o4'))
|
|
31
|
+
return 'OPENAI_API_KEY';
|
|
32
|
+
if (modelId.startsWith('gemini-'))
|
|
33
|
+
return 'GOOGLE_API_KEY';
|
|
34
|
+
if (modelId.startsWith('deepseek-'))
|
|
35
|
+
return 'DEEPSEEK_API_KEY';
|
|
36
|
+
if (modelId.startsWith('qwen-'))
|
|
37
|
+
return 'DASHSCOPE_API_KEY';
|
|
38
|
+
if (modelId.startsWith('codestral') || modelId.startsWith('mistral'))
|
|
39
|
+
return 'MISTRAL_API_KEY';
|
|
40
|
+
return 'OPENAI_API_KEY';
|
|
41
|
+
}
|
|
42
|
+
//# sourceMappingURL=factory.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"factory.js","sourceRoot":"","sources":["../../../src/core/providers/factory.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAE7C,MAAM,UAAU,cAAc,CAAC,OAAe;IAC5C,IAAI,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC;QAAE,OAAO,IAAI,iBAAiB,EAAE,CAAC;IAClE,IAAI,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC;QAAE,OAAO,IAAI,cAAc,EAAE,CAAC;IACpH,IAAI,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC;QAAE,OAAO,IAAI,cAAc,EAAE,CAAC;IAC/D,IAAI,OAAO,CAAC,UAAU,CAAC,WAAW,CAAC;QAAE,OAAO,IAAI,cAAc,CAAC,EAAE,OAAO,EAAE,6BAA6B,EAAE,SAAS,EAAE,kBAAkB,EAAE,CAAC,CAAC;IAC1I,IAAI,OAAO,CAAC,UAAU,CAAC,OAAO,CAAC;QAAE,OAAO,IAAI,cAAc,CAAC,EAAE,OAAO,EAAE,mDAAmD,EAAE,SAAS,EAAE,mBAAmB,EAAE,CAAC,CAAC;IAC7J,IAAI,OAAO,CAAC,UAAU,CAAC,QAAQ,CAAC;QAAE,OAAO,IAAI,cAAc,EAAE,CAAC,CAAC,kCAAkC;IACjG,IAAI,OAAO,CAAC,UAAU,CAAC,WAAW,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC;QAAE,OAAO,IAAI,cAAc,CAAC,EAAE,OAAO,EAAE,2BAA2B,EAAE,SAAS,EAAE,iBAAiB,EAAE,CAAC,CAAC;IACxK,IAAI,OAAO,KAAK,OAAO;QAAE,OAAO,IAAI,cAAc,CAAC,EAAE,OAAO,EAAE,2BAA2B,EAAE,SAAS,EAAE,gBAAgB,EAAE,CAAC,CAAC;IAE1H,iCAAiC;IACjC,OAAO,IAAI,cAAc,EAAE,CAAC;AAC9B,CAAC;AAED,MAAM,UAAU,iBAAiB,CAAC,OAAe;IAC/C,IAAI,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC;QAAE,OAAO,mBAAmB,CAAC;IAC9D,IAAI,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC;QAAE,OAAO,gBAAgB,CAAC;IAChH,IAAI,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC;QAAE,OAAO,gBAAgB,CAAC;IAC3D,IAAI,OAAO,CAAC,UAAU,CAAC,WAAW,CAAC;QAAE,OAAO,kBAAkB,CAAC;IAC/D,IAAI,OAAO,CAAC,UAAU,CAAC,OAAO,CAAC;QAAE,OAAO,mBAAmB,CAAC;IAC5D,IAAI,OAAO,CAAC,UAAU,CAAC,WAAW,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC;QAAE,OAAO,iBAAiB,CAAC;IAC/F,OAAO,gBAAgB,CAAC;AAC1B,CAAC"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Google Gemini provider — uses the REST API directly via fetch().
|
|
3
|
+
* No SDK dependency required.
|
|
4
|
+
*/
|
|
5
|
+
import type { ProviderClient, ProviderRequest, ProviderResponse } from './base.js';
|
|
6
|
+
export declare class GoogleProvider implements ProviderClient {
|
|
7
|
+
isAvailable(): Promise<boolean>;
|
|
8
|
+
call(request: ProviderRequest): Promise<ProviderResponse>;
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=google.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"google.d.ts","sourceRoot":"","sources":["../../../src/core/providers/google.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,eAAe,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAEnF,qBAAa,cAAe,YAAW,cAAc;IAC7C,WAAW,IAAI,OAAO,CAAC,OAAO,CAAC;IAI/B,IAAI,CAAC,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,gBAAgB,CAAC;CA+ChE"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Google Gemini provider — uses the REST API directly via fetch().
|
|
3
|
+
* No SDK dependency required.
|
|
4
|
+
*/
|
|
5
|
+
export class GoogleProvider {
|
|
6
|
+
async isAvailable() {
|
|
7
|
+
return !!process.env.GOOGLE_API_KEY;
|
|
8
|
+
}
|
|
9
|
+
async call(request) {
|
|
10
|
+
const apiKey = process.env.GOOGLE_API_KEY;
|
|
11
|
+
if (!apiKey) {
|
|
12
|
+
throw new Error('GOOGLE_API_KEY is required');
|
|
13
|
+
}
|
|
14
|
+
const contents = request.messages.map(m => ({
|
|
15
|
+
role: m.role === 'user' ? 'user' : 'model',
|
|
16
|
+
parts: [{ text: m.content }],
|
|
17
|
+
}));
|
|
18
|
+
const body = {
|
|
19
|
+
contents,
|
|
20
|
+
generationConfig: { maxOutputTokens: request.maxTokens ?? 1024 },
|
|
21
|
+
};
|
|
22
|
+
if (request.system) {
|
|
23
|
+
body.systemInstruction = { parts: [{ text: request.system }] };
|
|
24
|
+
}
|
|
25
|
+
const url = `https://generativelanguage.googleapis.com/v1beta/models/${request.model}:generateContent?key=${apiKey}`;
|
|
26
|
+
const response = await fetch(url, {
|
|
27
|
+
method: 'POST',
|
|
28
|
+
headers: { 'Content-Type': 'application/json' },
|
|
29
|
+
body: JSON.stringify(body),
|
|
30
|
+
});
|
|
31
|
+
if (!response.ok) {
|
|
32
|
+
const error = await response.text();
|
|
33
|
+
throw new Error(`Gemini API error (${response.status}): ${error}`);
|
|
34
|
+
}
|
|
35
|
+
const data = await response.json();
|
|
36
|
+
const candidates = data.candidates;
|
|
37
|
+
const usageMetadata = data.usageMetadata;
|
|
38
|
+
return {
|
|
39
|
+
content: candidates?.[0]?.content?.parts?.[0]?.text ?? '',
|
|
40
|
+
inputTokens: usageMetadata?.promptTokenCount ?? 0,
|
|
41
|
+
outputTokens: usageMetadata?.candidatesTokenCount ?? 0,
|
|
42
|
+
model: request.model,
|
|
43
|
+
stopReason: 'stop',
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
//# sourceMappingURL=google.js.map
|