{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "36a01ebd-1191-46b6-9b6e-4a9730e6ebb8",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "\n",
    "def subsets(arr,status,curr = 0):\n",
    "    global s_g\n",
    "    if(curr>=arr.shape[1]):\n",
    "        s_g.append(np.sum(arr*status, axis=1))\n",
    "        return\n",
    "    subsets(arr,status,curr+1)\n",
    "    status[curr] = 1\n",
    "    subsets(arr,status,curr+1)\n",
    "    status[curr] = 0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 161,
   "id": "143bdf66-076d-4f70-ad65-74c8e2494891",
   "metadata": {},
   "outputs": [
    {
     "ename": "IndexError",
     "evalue": "tuple index out of range",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mIndexError\u001b[0m                                Traceback (most recent call last)",
      "Cell \u001b[1;32mIn[161], line 3\u001b[0m\n\u001b[0;32m      1\u001b[0m rand_vars \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39mrandom\u001b[38;5;241m.\u001b[39muniform(\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m,\u001b[38;5;241m1\u001b[39m,\u001b[38;5;241m13\u001b[39m)\n\u001b[0;32m      2\u001b[0m s \u001b[38;5;241m=\u001b[39m []\n\u001b[1;32m----> 3\u001b[0m subsets(rand_vars, np\u001b[38;5;241m.\u001b[39mzeros_like(rand_vars))\n\u001b[0;32m      4\u001b[0m s \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39msort(s)\n",
      "Cell \u001b[1;32mIn[159], line 5\u001b[0m, in \u001b[0;36msubsets\u001b[1;34m(arr, status, curr)\u001b[0m\n\u001b[0;32m      3\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21msubsets\u001b[39m(arr,status,curr \u001b[38;5;241m=\u001b[39m \u001b[38;5;241m0\u001b[39m):\n\u001b[0;32m      4\u001b[0m     \u001b[38;5;28;01mglobal\u001b[39;00m s_g\n\u001b[1;32m----> 5\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m(curr\u001b[38;5;241m>\u001b[39m\u001b[38;5;241m=\u001b[39marr\u001b[38;5;241m.\u001b[39mshape[\u001b[38;5;241m1\u001b[39m]):\n\u001b[0;32m      6\u001b[0m         s_g\u001b[38;5;241m.\u001b[39mappend(np\u001b[38;5;241m.\u001b[39msum(arr\u001b[38;5;241m*\u001b[39mstatus, axis\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m1\u001b[39m))\n\u001b[0;32m      7\u001b[0m         \u001b[38;5;28;01mreturn\u001b[39;00m\n",
      "\u001b[1;31mIndexError\u001b[0m: tuple index out of range"
     ]
    }
   ],
   "source": [
    "rand_vars = np.random.uniform(-1,1,13)\n",
    "s = []\n",
    "subsets(rand_vars, np.zeros_like(rand_vars))\n",
    "s = np.sort(s)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "id": "1dbd3a18-17f5-4f87-a531-04aced010b84",
   "metadata": {},
   "outputs": [],
   "source": [
    "# VERIFICATION I: SSA sum exists close to each number in the range(-1,1) with precision 0.01\n",
    "epsilon = 0.01\n",
    "for i in range(1,200):\n",
    "    it = (i/100)-1\n",
    "    if(np.sum(((s-it)<=epsilon))==0):\n",
    "        print(i)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "id": "f6878abc-8b8d-4c31-8a49-fd8b955939b1",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Identifying indices for SSA sums which fall into the (-1,1) range\n",
    "start_ind = np.argwhere((s+1.0)>=epsilon)[0][0]\n",
    "end_ind = np.argwhere((1-s)>=epsilon)[-1][0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "id": "2ea61943-6487-4e81-9e76-5ce7200b0de6",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.00448854379462249\n"
     ]
    }
   ],
   "source": [
    "# Verification II: Difference between two consecutive sums is smaller than 2*epsilon\n",
    "s_short = s[start_ind:end_ind]\n",
    "err = 0\n",
    "for i in range(1,len(s_short)):\n",
    "    # print(i)\n",
    "    if(abs(s_short[i]-s_short[i-1])>err):\n",
    "        err = abs(s_short[i]-s_short[i-1])\n",
    "    assert(abs(s_short[i]-s_short[i-1])<=2*epsilon)\n",
    "print(err)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "c786a82c-a724-455a-ae55-c61bf5dc093f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Set size: 14\n",
      "0\n"
     ]
    },
    {
     "ename": "IndexError",
     "evalue": "tuple index out of range",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mIndexError\u001b[0m                                Traceback (most recent call last)",
      "Cell \u001b[1;32mIn[27], line 21\u001b[0m\n\u001b[0;32m     19\u001b[0m rand_vars \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39mrandom\u001b[38;5;241m.\u001b[39muniform(\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m,\u001b[38;5;241m1\u001b[39m,set_size)\n\u001b[0;32m     20\u001b[0m s \u001b[38;5;241m=\u001b[39m []\n\u001b[1;32m---> 21\u001b[0m subsets(rand_vars, np\u001b[38;5;241m.\u001b[39mzeros_like(rand_vars))\n\u001b[0;32m     22\u001b[0m s \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39msort(s)\n\u001b[0;32m     23\u001b[0m fail_1 \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n",
      "Cell \u001b[1;32mIn[27], line 3\u001b[0m, in \u001b[0;36msubsets\u001b[1;34m(arr, status, curr)\u001b[0m\n\u001b[0;32m      1\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21msubsets\u001b[39m(arr,status,curr \u001b[38;5;241m=\u001b[39m \u001b[38;5;241m0\u001b[39m):\n\u001b[0;32m      2\u001b[0m     \u001b[38;5;28;01mglobal\u001b[39;00m s_g\n\u001b[1;32m----> 3\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m(curr\u001b[38;5;241m>\u001b[39m\u001b[38;5;241m=\u001b[39marr\u001b[38;5;241m.\u001b[39mshape[\u001b[38;5;241m1\u001b[39m]):\n\u001b[0;32m      4\u001b[0m         s_g\u001b[38;5;241m.\u001b[39mappend(np\u001b[38;5;241m.\u001b[39msum(arr\u001b[38;5;241m*\u001b[39mstatus, axis\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m1\u001b[39m))\n\u001b[0;32m      5\u001b[0m         \u001b[38;5;28;01mreturn\u001b[39;00m\n",
      "\u001b[1;31mIndexError\u001b[0m: tuple index out of range"
     ]
    }
   ],
   "source": [
    "\n",
    "\n",
    "for set_size in reversed(range(10,15)):\n",
    "    fail_count_1 = 0\n",
    "    fail_count_2 = 0\n",
    "    epsilon = 0.01\n",
    "    print(\"Set size: \"+str(set_size))\n",
    "    for it in range(1000):\n",
    "        if(it%100 == 0):\n",
    "            print(it)\n",
    "        rand_vars = np.random.uniform(-1,1,set_size)\n",
    "        s = []\n",
    "        subsets(rand_vars, np.zeros_like(rand_vars))\n",
    "        s = np.sort(s)\n",
    "        fail_1 = False\n",
    "        for i in range(1,int(2/epsilon)):\n",
    "            it = (i*epsilon)-1\n",
    "            if(np.sum(((s-it)<=epsilon))==0):\n",
    "                fail_count_1+=1\n",
    "                fail_1 = True\n",
    "                break\n",
    "        if(fail_1):\n",
    "            continue\n",
    "        start_ind = np.argwhere((s+1.0)>=epsilon)[0][0]\n",
    "        end_ind = np.argwhere((1.0-s)>=epsilon)[-1][0]\n",
    "        s_short = s[start_ind:end_ind]\n",
    "        err = 0\n",
    "        fail_2 = False\n",
    "        for i in range(1,len(s_short)):\n",
    "            # print(i)\n",
    "            if(abs(s_short[i]-s_short[i-1])>err):\n",
    "                err = abs(s_short[i]-s_short[i-1])\n",
    "            if not (abs(s_short[i]-s_short[i-1])<=2*epsilon):\n",
    "                fail_count_2+=1\n",
    "                fail_2 = True\n",
    "                break\n",
    "        # print(err)\n",
    "    print(fail_count_1)\n",
    "    print(fail_count_2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 771,
   "id": "043bbc47-8bb7-4de3-8688-2cf5bf20de6f",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Set size: 10\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "40\n",
      "10\n"
     ]
    }
   ],
   "source": [
    "fail_count_1 = 0\n",
    "fail_count_2 = 0\n",
    "set_size = 10\n",
    "epsilon = 0.1\n",
    "print(\"Set size: \"+str(set_size))\n",
    "iterations = 1000\n",
    "rand_vars = np.random.uniform(-1,1,(iterations, set_size))\n",
    "s_g = []\n",
    "subsets(rand_vars, np.zeros_like(rand_vars[0]))\n",
    "s_g = np.array(s_g).T\n",
    "for it in range(iterations):\n",
    "    if(it%100 == 0):\n",
    "        print(it)\n",
    "    \n",
    "    #print(\"Finished\")\n",
    "    s = np.sort(s_g[it]).reshape(-1)\n",
    "    #print(\"Started_1\")\n",
    "    fail_1 = False\n",
    "    for i in range(1,int(2/epsilon)):\n",
    "        it = (i*epsilon)-1\n",
    "        if(np.sum(((s-it)<=epsilon))==0):\n",
    "            fail_count_1+=1\n",
    "            fail_1 = True\n",
    "            break\n",
    "    if(fail_1):\n",
    "        continue\n",
    "    #print(\"Finished_1\")\n",
    "    start_ind = np.argwhere((s+1.0)>=epsilon)[0][0]\n",
    "    end_ind = np.argwhere((1.0-s)>=epsilon)[-1][0]\n",
    "    #print(\"Started_2\")\n",
    "    if(np.sum(abs(s[start_ind+1:end_ind] - s[start_ind:end_ind-1])>=2*epsilon)>0):\n",
    "            fail_count_2+=1\n",
    "    \n",
    "print(fail_count_1)\n",
    "print(fail_count_2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 775,
   "id": "ed388e77-fd4d-4412-a6c0-92bdbe8d7842",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Set size: 5\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "314\n",
      "427\n",
      "Set size: 5\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "415\n",
      "585\n",
      "Set size: 5\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "423\n",
      "577\n",
      "Set size: 6\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "204\n",
      "263\n",
      "Set size: 6\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "284\n",
      "716\n",
      "Set size: 6\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "328\n",
      "672\n",
      "Set size: 7\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "119\n",
      "138\n",
      "Set size: 7\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "193\n",
      "807\n",
      "Set size: 7\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "196\n",
      "804\n",
      "Set size: 8\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "95\n",
      "67\n",
      "Set size: 8\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "145\n",
      "855\n",
      "Set size: 8\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "159\n",
      "841\n",
      "Set size: 9\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "45\n",
      "28\n",
      "Set size: 9\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "110\n",
      "882\n",
      "Set size: 9\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "98\n",
      "902\n",
      "Set size: 10\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "46\n",
      "21\n",
      "Set size: 10\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "51\n",
      "734\n",
      "Set size: 10\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "66\n",
      "934\n",
      "Set size: 11\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "19\n",
      "5\n",
      "Set size: 11\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "55\n",
      "447\n",
      "Set size: 11\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "58\n",
      "942\n",
      "Set size: 12\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "17\n",
      "3\n",
      "Set size: 12\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "23\n",
      "268\n",
      "Set size: 12\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "30\n",
      "970\n",
      "Set size: 13\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "13\n",
      "1\n",
      "Set size: 13\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "11\n",
      "134\n",
      "Set size: 13\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "23\n",
      "975\n",
      "Set size: 14\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "9\n",
      "3\n",
      "Set size: 14\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "6\n",
      "84\n",
      "Set size: 14\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "9\n",
      "833\n",
      "Set size: 15\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1\n",
      "0\n",
      "Set size: 15\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "10\n",
      "35\n",
      "Set size: 15\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "7\n",
      "460\n",
      "Set size: 16\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "2\n",
      "0\n",
      "Set size: 16\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "4\n",
      "19\n",
      "Set size: 16\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "2\n",
      "255\n",
      "Set size: 17\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "2\n",
      "0\n",
      "Set size: 17\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "3\n",
      "13\n",
      "Set size: 17\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "3\n",
      "158\n",
      "Set size: 18\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "2\n",
      "0\n",
      "Set size: 18\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "0\n",
      "4\n",
      "Set size: 18\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "0\n",
      "80\n",
      "Set size: 19\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1\n",
      "0\n",
      "Set size: 19\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "2\n",
      "8\n",
      "Set size: 19\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1\n",
      "49\n",
      "Set size: 20\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "0\n",
      "0\n",
      "Set size: 20\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "2\n",
      "2\n",
      "Set size: 20\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "0\n",
      "20\n",
      "Set size: 21\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "0\n",
      "0\n",
      "Set size: 21\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "0\n",
      "5\n",
      "Set size: 21\n",
      "0\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "0\n",
      "18\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "stats = {}\n",
    "iterations = 1000\n",
    "for set_size in range(5,22):\n",
    "    stats[set_size] = {}\n",
    "    for ep_exp in range(1,4):\n",
    "        fail_count_1 = 0\n",
    "        fail_count_2 = 0\n",
    "        epsilon = 10**(-ep_exp)\n",
    "        print(\"Set size: \"+str(set_size))\n",
    "        rand_vars = np.random.uniform(-1,1,(iterations, set_size))\n",
    "        s_g = []\n",
    "        subsets(rand_vars, np.zeros_like(rand_vars[0]))\n",
    "        s_g = np.array(s_g).T\n",
    "        for it in range(iterations):\n",
    "            if(it%100 == 0):\n",
    "                print(it)\n",
    "            \n",
    "            #print(\"Finished\")\n",
    "            s = np.sort(s_g[it]).reshape(-1)\n",
    "            #print(\"Started_1\")\n",
    "            fail_1 = False\n",
    "            for i in range(1,int(2/epsilon)):\n",
    "                it = (i*epsilon)-1\n",
    "                if(np.sum(((s-it)<=epsilon))==0):\n",
    "                    fail_count_1+=1\n",
    "                    fail_1 = True\n",
    "                    break\n",
    "            if(fail_1):\n",
    "                continue\n",
    "            #print(\"Finished_1\")\n",
    "            start_ind = np.argwhere((s+1.0)>=epsilon)[0][0]\n",
    "            end_ind = np.argwhere((1.0-s)>=epsilon)[-1][0]\n",
    "            #print(\"Started_2\")\n",
    "            if(np.sum(abs(s[start_ind+1:end_ind] - s[start_ind:end_ind-1])>=2*epsilon)>0):\n",
    "                    fail_count_2+=1\n",
    "            \n",
    "        print(fail_count_1)\n",
    "        print(fail_count_2)\n",
    "        stats[set_size][ep_exp] = [fail_count_1, fail_count_2]\n",
    "        torch.save(stats, \"SvsDelta.pth\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 777,
   "id": "08dcf8f6-cb54-457a-a6d7-83910193fef0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{5: {1: [314, 427], 2: [415, 585], 3: [423, 577]},\n",
       " 6: {1: [204, 263], 2: [284, 716], 3: [328, 672]},\n",
       " 7: {1: [119, 138], 2: [193, 807], 3: [196, 804]},\n",
       " 8: {1: [95, 67], 2: [145, 855], 3: [159, 841]},\n",
       " 9: {1: [45, 28], 2: [110, 882], 3: [98, 902]},\n",
       " 10: {1: [46, 21], 2: [51, 734], 3: [66, 934]},\n",
       " 11: {1: [19, 5], 2: [55, 447], 3: [58, 942]},\n",
       " 12: {1: [17, 3], 2: [23, 268], 3: [30, 970]},\n",
       " 13: {1: [13, 1], 2: [11, 134], 3: [23, 975]},\n",
       " 14: {1: [9, 3], 2: [6, 84], 3: [9, 833]},\n",
       " 15: {1: [1, 0], 2: [10, 35], 3: [7, 460]},\n",
       " 16: {1: [2, 0], 2: [4, 19], 3: [2, 255]},\n",
       " 17: {1: [2, 0], 2: [3, 13], 3: [3, 158]},\n",
       " 18: {1: [2, 0], 2: [0, 4], 3: [0, 80]},\n",
       " 19: {1: [1, 0], 2: [2, 8], 3: [1, 49]},\n",
       " 20: {1: [0, 0], 2: [2, 2], 3: [0, 20]},\n",
       " 21: {1: [0, 0], 2: [0, 5], 3: [0, 18]}}"
      ]
     },
     "execution_count": 777,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "stats"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "48e50647-702e-42ac-aed0-34351716d682",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[[0.741 0.467 0.257 0.162 0.073 0.067 0.024 0.02  0.014 0.012 0.001 0.002\n",
      "  0.002 0.002 0.001 0.    0.   ]\n",
      " [1.    1.    1.    1.    0.992 0.785 0.502 0.291 0.145 0.09  0.045 0.023\n",
      "  0.016 0.004 0.01  0.004 0.005]\n",
      " [1.    1.    1.    1.    1.    1.    1.    1.    0.998 0.842 0.467 0.257\n",
      "  0.161 0.08  0.05  0.02  0.018]]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "C:\\Users\\RahulN\\AppData\\Local\\Temp\\ipykernel_7728\\1171653098.py:5: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  stats = torch.load(\"SvsDelta.pth\")\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "stats = torch.load(\"SvsDelta.pth\")\n",
    "deltas = np.zeros((3, 17))\n",
    "for i in range(1,4):\n",
    "    for j in range(5,22):\n",
    "        deltas[i-1][j-5] = np.sum(stats[j][i])/1000\n",
    "print(deltas)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "4cc6f546-2cd1-453c-a595-1210aef8a2d1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0.5, 1.0, 'Scaling laws of Subset Sum Approximation')"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjcAAAHFCAYAAAAOmtghAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8fJSN1AAAACXBIWXMAAA9hAAAPYQGoP6dpAACJbElEQVR4nO3dd3yT1f7A8U/SJN0ttKWlhQ7KLLNQhiwB2SAoehVRGSpe8aoIiFeR6+L+FMWFC3AADsbFAaiAQJEtiKyyZRbKaCkFuunM+f0RGindIzxN+32/Xnn1yZMnOd+kTfLtc77nHJ1SSiGEEEIIUU3otQ5ACCGEEKIySXIjhBBCiGpFkhshhBBCVCuS3AghhBCiWpHkRgghhBDViiQ3QgghhKhWJLkRQgghRLUiyY0QQgghqhVJboQQQghRrUhyIwrYsWMHw4YNIygoCEdHR/z8/OjcuTPPPfeczdr86quv0Ol0nD592rpvzJgxhISE2KzN4mjZtq1cuXKFBx54AF9fX3Q6HXfffXeRx2ZnZ/PZZ5/RoUMHvLy8cHFxITg4mLvuuotly5aVq/2ePXvSsmXLckZfPunp6bz22mts3Lix1Pc5e/Ys//rXv2jSpAnOzs54eXnRqlUrHn/8cc6ePWu7YCtg0qRJ6HQ67rzzTq1DsYmQkBDGjBmjSduLFi1i5syZhd6m0+l47bXXbmk8onQMWgcgqpaVK1cydOhQevbsyYwZM/D39yc2NpZdu3bxv//9j/fee++WxfLyyy/z7LPP3rL2qrv//ve/LFu2jHnz5tGwYUO8vLyKPHbkyJEsXbqUCRMm8Prrr+Po6MipU6dYvXo1a9asYdiwYbcw8vJLT0/n9ddfByzJVUnOnTtHu3btqFWrFs899xxNmzYlKSmJw4cP891333Hq1CkCAwNtHHXZZGdns2DBAgBWr17N+fPnqVevnsZRVa5ly5bh4eGhSduLFi3i4MGDTJgwocBt27dvp379+rc+KFEyJcQNbr/9dtWwYUOVnZ1d4Lbc3FybtTt//nwFqOjoaJu1URajR49WwcHBWodRqfr06aPCwsJKPO7UqVMKUK+88kqht5f376BHjx6qRYsW5bpveV26dEkB6tVXXy3V8a+88ooC1KlTpwq93ZbvgfL6/vvvFaAGDx6sAPXGG29oFovZbFbp6ematW8LgwcPrnafBTWBdEuJfC5fvoyPjw8GQ8GTenp9wT+XRYsW0blzZ9zc3HBzcyM8PJy5c+dab4+MjOSuu+6ifv36ODk50ahRI5544gkSEhJKjKWwriGdTsfTTz/Nt99+S1hYGC4uLrRp04YVK1YUuP9PP/1E69atcXR0JDQ0lA8//JDXXnsNnU5XileioE8//ZTbb78dX19fXF1dadWqFTNmzCA7OzvfMXq9nvj4eOu+9957D51Ox1NPPWXdZzabqV27dr6uvtmzZ9OmTRvc3Nxwd3enWbNmvPTSSyXGdeXKFf71r39Rr149TCYToaGhTJ06lczMTABOnz6NTqdj3bp1HDlyBJ1Oh06nK7Kr5vLlywD4+/sXevuNfweFdScCbNy4scg2tmzZwm233YazszP16tXj5ZdfJjc3N98xpXkt4uLieOKJJ6hfvz4mk4kGDRrw+uuvk5OTY33ederUAeD111+3Pu/iujcuX76MXq/H19e3xOfes2fPQs8G3fx3m/f6v/POO7z99tuEhITg7OxMz549OXbsGNnZ2bz44osEBATg6enJsGHD8v39lGTu3LmYTCbmz59PYGAg8+fPR920HnLe72PBggVMmjSJunXr4uzsTI8ePdi7d2+B+N3c3Dh06BC9e/fG1dWVOnXq8PTTT5Oenp7v2Lz345w5cwgLC8PR0ZGvv/4agK1bt9K7d2/c3d1xcXGhS5curFy50nrfrVu3YjQamTx5cr7HzPubuvFz5OZuqbzns2jRIl544QX8/f1xc3NjyJAhXLx4kZSUFP75z3/i4+ODj48PjzzyCKmpqfnaKc37uWfPnqxcuZIzZ85Y/35u/PworFvq4MGD3HXXXdSuXRsnJyfCw8Otr8nN8S9evJipU6cSEBCAh4cHffr04ejRo4hKoHV2JaqWsWPHKkA988wz6o8//lBZWVlFHvvyyy8rQN1zzz3q+++/V2vXrlXvv/++evnll63HzJ49W02fPl39/PPPatOmTerrr79Wbdq0UU2bNs332IWduSns7AmgQkJCVMeOHdV3332nVq1apXr27KkMBoM6efKk9bhff/1V6fV61bNnT7Vs2TL1/fffq06dOqmQkBBVmj/7wtqeOHGimj17tlq9erVav369+uCDD5SPj4965JFHrMf89ddfClCLFi2y7hswYIBydnZWjRs3tu7bsWOHAtSqVauUUkotXrzY+rqvXbtW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7Ny5M18yBrB48eJ8ifCZM2fYtm1boaO+Fi5cmO/6okWLgJLnC3J1daVTp04sXbo03+tuNptZsGAB9evXp0mTJoDl9X744Yfp0aMH27ZtY+jQoTz22GNER0eX5eUos7K8n288K1yS3r17s379emsyk+ebb77BxcWF2267rQJRi7KQSfxEPv3796d+/foMGTKEZs2aYTabiYqK4r333sPNzc06qV5ISAgvvfQS//3vf7l27RojRozA09OTw4cPk5CQwOuvv06zZs1o2LAhL774IkopvLy8+OWXX0rsaqkM06ZNY/DgwfTv359nn32W3Nxc3nnnHdzc3Lhy5UqZH69v376YTCZGjBjBv//9bzIyMpg9ezZXr14t9PjevXvz8ccf4+zsTNeuXQFLPUSDBg1Yu3YtQ4cOzTfc/vHHH7ce6+/vT1xcHNOnT8fT07PIJAhg1KhRfPrpp4wePZrTp0/TqlUrtm7dyptvvsmgQYMKnB0qjaNHj9K/f38eeOABevTogb+/P1evXmXlypV8/vnn9OzZky5dugDQoUMHmjZtyuTJk8nJyaF27dosW7aMrVu3FvrY3t7ePPnkk8TExNCkSRNWrVrFF198wZNPPklQUFCpX4tp06YRGRlJly5dGD9+PE2bNiUjI4PTp0+zatUq5syZQ/369XF3dyc4OJiffvqJ3r174+XlhY+PT5GzT7/xxhv8/vvvDB8+nPDwcJydnYmOjuaTTz7h8uXLvPPOO9ZjR44cyWeffcbDDz/M448/zuXLl5kxY8YtmWxu4cKFZGRkMH78+EKTDW9vbxYuXMjcuXP54IMPrPvj4+MZNmwYjz/+OElJSbz66qs4OTkxZcqUfPc3mUy89957pKam0qFDB7Zt28b//d//MXDgQLp161ZifNOnT6dv37706tWLyZMnYzKZmDVrFgcPHmTx4sXodDpyc3MZMWKEdUi3g4MDX331FeHh4QwfPpytW7dW+ExKUcryfm7VqhVLly5l9uzZREREoNfrad++faGP++qrr1rrwV555RW8vLxYuHAhK1euZMaMGXh6etrk+YhCaFXJLKqmJUuWqAcffFA1btxYubm5KaPRqIKCgtTIkSPV4cOHCxz/zTffqA4dOignJyfl5uam2rZtq+bPn2+9/fDhw6pv377K3d1d1a5dW913330qJiamwCiDsoyWunHYcJ6bR1MopdSyZctUq1atrMNa33rrLTV+/HhVu3btEl+Hwtr+5ZdfVJs2bZSTk5OqV6+eev7559Wvv/5aYFSQUkr99NNPClB9+/bNt//xxx9XgProo4/y7f/6669Vr169lJ+fnzKZTCogIEDdf//9av/+/SXGevnyZTVu3Djl7++vDAaDCg4OVlOmTFEZGRn5jivtaKmrV6+q//u//1N33HGHqlevnjKZTMrV1VWFh4er//u//yswSduxY8dUv379lIeHh6pTp4565pln1MqVKwsdLdWiRQu1ceNG1b59e+Xo6Kj8/f3VSy+9lG/SyNK+FpcuXVLjx49XDRo0UEajUXl5eamIiAg1derUfKNt1q1bp9q2bascHR0VUODv5EZ//PGHeuqpp1SbNm2Ul5eXcnBwUHXq1FEDBgywDtu/0ddff63CwsKUk5OTat68uVqyZEmRo6XeeeedfPfNGzHz/fff59uf917YuXNnkXGGh4crX1/ffKMBb3bbbbcpHx8flZmZaW3r22+/VePHj1d16tRRjo6Oqnv37mrXrl357jd69Gjl6uqq9u/fr3r27KmcnZ2Vl5eXevLJJwuMYirq/aiUUlu2bFF33HGHcnV1Vc7Ozuq2225Tv/zyi/X2qVOnKr1er3777bd899u2bZsyGAzq2Wefte4rarRUaV+7V199VQHq0qVL1n2lfT9fuXJF/eMf/1C1atVSOp0u32jLmz/HlFLqwIEDasiQIcrT01OZTCbVpk2bfJ+JxcWf97dy8/Gi7HRK3dRZL0Q1lZ2dTXh4OPXq1WPt2rVahyPELbNx40Z69erF999/X2Bk3c3GjBnDDz/8UGTXpBD2QLqlRLX12GOP0bdvX2vXxpw5czhy5Agffvih1qEJIYSwIUluRLWVkpLC5MmTuXTpEkajkXbt2rFq1apy1aEIIYSwH9ItJYQQQohqRYaCCyGEEKJakeRGCCGEENWKJDdCCCGEqFZqXEGx2WzmwoULuLu7F5g2XAghhBBVk1KKlJQUAgIC8i1CW5gal9xcuHChwIqtQgghhLAPZ8+eLXbRZqiByY27uztgeXFuxTTpQgghhKi45ORkAgMDrd/jxalxyU1eV5SHh4ckN0IIIYSdKU1JiRQUCyGEEKJakeRGCCGEENWKJDdCCCGEqFZqXM2NEEKI6iU3N5fs7GytwxCVwGg04uDgUOHHkeRGCCGE3UpNTeXcuXPIMonVg06no379+ri5uVXocSS5EUIIYZdyc3M5d+4cLi4u1KlTRyZmtXNKKS5dusS5c+do3Lhxhc7gSHIjhBDCLmVnZ6OUok6dOjg7O2sdjqgEderU4fTp02RnZ1couZGCYiGEEHZNzthUH5X1u5TkRgghhBDViiQ3QgghhKhWNE1uNm/ezJAhQwgICECn07F8+fIS77Np0yYiIiJwcnIiNDSUOXPm2D5QIYQQQtgNTZObtLQ02rRpwyeffFKq46Ojoxk0aBDdu3dn7969vPTSS4wfP54ff/zRxpEKIYQQ9mvWrFk0aNAAJycnIiIi2LJlS7HHx8bG8uCDD9K0aVP0ej0TJky4NYFWEk1HSw0cOJCBAweW+vg5c+YQFBTEzJkzAQgLC2PXrl28++673HvvvYXeJzMzk8zMTOv15OTkCsVclJTk87y9YrRNHrtG0wEOjmB0AaMTGJzB6GzZ1hu1jk6Ugk6nw6g3Wi8mB5Nl28Fy3aA3FLzthtuLuu3G+xkdjBh0BiksFaIQS5YsYcKECcyaNYuuXbvy2WefMXDgQA4fPkxQUFCh98nMzKROnTpMnTqVDz744BZHXHF2NRR8+/bt9OvXL9++/v37M3fuXLKzszEaC37ZTZ8+nddff93msWVmJvNT9kWbt1NjXdM6AFHV6bAkUR6OHnzQ8wPCfcO1DkncYkoprmXnatK2s9Gh1Mn1H3/8wXPPPcf+/fsxm82EhoYyb948OnToYJPY3n//fR577DHGjh0LwMyZM1mzZg2zZ89m+vTphd4nJCSEDz/8EIB58+bZJC5bsqvkJi4uDj8/v3z7/Pz8yMnJISEhAX9//wL3mTJlCpMmTbJeT05OJjAwsNJjc3H2ZoJX+0p/3BpPmSErHbJSIDMVMlMgKxVyM0u+r5UOTK5gcgdHV3B0B5Pb39dN7pZ9jq6Ws0SiUikU2eZssnOzLT9v3i7n9SxzVoF2ssxZJFxLYPqf01k8eDF6nYyZqEmuZefS/JU1mrR9eFp/XEyl+0odMWIEXbt25csvv8TJyYlTp04V+G670bhx41iwYEHx7RdxFiYrK4vdu3fz4osv5tvfr18/tm3bVqp47ZFdJTdQcAx83pTbRWXMjo6OODra/gvLxc2Xx4bMt3k74rqsNEi9CKnxkBJn+Zl6EVLj8u9Li7ckSCQDsSU/rskder8Cnf5p62cgKkgpRY7KyZf4JGUm8dCqhzh8+TBrT69lQIMBWocpRAG5ubkEBQXRqFEjjEYjDRo0KPb4adOmMXny5GKPCQgIKHR/QkICubm5hZ4YiIuLK1vgdsSukpu6desW+GXEx8djMBjw9vbWKCqhCZMreIVaLsUx50L6ZUvik3LxegJ0w8W6L95ydigrBTa+Ce1GWep6RJWl0+kw6iw1N3l8nH0Y02IMn0Z9ykd7P6J3UG+MDlKbVVM4Gx04PK2/Zm2X1tKlSxk2bBgzZszAycmJ8+fP4+npWeTxvr6++Pr6Vii+wk4MVOcaNbtKbjp37swvv/ySb9/atWtp3759ofU2QqB3ADdfy6Vuq+KPzUyFWbdB0lk4/BO0GX5rYhSValTzUfzvr/9xNuUsPx7/kQeaPaB1SOIW0el0pe4a0tKUKVNo164dU6dOxcvLC3d392KPr0i3lI+PDw4ODoWeGCiuK8zeadohnZqaSlRUFFFRUYBlqHdUVBQxMTGA5Q9g1KhR1uPHjRvHmTNnmDRpEkeOHGHevHnMnTu3xNN1QpSKoxtEXB/xtsv+CuiEhYvRhXFtxgEwe99s0rPTNY5IiL8lJCSwbt06pk2bRseOHWnUqBF6ffFfxdOmTbN+VxZ1KapbymQyERERQWRkZL79kZGRdOnSpdKeV1WjaYq7a9cuevXqZb2eV/g7evRovvrqK2JjY62JDkCDBg1YtWoVEydO5NNPPyUgIICPPvqoyGHgQpRZ25Gw8S04+wdcPAR+LbSOSJTDvU3u5dvD3xKTEsPXh7/myTZPah2SEIDlTEpgYCCvvPIKr7zyCj4+PkRHR5OVlVVgNHCeinZLTZo0iZEjR9K+fXs6d+7M559/TkxMDOPGjbMeM2XKFM6fP88333xj3Zd34iE1NZVLly4RFRWFyWSiefPm5Y7lVtGpvIrcGiI5ORlPT0+SkpLw8PDQOhxRFS0ZCUd+hg6Pw+B3tY5GlNPq06t5ftPzuBhc+PXeX/Fy8tI6JFHJMjIyiI6Otk5OZy8OHTrEiy++yPbt20lJSSEoKIhXXnmFkSNH2qzNWbNmMWPGDGJjY2nZsiUffPABt99+u/X2MWPGcPr0aTZu3GjdV1hNTnBwMKdPn7ZZnMX9Tsvy/S3JjRA3O7kBvr0bHD3gub8sxcvC7piVmRErR3D48mEeCnuIFzu+WPKdhF2x1+RGFK2ykhuZBEKImzXoYRmFlZkMB2VpD3ul1+mZGDERgCVHl3A25azGEQkhbhVJboS4mV4PEY9YtqWw2K7d5n8bnf07k2PO4dOoT7UORwhxi0hyI0Rhwh8CBxNc2Gu5CLs1IWICACtPreTI5SPaBiOEuCUkuRGiMK7e0Pwuy/YumXnanjX3bs7ABpYFej/c86HG0QghbgVJboQoSvtHLT8P/AAZSdrGIirkmfBnMOgM/H7hd3bE7tA6HCGEjUlyI0RRgjqDT1PIToP932kdjaiAQI9A7mt6HwAf7P6AGjZIVIgaR5IbIYqi0/199mbXfJAvRLv2ROsncDG4cOjyIdaeWat1OEIIG5LkRojitBkOBmeIPwTndmodjagAb2dvRrewLK/x8d6PyTZnaxyREMJWJLkRojjOtaHl9eU9ZFi43RvdYjReTl6cST7DsuPLtA5HCGEjktwIUZL21+e8ObgU0q9oG4uoEFejK/9s/U9AFtUUojqT5EaIktSLgLqtIDcT9v1P62hEBd3f5H7qudUj4VoCC44s0DocIW6JWbNmWZc0iIiIYMuWLSXeZ9OmTURERODk5ERoaChz5szJd/uhQ4e49957CQkJQafTMXPmTBtFX3aS3AhRknyFxfOksNjOGR2MjG87HoB5B+dxNeOqxhEJYVtLlixhwoQJTJ06lb1799K9e3cGDhxITExMkfeJjo5m0KBBdO/enb179/LSSy8xfvx4fvzx7yVp0tPTCQ0N5a233qJu3bq34qmUmiQ3QpRGq/vA5AaXj8PprVpHIypoQIMBhHmFkZadxhcHvtA6HFFZlIKsNG0uZfin548//qBr1664u7vj6upKq1at2LnTdgMW3n//fR577DHGjh1LWFgYM2fOJDAwkNmzZxd5nzlz5hAUFMTMmTMJCwtj7NixPProo7z77rvWYzp06MA777zDAw88gKOjo83iLw+D1gEIYRcc3S0Jzu75lrM3DbprHZGoAL1Oz4R2E3hi3RP876//8VDYQ9Rzq6d1WKKistPhzQBt2n7pAphcS3XoiBEj6Nq1K19++SVOTk6cOnUKPz+/Io8fN24cCxYU34V6+PBhgoKCCuzPyspi9+7dvPjii/n29+vXj23bthX5eNu3b6dfv3759vXv35+5c+eSnZ2N0WgsNh6tSXIjRGm1f8SS3Bz5BVIvgVsdrSMSFdA5oDOd/DuxI3YHn+79lDe7v6l1SKKGyM3NJSgoiEaNGmE0GmnQoEGxx0+bNo3JkycXe0xAQOFJXUJCArm5uQWSJz8/P+Li4op8vLi4uELvk5OTQ0JCAv7+/sXGozVJboQoLf82UK89nN8FUQug20StIxIVoNPpmNhuIg+sfIAVp1YwusVomno11TosURFGF8sZFK3aLqWlS5cybNgwZsyYgZOTE+fPn8fT07PI4319ffH19a1QeDqdLt91pVSBfaW5T2H7qyKpuRGiLPKGhe+aD2aztrGICmvh04L+If1RKFlUszrQ6SxdQ1pcyvCFP2XKFNq1a8e2bduIiorC3d292OPHjRuHm5tbsZeiioN9fHxwcHAocJYmPj6+2K6wunXrFnofg8GAt7d3KZ+pduTMjRBl0eIeWP0SJJ6BU+uhUR+tIxIV9EzbZ/jtzG9sOb+FnXE76VC3g9YhiWosISGBdevWERUVRZs2bUp1n4p0S5lMJiIiIoiMjGTYsGHW/ZGRkdx1111FPl7nzp355Zdf8u1bu3Yt7du3r/L1NiBnboQoG5MLhI+wbO+ar20solIEewRzbxPLLNQzd8+URTWFTfn4+BAYGMgrr7zC7t27OXPmDBs3bmTt2qLXO/P19aVRo0bFXgyGos9VTJo0iS+//JJ58+Zx5MgRJk6cSExMDOPGjbMeM2XKFEaNGmW9Pm7cOM6cOcOkSZM4cuQI8+bNY+7cufmSrKysLKKiooiKiiIrK4vz588TFRXFiRMnKvgqVZwkN0KUVcT1rqmjv0KyRv37olKNazMOZ4Mz+xP281vMb1qHI6q5X3/9FbPZTP/+/WnSpAmPP/44Fy9etFl7w4cPZ+bMmUybNo3w8HA2b97MqlWrCA4Oth4TGxubr2urQYMGrFq1io0bNxIeHs5///tfPvroI+69917rMRcuXKBt27a0bduW2NhY3n33Xdq2bcvYsWNt9lxKS6dq2L8pycnJeHp6kpSUhIeHh9bhCHs1byDEbIOeL0HPF7SORlSCj/d+zOf7PyfEI4Rldy3DoJde+6ouIyOD6Oho68y7wv4V9zsty/e3nLkRojzyZize8zXk5mgbi6gUj7R4hFqOtTidfJrlJ5ZrHY4QogIkuRGiPJoPBRdvSD4PJyK1jkZUAjeTm3VRzVlRs7iWc03jiIQQ5SXJjRDlYXCE8Act27vmaRuLqDTDmw4nwDWAS9cusfDIQq3DEUKUkyQ3QpRXXmHx8Ui4ekbbWESlMDmYeLrt0wDMOzCPxIxEbQMSQpSLJDdClJd3QwjtCShL7Y2oFgaHDqZJ7SakZKfw5YEvtQ5HCFEOktwIURF5Z2/2fAu52drGIipF3qKaAIv+WkRsaqy2AQkhykySGyEqotlgcPODtHj4a6XW0YhK0q1eNzrU7UC2OZtPoz7VOhwhRBlJciNERTgYoe1Iy7YUFlcbOp3Oevbm55M/c+zqMW0DEkKUiSQ3QlRUxGhAB9Gb4PJJraMRlaR1ndb0De6LQvHRno+0DkcIUQaS3AhRUbWCoHFfy/ZuWW+qOnmm7TM46BzYdG4Tuy/u1jocIUQpSXIjRGXIm7F470LIztA2FlFpGng2YFhjy0rKH+z+QBbVFHZr1qxZ1iUNIiIi2LJlS4n32bRpExERETg5OREaGsqcOXMKHPPjjz/SvHlzHB0dad68OcuWLct3++bNmxkyZAgBAQHodDqWL19eWU+pWJLcCFEZGvcDj/pw7Qoc+VnraEQlerLNkzg5OLHv0j7Wn12vdThClNmSJUuYMGECU6dOZe/evXTv3p2BAwfmWyjzZtHR0QwaNIju3buzd+9eXnrpJcaPH8+PP/5oPWb79u0MHz6ckSNHsm/fPkaOHMn999/Pjh07rMekpaXRpk0bPvnkE5s+x5vJwplCVJaNb8PGNyGoCzz6q9bRiEr04Z4P+fLAl4R6hvLj0B9lUc0q4uZFFpVSmi2b4WxwRqfTlerYP/74g+eee479+/djNpsJDQ1l3rx5dOjQwSaxderUiXbt2jF79mzrvrCwMO6++26mT59e6H1eeOEFfv75Z44cOWLdN27cOPbt28f27dsBy2rjycnJ/Prr3593AwYMoHbt2ixevLjAY+p0OpYtW8bdd99dZKyVtXCmvEOFqCztRsKmty2rhccfAd8wrSMSleSRlo/w/bHvOZV0ip9P/sw9je/ROiRRiGs51+i0qJMmbe94cAcuRpdSHTtixAi6du3Kl19+iZOTE6dOncLPz6/I48eNG8eCBQuKfczDhw8TFBRUYH9WVha7d+/mxRdfzLe/X79+bNu2rcjH2759O/369cu3r3///sydO5fs7GyMRiPbt29n4sSJBY6ZOXNmsbHeCpLcCFFZPAKg6UD4awXsmg+DZmgdkagkHiYPHm/1OO/uepdPoz5lUINBOBmcSr6jEIXIzc0lKCiIRo0aYTQaadCgQbHHT5s2jcmTJxd7TEBAQKH7ExISyM3NLZA8+fn5ERcXV+TjxcXFFXqfnJwcEhIS8Pf3L/KY4h73VpHkRojK1P4RS3Kz73/Q5zUwle4/OVH1PdDsARYcWUBcWhyL/lrEoy0f1TokcRNngzM7HtxR8oE2aru0li5dyrBhw5gxYwZOTk6cP38eT0/PIo/39fXF19e3QvHd3GWmlCqxG62w+9y8vzyPeytIQbEQlSn0DqgVDJlJcGip1tGISuTo4MjT4ZZFNb888CVJmUkaRyRuptPpcDG6aHIpyxf6lClTaNeuHdu2bSMqKgp3d/dijx83bhxubm7FXooqDvbx8cHBwaHA2ZT4+Phiu8Lq1q1b6H0MBgPe3t7FHlPc494qktwIUZn0esvZG5AZi6uhO0PvpFGtRqRkpTD34FytwxF2KCEhgXXr1jFt2jQ6duxIo0aN0OuL/yqeNm0aUVFRxV6K6pYymUxEREQQGRmZb39kZCRdunQpss3OnTsXuM/atWtp3749RqOx2GOKe9xbRbqlhKhs4Q/D+jfg/G64EAUB4VpHJCqJg96BCe0m8PT6p1l4eCEPNnuQuq51tQ5L2BEfHx8CAwN55ZVXeOWVV/Dx8SE6OpqsrKwCBbx5KtotNWnSJEaOHEn79u3p3Lkzn3/+OTExMYwbN856zJQpUzh//jzffPMNYDlb9MknnzBp0iQef/xxtm/fzty5c/ONgnr22We5/fbbefvtt7nrrrv46aefWLduHVu3brUek5qayokTJ6zXo6OjiYqKwsvLq9AC6EqjapikpCQFqKSkJK1DEdXZd2OUetVDqZ/Hax2JqGRms1mNWjVKtfyqpXp568tah1OjXbt2TR0+fFhdu3ZN61DK5ODBg+rOO+9U3t7eymQyqUaNGqlvvvnGpm1++umnKjg4WJlMJtWuXTu1adOmfLePHj1a9ejRI9++jRs3qrZt2yqTyaRCQkLU7NmzCzzu999/r5o2baqMRqNq1qyZ+vHHH/PdvmHDBgUUuIwePbrQOIv7nZbl+1vmuRHCFqK3wNd3gskNnvsLHIvvUxf2JSo+ipG/jkSv07N06FIa1mqodUg1UnFzogj7VFnz3EjNjRC2ENINvBtDVirs/07raEQlC/cN547AOzArMx/u+VDrcIQQN5HkRghb0OluKCyeDzXrBGmN8Gy7Z9Hr9Gw4u4G98Xu1DkcIcQNJboSwlTYjwMERLh6Ac7u0jkZUstBaodzd6G4A/vfX/7QNRgiRjyQ3QtiKixe0vD5N/+752sYibGJAyADAUoMjhKg6JLkRwpbaX5/F9uCPcO2qtrGIStfKpxU6dFxIu8Cl9Etah1Nj1bBxMdVaZf0uJbkRwpbqdwDfFpCTYVmSQVQrbiY3GtVuBMC+S/s0jqbmcXBwACyLQ4rqIe93mfe7LS+ZxE8IW8orLF412VJY3GmcZZ+oNtrUacPxq8fZd2kffYL7aB1OjWIwGHBxceHSpUsYjcYSZ/oVVZvZbObSpUu4uLhgMFQsPZHkRghbaz0cIl+FhKNwZhuEdNU6IlGJwuuE88OxH6TuRgM6nQ5/f3+io6M5c+aM1uGISqDX6wkKCqrw4puS3Ahha04e0Ope2PONZb0pSW6qlTZ12gBw+PJhsnOzMToYNY6oZjGZTDRu3Fi6pqoJk8lUKWfgJLkR4lZo/6gluTn8E6S9Da4+WkckKkmwRzC1HGuRmJnIkStHaF2ntdYh1Th6vV5mKBb5SAelELdCQFvLxZwNUQu1jkZUIp1OZz17I0XFQlQNktwIcavkDQvfNR/MZm1jEZUqL7mRuhshqgZJboS4VVreC44ecDUaojdqHY2oRHLmRoiqRfPkZtasWdbVPyMiItiyZUuxxy9cuJA2bdrg4uKCv78/jzzyCJcvX75F0QpRASZXy8gpsBQWi2qjpU9L9Do9F9MvEpcWp3U4QtR4miY3S5YsYcKECUydOpW9e/fSvXt3Bg4cSExMTKHHb926lVGjRvHYY49x6NAhvv/+e3bu3MnYsWNvceRClFPeYpp/rYLkWG1jEZXGxehC09pNATl7I0RVoGly8/777/PYY48xduxYwsLCmDlzJoGBgcyePbvQ4//44w9CQkIYP348DRo0oFu3bjzxxBPs2iWLEgo74dcCAjuByoW9C7SORlSivFFSUncjhPY0S26ysrLYvXs3/fr1y7e/X79+bNu2rdD7dOnShXPnzrFq1SqUUly8eJEffviBwYMHF9lOZmYmycnJ+S5CaCqvsHj3V2DO1TQUUXny6m72X9qvcSRCCM2Sm4SEBHJzc/Hz88u338/Pj7i4wvusu3TpwsKFCxk+fDgmk4m6detSq1YtPv744yLbmT59Op6entZLYGBgpT4PIcqs+V3gXBuSz8HxSK2jEZUkvE44AIevHCYzN1PbYISo4TQvKL55imWlVJHTLh8+fJjx48fzyiuvsHv3blavXk10dDTjxo0r8vGnTJlCUlKS9XL27NlKjV+IMjM6Q/hDlu3d87WNRVSa+u718XLyIsecw+HLh7UOR4gaTbPkxsfHBwcHhwJnaeLj4wuczckzffp0unbtyvPPP0/r1q3p378/s2bNYt68ecTGFl6c6ejoiIeHR76LEJqLGGP5eWwNJBZeQC/sS77J/OKlqFgILWmW3JhMJiIiIoiMzH9aPjIyki5duhR6n/T09AJrTuQti66Usk2gQtiCT2MI6Q4oy7IMolqQ+W6EqBo07ZaaNGkSX375JfPmzePIkSNMnDiRmJgYazfTlClTGDVqlPX4IUOGsHTpUmbPns2pU6f4/fffGT9+PB07diQgIECrpyFE+eQVFu9dCJKcVwvWmYovRck/XEJoSNOFM4cPH87ly5eZNm0asbGxtGzZklWrVhEcHAxAbGxsvjlvxowZQ0pKCp988gnPPfcctWrV4o477uDtt9/W6ikIUX5NB4HRBVIuQNx+8G+jdUSiglr4tMCgM5BwLYELaReo51ZP65CEqJF0qob9e5GcnIynpydJSUlSfyO0t/hBOLoSev0HejyvdTSiEjyw4gEOXT7E293fZlDoIK3DEaLaKMv3t+ajpYSo0Zr0t/w8tlrbOESlkbobIbQnyY0QWmp8fRLL87shNV7bWESlkORGCO1JciOEljz8r9faKJnQr5oI9w0H4OiVo1zLuaZtMELUUJLcCKG1JgMsP6Vrqlrwd/WnjnMdclQOhxIOaR2OEDWSJDdCaC2v7ubkesjJ0jYWUWH5JvOTrikhNCHJjRBa828Lrr6QlQpnftc6GlEJ8rqmJLkRQhuS3AihNb0emlwvLD6+VttYRKW48cxNDZttQ4gqQZIbIaqCvLqbo7/KbMXVQJh3GAa9gSsZVziXck7rcISocSS5EaIqCO0JDia4Gg2XT2gdjaggRwdHmns1ByxLMQghbi1JboSoChzdIbirZVtGTVULbXylqFgIrUhyI0RVYR0SvkbbOESlkBFTQmhHkptKEp+cwftrjzJl6QGtQxH2Kq+o+Mw2uJaoaSii4vKSm2NXj5Gena5xNELULJLcVJLMHDMfrT/Bd7vOcjVN5ioR5eAVCj5NQeXCyd+0jkZUUF3Xuvi5+GFWZg4mHNQ6HCFqFEluKkmglwvN6rqTa1ZsPCZrBIlysi6kKV1T1YHMdyOENiS5qUR9m/sBEHn4osaRCLuVV3dzPBLMudrGIiosr2tKRkwJcWtJclOJ+oRZkptNRy+RmSNfTKIcAjuBkydcuwLndmkdjaigvORm/6X9MpmfELeQJDeVqFU9T3zdHUnLyuWPU1e0DkfYIwcDNOpr2ZYh4XYvzCsMk95EYmYiZ5LPaB2OEDWGJDeVSK/X0cfaNRWncTTCbkndTbVhdDDSwqcFIHU3QtxKktxUsr7Xu6bWHY6X09CifBr1AZ0e4g9BYozW0YgKkrobIW49SW4qWeeG3riYHIhLzuDQhWStwxH2yMXLUnsDcvamGpDJ/IS49SS5qWRORgdub1wHgLUyakqUl3RNVRt5yc2JqydIzUrVOBohagZJbmwgr+5mnSQ3orzyhoRHb4asNG1jERVSx6UO9dzqoVAcSJAZzIW4FSS5sYE7mvmi18Hh2GTOJ17TOhxhj+o0g1pBkJtpSXCEXWtdpzUgdTdC3CqS3NiAl6uJ9sFegJy9EeWk092wkKYMCbd3UncjxK0lyY2N9GnuC8C6I5LciHK6se5GRt7ZtbxlGPZf2o9ZmbUNRogaQJIbG+nbvC4Af5y6THJGtsbRCLsU3A2MLpASC3H7tY5GVECT2k1wcnAiJSuF6KRorcMRotqT5MZGGvi40rCOK9m5ik1HL2kdjrBHRicI7WXZllFTds2ol8n8hLiVJLmxIeuoKemaEuVl7ZqSuht7J3U3Qtw6ktzYUL/ryc2Gv+LJzpV+dlEOjftZfp7fDanx2sYiKiS8TjgA++IluRHC1iS5saHwwNp4u5pIzshhZ7QspCnKwcMf/MMt28fXahqKqJi84eAnk06SlJmkcTRCVG+S3NiQg17HHc0so6YipWtKlJd1SLjU3dgzb2dvAt0DAWQyPyFsTJIbG+trXSX8oiykKconr+7m5HrIydI2FlEhUncjxK0hyY2NdWvsg6NBz7mr1zh6MUXrcIQ98g8HV1/ISoUzv2sdjagAqbsR4taQ5MbGXEwGujXyAWS2YlFOej00uV5YLF1Tdq2Nr+XMzf6E/eSaczWORojqS5KbW+DGrikhysVad/OrzFZsxxrVaoSzwZm07DROJp3UOhwhqi1Jbm6BO8J80elg37kkLiZnaB2OsEehPcHBBFdPQ8JxraMR5WTQG2jl0wqQuhshbEmSm1vA192J8MBagEzoJ8rJ0R1Culm2ZUI/u2YtKpa6GyFsRpKbW6RP2PXZiqVrSpRXXteUzHdj1/IW0ZQzN0LYjiQ3t0he3c3vJy+TlpmjcTTCLuXNVnxmG1xL1DQUUX6tfSyT+Z1OPk1iRqK2wQhRTUlyc4s09nUj2NuFrBwzW47LQpqiHLwaQJ1moHLh5G9aRyPKqZZTLUI8QgDLqCkhROWT5OYW0el01q6pyMOyRpAop8YyJLw6yKu7iYqP0jYQIaopSW5uobzkZv1fF8k1y3BeUQ431t3IPCl2K2++G6m7EcI2JLm5hTqE1MbT2cjV9Gx2n7mqdTjCHgV2AidPuHYVzu3UOhpRTnlnbg4kHCDHLDV4QlQ2SW5uIYOD3rqQpgwJF+XiYIBGfS3bMiTcbjX0bIir0ZVrOdc4kXhC63CEqHYkubnFZEi4qDBZJdzuOegdrKOmZL4bISqfJDe3WI+mdTA56DmVkMaJ+FStwxH2qFFv0Okh/jAkxmgdjSinvLqbqEtR2gYiRDUkyc0t5uZo4LaG3oB0TYlycvGCwNss23L2xm5ZZyqWomIhKp0kNxrIm9BPuqZEuTXpb/kpyY3dal3H0i11NuUsl69d1jgaIaoXSW400CfMUlS8O+YqCamZGkcj7FJechO9GbLStI1FlIuHyYOGng0B2H9JJvMTojJJcqMBf09nWtbzQClY/5dM6CfKoU4zqBUEuZlwapPW0YhykrobIWxDkhuN9A2rC0CkdE2J8tDpbhg1JUPC7ZXU3QhhG5LcaKRPc0vX1Jbjl8jIlplmRTncWHejZMZrexReJxyAQwmHyDZnaxuMENWIJDcaae7vQb1azmRkm/n9RILW4Qh7FNwNjK6QGgex8p+/PQrxDMHd5E5GbgbHrh7TOhwhqg1JbjRiWUjTcvZGuqZEuRidoGEvy/bxtdrGIspFr9NbR03JIppCVB5JbjTUJ29I+JF4zLKQpigPa9eU1N3YK6m7EaLySXKjoU4NvHF3NJCQmsm+c4lahyPsUeN+lp/nd0OqjLyzR3l1NzIcXIjKo3lyM2vWLBo0aICTkxMRERFs2bKl2OMzMzOZOnUqwcHBODo60rBhQ+bNm3eLoq1cJoOeHk3rANI1JcrJvS74h1u2pWvKLrXyaYUOHedTz3Mp/ZLW4QhRLWia3CxZsoQJEyYwdepU9u7dS/fu3Rk4cCAxMUWvl3P//ffz22+/MXfuXI4ePcrixYtp1qzZLYy6cllnK5alGER5yZBwu+ZmcqNR7UaAdE0JUVk0TW7ef/99HnvsMcaOHUtYWBgzZ84kMDCQ2bNnF3r86tWr2bRpE6tWraJPnz6EhITQsWNHunTpcosjrzw9m/hi0Os4djGVM5dlpllRDnl1Nyc3QI7MeG2PpO5GiMqlWXKTlZXF7t276devX779/fr1Y9u2bYXe5+eff6Z9+/bMmDGDevXq0aRJEyZPnsy1a9eKbCczM5Pk5OR8l6rE08VIxwZegHRNiXLyDwc3P8hKhTO/ax2NKIe8uhtJboSoHJolNwkJCeTm5uLn55dvv5+fH3FxcYXe59SpU2zdupWDBw+ybNkyZs6cyQ8//MBTTz1VZDvTp0/H09PTegkMDKzU51EZ+oRJ15SoAL3+78JiWUjTLuWduTmUcIjsXJnMT4iK0rygWKfT5buulCqwL4/ZbEan07Fw4UI6duzIoEGDeP/99/nqq6+KPHszZcoUkpKSrJezZ89W+nOoqLy6m52nr5KYnqVxNMIu3Vh3I7MV251gj2BqOdYiy5zFkStHtA5HCLunWXLj4+ODg4NDgbM08fHxBc7m5PH396devXp4enpa94WFhaGU4ty5c4Xex9HREQ8Pj3yXqibQy4Vmdd3JNSs2HJXhvKIcQnuCgwmunoaE41pHI8pIp9NZJ/OTrikhKk6z5MZkMhEREUFkZGS+/ZGRkUUWCHft2pULFy6Qmppq3Xfs2DH0ej3169e3aby2Zu2aOizJjSgHRzcI6W7ZllFTdknqboSoPJp2S02aNIkvv/ySefPmceTIESZOnEhMTAzjxo0DLF1Ko0aNsh7/4IMP4u3tzSOPPMLhw4fZvHkzzz//PI8++ijOzs5aPY1Kkdc1tfFoPJk5spCmKIcbF9IUdiev7kaWYRCi4jRNboYPH87MmTOZNm0a4eHhbN68mVWrVhEcHAxAbGxsvjlv3NzciIyMJDExkfbt2/PQQw8xZMgQPvroI62eQqVpVc8TX3dH0rJy+ePUFa3DEfYor6g4Zjtcu6ptLKLMWvq0RK/TczH9InFphQ+qEEKUjk6pmlV9mJycjKenJ0lJSVWu/ualZQdYtCOGkbcF89+7W2odjrBHn3aCS3/BvXOh1T+0jkaU0f2/3M+RK0d4t8e79A/pr3U4QlQpZfn+1ny0lPhb3xuGhNewnFNUFumasmtSVCxE5ZDkpgrp3NAbF5MDsUkZHLpQtSYbFHYib0j4iUgwS+2WvbHOVBwvyY0QFSHJTRXiZHTg9saykKaogPodwamWpebm3E6toxFllDdi6vCVw2TmylIaQpSXJDdVTJ/ro6YkuRHl4mCAxn0t2zIk3O7Ud6+Pl5MXOeYcjlyWyfyEKC9JbqqYO5r5otfB4dhkzicWvWaWEEWyzlYsdTf2RqfTySKaQlSCCic3OTk5lRGHuM7L1UT7YMtCmr/JWlOiPBreATo9xB+Gq2e0jkaUkcx3I0TFlTq5SUxMJDExEYCrV68ycuRIvLy8cHJyIjQ0lE8++cRWMdY4fZr7AtI1JcrJxQsCb7NsH1+rbSyizG48cyOjJoUon1InN/fddx9bt24FYOzYsfj5+fHVV1/x448/MmzYMP7973/z8ccf2yzQmiRvKYY/Tl0mOUNWCBblYB0SLnU39qaFTwsMOgOXrl0iNi1W63CEsEulTm52795NWFgY58+fp1u3brz77rsMHTqUu+66i/fee49PP/2U2bNn2zLWGiO0jhsN67iSnavYdPSS1uEIe5RXdxO9BbLStI1FlImzwZmmXk0BqbsRorxKndxkZGTg6OjIhg0bePDBBwvc3q1btyJX5hZllzdqap3U3YjyqNMUagVDbiac2qR1NKKMpO5GiIopdXITEhLCggUL+PLLLzl69GiB2//3v//RpEmTSg2uJut3PbnZ8Fc82blmjaMRdkenu2HUlHRN2RsZMSVExRhKe+CkSZN44oknaNy4Mdu3b2fWrFm0adOGrKwstm7dyvr161m+fLkNQ61ZwgNr4+1q4nJaFjujr9ClkY/WIQl706Q//PmZZUi4UpaER9iFcN9wAI5eOUpGTgZOBidtAxLCzpT6zM3YsWNJTEzk4MGDvPDCCwwfPpz169fzww8/4O7uzoYNGxgyZIgtY61RHPQ67mh2fdSUdE2J8gjpBkZXSI2DWDkDYE/8Xf2p41yHHJXDocuHtA5HCLtTpnlu3N3dMRgsJ3uGDRtGZGQkBw4cYOnSpdx+++02CbAm69tcFtIUFWBwhIa9LNsyoZ9duXEyP6m7EaLsZIbiKqxbYx8cDXrOXrnG0YspWocj7JEMCbdbUncjRPlJclOFuZgMdLtea7NOJvQT5dG4n+XnhT2QIn9D9iSv7kYm8xOi7CS5qeLyuqYij8RrHImwS+51IaCtZVtmK7YrYd5hGPQGrmRc4VyqTLMhRFlIclPF3RHmi04H+84mcjE5Q+twhD3KGxJ+XOpu7ImjgyPNvZoDUncjRFmVKbnJycnBYDBw8OBBW8UjbuLr7kR4YC0AfpOzN6I88upuTm6AnExtYxFl0sZX6m6EKI8yJTcGg4Hg4GByc3NtFY8oRN5aU5GH4zSORNilum3ArS5kpcKZ37WORpRBXlHx/kv7NY5ECPtS5m6p//znP0yZMoUrV67YIh5RiLy6m99PXiYtM0fjaITd0euhyfXCYhkSblfykptjV4+Rnp2ucTRC2I8yJzcfffQRW7ZsISAggKZNm9KuXbt8F1H5Gvu6EeztQlaOmS3HE7QOR9ijxte7po7+apmtWNiFuq518XPxI1flcjBBygGEKK1SL7+Q5+6777ZBGKI4Op2OPmF+zN0aTeThiwxoWVfrkIS9Ce0JDiZIPAMJxywLawq7EO4bzprTa9h3aR8d/TtqHY4QdqHMyc2rr75qizhECfKSm/V/XSTXrHDQyzpBogwc3SCkO5z8zTKhnyQ3dqNNnTbW5EYIUTrlHgq+e/duFixYwMKFC9m7d29lxiQK0SGkNp7ORq6mZ7Mn5qrW4Qh7ZF0lXOpu7MmNMxXLZH5ClE6Zk5v4+HjuuOMOOnTowPjx43n66aeJiIigd+/eXLp0yRYxCsDgoP97IU2ZrViUR15RccwfkCrvVXsR5hWGSW8iMTORM8lntA5HCLtQ5uTmmWeeITk5mUOHDnHlyhWuXr3KwYMHSU5OZvz48baIUVyXNyRclmIQ5VI7BALagcqFfYu0jkaUktHBSAufFoDMdyNEaZU5uVm9ejWzZ88mLCzMuq958+Z8+umn/Prrr5UanMivR9M6mBz0nEpI4+SlVK3DEfYoYozl5+6vZdSUHZFFNIUomzInN2azGaPRWGC/0WjEbDZXSlCicG6OBm5r6A1I15Qop5b3gskNrpyE01u1jkaUUl5yE3UpSttAhLATZU5u7rjjDp599lkuXLhg3Xf+/HkmTpxI7969KzU4UVDehH7SNSXKxdENWt1n2d79laahiNLLS25OXD1BapactRWiJGVObj755BNSUlIICQmhYcOGNGrUiAYNGpCSksLHH39sixjFDfqEWYqKd8dc5XKqrBMkyiFitOXnkZ8h7bK2sYhSqeNSh3pu9VAoDiQc0DocIaq8Ms9zExgYyJ49e4iMjOSvv/5CKUXz5s3p06ePLeITN/H3dKZlPQ8Onk/mt7/iub99oNYhCXsT0Bb820DsPtj/P+j8lNYRiVJoXac151PPs+/SPjoHdNY6HCGqtHKvCt63b1+eeeYZxo8fL4nNLdY3zDJDsXRNiXKzFhZ/JYXFdkLqboQoPVkV3A71aW7pmtpyPIGMbPldiHJo+Q8wulqWYojZrnU0ohTC64QDlhXCzUoGbwhRHFkV3A419/egXi1nrmXn8vsJWUhTlIOTB7S617IthcV2oYlXE5wcnEjJSuF00mmtwxGiSpNVwe2QZSFNy9mbdUeka0qUU7sxlp+HlkO6/LNS1Rn1MpmfEKUlq4LbqT7N/fh6+xnWHYnnDbNCLwtpirKq1w78WsHFA7D/O7htnNYRiRK0qdOG3Rd3E3UpimGNh2kdjhBVVpmSm5ycHAAeffRRAgNllI6WOjXwxt3RwKWUTHZEX6Hz9cn9hCg1nc4yLHzVZEvXVKcnLPtElZVXd7P9wnZyzbk46B20DUiIKqrMBcXvvvuuFBRXASaDnjvbBADwv50xGkcj7Fbr+8HgDJeOwNk/tY5GlKBzQGc8TB7EpsWyPVYKwYUoSplrbnr37s3GjRttEIooq4c6BQHw64E4rqZlaRyNsEtOntDyHsv2nq+1jUWUyMngxNCGQwH44dgPGkcjRNVV5pqbgQMHMmXKFA4ePEhERASurq75bh86dGilBSeK17KeJ63qeXLgfBI/7jnH2O6hWock7FHEGIhaCAeXQv83wbmW1hGJYvyjyT9YcGQBG89uJD49Hl8XX61DEqLK0SlVthm89PqiT/bodLoq32WVnJyMp6cnSUlJeHh4aB1OhS3aEcNLyw4QWseV3yb1QCc1E6KslILZXSD+MAx6Fzo+rnVEogSjfx3Nnvg9PNP2Gf7Z+p9ahyPELVGW7+9yrQpe1KWqJzbV0dDwAFxMDpy6lMaf0TKcV5SDTvf3jMW75suMxXbgH03+AcCPx36UCf2EKESZkxtRtbg5Grgr3FJYvPhPKSwW5dT6fjA4QfwhOL9b62hECfoG98Xd5M6FtAtsvyCFxULcrNTJzaBBg0hKSrJef+ONN0hMTLRev3z5Ms2bN6/U4ETpPNgxGIBVB6WwWJSTc21ofrdlW2YsrvJuLCz+/tj3GkcjRNVT6uRmzZo1ZGZmWq+//fbb+ZZgyMnJ4ejRo5UbnSiVVvU9aVnPg6wcMz/uOad1OMJe5XVNHfwRMpI1DUWU7B+NLV1TG89u5FL6JW2DEaKKKXVyc3PdcRnrkIWNjehoGRa++M8Y+d2I8gm6DXyaQnY6HJCzAVVdo9qNaOvbllyVy/ITy7UOR4gqRWpuqomhbSyFxScvpbHz9FWtwxH26MbCYumasgvWwuLjUlgsxI1KndzodLoCw4xl2HHV4e5ktBYWL9pxRuNohN1q8wA4mCBuP1zYq3U0ogT9gvvhbnLnfOp5KSwW4galnsRPKcWYMWNwdHQEICMjg3Hjxlkn8buxHkdoY0THIBb/eZZVB+N4NS2L2q4mrUMS9sbFC5rfZemW2v0VBLTVOiJRjLzC4oVHFvLDsR/oWq+r1iEJUSWU+szN6NGj8fX1xdPTE09PTx5++GECAgKs1319fRk1apQtYxUlaFXPkxYBlsLipXvPax2OsFd5XVMHfoDMFE1DESW7t/G9gBQWC3GjUp+5mT9/vi3jEJVAp9MxomMQ/1l+kMV/xvBo1xDpOhRlF9wVvBvB5ROWkVN5yY6okhrXbkx4nXCiLkWx/MRyHm8tM0wLIQXF1cxd12csPhGfKoXFonx0Omg32rK9WxbTtAdSWCxEfpLcVDPuTkaGtpEZi0UFhT8IeiNc2AOx+7SORpSgf0h/a2HxHxf+0DocITQnyU01lDfnzcoDsSSmy4zFohxcfSBsiGVbzt5UeU4GJ4aEWn5fPxz/QeNohNCeJDfVUOv6njT3v15YvEcKi0U55dXa7P8OstI0DUWU7N4mlsLiDTEbSLiWoHE0QmhL8+Rm1qxZNGjQACcnJyIiItiyZUup7vf7779jMBgIDw+3bYB2SKfT8WAny9mbRTJjsSivkO5QuwFkpcChZVpHI0rQpHYT2tRpQ47KkRmLRY2naXKzZMkSJkyYwNSpU9m7dy/du3dn4MCBxMQUXyuSlJTEqFGj6N279y2K1P7cFR6As9FSWLzrjBQWi3LQ6yEir7D4K01DEaWTV1j8w7EfpLBY1GiaJjfvv/8+jz32GGPHjiUsLIyZM2cSGBjI7Nmzi73fE088wYMPPkjnzp1vUaT2J19h8Q4pLBblFP4Q6A1wbifEHdQ6GlGC/iH9cTdeLyyOlcJiUXNpltxkZWWxe/du+vXrl29/v3792LZtW5H3mz9/PidPnuTVV18tVTuZmZkkJyfnu9QUI653Ta2QwmJRXm6+0GywZXuPFBZXdc4GZ+5seCdgOXsjRE2lWXKTkJBAbm4ufn5++fb7+fkRFxdX6H2OHz/Oiy++yMKFCzEYSjf/4PTp062zKHt6ehIYGFjh2O1Fm/qehElhsaiovMLifUsgK13TUETJ8rqmpLBY1GSaFxTfPIOuUqrQWXVzc3N58MEHef3112nSpEmpH3/KlCkkJSVZL2fPnq1wzPbixsLixVJYLMqrQU+oFQyZSXD4J62jESVoUrsJreu0lsJiUaNpltz4+Pjg4OBQ4CxNfHx8gbM5ACkpKezatYunn34ag8GAwWBg2rRp7Nu3D4PBwPr16wttx9HREQ8Pj3yXmiSvsPh4fCq7pbBYlIdeD+2urxsnhcV24b4m9wHw4zGZsVjUTJolNyaTiYiICCIjI/Ptj4yMpEuXLgWO9/Dw4MCBA0RFRVkv48aNo2nTpkRFRdGpU6dbFbpd8XAyMqSNP2AZFi5EubR9GHQOcPYPiD+idTSiBHmFxedSz0lhsaiRNO2WmjRpEl9++SXz5s3jyJEjTJw4kZiYGMaNGwdYupTyVhrX6/W0bNky38XX1xcnJydatmyJq6urlk+lSrPOWLw/lqT0bI2jEXbJvS40HWjZlhmLqzxngzODQy2F4FJYLGoiTZOb4cOHM3PmTKZNm0Z4eDibN29m1apVBAcHAxAbG1vinDeiZOGBtQjz9yAzx8zSvee0DkfYK2th8WLIztA0FFEyKSwWNZlO1bAq0+TkZDw9PUlKSqpR9Tffbj/Nyz8doomfG2sm3F5o0bYQxTLnwodtIOks3PMFtL5f64hECR5a9RD7L+1nQrsJPNbqMa3DEaJCyvL9rfloKXFr3NW2Hk5GPccuprInRgqLRTnoHaSw2M78o7HMWCxqJkluaggPJyNDWltmLF4oMxaL8mr7MOj0cOZ3uHRM62hECfqH9MfN6Ma51HPsiN2hdThC3DKS3NQgeXPeSGGxKDePAGgywLItMxZXeS5GFyksFjWSJDc1SHhgLZrVdSczx8wyKSwW5dXu+mKaUYsgJ1PbWESJ8ua8WR+zXgqLRY0hyU0Nkn/G4rMyY7Eon0Z9wKMeXLsCR37ROhpRgqZeTWntY5mx+KcTMsO0qBkkualh7gq3FBYfvZjCnphErcMR9sjBAG1HWralsNgu5A0L//G4zFgsagZJbmoYT+e/C4sXSWGxKK+8wuLTW+DySa2jESXIKyw+m3KWP+P+1DocIWxOkpsaaMT1rqkV+y9IYbEon1qBlu4pkMJiOyCFxaKmkeSmBmp7Q2Hx8qjzWocj7FXejMV7F0JOlqahiJLlFRb/FvMbl69d1jgaIWxLkpsaSKfTWdebWvxnjBQWi/Jp3B/c6kJ6AhxdqXU0ogRNvZrSyqcVOeYcfjophcWiepPkpoa6+/qMxX/FSWGxKCcHA7STwmJ7Yi0sPiaFxaJ6k+SmhvJ0NnLn9cLixX9KYbEop7YjAR2c2ghXorWORpRgQMgAXI2uxKTEsDNup9bhCGEzktzUYHldUyv2XyDpmhQWi3KoHQwN77Bs7/lG21hEiVyMLtwZeicA3x/7XuNohLAdSW5qsHZBtWjq505GtpmfpLBYlJe1sHgB5EqSXNXldU1JYbGoziS5qcEshcWBgGXOGyksFuXSdCC4+kJaPBz9VetoRAmaeTWjpXdLcsw5/HzyZ63DEcImJLmp4Ya1q4+jwVJYvPdsotbhCHvkYLRM6gcy542dyDt788OxH+SfGlEtSXJTw+UrLJYZi0V55Y2aOvEbXD2jbSyiRAMbDLQWFsuMxaI6kuRG8GAnS9fUL/svkJwhNROiHLxCIbQnoGDvt1pHI0rgYnRhcAOZsVhUX5LcCNoF1aaJnxsZ2WaW75XCYlFO+QqLczQNRZQsr2tqXcw6rmRc0TgaISqXJDcCnU7Hg9eHhUthsSi3poPBxQdSYuH4Wq2jESUI8w6jhXcLS2HxCSksFtWLJDcCgGFt/y4sjpLCYlEeBhOEP2jZlhmL7ULeelM/HJfCYlG9SHIjAPB0MTK4tT8gMxaLCmg32vLzRCQkntU2FlGigQ0G4mJw4UzyGZmxWFQrktwIq7yuqV/2xUphsSgfn0YQ0h2U2VJ7I6o0F6MLg0OlsFhUP5LcCKuIYEth8bXsXH6SwmJRXtbC4m/BnKtpKKJkUlgsqiNJboSVZcZiy9mbhVJYLMorbAg4e0HyeTixTutoRAmaezenhXcLss3ZUlgsqg1JbkQ+w9rWsxYW7zuXpHU4wh4ZHKWw2M5YZyyWwmJRTUhyI/Kp5WJicCtLYfGiHTLTrCinvMLiY6sh+YK2sYgS3VhYvOviLq3DEaLCJLkRBYzoJIXFooLqNIHgrtcLixdqHY0ogavRlUGhgwD4/tj3GkcjRMVJciMKaB9cm8a+1wuLo+S/blFOeYXFe76RwmI7kDfnzboz67iacVXjaISoGEluRAE3FhbLjMWi3MKGglMtSIqBw8u1jkaUoLl3c5p7N7cUFp+UwmJh3yS5EYW6p109TAY9R2KTpbBYlI/RCW77l2U78lXIvqZtPKJE1sLiY1JYLOybJDeiUDcWFi/eITMWi3Lq8gx41Ieks7DtY62jESUY1GAQLgYXTieflsJiYdckuRFFevB6YfHP+y6QIoXFojxMLtD3dcv21g9k5FQVJ4XForqQ5EYUqX1wbRpJYbGoqJb3QuBtkJ0O617TOhpRAuuMxVJYLOyYJDeiSFJYLCqFTgcDplu29y+Bs7JAY1XWwrsFYV5hUlgs7JokN6JY914vLD4cm8x+KSwW5VWvHYQ/ZNle/SKYzdrGI4olhcXC3klyI4qVr7D4TyksFhXQ+xUwucH5XXBA6jmqssGhg3E2OEthsbBbktyIEuV1TUlhsagQ97rQfZJle91rkJWmaTiiaK5GVwY1sBQW/3DsB42jEaLsJLkRJeoQUpuGdVxJz5LCYlFBtz0FtYIh5QJsnal1NKIYeTMWR56JJDEjUdtghCgjSW5EiaSwWFQaoxP0+69le9tHkChdnVVVc+/mUlgs7JYkN6JU7m1X31pYvO3kZa3DEfYsbCgEd4OcDMvMxaJK0ul01sLi/x39H+nZ6RpHJETpSXIjSqW2q4l729UD4KlFezgRn6pxRMJuWYeG6+DQUjizXeuIRBEGNRiEl5MXZ1PO8u/N/ybHnKN1SEKUiiQ3otReubMF4YG1SEzPZsz8P4lPydA6JGGv/FtDu1GWbRkaXmW5mdz4sNeHODo4suncJt768y3plhZ2QZIbUWrOJgfmjm5PiLcL565e49GvdpKWKf/JiXK642Vw9IDYKNi3SOtoRBHCfcOZ3n06OnQsObqErw99rXVIQpRIkhtRJt5ujnz9aEe8XU0cPJ/MvxbuITtX/usW5eBWB25/3rK97nXISNY2HlGkvsF9eb6D5Xf13u73WH16tcYRCVE8SW5EmQV7uzJ3TAecjHo2HbvEf5YdlFPVonw6jQOvUEiLhy3vaR2NKMbI5iN5KMwyy/RLW15iz8U9GkckRNEkuRHlEh5Yi09GtEOvgyW7zvLRbye0DknYI4MJ+r9p2f5jFlw5pW08oljPt3+eOwLvINuczfgN44lOitY6JCEKJcmNKLc+zf34790tAfhg3TG+23VW44iEXWoyAEJ7QW4WrH1Z62hEMRz0Drx1+1u09mlNUmYST657koRrCVqHJUQBktyICnmoUzBP9WoIwJSlB9h07JLGEQm7kzc0XOcAf62A6M1aRySK4Wxw5qM7PqK+W33Op55n/PrxXMu5pnVYQuQjyY2osMn9mnJP23rkmhX/WrCbg+dl9XBRRr5h0P5Ry/bqKWDO1TYeUSxvZ29m95mNp6MnBxIO8MLmF8iV35moQiS5ERWm0+l4697WdG3kTVpWLo98tZOzV2Q2U1FGvV4Cp1pw8SDskeHGVV2IZwgf3/ExJr2JDWc3MGPnDBlYIKoMSW5EpTAZ9Mx+OIJmdd25lJLJmPl/kpiepXVYwp64eEHPKZbt9f8H1xI1DUeUrK1vW97sbikIX/TXIr49/K3GEQlhIcmNqDQeTka+eqQj/p5OnLyUxuPf7CIjW05VizLo8Bj4NIX0y7D5Ha2jEaXQP6Q/k9tPBuDdXe+y9vRajSMSQpIbUcnqejrx1SMdcXcysPP0VSZ9F4XZLKeqRSk5GP8eGr5jDiTIFAP2YFTzUYxoNgKFYsqWKUTFR2kdkqjhJLkRla5pXXc+H9kek4OeVQfieGPVEa1DEvakcR9o3A/MObB2qtbRiFLQ6XS80OEFegb2JMucxTPrn+FM8hmtwxI1mCQ3wiY6N/TmnftaAzB3azRfbpHJ2UQZ9H8T9AY4thpO/KZ1NKIUHPQOvN39bVp6tyQxM5En1z3JlYwrWoclaijNk5tZs2bRoEEDnJyciIiIYMuWLUUeu3TpUvr27UudOnXw8PCgc+fOrFmz5hZGK8rirvB6TBnYDIA3Vh1h5f5YjSMSdsOnMXT8p2V7zUuQKwu02gMXowsf9/6Yem71OJtylmfWP0NGTobWYYkaSNPkZsmSJUyYMIGpU6eyd+9eunfvzsCBA4mJiSn0+M2bN9O3b19WrVrF7t276dWrF0OGDGHv3r23OHJRWv+8PZTRnYNRCiZ+F8Wf0fKfnCilHv8GZy+49Bfsmqd1NKKUfJx9mN1nNh4mD/Zf2s+LW16UOXDELadTGk5M0KlTJ9q1a8fs2bOt+8LCwrj77ruZPn16qR6jRYsWDB8+nFdeeaVUxycnJ+Pp6UlSUhIeHh7liluUTa5Z8eSC3aw9fBFPZyM/PtmZRr7uWocl7MHOL2Hlc5b5b8bvtQwXF3Zh98XdPL72cbLN2Twc9jAvdHxB65CEnSvL97dmZ26ysrLYvXs3/fr1y7e/X79+bNu2rVSPYTabSUlJwcur6A+8zMxMkpOT813EreWg1/HRiLa0C6pF0rVsRs/bSXyynKoWpdBuDPg2h4xE2Fi6f3hE1RDhF8Gb3Swj3xYcWcCCwws0jkjUJJolNwkJCeTm5uLn55dvv5+fH3FxcaV6jPfee4+0tDTuv//+Io+ZPn06np6e1ktgYGCF4hbl42R04MvRHWjg48r5xGs88tVOUjOljkKUwMFgWXcKYOdciJeRd/ZkQIMBTIyYCMCMnTP47YwUh4tbQ/OCYp1Ol++6UqrAvsIsXryY1157jSVLluDr61vkcVOmTCEpKcl6OXtWVq7Wiperia8f6YiPm4lDF5L518I9ZOeatQ5LVHWhPaHZnaByLcXFMsW/XXmkxSMMbzocheKFLS+w79I+rUMSNYBmyY2Pjw8ODg4FztLEx8cXOJtzsyVLlvDYY4/x3Xff0adPn2KPdXR0xMPDI99FaCfI24W5ozvgbHRg87FLvLT0gKxHI0rW77/gYIKT6+GYjJC0Jzqdjhc7vkiP+j3IzM3kmd+eISa58EEjQlQWzZIbk8lEREQEkZGR+fZHRkbSpUuXIu+3ePFixowZw6JFixg8eLCtwxQ20CawFp8+1Ba9Dr7ffY6Z645rHZKo6rxC4bYnLdtrXoIcWbfMnhj0BmbcPoPm3s25mnmVf/32L65mXNU6LFGNadotNWnSJL788kvmzZvHkSNHmDhxIjExMYwbNw6wdCmNGjXKevzixYsZNWoU7733HrfddhtxcXHExcWRlJSk1VMQ5XRHMz/+7+5WAHz423GW7JT/5EQJuk8GV1+4chL+/FzraEQZuRhd+LT3pwS4BnAm+Qzj14+XOXCEzWia3AwfPpyZM2cybdo0wsPD2bx5M6tWrSI4OBiA2NjYfHPefPbZZ+Tk5PDUU0/h7+9vvTz77LNaPQVRAQ92CuKZOxoB8NKyg2w4Gq9xRKJKc/KA3i9btjfNgLQEbeMRZZY3B467yZ2oS1G8tPUlzErq7kTl03SeGy3IPDdVi1KK577fx9I953ExObDkn51pVd9T67BEVWXOhc97Qtx+iHgEhszUOiJRDjvjdvJE5BNkm7MZ3Xw0kztM1jokYQfsYp4bIcBSbPjWPa3p3tiH9KxcHvlqJ2evpGsdlqiq9A4w8G3L9p6vIe6gtvGIculQtwP/7fpfAL4+/DULjyzUOCJR3UhyIzRnMuiZ9VA7wvw9SEjNZPT8P7maJgWjogjBXaD53aDMsPpFGRpupwaHDubZdpaSgrf/fJv1Mes1jkhUJ5LciCrB3cnIV490IMDTiVOX0hj7zS4ysmU9GlGEvtPAwRFOb4G/VmgdjSinx1o+xj+a/MMyB87mFzhw6YDWIYlqQpIbUWX4eTjx1aMd8XAysPvMVSb8L4pcs/xXLgpROxi6PGPZXvsfyJZRN/ZIp9MxtdNUutfrTkZuBk+vf5qzKTLRqqg4SW5EldLEz53PR7XH5KBn9aE4/rvisEzyJwrXbSK41YWrp+GPWVpHI8rJoDfwbo93CfMK40rGFf617l8kZiRqHZawc5LciCrntlBv3ru/DQBfbTvNnE2nNI5IVEmObtDnNcv2lvcgpXRr0omqJ28OHH9Xf04nn+bZDc+Sni0DC0T5SXIjqqQhbQL4z+AwAN5e/RcL/jijcUSiSmo9HOpFQFYq/PZfraMRFVDHpQ6zes/C3ejOnvg9PLnuSVKzUrUOS9gpSW5ElTW2eyhP9WoIwMs/HWT53vMaRySqHL0eBrxl2Y5aCBf2ahuPqJBGtRsxu+9sa4LzROQTJGclax2WsEOS3IgqbXK/pozuHIxS8Nz3+1h7SLoexE0CO0Kr+wEFv8rQcHvXpk4bvuz/JZ6OnuxP2M/YNWNlHSpRZpLciCpNp9Px6pAW3NOuHrlmxdOL9vL7CZl2X9ykz2tgdIGzf8ChpVpHIyqouXdz5vWfh5eTF0euHOHRNY+ScE3e96L0JLkRVZ5er2PGva0Z0KIuWblmHv9mF7vPyH9y4gae9aDrBMt25KuQJcWo9q5J7SbMHzAfX2dfTiSe4JHVjxCXJmduRelIciPsgsFBz4cjwv9epmH+nxy+IH3x4gZdngGP+pB0Fub1g9j9WkckKijUM5SvBnxlHUU1ZvUYzqdK7Z0omSQ3wm44Ghz4bGQE7YNrk5yRw6h5Ozh1SUZTiOtMLjBsNjjXhrgD8EUvywiqnEytIxMVEOgRyFcDviLQPZDzqecZs3oMZ5Jl9KQoniQ3wq64mAzMe6QDLQI8SEjN4uEvd3A+8ZrWYYmqosHt8NSf0PwuMOfAlnfhs9vh3C6tIxMVEOAWwPz+82ng2YC4tDgeWf0IJxNPah2WqMIkuRF2x8PJyDePdqRhHVcuJGXw0Bd/EJ8i0++L69x84f5vLBfXOnDpL5jbF9ZMlVocO+bn6sf8/vNpXLsxl65d4tE1j3L0ylGtwxJVlCQ3wi55uzmyYGwn6tVy5vTldEbN/ZPEdFlJXNyg+V2WszitH7CsIL79E5jTFU7/rnVkopy8nb2Z128ezb2bcyXjCo+ueZRDCYe0DktUQZLcCLvl7+nMosc74evuyF9xKYyZv5PUzBytwxJViYsX3PMZPPgduAfAlVPw1SBYORkyU7SOTpRDLadafNHvC9rUaUNyVjJj144lKj5K67BEFSPJjbBrwd6uLBjbiVouRqLOJvL417vIyM7VOixR1TTpD0/9Ae1GW67v/AJmdYGT67WNS5SLh8mDz/p+RoRfBKnZqfwz8p/sjNupdViiCpHkRti9Jn7ufP1IR9wcDWw/dZmnF+0hO9esdViiqnHyhKEfwcjlUCsIkmLg22Hw01NwLVHr6EQZuRpdmd1nNp39O3Mt5xpPrnuSbee3aR2WqCIkuRHVQpvAWnw5uj2OBj3rjsTz3Hf7yDXLNPyiEA17wZPboeMTgA72LoBZt8HRX7WOTJSRs8GZj3t/TI/6PcjMzeTp9U+z8exGrcMSVYAkN6LauC3UmzkPR2DQ6/h53wX+s/wgStYZEoVxdINBM+CRX8G7EaTEwuIH4MexkHZZ6+hEGTg6OPJBzw/oG9yXbHM2EzdMZM3pNVqHJTQmyY2oVno182XmA+HodbD4zxim//qXJDiiaMGdYdxW6Pos6PRw4Hv4tCMcWiYLcNoRo4ORGbfPYFCDQeSoHP69+d/8cvIXrcMSGpLkRlQ7d7YOYPo9rQD4fPMpPll/QuOIRJVmdIa+02DsOqgTBukJ8P0YWPIwpFzUOjpRSga9gTe7vcmwRsMwKzNTt07lx2M/ah2W0IgkN6JaGt4hiP8MDgPgvchjzP89WuOIRJVXLwKe2AQ9XgC9Af5aYTmLE7VYzuLYCQe9A691eY3hTYejULy2/TUW/7VY67CEBiS5EdXW2O6hTOjTGIDXfznM97vOahyRqPIMjtDrJfjnRvBvAxmJsHwcLLwPks5pHZ0oBb1Oz9ROUxnVfBQAb+54k68OfqVtUOKWk+RGVGvP9m7MY90aAPDCj/tZdSBW44iEXajbCsauh96vgoMJTkTCp7fBrvlyFscO6HQ6JrefzOOtHgfgvd3v8dm+zzSOStxKktyIak2n0/GfwWEMbx+IWcGz/9vLxqPxWocl7IGDAbpPshQc1+8AWSmwYgJ8MxSuSDdnVafT6RjfbjxPhz8NwCdRn/DRno9kgEENIcmNqPZ0Oh1v3tOKwa39yc5VjFuwmz+jr2gdlrAXdZrCo2ug/3QwOEP0ZpjdBf6YDWaZDbuqe6LNE0xuPxmALw58wTu73pEEpwaQ5EbUCA56HR/cH06vpnXIyDbz6Fc7OXAuSeuwhL3QO0Dnf8G/tkFId8hOh9UvwvyBcOmY1tGJEoxuMZqXOr0EwLeHv+WNHW9gVjKLeXUmyY2oMUwGPbMfjqBTAy9SM3MYNW8Hxy/K4omiDLxCYdTPMPh9MLnD2R0wpxus/z/ISNY6OlGMEc1GMK3LNHToWHJ0Ca9ue5VcOfNWbUlyI2oUJ6MDX45uT5v6nlxNz+bhuTuIuZyudVjCnuj10OEx+Nd2aNgbcjNh8zvwYRvYPgtyMrWOUBRhWONhvNn9TRx0Diw/sZwpW6eQbc7WOixhA5LciBrH3cnIV490pImfGxeTM3lo7h/EJWVoHZawN7UC4eEf4f5vwbsxXLsCa6bAx+0tc+PIWYEq6c7QO3mnxzsYdAZ+jf6V5zc9T3auJDjVjSQ3okaq7WpiwWOdCPZ24eyVazw8dwdX0rK0DkvYG50Omg+Ff/0BQz4Ed3/LauPLx8Gc7nBsjQwdr4L6Bvflg14fYNQb+S3mN8ZvGM+Z5DNahyUqkU7VsLLx5ORkPD09SUpKwsPDQ+twhMbOXknnvjnbiUvOoFU9TxY+3gkPJ6PWYQl7lZUOf34GWz+AjOsF68Fdoc/rENhB29hEAdvOb+PZDc+SkWs5c9vKpxWDQwczIGQA3s7eGkcnblaW729JbkSNdyI+lfs/286VtCw6hnjx4YhwvF0dMRnkxKYop/QrlgRnx2eWmhyAZndC71csQ8tFlbHv0j5m75vN9gvbrSOoHHQOdA7ozJ2hd9IrsBcuRheNoxQgyU2xJLkRhTl4PokRn/9BSmaOdZ+LyYHaLiY8nY3UcjFatl2M1HK+advVRC1n4/XrJkmKxN+SzsHG6RC1CJTZsvJ4+EPQcwp41tM6OnGDhGsJrDm9hhUnV3Dw8kHrfmeDM72DejM4dDC3+d+GQW/QMMqaTZKbYkhyI4qy6/QVJn+/jzNX0itUJuFqcqCWi4laLpakqJazJRGqnW/7+u3ORoK9XSUhqu7i/4L1/7UsxglgcIKO/4RuE8HFS9vYRAGnk06zMnolK06u4Fzq32uKeTl5MbDBQO4MvZMW3i3Q6XQaRlnzSHJTDEluREnMZkVKRg5X07NIvJZNYnoWienXf17Lzrd9NT2bpOvbSdeyy5UU+bo78vQdjRjeIRBHg0PlPyFRdcTsgHWvQcw2y3UnT0uC02kcGJ01DU0UpJRif8J+VpxcwZrTa7iaedV6W4hHCINCB3FngzsJ9AjUMMqaQ5KbYkhyI2zFbFYkZ1xPfq5lczU9i6TridDVdEvyc3NSdCklk7Qsy5DhAE8nnundmH9E1MfoIGdyqi2l4PhaWPc6xB+y7HMPgJ4vWrqsHKTboyrKNmez/cJ2VpxawYaYDdYiZIDWdVpzZ+id9A/pj5eTnImzFUluiiHJjahKsnLMLNl1lk/WH+disqXwNMjLhWd7N+butvVw0Mtp72rLnAv7v4MNb0DSWcs+nyaWouNmd1qGmYsqKS07jfUx61lxagV/xP5hLUQ26Ax0qdeFwQ0G0yuoF84GORtXmSS5KYYkN6IqysjOZeGOGGZvPEFCqmW+ndA6rkzo04Q7W/mjlySn+srOgF1zYfO7lokAwbIKeZ/XIKSbpqGJkiVcS+DX6F9ZcWoFhy8ftu53MbjQJ7gPgxsMpqN/RylErgSS3BRDkhtRlaVn5fDN9jN8tukkV9Mts6Y29XNnYt8m9G/hJwWM1VlGEmz7GLZ/almYE6BxP+j9KtRtqW1solROJZ1i5amVrDy1kvOp5637fZx9GBAygDsb3klzr+byPi4nSW6KIcmNsAcpGdl89ftpPt9yipQMy/D0lvU8mNS3Cb2a+sqHY3WWEgebZsCer8GcA+ig9f3QayrUDtY6OlEKSin2XdrHilOWQuTEzETrbSEeIdwZeieDQgcR6C6FyGUhyU0xJLkR9iQpPZsvt55i3tZoa+FxeGAtnuvXhG6NfCTJqc4un7SsNn5oqeW6gwnaPwa3TwZXH21jE6WWnZvNtgvbLIXIZzeQmfv3wqrhdcKthci1nGppF6SdkOSmGJLcCHt0JS2Lzzad5Ovtp8nIthQvdmzgxXN9m9ApVKaJr9Yu7LUMHz+10XLd5A6t74PQnhDSXebJsSOpWan8FvMbK06t4M+4P/MVIner143BoYPpGdgTJ4OTxpFWTZLcFEOSG2HP4lMymL3xJAt3xJCVY/lg7NbIh0n9mtAuqLbG0QmbOrnekuTE7rthpw78W0ODHpZkJ6gzmGSpAHsQnx7Pr9G/svLUSo5cOWLd72p0pU9QHwaHDqZj3Y446GXuqzyS3BRDkhtRHcQmXeOT9Sf4btdZsnMtb+E7mvkyqW8TWtbz1Dg6YTNmM5xYZ7lEb4JLf+W/3cEE9TtCaA9LwlOvHTjIQrBV3anEU6w4tYJV0avyFSLXca5jnRG5mVezGt8NLclNMSS5EdXJ2SvpfLz+OD/uOU+u2fJW7t/Cj4l9m9Csrvx9V3spcRC92dJldWoTJJ/Lf7vJHUK6Xj+z0wN8m8v8OVWYUoqoS1GsOLmC1adXk5yVbL0t1DPUWohcz61mrksmyU0xJLkR1VF0Qhof/Xac5VHnUcry/XVn6wAm9GlMwzpuWocnbgWl4MopS6ITvcmS9Fy7mv8Y1zrQ4Pa/u7Fk9FWVlZ2bzdbzW1lxagUbz24ky5xlva2dbzsGhw6mf0h/PB1rzplaSW6KIcmNqM6OX0xh5rrjrDwQC4BeB3e3rcezvRsT7O2qcXTiljKbIW6/JdE5tQlitv89f06e2iF/n9Vp0ENGYVVRKVkprDuzjpWnVvJn3J8oLF/bBr2B7vW6Mzh0MD3q96j2hciS3BRDkhtRExy6kMQHkcdZd+QiAAa9jn9E1OeZ3o2pV0umhK+RcjLh3K6/z+yc2wUqN/8xfi3/PqsT3AUc5axfVXMx7aKlEDl6JX9d+bvmys3oRt/gvgwOHUx7v/bVshBZkptiSHIjapJ9ZxN5P/IYm45dAsDkoKdH0zrUcjbi6mjAxeSAq6MBV5MDLo4GXE0GXB0d/r7NZMDF0QE3RwPORocaX9BYrWSmwJltlrM6pzb+vYhnHr0B6rW3nNXxb2Pp0nLxtpzdcfSQ2p0q4PjV46w8tZJV0auITYu17vd18WVQg0HcGXonTWo3qTbvW0luiiHJjaiJdp2+wntrj7H91OVyP4ZOBy5GSxLkdlPyk5cUudzwM+8YdycDns4markYqeVipLaLCSdj9fuv0u6lXrpeq3O9GyvxTNHHOpjAxQdcva8nPT6Wn67eN2z7XE+G6oCjuyRDNmRWZvZc3MPK6JWsOb2GlKwU622NajVicOhggj2CMeqNmPQmjA5GjHrLxaA35LtucjBZt416Y5U6AyTJTTEkuRE12a7TVzh0IZm0rBzSM3NJy8ohLTOHtKxc0q//TMvMIf3Gn1k5VPanhKNBb0l2bkh6/t6+/tP5hu3rtzubqs4HbbV39bQlyYnebNlOuwTplyErteyP5WDKf+bHmhAVkRxJMlRuWblZbDm3hZXRK9l4diPZ5uwKPZ5ep8+X7NyYCOXbvik5quVUi1c7v1o5T+o6SW6KIcmNEGWjlOJadi5pmbmkZ+VYf6belARZrv99e16ilJKRQ9K1bBLTs0lMzyLHXP6PnNIkRR5ORhwNeowGPUYHHSYHPcbrF5NBh8nBAaNB9/c+B8txDnpdtTl9b1PZ1yAtAdITLD/TEq4nPgmQdvmG7euX7LSyt2F0Ba9Q8A61/PRqCN4NLdtufpL4lFJyVjLrzqxjfcx6UrJSyDZnWy652dbtrNysv/ebs8kx51RK277Ovvx2/2+V8lh5JLkphiQ3QmhHKUVaVi6J6VnXk51sEq9lcTU9m6Tr+66mZ5N07frt17Ktx1YkKSoNnQ5rsmO6nhj9nfzoCyREeceYDA74ezoRWNuZQC8XAr1cqF/bGUeDnGUCICv972Qn/XryY02ILt+UHCUUHNF1M5MbeDX4O+nxCr2e+DQEN19JfCpIKUWOOafw5OeGpKio2/Pua9QbGdZ4WKXGJslNMSS5EcL+5CVFV9OyrGeBrqZnkXitYFKUfC2HzFwz2TlmsnPzLoqsvO2cv6/bik4HdT2cCKztcj3hcSboeuIT5OVCHTdH9Hr5Ei5UVjokn7fM2XP5JFw5+fd20llQxfzebk588s722CrxUcpSmH3tKly7cv3nVUi/AtcSi9h/1XIfR3fLumDOta9fbth2qX3D/htuq+HddXaV3MyaNYt33nmH2NhYWrRowcyZM+nevXuRx2/atIlJkyZx6NAhAgIC+Pe//824ceNK3Z4kN0IIsCRM2bnKmgBlXU+C8pKirFwzWdcTIevtN1/PNXMtK5cLiRnEXEnn3NV0Yq6kk56VW2zbjgY99WvnT3gCvVyuJ0POuDvJkgmFysmExJjrSc8pS+KTt13qxOemsz1eoZbEx5qk3JyQXC1i//Xtm4fT25LeUHjSU2hSdMNtNyZFSoE51xJ3vp/mCuzPuWHbbPnpYIRGfSr16dtNcrNkyRJGjhzJrFmz6Nq1K5999hlffvklhw8fJigoqMDx0dHRtGzZkscff5wnnniC33//nX/9618sXryYe++9t1RtSnIjhLAlpRRX0rKIuZJ+PeG5RszldM5eT3xikzKsS2UUpbaLkSAvF+pfT3yCric+QV4u+Ndywuigz9derlmRqxRmM+QqRW6u5XquWWHOu/2GbctP8t+uFGbz39s33sce6HKzcE47h3PKGVxST1t+ppzBOfUMTukX0BWT+Ch06Cj/81QGZ8xOtVBOlmRCOddG52JJMHQutdE5e6F3vSHZMLlZCrNvTJJKSqpyMsodHzoH0OktSUdxCWBlcveH5/4q+bgysJvkplOnTrRr147Zs2db94WFhXH33Xczffr0Ase/8MIL/Pzzzxw58vcKquPGjWPfvn1s37690DYyMzPJzMy0Xk9OTiYwMFCSGyGEJrJzzcQmZliTnZgr6ZzNu1y9xpW0rGLvr9eBwUFvSUSUqvSRbNWRiWwCdfEE6y7SQBdHiC7Oul1Pl4BeZ3kRM5WRq7hxVbmRhBuJ6sZtV67iTqJyJen6MYnKjUTcyMRUqjgc9DocdDr0eq7/1N2wz/LTQf/37Q7Xb9frdDjrsvAk1XrxUCl4qBTcVQruKhV3lYybORU3cwpu5mRcc1NwNSdhVGUbLWVGh1nngEKPWeeAGT1Kp8d8/bplv76I2w3W6xmm2jSZ+GuZf1fFKUtyY6jUlssgKyuL3bt38+KLL+bb369fP7Zt21bofbZv306/fv3y7evfvz9z584lOzsbo7Hgqdzp06fz+uuvV17gQghRAUYHPUHeLgR5u9C1kNtTM3M4e1PSE3M98Tl7JZ3MHEt3WWnpdVi/IPN9kVr3Ff5Fa7jhPtWjzMOHBJqTAOy8Ya9BZeNuTiZJuZKBKf9ZrLyzYTftyzX/va3M4HB9X0lyzYpcFJS7J8sI1L5+KR0nMvEkDR2KXCxJyI0/b96Gyvll+7o78melPFL5aJbcJCQkkJubi5+fX779fn5+xMXFFXqfuLi4Qo/PyckhISEBf3//AveZMmUKkyZNsl7PO3MjhBBVkZujgTB/D8L8C/5najYrEtIyycoxF/iPPy8huTGR0euQ4e23iFIKsyq+qy+v2zD/vhsTJvJ1J+bkFv5Y1m7FIh7r731odobP1VHb0YKaJTd5bn7jKaWKfTMWdnxh+/M4Ojri6OhYwSiFEEJ7er0OX/fqvTiivdLpdDhcP0smtKcv+RDb8PHxwcHBocBZmvj4+AJnZ/LUrVu30OMNBgPe3t42i1UIIYQQ9kOz5MZkMhEREUFkZGS+/ZGRkXTp0qXQ+3Tu3LnA8WvXrqV9+/aF1tsIIYQQoubRLLkBmDRpEl9++SXz5s3jyJEjTJw4kZiYGOu8NVOmTGHUqFHW48eNG8eZM2eYNGkSR44cYd68ecydO5fJkydr9RSEEEIIUcVoWnMzfPhwLl++zLRp04iNjaVly5asWrWK4OBgAGJjY4mJibEe36BBA1atWsXEiRP59NNPCQgI4KOPPir1HDdCCCGEqP40n6H4VpNJ/IQQQgj7U5bvb027pYQQQgghKpskN0IIIYSoViS5EUIIIUS1IsmNEEIIIaoVSW6EEEIIUa1IciOEEEKIakWSGyGEEEJUK5LcCCGEEKJa0XxV8Fstb87C5ORkjSMRQgghRGnlfW+XZu7hGpfcpKSkABAYGKhxJEIIIYQoq5SUFDw9PYs9psYtv2A2m7lw4QLu7u7odLpKfezk5GQCAwM5e/bsLVnaQdqz/zalPftuT4s2pT1pr6q3aav2lFKkpKQQEBCAXl98VU2NO3Oj1+upX7++Tdvw8PC4petWSXv236a0Z9/tadGmtCftVfU2bdFeSWds8khBsRBCCCGqFUluhBBCCFGtSHJTiRwdHXn11VdxdHSU9uywPS3alPbsuz0t2pT2pL2q3qYWz/FmNa6gWAghhBDVm5y5EUIIIUS1IsmNEEIIIaoVSW6EEEIIUa1IciOEEEKIakWSGyGEEEJUK5LcVILXXnsNnU6X71K3bl2btnn+/HkefvhhvL29cXFxITw8nN27d9ukrZCQkALPT6fT8dRTT9mkvZycHP7zn//QoEEDnJ2dCQ0NZdq0aZjNZpu0B5a1SiZMmEBwcDDOzs506dKFnTt3Vtrjb968mSFDhhAQEIBOp2P58uX5bldK8dprrxEQEICzszM9e/bk0KFDNmtv6dKl9O/fHx8fH3Q6HVFRUeVuq6T2srOzeeGFF2jVqhWurq4EBAQwatQoLly4YJP2wPKebNasGa6urtSuXZs+ffqwY8cOm7V3oyeeeAKdTsfMmTNt1t6YMWMKvB9vu+22crdXmjYBjhw5wtChQ/H09MTd3Z3bbruNmJgYm7RX2GeOTqfjnXfesUl7qampPP3009SvXx9nZ2fCwsKYPXt2udoqTXsXL15kzJgxBAQE4OLiwoABAzh+/Hi525s+fTodOnTA3d0dX19f7r77bo4ePZrvmMr8nClNe5X9OVMWktxUkhYtWhAbG2u9HDhwwGZtXb16la5du2I0Gvn11185fPgw7733HrVq1bJJezt37sz33CIjIwG47777bNLe22+/zZw5c/jkk084cuQIM2bM4J133uHjjz+2SXsAY8eOJTIykm+//ZYDBw7Qr18/+vTpw/nz5yvl8dPS0mjTpg2ffPJJobfPmDGD999/n08++YSdO3dSt25d+vbta13otbLbS0tLo2vXrrz11lvlevyytJeens6ePXt4+eWX2bNnD0uXLuXYsWMMHTrUJu0BNGnShE8++YQDBw6wdetWQkJC6NevH5cuXbJJe3mWL1/Ojh07CAgIKFc7ZWlvwIAB+d6Xq1atsmmbJ0+epFu3bjRr1oyNGzeyb98+Xn75ZZycnGzS3o3PLTY2lnnz5qHT6bj33ntt0t7EiRNZvXo1CxYs4MiRI0ycOJFnnnmGn376qdLbU0px9913c+rUKX766Sf27t1LcHAwffr0IS0trVztbdq0iaeeeoo//viDyMhIcnJy6NevX77Hq8zPmdK0V9mfM2WiRIW9+uqrqk2bNresvRdeeEF169btlrV3s2effVY1bNhQmc1mmzz+4MGD1aOPPppv3z333KMefvhhm7SXnp6uHBwc1IoVK/Ltb9OmjZo6dWqltweoZcuWWa+bzWZVt25d9dZbb1n3ZWRkKE9PTzVnzpxKb+9G0dHRClB79+6tcDulaS/Pn3/+qQB15syZW9JeUlKSAtS6dets1t65c+dUvXr11MGDB1VwcLD64IMPKtxWUe2NHj1a3XXXXZXy+KVtc/jw4TZ7D5bmd3jXXXepO+64w2bttWjRQk2bNi3fvnbt2qn//Oc/ld7e0aNHFaAOHjxo3ZeTk6O8vLzUF198UeH2lFIqPj5eAWrTpk1KKdt/ztzc3o1s8TlTEjlzU0mOHz9OQEAADRo04IEHHuDUqVM2a+vnn3+mffv23Hffffj6+tK2bVu++OILm7V3o6ysLBYsWMCjjz5a6auq5+nWrRu//fYbx44dA2Dfvn1s3bqVQYMG2aS9nJwccnNzC/wH6uzszNatW23S5o2io6OJi4ujX79+1n2Ojo706NGDbdu22bx9LSQlJaHT6Wx2tvFGWVlZfP7553h6etKmTRubtGE2mxk5ciTPP/88LVq0sEkbN9u4cSO+vr40adKExx9/nPj4eJu1ZTabWblyJU2aNKF///74+vrSqVOnYrvnKtPFixdZuXIljz32mM3a6NatGz///DPnz59HKcWGDRs4duwY/fv3r/S2MjMzAfJ95jg4OGAymSrtMycpKQkALy8vwPafMze3pzVJbipBp06d+Oabb1izZg1ffPEFcXFxdOnShcuXL9ukvVOnTjF79mwaN27MmjVrGDduHOPHj+ebb76xSXs3Wr58OYmJiYwZM8ZmbbzwwguMGDGCZs2aYTQaadu2LRMmTGDEiBE2ac/d3Z3OnTvz3//+lwsXLpCbm8uCBQvYsWMHsbGxNmnzRnFxcQD4+fnl2+/n52e9rTrJyMjgxRdf5MEHH7TpCsUrVqzAzc0NJycnPvjgAyIjI/Hx8bFJW2+//TYGg4Hx48fb5PFvNnDgQBYuXMj69et577332LlzJ3fccYf1S7OyxcfHk5qayltvvcWAAQNYu3Ytw4YN45577mHTpk02afNGX3/9Ne7u7txzzz02a+Ojjz6iefPm1K9fH5PJxIABA5g1axbdunWr9LaaNWtGcHAwU6ZM4erVq2RlZfHWW28RFxdXKZ85SikmTZpEt27daNmyJWDbz5nC2tOaQesAqoOBAwdat1u1akXnzp1p2LAhX3/9NZMmTar09sxmM+3bt+fNN98EoG3bthw6dIjZs2czatSoSm/vRnPnzmXgwIEVrikozpIlS1iwYAGLFi2iRYsWREVFMWHCBAICAhg9erRN2vz222959NFHqVevHg4ODrRr144HH3yQPXv22KS9wtx8JkwpZbOzY1rJzs7mgQcewGw2M2vWLJu21atXL6KiokhISOCLL77g/vvvZ8eOHfj6+lZqO7t37+bDDz9kz549t+z3NXz4cOt2y5Ytad++PcHBwaxcudImCUBeMf9dd93FxIkTAQgPD2fbtm3MmTOHHj16VHqbN5o3bx4PPfRQuet7SuOjjz7ijz/+4OeffyY4OJjNmzfzr3/9C39/f/r06VOpbRmNRn788Ucee+wxvLy8cHBwoE+fPvm+Syri6aefZv/+/YWeBbLF50xx7WlFztzYgKurK61atapQ5Xtx/P39ad68eb59YWFh5R61UFpnzpxh3bp1jB071qbtPP/887z44os88MADtGrVipEjRzJx4kSmT59uszYbNmzIpk2bSE1N5ezZs/z5559kZ2fToEEDm7WZJ29k3c3/PcXHxxf4L8ueZWdnc//99xMdHU1kZKRNz9qA5X3YqFEjbrvtNubOnYvBYGDu3LmV3s6WLVuIj48nKCgIg8GAwWDgzJkzPPfcc4SEhFR6e4Xx9/cnODjYZp85Pj4+GAwGTT53tmzZwtGjR236uXPt2jVeeukl3n//fYYMGULr1q15+umnGT58OO+++65N2oyIiCAqKorExERiY2NZvXo1ly9frvBnzjPPPMPPP//Mhg0bqF+/vnW/rT5nimpPa5Lc2EBmZiZHjhzB39/fJo/ftWvXAkPujh07RnBwsE3ayzN//nx8fX0ZPHiwTdtJT09Hr8//p+ng4GDToeB5XF1d8ff35+rVq6xZs4a77rrL5m02aNCAunXrWkehgaVOZNOmTXTp0sXm7d8KeYnN8ePHWbduHd7e3rc8BqWUTbptRo4cyf79+4mKirJeAgICeP7551mzZk2lt1eYy5cvc/bsWZt95phMJjp06KDJ587cuXOJiIiwWb0UWP4+s7OzNfnc8fT0pE6dOhw/fpxdu3aV+zNHKcXTTz/N0qVLWb9+fYEkqbI/Z0pqT2vSLVUJJk+ezJAhQwgKCiI+Pp7/+7//Izk52WZdKBMnTqRLly68+eab3H///fz55598/vnnfP755zZpDyynpefPn8/o0aMxGGz7ZzNkyBDeeOMNgoKCaNGiBXv37uX999/n0UcftVmba9asQSlF06ZNOXHiBM8//zxNmzblkUceqZTHT01N5cSJE9br0dHRREVF4eXlRVBQEBMmTODNN9+kcePGNG7cmDfffBMXFxcefPBBm7R35coVYmJirHPN5H1p1a1bt1xzNBXXXkBAAP/4xz/Ys2cPK1asIDc31/rfo5eXFyaTqVLb8/b25o033mDo0KH4+/tz+fJlZs2axblz58o9fUFJr+fNyZrRaKRu3bo0bdq00tvz8vLitdde495778Xf35/Tp0/z0ksv4ePjw7Bhw8rVXkltBgUF8fzzzzN8+HBuv/12evXqxerVq/nll1/YuHGjTdoDSE5O5vvvv+e9994r9/MqbXs9evTg+eefx9nZmeDgYDZt2sQ333zD+++/b5P2vv/+e+rUqUNQUBAHDhzg2Wef5e67785X8FsWTz31FIsWLeKnn37C3d3d+h7z9PTE2dkZnU5XqZ8zJbUHVPrnTJncsnFZ1djw4cOVv7+/MhqNKiAgQN1zzz3q0KFDNm3zl19+US1btlSOjo6qWbNm6vPPP7dpe2vWrFGAOnr0qE3bUUqp5ORk9eyzz6qgoCDl5OSkQkND1dSpU1VmZqbN2lyyZIkKDQ1VJpNJ1a1bVz311FMqMTGx0h5/w4YNCihwGT16tFLKMkzz1VdfVXXr1lWOjo7q9ttvVwcOHLBZe/Pnzy/09ldffbXS28sbBlrYZcOGDZXe3rVr19SwYcNUQECAMplMyt/fXw0dOlT9+eef5WqrpPYKU9Gh4MW1l56ervr166fq1KmjjEajCgoKUqNHj1YxMTHlbq+kNvPMnTtXNWrUSDk5Oak2bdqo5cuX27S9zz77TDk7O1fKe7Gk9mJjY9WYMWNUQECAcnJyUk2bNlXvvfdeuae8KKm9Dz/8UNWvX9/6O/zPf/5Toc+4ot5j8+fPtx5TmZ8zpWmvsj9nykJ3PUghhBBCiGpBam6EEEIIUa1IciOEEEKIakWSGyGEEEJUK5LcCCGEEKJakeRGCCGEENWKJDdCCCGEqFYkuRFCCCFEtSLJjRBCCCGqFUluhBB2Z+PGjWVelDIkJKTcSwUIIeyLJDdCCLu3YcMGevXqhZeXFy4uLjRu3JjRo0eTk5OjdWhCCA1IciOEsGuHDh1i4MCBdOjQgc2bN3PgwAE+/vhjjEbjLVlJXghR9ciq4EIIuxYZGYm/vz8zZsyw7mvYsCEDBgzQMCohhJbkzI0Qwq7VrVuX2NhYNm/erHUoQogqQpIbIYRdu++++xgxYgQ9evTA39+fYcOG8cknn5CcnKx1aEIIjUhyI4Swaw4ODsyfP59z584xY8YMAgICeOONN2jRogWxsbFahyeE0IAkN0KIaqFevXqMHDmSTz/9lMOHD5ORkcGcOXO0DksIoQFJboQQ1U7t2rXx9/cnLS1N61CEEBqQ0VJCCLv22WefERUVxbBhw2jYsCEZGRl88803HDp0iI8//ljr8IQQGpDkRghh1zp27MjWrVsZN24cFy5cwM3NjRYtWrB8+XJ69OihdXhCCA1IciOEsGtt27bl22+/1ToMIUQVIjU3QgghhKhWJLkRQgghRLUiyY0Qwu6EhIQwYcKEMt1nwoQJZV5JXAhhn3RKKaV1EEIIIYQQlUXO3AghhBCiWpHkRgghhBDViiQ3QgghhKhWJLkRQgghRLUiyY0QQgghqhVJboQQQghRrUhyI4QQQohq5f8BpQqyCYkkxBYAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for i in range(3):\n",
    "    plt.plot(np.arange(22-5), deltas[i], label=\"\\u03B5 = \"+str(10**(-(i+1))))\n",
    "plt.legend()\n",
    "plt.xlabel(\"|S|\")\n",
    "plt.xticks(np.arange(22-5), np.arange(22-5)+5)\n",
    "plt.ylabel(\"Error \\u03B4\")\n",
    "plt.title(\"Scaling laws of Subset Sum Approximation\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 739,
   "id": "6aea22c5-83e4-43e2-a20b-c453fd976f67",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Set size: 15\n",
      "0.009509563446044922\n",
      "0.015209436416625977\n",
      "\n",
      "0.009105205535888672\n",
      "0.012586355209350586\n",
      "\n",
      "0.009000539779663086\n",
      "0.009999275207519531\n",
      "\n",
      "0.008000373840332031\n",
      "0.0050029754638671875\n",
      "\n",
      "0.009174823760986328\n",
      "0.005540609359741211\n",
      "\n",
      "0.01137542724609375\n",
      "0.011016130447387695\n",
      "\n",
      "0.010104179382324219\n",
      "0.014997720718383789\n",
      "\n",
      "0.010000228881835938\n",
      "0.0149993896484375\n",
      "\n",
      "0.008021354675292969\n",
      "0.0\n",
      "\n",
      "0.019041061401367188\n",
      "0.01907634735107422\n",
      "\n",
      "0\n",
      "1\n",
      "0\n"
     ]
    }
   ],
   "source": [
    "import time\n",
    "fail_count_1 = 0\n",
    "fail_count_2 = 0\n",
    "fail_count_3 = 0\n",
    "set_size = 15\n",
    "epsilon = 0.01\n",
    "print(\"Set size: \"+str(set_size))\n",
    "iterations = 10\n",
    "rand_vars = np.random.uniform(-1,1,(iterations, set_size))\n",
    "s_g = []\n",
    "subsets(rand_vars, np.zeros_like(rand_vars[0]))\n",
    "s_g = np.array(s_g).T\n",
    "for it in range(iterations):\n",
    "    # if(it%1 == 0):\n",
    "        # print(it)\n",
    "    \n",
    "    #print(\"Finished\")\n",
    "    s = np.sort(s_g[it]).reshape(-1)\n",
    "    #print(\"Started_1\")\n",
    "    start_time = time.time()\n",
    "    fail_1 = False\n",
    "    for i in range(1,int(2/epsilon)):\n",
    "        ij = (i*epsilon)-1\n",
    "        if(np.sum((abs(s-ij)<=epsilon))==0):\n",
    "            fail_count_1+=1\n",
    "            fail_1 = True\n",
    "            #print(i)\n",
    "            break\n",
    "    print(time.time()-start_time)\n",
    "    #if(fail_1):\n",
    "    #    continue\n",
    "    #print(\"Finished_1\")\n",
    "    start_ind = np.argwhere((s+1.0)>=0)[0][0]\n",
    "    end_ind = np.argwhere((s-1.0)<=0)[-1][0]\n",
    "    #print(start_ind)\n",
    "    #print(end_ind)\n",
    "    #print(\"Started_2\")\n",
    "    if(np.sum(abs(s[start_ind+1:end_ind] - s[start_ind:end_ind-1])>=2*epsilon)>0):\n",
    "            fail_count_2+=1\n",
    "    #print(start_ind)\n",
    "    #print(end_ind)\n",
    "    start_time = time.time()\n",
    "    k = 1\n",
    "    success = True\n",
    "    for i in range(start_ind, len(s)):\n",
    "        if(k>=int(2/epsilon)):\n",
    "            break\n",
    "        while(abs((k*epsilon-1)-s[i])<=epsilon):\n",
    "            k+=1\n",
    "        if(s[i]>(k*epsilon-1)):\n",
    "            success = False\n",
    "            #print(k)\n",
    "            #print()\n",
    "            break\n",
    "    if(not success):\n",
    "        fail_count_3 += 1\n",
    "    print(time.time()-start_time)\n",
    "    print()\n",
    "    # s_short = s[start_ind:end_ind]\n",
    "    # err = 0\n",
    "    # fail_2 = False\n",
    "    # for i in range(1,len(s_short)):\n",
    "    #     # print(i)\n",
    "    #     if(abs(s_short[i]-s_short[i-1])>err):\n",
    "    #         err = abs(s_short[i]-s_short[i-1])\n",
    "    #     if not (abs(s_short[i]-s_short[i-1])<=2*epsilon):\n",
    "    #         fail_count_2+=1\n",
    "    #         fail_2 = True\n",
    "    #         break\n",
    "    #print(\"Finished_2\")\n",
    "    # print(err)\n",
    "print(fail_count_1)\n",
    "\n",
    "print(fail_count_2)\n",
    "\n",
    "print(fail_count_3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "0fceb7e7-7c05-49b6-b1a8-3ba848eacdba",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Set size: 10\n",
      "1000\n",
      "0\n",
      "1000\n",
      "0\n",
      "1000\n",
      "0\n",
      "1000\n",
      "0\n",
      "Set size: 11\n",
      "1000\n",
      "0\n",
      "1000\n",
      "0\n",
      "1000\n",
      "0\n",
      "1000\n",
      "0\n",
      "Set size: 12\n",
      "1000\n",
      "0\n",
      "1000\n",
      "0\n",
      "1000\n",
      "0\n",
      "1000\n",
      "0\n",
      "Set size: 13\n",
      "995\n",
      "2\n",
      "993\n",
      "7\n",
      "986\n",
      "12\n",
      "1000\n",
      "0\n",
      "Set size: 14\n",
      "714\n",
      "105\n",
      "867\n",
      "51\n",
      "732\n",
      "77\n",
      "906\n",
      "64\n",
      "Set size: 15\n",
      "413\n",
      "69\n",
      "608\n",
      "36\n",
      "440\n",
      "52\n",
      "494\n",
      "81\n",
      "Set size: 16\n",
      "220\n",
      "40\n",
      "404\n",
      "43\n",
      "252\n",
      "50\n",
      "201\n",
      "50\n",
      "Set size: 17\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[1;32mIn[17], line 21\u001b[0m\n\u001b[0;32m     19\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m i \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(\u001b[38;5;241m1\u001b[39m,\u001b[38;5;28mint\u001b[39m(\u001b[38;5;241m2\u001b[39m\u001b[38;5;241m/\u001b[39mepsilon)):\n\u001b[0;32m     20\u001b[0m     ij \u001b[38;5;241m=\u001b[39m (i\u001b[38;5;241m*\u001b[39mepsilon)\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m\n\u001b[1;32m---> 21\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m(np\u001b[38;5;241m.\u001b[39msum((\u001b[38;5;28mabs\u001b[39m(s\u001b[38;5;241m-\u001b[39mij)\u001b[38;5;241m<\u001b[39m\u001b[38;5;241m=\u001b[39mepsilon))\u001b[38;5;241m==\u001b[39m\u001b[38;5;241m0\u001b[39m):\n\u001b[0;32m     22\u001b[0m         fail_count_1\u001b[38;5;241m+\u001b[39m\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m1\u001b[39m\n\u001b[0;32m     23\u001b[0m         fail_1 \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n",
      "\u001b[1;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "import time\n",
    "\n",
    "epsilon = 0.001\n",
    "iterations = 1000\n",
    "for set_size in range(10,20):\n",
    "    fail_count_1 = 0\n",
    "    fail_count_2 = 0\n",
    "    print(\"Set size: \"+str(set_size))\n",
    "    rand_vars = np.random.uniform(-1,1,(iterations, set_size))\n",
    "    s_g = []\n",
    "    subsets(rand_vars, np.zeros_like(rand_vars[0]))\n",
    "    s_g = np.array(s_g).T\n",
    "    #print(s_g.shape)\n",
    "    for it in range(iterations):\n",
    "        s = np.sort(s_g[it]).reshape(-1)\n",
    "        # print(s)\n",
    "        start_time = time.time()\n",
    "        fail_1 = False\n",
    "        for i in range(1,int(2/epsilon)):\n",
    "            ij = (i*epsilon)-1\n",
    "            if(np.sum((abs(s-ij)<=epsilon))==0):\n",
    "                fail_count_1+=1\n",
    "                fail_1 = True\n",
    "                break\n",
    "        if(fail_1):\n",
    "            continue\n",
    "        start_ind = np.argwhere((s+1.0)>=0)[0][0]\n",
    "        end_ind = np.argwhere((s-1.0)<=0)[-1][0]\n",
    "        if(np.sum(abs(s[start_ind+1:end_ind] - s[start_ind:end_ind-1])>=2*epsilon)>0):\n",
    "                fail_count_2+=1\n",
    "    print(fail_count_1)\n",
    "    print(fail_count_2)\n",
    "    \n",
    "    fail_count_1 = 0\n",
    "    fail_count_2 = 0\n",
    "    rand_vars = np.random.normal(0,1/3,(iterations, set_size))\n",
    "    # rand_vars = np.random.normal(0,1,(iterations, set_size))\n",
    "    s_g = []\n",
    "    subsets(rand_vars, np.zeros_like(rand_vars[0]))\n",
    "    s_g = np.array(s_g).T\n",
    "    for it in range(iterations):\n",
    "        s = np.sort(s_g[it]).reshape(-1)\n",
    "        start_time = time.time()\n",
    "        fail_1 = False\n",
    "        for i in range(1,int(2/epsilon)):\n",
    "            ij = (i*epsilon)-1\n",
    "            if(np.sum((abs(s-ij)<=epsilon))==0):\n",
    "                fail_count_1+=1\n",
    "                fail_1 = True\n",
    "                break\n",
    "        if(fail_1):\n",
    "            continue\n",
    "        start_ind = np.argwhere((s+1.0)>=0)[0][0]\n",
    "        end_ind = np.argwhere((s-1.0)<=0)[-1][0]\n",
    "        if(np.sum(abs(s[start_ind+1:end_ind] - s[start_ind:end_ind-1])>=2*epsilon)>0):\n",
    "                fail_count_2+=1\n",
    "    print(fail_count_1)\n",
    "    print(fail_count_2)\n",
    "    \n",
    "    fail_count_1 = 0\n",
    "    fail_count_2 = 0\n",
    "    rand_vars = np.random.normal(0,1/2,(iterations, set_size))\n",
    "    # rand_vars = np.random.normal(0,1,(iterations, set_size))\n",
    "    s_g = []\n",
    "    subsets(rand_vars, np.zeros_like(rand_vars[0]))\n",
    "    s_g = np.array(s_g).T\n",
    "    for it in range(iterations):\n",
    "        s = np.sort(s_g[it]).reshape(-1)\n",
    "        start_time = time.time()\n",
    "        fail_1 = False\n",
    "        for i in range(1,int(2/epsilon)):\n",
    "            ij = (i*epsilon)-1\n",
    "            if(np.sum((abs(s-ij)<=epsilon))==0):\n",
    "                fail_count_1+=1\n",
    "                fail_1 = True\n",
    "                break\n",
    "        if(fail_1):\n",
    "            continue\n",
    "        start_ind = np.argwhere((s+1.0)>=0)[0][0]\n",
    "        end_ind = np.argwhere((s-1.0)<=0)[-1][0]\n",
    "        if(np.sum(abs(s[start_ind+1:end_ind] - s[start_ind:end_ind-1])>=2*epsilon)>0):\n",
    "                fail_count_2+=1\n",
    "    print(fail_count_1)\n",
    "    print(fail_count_2)\n",
    "    \n",
    "    fail_count_1 = 0\n",
    "    fail_count_2 = 0\n",
    "    rand_vars = np.random.normal(0,1,(iterations, set_size))\n",
    "    # rand_vars = np.random.normal(0,1,(iterations, set_size))\n",
    "    s_g = []\n",
    "    subsets(rand_vars, np.zeros_like(rand_vars[0]))\n",
    "    s_g = np.array(s_g).T\n",
    "    for it in range(iterations):\n",
    "        s = np.sort(s_g[it]).reshape(-1)\n",
    "        start_time = time.time()\n",
    "        fail_1 = False\n",
    "        for i in range(1,int(2/epsilon)):\n",
    "            ij = (i*epsilon)-1\n",
    "            if(np.sum((abs(s-ij)<=epsilon))==0):\n",
    "                fail_count_1+=1\n",
    "                fail_1 = True\n",
    "                break\n",
    "        if(fail_1):\n",
    "            continue\n",
    "        start_ind = np.argwhere((s+1.0)>=0)[0][0]\n",
    "        end_ind = np.argwhere((s-1.0)<=0)[-1][0]\n",
    "        if(np.sum(abs(s[start_ind+1:end_ind] - s[start_ind:end_ind-1])>=2*epsilon)>0):\n",
    "                fail_count_2+=1\n",
    "    print(fail_count_1)\n",
    "    print(fail_count_2)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0955be93-65d1-47d4-9508-93e8a5103026",
   "metadata": {},
   "outputs": [],
   "source": [
    "Set size: 10\n",
    "712\n",
    "83\n",
    "\n",
    "798\n",
    "94\n",
    "\n",
    "946\n",
    "32\n",
    "\n",
    "986\n",
    "8\n",
    "Set size: 11\n",
    "440\n",
    "82\n",
    "\n",
    "461\n",
    "94\n",
    "\n",
    "628\n",
    "116\n",
    "\n",
    "807\n",
    "91\n",
    "Set size: 12\n",
    "228\n",
    "42\n",
    "\n",
    "243\n",
    "45\n",
    "\n",
    "256\n",
    "83\n",
    "\n",
    "384\n",
    "110\n",
    "Set size: 13\n",
    "161\n",
    "15\n",
    "\n",
    "96\n",
    "28\n",
    "\n",
    "100\n",
    "41\n",
    "\n",
    "141\n",
    "44\n",
    "Set size: 14\n",
    "84\n",
    "10\n",
    "\n",
    "70\n",
    "17\n",
    "\n",
    "51\n",
    "13\n",
    "\n",
    "71\n",
    "21\n",
    "Set size: 15\n",
    "37\n",
    "5\n",
    "\n",
    "41\n",
    "7\n",
    "\n",
    "20\n",
    "6\n",
    "\n",
    "30\n",
    "7\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 97,
   "id": "1832dcb4-08ae-4f61-b392-0a171d5a586b",
   "metadata": {},
   "outputs": [],
   "source": [
    "del s_g"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "36de3c6f-43da-4253-b4ab-d05398278b85",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.12.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
