From 276e38137585b53acbf2e143b054b0f8da29a622 Mon Sep 17 00:00:00 2001 From: JOASSART Edwin Date: Fri, 26 Apr 2024 00:18:00 +0200 Subject: [PATCH] patch: basic "Flash with URL" E2E tests for macOS --- .github/actions/test/action.yml | 24 +++++++- .gitignore | 8 ++- .../drive-selector/drive-selector.tsx | 1 + .../flash-results/flash-results.tsx | 2 +- .../progress-button/progress-button.tsx | 5 +- .../source-selector/source-selector.tsx | 4 ++ .../target-selector-button.tsx | 1 + lib/util/drive-scanner.ts | 2 + package.json | 2 + tests/e2e/e2e-flash-from-file.spec.ts | 42 +++++++++++++ tests/e2e/e2e-flash-from-url.spec.ts | 61 +++++++++++++++++++ tests/test.e2e.ts | 7 --- wdio.conf.ts | 29 ++++++--- 13 files changed, 166 insertions(+), 22 deletions(-) create mode 100644 tests/e2e/e2e-flash-from-file.spec.ts create mode 100644 tests/e2e/e2e-flash-from-url.spec.ts delete mode 100644 tests/test.e2e.ts diff --git a/.github/actions/test/action.yml b/.github/actions/test/action.yml index 80841b7b83..36ae5985ac 100644 --- a/.github/actions/test/action.yml +++ b/.github/actions/test/action.yml @@ -46,9 +46,20 @@ runs: with: python-version: '3.11' + - name: Setup Virtual Drive on MacOS + if: runner.os == 'macOS' + shell: bash + run: | + hdiutil create -size 4096m -layout NONE -o virtual_test_disk.dmg + virtual_path=$(hdiutil attach -nomount virtual_test_disk.dmg) + echo "TARGET_DRIVE=${virtual_path}" >> $GITHUB_ENV + echo "ETCHER_INCLUDE_VIRTUAL_DRIVES=1" >> $GITHUB_ENV + - name: Test release shell: bash run: | + # Build and Test release + ## FIXME: causes issues with `xxhash` which tries to load a debug build which doens't exist and cannot be compiled # if [[ '${{ inputs.VERBOSE }}' =~ on|On|Yes|yes|true|True ]]; then # export DEBUG='electron-forge:*,sidecar' @@ -57,8 +68,17 @@ runs: npm ci npm run lint npm run package - npm run wdio # test stage, note that it requires the package to be done first - + + # tests requires package to be built + if [[ '${{ runner.os }}' == 'macOS' ]]; then + # run all tests on macOS including E2E + # E2E tests can't input the administrative password, therefore the tests need to run as root + sudo TARGET_DRIVE=${{ env.TARGET_DRIVE }} ETCHER_INCLUDE_VIRTUAL_DRIVES=1 npm run wdio + else + # only run unit tests on Linux and Windows as E2E tests are not supported yet + npm run wdio:unit + fi + env: # https://www.electronjs.org/docs/latest/api/environment-variables ELECTRON_NO_ATTACH_CONSOLE: 'true' diff --git a/.gitignore b/.gitignore index f523e5fe17..eb19b1e4b9 100644 --- a/.gitignore +++ b/.gitignore @@ -120,4 +120,10 @@ secrets/WINDOWS_SIGNING.pfx #local development .yalc -yalc.lock \ No newline at end of file +yalc.lock + +# Test assets +virtual_test_disk.dmg +virtual_test_disk.img +virtual_test_disk.vhd +screenshots/ \ No newline at end of file diff --git a/lib/gui/app/components/drive-selector/drive-selector.tsx b/lib/gui/app/components/drive-selector/drive-selector.tsx index d3c2d38955..864ab91ff3 100644 --- a/lib/gui/app/components/drive-selector/drive-selector.tsx +++ b/lib/gui/app/components/drive-selector/drive-selector.tsx @@ -419,6 +419,7 @@ export class DriveSelector extends React.Component< primary: !showWarnings, warning: showWarnings, disabled: !hasAvailableDrives(), + 'data-testid': 'validate-target-button', }} {...props} > diff --git a/lib/gui/app/components/flash-results/flash-results.tsx b/lib/gui/app/components/flash-results/flash-results.tsx index 716c7acc38..b2f1469e06 100644 --- a/lib/gui/app/components/flash-results/flash-results.tsx +++ b/lib/gui/app/components/flash-results/flash-results.tsx @@ -163,7 +163,7 @@ export function FlashResults({ /> {middleEllipsis(image, 24)} - + {allFailed ? i18next.t('flash.flashFailed') : i18next.t('flash.flashCompleted')} diff --git a/lib/gui/app/components/progress-button/progress-button.tsx b/lib/gui/app/components/progress-button/progress-button.tsx index 0986bee642..7fb1cc95de 100644 --- a/lib/gui/app/components/progress-button/progress-button.tsx +++ b/lib/gui/app/components/progress-button/progress-button.tsx @@ -104,7 +104,9 @@ export class ProgressButton extends React.PureComponent { }} > - {status}  + + {status}  + {position} {type && ( @@ -125,6 +127,7 @@ export class ProgressButton extends React.PureComponent { warning={warning} onClick={this.props.callback} disabled={this.props.disabled} + data-testid={'flash-now-button'} style={{ marginTop: 30, }} diff --git a/lib/gui/app/components/source-selector/source-selector.tsx b/lib/gui/app/components/source-selector/source-selector.tsx index 9192e3df75..c41eb7cd9a 100644 --- a/lib/gui/app/components/source-selector/source-selector.tsx +++ b/lib/gui/app/components/source-selector/source-selector.tsx @@ -165,6 +165,7 @@ const URLSelector = ({ cancel={cancel} primaryButtonProps={{ disabled: loading || !imageURL, + 'data-testid': 'source-url-ok-button', }} action={loading ? : i18next.t('ok')} done={async () => { @@ -186,6 +187,7 @@ const URLSelector = ({ ) => @@ -655,6 +657,7 @@ export class SourceSelector extends React.Component< disabled={this.state.imageSelectorOpen} primary={this.state.defaultFlowActive} key="Flash from file" + data-testid="flash-from-file" flow={{ onClick: () => this.openImageSelector(), label: i18next.t('source.fromFile'), @@ -665,6 +668,7 @@ export class SourceSelector extends React.Component< /> this.openURLSelector(), label: i18next.t('source.fromURL'), diff --git a/lib/gui/app/components/target-selector/target-selector-button.tsx b/lib/gui/app/components/target-selector/target-selector-button.tsx index b2d62869c7..23d7ac4442 100644 --- a/lib/gui/app/components/target-selector/target-selector-button.tsx +++ b/lib/gui/app/components/target-selector/target-selector-button.tsx @@ -150,6 +150,7 @@ export function TargetSelectorButton(props: TargetSelectorProps) { tabIndex={targets.length > 0 ? -1 : 2} disabled={props.disabled} onClick={props.openDriveSelector} + data-testid="select-target-button" > {i18next.t('target.selectTarget')} diff --git a/lib/util/drive-scanner.ts b/lib/util/drive-scanner.ts index 30917a909c..8219f2f8dc 100644 --- a/lib/util/drive-scanner.ts +++ b/lib/util/drive-scanner.ts @@ -25,6 +25,8 @@ import { geteuid, platform } from 'process'; const adapters: Adapter[] = [ new BlockDeviceAdapter({ includeSystemDrives: () => true, + includeVirtualDrives: () => + process.env.ETCHER_INCLUDE_VIRTUAL_DRIVES === '1', }), ]; diff --git a/package.json b/package.json index 1c30bdefe1..24ac1b4168 100644 --- a/package.json +++ b/package.json @@ -20,6 +20,8 @@ "package": "electron-forge package", "start": "electron-forge start", "make": "electron-forge make", + "wdio:unit": "xvfb-maybe wdio run ./wdio.conf.ts --suite gui --suite shared", + "wdio:e2e": "xvfb-maybe wdio run ./wdio.conf.ts --suite e2e", "wdio": "xvfb-maybe wdio run ./wdio.conf.ts" }, "husky": { diff --git a/tests/e2e/e2e-flash-from-file.spec.ts b/tests/e2e/e2e-flash-from-file.spec.ts new file mode 100644 index 0000000000..fea720f966 --- /dev/null +++ b/tests/e2e/e2e-flash-from-file.spec.ts @@ -0,0 +1,42 @@ +import { browser } from '@wdio/globals'; + +describe('Electron Testing', () => { + it('should print application title', async () => { + console.log('Hello', await browser.getTitle(), 'application!'); + }); + + it('should "flash from file"', async () => { + const flashFromFileButton = $('button[data-testid="flash-from-file"]'); + await flashFromFileButton.waitForDisplayed({ timeout: 10000 }); + // const isDisplayed = await flashFromFileButton.isDisplayed(); + await flashFromFileButton.click(); + + const selectTargetButton = $('button[data-testid="select-target-button"]'); + await selectTargetButton.waitForClickable({ timeout: 30000 }); + await selectTargetButton.click(); + + // TODO: Select target using ENV variable for the drive + const targetVirtualDrive = $('=/dev/disk8'); + await targetVirtualDrive.waitForDisplayed({ timeout: 10000 }); + await targetVirtualDrive.click(); + + const validateTargetButton = $( + 'button[data-testid="validate-target-button"]', + ); + await validateTargetButton.waitForClickable({ timeout: 10000 }); + await validateTargetButton.click(); + + const flashNowButton = $('button[data-testid="flash-now-button"]'); + await flashNowButton.waitForClickable({ timeout: 10000 }); + await flashNowButton.click(); + + // FIXME: not able to find the flashResults :( + const flashResults = $('span[data-testid="flash-results"]'); + await flashResults.waitForDisplayed({ timeout: 20000 }); + + expect(flashResults.getText()).toBe('Flash Completed!'); + + // we're good; + // now we should check the content of the image but we can do that outside wdio + }); +}); diff --git a/tests/e2e/e2e-flash-from-url.spec.ts b/tests/e2e/e2e-flash-from-url.spec.ts new file mode 100644 index 0000000000..3663090833 --- /dev/null +++ b/tests/e2e/e2e-flash-from-url.spec.ts @@ -0,0 +1,61 @@ +import { browser } from '@wdio/globals'; + +describe('Electron Testing', () => { + it('should print application title', async () => { + console.log('Hello', await browser.getTitle(), 'application!'); + }); + + it('should "select an url source"', async () => { + const flashFromUrlButton = $('button[data-testid="flash-from-url"]'); + await flashFromUrlButton.waitForDisplayed({ timeout: 10000 }); + // const isDisplayed = await flashFromFileButton.isDisplayed(); + await flashFromUrlButton.click(); + + const enterValidUrlInput = $('input[data-testid="source-url-input"]'); + await enterValidUrlInput.waitForDisplayed({ timeout: 10000 }); + + // TODO: use an env variable for the URL + await enterValidUrlInput.setValue( + 'https://api.balena-cloud.com/download?deviceType=raspberrypi4-64&version=5.2.8&fileType=.zip&developmentMode=true', + ); + + const sourceUrlOkButton = $('button[data-testid="source-url-ok-button"]'); + await sourceUrlOkButton.waitForDisplayed({ timeout: 10000 }); + await sourceUrlOkButton.click(); + }); + + it('should "select a virtual target"', async () => { + const selectTargetButton = $('button[data-testid="select-target-button"]'); + await selectTargetButton.waitForClickable({ timeout: 30000 }); + await selectTargetButton.click(); + + // target drive is set in the github custom test action + // if you run the test locally, pass the varibale + const targetVirtualDrive = $(`=${process.env.TARGET_DRIVE}`); + await targetVirtualDrive.waitForDisplayed({ timeout: 10000 }); + await targetVirtualDrive.click(); + + const validateTargetButton = $( + 'button[data-testid="validate-target-button"]', + ); + await validateTargetButton.waitForClickable({ timeout: 10000 }); + await validateTargetButton.click(); + }); + + it('should "start flashing"', async () => { + const flashNowButton = $('button[data-testid="flash-now-button"]'); + await flashNowButton.waitForClickable({ timeout: 10000 }); + await flashNowButton.click(); + }); + + it('should get the "Flash Completed" screen', async () => { + const flashResults = $('[data-testid="flash-results"]'); + await flashResults.waitForDisplayed({ timeout: 180000 }); + + const flashResultsText = await flashResults.getText(); + expect(flashResultsText).toBe('Flash Completed!'); + + // we're good; + // now we should check the content of the image but we can do that outside wdio + }); +}); diff --git a/tests/test.e2e.ts b/tests/test.e2e.ts deleted file mode 100644 index f3a84194d4..0000000000 --- a/tests/test.e2e.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { browser } from '@wdio/globals'; - -describe('Electron Testing', () => { - it('should print application title', async () => { - console.log('Hello', await browser.getTitle(), 'application!'); - }); -}); diff --git a/wdio.conf.ts b/wdio.conf.ts index 4c1e8e47d8..8100a0c3c4 100644 --- a/wdio.conf.ts +++ b/wdio.conf.ts @@ -35,16 +35,25 @@ export const config: Options.Testrunner = { // Patterns to exclude. // FIXME: Remove the following exclusions once the tests are ported to WDIO exclude: [ - 'tests/gui/modules/image-writer.spec.ts', - 'tests/gui/os/window-progress.spec.ts', - 'tests/gui/models/available-drives.spec.ts', - 'tests/gui/models/flash-state.spec.ts', - 'tests/gui/models/selection-state.spec.ts', - 'tests/gui/models/settings.spec.ts', - 'tests/shared/drive-constraints.spec.ts', - 'tests/shared/messages.spec.ts', - 'tests/gui/modules/progress-status.spec.ts', + './tests/gui/modules/image-writer.spec.ts', + './tests/gui/os/window-progress.spec.ts', + './tests/gui/models/available-drives.spec.ts', + './tests/gui/models/flash-state.spec.ts', + './tests/gui/models/selection-state.spec.ts', + './tests/gui/models/settings.spec.ts', + './tests/shared/drive-constraints.spec.ts', + './tests/shared/messages.spec.ts', + './tests/gui/modules/progress-status.spec.ts', ], + + suites: { + 'gui': ['./tests/gui/**/*.spec.ts'], + 'shared': ['./tests/shared/**/*.spec.ts'], + 'e2e': [ + // 'tests/e2e/e2e-flash-from-file.spec.ts', + './tests/e2e/e2e-flash-from-url.spec.ts', + ], + }, // // ============ // Capabilities @@ -85,7 +94,7 @@ export const config: Options.Testrunner = { // Define all options that are relevant for the WebdriverIO instance here // // Level of logging verbosity: trace | debug | info | warn | error | silent - logLevel: 'info', + logLevel: 'warn', // // Set specific log levels per logger // loggers: