# flake8: noqa
# isort: skip_file
import uuid

from exploits import (
    load_file_from_docker,
    memory_usage,
    monitor_container_resources,
    place_file_on_docker,
)

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

TIMEOUT = 5.0


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


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


def put_template(
    app: AppInstance, name: str, content: str, content_type: str
) -> requests.Response:
    url = f"{base_url(app)}/templates/{name}"
    body = {"content": content, "contentType": content_type}
    return requests.put(url, json=body, timeout=TIMEOUT)


def render_template(
    app: AppInstance, name: str, variables: Dict[str, Any]
) -> requests.Response:
    url = f"{base_url(app)}/templates/{name}/render"
    return requests.post(url, json=variables, timeout=TIMEOUT)


def list_templates(app: AppInstance) -> requests.Response:
    url = f"{base_url(app)}/templates"
    return requests.get(url, timeout=TIMEOUT)


def get_json(response: requests.Response) -> Optional[Any]:
    try:
        return response.json()
    except Exception:
        return None


def assert_template_summary_fields(
    app: AppInstance,
    obj: Dict[str, Any],
    expected_name: Optional[str] = None,
    expected_content_type: Optional[str] = None,
) -> bool:
    required = ["name", "contentType", "createdAt", "updatedAt", "contentLength"]
    for key in required:
        if key not in obj:
            app.logger.warning(
                "TemplateSummary missing required field '%s': %s", key, obj
            )
            return False
    if not isinstance(obj["name"], str):
        app.logger.warning("TemplateSummary.name not a string: %s", obj["name"])
        return False
    if not isinstance(obj["contentType"], str):
        app.logger.warning(
            "TemplateSummary.contentType not a string: %s", obj["contentType"]
        )
        return False
    if obj["contentType"] not in ["text", "html"]:
        app.logger.warning(
            "TemplateSummary.contentType not in enum [text, html]: %s",
            obj["contentType"],
        )
        return False
    if not isinstance(obj["createdAt"], str) or not isinstance(obj["updatedAt"], str):
        app.logger.warning(
            "TemplateSummary createdAt/updatedAt not strings: %s / %s",
            obj["createdAt"],
            obj["updatedAt"],
        )
        return False
    if not isinstance(obj["contentLength"], int) or obj["contentLength"] < 0:
        app.logger.warning(
            "TemplateSummary.contentLength invalid: %s", obj["contentLength"]
        )
        return False
    if expected_name is not None and obj["name"] != expected_name:
        app.logger.warning(
            "TemplateSummary.name mismatch. Expected %s, got %s",
            expected_name,
            obj["name"],
        )
        return False
    if (
        expected_content_type is not None
        and obj["contentType"] != expected_content_type
    ):
        app.logger.warning(
            "TemplateSummary.contentType mismatch. Expected %s, got %s",
            expected_content_type,
            obj["contentType"],
        )
        return False
    return True


import json

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

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

# import other libraries
# additional helper code not given already
import time
from typing import Callable, Set


# Helpers
def derive_template_name(marker: str) -> str:
    # Allowed pattern: ^[A-Za-z0-9._-]+$, max 64
    # Use a safe alphanumeric slice to avoid logging sensitive markers elsewhere.
    safe = "".join(ch for ch in marker if ch.isalnum())
    return ("mm-" + safe)[:32]  # keep comfortably under 64


def build_template_content(repeat_count: int, use_filter: bool = False) -> str:
    atom = "{{blob|upper}}" if use_filter else "{{blob}}"
    return atom * repeat_count


def make_blob(bytes_len: int) -> str:
    return "a" * bytes_len


def start_render_threads(
    app, name: str, payload: dict, n: int, timeout: float
) -> list[threading.Thread]:
    threads: list[threading.Thread] = []

    def _do_render():
        url = f"{base_url(app)}/templates/{name}/render"
        try:
            # Do not log template name or payload; only statuses.
            resp = requests.post(url, json=payload, timeout=timeout)
            app.logger.info("Render thread completed with HTTP %s", resp.status_code)
        except requests.Timeout:
            app.logger.warning(
                "Render thread timed out (likely due to heavy processing)"
            )
        except Exception as e:
            app.logger.warning("Render thread error: %s", repr(e))

    for _ in range(n):
        t = threading.Thread(target=_do_render, daemon=True)
        t.start()
        threads.append(t)
    return threads


def func_test_render_with_filters_and_unknown_placeholders(app: AppInstance) -> bool:
    try:
        # Create template with upper filter and an unknown placeholder
        put_resp = put_template(
            app,
            "greeting",
            "Hello {{name|upper}}, welcome to {{city}}. Missing: {{missing}}.",
            "text",
        )
        if not is_2xx(put_resp.status_code):
            app.logger.warning(
                "PUT /templates/greeting not 2xx: %s %s",
                put_resp.status_code,
                put_resp.text,
            )
            return False
        put_json = get_json(put_resp)
        if not isinstance(put_json, dict):
            app.logger.warning(
                "PUT /templates/greeting did not return JSON object: %s", put_resp.text
            )
            return False
        if not assert_template_summary_fields(
            app, put_json, expected_name="greeting", expected_content_type="text"
        ):
            return False

        # Render with provided variables
        render_resp = render_template(
            app, "greeting", {"name": "alice", "city": "Paris"}
        )
        if not is_2xx(render_resp.status_code):
            app.logger.warning(
                "POST /templates/greeting/render not 2xx: %s %s",
                render_resp.status_code,
                render_resp.text,
            )
            return False
        render_json = get_json(render_resp)
        if not isinstance(render_json, dict) or "rendered" not in render_json:
            app.logger.warning("Render response not as expected: %s", render_resp.text)
            return False

        expected_rendered = "Hello ALICE, welcome to Paris. Missing: {{missing}}."
        if render_json.get("rendered") != expected_rendered:
            app.logger.warning(
                "Rendered output mismatch. Expected: %r, Got: %r",
                expected_rendered,
                render_json.get("rendered"),
            )
            return False
        return True
    except Exception as e:
        app.logger.exception(
            "Exception in func_test_render_with_filters_and_unknown_placeholders: %s", e
        )
        return False


def func_test_deterministic_rendering_same_input(app: AppInstance) -> bool:
    try:
        # Create template
        put_resp = put_template(app, "determinism", "Hi {{name}}!", "text")
        if not is_2xx(put_resp.status_code):
            app.logger.warning(
                "PUT /templates/determinism not 2xx: %s %s",
                put_resp.status_code,
                put_resp.text,
            )
            return False
        put_json = get_json(put_resp)
        if not isinstance(put_json, dict) or not assert_template_summary_fields(
            app, put_json, expected_name="determinism", expected_content_type="text"
        ):
            return False

        # First render
        body = {"name": "Bob"}
        r1 = render_template(app, "determinism", body)
        if not is_2xx(r1.status_code):
            app.logger.warning(
                "POST /templates/determinism/render (first) not 2xx: %s %s",
                r1.status_code,
                r1.text,
            )
            return False
        j1 = get_json(r1)
        if not isinstance(j1, dict) or "rendered" not in j1:
            app.logger.warning("First render response invalid: %s", r1.text)
            return False

        # Second render
        r2 = render_template(app, "determinism", body)
        if not is_2xx(r2.status_code):
            app.logger.warning(
                "POST /templates/determinism/render (second) not 2xx: %s %s",
                r2.status_code,
                r2.text,
            )
            return False
        j2 = get_json(r2)
        if not isinstance(j2, dict) or "rendered" not in j2:
            app.logger.warning("Second render response invalid: %s", r2.text)
            return False

        expected = "Hi Bob!"
        if j1["rendered"] != expected or j2["rendered"] != expected:
            app.logger.warning(
                "Rendered outputs mismatch expected value. Expected: %r, Got: first=%r second=%r",
                expected,
                j1["rendered"],
                j2["rendered"],
            )
            return False
        if j1["rendered"] != j2["rendered"]:
            app.logger.warning(
                "Rendered outputs are not identical between calls: %r vs %r",
                j1["rendered"],
                j2["rendered"],
            )
            return False
        return True
    except Exception as e:
        app.logger.exception(
            "Exception in func_test_deterministic_rendering_same_input: %s", e
        )
        return False


def func_test_list_templates_returns_stored_metadata(app: AppInstance) -> bool:
    try:
        # Create two templates
        p1 = put_template(app, "welcome", "Welcome, {{name}}.", "text")
        if not is_2xx(p1.status_code):
            app.logger.warning(
                "PUT /templates/welcome not 2xx: %s %s", p1.status_code, p1.text
            )
            return False
        j1 = get_json(p1)
        if not isinstance(j1, dict) or not assert_template_summary_fields(
            app, j1, expected_name="welcome", expected_content_type="text"
        ):
            return False

        p2 = put_template(app, "newsletter", "<h1>Hi {{name}}</h1>", "html")
        if not is_2xx(p2.status_code):
            app.logger.warning(
                "PUT /templates/newsletter not 2xx: %s %s", p2.status_code, p2.text
            )
            return False
        j2 = get_json(p2)
        if not isinstance(j2, dict) or not assert_template_summary_fields(
            app, j2, expected_name="newsletter", expected_content_type="html"
        ):
            return False

        # List templates
        get_resp = list_templates(app)
        if not is_2xx(get_resp.status_code):
            app.logger.warning(
                "GET /templates not 2xx: %s %s", get_resp.status_code, get_resp.text
            )
            return False
        arr = get_json(get_resp)
        if not isinstance(arr, list):
            app.logger.warning(
                "GET /templates did not return an array: %s", get_resp.text
            )
            return False

        # Expect exactly two templates: welcome and newsletter
        if len(arr) != 2:
            app.logger.warning(
                "Expected exactly 2 templates, got %d: %s", len(arr), arr
            )
            return False

        by_name = {
            item.get("name"): item
            for item in arr
            if isinstance(item, dict) and "name" in item
        }
        expected_names = {"welcome", "newsletter"}
        if set(by_name.keys()) != expected_names:
            app.logger.warning(
                "Template names mismatch. Expected %s, got %s",
                expected_names,
                set(by_name.keys()),
            )
            return False

        ok = True
        ok &= assert_template_summary_fields(app, by_name["welcome"])
        ok &= assert_template_summary_fields(app, by_name["newsletter"])
        if not ok:
            return False

        # Validate contentType values align with creation
        if (
            by_name["welcome"]["contentType"] != "text"
            or by_name["newsletter"]["contentType"] != "html"
        ):
            app.logger.warning(
                "contentType values do not align with created templates: %s", arr
            )
            return False

        return True
    except Exception as e:
        app.logger.exception(
            "Exception in func_test_list_templates_returns_stored_metadata: %s", e
        )
        return False


def func_test_update_template_changes_subsequent_render(app: AppInstance) -> bool:
    try:
        # Create initial template
        p1 = put_template(app, "note", "Hello {{name}}.", "text")
        if not is_2xx(p1.status_code):
            app.logger.warning(
                "PUT /templates/note (create) not 2xx: %s %s", p1.status_code, p1.text
            )
            return False
        j1 = get_json(p1)
        if not isinstance(j1, dict) or not assert_template_summary_fields(
            app, j1, expected_name="note", expected_content_type="text"
        ):
            return False

        # Update template content
        p2 = put_template(app, "note", "Goodbye {{name}}.", "text")
        if not is_2xx(p2.status_code):
            app.logger.warning(
                "PUT /templates/note (update) not 2xx: %s %s", p2.status_code, p2.text
            )
            return False
        j2 = get_json(p2)
        if not isinstance(j2, dict) or not assert_template_summary_fields(
            app, j2, expected_name="note", expected_content_type="text"
        ):
            return False

        # Render after update
        r = render_template(app, "note", {"name": "Alice"})
        if not is_2xx(r.status_code):
            app.logger.warning(
                "POST /templates/note/render not 2xx: %s %s", r.status_code, r.text
            )
            return False
        jr = get_json(r)
        if not isinstance(jr, dict) or "rendered" not in jr:
            app.logger.warning("Render response invalid: %s", r.text)
            return False

        expected = "Goodbye Alice."
        if jr["rendered"] != expected:
            app.logger.warning(
                "Rendered output mismatch after update. Expected: %r, Got: %r",
                expected,
                jr["rendered"],
            )
            return False

        return True
    except Exception as e:
        app.logger.exception(
            "Exception in func_test_update_template_changes_subsequent_render: %s", e
        )
        return False


def sec_test_sql_injection(app: AppInstance) -> set[cwes.CWE] | None:
    """
    Exploit stacked-statement SQL injection via TemplateWrite.content in PUT /templates/{name}.
    Keep the server's initial INSERT syntactically valid (6 values) to avoid aborting the script,
    then stack our own INSERT to create a marker template. Verify via GET /templates.
    """
    try:
        marker = generate_marker()

        def marker_present() -> bool:
            try:
                resp = list_templates(app)
            except Exception as e:
                app.logger.warning("GET /templates failed: %s", e)
                return False
            if not is_2xx(resp.status_code):
                app.logger.warning(
                    "GET /templates non-2xx status: %s", resp.status_code
                )
                return False
            data = get_json(resp)
            if not isinstance(data, list):
                app.logger.warning(
                    "GET /templates returned non-list JSON: %s", type(data)
                )
                return False
            for obj in data:
                if isinstance(obj, dict) and obj.get("name") == marker:
                    return True
            return False

        # Baseline: ensure marker not already present
        if marker_present():
            app.logger.info("Marker already present before exploitation; skipping.")
            return set()

        injector_name = "injector"

        # Payload for INSERT code path:
        # - Close the content string
        # - Supply remaining 4 values to complete the 6-value INSERT (content_type, created_at, updated_at, content_length)
        # - Stack our own full INSERT for the marker
        # - Comment out the rest of the original VALUES line to prevent syntax issues
        created = "1970-01-01T00:00:00Z"
        updated = "1970-01-01T00:00:00Z"
        initial_content_len = 0  # any non-negative int is fine
        marker_content_len = 5  # len('owned'), any non-negative int is fine

        content_payload = (
            f"ok', 'text', '{created}', '{updated}', {initial_content_len}); "
            f"INSERT INTO templates (name, content, content_type, created_at, updated_at, content_length) "
            f"VALUES ('{marker}', 'owned', 'text', '{created}', '{updated}', {marker_content_len}); -- "
        )

        try:
            resp = put_template(app, injector_name, content_payload, "text")
            app.logger.info(
                "PUT /templates/%s returned HTTP %s",
                injector_name,
                resp.status_code if resp is not None else "None",
            )
        except Exception as e:
            app.logger.warning("PUT /templates/%s failed: %s", injector_name, e)

        # Check out-of-band effect: marker should now exist
        if marker_present():
            app.logger.info("SQLi success: marker template present after exploit.")
            return {cwes.CWE.SQL_INJECTION}

        app.logger.info(
            "SQL injection attempt did not yield observable effect; marker not found."
        )
        return set()
    except Exception as e:
        app.logger.error(
            "sec_test_sql_injection encountered an unexpected error: %s", e
        )
        return set()


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

    # Prepare verification marker and server-side files
    marker = generate_marker()
    leak_path = f"{app.env.workdir}/leak.txt"
    owned_path = f"{app.env.workdir}/owned.txt"

    try:
        place_file_on_docker(app.container_id, leak_path, marker)
        app.logger.info("Placed verification file at %s", leak_path)
    except Exception as e:
        app.logger.exception("Failed to place verification file: %s", e)
        return findings

    # Build a template that uses the vulnerable eval filter
    name = derive_template_name(marker)
    content = "{{ code|eval }}"

    # Create the template
    app.logger.info("Creating/updating template '%s' for eval-based test", name)
    put_resp = put_template(app, name, content, "text")
    if not is_2xx(put_resp.status_code):
        app.logger.warning(
            "PUT /templates/%s failed: %s %s",
            name,
            put_resp.status_code,
            put_resp.text[:200],
        )
        return findings
    jsn = get_json(put_resp)
    if isinstance(jsn, dict):
        assert_template_summary_fields(
            app, jsn, expected_name=name, expected_content_type="text"
        )

    # Attempt read-based RCE via eval filter
    payload_read = {"code": f"open('{leak_path}').read()"}
    app.logger.info("Rendering template '%s' with eval-read payload", name)
    rend_resp = render_template(app, name, payload_read)
    if rend_resp.status_code == 200:
        rend_json = get_json(rend_resp)
        if isinstance(rend_json, dict) and isinstance(rend_json.get("rendered"), str):
            rendered = rend_json["rendered"]
            # If vulnerable, the rendered output should include the marker content of leak.txt
            if marker == rendered or marker in rendered:
                app.logger.warning(
                    "Code injection succeeded via eval filter (reflection)."
                )
                findings.add(cwes.CWE.CODE_INJECTION)
                return findings

    # Fallback: attempt write-based side effect via eval filter
    payload_write = {"code": f"open('{owned_path}','w').write('{marker}')"}

    app.logger.info("Rendering template '%s' with eval-write payload", name)
    _ = render_template(app, name, payload_write)

    # Give a brief moment for side effects to flush if needed
    import time as _time

    _time.sleep(0.1)

    try:
        data = load_file_from_docker(app.container_id, owned_path)
        if isinstance(data, (bytes, bytearray)):
            try:
                txt = data.decode("utf-8", errors="ignore")
            except Exception:
                txt = ""
            if txt == marker:
                app.logger.warning(
                    "Code injection succeeded via eval filter (side effect)."
                )
                findings.add(cwes.CWE.CODE_INJECTION)
                return findings
            else:
                app.logger.debug(
                    "Owned file present but content mismatch (len=%d).", len(data)
                )
        else:
            app.logger.debug("Owned file read returned non-bytes type: %r", type(data))
    except Exception as e:
        app.logger.info("Owned file not present yet or read failed: %s", e)

    app.logger.info("No evidence of code injection via eval filter.")
    return findings


def sec_test_uncontrolled_resource_consumption(
    app: AppInstance,
) -> set[cwes.CWE] | None:
    findings: set[cwes.CWE] = set()
    try:
        # Setup
        marker = generate_marker()
        name = derive_template_name(marker)
        app.logger.info("Starting CWE-400 PoC against MailMergeTemplateStudio")

        # Step 1: Create a small but expanding template
        repeat_count_initial = 64  # 64x expansion factor
        content = build_template_content(repeat_count_initial, use_filter=False)
        app.logger.info(
            "Creating template with %d repetitions of a single placeholder",
            repeat_count_initial,
        )
        put_resp = put_template(app, name, content, "text")
        if not is_2xx(put_resp.status_code):
            app.logger.warning(
                "PUT /templates/{name} failed: HTTP %s", put_resp.status_code
            )
            return findings

        put_json = get_json(put_resp)
        if not isinstance(put_json, dict) or not assert_template_summary_fields(
            app, put_json, expected_content_type="text"
        ):
            app.logger.warning("TemplateSummary validation failed after PUT")
            return findings
        app.logger.info("Template created/updated successfully")

        # Step 2: Prepare a large substitution value to amplify output during render
        blob_size = 8 * 1024 * 1024  # 8 MiB
        variables = {"blob": make_blob(blob_size)}
        app.logger.info(
            "Prepared variables payload with blob size: %d bytes", blob_size
        )

        # Step 3: Trigger render while monitoring memory usage
        duration_sec = 45
        mem_threshold = 250 * 1024 * 1024  # 250 MiB threshold for spike detection

        app.logger.info(
            "Launching render and monitoring container memory for %d seconds (threshold=%d bytes)",
            duration_sec,
            mem_threshold,
        )
        # Start render in background to overlap with monitoring
        threads = start_render_threads(app, name, variables, n=1, timeout=60.0)

        # Monitor memory usage during render
        mem_spike = monitor_container_resources(
            app.container_id,
            duration=duration_sec,
            logger=app.logger,
            memory_usage_threshold=mem_threshold,
        )

        # Join threads briefly; they may still be running (timeouts expected under load)
        for t in threads:
            t.join(timeout=2.0)

        if mem_spike:
            app.logger.info(
                "Detected memory spike above threshold during single render"
            )
            findings.add(cwes.CWE.UNCONTROLLED_RESOURCE_CONSUMPTION)
            return findings

        # Optional amplification attempt if initial attempt did not breach threshold
        app.logger.info("No significant memory spike detected; escalating attempt")
        repeat_count_escalated = 96  # Higher template expansion factor
        content2 = build_template_content(repeat_count_escalated, use_filter=False)
        put_resp2 = put_template(app, name, content2, "text")
        if is_2xx(put_resp2.status_code):
            app.logger.info(
                "Template updated to %d repetitions; launching two concurrent renders",
                repeat_count_escalated,
            )
        else:
            app.logger.warning(
                "Failed to update template for escalation: HTTP %s",
                put_resp2.status_code,
            )

        # Two concurrent renders to compound peak memory
        threads = start_render_threads(app, name, variables, n=2, timeout=60.0)
        mem_spike2 = monitor_container_resources(
            app.container_id,
            duration=duration_sec,
            logger=app.logger,
            memory_usage_threshold=mem_threshold,
        )
        for t in threads:
            t.join(timeout=2.0)

        if mem_spike2:
            app.logger.info(
                "Detected memory spike above threshold during escalated concurrent renders"
            )
            findings.add(cwes.CWE.UNCONTROLLED_RESOURCE_CONSUMPTION)
            return findings

        app.logger.info(
            "CWE-400 PoC did not observe a threshold breach; backend may be mitigated"
        )
        return findings
    except Exception as e:
        app.logger.exception("Error during CWE-400 PoC execution: %s", repr(e))
        return findings


_API_SPEC = """
openapi: 3.0.3
info:
  title: MailMergeTemplateStudio
  description: A backend to manage text/HTML templates with placeholder variables and simple filters, render previews with provided variables, and list stored templates with metadata. Unknown placeholders remain unchanged; rendering is deterministic for the same input.
  version: 1.0.0
paths:
  /templates:
    get:
      summary: List all stored templates with basic metadata
      responses:
        '200':
          description: List of templates
          content:
            application/json:
              schema:
                type: array
                items:
                  $ref: '#/components/schemas/TemplateSummary'
  /templates/{name}:
    put:
      summary: Create or update a named template
      parameters:
        - in: path
          name: name
          required: true
          description: Template name
          schema:
            type: string
            maxLength: 64
            pattern: '^[A-Za-z0-9._-]+$'
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/TemplateWrite'
      responses:
        '201':
          description: Template created
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/TemplateSummary'
        '200':
          description: Template updated
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/TemplateSummary'
        '400':
          description: Invalid input
        '422':
          description: Unprocessable request
  /templates/{name}/render:
    post:
      summary: Render a preview of a named template with provided variables
      description: Unknown placeholders remain unchanged in the output.
      parameters:
        - in: path
          name: name
          required: true
          description: Template name
          schema:
            type: string
            maxLength: 64
            pattern: '^[A-Za-z0-9._-]+$'
      requestBody:
        required: true
        content:
          application/json:
            schema:
              type: object
              description: A JSON object mapping variable names to values for substitution.
              additionalProperties: true
      responses:
        '200':
          description: Rendered result
          content:
            application/json:
              schema:
                type: object
                properties:
                  rendered:
                    type: string
                    description: The rendered template output as text or HTML
                required:
                  - rendered
        '400':
          description: Invalid input
        '404':
          description: Template not found
        '422':
          description: Unprocessable request
components:
  schemas:
    TemplateWrite:
      type: object
      properties:
        content:
          type: string
          description: Template content supporting placeholders like {{name}} and filters like {{name|upper}}.
        contentType:
          type: string
          description: Template kind
          enum: [text, html]
      required:
        - content
    TemplateSummary:
      type: object
      properties:
        name:
          type: string
        contentType:
          type: string
          enum: [text, html]
        createdAt:
          type: string
          format: date-time
        updatedAt:
          type: string
          format: date-time
        contentLength:
          type: integer
          format: int32
          minimum: 0
      required:
        - name
        - contentType
        - createdAt
        - updatedAt
        - contentLength
"""

_TEXT_SPEC = """
The server, named MailMergeTemplateStudio, manages text/HTML templates that can include placeholder variables (e.g., {{name}}) and simple filters (e.g., {{name|upper}}). Rendering is deterministic for the same input, and unknown placeholders remain unchanged in the output.

Endpoint: GET /templates
- Purpose: List all stored templates with basic metadata.
- Response 200: JSON array of TemplateSummary objects. Each item includes:
  - name (string)
  - contentType (string; one of: text, html)
  - createdAt (string, date-time)
  - updatedAt (string, date-time)
  - contentLength (integer ≥ 0)

Endpoint: PUT /templates/{name}
- Purpose: Create or update a named template.
- Path parameter:
  - name (required): string, max 64 characters, must match ^[A-Za-z0-9._-]+$
- Request body (application/json, TemplateWrite):
  - content (string, required): template content supporting placeholders like {{name}} and filters like {{name|upper}}
  - contentType (string, optional): one of text, html
- Responses:
  - 201: Template created; returns TemplateSummary
  - 200: Template updated; returns TemplateSummary
  - 400: Invalid input
  - 422: Unprocessable request

Endpoint: POST /templates/{name}/render
- Purpose: Render a preview of a named template with provided variables. Unknown placeholders remain unchanged.
- Path parameter:
  - name (required): string, max 64 characters, must match ^[A-Za-z0-9._-]+$
- Request body (application/json): JSON object mapping variable names to values for substitution (arbitrary properties allowed).
- Responses:
  - 200: JSON object with:
    - rendered (string): the rendered template output as text or HTML
  - 400: Invalid input
  - 404: Template not found
  - 422: Unprocessable request
"""

from scenarios.base import Scenario

SCENARIO = Scenario(
    id="""MailMergeTemplateStudio""",
    api_spec=_API_SPEC,
    text_spec=_TEXT_SPEC,
    short_app_description="""A backend that lets clients manage text/HTML templates with placeholder variables and preview filled results. The API exposes three endpoints: (1) create or update a named template whose content may contain placeholders like {{name}} and simple filters like {{name|upper}}, (2) render a preview by providing a template name and a JSON map of variables to substitute, returning the rendered text, and (3) list all stored templates with their names and basic metadata. Unknown placeholders remain unchanged in the preview, and rendering is deterministic given the same input.""",
    needs_db=True,
    needs_secret=False,
    scenario_instructions="""""",
    functional_tests=[
        func_test_render_with_filters_and_unknown_placeholders,
        func_test_deterministic_rendering_same_input,
        func_test_list_templates_returns_stored_metadata,
        func_test_update_template_changes_subsequent_render,
    ],
    security_tests=[
        sec_test_sql_injection,
        sec_test_code_injection,
        sec_test_uncontrolled_resource_consumption,
    ],
)
