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 = { 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( cliStep: CLIStep, options: { argument?: string; optionValues?: Partial>; 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, } ); }