Files
scadalink-design/docs/test_infra/test_infra_playwright.md
Joseph Doherty d3194e3634 feat: separate create/edit form pages, Playwright test infrastructure, /auth/token endpoint
Move all CRUD create/edit forms from inline on list pages to dedicated form pages
with back-button navigation and post-save redirect. Add Playwright Docker container
(browser server on port 3000) with 25 passing E2E tests covering login, navigation,
and site CRUD workflows. Add POST /auth/token endpoint for clean JWT retrieval.
2026-03-21 15:17:24 -04:00

3.8 KiB

Test Infrastructure: Playwright Browser Server

Overview

The Playwright browser server provides a remote headless browser (Chromium, Firefox, WebKit) that test scripts connect to over the network. It runs as a Playwright Server on port 3000, allowing UI tests for the Central UI (Blazor Server) to run from the host machine while the browser executes inside the container with access to the Docker network.

Image & Ports

  • Image: mcr.microsoft.com/playwright:v1.58.2-noble (Ubuntu 24.04 LTS)
  • Server port: 3000 (Playwright Server WebSocket endpoint)

Configuration

Setting Value Description
--host 0.0.0.0 Bind address Listen on all interfaces
--port 3000 Server port Playwright Server WebSocket port
ipc: host Docker IPC Shared IPC namespace (required for Chromium)

No additional config files are needed. The container runs npx playwright run-server on startup.

Connecting from Test Scripts

Test scripts run on the host and connect to the browser server via WebSocket. The connection URL is:

ws://localhost:3000

.NET (Microsoft.Playwright)

using var playwright = await Playwright.CreateAsync();
var browser = await playwright.Chromium.ConnectAsync("ws://localhost:3000");
var page = await browser.NewPageAsync();

// Browser runs inside Docker — use the Docker network hostname for Traefik.
await page.GotoAsync("http://scadalink-traefik");

Node.js

const { chromium } = require('playwright');
const browser = await chromium.connect('ws://localhost:3000');
const page = await browser.newPage();
await page.goto('http://scadalink-traefik');

Python

from playwright.sync_api import sync_playwright

with sync_playwright() as p:
    browser = p.chromium.connect("ws://localhost:3000")
    page = browser.new_page()
    page.goto("http://scadalink-traefik")

Central UI Access

The Playwright container is on the scadalink-net Docker network, so it can reach the Central UI cluster nodes directly:

Target URL in Test Scripts
Traefik LB http://scadalink-traefik
Central Node A http://scadalink-central-a:5000
Central Node B http://scadalink-central-b:5000

Important: The browser runs inside the Docker container, so page.goto() URLs must use Docker network hostnames (not localhost). The test script itself connects to the Playwright server via ws://localhost:3000 (host-mapped port), but all URLs navigated by the browser resolve inside the container.

Verification

  1. Check the container is running:
docker ps --filter name=scadalink-playwright
  1. Check the server is accepting connections (look for the WebSocket endpoint in logs):
docker logs scadalink-playwright 2>&1 | head -5
  1. Quick smoke test with a one-liner (requires npx and playwright on the host):
npx playwright@1.58.2 test --browser chromium --connect ws://localhost:3000
  • Central UI — end-to-end UI testing of all Blazor Server pages (login, admin, design, deployment, monitoring workflows).
  • Traefik Proxy — verify load balancer behavior, failover, and active node routing from a browser perspective.

Notes

  • The container includes Chromium, Firefox, and WebKit. Connect to the desired browser via playwright.chromium.connect(), playwright.firefox.connect(), or playwright.webkit.connect().
  • The ipc: host flag is required for Chromium to avoid out-of-memory crashes in the container.
  • The Playwright Server version (1.58.2) must match the @playwright package version used by test scripts on the host.
  • The container is stateless — no test data or browser state persists between restarts.
  • To stop only the Playwright container: cd infra && docker compose stop playwright.