{
 "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",
    "\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 Reshape,Conv2D, UpSampling1D, 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\"] = \"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=64 # This is the dimension of intermediate latent variable \n",
    "epochs = 50* 50\n",
    "epochstep = epochs/100\n",
    "nofclasses=10\n",
    "batchsize= 200 \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": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def MgGMMfit(encoderX,dataX_train,labelX_train,nofclasses):\n",
    "    gmmX = encoderX.predict(dataX_train)\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": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x7fc7ec11bf60>"
      ]
     },
     "execution_count": 8,
     "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": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "WARNING:tensorflow:From /nas/home/mrostami/.local/lib/python3.6/site-packages/keras/backend/tensorflow_backend.py:508: The name tf.placeholder is deprecated. Please use tf.compat.v1.placeholder instead.\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    " \n",
    "imgX = Input(shape=(28, 28, 1), name=\"input_img\")  # adapt this if using `channels_first` image data format\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": 68,
   "metadata": {},
   "outputs": [],
   "source": [
    "def encoderXNN(imgX,zdim):\n",
    "    x = Conv2D(64, (3, 3), activation='relu', padding='same', name='block1_conv1')(imgX)\n",
    "    x = Conv2D(64, (3, 3), activation='relu', padding='same', name='block1_conv2')(x)\n",
    "    x = MaxPooling2D((3, 3), strides=(2, 2), name='block1_pool')(x)\n",
    "\n",
    "    # Block 2\n",
    "    x = Conv2D(128, (3, 3), activation='relu', padding='same', name='block2_conv1')(x)\n",
    "    x = Conv2D(128, (3, 3), activation='relu', padding='same', name='block2_conv2')(x)\n",
    "    x = MaxPooling2D((3, 3), strides=(2, 2), name='block2_pool')(x)\n",
    "\n",
    "    # Block 3\n",
    "    x = Conv2D(256, (3, 3), activation='relu', padding='same', name='block3_conv1')(x)\n",
    "    x = Conv2D(256, (3, 3), activation='relu', padding='same', name='block3_conv2')(x)\n",
    "    x = Conv2D(256, (3, 3), activation='relu', padding='same', name='block3_conv3')(x)\n",
    "    x = MaxPooling2D((3, 3), strides=(2, 2), name='block3_pool')(x)\n",
    "    '''\n",
    "    # Block 4\n",
    "    x = Conv2D(512, (3, 3), activation='relu', padding='same', name='block4_conv1')(x)\n",
    "    x = Conv2D(512, (3, 3), activation='relu', padding='same', name='block4_conv2')(x)\n",
    "    x = Conv2D(512, (3, 3), activation='relu', padding='same', name='block4_conv3')(x)\n",
    "    x = MaxPooling2D((3, 3), strides=(2, 2), name='block4_pool')(x)\n",
    "    # Block 5\n",
    "    x = Conv2D(512, (2, 2), activation='relu', padding='same', name='block5_conv1')(x)\n",
    "    x = Conv2D(512, (2, 2), activation='relu', padding='same', name='block5_conv2')(x)\n",
    "    x = Conv2D(512, (2, 2), activation='relu', padding='valid', name='block5_conv3')(x)\n",
    "    x = MaxPooling2D((2, 2), strides=(2, 2), name='block5_pool')(x)\n",
    "    '''\n",
    "    # Top layers\n",
    "    x = Flatten(name='flatten')(x)\n",
    "    x = Dense(2048, activation='relu')(x)\n",
    "    x = Dropout(0.5)(x)\n",
    "    x = Dense(1024, activation='relu')(x)\n",
    "    x = Dropout(0.5)(x)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    embed = Dense(zdim, activation='relu')(x)\n",
    "    encoderX = Model(imgX, embed)\n",
    "\n",
    "    return encoderX\n",
    "\n",
    "    \n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    " "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 78,
   "metadata": {},
   "outputs": [],
   "source": [
    "def decoderXNN(yin):\n",
    "    x = Dense(512, activation='relu')(yin)\n",
    "    x = Dense(1048, activation='relu')(yin)\n",
    "    x = Reshape([1,1, 1048])(x)\n",
    "    xx = UpSampling2D((4, 4))(x)\n",
    "\n",
    "    xx = Conv2D(128, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    #encoded = UpSampling2D((2, 2))(yin)\n",
    "\n",
    "    xx = Dropout(0.3)(xx)\n",
    "    #x = UpSampling2D((2, 2))(x)\n",
    "    xx = Conv2D(64, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Conv2D(64, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Dropout(0.3)(xx)\n",
    "    xx = UpSampling2D((2, 2))(xx)\n",
    "    xx = Conv2D(32, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Conv2D(32, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Dropout(0.3)(xx)\n",
    "    xx = UpSampling2D((2, 2))(xx)\n",
    "    xx = Conv2D(16, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Conv2D(16, (3, 3), activation='relu')(xx)\n",
    "    xx = UpSampling2D((2, 2))(xx)\n",
    "    decoded = Conv2D(1, (2, 2), activation='sigmoid', padding='same',name=\"decoded\")(xx)\n",
    "\n",
    "    decoderX = Model(yin, decoded)\n",
    "    \n",
    "    return decoderX\n",
    "\n",
    "    #decoderX.summary()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "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",
    "    interdimX = 32\n",
    "    #y=Flatten()(yin)\n",
    "    #y=LeakyReLU(alpha=0.2)(yin)\n",
    "    #y=Dense(interdimX)(y)\n",
    "    #y=LeakyReLU(alpha=0.2)(y)\n",
    "    #y=BatchNormalization(momentum=0.8)(y)\n",
    "   # y=Dense(zdim)(y)\n",
    "    #y=LeakyReLU(alpha=0.2)(y)\n",
    "\n",
    "    probX=Dense(nofclasses,activation='softmax')(yin)\n",
    "    classifier=Model(inputs=[yin],outputs=[probX])\n",
    "    return classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 83,
   "metadata": {},
   "outputs": [],
   "source": [
    "encoderX = encoderXNN(imgX,zdim)\n",
    "yin =  Input(shape=(zdim,) )\n",
    "\n",
    "decoderX = decoderXNN(yin)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training on Task 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 84,
   "metadata": {},
   "outputs": [],
   "source": [
    "thres = .9\n",
    "lamda=1e0\n",
    "Classifier = []\n",
    "Encoder = []\n",
    "Decoder = []\n",
    "GMMs = []\n",
    "loss = []\n",
    "testXperf = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "metadata": {},
   "outputs": [],
   "source": [
    "LR = 3e-5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x720 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "model perforamnce on generated dataset\n",
      "99.65074217288263\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x720 with 4 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "\n",
    "fig=plt.figure(figsize=(10,10))\n",
    "\n",
    "\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_train[indCL,:,:,:] \n",
    "    else:\n",
    "        indCL = np.where(np.argmax(labelX_train,axis=1)==cl+1)[0]\n",
    "        xreptrainlabelonehotExtended = labelX_train[indCL,:cl+2] \n",
    "        extended_memory_label = np.concatenate([xreptrainlabelonehot,np.zeros([xreptrainlabelonehot.shape[0],1])],axis=1)\n",
    "        labelX_trainCL = np.concatenate([xreptrainlabelonehotExtended,extended_memory_label],axis=0)\n",
    "        dataX_train1CL =np.concatenate([ dataX_train[indCL,:,:,:],xreptrainData ],axis=0)\n",
    "\n",
    "        \n",
    "\n",
    "    labelX=K.placeholder(shape=(None,cl+2),dtype='float32') #labels of input images oneHot\n",
    "    theta=K.variable(generateTheta(nofprojections,zdim),)#Define a Keras Variable for \\theta_ls\n",
    "     \n",
    "\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",
    "    discriminationLoss=K.mean(K.categorical_crossentropy(labelX,classifier(encoderX(imgX))))\n",
    "    #discriminationLoss=K.mean(K.binary_crossentropy(labelX,classifier(encoderX(imgX))))\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<7:\n",
    "        \n",
    "        opt = Adam(lr=1e-4,decay = 1e-4 , amsgrad=True) # very important\n",
    "        updates = opt.get_updates(myLoss,params)\n",
    "        train = K.function(inputs=[imgX,labelX],outputs=[discriminationLoss,autorec],updates=updates)\n",
    "    if 7<=cl:\n",
    "        LR = 1e-6\n",
    "        opt = Adam(lr=LR,decay = LR, amsgrad=True) # very important\n",
    "        updates = opt.get_updates(myLoss,params)\n",
    "        train = K.function(inputs=[imgX,labelX],outputs=[discriminationLoss,autorec],updates=updates)\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",
    "        if cl==0:\n",
    "            loss.append(train(inputs=[trainDataX,trainLabelX]))\n",
    "        else:\n",
    "            loss.append(train(inputs=[trainDataX,trainLabelX]))\n",
    "            #loss.append(trainT(inputs=[trainDataY,trainDatasample,trainDataYtot,trainLabelY,trainLabelsample,trainLabelYtot,theta_]))   \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_test[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(decoderX.predict(encoderX.predict(trainDataX[imageind:imageind+1,:,:,:]))[0,:,:,0])\n",
    "\n",
    "            plt.subplot(4, 1, 4)\n",
    "            plt.imshow(trainDataX[imageind,:,:,0])\n",
    "            \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",
    "    labelX_train_EXP = np.argmax(np.squeeze(labelX_trainCL),axis=1)   \n",
    "    tempvar = labelX_trainCL\n",
    "    num_det_clusters = 0 \n",
    "    \n",
    "    gmm = MgGMMfit(encoderX,dataX_train1CL,labelX_trainCL,cl+2)\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 = (cl+1)*8000\n",
    "    xreptrainDataencoded, xreptrainlabel = gmm.sample(xnofsample)\n",
    "    \n",
    "    \n",
    "\n",
    "\n",
    "\n",
    "    xreptrainData = decoderX.predict(xreptrainDataencoded)\n",
    "\n",
    "    xreptrainData = xreptrainData.clip(min=0)\n",
    "    xreptrainData = xreptrainData.clip(max=1)\n",
    "\n",
    "    tres = .9\n",
    "\n",
    "    labpred = classifier.predict(encoderX.predict(xreptrainData))\n",
    "    indperd = (np.max(labpred,axis=1)>tres)\n",
    "    xreptrainData = xreptrainData[indperd,:,:,:] \n",
    "    xreptrainlabel = xreptrainlabel[indperd]    \n",
    "\n",
    "\n",
    "\n",
    "    xreptrainlabelonehot = keras.utils.to_categorical(xreptrainlabel)\n",
    "\n",
    "    xnofsampletest =  (cl+2)*1000\n",
    "    xreptestDataencoded, xreptestlabel = gmm.sample(xnofsampletest)\n",
    "    \n",
    "    \n",
    "    xreptestData = decoderX.predict(xreptestDataencoded)\n",
    "\n",
    "    xreptestData = xreptestData.clip(min=0)\n",
    "    xreptestData = xreptestData.clip(max=1)\n",
    "\n",
    "    labpred = classifier.predict(encoderX.predict(xreptestData))\n",
    "    indperd = (np.max(labpred,axis=1)>tres)\n",
    "    xrepestData = xreptestData[indperd,:,:,:] \n",
    "    xreptestlabel = xreptestlabel[indperd]    \n",
    "    xreptestData = xreptestData[indperd,:,:,:]\n",
    "    \n",
    "    xreptestlabelonehot = keras.utils.to_categorical(xreptestlabel)\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_X.shape[0])\n",
    "                                              \n",
    "                                              \n",
    "                                              \n",
    "                                              "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "import json\n",
    " \n",
    "testpermutedperf = testXperf \n",
    "with open('MNSIT-ICLA-CNN.json', 'w') as fp:\n",
    "    json.dump(testpermutedperf, fp)"
   ]
  }
 ],
 "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
}
