{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "f870191f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import torch, os\n",
    "from torch import nn\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "023736b8",
   "metadata": {},
   "outputs": [],
   "source": [
    "os.environ[\"CUDA_DEVICE_ORDER\"] = \"PCI_BUS_ID\"\n",
    "os.environ[\"CUDA_VISIBLE_DEVICES\"] = str(6)\n",
    "\n",
    "device = 'cuda' if torch.cuda.is_available() else 'cpu'\n",
    "if device == 'cuda':\n",
    "    torch.backends.cudnn.benchmark = True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "2f9215a0",
   "metadata": {},
   "outputs": [],
   "source": [
    "def mlp(dim, hidden_dim, output_dim, layers, activation='relu', normalization=None):\n",
    "    activation = {\n",
    "        'relu': nn.ReLU\n",
    "    }[activation]\n",
    "\n",
    "    if isinstance(dim, list):\n",
    "        dim = np.prod(dim)\n",
    "\n",
    "    if normalization == \"layernorm\":\n",
    "        seq = [nn.Linear(dim, hidden_dim), nn.LayerNorm(hidden_dim), activation()]\n",
    "        for _ in range(layers):\n",
    "            seq += [nn.Linear(hidden_dim, hidden_dim), nn.LayerNorm(hidden_dim), activation()]\n",
    "        seq += [nn.Linear(hidden_dim, output_dim)]\n",
    "    elif normalization == \"batchnorm\":\n",
    "        seq = [nn.Linear(dim, hidden_dim), nn.BatchNorm1d(hidden_dim), activation()]\n",
    "        for _ in range(layers):\n",
    "            seq += [nn.Linear(hidden_dim, hidden_dim), nn.BatchNorm1d(hidden_dim), activation()]\n",
    "        seq += [nn.Linear(hidden_dim, output_dim)]\n",
    "    else:\n",
    "        seq = [nn.Linear(dim, hidden_dim), activation()]\n",
    "        for _ in range(layers):\n",
    "            seq += [nn.Linear(hidden_dim, hidden_dim), activation()]\n",
    "        seq += [nn.Linear(hidden_dim, output_dim)]\n",
    "\n",
    "    return nn.Sequential(*seq)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "214e05c1",
   "metadata": {},
   "outputs": [],
   "source": [
    "class JointCritic(nn.Module):\n",
    "    \"\"\"Concat critic, where we concat the inputs and use one MLP to output the value.\"\"\"\n",
    "\n",
    "    def __init__(self, dim, hidden_dim, layers, activation=\"relu\", normalization=\"None\", **extra_kwargs):\n",
    "        super(JointCritic, self).__init__()\n",
    "        # output is scalar score\n",
    "        self._f = mlp(dim * 2, hidden_dim, 1, layers, activation, normalization)\n",
    "\n",
    "    def forward(self, x, y):\n",
    "        batch_size = x.shape[0]\n",
    "        # Tile all possible combinations of x and y\n",
    "        x_tiled = torch.stack([x] * batch_size, dim=0)\n",
    "        y_tiled = torch.stack([y] * batch_size, dim=1)\n",
    "        # xy is [batch_size * batch_size, x_dim + y_dim]\n",
    "        xy_pairs = torch.reshape(torch.cat((x_tiled, y_tiled), dim=2), [\n",
    "                                 batch_size * batch_size, -1])\n",
    "        # Compute scores for each x_i, y_j pair.\n",
    "        scores = self._f(xy_pairs)\n",
    "        return torch.reshape(scores, [batch_size, batch_size]).t()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "4791206c",
   "metadata": {},
   "outputs": [],
   "source": [
    "def estimate_mutual_information(estimator, x, y, critic_fn,\n",
    "                                baseline_fn=None, alpha_logit=None, buffer=None,\n",
    "                                return_scores=False,\n",
    "                                **kwargs):\n",
    "    \"\"\"Estimate variational lower bounds on mutual information.\n",
    "\n",
    "  Args:\n",
    "    estimator: string specifying estimator, one of:\n",
    "      'nwj', 'infonce', 'tuba', 'js', 'interpolated'\n",
    "    x: [batch_size, dim_x] Tensor\n",
    "    y: [batch_size, dim_y] Tensor\n",
    "    critic_fn: callable that takes x and y as input and outputs critic scores\n",
    "      output shape is a [batch_size, batch_size] matrix\n",
    "    baseline_fn (optional): callable that takes y as input\n",
    "      outputs a [batch_size]  or [batch_size, 1] vector\n",
    "    alpha_logit (optional): logit(alpha) for interpolated bound\n",
    "\n",
    "  Returns:\n",
    "    scalar estimate of mutual information\n",
    "    \"\"\"\n",
    "    x, y = x.to(device), y.to(device)\n",
    "    scores = critic_fn(x, y)\n",
    "    if baseline_fn is not None:\n",
    "        # Some baselines' output is (batch_size, 1) which we remove here.\n",
    "        log_baseline = torch.squeeze(baseline_fn(y))\n",
    "    if estimator == 'infonce':\n",
    "        mi = infonce_lower_bound(scores)\n",
    "    elif estimator == 'nwj':\n",
    "        mi = nwj_lower_bound(scores)\n",
    "    elif estimator == 'tuba':\n",
    "        mi = tuba_lower_bound(scores, log_baseline)\n",
    "    elif estimator == 'js':\n",
    "        mi = js_lower_bound(scores)\n",
    "    elif estimator == \"smile-inf\":\n",
    "        mi = smile_lower_bound(scores, clip=np.inf)\n",
    "    elif estimator.startswith('smile') & (not estimator.endswith(\"source\")):\n",
    "        clip = eval(estimator.split(\"-\")[-1])\n",
    "        mi = smile_lower_bound(scores, clip=clip)\n",
    "    elif estimator == 'dv':\n",
    "        mi = dv_upper_lower_bound(scores)\n",
    "    elif estimator == \"mine\":\n",
    "        mi, buffer = mine_lower_bound(scores, buffer=buffer, momentum=0.9)\n",
    "\n",
    "    if return_scores & (estimator == \"mine\"):\n",
    "        return mi, buffer, scores\n",
    "    elif return_scores:\n",
    "        return mi, scores\n",
    "    elif estimator == \"mine\":\n",
    "        return mi, buffer\n",
    "    else:\n",
    "        return mi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "525d8f42",
   "metadata": {},
   "outputs": [],
   "source": [
    "def mine_lower_bound(f, buffer=None, momentum=0.9):\n",
    "    \"\"\"\n",
    "    MINE lower bound based on DV inequality.\n",
    "    \"\"\"\n",
    "    if buffer is None:\n",
    "        buffer = torch.tensor(1.0).cuda()\n",
    "    first_term = f.diag().mean()\n",
    "\n",
    "    buffer_update = logmeanexp_nodiag(f).exp()\n",
    "    with torch.no_grad():\n",
    "        second_term = logmeanexp_nodiag(f)\n",
    "        buffer_new = buffer * momentum + buffer_update * (1 - momentum)\n",
    "        buffer_new = torch.clamp(buffer_new, min=1e-4)\n",
    "        third_term_no_grad = buffer_update / buffer_new\n",
    "\n",
    "    third_term_grad = buffer_update / buffer_new\n",
    "\n",
    "    return first_term - second_term - third_term_grad + third_term_no_grad, buffer_update\n",
    "\n",
    "def smile_lower_bound(f, clip=None):\n",
    "    if clip is not None:\n",
    "        f_ = torch.clamp(f, -clip, clip)\n",
    "    else:\n",
    "        f_ = f\n",
    "    z = logmeanexp_nodiag(f_, dim=(0, 1))\n",
    "    dv = f.diag().mean() - z\n",
    "\n",
    "    js = js_fgan_lower_bound(f)\n",
    "\n",
    "    with torch.no_grad():\n",
    "        dv_js = dv - js\n",
    "\n",
    "    return js + dv_js\n",
    "\n",
    "def js_fgan_lower_bound(f):\n",
    "    \"\"\"Lower bound on Jensen-Shannon divergence from Nowozin et al. (2016).\"\"\"\n",
    "    f_diag = f.diag()\n",
    "    first_term = -torch.nn.functional.softplus(-f_diag).mean()\n",
    "    n = f.size(0)\n",
    "    second_term = (torch.sum(torch.nn.functional.softplus(f)) -\n",
    "                   torch.sum(torch.nn.functional.softplus(f_diag))) / (n * (n - 1.))\n",
    "    return first_term - second_term"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "de1cccb1",
   "metadata": {},
   "outputs": [],
   "source": [
    "def logmeanexp_nodiag(x, dim=None, device='cuda'):\n",
    "    batch_size = x.size(0)\n",
    "    if dim is None:\n",
    "        dim = (0, 1)\n",
    "\n",
    "    logsumexp = torch.logsumexp(\n",
    "        x - torch.diag(np.inf * torch.ones(batch_size).to(device)), dim=dim)\n",
    "\n",
    "    try:\n",
    "        if len(dim) == 1:\n",
    "            num_elem = batch_size - 1.\n",
    "        else:\n",
    "            num_elem = batch_size * (batch_size - 1.)\n",
    "    except ValueError:\n",
    "        num_elem = batch_size - 1\n",
    "    return logsumexp - torch.log(torch.tensor(num_elem)).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7f4d3e78",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "84ba6668",
   "metadata": {},
   "outputs": [],
   "source": [
    "from libs.data import TabularDataset\n",
    "from libs.transform import *"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "0bcdd0b1",
   "metadata": {},
   "outputs": [],
   "source": [
    "def BinBoundaries(dataset, X_num, bin_size, device):\n",
    "    init_num = torch.linspace(0, 1, bin_size + 1, device=device)\n",
    "    init_bin_boundaries = dict()\n",
    "    for f in range(dataset.size(1)):\n",
    "        if f in X_num:\n",
    "            init_bin_boundaries[f] = init_num\n",
    "        else:\n",
    "            init_bin_boundaries[f] = None\n",
    "    return init_bin_boundaries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "6d7705a2",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/openml/datasets/functions.py:438: FutureWarning: Starting from Version 0.15 `download_data`, `download_qualities`, and `download_features_meta_data` will all be ``False`` instead of ``True`` by default to enable lazy loading. To disable this message until version 0.15 explicitly set `download_data`, `download_qualities`, and `download_features_meta_data` to a bool while calling `get_dataset`.\n",
      "  warnings.warn(\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:605: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype):\n",
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/utils/validation.py:614: FutureWarning: is_sparse is deprecated and will be removed in a future version. Check `isinstance(dtype, pd.SparseDtype)` instead.\n",
      "  if is_sparse(pd_dtype) or not is_extension_array_dtype(pd_dtype):\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset is loaded.. Data name: analcatdata_authorship, Target feature: Author\n",
      "input dim: 70, cat: 29, num: 41\n",
      "Classes:  4\n",
      "torch.Size([672, 70])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/opt/conda/envs/tabular/lib/python3.9/site-packages/sklearn/preprocessing/_data.py:2627: UserWarning: n_quantiles (1000) is greater than the total number of samples (672). n_quantiles is set to n_samples.\n",
      "  warnings.warn(\n"
     ]
    }
   ],
   "source": [
    "openml_id = 458\n",
    "tasktype = \"multiclass\"\n",
    "\n",
    "dataset = TabularDataset(openml_id, tasktype, device, labeled_data=0, cat_threshold=20, seed=1, modelname=\"xgboost\")\n",
    "(X_train, y_train), (X_val, y_val), (X_test, y_test) = dataset._indv_dataset()\n",
    "\n",
    "if tasktype == \"multiclass\":\n",
    "    print(\"Classes: \", y_train.size(1))\n",
    "    y_train = np.argmax(y_train.cpu().numpy(), axis=1)\n",
    "    y_val = np.argmax(y_val.cpu().numpy(), axis=1)\n",
    "    y_test = np.argmax(y_test.cpu().numpy(), axis=1)\n",
    "else:\n",
    "    y_train = y_train.cpu().numpy()\n",
    "    y_val = y_val.cpu().numpy()\n",
    "    y_test = y_test.cpu().numpy()\n",
    "print(X_train.size())\n",
    "\n",
    "X_train = X_train.cpu().numpy()\n",
    "X_val = X_val.cpu().numpy()\n",
    "X_test = X_test.cpu().numpy()\n",
    "\n",
    "dim = X_test.shape[1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "9621d663",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "# mis = pd.DataFrame(columns=(\"data_id\", \"numbins\", \"mis\", \"estimated_mi\", \"std\", \"min\", \"max\"))\n",
    "# mis.to_csv(\"mis-mine.csv\")\n",
    "# mis = pd.read_csv(\"mis-smile-inf.csv\", index_col=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "f819ed27",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0 0 0.020448685\n",
      "0 200 16.538137\n",
      "0 400 8.625386\n",
      "0 600 2.653543\n",
      "0 800 37.62349\n",
      "0 1000 26.600195\n",
      "0 1200 2.9550295\n",
      "0 1400 12.596814\n",
      "0 1600 5.883375\n",
      "0 1800 23.064278\n",
      "0 2000 24.617334\n",
      "1 0 0.007958531\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/home/tabsemi/libs/transform.py:192: UserWarning: torch.searchsorted(): input value tensor is non-contiguous, this will lower the performance due to extra data copy when converting non-contiguous tensor to contiguous, please use contiguous input value tensor if possible. This message will only appear once per program. (Triggered internally at /opt/conda/conda-bld/pytorch_1682343997789/work/aten/src/ATen/native/BucketizationUtils.h:33.)\n",
      "  idx = torch.bucketize(img[:, f], boundaries, right=True) - 1\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1 200 12.850822\n",
      "1 400 10.737989\n",
      "1 600 30.82373\n",
      "1 800 37.060467\n",
      "1 1000 47.06947\n",
      "1 1200 13.683464\n",
      "1 1400 6.880762\n",
      "1 1600 43.664295\n",
      "1 1800 22.69957\n",
      "1 2000 90.17601\n",
      "2 0 0.028280973\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2 200 18.810387\n",
      "2 400 11.948334\n",
      "2 600 3.8607903\n",
      "2 800 8.829183\n",
      "2 1000 19.6865\n",
      "2 1200 31.246777\n",
      "2 1400 62.325077\n",
      "2 1600 21.181084\n",
      "2 1800 nan\n",
      "2 2000 nan\n",
      "3 0 0.012694836\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "3 200 17.4529\n",
      "3 400 21.177668\n",
      "3 600 22.058681\n",
      "3 800 30.290493\n",
      "3 1000 35.54952\n",
      "3 1200 111.55815\n",
      "3 1400 22.629135\n",
      "3 1600 19.235264\n",
      "3 1800 10.936808\n",
      "3 2000 12.868852\n",
      "4 0 0.022105098\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "4 200 7.6177273\n",
      "4 400 27.49345\n",
      "4 600 12.414673\n",
      "4 800 6.4601116\n",
      "4 1000 20.791767\n",
      "4 1200 27.14509\n",
      "4 1400 17.615034\n",
      "4 1600 22.207066\n",
      "4 1800 44.19555\n",
      "4 2000 22.663826\n",
      "5 0 0.02323103\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "5 200 12.646904\n",
      "5 400 5.0379906\n",
      "5 600 7.080393\n",
      "5 800 19.308533\n",
      "5 1000 0.9953842\n",
      "5 1200 10.550507\n",
      "5 1400 21.755957\n",
      "5 1600 20.417595\n",
      "5 1800 16.098896\n",
      "5 2000 1.6564136\n",
      "6 0 0.010635614\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "6 200 9.579094\n",
      "6 400 23.235775\n",
      "6 600 11.56867\n",
      "6 800 39.25039\n",
      "6 1000 6.349003\n",
      "6 1200 28.642162\n",
      "6 1400 21.760563\n",
      "6 1600 65.464096\n",
      "6 1800 131.8916\n",
      "6 2000 10.826593\n",
      "7 0 0.012179136\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "7 200 12.395914\n",
      "7 400 33.262573\n",
      "7 600 8.940431\n",
      "7 800 16.025177\n",
      "7 1000 13.699283\n",
      "7 1200 8.808279\n",
      "7 1400 63.955376\n",
      "7 1600 11.2872505\n",
      "7 1800 47.942604\n",
      "7 2000 31.068972\n",
      "8 0 0.013889492\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "8 200 23.200413\n",
      "8 400 38.254257\n",
      "8 600 43.747017\n",
      "8 800 7.5148306\n",
      "8 1000 25.260902\n",
      "8 1200 17.293133\n",
      "8 1400 2.8421211\n",
      "8 1600 12.065629\n",
      "8 1800 34.571312\n",
      "8 2000 11.864893\n",
      "9 0 0.019058347\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "9 200 8.555689\n",
      "9 400 18.612701\n",
      "9 600 10.605102\n",
      "9 800 45.946762\n",
      "9 1000 9.827658\n",
      "9 1200 12.205917\n",
      "9 1400 14.809736\n",
      "9 1600 11.729019\n",
      "9 1800 37.676216\n",
      "9 2000 19.063276\n",
      "10 0 0.0073525906\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "10 200 21.621864\n",
      "10 400 18.39641\n",
      "10 600 30.862087\n",
      "10 800 36.69252\n",
      "10 1000 65.70222\n",
      "10 1200 36.477215\n",
      "10 1400 42.311867\n",
      "10 1600 76.48491\n",
      "10 1800 25.954624\n",
      "10 2000 22.748596\n",
      "11 0 0.02376914\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "11 200 20.503761\n",
      "11 400 32.681625\n",
      "11 600 30.471926\n",
      "11 800 11.701877\n",
      "11 1000 31.421711\n",
      "11 1200 21.539314\n",
      "11 1400 16.342148\n",
      "11 1600 27.507717\n",
      "11 1800 16.411911\n",
      "11 2000 119.791466\n",
      "12 0 0.024297774\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "12 200 19.295887\n",
      "12 400 12.601341\n",
      "12 600 10.106285\n",
      "12 800 10.347508\n",
      "12 1000 22.49419\n",
      "12 1200 -3.8603363\n",
      "12 1400 87.52559\n",
      "12 1600 21.012989\n",
      "12 1800 10.738381\n",
      "12 2000 17.27935\n",
      "13 0 0.028173804\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "13 200 25.543789\n",
      "13 400 41.35234\n",
      "13 600 31.563267\n",
      "13 800 10.079128\n",
      "13 1000 -3.7799835\n",
      "13 1200 33.49786\n",
      "13 1400 68.006836\n",
      "13 1600 -3.4865532\n",
      "13 1800 35.964066\n",
      "13 2000 56.145306\n",
      "14 0 0.011146963\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "14 200 15.488853\n",
      "14 400 33.636726\n",
      "14 600 23.017004\n",
      "14 800 14.615082\n",
      "14 1000 1.115674\n",
      "14 1200 15.025662\n",
      "14 1400 9.516499\n",
      "14 1600 13.778185\n",
      "14 1800 24.907597\n",
      "14 2000 7.673937\n",
      "15 0 0.02310121\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "15 200 22.107327\n",
      "15 400 34.96032\n",
      "15 600 6.281735\n",
      "15 800 13.515734\n",
      "15 1000 36.66394\n",
      "15 1200 11.203803\n",
      "15 1400 21.056625\n",
      "15 1600 3.8452072\n",
      "15 1800 39.33282\n",
      "15 2000 148.53336\n",
      "16 0 0.019071102\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "16 200 21.636347\n",
      "16 400 14.122879\n",
      "16 600 33.755863\n",
      "16 800 1.2058296\n",
      "16 1000 24.68666\n",
      "16 1200 98.1014\n",
      "16 1400 33.54059\n",
      "16 1600 23.536097\n",
      "16 1800 48.85627\n",
      "16 2000 18.185137\n",
      "17 0 0.022081971\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "17 200 17.482784\n",
      "17 400 13.613707\n",
      "17 600 45.48319\n",
      "17 800 11.743759\n",
      "17 1000 10.740321\n",
      "17 1200 8.293402\n",
      "17 1400 48.266052\n",
      "17 1600 11.368778\n",
      "17 1800 -20.91354\n",
      "17 2000 5.467993\n",
      "18 0 0.015907168\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "18 200 13.17353\n",
      "18 400 40.863953\n",
      "18 600 37.239\n",
      "18 800 28.184559\n",
      "18 1000 20.025312\n",
      "18 1200 -13.4621315\n",
      "18 1400 6.7096643\n",
      "18 1600 5.4506307\n",
      "18 1800 3.862577\n",
      "18 2000 9.466761\n",
      "19 0 0.019109964\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1703466/2700193231.py:29: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
      "/tmp/ipykernel_1703466/2700193231.py:30: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "19 200 17.137526\n",
      "19 400 7.2456603\n",
      "19 600 7.799402\n",
      "19 800 23.198795\n",
      "19 1000 34.132355\n",
      "19 1200 71.7827\n",
      "19 1400 21.963871\n",
      "19 1600 15.394323\n",
      "19 1800 19.917095\n",
      "19 2000 115.44333\n"
     ]
    }
   ],
   "source": [
    "estimator = \"mine\"\n",
    "fname = f'mis-{estimator}.csv'\n",
    "for b in np.arange(0, 20):\n",
    "    mis = pd.read_csv(fname, index_col=0)\n",
    "    if len(mis[(mis[\"numbins\"] == b) & (mis[\"data_id\"] == openml_id)]) == 0:\n",
    "        critic = JointCritic(dim, hidden_dim=256, layers=2).to(device)\n",
    "        opt_crit = torch.optim.Adam(critic.parameters(), lr=0.0005)\n",
    "\n",
    "        def train_step(x, y, buffer=None):\n",
    "            opt_crit.zero_grad()\n",
    "            if estimator == \"mine\":\n",
    "                mi, buffer = estimate_mutual_information(estimator, x, y, critic, baseline_fn=None, buffer=buffer)\n",
    "            else:\n",
    "                mi = estimate_mutual_information(estimator, x, y, critic, baseline_fn=None, buffer=buffer)\n",
    "            loss = -mi\n",
    "            loss.backward(); opt_crit.step()\n",
    "            return mi, buffer\n",
    "        \n",
    "        if b > 0:\n",
    "            bin_boundaries = BinBoundaries(torch.tensor(X_train), dataset.X_num, b, \"cpu\")\n",
    "            binshuffling = BinShuffling(1, bin_boundaries, sampling_alpha=False)\n",
    "\n",
    "        estimates = []; buffer = None\n",
    "\n",
    "        loader = torch.utils.data.DataLoader(dataset=X_train, batch_size=512, shuffle=True)\n",
    "        for i in range(4000 // len(loader) + 1):\n",
    "            for _, x in enumerate(loader):\n",
    "                if b > 0:\n",
    "                    sample = binshuffling({\"image\": torch.tensor(x), \"mask\": None})\n",
    "                    mi, buffer = train_step(x, torch.tensor(sample[\"image\"]), buffer=buffer)\n",
    "                else:\n",
    "                    mi, buffer = train_step(x, x, buffer=buffer)\n",
    "                \n",
    "                mi = mi.detach().cpu().numpy()\n",
    "                estimates.append(mi.item())\n",
    "\n",
    "            if i % 200 == 0:\n",
    "                print(b, i, mi)\n",
    "\n",
    "        mis.loc[len(mis)] = [openml_id, b, estimates, np.mean(estimates[-1000:]), np.std(estimates[-1000:]), \n",
    "                             np.mean(estimates[-1000:]) - np.std(estimates[-1000:]), np.mean(estimates[-1000:]) + np.std(estimates[-1000:])]\n",
    "        mis.to_csv(fname)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f6ea9f31",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "6027568a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# import matplotlib.pyplot as plt\n",
    "# q = mis[(mis[\"data_id\"] == 458) & (mis[\"numbins\"] == 10)][\"mis\"].values[0]\n",
    "# plt.plot(q)\n",
    "# plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "b20e5eeb",
   "metadata": {},
   "outputs": [],
   "source": [
    "# mis.loc[mis[\"data_id\"] == openml_id, 'mis'] = mis.loc[mis[\"data_id\"] == openml_id, 'mis'].apply(lambda x: [value for value in x if not np.isnan(value)])\n",
    "# mis.loc[mis[\"data_id\"] == openml_id, \"estimated_mi\"] = mis.loc[mis[\"data_id\"] == openml_id, 'mis'].apply(lambda x: np.mean(x[-1000:]))\n",
    "# mis.loc[mis[\"data_id\"] == openml_id, \"std\"] = mis.loc[mis[\"data_id\"] == openml_id, 'mis'].apply(lambda x: np.std(x[-1000:]))\n",
    "# mis.loc[mis[\"data_id\"] == openml_id, \"min\"] = mis.loc[mis[\"data_id\"] == openml_id, 'mis'].apply(lambda x: np.min(x[-1000:]))\n",
    "# mis.loc[mis[\"data_id\"] == openml_id, \"max\"] = mis.loc[mis[\"data_id\"] == openml_id, 'mis'].apply(lambda x: np.max(x[-1000:]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "3ff6295a",
   "metadata": {},
   "outputs": [],
   "source": [
    "mis = mis[mis[\"data_id\"] == openml_id]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "15393d0f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "base = mis[mis[\"numbins\"] == 0][\"estimated_mi\"].values[0]\n",
    "p = mis[mis[\"numbins\"] > 0]\n",
    "plt.figure(figsize=(4, 3))\n",
    "plt.plot(p[\"numbins\"].values, p[\"estimated_mi\"].values / base, marker=\"o\")\n",
    "plt.fill_between(p[\"numbins\"].values, p[\"min\"].values / base, p[\"max\"].values / base, alpha=0.3)\n",
    "plt.hlines(y=1, xmin=1, xmax=19, linewidth=1, linestyle=\"dashed\", color=\"red\")\n",
    "plt.xlabel(\"Number of bins\")\n",
    "plt.ylabel(\"Estimated MI / Base MI\")\n",
    "plt.xticks(np.arange(0, 20, 2))\n",
    "# plt.ylim(ymin=-0.1)\n",
    "plt.grid()\n",
    "plt.title(openml_id)\n",
    "plt.savefig(f'figs/mis/{openml_id}-{estimator}.png', bbox_inches=\"tight\", dpi=128)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "b7060453",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'mine'"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "estimator"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "83108f85",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "401edf41",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0393c01e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6e954d5c",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "47a0da7f",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "538ff542",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "f3dbe5bb",
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'z1' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Input \u001b[0;32mIn [14]\u001b[0m, in \u001b[0;36m<cell line: 4>\u001b[0;34m()\u001b[0m\n\u001b[1;32m      1\u001b[0m estimates \u001b[38;5;241m=\u001b[39m []\n\u001b[1;32m      2\u001b[0m buffer \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n\u001b[0;32m----> 4\u001b[0m mi, buffer \u001b[38;5;241m=\u001b[39m train_step(\u001b[43mz1\u001b[49m\u001b[38;5;241m.\u001b[39mview(args\u001b[38;5;241m.\u001b[39mbatch_size, \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m)\u001b[38;5;241m.\u001b[39mcuda(), z2\u001b[38;5;241m.\u001b[39mview(args\u001b[38;5;241m.\u001b[39mbatch_size, \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m)\u001b[38;5;241m.\u001b[39mcuda(), mi_params, buffer\u001b[38;5;241m=\u001b[39mbuffer)\n\u001b[1;32m      5\u001b[0m mi \u001b[38;5;241m=\u001b[39m mi\u001b[38;5;241m.\u001b[39mdetach()\u001b[38;5;241m.\u001b[39mcpu()\u001b[38;5;241m.\u001b[39mnumpy()\n",
      "\u001b[0;31mNameError\u001b[0m: name 'z1' is not defined"
     ]
    }
   ],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9b261718",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "tabular",
   "language": "python",
   "name": "tabular"
  },
  "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.16"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
