{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "6905dc4b",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Python Libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0a2b2a95",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Imports\n",
    "import numpy as np\n",
    "import math\n",
    "from scipy.optimize import linear_sum_assignment\n",
    "import random\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib.patches import Rectangle, Circle\n",
    "from keras.datasets import mnist\n",
    "from keras.datasets import cifar10"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "20ba19d0",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Costum Tree Library"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "84b362e1",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Tree:\n",
    "    def __init__(self, children):\n",
    "        self.children = children\n",
    "        self.n_vertices = len(children)\n",
    "        self.level_sets = Get_Level_Sets(self)\n",
    "        self.n_leaves = len(self.level_sets[0])\n",
    "        self.height = Get_Tree_Height(self)\n",
    "        self.subtree_leaves = Get_Subtree_Leaves(self)\n",
    "        self.parent = Get_Parents(self)\n",
    "        self.weight = Get_Weights(self)\n",
    "        self.uniform = Get_Uniform_Vector(self)\n",
    "    \n",
    "        \n",
    "    def subtree_requests(self, R):\n",
    "        return Get_Requests_In_Subtree(self, R)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0f8c5d30",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Get_Weights(t):\n",
    "    weight = np.zeros(t.n_vertices)\n",
    "    for l in range(0,t.height+1):\n",
    "        for v in t.level_sets[l]:\n",
    "            weight[v] = 2**l\n",
    "    return weight"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6367bd82",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Get_Uniform_Vector(t):\n",
    "    uni = np.zeros(t.n_vertices)\n",
    "    for l in range(0,t.height + 1):\n",
    "        for v in t.level_sets[l]:\n",
    "            if (l==0):\n",
    "                uni[v] = 1/t.n_leaves\n",
    "            else:\n",
    "                uni[v] = sum([uni[u] for u in t.children[v]])\n",
    "    return uni"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "04536b14",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Get_Level_Sets(t):\n",
    "    level_nodes = [[0]]\n",
    "    for dth in range(1,t.n_vertices):\n",
    "        d = []\n",
    "        for v in level_nodes[dth-1]:\n",
    "            for u in t.children[v]:\n",
    "                d.append(u)\n",
    "        if (d==[]):\n",
    "            break\n",
    "        level_nodes.append(d)\n",
    "    level_nodes.reverse()\n",
    "    return level_nodes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "438d1bf6",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Get_Tree_Height(t):\n",
    "    return len(t.level_sets) - 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "19c1d644",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Get_Subtree_Leaves(t):\n",
    "    subtree_leaves = [-1]*t.n_vertices\n",
    "    for l in range(0,t.height+1):\n",
    "        for v in t.level_sets[l]:\n",
    "            if (l==0):\n",
    "                subtree_leaves[v] = [v]\n",
    "            else:     \n",
    "                subtree_leaves[v] = []\n",
    "                for u in t.children[v]:\n",
    "                    subtree_leaves[v] += subtree_leaves[u]\n",
    "    return subtree_leaves"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f0fd97b6",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Get_Parents(t):\n",
    "    parent = [-1]*t.n_vertices\n",
    "    for i in range(t.n_vertices):\n",
    "        for u in t.children[i]:\n",
    "            parent[u] = i\n",
    "    return parent"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "34bfbb67",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Get_Requests_In_Subtree(t, R):\n",
    "    subtree_req = [0]*t.n_vertices\n",
    "    for l in range(0,t.height+1):\n",
    "        for v in t.level_sets[l]:\n",
    "            if (l==0):\n",
    "                if (v in R):\n",
    "                    subtree_req[v] = 1\n",
    "            else:     \n",
    "                subtree_req[v] = sum([subtree_req[u] for u in t.children[v]]) \n",
    "    return subtree_req"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "604001c3",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# HST Embedding Routines"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3259709a",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Tree_from_Frac(tree, t2g, y):\n",
    "    y_tree = np.zeros(tree.n_vertices)\n",
    "    for l in range(tree.height+1):\n",
    "        for v in tree.level_sets[l]:\n",
    "            if (l==0):\n",
    "                y_tree[v] = y[t2g[v]]\n",
    "            else:\n",
    "                y_tree[v] = sum([y_tree[u] for u in tree.children[v]])\n",
    "    return y_tree"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "071f98fd",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_ball(v,R,dist):\n",
    "    b = []\n",
    "    for i in range(len(dist)):\n",
    "        if (dist[v][i] <= R):\n",
    "            b.append(i)\n",
    "    return b"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e6584fac",
   "metadata": {},
   "outputs": [],
   "source": [
    "def decomp(S, delta, dist):\n",
    "    n = len(S)\n",
    "    R = np.random.uniform(delta/4,delta/2)\n",
    "    ball = []\n",
    "    for i in range(n):\n",
    "        b = get_ball(S[i],R,dist)\n",
    "        b_rev = [item for item in S if item in b]\n",
    "        ball.append(b_rev)\n",
    "    P = [i for i in range(n)]\n",
    "    np.random.shuffle(P)\n",
    "    sets = []\n",
    "    for i in range(n):\n",
    "        sets.append([])   \n",
    "    for i in range(n):\n",
    "        sets[i] = ball[P[i]]\n",
    "        for j in range(i):\n",
    "            sets[i] = [item for item in sets[i] if item not in sets[j]]\n",
    "    return [x for x in sets if x]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2a52ec6c",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Embed(dist):\n",
    "    \n",
    "    diam = np.max(dist)\n",
    "    dmin = diam\n",
    "    n = len(dist)\n",
    "    for i in range(n):\n",
    "        for j in range(n):\n",
    "            if (dist[i][j] < dmin and dist[i][j]>0):\n",
    "                dmin = dist[i][j]\n",
    "    tree = []\n",
    "    tree.append([i for i in range(n)])\n",
    "    delta = diam/2\n",
    "    active_level = decomp(tree[0], delta, dist)\n",
    "    tree = tree + active_level\n",
    "    children = []\n",
    "    children.append([i for i in range(1,len(active_level)+1)])\n",
    "    next_ind = len(active_level)+1\n",
    "\n",
    "    it = 0\n",
    "    while(delta>= 0.001*dmin):\n",
    "        delta = delta/2\n",
    "        active_level2 = []\n",
    "        for j in active_level:\n",
    "            s = decomp(j, delta, dist)\n",
    "            active_level2 += s\n",
    "            children.append([i for i in range(next_ind,next_ind + len(s))])\n",
    "            next_ind = next_ind + len(s)\n",
    "        tree = tree + active_level2\n",
    "        active_level = active_level2\n",
    "        flag = 1\n",
    "        for j in active_level:\n",
    "            if (len(j)>1):\n",
    "                flag=0\n",
    "                break\n",
    "        if (flag==1):\n",
    "            break\n",
    "            \n",
    "            \n",
    "    for j in active_level:\n",
    "        children.append([])\n",
    "        \n",
    "    n_tree_vertices = len(children)\n",
    "    tree_to_graph = [-1]*n_tree_vertices\n",
    "    for i in range(n_tree_vertices - n, n_tree_vertices):\n",
    "        for j in tree[i]:\n",
    "            tree_to_graph[i] = j\n",
    "    graph_to_tree = [-1]*n\n",
    "    for i in range(n_tree_vertices - n, n_tree_vertices):\n",
    "        graph_to_tree[tree_to_graph[i]] = i\n",
    "\n",
    "    return children, tree_to_graph, graph_to_tree"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2a246fc4",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Cut & Round"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fe5cdaae",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Alloc(y, a, y_rem, Y_rem):\n",
    "    d = y - np.floor(y)\n",
    "    d_rem = y_rem - np.floor(y_rem)\n",
    "    Y = np.floor(y)\n",
    "    if (Y_rem == np.floor(y_rem)):\n",
    "        if (d >= d_rem and a <= (d-d_rem)/(1-d_rem)):\n",
    "            Y += 1\n",
    "    else:\n",
    "        if (d < d_rem):\n",
    "            if (a <= d/d_rem):\n",
    "                Y += 1\n",
    "        else:\n",
    "            Y += 1\n",
    "    return Y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a28f5148",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Cut_and_Round(tree,y,a,n_fac):\n",
    "    y_n = y\n",
    "    Y = np.zeros(tree.n_vertices)\n",
    "    Y[0] = n_fac\n",
    "    for l in range(tree.height,0,-1):\n",
    "        for v in tree.level_sets[l]:\n",
    "            Y_rem = Y[v]\n",
    "            y_rem = y_n[v]\n",
    "            for u in tree.children[v]:\n",
    "                Y[u] = Alloc(y_n[u],a[u],y_rem,Y_rem)\n",
    "                Y_rem = Y_rem - Y[u]\n",
    "                y_rem = y_rem - y_n[u] \n",
    "    \n",
    "    F = []\n",
    "    for v in tree.level_sets[0]:\n",
    "        if (Y[v]>=1):\n",
    "            F.append(v)\n",
    "    return F"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5a31e3b",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Deterministic Rounding of Fotakis et al."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1152c3d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Frac_Cost(dist,y,j,ordered_distances):\n",
    "    cc = 0\n",
    "    rem = 1\n",
    "    for v in ordered_distances[j]:\n",
    "        amount = min(rem,y[v])\n",
    "        rem = rem - amount\n",
    "        cc += dist[j][v]*amount\n",
    "        if (rem==0):\n",
    "            break\n",
    "    return cc"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f2c268d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Round(dist, y, n_fac, ordered_distances):\n",
    "    n = len(dist)\n",
    "    beta = np.zeros(n)\n",
    "    for j in range(n):\n",
    "        beta[j] = Frac_Cost(dist, y, j, ordered_distances)\n",
    "\n",
    "    \n",
    "    order = sorted(range(len(beta)), key=lambda k: beta[k])\n",
    "    \n",
    "\n",
    "    F = []\n",
    "    for v in order:\n",
    "        if (F==[]):\n",
    "            F.append(v)\n",
    "        else:\n",
    "            cost_not_include = min([dist[j][v] for j in F])\n",
    "            if (cost_not_include > 6*n_fac*beta[v]):\n",
    "                F.append(v)\n",
    "        if (len(F)==n_fac):\n",
    "            break\n",
    "    \n",
    "\n",
    "    return F  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "07dcf127",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Augment_Placement(dist,F1,F2):\n",
    "    if (len(F1)==len(F2)):\n",
    "        return F2\n",
    "    else:\n",
    "        matching_matrix = np.array(dist[F1,:][:,F2])\n",
    "        row_ind, col_ind = linear_sum_assignment(matching_matrix)\n",
    "        to_copy = [F1[j] for j in range(len(F1)) if j not in row_ind]\n",
    "        return F2 + to_copy"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "25246282",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Cost Computations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2beee16d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Connection_Cost(dist,F,R):\n",
    "    cc = 0\n",
    "    for j in R:\n",
    "        cc += min([dist[j][i] for i in F])\n",
    "    return cc"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0e1e08f7",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Moving_Cost(dist,F1,F2):\n",
    "    matching_matrix = np.array(dist[F1,:][:,F2])\n",
    "    row_ind, col_ind = linear_sum_assignment(matching_matrix)\n",
    "    return matching_matrix[row_ind, col_ind].sum()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7b24f161",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Average_Cost(cost):\n",
    "    cost_total = np.cumsum(cost)\n",
    "    cost_avg = [cost_total[t]/(t+1) for t in range(len(cost))]\n",
    "    return cost_avg"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5d9460b7",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Follow the Regularized Leader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2ff60269",
   "metadata": {},
   "outputs": [],
   "source": [
    "def MirrorDescent(tree, y, c, n_fac,eta):\n",
    "    \n",
    "    \n",
    "    old_ratio = np.zeros(tree.n_vertices)\n",
    "    for v in range(1,tree.n_vertices):\n",
    "        p = tree.parent[v]\n",
    "        old_ratio[v] = y[v]/y[p]\n",
    "        \n",
    "    upd_ratio = np.zeros(tree.n_vertices)\n",
    "    for v in range(1,tree.n_vertices):\n",
    "        p = tree.parent[v]\n",
    "        upd_ratio[v] = old_ratio[v]*math.e**(-eta*c[v]/tree.weight[v])\n",
    "    \n",
    "    m=0.00000000001\n",
    "    M=1000000000000\n",
    "    for v in range(1,tree.n_vertices):\n",
    "        if (upd_ratio[v]<m):\n",
    "            upd_ratio[v] = m\n",
    "        if (upd_ratio[v]>M):\n",
    "            upd_ratio[v] = M\n",
    "    \n",
    "    \n",
    "    k = np.zeros(tree.n_vertices)\n",
    "    new_ratio = np.zeros(tree.n_vertices)\n",
    "    for l in range(tree.height+1):\n",
    "        for v in tree.level_sets[l]:\n",
    "            if (l==0):\n",
    "                k[v] = 1\n",
    "            else:\n",
    "                k[v] = np.sum([new_ratio[u] for u in tree.children[v]])\n",
    "            new_ratio[v] = upd_ratio[v]*math.sqrt(k[v])\n",
    "                             \n",
    "     \n",
    "    for l in range(tree.height,0,-1):\n",
    "        for v in tree.level_sets[l]:\n",
    "            s = sum([new_ratio[u] for u in tree.children[v]])\n",
    "            for u in tree.children[v]:\n",
    "                new_ratio[u] = new_ratio[u]/s\n",
    "    \n",
    "    global_ratio = np.zeros(tree.n_vertices)\n",
    "    global_ratio[0] = 1\n",
    "    for l in range(tree.height-1,-1,-1):\n",
    "        for v in tree.level_sets[l]:\n",
    "            p = tree.parent[v]\n",
    "            global_ratio[v] = new_ratio[v]*global_ratio[p]\n",
    "    \n",
    "        \n",
    "    y_next = np.zeros(tree.n_vertices)\n",
    "    for v in range(tree.n_vertices):\n",
    "        y_next[v] = np.max(n_fac*global_ratio[v], 0)\n",
    "       \n",
    "    \n",
    "    return y_next"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "26d952ff",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Solve_Fractional_FTRL(tree, r_seq_tree, n_fac, eta):\n",
    "    y_seq = [n_fac*tree.uniform]\n",
    "    total_req = [0]*tree.n_vertices\n",
    "    for t in range(len(r_seq_tree)):\n",
    "  \n",
    "        y_cur = y_seq[t]\n",
    "        \n",
    "        f_grad = np.zeros(tree.n_vertices)\n",
    "        for v in range(tree.n_vertices):\n",
    "            if (y_cur[v] < 1):\n",
    "                f_grad[v] = -2*tree.weight[v]*total_req[v]\n",
    "        \n",
    "        \n",
    "        y_next = MirrorDescent(tree, y_cur, f_grad, n_fac, eta)\n",
    "        y_seq.append(y_next)\n",
    "        \n",
    "        \n",
    "        new_req_subtree = tree.subtree_requests(r_seq_tree[t])\n",
    "        for i in range(tree.n_vertices):\n",
    "            total_req[i] += new_req_subtree[i]\n",
    "        \n",
    "        \n",
    "    return y_seq"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4b684c59",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Fractional Problem"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1ab2f57b",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Get_Duals(dist, y, R, ordered_distances): \n",
    "    n = len(dist)\n",
    "    g = np.zeros(n)\n",
    "    for j in R:\n",
    "        Rem = 1\n",
    "        x = np.zeros(n)\n",
    "        for v in ordered_distances[j]:\n",
    "            x[v] = min(y[v], Rem)\n",
    "            Rem = Rem - x[v]\n",
    "        Vp = [v for v in range(n) if x[v]>0]\n",
    "        D = max([dist[v][j] for v in Vp])\n",
    "        k = np.zeros(n)\n",
    "        for v in Vp:\n",
    "            k[v] = (x[v]/y[v])*(D - dist[v][j])\n",
    "        g = g - k\n",
    "    return g"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "339394dd",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Solve_Fractional(dist, r_seq, n_fac, ordered_distances, epsilon):\n",
    "    D = np.max(dist) \n",
    "    n = len(dist) \n",
    "    T = len(r_seq) \n",
    "    max_req_size = max(len(i) for i in r_seq)\n",
    "    y = [[n_fac/n for i in range(n)]]\n",
    "    \n",
    "    for t in range(T):\n",
    "        g = Get_Duals(dist, y[t], r_seq[t], ordered_distances)\n",
    "        y_new = [y[t][v]*math.e**(-epsilon*g[v]) for v in range(n)]\n",
    "        s = np.sum(y_new)\n",
    "        y.append([n_fac*y_new[v]/s for v in range(n)])\n",
    "        \n",
    "    return y"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "38428467",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Solver for Fotakis et al."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7733ee98",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Solver_Fotakis(dist, r_seq, n_fac, epsilon):\n",
    "    \n",
    "    ordered_distances = []\n",
    "    for p in range(len(dist)):\n",
    "        local = dist[p].copy()\n",
    "        ordered_distances.append(sorted(range(len(local)), key=lambda k: local[k]))\n",
    "        \n",
    "    \n",
    "    y_seq = Solve_Fractional(dist, r_seq, n_fac, ordered_distances, epsilon)\n",
    "    \n",
    "    for y in y_seq:\n",
    "        s = np.sum(y)\n",
    "        y[0] = y[0] + n_fac - s\n",
    "   \n",
    "    for y in y_seq:\n",
    "        s = np.sum(y)\n",
    "        y[0] = y[0] + n_fac - s\n",
    "        \n",
    "        \n",
    "    F_seq = [Round(dist, y, n_fac, ordered_distances) for y in y_seq]\n",
    "    \n",
    "    for F in F_seq:\n",
    "        while (len(F)< n_fac):\n",
    "            F.append(F[0])\n",
    "        \n",
    "    for t in range(1, len(r_seq)+1):\n",
    "        F_seq[t] = Augment_Placement(dist,F_seq[t-1],F_seq[t])        \n",
    "        \n",
    "    frac = []\n",
    "    for t in range(len(r_seq)):\n",
    "        f = 0\n",
    "        for j in r_seq[t]:\n",
    "            f+= Frac_Cost(dist,y_seq[t],j,ordered_distances)\n",
    "        frac.append(f)\n",
    "    \n",
    "    cc = Average_Cost([Connection_Cost(dist, F_seq[t], r_seq[t]) for t in range(len(r_seq))])\n",
    "    mc = Average_Cost([Moving_Cost(dist, F_seq[t], F_seq[t+1]) for t in range(len(r_seq))])\n",
    "    fc = Average_Cost(frac)\n",
    "        \n",
    "    return [F_seq, cc, mc, fc]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7f41503d",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Solver for FTRL"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0026beaf",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Solver_FTRL(dist, r_seq, n_fac, eta):\n",
    "\n",
    "    cld, t2g, g2t = Embed(dist)\n",
    "    tree = Tree(cld)\n",
    "    a = np.random.uniform(0,1,tree.n_vertices)\n",
    "\n",
    "    r_seq_tree = []\n",
    "    for r in r_seq:\n",
    "        new_r = [g2t[v] for v in r]\n",
    "        r_seq_tree.append(new_r)\n",
    "        \n",
    "    y_seq_tree = Solve_Fractional_FTRL(tree, r_seq_tree, n_fac, eta)\n",
    "    \n",
    "    F_seq_tree = [Cut_and_Round(tree, y, a, n_fac) for y in y_seq_tree]\n",
    "    \n",
    "    F_seq = []\n",
    "    for F in F_seq_tree:\n",
    "        F_seq.append([t2g[v] for v in F])\n",
    "    \n",
    "    cc = Average_Cost([Connection_Cost(dist, F_seq[t], r_seq[t]) for t in range(len(r_seq))])\n",
    "    mc = Average_Cost([Moving_Cost(dist, F_seq[t], F_seq[t+1]) for t in range(len(r_seq))])\n",
    "    \n",
    "    return [F_seq, cc, mc] "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "96620482",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Experiment 1: Periodically Moving Clients"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "968eb0c1",
   "metadata": {},
   "outputs": [],
   "source": [
    "llim = -1\n",
    "rlim = 1\n",
    "epsilon = 0.1\n",
    "line_points = int((rlim-llim)/epsilon + 1)\n",
    "n_points = line_points**2\n",
    "vertices = [i for i in range(n_points)]\n",
    "\n",
    "coord = []\n",
    "for i in range(line_points):\n",
    "    for j in range(line_points):\n",
    "        coord.append([round(llim + epsilon*i,1),round(llim+epsilon*j,1)])\n",
    "\n",
    "dist = np.zeros([n_points,n_points])\n",
    "for i in range(n_points):\n",
    "    for j in range(n_points):\n",
    "        dist[i][j] = math.sqrt((coord[i][0]-coord[j][0])**2 + (coord[i][1]-coord[j][1])**2)\n",
    "diam = np.max(dist)\n",
    "\n",
    "def Sample_In_Circle(x_center, y_center, R):\n",
    "    r = R * np.random.rand()\n",
    "    theta = 2 * math.pi * np.random.rand()\n",
    "    x = round(x_center + r*np.cos(theta),1)\n",
    "    y = round(y_center + r*np.sin(theta),1)\n",
    "    ind = int(round(21*(x+1)*10 + (y+1)*10,0))\n",
    "    return ind\n",
    "\n",
    "\n",
    "clique_centers = [66, 80, 360, 374]\n",
    "radius = 0.2\n",
    "n_cycles = 1000\n",
    "T = 4*n_cycles\n",
    "r_seq = []\n",
    "for i in range(n_cycles):\n",
    "    for j in clique_centers:\n",
    "        r_seq.append([Sample_In_Circle(coord[j][0], coord[j][1], radius)])\n",
    "        \n",
    "max_req_size =np.max([len(r) for r in r_seq])\n",
    "n_fac = 3\n",
    "\n",
    "hnd_opt = [66, 80, 360]\n",
    "hindsight_cost = np.zeros(T)\n",
    "for t in range(T):\n",
    "    hindsight_cost[t] = min(dist[r_seq[t][0]][j] for j in hnd_opt)\n",
    "hindsight_total_cost = np.cumsum(hindsight_cost)\n",
    "tc_opt = [hindsight_total_cost[t]/(t+1) for t in range(T)]\n",
    "\n",
    "epsilon = math.sqrt(math.log(n_points))/(np.max(dist)*max_req_size*math.sqrt(T)) \n",
    "[F_seq, cc, mc, fc] = Solver_Fotakis(dist, r_seq, n_fac, epsilon)\n",
    "\n",
    "gamma = 0\n",
    "eta = 1/(math.sqrt(n_points*T)*max(gamma,1))\n",
    "[F_seq0, cc0, mc0] = Solver_FTRL(dist, r_seq, n_fac, eta)\n",
    "\n",
    "\n",
    "gamma = 1\n",
    "eta = 1/(math.sqrt(n_points*T)*max(gamma,1))\n",
    "[F_seq1, cc1, mc1] = Solver_FTRL(dist, r_seq, n_fac, eta)\n",
    "\n",
    "gamma = 10\n",
    "eta = 1/(math.sqrt(n_points*T)*max(gamma,1))\n",
    "[F_seq10, cc10, mc10] = Solver_FTRL(dist, r_seq, n_fac, eta)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0d454921",
   "metadata": {},
   "outputs": [],
   "source": [
    "fig, axs = plt.subplots(nrows=2, ncols=2, figsize=(8, 7))\n",
    "\n",
    "\n",
    "gamma = 0\n",
    "tc_fot = [cc[t] + gamma*mc[t] for t in range(T)]\n",
    "tc_our = [cc0[t] + gamma*mc0[t] for t in range(T)]\n",
    "r_fot_0 = [tc_fot[t]/tc_opt[T-1] - 1 for t in range(T)]\n",
    "r_our_0 = [tc_our[t]/tc_opt[T-1] - 1 for t in range(T)]\n",
    "axs[0, 0].set_title(r' $\\gamma = 0$')\n",
    "axs[0, 0].loglog(r_fot_0, color='blue', label='Fotakis et al.')\n",
    "axs[0, 0].loglog(r_our_0, color='red', label='Our Algorithm')\n",
    "axs[0, 0].legend()\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "gamma = 1\n",
    "gamma = 0\n",
    "tc_fot = [cc[t] + gamma*mc[t] for t in range(T)]\n",
    "tc_our = [cc1[t] + gamma*mc1[t] for t in range(T)]\n",
    "r_fot_1 = [tc_fot[t]/tc_opt[T-1] - 1 for t in range(T)]\n",
    "r_our_1 = [tc_our[t]/tc_opt[T-1] - 1 for t in range(T)]\n",
    "axs[0, 1].set_title(r' $\\gamma = 1$')\n",
    "axs[0, 1].loglog(r_fot_1, color='blue', label='Fotakis et al.')\n",
    "axs[0, 1].loglog(r_our_1, color='red', label='Our Algorithm')\n",
    "axs[0, 1].legend()\n",
    "\n",
    "\n",
    "\n",
    "gamma = 10\n",
    "tc_fot = [cc[t] + gamma*mc[t] for t in range(T)]\n",
    "tc_our = [cc10[t] + gamma*mc10[t] for t in range(T)]\n",
    "r_fot_10 = [tc_fot[t]/tc_opt[T-1] - 1 for t in range(T)]\n",
    "r_our_10 = [tc_our[t]/tc_opt[T-1] - 1 for t in range(T)]\n",
    "axs[1, 0].set_title(r'$\\gamma = 10$')\n",
    "axs[1, 0].loglog(r_fot_10, color='blue', label='Fotakis et al.')\n",
    "axs[1, 0].loglog(r_our_10, color='red', label='Our Algorithm')\n",
    "axs[1, 0].legend()\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "axs[1, 1].set_title(\"Facility Placement\")\n",
    "axs[1, 1].add_patch(Rectangle((-1,-1),2,2,\n",
    "                    edgecolor='lightcyan',\n",
    "                    facecolor='lightcyan',\n",
    "                    fill = True,\n",
    "                    lw=1,\n",
    "                    zorder = -1))\n",
    "axs[1, 1].set_ylim(-1.1,1.1)\n",
    "axs[1, 1].set_xlim(-1.1,1.1)\n",
    "for i in clique_centers:\n",
    "    axs[1, 1].add_patch(Circle((coord[i][0],coord[i][1]), radius,\n",
    "                    edgecolor='red',\n",
    "                    facecolor='red',\n",
    "                    fill = True,\n",
    "                    lw=1,\n",
    "                    alpha = 0.2,\n",
    "                    zorder = -1))\n",
    "for i in F_seq1[T]:\n",
    "    axs[1, 1].scatter(coord[i][0],coord[i][1],color='green', zorder=1)\n",
    "\n",
    "fig.tight_layout()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "23841e5f",
   "metadata": {},
   "outputs": [],
   "source": [
    "fig, axs = plt.subplots(nrows=2, ncols=2, figsize=(8, 7))\n",
    "\n",
    "\n",
    "gamma = 0\n",
    "tc_fot = [cc[t] + gamma*mc[t] for t in range(T)]\n",
    "tc_our = [cc0[t] + gamma*mc0[t] for t in range(T)]\n",
    "r_fot_0 = [tc_fot[t]/tc_opt[T-1]  for t in range(T)]\n",
    "r_our_0 = [tc_our[t]/tc_opt[T-1]  for t in range(T)]\n",
    "axs[0, 0].set_title(r' $\\gamma = 0$')\n",
    "axs[0, 0].plot(r_fot_0, color='blue', label='Fotakis et al.')\n",
    "axs[0, 0].plot(r_our_0, color='red', label='Our Algorithm')\n",
    "axs[0, 0].legend()\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "gamma = 1\n",
    "gamma = 0\n",
    "tc_fot = [cc[t] + gamma*mc[t] for t in range(T)]\n",
    "tc_our = [cc1[t] + gamma*mc1[t] for t in range(T)]\n",
    "r_fot_1 = [tc_fot[t]/tc_opt[T-1]  for t in range(T)]\n",
    "r_our_1 = [tc_our[t]/tc_opt[T-1]  for t in range(T)]\n",
    "axs[0, 1].set_title(r' $\\gamma = 1$')\n",
    "axs[0, 1].plot(r_fot_1, color='blue', label='Fotakis et al.')\n",
    "axs[0, 1].plot(r_our_1, color='red', label='Our Algorithm')\n",
    "axs[0, 1].legend()\n",
    "\n",
    "\n",
    "\n",
    "gamma = 10\n",
    "tc_fot = [cc[t] + gamma*mc[t] for t in range(T)]\n",
    "tc_our = [cc10[t] + gamma*mc10[t] for t in range(T)]\n",
    "r_fot_10 = [tc_fot[t]/tc_opt[T-1]  for t in range(T)]\n",
    "r_our_10 = [tc_our[t]/tc_opt[T-1]  for t in range(T)]\n",
    "axs[1, 0].set_title(r'$\\gamma = 10$')\n",
    "axs[1, 0].plot(r_fot_10, color='blue', label='Fotakis et al.')\n",
    "axs[1, 0].plot(r_our_10, color='red', label='Our Algorithm')\n",
    "axs[1, 0].legend()\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "axs[1, 1].set_title(\"Facility Placement\")\n",
    "axs[1, 1].add_patch(Rectangle((-1,-1),2,2,\n",
    "                    edgecolor='lightcyan',\n",
    "                    facecolor='lightcyan',\n",
    "                    fill = True,\n",
    "                    lw=1,\n",
    "                    zorder = -1))\n",
    "axs[1, 1].set_ylim(-1.1,1.1)\n",
    "axs[1, 1].set_xlim(-1.1,1.1)\n",
    "for i in clique_centers:\n",
    "    axs[1, 1].add_patch(Circle((coord[i][0],coord[i][1]), radius,\n",
    "                    edgecolor='red',\n",
    "                    facecolor='red',\n",
    "                    fill = True,\n",
    "                    lw=1,\n",
    "                    alpha = 0.2,\n",
    "                    zorder = -1))\n",
    "for i in F_seq1[T]:\n",
    "    axs[1, 1].scatter(coord[i][0],coord[i][1],color='green', zorder=1)\n",
    "\n",
    "fig.tight_layout()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "07256a85",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Experiment 2: MNIST Dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "79421d84",
   "metadata": {},
   "outputs": [],
   "source": [
    "(x_train, y_train), (x_test, y_test) = mnist.load_data()\n",
    "n = 10000\n",
    "ds = np.random.randint(0, x_train.shape[0], n)\n",
    "\n",
    "w = np.zeros([n,n])\n",
    "for i in range(n):\n",
    "    for j in range(n):\n",
    "        if (i<j):\n",
    "            w[i][j] = np.linalg.norm(x_train[ds[i]] - x_train[ds[j]])\n",
    "        else: \n",
    "            w[i][j] = w[j][i]\n",
    "            \n",
    "\n",
    "dst = list(map(lambda p: list(map(lambda q: q, p)), w))\n",
    "for r in range(n):\n",
    "    for p in range(n):\n",
    "        for q in range(n):\n",
    "            dst[p][q] = min(dst[p][q], dst[p][r] + dst[r][q])\n",
    "dist = np.asarray(dst)\n",
    "D = np.max(dist)\n",
    "T = 3000\n",
    "r_seq = [[np.random.randint(0,n) for i in range(1)] for i in range(T)]\n",
    "r_max = np.max([len(r) for r in r_seq])\n",
    "n_fac = 10\n",
    "\n",
    "\n",
    "epsilon = math.sqrt(math.log(n_points))/(np.max(dist)*max_req_size*math.sqrt(T)) \n",
    "[F_seq, cc, mc, fc] = Solver_Fotakis(dist, r_seq, n_fac, epsilon)\n",
    "\n",
    "gamma = 0\n",
    "eta = 1/(math.sqrt(n_points*T)*max(gamma,1))\n",
    "[F_seq0, cc0, mc0] = Solver_FTRL(dist, r_seq, n_fac, eta)\n",
    "fot_ratio = (cc[T-1] + gamma*mc[T-1])/fc[T-1]\n",
    "our_ratio = (cc0[T-1] + gamma*mc0[T-1])/fc[T-1]\n",
    "print(\"g = 0 ----- Fotakis et al: \" + str(fot_ratio))\n",
    "print(\"g = 0 ----- FTRL: \" + str(our_ratio))\n",
    "\n",
    "gamma = 1\n",
    "eta = 1/(math.sqrt(n_points*T)*max(gamma,1))\n",
    "[F_seq1, cc1, mc1] = Solver_FTRL(dist, r_seq, n_fac, eta)\n",
    "fot_ratio = (cc[T-1] + gamma*mc[T-1])/fc[T-1]\n",
    "our_ratio = (cc1[T-1] + gamma*mc1[T-1])/fc[T-1]\n",
    "print(\"g = 1 ----- Fotakis et al: \" + str(fot_ratio))\n",
    "print(\"g = 1 ----- FTRL: \" + str(our_ratio))\n",
    "\n",
    "gamma = 10\n",
    "eta = 1/(math.sqrt(n_points*T)*max(gamma,1))\n",
    "[F_seq10, cc10, mc10] = Solver_FTRL(dist, r_seq, n_fac, eta)\n",
    "fot_ratio = (cc[T-1] + gamma*mc[T-1])/fc[T-1]\n",
    "our_ratio = (cc10[T-1] + gamma*mc10[T-1])/fc[T-1]\n",
    "print(\"g = 10 ---- Fotakis et al: \" + str(fot_ratio))\n",
    "print(\"g = 10 ---- FTRL: \" + str(our_ratio))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "41519e0f",
   "metadata": {},
   "source": [
    "# \n",
    "# \n",
    "# Experiment 3: CIFAR10 Dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "89c2295a",
   "metadata": {},
   "outputs": [],
   "source": [
    "(x_train, y_train), (x_test, y_test) = cifar10.load_data()\n",
    "n = 10000\n",
    "ds = np.random.randint(0, x_train.shape[0], n)\n",
    "\n",
    "w = np.zeros([n,n])\n",
    "for i in range(n):\n",
    "    for j in range(n):\n",
    "        if (i<j):\n",
    "            w[i][j] = np.linalg.norm(x_train[ds[i]] - x_train[ds[j]])\n",
    "        else: \n",
    "            w[i][j] = w[j][i]\n",
    "            \n",
    "\n",
    "dst = list(map(lambda p: list(map(lambda q: q, p)), w))\n",
    "for r in range(n):\n",
    "    for p in range(n):\n",
    "        for q in range(n):\n",
    "            dst[p][q] = min(dst[p][q], dst[p][r] + dst[r][q])\n",
    "dist = np.asarray(dst)\n",
    "D = np.max(dist)\n",
    "T = 3000\n",
    "r_seq = [[np.random.randint(0,n) for i in range(1)] for i in range(T)]\n",
    "r_max = np.max([len(r) for r in r_seq])\n",
    "n_fac = 10\n",
    "\n",
    "\n",
    "epsilon = math.sqrt(math.log(n_points))/(np.max(dist)*max_req_size*math.sqrt(T)) \n",
    "[F_seq, cc, mc, fc] = Solver_Fotakis(dist, r_seq, n_fac, epsilon)\n",
    "\n",
    "gamma = 0\n",
    "eta = 1/(math.sqrt(n_points*T)*max(gamma,1))\n",
    "[F_seq0, cc0, mc0] = Solver_FTRL(dist, r_seq, n_fac, eta)\n",
    "fot_ratio = (cc[T-1] + gamma*mc[T-1])/fc[T-1]\n",
    "our_ratio = (cc0[T-1] + gamma*mc0[T-1])/fc[T-1]\n",
    "print(\"g = 0 ----- Fotakis et al: \" + str(fot_ratio))\n",
    "print(\"g = 0 ----- FTRL: \" + str(our_ratio))\n",
    "\n",
    "gamma = 1\n",
    "eta = 1/(math.sqrt(n_points*T)*max(gamma,1))\n",
    "[F_seq1, cc1, mc1] = Solver_FTRL(dist, r_seq, n_fac, eta)\n",
    "fot_ratio = (cc[T-1] + gamma*mc[T-1])/fc[T-1]\n",
    "our_ratio = (cc1[T-1] + gamma*mc1[T-1])/fc[T-1]\n",
    "print(\"g = 1 ----- Fotakis et al: \" + str(fot_ratio))\n",
    "print(\"g = 1 ----- FTRL: \" + str(our_ratio))\n",
    "\n",
    "gamma = 10\n",
    "eta = 1/(math.sqrt(n_points*T)*max(gamma,1))\n",
    "[F_seq10, cc10, mc10] = Solver_FTRL(dist, r_seq, n_fac, eta)\n",
    "fot_ratio = (cc[T-1] + gamma*mc[T-1])/fc[T-1]\n",
    "our_ratio = (cc10[T-1] + gamma*mc10[T-1])/fc[T-1]\n",
    "print(\"g = 10 ---- Fotakis et al: \" + str(fot_ratio))\n",
    "print(\"g = 10 ---- FTRL: \" + str(our_ratio))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.8.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
