{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Tutorial for Disjoint Generative Models\n",
    "This notebook is a tutorial for using the disjoint generative models library, and should outline different functionalities to help you work with the library. The tutorial is split into three examples, the first one is a getting started example, the second one outlines the different joining strategies that are implemented in the library, and the third one demonstrates calibration and validator optimisation."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Example 1: Getting Started with DGMs\n",
    "\n",
    "First we do a very rudimentary example of DGMs on a simple dataset. We specify two models ```synthpop``` and ```privbayes``` to each be responsible for one part of the dataset. \n",
    "\n",
    "Unless otherwise specified, the dataset manager module will randomly split the dataset into equal parts for each model."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Imports\n",
    "import pandas as pd\n",
    "from disjoint_generation import DisjointGenerativeModels"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DataManager: The exterior correlations are 0.89 times that of the interiors.\n",
      "Threshold auto-set to: 0.5303119727915493\n",
      "{'split0': ['sex', 'thal', 'chol', 'thalach', 'target', 'exang', 'cp'], 'split1': ['age', 'slope', 'trestbps', 'ca', 'fbs', 'restecg', 'oldpeak']}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>age</th>\n",
       "      <th>sex</th>\n",
       "      <th>cp</th>\n",
       "      <th>trestbps</th>\n",
       "      <th>chol</th>\n",
       "      <th>fbs</th>\n",
       "      <th>restecg</th>\n",
       "      <th>thalach</th>\n",
       "      <th>exang</th>\n",
       "      <th>oldpeak</th>\n",
       "      <th>slope</th>\n",
       "      <th>ca</th>\n",
       "      <th>thal</th>\n",
       "      <th>target</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>61</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>104</td>\n",
       "      <td>256</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>141</td>\n",
       "      <td>1</td>\n",
       "      <td>3.8</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>67</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>110</td>\n",
       "      <td>269</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>99</td>\n",
       "      <td>1</td>\n",
       "      <td>2.4</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>51</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>192</td>\n",
       "      <td>318</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>117</td>\n",
       "      <td>1</td>\n",
       "      <td>2.2</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>58</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>192</td>\n",
       "      <td>249</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>96</td>\n",
       "      <td>1</td>\n",
       "      <td>2.2</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>74</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>102</td>\n",
       "      <td>302</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>125</td>\n",
       "      <td>1</td>\n",
       "      <td>3.2</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   age  sex  cp  trestbps  chol  fbs  restecg  thalach  exang  oldpeak  slope  \\\n",
       "0   61    1   0       104   256    1        0      141      1      3.8      0   \n",
       "1   67    1   0       110   269    0        1       99      1      2.4      1   \n",
       "2   51    0   0       192   318    0        2      117      1      2.2      0   \n",
       "3   58    1   0       192   249    0        2       96      1      2.2      2   \n",
       "4   74    1   0       102   302    1        0      125      1      3.2      1   \n",
       "\n",
       "   ca  thal  target  \n",
       "0   2     3       0  \n",
       "1   2     3       0  \n",
       "2   0     2       0  \n",
       "3   0     2       0  \n",
       "4   2     2       0  "
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Load the training data\n",
    "df_train = pd.read_csv('experiments/datasets/heart_train.csv')\n",
    "\n",
    "# Define a DGM using the Synthpop CART model and PrivBayes BN\n",
    "dgms = DisjointGenerativeModels(df_train, generative_models=['synthpop', 'privbayes'])\n",
    "df_syn = dgms.fit_generate(num_samples=20)\n",
    "print(dgms.used_splits)\n",
    "df_syn.head()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "If we want to specify the split, we can do so by passing a dictionary to the model containing the column names. E.g.,\n",
    "\n",
    "```python\t\n",
    "prepared_splits = {\n",
    "    \"part1\": [\"age\", \"sex\", \"cp\", \"trestbps\", \"chol\"],\n",
    "    \"part2\": [\"fbs\", \"restecg\", \"thalach\", \"exang\", \"oldpeak\", \"slope\", \"ca\", \"thal\", \"target\"]\n",
    "}\n",
    "\n",
    "dgms = DisjointGenerativeModels(df_train, generative_models=['synthpop', 'privbayes'], prepared_splits=prepared_splits)\n",
    "```\n",
    "Alternatively, we can specify the split by passing a dictionary with model names as keys and the corresponding column names as values (note that with this method one cannot specify using the same model for two different partitions).\n",
    "\n",
    "```python\n",
    "gms_splits = {\n",
    "    \"synthpop\": [\"age\", \"sex\", \"cp\", \"trestbps\", \"chol\"],\n",
    "    \"privbayes\": [\"fbs\", \"restecg\", \"thalach\", \"exang\", \"oldpeak\", \"slope\", \"ca\", \"thal\", \"target\"]\n",
    "}\n",
    "\n",
    "dgms = DisjointGenerativeModels(df_train, generative_models=gms_splits)\n",
    "```\n",
    "Finally, it is also possible to specify the number of equal-sized parts rather than the specific columns in both of the above methods. E.g. send 2 parts to the synthpop model and 1 part to the PrivBayes model:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "c:\\Users\\lautrup\\Documents\\GitHub\\disjoint-synthetic-data-generation\\.venv\\lib\\site-packages\\disjoint_generation\\utils\\dataset_manager.py:57: UserWarning: Split sizes adjusted to {'split0': 10, 'split1': 4}\n",
      "  warnings.warn(f\"Split sizes adjusted to {split_sizes}\")\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DataManager: The exterior correlations are 1.03 times that of the interiors.\n",
      "{'split0': ['chol', 'oldpeak', 'age', 'slope', 'target', 'thal', 'cp', 'sex', 'ca', 'fbs'], 'split1': ['trestbps', 'restecg', 'exang', 'thalach']}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>age</th>\n",
       "      <th>sex</th>\n",
       "      <th>cp</th>\n",
       "      <th>trestbps</th>\n",
       "      <th>chol</th>\n",
       "      <th>fbs</th>\n",
       "      <th>restecg</th>\n",
       "      <th>thalach</th>\n",
       "      <th>exang</th>\n",
       "      <th>oldpeak</th>\n",
       "      <th>slope</th>\n",
       "      <th>ca</th>\n",
       "      <th>thal</th>\n",
       "      <th>target</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>44</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>164</td>\n",
       "      <td>243</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>139</td>\n",
       "      <td>0</td>\n",
       "      <td>0.3</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>55</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>94</td>\n",
       "      <td>309</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>125</td>\n",
       "      <td>0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>54</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>114</td>\n",
       "      <td>254</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>114</td>\n",
       "      <td>0</td>\n",
       "      <td>0.8</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>62</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>145</td>\n",
       "      <td>340</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>165</td>\n",
       "      <td>0</td>\n",
       "      <td>1.8</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>64</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>156</td>\n",
       "      <td>211</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>169</td>\n",
       "      <td>1</td>\n",
       "      <td>1.4</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   age  sex  cp  trestbps  chol  fbs  restecg  thalach  exang  oldpeak  slope  \\\n",
       "0   44    1   0       164   243    0        1      139      0      0.3      1   \n",
       "1   55    1   0        94   309    0        0      125      0      2.0      1   \n",
       "2   54    1   0       114   254    0        0      114      0      0.8      1   \n",
       "3   62    1   0       145   340    0        0      165      0      1.8      1   \n",
       "4   64    1   2       156   211    0        0      169      1      1.4      1   \n",
       "\n",
       "   ca  thal  target  \n",
       "0   1     2       0  \n",
       "1   1     3       0  \n",
       "2   1     3       0  \n",
       "3   2     3       0  \n",
       "4   0     3       0  "
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dgms = DisjointGenerativeModels(df_train, generative_models={'synthpop': 2, 'privbayes': 1})\n",
    "df_syn = dgms.fit_generate(num_samples=5)\n",
    "print(dgms.used_splits)\n",
    "\n",
    "df_syn"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Finally, we can also import the method used for randomly splitting the dataset and use it to split the dataset ourselves. This is helpful if we want to use the same split for multiple models, but we don't want to specify the split manually."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'part1': ['sex', 'thalach', 'exang', 'chol', 'cp', 'fbs', 'age', 'slope'],\n",
       " 'part2': ['thal', 'restecg', 'target'],\n",
       " 'part3': ['ca', 'trestbps', 'oldpeak']}"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from disjoint_generation.utils.dataset_manager import random_split_columns\n",
    "\n",
    "random_split = random_split_columns(df_train, {'part1': 2, 'part2': 1, 'part3': 1})\n",
    "random_split"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Example 2: Joining Strategies\n",
    "\n",
    "The DGMs framework allows for virtually any sort of joining procedure. In this library the following joining strategies are implemented:\n",
    "\n",
    "- ```Concatenating```: Simply concatenates the synthetic data generated by each model. See also `RandomJoining`, which does the same but also shuffles the indecies (useful for debugging or partial synthesis).\n",
    "\n",
    "- ```UsingJoiningValidator```: Strategy for joining the synthetic data using a validator model. The validator model can use three different adapters ```JoiningValidator```, ```OneClassValidator``` and ```OutlierValidator```, the first admits binary classification model backends, the second one-class/outlier detection models, and the final uses outlier detection methods such as isolation forest. They assign prediction scores to querry joins on the synthetic samples repeadedly subject to various control parameters. Accepted joins are removed from the pool for the next round. \n",
    "\n",
    "The ```UsingJoiningValidator``` strategy has various control parameters that can be overwritten by the user, but for most regular use, the ```'behaviour'``` argument acts as a shorthand for selecting pre-configured option sets. The following behaviours are available:\n",
    "- ```'adaptive'```: The parameters are adjusted during the joining process to get more items, the selection threshold is automatically inferred. \n",
    "- ```'standard'```: Inherits the default settings from the adapter.\n",
    "- ```'strict'```: No parameters are changed during the joining process (likely to fail in getting enough good joins, consider adjusting the `'join_multiplier'` attribute of the DGMs object).\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Imports\n",
    "import pandas as pd\n",
    "\n",
    "from disjoint_generation import DisjointGenerativeModels\n",
    "from disjoint_generation.utils.joining_validator import JoiningValidator, OneClassValidator\n",
    "from disjoint_generation.utils.joining_strategies import UsingJoiningValidator"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DGMs (datamanager): Split sizes were adjusted to match the number of columns. requested_ratios={'split0': 2, 'split1': 1}, n_columns=14, adjusted_split_sizes={'split0': 10, 'split1': 4}\n",
      "DGMs (datamanager): The exterior correlations are 1.11 times that of the interiors.\n",
      "DGMs (validator): Threshold auto-set to: 0.7692307692307693\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>age</th>\n",
       "      <th>sex</th>\n",
       "      <th>cp</th>\n",
       "      <th>trestbps</th>\n",
       "      <th>chol</th>\n",
       "      <th>fbs</th>\n",
       "      <th>restecg</th>\n",
       "      <th>thalach</th>\n",
       "      <th>exang</th>\n",
       "      <th>oldpeak</th>\n",
       "      <th>slope</th>\n",
       "      <th>ca</th>\n",
       "      <th>thal</th>\n",
       "      <th>target</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>49</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>110</td>\n",
       "      <td>177</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>120</td>\n",
       "      <td>1</td>\n",
       "      <td>1.8</td>\n",
       "      <td>1</td>\n",
       "      <td>3</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>77</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>140</td>\n",
       "      <td>278</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>115</td>\n",
       "      <td>0</td>\n",
       "      <td>1.2</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>64</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>130</td>\n",
       "      <td>417</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>115</td>\n",
       "      <td>0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>44</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>134</td>\n",
       "      <td>219</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>194</td>\n",
       "      <td>0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>60</td>\n",
       "      <td>1</td>\n",
       "      <td>3</td>\n",
       "      <td>128</td>\n",
       "      <td>282</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>125</td>\n",
       "      <td>0</td>\n",
       "      <td>0.1</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>237</th>\n",
       "      <td>57</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>120</td>\n",
       "      <td>269</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>179</td>\n",
       "      <td>0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>238</th>\n",
       "      <td>66</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>140</td>\n",
       "      <td>228</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>103</td>\n",
       "      <td>0</td>\n",
       "      <td>1.3</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>239</th>\n",
       "      <td>53</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>138</td>\n",
       "      <td>299</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>162</td>\n",
       "      <td>0</td>\n",
       "      <td>1.5</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>240</th>\n",
       "      <td>48</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>125</td>\n",
       "      <td>271</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>180</td>\n",
       "      <td>0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>241</th>\n",
       "      <td>62</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>128</td>\n",
       "      <td>258</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>120</td>\n",
       "      <td>1</td>\n",
       "      <td>2.9</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>242 rows × 14 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "     age  sex  cp  trestbps  chol  fbs  restecg  thalach  exang  oldpeak  \\\n",
       "0     49    1   0       110   177    0        0      120      1      1.8   \n",
       "1     77    0   3       140   278    0        1      115      0      1.2   \n",
       "2     64    0   2       130   417    0        2      115      0      2.0   \n",
       "3     44    1   2       134   219    0        1      194      0      0.0   \n",
       "4     60    1   3       128   282    1        0      125      0      0.1   \n",
       "..   ...  ...  ..       ...   ...  ...      ...      ...    ...      ...   \n",
       "237   57    0   0       120   269    0        0      179      0      0.0   \n",
       "238   66    0   2       140   228    0        2      103      0      1.3   \n",
       "239   53    0   0       138   299    0        0      162      0      1.5   \n",
       "240   48    1   2       125   271    0        0      180      0      0.0   \n",
       "241   62    1   2       128   258    0        1      120      1      2.9   \n",
       "\n",
       "     slope  ca  thal  target  \n",
       "0        1   3     3       0  \n",
       "1        1   1     2       1  \n",
       "2        1   2     2       0  \n",
       "3        2   1     2       1  \n",
       "4        1   0     2       1  \n",
       "..     ...  ..   ...     ...  \n",
       "237      2   0     2       1  \n",
       "238      1   0     3       0  \n",
       "239      2   0     2       1  \n",
       "240      2   0     2       1  \n",
       "241      1   1     3       0  \n",
       "\n",
       "[242 rows x 14 columns]"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Load the training data\n",
    "df_train = pd.read_csv('experiments/datasets/heart_train.csv')\n",
    "\n",
    "gms = {'synthpop': 2, 'privbayes': 1}\n",
    "\n",
    "JS = UsingJoiningValidator()    # JoiningValidator with random forest model is used by default\n",
    "dgms1 = DisjointGenerativeModels(df_train, gms, joining_strategy=JS)\n",
    "\n",
    "df_syn1 = dgms1.fit_generate()\n",
    "\n",
    "df_syn1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DGMs (datamanager): Split sizes were adjusted to match the number of columns. requested_ratios={'split0': 2, 'split1': 1}, n_columns=14, adjusted_split_sizes={'split0': 10, 'split1': 4}\n",
      "DGMs (datamanager): The exterior correlations are 0.99 times that of the interiors.\n",
      "DGMs (validator): Verbose results:\n",
      "- F1-Score (Good Joins): [0.5858585858585859, 0.5052631578947369, 0.5111111111111111, 0.4742268041237113, 0.4631578947368421]\n",
      "- Mean F1: 0.5079235107449975\n",
      "Final model trained!\n",
      "DGMs (validator): Threshold auto-set to: 2.511968916847195\n",
      "- Predicted good joins fraction: 0.10055096418732783\n",
      "- Predicted good joins fraction: 0.042879019908116385\n",
      "- Predicted good joins fraction: 0.0192\n",
      "- Predicted good joins fraction: 0.01794453507340946\n",
      "- Predicted good joins fraction: 0.009966777408637873\n",
      "- Predicted good joins fraction: 0.003355704697986577\n",
      "- Predicted good joins fraction: 0.0\n",
      "- Predicted good joins fraction: 0.010101010101010102\n",
      "- Predicted good joins fraction: 0.008503401360544218\n",
      "- Predicted good joins fraction: 0.00686106346483705\n",
      "- Predicted good joins fraction: 0.0017271157167530224\n",
      "- Predicted good joins fraction: 0.005190311418685121\n",
      "- Predicted good joins fraction: 0.0034782608695652175\n",
      "- Predicted good joins fraction: 0.0034904013961605585\n",
      "- Predicted good joins fraction: 0.0017513134851138354\n",
      "- Predicted good joins fraction: 0.0\n",
      "- Predicted good joins fraction: 0.0017543859649122807\n",
      "- Predicted good joins fraction: 0.007029876977152899\n",
      "- Predicted good joins fraction: 0.0035398230088495575\n",
      "- Predicted good joins fraction: 0.0\n",
      "- Predicted good joins fraction: 0.014209591474245116\n",
      "- Predicted good joins fraction: 0.007207207207207207\n",
      "- Predicted good joins fraction: 0.003629764065335753\n",
      "- Predicted good joins fraction: 0.0018214936247723133\n",
      "- Predicted good joins fraction: 0.0018248175182481751\n",
      "- Predicted good joins fraction: 0.003656307129798903\n",
      "- Predicted good joins fraction: 0.005504587155963303\n",
      "- Predicted good joins fraction: 0.0\n",
      "- Predicted good joins fraction: 0.0036900369003690036\n",
      "- Predicted good joins fraction: 0.007407407407407408\n",
      "- Predicted good joins fraction: 0.007462686567164179\n",
      "- Predicted good joins fraction: 0.0018796992481203006\n",
      "- Predicted good joins fraction: 0.003766478342749529\n",
      "- Predicted good joins fraction: 0.007561436672967864\n",
      "- Predicted good joins fraction: 0.0\n",
      "- Predicted good joins fraction: 0.009523809523809525\n",
      "- Predicted good joins fraction: 0.0038461538461538464\n",
      "- Predicted good joins fraction: 0.0\n",
      "- Predicted good joins fraction: 0.005791505791505791\n",
      "- Predicted good joins fraction: 0.007766990291262136\n",
      "- Predicted good joins fraction: 0.011741682974559686\n",
      "- Predicted good joins fraction: 0.0\n",
      "- Predicted good joins fraction: 0.0039603960396039604\n",
      "- Predicted good joins fraction: 0.005964214711729622\n",
      "- Predicted good joins fraction: 0.006\n",
      "- Predicted good joins fraction: 0.006036217303822937\n",
      "- Predicted good joins fraction: 0.004048582995951417\n",
      "- Predicted good joins fraction: 0.006097560975609756\n",
      "- Predicted good joins fraction: 0.0\n",
      "- Predicted good joins fraction: 0.00408997955010225\n",
      "- Predicted good joins fraction: 0.004106776180698152\n",
      "- Predicted good joins fraction: 0.004123711340206186\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>age</th>\n",
       "      <th>sex</th>\n",
       "      <th>cp</th>\n",
       "      <th>trestbps</th>\n",
       "      <th>chol</th>\n",
       "      <th>fbs</th>\n",
       "      <th>restecg</th>\n",
       "      <th>thalach</th>\n",
       "      <th>exang</th>\n",
       "      <th>oldpeak</th>\n",
       "      <th>slope</th>\n",
       "      <th>ca</th>\n",
       "      <th>thal</th>\n",
       "      <th>target</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>53</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>120</td>\n",
       "      <td>261</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>149</td>\n",
       "      <td>1</td>\n",
       "      <td>1.4</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>62</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>120</td>\n",
       "      <td>253</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>140</td>\n",
       "      <td>1</td>\n",
       "      <td>2.8</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>67</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>128</td>\n",
       "      <td>236</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>152</td>\n",
       "      <td>0</td>\n",
       "      <td>1.5</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>46</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>118</td>\n",
       "      <td>246</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>162</td>\n",
       "      <td>1</td>\n",
       "      <td>1.4</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>67</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>120</td>\n",
       "      <td>255</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>146</td>\n",
       "      <td>1</td>\n",
       "      <td>2.2</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   age  sex  cp  trestbps  chol  fbs  restecg  thalach  exang  oldpeak  slope  \\\n",
       "0   53    1   2       120   261    0        0      149      1      1.4      2   \n",
       "1   62    1   0       120   253    0        1      140      1      2.8      0   \n",
       "2   67    1   2       128   236    0        0      152      0      1.5      1   \n",
       "3   46    0   3       118   246    0        0      162      1      1.4      2   \n",
       "4   67    0   0       120   255    0        0      146      1      2.2      1   \n",
       "\n",
       "   ca  thal  target  \n",
       "0   0     3       0  \n",
       "1   3     1       1  \n",
       "2   0     2       1  \n",
       "3   0     2       1  \n",
       "4   2     3       0  "
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "JS = UsingJoiningValidator(OneClassValidator(), behaviour='adaptive', threshold_decay=0.1)\n",
    "dgms2 = DisjointGenerativeModels(df_train, gms, joining_strategy=JS)\n",
    "\n",
    "df_syn2 = dgms2.fit_generate()\n",
    "\n",
    "df_syn2.head()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DGMs (datamanager): Split sizes were adjusted to match the number of columns. requested_ratios={'split0': 2, 'split1': 1}, n_columns=14, adjusted_split_sizes={'split0': 10, 'split1': 4}\n",
      "DGMs (datamanager): The exterior correlations are 1.17 times that of the interiors.\n",
      "DGMs (validator): Threshold auto-set to: 0.5591397849462365\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>age</th>\n",
       "      <th>sex</th>\n",
       "      <th>cp</th>\n",
       "      <th>trestbps</th>\n",
       "      <th>chol</th>\n",
       "      <th>fbs</th>\n",
       "      <th>restecg</th>\n",
       "      <th>thalach</th>\n",
       "      <th>exang</th>\n",
       "      <th>oldpeak</th>\n",
       "      <th>slope</th>\n",
       "      <th>ca</th>\n",
       "      <th>thal</th>\n",
       "      <th>target</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>64</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>164</td>\n",
       "      <td>212</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>133</td>\n",
       "      <td>0</td>\n",
       "      <td>0.8</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>41</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>130</td>\n",
       "      <td>256</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>148</td>\n",
       "      <td>0</td>\n",
       "      <td>3.2</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>59</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>165</td>\n",
       "      <td>284</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>160</td>\n",
       "      <td>0</td>\n",
       "      <td>3.4</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>61</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>125</td>\n",
       "      <td>254</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>147</td>\n",
       "      <td>0</td>\n",
       "      <td>4.4</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>56</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>110</td>\n",
       "      <td>205</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>184</td>\n",
       "      <td>0</td>\n",
       "      <td>2.5</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   age  sex  cp  trestbps  chol  fbs  restecg  thalach  exang  oldpeak  slope  \\\n",
       "0   64    1   0       164   212    1        0      133      0      0.8      1   \n",
       "1   41    1   2       130   256    0        1      148      0      3.2      0   \n",
       "2   59    1   0       165   284    0        1      160      0      3.4      1   \n",
       "3   61    1   2       125   254    0        1      147      0      4.4      0   \n",
       "4   56    1   0       110   205    0        1      184      0      2.5      0   \n",
       "\n",
       "   ca  thal  target  \n",
       "0   2     2       0  \n",
       "1   0     2       0  \n",
       "2   0     3       0  \n",
       "3   0     3       1  \n",
       "4   1     2       0  "
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Final example is how to change the validator model backend, the default choice\n",
    "# in the regular joining validator is a random forest classifier, but in the\n",
    "# example we use the SVM classifier model from sklearn\n",
    "from sklearn.svm import SVC\n",
    "\n",
    "JS = UsingJoiningValidator(\n",
    "    JoiningValidator(\n",
    "        classifier_model_base=SVC(kernel='linear', \n",
    "                                  probability=True, \n",
    "                                  class_weight='balanced'),\n",
    "        verbose=False)\n",
    "        ) \n",
    "dgms1 = DisjointGenerativeModels(df_train, gms, joining_strategy=JS)\n",
    "\n",
    "df_syn3 = dgms1.fit_generate()\n",
    "\n",
    "df_syn3.head()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We can compare the three generated datasets on a selection of metrics, using the [SynthEval Library](https://github.com/schneiderkamplab/syntheval/tree/main)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Inferred categorical columns (unique threshold: 10):\n",
      "['sex', 'cp', 'fbs', 'restecg', 'exang', 'slope', 'ca', 'thal', 'target']\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead tr th {\n",
       "        text-align: left;\n",
       "    }\n",
       "\n",
       "    .dataframe thead tr:last-of-type th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th colspan=\"2\" halign=\"left\">avg_h_dist</th>\n",
       "      <th colspan=\"2\" halign=\"left\">corr_mat_diff</th>\n",
       "      <th colspan=\"2\" halign=\"left\">auroc</th>\n",
       "      <th colspan=\"2\" halign=\"left\">avg_F1_diff</th>\n",
       "      <th colspan=\"2\" halign=\"left\">avg_F1_diff_hout</th>\n",
       "      <th>...</th>\n",
       "      <th colspan=\"2\" halign=\"left\">median_DCR</th>\n",
       "      <th colspan=\"2\" halign=\"left\">mia_recall</th>\n",
       "      <th colspan=\"2\" halign=\"left\">mia_precision</th>\n",
       "      <th>rank</th>\n",
       "      <th>u_rank</th>\n",
       "      <th>p_rank</th>\n",
       "      <th>f_rank</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th>value</th>\n",
       "      <th>error</th>\n",
       "      <th>value</th>\n",
       "      <th>error</th>\n",
       "      <th>value</th>\n",
       "      <th>error</th>\n",
       "      <th>value</th>\n",
       "      <th>error</th>\n",
       "      <th>value</th>\n",
       "      <th>error</th>\n",
       "      <th>...</th>\n",
       "      <th>value</th>\n",
       "      <th>error</th>\n",
       "      <th>value</th>\n",
       "      <th>error</th>\n",
       "      <th>value</th>\n",
       "      <th>error</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>dataset</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>cls_rf</th>\n",
       "      <td>0.053668</td>\n",
       "      <td>0.01815</td>\n",
       "      <td>1.269459</td>\n",
       "      <td>NaN</td>\n",
       "      <td>-0.009459</td>\n",
       "      <td>NaN</td>\n",
       "      <td>-0.09931</td>\n",
       "      <td>0.018206</td>\n",
       "      <td>-0.105849</td>\n",
       "      <td>0.012908</td>\n",
       "      <td>...</td>\n",
       "      <td>1.011259</td>\n",
       "      <td>NaN</td>\n",
       "      <td>0.475</td>\n",
       "      <td>0.067315</td>\n",
       "      <td>0.538846</td>\n",
       "      <td>0.050975</td>\n",
       "      <td>7.843108</td>\n",
       "      <td>4.480757</td>\n",
       "      <td>3.362351</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>cls_svm</th>\n",
       "      <td>0.088781</td>\n",
       "      <td>0.026512</td>\n",
       "      <td>1.276341</td>\n",
       "      <td>NaN</td>\n",
       "      <td>-0.369341</td>\n",
       "      <td>NaN</td>\n",
       "      <td>-0.463555</td>\n",
       "      <td>0.020505</td>\n",
       "      <td>-0.462908</td>\n",
       "      <td>0.009563</td>\n",
       "      <td>...</td>\n",
       "      <td>1.373413</td>\n",
       "      <td>NaN</td>\n",
       "      <td>0.3</td>\n",
       "      <td>0.063738</td>\n",
       "      <td>0.445501</td>\n",
       "      <td>0.057482</td>\n",
       "      <td>7.171934</td>\n",
       "      <td>3.226258</td>\n",
       "      <td>3.945677</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>occls_svm</th>\n",
       "      <td>0.049865</td>\n",
       "      <td>0.020996</td>\n",
       "      <td>1.263175</td>\n",
       "      <td>NaN</td>\n",
       "      <td>-0.281718</td>\n",
       "      <td>NaN</td>\n",
       "      <td>-0.371713</td>\n",
       "      <td>0.025893</td>\n",
       "      <td>-0.251984</td>\n",
       "      <td>0.017181</td>\n",
       "      <td>...</td>\n",
       "      <td>1.060856</td>\n",
       "      <td>NaN</td>\n",
       "      <td>0.3625</td>\n",
       "      <td>0.060596</td>\n",
       "      <td>0.506439</td>\n",
       "      <td>0.05303</td>\n",
       "      <td>7.336442</td>\n",
       "      <td>3.72335</td>\n",
       "      <td>3.613092</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>3 rows × 24 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "          avg_h_dist           corr_mat_diff           auroc        \\\n",
       "               value     error         value error     value error   \n",
       "dataset                                                              \n",
       "cls_rf      0.053668   0.01815      1.269459   NaN -0.009459   NaN   \n",
       "cls_svm     0.088781  0.026512      1.276341   NaN -0.369341   NaN   \n",
       "occls_svm   0.049865  0.020996      1.263175   NaN -0.281718   NaN   \n",
       "\n",
       "          avg_F1_diff           avg_F1_diff_hout            ... median_DCR  \\\n",
       "                value     error            value     error  ...      value   \n",
       "dataset                                                     ...              \n",
       "cls_rf       -0.09931  0.018206        -0.105849  0.012908  ...   1.011259   \n",
       "cls_svm     -0.463555  0.020505        -0.462908  0.009563  ...   1.373413   \n",
       "occls_svm   -0.371713  0.025893        -0.251984  0.017181  ...   1.060856   \n",
       "\n",
       "                mia_recall           mia_precision                rank  \\\n",
       "          error      value     error         value     error             \n",
       "dataset                                                                  \n",
       "cls_rf      NaN      0.475  0.067315      0.538846  0.050975  7.843108   \n",
       "cls_svm     NaN        0.3  0.063738      0.445501  0.057482  7.171934   \n",
       "occls_svm   NaN     0.3625  0.060596      0.506439   0.05303  7.336442   \n",
       "\n",
       "             u_rank    p_rank f_rank  \n",
       "                                      \n",
       "dataset                               \n",
       "cls_rf     4.480757  3.362351    0.0  \n",
       "cls_svm    3.226258  3.945677    0.0  \n",
       "occls_svm   3.72335  3.613092    0.0  \n",
       "\n",
       "[3 rows x 24 columns]"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from syntheval import SynthEval\n",
    "\n",
    "### Metrics\n",
    "metrics = {\n",
    "    \"h_dist\"    : {},\n",
    "    \"corr_diff\" : {\"mixed_corr\": True},\n",
    "    \"auroc_diff\" : {\"model\": \"rf_cls\"},\n",
    "    \"cls_acc\"   : {\"F1_type\": \"macro\"},\n",
    "    \"eps_risk\"  : {},\n",
    "    \"dcr\"       : {},\n",
    "    \"mia\"  : {\"num_eval_iter\": 5},\n",
    "}\n",
    "\n",
    "df_train = pd.read_csv('experiments/datasets/heart_train.csv')\n",
    "df_test = pd.read_csv('experiments/datasets/heart_test.csv')\n",
    "\n",
    "SE = SynthEval(df_train, df_test)\n",
    "res, _ = SE.benchmark({'cls_rf': df_syn1, \n",
    "                       'cls_svm': df_syn3,\n",
    "                       'occls_svm': df_syn2,}, \n",
    "                       analysis_target_var=\"target\",\n",
    "                       rank_strategy='summation', \n",
    "                       **metrics)\n",
    "\n",
    "res"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "According to this presentation, the dataset generated by the default model is slightly better on utility metrics, while theThe SVM classifier model is marginally better on privacy, and the one-class SVM model here gives a more balanced result.\n",
    "\n",
    "For more details on the different validator backends see the experiment codebook:\n",
    "\n",
    "- [04_joining_validator.ipynb](04_joining_validator.ipynb)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Example 3: Validator Optimisation and Calibration\n",
    "Clearly, in the DGMs framework, a lot depends on the validator model and its efficacy. In this part we show a basic example of how to optimise the validator model and calibrate its parameters. We use the same dataset and generative models as before, but we focus on the joining model backend. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Imports\n",
    "import pandas as pd\n",
    "\n",
    "df_train = pd.read_csv('experiments/datasets/heart_train.csv')\n",
    "df_test = pd.read_csv('experiments/datasets/heart_test.csv')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The `JoiningValidator` objects has a convinient `model_parameter_grid` argument that allows a dictionary of parameters to be passed for optimisation. The library will then do a grid search over the parameters and select the best one according to the validation scores."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DGMs (datamanager): The exterior correlations are 1.05 times that of the interiors.\n",
      "DGMs (validator): Grid search for hyperparameters\n",
      "DGMs (validator): Best score 0.690083 using {'max_depth': 10, 'max_features': 'sqrt', 'min_samples_leaf': 2, 'min_samples_split': 4, 'n_estimators': 15}\n",
      "DGMs (validator): Calibration improved the model from 0.9229 to 0.9236\n",
      "DGMs (validator): Threshold auto-set to: 0.7983137885730685\n",
      "- Predicted good joins fraction: 0.10055096418732783\n",
      "- Predicted good joins fraction: 0.11944869831546708\n",
      "- Predicted good joins fraction: 0.06782608695652174\n",
      "- Predicted good joins fraction: 0.06902985074626866\n",
      "- Predicted good joins fraction: 0.06012024048096192\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>age</th>\n",
       "      <th>sex</th>\n",
       "      <th>cp</th>\n",
       "      <th>trestbps</th>\n",
       "      <th>chol</th>\n",
       "      <th>fbs</th>\n",
       "      <th>restecg</th>\n",
       "      <th>thalach</th>\n",
       "      <th>exang</th>\n",
       "      <th>oldpeak</th>\n",
       "      <th>slope</th>\n",
       "      <th>ca</th>\n",
       "      <th>thal</th>\n",
       "      <th>target</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>56</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>145</td>\n",
       "      <td>213</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>195</td>\n",
       "      <td>0</td>\n",
       "      <td>0.2</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>59</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>130</td>\n",
       "      <td>257</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>144</td>\n",
       "      <td>1</td>\n",
       "      <td>1.8</td>\n",
       "      <td>0</td>\n",
       "      <td>4</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>70</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>140</td>\n",
       "      <td>224</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>141</td>\n",
       "      <td>0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>3</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>46</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>120</td>\n",
       "      <td>206</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>157</td>\n",
       "      <td>0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>58</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>200</td>\n",
       "      <td>227</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>95</td>\n",
       "      <td>1</td>\n",
       "      <td>3.6</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>3</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   age  sex  cp  trestbps  chol  fbs  restecg  thalach  exang  oldpeak  slope  \\\n",
       "0   56    0   3       145   213    1        0      195      0      0.2      1   \n",
       "1   59    1   0       130   257    0        0      144      1      1.8      0   \n",
       "2   70    0   1       140   224    0        2      141      0      4.0      0   \n",
       "3   46    1   2       120   206    0        0      157      0      0.0      2   \n",
       "4   58    1   1       200   227    0        0       95      1      3.6      1   \n",
       "\n",
       "   ca  thal  target  \n",
       "0   1     3       0  \n",
       "1   4     3       0  \n",
       "2   2     3       1  \n",
       "3   0     2       0  \n",
       "4   1     3       0  "
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 600x600 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from disjoint_generation import DisjointGenerativeModels\n",
    "from disjoint_generation.utils.joining_validator import JoiningValidator\n",
    "from disjoint_generation.utils.joining_strategies import UsingJoiningValidator\n",
    "\n",
    "from sklearn.ensemble import RandomForestClassifier\n",
    "\n",
    "validator_model = RandomForestClassifier()\n",
    "parameter_grid = {\n",
    "    'n_estimators': [5, 10, 15], 'max_depth': [5, 10],\n",
    "    'min_samples_split': [2, 3, 4], 'min_samples_leaf': [1, 2, 3], 'max_features': ['sqrt', 'log2']\n",
    "}\n",
    "calibration_method = 'sigmoid'\n",
    "\n",
    "JV = JoiningValidator(validator_model, model_parameter_grid = parameter_grid, calibration_method = 'sigmoid', verbose=True)\n",
    "JS = UsingJoiningValidator(JV)\n",
    "dgms_opt = DisjointGenerativeModels(df_train, {'synthpop': 1, 'privbayes': 1}, joining_strategy=JS)\n",
    "\n",
    "df_opt = dgms_opt.fit_generate()\n",
    "\n",
    "dgms_opt._make_calibration_plot(df_test, save = False)    # Special helper function which produces a plot of the calibration.\n",
    "\n",
    "df_opt.head()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "If the `fit_generate` method takes long to run and propper optimisation needs to be ensured before running the whole process, the `dgms_opt._setup()` helper method can be used to run the optimisation process without fitting the generative models.\n",
    "\n",
    "We can also save and load the optimised model for later use"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Save model\n",
    "import pickle\n",
    "\n",
    "with open('hd_rf_scikit.obj', 'wb') as file:\n",
    "    pickle.dump(dgms_opt._strategy.join_validator, file)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load model\n",
    "\n",
    "with open('hd_rf_scikit.obj', 'rb') as file:\n",
    "    joining_validator = pickle.load(file)\n",
    "\n",
    "JS = UsingJoiningValidator(joining_validator)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "For more details on the parameter tuning and calibration see the experiment codebooks:\n",
    "\n",
    "- [03_specified_splits.ipynb](03_specified_splits.ipynb)\n",
    "- [04_joining_validator.ipynb](04_joining_validator.ipynb)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".venv",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
