import numpy as np
import matplotlib.pyplot as plt
import matplotlib as mpl
from casadi import *
from casadi.tools import *
from plantODE_cpl import plantODE
from parameters import parameters
import time 

import scipy as scipy
from scipy import interpolate

import plotly.graph_objects as go
from plotly.subplots import make_subplots

import pandas as pd
import datetime
from datetime import datetime, timedelta

# Empty list to store the execution times of all days involved
complete_execution_time = []
infeasible_date = []
# Create an instance of the 'parameters' class
param_obj = parameters()
plant_obj = plantODE()

# Import the GH and environment(disturbance) data
# GH_data = scipy.io.loadmat('GH_24082011.mat')
GH_data = pd.read_csv(filepath_or_buffer='./modified_weather_2011.csv', sep=',')  # Update path as needed

#To extract 2011 weather file
# to populate the rows with date value
# numrows = GH_data.index.size
# i=0
# index=0
# while i <= numrows-1 and index <= numrows-1: 
#     index = (i+288)-1
#     GH_data.loc[i:index-1,'Datum'] = GH_data.loc[index,'Datum']
#     i=index+1

# Convert the 'Uhrzeit' column in GH_data DataFrame to datetime objects
GH_data['Time'] = pd.to_datetime(GH_data['Time'], format='%H:%M')

# Extract only the time part (hour and minute) from the 'Uhrzeit' column
GH_data['Time'] = GH_data['Time'].dt.strftime('%H:%M')

GH_data_selected = GH_data[['Date','Time','OutsideCO2level','OutdoorHumidity','GlobalRadiation','OutdoorTemp']]

# To extract 2012 weather file
# Define the expected timestamp formats
# expected_format_1 = '%m-%d-%y %H:%M'
# expected_format_2 = '%m/%d/%Y %H:%M'

# # Create empty 'Date' and 'Time' columns
# GH_data['Date'] = ""
# GH_data['Time'] = ""

# # Iterate through the timestamp column and extract 'Date' and 'Time'
# for index, row in GH_data.iterrows():
#     timestamp = row['Timestamp']
#     try:
#         # Try to convert to expected_format_1
#         dt = pd.to_datetime(timestamp, format=expected_format_1)
#     except ValueError:
#         # If it doesn't match expected_format_1, try expected_format_2
#         dt = pd.to_datetime(timestamp, format=expected_format_2)
    
#     # Extract 'Date' and 'Time' and assign to the new columns
#     GH_data.at[index, 'Date'] = dt.strftime('%d-%m-%y')
#     GH_data.at[index, 'Time'] = dt.strftime('%H:%M') 


#GH_data_selected = GH_data[['Date','Time','outside air temperature','outside relative humidity','outside global radiation','outside co2 concentration']]

#GHint_data = pd.read_excel(io = './CO2_application_CGH+RGH.xlsx', sheet_name='CO2 consumption')
#GHint_data['Date_timestamp_removed'] = GHint_data['Date'].dt.date.apply(lambda x: x.strftime('%d-%m-%Y'))

#To extract 2011 GH internal data
GHint_data = pd.read_csv(filepath_or_buffer='./gh2_2011.csv', sep=',')  # Update path as needed

custom_format_ghint = '%d-%m-%y %H:%M'  # Customize this format according to timestamp format
GHint_data['Timestamp'] = GHint_data['Timestamp'].apply(lambda x: datetime.strptime(x, custom_format_ghint))

GHint_data['Date'] = GHint_data['Timestamp'].dt.date.apply(lambda x: x.strftime('%d-%m-%Y'))
GHint_data['Time'] = GHint_data['Timestamp'].dt.strftime('%H:%M')

# To extract 2012 GH internal data
# custom_format_ghint = '%d-%m-%y %H:%M'  # Customize this format according to timestamp format
# GHint_data['Timestamp'] = GHint_data['Timestamp'].apply(lambda x: datetime.strptime(x, custom_format_ghint))

# GHint_data['Date'] = GHint_data['Timestamp'].dt.date.apply(lambda x: x.strftime('%d-%m-%y'))
# GHint_data['Time'] = GHint_data['Timestamp'].dt.strftime('%H:%M')

# Select specific columns including the new 'Date' and 'Time' columns
GHint_data_selected = GHint_data[['Date', 'Time', 'air temperature gh2', 'co2 concentration gh2', 'relative humidity gh2']]


#To intialize the climate data from 2011
# Cout_d = GH_data['Cout_d']
Cout= GH_data_selected['OutsideCO2level'] 
# Hout_d = GH_data['Hout_d']
Hout = GH_data_selected['OutdoorHumidity']
# Iout_d = GH_data['Iout_d']
Iout = GH_data_selected['GlobalRadiation']
# Tout_d = GH_data['Tout_d']
Tout = GH_data_selected['OutdoorTemp']

#To intialize the climate data from 2012
# # Cout_d = GH_data['Cout_d']
# Cout= GH_data_selected['outside co2 concentration'] 
# # Hout_d = GH_data['Hout_d']
# Hout = GH_data_selected['outside relative humidity']
# # Iout_d = GH_data['Iout_d']
# Iout = GH_data_selected['outside global radiation']
# # Tout_d = GH_data['Tout_d']
# Tout = GH_data_selected['outside air temperature']

Date_out = GH_data_selected['Date']
Time_out = GH_data_selected['Time']


# T = GHint_data['T_d'] # air temp
T = GHint_data['air temperature gh2']
# C = GHint_data['CO2_d'] # co2 concentration
C = GHint_data['co2 concentration gh2']
# H = GHint_data['H_d'] # relative humidity
H = GHint_data['relative humidity gh2'] 

Date = GHint_data['Date']
Time = GHint_data['Time']

old_dt = 300
dt_ocp = 300
dt_mpc = 60

mpl.rcParams['font.size'] = 16

def L(tau_col, tau, j):
    l = 1
    for k in range(len(tau_col)):
        if k!=j:
            l *= (tau-tau_col[k])/(tau_col[j]-tau_col[k]) 
    return l

def LgrInter(tau_col, tau, xk):
    z = 0
    for j in range(len(tau_col)):
        z += L(tau_col, tau, j)*xk[j,:]

    return z
	

INDEX, p = param_obj.get_all_parameters()

# Dimensions of x and u:
nx = 4
nu = p['INDEX']['U_max']

# simulation time
t0 = 0
tf = 86400

# OCP time
Nocp = int(tf/dt_ocp)

# MPC time
Nmpc = 5 #int(dt_ocp/dt_mpc)
Tmpc = Nmpc * dt_mpc
tmpc = np.arange(0, Tmpc + dt_mpc, dt_mpc)

tf_in = int(dt_ocp/dt_mpc)

# simulation time
tsim_in = np.arange(t0, dt_ocp + dt_mpc, dt_mpc)
ksim_in = len(tsim_in)
Nsim_in = ksim_in-1

# simulation time
tsim = np.arange(t0, tf + dt_mpc, dt_mpc)
ksim = len(tsim)

# Hierarchical control
# Cost matrix for open loop - economic objective function
# Price of electricity
pE = 0.834

#price of CO2
pCO2 = 0.052

# Initialize q as a column vector
q = 0 * np.ones(nx)
q[3] = 0.5653 * 3.84 * p['A_s']   # crop price

# Initialize Reco as a diagonal matrix with 0 on the diagonal
Reco = 0 * np.eye(nu)

# Initialize Reco2 as a column vector for final cost calculation
Reco2 = 0 * np.ones(nu)

# Modify specific diagonal elements in R

Reco[0, 0] = pE * p['vPower']  # Actuator cost for input 1 - vent
Reco[1, 1] = pCO2 * p['u_CO2_max']   # Actuator cost for input 2 - co2
Reco[2, 2] = pE * p['cPower']  # Actuator cost for input 3 - heat
Reco[3, 3] = pE * p['cPower']  # Actuator cost for input 4 - cool

Reco2[0] = pE * p['vPower']  # Actuator cost for input 1 - vent
Reco2[1] = pCO2 * p['u_CO2_max']   # Actuator cost for input 2 - co2
Reco2[2] = pE * p['cPower']  # Actuator cost for input 3 - heat
Reco2[3] = pE * p['cPower']  # Actuator cost for input 4 - cool# Define the state lower and upper bounds 

Teco = 0
Ceco = 0
Heco = 0
penT = 5 * 10**-2 # Penalty cost for temp constraint violation
penC = 1 * 10**-2 # Penalty cost for CO2 constraint violation
penH = 1.17 # Penalty cost for Humidity constraint violation

# p['state_lb'] = [18, plant_obj.ppm2gpmc(p['M_c'], 500, 18), plant_obj.rh2ab(60, 18), 0] # [5, plant_obj.ppm2gpmc(p['M_c'], 1, 0), plant_obj.rh2ab(1, 5)] 

# p['state_ub'] = [26, plant_obj.ppm2gpmc(p['M_c'], 900, 26), plant_obj.rh2ab(90, 26), 1000] # [40, plant_obj.ppm2gpmc(p['M_c'], 10000, 40), plant_obj.rh2ab(100, 40)]

# p['state_hlb'] = [14, plant_obj.ppm2gpmc(p['M_c'], 300, 14), plant_obj.rh2ab(10, 14), 0] # [5, plant_obj.ppm2gpmc(p['M_c'], 1, 0), plant_obj.rh2ab(1, 5)] 

# p['state_hub'] = [30, plant_obj.ppm2gpmc(p['M_c'], 1000, 30), plant_obj.rh2ab(100, 30), 1000] # [40, plant_obj.ppm2gpmc(p['M_c'], 10000, 40), plant_obj.rh2ab(100, 40)]

p['state_lb'] = [18, plant_obj.ppm2gpmc(p['M_c'], 500, 18), 60, 0] # [5, plant_obj.ppm2gpmc(p['M_c'], 1, 0), plant_obj.rh2ab(1, 5)] 

p['state_ub'] = [26, plant_obj.ppm2gpmc(p['M_c'], 900, 26), 90, 1000] # [40, plant_obj.ppm2gpmc(p['M_c'], 10000, 40), plant_obj.rh2ab(100, 40)]

p['state_hlb'] = [14, plant_obj.ppm2gpmc(p['M_c'], 300, 14), 10, 0] # [5, plant_obj.ppm2gpmc(p['M_c'], 1, 0), plant_obj.rh2ab(1, 5)] 

p['state_hub'] = [30, plant_obj.ppm2gpmc(p['M_c'], 1000, 30), 100, 1000] # [40, plant_obj.ppm2gpmc(p['M_c'], 10000, 40), plant_obj.rh2ab(100, 40)]


# Define the input lower and upper bounds 

p['input_lb'] = [0, 0, 0, 0]
p['input_ub'] = [1, 1, 1, 1]

# Initialize Jcl array - for cost calculation
Jcl = []

# Specify the start and end dates as strings in '%d-%m-%y' format
start_date_str = '24-08-2011'  # Replace with your desired start date
end_date_str = '25-08-2011'    # Replace with your desired end date

# Convert start and end date strings to datetime objects
start_date = datetime.strptime(start_date_str, '%d-%m-%Y')
end_date = datetime.strptime(end_date_str, '%d-%m-%Y')

current_date = start_date

format_first_date_str = start_date.strftime('%d-%m-%Y')
first_day_GHint_data = GHint_data_selected[GHint_data_selected['Date']== format_first_date_str]

T = first_day_GHint_data['air temperature gh2']
C = first_day_GHint_data['co2 concentration gh2']
H = first_day_GHint_data['relative humidity gh2']
T = np.array(T).reshape(-1,1)
C = np.array(C).reshape(-1,1)
H = np.array(H).reshape(-1,1)

# Load the intital pointand calculate the respective components
X_T = np.array(T[0, 0])
X_CO2 = plant_obj.ppm2gpmc(p['M_c'],C[0,0], T[0, 0]) # C[0,0]
X_H = np.array(H[0, 0]) # plant_obj.rh2ab(H[0, 0], T[0, 0])
X_B = 1

# Standard deviation for white noise
sd = 1e-3
desired_snr_db = 100


# Create an empty list to store the data for both days
all_day_data = []
all_day_flux = []
execution_data = []

while current_date <= end_date:
    # Convert the current date to the desired string format
    formatted_date_str = current_date.strftime('%d-%m-%Y')
    
    if formatted_date_str == '27-03-2011' or formatted_date_str == '14-04-2011':
        #formatted_date_str == '27-03-11' or formatted_date_str == '14-04-11'  or formatted_date_str == '25-07-11' or formatted_date_str == '26-07-11':
        #formatted_date_str == '02-03-12' or formatted_date_str == '13-05-12'  or formatted_date_str == '18-06-12' or formatted_date_str == '16-07-12' or formatted_date_str == '02-08-12' or formatted_date_str == '18-09-12':
        current_date += timedelta(days=1)
    else: 
        # Filter the data for the current date
        current_day_GH_data = GH_data_selected[GH_data_selected['Date'] == formatted_date_str]
        current_day_GHint_data = GHint_data_selected[GHint_data_selected['Date']== formatted_date_str]

        #For 2012 weather file
        # Cout_d = np.array(current_day_GH_data['outside co2 concentration'])
        # Hout_d = np.array(current_day_GH_data['outside relative humidity'])
        # Iout_d = np.array(current_day_GH_data['outside global radiation'])
        # Tout_d = np.array(current_day_GH_data['outside air temperature'])

        #For 2011 weather file
        Cout_d = np.array(current_day_GH_data['OutsideCO2level'])
        Hout_d = np.array(current_day_GH_data['OutdoorHumidity'])
        Iout_d = np.array(current_day_GH_data['GlobalRadiation'])
        Tout_d = np.array(current_day_GH_data['OutdoorTemp'])

        for i in range(len(Cout_d)):
            if np.isnan(Cout_d[i]):
                Cout_d[i] = Cout_d[i-1]
                Hout_d[i] = Hout_d[i-1]
                Iout_d[i] = Iout_d[i-1]
                Tout_d[i] = Tout_d[i-1]


        # Apply outlier removal
        Cout_d = plant_obj.remove_outliers(Cout_d)
        Tout_d = plant_obj.remove_outliers(Tout_d)
        Hout_d = plant_obj.remove_outliers(Hout_d)
        Iout_d = plant_obj.remove_outliers(Iout_d)

        # Interpolate missing data (NaNs) linearly
        Cout_d = pd.Series(Cout_d).interpolate(method='linear').to_numpy()
        Tout_d = pd.Series(Tout_d).interpolate(method='linear').to_numpy()
        Hout_d = pd.Series(Hout_d).interpolate(method='linear').to_numpy()
        Iout_d = pd.Series(Iout_d).interpolate(method='linear').to_numpy()

        # Define the original time points (assuming the data points are evenly spaced)
        original_time_points = np.arange(1, 86401, old_dt)

        # Define the new time points for interpolation (assuming 30 seconds interval)
        new_time_points = np.arange(1, 86401, dt_ocp)
        new_time_points_mpc = np.arange(1, 86401, dt_mpc)

        # Perform spline interpolation for each variable
        Cout_spline = interpolate.splrep(original_time_points, Cout_d)
        Hout_spline = interpolate.splrep(original_time_points, Hout_d)
        Iout_spline = interpolate.splrep(original_time_points, Iout_d)
        Tout_spline = interpolate.splrep(original_time_points, Tout_d)

        # Evaluate the splines at the new time points
        Cout = interpolate.splev(new_time_points, Cout_spline)
        Hout = interpolate.splev(new_time_points, Hout_spline)
        Iout = interpolate.splev(new_time_points, Iout_spline)
        Tout = interpolate.splev(new_time_points, Tout_spline)

        d_ocp = np.vstack((Tout, Cout, Hout, Iout))

        # Symbolic variables x and u:
        x = SX.sym('x', nx,1)
        u = SX.sym('u', nu,1)
        dist = SX.sym('dist',4)

        # right hand side of ODE:
        xdot = plant_obj.dynamicsODE(x,u,dist)

        # Create the CasADi function
        system = Function("sys",[x,u, dist],[xdot])

        ode = {'x': x, 'ode': xdot, 'p': vertcat(u, dist)}

        # By default the solver integrates from 0 to 1. We change the final time to dt.
        opts = {'tf': dt_mpc}

        # Create the solver object.
        ode_solver = integrator('F', 'idas', ode, opts)

        # collocation degree
        K = 4

        # collocation points (excluding 0)
        tau_col = collocation_points(K,'radau')

        # collocation points (including 0)
        tau_col = [0]+tau_col

        tau = SX.sym('tau')

        A = np.zeros((K+1,K+1))

        for j in range(K+1):
            dLj = gradient(L(tau_col, tau, j), tau)
            dLj_fcn = Function('dLj_fcn', [tau], [dLj])
            for k in range(K+1):
                A[j,k] = dLj_fcn(tau_col[k])
                
                
        D = np.zeros((K+1,1))

        for j in range(K+1):
            Lj = L(tau_col, tau, j)
            Lj_fcn = Function('Lj', [tau], [Lj])
            D[j] = Lj_fcn(1)

        # Create the p.x0 array
        p['x0'] = np.array([X_T, X_CO2, X_H, X_B])

        # Transpose the array to make it a column vector
        p['x0'] = p['x0'].reshape(-1, 1)

        # Create the p.u0 array
        p['u0'] = np.array([0.01, 0.08, 0.1, 0.1])

        # Transpose the array to make it a column vector
        p['u0'] = p['u0'].reshape(-1, 1)

        # state constraints
        lb_x = p['state_lb']
        lb_x = np.array(lb_x)
        ub_x = p['state_ub']
        ub_x = np.array(ub_x)

        # stage cost for open loop economic optimal control
        #dx = x-xss
        Penalty = plant_obj.penalty(x[0],lb_x[0],ub_x[0],penT) + plant_obj.penalty(x[1],lb_x[1],ub_x[1],penC) + plant_obj.penalty(x[2],lb_x[2],ub_x[2],penH)
        stage_cost_ocp = u.T@Reco2 + Penalty #u.T@Reco@u #(x-xss).T@Q@(x-xss)+ u.T@R@u
        stage_cost_ocp_fcn = Function('stage_cost_ocp',[x,u],[stage_cost_ocp]) #stage_cost_fcn = Function('stage_cost',[x,u,xss],[stage_cost])

        # terminal cost for open loop economic optimal control
        # terminal_cost = (x-xss).T@Q@(x-xss)
        terminal_cost_ocp = (x-X_B).T@q 
        terminal_cost_ocp_fcn = Function('terminal_cost_ocp',[x],[terminal_cost_ocp])

        lb_hx = p['state_hlb']
        lb_hx = np.array(lb_hx)
        ub_hx = p['state_hub']
        ub_hx = np.array(ub_hx)

        # input constraints
        lb_u = p['input_lb']
        lb_u = np.array(lb_u)
        ub_u = p['input_ub']
        ub_u = np.array(ub_u)


        # Structured Optimization Variable
        # Collocation points are included in the opt. problem hence the opt. variable
        # vector includes N+1 intervals with K+1 col. points in each 
        opt_x_ocp = struct_symSX([
            entry('x', shape=nx, repeat=[Nocp+1, K+1]),
            entry('u', shape=nu, repeat=[Nocp])
        ])

        #opt_u = struct_symSX([entry('u', shape=nu, repeat=[Nocp])])

        # Populating the structured optimization variable
        #lb_opt_x = opt_x_ocp(-np.inf)
        #ub_opt_x = opt_x_ocp(+np.inf)

        lb_opt_x = opt_x_ocp(0)
        ub_opt_x = opt_x_ocp(0)

        #lb_opt_x = opt_u(0)
        #ub_opt_x = opt_u(0)

        # Setting up the constraints
        lb_opt_x['x'] = lb_hx
        ub_opt_x['x'] = ub_hx

        lb_opt_x['u'] = lb_u
        ub_opt_x['u'] = ub_u

        # Formulate MPC optmization problem
        Jocp = 0
        gocp = []    # constraint expression gocp for OCP
        lb_g = []  # lower bound for constraint expression gocp
        ub_g = []  # upper bound for constraint expression gocp

        # Defining x_init as a symbolic variable such that you can pass the values of the
        # changing initial condition to the solver at every MPC iteration
        x_init = SX.sym('x_init', nx)

        x0 = opt_x_ocp['x', 0, 0]

        # Setting up the equality constraint for the initial condition of the states
        gocp.append(x0-x_init)
        lb_g.append(np.zeros((nx,1)))
        ub_g.append(np.zeros((nx,1)))

        for i in range(Nocp):
        
            # Adding the stage cost for the current iteration of the prediction horizon
            Jocp += stage_cost_ocp_fcn(opt_x_ocp['x',i,0],opt_x_ocp['u',i]) # J += stage_cost_fcn(x_k,u_k,xss)
        
            # equality constraints (system equation)
            for k in range(1,K+1):
                gk = -dt_ocp*system(opt_x_ocp['x',i,k], opt_x_ocp['u',i],d_ocp[:,i])
                for j in range(K+1):
                    gk += A[j,k]*opt_x_ocp['x',i,j]


                gocp.append(gk)
                lb_g.append(np.zeros((nx,1)))
                ub_g.append(np.zeros((nx,1)))

            # Integrating the system for a step and multiplying by D to get the state value
            # at the beginning of the next interval
            x_next = horzcat(*opt_x_ocp['x',i])@D
            gocp.append(x_next - opt_x_ocp['x', i+1, 0])
            lb_g.append(np.zeros((nx,1)))
            ub_g.append(np.zeros((nx,1)))


        # Terminal constraint
        Jocp -= terminal_cost_ocp_fcn(opt_x_ocp['x', Nocp, 0])

        # Vertically concatenating the constraint vectors for CasADi
        gocp = vertcat(*gocp)
        lb_g = vertcat(*lb_g)
        ub_g = vertcat(*ub_g)

        opts = {'ipopt.print_level':5, 'print_time':5}

        # Creating the problem structure as a dictionary
        prob = {'f':Jocp,'x':vertcat(opt_x_ocp),'g':gocp, 'p':vertcat(x_init,dist)} #, prob = {'f':J,'x':x,'g':g, 'p':vertcat(dist,xss)}
        solver = nlpsol('solver','ipopt',prob, opts)

        # Define the initial condition
        x_0 = np.array(p['x0'])

        # Solving once for open-loop predictions
        ocp_res = solver(p=vertcat(x_0,d_ocp[:,0]), lbg=lb_g, ubg=ub_g, lbx = lb_opt_x, ubx = ub_opt_x)
        #mpc_res = mpc_solver(p=vertcat(x_0,d[:,0],xSS[:,0]), lbg=lb_g, ubg=ub_g, lbx = lb_opt_x, ubx = ub_opt_x)

        if(solver.stats()['return_status']== "Solve_Succeeded"):
            skip = 1
        else: #(solver.stats()['return_status']=="Infeasible_Problem_Detected"):
            print("check once again for different initial condition")
            # Load the intital pointand calculate the respective components
            X_T = 19
            X_CO2 = plant_obj.ppm2gpmc(p['M_c'],800, X_T) # C[0,0]
            X_H = 75 #plant_obj.rh2ab(60, X_T)


            p['x0'] = np.array([X_T, X_CO2, X_H, X_B])
            #Transpose the array to make it a column vector
            p['x0'] = p['x0'].reshape(-1, 1)

            x_0 = np.array(p['x0'])

            ocp_res = solver(p=vertcat(x_0,d_ocp[:,0]), lbg=lb_g, ubg=ub_g, lbx = lb_opt_x, ubx = ub_opt_x)
            if(solver.stats()['return_status']== "Solve_Succeeded"):
                skip = 1
                print("Different initial condition worked")
            else: #(solver.stats()['return_status']=="Infeasible_Problem_Detected"):
                skip = 0
                print("Stop execution")

        # Extracting the Lagrange multipliers for the equality constraints
        lagrange_g = ocp_res['lam_g']
        lagrange_g = np.array(lagrange_g).flatten()
        print(lagrange_g.shape)

        # Step1 : Create empty list for selecting K=0 points of equality constraints (discarding other collocation points)
        lag_g_sel = []
        
        # Step 3: Iterate with a step size of nx*(K+1) (4 entries selected, 16 skipped)
        for i in range(0, len(lagrange_g), nx*(K+1)):
            lag_g_sel.append(lagrange_g[i:i+nx])  
        lag_g_sel_ar = np.array(lag_g_sel) # Converting to array for uniformity - shape [Nocp+1,nx]

        # Extracting the Lagrange multipliers for the state boundaries
        lagrange_x = ocp_res['lam_x']

        # Step 1: Split lagrange_x into two parts
        lag_x = lagrange_x[:nx*(Nocp+1)*(K+1)]  # First 5780 entries
        lag_x_ar = np.array(lag_x).flatten()
        lag_u = lagrange_x[nx*(Nocp+1)*(K+1):]  # Remaining 1152 entries
        lag_u_ar = np.array(lag_u).flatten()
        lag_u_ar = lag_u_ar.reshape(Nocp,nu)

        # Step 2: Create empty list for selecting K=0 points (discarding other collocation points)
        lag_x_sel = []

        # Step 3: Iterate with a step size of nx*(K+1) (4 entries selected, 16 skipped)
        for i in range(0, len(lag_x_ar), nx*(K+1)):
            lag_x_sel.append(lag_x_ar[i:i+nx])  
        lag_x_sel_ar = np.array(lag_x_sel) # Converting to array for uniformity - shape [Nocp+1,nx]

        print(current_date)

        opt_x_k = opt_x_ocp(ocp_res['x'])

        Xocp = horzcat(*opt_x_k['x',:,0,:])
        Uocp = horzcat(*opt_x_k['u',:])
        
        Xref = np.array(Xocp)
        Uref = np.array(Uocp)
        print(Xref.shape)
        print(Uref.shape)

        T_ref = np.array(Xref[0,:-1]).reshape(-1,)
        C_ref = np.array(Xocp[1,:-1]).reshape(-1,)
        H_ref = np.array(Xocp[2,:-1]).reshape(-1,)
        B_ref = np.array(Xocp[3,:-1]).reshape(-1,)
        B_fruit_ref = B_ref * 0.5653
        B_prune_ref = B_ref * 0.2678  * 0.05

        Uv_ref = np.array(Uref[0,:]).reshape(-1,)
        UCO2_ref = np.array(Uref[1,:]).reshape(-1,)
        Uh_ref = np.array(Uref[2,:]).reshape(-1,)
        Uc_ref = np.array(Uref[3,:]).reshape(-1,)

        # Creating individual state and input box inequality lagrangian
        Tieq_lg = lag_x_sel_ar[1:,0].reshape(-1,)
        Cieq_lg = lag_x_sel_ar[1:,1].reshape(-1,)
        Hieq_lg = lag_x_sel_ar[1:,2].reshape(-1,)
        Bieq_lg = lag_x_sel_ar[1:,3].reshape(-1,)

        uVieq_lg = lag_u_ar[:,0].reshape(-1,)
        uCO2ieq_lg = lag_u_ar[:,1].reshape(-1,)
        uHieq_lg = lag_u_ar[:,2].reshape(-1,)
        uCieq_lg = lag_u_ar[:,3].reshape(-1,)

        # Creating individual state equality constraints lagrangian
        Teq_lg = lag_g_sel_ar[:-1,0].reshape(-1,)
        Ceq_lg = lag_g_sel_ar[:-1,1].reshape(-1,)
        Heq_lg = lag_g_sel_ar[:-1,2].reshape(-1,)
        Beq_lg = lag_g_sel_ar[:-1,3].reshape(-1,)

        X_T = Xref[0,-1]
        X_CO2 = Xref[1,-1]
        X_H = Xref[2,-1]
        X_B = Xref[3,-1]

        phi_cond = []
        for i in range(Nocp):
            cond = plant_obj.condense(Xref[:,i],Uref[:,i],d_ocp[:,i])
            phi_cond.append(cond)

        results_df = pd.DataFrame({
        'Date': [current_date] * len(T_ref),  # Repeat the date for each row
        'Temp_ref' : T_ref,
        'CO2_ref' : plant_obj.gpmc2ppm(p['M_c'],C_ref, T_ref),
        'relhum_ref' : H_ref,
        'Bio_ref' : B_ref,
        'Vent_ref' : Uv_ref,
        'CO2_inj_ref' : UCO2_ref,
        'heat_ref' : Uh_ref,
        'cool_ref' : Uc_ref,
        'cond_ref' : phi_cond,
        'fruit_ref' : B_fruit_ref,
        'prune_ref' : B_prune_ref,
        'Tout' : Tout,
        'Cout': Cout,
        'Hout': Hout,
        'Iout': Iout,
        'Tieq_lg':Tieq_lg,
        'Cieq_lg':Cieq_lg,
        'Hieq_lg':Hieq_lg,
        'Bieq_lg':Bieq_lg,
        'uVieq_lg':uVieq_lg,
        'uCO2ieq_lg':uCO2ieq_lg,
        'uHieq_lg':uHieq_lg,
        'uCieq_lg':uCieq_lg,
        'Teq_lg':Teq_lg,
        'Ceq_lg':Ceq_lg,
        'Heq_lg':Heq_lg,
        'Beq_lg':Beq_lg       
        })

        if(skip == 0):
            infeasible_date.append(current_date) 

        flux_list = []
        for i in range(Nocp):
            computeflux = plant_obj.flux(Xref[:,i],Uref[:,i],d_ocp[:,i])
            flux_list.append(np.array(computeflux).flatten())

        flux = np.vstack(flux_list)  # Now make the full 2D array

        flux_df = pd.DataFrame(flux, columns=[
            'phi_QT_vent', 'phi_QT_sun', 'phi_QT_cov', 'phi_QT_trans',
            'phi_QT_heat', 'phi_QT_cool', 'phi_QCO2_inj', 'phi_QCO2_vent',
            'Phi_CO2_ac', 'phi_H_trans', 'phi_H_vent', 'phi_H_cov', 'phi_H_cool'
        ])
    # Append the DataFrame to the list
    all_day_data.append(results_df)

    # Append the DataFrame to the list
    all_day_flux.append(flux_df)

    # Increment the current date by one day
    current_date += timedelta(days=1)


# Concatenate the data for both days into a single DataFrame
final_result_df = pd.concat(all_day_data, ignore_index=True)

# Create a DataFrame for infeasible dates
infeasible_df = pd.DataFrame({'Infeasible_Date': infeasible_date})

# Create a DataFrame for flux
flux_df = pd.concat(all_day_flux, ignore_index=True)

# Concatenate the infeasible dates DataFrame with the final result DataFrame
final_result_df = pd.concat([final_result_df, flux_df, infeasible_df], axis=1)

# Save the final result DataFrame to a CSV file
csv_filename = 'HC_picn.csv'  # Provide the desired file name
final_result_df.to_csv(csv_filename, index=False)

# Print a message to confirm the file has been saved
print(f"DataFrame saved to {csv_filename}")

