import numpy as np
import tensorflow as tf
import json
import os
import time

from tensorflow.examples.tutorials.mnist import input_data

from core.hamInfnet_hei import HamInfNetHEI as HamInfNet
from decoder.vae_dcnn_mnist import VAE_DCNN_GPU, VAEQ_CONV
from util.constant import log_2pi
#from util.utils import dybinarize_mnist
from util.utils import binarise_fashion_mnist


def training_setting(z_dim):
    setting = {'mb_size': 128,
               'alpha': 0.0,   # alpha=0-divergence
               'z_dim': z_dim,
               'h_dim': 500,  
               'X_mnist_dim': 28 ** 2,
               'momentum_train_batch_size': 1,
               'z_train_sample_batch_size': 30,  
               'num_layers': 30,  
               'num_lfsteps': 5,
               'momentum_std': 1.0,
               'generator': 'dcnn_relu',
               'batches': 95000,  
               'dybin': True,
               'reg': 0.000001,
               'lr': 0.0002,
               'lr-decay': 0.97,
               }
    return setting


def train(setting, dataset, dataset_name='mnist', save_model=False, device='CPU', dtype=tf.float32):
    mb_size = setting['mb_size'] 
    alpha = setting['alpha']
    z_dim = setting['z_dim']  
    h_dim = setting['h_dim']  
    X_mnist_dim = setting['X_mnist_dim']  # 28**2
    momentum_train_batch_size = setting['momentum_train_batch_size']  
    z_train_sample_batch_size = setting['z_train_sample_batch_size']  
    generator = setting['generator']
    num_layers = setting['num_layers']  
    num_lfsteps = setting['num_lfsteps']  
    momentum_std = setting['momentum_std']  
    batches = setting['batches']  
    dybin = setting['dybin']  
    reg = setting['reg']
    lr = setting['lr']
    lr_decay = setting['lr-decay']
    if 'vfun' in setting.keys():
        vfun = setting['vfun']
    else:
        vfun = 'sigmoid'

    bin_label = 'dybin'
    if not dybin:
        bin_thresh = setting['bin_thresh']
        bin_label = 'stbin'

    setting = {'mb_size': mb_size,
               'alpha': alpha,
               'z_dim': z_dim,
               'h_dim': h_dim,
               'X_mnist_dim': X_mnist_dim,
               'momentum_train_batch_size': momentum_train_batch_size,
               'z_train_sample_batch_size': z_train_sample_batch_size,
               'num_layers': num_layers,
               'num_lfsteps': num_lfsteps,
               'momentum_std': momentum_std,
               'generator': generator,
               'vfun': vfun,
               'batches': batches,
               'dybin': dybin,
               'reg': reg,
               'lr': lr,
               'lr-decay': lr_decay,
               }
    if not dybin:
        setting['bin_thresh'] = bin_thresh
    model_name = "vae_{}-hei-{}-alpha{:.0e}-zd{}-hd{}-mbs{}-mbn{}-h{}-l{}-reg{:.0e}-{}-lr{:.0e}".format(generator,
                                                                                              dataset_name,
                                                                                              alpha,
                                                                                              z_dim,
                                                                                              h_dim, mb_size,
                                                                                              batches,
                                                                                              num_layers,
                                                                                              num_lfsteps,
                                                                                              reg, bin_label, lr)
    output_dir = "model/fashion_vae/"

    if save_model:
        #os.mkdir(output_dir + '{}'.format(model_name))
        ckpt_name = output_dir + '{}.ckpt'.format(model_name)
        #setting_filename = output_dir + '{}/setting.json'.format(model_name)
        setting_filename = output_dir + 'setting.json'.format(model_name)
        with open(setting_filename, 'w') as f:
            json.dump(setting, f)
        
            

    global_step = tf.Variable(0, trainable=False, name='global_step')
    global_step2 = tf.Variable(0, trainable=False, name='global_step2')

    device_config = '/device:{}:0'.format(device)
    with tf.device(device_config):
        X_batch_train = tf.placeholder(dtype, shape=[mb_size, X_mnist_dim])
        vaeq = VAEQ_CONV(alpha = alpha, h_dim=h_dim, z_dim=z_dim)
        vae = VAE_DCNN_GPU(h_dim=h_dim, z_dim=z_dim)
      

        
        hamInfNet_hm = HamInfNet(num_layers=num_layers, num_lfsteps=num_lfsteps,
                                 sample_dim=z_dim, dtype=dtype)
       
        q_loss = vaeq.create_loss_train(vae, X_batch_train, batch_size = 1, loss_only=True) # do not train decoder(trained by max E-log-target) params using alpha_divergence
        
        starter_learning_rate = lr
        learning_rate = tf.train.exponential_decay(starter_learning_rate, global_step,
                                                   1000, lr_decay, staircase=True)
        learning_rate2 = tf.train.exponential_decay(starter_learning_rate, global_step2,
                                                   1000, lr_decay, staircase=True)

        
        optimizer = tf.train.AdamOptimizer(learning_rate=learning_rate)
       
        train_op = optimizer.minimize(q_loss, global_step=global_step)

        
       
        optimizer2=tf.train.AdamOptimizer(learning_rate=learning_rate2)
        train_op2=optimizer2.minimize(q_loss,global_step=global_step2)


    loss_q_seq = []

    saved_variables = vae.get_parameters() + hamInfNet_hm.getParams() + vaeq.get_parameters()
    saver = tf.train.Saver(saved_variables, max_to_keep=10) 
    print(saved_variables)
    
    log = []

    with tf.Session() as sess:
        sess.run(tf.global_variables_initializer())
        checkpoint_batch = 5000 #1000
        total_time = 0
        
        for i in np.arange(0,100000):  # pretrain encoder+decoder using stadard VAE/IWAE training scheme
            X_mb_raw,_=dataset.train.next_batch(mb_size)

            X_mb = binarise_fashion_mnist(X_mb_raw)
            start=time.time()
            _,q_loss_i=sess.run([train_op,q_loss],feed_dict={X_batch_train:X_mb})

        
            end = time.time()
            total_time += end - start
        
            loss_q_seq.append(q_loss_i)
            
            if i % 1000 ==999:
                log_line = 'iter: {}, q_loss: {}, time: {}'.format(i+1, np.mean(np.asarray(loss_q_seq)),
                                                                                     total_time)
                print(log_line)
                log.append(log_line + '\n')
                
            
                loss_q_seq.clear()
                
                
                total_time = 0
            if i % checkpoint_batch == 4999:
                #with open(output_dir + '{}/training.cklog'.format(model_name), "a+") as log_file:
                with open(output_dir + 'training.cklog', "a+") as log_file:
                    log_file.writelines(log)
                    log.clear()
        
        for i in np.arange(0, batches):
            X_mb_raw, _ = dataset.train.next_batch(mb_size)
            #X_mb = dybinarize_mnist(X_mb_raw)
            X_mb = binarise_fashion_mnist(X_mb_raw)
            

            start = time.time()

            _,q_loss_i=sess.run([train_op2,q_loss],feed_dict={X_batch_train:X_mb})
            

            end = time.time()
            total_time += end - start

            loss_q_seq.append(q_loss_i)

            if i % 1000 ==999:
                log_line = 'iter: {}, q_loss: {}, time: {}'.format(i + 1,
                                                                                     #np.mean(np.array(loss_seq)),
                                                                                     np.mean(np.array(loss_q_seq)),
                                                                                     #np.mean(np.array(ksd_seq)),
                                                                                     #np.mean(np.array(pot_seq)),
                                                                                     #np.mean(np.array(recon_seq)),
                                                                                     #inflation_i,
                                                                                     total_time)
                print(log_line)
                log.append(log_line + '\n')

                loss_q_seq.clear()

                total_time = 0
            if save_model and i % checkpoint_batch == 4999:
                print("model saved at iter: {}".format(i + 1))
                saver.save(sess, ckpt_name, global_step=global_step2)
                #with open(output_dir + '{}/training.cklog'.format(model_name), "a+") as log_file:
                with open(output_dir + 'training.cklog', "a+") as log_file:
                    log_file.writelines(log)
                    log.clear()
        if save_model:
            saver.save(sess, ckpt_name, global_step=global_step2)
            #with open(output_dir + '{}/training.cklog'.format(model_name), "a+") as log_file:
            with open(output_dir + 'training.cklog', "a+") as log_file:
                log_file.writelines(log)
                log.clear()
    #return output_dir + '{}/'.format(model_name)
    return output_dir


if __name__ == '__main__':
    mnist = input_data.read_data_sets('data/fashion_mnist', one_hot=True)
   
    train(setting=training_setting(32), dataset=mnist, save_model=True, device="GPU") # 32