import gurobipy as gp
from gurobipy import GRB
from gurobipy import quicksum as qsum
import numpy as np

def stylized_invOptimization(env, f_list, qw_list, qs_list, w_list, lambda_ls, num, K, B, G):

    V_S = env.scenario.warehouse
    V_W = env.scenario.factory
    E = env.scenario.G.edges
    mT = np.array([env.random_graph.edges[(i,j)]['cost'] for (i,j) in E])
    cs = np.array([env.scenario.storage_capacities[i] for i in env.scenario.warehouse])
    cw = np.array([env.scenario.storage_capacities[i] for i in env.scenario.factory])

    m = gp.Model()
    # m.setParam('OutputFlag', 0)

    theta, mu, x, y, alpha, eta, v1, z1, z2, t, epsilon = create_variables(m, num, E, V_W, V_S, K)

    add_constraints(m, num, theta, mu, x, y, alpha, eta, v1, z1, z2, t, epsilon, B, G, f_list, qs_list, qw_list, w_list, lambda_ls, mT, cs, cw)
    # m.setObjective(sum(theta[i]@theta[i] for i in range(K)), GRB.MINIMIZE)
    m.setObjective(sum(theta[i]@theta[i] for i in range(K)) + 3*qsum(epsilon[i]*epsilon[i] for i in range(num)), GRB.MINIMIZE)
    m.setParam("TimeLimit", 120)  # 时间/限制
    # m.setParam('MIPGap', 0.2)
    m.optimize()

    # m.computeIIS()  # 计算 IIS
    # m.write("inventory.ilp")  # 保存 IIS 到文件（可选）

    return_ls = []
    theta_return_ls = []
    for i in range(K):
        theta_return_ls.append(theta[i].X)
        # thetaf_return_ls.append(thetaf_ls[i][-1])
    return_ls.append(theta_return_ls)
    mu_return_ls = []
    for i in range(K):
        mu_return_ls.append(mu[i].X)
        # mu_return_ls.append(mu_ls[i][-1])
    return_ls.append(mu_return_ls)
    
    return return_ls


def create_variables(m, num, E, V_W, V_S, K):
    # 创建变量
    theta = [m.addMVar(shape=len(E), vtype=GRB.CONTINUOUS, name=f'theta_{i}') for i in range(K)]
    mu = [m.addMVar(shape=1, name=f'mu{i}') for i in range(K)]

    x = m.addMVar(shape=(num, len(V_S)), lb=0, vtype=GRB.CONTINUOUS, name='x')
    y = m.addMVar(shape=(num, len(V_W)), lb=0, vtype=GRB.CONTINUOUS, name='y')
    alpha = m.addMVar(shape=(num, len(V_S)), lb=-10000, vtype=GRB.CONTINUOUS, name='alpha')
    v1 = m.addMVar(shape=(num, len(E)), lb=0, vtype=GRB.CONTINUOUS, name='v1')
    eta = m.addMVar(shape=(num, len(V_W)), lb=0, vtype=GRB.CONTINUOUS, name='eta')

    z1 = [m.addMVar(shape=num, name=f'z{i}', lb=0, ub=1) for i in range(K)]
    z2 = [m.addMVar(shape=num, name=f'z{i}', lb=0, ub=1) for i in range(K)]
    t = [m.addMVar(shape=num, name=f't{i}', lb=0) for i in range(K)]
    epsilon =  m.addMVar(shape = num, name = 'epsilon')

    return theta, mu, x, y, alpha, eta, v1, z1, z2, t, epsilon


def add_constraints(m, num, theta, mu, x, y, alpha, eta, v1, z1, z2, t, epsilon, B, G, f_list, qs_list, qw_list, w_list, lambda_ls, mT, cs, cw):
    K = len(theta)
    l1 = 10e-5
    for i in range(num):
        f = f_list[i]
        qs = qs_list[i]
        qw = qw_list[i]
        w = w_list[i]

        d = np.array(lambda_ls[i])

        # m.addConstr(v1[i,:]@f == 0)
        Gfq = np.dot(G, f) - qw
        # m.addConstr(Gfq@ y[i,:]== 0)
        Gfs = qs+np.dot(B,f)-d-cs
        # m.addConstr(Gfs@x[i,:]== 0)
        Gfw = qw + w - np.dot(G, f) - cw
        # m.addConstr(Gfw@eta[i,:] == 0)


        for j in range(K):
            # m.addConstr(z1[j][i]*(-theta[j]@f + mu[j] - t[j][i]) == 0)
            # m.addConstr(z2[j][i]*t[j][i] == 0)
            m.addConstr(t[j][i] >= -theta[j]@f + mu[j])
            m.addConstr(1 - z1[j][i] - z2[j][i] == 0)
        
        m.addConstr(mT + x[i, :]@B + y[i, :]@G -eta[i, :]@G+ alpha[i,:]@B - v1[i, :] - sum(z1[j][i]*theta[j] for j in range(K)) + 2*l1*f== 0)
        # m.addConstr(x[i, :]@B + y[i, :]@G + alpha[i,:]@B - v1[i, :] - sum(z1[j][i]*theta[j] for j in range(K)) == 0)
        
        #strong duality
        m.addConstr(Gfs@x[i,:]+Gfq@ y[i,:]-v1[i,:]@f+Gfw@eta[i,:]+sum(z1[j][i]*(-theta[j]@f+mu[j]-t[j][i]) for j in range(K)) - sum(z2[j][i]*t[j][i] for j in range(K)) + epsilon[i]== 0)


        # f = f_list[i]
        # qs = qs_list[i]
        # qw = qw_list[i]
        # d = np.array(lambda_ls[i])

        # m.addConstr(v1[i,:]@f == 0)
        # Gfq = np.dot(G, f) - qw
        # m.addConstr(Gfq@ y[i,:]== 0)
        # Gfs = qs+np.dot(B,f)-d-cs
        # m.addConstr(Gfs@x[i,:]== 0)

        # for j in range(K):
        #     m.addConstr(z1[j][i]*(-theta[j]@f + mu[j] - t[j][i]) == 0)
        #     m.addConstr(z2[j][i]*t[j][i] == 0)
        #     m.addConstr(t[j][i] >= -theta[j]@f + mu[j])
        #     m.addConstr(1 - z1[j][i] - z2[j][i] == 0)
        
        # m.addConstr(mT + x[i, :]@B + y[i, :]@G + alpha[i,:]@B - v1[i, :] - sum(z1[j][i]*theta[j] for j in range(K)) == 0)