Builder-vite: Fix storysource support

This commit is contained in:
Michael Shilman 2023-02-15 12:17:34 +08:00
parent 5ed049fb8d
commit ed975bb233
5 changed files with 111 additions and 0 deletions

View File

@ -52,6 +52,7 @@
"@storybook/node-logger": "7.0.0-beta.47",
"@storybook/preview": "7.0.0-beta.47",
"@storybook/preview-api": "7.0.0-beta.47",
"@storybook/source-loader": "7.0.0-beta.47",
"@storybook/types": "7.0.0-beta.47",
"browser-assert": "^1.2.1",
"es-module-lexer": "^0.9.3",

View File

@ -4,3 +4,4 @@ export * from './strip-story-hmr-boundaries';
export * from './code-generator-plugin';
export * from './csf-plugin';
export * from './external-globals-plugin';
export * from './source-loader-plugin';

View File

@ -0,0 +1,106 @@
import type { Plugin } from 'vite';
import sourceLoaderTransform from '@storybook/source-loader';
import MagicString from 'magic-string';
import type { ExtendedOptions } from '../types';
const storyPattern = /\.stories\.[jt]sx?$/;
const storySourcePattern = /var __STORY__ = "(.*)"/;
const storySourceReplacement = '--STORY_SOURCE_REPLACEMENT--';
const mockClassLoader = (id: string) => ({
// eslint-disable-next-line no-console
emitWarning: (message: string) => console.warn(message),
resourcePath: id,
getOptions: () => ({ injectStoryParameters: true }),
extension: `.${id.split('.').pop()}`,
});
// HACK: Until we can support only node 15+ and use string.prototype.replaceAll
const replaceAll = (str: string, search: string, replacement: string) => {
return str.split(search).join(replacement);
};
export function sourceLoaderPlugin(config: ExtendedOptions): Plugin | Plugin[] {
if (config.configType === 'DEVELOPMENT') {
return {
name: 'storybook:source-loader-plugin',
enforce: 'pre',
async transform(src: string, id: string) {
if (id.match(storyPattern)) {
const code: string = await sourceLoaderTransform.call(mockClassLoader(id), src);
const s = new MagicString(src);
// Entirely replace with new code
s.overwrite(0, src.length, code);
return {
code: s.toString(),
map: s.generateMap({ hires: true, source: id }),
};
}
return undefined;
},
};
}
// In production, we need to be fancier, to avoid vite:define plugin from replacing values inside the `__STORY__` string
const storySources = new WeakMap<ExtendedOptions, Map<string, string>>();
return [
{
name: 'storybook-vite-source-loader-plugin',
enforce: 'pre',
buildStart() {
storySources.set(config, new Map());
},
async transform(src: string, id: string) {
if (id.match(storyPattern)) {
let code: string = await sourceLoaderTransform.call(mockClassLoader(id), src);
// eslint-disable-next-line @typescript-eslint/naming-convention
const [_, sourceString] = code.match(storySourcePattern) ?? [null, null];
if (sourceString) {
const map = storySources.get(config);
map?.set(id, sourceString);
// Remove story source so that it is not processed by vite:define plugin
code = replaceAll(code, sourceString, storySourceReplacement);
}
const s = new MagicString(src);
// Entirely replace with new code
s.overwrite(0, src.length, code);
return {
code: s.toString(),
map: s.generateMap(),
};
}
return undefined;
},
},
{
name: 'storybook-vite-source-loader-plugin-post',
enforce: 'post',
buildStart() {
storySources.set(config, new Map());
},
async transform(src: string, id: string) {
if (id.match(storyPattern)) {
const s = new MagicString(src);
const map = storySources.get(config);
const storySourceStatement = map?.get(id);
// Put the previously-extracted source back in
if (storySourceStatement) {
const newCode = replaceAll(src, storySourceReplacement, storySourceStatement);
s.overwrite(0, src.length, newCode);
}
return {
code: s.toString(),
map: s.generateMap(),
};
}
return undefined;
},
},
];
}

View File

@ -17,6 +17,7 @@ import {
mdxPlugin,
stripStoryHMRBoundary,
externalGlobalsPlugin,
sourceLoaderPlugin,
} from './plugins';
import type { BuilderOptions } from './types';
@ -77,6 +78,7 @@ export async function pluginConfig(options: Options) {
const plugins = [
codeGeneratorPlugin(options),
sourceLoaderPlugin(options),
await csfPlugin(options),
await mdxPlugin(options),
injectExportOrderPlugin,

View File

@ -5195,6 +5195,7 @@ __metadata:
"@storybook/node-logger": 7.0.0-beta.47
"@storybook/preview": 7.0.0-beta.47
"@storybook/preview-api": 7.0.0-beta.47
"@storybook/source-loader": 7.0.0-beta.47
"@storybook/types": 7.0.0-beta.47
"@types/express": ^4.17.13
"@types/node": ^16.0.0