{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "c588fb80",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'/Users/nilssturma/git/multi-domain-crl'"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import os\n",
    "os.chdir('/Users/nilssturma/git/multi-domain-crl/')\n",
    "os.getcwd()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "0f097a89",
   "metadata": {},
   "outputs": [],
   "source": [
    "# === IMPORTS: BUILT-IN ===\n",
    "import pickle\n",
    "#from time import time\n",
    "\n",
    "# === IMPORTS: THIRD-PARTY ===\n",
    "import numpy as np\n",
    "from scipy.stats import beta, expon, lognorm, weibull_min, chi2, gumbel_r, skewnorm\n",
    "\n",
    "# === IMPORTS: LOCAL ===\n",
    "from src.learner import LinearMDCRL\n",
    "from experiments.rand import rand_model"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "55db7e52",
   "metadata": {},
   "source": [
    "## Define model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "11966d0a",
   "metadata": {},
   "outputs": [],
   "source": [
    "n=5000\n",
    "measure = \"ks-test\"\n",
    "\n",
    "rvs = [beta(2,3), beta(2,5), chi2(df=4), gumbel_r, lognorm(s=1),\n",
    "    weibull_min(c=2), expon(scale=0.1), skewnorm(a=6), skewnorm(a=12)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "89e5f306",
   "metadata": {},
   "outputs": [],
   "source": [
    "m = {\n",
    "    \"nr_doms\": 2,\n",
    "    \"joint_idx\": [0,1,2],\n",
    "    \"domain_spec_idx\": [[3,4],[5,6]],\n",
    "    \"noise_rvs\": rvs,\n",
    "    \"sample_sizes\":  [5000,5000],\n",
    "    \"dims\": [15,15],\n",
    "    \"graph_density\": 0.75,\n",
    "    \"mixing_density\": 0.9,\n",
    "    \"mixing_distribution\": 'unif',  # unif or normal\n",
    "    \"two_pure_children\": True\n",
    "}\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4427e35a",
   "metadata": {},
   "source": [
    "## Sample data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "bef84a6a",
   "metadata": {},
   "outputs": [],
   "source": [
    "m[\"sample_sizes\"] = [n for _ in range(m[\"nr_doms\"])]\n",
    "data, g, B_true = rand_model(m)\n",
    "A_true = np.transpose(g.to_amat())"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "dbb22298",
   "metadata": {},
   "source": [
    "## Fit model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "e5941515",
   "metadata": {},
   "outputs": [],
   "source": [
    "model = LinearMDCRL(measure=measure, gamma=0.1)\n",
    "model.fit(data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "8be9d93a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[-0.58, -0.57, -0.36,  0.82, -0.32,  0.  ,  0.  ],\n",
       "       [-1.03, -0.48,  0.47, -0.05, -0.82,  0.  ,  0.  ],\n",
       "       [ 0.65,  0.03,  0.  , -0.84,  0.61,  0.  ,  0.  ],\n",
       "       [ 0.76, -0.67, -1.39, -0.97, -0.71,  0.  ,  0.  ],\n",
       "       [ 1.64, -0.93, -1.  ,  0.01, -0.68,  0.  ,  0.  ],\n",
       "       [-0.61,  0.92,  0.18,  0.95, -0.37,  0.  ,  0.  ],\n",
       "       [ 0.41, -0.34, -0.75,  0.74,  0.34,  0.  ,  0.  ],\n",
       "       [-0.23,  0.75, -0.07, -0.63,  0.32,  0.  ,  0.  ],\n",
       "       [-0.34,  0.02,  0.34, -0.01,  0.01,  0.  ,  0.  ],\n",
       "       [ 0.5 ,  0.58, -0.55, -0.81,  0.75,  0.  ,  0.  ],\n",
       "       [-0.02, -0.98, -0.38, -0.79,  0.84,  0.  ,  0.  ],\n",
       "       [-0.26,  0.32,  0.23,  0.73,  0.78,  0.  ,  0.  ],\n",
       "       [-1.4 ,  0.68,  0.96,  0.93, -0.45,  0.  ,  0.  ],\n",
       "       [ 1.46, -0.73, -1.11,  0.01,  0.36,  0.  ,  0.  ],\n",
       "       [-1.34,  0.41,  0.53, -0.01, -0.68,  0.  ,  0.  ],\n",
       "       [ 0.77, -0.01, -0.35,  0.  ,  0.  ,  0.84,  0.46],\n",
       "       [-0.76, -0.8 ,  0.4 ,  0.  ,  0.  , -0.29,  0.4 ],\n",
       "       [-0.  ,  0.86, -0.77,  0.  ,  0.  ,  0.23,  1.02],\n",
       "       [ 0.28, -0.45, -0.37,  0.  ,  0.  , -0.62,  0.9 ],\n",
       "       [-0.67,  0.43,  0.87,  0.  ,  0.  , -0.85,  0.59],\n",
       "       [ 0.04,  0.65, -0.38,  0.  ,  0.  ,  0.29,  0.27],\n",
       "       [-0.82,  0.81,  0.2 ,  0.  ,  0.  , -1.28, -0.42],\n",
       "       [ 0.18, -0.66, -0.75,  0.  ,  0.  , -0.5 , -0.56],\n",
       "       [ 0.48,  0.02,  0.03,  0.  ,  0.  ,  0.51,  0.44],\n",
       "       [-0.03, -0.86, -0.21,  0.  ,  0.  ,  0.67, -0.78],\n",
       "       [ 0.19, -0.33, -0.31,  0.  ,  0.  ,  0.23,  0.4 ],\n",
       "       [-0.88,  0.62,  0.44,  0.  ,  0.  , -1.  , -1.  ],\n",
       "       [ 2.07, -0.46, -0.46,  0.  ,  0.  ,  1.31,  0.  ],\n",
       "       [-0.17,  0.63, -1.03,  0.  ,  0.  , -1.1 ,  0.03],\n",
       "       [ 0.5 ,  0.12,  1.  ,  0.  ,  0.  , -0.03,  0.5 ]])"
      ]
     },
     "execution_count": 31,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.joint_mixing.round(2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "5761b1a3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[[1, 3], [2, 0], [4, 4]]"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.joint_factors"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "9367d4fa",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[[1, 1], [1, 1], [1, 1]]"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.joint_signs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "d1005fd7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1008x504 with 10 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "fig, ax = plt.subplots(2, 5, figsize=(14, 7))\n",
    "\n",
    "for i in range(2):\n",
    "    for j in range(5):\n",
    "        ax[i,j].hist(model.indep_comps[i][:,j], bins=50, density=True)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bb4a1855",
   "metadata": {},
   "source": [
    "## Sample from joint distribution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "1b9722f0",
   "metadata": {},
   "outputs": [],
   "source": [
    "indep_comps_large = []\n",
    "\n",
    "# Shared components\n",
    "for i,f in enumerate(model.joint_factors):\n",
    "    shared_comp = np.array([])\n",
    "    for env in range(model.nr_env):\n",
    "        shared_comp = np.concatenate((shared_comp, \n",
    "                                      (model.joint_signs[i][env] * model.indep_comps[env][:,f[env]])))\n",
    "    indep_comps_large.append(shared_comp)\n",
    "    \n",
    "# Domain-specific components\n",
    "for env in range(model.nr_env):\n",
    "    ncomps = model.indep_comps[env].shape[1]\n",
    "    shared_components = [f[env] for f in model.joint_factors]\n",
    "    dom_spec_comps = set(np.arange(ncomps)) - set(shared_components)\n",
    "    for comp in dom_spec_comps:\n",
    "        indep_comps_large.append(model.indep_comps[env][:,comp])\n",
    "        \n",
    "# This should define a function \"get_components\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "c70b3776",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1296x180 with 7 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 7, figsize=(18, 2.5))\n",
    "\n",
    "for j in range(7):\n",
    "    ax[j].hist(indep_comps_large[j], bins=50, density=True)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "0c180784",
   "metadata": {},
   "outputs": [],
   "source": [
    "def sample_from_joint_distr(size=5000):\n",
    "    noise_mat = np.zeros((size, model.total_nr_lat))\n",
    "    for comp in range(model.total_nr_lat):\n",
    "        noise_mat[:,comp] = np.random.choice(indep_comps_large[comp], size = size, replace=True)\n",
    "        \n",
    "    samples = np.transpose(np.matmul(model.joint_mixing, np.transpose(noise_mat)))\n",
    "    \n",
    "    return(samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "a6823c2a",
   "metadata": {},
   "outputs": [],
   "source": [
    "samples = sample_from_joint_distr()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "96a7c211",
   "metadata": {},
   "source": [
    "## Apply LaHME on samples from joint distribution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "9eb9e7fe",
   "metadata": {},
   "outputs": [],
   "source": [
    "os.chdir('/Users/nilssturma/git/multi-domain-crl/code_for_LaHME')\n",
    "from Main_LiNGLaH_LaHME import LaHME\n",
    "import pandas as pd\n",
    "import time\n",
    "time.clock = time.time  # The LaHME code uses time.clock which is deprecated"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "id": "fc30965c",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.DataFrame(samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "db53c12d",
   "metadata": {},
   "outputs": [],
   "source": [
    "LaHME(df)  ### This is toooo slow, cannot do simulations! (Did not finish after several hours!)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2ced1487",
   "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.9.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
