@tangle-network/agent-runtime 0.44.0 → 0.46.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/README.md +95 -203
- package/dist/agent.d.ts +3 -2
- package/dist/agent.js +5 -7
- package/dist/agent.js.map +1 -1
- package/dist/analyst-loop.d.ts +28 -2
- package/dist/analyst-loop.js +4 -1
- package/dist/audit.d.ts +93 -0
- package/dist/audit.js +312 -0
- package/dist/audit.js.map +1 -0
- package/dist/chunk-4B6U4CVQ.js +15 -0
- package/dist/chunk-4B6U4CVQ.js.map +1 -0
- package/dist/chunk-65FQLI4V.js +4089 -0
- package/dist/chunk-65FQLI4V.js.map +1 -0
- package/dist/{chunk-GFKVVRQ7.js → chunk-GN75RGM6.js} +13 -12
- package/dist/chunk-GN75RGM6.js.map +1 -0
- package/dist/chunk-GSUO5QS6.js +146 -0
- package/dist/chunk-GSUO5QS6.js.map +1 -0
- package/dist/chunk-HNUXAZIJ.js +580 -0
- package/dist/chunk-HNUXAZIJ.js.map +1 -0
- package/dist/{chunk-SKUZZCHE.js → chunk-I42NHLKX.js} +5 -5
- package/dist/chunk-I42NHLKX.js.map +1 -0
- package/dist/{chunk-HVYOHJHK.js → chunk-JNPK46YH.js} +2 -2
- package/dist/chunk-JNPK46YH.js.map +1 -0
- package/dist/{chunk-3HMHSN22.js → chunk-KADIJAD4.js} +38 -24
- package/dist/chunk-KADIJAD4.js.map +1 -0
- package/dist/{chunk-KDMRUD2P.js → chunk-KPN7OQ64.js} +296 -8
- package/dist/chunk-KPN7OQ64.js.map +1 -0
- package/dist/{chunk-NRZOXCJK.js → chunk-VR4JIC5H.js} +2 -2
- package/dist/chunk-WIR4HOOJ.js +27 -0
- package/dist/chunk-WIR4HOOJ.js.map +1 -0
- package/dist/coder-DCWFQpmJ.d.ts +114 -0
- package/dist/driver-C-mtBo7h.d.ts +221 -0
- package/dist/improvement.d.ts +0 -1
- package/dist/improvement.js +0 -5
- package/dist/improvement.js.map +1 -1
- package/dist/index.d.ts +122 -9
- package/dist/index.js +398 -10
- package/dist/index.js.map +1 -1
- package/dist/{kb-gate-D0ZIhFOU.d.ts → kb-gate-2Gwpz_27.d.ts} +86 -9
- package/dist/{loop-runner-bin-BLMa8He3.d.ts → loop-runner-bin-D-K6bRp3.d.ts} +17 -13
- package/dist/loop-runner-bin.d.ts +8 -6
- package/dist/loop-runner-bin.js +6 -8
- package/dist/loops.d.ts +7 -393
- package/dist/loops.js +96 -27
- package/dist/mcp/bin.js +7 -7
- package/dist/mcp/bin.js.map +1 -1
- package/dist/mcp/index.d.ts +286 -13
- package/dist/mcp/index.js +341 -9
- package/dist/mcp/index.js.map +1 -1
- package/dist/{otel-export-wFDmmurL.d.ts → otel-export-nurzFwuJ.d.ts} +1 -1
- package/dist/profiles.d.ts +385 -86
- package/dist/profiles.js +549 -4
- package/dist/profiles.js.map +1 -1
- package/dist/{run-loop-C4L1Sted.d.ts → run-loop-CU2Y00Si.d.ts} +36 -13
- package/dist/runtime-hooks-C7JwKb9E.d.ts +70 -0
- package/dist/runtime.d.ts +1964 -0
- package/dist/runtime.js +114 -0
- package/dist/runtime.js.map +1 -0
- package/dist/substrate-CUgk7F7s.d.ts +77 -0
- package/dist/topology.d.ts +73 -0
- package/dist/topology.js +111 -0
- package/dist/topology.js.map +1 -0
- package/dist/types-BfoeiQRZ.d.ts +438 -0
- package/dist/{types-DbJzz2uf.d.ts → types-DnYoHvvZ.d.ts} +110 -4
- package/dist/workflow.d.ts +4 -3
- package/dist/workflow.js +4 -5
- package/dist/workflow.js.map +1 -1
- package/package.json +37 -28
- package/skills/agent-runtime-adoption/SKILL.md +32 -29
- package/skills/generate-eval/SKILL.md +60 -0
- package/dist/chunk-3HMHSN22.js.map +0 -1
- package/dist/chunk-GFKVVRQ7.js.map +0 -1
- package/dist/chunk-HVYOHJHK.js.map +0 -1
- package/dist/chunk-KDMRUD2P.js.map +0 -1
- package/dist/chunk-PY6NMZYX.js +0 -52
- package/dist/chunk-PY6NMZYX.js.map +0 -1
- package/dist/chunk-S7JXV32P.js +0 -947
- package/dist/chunk-S7JXV32P.js.map +0 -1
- package/dist/chunk-SKUZZCHE.js.map +0 -1
- package/dist/chunk-SQSCRJ7U.js +0 -65
- package/dist/chunk-SQSCRJ7U.js.map +0 -1
- package/dist/chunk-VOX6Z3II.js +0 -90
- package/dist/chunk-VOX6Z3II.js.map +0 -1
- package/dist/chunk-XBUG326M.js +0 -261
- package/dist/chunk-XBUG326M.js.map +0 -1
- package/dist/dynamic-wUgp6UKs.d.ts +0 -108
- package/dist/optimize-prompt-D-urF2wW.d.ts +0 -129
- /package/dist/{chunk-NRZOXCJK.js.map → chunk-VR4JIC5H.js.map} +0 -0
|
@@ -0,0 +1,580 @@
|
|
|
1
|
+
import {
|
|
2
|
+
AnalystError,
|
|
3
|
+
extractLlmCallEvent
|
|
4
|
+
} from "./chunk-GSUO5QS6.js";
|
|
5
|
+
|
|
6
|
+
// src/analyst-loop/iterations-to-trace-store.ts
|
|
7
|
+
import {
|
|
8
|
+
DEFAULT_TRACE_ANALYST_BUDGETS,
|
|
9
|
+
TRACE_ANALYST_TRUNCATION_MARKER_PREFIX
|
|
10
|
+
} from "@tangle-network/agent-eval";
|
|
11
|
+
var bytesOf = (v) => Buffer.byteLength(JSON.stringify(v) ?? "", "utf8");
|
|
12
|
+
var iso = (ms) => new Date(ms).toISOString();
|
|
13
|
+
function normalizeSignature(message) {
|
|
14
|
+
return message.replace(/0x[0-9a-fA-F]+/g, "HEX").replace(/[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}/g, "UUID").replace(/(\/[\w.-]+){2,}/g, "PATH").replace(/\b\d+(\.\d+)?(ms|s|m|h)\b/g, "DUR").replace(/\b\d+\b/g, "#").replace(/\s+/g, " ").trim().slice(0, 200);
|
|
15
|
+
}
|
|
16
|
+
function spanKindFor(event, agentRunName) {
|
|
17
|
+
const llm = extractLlmCallEvent(event, agentRunName);
|
|
18
|
+
if (llm) return { kind: "LLM", model: llm.model ?? null, tool: null };
|
|
19
|
+
const type = String(event?.type ?? "");
|
|
20
|
+
if (/tool/i.test(type)) {
|
|
21
|
+
const d = event?.data;
|
|
22
|
+
const tool = typeof d?.name === "string" ? d.name : typeof d?.tool === "string" ? d.tool : type;
|
|
23
|
+
return { kind: "TOOL", model: null, tool };
|
|
24
|
+
}
|
|
25
|
+
return { kind: "SPAN", model: null, tool: null };
|
|
26
|
+
}
|
|
27
|
+
function errorMessageOf(event) {
|
|
28
|
+
const type = String(event?.type ?? "");
|
|
29
|
+
const d = event?.data;
|
|
30
|
+
if (/error|fail/i.test(type) || d?.error) {
|
|
31
|
+
const m = d?.error ?? d?.message;
|
|
32
|
+
return typeof m === "string" ? m : `${type} error`;
|
|
33
|
+
}
|
|
34
|
+
return void 0;
|
|
35
|
+
}
|
|
36
|
+
function projectIteration(iter) {
|
|
37
|
+
const traceId = iter.events.find((e) => e?.data?.sandboxId)?.data?.sandboxId ?? `iter-${iter.index}`;
|
|
38
|
+
const start = iso(iter.startedAt);
|
|
39
|
+
const end = iso(iter.endedAt || iter.startedAt);
|
|
40
|
+
const durationMs = Math.max(0, (iter.endedAt || iter.startedAt) - iter.startedAt);
|
|
41
|
+
const rootId = `${traceId}:root`;
|
|
42
|
+
const iterErrored = Boolean(iter.error) || iter.verdict?.valid === false;
|
|
43
|
+
const spans = [
|
|
44
|
+
{
|
|
45
|
+
trace_id: traceId,
|
|
46
|
+
span_id: rootId,
|
|
47
|
+
parent_span_id: null,
|
|
48
|
+
name: iter.agentRunName,
|
|
49
|
+
kind: "AGENT",
|
|
50
|
+
start_time: start,
|
|
51
|
+
end_time: end,
|
|
52
|
+
duration_ms: durationMs,
|
|
53
|
+
status: iter.error ? "ERROR" : "OK",
|
|
54
|
+
status_message: iter.error?.message,
|
|
55
|
+
service_name: "agent-runtime",
|
|
56
|
+
agent_name: iter.agentRunName,
|
|
57
|
+
model_name: null,
|
|
58
|
+
tool_name: null,
|
|
59
|
+
attributes: {
|
|
60
|
+
"iteration.index": iter.index,
|
|
61
|
+
"verdict.valid": iter.verdict?.valid,
|
|
62
|
+
"verdict.score": iter.verdict?.score,
|
|
63
|
+
"output.preview": iter.output === void 0 ? void 0 : String(iter.output).slice(0, 2e3)
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
];
|
|
67
|
+
const models = /* @__PURE__ */ new Set();
|
|
68
|
+
const tools = /* @__PURE__ */ new Set();
|
|
69
|
+
iter.events.forEach((event, i) => {
|
|
70
|
+
const { kind, model, tool } = spanKindFor(event, iter.agentRunName);
|
|
71
|
+
if (model) models.add(model);
|
|
72
|
+
if (tool) tools.add(tool);
|
|
73
|
+
const errMsg = errorMessageOf(event);
|
|
74
|
+
spans.push({
|
|
75
|
+
trace_id: traceId,
|
|
76
|
+
span_id: `${traceId}:e${i}`,
|
|
77
|
+
parent_span_id: rootId,
|
|
78
|
+
name: String(event?.type ?? "event"),
|
|
79
|
+
kind,
|
|
80
|
+
start_time: start,
|
|
81
|
+
end_time: end,
|
|
82
|
+
duration_ms: 0,
|
|
83
|
+
status: errMsg ? "ERROR" : "OK",
|
|
84
|
+
status_message: errMsg,
|
|
85
|
+
service_name: "agent-runtime",
|
|
86
|
+
agent_name: iter.agentRunName,
|
|
87
|
+
model_name: model,
|
|
88
|
+
tool_name: tool,
|
|
89
|
+
attributes: event?.data ?? {}
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
const hasErrors = spans.some((s) => s.status === "ERROR") || iterErrored;
|
|
93
|
+
const summary = {
|
|
94
|
+
trace_id: traceId,
|
|
95
|
+
service_name: "agent-runtime",
|
|
96
|
+
agent_name: iter.agentRunName,
|
|
97
|
+
span_count: spans.length,
|
|
98
|
+
has_errors: hasErrors,
|
|
99
|
+
start_time: start,
|
|
100
|
+
end_time: end,
|
|
101
|
+
duration_ms: durationMs,
|
|
102
|
+
raw_jsonl_bytes: bytesOf(spans),
|
|
103
|
+
models: [...models],
|
|
104
|
+
tools: [...tools]
|
|
105
|
+
};
|
|
106
|
+
return { summary, spans, rawBytes: summary.raw_jsonl_bytes };
|
|
107
|
+
}
|
|
108
|
+
function matchesFilters(t, f) {
|
|
109
|
+
if (!f) return true;
|
|
110
|
+
if (f.has_errors !== void 0 && t.summary.has_errors !== f.has_errors) return false;
|
|
111
|
+
if (f.service_names?.length && !f.service_names.includes(t.summary.service_name ?? ""))
|
|
112
|
+
return false;
|
|
113
|
+
if (f.agent_names?.length && !f.agent_names.includes(t.summary.agent_name ?? "")) return false;
|
|
114
|
+
if (f.model_names?.length && !f.model_names.some((m) => t.summary.models.includes(m)))
|
|
115
|
+
return false;
|
|
116
|
+
if (f.tool_names?.length && !f.tool_names.some((tn) => t.summary.tools.includes(tn))) return false;
|
|
117
|
+
if (f.start_time_after && t.summary.start_time < f.start_time_after) return false;
|
|
118
|
+
if (f.start_time_before && t.summary.start_time > f.start_time_before) return false;
|
|
119
|
+
if (f.regex_pattern && !new RegExp(f.regex_pattern).test(JSON.stringify(t.spans))) return false;
|
|
120
|
+
return true;
|
|
121
|
+
}
|
|
122
|
+
function capAttributes(attributes, perAttrCap) {
|
|
123
|
+
let truncated = 0;
|
|
124
|
+
const capped = {};
|
|
125
|
+
for (const [k, v] of Object.entries(attributes)) {
|
|
126
|
+
const s = typeof v === "string" ? v : JSON.stringify(v);
|
|
127
|
+
if (typeof s === "string" && s.length > perAttrCap) {
|
|
128
|
+
truncated += 1;
|
|
129
|
+
capped[k] = `${TRACE_ANALYST_TRUNCATION_MARKER_PREFIX} ${s.length}b]${s.slice(0, perAttrCap)}`;
|
|
130
|
+
} else {
|
|
131
|
+
capped[k] = v;
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
return { capped, truncated };
|
|
135
|
+
}
|
|
136
|
+
function iterationsToTraceStore(iterations, budgets = DEFAULT_TRACE_ANALYST_BUDGETS) {
|
|
137
|
+
if (iterations.length === 0) {
|
|
138
|
+
throw new AnalystError("iterationsToTraceStore: no iterations to analyze (empty round)");
|
|
139
|
+
}
|
|
140
|
+
const traces = iterations.map((it) => projectIteration(it));
|
|
141
|
+
const byId = new Map(traces.map((t) => [t.summary.trace_id, t]));
|
|
142
|
+
const buildClusters = (set) => {
|
|
143
|
+
const map = /* @__PURE__ */ new Map();
|
|
144
|
+
for (const t of set) {
|
|
145
|
+
for (const s of t.spans) {
|
|
146
|
+
if (s.status !== "ERROR" || !s.status_message) continue;
|
|
147
|
+
const sig = normalizeSignature(s.status_message);
|
|
148
|
+
const c = map.get(sig) ?? {
|
|
149
|
+
signature: sig,
|
|
150
|
+
status_message_sample: s.status_message,
|
|
151
|
+
span_name: s.name,
|
|
152
|
+
tool_name: s.tool_name,
|
|
153
|
+
trace_count: 0,
|
|
154
|
+
span_count: 0,
|
|
155
|
+
prevalence: 0,
|
|
156
|
+
exemplar_trace_ids: [],
|
|
157
|
+
exemplar_span_ids: []
|
|
158
|
+
};
|
|
159
|
+
c.span_count += 1;
|
|
160
|
+
if (!c.exemplar_trace_ids.includes(t.summary.trace_id) && c.exemplar_trace_ids.length < 10) {
|
|
161
|
+
c.exemplar_trace_ids.push(t.summary.trace_id);
|
|
162
|
+
c.trace_count += 1;
|
|
163
|
+
}
|
|
164
|
+
if (c.exemplar_span_ids.length < 10) c.exemplar_span_ids.push(s.span_id);
|
|
165
|
+
map.set(sig, c);
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
const errorTraces = set.filter((t) => t.summary.has_errors).length || 1;
|
|
169
|
+
const clusters = [...map.values()].map((c) => ({
|
|
170
|
+
...c,
|
|
171
|
+
prevalence: c.trace_count / errorTraces
|
|
172
|
+
}));
|
|
173
|
+
return clusters.sort((a, b) => b.trace_count - a.trace_count);
|
|
174
|
+
};
|
|
175
|
+
return {
|
|
176
|
+
async getOverview(filters) {
|
|
177
|
+
const set = traces.filter((t) => matchesFilters(t, filters));
|
|
178
|
+
const services = /* @__PURE__ */ new Set();
|
|
179
|
+
const agents = /* @__PURE__ */ new Set();
|
|
180
|
+
const models = /* @__PURE__ */ new Set();
|
|
181
|
+
const tools = /* @__PURE__ */ new Set();
|
|
182
|
+
let errorSpans = 0;
|
|
183
|
+
for (const t of set) {
|
|
184
|
+
if (t.summary.service_name) services.add(t.summary.service_name);
|
|
185
|
+
if (t.summary.agent_name) agents.add(t.summary.agent_name);
|
|
186
|
+
for (const m of t.summary.models) models.add(m);
|
|
187
|
+
for (const tn of t.summary.tools) tools.add(tn);
|
|
188
|
+
errorSpans += t.spans.filter((s) => s.status === "ERROR").length;
|
|
189
|
+
}
|
|
190
|
+
const times = set.map((t) => t.summary.start_time).sort();
|
|
191
|
+
return {
|
|
192
|
+
total_traces: set.length,
|
|
193
|
+
raw_jsonl_bytes: set.reduce((n, t) => n + t.rawBytes, 0),
|
|
194
|
+
services: [...services],
|
|
195
|
+
agents: [...agents],
|
|
196
|
+
models: [...models],
|
|
197
|
+
tool_names: [...tools],
|
|
198
|
+
sample_trace_ids: set.slice(0, 20).map((t) => t.summary.trace_id),
|
|
199
|
+
errors: {
|
|
200
|
+
trace_count: set.filter((t) => t.summary.has_errors).length,
|
|
201
|
+
span_count: errorSpans
|
|
202
|
+
},
|
|
203
|
+
error_clusters: buildClusters(set),
|
|
204
|
+
time_range: times.length ? { earliest: times[0], latest: times[times.length - 1] } : null
|
|
205
|
+
};
|
|
206
|
+
},
|
|
207
|
+
async queryTraces(opts) {
|
|
208
|
+
const set = traces.filter((t) => matchesFilters(t, opts.filters));
|
|
209
|
+
const offset = opts.offset ?? 0;
|
|
210
|
+
const page = set.slice(offset, offset + opts.limit);
|
|
211
|
+
return {
|
|
212
|
+
traces: page.map((t) => t.summary),
|
|
213
|
+
total: set.length,
|
|
214
|
+
has_more: offset + opts.limit < set.length
|
|
215
|
+
};
|
|
216
|
+
},
|
|
217
|
+
async countTraces(filters) {
|
|
218
|
+
return traces.filter((t) => matchesFilters(t, filters)).length;
|
|
219
|
+
},
|
|
220
|
+
async viewTrace(opts) {
|
|
221
|
+
const t = byId.get(opts.trace_id);
|
|
222
|
+
if (!t) return { trace_id: opts.trace_id, spans: [] };
|
|
223
|
+
const cap = opts.per_attribute_byte_cap ?? budgets.perAttributeViewBudget;
|
|
224
|
+
const projected = t.spans.map((s) => ({
|
|
225
|
+
...s,
|
|
226
|
+
attributes: capAttributes(s.attributes, cap).capped
|
|
227
|
+
}));
|
|
228
|
+
if (bytesOf(projected) > budgets.perCallByteCeiling) {
|
|
229
|
+
const names = /* @__PURE__ */ new Map();
|
|
230
|
+
for (const s of t.spans) names.set(s.name, (names.get(s.name) ?? 0) + 1);
|
|
231
|
+
return {
|
|
232
|
+
trace_id: opts.trace_id,
|
|
233
|
+
oversized: {
|
|
234
|
+
span_count: t.spans.length,
|
|
235
|
+
top_span_names: [...names.entries()].sort((a, b) => b[1] - a[1]).slice(0, 20),
|
|
236
|
+
span_response_bytes_max: Math.max(...t.spans.map((s) => bytesOf(s))),
|
|
237
|
+
error_span_count: t.spans.filter((s) => s.status === "ERROR").length
|
|
238
|
+
}
|
|
239
|
+
};
|
|
240
|
+
}
|
|
241
|
+
return { trace_id: opts.trace_id, spans: projected };
|
|
242
|
+
},
|
|
243
|
+
async viewSpans(opts) {
|
|
244
|
+
const t = byId.get(opts.trace_id);
|
|
245
|
+
const cap = opts.per_attribute_byte_cap ?? budgets.perAttributeSpanBudget;
|
|
246
|
+
const want = new Set(opts.span_ids);
|
|
247
|
+
const found = (t?.spans ?? []).filter((s) => want.has(s.span_id));
|
|
248
|
+
let truncated = 0;
|
|
249
|
+
const spans = found.map((s) => {
|
|
250
|
+
const { capped, truncated: n } = capAttributes(s.attributes, cap);
|
|
251
|
+
truncated += n;
|
|
252
|
+
return { ...s, attributes: capped };
|
|
253
|
+
});
|
|
254
|
+
const foundIds = new Set(found.map((s) => s.span_id));
|
|
255
|
+
return {
|
|
256
|
+
trace_id: opts.trace_id,
|
|
257
|
+
spans,
|
|
258
|
+
missing_span_ids: opts.span_ids.filter((id) => !foundIds.has(id)),
|
|
259
|
+
truncated_attribute_count: truncated
|
|
260
|
+
};
|
|
261
|
+
},
|
|
262
|
+
async searchTrace(opts) {
|
|
263
|
+
const t = byId.get(opts.trace_id);
|
|
264
|
+
const max = opts.max_matches ?? 50;
|
|
265
|
+
const hits = [];
|
|
266
|
+
for (const s of t?.spans ?? []) {
|
|
267
|
+
for (const hit of searchSpanAttrs(s, opts.regex_pattern, budgets.perMatchTextBudget)) {
|
|
268
|
+
if (hits.length >= max) break;
|
|
269
|
+
hits.push(hit);
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
return {
|
|
273
|
+
trace_id: opts.trace_id,
|
|
274
|
+
hits,
|
|
275
|
+
total_matches: hits.length,
|
|
276
|
+
has_more: hits.length >= max
|
|
277
|
+
};
|
|
278
|
+
},
|
|
279
|
+
async searchSpan(opts) {
|
|
280
|
+
const t = byId.get(opts.trace_id);
|
|
281
|
+
const max = opts.max_matches ?? 50;
|
|
282
|
+
const span = (t?.spans ?? []).find((s) => s.span_id === opts.span_id);
|
|
283
|
+
const hits = span ? searchSpanAttrs(span, opts.regex_pattern, budgets.perMatchTextBudget).slice(0, max) : [];
|
|
284
|
+
return {
|
|
285
|
+
trace_id: opts.trace_id,
|
|
286
|
+
span_id: opts.span_id,
|
|
287
|
+
hits,
|
|
288
|
+
total_matches: hits.length,
|
|
289
|
+
has_more: false
|
|
290
|
+
};
|
|
291
|
+
}
|
|
292
|
+
};
|
|
293
|
+
}
|
|
294
|
+
function searchSpanAttrs(span, pattern, textCap) {
|
|
295
|
+
const re = new RegExp(pattern, "g");
|
|
296
|
+
const hits = [];
|
|
297
|
+
for (const [k, v] of Object.entries(span.attributes)) {
|
|
298
|
+
const text = typeof v === "string" ? v : JSON.stringify(v);
|
|
299
|
+
if (typeof text !== "string") continue;
|
|
300
|
+
re.lastIndex = 0;
|
|
301
|
+
const m = re.exec(text);
|
|
302
|
+
if (!m) continue;
|
|
303
|
+
const at = m.index;
|
|
304
|
+
hits.push({
|
|
305
|
+
trace_id: span.trace_id,
|
|
306
|
+
span_id: span.span_id,
|
|
307
|
+
span_name: span.name,
|
|
308
|
+
span_kind: span.kind,
|
|
309
|
+
attribute_path: `attributes.${k}`,
|
|
310
|
+
matched_text: m[0].slice(0, textCap),
|
|
311
|
+
context_before: text.slice(Math.max(0, at - textCap / 2), at),
|
|
312
|
+
context_after: text.slice(at + m[0].length, at + m[0].length + textCap / 2),
|
|
313
|
+
match_offset: at
|
|
314
|
+
});
|
|
315
|
+
}
|
|
316
|
+
return hits;
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
// src/analyst-loop/run-analyst-loop.ts
|
|
320
|
+
import { diffFindings } from "@tangle-network/agent-eval";
|
|
321
|
+
async function runAnalystLoop(opts) {
|
|
322
|
+
const log = opts.log ?? defaultLog;
|
|
323
|
+
const strategy = opts.priorFindingsStrategy ?? "per-kind";
|
|
324
|
+
const emit = makeEmitter(opts.onEvent);
|
|
325
|
+
const startedAt = Date.now();
|
|
326
|
+
const baselineRunId = resolveBaselineRunId(opts);
|
|
327
|
+
const priorAll = baselineRunId ? opts.findingsStore?.loadRun(baselineRunId) ?? [] : [];
|
|
328
|
+
log("baseline resolved", { baselineRunId, prior_findings: priorAll.length });
|
|
329
|
+
await emit({
|
|
330
|
+
type: "baseline-resolved",
|
|
331
|
+
runId: opts.runId,
|
|
332
|
+
baselineRunId,
|
|
333
|
+
priorFindingCount: priorAll.length
|
|
334
|
+
});
|
|
335
|
+
const priorFindings = buildPriorFindingsInput(priorAll, strategy, opts.registry.list());
|
|
336
|
+
const analystResult = await runRegistry(opts, priorFindings, emit);
|
|
337
|
+
log("analyst run complete", {
|
|
338
|
+
findings: analystResult.findings.length,
|
|
339
|
+
cost_usd: analystResult.total_cost_usd,
|
|
340
|
+
per_analyst: analystResult.per_analyst.map((s) => ({
|
|
341
|
+
id: s.analyst_id,
|
|
342
|
+
status: s.status,
|
|
343
|
+
n: s.findings_count
|
|
344
|
+
}))
|
|
345
|
+
});
|
|
346
|
+
if (opts.findingsStore && analystResult.findings.length > 0) {
|
|
347
|
+
await opts.findingsStore.append(opts.runId, analystResult.findings);
|
|
348
|
+
await emit({
|
|
349
|
+
type: "findings-persisted",
|
|
350
|
+
runId: opts.runId,
|
|
351
|
+
count: analystResult.findings.length
|
|
352
|
+
});
|
|
353
|
+
}
|
|
354
|
+
let diff = null;
|
|
355
|
+
if (baselineRunId && analystResult.findings.length > 0) {
|
|
356
|
+
diff = diffFindings(
|
|
357
|
+
priorAll.map((f) => ({ ...f })),
|
|
358
|
+
analystResult.findings.map((f) => ({ ...f, run_id: opts.runId }))
|
|
359
|
+
);
|
|
360
|
+
log("diff vs baseline", {
|
|
361
|
+
appeared: diff.appeared.length,
|
|
362
|
+
disappeared: diff.disappeared.length,
|
|
363
|
+
persisted: diff.persisted.length,
|
|
364
|
+
changed: diff.changed.length
|
|
365
|
+
});
|
|
366
|
+
await emit({
|
|
367
|
+
type: "diff-computed",
|
|
368
|
+
runId: opts.runId,
|
|
369
|
+
baselineRunId,
|
|
370
|
+
appeared: diff.appeared.length,
|
|
371
|
+
disappeared: diff.disappeared.length,
|
|
372
|
+
persisted: diff.persisted.length,
|
|
373
|
+
changed: diff.changed.length
|
|
374
|
+
});
|
|
375
|
+
}
|
|
376
|
+
let knowledge = null;
|
|
377
|
+
if (opts.knowledgeAdapter) {
|
|
378
|
+
knowledge = await runKnowledgeAdapter(opts, analystResult.findings, log, emit);
|
|
379
|
+
}
|
|
380
|
+
let improvement = null;
|
|
381
|
+
if (opts.improvementAdapter) {
|
|
382
|
+
improvement = await runImprovementAdapter(opts, analystResult.findings, log, emit);
|
|
383
|
+
}
|
|
384
|
+
await emit({
|
|
385
|
+
type: "loop-completed",
|
|
386
|
+
runId: opts.runId,
|
|
387
|
+
durationMs: Date.now() - startedAt
|
|
388
|
+
});
|
|
389
|
+
return {
|
|
390
|
+
runId: opts.runId,
|
|
391
|
+
baselineRunId,
|
|
392
|
+
analystResult,
|
|
393
|
+
diff,
|
|
394
|
+
knowledge,
|
|
395
|
+
improvement
|
|
396
|
+
};
|
|
397
|
+
}
|
|
398
|
+
function makeEmitter(onEvent) {
|
|
399
|
+
if (!onEvent) return async () => {
|
|
400
|
+
};
|
|
401
|
+
return async (event) => {
|
|
402
|
+
await onEvent(event);
|
|
403
|
+
};
|
|
404
|
+
}
|
|
405
|
+
async function runRegistry(opts, priorFindings, emit) {
|
|
406
|
+
const reg = opts.registry;
|
|
407
|
+
if (typeof reg.runStream === "function" && opts.onEvent) {
|
|
408
|
+
let final = null;
|
|
409
|
+
for await (const ev of reg.runStream(opts.runId, opts.inputs, { priorFindings })) {
|
|
410
|
+
await emit({ type: "analyst", runId: opts.runId, event: ev });
|
|
411
|
+
if (ev.type === "run-completed") final = ev.result;
|
|
412
|
+
}
|
|
413
|
+
if (!final) {
|
|
414
|
+
throw new Error("runAnalystLoop: registry.runStream ended without run-completed event");
|
|
415
|
+
}
|
|
416
|
+
return final;
|
|
417
|
+
}
|
|
418
|
+
return opts.registry.run(opts.runId, opts.inputs, { priorFindings });
|
|
419
|
+
}
|
|
420
|
+
function resolveBaselineRunId(opts) {
|
|
421
|
+
if (opts.baselineRunId === null) return null;
|
|
422
|
+
if (typeof opts.baselineRunId === "string") return opts.baselineRunId;
|
|
423
|
+
if (!opts.findingsStore) return null;
|
|
424
|
+
const all = opts.findingsStore.loadAll();
|
|
425
|
+
let last = null;
|
|
426
|
+
for (const row of all) {
|
|
427
|
+
if (row.run_id === opts.runId) continue;
|
|
428
|
+
last = row.run_id;
|
|
429
|
+
}
|
|
430
|
+
return last;
|
|
431
|
+
}
|
|
432
|
+
function buildPriorFindingsInput(prior, strategy, registry) {
|
|
433
|
+
if (strategy === "none" || prior.length === 0) return void 0;
|
|
434
|
+
const stripped = prior.map(({ run_id: _run_id, ...rest }) => rest);
|
|
435
|
+
if (strategy === "wildcard") {
|
|
436
|
+
return { "*": stripped };
|
|
437
|
+
}
|
|
438
|
+
void registry;
|
|
439
|
+
return stripped;
|
|
440
|
+
}
|
|
441
|
+
async function runKnowledgeAdapter(opts, findings, log, emit) {
|
|
442
|
+
const adapter = opts.knowledgeAdapter;
|
|
443
|
+
const batch = await adapter.proposeFromFindings(findings);
|
|
444
|
+
log("knowledge.proposeFromFindings", {
|
|
445
|
+
proposals: batch.proposals.length,
|
|
446
|
+
skipped: batch.skipped,
|
|
447
|
+
errors: batch.errors.length
|
|
448
|
+
});
|
|
449
|
+
await emit({
|
|
450
|
+
type: "knowledge-proposed",
|
|
451
|
+
runId: opts.runId,
|
|
452
|
+
proposalCount: batch.proposals.length,
|
|
453
|
+
skipped: batch.skipped,
|
|
454
|
+
errors: batch.errors.length
|
|
455
|
+
});
|
|
456
|
+
const auto = opts.autoApply?.knowledge ?? false;
|
|
457
|
+
const threshold = opts.autoApply?.knowledgeConfidenceThreshold ?? 0.85;
|
|
458
|
+
if (!auto || !adapter.apply) {
|
|
459
|
+
await emit({
|
|
460
|
+
type: "knowledge-applied",
|
|
461
|
+
runId: opts.runId,
|
|
462
|
+
writtenCount: 0,
|
|
463
|
+
withheldForReview: batch.proposals.length
|
|
464
|
+
});
|
|
465
|
+
return {
|
|
466
|
+
proposals: batch.proposals,
|
|
467
|
+
applied: [],
|
|
468
|
+
skipped: batch.skipped,
|
|
469
|
+
errors: batch.errors,
|
|
470
|
+
withheld_for_review: batch.proposals.length
|
|
471
|
+
};
|
|
472
|
+
}
|
|
473
|
+
const findingsById = new Map(findings.map((f) => [f.finding_id, f]));
|
|
474
|
+
const safe = [];
|
|
475
|
+
let withheld = 0;
|
|
476
|
+
for (const p of batch.proposals) {
|
|
477
|
+
const src = p.sourceFindingId ? findingsById.get(p.sourceFindingId) : void 0;
|
|
478
|
+
if (!src) {
|
|
479
|
+
withheld += 1;
|
|
480
|
+
continue;
|
|
481
|
+
}
|
|
482
|
+
if (src.confidence < threshold) {
|
|
483
|
+
withheld += 1;
|
|
484
|
+
continue;
|
|
485
|
+
}
|
|
486
|
+
safe.push(p);
|
|
487
|
+
}
|
|
488
|
+
const result = await adapter.apply(safe);
|
|
489
|
+
log("knowledge.apply", {
|
|
490
|
+
applied: result.written.length,
|
|
491
|
+
withheld_for_review: withheld,
|
|
492
|
+
warnings: result.warnings.length
|
|
493
|
+
});
|
|
494
|
+
await emit({
|
|
495
|
+
type: "knowledge-applied",
|
|
496
|
+
runId: opts.runId,
|
|
497
|
+
writtenCount: result.written.length,
|
|
498
|
+
withheldForReview: withheld
|
|
499
|
+
});
|
|
500
|
+
return {
|
|
501
|
+
proposals: batch.proposals,
|
|
502
|
+
applied: result.written,
|
|
503
|
+
skipped: batch.skipped,
|
|
504
|
+
errors: batch.errors,
|
|
505
|
+
withheld_for_review: withheld
|
|
506
|
+
};
|
|
507
|
+
}
|
|
508
|
+
async function runImprovementAdapter(opts, findings, log, emit) {
|
|
509
|
+
const adapter = opts.improvementAdapter;
|
|
510
|
+
const batch = await adapter.proposeFromFindings(findings);
|
|
511
|
+
log("improvement.proposeFromFindings", {
|
|
512
|
+
edits: batch.edits.length,
|
|
513
|
+
skipped: batch.skipped,
|
|
514
|
+
errors: batch.errors.length
|
|
515
|
+
});
|
|
516
|
+
await emit({
|
|
517
|
+
type: "improvement-proposed",
|
|
518
|
+
runId: opts.runId,
|
|
519
|
+
editCount: batch.edits.length,
|
|
520
|
+
skipped: batch.skipped,
|
|
521
|
+
errors: batch.errors.length
|
|
522
|
+
});
|
|
523
|
+
const auto = opts.autoApply?.improvement ?? false;
|
|
524
|
+
const threshold = opts.autoApply?.improvementConfidenceThreshold ?? 0.9;
|
|
525
|
+
if (!auto || !adapter.apply) {
|
|
526
|
+
await emit({
|
|
527
|
+
type: "improvement-applied",
|
|
528
|
+
runId: opts.runId,
|
|
529
|
+
appliedCount: 0,
|
|
530
|
+
withheldForReview: batch.edits.length
|
|
531
|
+
});
|
|
532
|
+
return {
|
|
533
|
+
edits: batch.edits,
|
|
534
|
+
applied: [],
|
|
535
|
+
skipped: batch.skipped,
|
|
536
|
+
errors: batch.errors,
|
|
537
|
+
withheld_for_review: batch.edits.length
|
|
538
|
+
};
|
|
539
|
+
}
|
|
540
|
+
const findingsById = new Map(findings.map((f) => [f.finding_id, f]));
|
|
541
|
+
const safe = [];
|
|
542
|
+
let withheld = 0;
|
|
543
|
+
for (const e of batch.edits) {
|
|
544
|
+
const src = e.sourceFindingId ? findingsById.get(e.sourceFindingId) : void 0;
|
|
545
|
+
if (!src || src.confidence < threshold) {
|
|
546
|
+
withheld += 1;
|
|
547
|
+
continue;
|
|
548
|
+
}
|
|
549
|
+
safe.push(e);
|
|
550
|
+
}
|
|
551
|
+
const result = await adapter.apply(safe);
|
|
552
|
+
log("improvement.apply", {
|
|
553
|
+
applied: result.applied.length,
|
|
554
|
+
withheld_for_review: withheld,
|
|
555
|
+
warnings: result.warnings.length
|
|
556
|
+
});
|
|
557
|
+
await emit({
|
|
558
|
+
type: "improvement-applied",
|
|
559
|
+
runId: opts.runId,
|
|
560
|
+
appliedCount: result.applied.length,
|
|
561
|
+
withheldForReview: withheld
|
|
562
|
+
});
|
|
563
|
+
return {
|
|
564
|
+
edits: batch.edits,
|
|
565
|
+
applied: result.applied,
|
|
566
|
+
skipped: batch.skipped,
|
|
567
|
+
errors: batch.errors,
|
|
568
|
+
withheld_for_review: withheld
|
|
569
|
+
};
|
|
570
|
+
}
|
|
571
|
+
function defaultLog(msg, fields) {
|
|
572
|
+
if (fields) console.log(`[analyst-loop] ${msg}`, fields);
|
|
573
|
+
else console.log(`[analyst-loop] ${msg}`);
|
|
574
|
+
}
|
|
575
|
+
|
|
576
|
+
export {
|
|
577
|
+
iterationsToTraceStore,
|
|
578
|
+
runAnalystLoop
|
|
579
|
+
};
|
|
580
|
+
//# sourceMappingURL=chunk-HNUXAZIJ.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/analyst-loop/iterations-to-trace-store.ts","../src/analyst-loop/run-analyst-loop.ts"],"sourcesContent":["/**\n * @experimental\n *\n * The read seam that closes the autonomous loop: project a round's `Iteration[]`\n * (each carrying its raw `SandboxEvent[]`) into an in-memory `TraceAnalysisStore`, the\n * read interface the trace analysts query. `runAnalystLoop` has had zero consumers\n * because nothing turned a loop's iterations into a store — this is that bridge.\n *\n * One iteration → one trace. The iteration is the root AGENT span; each `SandboxEvent`\n * becomes a child span (LLM for llm_call events, TOOL for tool events, else SPAN), so an\n * analyst can walk a shot's trace, cluster its errors, and emit findings the driver\n * steers on. Projection is best-effort over the FLAT SandboxEvent shape (no per-event\n * lineage yet — that's the richer-trace gap); it never fabricates — an errored iteration\n * surfaces a real ERROR span carrying the real message.\n */\n\nimport {\n type DatasetOverview,\n DEFAULT_TRACE_ANALYST_BUDGETS,\n type QueryTracesPage,\n type SearchSpanResult,\n type SearchTraceResult,\n type SpanMatchRecord,\n TRACE_ANALYST_TRUNCATION_MARKER_PREFIX,\n type TraceAnalysisStore,\n type TraceAnalystByteBudgets,\n type TraceAnalystFilters,\n type TraceAnalystSpan,\n type TraceAnalystSpanKind,\n type TraceAnalystTraceSummary,\n type ViewSpansResult,\n type ViewTraceResult,\n} from '@tangle-network/agent-eval'\nimport type { SandboxEvent } from '@tangle-network/sandbox'\nimport { AnalystError } from '../errors'\nimport { extractLlmCallEvent } from '../runtime/sandbox-events'\nimport type { Iteration } from '../runtime/types'\n\n/** ErrorCluster isn't re-exported from the agent-eval root; derive it from the overview. */\ntype ErrorCluster = DatasetOverview['error_clusters'][number]\n\ninterface ProjectedTrace {\n summary: TraceAnalystTraceSummary\n spans: TraceAnalystSpan[]\n /** raw JSONL bytes of this trace's spans — the byte-budget accounting unit. */\n rawBytes: number\n}\n\nconst bytesOf = (v: unknown): number => Buffer.byteLength(JSON.stringify(v) ?? '', 'utf8')\nconst iso = (ms: number): string => new Date(ms).toISOString()\n\n/** Normalize volatile tokens out of a status message so semantically identical failures\n * collapse to one signature (digits, hex/uuids, paths, durations → placeholders). */\nfunction normalizeSignature(message: string): string {\n return message\n .replace(/0x[0-9a-fA-F]+/g, 'HEX')\n .replace(/[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}/g, 'UUID')\n .replace(/(\\/[\\w.-]+){2,}/g, 'PATH')\n .replace(/\\b\\d+(\\.\\d+)?(ms|s|m|h)\\b/g, 'DUR')\n .replace(/\\b\\d+\\b/g, '#')\n .replace(/\\s+/g, ' ')\n .trim()\n .slice(0, 200)\n}\n\nfunction spanKindFor(\n event: SandboxEvent,\n agentRunName: string,\n): {\n kind: TraceAnalystSpanKind\n model: string | null\n tool: string | null\n} {\n const llm = extractLlmCallEvent(event, agentRunName)\n if (llm) return { kind: 'LLM', model: llm.model ?? null, tool: null }\n const type = String(event?.type ?? '')\n if (/tool/i.test(type)) {\n const d = event?.data as { name?: unknown; tool?: unknown } | undefined\n const tool = typeof d?.name === 'string' ? d.name : typeof d?.tool === 'string' ? d.tool : type\n return { kind: 'TOOL', model: null, tool }\n }\n return { kind: 'SPAN', model: null, tool: null }\n}\n\nfunction errorMessageOf(event: SandboxEvent): string | undefined {\n const type = String(event?.type ?? '')\n const d = event?.data as { error?: unknown; message?: unknown } | undefined\n if (/error|fail/i.test(type) || d?.error) {\n const m = d?.error ?? d?.message\n return typeof m === 'string' ? m : `${type} error`\n }\n return undefined\n}\n\n/** Project one iteration → one trace: root AGENT span + a child span per event. */\nfunction projectIteration<Task, Output>(iter: Iteration<Task, Output>): ProjectedTrace {\n const traceId =\n (\n iter.events.find((e) => (e?.data as { sandboxId?: string } | undefined)?.sandboxId)?.data as\n | { sandboxId?: string }\n | undefined\n )?.sandboxId ?? `iter-${iter.index}`\n const start = iso(iter.startedAt)\n const end = iso(iter.endedAt || iter.startedAt)\n const durationMs = Math.max(0, (iter.endedAt || iter.startedAt) - iter.startedAt)\n const rootId = `${traceId}:root`\n const iterErrored = Boolean(iter.error) || iter.verdict?.valid === false\n const spans: TraceAnalystSpan[] = [\n {\n trace_id: traceId,\n span_id: rootId,\n parent_span_id: null,\n name: iter.agentRunName,\n kind: 'AGENT',\n start_time: start,\n end_time: end,\n duration_ms: durationMs,\n status: iter.error ? 'ERROR' : 'OK',\n status_message: iter.error?.message,\n service_name: 'agent-runtime',\n agent_name: iter.agentRunName,\n model_name: null,\n tool_name: null,\n attributes: {\n 'iteration.index': iter.index,\n 'verdict.valid': iter.verdict?.valid,\n 'verdict.score': iter.verdict?.score,\n 'output.preview':\n iter.output === undefined ? undefined : String(iter.output).slice(0, 2000),\n },\n },\n ]\n const models = new Set<string>()\n const tools = new Set<string>()\n iter.events.forEach((event, i) => {\n const { kind, model, tool } = spanKindFor(event, iter.agentRunName)\n if (model) models.add(model)\n if (tool) tools.add(tool)\n const errMsg = errorMessageOf(event)\n spans.push({\n trace_id: traceId,\n span_id: `${traceId}:e${i}`,\n parent_span_id: rootId,\n name: String(event?.type ?? 'event'),\n kind,\n start_time: start,\n end_time: end,\n duration_ms: 0,\n status: errMsg ? 'ERROR' : 'OK',\n status_message: errMsg,\n service_name: 'agent-runtime',\n agent_name: iter.agentRunName,\n model_name: model,\n tool_name: tool,\n attributes: (event?.data as Record<string, unknown> | undefined) ?? {},\n })\n })\n const hasErrors = spans.some((s) => s.status === 'ERROR') || iterErrored\n const summary: TraceAnalystTraceSummary = {\n trace_id: traceId,\n service_name: 'agent-runtime',\n agent_name: iter.agentRunName,\n span_count: spans.length,\n has_errors: hasErrors,\n start_time: start,\n end_time: end,\n duration_ms: durationMs,\n raw_jsonl_bytes: bytesOf(spans),\n models: [...models],\n tools: [...tools],\n }\n return { summary, spans, rawBytes: summary.raw_jsonl_bytes }\n}\n\nfunction matchesFilters(t: ProjectedTrace, f?: TraceAnalystFilters): boolean {\n if (!f) return true\n if (f.has_errors !== undefined && t.summary.has_errors !== f.has_errors) return false\n if (f.service_names?.length && !f.service_names.includes(t.summary.service_name ?? ''))\n return false\n if (f.agent_names?.length && !f.agent_names.includes(t.summary.agent_name ?? '')) return false\n if (f.model_names?.length && !f.model_names.some((m) => t.summary.models.includes(m)))\n return false\n if (f.tool_names?.length && !f.tool_names.some((tn) => t.summary.tools.includes(tn))) return false\n if (f.start_time_after && t.summary.start_time < f.start_time_after) return false\n if (f.start_time_before && t.summary.start_time > f.start_time_before) return false\n if (f.regex_pattern && !new RegExp(f.regex_pattern).test(JSON.stringify(t.spans))) return false\n return true\n}\n\nfunction capAttributes(\n attributes: Record<string, unknown>,\n perAttrCap: number,\n): { capped: Record<string, unknown>; truncated: number } {\n let truncated = 0\n const capped: Record<string, unknown> = {}\n for (const [k, v] of Object.entries(attributes)) {\n const s = typeof v === 'string' ? v : JSON.stringify(v)\n if (typeof s === 'string' && s.length > perAttrCap) {\n truncated += 1\n capped[k] = `${TRACE_ANALYST_TRUNCATION_MARKER_PREFIX} ${s.length}b]${s.slice(0, perAttrCap)}`\n } else {\n capped[k] = v\n }\n }\n return { capped, truncated }\n}\n\n/**\n * Build an in-memory `TraceAnalysisStore` over a loop round's iterations. Fail-loud on an\n * empty round — there is nothing for an analyst to read, and a silent empty store would\n * mask a broken capture path.\n */\nexport function iterationsToTraceStore<Task, Output>(\n iterations: ReadonlyArray<Iteration<Task, Output>>,\n budgets: TraceAnalystByteBudgets = DEFAULT_TRACE_ANALYST_BUDGETS,\n): TraceAnalysisStore {\n if (iterations.length === 0) {\n throw new AnalystError('iterationsToTraceStore: no iterations to analyze (empty round)')\n }\n const traces = iterations.map((it) => projectIteration(it))\n const byId = new Map(traces.map((t) => [t.summary.trace_id, t]))\n\n const buildClusters = (set: ProjectedTrace[]): ErrorCluster[] => {\n const map = new Map<string, ErrorCluster>()\n for (const t of set) {\n for (const s of t.spans) {\n if (s.status !== 'ERROR' || !s.status_message) continue\n const sig = normalizeSignature(s.status_message)\n const c = map.get(sig) ?? {\n signature: sig,\n status_message_sample: s.status_message,\n span_name: s.name,\n tool_name: s.tool_name,\n trace_count: 0,\n span_count: 0,\n prevalence: 0,\n exemplar_trace_ids: [],\n exemplar_span_ids: [],\n }\n c.span_count += 1\n if (\n !c.exemplar_trace_ids.includes(t.summary.trace_id) &&\n c.exemplar_trace_ids.length < 10\n ) {\n c.exemplar_trace_ids.push(t.summary.trace_id)\n c.trace_count += 1\n }\n if (c.exemplar_span_ids.length < 10) c.exemplar_span_ids.push(s.span_id)\n map.set(sig, c)\n }\n }\n const errorTraces = set.filter((t) => t.summary.has_errors).length || 1\n const clusters = [...map.values()].map((c) => ({\n ...c,\n prevalence: c.trace_count / errorTraces,\n }))\n return clusters.sort((a, b) => b.trace_count - a.trace_count)\n }\n\n return {\n async getOverview(filters?: TraceAnalystFilters): Promise<DatasetOverview> {\n const set = traces.filter((t) => matchesFilters(t, filters))\n const services = new Set<string>()\n const agents = new Set<string>()\n const models = new Set<string>()\n const tools = new Set<string>()\n let errorSpans = 0\n for (const t of set) {\n if (t.summary.service_name) services.add(t.summary.service_name)\n if (t.summary.agent_name) agents.add(t.summary.agent_name)\n for (const m of t.summary.models) models.add(m)\n for (const tn of t.summary.tools) tools.add(tn)\n errorSpans += t.spans.filter((s) => s.status === 'ERROR').length\n }\n const times = set.map((t) => t.summary.start_time).sort()\n return {\n total_traces: set.length,\n raw_jsonl_bytes: set.reduce((n, t) => n + t.rawBytes, 0),\n services: [...services],\n agents: [...agents],\n models: [...models],\n tool_names: [...tools],\n sample_trace_ids: set.slice(0, 20).map((t) => t.summary.trace_id),\n errors: {\n trace_count: set.filter((t) => t.summary.has_errors).length,\n span_count: errorSpans,\n },\n error_clusters: buildClusters(set),\n time_range: times.length ? { earliest: times[0]!, latest: times[times.length - 1]! } : null,\n }\n },\n\n async queryTraces(opts): Promise<QueryTracesPage> {\n const set = traces.filter((t) => matchesFilters(t, opts.filters))\n const offset = opts.offset ?? 0\n const page = set.slice(offset, offset + opts.limit)\n return {\n traces: page.map((t) => t.summary),\n total: set.length,\n has_more: offset + opts.limit < set.length,\n }\n },\n\n async countTraces(filters?: TraceAnalystFilters): Promise<number> {\n return traces.filter((t) => matchesFilters(t, filters)).length\n },\n\n async viewTrace(opts): Promise<ViewTraceResult> {\n const t = byId.get(opts.trace_id)\n if (!t) return { trace_id: opts.trace_id, spans: [] }\n const cap = opts.per_attribute_byte_cap ?? budgets.perAttributeViewBudget\n const projected = t.spans.map((s) => ({\n ...s,\n attributes: capAttributes(s.attributes, cap).capped,\n }))\n if (bytesOf(projected) > budgets.perCallByteCeiling) {\n const names = new Map<string, number>()\n for (const s of t.spans) names.set(s.name, (names.get(s.name) ?? 0) + 1)\n return {\n trace_id: opts.trace_id,\n oversized: {\n span_count: t.spans.length,\n top_span_names: [...names.entries()].sort((a, b) => b[1] - a[1]).slice(0, 20),\n span_response_bytes_max: Math.max(...t.spans.map((s) => bytesOf(s))),\n error_span_count: t.spans.filter((s) => s.status === 'ERROR').length,\n },\n }\n }\n return { trace_id: opts.trace_id, spans: projected }\n },\n\n async viewSpans(opts): Promise<ViewSpansResult> {\n const t = byId.get(opts.trace_id)\n const cap = opts.per_attribute_byte_cap ?? budgets.perAttributeSpanBudget\n const want = new Set(opts.span_ids)\n const found = (t?.spans ?? []).filter((s) => want.has(s.span_id))\n let truncated = 0\n const spans = found.map((s) => {\n const { capped, truncated: n } = capAttributes(s.attributes, cap)\n truncated += n\n return { ...s, attributes: capped }\n })\n const foundIds = new Set(found.map((s) => s.span_id))\n return {\n trace_id: opts.trace_id,\n spans,\n missing_span_ids: opts.span_ids.filter((id) => !foundIds.has(id)),\n truncated_attribute_count: truncated,\n }\n },\n\n async searchTrace(opts): Promise<SearchTraceResult> {\n const t = byId.get(opts.trace_id)\n const max = opts.max_matches ?? 50\n const hits: SpanMatchRecord[] = []\n for (const s of t?.spans ?? []) {\n for (const hit of searchSpanAttrs(s, opts.regex_pattern, budgets.perMatchTextBudget)) {\n if (hits.length >= max) break\n hits.push(hit)\n }\n }\n return {\n trace_id: opts.trace_id,\n hits,\n total_matches: hits.length,\n has_more: hits.length >= max,\n }\n },\n\n async searchSpan(opts): Promise<SearchSpanResult> {\n const t = byId.get(opts.trace_id)\n const max = opts.max_matches ?? 50\n const span = (t?.spans ?? []).find((s) => s.span_id === opts.span_id)\n const hits = span\n ? searchSpanAttrs(span, opts.regex_pattern, budgets.perMatchTextBudget).slice(0, max)\n : []\n return {\n trace_id: opts.trace_id,\n span_id: opts.span_id,\n hits,\n total_matches: hits.length,\n has_more: false,\n }\n },\n }\n}\n\nfunction searchSpanAttrs(\n span: TraceAnalystSpan,\n pattern: string,\n textCap: number,\n): SpanMatchRecord[] {\n const re = new RegExp(pattern, 'g')\n const hits: SpanMatchRecord[] = []\n for (const [k, v] of Object.entries(span.attributes)) {\n const text = typeof v === 'string' ? v : JSON.stringify(v)\n if (typeof text !== 'string') continue\n re.lastIndex = 0\n const m = re.exec(text)\n if (!m) continue\n const at = m.index\n hits.push({\n trace_id: span.trace_id,\n span_id: span.span_id,\n span_name: span.name,\n span_kind: span.kind,\n attribute_path: `attributes.${k}`,\n matched_text: m[0].slice(0, textCap),\n context_before: text.slice(Math.max(0, at - textCap / 2), at),\n context_after: text.slice(at + m[0].length, at + m[0].length + textCap / 2),\n match_offset: at,\n })\n }\n return hits\n}\n","/**\n * `runAnalystLoop` — the one call agent apps reach for to close the\n * recursive-self-improvement loop.\n *\n * 1. Load baseline findings (last run, or the slice the caller specifies)\n * 2. Run the analyst registry with priorFindings injected\n * 3. Persist the new run's findings to the ledger\n * 4. Diff the new run against the baseline\n * 5. Hand the findings to the knowledge adapter → proposals (and\n * optionally apply them) → wiki edits\n * 6. Hand the findings to the improvement adapter → prompt / tool /\n * scaffolding edits (review-only by default)\n * 7. Return a single report the consumer renders / persists / acts on.\n *\n * Adapters are optional: the loop works as a \"run + diff + report\"\n * primitive when no adapters are wired; it closes end-to-end when\n * both adapters are wired.\n */\n\nimport type { AnalystFinding, AnalystRunResult, FindingsDiff } from '@tangle-network/agent-eval'\nimport { diffFindings } from '@tangle-network/agent-eval'\n\nimport type {\n AnalystLoopEvent,\n AnalystRegistryStreamingLike,\n ImprovementReport,\n KnowledgeReport,\n RunAnalystLoopOpts,\n RunAnalystLoopResult,\n} from './types'\n\nexport async function runAnalystLoop<TProposal = unknown, TEdit = unknown>(\n opts: RunAnalystLoopOpts,\n): Promise<RunAnalystLoopResult<TProposal, TEdit>> {\n const log = opts.log ?? defaultLog\n const strategy = opts.priorFindingsStrategy ?? 'per-kind'\n const emit = makeEmitter(opts.onEvent)\n const startedAt = Date.now()\n\n // 1. Resolve baseline + load prior findings.\n const baselineRunId = resolveBaselineRunId(opts)\n const priorAll: ReadonlyArray<AnalystFinding & { run_id: string }> = baselineRunId\n ? (opts.findingsStore?.loadRun(baselineRunId) ?? [])\n : []\n log('baseline resolved', { baselineRunId, prior_findings: priorAll.length })\n await emit({\n type: 'baseline-resolved',\n runId: opts.runId,\n baselineRunId,\n priorFindingCount: priorAll.length,\n })\n\n // 2. Run the registry. Strategy controls how analysts see priors.\n // When the registry exposes runStream, forward each event verbatim\n // so subscribers see per-analyst progress in real time.\n const priorFindings = buildPriorFindingsInput(priorAll, strategy, opts.registry.list())\n const analystResult = await runRegistry(opts, priorFindings, emit)\n log('analyst run complete', {\n findings: analystResult.findings.length,\n cost_usd: analystResult.total_cost_usd,\n per_analyst: analystResult.per_analyst.map((s) => ({\n id: s.analyst_id,\n status: s.status,\n n: s.findings_count,\n })),\n })\n\n // 3. Persist the new run before any side-effecting adapter runs so\n // the ledger is the source of truth even if an adapter throws.\n if (opts.findingsStore && analystResult.findings.length > 0) {\n await opts.findingsStore.append(opts.runId, analystResult.findings)\n await emit({\n type: 'findings-persisted',\n runId: opts.runId,\n count: analystResult.findings.length,\n })\n }\n\n // 4. Diff vs baseline.\n let diff: FindingsDiff | null = null\n if (baselineRunId && analystResult.findings.length > 0) {\n diff = diffFindings(\n priorAll.map((f) => ({ ...f })),\n analystResult.findings.map((f) => ({ ...f, run_id: opts.runId })),\n )\n log('diff vs baseline', {\n appeared: diff.appeared.length,\n disappeared: diff.disappeared.length,\n persisted: diff.persisted.length,\n changed: diff.changed.length,\n })\n await emit({\n type: 'diff-computed',\n runId: opts.runId,\n baselineRunId,\n appeared: diff.appeared.length,\n disappeared: diff.disappeared.length,\n persisted: diff.persisted.length,\n changed: diff.changed.length,\n })\n }\n\n // 5. Knowledge adapter — proposals + optional auto-apply.\n let knowledge: KnowledgeReport<TProposal> | null = null\n if (opts.knowledgeAdapter) {\n knowledge = await runKnowledgeAdapter(opts, analystResult.findings, log, emit)\n }\n\n // 6. Improvement adapter — prompt / tool / scaffolding edits.\n let improvement: ImprovementReport<TEdit> | null = null\n if (opts.improvementAdapter) {\n improvement = await runImprovementAdapter(opts, analystResult.findings, log, emit)\n }\n\n await emit({\n type: 'loop-completed',\n runId: opts.runId,\n durationMs: Date.now() - startedAt,\n })\n\n return {\n runId: opts.runId,\n baselineRunId,\n analystResult,\n diff,\n knowledge,\n improvement,\n }\n}\n\ntype Emitter = (event: AnalystLoopEvent) => Promise<void>\n\nfunction makeEmitter(onEvent: RunAnalystLoopOpts['onEvent']): Emitter {\n if (!onEvent) return async () => {}\n return async (event) => {\n await onEvent(event)\n }\n}\n\nasync function runRegistry(\n opts: RunAnalystLoopOpts,\n priorFindings: ReturnType<typeof buildPriorFindingsInput>,\n emit: Emitter,\n): Promise<AnalystRunResult> {\n const reg = opts.registry as AnalystRegistryStreamingLike\n if (typeof reg.runStream === 'function' && opts.onEvent) {\n let final: AnalystRunResult | null = null\n for await (const ev of reg.runStream(opts.runId, opts.inputs, { priorFindings })) {\n await emit({ type: 'analyst', runId: opts.runId, event: ev })\n if (ev.type === 'run-completed') final = ev.result\n }\n if (!final) {\n throw new Error('runAnalystLoop: registry.runStream ended without run-completed event')\n }\n return final\n }\n return opts.registry.run(opts.runId, opts.inputs, { priorFindings })\n}\n\nfunction resolveBaselineRunId(opts: RunAnalystLoopOpts): string | null {\n if (opts.baselineRunId === null) return null\n if (typeof opts.baselineRunId === 'string') return opts.baselineRunId\n if (!opts.findingsStore) return null\n const all = opts.findingsStore.loadAll()\n let last: string | null = null\n for (const row of all) {\n if (row.run_id === opts.runId) continue\n last = row.run_id\n }\n return last\n}\n\nfunction buildPriorFindingsInput(\n prior: ReadonlyArray<AnalystFinding & { run_id: string }>,\n strategy: 'per-kind' | 'wildcard' | 'none',\n registry: ReadonlyArray<{ id: string }>,\n): ReadonlyArray<AnalystFinding> | Record<string, ReadonlyArray<AnalystFinding>> | undefined {\n if (strategy === 'none' || prior.length === 0) return undefined\n const stripped = prior.map(({ run_id: _run_id, ...rest }) => rest as AnalystFinding)\n if (strategy === 'wildcard') {\n return { '*': stripped }\n }\n void registry\n return stripped\n}\n\nasync function runKnowledgeAdapter<TProposal>(\n opts: RunAnalystLoopOpts,\n findings: ReadonlyArray<AnalystFinding>,\n log: NonNullable<RunAnalystLoopOpts['log']>,\n emit: Emitter,\n): Promise<KnowledgeReport<TProposal>> {\n const adapter = opts.knowledgeAdapter!\n const batch = await adapter.proposeFromFindings(findings)\n log('knowledge.proposeFromFindings', {\n proposals: batch.proposals.length,\n skipped: batch.skipped,\n errors: batch.errors.length,\n })\n await emit({\n type: 'knowledge-proposed',\n runId: opts.runId,\n proposalCount: batch.proposals.length,\n skipped: batch.skipped,\n errors: batch.errors.length,\n })\n\n const auto = opts.autoApply?.knowledge ?? false\n const threshold = opts.autoApply?.knowledgeConfidenceThreshold ?? 0.85\n\n if (!auto || !adapter.apply) {\n await emit({\n type: 'knowledge-applied',\n runId: opts.runId,\n writtenCount: 0,\n withheldForReview: batch.proposals.length,\n })\n return {\n proposals: batch.proposals as TProposal[],\n applied: [],\n skipped: batch.skipped,\n errors: batch.errors,\n withheld_for_review: batch.proposals.length,\n }\n }\n\n const findingsById = new Map(findings.map((f) => [f.finding_id, f]))\n const safe: TProposal[] = []\n let withheld = 0\n for (const p of batch.proposals as Array<TProposal & { sourceFindingId?: string }>) {\n const src = p.sourceFindingId ? findingsById.get(p.sourceFindingId) : undefined\n if (!src) {\n withheld += 1\n continue\n }\n if (src.confidence < threshold) {\n withheld += 1\n continue\n }\n safe.push(p)\n }\n const result = await adapter.apply(safe)\n log('knowledge.apply', {\n applied: result.written.length,\n withheld_for_review: withheld,\n warnings: result.warnings.length,\n })\n await emit({\n type: 'knowledge-applied',\n runId: opts.runId,\n writtenCount: result.written.length,\n withheldForReview: withheld,\n })\n return {\n proposals: batch.proposals as TProposal[],\n applied: result.written,\n skipped: batch.skipped,\n errors: batch.errors,\n withheld_for_review: withheld,\n }\n}\n\nasync function runImprovementAdapter<TEdit>(\n opts: RunAnalystLoopOpts,\n findings: ReadonlyArray<AnalystFinding>,\n log: NonNullable<RunAnalystLoopOpts['log']>,\n emit: Emitter,\n): Promise<ImprovementReport<TEdit>> {\n const adapter = opts.improvementAdapter!\n const batch = await adapter.proposeFromFindings(findings)\n log('improvement.proposeFromFindings', {\n edits: batch.edits.length,\n skipped: batch.skipped,\n errors: batch.errors.length,\n })\n await emit({\n type: 'improvement-proposed',\n runId: opts.runId,\n editCount: batch.edits.length,\n skipped: batch.skipped,\n errors: batch.errors.length,\n })\n\n const auto = opts.autoApply?.improvement ?? false\n const threshold = opts.autoApply?.improvementConfidenceThreshold ?? 0.9\n\n if (!auto || !adapter.apply) {\n await emit({\n type: 'improvement-applied',\n runId: opts.runId,\n appliedCount: 0,\n withheldForReview: batch.edits.length,\n })\n return {\n edits: batch.edits as TEdit[],\n applied: [],\n skipped: batch.skipped,\n errors: batch.errors,\n withheld_for_review: batch.edits.length,\n }\n }\n\n const findingsById = new Map(findings.map((f) => [f.finding_id, f]))\n const safe: TEdit[] = []\n let withheld = 0\n for (const e of batch.edits as Array<TEdit & { sourceFindingId?: string }>) {\n const src = e.sourceFindingId ? findingsById.get(e.sourceFindingId) : undefined\n if (!src || src.confidence < threshold) {\n withheld += 1\n continue\n }\n safe.push(e)\n }\n const result = await adapter.apply(safe)\n log('improvement.apply', {\n applied: result.applied.length,\n withheld_for_review: withheld,\n warnings: result.warnings.length,\n })\n await emit({\n type: 'improvement-applied',\n runId: opts.runId,\n appliedCount: result.applied.length,\n withheldForReview: withheld,\n })\n return {\n edits: batch.edits as TEdit[],\n applied: result.applied,\n skipped: batch.skipped,\n errors: batch.errors,\n withheld_for_review: withheld,\n }\n}\n\nfunction defaultLog(msg: string, fields?: Record<string, unknown>): void {\n if (fields) console.log(`[analyst-loop] ${msg}`, fields)\n else console.log(`[analyst-loop] ${msg}`)\n}\n"],"mappings":"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","names":[]}
|