storybook/scripts/run-registry.ts

179 lines
5.0 KiB
TypeScript
Raw Normal View History

2020-05-09 11:34:50 +02:00
import { exec } from 'child_process';
import chalk from 'chalk';
2020-05-09 11:34:50 +02:00
import path from 'path';
import program from 'commander';
import detectFreePort from 'detect-port';
import dedent from 'ts-dedent';
import fs from 'fs';
2020-05-09 11:34:50 +02:00
import yaml from 'js-yaml';
import nodeCleanup from 'node-cleanup';
2020-05-09 11:34:50 +02:00
import startVerdaccioServer from 'verdaccio';
import pLimit from 'p-limit';
// @ts-ignore
import { maxConcurrentTasks } from './utils/concurrency';
2020-05-18 22:29:50 +02:00
import { listOfPackages, Package } from './utils/list-packages';
2019-04-25 11:44:33 +02:00
2020-05-09 11:34:50 +02:00
program
.option('-O, --open', 'keep process open')
.option('-P, --publish', 'should publish packages')
.option('-p, --port <port>', 'port to run https server on');
2020-05-09 11:34:50 +02:00
program.parse(process.argv);
const logger = console;
2020-05-18 14:58:55 +02:00
const freePort = (port?: number) => port || detectFreePort(port);
2019-04-25 11:44:33 +02:00
2020-05-18 14:58:55 +02:00
const startVerdaccio = (port: number) => {
let resolved = false;
return Promise.race([
2020-05-09 11:34:50 +02:00
new Promise((resolve) => {
const cache = path.join(__dirname, '..', '.verdaccio-cache');
const config = {
2020-11-26 16:55:47 +01:00
...(yaml.safeLoad(
fs.readFileSync(path.join(__dirname, 'verdaccio.yaml'), 'utf8')
) as Record<string, any>),
2020-05-09 11:34:50 +02:00
self_path: cache,
};
2020-05-18 15:09:22 +02:00
const onReady = (webServer: any) => {
2020-05-09 11:34:50 +02:00
webServer.listen(port, () => {
resolved = true;
2020-05-09 11:34:50 +02:00
resolve(webServer);
});
2020-05-09 11:34:50 +02:00
};
startVerdaccioServer(config, 6000, cache, '1.0.0', 'verdaccio', onReady);
}),
2020-05-18 22:29:50 +02:00
new Promise((_, rej) => {
setTimeout(() => {
if (!resolved) {
resolved = true;
rej(new Error(`TIMEOUT - verdaccio didn't start within 10s`));
}
}, 10000);
}),
]);
};
2020-05-18 15:09:22 +02:00
const registryUrl = (command: string, url?: string) =>
new Promise<string>((res, rej) => {
const args = url ? ['config', 'set', 'registry', url] : ['config', 'get', 'registry'];
exec(`${command} ${args.join(' ')}`, { cwd: path.join(process.cwd(), '..') }, (e, stdout) => {
if (e) {
rej(e);
} else {
res(url || stdout.toString().trim());
}
});
});
2020-05-18 15:09:22 +02:00
const registriesUrl = (yarnUrl?: string, npmUrl?: string) =>
Promise.all([registryUrl('/usr/local/bin/yarn', yarnUrl), registryUrl('npm', npmUrl || yarnUrl)]);
2020-05-18 14:58:55 +02:00
const applyRegistriesUrl = (
yarnUrl: string,
npmUrl: string,
originalYarnUrl: string,
originalNpmUrl: string
) => {
2019-04-25 11:44:33 +02:00
logger.log(`↪️ changing system config`);
nodeCleanup(() => {
registriesUrl(originalYarnUrl, originalNpmUrl);
logger.log(dedent`
Your registry config has been restored from:
npm: ${npmUrl} to ${originalNpmUrl}
yarn: ${yarnUrl} to ${originalYarnUrl}
`);
});
return registriesUrl(yarnUrl, npmUrl);
};
const currentVersion = async () => {
const { version } = (await import('../lerna.json')).default;
return version;
};
2020-05-18 14:58:55 +02:00
const publish = (packages: { name: string; location: string }[], url: string) => {
logger.log(`Publishing packages with a concurrency of ${maxConcurrentTasks}`);
const limit = pLimit(maxConcurrentTasks);
2020-09-14 22:20:57 +02:00
let i = 0;
2020-05-09 11:34:50 +02:00
return Promise.all(
packages.map(({ name, location }) =>
limit(
() =>
new Promise((res, rej) => {
logger.log(`🛫 publishing ${name} (${location})`);
const command = `cd ${location} && npm publish --registry ${url} --force --access restricted --ignore-scripts`;
2020-05-09 11:34:50 +02:00
exec(command, (e) => {
if (e) {
rej(e);
} else {
2020-09-14 22:20:57 +02:00
i += 1;
logger.log(`${i}/${packages.length} 🛬 successful publish of ${name}!`);
2020-11-26 10:58:09 +01:00
res(undefined);
2020-05-09 11:34:50 +02:00
}
});
})
)
)
);
};
const run = async () => {
2020-05-09 11:34:50 +02:00
const port = await freePort(program.port);
2019-04-25 11:44:33 +02:00
logger.log(`🌏 found a open port: ${port}`);
2020-05-09 11:34:50 +02:00
const verdaccioUrl = `http://localhost:${port}`;
2019-04-25 11:44:33 +02:00
logger.log(`🔖 reading current registry settings`);
let [originalYarnRegistryUrl, originalNpmRegistryUrl] = await registriesUrl();
if (
originalYarnRegistryUrl.includes('localhost') ||
originalNpmRegistryUrl.includes('localhost')
) {
originalYarnRegistryUrl = 'https://registry.npmjs.org/';
originalNpmRegistryUrl = 'https://registry.npmjs.org/';
}
2019-04-25 11:44:33 +02:00
logger.log(`📐 reading version of storybook`);
logger.log(`🚛 listing storybook packages`);
2020-05-09 11:34:50 +02:00
logger.log(`🎬 starting verdaccio (this takes ±5 seconds, so be patient)`);
2020-05-18 22:29:50 +02:00
const [verdaccioServer, packages, version] = await Promise.all<any, Package[], string>([
startVerdaccio(port),
listOfPackages(),
currentVersion(),
]);
2019-04-25 11:44:33 +02:00
logger.log(`🌿 verdaccio running on ${verdaccioUrl}`);
2020-05-09 11:34:50 +02:00
await applyRegistriesUrl(
verdaccioUrl,
verdaccioUrl,
originalYarnRegistryUrl,
originalNpmRegistryUrl
);
2020-05-09 11:34:50 +02:00
// await addUser(verdaccioUrl);
2020-05-09 11:34:50 +02:00
logger.log(`📦 found ${packages.length} storybook packages at version ${chalk.blue(version)}`);
2020-05-09 11:34:50 +02:00
if (program.publish) {
await publish(packages, verdaccioUrl);
}
2020-05-09 11:34:50 +02:00
if (!program.open) {
verdaccioServer.close();
2019-04-25 11:44:33 +02:00
}
};
2020-05-09 11:34:50 +02:00
run().catch((e) => {
logger.error(e);
process.exit(1);
});