import { exec } from 'node:child_process'; import { mkdir, rm } from 'node:fs/promises'; import http from 'node:http'; import type { Server } from 'node:http'; import { join, resolve as resolvePath } from 'node:path'; import { program } from 'commander'; // eslint-disable-next-line depend/ban-dependencies import { execa } from 'execa'; // eslint-disable-next-line depend/ban-dependencies import { pathExists, readJSON, remove } from 'fs-extra'; import pLimit from 'p-limit'; import picocolors from 'picocolors'; import { parseConfigFile, runServer } from 'verdaccio'; import { maxConcurrentTasks } from './utils/concurrency'; import { PACKS_DIRECTORY } from './utils/constants'; import { getWorkspaces } from './utils/workspace'; program .option('-O, --open', 'keep process open') .option('-P, --publish', 'should publish packages'); program.parse(process.argv); const logger = console; const root = resolvePath(__dirname, '..'); const opts = program.opts(); const startVerdaccio = async () => { const ready = { proxy: false, verdaccio: false, }; return Promise.race([ new Promise((resolve) => { /** * The proxy server will sit in front of verdaccio and tunnel traffic to either verdaccio or * the actual npm global registry We do this because tunneling all traffic through verdaccio * is slow (this might get fixed in verdaccio) With this heuristic we get the best of both * worlds: * * - Verdaccio for storybook packages (including unscoped packages such as `storybook` and `sb`) * - Npm global registry for all other packages * - The best performance for both * * The proxy server listens on port 6001 and verdaccio on port 6002 * * If you want to access the verdaccio UI, you can do so by visiting http://localhost:6002 */ const proxy = http.createServer((req, res) => { // if request contains "storybook" redirect to verdaccio if (req.url?.includes('storybook') || req.url?.includes('/sb') || req.method === 'PUT') { res.writeHead(302, { Location: 'http://localhost:6002' + req.url }); res.end(); } else { // forward to npm registry res.writeHead(302, { Location: 'https://registry.npmjs.org' + req.url }); res.end(); } }); let verdaccioApp: http.Server; proxy.listen(6001, () => { ready.proxy = true; if (ready.verdaccio) { resolve(verdaccioApp); } }); const cache = join(__dirname, '..', '.verdaccio-cache'); const config = { ...parseConfigFile(join(__dirname, 'verdaccio.yaml')), self_path: cache, }; // @ts-expect-error (verdaccio's interface is wrong) runServer(config).then((app: Server) => { verdaccioApp = app; app.listen(6002, () => { ready.verdaccio = true; if (ready.proxy) { resolve(verdaccioApp); } }); }); }), new Promise((_, rej) => { setTimeout(() => { if (!ready.verdaccio || !ready.proxy) { rej(new Error(`TIMEOUT - verdaccio didn't start within 10s`)); } }, 10000); }), ]) as Promise; }; const currentVersion = async () => { const { version } = await readJSON(join(__dirname, '..', 'code', 'package.json')); return version; }; const publish = async (packages: { name: string; location: string }[], url: string) => { logger.log(`Publishing packages with a concurrency of ${maxConcurrentTasks}`); const limit = pLimit(maxConcurrentTasks); let i = 0; /** * We need to "pack" our packages before publishing to npm because our package.json files contain * yarn specific version "ranges". such as "workspace:*" * * We can't publish to npm if the package.json contains these ranges. So with `yarn pack` we * create a tarball that we can publish. * * However this bug exists in NPM: https://github.com/npm/cli/issues/4533! Which causes the NPM * CLI to disregard the tarball CLI argument and instead re-create a tarball. But NPM doesn't * replace the yarn version ranges. * * So we create the tarball ourselves and move it to another location on the FS. Then we * change-directory to that directory and publish the tarball from there. */ await mkdir(PACKS_DIRECTORY, { recursive: true }).catch(() => {}); return Promise.all( packages.map(({ name, location }) => limit( () => new Promise((res, rej) => { logger.log( `🛫 publishing ${name} (${location.replace(resolvePath(join(__dirname, '..')), '.')})` ); const tarballFilename = `${name.replace('@', '').replace('/', '-')}.tgz`; const command = `cd ${resolvePath( '../code', location )} && yarn pack --out=${PACKS_DIRECTORY}/${tarballFilename} && cd ${PACKS_DIRECTORY} && npm publish ./${tarballFilename} --registry ${url} --force --tag="xyz" --ignore-scripts`; exec(command, (e) => { if (e) { rej(e); } else { i += 1; logger.log(`${i}/${packages.length} 🛬 successful publish of ${name}!`); res(undefined); } }); }) ) ) ); }; const run = async () => { const verdaccioUrl = `http://localhost:6001`; logger.log(`📐 reading version of storybook`); logger.log(`🚛 listing storybook packages`); if (!process.env.CI) { // when running e2e locally, clear cache to avoid EPUBLISHCONFLICT errors const verdaccioCache = resolvePath(__dirname, '..', '.verdaccio-cache'); if (await pathExists(verdaccioCache)) { logger.log(`🗑 cleaning up cache`); await remove(verdaccioCache); } } logger.log(`🎬 starting verdaccio (this takes ±5 seconds, so be patient)`); const [verdaccioServer, packages, version] = await Promise.all([ startVerdaccio(), getWorkspaces(false), currentVersion(), ]); logger.log(`🌿 verdaccio running on ${verdaccioUrl}`); logger.log(`👤 add temp user to verdaccio`); await execa( 'npx', // creates a .npmrc file in the root directory of the project [ 'npm-auth-to-token', '-u', 'foo', '-p', 's3cret', '-e', 'test@test.com', '-r', 'http://localhost:6002', ], { cwd: root, } ); logger.log( `📦 found ${packages.length} storybook packages at version ${picocolors.blue(version)}` ); if (opts.publish) { await publish(packages, 'http://localhost:6002'); } await rm(join(root, '.npmrc'), { force: true }); if (!opts.open) { verdaccioServer.close(); process.exit(0); } }; run().catch((e) => { logger.error(e); rm(join(root, '.npmrc'), { force: true }).then(() => { process.exit(1); }); });