# ============================================================
# sim_base.py — Minimal Euler simulator with edit markers
#
# PURPOSE:
# - Self-contained forward Euler simulator:
#       X(t+dt) = X(t) + dt * dX/dt
#
# HOW TO EDIT:
# - Only change code between the "BEGIN ... / END ..." markers below.
# - Do NOT modify imports, wrappers, loop structure, Euler integration, or OUTPUT section.
# - Keep STATE, ALGEBRAIC, HELPERS, and DERIVATIVES consistent.
#
# VARIABLE TYPES:
# - STATE: variables integrated over time (must have derivatives).
# - ALGEBRAIC: variables recomputed each step (not integrated).
# - INPUTS_T: pure functions of time (cannot depend on STATE).
# - INPUTS_RND: fresh random values each step (cannot depend on STATE).
# - HELPERS: intermediate reusable expressions, can also call wrappers
#            like sin(), exp(), graph() — e.g. nonlinear damping from a table.
# - DERIVATIVES: d/dt for each STATE variable.
#
# TRACE CONTENT (per step, in fixed order):
#   1) t            — current time
#   2) STATE + ALGEBRAIC variables
#   3) HELPERS
#   4) DERIVATIVES
#
# OUTPUT:
# - If --csv <path> is provided → write full simulation to that CSV file.
# - Otherwise → print the raw CSV (header + all rows) to stdout.
# - Columns/ordering come from the snapshot dict defined inside the loop.
# ============================================================

import math
import random
import argparse
import pandas as pd  # used only for CSV export
from typing import Iterable, Tuple

# ------------------ Wrapper Functions -----------------------
# These wrappers hide Python semantics; LLMs should ONLY call these.


def sin(x: float) -> float:
    """Sine function (angle in radians)."""
    return math.sin(x)


def cos(x: float) -> float:
    """Cosine function (angle in radians)."""
    return math.cos(x)


def exp(x: float) -> float:
    """Exponential function e^x."""
    return math.exp(x)


def tanh(x: float) -> float:
    """Hyperbolic tangent."""
    return math.tanh(x)


def sqrt(x: float) -> float:
    """Square root."""
    return math.sqrt(x)


def log(x: float) -> float:
    """Natural logarithm."""
    return math.log(x)


def gauss(std: float) -> float:
    """Gaussian random variable with mean=0 and standard deviation=std."""
    return random.gauss(0.0, std)


def uniform(lo: float, hi: float) -> float:
    """Uniform random variable between lo and hi."""
    return random.uniform(lo, hi)


def graph(v: float, table: Iterable[Tuple[float, float]]) -> float:
    """
    Lookup helper with linear interpolation.
    - table is an iterable of (x, y) points sorted by x.
    - If v < x0, return y0.
    - If v > xN, return yN.
    - Else, linearly interpolate between nearest points.
    """
    table = list(table)
    if not table:
        raise ValueError('graph() called with empty table')

    if v <= table[0][0]:
        return table[0][1]
    if v >= table[-1][0]:
        return table[-1][1]

    for i in range(len(table) - 1):
        x0, y0 = table[i]
        x1, y1 = table[i + 1]
        if x0 <= v <= x1:
            if x1 == x0:
                return y0
            frac = (v - x0) / (x1 - x0)
            return y0 + frac * (y1 - y0)
    return table[-1][1]


# ============================================================

# ===================== BEGIN CONFIG =========================
# Simulation configuration parameters
dt = 0.1  # time step size (weeks)
t0 = 0.0  # start time
tf = 520.0  # end time (10 years in weeks)
seed = 1234  # random seed (set None for nondeterministic runs)
# ====================== END CONFIG ==========================

if seed is not None:
    random.seed(seed)

# ====================== BEGIN STATE =========================
# STATE VARIABLES — integrated over time (must define derivatives below)
CONTAGIOUS = 1.0  # People
IMMUNE = 0.0  # People
NON_IMMUMNE = 1000000.0  # People
SICK = 0.0  # People
# ======================= END STATE ==========================

# ================ BEGIN ALGEBRAIC (OPTIONAL) ================
# ALGEBRAIC VARIABLES — recomputed each step (not integrated)
POPULATION = 0.0
# ================= END ALGEBRAIC (OPTIONAL) =================

# ====================== BEGIN PARAMS ========================
# CONSTANT PARAMETERS used in equations
VAC_FRACTION = 0.03
# ======================= END PARAMS =========================

# ----------------- Storage (engine; do not edit) ------------
trace = []  # list of dicts; each dict is one row of the output CSV

# ======================= MAIN LOOP ==========================
t = t0
while t <= tf + 1e-12:
    # =================== BEGIN INPUTS_T =====================
    # TIME-DEPENDENT INPUTS — use wrappers like sin(), exp(), graph()
    # ==================== END INPUTS_T ======================

    # ================= BEGIN INPUTS_RND =====================
    # RANDOM INPUTS — use gauss(), uniform() wrappers (no STATE usage).
    # ================== END INPUTS_RND ======================

    # ================== BEGIN ALGEBRAIC =====================
    # ALGEBRAIC VARIABLES — recomputed directly from STATE and inputs.
    POPULATION = CONTAGIOUS + IMMUNE + NON_IMMUMNE + SICK
    # =================== END ALGEBRAIC ======================

    # =================== BEGIN HELPERS ======================
    # HELPERS — intermediate expressions. These can also use
    # wrappers like graph(), sin(), exp().

    BIRTH_RATE = graph(
        POPULATION,
        (
            (0.0, 0.199),
            (166667, 0.189),
            (333333, 0.181),
            (500000, 0.171),
            (666667, 0.159),
            (833333, 0.146),
            (1e06, 0.138),
            (1.2e06, 0.133),
            (1.3e06, 0.129),
            (1.5e06, 0.125),
            (1.7e06, 0.123),
            (1.8e06, 0.122),
            (2e06, 0.12),
        ),
    )

    NAT_DEATH_RATE = graph(
        POPULATION,
        (
            (0.0, 0.0108),
            (208333, 0.0108),
            (416667, 0.0111),
            (625000, 0.0135),
            (833333, 0.0168),
            (1e06, 0.0207),
            (1.2e06, 0.0258),
            (1.5e06, 0.0321),
            (1.7e06, 0.0363),
            (1.9e06, 0.0387),
            (2.1e06, 0.0411),
            (2.3e06, 0.0423),
            (2.5e06, 0.0426),
        ),
    )

    CC = 0.000002 * (1 - 1 * IMMUNE / (IMMUNE + NON_IMMUMNE)) if (IMMUNE + NON_IMMUMNE) > 0 else 0.0

    # Flow calculations
    BIRTHING = POPULATION * BIRTH_RATE / 52
    IMMUNE_LOSS = 0.002 * IMMUNE
    CONTRACTING = CC * NON_IMMUMNE * CONTAGIOUS if IMMUNE > 0 else 0.0
    NONIMMUNE_DYING = NAT_DEATH_RATE / 52 * NON_IMMUMNE
    VACCIN_ATING = VAC_FRACTION * NON_IMMUMNE - 0.15 * IMMUNE

    CONTAGIOUS_DYING = CONTAGIOUS * NAT_DEATH_RATE / 52
    BED_DING = CONTAGIOUS - CONTAGIOUS_DYING

    RECOVERING = 0.9 * SICK
    IMMUNE__DYING = IMMUNE * NAT_DEATH_RATE / 52

    SICK_DYING = (0.1 * SICK) + (NAT_DEATH_RATE / 52 * SICK)
    # ==================== END HELPERS =======================

    # ================= BEGIN DERIVATIVES ====================
    # DERIVATIVES for each STATE variable.
    dCONTAGIOUS = CONTRACTING - CONTAGIOUS_DYING - BED_DING
    dIMMUNE = RECOVERING + VACCIN_ATING - IMMUNE__DYING - IMMUNE_LOSS
    dNON_IMMUMNE = BIRTHING + IMMUNE_LOSS - CONTRACTING - NONIMMUNE_DYING - VACCIN_ATING
    dSICK = BED_DING - RECOVERING - SICK_DYING
    # ================== END DERIVATIVES =====================

    # ---- SNAPSHOT (AFTER DERIVATIVES, BEFORE INTEGRATION) ----
    snapshot = {
        't': t,
        # STATE + ALGEBRAIC
        'CONTAGIOUS': CONTAGIOUS,
        'IMMUNE': IMMUNE,
        'NON_IMMUMNE': NON_IMMUMNE,
        'SICK': SICK,
        'POPULATION': POPULATION,
        # HELPERS
        'BIRTH_RATE': BIRTH_RATE,
        'NAT_DEATH_RATE': NAT_DEATH_RATE,
        'CC': CC,
        'BIRTHING': BIRTHING,
        'IMMUNE_LOSS': IMMUNE_LOSS,
        'CONTRACTING': CONTRACTING,
        'NONIMMUNE_DYING': NONIMMUNE_DYING,
        'VACCIN_ATING': VACCIN_ATING,
        'CONTAGIOUS_DYING': CONTAGIOUS_DYING,
        'BED_DING': BED_DING,
        'RECOVERING': RECOVERING,
        'IMMUNE__DYING': IMMUNE__DYING,
        'SICK_DYING': SICK_DYING,
        # DERIVATIVES
        'dCONTAGIOUS': dCONTAGIOUS,
        'dIMMUNE': dIMMUNE,
        'dNON_IMMUMNE': dNON_IMMUMNE,
        'dSICK': dSICK,
    }
    trace.append(snapshot)

    # --------------- Euler integration (engine; do not edit) ---------------
    CONTAGIOUS = CONTAGIOUS + dt * dCONTAGIOUS
    IMMUNE = IMMUNE + dt * dIMMUNE
    NON_IMMUMNE = NON_IMMUMNE + dt * dNON_IMMUMNE
    SICK = SICK + dt * dSICK

    # Advance time
    t += dt


# ========================= OUTPUT (FIXED) ===================
# DO NOT EDIT THIS SECTION.
def _parse_args():
    ap = argparse.ArgumentParser(description='Run Euler simulation and export CSV.')
    ap.add_argument(
        '--csv', dest='csv_path', default=None, help='Write results to this CSV path. If omitted, prints CSV to stdout.'
    )
    return ap.parse_args()


if __name__ == '__main__':
    args = _parse_args()
    df = pd.DataFrame(trace)  # column order = insertion order of dict
    if args.csv_path:
        df.to_csv(args.csv_path, index=False, na_rep='NaN')
        print(f'Simulation complete. Results written to {args.csv_path}')
    else:
        print(df.to_csv(index=False, na_rep='NaN'), end='')
