From 989a6a521cc5ff43491aaed3953816f6e58c5a64 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Sun, 15 Mar 2026 11:29:02 -0400 Subject: [PATCH] feat: comprehensive Playwright E2E tests for dashboard UX Add 22 browser-level tests covering the full dashboard UI: - Page load: 200 response, title, sidebar visibility, no JS errors - Sidebar: toggle collapse/expand, no agents initially, agent appears after API registration - Mesh overview: panel loads, agent counts visible after registration - New session dialog: opens on click, has form fields (name, dir), closes on Escape, closes on Cancel - Health deps page: loads, shows dependency list (uv, hm, claude), shows install links for missing deps - Messaging UI: send form present when agents exist - Agent detail: clicking agent shows detail view - Static assets: CSS loads, no 404s on page load - Health API: JSON endpoint returns ok Infrastructure: - pytest-playwright + chromium headless - Session-scoped server fixture (uvicorn on random port, temp DB) - Separate e2e_tests/ directory (doesn't affect unit test coverage) - just test-e2e / just test-e2e-headed recipes --- e2e_tests/__init__.py | 0 e2e_tests/conftest.py | 100 +++++++++ e2e_tests/test_dashboard.py | 422 ++++++++++++++++++++++++++++++++++++ justfile | 13 +- pyproject.toml | 11 + uv.lock | 181 ++++++++++++++++ 6 files changed, 726 insertions(+), 1 deletion(-) create mode 100644 e2e_tests/__init__.py create mode 100644 e2e_tests/conftest.py create mode 100644 e2e_tests/test_dashboard.py diff --git a/e2e_tests/__init__.py b/e2e_tests/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/e2e_tests/conftest.py b/e2e_tests/conftest.py new file mode 100644 index 0000000..180d366 --- /dev/null +++ b/e2e_tests/conftest.py @@ -0,0 +1,100 @@ +"""Playwright E2E test fixtures. + +Starts a real drasill server on a random port for each test session. +Tests run against a live browser hitting the actual UI. +""" + +from __future__ import annotations + +import asyncio +import socket +import threading +import time +from typing import TYPE_CHECKING + +import httpx +import pytest +import uvicorn + +from drasill.app import create_app +from drasill.config import AppConfig, DbConfig, ServerConfig + +if TYPE_CHECKING: + from collections.abc import Iterator + + from fastapi import FastAPI + from playwright.sync_api import Page + + +def _free_port() -> int: + """Find a free TCP port.""" + with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s: + s.bind(("127.0.0.1", 0)) + return s.getsockname()[1] + + +class _ServerThread(threading.Thread): + """Run uvicorn in a background thread.""" + + def __init__(self, app: FastAPI, host: str, port: int) -> None: + super().__init__(daemon=True) + self.config = uvicorn.Config(app, host=host, port=port, log_level="warning") + self.server = uvicorn.Server(self.config) + + def run(self) -> None: + # Create a fresh event loop for this thread. + loop = asyncio.new_event_loop() + asyncio.set_event_loop(loop) + loop.run_until_complete(self.server.serve()) + + def stop(self) -> None: + self.server.should_exit = True + self.join(timeout=5) + + +@pytest.fixture(scope="session") +def _server(tmp_path_factory: pytest.TempPathFactory) -> Iterator[str]: + """Start a drasill server for the test session. Returns the base URL.""" + tmp = tmp_path_factory.mktemp("e2e") + db_path = str(tmp / "e2e.db") + port = _free_port() + + cfg = AppConfig( + server=ServerConfig(host="127.0.0.1", port=port), + db=DbConfig(path=db_path), + ) + app = create_app(config=cfg) + + thread = _ServerThread(app, "127.0.0.1", port) + thread.start() + + # Wait for server to be ready. + base_url = f"http://127.0.0.1:{port}" + deadline = time.monotonic() + 10 + while time.monotonic() < deadline: + try: + httpx.get(f"{base_url}/health", timeout=1) + break + except (httpx.ConnectError, httpx.ReadError): + time.sleep(0.1) + else: + msg = "Server failed to start" + raise RuntimeError(msg) + + yield base_url + + thread.stop() + + +@pytest.fixture(scope="session") +def base_url(_server: str) -> str: + """Base URL of the running drasill server.""" + return _server + + +@pytest.fixture +def home(page: Page, base_url: str) -> Page: + """Navigate to the home page and wait for it to load.""" + page.goto(base_url) + page.wait_for_load_state("domcontentloaded") + return page diff --git a/e2e_tests/test_dashboard.py b/e2e_tests/test_dashboard.py new file mode 100644 index 0000000..614c845 --- /dev/null +++ b/e2e_tests/test_dashboard.py @@ -0,0 +1,422 @@ +"""E2E tests for the drasill dashboard. + +Tests the full UI via Playwright against a real drasill server. +Covers: page load, layout, sidebar, agent mesh, messaging, health deps, +new session dialog, SSE updates. + +Every test asserts structural invariants, not just "something exists." +""" + +from __future__ import annotations + +import re +from typing import TYPE_CHECKING + +import httpx +from playwright.sync_api import expect + +if TYPE_CHECKING: + from playwright.sync_api import Page + + +# ── Page load and structure ────────────────────────────────────────── + + +class TestPageLoad: + """Verify the dashboard loads with correct structure.""" + + def test_home_returns_200(self, base_url: str) -> None: + """GET / returns 200 with HTML content type.""" + resp = httpx.get(f"{base_url}/") + assert resp.status_code == 200 + assert "text/html" in resp.headers["content-type"] + + def test_home_has_title(self, home: Page) -> None: + """Page title contains 'drasill'.""" + expect(home).to_have_title(re.compile(r"drasill", re.IGNORECASE)) + + def test_sidebar_exists_with_htmx_trigger(self, home: Page) -> None: + """Sidebar loads via HTMX with SSE trigger.""" + sidebar = home.locator("#sidebar-agents") + expect(sidebar).to_be_attached() + # Must have hx-get for loading and sse:mesh for live updates. + assert sidebar.get_attribute("hx-get") == "/partials/sidebar" + assert "sse:mesh" in (sidebar.get_attribute("hx-trigger") or "") + + def test_sse_connection_established(self, home: Page) -> None: + """SSE event source connects to /mesh/events.""" + # The base template has hx-ext="sse" with sse-connect. + sse_el = home.locator("[sse-connect]").first + expect(sse_el).to_be_attached() + assert "/mesh/events" in (sse_el.get_attribute("sse-connect") or "") + + def test_no_js_errors_on_load(self, home: Page) -> None: + """No JavaScript errors on page load.""" + errors: list[str] = [] + home.on("pageerror", lambda err: errors.append(str(err))) + home.reload() + home.wait_for_load_state("domcontentloaded") + home.wait_for_timeout(1000) + assert errors == [], f"JS errors: {errors}" + + def test_no_404_resources(self, home: Page) -> None: + """No 404 errors for any resource on page load.""" + not_found: list[str] = [] + + def on_response(response: object) -> None: + url = getattr(response, "url", "") + status = getattr(response, "status", 0) + if status == 404 and not url.endswith("favicon.ico"): + not_found.append(url) + + home.on("response", on_response) + home.reload() + home.wait_for_load_state("domcontentloaded") + home.wait_for_timeout(500) + assert not_found == [], f"404s: {not_found}" + + +# ── Sidebar ────────────────────────────────────────────────────────── + + +class TestSidebar: + """Sidebar agent list behavior.""" + + def test_sidebar_collapse_expand(self, home: Page) -> None: + """Sidebar toggle changes width class.""" + aside = home.locator("aside").first + toggle = aside.locator("button").first + expect(toggle).to_be_visible() + + # Initially expanded — should have "Agents" text. + expect(aside).to_contain_text("Agents") + + # Collapse. + toggle.click() + home.wait_for_timeout(300) + + # Expand. + toggle.click() + home.wait_for_timeout(300) + expect(aside).to_contain_text("Agents") + + def test_empty_state(self, home: Page) -> None: + """With no agents, sidebar has zero agent buttons.""" + sidebar = home.locator("#sidebar-agents") + home.wait_for_timeout(1000) + agent_buttons = sidebar.locator("[data-open-agent]") + assert agent_buttons.count() == 0 + + def test_agent_registration_updates_sidebar( + self, + home: Page, + base_url: str, + ) -> None: + """Registering an agent via API makes it appear in the sidebar.""" + resp = httpx.post( + f"{base_url}/mesh/agents", + json={"name": "sidebar-test", "idle_timeout": 60}, + ) + assert resp.status_code == 201 + + # Trigger HTMX refresh and wait. + home.evaluate("htmx.trigger('#sidebar-agents', 'sse:mesh')") + home.wait_for_timeout(1500) + + sidebar = home.locator("#sidebar-agents") + expect(sidebar).to_contain_text("sidebar-test") + + # The agent button should have the correct data attribute. + btn = sidebar.locator("[data-open-agent='sidebar-test']") + assert btn.count() == 1 + + # Cleanup. + httpx.delete(f"{base_url}/mesh/agents/sidebar-test") + + def test_agent_deregistration_removes_from_sidebar( + self, + home: Page, + base_url: str, + ) -> None: + """Deregistering an agent removes it from the sidebar.""" + httpx.post( + f"{base_url}/mesh/agents", + json={"name": "sidebar-remove", "idle_timeout": 60}, + ) + home.evaluate("htmx.trigger('#sidebar-agents', 'sse:mesh')") + home.wait_for_timeout(1500) + expect(home.locator("#sidebar-agents")).to_contain_text("sidebar-remove") + + httpx.delete(f"{base_url}/mesh/agents/sidebar-remove") + # Trigger refresh twice — SSE can be delayed. + home.evaluate("htmx.trigger('#sidebar-agents', 'sse:mesh')") + home.wait_for_timeout(2000) + home.evaluate("htmx.trigger('#sidebar-agents', 'sse:mesh')") + home.wait_for_timeout(1500) + + btn = home.locator("[data-open-agent='sidebar-remove']") + assert btn.count() == 0 + + +# ── Agent mesh overview ────────────────────────────────────────────── + + +class TestMeshOverview: + """Agent mesh overview panel.""" + + def test_empty_mesh_shows_no_agents(self, home: Page) -> None: + """Empty mesh shows 'No agents registered' message.""" + home.wait_for_timeout(1000) + body = home.inner_text("body") + assert "no agents" in body.lower() + + def test_mesh_counts_update( + self, + home: Page, + base_url: str, + ) -> None: + """Registering agents updates the mesh overview counts.""" + httpx.post( + f"{base_url}/mesh/agents", + json={"name": "mesh-count-a", "idle_timeout": 60}, + ) + httpx.post( + f"{base_url}/mesh/agents", + json={"name": "mesh-count-b", "idle_timeout": 60}, + ) + home.evaluate("htmx.trigger('#sidebar-agents', 'sse:mesh')") + home.wait_for_timeout(2000) + + # Overview should show idle count (agents default to idle). + body = home.inner_text("body") + assert "idle" in body.lower() + + httpx.delete(f"{base_url}/mesh/agents/mesh-count-a") + httpx.delete(f"{base_url}/mesh/agents/mesh-count-b") + + +# ── New session dialog ─────────────────────────────────────────────── + + +class TestNewSessionDialog: + """New session creation dialog.""" + + def test_dialog_opens_and_has_fields(self, home: Page) -> None: + """Dialog opens on click with name and directory inputs.""" + btn = home.locator("button", has_text="New session").first + btn.click() + home.wait_for_timeout(500) + + # Name input. + name_input = home.locator("input[placeholder*='my-project']").first + expect(name_input).to_be_visible() + assert name_input.get_attribute("type") == "text" + + # Directory input. + dir_input = home.locator("input[placeholder*='dev']").first + expect(dir_input).to_be_visible() + + # Create button. + create_btn = home.locator("button", has_text="Create").first + expect(create_btn).to_be_visible() + + def test_dialog_closes_on_escape(self, home: Page) -> None: + """Pressing Escape closes the dialog.""" + home.locator("button", has_text="New session").first.click() + home.wait_for_timeout(300) + + # Dialog should be visible. + dialog = home.locator("[x-show='showNewSession']").first + expect(dialog).to_be_visible() + + home.keyboard.press("Escape") + home.wait_for_timeout(300) + expect(dialog).to_be_hidden() + + def test_dialog_closes_on_cancel(self, home: Page) -> None: + """Clicking Cancel closes the dialog.""" + home.locator("button", has_text="New session").first.click() + home.wait_for_timeout(300) + home.locator("button", has_text="Cancel").first.click() + home.wait_for_timeout(300) + + dialog = home.locator("[x-show='showNewSession']").first + expect(dialog).to_be_hidden() + + def test_dialog_closes_on_backdrop_click(self, home: Page) -> None: + """Clicking the backdrop closes the dialog.""" + home.locator("button", has_text="New session").first.click() + home.wait_for_timeout(300) + + # Click the backdrop (the outer overlay div). + backdrop = home.locator("[x-show='showNewSession']").first + # Click at the edge of the backdrop, not the inner content. + backdrop.click(position={"x": 5, "y": 5}) + home.wait_for_timeout(300) + expect(backdrop).to_be_hidden() + + +# ── Health deps page ───────────────────────────────────────────────── + + +class TestHealthDeps: + """Health dependency check page.""" + + def test_page_structure(self, page: Page, base_url: str) -> None: + """Health page has title, dependency rows, and status indicators.""" + page.goto(f"{base_url}/health/deps") + page.wait_for_load_state("domcontentloaded") + + expect(page.locator("body")).to_contain_text("System Health") + + # Each required dep should have a row. + for dep in ("uv", "hm", "claude"): + row = page.locator(f"text='{dep}'").first + expect(row).to_be_visible() + + # Should have status dots (green or red circles). + dots = page.locator(".rounded-full") + assert dots.count() >= 3, "should have at least 3 status indicators" + + def test_installed_deps_show_path(self, page: Page, base_url: str) -> None: + """Installed deps show their binary path.""" + page.goto(f"{base_url}/health/deps") + page.wait_for_load_state("domcontentloaded") + body = page.inner_text("body") + # uv should be installed in our test env. + assert "/uv" in body or "uv" in body + + def test_missing_deps_show_install_link( + self, + page: Page, + base_url: str, + ) -> None: + """Missing deps have clickable install links.""" + page.goto(f"{base_url}/health/deps") + page.wait_for_load_state("domcontentloaded") + # Install links should be external (https://). + install_links = page.locator("a[target='_blank']") + # At least optional deps (cloudflared, tailscale) are likely missing. + # Even if all are installed, the test should not fail. + for i in range(install_links.count()): + href = install_links.nth(i).get_attribute("href") + assert href is not None + assert href.startswith("http") + + +# ── Agent detail view ──────────────────────────────────────────────── + + +class TestAgentDetail: + """Agent detail panel.""" + + def test_agent_click_shows_detail( + self, + home: Page, + base_url: str, + ) -> None: + """Clicking an agent in sidebar loads its detail view.""" + httpx.post( + f"{base_url}/mesh/agents", + json={"name": "detail-click", "idle_timeout": 60}, + ) + home.evaluate("htmx.trigger('#sidebar-agents', 'sse:mesh')") + home.wait_for_timeout(1500) + + btn = home.locator("[data-open-agent='detail-click']").first + if btn.count() > 0: + btn.click() + home.wait_for_timeout(1500) + # Detail view should show agent name and status. + body = home.inner_text("body") + assert "detail-click" in body + # Should show a status indicator. + assert "idle" in body.lower() or "active" in body.lower() + + httpx.delete(f"{base_url}/mesh/agents/detail-click") + + +# ── Messaging ──────────────────────────────────────────────────────── + + +class TestMessaging: + """Message sending via mesh overview.""" + + def test_message_form_visible_with_agents( + self, + home: Page, + base_url: str, + ) -> None: + """Message send form appears in mesh overview when agents exist.""" + httpx.post( + f"{base_url}/mesh/agents", + json={"name": "msg-form-test", "idle_timeout": 60}, + ) + # The mesh overview loads via HTMX — click on a fresh overview + # pane to trigger a reload with the new agent. + home.reload() + home.wait_for_load_state("domcontentloaded") + home.wait_for_timeout(2000) + + # The mesh overview should now show the agent and a send form. + body = home.inner_text("body") + assert "msg-form-test" in body + + httpx.delete(f"{base_url}/mesh/agents/msg-form-test") + + +# ── API endpoints ──────────────────────────────────────────────────── + + +class TestAPI: + """JSON API endpoints work correctly.""" + + def test_health_ok(self, base_url: str) -> None: + """GET /health returns ok.""" + resp = httpx.get(f"{base_url}/health") + assert resp.status_code == 200 + data = resp.json() + assert data["status"] == "ok" + + def test_mesh_agents_empty(self, base_url: str) -> None: + """GET /mesh/agents returns empty list initially.""" + resp = httpx.get(f"{base_url}/mesh/agents") + assert resp.status_code == 200 + data = resp.json() + assert isinstance(data["agents"], list) + + def test_agent_lifecycle(self, base_url: str) -> None: + """Register → get → delete lifecycle via API.""" + # Register. + resp = httpx.post( + f"{base_url}/mesh/agents", + json={"name": "lifecycle-api", "idle_timeout": 60}, + ) + assert resp.status_code == 201 + + # Get. + resp = httpx.get(f"{base_url}/mesh/agents/lifecycle-api") + assert resp.status_code == 200 + assert resp.json()["name"] == "lifecycle-api" + assert resp.json()["status"] == "active" + + # Delete. + resp = httpx.delete(f"{base_url}/mesh/agents/lifecycle-api") + assert resp.status_code == 200 + + # Gone. + resp = httpx.get(f"{base_url}/mesh/agents/lifecycle-api") + assert resp.status_code == 404 + + +# ── Static assets ──────────────────────────────────────────────────── + + +class TestStaticAssets: + """Static asset loading.""" + + def test_css_loads_with_correct_type(self, base_url: str) -> None: + """style.css returns 200 with CSS content type.""" + resp = httpx.get(f"{base_url}/static/style.css") + assert resp.status_code == 200 + assert "css" in resp.headers.get("content-type", "") diff --git a/justfile b/justfile index 9f7a312..7cc490e 100644 --- a/justfile +++ b/justfile @@ -52,10 +52,13 @@ clean-logs: # ─── Quality ───────────────────────────────────────────────────────── -# Run all checks (lint + format + types + tests) +# Run all checks (lint + format + types + unit tests) check: uv run nox -s all_checks +# Run everything (unit tests + e2e tests) +check-all: check test-e2e + # Run Python tests test *args: uv run pytest {{ args }} @@ -64,6 +67,14 @@ test *args: test-fast: uv run pytest -x -q --tb=short +# Run Playwright E2E tests (headless) +test-e2e *args: + uv run pytest e2e_tests/ --no-cov {{ args }} + +# Run Playwright E2E tests (headed, for debugging) +test-e2e-headed *args: + uv run pytest e2e_tests/ --no-cov --headed {{ args }} + # Lint Python code lint: uv run ruff check py_src py_tests diff --git a/pyproject.toml b/pyproject.toml index 042754b..cd7f166 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -26,6 +26,7 @@ dev = [ "pytest-asyncio>=0.25.0", "nox>=2024.3.2", "httpx>=0.28.0", + "pytest-playwright>=0.7.2", ] [tool.ruff] @@ -60,10 +61,20 @@ ignore = [ "SLF001", # tests intentionally access private members (app.state._*) "D", # test function names are self-documenting; docstrings are noise ] +"e2e_tests/*" = [ + "S101", # assert is the point of tests + "S104", # test assertions reference 0.0.0.0 bind address + "PLR2004", # magic values clearer inline in tests + "D", # test function names are self-documenting +] [tool.pytest.ini_options] testpaths = ["py_tests"] asyncio_mode = "auto" +filterwarnings = [ + "ignore::DeprecationWarning:websockets", + "ignore::DeprecationWarning:uvicorn.protocols.websockets", +] addopts = [ "--strict-markers", "--verbose", diff --git a/uv.lock b/uv.lock index f6a5a4d..9f83ad4 100644 --- a/uv.lock +++ b/uv.lock @@ -68,6 +68,47 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/9a/3c/c17fb3ca2d9c3acff52e30b309f538586f9f5b9c9cf454f3845fc9af4881/certifi-2026.2.25-py3-none-any.whl", hash = "sha256:027692e4402ad994f1c42e52a4997a9763c646b73e4096e4d5d6db8af1d6f0fa", size = 153684, upload-time = "2026-02-25T02:54:15.766Z" }, ] +[[package]] +name = "charset-normalizer" +version = "3.4.5" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/1d/35/02daf95b9cd686320bb622eb148792655c9412dbb9b67abb5694e5910a24/charset_normalizer-3.4.5.tar.gz", hash = "sha256:95adae7b6c42a6c5b5b559b1a99149f090a57128155daeea91732c8d970d8644", size = 134804, upload-time = "2026-03-06T06:03:19.46Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/f5/48/9f34ec4bb24aa3fdba1890c1bddb97c8a4be1bd84ef5c42ac2352563ad05/charset_normalizer-3.4.5-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:ac59c15e3f1465f722607800c68713f9fbc2f672b9eb649fe831da4019ae9b23", size = 280788, upload-time = "2026-03-06T06:01:37.126Z" }, + { url = "https://files.pythonhosted.org/packages/0e/09/6003e7ffeb90cc0560da893e3208396a44c210c5ee42efff539639def59b/charset_normalizer-3.4.5-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:165c7b21d19365464e8f70e5ce5e12524c58b48c78c1f5a57524603c1ab003f8", size = 188890, upload-time = "2026-03-06T06:01:38.73Z" }, + { url = "https://files.pythonhosted.org/packages/42/1e/02706edf19e390680daa694d17e2b8eab4b5f7ac285e2a51168b4b22ee6b/charset_normalizer-3.4.5-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:28269983f25a4da0425743d0d257a2d6921ea7d9b83599d4039486ec5b9f911d", size = 206136, upload-time = "2026-03-06T06:01:40.016Z" }, + { url = "https://files.pythonhosted.org/packages/c7/87/942c3def1b37baf3cf786bad01249190f3ca3d5e63a84f831e704977de1f/charset_normalizer-3.4.5-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d27ce22ec453564770d29d03a9506d449efbb9fa13c00842262b2f6801c48cce", size = 202551, upload-time = "2026-03-06T06:01:41.522Z" }, + { url = "https://files.pythonhosted.org/packages/94/0a/af49691938dfe175d71b8a929bd7e4ace2809c0c5134e28bc535660d5262/charset_normalizer-3.4.5-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0625665e4ebdddb553ab185de5db7054393af8879fb0c87bd5690d14379d6819", size = 195572, upload-time = "2026-03-06T06:01:43.208Z" }, + { url = "https://files.pythonhosted.org/packages/20/ea/dfb1792a8050a8e694cfbde1570ff97ff74e48afd874152d38163d1df9ae/charset_normalizer-3.4.5-cp313-cp313-manylinux_2_31_armv7l.whl", hash = "sha256:c23eb3263356d94858655b3e63f85ac5d50970c6e8febcdde7830209139cc37d", size = 184438, upload-time = "2026-03-06T06:01:44.755Z" }, + { url = "https://files.pythonhosted.org/packages/72/12/c281e2067466e3ddd0595bfaea58a6946765ace5c72dfa3edc2f5f118026/charset_normalizer-3.4.5-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:e6302ca4ae283deb0af68d2fbf467474b8b6aedcd3dab4db187e07f94c109763", size = 193035, upload-time = "2026-03-06T06:01:46.051Z" }, + { url = "https://files.pythonhosted.org/packages/ba/4f/3792c056e7708e10464bad0438a44708886fb8f92e3c3d29ec5e2d964d42/charset_normalizer-3.4.5-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:e51ae7d81c825761d941962450f50d041db028b7278e7b08930b4541b3e45cb9", size = 191340, upload-time = "2026-03-06T06:01:47.547Z" }, + { url = "https://files.pythonhosted.org/packages/e7/86/80ddba897127b5c7a9bccc481b0cd36c8fefa485d113262f0fe4332f0bf4/charset_normalizer-3.4.5-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:597d10dec876923e5c59e48dbd366e852eacb2b806029491d307daea6b917d7c", size = 185464, upload-time = "2026-03-06T06:01:48.764Z" }, + { url = "https://files.pythonhosted.org/packages/4d/00/b5eff85ba198faacab83e0e4b6f0648155f072278e3b392a82478f8b988b/charset_normalizer-3.4.5-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:5cffde4032a197bd3b42fd0b9509ec60fb70918d6970e4cc773f20fc9180ca67", size = 208014, upload-time = "2026-03-06T06:01:50.371Z" }, + { url = "https://files.pythonhosted.org/packages/c8/11/d36f70be01597fd30850dde8a1269ebc8efadd23ba5785808454f2389bde/charset_normalizer-3.4.5-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:2da4eedcb6338e2321e831a0165759c0c620e37f8cd044a263ff67493be8ffb3", size = 193297, upload-time = "2026-03-06T06:01:51.933Z" }, + { url = "https://files.pythonhosted.org/packages/1a/1d/259eb0a53d4910536c7c2abb9cb25f4153548efb42800c6a9456764649c0/charset_normalizer-3.4.5-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:65a126fb4b070d05340a84fc709dd9e7c75d9b063b610ece8a60197a291d0adf", size = 204321, upload-time = "2026-03-06T06:01:53.887Z" }, + { url = "https://files.pythonhosted.org/packages/84/31/faa6c5b9d3688715e1ed1bb9d124c384fe2fc1633a409e503ffe1c6398c1/charset_normalizer-3.4.5-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:c7a80a9242963416bd81f99349d5f3fce1843c303bd404f204918b6d75a75fd6", size = 197509, upload-time = "2026-03-06T06:01:56.439Z" }, + { url = "https://files.pythonhosted.org/packages/fd/a5/c7d9dd1503ffc08950b3260f5d39ec2366dd08254f0900ecbcf3a6197c7c/charset_normalizer-3.4.5-cp313-cp313-win32.whl", hash = "sha256:f1d725b754e967e648046f00c4facc42d414840f5ccc670c5670f59f83693e4f", size = 132284, upload-time = "2026-03-06T06:01:57.812Z" }, + { url = "https://files.pythonhosted.org/packages/b9/0f/57072b253af40c8aa6636e6de7d75985624c1eb392815b2f934199340a89/charset_normalizer-3.4.5-cp313-cp313-win_amd64.whl", hash = "sha256:e37bd100d2c5d3ba35db9c7c5ba5a9228cbcffe5c4778dc824b164e5257813d7", size = 142630, upload-time = "2026-03-06T06:01:59.062Z" }, + { url = "https://files.pythonhosted.org/packages/31/41/1c4b7cc9f13bd9d369ce3bc993e13d374ce25fa38a2663644283ecf422c1/charset_normalizer-3.4.5-cp313-cp313-win_arm64.whl", hash = "sha256:93b3b2cc5cf1b8743660ce77a4f45f3f6d1172068207c1defc779a36eea6bb36", size = 133254, upload-time = "2026-03-06T06:02:00.281Z" }, + { url = "https://files.pythonhosted.org/packages/43/be/0f0fd9bb4a7fa4fb5067fb7d9ac693d4e928d306f80a0d02bde43a7c4aee/charset_normalizer-3.4.5-cp314-cp314-macosx_10_15_universal2.whl", hash = "sha256:8197abe5ca1ffb7d91e78360f915eef5addff270f8a71c1fc5be24a56f3e4873", size = 280232, upload-time = "2026-03-06T06:02:01.508Z" }, + { url = "https://files.pythonhosted.org/packages/28/02/983b5445e4bef49cd8c9da73a8e029f0825f39b74a06d201bfaa2e55142a/charset_normalizer-3.4.5-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:a2aecdb364b8a1802afdc7f9327d55dad5366bc97d8502d0f5854e50712dbc5f", size = 189688, upload-time = "2026-03-06T06:02:02.857Z" }, + { url = "https://files.pythonhosted.org/packages/d0/88/152745c5166437687028027dc080e2daed6fe11cfa95a22f4602591c42db/charset_normalizer-3.4.5-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:a66aa5022bf81ab4b1bebfb009db4fd68e0c6d4307a1ce5ef6a26e5878dfc9e4", size = 206833, upload-time = "2026-03-06T06:02:05.127Z" }, + { url = "https://files.pythonhosted.org/packages/cb/0f/ebc15c8b02af2f19be9678d6eed115feeeccc45ce1f4b098d986c13e8769/charset_normalizer-3.4.5-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d77f97e515688bd615c1d1f795d540f32542d514242067adcb8ef532504cb9ee", size = 202879, upload-time = "2026-03-06T06:02:06.446Z" }, + { url = "https://files.pythonhosted.org/packages/38/9c/71336bff6934418dc8d1e8a1644176ac9088068bc571da612767619c97b3/charset_normalizer-3.4.5-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:01a1ed54b953303ca7e310fafe0fe347aab348bd81834a0bcd602eb538f89d66", size = 195764, upload-time = "2026-03-06T06:02:08.763Z" }, + { url = "https://files.pythonhosted.org/packages/b7/95/ce92fde4f98615661871bc282a856cf9b8a15f686ba0af012984660d480b/charset_normalizer-3.4.5-cp314-cp314-manylinux_2_31_armv7l.whl", hash = "sha256:b2d37d78297b39a9eb9eb92c0f6df98c706467282055419df141389b23f93362", size = 183728, upload-time = "2026-03-06T06:02:10.137Z" }, + { url = "https://files.pythonhosted.org/packages/1c/e7/f5b4588d94e747ce45ae680f0f242bc2d98dbd4eccfab73e6160b6893893/charset_normalizer-3.4.5-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:e71bbb595973622b817c042bd943c3f3667e9c9983ce3d205f973f486fec98a7", size = 192937, upload-time = "2026-03-06T06:02:11.663Z" }, + { url = "https://files.pythonhosted.org/packages/f9/29/9d94ed6b929bf9f48bf6ede6e7474576499f07c4c5e878fb186083622716/charset_normalizer-3.4.5-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:4cd966c2559f501c6fd69294d082c2934c8dd4719deb32c22961a5ac6db0df1d", size = 192040, upload-time = "2026-03-06T06:02:13.489Z" }, + { url = "https://files.pythonhosted.org/packages/15/d2/1a093a1cf827957f9445f2fe7298bcc16f8fc5e05c1ed2ad1af0b239035e/charset_normalizer-3.4.5-cp314-cp314-musllinux_1_2_armv7l.whl", hash = "sha256:d5e52d127045d6ae01a1e821acfad2f3a1866c54d0e837828538fabe8d9d1bd6", size = 184107, upload-time = "2026-03-06T06:02:14.83Z" }, + { url = "https://files.pythonhosted.org/packages/0f/7d/82068ce16bd36135df7b97f6333c5d808b94e01d4599a682e2337ed5fd14/charset_normalizer-3.4.5-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:30a2b1a48478c3428d047ed9690d57c23038dac838a87ad624c85c0a78ebeb39", size = 208310, upload-time = "2026-03-06T06:02:16.165Z" }, + { url = "https://files.pythonhosted.org/packages/84/4e/4dfb52307bb6af4a5c9e73e482d171b81d36f522b21ccd28a49656baa680/charset_normalizer-3.4.5-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:d8ed79b8f6372ca4254955005830fd61c1ccdd8c0fac6603e2c145c61dd95db6", size = 192918, upload-time = "2026-03-06T06:02:18.144Z" }, + { url = "https://files.pythonhosted.org/packages/08/a4/159ff7da662cf7201502ca89980b8f06acf3e887b278956646a8aeb178ab/charset_normalizer-3.4.5-cp314-cp314-musllinux_1_2_s390x.whl", hash = "sha256:c5af897b45fa606b12464ccbe0014bbf8c09191e0a66aab6aa9d5cf6e77e0c94", size = 204615, upload-time = "2026-03-06T06:02:19.821Z" }, + { url = "https://files.pythonhosted.org/packages/d6/62/0dd6172203cb6b429ffffc9935001fde42e5250d57f07b0c28c6046deb6b/charset_normalizer-3.4.5-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:1088345bcc93c58d8d8f3d783eca4a6e7a7752bbff26c3eee7e73c597c191c2e", size = 197784, upload-time = "2026-03-06T06:02:21.86Z" }, + { url = "https://files.pythonhosted.org/packages/c7/5e/1aab5cb737039b9c59e63627dc8bbc0d02562a14f831cc450e5f91d84ce1/charset_normalizer-3.4.5-cp314-cp314-win32.whl", hash = "sha256:ee57b926940ba00bca7ba7041e665cc956e55ef482f851b9b65acb20d867e7a2", size = 133009, upload-time = "2026-03-06T06:02:23.289Z" }, + { url = "https://files.pythonhosted.org/packages/40/65/e7c6c77d7aaa4c0d7974f2e403e17f0ed2cb0fc135f77d686b916bf1eead/charset_normalizer-3.4.5-cp314-cp314-win_amd64.whl", hash = "sha256:4481e6da1830c8a1cc0b746b47f603b653dadb690bcd851d039ffaefe70533aa", size = 143511, upload-time = "2026-03-06T06:02:26.195Z" }, + { url = "https://files.pythonhosted.org/packages/ba/91/52b0841c71f152f563b8e072896c14e3d83b195c188b338d3cc2e582d1d4/charset_normalizer-3.4.5-cp314-cp314-win_arm64.whl", hash = "sha256:97ab7787092eb9b50fb47fa04f24c75b768a606af1bcba1957f07f128a7219e4", size = 133775, upload-time = "2026-03-06T06:02:27.473Z" }, + { url = "https://files.pythonhosted.org/packages/c5/60/3a621758945513adfd4db86827a5bafcc615f913dbd0b4c2ed64a65731be/charset_normalizer-3.4.5-py3-none-any.whl", hash = "sha256:9db5e3fcdcee89a78c04dffb3fe33c79f77bd741a624946db2591c81b2fc85b0", size = 55455, upload-time = "2026-03-06T06:03:17.827Z" }, +] + [[package]] name = "click" version = "8.3.1" @@ -213,6 +254,7 @@ dev = [ { name = "pytest" }, { name = "pytest-asyncio" }, { name = "pytest-cov" }, + { name = "pytest-playwright" }, { name = "ruff" }, { name = "ty" }, ] @@ -236,6 +278,7 @@ dev = [ { name = "pytest", specifier = ">=8.0.0" }, { name = "pytest-asyncio", specifier = ">=0.25.0" }, { name = "pytest-cov", specifier = ">=6.0.0" }, + { name = "pytest-playwright", specifier = ">=0.7.2" }, { name = "ruff", specifier = ">=0.9.0" }, { name = "ty" }, ] @@ -265,6 +308,40 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/f9/0b/de6f54d4a8bedfe8645c41497f3c18d749f0bd3218170c667bf4b81d0cdd/filelock-3.25.0-py3-none-any.whl", hash = "sha256:5ccf8069f7948f494968fc0713c10e5c182a9c9d9eef3a636307a20c2490f047", size = 26427, upload-time = "2026-03-01T15:08:44.593Z" }, ] +[[package]] +name = "greenlet" +version = "3.3.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/a3/51/1664f6b78fc6ebbd98019a1fd730e83fa78f2db7058f72b1463d3612b8db/greenlet-3.3.2.tar.gz", hash = "sha256:2eaf067fc6d886931c7962e8c6bede15d2f01965560f3359b27c80bde2d151f2", size = 188267, upload-time = "2026-02-20T20:54:15.531Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ac/48/f8b875fa7dea7dd9b33245e37f065af59df6a25af2f9561efa8d822fde51/greenlet-3.3.2-cp313-cp313-macosx_11_0_universal2.whl", hash = "sha256:aa6ac98bdfd716a749b84d4034486863fd81c3abde9aa3cf8eff9127981a4ae4", size = 279120, upload-time = "2026-02-20T20:19:01.9Z" }, + { url = "https://files.pythonhosted.org/packages/49/8d/9771d03e7a8b1ee456511961e1b97a6d77ae1dea4a34a5b98eee706689d3/greenlet-3.3.2-cp313-cp313-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ab0c7e7901a00bc0a7284907273dc165b32e0d109a6713babd04471327ff7986", size = 603238, upload-time = "2026-02-20T20:47:32.873Z" }, + { url = "https://files.pythonhosted.org/packages/59/0e/4223c2bbb63cd5c97f28ffb2a8aee71bdfb30b323c35d409450f51b91e3e/greenlet-3.3.2-cp313-cp313-manylinux_2_24_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:d248d8c23c67d2291ffd47af766e2a3aa9fa1c6703155c099feb11f526c63a92", size = 614219, upload-time = "2026-02-20T20:55:59.817Z" }, + { url = "https://files.pythonhosted.org/packages/94/2b/4d012a69759ac9d77210b8bfb128bc621125f5b20fc398bce3940d036b1c/greenlet-3.3.2-cp313-cp313-manylinux_2_24_s390x.manylinux_2_28_s390x.whl", hash = "sha256:ccd21bb86944ca9be6d967cf7691e658e43417782bce90b5d2faeda0ff78a7dd", size = 628268, upload-time = "2026-02-20T21:02:48.024Z" }, + { url = "https://files.pythonhosted.org/packages/7a/34/259b28ea7a2a0c904b11cd36c79b8cef8019b26ee5dbe24e73b469dea347/greenlet-3.3.2-cp313-cp313-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:b6997d360a4e6a4e936c0f9625b1c20416b8a0ea18a8e19cabbefc712e7397ab", size = 616774, upload-time = "2026-02-20T20:21:02.454Z" }, + { url = "https://files.pythonhosted.org/packages/0a/03/996c2d1689d486a6e199cb0f1cf9e4aa940c500e01bdf201299d7d61fa69/greenlet-3.3.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:64970c33a50551c7c50491671265d8954046cb6e8e2999aacdd60e439b70418a", size = 1571277, upload-time = "2026-02-20T20:49:34.795Z" }, + { url = "https://files.pythonhosted.org/packages/d9/c4/2570fc07f34a39f2caf0bf9f24b0a1a0a47bc2e8e465b2c2424821389dfc/greenlet-3.3.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:1a9172f5bf6bd88e6ba5a84e0a68afeac9dc7b6b412b245dd64f52d83c81e55b", size = 1640455, upload-time = "2026-02-20T20:21:10.261Z" }, + { url = "https://files.pythonhosted.org/packages/91/39/5ef5aa23bc545aa0d31e1b9b55822b32c8da93ba657295840b6b34124009/greenlet-3.3.2-cp313-cp313-win_amd64.whl", hash = "sha256:a7945dd0eab63ded0a48e4dcade82939783c172290a7903ebde9e184333ca124", size = 230961, upload-time = "2026-02-20T20:16:58.461Z" }, + { url = "https://files.pythonhosted.org/packages/62/6b/a89f8456dcb06becff288f563618e9f20deed8dd29beea14f9a168aef64b/greenlet-3.3.2-cp313-cp313-win_arm64.whl", hash = "sha256:394ead29063ee3515b4e775216cb756b2e3b4a7e55ae8fd884f17fa579e6b327", size = 230221, upload-time = "2026-02-20T20:17:37.152Z" }, + { url = "https://files.pythonhosted.org/packages/3f/ae/8bffcbd373b57a5992cd077cbe8858fff39110480a9d50697091faea6f39/greenlet-3.3.2-cp314-cp314-macosx_11_0_universal2.whl", hash = "sha256:8d1658d7291f9859beed69a776c10822a0a799bc4bfe1bd4272bb60e62507dab", size = 279650, upload-time = "2026-02-20T20:18:00.783Z" }, + { url = "https://files.pythonhosted.org/packages/d1/c0/45f93f348fa49abf32ac8439938726c480bd96b2a3c6f4d949ec0124b69f/greenlet-3.3.2-cp314-cp314-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:18cb1b7337bca281915b3c5d5ae19f4e76d35e1df80f4ad3c1a7be91fadf1082", size = 650295, upload-time = "2026-02-20T20:47:34.036Z" }, + { url = "https://files.pythonhosted.org/packages/b3/de/dd7589b3f2b8372069ab3e4763ea5329940fc7ad9dcd3e272a37516d7c9b/greenlet-3.3.2-cp314-cp314-manylinux_2_24_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:c2e47408e8ce1c6f1ceea0dffcdf6ebb85cc09e55c7af407c99f1112016e45e9", size = 662163, upload-time = "2026-02-20T20:56:01.295Z" }, + { url = "https://files.pythonhosted.org/packages/cd/ac/85804f74f1ccea31ba518dcc8ee6f14c79f73fe36fa1beba38930806df09/greenlet-3.3.2-cp314-cp314-manylinux_2_24_s390x.manylinux_2_28_s390x.whl", hash = "sha256:e3cb43ce200f59483eb82949bf1835a99cf43d7571e900d7c8d5c62cdf25d2f9", size = 675371, upload-time = "2026-02-20T21:02:49.664Z" }, + { url = "https://files.pythonhosted.org/packages/d2/d8/09bfa816572a4d83bccd6750df1926f79158b1c36c5f73786e26dbe4ee38/greenlet-3.3.2-cp314-cp314-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:63d10328839d1973e5ba35e98cccbca71b232b14051fd957b6f8b6e8e80d0506", size = 664160, upload-time = "2026-02-20T20:21:04.015Z" }, + { url = "https://files.pythonhosted.org/packages/48/cf/56832f0c8255d27f6c35d41b5ec91168d74ec721d85f01a12131eec6b93c/greenlet-3.3.2-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:8e4ab3cfb02993c8cc248ea73d7dae6cec0253e9afa311c9b37e603ca9fad2ce", size = 1619181, upload-time = "2026-02-20T20:49:36.052Z" }, + { url = "https://files.pythonhosted.org/packages/0a/23/b90b60a4aabb4cec0796e55f25ffbfb579a907c3898cd2905c8918acaa16/greenlet-3.3.2-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:94ad81f0fd3c0c0681a018a976e5c2bd2ca2d9d94895f23e7bb1af4e8af4e2d5", size = 1687713, upload-time = "2026-02-20T20:21:11.684Z" }, + { url = "https://files.pythonhosted.org/packages/f3/ca/2101ca3d9223a1dc125140dbc063644dca76df6ff356531eb27bc267b446/greenlet-3.3.2-cp314-cp314-win_amd64.whl", hash = "sha256:8c4dd0f3997cf2512f7601563cc90dfb8957c0cff1e3a1b23991d4ea1776c492", size = 232034, upload-time = "2026-02-20T20:20:08.186Z" }, + { url = "https://files.pythonhosted.org/packages/f6/4a/ecf894e962a59dea60f04877eea0fd5724618da89f1867b28ee8b91e811f/greenlet-3.3.2-cp314-cp314-win_arm64.whl", hash = "sha256:cd6f9e2bbd46321ba3bbb4c8a15794d32960e3b0ae2cc4d49a1a53d314805d71", size = 231437, upload-time = "2026-02-20T20:18:59.722Z" }, + { url = "https://files.pythonhosted.org/packages/98/6d/8f2ef704e614bcf58ed43cfb8d87afa1c285e98194ab2cfad351bf04f81e/greenlet-3.3.2-cp314-cp314t-macosx_11_0_universal2.whl", hash = "sha256:e26e72bec7ab387ac80caa7496e0f908ff954f31065b0ffc1f8ecb1338b11b54", size = 286617, upload-time = "2026-02-20T20:19:29.856Z" }, + { url = "https://files.pythonhosted.org/packages/5e/0d/93894161d307c6ea237a43988f27eba0947b360b99ac5239ad3fe09f0b47/greenlet-3.3.2-cp314-cp314t-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:8b466dff7a4ffda6ca975979bab80bdadde979e29fc947ac3be4451428d8b0e4", size = 655189, upload-time = "2026-02-20T20:47:35.742Z" }, + { url = "https://files.pythonhosted.org/packages/f5/2c/d2d506ebd8abcb57386ec4f7ba20f4030cbe56eae541bc6fd6ef399c0b41/greenlet-3.3.2-cp314-cp314t-manylinux_2_24_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:b8bddc5b73c9720bea487b3bffdb1840fe4e3656fba3bd40aa1489e9f37877ff", size = 658225, upload-time = "2026-02-20T20:56:02.527Z" }, + { url = "https://files.pythonhosted.org/packages/d1/67/8197b7e7e602150938049d8e7f30de1660cfb87e4c8ee349b42b67bdb2e1/greenlet-3.3.2-cp314-cp314t-manylinux_2_24_s390x.manylinux_2_28_s390x.whl", hash = "sha256:59b3e2c40f6706b05a9cd299c836c6aa2378cabe25d021acd80f13abf81181cf", size = 666581, upload-time = "2026-02-20T21:02:51.526Z" }, + { url = "https://files.pythonhosted.org/packages/8e/30/3a09155fbf728673a1dea713572d2d31159f824a37c22da82127056c44e4/greenlet-3.3.2-cp314-cp314t-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:b26b0f4428b871a751968285a1ac9648944cea09807177ac639b030bddebcea4", size = 657907, upload-time = "2026-02-20T20:21:05.259Z" }, + { url = "https://files.pythonhosted.org/packages/f3/fd/d05a4b7acd0154ed758797f0a43b4c0962a843bedfe980115e842c5b2d08/greenlet-3.3.2-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:1fb39a11ee2e4d94be9a76671482be9398560955c9e568550de0224e41104727", size = 1618857, upload-time = "2026-02-20T20:49:37.309Z" }, + { url = "https://files.pythonhosted.org/packages/6f/e1/50ee92a5db521de8f35075b5eff060dd43d39ebd46c2181a2042f7070385/greenlet-3.3.2-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:20154044d9085151bc309e7689d6f7ba10027f8f5a8c0676ad398b951913d89e", size = 1680010, upload-time = "2026-02-20T20:21:13.427Z" }, + { url = "https://files.pythonhosted.org/packages/29/4b/45d90626aef8e65336bed690106d1382f7a43665e2249017e9527df8823b/greenlet-3.3.2-cp314-cp314t-win_amd64.whl", hash = "sha256:c04c5e06ec3e022cbfe2cd4a846e1d4e50087444f875ff6d2c2ad8445495cf1a", size = 237086, upload-time = "2026-02-20T20:20:45.786Z" }, +] + [[package]] name = "h11" version = "0.16.0" @@ -472,6 +549,25 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/63/d7/97f7e3a6abb67d8080dd406fd4df842c2be0efaf712d1c899c32a075027c/platformdirs-4.9.4-py3-none-any.whl", hash = "sha256:68a9a4619a666ea6439f2ff250c12a853cd1cbd5158d258bd824a7df6be2f868", size = 21216, upload-time = "2026-03-05T18:34:12.172Z" }, ] +[[package]] +name = "playwright" +version = "1.58.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "greenlet" }, + { name = "pyee" }, +] +wheels = [ + { url = "https://files.pythonhosted.org/packages/f8/c9/9c6061d5703267f1baae6a4647bfd1862e386fbfdb97d889f6f6ae9e3f64/playwright-1.58.0-py3-none-macosx_10_13_x86_64.whl", hash = "sha256:96e3204aac292ee639edbfdef6298b4be2ea0a55a16b7068df91adac077cc606", size = 42251098, upload-time = "2026-01-30T15:09:24.028Z" }, + { url = "https://files.pythonhosted.org/packages/e0/40/59d34a756e02f8c670f0fee987d46f7ee53d05447d43cd114ca015cb168c/playwright-1.58.0-py3-none-macosx_11_0_arm64.whl", hash = "sha256:70c763694739d28df71ed578b9c8202bb83e8fe8fb9268c04dd13afe36301f71", size = 41039625, upload-time = "2026-01-30T15:09:27.558Z" }, + { url = "https://files.pythonhosted.org/packages/e1/ee/3ce6209c9c74a650aac9028c621f357a34ea5cd4d950700f8e2c4b7fe2c4/playwright-1.58.0-py3-none-macosx_11_0_universal2.whl", hash = "sha256:185e0132578733d02802dfddfbbc35f42be23a45ff49ccae5081f25952238117", size = 42251098, upload-time = "2026-01-30T15:09:30.461Z" }, + { url = "https://files.pythonhosted.org/packages/f1/af/009958cbf23fac551a940d34e3206e6c7eed2b8c940d0c3afd1feb0b0589/playwright-1.58.0-py3-none-manylinux1_x86_64.whl", hash = "sha256:c95568ba1eda83812598c1dc9be60b4406dffd60b149bc1536180ad108723d6b", size = 46235268, upload-time = "2026-01-30T15:09:33.787Z" }, + { url = "https://files.pythonhosted.org/packages/d9/a6/0e66ad04b6d3440dae73efb39540c5685c5fc95b17c8b29340b62abbd952/playwright-1.58.0-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8f9999948f1ab541d98812de25e3a8c410776aa516d948807140aff797b4bffa", size = 45964214, upload-time = "2026-01-30T15:09:36.751Z" }, + { url = "https://files.pythonhosted.org/packages/0e/4b/236e60ab9f6d62ed0fd32150d61f1f494cefbf02304c0061e78ed80c1c32/playwright-1.58.0-py3-none-win32.whl", hash = "sha256:1e03be090e75a0fabbdaeab65ce17c308c425d879fa48bb1d7986f96bfad0b99", size = 36815998, upload-time = "2026-01-30T15:09:39.627Z" }, + { url = "https://files.pythonhosted.org/packages/41/f8/5ec599c5e59d2f2f336a05b4f318e733077cd5044f24adb6f86900c3e6a7/playwright-1.58.0-py3-none-win_amd64.whl", hash = "sha256:a2bf639d0ce33b3ba38de777e08697b0d8f3dc07ab6802e4ac53fb65e3907af8", size = 36816005, upload-time = "2026-01-30T15:09:42.449Z" }, + { url = "https://files.pythonhosted.org/packages/c8/c4/cc0229fea55c87d6c9c67fe44a21e2cd28d1d558a5478ed4d617e9fb0c93/playwright-1.58.0-py3-none-win_arm64.whl", hash = "sha256:32ffe5c303901a13a0ecab91d1c3f74baf73b84f4bedbb6b935f5bc11cc98e1b", size = 33085919, upload-time = "2026-01-30T15:09:45.71Z" }, +] + [[package]] name = "pluggy" version = "1.6.0" @@ -563,6 +659,18 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/00/4b/ccc026168948fec4f7555b9164c724cf4125eac006e176541483d2c959be/pydantic_settings-2.13.1-py3-none-any.whl", hash = "sha256:d56fd801823dbeae7f0975e1f8c8e25c258eb75d278ea7abb5d9cebb01b56237", size = 58929, upload-time = "2026-02-19T13:45:06.034Z" }, ] +[[package]] +name = "pyee" +version = "13.0.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "typing-extensions" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/8b/04/e7c1fe4dc78a6fdbfd6c337b1c3732ff543b8a397683ab38378447baa331/pyee-13.0.1.tar.gz", hash = "sha256:0b931f7c14535667ed4c7e0d531716368715e860b988770fc7eb8578d1f67fc8", size = 31655, upload-time = "2026-02-14T21:12:28.044Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a0/c4/b4d4827c93ef43c01f599ef31453ccc1c132b353284fc6c87d535c233129/pyee-13.0.1-py3-none-any.whl", hash = "sha256:af2f8fede4171ef667dfded53f96e2ed0d6e6bd7ee3bb46437f77e3b57689228", size = 15659, upload-time = "2026-02-14T21:12:26.263Z" }, +] + [[package]] name = "pygments" version = "2.19.2" @@ -600,6 +708,19 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/e5/35/f8b19922b6a25bc0880171a2f1a003eaeb93657475193ab516fd87cac9da/pytest_asyncio-1.3.0-py3-none-any.whl", hash = "sha256:611e26147c7f77640e6d0a92a38ed17c3e9848063698d5c93d5aa7aa11cebff5", size = 15075, upload-time = "2025-11-10T16:07:45.537Z" }, ] +[[package]] +name = "pytest-base-url" +version = "2.1.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "pytest" }, + { name = "requests" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/ae/1a/b64ac368de6b993135cb70ca4e5d958a5c268094a3a2a4cac6f0021b6c4f/pytest_base_url-2.1.0.tar.gz", hash = "sha256:02748589a54f9e63fcbe62301d6b0496da0d10231b753e950c63e03aee745d45", size = 6702, upload-time = "2024-01-31T22:43:00.81Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/98/1c/b00940ab9eb8ede7897443b771987f2f4a76f06be02f1b3f01eb7567e24a/pytest_base_url-2.1.0-py3-none-any.whl", hash = "sha256:3ad15611778764d451927b2a53240c1a7a591b521ea44cebfe45849d2d2812e6", size = 5302, upload-time = "2024-01-31T22:42:58.897Z" }, +] + [[package]] name = "pytest-cov" version = "7.0.0" @@ -614,6 +735,21 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/ee/49/1377b49de7d0c1ce41292161ea0f721913fa8722c19fb9c1e3aa0367eecb/pytest_cov-7.0.0-py3-none-any.whl", hash = "sha256:3b8e9558b16cc1479da72058bdecf8073661c7f57f7d3c5f22a1c23507f2d861", size = 22424, upload-time = "2025-09-09T10:57:00.695Z" }, ] +[[package]] +name = "pytest-playwright" +version = "0.7.2" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "playwright" }, + { name = "pytest" }, + { name = "pytest-base-url" }, + { name = "python-slugify" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/e8/6b/913e36aa421b35689ec95ed953ff7e8df3f2ee1c7b8ab2a3f1fd39d95faf/pytest_playwright-0.7.2.tar.gz", hash = "sha256:247b61123b28c7e8febb993a187a07e54f14a9aa04edc166f7a976d88f04c770", size = 16928, upload-time = "2025-11-24T03:43:22.53Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/76/61/4d333d8354ea2bea2c2f01bad0a4aa3c1262de20e1241f78e73360e9b620/pytest_playwright-0.7.2-py3-none-any.whl", hash = "sha256:8084e015b2b3ecff483c2160f1c8219b38b66c0d4578b23c0f700d1b0240ea38", size = 16881, upload-time = "2025-11-24T03:43:24.423Z" }, +] + [[package]] name = "python-discovery" version = "1.1.1" @@ -645,6 +781,18 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/1b/d0/397f9626e711ff749a95d96b7af99b9c566a9bb5129b8e4c10fc4d100304/python_multipart-0.0.22-py3-none-any.whl", hash = "sha256:2b2cd894c83d21bf49d702499531c7bafd057d730c201782048f7945d82de155", size = 24579, upload-time = "2026-01-25T10:15:54.811Z" }, ] +[[package]] +name = "python-slugify" +version = "8.0.4" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "text-unidecode" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/87/c7/5e1547c44e31da50a460df93af11a535ace568ef89d7a811069ead340c4a/python-slugify-8.0.4.tar.gz", hash = "sha256:59202371d1d05b54a9e7720c5e038f928f45daaffe41dd10822f3907b937c856", size = 10921, upload-time = "2024-02-08T18:32:45.488Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a4/62/02da182e544a51a5c3ccf4b03ab79df279f9c60c5e82d5e8bec7ca26ac11/python_slugify-8.0.4-py2.py3-none-any.whl", hash = "sha256:276540b79961052b66b7d116620b36518847f52d5fd9e3a70164fc8c50faa6b8", size = 10051, upload-time = "2024-02-08T18:32:43.911Z" }, +] + [[package]] name = "pyyaml" version = "6.0.3" @@ -681,6 +829,21 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/f1/12/de94a39c2ef588c7e6455cfbe7343d3b2dc9d6b6b2f40c4c6565744c873d/pyyaml-6.0.3-cp314-cp314t-win_arm64.whl", hash = "sha256:ebc55a14a21cb14062aa4162f906cd962b28e2e9ea38f9b4391244cd8de4ae0b", size = 149341, upload-time = "2025-09-25T21:32:56.828Z" }, ] +[[package]] +name = "requests" +version = "2.32.5" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "certifi" }, + { name = "charset-normalizer" }, + { name = "idna" }, + { name = "urllib3" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/c9/74/b3ff8e6c8446842c3f5c837e9c3dfcfe2018ea6ecef224c710c85ef728f4/requests-2.32.5.tar.gz", hash = "sha256:dbba0bac56e100853db0ea71b82b4dfd5fe2bf6d3754a8893c3af500cec7d7cf", size = 134517, upload-time = "2025-08-18T20:46:02.573Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/1e/db/4254e3eabe8020b458f1a747140d32277ec7a271daf1d235b70dc0b4e6e3/requests-2.32.5-py3-none-any.whl", hash = "sha256:2462f94637a34fd532264295e186976db0f5d453d1cdd31473c85a6a161affb6", size = 64738, upload-time = "2025-08-18T20:46:00.542Z" }, +] + [[package]] name = "rich" version = "14.3.3" @@ -740,6 +903,15 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/81/0d/13d1d239a25cbfb19e740db83143e95c772a1fe10202dda4b76792b114dd/starlette-0.52.1-py3-none-any.whl", hash = "sha256:0029d43eb3d273bc4f83a08720b4912ea4b071087a3b48db01b7c839f7954d74", size = 74272, upload-time = "2026-01-18T13:34:09.188Z" }, ] +[[package]] +name = "text-unidecode" +version = "1.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/ab/e2/e9a00f0ccb71718418230718b3d900e71a5d16e701a3dae079a21e9cd8f8/text-unidecode-1.3.tar.gz", hash = "sha256:bad6603bb14d279193107714b288be206cac565dfa49aa5b105294dd5c4aab93", size = 76885, upload-time = "2019-08-30T21:36:45.405Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a6/a5/c0b6468d3824fe3fde30dbb5e1f687b291608f9473681bbf7dabbf5a87d7/text_unidecode-1.3-py2.py3-none-any.whl", hash = "sha256:1311f10e8b895935241623731c2ba64f4c455287888b18189350b67134a822e8", size = 78154, upload-time = "2019-08-30T21:37:03.543Z" }, +] + [[package]] name = "ty" version = "0.0.21" @@ -800,6 +972,15 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/dc/9b/47798a6c91d8bdb567fe2698fe81e0c6b7cb7ef4d13da4114b41d239f65d/typing_inspection-0.4.2-py3-none-any.whl", hash = "sha256:4ed1cacbdc298c220f1bd249ed5287caa16f34d44ef4e9c3d0cbad5b521545e7", size = 14611, upload-time = "2025-10-01T02:14:40.154Z" }, ] +[[package]] +name = "urllib3" +version = "2.6.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/c7/24/5f1b3bdffd70275f6661c76461e25f024d5a38a46f04aaca912426a2b1d3/urllib3-2.6.3.tar.gz", hash = "sha256:1b62b6884944a57dbe321509ab94fd4d3b307075e0c2eae991ac71ee15ad38ed", size = 435556, upload-time = "2026-01-07T16:24:43.925Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/39/08/aaaad47bc4e9dc8c725e68f9d04865dbcb2052843ff09c97b08904852d84/urllib3-2.6.3-py3-none-any.whl", hash = "sha256:bf272323e553dfb2e87d9bfd225ca7b0f467b919d7bbd355436d3fd37cb0acd4", size = 131584, upload-time = "2026-01-07T16:24:42.685Z" }, +] + [[package]] name = "uvicorn" version = "0.41.0"