mirror of
https://github.com/Tencent/tmagic-editor.git
synced 2025-04-05 19:41:40 +08:00
chore: 包内依赖改成workspace:*,更新release脚本
This commit is contained in:
parent
e39a7d140f
commit
6b081fdc53
@ -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",
|
||||
|
@ -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"
|
||||
},
|
||||
|
@ -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));
|
||||
};
|
||||
|
@ -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"
|
||||
},
|
||||
|
@ -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"
|
||||
},
|
||||
|
@ -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",
|
||||
|
@ -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"
|
||||
},
|
||||
|
@ -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": {
|
||||
|
@ -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"
|
||||
|
@ -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",
|
||||
|
@ -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": {
|
||||
|
@ -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"
|
||||
},
|
||||
|
@ -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
1223
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@ -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"
|
||||
},
|
||||
|
@ -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
386
scripts/release.mjs
Normal 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);
|
||||
});
|
Loading…
x
Reference in New Issue
Block a user