mirror of
https://github.com/badlogic/pi-mono.git
synced 2026-05-02 00:00:20 +00:00
196 lines
5.1 KiB
JavaScript
Executable file
196 lines
5.1 KiB
JavaScript
Executable file
#!/usr/bin/env node
|
|
/**
|
|
* Release script for pi-mono
|
|
*
|
|
* Usage:
|
|
* node scripts/release.mjs <major|minor|patch>
|
|
* node scripts/release.mjs <x.y.z>
|
|
*
|
|
* Steps:
|
|
* 1. Check for uncommitted changes
|
|
* 2. Bump version via npm run version:xxx or set an explicit version
|
|
* 3. Update CHANGELOG.md files: [Unreleased] -> [version] - date
|
|
* 4. Commit and tag
|
|
* 5. Publish to npm
|
|
* 6. Add new [Unreleased] section to changelogs
|
|
* 7. Commit
|
|
*/
|
|
|
|
import { execSync } from "child_process";
|
|
import { readFileSync, writeFileSync, readdirSync, existsSync } from "fs";
|
|
import { join } from "path";
|
|
|
|
const RELEASE_TARGET = process.argv[2];
|
|
const BUMP_TYPES = new Set(["major", "minor", "patch"]);
|
|
const SEMVER_RE = /^\d+\.\d+\.\d+$/;
|
|
|
|
if (!RELEASE_TARGET || (!BUMP_TYPES.has(RELEASE_TARGET) && !SEMVER_RE.test(RELEASE_TARGET))) {
|
|
console.error("Usage: node scripts/release.mjs <major|minor|patch|x.y.z>");
|
|
process.exit(1);
|
|
}
|
|
|
|
function run(cmd, options = {}) {
|
|
console.log(`$ ${cmd}`);
|
|
try {
|
|
return execSync(cmd, { encoding: "utf-8", stdio: options.silent ? "pipe" : "inherit", ...options });
|
|
} catch (e) {
|
|
if (!options.ignoreError) {
|
|
console.error(`Command failed: ${cmd}`);
|
|
process.exit(1);
|
|
}
|
|
return null;
|
|
}
|
|
}
|
|
|
|
function getVersion() {
|
|
const pkg = JSON.parse(readFileSync("packages/ai/package.json", "utf-8"));
|
|
return pkg.version;
|
|
}
|
|
|
|
function compareVersions(a, b) {
|
|
const aParts = a.split(".").map(Number);
|
|
const bParts = b.split(".").map(Number);
|
|
|
|
for (let i = 0; i < 3; i++) {
|
|
const diff = (aParts[i] || 0) - (bParts[i] || 0);
|
|
if (diff !== 0) {
|
|
return diff;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
function shellQuote(value) {
|
|
return `'${value.replace(/'/g, `'\\''`)}'`;
|
|
}
|
|
|
|
function stageChangedFiles() {
|
|
const output = run("git ls-files -m -o -d --exclude-standard", { silent: true });
|
|
const paths = [...new Set((output || "").split("\n").map((line) => line.trim()).filter(Boolean))];
|
|
if (paths.length === 0) {
|
|
return;
|
|
}
|
|
|
|
run(`git add -- ${paths.map(shellQuote).join(" ")}`);
|
|
}
|
|
|
|
function bumpOrSetVersion(target) {
|
|
const currentVersion = getVersion();
|
|
|
|
if (BUMP_TYPES.has(target)) {
|
|
console.log(`Bumping version (${target})...`);
|
|
run(`npm run version:${target}`);
|
|
return getVersion();
|
|
}
|
|
|
|
if (compareVersions(target, currentVersion) <= 0) {
|
|
console.error(`Error: explicit version ${target} must be greater than current version ${currentVersion}.`);
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log(`Setting explicit version (${target})...`);
|
|
run(
|
|
`npm version ${target} -ws --no-git-tag-version && node scripts/sync-versions.js && npx shx rm -rf node_modules packages/*/node_modules package-lock.json && npm install`,
|
|
);
|
|
return getVersion();
|
|
}
|
|
|
|
function getChangelogs() {
|
|
const packagesDir = "packages";
|
|
const packages = readdirSync(packagesDir);
|
|
return packages
|
|
.map((pkg) => join(packagesDir, pkg, "CHANGELOG.md"))
|
|
.filter((path) => existsSync(path));
|
|
}
|
|
|
|
function updateChangelogsForRelease(version) {
|
|
const date = new Date().toISOString().split("T")[0];
|
|
const changelogs = getChangelogs();
|
|
|
|
for (const changelog of changelogs) {
|
|
const content = readFileSync(changelog, "utf-8");
|
|
|
|
if (!content.includes("## [Unreleased]")) {
|
|
console.log(` Skipping ${changelog}: no [Unreleased] section`);
|
|
continue;
|
|
}
|
|
|
|
const updated = content.replace(
|
|
"## [Unreleased]",
|
|
`## [${version}] - ${date}`
|
|
);
|
|
writeFileSync(changelog, updated);
|
|
console.log(` Updated ${changelog}`);
|
|
}
|
|
}
|
|
|
|
function addUnreleasedSection() {
|
|
const changelogs = getChangelogs();
|
|
const unreleasedSection = "## [Unreleased]\n\n";
|
|
|
|
for (const changelog of changelogs) {
|
|
const content = readFileSync(changelog, "utf-8");
|
|
|
|
// Insert after "# Changelog\n\n"
|
|
const updated = content.replace(
|
|
/^(# Changelog\n\n)/,
|
|
`$1${unreleasedSection}`
|
|
);
|
|
writeFileSync(changelog, updated);
|
|
console.log(` Added [Unreleased] to ${changelog}`);
|
|
}
|
|
}
|
|
|
|
// Main flow
|
|
console.log("\n=== Release Script ===\n");
|
|
|
|
// 1. Check for uncommitted changes
|
|
console.log("Checking for uncommitted changes...");
|
|
const status = run("git status --porcelain", { silent: true });
|
|
if (status && status.trim()) {
|
|
console.error("Error: Uncommitted changes detected. Commit or stash first.");
|
|
console.error(status);
|
|
process.exit(1);
|
|
}
|
|
console.log(" Working directory clean\n");
|
|
|
|
// 2. Bump or set version
|
|
const version = bumpOrSetVersion(RELEASE_TARGET);
|
|
console.log(` New version: ${version}\n`);
|
|
|
|
// 3. Update changelogs
|
|
console.log("Updating CHANGELOG.md files...");
|
|
updateChangelogsForRelease(version);
|
|
console.log();
|
|
|
|
// 4. Commit and tag
|
|
console.log("Committing and tagging...");
|
|
stageChangedFiles();
|
|
run(`git commit -m "Release v${version}"`);
|
|
run(`git tag v${version}`);
|
|
console.log();
|
|
|
|
// 5. Publish
|
|
console.log("Publishing to npm...");
|
|
run("npm run publish");
|
|
console.log();
|
|
|
|
// 6. Add new [Unreleased] sections
|
|
console.log("Adding [Unreleased] sections for next cycle...");
|
|
addUnreleasedSection();
|
|
console.log();
|
|
|
|
// 7. Commit
|
|
console.log("Committing changelog updates...");
|
|
stageChangedFiles();
|
|
run(`git commit -m "Add [Unreleased] section for next cycle"`);
|
|
console.log();
|
|
|
|
// 8. Push
|
|
console.log("Pushing to remote...");
|
|
run("git push origin main");
|
|
run(`git push origin v${version}`);
|
|
console.log();
|
|
|
|
console.log(`=== Released v${version} ===`);
|