{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "65245462",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "env: TF_FORCE_UNIFIED_MEMORY=1\n"
     ]
    }
   ],
   "source": [
    "%env TF_FORCE_UNIFIED_MEMORY=1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "74d4816a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# sine config (from chelsea finn)\n",
    "config = {}\n",
    "config[\"n_epochs\"] = 70000\n",
    "config[\"n_tasks_per_epoch\"] = 24\n",
    "config[\"K\"] = 10\n",
    "config[\"L\"] = 10\n",
    "config[\"n_updates\"] = 5\n",
    "config[\"n_updates_test\"]= 10\n",
    "config[\"meta_lr\"] = 0.001\n",
    "config[\"inner_lr\"] = 1e-3\n",
    "config[\"data_noise\"] = 0.05 # not in cbfinn but we add it to compare to our algorithm\n",
    "config[\"n_test_tasks\"] = 100"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "38465e3b",
   "metadata": {},
   "outputs": [],
   "source": [
    "from jax import vmap\n",
    "from jax import numpy as np\n",
    "from flax import struct\n",
    "from typing import Any, Callable\n",
    "from flax import core\n",
    "import optax\n",
    "import models\n",
    "from jax import random\n",
    "import dataset_sines_finite\n",
    "from jax import value_and_grad, grad\n",
    "from functools import partial\n",
    "from jax.tree_util import tree_map\n",
    "from jax import jit\n",
    "import time\n",
    "from matplotlib import pyplot as plt\n",
    "from jax.lax import scan\n",
    "import pickle\n",
    "import dataset_sines_infinite\n",
    "import pickle"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b501dcce",
   "metadata": {},
   "source": [
    "## Inner and outer loss"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "57bfab88",
   "metadata": {},
   "outputs": [],
   "source": [
    "def error_fn(predictions, gt):\n",
    "    return np.mean( (predictions - gt)**2 )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "367d8046",
   "metadata": {},
   "outputs": [],
   "source": [
    "def inner_loss(current_params, x_a, y_a, apply_fn):\n",
    "    predictions = apply_fn(current_params, x_a)\n",
    "    \n",
    "    return error_fn(predictions, y_a)\n",
    "\n",
    "def gd_step0(inner_lr, param_value, param_grad):\n",
    "    return param_value - inner_lr * param_grad\n",
    "\n",
    "def inner_updates(current_params, x_a, y_a, n_updates, inner_lr, apply_fn):\n",
    "    def f(parameters, x):\n",
    "        inner_gradients = grad(inner_loss)(parameters, x_a, y_a, apply_fn)\n",
    "        parameters = tree_map(partial(gd_step0, inner_lr), parameters, inner_gradients)\n",
    "        \n",
    "        return parameters, None\n",
    "    \n",
    "    updated_params, _ = scan(f, current_params, None, n_updates)\n",
    "    \n",
    "    return updated_params"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "e73df2ff",
   "metadata": {},
   "outputs": [],
   "source": [
    "def outer_loss_single_task(current_params, x_a, y_a, x_b, y_b, n_updates, inner_lr, apply_fn):\n",
    "    updated_params = inner_updates(current_params, x_a, y_a, n_updates, inner_lr, apply_fn)\n",
    "    \n",
    "    predictions = apply_fn(updated_params, x_b)\n",
    "    \n",
    "    return error_fn(predictions, y_b)\n",
    "\n",
    "def outer_loss(current_params, x_a, y_a, x_b, y_b, n_updates, inner_lr, apply_fn):\n",
    "    unaveraged_losses = vmap(partial(outer_loss_single_task, current_params=current_params, n_updates=n_updates, inner_lr=inner_lr, apply_fn=apply_fn))(x_a=x_a, y_a=y_a, x_b=x_b, y_b=y_b)\n",
    "    return np.mean(unaveraged_losses)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "398ed5a0",
   "metadata": {},
   "source": [
    "## Dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "cb9e02e3",
   "metadata": {},
   "outputs": [],
   "source": [
    "# uncomment this cell to train on finite dataset\n",
    "#dataset_sines_finite.init_dataset(random.PRNGKey(0), 0.05, _n_train_tasks=10)\n",
    "#config[\"n_tasks_per_epoch\"] = 6"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "3401ab63",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_train_batch_fn(key):\n",
    "    # uncomment this line to train on infinite dataset\n",
    "    return dataset_sines_infinite.get_test_batch(key, config[\"n_tasks_per_epoch\"], config[\"K\"], config[\"L\"], config[\"data_noise\"])\n",
    "\n",
    "    # uncomment this line to train on finite dataset\n",
    "    #return sine_dataset_offset_finite.get_train_batch_as_val_batch(key, config[\"n_tasks_per_epoch\"], config[\"K\"], config[\"L\"], config[\"data_noise\"])\n",
    "    \n",
    "def get_test_batch_fn(key):\n",
    "    return dataset_sines_infinite.get_test_batch(key, config[\"n_test_tasks\"], config[\"K\"], config[\"L\"], config[\"data_noise\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "737f51e8",
   "metadata": {},
   "source": [
    "## State"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "584350e9",
   "metadata": {},
   "outputs": [],
   "source": [
    "class TrainStateNoProj(struct.PyTreeNode):\n",
    "    step: int\n",
    "    apply_fn: Callable = struct.field(pytree_node=False)\n",
    "    params: core.FrozenDict[str, Any]\n",
    "    tx_params: optax.GradientTransformation = struct.field(pytree_node=False)\n",
    "    opt_state_params: optax.OptState\n",
    "    inner_lr: float\n",
    "    n_updates: int\n",
    "    \n",
    "    def apply_gradients(self, *, grads_params, **kwargs):\n",
    "        \"\"\"\n",
    "        Updates both the params and the scaling matrix\n",
    "        Also requires new_batch_stats to keep track of what has been seen by the network\n",
    "        \"\"\"\n",
    "        # params part\n",
    "        updates_params, new_opt_state_params = self.tx_params.update(grads_params, self.opt_state_params, self.params)\n",
    "        new_params = optax.apply_updates(self.params, updates_params)\n",
    "\n",
    "        return self.replace(\n",
    "            step=self.step + 1,\n",
    "            params=new_params,\n",
    "            opt_state_params=new_opt_state_params,\n",
    "            **kwargs,\n",
    "        )\n",
    "\n",
    "\n",
    "    @classmethod\n",
    "    def create(cls, *, apply_fn, params, tx_params, inner_lr, n_updates, **kwargs):\n",
    "        opt_state_params = tx_params.init(params)\n",
    "        return cls(\n",
    "            step=0,\n",
    "            apply_fn=apply_fn,\n",
    "            params=params,\n",
    "            tx_params=tx_params,\n",
    "            opt_state_params=opt_state_params,\n",
    "            inner_lr = inner_lr,\n",
    "            n_updates = n_updates,\n",
    "            **kwargs,\n",
    "        )"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4c42d6d6",
   "metadata": {},
   "source": [
    "## Step"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "b337913e",
   "metadata": {},
   "outputs": [],
   "source": [
    "@partial(jit, static_argnums=(5, 6, 7))\n",
    "def get_loss_and_gradients(params, x_a, y_a, x_b, y_b, n_updates, inner_lr, apply_fn):\n",
    "    return value_and_grad(outer_loss)(params, x_a, y_a, x_b, y_b, n_updates, inner_lr, apply_fn)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "a9cbd323",
   "metadata": {},
   "outputs": [],
   "source": [
    "def step(key, state):\n",
    "    x_a, y_a, x_b, y_b = get_train_batch_fn(key)\n",
    "    \n",
    "    loss, gradients = get_loss_and_gradients(state.params, x_a, y_a, x_b, y_b, state.n_updates, state.inner_lr, state.apply_fn)\n",
    "    \n",
    "    state = state.apply_gradients(grads_params = gradients)\n",
    "    \n",
    "    return state, loss"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "17029a09",
   "metadata": {},
   "source": [
    "## Test during training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "86bd7c3d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def test_during_training(key, state):\n",
    "    x_a, y_a, x_b, y_b = get_test_batch_fn(key)\n",
    "    \n",
    "    def f(carry, task):\n",
    "        x_a, y_a, x_b, y_b = task\n",
    "        \n",
    "        updated_params = inner_updates(state.params, x_a, y_a, config[\"n_updates_test\"], state.inner_lr, state.apply_fn)\n",
    "        predictions = state.apply_fn(updated_params, x_b)\n",
    "        \n",
    "        return None, error_fn(predictions, y_b)\n",
    "    \n",
    "    _, errors = scan(f, None, (x_a, y_a, x_b, y_b))\n",
    "    return np.mean(errors)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "3ba73c78",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2022-08-27 14:06:45.085629: W external/org_tensorflow/tensorflow/stream_executor/platform/default/dso_loader.cc:64] Could not load dynamic library 'libcuda.so.1'; dlerror: libcuda.so.1: cannot open shared object file: No such file or directory; LD_LIBRARY_PATH: /state/partition1/llgrid/pkg/anaconda/anaconda3-2022a/pkgs/cudatoolkit-11.3.1-h2bc3f7f_2/lib\n",
      "2022-08-27 14:06:45.085662: W external/org_tensorflow/tensorflow/stream_executor/cuda/cuda_driver.cc:269] failed call to cuInit: UNKNOWN ERROR (303)\n",
      "WARNING:absl:No GPU/TPU found, falling back to CPU. (Set TF_CPP_MIN_LOG_LEVEL=0 and rerun for more info.)\n"
     ]
    }
   ],
   "source": [
    "key = random.PRNGKey(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "75c08a80",
   "metadata": {},
   "outputs": [],
   "source": [
    "model = models.small_network(40, \"relu\", 1)\n",
    "\n",
    "def apply_fn(params, inputs):\n",
    "    return model.apply({\"params\": params}, inputs)\n",
    "\n",
    "key, key_init0, key_init1 = random.split(key, 3)\n",
    "batch = get_train_batch_fn(key_init0)\n",
    "init_vars = model.init(key_init1, batch[0][0])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c4aa799f",
   "metadata": {},
   "source": [
    "## Option #1: Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "6ad756ca",
   "metadata": {},
   "outputs": [],
   "source": [
    "optimizer_params = optax.adam(learning_rate = config[\"meta_lr\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "8ee5691d",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0 | 6.9285 (2.7906 s / epoch)\n",
      "Error: 5.742314338684082\n",
      "100 | 4.8379 (0.0285 s / epoch)\n",
      "200 | 3.2989 (0.0289 s / epoch)\n",
      "300 | 2.5574 (0.0318 s / epoch)\n",
      "400 | 3.4344 (0.0417 s / epoch)\n",
      "500 | 4.0051 (0.0222 s / epoch)\n",
      "Error: 2.867670774459839\n",
      "600 | 3.1906 (0.0230 s / epoch)\n",
      "700 | 2.8186 (0.0259 s / epoch)\n",
      "800 | 2.8602 (0.0340 s / epoch)\n",
      "900 | 2.3430 (0.0352 s / epoch)\n",
      "1000 | 2.2166 (0.0275 s / epoch)\n",
      "Error: 1.8135133981704712\n",
      "1100 | 2.2618 (0.0293 s / epoch)\n",
      "1200 | 2.0090 (0.0314 s / epoch)\n",
      "1300 | 1.8723 (0.0312 s / epoch)\n",
      "1400 | 1.5975 (0.0328 s / epoch)\n",
      "1500 | 1.3584 (0.0382 s / epoch)\n",
      "Error: 1.0798664093017578\n",
      "1600 | 0.8626 (0.0258 s / epoch)\n",
      "1700 | 1.2260 (0.0369 s / epoch)\n",
      "1800 | 0.6464 (0.0223 s / epoch)\n",
      "1900 | 0.7116 (0.0339 s / epoch)\n",
      "2000 | 0.5786 (0.0244 s / epoch)\n",
      "Error: 0.8728893995285034\n",
      "2100 | 0.9676 (0.0370 s / epoch)\n",
      "2200 | 0.6963 (0.0262 s / epoch)\n",
      "2300 | 0.6230 (0.0409 s / epoch)\n",
      "2400 | 0.6647 (0.0319 s / epoch)\n",
      "2500 | 0.6583 (0.0374 s / epoch)\n",
      "Error: 0.5990662574768066\n",
      "2600 | 0.6036 (0.0273 s / epoch)\n",
      "2700 | 0.7255 (0.0411 s / epoch)\n",
      "2800 | 0.5672 (0.0305 s / epoch)\n",
      "2900 | 0.3776 (0.0402 s / epoch)\n",
      "3000 | 0.4132 (0.0292 s / epoch)\n",
      "Error: 0.4509776830673218\n",
      "3100 | 0.7281 (0.0225 s / epoch)\n",
      "3200 | 0.2993 (0.0448 s / epoch)\n",
      "3300 | 0.6759 (0.0372 s / epoch)\n",
      "3400 | 0.3636 (0.0388 s / epoch)\n",
      "3500 | 0.4313 (0.0382 s / epoch)\n",
      "Error: 0.524134635925293\n",
      "3600 | 0.5313 (0.0251 s / epoch)\n",
      "3700 | 0.4552 (0.0321 s / epoch)\n",
      "3800 | 0.5812 (0.0306 s / epoch)\n",
      "3900 | 0.5788 (0.0251 s / epoch)\n",
      "4000 | 0.3653 (0.0389 s / epoch)\n",
      "Error: 0.39830413460731506\n",
      "4100 | 0.3822 (0.0319 s / epoch)\n",
      "4200 | 0.3260 (0.0408 s / epoch)\n",
      "4300 | 0.4364 (0.0304 s / epoch)\n",
      "4400 | 0.6231 (0.0233 s / epoch)\n",
      "4500 | 0.4446 (0.0258 s / epoch)\n",
      "Error: 0.3106500506401062\n",
      "4600 | 0.5084 (0.0289 s / epoch)\n",
      "4700 | 0.4967 (0.0523 s / epoch)\n",
      "4800 | 0.7953 (0.0370 s / epoch)\n",
      "4900 | 0.5721 (0.0398 s / epoch)\n",
      "5000 | 0.3956 (0.0424 s / epoch)\n",
      "Error: 0.3179621398448944\n",
      "5100 | 0.3476 (0.0369 s / epoch)\n",
      "5200 | 0.4843 (0.0285 s / epoch)\n",
      "5300 | 0.5307 (0.0289 s / epoch)\n",
      "5400 | 0.4157 (0.0404 s / epoch)\n",
      "5500 | 0.4641 (0.0411 s / epoch)\n",
      "Error: 0.2871239483356476\n",
      "5600 | 0.3660 (0.0400 s / epoch)\n",
      "5700 | 0.3932 (0.0224 s / epoch)\n",
      "5800 | 0.3259 (0.0266 s / epoch)\n",
      "5900 | 0.3159 (0.0388 s / epoch)\n",
      "6000 | 0.4614 (0.0300 s / epoch)\n",
      "Error: 0.3875267803668976\n",
      "6100 | 0.3233 (0.0278 s / epoch)\n",
      "6200 | 0.3391 (0.0380 s / epoch)\n",
      "6300 | 0.3412 (0.0383 s / epoch)\n",
      "6400 | 0.3567 (0.0367 s / epoch)\n",
      "6500 | 0.5509 (0.0353 s / epoch)\n",
      "Error: 0.22064834833145142\n",
      "6600 | 0.3316 (0.0395 s / epoch)\n",
      "6700 | 0.2780 (0.0536 s / epoch)\n",
      "6800 | 0.3772 (0.0319 s / epoch)\n",
      "6900 | 0.2979 (0.0287 s / epoch)\n",
      "7000 | 0.3877 (0.0515 s / epoch)\n",
      "Error: 0.16515380144119263\n",
      "7100 | 0.4721 (0.0332 s / epoch)\n",
      "7200 | 0.6065 (0.0272 s / epoch)\n",
      "7300 | 0.4250 (0.0268 s / epoch)\n",
      "7400 | 0.4497 (0.0394 s / epoch)\n",
      "7500 | 0.2777 (0.0315 s / epoch)\n",
      "Error: 0.30868154764175415\n",
      "7600 | 0.3170 (0.0301 s / epoch)\n",
      "7700 | 0.3917 (0.0277 s / epoch)\n",
      "7800 | 0.2139 (0.0366 s / epoch)\n",
      "7900 | 0.3680 (0.0264 s / epoch)\n",
      "8000 | 0.4038 (0.0321 s / epoch)\n",
      "Error: 0.20403549075126648\n",
      "8100 | 0.3730 (0.0409 s / epoch)\n",
      "8200 | 0.3466 (0.0394 s / epoch)\n",
      "8300 | 0.3309 (0.0332 s / epoch)\n",
      "8400 | 0.4385 (0.0256 s / epoch)\n",
      "8500 | 0.2904 (0.0376 s / epoch)\n",
      "Error: 0.2076641023159027\n",
      "8600 | 0.2493 (0.0385 s / epoch)\n",
      "8700 | 0.3808 (0.0316 s / epoch)\n",
      "8800 | 0.3332 (0.0385 s / epoch)\n",
      "8900 | 0.3588 (0.0337 s / epoch)\n",
      "9000 | 0.3035 (0.0386 s / epoch)\n",
      "Error: 0.25990116596221924\n",
      "9100 | 0.4036 (0.0404 s / epoch)\n",
      "9200 | 0.3332 (0.0325 s / epoch)\n",
      "9300 | 0.2639 (0.0281 s / epoch)\n",
      "9400 | 0.2620 (0.0221 s / epoch)\n",
      "9500 | 0.2537 (0.0436 s / epoch)\n",
      "Error: 0.28925204277038574\n",
      "9600 | 0.2911 (0.0410 s / epoch)\n",
      "9700 | 0.2643 (0.0278 s / epoch)\n",
      "9800 | 0.3705 (0.0278 s / epoch)\n",
      "9900 | 0.2719 (0.0281 s / epoch)\n",
      "10000 | 0.2476 (0.0364 s / epoch)\n",
      "Error: 0.21840129792690277\n",
      "10100 | 0.2587 (0.0403 s / epoch)\n",
      "10200 | 0.3300 (0.0404 s / epoch)\n",
      "10300 | 0.2539 (0.0318 s / epoch)\n",
      "10400 | 0.3007 (0.0369 s / epoch)\n",
      "10500 | 0.2759 (0.0337 s / epoch)\n",
      "Error: 0.20744864642620087\n",
      "10600 | 0.1914 (0.0245 s / epoch)\n",
      "10700 | 0.1910 (0.0525 s / epoch)\n",
      "10800 | 0.1655 (0.0392 s / epoch)\n",
      "10900 | 0.2745 (0.0434 s / epoch)\n",
      "11000 | 0.2386 (0.0383 s / epoch)\n",
      "Error: 0.2384577840566635\n",
      "11100 | 0.2493 (0.0288 s / epoch)\n",
      "11200 | 0.2782 (0.0306 s / epoch)\n",
      "11300 | 0.1294 (0.0464 s / epoch)\n",
      "11400 | 0.2437 (0.0268 s / epoch)\n",
      "11500 | 0.1610 (0.0409 s / epoch)\n",
      "Error: 0.17040082812309265\n",
      "11600 | 0.1634 (0.0308 s / epoch)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "KeyboardInterrupt\n",
      "\n"
     ]
    }
   ],
   "source": [
    "state = TrainStateNoProj.create(apply_fn=apply_fn, params=init_vars[\"params\"], tx_params=optimizer_params, inner_lr=config[\"inner_lr\"], n_updates=config[\"n_updates\"])\n",
    "\n",
    "losses = []\n",
    "errors_test = []\n",
    "\n",
    "for epoch_index in range(config[\"n_epochs\"]):\n",
    "    t = time.time_ns()\n",
    "    key, subkey = random.split(key)\n",
    "    state, current_loss = step(subkey, state)\n",
    "    \n",
    "    if epoch_index % 100 == 0:\n",
    "        print(f\"{epoch_index} | {current_loss:.4f} ({(time.time_ns() - t) / 10**9:.4f} s / epoch)\")\n",
    "        \n",
    "    if epoch_index % 500 == 0:\n",
    "        # test time\n",
    "        key, subkey = random.split(key)\n",
    "        mse_test = test_during_training(subkey, state)\n",
    "        errors_test.append(mse_test)\n",
    "        print(f\"Error: {mse_test}\")\n",
    "    \n",
    "    losses.append(current_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "60ee4c91",
   "metadata": {},
   "outputs": [],
   "source": [
    "output = {}\n",
    "output[\"trained_params\"] = state.params\n",
    "output[\"losses\"]=losses\n",
    "output[\"errors_test\"]=errors_test\n",
    "output[\"config\"] = config"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "208d2f01",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(\"logs_final/maml_single_infinite.pickle\", \"wb\") as handle:\n",
    "    pickle.dump(output, handle, protocol=pickle.HIGHEST_PROTOCOL)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d12811c9",
   "metadata": {},
   "source": [
    "## Option #2: Loading a previously trained network"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "0b2aa1ce",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(\"logs_final/maml_single_infinite.pickle\", \"rb\") as handle:\n",
    "    output = pickle.load(handle)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "429e0381",
   "metadata": {},
   "outputs": [],
   "source": [
    "config = output[\"config\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "1d09aeea",
   "metadata": {},
   "outputs": [],
   "source": [
    "key, subkey = random.split(key)\n",
    "x_a, y_a, x_b, y_b = get_train_batch_fn(subkey)\n",
    "updated_params = inner_updates(output[\"trained_params\"], x_a[0], y_a[0], config[\"n_updates_test\"], config[\"inner_lr\"], apply_fn)\n",
    "predictions = apply_fn(updated_params, x_b[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "1df0b9d1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7fd8b03fa160>"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(x_a[0], y_a[0], \"ro\", label=\"Context\")\n",
    "plt.plot(x_b[0], y_b[0], \"rx\", label=\"Query\")\n",
    "plt.plot(x_b[0], predictions, \"+b\", label=\"Pred\")\n",
    "plt.plot(np.linspace(-5, 5, 100), apply_fn(output[\"trained_params\"], np.linspace(-5, 5, 100)[:, np.newaxis]), \"--\", label=\"Raw\", alpha=0.4)\n",
    "plt.legend()"
   ]
  }
 ],
 "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.8.13"
  },
  "latex_envs": {
   "LaTeX_envs_menu_present": true,
   "autoclose": false,
   "autocomplete": true,
   "bibliofile": "biblio.bib",
   "cite_by": "apalike",
   "current_citInitial": 1,
   "eqLabelWithNumbers": true,
   "eqNumInitial": 1,
   "hotkeys": {
    "equation": "Ctrl-E",
    "itemize": "Ctrl-I"
   },
   "labels_anchors": false,
   "latex_user_defs": false,
   "report_style_numbering": false,
   "user_envs_cfg": false
  },
  "toc": {
   "base_numbering": 1,
   "nav_menu": {},
   "number_sections": true,
   "sideBar": true,
   "skip_h1_title": false,
   "title_cell": "Table of Contents",
   "title_sidebar": "Contents",
   "toc_cell": false,
   "toc_position": {},
   "toc_section_display": true,
   "toc_window_display": false
  },
  "vscode": {
   "interpreter": {
    "hash": "87dee15dd5d19160a31e5ce9fa66a6098ea6ae58c7c999b60e5c6cbbacb7150d"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
