{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "#AVOID THE EXECUTION OF FIRST THREE BLOCKS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import autograd.numpy as np \n",
    "from autograd import grad  \n",
    "\n",
    "def sigmoid(x):\n",
    "    return ( 1/(1 + np.exp(-x)) )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def objective_noreg(A, mean_xs, zs, activation):\n",
    "    term_1=mean_xs\n",
    "    term_2=np.sum(np.maximum(activation(np.dot(A,zs.T)).T,0),axis=0)/len(zs)\n",
    "    term_3 =  term_2 - term_1 \n",
    "    return term_3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "def objective_reg(A, mean_xs, zs, activation):\n",
    "    term_1=mean_xs\n",
    "    term_2=np.sum(np.maximum(activation(np.dot(A,zs.T)).T,0),axis=0)/len(zs)\n",
    "    term_3 =  term_2 - term_1 \n",
    "    return np.inner(term_3, term_3)/2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def experiment(n,d,k,A_star,T, activation = lambda x:x):\n",
    "    z_true = np.random.normal(size=(10*n,k))\n",
    "    xs = activation(np.dot(A_star,z_true.T)).T\n",
    "    mean_xs = np.sum(np.maximum(xs,0),axis=0) /len(xs) \n",
    "       \n",
    "    A0 = np.random.rand(d,k)  \n",
    "    V0 = np.random.rand(d)\n",
    "    eta_A = 0.001\n",
    "    eta_V = 0.0001\n",
    "    eta = eta_A\n",
    "        \n",
    "    A_noreg = 0*A0\n",
    "    V_noreg = 0.0*V0\n",
    "    A_reg = A_noreg.copy()\n",
    "    iterates_noreg = []#[(A_noreg.item(),V_noreg.item())]\n",
    "    iterates_reg = []\n",
    "    for t in range(T):\n",
    "        z_sampled = np.random.normal(size=(n*4,k)) \n",
    "        my_f = lambda A : objective_noreg(A,  mean_xs, z_sampled, activation)\n",
    "        V_noreg =  V_noreg + eta_V * my_f(A_noreg)\n",
    "        for i in range(len(V_noreg)):\n",
    "            my_fi = lambda A : my_f(A)[i]\n",
    "            A_noreg = A_noreg-eta_A*V_noreg[i]* grad(my_fi)(A_noreg)\n",
    "        if t % 10 == 0 : iterates_noreg.append((A_noreg.item(),V_noreg.item())) \n",
    "            \n",
    "        my_f = lambda A : (objective_reg(A, mean_xs, z_sampled,activation))\n",
    "        A_reg = A_reg-eta * grad(my_f)(A_reg)\n",
    "        if t % 10 == 0 :iterates_reg.append((A_reg.item(),np.sqrt(2*my_f(A_reg))))\n",
    "            \n",
    "    return A_noreg, V_noreg, iterates_noreg, A_reg, iterates_reg\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "d = 1\n",
    "n = 10\n",
    "k = 1\n",
    "\n",
    "np.random.seed(0)\n",
    "\n",
    "#d=k= easy example\n",
    "#A_star= np.eye(k)\n",
    "#A_star=random_matrix(d,k)\n",
    "A_star= np.array([[1]], dtype = np.float64).T\n",
    "\n",
    "\n",
    "for T in [50000]:\n",
    "    A_noreg, V_noreg, iterates_noreg, A_reg, iterates_reg = experiment(n, d, k, A_star, T)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAewAAAHSCAYAAAAuWvi9AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8vihELAAAACXBIWXMAAAsTAAALEwEAmpwYAABDm0lEQVR4nO3deZgcVb3G8ffMZCMJSyALISQkbIG4IBBWQUFkvxJA9AJRQFH29QqK4MJFo7grLnAjIIsIooAGCURFENkT9h0SEBICJLKEkBAykzn3jzMnVV1T1V3dXdPVNf39PE+e6qW66qQlvv07deocY60VAABobm15NwAAAFRGYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAA/fJuQDnDhw+348ePz7sZAAA0xAMPPPAfa+2IuPeaOrDHjx+vOXPm5N0MAAAawhjzYtJ7dIkDAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABdA6gf3ii9KqVXm3AgCAmrRGYHd2SnvtJb3//dJ3vuPCGwCAAmmNwG5rk779bemtt6RzzpH220/65z+lq66i6gYAFEK/vBvQEG1t0qc+Je28s3TJJdK550q77ebeW75c+uIX82wdAAAVtUaF7Y0ZI33jG9JDD0lXXCHtuKN0+unSb36Td8sAACirNSrsqK22cn8+9jHpM5+RPv95aY01pE9+Ulq8WNpgg7xbCABAidaqsKPGjJH+/ndp++1dt/j220sTJkhz5gT7zJ8vHXywtGRJfu0EALS81g5sSWpvl/74R2mzzaSHH5ZWrpS220469VQ3IO3cc6UbbnD7AACQEwJbksaOlX7+c/f4c59z2wsukO6+2/2RpKVL82kbAAAisAMf/rD00ktuFPm//uVe+8hHpKefdo+ffbb85599Vlq2LP69lSuzaycAoCUR2GFjx0rGSJMnu1vBwp55Jnh8/fWum9zr6JAmTpQ+/emex7zmGmngQOm553qnzQCAlkBgxxk0SNp4Y/d46FBpm22kBx5wFfi0aW40+cEHBxX1ggVuO3Nm6XFOOEE67DD3+MknG9N2AECfRGAn+eEP3fZ3v5O+/GU3SnzTTaWvfS3Yxwe0n+q0vb30GBdeGDweOLD32goA6PNa8z7sNKZMkVascEG7eLF7raMjeL9fP9cFvv767rEk9e+ffLx33+29tgIA+jwq7HJ8VTxihLT11u7x2mtLd90l3XijtMsu0quvBl3i4Qq7s7P0WIwyBwDUgcBO6+Mfd9vzznNzku+zj3THHaWLhyxbFlTS8+eXfj7vwH71VWn8+NLBcwCAwiCw09pvP7fdZpvgNWPcaPKXX3ZzkkvusSS98ELp599+u/ZzL1rkZmF76aXaj3Htte5a+y9+UfsxAAC5IbDT2m036amnXDd41AYbSIce6h4/9pjbvvJK6T5Ll7p7uu+5xz1/7z03s9p550mvvVb+3FdeKc2eLf3kJ7W3398LXu46OwCgaRHY1dhii+T3PvABdw37wQfd81dfdds//9ltly6VzjgjmEntq19118W/+U3pS18qf15j3Nba2tvuB8wNGFD7MQAAuWGUeFbWWEOaNEm69VbX/b1ihbuf+xOfkMaNc4H92GPu9rDbbnMzqnldXeWP7SdxqSewfYXN7WUAUEiZBLYxZh9JP5PULulia+35CfttJ+leSf9tre17q2lsu6102WVBt/eECa46XnNN10Xur0F/7GOlnxs2LP54K1dKm2/uBotJ9QX2e++5LRU2ABRS3V3ixph2Sb+UtK+kSZIOM8ZMStjve5Jm1XvOpnXwwaXPR49227XWku6/P/lzs2e7YH/00dLXX3nFDRT75z/d80qVeDm+wiawAaCQsriGvb2kudba5621KyVdI2lKzH4nS7pO0qIMztmc9t7bdY1766/vtmuuKb31VvLnZs9228svL339zTdLn2dxDZtBZwBQSFkE9hhJ4ZuOF3S/tpoxZoykgyRdVOlgxphjjDFzjDFzFvsZxopiwAB3rXqTTdzz7bd32w02iN9/ww2lnXYKni+K/JaJ/v2zuIYdndAFAFAIWQS2iXktmiw/lfQVa+2qmH1LP2jtdGvtZGvt5BEjRmTQvAZrbw9mPDvoILedPDl+32HDpDGh3zazZ5eG9n/+U7r/ihW1t8tfww5PrwoAKIwsAnuBpLGh5xtKWhjZZ7Kka4wx/5Z0iKRfGWMOzODczekPf5B+8AM3YExKDmxr3Wpg3jPPSKNGBZOsRCvsd94pfX7vvdLFF7vHhx9e2h0f5StsAhsACimLUeKzJW1mjJkg6WVJh0o6PLyDtXaCf2yMuUzSX6y1f8rg3M3pgx90f7yttnLb008vnfykq8tNxiJJe+zhbgmTpOnTpTfeCBYV8fxynp7vTv/CF6Srry7fJj9lKoENAIVUd4Vtre2UdJLc6O+nJF1rrX3CGHOcMea4eo/fJwwa5K5t//CHblazK690r1srHdf9FX32s8H+Z54pffe7weQrXrTC9sKjx5O6zX3YVxvYK1ZIP/tZ6ZzpAICGy2SmM2vtTGvt5tbaTay107pfu8ha22OQmbX2qD55D3YlQ4e6CVBGjpTWXde91tUlHXWUC+5JPe6Ek+bOLX0erbC9JUuCx9GBa54Pe981nuT6690a4N4PfiCddpq7vxwAkBumJs3Dmmu6bbgyHjOm537RwF661E19OmiQdN11wetvvBE8js5Lbq3746+Ld3S4AWjTp8ff1/3JT0pTpwbPfdAn/RAAADQEgZ2HtdZy23Bgjh7t5hdfa61g5rP586Xhw4N9nn3Wzab23ntu/nE/Gt3PTy71DOxPf9pV9j7UOzqkadOkY4+VrrmmclsHD3Zbfw0cAJALAjsPPgTDgW2M9J3vuO7t8JrVm28uPfCA9NGPlh7jzTeDSv1f/wpe95XwW2+54//xj8H+kgtsH+pplvz0I88JbADIFYGdhyFD3HbTTePfHz48mEJ00CC3Bve225bu8/bbwSpeYYsWuTW5hw1zg8W85cvdttpBZ76KJ7ABIFes1pWHDTaQbrhB+shH4t83xt2PPX9+UOHG3WMdnbpUchW6X+Jz5sye7195pbTZZunb6oPaB36cn//c/YA455z0xwUAVIUKOy8HHhiMFo+z9tpuWy6w47z1lvTcc+7xqFHx+/j34yr0KB/Y5brPTzlF+trX0rUPAFATKuxm5Qem+aD2170reeQR6aLuu+kqzcVeLrBXrXLd4b6yfv31dOcHAPQKKuxm5SvsQYPcNlphR9fU9h54IHgcXa6zGr6y9lsCGwByRWA3q0pd4gMHum1b5H9CPzHKWmv1nCktqlyF7StrH9jPPVd50hUAQK8hsJtVtEs8Gth+FPmBB7p7sqPiJmKJKjfdaHSw2YoV0kMPVT5mkpdeku66q/bPA0CLI7CblV/FK65LfMCAILD79w9uvYr7/MiRyefwS27GCXeJ+/u9/WC1WkyYIO2yS/L7J5yQbhAcALQoArtZ+S5vv2JXOLCXLq0c2L6r/JRTks8R7uKOLhry7rvSrFnuOrifea2eLvG4aVDDLryw9mMDQAsgsJuVD2w/0Um0wg4Hul9vO7zu9v/9n1ty0y/BGccH8IsvSuuvL33rW8F7jz8u7bOPuxd8nXVK9+9NXCcHgFgEdrPq399tfYBFb+sKV9gHH+zW1T7yyOD9TTeVDj006FKP44/9/e+7CVe+8Y3gvSOOCB77AXBxYbp0aenzF1+Uzj67/EQrkjRnjruuHRVeeQwAsBqB3ax8IPuQTBp05oN9iy2CSlgKgrpcYPtr2PPnl29LNLBnzAimPfWD47wbbnBreZfripek7baTNtqo5+sENgDEYuKUZpU2sPuF/if0gR2+rp2mwq60CIifkc3vP2WK2556as99/Q+IuOpZckt9hgeXPfig6/bv10/q7CSwASABFXaz2nVXt91nH7etVGFLQWCHQzr82F/39nwAL1kibbll8PqXv1y6bzWDzvytYp2d5d/3tt1W2nHHoMu/lsB+7bXKg9oAoOAI7Ga19dZu5Pb++7vnSROnhEeI+8AO7xsO7Geflf785+B5uMKeNCl43d8SFn7ev382gZ20WlitgT13rhsw99OfVve53vLss64H4eab824JgD6GwG5m4So3qcK2NngtOp1p9PG4cdJeewXPX35Zuv12F9gjRwbXo/v1K+22HjrUnS8a2F/5Ss82+0o3KZg7O0vb7NUa2A8/7LZ33lnd53rL3Xe77e9/n287APQ5BHZR9IsMN/CVdbgruFKFLZX+CJg5U9p9dxeSa68tjRgRf66hQ93nfvSjYGERqWdV29UVVNhJgd3REf9eOLCvu0760IfSdXO/9prblpsgppHifowAQAYYdFYk06cH91X7CjgcEEOHuglTyl23jptNrKPDVdf+c9EKe8iQoKI//vjg9WjFvWpV5S7xzs74Gdb8j4wlS9xCI4884o5fbtCcFAR20lKijeb/92DWNgAZI7CL5ItfDB77mczCVagxrsoOV9hxs6DFCQd2eCCbFFzDrqSzs3KXeEdHzwVLpOBzS5YEFX5S6If5wB4+vPK+jURgA8gYXeJFFRfYkgvsuKr0fe8LHv/v//Z8f+21SyvssKFD03VPd3amq7DjBq/5gF+yJJgmNU1g+zW///73+L9Xo9ElDqCXENhFFdclLrmu4fAEKpKbBS08KOsb35A+8pHSfcp1iQ8eXH5lLy8c2D6An3jCXfv2Ojriu8T9/m+9FbyfJrD9vjNmSOeeW3n/3taMXeK/+IX09NN5twJAnQjsovIVdjSwL7tM+slPSl+LzoImBdekvWhgv//9wXvt7ekCe911g0r8pZekPfd0xznjjGCfpMD2VXVchb1woZvuNK4NzVbRNltgWyudfLK0/fZ5twRAnQjsovKBEO2q3nxzafz4yp+Pdpv7keCSC+ybbgrea2srDcvoLWZhfr9Vq1w3ddSTT7oqOsrPSb5kSc8K+6ST3HSnt93W83PNFthevYFtjJvApl7+v4/onO8ACofALqqka9hpRSvuoUNLQ3z48OAWrk02Ke2e9utjx6k0ucohh0gnntjz9Xfecdu4Ctv/XffcU1q0qPRzzRbYWbbnBz+o/xjMAAf0GQR2USVdw07LTzfqDRkSBLavcI85xoXmqFGlFXa5wK40L7kkPfdcz9fCXeLRCjv84+Lee0s/12yB1Gxd4nl/P4sXu9v0ANSNwC6q3qywfXgaE9wWljaw47q7ozbfPPm9t9/uWWH7GdyknrdvRX+w1PID5r33sl+Hu1kCO83Yg940cmTz3XIHFBSBXVR+opANN6zt8+UqbB+YYeEfBtElNcPSBPayZcnvrVwZVNiTJknvvlt6vmggR3+w1PIDZtiwYJa3tKyVvvQl6Zlner4ulQ/sCROkz3ymuvPVKu8KG0BmCOyimjJF+sMfpK9+tbbPx40a94PO4gI7XKmVC+w0c4H769VxVq0Kzm+t9Nhjpe9HJ2SJBvgdd1Q+f9S776bryg+bN0/68Y+DpUaj7SkX2P/+t3TVVdWdr1YENtBnENhFZYwbwJVmBrI40QpbcoPLpPhpPrPsEi8X2JILUG/58tLu6ui92dFA+tjHKp9fcj8K9txTevzxdPtH+e8jqQv+jTdKV0bLi/9+mqWLHkDNmJq0VQ0Z0vO1z3/eXW/8xCfKf7a3AtuvCBbuMn/33dLAjlbYaSrIF16QNtqodErUe+5xt52dfHLlz8fxgR2dZtUH+LXXuj8LF0qjR1d37CxHmud9DRtAZqiwW1V0URDJVWFTpsTP9R1WLrDTdIknzWDmewuWLw9eW768NKQ7O6X77w9CrdJsaPPmSRtv3HPa0npH2fsgjM7VHj1euLcgrSy7samwgT6DwG5V661X+2eHDUv+fNwsZmn5yjpcYb/zTmmF/ac/STvsIF1yiXteqYJ8+WW3/cc/Sl9PmngmLf+5SoFdyw+C3ghsAIVHYLeq973Pzb9di8GD3dSjEyfGv9/eLh1wgFvTuhbhqnTJktIK+5573PbFF922UoWddPtbvRWnP15Sl3h4v2uvdeebN6/0vaQfN1l2iRPYQJ9BYLeySteqw44+Onjcv78L7V//On7fcePcgKs012779UsOfskFdrjC9stp+go/rsIOv1bpfvU04bh4sQvc8A+cpC7xuNvMrr7aPX7kkdL3brrJHfeBB8ofox5cwwb6DAIb6UyfLu27r3vsrzXvuqt0xBE99/UhFjewLWrgQOmKK5Lff+stF9j+nG+8UXrsuAp75Urpmmukr389eZGUaq5h+6D92c+C13yFHw3saEBam3yr1x//6LbRW9G4hg0gBqPEIW27beV92tqCFb7Ct5LFBYEPyTSB3d4ubbedW2Xs5Zelc84pff/tt12X+ODBpQPa/L3acRVkR4d02GHu8X/9l9vW0yUeF7iHHOK20S7xaHu6upID2w+ui96aR5c4gBhU2K3uzTdL18pOo1JgRyvscl3vbW3uGEceGT+QbdkyVzEPHlz6enT60rDwNW/fhZ4UgmnCsdxkKP7vOm+eW10retvZqlXB588+u3TCmqTA7o0ucSpsoPAI7Fa3zjo9l9pM4oOn2sAePdp9duHC8hObRLuXpWDilKTATqqwPb/QSFKFnVVgH3ywW13riSdK3+/sDM791FOlvQQ+sPtFOroYJQ4gBoGN6vmu8SS+m9iHrA/40aN7/jgIh2BcYC9bFnSJh5UL7JUrpQ02cI+ff95t0wT2nDk9j5XUVs//Xf2PhGiF3dGR/KMgWmF3dbl1q3ujS5wKGyg8AhvVCwerD4Kdd3bLcYbf9xV2uCIvN5VqUmCvXNnzevi770qf+5y7vSyqoyM4j58jPCmww13q220Xf6tbuQBtb3fBG10S1OvsrBzYvsI+7zw3T3ua2eLSosIG+gwCG/XxwXfUUUFV6wMqLrB9dR7tBk56rdw17Msui2/T1lsH7fLToPpVv/zIbP9+dFnNhx/uebxol3g4gNvbpU03DSr5WgLbu+ACt33yyfj9a8FtXUCfQWCjeuEACoeYn+7Uh4QP7HAXuq+i112353GTrmGX6xKPs3Rpz8CeN8+9Hh2FHg3suFW7ooEdrlrnz5deeaX03GFpAtuHvN+v1gVJ4tAlDvQZBDbqExfYPoDiKmzfdRw3Irxcl/jAgaXv/+Y35dvl2+AD1AfXs89KY8cGz6PXnKOBK0m//a3bxgX2okWl+95+e892JHVL+8D2bVhjDbdduDB+/1rk0SU+bx5d8UAvILBRn3Bg+0q6XGD7aUd9hZ1m0NmKFe4YcV3mSXwFft99Pd9bsCAIlGiFHQ3sRYuCmcpqmX+8XIXtvwsf2GkXNKlGo4PzuefcJYLzzmvseYEWQGAjG+W6xOMC21fY4UCJC+y333bXhydNqm7t70qrZPmgjgb2G29If/1r8Dy8FGjWgR3eJ3zcLAO70fdh+8VWbrutMecDWgiBjeqF/88//DhNl7gP0o02cttwRRsX2JILvL33rq7CTlpz2/PXqqNd4rNmuXP5wAlfK69lSc5yXeJetMLOcqBYb13DnjdPWn/9nqP0k6aCBVA3Ahvp+ZANz9ZVbZe4v2678cZuGw6npMCW3HXnairsSqZNc9tohe35a9MXXtjzvXAAVwrCNIHtvy//XRShS3z6dDeLnL9c4FVabAVAzZhLHOl9//vSLrtIH/1o8Fq5QWfDhrn311472N9X2D6ww8oF9qBB1VXYlfhJUsotcfnee9IvfhG8FtclXqmSLDdxSngfKfje/OxsWWh0cNa7zjiARFTYSG/QIOnTny59rVyFPWqUdNdd0qc+FezvA3vChJ7HrxTYWVbYXrRL3LO2Z9f0jTdKDz1UfYWd9hq23951V/DeGWdIl15a/vPl9PZ92PWsggagKgQ26uNHew8Z0rPClqSddgpel4JQ95OshKWtsE89VTr00NrbnFZc2F1xRfXXsNNW2HHn+9GPStcir1bcNeyuLncNuh5Jwcw1bKDXZBLYxph9jDHPGGPmGmPOinl/ijHmUWPMw8aYOcaYXbI4L5rAOedIP/mJWxc7WmHHueUW6TvfST9ximdMENjDhknjx9fc5FTiKmyvt0aJZ3nt2otr63e/6269ynJGNY8ucaDX1B3Yxph2Sb+UtK+kSZIOM8ZMiux2q6StrLUfkvR5SRfXe140iUGDpNNOc2EbV2FHTZwoffWr5Ve+SupmDq8WttZaldu2996V9ykn6e9RTRhdcUXlucH9de7e6L6OO+Y//+m2/n70Rx6p/zzLlkl77inNneueE9hA5rKosLeXNNda+7y1dqWkayRNCe9grX3H2tVlxhBJ9Jf1RWkq7LCnn3ah4fnAHjTI3VYVtxCH5AJ7zTV7vj51avB44EBXzdfq6afjJ/+wtrowuvvunktuRnV09N615qQuccl1X3/ve9KHPiTdf391x43+qPrrX6W//106q7uDjS5xIHNZDLsdI2l+6PkCSTtEdzLGHCTpu5JGSto/6WDGmGMkHSNJ48aNy6B5aBhfYacNtIkTS5/7Lu+BA6Xddus5WtqHwIABwTSeYRMmSLNnu1W3oqt7hQ0YkHw7l/etbyW/Fw6jLLqxOzt7pztcKh/YxgSj5efPl7bfvrZzPPSQdP75peehwgYyl0WFHdd/2ePntbX2BmvtFpIOlJT4/4bW2unW2snW2skjRozIoHlomErrZFcSrrClnqEb7hIPD2QLf95/NrpYSNgee9TXznAYZRG0HR29H9hxr7Wl+Od///0uhGfPTt5nm22CCt3/b9TVVXm2OQBVySKwF0gaG3q+oaTE1QustXdI2sQYMzyDc6OZxIVoNXxg++NEA9tXb+UC2x+jXGAPHVpfO8MhWGlGtTQ6O3uvSzzuuD5U0wT2DTe47d/+Fv9+tOvbP7c2fnT7lVfGL7ACoKIsAnu2pM2MMROMMQMkHSqp5OKjMWZTY9z/2xpjtpE0QNLrGZwbzSTrCjsauj5oBwyIP1d7e/D61lsnnyfu+nc1su7ubUSFndQlXsmyZW5b7gdQ3Pm6uqR77y1977773N0EJ5yQ7lgAStR9Ddta22mMOUnSLEntki611j5hjDmu+/2LJH1S0hHGmA5J70r679AgNPQVWVfY0YlS/JSo/fu7eawl6QMfcNvHHnOf32QT6S9/cdfAk9RbYWf9n24jrmHHvdbWVvnv4qeSLTcmIMxX9NaW3qZnbTB/+6uvpjsWgBKZzPVorZ0paWbktYtCj78n6XtZnAtNLKvA9hV2lJ/itL1d2mEH6U9/kvbaS9p119LP7584ptHJsku8kjQD3LKosJcudVOpDo9caYpra1yXeLja7uhw4breekFgRyvspIlTygV2NdfOAfTAvxxkJ6su8aTg94Htu2mnTHGjxX1olJt4Jay3A/v554PHab6TLCrsTTeV4gZpxi2vWSk4DzssCP6kwE4S7hIP/+/R1dV7K4cBLYLARnbqrZySKuwttnBb3yW+ZEnp+80W2OF50tMEdhb3YfvVxaLKXcNOct11wWMf2Gl7T/zfo6ur9L+Hrq7qBruFzw1AEoGNZhJXYS9Z4u7zlYLr1tFqtNkCOyxN0FVTYS9eLC1MvAmjp3Jd4mn+Hj40kxb5SDqftT1/JFRTYd96q7tufscdlfcFWgSBjexVuoacJK7CXmut4Pnpp0tnny2deGLp51ascNty3bbHHhs8rneUeLmBWk89Vfo8bYWdNrBHjpTGjEm379tvS5/5TM/XfXBWquqffTZYOSztj5Rwl3i0wq4msO++221nzUp3XqAFENjI1rJlbjBYLSoNOhs0SJo2rWcw+yqwXOX8858Hj9OOeE5SLrx8972XdWBXIzwlargd/gfH3/4W3Gcd5+yzg8dpAzvcJR4O5lWrqusS9+MVopc/gBZGYCNbgwcHU4xWq9KgsyRpAjscEo0eJV5Jlrd1PfRQ/HXn8GMfnOHpV+Oq3nAFntSrUG6UeFJgp6mwk8YrAC2MwEbzqFRhJ/FTYKYN7Lh5yNMqt/hHXDg3osIOt2ebbYJu8PC5w4Fdbfd23GcqXcOOdok//nhtFXallc6AFkJgo3mEF/+oRprJPYwJutKjE7JUK67anDNHeumlnq+nrbDrGSUenerTL58Z/h793/nCC6VHH0133PBqaWkni0kadPbEE9Xdh+2/Nz/ZCoBsJk4BMlFrhe3DpFJX92OPuT/1BPYFF8Qvl7nllvGD3hpRYb8emeW33LGiA/bSSqrKy3WV1zNK3B+30qQzQAuhwkbz6NfPhbW/flmtSoG98cZuspV6J3i59daeryVVjY24hp0U2OGQrfbWt6i0XeLh/cP7WBu0YcWKnkunRvl9e2tRFKCAqLDRPNrb3W1Em21W2+fTDiart0s8Tq1BKNVeYS9c6K7HJwV2uPr1gduvX23nCh/rjTekb3+7/P7Ra9jha/833eT+rFyZ/L8FgQ30QIWN5rLNNrXfJ532dq3eCOykCjtpBrKwzs7aun7HjJF22qm6wE76YZGmYvbOOqty26Jd4nGD9To6yn9eIrCBEAIbfUfaKrfeLvE4SYF98MGVP9vREUz+Uk44gF97zW2fecZVvGHhkdrR15Juuas0qCz8fprbvaJd4uGpSaNtKofABlYjsNF6eqPCTqpQzz03WKwkSWdnusAO7/P002671lrJ1XlSl3ilfeNUuwZ4XJd4NHyTjnnZZW5SF6n3lh0FCojARvFtuWV1+1cb2OHFMGpRqfJ/803pnXcqHycc2D6kBwxIrkKrqbArBXK1a4DHdYlH25nU7s99LpiZjgobWI1BZyi+OXOCyVOifvYz6eabS1+LBnZbW/nA+sAHqm/Tj38crNpV6b7jVauk446rfMzw39FXnm1tyaEW141da2CH3097W9aGG5Y+j7YzTfVMYAOrEdgovsGDkxf+OOUU9ycsGloDBpTvkq5lBPjppwePw4G9ySbSvHnVH08qXWfbh50xlQN75MjKg86qCew0urqCNbX982hApwnj5593n6t1ulugD6FLHK0nWiFWmlmt3rAIB3a5FcUquecetzUmXYXtQ7Z///or7LRziYdftzbozbC2Z2C//HK6ucLrvSQB9BEENlBp1Hg991hLpT8Qal0prKsrGA2+3nrVdYn365dtl3ja9nZ1Bd9dXJf45MnBZYNyqr1+DvRRBDZQaRBavYEdVmtgr1oV3Lfc1RU8Lnf9PcvADodmdDBZ0vGigR13zfrNN8ufV8r2+wcKjMAG/IpQScHQDIHd2Vka2GmuYWfZJV5the27xP13F3cNOy2uXwOSCGwgmNJ0//3j388ysI86ym2rDaFohV3NoLN+/dy94KecUvugs2q7paMV9pIl6WZIi0NgA5IIbED69KfdkpTR0eReloG9224u/Kq9dzwpsNMOOpPcvc3vvVd+3yS1XMO2NhhwF7fCWVoENiCJwAZcIHzkI8nd1VkGtg/PNGtChyV1iacddBY9f1RvdIl3dQXnrvbvG5bmvm+gBRDYgJd0y1WWFZ4/hw+w//mf4L2TT07+3Gc/Ky1e7B5HB51VE9gTJ8bvm/WgMx/Y/sdOPYHN5CmAJAIbCAIhKbCzrLB9cPnQC4fp4Ycnf+7mm6WZM93jFSuk004LjpO2S9y/Fp7QJLpvUtd1vV3i9QQ284kDkpjpDAjCqBGB7cUFWS2LkhgTH6Zf+EIQdOEfBatWxfcYdHS4gL3mmvjz1HIvdLjCrqdbm8AGJFFhA8mBPWqU27a1SXvtle05swrspC7xSy6RLr+853G7uuLPc9ppbpR8mlHkacOXwAYyRWADPozWWKP09Xvvla6+2j2eNctVmd/6VjbnjHaNS7VX2JWu8YYr6vBAsKibb04O7DPOkP7yl56vl6u8w13iBDZQNwIb8IEdnqL0zDOl8eOlQw8t3TertbR9gNVbYcdN+RkV7RIvd55y3f/XXhsMdvPKXdv2a2IbU9/0ogw6AyQR2EAQJuEq8Pvfj983rjq9997qzxkX2JXmNI+zalXlQEvTJe73Kzci/sorXRtffz14bdq05P27utzfs9xI9jSosAFJBDZQ3QjouEDbYYfqz5lVhd3VVX2FXS6U0wywmz8/Xdt8l3iabvtyCGxAEoENVBcm9dyTvdtuweO4qr6WwF61qvIPjug17KTzrFolPfNM5XPef3+6toW7xO+4I91n4hDYgCQCG6iuwq71GvaKFdLf/tbz9WiF/dvfVnfcWrrEyy0A8utfV3f+cnyXuDGVV+VatSp52lQCG5BEYAP1d4mnMXBg6WeTusSnTq3uuGm6xMPd3JW6xLPku8TTTJry8Y9LgwbFv0dgA5IIbLSqk04KHocD++mnpeefT/5cVmHnu8TrvYadpsIOd7vfcYf06KPVn6cW4Qq7kttvd9u40eSMEgckMdMZWtXPfy7tvLObDjQc2ElzbXtZV6fhMPPHfvhh9/j976/8+TQVdtTbb1e3f61mzHDbpBnk4rzxRs/XoreSAS2KwEbr8tVtI65hR4W7xEePll55JXhtq63SH+e996Tly6s/f1tb9fOD16qaSVNefbXna8uWZdcWoMAIbLSu3XZzYeIX0kgjOhtavdra3Kjrxx6r7fNvvRV0JyeJC8zhw6VFi+qf1CSNNNewfTvieguWLMm+TUABcQ0brWvUKFdl7rxz+s+Eu3fHjKn93OFr2BtuKO27b+3HqiQusHfZxW233LL3zlvu/GG33BI8jlb9AwYQ2EA3AhuoRjiw77yzdNavWlQ7x/ZOO9V3Pi+8sElvq/R3vO664HG02l97bQIb6EZgA9UYMqT08brr1nacuNu60jjssPLvDxvmKvbRo3ueK8wPcKtnUY60Kp0j3A0erbAJbGA1AhuoRrjCrqc6jbutK41KI8Lb2tzo8pUry+/nA7sRFXalc4Tvs44G9jrrENhANwIbqEY4sLMYMV4uzLbbrueCIGnmDW9rKw3sIlXY0UlSLrlE+sc/sm8TUEAENlCNcGCvuWb9xysXZvfdJ737bulraWY1a29vrgo7/HccOLDn++G/U/TvN3Jk7ZcdgD6GwAaqEQ7seqrTNNew/dKUYWm6xNvbS+flLldhN7pLPG4J0XBVvWJF6XuN6AEACoLABqqR1cQpvXUN21fYYeUCu7edfXblFcnCgX3UUaXvEdjAagQ2kKdGBHYcH9i9PdvZrruWhm6lCvvll0vfI7CB1QhsIE/VBlKawI7+CIibGMYHdm8vrNHeXjmwWdwDSIXABqp1553SCy/UdwwfUtV2TVdaajKuwt5hB+mII4Ln11/fuAo7+gOiUpd4FBU2sBqBDVTrwx+Wxo+v7xh+FHfSGtBJaukSb2sLRrRvsYV00EHBPs1QYRPYQCoENpCHRgZ2eLS53zaywq4U2C++mLwACYENrEZgA3nIOrDDgRy9hh0ObB+AeV3DjusSL3d5gcAGVssksI0x+xhjnjHGzDXGnBXz/lRjzKPdf+42xlSx4C/QB2Ud2L6q7tcvvks8zwq70n3Y5RDYwGp1B7Yxpl3SLyXtK2mSpMOMMZMiu70g6aPW2g9K+pak6fWeFyg0H9hxM3+VMyn0TyschP5xUpe4fy1aYScF9iOPVNeuJGkqbACpZFFhby9prrX2eWvtSknXSJoS3sFae7e19s3up/dK2jCD8wLFVUuFfffd0rHHButHn3JK8F7WFXY1le2xxya/F20LFTZQsyymOxojaX7o+QJJO5TZ/2hJNye9aYw5RtIxkjRu3LgMmgc0oVoCe/hwF2B77y0tXOjm2R450s0mFq6ws7iGXc2ELmuvnfxee3vpgDICG6hZFhV23L+o2CGfxpjd5QL7K0kHs9ZOt9ZOttZOHjFiRAbNA5pQLV3i4fAaPbo0nMOVcxajxKsJynIh3N5eeo5qu8QJbGC1LCrsBZLGhp5vKGlhdCdjzAclXSxpX2vt6xmcFyiuWirsuKo37tp0NV3iWVTYlQKbChvIRBYV9mxJmxljJhhjBkg6VNKM8A7GmHGSrpf0WWvtsxmcEyg2H2LVBHZceEWDuNJ92GkHnVUTlKNHJ79HhQ1kpu4K21rbaYw5SdIsSe2SLrXWPmGMOa77/YskfUPSepJ+Zdw/wE5r7eR6zw0UXr0Vdjiopcpd4l6lwE5bYc+YIY0alfx+W1tphV3tVKwENrBaJmvsWWtnSpoZee2i0OMvSPpCFucC+pRquojLVdjhyjkatn6NbCkIz0pd4uFz3XuvtOOO8ft94hPS7Nnl211PYANYjZnOgDxVU0GWu4YdXkwkTYXt96lUYa+xhls8pJxKf4fwOaiwgZrxcxfIw803u+7kapQLr3Bg+4U+vPCgs2or7DSBWW4fawlsICMENpCHffZxf6oRV2H7APah2N4urbtu6T5xFfbgwW67bFnPY663Xs+u9nIq7cM1bCATBDZQFOXCyy9R2a+fC9zo56IV9tix6uGhh6S5c6XJk4NzpRl8Vm6ftdcuDexod30lBDawGoENFEW5YPRd23EVdlyX+PDhPY8xYYL0oQ+5xwsWuG00MEeMkBYvLn0tKVTXX9+dhy5xIBMMOgOKIi68fACHr2GvtVbPz0UD2x9riy2C/cI/CJKuYf/73+naJQVVPIENZIIKGyiKcteww4EdncgkfFtX2JtvutvKhgzpefyka9j+2rckHXVUcrvC6glsAKtRYQNFUa7C9vr1c93Wb71V+rm4UF1nndIADh8/zSjxSy9126R7yX3bGHQGZIKfu0BRpBkA5gMxvIJWXJd4peNHK+zvflfaYIPS/f17SQuYENhApghsoCjSVNhxXd/hQWfllLuGfdZZyZ+rVGGHu8QZJQ7UjC5xoCiqqbDD0lbYceGYJjArLRFKYAOZILCBokh7DTvuc9V2iUdHk5dTzTXsapbtBFCCLnGgKMqF3RFHuFnLTj2153vG9Fz8o9LxqwnsShU2gQ1kgsAGiqJchb3uutLllyd/Lk1Qho9fLtijkgaSxVXY1XaJA1iNwAaKotx92HFh3t4e3J+dpks8zFfNW2/d873f/U567rngedy5995bmjatZ7upsIGaEdhAUZSrsOPee+gh6a9/dY+rDcphw6Q77gimKg077LDKn7/lluBxuG0ENlAzAhsoimrD7gMfcH/Cn62mq3vXXas7XxIqbCAT/OsBiqKeW5zyDEoCG8gE/3qAoqj2GnbcZ6upsLNSrks8urIYgEQENlAU1V7DDktzW1dvCYd0dGESJkYBUiOwgSKrtsLuLRdfHIwKj/Jtmz27dI7z8HsAKiKwgSLr399tk2Yb83q7S/zoo6X99y9/7vb2nj8cCGwgNUaJA0V2wgnSwoXlF+eQGjPYK+kc/vW4dbkJbCA1AhsoskGDpB/+sPJ+jRh0lhS+VNhAJugSB1pBIyrspPD1rzMtKVAXAhtoBY2osCt1iYenSvWosIHUCGygFZS7rWvSpGzOkaZLvLOz9L2hQ7M5N9ACuIYNtIJyFfZ990lLltR/jkqB3dYWVNg77STttZe07bbSAQfUf26gBVBhA62g3DXsoUOlMWN67xw+yNvaggp7wADp3HOlkSOD/Xbeuf42AH0YgQ20gma4hm1t0AW+ySZuu+aawX5czwbKokscaAWNGCU+ZEj86z6Iu7qk7baT/vAHab/93GvhwGZhEKAsAhtoBY0Iw3XWKX/uri63PeSQ4D0CG0iNfyFAK2hEl/igQfGvf+Yzbrveej3fY5Q4kBqBDTQ7331cjzxX6/rmN6WlS6Vhw3q+1y/Uyee7zrfaqjHtAgqGLnGg2c2Y0fP+5WrluR52W1u6Sjrcxuuvl267rXfbBRQMFTbQ7NrbpYED6ztGo64P7723dPzx1X3mBz9wW389e8IE6aCDpAsuyLZtQMFRYQOtoFEV9i23VP8Z36bNNpNuuEHaffds2wT0EQQ20AqaeQS2Hz1ujHTggbk2BWhmTfyvGEBm8ryGXYlvUzP/qACaAP9CgFbQzIEdrrABJCKwgVaQ521dlfg2EdhAWQQ20AqaubuZwAZSaeJ/xQAyU4TAbuY2Ak2AfyFAK+AaNlB4BDbQCpo5DOkSB1IhsIFW0MwVNoENpEJgA63AhyGBDRQWgQ20kmYObAadAWXxLwRoBc1cvVJhA6kQ2EAraeYKm8AGyiKwgVbQzGHIbV1AKgQ20EqaucLmGjZQFv9CgFYwcqR05JHSjTfm3ZKeqLCBVDIJbGPMPsaYZ4wxc40xZ8W8v4Ux5h5jzHvGmDOyOCeAKrS1SZddJm2/fd4t6Ylr2EAq/eo9gDGmXdIvJe0paYGk2caYGdbaJ0O7vSHpFEkH1ns+AH0MgQ2kkkWFvb2kudba5621KyVdI2lKeAdr7SJr7WxJHRmcD0BfwjVsIJUs/oWMkTQ/9HxB92s1McYcY4yZY4yZs3jx4robB6DJcQ0bSCWLwI77V1bzUFRr7XRr7WRr7eQRI0bU0SwAhXDKKdLEidLhh+fdEqCp1X0NW66iHht6vqGkhRkcF0ArmDBBevrpvFsBNL0sKuzZkjYzxkwwxgyQdKikGRkcFwAAdKu7wrbWdhpjTpI0S1K7pEuttU8YY47rfv8iY8z6kuZIWktSlzHmNEmTrLVv13t+AABaQRZd4rLWzpQ0M/LaRaHHr8p1lQMAgBpwHwUAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFkElgG2P2McY8Y4yZa4w5K+Z9Y4y5oPv9R40x22RxXgAA8nDVVVdp/Pjxamtr0/jx43XVVVf1+jn71XsAY0y7pF9K2lPSAkmzjTEzrLVPhnbbV9Jm3X92kHRh9xYAgEK56qqrdMwxx2j58uWSpBdffFHHHHOMJGnq1Km9dt4sKuztJc211j5vrV0p6RpJUyL7TJF0hXXulbSOMWZ0BudObfp0adasRp4RANAXnXPOOavD2lu+fLnOOeecXj1vFoE9RtL80PMF3a9Vu48kyRhzjDFmjjFmzuLFizNonvPtb0u//31mhwMAtKiXXnqpqtezkkVgm5jXbA37uBetnW6tnWytnTxixIi6G+cNGCCtXJnZ4QAALWrcuHFVvZ6VLAJ7gaSxoecbSlpYwz69isAGAGRh2rRpGjx4cMlrgwcP1rRp03r1vFkE9mxJmxljJhhjBkg6VNKMyD4zJB3RPVp8R0lLrLWvZHDu1AhsAEAWpk6dqunTp2ujjTaSMUYbbbSRpk+f3qsDzqQMRolbazuNMSdJmiWpXdKl1tonjDHHdb9/kaSZkvaTNFfSckmfq/e81SKwAQBZmTp1aq8HdFTdgS1J1tqZcqEcfu2i0GMr6cQszlWrwYOlZcvybAEAALVrmZnOhg2T3nwz71YAAFCblgnsddeV3ngj71YAAFAbAhsAgAJomcAeNkx69133BwCAommZwB41ym1fey3fdgAAUIuWCWw/AU0vzxwHAECvaJnA3mgjtyWwAQBF1DKBPbZ7YlQCGwBQRC0T2Gus4a5jP/983i0BAKB6LRPYkjRxovT003m3AgCA6rVUYG+xBYENACimlgrsLbeUXn9dWrw475YAAFCdlgrsLbZwW6psAEDRtFRgv+99bvvII/m2AwCAarVUYG+4oTRypDRnTt4tAQCgOi0V2MZI221HYAMAiqelAluSJk+WnnpKeuedvFsCAEB6LRnYXV3Sgw/m3RIAANJrucDecUe3vfPOfNsBAEA1Wi6whw+XPvAB6fbb824JAADptVxgS9Juu0l33SWtXJl3SwAASKdlA3v5cmn27LxbAgBAOi0b2G1t0qxZebcEAIB0WjKw111X+vCHpRtvzLslAACk05KBLUkHHCA9/LD04ot5twQAgMpaOrAlqmwAQDG0bGBvvrk0caI0Y0beLQEAoLKWDWxJmjLF3Y+9ZEneLQEAoLyWDuwDDpA6OqSbbsq7JQAAlNfSgb3TTtLYsdJvf5t3SwAAKK+lA7utTfrMZ6S//lV69dW8WwMAQLKWDmxJ+uxnpVWrpKuvzrslAAAka/nA3nJLabvtpCuuyLslAAAka/nAlqQjjnCTqDz2WN4tAQAgHoEt6dBDpf79pd/8Ju+WAAAQj8CWWyN7yhTpyitZchMA0JwI7G6f/7z0n/9If/5z3i0BAKAnArvbXntJ48dLv/hF3i0BAKAnArtbe7t04onSHXe4AWgAADQTAjvk6KOlIUOkH/8475YAAFCKwA4ZNkw65hjpd79jnWwAQHMhsCNOP10yhiobANBcCOyIsWOlqVOlX//ajRoHAKAZENgxvvxlacUK6ac/zbslAAA4BHaMSZOkQw6RLrhAev31vFsDAACBneib35TeeUc6//y8WwIAAIGd6H3vc4uC/Pzn0oIFebcGANDqCOwyzj1X6uqSzjsv75YAAFodgV3G+PHS8cdLl14qPfNM3q0BALQyAruCc86R1lhDOvPMvFsCAGhlBHYFI0dKX/+6dOON0qxZebcGANCqCOwUTj1V2nRT6bTTWC8bAJAPAjuFgQPdJCpPPy395Cd5twYA0IrqCmxjzLrGmL8ZY57r3g5L2O9SY8wiY8zj9ZwvT/vvLx14oPS//yu98ELerQEAtJp6K+yzJN1qrd1M0q3dz+NcJmmfOs+VuwsucOtmn3CCZG3erQEAtJJ6A3uKpMu7H18u6cC4nay1d0h6o85z5W7sWGnaNOmWW6Tf/z7v1gAAWkm9gT3KWvuKJHVvR9bbIGPMMcaYOcaYOYsXL673cJk78URpu+3cQLQ3Cv8TBABQFBUD2xjzd2PM4zF/pvRGg6y10621k621k0eMGNEbp6hLe7s0fboL69NOy7s1AIBW0a/SDtbajye9Z4x5zRgz2lr7ijFmtKRFmbauSX3oQ9LZZ7spSw85RDrggLxbBADo6+rtEp8h6cjux0dK+nOdxyuMc86RttpK+uIXpUUt8TMFAJCnegP7fEl7GmOek7Rn93MZYzYwxsz0OxljrpZ0j6SJxpgFxpij6zxv7gYMkH77W2nJEunooxk1DgDoXRW7xMux1r4uaY+Y1xdK2i/0/LB6ztOs3v9+6Xvfc9eyp0+Xjj027xYBAPoqZjqr08knS3vuKZ1+Oit6AQB6D4Fdp7Y26bLL3IpeU6cy1zgAoHcQ2BnYYAPp4oulBx5wo8cBAMgagZ2Rgw6Sjj9e+tGPpD+3zFh5AECjENgZ+vGPpW23lY48Upo3L+/WAAD6EgI7Q4MGSX/8o7uufcgh0rvv5t0iAEBfQWBnbPx46corpYcfdiPIAQDIAoHdC/bf3w0+u+QS6Te/ybs1AIC+gMDuJeedJ33sY27t7Eceybs1AICiI7B7SXu79LvfScOGSZ/8pPTWW3m3CABQZAR2Lxo1Srr2WunFF6VPfUrq6Mi7RQCAoiKwe9kuu0i//rX097+7+7RZJAQAUIu6Fv9AOkcd5e7L/va33Sjyr30t7xYBAIqGwG6Q885zXeNf/7o0bpx0xBF5twgAUCQEdoMY4+YbX7DArZ89Zoy0R4+FSQEAiMc17AYaMEC6/npp4kTp4IOlxx/Pu0UAgKIgsBtsnXWkmTOlIUOkffeVXn457xYBAIqAwM7BuHHSTTe5e7P320968828WwQAaHYEdk623lq67jrpqaekffaR3n477xYBAJoZgZ2jvfaS/vAH6cEHXff40qV5twgA0KwI7JxNmSJdfbV0331u0ZBly/JuEQCgGRHYTeCQQ6Tf/la66y7pwAOlFSvybhEAoNkQ2E3i0EOlSy91U5h++tPMOw4AKEVgN5Ejj5R+9Svpxhulww+XOjvzbhEAoFkw01mTOf546b33pNNPl/r1k6680m0BAK2NKGhCp53musS//GW3rvbll7stAKB1EdhN6swzXZf42We7ecgvvVTq3z/vVgEA8kJgN7GvftWtn33OOW42tGuvlQYPzrtVAIA8MOisyZ19tnTRRdLNN0sf/7j0+ut5twgAkAcCuwCOPdbNiPbAA9Kuu0rz5+fdIgBAoxHYBXHwwdKsWW51r513lp58Mu8WAQAaicAukN12k+64ww1G22UX6Z578m4RAKBRCOyC2WorN4XpeutJe+zhlukEAPR9BHYBbbyxC+1Jk9ziIZdfnneLAAC9jcAuqJEjpdtuc93kRx0lnX++uwUMANA3EdgFtuaarkv8sMPcPdsnncT84wDQVzFxSsENHOiW5txwQ+kHP5Beekn63e9cmAMA+g4q7D6grU36/vfdSl833yztuKM0b17erQIAZInA7kOOP97dq/3qq9L220v/+EfeLQIAZIXA7mP22EO6/35p/fWlvfeWLr447xYBALJAYPdBm2ziJlXZYw/pi1+UvvIVadWqvFsFAKgHgd1HrbWW9Je/uG7y739fOuAA6a238m4VAKBWBHYf1q+fG4h24YXSX/8q7bCD9NRTebcKAFALArsFHHecG4D25psutK+9Nu8WAQCqRWC3iF13dctzvu990n//t5tk5b338m4VACAtAruFjB0r/fOf0v/8j/TLX0of/rD0wgt5twoAkAaB3WIGDJB+9CPpT3+S5s6VttlGmjEj71YBACohsFvUlCnSgw+6W8CmTJHOPFPq6Mi7VQCAJAR2C/PLdJ5wgvTDH0q77y4tWJB3qwAAcQjsFjdwoLueffXV0iOPSFtvLc2cmXerAABRBDYkSYceKs2ZI22wgbT//m7CleXL824VAMAjsLHaxIluHvIzzpD+7//cAiJMtAIAzYHARomBA9262rNmSYsWSZMnu/C2Nu+WAUBrqyuwjTHrGmP+Zox5rns7LGafscaY24wxTxljnjDGnFrPOdEYe+4pPfywtPPObqa0//ovt2wnACAf9VbYZ0m61Vq7maRbu59HdUr6krV2S0k7SjrRGDOpzvOiATbYwFXaF1zgpjZ9//ul66/Pu1UA0JrqDewpki7vfny5pAOjO1hrX7HWPtj9eKmkpySNqfO8aJC2Nunkk9092+PHS5/8pHTUUdKSJXm3DABaS72BPcpa+4rkglnSyHI7G2PGS9pa0n1l9jnGGDPHGDNn8eLFdTYPWdlyS7fG9te/Ll15pbTVVtLtt+fdKgBoHRUD2xjzd2PM4zF/plRzImPMUEnXSTrNWvt20n7W2unW2snW2skjRoyo5hToZf37S+edJ915p3u8++7SKadIq1bl3TIA6Pv6VdrBWvvxpPeMMa8ZY0Zba18xxoyWtChhv/5yYX2VtZaroAW3005uQNpXv+q6xtvb824RAPR9FQO7ghmSjpR0fvf2z9EdjDFG0iWSnrLW/rjO86FJDBniBqN1deXdEgBoDfVewz5f0p7GmOck7dn9XMaYDYwxfoLLD0v6rKSPGWMe7v6zX53nRZNo405+AGiIuipsa+3rkvaIeX2hpP26H98pydRzHgAAWh31EQAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAABDYAAAVAYAMAUAAENgAABUBgAwBQAAQ2AAAFQGADAFAAxlqbdxsSGWMWS3oxw0MOl/SfDI/XivgO68d3WD++w2zwPdYv6+9wI2vtiLg3mjqws2aMmWOtnZx3O4qM77B+fIf14zvMBt9j/Rr5HdIlDgBAARDYAAAUQKsF9vS8G9AH8B3Wj++wfnyH2eB7rF/DvsOWuoYNAEBRtVqFDQBAIfW5wDbG7GOMecYYM9cYc1bM+8YYc0H3+48aY7bJo53NLsX3OLX7+3vUGHO3MWarPNrZzCp9h6H9tjPGrDLGHNLI9hVBmu/QGLObMeZhY8wTxph/NrqNzS7Fv+W1jTE3GmMe6f4OP5dHO5uZMeZSY8wiY8zjCe83JlestX3mj6R2SfMkbSxpgKRHJE2K7LOfpJslGUk7Srov73Y325+U3+POkoZ1P96X77H67zC03z8kzZR0SN7tbqY/Kf87XEfSk5LGdT8fmXe7m+lPyu/wbEnf6348QtIbkgbk3fZm+iPpI5K2kfR4wvsNyZW+VmFvL2mutfZ5a+1KSddImhLZZ4qkK6xzr6R1jDGjG93QJlfxe7TW3m2tfbP76b2SNmxwG5tdmv8WJelkSddJWtTIxhVEmu/wcEnXW2tfkiRrLd9jqTTfoZW0pjHGSBoqF9idjW1mc7PW3iH3vSRpSK70tcAeI2l+6PmC7teq3afVVfsdHS336xKBit+hMWaMpIMkXdTAdhVJmv8ON5c0zBhzuzHmAWPMEQ1rXTGk+Q5/IWlLSQslPSbpVGttV2Oa12c0JFf6ZX3AnJmY16LD4NPs0+pSf0fGmN3lAnuXXm1R8aT5Dn8q6SvW2lWuuEFEmu+wn6RtJe0haQ1J9xhj7rXWPtvbjSuINN/h3pIelvQxSZtI+psx5l/W2rd7uW19SUNypa8F9gJJY0PPN5T71VjtPq0u1XdkjPmgpIsl7Wutfb1BbSuKNN/hZEnXdIf1cEn7GWM6rbV/akgLm1/af8//sdYuk7TMGHOHpK0kEdhOmu/wc5LOt+5i7FxjzAuStpB0f2Oa2Cc0JFf6Wpf4bEmbGWMmGGMGSDpU0ozIPjMkHdE9qm9HSUusta80uqFNruL3aIwZJ+l6SZ+lmolV8Tu01k6w1o631o6X9EdJJxDWJdL8e/6zpF2NMf2MMYMl7SDpqQa3s5ml+Q5fkuuhkDFmlKSJkp5vaCuLryG50qcqbGttpzHmJEmz5EZHXmqtfcIYc1z3+xfJjcbdT9JcScvlfl0iJOX3+A1J60n6VXeF2GlZRGC1lN8hykjzHVprnzLG3CLpUUldki621sbeetOKUv53+C1JlxljHpPr2v2KtZYVvEKMMVdL2k3ScGPMAknflNRfamyuMNMZAAAF0Ne6xAEA6JMIbAAACoDABgCgAAhsAAAKgMAGAKAACGwAAAqAwAYAoAAIbAAACuD/AbPNYaPWcqPgAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 576x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "f = plt.figure(figsize=(8,8))\n",
    "\n",
    "xs_noreg,ys_noreg=[],[]\n",
    "for v in iterates_noreg[:]:\n",
    "    xs_noreg.append(v[0])\n",
    "    ys_noreg.append(v[1])\n",
    "plt.plot(xs_noreg,ys_noreg,'-',color=\"blue\")\n",
    "\n",
    "xs_reg,ys_reg=[],[]\n",
    "for v in iterates_reg[:]:\n",
    "    xs_reg.append(v[0])\n",
    "    ys_reg.append(v[1])\n",
    "plt.plot(xs_reg,ys_reg,'-',color=\"red\")\n",
    "plt.plot([1],[0],'o',color=\"black\")\n",
    "\n",
    "\n",
    "#plt.xlim([-.5,2])\n",
    "#plt.ylim([-.012,.012])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib as mpl\n",
    "from mpl_toolkits.mplot3d import Axes3D\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "mpl.rcParams['legend.fontsize'] = 10\n",
    "\n",
    "fig = plt.figure()\n",
    "ax = fig.gca(projection='3d')\n",
    "zs = np.linspace(0, len(xs_noreg), len(xs_noreg))\n",
    "ax.plot(xs_noreg[0::20], ys_noreg[0::20], zs[0::20],color='red', label='parametric curve')\n",
    "ax.plot(xs_reg[0::20], ys_reg[0::20], zs[0::20],color='blue', label='parametric curve')\n",
    "\n",
    "ax.legend()\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'xs_noreg' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-2-aaa5437061c1>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      5\u001b[0m     \u001b[0mper\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;36m20\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      6\u001b[0m     \u001b[0mspamwriter\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mwriterow\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;34m'a'\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m'v'\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m'Time'\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 7\u001b[0;31m     \u001b[0;32mfor\u001b[0m \u001b[0mi\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mrange\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mint\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mlen\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mxs_noreg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m/\u001b[0m\u001b[0mper\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m      8\u001b[0m         \u001b[0ms\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;34m'1'\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m'0'\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0mstr\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mint\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mi\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      9\u001b[0m         \u001b[0mspamwriter\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mwriterow\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0ms\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mNameError\u001b[0m: name 'xs_noreg' is not defined"
     ]
    }
   ],
   "source": [
    "import csv\n",
    "with open('eq.csv', 'w', newline='') as csvfile:\n",
    "    spamwriter = csv.writer(csvfile, delimiter=',',\n",
    "                            quotechar='|', quoting=csv.QUOTE_MINIMAL)\n",
    "    per=20\n",
    "    spamwriter.writerow(['a','v','Time'])\n",
    "    for i in range(int(len(xs_noreg)/per)):\n",
    "        s=['1','0',str(int(i))]\n",
    "        spamwriter.writerow(s)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import csv\n",
    "with open('reg.csv', 'w', newline='') as csvfile:\n",
    "    spamwriter = csv.writer(csvfile, delimiter=',',\n",
    "                            quotechar='|', quoting=csv.QUOTE_MINIMAL)\n",
    "    per=20\n",
    "    spamwriter.writerow(['a','v','Time'])\n",
    "    for i in range(int(len(xs_noreg)/per)):\n",
    "        s=[\"{:.5f}\".format(xs_reg[i*per]),format(\"{:.5f}\".format(ys_reg[i*per])),str(int(i))]\n",
    "        spamwriter.writerow(s)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import csv\n",
    "with open('noreg.csv', 'w', newline='') as csvfile:\n",
    "    spamwriter = csv.writer(csvfile, delimiter=',',\n",
    "                            quotechar='|', quoting=csv.QUOTE_MINIMAL)\n",
    "    per=20\n",
    "    spamwriter.writerow(['a','v','Time'])\n",
    "    for i in range(int(len(xs_reg)/per)):\n",
    "        s=[\"{:.5f}\".format(xs_noreg[i*per]),format(\"{:.5f}\".format(ys_noreg[i*per])),str(int(i))]\n",
    "        spamwriter.writerow(s)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "reg=np.array([[xs_reg[i],ys_reg[i],i] for i in range(len(xs_noreg))])\n",
    "noreg=np.array([[xs_noreg[i],ys_noreg[i],i] for i in range(len(xs_noreg))])\n",
    "eq=np.array([[1,0,i] for i in range(len(xs_noreg))])\n",
    "\n",
    "import pandas as pd\n",
    "df_reg=pd.DataFrame(reg,columns=['a','v','Time'])\n",
    "df_noreg=pd.DataFrame(noreg,columns=['a','v','Time'])\n",
    "df_eq=pd.DataFrame(eq,columns=['a','v','Time'])\n",
    "\n",
    "import plotly.express as px\n",
    "import plotly.express as px\n",
    "fig = px.line_3d(df_noreg,x='a',y='v',z='Time')\n",
    "fig.update_layout(\n",
    "    margin=dict(l=0, r=0, t=0, b=0),\n",
    ")\n",
    "fig.show()\n",
    "fig.write_image(\"plotly.svg\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.1"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
