@rudderhq/run-intelligence-core 0.1.0-canary.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +186 -0
- package/dist/cli/analyze.d.ts +2 -0
- package/dist/cli/analyze.d.ts.map +1 -0
- package/dist/cli/analyze.js +35 -0
- package/dist/cli/analyze.js.map +1 -0
- package/dist/cli/common.d.ts +9 -0
- package/dist/cli/common.d.ts.map +1 -0
- package/dist/cli/common.js +28 -0
- package/dist/cli/common.js.map +1 -0
- package/dist/cli/compare.d.ts +2 -0
- package/dist/cli/compare.d.ts.map +1 -0
- package/dist/cli/compare.js +24 -0
- package/dist/cli/compare.js.map +1 -0
- package/dist/cli/trace-entry.d.ts +2 -0
- package/dist/cli/trace-entry.d.ts.map +1 -0
- package/dist/cli/trace-entry.js +53 -0
- package/dist/cli/trace-entry.js.map +1 -0
- package/dist/cli/trace-outline.d.ts +2 -0
- package/dist/cli/trace-outline.d.ts.map +1 -0
- package/dist/cli/trace-outline.js +43 -0
- package/dist/cli/trace-outline.js.map +1 -0
- package/dist/create-agent-benchmark.d.ts +113 -0
- package/dist/create-agent-benchmark.d.ts.map +1 -0
- package/dist/create-agent-benchmark.js +451 -0
- package/dist/create-agent-benchmark.js.map +1 -0
- package/dist/create-agent-benchmark.test.d.ts +2 -0
- package/dist/create-agent-benchmark.test.d.ts.map +1 -0
- package/dist/create-agent-benchmark.test.js +289 -0
- package/dist/create-agent-benchmark.test.js.map +1 -0
- package/dist/diagnosis.d.ts +4 -0
- package/dist/diagnosis.d.ts.map +1 -0
- package/dist/diagnosis.js +360 -0
- package/dist/diagnosis.js.map +1 -0
- package/dist/diagnosis.test.d.ts +2 -0
- package/dist/diagnosis.test.d.ts.map +1 -0
- package/dist/diagnosis.test.js +85 -0
- package/dist/diagnosis.test.js.map +1 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +10 -0
- package/dist/index.js.map +1 -0
- package/dist/langfuse-scores.d.ts +13 -0
- package/dist/langfuse-scores.d.ts.map +1 -0
- package/dist/langfuse-scores.js +95 -0
- package/dist/langfuse-scores.js.map +1 -0
- package/dist/langfuse-scores.test.d.ts +2 -0
- package/dist/langfuse-scores.test.d.ts.map +1 -0
- package/dist/langfuse-scores.test.js +121 -0
- package/dist/langfuse-scores.test.js.map +1 -0
- package/dist/loaders/filesystem.d.ts +26 -0
- package/dist/loaders/filesystem.d.ts.map +1 -0
- package/dist/loaders/filesystem.js +97 -0
- package/dist/loaders/filesystem.js.map +1 -0
- package/dist/loaders/rudder.d.ts +28 -0
- package/dist/loaders/rudder.d.ts.map +1 -0
- package/dist/loaders/rudder.js +81 -0
- package/dist/loaders/rudder.js.map +1 -0
- package/dist/parsers.d.ts +3 -0
- package/dist/parsers.d.ts.map +1 -0
- package/dist/parsers.js +23 -0
- package/dist/parsers.js.map +1 -0
- package/dist/trace.d.ts +42 -0
- package/dist/trace.d.ts.map +1 -0
- package/dist/trace.js +167 -0
- package/dist/trace.js.map +1 -0
- package/dist/trace.test.d.ts +2 -0
- package/dist/trace.test.d.ts.map +1 -0
- package/dist/trace.test.js +92 -0
- package/dist/trace.test.js.map +1 -0
- package/dist/transcript.d.ts +7 -0
- package/dist/transcript.d.ts.map +1 -0
- package/dist/transcript.js +70 -0
- package/dist/transcript.js.map +1 -0
- package/dist/types.d.ts +122 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/package.json +59 -0
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { diagnoseRun } from "../diagnosis.js";
|
|
2
|
+
import { getTranscriptParser } from "../parsers.js";
|
|
3
|
+
import { buildTranscript, parseNdjsonLog } from "../transcript.js";
|
|
4
|
+
async function fetchJson(url, init) {
|
|
5
|
+
const response = await fetch(url, init);
|
|
6
|
+
if (!response.ok) {
|
|
7
|
+
throw new Error(`Request failed (${response.status}) for ${url}`);
|
|
8
|
+
}
|
|
9
|
+
return response.json();
|
|
10
|
+
}
|
|
11
|
+
export async function listOrganizations(apiBaseUrl) {
|
|
12
|
+
return fetchJson(`${apiBaseUrl}/orgs`);
|
|
13
|
+
}
|
|
14
|
+
export async function listObservedRuns(apiBaseUrl, orgId, params) {
|
|
15
|
+
const qs = params && [...params.keys()].length > 0 ? `?${params.toString()}` : "";
|
|
16
|
+
return fetchJson(`${apiBaseUrl}/run-intelligence/orgs/${encodeURIComponent(orgId)}/runs${qs}`);
|
|
17
|
+
}
|
|
18
|
+
export async function getObservedRun(apiBaseUrl, runId) {
|
|
19
|
+
return fetchJson(`${apiBaseUrl}/run-intelligence/runs/${encodeURIComponent(runId)}`);
|
|
20
|
+
}
|
|
21
|
+
export async function getRunEvents(apiBaseUrl, runId) {
|
|
22
|
+
return fetchJson(`${apiBaseUrl}/run-intelligence/runs/${encodeURIComponent(runId)}/events`);
|
|
23
|
+
}
|
|
24
|
+
export async function getRunLog(apiBaseUrl, runId) {
|
|
25
|
+
return fetchJson(`${apiBaseUrl}/run-intelligence/runs/${encodeURIComponent(runId)}/log`);
|
|
26
|
+
}
|
|
27
|
+
export async function findObservedRunByPrefix(apiBaseUrl, runIdPrefix) {
|
|
28
|
+
const organizations = await listOrganizations(apiBaseUrl);
|
|
29
|
+
for (const organization of organizations) {
|
|
30
|
+
const params = new URLSearchParams({ limit: "200", runIdPrefix });
|
|
31
|
+
const rows = await listObservedRuns(apiBaseUrl, organization.id, params);
|
|
32
|
+
const match = rows.find((row) => row.run.id.toLowerCase().startsWith(runIdPrefix.toLowerCase()));
|
|
33
|
+
if (match)
|
|
34
|
+
return match;
|
|
35
|
+
}
|
|
36
|
+
return null;
|
|
37
|
+
}
|
|
38
|
+
export async function loadObservedRunDetail(apiBaseUrl, runId) {
|
|
39
|
+
const [observedRun, events, log] = await Promise.all([
|
|
40
|
+
getObservedRun(apiBaseUrl, runId),
|
|
41
|
+
getRunEvents(apiBaseUrl, runId),
|
|
42
|
+
getRunLog(apiBaseUrl, runId).catch(() => ({ content: "" })),
|
|
43
|
+
]);
|
|
44
|
+
const logChunks = parseNdjsonLog(log.content);
|
|
45
|
+
const transcript = buildTranscript(logChunks, getTranscriptParser(observedRun.bundle.agentRuntimeType));
|
|
46
|
+
return {
|
|
47
|
+
...observedRun,
|
|
48
|
+
events,
|
|
49
|
+
logContent: log.content,
|
|
50
|
+
logChunks,
|
|
51
|
+
transcript,
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
export async function diagnoseObservedRun(apiBaseUrl, runId, mode = "auto") {
|
|
55
|
+
const detail = await loadObservedRunDetail(apiBaseUrl, runId);
|
|
56
|
+
const diagnosis = diagnoseRun(detail, mode);
|
|
57
|
+
return { detail, diagnosis };
|
|
58
|
+
}
|
|
59
|
+
export function observedRunFromFilesystem(input) {
|
|
60
|
+
const bundle = input.bundle ?? {
|
|
61
|
+
agentRuntimeType: input.run.contextSnapshot?.agentRuntimeType ?? "process",
|
|
62
|
+
agentConfigRevisionId: null,
|
|
63
|
+
agentConfigRevisionCreatedAt: null,
|
|
64
|
+
agentConfigFingerprint: null,
|
|
65
|
+
runtimeConfigFingerprint: null,
|
|
66
|
+
};
|
|
67
|
+
const logChunks = parseNdjsonLog(input.logContent);
|
|
68
|
+
const transcript = buildTranscript(logChunks, getTranscriptParser(bundle.agentRuntimeType));
|
|
69
|
+
return {
|
|
70
|
+
run: input.run,
|
|
71
|
+
agentName: input.agentName,
|
|
72
|
+
orgName: input.orgName ?? null,
|
|
73
|
+
issue: input.issue ?? null,
|
|
74
|
+
bundle,
|
|
75
|
+
events: input.events ?? [],
|
|
76
|
+
logContent: input.logContent ?? null,
|
|
77
|
+
logChunks,
|
|
78
|
+
transcript,
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
//# sourceMappingURL=rudder.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"rudder.js","sourceRoot":"","sources":["../../src/loaders/rudder.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"parsers.d.ts","sourceRoot":"","sources":["../src/parsers.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,+BAA+B,CAAC;AAuBtE,wBAAgB,mBAAmB,CAAC,gBAAgB,EAAE,MAAM,GAAG,gBAAgB,CAE9E"}
|
package/dist/parsers.js
ADDED
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { parseClaudeStdoutLine } from "@rudderhq/agent-runtime-claude-local/ui";
|
|
2
|
+
import { parseCodexStdoutLine } from "@rudderhq/agent-runtime-codex-local/ui";
|
|
3
|
+
import { parseCursorStdoutLine } from "@rudderhq/agent-runtime-cursor-local/ui";
|
|
4
|
+
import { parseGeminiStdoutLine } from "@rudderhq/agent-runtime-gemini-local/ui";
|
|
5
|
+
import { parseOpenClawGatewayStdoutLine } from "@rudderhq/agent-runtime-openclaw-gateway/ui";
|
|
6
|
+
import { parseOpenCodeStdoutLine } from "@rudderhq/agent-runtime-opencode-local/ui";
|
|
7
|
+
import { parsePiStdoutLine } from "@rudderhq/agent-runtime-pi-local/ui";
|
|
8
|
+
const genericParser = (line, ts) => [{ kind: "stdout", ts, text: line }];
|
|
9
|
+
const parserByRuntimeType = {
|
|
10
|
+
claude_local: parseClaudeStdoutLine,
|
|
11
|
+
codex_local: parseCodexStdoutLine,
|
|
12
|
+
cursor: parseCursorStdoutLine,
|
|
13
|
+
gemini_local: parseGeminiStdoutLine,
|
|
14
|
+
openclaw_gateway: parseOpenClawGatewayStdoutLine,
|
|
15
|
+
opencode_local: parseOpenCodeStdoutLine,
|
|
16
|
+
pi_local: parsePiStdoutLine,
|
|
17
|
+
process: genericParser,
|
|
18
|
+
http: genericParser,
|
|
19
|
+
};
|
|
20
|
+
export function getTranscriptParser(agentRuntimeType) {
|
|
21
|
+
return parserByRuntimeType[agentRuntimeType] ?? genericParser;
|
|
22
|
+
}
|
|
23
|
+
//# sourceMappingURL=parsers.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"parsers.js","sourceRoot":"","sources":["../src/parsers.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,qBAAqB,EAAE,MAAM,yCAAyC,CAAC;AAChF,OAAO,EAAE,oBAAoB,EAAE,MAAM,wCAAwC,CAAC;AAC9E,OAAO,EAAE,qBAAqB,EAAE,MAAM,yCAAyC,CAAC;AAChF,OAAO,EAAE,qBAAqB,EAAE,MAAM,yCAAyC,CAAC;AAChF,OAAO,EAAE,8BAA8B,EAAE,MAAM,6CAA6C,CAAC;AAC7F,OAAO,EAAE,uBAAuB,EAAE,MAAM,2CAA2C,CAAC;AACpF,OAAO,EAAE,iBAAiB,EAAE,MAAM,qCAAqC,CAAC;AAExE,MAAM,aAAa,GAAqB,CAAC,IAAI,EAAE,EAAE,EAAE,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;AAE3F,MAAM,mBAAmB,GAAqC;IAC5D,YAAY,EAAE,qBAAqB;IACnC,WAAW,EAAE,oBAAoB;IACjC,MAAM,EAAE,qBAAqB;IAC7B,YAAY,EAAE,qBAAqB;IACnC,gBAAgB,EAAE,8BAA8B;IAChD,cAAc,EAAE,uBAAuB;IACvC,QAAQ,EAAE,iBAAiB;IAC3B,OAAO,EAAE,aAAa;IACtB,IAAI,EAAE,aAAa;CACpB,CAAC;AAEF,MAAM,UAAU,mBAAmB,CAAC,gBAAwB;IAC1D,OAAO,mBAAmB,CAAC,gBAAgB,CAAC,IAAI,aAAa,CAAC;AAChE,CAAC"}
|
package/dist/trace.d.ts
ADDED
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import type { TranscriptEntry } from "@rudderhq/agent-runtime-utils";
|
|
2
|
+
import type { ObservedRunDetail, ObservedRunTrace } from "./types.js";
|
|
3
|
+
export declare function isModelTranscriptEntry(entry: TranscriptEntry): entry is {
|
|
4
|
+
kind: "assistant";
|
|
5
|
+
ts: string;
|
|
6
|
+
text: string;
|
|
7
|
+
delta?: boolean;
|
|
8
|
+
} | {
|
|
9
|
+
kind: "thinking";
|
|
10
|
+
ts: string;
|
|
11
|
+
text: string;
|
|
12
|
+
delta?: boolean;
|
|
13
|
+
} | {
|
|
14
|
+
kind: "result";
|
|
15
|
+
ts: string;
|
|
16
|
+
text: string;
|
|
17
|
+
inputTokens: number;
|
|
18
|
+
outputTokens: number;
|
|
19
|
+
cachedTokens: number;
|
|
20
|
+
costUsd: number;
|
|
21
|
+
subtype: string;
|
|
22
|
+
isError: boolean;
|
|
23
|
+
errors: string[];
|
|
24
|
+
};
|
|
25
|
+
export declare function isPayloadTranscriptEntry(entry: TranscriptEntry): entry is {
|
|
26
|
+
kind: "tool_call";
|
|
27
|
+
ts: string;
|
|
28
|
+
name: string;
|
|
29
|
+
input: unknown;
|
|
30
|
+
toolUseId?: string;
|
|
31
|
+
} | {
|
|
32
|
+
kind: "tool_result";
|
|
33
|
+
ts: string;
|
|
34
|
+
toolUseId: string;
|
|
35
|
+
toolName?: string;
|
|
36
|
+
content: string;
|
|
37
|
+
isError: boolean;
|
|
38
|
+
};
|
|
39
|
+
export declare function detailTextForTranscriptEntry(entry: TranscriptEntry): string;
|
|
40
|
+
export declare function previewTextForTranscriptEntry(entry: TranscriptEntry, maxLength?: number): string;
|
|
41
|
+
export declare function buildObservedRunTrace(detailOrEntries: ObservedRunDetail | TranscriptEntry[]): ObservedRunTrace;
|
|
42
|
+
//# sourceMappingURL=trace.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"trace.d.ts","sourceRoot":"","sources":["../src/trace.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAC;AACrE,OAAO,KAAK,EAAE,iBAAiB,EAAmB,gBAAgB,EAAmB,MAAM,YAAY,CAAC;AA0BxG,wBAAgB,sBAAsB,CAAC,KAAK,EAAE,eAAe;;;;;;;;;;;;;;;;;;;;;EAE5D;AAED,wBAAgB,wBAAwB,CAAC,KAAK,EAAE,eAAe;;;;;;;;;;;;;EAE9D;AAED,wBAAgB,4BAA4B,CAAC,KAAK,EAAE,eAAe,UAWlE;AAED,wBAAgB,6BAA6B,CAAC,KAAK,EAAE,eAAe,EAAE,SAAS,SAAM,UAoBpF;AAgBD,wBAAgB,qBAAqB,CAAC,eAAe,EAAE,iBAAiB,GAAG,eAAe,EAAE,GAAG,gBAAgB,CAuF9G"}
|
package/dist/trace.js
ADDED
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
function truncate(value, maxLength = 180) {
|
|
2
|
+
const text = value.trim();
|
|
3
|
+
if (!text)
|
|
4
|
+
return "";
|
|
5
|
+
return text.length > maxLength ? `${text.slice(0, maxLength - 1)}…` : text;
|
|
6
|
+
}
|
|
7
|
+
function firstMeaningfulLine(value) {
|
|
8
|
+
for (const line of value.split(/\r?\n/)) {
|
|
9
|
+
const trimmed = line.trim();
|
|
10
|
+
if (trimmed)
|
|
11
|
+
return trimmed;
|
|
12
|
+
}
|
|
13
|
+
return "";
|
|
14
|
+
}
|
|
15
|
+
function formatJson(value) {
|
|
16
|
+
if (value === null || value === undefined)
|
|
17
|
+
return "";
|
|
18
|
+
if (typeof value === "string")
|
|
19
|
+
return value;
|
|
20
|
+
try {
|
|
21
|
+
return JSON.stringify(value, null, 2);
|
|
22
|
+
}
|
|
23
|
+
catch {
|
|
24
|
+
return String(value);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
export function isModelTranscriptEntry(entry) {
|
|
28
|
+
return entry.kind === "assistant" || entry.kind === "thinking" || entry.kind === "result";
|
|
29
|
+
}
|
|
30
|
+
export function isPayloadTranscriptEntry(entry) {
|
|
31
|
+
return entry.kind === "tool_call" || entry.kind === "tool_result";
|
|
32
|
+
}
|
|
33
|
+
export function detailTextForTranscriptEntry(entry) {
|
|
34
|
+
if (entry.kind === "tool_call")
|
|
35
|
+
return formatJson(entry.input);
|
|
36
|
+
if (entry.kind === "tool_result")
|
|
37
|
+
return entry.content || "";
|
|
38
|
+
if (entry.kind === "result") {
|
|
39
|
+
return [
|
|
40
|
+
entry.text || "",
|
|
41
|
+
entry.errors.length ? `Errors:\n${entry.errors.join("\n")}` : "",
|
|
42
|
+
].filter(Boolean).join("\n\n");
|
|
43
|
+
}
|
|
44
|
+
if (entry.kind === "init")
|
|
45
|
+
return `model=${entry.model}\nsession=${entry.sessionId}`;
|
|
46
|
+
return "text" in entry ? entry.text || "" : "";
|
|
47
|
+
}
|
|
48
|
+
export function previewTextForTranscriptEntry(entry, maxLength = 180) {
|
|
49
|
+
if (entry.kind === "tool_call") {
|
|
50
|
+
return truncate(`${entry.name}(${formatJson(entry.input)})`, maxLength);
|
|
51
|
+
}
|
|
52
|
+
if (entry.kind === "tool_result") {
|
|
53
|
+
const detail = detailTextForTranscriptEntry(entry);
|
|
54
|
+
return truncate(firstMeaningfulLine(detail) || entry.toolName || "(empty tool result)", maxLength);
|
|
55
|
+
}
|
|
56
|
+
if (entry.kind === "result") {
|
|
57
|
+
const summary = entry.text ? firstMeaningfulLine(entry.text) : "";
|
|
58
|
+
if (summary)
|
|
59
|
+
return truncate(summary, maxLength);
|
|
60
|
+
return truncate(`${entry.subtype} · tokens ${entry.inputTokens}/${entry.outputTokens} · $${Number(entry.costUsd || 0).toFixed(2)}`, maxLength);
|
|
61
|
+
}
|
|
62
|
+
if (entry.kind === "init") {
|
|
63
|
+
return truncate(`${entry.model} · ${entry.sessionId}`, maxLength);
|
|
64
|
+
}
|
|
65
|
+
if ("text" in entry) {
|
|
66
|
+
return truncate(firstMeaningfulLine(entry.text || "") || entry.text || "", maxLength);
|
|
67
|
+
}
|
|
68
|
+
return "";
|
|
69
|
+
}
|
|
70
|
+
function isTurnScopedEntry(entry) {
|
|
71
|
+
return entry.kind === "assistant"
|
|
72
|
+
|| entry.kind === "thinking"
|
|
73
|
+
|| entry.kind === "tool_call"
|
|
74
|
+
|| entry.kind === "tool_result"
|
|
75
|
+
|| entry.kind === "result";
|
|
76
|
+
}
|
|
77
|
+
function isErrorTranscriptEntry(entry) {
|
|
78
|
+
return entry.kind === "stderr"
|
|
79
|
+
|| (entry.kind === "tool_result" && entry.isError)
|
|
80
|
+
|| (entry.kind === "result" && entry.isError);
|
|
81
|
+
}
|
|
82
|
+
export function buildObservedRunTrace(detailOrEntries) {
|
|
83
|
+
const entries = Array.isArray(detailOrEntries) ? detailOrEntries : detailOrEntries.transcript;
|
|
84
|
+
const steps = [];
|
|
85
|
+
const looseSteps = [];
|
|
86
|
+
const turns = new Map();
|
|
87
|
+
let nextTurnIndex = 0;
|
|
88
|
+
let activeTurn = null;
|
|
89
|
+
for (const [rawIndex, entry] of entries.entries()) {
|
|
90
|
+
let turnIndex = null;
|
|
91
|
+
if (entry.kind === "assistant" || entry.kind === "thinking") {
|
|
92
|
+
if (activeTurn === null)
|
|
93
|
+
activeTurn = ++nextTurnIndex;
|
|
94
|
+
turnIndex = activeTurn;
|
|
95
|
+
}
|
|
96
|
+
else if (entry.kind === "tool_call" || entry.kind === "tool_result" || entry.kind === "result") {
|
|
97
|
+
if (activeTurn === null)
|
|
98
|
+
activeTurn = ++nextTurnIndex;
|
|
99
|
+
turnIndex = activeTurn;
|
|
100
|
+
if (entry.kind === "result")
|
|
101
|
+
activeTurn = null;
|
|
102
|
+
}
|
|
103
|
+
const detailText = detailTextForTranscriptEntry(entry);
|
|
104
|
+
const preview = previewTextForTranscriptEntry(entry);
|
|
105
|
+
const detailPreview = truncate(firstMeaningfulLine(detailText) || preview || "(empty)", 220);
|
|
106
|
+
const step = {
|
|
107
|
+
index: rawIndex + 1,
|
|
108
|
+
turnIndex,
|
|
109
|
+
kind: entry.kind,
|
|
110
|
+
ts: entry.ts,
|
|
111
|
+
label: turnIndex ? `Turn ${turnIndex}` : `Step ${rawIndex + 1}`,
|
|
112
|
+
preview,
|
|
113
|
+
detailPreview,
|
|
114
|
+
detailText,
|
|
115
|
+
isModelEntry: isModelTranscriptEntry(entry),
|
|
116
|
+
isPayloadEntry: isPayloadTranscriptEntry(entry),
|
|
117
|
+
hasExpandableDetail: Boolean(detailText && detailText.trim() && detailText.trim() !== preview.trim()),
|
|
118
|
+
isError: isErrorTranscriptEntry(entry),
|
|
119
|
+
};
|
|
120
|
+
steps.push(step);
|
|
121
|
+
if (!isTurnScopedEntry(entry) || turnIndex === null) {
|
|
122
|
+
looseSteps.push(step);
|
|
123
|
+
continue;
|
|
124
|
+
}
|
|
125
|
+
const turn = turns.get(turnIndex) ?? {
|
|
126
|
+
turnIndex,
|
|
127
|
+
label: `Turn ${turnIndex}`,
|
|
128
|
+
summary: "",
|
|
129
|
+
startedAt: step.ts,
|
|
130
|
+
endedAt: step.ts,
|
|
131
|
+
stepCount: 0,
|
|
132
|
+
toolCallCount: 0,
|
|
133
|
+
hasError: false,
|
|
134
|
+
steps: [],
|
|
135
|
+
};
|
|
136
|
+
turn.steps.push(step);
|
|
137
|
+
turn.stepCount += 1;
|
|
138
|
+
turn.endedAt = step.ts;
|
|
139
|
+
if (!turn.summary && step.preview) {
|
|
140
|
+
turn.summary = step.preview;
|
|
141
|
+
}
|
|
142
|
+
if (step.kind === "tool_call") {
|
|
143
|
+
turn.toolCallCount += 1;
|
|
144
|
+
}
|
|
145
|
+
if (step.isError) {
|
|
146
|
+
turn.hasError = true;
|
|
147
|
+
}
|
|
148
|
+
turns.set(turnIndex, turn);
|
|
149
|
+
}
|
|
150
|
+
const orderedTurns = [...turns.values()].map((turn) => {
|
|
151
|
+
const headline = turn.steps.find((step) => step.isModelEntry && step.preview)
|
|
152
|
+
?? turn.steps.find((step) => step.preview)
|
|
153
|
+
?? null;
|
|
154
|
+
return {
|
|
155
|
+
...turn,
|
|
156
|
+
summary: headline?.preview || "No transcript summary",
|
|
157
|
+
};
|
|
158
|
+
});
|
|
159
|
+
return {
|
|
160
|
+
steps,
|
|
161
|
+
looseSteps,
|
|
162
|
+
turns: orderedTurns,
|
|
163
|
+
turnCount: orderedTurns.length,
|
|
164
|
+
payloadStepCount: steps.filter((step) => step.isPayloadEntry).length,
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
//# sourceMappingURL=trace.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"trace.js","sourceRoot":"","sources":["../src/trace.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"trace.test.d.ts","sourceRoot":"","sources":["../src/trace.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { buildObservedRunTrace } from "./trace.js";
|
|
3
|
+
import { observedRunFromFilesystem } from "./loaders/rudder.js";
|
|
4
|
+
describe("buildObservedRunTrace", () => {
|
|
5
|
+
it("keeps tool payloads compact in previews while preserving full detail", () => {
|
|
6
|
+
const detail = observedRunFromFilesystem({
|
|
7
|
+
run: {
|
|
8
|
+
id: "run-trace-1",
|
|
9
|
+
orgId: "org-1",
|
|
10
|
+
agentId: "agent-1",
|
|
11
|
+
invocationSource: "on_demand",
|
|
12
|
+
triggerDetail: "manual",
|
|
13
|
+
status: "succeeded",
|
|
14
|
+
startedAt: new Date("2026-04-08T10:00:00.000Z"),
|
|
15
|
+
finishedAt: new Date("2026-04-08T10:01:00.000Z"),
|
|
16
|
+
error: null,
|
|
17
|
+
wakeupRequestId: null,
|
|
18
|
+
exitCode: 0,
|
|
19
|
+
signal: null,
|
|
20
|
+
usageJson: { inputTokens: 10, outputTokens: 20, costUsd: 0.12 },
|
|
21
|
+
resultJson: null,
|
|
22
|
+
sessionIdBefore: null,
|
|
23
|
+
sessionIdAfter: null,
|
|
24
|
+
logStore: "local_file",
|
|
25
|
+
logRef: "trace.ndjson",
|
|
26
|
+
logBytes: 100,
|
|
27
|
+
logSha256: null,
|
|
28
|
+
logCompressed: false,
|
|
29
|
+
stdoutExcerpt: null,
|
|
30
|
+
stderrExcerpt: null,
|
|
31
|
+
errorCode: null,
|
|
32
|
+
externalRunId: null,
|
|
33
|
+
processPid: null,
|
|
34
|
+
processStartedAt: null,
|
|
35
|
+
retryOfRunId: null,
|
|
36
|
+
processLossRetryCount: 0,
|
|
37
|
+
contextSnapshot: {},
|
|
38
|
+
createdAt: new Date("2026-04-08T10:00:00.000Z"),
|
|
39
|
+
updatedAt: new Date("2026-04-08T10:01:00.000Z"),
|
|
40
|
+
},
|
|
41
|
+
agentName: "Trace Agent",
|
|
42
|
+
bundle: {
|
|
43
|
+
agentRuntimeType: "codex_local",
|
|
44
|
+
agentConfigRevisionId: null,
|
|
45
|
+
agentConfigRevisionCreatedAt: null,
|
|
46
|
+
agentConfigFingerprint: null,
|
|
47
|
+
runtimeConfigFingerprint: null,
|
|
48
|
+
},
|
|
49
|
+
logContent: "",
|
|
50
|
+
});
|
|
51
|
+
detail.transcript = [
|
|
52
|
+
{ kind: "system", ts: "2026-04-08T10:00:00.000Z", text: "booted" },
|
|
53
|
+
{ kind: "assistant", ts: "2026-04-08T10:00:01.000Z", text: "I will inspect the codebase." },
|
|
54
|
+
{
|
|
55
|
+
kind: "tool_call",
|
|
56
|
+
ts: "2026-04-08T10:00:02.000Z",
|
|
57
|
+
name: "Read",
|
|
58
|
+
input: { filePath: "/tmp/big-file.ts" },
|
|
59
|
+
},
|
|
60
|
+
{
|
|
61
|
+
kind: "tool_result",
|
|
62
|
+
ts: "2026-04-08T10:00:03.000Z",
|
|
63
|
+
toolUseId: "tool-1",
|
|
64
|
+
toolName: "Read",
|
|
65
|
+
content: "first line\nsecond line\nthird line",
|
|
66
|
+
isError: false,
|
|
67
|
+
},
|
|
68
|
+
{
|
|
69
|
+
kind: "result",
|
|
70
|
+
ts: "2026-04-08T10:00:04.000Z",
|
|
71
|
+
text: "",
|
|
72
|
+
inputTokens: 10,
|
|
73
|
+
outputTokens: 20,
|
|
74
|
+
cachedTokens: 0,
|
|
75
|
+
costUsd: 0.12,
|
|
76
|
+
subtype: "completed",
|
|
77
|
+
isError: false,
|
|
78
|
+
errors: [],
|
|
79
|
+
},
|
|
80
|
+
];
|
|
81
|
+
const trace = buildObservedRunTrace(detail);
|
|
82
|
+
expect(trace.looseSteps).toHaveLength(1);
|
|
83
|
+
expect(trace.turns).toHaveLength(1);
|
|
84
|
+
expect(trace.turns[0]?.toolCallCount).toBe(1);
|
|
85
|
+
expect(trace.turns[0]?.summary).toContain("I will inspect the codebase.");
|
|
86
|
+
const toolResultStep = trace.steps.find((step) => step.kind === "tool_result");
|
|
87
|
+
expect(toolResultStep?.detailPreview).toBe("first line");
|
|
88
|
+
expect(toolResultStep?.detailText).toContain("second line");
|
|
89
|
+
expect(toolResultStep?.hasExpandableDetail).toBe(true);
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
//# sourceMappingURL=trace.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"trace.test.js","sourceRoot":"","sources":["../src/trace.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { type StdoutLineParser, type TranscriptEntry } from "@rudderhq/agent-runtime-utils";
|
|
2
|
+
import type { RunLogChunk } from "./types.js";
|
|
3
|
+
export declare function buildTranscript(chunks: RunLogChunk[], parser: StdoutLineParser, opts?: {
|
|
4
|
+
censorUsernameInLogs?: boolean;
|
|
5
|
+
}): TranscriptEntry[];
|
|
6
|
+
export declare function parseNdjsonLog(content: string | null | undefined): RunLogChunk[];
|
|
7
|
+
//# sourceMappingURL=transcript.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transcript.d.ts","sourceRoot":"","sources":["../src/transcript.ts"],"names":[],"mappings":"AAAA,OAAO,EAA0D,KAAK,gBAAgB,EAAE,KAAK,eAAe,EAAE,MAAM,+BAA+B,CAAC;AACpJ,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAE9C,wBAAgB,eAAe,CAC7B,MAAM,EAAE,WAAW,EAAE,EACrB,MAAM,EAAE,gBAAgB,EACxB,IAAI,CAAC,EAAE;IAAE,oBAAoB,CAAC,EAAE,OAAO,CAAA;CAAE,GACxC,eAAe,EAAE,CAgDnB;AAED,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI,GAAG,SAAS,GAAG,WAAW,EAAE,CAkBhF"}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import { redactHomePathUserSegments, redactTranscriptEntryPaths } from "@rudderhq/agent-runtime-utils";
|
|
2
|
+
export function buildTranscript(chunks, parser, opts) {
|
|
3
|
+
const entries = [];
|
|
4
|
+
let stdoutBuffer = "";
|
|
5
|
+
const redactionOptions = { enabled: opts?.censorUsernameInLogs ?? false };
|
|
6
|
+
const appendEntry = (entry) => {
|
|
7
|
+
if ((entry.kind === "thinking" || entry.kind === "assistant") && entry.delta) {
|
|
8
|
+
const last = entries[entries.length - 1];
|
|
9
|
+
if (last && last.kind === entry.kind && last.delta) {
|
|
10
|
+
last.text += entry.text;
|
|
11
|
+
last.ts = entry.ts;
|
|
12
|
+
return;
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
entries.push(entry);
|
|
16
|
+
};
|
|
17
|
+
for (const chunk of chunks) {
|
|
18
|
+
if (chunk.stream === "stderr") {
|
|
19
|
+
appendEntry({ kind: "stderr", ts: chunk.ts, text: redactHomePathUserSegments(chunk.chunk, redactionOptions) });
|
|
20
|
+
continue;
|
|
21
|
+
}
|
|
22
|
+
if (chunk.stream === "system") {
|
|
23
|
+
appendEntry({ kind: "system", ts: chunk.ts, text: redactHomePathUserSegments(chunk.chunk, redactionOptions) });
|
|
24
|
+
continue;
|
|
25
|
+
}
|
|
26
|
+
const combined = stdoutBuffer + chunk.chunk;
|
|
27
|
+
const lines = combined.split(/\r?\n/);
|
|
28
|
+
stdoutBuffer = lines.pop() ?? "";
|
|
29
|
+
for (const line of lines) {
|
|
30
|
+
const trimmed = line.trim();
|
|
31
|
+
if (!trimmed)
|
|
32
|
+
continue;
|
|
33
|
+
for (const entry of parser(trimmed, chunk.ts)) {
|
|
34
|
+
appendEntry(redactTranscriptEntryPaths(entry, redactionOptions));
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
const trailing = stdoutBuffer.trim();
|
|
39
|
+
if (trailing) {
|
|
40
|
+
const ts = chunks.length > 0 ? chunks[chunks.length - 1].ts : new Date().toISOString();
|
|
41
|
+
for (const entry of parser(trailing, ts)) {
|
|
42
|
+
appendEntry(redactTranscriptEntryPaths(entry, redactionOptions));
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return entries;
|
|
46
|
+
}
|
|
47
|
+
export function parseNdjsonLog(content) {
|
|
48
|
+
if (!content)
|
|
49
|
+
return [];
|
|
50
|
+
const chunks = [];
|
|
51
|
+
for (const line of content.split(/\r?\n/)) {
|
|
52
|
+
const trimmed = line.trim();
|
|
53
|
+
if (!trimmed)
|
|
54
|
+
continue;
|
|
55
|
+
try {
|
|
56
|
+
const parsed = JSON.parse(trimmed);
|
|
57
|
+
const ts = typeof parsed.ts === "string" ? parsed.ts : new Date().toISOString();
|
|
58
|
+
const stream = parsed.stream === "stderr" || parsed.stream === "system" ? parsed.stream : "stdout";
|
|
59
|
+
const chunk = typeof parsed.chunk === "string" ? parsed.chunk : "";
|
|
60
|
+
if (!chunk)
|
|
61
|
+
continue;
|
|
62
|
+
chunks.push({ ts, stream, chunk });
|
|
63
|
+
}
|
|
64
|
+
catch {
|
|
65
|
+
continue;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
return chunks;
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=transcript.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transcript.js","sourceRoot":"","sources":["../src/transcript.ts"],"names":[],"mappings":"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"}
|