test: add in-between assertions and more fixtures (#1224)
This commit is contained in:
3
.github/workflows/playwright.yml
vendored
3
.github/workflows/playwright.yml
vendored
@@ -18,10 +18,11 @@ jobs:
|
|||||||
- name: Install Playwright Browsers
|
- name: Install Playwright Browsers
|
||||||
run: yarn playwright install --with-deps
|
run: yarn playwright install --with-deps
|
||||||
- name: Run Playwright tests
|
- name: Run Playwright tests
|
||||||
|
working-directory: ./packages/e2e-tests
|
||||||
env:
|
env:
|
||||||
LOGIN_EMAIL: ${{ secrets.LOGIN_EMAIL }}
|
LOGIN_EMAIL: ${{ secrets.LOGIN_EMAIL }}
|
||||||
LOGIN_PASSWORD: ${{ secrets.LOGIN_PASSWORD }}
|
LOGIN_PASSWORD: ${{ secrets.LOGIN_PASSWORD }}
|
||||||
run: yarn playwright test
|
run: yarn test
|
||||||
- uses: actions/upload-artifact@v3
|
- uses: actions/upload-artifact@v3
|
||||||
if: always()
|
if: always()
|
||||||
with:
|
with:
|
||||||
|
@@ -2,6 +2,16 @@ const path = require('node:path');
|
|||||||
const { BasePage } = require('./base-page');
|
const { BasePage } = require('./base-page');
|
||||||
|
|
||||||
export class ApplicationsPage extends BasePage {
|
export class ApplicationsPage extends BasePage {
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
constructor(page) {
|
||||||
|
super(page);
|
||||||
|
|
||||||
|
this.drawerLink = this.page.getByTestId('apps-page-drawer-link');
|
||||||
|
this.addConnectionButton = this.page.getByTestId('add-connection-button');
|
||||||
|
}
|
||||||
|
|
||||||
async screenshot(options = {}) {
|
async screenshot(options = {}) {
|
||||||
const { path: plainPath, ...restOptions } = options;
|
const { path: plainPath, ...restOptions } = options;
|
||||||
|
|
||||||
|
@@ -2,8 +2,12 @@ const path = require('node:path');
|
|||||||
const { BasePage } = require('./base-page');
|
const { BasePage } = require('./base-page');
|
||||||
|
|
||||||
export class FlowEditorPage extends BasePage {
|
export class FlowEditorPage extends BasePage {
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
constructor(page) {
|
constructor(page) {
|
||||||
super(page);
|
super(page);
|
||||||
|
|
||||||
this.appAutocomplete = this.page.getByTestId('choose-app-autocomplete');
|
this.appAutocomplete = this.page.getByTestId('choose-app-autocomplete');
|
||||||
this.eventAutocomplete = this.page.getByTestId('choose-event-autocomplete');
|
this.eventAutocomplete = this.page.getByTestId('choose-event-autocomplete');
|
||||||
this.continueButton = this.page.getByTestId('flow-substep-continue-button');
|
this.continueButton = this.page.getByTestId('flow-substep-continue-button');
|
||||||
@@ -15,6 +19,7 @@ export class FlowEditorPage extends BasePage {
|
|||||||
this.publishFlowButton = this.page.getByTestId('publish-flow-button');
|
this.publishFlowButton = this.page.getByTestId('publish-flow-button');
|
||||||
this.infoSnackbar = this.page.getByTestId('flow-cannot-edit-info-snackbar');
|
this.infoSnackbar = this.page.getByTestId('flow-cannot-edit-info-snackbar');
|
||||||
this.trigger = this.page.getByLabel('Trigger on weekends?');
|
this.trigger = this.page.getByLabel('Trigger on weekends?');
|
||||||
|
this.stepCircularLoader = this.page.getByTestId('step-circular-loader');
|
||||||
}
|
}
|
||||||
|
|
||||||
async screenshot(options = {}) {
|
async screenshot(options = {}) {
|
||||||
|
@@ -1,11 +1,11 @@
|
|||||||
const base = require('@playwright/test');
|
const { test, expect} = require('@playwright/test');
|
||||||
const { ApplicationsPage } = require('./applications-page');
|
const { ApplicationsPage } = require('./applications-page');
|
||||||
const { ConnectionsPage } = require('./connections-page');
|
const { ConnectionsPage } = require('./connections-page');
|
||||||
const { ExecutionsPage } = require('./executions-page');
|
const { ExecutionsPage } = require('./executions-page');
|
||||||
const { FlowEditorPage } = require('./flow-editor-page');
|
const { FlowEditorPage } = require('./flow-editor-page');
|
||||||
const { LoginPage } = require('./login-page');
|
const { LoginPage } = require('./login-page');
|
||||||
|
|
||||||
exports.test = base.test.extend({
|
exports.test = test.extend({
|
||||||
page: async ({ page }, use) => {
|
page: async ({ page }, use) => {
|
||||||
await new LoginPage(page).login();
|
await new LoginPage(page).login();
|
||||||
|
|
||||||
@@ -25,4 +25,12 @@ exports.test = base.test.extend({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
exports.expect = base.expect;
|
expect.extend({
|
||||||
|
toBeClickableLink: async (locator) => {
|
||||||
|
await expect(locator).not.toHaveAttribute('aria-disabled', 'true');
|
||||||
|
|
||||||
|
return { pass: true };
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
exports.expect = expect;
|
||||||
|
@@ -1,4 +1,5 @@
|
|||||||
const path = require('node:path');
|
const path = require('node:path');
|
||||||
|
const { expect } = require('@playwright/test');
|
||||||
const { BasePage } = require('./base-page');
|
const { BasePage } = require('./base-page');
|
||||||
|
|
||||||
export class LoginPage extends BasePage {
|
export class LoginPage extends BasePage {
|
||||||
@@ -9,7 +10,6 @@ export class LoginPage extends BasePage {
|
|||||||
super(page);
|
super(page);
|
||||||
|
|
||||||
this.page = page;
|
this.page = page;
|
||||||
|
|
||||||
this.emailTextField = this.page.getByTestId('email-text-field');
|
this.emailTextField = this.page.getByTestId('email-text-field');
|
||||||
this.passwordTextField = this.page.getByTestId('password-text-field');
|
this.passwordTextField = this.page.getByTestId('password-text-field');
|
||||||
this.loginButton = this.page.getByTestId('login-button');
|
this.loginButton = this.page.getByTestId('login-button');
|
||||||
@@ -23,5 +23,7 @@ export class LoginPage extends BasePage {
|
|||||||
await this.passwordTextField.fill(process.env.LOGIN_PASSWORD);
|
await this.passwordTextField.fill(process.env.LOGIN_PASSWORD);
|
||||||
|
|
||||||
await this.loginButton.click();
|
await this.loginButton.click();
|
||||||
|
|
||||||
|
await expect(this.loginButton).not.toBeVisible();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -20,6 +20,8 @@ module.exports = defineConfig({
|
|||||||
retries: process.env.CI ? 2 : 0,
|
retries: process.env.CI ? 2 : 0,
|
||||||
/* Opt out of parallel tests on CI. */
|
/* Opt out of parallel tests on CI. */
|
||||||
workers: process.env.CI ? 1 : undefined,
|
workers: process.env.CI ? 1 : undefined,
|
||||||
|
/* Timeout threshold for each test */
|
||||||
|
timeout: 120000,
|
||||||
/* Reporter to use. See https://playwright.dev/docs/test-reporters */
|
/* Reporter to use. See https://playwright.dev/docs/test-reporters */
|
||||||
reporter: process.env.CI ? 'github' : 'html',
|
reporter: process.env.CI ? 'github' : 'html',
|
||||||
/* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */
|
/* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */
|
||||||
@@ -35,6 +37,11 @@ module.exports = defineConfig({
|
|||||||
viewport: { width: 1280, height: 720 },
|
viewport: { width: 1280, height: 720 },
|
||||||
},
|
},
|
||||||
|
|
||||||
|
expect: {
|
||||||
|
/* Timeout threshold for each assertion */
|
||||||
|
timeout: 30000,
|
||||||
|
},
|
||||||
|
|
||||||
/* Configure projects for major browsers */
|
/* Configure projects for major browsers */
|
||||||
projects: [
|
projects: [
|
||||||
{
|
{
|
||||||
|
@@ -2,15 +2,15 @@
|
|||||||
const { test, expect } = require('../../fixtures/index');
|
const { test, expect } = require('../../fixtures/index');
|
||||||
|
|
||||||
test.describe('Apps page', () => {
|
test.describe('Apps page', () => {
|
||||||
test.beforeEach(async ({ page, applicationsPage }) => {
|
test.beforeEach(async ({ applicationsPage }) => {
|
||||||
await page.getByTestId('apps-page-drawer-link').click();
|
await applicationsPage.drawerLink.click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('displays applications', async ({ page, applicationsPage }) => {
|
test('displays applications', async ({ applicationsPage }) => {
|
||||||
await page.getByTestId('apps-loader').waitFor({
|
await applicationsPage.page.getByTestId('apps-loader').waitFor({
|
||||||
state: 'detached',
|
state: 'detached',
|
||||||
});
|
});
|
||||||
await expect(page.getByTestId('app-row')).not.toHaveCount(0);
|
await expect(applicationsPage.page.getByTestId('app-row')).not.toHaveCount(0);
|
||||||
|
|
||||||
await applicationsPage.screenshot({
|
await applicationsPage.screenshot({
|
||||||
path: 'Applications.png',
|
path: 'Applications.png',
|
||||||
@@ -18,49 +18,56 @@ test.describe('Apps page', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test.describe('can add connection', () => {
|
test.describe('can add connection', () => {
|
||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ applicationsPage }) => {
|
||||||
await expect(page.getByTestId('add-connection-button')).toBeVisible();
|
await expect(applicationsPage.addConnectionButton).toBeClickableLink();
|
||||||
await page.getByTestId('add-connection-button').click();
|
await applicationsPage.addConnectionButton.click();
|
||||||
await page
|
await applicationsPage
|
||||||
|
.page
|
||||||
.getByTestId('search-for-app-loader')
|
.getByTestId('search-for-app-loader')
|
||||||
.waitFor({ state: 'detached' });
|
.waitFor({ state: 'detached' });
|
||||||
});
|
});
|
||||||
|
|
||||||
test('lists applications', async ({ page, applicationsPage }) => {
|
test('lists applications', async ({ applicationsPage }) => {
|
||||||
const appListItemCount = await page.getByTestId('app-list-item').count();
|
const appListItemCount = await applicationsPage.page.getByTestId('app-list-item').count();
|
||||||
expect(appListItemCount).toBeGreaterThan(10);
|
expect(appListItemCount).toBeGreaterThan(10);
|
||||||
|
|
||||||
await applicationsPage.clickAway();
|
await applicationsPage.clickAway();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('searches an application', async ({ page, applicationsPage }) => {
|
test('searches an application', async ({ applicationsPage }) => {
|
||||||
await page.getByTestId('search-for-app-text-field').fill('DeepL');
|
await applicationsPage.page.getByTestId('search-for-app-text-field').fill('DeepL');
|
||||||
await expect(page.getByTestId('app-list-item')).toHaveCount(1);
|
await applicationsPage
|
||||||
|
.page
|
||||||
|
.getByTestId('search-for-app-loader')
|
||||||
|
.waitFor({ state: 'detached' });
|
||||||
|
|
||||||
|
await expect(applicationsPage.page.getByTestId('app-list-item')).toHaveCount(1);
|
||||||
|
|
||||||
await applicationsPage.clickAway();
|
await applicationsPage.clickAway();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('goes to app page to create a connection', async ({
|
test('goes to app page to create a connection', async ({
|
||||||
page,
|
|
||||||
applicationsPage,
|
applicationsPage,
|
||||||
}) => {
|
}) => {
|
||||||
await page.getByTestId('app-list-item').first().click();
|
// loading app, app config, app auth clients take time
|
||||||
await expect(page).toHaveURL('/app/deepl/connections/add?shared=false');
|
test.setTimeout(60000);
|
||||||
await expect(page.getByTestId('add-app-connection-dialog')).toBeVisible();
|
|
||||||
|
await applicationsPage.page.getByTestId('app-list-item').first().click();
|
||||||
|
await expect(applicationsPage.page).toHaveURL('/app/deepl/connections/add?shared=false');
|
||||||
|
await expect(applicationsPage.page.getByTestId('add-app-connection-dialog')).toBeVisible();
|
||||||
|
|
||||||
await applicationsPage.clickAway();
|
await applicationsPage.clickAway();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('closes the dialog on backdrop click', async ({
|
test('closes the dialog on backdrop click', async ({
|
||||||
page,
|
|
||||||
applicationsPage,
|
applicationsPage,
|
||||||
}) => {
|
}) => {
|
||||||
await page.getByTestId('app-list-item').first().click();
|
await applicationsPage.page.getByTestId('app-list-item').first().click();
|
||||||
await expect(page).toHaveURL('/app/deepl/connections/add?shared=false');
|
await expect(applicationsPage.page).toHaveURL('/app/deepl/connections/add?shared=false');
|
||||||
await expect(page.getByTestId('add-app-connection-dialog')).toBeVisible();
|
await expect(applicationsPage.page.getByTestId('add-app-connection-dialog')).toBeVisible();
|
||||||
await applicationsPage.clickAway();
|
await applicationsPage.clickAway();
|
||||||
await expect(page).toHaveURL('/app/deepl/connections');
|
await expect(applicationsPage.page).toHaveURL('/app/deepl/connections');
|
||||||
await expect(page.getByTestId('add-app-connection-dialog')).toBeHidden();
|
await expect(applicationsPage.page.getByTestId('add-app-connection-dialog')).toBeHidden();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@@ -19,7 +19,7 @@ test.describe('Connections page', () => {
|
|||||||
|
|
||||||
test.describe('can add connection', () => {
|
test.describe('can add connection', () => {
|
||||||
test('has a button to open add connection dialog', async ({ page }) => {
|
test('has a button to open add connection dialog', async ({ page }) => {
|
||||||
await expect(page.getByTestId('add-connection-button')).toBeVisible();
|
await expect(page.getByTestId('add-connection-button')).toBeClickableLink();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('add connection button takes user to add connection page', async ({
|
test('add connection button takes user to add connection page', async ({
|
||||||
|
@@ -13,28 +13,30 @@ test.beforeAll(async ({ browser }) => {
|
|||||||
flowEditorPage = new FlowEditorPage(page);
|
flowEditorPage = new FlowEditorPage(page);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('create flow', async ({}) => {
|
test('create flow', async () => {
|
||||||
await flowEditorPage.page.getByTestId('create-flow-button').click();
|
await flowEditorPage.page.getByTestId('create-flow-button').click();
|
||||||
await expect(flowEditorPage.page).toHaveURL(/\/editor\/create/);
|
await expect(flowEditorPage.page).toHaveURL(/\/editor\/create/);
|
||||||
await expect(flowEditorPage.page).toHaveURL(
|
await expect(flowEditorPage.page).toHaveURL(
|
||||||
/\/editor\/[0-9a-fA-F]{8}\b-[0-9a-fA-F]{4}\b-[0-9a-fA-F]{4}\b-[0-9a-fA-F]{4}\b-[0-9a-fA-F]{12}/
|
/\/editor\/[0-9a-fA-F]{8}\b-[0-9a-fA-F]{4}\b-[0-9a-fA-F]{4}\b-[0-9a-fA-F]{4}\b-[0-9a-fA-F]{12}/
|
||||||
);
|
);
|
||||||
|
|
||||||
|
await expect(flowEditorPage.stepCircularLoader).not.toBeVisible();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('has two steps by default', async ({}) => {
|
test('has two steps by default', async () => {
|
||||||
await expect(flowEditorPage.page.getByTestId('flow-step')).toHaveCount(2);
|
await expect(flowEditorPage.page.getByTestId('flow-step')).toHaveCount(2);
|
||||||
});
|
});
|
||||||
|
|
||||||
test.describe('arrange Scheduler trigger', () => {
|
test.describe('arrange Scheduler trigger', () => {
|
||||||
test.describe('choose app and event substep', () => {
|
test.describe('choose app and event substep', () => {
|
||||||
test('choose application', async ({}) => {
|
test('choose application', async () => {
|
||||||
await flowEditorPage.appAutocomplete.click();
|
await flowEditorPage.appAutocomplete.click();
|
||||||
await flowEditorPage.page
|
await flowEditorPage.page
|
||||||
.getByRole('option', { name: 'Scheduler' })
|
.getByRole('option', { name: 'Scheduler' })
|
||||||
.click();
|
.click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('choose an event', async ({}) => {
|
test('choose an event', async () => {
|
||||||
await expect(flowEditorPage.eventAutocomplete).toBeVisible();
|
await expect(flowEditorPage.eventAutocomplete).toBeVisible();
|
||||||
await flowEditorPage.eventAutocomplete.click();
|
await flowEditorPage.eventAutocomplete.click();
|
||||||
await flowEditorPage.page
|
await flowEditorPage.page
|
||||||
@@ -42,34 +44,34 @@ test.describe('arrange Scheduler trigger', () => {
|
|||||||
.click();
|
.click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('continue to next step', async ({}) => {
|
test('continue to next step', async () => {
|
||||||
await flowEditorPage.continueButton.click();
|
await flowEditorPage.continueButton.click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('collapses the substep', async ({}) => {
|
test('collapses the substep', async () => {
|
||||||
await expect(flowEditorPage.appAutocomplete).not.toBeVisible();
|
await expect(flowEditorPage.appAutocomplete).not.toBeVisible();
|
||||||
await expect(flowEditorPage.eventAutocomplete).not.toBeVisible();
|
await expect(flowEditorPage.eventAutocomplete).not.toBeVisible();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test.describe('set up a trigger', () => {
|
test.describe('set up a trigger', () => {
|
||||||
test('choose "yes" in "trigger on weekends?"', async ({}) => {
|
test('choose "yes" in "trigger on weekends?"', async () => {
|
||||||
await expect(flowEditorPage.trigger).toBeVisible();
|
await expect(flowEditorPage.trigger).toBeVisible();
|
||||||
await flowEditorPage.trigger.click();
|
await flowEditorPage.trigger.click();
|
||||||
await flowEditorPage.page.getByRole('option', { name: 'Yes' }).click();
|
await flowEditorPage.page.getByRole('option', { name: 'Yes' }).click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('continue to next step', async ({}) => {
|
test('continue to next step', async () => {
|
||||||
await flowEditorPage.continueButton.click();
|
await flowEditorPage.continueButton.click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('collapses the substep', async ({}) => {
|
test('collapses the substep', async () => {
|
||||||
await expect(flowEditorPage.trigger).not.toBeVisible();
|
await expect(flowEditorPage.trigger).not.toBeVisible();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test.describe('test trigger', () => {
|
test.describe('test trigger', () => {
|
||||||
test('show sample output', async ({}) => {
|
test('show sample output', async () => {
|
||||||
await expect(flowEditorPage.testOuput).not.toBeVisible();
|
await expect(flowEditorPage.testOuput).not.toBeVisible();
|
||||||
await flowEditorPage.continueButton.click();
|
await flowEditorPage.continueButton.click();
|
||||||
await expect(flowEditorPage.testOuput).toBeVisible();
|
await expect(flowEditorPage.testOuput).toBeVisible();
|
||||||
@@ -83,12 +85,12 @@ test.describe('arrange Scheduler trigger', () => {
|
|||||||
|
|
||||||
test.describe('arrange Ntfy action', () => {
|
test.describe('arrange Ntfy action', () => {
|
||||||
test.describe('choose app and event substep', () => {
|
test.describe('choose app and event substep', () => {
|
||||||
test('choose application', async ({}) => {
|
test('choose application', async () => {
|
||||||
await flowEditorPage.appAutocomplete.click();
|
await flowEditorPage.appAutocomplete.click();
|
||||||
await flowEditorPage.page.getByRole('option', { name: 'Ntfy' }).click();
|
await flowEditorPage.page.getByRole('option', { name: 'Ntfy' }).click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('choose an event', async ({}) => {
|
test('choose an event', async () => {
|
||||||
await expect(flowEditorPage.eventAutocomplete).toBeVisible();
|
await expect(flowEditorPage.eventAutocomplete).toBeVisible();
|
||||||
await flowEditorPage.eventAutocomplete.click();
|
await flowEditorPage.eventAutocomplete.click();
|
||||||
await flowEditorPage.page
|
await flowEditorPage.page
|
||||||
@@ -96,33 +98,33 @@ test.describe('arrange Ntfy action', () => {
|
|||||||
.click();
|
.click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('continue to next step', async ({}) => {
|
test('continue to next step', async () => {
|
||||||
await flowEditorPage.continueButton.click();
|
await flowEditorPage.continueButton.click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('collapses the substep', async ({}) => {
|
test('collapses the substep', async () => {
|
||||||
await expect(flowEditorPage.appAutocomplete).not.toBeVisible();
|
await expect(flowEditorPage.appAutocomplete).not.toBeVisible();
|
||||||
await expect(flowEditorPage.eventAutocomplete).not.toBeVisible();
|
await expect(flowEditorPage.eventAutocomplete).not.toBeVisible();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test.describe('choose connection', () => {
|
test.describe('choose connection', () => {
|
||||||
test('choose connection list item', async ({}) => {
|
test('choose connection list item', async () => {
|
||||||
await flowEditorPage.connectionAutocomplete.click();
|
await flowEditorPage.connectionAutocomplete.click();
|
||||||
await flowEditorPage.page.getByRole('option').first().click();
|
await flowEditorPage.page.getByRole('option').first().click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('continue to next step', async ({}) => {
|
test('continue to next step', async () => {
|
||||||
await flowEditorPage.continueButton.click();
|
await flowEditorPage.continueButton.click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('collapses the substep', async ({}) => {
|
test('collapses the substep', async () => {
|
||||||
await expect(flowEditorPage.connectionAutocomplete).not.toBeVisible();
|
await expect(flowEditorPage.connectionAutocomplete).not.toBeVisible();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test.describe('set up action', () => {
|
test.describe('set up action', () => {
|
||||||
test('fill topic and message body', async ({}) => {
|
test('fill topic and message body', async () => {
|
||||||
await flowEditorPage.page
|
await flowEditorPage.page
|
||||||
.getByTestId('parameters.topic-power-input')
|
.getByTestId('parameters.topic-power-input')
|
||||||
.locator('[contenteditable]')
|
.locator('[contenteditable]')
|
||||||
@@ -133,17 +135,17 @@ test.describe('arrange Ntfy action', () => {
|
|||||||
.fill('Message body');
|
.fill('Message body');
|
||||||
});
|
});
|
||||||
|
|
||||||
test('continue to next step', async ({}) => {
|
test('continue to next step', async () => {
|
||||||
await flowEditorPage.continueButton.click();
|
await flowEditorPage.continueButton.click();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('collapses the substep', async ({}) => {
|
test('collapses the substep', async () => {
|
||||||
await expect(flowEditorPage.connectionAutocomplete).not.toBeVisible();
|
await expect(flowEditorPage.connectionAutocomplete).not.toBeVisible();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test.describe('test trigger', () => {
|
test.describe('test trigger', () => {
|
||||||
test('show sample output', async ({}) => {
|
test('show sample output', async () => {
|
||||||
await expect(flowEditorPage.testOuput).not.toBeVisible();
|
await expect(flowEditorPage.testOuput).not.toBeVisible();
|
||||||
await flowEditorPage.page
|
await flowEditorPage.page
|
||||||
.getByTestId('flow-substep-continue-button')
|
.getByTestId('flow-substep-continue-button')
|
||||||
@@ -159,34 +161,34 @@ test.describe('arrange Ntfy action', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test.describe('publish and unpublish', () => {
|
test.describe('publish and unpublish', () => {
|
||||||
test('publish flow', async ({}) => {
|
test('publish flow', async () => {
|
||||||
await expect(flowEditorPage.unpublishFlowButton).not.toBeVisible();
|
await expect(flowEditorPage.unpublishFlowButton).not.toBeVisible();
|
||||||
await expect(flowEditorPage.publishFlowButton).toBeVisible();
|
await expect(flowEditorPage.publishFlowButton).toBeVisible();
|
||||||
await flowEditorPage.publishFlowButton.click();
|
await flowEditorPage.publishFlowButton.click();
|
||||||
await expect(flowEditorPage.publishFlowButton).not.toBeVisible();
|
await expect(flowEditorPage.publishFlowButton).not.toBeVisible();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('shows read-only sticky snackbar', async ({}) => {
|
test('shows read-only sticky snackbar', async () => {
|
||||||
await expect(flowEditorPage.infoSnackbar).toBeVisible();
|
await expect(flowEditorPage.infoSnackbar).toBeVisible();
|
||||||
await flowEditorPage.screenshot({
|
await flowEditorPage.screenshot({
|
||||||
path: 'Published flow.png',
|
path: 'Published flow.png',
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test('unpublish from snackbar', async ({}) => {
|
test('unpublish from snackbar', async () => {
|
||||||
await flowEditorPage.page
|
await flowEditorPage.page
|
||||||
.getByTestId('unpublish-flow-from-snackbar')
|
.getByTestId('unpublish-flow-from-snackbar')
|
||||||
.click();
|
.click();
|
||||||
await expect(flowEditorPage.infoSnackbar).not.toBeVisible();
|
await expect(flowEditorPage.infoSnackbar).not.toBeVisible();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('publish once again', async ({}) => {
|
test('publish once again', async () => {
|
||||||
await expect(flowEditorPage.publishFlowButton).toBeVisible();
|
await expect(flowEditorPage.publishFlowButton).toBeVisible();
|
||||||
await flowEditorPage.publishFlowButton.click();
|
await flowEditorPage.publishFlowButton.click();
|
||||||
await expect(flowEditorPage.publishFlowButton).not.toBeVisible();
|
await expect(flowEditorPage.publishFlowButton).not.toBeVisible();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('unpublish from layout top bar', async ({}) => {
|
test('unpublish from layout top bar', async () => {
|
||||||
await expect(flowEditorPage.unpublishFlowButton).toBeVisible();
|
await expect(flowEditorPage.unpublishFlowButton).toBeVisible();
|
||||||
await flowEditorPage.unpublishFlowButton.click();
|
await flowEditorPage.unpublishFlowButton.click();
|
||||||
await expect(flowEditorPage.unpublishFlowButton).not.toBeVisible();
|
await expect(flowEditorPage.unpublishFlowButton).not.toBeVisible();
|
||||||
@@ -197,7 +199,7 @@ test.describe('publish and unpublish', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test.describe('in layout', () => {
|
test.describe('in layout', () => {
|
||||||
test('can go back to flows page', async ({}) => {
|
test('can go back to flows page', async () => {
|
||||||
await flowEditorPage.page.getByTestId('editor-go-back-button').click();
|
await flowEditorPage.page.getByTestId('editor-go-back-button').click();
|
||||||
await expect(flowEditorPage.page).toHaveURL('/flows');
|
await expect(flowEditorPage.page).toHaveURL('/flows');
|
||||||
});
|
});
|
||||||
|
@@ -71,17 +71,18 @@ function generateValidationSchema(substeps: ISubstep[]) {
|
|||||||
substepArgumentValidations[key] = yup.mixed();
|
substepArgumentValidations[key] = yup.mixed();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (typeof substepArgumentValidations[key] === 'object' && (arg.type === 'string' || arg.type === 'dropdown')) {
|
if (
|
||||||
|
typeof substepArgumentValidations[key] === 'object' &&
|
||||||
|
(arg.type === 'string' || arg.type === 'dropdown')
|
||||||
|
) {
|
||||||
// if the field is required, add the required validation
|
// if the field is required, add the required validation
|
||||||
if (required) {
|
if (required) {
|
||||||
substepArgumentValidations[key] = substepArgumentValidations[
|
substepArgumentValidations[key] = substepArgumentValidations[key]
|
||||||
key
|
|
||||||
]
|
|
||||||
.required(`${key} is required.`)
|
.required(`${key} is required.`)
|
||||||
.test(
|
.test(
|
||||||
'empty-check',
|
'empty-check',
|
||||||
`${key} must be not empty`,
|
`${key} must be not empty`,
|
||||||
(value: any) => !isEmpty(value),
|
(value: any) => !isEmpty(value)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -166,7 +167,9 @@ export default function FlowStep(
|
|||||||
|
|
||||||
const actionsOrTriggers: Array<ITrigger | IAction> =
|
const actionsOrTriggers: Array<ITrigger | IAction> =
|
||||||
(isTrigger ? app?.triggers : app?.actions) || [];
|
(isTrigger ? app?.triggers : app?.actions) || [];
|
||||||
const actionOrTrigger = actionsOrTriggers?.find(({ key }) => key === step.key);
|
const actionOrTrigger = actionsOrTriggers?.find(
|
||||||
|
({ key }) => key === step.key
|
||||||
|
);
|
||||||
const substeps = actionOrTrigger?.substeps || [];
|
const substeps = actionOrTrigger?.substeps || [];
|
||||||
|
|
||||||
const handleChange = React.useCallback(({ step }: { step: IStep }) => {
|
const handleChange = React.useCallback(({ step }: { step: IStep }) => {
|
||||||
@@ -187,7 +190,12 @@ export default function FlowStep(
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (!apps) {
|
if (!apps) {
|
||||||
return <CircularProgress sx={{ display: 'block', my: 2 }} />;
|
return (
|
||||||
|
<CircularProgress
|
||||||
|
data-test="step-circular-loader"
|
||||||
|
sx={{ display: 'block', my: 2 }}
|
||||||
|
/>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const onContextMenuClose = (event: React.SyntheticEvent) => {
|
const onContextMenuClose = (event: React.SyntheticEvent) => {
|
||||||
@@ -279,7 +287,8 @@ export default function FlowStep(
|
|||||||
step={step}
|
step={step}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{actionOrTrigger && substeps?.length > 0 &&
|
{actionOrTrigger &&
|
||||||
|
substeps?.length > 0 &&
|
||||||
substeps.map((substep: ISubstep, index: number) => (
|
substeps.map((substep: ISubstep, index: number) => (
|
||||||
<React.Fragment key={`${substep?.name}-${index}`}>
|
<React.Fragment key={`${substep?.name}-${index}`}>
|
||||||
{substep.key === 'chooseConnection' && app && (
|
{substep.key === 'chooseConnection' && app && (
|
||||||
@@ -304,7 +313,11 @@ export default function FlowStep(
|
|||||||
onSubmit={expandNextStep}
|
onSubmit={expandNextStep}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
onContinue={onContinue}
|
onContinue={onContinue}
|
||||||
showWebhookUrl={'showWebhookUrl' in actionOrTrigger ? actionOrTrigger.showWebhookUrl : false}
|
showWebhookUrl={
|
||||||
|
'showWebhookUrl' in actionOrTrigger
|
||||||
|
? actionOrTrigger.showWebhookUrl
|
||||||
|
: false
|
||||||
|
}
|
||||||
step={step}
|
step={step}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
Reference in New Issue
Block a user