{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "initial_id",
   "metadata": {
    "collapsed": true,
    "ExecuteTime": {
     "end_time": "2024-05-17T08:56:49.606458Z",
     "start_time": "2024-05-17T08:56:49.537274Z"
    }
   },
   "outputs": [],
   "source": [
    "# import infomec\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "\n",
    "from sklearn.preprocessing import LabelEncoder\n",
    "from sklearn.feature_selection import mutual_info_classif\n",
    "\n",
    "def entropy(x):\n",
    "    _, counts = np.unique(x, return_counts=True, axis=0)\n",
    "    p_x = counts / x.shape[0]\n",
    "    return -np.sum(p_x * np.log2(p_x))\n",
    "\n",
    "def conditional_entropy(x, y):\n",
    "    \"\"\"H(X|Y) = H(X, Y) - H(Y)\"\"\"\n",
    "    H_xy = entropy(np.concatenate([x, y], axis=1))\n",
    "    H_y = entropy(y)\n",
    "    return H_xy - H_y\n",
    "\n",
    "def compute_ncmi(s, z, n_neighbors=3):\n",
    "    \"\"\"\n",
    "    Normalized conditional mutual information between sources s and latents z.\n",
    "    The [i, j] element is the conditional mutual information between the i-th source and the j-th latent \n",
    "    given the rest of the sources, normalized by the conditional entropy of the i-th source given the rest\n",
    "    of the sources.\n",
    "    I(z_j; s_i | s_{-i}) = I(z_j; s_i, s_{-i}) - I(z_j; s_{-i}) = I(z_j; s) - I(z_j; s_{-i})\n",
    "    NCMI[i, j] = I(z_j; s_i | s_{-i}) / H(s_i | s_{-i})\n",
    "    \"\"\"\n",
    "    \n",
    "    ds = s.shape[1]\n",
    "    dz = z.shape[1]\n",
    "    \n",
    "    I_z_j_and_s_joint = np.empty(shape=(dz,))\n",
    "    s_joint = LabelEncoder().fit_transform([str(s_sample) for s_sample in s])\n",
    "    for j in range(dz):\n",
    "        I_z_j_and_s_joint[j] = mutual_info_classif(z[:, j].reshape(-1, 1), s_joint, n_neighbors=n_neighbors).squeeze()\n",
    "    \n",
    "    ncmi = np.empty(shape=(ds, dz))\n",
    "    for i in range(ds):\n",
    "        s_rest = s[:, np.arange(ds) != i]\n",
    "        s_rest_str = LabelEncoder().fit_transform([str(s_sample) for s_sample in s_rest])\n",
    "        for j in range(dz):\n",
    "            I_z_j_and_s_rest = mutual_info_classif(z[:, j].reshape(-1, 1), s_rest_str, n_neighbors=n_neighbors).squeeze()\n",
    "            ncmi[i, j] = I_z_j_and_s_joint[j] - I_z_j_and_s_rest\n",
    "        H_s_i_given_s_rest = conditional_entropy(s[:, i].reshape(-1, 1), s_rest)\n",
    "        ncmi[i, :] /= H_s_i_given_s_rest\n",
    "    return ncmi\n",
    "\n",
    "def compute_cinfomec(s, z):\n",
    "    \"\"\"\n",
    "    Compute CInfoM, CInfoC, and InfoE between discrete sources s and continuous latents z.\n",
    "    s: np.ndarray, shape=(n_samples, n_sources)\n",
    "    z: np.ndarray, shape=(n_samples, n_latents)\n",
    "    \n",
    "    Inactive latents are heuristically identified by their range being less than a proportion of the range of the most active latent. Inactive latents are ignored in the computation of CInfoM and CInfoC.\n",
    "    \"\"\"\n",
    "    ncmi = compute_ncmi(s, z)\n",
    "    ncmi[ncmi < 0] = 0\n",
    "    \n",
    "    z_ranges = np.max(z, axis=0) - np.min(z, axis=0)\n",
    "    max_z_range = np.max(z_ranges)\n",
    "    z_active = z_ranges > 0.1 * max_z_range\n",
    "    \n",
    "    ds = s.shape[1]\n",
    "    ncmi_active = ncmi[:, z_active]\n",
    "    dz_active = np.sum(z_active)\n",
    "    if dz_active == 0:\n",
    "        return {\n",
    "            'CInfoM': 0,\n",
    "            'CInfoC': 0,\n",
    "            # 'InfoE': 0,\n",
    "            'NCMI': ncmi,\n",
    "            'z_active': z_active\n",
    "        }\n",
    "    cinfom = (np.mean(np.max(ncmi_active, axis=0) / np.sum(ncmi_active, axis=0)) - 1 / ds) / (\n",
    "                1 - 1 / ds)\n",
    "    cinfoc = (np.mean(np.max(ncmi_active, axis=1) / np.sum(ncmi_active, axis=1)) - 1 / dz_active) / (\n",
    "                1 - 1 / dz_active)\n",
    "    # infoe = infomec.compute_infoe(s, z, 'discrete', 'continuous')\n",
    "    return {\n",
    "        'CInfoM': cinfom,\n",
    "        'CInfoC': cinfoc,\n",
    "        # 'InfoE': infoe,\n",
    "        'NCMI': ncmi,\n",
    "        'z_active': z_active\n",
    "    }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "outputs": [
    {
     "data": {
      "text/plain": "<Axes: >"
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "text/plain": "<Figure size 640x480 with 1 Axes>",
      "image/png": "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TUbOSX7mPDo6qq985SvKz8/XU089pfLycsNpc9dc57xq1SodPnxYzz///PQ/SfrWt76l++67z3Dq3OPn+VxXV6djx45lLPvFL36hK664wkTUnOZnzsuXL9fPf/7zjGW//OUv9bGPfcxE1EuGzePggiwhjuNow4YN2rlzp9555x3953/+p1paWnT77bdLOt26T506JUlau3atTp48qaamJnV3d6upqUmu62rdunU2f4Sc4GfOe/fu1a9+9Ss1NzdPr+vr6+PqmDmY65zD4bAqKysz/kmn/19ONBq1+SPkBD/P582bN+vYsWP6p3/6Jx0/flzf/va39eGHH+rGG2+0+SPkBD9z/tM//VM988wzev7553X8+HE9/vjj6unpUWNjo80fYUEIzHEw6xcBWzI6Ouo98MADXl1dnRePx73vfve70+uqq6szrn/u6OjwNmzY4MViMe/mm2/23n33XQuJc9Nc5/zHf/zHXnV19Yx/W7dutZQ8t/h5Pv8+vifEHz9z/u///m+vsbHR+/SnP+3deOON3ptvvmkhcW7yM+dnnnnGW7t2rVdXV+fdcsstXmdnp4XEue+jrwVBOQ7med7/P/8CAABg0IJ8OwYAAAQfJQQAAFhBCQEAAFZQQgAAgBWUEAAAYAUlBAAAWEEJAQAAVlBCAACAFZQQAABgBSUEAABYQQkBAABWUEIAAIAV/w9+Gw190X4PgAAAAABJRU5ErkJggg=="
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "s = np.mgrid[0:1:11j, 0:1:11j]\n",
    "s = s.reshape(2, -1).T\n",
    "slice = 0.99\n",
    "s = s[s[:, 1] > -s[:, 0] + slice]\n",
    "s = np.concatenate([s, s, s], axis=0)\n",
    "sns.scatterplot(x=s[:, 0], y=s[:, 1])"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2024-05-17T08:56:50.469317Z",
     "start_time": "2024-05-17T08:56:50.337766Z"
    }
   },
   "id": "ac2a47802311112a"
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "outputs": [
    {
     "data": {
      "text/plain": "<Figure size 800x400 with 8 Axes>",
      "image/png": "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YiLntYmJibyP7IoJBoP4ZDKuWMbjtGF9W5dQvaStdDqNoaEhBIPBnDWsM68bFTNB5WImKpuJQn24+HWqTh9fjiq+73bUwcg34fA8QctRaDxbznlC14nExo0bUVdXh4GBAdx8880AgNOnTyMYDOZ8WQgAOjs78fOf/xyyLKOmpgayLOPMmTP47//9vwv9ztraWvicNnS3B3CiwEd03e0B+Jw2niwMora21lT7ipmg5WImKpOJQn1otr41q2LjmddpM/REgucJqoRKjWe6ftna6XRi+/bteOqpp3D27Fm88847OHToEB544AEA8zPseHx+Brxt2zaEw2H09/djZGQE/f39kCQJd955p/DvbWl0on9HEN3tgZzXMysPlPrUTKJKYyaIcumVCTKuYuPZGp+xxzOeJ6iqyDqLxWLy7t275a6uLrm7u1t+8cUXs++tX79ePnr0aPbnwcFBefv27XIwGJR37twpf/jhhyX/nlQqJb///vtyKpXKvvb5VEz+aCwkv/fJhPzRWEj+fCpWkW0i9RXan0qvGwkzQeVgJuZVOhNKr1N1W2o8M8P+5HmCylVony4nEzWyLMt6T2a0kEqlMDg4yHtfTaLY/eCdnZ3Ze0epMGbCXJiJ5WMmzEfpfnBmojhmwnwqnQnLTCQSiYShv3BIYoLBYN6Sd5SLmbAWZqI4ZsJamInimAlrKScTlplIzM3NIZVKYcWKFdknPJL5yLKMubk51NXV5X3pjHIxE9bATJSOmbAGZqJ0zIQ1LCcTlplIEBERERFR5XAqTkREREREwjiRICIiIiIiYZxIEBERERGRME4kiIiIiIhIGCcSREREREQkjBMJIiIiIiISxokEEREREREJs+xEYnZ2Fvv27cPNN9+M7u5uHDp0SO8mkaBEIoG77roL7733nt5NMRyR4/+jjz7Cvffei87OTvT19eHcuXMatpQWEtlvu3btwoYNG3L+vfvuuxq21hx4rjA2nifKx/OEcWl5rqirRION6ODBgzh37hxeeukljI2NYc+ePWhpacG2bdv0bhqVYHZ2Fo899hiGh4f1boohlXr8x2IxPPTQQ7j77rtx4MABvPrqq3j44Yfx9ttvw+Vy6dR66xIZtz755BP84Ac/wG233ZZ9zefzadlcU+C5wrh4nlgenieMS9NzhWxB0WhUDgaD8n/8x39kX/uHf/gH+b777tOx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N954Axs2bMA777wDADh48CA2bNigc8uI9DE2NoYnnngCsixDlmXcd999zEMBXP6ViIiIiIiEcdUmIiIiIiISxokEEREREREJ40SCiIiIiIiEcSJBRERERETCOJEgIiIiIiJhnEgQEREREZEwTiSIiIiIiEgYJxJERERERCSMEwkiIiIiIhLGiQQREREREQnjRIKIiIiIiIRxIkFERERERMI4kSAiIiIiImGcSBARERERkTBOJIiIiIiISFid3g3QiizLmJub07sZpJEVK1agpqZG72ZUNWbCWpiJ4pgJa2EmimMmrKWcTFhmIpFOpzE4OKh3M0gjnZ2dqKuzzOFdFmbCWpiJ4pgJa2EmimMmrKWcTFRNghKJBHp7e/HEE0/g1ltvLVjmo48+wpNPPomPP/4Y7e3t+N73voebbrpJ6PcEg0HU1tZmf06n0xgaGsp7narbUvst87oZMBMkgpmYx0zQQoX2HTPBTFhZpTNRFROJ2dlZPPbYYxgeHl6yTCwWw0MPPYS7774bBw4cwKuvvoqHH34Yb7/9NlwuV9Hfkfmopra2tuBBv9TrVN2W2m9G/7iamaByMRPMBOUrtO+YCWbCyiqVCd0nEiMjI3jssccgy7JiuTfeeAP19fXYvXs3ampqsH//fhw/fhxvvvkment7y/rdo1MxROIppPxrMXw5CrejDq3+4sEygkvhOKaiCYTjKXiddfC77Gj2OgqWHZuWEJKSCEtJ+Jw2eJ02tDQ6q75sZr8plTUiM2ZCZP+yXnGZvKf8azE8HlXMuxHpmQkyLjOf4814niBj0n0icerUKdx66634H//jf6Crq2vJcoODg9iyZUt2tlRTU4PNmzdjYGCgrDBcnIxi37EhnByZzL7W3R5A/44g2gINwvVVk/+cjOLxAtv29I4gvrRo20T6wWhljcpsmWC96tYrknej0isTZFxKeWttNP4k22znCTIu3Zd//fa3v419+/bB6VT+q9z4+DhWr16d81ogEMDvf/974d85OhXLCwIAnBiZxP5jQxidignXWS0uheN5FxXA/LbtOzaES+F49rWxaUmxH8amJcOWNTIzZUKtfcZ654nk3cj0yAQZV7Hx7PNp457jM8x0niBj0/0TiVJJkgS73Z7zmt1uRyKREKonnU4jEk/lBSHjxMgkIvEU0ul02W3V01Q0obhtU9EEmhpsAICQlFQs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"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "std = 0.00001\n",
    "z0 = s[:, 0] + std * np.random.randn(s.shape[0])\n",
    "z0[z0 < 0.5] = 0\n",
    "z1 = s[:, 0] + std * np.random.randn(s.shape[0])\n",
    "z1[z1 >= 0.5] = 0\n",
    "z2 = s[:, 1] + std * np.random.randn(s.shape[0])\n",
    "z2[z2 < 0.5] = 0\n",
    "z3 = s[:, 1] + std * np.random.randn(s.shape[0])\n",
    "z3[z3 >= 0.5] = 0\n",
    "z = np.stack([z0, z1, z2, z3], axis=1)\n",
    "ns = s.shape[1]\n",
    "nz = z.shape[1]\n",
    "fig, axes = plt.subplots(ns, nz, figsize=(2 * nz, 2 * ns))\n",
    "for i in range(ns):\n",
    "    for j in range(nz):\n",
    "        ax = axes[i][j]\n",
    "        sns.scatterplot(\n",
    "            ax=ax,\n",
    "            x=z[:, j],\n",
    "            y=s[:, i],\n",
    "            rasterized=True\n",
    "        )\n",
    "        ax.set_xlabel(rf'$z_{{{j}}}$')\n",
    "        ax.set_ylabel(rf'$s_{{{i}}}$')\n",
    "fig.tight_layout()"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2024-05-17T08:56:53.450169Z",
     "start_time": "2024-05-17T08:56:51.211117Z"
    }
   },
   "id": "534aae6778f28c1e"
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "outputs": [
    {
     "data": {
      "text/plain": "{'CInfoM': 0.9831470158524429,\n 'CInfoC': 0.5772916483231224,\n 'NCMI': array([[0.63501964, 0.2882705 , 0.        , 0.        ],\n        [0.        , 0.01005536, 0.59488401, 0.27226574]]),\n 'z_active': array([ True,  True,  True,  True])}"
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "compute_cinfomec(s, z)"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2024-05-17T08:56:55.494853Z",
     "start_time": "2024-05-17T08:56:55.345780Z"
    }
   },
   "id": "c6d0136151652546"
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "outputs": [],
   "source": [],
   "metadata": {
    "collapsed": false
   },
   "id": "fa4d60fb4bb8624e"
  }
 ],
 "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": 5
}
