mirror of
https://github.com/actualbudget/actual.git
synced 2026-03-21 06:58:47 -05:00
* fix: use start:browser for playwright webServer to load correct browser-preload The playwright webServer was running 'yarn start' which doesn't set IS_GENERIC_BROWSER, causing Vite to not resolve .browser.js files. This resulted in browser-preload.js (empty electron stub) being loaded instead of browser-preload.browser.js, making window.Actual undefined. Changed to 'yarn start:browser' which properly sets IS_GENERIC_BROWSER=1 via the watch-browser script. * chore: update Playwright configuration and add blob-report to .gitignore - Modified Playwright config to use 'yarn start' with the correct working directory for local builds. - Added 'blob-report' to .gitignore to exclude it from version control. - Created release notes for running e2e tests against a local build instead of Netlify. * chore: update transaction test snapshots for split and transfer transactions - Updated binary snapshots for split and transfer test transactions in the e2e tests. - Ensured that the latest visual changes are reflected in the test suite for accurate regression testing. * chore: update transaction test snapshots for split and transfer transactions - Updated binary snapshots for split and transfer test transactions in the e2e tests to reflect recent changes. - Ensured visual consistency for accurate regression testing. * refactor: change test lifecycle hooks from beforeAll/afterAll to beforeEach/afterEach - Updated test files to use beforeEach and afterEach hooks for better isolation of tests. - This change ensures that each test starts with a fresh state, improving reliability and reducing side effects across tests. * chore: update e2e test workflow to disable translation downloads - Modified the e2e test workflow to include a new input parameter `download-translations` set to 'false' for the setup action. - This change aims to streamline the testing process by preventing unnecessary translation downloads during the test runs. * chore: update e2e test snapshots for settings page visuals - Updated binary snapshots for the settings page in e2e tests to reflect recent visual changes. - Ensured that the latest visual updates are accurately represented for regression testing. * fix: safely close page in e2e tests - Updated all e2e test files to use optional chaining when closing the page, ensuring that the close method is only called if the page is defined. - This change improves the robustness of the tests by preventing potential errors when the page object is not available.
194 lines
6.3 KiB
TypeScript
194 lines
6.3 KiB
TypeScript
import { join } from 'path';
|
|
|
|
import { type Page } from '@playwright/test';
|
|
|
|
import { expect, test } from './fixtures';
|
|
import { type AccountPage } from './page-models/account-page';
|
|
import { ConfigurationPage } from './page-models/configuration-page';
|
|
import { Navigation } from './page-models/navigation';
|
|
|
|
test.describe('Accounts', () => {
|
|
let page: Page;
|
|
let navigation: Navigation;
|
|
let configurationPage: ConfigurationPage;
|
|
let accountPage: AccountPage;
|
|
|
|
test.beforeEach(async ({ browser }) => {
|
|
page = await browser.newPage();
|
|
navigation = new Navigation(page);
|
|
configurationPage = new ConfigurationPage(page);
|
|
|
|
await page.goto('/');
|
|
await configurationPage.createTestFile();
|
|
});
|
|
|
|
test.afterEach(async () => {
|
|
await page?.close();
|
|
});
|
|
|
|
test('creates a new account and views the initial balance transaction', async () => {
|
|
accountPage = await navigation.createAccount({
|
|
name: 'New Account',
|
|
offBudget: false,
|
|
balance: 100,
|
|
});
|
|
|
|
const transaction = accountPage.getNthTransaction(0);
|
|
await expect(transaction.payee).toHaveText('Starting Balance');
|
|
await expect(transaction.notes).toHaveText('');
|
|
await expect(transaction.category).toHaveText('Starting Balances');
|
|
await expect(transaction.debit).toHaveText('');
|
|
await expect(transaction.credit).toHaveText('100.00');
|
|
await expect(page).toMatchThemeScreenshots();
|
|
});
|
|
|
|
test('closes an account', async () => {
|
|
accountPage = await navigation.goToAccountPage('Roth IRA');
|
|
|
|
await expect(accountPage.accountName).toHaveText('Roth IRA');
|
|
|
|
const modal = await accountPage.clickCloseAccount();
|
|
await modal.selectTransferAccount('Vanguard 401k');
|
|
await expect(page).toMatchThemeScreenshots();
|
|
await modal.closeAccount();
|
|
|
|
await expect(accountPage.accountName).toHaveText('Closed: Roth IRA');
|
|
await expect(page).toMatchThemeScreenshots();
|
|
});
|
|
|
|
test.describe('On Budget Accounts', () => {
|
|
// Reset filters
|
|
test.afterEach(async () => {
|
|
await accountPage.removeFilter(0);
|
|
});
|
|
|
|
test('creates a transfer from two existing transactions', async () => {
|
|
accountPage = await navigation.goToAccountPage('On budget');
|
|
await accountPage.waitFor();
|
|
|
|
await expect(accountPage.accountName).toHaveText('On Budget Accounts');
|
|
|
|
await accountPage.filterByNote('Test Acc Transfer');
|
|
|
|
await accountPage.createSingleTransaction({
|
|
account: 'Ally Savings',
|
|
payee: '',
|
|
notes: 'Test Acc Transfer',
|
|
category: 'Food',
|
|
debit: '34.56',
|
|
});
|
|
|
|
await accountPage.createSingleTransaction({
|
|
account: 'HSBC',
|
|
payee: '',
|
|
notes: 'Test Acc Transfer',
|
|
category: 'Food',
|
|
credit: '34.56',
|
|
});
|
|
|
|
await page.waitForTimeout(100); // Give time for the previous transaction to be rendered
|
|
|
|
await accountPage.selectNthTransaction(0);
|
|
await accountPage.selectNthTransaction(1);
|
|
await accountPage.clickSelectAction('Make transfer');
|
|
|
|
let transaction = accountPage.getNthTransaction(0);
|
|
await expect(transaction.payee).toHaveText('Ally Savings');
|
|
await expect(transaction.category).toHaveText('Transfer');
|
|
await expect(transaction.credit).toHaveText('34.56');
|
|
await expect(transaction.account).toHaveText('HSBC');
|
|
|
|
transaction = accountPage.getNthTransaction(1);
|
|
await expect(transaction.payee).toHaveText('HSBC');
|
|
await expect(transaction.category).toHaveText('Transfer');
|
|
await expect(transaction.debit).toHaveText('34.56');
|
|
await expect(transaction.account).toHaveText('Ally Savings');
|
|
});
|
|
});
|
|
|
|
test.describe('Import Transactions', () => {
|
|
test.beforeEach(async () => {
|
|
accountPage = await navigation.createAccount({
|
|
name: 'CSV import',
|
|
offBudget: false,
|
|
balance: 0,
|
|
});
|
|
await accountPage.waitFor();
|
|
});
|
|
|
|
async function importCsv(screenshot = false) {
|
|
const fileChooserPromise = page.waitForEvent('filechooser');
|
|
await accountPage.page.getByRole('button', { name: 'Import' }).click();
|
|
|
|
const fileChooser = await fileChooserPromise;
|
|
await fileChooser.setFiles(join(__dirname, 'data/test.csv'));
|
|
|
|
const importButton = accountPage.page.getByRole('button', {
|
|
name: /Import \d+ transactions/,
|
|
});
|
|
|
|
await importButton.waitFor({ state: 'visible' });
|
|
|
|
if (screenshot) await expect(page).toMatchThemeScreenshots();
|
|
|
|
await importButton.click();
|
|
|
|
await expect(importButton).not.toBeVisible();
|
|
}
|
|
|
|
test('imports transactions from a CSV file', async () => {
|
|
await importCsv(true);
|
|
});
|
|
|
|
test('import csv file twice', async () => {
|
|
await importCsv(false);
|
|
|
|
const fileChooserPromise = page.waitForEvent('filechooser');
|
|
await accountPage.page.getByRole('button', { name: 'Import' }).click();
|
|
|
|
const fileChooser = await fileChooserPromise;
|
|
await fileChooser.setFiles(join(__dirname, 'data/test.csv'));
|
|
|
|
const importButton = accountPage.page.getByRole('button', {
|
|
name: /Import \d+ transactions/,
|
|
});
|
|
|
|
await importButton.waitFor({ state: 'visible' });
|
|
|
|
await expect(page).toMatchThemeScreenshots();
|
|
|
|
await expect(importButton).toBeDisabled();
|
|
await expect(await importButton.innerText()).toMatch(
|
|
/Import 0 transactions/,
|
|
);
|
|
|
|
await accountPage.page.getByRole('button', { name: 'Close' }).click();
|
|
|
|
await expect(importButton).not.toBeVisible();
|
|
});
|
|
|
|
test('import notes checkbox is not shown for CSV files', async () => {
|
|
const fileChooserPromise = page.waitForEvent('filechooser');
|
|
await accountPage.page.getByRole('button', { name: 'Import' }).click();
|
|
|
|
const fileChooser = await fileChooserPromise;
|
|
await fileChooser.setFiles(join(__dirname, 'data/test.csv'));
|
|
|
|
// Verify the import notes checkbox is not visible for CSV files
|
|
const importNotesCheckbox = page.getByRole('checkbox', {
|
|
name: 'Import notes from file',
|
|
});
|
|
await expect(importNotesCheckbox).not.toBeVisible();
|
|
|
|
// Import the transactions
|
|
const importButton = page.getByRole('button', {
|
|
name: /Import \d+ transactions/,
|
|
});
|
|
await importButton.click();
|
|
|
|
// Verify the transactions were imported
|
|
await expect(importButton).not.toBeVisible();
|
|
});
|
|
});
|
|
});
|