{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.append('../..')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "\n",
    "import numpy as np\n",
    "from scipy import interpolate\n",
    "\n",
    "import random\n",
    "import tqdm\n",
    "from copy import deepcopy\n",
    "from argparse import Namespace\n",
    "import warnings\n",
    "\n",
    "%matplotlib inline\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "sns.set()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "from causal_meta.modules import networks as nets\n",
    "from causal_meta.utils.data_utils import RandomSplineSCM\n",
    "\n",
    "import train_utils as tu\n",
    "from xcoders import Rotor\n",
    "from mixture_density import MDN, GMM, mdn_nll\n",
    "from gmm import GaussianMixture"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "SEED = 91023\n",
    "torch.manual_seed(SEED)\n",
    "np.random.seed(SEED)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Boilerplate"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Utilities"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def normal(mean, std, N): \n",
    "    return torch.normal(torch.ones(N).mul_(mean), torch.ones(N).mul_(std)).view(-1, 1)\n",
    "\n",
    "def normal_like(X): \n",
    "    mean = X.mean()\n",
    "    std = X.std()\n",
    "    return normal(mean, std, X.size(0))\n",
    "\n",
    "def mlp(opt): \n",
    "    if opt.NUM_MID_LAYERS == -1: \n",
    "        return nn.Linear(opt.INP_DIM, opt.OUT_DIM)\n",
    "    else:\n",
    "        return nets.MLP(opt.INP_DIM, opt.OUT_DIM, opt.NUM_MID_LAYERS, \n",
    "                        opt.CAPACITY, opt.INP_NOISE)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Mixture Density"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "def mdn(opt): \n",
    "    return MDN(opt.CAPACITY, opt.NUM_COMPONENTS)\n",
    "\n",
    "def gmm(opt): \n",
    "    return GaussianMixture(opt.GMM_NUM_COMPONENTS)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Reversible Encoder"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def xcodergen(opt): \n",
    "    # Make rotor\n",
    "    return Rotor(opt.XCODER_INIT)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Ground Truth Generator"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We use a posterior of the form: \n",
    "\n",
    "$$\n",
    "P(Y | X) = \\sum_i \\pi_i(X) \\mathcal{N}(\\mu_i(X), \\sigma_i(X))\n",
    "$$\n",
    "\n",
    "which is a mixture of 10 gaussians. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "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": [
    "# Test\n",
    "rand_scm = RandomSplineSCM(False, True, 8, 8, 3, range_scale=1.)\n",
    "\n",
    "plt.figure()\n",
    "plt.title(\"SCM with Covariate Shift\")\n",
    "rand_scm.plot(X=normal(4, 2, 1000), show=False, label='Transfer 0', alpha=0.7)\n",
    "rand_scm.plot(X=normal(-4, 2, 1000), show=False, label='Transfer 1', alpha=0.7)\n",
    "rand_scm.plot(X=normal(0, 2, 1000), show=False, label='Train', alpha=0.7)\n",
    "plt.xlabel(\"X\")\n",
    "plt.ylabel(\"Y\")\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_key(frames, key, show=True, label=None, name=None): \n",
    "    its, vals = zip(*[(frame.iter_num, getattr(frame, key)) for frame in frames])\n",
    "    if show:\n",
    "        plt.figure()\n",
    "    plt.plot(its, vals, label=label)\n",
    "    if show:\n",
    "        plt.xlabel(\"Iterations\")\n",
    "        plt.ylabel(name if name is not None else key.title())\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "def gradnan_filter(model): \n",
    "    nan_found = False\n",
    "    for p in model.parameters(): \n",
    "        nan_mask = torch.isnan(p.grad.data)\n",
    "        nan_found = bool(nan_mask.any().item())\n",
    "        p.grad.data[nan_mask] = 0.\n",
    "    return nan_found"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "def marginal_nll(opt, inp, nll): \n",
    "    model_g = gmm(opt)\n",
    "    if opt.CUDA: \n",
    "        model_g = model_g.cuda()\n",
    "    model_g.fit(inp, n_iter=opt.EM_ITERS)\n",
    "    with torch.no_grad():\n",
    "        loss_marginal = nll(model_g(inp), inp)\n",
    "    return loss_marginal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "def encoder_train_shared_regret(opt, model_x2y, model_y2x, scm, encoder, decoder, alpha): \n",
    "    if opt.CUDA: \n",
    "        model_x2y = model_x2y.cuda()\n",
    "        model_y2x = model_y2x.cuda()\n",
    "        encoder = encoder.cuda()\n",
    "        decoder = decoder.cuda()\n",
    "    encoder_optim = torch.optim.Adam(encoder.parameters(), opt.ENCODER_LR)\n",
    "    alpha_optim = torch.optim.Adam([alpha], opt.ALPHA_LR)\n",
    "    frames = []\n",
    "    for meta_iter in tqdm.trange(opt.NUM_META_ITER): \n",
    "        # Preheat the models\n",
    "        _ = tu.train_nll(opt, model_x2y, scm, opt.TRAIN_DISTRY, 'X2Y', \n",
    "                         mdn_nll, decoder, encoder, quiet=True)\n",
    "        _ = tu.train_nll(opt, model_y2x, scm, opt.TRAIN_DISTRY, 'Y2X', \n",
    "                         mdn_nll, decoder, encoder, quiet=True)\n",
    "        # Sample from SCM\n",
    "        X = opt.TRANS_DISTRY()\n",
    "        Y = scm(X)\n",
    "        if opt.CUDA: \n",
    "            X, Y = X.cuda(), Y.cuda()\n",
    "        # Decode \n",
    "        with torch.no_grad(): \n",
    "            X, Y = decoder(X, Y)\n",
    "        # Encode\n",
    "        X, Y = encoder(X, Y)\n",
    "        with torch.no_grad():\n",
    "            if opt.USE_BASELINE:\n",
    "                baseline_y = marginal_nll(opt, Y, mdn_nll)\n",
    "                baseline_x = marginal_nll(opt, X, mdn_nll)\n",
    "            else:\n",
    "                baseline_y = 0.\n",
    "                baseline_x = 0.\n",
    "        # Save state dicts\n",
    "        state_x2y = deepcopy(model_x2y.state_dict())\n",
    "        state_y2x = deepcopy(model_y2x.state_dict())\n",
    "        # Inner loop \n",
    "        optim_x2y = torch.optim.Adam(model_x2y.parameters(), lr=opt.FINETUNE_LR)\n",
    "        optim_y2x = torch.optim.Adam(model_y2x.parameters(), lr=opt.FINETUNE_LR)\n",
    "        regrets_x2y = []\n",
    "        regrets_y2x = []\n",
    "        is_nan = False\n",
    "        # Evaluate regret discrepancy \n",
    "        for t in range(opt.FINETUNE_NUM_ITER):\n",
    "            loss_x2y = mdn_nll(model_x2y(X), Y)\n",
    "            loss_y2x = mdn_nll(model_y2x(Y), X)\n",
    "            if torch.isnan(loss_x2y).item() or torch.isnan(loss_y2x).item(): \n",
    "                is_nan = True\n",
    "                break\n",
    "            optim_x2y.zero_grad()\n",
    "            optim_y2x.zero_grad()\n",
    "            loss_x2y.backward(retain_graph=True) \n",
    "            loss_y2x.backward(retain_graph=True)\n",
    "            # Filter out NaNs that might have sneaked in\n",
    "            nan_in_x2y = gradnan_filter(model_x2y)\n",
    "            nan_in_y2x = gradnan_filter(model_y2x)\n",
    "            if nan_in_x2y or nan_in_y2x: \n",
    "                is_nan = True\n",
    "                break\n",
    "            optim_x2y.step()\n",
    "            optim_y2x.step()\n",
    "            # Store for encoder\n",
    "            regrets_x2y.append(loss_x2y + baseline_x)\n",
    "            regrets_y2x.append(loss_y2x + baseline_y)\n",
    "        if not is_nan:\n",
    "            # Evaluate total regret\n",
    "            regret_x2y = torch.stack(regrets_x2y).mean()\n",
    "            regret_y2x = torch.stack(regrets_y2x).mean()\n",
    "            # Evaluate losses\n",
    "            loss = torch.logsumexp(\n",
    "                torch.stack([F.logsigmoid(alpha) + regret_x2y,\n",
    "                             F.logsigmoid(-alpha) + regret_y2x]), \n",
    "                0)\n",
    "            # Optimize\n",
    "            encoder_optim.zero_grad()\n",
    "            alpha_optim.zero_grad()\n",
    "            loss.backward()\n",
    "            # Make sure no nans\n",
    "            if torch.isnan(encoder.theta.grad.data).any(): \n",
    "                encoder.theta.grad.data.zero_()\n",
    "            if torch.isnan(alpha.grad.data).any(): \n",
    "                alpha.grad.data.zero_()\n",
    "            encoder_optim.step()\n",
    "            alpha_optim.step()\n",
    "            # Load original state dicts\n",
    "            model_x2y.load_state_dict(state_x2y)\n",
    "            model_y2x.load_state_dict(state_y2x)\n",
    "            # Add info\n",
    "            frames.append(Namespace(iter_num=meta_iter, \n",
    "                                    regret_x2y=regret_x2y.item(), \n",
    "                                    regret_y2x=regret_y2x.item(),\n",
    "                                    loss=loss.item(),\n",
    "                                    alpha=alpha.item(), \n",
    "                                    theta=encoder.theta.item()))\n",
    "        else:\n",
    "            # Load original state dicts\n",
    "            model_x2y.load_state_dict(state_x2y)\n",
    "            model_y2x.load_state_dict(state_y2x)\n",
    "            # Add dummy info\n",
    "            frames.append(Namespace(iter_num=meta_iter, \n",
    "                                    regret_x2y=float('nan'), \n",
    "                                    regret_y2x=float('nan'),\n",
    "                                    loss=float('nan'),\n",
    "                                    alpha=float('nan'), \n",
    "                                    theta=float('nan')))\n",
    "            \n",
    "    return frames"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_theta(frames, gt_theta, save=False): \n",
    "    its, vals = zip(*[(frame.iter_num, frame.theta / (np.pi / 2)) for frame in frames])\n",
    "    gt_theta = -gt_theta.item() / (np.pi / 2)\n",
    "    plt.figure()\n",
    "    # plt.plot(its, vals, label=r'$\\theta_{\\mathcal{E}}$', c='black')\n",
    "    plt.plot(its, vals, label=r'$\\theta_{\\mathcal{E}}$')\n",
    "    plt.plot(its, [gt_theta] * len(its), linestyle='--', label=r'Solution 1 $\\left(+\\frac{\\pi}{4}\\right)$')\n",
    "    plt.plot(its, [gt_theta - 1] * len(its), linestyle='--', label=r'Solution 2 $\\left(-\\frac{\\pi}{4}\\right)$')\n",
    "    plt.xlabel(\"Iterations\")\n",
    "    plt.ylabel(\"Encoder Angle [π/2 rad]\")\n",
    "    plt.legend()\n",
    "    if save:\n",
    "        plt.savefig('fixed-encoder-evo.pdf', bbox_inches='tight', format='pdf')\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "def probe_xcoders(encoder, decoder):\n",
    "    # Test encoder and decoder\n",
    "    with torch.no_grad():\n",
    "        _X = torch.tensor([[1.]])\n",
    "        _Y = torch.tensor([[0.]])\n",
    "        if opt.CUDA:\n",
    "            _X, _Y = _X.to(encoder.theta.device), _Y.to(encoder.theta.device)\n",
    "        _X_d, _Y_d = decoder(_X, _Y)\n",
    "        _X_de, _Y_de = encoder(_X_d, _Y_d)\n",
    "    print(f\"Initial (A, B) = {_X.item()}, {_Y.item()}\")\n",
    "    print(f\"Decoded (X, Y) = {_X_d.item()}, {_Y_d.item()}\")\n",
    "    print(f\"Encoded (U, V) = {_X_de.item()}, {_Y_de.item()}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Experiments with Unfixed Encoders"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "opt = Namespace()\n",
    "\n",
    "# Model\n",
    "opt.CAPACITY = 32\n",
    "opt.NUM_COMPONENTS = 10\n",
    "opt.GMM_NUM_COMPONENTS = 10\n",
    "\n",
    "# Training\n",
    "opt.LR = 0.01\n",
    "opt.NUM_ITER = 20\n",
    "opt.NUM_META_ITER = 1000\n",
    "opt.ENCODER_LR = 0.01\n",
    "opt.ALPHA_LR = 0.001\n",
    "opt.CUDA = True\n",
    "opt.REC_FREQ = 10\n",
    "opt.ALPHA_INIT = 0.\n",
    "opt.USE_BASELINE = True\n",
    "\n",
    "# Fine tuning\n",
    "opt.FINETUNE_NUM_ITER = 5\n",
    "opt.FINETUNE_LR = 0.001\n",
    "opt.EM_ITERS = 500\n",
    "\n",
    "# Sampling \n",
    "opt.NUM_SAMPLES = 1000\n",
    "opt.TRAIN_DISTRY = lambda: normal(0, 2, opt.NUM_SAMPLES)\n",
    "opt.TRANS_DISTRY = lambda: normal(np.random.uniform(-4, 4), \n",
    "                                  2, opt.NUM_SAMPLES)\n",
    "\n",
    "# Encoder\n",
    "opt.DECODER_DEFAULT = -float(0.5 * np.pi/2)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Make Xcoders"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Make a ground-truth decoder with $\\theta_0$. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "gt_decoder = Rotor(opt.DECODER_DEFAULT)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The real encoder should have $\\theta = -\\theta_0$. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "encoder = Rotor(0. * np.pi/2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initial (A, B) = 1.0, 0.0\n",
      "Decoded (X, Y) = 0.7071067690849304, 0.7071067690849304\n",
      "Encoded (U, V) = 0.7071067690849304, 0.7071067690849304\n"
     ]
    }
   ],
   "source": [
    "probe_xcoders(encoder, gt_decoder)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Make Alpha and Models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "alpha = tu.make_alpha(opt)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "model_x2y = mdn(opt)\n",
    "model_y2x = mdn(opt)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Train Encoder\n",
    "\n",
    "If it doesn't work with the seed of your choice, try tuning the (hyper)parameters of ```opt.TRANS_DISTRY```. Since the SCM is generated randomly, it's not possible to know the correct hyper-parameters in advance (unless the seed is fixed). $\\sigma \\in (1, 2)$ should work fine, but if the range from which the means are sampled is too large, you can have numerical instabilities. If it's too small, the gradient signal to the encoder can be weak. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1000/1000 [22:56<00:00,  1.80s/it]\n"
     ]
    }
   ],
   "source": [
    "frames = encoder_train_shared_regret(opt, model_x2y, model_y2x, rand_scm, encoder, gt_decoder, alpha)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "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": [
    "plot_theta(frames, gt_decoder.theta)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "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": [
    "plot_key(frames, 'alpha', name='Structural Parameter')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.6.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
