model-test-bench 1.0.2
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 +201 -0
- package/README.md +389 -0
- package/dist/bin/mtb.d.ts +3 -0
- package/dist/bin/mtb.d.ts.map +1 -0
- package/dist/bin/mtb.js +148 -0
- package/dist/bin/mtb.js.map +1 -0
- package/dist/server/index.d.ts +13 -0
- package/dist/server/index.d.ts.map +1 -0
- package/dist/server/index.js +72 -0
- package/dist/server/index.js.map +1 -0
- package/dist/server/interfaces/evaluator.d.ts +15 -0
- package/dist/server/interfaces/evaluator.d.ts.map +1 -0
- package/dist/server/interfaces/evaluator.js +2 -0
- package/dist/server/interfaces/evaluator.js.map +1 -0
- package/dist/server/interfaces/logger.d.ts +9 -0
- package/dist/server/interfaces/logger.d.ts.map +1 -0
- package/dist/server/interfaces/logger.js +2 -0
- package/dist/server/interfaces/logger.js.map +1 -0
- package/dist/server/interfaces/runner.d.ts +9 -0
- package/dist/server/interfaces/runner.d.ts.map +1 -0
- package/dist/server/interfaces/runner.js +2 -0
- package/dist/server/interfaces/runner.js.map +1 -0
- package/dist/server/interfaces/storage.d.ts +36 -0
- package/dist/server/interfaces/storage.d.ts.map +1 -0
- package/dist/server/interfaces/storage.js +2 -0
- package/dist/server/interfaces/storage.js.map +1 -0
- package/dist/server/routes/eval-queue.d.ts +23 -0
- package/dist/server/routes/eval-queue.d.ts.map +1 -0
- package/dist/server/routes/eval-queue.js +45 -0
- package/dist/server/routes/eval-queue.js.map +1 -0
- package/dist/server/routes/evaluations.d.ts +8 -0
- package/dist/server/routes/evaluations.d.ts.map +1 -0
- package/dist/server/routes/evaluations.js +221 -0
- package/dist/server/routes/evaluations.js.map +1 -0
- package/dist/server/routes/providers.d.ts +5 -0
- package/dist/server/routes/providers.d.ts.map +1 -0
- package/dist/server/routes/providers.js +179 -0
- package/dist/server/routes/providers.js.map +1 -0
- package/dist/server/routes/run-queue.d.ts +17 -0
- package/dist/server/routes/run-queue.d.ts.map +1 -0
- package/dist/server/routes/run-queue.js +34 -0
- package/dist/server/routes/run-queue.js.map +1 -0
- package/dist/server/routes/run-sse.d.ts +18 -0
- package/dist/server/routes/run-sse.d.ts.map +1 -0
- package/dist/server/routes/run-sse.js +57 -0
- package/dist/server/routes/run-sse.js.map +1 -0
- package/dist/server/routes/runs.d.ts +9 -0
- package/dist/server/routes/runs.d.ts.map +1 -0
- package/dist/server/routes/runs.js +380 -0
- package/dist/server/routes/runs.js.map +1 -0
- package/dist/server/routes/scenarios.d.ts +5 -0
- package/dist/server/routes/scenarios.d.ts.map +1 -0
- package/dist/server/routes/scenarios.js +181 -0
- package/dist/server/routes/scenarios.js.map +1 -0
- package/dist/server/services/eval-helpers.d.ts +22 -0
- package/dist/server/services/eval-helpers.d.ts.map +1 -0
- package/dist/server/services/eval-helpers.js +75 -0
- package/dist/server/services/eval-helpers.js.map +1 -0
- package/dist/server/services/eval-parsers-debate-impl.d.ts +11 -0
- package/dist/server/services/eval-parsers-debate-impl.d.ts.map +1 -0
- package/dist/server/services/eval-parsers-debate-impl.js +133 -0
- package/dist/server/services/eval-parsers-debate-impl.js.map +1 -0
- package/dist/server/services/eval-parsers.d.ts +24 -0
- package/dist/server/services/eval-parsers.d.ts.map +1 -0
- package/dist/server/services/eval-parsers.js +153 -0
- package/dist/server/services/eval-parsers.js.map +1 -0
- package/dist/server/services/eval-prompts.d.ts +9 -0
- package/dist/server/services/eval-prompts.d.ts.map +1 -0
- package/dist/server/services/eval-prompts.js +164 -0
- package/dist/server/services/eval-prompts.js.map +1 -0
- package/dist/server/services/evaluator.d.ts +10 -0
- package/dist/server/services/evaluator.d.ts.map +1 -0
- package/dist/server/services/evaluator.js +143 -0
- package/dist/server/services/evaluator.js.map +1 -0
- package/dist/server/services/fs-adapter.d.ts +20 -0
- package/dist/server/services/fs-adapter.d.ts.map +1 -0
- package/dist/server/services/fs-adapter.js +13 -0
- package/dist/server/services/fs-adapter.js.map +1 -0
- package/dist/server/services/instruction-parser.d.ts +26 -0
- package/dist/server/services/instruction-parser.d.ts.map +1 -0
- package/dist/server/services/instruction-parser.js +121 -0
- package/dist/server/services/instruction-parser.js.map +1 -0
- package/dist/server/services/log-rotator.d.ts +20 -0
- package/dist/server/services/log-rotator.d.ts.map +1 -0
- package/dist/server/services/log-rotator.js +60 -0
- package/dist/server/services/log-rotator.js.map +1 -0
- package/dist/server/services/logger.d.ts +15 -0
- package/dist/server/services/logger.d.ts.map +1 -0
- package/dist/server/services/logger.js +69 -0
- package/dist/server/services/logger.js.map +1 -0
- package/dist/server/services/model-factory.d.ts +10 -0
- package/dist/server/services/model-factory.d.ts.map +1 -0
- package/dist/server/services/model-factory.js +33 -0
- package/dist/server/services/model-factory.js.map +1 -0
- package/dist/server/services/runner.d.ts +9 -0
- package/dist/server/services/runner.d.ts.map +1 -0
- package/dist/server/services/runner.js +99 -0
- package/dist/server/services/runner.js.map +1 -0
- package/dist/server/services/seeder.d.ts +5 -0
- package/dist/server/services/seeder.d.ts.map +1 -0
- package/dist/server/services/seeder.js +79 -0
- package/dist/server/services/seeder.js.map +1 -0
- package/dist/server/services/storage-test-helpers.d.ts +15 -0
- package/dist/server/services/storage-test-helpers.d.ts.map +1 -0
- package/dist/server/services/storage-test-helpers.js +151 -0
- package/dist/server/services/storage-test-helpers.js.map +1 -0
- package/dist/server/services/storage.d.ts +35 -0
- package/dist/server/services/storage.d.ts.map +1 -0
- package/dist/server/services/storage.js +219 -0
- package/dist/server/services/storage.js.map +1 -0
- package/dist/server/services/tools.d.ts +6 -0
- package/dist/server/services/tools.d.ts.map +1 -0
- package/dist/server/services/tools.js +94 -0
- package/dist/server/services/tools.js.map +1 -0
- package/dist/server/services/transcript-formatter.d.ts +18 -0
- package/dist/server/services/transcript-formatter.d.ts.map +1 -0
- package/dist/server/services/transcript-formatter.js +227 -0
- package/dist/server/services/transcript-formatter.js.map +1 -0
- package/dist/server/services/update-checker.d.ts +3 -0
- package/dist/server/services/update-checker.d.ts.map +1 -0
- package/dist/server/services/update-checker.js +34 -0
- package/dist/server/services/update-checker.js.map +1 -0
- package/dist/server/types/evaluation.d.ts +94 -0
- package/dist/server/types/evaluation.d.ts.map +1 -0
- package/dist/server/types/evaluation.js +5 -0
- package/dist/server/types/evaluation.js.map +1 -0
- package/dist/server/types/index.d.ts +5 -0
- package/dist/server/types/index.d.ts.map +1 -0
- package/dist/server/types/index.js +5 -0
- package/dist/server/types/index.js.map +1 -0
- package/dist/server/types/provider.d.ts +23 -0
- package/dist/server/types/provider.d.ts.map +1 -0
- package/dist/server/types/provider.js +5 -0
- package/dist/server/types/provider.js.map +1 -0
- package/dist/server/types/run.d.ts +31 -0
- package/dist/server/types/run.d.ts.map +1 -0
- package/dist/server/types/run.js +5 -0
- package/dist/server/types/run.js.map +1 -0
- package/dist/server/types/scenario.d.ts +19 -0
- package/dist/server/types/scenario.d.ts.map +1 -0
- package/dist/server/types/scenario.js +5 -0
- package/dist/server/types/scenario.js.map +1 -0
- package/dist/src/server/index.d.ts +13 -0
- package/dist/src/server/index.d.ts.map +1 -0
- package/dist/src/server/index.js +72 -0
- package/dist/src/server/index.js.map +1 -0
- package/dist/src/server/interfaces/evaluator.d.ts +15 -0
- package/dist/src/server/interfaces/evaluator.d.ts.map +1 -0
- package/dist/src/server/interfaces/evaluator.js +2 -0
- package/dist/src/server/interfaces/evaluator.js.map +1 -0
- package/dist/src/server/interfaces/logger.d.ts +9 -0
- package/dist/src/server/interfaces/logger.d.ts.map +1 -0
- package/dist/src/server/interfaces/logger.js +2 -0
- package/dist/src/server/interfaces/logger.js.map +1 -0
- package/dist/src/server/interfaces/runner.d.ts +9 -0
- package/dist/src/server/interfaces/runner.d.ts.map +1 -0
- package/dist/src/server/interfaces/runner.js +2 -0
- package/dist/src/server/interfaces/runner.js.map +1 -0
- package/dist/src/server/interfaces/storage.d.ts +36 -0
- package/dist/src/server/interfaces/storage.d.ts.map +1 -0
- package/dist/src/server/interfaces/storage.js +2 -0
- package/dist/src/server/interfaces/storage.js.map +1 -0
- package/dist/src/server/routes/eval-queue.d.ts +23 -0
- package/dist/src/server/routes/eval-queue.d.ts.map +1 -0
- package/dist/src/server/routes/eval-queue.js +45 -0
- package/dist/src/server/routes/eval-queue.js.map +1 -0
- package/dist/src/server/routes/evaluations.d.ts +8 -0
- package/dist/src/server/routes/evaluations.d.ts.map +1 -0
- package/dist/src/server/routes/evaluations.js +221 -0
- package/dist/src/server/routes/evaluations.js.map +1 -0
- package/dist/src/server/routes/providers.d.ts +5 -0
- package/dist/src/server/routes/providers.d.ts.map +1 -0
- package/dist/src/server/routes/providers.js +179 -0
- package/dist/src/server/routes/providers.js.map +1 -0
- package/dist/src/server/routes/run-queue.d.ts +17 -0
- package/dist/src/server/routes/run-queue.d.ts.map +1 -0
- package/dist/src/server/routes/run-queue.js +34 -0
- package/dist/src/server/routes/run-queue.js.map +1 -0
- package/dist/src/server/routes/run-sse.d.ts +18 -0
- package/dist/src/server/routes/run-sse.d.ts.map +1 -0
- package/dist/src/server/routes/run-sse.js +57 -0
- package/dist/src/server/routes/run-sse.js.map +1 -0
- package/dist/src/server/routes/runs.d.ts +9 -0
- package/dist/src/server/routes/runs.d.ts.map +1 -0
- package/dist/src/server/routes/runs.js +380 -0
- package/dist/src/server/routes/runs.js.map +1 -0
- package/dist/src/server/routes/scenarios.d.ts +5 -0
- package/dist/src/server/routes/scenarios.d.ts.map +1 -0
- package/dist/src/server/routes/scenarios.js +181 -0
- package/dist/src/server/routes/scenarios.js.map +1 -0
- package/dist/src/server/services/eval-helpers.d.ts +22 -0
- package/dist/src/server/services/eval-helpers.d.ts.map +1 -0
- package/dist/src/server/services/eval-helpers.js +75 -0
- package/dist/src/server/services/eval-helpers.js.map +1 -0
- package/dist/src/server/services/eval-parsers-debate-impl.d.ts +11 -0
- package/dist/src/server/services/eval-parsers-debate-impl.d.ts.map +1 -0
- package/dist/src/server/services/eval-parsers-debate-impl.js +133 -0
- package/dist/src/server/services/eval-parsers-debate-impl.js.map +1 -0
- package/dist/src/server/services/eval-parsers.d.ts +24 -0
- package/dist/src/server/services/eval-parsers.d.ts.map +1 -0
- package/dist/src/server/services/eval-parsers.js +153 -0
- package/dist/src/server/services/eval-parsers.js.map +1 -0
- package/dist/src/server/services/eval-prompts.d.ts +9 -0
- package/dist/src/server/services/eval-prompts.d.ts.map +1 -0
- package/dist/src/server/services/eval-prompts.js +164 -0
- package/dist/src/server/services/eval-prompts.js.map +1 -0
- package/dist/src/server/services/evaluator.d.ts +10 -0
- package/dist/src/server/services/evaluator.d.ts.map +1 -0
- package/dist/src/server/services/evaluator.js +143 -0
- package/dist/src/server/services/evaluator.js.map +1 -0
- package/dist/src/server/services/fs-adapter.d.ts +20 -0
- package/dist/src/server/services/fs-adapter.d.ts.map +1 -0
- package/dist/src/server/services/fs-adapter.js +13 -0
- package/dist/src/server/services/fs-adapter.js.map +1 -0
- package/dist/src/server/services/instruction-parser.d.ts +26 -0
- package/dist/src/server/services/instruction-parser.d.ts.map +1 -0
- package/dist/src/server/services/instruction-parser.js +121 -0
- package/dist/src/server/services/instruction-parser.js.map +1 -0
- package/dist/src/server/services/log-rotator.d.ts +20 -0
- package/dist/src/server/services/log-rotator.d.ts.map +1 -0
- package/dist/src/server/services/log-rotator.js +60 -0
- package/dist/src/server/services/log-rotator.js.map +1 -0
- package/dist/src/server/services/logger.d.ts +15 -0
- package/dist/src/server/services/logger.d.ts.map +1 -0
- package/dist/src/server/services/logger.js +69 -0
- package/dist/src/server/services/logger.js.map +1 -0
- package/dist/src/server/services/model-factory.d.ts +10 -0
- package/dist/src/server/services/model-factory.d.ts.map +1 -0
- package/dist/src/server/services/model-factory.js +33 -0
- package/dist/src/server/services/model-factory.js.map +1 -0
- package/dist/src/server/services/runner.d.ts +9 -0
- package/dist/src/server/services/runner.d.ts.map +1 -0
- package/dist/src/server/services/runner.js +99 -0
- package/dist/src/server/services/runner.js.map +1 -0
- package/dist/src/server/services/seeder.d.ts +5 -0
- package/dist/src/server/services/seeder.d.ts.map +1 -0
- package/dist/src/server/services/seeder.js +79 -0
- package/dist/src/server/services/seeder.js.map +1 -0
- package/dist/src/server/services/storage.d.ts +35 -0
- package/dist/src/server/services/storage.d.ts.map +1 -0
- package/dist/src/server/services/storage.js +219 -0
- package/dist/src/server/services/storage.js.map +1 -0
- package/dist/src/server/services/tools.d.ts +6 -0
- package/dist/src/server/services/tools.d.ts.map +1 -0
- package/dist/src/server/services/tools.js +94 -0
- package/dist/src/server/services/tools.js.map +1 -0
- package/dist/src/server/services/transcript-formatter.d.ts +18 -0
- package/dist/src/server/services/transcript-formatter.d.ts.map +1 -0
- package/dist/src/server/services/transcript-formatter.js +227 -0
- package/dist/src/server/services/transcript-formatter.js.map +1 -0
- package/dist/src/server/services/update-checker.d.ts +3 -0
- package/dist/src/server/services/update-checker.d.ts.map +1 -0
- package/dist/src/server/services/update-checker.js +34 -0
- package/dist/src/server/services/update-checker.js.map +1 -0
- package/dist/src/server/types/evaluation.d.ts +94 -0
- package/dist/src/server/types/evaluation.d.ts.map +1 -0
- package/dist/src/server/types/evaluation.js +5 -0
- package/dist/src/server/types/evaluation.js.map +1 -0
- package/dist/src/server/types/index.d.ts +5 -0
- package/dist/src/server/types/index.d.ts.map +1 -0
- package/dist/src/server/types/index.js +5 -0
- package/dist/src/server/types/index.js.map +1 -0
- package/dist/src/server/types/provider.d.ts +23 -0
- package/dist/src/server/types/provider.d.ts.map +1 -0
- package/dist/src/server/types/provider.js +5 -0
- package/dist/src/server/types/provider.js.map +1 -0
- package/dist/src/server/types/run.d.ts +31 -0
- package/dist/src/server/types/run.d.ts.map +1 -0
- package/dist/src/server/types/run.js +5 -0
- package/dist/src/server/types/run.js.map +1 -0
- package/dist/src/server/types/scenario.d.ts +19 -0
- package/dist/src/server/types/scenario.d.ts.map +1 -0
- package/dist/src/server/types/scenario.js +5 -0
- package/dist/src/server/types/scenario.js.map +1 -0
- package/dist/web/assets/index-AJu1Yn5F.js +70 -0
- package/dist/web/assets/index-C_ioEISr.css +1 -0
- package/dist/web/index.html +15 -0
- package/docs/schemas/provider-api.example.json +12 -0
- package/docs/schemas/provider-openai.example.json +11 -0
- package/docs/schemas/scenario-baseline.example.json +24 -0
- package/docs/schemas/scenario-carwash-baseline.example.json +22 -0
- package/docs/schemas/scenario-carwash-with-system-prompt.example.json +24 -0
- package/docs/schemas/scenario-golden-rules-baseline.example.json +24 -0
- package/docs/schemas/scenario-golden-rules-with-system-prompt.example.json +28 -0
- package/docs/schemas/scenario-negative-analysis-baseline.example.json +23 -0
- package/docs/schemas/scenario-negative-analysis-with-system-prompt.example.json +25 -0
- package/docs/schemas/scenario-with-system-prompt.example.json +25 -0
- package/package.json +97 -0
|
@@ -0,0 +1,221 @@
|
|
|
1
|
+
// ---------------------------------------------------------------------------
|
|
2
|
+
// Evaluation routes — start, list, get, stream evaluations
|
|
3
|
+
// ---------------------------------------------------------------------------
|
|
4
|
+
import { Router } from 'express';
|
|
5
|
+
import { v4 as uuidv4 } from 'uuid';
|
|
6
|
+
import { handleSSEConnection, broadcastSSE, closeSSE } from './run-sse.js';
|
|
7
|
+
import { EvalQueue, validateEvalEntry } from './eval-queue.js';
|
|
8
|
+
// Re-export so existing imports from this module continue to work
|
|
9
|
+
export { EvalQueue, validateEvalEntry } from './eval-queue.js';
|
|
10
|
+
// ---------------------------------------------------------------------------
|
|
11
|
+
// Helpers
|
|
12
|
+
// ---------------------------------------------------------------------------
|
|
13
|
+
function paramId(req) {
|
|
14
|
+
const raw = req.params.id;
|
|
15
|
+
return Array.isArray(raw) ? raw[0] : raw;
|
|
16
|
+
}
|
|
17
|
+
/** Strip rounds from evaluation for list endpoint summaries. */
|
|
18
|
+
function evalSummary(evaluation) {
|
|
19
|
+
const { rounds: _rounds, ...rest } = evaluation;
|
|
20
|
+
return rest;
|
|
21
|
+
}
|
|
22
|
+
// ---------------------------------------------------------------------------
|
|
23
|
+
// Router factory
|
|
24
|
+
// ---------------------------------------------------------------------------
|
|
25
|
+
export function createEvaluationRoutes(storage, evaluator, logger, queue) {
|
|
26
|
+
const router = Router();
|
|
27
|
+
const evalQueue = queue ?? new EvalQueue(1);
|
|
28
|
+
const sseSubscribers = new Map();
|
|
29
|
+
// POST / — start a new evaluation
|
|
30
|
+
router.post('/', async (req, res) => {
|
|
31
|
+
try {
|
|
32
|
+
const body = req.body;
|
|
33
|
+
const runId = body.runId;
|
|
34
|
+
if (!runId) {
|
|
35
|
+
res.status(400).json({ error: 'runId is required' });
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
const run = await storage.getRun(runId);
|
|
39
|
+
if (!run) {
|
|
40
|
+
res.status(404).json({ error: 'Run not found' });
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
// Validate evaluator entries (providerId + role)
|
|
44
|
+
const rawEvaluators = body.evaluators;
|
|
45
|
+
if (!Array.isArray(rawEvaluators) || rawEvaluators.length === 0) {
|
|
46
|
+
res.status(400).json({ error: 'evaluators must be a non-empty array' });
|
|
47
|
+
return;
|
|
48
|
+
}
|
|
49
|
+
const entries = [];
|
|
50
|
+
for (const raw of rawEvaluators) {
|
|
51
|
+
const result = validateEvalEntry(raw);
|
|
52
|
+
if (typeof result === 'string') {
|
|
53
|
+
res.status(400).json({ error: result });
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
entries.push(result);
|
|
57
|
+
}
|
|
58
|
+
// Resolve providerIds to real providers
|
|
59
|
+
const evaluators = [];
|
|
60
|
+
for (const entry of entries) {
|
|
61
|
+
const providerRecord = await storage.getProvider(entry.providerId);
|
|
62
|
+
if (!providerRecord) {
|
|
63
|
+
res.status(404).json({ error: `Provider not found: ${entry.providerId}` });
|
|
64
|
+
return;
|
|
65
|
+
}
|
|
66
|
+
evaluators.push({
|
|
67
|
+
providerName: providerRecord.providerName,
|
|
68
|
+
model: providerRecord.model,
|
|
69
|
+
apiKey: providerRecord.apiKey,
|
|
70
|
+
baseUrl: providerRecord.baseUrl,
|
|
71
|
+
role: entry.role,
|
|
72
|
+
});
|
|
73
|
+
}
|
|
74
|
+
// Validate maxRounds
|
|
75
|
+
const maxRounds = typeof body.maxRounds === 'number' ? body.maxRounds : 1;
|
|
76
|
+
if (maxRounds < 1 || maxRounds > 5) {
|
|
77
|
+
res.status(400).json({ error: 'maxRounds must be between 1 and 5' });
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
const maxBudgetUsd = typeof body.maxBudgetUsd === 'number' ? body.maxBudgetUsd : undefined;
|
|
81
|
+
const evalRequest = { runId, evaluators, maxRounds, maxBudgetUsd };
|
|
82
|
+
const now = new Date().toISOString();
|
|
83
|
+
const evaluation = {
|
|
84
|
+
id: uuidv4(),
|
|
85
|
+
runId,
|
|
86
|
+
status: 'pending',
|
|
87
|
+
evaluators,
|
|
88
|
+
rounds: [],
|
|
89
|
+
answerComparison: { matches: false, explanation: '', similarity: 0 },
|
|
90
|
+
criticalResults: [],
|
|
91
|
+
setupCompliance: {
|
|
92
|
+
instructionCompliance: { followed: [], violated: [], notApplicable: [], overallCompliance: 0 },
|
|
93
|
+
},
|
|
94
|
+
synthesis: { dimensionScores: {}, weightedTotal: 0, confidence: 0, dissenting: [] },
|
|
95
|
+
ledger: [],
|
|
96
|
+
totalCostUsd: 0,
|
|
97
|
+
createdAt: now,
|
|
98
|
+
updatedAt: now,
|
|
99
|
+
};
|
|
100
|
+
await storage.saveEvaluation(evaluation);
|
|
101
|
+
// Load related data for the evaluator
|
|
102
|
+
const providerSnapshot = run.providerSnapshot;
|
|
103
|
+
const scenario = run.scenarioSnapshot;
|
|
104
|
+
evalQueue.enqueue({
|
|
105
|
+
evaluation,
|
|
106
|
+
execute: async () => {
|
|
107
|
+
const callbacks = {
|
|
108
|
+
onStatusChange(status) {
|
|
109
|
+
const updatedEval = {
|
|
110
|
+
...evaluation,
|
|
111
|
+
status,
|
|
112
|
+
updatedAt: new Date().toISOString(),
|
|
113
|
+
};
|
|
114
|
+
broadcastSSE(evaluation.id, 'message', { type: 'status', status }, sseSubscribers);
|
|
115
|
+
storage.saveEvaluation(updatedEval).catch((saveErr) => {
|
|
116
|
+
logger.error('Failed to persist evaluation status change', {
|
|
117
|
+
evalId: evaluation.id,
|
|
118
|
+
status,
|
|
119
|
+
error: String(saveErr),
|
|
120
|
+
});
|
|
121
|
+
});
|
|
122
|
+
},
|
|
123
|
+
onProgress(step, detail) {
|
|
124
|
+
broadcastSSE(evaluation.id, 'message', { type: 'progress', step, detail }, sseSubscribers);
|
|
125
|
+
},
|
|
126
|
+
onMessage(info, msg) {
|
|
127
|
+
broadcastSSE(evaluation.id, 'message', {
|
|
128
|
+
type: 'sdkMessage',
|
|
129
|
+
phase: info.phase,
|
|
130
|
+
evaluatorRole: info.evaluatorRole,
|
|
131
|
+
roundNumber: info.roundNumber,
|
|
132
|
+
timestamp: msg.timestamp,
|
|
133
|
+
message: msg.message,
|
|
134
|
+
}, sseSubscribers);
|
|
135
|
+
},
|
|
136
|
+
};
|
|
137
|
+
try {
|
|
138
|
+
const result = await evaluator.evaluateRun(run, scenario, providerSnapshot, evalRequest, callbacks);
|
|
139
|
+
const finalEval = {
|
|
140
|
+
...result,
|
|
141
|
+
id: evaluation.id,
|
|
142
|
+
updatedAt: new Date().toISOString(),
|
|
143
|
+
};
|
|
144
|
+
await storage.saveEvaluation(finalEval);
|
|
145
|
+
}
|
|
146
|
+
catch (err) {
|
|
147
|
+
logger.error('Evaluation failed', { evalId: evaluation.id, error: String(err) });
|
|
148
|
+
broadcastSSE(evaluation.id, 'message', { type: 'status', status: 'failed' }, sseSubscribers);
|
|
149
|
+
const failedEval = {
|
|
150
|
+
...evaluation,
|
|
151
|
+
status: 'failed',
|
|
152
|
+
updatedAt: new Date().toISOString(),
|
|
153
|
+
};
|
|
154
|
+
await storage.saveEvaluation(failedEval).catch((saveErr) => {
|
|
155
|
+
logger.error('Failed to save failed evaluation', {
|
|
156
|
+
evalId: evaluation.id,
|
|
157
|
+
error: String(saveErr),
|
|
158
|
+
});
|
|
159
|
+
});
|
|
160
|
+
}
|
|
161
|
+
finally {
|
|
162
|
+
closeSSE(evaluation.id, sseSubscribers);
|
|
163
|
+
}
|
|
164
|
+
},
|
|
165
|
+
});
|
|
166
|
+
res.status(202).json({ id: evaluation.id, status: 'pending' });
|
|
167
|
+
}
|
|
168
|
+
catch (err) {
|
|
169
|
+
logger.error('Failed to start evaluation', { error: String(err) });
|
|
170
|
+
res.status(500).json({ error: 'Failed to start evaluation' });
|
|
171
|
+
}
|
|
172
|
+
});
|
|
173
|
+
// GET / — list evaluations
|
|
174
|
+
router.get('/', async (req, res) => {
|
|
175
|
+
try {
|
|
176
|
+
const filter = {};
|
|
177
|
+
if (typeof req.query.runId === 'string')
|
|
178
|
+
filter.runId = req.query.runId;
|
|
179
|
+
const evaluations = await storage.listEvaluations(filter);
|
|
180
|
+
res.json(evaluations.map(evalSummary));
|
|
181
|
+
}
|
|
182
|
+
catch (err) {
|
|
183
|
+
logger.error('Failed to list evaluations', { error: String(err) });
|
|
184
|
+
res.status(500).json({ error: 'Failed to list evaluations' });
|
|
185
|
+
}
|
|
186
|
+
});
|
|
187
|
+
// GET /:id — get full evaluation
|
|
188
|
+
router.get('/:id', async (req, res) => {
|
|
189
|
+
try {
|
|
190
|
+
const id = paramId(req);
|
|
191
|
+
const evaluation = await storage.getEvaluation(id);
|
|
192
|
+
if (!evaluation) {
|
|
193
|
+
res.status(404).json({ error: 'Evaluation not found' });
|
|
194
|
+
return;
|
|
195
|
+
}
|
|
196
|
+
res.json(evaluation);
|
|
197
|
+
}
|
|
198
|
+
catch (err) {
|
|
199
|
+
logger.error('Failed to get evaluation', { id: paramId(req), error: String(err) });
|
|
200
|
+
res.status(500).json({ error: 'Failed to get evaluation' });
|
|
201
|
+
}
|
|
202
|
+
});
|
|
203
|
+
// GET /:id/stream — SSE endpoint
|
|
204
|
+
router.get('/:id/stream', async (req, res) => {
|
|
205
|
+
try {
|
|
206
|
+
const id = paramId(req);
|
|
207
|
+
const evaluation = await storage.getEvaluation(id);
|
|
208
|
+
if (!evaluation) {
|
|
209
|
+
res.status(404).json({ error: 'Evaluation not found' });
|
|
210
|
+
return;
|
|
211
|
+
}
|
|
212
|
+
handleSSEConnection(req, res, id, sseSubscribers);
|
|
213
|
+
}
|
|
214
|
+
catch (err) {
|
|
215
|
+
logger.error('Failed to start eval SSE stream', { id: paramId(req), error: String(err) });
|
|
216
|
+
res.status(500).json({ error: 'Failed to start stream' });
|
|
217
|
+
}
|
|
218
|
+
});
|
|
219
|
+
return router;
|
|
220
|
+
}
|
|
221
|
+
//# sourceMappingURL=evaluations.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"evaluations.js","sourceRoot":"","sources":["../../../src/server/routes/evaluations.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import { Router } from 'express';
|
|
2
|
+
import type { IStorage } from '../interfaces/storage.js';
|
|
3
|
+
import type { ILogger } from '../interfaces/logger.js';
|
|
4
|
+
export declare function createProviderRoutes(storage: IStorage, logger: ILogger): Router;
|
|
5
|
+
//# sourceMappingURL=providers.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"providers.d.ts","sourceRoot":"","sources":["../../../src/server/routes/providers.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAGjC,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,yBAAyB,CAAC;AAyEvD,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,GAAG,MAAM,CAsH/E"}
|
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
import { Router } from 'express';
|
|
2
|
+
import { v4 as uuidv4 } from 'uuid';
|
|
3
|
+
// ─── Helpers ───────────────────────────────────────────────────────────
|
|
4
|
+
function paramId(req) {
|
|
5
|
+
const raw = req.params.id;
|
|
6
|
+
return Array.isArray(raw) ? raw[0] : raw;
|
|
7
|
+
}
|
|
8
|
+
function maskSecret(value) {
|
|
9
|
+
if (value.length <= 4)
|
|
10
|
+
return '****';
|
|
11
|
+
return '****' + value.slice(-4);
|
|
12
|
+
}
|
|
13
|
+
function maskProvider(provider) {
|
|
14
|
+
return { ...provider, apiKey: maskSecret(provider.apiKey) };
|
|
15
|
+
}
|
|
16
|
+
function maskProviderMetadata(provider) {
|
|
17
|
+
return {
|
|
18
|
+
id: provider.id,
|
|
19
|
+
name: provider.name,
|
|
20
|
+
providerName: provider.providerName,
|
|
21
|
+
model: provider.model,
|
|
22
|
+
createdAt: provider.createdAt,
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
function validateProviderBody(body) {
|
|
26
|
+
const errors = [];
|
|
27
|
+
if (typeof body !== 'object' || body === null) {
|
|
28
|
+
errors.push({ field: 'body', message: 'Request body must be a JSON object' });
|
|
29
|
+
return errors;
|
|
30
|
+
}
|
|
31
|
+
const b = body;
|
|
32
|
+
if (typeof b.name !== 'string' || b.name.trim().length === 0) {
|
|
33
|
+
errors.push({ field: 'name', message: 'name is required' });
|
|
34
|
+
}
|
|
35
|
+
if (typeof b.providerName !== 'string' || b.providerName.trim().length === 0) {
|
|
36
|
+
errors.push({ field: 'providerName', message: 'providerName is required' });
|
|
37
|
+
}
|
|
38
|
+
if (typeof b.model !== 'string' || b.model.trim().length === 0) {
|
|
39
|
+
errors.push({ field: 'model', message: 'model is required' });
|
|
40
|
+
}
|
|
41
|
+
if (typeof b.apiKey !== 'string' || b.apiKey.trim().length === 0) {
|
|
42
|
+
errors.push({ field: 'apiKey', message: 'apiKey is required' });
|
|
43
|
+
}
|
|
44
|
+
if (b.timeoutSeconds !== undefined) {
|
|
45
|
+
if (typeof b.timeoutSeconds !== 'number' || b.timeoutSeconds <= 0) {
|
|
46
|
+
errors.push({ field: 'timeoutSeconds', message: 'timeoutSeconds must be a positive number' });
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
if (b.temperature !== undefined && (typeof b.temperature !== 'number' || b.temperature < 0 || b.temperature > 2)) {
|
|
50
|
+
errors.push({ field: 'temperature', message: 'temperature must be a number between 0 and 2' });
|
|
51
|
+
}
|
|
52
|
+
if (b.maxTokens !== undefined && (typeof b.maxTokens !== 'number' || b.maxTokens <= 0)) {
|
|
53
|
+
errors.push({ field: 'maxTokens', message: 'maxTokens must be a positive number' });
|
|
54
|
+
}
|
|
55
|
+
if (b.topP !== undefined && (typeof b.topP !== 'number' || b.topP < 0 || b.topP > 1)) {
|
|
56
|
+
errors.push({ field: 'topP', message: 'topP must be a number between 0 and 1' });
|
|
57
|
+
}
|
|
58
|
+
return errors;
|
|
59
|
+
}
|
|
60
|
+
// ─── Router factory ────────────────────────────────────────────────────
|
|
61
|
+
export function createProviderRoutes(storage, logger) {
|
|
62
|
+
const router = Router();
|
|
63
|
+
// GET / — list all providers (metadata only)
|
|
64
|
+
router.get('/', async (_req, res) => {
|
|
65
|
+
try {
|
|
66
|
+
const providers = await storage.listProviders();
|
|
67
|
+
res.json(providers.map(maskProviderMetadata));
|
|
68
|
+
}
|
|
69
|
+
catch (err) {
|
|
70
|
+
logger.error('Failed to list providers', { error: String(err) });
|
|
71
|
+
res.status(500).json({ error: 'Failed to list providers' });
|
|
72
|
+
}
|
|
73
|
+
});
|
|
74
|
+
// GET /:id — get full provider (with masked secrets)
|
|
75
|
+
router.get('/:id', async (req, res) => {
|
|
76
|
+
try {
|
|
77
|
+
const id = paramId(req);
|
|
78
|
+
const provider = await storage.getProvider(id);
|
|
79
|
+
if (!provider) {
|
|
80
|
+
res.status(404).json({ error: 'Provider not found' });
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
res.json(maskProvider(provider));
|
|
84
|
+
}
|
|
85
|
+
catch (err) {
|
|
86
|
+
logger.error('Failed to get provider', { id: paramId(req), error: String(err) });
|
|
87
|
+
res.status(500).json({ error: 'Failed to get provider' });
|
|
88
|
+
}
|
|
89
|
+
});
|
|
90
|
+
// POST / — create provider
|
|
91
|
+
router.post('/', async (req, res) => {
|
|
92
|
+
const errors = validateProviderBody(req.body);
|
|
93
|
+
if (errors.length > 0) {
|
|
94
|
+
res.status(400).json({ errors });
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
97
|
+
try {
|
|
98
|
+
const body = req.body;
|
|
99
|
+
const now = new Date().toISOString();
|
|
100
|
+
const provider = {
|
|
101
|
+
id: uuidv4(),
|
|
102
|
+
name: body.name.trim(),
|
|
103
|
+
description: body.description ?? '',
|
|
104
|
+
providerName: body.providerName.trim(),
|
|
105
|
+
model: body.model.trim(),
|
|
106
|
+
apiKey: body.apiKey,
|
|
107
|
+
baseUrl: typeof body.baseUrl === 'string' ? body.baseUrl : undefined,
|
|
108
|
+
temperature: typeof body.temperature === 'number' ? body.temperature : undefined,
|
|
109
|
+
maxTokens: typeof body.maxTokens === 'number' ? body.maxTokens : undefined,
|
|
110
|
+
topP: typeof body.topP === 'number' ? body.topP : undefined,
|
|
111
|
+
timeoutSeconds: typeof body.timeoutSeconds === 'number' ? body.timeoutSeconds : 300,
|
|
112
|
+
createdAt: now,
|
|
113
|
+
updatedAt: now,
|
|
114
|
+
};
|
|
115
|
+
await storage.saveProvider(provider);
|
|
116
|
+
res.status(201).json(maskProvider(provider));
|
|
117
|
+
}
|
|
118
|
+
catch (err) {
|
|
119
|
+
logger.error('Failed to create provider', { error: String(err) });
|
|
120
|
+
res.status(500).json({ error: 'Failed to create provider' });
|
|
121
|
+
}
|
|
122
|
+
});
|
|
123
|
+
// PUT /:id — update provider
|
|
124
|
+
router.put('/:id', async (req, res) => {
|
|
125
|
+
try {
|
|
126
|
+
const id = paramId(req);
|
|
127
|
+
const existing = await storage.getProvider(id);
|
|
128
|
+
if (!existing) {
|
|
129
|
+
res.status(404).json({ error: 'Provider not found' });
|
|
130
|
+
return;
|
|
131
|
+
}
|
|
132
|
+
const errors = validateProviderBody(req.body);
|
|
133
|
+
if (errors.length > 0) {
|
|
134
|
+
res.status(400).json({ errors });
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
const body = req.body;
|
|
138
|
+
const updated = {
|
|
139
|
+
id: existing.id,
|
|
140
|
+
createdAt: existing.createdAt,
|
|
141
|
+
name: body.name.trim(),
|
|
142
|
+
description: body.description ?? '',
|
|
143
|
+
providerName: body.providerName.trim(),
|
|
144
|
+
model: body.model.trim(),
|
|
145
|
+
apiKey: body.apiKey,
|
|
146
|
+
baseUrl: typeof body.baseUrl === 'string' ? body.baseUrl : undefined,
|
|
147
|
+
temperature: typeof body.temperature === 'number' ? body.temperature : undefined,
|
|
148
|
+
maxTokens: typeof body.maxTokens === 'number' ? body.maxTokens : undefined,
|
|
149
|
+
topP: typeof body.topP === 'number' ? body.topP : undefined,
|
|
150
|
+
timeoutSeconds: typeof body.timeoutSeconds === 'number' ? body.timeoutSeconds : 300,
|
|
151
|
+
updatedAt: new Date().toISOString(),
|
|
152
|
+
};
|
|
153
|
+
await storage.saveProvider(updated);
|
|
154
|
+
res.json(maskProvider(updated));
|
|
155
|
+
}
|
|
156
|
+
catch (err) {
|
|
157
|
+
logger.error('Failed to update provider', { id: paramId(req), error: String(err) });
|
|
158
|
+
res.status(500).json({ error: 'Failed to update provider' });
|
|
159
|
+
}
|
|
160
|
+
});
|
|
161
|
+
// DELETE /:id — delete provider
|
|
162
|
+
router.delete('/:id', async (req, res) => {
|
|
163
|
+
try {
|
|
164
|
+
const id = paramId(req);
|
|
165
|
+
const deleted = await storage.deleteProvider(id);
|
|
166
|
+
if (!deleted) {
|
|
167
|
+
res.status(404).json({ error: 'Provider not found' });
|
|
168
|
+
return;
|
|
169
|
+
}
|
|
170
|
+
res.status(204).send();
|
|
171
|
+
}
|
|
172
|
+
catch (err) {
|
|
173
|
+
logger.error('Failed to delete provider', { id: paramId(req), error: String(err) });
|
|
174
|
+
res.status(500).json({ error: 'Failed to delete provider' });
|
|
175
|
+
}
|
|
176
|
+
});
|
|
177
|
+
return router;
|
|
178
|
+
}
|
|
179
|
+
//# sourceMappingURL=providers.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"providers.js","sourceRoot":"","sources":["../../../src/server/routes/providers.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,EAAE,IAAI,MAAM,EAAE,MAAM,MAAM,CAAC;AAMpC,0EAA0E;AAE1E,SAAS,OAAO,CAAC,GAAY;IAC3B,MAAM,GAAG,GAAG,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC;IAC1B,OAAO,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;AAC3C,CAAC;AAED,SAAS,UAAU,CAAC,KAAa;IAC/B,IAAI,KAAK,CAAC,MAAM,IAAI,CAAC;QAAE,OAAO,MAAM,CAAC;IACrC,OAAO,MAAM,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;AAClC,CAAC;AAED,SAAS,YAAY,CAAC,QAAkB;IACtC,OAAO,EAAE,GAAG,QAAQ,EAAE,MAAM,EAAE,UAAU,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;AAC9D,CAAC;AAED,SAAS,oBAAoB,CAAC,QAAkB;IAC9C,OAAO;QACL,EAAE,EAAE,QAAQ,CAAC,EAAE;QACf,IAAI,EAAE,QAAQ,CAAC,IAAI;QACnB,YAAY,EAAE,QAAQ,CAAC,YAAY;QACnC,KAAK,EAAE,QAAQ,CAAC,KAAK;QACrB,SAAS,EAAE,QAAQ,CAAC,SAAS;KAC9B,CAAC;AACJ,CAAC;AAOD,SAAS,oBAAoB,CAAC,IAAa;IACzC,MAAM,MAAM,GAAsB,EAAE,CAAC;IACrC,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,KAAK,IAAI,EAAE,CAAC;QAC9C,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,oCAAoC,EAAE,CAAC,CAAC;QAC9E,OAAO,MAAM,CAAC;IAChB,CAAC;IACD,MAAM,CAAC,GAAG,IAA+B,CAAC;IAE1C,IAAI,OAAO,CAAC,CAAC,IAAI,KAAK,QAAQ,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC7D,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,kBAAkB,EAAE,CAAC,CAAC;IAC9D,CAAC;IACD,IAAI,OAAO,CAAC,CAAC,YAAY,KAAK,QAAQ,IAAI,CAAC,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC7E,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,OAAO,EAAE,0BAA0B,EAAE,CAAC,CAAC;IAC9E,CAAC;IACD,IAAI,OAAO,CAAC,CAAC,KAAK,KAAK,QAAQ,IAAI,CAAC,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC/D,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,mBAAmB,EAAE,CAAC,CAAC;IAChE,CAAC;IACD,IAAI,OAAO,CAAC,CAAC,MAAM,KAAK,QAAQ,IAAI,CAAC,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACjE,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,OAAO,EAAE,oBAAoB,EAAE,CAAC,CAAC;IAClE,CAAC;IACD,IAAI,CAAC,CAAC,cAAc,KAAK,SAAS,EAAE,CAAC;QACnC,IAAI,OAAO,CAAC,CAAC,cAAc,KAAK,QAAQ,IAAI,CAAC,CAAC,cAAc,IAAI,CAAC,EAAE,CAAC;YAClE,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,gBAAgB,EAAE,OAAO,EAAE,0CAA0C,EAAE,CAAC,CAAC;QAChG,CAAC;IACH,CAAC;IACD,IAAI,CAAC,CAAC,WAAW,KAAK,SAAS,IAAI,CAAC,OAAO,CAAC,CAAC,WAAW,KAAK,QAAQ,IAAI,CAAC,CAAC,WAAW,GAAG,CAAC,IAAI,CAAC,CAAC,WAAW,GAAG,CAAC,CAAC,EAAE,CAAC;QACjH,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,aAAa,EAAE,OAAO,EAAE,8CAA8C,EAAE,CAAC,CAAC;IACjG,CAAC;IACD,IAAI,CAAC,CAAC,SAAS,KAAK,SAAS,IAAI,CAAC,OAAO,CAAC,CAAC,SAAS,KAAK,QAAQ,IAAI,CAAC,CAAC,SAAS,IAAI,CAAC,CAAC,EAAE,CAAC;QACvF,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,WAAW,EAAE,OAAO,EAAE,qCAAqC,EAAE,CAAC,CAAC;IACtF,CAAC;IACD,IAAI,CAAC,CAAC,IAAI,KAAK,SAAS,IAAI,CAAC,OAAO,CAAC,CAAC,IAAI,KAAK,QAAQ,IAAI,CAAC,CAAC,IAAI,GAAG,CAAC,IAAI,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,EAAE,CAAC;QACrF,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,uCAAuC,EAAE,CAAC,CAAC;IACnF,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,0EAA0E;AAE1E,MAAM,UAAU,oBAAoB,CAAC,OAAiB,EAAE,MAAe;IACrE,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC;IAExB,6CAA6C;IAC7C,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,EAAE,IAAa,EAAE,GAAa,EAAE,EAAE;QACrD,IAAI,CAAC;YACH,MAAM,SAAS,GAAG,MAAM,OAAO,CAAC,aAAa,EAAE,CAAC;YAChD,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC,CAAC;QAChD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,KAAK,CAAC,0BAA0B,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YACjE,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,0BAA0B,EAAE,CAAC,CAAC;QAC9D,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,qDAAqD;IACrD,MAAM,CAAC,GAAG,CAAC,MAAM,EAAE,KAAK,EAAE,GAAY,EAAE,GAAa,EAAE,EAAE;QACvD,IAAI,CAAC;YACH,MAAM,EAAE,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC;YACxB,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;YAC/C,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACd,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,oBAAoB,EAAE,CAAC,CAAC;gBACtD,OAAO;YACT,CAAC;YACD,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,CAAC;QACnC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,KAAK,CAAC,wBAAwB,EAAE,EAAE,EAAE,EAAE,OAAO,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YACjF,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,wBAAwB,EAAE,CAAC,CAAC;QAC5D,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,2BAA2B;IAC3B,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,EAAE,GAAY,EAAE,GAAa,EAAE,EAAE;QACrD,MAAM,MAAM,GAAG,oBAAoB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC9C,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;YACjC,OAAO;QACT,CAAC;QACD,IAAI,CAAC;YACH,MAAM,IAAI,GAAG,GAAG,CAAC,IAA+B,CAAC;YACjD,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;YACrC,MAAM,QAAQ,GAAa;gBACzB,EAAE,EAAE,MAAM,EAAE;gBACZ,IAAI,EAAG,IAAI,CAAC,IAAe,CAAC,IAAI,EAAE;gBAClC,WAAW,EAAG,IAAI,CAAC,WAAkC,IAAI,EAAE;gBAC3D,YAAY,EAAG,IAAI,CAAC,YAAuB,CAAC,IAAI,EAAE;gBAClD,KAAK,EAAG,IAAI,CAAC,KAAgB,CAAC,IAAI,EAAE;gBACpC,MAAM,EAAE,IAAI,CAAC,MAAgB;gBAC7B,OAAO,EAAE,OAAO,IAAI,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS;gBACpE,WAAW,EAAE,OAAO,IAAI,CAAC,WAAW,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,SAAS;gBAChF,SAAS,EAAE,OAAO,IAAI,CAAC,SAAS,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS;gBAC1E,IAAI,EAAE,OAAO,IAAI,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS;gBAC3D,cAAc,EAAE,OAAO,IAAI,CAAC,cAAc,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,GAAG;gBACnF,SAAS,EAAE,GAAG;gBACd,SAAS,EAAE,GAAG;aACf,CAAC;YACF,MAAM,OAAO,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;YACrC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,CAAC;QAC/C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,KAAK,CAAC,2BAA2B,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAClE,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC,CAAC;QAC/D,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,6BAA6B;IAC7B,MAAM,CAAC,GAAG,CAAC,MAAM,EAAE,KAAK,EAAE,GAAY,EAAE,GAAa,EAAE,EAAE;QACvD,IAAI,CAAC;YACH,MAAM,EAAE,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC;YACxB,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;YAC/C,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACd,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,oBAAoB,EAAE,CAAC,CAAC;gBACtD,OAAO;YACT,CAAC;YACD,MAAM,MAAM,GAAG,oBAAoB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YAC9C,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACtB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;gBACjC,OAAO;YACT,CAAC;YACD,MAAM,IAAI,GAAG,GAAG,CAAC,IAA+B,CAAC;YACjD,MAAM,OAAO,GAAa;gBACxB,EAAE,EAAE,QAAQ,CAAC,EAAE;gBACf,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAG,IAAI,CAAC,IAAe,CAAC,IAAI,EAAE;gBAClC,WAAW,EAAG,IAAI,CAAC,WAAkC,IAAI,EAAE;gBAC3D,YAAY,EAAG,IAAI,CAAC,YAAuB,CAAC,IAAI,EAAE;gBAClD,KAAK,EAAG,IAAI,CAAC,KAAgB,CAAC,IAAI,EAAE;gBACpC,MAAM,EAAE,IAAI,CAAC,MAAgB;gBAC7B,OAAO,EAAE,OAAO,IAAI,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS;gBACpE,WAAW,EAAE,OAAO,IAAI,CAAC,WAAW,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,SAAS;gBAChF,SAAS,EAAE,OAAO,IAAI,CAAC,SAAS,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS;gBAC1E,IAAI,EAAE,OAAO,IAAI,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS;gBAC3D,cAAc,EAAE,OAAO,IAAI,CAAC,cAAc,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,GAAG;gBACnF,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aACpC,CAAC;YACF,MAAM,OAAO,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;YACpC,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC;QAClC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,KAAK,CAAC,2BAA2B,EAAE,EAAE,EAAE,EAAE,OAAO,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YACpF,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC,CAAC;QAC/D,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,gCAAgC;IAChC,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,GAAY,EAAE,GAAa,EAAE,EAAE;QAC1D,IAAI,CAAC;YACH,MAAM,EAAE,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC;YACxB,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC,CAAC;YACjD,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,oBAAoB,EAAE,CAAC,CAAC;gBACtD,OAAO;YACT,CAAC;YACD,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC;QACzB,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,KAAK,CAAC,2BAA2B,EAAE,EAAE,EAAE,EAAE,OAAO,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YACpF,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC,CAAC;QAC/D,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,OAAO,MAAM,CAAC;AAChB,CAAC"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
export interface QueueEntry {
|
|
2
|
+
run: {
|
|
3
|
+
readonly id: string;
|
|
4
|
+
};
|
|
5
|
+
execute: () => Promise<void>;
|
|
6
|
+
}
|
|
7
|
+
export declare class RunQueue {
|
|
8
|
+
private readonly queue;
|
|
9
|
+
private active;
|
|
10
|
+
private readonly maxConcurrency;
|
|
11
|
+
constructor(maxConcurrency?: number);
|
|
12
|
+
enqueue(entry: QueueEntry): void;
|
|
13
|
+
get pendingCount(): number;
|
|
14
|
+
get activeCount(): number;
|
|
15
|
+
private drain;
|
|
16
|
+
}
|
|
17
|
+
//# sourceMappingURL=run-queue.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"run-queue.d.ts","sourceRoot":"","sources":["../../../src/server/routes/run-queue.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,UAAU;IACzB,GAAG,EAAE;QAAE,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAA;KAAE,CAAC;IAC7B,OAAO,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;CAC9B;AAED,qBAAa,QAAQ;IACnB,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAoB;IAC1C,OAAO,CAAC,MAAM,CAAK;IACnB,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;gBAE5B,cAAc,SAAI;IAI9B,OAAO,CAAC,KAAK,EAAE,UAAU,GAAG,IAAI;IAKhC,IAAI,YAAY,IAAI,MAAM,CAEzB;IAED,IAAI,WAAW,IAAI,MAAM,CAExB;YAEa,KAAK;CAWpB"}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
// ---------------------------------------------------------------------------
|
|
2
|
+
// RunQueue — in-memory concurrency limiter for scenario runs
|
|
3
|
+
// ---------------------------------------------------------------------------
|
|
4
|
+
export class RunQueue {
|
|
5
|
+
queue = [];
|
|
6
|
+
active = 0;
|
|
7
|
+
maxConcurrency;
|
|
8
|
+
constructor(maxConcurrency = 1) {
|
|
9
|
+
this.maxConcurrency = maxConcurrency;
|
|
10
|
+
}
|
|
11
|
+
enqueue(entry) {
|
|
12
|
+
this.queue.push(entry);
|
|
13
|
+
void this.drain();
|
|
14
|
+
}
|
|
15
|
+
get pendingCount() {
|
|
16
|
+
return this.queue.length;
|
|
17
|
+
}
|
|
18
|
+
get activeCount() {
|
|
19
|
+
return this.active;
|
|
20
|
+
}
|
|
21
|
+
async drain() {
|
|
22
|
+
while (this.active < this.maxConcurrency && this.queue.length > 0) {
|
|
23
|
+
const next = this.queue.shift();
|
|
24
|
+
if (!next)
|
|
25
|
+
break;
|
|
26
|
+
this.active++;
|
|
27
|
+
next.execute().finally(() => {
|
|
28
|
+
this.active--;
|
|
29
|
+
void this.drain();
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
//# sourceMappingURL=run-queue.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"run-queue.js","sourceRoot":"","sources":["../../../src/server/routes/run-queue.ts"],"names":[],"mappings":"AAAA,8EAA8E;AAC9E,6DAA6D;AAC7D,8EAA8E;AAO9E,MAAM,OAAO,QAAQ;IACF,KAAK,GAAiB,EAAE,CAAC;IAClC,MAAM,GAAG,CAAC,CAAC;IACF,cAAc,CAAS;IAExC,YAAY,cAAc,GAAG,CAAC;QAC5B,IAAI,CAAC,cAAc,GAAG,cAAc,CAAC;IACvC,CAAC;IAED,OAAO,CAAC,KAAiB;QACvB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACvB,KAAK,IAAI,CAAC,KAAK,EAAE,CAAC;IACpB,CAAC;IAED,IAAI,YAAY;QACd,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC;IAC3B,CAAC;IAED,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAEO,KAAK,CAAC,KAAK;QACjB,OAAO,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,cAAc,IAAI,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAClE,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;YAChC,IAAI,CAAC,IAAI;gBAAE,MAAM;YACjB,IAAI,CAAC,MAAM,EAAE,CAAC;YACd,IAAI,CAAC,OAAO,EAAE,CAAC,OAAO,CAAC,GAAG,EAAE;gBAC1B,IAAI,CAAC,MAAM,EAAE,CAAC;gBACd,KAAK,IAAI,CAAC,KAAK,EAAE,CAAC;YACpB,CAAC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type { Request, Response } from 'express';
|
|
2
|
+
import type { SDKMessageRecord } from '../types/index.js';
|
|
3
|
+
/** Per-run subscriber set for SSE clients. */
|
|
4
|
+
export type SSESubscriberMap = Map<string, Set<Response>>;
|
|
5
|
+
/**
|
|
6
|
+
* Set up an SSE connection for a given run ID.
|
|
7
|
+
* The response is kept open and messages are pushed via `sendSSE`.
|
|
8
|
+
*/
|
|
9
|
+
export declare function handleSSEConnection(req: Request, res: Response, runId: string, subscribers: SSESubscriberMap): void;
|
|
10
|
+
/**
|
|
11
|
+
* Broadcast a message to all SSE subscribers for a run.
|
|
12
|
+
*/
|
|
13
|
+
export declare function broadcastSSE(runId: string, event: string, data: SDKMessageRecord | Record<string, unknown> | string, subscribers: SSESubscriberMap): void;
|
|
14
|
+
/**
|
|
15
|
+
* Close all SSE connections for a run.
|
|
16
|
+
*/
|
|
17
|
+
export declare function closeSSE(runId: string, subscribers: SSESubscriberMap): void;
|
|
18
|
+
//# sourceMappingURL=run-sse.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"run-sse.d.ts","sourceRoot":"","sources":["../../../src/server/routes/run-sse.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AACjD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,mBAAmB,CAAC;AAE1D,8CAA8C;AAC9C,MAAM,MAAM,gBAAgB,GAAG,GAAG,CAAC,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC;AAE1D;;;GAGG;AACH,wBAAgB,mBAAmB,CACjC,GAAG,EAAE,OAAO,EACZ,GAAG,EAAE,QAAQ,EACb,KAAK,EAAE,MAAM,EACb,WAAW,EAAE,gBAAgB,GAC5B,IAAI,CAyBN;AAED;;GAEG;AACH,wBAAgB,YAAY,CAC1B,KAAK,EAAE,MAAM,EACb,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,gBAAgB,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM,EACzD,WAAW,EAAE,gBAAgB,GAC5B,IAAI,CAQN;AAED;;GAEG;AACH,wBAAgB,QAAQ,CAAC,KAAK,EAAE,MAAM,EAAE,WAAW,EAAE,gBAAgB,GAAG,IAAI,CAS3E"}
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
// ---------------------------------------------------------------------------
|
|
2
|
+
// SSE (Server-Sent Events) handler for real-time run streaming
|
|
3
|
+
// ---------------------------------------------------------------------------
|
|
4
|
+
/**
|
|
5
|
+
* Set up an SSE connection for a given run ID.
|
|
6
|
+
* The response is kept open and messages are pushed via `sendSSE`.
|
|
7
|
+
*/
|
|
8
|
+
export function handleSSEConnection(req, res, runId, subscribers) {
|
|
9
|
+
res.writeHead(200, {
|
|
10
|
+
'Content-Type': 'text/event-stream',
|
|
11
|
+
'Cache-Control': 'no-cache',
|
|
12
|
+
Connection: 'keep-alive',
|
|
13
|
+
'Access-Control-Allow-Origin': '*',
|
|
14
|
+
});
|
|
15
|
+
res.flushHeaders();
|
|
16
|
+
// Register subscriber
|
|
17
|
+
if (!subscribers.has(runId)) {
|
|
18
|
+
subscribers.set(runId, new Set());
|
|
19
|
+
}
|
|
20
|
+
subscribers.get(runId).add(res);
|
|
21
|
+
// Clean up on disconnect
|
|
22
|
+
req.on('close', () => {
|
|
23
|
+
const subs = subscribers.get(runId);
|
|
24
|
+
if (subs) {
|
|
25
|
+
subs.delete(res);
|
|
26
|
+
if (subs.size === 0) {
|
|
27
|
+
subscribers.delete(runId);
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Broadcast a message to all SSE subscribers for a run.
|
|
34
|
+
*/
|
|
35
|
+
export function broadcastSSE(runId, event, data, subscribers) {
|
|
36
|
+
const subs = subscribers.get(runId);
|
|
37
|
+
if (!subs || subs.size === 0)
|
|
38
|
+
return;
|
|
39
|
+
const payload = typeof data === 'string' ? data : JSON.stringify(data);
|
|
40
|
+
for (const res of subs) {
|
|
41
|
+
res.write(`event: ${event}\ndata: ${payload}\n\n`);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Close all SSE connections for a run.
|
|
46
|
+
*/
|
|
47
|
+
export function closeSSE(runId, subscribers) {
|
|
48
|
+
const subs = subscribers.get(runId);
|
|
49
|
+
if (!subs)
|
|
50
|
+
return;
|
|
51
|
+
for (const res of subs) {
|
|
52
|
+
res.write('event: done\ndata: {}\n\n');
|
|
53
|
+
res.end();
|
|
54
|
+
}
|
|
55
|
+
subscribers.delete(runId);
|
|
56
|
+
}
|
|
57
|
+
//# sourceMappingURL=run-sse.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"run-sse.js","sourceRoot":"","sources":["../../../src/server/routes/run-sse.ts"],"names":[],"mappings":"AAAA,8EAA8E;AAC9E,+DAA+D;AAC/D,8EAA8E;AAQ9E;;;GAGG;AACH,MAAM,UAAU,mBAAmB,CACjC,GAAY,EACZ,GAAa,EACb,KAAa,EACb,WAA6B;IAE7B,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE;QACjB,cAAc,EAAE,mBAAmB;QACnC,eAAe,EAAE,UAAU;QAC3B,UAAU,EAAE,YAAY;QACxB,6BAA6B,EAAE,GAAG;KACnC,CAAC,CAAC;IACH,GAAG,CAAC,YAAY,EAAE,CAAC;IAEnB,sBAAsB;IACtB,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;QAC5B,WAAW,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,GAAG,EAAE,CAAC,CAAC;IACpC,CAAC;IACD,WAAW,CAAC,GAAG,CAAC,KAAK,CAAE,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IAEjC,yBAAyB;IACzB,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;QACnB,MAAM,IAAI,GAAG,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACpC,IAAI,IAAI,EAAE,CAAC;YACT,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YACjB,IAAI,IAAI,CAAC,IAAI,KAAK,CAAC,EAAE,CAAC;gBACpB,WAAW,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAC5B,CAAC;QACH,CAAC;IACH,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,YAAY,CAC1B,KAAa,EACb,KAAa,EACb,IAAyD,EACzD,WAA6B;IAE7B,MAAM,IAAI,GAAG,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACpC,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,KAAK,CAAC;QAAE,OAAO;IAErC,MAAM,OAAO,GAAG,OAAO,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;IACvE,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,GAAG,CAAC,KAAK,CAAC,UAAU,KAAK,WAAW,OAAO,MAAM,CAAC,CAAC;IACrD,CAAC;AACH,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,QAAQ,CAAC,KAAa,EAAE,WAA6B;IACnE,MAAM,IAAI,GAAG,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACpC,IAAI,CAAC,IAAI;QAAE,OAAO;IAElB,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,GAAG,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAC;QACvC,GAAG,CAAC,GAAG,EAAE,CAAC;IACZ,CAAC;IACD,WAAW,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;AAC5B,CAAC"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { Router } from 'express';
|
|
2
|
+
import type { IStorage } from '../interfaces/storage.js';
|
|
3
|
+
import type { ILogger } from '../interfaces/logger.js';
|
|
4
|
+
import type { IRunner } from '../interfaces/runner.js';
|
|
5
|
+
import type { IEvaluator } from '../interfaces/evaluator.js';
|
|
6
|
+
import { RunQueue } from './run-queue.js';
|
|
7
|
+
export { RunQueue } from './run-queue.js';
|
|
8
|
+
export declare function createRunRoutes(storage: IStorage, runner: IRunner, logger: ILogger, evaluator?: IEvaluator, queue?: RunQueue): Router;
|
|
9
|
+
//# sourceMappingURL=runs.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runs.d.ts","sourceRoot":"","sources":["../../../src/server/routes/runs.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAGjC,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,yBAAyB,CAAC;AACvD,OAAO,KAAK,EAAE,OAAO,EAAgB,MAAM,yBAAyB,CAAC;AACrE,OAAO,KAAK,EAAE,UAAU,EAAuB,MAAM,4BAA4B,CAAC;AAmClF,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAC1C,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAY1C,wBAAgB,eAAe,CAC7B,OAAO,EAAE,QAAQ,EACjB,MAAM,EAAE,OAAO,EACf,MAAM,EAAE,OAAO,EACf,SAAS,CAAC,EAAE,UAAU,EACtB,KAAK,CAAC,EAAE,QAAQ,GACf,MAAM,CAuWR"}
|