openclaw/src/commands/providers/logs.ts

107 lines
3.2 KiB
TypeScript

import fs from "node:fs/promises";
import { parseLogLine } from "../../logging/parse-log-line.js";
import { getResolvedLoggerSettings } from "../../logging.js";
import { defaultRuntime, type RuntimeEnv } from "../../runtime.js";
import { theme } from "../../terminal/theme.js";
export type ProvidersLogsOptions = {
provider?: string;
lines?: string | number;
json?: boolean;
};
type LogLine = ReturnType<typeof parseLogLine>;
const DEFAULT_LIMIT = 200;
const MAX_BYTES = 1_000_000;
const PROVIDERS = new Set([
"whatsapp",
"telegram",
"discord",
"slack",
"signal",
"imessage",
"all",
]);
function parseProviderFilter(raw?: string) {
const trimmed = raw?.trim().toLowerCase();
if (!trimmed) return "all";
return PROVIDERS.has(trimmed) ? trimmed : "all";
}
function matchesProvider(line: NonNullable<LogLine>, provider: string) {
if (provider === "all") return true;
const needle = `gateway/providers/${provider}`;
if (line.subsystem?.includes(needle)) return true;
if (line.module?.includes(provider)) return true;
return false;
}
async function readTailLines(file: string, limit: number): Promise<string[]> {
const stat = await fs.stat(file).catch(() => null);
if (!stat) return [];
const size = stat.size;
const start = Math.max(0, size - MAX_BYTES);
const handle = await fs.open(file, "r");
try {
const length = Math.max(0, size - start);
if (length === 0) return [];
const buffer = Buffer.alloc(length);
const readResult = await handle.read(buffer, 0, length, start);
const text = buffer.toString("utf8", 0, readResult.bytesRead);
let lines = text.split("\n");
if (start > 0) lines = lines.slice(1);
if (lines.length && lines[lines.length - 1] === "") {
lines = lines.slice(0, -1);
}
if (lines.length > limit) {
lines = lines.slice(lines.length - limit);
}
return lines;
} finally {
await handle.close();
}
}
export async function providersLogsCommand(
opts: ProvidersLogsOptions,
runtime: RuntimeEnv = defaultRuntime,
) {
const provider = parseProviderFilter(opts.provider);
const limitRaw =
typeof opts.lines === "string" ? Number(opts.lines) : opts.lines;
const limit =
typeof limitRaw === "number" && Number.isFinite(limitRaw) && limitRaw > 0
? Math.floor(limitRaw)
: DEFAULT_LIMIT;
const file = getResolvedLoggerSettings().file;
const rawLines = await readTailLines(file, limit * 4);
const parsed = rawLines
.map(parseLogLine)
.filter((line): line is NonNullable<LogLine> => Boolean(line));
const filtered = parsed.filter((line) => matchesProvider(line, provider));
const lines = filtered.slice(Math.max(0, filtered.length - limit));
if (opts.json) {
runtime.log(JSON.stringify({ file, provider, lines }, null, 2));
return;
}
runtime.log(theme.info(`Log file: ${file}`));
if (provider !== "all") {
runtime.log(theme.info(`Provider: ${provider}`));
}
if (lines.length === 0) {
runtime.log(theme.muted("No matching log lines."));
return;
}
for (const line of lines) {
const ts = line.time ? `${line.time} ` : "";
const level = line.level ? `${line.level.toLowerCase()} ` : "";
runtime.log(`${ts}${level}${line.message}`.trim());
}
}