{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "589151ad-1aab-4611-b3cd-d9070466d4e9",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "from numba import njit\n",
    "import scipy.sparse as sp"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "ab49f4b8-e4b2-4d50-867a-5391947bea6f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "import numpy as np\n",
    "from numpy import bool_\n",
    "from numpy import sqrt\n",
    "from numpy import int64\n",
    "from numpy import float64\n",
    "from numba import njit\n",
    "from numba import objmode\n",
    "from numpy.linalg import norm\n",
    "@njit(cache=True)\n",
    "def local_appr(n, indptr, indices, degree, s, alpha, eps, opt_x=None):\n",
    "    with objmode(start='f8'):\n",
    "        start = time.perf_counter()\n",
    "    # --- initialization ---\n",
    "    xt = np.zeros(n, dtype=float64)\n",
    "    rt = np.zeros(n, dtype=float64)\n",
    "    rt[:] = s\n",
    "    eps_vec = eps * degree\n",
    "    # ----------------------\n",
    "    # queue data structure\n",
    "    st = np.nonzero(s)[0]\n",
    "    front = int64(0)\n",
    "    queue = np.zeros(n + 1, dtype=int64)\n",
    "    queue[:len(st)] = st\n",
    "    q_mark = np.zeros(n + 1, dtype=bool_)\n",
    "    q_mark[st] = True\n",
    "    rear = len(st)\n",
    "    queue[rear] = n  # iteration flag\n",
    "    q_mark[n] = True\n",
    "    rear += 1\n",
    "\n",
    "    # results\n",
    "    errs = []\n",
    "    opers = []\n",
    "    cd_xt = []\n",
    "    cd_rt = []\n",
    "    vol_st = []\n",
    "    vol_it = []\n",
    "    gamma_t = []\n",
    "    op_time = np.float64(0.)\n",
    "    oper = 0.\n",
    "    with objmode(debug_start='f8'):\n",
    "        debug_start = time.perf_counter()\n",
    "    gamma_pre = np.linalg.norm(s, 1)\n",
    "    gamma = 0.\n",
    "    with objmode(op_time='f8'):\n",
    "        op_time += (time.perf_counter() - debug_start)\n",
    "\n",
    "    while True:\n",
    "        u = queue[front]\n",
    "        q_mark[u] = False\n",
    "        front = (front + 1) % n\n",
    "        if u == n:  # one local iteration\n",
    "            # ------ debug time ------\n",
    "            with objmode(debug_start='f8'):\n",
    "                debug_start = time.perf_counter()\n",
    "            if opt_x is not None:\n",
    "                errs.append(norm(xt - opt_x, 1))\n",
    "            else:\n",
    "                errs.append(np.infty)  # fakes\n",
    "            opers.append(oper)\n",
    "            cd_xt.append(np.count_nonzero(xt))\n",
    "            cd_rt.append(np.count_nonzero(rt))\n",
    "            vol_st.append(oper)\n",
    "            vol_it.append(np.sum(degree[np.nonzero(rt)]))\n",
    "            gamma_t.append(gamma / gamma_pre)\n",
    "            oper = 0.\n",
    "            gamma = 0.\n",
    "            gamma_pre = np.linalg.norm(rt, 1)\n",
    "            with objmode(op_time='f8'):\n",
    "                op_time += (time.perf_counter() - debug_start)\n",
    "            # ------------------------\n",
    "\n",
    "            queue[rear] = n\n",
    "            rear = (rear + 1) % n\n",
    "            continue\n",
    "\n",
    "        oper += degree[u]\n",
    "        gamma += np.abs(rt[u])\n",
    "\n",
    "        delta = .5 * (1. - alpha) * rt[u]\n",
    "        xt[u] += alpha * rt[u]\n",
    "        rt[u] = delta\n",
    "        for v in indices[indptr[u]:indptr[u + 1]]:\n",
    "            rt[v] += delta / degree[u]\n",
    "            if not q_mark[v] and eps_vec[v] <= rt[v]:\n",
    "                queue[rear] = v\n",
    "                q_mark[v] = True\n",
    "                rear = (rear + 1) % n\n",
    "        # only iteration flag left, quit\n",
    "        if (rear - front) == 1:\n",
    "            if len(errs) != 0:\n",
    "                break\n",
    "            # ------ debug time ------\n",
    "            with objmode(debug_start='f8'):\n",
    "                debug_start = time.perf_counter()\n",
    "            if opt_x is not None:\n",
    "                errs.append(norm(xt - opt_x, 1))\n",
    "            else:\n",
    "                errs.append(np.infty)  # fakes\n",
    "            opers.append(oper)\n",
    "            cd_xt.append(np.count_nonzero(xt))\n",
    "            cd_rt.append(np.count_nonzero(rt))\n",
    "            vol_st.append(oper)\n",
    "            vol_it.append(np.sum(degree[np.nonzero(rt)]))\n",
    "            gamma_t.append(gamma / gamma_pre)\n",
    "            with objmode(op_time='f8'):\n",
    "                op_time += (time.perf_counter() - debug_start)\n",
    "            break\n",
    "            # ------------------------\n",
    "    with objmode(run_time='f8'):\n",
    "        run_time = time.perf_counter() - start\n",
    "    return xt, rt, errs, opers, cd_xt, cd_rt, vol_st, vol_it, gamma_t, run_time, op_time\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "80e68322-bbc8-4914-8e89-1fd278134b27",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "@njit(cache=True)\n",
    "def grad_gt(indptr, indices, sqrt_deg, s, xt, st, st_count, alpha, rho):\n",
    "    grad_xt = alpha * (rho * sqrt_deg - s / sqrt_deg)\n",
    "    for i in np.arange(st_count):\n",
    "        ii = st[i]\n",
    "        grad_xt[ii] += .5 * (1. + alpha) * xt[ii]\n",
    "        for v in indices[indptr[ii]:indptr[ii + 1]]:\n",
    "            demon = sqrt_deg[v] * sqrt_deg[ii]\n",
    "            grad_xt[v] -= .5 * (1. - alpha) * xt[ii] / demon\n",
    "    return grad_xt\n",
    "\n",
    "\n",
    "\n",
    "@njit(cache=True)\n",
    "def apgd(indptr, indices, sqrt_deg, alpha, st, st_count, x0, t, s, rho):\n",
    "    at_pre = 0.\n",
    "    at = 1.\n",
    "    yt = np.copy(x0)\n",
    "    zt = np.copy(x0)\n",
    "    kappa = 1. / alpha\n",
    "    for _ in np.arange(t):\n",
    "        at_next = at_pre + at\n",
    "        xt = (at_pre / at_next) * yt + (at / at_next) * zt\n",
    "        coeff_1 = (kappa - 1. + at_pre) / (kappa - 1. + at_next)\n",
    "        coeff_2 = at / (kappa - 1. + at_next)\n",
    "        grad_xt = grad_gt(indptr, indices, sqrt_deg, s, xt, st, st_count, alpha, rho)\n",
    "        tmp_zt = coeff_1 * zt + coeff_2 * (xt - grad_xt / alpha)\n",
    "        zt = np.zeros(len(xt))\n",
    "        for j in np.arange(st_count):\n",
    "            if tmp_zt[st[j]] > 0:\n",
    "                zt[st[j]] = tmp_zt[st[j]]\n",
    "        yt = (at_pre / at_next) * yt + (at / at_next) * zt\n",
    "        at = at_next * (2. * kappa / (2. * kappa + 1. - np.sqrt(1. + 4 * kappa)) - 1.)\n",
    "        at_pre = at_next\n",
    "    return yt\n",
    "\n",
    "\n",
    "\n",
    "@njit(cache=True)\n",
    "def var_appr_aspr(n, indptr, indices, degree, s, alpha, eps, rho, opt_x):\n",
    "    xt = np.zeros(n, dtype=np.float64)\n",
    "    st = np.zeros(n, dtype=np.int64)\n",
    "    st_flag = np.zeros(n, dtype=np.bool_)\n",
    "    sqrt_deg = np.sqrt(degree)\n",
    "    st_count = 0\n",
    "    new_counts = 0\n",
    "    for i in np.arange(n):\n",
    "        if s[i] > rho * degree[i]:\n",
    "            st[st_count] = i\n",
    "            st_flag[i] = True\n",
    "            st_count += 1\n",
    "            new_counts += 1\n",
    "    l1_error = []\n",
    "    nonzero_list = []\n",
    "    st_list = []\n",
    "    while new_counts != 0:\n",
    "        grad_xt = grad_gt(indptr, indices, sqrt_deg, s, xt, st, st_count, alpha, rho)\n",
    "        delta_t = np.sqrt((eps * alpha) / (1. + st_count))\n",
    "        eps_t_hat = (alpha * (delta_t ** 2.)) / 2.\n",
    "        num = (1. - alpha) * np.sum(grad_xt[st[:st_count]] ** 2.)\n",
    "        dem = 2. * eps_t_hat * (alpha ** 2.)\n",
    "        t = 1. + np.ceil(2. * np.sqrt(1. / alpha) * np.log(num / dem))\n",
    "        xt_bar = apgd(indptr, indices, sqrt_deg, alpha, st, st_count, xt, t, s, rho)\n",
    "        for i in np.arange(st_count):\n",
    "            node = st[i]\n",
    "            xt[node] = np.max(np.array([np.float64(0.), np.float64(xt_bar[node] - delta_t)]))\n",
    "        grad_xt = grad_gt(indptr, indices, sqrt_deg, s, xt, st, st_count, alpha, rho)\n",
    "        st_count_old = st_count\n",
    "        for i in np.arange(n):\n",
    "            if grad_xt[i] < 0. and not st_flag[i]:\n",
    "                st[st_count] = i\n",
    "                st_count += 1\n",
    "                st_flag[i] = True\n",
    "        new_counts = st_count - st_count_old\n",
    "        if opt_x is not None:\n",
    "            nonzero_list.append(np.count_nonzero(xt))\n",
    "            l1_error.append(np.linalg.norm(xt * sqrt_deg - opt_x, 1))\n",
    "            st_list.append((t + 1.) * np.sum(degree[st[:st_count]]))\n",
    "    return sqrt_deg * xt, l1_error, nonzero_list, st_list\n"
   ]
  },
  {
   "cell_type": "code",
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.9999952720403966\n",
      "0.9589211207215025\n",
      "0.9755076596470342\n"
     ]
    },
    {
     "data": {
      "image/png": "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S7M/DhsRERE5RLua82I0GrFp0yb07dsXo0aNQkhICBISErB+/foW96mpqUFFRUWTl1OYFqnTVzvn+ERERB1EuwovpaWluHTpEt544w2MHj0amzdvxoQJE3DfffchOzvb4j7p6enQaDTmV2RkpIOrbsBhIyIiIoeQHF4WLFgAQRBafeXl5VlVjNFoBACMHz8es2bNwo033og5c+Zg7NixWLp0qcV95s6dC51OZ34VFRVZdWybcdiIiIjIISTPeUlLS8OUKVNabRMdHW1VMcHBwfDw8ED//v2bbO/Xrx9+/fVXi/uo1Wqo1WqrjmdXnn71/9Zecm4dREREbk5yeAkODkZwcLActcDT0xPDhg1Dfn5+k+1Hjx5FVFSULMe0m54jgAkfAZ17OLsSIiIitybr3UaFhYW4ePEiCgsLYTAYoNVqAQC9e/eGn199T0VsbCzS09MxYcIEAMDs2bPxwAMP4JZbbsFtt92GrKwsfPPNN9i2bZucpdouuE/9i4iIiGQla3iZP38+li1bZv558ODBAICtW7dixIgRAID8/HzodDpzmwkTJmDp0qVIT0/HM888g5iYGKxZswbDhw+Xs1QiIiJyEYIoiqKzi7CniooKaDQa6HQ6BAQEOLscIiIiagMpf7/b1a3SRERERNfD8EJEREQuheGFiIiIXArDCxEREbkUhhciIiJyKQwvRERE5FIYXoiIiMilMLwQERGRS2F4ISIiIpfC8EJEREQuheGFiIiIXIqsD2Z0BtOjmioqKpxcCREREbWV6e92Wx656HbhpbKyEgAQGRnp5EqIiIhIqsrKSmg0mlbbuN1TpY1GI86cOQN/f38IgmDX766oqEBkZCSKioo61BOred4d57w74jkDPG+et/tzhXMWRRGVlZWIiIiAQtH6rBa363lRKBTo1q2brMcICAhotxdfTjzvjqMjnjPA8+5oOuJ5t/dzvl6Piwkn7BIREZFLYXghIiIil8LwIoFarcZrr70GtVrt7FIciufdcc67I54zwPPmebs/dztnt5uwS0RERO6NPS9ERETkUhheiIiIyKUwvBAREZFLYXghIiIil8Lwco0PP/wQPXr0gJeXF4YMGYJffvml1fbZ2dkYMmQIvLy80LNnTyxdutRBldqXlPPetm0bBEFo9jpy5IgDK7bNzz//jHvuuQcREREQBAHr16+/7j7ucK2lnrc7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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "adj_m = sp.load_npz('XXX/datasets/com-dblp/com-dblp_csr-mat.npz')\n",
    "degree = np.int32(adj_m.sum(1).A.flatten())\n",
    "indices = adj_m.indices\n",
    "indptr = adj_m.indptr\n",
    "n = len(degree)\n",
    "alpha = 0.15\n",
    "eps = 0.1 / n\n",
    "seed_node = 0\n",
    "s = np.zeros(n)\n",
    "s[seed_node] = 1.\n",
    "re_ = local_appr(n, indptr, indices, degree, s, alpha, eps=1e-6 / n, opt_x=None)\n",
    "opt_xt, rt, errs, opers, cd_xt, cd_rt, vol_st, vol_it, gamma_t, run_time, op_time = re_\n",
    "print(np.linalg.norm(opt_xt, 1))\n",
    "\n",
    "re_aspr = var_appr_aspr(n, indptr, indices, degree, s, alpha, eps=eps * 1e-1, rho=eps, opt_x=opt_xt)\n",
    "xt, l1_error, nonzero_list, st_list = re_aspr\n",
    "print(np.linalg.norm(xt, 1))\n",
    "plt.plot(np.cumsum(st_list), np.log10(l1_error), label=\"ASPR\")\n",
    "\n",
    "re_ = local_appr(n, indptr, indices, degree, s, alpha, eps, opt_x=opt_xt)\n",
    "xt, rt, errs, opers, cd_xt, cd_rt, vol_st, vol_it, gamma_t, run_time, op_time = re_\n",
    "print(np.linalg.norm(xt, 1))\n",
    "plt.plot(np.cumsum(opers), np.log10(errs), label=\"APPR\")\n",
    "plt.legend()\n",
    "plt.show()"
   ],
   "metadata": {
    "collapsed": false
   },
   "id": "046ed7a6-bf98-404f-a31f-258cbbfdb120",
   "execution_count": 10
  }
 ],
 "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.11.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
