import gurobipy as gp
import numpy as np
from gurobipy import GRB


def compute_credits(agent_num, coalition_set, advantage, advantage_grand):
    batch_size = advantage.shape[1]
    solutions = []
    results = []

    coalition_set = [list(coalition) for coalition in coalition_set]

    LP = gp.Model()
    LP.setParam('OutputFlag', 0)
    LP.setParam('Threads', 1)
    LP.setParam('Presolve', 2)
    LP.setParam('Method', 1)
    LP.setParam('OptimalityTol', 1e-9)
    LP.setParam('FeasibilityTol', 1e-9)
    LP.setParam('NumericFocus', 1)
    LP.setParam('LogToConsole', 0)
    LP.setParam('LogFile', '')

    eps = LP.addVar(lb=0, vtype=GRB.CONTINUOUS, name="eps")
    x = LP.addVars(range(agent_num), lb=-GRB.INFINITY, vtype=GRB.CONTINUOUS, name="x")

    obj = eps
    LP.setObjective(obj, GRB.MINIMIZE)

    constr = LP.addConstrs(
        (gp.quicksum(x[i] for i in coalition_set[j]) + eps >= 0 for j in range(len(coalition_set))),
        name="c"
    )
    constr_grand = LP.addConstr(gp.quicksum(x[i] for i in range(agent_num)) == 0, name="c_grand")

    LP.update()

    if not isinstance(advantage, np.ndarray):
        advantage = np.array(advantage)
    if not isinstance(advantage_grand, np.ndarray):
        advantage_grand = np.array(advantage_grand)

    for b in range(batch_size):
        # 更新约束的 RHS
        current_advantage = advantage[:, b]
        current_grand = advantage_grand[b]
        for j in range(len(coalition_set)):
            constr[j].RHS = current_advantage[j]
        constr_grand.RHS = current_grand

        if b > 0:
            last_solution = solutions[-1][:-1]
            for i in range(agent_num):
                x[i].Start = last_solution[i]
            eps.Start = solutions[-1][-1]

        LP.optimize()

        if LP.status == GRB.OPTIMAL:
            solution = [x[i].X for i in range(agent_num)] + [eps.X]
            result = [x[i].X for i in range(agent_num)]
        else:
            solution = [0.0] * (agent_num + 1)  # 处理不可行解
            result = [0.0] * agent_num

        solutions.append(solution)
        results.append(result)

    LP.dispose()
    return np.array(results)