@agentfare/core 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.
Files changed (102) hide show
  1. package/README.md +24 -0
  2. package/dist/analyzer/auto-model-selector.d.ts +3 -0
  3. package/dist/analyzer/auto-model-selector.d.ts.map +1 -0
  4. package/dist/analyzer/auto-model-selector.js +15 -0
  5. package/dist/analyzer/auto-model-selector.js.map +1 -0
  6. package/dist/analyzer/cache.d.ts +16 -0
  7. package/dist/analyzer/cache.d.ts.map +1 -0
  8. package/dist/analyzer/cache.js +113 -0
  9. package/dist/analyzer/cache.js.map +1 -0
  10. package/dist/analyzer/llm-analyzer.d.ts +4 -0
  11. package/dist/analyzer/llm-analyzer.d.ts.map +1 -0
  12. package/dist/analyzer/llm-analyzer.js +94 -0
  13. package/dist/analyzer/llm-analyzer.js.map +1 -0
  14. package/dist/analyzer/rules.d.ts +3 -0
  15. package/dist/analyzer/rules.d.ts.map +1 -0
  16. package/dist/analyzer/rules.js +223 -0
  17. package/dist/analyzer/rules.js.map +1 -0
  18. package/dist/analyzer/types.d.ts +52 -0
  19. package/dist/analyzer/types.d.ts.map +1 -0
  20. package/dist/analyzer/types.js +35 -0
  21. package/dist/analyzer/types.js.map +1 -0
  22. package/dist/config/defaults.d.ts +3 -0
  23. package/dist/config/defaults.d.ts.map +1 -0
  24. package/dist/config/defaults.js +41 -0
  25. package/dist/config/defaults.js.map +1 -0
  26. package/dist/config/enterprise.d.ts +7 -0
  27. package/dist/config/enterprise.d.ts.map +1 -0
  28. package/dist/config/enterprise.js +25 -0
  29. package/dist/config/enterprise.js.map +1 -0
  30. package/dist/config/loader.d.ts +10 -0
  31. package/dist/config/loader.d.ts.map +1 -0
  32. package/dist/config/loader.js +134 -0
  33. package/dist/config/loader.js.map +1 -0
  34. package/dist/config/types.d.ts +81 -0
  35. package/dist/config/types.d.ts.map +1 -0
  36. package/dist/config/types.js +3 -0
  37. package/dist/config/types.js.map +1 -0
  38. package/dist/errors.d.ts +14 -0
  39. package/dist/errors.d.ts.map +1 -0
  40. package/dist/errors.js +25 -0
  41. package/dist/errors.js.map +1 -0
  42. package/dist/index.d.ts +29 -0
  43. package/dist/index.d.ts.map +1 -0
  44. package/dist/index.js +63 -0
  45. package/dist/index.js.map +1 -0
  46. package/dist/optimizer/eval-runner.d.ts +23 -0
  47. package/dist/optimizer/eval-runner.d.ts.map +1 -0
  48. package/dist/optimizer/eval-runner.js +115 -0
  49. package/dist/optimizer/eval-runner.js.map +1 -0
  50. package/dist/optimizer/online-learning.d.ts +37 -0
  51. package/dist/optimizer/online-learning.d.ts.map +1 -0
  52. package/dist/optimizer/online-learning.js +128 -0
  53. package/dist/optimizer/online-learning.js.map +1 -0
  54. package/dist/optimizer/pipeline-parser.d.ts +4 -0
  55. package/dist/optimizer/pipeline-parser.d.ts.map +1 -0
  56. package/dist/optimizer/pipeline-parser.js +72 -0
  57. package/dist/optimizer/pipeline-parser.js.map +1 -0
  58. package/dist/optimizer/search.d.ts +7 -0
  59. package/dist/optimizer/search.d.ts.map +1 -0
  60. package/dist/optimizer/search.js +323 -0
  61. package/dist/optimizer/search.js.map +1 -0
  62. package/dist/optimizer/types.d.ts +31 -0
  63. package/dist/optimizer/types.d.ts.map +1 -0
  64. package/dist/optimizer/types.js +11 -0
  65. package/dist/optimizer/types.js.map +1 -0
  66. package/dist/routing/cross-provider.d.ts +8 -0
  67. package/dist/routing/cross-provider.d.ts.map +1 -0
  68. package/dist/routing/cross-provider.js +22 -0
  69. package/dist/routing/cross-provider.js.map +1 -0
  70. package/dist/routing/enterprise.d.ts +7 -0
  71. package/dist/routing/enterprise.d.ts.map +1 -0
  72. package/dist/routing/enterprise.js +16 -0
  73. package/dist/routing/enterprise.js.map +1 -0
  74. package/dist/routing/router.d.ts +19 -0
  75. package/dist/routing/router.d.ts.map +1 -0
  76. package/dist/routing/router.js +125 -0
  77. package/dist/routing/router.js.map +1 -0
  78. package/dist/routing/same-provider.d.ts +4 -0
  79. package/dist/routing/same-provider.d.ts.map +1 -0
  80. package/dist/routing/same-provider.js +24 -0
  81. package/dist/routing/same-provider.js.map +1 -0
  82. package/dist/tracker/cost-tracker.d.ts +13 -0
  83. package/dist/tracker/cost-tracker.d.ts.map +1 -0
  84. package/dist/tracker/cost-tracker.js +39 -0
  85. package/dist/tracker/cost-tracker.js.map +1 -0
  86. package/dist/tracker/database.d.ts +64 -0
  87. package/dist/tracker/database.d.ts.map +1 -0
  88. package/dist/tracker/database.js +194 -0
  89. package/dist/tracker/database.js.map +1 -0
  90. package/dist/tracker/quality-signal.d.ts +29 -0
  91. package/dist/tracker/quality-signal.d.ts.map +1 -0
  92. package/dist/tracker/quality-signal.js +95 -0
  93. package/dist/tracker/quality-signal.js.map +1 -0
  94. package/dist/tracker/report-exporter.d.ts +28 -0
  95. package/dist/tracker/report-exporter.d.ts.map +1 -0
  96. package/dist/tracker/report-exporter.js +39 -0
  97. package/dist/tracker/report-exporter.js.map +1 -0
  98. package/dist/utils/tokens.d.ts +10 -0
  99. package/dist/utils/tokens.d.ts.map +1 -0
  100. package/dist/utils/tokens.js +26 -0
  101. package/dist/utils/tokens.js.map +1 -0
  102. package/package.json +22 -0
@@ -0,0 +1,115 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.loadEvalDataset = loadEvalDataset;
37
+ exports.evaluateCombo = evaluateCombo;
38
+ const fs = __importStar(require("node:fs"));
39
+ const path = __importStar(require("node:path"));
40
+ function loadEvalDataset(datasetPath) {
41
+ const resolved = path.resolve(datasetPath);
42
+ if (!fs.existsSync(resolved))
43
+ throw new Error(`eval dataset 不存在: ${resolved}`);
44
+ const raw = JSON.parse(fs.readFileSync(resolved, "utf-8"));
45
+ if (!Array.isArray(raw))
46
+ throw new Error("eval dataset 必须是 JSON 数组");
47
+ return raw.map((item) => ({
48
+ input: typeof item.input === "string" ? item.input : JSON.stringify(item),
49
+ expected: item.expected ?? undefined,
50
+ metadata: item.metadata ?? undefined,
51
+ }));
52
+ }
53
+ async function evaluateCombo(combo, samples, fetchFn, options) {
54
+ // ISSUE-033: for multi-step pipelines, concatenate all models into the prompt
55
+ // so the eval considers the full combo rather than only the first model.
56
+ const modelId = Object.values(combo)[0] ?? "";
57
+ const comboDescription = Object.entries(combo).map(([step, model]) => `${step}=${model}`).join('|');
58
+ const baseUrl = options?.baseUrl ?? "https://api.openai.com/v1";
59
+ const url = `${baseUrl}/chat/completions`;
60
+ const results = [];
61
+ for (const sample of samples) {
62
+ const start = Date.now();
63
+ let output = "";
64
+ let passed = false;
65
+ try {
66
+ const response = await fetchFn(url, {
67
+ method: "POST",
68
+ headers: {
69
+ "Content-Type": "application/json",
70
+ ...(options?.apiKey
71
+ ? { Authorization: `Bearer ${options.apiKey}` }
72
+ : {}),
73
+ },
74
+ body: JSON.stringify({
75
+ model: modelId,
76
+ messages: [{ role: "user", content: `[pipeline: ${comboDescription}] ${sample.input}` }],
77
+ max_tokens: 100,
78
+ }),
79
+ });
80
+ const data = await response.json();
81
+ output = data.choices?.[0]?.message?.content ?? "";
82
+ if (sample.expected) {
83
+ const metric = options?.metric ?? "accuracy";
84
+ passed =
85
+ metric === "contains"
86
+ ? output.toLowerCase().includes(sample.expected.toLowerCase())
87
+ : output.trim().toLowerCase() ===
88
+ sample.expected.trim().toLowerCase() ||
89
+ output.includes(sample.expected);
90
+ }
91
+ else {
92
+ passed = true;
93
+ }
94
+ }
95
+ catch {
96
+ output = "[error]";
97
+ passed = false;
98
+ }
99
+ results.push({
100
+ input: sample.input,
101
+ output,
102
+ passed,
103
+ latencyMs: Date.now() - start,
104
+ });
105
+ }
106
+ const passedCount = results.filter((r) => r.passed).length;
107
+ const totalLatency = results.reduce((sum, r) => sum + r.latencyMs, 0);
108
+ return {
109
+ combo,
110
+ samples: results,
111
+ accuracy: results.length > 0 ? passedCount / results.length : 0,
112
+ avgLatencyMs: results.length > 0 ? totalLatency / results.length : 0,
113
+ };
114
+ }
115
+ //# sourceMappingURL=eval-runner.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eval-runner.js","sourceRoot":"","sources":["../../src/optimizer/eval-runner.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,37 @@
1
+ import type { TrackingDatabase } from "../tracker/database.js";
2
+ import type { QualitySignal } from "../tracker/quality-signal.js";
3
+ export interface ModelScore {
4
+ model: string;
5
+ stepType: string;
6
+ avgAccuracy: number;
7
+ avgLatencyMs: number;
8
+ avgCostPerTask: number;
9
+ sampleCount: number;
10
+ }
11
+ export interface OnlineLearnerConfig {
12
+ minSamplesBeforeSuggest: number;
13
+ suggestionChannel: "cli" | "log" | "off";
14
+ autoApply: boolean;
15
+ windowSize: number;
16
+ }
17
+ export declare class OnlineLearner {
18
+ private db;
19
+ private config;
20
+ private signals;
21
+ private modelScores;
22
+ private dirty;
23
+ constructor(db: TrackingDatabase, config?: Partial<OnlineLearnerConfig>);
24
+ recordSignal(model: string, stepType: string, signal: QualitySignal): void;
25
+ getScore(model: string, stepType: string): ModelScore | undefined;
26
+ getAllScores(): ModelScore[];
27
+ getSuggestions(): Array<{
28
+ from: string;
29
+ to: string;
30
+ stepType: string;
31
+ reason: string;
32
+ }>;
33
+ private updateModelScore;
34
+ private loadScoresFromDB;
35
+ persistToDB(): void;
36
+ }
37
+ //# sourceMappingURL=online-learning.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"online-learning.d.ts","sourceRoot":"","sources":["../../src/optimizer/online-learning.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAC/D,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,8BAA8B,CAAC;AAElE,MAAM,WAAW,UAAU;IACzB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;IACrB,cAAc,EAAE,MAAM,CAAC;IACvB,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,mBAAmB;IAClC,uBAAuB,EAAE,MAAM,CAAC;IAChC,iBAAiB,EAAE,KAAK,GAAG,KAAK,GAAG,KAAK,CAAC;IACzC,SAAS,EAAE,OAAO,CAAC;IACnB,UAAU,EAAE,MAAM,CAAC;CACpB;AAgBD,qBAAa,aAAa;IAOtB,OAAO,CAAC,EAAE;IANZ,OAAO,CAAC,MAAM,CAAsB;IACpC,OAAO,CAAC,OAAO,CAAsB;IACrC,OAAO,CAAC,WAAW,CAAsC;IACzD,OAAO,CAAC,KAAK,CAAkB;gBAGrB,EAAE,EAAE,gBAAgB,EAC5B,MAAM,CAAC,EAAE,OAAO,CAAC,mBAAmB,CAAC;IAQvC,YAAY,CACV,KAAK,EAAE,MAAM,EACb,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,aAAa,GACpB,IAAI;IAQP,QAAQ,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,UAAU,GAAG,SAAS;IAKjE,YAAY,IAAI,UAAU,EAAE;IAI5B,cAAc,IAAI,KAAK,CAAC;QACtB,IAAI,EAAE,MAAM,CAAC;QACb,EAAE,EAAE,MAAM,CAAC;QACX,QAAQ,EAAE,MAAM,CAAC;QACjB,MAAM,EAAE,MAAM,CAAC;KAChB,CAAC;IAuBF,OAAO,CAAC,gBAAgB;IAqCxB,OAAO,CAAC,gBAAgB;IAmBxB,WAAW,IAAI,IAAI;CAsBpB"}
@@ -0,0 +1,128 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.OnlineLearner = void 0;
4
+ const DEFAULT_CONFIG = {
5
+ minSamplesBeforeSuggest: 50,
6
+ suggestionChannel: "cli",
7
+ autoApply: false,
8
+ windowSize: 200,
9
+ };
10
+ class OnlineLearner {
11
+ db;
12
+ config;
13
+ signals = [];
14
+ modelScores = new Map();
15
+ dirty = false;
16
+ constructor(db, config) {
17
+ this.db = db;
18
+ this.config = { ...DEFAULT_CONFIG, ...config };
19
+ // ISSUE-038: Load historical scores from DB
20
+ this.loadScoresFromDB();
21
+ process.on("exit", () => this.persistToDB());
22
+ }
23
+ recordSignal(model, stepType, signal) {
24
+ this.signals.push({ model, stepType, signal, timestamp: Date.now() });
25
+ if (this.signals.length > this.config.windowSize) {
26
+ this.signals = this.signals.slice(-this.config.windowSize);
27
+ }
28
+ this.updateModelScore(model, stepType, signal);
29
+ }
30
+ getScore(model, stepType) {
31
+ const key = `${model}::${stepType}`;
32
+ return this.modelScores.get(key);
33
+ }
34
+ getAllScores() {
35
+ return Array.from(this.modelScores.values());
36
+ }
37
+ getSuggestions() {
38
+ const suggestions = [];
39
+ for (const score of this.modelScores.values()) {
40
+ if (score.sampleCount < this.config.minSamplesBeforeSuggest)
41
+ continue;
42
+ if (score.avgAccuracy < 0.5) {
43
+ suggestions.push({
44
+ from: score.model,
45
+ to: "",
46
+ stepType: score.stepType,
47
+ reason: `${score.model} 在 ${score.stepType} 步骤准确率仅 ${(score.avgAccuracy * 100).toFixed(0)}%,建议考虑其他模型`,
48
+ });
49
+ }
50
+ }
51
+ return suggestions;
52
+ }
53
+ updateModelScore(model, stepType, signal) {
54
+ const key = `${model}::${stepType}`;
55
+ const existing = this.modelScores.get(key);
56
+ const accuracy = signal === "success"
57
+ ? 1.0
58
+ : signal === "retry"
59
+ ? 0.3
60
+ : signal === "manual_switch"
61
+ ? 0.0
62
+ : signal === "error"
63
+ ? 0.0
64
+ : 0.5;
65
+ if (existing) {
66
+ const totalSamples = existing.sampleCount + 1;
67
+ existing.avgAccuracy =
68
+ (existing.avgAccuracy * existing.sampleCount + accuracy) / totalSamples;
69
+ existing.sampleCount = totalSamples;
70
+ }
71
+ else {
72
+ this.modelScores.set(key, {
73
+ model,
74
+ stepType,
75
+ avgAccuracy: accuracy,
76
+ avgLatencyMs: 0,
77
+ avgCostPerTask: 0,
78
+ sampleCount: 1,
79
+ });
80
+ }
81
+ this.dirty = true;
82
+ }
83
+ loadScoresFromDB() {
84
+ try {
85
+ const rows = this.db.db
86
+ .prepare("SELECT model, step_type, avg_accuracy, avg_latency_ms, avg_cost_per_task, sample_count FROM model_scores")
87
+ .all();
88
+ for (const row of rows) {
89
+ const key = `${row.model}::${row.step_type}`;
90
+ this.modelScores.set(key, {
91
+ model: row.model,
92
+ stepType: row.step_type,
93
+ avgAccuracy: row.avg_accuracy,
94
+ avgLatencyMs: row.avg_latency_ms,
95
+ avgCostPerTask: row.avg_cost_per_task,
96
+ sampleCount: row.sample_count,
97
+ });
98
+ }
99
+ }
100
+ catch { }
101
+ }
102
+ persistToDB() {
103
+ if (!this.dirty)
104
+ return;
105
+ try {
106
+ const upsert = this.db.db.prepare(`
107
+ INSERT INTO model_scores (model, step_type, avg_accuracy, avg_latency_ms, avg_cost_per_task, sample_count, last_updated)
108
+ VALUES (?, ?, ?, ?, ?, ?, datetime('now'))
109
+ ON CONFLICT(model, step_type) DO UPDATE SET
110
+ avg_accuracy = excluded.avg_accuracy,
111
+ avg_latency_ms = excluded.avg_latency_ms,
112
+ avg_cost_per_task = excluded.avg_cost_per_task,
113
+ sample_count = excluded.sample_count,
114
+ last_updated = datetime('now')
115
+ `);
116
+ const batch = this.db.db.transaction((scores) => {
117
+ for (const s of scores) {
118
+ upsert.run(s.model, s.stepType, s.avgAccuracy, s.avgLatencyMs, s.avgCostPerTask, s.sampleCount);
119
+ }
120
+ });
121
+ batch(Array.from(this.modelScores.values()));
122
+ this.dirty = false;
123
+ }
124
+ catch { }
125
+ }
126
+ }
127
+ exports.OnlineLearner = OnlineLearner;
128
+ //# sourceMappingURL=online-learning.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"online-learning.js","sourceRoot":"","sources":["../../src/optimizer/online-learning.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { Pipeline } from "./types.js";
2
+ export declare function parsePipelineYAML(yaml: string): Pipeline;
3
+ export declare function computeTotalCombinations(pipeline: Pipeline): number;
4
+ //# sourceMappingURL=pipeline-parser.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pipeline-parser.d.ts","sourceRoot":"","sources":["../../src/optimizer/pipeline-parser.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAC;AAiB3C,wBAAgB,iBAAiB,CAAC,IAAI,EAAE,MAAM,GAAG,QAAQ,CAmDxD;AAED,wBAAgB,wBAAwB,CAAC,QAAQ,EAAE,QAAQ,GAAG,MAAM,CAKnE"}
@@ -0,0 +1,72 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.parsePipelineYAML = parsePipelineYAML;
4
+ exports.computeTotalCombinations = computeTotalCombinations;
5
+ /**
6
+ * ISSUE-043: Robust key-value extraction that handles colons inside values.
7
+ */
8
+ function extractValue(line, key) {
9
+ const prefix = key + ":";
10
+ if (!line.startsWith(prefix))
11
+ return null;
12
+ let value = line.slice(prefix.length).trim();
13
+ if (value.startsWith('"') && value.endsWith('"')) {
14
+ value = value.slice(1, -1);
15
+ }
16
+ else if (value.startsWith("'") && value.endsWith("'")) {
17
+ value = value.slice(1, -1);
18
+ }
19
+ return value;
20
+ }
21
+ function parsePipelineYAML(yaml) {
22
+ const lines = yaml.split("\n");
23
+ let name = "";
24
+ const steps = [];
25
+ let currentStep = null;
26
+ let inCandidates = false;
27
+ for (const line of lines) {
28
+ const trimmed = line.trim();
29
+ if (!trimmed || trimmed.startsWith("#"))
30
+ continue;
31
+ const nameVal = extractValue(trimmed, "name");
32
+ if (nameVal !== null) {
33
+ name = nameVal;
34
+ continue;
35
+ }
36
+ const idVal = extractValue(trimmed, "- id");
37
+ if (idVal !== null) {
38
+ if (currentStep)
39
+ steps.push(currentStep);
40
+ currentStep = {
41
+ id: idVal,
42
+ description: "",
43
+ candidateModels: [],
44
+ };
45
+ inCandidates = false;
46
+ continue;
47
+ }
48
+ if (currentStep) {
49
+ const descVal = extractValue(trimmed, "description");
50
+ if (descVal !== null) {
51
+ currentStep.description = descVal;
52
+ continue;
53
+ }
54
+ }
55
+ if (currentStep && trimmed.startsWith("candidates:")) {
56
+ inCandidates = true;
57
+ continue;
58
+ }
59
+ if (inCandidates && currentStep && trimmed.startsWith("- "))
60
+ currentStep.candidateModels.push(trimmed
61
+ .slice(2)
62
+ .trim()
63
+ .replace(/"/g, ""));
64
+ }
65
+ if (currentStep)
66
+ steps.push(currentStep);
67
+ return { name, steps };
68
+ }
69
+ function computeTotalCombinations(pipeline) {
70
+ return pipeline.steps.reduce((total, step) => total * step.candidateModels.length, 1);
71
+ }
72
+ //# sourceMappingURL=pipeline-parser.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pipeline-parser.js","sourceRoot":"","sources":["../../src/optimizer/pipeline-parser.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ import type { Pipeline, RankedCombo, SearchConfig } from "./types.js";
2
+ export declare function bruteForceSearch(pipeline: Pipeline, costFn: (combo: Record<string, string>) => number, accuracyFn?: (combo: Record<string, string>) => number, config?: SearchConfig): RankedCombo[];
3
+ export declare function armEliminationSearch(pipeline: Pipeline, costFn: (combo: Record<string, string>) => number, accuracyFn?: (combo: Record<string, string>) => number, config?: SearchConfig): RankedCombo[];
4
+ export declare function epsilonLucbSearch(pipeline: Pipeline, costFn: (combo: Record<string, string>) => number, accuracyFn?: (combo: Record<string, string>) => number, config?: SearchConfig): RankedCombo[];
5
+ export declare function hillClimbingSearch(pipeline: Pipeline, costFn: (combo: Record<string, string>) => number, accuracyFn?: (combo: Record<string, string>) => number, config?: SearchConfig): RankedCombo[];
6
+ export declare function bayesianSearch(pipeline: Pipeline, costFn: (combo: Record<string, string>) => number, accuracyFn?: (combo: Record<string, string>) => number, config?: SearchConfig): RankedCombo[];
7
+ //# sourceMappingURL=search.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search.d.ts","sourceRoot":"","sources":["../../src/optimizer/search.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAItE,wBAAgB,gBAAgB,CAC9B,QAAQ,EAAE,QAAQ,EAClB,MAAM,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACjD,UAAU,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACtD,MAAM,GAAE,YAAoC,GAC3C,WAAW,EAAE,CAef;AAED,wBAAgB,oBAAoB,CAClC,QAAQ,EAAE,QAAQ,EAClB,MAAM,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACjD,UAAU,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACtD,MAAM,GAAE,YAAoC,GAC3C,WAAW,EAAE,CAwDf;AAED,wBAAgB,iBAAiB,CAC/B,QAAQ,EAAE,QAAQ,EAClB,MAAM,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACjD,UAAU,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACtD,MAAM,GAAE,YAAoC,GAC3C,WAAW,EAAE,CAmEf;AAED,wBAAgB,kBAAkB,CAChC,QAAQ,EAAE,QAAQ,EAClB,MAAM,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACjD,UAAU,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACtD,MAAM,GAAE,YAAoC,GAC3C,WAAW,EAAE,CA6Ef;AAED,wBAAgB,cAAc,CAC5B,QAAQ,EAAE,QAAQ,EAClB,MAAM,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACjD,UAAU,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,KAAK,MAAM,EACtD,MAAM,GAAE,YAAoC,GAC3C,WAAW,EAAE,CA6Ef"}