{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "1ac1beb0",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.optimize import linprog"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5164ffea",
   "metadata": {},
   "outputs": [],
   "source": [
    "def projsplx(y):\n",
    "    \"\"\"Python implementation of:\n",
    "    https://arxiv.org/abs/1101.6081\"\"\"\n",
    "    s = np.sort(y)\n",
    "    n = len(y) ; flag = False\n",
    "    \n",
    "    parsum = 0\n",
    "    tmax = -np.inf\n",
    "    for idx in range(n-2, -1, -1):\n",
    "        parsum += s[idx+1]\n",
    "        tmax = (parsum - 1) / (n - (idx + 1) )\n",
    "        if tmax >= s[idx]:\n",
    "            flag = True ; break\n",
    "    \n",
    "    if not flag:\n",
    "        tmax = (np.sum(s) - 1) / n\n",
    "    \n",
    "    return np.maximum(y - tmax, 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "5eaf2f26",
   "metadata": {},
   "outputs": [],
   "source": [
    "nstr = 2\n",
    "def init_player(k=nstr):\n",
    "    l = np.random.rand(k)\n",
    "    l /= np.sum(l)\n",
    "    return l\n",
    "\n",
    "def grad(T, _players, idx):\n",
    "    players = [x for i, x in enumerate(_players) if i!=idx ]\n",
    "    recipe = 'ijkl,'+','.join(['jkl','ikl','ijl','ijk'][idx])+'->'+ ('ijkl'[idx])\n",
    "    return np.einsum( recipe, T, *players)\n",
    "\n",
    "def utility(T, players):\n",
    "    return np.einsum('ijkl,i,j,k,l', T, *players)\n",
    "\n",
    "def nash_gap(T, players):\n",
    "    util = utility(T, players)\n",
    "    gap = 0.\n",
    "    for idx in range(len(players)):\n",
    "        if idx < 2:\n",
    "            u = -util\n",
    "            br = np.max(-grad(T, players, idx))\n",
    "            gap += (br - u)\n",
    "        else:\n",
    "            u = util\n",
    "            br = np.max(grad(T, players, idx))\n",
    "            gap += (br - u)\n",
    "    return gap"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "e3f93075",
   "metadata": {},
   "outputs": [],
   "source": [
    "nstr = 2\n",
    "np.random.seed(5)\n",
    "T = 2 * (np.random.rand(nstr, nstr, nstr, nstr) - 1/2)\n",
    "T /= np.max(np.abs(T))\n",
    "T *= 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "14c78fa7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7f7b1ee1d4c0>]"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "eta = 0.05\n",
    "k = -1.2\n",
    "p = 0.02\n",
    "# k = 0\n",
    "\n",
    "NEG = []\n",
    "\n",
    "x1, x2, y1, y2 = [init_player(nstr) for _ in range(4) ]\n",
    "thx1, thx2, thy1, thy2 = [init_player(nstr) for _ in range(4) ]\n",
    "\n",
    "for _ in range(20_000):\n",
    "    players = [x1, x2, y1, y2]\n",
    "    \n",
    "    x1new   = x1   - eta * grad(T, players, 0) + eta * k * (x1 - thx1)\n",
    "    thx1new = thx1 + eta * p * (x1 - thx1)\n",
    "    \n",
    "    x2new   = x2   - eta * grad(T, players, 1) + eta * k * (x2 - thx2)\n",
    "    thx2new = thx2 + eta * p * (x2 - thx2)\n",
    "    \n",
    "    y1new   = y1   + eta * grad(T, players, 2) + eta * k * (y1 - thy1)\n",
    "    thy1new = thy1 + eta * p * (y1 - thy1)\n",
    "    \n",
    "    y2new   = y2   + eta * grad(T, players, 3) + eta * k * (y2 - thy2)\n",
    "    thy2new = thy2 + eta * p * (y2 - thy2)\n",
    "    \n",
    "    x1, x2, y1, y2 = [projsplx(z) for z in [x1new, x2new, y1new, y2new]]\n",
    "    thx1, thx2, thy1, thy2 = [projsplx(z) for z in [thx1new, thx2new, thy1new, thy2new]]\n",
    "    \n",
    "    neg = nash_gap(T, [x1, x2, y1, y2])\n",
    "    \n",
    "    NEG.append(neg)\n",
    "#     X1.append(x1)\n",
    "#     X2.append(x2)\n",
    "#     Y1.append(y1)\n",
    "#     Y2.append(y2)\n",
    "\n",
    "plt.plot(NEG)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "376aaf6d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_kpv(seed):\n",
    "    np.random.seed(seed)\n",
    "    print(seed)\n",
    "    \n",
    "    T = 2 * (np.random.rand(nstr, nstr, nstr, nstr) - 1/2)\n",
    "    T /= np.max(np.abs(T))\n",
    "    T *= 2\n",
    "\n",
    "    eta = 0.05\n",
    "    k = -1.2\n",
    "    p = 0.02\n",
    "\n",
    "    NEG = []\n",
    "    \n",
    "    x1, x2, y1, y2 = [init_player(nstr) for _ in range(4) ]\n",
    "    thx1, thx2, thy1, thy2 = [init_player(nstr) for _ in range(4) ]\n",
    "\n",
    "    for _ in range(100_000):\n",
    "        players = [x1, x2, y1, y2]\n",
    "\n",
    "        x1new   = x1   - eta * grad(T, players, 0) + eta * k * (x1 - thx1)\n",
    "        thx1new = thx1 + eta * p * (x1 - thx1)\n",
    "\n",
    "        x2new   = x2   - eta * grad(T, players, 1) + eta * k * (x2 - thx2)\n",
    "        thx2new = thx2 + eta * p * (x2 - thx2)\n",
    "\n",
    "        y1new   = y1   + eta * grad(T, players, 2) + eta * k * (y1 - thy1)\n",
    "        thy1new = thy1 + eta * p * (y1 - thy1)\n",
    "\n",
    "        y2new   = y2   + eta * grad(T, players, 3) + eta * k * (y2 - thy2)\n",
    "        thy2new = thy2 + eta * p * (y2 - thy2)\n",
    "\n",
    "        x1, x2, y1, y2 = [projsplx(z) for z in [x1new, x2new, y1new, y2new]]\n",
    "        thx1, thx2, thy1, thy2 = [projsplx(z) for z in [thx1new, thx2new, thy1new, thy2new]]\n",
    "\n",
    "        neg = nash_gap(T, [x1, x2, y1, y2])\n",
    "\n",
    "        NEG.append(neg)\n",
    "    return (seed, NEG)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "a6c99d0e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "3\n"
     ]
    }
   ],
   "source": [
    "_, w = run_kpv(3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "d92e45c8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7f7b1e229040>]"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(w)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "8044c9b6",
   "metadata": {},
   "outputs": [],
   "source": [
    "from multiprocessing import Pool"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "f97dbdb4",
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "38776429443474324457\n",
      "1824079525\n",
      "16234720633960690481776908216\n",
      "\n",
      "36324723381713653108\n",
      "\n",
      "\n",
      "\n",
      "2806378439\n",
      "3380734117\n",
      "2395907837\n",
      "3064741550\n",
      "1552908833\n",
      "99289201\n",
      "966601729\n",
      "1356210005\n",
      "1745989749\n",
      "2417079159\n",
      "2014083018\n",
      "3213311723\n",
      "1156358012\n",
      "2618313700\n",
      "1253240419\n",
      "873401051\n",
      "33057203\n",
      "402377757419248843541784211496615818187825409994\n",
      "3575036440478831317\n",
      "\n",
      "\n",
      "\n",
      "\n",
      "\n",
      "42025867\n",
      "2189537827\n",
      "3762890437\n",
      "4630612\n",
      "1117381778\n",
      "4132826326\n",
      "1801425359\n",
      "654240901\n",
      "3999063397\n",
      "2309056134\n",
      "513754378\n",
      "1767072156\n",
      "624155476\n",
      "2326231873\n",
      "3880302106\n",
      "4008560692\n",
      "3508561601\n",
      "2448531460\n",
      "2638522359\n",
      "1602029156\n",
      "1448910120\n",
      "3617764669\n",
      "3164566373\n",
      "530975521\n",
      "4186452319\n",
      "1691516525\n",
      "404186467\n",
      "781102271\n",
      "2297629991\n",
      "1106757064\n",
      "1167894161\n",
      "3384411427\n",
      "3286699296\n",
      "4253949314\n",
      "4139439808\n",
      "1985337035\n",
      "3454319209\n",
      "4239532123\n",
      "3213905531\n",
      "2129287515\n",
      "178411408\n",
      "623411986\n",
      "461404521\n",
      "834142393\n",
      "3452376874\n",
      "282831727\n",
      "1758677892\n",
      "2440259719\n",
      "1577510944\n",
      "4151295079\n",
      "1906040217\n",
      "168828848\n",
      "382331446\n",
      "3376656669\n",
      "39661725\n",
      "3886951937\n",
      "3676093892\n",
      "3119709956\n",
      "736466804\n",
      "315841122\n",
      "3127368893\n",
      "1835733154\n",
      "3731157627\n",
      "4053121908\n",
      "1001989652\n",
      "2325670258\n",
      "772688861\n",
      "457201353\n",
      "3574218493\n",
      "3233958543\n",
      "1700439279\n",
      "4051353876\n",
      "3343677557\n",
      "2574188400\n",
      "3287657413\n",
      "2947270270\n",
      "4038900582\n",
      "944281464\n",
      "1782018145\n",
      "1719970840\n",
      "1696416641\n",
      "2621126505\n",
      "3678350977\n",
      "3138335834\n",
      "2394587974\n",
      "1895893501\n",
      "3189896553\n",
      "3938995715\n",
      "378317646\n",
      "2168237536\n",
      "569191808\n",
      "2409270\n",
      "3132331367\n",
      "3055491896\n",
      "4252008931\n",
      "4070502564\n",
      "3275420422\n",
      "1351475759\n",
      "1287159235\n",
      "2483653537\n",
      "1633225018\n",
      "790600414\n",
      "3608617389\n",
      "3767705036\n",
      "3137500059\n",
      "3316936099\n",
      "2897390516\n",
      "1014733443\n",
      "685101418\n",
      "1055761610\n",
      "1125915791\n",
      "3233727779\n",
      "1772927969\n",
      "3788294362\n",
      "3231694313\n",
      "3998048612\n",
      "2533796637\n",
      "2530376164\n",
      "3383514076\n",
      "683417121\n",
      "3768345021\n",
      "2210312673\n",
      "2060534907\n",
      "2176980973\n",
      "4221139634\n",
      "827417411\n",
      "3345919690\n",
      "1562467732\n",
      "2999448471\n",
      "1477663979\n",
      "2049853661\n",
      "4057981884\n",
      "86111281\n",
      "3825910615\n",
      "3441256076\n",
      "2540848793\n",
      "200380827\n",
      "3798978732\n",
      "3979429655\n",
      "541051187\n",
      "2283075074\n",
      "3938104878\n",
      "1265667805\n",
      "3876721206\n",
      "3815119775\n",
      "2823834962\n",
      "2013027414\n",
      "2205439647\n",
      "2442852364\n",
      "386831249\n",
      "828628183\n",
      "1588923956\n",
      "3627546473\n",
      "2839528934\n",
      "1119755176\n",
      "3480740315\n",
      "1050183741\n",
      "804135453\n",
      "2931444563\n",
      "2436125688\n",
      "3347490972\n",
      "553585389\n",
      "258251453\n",
      "4251429767\n",
      "518754790\n",
      "4022392477\n",
      "3334654150\n",
      "1855660270\n",
      "3695586783\n",
      "472671395\n",
      "3583972260\n",
      "1792604080\n",
      "2244047317\n",
      "1837897630\n",
      "452291103\n",
      "1153243169\n",
      "1444171366\n",
      "324343440\n",
      "4018190936\n",
      "2327132591\n",
      "1631472035\n",
      "911604904\n",
      "1439360732\n",
      "3981073462\n",
      "51177182\n",
      "3987558753\n",
      "3512752419\n",
      "1787502669\n",
      "1150092379\n",
      "64213151\n",
      "3823116844\n",
      "2582063717\n",
      "3578244811\n",
      "197129069\n",
      "1614247157\n",
      "1627508471\n",
      "2088675034\n",
      "977979253\n",
      "2383004067\n",
      "3298942389\n",
      "1472686221\n",
      "2154879842\n",
      "2966669207\n",
      "1874986936\n",
      "791374479\n",
      "2966425134\n",
      "2737941406\n",
      "2465102334\n",
      "4126774967\n",
      "3038984110\n",
      "1375209792\n",
      "1194298382\n",
      "2879226581\n",
      "1259538211\n",
      "1237083047\n",
      "3543200259\n",
      "3198070739\n",
      "356119260\n",
      "2155865223\n",
      "3268331186\n",
      "2060882549\n",
      "2004991380\n",
      "924599310\n",
      "3502600379\n",
      "1314165085\n",
      "3594840090\n",
      "390287546\n",
      "2899055931\n",
      "1436644890\n",
      "262927469\n",
      "868793314\n",
      "1400942146\n",
      "445488298\n",
      "2545083356\n",
      "1452852010\n",
      "474103952\n",
      "2526632692\n",
      "63192743\n",
      "4288030775\n",
      "3344144459\n",
      "3780217434\n",
      "423786835\n",
      "1738913592\n",
      "3177371280\n",
      "2327351780\n",
      "1540431308\n",
      "97359615\n",
      "3407649358\n",
      "2245300502\n",
      "1868841899\n",
      "2336043272\n",
      "4118457814\n",
      "2960790507\n",
      "1288242165\n",
      "3469212473\n",
      "3328446908\n",
      "1550557950\n",
      "2415406495\n",
      "3767892234\n",
      "1198053347\n",
      "3874790562\n",
      "1002751368\n",
      "805807104\n",
      "1307435519\n",
      "984717958\n",
      "1593033889\n"
     ]
    }
   ],
   "source": [
    "pool = Pool(8)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "aa66c0f6",
   "metadata": {},
   "outputs": [],
   "source": [
    "# res = pool.map(run_kpv, np.random.randint(0, 2**32 -1 , size=100 ) )\n",
    "\n",
    "res2 = pool.map(run_kpv, np.random.randint(0, 2**32 -1 , size=200 ) )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "afc2b282",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "for idx in range(len(res)):\n",
    "    print(res[idx][0])\n",
    "    plt.plot(res[idx][1], c='tab:blue')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "78a5cb1d",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "# from colorspacious import cspace_converter\n",
    "\n",
    "mpl.rcParams.update({'font.size': 14})\n",
    "\n",
    "# Indices to step through colormap.\n",
    "x = np.linspace(0.0, 1.0, 100)\n",
    "\n",
    "gradient = np.linspace(0, 1, 256)\n",
    "gradient = np.vstack((gradient, gradient))\n",
    "cmaps = {}\n",
    "\n",
    "def plot_color_gradients(category, cmap_list):\n",
    "    # Create figure and adjust figure height to number of colormaps\n",
    "    nrows = len(cmap_list)\n",
    "    figh = 0.35 + 0.15 + (nrows + (nrows - 1) * 0.1) * 0.22\n",
    "    fig, axs = plt.subplots(nrows=nrows + 1, figsize=(6.4, figh))\n",
    "    fig.subplots_adjust(top=1 - 0.35 / figh, bottom=0.15 / figh,\n",
    "                        left=0.2, right=0.99)\n",
    "    axs[0].set_title(f'{category} colormaps', fontsize=14)\n",
    "\n",
    "    for ax, name in zip(axs, cmap_list):\n",
    "        ax.imshow(gradient, aspect='auto', cmap=mpl.colormaps[name])\n",
    "        ax.text(-0.01, 0.5, name, va='center', ha='right', fontsize=10,\n",
    "                transform=ax.transAxes)\n",
    "\n",
    "    # Turn off *all* ticks & spines, not just the ones with colormaps.\n",
    "    for ax in axs:\n",
    "        ax.set_axis_off()\n",
    "\n",
    "    # Save colormap list for later.\n",
    "    cmaps[category] = cmap_list"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f217a2da",
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_color_gradients('Perceptually Uniform Sequential',\n",
    "                     ['viridis', 'plasma', 'inferno', 'magma', 'cividis'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8d0d2ca7",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib\n",
    "from matplotlib import cm\n",
    "\n",
    "\n",
    "#normalize item number values to colormap\n",
    "norm = matplotlib.colors.Normalize(vmin=0, vmax=2000)\n",
    "\n",
    "plt.figure(figsize=(10,5))\n",
    "for idx in range(len(res)):\n",
    "#     print(res[idx][0])\n",
    "    rgba_color = cm.jet(norm(10 * idx),bytes=True) \n",
    "#     plt.plot( np.random.rand(1000) + 2*idx, c=tuple( np.array(rgba_color) /255) )\n",
    "\n",
    "    plt.plot(res2[idx][1], c=tuple( np.array(rgba_color) /255))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "fb3a31ef",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(0, 0, 241, 255)"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7348990d",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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": 5
}
