import numpy as np
from parameters import parameters
from casadi import casadi

# Create an instance of the 'parameters' class
param_obj = parameters()

class plantODE:
    def __init__(self):
        pass

    def ab2rh(self, absh, T):
        # Constants
        R_W = 461.52  # gas constant water vapor J kg^-1 K^-1
        
        # Magnus-equation saturation vapor pressure
        e_s = 100 * 6.11 * np.exp((17.62 * T) / (243.12 + T))  # e_s in Pa
        
        # Calculate actual vapor pressure
        e = absh *10**-3 * (R_W * (273 + T))
        
        # Calculate relative humidity (RH)
        RH = (e / e_s) * 100
        
        return RH

    def rh2ab(self, RH, T):
        # Function to calculate absolute humidity from Relative humidity and temperature. Output in g/m3

        T = T + 273.15  # Convert temperature to Kelvin [K]

        a1 = -7.85951783
        a2 = 1.84408259
        a3 = -11.7866497
        a4 = 22.6807411
        a5 = -15.9618719
        a6 = 1.80122502

        Pc = 22.064  # Critical pressure of water [MPa]
        Tc = 647.096  # Critical temp of water [K]
        Rw = 461.5  # Specific gas constant for water vapor [J kg-1 K-1]

        tau = 1 - T / Tc

        Ps = Pc * np.exp(Tc / T * (a1 * tau + a2 * (tau ** 1.5) + a3 * (tau ** 3) + a4 * (tau ** 3.5) + a5 * (tau ** 4) + a6 * (tau ** 7.5)))

        ab = (RH * Ps * 10 ** 6) / (Rw * T * 100)  # Output in g/m3

        ab = ab * 1000
        return ab

    def dRH_dt(self, absh, T, dabsh_dt, dT_dt):
        # Constants
        R_W = 461.52  # J kg^-1 K^-1

        # Saturation vapor pressure (Pa)
        e_s = 100 * 6.11 * np.exp((17.62 * T) / (243.12 + T))
        # Derivative of e_s with respect to T
        des_dT = e_s * (17.62 * 243.12) / (243.12 + T)**2

        # Actual vapor pressure (Pa)
        e = absh * 1e-3 * (R_W * (273 + T))
        # Partial derivatives for chain rule
        de_dabsh = 1e-3 * (R_W * (273 + T))
        de_dT = absh * 1e-3 * R_W

        # Now, the chain rule for d(RH)/dt
        numerator = (de_dabsh * dabsh_dt + de_dT * dT_dt) * e_s - e * des_dT * dT_dt
        denominator = e_s ** 2

        dRH_dt = 100 * numerator / denominator
        return dRH_dt
    
    def ppm2kgpmc(self, mol_mass, ppm, T):
        # Conversion of gas concentration from parts per million to Kilogram per m3
        # The mass of a given gas is converted to the PPM value using the
        # given molecular mass of the given gas at the specified temperature

        # The molecular mass is simply the sum of the atomic masses
        # e.g. for CO2 (12.01 + 16.00 + 16.00 = 44.01 grams)
        # for O2 ( 15.9994 + 15.9994 = 31.99880 grams)

        T_K = T + 273.15  # °C to Kelvin
        V = ppm * 10 ** -6  # volume in m3
        P = 1 / (9.8692 * 10 ** -6)  # 1 Pa = 9.8692*10^-6 atm
        R = 8.31441  # universal gas constant
        n = (P * V) / (R * T_K)  # molecular number
        mass = n * mol_mass
        return mass
    
    def ppm2gpmc(self, mol_mass, ppm, T):
        # Conversion of gas concentration from parts per million to Kilogram per m3
        # The mass of a given gas is converted to the PPM value using the
        # given molecular mass of the given gas at the specified temperature

        # The molecular mass is simply the sum of the atomic masses
        # e.g. for CO2 (12.01 + 16.00 + 16.00 = 44.01 grams)
        # for O2 ( 15.9994 + 15.9994 = 31.99880 grams)

        T_K = T + 273.15  # °C to Kelvin
        V = ppm * 10 ** -6  # volume in m3
        P = 1 / (9.8692 * 10 ** -6)  # 1 Pa = 9.8692*10^-6 atm
        R = 8.31441  # universal gas constant
        n = (P * V) / (R * T_K)  # molecular number
        mass = n * mol_mass
        mass = mass * 10**3
        return mass    
    
    def kgpmc2ppm(self, mol_mass, mass, T):
    # Conversion of gas concentration from Kilogram per m3 to parts per million
    # The mass of a given gas is converted to the PPM value using the
    # given molecular mass of the given gas at the specified temperature

        T_K = T + 273.15  # °C to Kelvin
        P = 1 / (9.8692 * 10**-6)  # 1 Pa = 9.8692*10^-6 atm
        R = 8.31441  # universal gas constant
        n = mass / mol_mass  # molecular number
        V = (n * R * T_K) / P  # volume of the gas
        ppm = V / 10**-6
        return ppm

    def gpmc2ppm(self, mol_mass, mass, T):
    # Conversion of gas concentration from Kilogram per m3 to parts per million
    # The mass of a given gas is converted to the PPM value using the
    # given molecular mass of the given gas at the specified temperature

        T_K = T + 273.15  # °C to Kelvin
        P = 1 / (9.8692 * 10**-6)  # 1 Pa = 9.8692*10^-6 atm
        R = 8.31441  # universal gas constant
        n = mass / (mol_mass * 10**3) # molecular number
        V = (n * R * T_K) / P  # volume of the gas
        ppm = V / 10**-6
        return ppm
    
    def penalty(self, x, xMin, xMax,c):
        # soft constraint cost fucntion
        # beta = 1*10^-4
        beta = 10**-4
        P = c/2 * (np.sqrt((xMin - x)**2 + beta) + np.sqrt((xMax-x)**2 + beta) - (xMax - xMin)) 
        return P
    
    def dynamicsODE(self, X, U, D):
        INDEX, p = param_obj.get_all_parameters()  # Read in all parameters for the model from parameters.m

        # Input vector mapping
        X_T = X[INDEX['X_T']]
        X_CO2 = X[INDEX['X_CO2']]
        X_H = X[INDEX['X_H']]
        X_H = self.rh2ab(X_H,X_T)
        X_B = X[INDEX['X_B']]

        # input variables
        U_vent = U[INDEX['U_Vent']]
        U_CO2 = U[INDEX['U_CO2']]
        #U_temp = U[INDEX['U_temp']]
        U_heat = U[INDEX['U_heat']]
        U_cool = U[INDEX['U_cool']]

        # disturbance variables
        D_T_out = D[INDEX['D_T_out']]
        D_C_out = self.ppm2gpmc(p['M_c'], D[INDEX['D_C_out']], D[INDEX['D_T_out']])
        D_RH_out = D[INDEX['D_RH_out']]
        D_R_o = casadi.if_else(D[INDEX['D_R_o']] >= 0, D[INDEX['D_R_o']], 0)

        # Conversion factor for CO2 from µmol/mol to kg/m^3
        V_mol = (1 * p['R'] * (p['T0']+X_T)) / 101325
        f_ppm = p['MCO2'] / V_mol
    
        # CO2 concentration in ppm
        C_co2_ppm = (X_CO2 * 10**-3) / f_ppm

        # Calculation of time-varying parameters and constants - Temperature of air related terms
        K_C_gh = 8115.87  # Total heat capacity of the GH per unit area [J kg-1 K-1]
        cl = D_R_o < 3
        tau_tot = p['tau_cov'] * (1 - ((1 - p['tau_scr']) * cl * p['Cl_scr']) / 100) * (1 - ((1 - p['tau_scr2']) * cl * p['Cl_scr2']) / 100)
        phi_QT_sun = tau_tot * D_R_o
        k_u_vent = (50 * p['V_gh']) / 3600 / p['A_floor']

        phi_QT_vent = p['c_air'] * p['rho_air'] * k_u_vent * (1 - cl) * U_vent * (D_T_out - X_T)
        phi_QT_wall = p['u_wall'] * (D_T_out - X_T)
        phi_QT_roof = p['u_roof'] * (D_T_out - X_T)
        phi_QT_cov = (p['A_cov'] / p['A_floor']) * (phi_QT_wall + phi_QT_roof)
        phi_QT_heat = (U_heat * p['P_heat']) / p['A_floor']
        phi_QT_cool = (U_cool * p['P_cool']) / p['A_floor']
        #phi_QT_input = (U_temp * 10**3)/ p['A_floor']

        # Calculation of time-varying parameters and constants - Temperature of leaf related terms
        r_b = 200  # Boundary layer resistance - Stanghelline, Jong - A model of humidity and its application in a greenhouse
        phi_R_i = 0.86 * (1 - np.exp(-0.7 * p['LAI'])) * phi_QT_sun
        #r_s = (82 + 570 * np.exp(-p['gamma'] * (phi_R_i / p['LAI']))) * (1 + 0.0026 * (X_T - p['T_sp']) ** 2)
        r_st = 82 * ((phi_R_i / 2 * p['LAI']) + 4.30) / ((phi_R_i / 2 * p['LAI']) + 0.54) * (1 + 0.023 * (X_T - p['T_sp']) ** 2)
        epsilon = 0.7584 * np.exp(0.0518 * X_T)
        g_e = 2 * p['LAI'] / ((1 + epsilon) * r_b + r_st)
        H_i_sat = 5.5638 * np.exp(0.0572 * X_T)
        H_v = H_i_sat + epsilon * (r_b / 2 * p['LAI']) * (phi_R_i / p['L'])
        phi_QT_trans = g_e * p['L'] * (H_v - X_H)

        # Calculation of time-varying parameters and constants - CO2 related terms
        phi_QCO2_vent = k_u_vent * U_vent * (D_C_out - X_CO2)
        phi_QCO2_ass = 2.2 * 10 ** -3 * (1 / (1 + (0.42 / X_CO2))) * (1 - np.exp(-0.003 * phi_QT_sun))
        #vent10 = U_vent > 1
        phi_QCO2_inj = ((p['u_CO2_max'] * U_CO2) / p['A_floor'])/ (1 + np.exp(-2.5 * (D_R_o -3)))
        phi_QCO2_inj = phi_QCO2_inj/(1 + np.exp(-2.5 * (U_vent -0.1)))

        # Calculation of time-varying parameters and constants - Humidity/Water vapor related terms
        T_cov = (2 * D_T_out + X_T) / 3
        eps = 10**-4
        g_cb = ((p['A_cov'] / p['A_floor']) * 1.64 * 10 ** -3 * np.sign(X_T - T_cov) * casadi.fabs(X_T - T_cov) ** (1 / 3))
        # g_c = casadi.if_else(g_cb > 0, g_cb, 0)
        # Calculate g_c
        g_c = (0 + g_cb + np.sqrt((0 - g_cb)**2 + eps)) / 2

        phi_H_cov = (g_c) * (0.2522 * np.exp(0.0485 * X_T) * (X_T - D_T_out) - (H_i_sat - X_H))
        
        H_out = self.rh2ab(D_RH_out, D_T_out)
        phi_H_vent = k_u_vent * U_vent * (X_H - H_out)
        r_st = 82 * ((phi_R_i / 2 * p['LAI']) + 4.30) / ((phi_R_i / 2 * p['LAI']) + 0.54) * (1 + 0.023 * (X_T - p['T_sp']) ** 2)
        phi_H_trans = (2 * p['LAI']) / ((1 + epsilon) * r_b + r_st) * ((H_i_sat - X_H) + (epsilon * r_b) / (2 * p['LAI']) * phi_R_i / p['L'])
        
        # Change in vapour due to cooling

        rho_air = 1.29 * (p['T0']/(p['T0']+X_T))
        alpha_uc = 1.28 * (casadi.fabs(p['T_uc'] - (p['T0']+X_T))/p['d_uc'])**(1/4)
        k_h20_cool = alpha_uc/(rho_air * p['c_air'] * (p['Le'])**(2/3)) #mass transfer coeff of vapor from air to cooling pipe [m s-1]

        p_sat_uc = p['cs1'] * np.exp((p['cs2']*(p['T_uc']-p['T0']))/(p['cs3']+(p['T_uc']-p['T0']))) # saturation vapor pressure at T_uc - [N m-2]
        X_sat_uc = (p_sat_uc * p['M_h2o'])/(p['R'] * p['T_uc']) # [kg[h2o] m-3]
        X_sat_uc = X_sat_uc * 10**3 # [g[h2o] m-3]
        phi_H_uc = p['A_uc'] * k_h20_cool * (X_H - X_sat_uc) # [g[h2o] s-1]
        phi_H_cool = (0 + phi_H_uc + np.sqrt((0-phi_H_uc)**2 + eps))/2 # [g[h2o] s-1]
        phi_H_cool = (U_cool * phi_H_cool)/p['A_floor'] # [g[h2o] m-2 s-1]

        # Change in vapor due to direct heating
        phi_H_heat = p['neta_heat'] * phi_QT_heat # [g m-2 s-1]

        # Humidity related terms for photosynthesis
        f_I = ((phi_R_i / 2 * p['LAI']) + 4.30) / ((phi_R_i / 2 * p['LAI']) + 0.54)
        f_T = 1 + 2.2593 * 10**-2 * (X_T - p['T_sp'])**2
        f_CO2 = 1 + 6.08 * 10**-7 * (C_co2_ppm - 200)**2

        R_h2o_s = p['R_min'] * f_I * f_T * f_CO2
        R_h2o_b = (p['Le'])**(2/3) * r_b

        k_h20_ca = 1 / (R_h2o_b + ((p['R_cut'] * R_h2o_s) / (p['R_cut'] + R_h2o_s)))

        # Photosynthesis - Optimal control book
        psi = ((1 - p['Fp']) / 2) * p['zeta']
        Kc = p['Kc_25'] * np.exp(p['Ec'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25'])))
        Ko = p['Ko_25'] * np.exp(p['Eo'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25'])))
        Gamma = (Kc / (2 * Ko)) * p['p_o2i'] * p['f_oc']
        maxCG = (C_co2_ppm + Gamma + np.sqrt((C_co2_ppm-Gamma)**2 + eps))/2
        epsilonP = psi * (p['MCO2'] / 4) * ((maxCG - Gamma) / (maxCG + 2 * Gamma))

        Jmax_25 = 467 * p['rho_chl']
        Jmax = Jmax_25 * np.exp(p['Ej'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25']))) * ((1 + np.exp((p['S'] * p['T25'] - p['H']) / (p['R'] * p['T25']))) /
                        (1 + np.exp((p['S'] * (X_T + p['T0']) - p['H']) / (p['R'] * (X_T + p['T0'])))))
        # P_mm - Max. endogenous photosynthetic capacity 
        P_mm = (p['MCO2'] * Jmax)/4 # kg[CO2] m-1[leaf] s-1
        R_co2_s = 1.6*R_h2o_s # s m-1 - stomatal layer resistance to CO2 diffusion
        R_co2_b = 1.37*R_h2o_b # s m-1 - boundary layer resistance to CO2 diffusion

        # Km - effective Michaelis-Menten constant for carboxylation
        Km = Kc * (1 + (p['p_o2i']/Ko)) # µbar

        V_c_max25 = p['rho_chl'] * p['k_C'] * p['Et']
        V_c_max = V_c_max25 * np.exp(p['Evc'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25'])))
        
        rho_co2_t = p['rho_co2'] * p['T0'] / (X_T + p['T0'])
        R_co2_c = (Km / V_c_max) * (rho_co2_t / p['MCO2'])
        R_co2_tot = R_co2_s + R_co2_b + R_co2_c

        # P_nc - CO2 limited rate of net photosynthesis - kg[CO2] m-2 [leaf] s-1
        P_nc = (f_ppm / R_co2_tot) * (maxCG - Gamma)

        # P_n_max - max. net assimilation rate
        P_n_max = (P_mm + P_nc - np.sqrt((P_mm + P_nc) ** 2 - 4 * p['theta'] * P_mm * P_nc)) / (2 * p['theta']) 
        
        r_DL_mol = p['r_25DL_mol'] * np.exp(p['Ed'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25']))) # µmol[CO2] m-2[leaf] s-1
        r_DL = p['MCO2'] * r_DL_mol

        ## P_g_max - Maximum gross assimilation rate - kg[CO2] m-2[leaf] s-1
        P_g_max = P_n_max + r_DL
        P_gL = P_g_max * (1 - np.exp(-(epsilonP * phi_R_i) / P_g_max)) #  kg[CO2] m-2[gh] s-1

        #Assimilation + respiration
        # Gross canopy assimilation        
        P_g = P_gL * p['LAI'] #  kg[CO2] m-2[gh] s-1
        r_D = r_DL * p['LAI'] #  kg[CO2] m-2[gh] s-1
        P_cg = p['A_s'] * P_g #  kg[CO2] s-1
        r_c = p['A_s'] * r_D #  kg[CO2] s-1
        
        #Assimilation + respiration
        # Gross canopy assimilation
        Phi_CO2_ac = P_cg - r_c # kg[CO2] s-1
        f_CO2_B = (1 / (1 - p['p_w'] / 100)) * (p['c_f'] * p['c_cs']) / p['ASRQ'] # kg[fW] kg-1[CO2]

        # Model dynamics
        dT_dt = (phi_QT_vent + phi_QT_sun + phi_QT_cov - phi_QT_trans + phi_QT_heat - phi_QT_cool) / K_C_gh #
        dC_dt = (p['A_floor'] * (phi_QCO2_inj + phi_QCO2_vent) - (Phi_CO2_ac*10**3)) / p['V_gh']
        dH_dt = p['A_floor'] * (phi_H_trans - phi_H_vent - phi_H_cov - phi_H_cool ) / p['V_gh'] #+ phi_H_heat
        # dH_dt = self.ab2rh(dH_dt,dT_dt)
        dH_dt = self.dRH_dt(X_H, X_T, dH_dt,dT_dt)
        dB_dt = (f_CO2_B * Phi_CO2_ac)/p['A_s']

        X_dot = np.array([dT_dt, dC_dt, dH_dt, dB_dt])
        return X_dot
    
    def rk(self, ode, h, x, u, d):
        k1 = ode(x,u,d)
        k2 = ode(x+h/2*k1,u,d)
        k3 = ode(x+h/2*k2,u,d)
        k4 = ode(x+h*k3,u,d)
        xf = x + h/6 * (k1 + 2*k2 + 2*k3 + k4)
        return xf
    
    def condense(self, X, U, D):
        INDEX, p = param_obj.get_all_parameters()  # Read in all parameters for the model from parameters.m

        # Input vector mapping
        X_T = X[INDEX['X_T']]
        X_CO2 = X[INDEX['X_CO2']]
        X_H = X[INDEX['X_H']]
        X_H = self.rh2ab(X_H,X_T)
        X_B = X[INDEX['X_B']]

        # input variables
        U_vent = U[INDEX['U_Vent']]
        U_CO2 = U[INDEX['U_CO2']]
        #U_temp = U[INDEX['U_temp']]
        U_heat = U[INDEX['U_heat']]
        U_cool = U[INDEX['U_cool']]

        # disturbance variables
        D_T_out = D[INDEX['D_T_out']]
        D_C_out = self.ppm2gpmc(p['M_c'], D[INDEX['D_C_out']], D[INDEX['D_T_out']])
        D_RH_out = D[INDEX['D_RH_out']]
        D_R_o = casadi.if_else(D[INDEX['D_R_o']] >= 0, D[INDEX['D_R_o']], 0)

        H_i_sat = 5.5638 * np.exp(0.0572 * X_T)
        # Calculation of time-varying parameters and constants - Humidity/Water vapor related terms
        T_cov = (2 * D_T_out + X_T) / 3
        eps = 10**-4
        g_cb = ((p['A_cov'] / p['A_floor']) * 1.64 * 10 ** -3 * np.sign(X_T - T_cov) * casadi.fabs(X_T - T_cov) ** (1 / 3))
        # g_c = casadi.if_else(g_cb > 0, g_cb, 0)
        # Calculate g_c
        g_c = (0 + g_cb + np.sqrt((0 - g_cb)**2 + eps)) / 2

        phi_H_cov = (g_c) * (0.2522 * np.exp(0.0485 * X_T) * (X_T - D_T_out) - (H_i_sat - X_H))

        return phi_H_cov
    
    # Define the outliers using percentiles [1, 99]
    def remove_outliers(self, data, lower_percentile=1, upper_percentile=99):
        lower_bound = np.percentile(data, lower_percentile)
        upper_bound = np.percentile(data, upper_percentile)
        outliers = (data < lower_bound) | (data > upper_bound)
        data[outliers] = np.nan
        return data
    
    def flux(self, X, U, D):
        INDEX, p = param_obj.get_all_parameters()  # Read in all parameters for the model from parameters.m

        # Input vector mapping
        X_T = X[INDEX['X_T']]
        X_CO2 = X[INDEX['X_CO2']]
        X_H = X[INDEX['X_H']]
        X_H = self.rh2ab(X_H,X_T)
        X_B = X[INDEX['X_B']]

        # input variables
        U_vent = U[INDEX['U_Vent']]
        U_CO2 = U[INDEX['U_CO2']]
        #U_temp = U[INDEX['U_temp']]
        U_heat = U[INDEX['U_heat']]
        U_cool = U[INDEX['U_cool']]

        # disturbance variables
        D_T_out = D[INDEX['D_T_out']]
        D_C_out = self.ppm2gpmc(p['M_c'], D[INDEX['D_C_out']], D[INDEX['D_T_out']])
        D_RH_out = D[INDEX['D_RH_out']]
        D_R_o = casadi.if_else(D[INDEX['D_R_o']] >= 0, D[INDEX['D_R_o']], 0)

        # Conversion factor for CO2 from µmol/mol to kg/m^3
        V_mol = (1 * p['R'] * (p['T0']+X_T)) / 101325
        f_ppm = p['MCO2'] / V_mol
    
        # CO2 concentration in ppm
        C_co2_ppm = (X_CO2 * 10**-3) / f_ppm

        # Calculation of time-varying parameters and constants - Temperature of air related terms
        K_C_gh = 8115.87  # Total heat capacity of the GH per unit area [J kg-1 K-1]
        cl = D_R_o < 3
        tau_tot = p['tau_cov'] * (1 - ((1 - p['tau_scr']) * cl * p['Cl_scr']) / 100) * (1 - ((1 - p['tau_scr2']) * cl * p['Cl_scr2']) / 100)
        phi_QT_sun = tau_tot * D_R_o
        k_u_vent = (50 * p['V_gh']) / 3600 / p['A_floor']

        phi_QT_vent = p['c_air'] * p['rho_air'] * k_u_vent * (1 - cl) * U_vent * (D_T_out - X_T)
        phi_QT_wall = p['u_wall'] * (D_T_out - X_T)
        phi_QT_roof = p['u_roof'] * (D_T_out - X_T)
        phi_QT_cov = (p['A_cov'] / p['A_floor']) * (phi_QT_wall + phi_QT_roof)
        phi_QT_heat = (U_heat * p['P_heat']) / p['A_floor']
        phi_QT_cool = (U_cool * p['P_cool']) / p['A_floor']
        #phi_QT_input = (U_temp * 10**3)/ p['A_floor']

        # Calculation of time-varying parameters and constants - Temperature of leaf related terms
        r_b = 200  # Boundary layer resistance - Stanghelline, Jong - A model of humidity and its application in a greenhouse
        phi_R_i = 0.86 * (1 - np.exp(-0.7 * p['LAI'])) * phi_QT_sun
        #r_s = (82 + 570 * np.exp(-p['gamma'] * (phi_R_i / p['LAI']))) * (1 + 0.0026 * (X_T - p['T_sp']) ** 2)
        r_st = 82 * ((phi_R_i / 2 * p['LAI']) + 4.30) / ((phi_R_i / 2 * p['LAI']) + 0.54) * (1 + 0.023 * (X_T - p['T_sp']) ** 2)
        epsilon = 0.7584 * np.exp(0.0518 * X_T)
        g_e = 2 * p['LAI'] / ((1 + epsilon) * r_b + r_st)
        H_i_sat = 5.5638 * np.exp(0.0572 * X_T)
        H_v = H_i_sat + epsilon * (r_b / 2 * p['LAI']) * (phi_R_i / p['L'])
        phi_QT_trans = g_e * p['L'] * (H_v - X_H)

        # Calculation of time-varying parameters and constants - CO2 related terms
        phi_QCO2_vent = k_u_vent * U_vent * (D_C_out - X_CO2)
        phi_QCO2_ass = 2.2 * 10 ** -3 * (1 / (1 + (0.42 / X_CO2))) * (1 - np.exp(-0.003 * phi_QT_sun))
        #vent10 = U_vent > 1
        phi_QCO2_inj = ((p['u_CO2_max'] * U_CO2) / p['A_floor'])/ (1 + np.exp(-2.5 * (D_R_o -3)))
        phi_QCO2_inj = phi_QCO2_inj/(1 + np.exp(-2.5 * (U_vent -0.1)))

        # Calculation of time-varying parameters and constants - Humidity/Water vapor related terms
        T_cov = (2 * D_T_out + X_T) / 3
        eps = 10**-4
        g_cb = ((p['A_cov'] / p['A_floor']) * 1.64 * 10 ** -3 * np.sign(X_T - T_cov) * casadi.fabs(X_T - T_cov) ** (1 / 3))
        # g_c = casadi.if_else(g_cb > 0, g_cb, 0)
        # Calculate g_c
        g_c = (0 + g_cb + np.sqrt((0 - g_cb)**2 + eps)) / 2

        phi_H_cov = (g_c) * (0.2522 * np.exp(0.0485 * X_T) * (X_T - D_T_out) - (H_i_sat - X_H))
        
        H_out = self.rh2ab(D_RH_out, D_T_out)
        phi_H_vent = k_u_vent * U_vent * (X_H - H_out)
        r_st = 82 * ((phi_R_i / 2 * p['LAI']) + 4.30) / ((phi_R_i / 2 * p['LAI']) + 0.54) * (1 + 0.023 * (X_T - p['T_sp']) ** 2)
        phi_H_trans = (2 * p['LAI']) / ((1 + epsilon) * r_b + r_st) * ((H_i_sat - X_H) + (epsilon * r_b) / (2 * p['LAI']) * phi_R_i / p['L'])
        
        # Change in vapour due to cooling

        rho_air = 1.29 * (p['T0']/(p['T0']+X_T))
        alpha_uc = 1.28 * (casadi.fabs(p['T_uc'] - (p['T0']+X_T))/p['d_uc'])**(1/4)
        k_h20_cool = alpha_uc/(rho_air * p['c_air'] * (p['Le'])**(2/3)) #mass transfer coeff of vapor from air to cooling pipe [m s-1]

        p_sat_uc = p['cs1'] * np.exp((p['cs2']*(p['T_uc']-p['T0']))/(p['cs3']+(p['T_uc']-p['T0']))) # saturation vapor pressure at T_uc - [N m-2]
        X_sat_uc = (p_sat_uc * p['M_h2o'])/(p['R'] * p['T_uc']) # [kg[h2o] m-3]
        X_sat_uc = X_sat_uc * 10**3 # [g[h2o] m-3]
        phi_H_uc = p['A_uc'] * k_h20_cool * (X_H - X_sat_uc) # [g[h2o] s-1]
        phi_H_cool = (0 + phi_H_uc + np.sqrt((0-phi_H_uc)**2 + eps))/2 # [g[h2o] s-1]
        phi_H_cool = (U_cool * phi_H_cool)/p['A_floor'] # [g[h2o] m-2 s-1]

        # Change in vapor due to direct heating
        phi_H_heat = p['neta_heat'] * phi_QT_heat # [g m-2 s-1]

        # Humidity related terms for photosynthesis
        f_I = ((phi_R_i / 2 * p['LAI']) + 4.30) / ((phi_R_i / 2 * p['LAI']) + 0.54)
        f_T = 1 + 2.2593 * 10**-2 * (X_T - p['T_sp'])**2
        f_CO2 = 1 + 6.08 * 10**-7 * (C_co2_ppm - 200)**2

        R_h2o_s = p['R_min'] * f_I * f_T * f_CO2
        R_h2o_b = (p['Le'])**(2/3) * r_b

        k_h20_ca = 1 / (R_h2o_b + ((p['R_cut'] * R_h2o_s) / (p['R_cut'] + R_h2o_s)))

        # Photosynthesis - Optimal control book
        psi = ((1 - p['Fp']) / 2) * p['zeta']
        Kc = p['Kc_25'] * np.exp(p['Ec'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25'])))
        Ko = p['Ko_25'] * np.exp(p['Eo'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25'])))
        Gamma = (Kc / (2 * Ko)) * p['p_o2i'] * p['f_oc']
        maxCG = (C_co2_ppm + Gamma + np.sqrt((C_co2_ppm-Gamma)**2 + eps))/2
        epsilonP = psi * (p['MCO2'] / 4) * ((maxCG - Gamma) / (maxCG + 2 * Gamma))

        Jmax_25 = 467 * p['rho_chl']
        Jmax = Jmax_25 * np.exp(p['Ej'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25']))) * ((1 + np.exp((p['S'] * p['T25'] - p['H']) / (p['R'] * p['T25']))) /
                        (1 + np.exp((p['S'] * (X_T + p['T0']) - p['H']) / (p['R'] * (X_T + p['T0'])))))
        # P_mm - Max. endogenous photosynthetic capacity 
        P_mm = (p['MCO2'] * Jmax)/4 # kg[CO2] m-1[leaf] s-1
        R_co2_s = 1.6*R_h2o_s # s m-1 - stomatal layer resistance to CO2 diffusion
        R_co2_b = 1.37*R_h2o_b # s m-1 - boundary layer resistance to CO2 diffusion

        # Km - effective Michaelis-Menten constant for carboxylation
        Km = Kc * (1 + (p['p_o2i']/Ko)) # µbar

        V_c_max25 = p['rho_chl'] * p['k_C'] * p['Et']
        V_c_max = V_c_max25 * np.exp(p['Evc'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25'])))
        
        rho_co2_t = p['rho_co2'] * p['T0'] / (X_T + p['T0'])
        R_co2_c = (Km / V_c_max) * (rho_co2_t / p['MCO2'])
        R_co2_tot = R_co2_s + R_co2_b + R_co2_c

        # P_nc - CO2 limited rate of net photosynthesis - kg[CO2] m-2 [leaf] s-1
        P_nc = (f_ppm / R_co2_tot) * (maxCG - Gamma)

        # P_n_max - max. net assimilation rate
        P_n_max = (P_mm + P_nc - np.sqrt((P_mm + P_nc) ** 2 - 4 * p['theta'] * P_mm * P_nc)) / (2 * p['theta']) 
        
        r_DL_mol = p['r_25DL_mol'] * np.exp(p['Ed'] * ((X_T + p['T0'] - p['T25']) / ((X_T + p['T0']) * p['R'] * p['T25']))) # µmol[CO2] m-2[leaf] s-1
        r_DL = p['MCO2'] * r_DL_mol

        ## P_g_max - Maximum gross assimilation rate - kg[CO2] m-2[leaf] s-1
        P_g_max = P_n_max + r_DL
        P_gL = P_g_max * (1 - np.exp(-(epsilonP * phi_R_i) / P_g_max)) #  kg[CO2] m-2[gh] s-1

        #Assimilation + respiration
        # Gross canopy assimilation        
        P_g = P_gL * p['LAI'] #  kg[CO2] m-2[gh] s-1
        r_D = r_DL * p['LAI'] #  kg[CO2] m-2[gh] s-1
        P_cg = p['A_s'] * P_g #  kg[CO2] s-1
        r_c = p['A_s'] * r_D #  kg[CO2] s-1
        
        #Assimilation + respiration
        # Gross canopy assimilation
        Phi_CO2_ac = P_cg - r_c # kg[CO2] s-1
        f_CO2_B = (1 / (1 - p['p_w'] / 100)) * (p['c_f'] * p['c_cs']) / p['ASRQ'] # kg[fW] kg-1[CO2]

        # Model dynamics
        dT_dt = (phi_QT_vent + phi_QT_sun + phi_QT_cov - phi_QT_trans + phi_QT_heat - phi_QT_cool) / K_C_gh #
        dC_dt = (p['A_floor'] * (phi_QCO2_inj + phi_QCO2_vent) - (Phi_CO2_ac*10**3)) / p['V_gh']
        dH_dt = p['A_floor'] * (phi_H_trans - phi_H_vent - phi_H_cov - phi_H_cool ) / p['V_gh'] #+ phi_H_heat
        #dH_dt = self.ab2rh(dH_dt,dT_dt)
        dH_dt = self.dRH_dt(X_H, X_T, dH_dt,dT_dt)
        dB_dt = (f_CO2_B * Phi_CO2_ac)/p['A_s']

        flux = np.array([float(phi_QT_vent), float(phi_QT_sun), phi_QT_cov, -float(phi_QT_trans),phi_QT_heat, -phi_QT_cool, 
                        float(phi_QCO2_inj), phi_QCO2_vent, -float(Phi_CO2_ac*10**3),
                        float(phi_H_trans), -phi_H_vent, -phi_H_cov, -phi_H_cool])
        return flux
    


    


