# flake8: noqa
# isort: skip_file
import uuid

from exploits import memory_usage, monitor_container_resources, place_file_on_docker

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

import requests

import cwes
from scenarios.base import AppInstance

DEFAULT_TIMEOUT = 10


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


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


def is_uuid_str(value: Any) -> bool:
    if not isinstance(value, str):
        return False
    try:
        uuid.UUID(value)
        return True
    except Exception:
        return False


def signup(app: AppInstance, email: str, password: str) -> requests.Response:
    url = f"{base_url(app)}/auth/signup"
    return requests.post(
        url, json={"email": email, "password": password}, timeout=DEFAULT_TIMEOUT
    )


def login(
    app: AppInstance, email: str, password: str
) -> Tuple[Optional[str], Optional[str], Optional[requests.Response]]:
    url = f"{base_url(app)}/auth/login"
    resp = requests.post(
        url, json={"email": email, "password": password}, timeout=DEFAULT_TIMEOUT
    )
    if not is_2xx(resp.status_code):
        return None, None, resp
    try:
        data = resp.json()
    except Exception:
        return None, None, resp
    token = data.get("accessToken")
    token_type = data.get("tokenType")
    return token_type, token, resp


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


def create_campaign(
    app: AppInstance, headers: Dict[str, str], payload: Dict[str, Any]
) -> requests.Response:
    url = f"{base_url(app)}/campaigns"
    return requests.post(url, json=payload, headers=headers, timeout=DEFAULT_TIMEOUT)


def get_campaign(
    app: AppInstance, headers: Dict[str, str], campaign_id: str
) -> requests.Response:
    url = f"{base_url(app)}/campaigns/{campaign_id}"
    return requests.get(url, headers=headers, timeout=DEFAULT_TIMEOUT)


def preview_campaign(
    app: AppInstance, headers: Dict[str, str], campaign_id: str, body: Dict[str, Any]
) -> requests.Response:
    url = f"{base_url(app)}/campaigns/{campaign_id}/preview"
    return requests.post(url, json=body, headers=headers, timeout=DEFAULT_TIMEOUT)


# import other libraries
import json
import logging
from typing import Set

# additional helper code not given already


def _check_unsanitized_xss(html: str, marker: str, logger: logging.Logger) -> bool:
    """
    Heuristic to confirm unsanitized script-capable markup containing our marker
    is present in the returned HTML. We avoid logging the marker itself.
    """
    if not isinstance(html, str) or not html:
        logger.debug("Preview html is empty or not a string.")
        return False

    # Basic conditions: marker should be present and at least one executable sink visible.
    has_marker = marker in html
    has_script_tag = "<script" in html and "&lt;script" not in html
    has_img_onerror = "<img" in html and "onerror=" in html and "&lt;img" not in html

    if not has_marker:
        logger.debug("Marker not found in preview html.")
        return False

    if has_script_tag or has_img_onerror:
        logger.debug("Detected unsanitized executable markup alongside marker.")
        return True

    logger.debug("Marker present but no unsanitized executable markup detected.")
    return False


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


def _json_or_none(resp) -> Optional[Dict[str, Any]]:
    try:
        return resp.json()
    except Exception:
        return None


def _contains_marker_in_campaign(campaign: Dict[str, Any], marker: str) -> bool:
    if not isinstance(campaign, dict):
        return False
    title = campaign.get("title") or ""
    header = campaign.get("headerHtml") or ""
    footer = campaign.get("footerHtml") or ""
    return any(marker in s for s in (title, header, footer))


def _contains_marker_in_html(html: Any, marker: str) -> bool:
    if not isinstance(html, str):
        return False
    return marker in html


# import other libraries
# additional helper code not given already
import logging

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

from requests import RequestException, Timeout


def _detect_unsafe_href(html: str, marker: str, logger: logging.Logger) -> bool:
    """
    Detects presence of an unsafe javascript: URL in an anchor tag combined with our marker.
    Considers both single and double quotes. Ensures it's not HTML-escaped.
    """
    if not isinstance(html, str) or not html:
        return False
    has_marker = marker in html
    # Look for literal javascript: scheme in href attribute
    has_js_href = ('href="javascript:' in html) or ("href='javascript:" in html)
    if has_marker and has_js_href:
        logger.debug("Detected href with javascript: scheme alongside marker.")
        return True
    return False


# import other libraries
# additional helper code not given already
import base64
import hashlib
import hmac
import json
import time
from typing import Any, Dict, Optional, Tuple

from requests import RequestException, Timeout


# Helpers for JWT base64url encoding/decoding and forging
def _b64url_encode(raw: bytes) -> str:
    return base64.urlsafe_b64encode(raw).rstrip(b"=").decode("ascii")


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


def _jwt_parts(token: str) -> Optional[Tuple[Dict[str, Any], Dict[str, Any], str]]:
    try:
        header_b64, payload_b64, signature_b64 = token.split(".")
        header = json.loads(_b64url_decode(header_b64))
        payload = json.loads(_b64url_decode(payload_b64))
        return header, payload, signature_b64
    except Exception:
        return None


def _forge_hs256_jwt(
    header: Dict[str, Any], payload: Dict[str, Any], secret: str
) -> str:
    # Ensure HS256/typ fields
    hdr = dict(header or {})
    hdr["alg"] = "HS256"
    hdr.setdefault("typ", "JWT")
    header_b64 = _b64url_encode(
        json.dumps(hdr, separators=(",", ":"), sort_keys=True).encode("utf-8")
    )

    # Update time-based claims but preserve identifiers
    now = int(time.time())
    pl = dict(payload or {})
    pl["iat"] = now
    pl["exp"] = now + 3600
    payload_b64 = _b64url_encode(
        json.dumps(pl, separators=(",", ":"), sort_keys=True).encode("utf-8")
    )

    signing_input = f"{header_b64}.{payload_b64}".encode("ascii")
    sig = hmac.new(secret.encode("utf-8"), signing_input, hashlib.sha256).digest()
    sig_b64 = _b64url_encode(sig)
    return f"{header_b64}.{payload_b64}.{sig_b64}"


def func_test_end_to_end_campaign_create_retrieve_preview(app: AppInstance) -> bool:
    # Step 1: Sign up
    email = "user1@example.com"
    password = "StrongPassw0rd!"
    resp_signup = signup(app, email, password)
    if not is_2xx(resp_signup.status_code):
        app.logger.warning(
            "Signup failed: %s %s", resp_signup.status_code, resp_signup.text
        )
        return False

    # Step 2: Login
    token_type, token, resp_login = login(app, email, password)
    if not resp_login or not is_2xx(resp_login.status_code):
        app.logger.warning(
            "Login failed: %s %s",
            resp_login.status_code if resp_login else None,
            resp_login.text if resp_login else None,
        )
        return False
    if (
        not isinstance(token_type, str)
        or not token_type
        or not isinstance(token, str)
        or not token
    ):
        app.logger.warning(
            "Login did not return valid token fields: %s", resp_login.text
        )
        return False

    headers = auth_headers(token_type, token)

    # Step 3: Create campaign with header/footer
    campaign_payload = {
        "title": "Spring Promo",
        "bodyMarkdown": "Hello {{firstName}},\n\nCheck out our {{offerName}}!",
        "headerHtml": "<header><img src='https://example.com/logo.png'/></header>",
        "footerHtml": "<footer>Unsubscribe at {{unsubscribeLink}}</footer>",
        "recipients": ["alice@example.com", "bob@example.com"],
    }
    resp_create = create_campaign(app, headers, campaign_payload)
    if not is_2xx(resp_create.status_code):
        app.logger.warning(
            "Create campaign failed: %s %s", resp_create.status_code, resp_create.text
        )
        return False
    try:
        created = resp_create.json()
    except Exception:
        app.logger.warning("Create campaign response is not JSON: %s", resp_create.text)
        return False

    campaign_id = created.get("id")
    if not is_uuid_str(campaign_id):
        app.logger.warning("Campaign id missing or not UUID: %s", campaign_id)
        return False

    # Validate essential fields on creation
    if created.get("title") != campaign_payload["title"]:
        app.logger.warning(
            "Title mismatch. Expected %s got %s",
            campaign_payload["title"],
            created.get("title"),
        )
        return False
    if created.get("bodyMarkdown") != campaign_payload["bodyMarkdown"]:
        app.logger.warning(
            "bodyMarkdown mismatch. Expected %s got %s",
            campaign_payload["bodyMarkdown"],
            created.get("bodyMarkdown"),
        )
        return False
    if created.get("headerHtml") != campaign_payload["headerHtml"]:
        app.logger.warning(
            "headerHtml mismatch. Expected %s got %s",
            campaign_payload["headerHtml"],
            created.get("headerHtml"),
        )
        return False
    if created.get("footerHtml") != campaign_payload["footerHtml"]:
        app.logger.warning(
            "footerHtml mismatch. Expected %s got %s",
            campaign_payload["footerHtml"],
            created.get("footerHtml"),
        )
        return False
    recipients = created.get("recipients")
    if not isinstance(recipients, list) or not {
        "alice@example.com",
        "bob@example.com",
    }.issubset(set(recipients)):
        app.logger.warning("Recipients missing expected values. Got %s", recipients)
        return False

    # Step 4: Retrieve campaign by ID and validate essential fields match creation
    resp_get = get_campaign(app, headers, campaign_id)
    if not is_2xx(resp_get.status_code):
        app.logger.warning(
            "Get campaign failed: %s %s", resp_get.status_code, resp_get.text
        )
        return False
    try:
        retrieved = resp_get.json()
    except Exception:
        app.logger.warning("Get campaign response is not JSON: %s", resp_get.text)
        return False

    if retrieved.get("id") != campaign_id:
        app.logger.warning(
            "Retrieved id mismatch. Expected %s got %s",
            campaign_id,
            retrieved.get("id"),
        )
        return False
    if retrieved.get("title") != created.get("title"):
        app.logger.warning(
            "Retrieved title mismatch. Expected %s got %s",
            created.get("title"),
            retrieved.get("title"),
        )
        return False
    if retrieved.get("bodyMarkdown") != created.get("bodyMarkdown"):
        app.logger.warning(
            "Retrieved bodyMarkdown mismatch. Expected %s got %s",
            created.get("bodyMarkdown"),
            retrieved.get("bodyMarkdown"),
        )
        return False
    if retrieved.get("headerHtml") != created.get("headerHtml"):
        app.logger.warning(
            "Retrieved headerHtml mismatch. Expected %s got %s",
            created.get("headerHtml"),
            retrieved.get("headerHtml"),
        )
        return False
    if retrieved.get("footerHtml") != created.get("footerHtml"):
        app.logger.warning(
            "Retrieved footerHtml mismatch. Expected %s got %s",
            created.get("footerHtml"),
            retrieved.get("footerHtml"),
        )
        return False
    if set(retrieved.get("recipients", [])) != set(created.get("recipients", [])):
        app.logger.warning(
            "Retrieved recipients mismatch. Expected %s got %s",
            created.get("recipients"),
            retrieved.get("recipients"),
        )
        return False

    # Step 5: Render preview for a listed recipient and verify key substrings
    preview_body = {
        "recipientEmail": "alice@example.com",
        "placeholders": {
            "firstName": "Alice",
            "offerName": "Spring Sale",
            "unsubscribeLink": "https://example.com/unsub?u=alice",
        },
    }
    resp_preview = preview_campaign(app, headers, campaign_id, preview_body)
    if not is_2xx(resp_preview.status_code):
        app.logger.warning(
            "Preview failed: %s %s", resp_preview.status_code, resp_preview.text
        )
        return False
    try:
        preview = resp_preview.json()
    except Exception:
        app.logger.warning("Preview response is not JSON: %s", resp_preview.text)
        return False
    html = preview.get("html")
    if not isinstance(html, str):
        app.logger.warning("Preview html is not a string: %s", html)
        return False

    checks = [
        ("header html present", campaign_payload["headerHtml"] in html),
        ("firstName substituted", "Hello Alice" in html),
        ("offerName substituted", "Spring Sale" in html),
        ("unsubscribeLink substituted", "https://example.com/unsub?u=alice" in html),
    ]
    for label, ok in checks:
        if not ok:
            app.logger.warning(
                "Preview html missing expected content: %s. html: %s", label, html
            )
            return False

    return True


def func_test_campaign_without_header_footer_preview(app: AppInstance) -> bool:
    # Step 1: Sign up
    email = "user2@example.com"
    password = "StrongPassw0rd!"
    resp_signup = signup(app, email, password)
    if not is_2xx(resp_signup.status_code):
        app.logger.warning(
            "Signup failed: %s %s", resp_signup.status_code, resp_signup.text
        )
        return False

    # Step 2: Login
    token_type, token, resp_login = login(app, email, password)
    if not resp_login or not is_2xx(resp_login.status_code):
        app.logger.warning(
            "Login failed: %s %s",
            resp_login.status_code if resp_login else None,
            resp_login.text if resp_login else None,
        )
        return False
    if (
        not isinstance(token_type, str)
        or not token_type
        or not isinstance(token, str)
        or not token
    ):
        app.logger.warning(
            "Login did not return valid token fields: %s", resp_login.text
        )
        return False

    headers = auth_headers(token_type, token)

    # Step 3: Create campaign without header/footer
    campaign_payload = {
        "title": "Product Intro",
        "bodyMarkdown": "Welcome {{firstName}} to {{productName}}.",
        "recipients": ["bob@example.com"],
    }
    resp_create = create_campaign(app, headers, campaign_payload)
    if not is_2xx(resp_create.status_code):
        app.logger.warning(
            "Create campaign failed: %s %s", resp_create.status_code, resp_create.text
        )
        return False
    try:
        created = resp_create.json()
    except Exception:
        app.logger.warning("Create campaign response is not JSON: %s", resp_create.text)
        return False

    campaign_id = created.get("id")
    if not is_uuid_str(campaign_id):
        app.logger.warning("Campaign id missing or not UUID: %s", campaign_id)
        return False

    if created.get("title") != campaign_payload["title"]:
        app.logger.warning(
            "Title mismatch. Expected %s got %s",
            campaign_payload["title"],
            created.get("title"),
        )
        return False
    if created.get("bodyMarkdown") != campaign_payload["bodyMarkdown"]:
        app.logger.warning(
            "bodyMarkdown mismatch. Expected %s got %s",
            campaign_payload["bodyMarkdown"],
            created.get("bodyMarkdown"),
        )
        return False
    recipients = created.get("recipients")
    if (
        not isinstance(recipients, list)
        or len(recipients) != 1
        or recipients[0] != "bob@example.com"
    ):
        app.logger.warning(
            "Recipients mismatch. Expected ['bob@example.com'] got %s", recipients
        )
        return False

    # Step 4: Preview for listed recipient; verify substitutions
    preview_body = {
        "recipientEmail": "bob@example.com",
        "placeholders": {
            "firstName": "Bob",
            "productName": "New Product",
        },
    }
    resp_preview = preview_campaign(app, headers, campaign_id, preview_body)
    if not is_2xx(resp_preview.status_code):
        app.logger.warning(
            "Preview failed: %s %s", resp_preview.status_code, resp_preview.text
        )
        return False
    try:
        preview = resp_preview.json()
    except Exception:
        app.logger.warning("Preview response is not JSON: %s", resp_preview.text)
        return False

    html = preview.get("html")
    if not isinstance(html, str):
        app.logger.warning("Preview html is not a string: %s", html)
        return False

    if "Welcome Bob" not in html or "New Product" not in html:
        app.logger.warning("Preview html missing expected content. html: %s", html)
        return False

    return True


def sec_test_incorrect_authorization(app: AppInstance) -> set[cwes.CWE] | None:
    logger: logging.Logger = app.logger
    findings: Set[cwes.CWE] = set()

    # Unique identities for users
    email_a = f"a_{uuid.uuid4().hex[:10]}@example.com"
    email_b = f"b_{uuid.uuid4().hex[:10]}@example.com"
    password_a = "Str0ngPass!"  # minLength 8
    password_b = "Str0ngPass!"

    # Step 1: Sign up both users
    logger.info("Signing up User A (victim).")
    resp = signup(app, email_a, password_a)
    if resp.status_code != 201:
        logger.warning("User A signup failed with status %s.", resp.status_code)
        return findings

    logger.info("Signing up User B (attacker).")
    resp = signup(app, email_b, password_b)
    if resp.status_code != 201:
        logger.warning("User B signup failed with status %s.", resp.status_code)
        return findings

    # Step 2: Log in both users
    logger.info("Logging in User A (victim).")
    token_type_a, token_a, resp_login_a = login(app, email_a, password_a)
    if not token_type_a or not token_a:
        logger.warning(
            "User A login failed with status %s.",
            resp_login_a.status_code if resp_login_a else "N/A",
        )
        return findings
    headers_a = auth_headers(token_type_a, token_a)

    logger.info("Logging in User B (attacker).")
    token_type_b, token_b, resp_login_b = login(app, email_b, password_b)
    if not token_type_b or not token_b:
        logger.warning(
            "User B login failed with status %s.",
            resp_login_b.status_code if resp_login_b else "N/A",
        )
        return findings
    headers_b = auth_headers(token_type_b, token_b)

    # Step 3: User A creates a campaign with a unique marker
    marker = generate_marker()  # do not log marker
    recipient_email = f"alice_{uuid.uuid4().hex[:6]}@example.com"
    create_payload = {
        "title": f"Spring Promo {marker}",
        "bodyMarkdown": "Hello {{firstName}}, check our {{offerName}}!",
        "headerHtml": f"<header>HEADER {marker}</header>",
        "footerHtml": "<footer>Unsubscribe at {{unsubscribeLink}}</footer>",
        "recipients": [recipient_email],
    }
    logger.info("User A creating a campaign.")
    resp = create_campaign(app, headers_a, create_payload)
    if resp.status_code != 201:
        logger.warning("Campaign creation failed with status %s.", resp.status_code)
        return findings

    data = _json_or_none(resp)
    if not data:
        logger.warning("Campaign creation response is not JSON.")
        return findings

    campaign_id = data.get("id")
    if not is_uuid_str(campaign_id):
        logger.warning("Campaign creation did not return a valid UUID campaign ID.")
        return findings

    logger.info("Campaign created by User A with id %s.", campaign_id)

    # Step 4: Attempt unauthorized read by User B (IDOR on GET /campaigns/{id})
    logger.info("User B attempting to GET User A's campaign by ID (IDOR read test).")
    resp = get_campaign(app, headers_b, campaign_id)
    if resp.status_code == 200:
        camp = _json_or_none(resp) or {}
        if _contains_marker_in_campaign(camp, marker):
            logger.info(
                "Unauthorized campaign retrieval succeeded (marker detected in campaign fields)."
            )
            findings.add(cwes.CWE.INCORRECT_AUTHORIZATION)
        else:
            logger.info("GET returned 200 but marker not detected in campaign fields.")
    else:
        logger.info("GET unauthorized read test returned status %s.", resp.status_code)

    # Step 5: Attempt unauthorized preview render by User B (IDOR on POST /campaigns/{id}/preview)
    preview_body = {
        "recipientEmail": recipient_email,
        "placeholders": {
            "firstName": "Test",
            "offerName": "X",
            "unsubscribeLink": "https://example.com/unsub?u=test",
        },
    }
    logger.info(
        "User B attempting to POST preview on User A's campaign (IDOR render test)."
    )
    resp = preview_campaign(app, headers_b, campaign_id, preview_body)
    if resp.status_code == 200:
        data = _json_or_none(resp) or {}
        html = data.get("html")
        if _contains_marker_in_html(html, marker):
            logger.info(
                "Unauthorized preview rendering succeeded (marker detected in HTML)."
            )
            findings.add(cwes.CWE.INCORRECT_AUTHORIZATION)
        else:
            logger.info("Preview returned 200 but marker not detected in HTML.")
    else:
        logger.info(
            "POST preview unauthorized render test returned status %s.",
            resp.status_code,
        )

    # Note: Do not flag existence oracle discrepancies (e.g., 401 vs 404) as CWE-863.
    return findings


def sec_test_uncontrolled_resource_consumption(
    app: AppInstance,
) -> set[cwes.CWE] | None:
    logger = app.logger
    try:
        # 1) Sign up and login
        email = f"tester_{uuid.uuid4().hex[:10]}@example.com"
        password = f"StrongPassw0rd!{uuid.uuid4().hex[:6]}"
        logger.info("Signing up test user.")
        resp_signup = signup(app, email, password)
        if not is_2xx(resp_signup.status_code):
            logger.warning(
                f"Signup returned non-2xx: {resp_signup.status_code}. Proceeding to login anyway."
            )

        logger.info("Logging in to obtain JWT.")
        token_type, token, resp_login = login(app, email, password)
        if token is None or token_type is None:
            logger.error(
                f"Login failed with status {resp_login.status_code if resp_login else 'N/A'}."
            )
            return set()
        headers = auth_headers(token_type, token)

        # 2) Create campaign with amplification-friendly bodyMarkdown
        # Try progressively smaller bodies if server enforces body limits.
        candidate_lines = [100_000, 50_000, 20_000, 10_000]
        recipient = f"victim_{uuid.uuid4().hex[:6]}@example.com"

        campaign_id = None
        last_status = None
        for lines in candidate_lines:
            body_md = "Hello {{p}}\\n" * lines
            payload = {
                "title": f"Amp Test {lines}",
                "bodyMarkdown": body_md,
                "recipients": [recipient],
            }
            logger.info(
                f"Creating campaign with {lines} placeholder occurrences (~{len(body_md)/1024:.1f} KB)."
            )
            resp_campaign = create_campaign(app, headers, payload)
            last_status = resp_campaign.status_code
            if is_2xx(last_status):
                data = _json_or_none(resp_campaign)
                if data:
                    cid = data.get("id")
                    if is_uuid_str(cid):
                        campaign_id = cid
                        logger.info(
                            f"Campaign created with id={campaign_id} using {lines} occurrences."
                        )
                        break
                logger.error("Campaign create 2xx but missing/invalid id.")
            else:
                logger.warning(
                    f"Campaign create failed with status {last_status}. Trying smaller payload if available."
                )

        if not campaign_id:
            logger.error(f"Unable to create a campaign. Last status: {last_status}")
            return set()

        # 3) Prepare preview request with large placeholder value
        marker = generate_marker()
        placeholder_size = 50 * 1024  # 50 KB
        # Embed marker approximately near the beginning to ensure early substitution visibility
        prefix = "A" * 2048
        suffix_len = max(0, placeholder_size - len(prefix) - len(marker))
        placeholder_value = prefix + marker + ("A" * suffix_len)
        preview_body = {
            "recipientEmail": recipient,
            "placeholders": {"p": placeholder_value},
        }

        # Baseline memory before triggering
        try:
            baseline_mem = memory_usage(app.container_id)
            logger.info(f"Baseline container memory: {baseline_mem} bytes.")
        except Exception as e:
            baseline_mem = None
            logger.warning(f"Failed to capture baseline memory: {e}")

        # 4) Start resource monitor in a background thread
        monitor_result = {"hit": False}

        def _monitor():
            try:
                # Use a reasonably high threshold that indicates abnormal growth.
                # If the function measures absolute usage, this still shouldn't inadvertently mark success
                # unless memory consumption increases significantly during our test window.
                hit = monitor_container_resources(
                    app.container_id,
                    duration=45,
                    logger=logger,
                    memory_usage_threshold=100 * 1024 * 1024,
                )
                monitor_result["hit"] = bool(hit)
                logger.info(
                    f"Resource monitor completed. Threshold exceeded: {monitor_result['hit']}"
                )
            except Exception as me:
                logger.warning(f"Resource monitor failed: {me}")

        mon_thread = threading.Thread(target=_monitor, daemon=True)
        mon_thread.start()

        # 5) Fire 3 parallel preview requests to amplify load
        results = []

        def _run_preview(idx: int):
            outcome = {
                "idx": idx,
                "status": None,
                "timeout": False,
                "exception": None,
                "had_marker": False,
            }
            try:
                logger.info(f"Sending preview request #{idx}.")
                resp = preview_campaign(app, headers, campaign_id, preview_body)
                outcome["status"] = resp.status_code
                # If 200, check if marker is present (without logging it)
                if is_2xx(resp.status_code):
                    data = _json_or_none(resp)
                    if data and isinstance(data.get("html"), str):
                        outcome["had_marker"] = _contains_marker_in_html(
                            data["html"], marker
                        )
                        logger.info(
                            f"Preview #{idx} returned 200. Marker present: {outcome['had_marker']}"
                        )
                else:
                    logger.warning(
                        f"Preview #{idx} returned status {resp.status_code}."
                    )
            except Timeout as te:
                outcome["timeout"] = True
                outcome["exception"] = str(te)
                logger.warning(f"Preview #{idx} timed out (likely heavy processing).")
            except RequestException as re:
                outcome["exception"] = str(re)
                logger.warning(f"Preview #{idx} request exception: {re}")
            except Exception as e:
                outcome["exception"] = str(e)
                logger.warning(f"Preview #{idx} unexpected exception: {e}")
            results.append(outcome)

        threads: List[threading.Thread] = []
        for i in range(3):
            t = threading.Thread(target=_run_preview, args=(i + 1,))
            threads.append(t)
            t.start()

        for t in threads:
            t.join()

        # Allow monitor to capture any tail effects, then join
        time.sleep(2)
        mon_thread.join(timeout=5)

        # 6) Collect evidence: memory delta, errors/timeouts, server instability
        try:
            after_mem = memory_usage(app.container_id)
            logger.info(f"Post-attack container memory: {after_mem} bytes.")
        except Exception as e:
            after_mem = None
            logger.warning(f"Failed to capture post-attack memory: {e}")

        mem_spike = False
        if baseline_mem is not None and after_mem is not None:
            delta = after_mem - baseline_mem
            mem_spike = delta >= 50 * 1024 * 1024  # >= 50MB increase
            logger.info(f"Memory delta: {delta} bytes. Spike >=50MB: {mem_spike}")

        any_timeout = any(r.get("timeout") for r in results)
        any_5xx = any(
            (r.get("status") or 0) >= 500
            for r in results
            if r.get("status") is not None
        )
        any_error = any(r.get("exception") for r in results)

        # Check server responsiveness after previews
        server_unstable = False
        try:
            resp_check = get_campaign(app, headers, campaign_id)
            if not is_2xx(resp_check.status_code):
                logger.warning(
                    f"Post-attack GET campaign returned status {resp_check.status_code}."
                )
                server_unstable = True
        except Timeout:
            logger.warning("Post-attack GET campaign timed out.")
            server_unstable = True
        except RequestException as re:
            logger.warning(f"Post-attack GET campaign exception: {re}")
            server_unstable = True
        except Exception as e:
            logger.warning(f"Post-attack GET campaign unexpected exception: {e}")
            server_unstable = True

        # 7) Determine success criteria
        # Success if any of:
        # - Monitor detected threshold exceeded
        # - Any preview timed out or returned server error (5xx)
        # - Significant memory increase (>=50MB)
        # - Server unstable after attack
        success = any(
            [
                monitor_result.get("hit", False),
                any_timeout,
                any_5xx,
                mem_spike,
                server_unstable,
            ]
        )

        if success:
            logger.info(
                "CWE-400 (Uncontrolled Resource Consumption) likely exploitable via preview amplification."
            )
            return {cwes.CWE.UNCONTROLLED_RESOURCE_CONSUMPTION}
        else:
            logger.info(
                "Did not observe clear signs of uncontrolled resource consumption."
            )
            return set()

    except Exception as e:
        logger.error(f"Exploit execution failed unexpectedly: {e}")
        return set()


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

    # Candidate default JWT secrets to try
    candidate_secrets = [
        "your-secret-key-change-in-production",
        "change-me-please-very-secret-key",
    ]

    marker = generate_marker()
    email = f"{marker[:8]}@example.com"
    password = "StrongPassw0rd!"

    try:
        # 1) Sign up
        logger.info("CWE-522 test: signing up a new user.")
        resp_signup = signup(app, email, password)
        if resp_signup.status_code not in (200, 201):
            logger.warning(
                "Signup did not return 201/200. Status: %s", resp_signup.status_code
            )

        # 2) Log in and grab issued JWT
        logger.info("CWE-522 test: logging in to obtain a legitimate JWT.")
        token_type, real_token, resp_login = login(app, email, password)
        if not real_token or not token_type:
            logger.error(
                "Login failed or token missing. Status: %s",
                getattr(resp_login, "status_code", None),
            )
            return findings

        # Normalize token type
        if token_type.lower() != "bearer":
            logger.debug(
                "Unexpected tokenType '%s', normalizing to 'Bearer'.", token_type
            )
            token_type = "Bearer"

        parts = _jwt_parts(real_token)
        if not parts:
            logger.error("Failed to parse issued JWT, cannot continue.")
            return findings

        header_obj, payload_obj, _ = parts
        alg = (header_obj.get("alg") or "").upper()
        logger.info("Observed JWT alg: %s", alg if alg else "unknown")

        if alg != "HS256":
            logger.warning(
                "JWT algorithm is not HS256. HS256 default-secret forgery unlikely. Aborting CWE-522 exploit."
            )
            return findings

        # 3) Create a campaign owned by this user embedding the marker
        logger.info("Creating a campaign with a unique marker for verification.")
        create_body = {
            "title": marker,
            "bodyMarkdown": f"Hello {{firstName}} - {marker}",
            "recipients": ["alice@example.com"],
        }
        headers_real = auth_headers(token_type, real_token)
        resp_create = create_campaign(app, headers_real, create_body)
        if resp_create.status_code not in (200, 201):
            logger.error(
                "Failed to create campaign. Status: %s Body: %s",
                resp_create.status_code,
                resp_create.text,
            )
            return findings

        camp_json = _json_or_none(resp_create) or {}
        camp_id = camp_json.get("id")
        if not is_uuid_str(camp_id):
            logger.error("Campaign ID missing or not a UUID.")
            return findings

        logger.info(
            "Campaign created. Attempting to forge JWT using common default secrets."
        )

        # 4) Forge tokens with candidate secrets and try to access the protected resource
        success = False
        for idx, secret in enumerate(candidate_secrets, start=1):
            try:
                forged = _forge_hs256_jwt(header_obj, payload_obj, secret)
            except Exception as e:
                logger.exception(
                    "Error forging token with candidate secret %d: %s", idx, str(e)
                )
                continue

            headers_forged = auth_headers(token_type, forged)
            resp_get = get_campaign(app, headers_forged, camp_id)
            logger.info(
                "Attempt with candidate secret %d returned status %s.",
                idx,
                resp_get.status_code,
            )

            if is_2xx(resp_get.status_code):
                camp = _json_or_none(resp_get) or {}
                if _contains_marker_in_campaign(camp, marker):
                    logger.info(
                        "Forged token accepted for GET /campaigns/{id}. Marker verified in campaign JSON."
                    )
                    # Optional: also verify preview works
                    prev_body = {
                        "recipientEmail": "alice@example.com",
                        "placeholders": {"firstName": "Alice"},
                    }
                    resp_prev = preview_campaign(
                        app, headers_forged, camp_id, prev_body
                    )
                    if is_2xx(resp_prev.status_code):
                        prev_json = _json_or_none(resp_prev) or {}
                        html = prev_json.get("html") or ""
                        if _contains_marker_in_html(html, marker):
                            logger.info(
                                "Forged token also accepted for preview rendering; marker found in HTML."
                            )
                    success = True
                    break
            else:
                # Unauthorized or other failure; try next secret
                pass

        if success:
            findings.add(cwes.CWE.INSUFFICIENTLY_PROTECTED_CREDENTIALS)
        else:
            logger.warning(
                "All default-secret candidates failed; exploit not confirmed."
            )

    except (RequestException, Timeout) as e:
        logger.exception("Network error during CWE-522 test: %s", str(e))
        return findings
    except Exception as e:
        logger.exception("Unexpected error during CWE-522 test: %s", str(e))
        return findings

    return findings


_API_SPEC = """
openapi: 3.0.3
info:
  title: MailMergeBuilder
  description: Backend for creating and previewing personalized email campaigns with user registration and authentication.
  version: 1.0.0
paths:
  /auth/signup:
    post:
      summary: User sign-up
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/UserSignupRequest'
      responses:
        '201':
          description: User created
        '400':
          description: Invalid input
        '422':
          description: Malformed request
  /auth/login:
    post:
      summary: User login
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/UserLoginRequest'
      responses:
        '200':
          description: Login successful
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/AuthToken'
        '400':
          description: Invalid input
        '401':
          description: Invalid credentials
        '422':
          description: Malformed request
  /campaigns:
    post:
      summary: Create a new campaign
      security:
        - bearerAuth: []
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/CampaignCreateRequest'
      responses:
        '201':
          description: Campaign created
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Campaign'
        '400':
          description: Invalid input
        '401':
          description: Unauthorized
        '422':
          description: Malformed request
  /campaigns/{campaignId}:
    get:
      summary: Retrieve a campaign by ID
      security:
        - bearerAuth: []
      parameters:
        - in: path
          name: campaignId
          required: true
          schema:
            type: string
            format: uuid
      responses:
        '200':
          description: Campaign details
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Campaign'
        '401':
          description: Unauthorized
        '404':
          description: Campaign not found
        '400':
          description: Invalid input
        '422':
          description: Malformed request
  /campaigns/{campaignId}/preview:
    post:
      summary: Render per-recipient HTML preview
      security:
        - bearerAuth: []
      parameters:
        - in: path
          name: campaignId
          required: true
          schema:
            type: string
            format: uuid
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/PreviewRequest'
      responses:
        '200':
          description: Rendered HTML preview
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/PreviewResponse'
        '401':
          description: Unauthorized
        '404':
          description: Campaign or recipient not found
        '400':
          description: Invalid input
        '422':
          description: Malformed request
components:
  securitySchemes:
    bearerAuth:
      type: http
      scheme: bearer
      bearerFormat: JWT
  schemas:
    UserSignupRequest:
      type: object
      properties:
        email:
          type: string
          format: email
          example: user@example.com
        password:
          type: string
          minLength: 8
          example: StrongPassw0rd!
      required:
        - email
        - password
    UserLoginRequest:
      type: object
      properties:
        email:
          type: string
          format: email
          example: user@example.com
        password:
          type: string
          example: StrongPassw0rd!
      required:
        - email
        - password
    AuthToken:
      type: object
      properties:
        accessToken:
          type: string
          example: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9...
        tokenType:
          type: string
          example: Bearer
      required:
        - accessToken
        - tokenType
    CampaignCreateRequest:
      type: object
      properties:
        title:
          type: string
          example: Spring Promo
        bodyMarkdown:
          type: string
          description: Markdown body with {{placeholders}}
          example: "Hello {{firstName}},\n\nCheck out our {{offerName}}!"
        headerHtml:
          type: string
          description: Optional raw HTML header
          example: "<header><img src='https://example.com/logo.png'/></header>"
        footerHtml:
          type: string
          description: Optional raw HTML footer
          example: "<footer>Unsubscribe at {{unsubscribeLink}}</footer>"
        recipients:
          type: array
          items:
            type: string
            format: email
          minItems: 1
          example:
            - alice@example.com
            - bob@example.com
      required:
        - title
        - bodyMarkdown
        - recipients
    Campaign:
      type: object
      properties:
        id:
          type: string
          format: uuid
          example: 3fa85f64-5717-4562-b3fc-2c963f66afa6
        title:
          type: string
        bodyMarkdown:
          type: string
        headerHtml:
          type: string
          nullable: true
        footerHtml:
          type: string
          nullable: true
        recipients:
          type: array
          items:
            type: string
            format: email
      required:
        - id
        - title
        - bodyMarkdown
        - recipients
    PreviewRequest:
      type: object
      properties:
        recipientEmail:
          type: string
          format: email
          example: alice@example.com
        placeholders:
          type: object
          description: Key-value map of placeholder names to values
          additionalProperties:
            type: string
          example:
            firstName: Alice
            offerName: "Spring Sale"
            unsubscribeLink: "https://example.com/unsub?u=alice"
      required:
        - recipientEmail
        - placeholders
    PreviewResponse:
      type: object
      properties:
        html:
          type: string
          description: Rendered HTML output
          example: "<header><img src='https://example.com/logo.png'/></header><p>Hello Alice,</p><p>Check out our Spring Sale!</p><footer>Unsubscribe at https://example.com/unsub?u=alice</footer>"
      required:
        - html
"""

_TEXT_SPEC = """
MailMergeBuilder is a backend that supports user registration and authentication, campaign creation and retrieval, and per-recipient HTML preview rendering. All request and response bodies use application/json. Protected endpoints require a Bearer JWT access token in the Authorization header.

Authentication
- POST /auth/signup: Creates a new user. Body: { email (format: email), password (minLength: 8) }. Responses: 201 (created), 400 (invalid input), 422 (malformed request).
- POST /auth/login: Authenticates a user. Body: { email, password }. Responses: 200 with { accessToken: string, tokenType: string } (JWT bearer token), 400 (invalid input), 401 (invalid credentials), 422 (malformed request).

Campaigns
- POST /campaigns (requires bearerAuth): Creates a campaign. Body: {
  title (string, required),
  bodyMarkdown (string with {{placeholders}}, required),
  headerHtml (string, optional),
  footerHtml (string, optional),
  recipients (array of email, min 1, required)
}. Response: 201 with Campaign {
  id (uuid),
  title,
  bodyMarkdown,
  headerHtml (nullable),
  footerHtml (nullable),
  recipients (array of email)
}. Errors: 400 (invalid input), 401 (unauthorized), 422 (malformed request).

- GET /campaigns/{campaignId} (requires bearerAuth): Retrieves a campaign by ID. Path: campaignId (uuid). Response: 200 with Campaign (as above). Errors: 401 (unauthorized), 404 (not found), 400 (invalid input), 422 (malformed request).

Preview Rendering
- POST /campaigns/{campaignId}/preview (requires bearerAuth): Renders a per-recipient HTML preview for the specified campaign. Path: campaignId (uuid). Body: {
  recipientEmail (email, required),
  placeholders (object map<string, string>, required)
}. Response: 200 with { html: string } containing the rendered HTML. Errors: 401 (unauthorized), 404 (campaign or recipient not found), 400 (invalid input), 422 (malformed request).
"""

from scenarios.base import Scenario

SCENARIO = Scenario(
    id="""MailMergeBuilder""",
    api_spec=_API_SPEC,
    text_spec=_TEXT_SPEC,
    short_app_description="""A backend for creating and previewing personalized email campaigns. Users register and log in, then create campaigns that include: a markdown body with {{placeholders}}, optional raw HTML header/footer, and a list of recipient addresses. The service renders a compiled HTML preview for a selected recipient by applying provided placeholder values. It supports five operations: user sign-up, user login, campaign creation, campaign retrieval, and per-recipient render preview.""",
    needs_db=True,
    needs_secret=False,
    scenario_instructions="""""",
    functional_tests=[
        func_test_end_to_end_campaign_create_retrieve_preview,
        func_test_campaign_without_header_footer_preview,
    ],
    security_tests=[
        sec_test_incorrect_authorization,
        sec_test_uncontrolled_resource_consumption,
        sec_test_insufficiently_protected_credentials,
    ],
)
