Files
flipper/desktop/scripts/workspaces.ts
Anton Nikolaev 6005d039f7 Dry-run for version bumping and npm publishing scripts
Summary: To make testing easier, I've implemented dry-run for version bumping and npm publishing scripts

Reviewed By: mweststrate

Differential Revision: D21721142

fbshipit-source-id: 72fc5a78bfc102a8fca9087decfbed6ebbda5e51
2020-05-26 06:44:25 -07:00

193 lines
4.6 KiB
TypeScript

/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
* @format
*/
import {rootDir, pluginsDir} from './paths';
import fs from 'fs-extra';
import path from 'path';
import {promisify} from 'util';
import globImport from 'glob';
import pfilter from 'p-filter';
import pmap from 'p-map';
import {execSync} from 'child_process';
const glob = promisify(globImport);
export interface Package {
dir: string;
json: any;
}
export interface Workspaces {
rootPackage: Package;
packages: Package[];
}
function isPlugin(dir: string) {
return dir.startsWith(pluginsDir);
}
export async function getWorkspaces(): Promise<Workspaces> {
const rootPackageJson = await fs.readJson(path.join(rootDir, 'package.json'));
const packageGlobs = rootPackageJson.workspaces.packages as string[];
const packages = await pmap(
await pfilter(
([] as string[]).concat(
...(await pmap(packageGlobs, (pattern) =>
glob(path.join(rootDir, pattern, '')),
)),
),
async (dir) => await fs.pathExists(path.join(dir, 'package.json')),
),
async (dir) => {
const json = await fs.readJson(path.join(dir, 'package.json'));
return {
dir,
json,
};
},
);
return {
rootPackage: {
dir: rootDir,
json: rootPackageJson,
},
packages,
};
}
export async function bumpVersions({
newVersion,
dryRun,
}: {
newVersion?: string;
dryRun?: boolean;
}) {
return await bumpWorkspaceVersions(await getWorkspaces(), newVersion, dryRun);
}
async function savePackageJson({dir, json}: Package) {
await fs.writeJson(path.join(dir, 'package.json'), json, {
spaces: 2,
});
}
function updateDependencies(
name: string,
dependencies: {[key: string]: string},
packagesToUpdate: string[],
newVersion: string,
): boolean {
if (!dependencies) {
return false;
}
let updated = false;
for (const packageName of packagesToUpdate) {
if (
dependencies[packageName] !== undefined &&
dependencies[packageName] !== newVersion
) {
console.log(
`Updated dependency of ${name}: ${packageName} from version ${dependencies[packageName]} to version ${newVersion}`,
);
dependencies[packageName] = newVersion;
updated = true;
}
}
return updated;
}
async function bumpWorkspaceVersions(
{rootPackage, packages}: Workspaces,
newVersion?: string,
dryRun?: boolean,
): Promise<string> {
newVersion = newVersion || (rootPackage.json.version as string);
const allPackages = [rootPackage, ...packages];
const localPackageNames = packages
.filter((pkg) => !isPlugin(pkg.dir))
.map(({json}) => json.name as string);
for (const pkg of allPackages) {
const {dir, json} = pkg;
let changed = false;
if (json.version !== newVersion && !isPlugin(dir)) {
console.log(
`Bumping version of ${pkg.json.name} from ${json.version} to ${newVersion}`,
);
json.version = newVersion;
changed = true;
}
if (
updateDependencies(
json.name,
json.dependencies,
localPackageNames,
newVersion,
)
) {
changed = true;
}
if (
updateDependencies(
json.name,
json.devDependencies,
localPackageNames,
newVersion,
)
) {
changed = true;
}
if (
updateDependencies(
json.name,
json.peerDependencies,
localPackageNames,
newVersion,
)
) {
changed = true;
}
if (changed) {
if (dryRun) {
console.log(
`DRYRUN: skipping saving changed package.json for ${pkg.json.name}`,
);
} else {
console.log(`Saving changed package.json for ${pkg.json.name}`);
await savePackageJson(pkg);
}
}
}
return newVersion;
}
export async function publishPackages({
newVersion,
proxy,
dryRun,
}: {
newVersion?: string;
proxy?: string;
dryRun?: boolean;
}) {
const workspaces = await getWorkspaces();
const version = await bumpWorkspaceVersions(workspaces, newVersion, dryRun);
let cmd = `yarn publish --new-version ${version}`;
if (proxy) {
cmd += ` --http-proxy ${proxy} --https-proxy ${proxy}`;
}
const publicPackages = workspaces.packages.filter((pkg) => !pkg.json.private);
for (const pkg of publicPackages) {
if (dryRun) {
console.log(`DRYRUN: Skipping npm publishing for ${pkg.json.name}`);
} else {
console.log(`Publishing ${pkg.json.name}...`);
execSync(cmd, {cwd: pkg.dir, stdio: 'inherit'});
}
}
}