{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Using TensorFlow backend.\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "from utils import *\n",
    "import csv\n",
    "import keras.utils\n",
    "from tensorflow.keras import backend as K\n",
    "from tensorflow.keras.datasets import mnist\n",
    "from tensorflow.keras.models import save_model,load_model,Model\n",
    "from tensorflow.keras.optimizers import Adam, RMSprop\n",
    "from tensorflow.keras.layers import Conv2D, UpSampling2D, AveragePooling2D, MaxPooling2D, Dense,Input, Dropout\n",
    "from tensorflow.keras.layers import LeakyReLU,Reshape,BatchNormalization, Flatten\n",
    "from tensorflow.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",
    "from PIL import Image\n",
    "from sklearn.cluster import KMeans\n",
    "from tensorflow.keras import regularizers\n",
    "from tensorflow.keras.applications.resnet50 import ResNet50\n",
    "from tensorflow.keras.preprocessing import image\n",
    "from tensorflow.keras.applications.resnet50 import preprocess_input, decode_predictions\n",
    "\n",
    "from tensorflow.keras.layers import Input, Conv2D, Activation, BatchNormalization, GlobalAveragePooling2D, Dense, Dropout\n",
    "#from tensorflow.keras.layers.merge import add\n",
    "from tensorflow.keras.activations import relu, softmax\n",
    "from tensorflow.keras.models import Model\n",
    "from tensorflow.keras import regularizers\n",
    "\n",
    "from tensorflow.keras.models import Sequential\n",
    "from tensorflow.keras.layers import Dense, Dropout, Flatten\n",
    "from tensorflow.keras.applications.resnet50 import ResNet50\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import tensorflow.keras.layers as layers\n",
    "from tensorflow.keras.models import Model\n",
    "from tensorflow.keras.preprocessing import image\n",
    "from tensorflow.keras.initializers import glorot_uniform\n",
    "from tensorflow.keras.optimizers import SGD\n",
    "from array import array \n",
    "import json\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# import os\n",
    "# os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"2\"\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\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "zdim=128 # This is the dimension of intermediate latent variable \n",
    "epochs = 50   * 100\n",
    "epochstep = epochs/100\n",
    "nofclasses=10\n",
    "batchsize=100 \n",
    "nofprojections = 40"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def MgGMMfit(encoderX,dataX_train,labelX_train,nofclasses):\n",
    "    gmmX = encoderX.predict(dataX_train)\n",
    "    \n",
    "    gmmY = np.argmax(labelX_train,axis=1)\n",
    "    yper = classifier.predict(encoderX.predict(dataX_train))\n",
    "    yper = np.argmax(yper, axis=1)\n",
    "    gmmX = gmmX[gmmY==yper,:]\n",
    "    gmmY = gmmY[gmmY==yper]\n",
    "    gmmModel =  GaussianMixture(n_components=nofclasses,covariance_type='full', init_params='kmeans', tol=1e-04) \n",
    "    gmmModel.fit(gmmX,gmmY)\n",
    "    gmmModelSingle =  GaussianMixture(n_components=1,covariance_type='full') \n",
    "\n",
    "    for i in range(nofclasses):\n",
    "        a= gmmX[gmmY==i,:]\n",
    "        gmmModelSingle.fit(a)\n",
    "        gmmModel.weights_[i] = 1/float(nofclasses)#gmmY[gmmY==i].shape[0]/gmmY.shape[0]\n",
    "        gmmModel.covariances_[i] = gmmModelSingle.covariances_[0]\n",
    "        gmmModel.means_[i] = gmmModelSingle.means_[0]\n",
    "\n",
    "        gmmModel.precisions_cholesky_[i] = gmmModelSingle.precisions_cholesky_[0]\n",
    "        gmmModel.precisions_[i] = gmmModelSingle.precisions_[0]\n",
    "        \n",
    "    return gmmModel\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Data Generation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Read Fashion-MNIST\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": 7,
   "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": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(60000, 784)"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataX_train1.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x2b18a5f32250>"
      ]
     },
     "execution_count": 9,
     "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": [
    "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": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "WARNING:tensorflow:From /nas/home/mrostami/.local/lib/python3.7/site-packages/tensorflow/python/compat/v2_compat.py:96: disable_resource_variables (from tensorflow.python.ops.variable_scope) is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "non-resource variables are not supported in the long term\n"
     ]
    }
   ],
   "source": [
    "import tensorflow.compat.v1 as tf\n",
    "tf.disable_v2_behavior()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "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": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "def encoderXNN(imgX):\n",
    "\n",
    "\n",
    "\n",
    "    hidden_size = 800\n",
    "    x=Dense(hidden_size,activation = 'sigmoid')(imgX)\n",
    "\n",
    "    #x=Dense(int(hidden_size),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": 59,
   "metadata": {},
   "outputs": [],
   "source": [
    "def decoderXNN(yin):\n",
    "    hidden_size = 800\n",
    "\n",
    "    x=Dense(int(hidden_size),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 = 'sigmoid')(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": 78,
   "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": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "metadata": {},
   "outputs": [],
   "source": [
    "thres = .9\n",
    "lamda=1e1\n",
    "Classifier = []\n",
    "Encoder = []\n",
    "Decoder = []\n",
    "GMMs = []\n",
    "\n",
    "# reinitLayers(encoderX)\n",
    "# reinitLayers(encoderY)\n",
    "# reinitLayers(classifier)\n",
    "\n",
    "#seed=np.random.randint(0,high=100)\n",
    "loss = []\n",
    "testXperf = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "metadata": {},
   "outputs": [],
   "source": [
    "encoderX = encoderXNN(imgX)\n",
    "yin =  Input(shape=(zdim,) )\n",
    "\n",
    "decoderX = decoderXNN(yin)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "metadata": {},
   "outputs": [],
   "source": [
    "epochs = 200   * 100\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    " "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training on Task 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "metadata": {},
   "outputs": [],
   "source": [
    "XreptrainlabelonehotTot  = []\n",
    "XreptrainDataTot = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x720 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/nas/home/mrostami/anaconda3/envs/tf1/lib/python3.7/site-packages/ipykernel_launcher.py:94: MatplotlibDeprecationWarning: Adding an axes using the same arguments as a previous axes currently reuses the earlier instance.  In a future version, a new instance will always be created and returned.  Meanwhile, this warning can be suppressed, and the future behavior ensured, by passing a unique label to each axes instance.\n",
      "/nas/home/mrostami/anaconda3/envs/tf1/lib/python3.7/site-packages/ipykernel_launcher.py:97: MatplotlibDeprecationWarning: Adding an axes using the same arguments as a previous axes currently reuses the earlier instance.  In a future version, a new instance will always be created and returned.  Meanwhile, this warning can be suppressed, and the future behavior ensured, by passing a unique label to each axes instance.\n",
      "/nas/home/mrostami/anaconda3/envs/tf1/lib/python3.7/site-packages/ipykernel_launcher.py:102: MatplotlibDeprecationWarning: Adding an axes using the same arguments as a previous axes currently reuses the earlier instance.  In a future version, a new instance will always be created and returned.  Meanwhile, this warning can be suppressed, and the future behavior ensured, by passing a unique label to each axes instance.\n",
      "/nas/home/mrostami/anaconda3/envs/tf1/lib/python3.7/site-packages/ipykernel_launcher.py:105: MatplotlibDeprecationWarning: Adding an axes using the same arguments as a previous axes currently reuses the earlier instance.  In a future version, a new instance will always be created and returned.  Meanwhile, this warning can be suppressed, and the future behavior ensured, by passing a unique label to each axes instance.\n"
     ]
    }
   ],
   "source": [
    "fig=plt.figure(figsize=(10,10))\n",
    "LR = 3e-5\n",
    " \n",
    "\n",
    "\n",
    "\n",
    "for cl in range(5):\n",
    "    #epochs = int(epochs/(cl+1))\n",
    "    # extract the dataset\n",
    "    #for jj in range(cl+1):\n",
    "    classifier = classifierNN(5*cl+5,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],np.where(np.argmax(labelX_train,axis=1)==2)[0],np.where(np.argmax(labelX_train,axis=1)==3)[0],np.where(np.argmax(labelX_train,axis=1)==4)[0]])\n",
    "        labelX_trainCL =  labelX_train[indCL,:5*cl+5] \n",
    "        dataX_train1CL =  dataX_train1[indCL,:] \n",
    "    else:\n",
    "        indCL = np.concatenate([np.where(np.argmax(labelX_train,axis=1)==5)[0],np.where(np.argmax(labelX_train,axis=1)==6)[0],np.where(np.argmax(labelX_train,axis=1)==7)[0],np.where(np.argmax(labelX_train,axis=1)==8)[0],np.where(np.argmax(labelX_train,axis=1)==9)[0]])\n",
    "         \n",
    "        \n",
    "        xreptrainlabelonehotExtended = labelX_train[indCL,:5*cl+5] \n",
    "        extended_memory_label = np.concatenate([xreptrainlabelonehot,np.zeros([xreptrainlabelonehot.shape[0],5])],axis=1)\n",
    "        labelX_trainCL = np.concatenate([xreptrainlabelonehotExtended,extended_memory_label],axis=0)\n",
    "        dataX_train1CL =np.concatenate([ dataX_train1[indCL,:],xreptrainData ],axis=0)\n",
    "\n",
    "        \n",
    "\n",
    "    labelX=K.placeholder(shape=(None,5*cl+5),dtype='float32') #labels of input images oneHot\n",
    "    theta=K.variable(generateTheta(nofprojections,zdim),)#Define a Keras Variable for \\theta_ls\n",
    "    #swLoss=sWasserstein(encoderX(imgY),encoderX(imgZ),theta,Cp=labelX,Cq=labelZ)\n",
    "    \n",
    "\n",
    "    #autorec = K.mean(K.square(imgX-decoderX(encoderX(imgX))))\n",
    "    autorec = K.mean(K.square((imgX-decoderX(encoderX(imgX)))))\n",
    "    #autorec = K.mean(K.binary_crossentropy(imgX,decoderX(encoderX(imgX))))\n",
    "    \n",
    "    \n",
    "\n",
    "\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",
    "\n",
    "     \n",
    "    myLoss=  discriminationLoss+lamda*autorec\n",
    "    params=encoderX.weights + decoderX.weights + classifier.weights\n",
    "\n",
    "    \n",
    "    \n",
    "    #loss=[]\n",
    "    \n",
    "    \n",
    "    if cl == 0:\n",
    "        opt = Adam(lr=1e-5,decay = 1e-5,amsgrad=True)\n",
    "        updates = opt.get_updates(myLoss,params)\n",
    "        train = K.function(inputs=[imgX,labelX],outputs=[autorec],updates=updates)\n",
    "        LR=1e-5\n",
    "        \n",
    "    if cl >= 1:\n",
    "        LR = LR#/3\n",
    "        opt = Adam(lr=LR,decay = 1e-5,amsgrad=True) \n",
    "        #opt = Adam(lr=1e-5,decay = 1e-3,amsgrad=True) # very important\n",
    "        updates = opt.get_updates(myLoss,params)\n",
    "        train = K.function(inputs=[imgX,labelX],outputs=[autorec],updates=updates)\n",
    "        #train = K.function(inputs=[imgX,imgY,imgZ,labelX,labelY,labelZ,theta],outputs=[lamda*autorec,discriminationLoss,swLoss],updates=updatesT)\n",
    "\n",
    "\n",
    "    for itr in range(epochs):\n",
    "        \n",
    "        indTrainDataX,trainLabelX=batchGenerator(labelX_trainCL,batchsize,nofclasses=5*cl+5)\n",
    "        trainDataX=dataX_train1CL[indTrainDataX,...]\n",
    "        \n",
    "        \n",
    "        loss.append(train(inputs=[trainDataX,trainLabelX]))\n",
    "      \n",
    "        if itr%epochstep==0: \n",
    "\n",
    "            for jj in range(5*cl+5):\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(4, 1, 1)\n",
    "            plt.plot(np.asarray(loss))\n",
    "            \n",
    "            plt.subplot(4, 1, 2)\n",
    "            plt.plot(np.asarray(testXperf))\n",
    "            \n",
    "            imageind = np.random.randint(trainDataX.shape[0])\n",
    "            \n",
    "            plt.subplot(4, 1, 3)\n",
    "            plt.imshow(np.reshape(decoderX.predict(encoderX.predict(trainDataX[imageind:imageind+1,:])),[28,28]))\n",
    "\n",
    "            plt.subplot(4, 1, 4)\n",
    "            plt.imshow(np.reshape(trainDataX[imageind,:],[28,28]))\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",
    "        ### pseudo-datapoints\n",
    "    if cl<4:\n",
    "        labelX_train_EXP = np.argmax(np.squeeze(labelX_trainCL),axis=1)   \n",
    "        tempvar = labelX_trainCL\n",
    "        \n",
    "        \n",
    "        gmm = MgGMMfit(encoderX,dataX_train1CL,labelX_trainCL,5*cl+5)\n",
    "        \n",
    "\n",
    "        GMMs.append(gmm)\n",
    "\n",
    "  \n",
    "  \n",
    "        # GMM is used as a generative model so task 1 data is not saved\n",
    "        xnofsample = (5*cl+5)*4000\n",
    "        xreptrainDataencoded, xreptrainlabel = gmm.sample(xnofsample)\n",
    "\n",
    "\n",
    "\n",
    "        xtemper = np.zeros([xreptrainDataencoded.shape[0], xreptrainDataencoded.shape[1]])\n",
    "        xtemper  = xreptrainDataencoded\n",
    "\n",
    "\n",
    "        xreptrainData = decoderX.predict(xtemper)\n",
    "\n",
    "        xreptrainData = xreptrainData.clip(min=0)\n",
    "        xreptrainData = xreptrainData.clip(max=1)\n",
    "\n",
    "#         xreptrainData[xreptrainData>.5]=1\n",
    "#         xreptrainData[xreptrainData<.4]=0        \n",
    "        \n",
    " \n",
    "        xreptrainlabelonehot = keras.utils.to_categorical(xreptrainlabel)\n",
    "\n",
    "        xnofsampletest =  (5*cl+5)*1000\n",
    "        xreptestDataencoded, xreptestlabel = gmm.sample(xnofsampletest)\n",
    "\n",
    "        xtemper1 = np.zeros([xreptestDataencoded.shape[0],xreptestDataencoded.shape[1]])\n",
    "        xtemper1 = xreptestDataencoded\n",
    "\n",
    "        xreptestData = decoderX.predict(xtemper1)\n",
    "\n",
    "        xreptestData = xreptestData.clip(min=0)\n",
    "        xreptestData = xreptestData.clip(max=1)\n",
    "\n",
    "\n",
    "  \n",
    "        xreptestlabelonehot = keras.utils.to_categorical(xreptestlabel)\n",
    "    \n",
    "    \n",
    "        XreptrainlabelonehotTot.append(xreptrainlabelonehot)  \n",
    "        XreptrainDataTot.append(xreptrainData)  \n",
    "\n",
    "        # model perforamnce on generated dataset\n",
    "        perd_label_Sample = classifier.predict(encoderX.predict(xreptestData))\n",
    "        print('model perforamnce on generated dataset')\n",
    "        print(100*float(sum(1*(np.argmax(perd_label_Sample,axis=1)==np.argmax(np.squeeze(xreptestlabelonehot),axis=1))))/perd_label_Sample.shape[0])\n",
    "\n",
    "\n",
    "\n",
    "                                              "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1376"
      ]
     },
     "execution_count": 77,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    " "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# encoderX = Encoder[0]\n",
    "# decoderX = Decoder[0]\n",
    "# classifier = Classifier[0]\n",
    "# xreptrainlabelonehot = XreptrainlabelonehotTot[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x2b1b9c774b90>]"
      ]
     },
     "execution_count": 53,
     "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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# first three tasks 99.9, , 98.3 , 95.7 , 83.4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "97.8019069857949\n",
      "98.25063241875854\n",
      "98.59466822338977\n",
      "98.76629694493091\n",
      "98.83401439968867\n"
     ]
    }
   ],
   "source": [
    "perCperf = []\n",
    "for i in range(5):\n",
    "    perCperf.append(sum(testXperf[100*i+45:+100*i+95])/50)\n",
    "    print((sum(testXperf[100*i+45:+100*i+95])/50))\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "import json\n",
    " \n",
    "testpermutedperf = testXperf \n",
    "#with open('FMNIST-ICLA-CNN-06-16-2020.json', 'w') as fp:\n",
    "#    json.dump(testpermutedperf, fp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
