mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-06 07:21:16 +08:00
290 lines
7.0 KiB
TypeScript
290 lines
7.0 KiB
TypeScript
import { dirname, join, parse } from 'path';
|
|
import fs from 'fs-extra';
|
|
import express from 'express';
|
|
|
|
import { logger } from '@storybook/node-logger';
|
|
|
|
import { globalExternals } from '@fal-works/esbuild-plugin-global-externals';
|
|
import { pnpPlugin } from '@yarnpkg/esbuild-plugin-pnp';
|
|
import aliasPlugin from 'esbuild-plugin-alias';
|
|
|
|
import { getTemplatePath, renderHTML } from './utils/template';
|
|
import { definitions } from './utils/globals';
|
|
import type {
|
|
BuilderBuildResult,
|
|
BuilderFunction,
|
|
BuilderStartOptions,
|
|
BuilderStartResult,
|
|
Compilation,
|
|
ManagerBuilder,
|
|
StarterFunction,
|
|
} from './types';
|
|
// eslint-disable-next-line import/no-cycle
|
|
import { getData } from './utils/data';
|
|
import { safeResolve } from './utils/safeResolve';
|
|
import { readOrderedFiles } from './utils/files';
|
|
|
|
let compilation: Compilation;
|
|
let asyncIterator: ReturnType<StarterFunction> | ReturnType<BuilderFunction>;
|
|
|
|
export const getConfig: ManagerBuilder['getConfig'] = async (options) => {
|
|
const [addonsEntryPoints, customManagerEntryPoint, tsconfigPath] = await Promise.all([
|
|
options.presets.apply('managerEntries', []),
|
|
safeResolve(join(options.configDir, 'manager')),
|
|
getTemplatePath('addon.tsconfig.json'),
|
|
]);
|
|
|
|
return {
|
|
entryPoints: customManagerEntryPoint
|
|
? [...addonsEntryPoints, customManagerEntryPoint]
|
|
: addonsEntryPoints,
|
|
outdir: join(options.outputDir || './', 'sb-addons'),
|
|
format: 'esm',
|
|
write: false,
|
|
resolveExtensions: ['.ts', '.tsx', '.mjs', '.js', '.jsx'],
|
|
outExtension: { '.js': '.mjs' },
|
|
loader: {
|
|
'.js': 'jsx',
|
|
'.png': 'dataurl',
|
|
'.gif': 'dataurl',
|
|
'.jpg': 'dataurl',
|
|
'.jpeg': 'dataurl',
|
|
'.svg': 'dataurl',
|
|
'.webp': 'dataurl',
|
|
'.webm': 'dataurl',
|
|
},
|
|
target: ['chrome100'],
|
|
platform: 'browser',
|
|
bundle: true,
|
|
minify: true,
|
|
sourcemap: true,
|
|
conditions: ['browser', 'module', 'default'],
|
|
|
|
jsxFactory: 'React.createElement',
|
|
jsxFragment: 'React.Fragment',
|
|
jsx: 'transform',
|
|
jsxImportSource: 'react',
|
|
|
|
tsconfig: tsconfigPath,
|
|
|
|
legalComments: 'external',
|
|
plugins: [
|
|
aliasPlugin({
|
|
process: require.resolve('process/browser.js'),
|
|
util: require.resolve('util/util.js'),
|
|
assert: require.resolve('browser-assert'),
|
|
}),
|
|
globalExternals(definitions),
|
|
pnpPlugin(),
|
|
],
|
|
|
|
banner: {
|
|
js: 'try{',
|
|
},
|
|
footer: {
|
|
js: '}catch(e){ console.log("ONE OF YOUR MANAGER-ENTRIES FAILED: " + import.meta.url) }',
|
|
},
|
|
|
|
define: {
|
|
'process.env.NODE_ENV': "'production'",
|
|
'process.env': '{}',
|
|
global: 'window',
|
|
module: '{}',
|
|
},
|
|
};
|
|
};
|
|
|
|
export const executor = {
|
|
get: async () => {
|
|
const { build } = await import('esbuild');
|
|
return build;
|
|
},
|
|
};
|
|
|
|
/**
|
|
* This function is a generator so that we can abort it mid process
|
|
* in case of failure coming from other processes e.g. preview builder
|
|
*
|
|
* I am sorry for making you read about generators today :')
|
|
*/
|
|
const starter: StarterFunction = async function* starterGeneratorFn({
|
|
startTime,
|
|
options,
|
|
router,
|
|
}) {
|
|
logger.info('=> Starting manager..');
|
|
|
|
const { config, customHead, features, instance, refs, template, title, logLevel, docsOptions } =
|
|
await getData(options);
|
|
|
|
yield;
|
|
|
|
// make sure we clear output directory of addons dir before starting
|
|
// this could cause caching issues where addons are loaded when they shouldn't
|
|
const addonsDir = config.outdir;
|
|
await fs.remove(addonsDir);
|
|
|
|
yield;
|
|
|
|
compilation = await instance({
|
|
...config,
|
|
watch: true,
|
|
});
|
|
|
|
yield;
|
|
|
|
const coreDirOrigin = join(dirname(require.resolve('@storybook/manager/package.json')), 'dist');
|
|
|
|
router.use(`/sb-addons`, express.static(addonsDir, { immutable: true, maxAge: '5m' }));
|
|
router.use(`/sb-manager`, express.static(coreDirOrigin, { immutable: true, maxAge: '5m' }));
|
|
|
|
const { cssFiles, jsFiles } = await readOrderedFiles(addonsDir, compilation?.outputFiles);
|
|
|
|
yield;
|
|
|
|
const html = await renderHTML(
|
|
template,
|
|
title,
|
|
customHead,
|
|
cssFiles,
|
|
jsFiles,
|
|
features,
|
|
refs,
|
|
logLevel,
|
|
docsOptions,
|
|
options
|
|
);
|
|
|
|
yield;
|
|
|
|
router.use(`/`, ({ path }, res, next) => {
|
|
if (path === '/') {
|
|
res.status(200).send(html);
|
|
} else {
|
|
next();
|
|
}
|
|
});
|
|
|
|
return {
|
|
bail,
|
|
stats: {
|
|
toJson: () => ({}),
|
|
},
|
|
totalTime: process.hrtime(startTime),
|
|
} as BuilderStartResult;
|
|
};
|
|
|
|
/**
|
|
* This function is a generator so that we can abort it mid process
|
|
* in case of failure coming from other processes e.g. preview builder
|
|
*
|
|
* I am sorry for making you read about generators today :')
|
|
*/
|
|
const builder: BuilderFunction = async function* builderGeneratorFn({ startTime, options }) {
|
|
if (!options.outputDir) {
|
|
throw new Error('outputDir is required');
|
|
}
|
|
logger.info('=> Building manager..');
|
|
const { config, customHead, features, instance, refs, template, title, logLevel, docsOptions } =
|
|
await getData(options);
|
|
yield;
|
|
|
|
const addonsDir = config.outdir;
|
|
const coreDirOrigin = join(dirname(require.resolve('@storybook/manager/package.json')), 'dist');
|
|
const coreDirTarget = join(options.outputDir, `sb-manager`);
|
|
|
|
compilation = await instance({
|
|
...config,
|
|
|
|
minify: true,
|
|
watch: false,
|
|
});
|
|
|
|
yield;
|
|
|
|
const managerFiles = fs.copy(coreDirOrigin, coreDirTarget, {
|
|
filter: (src) => {
|
|
const { ext } = parse(src);
|
|
if (ext) {
|
|
return ext === '.mjs';
|
|
}
|
|
return true;
|
|
},
|
|
});
|
|
const { cssFiles, jsFiles } = await readOrderedFiles(addonsDir, compilation?.outputFiles);
|
|
|
|
yield;
|
|
|
|
const html = await renderHTML(
|
|
template,
|
|
title,
|
|
customHead,
|
|
cssFiles,
|
|
jsFiles,
|
|
features,
|
|
refs,
|
|
logLevel,
|
|
docsOptions,
|
|
options
|
|
);
|
|
|
|
await Promise.all([
|
|
//
|
|
fs.writeFile(join(options.outputDir, 'index.html'), html),
|
|
managerFiles,
|
|
]);
|
|
|
|
logger.trace({ message: '=> Manager built', time: process.hrtime(startTime) });
|
|
|
|
return {
|
|
toJson: () => ({}),
|
|
} as BuilderBuildResult;
|
|
};
|
|
|
|
export const bail: ManagerBuilder['bail'] = async () => {
|
|
if (asyncIterator) {
|
|
try {
|
|
// we tell the builder (that started) to stop ASAP and wait
|
|
await asyncIterator.throw(new Error());
|
|
} catch (e) {
|
|
//
|
|
}
|
|
}
|
|
|
|
if (compilation && compilation.stop) {
|
|
try {
|
|
compilation.stop();
|
|
logger.warn('Force closed manager build');
|
|
} catch (err) {
|
|
logger.warn('Unable to close manager build!');
|
|
}
|
|
}
|
|
};
|
|
|
|
export const start = async (options: BuilderStartOptions) => {
|
|
asyncIterator = starter(options);
|
|
let result;
|
|
|
|
do {
|
|
// eslint-disable-next-line no-await-in-loop
|
|
result = await asyncIterator.next();
|
|
} while (!result.done);
|
|
|
|
return result.value;
|
|
};
|
|
|
|
export const build = async (options: BuilderStartOptions) => {
|
|
asyncIterator = builder(options);
|
|
let result;
|
|
|
|
do {
|
|
// eslint-disable-next-line no-await-in-loop
|
|
result = await asyncIterator.next();
|
|
} while (!result.done);
|
|
|
|
return result.value;
|
|
};
|
|
|
|
export const corePresets: ManagerBuilder['corePresets'] = [];
|
|
export const overridePresets: ManagerBuilder['overridePresets'] = [];
|