storybook/scripts/utils/cli-step.ts
2025-02-19 12:33:27 +01:00

138 lines
3.5 KiB
TypeScript

import { createRequire } from 'module';
import { exec } from './exec';
import type { OptionSpecifier, OptionValues } from './options';
import { createOptions, getCommand } from './options';
const require = createRequire(import.meta.url);
const cliExecutable = require.resolve('../../code/core/bin/index.cjs');
const toolboxExecutable = require.resolve('../../code/lib/cli-storybook/bin/index.cjs');
const createStorybookExecutable = require.resolve('../../code/lib/create-storybook/bin/index.cjs');
export type CLIStep<TOptions extends OptionSpecifier> = {
command: string;
description: string;
hasArgument?: boolean;
icon: string;
// It would be kind of great to be able to share these with `lib/cli/src/generate.ts`
options: TOptions;
};
export const steps = {
repro: {
command: 'repro',
description: 'Bootstrapping Template',
icon: '👷',
hasArgument: true,
options: createOptions({
output: { type: 'string' },
// TODO allow default values for strings
branch: { type: 'string', values: ['main', 'next'] },
init: { type: 'boolean', inverse: true },
debug: { type: 'boolean' },
}),
},
init: {
command: 'init',
description: 'Initializing Storybook',
icon: '⚙️',
options: createOptions({
yes: { type: 'boolean' },
type: { type: 'string' },
debug: { type: 'boolean' },
'skip-install': { type: 'boolean' },
}),
},
add: {
command: 'add',
description: 'Adding addon',
icon: '+',
hasArgument: true,
options: createOptions({
yes: { type: 'boolean' },
}),
},
link: {
command: 'link',
description: 'Linking packages',
icon: '🔗',
hasArgument: true,
options: createOptions({
local: { type: 'boolean' },
start: { type: 'boolean', inverse: true },
}),
},
build: {
command: 'build',
description: 'Building Storybook',
icon: '🔨',
options: createOptions({}),
},
dev: {
command: 'dev',
description: 'Starting Storybook',
icon: '🖥 ',
options: createOptions({}),
},
migrate: {
command: 'migrate',
hasArgument: true,
description: 'Run codemods',
icon: '🚀',
options: createOptions({
glob: { type: 'string' },
}),
},
automigrate: {
command: 'automigrate',
hasArgument: true,
description: 'Run automigrations',
icon: '🤖',
options: createOptions({}),
},
};
export async function executeCLIStep<TOptions extends OptionSpecifier>(
cliStep: CLIStep<TOptions>,
options: {
argument?: string;
optionValues?: Partial<OptionValues<TOptions>>;
cwd: string;
dryRun?: boolean;
debug: boolean;
}
) {
if (cliStep.hasArgument && !options.argument) {
throw new Error(`Argument required for ${cliStep.command} command.`);
}
const cliCommand = cliStep.command;
const prefix = ['dev', 'build'].includes(cliCommand)
? `node ${cliExecutable} ${cliCommand}`
: cliCommand === 'init'
? `node ${createStorybookExecutable} ${cliCommand}`
: `node ${toolboxExecutable} ${cliCommand}`;
const command = getCommand(
cliStep.hasArgument ? `${prefix} ${options.argument}` : prefix,
cliStep.options,
options.optionValues || {}
);
await exec(
command,
{
cwd: options.cwd,
env: {
STORYBOOK_DISABLE_TELEMETRY: 'true',
},
},
{
startMessage: `${cliStep.icon} ${cliStep.description}`,
errorMessage: `🚨 ${cliStep.description} failed`,
dryRun: options.dryRun,
debug: options.debug,
}
);
}