Files
AutoGPT/autogpt_platform/frontend/playwright.config.ts
Nicholas Tindle 1a305db162 ci(frontend): add Playwright E2E coverage reporting to Codecov (#12665)
Co-authored-by: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-04 00:55:09 -05:00

148 lines
4.5 KiB
TypeScript

import { defineConfig, devices } from "@playwright/test";
/**
* Read environment variables from file.
* https://github.com/motdotla/dotenv
*/
import dotenv from "dotenv";
import fs from "fs";
import path from "path";
dotenv.config({ path: path.resolve(__dirname, ".env") });
dotenv.config({ path: path.resolve(__dirname, "../backend/.env") });
const frontendRoot = __dirname.replaceAll("\\", "/");
// Directory where CI copies .next/static from the Docker container
const staticCoverageDir = path.resolve(__dirname, ".next-static-coverage");
function normalizeCoverageSourcePath(filePath: string) {
const normalizedFilePath = filePath.replaceAll("\\", "/");
const withoutWebpackPrefix = normalizedFilePath.replace(
/^webpack:\/\/_N_E\//,
"",
);
if (withoutWebpackPrefix.startsWith("./")) {
return withoutWebpackPrefix.slice(2);
}
if (withoutWebpackPrefix.startsWith(frontendRoot)) {
return path.posix.relative(frontendRoot, withoutWebpackPrefix);
}
return withoutWebpackPrefix;
}
// Resolve source maps from the copied .next/static directory.
// Cache parsed results to avoid repeated disk reads during report generation.
const sourceMapCache = new Map<string, object | undefined>();
function resolveSourceMap(sourcePath: string) {
// sourcePath is the sourceMappingURL, e.g.:
// "http://localhost:3000/_next/static/chunks/abc123.js.map"
const match = sourcePath.match(/_next\/static\/(.+)$/);
if (!match) return undefined;
const mapFile = path.join(staticCoverageDir, match[1]);
if (sourceMapCache.has(mapFile)) return sourceMapCache.get(mapFile);
try {
const result = JSON.parse(fs.readFileSync(mapFile, "utf8")) as object;
sourceMapCache.set(mapFile, result);
return result;
} catch {
sourceMapCache.set(mapFile, undefined);
return undefined;
}
}
export default defineConfig({
testDir: "./src/tests",
/* Global setup file that runs before all tests */
globalSetup: "./src/tests/global-setup.ts",
/* Run tests in files in parallel */
fullyParallel: true,
/* Fail the build on CI if you accidentally left test.only in the source code. */
forbidOnly: !!process.env.CI,
/* Retry on CI only */
retries: process.env.CI ? 1 : 0,
/* use more workers on CI. */
workers: process.env.CI ? 4 : undefined,
/* Reporter to use. See https://playwright.dev/docs/test-reporters */
reporter: [
["list"],
["html", { open: "never" }],
[
"monocart-reporter",
{
name: "E2E Coverage Report",
outputFile: "./coverage/e2e/report.html",
coverage: {
reports: ["cobertura"],
outputDir: "./coverage/e2e",
entryFilter: (entry: { url: string }) =>
entry.url.includes("/_next/static/") &&
!entry.url.includes("node_modules"),
sourceFilter: (sourcePath: string) =>
sourcePath.includes("src/") && !sourcePath.includes("node_modules"),
sourcePath: (filePath: string) =>
normalizeCoverageSourcePath(filePath),
sourceMapResolver: (sourcePath: string) =>
resolveSourceMap(sourcePath),
},
},
],
],
/* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */
use: {
/* Base URL to use in actions like `await page.goto('/')`. */
baseURL: "http://localhost:3000/",
/* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */
screenshot: "only-on-failure",
bypassCSP: true,
/* Helps debugging failures */
trace: "retain-on-failure",
video: "retain-on-failure",
/* Auto-accept cookies in all tests to prevent banner interference */
storageState: {
cookies: [],
origins: [
{
origin: "http://localhost:3000",
localStorage: [
{
name: "autogpt_cookie_consent",
value: JSON.stringify({
hasConsented: true,
timestamp: Date.now(),
analytics: true,
monitoring: true,
}),
},
],
},
],
},
},
/* Maximum time one test can run for */
timeout: 25000,
/* Configure web server to start automatically (local dev only) */
webServer: {
command: "pnpm start",
url: "http://localhost:3000",
reuseExistingServer: true,
},
/* Configure projects for major browsers */
projects: [
{
name: "chromium",
use: { ...devices["Desktop Chrome"], channel: "chromium" },
},
],
});