chore: 包内依赖改成workspace:*,更新release脚本

This commit is contained in:
roymondchen 2024-04-26 17:15:28 +08:00
parent e39a7d140f
commit 6b081fdc53
17 changed files with 1437 additions and 475 deletions

View File

@ -28,7 +28,7 @@
"prepare": "husky install",
"commit": "git-cz",
"changelog": "conventional-changelog -p angular -i CHANGELOG.md -s",
"release": "node scripts/release.cjs"
"release": "node scripts/release.mjs"
},
"engines": {
"node": ">=18"
@ -51,7 +51,6 @@
"@vitejs/plugin-vue": "^4.5.2",
"@vitest/coverage-v8": "^1.0.4",
"c8": "^7.11.3",
"chalk": "^4.1.0",
"commitizen": "^4.3.0",
"conventional-changelog-cli": "^4.1.0",
"cosmiconfig": "^8.3.6",
@ -70,6 +69,7 @@
"jsdom": "^19.0.0",
"lint-staged": "^11.0.1",
"minimist": "^1.2.6",
"picocolors": "^1.0.0",
"prettier": "^2.8.7",
"recast": "^0.20.4",
"rimraf": "^3.0.2",

View File

@ -25,10 +25,10 @@
},
"dependencies": {
"cac": "^6.7.12",
"chalk": "^4.1.0",
"chokidar": "^3.5.3",
"esbuild": "^0.17.19",
"fs-extra": "^10.1.0",
"picocolors": "^1.0.0",
"recast": "^0.21.1",
"tslib": "^2.4.0"
},

View File

@ -1,17 +1,17 @@
import chalk from 'chalk';
import pico from 'picocolors';
export const info = (msg: string) => {
console.log(chalk.white(msg));
console.log(pico.white(msg));
};
export const error = (msg: string) => {
console.log(chalk.red(msg));
console.log(pico.red(msg));
};
export const success = (msg: string) => {
console.log(chalk.green(msg));
console.log(pico.green(msg));
};
export const execInfo = (msg: string) => {
console.log(chalk.blue(msg));
console.log(pico.blue(msg));
};

View File

@ -38,9 +38,9 @@
"vue"
],
"dependencies": {
"@tmagic/data-source": "1.4.4",
"@tmagic/schema": "1.4.4",
"@tmagic/utils": "1.4.4",
"@tmagic/data-source": "workspace:*",
"@tmagic/schema": "workspace:*",
"@tmagic/utils": "workspace:*",
"events": "^3.3.0",
"lodash-es": "^4.17.21"
},

View File

@ -34,9 +34,9 @@
"data-source"
],
"dependencies": {
"@tmagic/dep": "1.4.4",
"@tmagic/utils": "1.4.4",
"@tmagic/schema": "1.4.4",
"@tmagic/dep": "workspace:*",
"@tmagic/utils": "workspace:*",
"@tmagic/schema": "workspace:*",
"events": "^3.3.0",
"lodash-es": "^4.17.21"
},

View File

@ -31,8 +31,8 @@
"url": "https://github.com/Tencent/tmagic-editor.git"
},
"dependencies": {
"@tmagic/schema": "1.4.4",
"@tmagic/utils": "1.4.4"
"@tmagic/schema": "workspace:*",
"@tmagic/utils": "workspace:*"
},
"devDependencies": {
"@types/node": "^18.19.0",

View File

@ -1,5 +1,5 @@
{
"version": "1.4.4",
"version": "workspace:*",
"name": "@tmagic/editor",
"type": "module",
"sideEffects": [
@ -48,14 +48,14 @@
"dependencies": {
"@babel/core": "^7.18.0",
"@element-plus/icons-vue": "^2.3.1",
"@tmagic/core": "1.4.4",
"@tmagic/dep": "1.4.4",
"@tmagic/design": "1.4.4",
"@tmagic/form": "1.4.4",
"@tmagic/schema": "1.4.4",
"@tmagic/stage": "1.4.4",
"@tmagic/table": "1.4.4",
"@tmagic/utils": "1.4.4",
"@tmagic/core": "workspace:*",
"@tmagic/dep": "workspace:*",
"@tmagic/design": "workspace:*",
"@tmagic/form": "workspace:*",
"@tmagic/schema": "workspace:*",
"@tmagic/stage": "workspace:*",
"@tmagic/table": "workspace:*",
"@tmagic/utils": "workspace:*",
"buffer": "^6.0.3",
"color": "^3.1.3",
"emmet-monaco-es": "^5.3.0",
@ -69,8 +69,8 @@
"vue": "^3.4.21"
},
"peerDependencies": {
"@tmagic/design": "1.4.4",
"@tmagic/form": "1.4.4",
"@tmagic/design": "workspace:*",
"@tmagic/form": "workspace:*",
"monaco-editor": "^0.47.0",
"vue": "^3.4.21"
},

View File

@ -38,11 +38,11 @@
"typescript"
],
"dependencies": {
"@tmagic/design": "1.4.4",
"@tmagic/design": "workspace:*",
"element-plus": "^2.6.1"
},
"peerDependencies": {
"@tmagic/design": "1.4.4",
"@tmagic/design": "workspace:*",
"element-plus": "^2.6.1"
},
"devDependencies": {

View File

@ -38,8 +38,8 @@
},
"dependencies": {
"@element-plus/icons-vue": "^2.3.1",
"@tmagic/design": "1.4.4",
"@tmagic/utils": "1.4.4",
"@tmagic/design": "workspace:*",
"@tmagic/utils": "workspace:*",
"lodash-es": "^4.17.21",
"sortablejs": "^1.14.0",
"vue": "^3.4.21"

View File

@ -32,9 +32,9 @@
},
"dependencies": {
"@scena/guides": "^0.29.2",
"@tmagic/core": "1.4.4",
"@tmagic/schema": "1.4.4",
"@tmagic/utils": "1.4.4",
"@tmagic/core": "workspace:*",
"@tmagic/schema": "workspace:*",
"@tmagic/utils": "workspace:*",
"events": "^3.3.0",
"keycon": "^1.4.0",
"lodash-es": "^4.17.21",

View File

@ -35,14 +35,14 @@
"url": "https://github.com/Tencent/tmagic-editor.git"
},
"dependencies": {
"@tmagic/design": "1.4.4",
"@tmagic/form": "1.4.4",
"@tmagic/utils": "1.4.4",
"@tmagic/design": "workspace:*",
"@tmagic/form": "workspace:*",
"@tmagic/utils": "workspace:*",
"lodash-es": "^4.17.21",
"vue": "^3.4.21"
},
"peerDependencies": {
"@tmagic/form": "1.4.4",
"@tmagic/form": "workspace:*",
"vue": "^3.4.21"
},
"devDependencies": {

View File

@ -38,12 +38,12 @@
"typescript"
],
"dependencies": {
"@tmagic/design": "1.4.4",
"@tmagic/design": "workspace:*",
"tdesign-vue-next": "^1.8.1",
"vue": "^3.4.21"
},
"peerDependencies": {
"@tmagic/design": "1.4.4",
"@tmagic/design": "workspace:*",
"tdesign-vue-next": "^1.8.1",
"vue": "^3.4.21"
},

View File

@ -31,7 +31,7 @@
"url": "https://github.com/Tencent/tmagic-editor.git"
},
"dependencies": {
"@tmagic/schema": "1.4.4",
"@tmagic/schema": "workspace:*",
"dayjs": "^1.11.4",
"lodash-es": "^4.17.21"
},

1223
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@ -32,12 +32,12 @@
"url": "https://github.com/Tencent/tmagic-editor.git"
},
"dependencies": {
"@tmagic/core": "1.4.4",
"@tmagic/editor": "1.4.4",
"@tmagic/form": "1.4.4",
"@tmagic/utils": "1.4.4",
"@tmagic/schema": "1.4.4",
"@tmagic/stage": "1.4.4",
"@tmagic/core": "workspace:*",
"@tmagic/editor": "workspace:*",
"@tmagic/form": "workspace:*",
"@tmagic/utils": "workspace:*",
"@tmagic/schema": "workspace:*",
"@tmagic/stage": "workspace:*",
"element-plus": "^2.6.1",
"vue": "^3.4.21"
},

View File

@ -1,209 +0,0 @@
const args = require('minimist')(process.argv.slice(2));
const fs = require('fs');
const path = require('path');
const chalk = require('chalk');
const semver = require('semver');
const currentVersion = require('../package.json').version;
const { prompt } = require('enquirer');
const execa = require('execa');
const preId = args.preid || (semver.prerelease(currentVersion) && semver.prerelease(currentVersion)[0]);
const isDryRun = args.dry;
const { skipTests } = args;
const { skipBuild } = args;
const packages = fs
.readdirSync(path.resolve(__dirname, '../packages'))
.filter((p) => !p.endsWith('.ts') && !p.startsWith('.'));
const skippedPackages = [];
const versionIncrements = [
'patch',
'minor',
'major',
...(preId ? ['prepatch', 'preminor', 'premajor', 'prerelease'] : []),
];
const inc = (i) => semver.inc(currentVersion, i, preId);
const run = (bin, args, opts = {}) => execa(bin, args, { stdio: 'inherit', ...opts });
const dryRun = (bin, args, opts = {}) => console.log(chalk.blue(`[dryrun] ${bin} ${args.join(' ')}`), opts);
const runIfNotDry = isDryRun ? dryRun : run;
const getPkgRoot = (pkg) => path.resolve(__dirname, `../packages/${pkg}`);
const getRunTimeRoot = (pkg) => path.resolve(__dirname, `../runtime/${pkg}`);
const getPlayground = () => path.resolve(__dirname, `../playground`);
const step = (msg) => console.log(chalk.cyan(msg));
async function main() {
let targetVersion = args._[0];
if (!targetVersion) {
// no explicit version, offer suggestions
const { release } = await prompt({
type: 'select',
name: 'release',
message: 'Select release type',
choices: versionIncrements.map((i) => `${i} (${inc(i)})`).concat(['custom']),
});
if (release === 'custom') {
targetVersion = (
await prompt({
type: 'input',
name: 'version',
message: 'Input custom version',
initial: currentVersion,
})
).version;
} else {
[, targetVersion] = release.match(/\((.*)\)/);
}
}
if (!semver.valid(targetVersion)) {
throw new Error(`invalid target version: ${targetVersion}`);
}
const { yes } = await prompt({
type: 'confirm',
name: 'yes',
message: `Releasing v${targetVersion}. Confirm?`,
});
if (!yes) {
return;
}
// run tests before release
step('\nRunning tests...');
if (!skipTests && !isDryRun) {
await run('pnpm', ['test']);
} else {
console.log(`(skipped)`);
}
// update all package versions and inter-dependencies
step('\nUpdating cross dependencies...');
updateVersions(targetVersion);
// build all packages with types
step('\nBuilding all packages...');
if (!skipBuild && !isDryRun) {
await run('pnpm', ['run', 'build']);
} else {
console.log(`(skipped)`);
}
// generate changelog
step('\nGenerating changelog...');
await run(`pnpm`, ['run', 'changelog']);
// update pnpm-lock.yaml
step('\nUpdating lockfile...');
await run(`pnpm`, ['install', '--prefer-offline']);
const { stdout } = await run('git', ['diff'], { stdio: 'pipe' });
if (stdout) {
step('\nCommitting changes...');
await runIfNotDry('git', ['add', '-A']);
await runIfNotDry('git', ['commit', '-m', `chore: release v${targetVersion}`, '--no-verify']);
} else {
console.log('No changes to commit.');
}
// publish packages
step('\nPublishing packages...');
for (const pkg of packages) {
await publishPackage(pkg, targetVersion, runIfNotDry);
}
// push to GitHub
step('\nPushing to GitHub...');
await runIfNotDry('git', ['tag', `v${targetVersion}`]);
await runIfNotDry('git', ['push']);
if (isDryRun) {
console.log(`\nDry run finished - run git diff to see package changes.`);
}
if (skippedPackages.length) {
console.log(
chalk.yellow(`The following packages are skipped and NOT published:\n- ${skippedPackages.join('\n- ')}`),
);
}
console.log();
}
function updateVersions(version) {
// 1. update root package.json
updatePackage(path.resolve(__dirname, '..'), version);
// 2. update all packages
packages.forEach((p) => updatePackage(getPkgRoot(p), version));
['vue3', 'react', 'vue2'].forEach((p) => updatePackage(getRunTimeRoot(p), version));
updatePackage(getPlayground(), version);
updatePackage(getRunTimeRoot('tmagic-form'), version, false);
}
function updatePackage(pkgRoot, version, updateVersion = true) {
const pkgPath = path.resolve(pkgRoot, 'package.json');
const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'));
if (updateVersion) {
pkg.version = version;
}
updateDeps(pkg, 'dependencies', version);
updateDeps(pkg, 'peerDependencies', version);
fs.writeFileSync(pkgPath, `${JSON.stringify(pkg, null, 2)}\n`);
}
function updateDeps(pkg, depType, version) {
const deps = pkg[depType];
if (!deps) return;
Object.keys(deps).forEach((dep) => {
if (dep.startsWith('@tmagic') && packages.includes(dep.replace(/^@tmagic\//, ''))) {
console.log(chalk.yellow(`${pkg.name} -> ${depType} -> ${dep}@${version}`));
deps[dep] = version;
}
});
}
async function publishPackage(pkgName, version, runIfNotDry) {
if (skippedPackages.includes(pkgName)) {
return;
}
const pkgRoot = getPkgRoot(pkgName);
const pkgPath = path.resolve(pkgRoot, 'package.json');
const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'));
if (pkg.private) {
return;
}
let releaseTag = null;
if (args.tag) {
releaseTag = args.tag;
} else if (version.includes('alpha')) {
releaseTag = 'alpha';
} else if (version.includes('beta')) {
releaseTag = 'beta';
} else if (version.includes('rc')) {
releaseTag = 'rc';
}
step(`Publishing ${pkgName}...`);
try {
await runIfNotDry('pnpm', ['publish', ...(releaseTag ? ['--tag', releaseTag] : []), '--access', 'public'], {
cwd: pkgRoot,
stdio: 'pipe',
});
console.log(chalk.green(`Successfully published ${pkgName}@${version}`));
} catch (e) {
if (e.stderr.match(/previously published/)) {
console.log(chalk.red(`Skipping already published: ${pkgName}`));
} else {
throw e;
}
}
}
main().catch((err) => {
updateVersions(currentVersion);
console.error(err);
});

386
scripts/release.mjs Normal file
View File

@ -0,0 +1,386 @@
// @ts-check
import enquirer from 'enquirer';
import execa from 'execa';
import minimist from 'minimist';
import fs from 'node:fs';
import { createRequire } from 'node:module';
import path from 'node:path';
import { fileURLToPath } from 'node:url';
import pico from 'picocolors';
import semver from 'semver';
/**
* @typedef {{
* name: string
* version: string
* dependencies?: { [dependenciesPackageName: string]: string }
* peerDependencies?: { [peerDependenciesPackageName: string]: string }
* }} Package
*/
let versionUpdated = false;
const { prompt } = enquirer;
const currentVersion = createRequire(import.meta.url)('../package.json').version;
const __dirname = path.dirname(fileURLToPath(import.meta.url));
const args = minimist(process.argv.slice(2), {
alias: {
skipBuild: 'skip-build',
skipTests: 'skip-tests',
skipGit: 'skip-git',
skipPrompts: 'skip-prompts',
},
});
const preId = args.preid || semver.prerelease(currentVersion)?.[0];
const isDryRun = args.dry;
/** @type {boolean | undefined} */
let { skipTests } = args;
const { skipBuild } = args;
const skipPrompts = args.skipPrompts || args.canary;
const skipGit = args.skipGit || args.canary;
const packages = fs.readdirSync(path.resolve(__dirname, '../packages')).filter((p) => {
const pkgRoot = path.resolve(__dirname, '../packages', p);
if (fs.statSync(pkgRoot).isDirectory()) {
const pkg = JSON.parse(fs.readFileSync(path.resolve(pkgRoot, 'package.json'), 'utf-8'));
return !pkg.private;
}
});
const keepThePackageName = (/** @type {string} */ pkgName) => pkgName;
/** @type {string[]} */
const skippedPackages = [];
/** @type {ReadonlyArray<import('semver').ReleaseType>} */
const versionIncrements = [
'patch',
'minor',
'major',
...(preId ? /** @type {const} */ (['prepatch', 'preminor', 'premajor', 'prerelease']) : []),
];
const inc = (/** @type {import('semver').ReleaseType} */ i) => semver.inc(currentVersion, i, preId);
const run = async (
/** @type {string} */ bin,
/** @type {ReadonlyArray<string>} */ args,
/** @type {import('execa').Options} */ opts = {},
) => execa(bin, args, { stdio: 'inherit', ...opts });
const dryRun = async (
/** @type {string} */ bin,
/** @type {ReadonlyArray<string>} */ args,
/** @type {import('execa').Options} */ opts = {},
) => console.log(pico.blue(`[dryrun] ${bin} ${args.join(' ')}`), opts);
const runIfNotDry = isDryRun ? dryRun : run;
const getPkgRoot = (/** @type {string} */ pkg) => path.resolve(__dirname, `../packages/${pkg}`);
const getRunTimeRoot = (pkg) => path.resolve(__dirname, `../runtime/${pkg}`);
const getPlayground = () => path.resolve(__dirname, `../playground`);
const step = (/** @type {string} */ msg) => console.log(pico.cyan(msg));
async function main() {
if (!(await isInSyncWithRemote())) {
return;
}
console.log(`${pico.green(``)} commit is up-to-date with remote.\n`);
let targetVersion = args._[0];
if (!targetVersion) {
// no explicit version, offer suggestions
/** @type {{ release: string }} */
const { release } = await prompt({
type: 'select',
name: 'release',
message: 'Select release type',
choices: versionIncrements.map((i) => `${i} (${inc(i)})`).concat(['custom']),
});
if (release === 'custom') {
/** @type {{ version: string }} */
const result = await prompt({
type: 'input',
name: 'version',
message: 'Input custom version',
initial: currentVersion,
});
targetVersion = result.version;
} else {
targetVersion = release.match(/\((.*)\)/)?.[1] ?? '';
}
}
if (!semver.valid(targetVersion)) {
throw new Error(`invalid target version: ${targetVersion}`);
}
if (skipPrompts) {
step(`Releasing v${targetVersion}...`);
} else {
/** @type {{ yes: boolean }} */
const { yes: confirmRelease } = await prompt({
type: 'confirm',
name: 'yes',
message: `Releasing v${targetVersion}. Confirm?`,
});
if (!confirmRelease) {
return;
}
}
if (!skipTests) {
step('Checking CI status for HEAD...');
const isCIPassed = await getCIResult();
skipTests ||= isCIPassed;
if (isCIPassed && !skipPrompts) {
/** @type {{ yes: boolean }} */
const { yes: promptSkipTests } = await prompt({
type: 'confirm',
name: 'yes',
message: `CI for this commit passed. Skip local tests?`,
});
skipTests = promptSkipTests;
}
}
if (!skipTests) {
step('\nRunning tests...');
if (!isDryRun) {
await run('pnpm', ['run', 'test', '--run']);
} else {
console.log(`Skipped (dry run)`);
}
} else {
step('Tests skipped.');
}
// update all package versions and inter-dependencies
step('\nUpdating cross dependencies...');
updateVersions(targetVersion);
versionUpdated = true;
// build all packages with types
step('\nBuilding all packages...');
if (!skipBuild && !isDryRun) {
await run('pnpm', ['run', 'build']);
} else {
console.log(`(skipped)`);
}
// generate changelog
step('\nGenerating changelog...');
await run(`pnpm`, ['run', 'changelog']);
if (!skipPrompts) {
/** @type {{ yes: boolean }} */
const { yes: changelogOk } = await prompt({
type: 'confirm',
name: 'yes',
message: `Changelog generated. Does it look good?`,
});
if (!changelogOk) {
return;
}
}
if (!skipGit) {
const { stdout } = await run('git', ['diff'], { stdio: 'pipe' });
if (stdout) {
step('\nCommitting changes...');
await runIfNotDry('git', ['add', '-A']);
await runIfNotDry('git', ['commit', '-m', `release: v${targetVersion}`]);
} else {
console.log('No changes to commit.');
}
}
// publish packages
step('\nPublishing packages...');
const additionalPublishFlags = [];
if (isDryRun) {
additionalPublishFlags.push('--dry-run');
}
if (isDryRun || skipGit) {
additionalPublishFlags.push('--no-git-checks');
}
// bypass the pnpm --publish-branch restriction which isn't too useful to us
// otherwise it leads to a prompt and blocks the release script
const branch = await getBranch();
if (branch !== 'master') {
additionalPublishFlags.push('--publish-branch', branch);
}
for (const pkg of packages) {
await publishPackage(pkg, targetVersion, additionalPublishFlags);
}
// push to GitHub
if (!skipGit) {
step('\nPushing to GitHub...');
await runIfNotDry('git', ['tag', `v${targetVersion}`]);
await runIfNotDry('git', ['push', 'origin', `refs/tags/v${targetVersion}`]);
await runIfNotDry('git', ['push']);
}
if (isDryRun) {
console.log(`\nDry run finished - run git diff to see package changes.`);
}
if (skippedPackages.length) {
console.log(
pico.yellow(`The following packages are skipped and NOT published:\n- ${skippedPackages.join('\n- ')}`),
);
}
console.log();
}
async function getCIResult() {
try {
const sha = await getSha();
const res = await fetch(
`https://api.github.com/repos/vuejs/core/actions/runs?head_sha=${sha}` +
`&status=success&exclude_pull_requests=true`,
);
const data = await res.json();
return data.workflow_runs.length > 0;
} catch {
console.error('Failed to get CI status for current commit.');
return false;
}
}
async function isInSyncWithRemote() {
try {
const branch = await getBranch();
const res = await fetch(`https://api.github.com/repos/vuejs/core/commits/${branch}?per_page=1`);
const data = await res.json();
if (data.sha === (await getSha())) {
return true;
}
/** @type {{ yes: boolean }} */
const { yes } = await prompt({
type: 'confirm',
name: 'yes',
message: pico.red(`Local HEAD is not up-to-date with remote. Are you sure you want to continue?`),
});
return yes;
} catch {
console.error(pico.red('Failed to check whether local HEAD is up-to-date with remote.'));
return false;
}
}
async function getSha() {
return (await execa('git', ['rev-parse', 'HEAD'])).stdout;
}
async function getBranch() {
return (await execa('git', ['rev-parse', '--abbrev-ref', 'HEAD'])).stdout;
}
/**
* @param {string} version
* @param {(pkgName: string) => string} getNewPackageName
*/
function updateVersions(version, getNewPackageName = keepThePackageName) {
// 1. update root package.json
updatePackage(path.resolve(__dirname, '..'), version, getNewPackageName);
// 2. update all packages
packages.forEach((p) => updatePackage(getPkgRoot(p), version, getNewPackageName));
['vue3', 'react', 'vue2'].forEach((p) => updatePackage(getRunTimeRoot(p), version, getNewPackageName, true));
updatePackage(getPlayground(), version, getNewPackageName, true);
}
/**
* @param {string} pkgRoot
* @param {string} version
* @param {(pkgName: string) => string} getNewPackageName
*/
function updatePackage(pkgRoot, version, getNewPackageName, updateDep = false) {
const pkgPath = path.resolve(pkgRoot, 'package.json');
/** @type {Package} */
const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'));
pkg.name = getNewPackageName(pkg.name);
pkg.version = version;
if (updateDep) {
updateDeps(pkg, 'dependencies', version);
updateDeps(pkg, 'peerDependencies', version);
}
fs.writeFileSync(pkgPath, `${JSON.stringify(pkg, null, 2)}\n`);
}
/**
* @param {Package} pkg
* @param {'dependencies' | 'peerDependencies'} depType
* @param {string} version
*/
function updateDeps(pkg, depType, version) {
const deps = pkg[depType];
if (!deps) return;
Object.keys(deps).forEach((dep) => {
if (dep.startsWith('@tmagic') && packages.includes(dep.replace(/^@tmagic\//, ''))) {
console.log(pico.yellow(`${pkg.name} -> ${depType} -> ${dep}@${version}`));
deps[dep] = version;
}
});
}
/**
* @param {string} pkgName
* @param {string} version
* @param {ReadonlyArray<string>} additionalFlags
*/
async function publishPackage(pkgName, version, additionalFlags) {
if (skippedPackages.includes(pkgName)) {
return;
}
let releaseTag = null;
if (args.tag) {
releaseTag = args.tag;
} else if (version.includes('alpha')) {
releaseTag = 'alpha';
} else if (version.includes('beta')) {
releaseTag = 'beta';
} else if (version.includes('rc')) {
releaseTag = 'rc';
}
step(`Publishing ${pkgName}...`);
try {
// Don't change the package manager here as we rely on pnpm to handle
// workspace:* deps
await run(
'pnpm',
['publish', ...(releaseTag ? ['--tag', releaseTag] : []), '--access', 'public', ...additionalFlags],
{
cwd: getPkgRoot(pkgName),
stdio: 'pipe',
},
);
console.log(pico.green(`Successfully published ${pkgName}@${version}`));
} catch (/** @type {any} */ e) {
if (e.stderr.match(/previously published/)) {
console.log(pico.red(`Skipping already published: ${pkgName}`));
} else {
throw e;
}
}
}
main().catch((err) => {
if (versionUpdated) {
// revert to current version on failed releases
updateVersions(currentVersion);
}
console.error(err);
process.exit(1);
});