mirror of
https://github.com/danny-avila/LibreChat.git
synced 2026-06-16 07:51:32 +03:00
* feat: Add deployment skill directory * chore: Address deployment skill review feedback * fix: Include deployment skill file metadata * test: Add deployment skills e2e smoke test
134 lines
4.1 KiB
TypeScript
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,
|
|
},
|
|
],
|
|
});
|