mirror of https://github.com/openclaw/openclaw.git
172 lines
5.0 KiB
JavaScript
172 lines
5.0 KiB
JavaScript
#!/usr/bin/env node
|
|
import path from "node:path";
|
|
import { analyzeTopology } from "./lib/ts-topology/analyze.js";
|
|
import { renderTextReport } from "./lib/ts-topology/reports.js";
|
|
import {
|
|
createFilesystemPublicSurfaceScope,
|
|
createPluginSdkScope,
|
|
} from "./lib/ts-topology/scope.js";
|
|
import type { TopologyReportName, TopologyScope } from "./lib/ts-topology/types.js";
|
|
|
|
const VALID_REPORTS = new Set<TopologyReportName>([
|
|
"public-surface-usage",
|
|
"owner-map",
|
|
"single-owner-shared",
|
|
"unused-public-surface",
|
|
"consumer-topology",
|
|
]);
|
|
|
|
type IoLike = {
|
|
stdout: { write: (chunk: string) => void };
|
|
stderr: { write: (chunk: string) => void };
|
|
};
|
|
|
|
type CliOptions = {
|
|
repoRoot: string;
|
|
scopeId: string;
|
|
report: TopologyReportName;
|
|
json: boolean;
|
|
includeTests: boolean;
|
|
limit: number;
|
|
tsconfigName?: string;
|
|
customEntrypointRoot?: string;
|
|
customImportPrefix?: string;
|
|
};
|
|
|
|
function usage() {
|
|
return [
|
|
"Usage: ts-topology [analyze] [options]",
|
|
"",
|
|
"Options:",
|
|
" --scope=<plugin-sdk|custom> Built-in or custom scope",
|
|
" --entrypoint-root=<path> Required for --scope=custom",
|
|
" --import-prefix=<specifier> Required for --scope=custom",
|
|
" --report=<name> public-surface-usage | owner-map | single-owner-shared | unused-public-surface | consumer-topology",
|
|
" --json Emit JSON",
|
|
" --limit=<n> Limit ranked/text output (default: 25)",
|
|
" --exclude-tests Ignore test consumers",
|
|
" --repo-root=<path> Override repo root",
|
|
" --tsconfig=<name> Override tsconfig filename",
|
|
].join("\n");
|
|
}
|
|
|
|
function parseArgs(argv: string[]): CliOptions {
|
|
const args = [...argv];
|
|
if (args[0] === "analyze") {
|
|
args.shift();
|
|
}
|
|
const options: CliOptions = {
|
|
repoRoot: process.cwd(),
|
|
scopeId: "plugin-sdk",
|
|
report: "public-surface-usage",
|
|
json: false,
|
|
includeTests: true,
|
|
limit: 25,
|
|
};
|
|
|
|
for (const arg of args) {
|
|
if (arg === "--json") {
|
|
options.json = true;
|
|
continue;
|
|
}
|
|
if (arg === "--exclude-tests") {
|
|
options.includeTests = false;
|
|
continue;
|
|
}
|
|
if (arg === "--help" || arg === "-h") {
|
|
throw new Error(usage());
|
|
}
|
|
const [flag, value] = arg.split("=", 2);
|
|
switch (flag) {
|
|
case "--scope":
|
|
options.scopeId = value ?? options.scopeId;
|
|
break;
|
|
case "--report":
|
|
options.report = (value as TopologyReportName | undefined) ?? options.report;
|
|
break;
|
|
case "--limit":
|
|
options.limit = Math.max(1, Number.parseInt(value ?? "25", 10));
|
|
break;
|
|
case "--repo-root":
|
|
options.repoRoot = path.resolve(value ?? options.repoRoot);
|
|
break;
|
|
case "--entrypoint-root":
|
|
options.customEntrypointRoot = value;
|
|
break;
|
|
case "--import-prefix":
|
|
options.customImportPrefix = value;
|
|
break;
|
|
case "--tsconfig":
|
|
options.tsconfigName = value;
|
|
break;
|
|
default:
|
|
throw new Error(`Unknown argument: ${arg}\n\n${usage()}`);
|
|
}
|
|
}
|
|
return options;
|
|
}
|
|
|
|
function resolveScope(options: CliOptions): TopologyScope {
|
|
if (options.scopeId === "plugin-sdk") {
|
|
return createPluginSdkScope(options.repoRoot);
|
|
}
|
|
if (options.scopeId === "custom") {
|
|
if (!options.customEntrypointRoot || !options.customImportPrefix) {
|
|
throw new Error("--scope=custom requires --entrypoint-root and --import-prefix");
|
|
}
|
|
return createFilesystemPublicSurfaceScope(options.repoRoot, {
|
|
id: "custom",
|
|
entrypointRoot: options.customEntrypointRoot,
|
|
importPrefix: options.customImportPrefix,
|
|
});
|
|
}
|
|
throw new Error(`Unsupported scope: ${options.scopeId}`);
|
|
}
|
|
|
|
function assertValidReport(report: string): asserts report is TopologyReportName {
|
|
if (!VALID_REPORTS.has(report as TopologyReportName)) {
|
|
throw new Error(
|
|
`Unsupported report: ${report}\nValid reports: ${[...VALID_REPORTS].join(", ")}`,
|
|
);
|
|
}
|
|
}
|
|
|
|
export async function main(argv: string[], io: IoLike = process): Promise<number> {
|
|
let options: CliOptions;
|
|
try {
|
|
options = parseArgs(argv);
|
|
} catch (error) {
|
|
io.stderr.write(`${error instanceof Error ? error.message : String(error)}\n`);
|
|
return 1;
|
|
}
|
|
|
|
try {
|
|
assertValidReport(options.report);
|
|
const scope = resolveScope(options);
|
|
const envelope = analyzeTopology({
|
|
repoRoot: options.repoRoot,
|
|
scope,
|
|
report: options.report,
|
|
includeTests: options.includeTests,
|
|
limit: options.limit,
|
|
tsconfigName: options.tsconfigName,
|
|
});
|
|
if (options.json) {
|
|
io.stdout.write(`${JSON.stringify(envelope, null, 2)}\n`);
|
|
return 0;
|
|
}
|
|
io.stdout.write(`${renderTextReport(envelope, options.limit)}\n`);
|
|
return 0;
|
|
} catch (error) {
|
|
io.stderr.write(`${error instanceof Error ? error.message : String(error)}\n`);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
if (import.meta.url === `file://${process.argv[1]}`) {
|
|
const exitCode = await main(process.argv.slice(2));
|
|
if (exitCode !== 0) {
|
|
process.exit(exitCode);
|
|
}
|
|
}
|