{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from reluNets.plotting.network_plotting_functions import plot_single_metric_axis, plot_several_metrics, save_figure\n",
    "from reluNets.plotting.data_figs import plot_SVD, plot_data, plot_data_axis\n",
    "from reluNets.analytical_solutions.analytical_solutions import analytical_svs, ana_weights, ana_outputs, ana_loss\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "from torch.utils.data import DataLoader\n",
    "import matplotlib\n",
    "import math\n",
    "from sklearn.manifold import MDS\n",
    "import seaborn as sns\n",
    "from sklearn import linear_model\n",
    "from sklearn.metrics import r2_score,explained_variance_score\n",
    "# This notebook plot the TNR and TPR of the relu net"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "choice_temp = 0.2\n",
    "scale = 0.001\n",
    "bucket_size = 1\n",
    "log_interval = 1 * bucket_size\n",
    "model_name = \"relu_net\"\n",
    "folder = \"model_runs_local\"\n",
    "save_folder = \"relu_nets_bias\"\n",
    "bias_input = True\n",
    "metrics = [\"loss\", \"accuracy\",\"TPR_top_level\", \n",
    "            \"TPR_mid_level\", \"TPR_bottom_level\", \"TNR_top_level\",\n",
    "            \"TNR_mid_level\", \"TNR_bottom_level\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "n = 4\n",
    "train_inputs_test = np.array([[2, -1, 0], [2, 0, 1], [2, 0, -1], [2, 1, 0]]).T\n",
    "train_inputs_test = train_inputs_test/2\n",
    "train_labels_test = np.array([[1, 0, 0, 1],\n",
    "                              [0, 1, 1, 0],\n",
    "                              [1, 1, 0, 0],\n",
    "                              [0, 0, 1, 1]])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([1.   , 0.125, 0.125]),\n",
       " array([[1., 0., 0.],\n",
       "        [0., 1., 0.],\n",
       "        [0., 0., 1.]]))"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# lets get our exact solutions for the weights now\n",
    "# start with loading data\n",
    "n = 4\n",
    "\n",
    "output_covar = (1/n) * (train_labels_test @ train_inputs_test.T)\n",
    "input_covar = (1/n) * (train_inputs_test @ train_inputs_test.T)\n",
    "# get eigenvectors of input covariance matrix\n",
    "eigvals, eigvecs = np.linalg.eig(input_covar)\n",
    "eigvals, eigvecs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# get the decomposition of these matrices\n",
    "u, s, vt =  np.linalg.svd(output_covar, full_matrices=False)\n",
    "s = s[:,np.newaxis]\n",
    "_, delta, _ =  np.linalg.svd(input_covar, full_matrices=False)\n",
    "delta = delta[:,np.newaxis]\n",
    "delta = delta[:2,:]\n",
    "s = s[:2,:]\n",
    "vt = vt[:2,:]\n",
    "u = u[:, :2]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "# hyperparameters\n",
    "t= np.arange(0, 50000, 1)[np.newaxis]\n",
    "a_0 = 0.00000061\n",
    "step_size = 0.001 # should load this from the wandb config"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "ana_sol_svs = analytical_svs(t, s, delta, a_0, step_size=step_size, batch_size=n)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([[0.49999999, 0.49999999, 0.49999999, 0.49999999],\n",
       "        [0.49999999, 0.49999999, 0.49999999, 0.49999999],\n",
       "        [0.50045417, 0.50045417, 0.49954581, 0.49954581],\n",
       "        [0.49954581, 0.49954581, 0.50045417, 0.50045417]]),\n",
       " array([1.        , 1.98657497]))"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# get the analytical outputs (time x outputs x n)\n",
    "ana_outputs_test = ana_outputs(u, ana_sol_svs, vt, train_inputs_test)\n",
    "ana_outputs_test[4000], ana_sol_svs[:,10000]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "# calculate the analytical loss at each time step\n",
    "ana_loss_bias = ana_loss(u, ana_sol_svs, vt, train_inputs_test, train_labels_test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0, Loss 3.9999992847442627\n",
      "Epoch 1000, Loss 3.9930195808410645\n",
      "Epoch 2000, Loss 2.05222749710083\n",
      "Epoch 3000, Loss 1.9997642040252686\n",
      "Epoch 4000, Loss 1.9982722997665405\n",
      "Epoch 5000, Loss 1.9874166250228882\n",
      "Epoch 6000, Loss 1.9127869606018066\n",
      "Epoch 7000, Loss 1.563597559928894\n",
      "Epoch 8000, Loss 1.0915826559066772\n",
      "Epoch 9000, Loss 1.0035079717636108\n",
      "Epoch 10000, Loss 1.0000760555267334\n",
      "Epoch 11000, Loss 0.9999804496765137\n",
      "Epoch 12000, Loss 0.9999443888664246\n",
      "Epoch 13000, Loss 0.9998747706413269\n",
      "Epoch 14000, Loss 0.9997358322143555\n",
      "Epoch 15000, Loss 0.9994406700134277\n",
      "Epoch 16000, Loss 0.9988059997558594\n",
      "Epoch 17000, Loss 0.9974870085716248\n",
      "Epoch 18000, Loss 0.9947805404663086\n",
      "Epoch 19000, Loss 0.9892849326133728\n",
      "Epoch 20000, Loss 0.9782423973083496\n",
      "Epoch 21000, Loss 0.9568474888801575\n",
      "Epoch 22000, Loss 0.9167873859405518\n",
      "Epoch 23000, Loss 0.8477287888526917\n",
      "Epoch 24000, Loss 0.7421857714653015\n",
      "Epoch 25000, Loss 0.6065083742141724\n",
      "Epoch 26000, Loss 0.4580330550670624\n",
      "Epoch 27000, Loss 0.3103792667388916\n",
      "Epoch 28000, Loss 0.17602166533470154\n",
      "Epoch 29000, Loss 0.07902412116527557\n",
      "Epoch 30000, Loss 0.029016872867941856\n",
      "Epoch 31000, Loss 0.00930132158100605\n",
      "Epoch 32000, Loss 0.0027493576053529978\n",
      "Epoch 33000, Loss 0.0007831642869859934\n",
      "Epoch 34000, Loss 0.0002220337773906067\n",
      "Epoch 35000, Loss 6.417602708097547e-05\n",
      "Epoch 36000, Loss 1.928557685459964e-05\n",
      "Epoch 37000, Loss 6.0997695072728675e-06\n",
      "Epoch 38000, Loss 2.0542559013847494e-06\n",
      "Epoch 39000, Loss 7.34678565095237e-07\n",
      "Epoch 0, Loss 4.000000953674316\n",
      "Epoch 1000, Loss 3.9954586029052734\n",
      "Epoch 2000, Loss 2.1050925254821777\n",
      "Epoch 3000, Loss 1.9996836185455322\n",
      "Epoch 4000, Loss 1.9976726770401\n",
      "Epoch 5000, Loss 1.9831901788711548\n",
      "Epoch 6000, Loss 1.8871593475341797\n",
      "Epoch 7000, Loss 1.478605031967163\n",
      "Epoch 8000, Loss 1.0585781335830688\n",
      "Epoch 9000, Loss 1.001907229423523\n",
      "Epoch 10000, Loss 1.0000331401824951\n",
      "Epoch 11000, Loss 0.999975860118866\n",
      "Epoch 12000, Loss 0.9999349117279053\n",
      "Epoch 13000, Loss 0.9998596906661987\n",
      "Epoch 14000, Loss 0.9997087717056274\n",
      "Epoch 15000, Loss 0.9993940591812134\n",
      "Epoch 16000, Loss 0.9987325668334961\n",
      "Epoch 17000, Loss 0.9973335266113281\n",
      "Epoch 18000, Loss 0.9945416450500488\n",
      "Epoch 19000, Loss 0.9889215230941772\n",
      "Epoch 20000, Loss 0.9777570962905884\n",
      "Epoch 21000, Loss 0.9561660885810852\n",
      "Epoch 22000, Loss 0.9164984226226807\n",
      "Epoch 23000, Loss 0.850249171257019\n",
      "Epoch 24000, Loss 0.7556633949279785\n",
      "Epoch 25000, Loss 0.6464804410934448\n",
      "Epoch 26000, Loss 0.5393714904785156\n",
      "Epoch 27000, Loss 0.43650898337364197\n",
      "Epoch 28000, Loss 0.326948881149292\n",
      "Epoch 29000, Loss 0.20734329521656036\n",
      "Epoch 30000, Loss 0.10518361628055573\n",
      "Epoch 31000, Loss 0.0433117039501667\n",
      "Epoch 32000, Loss 0.01540104579180479\n",
      "Epoch 33000, Loss 0.005056489259004593\n",
      "Epoch 34000, Loss 0.0016161813400685787\n",
      "Epoch 35000, Loss 0.0005299021140672266\n",
      "Epoch 36000, Loss 0.00018097760039381683\n",
      "Epoch 37000, Loss 6.34161988273263e-05\n",
      "Epoch 38000, Loss 2.2664456992060877e-05\n",
      "Epoch 39000, Loss 8.206068741856143e-06\n",
      "Epoch 0, Loss 3.999999761581421\n",
      "Epoch 1000, Loss 3.992502212524414\n",
      "Epoch 2000, Loss 2.0462429523468018\n",
      "Epoch 3000, Loss 1.9998210668563843\n",
      "Epoch 4000, Loss 1.9986830949783325\n",
      "Epoch 5000, Loss 1.9904040098190308\n",
      "Epoch 6000, Loss 1.9337135553359985\n",
      "Epoch 7000, Loss 1.6420015096664429\n",
      "Epoch 8000, Loss 1.1327558755874634\n",
      "Epoch 9000, Loss 1.0057663917541504\n",
      "Epoch 10000, Loss 1.000136137008667\n",
      "Epoch 11000, Loss 1.000002384185791\n",
      "Epoch 12000, Loss 0.9999983906745911\n",
      "Epoch 13000, Loss 0.9999957084655762\n",
      "Epoch 14000, Loss 0.9999886751174927\n",
      "Epoch 15000, Loss 0.9999707937240601\n",
      "Epoch 16000, Loss 0.9999247789382935\n",
      "Epoch 17000, Loss 0.9998060464859009\n",
      "Epoch 18000, Loss 0.9995000958442688\n",
      "Epoch 19000, Loss 0.9987112879753113\n",
      "Epoch 20000, Loss 0.996813952922821\n",
      "Epoch 21000, Loss 0.9925723075866699\n",
      "Epoch 22000, Loss 0.9843677878379822\n",
      "Epoch 23000, Loss 0.9679442644119263\n",
      "Epoch 24000, Loss 0.9350446462631226\n",
      "Epoch 25000, Loss 0.8717202544212341\n",
      "Epoch 26000, Loss 0.7599756717681885\n",
      "Epoch 27000, Loss 0.5895175933837891\n",
      "Epoch 28000, Loss 0.38325852155685425\n",
      "Epoch 29000, Loss 0.20017428696155548\n",
      "Epoch 30000, Loss 0.08370570838451385\n",
      "Epoch 31000, Loss 0.028835976496338844\n",
      "Epoch 32000, Loss 0.008655032142996788\n",
      "Epoch 33000, Loss 0.002397079486399889\n",
      "Epoch 34000, Loss 0.0006401523714885116\n",
      "Epoch 35000, Loss 0.0001691158686298877\n",
      "Epoch 36000, Loss 4.519348294707015e-05\n",
      "Epoch 37000, Loss 1.2374616744637024e-05\n",
      "Epoch 38000, Loss 3.5199309422750957e-06\n",
      "Epoch 39000, Loss 1.0530737881708774e-06\n",
      "Epoch 0, Loss 3.999999523162842\n",
      "Epoch 1000, Loss 3.991495370864868\n",
      "Epoch 2000, Loss 2.0371978282928467\n",
      "Epoch 3000, Loss 1.9998319149017334\n",
      "Epoch 4000, Loss 1.998761534690857\n",
      "Epoch 5000, Loss 1.9909205436706543\n",
      "Epoch 6000, Loss 1.9358577728271484\n",
      "Epoch 7000, Loss 1.6426892280578613\n",
      "Epoch 8000, Loss 1.1300790309906006\n",
      "Epoch 9000, Loss 1.0054575204849243\n",
      "Epoch 10000, Loss 1.0001256465911865\n",
      "Epoch 11000, Loss 0.9999989867210388\n",
      "Epoch 12000, Loss 0.999991774559021\n",
      "Epoch 13000, Loss 0.9999826550483704\n",
      "Epoch 14000, Loss 0.9999639987945557\n",
      "Epoch 15000, Loss 0.999924898147583\n",
      "Epoch 16000, Loss 0.9998424053192139\n",
      "Epoch 17000, Loss 0.9996664524078369\n",
      "Epoch 18000, Loss 0.9992874264717102\n",
      "Epoch 19000, Loss 0.99846351146698\n",
      "Epoch 20000, Loss 0.9966524243354797\n",
      "Epoch 21000, Loss 0.9926356077194214\n",
      "Epoch 22000, Loss 0.9842503666877747\n",
      "Epoch 23000, Loss 0.9680790901184082\n",
      "Epoch 24000, Loss 0.9369775056838989\n",
      "Epoch 25000, Loss 0.8809062242507935\n",
      "Epoch 26000, Loss 0.7913265228271484\n",
      "Epoch 27000, Loss 0.6656336188316345\n",
      "Epoch 28000, Loss 0.5139911770820618\n",
      "Epoch 29000, Loss 0.35396867990493774\n",
      "Epoch 30000, Loss 0.20694217085838318\n",
      "Epoch 31000, Loss 0.09821267426013947\n",
      "Epoch 32000, Loss 0.03833944723010063\n",
      "Epoch 33000, Loss 0.013041873462498188\n",
      "Epoch 34000, Loss 0.004142359364777803\n",
      "Epoch 35000, Loss 0.001297002425417304\n",
      "Epoch 36000, Loss 0.0004160475800745189\n",
      "Epoch 37000, Loss 0.0001383348717354238\n",
      "Epoch 38000, Loss 4.751744927489199e-05\n",
      "Epoch 39000, Loss 1.6771506125223823e-05\n",
      "Epoch 0, Loss 3.9999990463256836\n",
      "Epoch 1000, Loss 3.991250514984131\n",
      "Epoch 2000, Loss 2.0353970527648926\n",
      "Epoch 3000, Loss 1.9997076988220215\n",
      "Epoch 4000, Loss 1.9978532791137695\n",
      "Epoch 5000, Loss 1.9843723773956299\n",
      "Epoch 6000, Loss 1.8936578035354614\n",
      "Epoch 7000, Loss 1.5027813911437988\n",
      "Epoch 8000, Loss 1.0666002035140991\n",
      "Epoch 9000, Loss 1.002286434173584\n",
      "Epoch 10000, Loss 1.000050663948059\n",
      "Epoch 11000, Loss 0.9999977946281433\n",
      "Epoch 12000, Loss 0.9999910593032837\n",
      "Epoch 13000, Loss 0.9999767541885376\n",
      "Epoch 14000, Loss 0.9999423027038574\n",
      "Epoch 15000, Loss 0.9998632669448853\n",
      "Epoch 16000, Loss 0.9996845722198486\n",
      "Epoch 17000, Loss 0.9993436336517334\n",
      "Epoch 18000, Loss 0.9986371994018555\n",
      "Epoch 19000, Loss 0.9971585869789124\n",
      "Epoch 20000, Loss 0.9941035509109497\n",
      "Epoch 21000, Loss 0.9877881407737732\n",
      "Epoch 22000, Loss 0.9747808575630188\n",
      "Epoch 23000, Loss 0.9483931064605713\n",
      "Epoch 24000, Loss 0.8967862129211426\n",
      "Epoch 25000, Loss 0.8049708008766174\n",
      "Epoch 26000, Loss 0.6736389398574829\n",
      "Epoch 27000, Loss 0.5170332193374634\n",
      "Epoch 28000, Loss 0.35702961683273315\n",
      "Epoch 29000, Loss 0.21175919473171234\n",
      "Epoch 30000, Loss 0.1008332371711731\n",
      "Epoch 31000, Loss 0.03847263380885124\n",
      "Epoch 32000, Loss 0.012489551678299904\n",
      "Epoch 33000, Loss 0.0036602430045604706\n",
      "Epoch 34000, Loss 0.001011459156870842\n",
      "Epoch 35000, Loss 0.00027186915394850075\n",
      "Epoch 36000, Loss 7.231041672639549e-05\n",
      "Epoch 37000, Loss 1.927204357343726e-05\n",
      "Epoch 38000, Loss 5.202175998419989e-06\n",
      "Epoch 39000, Loss 1.4413950566449785e-06\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.990720510482788\n",
      "Epoch 2000, Loss 2.031920909881592\n",
      "Epoch 3000, Loss 1.9997146129608154\n",
      "Epoch 4000, Loss 1.9979006052017212\n",
      "Epoch 5000, Loss 1.9847025871276855\n",
      "Epoch 6000, Loss 1.8969835042953491\n",
      "Epoch 7000, Loss 1.5145742893218994\n",
      "Epoch 8000, Loss 1.0711238384246826\n",
      "Epoch 9000, Loss 1.0024614334106445\n",
      "Epoch 10000, Loss 1.0000578165054321\n",
      "Epoch 11000, Loss 0.9999964833259583\n",
      "Epoch 12000, Loss 0.9999871253967285\n",
      "Epoch 13000, Loss 0.9999663233757019\n",
      "Epoch 14000, Loss 0.9999130964279175\n",
      "Epoch 15000, Loss 0.9997925162315369\n",
      "Epoch 16000, Loss 0.9995324611663818\n",
      "Epoch 17000, Loss 0.9990273714065552\n",
      "Epoch 18000, Loss 0.9980142712593079\n",
      "Epoch 19000, Loss 0.9959537982940674\n",
      "Epoch 20000, Loss 0.9917553663253784\n",
      "Epoch 21000, Loss 0.9832512736320496\n",
      "Epoch 22000, Loss 0.9661024212837219\n",
      "Epoch 23000, Loss 0.9322366118431091\n",
      "Epoch 24000, Loss 0.8679511547088623\n",
      "Epoch 25000, Loss 0.7550652623176575\n",
      "Epoch 26000, Loss 0.5844857096672058\n",
      "Epoch 27000, Loss 0.3791612982749939\n",
      "Epoch 28000, Loss 0.19627633690834045\n",
      "Epoch 29000, Loss 0.08090869337320328\n",
      "Epoch 30000, Loss 0.02766919881105423\n",
      "Epoch 31000, Loss 0.008367553353309631\n",
      "Epoch 32000, Loss 0.00236120936460793\n",
      "Epoch 33000, Loss 0.0006480882875621319\n",
      "Epoch 34000, Loss 0.0001772955438354984\n",
      "Epoch 35000, Loss 4.9974809371633455e-05\n",
      "Epoch 36000, Loss 1.4698438462801278e-05\n",
      "Epoch 37000, Loss 4.5663296077691484e-06\n",
      "Epoch 38000, Loss 1.505991349404212e-06\n",
      "Epoch 39000, Loss 5.21047297752375e-07\n",
      "Epoch 0, Loss 3.999999523162842\n",
      "Epoch 1000, Loss 3.9928760528564453\n",
      "Epoch 2000, Loss 2.0504634380340576\n",
      "Epoch 3000, Loss 1.9996670484542847\n",
      "Epoch 4000, Loss 1.9975471496582031\n",
      "Epoch 5000, Loss 1.9821481704711914\n",
      "Epoch 6000, Loss 1.8800712823867798\n",
      "Epoch 7000, Loss 1.4603548049926758\n",
      "Epoch 8000, Loss 1.0531108379364014\n",
      "Epoch 9000, Loss 1.0016613006591797\n",
      "Epoch 10000, Loss 1.0000361204147339\n",
      "Epoch 11000, Loss 1.0000007152557373\n",
      "Epoch 12000, Loss 0.9999999403953552\n",
      "Epoch 13000, Loss 0.9999998211860657\n",
      "Epoch 14000, Loss 0.9999992847442627\n",
      "Epoch 15000, Loss 0.9999982714653015\n",
      "Epoch 16000, Loss 0.9999958276748657\n",
      "Epoch 17000, Loss 0.9999895691871643\n",
      "Epoch 18000, Loss 0.9999737739562988\n",
      "Epoch 19000, Loss 0.9999337792396545\n",
      "Epoch 20000, Loss 0.999832272529602\n",
      "Epoch 21000, Loss 0.999573826789856\n",
      "Epoch 22000, Loss 0.9989168047904968\n",
      "Epoch 23000, Loss 0.9972459077835083\n",
      "Epoch 24000, Loss 0.9930046796798706\n",
      "Epoch 25000, Loss 0.9823107719421387\n",
      "Epoch 26000, Loss 0.9558143615722656\n",
      "Epoch 27000, Loss 0.8930383324623108\n",
      "Epoch 28000, Loss 0.7625097036361694\n",
      "Epoch 29000, Loss 0.570872962474823\n",
      "Epoch 30000, Loss 0.3499698340892792\n",
      "Epoch 31000, Loss 0.17134422063827515\n",
      "Epoch 32000, Loss 0.06714358925819397\n",
      "Epoch 33000, Loss 0.022214286029338837\n",
      "Epoch 34000, Loss 0.006597480271011591\n",
      "Epoch 35000, Loss 0.001832083216868341\n",
      "Epoch 36000, Loss 0.0004917050246149302\n",
      "Epoch 37000, Loss 0.00012995526776649058\n",
      "Epoch 38000, Loss 3.4229946322739124e-05\n",
      "Epoch 39000, Loss 9.061483979166951e-06\n",
      "Epoch 0, Loss 4.000000476837158\n",
      "Epoch 1000, Loss 3.9943246841430664\n",
      "Epoch 2000, Loss 2.0736472606658936\n",
      "Epoch 3000, Loss 1.9998266696929932\n",
      "Epoch 4000, Loss 1.9987239837646484\n",
      "Epoch 5000, Loss 1.9906889200210571\n",
      "Epoch 6000, Loss 1.935072898864746\n",
      "Epoch 7000, Loss 1.6417169570922852\n",
      "Epoch 8000, Loss 1.1341137886047363\n",
      "Epoch 9000, Loss 1.0061657428741455\n",
      "Epoch 10000, Loss 1.000154972076416\n",
      "Epoch 11000, Loss 1.0000033378601074\n",
      "Epoch 12000, Loss 0.9999995827674866\n",
      "Epoch 13000, Loss 0.9999986290931702\n",
      "Epoch 14000, Loss 0.9999963641166687\n",
      "Epoch 15000, Loss 0.9999906420707703\n",
      "Epoch 16000, Loss 0.9999758005142212\n",
      "Epoch 17000, Loss 0.9999377131462097\n",
      "Epoch 18000, Loss 0.9998397827148438\n",
      "Epoch 19000, Loss 0.9995872974395752\n",
      "Epoch 20000, Loss 0.9989368319511414\n",
      "Epoch 21000, Loss 0.9973642230033875\n",
      "Epoch 22000, Loss 0.99378502368927\n",
      "Epoch 23000, Loss 0.9855270385742188\n",
      "Epoch 24000, Loss 0.9683151841163635\n",
      "Epoch 25000, Loss 0.935553789138794\n",
      "Epoch 26000, Loss 0.8722370862960815\n",
      "Epoch 27000, Loss 0.7600606679916382\n",
      "Epoch 28000, Loss 0.5895523428916931\n",
      "Epoch 29000, Loss 0.3822186291217804\n",
      "Epoch 30000, Loss 0.19636285305023193\n",
      "Epoch 31000, Loss 0.07942871004343033\n",
      "Epoch 32000, Loss 0.0265754796564579\n",
      "Epoch 33000, Loss 0.0077968016266822815\n",
      "Epoch 34000, Loss 0.0021112116519361734\n",
      "Epoch 35000, Loss 0.0005474859499372542\n",
      "Epoch 36000, Loss 0.00013897883764002472\n",
      "Epoch 37000, Loss 3.4969503758475184e-05\n",
      "Epoch 38000, Loss 8.779127711022738e-06\n",
      "Epoch 39000, Loss 2.207957777500269e-06\n",
      "Epoch 0, Loss 4.000000476837158\n",
      "Epoch 1000, Loss 3.995946168899536\n",
      "Epoch 2000, Loss 2.12506365776062\n",
      "Epoch 3000, Loss 1.9997127056121826\n",
      "Epoch 4000, Loss 1.9978852272033691\n",
      "Epoch 5000, Loss 1.984615683555603\n",
      "Epoch 6000, Loss 1.8947067260742188\n",
      "Epoch 7000, Loss 1.5009962320327759\n",
      "Epoch 8000, Loss 1.0644537210464478\n",
      "Epoch 9000, Loss 1.0021288394927979\n",
      "Epoch 10000, Loss 1.000048041343689\n",
      "Epoch 11000, Loss 1.0000005960464478\n",
      "Epoch 12000, Loss 0.9999982714653015\n",
      "Epoch 13000, Loss 0.9999954104423523\n",
      "Epoch 14000, Loss 0.9999881982803345\n",
      "Epoch 15000, Loss 0.9999696016311646\n",
      "Epoch 16000, Loss 0.9999215602874756\n",
      "Epoch 17000, Loss 0.9997976422309875\n",
      "Epoch 18000, Loss 0.9995731115341187\n",
      "Epoch 19000, Loss 0.9991098642349243\n",
      "Epoch 20000, Loss 0.9981396794319153\n",
      "Epoch 21000, Loss 0.9960975050926208\n",
      "Epoch 22000, Loss 0.991769015789032\n",
      "Epoch 23000, Loss 0.9826337099075317\n",
      "Epoch 24000, Loss 0.9635412096977234\n",
      "Epoch 25000, Loss 0.9254511594772339\n",
      "Epoch 26000, Loss 0.858310878276825\n",
      "Epoch 27000, Loss 0.7607863545417786\n",
      "Epoch 28000, Loss 0.6466389298439026\n",
      "Epoch 29000, Loss 0.5488842725753784\n",
      "Epoch 30000, Loss 0.47157129645347595\n",
      "Epoch 31000, Loss 0.3918445110321045\n",
      "Epoch 32000, Loss 0.2866959869861603\n",
      "Epoch 33000, Loss 0.1699867844581604\n",
      "Epoch 34000, Loss 0.08070738613605499\n",
      "Epoch 35000, Loss 0.03258846327662468\n",
      "Epoch 36000, Loss 0.012045958079397678\n",
      "Epoch 37000, Loss 0.004310383461415768\n",
      "Epoch 38000, Loss 0.0015451040817424655\n",
      "Epoch 39000, Loss 0.000564100279007107\n",
      "Epoch 0, Loss 3.999999761581421\n",
      "Epoch 1000, Loss 3.9906773567199707\n",
      "Epoch 2000, Loss 2.031660318374634\n",
      "Epoch 3000, Loss 1.9997973442077637\n",
      "Epoch 4000, Loss 1.99850594997406\n",
      "Epoch 5000, Loss 1.9890761375427246\n",
      "Epoch 6000, Loss 1.9237427711486816\n",
      "Epoch 7000, Loss 1.5954009294509888\n",
      "Epoch 8000, Loss 1.1009628772735596\n",
      "Epoch 9000, Loss 1.0038597583770752\n",
      "Epoch 10000, Loss 1.0000874996185303\n",
      "Epoch 11000, Loss 1.0000016689300537\n",
      "Epoch 12000, Loss 0.9999995231628418\n",
      "Epoch 13000, Loss 0.9999986886978149\n",
      "Epoch 14000, Loss 0.9999964833259583\n",
      "Epoch 15000, Loss 0.9999911785125732\n",
      "Epoch 16000, Loss 0.999977171421051\n",
      "Epoch 17000, Loss 0.9999411702156067\n",
      "Epoch 18000, Loss 0.9998482465744019\n",
      "Epoch 19000, Loss 0.9996163845062256\n",
      "Epoch 20000, Loss 0.9991190433502197\n",
      "Epoch 21000, Loss 0.9979591369628906\n",
      "Epoch 22000, Loss 0.9952173233032227\n",
      "Epoch 23000, Loss 0.9887683391571045\n",
      "Epoch 24000, Loss 0.9741619825363159\n",
      "Epoch 25000, Loss 0.9459782242774963\n",
      "Epoch 26000, Loss 0.8917985558509827\n",
      "Epoch 27000, Loss 0.7925134897232056\n",
      "Epoch 28000, Loss 0.6380792260169983\n",
      "Epoch 29000, Loss 0.4392111301422119\n",
      "Epoch 30000, Loss 0.2466641366481781\n",
      "Epoch 31000, Loss 0.11209388077259064\n",
      "Epoch 32000, Loss 0.0426601767539978\n",
      "Epoch 33000, Loss 0.014553680084645748\n",
      "Epoch 34000, Loss 0.004711313638836145\n",
      "Epoch 35000, Loss 0.0015047367196530104\n",
      "Epoch 36000, Loss 0.0004890753771178424\n",
      "Epoch 37000, Loss 0.00016489977133460343\n",
      "Epoch 38000, Loss 5.819950456498191e-05\n",
      "Epoch 39000, Loss 2.1580575776170008e-05\n"
     ]
    }
   ],
   "source": [
    "# train a small relu network on the task xor task \n",
    "# hyperparameters\n",
    "input_size = 3\n",
    "hidden_size = 64\n",
    "output_size = 4\n",
    "n_epochs = 40000\n",
    "n_runs = 10\n",
    "\n",
    "# define a pytorch model and initialize weights as small gaussian\n",
    "class ReluNet(torch.nn.Module):\n",
    "    def __init__(self):\n",
    "        super(ReluNet, self).__init__()\n",
    "        self.fc1 = torch.nn.Linear(input_size, hidden_size, bias=False)\n",
    "        self.fc2 = torch.nn.Linear(hidden_size, output_size, bias=False)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = torch.relu(self.fc1(x))\n",
    "        x = self.fc2(x)\n",
    "        return x\n",
    "    \n",
    "    def get_hidden(self, x):\n",
    "        # x = self.fc1(x)\n",
    "        # return self.leaky_relu(x)\n",
    "        return torch.relu(self.fc1(x))\n",
    "    \n",
    "# initialize the model\n",
    "model = ReluNet()\n",
    "# initialize the weights\n",
    "model.fc1.weight.data = torch.randn_like(model.fc1.weight.data)*0.0002\n",
    "model.fc2.weight.data = torch.randn_like(model.fc2.weight.data)*0.0002\n",
    "\n",
    "# load the data we do not need test data with pytorch dataloaders\n",
    "train_loader = DataLoader(torch.utils.data.TensorDataset(torch.tensor(train_inputs_test.T, dtype=torch.float32), \n",
    "                                                          torch.tensor(train_labels_test.T, dtype=torch.float32)), \n",
    "                          batch_size=n, shuffle=False)\n",
    "\n",
    "# define the optimizer\n",
    "optimizer = torch.optim.SGD(model.parameters(), lr=0.001)\n",
    "\n",
    "# define the loss function\n",
    "loss = torch.nn.MSELoss(reduction='sum')\n",
    "\n",
    "# train the model\n",
    "def train(model, train_loader, optimizer):\n",
    "    losses = []\n",
    "    for epoch in range(n_epochs):\n",
    "        for inputs, labels in train_loader:\n",
    "            optimizer.zero_grad()\n",
    "            outputs = model(inputs)\n",
    "            loss_val = 1/2 * loss(outputs, labels)\n",
    "            loss_val.backward()\n",
    "            optimizer.step()\n",
    "            losses.append(loss_val.item())\n",
    "            # print every 1000th epoch\n",
    "            if epoch % 1000 == 0:\n",
    "                print(f\"Epoch {epoch}, Loss {loss_val.item()}\")\n",
    "    hidden_activation = model.get_hidden(inputs)\n",
    "    # convert to numpy array\n",
    "    hidden_activation = hidden_activation.detach().numpy()\n",
    "    return hidden_activation, losses\n",
    "\n",
    "# hidden_activations_relu_bias, losses = train(model, train_loader, optimizer)\n",
    "hidden_activations_relu_bias_list = []\n",
    "loss_list = []\n",
    "for run in range(n_runs):\n",
    "    # initialize the model\n",
    "    model = ReluNet()\n",
    "    # initialize the weights\n",
    "    model.fc1.weight.data = torch.randn_like(model.fc1.weight.data)*0.0002\n",
    "    model.fc2.weight.data = torch.randn_like(model.fc2.weight.data)*0.0002\n",
    "    # define the optimizer\n",
    "    optimizer = torch.optim.SGD(model.parameters(), lr=0.001)\n",
    "    # train the model\n",
    "    hidden_activations_relu_bias, losses = train(model, train_loader, optimizer)\n",
    "    hidden_activations_relu_bias_list.append(hidden_activations_relu_bias)\n",
    "    loss_list.append(losses)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0, Loss 3.9999990463256836\n",
      "Epoch 1000, Loss 3.9853758811950684\n",
      "Epoch 2000, Loss 2.0139646530151367\n",
      "Epoch 3000, Loss 1.9995099306106567\n",
      "Epoch 4000, Loss 1.9963918924331665\n",
      "Epoch 5000, Loss 1.973818302154541\n",
      "Epoch 6000, Loss 1.8285784721374512\n",
      "Epoch 7000, Loss 1.3348174095153809\n",
      "Epoch 8000, Loss 1.024521827697754\n",
      "Epoch 9000, Loss 1.000598669052124\n",
      "Epoch 10000, Loss 1.0000114440917969\n",
      "Epoch 11000, Loss 1.0\n",
      "Epoch 12000, Loss 1.0000001192092896\n",
      "Epoch 13000, Loss 1.0\n",
      "Epoch 14000, Loss 0.9999998807907104\n",
      "Epoch 15000, Loss 1.0\n",
      "Epoch 16000, Loss 0.9999999403953552\n",
      "Epoch 17000, Loss 1.0\n",
      "Epoch 18000, Loss 1.0\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 1.0\n",
      "Epoch 21000, Loss 1.0\n",
      "Epoch 22000, Loss 0.9999999403953552\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 1.0\n",
      "Epoch 25000, Loss 1.0\n",
      "Epoch 26000, Loss 1.0\n",
      "Epoch 27000, Loss 1.0\n",
      "Epoch 28000, Loss 1.0\n",
      "Epoch 29000, Loss 0.9999999403953552\n",
      "Epoch 30000, Loss 0.9999999403953552\n",
      "Epoch 31000, Loss 1.0\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 0.9999999403953552\n",
      "Epoch 34000, Loss 0.9999999403953552\n",
      "Epoch 35000, Loss 1.0\n",
      "Epoch 36000, Loss 1.0\n",
      "Epoch 37000, Loss 1.0\n",
      "Epoch 38000, Loss 1.0\n",
      "Epoch 39000, Loss 0.9999999403953552\n",
      "Epoch 0, Loss 4.000001430511475\n",
      "Epoch 1000, Loss 3.9890947341918945\n",
      "Epoch 2000, Loss 2.023911952972412\n",
      "Epoch 3000, Loss 1.9994170665740967\n",
      "Epoch 4000, Loss 1.9957084655761719\n",
      "Epoch 5000, Loss 1.9689610004425049\n",
      "Epoch 6000, Loss 1.8010306358337402\n",
      "Epoch 7000, Loss 1.2869969606399536\n",
      "Epoch 8000, Loss 1.0182095766067505\n",
      "Epoch 9000, Loss 1.0004258155822754\n",
      "Epoch 10000, Loss 1.0000081062316895\n",
      "Epoch 11000, Loss 1.0000001192092896\n",
      "Epoch 12000, Loss 1.0\n",
      "Epoch 13000, Loss 1.0\n",
      "Epoch 14000, Loss 0.9999999403953552\n",
      "Epoch 15000, Loss 1.0\n",
      "Epoch 16000, Loss 1.0\n",
      "Epoch 17000, Loss 1.0\n",
      "Epoch 18000, Loss 1.0\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 1.0\n",
      "Epoch 21000, Loss 1.0\n",
      "Epoch 22000, Loss 1.0\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 1.0\n",
      "Epoch 25000, Loss 1.0\n",
      "Epoch 26000, Loss 1.0\n",
      "Epoch 27000, Loss 0.9999999403953552\n",
      "Epoch 28000, Loss 1.0\n",
      "Epoch 29000, Loss 1.0\n",
      "Epoch 30000, Loss 1.0\n",
      "Epoch 31000, Loss 1.0\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 1.0\n",
      "Epoch 34000, Loss 1.0\n",
      "Epoch 35000, Loss 1.0\n",
      "Epoch 36000, Loss 1.0\n",
      "Epoch 37000, Loss 0.9999999403953552\n",
      "Epoch 38000, Loss 0.9999999403953552\n",
      "Epoch 39000, Loss 0.9999999403953552\n",
      "Epoch 0, Loss 3.999999761581421\n",
      "Epoch 1000, Loss 3.983060359954834\n",
      "Epoch 2000, Loss 2.0106024742126465\n",
      "Epoch 3000, Loss 1.9995416402816772\n",
      "Epoch 4000, Loss 1.9966232776641846\n",
      "Epoch 5000, Loss 1.9754712581634521\n",
      "Epoch 6000, Loss 1.8382282257080078\n",
      "Epoch 7000, Loss 1.3537436723709106\n",
      "Epoch 8000, Loss 1.0274193286895752\n",
      "Epoch 9000, Loss 1.000681757926941\n",
      "Epoch 10000, Loss 1.000012993812561\n",
      "Epoch 11000, Loss 1.000000238418579\n",
      "Epoch 12000, Loss 1.0\n",
      "Epoch 13000, Loss 1.0\n",
      "Epoch 14000, Loss 1.0\n",
      "Epoch 15000, Loss 1.0\n",
      "Epoch 16000, Loss 1.0\n",
      "Epoch 17000, Loss 1.0\n",
      "Epoch 18000, Loss 1.0\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 1.0\n",
      "Epoch 21000, Loss 1.0\n",
      "Epoch 22000, Loss 1.0\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 1.0\n",
      "Epoch 25000, Loss 1.0\n",
      "Epoch 26000, Loss 1.0\n",
      "Epoch 27000, Loss 1.0\n",
      "Epoch 28000, Loss 1.0\n",
      "Epoch 29000, Loss 1.0\n",
      "Epoch 30000, Loss 1.0\n",
      "Epoch 31000, Loss 1.0\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 1.0\n",
      "Epoch 34000, Loss 1.0\n",
      "Epoch 35000, Loss 1.0\n",
      "Epoch 36000, Loss 1.0\n",
      "Epoch 37000, Loss 1.0\n",
      "Epoch 38000, Loss 1.0\n",
      "Epoch 39000, Loss 1.0\n",
      "Epoch 0, Loss 3.999999523162842\n",
      "Epoch 1000, Loss 3.986994504928589\n",
      "Epoch 2000, Loss 2.0173592567443848\n",
      "Epoch 3000, Loss 1.9996027946472168\n",
      "Epoch 4000, Loss 1.9970723390579224\n",
      "Epoch 5000, Loss 1.978686809539795\n",
      "Epoch 6000, Loss 1.857414960861206\n",
      "Epoch 7000, Loss 1.395294427871704\n",
      "Epoch 8000, Loss 1.0347167253494263\n",
      "Epoch 9000, Loss 1.0009005069732666\n",
      "Epoch 10000, Loss 1.0000172853469849\n",
      "Epoch 11000, Loss 1.0000003576278687\n",
      "Epoch 12000, Loss 0.9999999403953552\n",
      "Epoch 13000, Loss 1.0\n",
      "Epoch 14000, Loss 1.0\n",
      "Epoch 15000, Loss 1.0\n",
      "Epoch 16000, Loss 1.0\n",
      "Epoch 17000, Loss 1.0\n",
      "Epoch 18000, Loss 1.0\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 1.0\n",
      "Epoch 21000, Loss 0.9999999403953552\n",
      "Epoch 22000, Loss 0.9999999403953552\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 1.0\n",
      "Epoch 25000, Loss 1.0\n",
      "Epoch 26000, Loss 1.0\n",
      "Epoch 27000, Loss 1.0\n",
      "Epoch 28000, Loss 1.0\n",
      "Epoch 29000, Loss 1.0\n",
      "Epoch 30000, Loss 1.0\n",
      "Epoch 31000, Loss 1.0\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 1.0\n",
      "Epoch 34000, Loss 1.0\n",
      "Epoch 35000, Loss 1.0\n",
      "Epoch 36000, Loss 1.0\n",
      "Epoch 37000, Loss 1.0\n",
      "Epoch 38000, Loss 1.0\n",
      "Epoch 39000, Loss 1.0\n",
      "Epoch 0, Loss 3.999999523162842\n",
      "Epoch 1000, Loss 3.985297203063965\n",
      "Epoch 2000, Loss 2.013828754425049\n",
      "Epoch 3000, Loss 1.9995495080947876\n",
      "Epoch 4000, Loss 1.9966816902160645\n",
      "Epoch 5000, Loss 1.9758896827697754\n",
      "Epoch 6000, Loss 1.8406922817230225\n",
      "Epoch 7000, Loss 1.3587779998779297\n",
      "Epoch 8000, Loss 1.0282325744628906\n",
      "Epoch 9000, Loss 1.0007054805755615\n",
      "Epoch 10000, Loss 1.0000133514404297\n",
      "Epoch 11000, Loss 1.0000003576278687\n",
      "Epoch 12000, Loss 1.0\n",
      "Epoch 13000, Loss 1.0\n",
      "Epoch 14000, Loss 1.0\n",
      "Epoch 15000, Loss 1.0\n",
      "Epoch 16000, Loss 1.0\n",
      "Epoch 17000, Loss 1.0\n",
      "Epoch 18000, Loss 1.0\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 1.0\n",
      "Epoch 21000, Loss 0.9999998807907104\n",
      "Epoch 22000, Loss 1.0\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 1.0\n",
      "Epoch 25000, Loss 1.0\n",
      "Epoch 26000, Loss 1.0\n",
      "Epoch 27000, Loss 1.0\n",
      "Epoch 28000, Loss 1.0\n",
      "Epoch 29000, Loss 1.0\n",
      "Epoch 30000, Loss 1.0\n",
      "Epoch 31000, Loss 1.0\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 0.9999998807907104\n",
      "Epoch 34000, Loss 1.0\n",
      "Epoch 35000, Loss 1.0\n",
      "Epoch 36000, Loss 1.0\n",
      "Epoch 37000, Loss 1.0\n",
      "Epoch 38000, Loss 1.0\n",
      "Epoch 39000, Loss 1.0\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.98787784576416\n",
      "Epoch 2000, Loss 2.0197317600250244\n",
      "Epoch 3000, Loss 1.9994049072265625\n",
      "Epoch 4000, Loss 1.9956191778182983\n",
      "Epoch 5000, Loss 1.9683268070220947\n",
      "Epoch 6000, Loss 1.7975201606750488\n",
      "Epoch 7000, Loss 1.281494140625\n",
      "Epoch 8000, Loss 1.0175668001174927\n",
      "Epoch 9000, Loss 1.0004088878631592\n",
      "Epoch 10000, Loss 1.0000076293945312\n",
      "Epoch 11000, Loss 1.000000238418579\n",
      "Epoch 12000, Loss 1.0\n",
      "Epoch 13000, Loss 0.9999999403953552\n",
      "Epoch 14000, Loss 1.0\n",
      "Epoch 15000, Loss 1.0\n",
      "Epoch 16000, Loss 1.0\n",
      "Epoch 17000, Loss 1.0\n",
      "Epoch 18000, Loss 1.0\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 1.0\n",
      "Epoch 21000, Loss 1.0\n",
      "Epoch 22000, Loss 1.0\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 1.0\n",
      "Epoch 25000, Loss 1.0\n",
      "Epoch 26000, Loss 1.0\n",
      "Epoch 27000, Loss 0.9999999403953552\n",
      "Epoch 28000, Loss 1.0\n",
      "Epoch 29000, Loss 1.0\n",
      "Epoch 30000, Loss 0.9999999403953552\n",
      "Epoch 31000, Loss 1.0\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 1.0\n",
      "Epoch 34000, Loss 1.0\n",
      "Epoch 35000, Loss 1.0\n",
      "Epoch 36000, Loss 1.0\n",
      "Epoch 37000, Loss 1.0\n",
      "Epoch 38000, Loss 0.9999998807907104\n",
      "Epoch 39000, Loss 1.0\n",
      "Epoch 0, Loss 4.000000476837158\n",
      "Epoch 1000, Loss 3.985311508178711\n",
      "Epoch 2000, Loss 2.013845205307007\n",
      "Epoch 3000, Loss 1.999459147453308\n",
      "Epoch 4000, Loss 1.9960172176361084\n",
      "Epoch 5000, Loss 1.9711508750915527\n",
      "Epoch 6000, Loss 1.8133025169372559\n",
      "Epoch 7000, Loss 1.307239294052124\n",
      "Epoch 8000, Loss 1.020716905593872\n",
      "Epoch 9000, Loss 1.0004931688308716\n",
      "Epoch 10000, Loss 1.0000094175338745\n",
      "Epoch 11000, Loss 1.0000001192092896\n",
      "Epoch 12000, Loss 1.0\n",
      "Epoch 13000, Loss 1.0\n",
      "Epoch 14000, Loss 1.0\n",
      "Epoch 15000, Loss 1.0\n",
      "Epoch 16000, Loss 1.0\n",
      "Epoch 17000, Loss 0.9999999403953552\n",
      "Epoch 18000, Loss 1.0\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 1.0\n",
      "Epoch 21000, Loss 1.0\n",
      "Epoch 22000, Loss 1.0\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 1.0\n",
      "Epoch 25000, Loss 1.0\n",
      "Epoch 26000, Loss 1.0\n",
      "Epoch 27000, Loss 1.0\n",
      "Epoch 28000, Loss 1.0\n",
      "Epoch 29000, Loss 1.0\n",
      "Epoch 30000, Loss 1.0\n",
      "Epoch 31000, Loss 1.0\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 1.0\n",
      "Epoch 34000, Loss 1.0\n",
      "Epoch 35000, Loss 1.0\n",
      "Epoch 36000, Loss 1.0\n",
      "Epoch 37000, Loss 1.0\n",
      "Epoch 38000, Loss 1.0\n",
      "Epoch 39000, Loss 1.0\n",
      "Epoch 0, Loss 3.9999992847442627\n",
      "Epoch 1000, Loss 3.9849331378936768\n",
      "Epoch 2000, Loss 2.0132052898406982\n",
      "Epoch 3000, Loss 1.9994770288467407\n",
      "Epoch 4000, Loss 1.9961485862731934\n",
      "Epoch 5000, Loss 1.9720842838287354\n",
      "Epoch 6000, Loss 1.8186070919036865\n",
      "Epoch 7000, Loss 1.3165028095245361\n",
      "Epoch 8000, Loss 1.0219427347183228\n",
      "Epoch 9000, Loss 1.000526785850525\n",
      "Epoch 10000, Loss 1.0000100135803223\n",
      "Epoch 11000, Loss 1.0000001192092896\n",
      "Epoch 12000, Loss 1.0\n",
      "Epoch 13000, Loss 1.0000001192092896\n",
      "Epoch 14000, Loss 1.0\n",
      "Epoch 15000, Loss 1.0\n",
      "Epoch 16000, Loss 1.0\n",
      "Epoch 17000, Loss 1.0\n",
      "Epoch 18000, Loss 1.0\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 1.0\n",
      "Epoch 21000, Loss 1.0\n",
      "Epoch 22000, Loss 1.0000001192092896\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 1.0000001192092896\n",
      "Epoch 25000, Loss 1.0\n",
      "Epoch 26000, Loss 1.0\n",
      "Epoch 27000, Loss 1.0\n",
      "Epoch 28000, Loss 1.0\n",
      "Epoch 29000, Loss 1.0\n",
      "Epoch 30000, Loss 1.0\n",
      "Epoch 31000, Loss 1.0\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 1.0\n",
      "Epoch 34000, Loss 1.0000001192092896\n",
      "Epoch 35000, Loss 1.0\n",
      "Epoch 36000, Loss 1.0\n",
      "Epoch 37000, Loss 1.0\n",
      "Epoch 38000, Loss 1.0\n",
      "Epoch 39000, Loss 1.0000001192092896\n",
      "Epoch 0, Loss 3.999998092651367\n",
      "Epoch 1000, Loss 3.976628303527832\n",
      "Epoch 2000, Loss 2.0057196617126465\n",
      "Epoch 3000, Loss 1.9995027780532837\n",
      "Epoch 4000, Loss 1.996337652206421\n",
      "Epoch 5000, Loss 1.9734324216842651\n",
      "Epoch 6000, Loss 1.8263455629348755\n",
      "Epoch 7000, Loss 1.3306101560592651\n",
      "Epoch 8000, Loss 1.0239102840423584\n",
      "Epoch 9000, Loss 1.0005816221237183\n",
      "Epoch 10000, Loss 1.0000109672546387\n",
      "Epoch 11000, Loss 1.0000001192092896\n",
      "Epoch 12000, Loss 1.0\n",
      "Epoch 13000, Loss 0.9999999403953552\n",
      "Epoch 14000, Loss 0.9999998807907104\n",
      "Epoch 15000, Loss 0.9999999403953552\n",
      "Epoch 16000, Loss 0.9999998807907104\n",
      "Epoch 17000, Loss 0.9999999403953552\n",
      "Epoch 18000, Loss 0.9999999403953552\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 0.9999999403953552\n",
      "Epoch 21000, Loss 1.0\n",
      "Epoch 22000, Loss 0.9999998807907104\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 0.9999999403953552\n",
      "Epoch 25000, Loss 0.9999999403953552\n",
      "Epoch 26000, Loss 0.9999999403953552\n",
      "Epoch 27000, Loss 0.9999999403953552\n",
      "Epoch 28000, Loss 0.9999998807907104\n",
      "Epoch 29000, Loss 1.0\n",
      "Epoch 30000, Loss 0.9999999403953552\n",
      "Epoch 31000, Loss 0.9999999403953552\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 1.0\n",
      "Epoch 34000, Loss 1.0\n",
      "Epoch 35000, Loss 0.9999999403953552\n",
      "Epoch 36000, Loss 0.9999999403953552\n",
      "Epoch 37000, Loss 0.9999999403953552\n",
      "Epoch 38000, Loss 1.0\n",
      "Epoch 39000, Loss 1.0\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.9859232902526855\n",
      "Epoch 2000, Loss 2.014991521835327\n",
      "Epoch 3000, Loss 1.9995193481445312\n",
      "Epoch 4000, Loss 1.9964594841003418\n",
      "Epoch 5000, Loss 1.9743013381958008\n",
      "Epoch 6000, Loss 1.8313831090927124\n",
      "Epoch 7000, Loss 1.3401920795440674\n",
      "Epoch 8000, Loss 1.0253196954727173\n",
      "Epoch 9000, Loss 1.0006214380264282\n",
      "Epoch 10000, Loss 1.000011920928955\n",
      "Epoch 11000, Loss 1.0000001192092896\n",
      "Epoch 12000, Loss 1.0\n",
      "Epoch 13000, Loss 0.9999998807907104\n",
      "Epoch 14000, Loss 1.0\n",
      "Epoch 15000, Loss 0.9999999403953552\n",
      "Epoch 16000, Loss 1.0\n",
      "Epoch 17000, Loss 1.0\n",
      "Epoch 18000, Loss 1.0\n",
      "Epoch 19000, Loss 1.0\n",
      "Epoch 20000, Loss 1.0\n",
      "Epoch 21000, Loss 1.0\n",
      "Epoch 22000, Loss 0.9999998807907104\n",
      "Epoch 23000, Loss 1.0\n",
      "Epoch 24000, Loss 1.0\n",
      "Epoch 25000, Loss 1.0\n",
      "Epoch 26000, Loss 1.0\n",
      "Epoch 27000, Loss 0.9999999403953552\n",
      "Epoch 28000, Loss 0.9999999403953552\n",
      "Epoch 29000, Loss 1.0\n",
      "Epoch 30000, Loss 0.9999998807907104\n",
      "Epoch 31000, Loss 0.9999999403953552\n",
      "Epoch 32000, Loss 1.0\n",
      "Epoch 33000, Loss 1.0\n",
      "Epoch 34000, Loss 1.0\n",
      "Epoch 35000, Loss 1.0\n",
      "Epoch 36000, Loss 1.0\n",
      "Epoch 37000, Loss 0.9999999403953552\n",
      "Epoch 38000, Loss 0.9999999403953552\n",
      "Epoch 39000, Loss 1.0\n"
     ]
    }
   ],
   "source": [
    "\n",
    "hidden_size = 64\n",
    "\n",
    "class LinearNet(torch.nn.Module):\n",
    "    def __init__(self):\n",
    "        super(LinearNet, self).__init__()\n",
    "        self.fc1 = torch.nn.Linear(input_size, hidden_size, bias=False)\n",
    "        self.fc2 = torch.nn.Linear(hidden_size, output_size, bias=False)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = self.fc1(x)\n",
    "        x = self.fc2(x)\n",
    "        return x\n",
    "    \n",
    "    def get_hidden(self, x):\n",
    "        return self.fc1(x)\n",
    "model = LinearNet()\n",
    "# initialize the weights\n",
    "model.fc1.weight.data = torch.randn_like(model.fc1.weight.data)*0.0002\n",
    "model.fc2.weight.data = torch.randn_like(model.fc2.weight.data)*0.0002\n",
    "\n",
    "# load the data we do not need test data with pytorch dataloaders\n",
    "train_loader = DataLoader(torch.utils.data.TensorDataset(torch.tensor(train_inputs_test.T, dtype=torch.float32), \n",
    "                                                          torch.tensor(train_labels_test.T, dtype=torch.float32)), \n",
    "                          batch_size=n, shuffle=False)\n",
    "\n",
    "# define the optimizer\n",
    "optimizer = torch.optim.SGD(model.parameters(), lr=0.001)\n",
    "\n",
    "loss_list_linear_bias = []\n",
    "hidden_activations_linear_bias_list = []\n",
    "# n_runs = 10\n",
    "for run in range(n_runs):\n",
    "    # initialize the model\n",
    "    model = LinearNet()\n",
    "    # initialize the weights\n",
    "    model.fc1.weight.data = torch.randn_like(model.fc1.weight.data)*0.0002\n",
    "    model.fc2.weight.data = torch.randn_like(model.fc2.weight.data)*0.0002\n",
    "    # define the optimizer\n",
    "    optimizer = torch.optim.SGD(model.parameters(), lr=0.001)\n",
    "    # train the model\n",
    "    hidden_activations_linear_bias, losses_linear_bias = train(model, train_loader, optimizer)\n",
    "    hidden_activations_linear_bias_list.append(hidden_activations_linear_bias)\n",
    "    loss_list_linear_bias.append(losses_linear_bias)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.9999935626983643\n",
      "Epoch 2000, Loss 3.999967098236084\n",
      "Epoch 3000, Loss 3.9998342990875244\n",
      "Epoch 4000, Loss 3.999135971069336\n",
      "Epoch 5000, Loss 3.9953877925872803\n",
      "Epoch 6000, Loss 3.9750895500183105\n",
      "Epoch 7000, Loss 3.8671510219573975\n",
      "Epoch 8000, Loss 3.3677570819854736\n",
      "Epoch 9000, Loss 2.0634970664978027\n",
      "Epoch 10000, Loss 1.1046644449234009\n",
      "Epoch 11000, Loss 0.9087619781494141\n",
      "Epoch 12000, Loss 0.7926420569419861\n",
      "Epoch 13000, Loss 0.618332028388977\n",
      "Epoch 14000, Loss 0.39426177740097046\n",
      "Epoch 15000, Loss 0.1895645558834076\n",
      "Epoch 16000, Loss 0.06937024742364883\n",
      "Epoch 17000, Loss 0.021137788891792297\n",
      "Epoch 18000, Loss 0.005700846202671528\n",
      "Epoch 19000, Loss 0.0014407754642888904\n",
      "Epoch 20000, Loss 0.0003529682580847293\n",
      "Epoch 21000, Loss 8.513105422025546e-05\n",
      "Epoch 22000, Loss 2.0426876290002838e-05\n",
      "Epoch 23000, Loss 4.980412995791994e-06\n",
      "Epoch 24000, Loss 1.2523896657512523e-06\n",
      "Epoch 25000, Loss 3.219928146336315e-07\n",
      "Epoch 26000, Loss 8.37169835676832e-08\n",
      "Epoch 27000, Loss 2.4619337679609998e-08\n",
      "Epoch 28000, Loss 1.3643650653705208e-08\n",
      "Epoch 29000, Loss 1.0449464404871378e-08\n",
      "Epoch 30000, Loss 8.810040696971555e-09\n",
      "Epoch 31000, Loss 7.887277497786727e-09\n",
      "Epoch 32000, Loss 7.262213941316986e-09\n",
      "Epoch 33000, Loss 6.821858633543343e-09\n",
      "Epoch 34000, Loss 6.519144779559838e-09\n",
      "Epoch 35000, Loss 6.306011268719658e-09\n",
      "Epoch 36000, Loss 6.134315722050587e-09\n",
      "Epoch 37000, Loss 5.968452843063687e-09\n",
      "Epoch 38000, Loss 5.859256191342865e-09\n",
      "Epoch 39000, Loss 5.787642809451654e-09\n",
      "Epoch 0, Loss 3.999999761581421\n",
      "Epoch 1000, Loss 3.999992847442627\n",
      "Epoch 2000, Loss 3.9999632835388184\n",
      "Epoch 3000, Loss 3.9998114109039307\n",
      "Epoch 4000, Loss 3.9990077018737793\n",
      "Epoch 5000, Loss 3.9946837425231934\n",
      "Epoch 6000, Loss 3.971257209777832\n",
      "Epoch 7000, Loss 3.8470377922058105\n",
      "Epoch 8000, Loss 3.2809927463531494\n",
      "Epoch 9000, Loss 1.9041787385940552\n",
      "Epoch 10000, Loss 1.075798749923706\n",
      "Epoch 11000, Loss 0.9489186406135559\n",
      "Epoch 12000, Loss 0.8713619709014893\n",
      "Epoch 13000, Loss 0.7366525530815125\n",
      "Epoch 14000, Loss 0.5393539667129517\n",
      "Epoch 15000, Loss 0.3124341368675232\n",
      "Epoch 16000, Loss 0.1368260383605957\n",
      "Epoch 17000, Loss 0.048147041350603104\n",
      "Epoch 18000, Loss 0.014321568422019482\n",
      "Epoch 19000, Loss 0.0038179531693458557\n",
      "Epoch 20000, Loss 0.0009588118991814554\n",
      "Epoch 21000, Loss 0.00023375217278953642\n",
      "Epoch 22000, Loss 5.6213666539406404e-05\n",
      "Epoch 23000, Loss 1.345323289569933e-05\n",
      "Epoch 24000, Loss 3.2217301395576214e-06\n",
      "Epoch 25000, Loss 7.779428301546432e-07\n",
      "Epoch 26000, Loss 1.9078984792031406e-07\n",
      "Epoch 27000, Loss 4.952414656145265e-08\n",
      "Epoch 28000, Loss 1.5917546392074655e-08\n",
      "Epoch 29000, Loss 1.0809176664849929e-08\n",
      "Epoch 30000, Loss 8.897012016007011e-09\n",
      "Epoch 31000, Loss 7.999330087216094e-09\n",
      "Epoch 32000, Loss 7.466816498435946e-09\n",
      "Epoch 33000, Loss 7.120159128959358e-09\n",
      "Epoch 34000, Loss 6.868020818728837e-09\n",
      "Epoch 35000, Loss 6.64421673235438e-09\n",
      "Epoch 36000, Loss 6.453502177095061e-09\n",
      "Epoch 37000, Loss 6.313069622621015e-09\n",
      "Epoch 38000, Loss 6.192340862298806e-09\n",
      "Epoch 39000, Loss 6.076722680603552e-09\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.999992847442627\n",
      "Epoch 2000, Loss 3.999962091445923\n",
      "Epoch 3000, Loss 3.999800682067871\n",
      "Epoch 4000, Loss 3.998932123184204\n",
      "Epoch 5000, Loss 3.994192600250244\n",
      "Epoch 6000, Loss 3.9682748317718506\n",
      "Epoch 7000, Loss 3.8312556743621826\n",
      "Epoch 8000, Loss 3.2353105545043945\n",
      "Epoch 9000, Loss 1.923197627067566\n",
      "Epoch 10000, Loss 1.0976202487945557\n",
      "Epoch 11000, Loss 0.9452170729637146\n",
      "Epoch 12000, Loss 0.8671389818191528\n",
      "Epoch 13000, Loss 0.7353589534759521\n",
      "Epoch 14000, Loss 0.5339340567588806\n",
      "Epoch 15000, Loss 0.306081086397171\n",
      "Epoch 16000, Loss 0.13300272822380066\n",
      "Epoch 17000, Loss 0.043957822024822235\n",
      "Epoch 18000, Loss 0.011958476155996323\n",
      "Epoch 19000, Loss 0.0028870280366390944\n",
      "Epoch 20000, Loss 0.0006570040713995695\n",
      "Epoch 21000, Loss 0.00014600437134504318\n",
      "Epoch 22000, Loss 3.22876003338024e-05\n",
      "Epoch 23000, Loss 7.197335435193963e-06\n",
      "Epoch 24000, Loss 1.6356763126168516e-06\n",
      "Epoch 25000, Loss 3.8568666127503093e-07\n",
      "Epoch 26000, Loss 9.958952063016113e-08\n",
      "Epoch 27000, Loss 2.7749521791520237e-08\n",
      "Epoch 28000, Loss 1.3082471994607658e-08\n",
      "Epoch 29000, Loss 8.63539817430592e-09\n",
      "Epoch 30000, Loss 6.5872884924544906e-09\n",
      "Epoch 31000, Loss 5.687825321842865e-09\n",
      "Epoch 32000, Loss 5.256679092013883e-09\n",
      "Epoch 33000, Loss 4.96124386017982e-09\n",
      "Epoch 34000, Loss 4.792424235233739e-09\n",
      "Epoch 35000, Loss 4.660778873954996e-09\n",
      "Epoch 36000, Loss 4.543155629477269e-09\n",
      "Epoch 37000, Loss 4.4813437405366585e-09\n",
      "Epoch 38000, Loss 4.432850086999451e-09\n",
      "Epoch 39000, Loss 4.395944941393282e-09\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.999993085861206\n",
      "Epoch 2000, Loss 3.999966621398926\n",
      "Epoch 3000, Loss 3.9998371601104736\n",
      "Epoch 4000, Loss 3.9991815090179443\n",
      "Epoch 5000, Loss 3.9957633018493652\n",
      "Epoch 6000, Loss 3.977647304534912\n",
      "Epoch 7000, Loss 3.8823952674865723\n",
      "Epoch 8000, Loss 3.4331274032592773\n",
      "Epoch 9000, Loss 2.1489949226379395\n",
      "Epoch 10000, Loss 1.098403811454773\n",
      "Epoch 11000, Loss 0.8623275756835938\n",
      "Epoch 12000, Loss 0.7095988392829895\n",
      "Epoch 13000, Loss 0.5027586221694946\n",
      "Epoch 14000, Loss 0.28190383315086365\n",
      "Epoch 15000, Loss 0.12126004695892334\n",
      "Epoch 16000, Loss 0.042302437126636505\n",
      "Epoch 17000, Loss 0.013102016411721706\n",
      "Epoch 18000, Loss 0.0038930410519242287\n",
      "Epoch 19000, Loss 0.0011568069458007812\n",
      "Epoch 20000, Loss 0.0003486002387944609\n",
      "Epoch 21000, Loss 0.00010688933980418369\n",
      "Epoch 22000, Loss 3.324164936202578e-05\n",
      "Epoch 23000, Loss 1.0444487998029217e-05\n",
      "Epoch 24000, Loss 3.316132506370195e-06\n",
      "Epoch 25000, Loss 1.0658120572770713e-06\n",
      "Epoch 26000, Loss 3.4835886708606267e-07\n",
      "Epoch 27000, Loss 1.180581250537216e-07\n",
      "Epoch 28000, Loss 4.1604621969781874e-08\n",
      "Epoch 29000, Loss 1.8166403492614336e-08\n",
      "Epoch 30000, Loss 1.1378604725109653e-08\n",
      "Epoch 31000, Loss 8.958406461090362e-09\n",
      "Epoch 32000, Loss 7.836458593146745e-09\n",
      "Epoch 33000, Loss 7.117349376528637e-09\n",
      "Epoch 34000, Loss 6.568341426316238e-09\n",
      "Epoch 35000, Loss 6.112117034717812e-09\n",
      "Epoch 36000, Loss 5.731730645663902e-09\n",
      "Epoch 37000, Loss 5.4208664224120184e-09\n",
      "Epoch 38000, Loss 5.262672964079229e-09\n",
      "Epoch 39000, Loss 5.151321147423005e-09\n",
      "Epoch 0, Loss 3.9999990463256836\n",
      "Epoch 1000, Loss 3.999990940093994\n",
      "Epoch 2000, Loss 3.999955892562866\n",
      "Epoch 3000, Loss 3.999779224395752\n",
      "Epoch 4000, Loss 3.9988622665405273\n",
      "Epoch 5000, Loss 3.9940133094787598\n",
      "Epoch 6000, Loss 3.968118667602539\n",
      "Epoch 7000, Loss 3.8331286907196045\n",
      "Epoch 8000, Loss 3.240668296813965\n",
      "Epoch 9000, Loss 1.8846499919891357\n",
      "Epoch 10000, Loss 1.022364616394043\n",
      "Epoch 11000, Loss 0.8178681135177612\n",
      "Epoch 12000, Loss 0.6565268635749817\n",
      "Epoch 13000, Loss 0.4525902569293976\n",
      "Epoch 14000, Loss 0.24810734391212463\n",
      "Epoch 15000, Loss 0.10528590530157089\n",
      "Epoch 16000, Loss 0.0357329435646534\n",
      "Epoch 17000, Loss 0.010368399322032928\n",
      "Epoch 18000, Loss 0.0027558556757867336\n",
      "Epoch 19000, Loss 0.0007028941763564944\n",
      "Epoch 20000, Loss 0.00017674057744443417\n",
      "Epoch 21000, Loss 4.4429139961721376e-05\n",
      "Epoch 22000, Loss 1.1250659554207232e-05\n",
      "Epoch 23000, Loss 2.88349951915734e-06\n",
      "Epoch 24000, Loss 7.534277983722859e-07\n",
      "Epoch 25000, Loss 2.0366449859920976e-07\n",
      "Epoch 26000, Loss 5.593731344788466e-08\n",
      "Epoch 27000, Loss 1.8514887401011038e-08\n",
      "Epoch 28000, Loss 1.0545502249215133e-08\n",
      "Epoch 29000, Loss 8.21309686926952e-09\n",
      "Epoch 30000, Loss 7.201943041934555e-09\n",
      "Epoch 31000, Loss 6.568322774569424e-09\n",
      "Epoch 32000, Loss 6.130340679533219e-09\n",
      "Epoch 33000, Loss 5.804058567093762e-09\n",
      "Epoch 34000, Loss 5.624846366458769e-09\n",
      "Epoch 35000, Loss 5.507097444734654e-09\n",
      "Epoch 36000, Loss 5.3808482114448e-09\n",
      "Epoch 37000, Loss 5.274655823228613e-09\n",
      "Epoch 38000, Loss 5.230649691156941e-09\n",
      "Epoch 39000, Loss 5.188372842468425e-09\n",
      "Epoch 0, Loss 3.999999761581421\n",
      "Epoch 1000, Loss 3.9999914169311523\n",
      "Epoch 2000, Loss 3.999955177307129\n",
      "Epoch 3000, Loss 3.9997684955596924\n",
      "Epoch 4000, Loss 3.9987776279449463\n",
      "Epoch 5000, Loss 3.993434190750122\n",
      "Epoch 6000, Loss 3.9644854068756104\n",
      "Epoch 7000, Loss 3.8125011920928955\n",
      "Epoch 8000, Loss 3.154996871948242\n",
      "Epoch 9000, Loss 1.760713815689087\n",
      "Epoch 10000, Loss 1.0174803733825684\n",
      "Epoch 11000, Loss 0.8733782172203064\n",
      "Epoch 12000, Loss 0.7580751776695251\n",
      "Epoch 13000, Loss 0.5938957333564758\n",
      "Epoch 14000, Loss 0.39947861433029175\n",
      "Epoch 15000, Loss 0.22084584832191467\n",
      "Epoch 16000, Loss 0.09657002985477448\n",
      "Epoch 17000, Loss 0.03327227383852005\n",
      "Epoch 18000, Loss 0.009651953354477882\n",
      "Epoch 19000, Loss 0.002511846600100398\n",
      "Epoch 20000, Loss 0.0006156062008813024\n",
      "Epoch 21000, Loss 0.00014655498671345413\n",
      "Epoch 22000, Loss 3.447549170232378e-05\n",
      "Epoch 23000, Loss 8.099912520265207e-06\n",
      "Epoch 24000, Loss 1.9193200841982616e-06\n",
      "Epoch 25000, Loss 4.66388314634969e-07\n",
      "Epoch 26000, Loss 1.178560680159535e-07\n",
      "Epoch 27000, Loss 3.373554946506374e-08\n",
      "Epoch 28000, Loss 1.4791306846007046e-08\n",
      "Epoch 29000, Loss 9.433215097942593e-09\n",
      "Epoch 30000, Loss 7.824644931986313e-09\n",
      "Epoch 31000, Loss 7.07155001222759e-09\n",
      "Epoch 32000, Loss 6.597848045686305e-09\n",
      "Epoch 33000, Loss 6.277208086657993e-09\n",
      "Epoch 34000, Loss 6.060048907130522e-09\n",
      "Epoch 35000, Loss 5.936224400926449e-09\n",
      "Epoch 36000, Loss 5.835796734743326e-09\n",
      "Epoch 37000, Loss 5.74435210509705e-09\n",
      "Epoch 38000, Loss 5.655180768115997e-09\n",
      "Epoch 39000, Loss 5.586176410332655e-09\n",
      "Epoch 0, Loss 3.999999523162842\n",
      "Epoch 1000, Loss 3.999990463256836\n",
      "Epoch 2000, Loss 3.9999523162841797\n",
      "Epoch 3000, Loss 3.999758243560791\n",
      "Epoch 4000, Loss 3.998746395111084\n",
      "Epoch 5000, Loss 3.993370771408081\n",
      "Epoch 6000, Loss 3.964587688446045\n",
      "Epoch 7000, Loss 3.8145012855529785\n",
      "Epoch 8000, Loss 3.161555290222168\n",
      "Epoch 9000, Loss 1.754595398902893\n",
      "Epoch 10000, Loss 1.0111615657806396\n",
      "Epoch 11000, Loss 0.8542441129684448\n",
      "Epoch 12000, Loss 0.7127319574356079\n",
      "Epoch 13000, Loss 0.5215491056442261\n",
      "Epoch 14000, Loss 0.31059372425079346\n",
      "Epoch 15000, Loss 0.1445522904396057\n",
      "Epoch 16000, Loss 0.053639207035303116\n",
      "Epoch 17000, Loss 0.016782136633992195\n",
      "Epoch 18000, Loss 0.0047433157451450825\n",
      "Epoch 19000, Loss 0.0012833537766709924\n",
      "Epoch 20000, Loss 0.0003420316497795284\n",
      "Epoch 21000, Loss 9.12972682272084e-05\n",
      "Epoch 22000, Loss 2.4691165890544653e-05\n",
      "Epoch 23000, Loss 6.835693056927994e-06\n",
      "Epoch 24000, Loss 1.95404868463811e-06\n",
      "Epoch 25000, Loss 5.864552576895221e-07\n",
      "Epoch 26000, Loss 1.8780622212943854e-07\n",
      "Epoch 27000, Loss 6.256745876953573e-08\n",
      "Epoch 28000, Loss 2.6244807216357913e-08\n",
      "Epoch 29000, Loss 1.4973045026067666e-08\n",
      "Epoch 30000, Loss 1.0737426059392874e-08\n",
      "Epoch 31000, Loss 8.793640482451792e-09\n",
      "Epoch 32000, Loss 7.820028180560712e-09\n",
      "Epoch 33000, Loss 7.229483234283407e-09\n",
      "Epoch 34000, Loss 6.839136812430979e-09\n",
      "Epoch 35000, Loss 6.514775829913333e-09\n",
      "Epoch 36000, Loss 6.281584141731855e-09\n",
      "Epoch 37000, Loss 6.085659975951785e-09\n",
      "Epoch 38000, Loss 5.905319344634563e-09\n",
      "Epoch 39000, Loss 5.755974363808036e-09\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.9999914169311523\n",
      "Epoch 2000, Loss 3.999955892562866\n",
      "Epoch 3000, Loss 3.9997758865356445\n",
      "Epoch 4000, Loss 3.9988293647766113\n",
      "Epoch 5000, Loss 3.993770122528076\n",
      "Epoch 6000, Loss 3.966533660888672\n",
      "Epoch 7000, Loss 3.8238022327423096\n",
      "Epoch 8000, Loss 3.198455572128296\n",
      "Epoch 9000, Loss 1.8158118724822998\n",
      "Epoch 10000, Loss 1.0461703538894653\n",
      "Epoch 11000, Loss 0.9068294763565063\n",
      "Epoch 12000, Loss 0.7936112880706787\n",
      "Epoch 13000, Loss 0.6194851994514465\n",
      "Epoch 14000, Loss 0.40269356966018677\n",
      "Epoch 15000, Loss 0.20310741662979126\n",
      "Epoch 16000, Loss 0.07958067953586578\n",
      "Epoch 17000, Loss 0.02609134465456009\n",
      "Epoch 18000, Loss 0.007527042180299759\n",
      "Epoch 19000, Loss 0.0020326850935816765\n",
      "Epoch 20000, Loss 0.0005303571815602481\n",
      "Epoch 21000, Loss 0.00013598083751276135\n",
      "Epoch 22000, Loss 3.471162563073449e-05\n",
      "Epoch 23000, Loss 8.876677384250797e-06\n",
      "Epoch 24000, Loss 2.2847091258881846e-06\n",
      "Epoch 25000, Loss 5.974662826702115e-07\n",
      "Epoch 26000, Loss 1.6082991294297244e-07\n",
      "Epoch 27000, Loss 4.5366441270289215e-08\n",
      "Epoch 28000, Loss 1.722284004301855e-08\n",
      "Epoch 29000, Loss 1.0514308534936845e-08\n",
      "Epoch 30000, Loss 8.684161834082715e-09\n",
      "Epoch 31000, Loss 7.858727890663886e-09\n",
      "Epoch 32000, Loss 7.346868891033864e-09\n",
      "Epoch 33000, Loss 7.034421933838075e-09\n",
      "Epoch 34000, Loss 6.7625522959247064e-09\n",
      "Epoch 35000, Loss 6.565246124523583e-09\n",
      "Epoch 36000, Loss 6.446763567424796e-09\n",
      "Epoch 37000, Loss 6.383463979631188e-09\n",
      "Epoch 38000, Loss 6.344781589007198e-09\n",
      "Epoch 39000, Loss 6.300162169736723e-09\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.9999914169311523\n",
      "Epoch 2000, Loss 3.9999568462371826\n",
      "Epoch 3000, Loss 3.9997830390930176\n",
      "Epoch 4000, Loss 3.9988811016082764\n",
      "Epoch 5000, Loss 3.994105100631714\n",
      "Epoch 6000, Loss 3.968569755554199\n",
      "Epoch 7000, Loss 3.8348522186279297\n",
      "Epoch 8000, Loss 3.23704195022583\n",
      "Epoch 9000, Loss 1.8401103019714355\n",
      "Epoch 10000, Loss 1.0286060571670532\n",
      "Epoch 11000, Loss 0.8612250089645386\n",
      "Epoch 12000, Loss 0.7211833596229553\n",
      "Epoch 13000, Loss 0.5242539644241333\n",
      "Epoch 14000, Loss 0.30664998292922974\n",
      "Epoch 15000, Loss 0.1381736546754837\n",
      "Epoch 16000, Loss 0.049252405762672424\n",
      "Epoch 17000, Loss 0.01478959247469902\n",
      "Epoch 18000, Loss 0.004011142998933792\n",
      "Epoch 19000, Loss 0.0010389413218945265\n",
      "Epoch 20000, Loss 0.0002637890574987978\n",
      "Epoch 21000, Loss 6.673777534160763e-05\n",
      "Epoch 22000, Loss 1.704496389720589e-05\n",
      "Epoch 23000, Loss 4.458000603335677e-06\n",
      "Epoch 24000, Loss 1.2188022537884535e-06\n",
      "Epoch 25000, Loss 3.5806769460577925e-07\n",
      "Epoch 26000, Loss 1.1299169955236721e-07\n",
      "Epoch 27000, Loss 4.252730789744419e-08\n",
      "Epoch 28000, Loss 1.9741143830742658e-08\n",
      "Epoch 29000, Loss 1.1259896126603053e-08\n",
      "Epoch 30000, Loss 7.545461144786714e-09\n",
      "Epoch 31000, Loss 6.099897476019578e-09\n",
      "Epoch 32000, Loss 5.368325783905448e-09\n",
      "Epoch 33000, Loss 4.970921452240873e-09\n",
      "Epoch 34000, Loss 4.745365878022767e-09\n",
      "Epoch 35000, Loss 4.622763949413411e-09\n",
      "Epoch 36000, Loss 4.525772201446898e-09\n",
      "Epoch 37000, Loss 4.4467940440995335e-09\n",
      "Epoch 38000, Loss 4.382182616780028e-09\n",
      "Epoch 39000, Loss 4.327840308349096e-09\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.9999923706054688\n",
      "Epoch 2000, Loss 3.9999632835388184\n",
      "Epoch 3000, Loss 3.9998202323913574\n",
      "Epoch 4000, Loss 3.9990921020507812\n",
      "Epoch 5000, Loss 3.99528431892395\n",
      "Epoch 6000, Loss 3.9750754833221436\n",
      "Epoch 7000, Loss 3.869020938873291\n",
      "Epoch 8000, Loss 3.3768491744995117\n",
      "Epoch 9000, Loss 2.049617290496826\n",
      "Epoch 10000, Loss 1.0502146482467651\n",
      "Epoch 11000, Loss 0.8195293545722961\n",
      "Epoch 12000, Loss 0.6621530055999756\n",
      "Epoch 13000, Loss 0.47464630007743835\n",
      "Epoch 14000, Loss 0.28336283564567566\n",
      "Epoch 15000, Loss 0.1377609521150589\n",
      "Epoch 16000, Loss 0.05470215529203415\n",
      "Epoch 17000, Loss 0.01926812157034874\n",
      "Epoch 18000, Loss 0.006303170695900917\n",
      "Epoch 19000, Loss 0.0019835173152387142\n",
      "Epoch 20000, Loss 0.0006127297529019415\n",
      "Epoch 21000, Loss 0.0001878147741081193\n",
      "Epoch 22000, Loss 5.747636532760225e-05\n",
      "Epoch 23000, Loss 1.763365617080126e-05\n",
      "Epoch 24000, Loss 5.446404884423828e-06\n",
      "Epoch 25000, Loss 1.6981565522655728e-06\n",
      "Epoch 26000, Loss 5.398794655775419e-07\n",
      "Epoch 27000, Loss 1.7785004047254915e-07\n",
      "Epoch 28000, Loss 6.131242713536267e-08\n",
      "Epoch 29000, Loss 2.4877612858631437e-08\n",
      "Epoch 30000, Loss 1.4486172261740649e-08\n",
      "Epoch 31000, Loss 1.0836060049257412e-08\n",
      "Epoch 32000, Loss 9.244333298852325e-09\n",
      "Epoch 33000, Loss 8.324263056636028e-09\n",
      "Epoch 34000, Loss 7.708420568519614e-09\n",
      "Epoch 35000, Loss 7.345863028973554e-09\n",
      "Epoch 36000, Loss 7.083735376056666e-09\n",
      "Epoch 37000, Loss 6.882087788540048e-09\n",
      "Epoch 38000, Loss 6.773636318513354e-09\n",
      "Epoch 39000, Loss 6.678932962245199e-09\n"
     ]
    }
   ],
   "source": [
    "input_size = 2\n",
    "hidden_size = 64\n",
    "output_size = 4\n",
    "\n",
    "train_inputs = train_inputs_test[1:,:]\n",
    "train_labels = train_labels_test\n",
    "\n",
    "class ReluNet(torch.nn.Module):\n",
    "    def __init__(self):\n",
    "        super(ReluNet, self).__init__()\n",
    "        self.fc1 = torch.nn.Linear(input_size, hidden_size, bias=False)\n",
    "        self.fc2 = torch.nn.Linear(hidden_size, output_size, bias=False)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = torch.relu(self.fc1(x))\n",
    "        x = self.fc2(x)\n",
    "        return x\n",
    "    \n",
    "    def get_hidden(self, x):\n",
    "        return torch.relu(self.fc1(x))\n",
    "\n",
    "model = ReluNet()\n",
    "# initialize the weights\n",
    "model.fc1.weight.data = torch.randn_like(model.fc1.weight.data)*0.0002\n",
    "model.fc2.weight.data = torch.randn_like(model.fc2.weight.data)*0.0002\n",
    "\n",
    "# load the data we do not need test data with pytorch dataloaders\n",
    "train_loader = DataLoader(torch.utils.data.TensorDataset(torch.tensor(train_inputs.T, dtype=torch.float32), \n",
    "                                                          torch.tensor(train_labels.T, dtype=torch.float32)), \n",
    "                          batch_size=n, shuffle=False)\n",
    "\n",
    "# define the optimizer\n",
    "optimizer = torch.optim.SGD(model.parameters(), lr=0.001)\n",
    "\n",
    "loss_list_relu_no_bias = []\n",
    "hidden_activations_relu_no_bias_list = []\n",
    "# n_runs = 10\n",
    "for run in range(n_runs):\n",
    "    # initialize the model\n",
    "    model = ReluNet()\n",
    "    # initialize the weights\n",
    "    model.fc1.weight.data = torch.randn_like(model.fc1.weight.data)*0.0002\n",
    "    model.fc2.weight.data = torch.randn_like(model.fc2.weight.data)*0.0002\n",
    "    # define the optimizer\n",
    "    optimizer = torch.optim.SGD(model.parameters(), lr=0.001)\n",
    "    # train the model\n",
    "    hidden_activations_relu_no_bias, losses_relu_no_bias = train(model, train_loader, optimizer)\n",
    "    hidden_activations_relu_no_bias_list.append(hidden_activations_relu_no_bias)\n",
    "    loss_list_relu_no_bias.append(losses_relu_no_bias)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0, Loss 4.000000476837158\n",
      "Epoch 1000, Loss 3.999993085861206\n",
      "Epoch 2000, Loss 3.9999475479125977\n",
      "Epoch 3000, Loss 3.99961256980896\n",
      "Epoch 4000, Loss 3.997144937515259\n",
      "Epoch 5000, Loss 3.979210138320923\n",
      "Epoch 6000, Loss 3.860591173171997\n",
      "Epoch 7000, Loss 3.402724027633667\n",
      "Epoch 8000, Loss 3.0361716747283936\n",
      "Epoch 9000, Loss 3.000945568084717\n",
      "Epoch 10000, Loss 3.000018358230591\n",
      "Epoch 11000, Loss 3.000000476837158\n",
      "Epoch 12000, Loss 3.0\n",
      "Epoch 13000, Loss 2.999999761581421\n",
      "Epoch 14000, Loss 3.0\n",
      "Epoch 15000, Loss 3.0\n",
      "Epoch 16000, Loss 3.0\n",
      "Epoch 17000, Loss 3.0\n",
      "Epoch 18000, Loss 3.0\n",
      "Epoch 19000, Loss 3.0\n",
      "Epoch 20000, Loss 3.0\n",
      "Epoch 21000, Loss 3.0\n",
      "Epoch 22000, Loss 3.0\n",
      "Epoch 23000, Loss 3.0\n",
      "Epoch 24000, Loss 3.0\n",
      "Epoch 25000, Loss 3.0\n",
      "Epoch 26000, Loss 3.0\n",
      "Epoch 27000, Loss 3.0\n",
      "Epoch 28000, Loss 3.0\n",
      "Epoch 29000, Loss 3.0\n",
      "Epoch 30000, Loss 3.0\n",
      "Epoch 31000, Loss 3.0\n",
      "Epoch 32000, Loss 3.0\n",
      "Epoch 33000, Loss 3.0\n",
      "Epoch 34000, Loss 3.0\n",
      "Epoch 35000, Loss 3.0\n",
      "Epoch 36000, Loss 3.0\n",
      "Epoch 37000, Loss 3.0\n",
      "Epoch 38000, Loss 3.0\n",
      "Epoch 39000, Loss 3.0\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.999990701675415\n",
      "Epoch 2000, Loss 3.999929666519165\n",
      "Epoch 3000, Loss 3.999479293823242\n",
      "Epoch 4000, Loss 3.996166706085205\n",
      "Epoch 5000, Loss 3.9722135066986084\n",
      "Epoch 6000, Loss 3.819345712661743\n",
      "Epoch 7000, Loss 3.3178176879882812\n",
      "Epoch 8000, Loss 3.022120952606201\n",
      "Epoch 9000, Loss 3.0005316734313965\n",
      "Epoch 10000, Loss 3.0000102519989014\n",
      "Epoch 11000, Loss 3.000000238418579\n",
      "Epoch 12000, Loss 3.000000238418579\n",
      "Epoch 13000, Loss 2.999999761581421\n",
      "Epoch 14000, Loss 3.0\n",
      "Epoch 15000, Loss 3.0\n",
      "Epoch 16000, Loss 3.0\n",
      "Epoch 17000, Loss 3.0\n",
      "Epoch 18000, Loss 3.0\n",
      "Epoch 19000, Loss 3.0\n",
      "Epoch 20000, Loss 3.0\n",
      "Epoch 21000, Loss 3.0\n",
      "Epoch 22000, Loss 3.0\n",
      "Epoch 23000, Loss 3.0\n",
      "Epoch 24000, Loss 3.0\n",
      "Epoch 25000, Loss 3.0\n",
      "Epoch 26000, Loss 3.0\n",
      "Epoch 27000, Loss 3.0\n",
      "Epoch 28000, Loss 3.0\n",
      "Epoch 29000, Loss 3.0\n",
      "Epoch 30000, Loss 3.0\n",
      "Epoch 31000, Loss 3.0\n",
      "Epoch 32000, Loss 3.0\n",
      "Epoch 33000, Loss 3.0\n",
      "Epoch 34000, Loss 3.0\n",
      "Epoch 35000, Loss 3.0\n",
      "Epoch 36000, Loss 3.0\n",
      "Epoch 37000, Loss 3.0\n",
      "Epoch 38000, Loss 3.0\n",
      "Epoch 39000, Loss 3.0\n",
      "Epoch 0, Loss 3.999999761581421\n",
      "Epoch 1000, Loss 3.9999914169311523\n",
      "Epoch 2000, Loss 3.9999375343322754\n",
      "Epoch 3000, Loss 3.9995388984680176\n",
      "Epoch 4000, Loss 3.9966073036193848\n",
      "Epoch 5000, Loss 3.975355863571167\n",
      "Epoch 6000, Loss 3.8375492095947266\n",
      "Epoch 7000, Loss 3.3523712158203125\n",
      "Epoch 8000, Loss 3.027200937271118\n",
      "Epoch 9000, Loss 3.000675678253174\n",
      "Epoch 10000, Loss 3.0000128746032715\n",
      "Epoch 11000, Loss 3.000000476837158\n",
      "Epoch 12000, Loss 3.0\n",
      "Epoch 13000, Loss 3.000000238418579\n",
      "Epoch 14000, Loss 3.0\n",
      "Epoch 15000, Loss 3.000000238418579\n",
      "Epoch 16000, Loss 3.000000238418579\n",
      "Epoch 17000, Loss 3.0\n",
      "Epoch 18000, Loss 3.0\n",
      "Epoch 19000, Loss 3.0\n",
      "Epoch 20000, Loss 3.0\n",
      "Epoch 21000, Loss 3.0\n",
      "Epoch 22000, Loss 3.0\n",
      "Epoch 23000, Loss 3.0\n",
      "Epoch 24000, Loss 3.0\n",
      "Epoch 25000, Loss 3.0\n",
      "Epoch 26000, Loss 3.0\n",
      "Epoch 27000, Loss 3.0\n",
      "Epoch 28000, Loss 3.0\n",
      "Epoch 29000, Loss 3.0\n",
      "Epoch 30000, Loss 3.0\n",
      "Epoch 31000, Loss 3.0\n",
      "Epoch 32000, Loss 3.0\n",
      "Epoch 33000, Loss 3.0\n",
      "Epoch 34000, Loss 3.0\n",
      "Epoch 35000, Loss 3.0\n",
      "Epoch 36000, Loss 3.0\n",
      "Epoch 37000, Loss 3.0\n",
      "Epoch 38000, Loss 3.0\n",
      "Epoch 39000, Loss 3.0\n",
      "Epoch 0, Loss 3.9999992847442627\n",
      "Epoch 1000, Loss 3.999988555908203\n",
      "Epoch 2000, Loss 3.9999136924743652\n",
      "Epoch 3000, Loss 3.99936580657959\n",
      "Epoch 4000, Loss 3.9953360557556152\n",
      "Epoch 5000, Loss 3.9663257598876953\n",
      "Epoch 6000, Loss 3.7865753173828125\n",
      "Epoch 7000, Loss 3.2651052474975586\n",
      "Epoch 8000, Loss 3.015744209289551\n",
      "Epoch 9000, Loss 3.000361680984497\n",
      "Epoch 10000, Loss 3.000006675720215\n",
      "Epoch 11000, Loss 3.0\n",
      "Epoch 12000, Loss 3.0\n",
      "Epoch 13000, Loss 3.0\n",
      "Epoch 14000, Loss 3.0\n",
      "Epoch 15000, Loss 3.0\n",
      "Epoch 16000, Loss 3.0\n",
      "Epoch 17000, Loss 3.0\n",
      "Epoch 18000, Loss 3.0\n",
      "Epoch 19000, Loss 3.0\n",
      "Epoch 20000, Loss 3.0\n",
      "Epoch 21000, Loss 3.0\n",
      "Epoch 22000, Loss 3.0\n",
      "Epoch 23000, Loss 3.0\n",
      "Epoch 24000, Loss 3.0\n",
      "Epoch 25000, Loss 3.0\n",
      "Epoch 26000, Loss 3.0\n",
      "Epoch 27000, Loss 3.0\n",
      "Epoch 28000, Loss 3.0\n",
      "Epoch 29000, Loss 3.000000238418579\n",
      "Epoch 30000, Loss 3.0\n",
      "Epoch 31000, Loss 3.0\n",
      "Epoch 32000, Loss 3.0\n",
      "Epoch 33000, Loss 3.0\n",
      "Epoch 34000, Loss 3.0\n",
      "Epoch 35000, Loss 3.0\n",
      "Epoch 36000, Loss 3.000000238418579\n",
      "Epoch 37000, Loss 3.000000238418579\n",
      "Epoch 38000, Loss 3.000000238418579\n",
      "Epoch 39000, Loss 3.0\n",
      "Epoch 0, Loss 4.000000953674316\n",
      "Epoch 1000, Loss 3.999993085861206\n",
      "Epoch 2000, Loss 3.999945878982544\n",
      "Epoch 3000, Loss 3.9996016025543213\n",
      "Epoch 4000, Loss 3.997063636779785\n",
      "Epoch 5000, Loss 3.9786264896392822\n",
      "Epoch 6000, Loss 3.857048511505127\n",
      "Epoch 7000, Loss 3.3944478034973145\n",
      "Epoch 8000, Loss 3.0345540046691895\n",
      "Epoch 9000, Loss 3.0008952617645264\n",
      "Epoch 10000, Loss 3.0000171661376953\n",
      "Epoch 11000, Loss 3.000000476837158\n",
      "Epoch 12000, Loss 3.0\n",
      "Epoch 13000, Loss 2.999999761581421\n",
      "Epoch 14000, Loss 3.0\n",
      "Epoch 15000, Loss 3.0\n",
      "Epoch 16000, Loss 3.0\n",
      "Epoch 17000, Loss 3.0\n",
      "Epoch 18000, Loss 3.0\n",
      "Epoch 19000, Loss 3.0\n",
      "Epoch 20000, Loss 3.0\n",
      "Epoch 21000, Loss 3.0\n",
      "Epoch 22000, Loss 3.0\n",
      "Epoch 23000, Loss 3.0\n",
      "Epoch 24000, Loss 3.0\n",
      "Epoch 25000, Loss 3.0\n",
      "Epoch 26000, Loss 3.0\n",
      "Epoch 27000, Loss 3.0\n",
      "Epoch 28000, Loss 3.0\n",
      "Epoch 29000, Loss 3.0\n",
      "Epoch 30000, Loss 3.0\n",
      "Epoch 31000, Loss 3.0\n",
      "Epoch 32000, Loss 3.0\n",
      "Epoch 33000, Loss 3.0\n",
      "Epoch 34000, Loss 3.0\n",
      "Epoch 35000, Loss 3.0\n",
      "Epoch 36000, Loss 3.0\n",
      "Epoch 37000, Loss 3.0\n",
      "Epoch 38000, Loss 3.0\n",
      "Epoch 39000, Loss 3.0\n",
      "Epoch 0, Loss 3.999999523162842\n",
      "Epoch 1000, Loss 3.9999899864196777\n",
      "Epoch 2000, Loss 3.9999213218688965\n",
      "Epoch 3000, Loss 3.999420166015625\n",
      "Epoch 4000, Loss 3.9957327842712402\n",
      "Epoch 5000, Loss 3.9691314697265625\n",
      "Epoch 6000, Loss 3.80197811126709\n",
      "Epoch 7000, Loss 3.288503408432007\n",
      "Epoch 8000, Loss 3.0183887481689453\n",
      "Epoch 9000, Loss 3.0004305839538574\n",
      "Epoch 10000, Loss 3.0000083446502686\n",
      "Epoch 11000, Loss 3.000000238418579\n",
      "Epoch 12000, Loss 3.0\n",
      "Epoch 13000, Loss 2.999999761581421\n",
      "Epoch 14000, Loss 3.000000238418579\n",
      "Epoch 15000, Loss 2.999999761581421\n",
      "Epoch 16000, Loss 2.999999761581421\n",
      "Epoch 17000, Loss 2.999999761581421\n",
      "Epoch 18000, Loss 2.999999761581421\n",
      "Epoch 19000, Loss 2.999999761581421\n",
      "Epoch 20000, Loss 2.999999761581421\n",
      "Epoch 21000, Loss 2.999999761581421\n",
      "Epoch 22000, Loss 2.999999761581421\n",
      "Epoch 23000, Loss 2.999999761581421\n",
      "Epoch 24000, Loss 2.999999761581421\n",
      "Epoch 25000, Loss 2.999999761581421\n",
      "Epoch 26000, Loss 2.999999761581421\n",
      "Epoch 27000, Loss 2.999999761581421\n",
      "Epoch 28000, Loss 2.999999761581421\n",
      "Epoch 29000, Loss 2.999999761581421\n",
      "Epoch 30000, Loss 2.999999761581421\n",
      "Epoch 31000, Loss 2.999999761581421\n",
      "Epoch 32000, Loss 2.999999761581421\n",
      "Epoch 33000, Loss 2.999999761581421\n",
      "Epoch 34000, Loss 2.999999761581421\n",
      "Epoch 35000, Loss 2.999999761581421\n",
      "Epoch 36000, Loss 2.999999761581421\n",
      "Epoch 37000, Loss 2.999999761581421\n",
      "Epoch 38000, Loss 2.999999761581421\n",
      "Epoch 39000, Loss 2.999999761581421\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.9999914169311523\n",
      "Epoch 2000, Loss 3.999936103820801\n",
      "Epoch 3000, Loss 3.9995272159576416\n",
      "Epoch 4000, Loss 3.996518611907959\n",
      "Epoch 5000, Loss 3.974724292755127\n",
      "Epoch 6000, Loss 3.833850860595703\n",
      "Epoch 7000, Loss 3.3450047969818115\n",
      "Epoch 8000, Loss 3.0260508060455322\n",
      "Epoch 9000, Loss 3.0006422996520996\n",
      "Epoch 10000, Loss 3.0000126361846924\n",
      "Epoch 11000, Loss 3.000000238418579\n",
      "Epoch 12000, Loss 3.000000238418579\n",
      "Epoch 13000, Loss 3.000000238418579\n",
      "Epoch 14000, Loss 3.0\n",
      "Epoch 15000, Loss 3.0\n",
      "Epoch 16000, Loss 3.0\n",
      "Epoch 17000, Loss 3.0\n",
      "Epoch 18000, Loss 3.0\n",
      "Epoch 19000, Loss 3.0\n",
      "Epoch 20000, Loss 2.999999523162842\n",
      "Epoch 21000, Loss 3.0\n",
      "Epoch 22000, Loss 3.0\n",
      "Epoch 23000, Loss 3.0\n",
      "Epoch 24000, Loss 3.0\n",
      "Epoch 25000, Loss 3.0\n",
      "Epoch 26000, Loss 3.0\n",
      "Epoch 27000, Loss 3.0\n",
      "Epoch 28000, Loss 3.0\n",
      "Epoch 29000, Loss 3.0\n",
      "Epoch 30000, Loss 3.0\n",
      "Epoch 31000, Loss 3.0\n",
      "Epoch 32000, Loss 3.0\n",
      "Epoch 33000, Loss 3.0\n",
      "Epoch 34000, Loss 2.999999523162842\n",
      "Epoch 35000, Loss 3.0\n",
      "Epoch 36000, Loss 3.0\n",
      "Epoch 37000, Loss 3.0\n",
      "Epoch 38000, Loss 3.0\n",
      "Epoch 39000, Loss 3.0\n",
      "Epoch 0, Loss 3.9999992847442627\n",
      "Epoch 1000, Loss 3.9999887943267822\n",
      "Epoch 2000, Loss 3.9999160766601562\n",
      "Epoch 3000, Loss 3.9993810653686523\n",
      "Epoch 4000, Loss 3.9954450130462646\n",
      "Epoch 5000, Loss 3.9670958518981934\n",
      "Epoch 6000, Loss 3.790764570236206\n",
      "Epoch 7000, Loss 3.271244764328003\n",
      "Epoch 8000, Loss 3.016411304473877\n",
      "Epoch 9000, Loss 3.0003786087036133\n",
      "Epoch 10000, Loss 3.000007152557373\n",
      "Epoch 11000, Loss 3.000000238418579\n",
      "Epoch 12000, Loss 3.000000238418579\n",
      "Epoch 13000, Loss 3.0\n",
      "Epoch 14000, Loss 3.0\n",
      "Epoch 15000, Loss 3.0\n",
      "Epoch 16000, Loss 3.0\n",
      "Epoch 17000, Loss 3.0\n",
      "Epoch 18000, Loss 3.0\n",
      "Epoch 19000, Loss 3.0\n",
      "Epoch 20000, Loss 3.0\n",
      "Epoch 21000, Loss 3.0\n",
      "Epoch 22000, Loss 3.0\n",
      "Epoch 23000, Loss 3.0\n",
      "Epoch 24000, Loss 3.0\n",
      "Epoch 25000, Loss 3.0\n",
      "Epoch 26000, Loss 3.0\n",
      "Epoch 27000, Loss 3.0\n",
      "Epoch 28000, Loss 3.0\n",
      "Epoch 29000, Loss 3.0\n",
      "Epoch 30000, Loss 3.0\n",
      "Epoch 31000, Loss 3.0\n",
      "Epoch 32000, Loss 3.0\n",
      "Epoch 33000, Loss 3.0\n",
      "Epoch 34000, Loss 3.0\n",
      "Epoch 35000, Loss 3.0\n",
      "Epoch 36000, Loss 3.0\n",
      "Epoch 37000, Loss 3.0\n",
      "Epoch 38000, Loss 3.0\n",
      "Epoch 39000, Loss 3.0\n",
      "Epoch 0, Loss 4.000000476837158\n",
      "Epoch 1000, Loss 3.9999899864196777\n",
      "Epoch 2000, Loss 3.999925136566162\n",
      "Epoch 3000, Loss 3.999446392059326\n",
      "Epoch 4000, Loss 3.9959218502044678\n",
      "Epoch 5000, Loss 3.9704747200012207\n",
      "Epoch 6000, Loss 3.8094871044158936\n",
      "Epoch 7000, Loss 3.3007733821868896\n",
      "Epoch 8000, Loss 3.0198910236358643\n",
      "Epoch 9000, Loss 3.0004706382751465\n",
      "Epoch 10000, Loss 3.000009059906006\n",
      "Epoch 11000, Loss 3.000000476837158\n",
      "Epoch 12000, Loss 2.999999761581421\n",
      "Epoch 13000, Loss 2.999999761581421\n",
      "Epoch 14000, Loss 2.999999523162842\n",
      "Epoch 15000, Loss 2.999999523162842\n",
      "Epoch 16000, Loss 3.0\n",
      "Epoch 17000, Loss 2.999999523162842\n",
      "Epoch 18000, Loss 3.0\n",
      "Epoch 19000, Loss 3.0\n",
      "Epoch 20000, Loss 3.0\n",
      "Epoch 21000, Loss 3.0\n",
      "Epoch 22000, Loss 3.000000238418579\n",
      "Epoch 23000, Loss 3.000000238418579\n",
      "Epoch 24000, Loss 3.000000238418579\n",
      "Epoch 25000, Loss 3.000000238418579\n",
      "Epoch 26000, Loss 3.000000238418579\n",
      "Epoch 27000, Loss 3.000000238418579\n",
      "Epoch 28000, Loss 3.000000238418579\n",
      "Epoch 29000, Loss 3.000000238418579\n",
      "Epoch 30000, Loss 3.000000238418579\n",
      "Epoch 31000, Loss 3.000000238418579\n",
      "Epoch 32000, Loss 3.000000238418579\n",
      "Epoch 33000, Loss 3.000000238418579\n",
      "Epoch 34000, Loss 3.000000238418579\n",
      "Epoch 35000, Loss 3.000000238418579\n",
      "Epoch 36000, Loss 3.000000238418579\n",
      "Epoch 37000, Loss 3.000000238418579\n",
      "Epoch 38000, Loss 3.000000238418579\n",
      "Epoch 39000, Loss 3.000000238418579\n",
      "Epoch 0, Loss 4.0\n",
      "Epoch 1000, Loss 3.999990463256836\n",
      "Epoch 2000, Loss 3.9999260902404785\n",
      "Epoch 3000, Loss 3.9994559288024902\n",
      "Epoch 4000, Loss 3.9959938526153564\n",
      "Epoch 5000, Loss 3.9709835052490234\n",
      "Epoch 6000, Loss 3.812356948852539\n",
      "Epoch 7000, Loss 3.305621385574341\n",
      "Epoch 8000, Loss 3.0205078125\n",
      "Epoch 9000, Loss 3.000487804412842\n",
      "Epoch 10000, Loss 3.000009298324585\n",
      "Epoch 11000, Loss 3.0\n",
      "Epoch 12000, Loss 3.000000238418579\n",
      "Epoch 13000, Loss 3.0\n",
      "Epoch 14000, Loss 3.0\n",
      "Epoch 15000, Loss 3.0\n",
      "Epoch 16000, Loss 3.0\n",
      "Epoch 17000, Loss 3.0\n",
      "Epoch 18000, Loss 3.0\n",
      "Epoch 19000, Loss 3.0\n",
      "Epoch 20000, Loss 3.0\n",
      "Epoch 21000, Loss 3.0\n",
      "Epoch 22000, Loss 3.0\n",
      "Epoch 23000, Loss 3.0\n",
      "Epoch 24000, Loss 3.0\n",
      "Epoch 25000, Loss 3.0\n",
      "Epoch 26000, Loss 3.0\n",
      "Epoch 27000, Loss 3.0\n",
      "Epoch 28000, Loss 3.0\n",
      "Epoch 29000, Loss 3.0\n",
      "Epoch 30000, Loss 3.0\n",
      "Epoch 31000, Loss 3.0\n",
      "Epoch 32000, Loss 3.0\n",
      "Epoch 33000, Loss 3.0\n",
      "Epoch 34000, Loss 3.0\n",
      "Epoch 35000, Loss 3.0\n",
      "Epoch 36000, Loss 3.0\n",
      "Epoch 37000, Loss 3.0\n",
      "Epoch 38000, Loss 3.0\n",
      "Epoch 39000, Loss 3.0\n"
     ]
    }
   ],
   "source": [
    "class LinearNet(torch.nn.Module):\n",
    "    def __init__(self):\n",
    "        super(LinearNet, self).__init__()\n",
    "        self.fc1 = torch.nn.Linear(input_size, hidden_size, bias=False)\n",
    "        self.fc2 = torch.nn.Linear(hidden_size, output_size, bias=False)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = self.fc1(x)\n",
    "        x = self.fc2(x)\n",
    "        return x\n",
    "    \n",
    "    def get_hidden(self, x):\n",
    "        return self.fc1(x)\n",
    "    \n",
    "model = LinearNet()\n",
    "# initialize the weights\n",
    "model.fc1.weight.data = torch.randn_like(model.fc1.weight.data)*0.0002\n",
    "model.fc2.weight.data = torch.randn_like(model.fc2.weight.data)*0.0002\n",
    "\n",
    "# load the data we do not need test data with pytorch dataloaders\n",
    "train_loader = DataLoader(torch.utils.data.TensorDataset(torch.tensor(train_inputs.T, dtype=torch.float32), \n",
    "                                                          torch.tensor(train_labels.T, dtype=torch.float32)), \n",
    "                          batch_size=n, shuffle=False)\n",
    "\n",
    "# define the optimizer\n",
    "optimizer = torch.optim.SGD(model.parameters(), lr=0.001)\n",
    "\n",
    "# hidden_activations_linear_no_bias, losses_linear = train(model, train_loader, optimizer)\n",
    "loss_list_linear_no_bias = []\n",
    "hidden_activations_linear_no_bias_list = []\n",
    "# n_runs = 10\n",
    "for run in range(n_runs):\n",
    "    # initialize the model\n",
    "    model = LinearNet()\n",
    "    # initialize the weights\n",
    "    model.fc1.weight.data = torch.randn_like(model.fc1.weight.data)*0.0002\n",
    "    model.fc2.weight.data = torch.randn_like(model.fc2.weight.data)*0.0002\n",
    "    # define the optimizer\n",
    "    optimizer = torch.optim.SGD(model.parameters(), lr=0.001)\n",
    "    # train the model\n",
    "    hidden_activations_linear_no_bias, losses_linear_no_bias = train(model, train_loader, optimizer)\n",
    "    hidden_activations_linear_no_bias_list.append(hidden_activations_linear_no_bias)\n",
    "    loss_list_linear_no_bias.append(losses_linear_no_bias)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "from itertools import combinations\n",
    "\n",
    "hidden_activations_linear_bias_list = np.array(hidden_activations_linear_bias_list)\n",
    "hidden_activations_linear_no_bias_list = np.array(hidden_activations_linear_no_bias_list)\n",
    "hidden_activations_relu_bias_list = np.array(hidden_activations_relu_bias_list)\n",
    "hidden_activations_relu_no_bias_list = np.array(hidden_activations_relu_no_bias_list)\n",
    "activation_list = [hidden_activations_linear_bias_list, hidden_activations_relu_bias_list, hidden_activations_relu_no_bias_list]\n",
    "model_names = [\"linear + bias\", \"relu + bias\", \"relu no bias\"]\n",
    "\n",
    "regr = linear_model.LinearRegression()\n",
    "simple_features = train_labels[2:,:].T\n",
    "complex_features = train_labels[:2,:].T\n",
    "combined_features = train_labels[1:3,:].T"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n",
      "-----\n"
     ]
    }
   ],
   "source": [
    "global_complex_r2_list = []\n",
    "global_simple_r2_list = []\n",
    "global_combined_r2_list = []\n",
    "for sub_list in activation_list:\n",
    "    r_2_list_complex = []\n",
    "    r_2_list_simple = []\n",
    "    r_2_list_combined = []\n",
    "    for activations in sub_list:\n",
    "        regr.fit(complex_features, activations)\n",
    "        r_2_list_complex.append(regr.score(complex_features, activations))\n",
    "        regr.fit(simple_features, activations)\n",
    "        r_2_list_simple.append(regr.score(simple_features, activations))\n",
    "        regr.fit(combined_features, activations)\n",
    "        r_2_list_combined.append(regr.score(combined_features, activations))\n",
    "        print(\"-----\")\n",
    "    global_complex_r2_list.append(r_2_list_complex)\n",
    "    global_simple_r2_list.append(r_2_list_simple)\n",
    "    global_combined_r2_list.append(r_2_list_combined)\n",
    "global_complex_r2_list = np.array(global_complex_r2_list)\n",
    "global_simple_r2_list = np.array(global_simple_r2_list)\n",
    "global_combined_r2_list = np.array(global_combined_r2_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 500x350 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import pandas as pd\n",
    "sns.set_style(\"whitegrid\")\n",
    "plt.rcParams[\"font.family\"] = \"sans-serif\"\n",
    "df_list = []\n",
    "for i, model_name in enumerate(model_names):\n",
    "    df = pd.DataFrame({\"linear\": global_simple_r2_list[i],\n",
    "                       \"non-linear\": global_complex_r2_list[i], \n",
    "                       \"combined\": global_combined_r2_list[i]})\n",
    "    df = df.melt(var_name=\"feature\", value_name=\"r2_score\")\n",
    "    df[\"model\"] = model_name\n",
    "    df_list.append(df)\n",
    "df = pd.concat(df_list, ignore_index=True)\n",
    "fig, ax = plt.subplots(figsize=(5,3.5))\n",
    "sns.barplot(data=df, x=\"feature\", y=\"r2_score\", hue=\"model\", capsize=.4, palette=\"Set2\", ax=ax)\n",
    "ax.set_ylabel(\"Rep. variance explained, $R^2$\", fontsize=14)\n",
    "ax.set_xlabel(\"Regressor\", fontsize=14)\n",
    "# increase the x tick labels \n",
    "ax.tick_params(axis='x', labelsize=12)\n",
    "# increase the y tick labels\n",
    "ax.tick_params(axis='y', labelsize=12)\n",
    "ax.set_ylim(-0.02,1.1)\n",
    "sns.despine(left=True)\n",
    "# Now you can save with\n",
    "# save_figure('/relu_nets_bias/', 'Representational_variance_explained', fig)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "def effective_rank(H):\n",
    "    S = np.linalg.svd(H, full_matrices=False, compute_uv=False)\n",
    "    p = (S**2) / (np.sum(S**2) + 1e-12)\n",
    "    H = -np.sum(p * np.log(p + 1e-12))\n",
    "    return np.exp(H)  # eRank in [1, min(N_out, N_hid)]\n",
    "\n",
    "def compute_and_sd_effective_rank(hidden_activations_list):\n",
    "    e_rank_list = []\n",
    "    for hidden_activations in hidden_activations_list:\n",
    "        H = hidden_activations @ hidden_activations.T\n",
    "        e_rank = effective_rank(H)\n",
    "        e_rank_list.append(e_rank)\n",
    "    e_rank_list = np.array(e_rank_list)\n",
    "    return e_rank_list\n",
    "\n",
    "e_rank_list_relu_no_bias = compute_and_sd_effective_rank(hidden_activations_relu_no_bias_list)\n",
    "e_rank_list_linear_bias = compute_and_sd_effective_rank(hidden_activations_linear_bias_list)\n",
    "e_rank_list_relu_bias = compute_and_sd_effective_rank(hidden_activations_relu_bias_list)\n",
    "e_rank_list_linear_no_bias = compute_and_sd_effective_rank(hidden_activations_linear_no_bias_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/zm/n63_ssc512d4ygg3lbvwql400000gq/T/ipykernel_41921/735238638.py:13: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.barplot(x=\"model\", y=\"effective_rank\", data=df_e_rank,capsize=.3, palette=\"Set2\", ax=ax)\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x350 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Create a long-format DataFrame where each row is one observation\n",
    "e_ranks = [e_rank_list_linear_bias, e_rank_list_relu_bias, e_rank_list_relu_no_bias]\n",
    "model_names = [\"linear + bias\", \"relu + bias\", \"relu no bias\"]\n",
    "df_data = []\n",
    "for model_name, e_rank_array in zip(model_names, e_ranks):\n",
    "    for e_rank_value in e_rank_array:\n",
    "        df_data.append({\"model\": model_name, \"effective_rank\": e_rank_value})\n",
    "\n",
    "df_e_rank = pd.DataFrame(df_data)\n",
    "\n",
    "# Create the barplot - seaborn will automatically compute confidence intervals\n",
    "fig, ax = plt.subplots(figsize=(4,3.5))\n",
    "sns.barplot(x=\"model\", y=\"effective_rank\", data=df_e_rank,capsize=.3, palette=\"Set2\", ax=ax)\n",
    "\n",
    "ax.set_ylabel(\"Effective rank, $\\\\rho(H^TH)$\", fontsize=14)\n",
    "ax.tick_params(axis='x', labelsize=12)\n",
    "ax.tick_params(axis='y', labelsize=14)\n",
    "\n",
    "sns.despine(left=True)\n",
    "plt.tight_layout()\n",
    "# save_figure('/relu_nets_bias/', 'Effective_rank_toy', fig)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAksAAAE9CAYAAAAbLiZDAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/H5lhTAAAACXBIWXMAABcSAAAXEgFnn9JSAABpk0lEQVR4nO3dd3xT1fsH8M/NTveipS10sVoopVAomxaZAgKKwBcXCI4vQ8CfKIgooIh8BQcoiqBSZImAIBtBhgzZe5TVllHo3iNJk5zfHzVpbzPoSJukPO+Xfdl777n3nlySp0/OPfccjjHGQAghhBBCjBJYuwKEEEIIIbaMkiVCCCGEEDMoWSKEEEIIMYOSJUIIIYQQMyhZIoQQQggxg5IlQgghhBAzKFkihBBCCDGDkiVCCCGEEDMoWSKEEEIIMYOSJUIIIYQQMyhZIoQQQggxg5IlQgghhBAzKFkihBBCCDGDkiVCCCGEEDMoWSJVxnGcwY9EIkHjxo3x4osv4vLly0b3i42NBcdxSEpKqtsKE0LqnC42uLu7Iycnx2iZOXPmgOM4LFiwoNbqkZSUZDRmSaVShISE4I033jAZk4KCgsBxXK3VjdgPkbUrQOzX6NGj9b/n5ubi7NmzWLduHTZt2oQ9e/agZ8+eVqwdIcQW5OTk4KuvvsLcuXOtWg9HR0c8//zz+uWMjAycPn0aK1aswMaNG3H06FG0atXKijUktoySJVJtcXFxvOWSkhKMGzcOq1evxpQpU3Dp0iXe9l9++QVFRUXw9/evw1oSQqxFIBBAJBLh66+/xtSpU+Hu7m61unh5eRnErMLCQgwZMgR//fUX3n//fWzbto23/a+//kJJSUkd1pLYKroNRyxGLBZjzpw5AIDLly8bNL0HBAQgNDQUYrG47itHCKlzYrEYr732GvLy8vDll19auzoGHB0dMXPmTADA33//bbC9SZMmCA0NretqERtEyRKxKB8fH/3varWat81Un6UjR45g0qRJiIiIgLu7O+RyOUJDQzFjxgyTfR1OnjyJZ599FoGBgZBKpWjYsCGio6Px/vvvo6CgwNIvixBSTTNnzoRUKsXixYuRlZVV6f2KiorwySefIDw8HHK5HK6urujRowd+/fVXi9ZPF7MqxivAdJ+lnTt3YuzYsQgLC4OLiwscHR3Rpk0bzJ8/H0ql0uh59u7di379+qFRo0aQSqXw8/NDt27drH57klQOJUvEos6ePQugtMnby8urUvu8++67+PHHHyGRSPDUU0+hV69eyMvLw//+9z9069bNIPnZuXMnunTpgu3btyMoKAjPPfccIiMjkZGRgQULFiAjI8Pir4sQUj3+/v54/fXXkZ+fjy+++KJS++Tn56NHjx746KOPkJaWhkGDBqFr1644deoURo0ahalTp1qsfrqYFRYWVul9xo0bh40bN8LV1RX9+/dH9+7dcf/+fXzwwQcYMGAANBoNr/yyZcvQv39/HD58GGFhYRg2bBhatWqFpKQkfWs8sXGMkCoCwCq+dXJyctiff/7JmjdvzgCwL7/80mC/mJgYBoAlJiby1u/cuZNlZWXx1ikUCvbGG28wAGzu3LkGx+E4jp05c8bgHCdPnmR5eXnVfGWEEEsBwKRSKWOMseTkZCaTyZizszPLyMjQl5k9ezYDwD777DPevpMmTWIAWO/evVl+fr5+/fXr15m3tzcDwHbu3FmpeiQmJjIALDAwkLc+IyODbd68mfn4+DCO49imTZsM9g0MDDSIdYwxtmXLFlZQUMBbl5eXxwYNGsQAsFWrVhkcx8XFxSD2abVaduDAgUq9DmJdlCyRKtMlS8Z+vL292bp164zuZypZMqWoqIiJRCLWrl073vqwsDDm5uZW05dBCKlF5ZMlxhibPHkyA8BmzJihX2csWSooKGByuZwJBAJ28+ZNg+MuWbKEAWD9+vWrVD10yZKpnyZNmrC9e/ca3ddUsmTKrVu3GAD23HPP8dbL5XLWpk2bSh+H2B56Go5UW/mhA5RKJe7evYuTJ0/ivffeg5+fH2JiYip9rOTkZGzfvh3x8fHIy8uDVqsFAEgkEty6dYtXNioqCmvWrMG4cePw9ttvIzw83DIviBBSa2bMmIHly5fj22+/xTvvvGPyNv3Zs2dRXFyMTp06oVmzZgbbX375ZUyePBnHjh0DY6zS4yBVHDqguLgYt2/fxrlz5zB16lT89ttvVYolt27dwq5du3D79m0UFhZCq9WCMabfVl5UVBSOHj2KGTNm4PXXX0eTJk0qfR5iI6ydrRH7AyO34XTOnTvHnJ2dmVQqZQkJCbxtplqWvvjiCyaRSMx++yvv3r17rE2bNvptXl5ebPDgweznn39mCoXCoq+VEFI9qNCyxBhjU6dOZQDYe++9xxgz3rK0fv16BoD95z//MXlsV1dXBoDl5OQ8th6mbsPp7Nmzh4lEItagQQOD4xlrWdJqtez//u//GMdxJuNVUFAQb5+LFy+y4OBg/XY/Pz82cuRItnHjRqbRaB77Goj1UQdvYlFt27bFm2++CaVSiW+//fax5U+cOIF33nkHcrkccXFxSEpKgkKhACu9RQxfX1+DfRo3bowzZ85g7969eOutt+Dn54ft27dj7NixiIyMRHZ2dm28NEJIDU2fPh1yuRxLly5Fenq62bKVaTGyxOja/fr1w9ChQ5Geno5ffvnlseU3bNiAL7/8Ev7+/ti0aROSk5OhUqnAGNM/Ccf+bWHSiYiIwLVr17Blyxa8/vrrcHJywoYNGzB8+HDExMRApVLV+HWQ2kXJErG44OBgAMCNGzceW3bLli0AgHnz5mH06NH6oQCA0mbylJQUo/uJRCL07dsXS5YswcWLF5GUlISnnnoK8fHxtTp1AiGk+ho2bIjx48ejsLAQn3/+udEyfn5+AIDExESj23Nzc5GbmwtHR0c4OztbpF7ViVnff/89hg0bBj8/P/3YcQkJCSb3k8lkGDp0KJYvX44bN27g6tWriIiIwNGjR/HTTz9Z4FWQ2kTJErE4XcBwdHR8bFldK1Djxo0Ntm3cuNHgG5opAQEBmD59OgCYnJuOEGJ906dPh4ODA7777jukpqYabI+KioJcLsepU6cM+v4AwJo1awAA3bp1s9i8bZaKWb/99lulz9myZUtMnDgRAMUse0DJErGo8+fPY/ny5QCAAQMGPLZ88+bNAQA//fQTb1qBa9eu6ZOfir766iujQXbPnj0AShMnQoht8vb2xoQJE1BUVIRVq1YZbHd0dMTYsWOh1WoxceJEFBYW6rfdvHkT8+bNAwC89dZbFqnP3r17sXXrVgBVi1nLly/nfZk7cuQIFi5caFC+qKgIS5YsMRhgV6vV4s8//wRAMcsecKyyX90J+Zfu21z5p+FUKhXu3r2LEydOQKvV4plnnsHWrVshEJTl47GxsTh8+DASExMRFBQEAMjMzER4eDhSUlIQHByMDh06ICsrC4cPH8bQoUNx6tQp3L17lxeU3NzckJ+fjzZt2qBZs2ZgjOHSpUu4ceMGvLy8cOLECXrahBAr4zgOUqkUCoXCYFt6ejqCg4P1idBnn32GGTNm6Lfn5+ejZ8+eOHv2LLy9vRETE4PCwkIcOHAACoUCkydPxuLFiytVj6SkJAQHB5t9Gg4AJkyYgKVLl/L2DQoKMog/N2/eRLt27VBYWIiWLVsiIiICycnJOHr0KN555x0sWrQIgYGB+pkKcnJy4O7uDolEgnbt2iEoKAgqlQpnzpzBvXv3EBISgjNnzlh13jxSCVbqWE7sGIw8/SEQCJiHhweLjY1lP/30k9EnPEw9DXf//n32wgsvMH9/fyaTyVhYWBj77LPPmFqtNvo0yi+//MJeeOEF1qJFC+bs7MycnZ1Zy5Yt2bRp09jDhw9r86UTQioJRp6GK++9997Tx4+Kg1IyVjre0ty5c1nLli2ZVCplzs7OrFu3bibHcTPF1DhLQqGQNWjQgPXv359t3rzZ6L6mxlm6du0ae+aZZ5i3tzdzcHBgbdu2ZcuXL9e/7vJP3pWUlLClS5ey5557jjVp0oQ5ODgwNzc31qZNG/bJJ5+w7OzsKr0eYh3UskQIIYQQYgb1WSKEEEIIMYOSJUIIIYQQMyhZIoQQQggxg5IlQgghhBAzKFkihBBCCDGDkiVCCCGEEDMoWSKEEEIIMYOSJUIIIYQQMyhZIoQQQggxg5IlQgghhBAzRNaugIeHBwoKCiASieDv72/t6hBC6lhycjLUajWcnJyQlZVVa+ehWEPIk6umccbqc8NJJBKUlJRYswqEEBsgFouhUqlq7fgUawgh1Y0zVm9ZEolEKCkpgUgkQlBQkLWrY3N0/6gSicTKNbE9dG2Ms7frcvfuXX0MqE26WCMWixEYGFir57I39vaeqUt0bUyzp2tT0zhj9WTJ398ft2/fhp+fH27dumXt6tic+Ph4AEBoaKiVa1K7GGOoaiPnjRs3AAAtWrSojSrZLVu+LhzHgeM43rpmzZrh9u3btX5rTBdrAgMDKdZU8KTEmeqozWtTnbhnS+wp1tQ0zlg9WSJPNq1Wi4cPH6KgoKDKQUOhUAAo+8CSUrZ8XTiOg5OTE/z8/CAQ0PMl5MlUk7hnS2w51giFQvj7+8PR0dEix6uVaPXjjz/qgyIh5jx8+BD5+fnVChgSicQumn/rmi1fF8YY8vPz8fDhQ2tXhRCrqUncsyW2HGs0Gg3S0tIsdjyLtywlJydj2rRp8PPzQ25ubqX302i0uP8wEyKhACKREEKhAGKxCE4OUoNme1I/MMZQUFAAAPDz84Ozs3OV9i8uLgYAyOVyi9fNntnyddElSrpv1PTZJk+amsY9W2KrsaakpAQJCQlQKBQWizMWT5b++9//okePHvDw8MCmTZsqvV9aZj6eGvmxwXpHuRTtWgdj+KBO6BfThpru65Hy9+udnZ2r/G+rK0/vCT5bvi66Pwy6f3tKlsiTpqZxz5bYaqwRi8X6320yWVqzZg0OHz6Ma9euYdasWRY5ZmGxEkdOxePIqXjEdGqJr+eOhpODzCLHJoQ8ebJy8zD7m58BDuAAgONK/69fAXDggH8DrC7QcmU7/Ptf2TJX8f/gAKEQnFAMTiQBJ5RAIJFDKHOBVCaDk1QIB4kIjlIRHKVCeDlJ4ecmh4tMRAkkITbIYslSWloapk6digULFqBRo0aWOizP4RPX8Pq0pfjorQFPTEDJyMgAUPZERn3CGINCoYBEIkFxcXGVv52o1WoAZU3BlXH37l10794dYWFhUCgU+Oqrr9CuXTujZefNm4d27dphwIABAIDVq1ejoKAA48ePBwB8+OGH6NOnD3r06FGpc/fr1w+bN2/m9eX7888/UVxcjCFDhlT6NTxOda5LXdFqtdBqtVCpVLhx4wY4jqvVsZWM0pRAnH27xodhJn5/nHQmRrrWCQ81rkjUeKGQSfXbHMQcGrlKENpAhlbeMnRo5AAHce1/a6/PcaamLH1tahr3qqO24l5NYk3F8wBASkoKVqxYgQ8//LDKxyuvNuKMxZKlCRMmoEWLFvoLWlvOXL6HwydvIbZT81o9D6lbWsaQU1i1N7PugyoSaY1ud3MQQ2Akqe7WrRvWrVuHEydO4IsvvsDatWurXmET3njjDSxfvrzS5fv27WuxcxP7IOdKECDMRoAwGx1ZEu5r3XFGFYgs5oiiEoabGUrczFBi2/VcSIQcOgc4YmRrN4R4SB9/cGI3tIwhq1Bl8WTJGnHPUho2bFjjRKm2WCRZ2rx5M7Zv347z58/XSYvPzsPx+O+YwbV+HltQn8c/0Wq1+kdOFVohui46bNHjn53VG55O/D8wMpkMQqEQcrlc/61OLpdjz549+PTTT6HRaPDWW29h1KhREIvFkEql+s6Luic/dMsikYi3HYD+2MYIBAJ8/vnnOHXqFCIiIvDNN98gLi4OBQUFmDRpEl588UX9kPzr1q1Dw4YNMWzYMOTl5YExhj///BMy2eNvQdtqp0ug9N9cIBBAJpOhRYsWEAgENvs0TV3gOCBAmI1GsmxcVvvjTEkgGMpiqErDcDixAIcTC/B8VCN89ExLuMjEZo5YPfU5ztSUpa+NLu7lKDR45svjFjlmeXUZ93SxZvz48RCLxUhMTISjoyO2bt0KjUaDV155BcnJyXB0dMSaNWvg4eGhr5NYLMb27duxYsUKMMawceNGZGVlYdq0adi0aRMWLlyIXbt2IS8vDwsWLECfPn0wa9YsHDhwACKRCJ9//jk6depk8hpbOs7UOFkqKCjAxIkT8dZbb8HPzw85OTkAykb2zMnJgVgstthYBwAQfzsZCfdSERLgY7FjkifH4cOH0bFjR9y5cwcHDhyAVqvFxx9/jEOHDkEkEqFnz54YMWJErZy7T58+WLhwIV588UWcPXuWt23FihVwcHDAtm3b8MMPP2Ds2LGQSqU4fPgwdYa2JKEYGs9m+kVdZ1sGBo7pbqkxQLde/3g3Q9mT3rqCFcvg3/0YwDQQaErAaUsg0JZAoFHA1L+ggAPaiJPhISjEfmUoNBAalNl09gH+uZOJuFc7oJmP/T5BRayjtuNe9+7dsWLFCrz44ou4fPkybty4gYCAAKxbtw6rV6/GN998g9mzZ/P28fPzQ1xcHFasWIEVK1Zg2LBh+m0TJ07Eu+++i4yMDAwfPhx9+vTB3r178c8//0AkEkGrNX5HobbUOFnKyMhAamoqvvjiC3zxxRcG293d3TFkyBBs3bq1pqfiOX7mJiVLpFpiYmKwadMmLF68GCdOnEDDhg1x69Yt/S2xjIwMpKenG+wnk8mQmZmpX1YoFJDL5UhOTsaLL74IoPRbaGxsLADgr7/+glDI/6MXFRUFAIiOjsbt22X9ZjQaDaZPn44LFy5AqVSiVatWaNKkCWJiYjBmzBj4+/vj448/NjgeqToPVxfMnvhqnZ9XrS5BTk42Uh49RGLiHcRfvwKlUskr01iYgzcaJeOqvC2O38mCWsvvDZWcU4zhP/yDta91RCs/17qsPrFzlo57FbVt2xYA0LhxY2RnZ+POnTvo0KEDAKBDhw74888/DfYpHw+XLVvG27Z27Vr88ssvEAgESElJAVDaz+nNN9+ESCTC3Llz0bBhw+pcimqp8c3Shg0b4uDBgwY//fr1g0wmw8GDBzFv3rzHHsfb0xm7fnkf21a+h99XvIPfvp8Ksdj0H4ZL1+/WtOrkCTdhwgQsW7YMrq6uCAsLw759+3Do0CFcuHDB6IcwPDwcR48eBVCa3Jw/fx7NmjWDv78/Dh06hEOHDqF///76340lNufPnwcAnDlzBk2bNtWvv3DhAlJTU3HkyBHMmjULjDEolUpMnDgRcXFxSE9Px7Fjx2rpSpC6IBKJ4eXljfDWkXhm8DBMeft99IjpBYGA/z4pybqPsSH5OP1Bb7zTpzmcpfzvtDlFJRgXdwapeYq6rD6pJywV9yoq3/LNGEPTpk1x+vRpAMDp06eN7mMqHgLAokWLcPDgQd4QRD169MBPP/2EmJiYKvUNtYQatyzJZDL9N+ny4uLiIBQKjW4zRigUoFkw/x9qSN/22LTzpNHyV28+qGpViQ1zd5Dg7KzeVdpHN9S+qX487g7m71GLxWL069cPW7ZswQcffIDevXtDIBCgQYMG+O233wAAs2bNwtdffw0A+OKLLxAZGYlu3bqBMYZx48bx7sFXxu7du/Hxxx+jTZs2iIqKwuXLlwGU9od49OgR+vTpg7CwMAClT7CMGzcOWq0WLi4uJp9eIfZJIpGgR0wvBIc0xa/rVkGpLEt+jh09hKZNW+CtXs3wXFQjTFx7Dhfu5+i3p+QpMOXX81j3WicIBHR71h65SAU4PfMpi3fwtpW4N3ToUPz+++/o0aMHHB0djXYov3//Pvr16wcA+j5LOj179kT37t0RHR0NFxcX/TGLi4uhVCrx448/Vup6WArHamm89TFjxmDTpk36kUpN0U1uFxAQgLt3+a1Fa7ccxZwvNxrdTyYV49K+hfW+H0d97nhZvoO3rhNeVdhyR2ZrsuXrYuzfXBcDmjZtWqsT3NbVearj3r0krPnlR14/DF8/f4wdNx4cJ0CRSo0xP5/GqaQs3n7zhobjpU6BNT5/fY4zNVVbHbyB6sU9W2KrsaY24kyt/SvpnvKpieDGDUxuUyhLkJGVX6PjE0KILQgICEJMLL9l9dHDZFy5fBEA4CARYdnLUfB34/9RWvTnDeQpSuqsnoQ8qWw6pW3s52V2+73kjDqqCSGE1K5OnbvD04v/BfH4scP6J+08HCX47LnWvO05RSX4+WhindWRkCeVTSdLvt5uEJppokzLrPxEvYQQYsuEQiF69erPW5eenoY7t2/ql3s0b4B+rfhPAf98NBFFKnWd1JGQJ5VNJ0sikRCe7k4mt9NtOEJIfdKseQt4VWhdOnP6BG/5nb4tUL6rZp5CjW0XHtZF9Qh5Ytl0sgQAHu6mB1/LzK5ZnyhCCLElHCdAx87deevu3LmF/Pw8/XJzH2fENucnVKtP0FAqhNQmm0+WzLUsZeZQyxKpmqSkJDz//PO8dQsWLEBiovX7fbRv395gXVxcHP755x8r1IZYS6tWrSEWlz3+zZhW39Fb5+XO/Cfgrj7Mw+00iofEOFuOe+UdOnQI06ZNM1j/5ptvWqE2fBabSLe2eLqZSZaoZaneYEyLwsKiKu2jG2dJq9UY3e7g4ACOe/z3gRkzZlTpvJWhm5uoojFjxiAuLq7SxxkzZozlKkXsgkQiRVjLcFy6eE6/7sqVC+jcpazFKaa5N7ydpUjLLxsBfOelFEzpTdOg2AvGGAoLCy0+TpYtxr2a+uGHHyx+zKqy/WTJzG247BxKluqLoqJiLP7qM4se8+13ZsLR0XSyrTNmzBhMmzYNGRkZWLBgASQSCe7du4fVq1ejdevWRiebXL16NVauXImcnBy8/fbbePnllzFnzhwkJiYiLS0Nn3/+OVq3bv3Yc5en1WoxefJknD59GkOHDsX06dMxZ84ctG/fHgMGDED//v2hVCohkUiwefNmqNVqPPfccwAAFxcXbNu2rVrXidie1hFteclSasoj5ORkw83NHQAgFHAY0NoXcceT9GV2Xn6IKb0NR0kmtkmlUlo85gF1G/eef/55zJs3Dw8ePDCIe3Fxcdi2bRuUSiXS0tLwxx9/wM/PD4sWLcLmzZvBcRy++eYb/ZQnOjdu3MDgwYPx6NEj/PTTT4iIiED79u1x5swZ7Nu3D/Pnz0dBQQGGDRuGGTNmYOvWrZg/fz4cHBwwcuRIjB8/3uLXFLCD23Auzg4mt+UXFNdhTciToKSkBNu2bcPChQuxcuVK/WSTf/31F44ePYply5ZBo9Fg2LBhOHDgAP755x8sWbJEv39AQAB2795d5UQJALKzszFx4kQcP34ce/bsQWpqqn6bQCDAH3/8gcOHD+OZZ57Bhg0bcP78ebRv3x6HDh2y+NyLxLoCA4Mgl/Nj380b13nLA1r78renFuB+VtVaZwkBai/uubm5YefOnXjjjTewadMmpKSkYNu2bTh27BjWrVuH6dOnG9QlOzsbf/zxB9auXYsPP/yQt61r1644ePAgTp06ha1bt6K4uBibN2/Gzz//jEOHDtXq7TqbT5acnYxPZQEA+YU0NxKxrMjISABlk0FmZGToJ5t86qmn9JNN7tu3DzExMejbty9u3ix7tFs3cWR5Z8+eRWxsLGJjY7Fnzx7ExsZi+PDhBuWcnJzQokULcByHyMhIJCUl6bcVFhbi9ddfR48ePfDjjz/i4cOHiImJgaurK0aPHo0vv/zS4teCWI9AIESz5vwRo2/cuMZbbh/oDg9H/tQWR2/T2HOk6moj7gGGk+smJSWhTZs2EAgECAoKQm6u4fA/bdu2BcdxaN68OdLS0njbzp8/j969eyM2NhYJCQlIS0vDhx9+iKVLl+KVV17BqVOnLHRFDNn8bThnR9PDqFOyRCyt4mSQXl5e+skmxWIxSkpKIBaLMXfuXBw8eBBSqRRNmjTR72Psfn1UVBQOHToEwHyfpYKCAty6dQtNmzbFpUuXeH0K9uzZAz8/P6xZswZLlixBVlYWSkpK9N+8+vbtixEjRiAgIMACV4HYghahLXm34u7dTUJxcbF+agmBgEPXpl7YfrFs2IAjt9IxKpreA6RqaiPuGTtuUFAQLly4AK1Wi3v37sHNzc1gnwsXLoAxhjt37sDb25u3bcGCBViyZAlCQ0MRHR0NxhgaN26M77//HsnJyXj55Zdx4MCBGl4N4+wgWTLdslRQqABjrN7PD/ckcHCQ4+13ZlZpn8dNpOvgYPwW7pEjR9C7d+nUErpJHE0RCARGJ5scNmwYevbsicjISLi7u1ep3qa4u7vj66+/xtmzZ/Hss8/Cx6ds8MFOnTrh008/xYABA+Dr64vGjRvj9OnT+OCDD6BWqxEcHIxGjRpZpB7ENoSENIVIJIJaXTrgJGNa3E26g9CwcH2Z7hWSpWO3M6HRMghpcl2bJ5FIMeXt92ulg7cxthD3GjZsiCFDhqBr167gOI53K0/H1dUVzzzzDFJTU/HTTz/xtg0bNgwjR45Eq1at4OjoCACYO3cu/vnnH+Tn5xt9ks5Sam0i3coyN5EuAJw4dwuj314KsZBBLOQgFnEQijg0CgpBoLcrPp0+ClKJzed81VafJ7ikiXRrhy1fF5pIt2rWrv4ZiYm39cvtoqIxYOBQ/fLDnGJ0WcD/Jr17SneE+bpU6Tz1Oc7UFE2ka5qtxpraiDM2n2VERzbBX2v+D3Erl+nXqZkA0c8NM+jgSAgh9UlwSFNespSYcJu33c9NjkbucjzILnvY5dy97ConS4QQ82w+pRUIBJDJ+bdZRJwWeUXUX4kQUr+FhDTlLWdnZyE7O4u3rl0A/3bIubs5tV0tQp44Nt+yBABSqWGflLwCekSWEFK/+TRsCAcHRxQVFerXJSbchntUtH65XYAbtpXrt3T+fnad1pFUDcdx4DgOjDHk5+fD2dl+BxLVarW8/9sKlUql/91SfZrtIlmSSKQG64oUSiMlCSGk/uA4AYKCm+Da1Uv6dYkJt9GufLIUyG9ZSkgvRE6RCm4O/GEFiG3gOA5OTk7Iz8/Hw4f2PQGyLkmy1X5XMpnMYsmSbb7CCsRiscG6YqXKSElCCKlfKt6KS0q6A8bKvsmH+bpAKuKH8qsP80Bsl5+fH5ydne3+SW6VSsVrxbElQqHQYOiBmrCLliWBQADGCcCVCxBKG/0HIjWXnVuI3LzCx5ZTKEtbF2VSfsujq4sj3F0dze7bpUsXDBo0CDNnVm24AgD6ofeNiYuLwwsvvACJRIK4uDi0aNECnTt3rtRx4+LiUFBQgEmTJunXHTp0CDt27MCiRYv066ZOnYrPPvuszp5AMVYHoHRyS1uYs6m+C66QLBUXFyMl5RF8ff0BAGKhAM19nHE5uWyAv+uP8tC1qVed1pNUnkAgQKNGjcAYg5UfSK+R8k+c2Rrd7U5LsYtkCQAgEAGasgRJqaTbcPXV6s1/45uVe6q9/1uv9sfksU+b3H7//n0EBgbir7/+qlayZE5cXByef/55SCSSWpsE9+uvv7bo8XTBWhdYKjvZLyVKdcPV1Q0eHp7IysrUr0tMuKNPlgAgzLdispRfp3Uk1WPpP+h1TVd3W70NZ0l28wo5IT+vU5WUWKkmxN5t2rQJL730Epo0aYLbt0sfxZ4zZw5efPFF9O/fHz169EBRURG0Wi369u2LmJgY9OnTB3l5Zbc2SkpK0K1bN/3yyJEjcezYMVy4cAFPP/00Fi9ejDlz5mDHjh0AgE8//RRdunRBTEwMrl27hn379qFnz57o0KEDFixYUKX6x8bGoqCgAHFxcXjuuecwcOBAdOjQQd//YfXq1ejduze6dOmiH8124cKF6NmzJ6KiorBv3z4ApUnR+PHj0bt3b+Tk5Jg9p25yyw4dOuDSpdL+M+3btwcAo69l69atiI6ORmxsLL7//vsqvT5iqGLrUvnhBAAYDBVw/RHdhiPEkuwmWRJUSJZK6DYcqaa//voLffv2xahRo7Bx40b9+hYtWmDPnj3o3r079u/fb3TyWh2xWIy2bdvizJkzyMvLQ1ZWFrp27YrIyEjs3r0bU6ZM0Ze9ePEizpw5g2PHjuHw4cMIDQ01OiFkdVScqDIjIwMbN27Evn37sH//fnz66acAgIkTJ+LgwYPYu3cv5s+fr9+/ffv2+Ouvvx47Gq+tTm75pAgObsJbvn8vCWp12RfGisnS7bQClGhs6wklQuyZ3dyGEwjFKP/RLx8oCKmsBw8e4NKlS3jmmWeg1WpRWFiI999/H4DhpI+FhYV48803ce/ePeTk5GDYsGG8Y73yyitYs2YN2rRpY7CtvPj4eP3w/kBpk/X58+cxe/ZslJSU6CeErI7ydT558iQSEhJw/fp19O/fHwKBAOnp6QCAtWvX4pdffoFAIEBKSop+f90EmBqNBr169dLXNzY2Vr+f7jzmJres+Fo+/PBDfPXVVygsLMSECRPQqVOnar0+UiowqAkADkDpLVO1Wo379+/pk6iwhvxkSaXR4k56AUIb0uCUhFiC3SRLIpEY6nLLJXQbjlTDpk2bsHjxYjz77LMAgHHjxulvxVWc9NHY5LXldejQAdOnT8f169exfv16AKUtThqNhlcuNDSU1yql1WqNTghZHRXrHBISgvDwcPz+++9wcHDQf04WLVqEq1evIjs7m3f7UNfXQCgUmpzs99atWzY7ueWTQi6Xw8/PHw8fPtCvS0y4rU+WXB3E8HOV4WFu2WC9t1IpWSLEUuwiWTpy8jqu3slHYZ4Kag2g0TBk4Q6mfbIaUREhGDWkq7WrSOzE5s2b8ccff+iXe/XqxbsVV56xyWsr6tOnD06dOgUPDw8AwODBgzFixAiMGDFCX6ZNmzZo27YtOnfuDLlcjm+//dbohJCmbNy4ERcuXABQ2ppljpeXF4YPH46+fftCLBajdevWWLJkCXr27Inu3bsjOjoaLi5V/wNqq5NbPkmCgpvwkqWkxDu87U28nXjJUkL6458oJYRUjs1PpAsAkz9aid0HLxjdNqRveyz68OVarKF11ecJLk1NKLnk5921+jScJf3vf/9D8+bN9S1VtsBWJ7cEaCLdmkhMvIO1q8snqhzeeXeW/t959h9XsOqfshg6JNIPi//TtlLHrs9xpqbo2phmT9em3k+kCwBymemRaGlwyvrn5WE9MLhP1GPLmRtnqS58/PHHOHbsGN555506OR95sjVuHACRSAS1WtchgeFu0h2EhoUDAEIaOPHKU8sSIZZj98mSQkF9l+obd9fHDyoJWL8F5aOPPrLKecmTSSQSo3FAEBITyoYNSEwsnyzxPzMJ6QVgjNn1OD6E2Aq7GDpAJjWdLBUpqGWJEPJkqDiEQPnEqWLLUqFKg7R8GryXEEuwi2TJQW7mNlwxBQNCyJMhOJg/OGVWViYyM0qHh/B1kUEm5of0O+kFdVY3Quozu0iWzLUsKZR0G44Q8mTwaegLR0d+C9LVq6UjqgsEHIK9qN8SIbXBTpIlscltShUlS4SQJ4NAIEBYy3DeumvXLut/D/Hi91u6n1VUJ/UipL6zi2RJLjOdLBVTn6V6Jz09nfdT9vTP48s+brDSQ4cO6cf9sdVpOHRzv5UXFxeHb7/9ts7qUH5eu+qgEbtrT8tWEbzljPQ0pKSUzgvYyIP/sMP9bEqWCLEEu0iWpGZalug2XP3j7e3N+9GNy2NMWFgYr6xu8MbK+OGHHyxQ2zJabdXm4qpqeUKA0iEEnF1ceetOn/oHANDI3YG3/n5W9eYcJITw2UWyJH9MnyUrj6tJ7FT79u0BlLaivPjii+jfvz969OiBoqLSb+Pz589HTEwMevTogcuXS291vPPOO4iNjUV0dLQ+MYuNjcU777yDp5/mD4Q5ZswYTJw4Eb1798aIESOg0Whw6NAhDBo0CIMHD8bq1auxbt06dOzYER07dsSePWUDcc6dOxcxMTF46623DOodFxeH7t27o0uXLvppRGJjYzFlyhR07NgR//vf//DOO++gU6dOWLhwIW/frKwsxMbGIjY2FoMHDwYAXLp0CV27dkWXLl0wb948XvmFCxfit99+AwDcuHEDr7zyChhjeOutt9CzZ0/06dMHDx480F+vzp07Y9KkSQZTvhDL4TgB2rZtz1t35fJFFBTko7E7v2XpAbUsEWIRNU6WLly4gIEDByIgIAByuRweHh7o3Lkz1qxZY4n6AQBkZsZZ0mi0KFFTYCY106JFC+zZswfdu3fH/v37cfnyZdy4cQOHDx/Gb7/9ph9T6ZNPPsGhQ4fw448/8hKRAQMGYO/evQbHbdOmDfbv349mzZph69atAIC8vDz88ccfeOmll7BgwQL8/fff2LdvHz744AP9fn369MHhw4eRlZWFs2fP6tdnZGRg/fr1+Pvvv7F//358+umn+m0jRozAiRMnsHLlSrzyyis4duyYwefw/PnzaN++PQ4dOqSvz8yZM/Hjjz/i2LFj+Pvvv5GUlKQvP3LkSH2ytGHDBowcORI7d+6Eu7s7Dh48iAULFmDBggVISUnB3r17cfz4cUyePBmZmZnV+4cgldIuKhpCoVC/rNGocejgPjT24LcsZReVoEBp+jY2IaRyajwoZU5ODho3boxRo0bB398fhYWFWLt2LV5++WUkJSVh1qxZNa6kuQ7eAKBUlkAitovxNYmNatu2dFqIxo0bIzs7GwqFAsePH0dsbCwA6P8wffHFF9i7dy8EAgHvj1WHDh2MHjcqqnQk8ujoaMTHx8PT0xPt27cHx3FIT09HYGAgpFIppFIpJBKJvn9W+f10E/0CQEJCAq5du4aePXsCKO2zpRMREQGO4+Dj44OIiAgIhUKIxfzPTkxMDI4fP47Ro0ejdevWmDZtGlJTUxEWFgYAaNeuHe7cKZtzLCAgANnZ2SgoKMD+/fvx/vvv46uvvsKWLVvw999/6yfNTUpK0p+/efPmcHXl3yYiluXk5IzWEW1x4fwZ/boL58+iSbOWBmXvZxUhzJcm1CWkJmqcYeia9MsbNGgQEhMTsXz5coskS+ZG8AZKO3k7O9nePFiketLS0njL7u7uJstev36dN4K3m5tbtc5ZfpRjxhhCQ0MRExODH3/8EQBQUlKCzMxM7NixAydOnMDly5cxefJk/T66ee0qOn/+PKKionDmzBlERkbyyjZo0AB3796FUqmEUqmESqWCSCTS79e7d2+cOXMGU6dO1d8GDAkJQUREBHbs2AGO43gd2su/BlOjNpeUlODDDz8EAPTt2xcjRoyAj48Prl+/jtDQUJw7dw7//e9/ceTIEf0+gwcPxmeffYYWLVpALBYjNDQUI0aM0B9Hd22uXLkCxhju3LmD3NzcSlx1UhMxsb1x9collJToHnJh2L71V7RxaoGLBWXJESVLhNRcrTXHeHl5GfzRqy6pxHzLkoKGD6hXGjRoUKWytTHdSUREBJo1a4aYmBgIBAL06dMHM2bMgI+PD3r27Fnpp73Onj2L9evXw9PTE3PmzOElIUKhEDNmzECPHj0AgNdfaPfu3fj444/Rpk0bREVF6ZMlLy8v/Oc//0FMTAyEQiFat26NJUuWVPp1nT59Gh988AHUajWCg4PRqFEjfPrpp3jttdfAGMPAgQMRFBTE22f48OEICgrC7t27AQDPPPMMDhw4oG/deumllzBu3Dj06dMHnTt3Rrt27eDp6VnpOpHqcXZ2QWzPPtj35079OpVKhQ64jGCZIx5q3JCrleHWzetoJi+AQCiEUCCEQCgwSKazs0tvmz56lGzyfByqMW1KFXep1jlqWU5OFgAgNeWRlWtie+zp2tS0HyXHLNQ7WqvVQqvVIjs7Gxs3bsSUKVPw7bffPvbxbN1MwAEBAbh7967RMg8eZaLniI9NHmPnquloHuJXo/rbKnua1bmqjM1AXxXWnhvuccaMGYNp06YhPDz88YUtyJavi7F/85rOBl5ZdXWeusQYw++b1+P6tSvWrgohNu2rxcuQlZVd7c+/xVqWJkyYoH8UWyKRYMmSJVUax0ar1eoTg4qy88w/0RF/4za0qrzKV9aOZGRkAIDJa2PPGGNQKBSQSCQoLi6ucrKk69+jSw5sjUajgUKhqPP62fJ10X2pUqlUuHHjBjiOg0pVt2OlqVSqevV5Cg1rg/z8Ajy4n2TtqhBSb1ksWZo5cyZee+01pKWlYfv27Zg0aRIKCwv1AwDWhFRivppKFT3tYe/UarW+v05l6cYpstXxipYtWwag7utny9fF3ACjpHqEQiE6duoOX19/XLlyHsVFNFwAIZZmsWQpICAAAQEBAEofowaA999/H6NHj65UHxSBQGDyVpP6MUMDePs0rJe3qYD6fRuOMYZbt25Bo9Hg4cOHVd5flwxUtUWqvrP16yIQCODo6IhmzZqB4zhIJOYf4LA0iURSLz9PYWFh6NP3aWw/eg6r9p2CJ1cEOaeCo1ANb0cRNBoNtFoNNBotAH7vC11vDFMPBphi2SHubHO8vOpemyeBPV2bmtaw1jp4R0dHY9myZUhISKhSh11jRCIhxCKhyfGUaBRv+8RxHPz9/ZGWlgaFQlHl/XW3b2QymaWrZtds/brIZDJ4e3vbRYC1NwKBAC1btMC5Xam89fHv94dMLDSxV/3+UlZTdG1Ms6drs3LVBmRmZVd7/1pLlg4ePAiBQICQkBCLHE8mk6CkwHgfDAXND2e3HB0dERwcDMZYlUdiL99RmJSx5evCcRwlSbWsoathkpySq0BQhUl2CSGVV+Nk6Y033oCLiwuio6Ph4+ODjIwMbNy4ERs2bMC7775b41YlHZlEjHyYSJaoZcnuVeePqK68rd5usha6Lk82Z5kYTlIRb+TuR5QsEVIjNU6WOnfujJUrV2LVqlXIycmBk5MT2rRpg9WrV+Oll16yRB0BADKZ6bGWipXUskQIITq+rjLcSivQL6fk2d6TkYTYkxonS6+++ipeffVVS9TFLHNTniipZYkQQvQaVkiWHuVWvU8gIaSM3bTTm5vypFhByRIhhOj4Vui39CiHkiVCasJukiWZ1HSypKDbcIQQoufryh+9nVqWCKkZO0qWTN+Gow7ehBBSpmLLEvVZIqRmam3oAEt7umckigozoC7OgUjIQSgAsiQ+eOv5XghsZJkn7gghpD6oOHwA3YYjpGbsJll6fmAnZKTFI+N+2VD+twQueLpnWyvWihBCbE/F23CZhSooSjRmB6YkhJhmN7fhAEAs4d+KY1qaZ4oQQirydTMcmDI1j1qXCKkuu0qWJOIKnbw1lCwRQkhFzlIRHCT8VqTUPKWVakOI/bOrZElaccJNpoFWa5uTLxJCiLVwHAcfF37rUlo+tSwRUl12lSzJKiRLImihVGutVBtCCLFdDZylvGVqWSKk+uwrWaow1pKI00JRorFSbQghxHZRyxIhlmNfyZLMsGWpmJIlQggx4F2hZSmNWpYIqTa7SpbkUv6HXwQNtSwRQogRBskStSwRUm12lSzJKgwdIOSoZYkQQowxuA1HLUuEVJtdJUtiIx28qWWJEEIMVWxZonGWCKk++0qWRPyWJRG0KFZpoFSWgDEaQoAQQnS8XfjJUp5CTV8uCakmu5nu5Madh5gy5zekpxdBowXUGga1phCbD8wDY8DpnfPh5uJo7WoSQohN8HYxHMU7LU+JAE8HK9SGEPtmNy1Lao0Wd+6mI6+IoVDBoCwBNFpA16BUrCixbgUJIcSGOEtFkIn5IZ46eRNSPXaTLMllYrPblUpVHdWEEEJsn/FRvKmTNyHVYT/JUoUBKSsqVlLLEiGElEedvAmxDLtJlqRS8y1LCkqWCCGEx9uZWpYIsQS7SZbkMvMtSwoF3YYjhJDyKj4RRy1LhFSP3SRLUon5B/eKKVkihBCeii1L6dSyREi12E2yJBAIIJWYvhWnVNFtOEIIKY/mhyPEMuwmWQIAmZl+SzR0ACGE8FV8Gi6Vhg4gpFrsK1kyM3yAgoYOIIQQnop9lnKKSqBU0yjehFSVfSVLZlqWFNSyRAghPBVvwwF0K46Q6rCrZMncWEsK6rNECCE8rnIxJKKKo3hTskRIVdlVsmRurCUaOoAQQvg4jjNoXUqnfkuEVJldJUvmxlqioQMIIcRQxU7eKbmULBFSVXaVLJkbOoBuwxFCiCGfCp286TYcIVVnV8mSuZaloiIKAIQQUlHFgSlTqYM3IVVmV8mSq7Pc5LasnMI6rAkhhNiHirfh0qjPEiFVZlfJkqeHs8ltGdn5dVgTQgixDxVvw9H8cIRUnV0lS17uppOlzCxKlgghpCKDUbzpNhwhVWZfyZKZlqWs3AJotdo6rA0hhNi+ii1LucUlUJTQKN6EVEWNk6UDBw5g7NixCA0NhaOjI/z9/TFkyBCcPXvWEvXj8TTTsqTRaJFOrUuEEMLjXaFlCaBRvAmpqhonS99//z2SkpIwZcoU7Nq1C4sXL0ZaWho6deqEAwcOWKKOer7e7ma3305KqfSxbiU+wpUb95F4Lw2Z2QVQ03xJhJB6yFkqglws5K2jTt6EVI2opgdYunQpvL29eev69++Ppk2bYv78+Xjqqadqego9/4bukMskRgegHDsyFk4Oht+gTHl7zircSHjEW+fkKEOjhh5oEuiDpsG+6NSuKSJbBkEkEpo4CiGE2DaO4+DjIkVSZpF+HfVbIqRqapwsVUyUAMDJyQktW7bE/fv3a3p4HoFAgGbBDXHp+j2DbcfPnEVO7i2s/0MAxgnAgQM4ARjHgeNKfy/7vwCPMrIMjlFQqED8nYeIv/MQwHks/glwcZJjcN/2eGVYDwQHGL5WQgixdd4usgrJErUsEVIVNU6WjMnNzcW5c+eq1Kqk1WoRHx//2HIhjd2NJksypoCzRgtU9m6aunLTo+QVFGPN70ewbstRPB3bCi8PjYajg+FM3rUlIyMDACp1bZ40dG2Ms7frolLV7VRFKpXKbq6NpcjBv8bXkx4i3qusdcne3jN1ia6NafZ0bWoaZ2rlabiJEyeisLAQH3zwgcWPHduxmdH1XBVfCWNVK69lDDsPXsHkjzfibnJm1XYmhBAr8nTgdyXILFJbqSaE2CeLtyx9+OGHWLt2Lb755htERUVVej+BQIDQ0NDHlgsNDcXeo7ex++AF/v5c1epZxVxJLy0zH9M//wM/LxqPdq2Dq3mUytNl7JW5Nk8aujbG2dt1kUhMT2NUW+ezl2tjKWFpCcDVXP2ygpPyroG9vWfqEl0b0+zp2tQ0zlg0WZo7dy7mzZuHTz/9FJMmTbLkoXk+nT4KuflFOH7mpn6di28TeLUMglarhZZpodVowRgD02r163S/M6YF4y4AqN7ku4VFSrw+fTk2fDcFTYMaWuZFEUJILfGmUbwJqRGLJUtz587FnDlzMGfOHMycOdNShzXK2VGGHxf+F1v3nMLOA+dxK+ERYju0xgtDu1X6GDt230RhUe7jC5qQl1+EiR/8hD9+fhcyad1+MyaEkKowmB+OnoYjpEoskix98sknmDNnDmbNmoXZs2db4pCPJRYJMXxQZwwf1Lla+2/7+T2UqDXILyhGbl4RcvOL8CgtBwn3UnHhahIux997bL+mhHtp+OKHHfhg8nPVqgMhhNSFislSvlKNQqUajtJaecaHkHqnxp+UL774Ah999BH69++PgQMH4sSJE7ztnTp1qukpaoWHmxMAwMfL1ej2tIxcLPpuI7buv2w2afpl898YPqgTmof41UY1CSGkxrydDZ/gTctXIpiSJUIqpcaflO3btwMA9uzZgz179hhsZ1V97MxGeHu54vOPXkNEi+2Yv2w/Skw8PKLVMixcth0rPn+zbitICCGV5CgVwVkqQr6yLJCl5ikQ7OVoxVoRYj9qPHTAoUOHSjtSm/ixdy+OGIQXnzY+XIHOoX+u4dL1u3VUI0IIqTrq5E1I9dXKOEv1CcdxmPjGiwgPMt+JO27j4TqqESGEVB118iak+ihZqgQ3N3eMGd4VUrHpMrsOnEdKek6d1YkQQqqiYrJELUuEVB4lS5UUGxOL1sGmpznRaLT4Y+/pOqwRIYRUnsFtuHxqWSKksihZqiRXVzcM6hkOkdB0mS17T9eLflqEkPrHx5lalgipLkqWqqBLly5o6mf6AcI7Sam4diu5DmtECCGVU/E2XEouJUuEVBYlS1UQGBSC1k2Mj8uks/Ovc3VUG0IIqTxfN36y9Ci3GFottYQTUhmULFWBQCBAbLcouDoaztrLxFII/Bqjawfbn1CQEPLkaeQm5y2XaBjSC6jfEiGVQclSFYWHt0Fww9JbcR7OArQJEUPaJhLqDj2gDA4Dc/Wwcg0JIcSQl5MUEiE/5D/ILrZSbQixL5QsVZGvnx/atnDDsG5yPNNJjsgmEgS5lQBcaWvT/uupVq4hIYQYEgg4+FW4FZecQ8kSIZVByVIVcZwAEeEt4SQvu3QBwmz97/uvp9ITcYQQm+Tvzr8Vl0wtS4RUCiVL1dCsOb9fUkNBHsQonXPpQXYxbqcVWKNahBBiln+FfkvJOUVWqgkh9oWSpWoIDm4KobBsCAEBx+ArzNUvH7udYY1qEUKIWf5uDrxlalkipHIoWaoGiUSCRo0DeOv8BGXJ0tHbmXVdJUIIeSyD23DUZ4mQSqFkqZqCg5vwlv3KtSydSMiEWqOt6yoRQohZBrfhsoupjyUhlUDJUjUFVUiWPARFkEMFAChQqnHxQa6x3QghxGoaVWhZKlRpkFtcYqXaEGI/KFmqJj8/f0il/Ikpy7cuHad+S4QQG9PQVaYb5USPxloi5PEoWaomgUCIgMBg3jo/YY7+9yOULBFCbIxYKIBvhTni7mXRE3GEPA4lSzUQFFSh35IgF0Dp/f/zCWn488hlK9SKEEJMC27gyFtOzCi0Uk0IsR+ULNVA+U7eGi1DQU4RZHfjIbx4Euz4AUyc+SPuPaQWJkKI7Qjy5CdLCemULBHyOKLHFyGmNPD2gVgix64T2UjN1qD0Abj7vAz0nzM3ETDYy0o1JIQQvmAvfrKUlFkIQGq8MCEEALUs1QjHcWjWrBlyCrQwNVLAP2dv1m2lCCHEjBC6DUdIlVGyVEMhTZrB11NocvuxMzeh1dKYS4QQ21DxNlxWoQr5So2VakOIfaBkqYaCg5vA18N0spSTV4j4Ow/rsEaEEGJaYw8HCAX88QMe5tFYS4SYQ8lSDbm6uiGsifk+SXQrjhBiK8RCAQI8+HPE3adkiRCzKFmygNYtW8DVkTO5/diZG3VYG0IIMS+kQifvu9kqK9WEEPtAyZIFBAc3gZ+ZW3GnLtyBqkRdhzUihBDTQn2decsJWUor1YQQ+0DJkgUEBjWBr6fpURiUyhKcv5JUdxUihBAzwnxdeMsJ1LJEiFmULFmAXC5HZFgjgzmXytt/5FLdVYgQQsyomCxlF2uQU0yt34SYQsmShbRsGQYvF9OXc/fBCzSEACHEJgR5OkIm5scral0ixDRKliwkNKwVArxN91tKzcjFpev36rBGhBBinFDAoUVDfuvSnUzqt0SIKZQsWYi3d0O0bm5+CIFtf56po9oQQoh5rf35ydK1NIWVakKI7aNkyUI4jkPnDm3h4Wz6kv6+5xQKi+jbGyHE+toHevCWr6UpwBizUm0IsW2ULFlQ69aRCPIxfSuusEiJ33efrMMaEUKIcVGB7rzlXKUWCTRPHCFGUbJkQQ28fdC9fbDZp+KWrvoTBUXU3E0Isa5G7nI0dJHx1p1OzLJSbQixbRZJlvLz8/Hee++hb9++aNCgATiOw5w5cyxxaLvTM6YbAhqYbl3KzM7H7EW/0ZNxhBCr4jgO7YP4rUuHbqRbqTaE2DaLJEuZmZlYvnw5lEolhg4daolD2q2wluHo1LqB2TLb9p3FuGk/YO/hi0i6nwaFkh7ZJYTUvZjm/Fh15FY6lGqNlWpDiO0yPex0FQQGBiI7OxscxyEjIwM//vijJQ5rlwQCIV4ZNRRHLqzAw0zTQefo6XgcPR0PAHj9hV54b/zgSp9j0OgFSM3IBVD67bA27V79PjzdnR9fEMD5K4l4c8aKWq2PTofIJnjrpa6VLv/Zt1uwZc/pWqxRmVlTnsPgPu0rXT52+FwUFVuu479GU/q+EwoNWziP/P4xpJLKfez/Pnkd0z5ZbbF6mZKemVfr5yDG9Qz1BscBun7dhSoNTiRkGSRRhDzpLJIsWeIPtlarRXp6aROwu7s7RCLjVdOV0XFzc4NYLDZaNiMjg/d0h6urKyQSidGymZmZvFtjLi4ukEqlRstmZWXp/yAZK9useShGPt0KX6+9BHWJAmBlx+WEEggE/Nd24todfLF+F0oURRAwLUSC0usplctRXFzav+nYjWTorvLD9GzkZOcCrKwOnEAMgdD4ddColZUuq1UrwcqVXbNjP7w8y5rqy/9TFxcVoaREBe7fmj1Ky0d2rvEOolqNCkxbNkIwxwkhEBm/vlpNCZi23CzonBDCCmUT7qXg7wvxUKmU+OfyrX/LASKxCI6OTvo6CQWASMjhRsJ9ZGblgmlVZo+rr4NWDaYpX1YAoUhmtCzTqqEtV/by5UsIbmj8/atSqVBQUKBfFggEyMzOg0JpOHoy02qg1ZRPojgIxXITdahYFhCKHQzKXb1yERzHkJfHT1A8PDwMyt66fQ9ZOfnQqvl97AQiudHPPGMMWnVxlcuWqOt2xnuNRlNvYk152dnZUKvL3kfOzs6QyYy/Z3NyclBSUnrdw9wYrqSpIBCXHvePC8m8ZCk3NxcqVdn729HREQ4Ohu8tY2UdHBzg6OhotGxeXh6UyrL3rFwuh5OTk9Gy+fn5UCjK3ocymQzOzsa/xBUUFKC4uOx9KJVK4eLiYrRsYWEhioqK9MsSiQSurq5GyxYVFaGwsLBSZYuLi3mfc7FYDDc3N6NlFQoF8vPz9csikQju7u5GyyqVSt5nVygUGv3sGisrEAjg6elptKxKpUJubq5+meM4eHkZHwqnpKQEOTk5AErfn+ao1WpkZ2fz1jVoYDwR12g0Bsfz8vIyGj+0Wi0yMzN56zw9PSEQGN4sY4whIyOD9zmqDoskS5bw4MEDeHt7AwCuXLmCVq1aGS0XFBTEe3OfOnUKHTp0MFo2LCwMGRkZ+uWDBw8iNjbWaNmoqCjcvXtXv7xjxw4MHDjQaNnu3bvj2rVr+uXffvsNw4cP1y9zHIdxY17E6csPsH7Vd1AWpOm3eTfrDReflrzjiQqSUXwrHT+tXIukpLKBKwcN6IuOHaMAAI/KjWepUSmRemMvirKT9Os8g7rCvVGU0fqm3foLhZm39cvujaPhGdjJaNn0hL+Rn3Zdv7welzBoQC+jZbfv2ItTp8/pl8Mj2gIu3Y2Wzbp3CjnJZWWdvcPg07yP0bLZD84i+/4p/bKjZ1P4hg3glSnMzUDazVM4evQE9u47qF/fvFkTvPzSCMPX9VCJvNSryEg4rF8nc/FHo4hhRutQkH4Tabf265elTt5oHPkfo2ULsxKREr9bv7zw5hawgjeMlr1x8zbWrN2oX3Zzc0XDNqONli3Oe4SHV37XLwtEMoR0Mn5cZWEGHlzcoF/mBCI06TLBoNz2bZuQkZGBb5fyW38/mfu+Qdl7aWqolQW4eyaOtz6k83hwRpJtrboYiSf5xw3u+JrRpI1p1QZl60piYmK9iTXl9e3bF2fOlI3ltnLlSowZM8Zo2aFDh+Lw4bLPgkef/8K53SAAwK7LjzBncCu4yEr/jV944QXs2rVLX/bzzz/Hu+++a/S4r732GjZt2qRfnj17tsn+q5MnT8aqVav0y9OmTcPChQuNlp0xYwa+++47/fKECROwdOlSo2Xnzp2LRYsW6ZdHjx6NuLg4o2UXLlyIuXPn6peff/55bNy40WjZpUuX4r333tMvDxgwADt37jRa9ueff8akSZP0yzExMTh06JDRsr/++iteffVV/XL79u1x+rTxVvBt27ZhxIiy+NayZUtcvXrVaNn9+/dj0KBB+uXAwEAkJSUZLXv8+HH07NlTv+zl5WXwZUHnwoULiI6O1i/L5XLe56S8GzduIDw8nLfO1PAU9+/fR3BwMG9dQUGB0WQ7MzNT/xnWSUtLM5qIFRUVGZStDptJlspLSEgwegsBMLzQSUlJJr9hVMwk7927h/j4eKNldd+ydO7fv2+ybPlvQwCQnJxstOzkcYPx+7pleNwNFmEt30ojhBhXX2INAF7LCwA8evTIZNmKf9zKRyBFiRbf7TqLoS3dAIDXQgKU/lEyddzyLSRAaYubqbLlWzKA0lYKU2Urtk5kZ2ebLFuxdSI3N9dk2fIJLlBa/4pldWXS0tJ46wsKCkweNyUlhbdcVFRksuyjR494ywqFwmTZ5ORk3rJSqTRZ9v79+7zlkpISk2Xv3ePPLqHRaEyWrZhwMcZMlk1ISDBYV9nXBpQmW8ZaMY21aN26dcugtQkwfK9XFw0dUIvkcjka+xlv9izPRKw2iYaNI4RYUqMKg+luvJIDlZqe2CVExyZblkJCQhAaGmp0W8X7l0FBQSbLVvzGGBAQYLJsxb4IjRs3Nlm2Yp8Bf39/k2XlcuN9TPgHdIJS4gAtx//nUApkyOdKv8kKBBwYAA4MDMVGDlI7Sjixvg4V07QSjn/N1Khi1lcDaoiQwxxRDH6/kBIIkQNdvwcGxko7rypY3T3ho2UcioQucJKJ4CoXQyIs+0OUmsb/RiQUCP99T9dNCuzt7Qsww5ZMHx9fg3V5qkIA9ftR8voUayr2T/L19TVZtuK39faNnbG/3HJmkQZHMySYENvUoB+Rt7e3yeNWbHnz8vIyWbZifx8PDw+TZSv24XF3dzdZtmIfHldXV5NlK/bLcXZ2NiirawmpeCvHycnJ5HEbNmzIW3ZwcDBZ1teX/9mTyWQmy16+fJm3LJVKTZa9c+cOb1ksFpssW7ElTCgUmixbsfWQ4ziTZY31E6rs+xcAWrRoYfQ2nLFbhM2aNTN6G658P7Oa4JiFx7fPyMhAgwYNzN6rLq9Zs2a4ffs2GjVqhHPnSvu01NdOl0XFShw5fRuHT97AucuJyCsoTXoWzXoJQ/p14HW6BEo/jLq+DeXfYN2e/QgPU9LqpIP3gd/mIijAz2jZip0ur9xMxhszVho/roU7eHeJao73/9sLxcXFaNSokX69qU6Xsz7/Feu3HqmTDt7qkJYQBDT/93UCvUK9MbV3c4T7uxrtdPnUqAUoKjYcPqI2Onhf+esLCAWGtzWMBZm/jl7GmzOW13oH7/TrW1CQl4mmTZvi1q1bRl+fJehiTXBwME6eLB1Jv77GGqDyHbyB0lsgsw9l4mJK2b+1VCTAnqk94CFWP9EdvHXJUkBAAHXwBr+Dt+7z2qVLF6NlbamDd8eOHZGYmFjtOGMzyVJAQACv02N9xxhDVk4BUtJz4ePlAi8P4x9m3Qe1fLK0++B5KJQl+sd9a3M+p4G92kImNR70K0rPzMPfJ68/vqAFeHu6oIFr6YfI1DeV8s5dTkTi/bTHljNFyxiyC0uQXqBAWr4S6eV+cov5fVCYixsg5/+B4Djg2Uh/vNOvBfzd+AnPH3tPQ62x3C0PXR+Iit9YAWBovw4QCit39/1hahb+OVt7yYvOlDdHIPnBvTpLlmr7PPYoPj4eNzIUmLozGeXDSZivC34f3wVySd21GtsaYzGYlLKna1PTz7/FbsPt3r0bhYWF+gz52rVr+qciBgwYYPKbyJOK4zh4ujtXegyj8p7u2bYWalRzDTxdMGxAxzo7n6mOgsa0ax2Mdq2DH1+wGlJyFTh4Iw0H4tNw7HYGilSGTc+MAb+fT8aOy48wtmswJvRson/aaEg/409YVZelApifj0ed/HvOkFUuGSe1q4WXDKM7ByHueJJ+3fVHeZjy63ksfbEdxJVMsgmpjyyWLI0fP57XMrRx40b9I5iJiYkICgqy1KkIsSkNXWUYFR2AUdEBUKo1OJmQhT+vpeCP8w+RX2H8JJVai2WH72DD6XuY/UwrDIn0q/WBRQmprHf7tcDR2xm4nVZ2C+nPa6l4e8MFfD0yEiJKmMgTymLv/KSkJDDGjP5QokSeFFKRED2aN8C8oa1xYmYvvNuvBZykht9JsotKMHXDBbzz20UUqQwHpCTEGhylIvzwcpTBe3bHpUd4b9MlaLT0LC55MtHXBEJqiaNUhIk9m+LQu7F4pXOgfmT28n4/n4znvjuO+1mWGQuEkJpq0sAJP41uD5mY/+fh9/PJmLGZEibyZKJkiZBa5uUkxcdDwvHn2z3Qr5WPwfb4lHw8+90xXHqQU/eVI8SIjiGe+PGVDpCIKoy/dPYB3t10EVpKmMgThpIlQupISAMn/PBye3z7Qls4V7jNkVGgwn+Wn8Cx2xkm9iakbnVr5oXlL0fxxgkDgN/PJePjHddq9SlcQmwNJUuE1LFBEX7Y9lY3NPPmjylTpNJg3KrTOJlgOGQ/IdYQ28Ib37/UziBhijuehB/+NpzKgpD6ipIlQqwg2MsRmyd0Qdem/EHiFCVajI07jXP3sk3sSUjd6hXmgx9ejoJYyO9zt2B3PDaffWClWhFStyhZIsRKXGRirBwTjYGt+YNHFqo0GP3zKVxJzjWxJyF1q2eoNxYNb2OwfvrmSzh0o/qDvRJiLyhZIsSKJCIBvv5PpEHH73yFGi//dBI3U/NN7ElI3RoS6Y9ZA8N469Rahglrz+Hi/RzrVIqQOkLJEiFWJhYKsGRUW8S24M+ZlF1UgrFxp5FZoDSxJyF167XuIXi9O38k/CKVBmPjTiMxwzITlhJiiyhZIsQGSEVCLHspCl2a8PswPcguxvg156BSW27uOEJq4v2nwzAkkj+5dmahCq/8fBJp+QoTexFi3yhZIsRGyMRC/Di6PSIbu/HWn0rKwqytl+lRbWITBAIOC59vg25N+bPS388qxqsrT6NASSPSk/qHkiVCbIiDRITlr0TBz1XGW//bmQdYfeKuib0IqVsSkQDLXo5CuL8Lb/3Vh3mYtO4cjfJN6h1KlgixMd7OMix/pT3kYiFv/Sc7ruECdaQlNsJJKsLKMdEI8HDgrT90Ix1f7rthpVoRUjsoWSLEBoX7u+LLEfxHtUs0DBPXnkNOkcpKtSKEr4GzFL+MjYano4S3funBO/jzaoqVakWI5VGyRIiNerq1L97sEcJbl5xTjPc2XaL+S8RmBHk54vuXogwmin7nt4tISC+wUq0IsSxKlgixYdP6tUB0kAdv3Z/XUrGG+i8RGxId7GEwBlO+Uo3/rjmLQurwTeoBSpYIsWFioQDfvNAWHhVuc3yy8zquPcyzUq0IMTS6S5DBkAI3UwswfTO1hBL7R8kSITbOx0WGRcMjeOtUai3eWn8ORSr61k5sA8dx+Oy51ght6Mxbv+PSI/x0NNFKtSLEMihZIsQOPBXqg7Fd+SMn30kvxMfbr1mpRoQYcpCIsOylKDjLRLz1n+2Ox4mETCvVipCao2SJEDsx/ekWBuPa/Hr6PnZeemSlGhFiKMjLEV+PjOSt02gZJq07h5RcGuGb2CdKlgixE1KREN+MagcHCX/8pfd/v4QH2UVWqhUhhnqF+WByr2a8dRkFKoxfe5am7iF2iZIlQuxIsJcjPh4SzluXp1DjrfXnUaKhP0LEdkzt1cxgcujz93LwyQ66dUzsDyVLhNiZYe38DZ46On8vB4v+pFGTie0QCDh8PTISjT3kvPWrT9zF5rMPrFQrQqqHkiVC7AzHcZg3NByBnvxpJn44nIC/rqdaqVaEGHJzkOD7F6MgFfH/1MzcchlXH+ZaqVaEVB0lS4TYIWeZGN+OageJkP8RfnvDBTzKL7FSrQgxFO7vivnPtuatU6q1+O+aszR1D7EblCwRYqdaN3LFzAGhvHV5CjXmHkiBooT6LxHbMSyqEV7uFMhbdz+rGFM3XIBWSwNWEttHyRIhdmx0lyAMjPDlrUvKVmHR0TT6I0RsyoeDWqJdgBtv3aEb6fj6r1vWqRAhVUDJEiF2jOM4fD4sAs28nXjrj94txJf7blqpVoQYkogE+O7FKHg58afuWfLXLey+TGOFEdtGyRIhds5RKsLyV9objJr87cHbWHuSJtwltqOhqwzfvtAOQgHHWz/l1ws4fjvDSrUi5PEoWSKkHgj2csR3Lxr+EZq19Qr+uJBspVoRYqhTiCfef5rf106l0WLcqjM0JQqxWZQsEVJPdG/WAB8PacVbxxjwf79dxPaLD61UK0IMjesWjBc6BvDWFZdo8OrK0zh0I81KtSLENEqWCKlHXuwYiBfbuPPWabQMU349j9X/JFmnUoRUwHEcPhkSjoGt+Q8nFJdo8NqqM9hEg1YSG0PJEiH1zEuR7hga5spbp2XAh39cxUd/XIGapkUhNkAo4PD1fyLRr5UPb71ayzBt40Us2nsDGnqik9gISpYIqWc4jsOb0Z4Y3TnQYNsv/9zFSz+dxKPcYivUjBA+sVCAb0a1w9AK0/cApQ8ovPzTSTzMofcqsT5KlgiphziOw5zBrQxmfgeAEwlZ6PvV31h78i59cydWJxEJ8OWISLzRI8Rg2/E7mej39d9YfYLeq8S6KFkipJ7iOA7/16c5Fv8nEpIKc3PlK9T4YMsVDFxyBH9eTaEBLIlVCQQcZg4Iw9zBrSCq8ERnvkKND7deQf+v/8auy48oaSJWYZFkqaCgAFOnToWfnx9kMhkiIyPx66+/WuLQhJAaGhLpj1/f6ARfV5nBtviUfLyx+ix6fXkYyw7fQTLd8iBWNLpLENa9bvy9eiutABPWnkOPzw9i6cHbuJdZZIUakieV6PFFHu+5557D6dOnsWDBAjRv3hzr1q3DqFGjoNVq8cILL1jiFISQGmgX4I49U3pgzvar2HLecNylxIxCLNgdjwW74xHm64J2AW5o08gNrRu5opm3E0RCaoQmdSM62AN7pvbA3O1X8fs5w/dqck4xFu69gYV7byC0oTOigz0QFeiOyMZuaOTuYDDWGCGWUONkadeuXdi3b58+QQKAnj174u7du3j33XcxcuRICIXCGleUEFIzrg5ifDUyEsPaNcInO67hRmq+0XLXH+Xh+qM8rD15DwAgEwvQ2N0B7g4SuDqI4e4ghruDBG4OErj9u6z73UkqgkgggEjIQSTgIBIKSv8v4CAUcOA4+kNGHs9VLsaXIyIxPKox5m6/ivgU4+/V+JR8xKfk45d/Skeql4oECPZyhL+bHN4uUjRwlqGBsxQuMhEcJSI4SIVwlIjgKBVBKtK9T/99j+p+//e9S+9VUl6Nk6UtW7bAyckJw4cP561/9dVX8cILL+DkyZPo0qVLTU9DCLGQbs28sGtKd2y/+BDLDt8x+YdIR1Gixa20Aoucm+MADoCA4/79ncNdup1CTOjcxBO7JnfHriuP8P2hO7j6MM9seaVaq0+gakqXK3H65YRyv5e+d3UFjK0vv399Tby0Wg0AQCCw/WmV7mXVLM7UOFm6cuUKwsLCIBLxDxUREaHfXplkSavVIj4+vqbVqXcyMkrnS6JrY4iujXGVvS6hcuCrfg1wNc0Z+27n49jdQhSoancMJsYABkDLdJ10GRjqtsOuSqWi90wFtv5ZaioBFvXxwo0MZ+y5mYd/7hciV1H771UAZe9OVvF9Sh3Ny6itXYHH0hr8+1VNjZOlzMxMhIQYPvLp4eGh304IsU0cxyHcR45wHzne6sxwPU2B+HQFbmYqcTNDidQC2w+C5MnAcRxCG8gQ2kCGt7QM8ekKXElV4FqaAtfTFchT0mCrpPZYpIO3uSbGyjY/CgQChIaGPr7gE0b3TY+ujSG6NsbV5LqEt+QvZxWqcP1RHjIKlMgpKkFOUQmyi1TIKVIhu6gEOcUlpb8XqpCnsI/ESiKR0HumAnv8LLUCMOzf3xljyCxU4U5aARIzCpGWr0RqngJp+UpkFihRqNSgUKVGoVKNQpUGKjUlVqRqapwseXp6Gm09ysrKAlDWwkQIsT8ejhJ0bepVqbJqjRZKtRZqDYNaq4VGy1CiZdD8u6zWMmgZA2PQ/58xYNAGOe5l1/ILIfUax3HwcpLCy0mKjiGejy1fotFCpS59T6o1/Pdqibb0PcxQ+h5NTEwEAAQGBQHg340rva3Myv2uW//vOhi5e1eP3L1b2lcpMNBwtgBb8+wGOe7XIM7UOFlq3bo11q9fD7Vazeu3dPnyZQBAeHh4TU9BCLEDIqGgWkMMVBwwk5DaJhYKIK7ke5XLlQIAQv1cH1PyyeNYnAoACA10f0xJ65PWMM7UOEo9++yzKCgowObNm3nrV61aBT8/P3Ts2LGmpyCEEEIIsZoatyw9/fTT6NOnD8aPH4+8vDw0bdoU69evx549e7BmzRoaY4kQQgghds0iHbx///13fPDBB/joo4+QlZWF0NBQrF+/Hv/5z38scXhCCCGEEKuxSLLk5OSExYsXY/HixZY4HCGEEEKIzaCelYQQQgghZnCMWffBRgcHBxQXF0MkEiHo30czSRmVSgWgdGwYwkfXxjh7uy53795FSUkJ5HI5iopqb+oTXawRi8V28ahzXbK390xdomtjmj1dm5rGGasnSxKJBCUlJdasAiHEBojFYn3wrQ0Uawgh1Y0zFumzVBNOTk4oKCiASCSCv7+/tatDCKljycnJUKvVcHJyqtXzUKwh5MlV0zhj9ZYlQgghhBBbRh28CSGEEELMoGSJEEIIIcQMSpYIIYQQQsygZIkQQgghxAyrJUsFBQWYOnUq/Pz8IJPJEBkZiV9//dVa1bEp+fn5eO+999C3b180aNAAHMdhzpw51q6W1R04cABjx45FaGgoHB0d4e/vjyFDhuDs2bPWrprVXbhwAQMHDkRAQADkcjk8PDzQuXNnrFmzxtpVszqKNaZRrDGOYo1pT2qssVqy9Nxzz2HVqlWYPXs2du/ejQ4dOmDUqFFYt26dtapkMzIzM7F8+XIolUoMHTrU2tWxGd9//z2SkpIwZcoU7Nq1C4sXL0ZaWho6deqEAwcOWLt6VpWTk4PGjRtj/vz52LVrF3755RcEBQXh5Zdfxrx586xdPauiWGMaxRrjKNaY9sTGGmYFO3fuZADYunXreOv79OnD/Pz8mFqttka1bIZWq2VarZYxxlh6ejoDwGbPnm3dStmA1NRUg3X5+fnMx8eH9erVywo1sn0dO3ZkjRs3tnY1rIZijXkUa4yjWFN19T3WWKVlacuWLXBycsLw4cN561999VU8fPgQJ0+etEa1bAbHceA4ztrVsDne3t4G65ycnNCyZUvcv3/fCjWyfV5eXhCJrD72rNVQrDGPYo1xFGuqrr7HGqskS1euXEFYWJjBhY2IiNBvJ6QycnNzce7cObRq1craVbEJWq0WarUa6enp+O6777B3715Mnz7d2tWyGoo1xFIo1vA9abHGKmlgZmYmQkJCDNZ7eHjotxNSGRMnTkRhYSE++OADa1fFJkyYMAE//PADgNK50JYsWYI333zTyrWyHoo1xFIo1vA9abHGam1m5pp+qVmYVMaHH36ItWvX4ptvvkFUVJS1q2MTZs6ciddeew1paWnYvn07Jk2ahMLCQkybNs3aVbMaijWkpijWGHrSYo1VkiVPT0+j3+iysrIAlH3rI8SUuXPnYt68efj0008xadIka1fHZgQEBCAgIAAAMGDAAADA+++/j9GjR6NBgwbWrJpVUKwhNUWxxrgnLdZYpc9S69atcf36dajVat76y5cvAwDCw8OtUS1iJ+bOnYs5c+Zgzpw5mDlzprWrY9Oio6OhVquRkJBg7apYBcUaUhMUayqvvscaqyRLzz77LAoKCrB582be+lWrVsHPzw8dO3a0RrWIHfjkk08wZ84czJo1C7Nnz7Z2dWzewYMHIRAIjPbbeRJQrCHVRbGmaup7rLHKbbinn34affr0wfjx45GXl4emTZti/fr12LNnD9asWQOhUGiNatmU3bt3o7CwEPn5+QCAa9euYdOmTQBKmzwdHBysWT2r+OKLL/DRRx+hf//+GDhwIE6cOMHb3qlTJyvVzPreeOMNuLi4IDo6Gj4+PsjIyMDGjRuxYcMGvPvuu/WyWbwyKNY8HsUaQxRrTHtiY421BnjKz89nkydPZg0bNmQSiYRFRESw9evXW6s6NicwMJABMPqTmJho7epZRUxMjMlrYsW3sk34+eefWffu3ZmXlxcTiUTMzc2NxcTEsNWrV1u7alZHscY8ijWGKNaY9qTGGo4xxuoqMSOEEEIIsTdWmxuOEEIIIcQeULJECCGEEGIGJUuEEEIIIWZQskQIIYQQYgYlS4QQQgghZlCyRAghhBBihtUm0iV1IyUlBd988w12796N27dvQ6FQwNPTEz4+Pmjfvj1iYmLw7LPPwsnJSb/P119/jZycHIwZMwZBQUHWq7yVffvtt8jIyMCbb74JX19fa1eHEJtGsab6KNbYPhpnqR47duwYBg0ahJycHHAcB39/f/j6+qKwsBC3b9+GSqUCABw5cgTdunXT7xcUFIS7d+/i4MGDiI2NtVLtrUulUsHZ2RkqlQo5OTlwdXW1dpUIsVkUa6qPYo19oJaleqqgoADPP/88cnJy0KdPH3z77bdo3ry5frtSqcTBgwcRFxcHsVhsxZrapsuXL0OlUiEkJISCFyFmUKypGYo19oGSpXpq586dSElJgZOTE7Zs2QJHR0fedqlUiv79+6N///5WqqFtO3v2LACgbdu2Vq6J5Z09exb79u3DqVOncPLkSTx8+BBSqRQKhaLWzpmVlYXQ0FCkp6ejRYsWiI+Pr7VzkbpFsaZmKNZYVq3FGuvOtkJqy/z58xkAFhUVVel9Vq5caXY+pNmzZ/PKa7VatnbtWtanTx/m6enJxGIx8/f3Z6+88gq7fv260XPo5qE6ePAgu3z5Mhs2bBhr0KABk8lkrE2bNmzp0qVMo9EY3ffhw4ds8uTJrHnz5kwmkzG5XM4aN27MnnrqKbZgwQKmUqkq/VpN2b59u9lr8Oabb9b4HNY2ZMgQg9cllUpr9ZyjR49mHMcxAKxFixa1ei5StyjWVA/FmtpRW7GGkqV66ttvv2UAmIuLC8vMzKzUPrt27WJdu3ZlUqmUAWDh4eGsa9eu+p+ffvpJX7aoqIgNHDhQ/wHw9fVlkZGRzNHRkQFgcrmc7dq1y+AcugC2YMECJpfLmYODA4uKiuJN5jlq1Cim1Wp5+92/f581bNiQAWBisZiFhYWx9u3bM19fX/0HIzs7u0bXjDHG1q5dy7p27ap/Ha1bt+Zdgw0bNtT4HNa2YMEC9tFHH7Ht27ezlJSUWg9g+/fvZwDYG2+8QclSPUSxpnoo1lhebcYaSpbqqVu3bjGhUMgAsMjISLZu3TqWnp5eqX3LfyMzZezYsfpvk+fPn9evV6lUbNasWQwAc3d3Z2lpaUaPLRaL2XPPPcdycnL02zZv3qwPnsuXL+ftN2XKFAaA9e7d2+B1pKSksK+++ooVFhZW6vVVhqenJwPAkpKSLHZMW1WbAayoqIg1bdqUtWzZkt28eZOSpXqIYk3NUKyxjNqONZQs1WOLFi3SfxPS/QQFBbHnn3+efffddyYD2uMC2LVr1xjHcczLy4s9evTIaBld8+v8+fONHrtBgwasqKjIYD9d8AsJCeF94+vXrx8DwLZu3VrJV199d+/e1QfgJ0FlAlh+fj6bO3cuCw8PZ3K5nDk7O7MePXqwLVu2mN1v+vTpjOM4dvjwYZaYmEjJUj1FsaZ6KNYYstVYQ8lSPXfs2DH27LPPMplMZnDv2MHBgS1YsMBgn8cFsE8++YQBYOPGjTN53ri4OAaA9enTx+ixZ8yYYXS/1NRUff3i4+P161977TUGgL300ktMqVRW4pVX39atWxkA1rNnz1o9j614XABLSUlhLVu2ZACYv78/Gzx4MOvdu7f+9sFnn31mdL+LFy8ykUjExo4dyxhjlCzVcxRrqo5iDZ8txxpKlp4QCoWCHT9+nC1dupQNHTqUF9AqBrHHBbDnn39e/82x/D328j+tWrViAFhYWJjRY//yyy8m6+rl5cUAsD/++EO/7uLFi0wikTAAzNPTk7300kvsu+++Y9euXav+RTFh9uzZDAD7v//7P4sf2xY9LoA9/fTTDAB77733eB1b79y5w5o0acKEQiG7ePEibx+NRsOio6OZl5cXy8jIYIxRsvSkoFhTeRRr+Gw51lCy9IRKTEzUBxlnZ2emUCj02x4XwHr37m3wzdHUT2BgIG9f3bH37Nljsm66eq1Zs4a3/tSpU2zgwIH6QKb7iYiIYLt37672taho8ODBjw2y9Ym5AHb+/HkGgHXp0sWgIyxjZd+M33rrLd76r7/+mgFgK1eu1K+jZOnJRLHGNIo1ZWw91tDccE+ooKAg/O9//wMA5Ofn49q1a5XeVzddwcKFC8FKE26TP0lJSUaPkZaWZvL4qampAABnZ2fe+g4dOmDHjh3IycnBwYMHMXv2bISFheHSpUsYNGgQTpw4UenXYM65c+cA1M9xT6pq3759AIAhQ4aA4ziD7brRmE+fPq1fd//+fcyaNQsxMTEYM2ZMndST2C6KNaZRrClj67GGkqUnWJMmTfS/66YjAGD0jVpeeHg4gNKRZ6vLVMBMT09HRkYGAKBFixZGy8jlcsTGxmLOnDm4cuUKnn76aWg0GqxYsaLa9dHJzMzEgwcPIJVKERoaWuPj2TvdH6Dp06eD4ziDHy8vLwDQ/5sBwIQJE6BSqfD9999bo8rEBlGsMUSxhs/WYw2N4F1Ppaenw8vLy2wwOn78OABAIBDwgplcLgcAFBcXG93v+eefx7x587Bp0yZ88sknCAgIqHL9fv75Z8yePRsymYy3/ptvvgEABAcH86ZMMEUgEKBLly7YvXs3Hj16VOV6VHTnzh0Apd+GRSL6eGg0GgBA9+7dERISYrKcLpABwI4dO+Dm5obx48fzyuhG7b13755+HrAdO3bwJlYl9odiTfVQrOGz+VhjsRt6xKZ89dVXLDw8nH333XcsJSWFt02lUrFffvmFubm5MQBs6NChvO2DBg1iANi7775r8vhjxoxhAFjTpk2N9je4fv06++ijjwwe9yw/9snw4cNZbm6uftvWrVv1nUGXLVvG2++NN95ga9euZXl5ebz1N27cYE2aNGEA2IcffsjbVn6U4Mq6cuUKA8BcXV1ZVlZWpfezZzDTj0D3NNLixYurdLzK/lhicD9iXRRrKNZUlj3HGkqW6ildpzfdT+PGjVmHDh1Yy5YtmZOTk359ZGQkS01N5e27fv16/famTZuyHj16sJiYGF4HuuLiYjZs2DB9OW9vbxYdHc3atm3LPDw89OtXrFjBO7YugM2fP5/JZDLm4ODA2rdvz4KCgvT7jBgxwqCDX5s2bRgAJhQKWfPmzVnHjh1Zs2bN9GO7tGnThhcMGateAFOpVPqA6Orqyjp27Mi6du3KPv3000ofw96YC2D//PMPAyzzaDN18K6fKNZQrKkse441lCzVUyqVih06dIjNnDmT9ejRgwUHBzO5XM6kUilr1KgRGzhwIPv5559ZSUmJ0f2XLVvGoqKi9ONbAIbzNTHG2LZt29izzz7L/Pz8mFgsZp6eniwiIoKNHj2abdmyxWAwuPJPv1y4cIENHTqUeXp6MqlUylq3bs2++eYbo/M1HThwgL399tusffv2rGHDhkwkEjFnZ2cWHR3NPv/8c6ODzn322WcMAGvbtm2Vrl18fDwbNmwY8/X1ZQKBgAFgCxcurNIx7Im5AMYYY7169WIA2NSpU1l+fj5vm0ajYXv37mVHjhx57HkoWaqfKNZQrKkse441lCyROlWZ6Q0sZcCAAQwA++2332r9XPZkx44drGPHjvofAIzjON66HTt26MunpKSwiIgIBoB5eHiwp556io0cOZJ169aNNWjQgAFgX3311WPPS8kSqUsUa6yvPsUa6lVG6iWtVovjx4+jWbNmGDZsmLWrY1PS09Nx8uRJ3jrGGG9denq6/ncfHx+cOHECy5Ytw4YNG3D69GmoVCr4+vqibdu2GDJkCEaMGFFn9SfEllCsMa0+xRqOMcascmbyRAoKCsLdu3dx8OBB/VMKteHSpUto06YNVqxYgddee63WzkMIsU0Ua4glUcsSqZciIiJA3wMIIbWNYs2TgQalJIQQQggxg27DEUIIIYSYQS1LhBBCCCFmULJECCGEEGIGJUuEEEIIIWZQskQIIYQQYgYlS4QQQgghZlCyRAghhBBiBiVLhBBCCCFmULJECCGEEGIGJUuEEEIIIWZQskQIIYQQYgYlS4QQQgghZlCyRAghhBBiBiVLhBBCCCFmULJECCGEEGIGJUuEEEIIIWZQskQIIYQQYsb/AzE4RzfjB2+pAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 570x300 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# remove any seaborn themes\n",
    "sns.reset_orig()\n",
    "color_list = matplotlib.colormaps['cividis_r'](np.linspace(.09, .85,3))\n",
    "mosaic = \"\"\"\n",
    "aadd\n",
    "aadd\n",
    "\"\"\"\n",
    "figsize = (3.8, 2)\n",
    "y_tick_fontsize = 8\n",
    "x_tick_fontsize = 8\n",
    "fig, axd = plt.subplot_mosaic(mosaic, \n",
    "                              layout='constrained', # will make spacing nice\n",
    "                              figsize=figsize,\n",
    "                              dpi=150,\n",
    "                              facecolor='w')\n",
    "fig.get_layout_engine().set(hspace=0.08, wspace=0.15)\n",
    "\n",
    "plot_single_metric_axis(axd[\"a\"],\n",
    "                        metric_array=np.array(losses), \n",
    "                        y_label=\"\",\n",
    "                        fig_title = \"\",\n",
    "                        log_interval=log_interval, \n",
    "                        legend_str=\"ReLU+ bias\",\n",
    "                        chance_level=None, \n",
    "                        line_width=2, \n",
    "                        x_tick_fontsize=y_tick_fontsize,\n",
    "                        y_tick_fontsize=y_tick_fontsize, \n",
    "                        x_label_fontsize=11,\n",
    "                        y_label_fontsize=11)\n",
    "\n",
    "axd[\"a\"].plot(losses_linear_bias, color=color_list[1], label=\"Linear + bias\", linewidth=2)\n",
    "axd[\"a\"].plot(ana_loss_bias, color=color_list[2], linestyle=\"--\", label=\"Analytical Linear+bias\", linewidth=3.5)\n",
    "# make vertical dotted line at y = 1 \n",
    "axd[\"a\"].axhline(y=1, color='black', linestyle=':', linewidth=2, label=\"linear problem solved\")\n",
    "\n",
    "axd[\"a\"].legend(prop={'size': 4})\n",
    "axd[\"a\"].set_xlim([-100, 36100])\n",
    "# # set title \n",
    "axd[\"a\"].set_title(\"Bias\", fontsize=10)\n",
    "\n",
    "plot_single_metric_axis(axd[\"d\"],\n",
    "                        metric_array=np.array(losses_relu_no_bias), \n",
    "                        y_label=\"\",\n",
    "                        fig_title = \"\",\n",
    "                        log_interval=log_interval, \n",
    "                        legend_str=\"ReLU no bias\",\n",
    "                        chance_level=None, \n",
    "                        line_width=2, \n",
    "                        x_tick_fontsize=y_tick_fontsize,\n",
    "                        y_tick_fontsize=y_tick_fontsize, \n",
    "                        x_label_fontsize=11,\n",
    "                        y_label_fontsize=11)\n",
    "\n",
    "axd[\"d\"].plot(losses_linear_no_bias, color=color_list[1], label=\"Linear no bias\", linewidth=2)\n",
    "axd[\"d\"].yaxis.set_tick_params(labelleft=False)\n",
    "axd[\"d\"].axhline(y=1, color='black', linestyle=':', linewidth=2)\n",
    "\n",
    "axd[\"d\"].legend(prop={'size': 4})\n",
    "axd[\"d\"].set_xlim([-100, 36100])\n",
    "# # set title \n",
    "axd[\"d\"].set_title(\"No Bias\", fontsize=10)\n",
    "plt.show()\n",
    "# save_figure('/relu_nets_bias/', 'relu_vs_linear_linear_and_nonlinear_problem_simple', fig)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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mjwpIiIiIrI9kj1uqP7sJRERERG1TmZ5F6BarUZ5eTlqamoM1nfr1s0cwxIRERGRBUmaWO7Zswevv/46du/ejdra2gbb1dXVSTksEREREVkByX5SvHfvXgwePBjHjx/HM888AyEEBgwYgClTpiA8PBxCCAwfPhyzZ8+WakgiIiIisiKSJZYLFixA27ZtcfDgQSxevBgAEB0djSVLluDw4cN47733sGXLFsTGxko1JBERERFZEckSy3379mH06NFo3769tkyj0Wj//sILL6Bfv354/fXXpRqSiIiIiKyIZInl9evX0alTJ+1nJycnXL58WadNnz59UFBQINWQRERERGRFJEsslUolfv31V+3n7t2749ChQzptfvnlFzg6Oko1JBERERFZEckSy6ioKHz//ffazyNGjMB3332Ht99+G0eOHMGyZcuwZs0a9O/fX6ohiYiIiMiKSJZYJiUlwcXFBaWlpQCAV155BT179sQrr7yC8PBwTJkyBZ06dcJ7770n1ZBEREREZEUke45lbGyszi++3d3d8f3332P9+vU4deoUunbtihEjRsDV1VWqIYmIiIjIipjlzTv15HI5EhISzDkEEREREVkJsySWP/30E44dO4arV6/iySefNMcQRERERGRlJLvHEgC+++47hIWFISIiAgkJCUhOTtbW7dq1C23atMEXX3wh5ZBEREREZCUkSywLCwsRExODsrIyvPjii3pv2HnwwQfh7e2NVatWSTUkEREREVkRyRLLOXPmoE2bNvjhhx/wzjvvoG/fvnptoqKicODAAamGJCIiIiIrIlliuWfPHsTHx6Nr164NtunatSvKysqkGrJZqqurMXPmTCiVSjg5OSEyMtLos6dr167FuHHjEBAQAGdnZ/j5+eGJJ57A8ePHzTxrIqLmYawjIkuS7Mc7tbW1cHNza7TNpUuXLPbmnfj4eBQWFiItLQ2BgYHIycnBuHHjoNFoMH78+Eb7vv322+jcuTNSU1Ph7++Pc+fO4c0330Tv3r2xb98+hIaG3qa1ICJqHGMdEVmSZIllUFAQdu/e3WB9XV0dduzYgbCwMKmGNNqmTZuQl5enDbAAEB0djdLSUsyaNQtjx46Fvb19g/03btyIjh076pQNHjwYfn5+WLRoEZYvX27W+RMRGYOxjogsTbJL4cnJydi/fz/mz5+vV6dSqTBjxgycOHEC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6tWrDLZErRBjHdkCJpZk81xdXQEADz30EPLy8ppsv379ehQVFeGf//wnXn31VZ26qVOnYseOHUaPbWd348EKarVar66qqqrBfjKZTK+sfj3eeOMNpKamGj0HImodGOvIFvBxQ2Tz3NzcEBgYiB9//FHvMo8hp06dAgCMHDlSp1wIgb179+q1rw+odXV1enXu7u4AYPDxGIcOHWp68jfp06cPZDIZCgoKmtWPiFoHxjqyBUws6Y4wdepU/P7773jxxRf1vlELIbBz507t565duwIAdu/erdNu8eLF+OGHH/SW7enpCcBwQHV3d0dAQAA2bNiAyspKbfmpU6fw/vvvN2sdlEolRo4ciY0bN+Kzzz7Tq79+/brB59QRUevBWEfWjpfC6Y4wffp07NixA0uWLEFeXh4GDhwIDw8PnDt3DgUFBSgrK9N+wx8xYgS6dOmC5557Dtu3b4ePjw++//577N27F8OHD8emTZt0lt2vXz8oFAosWrQIly5dgqenJ3x9ffHkk08CAGbMmIHnn38evXv3xqhRo1BRUYF169ZhyJAh+OKLL5q1Hp988gmOHDmCpKQkLFmyBPfccw/s7e1x5swZ5OfnIyoqCps3b25yOcuXL8d3330HANr/vvLKK9qzDi+//DKCg4ObNTcisjzGOl2MdVbIkj9JJ2pK/SMrxo4d22Tburo6sWzZMtG/f3/h6uoqnJychL+/v0hISBCrV6/WaVtSUiIeeeQR4enpKVxdXcWQIUPE3r17xZw5c3Qe7VFv7dq14u677xYKhUIAEAMHDtSp/9e//iV8fX2FXC4XISEhYuXKlY0+gmPChAkNrkdlZaV4/fXXRc+ePYWTk5Nwc3MTPXv2FM8++6zYv39/k9tBiL8eu9HQn+3btxu1HCK6PRjrGOvuFDIhhLh9aSwRERER3al4jyURERERSYKJJRERERFJgoklEREREUmCiSURERERSYKJJRERERFJgoklEREREUmCiSURERERSYKJJRERERFJgoklEREREUmCiSURERERSYKJJRERERFJgoklEREREUmCiSURERERSYKJJRERERFJgoklEREREUmCiSURERERSYKJJRERERFJgoklEREREUni/wHIk9/swRhLcAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 645x300 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "mosaic = \"\"\"\n",
    "aabb\n",
    "aabb\n",
    "\"\"\"\n",
    "figsize = (4.3, 2)\n",
    "sns.reset_orig()\n",
    "fig, axd = plt.subplot_mosaic(mosaic, \n",
    "                              layout='constrained', # will make spacing nice\n",
    "                              figsize=figsize,\n",
    "                              dpi=150,\n",
    "                              facecolor='w')\n",
    "fig.get_layout_engine().set(hspace=0.08, wspace=0.15)\n",
    "# plot the dataset\n",
    "y_tick_fontsize = 8\n",
    "x_tick_fontsize = 8\n",
    "\n",
    "symbol_list = [\"+\", \"x\"]\n",
    "axd['a'].scatter(train_inputs_test[2,:], train_inputs_test[1, :], c=train_labels_test[2], cmap=\"coolwarm\", s=100, marker=symbol_list[0])\n",
    "axd['a'].set_title(\"linear problem\", fontsize=10)\n",
    "for spine in axd['a'].spines.values():\n",
    "    spine.set_linewidth(1.3)\n",
    "axd['a'].tick_params(axis='x', which='both', length=0, labelsize=y_tick_fontsize)\n",
    "axd['a'].tick_params(axis='y', which='both', length=0, labelsize=x_tick_fontsize)\n",
    "# ad a grid\n",
    "axd['a'].grid(alpha=0.6)\n",
    "# set x and y limits\n",
    "axd['a'].set_xlim([-0.6, 0.6])\n",
    "axd['a'].set_ylim([-0.6,0.6])\n",
    "axd['a'].set_xlabel(\"Feature 1\", fontsize=9)\n",
    "axd['a'].set_ylabel(\"Feature 2\", fontsize=9)\n",
    "\n",
    "axd['b'].scatter(train_inputs_test[2,:], train_inputs_test[1, :], c=train_labels_test[1], cmap=\"coolwarm\", s=100, marker=symbol_list[0])\n",
    "axd['b'].set_title(\"non-linear problem\", fontsize=10)\n",
    "for spine in axd['b'].spines.values():\n",
    "    spine.set_linewidth(1.3)\n",
    "axd['b'].tick_params(axis='x', which='both', length=0, labelsize=y_tick_fontsize)\n",
    "axd['b'].tick_params(axis='y', which='both', length=0, labelsize=x_tick_fontsize)\n",
    "axd['b'].set_xlabel(\"Feature 1\", fontsize=9)\n",
    "# ad a grid\n",
    "axd['b'].grid(alpha=0.6)\n",
    "# set x and y limits\n",
    "axd['b'].set_xlim([-0.6, 0.6])\n",
    "axd['b'].set_ylim([-0.6, 0.6])\n",
    "# hide tick labels\n",
    "# axd['b'].yaxis.set_tick_params(labelleft=False)\n",
    "axd['b'].xaxis.set_tick_params(labelbottom=True)\n",
    "\n",
    "# save_figure('/relu_nets_bias/', 'linear_and_nonlinear_problem_simple', fig)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "behav-analysis-fmri",
   "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.10.17"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
