mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-27 09:21:35 +07:00
74 lines
2.4 KiB
TypeScript
74 lines
2.4 KiB
TypeScript
import fs from "node:fs";
|
|
import path from "node:path";
|
|
import { collectFilesSync, relativeToCwd } from "./check-file-utils.js";
|
|
|
|
const FORBIDDEN_PATTERNS: Array<{ pattern: RegExp; hint: string }> = [
|
|
{
|
|
pattern: /["']openclaw\/plugin-sdk["']/,
|
|
hint: "Use openclaw/plugin-sdk/<subpath> instead of the monolithic root entry.",
|
|
},
|
|
{
|
|
pattern: /["']openclaw\/plugin-sdk\/test-utils["']/,
|
|
hint: "Use openclaw/plugin-sdk/testing for the public extension test surface.",
|
|
},
|
|
{
|
|
pattern: /["']openclaw\/plugin-sdk\/compat["']/,
|
|
hint: "Use a focused public plugin-sdk subpath instead of compat.",
|
|
},
|
|
{
|
|
pattern: /["'](?:\.\.\/)+(?:test-utils\/)[^"']+["']/,
|
|
hint: "Use test/helpers/extensions/* for repo-only bundled extension test helpers.",
|
|
},
|
|
{
|
|
pattern: /["'](?:\.\.\/)+(?:src\/test-utils\/)[^"']+["']/,
|
|
hint: "Use test/helpers/extensions/* for repo-only helpers, or openclaw/plugin-sdk/testing for public surfaces.",
|
|
},
|
|
{
|
|
pattern: /["'](?:\.\.\/)+(?:src\/plugins\/types\.js)["']/,
|
|
hint: "Use public plugin-sdk/core types or test/helpers/extensions/* instead.",
|
|
},
|
|
];
|
|
|
|
function isExtensionTestFile(filePath: string): boolean {
|
|
return /\.test\.[cm]?[jt]sx?$/u.test(filePath) || /\.e2e\.test\.[cm]?[jt]sx?$/u.test(filePath);
|
|
}
|
|
|
|
function collectExtensionTestFiles(rootDir: string): string[] {
|
|
return collectFilesSync(rootDir, {
|
|
includeFile: (filePath) => isExtensionTestFile(filePath),
|
|
});
|
|
}
|
|
|
|
function main() {
|
|
const extensionsDir = path.join(process.cwd(), "extensions");
|
|
const files = collectExtensionTestFiles(extensionsDir);
|
|
const offenders: Array<{ file: string; hint: string }> = [];
|
|
|
|
for (const file of files) {
|
|
const content = fs.readFileSync(file, "utf8");
|
|
for (const rule of FORBIDDEN_PATTERNS) {
|
|
if (!rule.pattern.test(content)) {
|
|
continue;
|
|
}
|
|
offenders.push({ file, hint: rule.hint });
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (offenders.length > 0) {
|
|
console.error(
|
|
"Extension test files must stay on extension test bridges or public plugin-sdk surfaces.",
|
|
);
|
|
for (const offender of offenders.toSorted((a, b) => a.file.localeCompare(b.file))) {
|
|
console.error(`- ${relativeToCwd(offender.file)}: ${offender.hint}`);
|
|
}
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log(
|
|
`OK: extension test files avoid direct core test/internal imports (${files.length} checked).`,
|
|
);
|
|
}
|
|
|
|
main();
|