{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 118,
   "metadata": {
    "id": "VVtKzfOWJc7Z"
   },
   "outputs": [],
   "source": [
    "import cvxpy as cp\n",
    "import numpy as np\n",
    "import numpy as np\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "metadata": {
    "id": "KqYpAuqHZyfd"
   },
   "outputs": [],
   "source": [
    "def simplex_projection(s):\n",
    "    \"\"\"Projection onto the unit simplex.\"\"\"\n",
    "    if np.sum(s) <=1 and np.alltrue(s >= 0):\n",
    "        return s\n",
    "    u = np.sort(s)[::-1]\n",
    "    cssv = np.cumsum(u)\n",
    "    # get the number of > 0 components of the optimal solution\n",
    "    rho = np.nonzero(u * np.arange(1, len(u)+1) > (cssv - 1))[0][-1]\n",
    "    # compute the Lagrange multiplier associated to the simplex constraint\n",
    "    theta = (cssv[rho] - 1) / (rho + 1.0)\n",
    "    # compute the projection by thresholding v using theta\n",
    "    return np.maximum(s-theta, 0)\n",
    "\n",
    "def nuclear_projection(A):\n",
    "    \"\"\"Projection onto nuclear norm ball.\"\"\"\n",
    "    U, s, V = np.linalg.svd(A, full_matrices=False)\n",
    "    s = simplex_projection(s)\n",
    "    return U.dot(np.diag(s).dot(V))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 120,
   "metadata": {
    "id": "rNNqMg8iwk-k"
   },
   "outputs": [],
   "source": [
    "class squared_SIM_fn():\n",
    "\n",
    "    def __init__(self, B_star):\n",
    "        self.B_star = B_star\n",
    "\n",
    "    def fn_value(self, B):\n",
    "        return None\n",
    "\n",
    "    def grad(self, B):\n",
    "        return None\n",
    "    \n",
    "    def stoc_fn_value(self, B):\n",
    "        A = np.random.normal(scale=0.3, size=self.B_star.shape) + np.identity(self.B_star.shape[0])\n",
    "        noise = np.random.normal(scale=0.1, size=1)\n",
    "        y = np.trace(A.T @ self.B_star)**2 + noise\n",
    "        fn_value = (y -  np.trace(A.T @ B)**2)**2\n",
    "        return fn_value\n",
    "\n",
    "    def stoc_grad(self, B):\n",
    "        A = np.random.normal(scale=0.3, size=self.B_star.shape) + np.identity(self.B_star.shape[0])\n",
    "        noise = np.random.normal(scale=0.1, size=1)\n",
    "        y = np.trace(A.T @ self.B_star)**2 + noise\n",
    "        return 4*(np.trace(A.T @ B)**2 - y)*A\n",
    "\n",
    "    def stoc_grad_2pts(self, B1, B2):\n",
    "        A = np.random.normal(scale=0.3, size=self.B_star.shape) + np.identity(self.B_star.shape[0])\n",
    "        noise = np.random.normal(scale=0.1, size=1)\n",
    "        y = np.trace(A.T @ self.B_star)**2 + noise\n",
    "        return 4*(np.trace(A.T @ B1)**2 - y)*A, 4*(np.trace(A.T @ B2)**2 - y)*A\n",
    "\n",
    "    def grad_mapping(self, B):\n",
    "        grad = np.mean([self.stoc_grad(B) for _ in range(300)], axis=0)\n",
    "        ret = np.linalg.norm(nuclear_projection(B-grad) - B, 'fro')\n",
    "        return ret\n",
    "\n",
    "    def FW_gap(self, B):\n",
    "        grad = np.mean([self.stoc_grad(B) for _ in range(300)], axis=0)\n",
    "        v = cp.Variable(B.shape)\n",
    "        prob = cp.Problem(cp.Minimize( cp.trace(grad.T @ v-B)), [cp.norm(v, \"nuc\")<=1])\n",
    "        prob.solve()\n",
    "        v = v.value\n",
    "        return - np.trace(grad.T @ v-B)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "metadata": {
    "id": "wgjFcHXD3x21"
   },
   "outputs": [],
   "source": [
    "class CG_ASA:\n",
    "    \n",
    "    def __init__(self, fn, x0, beta, N=100, s=1):\n",
    "        \"\"\"\n",
    "        args:\n",
    "            fn: a fun object \n",
    "            x0: np.array\n",
    "        \"\"\"\n",
    "        self.sfo, self.lmo = 0, 0\n",
    "        self.fn = fn\n",
    "        self.x = [x0]\n",
    "        self.dim = x0.shape\n",
    "        self.beta = beta\n",
    "\n",
    "        self.z = [fn.stoc_grad(x0)]\n",
    "        self.c = 1 # constant \n",
    "        self.N = N # total number of iterations\n",
    "        self.k = 0 # step\n",
    "\n",
    "        self.s = s # nuclear norm constraint\n",
    "\n",
    "    def ICG_for_nuc_ball(self, x, z, beta, num_iter):\n",
    "        if np.linalg.norm(x-z/beta,'nuc')<=self.s:\n",
    "            return x-z/beta\n",
    "        y=x\n",
    "        for _ in range(num_iter):\n",
    "            v = cp.Variable(self.dim)\n",
    "            prob = cp.Problem(cp.Minimize(cp.trace(v.T @ z+beta*(y-x))), [cp.norm(v, \"nuc\")<=self.s])\n",
    "            prob.solve()\n",
    "            v = v.value\n",
    "            if np.linalg.norm(v-y, 'fro') < 1e-4: # for numeric issues\n",
    "                return y\n",
    "            else:\n",
    "                mu = min(1, np.trace((beta*(x-y) - z).T @ (v-y)) / beta / np.linalg.norm(v-y, 'fro')**2)\n",
    "            y = (1-mu)*y + mu*v\n",
    "        return y\n",
    "\n",
    "    def update(self):\n",
    "        \n",
    "        y = self.ICG_for_nuc_ball(self.x[-1], self.z[-1], self.beta, int(np.sqrt(self.k)))\n",
    "\n",
    "        x_next = (1-self.c/np.sqrt(self.N)) * self.x[-1] + self.c/np.sqrt(self.N) * y\n",
    "        z_next = (1-self.c/np.sqrt(self.N)) * self.z[-1] + self.c/np.sqrt(self.N) * self.fn.stoc_grad(self.x[-1])\n",
    "\n",
    "        self.z.append(z_next)\n",
    "        self.x.append(x_next)\n",
    "\n",
    "        self.k += 1\n",
    "        self.sfo += 1\n",
    "        self.lmo += int(np.sqrt(self.k))\n",
    "\n",
    "    def run(self):\n",
    "        for _ in range(self.N):\n",
    "            self.update()\n",
    "        "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 122,
   "metadata": {
    "id": "-84szqo4-uXr"
   },
   "outputs": [],
   "source": [
    "class One_SFW:\n",
    "    \n",
    "    def __init__(self, fn, x0, N=100, s=1):\n",
    "        \"\"\"\n",
    "        args:\n",
    "            fn: a fun object \n",
    "            x0: np.array\n",
    "        \"\"\"\n",
    "        self.sfo, self.lmo = 0, 0\n",
    "        self.fn = fn\n",
    "        self.x = [x0]\n",
    "        self.d = [fn.stoc_grad(x0)]\n",
    "        self.dim = x0.shape\n",
    "\n",
    "        self.eta = N**(-2/3)\n",
    "        self.rho = lambda k: (k-1)**(-2/3)\n",
    "        self.N = N\n",
    "        self.k = 0\n",
    "        self.s = s\n",
    "\n",
    "    def update(self):\n",
    "        self.k += 1\n",
    "        if self.k > 1:\n",
    "            Grad1, Grad2 = self.fn.stoc_grad_2pts(self.x[-1], self.x[-2])\n",
    "            d_next = (1-self.rho(self.k)) * (self.d[-1] + Grad1 - Grad2) + self.rho(self.k) * Grad1\n",
    "        else:\n",
    "            d_next = self.fn.stoc_grad(self.x[-1])\n",
    "\n",
    "        v = cp.Variable(self.dim)\n",
    "        prob = cp.Problem(cp.Minimize( cp.trace(v.T @ d_next)), [cp.norm(v, \"nuc\")<=self.s])\n",
    "        prob.solve()\n",
    "        v = v.value\n",
    "\n",
    "        x_next = (1-self.eta) * self.x[-1] + self.eta * v\n",
    "\n",
    "        self.x.append(x_next)\n",
    "        self.d.append(d_next)\n",
    "        \n",
    "        self.sfo += 1\n",
    "        self.lmo += int(np.sqrt(self.k))\n",
    "\n",
    "    def run(self):\n",
    "        for _ in range(self.N):\n",
    "            self.update()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 125,
   "metadata": {
    "id": "gxIoXdv9tol2"
   },
   "outputs": [],
   "source": [
    "v = np.random.normal(size=(4,1)) \n",
    "B_star = v @ v.T\n",
    "B_star = B_star / np.linalg.norm(B_star, 'nuc')\n",
    "fn = squared_SIM_fn(B_star)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 128,
   "metadata": {
    "id": "30qe6D6_HWK0"
   },
   "outputs": [],
   "source": [
    "alg1 = CG_ASA(fn, x0 = np.zeros((4,4)), beta=1, N=500, s=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 129,
   "metadata": {
    "id": "NfD4Ao68H7L4"
   },
   "outputs": [],
   "source": [
    "alg1.run()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 131,
   "metadata": {
    "id": "4zdsszzoQTmF"
   },
   "outputs": [],
   "source": [
    "alg2 = One_SFW(fn, x0 = np.zeros((4,4)), N=500, s=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 132,
   "metadata": {
    "id": "NwhROjFvQZjU"
   },
   "outputs": [],
   "source": [
    "alg2.run()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 133,
   "metadata": {
    "id": "BwhcVVXZ-BnQ"
   },
   "outputs": [],
   "source": [
    "GM1 = [fn.grad_mapping(x) for x in alg1.x]\n",
    "GM2 = [fn.grad_mapping(x) for x in alg2.x]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 135,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "jye3aKlGYGX3",
    "outputId": "856978a2-f07a-4c87-8732-85785540d7b4"
   },
   "outputs": [
    {
     "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(np.arange(501), GM1, label='ASA+ICG')\n",
    "plt.plot(np.arange(501), GM2, label='1-SFW')\n",
    "plt.xlabel('number of iterations')\n",
    "plt.ylabel('gradient mapping')\n",
    "plt.legend()\n",
    "plt.savefig('output.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "colab": {
   "collapsed_sections": [],
   "name": "SFW.ipynb",
   "provenance": []
  },
  "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.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
