{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# hyperparameter\n",
    "# we will fix sigma to 1\n",
    "# mu is SNR\n",
    "# rho is label noise ratio\n",
    "import numpy as np\n",
    "import torch \n",
    "p = 500\n",
    "n = 8\n",
    "snr = 50\n",
    "rho = 0.2\n",
    "\n",
    "lr = 0.001\n",
    "epochs = 100"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.1027360021211954\n",
      "0.9741356342540504\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# data generation\n",
    "# signal, different direction are the \n",
    "mu = np.zeros((p,))\n",
    "mu[0] = snr\n",
    "features = []\n",
    "labels = []\n",
    "for i in range(n):\n",
    "    label = np.random.uniform()\n",
    "    label = 2*(label > 0.5) - 1\n",
    "    labels.append(label)\n",
    "    flip = np.random.uniform()\n",
    "    if(flip < rho):\n",
    "        labels[-1]  = -labels[-1]\n",
    "    feature = mu*label + np.random.normal(size = (p,))\n",
    "    feature = torch.tensor(feature)\n",
    "    features.append(feature)\n",
    "features = torch.stack(features)\n",
    "labels = torch.tensor(labels)\n",
    "onestep = (features*labels.unsqueeze(dim = 1)).mean(dim = 0)\n",
    "best = onestep[0]/onestep.norm()\n",
    "# training\n",
    "hatmu = torch.zeros((p,),dtype = float)\n",
    "hatmu.requires_grad = True\n",
    "epochs = 200\n",
    "lr = 1e-5\n",
    "hatmus = []\n",
    "losses = []\n",
    "for epoch in range(epochs):\n",
    "    running_loss = 0\n",
    "    for iter in range(n):\n",
    "        loss = torch.exp(-labels[iter]*torch.dot(hatmu,features[iter]))\n",
    "        loss.backward()\n",
    "        with torch.no_grad():\n",
    "            hatmu = hatmu - lr*hatmu.grad\n",
    "            running_loss += loss.item()\n",
    "        hatmu.requires_grad = True\n",
    "    hatmus.append((hatmu[0]/hatmu.norm()).detach().numpy())\n",
    "    losses.append(running_loss/n)\n",
    "from matplotlib import pyplot as plt\n",
    "plt.plot(hatmus)\n",
    "plt.plot(losses)\n",
    "plt.plot([best]*len(losses))\n",
    "print(hatmus[-1])\n",
    "print(max(hatmus))\n",
    "# print(losses[-1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "cmd = []\n",
    "for _ in range(15):\n",
    "    n = 8*pow(2,_)\n",
    "    for __ in range(_,15):\n",
    "        p = 8*pow(2,__)\n",
    "        for snr in [5,10,20,40]:\n",
    "            for rho in [0,0.4]:\n",
    "                cmd.append('python3 fit_l.py -n '+str(n)+' -p '+str(p)+' -snr '+str(snr)+' -rho '+str(rho)+' -save_path '+\\\n",
    "                    str(n)+'_'+str(p)+'_'+str(snr)+'_'+str(rho)+'_logistic')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "cmd = []\n",
    "perfdict = {}\n",
    "for _ in range(8):\n",
    "    n = 8*pow(2,_)\n",
    "    for __ in range(_,12):\n",
    "        p = 8*pow(2,__)\n",
    "        for snr in [40]:\n",
    "            for rho in [0, 0.4]:\n",
    "                hmax = 0\n",
    "                cnt = 0\n",
    "                hfinal = 0\n",
    "                for ___ in range(8):\n",
    "                    try:\n",
    "                        h = torch.load(str(n)+'_'+str(p)+'_'+str(snr)+'_'+str(rho)+'_logistic/stats'+str(___))\n",
    "                        cnt += 1\n",
    "                        hmax += max(h[1])\n",
    "                        hfinal += h[1][-1]\n",
    "                    except:\n",
    "                        continue\n",
    "                if(cnt > 0):\n",
    "                    hmax /= cnt\n",
    "                    hfinal /= cnt\n",
    "                if(cnt < 8):\n",
    "                    cmd.append('python3 fit.py -n '+str(n)+' -p '+str(p)+' -snr '+str(snr)+' -rho '+str(rho)+' -save_path '+\\\n",
    "                    str(n)+'_'+str(p)+'_'+str(snr)+'_'+str(rho))\n",
    "                perfdict[(n,p,snr,rho)] = (hmax,hfinal)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['python3 fit.py -n 256 -p 256 -snr 40 -rho 0.4 -save_path 256_256_40_0.4']"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "cmd"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy import special\n",
    "def toacc(snr,x):\n",
    "    bound = x*snr\n",
    "    acc = 1/2 + (1/2) * special.erf(bound/np.sqrt(2))\n",
    "    return acc\n",
    "cmd = []\n",
    "import  numpy as np\n",
    "for _ in range(8):\n",
    "    for __ in range(_,12):\n",
    "        n = 8*pow(2,_)\n",
    "        p = 8*pow(2,__)\n",
    "        for snr in [40]:\n",
    "            for rho in [0,0.4]:\n",
    "                # cmd.append('python3 fit_l.py -n '+str(n)+' -p '+str(p)+' -snr '+str(snr)+' -rho '+str(rho)+' -save_path '+\\\n",
    "                #     str(n)+'_'+str(p)+'_'+str(snr)+'_'+str(rho)+'_logistic')\n",
    "                perfdict[(n,p,snr,rho)] = (toacc(snr,perfdict[(n,p,snr,rho)][0]),toacc(snr, perfdict[(n,p,snr,rho)][1]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "17.0"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(cmd)/8"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {},
   "outputs": [],
   "source": [
    "for k in range(8):\n",
    "    with open(str(k),'w') as f:\n",
    "        for i in range(17):\n",
    "            try:\n",
    "                f.write(cmd[17*k + i]+'\\n')\n",
    "            except:\n",
    "                continue"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "40 0.4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_90409/1753658773.py:21: DeprecationWarning: `np.bool` is a deprecated alias for the builtin `bool`. To silence this warning, use `bool` by itself. Doing this will not modify any behavior and is safe. If you specifically wanted the numpy scalar type, use `np.bool_` here.\n",
      "Deprecated in NumPy 1.20; for more details and guidance: https://numpy.org/devdocs/release/1.20.0-notes.html#deprecations\n",
      "  mask = np.zeros_like(diff_matrix, dtype=np.bool)\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import seaborn as sns\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "len_p = 12\n",
    "len_n = 7\n",
    "for snr in [40]:\n",
    "    for rho in [0.4]:\n",
    "        best_matrix = np.zeros((len_n,len_p))\n",
    "        final_matrix = np.zeros((len_n,len_p))\n",
    "        diff_matrix = np.zeros((len_n,len_p))\n",
    "        for _ in range(len_n):\n",
    "            n = 8*pow(2,_)\n",
    "            for __ in range(_,len_p):\n",
    "                p = 8*pow(2,__)\n",
    "                best_matrix[_ ,__] = perfdict[(n,p,snr,rho)][0]\n",
    "                final_matrix[_ ,__] = perfdict[(n,p,snr,rho)][1]\n",
    "                # print(best_matrix[_,__])\n",
    "        print(snr,rho)\n",
    "        # diff_matrix = best_matrix - final_matrix\n",
    "        diff_matrix = (0.6 - 0.6*final_matrix - 0.4*(1-final_matrix))[1:,1:]\n",
    "        mask = np.zeros_like(diff_matrix, dtype=np.bool)\n",
    "        mask[np.tril_indices_from(mask)] = True\n",
    "        sns_plot = sns.heatmap(diff_matrix,cmap=\"Blues\",mask = mask,vmin = -0.001,vmax = 0.08)\n",
    "        # sns_plot.set_bed('w')\n",
    "        plt.xticks(np.arange(len_p -1)+0.5,[str(_ + 4) for _ in range(len_p -1)],rotation = 0,fontsize = 15)\n",
    "        plt.xlabel('log p',fontsize = 20)\n",
    "        plt.ylabel('log n',fontsize = 20)\n",
    "        plt.yticks(np.arange(len_n-1)+0.5,[str(_ + 4) for _ in range(len_n-1)],rotation = 0,fontsize = 15)\n",
    "        arange = np.linspace(0,10)\n",
    "        output = arange - 4\n",
    "        plt.plot(arange,output,color = 'Blue')\n",
    "        plt.text(4.5,3.8,'Mild', fontsize = 30, color = 'White') \n",
    "        plt.text(7.5,2.5,'Heavy', fontsize = 30, color = 'Black') \n",
    "        plt.tight_layout()\n",
    "        plt.show()\n",
    "        plt.savefig(str(snr)+'-'+str(rho)+'-sim-logistic.pdf',format = 'pdf')\n",
    "        plt.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "40 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_90409/3334184724.py:21: DeprecationWarning: `np.bool` is a deprecated alias for the builtin `bool`. To silence this warning, use `bool` by itself. Doing this will not modify any behavior and is safe. If you specifically wanted the numpy scalar type, use `np.bool_` here.\n",
      "Deprecated in NumPy 1.20; for more details and guidance: https://numpy.org/devdocs/release/1.20.0-notes.html#deprecations\n",
      "  mask = np.zeros_like(diff_matrix, dtype=np.bool)\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import seaborn as sns\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "len_p = 12\n",
    "len_n = 7\n",
    "for snr in [40]:\n",
    "    for rho in [0]:\n",
    "        best_matrix = np.zeros((len_n,len_p))\n",
    "        final_matrix = np.zeros((len_n,len_p))\n",
    "        diff_matrix = np.zeros((len_n,len_p))\n",
    "        for _ in range(len_n):\n",
    "            n = 8*pow(2,_)\n",
    "            for __ in range(_,len_p):\n",
    "                p = 8*pow(2,__)\n",
    "                best_matrix[_ ,__] = perfdict[(n,p,snr,rho)][0]\n",
    "                final_matrix[_ ,__] = perfdict[(n,p,snr,rho)][1]\n",
    "                # print(best_matrix[_,__])\n",
    "        print(snr,rho)\n",
    "        # diff_matrix = best_matrix - final_matrix\n",
    "        diff_matrix = (1.01 - final_matrix)[1:,1:]\n",
    "        mask = np.zeros_like(diff_matrix, dtype=np.bool)\n",
    "        mask[np.tril_indices_from(mask)] = True\n",
    "        # for k in range(len_n -1):\n",
    "        #     mask[k,k] = False\n",
    "        sns_plot = sns.heatmap(diff_matrix,cmap=\"Blues\",mask = mask,vmin = -0.001, vmax = 0.08)\n",
    "        # sns_plot.set_bed('w')\n",
    "        plt.xticks(np.arange(len_p -1)+0.5,[str(_ + 4) for _ in range(len_p -1)],rotation = 0,fontsize = 15)\n",
    "        plt.xlabel('log p',fontsize = 20)\n",
    "        plt.ylabel('log n',fontsize = 20)\n",
    "        plt.yticks(np.arange(len_n-1)+0.5,[str(_ + 4) for _ in range(len_n-1)],rotation = 0,fontsize = 15)\n",
    "        arange = np.linspace(0,10)\n",
    "        output = arange - 4\n",
    "        # plt.plot(arange,output,color = 'Blue')\n",
    "        # plt.text(4.5,3.8,'Mild', fontsize = 30, color = 'White') \n",
    "        # plt.text(7.5,2.5,'Heavy', fontsize = 30, color = 'Black') \n",
    "        plt.tight_layout()\n",
    "        plt.show()\n",
    "        plt.savefig(str(snr)+'-'+str(rho)+'-sim-logistic.pdf',format = 'pdf')\n",
    "        plt.close()"
   ]
  }
 ],
 "metadata": {
  "interpreter": {
   "hash": "e72f24b787c0d487131ca6ac3f3c79044e81d11bd62401f3bd21201feea542ed"
  },
  "kernelspec": {
   "display_name": "Python 3.8.13 ('JiayeTeng')",
   "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.13"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
