From 8d509fc6f1308f45f5f8bfd82d97649a5e040ec8 Mon Sep 17 00:00:00 2001 From: tegwick Date: Tue, 23 Jun 2026 21:37:07 +0200 Subject: [PATCH] Implement SAND-WP-0003: validation meta-framework extraction Port e2e-framework schema, runner, and reporter into wise-validator with sand-boxer CLI integration, validate run CLI, unit tests, registry capability, and operator docs. --- Makefile | 29 ++ README.md | 42 +- docs/integrations/sand-boxer.md | 45 ++ docs/runbooks/validate-compose-e2e.md | 87 ++++ pyproject.toml | 41 ++ .../capabilities/validation.compose-e2e.md | 42 ++ registry/indexes/capabilities.yaml | 8 +- scripts/smoke-validate-run.sh | 17 + src/wisevalidator/__init__.py | 3 + src/wisevalidator/cli.py | 70 ++++ src/wisevalidator/models.py | 17 + src/wisevalidator/reporter.py | 53 +++ src/wisevalidator/runner.py | 113 +++++ src/wisevalidator/sandbox_client.py | 97 +++++ src/wisevalidator/schema.py | 63 +++ src/wisevalidator/ssh.py | 87 ++++ .../fixtures/minimal_repo/docker-compose.yml | 5 + tests/fixtures/minimal_repo/e2e/e2e.yml | 9 + tests/test_reporter.py | 37 ++ tests/test_runner.py | 104 +++++ tests/test_sandbox_client.py | 59 +++ tests/test_schema.py | 25 ++ uv.lock | 386 ++++++++++++++++++ 23 files changed, 1435 insertions(+), 4 deletions(-) create mode 100644 Makefile create mode 100644 docs/integrations/sand-boxer.md create mode 100644 docs/runbooks/validate-compose-e2e.md create mode 100644 pyproject.toml create mode 100644 registry/capabilities/validation.compose-e2e.md create mode 100755 scripts/smoke-validate-run.sh create mode 100644 src/wisevalidator/__init__.py create mode 100644 src/wisevalidator/cli.py create mode 100644 src/wisevalidator/models.py create mode 100644 src/wisevalidator/reporter.py create mode 100644 src/wisevalidator/runner.py create mode 100644 src/wisevalidator/sandbox_client.py create mode 100644 src/wisevalidator/schema.py create mode 100644 src/wisevalidator/ssh.py create mode 100644 tests/fixtures/minimal_repo/docker-compose.yml create mode 100644 tests/fixtures/minimal_repo/e2e/e2e.yml create mode 100644 tests/test_reporter.py create mode 100644 tests/test_runner.py create mode 100644 tests/test_sandbox_client.py create mode 100644 tests/test_schema.py create mode 100644 uv.lock diff --git a/Makefile b/Makefile new file mode 100644 index 0000000..a270985 --- /dev/null +++ b/Makefile @@ -0,0 +1,29 @@ +.DEFAULT_GOAL := help + +.PHONY: help setup install test lint format build check cli-version + +help: ## List available make targets + @awk 'BEGIN {FS = ":.*## "}; /^[a-zA-Z0-9_.-]+:.*## / {printf " %-16s %s\n", $$1, $$2}' $(MAKEFILE_LIST) + +setup: ## Sync dependencies into .venv + uv sync --all-groups + +install: ## Install validate CLI to ~/.local/bin (editable) + uv tool install -e . --force + +test: ## Run the test suite + uv run pytest + +lint: ## Run ruff lint checks + uv run ruff check . + +format: ## Auto-format with ruff + uv run ruff format . + +build: ## Build wheel and sdist + uv build + +check: lint test ## Run lint and tests (CI gate) + +cli-version: ## Smoke test: validate --help + uv run validate --help \ No newline at end of file diff --git a/README.md b/README.md index 9d0f190..f6d205b 100644 --- a/README.md +++ b/README.md @@ -1 +1,41 @@ -Agent first usecase based e2e test and health framework. \ No newline at end of file +# wise-validator + +Cross-repo use-case validation for Coulomb — health polling, test execution, and +State Hub reporting on environments established by [sand-boxer](https://gitea.coulomb). + +**Charter:** `INTENT.md` + +## Quick start + +```bash +make setup && make install # validate CLI → ~/.local/bin + +# Requires sand-boxer on PATH: cd ~/sand-boxer && make install +export SANDBOXER_HOST=coulombcore +export SANDBOXER_COMPOSE_CMD=podman-compose + +validate run ~/activity-core +``` + +## Commands + +```bash +validate run # full e2e validation +validate run --keep # skip sandbox destroy +validate run --no-report # skip State Hub +validate version +``` + +## Layout + +- `src/wisevalidator/` — schema, runner, reporter, sand-boxer client +- `docs/runbooks/validate-compose-e2e.md` — operator runbook +- `docs/integrations/sand-boxer.md` — sibling boundary + +## Development + +```bash +make check # lint + test +make test +make lint +``` \ No newline at end of file diff --git a/docs/integrations/sand-boxer.md b/docs/integrations/sand-boxer.md new file mode 100644 index 0000000..61aed29 --- /dev/null +++ b/docs/integrations/sand-boxer.md @@ -0,0 +1,45 @@ +# sand-boxer integration + +wise-validator **consumes** sand-boxer for isolated execution environments. +sand-boxer is self-sustained and does not depend on wise-validator. + +## Provision + +```bash +# Invoked internally by `validate run`; equivalent manual call: +sandboxer create \ + --profile profile.compose-e2e \ + --input repo=/path/to/repo \ + --actor atm \ + --project wise-validator \ + --host "${SANDBOXER_HOST:-coulombcore}" +``` + +Environment variables: + +| Variable | Purpose | +|----------|---------| +| `SANDBOXER_HOST` | Default sandbox host | +| `SANDBOXER_SSH_USER` | SSH user when not in reachability | +| `SANDBOXER_SSH_KEY` | SSH private key path | +| `SANDBOXER_COMPOSE_CMD` | `podman-compose` on CoulombCore | +| `SANDBOXER_BIN` | Override `sandboxer` binary path | + +## wise-validator steps after `ready` + +1. Poll `health_checks` from repo `e2e/e2e.yml` via SSH `curl` on remote host +2. Run `test_command` in `reachability.remote_dir` +3. `destroy` per `cleanup` policy (`always` / `on_success` / `never`, or `--keep`) + +## Ownership + +| Concern | Owner | +|---------|-------| +| rsync + compose up + reachability | sand-boxer | +| `e2e/e2e.yml` parsing | wise-validator | +| HTTP health polling | wise-validator | +| Test command + pass/fail | wise-validator | +| State Hub `e2e_result` events | wise-validator | +| Sandbox lifecycle events | sand-boxer | + +Canon in sand-boxer: `sand-boxer/docs/integrations/wise-validator.md` \ No newline at end of file diff --git a/docs/runbooks/validate-compose-e2e.md b/docs/runbooks/validate-compose-e2e.md new file mode 100644 index 0000000..0177a4b --- /dev/null +++ b/docs/runbooks/validate-compose-e2e.md @@ -0,0 +1,87 @@ +# validate compose-e2e — Runbook + +Cross-repo e2e validation using wise-validator + sand-boxer. + +## Prerequisites + +**Workstation:** + +- `sandboxer` on PATH (`cd ~/sand-boxer && make install`) +- `validate` on PATH (`cd ~/wise-validator && make install`) +- `ssh` available (BatchMode; respects `~/.ssh/config`) +- State Hub on `:8000` for result reporting (optional) + +**Sandbox host (CoulombCore / sandboxer01):** + +- SSH access +- `podman-compose` or `docker compose` +- Sufficient disk for images + +## First run + +```bash +export SANDBOXER_HOST=92.205.130.254 # or coulombcore alias +export SANDBOXER_COMPOSE_CMD=podman-compose + +validate run ~/activity-core +``` + +Output: sandbox create → health wait → test → destroy. Exit 0 = pass, 1 = fail. + +## Options + +```bash +# Keep sandbox for debugging +validate run ~/activity-core --keep + +# Attach State Hub workstream +validate run ~/activity-core --workstream-id + +# Skip State Hub reporting +validate run ~/activity-core --no-report + +# Override host +validate run ~/activity-core --host 92.205.130.254 +``` + +## Adding a repo contract + +Create `/e2e/e2e.yml`: + +```yaml +name: my-repo +compose_file: docker-compose.dev.yml +health_checks: + - name: api + url: http://localhost:8080 + timeout: 120 +test_command: uv run python -m pytest e2e/tests/ -v +timeout: 300 +cleanup: always +``` + +Run: `validate run ~/my-repo` + +## Self-smoke (sand-boxer repo) + +```bash +validate run ~/sand-boxer +``` + +Uses `e2e/docker-compose.smoke.yml` (nginx on `127.0.0.1:18080`). + +## Troubleshooting + +**`sandboxer not found`:** Install sand-boxer CLI. + +**Health check timeout:** SSH to host and `curl` the URL from inside the sandbox dir. + +**Stale sandboxes:** `sandboxer inspect stale` / `sandboxer reap-stale --apply` + +## Migration from the-custodian + +| Legacy | Replacement | +|--------|-------------| +| `make e2e REPO=` | `validate run ~/REPO` (SAND-WP-0004 shim pending) | +| `python -m e2e_framework` | `validate run` | +| Embedded SSH provision | `sandboxer create` (sand-boxer) | \ No newline at end of file diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000..966e03c --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,41 @@ +[build-system] +requires = ["hatchling"] +build-backend = "hatchling.build" + +[project] +name = "wise-validator" +version = "0.0.0" +description = "Cross-repo use-case validation meta-framework for Coulomb" +readme = "README.md" +requires-python = ">=3.11" +license = "MIT-0" +dependencies = [ + "typer>=0.12", + "pydantic>=2.0", + "pyyaml>=6.0", +] + +[project.scripts] +validate = "wisevalidator.cli:app" + +[tool.hatch.build.targets.wheel] +packages = ["src/wisevalidator"] + +[tool.pytest.ini_options] +testpaths = ["tests"] +pythonpath = ["src"] +addopts = "-ra --strict-markers --strict-config" + +[tool.ruff] +src = ["src", "tests"] +target-version = "py311" +line-length = 100 + +[tool.ruff.lint] +select = ["E", "F", "I", "UP", "B", "SIM"] + +[dependency-groups] +dev = [ + "pytest>=8.0", + "ruff>=0.6", +] \ No newline at end of file diff --git a/registry/capabilities/validation.compose-e2e.md b/registry/capabilities/validation.compose-e2e.md new file mode 100644 index 0000000..dc70c02 --- /dev/null +++ b/registry/capabilities/validation.compose-e2e.md @@ -0,0 +1,42 @@ +--- +id: capability.validation.compose-e2e +name: Compose E2E Validation +summary: Cross-repo use-case validation — health polling, test execution, and State Hub result reporting on sand-boxer-established compose environments. +owner: wise-validator +status: draft +domain: infotech +tags: [validation, e2e, health, compose, cross-repo, use-case] + +maturity: + discovery: + current: D4 + target: D6 + confidence: high + rationale: > + Charter (INTENT.md) and sand-boxer integration contract define scope. + Extraction from the-custodian/e2e-framework completes the validation half. + availability: + current: A2 + target: A5 + confidence: medium + rationale: > + validate run CLI v0 and sand-boxer profile.compose-e2e integration land + in SAND-WP-0003. the-custodian shim deferred to SAND-WP-0004. + +external_evidence: + completeness: + level: C2 + name: Partial + confidence: medium + basis: scope_vs_intent_and_consumer_expectations + satisfied_expectations: + - e2e.yml v1 contract parsing + - health-wait and test_command on ready sandbox + - State Hub e2e_result events + - sand-boxer create/destroy without embedded provision + broken_expectations: + - the-custodian make e2e shim not yet migrated + - activity-core scheduled validation not wired + out_of_scope_expectations: + - sandbox provision (sand-boxer) + - agent harness (glas-harness) \ No newline at end of file diff --git a/registry/indexes/capabilities.yaml b/registry/indexes/capabilities.yaml index f944e47..6356b6c 100644 --- a/registry/indexes/capabilities.yaml +++ b/registry/indexes/capabilities.yaml @@ -1,4 +1,6 @@ version: 1 -updated: '2026-06-16' -domain: helix_forge -capabilities: [] +updated: '2026-06-23' +domain: infotech +capabilities: + - id: capability.validation.compose-e2e + path: registry/capabilities/validation.compose-e2e.md diff --git a/scripts/smoke-validate-run.sh b/scripts/smoke-validate-run.sh new file mode 100755 index 0000000..cfbb58b --- /dev/null +++ b/scripts/smoke-validate-run.sh @@ -0,0 +1,17 @@ +#!/usr/bin/env bash +# Remote smoke: validate run against sand-boxer self-deploy contract (SAND-WP-0003-T09). +set -euo pipefail + +HOST="${SANDBOXER_HOST:-coulombcore}" +REPO="${SMOKE_REPO:-$(cd "$(dirname "$0")/.." && pwd)/../sand-boxer}" +export SANDBOXER_COMPOSE_CMD="${SANDBOXER_COMPOSE_CMD:-podman-compose}" + +if [[ ! -d "${REPO}/e2e/e2e.yml" && ! -f "${REPO}/e2e/e2e.yml" ]]; then + echo "FAIL: sand-boxer repo not found at ${REPO}" >&2 + exit 1 +fi + +echo "==> validate run host=${HOST} repo=${REPO}" +validate run "${REPO}" --host "${HOST}" --no-report + +echo "==> PASS" \ No newline at end of file diff --git a/src/wisevalidator/__init__.py b/src/wisevalidator/__init__.py new file mode 100644 index 0000000..91e40fc --- /dev/null +++ b/src/wisevalidator/__init__.py @@ -0,0 +1,3 @@ +"""wise-validator — cross-repo use-case validation for Coulomb.""" + +__version__ = "0.0.0" \ No newline at end of file diff --git a/src/wisevalidator/cli.py b/src/wisevalidator/cli.py new file mode 100644 index 0000000..5be58d8 --- /dev/null +++ b/src/wisevalidator/cli.py @@ -0,0 +1,70 @@ +"""wise-validator CLI — validation orchestration surface (v0).""" + +from __future__ import annotations + +import os +from pathlib import Path +from typing import Annotated + +import typer + +from wisevalidator import __version__ +from wisevalidator.reporter import report +from wisevalidator.runner import run_validation + +app = typer.Typer( + name="validate", + help="Cross-repo use-case validation for Coulomb.", + no_args_is_help=True, +) + + +@app.callback() +def main() -> None: + """validate CLI root.""" + + +@app.command() +def version() -> None: + """Print the installed package version.""" + typer.echo(__version__) + + +@app.command("run") +def validate_run( + repo_path: Annotated[Path, typer.Argument(help="Repo root containing e2e/e2e.yml")], + host: Annotated[ + str | None, + typer.Option(help="Sandbox host override. Env: SANDBOXER_HOST"), + ] = None, + profile: Annotated[str, typer.Option(help="sand-boxer profile id")] = "profile.compose-e2e", + keep: Annotated[bool, typer.Option(help="Keep sandbox after run (skip destroy)")] = False, + workstream_id: Annotated[ + str | None, typer.Option(help="State Hub workstream id for progress event") + ] = None, + no_report: Annotated[bool, typer.Option(help="Skip posting results to State Hub")] = False, +) -> None: + """Run e2e validation for a repo (sand-boxer + health + test).""" + resolved_host = host or os.environ.get("SANDBOXER_HOST") or None + repo = repo_path.expanduser().resolve() + if not repo.exists(): + typer.echo(f"ERROR: repo path does not exist: {repo}", err=True) + raise typer.Exit(code=1) + + try: + result = run_validation(repo, host=resolved_host, keep=keep, profile=profile) + except FileNotFoundError as exc: + typer.echo(f"ERROR: {exc}", err=True) + raise typer.Exit(code=1) from exc + except RuntimeError as exc: + typer.echo(f"ERROR: {exc}", err=True) + raise typer.Exit(code=1) from exc + + if not no_report: + report(result, workstream_id=workstream_id) + + raise typer.Exit(code=0 if result.passed else 1) + + +if __name__ == "__main__": + app() \ No newline at end of file diff --git a/src/wisevalidator/models.py b/src/wisevalidator/models.py new file mode 100644 index 0000000..a48a73c --- /dev/null +++ b/src/wisevalidator/models.py @@ -0,0 +1,17 @@ +"""Validation run result models.""" + +from __future__ import annotations + +from dataclasses import dataclass, field + + +@dataclass +class RunResult: + sandbox_id: str + repo: str + passed: bool + exit_code: int + duration_s: float + output: str + error: str = "" + health_outcomes: list[dict[str, object]] = field(default_factory=list) \ No newline at end of file diff --git a/src/wisevalidator/reporter.py b/src/wisevalidator/reporter.py new file mode 100644 index 0000000..46822d6 --- /dev/null +++ b/src/wisevalidator/reporter.py @@ -0,0 +1,53 @@ +"""Push validation run results to State Hub as progress events.""" + +from __future__ import annotations + +import json +import os +import urllib.error +import urllib.request + +from wisevalidator.models import RunResult + +STATE_HUB_URL = os.environ.get("STATE_HUB_URL", "http://127.0.0.1:8000") + + +def report(result: RunResult, workstream_id: str | None = None) -> bool: + """POST result to state-hub /progress/. Returns True on success.""" + body = { + "event_type": "e2e_result", + "repo": result.repo, + "sandbox_id": result.sandbox_id, + "passed": result.passed, + "exit_code": result.exit_code, + "duration_s": round(result.duration_s, 1), + } + if result.error: + body["error"] = result.error + if result.health_outcomes: + body["health_outcomes"] = result.health_outcomes + + payload = { + "summary": ( + f"E2E {'PASSED' if result.passed else 'FAILED'}: {result.repo} " + f"(sandbox={result.sandbox_id}, {result.duration_s:.0f}s)" + ), + "details": json.dumps(body), + "event_type": "e2e_result", + } + if workstream_id: + payload["workstream_id"] = workstream_id + + try: + req = urllib.request.Request( + f"{STATE_HUB_URL}/progress/", + data=json.dumps(payload).encode(), + headers={"Content-Type": "application/json"}, + method="POST", + ) + with urllib.request.urlopen(req, timeout=10) as resp: + print(f"[reporter] progress event recorded (status={resp.status})") + return True + except urllib.error.URLError as exc: + print(f"[reporter] WARNING: could not reach state-hub: {exc}") + return False \ No newline at end of file diff --git a/src/wisevalidator/runner.py b/src/wisevalidator/runner.py new file mode 100644 index 0000000..70df337 --- /dev/null +++ b/src/wisevalidator/runner.py @@ -0,0 +1,113 @@ +"""Validation lifecycle: sandbox → health-wait → test → cleanup → result.""" + +from __future__ import annotations + +import time +from pathlib import Path + +from wisevalidator.models import RunResult +from wisevalidator.sandbox_client import SandboxHandle, create_sandbox, destroy_sandbox +from wisevalidator.schema import E2EConfig +from wisevalidator.ssh import SSHSession + + +def _should_cleanup(config: E2EConfig, *, keep: bool, passed: bool) -> bool: + if keep: + return False + if config.cleanup == "never": + return False + if config.cleanup == "on_success": + return passed + return True + + +def run_validation( + repo_path: Path, + *, + host: str | None = None, + keep: bool = False, + profile: str = "profile.compose-e2e", +) -> RunResult: + config = E2EConfig.load(repo_path) + started = time.time() + output_lines: list[str] = [] + sandbox: SandboxHandle | None = None + passed = False + exit_code = -1 + error = "" + + def log(msg: str) -> None: + print(msg) + output_lines.append(msg) + + log(f"\n{'=' * 60}") + log(f"Validation run: {config.name}") + log(f"{'=' * 60}\n") + + health_outcomes: list[dict[str, object]] = [] + + try: + log("[runner] requesting sandbox via sand-boxer") + sandbox = create_sandbox(repo_path, profile=profile, host=host) + log(f"[runner] sandbox ready: {sandbox.sandbox_id} on {sandbox.host}") + + ssh = SSHSession.from_reachability( + ssh=sandbox.ssh, + host=sandbox.host, + remote_dir=sandbox.remote_dir, + ) + + for hc in config.health_checks: + ok = ssh.wait_for_url(hc.url, timeout=hc.timeout) + health_outcomes.append({"name": hc.name, "url": hc.url, "passed": ok}) + if not ok: + raise RuntimeError(f"Health check failed: {hc.name} ({hc.url})") + + log(f"\n[runner] running: {config.test_command}") + test_cmd = f"cd {ssh.remote_dir} && {config.test_command}" + exit_code, test_out = ssh.run(test_cmd, timeout=config.timeout) + output_lines.append(test_out) + if test_out.strip(): + print(test_out) + + passed = exit_code == 0 + duration = time.time() - started + log(f"\n[runner] {'PASSED' if passed else 'FAILED'} (exit={exit_code}, {duration:.1f}s)") + + return RunResult( + sandbox_id=sandbox.sandbox_id, + repo=config.name, + passed=passed, + exit_code=exit_code, + duration_s=duration, + output="\n".join(output_lines), + health_outcomes=health_outcomes, + ) + + except Exception as exc: + duration = time.time() - started + error = str(exc) + log(f"\n[runner] ERROR: {exc}") + return RunResult( + sandbox_id=sandbox.sandbox_id if sandbox else "none", + repo=config.name, + passed=False, + exit_code=exit_code, + duration_s=duration, + output="\n".join(output_lines), + error=error, + health_outcomes=health_outcomes, + ) + + finally: + if sandbox and _should_cleanup(config, keep=keep, passed=passed): + log(f"[runner] destroying sandbox {sandbox.sandbox_id}") + try: + destroy_sandbox(sandbox.sandbox_id) + except Exception as exc: + log(f"[runner] WARNING: destroy failed: {exc}") + elif sandbox: + log( + f"[runner] skipping cleanup (keep={keep}, cleanup={config.cleanup}, " + f"passed={passed})" + ) \ No newline at end of file diff --git a/src/wisevalidator/sandbox_client.py b/src/wisevalidator/sandbox_client.py new file mode 100644 index 0000000..0c9e0dd --- /dev/null +++ b/src/wisevalidator/sandbox_client.py @@ -0,0 +1,97 @@ +"""sand-boxer CLI client — provision and teardown only.""" + +from __future__ import annotations + +import json +import os +import shutil +import subprocess +from dataclasses import dataclass +from pathlib import Path +from typing import Any + + +@dataclass +class SandboxHandle: + sandbox_id: str + host: str + ssh: str | None + remote_dir: str + compose_project: str | None + raw: dict[str, Any] + + +def _sandboxer_bin() -> str: + return os.environ.get("SANDBOXER_BIN", "sandboxer") + + +def _run_sandboxer(args: list[str]) -> dict[str, Any]: + cmd = [_sandboxer_bin(), *args] + try: + result = subprocess.run(cmd, capture_output=True, text=True, check=False) + except FileNotFoundError as exc: + raise RuntimeError( + f"{_sandboxer_bin()} not found on PATH; install sand-boxer (make install)" + ) from exc + + if result.returncode != 0: + detail = result.stderr.strip() or result.stdout.strip() or "unknown error" + raise RuntimeError(f"sandboxer {' '.join(args)} failed: {detail}") + + try: + return json.loads(result.stdout) + except json.JSONDecodeError as exc: + raise RuntimeError(f"sandboxer returned non-JSON output: {result.stdout[:200]}") from exc + + +def create_sandbox( + repo_path: Path, + *, + profile: str = "profile.compose-e2e", + host: str | None = None, + run_id: str | None = None, +) -> SandboxHandle: + if not shutil.which(_sandboxer_bin()): + raise RuntimeError( + f"{_sandboxer_bin()} not found on PATH; install sand-boxer (make install)" + ) + + args = [ + "create", + "--profile", + profile, + "--input", + f"repo={repo_path}", + "--actor", + "atm", + "--project", + "wise-validator", + ] + if host: + args += ["--host", host] + elif os.environ.get("SANDBOXER_HOST"): + args += ["--host", os.environ["SANDBOXER_HOST"]] + + payload = _run_sandboxer(args) + if payload.get("state") != "ready": + raise RuntimeError( + f"sandbox not ready: state={payload.get('state')} error={payload.get('error')}" + ) + + reach = payload.get("reachability") or {} + sandbox_id = payload.get("sandbox_id") + if not sandbox_id: + raise RuntimeError("sandboxer create response missing sandbox_id") + + return SandboxHandle( + sandbox_id=sandbox_id, + host=reach.get("host") or payload.get("host") or host or "", + ssh=reach.get("ssh"), + remote_dir=reach.get("remote_dir") or "", + compose_project=reach.get("compose_project"), + raw=payload, + ) + + +def destroy_sandbox(sandbox_id: str) -> dict[str, Any]: + return _run_sandboxer(["destroy", sandbox_id]) \ No newline at end of file diff --git a/src/wisevalidator/schema.py b/src/wisevalidator/schema.py new file mode 100644 index 0000000..a914f2e --- /dev/null +++ b/src/wisevalidator/schema.py @@ -0,0 +1,63 @@ +"""Parse and validate e2e.yml — the per-repo test contract.""" + +from __future__ import annotations + +from dataclasses import dataclass, field +from pathlib import Path +from typing import Literal + +import yaml + + +@dataclass +class HealthCheck: + name: str + url: str + timeout: int = 120 + + +@dataclass +class E2EConfig: + name: str + compose_file: str + test_command: str + health_checks: list[HealthCheck] = field(default_factory=list) + timeout: int = 300 + cleanup: Literal["always", "on_success", "never"] = "always" + env: dict[str, str] = field(default_factory=dict) + + @classmethod + def load(cls, repo_root: Path) -> E2EConfig: + config_path = repo_root / "e2e" / "e2e.yml" + if not config_path.exists(): + raise FileNotFoundError(f"No e2e.yml found at {config_path}") + + raw = yaml.safe_load(config_path.read_text()) or {} + if not isinstance(raw, dict): + raise ValueError(f"Invalid e2e.yml at {config_path}: expected mapping") + + health_checks = [ + HealthCheck( + name=hc.get("name", hc["url"]), + url=hc["url"], + timeout=int(hc.get("timeout", 120)), + ) + for hc in raw.get("health_checks", []) + ] + + env: dict[str, str] = {} + for item in raw.get("env", []): + if isinstance(item, dict) and "key" in item: + env[item["key"]] = str(item.get("value", "")) + elif isinstance(item, dict): + env.update({str(k): str(v) for k, v in item.items()}) + + return cls( + name=raw["name"], + compose_file=raw["compose_file"], + test_command=raw["test_command"], + health_checks=health_checks, + timeout=int(raw.get("timeout", 300)), + cleanup=raw.get("cleanup", "always"), + env=env, + ) \ No newline at end of file diff --git a/src/wisevalidator/ssh.py b/src/wisevalidator/ssh.py new file mode 100644 index 0000000..4c6aa52 --- /dev/null +++ b/src/wisevalidator/ssh.py @@ -0,0 +1,87 @@ +"""SSH session for health polling and test execution on a ready sandbox.""" + +from __future__ import annotations + +import os +import subprocess +import time +from dataclasses import dataclass + + +@dataclass +class SSHSession: + host: str + user: str | None = None + key: str | None = None + remote_dir: str = "" + connect_timeout: int = 15 + + @property + def destination(self) -> str: + if self.user: + return f"{self.user}@{self.host}" + return self.host + + @classmethod + def from_reachability( + cls, + *, + ssh: str | None, + host: str | None, + remote_dir: str | None, + ) -> SSHSession: + parsed_host = host or "" + parsed_user: str | None = None + if ssh: + if "@" in ssh: + parsed_user, parsed_host = ssh.split("@", 1) + else: + parsed_host = ssh + if not parsed_host: + raise ValueError("reachability missing host/ssh target") + + return cls( + host=parsed_host, + user=parsed_user or os.environ.get("SANDBOXER_SSH_USER"), + key=os.environ.get("SANDBOXER_SSH_KEY"), + remote_dir=remote_dir or "", + ) + + def ssh_base(self) -> list[str]: + args = [ + "ssh", + "-o", + "StrictHostKeyChecking=no", + "-o", + "BatchMode=yes", + "-o", + f"ConnectTimeout={self.connect_timeout}", + ] + if self.key: + args += ["-i", self.key] + args.append(self.destination) + return args + + def run(self, cmd: str, *, timeout: int = 60) -> tuple[int, str]: + result = subprocess.run( + self.ssh_base() + [cmd], + capture_output=True, + text=True, + timeout=timeout, + ) + return result.returncode, result.stdout + result.stderr + + def wait_for_url(self, url: str, timeout: int = 120, interval: int = 5) -> bool: + print(f"[validator] waiting for {url} (timeout={timeout}s)") + deadline = time.time() + timeout + while time.time() < deadline: + rc, _ = self.run( + f"curl -sf --max-time 5 {url} > /dev/null 2>&1", + timeout=15, + ) + if rc == 0: + print(f"[validator] {url} is up") + return True + time.sleep(interval) + print(f"[validator] TIMEOUT waiting for {url}") + return False \ No newline at end of file diff --git a/tests/fixtures/minimal_repo/docker-compose.yml b/tests/fixtures/minimal_repo/docker-compose.yml new file mode 100644 index 0000000..1a5f982 --- /dev/null +++ b/tests/fixtures/minimal_repo/docker-compose.yml @@ -0,0 +1,5 @@ +services: + web: + image: docker.io/library/nginx:alpine + ports: + - "127.0.0.1:8080:80" \ No newline at end of file diff --git a/tests/fixtures/minimal_repo/e2e/e2e.yml b/tests/fixtures/minimal_repo/e2e/e2e.yml new file mode 100644 index 0000000..c4c97cd --- /dev/null +++ b/tests/fixtures/minimal_repo/e2e/e2e.yml @@ -0,0 +1,9 @@ +name: fixture-repo +compose_file: docker-compose.yml +health_checks: + - name: web + url: http://localhost:8080 + timeout: 30 +test_command: "echo ok" +timeout: 60 +cleanup: always \ No newline at end of file diff --git a/tests/test_reporter.py b/tests/test_reporter.py new file mode 100644 index 0000000..b6b95d6 --- /dev/null +++ b/tests/test_reporter.py @@ -0,0 +1,37 @@ +"""State Hub reporter tests.""" + +import json +from unittest.mock import MagicMock, patch + +from wisevalidator.models import RunResult +from wisevalidator.reporter import report + + +def test_report_posts_e2e_result() -> None: + result = RunResult( + sandbox_id="abc12345", + repo="fixture-repo", + passed=True, + exit_code=0, + duration_s=12.3, + output="ok", + ) + captured: dict = {} + + def fake_urlopen(req, timeout=10): + captured["url"] = req.full_url + captured["body"] = json.loads(req.data.decode()) + resp = MagicMock() + resp.status = 201 + resp.__enter__ = MagicMock(return_value=resp) + resp.__exit__ = MagicMock(return_value=False) + return resp + + with patch("wisevalidator.reporter.urllib.request.urlopen", fake_urlopen): + ok = report(result, workstream_id="ws-uuid") + + assert ok is True + assert captured["url"].endswith("/progress/") + assert captured["body"]["event_type"] == "e2e_result" + assert "PASSED" in captured["body"]["summary"] + assert captured["body"]["workstream_id"] == "ws-uuid" \ No newline at end of file diff --git a/tests/test_runner.py b/tests/test_runner.py new file mode 100644 index 0000000..5460c79 --- /dev/null +++ b/tests/test_runner.py @@ -0,0 +1,104 @@ +"""Validation runner tests with mocked sand-boxer and SSH.""" + +from __future__ import annotations + +from pathlib import Path +from unittest.mock import patch + +from wisevalidator.runner import run_validation +from wisevalidator.sandbox_client import SandboxHandle + +FIXTURE_REPO = Path(__file__).parent / "fixtures" / "minimal_repo" + + +def test_run_validation_success() -> None: + handle = SandboxHandle( + sandbox_id="run12345", + host="coulombcore", + ssh="root@coulombcore", + remote_dir="/tmp/sandboxer/run12345", + compose_project="sbx-e2e-run12345", + raw={}, + ) + + class FakeSSH: + remote_dir = handle.remote_dir + + def wait_for_url(self, url: str, timeout: int = 120, interval: int = 5) -> bool: + return True + + def run(self, cmd: str, *, timeout: int = 60) -> tuple[int, str]: + return 0, "test output\n" + + with ( + patch("wisevalidator.runner.create_sandbox", return_value=handle), + patch("wisevalidator.runner.destroy_sandbox") as destroy, + patch("wisevalidator.runner.SSHSession.from_reachability", return_value=FakeSSH()), + ): + result = run_validation(FIXTURE_REPO, host="coulombcore") + + assert result.passed is True + assert result.exit_code == 0 + assert result.sandbox_id == "run12345" + destroy.assert_called_once_with("run12345") + + +def test_run_validation_keep_skips_destroy() -> None: + handle = SandboxHandle( + sandbox_id="keep1234", + host="coulombcore", + ssh="coulombcore", + remote_dir="/tmp/sandboxer/keep1234", + compose_project="sbx-e2e-keep1234", + raw={}, + ) + + class FakeSSH: + remote_dir = handle.remote_dir + + def wait_for_url(self, url: str, timeout: int = 120, interval: int = 5) -> bool: + return True + + def run(self, cmd: str, *, timeout: int = 60) -> tuple[int, str]: + return 0, "" + + with ( + patch("wisevalidator.runner.create_sandbox", return_value=handle), + patch("wisevalidator.runner.destroy_sandbox") as destroy, + patch("wisevalidator.runner.SSHSession.from_reachability", return_value=FakeSSH()), + ): + result = run_validation(FIXTURE_REPO, keep=True) + + assert result.passed is True + destroy.assert_not_called() + + +def test_run_validation_health_failure() -> None: + handle = SandboxHandle( + sandbox_id="fail1234", + host="coulombcore", + ssh="coulombcore", + remote_dir="/tmp/sandboxer/fail1234", + compose_project=None, + raw={}, + ) + + class FakeSSH: + remote_dir = handle.remote_dir + + def wait_for_url(self, url: str, timeout: int = 120, interval: int = 5) -> bool: + return False + + def run(self, cmd: str, *, timeout: int = 60) -> tuple[int, str]: + return 0, "" + + with ( + patch("wisevalidator.runner.create_sandbox", return_value=handle), + patch("wisevalidator.runner.destroy_sandbox") as destroy, + patch("wisevalidator.runner.SSHSession.from_reachability", return_value=FakeSSH()), + ): + result = run_validation(FIXTURE_REPO) + + assert result.passed is False + assert "Health check failed" in result.error + destroy.assert_called_once_with("fail1234") \ No newline at end of file diff --git a/tests/test_sandbox_client.py b/tests/test_sandbox_client.py new file mode 100644 index 0000000..3acbe87 --- /dev/null +++ b/tests/test_sandbox_client.py @@ -0,0 +1,59 @@ +"""sand-boxer CLI client tests.""" + +import json +from unittest.mock import MagicMock, patch + +import pytest + +from wisevalidator.sandbox_client import create_sandbox, destroy_sandbox + + +def test_create_sandbox_parses_ready_status(tmp_path) -> None: + repo = tmp_path / "repo" + repo.mkdir() + payload = { + "sandbox_id": "cli12345", + "state": "ready", + "host": "coulombcore", + "reachability": { + "ssh": "root@coulombcore", + "host": "coulombcore", + "remote_dir": "/tmp/sandboxer/cli12345", + "compose_project": "sbx-e2e-cli12345", + }, + } + + proc = MagicMock(returncode=0, stdout=json.dumps(payload), stderr="") + + with ( + patch("wisevalidator.sandbox_client.shutil.which", return_value="/bin/sandboxer"), + patch("wisevalidator.sandbox_client.subprocess.run", return_value=proc), + ): + handle = create_sandbox(repo) + + assert handle.sandbox_id == "cli12345" + assert handle.remote_dir == "/tmp/sandboxer/cli12345" + + +def test_create_sandbox_not_ready_raises(tmp_path) -> None: + repo = tmp_path / "repo" + repo.mkdir() + payload = {"sandbox_id": "bad12345", "state": "failed", "error": "compose up failed"} + proc = MagicMock(returncode=0, stdout=json.dumps(payload), stderr="") + + with ( + patch("wisevalidator.sandbox_client.shutil.which", return_value="/bin/sandboxer"), + patch("wisevalidator.sandbox_client.subprocess.run", return_value=proc), + pytest.raises(RuntimeError, match="sandbox not ready"), + ): + create_sandbox(repo) + + +def test_destroy_sandbox() -> None: + payload = {"sandbox_id": "cli12345", "state": "destroyed"} + proc = MagicMock(returncode=0, stdout=json.dumps(payload), stderr="") + + with patch("wisevalidator.sandbox_client.subprocess.run", return_value=proc): + out = destroy_sandbox("cli12345") + + assert out["state"] == "destroyed" \ No newline at end of file diff --git a/tests/test_schema.py b/tests/test_schema.py new file mode 100644 index 0000000..db652a6 --- /dev/null +++ b/tests/test_schema.py @@ -0,0 +1,25 @@ +"""E2EConfig schema tests.""" + +from pathlib import Path + +import pytest + +from wisevalidator.schema import E2EConfig, HealthCheck + +FIXTURE_REPO = Path(__file__).parent / "fixtures" / "minimal_repo" + + +def test_load_minimal_fixture() -> None: + config = E2EConfig.load(FIXTURE_REPO) + assert config.name == "fixture-repo" + assert config.compose_file == "docker-compose.yml" + assert config.test_command == "echo ok" + assert len(config.health_checks) == 1 + assert config.health_checks[0] == HealthCheck( + name="web", url="http://localhost:8080", timeout=30 + ) + + +def test_missing_e2e_yml(tmp_path: Path) -> None: + with pytest.raises(FileNotFoundError, match="No e2e.yml"): + E2EConfig.load(tmp_path) \ No newline at end of file diff --git a/uv.lock b/uv.lock new file mode 100644 index 0000000..4c7fefa --- /dev/null +++ b/uv.lock @@ -0,0 +1,386 @@ +version = 1 +requires-python = ">=3.11" + +[[package]] +name = "annotated-doc" +version = "0.0.4" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/57/ba/046ceea27344560984e26a590f90bc7f4a75b06701f653222458922b558c/annotated_doc-0.0.4.tar.gz", hash = "sha256:fbcda96e87e9c92ad167c2e53839e57503ecfda18804ea28102353485033faa4", size = 7288 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/1e/d3/26bf1008eb3d2daa8ef4cacc7f3bfdc11818d111f7e2d0201bc6e3b49d45/annotated_doc-0.0.4-py3-none-any.whl", hash = "sha256:571ac1dc6991c450b25a9c2d84a3705e2ae7a53467b5d111c24fa8baabbed320", size = 5303 }, +] + +[[package]] +name = "annotated-types" +version = "0.7.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/ee/67/531ea369ba64dcff5ec9c3402f9f51bf748cec26dde048a2f973a4eea7f5/annotated_types-0.7.0.tar.gz", hash = "sha256:aff07c09a53a08bc8cfccb9c85b05f1aa9a2a6f23728d790723543408344ce89", size = 16081 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/78/b6/6307fbef88d9b5ee7421e68d78a9f162e0da4900bc5f5793f6d3d0e34fb8/annotated_types-0.7.0-py3-none-any.whl", hash = "sha256:1f02e8b43a8fbbc3f3e0d4f0f4bfc8131bcb4eebe8849b8e5c773f3a1c582a53", size = 13643 }, +] + +[[package]] +name = "colorama" +version = "0.4.6" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335 }, +] + +[[package]] +name = "iniconfig" +version = "2.3.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/72/34/14ca021ce8e5dfedc35312d08ba8bf51fdd999c576889fc2c24cb97f4f10/iniconfig-2.3.0.tar.gz", hash = "sha256:c76315c77db068650d49c5b56314774a7804df16fee4402c1f19d6d15d8c4730", size = 20503 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/cb/b1/3846dd7f199d53cb17f49cba7e651e9ce294d8497c8c150530ed11865bb8/iniconfig-2.3.0-py3-none-any.whl", hash = "sha256:f631c04d2c48c52b84d0d0549c99ff3859c98df65b3101406327ecc7d53fbf12", size = 7484 }, +] + +[[package]] +name = "markdown-it-py" +version = "4.2.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "mdurl" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/06/ff/7841249c247aa650a76b9ee4bbaeae59370dc8bfd2f6c01f3630c35eb134/markdown_it_py-4.2.0.tar.gz", hash = "sha256:04a21681d6fbb623de53f6f364d352309d4094dd4194040a10fd51833e418d49", size = 82454 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/b3/81/4da04ced5a082363ecfa159c010d200ecbd959ae410c10c0264a38cac0f5/markdown_it_py-4.2.0-py3-none-any.whl", hash = "sha256:9f7ebbcd14fe59494226453aed97c1070d83f8d24b6fc3a3bcf9a38092641c4a", size = 91687 }, +] + +[[package]] +name = "mdurl" +version = "0.1.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d6/54/cfe61301667036ec958cb99bd3efefba235e65cdeb9c84d24a8293ba1d90/mdurl-0.1.2.tar.gz", hash = "sha256:bb413d29f5eea38f31dd4754dd7377d4465116fb207585f97bf925588687c1ba", size = 8729 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/b3/38/89ba8ad64ae25be8de66a6d463314cf1eb366222074cfda9ee839c56a4b4/mdurl-0.1.2-py3-none-any.whl", hash = "sha256:84008a41e51615a49fc9966191ff91509e3c40b939176e643fd50a5c2196b8f8", size = 9979 }, +] + +[[package]] +name = "packaging" +version = "26.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d7/f1/e7a6dd94a8d4a5626c03e4e99c87f241ba9e350cd9e6d75123f992427270/packaging-26.2.tar.gz", hash = "sha256:ff452ff5a3e828ce110190feff1178bb1f2ea2281fa2075aadb987c2fb221661", size = 228134 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/df/b2/87e62e8c3e2f4b32e5fe99e0b86d576da1312593b39f47d8ceef365e95ed/packaging-26.2-py3-none-any.whl", hash = "sha256:5fc45236b9446107ff2415ce77c807cee2862cb6fac22b8a73826d0693b0980e", size = 100195 }, +] + +[[package]] +name = "pluggy" +version = "1.6.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/f9/e2/3e91f31a7d2b083fe6ef3fa267035b518369d9511ffab804f839851d2779/pluggy-1.6.0.tar.gz", hash = "sha256:7dcc130b76258d33b90f61b658791dede3486c3e6bfb003ee5c9bfb396dd22f3", size = 69412 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/54/20/4d324d65cc6d9205fabedc306948156824eb9f0ee1633355a8f7ec5c66bf/pluggy-1.6.0-py3-none-any.whl", hash = "sha256:e920276dd6813095e9377c0bc5566d94c932c33b27a3e3945d8389c374dd4746", size = 20538 }, +] + +[[package]] +name = "pydantic" +version = "2.13.4" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "annotated-types" }, + { name = "pydantic-core" }, + { name = "typing-extensions" }, + { name = "typing-inspection" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/18/a5/b60d21ac674192f8ab0ba4e9fd860690f9b4a6e51ca5df118733b487d8d6/pydantic-2.13.4.tar.gz", hash = "sha256:c40756b57adaa8b1efeeced5c196f3f3b7c435f90e84ea7f443901bec8099ef6", size = 844775 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/fd/7b/122376b1fd3c62c1ed9dc80c931ace4844b3c55407b6fb2d199377c9736f/pydantic-2.13.4-py3-none-any.whl", hash = "sha256:45a282cde31d808236fd7ea9d919b128653c8b38b393d1c4ab335c62924d9aba", size = 472262 }, +] + +[[package]] +name = "pydantic-core" +version = "2.46.4" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "typing-extensions" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/9d/56/921726b776ace8d8f5db44c4ef961006580d91dc52b803c489fafd1aa249/pydantic_core-2.46.4.tar.gz", hash = "sha256:62f875393d7f270851f20523dd2e29f082bcc82292d66db2b64ea71f64b6e1c1", size = 471464 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/5c/fa/6d7708d2cfc1a832acb6aeb0cd16e801902df8a0f583bb3b4b527fde022e/pydantic_core-2.46.4-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:0e96592440881c74a213e5ad528e2b24d3d4f940de2766bed9010ab1d9e51594", size = 2111872 }, + { url = "https://files.pythonhosted.org/packages/ae/6f/aa064a3e74b5745afbdf250594f38e7ead05e2d651bcb35994b9417a0d4d/pydantic_core-2.46.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e0d65b8c354be7fb5f720c3caa8bc940bc2d20ce749c8e06135f07f8ed95dd7c", size = 1948255 }, + { url = "https://files.pythonhosted.org/packages/43/3a/41114a9f7569b84b4d84e7a018c57c56347dac30c0d4a872946ec4e36c46/pydantic_core-2.46.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7bfb192b3f4b9e8a89b6277b6ce787564f62cfd272055f6e685726b111dc7826", size = 1972827 }, + { url = "https://files.pythonhosted.org/packages/ef/25/1ab42e8048fe551934d9884e8d64daa7e990ad386f310a15981aeb6a5b08/pydantic_core-2.46.4-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:9037063db01f09b09e237c282b6792bd4da634b5402c4e7f0c61effed7701a04", size = 2041051 }, + { url = "https://files.pythonhosted.org/packages/94/c2/1a934597ddf08da410385b3b7aae91956a5a76c635effef456074fad7e88/pydantic_core-2.46.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fc010ab034c8c7452522748bf937df58020d256ccae0874463d1f4d01758af8e", size = 2221314 }, + { url = "https://files.pythonhosted.org/packages/02/6d/9e8ad178c9c4df27ad3c8f25d1fe2a7ab0d2ba0559fad4aee5d3d1f16771/pydantic_core-2.46.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8c5dac79fa1614d1e06ca695109c6105923bd9c7d1d6c918d4e637b7e6b32fd3", size = 2285146 }, + { url = "https://files.pythonhosted.org/packages/80/50/540cd3aeefc041beb111125c4bff779831a2111fc6b15a9138cda277d32c/pydantic_core-2.46.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f9fa868638bf362d3d138ea55829cefb3d5f4b0d7f142234382a15e2485dbec4", size = 2089685 }, + { url = "https://files.pythonhosted.org/packages/6b/a4/b440ad35f05f6a38f89fa0f149accb3f0e02be94ca5e15f3c449a61b4bc9/pydantic_core-2.46.4-cp311-cp311-manylinux_2_31_riscv64.whl", hash = "sha256:17299feefe090f2caa5b8e37222bb5f663e4935a8bfa6931d4102e5df1a9f398", size = 2115420 }, + { url = "https://files.pythonhosted.org/packages/99/61/de4f55db8dfd57bfdfa9a12ec90fe1b57c4f41062f7ca86f08586b3e0ac0/pydantic_core-2.46.4-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:4c63ebc82684aa89d9a3bcbd13d515b3be44250dc68dd3bd81526c1cb31286c3", size = 2165122 }, + { url = "https://files.pythonhosted.org/packages/f7/52/7c529d7bdb2d1068bd52f51fe32572c8301f9a4febf1948f10639f1436f5/pydantic_core-2.46.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:aaa2a54443eff1950ba5ddc6b6ccda0d9c84a364276a62f969bdf2a390650848", size = 2182573 }, + { url = "https://files.pythonhosted.org/packages/37/b3/7c40325848ba78247f2812dcf9c7274e38cd801820ca6dd9fe63bcfb0eb4/pydantic_core-2.46.4-cp311-cp311-musllinux_1_1_armv7l.whl", hash = "sha256:18e5ceec2ab67e6d5f1a9085e5a24c9c4e2ac4545730bfe668680bca05e555f3", size = 2317139 }, + { url = "https://files.pythonhosted.org/packages/d9/37/f913f81a657c865b75da6c0dbed79876073c2a43b5bd9edbe8da785e4d49/pydantic_core-2.46.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:a0f62d0a58f4e7da165457e995725421e0064f2255d8eccebc49f41bbc23b109", size = 2360433 }, + { url = "https://files.pythonhosted.org/packages/c4/67/6acaa1be2567f9256b056d8477158cac7240813956ce86e49deae8e173b4/pydantic_core-2.46.4-cp311-cp311-win32.whl", hash = "sha256:041bde0a48fd37cf71cab1c9d56d3e8625a3793fef1f7dd232b3ff37e978ecda", size = 1985513 }, + { url = "https://files.pythonhosted.org/packages/aa/e6/c505f83dfeda9a2e5c995cfd872949e4d05e12f7feb3dca72f633daefa94/pydantic_core-2.46.4-cp311-cp311-win_amd64.whl", hash = "sha256:6f2eeda33a839975441c86a4119e1383c50b47faf0cbb5176985565c6bb02c33", size = 2071114 }, + { url = "https://files.pythonhosted.org/packages/0f/da/7a263a96d965d9d0df5e8de8a475f33495451117035b09acb110288c381f/pydantic_core-2.46.4-cp311-cp311-win_arm64.whl", hash = "sha256:14f4c5d6db102bd796a627bbb3a17b4cf4574b9ae861d8b7c9a9661c6dd3362d", size = 2044298 }, + { url = "https://files.pythonhosted.org/packages/ce/8c/af022f0af448d7747c5154288d46b5f2bc5f17366eaa0e23e9aa04d59f3b/pydantic_core-2.46.4-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:3245406455a5d98187ec35530fd772b1d799b26667980872c8d4614991e2c4a2", size = 2106158 }, + { url = "https://files.pythonhosted.org/packages/19/95/6195171e385007300f0f5574592e467c568becce2d937a0b6804f218bc49/pydantic_core-2.46.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:962ccbab7b642487b1d8b7df90ef677e03134cf1fd8880bf698649b22a69371f", size = 1951724 }, + { url = "https://files.pythonhosted.org/packages/8e/bc/f47d1ff9cbb1620e1b5b697eef06010035735f07820180e74178226b27b3/pydantic_core-2.46.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8233f2947cf85404441fd7e0085f53b10c93e0ee78611099b5c7237e36aacbf7", size = 1975742 }, + { url = "https://files.pythonhosted.org/packages/5b/11/9b9a5b0306345664a2da6410877af6e8082481b5884b3ddd78d47c6013ce/pydantic_core-2.46.4-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:3a233125ac121aa3ffba9a2b59edfc4a985a76092dc8279586ab4b71390875e7", size = 2052418 }, + { url = "https://files.pythonhosted.org/packages/f1/b7/a65fec226f5d78fc39f4a13c4cc0c768c22b113438f60c14adc9d2865038/pydantic_core-2.46.4-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5b712b53160b79a5850310b912a5ef8e57e56947c8ad690c227f5c9d7e561712", size = 2232274 }, + { url = "https://files.pythonhosted.org/packages/68/f0/92039db98b907ef49269a8271f67db9cb78ae2fc68062ef7e4e77adb5f61/pydantic_core-2.46.4-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9401557acd873c3a7f3eb9383edef8ac4968f9510e340f4808d427e75667e7b4", size = 2309940 }, + { url = "https://files.pythonhosted.org/packages/5f/97/2aab507d3d00ca626e8e57c1eac6a79e4e5fbcc63eb99733ff55d1717f65/pydantic_core-2.46.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:926c9541b14b12b1681dca8a0b75feb510b06c6341b70a8e500c2fdcff837cce", size = 2094516 }, + { url = "https://files.pythonhosted.org/packages/22/37/a8aca44d40d737dde2bc05b3c6c07dff0de07ce6f82e9f3167aeaf4d5dea/pydantic_core-2.46.4-cp312-cp312-manylinux_2_31_riscv64.whl", hash = "sha256:56cb4851bcaf3d117eddcef4fe66afd750a50274b0da8e22be256d10e5611987", size = 2136854 }, + { url = "https://files.pythonhosted.org/packages/24/99/fcef1b79238c06a8cbec70819ac722ba76e02bc8ada9b0fd66eba40da01b/pydantic_core-2.46.4-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:c68fcd102d71ea85c5b2dfac3f4f8476eff42a9e078fd5faefff6d145063536b", size = 2180306 }, + { url = "https://files.pythonhosted.org/packages/ae/6c/fc44000918855b42779d007ae63b0532794739027b2f417321cddbc44f6a/pydantic_core-2.46.4-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:b2f69dec1725e79a012d920df1707de5caf7ed5e08f3be4435e25803efc47458", size = 2190044 }, + { url = "https://files.pythonhosted.org/packages/6b/65/d9cadc9f1920d7a127ad2edba16c1db7916e59719285cd6c94600b0080ba/pydantic_core-2.46.4-cp312-cp312-musllinux_1_1_armv7l.whl", hash = "sha256:8d0820e8192167f80d88d64038e609c31452eeca865b4e1d9950a27a4609b00b", size = 2329133 }, + { url = "https://files.pythonhosted.org/packages/d0/cf/c873d91679f3a30bcf5e7ac280ce5573483e72295307685120d0d5ad3416/pydantic_core-2.46.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:fbdb89b3e1c94a30cc5edfce477c6e6a5dc4d8f84665b455c27582f211a1c72c", size = 2374464 }, + { url = "https://files.pythonhosted.org/packages/47/bd/6f2fc8188f31bf10590f1e98e7b306336161fac930a8c514cd7bd828c7dc/pydantic_core-2.46.4-cp312-cp312-win32.whl", hash = "sha256:9aa768456404a8bf48a4406685ac2bec8e72b62c69313734fa3b73cf33b3a894", size = 1974823 }, + { url = "https://files.pythonhosted.org/packages/40/8c/985c1d41ea1107c2534abd9870e4ed5c8e7669b5c308297835c001e7a1c4/pydantic_core-2.46.4-cp312-cp312-win_amd64.whl", hash = "sha256:e9c26f834c65f5752f3f06cb08cb86a913ceb7274d0db6e267808a708b46bc89", size = 2072919 }, + { url = "https://files.pythonhosted.org/packages/c4/ba/f463d006e0c47373ca7ec5e1a261c59dc01ef4d62b2657af925fb0deee3a/pydantic_core-2.46.4-cp312-cp312-win_arm64.whl", hash = "sha256:4fc73cb559bdb54b1134a706a2802a4cddd27a0633f5abb7e53056268751ac6a", size = 2027604 }, + { url = "https://files.pythonhosted.org/packages/51/a2/5d30b469c5267a17b39dec53208222f76a8d351dfac4af661888c5aee77d/pydantic_core-2.46.4-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:5d5902252db0d3cedf8d4a1bc68f70eeb430f7e4c7104c8c476753519b423008", size = 2106306 }, + { url = "https://files.pythonhosted.org/packages/c1/81/4fa520eaffa8bd7d1525e644cd6d39e7d60b1592bc5b516693c7340b50f1/pydantic_core-2.46.4-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:c94f0688e7b8d0a67abf40e57a7eaaecd17cc9586706a31b76c031f63df052b4", size = 1951906 }, + { url = "https://files.pythonhosted.org/packages/03/d5/fd02da45b659668b05923b17ba3a0100a0a3d5541e3bd8fcc4ecb711309e/pydantic_core-2.46.4-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f027324c56cd5406ca49c124b0db10e56c69064fec039acc571c29020cc87c76", size = 1976802 }, + { url = "https://files.pythonhosted.org/packages/21/f2/95727e1368be3d3ed485eaab7adbd7dda408f33f7a36e8b48e0144002b91/pydantic_core-2.46.4-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:e739fee756ba1010f8bcccb534252e85a35fe45ae92c295a06059ce58b74ccd3", size = 2052446 }, + { url = "https://files.pythonhosted.org/packages/9c/86/5d99feea3f77c7234b8718075b23db11532773c1a0dbd9b9490215dc2eeb/pydantic_core-2.46.4-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9d56801be94b86a9da183e5f3766e6310752b99ff647e38b09a9500d88e46e76", size = 2232757 }, + { url = "https://files.pythonhosted.org/packages/d2/3a/508ac615935ef7588cf6d9e9b91309fdc2da751af865e02a9098de88258c/pydantic_core-2.46.4-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2412e734dcb48da14d4e4006b82b46b74f2518b8a26ee7e58c6844a6cd6d03c4", size = 2309275 }, + { url = "https://files.pythonhosted.org/packages/07/f8/41db9de19d7987d6b04715a02b3b40aea467000275d9d758ffaa31af7d50/pydantic_core-2.46.4-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9551187363ffc0de2a00b2e47c25aeaeb1020b69b668762966df15fc5659dd5a", size = 2094467 }, + { url = "https://files.pythonhosted.org/packages/2c/e2/f35033184cb11d0052daf4416e8e10a502ea2ac006fc4f459aee872727d1/pydantic_core-2.46.4-cp313-cp313-manylinux_2_31_riscv64.whl", hash = "sha256:0186750b482eefa11d7f435892b09c5c606193ef3375bcf94aa00ae6bfb66262", size = 2134417 }, + { url = "https://files.pythonhosted.org/packages/7e/7b/6ceeb1cc90e193862f444ebe373d8fdf613f0a82572dde03fb10734c6c71/pydantic_core-2.46.4-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:5855698a4856556d86e8e6cd8434bc3ac0314ee8e12089ae0e143f64c6256e4e", size = 2179782 }, + { url = "https://files.pythonhosted.org/packages/5a/f2/c8d7773ede6af08036423a00ae0ceffce266c3c52a096c435d68c896083f/pydantic_core-2.46.4-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:cbaf13819775b7f769bf4a1f066cb6df7a28d4480081a589828ef190226881cd", size = 2188782 }, + { url = "https://files.pythonhosted.org/packages/59/31/0c864784e31f09f05cdd87606f08923b9c9e7f6e51dd27f20f62f975ce9f/pydantic_core-2.46.4-cp313-cp313-musllinux_1_1_armv7l.whl", hash = "sha256:633147d34cf4550417f12e2b1a0383973bdf5cdfde212cb09e9a581cf10820be", size = 2328334 }, + { url = "https://files.pythonhosted.org/packages/c2/eb/4f6c8a41efa30baa755590f4141abf3a8c370fab610915733e74134a7270/pydantic_core-2.46.4-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:82cf5301172168103724d49a1444d3378cb20cdee30b116a1bd6031236298a5d", size = 2372986 }, + { url = "https://files.pythonhosted.org/packages/5b/24/b375a480d53113860c299764bfe9f349a3dc9108b3adc0d7f0d786492ebf/pydantic_core-2.46.4-cp313-cp313-win32.whl", hash = "sha256:9fa8ae11da9e2b3126c6426f147e0fba88d96d65921799bb30c6abd1cb2c97fb", size = 1973693 }, + { url = "https://files.pythonhosted.org/packages/7e/e8/cff247591966f2d22ec8c003cd7587e27b7ba7b81ab2fb888e3ab75dc285/pydantic_core-2.46.4-cp313-cp313-win_amd64.whl", hash = "sha256:6b3ace8194b0e5204818c92802dcdca7fc6d88aabbb799d7c795540d9cd6d292", size = 2071819 }, + { url = "https://files.pythonhosted.org/packages/c6/1a/f4aee670d5670e9e148e0c82c7db98d780be566c6e6a97ee8035528ca0b3/pydantic_core-2.46.4-cp313-cp313-win_arm64.whl", hash = "sha256:184c081504d17f1c1066e430e117142b2c77d9448a97f7b65c6ac9fd9aee238d", size = 2027411 }, + { url = "https://files.pythonhosted.org/packages/8d/74/228a26ddad29c6672b805d9fd78e8d251cd04004fa7eed0e622096cd0250/pydantic_core-2.46.4-cp314-cp314-macosx_10_12_x86_64.whl", hash = "sha256:428e04521a40150c85216fc8b85e8d39fece235a9cf5e383761238c7fa9b96fb", size = 2102079 }, + { url = "https://files.pythonhosted.org/packages/ad/1f/8970b150a4b4365623ae00fc88603491f763c627311ae8031e3111356d6e/pydantic_core-2.46.4-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:23ace664830ee0bfe014a0c7bc248b1f7f25ed7ad103852c317624a1083af462", size = 1952179 }, + { url = "https://files.pythonhosted.org/packages/95/30/5211a831ae054928054b2f79731661087a2bc5c01e825c672b3a4a8f1b3e/pydantic_core-2.46.4-cp314-cp314-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ce5c1d2a8b27468f433ca974829c44060b8097eedc39933e3c206a90ee49c4a9", size = 1978926 }, + { url = "https://files.pythonhosted.org/packages/57/e9/689668733b1eb67adeef047db3c2e8788fcf65a7fd9c9e2b46b7744fe245/pydantic_core-2.46.4-cp314-cp314-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:7283d57845ecf5a163403eb0702dfc220cc4fbdd18919cb5ccea4f95ee1cdab4", size = 2046785 }, + { url = "https://files.pythonhosted.org/packages/60/d9/6715260422ff50a2109878fd24d948a6c3446bb2664f34ee78cd972b3acd/pydantic_core-2.46.4-cp314-cp314-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8daafc69c93ee8a0204506a3b6b30f586ef54028f52aeeeb5c4cfc5184fd5914", size = 2228733 }, + { url = "https://files.pythonhosted.org/packages/18/ae/fdb2f64316afca925640f8e70bb1a564b0ec2721c1389e25b8eb4bf9a299/pydantic_core-2.46.4-cp314-cp314-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cd2213145bcc2ba85884d0ac63d222fece9209678f77b9b4d76f054c561adb28", size = 2307534 }, + { url = "https://files.pythonhosted.org/packages/89/1d/8eff589b45bb8190a9d12c49cfad0f176a5cbd1534908a6b5125e2886239/pydantic_core-2.46.4-cp314-cp314-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7a5f930472650a82629163023e630d160863fce524c616f4e5186e5de9d9a49b", size = 2099732 }, + { url = "https://files.pythonhosted.org/packages/06/d5/ee5a3366637fee41dee51a1fc91562dcf12ddbc68fda34e6b253da2324bb/pydantic_core-2.46.4-cp314-cp314-manylinux_2_31_riscv64.whl", hash = "sha256:c1b3f518abeca3aa13c712fd202306e145abf59a18b094a6bafb2d2bbf59192c", size = 2129627 }, + { url = "https://files.pythonhosted.org/packages/94/33/2414be571d2c6a6c4d08be21f9292b6d3fdb08949a97b6dfe985017821db/pydantic_core-2.46.4-cp314-cp314-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:1a7dd0b3ee80d90150e3495a3a13ac34dbcbfd4f012996a6a1d8900e91b5c0fb", size = 2179141 }, + { url = "https://files.pythonhosted.org/packages/7b/79/7daa95be995be0eecc4cf75064cb33f9bbbfe3fe0158caf2f0d4a996a5c7/pydantic_core-2.46.4-cp314-cp314-musllinux_1_1_aarch64.whl", hash = "sha256:3fb702cd90b0446a3a1c5e470bfa0dd23c0233b676a9099ddcc964fa6ca13898", size = 2184325 }, + { url = "https://files.pythonhosted.org/packages/9f/cb/d0a382f5c0de8a222dc61c65348e0ce831b1f68e0a018450d31c2cace3a5/pydantic_core-2.46.4-cp314-cp314-musllinux_1_1_armv7l.whl", hash = "sha256:b8458003118a712e66286df6a707db01c52c0f52f7db8e4a38f0da1d3b94fc4e", size = 2323990 }, + { url = "https://files.pythonhosted.org/packages/05/db/d9ba624cc4a5aced1598e88c04fdbd8310c8a69b9d38b9a3d39ce3a61ed7/pydantic_core-2.46.4-cp314-cp314-musllinux_1_1_x86_64.whl", hash = "sha256:372429a130e469c9cd698925ce5fc50940b7a1336b0d82038e63d5bbc4edc519", size = 2369978 }, + { url = "https://files.pythonhosted.org/packages/f2/20/d15df15ba918c423461905802bfd2981c3af0bfa0e40d05e13edbfa48bc3/pydantic_core-2.46.4-cp314-cp314-win32.whl", hash = "sha256:85bb3611ff1802f3ee7fdd7dbff26b56f343fb432d57a4728fdd49b6ef35e2f4", size = 1966354 }, + { url = "https://files.pythonhosted.org/packages/fc/b6/6b8de4c0a7d7ab3004c439c80c5c1e0a3e8d78bbae19379b01960383d9e5/pydantic_core-2.46.4-cp314-cp314-win_amd64.whl", hash = "sha256:811ff8e9c313ab425368bcbb36e5c4ebd7108c2bbf4e4089cfbb0b01eff63fac", size = 2072238 }, + { url = "https://files.pythonhosted.org/packages/32/36/51eb763beec1f4cf59b1db243a7dcc39cbb41230f050a09b9d69faaf0a48/pydantic_core-2.46.4-cp314-cp314-win_arm64.whl", hash = "sha256:bfec22eab3c8cc2ceec0248aec886624116dc079afa027ecc8ad4a7e62010f8a", size = 2018251 }, + { url = "https://files.pythonhosted.org/packages/e8/91/855af51d625b23aa987116a19e231d2aaef9c4a415273ddc189b79a45fee/pydantic_core-2.46.4-cp314-cp314t-macosx_10_12_x86_64.whl", hash = "sha256:af8244b2bef6aaad6d92cda81372de7f8c8d36c9f0c3ea36e827c60e7d9467a0", size = 2099593 }, + { url = "https://files.pythonhosted.org/packages/fb/1b/8784a54c65edb5f49f0a14d6977cf1b209bba85a4c77445b255c2de58ab3/pydantic_core-2.46.4-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:5a4330cdbc57162e4b3aa303f588ba752257694c9c9be3e7ebb11b4aca659b5d", size = 1935226 }, + { url = "https://files.pythonhosted.org/packages/e8/e7/1955d28d1afc56dd4b3ad7cc0cf39df1b9852964cf16e5d13912756d6d6b/pydantic_core-2.46.4-cp314-cp314t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:29c61fc04a3d840155ff08e475a04809278972fe6aef51e2720554e96367e34b", size = 1974605 }, + { url = "https://files.pythonhosted.org/packages/93/e2/3fedbf0ba7a22850e6e9fd78117f1c0f10f950182344d8a6c535d468fdd8/pydantic_core-2.46.4-cp314-cp314t-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:c50f2528cf200c5eed56faf3f4e22fcd5f38c157a8b78576e6ba3168ec35f000", size = 2030777 }, + { url = "https://files.pythonhosted.org/packages/f8/61/46be275fcaaba0b4f5b9669dd852267ce1ff616592dccf7a7845588df091/pydantic_core-2.46.4-cp314-cp314t-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:0cbe8b01f948de4286c74cdd6c667aceb38f5c1e26f0693b3983d9d74887c65e", size = 2236641 }, + { url = "https://files.pythonhosted.org/packages/60/db/12e93e46a8bac9988be3c016860f83293daea8c716c029c9ace279036f2f/pydantic_core-2.46.4-cp314-cp314t-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:617d7e2ca7dcb8c5cf6bcb8c59b8832c94b36196bbf1cbd1bfb56ed341905edd", size = 2286404 }, + { url = "https://files.pythonhosted.org/packages/e2/4a/4d8b19008f38d31c53b8219cfedc2e3d5de5fe99d90076b7e767de29274f/pydantic_core-2.46.4-cp314-cp314t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7027560ee92211647d0d34e3f7cd6f50da56399d26a9c8ad0da286d3869a53f3", size = 2109219 }, + { url = "https://files.pythonhosted.org/packages/88/70/3cbc40978fefb7bb09c6708d40d4ad1a5d70fd7213c3d17f971de868ec1f/pydantic_core-2.46.4-cp314-cp314t-manylinux_2_31_riscv64.whl", hash = "sha256:f99626688942fb746e545232e7726926f3be91b5975f8b55327665fafda991c7", size = 2110594 }, + { url = "https://files.pythonhosted.org/packages/9d/20/b8d36736216e29491125531685b2f9e61aa5b4b2599893f8268551da3338/pydantic_core-2.46.4-cp314-cp314t-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:fc3e9034a63de20e15e8ade85358bc6efc614008cab72898b4b4952bea0509ff", size = 2159542 }, + { url = "https://files.pythonhosted.org/packages/1d/a2/367df868eb584dacf6bf82a389272406d7178e301c4ac82545ab98bc2dd9/pydantic_core-2.46.4-cp314-cp314t-musllinux_1_1_aarch64.whl", hash = "sha256:97e7cf2be5c77b7d1a9713a05605d49460d02c6078d38d8bef3cbe323c548424", size = 2168146 }, + { url = "https://files.pythonhosted.org/packages/c1/b8/4460f77f7e201893f649a29ab355dddd3beee8a97bcb1a320db414f9a06e/pydantic_core-2.46.4-cp314-cp314t-musllinux_1_1_armv7l.whl", hash = "sha256:3bf92c5d0e00fefaab325a4d27828fe6b6e2a21848686b5b60d2d9eeb09d76c6", size = 2306309 }, + { url = "https://files.pythonhosted.org/packages/64/c4/be2639293acd87dc8ddbcec41a73cee9b2ebf996fe6d892a1a74e88ad3f7/pydantic_core-2.46.4-cp314-cp314t-musllinux_1_1_x86_64.whl", hash = "sha256:3ecbc122d18468d06ca279dc26a8c2e2d5acb10943bb35e36ae92096dc3b5565", size = 2369736 }, + { url = "https://files.pythonhosted.org/packages/30/a6/9f9f380dbb301f67023bf8f707aaa75daadf84f7152d95c410fd7e81d994/pydantic_core-2.46.4-cp314-cp314t-win32.whl", hash = "sha256:e846ae7835bf0703ae43f534ab79a867146dadd59dc9ca5c8b53d5c8f7c9ef02", size = 1955575 }, + { url = "https://files.pythonhosted.org/packages/40/1f/f1eb9eb350e795d1af8586289746f5c5677d16043040d63710e22abc43c9/pydantic_core-2.46.4-cp314-cp314t-win_amd64.whl", hash = "sha256:2108ba5c1c1eca18030634489dc544844144ee36357f2f9f780b93e7ddbb44b5", size = 2051624 }, + { url = "https://files.pythonhosted.org/packages/f6/d2/42dd53d0a85c27606f316d3aa5d2869c4e8470a5ed6dec30e4a1abe19192/pydantic_core-2.46.4-cp314-cp314t-win_arm64.whl", hash = "sha256:4fcbe087dbc2068af7eda3aa87634eba216dbda64d1ae73c8684b621d33f6596", size = 2017325 }, + { url = "https://files.pythonhosted.org/packages/ee/a4/73995fd4ebbb46ba0ee51e6fa049b8f02c40daebb762208feda8a6b7894d/pydantic_core-2.46.4-graalpy311-graalpy242_311_native-macosx_10_12_x86_64.whl", hash = "sha256:14d4edf427bdcf950a8a02d7cb44a08614388dd6e1bdcbf4f67504fa7887da9c", size = 2111589 }, + { url = "https://files.pythonhosted.org/packages/fb/7f/f37d3a5e8bfcc2e403f5c57a730f2d815693fb42119e8ea48b3789335af1/pydantic_core-2.46.4-graalpy311-graalpy242_311_native-macosx_11_0_arm64.whl", hash = "sha256:0ce40cd7b21210e99342afafbd4d0f76d784eb5b1d60f3bdc566be4983c6c73b", size = 1944552 }, + { url = "https://files.pythonhosted.org/packages/15/3c/d7eb777b3ff43e8433a4efb39a17aa8fd98a4ee8561a24a67ef5db07b2d6/pydantic_core-2.46.4-graalpy311-graalpy242_311_native-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:90884113d8b48f760e9587002789ddd741e76ab9f89518cd1e43b1f1a52ec44b", size = 1982984 }, + { url = "https://files.pythonhosted.org/packages/63/87/70b9f40170a81afd55ca26c9b2acb25c20d64bcfbf888fafecb3ba077d4c/pydantic_core-2.46.4-graalpy311-graalpy242_311_native-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:66ce7632c22d837c95301830e111ad0128a32b8207533b60896a96c4915192ea", size = 2138417 }, + { url = "https://files.pythonhosted.org/packages/9d/1d/8987ad40f65ae1432753072f214fb5c74fe47ffbd0698bb9cbbb585664f8/pydantic_core-2.46.4-graalpy312-graalpy250_312_native-macosx_10_12_x86_64.whl", hash = "sha256:1d8ba486450b14f3b1d63bc521d410ec7565e52f887b9fb671791886436a42f7", size = 2095527 }, + { url = "https://files.pythonhosted.org/packages/64/d3/84c282a7eee1d3ac4c0377546ef5a1ea436ce26840d9ac3b7ed54a377507/pydantic_core-2.46.4-graalpy312-graalpy250_312_native-macosx_11_0_arm64.whl", hash = "sha256:3009f12e4e90b7f88b4f9adb1b0c4a3d58fe7820f3238c190047209d148026df", size = 1936024 }, + { url = "https://files.pythonhosted.org/packages/d7/ca/eac61596cdeb4d7e174d3dc0bd8a6238f14f75f97a24e7b7db4c7e7340a0/pydantic_core-2.46.4-graalpy312-graalpy250_312_native-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ad785e92e6dc634c21555edc8bd6b64957ab844541bcb96a1366c202951ae526", size = 1990696 }, + { url = "https://files.pythonhosted.org/packages/fa/c3/7c8b240552251faf6b3a957db200fcfbbcec36763c050428b601e0c9b83b/pydantic_core-2.46.4-graalpy312-graalpy250_312_native-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:00c603d540afdd6b80eb39f078f33ebd46211f02f33e34a32d9f053bba711de0", size = 2147590 }, + { url = "https://files.pythonhosted.org/packages/11/cb/428de0385b6c8d44b716feba566abfacfbd23ee3c4439faa789a1456242f/pydantic_core-2.46.4-pp311-pypy311_pp73-macosx_10_12_x86_64.whl", hash = "sha256:0c563b08bca408dc7f65f700633d8442fffb2421fc47b8101377e9fd65051ff0", size = 2112782 }, + { url = "https://files.pythonhosted.org/packages/0b/b5/6a17bdadd0fc1f170adfd05a20d37c832f52b117b4d9131da1f41bb097ce/pydantic_core-2.46.4-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:db06ffe51636ffe9ca531fe9023dd64bdd794be8754cb5df57c5498ae5b518a7", size = 1952146 }, + { url = "https://files.pythonhosted.org/packages/2a/dc/03734d80e362cd43ef65428e9de77c730ce7f2f11c60d2b1e1b39f0fbf99/pydantic_core-2.46.4-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:133878133d271ade3d41d1bfb2a45ec38dbdbda40bc065921c6b04e4630127e2", size = 2134492 }, + { url = "https://files.pythonhosted.org/packages/de/df/5e5ffc085ed07cc22d298134d3d911c63e91f6a0eb91fe646750a3209910/pydantic_core-2.46.4-pp311-pypy311_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:9bc519fbf2b7578398853d815009ae5e4d4603d12f4e3f91da8c06852d3da3e9", size = 2156604 }, + { url = "https://files.pythonhosted.org/packages/81/44/6e112a4253e56f5705467cbab7ab5e91ee7398ba3d56d358635958893d3e/pydantic_core-2.46.4-pp311-pypy311_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:c7a7bd4e39e8e4c12c39cd480356842b6a8a06e41b23a55a5e3e191718838ddf", size = 2183828 }, + { url = "https://files.pythonhosted.org/packages/ac/ad/5565071e937d8e752842ac241463944c9eb14c87e2d269f2658a5bd05e98/pydantic_core-2.46.4-pp311-pypy311_pp73-musllinux_1_1_armv7l.whl", hash = "sha256:d396ec2b979760aaf3218e76c24e65bd0aca24983298653b3a9d7a45f9e47b30", size = 2310000 }, + { url = "https://files.pythonhosted.org/packages/4f/c3/66883a5cec183e7fba4d024b4cbbe61851a63750ef606b0afecc46d1f2bf/pydantic_core-2.46.4-pp311-pypy311_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:86e1a4418c6cd97d60c95c71164158eaf7324fae7b0923264016baa993eba6fc", size = 2361286 }, + { url = "https://files.pythonhosted.org/packages/4b/2d/69abac8f838090bbecd5df894befb2c2619e7996a98ddb949db9f3b93225/pydantic_core-2.46.4-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:d51026d73fcfd93610abc7b27789c26b313920fcfb20e27462d74a7f8b06e983", size = 2193071 }, +] + +[[package]] +name = "pygments" +version = "2.20.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/c3/b2/bc9c9196916376152d655522fdcebac55e66de6603a76a02bca1b6414f6c/pygments-2.20.0.tar.gz", hash = "sha256:6757cd03768053ff99f3039c1a36d6c0aa0b263438fcab17520b30a303a82b5f", size = 4955991 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/f4/7e/a72dd26f3b0f4f2bf1dd8923c85f7ceb43172af56d63c7383eb62b332364/pygments-2.20.0-py3-none-any.whl", hash = "sha256:81a9e26dd42fd28a23a2d169d86d7ac03b46e2f8b59ed4698fb4785f946d0176", size = 1231151 }, +] + +[[package]] +name = "pytest" +version = "9.1.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "colorama", marker = "sys_platform == 'win32'" }, + { name = "iniconfig" }, + { name = "packaging" }, + { name = "pluggy" }, + { name = "pygments" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/e4/47/b9efed96c114afcfa3c9d3fe98a76a1d14c74a9e266d397cf6eb64be5e01/pytest-9.1.1.tar.gz", hash = "sha256:1088fbde8f2b49d95a549a195707afa7a76a3ce9bcadc26b6d71f0ffda5fe313", size = 1636369 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/24/25/1de2678b631f5a49215c6c96fff41ba892b0a34df68d6d80292b1b48aa7f/pytest-9.1.1-py3-none-any.whl", hash = "sha256:37a86b45efb9a47a61a36449063e8e18d0cab3161329fc099eb21783169c4f0c", size = 386536 }, +] + +[[package]] +name = "pyyaml" +version = "6.0.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/05/8e/961c0007c59b8dd7729d542c61a4d537767a59645b82a0b521206e1e25c2/pyyaml-6.0.3.tar.gz", hash = "sha256:d76623373421df22fb4cf8817020cbb7ef15c725b9d5e45f17e189bfc384190f", size = 130960 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/6d/16/a95b6757765b7b031c9374925bb718d55e0a9ba8a1b6a12d25962ea44347/pyyaml-6.0.3-cp311-cp311-macosx_10_13_x86_64.whl", hash = "sha256:44edc647873928551a01e7a563d7452ccdebee747728c1080d881d68af7b997e", size = 185826 }, + { url = "https://files.pythonhosted.org/packages/16/19/13de8e4377ed53079ee996e1ab0a9c33ec2faf808a4647b7b4c0d46dd239/pyyaml-6.0.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:652cb6edd41e718550aad172851962662ff2681490a8a711af6a4d288dd96824", size = 175577 }, + { url = "https://files.pythonhosted.org/packages/0c/62/d2eb46264d4b157dae1275b573017abec435397aa59cbcdab6fc978a8af4/pyyaml-6.0.3-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:10892704fc220243f5305762e276552a0395f7beb4dbf9b14ec8fd43b57f126c", size = 775556 }, + { url = "https://files.pythonhosted.org/packages/10/cb/16c3f2cf3266edd25aaa00d6c4350381c8b012ed6f5276675b9eba8d9ff4/pyyaml-6.0.3-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:850774a7879607d3a6f50d36d04f00ee69e7fc816450e5f7e58d7f17f1ae5c00", size = 882114 }, + { url = "https://files.pythonhosted.org/packages/71/60/917329f640924b18ff085ab889a11c763e0b573da888e8404ff486657602/pyyaml-6.0.3-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:b8bb0864c5a28024fac8a632c443c87c5aa6f215c0b126c449ae1a150412f31d", size = 806638 }, + { url = "https://files.pythonhosted.org/packages/dd/6f/529b0f316a9fd167281a6c3826b5583e6192dba792dd55e3203d3f8e655a/pyyaml-6.0.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:1d37d57ad971609cf3c53ba6a7e365e40660e3be0e5175fa9f2365a379d6095a", size = 767463 }, + { url = "https://files.pythonhosted.org/packages/f2/6a/b627b4e0c1dd03718543519ffb2f1deea4a1e6d42fbab8021936a4d22589/pyyaml-6.0.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:37503bfbfc9d2c40b344d06b2199cf0e96e97957ab1c1b546fd4f87e53e5d3e4", size = 794986 }, + { url = "https://files.pythonhosted.org/packages/45/91/47a6e1c42d9ee337c4839208f30d9f09caa9f720ec7582917b264defc875/pyyaml-6.0.3-cp311-cp311-win32.whl", hash = "sha256:8098f252adfa6c80ab48096053f512f2321f0b998f98150cea9bd23d83e1467b", size = 142543 }, + { url = "https://files.pythonhosted.org/packages/da/e3/ea007450a105ae919a72393cb06f122f288ef60bba2dc64b26e2646fa315/pyyaml-6.0.3-cp311-cp311-win_amd64.whl", hash = "sha256:9f3bfb4965eb874431221a3ff3fdcddc7e74e3b07799e0e84ca4a0f867d449bf", size = 158763 }, + { url = "https://files.pythonhosted.org/packages/d1/33/422b98d2195232ca1826284a76852ad5a86fe23e31b009c9886b2d0fb8b2/pyyaml-6.0.3-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:7f047e29dcae44602496db43be01ad42fc6f1cc0d8cd6c83d342306c32270196", size = 182063 }, + { url = "https://files.pythonhosted.org/packages/89/a0/6cf41a19a1f2f3feab0e9c0b74134aa2ce6849093d5517a0c550fe37a648/pyyaml-6.0.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:fc09d0aa354569bc501d4e787133afc08552722d3ab34836a80547331bb5d4a0", size = 173973 }, + { url = "https://files.pythonhosted.org/packages/ed/23/7a778b6bd0b9a8039df8b1b1d80e2e2ad78aa04171592c8a5c43a56a6af4/pyyaml-6.0.3-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:9149cad251584d5fb4981be1ecde53a1ca46c891a79788c0df828d2f166bda28", size = 775116 }, + { url = "https://files.pythonhosted.org/packages/65/30/d7353c338e12baef4ecc1b09e877c1970bd3382789c159b4f89d6a70dc09/pyyaml-6.0.3-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:5fdec68f91a0c6739b380c83b951e2c72ac0197ace422360e6d5a959d8d97b2c", size = 844011 }, + { url = "https://files.pythonhosted.org/packages/8b/9d/b3589d3877982d4f2329302ef98a8026e7f4443c765c46cfecc8858c6b4b/pyyaml-6.0.3-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ba1cc08a7ccde2d2ec775841541641e4548226580ab850948cbfda66a1befcdc", size = 807870 }, + { url = "https://files.pythonhosted.org/packages/05/c0/b3be26a015601b822b97d9149ff8cb5ead58c66f981e04fedf4e762f4bd4/pyyaml-6.0.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:8dc52c23056b9ddd46818a57b78404882310fb473d63f17b07d5c40421e47f8e", size = 761089 }, + { url = "https://files.pythonhosted.org/packages/be/8e/98435a21d1d4b46590d5459a22d88128103f8da4c2d4cb8f14f2a96504e1/pyyaml-6.0.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:41715c910c881bc081f1e8872880d3c650acf13dfa8214bad49ed4cede7c34ea", size = 790181 }, + { url = "https://files.pythonhosted.org/packages/74/93/7baea19427dcfbe1e5a372d81473250b379f04b1bd3c4c5ff825e2327202/pyyaml-6.0.3-cp312-cp312-win32.whl", hash = "sha256:96b533f0e99f6579b3d4d4995707cf36df9100d67e0c8303a0c55b27b5f99bc5", size = 137658 }, + { url = "https://files.pythonhosted.org/packages/86/bf/899e81e4cce32febab4fb42bb97dcdf66bc135272882d1987881a4b519e9/pyyaml-6.0.3-cp312-cp312-win_amd64.whl", hash = "sha256:5fcd34e47f6e0b794d17de1b4ff496c00986e1c83f7ab2fb8fcfe9616ff7477b", size = 154003 }, + { url = "https://files.pythonhosted.org/packages/1a/08/67bd04656199bbb51dbed1439b7f27601dfb576fb864099c7ef0c3e55531/pyyaml-6.0.3-cp312-cp312-win_arm64.whl", hash = "sha256:64386e5e707d03a7e172c0701abfb7e10f0fb753ee1d773128192742712a98fd", size = 140344 }, + { url = "https://files.pythonhosted.org/packages/d1/11/0fd08f8192109f7169db964b5707a2f1e8b745d4e239b784a5a1dd80d1db/pyyaml-6.0.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:8da9669d359f02c0b91ccc01cac4a67f16afec0dac22c2ad09f46bee0697eba8", size = 181669 }, + { url = "https://files.pythonhosted.org/packages/b1/16/95309993f1d3748cd644e02e38b75d50cbc0d9561d21f390a76242ce073f/pyyaml-6.0.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:2283a07e2c21a2aa78d9c4442724ec1eb15f5e42a723b99cb3d822d48f5f7ad1", size = 173252 }, + { url = "https://files.pythonhosted.org/packages/50/31/b20f376d3f810b9b2371e72ef5adb33879b25edb7a6d072cb7ca0c486398/pyyaml-6.0.3-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ee2922902c45ae8ccada2c5b501ab86c36525b883eff4255313a253a3160861c", size = 767081 }, + { url = "https://files.pythonhosted.org/packages/49/1e/a55ca81e949270d5d4432fbbd19dfea5321eda7c41a849d443dc92fd1ff7/pyyaml-6.0.3-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:a33284e20b78bd4a18c8c2282d549d10bc8408a2a7ff57653c0cf0b9be0afce5", size = 841159 }, + { url = "https://files.pythonhosted.org/packages/74/27/e5b8f34d02d9995b80abcef563ea1f8b56d20134d8f4e5e81733b1feceb2/pyyaml-6.0.3-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0f29edc409a6392443abf94b9cf89ce99889a1dd5376d94316ae5145dfedd5d6", size = 801626 }, + { url = "https://files.pythonhosted.org/packages/f9/11/ba845c23988798f40e52ba45f34849aa8a1f2d4af4b798588010792ebad6/pyyaml-6.0.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:f7057c9a337546edc7973c0d3ba84ddcdf0daa14533c2065749c9075001090e6", size = 753613 }, + { url = "https://files.pythonhosted.org/packages/3d/e0/7966e1a7bfc0a45bf0a7fb6b98ea03fc9b8d84fa7f2229e9659680b69ee3/pyyaml-6.0.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:eda16858a3cab07b80edaf74336ece1f986ba330fdb8ee0d6c0d68fe82bc96be", size = 794115 }, + { url = "https://files.pythonhosted.org/packages/de/94/980b50a6531b3019e45ddeada0626d45fa85cbe22300844a7983285bed3b/pyyaml-6.0.3-cp313-cp313-win32.whl", hash = "sha256:d0eae10f8159e8fdad514efdc92d74fd8d682c933a6dd088030f3834bc8e6b26", size = 137427 }, + { url = "https://files.pythonhosted.org/packages/97/c9/39d5b874e8b28845e4ec2202b5da735d0199dbe5b8fb85f91398814a9a46/pyyaml-6.0.3-cp313-cp313-win_amd64.whl", hash = "sha256:79005a0d97d5ddabfeeea4cf676af11e647e41d81c9a7722a193022accdb6b7c", size = 154090 }, + { url = "https://files.pythonhosted.org/packages/73/e8/2bdf3ca2090f68bb3d75b44da7bbc71843b19c9f2b9cb9b0f4ab7a5a4329/pyyaml-6.0.3-cp313-cp313-win_arm64.whl", hash = "sha256:5498cd1645aa724a7c71c8f378eb29ebe23da2fc0d7a08071d89469bf1d2defb", size = 140246 }, + { url = "https://files.pythonhosted.org/packages/9d/8c/f4bd7f6465179953d3ac9bc44ac1a8a3e6122cf8ada906b4f96c60172d43/pyyaml-6.0.3-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:8d1fab6bb153a416f9aeb4b8763bc0f22a5586065f86f7664fc23339fc1c1fac", size = 181814 }, + { url = "https://files.pythonhosted.org/packages/bd/9c/4d95bb87eb2063d20db7b60faa3840c1b18025517ae857371c4dd55a6b3a/pyyaml-6.0.3-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:34d5fcd24b8445fadc33f9cf348c1047101756fd760b4dacb5c3e99755703310", size = 173809 }, + { url = "https://files.pythonhosted.org/packages/92/b5/47e807c2623074914e29dabd16cbbdd4bf5e9b2db9f8090fa64411fc5382/pyyaml-6.0.3-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:501a031947e3a9025ed4405a168e6ef5ae3126c59f90ce0cd6f2bfc477be31b7", size = 766454 }, + { url = "https://files.pythonhosted.org/packages/02/9e/e5e9b168be58564121efb3de6859c452fccde0ab093d8438905899a3a483/pyyaml-6.0.3-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:b3bc83488de33889877a0f2543ade9f70c67d66d9ebb4ac959502e12de895788", size = 836355 }, + { url = "https://files.pythonhosted.org/packages/88/f9/16491d7ed2a919954993e48aa941b200f38040928474c9e85ea9e64222c3/pyyaml-6.0.3-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c458b6d084f9b935061bc36216e8a69a7e293a2f1e68bf956dcd9e6cbcd143f5", size = 794175 }, + { url = "https://files.pythonhosted.org/packages/dd/3f/5989debef34dc6397317802b527dbbafb2b4760878a53d4166579111411e/pyyaml-6.0.3-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:7c6610def4f163542a622a73fb39f534f8c101d690126992300bf3207eab9764", size = 755228 }, + { url = "https://files.pythonhosted.org/packages/d7/ce/af88a49043cd2e265be63d083fc75b27b6ed062f5f9fd6cdc223ad62f03e/pyyaml-6.0.3-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:5190d403f121660ce8d1d2c1bb2ef1bd05b5f68533fc5c2ea899bd15f4399b35", size = 789194 }, + { url = "https://files.pythonhosted.org/packages/23/20/bb6982b26a40bb43951265ba29d4c246ef0ff59c9fdcdf0ed04e0687de4d/pyyaml-6.0.3-cp314-cp314-win_amd64.whl", hash = "sha256:4a2e8cebe2ff6ab7d1050ecd59c25d4c8bd7e6f400f5f82b96557ac0abafd0ac", size = 156429 }, + { url = "https://files.pythonhosted.org/packages/f4/f4/a4541072bb9422c8a883ab55255f918fa378ecf083f5b85e87fc2b4eda1b/pyyaml-6.0.3-cp314-cp314-win_arm64.whl", hash = "sha256:93dda82c9c22deb0a405ea4dc5f2d0cda384168e466364dec6255b293923b2f3", size = 143912 }, + { url = "https://files.pythonhosted.org/packages/7c/f9/07dd09ae774e4616edf6cda684ee78f97777bdd15847253637a6f052a62f/pyyaml-6.0.3-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:02893d100e99e03eda1c8fd5c441d8c60103fd175728e23e431db1b589cf5ab3", size = 189108 }, + { url = "https://files.pythonhosted.org/packages/4e/78/8d08c9fb7ce09ad8c38ad533c1191cf27f7ae1effe5bb9400a46d9437fcf/pyyaml-6.0.3-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:c1ff362665ae507275af2853520967820d9124984e0f7466736aea23d8611fba", size = 183641 }, + { url = "https://files.pythonhosted.org/packages/7b/5b/3babb19104a46945cf816d047db2788bcaf8c94527a805610b0289a01c6b/pyyaml-6.0.3-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6adc77889b628398debc7b65c073bcb99c4a0237b248cacaf3fe8a557563ef6c", size = 831901 }, + { url = "https://files.pythonhosted.org/packages/8b/cc/dff0684d8dc44da4d22a13f35f073d558c268780ce3c6ba1b87055bb0b87/pyyaml-6.0.3-cp314-cp314t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:a80cb027f6b349846a3bf6d73b5e95e782175e52f22108cfa17876aaeff93702", size = 861132 }, + { url = "https://files.pythonhosted.org/packages/b1/5e/f77dc6b9036943e285ba76b49e118d9ea929885becb0a29ba8a7c75e29fe/pyyaml-6.0.3-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:00c4bdeba853cc34e7dd471f16b4114f4162dc03e6b7afcc2128711f0eca823c", size = 839261 }, + { url = "https://files.pythonhosted.org/packages/ce/88/a9db1376aa2a228197c58b37302f284b5617f56a5d959fd1763fb1675ce6/pyyaml-6.0.3-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:66e1674c3ef6f541c35191caae2d429b967b99e02040f5ba928632d9a7f0f065", size = 805272 }, + { url = "https://files.pythonhosted.org/packages/da/92/1446574745d74df0c92e6aa4a7b0b3130706a4142b2d1a5869f2eaa423c6/pyyaml-6.0.3-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:16249ee61e95f858e83976573de0f5b2893b3677ba71c9dd36b9cf8be9ac6d65", size = 829923 }, + { url = "https://files.pythonhosted.org/packages/f0/7a/1c7270340330e575b92f397352af856a8c06f230aa3e76f86b39d01b416a/pyyaml-6.0.3-cp314-cp314t-win_amd64.whl", hash = "sha256:4ad1906908f2f5ae4e5a8ddfce73c320c2a1429ec52eafd27138b7f1cbe341c9", size = 174062 }, + { url = "https://files.pythonhosted.org/packages/f1/12/de94a39c2ef588c7e6455cfbe7343d3b2dc9d6b6b2f40c4c6565744c873d/pyyaml-6.0.3-cp314-cp314t-win_arm64.whl", hash = "sha256:ebc55a14a21cb14062aa4162f906cd962b28e2e9ea38f9b4391244cd8de4ae0b", size = 149341 }, +] + +[[package]] +name = "rich" +version = "15.0.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "markdown-it-py" }, + { name = "pygments" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/c0/8f/0722ca900cc807c13a6a0c696dacf35430f72e0ec571c4275d2371fca3e9/rich-15.0.0.tar.gz", hash = "sha256:edd07a4824c6b40189fb7ac9bc4c52536e9780fbbfbddf6f1e2502c31b068c36", size = 230680 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/82/3b/64d4899d73f91ba49a8c18a8ff3f0ea8f1c1d75481760df8c68ef5235bf5/rich-15.0.0-py3-none-any.whl", hash = "sha256:33bd4ef74232fb73fe9279a257718407f169c09b78a87ad3d296f548e27de0bb", size = 310654 }, +] + +[[package]] +name = "ruff" +version = "0.15.18" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/74/98/1295ad5a5aa9bc85bdcdfa5d82fe7b49c61af5657df4f227637ff9de0da6/ruff-0.15.18.tar.gz", hash = "sha256:2698a964c70e8bf402dcb99c8810472d270d141e7aa8c4e13599fd52033a2f33", size = 4761437 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/b9/d0/686e984941269621e2be72612d5c1e461f8f7b38415a2a7d7a81c8ae6715/ruff-0.15.18-py3-none-linux_armv6l.whl", hash = "sha256:8b6850172348c8381b8b3084c5915a4393c2373b9b54cd5b5e1ea15812bc10df", size = 10887308 }, + { url = "https://files.pythonhosted.org/packages/ed/21/bc4123e3f5515ee99f8ce1eb93a14a0628fe4d1678663cd08f933ac16931/ruff-0.15.18-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:3fccc153a85417dcd976883160cacce486997b0a0058dd18f54b8aaaac7d1ce2", size = 11281305 }, + { url = "https://files.pythonhosted.org/packages/51/93/4769464c25cf7ab2acb3c7dda9cad3d867eb41c59565b3e2a9d17249c90c/ruff-0.15.18-py3-none-macosx_11_0_arm64.whl", hash = "sha256:08d4c86a68f2c3ec2c9d56380a71fb4a4f65373055cbb8caabd645e9102f38d4", size = 10641215 }, + { url = "https://files.pythonhosted.org/packages/6c/42/56926d17120db2c208d76bf60a1a019644dd9e91dc27f0f95c9caddb1366/ruff-0.15.18-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:37e5108745c2c0705da916d7d4de533ddf547051ef45f62888c31bae73f66318", size = 10957224 }, + { url = "https://files.pythonhosted.org/packages/22/4f/d43fab8d8189afde803103022d000a8ef9f230616d436d52a8b2b8d63b50/ruff-0.15.18-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:56949a6ce8b3abde54c0bcb22cebfe57e8771cadc84b407ae8b8eaf67ebdcd43", size = 10699024 }, + { url = "https://files.pythonhosted.org/packages/63/42/1e3e4c68bd408b9768cf3e439acbe2c78245225faef253f7028a0cdb63e0/ruff-0.15.18-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:01a754cd6a1b630d3f97e33eb452cf7a98040482318e870f8bc52a5a30e62657", size = 11491458 }, + { url = "https://files.pythonhosted.org/packages/20/77/47a3484bea8521e14a203d98c389c5c97846675e4f02734672da4a69b52a/ruff-0.15.18-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:6ba7a07e03a44dbf10bb086ee06705b173625014ec99f73a7e6836a5e5590a0c", size = 12383752 }, + { url = "https://files.pythonhosted.org/packages/0a/ca/054159590787023d83b658a1a1819c4c8910114e7015069340b71c0961cb/ruff-0.15.18-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5a2c40a41a4cadbcf5897b548ab29dfe248b20c540961c0247d98a3973c70403", size = 11577923 }, + { url = "https://files.pythonhosted.org/packages/6d/ff/d353d6b7bbd73cc0ec37f4463d7540e45e894338abdd9964eee0de332708/ruff-0.15.18-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5f0480ce690cbb6c4db6e5d08f19fce98e10ba131a8b60c1bcdac42771e3ae2d", size = 11583925 }, + { url = "https://files.pythonhosted.org/packages/c1/4a/891f89b9c296ed3e5f3ece1a5629badc989d9a8fdaa30431aaf4774bc1c2/ruff-0.15.18-py3-none-manylinux_2_31_riscv64.whl", hash = "sha256:2330215f1f393fa8733f55edce04fcf94c36a2c460fcde31f78cc84e4951e9b1", size = 11582834 }, + { url = "https://files.pythonhosted.org/packages/32/a3/ed9e370154bf85de360b93c03026157f02d4943b2d01ff4945f4429f8e8a/ruff-0.15.18-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:a6aa6a3d979e48ae617578183674bf264fbe7d0114a796a26bd678d67963c7ff", size = 10927328 }, + { url = "https://files.pythonhosted.org/packages/f5/d1/5cf5909329fedb5d39d555ee818ba5cf4638e1a301b89785d34f2905bfcb/ruff-0.15.18-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:a81beadbbff2c9c245561ae3f77b16709d87f35eec650d0501679239d3449b22", size = 10693187 }, + { url = "https://files.pythonhosted.org/packages/fd/44/ff6c635cf2c4f4e7b618b6640da057376baa36014695487d88aed4794268/ruff-0.15.18-py3-none-musllinux_1_2_i686.whl", hash = "sha256:2186d9e940ae332ab293623a75b5f4fe49565f449954d50a72a046683aa6b809", size = 11208721 }, + { url = "https://files.pythonhosted.org/packages/88/d9/5baa2a30861adfb7022cf33c1e35b2fc18085b08c16f83eff4c7b99a5f48/ruff-0.15.18-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:5c2abf140438032bc77b2284a6c9944ecd8a19e5f1c7b52b1b8e4a0a80d19a7a", size = 11678599 }, + { url = "https://files.pythonhosted.org/packages/c3/1a/0725a7cfdc32ff769efb96ee782bec882e16448c5d9e3be947ec4c04ce27/ruff-0.15.18-py3-none-win32.whl", hash = "sha256:02299e6e9fa5b297a3f6d5d10d7bcd655c925b028bb8b9d4588214549c6b9ec4", size = 10901903 }, + { url = "https://files.pythonhosted.org/packages/f3/51/805d9f6fb7970505c3504794a5ec350f605361b807fef4dcf214ebd35e72/ruff-0.15.18-py3-none-win_amd64.whl", hash = "sha256:dac80dc8d26b2257dbefabed62f5d255c3937b4ccb122da1fc634794fa3578b3", size = 12041189 }, + { url = "https://files.pythonhosted.org/packages/29/4c/67bb45e41609eb4726f1bfeb59e083cf91d14c696d4bd14c234a980be93d/ruff-0.15.18-py3-none-win_arm64.whl", hash = "sha256:b2c9257fcbd4a3e5b977a1904e6facca016bafe2edc17df24db67cfaee03b4e4", size = 11329958 }, +] + +[[package]] +name = "shellingham" +version = "1.5.4" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/58/15/8b3609fd3830ef7b27b655beb4b4e9c62313a4e8da8c676e142cc210d58e/shellingham-1.5.4.tar.gz", hash = "sha256:8dbca0739d487e5bd35ab3ca4b36e11c4078f3a234bfce294b0a0291363404de", size = 10310 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/e0/f9/0595336914c5619e5f28a1fb793285925a8cd4b432c9da0a987836c7f822/shellingham-1.5.4-py2.py3-none-any.whl", hash = "sha256:7ecfff8f2fd72616f7481040475a65b2bf8af90a56c89140852d1120324e8686", size = 9755 }, +] + +[[package]] +name = "typer" +version = "0.26.7" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "annotated-doc" }, + { name = "colorama", marker = "platform_system == 'Windows'" }, + { name = "rich" }, + { name = "shellingham" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/5e/ed/ef06584ccdd5c410df0837951ecd7e15d9a6144ea1bd4c73cecab1a89891/typer-0.26.7.tar.gz", hash = "sha256:e314a34c617e419c091b2830dda3ea1f257134ff593061a8f5b9717ab8dddb3a", size = 201709 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/24/25/2201973529af2c954de0bb725323c3aaed6d7f0ceee8f550dec9185df013/typer-0.26.7-py3-none-any.whl", hash = "sha256:5c87cfbc5d34491c5346ebf49c23e18d56ccb863268d3a8d592b26087c2f5e58", size = 122456 }, +] + +[[package]] +name = "typing-extensions" +version = "4.15.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/72/94/1a15dd82efb362ac84269196e94cf00f187f7ed21c242792a923cdb1c61f/typing_extensions-4.15.0.tar.gz", hash = "sha256:0cea48d173cc12fa28ecabc3b837ea3cf6f38c6d1136f85cbaaf598984861466", size = 109391 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/18/67/36e9267722cc04a6b9f15c7f3441c2363321a3ea07da7ae0c0707beb2a9c/typing_extensions-4.15.0-py3-none-any.whl", hash = "sha256:f0fa19c6845758ab08074a0cfa8b7aecb71c999ca73d62883bc25cc018c4e548", size = 44614 }, +] + +[[package]] +name = "typing-inspection" +version = "0.4.2" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "typing-extensions" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/55/e3/70399cb7dd41c10ac53367ae42139cf4b1ca5f36bb3dc6c9d33acdb43655/typing_inspection-0.4.2.tar.gz", hash = "sha256:ba561c48a67c5958007083d386c3295464928b01faa735ab8547c5692e87f464", size = 75949 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/dc/9b/47798a6c91d8bdb567fe2698fe81e0c6b7cb7ef4d13da4114b41d239f65d/typing_inspection-0.4.2-py3-none-any.whl", hash = "sha256:4ed1cacbdc298c220f1bd249ed5287caa16f34d44ef4e9c3d0cbad5b521545e7", size = 14611 }, +] + +[[package]] +name = "wise-validator" +version = "0.0.0" +source = { editable = "." } +dependencies = [ + { name = "pydantic" }, + { name = "pyyaml" }, + { name = "typer" }, +] + +[package.dev-dependencies] +dev = [ + { name = "pytest" }, + { name = "ruff" }, +] + +[package.metadata] +requires-dist = [ + { name = "pydantic", specifier = ">=2.0" }, + { name = "pyyaml", specifier = ">=6.0" }, + { name = "typer", specifier = ">=0.12" }, +] + +[package.metadata.requires-dev] +dev = [ + { name = "pytest", specifier = ">=8.0" }, + { name = "ruff", specifier = ">=0.6" }, +]