mirror of
https://fastgit.cc/github.com/openclaw/openclaw
synced 2026-04-21 05:12:57 +08:00
463 lines
14 KiB
JavaScript
463 lines
14 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
import { spawnSync } from "node:child_process";
|
|
import fs from "node:fs";
|
|
import path from "node:path";
|
|
import {
|
|
acquireLocalHeavyCheckLockSync,
|
|
applyLocalTsgoPolicy,
|
|
shouldAcquireLocalHeavyCheckLockForTsgo,
|
|
} from "./lib/local-heavy-check-runtime.mjs";
|
|
|
|
const repoRoot = path.resolve(import.meta.dirname, "..");
|
|
const artifactRoot = path.resolve(repoRoot, ".artifacts/tsgo-profile");
|
|
const tsgoPath = path.resolve(repoRoot, "node_modules", ".bin", "tsgo");
|
|
|
|
const GRAPH_DEFINITIONS = {
|
|
core: {
|
|
config: "tsconfig.core.json",
|
|
description: "core production graph",
|
|
},
|
|
"core-test": {
|
|
config: "tsconfig.core.test.json",
|
|
description: "core colocated test graph",
|
|
},
|
|
"core-test-agents": {
|
|
config: "tsconfig.core.test.agents.json",
|
|
description: "diagnostic slice: core agent colocated tests",
|
|
},
|
|
"core-test-non-agents": {
|
|
config: "tsconfig.core.test.non-agents.json",
|
|
description: "diagnostic slice: core tests excluding agent test roots",
|
|
},
|
|
extensions: {
|
|
config: "tsconfig.extensions.json",
|
|
description: "bundled extension production graph",
|
|
},
|
|
"extensions-test": {
|
|
config: "tsconfig.extensions.test.json",
|
|
description: "bundled extension colocated test graph",
|
|
},
|
|
};
|
|
|
|
function usage() {
|
|
return [
|
|
"Usage: pnpm tsgo:profile [graph...] [options]",
|
|
"",
|
|
"Graphs:",
|
|
...Object.entries(GRAPH_DEFINITIONS).map(
|
|
([name, graph]) => ` ${name.padEnd(16)} ${graph.description}`,
|
|
),
|
|
"",
|
|
"Options:",
|
|
" --all Profile all graphs",
|
|
" --reuse Reuse profile tsbuildinfo files instead of forcing fresh checks",
|
|
" --deep Also write --generateTrace and --generateCpuProfile artifacts",
|
|
" --explain Also write --explainFiles artifacts",
|
|
" --out=<dir> Output directory (default: .artifacts/tsgo-profile)",
|
|
" --json Print JSON report to stdout",
|
|
" --help Show this help",
|
|
"",
|
|
"Default graphs: core-test extensions-test",
|
|
].join("\n");
|
|
}
|
|
|
|
function parseArgs(argv) {
|
|
const graphNames = [];
|
|
const options = {
|
|
all: false,
|
|
deep: false,
|
|
explain: false,
|
|
json: false,
|
|
reuse: false,
|
|
outDir: artifactRoot,
|
|
};
|
|
|
|
for (const arg of argv) {
|
|
if (arg === "--help" || arg === "-h") {
|
|
throw new Error(usage());
|
|
}
|
|
if (arg === "--all") {
|
|
options.all = true;
|
|
continue;
|
|
}
|
|
if (arg === "--deep") {
|
|
options.deep = true;
|
|
continue;
|
|
}
|
|
if (arg === "--explain") {
|
|
options.explain = true;
|
|
continue;
|
|
}
|
|
if (arg === "--json") {
|
|
options.json = true;
|
|
continue;
|
|
}
|
|
if (arg === "--reuse") {
|
|
options.reuse = true;
|
|
continue;
|
|
}
|
|
if (arg.startsWith("--out=")) {
|
|
options.outDir = path.resolve(repoRoot, arg.slice("--out=".length));
|
|
continue;
|
|
}
|
|
if (!GRAPH_DEFINITIONS[arg]) {
|
|
throw new Error(`Unknown graph: ${arg}\n\n${usage()}`);
|
|
}
|
|
graphNames.push(arg);
|
|
}
|
|
|
|
const selectedGraphs = options.all
|
|
? Object.keys(GRAPH_DEFINITIONS)
|
|
: graphNames.length > 0
|
|
? graphNames
|
|
: ["core-test", "extensions-test"];
|
|
|
|
return { options, selectedGraphs };
|
|
}
|
|
|
|
function ensureDirs(outDir) {
|
|
fs.mkdirSync(outDir, { recursive: true });
|
|
fs.mkdirSync(path.join(outDir, "cache"), { recursive: true });
|
|
}
|
|
|
|
function removeIfFreshMode(filePath, reuse) {
|
|
if (!reuse) {
|
|
fs.rmSync(filePath, { force: true });
|
|
}
|
|
}
|
|
|
|
function runTsgo(label, args, params = {}) {
|
|
const { args: finalArgs, env } = applyLocalTsgoPolicy(args, process.env);
|
|
const releaseLock = shouldAcquireLocalHeavyCheckLockForTsgo(finalArgs, env)
|
|
? acquireLocalHeavyCheckLockSync({
|
|
cwd: repoRoot,
|
|
env,
|
|
toolName: "tsgo-profile",
|
|
})
|
|
: () => {};
|
|
|
|
const startedAt = Date.now();
|
|
try {
|
|
const result = spawnSync(tsgoPath, finalArgs, {
|
|
cwd: repoRoot,
|
|
env,
|
|
encoding: "utf8",
|
|
maxBuffer: params.maxBuffer ?? 128 * 1024 * 1024,
|
|
shell: process.platform === "win32",
|
|
});
|
|
const elapsedMs = Date.now() - startedAt;
|
|
const stdout = result.stdout ?? "";
|
|
const stderr = result.stderr ?? "";
|
|
if (result.error) {
|
|
throw result.error;
|
|
}
|
|
if ((result.status ?? 1) !== 0) {
|
|
const output = [stdout, stderr].filter(Boolean).join("\n");
|
|
throw new Error(`${label} failed with exit code ${result.status ?? 1}\n${output}`);
|
|
}
|
|
return { elapsedMs, stdout, stderr };
|
|
} finally {
|
|
releaseLock();
|
|
}
|
|
}
|
|
|
|
function parseDiagnostics(output) {
|
|
const diagnostics = {};
|
|
for (const line of output.split(/\r?\n/u)) {
|
|
const match = /^(.+?):\s+([0-9.]+)(K|s)?\s*$/u.exec(line.trim());
|
|
if (!match) {
|
|
continue;
|
|
}
|
|
const [, rawKey, rawValue, unit] = match;
|
|
const key = rawKey.trim().replaceAll(/\s+/gu, " ");
|
|
const value = Number(rawValue);
|
|
diagnostics[key] = unit === "K" ? value * 1024 : value;
|
|
}
|
|
return diagnostics;
|
|
}
|
|
|
|
function normalizeFilePath(filePath) {
|
|
const normalized = filePath.trim().replaceAll("\\", "/");
|
|
const normalizedRoot = repoRoot.replaceAll("\\", "/");
|
|
if (normalized.startsWith(`${normalizedRoot}/`)) {
|
|
return normalized.slice(normalizedRoot.length + 1);
|
|
}
|
|
return normalized;
|
|
}
|
|
|
|
function packageNameFromNodeModule(parts, startIndex) {
|
|
const first = parts[startIndex + 1];
|
|
if (!first) {
|
|
return "node_modules";
|
|
}
|
|
if (first.startsWith("@")) {
|
|
return `${first}/${parts[startIndex + 2] ?? ""}`.replace(/\/$/u, "");
|
|
}
|
|
return first;
|
|
}
|
|
|
|
function classifyFile(relativePath) {
|
|
const parts = relativePath.split("/");
|
|
const first = parts[0];
|
|
if (relativePath.includes("/node_modules/") || first === "node_modules") {
|
|
const nodeModulesIndex = parts.indexOf("node_modules");
|
|
return `node_modules/${packageNameFromNodeModule(parts, nodeModulesIndex)}`;
|
|
}
|
|
if (first === "extensions") {
|
|
return `extensions/${parts[1] ?? "(root)"}`;
|
|
}
|
|
if (first === "packages") {
|
|
return `packages/${parts[1] ?? "(root)"}`;
|
|
}
|
|
if (first === "src") {
|
|
return `src/${parts[1] ?? "(root)"}`;
|
|
}
|
|
if (first === "ui") {
|
|
return `ui/${parts[1] ?? "(root)"}`;
|
|
}
|
|
if (first === "test") {
|
|
return `test/${parts[1] ?? "(root)"}`;
|
|
}
|
|
if (first.startsWith("/") || /^[A-Za-z]:/u.test(first)) {
|
|
return "(external)";
|
|
}
|
|
return first || "(unknown)";
|
|
}
|
|
|
|
function countBy(values, keyFn) {
|
|
const counts = new Map();
|
|
for (const value of values) {
|
|
const key = keyFn(value);
|
|
counts.set(key, (counts.get(key) ?? 0) + 1);
|
|
}
|
|
return [...counts.entries()]
|
|
.map(([key, count]) => ({ key, count }))
|
|
.toSorted((left, right) => right.count - left.count || left.key.localeCompare(right.key));
|
|
}
|
|
|
|
function summarizeFiles(stdout) {
|
|
const files = stdout
|
|
.split(/\r?\n/u)
|
|
.map(normalizeFilePath)
|
|
.filter(Boolean)
|
|
.filter((line) => !line.startsWith("Files:"));
|
|
|
|
const projectRelativeFiles = files.filter(
|
|
(file) => !path.isAbsolute(file) && !/^[A-Za-z]:/u.test(file),
|
|
);
|
|
const testFiles = projectRelativeFiles.filter((file) => /\.test\.[cm]?[tj]sx?$/u.test(file));
|
|
return {
|
|
totalFiles: files.length,
|
|
projectRelativeFiles: projectRelativeFiles.length,
|
|
testFiles: testFiles.length,
|
|
groups: countBy(projectRelativeFiles, classifyFile).slice(0, 40),
|
|
};
|
|
}
|
|
|
|
function diffDiagnostics(check, noCheck) {
|
|
const totalDelta = (check["Total time"] ?? 0) - (noCheck["Total time"] ?? 0);
|
|
const checkTime = check["Check time"] ?? 0;
|
|
return {
|
|
checkTimeSeconds: checkTime,
|
|
totalDeltaSeconds: totalDelta,
|
|
typeShareOfTotal:
|
|
check["Total time"] && checkTime ? Number((checkTime / check["Total time"]).toFixed(3)) : 0,
|
|
};
|
|
}
|
|
|
|
function formatSeconds(value) {
|
|
return `${value.toFixed(2)}s`;
|
|
}
|
|
|
|
function renderTextReport(report) {
|
|
const lines = [
|
|
"# tsgo profile",
|
|
"",
|
|
`Generated: ${report.generatedAt}`,
|
|
`Fresh profile caches: ${report.options.reuse ? "no" : "yes"}`,
|
|
"",
|
|
];
|
|
|
|
for (const graph of report.graphs) {
|
|
const check = graph.check.diagnostics;
|
|
const noCheck = graph.noCheck.diagnostics;
|
|
lines.push(`## ${graph.name}`);
|
|
lines.push(`Config: ${graph.config}`);
|
|
lines.push(
|
|
`Check: wall ${formatSeconds(graph.check.elapsedMs / 1000)}, compiler total ${formatSeconds(
|
|
check["Total time"] ?? 0,
|
|
)}, check ${formatSeconds(check["Check time"] ?? 0)}, memory ${Math.round(
|
|
(check["Memory used"] ?? 0) / 1024 / 1024,
|
|
)} MiB`,
|
|
);
|
|
lines.push(
|
|
`NoCheck: wall ${formatSeconds(
|
|
graph.noCheck.elapsedMs / 1000,
|
|
)}, compiler total ${formatSeconds(noCheck["Total time"] ?? 0)}`,
|
|
);
|
|
lines.push(
|
|
`Files: compiler ${check.Files ?? "?"}, listed ${graph.files.totalFiles}, project-relative ${graph.files.projectRelativeFiles}, tests ${graph.files.testFiles}`,
|
|
);
|
|
lines.push(`File list: ${graph.files.artifact}`);
|
|
lines.push(
|
|
`Type cost: check ${formatSeconds(graph.typeCost.checkTimeSeconds)}, total delta ${formatSeconds(
|
|
graph.typeCost.totalDeltaSeconds,
|
|
)}, share ${(graph.typeCost.typeShareOfTotal * 100).toFixed(1)}%`,
|
|
);
|
|
lines.push("Top file groups:");
|
|
for (const group of graph.files.groups.slice(0, 15)) {
|
|
lines.push(`- ${group.key}: ${group.count}`);
|
|
}
|
|
if (graph.deep) {
|
|
lines.push(`Deep artifacts: ${graph.deep.traceDir}, ${graph.deep.cpuProfile}`);
|
|
}
|
|
if (graph.explain) {
|
|
lines.push(`Explain: ${graph.explain.artifact}`);
|
|
}
|
|
lines.push("");
|
|
}
|
|
|
|
lines.push(`JSON: ${report.paths.json}`);
|
|
lines.push("");
|
|
return `${lines.join("\n")}\n`;
|
|
}
|
|
|
|
function profileGraph(name, options) {
|
|
const graph = GRAPH_DEFINITIONS[name];
|
|
const outDir = options.outDir;
|
|
const graphCacheRoot = path.join(outDir, "cache");
|
|
const checkBuildInfo = path.join(graphCacheRoot, `${name}-check.tsbuildinfo`);
|
|
const noCheckBuildInfo = path.join(graphCacheRoot, `${name}-nocheck.tsbuildinfo`);
|
|
const configPath = graph.config;
|
|
|
|
removeIfFreshMode(checkBuildInfo, options.reuse);
|
|
removeIfFreshMode(noCheckBuildInfo, options.reuse);
|
|
|
|
const baseArgs = ["-p", configPath, "--pretty", "false"];
|
|
const listFiles = runTsgo(`${name}:listFilesOnly`, [...baseArgs, "--listFilesOnly"], {
|
|
maxBuffer: 256 * 1024 * 1024,
|
|
});
|
|
const filesArtifact = path.join(outDir, `${name}.files.txt`);
|
|
fs.writeFileSync(filesArtifact, listFiles.stdout);
|
|
const noCheck = runTsgo(`${name}:noCheck`, [
|
|
...baseArgs,
|
|
"--noCheck",
|
|
"--incremental",
|
|
"--tsBuildInfoFile",
|
|
noCheckBuildInfo,
|
|
"--extendedDiagnostics",
|
|
]);
|
|
|
|
const checkArgs = [
|
|
...baseArgs,
|
|
"--incremental",
|
|
"--tsBuildInfoFile",
|
|
checkBuildInfo,
|
|
"--extendedDiagnostics",
|
|
];
|
|
let deep;
|
|
if (options.deep) {
|
|
const traceDir = path.join(outDir, `${name}-trace`);
|
|
const cpuProfile = path.join(outDir, `${name}.cpuprofile`);
|
|
fs.rmSync(traceDir, { force: true, recursive: true });
|
|
fs.rmSync(cpuProfile, { force: true });
|
|
checkArgs.push("--generateTrace", traceDir, "--generateCpuProfile", cpuProfile);
|
|
deep = {
|
|
traceDir: path.relative(repoRoot, traceDir),
|
|
cpuProfile: path.relative(repoRoot, cpuProfile),
|
|
};
|
|
}
|
|
const check = runTsgo(`${name}:check`, checkArgs);
|
|
let explain;
|
|
if (options.explain) {
|
|
const explainArtifact = path.join(outDir, `${name}.explain.txt`);
|
|
const explainResult = runTsgo(`${name}:explainFiles`, [...baseArgs, "--explainFiles"], {
|
|
maxBuffer: 256 * 1024 * 1024,
|
|
});
|
|
fs.writeFileSync(explainArtifact, `${explainResult.stdout}${explainResult.stderr}`);
|
|
explain = {
|
|
artifact: path.relative(repoRoot, explainArtifact),
|
|
elapsedMs: explainResult.elapsedMs,
|
|
};
|
|
}
|
|
|
|
const checkDiagnostics = parseDiagnostics(`${check.stdout}\n${check.stderr}`);
|
|
const noCheckDiagnostics = parseDiagnostics(`${noCheck.stdout}\n${noCheck.stderr}`);
|
|
return {
|
|
name,
|
|
config: configPath,
|
|
description: graph.description,
|
|
files: {
|
|
...summarizeFiles(listFiles.stdout),
|
|
artifact: path.relative(repoRoot, filesArtifact),
|
|
},
|
|
noCheck: {
|
|
elapsedMs: noCheck.elapsedMs,
|
|
diagnostics: noCheckDiagnostics,
|
|
},
|
|
check: {
|
|
elapsedMs: check.elapsedMs,
|
|
diagnostics: checkDiagnostics,
|
|
},
|
|
typeCost: diffDiagnostics(checkDiagnostics, noCheckDiagnostics),
|
|
...(deep ? { deep } : {}),
|
|
...(explain ? { explain } : {}),
|
|
};
|
|
}
|
|
|
|
async function main(argv) {
|
|
const { options, selectedGraphs } = parseArgs(argv);
|
|
ensureDirs(options.outDir);
|
|
const report = {
|
|
generatedAt: new Date().toISOString(),
|
|
options: {
|
|
graphs: selectedGraphs,
|
|
deep: options.deep,
|
|
explain: options.explain,
|
|
reuse: options.reuse,
|
|
},
|
|
graphs: [],
|
|
paths: {},
|
|
};
|
|
|
|
for (const graphName of selectedGraphs) {
|
|
process.stderr.write(`[tsgo-profile] profiling ${graphName}\n`);
|
|
report.graphs.push(profileGraph(graphName, options));
|
|
}
|
|
|
|
const timestamp = new Date()
|
|
.toISOString()
|
|
.replaceAll(":", "")
|
|
.replaceAll(".", "")
|
|
.replace("T", "-")
|
|
.replace("Z", "");
|
|
const jsonPath = path.join(options.outDir, `tsgo-profile-${timestamp}.json`);
|
|
const textPath = path.join(options.outDir, `tsgo-profile-${timestamp}.md`);
|
|
report.paths = {
|
|
json: path.relative(repoRoot, jsonPath),
|
|
text: path.relative(repoRoot, textPath),
|
|
};
|
|
|
|
fs.writeFileSync(jsonPath, `${JSON.stringify(report, null, 2)}\n`);
|
|
fs.writeFileSync(textPath, renderTextReport(report));
|
|
fs.writeFileSync(
|
|
path.join(options.outDir, "latest.json"),
|
|
`${JSON.stringify(report, null, 2)}\n`,
|
|
);
|
|
fs.writeFileSync(path.join(options.outDir, "latest.md"), renderTextReport(report));
|
|
|
|
if (options.json) {
|
|
process.stdout.write(`${JSON.stringify(report, null, 2)}\n`);
|
|
} else {
|
|
process.stdout.write(renderTextReport(report));
|
|
}
|
|
}
|
|
|
|
try {
|
|
await main(process.argv.slice(2));
|
|
} catch (error) {
|
|
process.stderr.write(`${error instanceof Error ? error.message : String(error)}\n`);
|
|
process.exit(1);
|
|
}
|