{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "33eecf77",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Importing the libraries \n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "from sklearn import metrics\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "1d2e3d3c",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torch.optim as optim\n",
    "from torch.utils.data import *\n",
    "from torchvision import transforms\n",
    "import torchvision\n",
    "from tqdm import tqdm\n",
    "from torchvision.utils import save_image\n",
    "to_pil_image = transforms.ToPILImage()\n",
    "def image_to_vid(images):\n",
    "    imgs = [np.array(to_pil_image(img)) for img in images]\n",
    "    imageio.mimsave('../outputs/generated_images.gif', imgs)\n",
    "def save_reconstructed_imagesVAE(recon_images, epoch):\n",
    "    save_image(recon_images.cpu(), f\"VAE{epoch}.jpg\")\n",
    "def save_reconstructed_imagesEVAE(recon_images, epoch):\n",
    "    save_image(recon_images.cpu(), f\"EVAE{epoch}.jpg\")\n",
    "def save_reconstructed_imagesREAL(images, epoch):\n",
    "    save_image(images.cpu(), f\"REAL{epoch}.jpg\")\n",
    "def save_train_loss_plot(train_loss, valid_loss):\n",
    "    # loss plots\n",
    "    plt.figure(figsize=(10, 7))\n",
    "    plt.plot(train_loss, color='orange', label='train loss VAE')\n",
    "    plt.plot(valid_loss, color='red', label='train loss EVAE')\n",
    "    plt.xlabel('Epochs')\n",
    "    plt.ylabel('Total loss (Reconstruction+B*I(K))/M')\n",
    "    plt.legend()\n",
    "    plt.savefig('lossMNIST.jpg')\n",
    "    plt.show()\n",
    "def save_valid_loss_plot(valid_lossVAE, valid_lossEVAE):\n",
    "    # loss plots\n",
    "    plt.figure(figsize=(10, 7))\n",
    "    plt.plot(valid_lossVAE, color='orange', label='validation loss VAE')\n",
    "    plt.plot(valid_lossEVAE, color='red', label='validation loss EVAE')\n",
    "    plt.xlabel('Epochs')\n",
    "    plt.ylabel('Reconstruction loss')\n",
    "    plt.legend()\n",
    "    plt.savefig('lossMNIST_valid.jpg')\n",
    "    plt.show()\n",
    "\n",
    "from torchvision.utils import make_grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "77b735dc",
   "metadata": {},
   "outputs": [],
   "source": [
    "class HypersphericalUniform(torch.distributions.Distribution):\n",
    "\n",
    "    support = torch.distributions.constraints.real\n",
    "    has_rsample = False\n",
    "    _mean_carrier_measure = 0\n",
    "\n",
    "    @property\n",
    "    def dim(self):\n",
    "        return self._dim\n",
    "\n",
    "    @property\n",
    "    def device(self):\n",
    "        return self._device\n",
    "\n",
    "    @device.setter\n",
    "    def device(self, val):\n",
    "        self._device = val if isinstance(val, torch.device) else torch.device(val)\n",
    "\n",
    "    def __init__(self, dim, validate_args=None, device=\"cpu\"):\n",
    "        super(HypersphericalUniform, self).__init__(\n",
    "            torch.Size([dim]), validate_args=validate_args\n",
    "        )\n",
    "        self._dim = dim\n",
    "        self.device = device\n",
    "\n",
    "    def sample(self, shape=torch.Size()):\n",
    "        output = (\n",
    "            torch.distributions.Normal(0, 1)\n",
    "            .sample(\n",
    "                (shape if isinstance(shape, torch.Size) else torch.Size([shape]))\n",
    "                + torch.Size([self._dim + 1])\n",
    "            )\n",
    "            .to(self.device)\n",
    "        )\n",
    "\n",
    "        return output / output.norm(dim=-1, keepdim=True)\n",
    "\n",
    "    def entropy(self):\n",
    "        \n",
    "        return self.__log_surface_area().to(device)\n",
    "\n",
    "    def log_prob(self, x):\n",
    "        return -torch.ones(x.shape[:-1], device=self.device) * self.__log_surface_area()\n",
    "\n",
    "    def __log_surface_area(self):\n",
    "        if torch.__version__ >= \"1.0.0\":\n",
    "            lgamma = torch.lgamma(torch.tensor([(self._dim + 1) / 2]).to(self.device))\n",
    "        else:\n",
    "            lgamma = torch.lgamma(\n",
    "                torch.Tensor([(self._dim + 1) / 2], device=self.device)\n",
    "            )\n",
    "        return math.log(2) + ((self._dim + 1) / 2) * math.log(math.pi) - lgamma"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "8693f4c8",
   "metadata": {},
   "outputs": [],
   "source": [
    "import scipy.special\n",
    "from numbers import Number\n",
    "\n",
    "\n",
    "class IveFunction(torch.autograd.Function):\n",
    "    @staticmethod\n",
    "    def forward(self, v, z):\n",
    "\n",
    "        assert isinstance(v, Number), \"v must be a scalar\"\n",
    "\n",
    "        self.save_for_backward(z)\n",
    "        self.v = v\n",
    "        z_cpu = z.data.cpu().numpy()\n",
    "\n",
    "        if np.isclose(v, 0):\n",
    "            output = scipy.special.i0e(z_cpu, dtype=z_cpu.dtype)\n",
    "        elif np.isclose(v, 1):\n",
    "            output = scipy.special.i1e(z_cpu, dtype=z_cpu.dtype)\n",
    "        else:  #  v > 0\n",
    "            output = scipy.special.ive(v, z_cpu, dtype=z_cpu.dtype)\n",
    "        #         else:\n",
    "        #             print(v, type(v), np.isclose(v, 0))\n",
    "        #             raise RuntimeError('v must be >= 0, it is {}'.format(v))\n",
    "\n",
    "        return torch.Tensor(output).to(z.device)\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(self, grad_output):\n",
    "        z = self.saved_tensors[-1]\n",
    "        return (\n",
    "            None,\n",
    "            grad_output * (ive(self.v - 1, z) - ive(self.v, z) * (self.v + z) / z),\n",
    "        )\n",
    "\n",
    "\n",
    "class Ive(torch.nn.Module):\n",
    "    def __init__(self, v):\n",
    "        super(Ive, self).__init__()\n",
    "        self.v = v\n",
    "\n",
    "    def forward(self, z):\n",
    "        return ive(self.v, z)\n",
    "\n",
    "\n",
    "ive = IveFunction.apply\n",
    "\n",
    "\n",
    "##########\n",
    "# The below provided approximations were provided in the\n",
    "# respective source papers, to improve the stability of\n",
    "# the Bessel fractions.\n",
    "# I_(v/2)(k) / I_(v/2 - 1)(k)\n",
    "\n",
    "# source: https://arxiv.org/pdf/1606.02008.pdf\n",
    "def ive_fraction_approx(v, z):\n",
    "    # I_(v/2)(k) / I_(v/2 - 1)(k) >= z / (v-1 + ((v+1)^2 + z^2)^0.5\n",
    "    return z / (v - 1 + torch.pow(torch.pow(v + 1, 2) + torch.pow(z, 2), 0.5))\n",
    "\n",
    "\n",
    "# source: https://arxiv.org/pdf/1902.02603.pdf\n",
    "def ive_fraction_approx2(v, z, eps=1e-20):\n",
    "    def delta_a(a):\n",
    "        lamb = v + (a - 1.0) / 2.0\n",
    "        return (v - 0.5) + lamb / (\n",
    "            2 * torch.sqrt((torch.pow(lamb, 2) + torch.pow(z, 2)).clamp(eps))\n",
    "        )\n",
    "\n",
    "    delta_0 = delta_a(0.0)\n",
    "    delta_2 = delta_a(2.0)\n",
    "    B_0 = z / (\n",
    "        delta_0 + torch.sqrt((torch.pow(delta_0, 2) + torch.pow(z, 2))).clamp(eps)\n",
    "    )\n",
    "    B_2 = z / (\n",
    "        delta_2 + torch.sqrt((torch.pow(delta_2, 2) + torch.pow(z, 2))).clamp(eps)\n",
    "    )\n",
    "\n",
    "    return (B_0 + B_2) / 2.0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "bbbbc408",
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch.distributions.kl import register_kl\n",
    "class VonMisesFisher(torch.distributions.Distribution):\n",
    "\n",
    "    arg_constraints = {\n",
    "        \"loc\": torch.distributions.constraints.real,\n",
    "        \"scale\": torch.distributions.constraints.positive,\n",
    "    }\n",
    "    support = torch.distributions.constraints.real\n",
    "    has_rsample = True\n",
    "    _mean_carrier_measure = 0\n",
    "\n",
    "    @property\n",
    "    def mean(self):\n",
    "        # option 1:\n",
    "        return self.loc * (\n",
    "            ive(self.__m / 2, self.scale) / ive(self.__m / 2 - 1, self.scale)\n",
    "        )\n",
    "        # option 2:\n",
    "        # return self.loc * ive_fraction_approx(torch.tensor(self.__m / 2), self.scale)\n",
    "        # options 3:\n",
    "        # return self.loc * ive_fraction_approx2(torch.tensor(self.__m / 2), self.scale)\n",
    "\n",
    "    @property\n",
    "    def stddev(self):\n",
    "        return self.scale\n",
    "\n",
    "    def __init__(self, loc, scale, validate_args=None, k=1):\n",
    "        self.dtype = loc.dtype\n",
    "        self.loc = loc\n",
    "        self.scale = scale\n",
    "        self.device = loc.device\n",
    "        self.__m = loc.shape[-1]\n",
    "        self.__e1 = (torch.Tensor([1.0] + [0] * (loc.shape[-1] - 1))).to(self.device)\n",
    "        self.k = k\n",
    "\n",
    "        super().__init__(self.loc.size(), validate_args=validate_args)\n",
    "\n",
    "    def sample(self, shape=torch.Size()):\n",
    "        with torch.no_grad():\n",
    "            return self.rsample(shape)\n",
    "\n",
    "    def rsample(self, shape=torch.Size()):\n",
    "        shape = shape if isinstance(shape, torch.Size) else torch.Size([shape])\n",
    "\n",
    "        w = (\n",
    "            self.__sample_w3(shape=shape)\n",
    "            if self.__m == 3\n",
    "            else self.__sample_w_rej(shape=shape)\n",
    "        )\n",
    "\n",
    "        v = (\n",
    "            torch.distributions.Normal(0, 1)\n",
    "            .sample(shape + torch.Size(self.loc.shape))\n",
    "            .to(self.device)\n",
    "            .transpose(0, -1)[1:]\n",
    "        ).transpose(0, -1)\n",
    "\n",
    "        v = v / v.norm(dim=-1, keepdim=True)\n",
    "#[1:]\n",
    "        w_ = torch.sqrt(torch.clamp(1 - (w ** 2), 1e-10))\n",
    "        x = torch.cat((w, w_ * v), -1)\n",
    "        z = self.__householder_rotation(x)\n",
    "\n",
    "        return z.type(self.dtype)\n",
    "\n",
    "    def __sample_w3(self, shape):\n",
    "        shape = shape + torch.Size(self.scale.shape)\n",
    "        u = torch.distributions.Uniform(0, 1).sample(shape).to(self.device)\n",
    "        self.__w = (\n",
    "            1\n",
    "            + torch.stack(\n",
    "                [torch.log(u), torch.log(1 - u) - 2 * self.scale], dim=0\n",
    "            ).logsumexp(0)\n",
    "            / self.scale\n",
    "        )\n",
    "        return self.__w\n",
    "\n",
    "    def __sample_w_rej(self, shape):\n",
    "        c = torch.sqrt((4 * (self.scale ** 2)) + (self.__m - 1) ** 2)\n",
    "        b_true = (-2 * self.scale + c) / (self.__m - 1)\n",
    "\n",
    "        # using Taylor approximation with a smooth swift from 10 < scale < 11\n",
    "        # to avoid numerical errors for large scale\n",
    "        b_app = (self.__m - 1) / (4 * self.scale)\n",
    "        s = torch.min(\n",
    "            torch.max(\n",
    "                torch.tensor([0.0], dtype=self.dtype, device=self.device),\n",
    "                self.scale - 10,\n",
    "            ),\n",
    "            torch.tensor([1.0], dtype=self.dtype, device=self.device),\n",
    "        )\n",
    "        b = b_app * s + b_true * (1 - s)\n",
    "\n",
    "        a = (self.__m - 1 + 2 * self.scale + c) / 4\n",
    "        d = (4 * a * b) / (1 + b) - (self.__m - 1) * math.log(self.__m - 1)\n",
    "\n",
    "        self.__b, (self.__e, self.__w) = b, self.__while_loop(b, a, d, shape, k=self.k)\n",
    "        return self.__w\n",
    "\n",
    "    @staticmethod\n",
    "    def first_nonzero(x, dim, invalid_val=-1):\n",
    "        mask = x > 0\n",
    "        idx = torch.where(\n",
    "            mask.any(dim=dim),\n",
    "            mask.float().argmax(dim=1).squeeze(),\n",
    "            torch.tensor(invalid_val, device=x.device),\n",
    "        )\n",
    "        return idx\n",
    "\n",
    "    def __while_loop(self, b, a, d, shape, k=20, eps=1e-20):\n",
    "        #  matrix while loop: samples a matrix of [A, k] samples, to avoid looping all together\n",
    "        b, a, d = [\n",
    "            e.repeat(*shape, *([1] * len(self.scale.shape))).reshape(-1, 1)\n",
    "            for e in (b, a, d)\n",
    "        ]\n",
    "        w, e, bool_mask = (\n",
    "            torch.zeros_like(b).to(self.device),\n",
    "            torch.zeros_like(b).to(self.device),\n",
    "            (torch.ones_like(b) == 1).to(self.device),\n",
    "        )\n",
    "\n",
    "        sample_shape = torch.Size([b.shape[0], k])\n",
    "        shape = shape + torch.Size(self.scale.shape)\n",
    "\n",
    "        while bool_mask.sum() != 0:\n",
    "            con1 = torch.tensor((self.__m - 1) / 2, dtype=torch.float64)\n",
    "            con2 = torch.tensor((self.__m - 1) / 2, dtype=torch.float64)\n",
    "            e_ = (\n",
    "                torch.distributions.Beta(con1, con2)\n",
    "                .sample(sample_shape)\n",
    "                .to(self.device)\n",
    "                .type(self.dtype)\n",
    "            )\n",
    "\n",
    "            u = (\n",
    "                torch.distributions.Uniform(0 + eps, 1 - eps)\n",
    "                .sample(sample_shape)\n",
    "                .to(self.device)\n",
    "                .type(self.dtype)\n",
    "            )\n",
    "\n",
    "            w_ = (1 - (1 + b) * e_) / (1 - (1 - b) * e_)\n",
    "            t = (2 * a * b) / (1 - (1 - b) * e_)\n",
    "\n",
    "            accept = ((self.__m - 1.0) * t.log() - t + d) > torch.log(u)\n",
    "            accept_idx = self.first_nonzero(accept, dim=-1, invalid_val=-1).unsqueeze(1)\n",
    "            accept_idx_clamped = accept_idx.clamp(0)\n",
    "            # we use .abs(), in order to not get -1 index issues, the -1 is still used afterwards\n",
    "            w_ = w_.gather(1, accept_idx_clamped.view(-1, 1))\n",
    "            e_ = e_.gather(1, accept_idx_clamped.view(-1, 1))\n",
    "\n",
    "            reject = accept_idx < 0\n",
    "            accept = ~reject if torch.__version__ >= \"1.2.0\" else 1 - reject\n",
    "\n",
    "            w[bool_mask * accept] = w_[bool_mask * accept]\n",
    "            e[bool_mask * accept] = e_[bool_mask * accept]\n",
    "\n",
    "            bool_mask[bool_mask * accept] = reject[bool_mask * accept]\n",
    "\n",
    "        return e.reshape(shape), w.reshape(shape)\n",
    "\n",
    "    def __householder_rotation(self, x):\n",
    "        u = self.__e1 - self.loc\n",
    "        u = u / (u.norm(dim=-1, keepdim=True) + 1e-5)\n",
    "\n",
    "        z = x - 2 * (x * u).sum(-1, keepdim=True) * u\n",
    "        return z\n",
    "\n",
    "    def entropy(self):\n",
    "        # option 1:\n",
    "        output = (\n",
    "            -self.scale\n",
    "            * ive(self.__m / 2, self.scale)\n",
    "            / ive((self.__m / 2) - 1, self.scale)\n",
    "        ).to(self.device)\n",
    "        # option 2:\n",
    "        # output = - self.scale * ive_fraction_approx(torch.tensor(self.__m / 2), self.scale)\n",
    "        # option 3:\n",
    "        # output = - self.scale * ive_fraction_approx2(torch.tensor(self.__m / 2), self.scale)\n",
    "        \n",
    "        return output.view(*(output.shape[:-1]))+ self._log_normalization()\n",
    "\n",
    "    def log_prob(self, x):\n",
    "        return self._log_unnormalized_prob(x) - self._log_normalization()\n",
    "\n",
    "    def _log_unnormalized_prob(self, x):\n",
    "        output = self.scale * (self.loc * x).sum(-1, keepdim=True)\n",
    "\n",
    "        return output.view(*(output.shape[:-1]))\n",
    "\n",
    "    def _log_normalization(self):\n",
    "        output = -(\n",
    "            (self.__m / 2 - 1) * torch.log(self.scale)\n",
    "            - (self.__m / 2) * math.log(2 * math.pi)\n",
    "            - (self.scale + torch.log(ive(self.__m / 2 - 1, self.scale)))\n",
    "        )\n",
    "        #print(output.view(*(output.shape[:-1])).to(device))\n",
    "\n",
    "        return output.view(*(output.shape[:-1])).to(device)\n",
    "\n",
    "\n",
    "@register_kl(VonMisesFisher, HypersphericalUniform)\n",
    "def _kl_vmf_uniform(vmf, hyu):\n",
    "    return -vmf.entropy() + hyu.entropy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "02d7c481",
   "metadata": {},
   "outputs": [],
   "source": [
    "class HVAE(nn.Module):\n",
    "    def __init__(self,image_channel=3,kernel_size=4,latent_dim=10,init_channel=32,reg_weight=100,kernel_type='imq',latent_var=2.0):\n",
    "        super(HVAE, self).__init__()\n",
    "\n",
    "      \n",
    "        #self.latent_dim = latent_dim\n",
    "\n",
    "        self.latent_dim=latent_dim\n",
    "        # encoder\n",
    "        self.encoder = nn.Sequential(\n",
    "            nn.Conv2d(image_channel,out_channels=init_channel,kernel_size=kernel_size,stride=2,padding=1),            \n",
    "            nn.BatchNorm2d(init_channel),\n",
    "            nn.ReLU(inplace=True),\n",
    "\n",
    "\n",
    "            nn.Conv2d(init_channel,out_channels=init_channel*2,kernel_size=kernel_size,stride=2,padding=1),\n",
    "            nn.BatchNorm2d(init_channel*2),\n",
    "            nn.ReLU(inplace=True),\n",
    "\n",
    "\n",
    "            nn.Conv2d(init_channel*2,out_channels=init_channel*4,kernel_size=kernel_size,stride=2,padding=1),\n",
    "            nn.BatchNorm2d(init_channel*4),\n",
    "            nn.ReLU(inplace=True),\n",
    "\n",
    "            nn.Conv2d(init_channel*4,out_channels=init_channel*8,kernel_size=kernel_size,stride=2,padding=0)   ,            \n",
    "            nn.BatchNorm2d(init_channel*8),\n",
    "            nn.ReLU(inplace=True),\n",
    "\n",
    "            \n",
    "            #nn.AdaptiveAvgPool2d((100,init_channel*8)),\n",
    "            nn.Flatten( start_dim=1)\n",
    "            \n",
    "            #nn.Linear(init_channel*8,init_channel*8) #hidden\n",
    "            \n",
    "        )\n",
    "        \n",
    "        \n",
    "\n",
    "        ## fully connected laeyer for learning representation\n",
    "        #self.fc_z  = nn.Linear(init_channel*8,latent_dim)\n",
    "        #self.fc_var = nn.Linear(init_channel*8,latent_dim)\n",
    "        self.fully_connected_layer_a=nn.Linear(init_channel*8,latent_dim)\n",
    "        #self.fully_connected_layer_b=nn.Linear(init_channel*8,latent_dim)\n",
    "#         self.fully_connected_layer_beta=nn.Linear(init_channel*8,latent_dim)\n",
    "        self.fc_mean = nn.Linear(latent_dim,20)\n",
    "        self.fc_var = nn.Linear(latent_dim, 1)\n",
    "        self.decoder_input =nn.Linear(20,init_channel*8)\n",
    "        ## decoder\n",
    "        \n",
    "                # encoder\n",
    "        self.decoder = nn.Sequential(\n",
    "            \n",
    "            \n",
    "            \n",
    "            nn.ConvTranspose2d(init_channel*8,out_channels=init_channel*4,kernel_size=kernel_size,stride=1,padding=0),            \n",
    "            nn.BatchNorm2d(init_channel*4),\n",
    "            nn.ReLU(inplace=True),\n",
    "\n",
    "            nn.ConvTranspose2d(init_channel*4,out_channels=init_channel*2,kernel_size=kernel_size,stride=2,padding=1),\n",
    "            nn.BatchNorm2d(init_channel*2),\n",
    "            nn.ReLU(inplace=True),\n",
    "\n",
    "\n",
    "            nn.ConvTranspose2d(init_channel*2,out_channels=init_channel*1,kernel_size=kernel_size,stride=2,padding=1),\n",
    "            nn.BatchNorm2d(init_channel*1) ,\n",
    "            nn.ReLU(inplace=True),\n",
    "\n",
    "            \n",
    "\n",
    "            nn.ConvTranspose2d(init_channel*1,out_channels=image_channel,kernel_size=kernel_size,stride=2,padding=1),\n",
    "            nn.Sigmoid()\n",
    "        )\n",
    "#         self.pseudo_input = torch.eye(self.num_components, requires_grad= False)\n",
    "#         self.embed_pseudo = nn.Sequential(nn.Linear(self.num_components, 1024),\n",
    "#                                           nn.Hardtanh(0.0, 1.0)) # 1x32x32 = 3072\n",
    "        \n",
    "    def reparametrize(self,mu,log_var):\n",
    "        \n",
    "        q_z = VonMisesFisher(mu, log_var)\n",
    "        p_z = HypersphericalUniform(self.latent_dim - 1)    \n",
    "        return q_z, p_z\n",
    " \n",
    "\n",
    "\n",
    "    def forward(self,x):\n",
    "        #encoding\n",
    "\n",
    "        hidden =self.encoder(x)\n",
    "        \n",
    "        #hidden=torch.flatten(x, start_dim=1)\n",
    "     \n",
    "        x=self.fully_connected_layer_a(hidden)\n",
    "       # y=self.fully_connected_layer_b(hidden)\n",
    "        \n",
    "        z_mean = self.fc_mean(x)\n",
    "        z_mean = z_mean / z_mean.norm(dim=-1, keepdim=True)\n",
    "        z_mean =z_mean.to(device)\n",
    "        # the `+ 1` prevent collapsing behaviors\n",
    "        z_var = (F.softplus(self.fc_var(x)) + 1).to(device)\n",
    "        \n",
    "        q_z, p_z = self.reparametrize(z_mean, z_var)\n",
    "        #print(z_var.size())\n",
    "        z = q_z.rsample()\n",
    "        \n",
    "        \n",
    "        z=self.decoder_input(z)\n",
    "      \n",
    "        z=z.view(-1,256,1,1)\n",
    "        #decoding\n",
    "        reconstruction=self.decoder(z)\n",
    "\n",
    "        return reconstruction,z_mean,z_var,q_z,p_z"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "fbd422a2",
   "metadata": {},
   "outputs": [],
   "source": [
    "def model_trainHVAE(model,dataloader,dataset,device,optimizer,criterion):\n",
    "    model.train()\n",
    "    running_loss=0.0\n",
    "    counter=0\n",
    "    for i, data in tqdm(enumerate(dataloader),total=int(len(dataset)/dataloader.batch_size)):\n",
    "        counter+=1\n",
    "        data=data[0]\n",
    "        data=data.to(device)\n",
    "        optimizer.zero_grad()\n",
    "        #output=model(data)\n",
    "        #print(len(output))\n",
    "        recons,mean,log_var,q_z,p_z =model(data)\n",
    "\n",
    "        #reconstruction_error=criterion(reconstruction,data)\n",
    "        # Combining the loss functions\n",
    "        #print(reconstruction_error)\n",
    "        #print(p_z.device())\n",
    "        loss =criterion( recons.reshape(-1,1024),data.reshape(-1, 1024)  ).sum(-1).mean()+torch.distributions.kl.kl_divergence(q_z, p_z).mean()#mse_u + mse_f+reconstruction_error+mse_fv#+mse_F_f#+reconstruction_manifold#+mse_ulb+mse_uub\n",
    "        #loss=final_loss(bce_loss,mu,log_var)\n",
    "        #print(loss)\n",
    "        #print(loss)\n",
    "        loss.backward()\n",
    "        running_loss+=loss.item()\n",
    "        optimizer.step()\n",
    "    #print(mean)\n",
    "    #print(log_var)\n",
    "    train_loss=running_loss/counter\n",
    "    return train_loss\n",
    "\n",
    "def model_validateHVAE(model,dataloader,dataset,device,optimizer,criterion):\n",
    "    model.eval()\n",
    "    running_loss=0.0\n",
    "    counter=0\n",
    "    with torch.no_grad():\n",
    "        for i,data in tqdm(enumerate(dataloader),total=int(len(dataset)/dataloader.batch_size)):\n",
    "            counter+=1\n",
    "            data=data[0]\n",
    "            data=data.to(device)\n",
    "            optimizer.zero_grad()\n",
    "            #reconstruction,mu,log_var=model(data)\n",
    "            recons,*_=model(data)\n",
    "            #reconstruction,mean,log_var =model(data)\n",
    "            #bce_loss= criterion(reconstruction,data)\n",
    "            \n",
    "\n",
    "            #reconstruction_error=criterion(reconstruction,data)\n",
    "\n",
    "            #values.append(fid.compute())\n",
    "            \n",
    "\n",
    "            loss =criterion(recons,data).sum(-1).mean()#reconstruction_error#final_loss(reconstruction_error,mean,log_var)\n",
    "            running_loss+=loss.item()\n",
    "            \n",
    "\n",
    "        valid_loss=running_loss/counter\n",
    "        #print(counter)\n",
    "        return valid_loss\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9e816c1d",
   "metadata": {},
   "source": [
    "##EVAE\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a445ce36",
   "metadata": {},
   "source": [
    "## dataset\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "0f8a070e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Dataset MNIST\n",
       "    Number of datapoints: 60000\n",
       "    Root location: ./\n",
       "    Split: Train"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torchvision.datasets.MNIST('./',download=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "788b980d",
   "metadata": {},
   "source": [
    "# training VAE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "b72a8856",
   "metadata": {},
   "outputs": [],
   "source": [
    "device=torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "latent_dim=64\n",
    "HVAEmodel=HVAE(latent_dim=latent_dim).to(device)\n",
    "#EVAEmodel=EVAE(latent_dim=latent_dim,B=10).to(device)\n",
    "lr=0.0003\n",
    "epochs=50\n",
    "batch_size=100\n",
    "\n",
    "transform=transforms.Compose([transforms.Resize((32,32)),transforms.ToTensor()])\n",
    "\n",
    "#training set transforms.Resize((32,32)),\n",
    "train_set=torchvision.datasets.CIFAR10(root='./',train=True,download=False,transform=transform)\n",
    "train_loader=torch.utils.data.DataLoader(train_set,batch_size=batch_size,shuffle=True)\n",
    "\n",
    "#test set\n",
    "test_set=torchvision.datasets.CIFAR10(root='./',train=False,download=False,transform=transform)\n",
    "test_loader=torch.utils.data.DataLoader(test_set,batch_size=batch_size,shuffle=True)\n",
    "optimizerVAE=optim.Adam(HVAEmodel.parameters(),lr=lr)\n",
    "#optimizerEVAE=optim.Adam(EVAEmodel.parameters(),lr=lr)\n",
    "criterion=nn.BCELoss(reduction='none')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "d2994894",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch1 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:13<00:00, 36.48it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 64.20it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:627.6968\n",
      "valid loss:19.9942\n",
      "Epoch2 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 34.79it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 61.10it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:603.9423\n",
      "valid loss:19.5677\n",
      "Epoch3 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 33.54it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.91it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:597.8860\n",
      "valid loss:19.4222\n",
      "Epoch4 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.90it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 41.72it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:596.0655\n",
      "valid loss:19.3277\n",
      "Epoch5 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.14it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 58.37it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:594.3368\n",
      "valid loss:19.2743\n",
      "Epoch6 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.59it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 58.43it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:593.2000\n",
      "valid loss:19.2178\n",
      "Epoch7 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 33.88it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 55.44it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:592.5968\n",
      "valid loss:19.2047\n",
      "Epoch8 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 33.44it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 43.17it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:592.1173\n",
      "valid loss:19.1888\n",
      "Epoch9 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.65it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 59.79it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:591.8514\n",
      "valid loss:19.1679\n",
      "Epoch10 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 33.35it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 45.16it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:591.4183\n",
      "valid loss:19.1684\n",
      "Epoch11 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.69it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 59.57it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:591.2315\n",
      "valid loss:19.1341\n",
      "Epoch12 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.16it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 53.15it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:590.8159\n",
      "valid loss:19.1313\n",
      "Epoch13 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.85it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 58.49it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:590.5560\n",
      "valid loss:19.1136\n",
      "Epoch14 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 34.22it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 57.13it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:590.3085\n",
      "valid loss:19.1236\n",
      "Epoch15 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.58it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 55.18it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:590.2105\n",
      "valid loss:19.1258\n",
      "Epoch16 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.68it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 53.27it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:590.0836\n",
      "valid loss:19.1021\n",
      "Epoch17 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 31.41it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 55.35it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.9352\n",
      "valid loss:19.0931\n",
      "Epoch18 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 31.24it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 55.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.8240\n",
      "valid loss:19.0961\n",
      "Epoch19 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.20it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.59it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.7557\n",
      "valid loss:19.0948\n",
      "Epoch20 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 33.06it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 44.69it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.7339\n",
      "valid loss:19.0890\n",
      "Epoch21 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.21it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.85it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.5858\n",
      "valid loss:19.0885\n",
      "Epoch22 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 30.99it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 50.70it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.5290\n",
      "valid loss:19.0828\n",
      "Epoch23 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 30.61it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.69it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.4991\n",
      "valid loss:19.0716\n",
      "Epoch24 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.34it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.83it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.3591\n",
      "valid loss:19.0799\n",
      "Epoch25 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 31.57it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 43.57it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.3241\n",
      "valid loss:19.0711\n",
      "Epoch26 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 31.64it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 59.70it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.3093\n",
      "valid loss:19.0877\n",
      "Epoch27 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 35.64it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.90it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.1855\n",
      "valid loss:19.0717\n",
      "Epoch28 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:13<00:00, 36.08it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 57.33it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.1740\n",
      "valid loss:19.0753\n",
      "Epoch29 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 34.85it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 57.19it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.0808\n",
      "valid loss:19.0637\n",
      "Epoch30 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 34.43it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 52.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.9571\n",
      "valid loss:19.0742\n",
      "Epoch31 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 34.33it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 54.08it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:589.0038\n",
      "valid loss:19.0720\n",
      "Epoch32 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 31.92it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 43.35it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.9112\n",
      "valid loss:19.0797\n",
      "Epoch33 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 30.55it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 49.74it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.8790\n",
      "valid loss:19.0644\n",
      "Epoch34 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 33.88it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 54.65it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.8483\n",
      "valid loss:19.0745\n",
      "Epoch35 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 34.41it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 50.95it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.7658\n",
      "valid loss:19.0727\n",
      "Epoch36 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 30.87it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 47.86it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.7158\n",
      "valid loss:19.0534\n",
      "Epoch37 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 30.73it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 46.62it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.6827\n",
      "valid loss:19.0586\n",
      "Epoch38 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 30.24it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 47.13it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.6578\n",
      "valid loss:19.0660\n",
      "Epoch39 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 30.99it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 50.12it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.7044\n",
      "valid loss:19.0588\n",
      "Epoch40 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 32.34it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.08it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.5991\n",
      "valid loss:19.0683\n",
      "Epoch41 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:13<00:00, 35.74it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 59.32it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.5185\n",
      "valid loss:19.0658\n",
      "Epoch42 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 34.55it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 55.86it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.5027\n",
      "valid loss:19.0574\n",
      "Epoch43 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 34.69it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.70it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.5002\n",
      "valid loss:19.0579\n",
      "Epoch44 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 35.20it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 57.08it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.4706\n",
      "valid loss:19.0547\n",
      "Epoch45 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:14<00:00, 34.06it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 55.33it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.3768\n",
      "valid loss:19.0576\n",
      "Epoch46 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 31.26it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 50.58it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.4814\n",
      "valid loss:19.0642\n",
      "Epoch47 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 30.60it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 55.03it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.3488\n",
      "valid loss:19.0590\n",
      "Epoch48 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 31.15it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 52.54it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.3478\n",
      "valid loss:19.0506\n",
      "Epoch49 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:15<00:00, 31.90it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.14it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.2789\n",
      "valid loss:19.0480\n",
      "Epoch50 of 50\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:16<00:00, 31.03it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 54.42it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train loss:588.3245\n",
      "valid loss:19.0569\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "import math\n",
    "grid_imagesVAE=[]\n",
    "train_lossVAE=[]\n",
    "valid_lossVAE=[]\n",
    "\n",
    "grid_imagesEVAE=[]\n",
    "train_lossEVAE=[]\n",
    "valid_lossEVAE=[]\n",
    "\n",
    "for epoch in range(epochs):\n",
    "    print(f\"Epoch{epoch+1} of {epochs}\")\n",
    "\n",
    "    \n",
    "    train_epoch_lossVAE=model_trainHVAE(HVAEmodel,train_loader,train_set,device,optimizerVAE,criterion)\n",
    "    valid_epoch_lossVAE=model_validateHVAE(HVAEmodel,test_loader,test_set,device,optimizerVAE,criterion)\n",
    "    train_lossVAE.append(train_epoch_lossVAE)\n",
    "    valid_lossVAE.append(valid_epoch_lossVAE)\n",
    "    print(f\"train loss:{train_epoch_lossVAE:.4f}\")\n",
    "    print(f\"valid loss:{valid_epoch_lossVAE:.4f}\")\n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f97eaddc",
   "metadata": {},
   "source": [
    "#Reconstructed image"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8d80202a",
   "metadata": {},
   "outputs": [],
   "source": [
    "VAEmodel.eval()\n",
    "EVAEmodel.eval()\n",
    "index_batch=torch.randperm(train_loader.batch_size)[0]\n",
    "index_sample=torch.randperm(train_loader.batch_size)[:64]\n",
    "with torch.no_grad():\n",
    "    for i,data in tqdm(enumerate(test_loader),total=int(len(test_set)/test_loader.batch_size)):\n",
    "        \n",
    "        data=data[0]\n",
    "        data=data.to(device)\n",
    "        optimizerVAE.zero_grad()\n",
    "        optimizerEVAE.zero_grad()\n",
    "        if i==index_batch.cpu():\n",
    "            reconstruction_VAE,mu,log_r=VAEmodel(data[index_sample])\n",
    "            reconstruction_EVAE,mu,log_r=EVAEmodel(data[index_sample])\n",
    "            real_images=data[index_sample]\n",
    "            \n",
    "            save_reconstructed_imagesVAE(reconstruction_VAE,0)\n",
    "            save_reconstructed_imagesEVAE(reconstruction_EVAE,0)\n",
    "            save_reconstructed_imagesREAL(real_images,0)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "d7710d1a",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  1%|          | 1/100 [00:01<02:56,  1.78s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(220.6418, device='cuda:0')\n",
      "3.287424243525539\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  2%|▏         | 2/100 [00:03<03:08,  1.92s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(213.5926, device='cuda:0')\n",
      "6.657649930089505\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  3%|▎         | 3/100 [00:06<03:32,  2.19s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(208.5847, device='cuda:0')\n",
      "9.942700957585298\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  4%|▍         | 4/100 [00:10<04:33,  2.85s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(207.2882, device='cuda:0')\n",
      "13.373491522547118\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  5%|▌         | 5/100 [00:13<04:57,  3.13s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(207.1350, device='cuda:0')\n",
      "16.82397122287099\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  6%|▌         | 6/100 [00:18<05:52,  3.75s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(206.9123, device='cuda:0')\n",
      "20.138305074478506\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  7%|▋         | 7/100 [00:22<05:58,  3.86s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(207.0644, device='cuda:0')\n",
      "23.350443368442352\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  8%|▊         | 8/100 [00:27<06:28,  4.22s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(206.1119, device='cuda:0')\n",
      "26.62812724373597\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  9%|▉         | 9/100 [00:33<07:16,  4.80s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(204.9682, device='cuda:0')\n",
      "30.276790297604514\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 10%|█         | 10/100 [00:39<07:41,  5.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(205.3398, device='cuda:0')\n",
      "34.03080154025984\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 11%|█         | 11/100 [00:45<08:02,  5.42s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(205.3677, device='cuda:0')\n",
      "37.33696896342124\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 12%|█▏        | 12/100 [00:52<08:16,  5.64s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(205.2000, device='cuda:0')\n",
      "40.68161593451881\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 13%|█▎        | 13/100 [00:58<08:20,  5.76s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(204.3864, device='cuda:0')\n",
      "44.32108731178452\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 14%|█▍        | 14/100 [01:04<08:28,  5.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(204.0481, device='cuda:0')\n",
      "47.83037260303535\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 15%|█▌        | 15/100 [01:10<08:29,  5.99s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(203.4926, device='cuda:0')\n",
      "51.22727388372851\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 16%|█▌        | 16/100 [01:17<08:39,  6.19s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(202.9118, device='cuda:0')\n",
      "54.82636343259897\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 17%|█▋        | 17/100 [01:23<08:33,  6.19s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(202.9613, device='cuda:0')\n",
      "58.20037440493692\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 18%|█▊        | 18/100 [01:29<08:29,  6.22s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(202.1507, device='cuda:0')\n",
      "62.15500643840451\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 19%|█▉        | 19/100 [01:35<08:12,  6.08s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(201.8468, device='cuda:0')\n",
      "65.70426980622226\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 20%|██        | 20/100 [01:41<07:58,  5.98s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(201.8913, device='cuda:0')\n",
      "69.09400511537517\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 21%|██        | 21/100 [01:47<08:03,  6.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(200.5730, device='cuda:0')\n",
      "72.72162385917598\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 22%|██▏       | 22/100 [01:53<08:02,  6.19s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(200.5417, device='cuda:0')\n",
      "76.25771377844693\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 23%|██▎       | 23/100 [02:00<08:03,  6.28s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(200.1129, device='cuda:0')\n",
      "79.87996552399055\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 24%|██▍       | 24/100 [02:06<07:58,  6.29s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(200.0880, device='cuda:0')\n",
      "83.040849537437\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 25%|██▌       | 25/100 [02:12<07:50,  6.27s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(199.7983, device='cuda:0')\n",
      "86.65319338593116\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 26%|██▌       | 26/100 [02:19<07:41,  6.23s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(199.4106, device='cuda:0')\n",
      "90.1118999711596\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 27%|██▋       | 27/100 [02:25<07:31,  6.18s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(198.9823, device='cuda:0')\n",
      "93.39865168595654\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 28%|██▊       | 28/100 [02:31<07:32,  6.28s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(199.0121, device='cuda:0')\n",
      "96.94936616329731\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 29%|██▉       | 29/100 [02:38<07:30,  6.34s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(198.4491, device='cuda:0')\n",
      "100.61802749704795\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 30%|███       | 30/100 [02:43<07:08,  6.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(198.6896, device='cuda:0')\n",
      "104.08766681519955\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 31%|███       | 31/100 [02:50<07:07,  6.20s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(198.5609, device='cuda:0')\n",
      "107.6946352832121\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 32%|███▏      | 32/100 [02:56<07:06,  6.27s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(198.2892, device='cuda:0')\n",
      "111.40323500614394\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 33%|███▎      | 33/100 [03:02<07:00,  6.27s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(198.0479, device='cuda:0')\n",
      "115.0950533617056\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 34%|███▍      | 34/100 [03:09<07:02,  6.41s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(197.5531, device='cuda:0')\n",
      "118.78530222671311\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 35%|███▌      | 35/100 [03:15<06:52,  6.35s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(197.3673, device='cuda:0')\n",
      "122.38333714618413\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 36%|███▌      | 36/100 [03:22<06:55,  6.50s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(197.2574, device='cuda:0')\n",
      "125.73780347133714\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 37%|███▋      | 37/100 [03:29<06:47,  6.47s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.9243, device='cuda:0')\n",
      "129.51882295130477\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 38%|███▊      | 38/100 [03:35<06:36,  6.40s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(197.0759, device='cuda:0')\n",
      "133.0507110445139\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 39%|███▉      | 39/100 [03:41<06:28,  6.36s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.7958, device='cuda:0')\n",
      "136.960683381414\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 40%|████      | 40/100 [03:47<06:22,  6.38s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.6130, device='cuda:0')\n",
      "140.2221828028888\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 41%|████      | 41/100 [03:54<06:20,  6.46s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.6553, device='cuda:0')\n",
      "143.607052240405\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 42%|████▏     | 42/100 [04:01<06:19,  6.55s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.4684, device='cuda:0')\n",
      "147.33369171215801\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 43%|████▎     | 43/100 [04:07<06:07,  6.45s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.4041, device='cuda:0')\n",
      "151.02520183670117\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 44%|████▍     | 44/100 [04:13<05:58,  6.41s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.4921, device='cuda:0')\n",
      "154.5003493558022\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 45%|████▌     | 45/100 [04:20<05:54,  6.45s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.6976, device='cuda:0')\n",
      "157.95659432035637\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 46%|████▌     | 46/100 [04:26<05:42,  6.35s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.7276, device='cuda:0')\n",
      "161.38339333855896\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 47%|████▋     | 47/100 [04:33<05:39,  6.41s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.7228, device='cuda:0')\n",
      "165.12609707227307\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 48%|████▊     | 48/100 [04:38<05:19,  6.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.4839, device='cuda:0')\n",
      "169.26397952354998\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 49%|████▉     | 49/100 [04:44<05:13,  6.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.3178, device='cuda:0')\n",
      "172.83250809773753\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 50%|█████     | 50/100 [04:50<04:57,  5.95s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.2493, device='cuda:0')\n",
      "176.54670961673617\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 51%|█████     | 51/100 [04:56<04:58,  6.09s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.3076, device='cuda:0')\n",
      "180.15622040298643\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 52%|█████▏    | 52/100 [05:03<04:56,  6.18s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.3142, device='cuda:0')\n",
      "183.64848759317326\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 53%|█████▎    | 53/100 [05:10<05:02,  6.44s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.2684, device='cuda:0')\n",
      "187.1768409645303\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 54%|█████▍    | 54/100 [05:16<04:57,  6.47s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.2246, device='cuda:0')\n",
      "190.7188534771995\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 55%|█████▌    | 55/100 [05:22<04:38,  6.19s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.3701, device='cuda:0')\n",
      "194.08622739323016\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 56%|█████▌    | 56/100 [05:27<04:22,  5.95s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.3772, device='cuda:0')\n",
      "197.30548790129257\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 57%|█████▋    | 57/100 [05:34<04:24,  6.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.3690, device='cuda:0')\n",
      "201.05941447617855\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 58%|█████▊    | 58/100 [05:41<04:27,  6.38s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.3378, device='cuda:0')\n",
      "204.82448052839135\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 59%|█████▉    | 59/100 [05:46<04:14,  6.21s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.2180, device='cuda:0')\n",
      "208.4645631691637\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 60%|██████    | 60/100 [05:52<04:03,  6.09s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9272, device='cuda:0')\n",
      "212.40213365017502\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 61%|██████    | 61/100 [05:58<03:57,  6.10s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8552, device='cuda:0')\n",
      "215.7534234748233\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 62%|██████▏   | 62/100 [06:05<03:52,  6.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9149, device='cuda:0')\n",
      "219.21548880838873\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 63%|██████▎   | 63/100 [06:10<03:43,  6.04s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9518, device='cuda:0')\n",
      "222.78570084421227\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 64%|██████▍   | 64/100 [06:16<03:35,  5.99s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8311, device='cuda:0')\n",
      "226.35997174324\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 65%|██████▌   | 65/100 [06:22<03:29,  5.99s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8161, device='cuda:0')\n",
      "230.31044102881984\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 66%|██████▌   | 66/100 [06:28<03:22,  5.96s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8266, device='cuda:0')\n",
      "233.70120502413627\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 67%|██████▋   | 67/100 [06:34<03:13,  5.86s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8757, device='cuda:0')\n",
      "237.1228593378206\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 68%|██████▊   | 68/100 [06:39<03:02,  5.71s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.7902, device='cuda:0')\n",
      "240.78727111074446\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 69%|██████▉   | 69/100 [06:45<02:57,  5.72s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8601, device='cuda:0')\n",
      "244.03509292895504\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 70%|███████   | 70/100 [06:51<02:54,  5.81s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9316, device='cuda:0')\n",
      "247.42616592445208\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 71%|███████   | 71/100 [06:57<02:55,  6.04s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.0416, device='cuda:0')\n",
      "250.77214546326155\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 72%|███████▏  | 72/100 [07:04<02:50,  6.09s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.0310, device='cuda:0')\n",
      "254.18140450109087\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 73%|███████▎  | 73/100 [07:09<02:38,  5.87s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8815, device='cuda:0')\n",
      "258.04608546328353\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 74%|███████▍  | 74/100 [07:15<02:32,  5.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8640, device='cuda:0')\n",
      "261.30892307176606\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 75%|███████▌  | 75/100 [07:21<02:26,  5.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9687, device='cuda:0')\n",
      "264.3213599361758\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 76%|███████▌  | 76/100 [07:27<02:19,  5.83s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9903, device='cuda:0')\n",
      "267.47076221211523\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 77%|███████▋  | 77/100 [07:32<02:12,  5.74s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.0137, device='cuda:0')\n",
      "271.1204990630088\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 78%|███████▊  | 78/100 [07:38<02:04,  5.66s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.0587, device='cuda:0')\n",
      "274.5413697990952\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 79%|███████▉  | 79/100 [07:44<02:02,  5.85s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.0241, device='cuda:0')\n",
      "278.0723307228457\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 80%|████████  | 80/100 [07:50<01:57,  5.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.0808, device='cuda:0')\n",
      "281.7816762444218\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 81%|████████  | 81/100 [07:55<01:50,  5.82s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.1329, device='cuda:0')\n",
      "285.0978747171496\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 82%|████████▏ | 82/100 [08:01<01:45,  5.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.1401, device='cuda:0')\n",
      "288.4862187507254\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 83%|████████▎ | 83/100 [08:08<01:42,  6.05s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.0506, device='cuda:0')\n",
      "292.0670531258054\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 84%|████████▍ | 84/100 [08:14<01:38,  6.14s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.1388, device='cuda:0')\n",
      "295.2267515500417\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 85%|████████▌ | 85/100 [08:20<01:31,  6.07s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.0903, device='cuda:0')\n",
      "298.3936959821976\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 86%|████████▌ | 86/100 [08:26<01:24,  6.00s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9833, device='cuda:0')\n",
      "301.91245848468714\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 87%|████████▋ | 87/100 [08:32<01:19,  6.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9808, device='cuda:0')\n",
      "305.3300225958058\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 88%|████████▊ | 88/100 [08:39<01:14,  6.20s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8716, device='cuda:0')\n",
      "309.20065367047243\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 89%|████████▉ | 89/100 [08:45<01:06,  6.09s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9755, device='cuda:0')\n",
      "312.38455318001894\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 90%|█████████ | 90/100 [08:51<01:01,  6.13s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(196.0150, device='cuda:0')\n",
      "315.8872164780213\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 91%|█████████ | 91/100 [08:56<00:53,  5.92s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9535, device='cuda:0')\n",
      "319.17425381327274\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 92%|█████████▏| 92/100 [09:03<00:48,  6.11s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9098, device='cuda:0')\n",
      "322.5343675272192\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 93%|█████████▎| 93/100 [09:09<00:43,  6.18s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8726, device='cuda:0')\n",
      "325.7648229924701\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 94%|█████████▍| 94/100 [09:15<00:36,  6.17s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8478, device='cuda:0')\n",
      "328.850708728192\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 95%|█████████▌| 95/100 [09:21<00:30,  6.01s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.7907, device='cuda:0')\n",
      "332.408134722841\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 96%|█████████▌| 96/100 [09:27<00:23,  5.89s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.7435, device='cuda:0')\n",
      "336.192076555614\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 97%|█████████▋| 97/100 [09:32<00:17,  5.90s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.7970, device='cuda:0')\n",
      "340.077578133056\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 98%|█████████▊| 98/100 [09:38<00:11,  5.65s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.9308, device='cuda:0')\n",
      "343.7045426055252\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 99%|█████████▉| 99/100 [09:44<00:05,  5.75s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8676, device='cuda:0')\n",
      "347.11931000433185\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [09:49<00:00,  5.90s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(195.8053, device='cuda:0')\n",
      "350.31353971446526\n",
      "FIDEVAE: 195.8052978515625\n",
      "shaprnessEVAE:0.0350\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "#VAEmodel.train()\n",
    "#EVAEmodel.train()\n",
    "HVAEmodel.eval()\n",
    "running_loss=0.0\n",
    "counter=0\n",
    "#test set\n",
    "tota_sharpVAE=0.0\n",
    "tota_sharpHVAEmodel=0.0\n",
    "from scipy import signal\n",
    "#laplace\n",
    "kernel=np.array([[0 ,1, 0],[1, -4,1],[0, 1 ,0]])\n",
    "\n",
    "\n",
    "from torcheval.metrics import FrechetInceptionDistance\n",
    "#fidVAE = FrechetInceptionDistance(device=device)\n",
    "fidHVAE = FrechetInceptionDistance(device=device)            \n",
    "\n",
    "with torch.no_grad():\n",
    "    for i,data in tqdm(enumerate(test_loader),total=int(len(test_set)/test_loader.batch_size)):\n",
    "        counter+=1\n",
    "        data=data[0]\n",
    "        data=data.to(device)\n",
    "        #optimizerVAE.zero_grad()\n",
    "        optimizerVAE.zero_grad()\n",
    "\n",
    "        #reconstruction_VAE,mu,log_r=VAEmodel(data)\n",
    "        reconstruction,*_=HVAEmodel(data)\n",
    "        \n",
    "        #mage_grid_VAE=reconstruction_VAE.detach().cpu()\n",
    "        image_grid_HVAE=reconstruction.detach().cpu()\n",
    "        for i in range(data.size(dim=0)):\n",
    "\n",
    "#             sharpnessVAE = np.var(np.abs(signal.convolve2d(image_grid_VAE[i][0].cpu(), kernel, mode=\"same\")))\n",
    "#             tota_sharpVAE+=sharpnessVAE\n",
    "            sharpnessHVAE = np.var(np.abs(signal.convolve2d(image_grid_HVAE[i][0].cpu(), kernel, mode=\"same\")))\n",
    "            tota_sharpHVAEmodel+=sharpnessHVAE\n",
    "\n",
    "#         fidVAE.update(data.repeat(1, 3, 1,1), is_real=True)\n",
    "#         fidVAE.update(reconstruction_VAE.repeat(1, 3, 1,1), is_real=False)\n",
    "\n",
    "        fidHVAE.update(data, is_real=True)\n",
    "        fidHVAE.update(reconstruction, is_real=False)\n",
    "        \n",
    "        #lossVAE=fidVAE.compute()#reconstruction_error#final_loss(reconstruction_error,mean,log_var)\n",
    "        lossHVAE=fidHVAE.compute()\n",
    "        #print(lossVAE)\n",
    "        #print(tota_sharpVAE)\n",
    "        print(lossHVAE)\n",
    "        print(tota_sharpHVAEmodel)\n",
    "\n",
    "\n",
    "print(f\"FIDEVAE: {float(lossHVAE)}\")\n",
    "print(f\"shaprnessEVAE:{tota_sharpHVAEmodel/len(test_set):.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 97,
   "id": "f91b1656",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 5, 1, 32, 32])"
      ]
     },
     "execution_count": 97,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "reconstruction.size()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "72d7d9f7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA2QAAAJaCAYAAABA74ShAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAACCoElEQVR4nO3dd3hUZfrG8XtSSUIyJIEQIlGQJjHIKiDNFRBMQIrYQMEIimABskhR8bcWXBULCpa1rAXssC5lURQBQcpSBaMiiI0ShFAkTCCE1PP7Y5hJJgkwAzM5Q/L9XNe55sw5Z2aeyc7levu873sshmEYAgAAAABUuQCzCwAAAACAmopABgAAAAAmIZABAAAAgEkIZAAAAABgEgIZAAAAAJiEQAYAAAAAJiGQAQAAAIBJCGQAAAAAYJIgswuoTkpKSrRnzx5FRkbKYrGYXQ4AAAAAkxiGoSNHjighIUEBASfvgxHIvGjPnj1KTEw0uwwAAAAAfiIzM1MNGzY86XkCmRdFRkZKsv/Ro6KiTK4GAAAAgFlycnKUmJjozAgnQyDzIscwxaioKAIZAAAAgNNOZWJRDwAAAAAwCYEMAAAAAExCIAMAAAAAkzCHDAAAANWWYRgqKipScXGx2aWgmgkMDFRQUNBZ3+6KQAYAAIBqqaCgQHv37tWxY8fMLgXVVHh4uBo0aKCQkJAzfg8CGQAAAKqdkpISbd++XYGBgUpISFBISMhZdzIAB8MwVFBQoAMHDmj79u1q1qzZKW/+fCoEMgAAAFQ7BQUFKikpUWJiosLDw80uB9VQWFiYgoODtXPnThUUFKhWrVpn9D4s6gEAAIBq60y7FoA7vPH74hcKAAAAACYhkAEAAACASQhkAAAAQDXWqFEjTZs2zfT3QOUIZAAAAIAf6dq1q8aMGeO199uwYYNGjBjhtferSq1atdKdd95Z6bmPP/5YwcHB2rdvn/NYSkqKAgMDtXbt2grXDx06VBaLpcLWs2dPn9XvDgIZAAAAcI5x3PDaHfXq1TtnV5ocNmyY/v3vf1d6L7l33nlHffr0Uf369SVJu3bt0po1azRq1Ci9/fbblb5fz549tXfvXpft448/9ul3OB0CGQAAAGoGw5CKcs3ZDMOtEocOHarly5frxRdfdHZwduzYoa+//loWi0Vffvml2rZtq9DQUK1cuVK//fabrr32WtWvX1+1a9dWu3bttGTJEpf3LD/c0GKx6K233tJ1112n8PBwNWvWTPPnz/foT7lr1y5de+21ql27tqKiojRgwACXTtV3332nbt26KTIyUlFRUWrTpo2++eYbSdLOnTvVt29fRUdHKyIiQhdffLE+//zzSj8nLS1N+fn5+uSTTyp8/tKlSzVs2DDnsenTp6tPnz665557NGvWLOXm5lZ4v9DQUMXHx7ts0dHRHn13b+M+ZAAAAKgZio9J/65tzmcPOCoFRZz2shdffFE///yzkpOT9fjjj0uyd7h27NghSbr//vs1ZcoUXXjhhapTp452796ta665Rk888YRq1aqld999V3379tW2bdt0/vnnn/RzJk2apGeffVbPPfecXn75ZQ0ePFg7d+5UTEzMaWs0DEP9+/dXRESEli9frqKiIt17770aOHCgvv76a0nS4MGDdemll+q1115TYGCgMjIyFBwcLEkaOXKkCgoKtGLFCkVERGjLli2qXbvy/11iY2N17bXXavr06RoyZIjz+PTp01W/fn316tXLWdP06dP1z3/+UxdddJGaN2+uf//737r99ttP+33MZmqHbPLkyWrXrp0iIyMVFxen/v37a9u2bSe9/q677pLFYqkwoTA/P1+jR49W3bp1FRERoX79+mn37t0u12RnZystLU1Wq1VWq1VpaWk6fPiwyzW7du1S3759FRERobp16yo9PV0FBQXe+roAAADAKVmtVoWEhCg8PNzZwQkMDHSef/zxx3X11VerSZMmio2NVevWrXXXXXepVatWatasmZ544gldeOGFp+14DR06VLfccouaNm2qp556Srm5uVq/fr1bNS5ZskTff/+9PvroI7Vp00bt27fX+++/r+XLl2vDhg2S7P9e3aNHD1100UVq1qyZbrrpJrVu3dp5rnPnzmrVqpUuvPBC9enTR1deeeVJP++OO+7QihUr9Pvvv0uyh68ZM2Zo6NChzr/NkiVLdOzYMaWmpkqSbr311kqHLX722WeqXbu2y/aPf/zDre/tK6Z2yJYvX66RI0eqXbt2Kioq0v/93/8pJSVFW7ZsUUSE639BmDdvntatW6eEhIQK7zNmzBh9+umnmjlzpmJjYzVu3Dj16dNHGzdudP6PNGjQIO3evVsLFy6UJI0YMUJpaWn69NNPJUnFxcXq3bu36tWrp1WrVunPP//UkCFDZBiGXn75ZR//JQAAAOBzgeH2TpVZn+0Fbdu2dXmem5urSZMm6bPPPtOePXtUVFSkvLw87dq165Tvc8kllzj3IyIiFBkZqf3797tVw9atW5WYmKjExETnsaSkJNWpU0dbt25Vu3btNHbsWN155516//331aNHD910001q0qSJJCk9PV333HOPFi1apB49euiGG25wqae8lJQUNWzYUNOnT9c//vEPLV26VDt27HDpfr399tsaOHCggoLs8eaWW27RhAkTtG3bNrVo0cJ5Xbdu3fTaa6+5vL87XUFfMrVDtnDhQg0dOlQXX3yxWrdurenTp2vXrl3auHGjy3V//PGHRo0apQ8//NDZ6nSw2Wx6++239fzzz6tHjx669NJL9cEHH+iHH35wjp/dunWrFi5cqLfeeksdO3ZUx44d9eabb+qzzz5zduQWLVqkLVu26IMPPtCll16qHj166Pnnn9ebb76pnJycqvmDAAAAwHcsFvuwQTM2i8UrX6F802LChAmaPXu2nnzySa1cuVIZGRlq1arVaUd5lf93aovFopKSErdqMAxDlkq+T9njjz32mH788Uf17t1bS5cuVVJSkubOnStJuvPOO/X7778rLS1NP/zwg9q2bXvKBkhAQICGDh2qd999VyUlJZo+fbquvPJKNWvWTJJ06NAhzZs3T6+++qqCgoIUFBSk8847T0VFRXrnnXdc3isiIkJNmzZ12Wp0ICvPZrNJck2pJSUlSktL04QJE3TxxRdXeM3GjRtVWFiolJQU57GEhAQlJydr9erVkqQ1a9bIarWqffv2zms6dOggq9Xqck1ycrJLBy41NVX5+fkVAqJDfn6+cnJyXDYAAADgbISEhKi4uNita1euXKmhQ4fquuuuU6tWrRQfH++cb+YrSUlJ2rVrlzIzM53HtmzZIpvNppYtWzqPNW/eXPfdd58WLVqk66+/XtOnT3eeS0xM1N133605c+Zo3LhxevPNN0/5mbfffrt2796tOXPmaM6cOS6LeXz44Ydq2LChvvvuO2VkZDi3adOm6d1333V7NUqz+E0gMwxDY8eO1RVXXKHk5GTn8WeeeUZBQUFKT0+v9HVZWVkKCQmpsDpK/fr1lZWV5bwmLi6uwmvj4uJcrnEsmekQHR2tkJAQ5zXlTZ482TknzWq1urRtAQAAgDPRqFEjrVu3Tjt27NDBgwdP2blq2rSp5syZ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JtWubc527JhbJVqtVoWEhLh0hAIDA53nH3/8cV199dVq0qSJYmNj1bp1a911111q1aqVmjVrpieeeEIXXnjhaTteQ4cO1S233KKmTZvqqaeeUm5urtavX+9WjUuWLNH333+vjz76SG3atFH79u31/vvva/ny5dqwYYMkewetR48euuiii9SsWTPddNNNat26tfNc586d1apVK1144YXq06ePrrzyylN+5quvvqratWu7bO+++64kKTAwULfddptmzJgh40Tw3b59u5YvX65hw4Y53+Odd95Rr169nHPIevbsqXfeeafCZz3wwAMVPuvrr792629zJghk1RVzyAAAAKqdtm3bujzPzc3V/fffr6SkJNWpU0e1a9fWTz/9dNoOWdmFLiIiIhQZGan9+/e7VcPWrVuVmJjo0rlzfP7WrVslSWPHjtWdd96pHj166Omnn9Zvv/3mvDY9PV1PPPGEOnfurEcffVTff//9aT9z8ODBysjIcNmuu+465/lhw4Zp586dWrp0qSR7+GrYsKF69OghSSouLta7777rMnzx1ltv1bvvvqvi4mKXz5owYUKFz2rfvr1bf5szwaIe1VUQgQwAAMBFeLh09Kh5n+0F5ecyTZgwQV9++aWmTJmipk2bKiwsTDfeeKMKCgpO+T6O4YMOFotFJSUlbtVgGIYsFsspjz/22GMaNGiQFixYoC+++EKPPvqoZs6cqeuuu0533nmnUlNTtWDBAi1atEiTJ0/W888/r9GjR5/0M61Wq5o2bXrS882aNdNf//pXTZ8+Xd26ddO7776r22+/XQEB9v7Tl19+qT/++MNlWKVkD2qLFi1Sr169nMfq1q17ys/yNjpk1ZVjyGIRQxYBAAAkSRaLFBFhzlZJgDmZkJCQCl2bk1m5cqWGDh2q6667Tq1atVJ8fHyFxS28LSkpSbt27VJmZqbz2JYtW2Sz2dSyZUvnsebNm+u+++7TokWLdP3112v69OnOc4mJibr77rs1Z84cjRs3Tm+++eZZ1zVs2DDNmTNHs2fP1u7du3X77bc7z7399tu6+eabK3S+Bg8efNLFPaoKHbLqiiGLAAAA56RGjRpp3bp12rFjh2rXrq2YmJiTXtu0aVPNmTNHffv2lcVi0cMPP+x2p+tM9ejRQ5dccokGDx6sadOmqaioSPfee6+6dOmitm3bKi8vTxMmTNCNN96o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UT399NPq2bOnFi1apOuuu07XX3+9li9fLknKyspSSEiIoqOjXV5bv359ZWVlOa+Ji4ur8P5xcXEu19SvX9/lfHR0tEJCQpzXVGby5MnOeWlWq1WJiYln9Z194kzmkDkEhkrtXrXv//KKdGiT9+oCAAAA4Hkg+/7779W8eXM988wzeu6555yLY8ydO1cTJ070WmElJSWSpGuvvVb33Xef/vKXv+jBBx9Unz599Prrr5/ytYZhyGKxOJ+X3T+ba8qbOHGibDabc8vMzDzt96pyZ9Mhk6T4HtIFN0tGibThHqmk2Hu1AQAAADWcx4Fs7NixGjp0qH755ReXxTV69erl1fuQ1a1bV0FBQRVWbmzZsqVzlcX4+HgVFBQoOzvb5Zr9+/c7O17x8fHat29fhfc/cOCAyzXlO2HZ2dkqLCys0DkrKzQ0VFFRUS6b3zmbDpnDZS9IQZHSn+ul397yTl0AAAAAPA9kGzZs0F133VXh+HnnnXfK4X2eCgkJUbt27bRt2zaX4z///LMuuOACSVKbNm0UHBysxYsXO8/v3btXmzdvVqdOnSRJHTt2lM1m0/r1653XrFu3TjabzeWazZs3a+/evc5rFi1apNDQULVp08Zr38kUZ9shk6SwBlLrJ+z7GQ9Kx/effV0AAAAAPF/2vlatWpUuXrFt2zbVq1fPo/c6evSofv31V+fz7du3KyMjQzExMTr//PM1YcIEDRw4UFdeeaW6deumhQsX6tNPP9XXX38tSbJarRo2bJjGjRun2NhYxcTEaPz48WrVqpV69Oghyd5R69mzp4YPH6433nhDkn3Z+z59+qhFixaSpJSUFCUlJSktLU3PPfecDh06pPHjx2v48OH+2fXyhKNDdvSoVFAgVXJTb7c0u1f6fYaU/a307QSp47teKxEAAACoqTzukF177bV6/PHHVVhYKMk+92rXrl168MEHK72X2Kl88803uvTSS3XppZdKsg+HvPTSS/XII/YV/a677jq9/vrrevbZZ9WqVSu99dZbmj17tq644grne0ydOlX9+/fXgAED1LlzZ4WHh+vTTz913oNMkj788EO1atVKKSkpSklJ0SWXXKL333/feT4wMFALFixQrVq11LlzZw0YMED9+/fXlClTPP3z+B+r1X5fMensumQBQVK71yVZpO3vSfuWe6U8AAAAoCazGIZhePKCnJwcXXPN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AQAAAIBJCGQAAAAAYBICGQAAAACYhEAGAAAAACYhkAEAAACASQhkAACYxGKxaN68eWaXAQAwEYEMAFAjDR06VBaLpcLWs2dPs0sDANQgQWYXAACAWXr27Knp06e7HAsNDTWpGgBATUSHDABQY4WGhio+Pt5li46OlmQfTvjaa6+pV69eCgsLU+PGjfXJJ5+4vP6HH37QVVddpbCwMMXGxmrEiBE6evSoyzXvvPOOLr74YoWGhqpBgwYaNWqUy/mDBw/quuuuU3h4uJo1a6b58+c7z2VnZ2vw4MGqV6+ewsLC1KxZswoBEgBwbiOQAQBwEg8//LBuuOEGfffdd7r11lt1yy23aOvWrZKkY8eOqWfPnoqOjtaGDRv0ySefaMmSJS6B67XXXtPIkSM1YsQI/fDDD5o/f76aNm3q8hmTJk3SgAED9P333+uaa67R4MGDdejQIefnb9myRV988YW2bt2q1157TXXr1q26PwAAwOcshmEYZhcBAEBVGzp0qD744APVqlXL5fgDDzyghx9+WBaLRXfffbdee+0157kOHTrosssu06uvvqo333xTDzzwgDIzMxURESFJ+vzzz9W3b1/t2bNH9evX13nnnafbb79dTzzxRKU1WCwW/f3vf9c//vEPSVJubq4iIyP1+eefq2fPnurXr5/q1q2rd955x0d/BQCA2ZhDBgCosbp16+YSuCQpJibGud+xY0eXcx07dlRGRoYkaevWrWrdurUzjElS586dVVJSom3btslisWjPnj3q3r37KWu45JJLnPsRERGKjIzU/v37JUn33HOPbrjhBm3atEkpKSnq37+/OnXqdEbfFQDgnwhkAIAaKyIiosIQwtOxWCySJMMwnPuVXRMWFubW+wUHB1d4bUlJiSSpV69e2rlzpxYsWKAlS5aoe/fuGjlypKZMmeJRzQAA/8UcMgAATmLt2rUVnl900UWSpKSkJGVkZCg3N9d5/n//+58CAgLUvHlzRUZGqlGjRvrqq6/OqoZ69eo5h1dOmzZN//rXv87q/QAA/oUOGQCgxsrPz1dWVpbLsaCgIOfCGZ988onatm2rK664Qh9++KHWr1+vt99+W5I0ePBgPfrooxoyZIgee+wxHThwQKNHj1ZaWprq168vSXrsscd09913Ky4uTr169dKRI0f0v//9T6NHj3arvkceeURt2rTRxRdfrPz8fH322Wdq2bKlF/8CAACzEcgAADXWwoUL1aBBA5djLVq00E8//STJvgLizJkzde+99yo+Pl4ffvihkpKSJEnh4eH68ssv9be//U3t2rVTeHi4brjhBr3wwgvO9xoyZIiOHz+uqVOnavz48apbt65uvPFGt+sLCQnRxIkTtWPHDoWFhemvf/2rZs6c6YVvDgDwF6yyCABAJSwWi+bOnav+/fubXQoAoBpjDhkAAAAAmIRABgAAAAAmYQ4ZAACVYEQ/AKAq0CEDAAAAAJMQyAAAAADAJAQyAAAAADAJgQwAAAAATEIgAwAAAACTEMgAAAAAwCQEMgAAAAAwCYEMAAAAAExCIAMAAAAAk/w/APLAOA81ht8AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1000x700 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "save_train_loss_plot(train_lossVAE,train_lossEVAE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "34268447",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA2QAAAJaCAYAAABA74ShAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAAB9gUlEQVR4nO3deVhWdf7/8dfNvogIIiCKWyqGW6RpaKsWai5ZNlaao+bot9IcU9umqaxfTZtlTU5ONZZmTlajlmauLZaZa1GaS2moqCimLAKyn98fR264BRRkOTfcz8d1nes+9zmHc943c9v48rPZDMMwBAAAAACodW5WFwAAAAAAropABgAAAAAWIZABAAAAgEUIZAAAAABgEQIZAAAAAFiEQAYAAAAAFiGQAQAAAIBFCGQAAAAAYBEPqwuoTwoLC3X06FEFBATIZrNZXQ4AAAAAixiGodOnTysiIkJubuW3gxHIqtHRo0cVGRlpdRkAAAAAnERiYqKaN29e7nkCWTUKCAiQZP7SGzZsaHE1AAAAAKySnp6uyMhIe0YoD4GsGhV1U2zYsCGBDAAAAMAFhzIxqQcAAAAAWIRABgAAAAAWIZABAAAAgEUYQwYAAIBqYxiG8vPzVVBQYHUpQI1yd3eXh4dHlZe7IpABAACgWuTm5iopKUlZWVlWlwLUCj8/PzVt2lReXl4XfQ8CGQAAAKqssLBQCQkJcnd3V0REhLy8vKrccgA4K8MwlJubqxMnTighIUHt2rU77+LP50MgAwAAQJXl5uaqsLBQkZGR8vPzs7ocoMb5+vrK09NTBw8eVG5urnx8fC7qPkzqAQAAgGpzsa0EQF1UHd93/sQAAAAAgEUIZAAAAABgEQIZAAAAUAWtWrXSq6++an9vs9n0ySeflHv9gQMHZLPZFB8fX6XnVtd9LmTMmDEaOnRojT7DlRHIAAAAgGqUlJSkAQMGVOs9ywpFkZGRSkpKUqdOnar1Wc7i/vvvV7t27co8d+TIEbm7u2vJkiX2Y//4xz/k7u6u559/vtT18+bNk81mK7Vd7EQc1YlABgAAAFSj8PBweXt71/hz3N3dFR4eLg+P+jlx+rhx47Rv3z59++23pc7NmzdPjRs31uDBg+3H3n33XT300EN65513yrxfw4YNlZSU5LAdPHiwxuqvKAIZAAAAaoZhSPmZtb8ZRoXKe/PNN9WsWTMVFhY6HB8yZIhGjx4tSdq/f79uvvlmhYWFqUGDBrriiiu0bt2689733C6LW7ZsUUxMjHx8fNS9e3f9+OOPDtcXFBRo3Lhxat26tXx9fRUVFaXXXnvNfn7GjBmaP3++Pv30U3vLztdff11ml8X169erR48e8vb2VtOmTfXII48oPz/ffv66667T5MmT9dBDDyk4OFjh4eGaMWNGhX5fRXJycjR58mSFhobKx8dHV111lbZu3Wo/n5KSopEjR6pJkyby9fVVu3bt9O6770oyl0eYNGmSmjZtKh8fH7Vq1UrPPfdcmc+57LLLdPnll5cZsObNm6c///nP8vT0tH/uM2fO6Omnn1ZmZqa++eabUj9js9kUHh7usIWFhVXqs9eE+hmnAQAAYL2CLOmjBrX/3OEZkof/BS/705/+pMmTJ+urr75S3759JZlhYvXq1Vq+fLkkKSMjQzfddJOeeeYZ+fj4aP78+Ro8eLD27t2rFi1aXPAZmZmZGjRokPr06aP3339fCQkJ+utf/+pwTWFhoZo3b66PPvpIISEh2rhxoyZMmKCmTZtq+PDhmj59unbv3q309HR7sAkODtbRo0cd7nPkyBHddNNNGjNmjN577z3t2bNH48ePl4+Pj0Pomj9/vqZOnarNmzfr+++/15gxY9S7d2/deOONF/w8kvTQQw9p8eLFmj9/vlq2bKkXX3xR/fr10759+xQcHKzHH39cu3bt0sqVKxUSEqJ9+/bpzJkzkqR//vOfWrZsmT766CO1aNFCiYmJSkxMLPdZ48aN00MPPaTXX39dDRqY36X169dr3759uvvuu+3XzZ07V3feeac8PT115513au7cubrmmmsq9HmsRiADAACASwoODlb//v313//+1x7IPv74YwUHB9vfd+3aVV27drX/zDPPPKOlS5dq2bJlmjRp0gWfsXDhQhUUFOidd96Rn5+fOnbsqMOHD+vee++1X+Pp6amnnnrK/r5169bauHGjPvroIw0fPlwNGjSQr6+vcnJyFB4eXu6z3njjDUVGRmr27Nmy2Wzq0KGDjh49qocfflhPPPGEfc2sLl266Mknn5QktWvXTrNnz9YXX3xRoUCWmZmpOXPmaN68efZxcm+//bbWrl2ruXPn6sEHH9ShQ4cUExOj7t27SzInPSly6NAhtWvXTldddZVsNptatmx53ueNGDFC06ZN08cff6yxY8dKkt555x3FxsYqOjpakpSenq7Fixdr48aNkqS77rpLvXv31uuvv66GDRva75WWlmYPdUV69eqlNWvWXPBz1yQCGQAAAGqGu5/ZWmXFcyto5MiRmjBhgt544w15e3tr4cKFuuOOO+Tu7i7JDCBPPfWUPvvsMx09elT5+fk6c+aMDh06VKH77969W127dpWfX3FNsbGxpa7797//rf/85z86ePCgzpw5o9zcXF122WUV/hxFz4qNjZXNZrMf6927tzIyMnT48GF7i16XLl0cfq5p06ZKTk6u0DP279+vvLw89e7d237M09NTPXr00O7duyVJ9957r4YNG6YffvhBcXFxGjp0qHr16iXJnJzkxhtvVFRUlPr3769BgwYpLi6u3Oc1atRIt956q9555x2NHTtWp0+f1uLFix1mtfzvf/+rNm3a2IPzZZddpjZt2mjRokWaMGGC/bqAgAD98MMPDvf39fWt0OeuSQQyAAAA1AybrUJdB600ePBgFRYWasWKFbriiiv07bff6pVXXrGff/DBB7V69WrNnDlTbdu2la+vr2677Tbl5uZW6P5GBcazffTRR3rggQf08ssvKzY2VgEBAXrppZe0efPmSn0WwzAcwljJ55c8XjTuqojNZis1ju58zzj3fuc+e8CAATp48KBWrFihdevWqW/fvpo4caJmzpypyy+/XAkJCVq5cqXWrVun4cOH64YbbtD//ve/cp85btw49e3bV7/99pvWr18vSbr99tvt59955x398ssvDpObFBYWau7cuQ6BzM3NTW3btq3Q56xNBDIAAAC4LF9fX916661auHCh9u3bp/bt26tbt272899++63GjBmjW265RZI5puzAgQMVvn90dLQWLFigM2fO2FtjNm3a5HDNt99+q169eum+++6zH9u/f7/DNV5eXiooKLjgsxYvXuwQjjZu3KiAgAA1a9aswjWfT9u2beXl5aUNGzZoxIgRkqS8vDxt27ZNU6ZMsV/XpEkTjRkzRmPGjNHVV1+tBx98UDNnzpRkznZ4++236/bbb9dtt92m/v3769SpUwoODi7zmddff73atGmjefPm6auvvtLw4cMVEBAgSdqxY4e2bdumr7/+2uHnU1NTdc0112jnzp1OvywAgQwAAAAubeTIkRo8eLB++eUX3XXXXQ7n2rZtqyVLlmjw4MGy2Wx6/PHHK9yaJJljoB577DGNGzdOf//733XgwAF7MCn5jPfee0+rV69W69attWDBAm3dulWtW7e2X9OqVSutXr1ae/fuVePGjRUYGFjqWffdd59effVV3X///Zo0aZL27t2rJ598UlOnTrWPH6sqf39/3XvvvXrwwQcVHBysFi1a6MUXX1RWVpbGjRsnSXriiSfUrVs3dezYUTk5Ofrss8906aWXSpJmzZqlpk2b6rLLLpObm5s+/vhjhYeHq1GjRuU+02azaezYsXrllVeUkpKil156yX5u7ty56tGjR5kTeMTGxmru3LmaNWuWJLMV79ixY6WuCw0Nrbbfz8Vg2vv6KG2XtKqHtO5aqysBAABwen369FFwcLD27t1rb/UpMmvWLAUFBalXr14aPHiw+vXrp8svv7zC927QoIGWL1+uXbt2KSYmRo899pheeOEFh2vuuece3Xrrrbr99tvVs2dPnTx50qG1TJLGjx+vqKgode/eXU2aNNF3331X6lnNmjXT559/ri1btqhr166655577EGwOj3//PMaNmyYRo0apcsvv1z79u3T6tWrFRQUJMlszXv00UfVpUsXXXPNNXJ3d9eiRYvsv48XXnhB3bt31xVXXKEDBw7o888/v2AgGjNmjNLS0hQVFWUfv5abm6v3339fw4YNK/Nnhg0bpvfff9/evTQ9PV1NmzYttVV0/FxNsRkV6diKCklPT1dgYKDS0tIcZnSpdaf3ScvbSR4NpOGnrasDAAC4jOzsbCUkJKh169by8fGxuhygVpzve1/RbEALWX3kc3Y61PwMKc+CmY0AAAAAVAiBrD7ybFA8o1H2cWtrAQAAAFAuAll9VdRKll164CIAAAAA50Agq698zwayMwQyAAAAwFkRyOorWsgAAAAAp0cgq698aCEDAAAAnB2BrL7ypYUMAAAAcHYEsvqKFjIAAADA6RHI6itayAAAAGpFq1at9Oqrr9rf22w2ffLJJ+Vef+DAAdlsNsXHx1fpudV1nwsZM2aMhg4dWqPPcGUEsvrKJ8x8JZABAADUqqSkJA0YMKBa71lWKIqMjFRSUpI6depUrc9yJjNmzJDNZiu1dejQQZLUuXNn/eUvfynzZz/44AN5enrq+PHidXnj4uLk7u6uTZs2lbp+zJgxZT6rf//+NfPhziKQ1Vf2WRaPS0ahtbUAAAC4kPDwcHl7e9f4c9zd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Hs2ZM8d+7sorr9Tll1+uN954Q2+//bYefvhhJSYmyt/fX5L0+eefa/DgwTp69KjCwsLUrFkzjR07Vs8880yZNdhsNv3973/X//t//0+SlJmZqYCAAH3++efq37+/hgwZopCQEL3zzjs19FsAAFiNMWQAAJd1/fXXOwQuSQoODrbvx8bGOpyLjY1VfHy8JGn37t3q2rWrPYxJUu/evVVYWKi9e/fKZrPp6NGj6tu373lr6NKli33f399fAQEBSk5OliTde++9GjZsmH744QfFxcVp6NCh6tWr10V9VgCAcyKQAQBclr+/f6kuhBdis9kkSYZh2PfLusbX17dC9/P09Cz1s4WFhZKkAQMG6ODBg1qxYoXWrVunvn37auLEiZo5c2alagYAOC/GkAEAUI5NmzaVet+hQwdJUnR0tOLj45WZmWk//91338nNzU3t27dXQECAWrVqpS+++KJKNTRp0sTevfLVV1/VW2+9VaX7AQCcCy1kAACXlZOTo2PHjjkc8/DwsE+c8fHHH6t79+666qqrtHDhQm3ZskVz586VJI0cOVJPPvmkRo8erRkzZujEiRO6//77NWrUKIWFhUmSZsyYoXvuuUehoaEaMGCATp8+re+++073339/hep74okn1K1bN3Xs2FE5OTn67LPPdOmll1bjbwAAYDUCGQDAZa1atUpNmzZ1OBYVFaU9e/ZIMmdAXLRoke677z6Fh4dr4cKFio6OliT5+flp9erV+utf/6orrrhCfn5+GjZsmF555RX7vUaPHq3s7GzNmjVL06dPV0hIiG677bYK1+fl5aVHH31UBw4ckK+vr66++motWrSoGj45AMBZMMsiAABlsNlsWrp0qYYOHWp1KQCAeowxZAAAAABgEQIZAAAAAFiEMWQAAJSBHv0AgNpACxkAAAAAWIRABgAAAAAWIZABAAAAgEUIZAAAAABgEQIZAAAAAFiEQAYAAAAAFiGQAQAAAIBFCGQAAAAAYBECGQAAAABY5P8DvyyieYHvq3oAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1000x700 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "save_valid_loss_plot(valid_lossVAE, valid_lossEVAE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3b764d9e",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
