storybook/scripts/run-registry.ts
2025-01-28 16:17:39 +01:00

224 lines
6.9 KiB
TypeScript
Executable File

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<typeof http.IncomingMessage, typeof http.ServerResponse>;
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<Server>;
};
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);
});
});