{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "311d2f8e",
   "metadata": {},
   "source": [
    "# Import libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "9f3ee380",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[ENV WARNING] deactivate virtualenv to allow for testing Actionable Recourse\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from dirrac.data.synthetic_data import DataSynthesizer\n",
    "from dirrac.classifier.logistic import logistic_classifier\n",
    "from dirrac.optim.opt import Optimization\n",
    "from utils import pad_ones"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "e4d184fa",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Matplotlib config\n",
    "SMALL_SIZE = 8\n",
    "MEDIUM_SIZE = 10\n",
    "BIGGER_SIZE = 14\n",
    "\n",
    "plt.rc('font', size=BIGGER_SIZE)          # controls default text sizes\n",
    "plt.rc('axes', titlesize=BIGGER_SIZE)     # fontsize of the axes title\n",
    "plt.rc('axes', labelsize=BIGGER_SIZE)    # fontsize of the x and y labels\n",
    "plt.rc('xtick', labelsize=BIGGER_SIZE)    # fontsize of the tick labels\n",
    "plt.rc('ytick', labelsize=BIGGER_SIZE)    # fontsize of the tick labels\n",
    "plt.rc('legend', fontsize=MEDIUM_SIZE)    # legend fontsize"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e04c1168",
   "metadata": {},
   "source": [
    "# Synthesize data and train classifiers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "7c0781da",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Original data parameters\n",
    "mean_0 = np.ones(2) * (-3)\n",
    "mean_1 = np.ones(2) * 3\n",
    "cov_0 = cov_1 = np.identity(2)\n",
    "n = 1000\n",
    "\n",
    "\"\"\" Here we synthesize original data, \n",
    "and 100 data distribution shifts \n",
    "with 50 mean shifts, 50 covariance shifts with\n",
    "shifted parameters \\alpha and \\beta adaptive to iteration with factor 0.1 \"\"\"\n",
    "sd = DataSynthesizer(mean_0, cov_0, mean_1, cov_1, n)\n",
    "features, labels = sd.synthesize_modes_data(100, [0.5, 0.5, 0], [0.1, 0.1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "e9738222",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Fist we split the original data, and train the original classifier on training data\n",
    "X_train, X_test, y_train, y_test = train_test_split(pad_ones(features[0]), labels[0], test_size=0.2)\n",
    "clf, orig_theta = logistic_classifier(X_train, y_train, intercept=True)\n",
    "\n",
    "# X_recourse is all the instances that are classified as negative class\n",
    "X_recourse = X_test[clf.predict(X_test) == 0]\n",
    "\n",
    "# We choose the first instance to visualize\n",
    "# x_0 = X_recourse[np.random.choice(X_recourse.shape[0], 1, replace=False)].squeeze()\n",
    "\n",
    "# For illustration, we choose $x_0 = [-1, -1]$\n",
    "x_0 = np.array([0, 0, 1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "1b9fd0fe",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train 101 classifiers, the first is on the original dataset and the others are on shifted data\n",
    "all_coef = np.zeros((101, 3))\n",
    "for i in range(101):\n",
    "    all_coef[i] = logistic_classifier(pad_ones(features[i]), labels[i])[1]\n",
    "    \n",
    "# Get theta and sigma with 2 types of distribution shifts: mean shifts and covariance shifts\n",
    "mean_shift = all_coef[1:51]\n",
    "cov_shift = all_coef[51:101]\n",
    "\n",
    "theta_0, sigma_0 = np.mean(mean_shift, axis=0), np.cov(mean_shift.T)\n",
    "theta_1, sigma_1 = np.mean(cov_shift, axis=0), np.cov(cov_shift.T)\n",
    "\n",
    "theta = np.zeros((2, 3))\n",
    "sigma = np.zeros((2, 3, 3))\n",
    "\n",
    "theta[0] = theta_0\n",
    "theta[1] = theta_1\n",
    "\n",
    "sigma[0] = sigma_0\n",
    "sigma[1] = sigma_1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "e7e7c16b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# For illustration, we choose different theta\n",
    "theta = np.array([[ 1.38772071,  2.05998749, -0.59898248], [2.66331018,  1.65496587, -1.92942816]])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "51f72fc9",
   "metadata": {},
   "source": [
    "# Initialize model parameters and find take projection to feasible set"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "846ed904",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Model parameters\n",
    "delta_add = 2\n",
    "K = 2\n",
    "dim = 3\n",
    "p = np.array([0.5, 0.5])\n",
    "rho = np.array([1, 1])\n",
    "lmbda = 0.7\n",
    "zeta = 1\n",
    "opt = Optimization(delta_add, K, dim, p, theta, sigma, rho, lmbda, zeta, padding=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "4dc08876",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Find delta_min\n",
    "delta_min = opt.find_delta_min(x_0)\n",
    "delta = delta_min + delta_add"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "1bc8aa55",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "3.086185448508999"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "delta"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "3c67c41f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Find x w.r.t to delta_min\n",
    "x_delta_min = opt.projection_moments_infor(x_0, X_recourse[0], delta)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "e0857ff8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([0.66484143, 1.06729751, 1.        ])"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_delta_min"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "3915201f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Constraint to visualize\n",
    "def constraint1(x0, x1):\n",
    "    return np.sqrt((x0 - x_0[0]) ** 2 + (x1 - x_0[1]) ** 2 + 1) - delta \n",
    "\n",
    "\n",
    "def constraint2(x0, x1, k, rho):\n",
    "    f_val = 0\n",
    "    A_k = -(theta[k][0] * x0 + theta[k][1] * x1 + theta[k][2] * 1)\n",
    "    C_k = rho[k] * np.sqrt(x0 ** 2 + x1 ** 2 + 1)\n",
    "    f_val = (A_k + C_k)\n",
    "        \n",
    "    return f_val"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1b0155d0",
   "metadata": {},
   "source": [
    "# Visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "1aa7ffef",
   "metadata": {},
   "outputs": [],
   "source": [
    "def visualize(x_0, delta):\n",
    "    \n",
    "    fig, ax = plt.subplots()\n",
    "    xmin, xmax = -1, 4\n",
    "    ymin, ymax = -1, 3\n",
    "    \n",
    "    x = np.linspace(xmin, xmax, 500)\n",
    "    y = np.linspace(ymin, ymax, 500)\n",
    "    X, Y = np.meshgrid(x, y)\n",
    "    \n",
    "    # Calculate constraint values\n",
    "    J_grid1 = constraint1(X, Y)\n",
    "    J_grid2 = constraint2(X, Y, 0, rho=np.array([0, 0]))\n",
    "    J_grid3 = constraint2(X, Y, 1, rho=np.array([0, 0]))\n",
    "    \n",
    "    J_grid4 = constraint2(X, Y, 0, rho=np.array([1, 1]))\n",
    "    J_grid5 = constraint2(X, Y, 1, rho=np.array([1, 1]))\n",
    "\n",
    "    # Plot x_0 and x_projection\n",
    "    ax.scatter(*(x_0[0], x_0[1]), zorder=5, s=5, c='black')\n",
    "    ax.annotate('$x_{0}$', (x_0[0] - 0.3, x_0[1] - 0.4))\n",
    "    ax.scatter(*(x_delta_min[0], x_delta_min[1]), zorder=5, s=5, c='black')\n",
    "    ax.annotate('$x_{proj}$', (x_delta_min[0]- 0.3, x_delta_min[1] - 0.4))\n",
    "\n",
    "    # Visualize contour of constraint\n",
    "    ax.contour(X, Y, J_grid1, 0)\n",
    "\n",
    "    CS = ax.contour(X, Y, J_grid2, 0, colors='blue', linestyles='dashed')\n",
    "    CS.collections[0].set_label(r'$-\\hat\\theta_1^\\top x = 0$')\n",
    "\n",
    "    CS = ax.contour(X, Y, J_grid3, 0, colors='orange', linestyles='dashed')\n",
    "    CS.collections[0].set_label(r'$-\\hat\\theta_2^\\top x = 0$')\n",
    "    \n",
    "    CS = ax.contour(X, Y, J_grid4, 0, colors='blue')\n",
    "    CS.collections[0].set_label(r'$-\\hat\\theta_1^\\top x + \\rho_1 \\|x \\|_2 = 0$')\n",
    "\n",
    "    CS = ax.contour(X, Y, J_grid5, 0, colors='orange')\n",
    "    CS.collections[0].set_label(r'$-\\hat\\theta_2^\\top x + \\rho_2 \\|x \\|_2 = 0$')\n",
    "\n",
    "    ax.set_xlim(xmin, xmax)\n",
    "    ax.set_ylim(ymin, ymax)\n",
    "    \n",
    "    ax.set_xlabel('$x_{1}$')\n",
    "    ax.set_ylabel('$x_{2}$')\n",
    "    \n",
    "    plt.gca().set_aspect('equal', adjustable='box')\n",
    "    # ax.legend(frameon=False, loc='upper right', bbox_to_anchor=(0.4, 1.05, 1.1, .11))# , ncol=5)\n",
    "    plt.axis('off')\n",
    "\n",
    "    if not os.path.exists('result/figure2/'):\n",
    "        os.makedirs('result/figure2/')\n",
    "    plt.savefig('result/figure2/feasible_set.png', dpi=1000, transparent=True)\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "de1cf513",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAR4AAADnCAYAAADIIzmzAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/MnkTPAAAACXBIWXMAAAsTAAALEwEAmpwYAAA09ElEQVR4nO2dd5gU1dKH3wIE8zVg1iuOqJhQVPAKBsyY9TM3BozXdMcwoqJezNlRGRUVMWMbEL1gAEXFiAHBBGJiVEQwoYIiEuv7o87CgrC7sxO6Z+a8z7NP4+50d83aU3tOhV+JquLxeDylpEnUBng8nurDOx6Px1NyvOPxeDwlxzsej8dTcrzj8Xg8Jae6HU8oexDKM4TSPGpTPJ5qorodDywL7ANsG7UhHk81Ue2O52VgLrBb1IZ4PNWEVEsBYSKTFqApIO4LYO64lc99U4Q5BNoxOus8nuqirBxPIpNuBqwOrAWs4f69KtASWAlYEVgBWM59LQ0sBbQAFhnHOWfpwZyy1DC2/uWyP37XpaYDfwJ/AL8DvwG/Ar8APwM/Aj8Ak4DvgEnZZGpmEd6qx1PRNIvagIVJZNLLARu6rw2A9YH1gFaYw1nU9nAq5hh+BaZgDuIPYBowHZgBzARmAYptrwRo8t2cFddvJnOPPWzJd16+Z3rniZizWhZYHnNorYGVMae2MJrIpL8Hvga+AsYBXwKfA59lk6lf8/hVeDwVS2QrHrd62QjYEtgC2BzYFFhnoZdOwD7UXwPjgW+x1cZEbPXxU16rDsto/QrcR6Bn1GNvS2yVtQbmBNcG/ok5xfXcv2s7xu+BMcDHwEfAB8AYv0ryVDslcTwuvrIusB2WQWoPtMO2QWCrkbHYh/QT9+/PgS+zydRfRTcwlGeB1gS6UT6XSWTSzTEHtCHQBtgE2AxzqLXf68fAu8A7wNvA59lkqnz2vB5PnhTF8ThHswnQGdgR2B5Y0/14OjAKeA8YCbyPbUtmFdyQhhLK2cBNQCsC/abQl09k0k2xLduWwNbANu64vHvJZOAN4DXgVeCDbDI1p9B2eDxxoWCOJ5FJrwXsAewO7AKs5n40AXgdeBMYDoyO1MksilA2BUYDJxFo31LcMpFJNwE2xlaBnYAdsHgWWFB7GDAUeCGbTI0rhU0eT6lotONxf8W3BfYD9gbauh/9ALyI1ci8AnwV+21EKII5yDcI9PCozEhk0mtiq8RdsNqidd2PvgCeA54BXvMxIk+5k5PjSWTSLYBdgYMxh7MKMBvbJgwGhgAfx97RLIpQ7sfe06oEGvk2x21XNwD2xBx7Z2BJLIP3LPAkMDibTE2LykaPp7HU7XhCufXL2atet8dv57UFDgcOAP7B/Id/EDAkm0z9VnxTi0woXYF+QHsCfS9qcxYmkUkvja2CDgD2xzJs07GV0GPAM9lkanp0Fno8DadOxzP74aZzXpi52ewzfj+mORZ3+B/QH3ix4pb7oayKbRMvItCrozanLtw2dwfgUGz1uRpWtzQAeAh4xQenPXGmTsfz0D2dPjt6qeEb3vrnbhfe/GeXm7LJ1IwS2lZ6QnkfmEKgnaM2paE4J9QZCIBDsEzZBOAB4N5sMpWNzjqPZ9HU6XgmPLRSi7Wb/no0cD+Bzi6dWRERyvXAWcBKBPpHxNbkTCKTXgrbhnXDMoxNsEB/H2Bgxa1SPWVLw4PLoTQn0Mp+cEPZFfug7kegz0RtTj4kMum1geOAE7Ds2PfA3cCd2WRqYpS2eTwNczyhbA88CuxBoJ8U26jICKUF1j7Rl0CTUZtTCNxWbE/gdGAvYA7wOHBzNpmKXRDdUx001PGsAnyKtTTsRFBGLe25EspgIJFv+0QcSWTS6wNnYKug5bBK6RuA57LJ1NwobfNUF7lstU4A+gInEOi9xTQqUkI5C7gZWI9Av47WmOKQyKSXB07E4lnrYFXb1wKPZZOpyo/leSInF8fTBOsj2gRoQ6A/FdGu6AhlY6xR9d8E2idqc4pJIpNeAjgCOB9rZP0SuAro5x2Qp5jk1jJhPU0fACkCzRTJpmix9olvgBEEenDU5pQC1zd2AHAxsBWmK3Q58LCvB/IUg9x7tUJpQ6CfFsecmBBKX6wmpmVVlBE4XJvGfsClmGzJWMwZPVWWbTCe2JK72HuN0wllfZcFqkSex1pDqmr6RDaZ0mwyNQiT7DgEU2kcALyVyKR3iNQ4T0XRuCkTobTCMlznFdKYGPEiJo+6Z9SGRIFzQAMwVcgTsQD0a4lM+qlEJt06Wus8lUDj9XhCeQyLC2xOoF8U0qhYEMpwoBmBdojalKhxDarnYEHoFkAv4IpsMjU1UsM8ZUs+jmcNrLbnXaywsLJiAKH0xGIdqxLozxFbEwsSmfQaWNarG9ZQex6WAaus//eeotP4gX6BTgIuxKQajiyUQTHieSzGsXvUhsSFbDI1KZtMHY/FvsYDDwKvJDLpTaO1zFNu5DtJ9E7gLUxPuNJ4D9NC7hK1IXEjm0yNwCRbT8LE7D9IZNJXuSZVj6de8tdcDmUJAo2XhnKhCCXEZEjXJFDfUrAIEpl0S+BG4FisAPHEbDL1arRWeeJO4aZMhLIj8Gcc1fsaTSjHYLo2WxPoqKjNiTOJTHpXTH4jAdwBnJ9Npn6P1ipPXMl3q2XYULyHgHsJZYmCXDMePO+Oe0VqRRmQTaZewtLvNwOnAB8lMunOkRrliS2FXPEcgEmjnk+g1xfmojEglJHYSs4X0DWQRCbdCbgfG9dzC3BhSQYzesqGwqx4AAIdCAwELnUFhpXCYGA7QlkhakPKhWwy9SY2vPAO4GxgRCKT3jxSozyxorCTRENZB+vveQVT8Sv/+o5QOmHjew4j0P5Rm1NuJDLpvYD7gBWAFNDb1/14CrfiAQj0W6A71nIgBb12dLyDTdjwcZ5GkE2mBmPDHl8GbgOeTGTSK0ZrlSdqijI7veKw9pAdsbS6/4U1Atf5fg4mODYBONRLr1YvxXM8NiCvNYFeVpwblJBQjsWCpT6tnieJTHpbTPN5dSAJ9PFbr+qjsFutBekI9CSUrYt4j1IxxB33jtSKCiCbTL2DiY29jFW+35PIpJeM1ipPqSmm47kQ+BHoQyhNi3if4hPoD1gLhXc8BSCbTE0G9gGuwEbwvO7G8XiqhDodjwjdRVipUVcOdAomJr4VNlql3HkO2JZQVo7akEogm0zNzSZTPYEDgTZYyv1f0VrlKRX1rXiuBm7P4/qPY9uUqwilZR7XiQPPYr8v3zRaQLLJ1EDgX8CfWKd7ELFJnhJQn+O5DDhChCMadXXLAJ0OHF0BmjbvAT9hWwRPAckmU2OADsDbwMOJTPoSlwXzVCh1ZrVEaIYVz20IbK7Kd3ndrdw72UN5ABNDX7WaROBLRSKTbo41mh6LNeee7Oe9VyZ1rnhUmQ0cjcld3iuSR1FgKCcCHxPKMo2+RvQ8A6yIbQ08BcY5meOASzDn82wik14uWqs8xaDerJYqXwDnAnsAp+Vxr8+AjbCHqlx5AZgN7Bu1IZWKE5q/HHNAO2Nxn9UiNstTYBpUQOhWOs8BOwHtVPmsUXezeVXdgHYE+nGjrhE1obyMbbU2i9qUSieRSe8NPAF8B+yeTaa+jtYiT6FoUB2PKgqcAEwHHhKhsZo75wO/Ane5kcjlyDPAphXWgR9LssnUc5imd0vgjUQm3SZikzwFosEfflUmAv8G2gMXNepugU7GOpS3pXzjJE+7o99ulYBsMjUcW2k3w2Z7tY3YJE8ByGnVocoTmNLgxSKNnrL5EDaLa3gjz48WmyH2OZbd8pSAbDL1EdakOwOL+VRCG05Vk3OTqAj/AD4CZgJbqjKt0Xcv1znsodyINTiuTKBeV7hEJDLp9YBh2Hjp3X13e/mSc5xFlSnAMZis5U2NvnMoBwJjCaUc51Y9DSyBZfo8JSKbTH2Fbbt+A4b6lU/50qgAryqvAjcAJ4s0essxBPgC6E0o5dad/CYWJN8/akOqjWwy9Q3QGZiCOR8f8ylD8sks9QQ+BO4RIfc6i0D/Ak7FhgFemIcdpceqlp8D9in7zvsyxDmfnYFpwIs+21V+NNrxqDID6Aosjzmf3KuaA30JeBi4gFDK7eEZBKyM6Q55Sozbdu0KKOZ81o3YJE8O5FVLo8oY4DyscfKURl7mHGASsEk+tkTAEGAWfrsVGdlk6nNstv0ymPPxFc5lQt7Sp26lMxhLd26lSu5ZqlCaE2j5NQOG8jywHrCR12KOjkQm3REbMDAW2DmbTE2N2CRPPeRdPeyqmo/D9tuhCM1zvkigMwlFCCUoM92egcAGmJCVJyJckeEhwBbAE67L3RNjCtK2oMok4ESgHXBlIy+zPiaFcF0hbCoRg9zxwCiN8MxrrzgJ23r18Xo+8aZg/VKqDMS0VM4VYZecLxDol0AaOJ5QdiyUXUUl0AmYQNgBUZvigWwydR/zJTV6RmyOpw4K3ah5DtZO8KAIjdEmvhz4GriTUMplufw/TIt5zagN8QAmIP8AcKmXUY0vBXU8rn0iAFYF+uScYg/0T0wqdWNsImk58JQ7+lVPDHAzuk4GXgPu9QLy8aTg0hSqjMIKAv8Pi/vkRqDPARmgXPR6xmIV2AdFbYjHcEqGB2M6Pk/50TnxoyiTREVoAjwPdKKxKfZyIpRrMbmPVQn016jN8RiJTHpTTED+U2CHbDL1V8QmeRxFEeNSZS4W4PsTeESEFjlfJJQlCOViQjm00PYVgacwvRiv0RMj3PSKo4BtgN4+0xUfiqYC6ITDjgO2BK5txCXmYnGTWwllhcJZVhRGYMv6g6M2xLMgbm5XzcTSkyM2x+MoqvyoKk8DtwFnibBXTicHOgdTPFwFGywYXwKdi6169iSUZaM2x/M3LsVaXDKJTHqbiG3xUGTH4+iOCYc9IMIaOZ0Z6CjgVuAUQol7dmIAsCTk6GA9RSebTM3FtlzfA/0TmfSKEZtU9RTd8ajyF3AksCxW35PrPf8LTMR0e+K8R38d+BEr3ffEjGwyNRk4DFgbuMfHe6KlJJMeVPkEOBObGHBeTiebtOgxwEmxbsS0reFTmEbPUlGb4/k72WTqHeACrPQhnxlxnjwp5YiZvsDjwJUibJfTmYG+TKAjAQilWeFNKxhPYBINXaI2xLNYbsZE3NKJTHrzqI2pVkrmeFwX+8nAeCzFnvs+O5ReWCwlrrwCTAbKoQSgKnHxnuMw3eYwkUmXm+xuRVDSoXpOKP5IYC2gbyNUC78F9ieUeFYJmyTqk8B+frsVX7LJ1I+Y89kMuCpic6qSkk/zVOUdoAfWUpHrPrsXliHLEMpyhbatQPTHAuk+uxVjssnUYOAO4OxEJt05YnOqjqjGCN+E7bNvEqFdg88KdBZW27MW1skeR4YBP2MZFE+86Q6MA+5LZNK+/qqEROJ4arVU/AT0F2H5Bp8c6NvAXcBxhBK/egzbbg3AtlvLRG2OZ/Fkk6lpQDdgXeD6aK2pLqJa8aDKz8ARQCvg7hzjPT2wMchxbch8FFgaE8H3xJhsMvUmcAtwaiKT3ilic6qGyBwPgCpvABdj25JTG3xioL8R6LdOp3mDYtmXB69jVbJHRG2Ip0FcDGSBvolM2icFSkCkjsdxPRbvuVmEXEfSXgGMjJ36nxUTPgbsTSj/iNocT91kk6k/sVKP1pgT8hSZyB1PrXjPD1i8Z4UcTr8PaI4tlePGo0ALLHvniTnZZOol4EHgPKfj4ykikTsemBfvOQxYB7i/wfGeQMdhUy0OJZS4pa/fwZbvR0ZtiKfBnAtMxWv3FJ1YOB4AVd7G0psHYA9AQ7kBU5jrTShLF8O2RmF9ZSGwK6GsHrU5nvrJJlM/AedjwymPiticiiY2jsfRC+t3ukaEho24CXQGNj65KbZHjxMh9jv2Qeby4V7gXeD6RCbd8DIPT07EyvG4fq4TsKKuxxqs3xPoq0BrAv2oiOblTqBjgVH4v55lg+vlOgNYHR9oLhqxcjwAqkzFJESXBx4XYYkGnWhjkFsQygmEEqf31Q/YmlA2jtoQT8PIJlMjsMTFWYlMOm6r6IogTh/QeagyGhuNsz25VZQegMlvnFAMuxrJI5h+9NFRG+LJiYuAGfiK5qIQS8cDoMoj2Hyts0QanBnqD7wKXEcoqxbNuFwI9Hts1M9RMVuJeeogm0xNwoYUHJTIpLeP2p5KI+4fhHOBNzAJjbb1vtoySadi3eHp4pqWEw9ipQI7R22IJyduxmR3b/Dp9cISa8ejyiysvuc34CkRVqr3JAvoXoetMHYtqoENZyAwBSuU9JQJrqK5J/Av/KTYghJrxwOgyiRMQH0dIBShaQNOuxpLy08upm0NJtDpWAvFIYTiU7TlxQNYndiViUy6Ic+epwHE3vEAqPIWcDqwJw2ZsRXodAI9lEA/KLJpuXAfsBRep6esyCZTs7FJJxsDXSM2p2IoC8cDoMrdwJ3AeQ0ONoeyIqHcRSgbFdW4hvEOMBY4PmpDPDnzJPA+0DORScd52EDZUDaOx3EmJjlxrwgNmQjZAjgcuCPymVwW+L4X2M7X9JQXrqjwUmB9fDFoQSgrx6PKTCze8yPwv3ormy2VfQGWTYrDA/MQMJt41Rl5GsbT2KrnIr/qyZ+ycjwAqvwI7A+siGW66htP0gd4G7iJUFYutn11EugPwCDgWEJpEaktnpzIJlOK6T+1xsfp8qbsHA+AKh9ilcDbAvfUKaMR6FxMIH5F7MGJmruBlsCBEdvhyZ2BwCdAj0QmXZafnbhQtr88VZ7EytoDd1w81jzaDbim6IbVzwvAN5jinaeMcLGea7F5XHtHbE5ZU7aOx3EN1oR5hQiH1/nKQPvV0mmOrh7DVmB3A7sQyoaR2eFpLI9igyW7R21IOVPWjsfJaJyItVU8UO9MdhMKGwqcV3zr6uQeLMj874jt8ORINpmahUnt7pjIpBuSWfUsgrJ2PACqzMDK2b8FBomw/mJfHOifWOtCT0JJlMbCRdrxPfAUNhvMTzUoP/oCvwNnR21IuVL2jgfmaTbvg72f50SoK3t1JrbauD3i2p7bsYC312QuM7LJ1FRs1XpYIpOO14STMqEiHA+AKp9jejzrAgNFWPRKItAJmLJcF+DQkhn4d14DxgBnRF7c6GkMt2Fyu3673AgqxvHAvAGBRwMdgX51NJTejkmSnhvZh94qmW8H2kE9sSlP7MgmU+OAwcDJiUy6YSqZnnlUlOMBUKU/cA42z+rWRdb42HzzQ4FdnQOIigeBKbPnkhSR1UT8yqfM6I1pMx8YsR1lR8U5HgBVbsEkK0/FOov/TqBZAv3d6TT/s4Tm1bZh2uw59N31Kg4DJgDDRLxKYRkxBF+T1Sgq+SG/ANNSuUykzrnszwADCSWS/ptbX+DR4Z8jQDOgE7BKFHZ4ciebTM3Bgsy7JTLp6LKkZUjFOh5X43MS5lhur0NK4y5gS+A/JTJtAc7px8iN1+KnZk2gaRPexhpgPeXD/ZiY/3ER21FWiEYa4ig+Lrs1BAs4H6TKMwu8wILLTwOdgY0J9NtS2zj9fuk4dTpvtlyOZNOj9NZS39+TH4lMejCwKbCeWwV56qFiVzw1qDId2A/4AHhChAV1mC24fAb2u4jkQ79UNx2+2j8Y3rQJZ0e15fPkxQOYNG/niO0oGyre8cC8IYFdgC+w6uYdFnhBoF9jQk/rEMo/Sm2f4wZgPUxvyFNeDASm4menNZg6t1oidFDl3RLaU1REWA2bu7UWsIfTcjZqVhqWai89NnPrE+BPYOuI0/yeHElk0vdgJRqrZZOp6VHbE3fqW/G0ABChSZ2aN2WCKj8AuwCTgOdF+Ne8HwY6m0BnE8pKhFL6imbrWr8eKyjcs+T39+RLCCyHte546qFOx6PK6+6fl2FzzMs+1avKREwK9QfgBRE6LvSSi4FHCGXLUtuGSXxMAC6M4N6e/HgFe6aOiNiOsqChMZ4pmNzoGBH+r4j2lARVvsMCgd9jzmenWj++ApvHdWfJdXsCnYnFenYglB1Lem9PXrhs1gBg70QmvUzU9sSdBjkeVW4EtsakJwaIENbTAR57nPPZCRgPDBZhDwAC/RVrudiWaBoA+2K1PIuuuPbEmf7Y7LS9ojYk7jQ4q6XKaGyUa0+sC3ydYhlVKtyU0p2Az4Cna63mQuAl4BpCqXuSRaExzaAbgN0IZeFtoCfevA78DOW/Kyg2OaXTVZmlyhXAP1X5AECE40VYoQi2lQRVfsICziOB/iIc7zJKp2L6yFEE1e8AfsJS/J4ywW23BgH7JDLp5lHbE2caVcejajPJndpfH2C0CF0KaVgpUeVXYHfgRWxqRQ/pql+6McgTS25QoNOwDNfuhLJ9ye/vyYdBwPKAj9HVQV4FhKqMw2IhU7A4yd0iLF8Qy0qMKtOwCucQm8+eEaEpobQmlPsikCjtjQW/r/RCYWXFi8AMYN+oDYkzeVcuqzISCzxfh80Ff0OkPCui3aTSo4EbsTaKAZ9N3LA1Nhrn4vrOF5FDRGSGiKxb63u9RGSciKyWkzEW67kai0HtntO5nsjIJlPTgGH48Td1UhAHocpfqlyAyTpcqspcEUSEsksrqjJXle5Yt/p+bbp/duWnEzd6HDiPUDap5/QBwMc4JyUi52Kayl1U9YdGmNMH03u52lU2e8qD54ANvFTG4inow6zK227QHpg40scL1ciUDarchmXv2mx63pgd3hnXYRpW27PY35la/8mFQDcRuQDLAO6jql80yohAZwCXYCvKQ0Wko4jUPbzQEweed0dfgb4YivlXdAymU/KKCLeIsHQR71UUnIRGp7nadFanS99c6pHhR+wAHFX3OfoCMAK4EjhcVUfkck+RvxUt9sNWUVfpw7ynqlflcj1PJHyBrVT9FnkxFM3xOOH1LTA1/jOBDxfojSoT3Jz29nPmNn03uP0Rljnuj61FWKx0hYjsgr1vwUroa//sKRF5VERGuLjPNu77g0Skt4iMAI4TkbYi8rqIfChdeeKHKVwMrL/5BbxTc44nvmSTKcXqwHZOZNLRTa2NMUWNG6gyTZX/YHUyTYGoJCfyQpUfQXYFev85c5mkMHeoCKsv/DoR2QIb1Pcf4H/8fVZ7W2CUqrbHtmEp9/3Ngc/c9/thY3JPVNUtgIlrnE5L4KVZs9m832l8V4S36Ck8LwMrYOqWnoUoScBSlWFAG1Xb+4pwmggdSnHvQqHKTFVOb9dqVPcWS8zYqXmzGWNE5gs/uUzWYCCtqvdisZndRaSz+/mywJJA2p0yFlhRRJYDmqpqL/f9A4HBqvqZ++9PVVnlw/Fc1LQJTbt28jO7y4RX3LEsY5zFpmSZEpeqrpEiPRd4S4SrRUx6o1wYddXWt79xyfbfrb3ShGVBXxLhUpHNVsHkVZ9W1csBVHU01rtTs+rZHBijqjXSmFthsZtNgeG1brExpstTw6bAJ1v2QFdahq+A/xDKhkV7g56CkE2mvgPG4QsJF0nJU7ROirQdJhfZA3hPhK1KbUejCXT61uuNOunDa7Zo/q/Wb38MXAKjnwTdS1UXaCpV1cNVtWZYX1tgXRFZQkRWxrZjd2IO6eNap00C2gCIyJaYVvRgoO3n3/MYMB24uYjv0FM4Xgc6JTJpXwC6EJHUhqgyRZXjserOlYHXRWgZhS2NItAhyy457bG3LuvYZpv1RpyLOZWPRDi2DsG0tlh9x0hsGX6Bqo7j747nIWATERmNBeYPV9XZQNsfp/IWpo20N6H4ytj4MxxoCfgV6kJEWpSmyrPYVuJIVX4GEGHtKG3KgbOBGSOu7NAKCyB+gI06eWpRgWfM8dykqm1VdXNVHQygqklVHVTzIlWdpqr7qOpmqrq9qo51P9oeS9NnsPhQrwjaODy5UbOFLrtsbrGJvBpWlV9VGQQgwj5AVoSL6kpZx4JAJwE7AGer8hWmanguJir/iQjHLLT6WUs199E5ItJcREYBQ1R1EoHOwto5EsD5eb8PTzEZC/yO9TN6ahGruVpuu3UbcDjwHnCs6gKB1ngSykrAXAL9TYSNsOmSnYChwKmumbbQ93wEOAhoS6CfF/z6noKQyKRfBpbLJlPto7YlTkS+4qmNKj+rcgRwGNAKeF+EM6K1qh5CWRYYjclYoMpnWCbjDGyJPVqEi4uQvTsH64Lu7bvXY817QFuvz7MgsXI8NajSH4v9PAtMi9icugn0D0xK46QaxUDXaHo7lhp/BtNxHu22koW67yRsPvyu+HlOcWYU0Byor8G4qoil44GaamEOxgK2NUqHZ8dUcuNSTI/6LkJZouabqnynyqHAHsAc4BkRBosU7CG8Cwtg3kQoqxbomp7C8r47tovUipgRxw/xPFRRVWqCULsBN2FNp+tHaNbfsVXPGcBm2BZoAVQZimW1UsB2WOr9jsVkv3K571zgJGyeUyava3mKxZdY7VXbqA2JE7F2PAvRFTiO+TUzp8dq9RPoIKw/q+2iYi6u5eImoDWmqXwi8KUIV4jk0cMW6Ce4TnhCObDR1/EUBafDPAar1/I44vPBrQe3+rkfW1W8hmW/4tbvdSSBdq1r/LALoP8H2/M/i4mGfSVCDxGWbeR9r8XqiO4klLIeO1ShjMFilh5H2TieGlSZgMlK7qjK2wAibBOLEcuB/gVAKBsQSp3Ngap8ocrhWM/WcEzm9GvngHLTrbbanm7ASphD9sSLT4DVE5n0ClEbEhfKzvHAvNXP6wAuUPs2MEQkBrO+bJv1IBASSr0ORJX3VdkXS72/izmgb0S4PKc2kkA/BC4HjiCUwxtlu6dYfOqObSK1IkaUpeNZiE+BJNZSMFqEbpGufmybdSawBhZ7aRCqvKPK3kB7rJfrv8B4EW7LIZh+Lea87iCUNXOy21NMaiROfM+Wo+wdj6uZ6Y0FnT8A7sPGLEfpfN7FxtOcTpibYqAq76lyEBYDehTTrv5ChCdF2LHO9xXobKympwVwvxeIjw1fYTLAraM2JC5UzIPp2hJ2xpo336qVho+KizDp07sIJee+M1XGug7+Vpimz07Aq1g19wmL1bC29olzML3fsxpluaegZJOpmcB4iFkZSIRUjOOBeaufW1S5AUCEfd1KIbeZVoUg0CnYFvB1YIl6Xr1YVJmoykXYrPqTsf9nfYHvnIj+xos4rQ8wEJv97gvX4sFXwHpRGxEXKsrxLII1sQzYGBEOK/ndA32CQM8i0On5XkqVP1W5GxOS3wlTPDwN64R/3XXDL+3uq1id0M/Ao66fzBMtX2OrVw8V7nhU6YOlq78CHhPhsUgEx0LZgVBuKMSlXEbvNVWOBNbGpDFWxxQdJ4lwlwgdpatOBgIsrtC7EPf25MV4LKXum0WpcMcD4GQ1tsNiLgcRzWjZTsC5hLJfIS+qyo+qXI9lSzpjldNHAW8Cn0lX3XH8z+vcBhxNKMcV8t6enJmAjTxaI2pD4kDFOx4AVWarcjXWLf4QgAi7iLBSiUy4CatevY1QCj7W2a2CXlXlWGz1czzwHXD5emd9lXznyw5TZs1pdtcz3ff1Ew+io2Ys0VqRWhETqsLx1KDKOFXUtSY8QaGlKhZHoDOBU4B/Yp3sRUOV31W5T5WdgXXnatMeh/R6YuLPU1suscHqX7yy8nKTXxbhpLLSuK4MJrmjX/FQZY6nBlX+wHRsJmNSFffl1ajZEAJ9A8tGnU0oJdFmUWW8Ktd+O3mdTYaN3blb69W+1AdOObY9aB/gexFeEOHkSLJ+1cf37uh/11Sp4wFrVQC2wVoUjgE+LrrzsUDwWUDJpUqD28IHmjaZ233fds8uO+qqrXphionrYZo+E0V4zekdtSq1bVXCz+7odZOImeZyVLippru5OBAiNFNldlFvGkoTp6dTOqyP7DFMYG1P6aovYRXf/4cF3mukGz7C6oAGAaNUKa2dFUoik/4FCLPJVLzlfEtA1a54aqPKu7WcTntgrAi7FO2GoewOjCWU/ITAcsXqe47Hph88pg9LK1U+VOUSVdoCG2CTMqZgWcARwAQR+oiwvwgFD4xXGb9AyRIascY7nkWjwEuuQbMYH7ZvsGKym4pw7boxtcQDsf/3/6tdXKjKl6qkVdkRi0Uci6Xmj8BWQJNFGCJCUoQNSm57+fMrsGLURsQB73gWQpUR2IC+Xlhl8Ici7FDQm1g/1TXAkYSyR0Gv3bD7f4mNENoMeGBRzaROsOxBpxndEpOe7Y05zF7A5yKMc855vzxEzKqJ36DoccSywDueReDaE87CivKaYC0KheZaLMjcO5KJoIG+gG2r/g+4pK6XOtnWl1Q5R5U2WLPjGZjA1XFYLOgXEV4R4UInzOafrb/zO+Qo8lah+IejDlR5DQu+Xgfzig4LM47W1ApPwz7EhxTkmrlzC3Av0JNQjmjoSapkVbldlf2wmMVuwM3ACsBVWGzoJxEed+n6RMEtL09+Bx8nA+946kWVP1SZ5XRwrgLeFOE6EZbM++KBvgS0J9CH8r5W4+6vwKlYB/39hJKzU1VlhlsNna/Klljl9FHA00BHLF0/ToSvROgrwpF5T9coX/7EOx7AO54G4/R99sRWCOcBI0XISeRrkQT6HgChrBvJRFCrqv4/rJdoEKHkJd2gyg+qPKxKN0zKYxPgP5hI28HY8MNJInzi4kOHVFEV9XSg9NvqGOIdTw6oMlWVk4C9sCDh2yIFGNQWypaYPGa3vK/VGAL9GdgH0w16llBWKMRlXQ/ZWFVuc6qKLTFp1/OxzF43oD+2LftYhFtFOFiEVQpx/xgyAwo+yros8Y6nEagyBMsInYv9JSfnyRAL8hE2Y/sGQonmr3+gn2FFhK2BJwml4PINqsxx0q7Xq7IXllruhNUMTcJqjJ4AfhRhjBt6eIQIlaIfPQtYIpFJRz8RJWKq0vGISBMRWU2k8VsbVX5zaocqwnrYaJqeIo1QG7QK5lOwVdT1jbUpbwJ9Bfvw7wzcU+ytnyqzVBmuytWq7IEFpzsCPTD9mqOARzC1xS9FuFeE40RoHYtxRrlTUw1flZ+72lTdL0BEmgAvYzGNYe6/82UKMBi4DNt+bZbzFQIdDdwIHFffTK6iEmg/bMjgUVgwvWQ4R/SWKtfWWhG1x0Y/jwb2x2JsX2D9Zf1dMeNWIuSsax0Bc9yxaaRWxICqczzAKtjyvpk75h1PUOUXVbpiwdN1sMDzBY34q3wFNms7/6B1flyN6Tb3IJTTozLC6Si9p8pNqhyINVhuhmXiXsYmyfYCRgK/ijBUhEtE2F2E5aKyuw58Y6SjGh3Pj1gbwGx3/LFQF1blSWxU7SCgVc6TLgL9E2hLoOlC2dQoLM1+OvY+biWUgyO1x+HE/MeocqcqXVVZF9M4CrAhiqtgxZAvAL+J8L7LnAUitIrB9izq+8eGquxOd9urVYAftQi/APeAN3P1P+2BHYFbVOcttesnlB2B7wh0XKHty8GGpYGh2ApsTxcDijUuyL8dtprtiE1oramdmQi8hY2MfgvrvJ9RKtsSmXRPbDveLJtMNfxZqECq0vGUEhFuxGIUw4FuqnxR70k2+ng8NhV0T7cCiYZQVsIKDNcGOhPo+5HZ0ghc7GdzzAnVfLVyP54BjMKc0FvA26pMKJYtiUz6cuC/2WSq6lc+1bjVKjXdsemem2ANp8l6+5gCnYoFeHfHOsOjI9BfsMLJKcAQQimrrnQXJ3rftXh0VWU9bOzRwcCtWNzldKye6FsRvnVB63NE6FiQCvX5NAdmFvB6ZYt3PEXGFdH1w2I/w7Bg6LENOPUOrOfp5kIV9DWaQCdgTrAJMJRQ1o7UnjxRZZIqT6rSXZVOWOPmttjM+zewrWUaiwFOFWGEK248SoQN8ogVLQml29rFGe94SoQqE4F9sYbQfgB1BjwDnQP8G4tFXVMiMxePFRh2wZpCh0ZW6FgEXPf9u6pkVDnSrYrWwHSL0sAfWBf+Q5iiwE8iPOsyaF1ymFayJNY2UfV4x1NC3OpngAs6r4DFFV4QYd1FnmDxlKsxxcDoCXQksB8WIxlCKBWrLaPK96oMVKWHm9jxD2yK678xUbR1sQzaYEwg7XMR+onwHxG2FVlka8QywLRSvYc444PLEeFWOidjf1HnAmcD9+acgo+CUPbBhge+DXQh0Kr8MLkM2jZYPdG27qtmfM0s4EMsQfAu8G6rm2+6Wpro+tlkqm0U9sYJ73gixrVb3IuJjg0GDlVd6K+itS4cBSxNoHeV2sZFEsphWDvDS8B+BOpjF4AIazPfEXXAHNOyANJ85pzma0z+Y8Y3a9yJxe9GAN+WxR+bAuMdTwxwWa7TsdqTI//2IJrjGYjNAtuEQL8puZGLIpRuwH2Y9s7BBDorWoPihwhNgTZA+2W2+eT6meNXbz7rx5WWhnk9fT8w3wmNAEaozhuFU7F4xxMjRJBaTafXAGepukFwofwTkxodBuwfaW1PbUI5Dbgd6yo/kkCLOxaojElk0llg+Fdnpk7ElC07YL1o7THnVJNo+BpzQu+540hVppbc4CJSDo11VUOtlc5WwAHA7iKcATyqquMJ5RKskfQg4MmIzFyQQHsTypJYrGoGoRzrMnKev7MSMFmVv5gf+wHmxYu2Yv72rD1wqPuxivAZ5ohqvt5X5c8S2l5Q/IonpojQBrgfixUMAE7Vh+VX7KFrCWzoerviQSg9sAzcfcCJJR9WGHMSmXRzrIanZzaZuqIh5zhlxhontI37qtEmmguMwRpka1ZHHzmnFnvKOp0uIoeIyAwRWbfW93qJyDgRKesZ1ap8CmwPXIClsHu4bcxJwBmxcjoAgV4DXIrVu9y1qJE5VU6NCkKDm5LdiKEhqlyhygGqrAWsha2Gr8KkXfbBtrrvAL+7xti7Rfi3CFsvJq0fOeW+1RqAfTAvBk4SkXOBI4FOqvpDpJYVALUxyteJ8AzWu4V01d+B1zUgmjHIdXM5FjS9CJhDKKfFzL4oqRG4z+u5dIWog9xXTVnGOtiqaGtsVXQwcKI7ZZYIH2MropHu62PVaFs3ytrxqKqKyIXAsyIyDrgQ2FVVvwAQkX2x2EMT4DpV7RudtY1HlTEw7yF7BFjjme77Dty3HZsQSufYxFQCVUL5LyZ0dQEwl1BOj00gPFpq6nsmFfKiLi443n0NgHnPSSvmb8+2AQ7D6sYAZtZyRqPccXQpnVHZL4dV9QVsj3slcLiqjgAQkWbYiOBdgHZAdxFZOTJDC4B7yLoB3/d746iTge1/mtoyFa1VC2FO5kJMwvVU4PZIpmfEj7Xc8bti38hVyH+lSn83dmhXLLDdGms6vgWbano4Nn5oJLZNGynCXW6btk0xt2ll73hEZBeslF1YcBnbARijqt+p6h9YcV7pxwUXGFU+BDo8/s5hVzz/0R665BJ/XXtml17bR23XApjzuYAFnU/ZP2t5sjYmffp9FDd3zmicKo85Z7Qb853R4Zgz+hXLpNUUOP4uwigXMzpVhA4ihRnPU9YPg4hsATyFzW36Hws2U67Jgn9dvmP+X52yRpWZc+c26Tnkoy4HtVhixtybjjrnLJi3xI4Hf3c+vavc+awLTMwmU7Gpc6rljB6v5YxWxqbbHoqFKX7G5q71Zn4A+0MR7hPhDCcdkvOQwrJ9EFwmazCQVtV7sYa93UWkc5R2lZKbnzt7YPNmsy5p2mTuQUN77P4v4C0RdovarnnMdz7XYM2V1ZztaoUVBsYa54yyqjzhGmT3wMo3WmFB6+swJcd9MD2jGumQMSI8JMLZIuxU37insgwui8hKwBDgaVW9HEBVR4tIf+wh3w775dRe4axFrYKtCuIG4Nk9rh06AxsPM1SEO4HuqvwRqWVQE3C2LJdlH5sRyomxCYiXjgTW11Z2uNjiN+7rSZi3ul4LK3rc2h13xXoKwQTWFvtHpmILCF1weSzWfDkFC6B1VNXJUdpVTN69osMG2/Z89xSs0/0bTGr11YjNMizA3BOr9XkY6FYt7RWJTHopbG76Jdlk6vKo7SkmIqyOOaENVOm1uNdV7LJXVWdjWsfDsGmf6Up2OoRybIf1R4zRh+V+TFx+Dvb+40GgSqCXYRmvrsDDhJL78MPyZH13/DJSK0qA0zF6ri6nAxXseABUdZCqbqiqrVW1T9T2FJlngKnAnfqwDMcyfceDSW+I0ClK4+ZhFc7dsbqS/oQSy8raArOhO34WqRUxoqIdT1UR6GRshdMROFGVabXkFS4HXhfhxkKlQ/Mi0BuxTOQBwFOEEr1NxaWNO34eqRUxwjueyuJB4BXgOsIFetVOxQrFUsAoETpEYNuCBHob1nfWBXiWUJaN2KJisinwbTaZ+j1qQ+KCdzyVhKWvT8U6l9vVfFuVP1Q5FSugXAZLux8QjZG1CLQvcAywE/B8BWs4b4bNfvc4vOOpNAL9FPgngQ5Z+EeqDMWG26Wx2eOIEG2AN9B+WLynPfByJU2vAEhk0ktgWy3veGrhHU8lEug0QhFCOdSJdM1DlSmqnKfK725Y3XsiXCZC84ishUAHYKNkNgVeIZQ16j6hrGiDDfL7MGpD4oR3PJVLe+BxoEcdr6n5QPQE3hVhi1IYtkgCfQ7YC6uQfZ1wvsZSmbOlO34QoQ2xwzueSiXQd4EQuIBQNlrUS1SZqsox2GpjdWCECBe7eeOlJ9BhMK9f6I3F2V1mbI0N8fs0akPihHc8lc05WMXsHXVJU6gyENvmDICIg86Bvo1VmzfHVj5bRmpP/mwDvJ9NpqqtRaROvOOpZAL9AWvS3Bk4uq6XqjJZlSOBXVSZLcLKIpzlxrOUlkA/BHYA/sJiPvEofswRF1jeCpOY8NTCO57K526gP9CgdhFVampNugI3A2+IUPotT6CfY5rTPwIvEMqeJbchfzYHlsLkJDy18I6n0gl0LoEeRqDP5njmrZjz2Qj4wMkdlPZ5CXQ8tvL5AniaUA4p6f3zZzt3fCtSK2KIdzzVQijNCeViQtmhIS93uiwhFvsZisnINmgsS0Gx7WJnTNLkMUI5se4TYkUnTJ4lHpNfY0TFymJ4FiKUZbA5TNOAdgTaYGFvp71yFPCSKhNFWAGYqkrpJkiEsjQW/O4CnE+g15fs3o0gkUkL8C3wZjaZOjxqe+KGX/FUC4FOw+azb0KOchlu9fOQczpNMLnZF0VoVXhDF4PNETsAeAzrRbsu5iLy62FCWa9FbUgc8Y6nmrA4zwCgJ6EkGnkVxYS8tgE+FuHkkmk92yqtKyZGfh7Qh1BKn3VrGDu74ytRGhFXvOOpPs4EZmHB45xxq5++WMbmHazrfYgIpZncapKpp2HjjE4EHl+4LSQm7IJl5D6J2pA44mM81UgohwJZAh2Zz2XctusUrCO+Y61UfGkI5UxsLMsw4EACnVrS+y8GF9+ZBLycTaaCqO2JI37FU40E2n+e08lj6oMqc1XpDWxZ03Qqwq0irFkoU+sk0F5YYeQOwDBCWbUk962ftsBqWDbQswi846lWrHv9LqhbG7chqFLTDrANcAIwRoSjShL7MVmNA4GNsf6uVkW/Z/3UFDu+EKkVMcY7nmrFRMP+Ak4nlG0LcUlV3sC0nscCDwFPliT2Y0Hz3bD5T8MJZfOi37Nu9gY+yiZTRR9XXK54x1Pd/BeLRdxJKAXpSFflC2zr0x2TuXigENetl0CHY9M1FGsubVChZKFJZNIrYIWDz0Vx/3LBO55qxoKxSUwzJlmoy6oyR5UbMfnVswBEWFGE4qoLBjoaE7v/HuvviqLTvgs2KPPpCO5dNnjH43kSeBboXui0tCpjVefp0NyCxX4OKuQ9/kag32DNpR8BT0bQYnEA8BO+MbROfDrdA6GsDQiBflusW4iwObbtaocVICZV+aVY93NTK/pjK5D/Ale5uFbRSGTSLbDanf7ZZKqcespKjl/xeCDQCQT6rct0rVX/CbmjysfAttgI48OB0SL8qxj3AiDQP4D9sSD3FcCtJahy3hVYHjdf3LN4vOPxGNb3dBvwVrFmXKkyS5XLgA7YcLvxxbjPPAKdBXQDbsT61B4tcpXzocAU4KUi3qMi8I7HY9g2pB+wDjZ5tGio8r4qnWuaTkXoJ0JxhL5Mj6g7JgN7CDCEUFYo9G0SmXRzrJ5oYDaZmlHo61ca3vF45hPoW1jv1ZmE0q6+lxeI1TB50CEi9BFh+aLcJdCbsQbTjsBrRdhSdgFWwLrnPfXgHY9nYXoAPwN3laLzW5VJmOO5Aat6/kiEXYpys0BDrLivFVZouHEhr4793nybRAPwjsezIIH+CpwNrAGUZLaVKn+pch6WBp8B9C3agMFAX8RGJrfAutzzxjWFLg08mk2mZhXimpWOT6d7PJ6S41c8Ho+n5HjH4/F4So53PB6Pp+R4x+PxeEqOdzwej6fkeMfj8XhKzv8D8wNmp14sny4AAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "visualize(x_0, delta)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
