# ============================================================
# 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.001  # time step size
t0 = 0.0  # start time
tf = 5.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)
A = 1.0  # example: position
B = 0.0  # example: velocity
# ======================= END STATE ==========================

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

# ====================== BEGIN PARAMS ========================
# CONSTANT PARAMETERS used in equations
m = 1.0
c = 0.2
k = 4.0
F0 = 1.0
w = 2.0 * math.pi * 1.0
sigma_eta = 0.0
# ======================= 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()
    # Example: oscillatory force with time-varying amplitude (lookup table inlined).
    F_amp = graph(
        t,
        (
            (0.0, 0.0),
            (1.0, 0.5),
            (2.0, 1.0),
            (3.0, 0.8),
            (5.0, 0.2),
        ),
    )
    F_t = F_amp * sin(w * t)
    # ==================== 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.
    #
    # DELAY/SMOOTH FUNCTIONS require a 'name' parameter for unique identification.
    # Use descriptive names, preferably matching the delayed variable name:
    #   delayed_population = delay(population, 5.0, "population")
    #   smoothed_temp = smth1(raw_temperature, 1.5, "temperature")
    #
    # Variables computed each step (order matters for dependencies):
    C = A + 0.5 * F_t  # Algebraic variable computed from STATE and inputs

    # Example: nonlinear damping coefficient depending on velocity magnitude.
    damping_coeff = graph(
        abs(B),
        (
            (0.0, c),  # low velocity → base damping
            (1.0, c * 1.5),
            (2.0, c * 2.0),
        ),
    )

    spring_force = -k * A
    damping_force = -damping_coeff * B  # uses velocity-dependent damping
    drive_force = F_t
    # Random inputs can now depend on helper variables and be computed in dependency order
    eta = gauss(0.0, sigma_eta) if sigma_eta > 0.0 else 0.0
    noise_force = eta
    net_force = spring_force + damping_force + drive_force + noise_force
    acceleration = net_force / m
    # ==================== END HELPERS =======================

    # ================= BEGIN DERIVATIVES ====================
    # DERIVATIVES for each STATE variable.
    dA = B
    dB = acceleration
    # ================== END DERIVATIVES =====================

    # ---- SNAPSHOT (AFTER DERIVATIVES, BEFORE INTEGRATION) ----
    snapshot = {
        't': t,
        # STATE
        'A': A,
        'B': B,
        # HELPERS (includes all computed variables)
        'C': C,
        'damping_coeff': damping_coeff,
        'spring_force': spring_force,
        'damping_force': damping_force,
        'drive_force': drive_force,
        'noise_force': noise_force,
        'net_force': net_force,
        'acceleration': acceleration,
        # DERIVATIVES
        'dA': dA,
        'dB': dB,
    }
    trace.append(snapshot)

    # --------------- Euler integration (engine; do not edit) ---------------
    A = A + dt * dA
    B = B + dt * dB

    # 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='')
