Files
openclaw/src/agents/bootstrap-budget.ts
Gustavo Madeira Santana e4b4486a96 Agent: unify bootstrap truncation warning handling (#32769)
Merged via squash.

Prepared head SHA: 5d6d4ddfa6
Co-authored-by: gumadeiras <5599352+gumadeiras@users.noreply.github.com>
Co-authored-by: gumadeiras <5599352+gumadeiras@users.noreply.github.com>
Reviewed-by: @gumadeiras
2026-03-03 16:28:38 -05:00

350 lines
11 KiB
TypeScript

import path from "node:path";
import type { EmbeddedContextFile } from "./pi-embedded-helpers.js";
import type { WorkspaceBootstrapFile } from "./workspace.js";
export const DEFAULT_BOOTSTRAP_NEAR_LIMIT_RATIO = 0.85;
export const DEFAULT_BOOTSTRAP_PROMPT_WARNING_MAX_FILES = 3;
export const DEFAULT_BOOTSTRAP_PROMPT_WARNING_SIGNATURE_HISTORY_MAX = 32;
export type BootstrapTruncationCause = "per-file-limit" | "total-limit";
export type BootstrapPromptWarningMode = "off" | "once" | "always";
export type BootstrapInjectionStat = {
name: string;
path: string;
missing: boolean;
rawChars: number;
injectedChars: number;
truncated: boolean;
};
export type BootstrapAnalyzedFile = BootstrapInjectionStat & {
nearLimit: boolean;
causes: BootstrapTruncationCause[];
};
export type BootstrapBudgetAnalysis = {
files: BootstrapAnalyzedFile[];
truncatedFiles: BootstrapAnalyzedFile[];
nearLimitFiles: BootstrapAnalyzedFile[];
totalNearLimit: boolean;
hasTruncation: boolean;
totals: {
rawChars: number;
injectedChars: number;
truncatedChars: number;
bootstrapMaxChars: number;
bootstrapTotalMaxChars: number;
nearLimitRatio: number;
};
};
export type BootstrapPromptWarning = {
signature?: string;
warningShown: boolean;
lines: string[];
warningSignaturesSeen: string[];
};
export type BootstrapTruncationReportMeta = {
warningMode: BootstrapPromptWarningMode;
warningShown: boolean;
promptWarningSignature?: string;
warningSignaturesSeen?: string[];
truncatedFiles: number;
nearLimitFiles: number;
totalNearLimit: boolean;
};
function normalizePositiveLimit(value: number): number {
if (!Number.isFinite(value) || value <= 0) {
return 1;
}
return Math.floor(value);
}
function formatWarningCause(cause: BootstrapTruncationCause): string {
return cause === "per-file-limit" ? "max/file" : "max/total";
}
function normalizeSeenSignatures(signatures?: string[]): string[] {
if (!Array.isArray(signatures) || signatures.length === 0) {
return [];
}
const seen = new Set<string>();
const result: string[] = [];
for (const signature of signatures) {
const value = typeof signature === "string" ? signature.trim() : "";
if (!value || seen.has(value)) {
continue;
}
seen.add(value);
result.push(value);
}
return result;
}
function appendSeenSignature(signatures: string[], signature: string): string[] {
if (!signature.trim()) {
return signatures;
}
if (signatures.includes(signature)) {
return signatures;
}
const next = [...signatures, signature];
if (next.length <= DEFAULT_BOOTSTRAP_PROMPT_WARNING_SIGNATURE_HISTORY_MAX) {
return next;
}
return next.slice(-DEFAULT_BOOTSTRAP_PROMPT_WARNING_SIGNATURE_HISTORY_MAX);
}
export function resolveBootstrapWarningSignaturesSeen(report?: {
bootstrapTruncation?: {
warningMode?: BootstrapPromptWarningMode;
warningSignaturesSeen?: string[];
promptWarningSignature?: string;
};
}): string[] {
const truncation = report?.bootstrapTruncation;
const seenFromReport = normalizeSeenSignatures(truncation?.warningSignaturesSeen);
if (seenFromReport.length > 0) {
return seenFromReport;
}
// In off mode, signature metadata should not seed once-mode dedupe state.
if (truncation?.warningMode === "off") {
return [];
}
const single =
typeof truncation?.promptWarningSignature === "string"
? truncation.promptWarningSignature.trim()
: "";
return single ? [single] : [];
}
export function buildBootstrapInjectionStats(params: {
bootstrapFiles: WorkspaceBootstrapFile[];
injectedFiles: EmbeddedContextFile[];
}): BootstrapInjectionStat[] {
const injectedByPath = new Map<string, string>();
const injectedByBaseName = new Map<string, string>();
for (const file of params.injectedFiles) {
const pathValue = typeof file.path === "string" ? file.path.trim() : "";
if (!pathValue) {
continue;
}
if (!injectedByPath.has(pathValue)) {
injectedByPath.set(pathValue, file.content);
}
const normalizedPath = pathValue.replace(/\\/g, "/");
const baseName = path.posix.basename(normalizedPath);
if (!injectedByBaseName.has(baseName)) {
injectedByBaseName.set(baseName, file.content);
}
}
return params.bootstrapFiles.map((file) => {
const pathValue = typeof file.path === "string" ? file.path.trim() : "";
const rawChars = file.missing ? 0 : (file.content ?? "").trimEnd().length;
const injected =
(pathValue ? injectedByPath.get(pathValue) : undefined) ??
injectedByPath.get(file.name) ??
injectedByBaseName.get(file.name);
const injectedChars = injected ? injected.length : 0;
const truncated = !file.missing && injectedChars < rawChars;
return {
name: file.name,
path: pathValue || file.name,
missing: file.missing,
rawChars,
injectedChars,
truncated,
};
});
}
export function analyzeBootstrapBudget(params: {
files: BootstrapInjectionStat[];
bootstrapMaxChars: number;
bootstrapTotalMaxChars: number;
nearLimitRatio?: number;
}): BootstrapBudgetAnalysis {
const bootstrapMaxChars = normalizePositiveLimit(params.bootstrapMaxChars);
const bootstrapTotalMaxChars = normalizePositiveLimit(params.bootstrapTotalMaxChars);
const nearLimitRatio =
typeof params.nearLimitRatio === "number" &&
Number.isFinite(params.nearLimitRatio) &&
params.nearLimitRatio > 0 &&
params.nearLimitRatio < 1
? params.nearLimitRatio
: DEFAULT_BOOTSTRAP_NEAR_LIMIT_RATIO;
const nonMissing = params.files.filter((file) => !file.missing);
const rawChars = nonMissing.reduce((sum, file) => sum + file.rawChars, 0);
const injectedChars = nonMissing.reduce((sum, file) => sum + file.injectedChars, 0);
const totalNearLimit = injectedChars >= Math.ceil(bootstrapTotalMaxChars * nearLimitRatio);
const totalOverLimit = injectedChars >= bootstrapTotalMaxChars;
const files = params.files.map((file) => {
if (file.missing) {
return { ...file, nearLimit: false, causes: [] };
}
const perFileOverLimit = file.rawChars > bootstrapMaxChars;
const nearLimit = file.rawChars >= Math.ceil(bootstrapMaxChars * nearLimitRatio);
const causes: BootstrapTruncationCause[] = [];
if (file.truncated) {
if (perFileOverLimit) {
causes.push("per-file-limit");
}
if (totalOverLimit) {
causes.push("total-limit");
}
}
return { ...file, nearLimit, causes };
});
const truncatedFiles = files.filter((file) => file.truncated);
const nearLimitFiles = files.filter((file) => file.nearLimit);
return {
files,
truncatedFiles,
nearLimitFiles,
totalNearLimit,
hasTruncation: truncatedFiles.length > 0,
totals: {
rawChars,
injectedChars,
truncatedChars: Math.max(0, rawChars - injectedChars),
bootstrapMaxChars,
bootstrapTotalMaxChars,
nearLimitRatio,
},
};
}
export function buildBootstrapTruncationSignature(
analysis: BootstrapBudgetAnalysis,
): string | undefined {
if (!analysis.hasTruncation) {
return undefined;
}
const files = analysis.truncatedFiles
.map((file) => ({
path: file.path || file.name,
rawChars: file.rawChars,
injectedChars: file.injectedChars,
causes: [...file.causes].toSorted(),
}))
.toSorted((a, b) => {
const pathCmp = a.path.localeCompare(b.path);
if (pathCmp !== 0) {
return pathCmp;
}
if (a.rawChars !== b.rawChars) {
return a.rawChars - b.rawChars;
}
if (a.injectedChars !== b.injectedChars) {
return a.injectedChars - b.injectedChars;
}
return a.causes.join("+").localeCompare(b.causes.join("+"));
});
return JSON.stringify({
bootstrapMaxChars: analysis.totals.bootstrapMaxChars,
bootstrapTotalMaxChars: analysis.totals.bootstrapTotalMaxChars,
files,
});
}
export function formatBootstrapTruncationWarningLines(params: {
analysis: BootstrapBudgetAnalysis;
maxFiles?: number;
}): string[] {
if (!params.analysis.hasTruncation) {
return [];
}
const maxFiles =
typeof params.maxFiles === "number" && Number.isFinite(params.maxFiles) && params.maxFiles > 0
? Math.floor(params.maxFiles)
: DEFAULT_BOOTSTRAP_PROMPT_WARNING_MAX_FILES;
const lines: string[] = [];
const duplicateNameCounts = params.analysis.truncatedFiles.reduce((acc, file) => {
acc.set(file.name, (acc.get(file.name) ?? 0) + 1);
return acc;
}, new Map<string, number>());
const topFiles = params.analysis.truncatedFiles.slice(0, maxFiles);
for (const file of topFiles) {
const pct =
file.rawChars > 0
? Math.round(((file.rawChars - file.injectedChars) / file.rawChars) * 100)
: 0;
const causeText =
file.causes.length > 0
? file.causes.map((cause) => formatWarningCause(cause)).join(", ")
: "";
const nameLabel =
(duplicateNameCounts.get(file.name) ?? 0) > 1 && file.path.trim().length > 0
? `${file.name} (${file.path})`
: file.name;
lines.push(
`${nameLabel}: ${file.rawChars} raw -> ${file.injectedChars} injected (~${Math.max(0, pct)}% removed${causeText ? `; ${causeText}` : ""}).`,
);
}
if (params.analysis.truncatedFiles.length > topFiles.length) {
lines.push(
`+${params.analysis.truncatedFiles.length - topFiles.length} more truncated file(s).`,
);
}
lines.push(
"If unintentional, raise agents.defaults.bootstrapMaxChars and/or agents.defaults.bootstrapTotalMaxChars.",
);
return lines;
}
export function buildBootstrapPromptWarning(params: {
analysis: BootstrapBudgetAnalysis;
mode: BootstrapPromptWarningMode;
previousSignature?: string;
seenSignatures?: string[];
maxFiles?: number;
}): BootstrapPromptWarning {
const signature = buildBootstrapTruncationSignature(params.analysis);
let seenSignatures = normalizeSeenSignatures(params.seenSignatures);
if (params.previousSignature && !seenSignatures.includes(params.previousSignature)) {
seenSignatures = appendSeenSignature(seenSignatures, params.previousSignature);
}
const hasSeenSignature = Boolean(signature && seenSignatures.includes(signature));
const warningShown =
params.mode !== "off" && Boolean(signature) && (params.mode === "always" || !hasSeenSignature);
const warningSignaturesSeen =
signature && params.mode !== "off"
? appendSeenSignature(seenSignatures, signature)
: seenSignatures;
return {
signature,
warningShown,
lines: warningShown
? formatBootstrapTruncationWarningLines({
analysis: params.analysis,
maxFiles: params.maxFiles,
})
: [],
warningSignaturesSeen,
};
}
export function buildBootstrapTruncationReportMeta(params: {
analysis: BootstrapBudgetAnalysis;
warningMode: BootstrapPromptWarningMode;
warning: BootstrapPromptWarning;
}): BootstrapTruncationReportMeta {
return {
warningMode: params.warningMode,
warningShown: params.warning.warningShown,
promptWarningSignature: params.warning.signature,
...(params.warning.warningSignaturesSeen.length > 0
? { warningSignaturesSeen: params.warning.warningSignaturesSeen }
: {}),
truncatedFiles: params.analysis.truncatedFiles.length,
nearLimitFiles: params.analysis.nearLimitFiles.length,
totalNearLimit: params.analysis.totalNearLimit,
};
}