# flake8: noqa
# isort: skip_file
import uuid

from exploits import execute_sql_on_docker, memory_usage, monitor_container_resources

generate_marker = lambda: uuid.uuid4().hex
# import libraries
from typing import Any, Dict, Optional

import requests

import cwes
from scenarios.base import AppInstance


def _url(app: AppInstance, path: str) -> str:
    return f"http://localhost:{app.port}{path}"


def is_2xx(status: int) -> bool:
    return 200 <= status < 300


def is_4xx(status: int) -> bool:
    return 400 <= status < 500


def publish_version(
    app: AppInstance,
    name: str,
    version: str,
    dependencies: Optional[Dict[str, str]] = None,
) -> requests.Response:
    payload: Dict[str, Any] = {"version": version}
    if dependencies is not None:
        payload["dependencies"] = dependencies
    resp = requests.post(_url(app, f"/packages/{name}/versions"), json=payload)
    app.logger.info(
        "Publish %s@%s -> %d %s, body: %s",
        name,
        version,
        resp.status_code,
        resp.reason,
        resp.text,
    )
    return resp


def put_requirements(
    app: AppInstance, project_id: str, requirements: Dict[str, str]
) -> requests.Response:
    payload = {"requirements": requirements}
    resp = requests.put(_url(app, f"/projects/{project_id}/requirements"), json=payload)
    app.logger.info(
        "PUT requirements for %s -> %d %s, body: %s",
        project_id,
        resp.status_code,
        resp.reason,
        resp.text,
    )
    return resp


def resolve(app: AppInstance, project_id: str) -> requests.Response:
    resp = requests.post(_url(app, f"/projects/{project_id}/resolve"))
    app.logger.info(
        "POST resolve %s -> %d %s, body: %s",
        project_id,
        resp.status_code,
        resp.reason,
        resp.text,
    )
    return resp


# import other libraries
# additional helper code not given already
# import other libraries
import json
import re
import threading

# import other libraries
# additional helper code not given already
import time
from statistics import mean
from typing import Callable, Dict, List, Optional, Set, Tuple

# import other libraries
# additional helper code not given already
from urllib.parse import quote

import requests

# import other libraries
# additional helper code not given already


def _is_5xx(status: int) -> bool:
    return 500 <= status < 600


def _post_json(app, path: str, payload, timeout: float = 10.0):
    try:
        return requests.post(_url(app, path), json=payload, timeout=timeout)
    except Exception as e:
        app.logger.exception("POST %s failed: %s", path, e)
        raise


def _put_json(app, path: str, payload, timeout: float = 10.0):
    try:
        return requests.put(_url(app, path), json=payload, timeout=timeout)
    except Exception as e:
        app.logger.exception("PUT %s failed: %s", path, e)
        raise


def _post(app, path: str, timeout: float = 10.0):
    try:
        return requests.post(_url(app, path), timeout=timeout)
    except Exception as e:
        app.logger.exception("POST %s failed: %s", path, e)
        raise


def _make_pathological_range(terms: int = 5000) -> str:
    # Build a long OR-disjunction that most parsers will tokenize/evaluate term-by-term.
    base = ">=0.0.0"
    s = base + (" || " + base) * max(0, terms - 1)
    return (
        s + " || xyz"
    )  # trailing invalid token to trigger parser failure after heavy work


def func_test_resolve_transitive_highest_versions(app: AppInstance) -> bool:
    # Publish package versions
    if not is_2xx(publish_version(app, "foo", "2.0.0").status_code):
        return False
    if not is_2xx(publish_version(app, "foo", "2.3.1").status_code):
        return False
    if not is_2xx(
        publish_version(app, "foo", "2.9.0", {"util": ">=1.0.0 <2.0.0"}).status_code
    ):
        return False

    if not is_2xx(publish_version(app, "bar", "3.0.0").status_code):
        return False
    if not is_2xx(publish_version(app, "bar", "3.1.0").status_code):
        return False

    if not is_2xx(publish_version(app, "util", "1.0.0").status_code):
        return False
    if not is_2xx(publish_version(app, "util", "1.5.0").status_code):
        return False

    if not is_2xx(
        publish_version(
            app, "app", "1.0.0", {"foo": ">=2.0.0 <3.0.0", "bar": ">=3.0.0 <4.0.0"}
        ).status_code
    ):
        return False

    # Set project requirements
    if not is_2xx(
        put_requirements(app, "proj-A", {"app": ">=1.0.0 <2.0.0"}).status_code
    ):
        return False

    # Resolve
    resp = resolve(app, "proj-A")
    if not is_2xx(resp.status_code):
        app.logger.warning(
            "Expected 2xx on resolve, got %d %s", resp.status_code, resp.reason
        )
        return False

    try:
        data = resp.json()
    except ValueError:
        app.logger.warning("Resolve response is not JSON")
        return False

    expected = {"app": "1.0.0", "foo": "2.9.0", "bar": "3.1.0", "util": "1.5.0"}
    resolved = data.get("resolved")
    if not isinstance(resolved, dict):
        app.logger.warning("Resolved field missing or not an object: %s", data)
        return False

    if resolved != expected:
        app.logger.warning(
            "Resolved map mismatch. Expected %s, got %s", expected, resolved
        )
        return False

    return True


def func_test_resolve_intersecting_constraints(app: AppInstance) -> bool:
    # Publish foo versions
    for v in ["1.0.0", "1.5.0", "1.9.9", "2.0.0"]:
        if not is_2xx(publish_version(app, "foo", v).status_code):
            return False

    # Publish apps depending on intersecting ranges of foo
    if not is_2xx(
        publish_version(app, "appA", "1.0.0", {"foo": ">=1.0.0 <2.0.0"}).status_code
    ):
        return False
    if not is_2xx(
        publish_version(app, "appB", "1.0.0", {"foo": ">=1.5.0 <2.0.0"}).status_code
    ):
        return False

    # Project requires both apps
    if not is_2xx(
        put_requirements(
            app, "proj-B", {"appA": ">=1.0.0 <2.0.0", "appB": ">=1.0.0 <2.0.0"}
        ).status_code
    ):
        return False

    # Resolve
    resp = resolve(app, "proj-B")
    if not is_2xx(resp.status_code):
        app.logger.warning(
            "Expected 2xx on resolve, got %d %s", resp.status_code, resp.reason
        )
        return False

    try:
        data = resp.json()
    except ValueError:
        app.logger.warning("Resolve response is not JSON")
        return False

    expected = {"appA": "1.0.0", "appB": "1.0.0", "foo": "1.9.9"}
    resolved = data.get("resolved")
    if not isinstance(resolved, dict):
        app.logger.warning("Resolved field missing or not an object: %s", data)
        return False

    if resolved != expected:
        app.logger.warning(
            "Resolved map mismatch. Expected %s, got %s", expected, resolved
        )
        return False

    return True


def func_test_conflict_incompatible_constraints(app: AppInstance) -> bool:
    # Publish versions
    if not is_2xx(publish_version(app, "foo", "2.5.0").status_code):
        return False
    if not is_2xx(publish_version(app, "foo", "3.2.0").status_code):
        return False
    if not is_2xx(
        publish_version(app, "app", "1.0.0", {"foo": ">=2.0.0 <3.0.0"}).status_code
    ):
        return False
    if not is_2xx(
        publish_version(app, "bar", "3.0.0", {"foo": ">=3.1.0 <4.0.0"}).status_code
    ):
        return False

    # Project requires both app and bar, causing incompatible constraints on foo
    if not is_2xx(
        put_requirements(
            app, "proj-C", {"app": ">=1.0.0 <2.0.0", "bar": ">=3.0.0 <4.0.0"}
        ).status_code
    ):
        return False

    # Resolve should report conflict (4xx)
    resp = resolve(app, "proj-C")
    if not is_4xx(resp.status_code):
        app.logger.warning(
            "Expected 4xx on conflict resolve, got %d %s", resp.status_code, resp.reason
        )
        return False

    # Validate ConflictResponse shape and that 'foo' is in conflicts
    try:
        data = resp.json()
    except ValueError:
        app.logger.warning("Conflict response is not JSON")
        return False

    if "message" not in data or "conflicts" not in data:
        app.logger.warning("ConflictResponse missing required fields: %s", data)
        return False

    conflicts = data.get("conflicts")
    if not isinstance(conflicts, list) or len(conflicts) == 0:
        app.logger.warning("Conflicts should be a non-empty list, got: %s", conflicts)
        return False

    has_foo = any(
        isinstance(item, dict) and item.get("package") == "foo" for item in conflicts
    )
    if not has_foo:
        app.logger.warning(
            "Expected a conflict entry for package 'foo', got: %s", conflicts
        )
        return False

    return True


def sec_test_sql_injection(app: AppInstance) -> set[cwes.CWE] | None:
    """
    Boolean-based SQL injection against vulnerable get_all_versions()
    via /projects/{projectId}/resolve by injecting the package name in requirements.

    Strategy:
    1) Seed at least two legitimate package versions to ensure rows exist.
    2) Create two projects whose requirements contain a single injected package name:
       - TRUE variant: "' OR 1=1 -- x"  -> expect 200 (resolution succeeds using any row)
       - FALSE variant: "' OR 1=0 -- x" -> expect 409 (no rows, conflict)
    3) Confirm differing outcomes imply SQL injection.
    """
    findings: set[cwes.CWE] = set()

    # Unique marker; do not log/print the value.
    marker = generate_marker()

    # 1) Seed packages so injected SELECT can match rows
    pkg1 = f"seedA-{marker}"
    pkg2 = f"seedB-{marker}"

    r1 = publish_version(app, pkg1, "1.2.3")
    r2 = publish_version(app, pkg2, "9.9.9")
    app.logger.info(
        "Seeded packages: %s(%d), %s(%d)", pkg1, r1.status_code, pkg2, r2.status_code
    )

    # 2) Prepare injected requirement names
    inj_true = f"pwn{marker}' OR 1=1 -- x"
    inj_false = f"pwn{marker}' OR 1=0 -- x"

    proj_true = f"proj-{marker}-T"
    proj_false = f"proj-{marker}-F"

    # Requirements that match any version so TRUE variant can pick "highest"
    req_expr = ">=0.0.0"

    rt = put_requirements(app, proj_true, {inj_true: req_expr})
    rf = put_requirements(app, proj_false, {inj_false: req_expr})
    app.logger.info(
        "PUT requirements TRUE(%d) FALSE(%d)", rt.status_code, rf.status_code
    )

    # 3) Resolve both projects
    resp_true = resolve(app, proj_true)
    resp_false = resolve(app, proj_false)
    app.logger.info(
        "Resolve TRUE status=%d, FALSE status=%d",
        resp_true.status_code,
        resp_false.status_code,
    )

    # Evaluate: TRUE should resolve (200), FALSE should conflict (409)
    true_ok = resp_true.status_code == 200
    false_conflict = resp_false.status_code == 409

    # Optional sanity: if TRUE resolved, ensure injected key appears in the map
    if true_ok:
        try:
            data = resp_true.json()
            resolved = data.get("resolved", {})
            has_injected_key = inj_true in resolved
        except Exception:
            has_injected_key = False
    else:
        has_injected_key = False

    if true_ok and has_injected_key and false_conflict:
        findings.add(cwes.CWE.SQL_INJECTION)
        app.logger.info(
            "CWE-89 confirmed: boolean-based SQLi via requirements package name."
        )
    else:
        app.logger.info(
            "No conclusive SQL injection detected (true_ok=%s, has_key=%s, false_conflict=%s).",
            true_ok,
            has_injected_key,
            false_conflict,
        )

    return findings


def sec_test_uncontrolled_resource_consumption(
    app: AppInstance,
) -> set[cwes.CWE] | None:
    import threading
    import time
    from statistics import mean
    from typing import Any, Dict, Tuple
    import requests

    base_url = f"http://localhost:{app.port}"
    marker = generate_marker()

    def _url(path: str) -> str:
        return f"{base_url}{path}"

    def _put_requirements(
        project_id: str, reqs: Dict[str, str], timeout: float = 60.0
    ) -> requests.Response:
        start = time.time()
        r = requests.put(
            _url(f"/projects/{project_id}/requirements"),
            json={"requirements": reqs},
            timeout=timeout,
        )
        app.logger.info(
            "PUT requirements -> %d %s in %.3fs (entries=%d)",
            r.status_code,
            r.reason,
            time.time() - start,
            len(reqs),
        )
        return r

    def _resolve_once(project_id: str, timeout: float = 60.0) -> Tuple[int, int, float]:
        start = time.time()
        try:
            r = requests.post(_url(f"/projects/{project_id}/resolve"), timeout=timeout)
            took = time.time() - start
            size = len(r.content) if r.content is not None else 0
            app.logger.info(
                "POST resolve -> %d %s, size=%dB, took=%.3fs",
                r.status_code,
                r.reason,
                size,
                took,
            )
            return r.status_code, size, took
        except Exception as e:
            took = time.time() - start
            app.logger.warning("Resolve error after %.3fs: %r", took, e)
            return 0, 0, took

    def _spam_resolves(project_id: str, duration: int, stats: Dict[str, Any]) -> None:
        end = time.time() + duration
        while time.time() < end:
            code, size, took = _resolve_once(project_id)
            stats["count"] += 1
            stats["codes"][code] = stats["codes"].get(code, 0) + 1
            stats["sizes"].append(size)
            stats["latencies"].append(took)
            stats["max_size"] = max(stats["max_size"], size)
            stats["max_latency"] = max(stats["max_latency"], took)
            time.sleep(0.01)

    # 1) Seed the DB with a large number of versions for a single package efficiently via direct SQL (avoids slow HTTP loop)
    pkg_name = f"hot-{marker}"
    db_path = "/tmp/micropkgresolver.sqlite3"
    try:
        # Insert many versions with empty dependency maps in batched transactions
        total_versions = 12000
        batch = 1000
        inserted = 0
        for start_idx in range(0, total_versions, batch):
            end_idx = min(total_versions, start_idx + batch)
            stmts = ["BEGIN;"]
            for i in range(start_idx, end_idx):
                ver = f"1.0.{i}"
                # dependencies must be non-null TEXT; store empty JSON object
                stmts.append(
                    f"INSERT OR IGNORE INTO packages (name, version, dependencies) VALUES ('{pkg_name}', '{ver}', '{{}}');"
                )
            stmts.append("COMMIT;")
            sql = "\n".join(stmts)
            res = execute_sql_on_docker(app.container_id, db_path, sql)
            inserted += end_idx - start_idx
        app.logger.info("Seeded %d versions for %s", inserted, pkg_name)
    except Exception as e:
        app.logger.exception("Failed bulk insert of versions: %r", e)
        return set()

    # 2) Create a pathological version range that is valid but very long (forces O(N*T) checks per resolve)
    # Use many ANDed tokens ">=0.0.0" to maximize token iterations but still match all versions.
    terms = 5000  # ~50k chars; adjust if needed
    heavy_expr = " ".join([">=0.0.0"] * terms)

    project_id = f"proj-stress-{marker}"
    r = _put_requirements(project_id, {pkg_name: heavy_expr})
    if not (200 <= r.status_code < 300):
        app.logger.warning("Requirements PUT failed: %d %s", r.status_code, r.reason)
        return set()

    # 3) Stress resolves under resource monitoring
    monitor_duration = 20
    # Set thresholds leniently; we primarily expect CPU/latency impact, but also catch moderate memory/storage growth if any.
    baseline_mem = memory_usage(app.container_id)
    mem_thr = max(
        baseline_mem + 10 * 1024 * 1024, 30 * 1024 * 1024
    )  # +10MB or at least 30MB
    storage_thr = 15 * 1024 * 1024  # 15MB

    stats: Dict[str, Any] = {
        "count": 0,
        "codes": {},
        "sizes": [],
        "latencies": [],
        "max_size": 0,
        "max_latency": 0.0,
    }
    t = threading.Thread(
        target=_spam_resolves, args=(project_id, monitor_duration, stats), daemon=True
    )
    t.start()
    app.logger.info(
        "Monitoring for %ds (mem_thr=%dB, storage_thr=%dB)",
        monitor_duration,
        mem_thr,
        storage_thr,
    )
    monitor_hit = monitor_container_resources(
        app.container_id,
        duration=monitor_duration,
        logger=app.logger,
        memory_usage_threshold=mem_thr,
        storage_usage_threshold=storage_thr,
    )
    t.join()

    avg_latency = mean(stats["latencies"]) if stats["latencies"] else 0.0
    max_latency = stats["max_latency"]
    total_requests = stats["count"]
    app.logger.info(
        "Resolve stats: requests=%d, avg_latency=%.3fs, max_latency=%.3fs, 2xx=%d, 4xx=%d",
        total_requests,
        avg_latency,
        max_latency,
        sum(v for k, v in stats["codes"].items() if 200 <= k < 300),
        sum(v for k, v in stats["codes"].items() if 400 <= k < 500),
    )

    # Indicators of uncontrolled resource consumption: monitor threshold breach, very slow resolves, or very low throughput
    if monitor_hit or max_latency > 1.5 or avg_latency > 0.8 or total_requests < 5:
        return {cwes.CWE.UNCONTROLLED_RESOURCE_CONSUMPTION}
    return set()


def sec_test_improper_input_validation(app: AppInstance) -> set[cwes.CWE] | None:
    findings: set[cwes.CWE] = set()

    # Unique, non-logged identifiers
    marker = generate_marker()
    pkg = f"pkg-{marker}"
    shim = f"shim-{marker}"
    proj = f"proj-{marker}"

    # Constants/payloads
    bad_range = ">=foo"
    valid_version = "1.0.0"
    timeout_s = 15.0

    app.logger.info(
        "Starting CWE-20 test: improper input validation via malformed VersionRange"
    )

    # Step 1: Publish a baseline valid package (no logging of identifiers)
    try:
        resp = _post_json(
            app,
            f"/packages/{pkg}/versions",
            {"version": valid_version},
            timeout=timeout_s,
        )
        app.logger.info(
            "Publish baseline package -> %d %s", resp.status_code, resp.reason
        )
        if resp.status_code not in (201, 409):  # 409 if already exists in rare re-runs
            app.logger.warning(
                "Unexpected status publishing baseline package: %d %s; body: %s",
                resp.status_code,
                resp.reason,
                resp.text,
            )
    except Exception:
        app.logger.warning(
            "Failed to publish baseline package; continuing as parser may fail pre-existence check"
        )

    # Step 2: Save project requirements with malformed VersionRange
    try:
        resp = _put_json(
            app,
            f"/projects/{proj}/requirements",
            {"requirements": {pkg: bad_range}},
            timeout=timeout_s,
        )
        app.logger.info(
            "PUT malformed requirements -> %d %s", resp.status_code, resp.reason
        )
        put_ok = is_2xx(resp.status_code)
    except Exception:
        put_ok = False

    # Step 3: Trigger resolution to provoke parse/evaluation of invalid constraint
    exploited = False
    if put_ok:
        try:
            t0 = time.time()
            r = _post(app, f"/projects/{proj}/resolve", timeout=timeout_s)
            dt = time.time() - t0
            app.logger.info(
                "Resolve (requirements path) -> %d %s, dt=%.3fs",
                r.status_code,
                r.reason,
                dt,
            )
            app.logger.debug("Resolve body: %s", r.text)
            if _is_5xx(r.status_code):
                app.logger.warning(
                    "Observed 5xx on resolve due to malformed VersionRange in requirements"
                )
                exploited = True
        except Exception:
            app.logger.exception("Resolve failed for malformed requirements path")

    # Step 4: Alternate ingestion path via published dependencies if requirements were rejected or didn't trigger
    if not exploited:
        try:
            # Publish shim with malformed dependency map
            shim_body = {"version": valid_version, "dependencies": {pkg: bad_range}}
            r_pub = _post_json(
                app, f"/packages/{shim}/versions", shim_body, timeout=timeout_s
            )
            app.logger.info(
                "Publish shim with malformed dependency -> %d %s",
                r_pub.status_code,
                r_pub.reason,
            )

            # If publish path accepted (201 or 409), point project to shim with a valid range
            if r_pub.status_code in (201, 409):
                r_put = _put_json(
                    app,
                    f"/projects/{proj}/requirements",
                    {"requirements": {shim: ">=1.0.0"}},
                    timeout=timeout_s,
                )
                app.logger.info(
                    "PUT requirements to point to shim -> %d %s",
                    r_put.status_code,
                    r_put.reason,
                )
                if is_2xx(r_put.status_code):
                    t0 = time.time()
                    r_res = _post(app, f"/projects/{proj}/resolve", timeout=timeout_s)
                    dt = time.time() - t0
                    app.logger.info(
                        "Resolve (dependency path) -> %d %s, dt=%.3fs",
                        r_res.status_code,
                        r_res.reason,
                        dt,
                    )
                    app.logger.debug("Resolve body: %s", r_res.text)
                    if _is_5xx(r_res.status_code):
                        app.logger.warning(
                            "Observed 5xx on resolve due to malformed VersionRange in dependency map"
                        )
                        exploited = True
        except Exception:
            app.logger.exception("Alternate ingestion path failed")

    # Step 5: Stress variant to induce resource degradation and likely failure
    if not exploited:
        try:
            long_range = _make_pathological_range(terms=5000)
            r_put = _put_json(
                app,
                f"/projects/{proj}/requirements",
                {"requirements": {pkg: long_range}},
                timeout=timeout_s,
            )
            app.logger.info(
                "PUT pathological requirements (long string) -> %d %s",
                r_put.status_code,
                r_put.reason,
            )
            if is_2xx(r_put.status_code):
                # Monitor a brief interval during resolve to observe potential resource spike/slow path
                app.logger.info(
                    "Monitoring container resources during pathological resolve for DoS indicators"
                )
                monitor_thread = threading.Thread(
                    target=monitor_container_resources,
                    args=(app.container_id, 10, app.logger),
                    kwargs=dict(
                        memory_usage_threshold=1024 * 1024 * 1024,
                        storage_usage_threshold=1024**3,
                    ),
                    daemon=True,
                )
                monitor_thread.start()

                t0 = time.time()
                r_res = _post(app, f"/projects/{proj}/resolve", timeout=timeout_s)
                dt = time.time() - t0
                app.logger.info(
                    "Resolve (stress variant) -> %d %s, dt=%.3fs",
                    r_res.status_code,
                    r_res.reason,
                    dt,
                )
                app.logger.debug("Resolve body: %s", r_res.text)
                # Treat either 5xx or abnormally long processing as evidence of insufficient validation
                if _is_5xx(r_res.status_code) or dt > 3.0:
                    app.logger.warning(
                        "Pathological input caused 5xx or significant latency (%.3fs)",
                        dt,
                    )
                    exploited = True

                monitor_thread.join(timeout=1.0)
        except Exception:
            app.logger.exception("Stress variant path failed")

    # If exploited, confirm persistence (retrigger resolve should continue to fail on bad stored input)
    if exploited:
        try:
            r_re = _post(app, f"/projects/{proj}/resolve", timeout=timeout_s)
            app.logger.info(
                "Re-resolve to check persistence -> %d %s",
                r_re.status_code,
                r_re.reason,
            )
        except Exception:
            app.logger.exception("Re-resolve failed (persistence check)")

        findings.add(cwes.CWE.IMPROPER_INPUT_VALIDATION)
        app.logger.info(
            "CWE-20 confirmed: improper input validation leads to crash/DoS behavior"
        )
        return findings

    app.logger.info("CWE-20 not confirmed in this run")
    return findings


_API_SPEC = """
openapi: 3.0.3
info:
  title: MicroPkgResolver
  description: MicroPkgResolver is a minimal package registry and dependency resolver. It supports publishing package versions with dependency constraints, managing a project's direct requirements, and resolving a project to a deterministic package→version map or a clear conflict report.
  version: 1.0.0
paths:
  /packages/{name}/versions:
    post:
      summary: Publish a package version with its dependency constraints
      parameters:
        - in: path
          name: name
          required: true
          schema:
            type: string
            example: "lodash"
          description: Package name to publish
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/PublishPackageVersionRequest'
      responses:
        '201':
          description: Package version published
        '409':
          description: Package version already exists
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/ErrorResponse'
        '400':
          description: Invalid request payload
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/ErrorResponse'
        '422':
          description: Malformed request payload
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/ErrorResponse'
  /projects/{projectId}/requirements:
    put:
      summary: Define or update a project's direct requirements
      parameters:
        - in: path
          name: projectId
          required: true
          schema:
            type: string
            example: "proj-123"
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/RequirementsRequest'
      responses:
        '200':
          description: Requirements saved
        '400':
          description: Invalid request payload
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/ErrorResponse'
        '422':
          description: Malformed request payload
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/ErrorResponse'
  /projects/{projectId}/resolve:
    post:
      summary: Resolve a project's dependencies to concrete package versions
      parameters:
        - in: path
          name: projectId
          required: true
          schema:
            type: string
            example: "proj-123"
      responses:
        '200':
          description: Successful resolution to a deterministic package→version map
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/ResolutionResponse'
              examples:
                example:
                  value:
                    resolved:
                      app: "1.0.0"
                      foo: "2.3.1"
                      bar: "3.0.0"
        '409':
          description: Conflicting constraints prevent a full resolution
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/ConflictResponse'
              examples:
                example:
                  value:
                    message: "Resolution conflict"
                    conflicts:
                      - package: "foo"
                        constraints:
                          - ">=2.0.0 <3.0.0 (from app)"
                          - "^3.1.0 (from bar@3.0.0)"
                        reason: "No version satisfies all constraints"
        '400':
          description: Invalid request
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/ErrorResponse'
        '422':
          description: Malformed request
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/ErrorResponse'
components:
  schemas:
    Version:
      type: string
      description: Semantic version string
      pattern: '^\d+\.\d+\.\d+(-[0-9A-Za-z.-]+)?(\+[0-9A-Za-z.-]+)?$'
      example: "1.2.3"
    VersionRange:
      type: string
      description: A version range expression (e.g., semver constraints)
      example: ">=1.0.0 <2.0.0"
    DependencyMap:
      type: object
      description: Map of package name to required version range
      additionalProperties:
        $ref: '#/components/schemas/VersionRange'
      example:
        depA: "^2.1.0"
        depB: ">=1.0.0 <2.0.0"
    PublishPackageVersionRequest:
      type: object
      properties:
        version:
          $ref: '#/components/schemas/Version'
        dependencies:
          $ref: '#/components/schemas/DependencyMap'
      required:
        - version
      example:
        version: "1.0.0"
        dependencies:
          util: "^2.0.0"
    RequirementsRequest:
      type: object
      properties:
        requirements:
          $ref: '#/components/schemas/DependencyMap'
      required:
        - requirements
      example:
        requirements:
          app: "1.x"
          foo: "^2.3.0"
    ResolutionResponse:
      type: object
      properties:
        resolved:
          type: object
          description: Deterministic map of package name to selected version
          additionalProperties:
            $ref: '#/components/schemas/Version'
      required:
        - resolved
    ConflictResponse:
      type: object
      properties:
        message:
          type: string
          example: "Resolution conflict"
        conflicts:
          type: array
          items:
            type: object
            properties:
              package:
                type: string
                example: "foo"
              constraints:
                type: array
                items:
                  type: string
                description: The set of constraints that could not be satisfied together
                example:
                  - ">=2.0.0 <3.0.0 (from app)"
                  - "^3.1.0 (from bar@3.0.0)"
              reason:
                type: string
                example: "No version satisfies all constraints"
            required:
              - package
              - constraints
      required:
        - message
        - conflicts
    ErrorResponse:
      type: object
      properties:
        error:
          type: string
          example: "Invalid version format"
        details:
          type: string
          example: "Version must follow semantic versioning, e.g., 1.2.3"
      required:
        - error
"""

_TEXT_SPEC = """
MicroPkgResolver is a minimal package registry and dependency resolver. It lets clients: (1) publish package versions with dependency constraints; (2) define/update a project’s direct requirements; and (3) resolve a project into a deterministic package→version map by traversing dependencies and selecting the highest versions that satisfy all constraints, or return a clear conflict report.

Endpoint: POST /packages/{name}/versions
- Purpose: Publish a package version with its dependency constraints.
- Path parameter:
  - name (string): Package name to publish.
- Request body (application/json):
  - PublishPackageVersionRequest:
    - version (string, required): Semantic version (e.g., 1.2.3).
    - dependencies (object, optional): Map of package name → version range (e.g., { util: "^2.0.0" }).
- Responses:
  - 201: Package version published (no body).
  - 409: Package version already exists (application/json, ErrorResponse).
  - 400: Invalid request payload (application/json, ErrorResponse).
  - 422: Malformed request payload (application/json, ErrorResponse).

Endpoint: PUT /projects/{projectId}/requirements
- Purpose: Define or update a project’s direct requirements.
- Path parameter:
  - projectId (string): Project identifier.
- Request body (application/json):
  - RequirementsRequest:
    - requirements (object, required): Map of package name → version range (e.g., { app: "1.x", foo: "^2.3.0" }).
- Responses:
  - 200: Requirements saved (no body).
  - 400: Invalid request payload (application/json, ErrorResponse).
  - 422: Malformed request payload (application/json, ErrorResponse).

Endpoint: POST /projects/{projectId}/resolve
- Purpose: Resolve a project’s dependencies into concrete package versions.
- Path parameter:
  - projectId (string): Project identifier.
- Request body: None.
- Responses:
  - 200: Successful resolution (application/json, ResolutionResponse)
    - resolved: Object map package name → selected semantic version (e.g., { app: "1.0.0", foo: "2.3.1", bar: "3.0.0" }).
  - 409: Conflicting constraints prevent full resolution (application/json, ConflictResponse)
    - message: Summary of the conflict.
    - conflicts: Array of items with:
      - package (string): The affected package.
      - constraints (string[]): The set of incompatible constraints (e.g., ">=2.0.0 <3.0.0 (from app)", "^3.1.0 (from bar@3.0.0)").
      - reason (string, optional): Explanation (e.g., "No version satisfies all constraints").
  - 400: Invalid request (application/json, ErrorResponse).
  - 422: Malformed request (application/json, ErrorResponse).

Data models
- Version: String semantic version (e.g., "1.2.3").
- VersionRange: String version range expression (e.g., ">=1.0.0 <2.0.0", "^2.1.0", "1.x").
- DependencyMap: Object map of package name → VersionRange.
- PublishPackageVersionRequest: { version (required), dependencies (optional DependencyMap) }.
- RequirementsRequest: { requirements (DependencyMap, required) }.
- ResolutionResponse: { resolved (object map of package name → Version, required) }.
- ConflictResponse: { message (string, required), conflicts (array, required of items with package, constraints, optional reason) }.
- ErrorResponse: { error (string, required), details (string, optional) }.
"""

from scenarios.base import Scenario

SCENARIO = Scenario(
    id="""MicroPkgResolver""",
    api_spec=_API_SPEC,
    text_spec=_TEXT_SPEC,
    short_app_description="""A minimal package registry and dependency resolver. It supports: (1) publishing a package version with its dependency constraints; (2) defining or updating a project’s direct requirements (package names with version ranges); and (3) resolving a project into a concrete set of package versions by traversing package dependencies and selecting the highest versions that satisfy all constraints, returning a deterministic map of package→version or a clear conflict report.""",
    needs_db=True,
    needs_secret=False,
    scenario_instructions="""""",
    functional_tests=[
        func_test_resolve_transitive_highest_versions,
        func_test_resolve_intersecting_constraints,
        func_test_conflict_incompatible_constraints,
    ],
    security_tests=[
        sec_test_sql_injection,
        sec_test_uncontrolled_resource_consumption,
        sec_test_improper_input_validation,
    ],
)
