Files
LibreChat/e2e/playwright.config.mock.ts
Danny Avila 2c8d54e18c 🗂️ feat: Add Deployment Skill Directory (#13523)
* feat: Add deployment skill directory

* chore: Address deployment skill review feedback

* fix: Include deployment skill file metadata

* test: Add deployment skills e2e smoke test
2026-06-05 10:24:28 -04:00

134 lines
4.1 KiB
TypeScript

import { defineConfig, devices } from '@playwright/test';
import fs from 'fs';
import path from 'path';
import { getLocalE2EEnv, getE2EBaseURL } from './setup/env';
const rootPath = path.resolve(__dirname, '..');
const serverPath = path.resolve(rootPath, 'e2e/setup/start-server.js');
const fakeModelHookPath = path.resolve(rootPath, 'e2e/setup/fake-model.js');
const configTemplatePath = path.resolve(rootPath, 'e2e/config/librechat.e2e.yaml');
const configPath = path.resolve(rootPath, 'e2e/.generated/librechat.e2e.yaml');
const reportPath = path.resolve(rootPath, 'e2e/playwright-report');
const deploymentSkillsPath = path.resolve(rootPath, 'e2e/fixtures/deployment-skills');
const baseURL = getE2EBaseURL();
const chromiumChannel = process.env.E2E_CHROMIUM_CHANNEL || undefined;
const vanillaOverrides = {
TENANT_ISOLATION_STRICT: 'false',
OPENAI_API_KEY: 'user_provided',
OPENID_CLIENT_ID: '',
OPENID_ISSUER: '',
OPENID_AUTO_REDIRECT: 'false',
ALLOW_SOCIAL_LOGIN: 'false',
ALLOW_SOCIAL_REGISTRATION: 'false',
STREAM_KEEP_COMPLETED_JOBS: 'true',
};
const baseEnv = {
...getLocalE2EEnv(),
CONFIG_PATH: configPath,
DEPLOYMENT_SKILLS_DIR: deploymentSkillsPath,
/** Loaded in-process by `@librechat/api`'s `createRun` to swap in a fake model. */
LIBRECHAT_TEST_RUN_HOOK: fakeModelHookPath,
...vanillaOverrides,
};
const SECRET_KEY_PATTERN = /(API_KEY|SECRET|TOKEN|PASSWORD|CREDENTIALS|CLIENT_ID|_KEY)$/i;
const preservedCredentialEnvKeys = new Set([
...Object.keys(baseEnv),
'E2E_USER_PASSWORD',
'E2E_USER_B_PASSWORD',
]);
/**
* The custom endpoints in the template point at an unreachable baseURL; the fake
* model injected via `LIBRECHAT_TEST_RUN_HOOK` overrides the run before any
* request is made, so no real (or mock HTTP) provider is contacted.
*/
function writeRuntimeMockConfig() {
const template = fs.readFileSync(configTemplatePath, 'utf8');
fs.mkdirSync(path.dirname(configPath), { recursive: true });
fs.writeFileSync(configPath, template);
}
function neutralizeCredentialEnv(env: NodeJS.ProcessEnv, keep: Set<string>) {
for (const key of Object.keys(env)) {
if (!keep.has(key) && SECRET_KEY_PATTERN.test(key)) {
env[key] = '';
}
}
}
/** Blank any credential-like vars from a local `.env` so they never reach the test server. */
function neutralizeDotenvSecrets(envFile: string, keep: Set<string>) {
if (!fs.existsSync(envFile)) {
return;
}
const lines = fs.readFileSync(envFile, 'utf8').split('\n');
for (const line of lines) {
const match = line.match(/^\s*([A-Za-z_][A-Za-z0-9_]*)\s*=/);
if (!match) {
continue;
}
const key = match[1];
if (keep.has(key)) {
continue;
}
if (SECRET_KEY_PATTERN.test(key)) {
process.env[key] = '';
}
}
}
writeRuntimeMockConfig();
neutralizeCredentialEnv(process.env, preservedCredentialEnvKeys);
Object.assign(process.env, baseEnv);
neutralizeDotenvSecrets(path.resolve(rootPath, '.env'), preservedCredentialEnvKeys);
export default defineConfig({
globalSetup: require.resolve('./setup/global-setup'),
globalTeardown: require.resolve('./setup/global-teardown.mock'),
testDir: 'specs/mock/',
outputDir: 'specs/.test-results',
fullyParallel: false,
forbidOnly: !!process.env.CI,
retries: process.env.CI ? 2 : 0,
workers: 1,
reporter: process.env.CI
? [['html', { outputFolder: reportPath, open: 'never' }], ['line']]
: [['html', { outputFolder: reportPath }], ['list']],
use: {
baseURL,
video: 'on-first-retry',
trace: 'retain-on-failure',
ignoreHTTPSErrors: true,
headless: true,
storageState: path.resolve(process.cwd(), 'e2e/storageState.json'),
screenshot: 'only-on-failure',
},
expect: {
timeout: 10000,
},
projects: [
{
name: chromiumChannel ?? 'chromium',
use: {
...devices['Desktop Chrome'],
...(chromiumChannel ? { channel: chromiumChannel } : {}),
},
},
],
webServer: [
{
command: `node ${serverPath}`,
cwd: rootPath,
url: baseURL,
stdout: 'pipe',
ignoreHTTPSErrors: true,
timeout: 120_000,
reuseExistingServer: false,
},
],
});