LibreChat/e2e/setup/authenticate.ts
Danny Avila b993d9fb28
🛟 test: Restore Playwright Smoke E2E (#13020)
* test: restore Playwright smoke e2e

* test: harden e2e smoke setup

* test: sync e2e server bindings

* test: normalize e2e auth urls
2026-05-14 09:49:26 -04:00

103 lines
3.7 KiB
TypeScript

import { chromium } from '@playwright/test';
import type { FullConfig, Page } from '@playwright/test';
import type { User } from '../types';
import cleanupUser from './cleanupUser';
import dotenv from 'dotenv';
dotenv.config();
const timeout = Number(process.env.E2E_AUTH_TIMEOUT ?? 15000);
async function register(page: Page, user: User) {
await page.getByRole('link', { name: 'Sign up' }).click();
await page.getByLabel('Full name').click();
await page.getByLabel('Full name').fill(user.name);
await page.getByLabel('Email').click();
await page.getByLabel('Email').fill(user.email);
await page.getByLabel('Email').press('Tab');
await page.getByTestId('password').click();
await page.getByTestId('password').fill(user.password);
await page.getByTestId('confirm_password').click();
await page.getByTestId('confirm_password').fill(user.password);
await page.getByLabel('Submit registration').click();
}
async function registrationErrorIsVisible(page: Page) {
return page
.getByTestId('registration-error')
.isVisible({ timeout: 500 })
.catch(() => false);
}
async function login(page: Page, user: User) {
await page.getByLabel('Email').fill(user.email);
await page.getByLabel('Password').fill(user.password);
await page.getByTestId('login-button').click();
}
function appURL(baseURL: string, pathname = '') {
const normalizedBaseURL = baseURL.endsWith('/') ? baseURL : `${baseURL}/`;
return new URL(pathname.replace(/^\/+/, ''), normalizedBaseURL).toString();
}
async function authenticate(config: FullConfig, user: User) {
console.log('🤖: global setup has been started');
const { baseURL, storageState } = config.projects[0].use;
console.log('🤖: using baseURL', baseURL);
console.log('🤖: using E2E user:', user.email);
if (typeof storageState !== 'string') {
throw new Error('🤖: storageState must be a file path');
}
const browser = await chromium.launch({
headless: config.projects[0].use.headless ?? true,
});
try {
const page = await browser.newPage();
console.log('🤖: 🗝 authenticating user:', user.email);
if (typeof baseURL !== 'string') {
throw new Error('🤖: baseURL is not defined');
}
const conversationURL = appURL(baseURL, 'c/new');
const loginURL = appURL(baseURL, 'login');
// Set localStorage before navigating to the page
await page.context().addInitScript(() => {
localStorage.setItem('navVisible', 'true');
});
console.log('🤖: ✔️ localStorage: set Nav as Visible', storageState);
await page.goto(baseURL, { timeout });
await register(page, user);
try {
await page.waitForURL(conversationURL, { timeout });
} catch (error) {
console.error('Error:', error);
if (await registrationErrorIsVisible(page)) {
console.log('🤖: 🚨 user already exists');
await cleanupUser(user);
await page.goto(baseURL, { timeout });
await register(page, user);
await page.waitForURL(conversationURL, { timeout });
} else {
throw new Error('🤖: 🚨 user failed to register');
}
}
console.log('🤖: ✔️ user successfully registered');
await page.goto(loginURL, { timeout });
await login(page, user);
await page.waitForURL(conversationURL, { timeout });
console.log('🤖: ✔️ user successfully authenticated');
await page.context().storageState({ path: storageState });
console.log('🤖: ✔️ authentication state successfully saved in', storageState);
// await browser.close();
// console.log('🤖: global setup has been finished');
} finally {
await browser.close();
console.log('🤖: global setup has been finished');
}
}
export default authenticate;