{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from sklearn.linear_model import Ridge, RidgeCV\n",
    "\n",
    "from src.model import *\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib.lines import Line2D\n",
    "plt.style.use('seaborn-whitegrid')\n",
    "\n",
    "# PLOT PREAMBLE: LIKE LATEX\n",
    "LINEWIDTH = 6.00117\n",
    "\n",
    "SCRIPT_SIZE = 8\n",
    "SMALL = 10\n",
    "NORMAL_SIZE = 10.95\n",
    "LARGE = 12\n",
    "Huge = 24.88\n",
    "\n",
    "#Direct input\n",
    "plt.rcParams['text.latex.preamble']=r\"\\usepackage{lmodern}\"\n",
    "#Options\n",
    "params = {'text.usetex' : True,\n",
    "          'font.size' : NORMAL_SIZE,\n",
    "          'axes.labelsize' : NORMAL_SIZE,\n",
    "          'xtick.labelsize' : NORMAL_SIZE,\n",
    "          'ytick.labelsize' : NORMAL_SIZE,\n",
    "          'legend.fontsize' : NORMAL_SIZE,\n",
    "          'figure.titlesize' : LARGE,\n",
    "          'font.family' : 'serif',\n",
    "          'font.serif': ['Computer Modern Roman'],\n",
    "          'lines.linewidth': 1.5\n",
    "          }\n",
    "plt.rcParams.update(params)\n",
    "\n",
    "# For the warning when hiding yticklabels\n",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "outputs": [],
   "source": [
    "def causal_excess_risk(beta_hat, model, interventional_validation=None):\n",
    "    \"\"\"Computes causal risk of a linear predictor beta_hat based on samples X_int, Y_int from the interventional distribution\"\"\"\n",
    "    if interventional_validation is None:\n",
    "        return weighted_dot(u=model.beta.flatten()-beta_hat.flatten(), v=None, A=model.Sigma)\n",
    "    else:\n",
    "        X_int, Y_int = interventional_validation\n",
    "        return np.mean((X_int @ beta_hat.flatten() - Y_int.flatten()) ** 2, axis=0) - model.bayes_risk_caus"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "outputs": [],
   "source": [
    "# Fixed hyperparameters of statistical model\n",
    "d, l = 300, 350\n",
    "r_sq_stat = 5\n",
    "sigma_sq_stat = 1\n",
    "sigma_sq = .5\n",
    "\n",
    "# Generate a single M to use throughout all models\n",
    "U, V = ortho_group.rvs(d), ortho_group.rvs(l)\n",
    "Lambda = np.concatenate((np.eye(d), np.zeros((d, l-d))), axis=1)\n",
    "M = (U @ Lambda) @ V.T\n",
    "M_params = (U, V, M)\n",
    "# Iterate over different confounding strengths (different causal models with same statistical model)\n",
    "gam = 1e-2\n",
    "n_samples = int(d / gam)\n",
    "n_samples_intervention = n_samples\n",
    "\n",
    "risks_cv = []\n",
    "risks_intervention = []\n",
    "risks_null_finite = []\n",
    "\n",
    "conf_strength_list = np.linspace(.001, .99, 15)\n",
    "lams = [10 ** k for k in np.linspace(-4, 4, 30)]\n",
    "for conf_strength in conf_strength_list:\n",
    "    # Create model with given confounding strength\n",
    "    _, alpha, beta, sigma_sq = generate_causal_params_for_fixed_statistical(d=d, l=l, r_sq_stat=r_sq_stat,\n",
    "                                                                                    sigma_sq_stat=sigma_sq_stat,\n",
    "                                                                                    sigma_sq=sigma_sq,\n",
    "                                                                                    conf_strength=conf_strength,\n",
    "                                                                                    eta=0,\n",
    "                                                                                    M_params=M_params)\n",
    "    ridge = RidgeLimit(alpha=alpha, beta=beta, sigma_sq=sigma_sq, M=M)\n",
    "\n",
    "    # Empirical quantities\n",
    "    X, Y = ridge.sample(n_samples=n_samples)\n",
    "    X_int, _ = ridge.sample(n_samples=n_samples_intervention)\n",
    "    Y_int = X_int @ ridge.beta.reshape(-1, 1) + np.random.normal(loc=0, scale=np.sqrt(ridge.sigma_sq), size=n_samples_intervention).reshape(-1, 1)\n",
    "\n",
    "    # Statistical regression\n",
    "    clf_cv = RidgeCV([lam * n_samples for lam in lams], fit_intercept=False).fit(X,Y)\n",
    "    risks_cv.append(causal_excess_risk(clf_cv.coef_, model=ridge))\n",
    "\n",
    "    # Null risk\n",
    "    risks_null_finite.append(causal_excess_risk(np.zeros(d), model=ridge))\n",
    "\n",
    "    # Regression with interventional validation\n",
    "    best_val_risk = np.infty\n",
    "    best_pred = None\n",
    "    for lam in lams:\n",
    "        clf = Ridge(lam * n_samples).fit(X, Y)\n",
    "        val_risk = causal_excess_risk(clf.coef_, model=ridge, interventional_validation=(X_int, Y_int))\n",
    "        if val_risk < best_val_risk:\n",
    "            best_val_risk = val_risk\n",
    "            best_pred = clf.coef_\n",
    "    risks_intervention.append(causal_excess_risk(best_pred, model=ridge))"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "outputs": [
    {
     "data": {
      "text/plain": "<Figure size 360.07x225.044 with 1 Axes>",
      "image/png": "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\n"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "save_path = '../fig/figure2.pdf'\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(.6*LINEWIDTH, .375*LINEWIDTH))\n",
    "\n",
    "ax.plot(conf_strength_list, risks_intervention, color='red', marker='x', markersize=6)\n",
    "ax.plot(conf_strength_list, risks_cv, color='black', marker='o', markersize=6)\n",
    "ax.plot(conf_strength_list, risks_null_finite, color='k', linestyle='--')\n",
    "\n",
    "ax.set_xlim([0, 1])\n",
    "ax.set_xlabel(r'Confounding strength $\\zeta$')\n",
    "ax.set_ylabel(r'Causal excess risk $||\\hat{\\beta}-\\beta||_\\Sigma^2$')\n",
    "\n",
    "ax.spines['left'].set_position('zero')\n",
    "ax.spines['bottom'].set_position('zero')\n",
    "\n",
    "handles = [Line2D([], [], label=r'Interventional validation', color='red', marker='x'),\n",
    "           Line2D([], [], label=r'Cross validation', color='k', marker='o'),\n",
    "           Line2D([], [], label=r'Null predictor', color='k', linestyle='--')]\n",
    "\n",
    "plt.legend(handles=handles, framealpha=1, frameon=True, ncol=1, prop={'size': 9}, loc='upper center')\n",
    "if save_path is not None:\n",
    "    plt.savefig(fname=save_path, bbox_inches='tight', dpi=300)\n",
    "plt.show()"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 2
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython2",
   "version": "2.7.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 0
}