# -*- coding: utf-8 -*-

import numpy as np
import matplotlib.pyplot as plt
import time
import pandas as pd
from sklearn.model_selection import train_test_split
#import itertools
#from scipy.sparse import csr_matrix

###Parameters 

Epsilon=0.1
NTime=24
dt=15
Nt=(60//dt)*NTime+1
NP=4
Td=np.linspace(0,NTime,Nt)

###Import Data

DataTransaction=pd.read_excel('./elaadnl_open_ev_datasets.xlsx',sheet_name="open_transactions")
#DataMeter=pd.read_excel('./elaadnl_open_ev_datasets.xlsx',sheet_name="open_metervalues")
#DataMeter['Hour'] = DataMeter['UTCTime'].dt.hour
#DataMeter['WeekDay'] = DataMeter['UTCTime'].dt.dayofweek
DataTransaction['WeekDay'] = DataTransaction['UTCTransactionStart'].dt.dayofweek
DataTransaction['ConnectedTime4'] = ((DataTransaction['ConnectedTime']*60)//dt)
DataTransaction['ChargeTime4'] = ((DataTransaction['ChargeTime']*60)//dt)

def Pow(lP):
    return(np.select([lP < 2, (2 <= lP) & (lP < 5), (5 <= lP) & (lP <= 10), lP > 10],[0, 1, 2, 3]))

DataTransaction['Start4'] = DataTransaction['UTCTransactionStart'].dt.hour*(60//dt)+((DataTransaction['UTCTransactionStart'].dt.minute)//dt)
DataTransaction['Stop4'] = DataTransaction['UTCTransactionStop'].dt.hour*(60//dt)+((DataTransaction['UTCTransactionStop'].dt.minute)//dt)

#These 2 lines allows to deal with vehicles arriving and leaving on different days. We are forcing the algo to charge them on the first day if possible.
DataTransaction['ChargeTime4'] = DataTransaction.apply(lambda row: max(0,min(row['ChargeTime4'], Nt - row['Start4'])), axis=1) #Comment this line to generate rHist
DataTransaction['Stop4'] = DataTransaction.apply(lambda row: Nt if row['UTCTransactionStop'].date() != row['UTCTransactionStart'].date() else row['Stop4'],axis=1)#Comment this line to generate rHist

DataTransactionOld=DataTransaction
DataTransaction['MaxPower'] = Pow(DataTransaction['MaxPower'])

DataTransaction['Day'] = DataTransaction['UTCTransactionStart'].dt.day_of_year
DataTransaction=DataTransaction.drop(['TransactionId', 'ChargePoint', 'Connector', 'UTCTransactionStart','UTCTransactionStop', 'StartCard', 'ConnectedTime', 'ChargeTime','TotalEnergy'],axis=1)
WeekendData = DataTransaction[DataTransaction['WeekDay'].isin([5, 6])]
WeekdayData = DataTransaction[~DataTransaction['WeekDay'].isin([5, 6])]

#We delete EVs with ChargeTime4=0 or low power or too long connected time:
WeekdayData=WeekdayData[WeekdayData.ChargeTime4>0]
WeekdayData=WeekdayData[WeekdayData.MaxPower>0]
WeekdayData=WeekdayData[WeekdayData['ConnectedTime4']<Nt]

rate=0.1
TrainDay,TestDay=train_test_split(np.unique(WeekdayData.Day), test_size=rate, random_state=0)

WeekdayTest = WeekdayData[WeekdayData['Day'].isin(TestDay)]
WeekdayTrain = WeekdayData[WeekdayData['Day'].isin(TrainDay)]
WeekdayTest=WeekdayTest.drop(['Day'],axis=1)
WeekdayTrain=WeekdayTrain.drop(['Day'],axis=1)



#WeekendTrain, WeekendTest = train_test_split(WeekendData, test_size=0.1, random_state=42)
#WeekdayTrain, WeekdayTest = train_test_split(WeekdayData, test_size=0.1, random_state=42)

#rHist=np.zeros(Nt)+0.1

dtTrain=WeekdayTrain
dtTest=WeekdayTest

def SaverHist(): ##Need to set NTime=48 and comment rHist= before running this function
    rHist24=ProdFM(fT,mu1)[24*(60//dt):]
    np.save("rHist"+str(dt)+".npy",rHist24)
    
rHist=np.load("rHist"+str(dt)+".npy")*len(dtTrain)/9

### PlotHeatmap of behaviors
#DataMeter=pd.read_excel('./elaadnl_open_ev_datasets.xlsx',sheet_name="open_metervalues")
#DataMeter['Hour'] = DataMeter['UTCTime'].dt.hour
#DataMeter['WeekDay'] = DataMeter['UTCTime'].dt.dayofweek
# DataTransaction['Hour'] = DataTransaction['UTCTransactionStart'].dt.hour
# Tab = DataTransaction.groupby(['ChargeTime4', 'Hour']).count()
# Mz=Tab.pivot_table(index='ChargeTime4', columns='Hour', values='TransactionId')
# Mz.fillna(0, inplace=True)
# plt.imshow(Mz,aspect="auto", cmap='Blues',origin='lower')
# plt.colorbar(label='number')
# plt.xlabel("Hour of the day")
# plt.ylabel("ChargeTime")
# plt.xticks(ticks=np.arange(len(Mz.columns))[::2], labels=Mz.columns[::2])
# plt.yticks(ticks=np.arange(len(Mz.index))[::4], labels=Mz.index[::4])

### Plot Consumption

#DataMeter=pd.read_excel('./elaadnl_open_ev_datasets.xlsx',sheet_name="open_metervalues")
#DataMeter['Hour'] = DataMeter['UTCTime'].dt.hour
#DataMeter['WeekDay'] = DataMeter['UTCTime'].dt.dayofweek
# total_consumption = DataMeter.groupby(['WeekDay', 'Hour'])["EnergyInterval"].sum()
# total_consumption = total_consumption.sort_index()
# plot_data = total_consumption.unstack(level=0)
# plt.plot(plot_data)
# plt.xlabel('Heure de la journée')
# plt.ylabel('Consommation d\'énergie totale')
# plt.title('Energy Consumption')
# plt.legend(title='Jour de la semaine', labels=["Monday","Tuesday","Wednesday","Thursday","Friday","Saturday","Sunday"])
# plt.show()

###Functions

cT=np.zeros((Nt,Nt,Nt,NP,Nt))
for ta in range(0,Nt):
    for tc in range(0,Nt):
        cT[ta,:,:,:,tc]+=(ta*NTime/Nt-tc*NTime/Nt)**2

fT=np.zeros((Nt,Nt,Nt,NP,Nt))
gT=np.zeros((Nt-1,Nt,Nt,NP,Nt))
for ta in range(Nt):
    print(str(ta+1)+"/"+str(Nt),flush=True)
    for tch in range(Nt):
        for tc in range(Nt):
            for p in range(NP):
                for T in range(tc,min(Nt,tc+tch)):
                    fT[T,ta,tch,p,tc]+=p*4
                gT[:,ta,tch,p,tc]=fT[1:,ta,tch,p,tc]-fT[:-1,ta,tch,p,tc]

D=np.zeros((Nt-1,Nt))
for i in range(Nt-1):
    D[i,i]=-1
    D[i,i+1]=1
gT=np.tensordot(D,fT,axes=[-1,0])  #Discretization of the second general moment in the 1S-RCMOT problem, representing the gradient of the global consumption

S0=np.zeros(Nt-1)+(120/726)*NTime/(Nt-1) # Bounds for the gradient overall consumption

### Definition of mu1 and mu2

nu0=np.zeros((Nt,Nt,Nt,NP))

for row in dtTrain.itertuples(index=False):
    if row[4] <Nt and row[2] < Nt and row[3]<Nt and row[0]<NP:
        nu0[int(row[4]),int(row[2]),int(row[3]),int(row[0])]+=1
nu0=nu0/np.sum(nu0)

nuReal=np.zeros((Nt,Nt,Nt,NP))

for row in dtTest.itertuples(index=False):
    if row[4] <Nt and row[2] < Nt and row[3]<Nt and row[0]<NP:
        nuReal[int(row[4]),int(row[2]),int(row[3]),int(row[0])]+=1
nuReal=nuReal/np.sum(nuReal)

muReal=np.zeros((Nt,Nt,Nt,NP,Nt))

for t in range(0,Nt):
    muReal[t,:,:,:,t]=nuReal[t]

mu1=np.zeros((Nt,Nt,Nt,NP,Nt))

for t in range(0,Nt):
    mu1[t,:,:,:,t]=nu0[t]

mu2=np.zeros((Nt,Nt,Nt,NP,Nt))

for ta in range(0,Nt):
    for tco in range(0,Nt):
        for tch in range(0,Nt):
            for p in range(NP):
                for tc in range(ta,min(ta+tco-tch+1,Nt)):
                    mu2[ta,tco,tch,p,tc]+=1
                if np.sum(mu2[ta,tco,tch,p])!=0:
                    mu2[ta,tco,tch,p]=mu2[ta,tco,tch,p]*nu0[ta,tco,tch,p]/np.sum(mu2[ta,tco,tch,p])

###Useful functions

def ProdFM(f,m):
    return(np.tensordot(f,np.sum(m,1),axes=[(1,2,3,4),(0,1,2,3)]))

NominalConsumption=ProdFM(fT,mu1)*len(dtTest)
PredictedConsumption=ProdFM(fT,muReal)*len(dtTest)

# def gradJL(tOn,lbda,beta,nu_0,mu_2,rHist):
#     S=np.tensordot(lbda[tOn:]+np.dot(beta[max(tOn-1,0):],D[max(tOn-1,0):,tOn:]),fT[tOn:,tOn:,:,:,tOn:],axes=[0,0])[:,np.newaxis]+cT[tOn:,:,:,:,tOn:]
#     B=np.exp(-S/Epsilon)*mu_2
#     UB=np.sum(B,-1)
#     MuL=B*(np.divide(nu_0,UB,out=np.zeros_like(nu_0), where=UB!=0)[..., np.newaxis])
#     return(ProdFM(fT[:,tOn:,:,:,tOn:],MuL,rHist))

# def gradJB(tOn,lbda,beta,nu_0,mu_2):
#     S=np.tensordot(lbda[tOn:],fT[tOn:,tOn:,:,:,tOn:],axes=[0,0])[:,np.newaxis]+cT[tOn:,:,:,:,tOn:]+np.tensordot(beta[tOn:],gT[tOn:,tOn:,:,:,tOn:],axes=[0,0])[:,np.newaxis]+cT[tOn:,:,:,:,tOn:]
#     B=np.exp(-S/Epsilon)*mu_2
#     UB=np.sum(B,-1)
#     MuL=B*(np.divide(nu_0,UB,out=np.zeros_like(nu_0), where=UB!=0)[..., np.newaxis])
#     return(-np.sign(beta)*S0+ProdFM(gT[:,tOn:,:,:,tOn:],MuL))

# def gradJBFromMu(tOn,beta,MuL):
#     return(-np.sign(beta)*S0+ProdFM(gT[:,tOn:,:,:,tOn:],MuL))

def gradJLFromMu(tOn,MuL):
    return(ProdFM(fT[:,:,:,:,tOn:],MuL))

    # S=np.tensordot(lbda,fT,axes=[0,0])
    # B=np.tensordot(np.exp(-S/Epsilon)*mu_2,np.exp(-cT/Epsilon)[tOn:],axes=[4,1])
    # fB=np.tensordot(fT*np.exp(-S/Epsilon)*mu_2,np.exp(-cT/Epsilon)[tOn:],axes=[4,1])
    # return(ProdFM(np.divide(fB, B, out=np.zeros_like(fB), where=B!=0),mu_1[:,:,:,:,tOn:]))

def RangeR(tOn,rangeR):
    l=[]
    for i in range(tOn,Nt):
        if i in rangeR:
            l.append(i)
    return(l)

def NotRangeR(tOn,rangeR):
    l=[]
    for i in range(max(0,tOn-1),Nt-1):
        if i not in rangeR:
            l.append(i)
    return(l)

RTest=np.zeros(Nt)+np.mean(ProdFM(fT,mu1))

###Algo

lbd0=np.zeros(Nt)
bet0=np.zeros(Nt-1)

def prho(tOn,k):
    if tOn==0:
        return(min(0.5,5/((1+k)**0.7)))
    else:
        return(0.5/((1+k)**0.7))

def AlgoOnTimeAtStept(tOn,nu_0,mu_2,R,rangeR,Klimit=300,plot=True,prin=True,lbd0A=lbd0,bet0A=bet0,S0=[],NomC1=NominalConsumption,Pred=PredictedConsumption):
    lbda=lbd0A.copy()
    beta=bet0A.copy()
    k=0
    GL=lbd0.copy()+np.inf
    GB=bet0.copy()+np.inf
    rangeR2=RangeR(tOn, rangeR)
    while k<Klimit and (np.linalg.norm((GL[rangeR2]>R[rangeR2])*(GL[rangeR2]-R[rangeR2]))>0.03 ): #or np.linalg.norm(((GB-S0)*(GB>S0)+(GB+S0)*(GB<-S0))[max(0,tOn-1):])>0.10
        MuL=MuLambda(tOn, lbda, beta, nu_0, mu_2)
        GL=gradJLFromMu(tOn, MuL)
        GB=np.dot(D,GL)
        lbda[rangeR2]+=prho(tOn,k)*(GL[rangeR2]-R[rangeR2])
        lbda=np.max([lbda,lbd0],0)
        if len(S0)>0:
            beta[max(0,tOn-1):]+=prho(tOn,k)*(-np.sign(beta)*S0+GB)[max(0,tOn-1):]
        if plot:
            Display(tOn,GL,R,rangeR2,it=k,Nom=NomC1,Pred=Pred)
            # plt.plot(np.dot(D,GL))
            # plt.plot(S0)
            # plt.plot(-S0)
            # plt.show()
        if prin:
            print("Step "+str(k)+ " GL:" +str(np.linalg.norm((GL[rangeR2]>R[rangeR2])*(GL[rangeR2]-R[rangeR2])))+" GB:"+str(np.linalg.norm(((GB-S0)*(GB>S0)+(GB+S0)*(GB<-S0))[max(0,tOn-1):])),flush=True)
        k+=1
    return(lbda,beta)

def VehiclesAtStepT(t):
    l=[]
    for row in dtTest[dtTest['Start4']==t].itertuples(index=False):
        if row[4] <Nt and row[2] < Nt and row[3]<Nt and row[0]<NP:
            l.append([int(row[4]),int(row[2]),int(row[3]),int(row[0])])
    return(l)

def InitT(tOn,listX,N):
    nu_0=np.zeros((Nt,Nt,Nt,NP))
    mu_2=np.zeros((Nt,Nt,Nt,NP,Nt-tOn))
    N1t=len(listX)
    N2t=np.sum(nu0[tOn+1:])*N
    nu_0[0:tOn+1]=nu0[0:tOn+1]*0
    nu_0[tOn+1:]=nu0[tOn+1:]*N2t/(N1t+N2t)
    for V in listX:
        nu_0[V[0],V[1],V[2],V[3]]+=1/(N1t+N2t)
    for ta in range(0,Nt):
        for tco in range(0,Nt):
            for tch in range(0,Nt):
                for p in range(NP):
                    for tc in range(max(0,ta-tOn),min(ta-tOn+tco-tch+1,Nt-tOn)):
                        mu_2[ta,tco,tch,p,tc]+=1
                    if np.sum(mu_2[ta,tco,tch,p])!=0:
                        mu_2[ta,tco,tch,p]=mu_2[ta,tco,tch,p]*nu_0[ta,tco,tch,p]/np.sum(mu_2[ta,tco,tch,p])
    return(nu_0,mu_2)

def AlgoOnTime(R2,rangeR,N=len(dtTrain)*rate/(1-rate),Klimit=75,S0=[]):
    lbda_star=lbd0.copy()
    beta_star=bet0.copy()
    muFinal=np.zeros((Nt,Nt,Nt,NP,Nt))
    t0=time.time()
    listX=[]
    for tOn in range(Nt):
        print(str((tOn*dt)//60)+"h"+str((tOn*dt)%60),flush=True)
        listX+=VehiclesAtStepT(tOn)
        N1t=len(listX)
        N2t=np.sum(nu0[tOn+1:])*N
        NomC1=NominalConsumption/(N1t+N2t)
        Pred=PredictedConsumption/(N1t+N2t)
        r=(R2-ProdFM(fT, muFinal)-rHist)/(N1t+N2t)
        nu_0,mu_2=InitT(tOn, listX,N)
        lbda_star,beta_star=AlgoOnTimeAtStept(tOn, nu_0, mu_2, r, rangeR,prin=False,plot=True,Klimit=Klimit,lbd0A=lbda_star,bet0A=beta_star,S0=S0,NomC1=NomC1,Pred=Pred) #range(max(np.min(rangeR),tOn),np.max(rangeR))
        print("Running Time MuComputing: "+str(round(time.time()-t0))+"s",flush=True)
        MuL=MuLambda(tOn, lbda_star, beta_star, nu_0, mu_2)
        listXNew=[]
        for V in listX:
            if np.sum(MuL[V[0],V[1],V[2],V[3]])!=0:
                tc=np.random.choice(range(Nt-tOn),p=MuL[V[0],V[1],V[2],V[3]]/np.sum(MuL[V[0],V[1],V[2],V[3]]))
                if tc==0:
                    muFinal+=np.array([[[[[t1==V[0] and tch1==V[2] and tco1==V[1] and p1==V[3] and tc1==tc+tOn for tc1 in range(Nt)] for p1 in range(NP)] for tch1 in range(Nt) ]for tco1 in range(Nt)]for t1 in range(Nt)])
                else:
                    listXNew.append(V)
            else:
                tc=np.random.choice(range(Nt-tOn))
                print("Error",flush=True)
        listX=listXNew
        Display(tOn,ProdFM(fT, muFinal)+rHist, R2, rangeR,Pred=PredictedConsumption+rHist,Nom=NominalConsumption+rHist)
        print("Running Time Vehicles Plugging: "+str(round(time.time()-t0))+"s",flush=True)
        if len(S0)>0:
            np.save("MuFinalGB.npy",muFinal)
        else:
            np.save("MuFinal.npy",muFinal)    
    return(muFinal)

def AlgoOnTimeReject(R2,rangeR,N=len(dtTrain)*rate/(1-rate),Klimit=75,S0=[]):
    lbda_star=lbd0.copy()
    beta_star=bet0.copy()
    muFinal=np.zeros((Nt,Nt,Nt,NP,Nt))
    t0=time.time()
    listX=[]
    listR=[]
    p_rest=R2-rHist
    for tOn in range(Nt):
        print(str((tOn*dt)//60)+"h"+str((tOn*dt)%60),flush=True)
        listX+=VehiclesAtStepT(tOn)
        N1t=len(listX)
        N2t=np.sum(nu0[tOn+1:])*N
        NomC1=NominalConsumption/(N1t+N2t)
        Pred=PredictedConsumption/(N1t+N2t)
        r=p_rest/(N1t+N2t)
        nu_0,mu_2=InitT(tOn, listX,N)
        lbda_star,beta_star=AlgoOnTimeAtStept(tOn, nu_0, mu_2, r, rangeR,prin=False,plot=False,Klimit=Klimit,lbd0A=lbda_star,bet0A=beta_star,S0=S0,NomC1=NomC1,Pred=Pred) #range(max(np.min(rangeR),tOn),np.max(rangeR))
        print("Running Time MuComputing: "+str(round(time.time()-t0))+"s",flush=True)
        MuL=MuLambda(tOn, lbda_star, beta_star, nu_0, mu_2)
        listXNew=[]
        for V in listX:
            posstc=[]
            for t in range(0,Nt-tOn):
                if np.min(p_rest[t+tOn:]-fT[t+tOn:,V[1],V[2],V[3],t+tOn])>=0:
                    posstc.append(t)
            if len(posstc)>0 and np.sum(MuL[V[0],V[1],V[2],V[3],posstc])!=0:
                tc=np.random.choice(posstc,p=MuL[V[0],V[1],V[2],V[3],posstc]/np.sum(MuL[V[0],V[1],V[2],V[3],posstc]))
                if tc==0:
                    muFinal+=np.array([[[[[t1==V[0] and tch1==V[2] and tco1==V[1] and p1==V[3] and tc1==tc+tOn for tc1 in range(Nt)] for p1 in range(NP)] for tch1 in range(Nt) ]for tco1 in range(Nt)]for t1 in range(Nt)])
                    p_rest-=fT[:,V[1],V[2],V[3],tOn]
                else:
                    listXNew.append(V)
            else:
                listR.append(V)
                print("Not Charged",flush=True)
        listX=listXNew
        #Display(tOn,ProdFM(fT, muFinal)+rHist, R2, rangeR,Pred=PredictedConsumption+rHist,Nom=NominalConsumption+rHist)
        print("Running Time Vehicles Plugging: "+str(round(time.time()-t0))+"s",flush=True)
        np.save("MuFinalReject.npy",muFinal)
        np.save("listR.npy",listR)
    return(muFinal,listR)

###Display

def Display(tOn,G,R2,rangeR,it="",Nom=NominalConsumption,Pred=PredictedConsumption):
    plt.plot(Td,Nom,label='Nominal consumption')
    plt.plot(Td,Pred,label="Predicted consumption")
    plt.plot(Td,G,label="Optimized consumption")
    plt.plot(Td[rangeR],R2[rangeR],label="Constraint")
    plt.xlabel("Time")
    plt.ylabel("Aggregated consumption")
    plt.grid()
    plt.legend()
    if it!="":
        plt.title("Time: "+str(tOn)+ " Iteration: "+str(it))
    plt.show()

def MuLambda(tOn,lbda,beta,nu_0,mu_2):
    S=np.tensordot(lbda[tOn:]+np.dot(beta[max(tOn-1,0):],D[max(tOn-1,0):,tOn:]),fT[tOn:,:,:,:,tOn:],axes=[0,0])[:,np.newaxis]+cT[:,:,:,:,tOn:]
    B=np.exp(-S/Epsilon)*mu_2
    UB=np.sum(B,-1)
    MuL=B*(np.divide(nu_0,UB,out=np.zeros_like(nu_0), where=UB!=0)[..., np.newaxis])
    return(MuL)   

def SaveWithTD(Name,Arr):
    Mat=np.transpose([Td,Arr])
    np.savetxt(Name,Mat)

#############################################################################################
############################      IOCS Algorithm      #######################################
#############################################################################################

def ICS(df,ptotal,rHist):
    df = df.copy()
    df.loc[:, 'Di'] = df['ChargeTime4'] * df['MaxPower']*4*dt/60
    df.loc[:, 'vi'] = df['ChargeTime4']
    y={t: [] for t in range(0, T + 1)}
    beta=np.zeros(Nt)
    notSelected=[]
    for i, ev in df.sort_values(by="vi", ascending=False).iterrows():
        ai=ev.Start4
        di=ev.Stop4
        sig=ptotal-np.array([sum(power for _, power in y[t]) for t in range(Nt)])-rHist
        inter=0
        ts=int(ai)
        while inter<ev.ChargeTime4 and ts<di:
            if ev.MaxPower*4<=sig[ts]:
                inter+=1
            else:
                inter=0
            ts+=1
        if inter==ev.ChargeTime4:
            for t in range(ts-inter,ts):
                y[t].append((ev.name,ev.MaxPower*4))
    return(y)


def IOCS(df, ptotal, rHist):
    df = df.copy()
    allocation = {t: [] for t in range(0, T + 1)}
    active_evs = []
    s=0
    p_rest=ptotal-rHist
    for t in range(0, Nt):
        new_evs = df[(df['Start4']==t) & (~df.index.isin(active_evs))]
        s+=len(new_evs)
        active_evs += new_evs.index.tolist()
        y=ICS(df[df.index.isin(active_evs)], p_rest, np.zeros(Nt))
        plugged_evs=[y[t][i][0] for i in range(len(y[t]))]
        p_rest-=np.array([sum(power*(ev in plugged_evs) for ev, power in y[t]) for t in range(Nt)])
        for t1 in range(t,Nt):
            allocation[t1]+=[evp for evp in y[t1] if evp[0] in plugged_evs]
        active_evs=[ev for ev in active_evs if ev not in plugged_evs]
    return allocation

def AfficheIOCS(allocation,ptotal,muF=""):
    time_slots = sorted(allocation.keys())
    consumption = [sum(power for _, power in allocation[t]) for t in time_slots]
    plt.figure(figsize=(10, 6))
    plt.plot(time_slots, consumption+rHist, marker='o', label='Total Consumption')
    plt.plot(time_slots,np.zeros(Nt)+ptotal, color='r', linestyle='--', label='Constraint')
    plt.plot(ProdFM(fT,muReal)*len(dtTest)+rHist,label='Nominal')
    if muF!="":
        plt.plot(ProdFM(fT,np.load(muF))+rHist,label='Online MCOT')
    plt.xlabel('Time')
    plt.ylabel('Consumption (kW)')
    plt.legend()
    plt.grid(True)
    plt.show()

def Charged(allocation):
    return({a for t in allocation for a, b in allocation[t]})

def NotCharged(allocation):
    evCharged={a for t in allocation for a, b in allocation[t]}
    return([x for x in WeekdayTest.index.tolist()if x not in evCharged])

print(S0)
R2=650+np.zeros(Nt)
rangeR=range(Nt)
muFinal,listR=AlgoOnTimeReject(R2,rangeR,N=len(dtTrain)*rate/(1-rate),Klimit=0*75,S0=S0)
plt.plot(Td,ProdFM(fT, muFinal)+rHist,label="MCOT")
allocation=IOCS(dtTest, 650, rHist)
time_slots = sorted(allocation.keys())
consumption = [sum(power for _, power in allocation[t]) for t in time_slots]
plt.plot(Td,consumption,label="IOCS")
plt.plot(Td[rangeR],R2[rangeR],label="Constraint")
plt.xlabel("Time")
plt.ylabel("Aggregated consumption")
plt.grid()
plt.legend()
