1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-11 18:24:43 +02:00

Chore: Tests: Improve Playwright test reliability (#10981)

This commit is contained in:
Henry Heino 2024-09-04 04:14:12 -07:00 committed by GitHub
parent 0b13dbddd8
commit e41394b57f
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
10 changed files with 66 additions and 52 deletions

View File

@ -13,11 +13,16 @@ test.describe('markdownEditor', () => {
await mainScreen.importHtmlDirectory(electronApp, join(__dirname, 'resources', 'html-import')); await mainScreen.importHtmlDirectory(electronApp, join(__dirname, 'resources', 'html-import'));
const importedFolder = mainScreen.sidebar.container.getByText('html-import'); const importedFolder = mainScreen.sidebar.container.getByText('html-import');
await importedFolder.waitFor(); await importedFolder.waitFor();
await importedFolder.click();
await mainScreen.noteList.focusContent(electronApp); // Retry -- focusing the imported-folder may fail in some cases
const importedHtmlFileItem = mainScreen.noteList.getNoteItemByTitle('test-html-file-with-image'); await expect(async () => {
await importedHtmlFileItem.click(); await importedFolder.click();
await mainScreen.noteList.focusContent(electronApp);
const importedHtmlFileItem = mainScreen.noteList.getNoteItemByTitle('test-html-file-with-image');
await importedHtmlFileItem.click({ timeout: 300 });
}).toPass();
const viewerFrame = mainScreen.noteEditor.getNoteViewerIframe(); const viewerFrame = mainScreen.noteEditor.getNoteViewerIframe();
// Should render headers // Should render headers

View File

@ -14,10 +14,7 @@ export default class GoToAnything {
public async open(electronApp: ElectronApplication) { public async open(electronApp: ElectronApplication) {
await this.mainScreen.waitFor(); await this.mainScreen.waitFor();
await activateMainMenuItem(electronApp, 'Goto Anything...');
if (!await activateMainMenuItem(electronApp, 'Goto Anything...')) {
throw new Error('Menu item for opening Goto Anything not found');
}
return this.waitFor(); return this.waitFor();
} }

View File

@ -46,12 +46,7 @@ export default class MainScreen {
public async openSettings(electronApp: ElectronApplication) { public async openSettings(electronApp: ElectronApplication) {
// Check both labels so this works on MacOS // Check both labels so this works on MacOS
const openedWithPreferences = await activateMainMenuItem(electronApp, 'Preferences...'); await activateMainMenuItem(electronApp, /^(Preferences\.\.\.|Options)$/);
const openedWithOptions = await activateMainMenuItem(electronApp, 'Options');
if (!openedWithOptions && !openedWithPreferences) {
throw new Error('Unable to find settings menu item in application menus.');
}
} }
public async search(text: string) { public async search(text: string) {
@ -61,10 +56,6 @@ export default class MainScreen {
public async importHtmlDirectory(electronApp: ElectronApplication, path: string) { public async importHtmlDirectory(electronApp: ElectronApplication, path: string) {
await setFilePickerResponse(electronApp, [path]); await setFilePickerResponse(electronApp, [path]);
const startedImport = await activateMainMenuItem(electronApp, 'HTML - HTML document (Directory)', 'Import'); await activateMainMenuItem(electronApp, 'HTML - HTML document (Directory)', 'Import');
if (!startedImport) {
throw new Error('Unable to find HTML directory import menu item.');
}
} }
} }

View File

@ -3,9 +3,11 @@ import { ElectronApplication, Locator, Page, expect } from '@playwright/test';
export default class NoteList { export default class NoteList {
public readonly container: Locator; public readonly container: Locator;
public readonly sortOrderButton: Locator;
public constructor(page: Page) { public constructor(page: Page) {
this.container = page.locator('.rli-noteList'); this.container = page.locator('.rli-noteList');
this.sortOrderButton = this.container.getByRole('button', { name: 'Toggle sort order' });
} }
public waitFor() { public waitFor() {
@ -13,14 +15,12 @@ export default class NoteList {
} }
private async sortBy(electronApp: ElectronApplication, sortMethod: string) { private async sortBy(electronApp: ElectronApplication, sortMethod: string) {
const success = await activateMainMenuItem(electronApp, sortMethod, 'Sort notes by'); await activateMainMenuItem(electronApp, sortMethod, 'Sort notes by');
if (!success) {
throw new Error(`Unable to find sorting menu item: ${sortMethod}`);
}
} }
public async sortByTitle(electronApp: ElectronApplication) { public async sortByTitle(electronApp: ElectronApplication) {
return this.sortBy(electronApp, 'Title'); await this.sortBy(electronApp, 'Title');
await expect(this.sortOrderButton).toHaveAttribute('title', /Toggle sort order field:[\n ]*title ->/);
} }
public async focusContent(electronApp: ElectronApplication) { public async focusContent(electronApp: ElectronApplication) {

View File

@ -23,10 +23,7 @@ export default class Sidebar {
} }
private async sortBy(electronApp: ElectronApplication, option: string) { private async sortBy(electronApp: ElectronApplication, option: string) {
const success = await activateMainMenuItem(electronApp, option, 'Sort notebooks by'); await activateMainMenuItem(electronApp, option, 'Sort notebooks by');
if (!success) {
throw new Error(`Failed to find menu item: ${option}`);
}
} }
public async sortByDate(electronApp: ElectronApplication) { public async sortByDate(electronApp: ElectronApplication) {

View File

@ -91,7 +91,14 @@ test.describe('noteList', () => {
await noteList.focusContent(electronApp); await noteList.focusContent(electronApp);
// The most recently-created note should be visible // The most recently-created note should be visible
const note4Item = noteList.getNoteItemByTitle('note_4'); const note4Item = noteList.getNoteItemByTitle('note_4');
const note3Item = noteList.getNoteItemByTitle('note_3');
const note2Item = noteList.getNoteItemByTitle('note_2');
const note1Item = noteList.getNoteItemByTitle('note_1');
await expect(note4Item).toBeVisible(); await expect(note4Item).toBeVisible();
await expect(note3Item).toBeVisible();
await expect(note2Item).toBeVisible();
await expect(note1Item).toBeVisible();
await noteList.expectNoteToBeSelected('note_4'); await noteList.expectNoteToBeSelected('note_4');
await noteList.container.press('ArrowUp'); await noteList.container.press('ArrowUp');

View File

@ -8,7 +8,7 @@ test.describe('settings', () => {
await mainScreen.waitFor(); await mainScreen.waitFor();
// Sort order buttons should be visible by default // Sort order buttons should be visible by default
const sortOrderLocator = mainScreen.noteList.container.getByRole('button', { name: 'Toggle sort order' }); const sortOrderLocator = mainScreen.noteList.sortOrderButton;
await expect(sortOrderLocator).toBeVisible(); await expect(sortOrderLocator).toBeVisible();
await mainScreen.openSettings(electronApp); await mainScreen.openSettings(electronApp);

View File

@ -1,4 +1,4 @@
import { test, expect } from './util/test'; import { test } from './util/test';
import MainScreen from './models/MainScreen'; import MainScreen from './models/MainScreen';
import SettingsScreen from './models/SettingsScreen'; import SettingsScreen from './models/SettingsScreen';
import activateMainMenuItem from './util/activateMainMenuItem'; import activateMainMenuItem from './util/activateMainMenuItem';
@ -28,7 +28,7 @@ test.describe('simpleBackup', () => {
await mainScreen.waitFor(); await mainScreen.waitFor();
// Backups should work // Backups should work
expect(await activateMainMenuItem(electronApp, 'Create backup')).toBe(true); await activateMainMenuItem(electronApp, 'Create backup');
const successDialog = mainWindow.locator('iframe[id$=backup-backupDialog]'); const successDialog = mainWindow.locator('iframe[id$=backup-backupDialog]');
await successDialog.waitFor(); await successDialog.waitFor();

View File

@ -1,5 +1,5 @@
import type { ElectronApplication } from '@playwright/test'; import { expect, type ElectronApplication } from '@playwright/test';
import type { MenuItem } from 'electron'; import type { MenuItem } from 'electron';
@ -7,35 +7,45 @@ import type { MenuItem } from 'electron';
// https://github.com/spaceagetv/electron-playwright-helpers/blob/main/src/menu_helpers.ts // https://github.com/spaceagetv/electron-playwright-helpers/blob/main/src/menu_helpers.ts
// If given, `parentMenuLabel` should be the label of the menu containing the target item. // If given, `parentMenuLabel` should be the label of the menu containing the target item.
const activateMainMenuItem = ( const activateMainMenuItem = async (
electronApp: ElectronApplication, electronApp: ElectronApplication,
targetItemLabel: string, targetItemLabel: string|RegExp,
parentMenuLabel?: string, parentMenuLabel?: string,
) => { ) => {
return electronApp.evaluate(async ({ Menu }, [targetItemLabel, parentMenuLabel]) => { await expect.poll(() => {
const activateItemInSubmenu = (submenu: MenuItem[], parentLabel: string) => { return electronApp.evaluate(async ({ Menu }, [targetItemLabel, parentMenuLabel]) => {
for (const item of submenu) { const activateItemInSubmenu = (submenu: MenuItem[], parentLabel: string) => {
const matchesParent = !parentMenuLabel || parentLabel === parentMenuLabel; for (const item of submenu) {
if (item.label === targetItemLabel && matchesParent && item.visible) { const matchesParent = !parentMenuLabel || parentLabel === parentMenuLabel;
// Found! const matchesLabel = typeof targetItemLabel === 'string' ? (
item.click(); targetItemLabel === item.label
return true; ) : (
} else if (item.submenu) { item.label.match(targetItemLabel)
const foundItem = activateItemInSubmenu(item.submenu.items, item.label); );
if (foundItem) { if (matchesLabel && matchesParent && item.visible) {
// Found!
item.click();
return true; return true;
} else if (item.submenu) {
const foundItem = activateItemInSubmenu(item.submenu.items, item.label);
if (foundItem) {
return true;
}
} }
} }
}
// No item found // No item found
return false; return false;
}; };
const appMenu = Menu.getApplicationMenu(); const appMenu = Menu.getApplicationMenu();
return activateItemInSubmenu(appMenu.items, ''); return activateItemInSubmenu(appMenu.items, '');
}, [targetItemLabel, parentMenuLabel]); }, [targetItemLabel, parentMenuLabel]);
}, {
message: `should find and activate menu item with label ${JSON.stringify(targetItemLabel)}`,
}).toBe(true);
}; };
export default activateMainMenuItem; export default activateMainMenuItem;

View File

@ -56,6 +56,13 @@ export const test = base.extend<JoplinFixtures>({
mainWindow: async ({ electronApp }, use) => { mainWindow: async ({ electronApp }, use) => {
const mainWindow = await firstNonDevToolsWindow(electronApp); const mainWindow = await firstNonDevToolsWindow(electronApp);
// Setting the viewport size helps keep test environments consistent.
await mainWindow.setViewportSize({
width: 1200,
height: 800,
});
await use(mainWindow); await use(mainWindow);
}, },
}); });