OCRテストサービス
@wdio/ocr-serviceはサードパーティのパッケージです。詳細についてはGitHub | npmをご覧ください。
WebdriverIOでのビジュアルテストに関するドキュメントはドキュメントを参照してください。このプロジェクトにはWebdriverIOでビジュアルテストを実行するための関連モジュールがすべて含まれています。./packages
ディレクトリには以下が含まれています:
@wdio/visual-testing
: ビジュアルテストを統合するためのWebdriverIOサービスwebdriver-image-comparison
: WebDriverプロトコルをサポートする様々なNodeJSテスト自動化フレームワークで使用できる画像比較モジュール
Storybookランナー(ベータ版)
Storybookランナーベータ版についての詳細情報を見るにはクリックしてください
Storybookランナーはまだベータ版です。ドキュメントは後ほどWebdriverIOのドキュメントページに移動します。
このモジュールは新しいビジュアルランナーでStorybookをサポートするようになりました。このランナーはローカル/リモートのStorybookインスタンスを自動的にスキャンし、各コンポーネントの要素スクリーンショットを作成します。これは以下のように
export const config: WebdriverIO.Config = {
// ...
services: ["visual"],
// ....
};
をservices
に追加し、コマンドラインからnpx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook
を実行することでできます。
デフォルトではヘッドレスモードのChromeをブラウザとして使用します。
[!NOTE]
- ビジュアルテストのオプションの多くはStorybookランナーでも機能します。WebdriverIOのドキュメントを参照してください。
- Storybookランナーはすべてのcapabilitiesを上書きし、サポートするブラウザでのみ実行できます。
--browsers
を参照してください。- StorybookランナーはMultiremote capabilitiesを使用する既存の設定をサポートせず、エラーがスローされます。
- StorybookランナーはデスクトップWebのみをサポートし、モバイルWebはサポートしていません。
Storybookランナーサービスオプション
サービスオプションは次のように提供できます
export const config: WebdriverIO.Config = {
// ...
services: [
[
'visual',
{
// いくつかのデフォルトオプション
baselineFolder: join(process.cwd(), './__snapshots__/'),
debug: true,
// Storybookオプション、説明はCLIオプションを参照
storybook: {
additionalSearchParams: new URLSearchParams({foo: 'bar', abc: 'def'}),
clip: false,
clipSelector: ''#some-id,
numShards: 4,
// `skipStories`は文字列('example-button--secondary')、
// 配列(['example-button--secondary', 'example-button--small'])
// または文字列として提供する必要があるregex ("/.*button.*/gm")が使用できます
skipStories: ['example-button--secondary', 'example-button--small'],
url: 'https://www.bbc.co.uk/iplayer/storybook/',
version: 6,
// オプション - ベースラインパスの上書きを可能にします。デフォルトではカテゴリとコンポーネントでベースラインをグループ化します(例:forms/input/baseline.png)
getStoriesBaselinePath: (category, component) => `path__${category}__${component}`,
},
},
],
],
// ....
}
Storybookランナー CLIオプション
--additionalSearchParams
- タイプ:
string
- 必須: いいえ
- デフォルト: ''
- 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --additionalSearchParams="foo=bar&abc=def"
StorybookのURLに追加の検索パラメータを追加します。 詳細についてはURLSearchParamsのドキュメントを参照し てください。文字列は有効なURLSearchParamsの文字列である必要があります。
[!NOTE] ダブルクォートは
&
がコマンド区切り文字として解釈されるのを防ぐために必要です。 例えば--additionalSearchParams="foo=bar&abc=def"
では、ストーリーテスト用に次のようなStorybookのURLが生成されます:http://storybook.url/iframe.html?id=story-id&foo=bar&abc=def
。
--browsers
- タイプ:
string
- 必須: いいえ
- デフォルト:
chrome
、chrome|firefox|edge|safari
から選択可能 - 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --browsers=chrome,firefox,edge,safari
- 注意: CLIからのみ利用可能
提供されたブラウザを使用してコンポーネントのスクリーンショットを撮影します
[!NOTE] 実行したいブラウザがローカルマシンにインストールされていることを確認してください
--clip
- タイプ:
boolean
- 必須: いいえ
- デフォルト:
true
- 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --clip=false
無効にするとビューポートスクリーンショットを作成します。有効にすると--clipSelector
に基づいて要素スクリーンショットを作成し、コンポーネントスクリーンショット周囲の余白を減らし、スクリーンショットのサイズを縮小します。
--clipSelector
- タイプ:
string
- 必須: いいえ
- デフォルト: Storybook V7では
#storybook-root > :first-child
、Storybook V6では#root > :first-child:not(script):not(style)
、--version
も参照 - 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --clipSelector="#some-id"
このセレクタは以下の用途に使用されます:
- スクリーンショットを撮影する要素の選択
- スクリーンショットを撮影する前に表示されるのを待つ要素
--devices
- タイプ:
string
- 必須: いいえ
- デフォルト:
deviceDescriptors.ts
から選択可能 - 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --devices="iPhone 14 Pro Max","Pixel 3 XL"
- 注意: CLIからのみ利用可能
deviceDescriptors.ts
に一致する提供されたデバイスを使用してコンポーネントのスクリーンショットを撮影します
[!NOTE]
- デバイス設定が見つからない場合は、機能リクエストを提出してください
- これはChromeでのみ動作します:
--devices
を提供すると、すべてのChromeインスタンスはモバイルエミュレーションモードで実行されます- Chromeの他にも
--devices --browsers=firefox,safari,edge
のような他のブラウザを提供すると、モバイルエミュレーションモードのChromeが自動的に追加されます- Storybookランナーはデフォルトで要素スナップショットを作成します。完全なモバイルエミュレーションスクリーンショットを見たい場合は、コマンドラインで
--clip=false
を提供してください- ファイル名は例えば
__snapshots__/example/button/desktop_chrome/example-button--large-local-chrome-iPhone-14-Pro-Max-430x932-dpr-3.png
のようになります- 出典: モバイルエミュレーションを使用してデスクトップでモバイルウェブサイトをテストすることは有用ですが、テスターは以下のような多くの微妙な違いに注意すべきです:
- まったく異なるGPUがあり、大きなパフォーマンスの変化につながる可能性があります
- モバイルUIはエミュレートされません(特にURLバーの非表示化はページの高さに影響します)
- ディスアンビギュエーションポップアップ(複数のタッチターゲットから1つを選択する)はサポートされていません
- 多くのハードウェアAPI(例えば、orientationchangeイベント)は利用できません
--headless
- タイプ:
boolean
- 必須: いいえ
- デフォルト:
true
- 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --headless=false
- 注意: CLIからのみ利用可能
これはデフォルトでテストをヘッドレスモード(ブラウザがサポートしている場合)で実行するか、無効にするかを設定します
--numShards
- タイプ:
number
- 必須: いいえ
- デフォルト:
true
- 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --numShards=10
ストーリーを実行するために使用される並列インスタンスの数です。これはwdio.conf
ファイルのmaxInstances
によって制限されます。
[!IMPORTANT]
headless
モードで実行する場合、リソース制限による不安定さを防ぐために数を20以上に増やさないでください
--skipStories
- タイプ:
string|regex
- 必須: いいえ
- デフォルト: null
- 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --skipStories="/.*button.*/gm"
これは以下のいずれかです:
- 文字列 (
example-button--secondary,example-button--small
) - または正規表現 (
"/.*button.*/gm"
)
特定のストーリーをスキップするために使用します。ストーリーのURLで見つけることができるid
を使用します。例えば、URL http://localhost:6006/?path=/story/example-page--logged-out
のid
はexample-page--logged-out
です
--url
- タイプ:
string
- 必須: いいえ
- デフォルト:
http://127.0.0.1:6006
- 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --url="https://example.com"
StorybookインスタンスがホストされているURL。
--version
- タイプ:
number
- 必須: いいえ
- デフォルト: 7
- 例:
npx wdio tests/configs/wdio.local.desktop.storybook.conf.ts --storybook --version=6
Storybookのバージョンで、デフォルトは7
です。V6のclipSelector
を使用する必要があるかどうかを知るために必要です。
Storybookインタラクションテスト
Storybookインタラクションテストを使用すると、WDIOコマンドを使用してカスタムスクリプトを作成し、コンポーネントを特定の状態にするためのインタラクションが可能になります。例えば、以下のコードスニペットをご覧ください:
import { browser, expect } from "@wdio/globals";
describe("Storybook Interaction", () => {
it("should create screenshots for the logged in state when it logs out", async () => {
const componentId = "example-page--logged-in";
await browser.waitForStorybookComponentToBeLoaded({ id: componentId });
await expect($("header")).toMatchElementSnapshot(
`${componentId}-logged-in-state`
);
await $("button=Log out").click();
await expect($("header")).toMatchElementSnapshot(
`${componentId}-logged-out-state`
);
});
it("should create screenshots for the logged out state when it logs in", async () => {
const componentId = "example-page--logged-out";
await browser.waitForStorybookComponentToBeLoaded({ id: componentId });
await expect($("header")).toMatchElementSnapshot(
`${componentId}-logged-out-state`
);
await $("button=Log in").click();
await expect($("header")).toMatchElementSnapshot(
`${componentId}-logged-in-state`
);
});
});
2つの異なるコンポーネントに対する2つのテストが実行されます。各テストはまず状態を設定し、次にスクリーンショットを撮ります。また、新しいカスタムコマンドが導入されていることにも気づくでしょう。これについてはこちらで詳しく説明されています。
上記のspecファイルをフォルダに保存し、以下のコマンドでコマンドラインに追加できます:
pnpm run test.local.desktop.storybook.localhost -- --spec='tests/specs/storybook-interaction/*.ts'
Storybookランナーはまず自動的にStorybookインスタンスをスキャンし、比較する必要があるストーリーにテストを追加します。インタラクションテストに使用 するコンポーネントが2回比較されないようにするには、--skipStories
フィルターを提供して「デフォルト」ストーリーをスキャンから除外できます。これは以下のようになります:
pnpm run test.local.desktop.storybook.localhost -- --skipStories="/example-page.*/gm" --spec='tests/specs/storybook-interaction/*.ts'
新しいカスタムコマンド
browser.waitForStorybookComponentToBeLoaded({ id: 'componentId' })
という新しいカスタムコマンドがbrowser/driver
オブジェクトに追加され、コンポーネントを自動的に読み 込み、それが完了するのを待つため、browser.url('url.com')
メソッドを使用する必要はありません。次のように使用できます:
import { browser, expect } from "@wdio/globals";
describe("Storybook Interaction", () => {
it("should create screenshots for the logged in state when it logs out", async () => {
const componentId = "example-page--logged-in";
await browser.waitForStorybookComponentToBeLoaded({ id: componentId });
await expect($("header")).toMatchElementSnapshot(
`${componentId}-logged-in-state`
);
await $("button=Log out").click();
await expect($("header")).toMatchElementSnapshot(
`${componentId}-logged-out-state`
);
});
it("should create screenshots for the logged out state when it logs in", async () => {
const componentId = "example-page--logged-out";
await browser.waitForStorybookComponentToBeLoaded({ id: componentId });
await expect($("header")).toMatchElementSnapshot(
`${componentId}-logged-out-state`
);
await $("button=Log in").click();
await expect($("header")).toMatchElementSnapshot(
`${componentId}-logged-in-state`
);
});
});
オプションは以下の通りです:
additionalSearchParams
- タイプ:
URLSearchParams
- 必須: いいえ
- デフォルト:
new URLSearchParams()
- 例:
await browser.waitForStorybookComponentToBeLoaded({
additionalSearchParams: new URLSearchParams({ foo: "bar", abc: "def" }),
id: "componentId",
});
これによりStorybookのURLに追加の検索パラメータが追加されます。上記の例では、URLはhttp://storybook.url/iframe.html?id=story-id&foo=bar&abc=def
になります。
詳細についてはURLSearchParamsのドキュメントを参照してください。
clipSelector
- タイプ:
string
- 必須: いいえ
- デフォルト: Storybook V7では
#storybook-root > :first-child
、Storybook V6では#root > :first-child:not(script):not(style)
- 例:
await browser.waitForStorybookComponentToBeLoaded({
clipSelector: "#your-selector",
id: "componentId",
});
このセレクタは以下の用途に使用されます:
- スクリーンショットを撮影する要素の選択
- スクリーンショットを撮影する前に表示されるのを待つ要素
id
- タイプ:
string
- 必須: はい
- 例:
await browser.waitForStorybookComponentToBeLoaded({ '#your-selector', id: 'componentId' })
ストーリーのURLで見つけることができるid
を使用します。例えば、URL http://localhost:6006/?path=/story/example-page--logged-out
のid
はexample-page--logged-out
です
timeout
- タイプ:
number
- 必須: いいえ
- デフォルト: 1100ミリ秒
- 例:
await browser.waitForStorybookComponentToBeLoaded({
id: "componentId",
timeout: 20000,
});
ページ読み込み後にコンポーネントが表示されるのを待つ最大タイムアウト時間