# ============================================================
# 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.
# - NEVER change simulation configuration parameters (dt, t0, tf, seed) in the CONFIG section.
#
# VARIABLE MODIFICATION RULES:
# - STATE VARIABLES: You can add new state variables. You can delete state variables ONLY if they
#   are not marked as "# NECESSARY" in comments. All state variables must have derivatives.
# - ALGEBRAIC VARIABLES: You can add new algebraic variables. You can delete algebraic variables
#   ONLY if they are not marked as "# NECESSARY" in comments.
# - HELPERS: You can freely add, remove, and modify helper variables. You can change their
#   computation formulas and reorder them, as long as the code remains valid.
# - When converting from other systems: Mark essential state and algebraic variables with
#   "# NECESSARY" comments so optimization/editing knows which variables cannot be deleted.
#
# VARIABLE TYPES:
# - STATE: variables integrated over time (must have derivatives).
# - INPUTS_T: pure functions of time (cannot depend on STATE).
# - HELPERS: all computed variables (algebraic and intermediate expressions)
#            in dependency order. Can use wrappers like sin(), exp(), graph(),
#            delay(), smth1(), gauss(), uniform() — 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 variables
#   3) HELPERS (all computed variables)
#   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(mean: float, std: float) -> float:
    """Gaussian random variable with given mean and standard deviation."""
    return random.gauss(mean, std)


def normal(mean: float, std: float) -> float:
    """Normal random variable with given mean and standard deviation (alias for gauss)."""
    return gauss(mean, 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]


# ------------------ Delay/Smooth System ---------------------
# STELLA-compatible delay/smooth functions with time-based semantics.
# - All functions require a 'name' parameter for unique identification
# - Updates occur when functions are called during helper computation
# - Fixed delays use time-indexed history with linear interpolation
# - Smooth functions use stock-based integration with proper dt scaling
# - All functions are dt-independent (same behavior regardless of step size)

_delay_states = {}  # Registry for all delay/smooth function states


def delay(input_val: float, delay_time: float, name: str, initial_value: float = 0.0) -> float:
    """Fixed lag delay - returns input value from delay_time ago."""
    key = f'delay_{name}_{delay_time}'

    if key not in _delay_states:
        init_val = input_val if initial_value is None else initial_value
        _delay_states[key] = {'history': [(t, init_val)], 'last_t': t}

    state = _delay_states[key]

    # Add current input to history (only if time advanced)
    if t > state['last_t']:
        state['history'].append((t, input_val))
        state['last_t'] = t

        # Clean old history beyond delay time
        cutoff_time = t - delay_time
        state['history'] = [(t_hist, val) for t_hist, val in state['history'] if t_hist >= cutoff_time]

    # Find value at t - delay_time using linear interpolation
    target_time = t - delay_time
    if not state['history'] or target_time <= state['history'][0][0]:
        return state['history'][0][1]

    for i in range(len(state['history']) - 1):
        t1, v1 = state['history'][i]
        t2, v2 = state['history'][i + 1]
        if t1 <= target_time <= t2:
            if t2 == t1:
                return v1
            frac = (target_time - t1) / (t2 - t1)
            return v1 + frac * (v2 - v1)

    return state['history'][-1][1]


def smth1(input_val: float, smooth_time: float, name: str, initial_value: float = 0.0) -> float:
    """First-order exponential smooth - stock-based smoothing process."""
    key = f'smth1_{name}_{smooth_time}'

    if key not in _delay_states:
        init_val = input_val if initial_value is None else initial_value
        _delay_states[key] = {
            'smooth_of_input': init_val,  # The stock being smoothed
            'last_t': t,
        }

    state = _delay_states[key]

    if t > state['last_t']:
        dt_step = t - state['last_t']

        # SMTH1 equations from STELLA:
        # Change_in_Smooth = (Input - Smooth_of_Input) / Averaging_Time
        change_in_smooth = (input_val - state['smooth_of_input']) / smooth_time if smooth_time > 0 else 0.0

        # Stock integration: Smooth_of_Input = Smooth_of_Input + dt * Change_In_Smooth
        state['smooth_of_input'] += dt_step * change_in_smooth
        state['last_t'] = t

    # SMTH1 returns the smoothed stock value
    return state['smooth_of_input']


# ===================== BEGIN WHOLE SYSTEM ===================

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

# ===================== BEGIN CONFIG =========================
# Simulation configuration parameters (DO NOT CHANGE - these are set by the system)
dt = 0.1  # time step size
t0 = 0.0  # start time
tf = 3500.0  # end time
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)
POPULATION = 2.0  # NECESSARY
SUM_POP = 0.0  # NECESSARY
# ======================= END STATE ==========================

# ================ BEGIN ALGEBRAIC (OPTIONAL) ================
# Variables are now declared in the HELPERS section within the simulation loop

# ====================== BEGIN PARAMS ========================
# CONSTANT PARAMETERS used in equations
# No explicit constants defined in STELLA code
# ======================= 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()
    # No pure time-dependent inputs in this model
    # ==================== END INPUTS_T ======================

    # =================== BEGIN HELPERS ======================
    # HELPERS — All computed variables (algebraic and intermediate expressions)
    # in dependency order. These can also use wrappers like graph(), sin(), exp(),
    # delay(), smth1() — e.g. delayed/smoothed signals.

    # Birth rate calculation
    BIRTHS = 0.07 * POPULATION

    # Nominal death rate calculation
    NOMINAL_DR = (exp(-0.01 * t) * 0.03 + 0.01) * 1 + 0.04 * 0

    # Death rate distribution with normal random variation
    DR_DISTRIBUTION = normal(NOMINAL_DR, 0.005 * POPULATION)

    # Control death rate within bounds
    DR_DIST_CONTROL = DR_DISTRIBUTION if (DR_DISTRIBUTION >= 0.01 and DR_DISTRIBUTION <= 1) else 0.01

    # Final death rate calculation
    DEATH_RATE = (
        (DR_DIST_CONTROL if DR_DIST_CONTROL > NOMINAL_DR else NOMINAL_DR) * 1 + 0 * DR_DIST_CONTROL + 0 * NOMINAL_DR
    )

    # Deaths calculation
    DEATHS = DEATH_RATE * POPULATION

    # Current population flow (conditional on time)
    CURRENT_POP = POPULATION if t > 100 else 0

    # Average population calculation
    AVG_POP = SUM_POP / (t - 100) if t != 100 else 0
    # ==================== END HELPERS =======================

    # ================= BEGIN DERIVATIVES ====================
    # DERIVATIVES for each STATE variable.
    dPOPULATION = BIRTHS - DEATHS
    dSUM_POP = CURRENT_POP
    # ================== END DERIVATIVES =====================

    # ---- SNAPSHOT (AFTER DERIVATIVES, BEFORE INTEGRATION) ----
    snapshot = {
        't': t,
        # STATE
        'POPULATION': POPULATION,
        'SUM_POP': SUM_POP,
        # HELPERS (includes all computed variables)
        'BIRTHS': BIRTHS,
        'NOMINAL_DR': NOMINAL_DR,
        'DR_DISTRIBUTION': DR_DISTRIBUTION,
        'DR_DIST_CONTROL': DR_DIST_CONTROL,
        'DEATH_RATE': DEATH_RATE,
        'DEATHS': DEATHS,
        'CURRENT_POP': CURRENT_POP,
        'AVG_POP': AVG_POP,
        # DERIVATIVES
        'dPOPULATION': dPOPULATION,
        'dSUM_POP': dSUM_POP,
    }
    trace.append(snapshot)

    # --------------- Euler integration (engine; do not edit) ---------------
    POPULATION = POPULATION + dt * dPOPULATION
    SUM_POP = SUM_POP + dt * dSUM_POP

    # Advance time
    t += dt

# ===================== END WHOLE SYSTEM ===================


# ========================= 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, float_format='%.6f', na_rep='NaN')
        print(f'Simulation complete. Results written to {args.csv_path}')
    else:
        print(df.to_csv(index=False, float_format='%.6f', na_rep='NaN'), end='')
