{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "4886f3bf-c733-4976-bb2a-fc0fee472b86",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import xgboost as xgb\n",
    "import sklearn\n",
    "from sklearn.ensemble import RandomForestClassifier\n",
    "import sklearn.model_selection as ms\n",
    "import numpy as np\n",
    "import scipy\n",
    "\n",
    "import torch\n",
    "from nnlib.nnlib import utils\n",
    "\n",
    "import warnings\n",
    "from tqdm.auto import tqdm\n",
    "\n",
    "import h5py\n",
    "import gpflow"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "47bdbfdd-3e99-4560-a93f-29ca06a6f5a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "class CalibrationMethod(sklearn.base.BaseEstimator):\n",
    "    \"\"\"\n",
    "    A generic class for probability calibration\n",
    "\n",
    "    A calibration method takes a set of posterior class probabilities and transform them into calibrated posterior\n",
    "    probabilities. Calibrated in this sense means that the empirical frequency of a correct class prediction matches its\n",
    "    predicted posterior probability.\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "\n",
    "    def fit(self, X, y):\n",
    "        \"\"\"\n",
    "        Fit the calibration method based on the given uncalibrated class probabilities X and ground truth labels y.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        X : array-like, shape (n_samples, n_classes)\n",
    "            Training data, i.e. predicted probabilities of the base classifier on the calibration set.\n",
    "        y : array-like, shape (n_samples,)\n",
    "            Target classes.\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        self : object\n",
    "            Returns an instance of self.\n",
    "        \"\"\"\n",
    "        raise NotImplementedError(\"Subclass must implement this method.\")\n",
    "\n",
    "    def predict_proba(self, X):\n",
    "        \"\"\"\n",
    "        Compute calibrated posterior probabilities for a given array of posterior probabilities from an arbitrary\n",
    "        classifier.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        X : array-like, shape (n_samples, n_classes)\n",
    "            The uncalibrated posterior probabilities.\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        P : array, shape (n_samples, n_classes)\n",
    "            The predicted probabilities.\n",
    "        \"\"\"\n",
    "        raise NotImplementedError(\"Subclass must implement this method.\")\n",
    "\n",
    "    def predict(self, X):\n",
    "        \"\"\"\n",
    "        Predict the class of new samples after scaling. Predictions are identical to the ones from the uncalibrated\n",
    "        classifier.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        X : array-like, shape (n_samples, n_classes)\n",
    "            The uncalibrated posterior probabilities.\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        C : array, shape (n_samples,)\n",
    "            The predicted classes.\n",
    "        \"\"\"\n",
    "        return np.argmax(self.predict_proba(X), axis=1)\n",
    "\n",
    "    def plot(self, filename, xlim=[0, 1], **kwargs):\n",
    "        \"\"\"\n",
    "        Plot the calibration map.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        xlim : array-like\n",
    "            Range of inputs of the calibration map to be plotted.\n",
    "\n",
    "        **kwargs :\n",
    "            Additional arguments passed on to :func:`matplotlib.plot`.\n",
    "        \"\"\"\n",
    "        # TODO: Fix this plotting function\n",
    "\n",
    "        # Generate data and transform\n",
    "        x = np.linspace(0, 1, 10000)\n",
    "        y = self.predict_proba(np.column_stack([1 - x, x]))[:, 1]\n",
    "\n",
    "        # Plot and label\n",
    "        plt.plot(x, y, **kwargs)\n",
    "        plt.xlim(xlim)\n",
    "        plt.xlabel(\"p(y=1|x)\")\n",
    "        plt.ylabel(\"f(p(y=1|x))\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "c0f3b7e3-c80f-4bb8-be0d-e5103d393a01",
   "metadata": {},
   "outputs": [],
   "source": [
    "from gpflow.mean_functions import MeanFunction\n",
    "\n",
    "class Log(MeanFunction):\n",
    "    \"\"\"\n",
    "    Natural logarithm prior mean function.\n",
    "\n",
    "    :math:`y_i = \\log(x_i)`\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self):\n",
    "        MeanFunction.__init__(self)\n",
    "\n",
    "    def __call__(self, X):\n",
    "        # Avoid -inf = log(0)\n",
    "        tiny = np.finfo(np.float).tiny\n",
    "        X = tf.clip_by_value(X, clip_value_min=tiny, clip_value_max=np.inf)\n",
    "        # Returns the natural logarithm of the input\n",
    "        return tf.log(X)\n",
    "\n",
    "\n",
    "class ScalarMult(MeanFunction):\n",
    "    \"\"\"\n",
    "    Scalar multiplication mean function.\n",
    "\n",
    "    :math:`y_i = \\\\alpha x_i`\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self, alpha=1):\n",
    "        MeanFunction.__init__(self)\n",
    "        self.alpha = Parameter(alpha, dtype=gpflow.config.default_float())\n",
    "\n",
    "    def __call__(self, X):\n",
    "        # Scalar multiplication\n",
    "        return tf.multiply(self.alpha, X)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e19cfab4-ecf7-43e9-9d23-8134a5f02cf8",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "389ac9e8-6401-41cb-adf7-b98c6e5e3f8a",
   "metadata": {},
   "outputs": [],
   "source": [
    "class SoftArgMax(gpflow.Module):\n",
    "    \"\"\"\n",
    "    This class implements the multi-class softargmax inverse-link function. Given a vector :math:`f=[f_1, f_2, ... f_k]`,\n",
    "    then result of the mapping is :math:`y = [y_1 ... y_k]`, where\n",
    "    :math:`y_i = \\\\frac{\\\\exp(f_i)}{\\\\sum_{j=1}^k\\\\exp(f_j)}`.\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self, num_classes, **kwargs):\n",
    "        super().__init__(**kwargs)\n",
    "        self.num_classes = num_classes\n",
    "\n",
    "    def __call__(self, F):\n",
    "        return tf.nn.softmax(F)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "61e295aa-5d7c-4846-b527-e559eafeacf6",
   "metadata": {},
   "outputs": [],
   "source": [
    "class MultiCal(gpflow.likelihoods.Likelihood):\n",
    "    def __init__(self, num_classes, input_dim, latent_dim, observation_dim, invlink=None, num_monte_carlo_points=100, **kwargs):\n",
    "        \"\"\"\n",
    "        A likelihood that performs multiclass calibration using the softargmax link function and a single latent\n",
    "        process.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        num_classes : int\n",
    "            Number of classes.\n",
    "        invlink : default=None\n",
    "            Inverse link function :math:`p(y \\mid f)`.\n",
    "        num_monte_carlo_points : int, default=100\n",
    "            Number of Monte-Carlo points for prediction, i.e. for the integral :math:`\\int p(y=Y|f)q(f) df`, where\n",
    "            :math:`q(f)` is a Gaussian.\n",
    "        kwargs\n",
    "        \"\"\"\n",
    "        super().__init__(input_dim=input_dim, latent_dim=latent_dim, observation_dim=observation_dim, **kwargs)\n",
    "        self.num_classes = num_classes\n",
    "        if invlink is None:\n",
    "            invlink = SoftArgMax(self.num_classes)\n",
    "        elif not isinstance(invlink, SoftArgMax):\n",
    "            raise NotImplementedError\n",
    "        self.invlink = invlink\n",
    "        self.num_monte_carlo_points = num_monte_carlo_points\n",
    "\n",
    "    def _log_prob(self, F, Y):\n",
    "        \"\"\"\n",
    "        Computes the log softargmax at indices from Y.\n",
    "\n",
    "        :math:`\\\\sigma(F)_y = \\\\frac{exp(F_y)}{\\\\sum_{k=1}^K \\\\exp(F_k)}`\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        F : tf.tensor, shape=(N, K)\n",
    "            Inputs to softargmax.\n",
    "        Y : tf.tensor, shape=(N, 1)\n",
    "            Indices of softargmax output.\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        log_sigma_y : tf.tensor, shape=()\n",
    "            log softargmax at y\n",
    "\n",
    "        \"\"\"\n",
    "        if isinstance(self.invlink, SoftArgMax):\n",
    "            with tf.control_dependencies(\n",
    "                    [\n",
    "                        tf.assert_equal(tf.shape(Y)[1], 1),\n",
    "                        tf.assert_equal(tf.cast(tf.shape(F)[1], settings.int_type),\n",
    "                                        tf.cast(self.num_classes, settings.int_type))\n",
    "                    ]):\n",
    "                return -tf.nn.sparse_softmax_cross_entropy_with_logits(logits=F, labels=Y[:, 0])[:, None]\n",
    "        else:\n",
    "            raise NotImplementedError\n",
    "\n",
    "    def _variational_expectations(self, Fmus, Fvars, Y, full_cov=False):\n",
    "        \"\"\"\n",
    "        Computes an approximation to the expectation terms in the variational objective.\n",
    "\n",
    "        This function approximates the :math:`n` expectation terms :math:`\\\\mathbb{E}_{q(f_n)}[\\\\log p(y_n \\\\mid f_n)]`\n",
    "        in the variational objective function.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        Fmus : tf.tensor, shape=(N, K)\n",
    "            Means of the latent GP at input locations X. Dimension N x K.\n",
    "        Fvars : tf.tensor, shape=(N, K(, K))\n",
    "            Variances of the latent GP at input locations X. Dimension N x K (x K).\n",
    "        Y : tf.tensor, shape=(N,)\n",
    "            Output vector.\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        ve : tf.tensor, shape=(N,)\n",
    "            The variational expectation assuming a Gaussian approximation q.\n",
    "        \"\"\"\n",
    "        if isinstance(self.invlink, SoftArgMax):\n",
    "            # Compute variational expectations by 2nd order Taylor approximation\n",
    "            sigma_mu = tf.nn.softmax(Fmus, axis=1)\n",
    "\n",
    "            if full_cov:\n",
    "                sigSsig = tf.einsum(\"nk, nkl, nl -> n\", sigma_mu, Fvars, sigma_mu)\n",
    "            else:\n",
    "                sigSsig = tf.reduce_sum(tf.multiply(tf.multiply(sigma_mu, Fvars), sigma_mu), axis=1)\n",
    "\n",
    "            diagSsig = tf.reduce_sum(tf.multiply(sigma_mu, Fvars), axis=1)\n",
    "            logsoftargmax_y = tf.squeeze(self.logp(Fmus, Y))\n",
    "\n",
    "            # Objective function\n",
    "            return logsoftargmax_y + 0.5 * (sigSsig - diagSsig)\n",
    "\n",
    "        else:\n",
    "            raise NotImplementedError\n",
    "\n",
    "    def _predict_mean_and_var(self, Fmus, Fvars):\n",
    "        \"\"\"\n",
    "        Given a Normal distribution for the latent function, return the mean of :math:`Y`, if\n",
    "        :math:`q(f) = N(Fmu, Fvar)` and this object represents :math:`p(y|f)`, then this method computes the predictive\n",
    "        mean :math:`\\\\int\\\\int y p(y|f)q(f) df dy` and the predictive variance\n",
    "        :math:`\\\\int\\\\int y^2 p(y|f)q(f) df dy  - [ \\\\int\\\\int y p(y|f)q(f) df dy ]^2`.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        Fmus : array/tensor, shape=(N, D)\n",
    "            Mean(s) of Gaussian density.\n",
    "        Fvars : array/tensor, shape=(N, D(, D))\n",
    "            Covariance(s) of Gaussian density.\n",
    "        \"\"\"\n",
    "        raise NotImplementedError\n",
    "\n",
    "    def _predict_log_density(self, Fmus, Fvars, Y):\n",
    "        \"\"\"\n",
    "        Given a Normal distribution for the latent function, and a datum Y, compute the log predictive density of Y.\n",
    "        i.e. if :math:`p(f_* | y) = \\\\mathcal{N}(Fmu, Fvar)` and :math:`p(y_*|f_*)` is the likelihood, then this\n",
    "        method computes the log predictive density :math:`\\\\log \\\\int p(y_*|f)p(f_* | y) df`. Here, we implement a\n",
    "        Monte-Carlo routine.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        Fmus : array/tensor, shape=(N, K)\n",
    "            Mean(s) of Gaussian density.\n",
    "        Fvars : array/tensor, shape=(N, K(, K))\n",
    "            Covariance(s) of Gaussian density.\n",
    "        Y : arrays/tensors, shape=(N(, K))\n",
    "            Deterministic arguments to be passed by name to funcs.\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        log_density : array/tensor, shape=(N(, K))\n",
    "            Log predictive density.\n",
    "        \"\"\"\n",
    "        if isinstance(self.invlink, SoftArgMax):\n",
    "            return ndiag_mc(self.logp, self.num_monte_carlo_points, Fmus, Fvars,\n",
    "                            logspace=True, epsilon=None, Y=Y)\n",
    "        else:\n",
    "            raise NotImplementedError\n",
    "\n",
    "    def _predict_full_density(self, Fmus, Fvars):\n",
    "        if isinstance(self.invlink, SoftArgMax):\n",
    "            # Sample from standard normal\n",
    "            N = tf.shape(Fmus)[0]\n",
    "            epsilon = tf.random_normal((self.num_monte_carlo_points, N, self.num_classes),\n",
    "                                       dtype=gpflow.config.default_float())\n",
    "\n",
    "            # Transform to correct mean and covariance\n",
    "            f_star = Fmus[None, :, :] + tf.sqrt(Fvars[None, :, :]) * epsilon  # S x N x K\n",
    "\n",
    "            # Compute Softmax\n",
    "            p_y_f_star = tf.nn.softmax(f_star, axis=2)\n",
    "\n",
    "            # Average to obtain log Monte-Carlo estimate\n",
    "            return tf.log(tf.reduce_mean(p_y_f_star, axis=0))\n",
    "        else:\n",
    "            raise NotImplementedError"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "9c13fa4f-573f-4331-9be4-8d3375bdcb42",
   "metadata": {},
   "outputs": [],
   "source": [
    "import tensorflow as tf\n",
    "import gpflow\n",
    "\n",
    "import tensorflow_probability as tfp\n",
    "import tensorflow as tf\n",
    "\n",
    "class BoundedPositiveBijector(tfp.bijectors.Bijector):\n",
    "    def __init__(self, lower, upper, validate_args=False, name=\"bounded_positive\"):\n",
    "        super().__init__(forward_min_event_ndims=0, validate_args=validate_args, name=name)\n",
    "        self.lower = tf.cast(lower, tf.float64)\n",
    "        self.upper = tf.cast(upper, tf.float64)\n",
    "\n",
    "    def _forward(self, x):\n",
    "        # [0, 1]にスケーリングされた値\n",
    "        scaled_x = tf.sigmoid(x)\n",
    "        # [lower, upper]の範囲に変換\n",
    "        return self.lower + (self.upper - self.lower) * scaled_x\n",
    "\n",
    "    def _inverse(self, y):\n",
    "        # yを[0, 1]に変換\n",
    "        scaled_y = (y - self.lower) / (self.upper - self.lower)\n",
    "        # sigmoidの逆関数はlogit\n",
    "        return tf.math.log(scaled_y / (1 - scaled_y))\n",
    "\n",
    "    def _forward_log_det_jacobian(self, x):\n",
    "        return tf.constant(0., dtype=x.dtype)\n",
    "\n",
    "class GPCalibration(CalibrationMethod):\n",
    "    \"\"\"\n",
    "    Probability calibration using a latent Gaussian process\n",
    "\n",
    "    Gaussian process calibration [1]_ is a non-parametric approach to calibrate posterior probabilities from an arbitrary\n",
    "    classifier based on a hold-out data set. Inference is performed using a sparse variational Gaussian process\n",
    "    (SVGP) [2]_ implemented in `gpflow` [3]_.\n",
    "\n",
    "    Parameters\n",
    "    ----------\n",
    "    n_classes : int\n",
    "        Number of classes in calibration data.\n",
    "    logits : bool, default=False\n",
    "        Are the inputs for calibration logits (e.g. from a neural network)?\n",
    "    mean_function : GPflow object\n",
    "        Mean function of the latent GP.\n",
    "    kernel : GPflow object\n",
    "        Kernel function of the latent GP.\n",
    "    likelihood : GPflow object\n",
    "        Likelihood giving a prior on the class prediction.\n",
    "    n_inducing_points : int, default=100\n",
    "        Number of inducing points for the variational approximation.\n",
    "    maxiter : int, default=1000\n",
    "        Maximum number of iterations for the likelihood optimization procedure.\n",
    "    n_monte_carlo : int, default=100\n",
    "        Number of Monte Carlo samples for the inference procedure.\n",
    "    max_samples_monte_carlo : int, default=10**7\n",
    "        Maximum number of Monte Carlo samples to draw in one batch when predicting. Setting this value too large can\n",
    "        cause memory issues.\n",
    "    inf_mean_approx : bool, default=False\n",
    "        If True, when inferring calibrated probabilities, only the mean of the latent Gaussian process is taken into\n",
    "        account, not its covariance.\n",
    "    session : tf.Session, default=None\n",
    "        `tensorflow` session to use.\n",
    "    random_state : int, default=0\n",
    "        Random seed for reproducibility. Needed for Monte-Carlo sampling routine.\n",
    "    verbose : bool\n",
    "        Print information on optimization routine.\n",
    "\n",
    "    References\n",
    "    ----------\n",
    "    .. [1] Wenger, J., Kjellström H. & Triebel, R. Non-Parametric Calibration for Classification in\n",
    "           Proceedings of AISTATS (2020)\n",
    "    .. [2] Hensman, J., Matthews, A. G. d. G. & Ghahramani, Z. Scalable Variational Gaussian Process Classification in\n",
    "           Proceedings of AISTATS (2015)\n",
    "    .. [3] Matthews, A. G. d. G., van der Wilk, M., et al. GPflow: A Gaussian process library using TensorFlow. Journal\n",
    "           of Machine Learning Research 18, 1–6 (Apr. 2017)\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self,\n",
    "                 n_classes,\n",
    "                 logits=False,\n",
    "                 mean_function=None,\n",
    "                 kernel=None,\n",
    "                 likelihood=None,\n",
    "                 n_inducing_points=10,\n",
    "                 maxiter=1000,\n",
    "                 n_monte_carlo=100,\n",
    "                 max_samples_monte_carlo=10 ** 7,\n",
    "                 inf_mean_approx=False,\n",
    "                 session=None,\n",
    "                 random_state=1,\n",
    "                 verbose=False):\n",
    "        super().__init__()\n",
    "\n",
    "        # Parameter initialization\n",
    "        self.n_classes = n_classes\n",
    "        self.verbose = verbose\n",
    "        self.logits = logits\n",
    "\n",
    "        # Initialization of tensorflow session\n",
    "        self.session = session\n",
    "\n",
    "        # Initialization of Gaussian process components and inference parameters\n",
    "        self.input_dim = n_classes\n",
    "        self.n_monte_carlo = n_monte_carlo\n",
    "        self.max_samples_monte_carlo = max_samples_monte_carlo\n",
    "        self.n_inducing_points = n_inducing_points\n",
    "        self.maxiter = maxiter\n",
    "        self.inf_mean_approx = inf_mean_approx\n",
    "        self.random_state = random_state\n",
    "        np.random.seed(self.random_state)  # Set seed for optimization of hyperparameters\n",
    "\n",
    "        # Set likelihood\n",
    "        if likelihood is None:\n",
    "            self.likelihood = MultiCal(num_classes=self.n_classes, input_dim=self.input_dim, latent_dim=1, observation_dim=1,\n",
    "                                                      num_monte_carlo_points=self.n_monte_carlo)\n",
    "        else:\n",
    "            self.likelihood = likelihood\n",
    "\n",
    "        # Set mean function\n",
    "        if mean_function is None:\n",
    "            if logits:\n",
    "                self.mean_function = gpflow.conditionals.mean_functions.Identity()\n",
    "            else:\n",
    "                #self.mean_function = gp_classes.Log()\n",
    "                self.mean_function = Log()\n",
    "        else:\n",
    "            self.mean_function = mean_function\n",
    "\n",
    "        # Set kernel\n",
    "        if kernel is None:\n",
    "            k_white = gpflow.kernels.White(variance=0.01, active_dims=1)\n",
    "            if logits:\n",
    "                kernel_lengthscale = 10\n",
    "                k_rbf = gpflow.kernels.RBF(lengthscales=kernel_lengthscale, variance=1)\n",
    "            else:\n",
    "                kernel_lengthscale = 0.5\n",
    "                k_rbf = gpflow.kernels.RBF(lengthscales=kernel_lengthscale, variance=1)\n",
    "                \n",
    "                # Place constraints [a,b] on kernel parameters\n",
    "                transform_lengthscale = BoundedPositiveBijector(.001, 10)\n",
    "                transform_variance = BoundedPositiveBijector(0.01, 5)\n",
    "                \n",
    "                k_rbf.lengthscales = gpflow.Parameter(kernel_lengthscale, transform=transform_lengthscale)\n",
    "                k_rbf.variance = gpflow.Parameter(1, transform=transform_variance)\n",
    "                \n",
    "                #k_rbf.lengthscales = gpflow.Parameter(kernel_lengthscale, transform=bounded_positive(0.001, 10))\n",
    "                #k_rbf.variance = gpflow.Parameter(1, transform=bounded_positive(0, 5))\n",
    "                \n",
    "                #k_rbf.lengthscales = gpflow.Parameter(kernel_lengthscale, transform=gpflow.transforms.Logistic(low=.001, high=10))\n",
    "                #k_rbf.lengthscales.transform = gpflow.transforms.Logistic(a=.001, b=10)\n",
    "                #k_rbf.variance.transform = gpflow.transforms.Logistic(a=0, b=5)\n",
    "                self.kernel = k_rbf + k_white\n",
    "        else:\n",
    "            self.kernel = kernel\n",
    "\n",
    "    def fit(self, X, y):\n",
    "        \"\"\"\n",
    "        Fit the calibration method based on the given uncalibrated class probabilities or logits X and ground truth\n",
    "        labels y.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        X : array-like, shape (n_samples, n_classes)\n",
    "            Training data, i.e. predicted probabilities or logits of the base classifier on the calibration set.\n",
    "        y : array-like, shape (n_samples,)\n",
    "            Target classes.\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        self : object\n",
    "            Returns an instance of self.\n",
    "        \"\"\"\n",
    "        # Check for correct dimensions\n",
    "        if X.ndim == 1 or np.shape(X)[1] != self.n_classes:\n",
    "            raise ValueError(\"Calibration data must have shape (n_samples, n_classes).\")\n",
    "\n",
    "        # Create a new TF session if none is given\n",
    "        #if self.session is None:\n",
    "        #    self.session = tf.Session(graph=tf.Graph())\n",
    "\n",
    "        # Fit GP in TF session\n",
    "        #with self.session.as_default(), self.session.graph.as_default():\n",
    "        #    if X.ndim == 1:\n",
    "        #        raise ValueError(\"Calibration training data must have shape (n_samples, n_classes).\")\n",
    "        #    else:\n",
    "        #        self._fit_multiclass(X, y)\n",
    "        if X.ndim == 1:\n",
    "            raise ValueError(\"Calibration training data must have shape (n_samples, n_classes).\")\n",
    "        else:\n",
    "            self._fit_multiclass(X, y)\n",
    "        \n",
    "        return self\n",
    "\n",
    "    def _fit_multiclass(self, X, y):\n",
    "        # Setup\n",
    "        y = y.reshape(-1, 1)\n",
    "\n",
    "        # Select inducing points through scipy.cluster.vq.kmeans\n",
    "        Z = scipy.cluster.vq.kmeans(obs=X.flatten().reshape(-1, 1),\n",
    "                                    k_or_guess=min(X.shape[0] * X.shape[1], self.n_inducing_points, ))[0]\n",
    "\n",
    "        # Define SVGP calibration model with multiclass softargmax calibration likelihood\n",
    "        self.model = SVGPcal(X=X, Y=y, Z=Z,\n",
    "                                        mean_function=self.mean_function,\n",
    "                                        kern=self.kernel,\n",
    "                                        likelihood=self.likelihood,\n",
    "                                        whiten=True,\n",
    "                                        q_diag=True)\n",
    "\n",
    "        # Optimize parameters\n",
    "        #opt = gpflow.train.ScipyOptimizer()\n",
    "        #opt.minimize(self.model, maxiter=self.maxiter, disp=self.verbose)\n",
    "        optimizer = gpflow.optimizers.Scipy()\n",
    "        optimizer.minimize(self.model._build_likelihood, variables=self.model.trainable_variables, \n",
    "                           options=dict(disp=True, maxiter=self.maxiter))\n",
    "        \n",
    "        #@tf.function\n",
    "        #def optimization_step():\n",
    "        #    with tf.GradientTape() as tape:\n",
    "        #        loss = self.model\n",
    "        #    grads = tape.gradient(loss, self.model.trainable_variables)\n",
    "        #    opt.apply_gradients(zip(grads, self.model.trainable_variables))\n",
    "        \n",
    "        #for _ in range(self.maxiter):\n",
    "        #    optimization_step()\n",
    "\n",
    "        return self\n",
    "\n",
    "    def predict_proba(self, X, mean_approx=False):\n",
    "        \"\"\"\n",
    "        Compute calibrated posterior probabilities for a given array of posterior probabilities from an arbitrary\n",
    "        classifier.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        X : array-like, shape=(n_samples, n_classes)\n",
    "            The uncalibrated posterior probabilities.\n",
    "        mean_approx : bool, default=False\n",
    "            If True, inference is performed using only the mean of the latent Gaussian process, not its covariance.\n",
    "            Note, if `self.inference_mean_approximation==True`, then the logical value of this option is not considered.\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        P : array, shape (n_samples, n_classes)\n",
    "            The predicted probabilities.\n",
    "        \"\"\"\n",
    "        check_is_fitted(self, \"model\")\n",
    "\n",
    "        if mean_approx or self.inf_mean_approx:\n",
    "\n",
    "            # Evaluate latent GP\n",
    "            with self.session.as_default(), self.session.graph.as_default():\n",
    "                f, _ = self.model.predict_f(X_onedim=X.reshape(-1, 1))\n",
    "                latent = f.eval().reshape(np.shape(X))\n",
    "\n",
    "            # Return softargmax of fitted GP at input\n",
    "            return scipy.special.softmax(latent, axis=1)\n",
    "        else:\n",
    "\n",
    "            with self.session.as_default(), self.session.graph.as_default():\n",
    "                # Seed for Monte_Carlo\n",
    "                tf.set_random_seed(self.random_state)\n",
    "\n",
    "                if X.ndim == 1 or np.shape(X)[1] != self.n_classes:\n",
    "                    raise ValueError(\"Calibration data must have shape (n_samples, n_classes).\")\n",
    "                else:\n",
    "                    # Predict in batches to keep memory usage in Monte-Carlo sampling low\n",
    "                    n_data = np.shape(X)[0]\n",
    "                    samples_monte_carlo = self.n_classes * self.n_monte_carlo * n_data\n",
    "                    if samples_monte_carlo >= self.max_samples_monte_carlo:\n",
    "                        n_pred_batches = np.divmod(samples_monte_carlo, self.max_samples_monte_carlo)[0]\n",
    "                    else:\n",
    "                        n_pred_batches = 1\n",
    "\n",
    "                    p_pred_list = []\n",
    "                    for i in range(n_pred_batches):\n",
    "                        if self.verbose:\n",
    "                            print(\"Predicting batch {}/{}.\".format(i + 1, n_pred_batches))\n",
    "                        ind_range = np.arange(start=self.max_samples_monte_carlo * i,\n",
    "                                              stop=np.minimum(self.max_samples_monte_carlo * (i + 1), n_data))\n",
    "                        p_pred_list.append(tf.exp(self.model.predict_full_density(Xnew=X[ind_range, :])).eval())\n",
    "\n",
    "                    return np.concatenate(p_pred_list, axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "4085ee06-ea8a-4857-8a1b-a1b3eb5ed594",
   "metadata": {},
   "outputs": [],
   "source": [
    "#from gpflow.decors import params_as_tensors\n",
    "#from gpflow import params_as_tensors\n",
    "from gpflow.models.model import GPModel\n",
    "from gpflow import Parameter\n",
    "from gpflow.utilities import positive\n",
    "from gpflow import kullback_leiblers\n",
    "from gpflow.conditionals import conditional\n",
    "\n",
    "class SVGPcal(gpflow.models.GPModel):\n",
    "    \"\"\"\n",
    "    Probability calibration using a sparse variational latent Gaussian process.\n",
    "\n",
    "    This is the Sparse Variational GP [1]_ calibration model. It has a single one-dimensional GP as a latent function\n",
    "    which is applied to all inputs individually.\n",
    "\n",
    "    .. [1] Hensman, J., Matthews, A. G. d. G. & Ghahramani, Z. Scalable Variational Gaussian Process Classification in\n",
    "           Proceedings of AISTATS (2015)\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self, X, Y, kern, likelihood, feat=None,\n",
    "                 mean_function=None,\n",
    "                 num_latent=None,\n",
    "                 q_diag=False,\n",
    "                 whiten=True,\n",
    "                 minibatch_size=None,\n",
    "                 Z=None,\n",
    "                 num_data=None,\n",
    "                 q_mu=None,\n",
    "                 q_sqrt=None,\n",
    "                 **kwargs):\n",
    "        \"\"\"\n",
    "        - X is a data matrix, size N x D\n",
    "        - Y is a data matrix, size N x P\n",
    "        - kern, likelihood, mean_function are appropriate GPflow objects\n",
    "        - Z is a matrix of pseudo inputs, size M x D\n",
    "        - num_latent is the number of latent process to use, defaults to one.\n",
    "        - q_diag is a boolean. If True, the covariance is approximated by a\n",
    "          diagonal matrix.\n",
    "        - whiten is a boolean. If True, we use the whitened representation of\n",
    "          the inducing points.\n",
    "        - minibatch_size, if not None, turns on mini-batching with that size.\n",
    "        - num_data is the total number of observations, default to X.shape[0]\n",
    "          (relevant when feeding in external minibatches)\n",
    "        \"\"\"\n",
    "        # sort out the X, Y into MiniBatch objects if required.\n",
    "        if minibatch_size is None:\n",
    "            #X = DataHolder(X)\n",
    "            #Y = DataHolder(Y)\n",
    "            self.X = tf.constant(X)\n",
    "            self.Y = tf.constant(Y)\n",
    "        else:\n",
    "            self.X = Minibatch(X, batch_size=minibatch_size, seed=0)\n",
    "            self.Y = Minibatch(Y, batch_size=minibatch_size, seed=0)\n",
    "\n",
    "        # init the super class, accept args\n",
    "        if num_latent is None:\n",
    "            #self.num_latent = 1\n",
    "            self.num_latent = self.Y.shape[1]\n",
    "        GPModel.__init__(self, kern, likelihood, mean_function, self.num_latent, **kwargs)\n",
    "        self.kern = kern\n",
    "        self.num_data = num_data or self.X.shape[0]\n",
    "        self.num_classes = self.X.shape[1]\n",
    "        self.q_diag, self.whiten = q_diag, whiten\n",
    "        self.feature = gpflow.inducing_variables.InducingPoints(Z)\n",
    "\n",
    "        # init variational parameters\n",
    "        num_inducing = len(self.feature)\n",
    "\n",
    "        self._init_variational_parameters(num_inducing, q_mu, q_sqrt, q_diag)\n",
    "\n",
    "    def _init_variational_parameters(self, num_inducing, q_mu, q_sqrt, q_diag):\n",
    "        \"\"\"\n",
    "        Constructs the mean and cholesky of the covariance of the variational Gaussian posterior.\n",
    "        If a user passes values for `q_mu` and `q_sqrt` the routine checks if they have consistent\n",
    "        and correct shapes. If a user does not specify any values for `q_mu` and `q_sqrt`, the routine\n",
    "        initializes them, their shape depends on `num_inducing` and `q_diag`.\n",
    "        Note: most often the comments refer to the number of observations (=output dimensions) with P,\n",
    "        number of latent GPs with L, and number of inducing points M. Typically P equals L,\n",
    "        but when certain multi-output kernels are used, this can change.\n",
    "        Parameters\n",
    "        ----------\n",
    "        :param num_inducing: int\n",
    "            Number of inducing variables, typically referred to as M.\n",
    "        :param q_mu: np.array or None\n",
    "            Mean of the variational Gaussian posterior. If None the function will initialise\n",
    "            the mean with zeros. If not None, the shape of `q_mu` is checked.\n",
    "        :param q_sqrt: np.array or None\n",
    "            Cholesky of the covariance of the variational Gaussian posterior.\n",
    "            If None the function will initialise `q_sqrt` with identity matrix.\n",
    "            If not None, the shape of `q_sqrt` is checked, depending on `q_diag`.\n",
    "        :param q_diag: bool\n",
    "            Used to check if `q_mu` and `q_sqrt` have the correct shape or to\n",
    "            construct them with the correct shape. If `q_diag` is true,\n",
    "            `q_sqrt` is two dimensional and only holds the square root of the\n",
    "            covariance diagonal elements. If False, `q_sqrt` is three dimensional.\n",
    "        \"\"\"\n",
    "        dtype = gpflow.config.default_float() #gpflow.config.default_float()\n",
    "        q_mu = np.zeros((num_inducing, self.num_latent)) if q_mu is None else q_mu\n",
    "        self.q_mu = Parameter(q_mu, dtype=dtype)  # M x P\n",
    "\n",
    "        if q_sqrt is None:\n",
    "            if self.q_diag:\n",
    "                self.q_sqrt = Parameter(np.ones((num_inducing, self.num_latent), dtype=dtype),\n",
    "                                        transform=positive())  # M x P\n",
    "            else:\n",
    "                q_sqrt = np.array([np.eye(num_inducing, dtype=dtype) for _ in range(self.num_latent)])\n",
    "                self.q_sqrt = Parameter(q_sqrt, transform=transforms.LowerTriangular(num_inducing,\n",
    "                                                                                     self.num_latent))  # P x M x M\n",
    "        else:\n",
    "            if q_diag:\n",
    "                assert q_sqrt.ndim == 2\n",
    "                self.num_latent = q_sqrt.shape[1]\n",
    "                self.q_sqrt = Parameter(q_sqrt, transform=positive())  # M x L/P\n",
    "            else:\n",
    "                assert q_sqrt.ndim == 3\n",
    "                self.num_latent = q_sqrt.shape[0]\n",
    "                num_inducing = q_sqrt.shape[1]\n",
    "                self.q_sqrt = Parameter(q_sqrt, transform=transforms.LowerTriangular(num_inducing,\n",
    "                                                                                     self.num_latent))  # L/P x M x M\n",
    "    def build_prior_KL(self):\n",
    "        if self.whiten:\n",
    "            K = None\n",
    "        else:\n",
    "            K = Kuu(self.feature, self.kern, jitter=settings.numerics.jitter_level)  # (P x) x M x M\n",
    "\n",
    "        return kullback_leiblers.gauss_kl(self.q_mu, self.q_sqrt, K)\n",
    "\n",
    "    def _build_likelihood(self):\n",
    "        \"\"\"\n",
    "        This gives a variational bound on the model likelihood.\n",
    "        \"\"\"\n",
    "\n",
    "        # Get prior KL\n",
    "        KL = self.build_prior_KL()\n",
    "\n",
    "        # Get conditionals\n",
    "        # TODO: allow for block-diagonal covariance\n",
    "        fmeans, fvars = self._build_predict(self.X, full_cov=False, full_output_cov=False)\n",
    "\n",
    "        # Get variational expectations\n",
    "        var_exp = self.likelihood.variational_expectations(fmeans, fvars, self.Y, full_cov=False)\n",
    "\n",
    "        # re-scale for minibatch size\n",
    "        scale = tf.cast(self.num_data, gpflow.config.default_float()) / tf.cast(tf.shape(self.X)[0], settings.float_type)\n",
    "\n",
    "        return tf.reduce_sum(var_exp) * scale - KL\n",
    "\n",
    "    def _build_predict(self, Xnew, full_cov=False, full_output_cov=False):\n",
    "        \"\"\"\n",
    "        Compute the mean and variance of :math:`p(f_* \\\\mid y)`.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        Xnew : np.array, shape=(N, K)\n",
    "        full_cov : bool\n",
    "        full_output_cov : bool\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        mus, vars :\n",
    "            Mean and covariances of the variational approximation to the GP applied to the K input dimensions of Xnew.\n",
    "            Dimensions: mus= N x K and vars= N x K (x K)\n",
    "\n",
    "        \"\"\"\n",
    "        # Reshape to obtain correct covariance\n",
    "        num_data_new = tf.shape(Xnew)[0]\n",
    "        Xnew = tf.reshape(Xnew, [-1, 1])\n",
    "\n",
    "        # Compute conditional\n",
    "        mu_tmp, var_tmp = conditional(Xnew, self.feature, self.kern, self.q_mu, q_sqrt=self.q_sqrt,\n",
    "                                      full_cov=full_cov,\n",
    "                                      white=self.whiten, full_output_cov=full_output_cov)\n",
    "\n",
    "        # Reshape to N x K\n",
    "        mu = tf.reshape(mu_tmp + self.mean_function(Xnew), [num_data_new, self.num_classes])\n",
    "        var = tf.reshape(var_tmp, [num_data_new, self.num_classes])\n",
    "\n",
    "        return mu, var\n",
    "\n",
    "    def predict_f(self, X_onedim, full_cov=False, full_output_cov=False):\n",
    "        \"\"\"\n",
    "        Predict the one-dimensional latent function\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        X_onedim\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "\n",
    "        \"\"\"\n",
    "        # Compute conditional\n",
    "        mu, var = conditional(X_onedim, self.feature, self.kern, self.q_mu, q_sqrt=self.q_sqrt,\n",
    "                              full_cov=full_cov,\n",
    "                              white=self.whiten, full_output_cov=full_output_cov)\n",
    "\n",
    "        return mu + self.mean_function(X_onedim), var\n",
    "\n",
    "    def predict_full_density(self, Xnew):\n",
    "        pred_f_mean, pred_f_var = self._build_predict(Xnew)\n",
    "        return self.likelihood.predict_full_density(pred_f_mean, pred_f_var)\n",
    "    \n",
    "    def maximum_log_likelihood_objective(self):\n",
    "        NotImplemented\n",
    "    #    return -self.log_marginal_likelihood()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "36338265-4ec7-4664-9f59-437b010169d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "random_state = 1\n",
    "recal_f = GPCalibration(n_classes=2, maxiter=1000, n_inducing_points=10, logits=False, random_state=random_state)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dae6f1fc-4d54-4848-8aa4-0e4af120d408",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "e280e6e8-25b9-45d3-b0d9-dd5aea435f02",
   "metadata": {},
   "outputs": [],
   "source": [
    "files = ['kitti_all_train.data', 'kitti_all_train.labels', 'kitti_all_test.data', 'kitti_all_test.labels']\n",
    "file_path = os.getcwd() + '/kitti_features/'\n",
    "\n",
    "X_train = np.loadtxt(os.path.join(file_path, files[0]), np.float64, skiprows=1)\n",
    "y_train = np.loadtxt(os.path.join(file_path, files[1]), np.int32, skiprows=1)\n",
    "X_test = np.loadtxt(os.path.join(file_path, files[2]), np.float64, skiprows=1)\n",
    "y_test = np.loadtxt(os.path.join(file_path, files[3]), np.int32, skiprows=1)\n",
    "\n",
    "y_train = np.where(y_train > 0, 1, 0)\n",
    "y_test = np.where(y_test > 0, 1, 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "4cfb1d46-95d2-40b9-be18-9c031a2ad1d1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style>#sk-container-id-1 {color: black;}#sk-container-id-1 pre{padding: 0;}#sk-container-id-1 div.sk-toggleable {background-color: white;}#sk-container-id-1 label.sk-toggleable__label {cursor: pointer;display: block;width: 100%;margin-bottom: 0;padding: 0.3em;box-sizing: border-box;text-align: center;}#sk-container-id-1 label.sk-toggleable__label-arrow:before {content: \"▸\";float: left;margin-right: 0.25em;color: #696969;}#sk-container-id-1 label.sk-toggleable__label-arrow:hover:before {color: black;}#sk-container-id-1 div.sk-estimator:hover label.sk-toggleable__label-arrow:before {color: black;}#sk-container-id-1 div.sk-toggleable__content {max-height: 0;max-width: 0;overflow: hidden;text-align: left;background-color: #f0f8ff;}#sk-container-id-1 div.sk-toggleable__content pre {margin: 0.2em;color: black;border-radius: 0.25em;background-color: #f0f8ff;}#sk-container-id-1 input.sk-toggleable__control:checked~div.sk-toggleable__content {max-height: 200px;max-width: 100%;overflow: auto;}#sk-container-id-1 input.sk-toggleable__control:checked~label.sk-toggleable__label-arrow:before {content: \"▾\";}#sk-container-id-1 div.sk-estimator input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-1 div.sk-label input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-1 input.sk-hidden--visually {border: 0;clip: rect(1px 1px 1px 1px);clip: rect(1px, 1px, 1px, 1px);height: 1px;margin: -1px;overflow: hidden;padding: 0;position: absolute;width: 1px;}#sk-container-id-1 div.sk-estimator {font-family: monospace;background-color: #f0f8ff;border: 1px dotted black;border-radius: 0.25em;box-sizing: border-box;margin-bottom: 0.5em;}#sk-container-id-1 div.sk-estimator:hover {background-color: #d4ebff;}#sk-container-id-1 div.sk-parallel-item::after {content: \"\";width: 100%;border-bottom: 1px solid gray;flex-grow: 1;}#sk-container-id-1 div.sk-label:hover label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-1 div.sk-serial::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: 0;}#sk-container-id-1 div.sk-serial {display: flex;flex-direction: column;align-items: center;background-color: white;padding-right: 0.2em;padding-left: 0.2em;position: relative;}#sk-container-id-1 div.sk-item {position: relative;z-index: 1;}#sk-container-id-1 div.sk-parallel {display: flex;align-items: stretch;justify-content: center;background-color: white;position: relative;}#sk-container-id-1 div.sk-item::before, #sk-container-id-1 div.sk-parallel-item::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: -1;}#sk-container-id-1 div.sk-parallel-item {display: flex;flex-direction: column;z-index: 1;position: relative;background-color: white;}#sk-container-id-1 div.sk-parallel-item:first-child::after {align-self: flex-end;width: 50%;}#sk-container-id-1 div.sk-parallel-item:last-child::after {align-self: flex-start;width: 50%;}#sk-container-id-1 div.sk-parallel-item:only-child::after {width: 0;}#sk-container-id-1 div.sk-dashed-wrapped {border: 1px dashed gray;margin: 0 0.4em 0.5em 0.4em;box-sizing: border-box;padding-bottom: 0.4em;background-color: white;}#sk-container-id-1 div.sk-label label {font-family: monospace;font-weight: bold;display: inline-block;line-height: 1.2em;}#sk-container-id-1 div.sk-label-container {text-align: center;}#sk-container-id-1 div.sk-container {/* jupyter's `normalize.less` sets `[hidden] { display: none; }` but bootstrap.min.css set `[hidden] { display: none !important; }` so we also need the `!important` here to be able to override the default hidden behavior on the sphinx rendered scikit-learn.org. See: https://github.com/scikit-learn/scikit-learn/issues/21755 */display: inline-block !important;position: relative;}#sk-container-id-1 div.sk-text-repr-fallback {display: none;}</style><div id=\"sk-container-id-1\" class=\"sk-top-container\"><div class=\"sk-text-repr-fallback\"><pre>XGBClassifier(base_score=None, booster=&#x27;gbtree&#x27;, callbacks=None,\n",
       "              colsample_bylevel=None, colsample_bynode=None,\n",
       "              colsample_bytree=None, device=None, early_stopping_rounds=None,\n",
       "              enable_categorical=False, eval_metric=None, feature_types=None,\n",
       "              gamma=None, grow_policy=None, importance_type=None,\n",
       "              interaction_constraints=None, learning_rate=None, max_bin=None,\n",
       "              max_cat_threshold=None, max_cat_to_onehot=None,\n",
       "              max_delta_step=None, max_depth=None, max_leaves=None,\n",
       "              min_child_weight=None, missing=nan, monotone_constraints=None,\n",
       "              multi_strategy=None, n_estimators=100, n_jobs=-1,\n",
       "              num_parallel_tree=None, random_state=1, ...)</pre><b>In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook. <br />On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.</b></div><div class=\"sk-container\" hidden><div class=\"sk-item\"><div class=\"sk-estimator sk-toggleable\"><input class=\"sk-toggleable__control sk-hidden--visually\" id=\"sk-estimator-id-1\" type=\"checkbox\" checked><label for=\"sk-estimator-id-1\" class=\"sk-toggleable__label sk-toggleable__label-arrow\">XGBClassifier</label><div class=\"sk-toggleable__content\"><pre>XGBClassifier(base_score=None, booster=&#x27;gbtree&#x27;, callbacks=None,\n",
       "              colsample_bylevel=None, colsample_bynode=None,\n",
       "              colsample_bytree=None, device=None, early_stopping_rounds=None,\n",
       "              enable_categorical=False, eval_metric=None, feature_types=None,\n",
       "              gamma=None, grow_policy=None, importance_type=None,\n",
       "              interaction_constraints=None, learning_rate=None, max_bin=None,\n",
       "              max_cat_threshold=None, max_cat_to_onehot=None,\n",
       "              max_delta_step=None, max_depth=None, max_leaves=None,\n",
       "              min_child_weight=None, missing=nan, monotone_constraints=None,\n",
       "              multi_strategy=None, n_estimators=100, n_jobs=-1,\n",
       "              num_parallel_tree=None, random_state=1, ...)</pre></div></div></div></div></div>"
      ],
      "text/plain": [
       "XGBClassifier(base_score=None, booster='gbtree', callbacks=None,\n",
       "              colsample_bylevel=None, colsample_bynode=None,\n",
       "              colsample_bytree=None, device=None, early_stopping_rounds=None,\n",
       "              enable_categorical=False, eval_metric=None, feature_types=None,\n",
       "              gamma=None, grow_policy=None, importance_type=None,\n",
       "              interaction_constraints=None, learning_rate=None, max_bin=None,\n",
       "              max_cat_threshold=None, max_cat_to_onehot=None,\n",
       "              max_delta_step=None, max_depth=None, max_leaves=None,\n",
       "              min_child_weight=None, missing=nan, monotone_constraints=None,\n",
       "              multi_strategy=None, n_estimators=100, n_jobs=-1,\n",
       "              num_parallel_tree=None, random_state=1, ...)"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Classify KITTI validation data with selected classifiers\n",
    "random_state = 1\n",
    "model = xgb.XGBClassifier(booster=\"gbtree\", n_estimators=100, random_state=random_state, n_jobs=-1)\n",
    "#model = RandomForestClassifier(n_estimators=100, criterion=\"gini\", min_samples_split=2, bootstrap=True, n_jobs=-1, random_state=random_state)\n",
    "\n",
    "model.fit(X=X_train, y=y_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "0a24087d-2017-47f5-8ffb-aadc4b5802f8",
   "metadata": {},
   "outputs": [],
   "source": [
    "p_pred_train = model.predict_proba(X_train)\n",
    "p_pred = model.predict_proba(X_test)\n",
    "y_y_pred = np.column_stack([y_test, np.argmax(p_pred, axis=1)])\n",
    "# Remove possible nan or inf\n",
    "isnotnan_ind = ~np.isnan(p_pred).any(axis=1)\n",
    "p_pred = p_pred[isnotnan_ind, :]\n",
    "y_y_pred = y_y_pred[isnotnan_ind, :]\n",
    "\n",
    "isnotnan_ind = ~np.isnan(p_pred_train).any(axis=1)\n",
    "p_pred_train = p_pred_train[isnotnan_ind, :]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 264,
   "id": "20b9f17e-8502-438b-a3af-e96c6dc277b4",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/ipykernel_93113/2086292548.py:67: DeprecationWarning: Call to deprecated method __len__. (len(iv) should return an `int`, but this actually returns a `tf.Tensor`. Use `iv.num_inducing` instead.)\n",
      "  num_inducing = len(self.feature)\n"
     ]
    },
    {
     "ename": "ValueError",
     "evalue": "in user code:\n\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/optimizers/scipy.py\", line 192, in _tf_eval  *\n        loss, grads = _compute_loss_and_gradients(\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/optimizers/scipy.py\", line 329, in _compute_loss_and_gradients  *\n        loss = loss_closure()\n    File \"/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/ipykernel_93113/2086292548.py\", line 138, in _build_likelihood  *\n        fmeans, fvars = self._build_predict(self.X, full_cov=False, full_output_cov=False)\n    File \"/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/ipykernel_93113/2086292548.py\", line 170, in _build_predict  *\n        mu_tmp, var_tmp = conditional(Xnew, self.feature, self.kern, self.q_mu, q_sqrt=self.q_sqrt,\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/multipledispatch/dispatcher.py\", line 279, in __call__  *\n        return func(*args, **kwargs)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/decorator.py\", line 53, in wrapped_function  *\n        if not get_enable_check_shapes():\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/conditionals/conditionals.py\", line 87, in _sparse_conditional  *\n        return posterior.fused_predict_f(Xnew, full_cov=full_cov, full_output_cov=full_output_cov)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/integration/tf.py\", line 241, in wrapped_method  *\n        return wrapped_function(self, *args, **kwargs)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/decorator.py\", line 53, in wrapped_function  *\n        if not get_enable_check_shapes():\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/posteriors.py\", line 255, in fused_predict_f  *\n        mean, cov = self._conditional_fused(\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/integration/tf.py\", line 241, in wrapped_method  *\n        return wrapped_function(self, *args, **kwargs)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/decorator.py\", line 53, in wrapped_function  *\n        if not get_enable_check_shapes():\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/posteriors.py\", line 833, in _conditional_fused  *\n        Knn = self.kernel(Xnew, full_cov=full_cov)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/kernels/base.py\", line 290, in __call__  *\n        [k(X, X2, full_cov=full_cov, presliced=presliced) for k in self.kernels]\n\n    ValueError: Shapes must be equal rank, but are 1 and 0\n    \tFrom merging shape 0 with other shapes. for '{{node AddN}} = AddN[N=2, T=DT_DOUBLE](Fill, Fill_1)' with input shapes: [32000], [].\n",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mValueError\u001b[0m                                Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[264], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m \u001b[43mrecal_f\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mfit\u001b[49m\u001b[43m(\u001b[49m\u001b[43mp_pred_train\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43my_train\u001b[49m\u001b[43m)\u001b[49m\n",
      "Cell \u001b[0;32mIn[258], line 189\u001b[0m, in \u001b[0;36mGPCalibration.fit\u001b[0;34m(self, X, y)\u001b[0m\n\u001b[1;32m    187\u001b[0m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mValueError\u001b[39;00m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mCalibration training data must have shape (n_samples, n_classes).\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n\u001b[1;32m    188\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m--> 189\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_fit_multiclass\u001b[49m\u001b[43m(\u001b[49m\u001b[43mX\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43my\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    191\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\n",
      "Cell \u001b[0;32mIn[258], line 213\u001b[0m, in \u001b[0;36mGPCalibration._fit_multiclass\u001b[0;34m(self, X, y)\u001b[0m\n\u001b[1;32m    209\u001b[0m \u001b[38;5;66;03m# Optimize parameters\u001b[39;00m\n\u001b[1;32m    210\u001b[0m \u001b[38;5;66;03m#opt = gpflow.train.ScipyOptimizer()\u001b[39;00m\n\u001b[1;32m    211\u001b[0m \u001b[38;5;66;03m#opt.minimize(self.model, maxiter=self.maxiter, disp=self.verbose)\u001b[39;00m\n\u001b[1;32m    212\u001b[0m optimizer \u001b[38;5;241m=\u001b[39m gpflow\u001b[38;5;241m.\u001b[39moptimizers\u001b[38;5;241m.\u001b[39mScipy()\n\u001b[0;32m--> 213\u001b[0m \u001b[43moptimizer\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mminimize\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mmodel\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_build_likelihood\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mvariables\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mmodel\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mtrainable_variables\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\n\u001b[1;32m    214\u001b[0m \u001b[43m                   \u001b[49m\u001b[43moptions\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mdict\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mdisp\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mmaxiter\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mmaxiter\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    216\u001b[0m \u001b[38;5;66;03m#@tf.function\u001b[39;00m\n\u001b[1;32m    217\u001b[0m \u001b[38;5;66;03m#def optimization_step():\u001b[39;00m\n\u001b[1;32m    218\u001b[0m \u001b[38;5;66;03m#    with tf.GradientTape() as tape:\u001b[39;00m\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m    223\u001b[0m \u001b[38;5;66;03m#for _ in range(self.maxiter):\u001b[39;00m\n\u001b[1;32m    224\u001b[0m \u001b[38;5;66;03m#    optimization_step()\u001b[39;00m\n\u001b[1;32m    226\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/optimizers/scipy.py:159\u001b[0m, in \u001b[0;36mScipy.minimize\u001b[0;34m(self, closure, variables, method, step_callback, compile, allow_unused_variables, tf_fun_args, track_loss_history, **scipy_kwargs)\u001b[0m\n\u001b[1;32m    156\u001b[0m     callback \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mloss_history_callback_func(func, history, scipy_kwargs\u001b[38;5;241m.\u001b[39mget(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mcallback\u001b[39m\u001b[38;5;124m\"\u001b[39m))\n\u001b[1;32m    157\u001b[0m     scipy_kwargs[\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mcallback\u001b[39m\u001b[38;5;124m\"\u001b[39m] \u001b[38;5;241m=\u001b[39m callback\n\u001b[0;32m--> 159\u001b[0m opt_result \u001b[38;5;241m=\u001b[39m \u001b[43mscipy\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43moptimize\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mminimize\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m    160\u001b[0m \u001b[43m    \u001b[49m\u001b[43mfunc\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43minitial_params\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mjac\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mmethod\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mmethod\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mscipy_kwargs\u001b[49m\n\u001b[1;32m    161\u001b[0m \u001b[43m\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    163\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m track_loss_history:\n\u001b[1;32m    164\u001b[0m     opt_result[\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mloss_history\u001b[39m\u001b[38;5;124m\"\u001b[39m] \u001b[38;5;241m=\u001b[39m history\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/scipy/optimize/_minimize.py:710\u001b[0m, in \u001b[0;36mminimize\u001b[0;34m(fun, x0, args, method, jac, hess, hessp, bounds, constraints, tol, callback, options)\u001b[0m\n\u001b[1;32m    707\u001b[0m     res \u001b[38;5;241m=\u001b[39m _minimize_newtoncg(fun, x0, args, jac, hess, hessp, callback,\n\u001b[1;32m    708\u001b[0m                              \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39moptions)\n\u001b[1;32m    709\u001b[0m \u001b[38;5;28;01melif\u001b[39;00m meth \u001b[38;5;241m==\u001b[39m \u001b[38;5;124m'\u001b[39m\u001b[38;5;124ml-bfgs-b\u001b[39m\u001b[38;5;124m'\u001b[39m:\n\u001b[0;32m--> 710\u001b[0m     res \u001b[38;5;241m=\u001b[39m \u001b[43m_minimize_lbfgsb\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfun\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mx0\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mjac\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mbounds\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    711\u001b[0m \u001b[43m                           \u001b[49m\u001b[43mcallback\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mcallback\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43moptions\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    712\u001b[0m \u001b[38;5;28;01melif\u001b[39;00m meth \u001b[38;5;241m==\u001b[39m \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mtnc\u001b[39m\u001b[38;5;124m'\u001b[39m:\n\u001b[1;32m    713\u001b[0m     res \u001b[38;5;241m=\u001b[39m _minimize_tnc(fun, x0, args, jac, bounds, callback\u001b[38;5;241m=\u001b[39mcallback,\n\u001b[1;32m    714\u001b[0m                         \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39moptions)\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/scipy/optimize/_lbfgsb_py.py:307\u001b[0m, in \u001b[0;36m_minimize_lbfgsb\u001b[0;34m(fun, x0, args, jac, bounds, disp, maxcor, ftol, gtol, eps, maxfun, maxiter, iprint, callback, maxls, finite_diff_rel_step, **unknown_options)\u001b[0m\n\u001b[1;32m    304\u001b[0m     \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m    305\u001b[0m         iprint \u001b[38;5;241m=\u001b[39m disp\n\u001b[0;32m--> 307\u001b[0m sf \u001b[38;5;241m=\u001b[39m \u001b[43m_prepare_scalar_function\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfun\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mx0\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mjac\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mjac\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43margs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mepsilon\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43meps\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    308\u001b[0m \u001b[43m                              \u001b[49m\u001b[43mbounds\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mnew_bounds\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    309\u001b[0m \u001b[43m                              \u001b[49m\u001b[43mfinite_diff_rel_step\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mfinite_diff_rel_step\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    311\u001b[0m func_and_grad \u001b[38;5;241m=\u001b[39m sf\u001b[38;5;241m.\u001b[39mfun_and_grad\n\u001b[1;32m    313\u001b[0m fortran_int \u001b[38;5;241m=\u001b[39m _lbfgsb\u001b[38;5;241m.\u001b[39mtypes\u001b[38;5;241m.\u001b[39mintvar\u001b[38;5;241m.\u001b[39mdtype\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/scipy/optimize/_optimize.py:383\u001b[0m, in \u001b[0;36m_prepare_scalar_function\u001b[0;34m(fun, x0, jac, args, bounds, epsilon, finite_diff_rel_step, hess)\u001b[0m\n\u001b[1;32m    379\u001b[0m     bounds \u001b[38;5;241m=\u001b[39m (\u001b[38;5;241m-\u001b[39mnp\u001b[38;5;241m.\u001b[39minf, np\u001b[38;5;241m.\u001b[39minf)\n\u001b[1;32m    381\u001b[0m \u001b[38;5;66;03m# ScalarFunction caches. Reuse of fun(x) during grad\u001b[39;00m\n\u001b[1;32m    382\u001b[0m \u001b[38;5;66;03m# calculation reduces overall function evaluations.\u001b[39;00m\n\u001b[0;32m--> 383\u001b[0m sf \u001b[38;5;241m=\u001b[39m \u001b[43mScalarFunction\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfun\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mx0\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mgrad\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mhess\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    384\u001b[0m \u001b[43m                    \u001b[49m\u001b[43mfinite_diff_rel_step\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mbounds\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mepsilon\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mepsilon\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    386\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m sf\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/scipy/optimize/_differentiable_functions.py:158\u001b[0m, in \u001b[0;36mScalarFunction.__init__\u001b[0;34m(self, fun, x0, args, grad, hess, finite_diff_rel_step, finite_diff_bounds, epsilon)\u001b[0m\n\u001b[1;32m    155\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mf \u001b[38;5;241m=\u001b[39m fun_wrapped(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mx)\n\u001b[1;32m    157\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_update_fun_impl \u001b[38;5;241m=\u001b[39m update_fun\n\u001b[0;32m--> 158\u001b[0m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_update_fun\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    160\u001b[0m \u001b[38;5;66;03m# Gradient evaluation\u001b[39;00m\n\u001b[1;32m    161\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mcallable\u001b[39m(grad):\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/scipy/optimize/_differentiable_functions.py:251\u001b[0m, in \u001b[0;36mScalarFunction._update_fun\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    249\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21m_update_fun\u001b[39m(\u001b[38;5;28mself\u001b[39m):\n\u001b[1;32m    250\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mf_updated:\n\u001b[0;32m--> 251\u001b[0m         \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_update_fun_impl\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    252\u001b[0m         \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mf_updated \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/scipy/optimize/_differentiable_functions.py:155\u001b[0m, in \u001b[0;36mScalarFunction.__init__.<locals>.update_fun\u001b[0;34m()\u001b[0m\n\u001b[1;32m    154\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mupdate_fun\u001b[39m():\n\u001b[0;32m--> 155\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mf \u001b[38;5;241m=\u001b[39m \u001b[43mfun_wrapped\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mx\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/scipy/optimize/_differentiable_functions.py:137\u001b[0m, in \u001b[0;36mScalarFunction.__init__.<locals>.fun_wrapped\u001b[0;34m(x)\u001b[0m\n\u001b[1;32m    133\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mnfev \u001b[38;5;241m+\u001b[39m\u001b[38;5;241m=\u001b[39m \u001b[38;5;241m1\u001b[39m\n\u001b[1;32m    134\u001b[0m \u001b[38;5;66;03m# Send a copy because the user may overwrite it.\u001b[39;00m\n\u001b[1;32m    135\u001b[0m \u001b[38;5;66;03m# Overwriting results in undefined behaviour because\u001b[39;00m\n\u001b[1;32m    136\u001b[0m \u001b[38;5;66;03m# fun(self.x) will change self.x, with the two no longer linked.\u001b[39;00m\n\u001b[0;32m--> 137\u001b[0m fx \u001b[38;5;241m=\u001b[39m \u001b[43mfun\u001b[49m\u001b[43m(\u001b[49m\u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mcopy\u001b[49m\u001b[43m(\u001b[49m\u001b[43mx\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    138\u001b[0m \u001b[38;5;66;03m# Make sure the function returns a true scalar\u001b[39;00m\n\u001b[1;32m    139\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m np\u001b[38;5;241m.\u001b[39misscalar(fx):\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/scipy/optimize/_optimize.py:77\u001b[0m, in \u001b[0;36mMemoizeJac.__call__\u001b[0;34m(self, x, *args)\u001b[0m\n\u001b[1;32m     75\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21m__call__\u001b[39m(\u001b[38;5;28mself\u001b[39m, x, \u001b[38;5;241m*\u001b[39margs):\n\u001b[1;32m     76\u001b[0m \u001b[38;5;250m    \u001b[39m\u001b[38;5;124;03m\"\"\" returns the function value \"\"\"\u001b[39;00m\n\u001b[0;32m---> 77\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_compute_if_needed\u001b[49m\u001b[43m(\u001b[49m\u001b[43mx\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     78\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_value\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/scipy/optimize/_optimize.py:71\u001b[0m, in \u001b[0;36mMemoizeJac._compute_if_needed\u001b[0;34m(self, x, *args)\u001b[0m\n\u001b[1;32m     69\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m np\u001b[38;5;241m.\u001b[39mall(x \u001b[38;5;241m==\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mx) \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_value \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mjac \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m     70\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mx \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39masarray(x)\u001b[38;5;241m.\u001b[39mcopy()\n\u001b[0;32m---> 71\u001b[0m     fg \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mfun\u001b[49m\u001b[43m(\u001b[49m\u001b[43mx\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     72\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mjac \u001b[38;5;241m=\u001b[39m fg[\u001b[38;5;241m1\u001b[39m]\n\u001b[1;32m     73\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_value \u001b[38;5;241m=\u001b[39m fg[\u001b[38;5;241m0\u001b[39m]\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/optimizers/scipy.py:224\u001b[0m, in \u001b[0;36mScipy.eval_func.<locals>._eval\u001b[0;34m(x)\u001b[0m\n\u001b[1;32m    223\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21m_eval\u001b[39m(x: AnyNDArray) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m Tuple[AnyNDArray, AnyNDArray]:\n\u001b[0;32m--> 224\u001b[0m     loss, grad \u001b[38;5;241m=\u001b[39m \u001b[43m_tf_eval\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtf\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconvert_to_tensor\u001b[49m\u001b[43m(\u001b[49m\u001b[43mx\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    225\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m loss\u001b[38;5;241m.\u001b[39mnumpy()\u001b[38;5;241m.\u001b[39mastype(np\u001b[38;5;241m.\u001b[39mfloat64), grad\u001b[38;5;241m.\u001b[39mnumpy()\u001b[38;5;241m.\u001b[39mastype(np\u001b[38;5;241m.\u001b[39mfloat64)\n",
      "File \u001b[0;32m~/.pyenv/versions/3.11.6/lib/python3.11/site-packages/tensorflow/python/util/traceback_utils.py:153\u001b[0m, in \u001b[0;36mfilter_traceback.<locals>.error_handler\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m    151\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m \u001b[38;5;167;01mException\u001b[39;00m \u001b[38;5;28;01mas\u001b[39;00m e:\n\u001b[1;32m    152\u001b[0m   filtered_tb \u001b[38;5;241m=\u001b[39m _process_traceback_frames(e\u001b[38;5;241m.\u001b[39m__traceback__)\n\u001b[0;32m--> 153\u001b[0m   \u001b[38;5;28;01mraise\u001b[39;00m e\u001b[38;5;241m.\u001b[39mwith_traceback(filtered_tb) \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m\n\u001b[1;32m    154\u001b[0m \u001b[38;5;28;01mfinally\u001b[39;00m:\n\u001b[1;32m    155\u001b[0m   \u001b[38;5;28;01mdel\u001b[39;00m filtered_tb\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_file2pxmbj_4.py:37\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf___tf_eval\u001b[0;34m(x)\u001b[0m\n\u001b[1;32m     35\u001b[0m grads \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mUndefined(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mgrads\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m     36\u001b[0m loss \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mUndefined(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mloss\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[0;32m---> 37\u001b[0m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mif_stmt\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfirst_call\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mif_body\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43melse_body\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mget_state\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mset_state\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mgrads\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mloss\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mfirst_call\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m2\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m     38\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m     39\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_file2pxmbj_4.py:28\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf___tf_eval.<locals>.if_body\u001b[0;34m()\u001b[0m\n\u001b[1;32m     26\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mif_body\u001b[39m():\n\u001b[1;32m     27\u001b[0m     \u001b[38;5;28;01mnonlocal\u001b[39;00m grads, first_call, loss\n\u001b[0;32m---> 28\u001b[0m     loss, grads \u001b[38;5;241m=\u001b[39m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconverted_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43m_compute_loss_and_gradients\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mclosure\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mvariables\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtf\u001b[49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mUnconnectedGradients\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mNONE\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mNone\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfscope\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     29\u001b[0m     grads \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39m_filter_unused_variables, (ag__\u001b[38;5;241m.\u001b[39mld(variables), ag__\u001b[38;5;241m.\u001b[39mld(grads), ag__\u001b[38;5;241m.\u001b[39mld(allow_unused_variables)), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)\n\u001b[1;32m     30\u001b[0m     first_call \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_filey6aruf3_.py:12\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf___compute_loss_and_gradients\u001b[0;34m(loss_closure, variables, unconnected_gradients)\u001b[0m\n\u001b[1;32m     10\u001b[0m \u001b[38;5;28;01mwith\u001b[39;00m ag__\u001b[38;5;241m.\u001b[39mld(tf)\u001b[38;5;241m.\u001b[39mGradientTape(watch_accessed_variables\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m) \u001b[38;5;28;01mas\u001b[39;00m tape:\n\u001b[1;32m     11\u001b[0m     ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(tape)\u001b[38;5;241m.\u001b[39mwatch, (ag__\u001b[38;5;241m.\u001b[39mld(variables),), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)\n\u001b[0;32m---> 12\u001b[0m     loss \u001b[38;5;241m=\u001b[39m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconverted_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mloss_closure\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mNone\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfscope\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     13\u001b[0m grads \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(tape)\u001b[38;5;241m.\u001b[39mgradient, (ag__\u001b[38;5;241m.\u001b[39mld(loss), ag__\u001b[38;5;241m.\u001b[39mld(variables)), \u001b[38;5;28mdict\u001b[39m(unconnected_gradients\u001b[38;5;241m=\u001b[39mag__\u001b[38;5;241m.\u001b[39mld(unconnected_gradients)), fscope)\n\u001b[1;32m     14\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_fileooulxlvb.py:14\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf___build_likelihood\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m     12\u001b[0m retval_ \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mUndefinedReturnValue()\n\u001b[1;32m     13\u001b[0m KL \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mbuild_prior_KL, (), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)\n\u001b[0;32m---> 14\u001b[0m fmeans, fvars \u001b[38;5;241m=\u001b[39m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconverted_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_build_predict\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mX\u001b[49m\u001b[43m,\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mdict\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mfull_cov\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfull_output_cov\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfscope\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     15\u001b[0m var_exp \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mlikelihood\u001b[38;5;241m.\u001b[39mvariational_expectations, (ag__\u001b[38;5;241m.\u001b[39mld(fmeans), ag__\u001b[38;5;241m.\u001b[39mld(fvars), ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mY), \u001b[38;5;28mdict\u001b[39m(full_cov\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m), fscope)\n\u001b[1;32m     16\u001b[0m scale \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(tf)\u001b[38;5;241m.\u001b[39mcast, (ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mnum_data, ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(gpflow)\u001b[38;5;241m.\u001b[39mconfig\u001b[38;5;241m.\u001b[39mdefault_float, (), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)), \u001b[38;5;28;01mNone\u001b[39;00m, fscope) \u001b[38;5;241m/\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(tf)\u001b[38;5;241m.\u001b[39mcast, (ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(tf)\u001b[38;5;241m.\u001b[39mshape, (ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mX,), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)[\u001b[38;5;241m0\u001b[39m], ag__\u001b[38;5;241m.\u001b[39mld(settings)\u001b[38;5;241m.\u001b[39mfloat_type), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_file8ju86kre.py:28\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf___build_predict\u001b[0;34m(self, Xnew, full_cov, full_output_cov)\u001b[0m\n\u001b[1;32m     26\u001b[0m num_data_new \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(tf)\u001b[38;5;241m.\u001b[39mshape, (ag__\u001b[38;5;241m.\u001b[39mld(Xnew),), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)[\u001b[38;5;241m0\u001b[39m]\n\u001b[1;32m     27\u001b[0m Xnew \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(tf)\u001b[38;5;241m.\u001b[39mreshape, (ag__\u001b[38;5;241m.\u001b[39mld(Xnew), [\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m, \u001b[38;5;241m1\u001b[39m]), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)\n\u001b[0;32m---> 28\u001b[0m mu_tmp, var_tmp \u001b[38;5;241m=\u001b[39m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconverted_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mconditional\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mXnew\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mfeature\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mkern\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mq_mu\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mdict\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mq_sqrt\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mq_sqrt\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfull_cov\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfull_cov\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mwhite\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mwhiten\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfull_output_cov\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfull_output_cov\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfscope\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     29\u001b[0m mu \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(tf)\u001b[38;5;241m.\u001b[39mreshape, (ag__\u001b[38;5;241m.\u001b[39mld(mu_tmp) \u001b[38;5;241m+\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mmean_function, (ag__\u001b[38;5;241m.\u001b[39mld(Xnew),), \u001b[38;5;28;01mNone\u001b[39;00m, fscope), [ag__\u001b[38;5;241m.\u001b[39mld(num_data_new), ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mnum_classes]), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)\n\u001b[1;32m     30\u001b[0m var \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(tf)\u001b[38;5;241m.\u001b[39mreshape, (ag__\u001b[38;5;241m.\u001b[39mld(var_tmp), [ag__\u001b[38;5;241m.\u001b[39mld(num_data_new), ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mnum_classes]), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_file23gu21ef.py:32\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf____call__\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m     30\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m     31\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[0;32m---> 32\u001b[0m     retval_ \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(func), \u001b[38;5;28mtuple\u001b[39m(ag__\u001b[38;5;241m.\u001b[39mld(args)), \u001b[38;5;28mdict\u001b[39m(\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mag__\u001b[38;5;241m.\u001b[39mld(kwargs)), fscope)\n\u001b[1;32m     33\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m:\n\u001b[1;32m     34\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_fileosceii7d.py:40\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf__wrapped_function\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m     38\u001b[0m     \u001b[38;5;28;01mnonlocal\u001b[39;00m do_return, retval_\n\u001b[1;32m     39\u001b[0m     \u001b[38;5;28;01mpass\u001b[39;00m\n\u001b[0;32m---> 40\u001b[0m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mif_stmt\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mnot_\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconverted_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mget_enable_check_shapes\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mNone\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfscope\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mif_body\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43melse_body\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mget_state\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mset_state\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mdo_return\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mretval_\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m2\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m     42\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mget_state_8\u001b[39m():\n\u001b[1;32m     43\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m (arg_map, ag__\u001b[38;5;241m.\u001b[39mldu(\u001b[38;5;28;01mlambda\u001b[39;00m: arg_map[RESULT_TOKEN], \u001b[38;5;124m'\u001b[39m\u001b[38;5;124marg_map[RESULT_TOKEN]\u001b[39m\u001b[38;5;124m'\u001b[39m), checker, do_return, retval_)\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_filepf4l6fhw.py:40\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf___sparse_conditional\u001b[0;34m(Xnew, inducing_variable, kernel, f, full_cov, full_output_cov, q_sqrt, white)\u001b[0m\n\u001b[1;32m     38\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m     39\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[0;32m---> 40\u001b[0m     retval_ \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(posterior)\u001b[38;5;241m.\u001b[39mfused_predict_f, (ag__\u001b[38;5;241m.\u001b[39mld(Xnew),), \u001b[38;5;28mdict\u001b[39m(full_cov\u001b[38;5;241m=\u001b[39mag__\u001b[38;5;241m.\u001b[39mld(full_cov), full_output_cov\u001b[38;5;241m=\u001b[39mag__\u001b[38;5;241m.\u001b[39mld(full_output_cov)), fscope)\n\u001b[1;32m     41\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m:\n\u001b[1;32m     42\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_fileyazwcc6d.py:13\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf__wrapped_method\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m     11\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m     12\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[0;32m---> 13\u001b[0m     retval_ \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(wrapped_function), (ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m),) \u001b[38;5;241m+\u001b[39m \u001b[38;5;28mtuple\u001b[39m(ag__\u001b[38;5;241m.\u001b[39mld(args)), \u001b[38;5;28mdict\u001b[39m(\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mag__\u001b[38;5;241m.\u001b[39mld(kwargs)), fscope)\n\u001b[1;32m     14\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m:\n\u001b[1;32m     15\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_fileosceii7d.py:40\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf__wrapped_function\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m     38\u001b[0m     \u001b[38;5;28;01mnonlocal\u001b[39;00m do_return, retval_\n\u001b[1;32m     39\u001b[0m     \u001b[38;5;28;01mpass\u001b[39;00m\n\u001b[0;32m---> 40\u001b[0m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mif_stmt\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mnot_\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconverted_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mget_enable_check_shapes\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mNone\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfscope\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mif_body\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43melse_body\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mget_state\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mset_state\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mdo_return\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mretval_\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m2\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m     42\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mget_state_8\u001b[39m():\n\u001b[1;32m     43\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m (arg_map, ag__\u001b[38;5;241m.\u001b[39mldu(\u001b[38;5;28;01mlambda\u001b[39;00m: arg_map[RESULT_TOKEN], \u001b[38;5;124m'\u001b[39m\u001b[38;5;124marg_map[RESULT_TOKEN]\u001b[39m\u001b[38;5;124m'\u001b[39m), checker, do_return, retval_)\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_fileoixhtcmk.py:14\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf__fused_predict_f\u001b[0;34m(self, Xnew, full_cov, full_output_cov)\u001b[0m\n\u001b[1;32m     12\u001b[0m do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n\u001b[1;32m     13\u001b[0m retval_ \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mUndefinedReturnValue()\n\u001b[0;32m---> 14\u001b[0m mean, cov \u001b[38;5;241m=\u001b[39m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconverted_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_conditional_fused\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mXnew\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mdict\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mfull_cov\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfull_cov\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfull_output_cov\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfull_output_cov\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfscope\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     15\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m     16\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_fileyazwcc6d.py:13\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf__wrapped_method\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m     11\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m     12\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[0;32m---> 13\u001b[0m     retval_ \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(wrapped_function), (ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m),) \u001b[38;5;241m+\u001b[39m \u001b[38;5;28mtuple\u001b[39m(ag__\u001b[38;5;241m.\u001b[39mld(args)), \u001b[38;5;28mdict\u001b[39m(\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mag__\u001b[38;5;241m.\u001b[39mld(kwargs)), fscope)\n\u001b[1;32m     14\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m:\n\u001b[1;32m     15\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_fileosceii7d.py:40\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf__wrapped_function\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m     38\u001b[0m     \u001b[38;5;28;01mnonlocal\u001b[39;00m do_return, retval_\n\u001b[1;32m     39\u001b[0m     \u001b[38;5;28;01mpass\u001b[39;00m\n\u001b[0;32m---> 40\u001b[0m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mif_stmt\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mnot_\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconverted_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mget_enable_check_shapes\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mNone\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfscope\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mif_body\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43melse_body\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mget_state\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mset_state\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mdo_return\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mretval_\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m2\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m     42\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mget_state_8\u001b[39m():\n\u001b[1;32m     43\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m (arg_map, ag__\u001b[38;5;241m.\u001b[39mldu(\u001b[38;5;28;01mlambda\u001b[39;00m: arg_map[RESULT_TOKEN], \u001b[38;5;124m'\u001b[39m\u001b[38;5;124marg_map[RESULT_TOKEN]\u001b[39m\u001b[38;5;124m'\u001b[39m), checker, do_return, retval_)\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_file78q4qcjy.py:10\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf___conditional_fused\u001b[0;34m(self, Xnew, full_cov, full_output_cov)\u001b[0m\n\u001b[1;32m      8\u001b[0m do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n\u001b[1;32m      9\u001b[0m retval_ \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mUndefinedReturnValue()\n\u001b[0;32m---> 10\u001b[0m Knn \u001b[38;5;241m=\u001b[39m \u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconverted_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mkernel\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mXnew\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mdict\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mfull_cov\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mag__\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mld\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfull_cov\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfscope\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     11\u001b[0m Kmm \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(covariances)\u001b[38;5;241m.\u001b[39mKuu, (ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mX_data, ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mkernel), \u001b[38;5;28mdict\u001b[39m(jitter\u001b[38;5;241m=\u001b[39mag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(default_jitter), (), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)), fscope)\n\u001b[1;32m     12\u001b[0m Kmn \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(covariances)\u001b[38;5;241m.\u001b[39mKuf, (ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mX_data, ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mkernel, ag__\u001b[38;5;241m.\u001b[39mld(Xnew)), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)\n",
      "File \u001b[0;32m/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/__autograph_generated_filei0lm38ga.py:12\u001b[0m, in \u001b[0;36mouter_factory.<locals>.inner_factory.<locals>.tf____call__\u001b[0;34m(self, X, X2, full_cov, presliced)\u001b[0m\n\u001b[1;32m     10\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m     11\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[0;32m---> 12\u001b[0m     retval_ \u001b[38;5;241m=\u001b[39m ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39m_reduce, ([ag__\u001b[38;5;241m.\u001b[39mconverted_call(ag__\u001b[38;5;241m.\u001b[39mld(k), (ag__\u001b[38;5;241m.\u001b[39mld(X), ag__\u001b[38;5;241m.\u001b[39mld(X2)), \u001b[38;5;28mdict\u001b[39m(full_cov\u001b[38;5;241m=\u001b[39mag__\u001b[38;5;241m.\u001b[39mld(full_cov), presliced\u001b[38;5;241m=\u001b[39mag__\u001b[38;5;241m.\u001b[39mld(presliced)), fscope) \u001b[38;5;28;01mfor\u001b[39;00m k \u001b[38;5;129;01min\u001b[39;00m ag__\u001b[38;5;241m.\u001b[39mld(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39mkernels],), \u001b[38;5;28;01mNone\u001b[39;00m, fscope)\n\u001b[1;32m     13\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m:\n\u001b[1;32m     14\u001b[0m     do_return \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n",
      "\u001b[0;31mValueError\u001b[0m: in user code:\n\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/optimizers/scipy.py\", line 192, in _tf_eval  *\n        loss, grads = _compute_loss_and_gradients(\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/optimizers/scipy.py\", line 329, in _compute_loss_and_gradients  *\n        loss = loss_closure()\n    File \"/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/ipykernel_93113/2086292548.py\", line 138, in _build_likelihood  *\n        fmeans, fvars = self._build_predict(self.X, full_cov=False, full_output_cov=False)\n    File \"/var/folders/m_/mrnp6k153dq_mxbk9450fv5c0000gn/T/ipykernel_93113/2086292548.py\", line 170, in _build_predict  *\n        mu_tmp, var_tmp = conditional(Xnew, self.feature, self.kern, self.q_mu, q_sqrt=self.q_sqrt,\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/multipledispatch/dispatcher.py\", line 279, in __call__  *\n        return func(*args, **kwargs)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/decorator.py\", line 53, in wrapped_function  *\n        if not get_enable_check_shapes():\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/conditionals/conditionals.py\", line 87, in _sparse_conditional  *\n        return posterior.fused_predict_f(Xnew, full_cov=full_cov, full_output_cov=full_output_cov)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/integration/tf.py\", line 241, in wrapped_method  *\n        return wrapped_function(self, *args, **kwargs)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/decorator.py\", line 53, in wrapped_function  *\n        if not get_enable_check_shapes():\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/posteriors.py\", line 255, in fused_predict_f  *\n        mean, cov = self._conditional_fused(\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/integration/tf.py\", line 241, in wrapped_method  *\n        return wrapped_function(self, *args, **kwargs)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/check_shapes/decorator.py\", line 53, in wrapped_function  *\n        if not get_enable_check_shapes():\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/posteriors.py\", line 833, in _conditional_fused  *\n        Knn = self.kernel(Xnew, full_cov=full_cov)\n    File \"/Users/fujisawa/.pyenv/versions/3.11.6/lib/python3.11/site-packages/gpflow/kernels/base.py\", line 290, in __call__  *\n        [k(X, X2, full_cov=full_cov, presliced=presliced) for k in self.kernels]\n\n    ValueError: Shapes must be equal rank, but are 1 and 0\n    \tFrom merging shape 0 with other shapes. for '{{node AddN}} = AddN[N=2, T=DT_DOUBLE](Fill, Fill_1)' with input shapes: [32000], [].\n"
     ]
    }
   ],
   "source": [
    "recal_f.fit(p_pred_train, y_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dbfd5446-0025-4a23-ac09-de7dbefbfc5e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "418798b9-c0b3-4053-8338-724786e80634",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjcAAAHHCAYAAABDUnkqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/SrBM8AAAACXBIWXMAAA9hAAAPYQGoP6dpAABGFElEQVR4nO3dd3wUdf7H8ffsJrubQgIRSCiBIBY6KCUiIiKRiIh6NqyEqOgpoILnnYiAYonogaigqKdw550/EbuCtAg2UBDEDoiAIJoAAgkkkLL7/f2xycqSQoAkC8Pr+XjsAzL7nZnPzO7MvPOdEssYYwQAAGATjlAXAAAAUJ0INwAAwFYINwAAwFYINwAAwFYINwAAwFYINwAAwFYINwAAwFYINwAAwFYINwAAwFYIN8BRzrIs3X///aEuI6T27Nmjm266SQkJCbIsS3feeWeoSwqYMWOGLMvSxo0bQzL/jRs3yrIszZgxI2j43Llz1alTJ3k8HlmWpV27dmnw4MFKSkqq9RoXL14sy7K0ePHiWp83jk+EGxwXvv32W11++eVq3ry5PB6PmjRpovPOO09PP/10qEurdUlJSbIsS5ZlyeFwqG7dumrfvr1uvvlmffHFF0c07UceeURvv/129RR6wHRnzJihW2+9VS+//LKuv/76ap/Hgbxer6ZPn65zzjlHcXFxcrvdSkpKUnp6ur788ssan/+R+OOPP3TllVcqIiJCU6dO1csvv6yoqKgan+8zzzxTJmQBoWDxt6Vgd0uWLFHv3r3VrFkzpaWlKSEhQZs3b9bnn3+un3/+WevWrQt1iZWyLEvjxo2rtt6bpKQk1atXT3fddZckaffu3frxxx81a9YsZWVlacSIEZo0adJhTTs6OlqXX355tR/gzjjjDIWFhenTTz+t1ulWZO/evbr00ks1d+5cnX322RowYIDi4uK0ceNGvfbaa1q7dq02bdqkpk2basaMGUpPT9eGDRtC0itijFFBQYHCw8PldDol+Xtt+vXrpwULFiglJSXQtqioSD6fT263u0ZqadeunerXr1+mh8bn86mwsFAul0sOB79To+aFhboAoKY9/PDDio2N1fLly1W3bt2g97Zu3RqaokKsSZMmuu6664KGTZgwQddcc42eeOIJnXzyybr11ltDVF1ZW7duVZs2baptesXFxfL5fHK5XOW+f/fdd2vu3Ll64oknypwCGzdunJ544olqq+VIWZYlj8cTNKz0e33g9z08PLy2ygricDjK1AjUKAPY3KmnnmrOOeecKrV96aWXTO/evU2DBg2My+UyrVu3Ns8880yZds2bNzf9+/c3ixYtMp07dzYej8e0a9fOLFq0yBhjzBtvvGHatWtn3G63Of30083KlSuDxk9LSzNRUVHm559/Nn379jWRkZGmUaNG5oEHHjA+ny+orSQzbty4oGG//vqrSU9PNw0bNjQul8u0adPGvPjii1VaxtLay7N7924TFxdnmjRpElTH448/brp3727i4uKMx+Mxp59+upk1a1aZOg98paWlGWOM2bhxo7n11lvNKaecYjwej4mLizOXX3652bBhQ6W1Llq0qNzplo6XnZ1tbrjhBtOwYUPjdrtNhw4dzIwZM4KmsWHDBiPJPP744+aJJ54wJ554onE4HOarr74qd56bN282YWFh5rzzzqu0tlLTp08PqskYY95++21zwQUXmEaNGhmXy2VOPPFEM378eFNcXBw07tq1a82ll15q4uPjjdvtNk2aNDEDBw40u3btCrSZP3++6dGjh4mNjTVRUVHmlFNOMaNGjSqzfNOnTzfGGNOrV68KP4e0tDTTvHnzoBq8Xq+ZPHly4Ptav359k5qaapYvXx5oU5Xtonnz5mXm26tXL2PMn59j6fZR6rXXXjOnn3668Xg85oQTTjDXXnut+fXXX4PalG4rv/76q7n44otNVFSUqV+/vrnrrrvKrE+gFD03sL3mzZtr6dKl+u6779SuXbtK2z777LNq27atLrroIoWFhem9997TbbfdJp/Pp6FDhwa1Xbduna655hrdcsstuu666/TPf/5TAwYM0LRp03TvvffqtttukyRlZGToyiuv1Jo1a4K65L1er84//3ydccYZeuyxxzR37lyNGzdOxcXFGj9+fIU1Zmdn64wzzpBlWRo2bJgaNGigDz74QDfeeKNyc3OP6GLb6Oho/eUvf9GLL76oH374QW3btpUkPfnkk7rooot07bXXqrCwUK+++qquuOIKvf/+++rfv78k6eWXX9ZNN92kbt266eabb5YktWzZUpK0fPlyLVmyRFdddZWaNm2qjRs36tlnn9U555yjH374QZGRkeXW07p1a7388ssaMWKEmjZtGjiV1qBBA+3du1fnnHOO1q1bp2HDhqlFixaaNWuWBg8erF27dumOO+4Imtb06dO1b98+3XzzzXK73YqLiyt3nh988IGKi4uP6LqeGTNmKDo6WiNHjlR0dLQ+/PBDjR07Vrm5uXr88cclSYWFhUpNTVVBQYGGDx+uhIQEbdmyRe+//7527dql2NhYff/997rwwgvVoUMHjR8/Xm63W+vWrdNnn31W4bxHjx6tU089Vc8//7zGjx+vFi1aBD6H8tx4442aMWOG+vXrp5tuuknFxcX65JNP9Pnnn6tLly6SqrZdTJ48WcOHD1d0dLRGjx4tSYqPj690HaWnp6tr167KyMhQdna2nnzySX322Wf66quvgnqdvF6vUlNTlZycrH/+859auHChJk6cqJYtWx5VPYw4ioQ6XQE1bf78+cbpdBqn02m6d+9u/v73v5t58+aZwsLCMm3z8/PLDEtNTTUnnnhi0LDS31KXLFkSGDZv3jwjyURERJhffvklMPy5554r81trWlqakWSGDx8eGObz+Uz//v2Ny+Uy27ZtCwzXAT03N954o2nUqJHZvn17UE1XXXWViY2NLXcZDqy9op4bY4x54oknjCTzzjvvBIYdOM3CwkLTrl07c+655wYNj4qKCvQS7K+8mpYuXWokmf/85z+V1ltRzZMnTzaSzH//+9+gurp3726io6NNbm6uMebPno2YmBizdevWg85rxIgRRlKFPTsHKq/nprzlveWWW0xkZKTZt2+fMcaYr776ykgq0wO2v9LPYv/vw4EO7LnZv6b9e1+MKdtz8+GHHxpJ5vbbby8z3f177qq6XbRt2zbQW7O/A3tuCgsLTcOGDU27du3M3r17A+3ef/99I8mMHTs2qGZJZvz48UHTPO2000znzp3LzAswxhiu7ILtnXfeeVq6dKkuuugiff3113rssceUmpqqJk2a6N133w1qGxEREfh/Tk6Otm/frl69emn9+vXKyckJatumTRt179498HNycrIk6dxzz1WzZs3KDF+/fn2Z2oYNGxb4f2lPTGFhoRYuXFjushhj9MYbb2jAgAEyxmj79u2BV2pqqnJycrRy5cqqrppyRUdHS/JfaFxq//Wyc+dO5eTkqGfPnlWe1/7jFxUV6Y8//tBJJ52kunXrHna9c+bMUUJCgq6++urAsPDwcN1+++3as2ePPvroo6D2l112mRo0aHDQ6ebm5kqS6tSpc1h1ScHLu3v3bm3fvl09e/ZUfn6+Vq9eLUmKjY2VJM2bN0/5+fnlTqe09+Kdd96Rz+c77Hoq8sYbbwQuWD+QZVmB/x/KdlEVX375pbZu3arbbrst6Fqc/v37q1WrVpo9e3aZcf76178G/dyzZ89ytylA4lZwHCe6du2qN998Uzt37tSyZcs0atQo7d69W5dffrl++OGHQLvPPvtMKSkpioqKUt26ddWgQQPde++9klRmJ75/gJH+PFglJiaWO3znzp1Bwx0Oh0488cSgYaeccookVfjMlG3btmnXrl16/vnn1aBBg6BXenq6pCO/SHrPnj2Sgg/u77//vs444wx5PB7FxcWpQYMGevbZZ6t8YNu7d6/Gjh2rxMREud1u1a9fXw0aNNCuXbsO6+AoSb/88otOPvnkMnfftG7dOvD+/lq0aFGl6cbExEgKDneH6vvvv9df/vIXxcbGKiYmRg0aNAhcwF26vC1atNDIkSP1r3/9S/Xr11dqaqqmTp0atD4GDhyoHj166KabblJ8fLyuuuoqvfbaa9UWdH7++Wc1bty4wlN0pQ5lu6iK0s/m1FNPLfNeq1atynx2Ho+nTDCtV69emW0KKMU1NziuuFwude3aVV27dtUpp5yi9PR0zZo1S+PGjdPPP/+sPn36qFWrVpo0aZISExPlcrk0Z84cPfHEE2UOKKW33R6oouGmGp66UFrDddddp7S0tHLbdOjQ4Yjm8d1330mSTjrpJEnSJ598oosuukhnn322nnnmGTVq1Ejh4eGaPn26XnnllSpNc/jw4Zo+fbruvPNOde/eXbGxsbIsS1dddVWN9EiUZ//eh8q0atVKkv/ZSJ06dTrk+ezatUu9evVSTEyMxo8fr5YtW8rj8WjlypX6xz/+EbS8EydO1ODBg/XOO+9o/vz5uv3225WRkaHPP/9cTZs2VUREhD7++GMtWrRIs2fP1ty5czVz5kyde+65mj9/foXftep0qNtFTaiN5YS9EG5w3Cq9WPL333+XJL333nsqKCjQu+++G9Qrs2jRohqZv8/n0/r16wO9NZK0du1aSarweSkNGjRQnTp15PV6g55fUl327Nmjt956S4mJiYEekDfeeEMej0fz5s0Lej7K9OnTy4y//6mM/b3++utKS0vTxIkTA8P27dunXbt2HXatzZs31zfffCOfzxfUe1N62qd58+aHNd1+/frJ6XTqv//972FdVLx48WL98ccfevPNN3X22WcHhm/YsKHc9u3bt1f79u113333acmSJerRo4emTZumhx56SJK/h69Pnz7q06ePJk2apEceeUSjR4/WokWLjvg70LJlS82bN087duyosPfmULaLij7/A5V+NmvWrNG5554b9N6aNWsO+7MDSnFaCra3aNGicntN5syZI+nPrvHS3w73b5uTk1PuQby6TJkyJfB/Y4ymTJmi8PBw9enTp9z2TqdTl112md54441AD8v+tm3bdti17N27V9dff7127Nih0aNHBw5UTqdTlmXJ6/UG2m7cuLHcJxFHRUWVG1icTmeZz+Dpp58OmuahuuCCC5SVlaWZM2cGhhUXF+vpp59WdHS0evXqdVjTTUxM1JAhQzR//vxyn2Dt8/k0ceJE/frrr+WOX973qLCwUM8880xQu9zcXBUXFwcNa9++vRwOhwoKCiRJO3bsKDP90t6k0jZH4rLLLpMxRg888ECZ90rrP5TtoqLP/0BdunRRw4YNNW3atKDl+OCDD/Tjjz8G7sADDhc9N7C94cOHKz8/X3/5y1/UqlUrFRYWasmSJZo5c2bgcfqS1LdvX7lcLg0YMEC33HKL9uzZoxdeeEENGzYM9O5UJ4/Ho7lz5yotLU3Jycn64IMPNHv2bN17772VXvj66KOPatGiRUpOTtaQIUPUpk0b7dixQytXrtTChQvLPSAeaMuWLfrvf/8ryd9b88MPPwSeUHzXXXfplltuCbTt37+/Jk2apPPPP1/XXHONtm7dqqlTp+qkk07SN998EzTdzp07a+HChZo0aZIaN26sFi1aKDk5WRdeeKFefvllxcbGqk2bNlq6dKkWLlyoE0444TDXnnTzzTfrueee0+DBg7VixQolJSXp9ddf12effabJkycf0QXBEydO1M8//6zbb79db775pi688ELVq1dPmzZt0qxZs7R69WpdddVV5Y575plnql69ekpLS9Ptt98uy7L08ssvlwl3H374oYYNG6YrrrhCp5xyioqLi/Xyyy8HAqwkjR8/Xh9//LH69++v5s2ba+vWrXrmmWfUtGlTnXXWWYe9fKV69+6t66+/Xk899ZR++uknnX/++fL5fPrkk0/Uu3dvDRs27JC2i86dO+vZZ5/VQw89pJNOOkkNGzYs0zMj+S/8njBhgtLT09WrVy9dffXVgVvBk5KSNGLEiCNeNhzn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wlrLihzo3+bKa1+4yv9d8/4sWTGyPBdIEf1lWZ4/2xgjFX3l/9571/vbRVwoeS6QZbmDpucr3ibteUIq+NA/f7kkR4zkbOh/iF/kQFnOBPm8u6U9k6R975U8UsIhOZpJ0bfKijhfllVx+DHF60q292/8+5zw3pJve8lDPvfI3+vhkZydpejLpeKN/v2bfP6H+EVcJSusacm0NvgfJlq0yr9Nlt6NZXn8D+fz5UiW03/aL+JK//t7Z8nsmyd5t0qWQ3I29T+Y1HPRQbdRU/yz/061wk9Kph0tubrKihokK7z9AW03B++7nY0k3y7/Rd6OprIir/CfujM7pfzXZQo+klQshZ9e0pNWXLKeVkmWa7/9Yl3/9L1b9tuXOUqW8QpZzoaVLkOoHcrxO6ThZvHixerdu3eZ4WlpaZoxY4YGDx6sjRs3avHixUHjjBgxQj/88IOaNm2qMWPGhOwhfgAAoHYcM+EmFAg3AAAce46ZJxQDAABUN8INAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwlZCHm6lTpyopKUkej0fJyclatmxZpe0nT56sU089VREREUpMTNSIESO0b9++WqoWAAAc7UIabmbOnKmRI0dq3LhxWrlypTp27KjU1FRt3bq13PavvPKK7rnnHo0bN04//vijXnzxRc2cOVP33ntvLVcOAACOViENN5MmTdKQIUOUnp6uNm3aaNq0aYqMjNRLL71UbvslS5aoR48euuaaa5SUlKS+ffvq6quvPmhvDwAAOH6ELNwUFhZqxYoVSklJ+bMYh0Mp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ddfphBNOUEREhNq3b68vv/wy1GXZjtfr1ZgxY9SiRQtFRESoZcuWevDBB6v094pQNR9//LEGDBigxo0by7Isvf3220HvG2M0duxYNWrUSBEREUpJSdFPP/1UrTUQbo7AzJkzNXLkSI0bN04rV65Ux44dlZqaqq1bt4a6NNv56KOPNHToUH3++edasGCBioqK1LdvX+Xl5YW6NFtbvny5nnvuOXXo0CHUpdjazp071aNHD4WHh+uDDz7QDz/8oIkTJ6pevXqhLs12JkyYoGeffVZTpkzRjz/+qAkTJuixxx7T008/HerSbCMvL08dO3bU1KlTy33/scce01NPPaVp06bpiy++UFRUlFJTU7Vv377qK8LgsHXr1s0MHTo08LPX6zWNGzc2GRkZIazq+LB161YjyXz00UehLsW2du/ebU4++WSzYMEC06tXL3PHHXeEuiTb+sc//mHOOuusUJdxXOjfv7+54YYbgoZdeuml5tprrw1RRfYmybz11luBn30+n0lISDCPP/54YNiuXbuM2+02//d//1dt86Xn5jAVFhZqxYoVSklJCQxzOBxKSUnR0qVLQ1jZ8SEnJ0eSFBcXF+JK7Gvo0KHq379/0HccNePdd99Vly5ddMUVV6hhw4Y67bTT9MILL4S6LFs688wzlZmZqbVr10qSvv76a3366afq169fiCs7PmzYsEFZWVlB+5XY2FglJydX67HzuPvDmdVl+/bt8nq9io+PDxoeHx+v1atXh6iq44PP59Odd96pHj16qF27dqEux5ZeffVVrVy5UsuXLw91KceF9evX69lnn9XIkSN17733avny5br99tvlcrmUlpYW6vJs5Z577lFubq5atWolp9Mpr9erhx9+WNdee22oSzsuZGVlSVK5x87S96oD4QbHnKFDh+q7777Tp59+GupSbGnz5s264447tGDBAnk8nlCXc1zw+Xzq0qWLHnnkEUnSaaedpu+++07Tpk0j3FSz1157Tf/73//0yiuvqG3btlq1apXuvPNONW7cmHVtI5yWOkz169eX0+lUdnZ20PDs7GwlJCSEqCr7GzZsmN5//30tWrRITZs2DXU5trRixQpt3bpVp59+usLCwhQWFqaPPvpITz31lMLCwuT1ekNdou00atRIbdq0CRrWunVrbdq0KUQV2dfdd9+te+65R1dddZXat2+v66+/XiNGjFBGRkaoSzsulB4fa/rYSbg5TC6XS507d1ZmZmZgmM/nU2Zmprp37x7CyuzJGKNhw4bprbfe0ocffqgWLVqEuiTb6tOnj7799lutWrUq8OrSpYuuvfZarVq1Sk6nM9Ql2k6PHj3KPNpg7dq1at68eYgqsq/8/Hw5HMGHPqfTKZ/PF6KKji8tWrRQQkJC0LEzNzdXX3zxRbUeOzktdQRGjhyptLQ0denSRd26ddPkyZOVl5en9PT0UJdmO0OHDtUrr7yid955R3Xq1Amcm42NjVVERESIq7OXOnXqlLmWKSoqSieccALXONWQESNG6Mwzz9QjjzyiK6+8UsuWLdPzzz+v559/PtSl2c6AAQP08MMPq1mzZmrbtq2++uorTZo0STfccEOoS7ONPXv2aN26dYGfN2zYoFWrVikuLk7NmjXTnXfeqYceekgnn3yyWrRooTFjxqhx48a65JJLqq+Iarvv6jj19NNPm2bNmhmXy2W6detmPv/881CXZEuSyn1Nnz491KUdF7gVvOa99957pl27dsbtdptWrVqZ559/PtQl2VJubq654447TLNmzYzH4zEnnniiGT16tCkoKAh1abaxaNGicvfXaWlpxhj/7eBjxowx8fHxxu12mz59+pg1a9ZUaw2WMTyWEQAA2AfX3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AA4pnm9Xp155pm69NJLg4bn5OQoMTFRo0ePDlFlAEKFJxQDOOatXbtWnTp10gsvvKBrr71WkjRo0CB9/fXXWr58uVwuV4grBFCbCDcAbOGpp57S/fffr++//17Lli3TFVdcoeXLl6tjx46hLg1ALSPcALAFY4zOPfdcOZ1Offvttxo+fLjuu+++UJcFIAQINwBsY/Xq1WrdurXat2+vlStXKiwsLNQlAQgBLigGYBsvvfSSIiMjtWHDBv3666+hLgdAiNBzA8AWlixZol69emn+/Pl66KGHJEkLFy6UZVkhrgxAbaPnBsAxLz8/X4MHD9att96q3r1768UXX9SyZcs0bdq0UJcGIATouQFwzLvjjjs0Z84cff3114qMjJQkPffcc/rb3/6mb7/9VklJSaEtEECtItwAOKZ99NFH6tOnjxYvXqyzzjor6L3U1FQVFxdzego4zhBuAACArXDNDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsJX/B8mK1g1zABOOAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import gpflow\n",
    "import tensorflow as tf\n",
    "from gpflow.utilities import print_summary, set_trainable\n",
    "\n",
    "# バイナリ分類のためのトイデータ生成\n",
    "np.random.seed(42)\n",
    "X = np.random.rand(100, 1) * 10  # 0から10の範囲の100個のサンプル\n",
    "Y = (np.sin(X) + np.random.randn(100, 1) * 0.1 > 0).astype(int)  # sin関数に基づくバイナリラベル\n",
    "\n",
    "# プロットして確認\n",
    "plt.scatter(X, Y, c=Y, cmap='viridis')\n",
    "plt.title(\"Sample Data for Classification\")\n",
    "plt.xlabel(\"X\")\n",
    "plt.ylabel(\"Y\")\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 165,
   "id": "e7fd7650-2f9e-4f67-95b4-670329e8a127",
   "metadata": {},
   "outputs": [],
   "source": [
    "import tensorflow_probability as tfp\n",
    "class BoundedPositiveBijector(tfp.bijectors.Bijector):\n",
    "    def __init__(self, lower, upper, validate_args=False, name=\"bounded_positive\"):\n",
    "        super().__init__(forward_min_event_ndims=0, validate_args=validate_args, name=name)\n",
    "        self.lower = tf.cast(lower, tf.float64)\n",
    "        self.upper = tf.cast(upper, tf.float64)\n",
    "\n",
    "    def _forward(self, x):\n",
    "        # [0, 1]にスケーリングされた値\n",
    "        scaled_x = tf.sigmoid(x)\n",
    "        # [lower, upper]の範囲に変換\n",
    "        return self.lower + (self.upper - self.lower) * scaled_x\n",
    "\n",
    "    def _inverse(self, y):\n",
    "        # yを[0, 1]に変換\n",
    "        scaled_y = (y - self.lower) / (self.upper - self.lower)\n",
    "        # sigmoidの逆関数はlogit\n",
    "        return tf.math.log(scaled_y / (1 - scaled_y))\n",
    "\n",
    "    def _forward_log_det_jacobian(self, x):\n",
    "        return tf.constant(0., dtype=x.dtype)\n",
    "\n",
    "k_white = gpflow.kernels.White(variance=0.01)\n",
    "\n",
    "kernel_lengthscale = 0.5\n",
    "k_rbf = gpflow.kernels.RBF(lengthscales=kernel_lengthscale, variance=1)\n",
    "                \n",
    "# Place constraints [a,b] on kernel parameters\n",
    "transform_lengthscale = BoundedPositiveBijector(.001, 10)\n",
    "transform_variance = BoundedPositiveBijector(0.01, 5)\n",
    "                \n",
    "k_rbf.lengthscales = gpflow.Parameter(kernel_lengthscale, transform=transform_lengthscale)\n",
    "k_rbf.variance = gpflow.Parameter(1, transform=transform_variance)\n",
    "kernel = k_rbf + k_white"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 166,
   "id": "7ce07636-c2ec-407d-ab64-2a18e963baf9",
   "metadata": {},
   "outputs": [],
   "source": [
    "class LogMeanFunction(gpflow.mean_functions.MeanFunction):\n",
    "    def __call__(self, X):\n",
    "        X = tf.cast(X, dtype=tf.float64) \n",
    "        return tf.math.log(X + 1e-6)  # 微小値を加えてゼロ除算を防ぐ"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d03faef9-e201-44fc-9f9f-2787e8823c81",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "id": "71e3c832-b6b4-4d8c-9418-9da667d09f4e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjcAAAHHCAYAAABDUnkqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/SrBM8AAAACXBIWXMAAA9hAAAPYQGoP6dpAABGFElEQVR4nO3dd3wUdf7H8ffsJrubQgIRSCiBIBY6KCUiIiKRiIh6NqyEqOgpoILnnYiAYonogaigqKdw550/EbuCtAg2UBDEDoiAIJoAAgkkkLL7/f2xycqSQoAkC8Pr+XjsAzL7nZnPzO7MvPOdEssYYwQAAGATjlAXAAAAUJ0INwAAwFYINwAAwFYINwAAwFYINwAAwFYINwAAwFYINwAAwFYINwAAwFYINwAAwFYIN8BRzrIs3X///aEuI6T27Nmjm266SQkJCbIsS3feeWeoSwqYMWOGLMvSxo0bQzL/jRs3yrIszZgxI2j43Llz1alTJ3k8HlmWpV27dmnw4MFKSkqq9RoXL14sy7K0ePHiWp83jk+EGxwXvv32W11++eVq3ry5PB6PmjRpovPOO09PP/10qEurdUlJSbIsS5ZlyeFwqG7dumrfvr1uvvlmffHFF0c07UceeURvv/129RR6wHRnzJihW2+9VS+//LKuv/76ap/Hgbxer6ZPn65zzjlHcXFxcrvdSkpKUnp6ur788ssan/+R+OOPP3TllVcqIiJCU6dO1csvv6yoqKgan+8zzzxTJmQBoWDxt6Vgd0uWLFHv3r3VrFkzpaWlKSEhQZs3b9bnn3+un3/+WevWrQt1iZWyLEvjxo2rtt6bpKQk1atXT3fddZckaffu3frxxx81a9YsZWVlacSIEZo0adJhTTs6OlqXX355tR/gzjjjDIWFhenTTz+t1ulWZO/evbr00ks1d+5cnX322RowYIDi4uK0ceNGvfbaa1q7dq02bdqkpk2basaMGUpPT9eGDRtC0itijFFBQYHCw8PldDol+Xtt+vXrpwULFiglJSXQtqioSD6fT263u0ZqadeunerXr1+mh8bn86mwsFAul0sOB79To+aFhboAoKY9/PDDio2N1fLly1W3bt2g97Zu3RqaokKsSZMmuu6664KGTZgwQddcc42eeOIJnXzyybr11ltDVF1ZW7duVZs2baptesXFxfL5fHK5XOW+f/fdd2vu3Ll64oknypwCGzdunJ544olqq+VIWZYlj8cTNKz0e33g9z08PLy2ygricDjK1AjUKAPY3KmnnmrOOeecKrV96aWXTO/evU2DBg2My+UyrVu3Ns8880yZds2bNzf9+/c3ixYtMp07dzYej8e0a9fOLFq0yBhjzBtvvGHatWtn3G63Of30083KlSuDxk9LSzNRUVHm559/Nn379jWRkZGmUaNG5oEHHjA+ny+orSQzbty4oGG//vqrSU9PNw0bNjQul8u0adPGvPjii1VaxtLay7N7924TFxdnmjRpElTH448/brp3727i4uKMx+Mxp59+upk1a1aZOg98paWlGWOM2bhxo7n11lvNKaecYjwej4mLizOXX3652bBhQ6W1Llq0qNzplo6XnZ1tbrjhBtOwYUPjdrtNhw4dzIwZM4KmsWHDBiPJPP744+aJJ54wJ554onE4HOarr74qd56bN282YWFh5rzzzqu0tlLTp08PqskYY95++21zwQUXmEaNGhmXy2VOPPFEM378eFNcXBw07tq1a82ll15q4uPjjdvtNk2aNDEDBw40u3btCrSZP3++6dGjh4mNjTVRUVHmlFNOMaNGjSqzfNOnTzfGGNOrV68KP4e0tDTTvHnzoBq8Xq+ZPHly4Ptav359k5qaapYvXx5oU5Xtonnz5mXm26tXL2PMn59j6fZR6rXXXjOnn3668Xg85oQTTjDXXnut+fXXX4PalG4rv/76q7n44otNVFSUqV+/vrnrrrvKrE+gFD03sL3mzZtr6dKl+u6779SuXbtK2z777LNq27atLrroIoWFhem9997TbbfdJp/Pp6FDhwa1Xbduna655hrdcsstuu666/TPf/5TAwYM0LRp03TvvffqtttukyRlZGToyiuv1Jo1a4K65L1er84//3ydccYZeuyxxzR37lyNGzdOxcXFGj9+fIU1Zmdn64wzzpBlWRo2bJgaNGigDz74QDfeeKNyc3OP6GLb6Oho/eUvf9GLL76oH374QW3btpUkPfnkk7rooot07bXXqrCwUK+++qquuOIKvf/+++rfv78k6eWXX9ZNN92kbt266eabb5YktWzZUpK0fPlyLVmyRFdddZWaNm2qjRs36tlnn9U555yjH374QZGRkeXW07p1a7388ssaMWKEmjZtGjiV1qBBA+3du1fnnHOO1q1bp2HDhqlFixaaNWuWBg8erF27dumOO+4Imtb06dO1b98+3XzzzXK73YqLiyt3nh988IGKi4uP6LqeGTNmKDo6WiNHjlR0dLQ+/PBDjR07Vrm5uXr88cclSYWFhUpNTVVBQYGGDx+uhIQEbdmyRe+//7527dql2NhYff/997rwwgvVoUMHjR8/Xm63W+vWrdNnn31W4bxHjx6tU089Vc8//7zGjx+vFi1aBD6H8tx4442aMWOG+vXrp5tuuknFxcX65JNP9Pnnn6tLly6SqrZdTJ48WcOHD1d0dLRGjx4tSYqPj690HaWnp6tr167KyMhQdna2nnzySX322Wf66quvgnqdvF6vUlNTlZycrH/+859auHChJk6cqJYtWx5VPYw4ioQ6XQE1bf78+cbpdBqn02m6d+9u/v73v5t58+aZwsLCMm3z8/PLDEtNTTUnnnhi0LDS31KXLFkSGDZv3jwjyURERJhffvklMPy5554r81trWlqakWSGDx8eGObz+Uz//v2Ny+Uy27ZtCwzXAT03N954o2nUqJHZvn17UE1XXXWViY2NLXcZDqy9op4bY4x54oknjCTzzjvvBIYdOM3CwkLTrl07c+655wYNj4qKCvQS7K+8mpYuXWokmf/85z+V1ltRzZMnTzaSzH//+9+gurp3726io6NNbm6uMebPno2YmBizdevWg85rxIgRRlKFPTsHKq/nprzlveWWW0xkZKTZt2+fMcaYr776ykgq0wO2v9LPYv/vw4EO7LnZv6b9e1+MKdtz8+GHHxpJ5vbbby8z3f177qq6XbRt2zbQW7O/A3tuCgsLTcOGDU27du3M3r17A+3ef/99I8mMHTs2qGZJZvz48UHTPO2000znzp3LzAswxhiu7ILtnXfeeVq6dKkuuugiff3113rssceUmpqqJk2a6N133w1qGxEREfh/Tk6Otm/frl69emn9+vXKyckJatumTRt179498HNycrIk6dxzz1WzZs3KDF+/fn2Z2oYNGxb4f2lPTGFhoRYuXFjushhj9MYbb2jAgAEyxmj79u2BV2pqqnJycrRy5cqqrppyRUdHS/JfaFxq//Wyc+dO5eTkqGfPnlWe1/7jFxUV6Y8//tBJJ52kunXrHna9c+bMUUJCgq6++urAsPDwcN1+++3as2ePPvroo6D2l112mRo0aHDQ6ebm5kqS6tSpc1h1ScHLu3v3bm3fvl09e/ZUfn6+Vq9eLUmKjY2VJM2bN0/5+fnlTqe09+Kdd96Rz+c77Hoq8sYbbwQuWD+QZVmB/x/KdlEVX375pbZu3arbbrst6Fqc/v37q1WrVpo9e3aZcf76178G/dyzZ89ytylA4lZwHCe6du2qN998Uzt37tSyZcs0atQo7d69W5dffrl++OGHQLvPPvtMKSkpioqKUt26ddWgQQPde++9klRmJ75/gJH+PFglJiaWO3znzp1Bwx0Oh0488cSgYaeccookVfjMlG3btmnXrl16/vnn1aBBg6BXenq6pCO/SHrPnj2Sgg/u77//vs444wx5PB7FxcWpQYMGevbZZ6t8YNu7d6/Gjh2rxMREud1u1a9fXw0aNNCuXbsO6+AoSb/88otOPvnkMnfftG7dOvD+/lq0aFGl6cbExEgKDneH6vvvv9df/vIXxcbGKiYmRg0aNAhcwF26vC1atNDIkSP1r3/9S/Xr11dqaqqmTp0atD4GDhyoHj166KabblJ8fLyuuuoqvfbaa9UWdH7++Wc1bty4wlN0pQ5lu6iK0s/m1FNPLfNeq1atynx2Ho+nTDCtV69emW0KKMU1NziuuFwude3aVV27dtUpp5yi9PR0zZo1S+PGjdPPP/+sPn36qFWrVpo0aZISExPlcrk0Z84cPfHEE2UOKKW33R6oouGmGp66UFrDddddp7S0tHLbdOjQ4Yjm8d1330mSTjrpJEnSJ598oosuukhnn322nnnmGTVq1Ejh4eGaPn26XnnllSpNc/jw4Zo+fbruvPNOde/eXbGxsbIsS1dddVWN9EiUZ//eh8q0atVKkv/ZSJ06dTrk+ezatUu9evVSTEyMxo8fr5YtW8rj8WjlypX6xz/+EbS8EydO1ODBg/XOO+9o/vz5uv3225WRkaHPP/9cTZs2VUREhD7++GMtWrRIs2fP1ty5czVz5kyde+65mj9/foXftep0qNtFTaiN5YS9EG5w3Cq9WPL333+XJL333nsqKCjQu+++G9Qrs2jRohqZv8/n0/r16wO9NZK0du1aSarweSkNGjRQnTp15PV6g55fUl327Nmjt956S4mJiYEekDfeeEMej0fz5s0Lej7K9OnTy4y//6mM/b3++utKS0vTxIkTA8P27dunXbt2HXatzZs31zfffCOfzxfUe1N62qd58+aHNd1+/frJ6XTqv//972FdVLx48WL98ccfevPNN3X22WcHhm/YsKHc9u3bt1f79u113333acmSJerRo4emTZumhx56SJK/h69Pnz7q06ePJk2apEceeUSjR4/WokWLjvg70LJlS82bN087duyosPfmULaLij7/A5V+NmvWrNG5554b9N6aNWsO+7MDSnFaCra3aNGicntN5syZI+nPrvHS3w73b5uTk1PuQby6TJkyJfB/Y4ymTJmi8PBw9enTp9z2TqdTl112md54441AD8v+tm3bdti17N27V9dff7127Nih0aNHBw5UTqdTlmXJ6/UG2m7cuLHcJxFHRUWVG1icTmeZz+Dpp58OmuahuuCCC5SVlaWZM2cGhhUXF+vpp59WdHS0evXqdVjTTUxM1JAhQzR//vxyn2Dt8/k0ceJE/frrr+WOX973qLCwUM8880xQu9zcXBUXFwcNa9++vRwOhwoKCiRJO3bsKDP90t6k0jZH4rLLLpMxRg888ECZ90rrP5TtoqLP/0BdunRRw4YNNW3atKDl+OCDD/Tjjz8G7sADDhc9N7C94cOHKz8/X3/5y1/UqlUrFRYWasmSJZo5c2bgcfqS1LdvX7lcLg0YMEC33HKL9uzZoxdeeEENGzYM9O5UJ4/Ho7lz5yotLU3Jycn64IMPNHv2bN17772VXvj66KOPatGiRUpOTtaQIUPUpk0b7dixQytXrtTChQvLPSAeaMuWLfrvf/8ryd9b88MPPwSeUHzXXXfplltuCbTt37+/Jk2apPPPP1/XXHONtm7dqqlTp+qkk07SN998EzTdzp07a+HChZo0aZIaN26sFi1aKDk5WRdeeKFefvllxcbGqk2bNlq6dKkWLlyoE0444TDXnnTzzTfrueee0+DBg7VixQolJSXp9ddf12effabJkycf0QXBEydO1M8//6zbb79db775pi688ELVq1dPmzZt0qxZs7R69WpdddVV5Y575plnql69ekpLS9Ptt98uy7L08ssvlwl3H374oYYNG6YrrrhCp5xyioqLi/Xyyy8HAqwkjR8/Xh9//LH69++v5s2ba+vWrXrmmWfUtGlTnXXWWYe9fKV69+6t66+/Xk899ZR++uknnX/++fL5fPrkk0/Uu3dvDRs27JC2i86dO+vZZ5/VQw89pJNOOkkNGzYs0zMj+S/8njBhgtLT09WrVy9dffXVgVvBk5KSNGLEiCNeNhzn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wlrLihzo3+bKa1+4yv9d8/4sWTGyPBdIEf1lWZ4/2xgjFX3l/9571/vbRVwoeS6QZbmDpucr3ibteUIq+NA/f7kkR4zkbOh/iF/kQFnOBPm8u6U9k6R975U8UsIhOZpJ0bfKijhfllVx+DHF60q292/8+5zw3pJve8lDPvfI3+vhkZydpejLpeKN/v2bfP6H+EVcJSusacm0NvgfJlq0yr9Nlt6NZXn8D+fz5UiW03/aL+JK//t7Z8nsmyd5t0qWQ3I29T+Y1HPRQbdRU/yz/061wk9Kph0tubrKihokK7z9AW03B++7nY0k3y7/Rd6OprIir/CfujM7pfzXZQo+klQshZ9e0pNWXLKeVkmWa7/9Yl3/9L1b9tuXOUqW8QpZzoaVLkOoHcrxO6ThZvHixerdu3eZ4WlpaZoxY4YGDx6sjRs3avHixUHjjBgxQj/88IOaNm2qMWPGhOwhfgAAoHYcM+EmFAg3AAAce46ZJxQDAABUN8INAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwFcINAACwlZCHm6lTpyopKUkej0fJyclatmxZpe0nT56sU089VREREUpMTNSIESO0b9++WqoWAAAc7UIabmbOnKmRI0dq3LhxWrlypTp27KjU1FRt3bq13PavvPKK7rnnHo0bN04//vijXnzxRc2cOVP33ntvLVcOAACOViENN5MmTdKQIUOUnp6uNm3aaNq0aYqMjNRLL71UbvslS5aoR48euuaaa5SUlKS+ffvq6quvPmhvDwAAOH6ELNwUFhZqxYoVSklJ+bMYh0Mp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ddfphBNOUEREhNq3b68vv/wy1GXZjtfr1ZgxY9SiRQtFRESoZcuWevDBB6v094pQNR9//LEGDBigxo0by7Isvf3220HvG2M0duxYNWrUSBEREUpJSdFPP/1UrTUQbo7AzJkzNXLkSI0bN04rV65Ux44dlZqaqq1bt4a6NNv56KOPNHToUH3++edasGCBioqK1LdvX+Xl5YW6NFtbvny5nnvuOXXo0CHUpdjazp071aNHD4WHh+uDDz7QDz/8oIkTJ6pevXqhLs12JkyYoGeffVZTpkzRjz/+qAkTJuixxx7T008/HerSbCMvL08dO3bU1KlTy33/scce01NPPaVp06bpiy++UFRUlFJTU7Vv377qK8LgsHXr1s0MHTo08LPX6zWNGzc2GRkZIazq+LB161YjyXz00UehLsW2du/ebU4++WSzYMEC06tXL3PHHXeEuiTb+sc//mHOOuusUJdxXOjfv7+54YYbgoZdeuml5tprrw1RRfYmybz11luBn30+n0lISDCPP/54YNiuXbuM2+02//d//1dt86Xn5jAVFhZqxYoVSklJCQxzOBxKSUnR0qVLQ1jZ8SEnJ0eSFBcXF+JK7Gvo0KHq379/0HccNePdd99Vly5ddMUVV6hhw4Y67bTT9MILL4S6LFs688wzlZmZqbVr10qSvv76a3366afq169fiCs7PmzYsEFZWVlB+5XY2FglJydX67HzuPvDmdVl+/bt8nq9io+PDxoeHx+v1atXh6iq44PP59Odd96pHj16qF27dqEux5ZeffVVrVy5UsuXLw91KceF9evX69lnn9XIkSN17733avny5br99tvlcrmUlpYW6vJs5Z577lFubq5atWolp9Mpr9erhx9+WNdee22oSzsuZGVlSVK5x87S96oD4QbHnKFDh+q7777Tp59+GupSbGnz5s264447tGDBAnk8nlCXc1zw+Xzq0qWLHnnkEUnSaaedpu+++07Tpk0j3FSz1157Tf/73//0yiuvqG3btlq1apXuvPNONW7cmHVtI5yWOkz169eX0+lUdnZ20PDs7GwlJCSEqCr7GzZsmN5//30tWrRITZs2DXU5trRixQpt3bpVp59+usLCwhQWFqaPPvpITz31lMLCwuT1ekNdou00atRIbdq0CRrWunVrbdq0KUQV2dfdd9+te+65R1dddZXat2+v66+/XiNGjFBGRkaoSzsulB4fa/rYSbg5TC6XS507d1ZmZmZgmM/nU2Zmprp37x7CyuzJGKNhw4bprbfe0ocffqgWLVqEuiTb6tOnj7799lutWrUq8OrSpYuuvfZarVq1Sk6nM9Ql2k6PHj3KPNpg7dq1at68eYgqsq/8/Hw5HMGHPqfTKZ/PF6KKji8tWrRQQkJC0LEzNzdXX3zxRbUeOzktdQRGjhyptLQ0denSRd26ddPkyZOVl5en9PT0UJdmO0OHDtUrr7yid955R3Xq1Amcm42NjVVERESIq7OXOnXqlLmWKSoqSieccALXONWQESNG6Mwzz9QjjzyiK6+8UsuWLdPzzz+v559/PtSl2c6AAQP08MMPq1mzZmrbtq2++uorTZo0STfccEOoS7ONPXv2aN26dYGfN2zYoFWrVikuLk7NmjXTnXfeqYceekgnn3yyWrRooTFjxqhx48a65JJLqq+Iarvv6jj19NNPm2bNmhmXy2W6detmPv/881CXZEuSyn1Nnz491KUdF7gVvOa99957pl27dsbtdptWrVqZ559/PtQl2VJubq654447TLNmzYzH4zEnnniiGT16tCkoKAh1abaxaNGicvfXaWlpxhj/7eBjxowx8fHxxu12mz59+pg1a9ZUaw2WMTyWEQAA2AfX3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AAAAFsh3AA4pnm9Xp155pm69NJLg4bn5OQoMTFRo0ePDlFlAEKFJxQDOOatXbtWnTp10gsvvKBrr71WkjRo0CB9/fXXWr58uVwuV4grBFCbCDcAbOGpp57S/fffr++//17Lli3TFVdcoeXLl6tjx46hLg1ALSPcALAFY4zOPfdcOZ1Offvttxo+fLjuu+++UJcFIAQINwBsY/Xq1WrdurXat2+vlStXKiwsLNQlAQgBLigGYBsvvfSSIiMjtWHDBv3666+hLgdAiNBzA8AWlixZol69emn+/Pl66KGHJEkLFy6UZVkhrgxAbaPnBsAxLz8/X4MHD9att96q3r1768UXX9SyZcs0bdq0UJcGIATouQFwzLvjjjs0Z84cff3114qMjJQkPffcc/rb3/6mb7/9VklJSaEtEECtItwAOKZ99NFH6tOnjxYvXqyzzjor6L3U1FQVFxdzego4zhBuAACArXDNDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsBXCDQAAsJX/B8mK1g1zABOOAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1000x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import gpflow\n",
    "from gpflow.utilities import set_trainable\n",
    "\n",
    "class GPCalibration(CalibrationMethod):\n",
    "    def __init__(self, kernel=None, mean_function=None, model_type='SVGP', num_inducing=10):\n",
    "        self.kernel = kernel if kernel is not None else gpflow.kernels.SquaredExponential()\n",
    "        self.mean_function = mean_function if mean_function is not None else gpflow.mean_functions.Zero()\n",
    "        self.model_type = model_type\n",
    "        self.num_inducing = num_inducing\n",
    "        self.model = None\n",
    "\n",
    "    def fit(self, X, Y):\n",
    "        Y = Y.reshape(-1, 1)\n",
    "        # 多クラス分類のためのSVGPモデル\n",
    "        if self.model_type == 'SVGP':\n",
    "            num_classes = len(np.unique(Y))\n",
    "            inducing_variable = scipy.cluster.vq.kmeans(obs=X,\n",
    "                                    k_or_guess=min(X.shape[0] * X.shape[1], self.num_inducing, ))[0]\n",
    "            likelihood = gpflow.likelihoods.Softmax(num_classes)\n",
    "            self.model = gpflow.models.SVGP(kernel=self.kernel, likelihood=likelihood, inducing_variable=inducing_variable, mean_function=self.mean_function, num_latent_gps=num_classes, whiten=True, q_diag=True)\n",
    "            # 最適化\n",
    "            optimizer = gpflow.optimizers.Scipy()\n",
    "            #set_trainable(self.model.inducing_variable, False)  # インデューシングポイントを固定\n",
    "            data = (X, Y)\n",
    "            training_loss = self.model.training_loss_closure(data, compile=False)\n",
    "            optimizer.minimize(training_loss, variables=self.model.trainable_variables, options=dict(maxiter=1000))\n",
    "        \n",
    "        return self\n",
    "        \n",
    "    def predict(self, X):\n",
    "        mean, var = self.model.predict_y(X)  # Yに対する予測確率を返す\n",
    "        return mean, var\n",
    "\n",
    "# トイデータ生成とモデルの使用\n",
    "#np.random.seed(42)\n",
    "#X = np.random.rand(100, 2) * 10\n",
    "# 多クラスのラベル生成（3クラス）\n",
    "\n",
    "#Y = np.digitize(X, bins=np.quantile(X, [1/3, 2/3])) - 1\n",
    "# バイナリ分類のためのトイデータ生成\n",
    "np.random.seed(42)\n",
    "X = np.random.rand(100, 1) * 10  # 0から10の範囲の100個のサンプル\n",
    "Y = (np.sin(X) + np.random.randn(100, 1) * 0.1 > 0).astype(int)  # sin関数に基づくバイナリラベル\n",
    "\n",
    "# プロットして確認\n",
    "plt.scatter(X, Y, c=Y, cmap='viridis')\n",
    "plt.title(\"Sample Data for Classification\")\n",
    "plt.xlabel(\"X\")\n",
    "plt.ylabel(\"Y\")\n",
    "plt.show()\n",
    "\n",
    "#kernel = gpflow.kernels.SquaredExponential()\n",
    "#calibrator = GPCalibration(kernel=kernel, mean_function=gpflow.mean_functions.Identity(), model_type='SVGP', num_inducing=10)\n",
    "calibrator = GPCalibration(kernel=kernel, mean_function=LogMeanFunction(), model_type='SVGP', num_inducing=10)\n",
    "#calibrator = GPCalibration(kernel=kernel, mean_function=Log(), model_type='SVGP', num_inducing=10)\n",
    "calibrator.fit(X, Y)\n",
    "\n",
    "X_test = np.linspace(0, 10, 100).reshape(-1, 1)\n",
    "mean, var = calibrator.predict(X_test)\n",
    "\n",
    "# 結果のプロット\n",
    "plt.figure(figsize=(10, 5))\n",
    "for i in range(mean.shape[1]):\n",
    "    plt.plot(X_test, mean[:, i], label=f'Class {i}')\n",
    "plt.scatter(X, Y, c=Y, cmap='viridis', label='True data')\n",
    "plt.title(\"GP Classification Result\")\n",
    "plt.xlabel(\"X\")\n",
    "plt.ylabel(\"Predicted Probability\")\n",
    "plt.legend()\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "d019a068-c636-4a08-bf0d-0622d4975ad8",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "#X = np.random.rand(100, 1) * 10\n",
    "#scipy.cluster.vq.kmeans(obs=X, k_or_guess=min(X.shape[0] * X.shape[1], 10, ))[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "91274383-ca5f-4864-92bd-02565a9dd8f6",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 167,
   "id": "bb13633a-c5c3-4aaf-b461-226a12a05364",
   "metadata": {},
   "outputs": [],
   "source": [
    "files = ['kitti_all_train.data', 'kitti_all_train.labels', 'kitti_all_test.data', 'kitti_all_test.labels']\n",
    "file_path = os.getcwd() + '/kitti_features/'\n",
    "\n",
    "X_train = np.loadtxt(os.path.join(file_path, files[0]), np.float64, skiprows=1)\n",
    "y_train = np.loadtxt(os.path.join(file_path, files[1]), np.int32, skiprows=1)\n",
    "X_test = np.loadtxt(os.path.join(file_path, files[2]), np.float64, skiprows=1)\n",
    "y_test = np.loadtxt(os.path.join(file_path, files[3]), np.int32, skiprows=1)\n",
    "\n",
    "y_train = np.where(y_train > 0, 1, 0)\n",
    "y_test = np.where(y_test > 0, 1, 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 168,
   "id": "0757054a-c5cf-40af-b93b-8f3706c3a952",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style>#sk-container-id-7 {color: black;}#sk-container-id-7 pre{padding: 0;}#sk-container-id-7 div.sk-toggleable {background-color: white;}#sk-container-id-7 label.sk-toggleable__label {cursor: pointer;display: block;width: 100%;margin-bottom: 0;padding: 0.3em;box-sizing: border-box;text-align: center;}#sk-container-id-7 label.sk-toggleable__label-arrow:before {content: \"▸\";float: left;margin-right: 0.25em;color: #696969;}#sk-container-id-7 label.sk-toggleable__label-arrow:hover:before {color: black;}#sk-container-id-7 div.sk-estimator:hover label.sk-toggleable__label-arrow:before {color: black;}#sk-container-id-7 div.sk-toggleable__content {max-height: 0;max-width: 0;overflow: hidden;text-align: left;background-color: #f0f8ff;}#sk-container-id-7 div.sk-toggleable__content pre {margin: 0.2em;color: black;border-radius: 0.25em;background-color: #f0f8ff;}#sk-container-id-7 input.sk-toggleable__control:checked~div.sk-toggleable__content {max-height: 200px;max-width: 100%;overflow: auto;}#sk-container-id-7 input.sk-toggleable__control:checked~label.sk-toggleable__label-arrow:before {content: \"▾\";}#sk-container-id-7 div.sk-estimator input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-7 div.sk-label input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-7 input.sk-hidden--visually {border: 0;clip: rect(1px 1px 1px 1px);clip: rect(1px, 1px, 1px, 1px);height: 1px;margin: -1px;overflow: hidden;padding: 0;position: absolute;width: 1px;}#sk-container-id-7 div.sk-estimator {font-family: monospace;background-color: #f0f8ff;border: 1px dotted black;border-radius: 0.25em;box-sizing: border-box;margin-bottom: 0.5em;}#sk-container-id-7 div.sk-estimator:hover {background-color: #d4ebff;}#sk-container-id-7 div.sk-parallel-item::after {content: \"\";width: 100%;border-bottom: 1px solid gray;flex-grow: 1;}#sk-container-id-7 div.sk-label:hover label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-7 div.sk-serial::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: 0;}#sk-container-id-7 div.sk-serial {display: flex;flex-direction: column;align-items: center;background-color: white;padding-right: 0.2em;padding-left: 0.2em;position: relative;}#sk-container-id-7 div.sk-item {position: relative;z-index: 1;}#sk-container-id-7 div.sk-parallel {display: flex;align-items: stretch;justify-content: center;background-color: white;position: relative;}#sk-container-id-7 div.sk-item::before, #sk-container-id-7 div.sk-parallel-item::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: -1;}#sk-container-id-7 div.sk-parallel-item {display: flex;flex-direction: column;z-index: 1;position: relative;background-color: white;}#sk-container-id-7 div.sk-parallel-item:first-child::after {align-self: flex-end;width: 50%;}#sk-container-id-7 div.sk-parallel-item:last-child::after {align-self: flex-start;width: 50%;}#sk-container-id-7 div.sk-parallel-item:only-child::after {width: 0;}#sk-container-id-7 div.sk-dashed-wrapped {border: 1px dashed gray;margin: 0 0.4em 0.5em 0.4em;box-sizing: border-box;padding-bottom: 0.4em;background-color: white;}#sk-container-id-7 div.sk-label label {font-family: monospace;font-weight: bold;display: inline-block;line-height: 1.2em;}#sk-container-id-7 div.sk-label-container {text-align: center;}#sk-container-id-7 div.sk-container {/* jupyter's `normalize.less` sets `[hidden] { display: none; }` but bootstrap.min.css set `[hidden] { display: none !important; }` so we also need the `!important` here to be able to override the default hidden behavior on the sphinx rendered scikit-learn.org. See: https://github.com/scikit-learn/scikit-learn/issues/21755 */display: inline-block !important;position: relative;}#sk-container-id-7 div.sk-text-repr-fallback {display: none;}</style><div id=\"sk-container-id-7\" class=\"sk-top-container\"><div class=\"sk-text-repr-fallback\"><pre>XGBClassifier(base_score=None, booster=&#x27;gbtree&#x27;, callbacks=None,\n",
       "              colsample_bylevel=None, colsample_bynode=None,\n",
       "              colsample_bytree=None, device=None, early_stopping_rounds=None,\n",
       "              enable_categorical=False, eval_metric=None, feature_types=None,\n",
       "              gamma=None, grow_policy=None, importance_type=None,\n",
       "              interaction_constraints=None, learning_rate=None, max_bin=None,\n",
       "              max_cat_threshold=None, max_cat_to_onehot=None,\n",
       "              max_delta_step=None, max_depth=None, max_leaves=None,\n",
       "              min_child_weight=None, missing=nan, monotone_constraints=None,\n",
       "              multi_strategy=None, n_estimators=100, n_jobs=-1,\n",
       "              num_parallel_tree=None, random_state=1, ...)</pre><b>In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook. <br />On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.</b></div><div class=\"sk-container\" hidden><div class=\"sk-item\"><div class=\"sk-estimator sk-toggleable\"><input class=\"sk-toggleable__control sk-hidden--visually\" id=\"sk-estimator-id-7\" type=\"checkbox\" checked><label for=\"sk-estimator-id-7\" class=\"sk-toggleable__label sk-toggleable__label-arrow\">XGBClassifier</label><div class=\"sk-toggleable__content\"><pre>XGBClassifier(base_score=None, booster=&#x27;gbtree&#x27;, callbacks=None,\n",
       "              colsample_bylevel=None, colsample_bynode=None,\n",
       "              colsample_bytree=None, device=None, early_stopping_rounds=None,\n",
       "              enable_categorical=False, eval_metric=None, feature_types=None,\n",
       "              gamma=None, grow_policy=None, importance_type=None,\n",
       "              interaction_constraints=None, learning_rate=None, max_bin=None,\n",
       "              max_cat_threshold=None, max_cat_to_onehot=None,\n",
       "              max_delta_step=None, max_depth=None, max_leaves=None,\n",
       "              min_child_weight=None, missing=nan, monotone_constraints=None,\n",
       "              multi_strategy=None, n_estimators=100, n_jobs=-1,\n",
       "              num_parallel_tree=None, random_state=1, ...)</pre></div></div></div></div></div>"
      ],
      "text/plain": [
       "XGBClassifier(base_score=None, booster='gbtree', callbacks=None,\n",
       "              colsample_bylevel=None, colsample_bynode=None,\n",
       "              colsample_bytree=None, device=None, early_stopping_rounds=None,\n",
       "              enable_categorical=False, eval_metric=None, feature_types=None,\n",
       "              gamma=None, grow_policy=None, importance_type=None,\n",
       "              interaction_constraints=None, learning_rate=None, max_bin=None,\n",
       "              max_cat_threshold=None, max_cat_to_onehot=None,\n",
       "              max_delta_step=None, max_depth=None, max_leaves=None,\n",
       "              min_child_weight=None, missing=nan, monotone_constraints=None,\n",
       "              multi_strategy=None, n_estimators=100, n_jobs=-1,\n",
       "              num_parallel_tree=None, random_state=1, ...)"
      ]
     },
     "execution_count": 168,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Classify KITTI validation data with selected classifiers\n",
    "random_state = 1\n",
    "model = xgb.XGBClassifier(booster=\"gbtree\", n_estimators=100, random_state=random_state, n_jobs=-1)\n",
    "#model = RandomForestClassifier(n_estimators=100, criterion=\"gini\", min_samples_split=2, bootstrap=True, n_jobs=-1, random_state=random_state)\n",
    "\n",
    "model.fit(X=X_train, y=y_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 169,
   "id": "966e4ae4-be6f-4ee4-88f0-10df02e4b747",
   "metadata": {},
   "outputs": [],
   "source": [
    "p_pred_train = model.predict_proba(X_train)\n",
    "p_pred = model.predict_proba(X_test)\n",
    "y_y_pred = np.column_stack([y_test, np.argmax(p_pred, axis=1)])\n",
    "# Remove possible nan or inf\n",
    "isnotnan_ind = ~np.isnan(p_pred).any(axis=1)\n",
    "p_pred = p_pred[isnotnan_ind, :]\n",
    "y_y_pred = y_y_pred[isnotnan_ind, :]\n",
    "\n",
    "isnotnan_ind = ~np.isnan(p_pred_train).any(axis=1)\n",
    "p_pred_train = p_pred_train[isnotnan_ind, :]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 172,
   "id": "9552045d-c87c-48be-ac0d-97c2e7e6ab4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "#p_pred_train.reshape(-1,1)\n",
    "#p_pred_train.shape\n",
    "#y_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 173,
   "id": "b1708a0a-8471-4005-a314-99e2bac0958a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style>#sk-container-id-8 {color: black;}#sk-container-id-8 pre{padding: 0;}#sk-container-id-8 div.sk-toggleable {background-color: white;}#sk-container-id-8 label.sk-toggleable__label {cursor: pointer;display: block;width: 100%;margin-bottom: 0;padding: 0.3em;box-sizing: border-box;text-align: center;}#sk-container-id-8 label.sk-toggleable__label-arrow:before {content: \"▸\";float: left;margin-right: 0.25em;color: #696969;}#sk-container-id-8 label.sk-toggleable__label-arrow:hover:before {color: black;}#sk-container-id-8 div.sk-estimator:hover label.sk-toggleable__label-arrow:before {color: black;}#sk-container-id-8 div.sk-toggleable__content {max-height: 0;max-width: 0;overflow: hidden;text-align: left;background-color: #f0f8ff;}#sk-container-id-8 div.sk-toggleable__content pre {margin: 0.2em;color: black;border-radius: 0.25em;background-color: #f0f8ff;}#sk-container-id-8 input.sk-toggleable__control:checked~div.sk-toggleable__content {max-height: 200px;max-width: 100%;overflow: auto;}#sk-container-id-8 input.sk-toggleable__control:checked~label.sk-toggleable__label-arrow:before {content: \"▾\";}#sk-container-id-8 div.sk-estimator input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-8 div.sk-label input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-8 input.sk-hidden--visually {border: 0;clip: rect(1px 1px 1px 1px);clip: rect(1px, 1px, 1px, 1px);height: 1px;margin: -1px;overflow: hidden;padding: 0;position: absolute;width: 1px;}#sk-container-id-8 div.sk-estimator {font-family: monospace;background-color: #f0f8ff;border: 1px dotted black;border-radius: 0.25em;box-sizing: border-box;margin-bottom: 0.5em;}#sk-container-id-8 div.sk-estimator:hover {background-color: #d4ebff;}#sk-container-id-8 div.sk-parallel-item::after {content: \"\";width: 100%;border-bottom: 1px solid gray;flex-grow: 1;}#sk-container-id-8 div.sk-label:hover label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-8 div.sk-serial::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: 0;}#sk-container-id-8 div.sk-serial {display: flex;flex-direction: column;align-items: center;background-color: white;padding-right: 0.2em;padding-left: 0.2em;position: relative;}#sk-container-id-8 div.sk-item {position: relative;z-index: 1;}#sk-container-id-8 div.sk-parallel {display: flex;align-items: stretch;justify-content: center;background-color: white;position: relative;}#sk-container-id-8 div.sk-item::before, #sk-container-id-8 div.sk-parallel-item::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: -1;}#sk-container-id-8 div.sk-parallel-item {display: flex;flex-direction: column;z-index: 1;position: relative;background-color: white;}#sk-container-id-8 div.sk-parallel-item:first-child::after {align-self: flex-end;width: 50%;}#sk-container-id-8 div.sk-parallel-item:last-child::after {align-self: flex-start;width: 50%;}#sk-container-id-8 div.sk-parallel-item:only-child::after {width: 0;}#sk-container-id-8 div.sk-dashed-wrapped {border: 1px dashed gray;margin: 0 0.4em 0.5em 0.4em;box-sizing: border-box;padding-bottom: 0.4em;background-color: white;}#sk-container-id-8 div.sk-label label {font-family: monospace;font-weight: bold;display: inline-block;line-height: 1.2em;}#sk-container-id-8 div.sk-label-container {text-align: center;}#sk-container-id-8 div.sk-container {/* jupyter's `normalize.less` sets `[hidden] { display: none; }` but bootstrap.min.css set `[hidden] { display: none !important; }` so we also need the `!important` here to be able to override the default hidden behavior on the sphinx rendered scikit-learn.org. See: https://github.com/scikit-learn/scikit-learn/issues/21755 */display: inline-block !important;position: relative;}#sk-container-id-8 div.sk-text-repr-fallback {display: none;}</style><div id=\"sk-container-id-8\" class=\"sk-top-container\"><div class=\"sk-text-repr-fallback\"><pre>GPCalibration(kernel=&lt;gpflow.kernels.base.Sum object at 0x3183a5010&gt;,\n",
       "              mean_function=&lt;__main__.LogMeanFunction object at 0x2e3f9cf90&gt;)</pre><b>In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook. <br />On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.</b></div><div class=\"sk-container\" hidden><div class=\"sk-item\"><div class=\"sk-estimator sk-toggleable\"><input class=\"sk-toggleable__control sk-hidden--visually\" id=\"sk-estimator-id-8\" type=\"checkbox\" checked><label for=\"sk-estimator-id-8\" class=\"sk-toggleable__label sk-toggleable__label-arrow\">GPCalibration</label><div class=\"sk-toggleable__content\"><pre>GPCalibration(kernel=&lt;gpflow.kernels.base.Sum object at 0x3183a5010&gt;,\n",
       "              mean_function=&lt;__main__.LogMeanFunction object at 0x2e3f9cf90&gt;)</pre></div></div></div></div></div>"
      ],
      "text/plain": [
       "GPCalibration(kernel=<gpflow.kernels.base.Sum object at 0x3183a5010>,\n",
       "              mean_function=<__main__.LogMeanFunction object at 0x2e3f9cf90>)"
      ]
     },
     "execution_count": 173,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "recal_f = GPCalibration(kernel=kernel, mean_function=LogMeanFunction(), model_type='SVGP', num_inducing=10)\n",
    "#recal_f.fit(p_pred_train.max(1).reshape(-1,1), y_train.reshape(-1,1))\n",
    "#recal_f.fit(p_pred_train[:,0].reshape(-1,1), y_train)\n",
    "recal_f.fit(p_pred_train, y_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 154,
   "id": "fdccf784-3ba1-4ff9-858b-406cba1cd4ad",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(16000, 2)"
      ]
     },
     "execution_count": 154,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "p_pred_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 174,
   "id": "7a679e20-03a5-4a0d-aa35-c37bc32d938c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[9.93194980e-01, 6.80501959e-03],\n",
       "       [9.99036903e-01, 9.63096753e-04],\n",
       "       [9.99787478e-01, 2.12522022e-04],\n",
       "       ...,\n",
       "       [9.99940954e-01, 5.90459797e-05],\n",
       "       [9.99549411e-01, 4.50588728e-04],\n",
       "       [9.99779174e-01, 2.20826234e-04]])"
      ]
     },
     "execution_count": 174,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#recal_f.predict(p_pred_train[:,0].reshape(-1,1))[0].numpy().max(1)\n",
    "#recal_f.predict(p_pred_train)\n",
    "recal_f.predict(p_pred_train)[0].numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "id": "6ef0e7f2-ab11-4d1d-b463-6d681c1a5a8c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "10.72220393740328"
      ]
     },
     "execution_count": 79,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "kl_divergence = recal_f.model.prior_kl().numpy()\n",
    "kl_divergence\n",
    "#print(f\"KL divergence: {kl_divergence}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 111,
   "id": "6e5b71b6-ba8a-484e-8c78-a0cf186831eb",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[9.93729147e-01, 6.27085313e-03],\n",
       "       [9.99096444e-01, 9.03555977e-04],\n",
       "       [9.99797788e-01, 2.02212374e-04],\n",
       "       ...,\n",
       "       [9.99945715e-01, 5.42845405e-05],\n",
       "       [9.99594175e-01, 4.05825331e-04],\n",
       "       [9.99794755e-01, 2.05244930e-04]])"
      ]
     },
     "execution_count": 111,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#recal_f.model.predict_f(p_pred_train)\n",
    "f, _ = recal_f.model.predict_f(p_pred_train)\n",
    "latent = f.numpy().reshape(np.shape(p_pred_train))\n",
    "\n",
    "# Return softargmax of fitted GP at input\n",
    "scipy.special.softmax(latent, axis=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 108,
   "id": "41f3e101-7719-4cad-a4b0-b805d983d3e8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(<tf.Tensor: shape=(16000, 2), dtype=float64, numpy=\n",
       " array([[9.93210176e-01, 6.78982419e-03],\n",
       "        [9.98993752e-01, 1.00624782e-03],\n",
       "        [9.99775748e-01, 2.24251510e-04],\n",
       "        ...,\n",
       "        [9.99937432e-01, 6.25682447e-05],\n",
       "        [9.99548404e-01, 4.51595530e-04],\n",
       "        [9.99756867e-01, 2.43132586e-04]])>,\n",
       " <tf.Tensor: shape=(16000, 2), dtype=float64, numpy=\n",
       " array([[6.74372248e-03, 6.74372248e-03],\n",
       "        [1.00523529e-03, 1.00523529e-03],\n",
       "        [2.24201222e-04, 2.24201222e-04],\n",
       "        ...,\n",
       "        [6.25643299e-05, 6.25643299e-05],\n",
       "        [4.51391591e-04, 4.51391591e-04],\n",
       "        [2.43073472e-04, 2.43073472e-04]])>)"
      ]
     },
     "execution_count": 108,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "recal_f.model.predict_y(p_pred_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 151,
   "id": "ecd5102e-4ae5-4db3-a6f8-16d96df2269c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[9.92871957e-01, 7.12804270e-03],\n",
       "       [9.98979742e-01, 1.02025756e-03],\n",
       "       [9.99785166e-01, 2.14834277e-04],\n",
       "       ...,\n",
       "       [9.99939612e-01, 6.03881554e-05],\n",
       "       [9.99577946e-01, 4.22054016e-04],\n",
       "       [9.99769628e-01, 2.30371663e-04]])"
      ]
     },
     "execution_count": 151,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#from gpflow.conditionals import conditional\n",
    "#recal_f.model.likelihood.predict_log_density((p_pred_train,y_train))\n",
    "#np.exp(recal_f.model.predict_log_density((p_pred_train, y_train.reshape(-1,1))).numpy())\n",
    "#np.exp(recal_f.model.predict_log_density(p_pred_train).numpy())\n",
    "#mu, var = recal_f.model.predict_y(p_pred_train)\n",
    "\n",
    "mu, var = recal_f.model.predict_f(p_pred_train)\n",
    "\n",
    "N = tf.shape(mu)[0]\n",
    "epsilon = tf.random.normal((100, N, 2), dtype=tf.float64)\n",
    "\n",
    "f_star = mu[None, :, :] + tf.sqrt(var[None, :, :]) * epsilon  # S x N x K\n",
    "p_y_f_star = tf.nn.softmax(f_star, axis=2)\n",
    "tf.math.log(tf.reduce_mean(p_y_f_star, axis=0)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 164,
   "id": "cf57bd5a-d815-4df6-a534-18a47a413286",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.float32"
      ]
     },
     "execution_count": 164,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.tensor(recal_f.model.predict_y(p_pred_train)[0].numpy()).float().dtype"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d34ea64f-8126-4e76-8cff-8fbf71fc48b9",
   "metadata": {},
   "outputs": [],
   "source": [
    "def predict_full_density(self, Fmus, Fvars):\n",
    "        if isinstance(self.invlink, SoftArgMax):\n",
    "            # Sample from standard normal\n",
    "            N = tf.shape(Fmus)[0]\n",
    "            epsilon = tf.random_normal((self.num_monte_carlo_points, N, self.num_classes),\n",
    "                                       dtype=settings.float_type)\n",
    "\n",
    "            # Transform to correct mean and covariance\n",
    "            f_star = Fmus[None, :, :] + tf.sqrt(Fvars[None, :, :]) * epsilon  # S x N x K\n",
    "\n",
    "            # Compute Softmax\n",
    "            p_y_f_star = tf.nn.softmax(f_star, axis=2)\n",
    "\n",
    "            # Average to obtain log Monte-Carlo estimate\n",
    "            return tf.log(tf.reduce_mean(p_y_f_star, axis=0))\n",
    "        else:\n",
    "            raise NotImplementedError"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f482ee91-e90e-4b6b-aef4-ba731a207831",
   "metadata": {},
   "outputs": [],
   "source": [
    "    @params_as_tensors\n",
    "    def _build_predict(self, Xnew, full_cov=False, full_output_cov=False):\n",
    "        \"\"\"\n",
    "        Compute the mean and variance of :math:`p(f_* \\\\mid y)`.\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        Xnew : np.array, shape=(N, K)\n",
    "        full_cov : bool\n",
    "        full_output_cov : bool\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "        mus, vars :\n",
    "            Mean and covariances of the variational approximation to the GP applied to the K input dimensions of Xnew.\n",
    "            Dimensions: mus= N x K and vars= N x K (x K)\n",
    "\n",
    "        \"\"\"\n",
    "        # Reshape to obtain correct covariance\n",
    "        num_data_new = tf.shape(Xnew)[0]\n",
    "        Xnew = tf.reshape(Xnew, [-1, 1])\n",
    "\n",
    "        # Compute conditional\n",
    "        mu_tmp, var_tmp = conditional(Xnew, self.feature, self.kern, self.q_mu, q_sqrt=self.q_sqrt,\n",
    "                                      full_cov=full_cov,\n",
    "                                      white=self.whiten, full_output_cov=full_output_cov)\n",
    "\n",
    "        # Reshape to N x K\n",
    "        mu = tf.reshape(mu_tmp + self.mean_function(Xnew), [num_data_new, self.num_classes])\n",
    "        var = tf.reshape(var_tmp, [num_data_new, self.num_classes])\n",
    "\n",
    "        return mu, var\n",
    "    def predict_full_density(self, Xnew):\n",
    "        pred_f_mean, pred_f_var = self._build_predict(Xnew)\n",
    "        return self.likelihood.predict_full_density(pred_f_mean, pred_f_var)\n",
    "\n",
    "    @params_as_tensors\n",
    "    def predict_f(self, X_onedim, full_cov=False, full_output_cov=False):\n",
    "        \"\"\"\n",
    "        Predict the one-dimensional latent function\n",
    "\n",
    "        Parameters\n",
    "        ----------\n",
    "        X_onedim\n",
    "\n",
    "        Returns\n",
    "        -------\n",
    "\n",
    "        \"\"\"\n",
    "        # Compute conditional\n",
    "        mu, var = conditional(X_onedim, self.feature, self.kern, self.q_mu, q_sqrt=self.q_sqrt,\n",
    "                              full_cov=full_cov,\n",
    "                              white=self.whiten, full_output_cov=full_output_cov)\n",
    "\n",
    "        return mu + self.mean_function(X_onedim), var\n",
    "\n",
    "    @params_as_tensors\n",
    "    def predict_full_density(self, Xnew):\n",
    "        pred_f_mean, pred_f_var = self._build_predict(Xnew)\n",
    "        return self.likelihood.predict_full_density(pred_f_mean, pred_f_var)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
