Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Addon Test: Prompt switch to experimental-nextjs-vite #29814

Merged
merged 16 commits into from
Dec 10, 2024
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 0 additions & 1 deletion code/addons/test/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -85,7 +85,6 @@
"@storybook/icons": "^1.2.12",
"@storybook/instrumenter": "workspace:*",
"@storybook/test": "workspace:*",
"@storybook/theming": "workspace:*",
"polished": "^4.2.2",
"prompts": "^2.4.0",
"ts-dedent": "^2.2.0"
Expand Down
72 changes: 63 additions & 9 deletions code/addons/test/src/postinstall.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,13 +2,17 @@ import { existsSync } from 'node:fs';
import * as fs from 'node:fs/promises';
import { writeFile } from 'node:fs/promises';

import type { types } from 'storybook/internal/babel';
import { traverse } from 'storybook/internal/babel';
import {
JsPackageManagerFactory,
extractProperFrameworkName,
loadAllPresets,
loadMainConfig,
serverResolve,
validateFrameworkName,
} from 'storybook/internal/common';
import { readConfig, writeConfig } from 'storybook/internal/csf-tools';
import { colors, logger } from 'storybook/internal/node-logger';

// eslint-disable-next-line depend/ban-dependencies
Expand Down Expand Up @@ -57,6 +61,52 @@ export default async function postInstall(options: PostinstallOptions) {
// if Vitest is installed, we use the same version to keep consistency across Vitest packages
const vitestVersionToInstall = vitestVersionSpecifier ?? 'latest';

const mainJsPath = serverResolve(resolve(options.configDir, 'main')) as string;
const config = await readConfig(mainJsPath);

const hasCustomWebpackConfig = !!config.getFieldNode(['webpackFinal']);

if (info.frameworkPackageName === '@storybook/nextjs' && !hasCustomWebpackConfig) {
const out = options.yes
? {
migrateToExperimentalNextjsVite: true,
}
: await prompts({
type: 'confirm',
name: 'migrateToExperimentalNextjsVite',
message: dedent`
The addon requires the use of @storybook/experimental-nextjs-vite to work with Next.js.
https://storybook.js.org/docs/writing-tests/test-addon#install-and-set-up

Do you want to migrate?
`,
initial: true,
});

if (out.migrateToExperimentalNextjsVite) {
console.log('running migrateToExperimentalNextjsVite');
ndelangen marked this conversation as resolved.
Show resolved Hide resolved
await packageManager.addDependencies({ installAsDevDependencies: true }, [
'@storybook/experimental-nextjs-vite',
ndelangen marked this conversation as resolved.
Show resolved Hide resolved
]);

await packageManager.removeDependencies({}, ['@storybook/nextjs']);

// eslint-disable-next-line no-underscore-dangle
traverse(config._ast, {
StringLiteral(path) {
if (path.node.value === '@storybook/nextjs') {
path.node.value = '@storybook/experimental-nextjs-vite';
}
},
});

await writeConfig(config, mainJsPath);

info.frameworkPackageName = '@storybook/experimental-nextjs-vite';
info.builderPackageName = '@storybook/builder-vite';
}
}

const annotationsImport = [
'@storybook/nextjs',
'@storybook/experimental-nextjs-vite',
Expand All @@ -71,11 +121,16 @@ export default async function postInstall(options: PostinstallOptions) {
)
? info.rendererPackageName
: null;

const isRendererSupported = !!annotationsImport;

const prerequisiteCheck = async () => {
const reasons = [];

if (hasCustomWebpackConfig) {
reasons.push('• The addon can not be used with a custom Webpack configuration.');
}

if (
info.frameworkPackageName !== '@storybook/nextjs' &&
info.builderPackageName !== '@storybook/builder-vite'
Expand Down Expand Up @@ -149,9 +204,7 @@ export default async function postInstall(options: PostinstallOptions) {
return;
}

const interactionsAddon = info.addons.find((addon) => addon.includes(addonInteractionsName));

if (!!interactionsAddon) {
if (info.hasAddonInteractions) {
let shouldUninstall = options.yes;
if (!options.yes) {
printInfo(
Expand Down Expand Up @@ -188,10 +241,9 @@ export default async function postInstall(options: PostinstallOptions) {
],
{
shell: true,
stdio: 'inherit',
}
);
} else {
return;
}
}

Expand Down Expand Up @@ -267,7 +319,7 @@ export default async function postInstall(options: PostinstallOptions) {
logger.plain(colors.gray(` ${vitestSetupFile}`));

const previewExists = EXTENSIONS.map((ext) => resolve(options.configDir, `preview${ext}`)).some(
(config) => existsSync(config)
existsSync
);

const a11yAddon = info.addons.find((addon) => addon.includes(addonA11yName));
Expand Down Expand Up @@ -474,7 +526,7 @@ const getVitestPluginInfo = (framework: string) => {
frameworkPluginCall = 'storybookReactNativeWeb()';
}

// spaces for file identation
// spaces for file indentation
frameworkPluginImport = `\n${frameworkPluginImport}`;
frameworkPluginDocs = frameworkPluginDocs ? `\n ${frameworkPluginDocs}` : '';
frameworkPluginCall = frameworkPluginCall ? `\n ${frameworkPluginCall},` : '';
Expand All @@ -492,7 +544,6 @@ async function getStorybookInfo({ configDir, packageManager: pkgMgr }: Postinsta
const frameworkName = typeof framework === 'string' ? framework : framework?.name;
validateFrameworkName(frameworkName);
const frameworkPackageName = extractProperFrameworkName(frameworkName);
const addons = getAddonNames(config);

const presets = await loadAllPresets({
corePresets: [join(frameworkName, 'preset')],
Expand All @@ -504,6 +555,8 @@ async function getStorybookInfo({ configDir, packageManager: pkgMgr }: Postinsta
isCritical: true,
});

const hasAddonInteractions = !!(await presets.apply('ADDON_INTERACTIONS_IN_USE', false));

const core = await presets.apply('core', {});

const { builder, renderer } = core;
Expand Down Expand Up @@ -531,6 +584,7 @@ async function getStorybookInfo({ configDir, packageManager: pkgMgr }: Postinsta
frameworkPackageName,
builderPackageName,
rendererPackageName,
addons,
hasAddonInteractions,
addons: getAddonNames(config),
};
}
3 changes: 1 addition & 2 deletions code/addons/test/src/utils.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import { type StorybookTheme, useTheme } from 'storybook/internal/theming';

import type { StorybookConfig } from '@storybook/types';
import type { StorybookConfig } from 'storybook/internal/types';

import Filter from 'ansi-to-html';
import stripAnsi from 'strip-ansi';
Expand Down
1 change: 0 additions & 1 deletion code/yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -6403,7 +6403,6 @@ __metadata:
"@storybook/icons": "npm:^1.2.12"
"@storybook/instrumenter": "workspace:*"
"@storybook/test": "workspace:*"
"@storybook/theming": "workspace:*"
"@types/istanbul-lib-report": "npm:^3.0.3"
"@types/node": "npm:^22.0.0"
"@types/semver": "npm:^7"
Expand Down
Loading