文档
Storybook 文档

Test runner

Storybook test runner 将您的所有 story 转换为可执行的测试。它由 JestPlaywright 驱动。

  • 对于那些没有 play 函数的用户:它验证 story 是否在没有任何错误的情况下渲染。
  • 对于那些有 play 函数的用户:它还会检查 play 函数中的错误以及所有断言是否通过。

这些测试在实时浏览器中运行,可以通过命令行或您的 CI 服务器执行。

设置

test-runner 是一个独立的、与框架无关的实用程序,与您的 Storybook 并行运行。您需要采取一些额外的步骤来正确设置它。下面详细介绍了我们配置和执行它的建议。

运行以下命令进行安装。

npm install @storybook/test-runner --save-dev

更新您的 package.json 脚本并启用 test runner。

package.json
{
  "scripts": {
    "test-storybook": "test-storybook"
  }
}

使用以下命令启动您的 Storybook

npm run storybook

Storybook 的 test runner 需要本地运行的 Storybook 实例或已发布的 Storybook 才能运行所有现有测试。

最后,打开一个新的终端窗口并使用以下命令运行 test-runner

npm run test-storybook

配置

Test runner 为 Storybook 提供零配置支持。但是,您可以运行 test-storybook --eject 以获得更细粒度的控制。它会在项目的根目录生成一个 test-runner-jest.config.js 文件,您可以修改该文件。此外,您可以扩展生成的配置文件并提供 testEnvironmentOptions,因为 test runner 在底层也使用了 jest-playwright

CLI 选项

Test-runner 由 Jest 驱动,并接受其 CLI 选项 的子集(例如,--watch--maxWorkers)。如果您已经在项目中使用这些标志中的任何一个,您应该能够将它们迁移到 Storybook 的 test-runner 中,而不会有任何问题。下面列出了所有可用的标志以及使用它们的示例。

选项描述
--help输出使用信息
test-storybook --help
-s, --index-json在 index json 模式下运行。自动检测(需要兼容的 Storybook)
test-storybook --index-json
--no-index-json禁用 index json 模式
test-storybook --no-index-json
-c, --config-dir [目录名]从中加载 Storybook 配置的目录
test-storybook -c .storybook
--watch在 watch 模式下运行
test-storybook --watch
--watchAll监视文件更改,并在发生更改时重新运行所有测试。
test-storybook --watchAll
--coverage对您的 stories 和组件运行覆盖率测试
test-storybook --coverage
--coverageDirectory写入覆盖率报告输出的目录
test-storybook --coverage --coverageDirectory coverage/ui/storybook
--url定义在其中运行测试的 URL。对于自定义 Storybook URL 很有用
test-storybook --url http://the-storybook-url-here.com
--browsers定义在其中运行测试的浏览器。一个或多个:chromium、firefox、webkit
test-storybook --browsers firefox chromium
--maxWorkers [数量]指定 worker 池将为运行测试生成的最大 worker 数量
test-storybook --maxWorkers=2
--testTimeout [数量]定义测试可以运行的最大时间(以毫秒为单位),超过此时间将自动标记为失败。对于长时间运行的测试很有用
test-storybook --testTimeout=60000
--no-cache禁用缓存
test-storybook --no-cache
--clearCache删除 Jest 缓存目录,然后退出而不运行测试
test-storybook --clearCache
--verbose使用测试套件层级结构显示各个测试结果
test-storybook --verbose
-u, --updateSnapshot使用此标志重新记录在此测试运行期间失败的每个快照
test-storybook -u
--eject创建一个本地配置文件以覆盖 test-runner 的默认值
test-storybook --eject
--json以 JSON 格式打印测试结果。此模式会将所有其他测试输出和用户消息发送到 stderr。
test-storybook --json
--outputFile当同时指定 --json 选项时,将测试结果写入文件。
test-storybook --json --outputFile results.json
--junit指示应在 junit 文件中报告测试信息。
test-storybook --**junit**
--ci它不会自动存储新快照,而是会使测试失败,并要求使用 --updateSnapshot 运行 Jest。
test-storybook --ci
--shard [索引/计数]需要 CI。将测试套件执行拆分为多台机器
test-storybook --shard=1/8
--failOnConsole使测试在浏览器控制台错误时失败
test-storybook --failOnConsole
--includeTags实验性功能
定义要测试的 stories 子集(如果它们与启用的标签匹配)。
test-storybook --includeTags="test-only, pages"
--excludeTags实验性功能
阻止测试与提供的标签匹配的 stories。
test-storybook --excludeTags="no-tests, tokens"
--skipTags实验性功能
配置 test runner 以跳过运行与提供的标签匹配的 stories 的测试。
test-storybook --skipTags="skip-test, layout"
npm run test-storybook -- --watch

针对已部署的 Storybook 运行测试

默认情况下,test-runner 假定您正在针对本地服务的 Storybook 端口 6006 运行它。如果您想定义一个目标 URL 以针对已部署的 Storybook 运行,您可以使用 --url 标志

npm run test-storybook -- --url https://the-storybook-url-here.com

或者,您可以设置 TARGET_URL 环境变量并运行 test-runner

TARGET_URL=https://the-storybook-url-here.com yarn test-storybook

设置 CI 以运行测试

您还可以配置 test-runner 以在 CI 环境中运行测试。下面记录了一些帮助您入门的方案。

通过 Github Actions 部署针对已部署的 Storybook 运行

如果你正在使用 VercelNetlify 等服务发布你的 Storybook,它们会在 GitHub Actions 中发出 deployment_status 事件。你可以使用它并将 deployment_status.target_url 设置为 TARGET_URL 环境变量。方法如下:

.github/workflows/storybook-tests.yml
name: Storybook Tests
 
on: deployment_status
 
jobs:
  test:
    timeout-minutes: 60
    runs-on: ubuntu-latest
    if: github.event.deployment_status.state == 'success'
    steps:
      - uses: actions/checkout@v4
      - uses: actions/setup-node@v4
        with:
          node-version-file: '.nvmrc'
      - name: Install dependencies
        run: yarn
      - name: Install Playwright
        run: npx playwright install --with-deps
      - name: Run Storybook tests
        run: yarn test-storybook
        env:
          TARGET_URL: '${{ github.event.deployment_status.target_url }}'

为了使此示例正常工作,发布的 Storybook 必须是公开可用的。如果它需要身份验证,我们建议使用下面的方法运行测试服务器。

针对未部署的 Storybook 运行

你可以使用你的 CI 提供商(例如,GitHub Actions, GitLab Pipelines, CircleCI)来构建 Storybook 并针对已构建的 Storybook 运行测试运行器。这是一个依赖于第三方库(即 concurrently, http-server, 和 wait-on)构建 Storybook 并使用测试运行器运行测试的方法。

.github/workflows/storybook-tests.yml
name: 'Storybook Tests'
 
on: push
 
jobs:
  test:
    timeout-minutes: 60
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v4
      - uses: actions/setup-node@v4
        with:
          node-version-file: '.nvmrc'
      - name: Install dependencies
        run: yarn
      - name: Install Playwright
        run: npx playwright install --with-deps
      - name: Build Storybook
        run: yarn build-storybook --quiet
      - name: Serve Storybook and run tests
        run: |
          npx concurrently -k -s first -n "SB,TEST" -c "magenta,blue" \
            "npx http-server storybook-static --port 6006 --silent" \
            "npx wait-on tcp:127.0.0.1:6006 && yarn test-storybook"

默认情况下,Storybook 将构建输出到 storybook-static 目录。如果你正在使用不同的构建目录,你需要相应地调整此方法。

Chromatic 和测试运行器之间有什么区别?

测试运行器是一个通用的测试工具,可以在本地或 CI 上运行,并且可以配置或扩展以运行各种类型的测试。

Chromatic 是一种基于云的服务,无需设置测试运行器即可运行视觉组件测试(以及即将推出的可访问性测试)。它还与你的 git 提供商同步,并管理私有项目的访问控制。

然而,在某些情况下,你可能希望将测试运行器和 Chromatic 结合使用。

  • 在本地使用它,并在你的 CI 上使用 Chromatic。
  • 使用 Chromatic 进行视觉和组件测试,并使用测试运行器运行其他自定义测试。

高级配置

测试钩子 API

测试运行器渲染一个 story 并执行其 play 函数(如果存在)。然而,某些行为无法通过在浏览器中执行的 play 函数来实现。例如,如果你希望测试运行器为你拍摄视觉快照,这可以通过 Playwright/Jest 实现,但必须在 Node 中执行。

测试运行器导出可以全局覆盖的测试钩子,以启用诸如视觉或 DOM 快照之类的用例。这些钩子使你可以在 story 渲染之前和之后访问测试生命周期。下面列出了可用的钩子以及如何使用它们的概述。

钩子描述
准备为测试准备浏览器
async prepare({ page, browserContext, testRunnerConfig }) {}
设置在所有测试运行之前执行一次
setup() {}
访问前在 story 首次被访问并在浏览器中渲染之前执行
async preVisit(page, context) {}
访问后在 story 被访问并完全渲染之后执行
async postVisit(page, context) {}

这些测试钩子是实验性的,可能会发生重大更改。我们鼓励你在 story 的 play 函数中尽可能多地进行测试。

要启用钩子 API,你需要在你的 Storybook 目录中添加一个新的配置文件,并按如下方式设置它们:

.storybook/test-runner.ts
import type { TestRunnerConfig } from '@storybook/test-runner';
 
const config: TestRunnerConfig = {
  // Hook that is executed before the test runner starts running tests
  setup() {
    // Add your configuration here.
  },
  /* Hook to execute before a story is initially visited before being rendered in the browser.
   * The page argument is the Playwright's page object for the story.
   * The context argument is a Storybook object containing the story's id, title, and name.
   */
  async preVisit(page, context) {
    // Add your configuration here.
  },
  /* Hook to execute after a story is visited and fully rendered.
   * The page argument is the Playwright's page object for the story
   * The context argument is a Storybook object containing the story's id, title, and name.
   */
  async postVisit(page, context) {
    // Add your configuration here.
  },
};
 
export default config;

除了 setup 函数外,所有其他函数都异步运行。preVisitpostVisit 函数都包含两个额外的参数:Playwright page 和一个 context 对象,其中包含 story 的 idtitlename

当测试运行器执行时,你现有的测试将经历以下生命周期:

  • setup 函数在所有测试运行之前执行。
  • 生成包含所需信息的 context 对象。
  • Playwright 导航到 story 的页面。
  • preVisit 函数被执行。
  • story 被渲染,并且任何现有的 play 函数都被执行。
  • postVisit 函数被执行。

(实验性)过滤测试

当你在 Storybook 上运行测试运行器时,默认情况下它会测试每个 story。但是,如果你想过滤测试,可以使用 tags 配置选项。Storybook 最初引入此功能是为了为 story 生成自动文档。但它可以进一步扩展,以配置测试运行器根据提供的标签使用类似的配置选项或通过 CLI 标志(例如,--includeTags, --excludeTags, --skipTags)运行测试,这些标志仅在最新的稳定版本(0.15 或更高版本)中可用。下面列出了可用的选项以及如何使用它们的概述。

选项描述
排除阻止与提供的标签匹配的 story 被测试。
包含定义一个 story 子集,仅当它们与启用的标签匹配时才会被测试。
跳过如果 story 与提供的标签匹配,则跳过对其进行测试。
.storybook/test-runner.ts
import type { TestRunnerConfig } from '@storybook/test-runner';
 
const config: TestRunnerConfig = {
  tags: {
    include: ['test-only', 'pages'],
    exclude: ['no-tests', 'tokens'],
    skip: ['skip-test', 'layout'],
  },
};
 
export default config;

使用 CLI 标志运行测试优先于配置文件中提供的选项,并将覆盖配置文件中可用的选项。

禁用测试

如果你想阻止特定的 story 被测试运行器测试,你可以使用自定义标签配置你的 story,将其启用到测试运行器配置文件中,或者使用 --excludeTags CLI 标志运行测试运行器并将它们从测试中排除。当你想要排除尚未准备好进行测试或与你的测试无关的 story 时,这很有用。例如:

MyComponent.stories.ts|tsx
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
 
import { MyComponent } from './MyComponent';
 
const meta: Meta<typeof MyComponent> = {
  component: MyComponent,
  tags: ['no-tests'], // 👈 Provides the `no-tests` tag to all stories in this file
};
 
export default meta;
type Story = StoryObj<typeof MyComponent>;
 
export const ExcludeStory: Story = {
  //👇 Adds the `no-tests` tag to this story to exclude it from the tests when enabled in the test-runner configuration
  tags: ['no-tests'],
};

为 story 的子集运行测试

为了允许测试运行器仅对特定的 story 或 story 子集运行测试,你可以使用自定义标签配置 story,在测试运行器配置文件中启用它,或者使用 --includeTags CLI 标志运行测试运行器并将它们包含在你的测试中。例如,如果你想基于 test-only 标签运行测试,你可以按如下方式调整你的配置:

MyComponent.stories.ts|tsx
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
 
import { MyComponent } from './MyComponent';
 
const meta: Meta<typeof MyComponent> = {
  component: MyComponent,
  tags: ['test-only'], // 👈 Provides the `test-only` tag to all stories in this file
};
 
export default meta;
type Story = StoryObj<typeof MyComponent>;
 
export const IncludeStory: Story = {
  //👇 Adds the `test-only` tag to this story to be included in the tests when enabled in the test-runner configuration
  tags: ['test-only'],
};

应用于组件 story 的标签应在组件级别(使用 meta)或 story 级别完成。 Storybook 不支持跨 story 导入标签,并且无法按预期工作。

跳过测试

如果你想跳过对特定 story 或 story 子集运行测试,你可以使用自定义标签配置你的 story,在测试运行器配置文件中启用它,或者使用 --skipTags CLI 标志运行测试运行器。使用此选项运行测试将导致测试运行器忽略它们并在测试结果中相应地标记它们,表明测试已暂时禁用。例如:

MyComponent.stories.ts|tsx
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
 
import { MyComponent } from './MyComponent';
 
const meta: Meta<typeof MyComponent> = {
  component: MyComponent,
  tags: ['skip-test'], // 👈 Provides the `skip-test` tag to all stories in this file
};
 
export default meta;
type Story = StoryObj<typeof MyComponent>;
 
export const SkipStory: Story = {
  //👇 Adds the `skip-test` tag to this story to allow it to be skipped in the tests when enabled in the test-runner configuration
  tags: ['skip-test'],
};

已部署 Storybook 的身份验证

如果你使用需要身份验证的安全托管提供商来托管你的 Storybook,你可能需要设置 HTTP 标头。这主要是因为测试运行器通过 fetch 请求和 Playwright 检查实例状态及其 story 索引的方式。为此,你可以修改测试运行器配置文件以包含 getHttpHeaders 函数。此函数将 fetch 调用和页面访问的 URL 作为输入,并返回一个包含需要设置的标头的对象。

.storybook/test-runner.ts
import type { TestRunnerConfig } from '@storybook/test-runner';
 
const config: TestRunnerConfig = {
  getHttpHeaders: async (url) => {
    const token = url.includes('prod') ? 'prod-token' : 'dev-token';
    return {
      Authorization: `Bearer ${token}`,
    };
  },
};
 
export default config;

辅助工具

测试运行器导出了一些辅助工具,可以通过访问 Storybook 的内部结构(例如,argsparameters)来使你的测试更具可读性和可维护性。下面列出了可用的辅助工具以及如何使用它们的概述。

.storybook/test-runner.ts
import type { TestRunnerConfig } from '@storybook/test-runner';
import { getStoryContext, waitForPageReady } from '@storybook/test-runner';
 
const config: TestRunnerConfig = {
  // Hook that is executed before the test runner starts running tests
  setup() {
    // Add your configuration here.
  },
  /* Hook to execute before a story is initially visited before being rendered in the browser.
   * The page argument is the Playwright's page object for the story.
   * The context argument is a Storybook object containing the story's id, title, and name.
   */
  async preVisit(page, context) {
    // Add your configuration here.
  },
  /* Hook to execute after a story is visited and fully rendered.
   * The page argument is the Playwright's page object for the story
   * The context argument is a Storybook object containing the story's id, title, and name.
   */
  async postVisit(page, context) {
    // Get the entire context of a story, including parameters, args, argTypes, etc.
    const storyContext = await getStoryContext(page, context);
 
    // This utility function is designed for image snapshot testing. It will wait for the page to be fully loaded, including all the async items (e.g., images, fonts, etc.).
    await waitForPageReady(page);
 
    // Add your configuration here.
  },
};
 
export default config;

使用测试运行器访问 Story 信息

如果你需要访问关于 Story 的信息,例如其参数,测试运行器包含一个名为 getStoryContext 的辅助函数,你可以使用它来检索信息。然后你可以根据需要使用它来进一步自定义你的测试。例如,如果你需要配置 Playwright 页面的视口大小以使用 Story 参数中定义的视口大小,你可以这样做,如下所示

.storybook/test-runner.ts
import type { TestRunnerConfig } from '@storybook/test-runner';
import { getStoryContext } from '@storybook/test-runner';
 
const { MINIMAL_VIEWPORTS } = require('@storybook/addon-viewport');
 
const DEFAULT_VIEWPORT_SIZE = { width: 1280, height: 720 };
 
const config: TestRunnerConfig = {
  async preVisit(page, story) {
    // Accesses the story's parameters and retrieves the viewport used to render it
    const context = await getStoryContext(page, story);
    const viewportName = context.parameters?.viewport?.defaultViewport;
    const viewportParameter = MINIMAL_VIEWPORTS[viewportName];
 
    if (viewportParameter) {
      const viewportSize = Object.entries(viewportParameter.styles).reduce(
        (acc, [screen, size]) => ({
          ...acc,
          // Converts the viewport size from percentages to numbers
          [screen]: parseInt(size),
        }),
        {},
      );
      // Configures the Playwright page to use the viewport size
      page.setViewportSize(viewportSize);
    } else {
      page.setViewportSize(DEFAULT_VIEWPORT_SIZE);
    }
  },
};
 
export default config;

使用 assets

如果你正在运行一组特定的测试(例如,图像快照测试),测试运行器提供一个名为 waitForPageReady 的辅助函数,你可以使用它来确保页面完全加载并准备好再运行测试。例如

.storybook/test-runner.ts
import type { TestRunnerConfig } from '@storybook/test-runner';
 
import { waitForPageReady } from '@storybook/test-runner';
 
import { toMatchImageSnapshot } from 'jest-image-snapshot';
 
const customSnapshotsDir = `${process.cwd()}/__snapshots__`;
 
const config: TestRunnerConfig = {
  setup() {
    expect.extend({ toMatchImageSnapshot });
  },
  async postVisit(page, context) {
    // Awaits for the page to be loaded and available including assets (e.g., fonts)
    await waitForPageReady(page);
 
    // Generates a snapshot file based on the story identifier
    const image = await page.screenshot();
    expect(image).toMatchImageSnapshot({
      customSnapshotsDir,
      customSnapshotIdentifier: context.id,
    });
  },
};
 
export default config;

Index.json 模式

当测试本地 Storybook 时,测试运行器会将你的 story 文件转换为测试。对于远程 Storybook,它使用 Storybook 的 index.json 文件(以前是 stories.json)(所有 story 的静态索引)来运行测试。

为什么?

假设你遇到本地和远程 Storybook 似乎不同步的情况,或者你甚至可能无法访问代码。在这种情况下,index.json 文件保证是你正在测试的已部署 Storybook 的最准确表示。要使用此功能测试本地 Storybook,请使用 --index-json 标志,如下所示

npm run test-storybook -- --index-json

index.json 模式与 watch 模式不兼容。

如果你需要禁用它,请使用 --no-index-json 标志

npm run test-storybook -- --no-index-json

如何检查我的 Storybook 是否有 index.json 文件?

Index.json 模式需要 index.json 文件。打开浏览器窗口并导航到你部署的 Storybook 实例(例如,https://your-storybook-url-here.com/index.json)。你应该看到一个 JSON 文件,它以 "v": 3 键开头,紧接着是另一个名为 "stories" 的键,其中包含 story ID 到 JSON 对象的映射。如果是这种情况,你的 Storybook 支持 index.json 模式


故障排除

测试运行器似乎不稳定并且一直超时

如果你的测试超时并显示以下消息

Timeout - Async callback was not invoked within the 15000 ms timeout specified by jest.setTimeout

可能是 Playwright 无法处理测试你的项目中 story 的数量。也许你有大量的 story,或者你的 CI 环境的 RAM 配置非常低。在这种情况下,你应该通过调整你的命令来限制并行运行的 worker 的数量,如下所示

package.json
{
  "scripts": {
    "test-storybook:ci": "yarn test-storybook --maxWorkers=2"
  }
}

CLI 中的错误输出太短

默认情况下,测试运行器将错误输出截断为 1000 个字符,你可以在浏览器中的 Storybook 中直接查看完整输出。但是,如果你想更改该限制,你可以通过将 DEBUG_PRINT_LIMIT 环境变量设置为你选择的数字来做到这一点,例如,DEBUG_PRINT_LIMIT=5000 yarn test-storybook

在其他 CI 环境中运行测试运行器

由于测试运行器基于 Playwright,你可能需要使用特定的 docker 镜像或其他配置,具体取决于你的 CI 设置。在这种情况下,你可以参考 Playwright CI 文档以获取更多信息。

按标签过滤的测试执行不正确

如果你已启用使用标签过滤测试,并为 includeexclude 列表提供了相似的标签,则测试运行器将根据 exclude 列表执行测试,并忽略 include 列表。为避免这种情况,请确保提供给 includeexclude 列表的标签不同。

测试运行器不直接支持 Yarn PnP

如果你在启用了 Plug'n'Play (PnP) 的较新版本 Yarn 上运行的项目中启用了测试运行器,则测试运行器可能无法按预期工作,并且在运行测试时可能会生成以下错误

PlaywrightError: jest-playwright-preset: Cannot find playwright package to use chromium

这是因为测试运行器使用了社区维护的包 jest-playwright-preset,该包仍需要支持此功能。为了解决这个问题,你可以将 nodeLinker 设置切换到 node-modules,或者在你的项目中直接安装 Playwright 作为依赖项,然后通过 install 命令添加浏览器二进制文件。

了解其他 UI 测试