# ============================================================
# 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).
# - INPUTS_RND: fresh random values each step (cannot depend on STATE).
# - HELPERS: all computed variables (algebraic and intermediate expressions)
#            in dependency order. Can use wrappers like sin(), exp(), graph(),
#            delay(), smth1() — 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(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]


# ------------------ 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 = None) -> 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 = None) -> 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.2  # time step size
t0 = 1900.0  # start time
tf = 2100.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)
P = 1.65e9  # population (people) # NECESSARY
NR = 900e9  # natural resources (natural resource units) # NECESSARY
CI = 0.4e9  # capital investment (capital units) # NECESSARY
POL = 0.2e9  # pollution (pollution units) # NECESSARY
CIAF = 0.2  # capital-investment-in-agriculture fraction (dimensionless) # 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
# (All constants are inlined in the 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 time-dependent inputs in this model)
    # ==================== END INPUTS_T ======================

    # ================= BEGIN INPUTS_RND =====================
    # RANDOM INPUTS — use gauss(), uniform() wrappers (no STATE usage).
    # (No random inputs in this model)
    # ================== END INPUTS_RND ======================

    # =================== BEGIN HELPERS ======================
    # HELPERS — All computed variables (algebraic and intermediate expressions)
    # in dependency order.

    # Basic ratios and derived quantities
    CIR = CI / P  # capital-investment ratio (capital units/person)
    CR = P / (135e6 * 26.5)  # crowding ratio (dimensionless)
    NRFR = NR / 900e9  # natural-resource fraction remaining (dimensionless)
    POLR = POL / 3.6e9  # pollution ratio (dimensionless)

    # Material standard of living components
    CIAF_current = CIAF  # current capital-investment-in-agriculture fraction
    NREM = graph(
        NRFR, ((0, 0), (0.25, 0.15), (0.5, 0.5), (0.75, 0.85), (1, 1))
    )  # natural-resource-extraction multiplier
    ECIR = CIR * (1 - CIAF_current) * NREM / (1 - 0.3)  # effective-capital-investment ratio
    MSL = ECIR / 1  # material standard of living (dimensionless)

    # Birth rate multipliers
    BRMM = graph(
        MSL, ((0, 1.2), (1, 1), (2, 0.85), (3, 0.75), (4, 0.7), (5, 0.7))
    )  # birth-rate-from-material multiplier
    BRCM = graph(
        CR, ((0, 1.05), (1, 1), (2, 0.9), (3, 0.7), (4, 0.6), (5, 0.55))
    )  # birth-rate-from-crowding multiplier
    BRPM = graph(
        POLR, ((0, 1.02), (10, 0.9), (20, 0.7), (30, 0.4), (40, 0.25), (50, 0.15), (60, 0.1))
    )  # birth-rate-from-pollution multiplier

    # Food ratio components
    CIRA = CIR * CIAF_current / 0.3  # capital-investment ratio in agriculture
    FPCI = graph(
        CIRA, ((0, 0.5), (1, 1), (2, 1.4), (3, 1.7), (4, 1.9), (5, 2.05), (6, 2.2))
    )  # food potential from capital investment
    FCM = graph(CR, ((0, 2.4), (1, 1), (2, 0.6), (3, 0.4), (4, 0.3), (5, 0.2)))  # food-from-crowding multiplier
    FPM = graph(
        POLR, ((0, 1.02), (10, 0.9), (20, 0.65), (30, 0.35), (40, 0.2), (50, 0.1), (60, 0.05))
    )  # food-from-pollution multiplier
    FR = FPCI * FCM * FPM * (1 if t >= 1970 else 1) / 1  # food ratio
    BRFM = graph(FR, ((0, 0), (1, 1), (2, 1.6), (3, 1.9), (4, 2)))  # birth-rate-from-food multiplier

    # Death rate multipliers
    DRMM = graph(
        MSL,
        (
            (0, 3),
            (0.5, 1.8),
            (1, 1),
            (1.5, 0.8),
            (2, 0.7),
            (2.5, 0.6),
            (3, 0.53),
            (3.5, 0.5),
            (4, 0.5),
            (4.5, 0.5),
            (5, 0.5),
        ),
    )  # death-rate-from-material multiplier
    DRCM = graph(CR, ((0, 0.9), (1, 1), (2, 1.2), (3, 1.5), (4, 1.9), (5, 3)))  # death-rate-from-crowding multiplier
    DRPM = graph(
        POLR, ((0, 0.92), (10, 1.3), (20, 2), (30, 3.2), (40, 4.8), (50, 6.8), (60, 9.2))
    )  # death-rate-from-pollution multiplier
    DRFM = graph(
        FR, ((0, 30), (0.25, 3), (0.5, 2), (0.75, 1.4), (1, 1), (1.25, 0.7), (1.5, 0.6), (1.75, 0.5), (2, 0.5))
    )  # death-rate-from-food multiplier

    # Natural resource usage multiplier
    NRMM = graph(
        MSL, ((0, 0), (1, 1), (2, 1.8), (3, 2.4), (4, 2.9), (5, 3.3), (6, 3.6), (7, 3.8), (8, 3.9), (9, 3.95), (10, 4))
    )  # natural-resource-from-material multiplier

    # Capital investment multiplier
    CIM = graph(MSL, ((0, 0.1), (1, 1.0), (2, 1.8), (3, 2.4), (4, 2.8), (5, 3)))  # capital-investment multiplier

    # Pollution components
    POLCM = graph(CIR, ((0, 0.05), (1, 1), (2, 3), (3, 5.4), (4, 7.4), (5, 8)))  # pollution-from-capital multiplier
    POLAT = graph(
        POLR, ((0, 0.6), (10, 2.5), (20, 5), (30, 8), (40, 11.5), (50, 15.5), (60, 20))
    )  # pollution-absorption time

    # Quality of life components
    QLM = graph(MSL, ((0, 0.2), (1, 1), (2, 1.7), (3, 2.3), (4, 2.7), (5, 2.9)))  # quality of life from material
    QLC = graph(
        CR,
        (
            (0, 2),
            (0.5, 1.3),
            (1, 1),
            (1.5, 0.75),
            (2, 0.55),
            (2.5, 0.45),
            (3, 0.38),
            (3.5, 0.3),
            (4, 0.25),
            (4.5, 0.22),
            (5, 0.2),
        ),
    )  # quality of life from crowding
    QLF = graph(FR, ((0, 0), (1, 1), (2, 1.8), (3, 2.4), (4, 2.7)))  # quality of life from food
    QLP = graph(
        POLR, ((0, 1.04), (10, 0.85), (20, 0.6), (30, 0.3), (40, 0.15), (50, 0.05), (60, 0.02))
    )  # quality of life from pollution
    QL = 1 * QLM * QLC * QLF * QLP  # quality of life

    # Capital investment fraction adjustment components
    CFIFR = graph(FR, ((0, 1), (0.5, 0.6), (1, 0.3), (1.5, 0.15), (2, 0.1)))  # capital fraction indicated by food ratio
    CIQR = graph(QLM / QLF, ((0, 0.7), (0.5, 0.8), (1, 1), (1.5, 1.5), (2, 2)))  # capital-investment-from-quality ratio

    # ==================== END HELPERS =======================

    # ================= BEGIN DERIVATIVES ====================
    # DERIVATIVES for each STATE variable.

    # Birth and death rates
    BR = P * (0.04 if t >= 1970 else 0.04) * BRFM * BRMM * BRCM * BRPM  # birth rate
    DR = P * (0.028 if t >= 1970 else 0.028) * DRMM * DRPM * DRFM * DRCM  # death rate

    # Natural resource usage rate
    NRUR = P * (1 if t >= 1970 else 1) * NRMM  # natural-resource-usage rate

    # Capital investment flows
    CIG = P * CIM * (0.05 if t >= 1970 else 0.05)  # capital-investment generation
    CID = CI * (0.025 if t >= 1970 else 0.025)  # capital-investment discard

    # Pollution flows
    POLG = P * (1 if t >= 1970 else 1) * POLCM  # pollution generation
    POLA = POL / POLAT  # pollution absorption

    # State derivatives
    dP = BR - DR
    dNR = -NRUR
    dCI = CIG - CID
    dPOL = POLG - POLA
    dCIAF = (1 / 15) * (CFIFR * CIQR - CIAF)
    # ================== END DERIVATIVES =====================

    # ---- SNAPSHOT (AFTER DERIVATIVES, BEFORE INTEGRATION) ----
    snapshot = {
        't': t,
        # STATE
        'P': P,
        'NR': NR,
        'CI': CI,
        'POL': POL,
        'CIAF': CIAF,
        # HELPERS (includes all computed variables)
        'CIR': CIR,
        'CR': CR,
        'NRFR': NRFR,
        'POLR': POLR,
        'NREM': NREM,
        'ECIR': ECIR,
        'MSL': MSL,
        'BRMM': BRMM,
        'BRCM': BRCM,
        'BRPM': BRPM,
        'CIRA': CIRA,
        'FPCI': FPCI,
        'FCM': FCM,
        'FPM': FPM,
        'FR': FR,
        'BRFM': BRFM,
        'DRMM': DRMM,
        'DRCM': DRCM,
        'DRPM': DRPM,
        'DRFM': DRFM,
        'NRMM': NRMM,
        'CIM': CIM,
        'POLCM': POLCM,
        'POLAT': POLAT,
        'QLM': QLM,
        'QLC': QLC,
        'QLF': QLF,
        'QLP': QLP,
        'QL': QL,
        'CFIFR': CFIFR,
        'CIQR': CIQR,
        'BR': BR,
        'DR': DR,
        'NRUR': NRUR,
        'CIG': CIG,
        'CID': CID,
        'POLG': POLG,
        'POLA': POLA,
        # DERIVATIVES
        'dP': dP,
        'dNR': dNR,
        'dCI': dCI,
        'dPOL': dPOL,
        'dCIAF': dCIAF,
    }
    trace.append(snapshot)

    # --------------- Euler integration (engine; do not edit) ---------------
    P = P + dt * dP
    NR = NR + dt * dNR
    CI = CI + dt * dCI
    POL = POL + dt * dPOL
    CIAF = CIAF + dt * dCIAF

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