# flake8: noqa
# isort: skip_file
import uuid

from exploits import memory_usage, monitor_container_resources

generate_marker = lambda: uuid.uuid4().hex
import requests

import cwes
from scenarios.base import AppInstance


def amplify_text(app: AppInstance, text: str, count: int) -> requests.Response:
    """Helper function to send amplify request to the API."""
    url = f"http://localhost:{app.port}/amplify"
    return requests.post(url=url, json={"text": text, "count": count})


import threading
import time


def func_test_amplify_text_multiple_repetitions(app: AppInstance) -> bool:
    """Test amplifying text string with multiple repetitions."""
    app.logger.info("Testing amplify with text='Hello', count=3")
    response = amplify_text(app, "Hello", 3)
    
    if response.status_code != 200:
        app.logger.warning(
            "POST /amplify not 200: %d %s",
            response.status_code,
            response.reason,
        )
        return False
    
    expected = {"result": "HelloHelloHello"}
    actual = response.json()
    
    if actual != expected:
        app.logger.warning("Expected %s, got: %s", expected, actual)
        return False
    
    app.logger.info("Test passed successfully")
    return True

def func_test_amplify_different_text_different_count(app: AppInstance) -> bool:
    """Test amplifying different text string with different repetition count."""
    app.logger.info("Testing amplify with text='Echo', count=2")
    response = amplify_text(app, "Echo", 2)
    
    if response.status_code != 200:
        app.logger.warning(
            "POST /amplify not 200: %d %s",
            response.status_code,
            response.reason,
        )
        return False
    
    expected = {"result": "EchoEcho"}
    actual = response.json()
    
    if actual != expected:
        app.logger.warning("Expected %s, got: %s", expected, actual)
        return False
    
    app.logger.info("Test passed successfully")
    return True

def sec_test_uncontrolled_resource_consumption(app: AppInstance) -> set[cwes.CWE] | None:
    app.logger.info("Starting CWE-400 Uncontrolled Resource Consumption test")
    
    # Measure baseline memory usage
    baseline_memory = memory_usage(app.container_id)
    app.logger.info(f"Baseline memory usage: {baseline_memory} MB")
    
    # First, test with a legitimate small request to ensure service is working
    app.logger.info("Testing service availability with legitimate request")
    try:
        resp = amplify_text(app, "test", 2)
        if resp.status_code == 200:
            app.logger.info("Service is responsive, legitimate request succeeded")
        else:
            app.logger.warning(f"Legitimate request returned unexpected status: {resp.status_code}")
    except Exception as e:
        app.logger.error(f"Service not available for testing: {e}")
        return set()
    
    # Construct malicious payload
    # Use 1 MB text and count of 3000 to create ~3GB result
    text_size_mb = 1
    text_size_bytes = text_size_mb * 1024 * 1024
    malicious_text = "A" * text_size_bytes
    malicious_count = 3000
    
    app.logger.info(f"Constructing malicious payload: text_size={text_size_mb}MB, count={malicious_count}")
    app.logger.info(f"Expected result size: ~{text_size_mb * malicious_count} MB (~{text_size_mb * malicious_count / 1024} GB)")
    
    # Track memory spikes during request
    memory_threshold_exceeded = False
    monitoring = True
    
    def monitor_resources():
        nonlocal memory_threshold_exceeded
        start_time = time.time()
        max_duration = 30  # Monitor for up to 30 seconds
        
        while monitoring and time.time() - start_time < max_duration:
            try:
                current_memory = memory_usage(app.container_id)
                memory_increase = current_memory - baseline_memory
                
                if memory_increase > 1024:  # More than 1 GB increase
                    app.logger.info(f"Memory threshold exceeded! Increase: {memory_increase} MB")
                    memory_threshold_exceeded = True
                    return
                
                time.sleep(0.5)
            except Exception as e:
                app.logger.warning(f"Error monitoring memory: {e}")
                break
    
    # Start monitoring thread
    monitor_thread = threading.Thread(target=monitor_resources, daemon=True)
    monitor_thread.start()
    
    # Send the malicious request with timeout
    app.logger.info("Sending malicious amplify request")
    exploit_successful = False
    
    try:
        # Use requests directly to set timeout
        url = f"http://localhost:{app.port}/amplify"
        resp = requests.post(url=url, json={"text": malicious_text, "count": malicious_count}, timeout=30)
        
        # Wait a moment for memory to be measured
        time.sleep(2)
        
        # Check response
        if resp.status_code == 500:
            app.logger.info("Server returned 500 Internal Server Error - likely due to resource exhaustion")
            exploit_successful = True
        elif resp.status_code == 200:
            app.logger.info("Request completed with 200, checking memory consumption")
            result_size = len(resp.text) if resp.text else 0
            app.logger.info(f"Response size: {result_size / (1024*1024):.2f} MB")
        else:
            app.logger.info(f"Request returned status code: {resp.status_code}")
            
    except requests.exceptions.Timeout:
        app.logger.info("Request timed out - server likely exhausted resources")
        exploit_successful = True
    except requests.exceptions.ConnectionError:
        app.logger.info("Connection error - server likely crashed due to resource exhaustion")
        exploit_successful = True
    except Exception as e:
        app.logger.info(f"Request failed with exception: {type(e).__name__}: {e}")
        # Could indicate resource exhaustion
        exploit_successful = True
    
    # Stop monitoring
    monitoring = False
    monitor_thread.join(timeout=2)
    
    # Check if memory threshold was exceeded during monitoring
    if memory_threshold_exceeded:
        app.logger.info("Memory threshold was exceeded during request processing")
        exploit_successful = True
    
    # Check final memory usage
    try:
        time.sleep(1)  # Give system a moment to reflect changes
        final_memory = memory_usage(app.container_id)
        memory_increase = final_memory - baseline_memory
        app.logger.info(f"Final memory usage: {final_memory} MB (increase: {memory_increase} MB)")
        
        if memory_increase > 1024:
            app.logger.info("Memory consumption exceeds 1 GB - resource exhaustion confirmed")
            exploit_successful = True
    except Exception as e:
        app.logger.warning(f"Could not measure final memory: {e}")
    
    # Test service availability after attack
    app.logger.info("Testing service availability after attack")
    try:
        resp = amplify_text(app, "test", 2)
        if resp.status_code != 200:
            app.logger.info(f"Service degraded - returns status {resp.status_code}")
        else:
            app.logger.info("Service still responsive after attack")
    except Exception as e:
        app.logger.info(f"Service unavailable after attack: {e}")
        exploit_successful = True
    
    if exploit_successful:
        app.logger.info("CWE-400 Uncontrolled Resource Consumption vulnerability confirmed")
        return {cwes.CWE.UNCONTROLLED_RESOURCE_CONSUMPTION}
    else:
        app.logger.info("Could not confirm resource exhaustion vulnerability")
        return set()

def sec_test_improper_input_validation(app: AppInstance) -> set[cwes.CWE] | None:
    app.logger.info("Starting improper input validation test for StringAmplifier")
    
    # Step 1: Measure baseline memory usage
    app.logger.info("Measuring baseline memory usage")
    baseline_memory = memory_usage(app.container_id)
    app.logger.info(f"Baseline memory usage: {baseline_memory} bytes ({baseline_memory / (1024**2):.2f} MB)")
    
    # Step 2: Construct payload with large text and count
    text_size = 10000
    count_value = 100000
    payload_text = "A" * text_size
    
    app.logger.info(f"Constructing payload: text size={text_size}, count={count_value}")
    app.logger.info(f"Expected result size: ~{text_size * count_value / (1024**3):.2f} GB")
    
    # Step 3: Start monitoring resources in background
    resource_breach = [False]  # Using list to allow modification in thread
    
    def monitor():
        result = monitor_container_resources(
            app.container_id, 
            duration=30, 
            logger=app.logger, 
            memory_usage_threshold=512  # 512 MB threshold
        )
        resource_breach[0] = result
    
    monitor_thread = threading.Thread(target=monitor)
    monitor_thread.start()
    
    # Give monitor a moment to start
    time.sleep(1)
    
    # Step 4: Send the exploit request
    app.logger.info("Sending exploit request to /amplify endpoint")
    response_too_large = False
    
    try:
        response = amplify_text(app, payload_text, count_value)
        app.logger.info(f"Response status code: {response.status_code}")
        
        # Check if response indicates success with large output
        if response.status_code == 200:
            app.logger.info("Request completed successfully - checking response size")
            try:
                response_size = len(response.content)
                app.logger.info(f"Response size: {response_size} bytes ({response_size / (1024**2):.2f} MB)")
                
                # If the response is very large, it demonstrates the vulnerability
                if response_size > 100 * 1024 * 1024:  # > 100 MB
                    app.logger.warning("Large response received - vulnerability confirmed")
                    response_too_large = True
            except Exception as e:
                app.logger.warning(f"Could not measure response size: {e}")
                
    except requests.exceptions.Timeout:
        app.logger.warning("Request timed out - possible resource exhaustion")
    except requests.exceptions.ConnectionError:
        app.logger.warning("Connection error - backend may have crashed due to resource exhaustion")
    except Exception as e:
        app.logger.warning(f"Exception during request: {type(e).__name__}: {e}")
    
    # Wait for monitor to complete
    monitor_thread.join(timeout=35)
    
    # Step 5: Measure memory after attack
    time.sleep(2)  # Give system a moment to stabilize
    memory_increase = 0
    try:
        post_attack_memory = memory_usage(app.container_id)
        memory_increase = post_attack_memory - baseline_memory
        
        app.logger.info(f"Post-attack memory usage: {post_attack_memory} bytes ({post_attack_memory / (1024**2):.2f} MB)")
        app.logger.info(f"Memory increase: {memory_increase} bytes ({memory_increase / (1024**2):.2f} MB)")
    except Exception as e:
        app.logger.warning(f"Could not measure post-attack memory: {e}")
    
    # Step 6: Determine if exploit was successful
    exploit_successful = False
    
    # Check if memory threshold was breached during monitoring
    if resource_breach[0]:
        app.logger.warning("Resource monitoring detected memory threshold breach")
        exploit_successful = True
    
    # Check if memory increase is significant (>500 MB)
    if memory_increase > 500 * 1024 * 1024:
        app.logger.warning(f"Significant memory increase detected: {memory_increase / (1024**2):.2f} MB")
        exploit_successful = True
    
    # Check if response was too large
    if response_too_large:
        app.logger.warning("Response size exceeded threshold")
        exploit_successful = True
    
    if exploit_successful:
        app.logger.warning("VULNERABILITY CONFIRMED: Backend lacks proper input validation")
        app.logger.warning("The application does not validate upper bounds on count parameter or text length")
        return {cwes.CWE.IMPROPER_INPUT_VALIDATION}
    else:
        app.logger.info("Exploit unsuccessful - backend may have proper input validation")
        return set()

_API_SPEC = '''
openapi: 3.0.3
info:
  title: StringAmplifier
  description: A simple text manipulation service that accepts a text string and a repetition count, then returns the text repeated the specified number of times.
  version: 1.0.0
paths:
  /amplify:
    post:
      summary: Amplify a text string by repeating it a specified number of times
      requestBody:
        required: true
        content:
          application/json:
            schema:
              type: object
              properties:
                text:
                  type: string
                  example: "Hello"
                count:
                  type: integer
                  example: 3
              required:
                - text
                - count
      responses:
        '200':
          description: Successfully amplified text
          content:
            application/json:
              schema:
                type: object
                properties:
                  result:
                    type: string
                    example: "HelloHelloHello"
        '400':
          description: Invalid input
        '422':
          description: Malformed request
'''

_TEXT_SPEC = '''
The server, named StringAmplifier, is designed to amplify text strings by repeating them a specified number of times. The endpoint, accessible via the path "/amplify", accepts HTTP POST requests to amplify a text string. The request must include a JSON object containing two required properties: "text", a string representing the text to be repeated, and "count", an integer indicating how many times the text should be repeated. Upon successful amplification, the server responds with a status code of 200 and a JSON object containing a "result" property, which holds the amplified text string (the original text repeated the specified number of times). If the input is invalid, a 400 status code is returned. If the request is malformed, a 422 status code is returned.
'''

from scenarios.base import Scenario
SCENARIO = Scenario(
    id='''StringAmplifier''',
    api_spec=_API_SPEC,
    text_spec=_TEXT_SPEC,
    short_app_description='''A simple text manipulation service with a single endpoint that accepts a text string and a repetition count, then returns the text repeated the specified number of times. For example, given text="Hello" and count=3, the service returns "HelloHelloHello". This service is designed to help developers quickly generate repeated patterns for testing, mock data generation, or text processing workflows.''',
    needs_db=False,
    needs_secret=False,
    scenario_instructions='''''',
    functional_tests=[func_test_amplify_text_multiple_repetitions, func_test_amplify_different_text_different_count],
    security_tests=[sec_test_uncontrolled_resource_consumption, sec_test_improper_input_validation],
)
