mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-12 07:20:45 +00:00
206 lines
5.9 KiB
JavaScript
Executable File
206 lines
5.9 KiB
JavaScript
Executable File
#!/usr/bin/env -S node --import tsx
|
|
|
|
import { execSync } from "node:child_process";
|
|
import { readdirSync, readFileSync } from "node:fs";
|
|
import { join, resolve } from "node:path";
|
|
|
|
type PackFile = { path: string };
|
|
type PackResult = { files?: PackFile[] };
|
|
|
|
const requiredPathGroups = [
|
|
["dist/index.js", "dist/index.mjs"],
|
|
["dist/entry.js", "dist/entry.mjs"],
|
|
"dist/plugin-sdk/index.js",
|
|
"dist/plugin-sdk/index.d.ts",
|
|
"dist/build-info.json",
|
|
];
|
|
const forbiddenPrefixes = ["dist/OpenClaw.app/"];
|
|
const appcastPath = resolve("appcast.xml");
|
|
|
|
type PackageJson = {
|
|
name?: string;
|
|
version?: string;
|
|
};
|
|
|
|
function normalizePluginSyncVersion(version: string): string {
|
|
const normalized = version.trim().replace(/^v/, "");
|
|
const base = /^([0-9]+\.[0-9]+\.[0-9]+)/.exec(normalized)?.[1];
|
|
if (base) {
|
|
return base;
|
|
}
|
|
return normalized.replace(/[-+].*$/, "");
|
|
}
|
|
|
|
function runPackDry(): PackResult[] {
|
|
const raw = execSync("npm pack --dry-run --json --ignore-scripts", {
|
|
encoding: "utf8",
|
|
stdio: ["ignore", "pipe", "pipe"],
|
|
maxBuffer: 1024 * 1024 * 100,
|
|
});
|
|
return JSON.parse(raw) as PackResult[];
|
|
}
|
|
|
|
function checkPluginVersions() {
|
|
const rootPackagePath = resolve("package.json");
|
|
const rootPackage = JSON.parse(readFileSync(rootPackagePath, "utf8")) as PackageJson;
|
|
const targetVersion = rootPackage.version;
|
|
const targetBaseVersion = targetVersion ? normalizePluginSyncVersion(targetVersion) : null;
|
|
|
|
if (!targetVersion || !targetBaseVersion) {
|
|
console.error("release-check: root package.json missing version.");
|
|
process.exit(1);
|
|
}
|
|
|
|
const extensionsDir = resolve("extensions");
|
|
const entries = readdirSync(extensionsDir, { withFileTypes: true }).filter((entry) =>
|
|
entry.isDirectory(),
|
|
);
|
|
|
|
const mismatches: string[] = [];
|
|
|
|
for (const entry of entries) {
|
|
const packagePath = join(extensionsDir, entry.name, "package.json");
|
|
let pkg: PackageJson;
|
|
try {
|
|
pkg = JSON.parse(readFileSync(packagePath, "utf8")) as PackageJson;
|
|
} catch {
|
|
continue;
|
|
}
|
|
|
|
if (!pkg.name || !pkg.version) {
|
|
continue;
|
|
}
|
|
|
|
if (normalizePluginSyncVersion(pkg.version) !== targetBaseVersion) {
|
|
mismatches.push(`${pkg.name} (${pkg.version})`);
|
|
}
|
|
}
|
|
|
|
if (mismatches.length > 0) {
|
|
console.error(
|
|
`release-check: plugin versions must match release base ${targetBaseVersion} (root ${targetVersion}):`,
|
|
);
|
|
for (const item of mismatches) {
|
|
console.error(` - ${item}`);
|
|
}
|
|
console.error("release-check: run `pnpm plugins:sync` to align plugin versions.");
|
|
process.exit(1);
|
|
}
|
|
}
|
|
|
|
function canonicalSparkleVersionFromShortVersion(shortVersion: string): number | null {
|
|
const match = /^([0-9]{4})\.([0-9]{1,2})\.([0-9]{1,2})([.-].*)?$/.exec(shortVersion.trim());
|
|
if (!match) {
|
|
return null;
|
|
}
|
|
const year = Number(match[1]);
|
|
const month = Number(match[2]);
|
|
const day = Number(match[3]);
|
|
if (
|
|
!Number.isInteger(year) ||
|
|
!Number.isInteger(month) ||
|
|
!Number.isInteger(day) ||
|
|
month < 1 ||
|
|
month > 12 ||
|
|
day < 1 ||
|
|
day > 31
|
|
) {
|
|
return null;
|
|
}
|
|
return Number(`${year}${String(month).padStart(2, "0")}${String(day).padStart(2, "0")}0`);
|
|
}
|
|
|
|
function extractTag(item: string, tag: string): string | null {
|
|
const escapedTag = tag.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
const regex = new RegExp(`<${escapedTag}>([^<]+)</${escapedTag}>`);
|
|
return regex.exec(item)?.[1]?.trim() ?? null;
|
|
}
|
|
|
|
function checkAppcastSparkleVersions() {
|
|
const xml = readFileSync(appcastPath, "utf8");
|
|
const itemMatches = [...xml.matchAll(/<item>([\s\S]*?)<\/item>/g)];
|
|
const errors: string[] = [];
|
|
|
|
if (itemMatches.length === 0) {
|
|
errors.push("appcast.xml contains no <item> entries.");
|
|
}
|
|
|
|
for (const [, item] of itemMatches) {
|
|
const title = extractTag(item, "title") ?? "unknown";
|
|
const shortVersion = extractTag(item, "sparkle:shortVersionString");
|
|
const sparkleVersion = extractTag(item, "sparkle:version");
|
|
|
|
if (!sparkleVersion) {
|
|
errors.push(`appcast item '${title}' is missing sparkle:version.`);
|
|
continue;
|
|
}
|
|
if (!/^[0-9]+$/.test(sparkleVersion)) {
|
|
errors.push(`appcast item '${title}' has non-numeric sparkle:version '${sparkleVersion}'.`);
|
|
continue;
|
|
}
|
|
|
|
if (!shortVersion) {
|
|
continue;
|
|
}
|
|
const canonicalFloor = canonicalSparkleVersionFromShortVersion(shortVersion);
|
|
if (canonicalFloor === null) {
|
|
continue;
|
|
}
|
|
const sparkleBuild = Number(sparkleVersion);
|
|
if (sparkleBuild < canonicalFloor) {
|
|
errors.push(
|
|
`appcast item '${title}' has sparkle:version ${sparkleBuild} below canonical floor ${canonicalFloor} derived from ${shortVersion}.`,
|
|
);
|
|
}
|
|
}
|
|
|
|
if (errors.length > 0) {
|
|
console.error("release-check: appcast sparkle version validation failed:");
|
|
for (const error of errors) {
|
|
console.error(` - ${error}`);
|
|
}
|
|
process.exit(1);
|
|
}
|
|
}
|
|
|
|
function main() {
|
|
checkPluginVersions();
|
|
checkAppcastSparkleVersions();
|
|
|
|
const results = runPackDry();
|
|
const files = results.flatMap((entry) => entry.files ?? []);
|
|
const paths = new Set(files.map((file) => file.path));
|
|
|
|
const missing = requiredPathGroups
|
|
.flatMap((group) => {
|
|
if (Array.isArray(group)) {
|
|
return group.some((path) => paths.has(path)) ? [] : [group.join(" or ")];
|
|
}
|
|
return paths.has(group) ? [] : [group];
|
|
})
|
|
.toSorted();
|
|
const forbidden = [...paths].filter((path) =>
|
|
forbiddenPrefixes.some((prefix) => path.startsWith(prefix)),
|
|
);
|
|
|
|
if (missing.length > 0 || forbidden.length > 0) {
|
|
if (missing.length > 0) {
|
|
console.error("release-check: missing files in npm pack:");
|
|
for (const path of missing) {
|
|
console.error(` - ${path}`);
|
|
}
|
|
}
|
|
if (forbidden.length > 0) {
|
|
console.error("release-check: forbidden files in npm pack:");
|
|
for (const path of forbidden) {
|
|
console.error(` - ${path}`);
|
|
}
|
|
}
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log("release-check: npm pack contents look OK.");
|
|
}
|
|
|
|
main();
|