{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import gurobipy as gp\n",
    "from gurobipy import GRB\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "def soft_grouping(S, n):\n",
    "    m = len(S)\n",
    "\n",
    "    # Create a Gurobi model\n",
    "    model = gp.Model()\n",
    "    model.setParam(\"NonConvex\", 2)\n",
    "    model.setParam(\"OutputFlag\", 0)\n",
    "\n",
    "    # Create binary decision variables\n",
    "    X = model.addVars(m, n, vtype=GRB.BINARY, name=\"X\")\n",
    "\n",
    "    # Create continuous decision variables for y\n",
    "    y = model.addVars(n, vtype=GRB.CONTINUOUS, name=\"y\")\n",
    "\n",
    "    z = model.addVars(m, n, m, vtype=GRB.BINARY, name=\"z\")\n",
    "\n",
    "    objective_expr = gp.quicksum(\n",
    "        y[j]\n",
    "        * gp.quicksum(\n",
    "            [gp.quicksum([z[i, j, k] * S[i, k] for i in range(m)]) for k in range(m)]\n",
    "        )\n",
    "        for j in range(n)\n",
    "    )\n",
    "\n",
    "    model.setObjective(objective_expr, sense=GRB.MAXIMIZE)\n",
    "\n",
    "    # Each task is assigned to exactly one group\n",
    "    for i in range(m):\n",
    "        model.addConstr(\n",
    "            gp.quicksum(X[i, j] for j in range(n)) >= 1, name=f\"constraint1_{i}\"\n",
    "        )\n",
    "\n",
    "    # introduce the auxiliary variable y\n",
    "    for j in range(n):\n",
    "        model.addConstr(\n",
    "            gp.quicksum(X[i, j] for i in range(m)) * y[j] == 1,\n",
    "            name=f\"constraint2_{j}\",\n",
    "        )\n",
    "\n",
    "    # each group has at least one task\n",
    "    for j in range(n):\n",
    "        model.addConstr(\n",
    "            gp.quicksum(X[i, j] for i in range(m)) >= 1, name=f\"constraint3_{j}\"\n",
    "        )\n",
    "\n",
    "    # no same groups\n",
    "    for j1 in range(n):\n",
    "        for j2 in range(j1, n):\n",
    "            if j1 != j2:\n",
    "                model.addConstr(\n",
    "                    gp.quicksum(\n",
    "                        X[i, j1] + X[i, j2] - 2 * X[i, j1] * X[i, j2] for i in range(m)\n",
    "                    )\n",
    "                    >= 1,\n",
    "                    name=f\"constraint4_{j1}_{j2}\",\n",
    "                )\n",
    "\n",
    "    # introduce z to relax to a quadratic program\n",
    "    for i in range(m):\n",
    "        for j in range(n):\n",
    "            for k in range(m):\n",
    "                model.addConstr(\n",
    "                    z[i, j, k] - X[i, j] * X[k, j] == 0, name=f\"constraint5_{i}_{j}_{k}\"\n",
    "                )\n",
    "\n",
    "    # Optimize the model\n",
    "    model.optimize()\n",
    "    print('Sovling Time: {:.3f}s'.format(model.Runtime))\n",
    "\n",
    "    groups = [[] for i in range(n)]\n",
    "    soln = np.zeros((m, n))\n",
    "    # Print the solution\n",
    "    if model.status == GRB.OPTIMAL:\n",
    "        for j in range(n):\n",
    "            for i in range(m):\n",
    "                soln[i, j] = X[i, j].X\n",
    "                if np.isclose(X[i, j].X, 1):\n",
    "                    groups[j].append(i)\n",
    "    else:\n",
    "        print(\"No optimal solution found.\")\n",
    "    return groups, model.objVal\n",
    "\n",
    "\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "game",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
