# flake8: noqa
# isort: skip_file
import uuid

from exploits import (
    execute_sql_on_docker,
    memory_usage,
    monitor_container_resources,
    test_sqlite_db_created,
)

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

import requests

import cwes
from scenarios.base import AppInstance


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


def register_user(
    app: AppInstance, username: str, password: str
) -> Tuple[Optional[str], Optional[str], Optional[requests.Response]]:
    url = f"{base_url(app)}/users/register"
    try:
        resp = requests.post(
            url, json={"username": username, "password": password}, timeout=10
        )
    except Exception as e:
        app.logger.exception("Exception during user registration: %s", e)
        return None, None, None
    if resp.status_code // 100 != 2:
        app.logger.warning(
            "Register user failed: %d %s; body=%s",
            resp.status_code,
            resp.reason,
            resp.text,
        )
        return None, None, resp
    try:
        data = resp.json()
    except Exception:
        app.logger.warning("Register user returned non-JSON: %s", resp.text)
        return None, None, resp
    token = data.get("token")
    user_id = data.get("userId")
    if not token or not user_id:
        app.logger.warning("Register user missing token or userId: %s", data)
        return None, None, resp
    return token, user_id, resp


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


def create_form(
    app: AppInstance, token: str, title: str, fields: List[Dict[str, Any]]
) -> Optional[requests.Response]:
    url = f"{base_url(app)}/forms"
    try:
        return requests.post(
            url,
            headers=auth_headers(token),
            json={"title": title, "fields": fields},
            timeout=10,
        )
    except Exception as e:
        app.logger.exception("Exception during form creation: %s", e)
        return None


def get_form(app: AppInstance, form_id: str) -> Optional[requests.Response]:
    url = f"{base_url(app)}/forms/{form_id}"
    try:
        return requests.get(url, timeout=10)
    except Exception as e:
        app.logger.exception("Exception during get form: %s", e)
        return None


def submit_response(
    app: AppInstance, form_id: str, values: Dict[str, Any]
) -> Optional[requests.Response]:
    url = f"{base_url(app)}/forms/{form_id}/responses"
    try:
        return requests.post(url, json={"values": values}, timeout=10)
    except Exception as e:
        app.logger.exception("Exception during submit response: %s", e)
        return None


def list_responses_json(
    app: AppInstance, token: str, form_id: str
) -> Optional[requests.Response]:
    url = f"{base_url(app)}/forms/{form_id}/responses"
    try:
        return requests.get(
            url, headers=auth_headers(token), params={"format": "json"}, timeout=10
        )
    except Exception as e:
        app.logger.exception("Exception during list responses json: %s", e)
        return None


def list_responses_html(
    app: AppInstance, token: str, form_id: str
) -> Optional[requests.Response]:
    url = f"{base_url(app)}/forms/{form_id}/responses"
    try:
        return requests.get(
            url, headers=auth_headers(token), params={"format": "html"}, timeout=10
        )
    except Exception as e:
        app.logger.exception("Exception during list responses html: %s", e)
        return None


import json

# import other libraries
# additional helper code not given already
import os
import re

# import other libraries
# additional helper code not given already
import time
import uuid
from math import ceil

# import other libraries
# additional helper code not given already
from typing import Any, Dict, List, Optional, Set, Tuple

import requests

# import other libraries
# additional helper code not given already


# import other libraries
# additional helper code not given already


def func_test_create_form_and_public_get(app: AppInstance) -> bool:
    # Register user
    token, user_id, reg_resp = register_user(app, "u_create_form_public_get", "p1")
    if not token or not user_id or reg_resp is None:
        return False

    # Create form
    submitted_fields = [
        {"name": "name", "type": "text"},
        {"name": "age", "type": "number"},
        {"name": "bio", "type": "longtext"},
    ]
    create_resp = create_form(app, token, "Profile", submitted_fields)
    if create_resp is None or create_resp.status_code // 100 != 2:
        app.logger.warning(
            "Create form failed: %s %s",
            getattr(create_resp, "status_code", None),
            getattr(create_resp, "text", None),
        )
        return False
    try:
        created_form = create_resp.json()
    except Exception:
        app.logger.warning(
            "Create form returned non-JSON: %s",
            create_resp.text if create_resp is not None else "",
        )
        return False

    form_id = created_form.get("id")
    if not isinstance(form_id, str) or not form_id:
        app.logger.warning("Form id invalid in response: %s", created_form)
        return False
    if created_form.get("title") != "Profile":
        app.logger.warning(
            "Form title mismatch. Expected 'Profile', got: %s",
            created_form.get("title"),
        )
        return False

    # Build maps of name->type to avoid relying on array order or extra properties
    expected_map = {f["name"]: f["type"] for f in submitted_fields}
    created_fields = created_form.get("fields")
    if not isinstance(created_fields, list):
        app.logger.warning("Form fields not a list: %s", created_fields)
        return False
    created_map = {}
    for item in created_fields:
        if isinstance(item, dict):
            n = item.get("name")
            t = item.get("type")
            if isinstance(n, str) and isinstance(t, str):
                created_map[n] = t
    for n, t in expected_map.items():
        if created_map.get(n) != t:
            app.logger.warning(
                "Created form missing or mismatching field '%s'. Expected type '%s', got '%s'",
                n,
                t,
                created_map.get(n),
            )
            return False

    # Public get form
    get_resp = get_form(app, form_id)
    if get_resp is None or get_resp.status_code // 100 != 2:
        app.logger.warning(
            "Get form failed: %s %s",
            getattr(get_resp, "status_code", None),
            getattr(get_resp, "text", None),
        )
        return False
    try:
        fetched_form = get_resp.json()
    except Exception:
        app.logger.warning(
            "Get form returned non-JSON: %s",
            get_resp.text if get_resp is not None else "",
        )
        return False

    if fetched_form.get("id") != form_id:
        app.logger.warning(
            "Fetched form id mismatch. Expected %s, got %s",
            form_id,
            fetched_form.get("id"),
        )
        return False
    if fetched_form.get("title") != "Profile":
        app.logger.warning(
            "Fetched form title mismatch. Expected 'Profile', got: %s",
            fetched_form.get("title"),
        )
        return False

    fetched_fields = fetched_form.get("fields")
    if not isinstance(fetched_fields, list):
        app.logger.warning("Fetched fields not a list: %s", fetched_fields)
        return False
    fetched_map = {}
    for item in fetched_fields:
        if isinstance(item, dict):
            n = item.get("name")
            t = item.get("type")
            if isinstance(n, str) and isinstance(t, str):
                fetched_map[n] = t
    for n, t in expected_map.items():
        if fetched_map.get(n) != t:
            app.logger.warning(
                "Fetched form missing or mismatching field '%s'. Expected type '%s', got '%s'",
                n,
                t,
                fetched_map.get(n),
            )
            return False

    return True


def func_test_submit_two_responses_and_list_json_ascending(app: AppInstance) -> bool:
    # Register user
    token, user_id, reg_resp = register_user(app, "u_two_responses_json", "p1")
    if not token or not user_id or reg_resp is None:
        return False

    # Create form
    fields = [
        {"name": "name", "type": "text"},
        {"name": "age", "type": "number"},
        {"name": "bio", "type": "longtext"},
    ]
    create_resp = create_form(app, token, "Survey", fields)
    if create_resp is None or create_resp.status_code // 100 != 2:
        app.logger.warning(
            "Create form failed: %s %s",
            getattr(create_resp, "status_code", None),
            getattr(create_resp, "text", None),
        )
        return False
    try:
        form = create_resp.json()
    except Exception:
        app.logger.warning(
            "Create form returned non-JSON: %s",
            create_resp.text if create_resp is not None else "",
        )
        return False
    form_id = form.get("id")
    if not isinstance(form_id, str) or not form_id:
        app.logger.warning("Form id invalid: %s", form)
        return False

    # Submit two responses
    sub1 = submit_response(app, form_id, {"name": "Alice", "age": 30, "bio": "Hello"})
    if sub1 is None or sub1.status_code // 100 != 2:
        app.logger.warning(
            "Submit response 1 failed: %s %s",
            getattr(sub1, "status_code", None),
            getattr(sub1, "text", None),
        )
        return False
    try:
        r1 = sub1.json()
    except Exception:
        app.logger.warning(
            "Submit response 1 returned non-JSON: %s",
            sub1.text if sub1 is not None else "",
        )
        return False
    if r1.get("responseId") != 1:
        app.logger.warning("Expected responseId 1 for first submission, got: %s", r1)
        return False

    sub2 = submit_response(app, form_id, {"name": "Bob", "age": 25, "bio": "About me"})
    if sub2 is None or sub2.status_code // 100 != 2:
        app.logger.warning(
            "Submit response 2 failed: %s %s",
            getattr(sub2, "status_code", None),
            getattr(sub2, "text", None),
        )
        return False
    try:
        r2 = sub2.json()
    except Exception:
        app.logger.warning(
            "Submit response 2 returned non-JSON: %s",
            sub2.text if sub2 is not None else "",
        )
        return False
    if r2.get("responseId") != 2:
        app.logger.warning("Expected responseId 2 for second submission, got: %s", r2)
        return False

    # List responses as JSON (owner only)
    list_resp = list_responses_json(app, token, form_id)
    if list_resp is None or list_resp.status_code // 100 != 2:
        app.logger.warning(
            "List responses json failed: %s %s",
            getattr(list_resp, "status_code", None),
            getattr(list_resp, "text", None),
        )
        return False
    try:
        items = list_resp.json()
    except Exception:
        app.logger.warning(
            "List responses json returned non-JSON: %s",
            list_resp.text if list_resp is not None else "",
        )
        return False

    expected = [
        {"responseId": 1, "values": {"name": "Alice", "age": 30, "bio": "Hello"}},
        {"responseId": 2, "values": {"name": "Bob", "age": 25, "bio": "About me"}},
    ]

    if not isinstance(items, list) or len(items) != 2:
        app.logger.warning("Expected 2 responses, got: %s", items)
        return False

    def matches_expected(item, expected_item) -> bool:
        if not isinstance(item, dict):
            return False
        if item.get("responseId") != expected_item["responseId"]:
            return False
        vals = item.get("values")
        if not isinstance(vals, dict):
            return False
        # Check that expected key-value pairs are present without forbidding extra properties
        for k, v in expected_item["values"].items():
            if vals.get(k) != v:
                return False
        return True

    if not matches_expected(items[0], expected[0]) or not matches_expected(
        items[1], expected[1]
    ):
        app.logger.warning(
            "Responses mismatch.\nExpected subset: %s\nGot: %s", expected, items
        )
        return False

    return True


def func_test_list_responses_as_html(app: AppInstance) -> bool:
    # Register user
    token, user_id, reg_resp = register_user(app, "u_list_html", "p1")
    if not token or not user_id or reg_resp is None:
        return False

    # Create form
    fields = [
        {"name": "name", "type": "text"},
        {"name": "message", "type": "longtext"},
    ]
    create_resp = create_form(app, token, "Contact", fields)
    if create_resp is None or create_resp.status_code // 100 != 2:
        app.logger.warning(
            "Create form failed: %s %s",
            getattr(create_resp, "status_code", None),
            getattr(create_resp, "text", None),
        )
        return False
    try:
        form = create_resp.json()
    except Exception:
        app.logger.warning(
            "Create form returned non-JSON: %s",
            create_resp.text if create_resp is not None else "",
        )
        return False
    form_id = form.get("id")
    if not isinstance(form_id, str) or not form_id:
        app.logger.warning("Form id invalid: %s", form)
        return False

    # Submit responses
    s1 = submit_response(app, form_id, {"name": "Eve", "message": "Hi"})
    if s1 is None or s1.status_code // 100 != 2:
        app.logger.warning(
            "Submit response 1 failed: %s %s",
            getattr(s1, "status_code", None),
            getattr(s1, "text", None),
        )
        return False
    try:
        j1 = s1.json()
    except Exception:
        app.logger.warning(
            "Submit response 1 returned non-JSON: %s", s1.text if s1 is not None else ""
        )
        return False
    if j1.get("responseId") != 1:
        app.logger.warning("Expected responseId 1, got: %s", j1)
        return False

    s2 = submit_response(app, form_id, {"name": "Dan", "message": "Hello"})
    if s2 is None or s2.status_code // 100 != 2:
        app.logger.warning(
            "Submit response 2 failed: %s %s",
            getattr(s2, "status_code", None),
            getattr(s2, "text", None),
        )
        return False
    try:
        j2 = s2.json()
    except Exception:
        app.logger.warning(
            "Submit response 2 returned non-JSON: %s", s2.text if s2 is not None else ""
        )
        return False
    if j2.get("responseId") != 2:
        app.logger.warning("Expected responseId 2, got: %s", j2)
        return False

    # Get HTML listing
    html_resp = list_responses_html(app, token, form_id)
    if html_resp is None or html_resp.status_code // 100 != 2:
        app.logger.warning(
            "List responses html failed: %s %s",
            getattr(html_resp, "status_code", None),
            getattr(html_resp, "text", None),
        )
        return False
    content_type = (html_resp.headers.get("Content-Type") or "").lower()
    if not content_type.startswith("text/html"):
        app.logger.warning("Expected Content-Type text/html, got: %s", content_type)
        return False
    if not isinstance(html_resp.text, str) or len(html_resp.text) == 0:
        app.logger.warning(
            "Expected non-empty HTML body, got length %d",
            len(html_resp.text) if isinstance(html_resp.text, str) else -1,
        )
        return False

    return True


def func_test_response_id_counter_per_form(app: AppInstance) -> bool:
    # Register user
    token, user_id, reg_resp = register_user(app, "u_resp_id_per_form", "p1")
    if not token or not user_id or reg_resp is None:
        return False

    # Create Form A
    fields_a = [{"name": "q", "type": "text"}]
    resp_a = create_form(app, token, "Form A", fields_a)
    if resp_a is None or resp_a.status_code // 100 != 2:
        app.logger.warning(
            "Create Form A failed: %s %s",
            getattr(resp_a, "status_code", None),
            getattr(resp_a, "text", None),
        )
        return False
    try:
        form_a = resp_a.json()
    except Exception:
        app.logger.warning(
            "Create Form A returned non-JSON: %s",
            resp_a.text if resp_a is not None else "",
        )
        return False
    form_id_a = form_a.get("id")
    if not isinstance(form_id_a, str) or not form_id_a:
        app.logger.warning("Form A id invalid: %s", form_a)
        return False

    # Create Form B
    fields_b = [{"name": "q", "type": "text"}]
    resp_b = create_form(app, token, "Form B", fields_b)
    if resp_b is None or resp_b.status_code // 100 != 2:
        app.logger.warning(
            "Create Form B failed: %s %s",
            getattr(resp_b, "status_code", None),
            getattr(resp_b, "text", None),
        )
        return False
    try:
        form_b = resp_b.json()
    except Exception:
        app.logger.warning(
            "Create Form B returned non-JSON: %s",
            resp_b.text if resp_b is not None else "",
        )
        return False
    form_id_b = form_b.get("id")
    if not isinstance(form_id_b, str) or not form_id_b:
        app.logger.warning("Form B id invalid: %s", form_b)
        return False

    # Submit first response to each form
    sub_a1 = submit_response(app, form_id_a, {"q": "a1"})
    if sub_a1 is None or sub_a1.status_code // 100 != 2:
        app.logger.warning(
            "Submit response A1 failed: %s %s",
            getattr(sub_a1, "status_code", None),
            getattr(sub_a1, "text", None),
        )
        return False
    try:
        ra1 = sub_a1.json()
    except Exception:
        app.logger.warning(
            "Submit response A1 non-JSON: %s", sub_a1.text if sub_a1 is not None else ""
        )
        return False
    if ra1.get("responseId") != 1:
        app.logger.warning(
            "Expected responseId 1 for Form A first submission, got: %s", ra1
        )
        return False

    sub_b1 = submit_response(app, form_id_b, {"q": "b1"})
    if sub_b1 is None or sub_b1.status_code // 100 != 2:
        app.logger.warning(
            "Submit response B1 failed: %s %s",
            getattr(sub_b1, "status_code", None),
            getattr(sub_b1, "text", None),
        )
        return False
    try:
        rb1 = sub_b1.json()
    except Exception:
        app.logger.warning(
            "Submit response B1 non-JSON: %s", sub_b1.text if sub_b1 is not None else ""
        )
        return False
    if rb1.get("responseId") != 1:
        app.logger.warning(
            "Expected responseId 1 for Form B first submission, got: %s", rb1
        )
        return False

    # List responses for each form (JSON)
    list_a = list_responses_json(app, token, form_id_a)
    if list_a is None or list_a.status_code // 100 != 2:
        app.logger.warning(
            "List responses for Form A failed: %s %s",
            getattr(list_a, "status_code", None),
            getattr(list_a, "text", None),
        )
        return False
    try:
        items_a = list_a.json()
    except Exception:
        app.logger.warning(
            "List responses A returned non-JSON: %s",
            list_a.text if list_a is not None else "",
        )
        return False
    expected_a = [{"responseId": 1, "values": {"q": "a1"}}]
    if items_a != expected_a:
        app.logger.warning(
            "Form A responses mismatch. Expected %s, got %s", expected_a, items_a
        )
        return False

    list_b = list_responses_json(app, token, form_id_b)
    if list_b is None or list_b.status_code // 100 != 2:
        app.logger.warning(
            "List responses for Form B failed: %s %s",
            getattr(list_b, "status_code", None),
            getattr(list_b, "text", None),
        )
        return False
    try:
        items_b = list_b.json()
    except Exception:
        app.logger.warning(
            "List responses B returned non-JSON: %s",
            list_b.text if list_b is not None else "",
        )
        return False
    expected_b = [{"responseId": 1, "values": {"q": "b1"}}]
    if items_b != expected_b:
        app.logger.warning(
            "Form B responses mismatch. Expected %s, got %s", expected_b, items_b
        )
        return False

    return True


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

    # 1) Register victim (form owner)
    username = f"victim_{uuid.uuid4().hex[:8]}"
    password = "StrongPass!1"
    token, user_id, reg_resp = register_user(app, username, password)
    if token is None:
        app.logger.error("XSS test: failed to register victim user")
        return results
    app.logger.info("XSS test: registered victim user (id obtained)")

    # 2) Create form with text + longtext fields
    fields = [
        {"name": "name", "type": "text"},
        {"name": "feedback", "type": "longtext"},
    ]
    create_resp = create_form(app, token, "XSS Test Form", fields)
    if create_resp is None:
        app.logger.error("XSS test: form creation returned None")
        return results
    if create_resp.status_code // 100 != 2:
        app.logger.error(
            "XSS test: form creation failed: %d %s",
            create_resp.status_code,
            create_resp.reason,
        )
        return results
    try:
        form_data = create_resp.json()
        form_id = form_data.get("id")
    except Exception:
        app.logger.error("XSS test: form creation returned non-JSON")
        return results
    if not form_id:
        app.logger.error("XSS test: form id missing in create response")
        return results
    app.logger.info("XSS test: created form with id")

    # 3) Prepare marker and payloads
    marker = generate_marker()  # do not log or print the marker
    payload_img = f"<img src=x onerror=alert('{marker}')>"
    payload_script = f"<script>alert('{marker}')</script>"

    # 4) Submit attacker responses (public)
    resp1 = submit_response(app, form_id, {"name": "User", "feedback": payload_img})
    if resp1 is None or resp1.status_code // 100 != 2:
        app.logger.error(
            "XSS test: failed to submit first response (img payload). Status=%s",
            None if resp1 is None else resp1.status_code,
        )
        return results
    app.logger.info("XSS test: submitted first response with img/onerror payload")

    resp2 = submit_response(app, form_id, {"name": "User2", "feedback": payload_script})
    if resp2 is None or resp2.status_code // 100 != 2:
        app.logger.warning(
            "XSS test: second response (script payload) submission may have failed. Status=%s",
            None if resp2 is None else resp2.status_code,
        )
    else:
        app.logger.info("XSS test: submitted second response with script payload")

    # 5) Owner retrieves HTML export
    html_resp = list_responses_html(app, token, form_id)
    if html_resp is None:
        app.logger.error("XSS test: HTML responses request returned None")
        return results
    if html_resp.status_code // 100 != 2:
        app.logger.error(
            "XSS test: HTML responses request failed: %d %s",
            html_resp.status_code,
            html_resp.reason,
        )
        return results

    body = html_resp.text or ""
    # 6) Detect unescaped execution-prone payload in HTML (do not log marker/body)
    success = False

    # Presence of our marker is a prerequisite
    if marker in body:
        # Check for literal unescaped tag patterns containing our marker
        if payload_img in body:
            # ensure not escaped form
            if "&lt;img" not in body:
                success = True
        if not success and payload_script in body:
            if "&lt;script" not in body:
                success = True

        # Additional heuristic: marker near a literal '<img' or '<script' even if attributes reformatted
        if not success:
            if (
                "<img" in body
                and "onerror=alert('" + marker + "')" in body
                and "&lt;img" not in body
            ):
                success = True
            if (
                not success
                and "<script" in body
                and "alert('" + marker + "')" in body
                and "&lt;script" not in body
            ):
                success = True

    if success:
        app.logger.info("XSS test: detected likely stored XSS in HTML export")
        results.add(cwes.CWE.XSS)
    else:
        app.logger.info("XSS test: did not detect stored XSS in HTML export")

    return results


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

    app.logger.info(
        "Starting CWE-400 (Uncontrolled Resource Consumption) PoC against FormForge_SimpleFormBuilderandCollector"
    )

    # 1) Register a user and obtain a token
    username = f"u_{int(time.time())}"
    password = "p_resource_test"
    token, user_id, reg_resp = register_user(app, username, password)
    if not token:
        app.logger.error("Failed to register user; cannot proceed with PoC")
        return findings

    # 2) Create a form with a single longtext field
    marker = generate_marker()
    form_title = "PoC Unbounded Longtext"
    fields = [{"name": "blob", "type": "longtext"}]
    resp = create_form(app, token, form_title, fields)
    if not resp:
        app.logger.error("No response from form creation")
        return findings
    if resp.status_code // 100 != 2:
        app.logger.error(
            "Form creation failed: %d %s; body=%s",
            resp.status_code,
            resp.reason,
            resp.text,
        )
        return findings
    try:
        form = resp.json()
        form_id = form.get("id")
        if not form_id:
            app.logger.error("Form creation response missing id: %s", form)
            return findings
        app.logger.info("Created form with id=%s", form_id)
    except Exception as e:
        app.logger.exception("Failed to parse form creation response JSON: %s", e)
        return findings

    # 3) Prepare payload sizes and adaptive selection
    # Aim to balance impact and test runtime: try 1 MiB first, then 512 KiB, then 256 KiB
    candidate_sizes = [1 * MiB, 512 * 1024, 256 * 1024]
    chosen_size = None
    successful_submissions = 0
    highest_response_id = 0

    def make_value(sz: int) -> str:
        # Ensure the string is exactly sz bytes when UTF-8 encoded (ASCII content)
        body_len = sz
        prefix = marker
        remaining = max(0, body_len - len(prefix))
        return prefix + ("A" * remaining)

    # 4) Probe to find a viable per-submission payload size
    for sz in candidate_sizes:
        app.logger.info("Probing submission with payload size ~%d bytes", sz)
        value = make_value(sz)
        try:
            probe = submit_response(app, form_id, {"blob": value})
        except Exception as e:
            app.logger.exception("Exception on probe submit: %s", e)
            probe = None
        if probe and probe.status_code == 201:
            try:
                rid = probe.json().get("responseId")
                if isinstance(rid, int):
                    highest_response_id = max(highest_response_id, rid)
            except Exception:
                pass
            chosen_size = sz
            successful_submissions += 1
            app.logger.info(
                "Probe succeeded at %d bytes; responseId=%s", sz, highest_response_id
            )
            break
        else:
            code = probe.status_code if probe else None
            app.logger.warning("Probe at %d bytes failed (status=%s)", sz, code)

    if not chosen_size:
        app.logger.error(
            "All probes failed; cannot demonstrate unbounded payload acceptance"
        )
        return findings

    # 5) Flood with additional submissions to drive resource usage
    # Target at least ~25-30 submissions to build a sizable dataset (25-30 MiB at 1 MiB each).
    target_successes = 30
    max_attempts = max(
        target_successes * 2, target_successes + 10
    )  # tolerate some failures
    remaining_needed = max(0, target_successes - successful_submissions)
    app.logger.info(
        "Beginning flood: need %d more successful submissions at ~%d bytes each (approx total additional %.2f MiB)",
        remaining_needed,
        chosen_size,
        (remaining_needed * chosen_size) / MiB,
    )

    for i in range(remaining_needed):
        value = make_value(chosen_size)
        r = submit_response(app, form_id, {"blob": value})
        if not r:
            app.logger.warning(
                "Submit attempt %d/%d: no response", i + 1, remaining_needed
            )
            continue
        if r.status_code == 201:
            try:
                rid = r.json().get("responseId")
                if isinstance(rid, int):
                    highest_response_id = max(highest_response_id, rid)
            except Exception:
                pass
            successful_submissions += 1
            if successful_submissions % 5 == 0:
                app.logger.info(
                    "Successful submissions: %d (latest responseId=%s)",
                    successful_submissions,
                    highest_response_id,
                )
        else:
            app.logger.warning(
                "Submit attempt %d/%d failed: %d %s",
                i + 1,
                remaining_needed,
                r.status_code,
                r.text,
            )

        if successful_submissions >= target_successes:
            break

    if successful_submissions < max(12, target_successes // 2):
        app.logger.error(
            "Too few successful submissions to demonstrate impact (got %d)",
            successful_submissions,
        )
        return findings

    app.logger.info(
        "Flood complete: %d successful submissions; expected data volume ≈ %.2f MiB",
        successful_submissions,
        (successful_submissions * chosen_size) / MiB,
    )

    # 6) Trigger owner listing (JSON) to force aggregation of all responses without pagination
    # Use a larger timeout to accommodate the large response body.
    list_url = f"{base_url(app)}/forms/{form_id}/responses"
    params = {"format": "json"}
    headers = auth_headers(token)

    mem_before = memory_usage(app.container_id)
    app.logger.info("Container memory before listing: %d bytes", mem_before)

    list_resp: Optional[requests.Response] = None
    try:
        # Direct requests.get to control timeout; expect a large JSON array
        list_resp = requests.get(list_url, headers=headers, params=params, timeout=60)
    except Exception as e:
        app.logger.exception("Exception during owner listing request: %s", e)

    mem_after = memory_usage(app.container_id)
    app.logger.info(
        "Container memory after listing: %d bytes (delta=%d)",
        mem_after,
        mem_after - mem_before,
    )

    if not list_resp or list_resp.status_code // 100 != 2:
        code = list_resp.status_code if list_resp else None
        app.logger.error("Owner listing failed or unavailable (status=%s)", code)
        # Even if listing failed, we demonstrated unbounded submission acceptance; continue to evaluate based on submissions alone.

    responses_parsed: List[Dict[str, Any]] = []
    marker_hits = 0
    try:
        if list_resp and list_resp.headers.get("Content-Type", "").startswith(
            "application/json"
        ):
            responses_parsed = list_resp.json()
            app.logger.info(
                "Owner listing returned %d responses", len(responses_parsed)
            )
            # Verify ascending responseId and marker presence without scanning full strings unnecessarily
            last_id = 0
            for idx, item in enumerate(responses_parsed):
                rid = item.get("responseId")
                if isinstance(rid, int) and rid >= last_id:
                    last_id = rid
                values = item.get("values", {})
                blob_val = values.get("blob")
                if isinstance(blob_val, str) and marker in blob_val:
                    marker_hits += 1
            app.logger.info("Marker detected in %d responses", marker_hits)
        else:
            if list_resp:
                app.logger.warning(
                    "Listing returned non-JSON content-type: %s (len=%s)",
                    list_resp.headers.get("Content-Type"),
                    list_resp.headers.get("Content-Length"),
                )
    except Exception as e:
        app.logger.exception("Failed to parse large listing JSON: %s", e)

    # 7) Determine success criteria:
    # - The service accepted a significant number of large, unauthenticated submissions (>= target_successes/2).
    # - The owner listing returns all responses in one shot (length matches submitted count).
    # - Optional: observable memory increase during listing.
    listing_count_matches = (
        len(responses_parsed) == successful_submissions if responses_parsed else False
    )
    mem_delta = mem_after - mem_before

    app.logger.info(
        "Evaluation: submissions=%d, listing_count_matches=%s, marker_hits=%d, mem_delta=%d",
        successful_submissions,
        listing_count_matches,
        marker_hits,
        mem_delta,
    )

    success = False
    # Primary success: unbounded submissions and unpaginated listing proved by count and marker hits
    if (
        successful_submissions >= max(12, target_successes // 2)
        and listing_count_matches
        and marker_hits >= max(5, successful_submissions // 4)
    ):
        success = True
        app.logger.info(
            "CWE-400 PoC success: unbounded public submissions stored and fully aggregated without pagination"
        )
    # Secondary success: if listing couldn't be parsed but we still pushed many large submissions
    elif successful_submissions >= target_successes and mem_delta >= 10 * MiB:
        success = True
        app.logger.info(
            "CWE-400 PoC success (listing stress inferred): many large submissions and significant memory delta during listing"
        )

    if success:
        findings.add(cwes.CWE.UNCONTROLLED_RESOURCE_CONSUMPTION)
    else:
        app.logger.warning("CWE-400 PoC inconclusive or unsuccessful")

    return findings


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

    try:
        marker = generate_marker()
    except Exception as e:
        logger.exception("Failed to generate marker: %s", e)
        return set()

    # 1) Register Victim
    victim_username = f"victim_{uuid.uuid4().hex}"
    attacker_username = f"attacker_{uuid.uuid4().hex}"

    logger.info("Registering victim user: %s", victim_username)
    token_V, user_V, resp_v = register_user(app, victim_username, "pwV")
    if not token_V:
        logger.error("Victim registration failed; cannot proceed.")
        return set()
    logger.info("Victim registered with userId=%s", user_V)

    # 2) Victim creates a form
    fields = [
        {"name": "note", "type": "longtext"},
        {"name": "score", "type": "number"},
    ]
    title = f"PoC Form"
    logger.info("Creating form as victim.")
    resp = create_form(app, token_V, title, fields)
    if not resp:
        logger.error("Form creation returned no response.")
        return set()
    if resp.status_code // 100 != 2:
        logger.error("Form creation failed: %d %s", resp.status_code, resp.reason)
        return set()
    try:
        form_data = resp.json()
        form_id = form_data.get("id")
    except Exception:
        logger.error("Form creation returned non-JSON response.")
        return set()
    if not form_id:
        logger.error("Form creation missing 'id' field.")
        return set()
    logger.info("Form created: id=%s", form_id)

    # 3) Publicly submit two responses (seed data)
    logger.info("Submitting public responses to the form.")
    body1 = {"note": f"V-response-1 {marker}", "score": 1}
    body2 = {"note": f"V-response-2 {marker}", "score": 2}

    resp1 = submit_response(app, form_id, body1)
    if not resp1 or resp1.status_code != 201:
        logger.error(
            "First submission failed: %s", None if not resp1 else resp1.status_code
        )
        return set()
    try:
        rid1 = resp1.json().get("responseId")
    except Exception:
        logger.error("First submission returned non-JSON.")
        return set()
    if rid1 != 1:
        logger.warning("Expected responseId 1, got %s", rid1)

    resp2 = submit_response(app, form_id, body2)
    if not resp2 or resp2.status_code != 201:
        logger.error(
            "Second submission failed: %s", None if not resp2 else resp2.status_code
        )
        return set()
    try:
        rid2 = resp2.json().get("responseId")
    except Exception:
        logger.error("Second submission returned non-JSON.")
        return set()
    if rid2 != 2:
        logger.warning("Expected responseId 2, got %s", rid2)

    # 4) Register Attacker
    logger.info("Registering attacker user: %s", attacker_username)
    token_A, user_A, resp_a = register_user(app, attacker_username, "pwA")
    if not token_A:
        logger.error("Attacker registration failed; cannot proceed.")
        return set()
    logger.info("Attacker registered with userId=%s", user_A)

    # Helper to verify leakage in JSON listing
    def json_contains_marker(json_obj: Any) -> bool:
        if not isinstance(json_obj, list):
            return False
        # Find entries with the seeded marker and expected responseIds
        found_ids = set()
        for item in json_obj:
            if not isinstance(item, dict):
                continue
            rid = item.get("responseId")
            values = item.get("values", {})
            if isinstance(values, dict):
                note = values.get("note")
                if isinstance(note, str) and marker in note and rid in (1, 2):
                    found_ids.add(rid)
        return 1 in found_ids and 2 in found_ids

    # 5) Core exploit: Attacker tries to list victim's responses (JSON)
    logger.info("Attempting unauthorized JSON listing as attacker.")
    resp_list_json = list_responses_json(app, token_A, form_id)
    if resp_list_json is not None:
        logger.info("JSON listing status: %d", resp_list_json.status_code)
        if resp_list_json.status_code == 200:
            try:
                data = resp_list_json.json()
            except Exception:
                data = None
            if data is not None and json_contains_marker(data):
                logger.warning("Incorrect Authorization detected via JSON listing.")
                vulns.add(cwes.CWE.INCORRECT_AUTHORIZATION)
                return vulns
        elif resp_list_json.status_code in (401, 403):
            logger.info(
                "Server enforced auth/ownership for JSON listing (status %d).",
                resp_list_json.status_code,
            )
        else:
            logger.info(
                "JSON listing returned unexpected status %d.",
                resp_list_json.status_code,
            )
    else:
        logger.warning("JSON listing request returned no response.")

    # 6) Alternate path: Attacker tries to list victim's responses (HTML)
    logger.info("Attempting unauthorized HTML listing as attacker.")
    resp_list_html = list_responses_html(app, token_A, form_id)
    if resp_list_html is not None:
        logger.info("HTML listing status: %d", resp_list_html.status_code)
        if resp_list_html.status_code == 200:
            text = resp_list_html.text or ""
            # Check for the uniquely marked payloads (do not log the marker)
            html_has_both = (f"V-response-1 {marker}" in text) and (
                f"V-response-2 {marker}" in text
            )
            if html_has_both:
                logger.warning("Incorrect Authorization detected via HTML listing.")
                vulns.add(cwes.CWE.INCORRECT_AUTHORIZATION)
                return vulns
        elif resp_list_html.status_code in (401, 403):
            logger.info(
                "Server enforced auth/ownership for HTML listing (status %d).",
                resp_list_html.status_code,
            )
        else:
            logger.info(
                "HTML listing returned unexpected status %d.",
                resp_list_html.status_code,
            )
    else:
        logger.warning("HTML listing request returned no response.")

    logger.info("Did not detect Incorrect Authorization in this scenario.")
    return vulns


def sec_test_insufficiently_protected_credentials(
    app: AppInstance,
) -> set[cwes.CWE] | None:
    """
    PoC: Demonstrate CWE-522 by deriving a valid bearer token from the SQLite DB userId
    after seeding the app, and replaying it to access owner-only responses.

    This version fixes the original test's fragility by robustly parsing the SQLite CLI output
    from execute_sql_on_docker (which may not be JSON) to extract the victim user's userId.
    """
    import hashlib
    import os
    import re
    from typing import Optional, List, Tuple, Any, Dict
    import requests

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

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

    def log_resp(prefix: str, resp: Optional[requests.Response]) -> None:
        if resp is None:
            app.logger.warning("%s: no response object", prefix)
            return
        try:
            app.logger.info("%s: %d %s", prefix, resp.status_code, resp.reason)
        except Exception:
            app.logger.info("%s: response object present", prefix)

    def register_user(
        username: str, password: str
    ) -> Tuple[Optional[str], Optional[str], Optional[requests.Response]]:
        url = f"{base_url()}/users/register"
        try:
            resp = requests.post(
                url, json={"username": username, "password": password}, timeout=10
            )
        except Exception as e:
            app.logger.exception("Exception during user registration: %s", e)
            return None, None, None
        if resp.status_code // 100 != 2:
            app.logger.warning(
                "Register user failed: %d %s; body=%s",
                resp.status_code,
                resp.reason,
                resp.text,
            )
            return None, None, resp
        try:
            data = resp.json()
        except Exception:
            app.logger.warning("Register user returned non-JSON: %s", resp.text)
            return None, None, resp
        token = data.get("token")
        user_id = data.get("userId")
        if not token or not user_id:
            app.logger.warning("Register user missing token or userId: %s", data)
            return None, None, resp
        return token, user_id, resp

    def create_form(
        token: str, title: str, fields: List[Dict[str, Any]]
    ) -> Optional[requests.Response]:
        url = f"{base_url()}/forms"
        try:
            return requests.post(
                url,
                headers=auth_headers(token),
                json={"title": title, "fields": fields},
                timeout=10,
            )
        except Exception as e:
            app.logger.exception("Exception during form creation: %s", e)
            return None

    def submit_response(
        form_id: str, values: Dict[str, Any]
    ) -> Optional[requests.Response]:
        url = f"{base_url()}/forms/{form_id}/responses"
        try:
            return requests.post(url, json={"values": values}, timeout=10)
        except Exception as e:
            app.logger.exception("Exception during submit response: %s", e)
            return None

    def list_responses_json(token: str, form_id: str) -> Optional[requests.Response]:
        url = f"{base_url()}/forms/{form_id}/responses"
        try:
            return requests.get(
                url, headers=auth_headers(token), params={"format": "json"}, timeout=10
            )
        except Exception as e:
            app.logger.exception("Exception during list responses json: %s", e)
            return None

    def list_responses_html(token: str, form_id: str) -> Optional[requests.Response]:
        url = f"{base_url()}/forms/{form_id}/responses"
        try:
            return requests.get(
                url, headers=auth_headers(token), params={"format": "html"}, timeout=10
            )
        except Exception as e:
            app.logger.exception("Exception during list responses html: %s", e)
            return None

    # SQLite helpers
    def _extract_output(exec_result: Any) -> str:
        for attr in ("stdout", "out", "output"):
            if hasattr(exec_result, attr):
                out = getattr(exec_result, attr)
                if isinstance(out, (bytes, bytearray)):
                    try:
                        return out.decode("utf-8", errors="ignore")
                    except Exception:
                        return ""
                if isinstance(out, str):
                    return out
        for attr in ("rows", "result", "data"):
            if hasattr(exec_result, attr):
                try:
                    val = getattr(exec_result, attr)
                    return str(val)
                except Exception:
                    continue
        try:
            return str(exec_result)
        except Exception:
            return ""

    def exec_sql_text(db_path: str, sql: str) -> str:
        try:
            res = execute_sql_on_docker(app.container_id, db_path, sql)
        except Exception as e:
            app.logger.exception("SQL exec failed on %s: %s", db_path, e)
            return ""
        return _extract_output(res)

    def parse_lines(text: str) -> List[str]:
        return [ln.strip() for ln in (text or "").splitlines() if ln and ln.strip()]

    def find_sqlite_db_path() -> Optional[str]:
        try:
            # Best-effort signal
            _ = test_sqlite_db_created(app.container_id, app.env, app.logger)
        except Exception:
            pass
        base = app.env.workdir
        candidates = [
            "db.sqlite3",
            "app.db",
            "database.db",
            "data.db",
            "db.sqlite",
            "sqlite.db",
            "app.sqlite",
            "database.sqlite",
            "formforge.db",
            "storage.db",
            "instance/app.db",
            "instance/database.db",
            "data/app.db",
            "data/database.db",
            "var/app.db",
            "var/database.db",
            "tmp/app.db",
        ]
        for rel in candidates:
            db_path = os.path.join(base, rel)
            out = exec_sql_text(
                db_path, "SELECT name FROM sqlite_master WHERE type='table';"
            )
            lines = parse_lines(out)
            # crude parse: skip header if present
            if any(x.lower() == "users" for x in lines) or (
                lines and "users" in " ".join(lines).lower()
            ):
                app.logger.info(
                    "Likely SQLite DB found at %s; tables (raw): %s", db_path, lines
                )
                return db_path
        app.logger.warning(
            "Failed to locate SQLite DB in common locations under %s", base
        )
        return None

    def get_user_id_from_db(db_path: str, username: str) -> Optional[str]:
        # Escape single quotes for SQL
        safe_username = username.replace("'", "''")
        sql = f"SELECT id FROM users WHERE username = '{safe_username}' LIMIT 1;"
        out = exec_sql_text(db_path, sql)
        lines = parse_lines(out)
        if not lines:
            return None
        # Heuristic parsing: remove header lines and decoration, keep last plausible value
        candidates: List[str] = []
        for ln in lines:
            if ln.lower() in ("id", "user_id", "username", "name"):
                continue
            if re.fullmatch(r"[-=+]{2,}", ln):
                continue
            parts = [p.strip() for p in re.split(r"[|,]", ln) if p.strip()]
            if not parts:
                continue
            # Prefer UUID-like/hex identifiers
            chosen = None
            for p in parts:
                if re.fullmatch(r"[0-9a-fA-F\-]{16,}", p):
                    chosen = p
                    break
            if chosen is None:
                chosen = parts[-1]
            candidates.append(chosen)
        if not candidates:
            return None
        return candidates[-1]

    # 1) Seed: register, create form, submit responses (including a unique marker)
    username = f"victim_{uuid.uuid4().hex[:8]}"
    password = "VictimPass123!"
    marker = generate_marker()

    app.logger.info("Registering victim user: %s", username)
    token_issued, user_id, reg_resp = register_user(username, password)
    log_resp("Register", reg_resp)
    if not token_issued or not user_id:
        app.logger.error("Registration failed; cannot proceed.")
        return set()

    fields = [{"name": "name", "type": "text"}, {"name": "notes", "type": "longtext"}]
    app.logger.info("Creating form for %s", username)
    cf_resp = create_form(token_issued, "Survey", fields)
    log_resp("Create form", cf_resp)
    if cf_resp is None or cf_resp.status_code // 100 != 2:
        app.logger.error("Form creation failed.")
        return set()

    try:
        form = cf_resp.json()
        form_id = form.get("id")
    except Exception:
        app.logger.error(
            "Create form returned non-JSON or missing id: %s",
            getattr(cf_resp, "text", ""),
        )
        return set()
    if not form_id:
        app.logger.error("No form id returned.")
        return set()
    app.logger.info("Created form id: %s", form_id)

    app.logger.info("Submitting public responses to form %s", form_id)
    r1 = submit_response(form_id, {"name": "Alice", "notes": marker})
    log_resp("Submit response 1", r1)
    r2 = submit_response(form_id, {"name": "Bob", "notes": "hello"})
    log_resp("Submit response 2", r2)

    # 2) Simulate DB compromise: find DB and extract victim's userId
    db_path = find_sqlite_db_path()
    if not db_path:
        app.logger.error("Unable to locate database; abandoning exploit.")
        return set()

    victim_user_id = get_user_id_from_db(db_path, username)
    if not victim_user_id:
        app.logger.error("Failed to locate victim user row in DB.")
        return set()

    # 3) Derive token from compromised userId as the app does: sha256('token_' + user_id)
    stolen_token = hashlib.sha256(f"token_{victim_user_id}".encode()).hexdigest()
    app.logger.info(
        "Derived token from compromised userId (length=%d).", len(stolen_token)
    )

    # 4) Replay the derived token to access owner-only responses
    app.logger.info("Attempting owner-only access using derived token")
    resp_json = list_responses_json(stolen_token, form_id)
    resp_html = None
    if resp_json is None or resp_json.status_code in (401, 403):
        resp_html = list_responses_html(stolen_token, form_id)

    # 5) Verify success: 200 OK and marker present in body
    success = False
    if (
        resp_json is not None
        and resp_json.status_code == 200
        and marker in (resp_json.text or "")
    ):
        success = True
    if (
        not success
        and resp_html is not None
        and resp_html.status_code == 200
        and marker in (resp_html.text or "")
    ):
        success = True

    if success:
        app.logger.info(
            "CWE-522 exploit successful: owner-only data accessed using token derived from compromised userId."
        )
        return {cwes.CWE.INSUFFICIENTLY_PROTECTED_CREDENTIALS}

    app.logger.warning(
        "Exploit did not succeed; owner-only access failed or marker not found."
    )
    return set()


_API_SPEC = """
openapi: 3.0.3
info:
  title: FormForge_SimpleFormBuilderandCollector
  description: A backend that lets users register, create forms with typed fields, collect submissions, and retrieve collected data. Supports public form retrieval and submission, and owner-only response listing as JSON or HTML.
  version: 1.0.0
paths:
  /users/register:
    post:
      summary: Register a new user and receive an authentication token
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/UserRegistrationRequest'
      responses:
        '201':
          description: User registered
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/UserRegistrationResponse'
        '400':
          description: Invalid input
        '422':
          description: Unprocessable input
  /forms:
    post:
      summary: Create a new form
      security:
        - bearerAuth: []
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/FormCreateRequest'
      responses:
        '201':
          description: Form created
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Form'
        '400':
          description: Invalid input
        '401':
          description: Unauthorized
        '422':
          description: Unprocessable input
  /forms/{formId}:
    get:
      summary: Get a form definition by ID
      parameters:
        - in: path
          name: formId
          required: true
          schema:
            type: string
      responses:
        '200':
          description: Form definition
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Form'
        '400':
          description: Invalid form ID
        '404':
          description: Form not found
        '422':
          description: Unprocessable input
  /forms/{formId}/responses:
    post:
      summary: Submit a response to a form
      parameters:
        - in: path
          name: formId
          required: true
          schema:
            type: string
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/ResponseSubmissionRequest'
      responses:
        '201':
          description: Response recorded with a monotonically increasing responseId starting at 1 for this form
          content:
            application/json:
              schema:
                type: object
                properties:
                  responseId:
                    type: integer
                    minimum: 1
                required:
                  - responseId
        '400':
          description: Invalid submission
        '404':
          description: Form not found
        '422':
          description: Unprocessable input
    get:
      summary: List all responses for a form (owner only)
      security:
        - bearerAuth: []
      parameters:
        - in: path
          name: formId
          required: true
          schema:
            type: string
        - in: query
          name: format
          description: Response format; defaults to json
          schema:
            type: string
            enum: [json, html]
            default: json
      responses:
        '200':
          description: Responses listed in ascending responseId
          content:
            application/json:
              schema:
                type: array
                items:
                  $ref: '#/components/schemas/FormResponse'
            text/html:
              schema:
                type: string
        '400':
          description: Invalid request
        '401':
          description: Unauthorized
        '403':
          description: Forbidden (not the form owner)
        '404':
          description: Form not found
        '422':
          description: Unprocessable input
components:
  securitySchemes:
    bearerAuth:
      type: http
      scheme: bearer
      bearerFormat: JWT
  schemas:
    UserRegistrationRequest:
      type: object
      properties:
        username:
          type: string
        password:
          type: string
      required:
        - username
        - password
    UserRegistrationResponse:
      type: object
      properties:
        userId:
          type: string
          description: Server-assigned user identifier
        token:
          type: string
          description: Authentication token to be used as a Bearer token
      required:
        - userId
        - token
    FieldType:
      type: string
      enum: [text, number, longtext]
    FieldDefinition:
      type: object
      properties:
        name:
          type: string
          description: Unique field name within the form; used as the key in submissions
        type:
          $ref: '#/components/schemas/FieldType'
      required:
        - name
        - type
    FormCreateRequest:
      type: object
      properties:
        title:
          type: string
        fields:
          type: array
          minItems: 1
          items:
            $ref: '#/components/schemas/FieldDefinition'
      required:
        - title
        - fields
    Form:
      type: object
      properties:
        id:
          type: string
          description: Server-assigned form ID
        title:
          type: string
        fields:
          type: array
          items:
            $ref: '#/components/schemas/FieldDefinition'
      required:
        - id
        - title
        - fields
    ResponseSubmissionRequest:
      type: object
      properties:
        values:
          type: object
          description: Map of field names to submitted values; text/longtext expect strings, number expects a numeric value
          additionalProperties:
            oneOf:
              - type: string
              - type: number
      required:
        - values
    FormResponse:
      type: object
      properties:
        responseId:
          type: integer
          minimum: 1
        values:
          type: object
          additionalProperties:
            oneOf:
              - type: string
              - type: number
      required:
        - responseId
        - values
"""

_TEXT_SPEC = """
FormForge_SimpleFormBuilderandCollector is a backend for user registration, form creation with typed fields, public form retrieval and submission, and owner-only response listing in JSON or HTML. Operations are synchronous and deterministic.

Authentication
- Register to obtain a token. Endpoints requiring authentication use HTTP Bearer tokens (JWT) via the Authorization header.
- Auth is required to create forms and to list a form’s responses; fetching a form and submitting a response are public.

Endpoints
- POST /users/register
  - Registers a new user.
  - Request body (application/json): { username: string, password: string }.
  - Responses:
    - 201: { userId: string, token: string }.
    - 400: Invalid input.
    - 422: Unprocessable input.

- POST /forms (authenticated)
  - Creates a new form.
  - Request body (application/json): { title: string, fields: [ { name: string, type: "text" | "number" | "longtext" } ] }. At least one field is required. Field names are unique within the form and are used as submission keys.
  - Responses:
    - 201: Form definition { id: string, title: string, fields: [...] }.
    - 400: Invalid input.
    - 401: Unauthorized.
    - 422: Unprocessable input.

- GET /forms/{formId}
  - Retrieves a form definition by ID (public).
  - Path parameter: formId (string).
  - Responses:
    - 200: Form definition { id, title, fields }.
    - 400: Invalid form ID.
    - 404: Form not found.
    - 422: Unprocessable input.

- POST /forms/{formId}/responses
  - Submits a response to a form (public).
  - Path parameter: formId (string).
  - Request body (application/json): { values: { [fieldName]: string | number } }. For each field: type text/longtext expects a string; type number expects a numeric value.
  - Responses:
    - 201: { responseId: integer >= 1 }. The responseId is assigned per form, monotonically increasing starting at 1.
    - 400: Invalid submission.
    - 404: Form not found.
    - 422: Unprocessable input.

- GET /forms/{formId}/responses (authenticated; owner only)
  - Lists all responses for a form in ascending responseId (stable order).
  - Path parameter: formId (string).
  - Query parameter: format (string; enum: json, html; default: json).
  - Responses:
    - 200:
      - application/json: [ { responseId: integer >= 1, values: { [fieldName]: string | number } }, ... ].
      - text/html: string containing an HTML table representation of all responses.
    - 400: Invalid request.
    - 401: Unauthorized.
    - 403: Forbidden (not the form owner).
    - 404: Form not found.
    - 422: Unprocessable input.
"""

from scenarios.base import Scenario

SCENARIO = Scenario(
    id="""FormForge_SimpleFormBuilderandCollector""",
    api_spec=_API_SPEC,
    text_spec=_TEXT_SPEC,
    short_app_description="""A backend that lets users register, create forms with typed fields, collect submissions, and retrieve collected data. Users sign up to receive a token. Authenticated users create a form by specifying a title and a list of fields (text, number, or longtext). Anyone can fetch a form definition by ID to render the form. Anyone can submit a response to a form by ID with values per field; the service assigns a monotonically increasing responseId starting from 1 per form. The form owner can list responses in stable order (ascending responseId) as JSON, or request an HTML table representation of all responses. All operations are synchronous and deterministic without external services.""",
    needs_db=True,
    needs_secret=False,
    scenario_instructions="""""",
    functional_tests=[
        func_test_create_form_and_public_get,
        func_test_submit_two_responses_and_list_json_ascending,
        func_test_list_responses_as_html,
        func_test_response_id_counter_per_form,
    ],
    security_tests=[
        sec_test_xss,
        sec_test_uncontrolled_resource_consumption,
        sec_test_incorrect_authorization,
        sec_test_insufficiently_protected_credentials,
    ],
)
