{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/usr/local/lib64/python3.6/site-packages/h5py/__init__.py:36: FutureWarning: Conversion of the second argument of issubdtype from `float` to `np.floating` is deprecated. In future, it will be treated as `np.float64 == np.dtype(float).type`.\n",
      "  from ._conv import register_converters as _register_converters\n",
      "Using TensorFlow backend.\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "from utils import *\n",
    "\n",
    "import keras.utils\n",
    "from keras import backend as K\n",
    "from keras.datasets import mnist\n",
    "from keras.models import save_model,load_model,Model\n",
    "from keras.optimizers import Adam, RMSprop\n",
    "from keras.layers import Conv2D, UpSampling2D, AveragePooling2D, MaxPooling2D, Dense,Input, Dropout\n",
    "from keras.layers import LeakyReLU,Reshape,BatchNormalization, Flatten\n",
    "from keras.models import save_model\n",
    "from tempfile import TemporaryFile\n",
    "import tensorflow as tf\n",
    "import pickle\n",
    "import scipy.io as sio\n",
    "import matplotlib.pyplot as plt\n",
    "from IPython import display\n",
    "import time\n",
    "from sklearn.mixture import GaussianMixture\n",
    "import umap\n",
    "from copy import deepcopy\n",
    "\n",
    "import numpy as np\n",
    "from utils import *\n",
    "\n",
    "import keras.utils\n",
    "from keras import backend as K\n",
    "from keras.datasets import mnist\n",
    "from keras.models import save_model,load_model,Model\n",
    "from keras.optimizers import Adam, RMSprop\n",
    "from keras.layers import Conv2D, UpSampling2D, AveragePooling2D, MaxPooling2D, Dense,Input, Dropout\n",
    "from keras.layers import LeakyReLU,Reshape,BatchNormalization, Flatten\n",
    "from keras.models import save_model\n",
    "from tempfile import TemporaryFile\n",
    "import tensorflow as tf\n",
    "import pickle\n",
    "import scipy.io as sio\n",
    "import matplotlib.pyplot as plt\n",
    "from IPython import display\n",
    "import time\n",
    "from sklearn.mixture import GaussianMixture\n",
    "import umap\n",
    "import scipy.io as sio\n",
    "from copy import deepcopy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# import os\n",
    "# os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"0\"\n",
    "# from keras import backend as K\n",
    "# config = tf.ConfigProto()\n",
    "# config.gpu_options.allow_growth = True\n",
    "# K.tensorflow_backend.set_session(tf.Session(config=config))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ! pip install umap-learn\n",
    "# ! pip install keras\n",
    "# ! pip install matplotlib\n",
    "# ! pip install np_utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "zdim=16 # This is the dimension of intermediate latent variable \n",
    "epochs = 10000\n",
    "epochstep = epochs/100\n",
    "nofclasses=10\n",
    "batchsize=2000 \n",
    "nofprojections = 40"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Data Generation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Read MNIST\n",
    "\n",
    "(x_train,y_train),(x_test,y_test)=mnist.load_data()\n",
    "dataX_train=np.expand_dims(x_train.astype('float32')/255.,3)\n",
    "dataX_test=np.expand_dims(x_test.astype('float32')/255.,3)\n",
    "\n",
    "labelX_train = keras.utils.to_categorical(y_train, nofclasses)\n",
    "labelX_test = keras.utils.to_categorical(y_test, nofclasses)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "numdataTrain = labelX_train.shape[0]\n",
    "numdataTest = labelX_test.shape[0]\n",
    "\n",
    "dataX_train1 = np.reshape(dataX_train,[numdataTrain,28*28]) \n",
    "dataX_test1 = np.reshape(dataX_test,[numdataTest,28*28]) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x7fb49c684588>"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAPsAAAD4CAYAAAAq5pAIAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8li6FKAAAOj0lEQVR4nO3df7BU5X3H8c8HvIISSUCUoDKKFqOkbTC9JWl0MrZOjTpp0PywMqkxHVMyVTvqpDaOmWn8pw01oRnHRCf4I8GMxWYKVjpDq4RmhkkajVcHFaUGJVIhCCqZCjYg9/LtH/eQueo9z172nN2z8LxfM3d293z37PnOkY9n9zy753FECMDhb1zTDQDoDsIOZIKwA5kg7EAmCDuQiSO6ubEjPSEmalI3NwlkZY/e0Jux16PVKoXd9gWSbpU0XtJdEbEo9fyJmqQP+bwqmwSQ8GisKa21/Tbe9nhJ35Z0oaQ5khbYntPu6wHorCqf2edJej4iNkXEm5LulzS/nrYA1K1K2E+U9NKIx1uKZW9he6HtAdsD+7S3wuYAVNHxs/ERsSQi+iOiv08TOr05ACWqhH2rpJkjHp9ULAPQg6qE/TFJs23Psn2kpMskraynLQB1a3voLSIGbV8j6SEND73dExHP1NYZgFpVGmePiFWSVtXUC4AO4uuyQCYIO5AJwg5kgrADmSDsQCYIO5AJwg5kgrADmSDsQCYIO5AJwg5kgrADmSDsQCYIO5AJwg5kgrADmSDsQCYIO5AJwg5kgrADmSDsQCYIO5AJwg5kgrADmSDsQCYIO5AJwg5kgrADmSDsQCYqzeKKesTZc5P1V846ukudHDwPRbJ+3B0/7VInaKVS2G2/KGmXpCFJgxHRX0dTAOpXx5H9DyPi1RpeB0AH8ZkdyETVsIekh20/bnvhaE+wvdD2gO2BfdpbcXMA2lX1bfw5EbHV9vGSVtv+74hYO/IJEbFE0hJJmuyp6bM5ADqm0pE9IrYWtzskPSBpXh1NAahf22G3Pcn2MQfuSzpf0vq6GgNQrypv46dLesD2gdf5p4j4j1q6OsSMOzo9Dh5nzkrWP3XXw8n6n09+6aB76pZ9MZSsf/Izl5TWtr0+ObnuCZdvSdb379qVrOOt2g57RGyS9IEaewHQQQy9AZkg7EAmCDuQCcIOZIKwA5ngJ6412Hx9+ieqT151W5c66b4+j0/W/+19K9t+7QWrPpasv3D/7yTr73lhX2ntqIFNyXWHXn0tWT8UcWQHMkHYgUwQdiAThB3IBGEHMkHYgUwQdiATjLOjZy079aFkffuX/zVZf/LNaaW1W67/XHLdSS2ugH0ojsNzZAcyQdiBTBB2IBOEHcgEYQcyQdiBTBB2IBOMs4/Rjqs+Ulo78Y9691LPh7Pp449K1s8/6o3S2teuTs9FuvfaScn6kfP3JOv73yjfdlM4sgOZIOxAJgg7kAnCDmSCsAOZIOxAJgg7kAnG2cdo+qc3l9aqXBu9Dg//unxM+Ce7Tk+uO86RrO8PJ+uXTflZsn5mX1+y3pRTJu9M1l/6+/R+8++X/1ZekiY898tkfXDby8l6J7Q8stu+x/YO2+tHLJtqe7XtjcXtlM62CaCqsbyN/56kC9627EZJayJitqQ1xWMAPaxl2CNiraS3v+eZL2lpcX+ppItr7gtAzdr9zD49IrYV91+WNL3sibYXSlooSRN1dJubA1BV5bPxERGSSs/yRMSSiOiPiP4+Tai6OQBtajfs223PkKTidkd9LQHohHbDvlLSFcX9KyQ9WE87ADql5Wd228sknStpmu0tkr4qaZGkH9i+UtJmSZd2ssncnf7gXybrp/7LUGntiP98vO523uKHV1yfrA99uvz66o98cFnd7YzZd09ek6x/9sb0vPPfOjn93YqPfe2GZP3427s/zt4y7BGxoKR0Xs29AOggvi4LZIKwA5kg7EAmCDuQCcIOZIKfuBZe+8IfJOurZn8jUZ1YadvLd6d/Lnnmbb9K1oc2bKy0/SqmLE3PbTxuxTGltT854/PJdT9z7+pk/XOTtybrVdw36+EWz6j237wJHNmBTBB2IBOEHcgEYQcyQdiBTBB2IBOEHcgE4+yFoYnpSyZPGde5cdWv/PufJuu/teGRjm270/bv2lVefOzp5Lr//IW3X+f0rd7z3RXJ+icmpb+f0Env/7Nnk/VXbu9SIyNwZAcyQdiBTBB2IBOEHcgEYQcyQdiBTBB2IBOMs6Nn+SfrkvW7fndOsn7D4g+U1p67uLMD3a0uVf1x/V5Htz8ajuxAJgg7kAnCDmSCsAOZIOxAJgg7kAnCDmSCcfbC/r6mO8DB2r9nT7I++/u/Lq397ML09QvmTYi2euplLY/stu+xvcP2+hHLbra91fa64u+izrYJoKqxvI3/nqTRLhnyzYiYW/ytqrctAHVrGfaIWCtpZxd6AdBBVU7QXWP7qeJt/pSyJ9leaHvA9sA+7a2wOQBVtBv2OySdJmmupG2SFpc9MSKWRER/RPT3aUKbmwNQVVthj4jtETEUEfsl3SlpXr1tAahbW2G3PWPEw0skrS97LoDe0HKc3fYySedKmmZ7i6SvSjrX9lxJIelFSV/sYI9d8eQN6d837zv8hl0Pf488VVraOlh6mmnYhMPvnHTLsEfEglEW392BXgB0EF+XBTJB2IFMEHYgE4QdyARhBzLBT1wLpy5Pjx4+98kG5tgFasSRHcgEYQcyQdiBTBB2IBOEHcgEYQcyQdiBTDDOXnj3c+ObbgE12/Px8muqnN73aIu1D79ocGQHMkHYgUwQdiAThB3IBGEHMkHYgUwQdiATh99gYptOWL4pWb/q8o+W1m4/aW21jR+bnhbrfz/74WT9V2eUTz982rLXkusOPfvzZL2Txs85PVkfmjwxWd8986hk/e8W3Vlae/+R1f7p/89g+XTQkvSJ7/xNsn6S/qvS9tvBkR3IBGEHMkHYgUwQdiAThB3IBGEHMkHYgUw4ontzEU/21PiQz+va9ur08vUfKa0N/PVtXezk4CzfPS1Zv+mhS7vUyTstuuD+ZP2SSb07bXL/LX+VrL/31u6Po0vSo7FGr8fOUb940fLIbnum7R/Zftb2M7avLZZPtb3a9sbitsWE1wCaNJa38YOSvhQRcyR9WNLVtudIulHSmoiYLWlN8RhAj2oZ9ojYFhFPFPd3Sdog6URJ8yUtLZ62VNLFnWoSQHUH9QVh26dIOkvSo5KmR8S2ovSypOkl6yyUtFCSJurodvsEUNGYz8bbfpek5ZKui4jXR9Zi+CzfqGf6ImJJRPRHRH+fJlRqFkD7xhR2230aDvp9EbGiWLzd9oyiPkPSjs60CKAOLYfebFvDn8l3RsR1I5Z/XdJrEbHI9o2SpkZE8nd9h/LQm48o/8Tz/D/0J9fdcNm3624HFf1icE+y/qlv3ZCsn7C4xaWo9w8dbEu1SA29jeUz+9mSLpf0tO11xbKbJC2S9APbV0raLKm5AVsALbUMe0T8WFLZ1REOzcM0kCG+LgtkgrADmSDsQCYIO5AJwg5kgktJj1EMDpbWTl2RHrM9M65O1jcsYBy+21qOo3+9mZ+odhJHdiAThB3IBGEHMkHYgUwQdiAThB3IBGEHMsGlpLth3PhkefyxU5P1fWeclH75m18prb3v3duT6y6e8Uiy3sqrQ+mpi+f/bfl49pTLX6q07dfvTu+X/zuu/Fj23tt68/foVVW6lDSAwwNhBzJB2IFMEHYgE4QdyARhBzJB2IFMMM5+mDti1snJ+i8vPLHS648bSv/7mfadn1Z6fRwcxtkBEHYgF4QdyARhBzJB2IFMEHYgE4QdyETL68bbninpXknTJYWkJRFxq+2bJf2FpAM/pr4pIlZ1qlG0Z/AXm5P1429P13H4GMskEYOSvhQRT9g+RtLjtlcXtW9GxDc61x6AuoxlfvZtkrYV93fZ3iCp2teuAHTdQX1mt32KpLMkHbimzzW2n7J9j+0pJesstD1ge2Cf9lZqFkD7xhx22++StFzSdRHxuqQ7JJ0maa6Gj/yLR1svIpZERH9E9PdpQg0tA2jHmMJuu0/DQb8vIlZIUkRsj4ihiNgv6U5J8zrXJoCqWobdtiXdLWlDRPzjiOUzRjztEknr628PQF3Gcjb+bEmXS3ra9rpi2U2SFtieq+HhuBclfbEjHQKoxVjOxv9Y0mi/j2VMHTiE8A06IBOEHcgEYQcyQdiBTBB2IBOEHcgEYQcyQdiBTBB2IBOEHcgEYQcyQdiBTBB2IBOEHchEV6dstv2KpJHXLp4m6dWuNXBwerW3Xu1Lord21dnbyRFx3GiFrob9HRu3ByKiv7EGEnq1t17tS6K3dnWrN97GA5kg7EAmmg77koa3n9KrvfVqXxK9tasrvTX6mR1A9zR9ZAfQJYQdyEQjYbd9ge3nbD9v+8Ymeihj+0XbT9teZ3ug4V7usb3D9voRy6baXm17Y3E76hx7DfV2s+2txb5bZ/uihnqbaftHtp+1/Yzta4vlje67RF9d2W9d/8xue7ykn0v6Y0lbJD0maUFEPNvVRkrYflFSf0Q0/gUM2x+VtFvSvRHx28WyWyTtjIhFxf8op0TEl3ukt5sl7W56Gu9itqIZI6cZl3SxpM+rwX2X6OtSdWG/NXFknyfp+YjYFBFvSrpf0vwG+uh5EbFW0s63LZ4vaWlxf6mG/7F0XUlvPSEitkXEE8X9XZIOTDPe6L5L9NUVTYT9REkvjXi8Rb0133tIetj247YXNt3MKKZHxLbi/suSpjfZzChaTuPdTW+bZrxn9l07059XxQm6dzonIj4o6UJJVxdvV3tSDH8G66Wx0zFN490to0wz/htN7rt2pz+vqomwb5U0c8Tjk4plPSEitha3OyQ9oN6binr7gRl0i9sdDffzG700jfdo04yrB/Zdk9OfNxH2xyTNtj3L9pGSLpO0soE+3sH2pOLEiWxPknS+em8q6pWSrijuXyHpwQZ7eYtemca7bJpxNbzvGp/+PCK6/ifpIg2fkX9B0lea6KGkr1MlPVn8PdN0b5KWafht3T4Nn9u4UtKxktZI2ijph5Km9lBv35f0tKSnNBysGQ31do6G36I/JWld8XdR0/su0VdX9htflwUywQk6IBOEHcgEYQcyQdiBTBB2IBOEHcgEYQcy8f/C9VzKuVoa/QAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "iii = 55110\n",
    "trainim = np.reshape(dataX_train1[iii,:],(28,28))\n",
    "plt.imshow(trainim)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "##  X is input and Z the embedding space"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    " \n",
    "imgX=Input(((28*28),)) #Input image \n",
    "\n",
    "\n",
    " \n",
    "labelX=K.placeholder(shape=(None,nofclasses),dtype='float32') #labels of input images oneHot\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Encoder, Decoder and Classifier NN"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "def encoderXNN(imgX):\n",
    "\n",
    "\n",
    "\n",
    "    hidden_size = 512\n",
    "    x=Dense(hidden_size,activation = 'relu')(imgX)\n",
    "\n",
    "    x=Dense(int(hidden_size/2),activation = 'relu')(x)\n",
    "    #x=Dense(hidden_size/32,activation = 'sigmoid')(x)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    encodedX=Dense(zdim)(x)\n",
    "\n",
    "    encoderX=Model(inputs=[imgX],outputs=[encodedX])\n",
    "\n",
    "\n",
    "\n",
    "    return encoderX\n",
    "    #encoderX.summary()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "def decoderXNN(yin):\n",
    "    hidden_size = 512\n",
    "\n",
    "    x=Dense(int(hidden_size/2),activation = 'relu')(yin)\n",
    "    #x=Dense(hidden_size/2,activation = 'sigmoid')(x)\n",
    "    x=Dense(hidden_size,activation = 'relu')(x)\n",
    "    #x=Dense(hidden_size,activation = 'sigmoid')(x)\n",
    "\n",
    "\n",
    "    decodedX=Dense(28*28,activation = 'relu')(x)\n",
    "\n",
    "\n",
    "    decoderX=Model(inputs=[yin],outputs=[decodedX])\n",
    "\n",
    "    \n",
    "    return decoderX\n",
    "\n",
    "    #decoderX.summary()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "# def classifierNN(nofclasses,z):\n",
    "#     probX=Dense(nofclasses,activation='softmax')(z)\n",
    "#     classifier=Model(inputs=[z],outputs=[probX])\n",
    "#     return classifier\n",
    "\n",
    "\n",
    "def classifierNN(nofclasses,yin):\n",
    "    \n",
    "    \n",
    "    probX=Dense(nofclasses,activation='softmax')(yin)\n",
    "    classifier=Model(inputs=[yin],outputs=[probX])\n",
    "    return classifier\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "WARNING: Logging before flag parsing goes to stderr.\n",
      "W0206 16:11:23.185524 140415288272704 deprecation.py:506] From /usr/local/lib64/python3.6/site-packages/tensorflow_core/python/ops/resource_variable_ops.py:1630: calling BaseResourceVariable.__init__ (from tensorflow.python.ops.resource_variable_ops) with constraint is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "If using Keras pass *_constraint arguments to layers.\n"
     ]
    }
   ],
   "source": [
    "encoderX = encoderXNN(imgX)\n",
    "yin = Input((zdim,))\n",
    "\n",
    "\n",
    "decoderX = decoderXNN(yin)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "def step_decay(epoch):\n",
    "   initial_lrate = 1e-4\n",
    "   drop = 0.5\n",
    "   epochs_drop = 10.0\n",
    "   lrate = initial_lrate * math.pow(drop,  \n",
    "           math.floor((1+epoch)/epochs_drop))\n",
    "   return lrate"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training on Task 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x720 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "lamda2=1e-3\n",
    "\n",
    "thres = .95\n",
    "lamda=4e2\n",
    "Classifier = []\n",
    "Encoder = []\n",
    "Decoder = []\n",
    "GMMs = []\n",
    "\n",
    "fig=plt.figure(figsize=(10,10))\n",
    "# reinitLayers(encoderX)\n",
    "# reinitLayers(encoderY)\n",
    "# reinitLayers(classifier)\n",
    "\n",
    "#seed=np.random.randint(0,high=100)\n",
    "loss = []\n",
    "testXperf = []\n",
    "for cl in range(9):\n",
    "    # extract the dataset\n",
    "    #for jj in range(cl+1):\n",
    "    classifier = classifierNN(cl+2,yin)\n",
    "    if cl == 0:\n",
    "        indCL = np.concatenate([np.where(np.argmax(labelX_train,axis=1)==0)[0],np.where(np.argmax(labelX_train,axis=1)==1)[0]])\n",
    "        labelX_trainCL =  labelX_train[indCL,:cl+2] \n",
    "        dataX_train1CL =  dataX_train1[indCL,:] \n",
    "    else:\n",
    "        indCL = np.concatenate([indCL,np.where(np.argmax(labelX_train,axis=1)==cl+1)[0]])\n",
    "        xreptrainlabelonehotExtended = labelX_train[indCL,:cl+2] \n",
    "        labelX_trainCL =  labelX_train[indCL,:cl+2] \n",
    "        dataX_train1CL =  dataX_train1[indCL,:] \n",
    "        \n",
    "\n",
    "    labelX=K.placeholder(shape=(None,cl+2),dtype='float32') #labels of input images oneHot\n",
    "   \n",
    "     \n",
    "    #discriminationLoss=K.mean(K.categorical_crossentropy(labelX,classifier(encoderX(imgX))))\n",
    "    discriminationLoss=K.mean(K.binary_crossentropy(labelX,classifier(encoderX(imgX))))\n",
    "    \n",
    "    myLoss=  discriminationLoss\n",
    "     \n",
    "  \n",
    "    params=encoderX.weights   + classifier.weights\n",
    "\n",
    "    \n",
    "    \n",
    "    #loss=[]\n",
    "    \n",
    "    \n",
    "    \n",
    "    opt = Adam(lr=1e-4)\n",
    "    updates = opt.get_updates(myLoss,params)\n",
    "    train = K.function(inputs=[imgX,labelX],outputs=[discriminationLoss],updates=updates)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    for itr in range(epochs):\n",
    "        \n",
    "        indTrainDataX,trainLabelX=batchGenerator(labelX_trainCL,batchsize,nofclasses=cl+2)\n",
    "        trainDataX=dataX_train1CL[indTrainDataX,...]\n",
    "        \n",
    "        \n",
    "  \n",
    "        loss.append(train(inputs=[trainDataX,trainLabelX]))\n",
    "  \n",
    "        if itr%epochstep==0: \n",
    "\n",
    "            for jj in range(cl+2):\n",
    "                if jj == 0:\n",
    "                    indCLtest = np.where(np.argmax(labelX_test,axis=1)==0)[0]\n",
    "                else:\n",
    "                    indCLtest = np.concatenate([indCLtest,np.where(np.argmax(labelX_test,axis=1)==jj)[0]])\n",
    "            labelX_testCL = labelX_test[indCLtest,:]\n",
    "            dataX_test1CL = dataX_test1[indCLtest,:]   \n",
    "\n",
    "            perd_label_X = classifier.predict(encoderX.predict(dataX_test1CL))\n",
    "\n",
    "\n",
    "\n",
    "            \n",
    "            testXperf.append(100*float(sum(1*(np.argmax(perd_label_X,axis=1)==np.argmax(np.squeeze(labelX_testCL),axis=1))))/labelX_testCL.shape[0])\n",
    "            \n",
    "            plt.subplot(2, 1, 1)\n",
    "            plt.plot(np.asarray(loss))\n",
    "            \n",
    "            plt.subplot(2, 1, 2)\n",
    "            plt.plot(np.asarray(testXperf))\n",
    "            \n",
    "            display.clear_output(wait=True)\n",
    "            display.display(plt.gcf()) \n",
    "            time.sleep(1e-3) \n",
    "    Classifier.append(classifier)  \n",
    "    Encoder.append(encoderX)  \n",
    "    Decoder.append(decoderX)  \n",
    "    \n",
    " "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7fb1d018dc88>]"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(testXperf)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
