{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "name": "Code_for_Neurips.ipynb",
      "provenance": [],
      "collapsed_sections": []
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    },
    "accelerator": "GPU"
  },
  "cells": [
    {
      "cell_type": "code",
      "metadata": {
        "id": "YZhlN1xA81nx",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "7b8ac67d-d8d0-4201-b9c8-d26abff8eea9"
      },
      "source": [
        "import pickle\n",
        "import torch\n",
        "import torch.nn as nn\n",
        "import math\n",
        "import torch.optim as optim\n",
        "import matplotlib.pyplot as plt\n",
        "import time\n",
        "import numpy as np\n",
        "import seaborn as sns\n",
        "torch.manual_seed(0)\n",
        "\n",
        "torch.cuda.is_available()\n",
        "\n",
        "if torch.cuda.is_available:\n",
        "  dev = 'cuda:0'\n",
        "else:\n",
        "  dev = 'CPU'\n",
        "\n",
        "device = torch.device(dev)\n",
        "\n",
        "from tqdm import tqdm\n",
        "!pip install tensorly\n",
        "import tensorly as tl\n",
        "tl.set_backend('pytorch')"
      ],
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "Collecting tensorly\n",
            "\u001b[?25l  Downloading https://files.pythonhosted.org/packages/42/3a/bf39a426e0d13a60d8cdee42419edeaa24aed2ec82f90a820b040e7db190/tensorly-0.6.0-py3-none-any.whl (160kB)\n",
            "\r\u001b[K     |██                              | 10kB 22.1MB/s eta 0:00:01\r\u001b[K     |████                            | 20kB 29.2MB/s eta 0:00:01\r\u001b[K     |██████▏                         | 30kB 34.0MB/s eta 0:00:01\r\u001b[K     |████████▏                       | 40kB 29.1MB/s eta 0:00:01\r\u001b[K     |██████████▏                     | 51kB 30.4MB/s eta 0:00:01\r\u001b[K     |████████████▎                   | 61kB 32.4MB/s eta 0:00:01\r\u001b[K     |██████████████▎                 | 71kB 33.8MB/s eta 0:00:01\r\u001b[K     |████████████████▎               | 81kB 28.7MB/s eta 0:00:01\r\u001b[K     |██████████████████▍             | 92kB 30.2MB/s eta 0:00:01\r\u001b[K     |████████████████████▍           | 102kB 28.1MB/s eta 0:00:01\r\u001b[K     |██████████████████████▍         | 112kB 28.1MB/s eta 0:00:01\r\u001b[K     |████████████████████████▌       | 122kB 28.1MB/s eta 0:00:01\r\u001b[K     |██████████████████████████▌     | 133kB 28.1MB/s eta 0:00:01\r\u001b[K     |████████████████████████████▌   | 143kB 28.1MB/s eta 0:00:01\r\u001b[K     |██████████████████████████████▋ | 153kB 28.1MB/s eta 0:00:01\r\u001b[K     |████████████████████████████████| 163kB 28.1MB/s \n",
            "\u001b[?25hRequirement already satisfied: scipy in /usr/local/lib/python3.7/dist-packages (from tensorly) (1.4.1)\n",
            "Requirement already satisfied: numpy in /usr/local/lib/python3.7/dist-packages (from tensorly) (1.19.5)\n",
            "Collecting nose\n",
            "\u001b[?25l  Downloading https://files.pythonhosted.org/packages/15/d8/dd071918c040f50fa1cf80da16423af51ff8ce4a0f2399b7bf8de45ac3d9/nose-1.3.7-py3-none-any.whl (154kB)\n",
            "\u001b[K     |████████████████████████████████| 163kB 41.0MB/s \n",
            "\u001b[?25hInstalling collected packages: nose, tensorly\n",
            "Successfully installed nose-1.3.7 tensorly-0.6.0\n"
          ],
          "name": "stdout"
        }
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "IvtV9Lo1KrER"
      },
      "source": [
        "def inner_product(data_tensor, weight_tensor):\n",
        "  \"\"\"\n",
        "  Computes the inner products of between a batch of tensors and a TT tensor\n",
        "\n",
        "  Parameters\n",
        "  -------\n",
        "  data_tensor: torch tensor\n",
        "               batch of input tensors of size n x d_1 x ... x d_p\n",
        "  weight_tensor: list\n",
        "               list of length p containing the cores of the TT tensor\n",
        "\n",
        "  Returns\n",
        "  ------\n",
        "  vector\n",
        "    Vector of size n containing the inner products \n",
        "  \"\"\"\n",
        "  W = tl.tt_to_tensor(weight_tensor)\n",
        "  return torch.matmul(data_tensor.reshape([data_tensor.shape[0],-1]),W.ravel())"
      ],
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "hHFvOKDc9CNm"
      },
      "source": [
        "class TTModel(nn.Module):      \n",
        "    \"\"\" \n",
        "    Class for a linear binary classifier parameterized by a TT tensor \n",
        "    \"\"\"\n",
        "    def __init__(self, order, dimension, rank, init='normal'): \n",
        "        \"\"\"\n",
        "        Parameters\n",
        "        ---------\n",
        "        order: int\n",
        "          Order of the TT tensor\n",
        "        dimension: int\n",
        "          Uniform dimension of the the TT tensor \n",
        "        rank: int\n",
        "          Uniform rank of the the TT tensor\n",
        "        init: string\n",
        "          initialization method for the core tensors of the TT parameters\n",
        "          choices are \"uniform\" and \"normal\"\n",
        "        \"\"\"\n",
        "        super().__init__()\n",
        "        self.order, self.dim, self.bond = order, dimension, rank\n",
        "\n",
        "        cores = [None for i in range(self.order)]\n",
        "        self.weights = [None for i in range(self.order)]\n",
        "        cores[0] = torch.Tensor(1,self.dim, self.bond)\n",
        "        for l in range(1,self.order-1):\n",
        "          cores[l] = torch.Tensor(self.bond, self.dim, self.bond)\n",
        "        cores[-1] = torch.Tensor(self.bond, self.dim,1)\n",
        "        for l in range(self.order):\n",
        "          self.weights[l] = nn.Parameter(cores[l])\n",
        "        self.params = nn.ParameterList(self.weights)\n",
        "\n",
        "        # initialize weights and biases\n",
        "        for weight in self.params:\n",
        "          if init=='normal':\n",
        "            torch.nn.init.normal_( weight, mean=0.0, std= .5)\n",
        "          elif init=='uniform':\n",
        "            torch.nn.init.uniform_(weight,-1,1)\n",
        "\n",
        "    def forward(self, x): \n",
        "        \"\"\"\n",
        "        Computes the forward pass\n",
        "\n",
        "        Parameters\n",
        "        ---------\n",
        "        self\n",
        "\n",
        "        x: Batches of n input tensors of order [order]\n",
        "            \n",
        "        Returns\n",
        "        ------\n",
        "        torch tensor of shape [n,2]\n",
        "          Tensor containing the scores of each input for the two classes\n",
        "    \n",
        "        \"\"\"\n",
        "        prod = inner_product(x, self.weights)\n",
        "        return torch.stack((prod,-1*prod),0).T  # w times x + b       \n",
        "\n"
      ],
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "21-m1zMME2Vv"
      },
      "source": [
        "def train_TT_model(training_data, rank, loss, optimizer, lr, epochs):\n",
        "  \"\"\"\n",
        "  Trains a TT classifier and returns the trained model.\n",
        "\n",
        "  Parameters\n",
        "  --------\n",
        "  training_data: tuple of two tensors, inputs (size n x d_1 x ... x d_p) and \n",
        "                 labels (size n vector of 0s and 1s)\n",
        "  rank: int\n",
        "        the model's rank\n",
        "  loss: \n",
        "        loss function.\n",
        "  optimizer:\n",
        "        pytorch optimizer \n",
        "  lr: float\n",
        "        learning rate\n",
        "  epochs: int\n",
        "        number of learning epochs\n",
        "\n",
        "  Returns\n",
        "  ------\n",
        "  TTModel\n",
        "    Trained model\n",
        "      \n",
        "  \"\"\"\n",
        "  inputs = training_data[0]\n",
        "  labels_train = training_data[1]\n",
        "  dimension = inputs.shape[1]\n",
        "  order = len(inputs.shape) - 1\n",
        "  model = TTModel(order, dimension, rank)#.to(device)\n",
        "  optimizer = optimizer(model.parameters(), lr)\n",
        "  for e in range(epochs):           \n",
        "    result = model(inputs)#.to(device)                                    \n",
        "    #pred = torch.argmax(result, axis = 1)\n",
        "    loss_train = loss(result, labels_train)\n",
        "    loss_train.backward()\n",
        "    optimizer.step()\n",
        "    optimizer.zero_grad()\n",
        "  return model\n"
      ],
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "nkIvoLhJcdzI"
      },
      "source": [
        "# First Experiment: Different Sample Sizes "
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "2RA7HYQh9D_H",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "c9482284-7b82-4060-88b0-edd6bf340bc1"
      },
      "source": [
        "dimension = 4\n",
        "order = 4\n",
        "target_rank = 8\n",
        "test_size = 4000\n",
        "n_runs = 20\n",
        "train_sample_sizes = [100,200,400,800,1400,2000,3000]  \n",
        "ranks = [2,4]\n",
        "learning_rate = 0.01\n",
        "epochs = 1000\n",
        "loss = nn.CrossEntropyLoss()\n",
        "optimizer = optim.SGD\n",
        "# generate target model and test data\n",
        "target_model = TTModel(order,dimension,target_rank,init='uniform')\n",
        "test_inputs = torch.randn([test_size] + [dimension]*order)\n",
        "test_labels = torch.argmax(target_model.forward(test_inputs),axis=1)\n",
        "\n",
        "\n",
        "\n",
        "results = []\n",
        "\n",
        "for n in tqdm(range(n_runs),position=0, leave=True):\n",
        "  result = {}\n",
        "  for rank in ranks:\n",
        "    result[rank] = []\n",
        "    for size in train_sample_sizes:\n",
        "      # generate training data\n",
        "      inputs = torch.randn([size]+[dimension]*order )\n",
        "      labels = torch.argmax(target_model.forward(inputs),axis=1)\n",
        "\n",
        "      # train the model\n",
        "      model = train_TT_model((inputs, labels), rank, loss, optimizer, lr=learning_rate, epochs=epochs)\n",
        "      test_loss = loss(model.forward(test_inputs), test_labels)\n",
        "      train_loss = loss(model.forward(inputs), labels)\n",
        "      result[rank].append((size,train_loss,test_loss))\n",
        "  results.append(result)              \n",
        "                                        "
      ],
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "100%|██████████| 20/20 [03:19<00:00,  9.98s/it]\n"
          ],
          "name": "stderr"
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "tuUTAsofTcaR"
      },
      "source": [
        "### Plotting the results"
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 251
        },
        "id": "sSARrtw9EOBa",
        "outputId": "d883d6dc-c0a6-4f10-cf22-255c9da28c9b"
      },
      "source": [
        "log_gen_gaps = {}\n",
        "upper_bound = {}\n",
        "with sns.axes_style('darkgrid'):\n",
        "  colors = sns.color_palette('husl' , 2)\n",
        "  figure = plt.figure(figsize=(4,2.3))\n",
        "  c = 0\n",
        "  for rank in ranks:\n",
        "    log_gen_gaps[rank] = np.array([[((losses[2]-losses[1]).detach()) for losses in result[rank]] for result in results])\n",
        "    #upper_bound[rank] = np.array( [(np.sqrt(order*dimension*rank**2 *math.log(sample_size)/sample_size))  for sample_size in train_sample_sizes])\n",
        "    n_params = order*dimension*rank**2\n",
        "    upper_bound[rank] = np.array( [2*(np.sqrt( n_params*math.log(8*sample_size*order*np.e / n_params)/sample_size))  for sample_size in train_sample_sizes])\n",
        "    ys = np.mean(log_gen_gaps[rank], axis = 0) #maybe add axis=1\n",
        "    stds = np.std(log_gen_gaps[rank], axis = 0)\n",
        "    print(ys)\n",
        "    plt.xlabel('sample size')\n",
        "    plt.ylabel('generalization')\n",
        "    plt.yscale('log')\n",
        "    plt.plot(train_sample_sizes, ys, color = colors[c]  , label ='experiment')\n",
        "    plt.fill_between(train_sample_sizes, list(torch.tensor(ys) - torch.tensor(stds)) , list(torch.tensor(ys) + torch.tensor(stds)), alpha = .3, facecolor = colors[c], label='_nolegend_')\n",
        "    plt.plot(train_sample_sizes,list(torch.tensor(upper_bound[rank])), color = colors[c] , linestyle ='dashed', label='_nolegend_') # The factor 1/4 is to reduce the distance between the bound curve from the theory and the one from our experiment.\n",
        "    c+=1\n",
        "  plt.legend(\"r=2 r=4\".split())"
      ],
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "[0.7026426  0.1959885  0.07610572 0.03401656 0.0130482  0.00936842\n",
            " 0.00555066]\n",
            "[3.4085011  0.699389   0.2122777  0.08208586 0.03694681 0.02745454\n",
            " 0.01571509]\n"
          ],
          "name": "stdout"
        },
        {
          "output_type": "display_data",
          "data": {
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAR0AAACqCAYAAACQ0psCAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAgAElEQVR4nOy9eZwU9Z3///x8qqrv7um5LwYUBETwCioqHhECKoooQhKNJl83idmYxDXZuBvX1Y0mq2tiNv7U3awx+0vUdePm8NhgvjGKV9R4axAvQDlmgJmBufvuqvp8/6ienhmYowd6ZoCp5+PRdFPVVfXumplXv9/vz+fzfgullMLFxcVlnJATbYCLi8vkwhUdFxeXccUVHRcXl3HFFR0XF5dxxRUdFxeXccUVHRcXl3FFn2gDxgLbtrEsZyaApon864nGtWVwXFsG52C0xTC0Ed9zSIqOZSk6OxMARKOB/OuJxrVlcFxbBudgtKWyMjzie9zwysXFZVxxRcfFxWVcmdSi053NTLQJLi6TjkkrOh2ZFFe8+iTff+9V1ne14S5Bc3EZHw7JRHIh6EJyYf10fr9zCy+3NTMzFOXC+hksrKhFl5NWi11cxpxJ+9cVNjxcftgcfn7SEq464hgSVpbbP3yDXekkAKZtT7CFLi6HJpPW0+nFp+ksqz2Mc2qmsTHWSa0/CMBN777C7nSSo0rKOSpSxtxIGdW+AEKICbbYxeXgZtKLTi9SCGaHS/P/P7Gsmjc7Wnlh13aeaN4KwMKKWq6bcyIAjYke6vwhNFeEXFxGxaQWHU0XWObgCeQL6qdzQf10bKXYlujhva42SjxeABKmydfeeAavpnNkuJSjSsqYGylnVjiKT5vUt9TFZUQm7V9IY6wL/59eo+bYeZh1NUO+TwrBYcEIhwUj/bbB387+BO91t/NuVxv/vfVDFPDl6fNYUT+d5mSch7d/RJXXT6U3QJXPT5U3QET5x+GTubgc2Exa0bl/43paZDc/v+8RzGn1ZBbOx5paBwWESz5N58yqKZxZNQWAWDbD+z0dTAs4U8B3pZP8add2eszsgONuO+l05npL+bC7gz80b6XK688LUpUvQIXX74ZrLoc8k1Z0jq2u4z9ibbzxyeM4/rUNBP77fzGn1JC66GxUKDCqc4UMDyeWVef/f3S0gl+eci4J06Q1nWBXOklrKsHskjJIKXZnkrzR0UJ7Jj3gPHd94kwOD5bwSlszf9q1nWpfYICnVOsPuqLkctAzaUXn9Mo67v14PY+VeZn99cvQ33offcNmVMAHgLZ1O3Y0gioZeQHbUAR0ncP0vtAs6vPTmUqwsKKOhRV1ZGyLXekku1JJWtMJan3OyFl7JsV73e08v2sHNn05p4dOPoeQ4eH/7tzCXzp3U+X1E/V4CekeQrrBKeU1CCGIm1k0IfBKzR1tczngmLSiU2J4Oba0kjfbW+gRNqH588jOn+fsVAr/w08gUmnskjDW1DrMqXVYh01BRUJFs8EjNer9Ier9A895bu1hnFt7GJayaUunaE0n2ZVOEtQNAGJmlo/jXbzS1kxWOfOJ/JrGqaeeB8C/b1rHc7u2owtBSDcI6R6qfQFumncyAE80b6Wn2US3IKQbhHUPpR4vs3Kjd6ZtuxMkXcaMSSs6AIuqpvBmeysv79rJkuppA5ZCJD63Am3rdrRtO9A3bsF450MyC44lvehUMC309zdhTa3bL09oJDQhqfI5+Z7+rG6YyeqGmSilSFgmMTNL0jL7fa4GDg9GiJnZ/MMr++qcvNzWzOvtLfQftzssEObu+WcB8PfrXuTjWJcjSIbjRc0KR/nSdEeUf79zC1nbJqwbOVEzKPf6qfaNLix1mZxMatFZWFnH3Rv/wrOt21lUPbVverYQ2FXl2FXlZE88BpRC7mpHGY6noTW34l/zNAB2NIw1tR5zah3mjGmQC8/GAyEEQd3Ie0C9zC+rYn5Z1ZDH/dPcBYRLfOzc3U1PTpRUPwlaUj2VHSUxR7Cyzv5svxnav23cREt6YG2Vk8tr+MejTgLgi689hWnbeUEK6R7ml1ayrO5wAH657UMMIfFpOj5NoywRoEx5OCwYQSlFUzKGV2r5/YaQbph4CDGpRcdAsqCihpdad7I7m6RKH2JIOydCvVj1NcS/+Ok+T2jDZox1H5C4bAVWoA7to20Y73yAXRJBRcPYJWHndXj8BGkkNCEJGR5ChmevfefUThv22J+euIi4aRIzM8TMLD1mlpDWJ3ynV9TRlc3k97ekE7RlUgDYSvHLrR+y5yKTFfXT+fL0eaRsi6++8cyAfRLBJdNmccnU2XRnM1z/zkv4NT0nTI44fbJyCvPLqujJZniieSteTccnNed9msbhwQgVXj9Z26Izm8GXO1Z3BW3cmdSiA/Cpmqk817KdZ5obuWTqbGy7gNXmQ3hCdnnU2Z1MoTXvRt+wGWH1+/O67kpAQ1+/Af3jRuycIKloxHkuCRc0ZD/RaEISMTxEBhEsgP9z+FFDHiuF4NHTlpOxLVKWRco28QQM7KQFOAtxr509n5RtkrIs0pZJyraYEy4DQClFlTdAyjZJWCbtmRQp2+LoEudLoS2T4hdb3t/rulfPPI6lNVP5ONbN3/7lT332IPBpGt+cdTynVNTybkcbt7/9Gh4p8UgNXUo8UvKZhllMD5WwOd7F0y1NGFJi5N7jEZKFlXWUeXy0phJ8HO/CEFpuv8SQGlP8IbyaRsoyydp27nhtUo5GTnrROS5SQZnHx0u7dnJRwxF49mUN7B6ekDlvFua8WaAUoieO7OpGdPYQjISgO4mIJ9CadqK/txGRyyMpIYj93ZUgBMZr65CtbaiSsDOCFu31lILF+tgTihQiFzrpgJdoJECn7YRrhpScWVU/5LElHi83zD1pyP3TAmF+c+qyvKA5wmVR5XO82Cqfn6tnHuvszwlayrLy+ShNCCKGh6xtk7RMsqZN1rZJ244o7kwm+L87t5Cx7QEji7MipZR5fLzduYs7N/5lL7v+7ROfZFowwhPN27j34/X57ZoQGELykxMWUen18/sdm1mzcwuGlAQMA2E79+TvjzyBgK7z4u4dvN2xC4/U8sJnSI2LpxyBJgQfdLfTnErkxc4QEq+mMSfiiHZnJo2pbAzh7PdIiSbEuHp7k150BILTq+r436aP2Zbo5gh/tIgnF6hICCsSggYQ0vnBZhccR3bBcWBZiO6cKMWTkBsxkt0x9I+3IWN9eRM7EiL+tcsB8Dz/KiKedMQoFET5vKhICLu6IvdmO3+uyYbYQ9D2pNTjY2nN0OHjkdGy/CjfYJxaUcupFc4ooaUcQcraNv7c8pdTymuZHirJb8/aNhnbotLriNrRJeV8efo8srbl7FM2WdsikDs+YniZ4g+RVTa2hKSVJZk1yf3qsDXew0ttO3PntTFzo5erpxwBwJMtjfm1gr34pMZvFjo23/vxep7btX3A/jKPl/sXnA3AHRve4p3OtgGeXI0vwD+ffPqQ92S0THrRAVhSM5XHmj7mqR3bOGJmKRQSYhUDTUOVRrBKIwM2pxefSnrxqWCaiK4eZGcPwrLy+2VHF9rmJmQyld9mTqkheflFAAT+81fI7pgjRn4fyufFmlpH5rQTADDeeg876EVXIr9fhQKogLtMYzRoQqJpEl+/Bghhw0N4iLATYHqohOmhkiH3n1ZZx2mVdcDgxdAvnTabS6fNzv/fVgpT2XlP5fPTjuSi+hlkbItsThTtfqOyZ9dM4+iSitw+Kx/q9XJ4MIKl1ADB1EVxv8Bc0QGmB0uYFgzzWnsrX7Cz+A+U26LrqPJSrPLSAZtTK5Y4L9IZRCKJSKbJfxUC2WPnILtjiFQakUxBKo1I95Vm9T77Z1QqQ3+Jyc6bRWr5YgCCP/kvlK7nRQufF3PGNMw5M0Ap9Pc3oXw+lN+bf8brOSjyUYcaUgg8ok/1Sjze/MLkwTgmWsEx0Yoh96+on1FU+wbjAPnrGppEIsFNN92EYRicdNJJXHDBBUW/hm0rPlk9hfs+fp+/dOzmlLLag6N8qdeD8npQAzWJ7EnHDntY7KrLKfFIelo7EcmUMwkylMsX2TbmtCmIVAqRTCO7ehDNu7AjIZgzA9IZ/I89tdc506edQOb0ExGJJP5f/94RI583J0xezJmHYddWQSqNvm0HytCdKQgeA2VlwBZgGHud1+XQY0JE57rrruPZZ5+lvLycNWvW5Lc///zz/PM//zO2bbN69WquvPJK/vjHP3L22WezaNEirrnmmqKKjhDQqy2Lqxt44OMPeLaliRPLqtE4hL+1vR5ENIAtBvnxS0l62SeHPtZjEP/yZxGpFCTTedGy6nJrzywL5fUikilkeycilYZUGhUOYtdWITu78f/2DwNOaQP6+Yswj56N1rQT/0NrUB4DDANl6OAxSJ91ClZDLbK1DePN9fl9ve8zZx2OCgcR8QSyvWvAPuVxxM31xA4MJkR0Vq5cyWWXXcbf//3f57dZlsXNN9/Mz3/+c6qrq1m1ahWLFi2ipaWF2bOdGFbTRu4eOBoMAZmc6FR4/BwVLePtjl30mFmi2tBx+aRGSuyK0iF3q3CI5GfPH7ix38RCuzxK/IpViGwWMiYimyVgSKwyZ3TFDgTIHj8XsllEJgtZE5HJonJ5B9ETQ/9wMyKbRWT7ZmEnqsqxwkG0jxvzEzf7E79iFXZNJfq6D/C+8LojZoaB8jjPqXPPRIWCqE3b8LyzETQNdA2l66BpZI85Egwdubsd0dkDem6/poOuOfdECMhmQeHsn6TJ/JGYENE58cQTaWpqGrBt3bp1TJs2jYaGBgDOO+881q5dS3V1Nc3NzcyZMwe7wLrFmiaIRnNDoJrMv96L7hiBSCD/y7Fs6uH8YN3rvN7dyqrDZ+3jpxvOrmFsGWcm1JbKgYlUTZOU9M5nigZgeu1eh+SzFPPnOA9A2QpME9IZQn4vQtdRx82C6lLIZFGZLOQe4YZKRDCAqi1HzWgYsI9EgkhJABEOwLpWvG+sh36CBhA49RiE34f90uuo517byz55yzcRmsR+9CnUn3ND5lI64uPzol3/FQDsP7yA2rAFDA10HXQdEfQjP32Os//VddDaDjkPLyIlhALI+XOdz7y5CZLpnOjpznl8XkRlbh5TIuXk93QdtOJNfCzm78sBk9NpaWmhpqavmFZ1dTXr1q3j8ssv53vf+x7PPvssZ511VkHnKrStsDeVRqWyZHKJtxNLqvBJjWeatnFGWR0eVdxvqoOxTex4sP+2CIhlgIzzumqQJSBZoDMBlZVw9pl777ec/dEzTqDnmKOcuNu2wbQQpolKWZBOII4+EnFYA8K0wLTAMhGmhdnjjCRq06YifX5nv2U5z0A69/kMIdF9Xue8yQxYCVQiRTK33/feZvRNW8C08nlFu6yE7hnOEhL/7/+Evm3HQNNrKkhcsRqAwM9/g9a8K79P6RrWtHqSn3aGzP3/8zginnBGTnPemjWllszC+QB4nn0ZkTUdD693f2U54RPmFK2tcEGik8lkeOKJJ9i+fTum2fcN8PWvf72Qw/eLQCDArbfeOjYnVwq6Y8gqH7atCEqD+WVVvNrWwpZYl7Pq2m0KMTkRwgmxNA3l7Qu1VTiECg9dacA6fArW4VOG3J+fozUEqQuX5F+XhH10tfUMCE9Ty85yRiUt0xE903K8ohyZBcciYgmEaeYE08LutyjZjkYQmsyJponI5eV60TdsduaHmVZ+mkZ23iw4Yc6QNo+WgkTnq1/9KuFwmLlz5+LxjE2uozeM6qWlpYXq6uphjthPMlnIZrFjSbRSE1vTsG3F4tqpvLh7J883b8evGzT4QnAQDGS5HHoITToJ8H6o0siwv47mUTOHPWf67OEn+SWuvKTfxRRYFigo5qrBgkSnpaWF//zP/yziZffm6KOPZsuWLTQ2NlJdXc3jjz/Oj370ozG7nm/NWrRYAvOUT6AlkohICKXg+GglpR4vz+/ewYxQCTKKU+/G9XhcJhsilxsqMgUlLY4//ng+/PDDol30W9/6Fp/97GfZvHkzZ5xxBr/+9a/RdZ0bb7yRL33pSyxbtoxzzz2XmTOHV+39wZpaj9zegmzrwO7qQebiZ6/U+D/Tj6I7m+GBrR+wsbODHan4gMl3Li4u+45QBcyCW7ZsGdu2baO+vn5AePW73/1uTI3bV7JZa+REciZL6O77sSpKyZw6H21KNRmf40TGVJY1jZv55bYPqfL6+fxhczgiWkqdN4DaD4/n0EreFg/XlsE5GG0pWiL53nvvLeRtBxceA/uY2WivvYPoiUFXEOn3YSsIagZHRctYZR/Brxs38eDWD7mMIxFRqPEE3ByPi8t+UFB4VV9fT09PD8888wzPPPMMPT091NcPXX7gYME69kgQAn3jVuxEEj3rtIyRSlDm9TErXMrKKTPYnozxy20f8nFXJ63ZpDux1cVlPyhIdO677z6+/e1v09bWRltbG9deey0PPPDAWNs29oQCWFPr0Lc0oRJJiCUQQqCUosYXIOL1MCdSxor66WxN9PDQtg181OkKj4vL/lBQePWb3/yGX/3qVwQCzozEL3/5y3zmM5/h8ssvH1PjxoPsrMPRt25H/6gROxxEKwljComuJDPCUTaqTo4uqcBSit/t2MyvGjfyGTETGRVUGv7CKg26uLjkKXjKbf91T8VeAzWRqJIwVnUFxkdbsRMptH4TpTxKMjMcJeQxOC5aybk109gY6+Q3jZv4qKuL3dmUW1/XxWWUFOTprFy5ktWrV7NkiTNb8qmnnuLiiy8eU8PGk+ysw/H96TW0xp2oSAgtGMDKOTAeJDMiUTZ1dXJCWTWmUjzZso2HGzdxMUcgS0op1b0HRykMF5cDgIJE54orruCkk07ijTfeAODWW2/lqKOGLr59sGFXlWOXhDE2bCZ1eANaJoPVr/qbT2kcURJlQ2cHJ5fXYCqbZ1qb0LcLLmQGR0RLKdU8uLrj4jIyw4pOLBYjFArR2dlJfX39gBGrzs5OotEi1hOeSIQgO+twvK+tQ+5sRZSFkRXlA/I1PqUxsyTKhq4OTquow7Rt/rR7B7qQLBcwM1pKiXSFx8VlJIYVnb/927/lnnvuYeXKlQNyF0ophBCsXbt2zA0cL6yGWux3PsTYsJlMQy0yWoItB+auAkLniEgpG7s6OLOyHlPZ/Lmt2emdhGBWtJSwNFzhcXEZhmFF55577gHg6af3Lop0SNA/CSwl5hHT8KzfALva0cJBKI9i0l9sISR1ZpaUsqGrg8VVDZi2zcvtzehSIoXg8EiUiG4gFa74uLgMQkGjV1/4whcK2nawYfl8yH5tgM3pU1GahrFhM2Z7F7J5F4ZtDTimT3iieHWds2umcXy0khd27+Cp5kY2dLbzflc7rdkkaSyku2bLxWUAw3o66XSaZDJJR0cHXV1d+RGaWCxGS0vLuBg4lpgIPJWliKYWlGWDx8A8fAr6R9sQ82ZhATJr4qmuIOvty9coBWFpMKOkhE2dnSyrPQxT2Ty7q4mElWVBWQ09mQweTSPi8VDpCxDW3aLjLi4wgug89NBD3HfffbS2trJy5cq86IRCIS677LJxMXCsMX0+jLII5q5O5/8zD0PftBXjvU1k5s/DTmUQ21vwVJWRDQXzLbGUghLpYXpJlI+7Ormgbjq6kLzW3sJr7S0cEYpyQlkVM4IltCWT+A2dqTKKF4Ffavu1cNTF5WCmoFXmDzzwwEE1+7igVeb90JWN2N6CnUwDYLz9PsamLZhTasiccLRTU0QKtPIoVmkES/WFTEIIOsw0H3V3Yto2Xdk0b3bs4q2OVuKWSanh5ROlVRwXraCyJEQmmSXs8VDpDxDWDHTkhMzxORhXMI8Hri2DU8xV5gWJDsCGDRvYtGkTmUxf07YLL7ywkEPHndGKDoAnlcLa3uqUhlQKfeMWjHUfoKIR0qd+It/9UouGsCvKMPt1PRRC0Gam2NzdhZkrLWkpm/e7O3ijo5VtiR40ITimrJLjwhXU+4NIIfDqOuU+H2VeHwFpjF9nUQ7OX+jxwLVlcMa9tMXdd9/NK6+8wkcffcSZZ57J888/z/z58w9Y0dkXTL8PIxrGbO8CITBnHY4dDuF95W18a18ifconsCtKsTpjyHQWT00FWcMZHldKUWH4UBHF5u4uLFuhCcm8knLmlZTTmkrwRkcr73S08VZbK9W+ACeUVjGvpJyUadKciBM2PFT6/UR0Lx4h3TVdLocsBY1ePfHEE9x3331UVFRw66238thjj9HT0zPWto0rtgKrrATp7Uv42rWVpBadgjJ0vM+9gra50dmeTGM3tWAkk/nRKdtWVOg+pkeihDyeAa36qnwBzq09jH847iSW1RyGUorHd27hjg1v84fmrTQnE3Sm02zs7GR95242J7qJkQXh9odzOfQoyNPxer1IKdF1nVgsRnl5OTt37hxr28YdU0i8VWXY21vzoY6KhEgtOhXvy2/hfWM92a4Y2WNmo7Im1o5dGBWlmCVhrNy8nHLdS2mJl24zQ3MiTncmk29g79V05pdV8YnSSpqSMV7vaOXNjlZea29hWiDMCWXVzA5HSZsWrYkEQY9Bpc9PieHFJzTX+3E5JChIdObNm0d3dzerV69m5cqVBAIBjj/++LG2DYDGxkZ+8pOfEIvFuPPOO8f8emYggFYSwuro58l5DNKnnYCx7kOMTVuQ3T2kTz7eGWJvbUfPZBEVUUwkSoHIjWyVRDz0WFlac55ML0IIGgJhGgJhllZP5e3OXbzZ0cpvmzYR0g2Oj1byidIqbKXoSTtD71Gvlwqfn5DmTjx0ObgpOJHcS1NTE7FYjCOPPHLE946mZ/lIXH311QWLzr4kkvtj2BaqcScqY+61T9vciOfNd1FBP+lT56MiTg8kLeSHqnIy2t46LiQkbJOYsGjq6CZjWXu9x1aKj2JdvNHRwsZYFwKYFS7lhNIqDg9GEEIghcCv61T6He8nIDUKbHq6FwdjknI8cG0ZnHFPJH/hC1/gr/7qrzjzzDOZMsVpJHbDDTfwve99b9jjRtOz3LIs/vVf/3XA8bfccgvl5eWFmFhUTE3DU12OtXM3yhwoENbhDaTDIbx/fhPf038mveA47NpKrFgSmXXm89heb76PFoCywY9ObUmEiNJpz6RoTSZJ9WtcKIVgZjjKzHCUjkyaNztaebtzFx/2dFDm8TK/tIpjo5XYShHPZjGkJOL1UOEL4JUaPqmh4VQ9dL0glwOZgkSnqamJe++9l3feeSff1XP9+vUjHjeanuVf+cpX8mu99peCe5kPR0kAQj5U827Y0+MJ1qDKP4l65hV8L76OmD8X5h7hLIrt6AJdQwR8EAqA35dvmKZpkupomGrCHG6ZtKVTNCfipExzQK33YNDLlNIIy+zpvNOxm5dbd/JkSyPPtG7n2PJKTq6spcEfJoWiKRNHEwJdSgK6QdhjENAMfJqOX9fQxOBjBW5f9cFxbRmcce9lHolE+MUvfsH3v/99/vqv/5of/vCH+3zBoXqWD0VHRwc//vGPee+997jnnnv4yle+MuI1Cu1lPjICT1kpNLdhxZN77NLgjJPwvP4O+hvvYu7qIDN/ntOKFqAzDrQhDB3p9yHCAfxlYbozVj4kCgnJ4UaYLpmmJRGnJ5PNJ517meUrYdbUElpyw+7r2nbxxu4Wan0BTiitZk6kDG/umu15q0HXJIbUCBkGYcODX9fxCIlHaKCgpMR/0Lnu44Fry+CMe3illELXdb773e/y8MMPc+mll9LV1VXIoftNaWkpN99887hcazAyUkevq0Lb1YbVGRu4U9fJLDgOu+QjPO9uRMTiZE4+Pj+REHBGubIx6I6hemLolkJEgiifF8twPKBS6aU07KXHztKSjNOVzuQnGfZS7QuwrPYwFlc1sK5rN290tPK7nZtZs3Mz1b6Ak5j2h2gIhIkYHrKWTdaySWSztJJACoEhJR5dI2x4qPLYWFh4hYYhpBuWuYwbBYnOZz/72fzrlStXMmvWLB588MF9uuC49ywvAiYCraoczTCw2joHzhwWAnPOEahICM+r6/D9/lns6grMhlqs+mow+i30tCyseBriSYQmkR7DKaER8GF5DCLSQyRkEA+YtKaSdKRSeyWdvZrGiWXVnFBaRWMyxkexLpoSPbzdsYvX2p1FuCWGhwZ/mIZAiCmBMFVeRwTTlkXasuhJZ+hSWVKJDIam4dN0wh6DkO7BIyVeNz/kMoYUVDnwnHPOobOzM799ypQpA5LDo2G8e5YXC0sJVGkJhqFjtbbvnWCuryG1JIK+pQmtcQfe199BvfkuVm0lVkMtVm3VgPcry0Yl05BMg5RIj44WCkDAT8TrIRIqIe4L0pZJsiuZILXH9YQQTA2EmRoI5+yzaUklaEzEaEz0sCXRzfruNsBplVzvD+a9ofpAiCBgKYVlmqRMk860U5DekBJdkwR1JywL6DoeoeGRGhLcuUIu+82wQ+a9yd1Fixbl+0HlDyygcuC3vvUtXn31VTo6OigvL+cb3/gGq1ev5rnnnuOWW27BsiwuvvhivvrVrxbvE7H/Q+bDIYTASKVQzbuxM9nB36QUsr0LrXEHeuNORDqD0jXEtDpStdXYVeUgh5gMLgTSoyOCfggGUD4vKQ3ac0nnZNYsqMGoUorObIbGRA9NyRjbEj3sSjt5KQHUBULU+4JM6ReSDWoOYGgaHikJGgZhjwef1PFKJz/UuwxkfzgYcxfjwcFoS1EXfB5MjKXo9OKxsqiWNux4avg3KoVsbUNv3Im+vRmyJsrrwZxSg9VQh10eHXqtg6BfIjpI2mfQicWOeJz4IEnnkUhZJk2JGI3JGDvScbbFejBzNTb6h2QNgTCVXj9yCLukEBiaxKtpRDxegrqBV2p4hYYuRh+WHYx/XOPBwWjLfovOu+++O+zBc+fOHfECE8F4iA44JTHk7va9E8xDEPTppD7ajta4A21HK8K2sQN+rIZazIZaVEl42MVWwtCQfh9EQ3Rqgh2ZNN2ZNFlr9DMEg0Ev3bEkzf1CsqZkjJjpeG9eqTHFH2JKToTq/UE8cuh+Z7qUuWF7nZDH4wiRkHiFhhxBiA7GP67x4GC0Zb9FZ7gaOkII7r///hEvMBGMl+gAaEKhd3RhtnWNWJoiFPQSi+eWQ4bpJKQAACAASURBVGRNtB0t6I07kS27EUphR0JOArqhDhUafk6EMDRE0E886CUmod3MErdsMpZZUIWMYNBLPJ4esM0JydKOCCV7aEzEBoRkNb2jZLncUHiIkKwXQ0oMTSNo6IR0D7qUaFKg4cyudsrZC0rCPhKxDJogJ1C99gCMbzL7YPxDHw/c8GoExlN0AKQAIxYfNMHcnwGi0590Gr2pGW3bTrS2DsBZ8W411GFOqXEmGA6HpiF9BhmvTspn0IWiw8ySUpC17UFzLoOJzmAk8yFZD02JGNuT8XxIFjU8TOk3VD9cSNYfAfllHUJAKOAlmcwihXBESUg8UqILmRcqXQwuWL1CJZFogv0WrIPxD308cIt4jcB4iw70JpiTqOa2IRPMQ4pO//MkkmiNO9EbdyA7e1A4zQCdIfia/OzmYc+ha+DzkPRopDwaHbZFj22SAmwlsJUqWHT2xFJ2LiTryYVlMeLWwJCsNy9UN0JI1stobNlTsCQi//9ecTL2FKzc/yWMKFiRiJ+ursQBMVVgUovOUEW8xmPV974wEaIDTjpGNy1ETwy7q2evBaOFiM6A83XH0Bt3oG3biYwnUFJg1VT1DcHrBfSUFwJp6GS9GilDIyYFncrC8mh0p7Mg5KgT0v1RStGRC8maEj00JvcMyYI0BEJUev0EdYOgphPQDYKagUdKhBD7LICjpRDBKgn5sNIWHqlhaBId6SwzEQIt97r3c4+1MB2qolPQ5MAnnniCxx57jAsvvJBbb72V3bt3c+211xZy6KRCKchqGqI0ihYJoceT2B1d2OkhhtZHOl8kRHbuLLJHzUR2dDnhV9NO9B0tKF3DqqvGqq/BqiobOAlxD6PsTBYtkyUIhDRJja4hIgYdaHRbNt3YpABTCJSUqFGEJUIIyjw+yjw+jo1WAP1CspwIvdnRijnICXUh8vOBfFIjqBmOIOk6Qc15DmiGI1a6PuQ6skJROGIxnMhmNDVAAKUQaDlhktKZ1e3VNLyajkdqeDTHi9IQeXHShcxNJdj/6QSHIm4RrzFAKYUpNaxICC0cQE8ksdu7970MoBDYZVHssijZY49E7mp3PKCmZvRtO1BCYJdHsaorsKsrsUsjQ15LWTZYNsGExI6niRo6ti5JGpKEgM5shriAjARbajkRGl39Hr+m51fMgxOSxcwsCdMkbmaJW1nipknCyhI3s6Sw6U6naU0liVtZrCEu5pNazkvSc0JkEOj3us+L0vFr+oCutPuK3V+kLEgO8h7Z6y3lhEnPzer2ahoezZlYqUvHU9Jz79NzRTsnIlk+0RzwRbwOZpQCE4kZCKIFAwhho23fjZVI7XsRdiGwq8rJVJXD8XORuzvQWnajtezG8+5GeHcjymNgVVfkH8MlolXWRGQhkISAFFQZOmlNkBKSnqxFl5UiJSCrSZSmoTQ97zEUiiYkJYaXEsM76P7+4ZVSirRtDRCluGWSyD3HzSwJM0tbJsW2RA8Ja++aR+CEUgNEqddz6ve6v2dVSO5pKHqFyQTIjSPsWcxXAFquC6yWC+e8mp4XJ0OTGEKi5bwmPbceTkpxyAnTiDkdpRTNzc3U1tYCoyviNVFMVE5nJKLRAD1dCbR0Gjp7sGMJx/MoFql0XoC0lt2ItJP0t0vCeQGyK0pB0wpLausa6BpJXZAU0GlmiFkmWSmwDB0lJbauA2K/8kL7k9OxlSJpmXlx6hWluNXnVfX3sDJDVD0zhCSo64Q9XgJSJ6QbAx5h3UMo51EVMkK3LwhA5kbvpBCEQ16slNXnMWlaTpj6vCZdSARj7zGNeyJ5+fLl/O53vyvMugOAA1l0em2RAvRsFtHdg9WdQGUH/8beZ5RCdHbnBUju7kAohdI07Moy9Kk1JEqjqFCwsLBPCIShYemSpCaI2xYdZpakaZKVgMeD0jVsTRt1cnq8EsngTCHIe1B5b6pPlFLKojOdJmZmSQ7hRQU1Z95RSDcIGf3FyUO432tjqKUuBTLUfekN47ScADmLdh1R8koNQ+aEKRfGabmRO9j3BPi4J5KPOuoo1q1bxzHHHFPI210KwFaQ0Q1kRTlaSQQZi6M6Y0Ov5xotwlmgapaWYB45A0wTrbUNmRMh9eo7+MGZEV1dgV1TgVVVPmxCWmVMZAaCQFBKqnUPaZ+PpITubIaudJKMZWFKieb1oAwdlc8LHRjhgSElJbKwUK83F5V/ZLP0mJkB21rTCeKmiT3Iijiv1PbyloL51wYhwxEun9RGlX+ylMKyFFlyXlt279+Z/mGcFBKP1PBqEq+m55PhktyonJAYwplGAGM/MleQp3POOeewbds26urq8Pv7asUcqN7PweDp7IkQAs220OIJ7PZu7HRm0PcVi6Btkt683RGh1jaEaTkJ6bJoXoTs0pKCk99C11C5UCyBojOdpsdMY1o2tq6Bx0BoGkoKp2i0wBEjIfAHvSQTzh/ORIvTvnhdSikSlpkTIkeUegaIVZ9QZQfpJ60JMcBb6n1dHgzgsfr2FTu065/81oTAyOWXvLnkt0fT8qNyFSUBYl0j35eihVfbt28fdHt9ff2IF5gIDkbR6UUI0JRCSySxO7qdVsdj8Fc4IKdj28i2TicMa9mF7Oh28gQeA6uqAqumAru6AjXSzOh+H0IaOtlcKNZjm3Sl06RNE8uyMJWNshVI59vV6zNIWzZIDXQJUiI0iZLSuSFCoIToEyxFzq8ovkCNZainlCJj24MI00DvKWZmSA5SvF8AAc0J6cI5EeqfbwoVMbTrpbf423F11ZAYOf9YtPCqvr6e119/na1bt3LxxRfT3t5OPB4v5FCXUeKMeAmnFU7Qj5ZMQ2c3VjzFPrd+GAkpsSvLsCvLYN4sSKfRWtry+SC9yZkeYUdCWNWVjgjlEtJDfQhnbhCEgLAmqdf9mAZkJZhSkBVOUbGkaWJLwExjZbKYSRu7/+eUIHKekRTCKQmi9T2E1FDaQHFCSEegesOF3L8THd4JIXKehJ9yr3/Y95q2jfIIWrvjOXHaW5haUgliZnbQUie9oV2vOIUH5KD6RGqk0M5WimyRf+8Kbiu8fv16Nm/ezMUXX0w2m+Xaa6/loYceKqoxLgOxlMDy+dDq/LkRr27sniKPeA2G14s1tQ5rap2TkO7qcQSoeTf6R1swNm52Rq4qy7BqKrCqK1HhoRPSyrJRVgYJeHMPIOfR6AQDPrrxDCpICTNL1rKxLAvLsrEGyV8AvdONEb0PKZxsvaY5j7z3JPK1jHrDO3KTDg+AlFMeXUqCXi9GYPhwqnf0boAwZbPErL6wbkcyTo/ZmV8z1x9NiFzy25MTp73DvIjhQSlFsQK7gkTnySef5NFHH+Wiiy4CnJKjrqczfli2wjI8yKoK9NIsojuG1R0v/ojXYAiBikYwoxHM2dPBNJG72vvmBv3lA+ADbL/PSUZXV2BVVRS0RgzbRtk2ZLLIZGZQQRK6B8vjTFbcU5BSlpUL2XKiZNtOTmjARQYRKYmTNJXOpD4nptUc7ynjQTdtR6BEv/xT3oMiH+LBxOegZG5Wd1A3gKErE/SGdgMT4QO9J2fuU/egoV2PNFlReXhRbC5IdAzDyH+DACQSB0aOZLLRN+JV1m/Eq6d4I16FoOvYtVXYtVVkARFP5EfEtMZm9M1NziLVsmg+F2SXDVOobChygqRwnJhBBUkzsAyRFyRTQMqySJkWScvcS5D6zu2IBXZ/gcoJuGlip/rdT+GERU741i/E6w3zpBzoRQ0I83J5KPryKxMV6vUP7SoKCO36C1PCMjmlqrZormBBonPuuedy44030t3dza9+9St++9vf8ulPf7o4FriMGttW2JqGiJagRUJosTiqo3uf13jtDyoYwJo+FWv6VCch3d6F1rIL2bwb471NiPc2oQwdOxpB+byDP3qXeBcqTMMIUomUCE1DaPq+CdJeH7B39nWvXEB+2vFg9Avz8iIlRJ9AaZqTQJdyQC4qnyzvF+5BTvDGWaR0KYl6vEQ9zl2VQlDlDxSUSC6EgktbvPjii7zwwgsAnHbaaSxcuLAoBozEU089xbPPPkssFmPVqlWcdtppIx5zMI9e7QtCgGbbaLFEwQtMR7vifZ9IZ9BanYS06IkjUmnnMYj7roQYKER+L3id5z1Fasj60iOR80iEFFhSkpEKU0pMoZyQzbJQGsSTGazc0gZlq9wiUTs3f0UVtzj9HiKV96qkxBvwkM7auYS541mpXFgIuZxU/v9jlzSXQnDilNqijV6NaRGvYvYy7+rq4rbbbuOWW24Z8b2TTXR6yYtPPIHdPrz4jIvoDIZSYFp9ApRK47MtMt3xAdtEKp1fxrHXKTzG0F5Tf9HS9cK9JykQmkYw6CWRcqbd2QJswJKOKOa3CYGNwrIVprL7npWNadtkbWeb3StSuec+AVMFrV3z+wySqeFzUr3PTl2V/uGe41EpIftESsi+kE+IgqceFFt0Cgqv/vjHP3L77bfT1taWV3shBG+++eawxxWzl/lPfvITPve5zxVi7qRFKTCFxAqH0IIB9ALEZ9wRAgzdma0cDjqbgl6ygwmgbSPSGUQyDXsKUu4hd3c4rwcJkZSUjqc0hMekfF7weVFeDyBRtglZDTvliJ3MPYb8IxngnUgQueFnPSdS/QTKFgJLgBJgI1AiN7NY2Vi5YWnLzj3nxMvwaIDEzolVn2DlvC5zj6T5UN8h/UQKQPYKUf/pB715qUGS5/tZUWQvChKdH/7wh/zHf/wHM2bMGNXJi9HLXCnF7bffzhlnnFFwIfii9DIfA8bdltIQ1JQ53UU7uqGf+EgpCAUHXwow3gxrS3j4pCfkci6ZrNNDLJnq90gjcs/EErC7fcA9GIDPC34vyu8j4HfESPi9zgr93D58PvB59qFkRu79+dRQ/2aNmqNsWs77QORDLqTADCgsch5XzvNyxMwZLrfUQG/LtHtfq3yp2j0fec9L2WDa+Rz6QHP7Jc91ichkKYmO7MUUQkGiU15ePmrBGYrR9jJ/4IEH+POf/0xPTw9bt27lkksuGfEaxetlXlwmyhahedDKygaEXRMWXg1C0WwxPM4jEhn6PZaV955EOu08p/o8KT2Txe7sRqQyg07GVEKA14Pyevo8pvxrj+NV9XpPXs++11DCuS+pPe5Lr/eVZ89ckBD5yZTkvC1L6ydWONvynhjKESqlMHuFy3K8rWxOzDBzRfvHc8HnvHnzuOaaa/jUpz6Fx9PXAWDp0qWFHL5ffP7zn+fzn//8mF/nUGZA2BUKoMeTkErBASI644qmoQL+Af3m++MJeknEc0tPTBORyuTyS+kBr8m9lj0xZ/toBWqPbfssUErlcjN93tOe6Rkt9xiSvFiRS9JriJznpQBbE4Q0g26rOOnfgkQnHo/j9/t58cUXB2zfF9E5GHuZHyr0FhWzQkH81aXoPl/f+i6XgQgBhoEyjHzuaUjGSKDskB9d04snUMPZr3KylRtY7C8vQgpkEcebChKdW2+9tWgXnKhe5pZl0tGxC9Mc29Xbw9HSIgoatRgNuu6htLQSTSvoRwnk1o/qOplQsC/h7IrPvjNGAkVjBs9gAgX5BPiYeVBjSEG/qZs3b+a73/0ubW1trFmzhg8++ICnn36aq666atjj+vcyP+OMM/K9zG+88Ua+9KUv5XuZz5w5sygfZjg6Onbh8wUIBmuKUjt3X9A0iVXEdVNKKeLxbjo6dlFRUbsPxzuLS609xSeVPrAWIh1KjEKgggEP8a54AQIVH3oEDw44gSpons5ll13G3/3d33HjjTfy6KOPAnD++ecPmHtzIDHYPJ3m5q1UV0+dMMGB4osOOMLT0rKNmpppozpuqKS2jkJL5Gr6jJP4HJJJ7SIwKlsK9KBEOlO4QPU++734Fi+gyzvyyGvREsnJZHKvqoHaUGUNDmAmUnDGimJ/JqesRhA90NvFosv1fA4GxiTE6/OgVGUpnHhcUUwtSHRKS0vZtm1b/hf8D3/4A5WVlUUxwGVoHnrov1iz5jE0TSMaLeW6626kpmb0YdS+kK/pEwigJxKu+BxKjFagbJvw0TMGXbC/LxQkOv/0T//EDTfcwMcff8zpp5/OlClTuP3224tjwSSld6KWHGYd0axZR/Kzn63C5/PxyCO/4d///U5uvrl4Sf1CsAArEEAL+NGTSey2LmfG7kRXxHIZH3IzyItJQWd76qmnOPPMM1mwYAG2bRMIBHjppZeYN28ec+bMKapBhzI7d+7gb/7mKo46ah4ffvgBt9/+/w3ruXziEyfkX8+dO48//vH342HmoFgILH8ArcGPnkg5ns8YlVJ1ObQpSHTWr1/P+vXrWbRoEUop/vd//5fZs2fz0EMPcc455/DlL395rO0sKvo7H2Ks+6Co58wecyTm0bNHfF9TUyPXX38T8+YdzY03Xse2bVv3es9nPnMp5557/oBta9Y8xoIFpxbN3n3FUgLL70eb4kNP5sQn4YqPS+EUJDrNzc08/PDDBINO/PeNb3yDr3zlKzz44IOsXLnyoBOdiaSmppZ5844GKDhUeuKJ3/PBB+9z990/HUvTRoVTStWPVu+Ij2rvwnLFx6UAChKdtra2AcsfDMNg9+7d+Hy+AdsPFsyjZxfklYwFPl9fR4VCPJ3XXnuF++///7n77p8ekPe6v/hoyRS0dzttk13xcRmCgkRn+fLlfPrTn2bx4sUAPP3005x//vkkEomiLQSdjIzk6WzY8AE//OEt/OhHd1FaWjZOVu0bA8UnDe1drvi4DEpBovO1r32NM844I18/56abbuLoo50QYTyWMExW/u3f7iSZTHLDDd8BnHVrt9324wm2anjyHSzqva74uAzKmFYOnCiGmpE82lm7xWYsZiQD+/TZxqvMhiZwwq7OblTGdLo/KOU858p+HrSzgMeYA8YWKQgfOY3OAubpFG1GsovLvmIpsHw+ZJ0foRTCshC2jbQV2BbCtBCGRDMSkDFRluV0/7RzXUAPve/ESY8rOi7jQr6YudScRz/80QCmPwmAUDbCsvPCJGwLTMvp8ZU1nWdXmA5qXNFxOSDo67AgBhUmIfrWmQ0Qppz3pLImmJYrTAcBrui4HBSofP8p2EuYDMDXJ0xC4BR1z3tMznOfMGVRWcsVpgnCFR2XQ4ZhhQlywiTy/e3IiVJ/YRK6QGuPoVJplGmNfd/4SYgrOi6TCmehbe//9hYmfzSA6fE7QmSaaKaJSqUhkcbOZlGmlR91c9k3itzRxmUsePbZtZx22gl88MF7E23KpMC2FRaCrG6Q9vnJlkYxp9TA1Dq0qbXodZVopRGk34swtAOuHOiBzgHv6Xz00Ufcd999dHZ2cvLJJ3PppZdOtElFoZDSFgCJRJxf//ohjjpq3jhZ5rInvWGbLWS+zY0IhZzGmtksmmlBOgOptBuWFcCYejrXXXcdp5xyCuefP3DF9PPPP8/ZZ5/NkiVL+OlPh1/EOGPGDG6++WbuuOOOETuKHujs3LmDSy5Zyfe+dyOXX/4ZWltbRjzm3nv/g8997gsH5LqryYxSubbCmk7G6yVbEsasqcSeWoeYWoveUI1WGUUL+hEePd9d02WMPZ1itRVeu3Ytv/zlL1mxYkVR7Frb0siTLduKcq5ellRPZXF1w4jvG01piw8//IDW1mZOPfU0/vu/7y+qvS7FpS+JLbB0A3QD4fcjygTSstBME5E1UYlUzhsyUaY9KUfMxlR0itFWGGDx4sUsXryYK6+8kuXLl4+lyWNOoaUtbNvmrrv+leuv/+44WeZSbAYNy4LBXFjmJKnJZCA5ucKycc/pjLat8CuvvMKTTz5JJpPhzDPPLOgag/Uyb2kRaJoTTS6tm8bSuolZh+X3+/N2/OM//v2gns5nP3sZZ5xxJps3f8w3vvEVANrb2/jOd77FD35wB3PmHDXg/UKIUfdIn9Q93odhwmyxbEeAsmZfH3bTIiTFxI+WCYGUgmh05L7yhXDAJ5IXLFjAggULRnXMYL3MlVJjsthytPS346abhi9t8fjjT+Vff/3rV/L1r1/DrFlH7vU5Cu0z3R+3x/vgTLwtEuELIPxBIgGDZEfMCcuSKVRygsIyKQjb49zLvJi4bYVdXIYnnx/SdTKHYFg27qIzUW2FDwRqa+t44IFf7dOxB1KpUpfxRymFpQBNA01D+LyIkghC2cisiTRNOEgmMY6p6BxIbYVdXA4lCh4ty4dlliNEB8Bo2ZiKzp5D4L2ceeaZBSeFXVxcCmPE0TLLdCYxTnBYdsAnkl1cXPadPcMyPF7kXmFZBhKpXFhmwyB9zovJpBIdpdQh18/8EKw26zLGOLWL9gzLokOGZcUWoUkjOrruIR7vJhiMHDLCo5QiHu9G190lEi77zkhhmbRMp7Vw1izK9SaN6JSWVtLRsYtYrHPCbBBCFN0z0XUPpaWVRT2ni8ueYVnA44GEKzqjQtN0KiqG7hs+Hkz8xDMXl4nHrafj4uIyrrii4+LiMq64ouPi4jKuHJIdPl1cXA5cXE/HxcVlXHFFx8XFZVxxRcfFxWVccUXHxcVlXHFFx8XFZVxxRcfFxWVcOWRFZzS9tYrFokWLWL58OStWrGDlypUAdHZ2csUVV7B06VKuuOIKurq6AGdty/e//32WLFnC8uXLeffdd/fr2oP1GNuXaz/yyCMsXbqUpUuX8sgjjxTNlrvuuovTTz+dFStWsGLFCp577rn8vnvuuYclS5Zw9tln86c//Sm/vRg/w507d3L55ZezbNkyzjvvPO677z5gYu7NULZMxL1Jp9OsWrWKCy64gPPOO48777wTgMbGRlavXs2SJUu45ppryGQyAGQyGa655hqWLFnC6tWrB3R5GcrGIVGHIKZpqsWLF6tt27apdDqtli9frjZu3Djm1z3rrLNUW1vbgG233Xabuueee5RSSt1zzz3qBz/4gVJKqWeffVZ98YtfVLZtq7feekutWrVqv6796quvqvXr16vzzjtvn6/d0dGhFi1apDo6OlRnZ6datGiR6uzsLIotd955p/rZz36213s3btyoli9frtLptNq2bZtavHixMk2zaD/DlpYWtX79eqWUUj09PWrp0qVq48aNE3JvhrJlIu6NbdsqFosppZTKZDJq1apV6q233lJXX321WrNmjVJKqRtuuEE9+OCDSiml/uu//kvdcMMNSiml1qxZo/7mb/5mWBuH45D0dPr31vJ4PPneWhPB2rVrufDCCwG48MILeeqppwZsF0Jw3HHH0d3dTWtr6z5f58QTT6SkpGS/rv3CCy+wcOFCotEoJSUlLFy4sLBvrgJsGYq1a9dy3nnn4fF4aGhoYNq0aaxbt65oP8Oqqirmzp0LQCgUYvr06bS0tEzIvRnKlom4N0IIgsEgAKZpYpomQghefvllzj77bAAuuuii/HmffvppLrroIgDOPvts/vznP6OUGtLG4TgkRWew3lrD/XCLyRe/+EVWrlzJ//zP/wDQ1tZGVVUVAJWVlbS1tQ1qY01NTdFtHO21x/q+PfjggyxfvpzrrrsuH84Mdc2xsKWpqYn333+fY489dsLvTX9bYGLujWVZrFixglNPPZVTTz2VhoYGIpEIuu4Un+j/O9nS0kJtrVOlQdd1wuEwHR0d+2TLISk6E8Uvf/lLHnnkEe69914efPBBXnvttQH7hRATVkBsIq8NcMkll/Dkk0/y2GOPUVVVxb/8y7+M6/Xj8ThXX301//AP/0AoFBqwb7zvzZ62TNS90TSNxx57jOeee45169bx8ccfj8t1D0nRmajeWr3XKC8vZ8mSJaxbt47y8vJ82NTa2kpZWdmgNjY3NxfdxtFeeyzvW0VFBZqmIaVk9erVvPPOO4Pa0nvNYtqSzWa5+uqrWb58OUuXLgUm7t4MZstE3huASCTCggULePvtt+nu7sY0nWJd/X8nq6ur2blzJ+CEYz09PZSWlu6TLYek6PTvrZXJZHj88cdZtGjRmF4zkUgQi8Xyr1988UVmzpzJokWLePTRRwF49NFHWbx4MUB+u1KKt99+m3A4nHf3i8Vor33aaafxwgsv0NXVRVdXFy+88AKnnXZaUWzpn6966qmn8m2HFi1axOOPP04mk6GxsZEtW7ZwzDHHFO1nqJTi+uuvZ/r06VxxxRX57RNxb4ayZSLuTXt7O93d3QCkUileeuklZsyYwYIFC3jiiScAZ7Su97yLFi3Kj9g98cQTnHzyyQghhrRxOA7JyoG6ro97b622tja+9rWvAU6sfP7553PGGWdw9NFHc8011/Cb3/yGuro67rjjDsBpw/Pcc8+xZMkS/H4/t9xyy35df7AeY1deeeWorh2NRrnqqqtYtWoVAF/72teIRqNFseXVV1/lgw8+AKC+vp6bb74ZgJkzZ3LuueeybNkyNE3jxhtvRNM0gKL8DN944w0ee+wxZs2axYoVK/L2TcS9GcqWNWvWjPu9aW1t5Tvf+Q6WZaGU4pxzzuGss87iiCOO4Jvf/CZ33HEHc+bMYfXq1QCsWrWKa6+9liVLllBSUsKPf/zjEW0cCre0hYuLy7hySIZXLi4uBy6u6Li4uIwrrui4uLiMK67ouLi4jCuu6Li4uIwrrui4HDBcfvnl+Ylx+8ratWvHraqAy75xSM7TcZm8LF68OD/Rz+XAxBUdlyFJJBJcc801NDc3Y9s2V111FcuWLePuu+/mmWeeIZ1Oc/zxx3PzzTcjhODyyy9nzpw5vP766ySTSW677TZ++tOfsmHDBs4991y++c1v0tTUxJe+9CXmzp3Le++9x8yZM7ntttvw+/0Drv3CCy9w1113kclkaGho4NZbb82viu7l/vvv56GHHkLTNI444gh+/OMf8/DDD7N+/XpuvPHG/AQ8gM2bN/Ozn/2MefPm8b3vfY+NGzdimiZf//rX+dSnPjUu99Mlx6iKcLhMKv7whz+o66+/Pv//7u5upZRTW6aXb3/722rt2rVKKaUuu+yyfF2aX/ziF2rhwoWqpaVFpdNpdfrpp6v29nbVEFa74wAAAvhJREFU2NioZs2apV5//XWllFLf+c538rVkLrvsMrVu3TrV1tamLr30UhWPx5VSTr2bu+66ay/7Fi5cqNLptFJKqa6uLqWUUr/97W/VTTfdNOB9a9euVZdcconKZDLqRz/6kXr00UfzxyxdujR/HZfxwc3puAzJrFmzeOmll/jhD3/I66+/TjgcBuCVV15h9erVLF++nJdffplNmzblj+ldqzNr1ixmzpxJVVVVvtZK78LA2tpa5s+fD8AFF1zAG2+8MeC6f/nLX9i0aROXXHIJK1as4NFHH2XHjh172Td79my+/e1v89hjjw059X7Lli384Ac/4I477uD/tXO/LqtDYRzAvwwVLAPFIlgmVsFgEPRPEAeCwaBFBDEN0WJSGAoGES3+qBpnccUkgk3QtmBRwR+Y/QELuhvudeALXvC9l73l+aSdnbPD4YSH5wyeYzabMZvN0Ov1wPM8EokEVFXVCxmJMeh4Rd7iOA7D4RDT6RSNRgOBQADpdBrlchmSJMHpdKLVakFVVf0bi8UCAGAYRn9+tp/Vy1+vkfja1jQNwWAQ9Xr9r+vrdruYz+eYTCZot9sYjUYv/dfrFYIgQBTFl2LaZrMJt9v9wU6Q/4kyHfLW6XSC1WoFz/NIpVJQFEUPMDabDdfrVa9I/sThcMByuQQAyLKsZz1PPp8Pi8UC2+0WwO9/S+v1+mXM4/HA8XhEIBBAPp/H+XzG7XZ7GVMsFhGNRuH3+/V3oVAI/X4f2p+SQ0VRPl4/+TeU6ZC3VqsVarUaGIaByWRCqVQCy7KIxWIIh8NwOBzwer0fz8txHAaDAYrFIjweD+Lx+Eu/3W5HtVpFLpfTLwYXBAEcx+lj7vc7CoUCLpcLNE1DMpkEy7J6/36/x3g8xmazgSRJAABRFJHNZlGpVBCJRPB4POByudDpdL6zPeSbqMqcGGq32yGTyUCW5Z9eCvkhdLwihBiKMh1CiKEo0yGEGIqCDiHEUBR0CCGGoqBDCDEUBR1CiKEo6BBCDPULqPmBgCYO7dgAAAAASUVORK5CYII=\n",
            "text/plain": [
              "<Figure size 288x165.6 with 1 Axes>"
            ]
          },
          "metadata": {
            "tags": []
          }
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "FvW_-4OArVcS"
      },
      "source": [
        "# Second Experiment: Different Ranks  "
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "kPVhX2f4pUXW"
      },
      "source": [
        "dimension = 4\n",
        "order = 4\n",
        "target_rank = 8\n",
        "test_size = 4000\n",
        "n_runs = 20\n",
        "train_sample_sizes = [2000, 4000]\n",
        "ranks = [2, 4, 6 , 8 , 10, 12, 14]\n",
        "learning_rate = 0.01\n",
        "epochs = 1000\n",
        "loss = nn.CrossEntropyLoss()\n",
        "optimizer = optim.SGD\n",
        "\n",
        "target_model = TTModel(order,dimension,target_rank,init='uniform')\n",
        "test_inputs = torch.randn([test_size] + [dimension]*order)\n",
        "test_labels = torch.argmax(target_model.forward(test_inputs),axis=1)\n",
        "\n",
        "\n",
        "results = []\n",
        "for n in range(n_runs):\n",
        "  result = {}\n",
        "  for sample_size in train_sample_sizes:\n",
        "    result[sample_size] = []\n",
        "    for rank in ranks:\n",
        "      # generate training data\n",
        "      inputs = torch.randn([sample_size] + [dimension]*order)\n",
        "      labels = torch.argmax(target_model.forward(inputs), axis = 1)\n",
        "      # train the model\n",
        "      model = train_TT_model((inputs, labels), rank, loss, optimizer, lr=learning_rate, epochs=epochs)\n",
        "      test_loss = loss(model.forward(test_inputs), test_labels)\n",
        "      train_loss = loss(model.forward(inputs), labels)\n",
        "      result[sample_size].append((sample_size,train_loss,test_loss))\n",
        "  results.append(result)        \n",
        "\n",
        "  "
      ],
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "A0kOpOEKTmA7"
      },
      "source": [
        "### Plotting the results"
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "D1LEDNS3eSsZ",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 240
        },
        "outputId": "0d83366c-d702-4e0b-e44f-2451680d8020"
      },
      "source": [
        "log_gen_gaps = {}\n",
        "upper_bound = {}\n",
        "with sns.axes_style('darkgrid'):\n",
        "  colors = sns.color_palette('husl' , 2)\n",
        "  figure = plt.figure(figsize=(4,2.3))\n",
        "  c = 0\n",
        "  for sample_size in train_sample_sizes:\n",
        "    log_gen_gaps[sample_size] = np.array([[((losses[2]-losses[1]).detach()) for losses in result[sample_size]] for result in results])\n",
        "    n_params = order*dimension\n",
        "    upper_bound[sample_size] = np.array( [2*(np.sqrt( n_params*rank**2*math.log(8*sample_size*order*np.e / (n_params*rank**2))/sample_size))  for rank in ranks])\n",
        "    ys = np.mean(log_gen_gaps[sample_size], axis = 0) \n",
        "    stds = np.std(log_gen_gaps[sample_size], axis = 0)\n",
        "    print(ys)\n",
        "    plt.xlabel('rank')\n",
        "    plt.ylabel('generalization')\n",
        "    plt.yscale('log')\n",
        "    plt.plot(ranks, ys, color = colors[c]  , label ='experiment')\n",
        "    plt.fill_between(ranks, list(torch.tensor(ys) - torch.tensor(stds)) , list(torch.tensor(ys) + torch.tensor(stds)), alpha = .3, facecolor = colors[c], label='_nolegend_')\n",
        "    plt.plot(ranks,list(torch.tensor(upper_bound[sample_size])), color = colors[c] , linestyle ='dashed', label='_nolegend_') # The factor 1/4 is to reduce the distance between the bound curve from the theory and the one from our experiment.\n",
        "    c+=1\n",
        "    plt.tight_layout()\n",
        "\n",
        "  plt.legend(\"n=2000 n=4000\".split())"
      ],
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "[0.00640478 0.02303103 0.03646769 0.05027365 0.07151561 0.08757969\n",
            " 0.10857521]\n",
            "[0.00557374 0.01454905 0.01445679 0.02332144 0.02879337 0.03536797\n",
            " 0.03862754]\n"
          ],
          "name": "stdout"
        },
        {
          "output_type": "display_data",
          "data": {
            "image/png": "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\n",
            "text/plain": [
              "<Figure size 288x165.6 with 1 Axes>"
            ]
          },
          "metadata": {
            "tags": []
          }
        }
      ]
    }
  ]
}