#!/usr/bin/env -S node --import tsx import { execSync } from "node:child_process"; type PackFile = { path: string }; type PackResult = { files?: PackFile[] }; const requiredPathGroups = [["dist/entry.js", "dist/entry.mjs"], "dist/build-info.json"]; const forbiddenPrefixes = ["dist/OpenClaw.app/"]; 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 main() { 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();