{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "from scipy.spatial.distance import cdist\n",
    "# from cplex import Cplex\n",
    "from gurobipy import *"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[0.07212333, 0.39643502, 0.78429519, 0.33299864],\n",
       "       [0.39824652, 0.51978404, 0.22256341, 0.57773067],\n",
       "       [0.39648451, 0.6963166 , 0.95354293, 0.12625811],\n",
       "       [0.59531928, 0.33496655, 0.79738176, 0.48429561],\n",
       "       [0.96170103, 0.15688933, 0.28185969, 0.63001982]])"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "a = np.random.random([4,5])\n",
    "a.sum(0)\n",
    "a = a.T\n",
    "a"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([2.42387466, 2.10439154, 3.03964298, 2.15130285])"
      ]
     },
     "execution_count": 24,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "a.sum(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[1., 1., 1., 1.],\n",
       "       [2., 2., 2., 2.],\n",
       "       [3., 3., 3., 3.],\n",
       "       [4., 4., 4., 4.],\n",
       "       [5., 5., 5., 5.]])"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "b = np.array([1,2,3,4,5]).reshape([-1, 1])\n",
    "b\n",
    "c = b @ np.ones([b.shape[1], a.shape[1]])\n",
    "c"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[0.07212333, 0.39643502, 0.78429519, 0.33299864],\n",
       "       [0.79649303, 1.03956808, 0.44512682, 1.15546133],\n",
       "       [1.18945352, 2.08894979, 2.8606288 , 0.37877433],\n",
       "       [2.38127713, 1.33986621, 3.18952704, 1.93718245],\n",
       "       [4.80850514, 0.78444666, 1.40929844, 3.15009911]])"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "a * c"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def rounding(df, centers, alpha, beta, color_flag, phi):\n",
    "    \n",
    "    data_center_dist = cdist(df.values, centers)\n",
    "    data_center_cost = data_center_dist * data_center_dist\n",
    "    \n",
    "    for var in color_flag:\n",
    "        for color in range(min(color_flag[var]), max(color_flag[var])+1):\n",
    "            labels = np.zeros(df.shape[0])\n",
    "            # clients_belong_color is a vector with the same size with clients\n",
    "            # if client_belong_color[i] = 1, that means the i th client belong to this color\n",
    "            clients_belong_color = [i == color for i in color_flag[var]]\n",
    "            clients_belong_color_index = np.nonzero(clients_belong_color)[0]\n",
    "            \n",
    "            # clients_belong_color_points = df.values[clients_belong_color_index]\n",
    "            color_phi = phi[clients_belong_color_index]\n",
    "            color_cost_matrix = data_center_cost[clients_belong_color_index]\n",
    "            \n",
    "            rounded_color_phi = rounding_color(color_cost_matrix, color_phi)\n",
    "            phi[clients_belong_color_index] = rounded_color_phi\n",
    "            \n",
    "    return phi\n",
    "\n",
    "def rounding_color(color_cost_matrix, color_phi):\n",
    "    \n",
    "    # Gamma is the weight vector of centers\n",
    "    gamma = color_phi.sum(0).reshape([-1, 1])\n",
    "    floor_gamma = np.floor(gamma)\n",
    "    fractional_gamma = gamma - floor_gamma\n",
    "    \n",
    "    ratio_floor = floor_gamma / gamma\n",
    "    ratio_floor_phi = ratio_floor @ np.ones([ratio_floor.shape[1], color_phi.shape[1]])\n",
    "    \n",
    "    # The transmission matrix from center to clients\n",
    "    # reversed_phi = color_phi.T\n",
    "    reversed_cost = color_cost_matrix.T\n",
    "    k, n = reversed_cost.shape\n",
    "    \n",
    "    # floor_phi = ratio_floor_phi * reversed_phi\n",
    "    # fractional_phi = reversed_phi - floor_phi\n",
    "    \n",
    "    problem = Model('mip')\n",
    "    integral_x = problem.addVars(k, n, vtype = GRB.BINARY, ub = 1)\n",
    "    fractional_x = problem.addVars(k, n, vtype = GRB.BINARY, ub = 1)\n",
    "    \n",
    "    problem.setObjective(sum(integral_x[(i,j)]* reversed_cost[i,j] for i in range(n) for j in range(k)) + sum(fractional_x[(i,j)]* reversed_cost[i,j] for i in range(n) for j in range(k)), GRB.MINIMIZE)\n",
    "    \n",
    "    problem.addConstrs(sum(integral_x[(j, i)] for i in range(n)) == floor_gamma[j] for j in range(k))\n",
    "    problem.addConstrs(sum(fractional_x[(j, i)] for i in range(n)) <= 1 for j in range(k))\n",
    "    problem.addConstrs(sum(integral_x[(j, i)] + fractional_x[(j, i)] for j in range(k)) == 1 for i in range(n))\n",
    "    \n",
    "    # close the output\n",
    "    problem.setParam('outPutFlag',0)\n",
    "    \n",
    "    # optimize the model\n",
    "    problem.optimize()\n",
    "    \n",
    "    res = {\n",
    "            \"status\": problem.Status,\n",
    "            \"objective\": problem.ObjVal,\n",
    "            'assignment': (np.array(list(problem.getAttr('x', integral_x).values())).reshape(k, n) + np.array(list(problem.getAttr('x', fractional_x).values())).reshape(k, n)).T\n",
    "        }\n",
    "    \n",
    "    return res\n",
    "    "
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "ph",
   "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.10.13"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
