personal memory agent
0
fork

Configure Feed

Select the types of activity you want to include in your feed.

chore: remove playwright and sol screenshot CLI

Visual review of the Convey UI now happens via external pinchtab; the in-repo sol screenshot tool and its Playwright dependency are dead weight.

- Delete convey/screenshot.py and apps/chat/tests/test_markdown_bootstrap_browser.py
- Drop playwright from pyproject.toml; uv.lock loses playwright, pyee, greenlet
- Remove the playwright install chromium step from make install
- Trim screenshot from think/sol_cli.py COMMANDS and the Convey help group
- Trim doc references in AGENTS.md, README.md, docs/testing.md, docs/CONVEY.md, docs/SOLCLI.md, docs/BACKLOG.md, and scripts/doctor.py
- Trim the stale make screenshot clause from tests/integration/test_apps.py

+8 -457
+2 -3
AGENTS.md
··· 36 36 | `think/sol_cli.py` | CLI entry point — `COMMANDS` / `ALIASES` / `GROUPS` dicts | adding a top-level `sol <cmd>` | `docs/SOLCLI.md` | 37 37 | `observe/` | Multimodal capture — screen, audio, transcribe, describe, sense, transfer | capture-side bugs, new input modalities | `docs/OBSERVE.md` | 38 38 | `think/` | Post-processing core — cortex, talent, callosum, indexer, entities, facets, activities, scheduler, heartbeat, supervisor | anything downstream of capture; most coder work lives here | `docs/THINK.md`, `docs/CORTEX.md`, `docs/CALLOSUM.md` | 39 - | `convey/` | Web app framework — app discovery, routing, bridge, screenshot tooling | layout / framework-level UI changes | `docs/CONVEY.md` | 39 + | `convey/` | Web app framework — app discovery, routing, bridge | layout / framework-level UI changes | `docs/CONVEY.md` | 40 40 | `apps/` | Convey apps — each self-contained (`call.py` Typer sub-app + `routes.py` + `templates/`) | adding a user-facing feature, a `sol call <app>` verb, a UI surface | `docs/APPS.md` (required reading before modifying `apps/`) | 41 41 | `talent/` | AI talent configs (markdown prompts + optional `.py` post-hooks) + `SKILL.md`s (journal, coder, partner, …) | defining or tuning a talent; adding a journal-side skill | `talent/journal/SKILL.md`, `docs/PROMPT_TEMPLATES.md` | 42 42 | `scripts/` | Repo maintenance scripts — `check_layer_hygiene.py` | tooling that guards the codebase; wired into `make ci` | (none) | ··· 86 86 87 87 | Target | When to use | 88 88 |--------|-------------| 89 - | `make install` | First setup and whenever `pyproject.toml` or `uv.lock` changes. Creates `.venv/`, syncs deps, installs Playwright chromium, runs `make skills`. | 89 + | `make install` | First setup and whenever `pyproject.toml` or `uv.lock` changes. Creates `.venv/`, syncs deps, runs `make skills`. | 90 90 | `make skills` | After adding or renaming a `SKILL.md` under `talent/` or `apps/*/talent/`. Rewrites the `.claude/` + `.agents/` skill symlinks into `journal/`. (`make install` depends on this; rarely run alone.) | 91 91 | `make update` | Upgrade all deps to latest, regenerate `uv.lock`. Expect test churn. | 92 92 | `make update-prices` | Refresh genai-prices model-cost data when adding a new provider model or when pricing tests fail. | ··· 161 161 - **Run app tests:** `make test-apps` or `make test-app APP=<name>`. 162 162 - **Integration tests** (`tests/integration/`): hit real provider APIs, require `.env` keys, run via `make test-integration`. 163 163 - **After editing `convey/` or `apps/`:** `sol restart-convey` to reload code in a running stack. 164 - - **Screenshots for UI review:** `sol screenshot <route>` (captures into `scratch/`). 165 164 - **`make dev` + `make sandbox`** both write runtime artifacts into the fixtures journal; `tests/fixtures/journal/.gitignore` covers those — never commit them. 166 165 167 166 Full depth: `docs/testing.md`.
-2
Makefile
··· 68 68 $(UV) sync --extra parakeet-onnx-$(PARAKEET_ONNX_VARIANT) || { echo "parakeet install: uv sync --extra parakeet-onnx-$(PARAKEET_ONNX_VARIANT) failed" >&2; exit 1; }; \ 69 69 fi 70 70 @$(VENV_BIN)/python -c "from observe.transcribe.main import PYANNOTE_OVERLAP_MODEL_PATH, PYANNOTE_OVERLAP_MODEL_SHA256, WESPEAKER_MODEL_PATH, WESPEAKER_MODEL_SHA256; from observe.utils import compute_file_sha256; actual = compute_file_sha256(WESPEAKER_MODEL_PATH); assert actual == WESPEAKER_MODEL_SHA256, f'WeSpeaker asset hash mismatch: got {actual}, expected {WESPEAKER_MODEL_SHA256}'; print(f'wespeaker asset ok ({actual[:12]}...)'); actual = compute_file_sha256(PYANNOTE_OVERLAP_MODEL_PATH); assert actual == PYANNOTE_OVERLAP_MODEL_SHA256, f'pyannote asset hash mismatch: got {actual}, expected {PYANNOTE_OVERLAP_MODEL_SHA256}'; print(f'pyannote asset ok ({actual[:12]}...)')" 71 - @echo "Installing Playwright browser for sol screenshot..." 72 - $(VENV_BIN)/playwright install chromium 73 71 @$(MAKE) --no-print-directory skills 74 72 @touch .installed 75 73
+1 -2
README.md
··· 95 95 sol chat # Interactive AI chat from the terminal 96 96 sol transcribe <file> # Transcribe an audio file 97 97 sol indexer # Rebuild the search index 98 - sol screenshot / # Capture a screenshot of the web UI 99 98 ``` 100 99 101 100 Run `sol help` for the full command reference. ··· 120 119 121 120 See [AGENTS.md](AGENTS.md) for development guidelines, coding standards, and testing instructions. 122 121 123 - Use `make dev` to run the full stack against test fixtures, `make ci` for pre-commit checks, and `sol screenshot` for UI testing workflows. 122 + Use `make dev` to run the full stack against test fixtures and `make ci` for pre-commit checks. 124 123 125 124 ## contributing 126 125
-136
apps/chat/tests/test_markdown_bootstrap_browser.py
··· 1 - # SPDX-License-Identifier: AGPL-3.0-only 2 - # Copyright (c) 2026 sol pbc 3 - 4 - from __future__ import annotations 5 - 6 - import contextlib 7 - import json 8 - import os 9 - import shutil 10 - import subprocess 11 - import threading 12 - from collections.abc import Iterator 13 - from datetime import datetime 14 - from pathlib import Path 15 - from typing import Any 16 - 17 - import pytest 18 - from werkzeug.serving import make_server 19 - 20 - from convey import create_app 21 - from convey.chat_stream import append_chat_event 22 - 23 - playwright_sync = pytest.importorskip("playwright.sync_api") 24 - sync_playwright = playwright_sync.sync_playwright 25 - PlaywrightError = playwright_sync.Error 26 - 27 - 28 - def _ms(year: int, month: int, day: int, hour: int, minute: int) -> int: 29 - return int(datetime(year, month, day, hour, minute).timestamp() * 1000) 30 - 31 - 32 - def _copytree_tracked(src: Path, dst: Path) -> None: 33 - result = subprocess.run( 34 - ["git", "ls-files", "."], 35 - cwd=str(src), 36 - capture_output=True, 37 - text=True, 38 - check=True, 39 - ) 40 - for rel in result.stdout.splitlines(): 41 - if not rel: 42 - continue 43 - src_file = src / rel 44 - dst_file = dst / rel 45 - dst_file.parent.mkdir(parents=True, exist_ok=True) 46 - if src_file.is_symlink(): 47 - os.symlink(os.readlink(src_file), dst_file) 48 - else: 49 - shutil.copy2(src_file, dst_file) 50 - 51 - 52 - @contextlib.contextmanager 53 - def _isolated_app_env(journal: Path) -> Iterator[Path]: 54 - previous = os.environ.get("SOLSTONE_JOURNAL") 55 - os.environ["SOLSTONE_JOURNAL"] = str(journal.resolve()) 56 - try: 57 - yield journal 58 - finally: 59 - if previous is None: 60 - os.environ.pop("SOLSTONE_JOURNAL", None) 61 - else: 62 - os.environ["SOLSTONE_JOURNAL"] = previous 63 - 64 - 65 - @contextlib.contextmanager 66 - def _running_chat_server(journal: Path) -> Iterator[str]: 67 - app = create_app(str(journal)) 68 - server = make_server("127.0.0.1", 0, app, threaded=True) 69 - thread = threading.Thread(target=server.serve_forever, daemon=True) 70 - thread.start() 71 - try: 72 - yield f"http://127.0.0.1:{server.server_port}" 73 - finally: 74 - server.shutdown() 75 - thread.join(timeout=5) 76 - 77 - 78 - def _write_localhost_auth_config(journal: Path) -> None: 79 - config_path = journal / "config" / "journal.json" 80 - if config_path.exists(): 81 - config = json.loads(config_path.read_text(encoding="utf-8")) 82 - else: 83 - config = {} 84 - config.setdefault("setup", {})["completed_at"] = 1 85 - config.setdefault("convey", {})["trust_localhost"] = True 86 - config_path.parent.mkdir(parents=True, exist_ok=True) 87 - config_path.write_text( 88 - json.dumps(config, indent=2, sort_keys=True) + "\n", 89 - encoding="utf-8", 90 - ) 91 - 92 - 93 - def _launch_chromium(playwright: Any) -> Any: 94 - try: 95 - return playwright.chromium.launch() 96 - except PlaywrightError as exc: 97 - message = str(exc) 98 - if "Executable doesn't exist" in message or "playwright install" in message: 99 - pytest.skip("Playwright chromium is not installed") 100 - raise 101 - 102 - 103 - def test_ssr_talent_markdown_bootstraps_to_rendered_dom(tmp_path, monkeypatch): 104 - journal = tmp_path / "journal" 105 - _copytree_tracked(Path("tests/fixtures/journal").resolve(), journal) 106 - monkeypatch.setenv("SOLSTONE_JOURNAL", str(journal)) 107 - _write_localhost_auth_config(journal) 108 - 109 - day = "20990102" 110 - append_chat_event( 111 - "talent_finished", 112 - ts=_ms(2099, 1, 2, 9, 3), 113 - use_id="use-md-browser-1", 114 - name="exec", 115 - summary="**done**", 116 - ) 117 - 118 - with _isolated_app_env(journal): 119 - with _running_chat_server(journal) as base_url: 120 - with sync_playwright() as playwright: 121 - browser = _launch_chromium(playwright) 122 - try: 123 - page = browser.new_page() 124 - page.goto( 125 - f"{base_url}/app/chat/{day}", wait_until="domcontentloaded" 126 - ) 127 - 128 - assert ( 129 - page.locator( 130 - ".chat-talent-card-detail--markdown strong" 131 - ).count() 132 - > 0 133 - ) 134 - assert page.locator('[data-markdown="1"]').count() == 0 135 - finally: 136 - browser.close()
-209
convey/screenshot.py
··· 1 - #!/usr/bin/env python3 2 - # SPDX-License-Identifier: AGPL-3.0-only 3 - # Copyright (c) 2026 sol pbc 4 - 5 - """Screenshot utility for Convey web views.""" 6 - 7 - from __future__ import annotations 8 - 9 - import argparse 10 - import os 11 - import sys 12 - 13 - from playwright.sync_api import sync_playwright 14 - 15 - from think.utils import read_service_port, require_solstone, setup_cli 16 - 17 - 18 - class _HelpOnErrorParser(argparse.ArgumentParser): 19 - """ArgumentParser that shows full help on any error.""" 20 - 21 - def error(self, message: str) -> None: 22 - self.print_help(sys.stderr) 23 - sys.stderr.write(f"\nerror: {message}\n") 24 - sys.exit(2) 25 - 26 - 27 - def screenshot( 28 - route: str, 29 - port: int, 30 - output_path: str = "logs/screenshot.png", 31 - host: str = "localhost", 32 - width: int = 1440, 33 - height: int = 900, 34 - script: str | None = None, 35 - facet: str | None = None, 36 - delay: int = 0, 37 - password: str | None = None, 38 - ) -> None: 39 - """ 40 - Capture screenshot of a Convey view. 41 - 42 - Args: 43 - route: The route to screenshot (e.g., "/", "/facets") 44 - port: Server port 45 - output_path: Where to save the screenshot (default: logs/screenshot.png) 46 - host: Server host (default: localhost) 47 - width: Viewport width (default: 1440) 48 - height: Viewport height (default: 900) 49 - script: Optional JavaScript to execute before taking screenshot 50 - facet: Optional facet to select (use "all" for all-facet mode) 51 - delay: Milliseconds to wait after page load before screenshot (default: 0) 52 - password: Optional password for Basic Auth 53 - """ 54 - # Ensure route has leading slash 55 - if not route.startswith("/"): 56 - route = "/" + route 57 - 58 - url = f"http://{host}:{port}{route}" 59 - 60 - # Ensure output directory exists 61 - os.makedirs(os.path.dirname(os.path.abspath(output_path)), exist_ok=True) 62 - 63 - with sync_playwright() as p: 64 - browser = p.chromium.launch() 65 - context_kwargs: dict = {"viewport": {"width": width, "height": height}} 66 - if password: 67 - context_kwargs["http_credentials"] = {"username": "", "password": password} 68 - context = browser.new_context(**context_kwargs) 69 - 70 - # Set facet selection cookie if specified 71 - if facet: 72 - if facet.lower() == "all": 73 - # Clear cookie for all-facet mode (empty value) 74 - cookie_value = "" 75 - else: 76 - cookie_value = facet 77 - context.add_cookies( 78 - [ 79 - { 80 - "name": "selectedFacet", 81 - "value": cookie_value, 82 - "domain": host, 83 - "path": "/", 84 - } 85 - ] 86 - ) 87 - 88 - page = context.new_page() 89 - page.goto(url) 90 - 91 - # Wait for page to be fully loaded 92 - page.wait_for_load_state("networkidle") 93 - 94 - # Wait additional delay if specified (for JS to process hash, animations, etc.) 95 - if delay > 0: 96 - page.wait_for_timeout(delay) 97 - 98 - # Execute custom JavaScript if provided 99 - if script: 100 - page.evaluate(script) 101 - # Small delay for script effects to render 102 - page.wait_for_timeout(100) 103 - 104 - page.screenshot(path=output_path, full_page=True) 105 - browser.close() 106 - 107 - print(f"Screenshot saved to {output_path}") 108 - 109 - 110 - def _get_available_facets() -> list[str]: 111 - """Return list of available facet names.""" 112 - try: 113 - from think.facets import get_facets 114 - 115 - return sorted(get_facets().keys()) 116 - except Exception: 117 - return [] 118 - 119 - 120 - def _validate_facet(facet: str, available: list[str]) -> None: 121 - """Warn if facet doesn't exist (non-fatal).""" 122 - if facet.lower() == "all": 123 - return 124 - 125 - if facet not in available: 126 - print(f"Warning: facet '{facet}' not found in journal") 127 - 128 - 129 - def main() -> None: 130 - """CLI entry point for screenshot utility.""" 131 - # Get available facets for help text and validation 132 - available_facets = _get_available_facets() 133 - facet_choices = ", ".join(available_facets) if available_facets else "none found" 134 - facet_help = f"Facet to select: {facet_choices}, or 'all' for all-facet mode" 135 - 136 - parser = _HelpOnErrorParser(description="Capture screenshots of Convey web views") 137 - parser.add_argument("route", help="Route to screenshot (e.g., /, /facets)") 138 - parser.add_argument( 139 - "-o", 140 - "--output", 141 - default="logs/screenshot.png", 142 - help="Output path (default: logs/screenshot.png)", 143 - ) 144 - parser.add_argument( 145 - "--port", 146 - type=int, 147 - default=None, 148 - help="Server port (default: read from Convey port file)", 149 - ) 150 - parser.add_argument("--width", type=int, default=1440, help="Viewport width") 151 - parser.add_argument("--height", type=int, default=900, help="Viewport height") 152 - parser.add_argument( 153 - "--script", 154 - help="JavaScript to execute before taking screenshot", 155 - ) 156 - parser.add_argument( 157 - "--facet", 158 - help=facet_help, 159 - ) 160 - parser.add_argument( 161 - "--delay", 162 - type=int, 163 - default=None, 164 - help="Delay in ms after page load (default: 500 if route has #fragment, else 0)", 165 - ) 166 - parser.add_argument( 167 - "--password", 168 - help="Password for Basic Auth", 169 - ) 170 - 171 - args = setup_cli(parser) 172 - require_solstone() 173 - 174 - # Determine port: CLI arg takes precedence, then port file, then error 175 - if args.port is not None: 176 - port = args.port 177 - else: 178 - port = read_service_port("convey") 179 - if port is None: 180 - print( 181 - "Error: Convey port not found. Is Convey running? " 182 - "Use --port to specify manually.", 183 - file=sys.stderr, 184 - ) 185 - sys.exit(1) 186 - 187 - # Auto-set delay for fragment routes if not explicitly specified 188 - if args.delay is None: 189 - args.delay = 500 if "#" in args.route else 0 190 - 191 - # Validate facet if specified 192 - if args.facet: 193 - _validate_facet(args.facet, available_facets) 194 - 195 - screenshot( 196 - route=args.route, 197 - output_path=args.output, 198 - port=port, 199 - width=args.width, 200 - height=args.height, 201 - script=args.script, 202 - facet=args.facet, 203 - delay=args.delay, 204 - password=args.password, 205 - ) 206 - 207 - 208 - if __name__ == "__main__": 209 - main()
+1 -1
docs/BACKLOG.md
··· 35 35 ## Testing 36 36 37 37 - [x] Move fixtures/ into tests/ 38 - - [ ] Enable clean fixture-based service startup (Convey on dynamic port) for integration/dev testing with screenshots 38 + - [ ] Enable clean fixture-based service startup (Convey on dynamic port) for integration/dev testing
-1
docs/CONVEY.md
··· 38 38 state.py - global state (journal_root) 39 39 bridge.py - Callosum WebSocket bridge for real-time events 40 40 utils.py - shared helpers (format_date, spawn_agent, etc.) 41 - screenshot.py - screenshot utility for testing 42 41 views/ 43 42 __init__.py - blueprint registration 44 43 home.py - authentication (login/logout) and root redirect
+1 -1
docs/SOLCLI.md
··· 297 297 | Service | `service` (+ aliases `up`, `down`, `start`) | 298 298 | Observe (capture) | `transcribe`, `describe`, `sense`, `transfer`, `observer` | 299 299 | Talent (AI agents) | `agents`, `cortex`, `talent`, `call`, `engage` | 300 - | Convey (web UI) | `convey`, `restart-convey`, `screenshot`, `maint` | 300 + | Convey (web UI) | `convey`, `restart-convey`, `maint` | 301 301 | Specialized | `config`, `streams`, `journal-stats`, `formatter`, `detect-created` | 302 302 | Help | `help`, `chat` | 303 303
+1 -3
docs/testing.md
··· 35 35 - `make coverage` to generate a coverage report 36 36 - `make ci` before committing (formats, lints, tests) 37 37 - Always run `sol restart-convey` after editing `convey/` or `apps/` to reload code 38 - - Use `sol screenshot <route>` to capture UI screenshots for visual testing 39 38 40 39 ## Worktree Development 41 40 ··· 45 44 make dev # Start stack (Ctrl+C to stop) 46 45 ``` 47 46 48 - In a second terminal, take screenshots or hit endpoints: 47 + In a second terminal, hit endpoints: 49 48 50 49 ```bash 51 50 export SOLSTONE_JOURNAL=tests/fixtures/journal 52 51 export PATH=$(pwd)/.venv/bin:$PATH 53 - sol screenshot / -o scratch/home.png 54 52 curl -s http://localhost:$(cat tests/fixtures/journal/health/convey.port)/ 55 53 ``` 56 54
-1
pyproject.toml
··· 91 91 "pytest-cov", 92 92 "pytest-timeout", 93 93 "freezegun", 94 - "playwright>=1.40.0", 95 94 "requests", 96 95 ] 97 96
+1 -1
scripts/doctor.py
··· 12 12 Decision log: 13 13 - uv floor: 0.7.12 — `uv.lock` revision=3 requires >= 0.7.12 per 14 14 astral-sh/uv#15220. 15 - - disk threshold: 10 GiB — measured `.venv`=7.88 GiB + playwright=0.61 GiB + 15 + - disk threshold: 10 GiB — measured `.venv`=7.88 GiB + 16 16 uv-cache first-install growth ~1 GiB + buffer. 17 17 - Makefile UV-guard strategy: MAKECMDGOALS filter; prep verified the 18 18 doctor-only matrix on GNU make.
+1 -2
tests/integration/test_apps.py
··· 47 47 response = convey_client.get(route, follow_redirects=True) 48 48 49 49 assert response.status_code == 200, ( 50 - f"App '{app_name}' at {route} returned {response.status_code}. " 51 - "Always use 'make screenshot' to validate this failure" 50 + f"App '{app_name}' at {route} returned {response.status_code}." 52 51 )
-2
think/sol_cli.py
··· 71 71 # convey package - web UI 72 72 "convey": "convey.cli", 73 73 "restart-convey": "convey.restart", 74 - "screenshot": "convey.screenshot", 75 74 "maint": "convey.maint_cli", 76 75 "service": "think.service", 77 76 } ··· 124 123 "Convey (web UI)": [ 125 124 "convey", 126 125 "restart-convey", 127 - "screenshot", 128 126 "maint", 129 127 ], 130 128 "Specialized tools": [
-93
uv.lock
··· 941 941 ] 942 942 943 943 [[package]] 944 - name = "greenlet" 945 - version = "3.3.1" 946 - source = { registry = "https://pypi.org/simple" } 947 - sdist = { url = "https://files.pythonhosted.org/packages/8a/99/1cd3411c56a410994669062bd73dd58270c00cc074cac15f385a1fd91f8a/greenlet-3.3.1.tar.gz", hash = "sha256:41848f3230b58c08bb43dee542e74a2a2e34d3c59dc3076cec9151aeeedcae98", size = 184690, upload-time = "2026-01-23T15:31:02.076Z" } 948 - wheels = [ 949 - { url = "https://files.pythonhosted.org/packages/fe/65/5b235b40581ad75ab97dcd8b4218022ae8e3ab77c13c919f1a1dfe9171fd/greenlet-3.3.1-cp310-cp310-macosx_11_0_universal2.whl", hash = "sha256:04bee4775f40ecefcdaa9d115ab44736cd4b9c5fba733575bfe9379419582e13", size = 273723, upload-time = "2026-01-23T15:30:37.521Z" }, 950 - { url = "https://files.pythonhosted.org/packages/ce/ad/eb4729b85cba2d29499e0a04ca6fbdd8f540afd7be142fd571eea43d712f/greenlet-3.3.1-cp310-cp310-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:50e1457f4fed12a50e427988a07f0f9df53cf0ee8da23fab16e6732c2ec909d4", size = 574874, upload-time = "2026-01-23T16:00:54.551Z" }, 951 - { url = "https://files.pythonhosted.org/packages/87/32/57cad7fe4c8b82fdaa098c89498ef85ad92dfbb09d5eb713adedfc2ae1f5/greenlet-3.3.1-cp310-cp310-manylinux_2_24_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:070472cd156f0656f86f92e954591644e158fd65aa415ffbe2d44ca77656a8f5", size = 586309, upload-time = "2026-01-23T16:05:25.18Z" }, 952 - { url = "https://files.pythonhosted.org/packages/66/66/f041005cb87055e62b0d68680e88ec1a57f4688523d5e2fb305841bc8307/greenlet-3.3.1-cp310-cp310-manylinux_2_24_s390x.manylinux_2_28_s390x.whl", hash = "sha256:1108b61b06b5224656121c3c8ee8876161c491cbe74e5c519e0634c837cf93d5", size = 597461, upload-time = "2026-01-23T16:15:51.943Z" }, 953 - { url = "https://files.pythonhosted.org/packages/87/eb/8a1ec2da4d55824f160594a75a9d8354a5fe0a300fb1c48e7944265217e1/greenlet-3.3.1-cp310-cp310-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:3a300354f27dd86bae5fbf7002e6dd2b3255cd372e9242c933faf5e859b703fe", size = 586985, upload-time = "2026-01-23T15:32:47.968Z" }, 954 - { url = "https://files.pythonhosted.org/packages/15/1c/0621dd4321dd8c351372ee8f9308136acb628600658a49be1b7504208738/greenlet-3.3.1-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:e84b51cbebf9ae573b5fbd15df88887815e3253fc000a7d0ff95170e8f7e9729", size = 1547271, upload-time = "2026-01-23T16:04:18.977Z" }, 955 - { url = "https://files.pythonhosted.org/packages/9d/53/24047f8924c83bea7a59c8678d9571209c6bfe5f4c17c94a78c06024e9f2/greenlet-3.3.1-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:e0093bd1a06d899892427217f0ff2a3c8f306182b8c754336d32e2d587c131b4", size = 1613427, upload-time = "2026-01-23T15:33:44.428Z" }, 956 - { url = "https://files.pythonhosted.org/packages/ff/07/ac9bf1ec008916d1a3373cae212884c1dcff4a4ba0d41127ce81a8deb4e9/greenlet-3.3.1-cp310-cp310-win_amd64.whl", hash = "sha256:7932f5f57609b6a3b82cc11877709aa7a98e3308983ed93552a1c377069b20c8", size = 226100, upload-time = "2026-01-23T15:30:56.957Z" }, 957 - { url = "https://files.pythonhosted.org/packages/ec/e8/2e1462c8fdbe0f210feb5ac7ad2d9029af8be3bf45bd9fa39765f821642f/greenlet-3.3.1-cp311-cp311-macosx_11_0_universal2.whl", hash = "sha256:5fd23b9bc6d37b563211c6abbb1b3cab27db385a4449af5c32e932f93017080c", size = 274974, upload-time = "2026-01-23T15:31:02.891Z" }, 958 - { url = "https://files.pythonhosted.org/packages/7e/a8/530a401419a6b302af59f67aaf0b9ba1015855ea7e56c036b5928793c5bd/greenlet-3.3.1-cp311-cp311-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:09f51496a0bfbaa9d74d36a52d2580d1ef5ed4fdfcff0a73730abfbbbe1403dd", size = 577175, upload-time = "2026-01-23T16:00:56.213Z" }, 959 - { url = "https://files.pythonhosted.org/packages/8e/89/7e812bb9c05e1aaef9b597ac1d0962b9021d2c6269354966451e885c4e6b/greenlet-3.3.1-cp311-cp311-manylinux_2_24_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:cb0feb07fe6e6a74615ee62a880007d976cf739b6669cce95daa7373d4fc69c5", size = 590401, upload-time = "2026-01-23T16:05:26.365Z" }, 960 - { url = "https://files.pythonhosted.org/packages/70/ae/e2d5f0e59b94a2269b68a629173263fa40b63da32f5c231307c349315871/greenlet-3.3.1-cp311-cp311-manylinux_2_24_s390x.manylinux_2_28_s390x.whl", hash = "sha256:67ea3fc73c8cd92f42467a72b75e8f05ed51a0e9b1d15398c913416f2dafd49f", size = 601161, upload-time = "2026-01-23T16:15:53.456Z" }, 961 - { url = "https://files.pythonhosted.org/packages/5c/ae/8d472e1f5ac5efe55c563f3eabb38c98a44b832602e12910750a7c025802/greenlet-3.3.1-cp311-cp311-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:39eda9ba259cc9801da05351eaa8576e9aa83eb9411e8f0c299e05d712a210f2", size = 590272, upload-time = "2026-01-23T15:32:49.411Z" }, 962 - { url = "https://files.pythonhosted.org/packages/a8/51/0fde34bebfcadc833550717eade64e35ec8738e6b097d5d248274a01258b/greenlet-3.3.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:e2e7e882f83149f0a71ac822ebf156d902e7a5d22c9045e3e0d1daf59cee2cc9", size = 1550729, upload-time = "2026-01-23T16:04:20.867Z" }, 963 - { url = "https://files.pythonhosted.org/packages/16/c9/2fb47bee83b25b119d5a35d580807bb8b92480a54b68fef009a02945629f/greenlet-3.3.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:80aa4d79eb5564f2e0a6144fcc744b5a37c56c4a92d60920720e99210d88db0f", size = 1615552, upload-time = "2026-01-23T15:33:45.743Z" }, 964 - { url = "https://files.pythonhosted.org/packages/1f/54/dcf9f737b96606f82f8dd05becfb8d238db0633dd7397d542a296fe9cad3/greenlet-3.3.1-cp311-cp311-win_amd64.whl", hash = "sha256:32e4ca9777c5addcbf42ff3915d99030d8e00173a56f80001fb3875998fe410b", size = 226462, upload-time = "2026-01-23T15:36:50.422Z" }, 965 - { url = "https://files.pythonhosted.org/packages/91/37/61e1015cf944ddd2337447d8e97fb423ac9bc21f9963fb5f206b53d65649/greenlet-3.3.1-cp311-cp311-win_arm64.whl", hash = "sha256:da19609432f353fed186cc1b85e9440db93d489f198b4bdf42ae19cc9d9ac9b4", size = 225715, upload-time = "2026-01-23T15:33:17.298Z" }, 966 - { url = "https://files.pythonhosted.org/packages/f9/c8/9d76a66421d1ae24340dfae7e79c313957f6e3195c144d2c73333b5bfe34/greenlet-3.3.1-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:7e806ca53acf6d15a888405880766ec84721aa4181261cd11a457dfe9a7a4975", size = 276443, upload-time = "2026-01-23T15:30:10.066Z" }, 967 - { url = "https://files.pythonhosted.org/packages/81/99/401ff34bb3c032d1f10477d199724f5e5f6fbfb59816ad1455c79c1eb8e7/greenlet-3.3.1-cp312-cp312-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:d842c94b9155f1c9b3058036c24ffb8ff78b428414a19792b2380be9cecf4f36", size = 597359, upload-time = "2026-01-23T16:00:57.394Z" }, 968 - { url = "https://files.pythonhosted.org/packages/2b/bc/4dcc0871ed557792d304f50be0f7487a14e017952ec689effe2180a6ff35/greenlet-3.3.1-cp312-cp312-manylinux_2_24_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:20fedaadd422fa02695f82093f9a98bad3dab5fcda793c658b945fcde2ab27ba", size = 607805, upload-time = "2026-01-23T16:05:28.068Z" }, 969 - { url = "https://files.pythonhosted.org/packages/3b/cd/7a7ca57588dac3389e97f7c9521cb6641fd8b6602faf1eaa4188384757df/greenlet-3.3.1-cp312-cp312-manylinux_2_24_s390x.manylinux_2_28_s390x.whl", hash = "sha256:c620051669fd04ac6b60ebc70478210119c56e2d5d5df848baec4312e260e4ca", size = 622363, upload-time = "2026-01-23T16:15:54.754Z" }, 970 - { url = "https://files.pythonhosted.org/packages/cf/05/821587cf19e2ce1f2b24945d890b164401e5085f9d09cbd969b0c193cd20/greenlet-3.3.1-cp312-cp312-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:14194f5f4305800ff329cbf02c5fcc88f01886cadd29941b807668a45f0d2336", size = 609947, upload-time = "2026-01-23T15:32:51.004Z" }, 971 - { url = "https://files.pythonhosted.org/packages/a4/52/ee8c46ed9f8babaa93a19e577f26e3d28a519feac6350ed6f25f1afee7e9/greenlet-3.3.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:7b2fe4150a0cf59f847a67db8c155ac36aed89080a6a639e9f16df5d6c6096f1", size = 1567487, upload-time = "2026-01-23T16:04:22.125Z" }, 972 - { url = "https://files.pythonhosted.org/packages/8f/7c/456a74f07029597626f3a6db71b273a3632aecb9afafeeca452cfa633197/greenlet-3.3.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:49f4ad195d45f4a66a0eb9c1ba4832bb380570d361912fa3554746830d332149", size = 1636087, upload-time = "2026-01-23T15:33:47.486Z" }, 973 - { url = "https://files.pythonhosted.org/packages/34/2f/5e0e41f33c69655300a5e54aeb637cf8ff57f1786a3aba374eacc0228c1d/greenlet-3.3.1-cp312-cp312-win_amd64.whl", hash = "sha256:cc98b9c4e4870fa983436afa999d4eb16b12872fab7071423d5262fa7120d57a", size = 227156, upload-time = "2026-01-23T15:34:34.808Z" }, 974 - { url = "https://files.pythonhosted.org/packages/c8/ab/717c58343cf02c5265b531384b248787e04d8160b8afe53d9eec053d7b44/greenlet-3.3.1-cp312-cp312-win_arm64.whl", hash = "sha256:bfb2d1763d777de5ee495c85309460f6fd8146e50ec9d0ae0183dbf6f0a829d1", size = 226403, upload-time = "2026-01-23T15:31:39.372Z" }, 975 - { url = "https://files.pythonhosted.org/packages/ec/ab/d26750f2b7242c2b90ea2ad71de70cfcd73a948a49513188a0fc0d6fc15a/greenlet-3.3.1-cp313-cp313-macosx_11_0_universal2.whl", hash = "sha256:7ab327905cabb0622adca5971e488064e35115430cec2c35a50fd36e72a315b3", size = 275205, upload-time = "2026-01-23T15:30:24.556Z" }, 976 - { url = "https://files.pythonhosted.org/packages/10/d3/be7d19e8fad7c5a78eeefb2d896a08cd4643e1e90c605c4be3b46264998f/greenlet-3.3.1-cp313-cp313-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:65be2f026ca6a176f88fb935ee23c18333ccea97048076aef4db1ef5bc0713ac", size = 599284, upload-time = "2026-01-23T16:00:58.584Z" }, 977 - { url = "https://files.pythonhosted.org/packages/ae/21/fe703aaa056fdb0f17e5afd4b5c80195bbdab701208918938bd15b00d39b/greenlet-3.3.1-cp313-cp313-manylinux_2_24_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:7a3ae05b3d225b4155bda56b072ceb09d05e974bc74be6c3fc15463cf69f33fd", size = 610274, upload-time = "2026-01-23T16:05:29.312Z" }, 978 - { url = "https://files.pythonhosted.org/packages/06/00/95df0b6a935103c0452dad2203f5be8377e551b8466a29650c4c5a5af6cc/greenlet-3.3.1-cp313-cp313-manylinux_2_24_s390x.manylinux_2_28_s390x.whl", hash = "sha256:12184c61e5d64268a160226fb4818af4df02cfead8379d7f8b99a56c3a54ff3e", size = 624375, upload-time = "2026-01-23T16:15:55.915Z" }, 979 - { url = "https://files.pythonhosted.org/packages/cb/86/5c6ab23bb3c28c21ed6bebad006515cfe08b04613eb105ca0041fecca852/greenlet-3.3.1-cp313-cp313-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:6423481193bbbe871313de5fd06a082f2649e7ce6e08015d2a76c1e9186ca5b3", size = 612904, upload-time = "2026-01-23T15:32:52.317Z" }, 980 - { url = "https://files.pythonhosted.org/packages/c2/f3/7949994264e22639e40718c2daf6f6df5169bf48fb038c008a489ec53a50/greenlet-3.3.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:33a956fe78bbbda82bfc95e128d61129b32d66bcf0a20a1f0c08aa4839ffa951", size = 1567316, upload-time = "2026-01-23T16:04:23.316Z" }, 981 - { url = "https://files.pythonhosted.org/packages/8d/6e/d73c94d13b6465e9f7cd6231c68abde838bb22408596c05d9059830b7872/greenlet-3.3.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:4b065d3284be43728dd280f6f9a13990b56470b81be20375a207cdc814a983f2", size = 1636549, upload-time = "2026-01-23T15:33:48.643Z" }, 982 - { url = "https://files.pythonhosted.org/packages/5e/b3/c9c23a6478b3bcc91f979ce4ca50879e4d0b2bd7b9a53d8ecded719b92e2/greenlet-3.3.1-cp313-cp313-win_amd64.whl", hash = "sha256:27289986f4e5b0edec7b5a91063c109f0276abb09a7e9bdab08437525977c946", size = 227042, upload-time = "2026-01-23T15:33:58.216Z" }, 983 - { url = "https://files.pythonhosted.org/packages/90/e7/824beda656097edee36ab15809fd063447b200cc03a7f6a24c34d520bc88/greenlet-3.3.1-cp313-cp313-win_arm64.whl", hash = "sha256:2f080e028001c5273e0b42690eaf359aeef9cb1389da0f171ea51a5dc3c7608d", size = 226294, upload-time = "2026-01-23T15:30:52.73Z" }, 984 - { url = "https://files.pythonhosted.org/packages/ae/fb/011c7c717213182caf78084a9bea51c8590b0afda98001f69d9f853a495b/greenlet-3.3.1-cp314-cp314-macosx_11_0_universal2.whl", hash = "sha256:bd59acd8529b372775cd0fcbc5f420ae20681c5b045ce25bd453ed8455ab99b5", size = 275737, upload-time = "2026-01-23T15:32:16.889Z" }, 985 - { url = "https://files.pythonhosted.org/packages/41/2e/a3a417d620363fdbb08a48b1dd582956a46a61bf8fd27ee8164f9dfe87c2/greenlet-3.3.1-cp314-cp314-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b31c05dd84ef6871dd47120386aed35323c944d86c3d91a17c4b8d23df62f15b", size = 646422, upload-time = "2026-01-23T16:01:00.354Z" }, 986 - { url = "https://files.pythonhosted.org/packages/b4/09/c6c4a0db47defafd2d6bab8ddfe47ad19963b4e30f5bed84d75328059f8c/greenlet-3.3.1-cp314-cp314-manylinux_2_24_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:02925a0bfffc41e542c70aa14c7eda3593e4d7e274bfcccca1827e6c0875902e", size = 658219, upload-time = "2026-01-23T16:05:30.956Z" }, 987 - { url = "https://files.pythonhosted.org/packages/e2/89/b95f2ddcc5f3c2bc09c8ee8d77be312df7f9e7175703ab780f2014a0e781/greenlet-3.3.1-cp314-cp314-manylinux_2_24_s390x.manylinux_2_28_s390x.whl", hash = "sha256:3e0f3878ca3a3ff63ab4ea478585942b53df66ddde327b59ecb191b19dbbd62d", size = 671455, upload-time = "2026-01-23T16:15:57.232Z" }, 988 - { url = "https://files.pythonhosted.org/packages/80/38/9d42d60dffb04b45f03dbab9430898352dba277758640751dc5cc316c521/greenlet-3.3.1-cp314-cp314-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:34a729e2e4e4ffe9ae2408d5ecaf12f944853f40ad724929b7585bca808a9d6f", size = 660237, upload-time = "2026-01-23T15:32:53.967Z" }, 989 - { url = "https://files.pythonhosted.org/packages/96/61/373c30b7197f9e756e4c81ae90a8d55dc3598c17673f91f4d31c3c689c3f/greenlet-3.3.1-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:aec9ab04e82918e623415947921dea15851b152b822661cce3f8e4393c3df683", size = 1615261, upload-time = "2026-01-23T16:04:25.066Z" }, 990 - { url = "https://files.pythonhosted.org/packages/fd/d3/ca534310343f5945316f9451e953dcd89b36fe7a19de652a1dc5a0eeef3f/greenlet-3.3.1-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:71c767cf281a80d02b6c1bdc41c9468e1f5a494fb11bc8688c360524e273d7b1", size = 1683719, upload-time = "2026-01-23T15:33:50.61Z" }, 991 - { url = "https://files.pythonhosted.org/packages/52/cb/c21a3fd5d2c9c8b622e7bede6d6d00e00551a5ee474ea6d831b5f567a8b4/greenlet-3.3.1-cp314-cp314-win_amd64.whl", hash = "sha256:96aff77af063b607f2489473484e39a0bbae730f2ea90c9e5606c9b73c44174a", size = 228125, upload-time = "2026-01-23T15:32:45.265Z" }, 992 - { url = "https://files.pythonhosted.org/packages/6a/8e/8a2db6d11491837af1de64b8aff23707c6e85241be13c60ed399a72e2ef8/greenlet-3.3.1-cp314-cp314-win_arm64.whl", hash = "sha256:b066e8b50e28b503f604fa538adc764a638b38cf8e81e025011d26e8a627fa79", size = 227519, upload-time = "2026-01-23T15:31:47.284Z" }, 993 - { url = "https://files.pythonhosted.org/packages/28/24/cbbec49bacdcc9ec652a81d3efef7b59f326697e7edf6ed775a5e08e54c2/greenlet-3.3.1-cp314-cp314t-macosx_11_0_universal2.whl", hash = "sha256:3e63252943c921b90abb035ebe9de832c436401d9c45f262d80e2d06cc659242", size = 282706, upload-time = "2026-01-23T15:33:05.525Z" }, 994 - { url = "https://files.pythonhosted.org/packages/86/2e/4f2b9323c144c4fe8842a4e0d92121465485c3c2c5b9e9b30a52e80f523f/greenlet-3.3.1-cp314-cp314t-manylinux_2_24_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:76e39058e68eb125de10c92524573924e827927df5d3891fbc97bd55764a8774", size = 651209, upload-time = "2026-01-23T16:01:01.517Z" }, 995 - { url = "https://files.pythonhosted.org/packages/d9/87/50ca60e515f5bb55a2fbc5f0c9b5b156de7d2fc51a0a69abc9d23914a237/greenlet-3.3.1-cp314-cp314t-manylinux_2_24_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:c9f9d5e7a9310b7a2f416dd13d2e3fd8b42d803968ea580b7c0f322ccb389b97", size = 654300, upload-time = "2026-01-23T16:05:32.199Z" }, 996 - { url = "https://files.pythonhosted.org/packages/7c/25/c51a63f3f463171e09cb586eb64db0861eb06667ab01a7968371a24c4f3b/greenlet-3.3.1-cp314-cp314t-manylinux_2_24_s390x.manylinux_2_28_s390x.whl", hash = "sha256:4b9721549a95db96689458a1e0ae32412ca18776ed004463df3a9299c1b257ab", size = 662574, upload-time = "2026-01-23T16:15:58.364Z" }, 997 - { url = "https://files.pythonhosted.org/packages/1d/94/74310866dfa2b73dd08659a3d18762f83985ad3281901ba0ee9a815194fb/greenlet-3.3.1-cp314-cp314t-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:92497c78adf3ac703b57f1e3813c2d874f27f71a178f9ea5887855da413cd6d2", size = 653842, upload-time = "2026-01-23T15:32:55.671Z" }, 998 - { url = "https://files.pythonhosted.org/packages/97/43/8bf0ffa3d498eeee4c58c212a3905dd6146c01c8dc0b0a046481ca29b18c/greenlet-3.3.1-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:ed6b402bc74d6557a705e197d47f9063733091ed6357b3de33619d8a8d93ac53", size = 1614917, upload-time = "2026-01-23T16:04:26.276Z" }, 999 - { url = "https://files.pythonhosted.org/packages/89/90/a3be7a5f378fc6e84abe4dcfb2ba32b07786861172e502388b4c90000d1b/greenlet-3.3.1-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:59913f1e5ada20fde795ba906916aea25d442abcc0593fba7e26c92b7ad76249", size = 1676092, upload-time = "2026-01-23T15:33:52.176Z" }, 1000 - { url = "https://files.pythonhosted.org/packages/e1/2b/98c7f93e6db9977aaee07eb1e51ca63bd5f779b900d362791d3252e60558/greenlet-3.3.1-cp314-cp314t-win_amd64.whl", hash = "sha256:301860987846c24cb8964bdec0e31a96ad4a2a801b41b4ef40963c1b44f33451", size = 233181, upload-time = "2026-01-23T15:33:00.29Z" }, 1001 - ] 1002 - 1003 - [[package]] 1004 944 name = "h11" 1005 945 version = "0.16.0" 1006 946 source = { registry = "https://pypi.org/simple" } ··· 2239 2179 ] 2240 2180 2241 2181 [[package]] 2242 - name = "playwright" 2243 - version = "1.58.0" 2244 - source = { registry = "https://pypi.org/simple" } 2245 - dependencies = [ 2246 - { name = "greenlet" }, 2247 - { name = "pyee" }, 2248 - ] 2249 - wheels = [ 2250 - { 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" }, 2251 - { 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" }, 2252 - { 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" }, 2253 - { 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" }, 2254 - { 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" }, 2255 - { 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" }, 2256 - { 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" }, 2257 - { 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" }, 2258 - ] 2259 - 2260 - [[package]] 2261 2182 name = "pluggy" 2262 2183 version = "1.6.0" 2263 2184 source = { registry = "https://pypi.org/simple" } ··· 2478 2399 sdist = { url = "https://files.pythonhosted.org/packages/36/ee/fb410c5c854b6a081a49077912a9765aeffd8e07cbb0663cfda310b01fb4/pydyf-0.12.1.tar.gz", hash = "sha256:fbd7e759541ac725c29c506612003de393249b94310ea78ae44cb1d04b220095", size = 17716, upload-time = "2025-12-02T14:52:14.244Z" } 2479 2400 wheels = [ 2480 2401 { url = "https://files.pythonhosted.org/packages/22/11/47efe2f66ba848a107adfd490b508f5c0cedc82127950553dca44d29e6c4/pydyf-0.12.1-py3-none-any.whl", hash = "sha256:ea25b4e1fe7911195cb57067560daaa266639184e8335365cc3ee5214e7eaadc", size = 8028, upload-time = "2025-12-02T14:52:12.938Z" }, 2481 - ] 2482 - 2483 - [[package]] 2484 - name = "pyee" 2485 - version = "13.0.0" 2486 - source = { registry = "https://pypi.org/simple" } 2487 - dependencies = [ 2488 - { name = "typing-extensions" }, 2489 - ] 2490 - sdist = { url = "https://files.pythonhosted.org/packages/95/03/1fd98d5841cd7964a27d729ccf2199602fe05eb7a405c1462eb7277945ed/pyee-13.0.0.tar.gz", hash = "sha256:b391e3c5a434d1f5118a25615001dbc8f669cf410ab67d04c4d4e07c55481c37", size = 31250, upload-time = "2025-03-17T18:53:15.955Z" } 2491 - wheels = [ 2492 - { url = "https://files.pythonhosted.org/packages/9b/4d/b9add7c84060d4c1906abe9a7e5359f2a60f7a9a4f67268b2766673427d8/pyee-13.0.0-py3-none-any.whl", hash = "sha256:48195a3cddb3b1515ce0695ed76036b5ccc2ef3a9f963ff9f77aec0139845498", size = 15730, upload-time = "2025-03-17T18:53:14.532Z" }, 2493 2402 ] 2494 2403 2495 2404 [[package]] ··· 3455 3364 { name = "opencv-python-headless" }, 3456 3365 { name = "pdf2image" }, 3457 3366 { name = "pillow" }, 3458 - { name = "playwright" }, 3459 3367 { name = "psutil" }, 3460 3368 { name = "pyjwt" }, 3461 3369 { name = "pyopenssl" }, ··· 3544 3452 { name = "opencv-python-headless" }, 3545 3453 { name = "pdf2image" }, 3546 3454 { name = "pillow" }, 3547 - { name = "playwright", specifier = ">=1.40.0" }, 3548 3455 { name = "psutil" }, 3549 3456 { name = "pyjwt", specifier = ">=2.8" }, 3550 3457 { name = "pyopenssl", specifier = ">=24.0" },