fc9cda72d2
* build: convert all release scripts to typescript (#44035) * build: convert all release scripts to typescript * fix test imports * build: fix version bumper export * refactor: use as const * spec: fix bad type spec * build: use ts-node to spawn the version-bumper (#44057) Missed this in the tsification, we should probably call this via API instead of spawning a sub-proc? * build: still colors
229 lines
8.1 KiB
TypeScript
229 lines
8.1 KiB
TypeScript
import { Octokit } from '@octokit/rest';
|
|
import * as childProcess from 'node:child_process';
|
|
import * as fs from 'node:fs';
|
|
import * as path from 'node:path';
|
|
import * as semver from 'semver';
|
|
import * as temp from 'temp';
|
|
|
|
import { getCurrentBranch, ELECTRON_DIR } from '../lib/utils';
|
|
import { getElectronVersion } from '../lib/get-version';
|
|
|
|
import { getAssetContents } from './get-asset';
|
|
import { createGitHubTokenStrategy } from './github-token';
|
|
import { ELECTRON_ORG, ELECTRON_REPO, ElectronReleaseRepo, NIGHTLY_REPO } from './types';
|
|
|
|
const rootPackageJson = JSON.parse(fs.readFileSync(path.resolve(__dirname, '../../package.json'), 'utf-8'));
|
|
|
|
if (!process.env.ELECTRON_NPM_OTP) {
|
|
console.error('Please set ELECTRON_NPM_OTP');
|
|
process.exit(1);
|
|
}
|
|
|
|
let tempDir: string;
|
|
temp.track(); // track and cleanup files at exit
|
|
|
|
const files = [
|
|
'cli.js',
|
|
'index.js',
|
|
'install.js',
|
|
'package.json',
|
|
'README.md',
|
|
'LICENSE'
|
|
];
|
|
|
|
const jsonFields = [
|
|
'name',
|
|
'repository',
|
|
'description',
|
|
'license',
|
|
'author',
|
|
'keywords'
|
|
];
|
|
|
|
let npmTag = '';
|
|
|
|
const currentElectronVersion = getElectronVersion();
|
|
const isNightlyElectronVersion = currentElectronVersion.includes('nightly');
|
|
const targetRepo = getRepo();
|
|
|
|
const octokit = new Octokit({
|
|
userAgent: 'electron-npm-publisher',
|
|
authStrategy: createGitHubTokenStrategy(targetRepo)
|
|
});
|
|
|
|
function getRepo (): ElectronReleaseRepo {
|
|
return isNightlyElectronVersion ? NIGHTLY_REPO : ELECTRON_REPO;
|
|
}
|
|
|
|
new Promise<string>((resolve, reject) => {
|
|
temp.mkdir('electron-npm', (err, dirPath) => {
|
|
if (err) {
|
|
reject(err);
|
|
} else {
|
|
resolve(dirPath);
|
|
}
|
|
});
|
|
})
|
|
.then((dirPath) => {
|
|
tempDir = dirPath;
|
|
// copy files from `/npm` to temp directory
|
|
for (const name of files) {
|
|
const noThirdSegment = name === 'README.md' || name === 'LICENSE';
|
|
fs.writeFileSync(
|
|
path.join(tempDir, name),
|
|
fs.readFileSync(path.join(ELECTRON_DIR, noThirdSegment ? '' : 'npm', name))
|
|
);
|
|
}
|
|
// copy from root package.json to temp/package.json
|
|
const packageJson = require(path.join(tempDir, 'package.json'));
|
|
for (const fieldName of jsonFields) {
|
|
packageJson[fieldName] = rootPackageJson[fieldName];
|
|
}
|
|
packageJson.version = currentElectronVersion;
|
|
fs.writeFileSync(
|
|
path.join(tempDir, 'package.json'),
|
|
JSON.stringify(packageJson, null, 2)
|
|
);
|
|
|
|
return octokit.repos.listReleases({
|
|
owner: ELECTRON_ORG,
|
|
repo: targetRepo
|
|
});
|
|
})
|
|
.then((releases) => {
|
|
// download electron.d.ts from release
|
|
const release = releases.data.find(
|
|
(release) => release.tag_name === `v${currentElectronVersion}`
|
|
);
|
|
if (!release) {
|
|
throw new Error(`cannot find release with tag v${currentElectronVersion}`);
|
|
}
|
|
return release;
|
|
})
|
|
.then(async (release) => {
|
|
const tsdAsset = release.assets.find((asset) => asset.name === 'electron.d.ts');
|
|
if (!tsdAsset) {
|
|
throw new Error(`cannot find electron.d.ts from v${currentElectronVersion} release assets`);
|
|
}
|
|
|
|
const typingsContent = await getAssetContents(
|
|
targetRepo,
|
|
tsdAsset.id
|
|
);
|
|
|
|
fs.writeFileSync(path.join(tempDir, 'electron.d.ts'), typingsContent);
|
|
|
|
return release;
|
|
})
|
|
.then(async (release) => {
|
|
const checksumsAsset = release.assets.find((asset) => asset.name === 'SHASUMS256.txt');
|
|
if (!checksumsAsset) {
|
|
throw new Error(`cannot find SHASUMS256.txt from v${currentElectronVersion} release assets`);
|
|
}
|
|
|
|
const checksumsContent = await getAssetContents(
|
|
targetRepo,
|
|
checksumsAsset.id
|
|
);
|
|
|
|
const checksumsObject: Record<string, string> = Object.create(null);
|
|
for (const line of checksumsContent.trim().split('\n')) {
|
|
const [checksum, file] = line.split(' *');
|
|
checksumsObject[file] = checksum;
|
|
}
|
|
|
|
fs.writeFileSync(path.join(tempDir, 'checksums.json'), JSON.stringify(checksumsObject, null, 2));
|
|
|
|
return release;
|
|
})
|
|
.then(async (release) => {
|
|
const currentBranch = await getCurrentBranch();
|
|
|
|
if (isNightlyElectronVersion) {
|
|
// Nightlies get published to their own module, so they should be tagged as latest
|
|
npmTag = currentBranch === 'main' ? 'latest' : `nightly-${currentBranch}`;
|
|
|
|
const currentJson = JSON.parse(fs.readFileSync(path.join(tempDir, 'package.json'), 'utf8'));
|
|
currentJson.name = 'electron-nightly';
|
|
rootPackageJson.name = 'electron-nightly';
|
|
|
|
fs.writeFileSync(
|
|
path.join(tempDir, 'package.json'),
|
|
JSON.stringify(currentJson, null, 2)
|
|
);
|
|
} else {
|
|
if (currentBranch === 'main') {
|
|
// This should never happen, main releases should be nightly releases
|
|
// this is here just-in-case
|
|
throw new Error('Unreachable release phase, can\'t tag a non-nightly release on the main branch');
|
|
} else if (!release.prerelease) {
|
|
// Tag the release with a `2-0-x` style tag
|
|
npmTag = currentBranch;
|
|
} else if (release.tag_name.indexOf('alpha') > 0) {
|
|
// Tag the release with an `alpha-3-0-x` style tag
|
|
npmTag = `alpha-${currentBranch}`;
|
|
} else {
|
|
// Tag the release with a `beta-3-0-x` style tag
|
|
npmTag = `beta-${currentBranch}`;
|
|
}
|
|
}
|
|
})
|
|
.then(() => childProcess.execSync('npm pack', { cwd: tempDir }))
|
|
.then(() => {
|
|
// test that the package can install electron prebuilt from github release
|
|
const tarballPath = path.join(tempDir, `${rootPackageJson.name}-${currentElectronVersion}.tgz`);
|
|
return new Promise((resolve, reject) => {
|
|
const result = childProcess.spawnSync('npm', ['install', tarballPath, '--force', '--silent'], {
|
|
env: { ...process.env, electron_config_cache: tempDir },
|
|
cwd: tempDir,
|
|
stdio: 'inherit'
|
|
});
|
|
if (result.status !== 0) {
|
|
return reject(new Error(`npm install failed with status ${result.status}`));
|
|
}
|
|
try {
|
|
const electronPath = require(path.resolve(tempDir, 'node_modules', rootPackageJson.name));
|
|
if (typeof electronPath !== 'string') {
|
|
return reject(new Error(`path to electron binary (${electronPath}) returned by the ${rootPackageJson.name} module is not a string`));
|
|
}
|
|
if (!fs.existsSync(electronPath)) {
|
|
return reject(new Error(`path to electron binary (${electronPath}) returned by the ${rootPackageJson.name} module does not exist on disk`));
|
|
}
|
|
} catch (e) {
|
|
console.error(e);
|
|
return reject(new Error(`loading the generated ${rootPackageJson.name} module failed with an error`));
|
|
}
|
|
resolve(tarballPath);
|
|
});
|
|
})
|
|
.then((tarballPath) => {
|
|
const existingVersionJSON = childProcess.execSync(`npx npm@7 view ${rootPackageJson.name}@${currentElectronVersion} --json`).toString('utf-8');
|
|
// It's possible this is a re-run and we already have published the package, if not we just publish like normal
|
|
if (!existingVersionJSON) {
|
|
childProcess.execSync(`npm publish ${tarballPath} --tag ${npmTag} --otp=${process.env.ELECTRON_NPM_OTP}`);
|
|
}
|
|
})
|
|
.then(() => {
|
|
const currentTags = JSON.parse(childProcess.execSync('npm show electron dist-tags --json').toString());
|
|
const parsedLocalVersion = semver.parse(currentElectronVersion)!;
|
|
if (rootPackageJson.name === 'electron') {
|
|
// We should only customly add dist tags for non-nightly releases where the package name is still
|
|
// "electron"
|
|
if (parsedLocalVersion.prerelease.length === 0 &&
|
|
semver.gt(currentElectronVersion, currentTags.latest)) {
|
|
childProcess.execSync(`npm dist-tag add electron@${currentElectronVersion} latest --otp=${process.env.ELECTRON_NPM_OTP}`);
|
|
}
|
|
if (parsedLocalVersion.prerelease[0] === 'beta' &&
|
|
semver.gt(currentElectronVersion, currentTags.beta)) {
|
|
childProcess.execSync(`npm dist-tag add electron@${currentElectronVersion} beta --otp=${process.env.ELECTRON_NPM_OTP}`);
|
|
}
|
|
if (parsedLocalVersion.prerelease[0] === 'alpha' &&
|
|
semver.gt(currentElectronVersion, currentTags.alpha)) {
|
|
childProcess.execSync(`npm dist-tag add electron@${currentElectronVersion} alpha --otp=${process.env.ELECTRON_NPM_OTP}`);
|
|
}
|
|
}
|
|
})
|
|
.catch((err) => {
|
|
console.error('Error:', err);
|
|
process.exit(1);
|
|
});
|