Workflow e2e tests – 1st batch (#9713)
- Clean Playwright's configuration:
- Remove artificial 500ms delay between each step
- Group all tests under a `chrome` project relying on a `setup` project
to get an authentication state which all tests can reuse
- Changes on the `Sign up with invite link via email` test:
- Generate a new email for each test trial, as previously it was failing
when run many times
- Make deleting the account part of the test; if we write other tests
for account sign-up, we'll prefer to delete the accounts with an HTTP
call to speed up things
- Added some assertions to ensure we reached steps when expected, as we
removed the 500ms delay between each step, and it made some assertions
fail
- Wrote new tests for workflows:
- Created `Create workflow`, a test asserting we can create a workflow
from the record table
- Created `Create simple workflow`, a test asserting we can create a
simple flow; I will add more assertions to this test and write other
tests once this first PR is approved
- I make HTTP calls to delete and destroy workflows after they run to
keep the database clean
- Added a data-testid to ensure we focus elements from the Cmd+K; our
selectors are not strong – see `getByRole('textbox')` – and I preferred
to scope them to a root element
- Added an `aria-label` to a button
---------
Co-authored-by: prastoin <paul@twenty.com>
This commit is contained in:
committed by
GitHub
parent
d50294d39a
commit
7ed2c12e7a
8
.github/workflows/ci-e2e.yaml
vendored
8
.github/workflows/ci-e2e.yaml
vendored
@ -69,11 +69,12 @@ jobs:
|
|||||||
- name: Build twenty-shared
|
- name: Build twenty-shared
|
||||||
run: npx nx build twenty-shared
|
run: npx nx build twenty-shared
|
||||||
|
|
||||||
|
- name: Install Playwright Browsers
|
||||||
|
run: npx nx setup twenty-e2e-testing
|
||||||
|
|
||||||
- name: Setup environment files
|
- name: Setup environment files
|
||||||
run: |
|
run: |
|
||||||
cp packages/twenty-e2e-testing/.env.example packages/twenty-e2e-testing/.env
|
|
||||||
cp packages/twenty-front/.env.example packages/twenty-front/.env
|
cp packages/twenty-front/.env.example packages/twenty-front/.env
|
||||||
cp packages/twenty-e2e-testing/.env.example packages/twenty-e2e-testing/.env
|
|
||||||
npx nx reset:env twenty-server
|
npx nx reset:env twenty-server
|
||||||
|
|
||||||
- name: Build frontend
|
- name: Build frontend
|
||||||
@ -105,9 +106,6 @@ jobs:
|
|||||||
npx nx run twenty-server:worker:ci &
|
npx nx run twenty-server:worker:ci &
|
||||||
echo "Worker started"
|
echo "Worker started"
|
||||||
|
|
||||||
- name: Install Playwright Browsers
|
|
||||||
run: npx nx setup twenty-e2e-testing
|
|
||||||
|
|
||||||
- name: Run Playwright tests
|
- name: Run Playwright tests
|
||||||
run: npx nx test twenty-e2e-testing
|
run: npx nx test twenty-e2e-testing
|
||||||
|
|
||||||
|
|||||||
@ -1,5 +1,6 @@
|
|||||||
# Note that provide always without trailing forward slash to have expected behaviour
|
# Note that provide always without trailing forward slash to have expected behaviour
|
||||||
FRONTEND_BASE_URL=http://localhost:3001
|
FRONTEND_BASE_URL=http://localhost:3001
|
||||||
|
BACKEND_BASE_URL=http://localhost:3000
|
||||||
DEFAULT_LOGIN=tim@apple.dev
|
DEFAULT_LOGIN=tim@apple.dev
|
||||||
DEFAULT_PASSWORD=Applecar2025
|
DEFAULT_PASSWORD=Applecar2025
|
||||||
WEBSITE_URL=https://twenty.com
|
WEBSITE_URL=https://twenty.com
|
||||||
|
|||||||
@ -1,33 +0,0 @@
|
|||||||
import {
|
|
||||||
Reporter,
|
|
||||||
FullConfig,
|
|
||||||
Suite,
|
|
||||||
TestCase,
|
|
||||||
TestResult,
|
|
||||||
FullResult,
|
|
||||||
} from '@playwright/test/reporter';
|
|
||||||
|
|
||||||
class CustomReporter implements Reporter {
|
|
||||||
constructor(options: { customOption?: string } = {}) {
|
|
||||||
console.log(
|
|
||||||
`my-awesome-reporter setup with customOption set to ${options.customOption}`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
onBegin(config: FullConfig, suite: Suite) {
|
|
||||||
console.log(`Starting the run with ${suite.allTests().length} tests`);
|
|
||||||
}
|
|
||||||
|
|
||||||
onTestBegin(test: TestCase) {
|
|
||||||
console.log(`Starting test ${test.title}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
onTestEnd(test: TestCase, result: TestResult) {
|
|
||||||
console.log(`Finished test ${test.title}: ${result.status}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
onEnd(result: FullResult) {
|
|
||||||
console.log(`Finished the run: ${result.status}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
export default CustomReporter;
|
|
||||||
@ -1,21 +0,0 @@
|
|||||||
import * as fs from 'fs';
|
|
||||||
import path from 'path';
|
|
||||||
|
|
||||||
export const envVariables = (variables: string) => {
|
|
||||||
let payload = `
|
|
||||||
PG_DATABASE_URL=postgres://postgres:postgres@localhost:5432/default
|
|
||||||
ACCESS_TOKEN_SECRET=replace_me_with_a_random_string_access
|
|
||||||
LOGIN_TOKEN_SECRET=replace_me_with_a_random_string_login
|
|
||||||
REFRESH_TOKEN_SECRET=replace_me_with_a_random_string_refresh
|
|
||||||
FILE_TOKEN_SECRET=replace_me_with_a_random_string_refresh
|
|
||||||
REDIS_URL=redis://localhost:6379
|
|
||||||
`;
|
|
||||||
payload = payload.concat(variables);
|
|
||||||
fs.writeFile(
|
|
||||||
path.join(__dirname, '..', '..', 'twenty-server', '.env'),
|
|
||||||
payload,
|
|
||||||
(err) => {
|
|
||||||
throw err;
|
|
||||||
},
|
|
||||||
);
|
|
||||||
};
|
|
||||||
@ -1,13 +0,0 @@
|
|||||||
import { exec } from 'child_process';
|
|
||||||
|
|
||||||
export async function sh(cmd) {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
exec(cmd, (err, stdout, stderr) => {
|
|
||||||
if (err) {
|
|
||||||
reject(err);
|
|
||||||
} else {
|
|
||||||
resolve({ stdout, stderr });
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
69
packages/twenty-e2e-testing/lib/fixtures/blank-workflow.ts
Normal file
69
packages/twenty-e2e-testing/lib/fixtures/blank-workflow.ts
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
import { test as base, expect, Page } from '@playwright/test';
|
||||||
|
import { randomUUID } from 'node:crypto';
|
||||||
|
import { createWorkflow } from '../requests/create-workflow';
|
||||||
|
import { deleteWorkflow } from '../requests/delete-workflow';
|
||||||
|
import { destroyWorkflow } from '../requests/destroy-workflow';
|
||||||
|
|
||||||
|
export class WorkflowVisualizerPage {
|
||||||
|
#page: Page;
|
||||||
|
|
||||||
|
workflowId: string;
|
||||||
|
workflowName: string;
|
||||||
|
|
||||||
|
constructor({ page, workflowName }: { page: Page; workflowName: string }) {
|
||||||
|
this.#page = page;
|
||||||
|
this.workflowName = workflowName;
|
||||||
|
}
|
||||||
|
|
||||||
|
async createOneWorkflow() {
|
||||||
|
const id = randomUUID();
|
||||||
|
|
||||||
|
const response = await createWorkflow({
|
||||||
|
page: this.#page,
|
||||||
|
workflowId: id,
|
||||||
|
workflowName: this.workflowName,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(response.status()).toBe(200);
|
||||||
|
|
||||||
|
const responseBody = await response.json();
|
||||||
|
expect(responseBody.data.createWorkflow.id).toBe(id);
|
||||||
|
|
||||||
|
this.workflowId = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
async goToWorkflowVisualizerPage() {
|
||||||
|
await this.#page.goto(`/object/workflow/${this.workflowId}`);
|
||||||
|
|
||||||
|
const workflowName = this.#page.getByRole('button', {
|
||||||
|
name: this.workflowName,
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect(workflowName).toBeVisible();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const test = base.extend<{ workflowVisualizer: WorkflowVisualizerPage }>(
|
||||||
|
{
|
||||||
|
workflowVisualizer: async ({ page }, use) => {
|
||||||
|
const workflowVisualizer = new WorkflowVisualizerPage({
|
||||||
|
page,
|
||||||
|
workflowName: 'Test Workflow',
|
||||||
|
});
|
||||||
|
|
||||||
|
await workflowVisualizer.createOneWorkflow();
|
||||||
|
await workflowVisualizer.goToWorkflowVisualizerPage();
|
||||||
|
|
||||||
|
await use(workflowVisualizer);
|
||||||
|
|
||||||
|
await deleteWorkflow({
|
||||||
|
page,
|
||||||
|
workflowId: workflowVisualizer.workflowId,
|
||||||
|
});
|
||||||
|
await destroyWorkflow({
|
||||||
|
page,
|
||||||
|
workflowId: workflowVisualizer.workflowId,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
@ -1,4 +1,4 @@
|
|||||||
import { Locator, Page } from '@playwright/test';
|
import { expect, Locator, Page } from '@playwright/test';
|
||||||
|
|
||||||
export class LoginPage {
|
export class LoginPage {
|
||||||
private readonly loginWithGoogleButton: Locator;
|
private readonly loginWithGoogleButton: Locator;
|
||||||
@ -98,6 +98,8 @@ export class LoginPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async typeEmail(email: string) {
|
async typeEmail(email: string) {
|
||||||
|
await expect(this.emailField).toBeVisible();
|
||||||
|
|
||||||
await this.emailField.fill(email);
|
await this.emailField.fill(email);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
4
packages/twenty-e2e-testing/lib/requests/backend.ts
Normal file
4
packages/twenty-e2e-testing/lib/requests/backend.ts
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
export const backendGraphQLUrl = new URL(
|
||||||
|
'/graphql',
|
||||||
|
process.env.BACKEND_BASE_URL,
|
||||||
|
).toString();
|
||||||
32
packages/twenty-e2e-testing/lib/requests/create-workflow.ts
Normal file
32
packages/twenty-e2e-testing/lib/requests/create-workflow.ts
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
import { Page } from '@playwright/test';
|
||||||
|
import { getAuthToken } from '../utils/getAuthToken';
|
||||||
|
import { backendGraphQLUrl } from './backend';
|
||||||
|
|
||||||
|
export const createWorkflow = async ({
|
||||||
|
page,
|
||||||
|
workflowId,
|
||||||
|
workflowName,
|
||||||
|
}: {
|
||||||
|
page: Page;
|
||||||
|
workflowId: string;
|
||||||
|
workflowName: string;
|
||||||
|
}) => {
|
||||||
|
const { authToken } = await getAuthToken(page);
|
||||||
|
|
||||||
|
return page.request.post(backendGraphQLUrl, {
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${authToken}`,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
operationName: 'CreateOneWorkflow',
|
||||||
|
query:
|
||||||
|
'mutation CreateOneWorkflow($input: WorkflowCreateInput!) { createWorkflow(data: $input) { __typename id } }',
|
||||||
|
variables: {
|
||||||
|
input: {
|
||||||
|
id: workflowId,
|
||||||
|
name: workflowName,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
25
packages/twenty-e2e-testing/lib/requests/delete-workflow.ts
Normal file
25
packages/twenty-e2e-testing/lib/requests/delete-workflow.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import { Page } from '@playwright/test';
|
||||||
|
import { getAuthToken } from '../utils/getAuthToken';
|
||||||
|
import { backendGraphQLUrl } from './backend';
|
||||||
|
|
||||||
|
export const deleteWorkflow = async ({
|
||||||
|
page,
|
||||||
|
workflowId,
|
||||||
|
}: {
|
||||||
|
page: Page;
|
||||||
|
workflowId: string;
|
||||||
|
}) => {
|
||||||
|
const { authToken } = await getAuthToken(page);
|
||||||
|
|
||||||
|
return page.request.post(backendGraphQLUrl, {
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${authToken}`,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
operationName: 'DeleteOneWorkflow',
|
||||||
|
variables: { idToDelete: workflowId },
|
||||||
|
query:
|
||||||
|
'mutation DeleteOneWorkflow($idToDelete: ID!) {\n deleteWorkflow(id: $idToDelete) {\n __typename\n deletedAt\n id\n }\n}',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
25
packages/twenty-e2e-testing/lib/requests/destroy-workflow.ts
Normal file
25
packages/twenty-e2e-testing/lib/requests/destroy-workflow.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import { Page } from '@playwright/test';
|
||||||
|
import { getAuthToken } from '../utils/getAuthToken';
|
||||||
|
import { backendGraphQLUrl } from './backend';
|
||||||
|
|
||||||
|
export const destroyWorkflow = async ({
|
||||||
|
page,
|
||||||
|
workflowId,
|
||||||
|
}: {
|
||||||
|
page: Page;
|
||||||
|
workflowId: string;
|
||||||
|
}) => {
|
||||||
|
const { authToken } = await getAuthToken(page);
|
||||||
|
|
||||||
|
return page.request.post(backendGraphQLUrl, {
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${authToken}`,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
operationName: 'DestroyOneWorkflow',
|
||||||
|
variables: { idToDestroy: workflowId },
|
||||||
|
query:
|
||||||
|
'mutation DestroyOneWorkflow($idToDestroy: ID!) {\n destroyWorkflow(id: $idToDestroy) {\n id\n __typename\n }\n}',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
15
packages/twenty-e2e-testing/lib/utils/getAuthToken.ts
Normal file
15
packages/twenty-e2e-testing/lib/utils/getAuthToken.ts
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
import { Page } from '@playwright/test';
|
||||||
|
|
||||||
|
export const getAuthToken = async (page: Page) => {
|
||||||
|
const storageState = await page.context().storageState();
|
||||||
|
const authCookie = storageState.cookies.find(
|
||||||
|
(cookie) => cookie.name === 'tokenPair',
|
||||||
|
);
|
||||||
|
if (!authCookie) {
|
||||||
|
throw new Error('No auth cookie found');
|
||||||
|
}
|
||||||
|
const token = JSON.parse(decodeURIComponent(authCookie.value)).accessToken
|
||||||
|
.token;
|
||||||
|
|
||||||
|
return { authToken: token };
|
||||||
|
};
|
||||||
@ -30,10 +30,6 @@ export default defineConfig({
|
|||||||
screenshot: 'on', // either 'on' here or in different method in modules, if 'on' all screenshots are overwritten each time the test is run
|
screenshot: 'on', // either 'on' here or in different method in modules, if 'on' all screenshots are overwritten each time the test is run
|
||||||
headless: true, // instead of changing it to false, run 'yarn test:e2e:debug' or 'yarn test:e2e:ui'
|
headless: true, // instead of changing it to false, run 'yarn test:e2e:debug' or 'yarn test:e2e:ui'
|
||||||
testIdAttribute: 'data-testid', // taken from Twenty source
|
testIdAttribute: 'data-testid', // taken from Twenty source
|
||||||
viewport: { width: 1920, height: 1080 }, // most laptops use this resolution
|
|
||||||
launchOptions: {
|
|
||||||
slowMo: 500, // time in milliseconds between each step, better to use it than explicitly define timeout in tests
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
expect: {
|
expect: {
|
||||||
timeout: 5000,
|
timeout: 5000,
|
||||||
@ -41,24 +37,17 @@ export default defineConfig({
|
|||||||
reporter: process.env.CI ? 'github' : 'list',
|
reporter: process.env.CI ? 'github' : 'list',
|
||||||
projects: [
|
projects: [
|
||||||
{
|
{
|
||||||
name: 'Login setup',
|
name: 'setup',
|
||||||
testMatch: /login\.setup\.ts/, // finds all tests matching this regex, in this case only 1 test should be found
|
testMatch: /.*\.setup\.ts/,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'Demo check',
|
name: 'chrome',
|
||||||
use: {
|
use: {
|
||||||
...devices['Desktop Chrome'],
|
...devices['Desktop Chrome'],
|
||||||
},
|
|
||||||
testMatch: /demo\/demo_basic\.e2e-spec\.ts/,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: 'Authentication',
|
|
||||||
use: {
|
|
||||||
permissions: ['clipboard-read', 'clipboard-write'],
|
permissions: ['clipboard-read', 'clipboard-write'],
|
||||||
storageState: path.resolve(__dirname, '.auth', 'user.json'), // takes saved cookies from directory
|
storageState: path.resolve(__dirname, '.auth', 'user.json'), // takes saved cookies from directory
|
||||||
},
|
},
|
||||||
dependencies: ['Login setup'],
|
dependencies: ['setup'],
|
||||||
testMatch: /authentication\/.+\.e2e-spec\.ts/, // forces to run login setup before running tests from this project - CASE SENSITIVE
|
|
||||||
},
|
},
|
||||||
|
|
||||||
//{
|
//{
|
||||||
|
|||||||
@ -8,7 +8,8 @@
|
|||||||
"options": {
|
"options": {
|
||||||
"cwd": "packages/twenty-e2e-testing",
|
"cwd": "packages/twenty-e2e-testing",
|
||||||
"commands": [
|
"commands": [
|
||||||
"yarn playwright install"
|
"yarn playwright install",
|
||||||
|
"cp .env.example .env"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@ -1,64 +0,0 @@
|
|||||||
import { expect, test } from './fixture';
|
|
||||||
|
|
||||||
test.describe('Authentication test', () => {
|
|
||||||
const email = 'test@apple.dev';
|
|
||||||
const firstName = 'John';
|
|
||||||
const lastName = 'Doe';
|
|
||||||
let testCompleted = false;
|
|
||||||
|
|
||||||
test('Sign up with invite link via email', async ({
|
|
||||||
page,
|
|
||||||
loginPage,
|
|
||||||
leftMenu,
|
|
||||||
membersSection,
|
|
||||||
settingsPage,
|
|
||||||
}) => {
|
|
||||||
const inviteLink: string =
|
|
||||||
await test.step('Go to Settings and copy invite link', async () => {
|
|
||||||
await page.goto(process.env.LINK); // skip login page (and redirect) when running on environments with multi-workspace enabled
|
|
||||||
await leftMenu.goToSettings();
|
|
||||||
await settingsPage.goToMembersSection();
|
|
||||||
await membersSection.copyInviteLink();
|
|
||||||
return await page.evaluate('navigator.clipboard.readText()');
|
|
||||||
});
|
|
||||||
await test.step('Go to invite link', async () => {
|
|
||||||
await settingsPage.logout();
|
|
||||||
await page.goto(inviteLink);
|
|
||||||
});
|
|
||||||
await test.step('Create new account', async () => {
|
|
||||||
await loginPage.clickLoginWithEmail();
|
|
||||||
await loginPage.typeEmail(email);
|
|
||||||
await loginPage.clickContinueButton();
|
|
||||||
await loginPage.typePassword(process.env.DEFAULT_PASSWORD);
|
|
||||||
await loginPage.clickSignUpButton();
|
|
||||||
await loginPage.typeFirstName(firstName);
|
|
||||||
await loginPage.typeLastName(lastName);
|
|
||||||
await loginPage.clickContinueButton();
|
|
||||||
await loginPage.noSyncWithGoogle();
|
|
||||||
testCompleted = true;
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
test.afterEach(
|
|
||||||
async ({
|
|
||||||
page,
|
|
||||||
confirmationModal,
|
|
||||||
leftMenu,
|
|
||||||
profileSection,
|
|
||||||
settingsPage,
|
|
||||||
}) => {
|
|
||||||
if (testCompleted) {
|
|
||||||
// security measurement to clean up only after test is completed,
|
|
||||||
// otherwise default account used for tests may be deleted and resetting database will be necessary
|
|
||||||
await test.step('Cleanup - deleting account', async () => {
|
|
||||||
await leftMenu.goToSettings();
|
|
||||||
await settingsPage.goToProfileSection();
|
|
||||||
await profileSection.deleteAccount();
|
|
||||||
await confirmationModal.typePlaceholderToInput();
|
|
||||||
await confirmationModal.clickConfirmButton();
|
|
||||||
expect(page.url()).toContain('/welcome');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
});
|
|
||||||
@ -0,0 +1,60 @@
|
|||||||
|
import { randomUUID } from 'crypto';
|
||||||
|
import { expect, test } from './fixture';
|
||||||
|
|
||||||
|
test('Sign up with invite link via email', async ({
|
||||||
|
page,
|
||||||
|
loginPage,
|
||||||
|
leftMenu,
|
||||||
|
membersSection,
|
||||||
|
settingsPage,
|
||||||
|
profileSection,
|
||||||
|
confirmationModal,
|
||||||
|
}) => {
|
||||||
|
const email = `test${randomUUID().replaceAll('-', '')}@apple.dev`;
|
||||||
|
const firstName = 'John';
|
||||||
|
const lastName = 'Doe';
|
||||||
|
|
||||||
|
const inviteLink: string =
|
||||||
|
await test.step('Go to Settings and copy invite link', async () => {
|
||||||
|
await page.goto(process.env.LINK); // skip login page (and redirect) when running on environments with multi-workspace enabled
|
||||||
|
await leftMenu.goToSettings();
|
||||||
|
await settingsPage.goToMembersSection();
|
||||||
|
await membersSection.copyInviteLink();
|
||||||
|
return await page.evaluate('navigator.clipboard.readText()');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('Go to invite link', async () => {
|
||||||
|
await settingsPage.logout();
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
expect(page.getByText(/Join .+ team/)).toBeVisible(),
|
||||||
|
|
||||||
|
page.goto(inviteLink),
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('Create new account', async () => {
|
||||||
|
await loginPage.clickLoginWithEmail();
|
||||||
|
await loginPage.typeEmail(email);
|
||||||
|
await loginPage.clickContinueButton();
|
||||||
|
await loginPage.typePassword(process.env.DEFAULT_PASSWORD);
|
||||||
|
await loginPage.clickSignUpButton();
|
||||||
|
await loginPage.typeFirstName(firstName);
|
||||||
|
await loginPage.typeLastName(lastName);
|
||||||
|
await loginPage.clickContinueButton();
|
||||||
|
await loginPage.noSyncWithGoogle();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('Delete account from workspace', async () => {
|
||||||
|
await leftMenu.goToSettings();
|
||||||
|
await settingsPage.goToProfileSection();
|
||||||
|
await profileSection.deleteAccount();
|
||||||
|
await confirmationModal.typePlaceholderToInput();
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForURL('/welcome'),
|
||||||
|
|
||||||
|
confirmationModal.clickConfirmButton(),
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -1,6 +1,6 @@
|
|||||||
import { expect, test as base } from '@playwright/test';
|
import { test as base, expect } from '@playwright/test';
|
||||||
import { LoginPage } from '../lib/pom/loginPage';
|
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
|
import { LoginPage } from '../lib/pom/loginPage';
|
||||||
|
|
||||||
// fixture
|
// fixture
|
||||||
const test = base.extend<{ loginPage: LoginPage }>({
|
const test = base.extend<{ loginPage: LoginPage }>({
|
||||||
@ -29,7 +29,7 @@ test('Login test', async ({ loginPage, page }) => {
|
|||||||
await loginPage.typePassword(process.env.DEFAULT_PASSWORD);
|
await loginPage.typePassword(process.env.DEFAULT_PASSWORD);
|
||||||
await page.waitForLoadState('networkidle');
|
await page.waitForLoadState('networkidle');
|
||||||
await loginPage.clickSignInButton();
|
await loginPage.clickSignInButton();
|
||||||
await expect(page.getByText('Welcome to Twenty')).not.toBeVisible();
|
await expect(page.getByText(/Welcome to .+/)).not.toBeVisible();
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
67
packages/twenty-e2e-testing/tests/workflow-creation.spec.ts
Normal file
67
packages/twenty-e2e-testing/tests/workflow-creation.spec.ts
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
import { expect, test } from '@playwright/test';
|
||||||
|
import { deleteWorkflow } from '../lib/requests/delete-workflow';
|
||||||
|
import { destroyWorkflow } from '../lib/requests/destroy-workflow';
|
||||||
|
|
||||||
|
test('Create workflow', async ({ page }) => {
|
||||||
|
const NEW_WORKFLOW_NAME = 'Test Workflow';
|
||||||
|
|
||||||
|
await page.goto('/');
|
||||||
|
|
||||||
|
const workflowsLink = page.getByRole('link', { name: 'Workflows' });
|
||||||
|
await workflowsLink.click();
|
||||||
|
|
||||||
|
const createWorkflowButton = page.getByRole('button', { name: 'New record' });
|
||||||
|
|
||||||
|
const [createWorkflowResponse] = await Promise.all([
|
||||||
|
page.waitForResponse(async (response) => {
|
||||||
|
if (!response.url().endsWith('/graphql')) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const requestBody = response.request().postDataJSON();
|
||||||
|
|
||||||
|
return requestBody.operationName === 'CreateOneWorkflow';
|
||||||
|
}),
|
||||||
|
|
||||||
|
await createWorkflowButton.click(),
|
||||||
|
]);
|
||||||
|
|
||||||
|
const nameInputClosedState = page.getByText('Name').first();
|
||||||
|
await nameInputClosedState.click();
|
||||||
|
|
||||||
|
const nameInput = page.getByRole('textbox');
|
||||||
|
await nameInput.fill(NEW_WORKFLOW_NAME);
|
||||||
|
await nameInput.press('Enter');
|
||||||
|
|
||||||
|
const body = await createWorkflowResponse.json();
|
||||||
|
const newWorkflowId = body.data.createWorkflow.id;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const newWorkflowRowEntryName = page
|
||||||
|
.getByTestId(`row-id-${newWorkflowId}`)
|
||||||
|
.locator('div')
|
||||||
|
.filter({ hasText: NEW_WORKFLOW_NAME })
|
||||||
|
.nth(2);
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForURL(
|
||||||
|
(url) => url.pathname === `/object/workflow/${newWorkflowId}`,
|
||||||
|
),
|
||||||
|
|
||||||
|
newWorkflowRowEntryName.click(),
|
||||||
|
]);
|
||||||
|
|
||||||
|
const workflowName = page.getByRole('button', { name: NEW_WORKFLOW_NAME });
|
||||||
|
|
||||||
|
await expect(workflowName).toBeVisible();
|
||||||
|
} finally {
|
||||||
|
await deleteWorkflow({
|
||||||
|
page,
|
||||||
|
workflowId: newWorkflowId,
|
||||||
|
});
|
||||||
|
await destroyWorkflow({
|
||||||
|
page,
|
||||||
|
workflowId: newWorkflowId,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
@ -0,0 +1,35 @@
|
|||||||
|
import { expect } from '@playwright/test';
|
||||||
|
import { test } from '../lib/fixtures/blank-workflow';
|
||||||
|
|
||||||
|
test('Create simple workflow', async ({ workflowVisualizer, page }) => {
|
||||||
|
const addTriggerButton = page.getByText('Add a Trigger');
|
||||||
|
await addTriggerButton.click();
|
||||||
|
|
||||||
|
const triggerOption = page.getByText('Database Event');
|
||||||
|
await triggerOption.click();
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
page.getByTestId('command-menu').getByRole('textbox'),
|
||||||
|
).toHaveValue('When a Company is Created');
|
||||||
|
|
||||||
|
const triggerNode = page.getByTestId('rf__node-trigger');
|
||||||
|
await expect(triggerNode).toHaveClass(/selected/);
|
||||||
|
await expect(triggerNode).toHaveText(/Company is Created/);
|
||||||
|
|
||||||
|
const addStepButton = page.getByLabel('Add a step');
|
||||||
|
await addStepButton.click();
|
||||||
|
|
||||||
|
const createRecordOption = page.getByText('Create Record');
|
||||||
|
|
||||||
|
await createRecordOption.click();
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
page.getByTestId('command-menu').getByRole('textbox').first(),
|
||||||
|
).toHaveValue('Create Record');
|
||||||
|
|
||||||
|
const createRecordNode = page
|
||||||
|
.locator('.react-flow__node.selected')
|
||||||
|
.getByText('Create Record');
|
||||||
|
await expect(createRecordNode).toBeVisible();
|
||||||
|
await expect(triggerNode).not.toHaveClass(/selected/);
|
||||||
|
});
|
||||||
@ -91,6 +91,7 @@ export const CommandMenuContainer = ({
|
|||||||
<ActionMenuConfirmationModals />
|
<ActionMenuConfirmationModals />
|
||||||
{isCommandMenuOpened && (
|
{isCommandMenuOpened && (
|
||||||
<StyledCommandMenu
|
<StyledCommandMenu
|
||||||
|
data-testid="command-menu"
|
||||||
ref={commandMenuRef}
|
ref={commandMenuRef}
|
||||||
className="command-menu"
|
className="command-menu"
|
||||||
animate={targetVariantForAnimation}
|
animate={targetVariantForAnimation}
|
||||||
|
|||||||
@ -15,7 +15,7 @@ export const WorkflowDiagramCreateStepNode = () => {
|
|||||||
<StyledContainer>
|
<StyledContainer>
|
||||||
<StyledTargetHandle type="target" position={Position.Top} />
|
<StyledTargetHandle type="target" position={Position.Top} />
|
||||||
|
|
||||||
<IconButton Icon={IconPlus} size="medium" />
|
<IconButton Icon={IconPlus} size="medium" ariaLabel="Add a step" />
|
||||||
</StyledContainer>
|
</StyledContainer>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user