import argparse
import numpy as np
import numpy.lib as npl
import torch
from torch import nn, optim, autograd
import random

from load import get_years, YEARS
from utils import *
from densratio import densratio

parser = argparse.ArgumentParser(description='Colored MNIST')
parser.add_argument('--hidden_dim', type=int, default=256)
parser.add_argument('--l2_regularizer_weight', type=float,default=0.001)
parser.add_argument('--lr', type=float, default=0.001)
parser.add_argument('--n_restarts', type=int, default=1)
parser.add_argument('--penalty_anneal_iters', type=int, default=1)
parser.add_argument('--rex_penalty_weight', type=float, default=1e4)
parser.add_argument('--steps', type=int, default=500)
parser.add_argument('--plot', action='store_true')
parser.add_argument('--save', type=str, default='')
parser.add_argument('--train_envs', type=str, default='2014,2015,2016')
parser.add_argument('--test_envs', type=str, default='2017,2018')
parser.add_argument('--psel', type=float, default=0.5)
parser.add_argument('--alpha', type=float, default=0.5)
parser.add_argument('--beta', type=float, default=0.5)
flags = parser.parse_args()

train_env_ids = [int(s.strip()) for s in flags.train_envs.split(',')]
if flags.test_envs:
    test_env_ids = [int(s.strip()) for s in flags.test_envs.split(',')]
else:
    test_env_ids = npl.setxor1d(YEARS, train_env_ids)


print('Flags:')
for k, v in sorted(vars(flags).items()):
    print("    {}: {}".format(k, v))

def whiten(x):
    with torch.no_grad():
        x -= x.mean()
        x /= x.std()
    return x

def mean_nll(logits, y):
    return nn.functional.binary_cross_entropy_with_logits(logits, y)

def mean_accuracy(logits, y):
    preds = (logits > 0.).float()
    return ((preds - y).abs() < 1e-2).float().mean()


def get_rex_penalty(train_envs):
    losses = torch.stack([e['nll'] for e in train_envs])
    if len(losses) > 0: 
        penalty = torch.var(losses)
    else:
        penalty = torch.tensor(0.0)
    return penalty


class MLP(nn.Module):
    def __init__(self, input_size):
        super(MLP, self).__init__()
        self.input_size = input_size
        lin1 = nn.Linear(input_size, flags.hidden_dim)
        lin2 = nn.Linear(flags.hidden_dim, flags.hidden_dim)
        lin3 = nn.Linear(flags.hidden_dim, 1)
        # for lin in [lin1, lin2, lin3]:
        #     nn.init.xavier_uniform_(lin.weight)
        #     nn.init.zeros_(lin.bias)
        self._main = nn.Sequential(
            lin1, nn.ReLU(True), #nn.Tanh(), #nn.ReLU(True),
            nn.Dropout(),
            lin2, nn.ReLU(True), #nn.Tanh(), #nn.ReLU(True),
            nn.Dropout(),
            lin3)
        
    def forward(self, x):
        x = x.view(x.shape[0], self.input_size)
        out = self._main(x)
        return out
    
def Q_CRIC(env):
    Q_num=0
    Q_phi=0
    # y_e_E=[]
    # q_ee=[]
    # dr_co_list=[]
    # err=torch.zeros(len(env[]))
    assert type(env)==list
    for i in range(len(env)):
        #x, _ = env[i]
        train_e=env[i]
        y = train_e['logits']
        x = train_e['images']
        q_e=torch.mean(y)
        # q_ee.append(q_e)
        # for j in env[-i]:
        for j in range(len(env)):
            if j == i:
                continue
            #x_t, _ = env[j]
            train_t=env[j]
            y_t=train_t['logits']
            x_t=train_t['images']
            dr_co=densratio(x.numpy(),x_t.numpy())

            w_co=torch.from_numpy(dr_co.compute_density_ratio(x_t.numpy())) 
            y_t = w_co*train_t['logits']
            q_e_co=torch.mean(y_t)  #calculate the weighted q
        
            Q_num=Q_num+(q_e_co-q_e)**2
            Q_num=Q_num.detach()
      
    return Q_num

final_train_accs = []
final_test_accs = []
logs = []
for restart in range(flags.n_restarts):
    print("Restart", restart)

    train_envs = get_years(train_env_ids)
    test_envs = get_years(test_env_ids)
    # preprocess
    for e in train_envs + test_envs:
        e['images'] = whiten(e['images'])
    
    # init
    logger = Logger()
    mlp = MLP(train_envs[0]['images'].shape[1])
    optimizer = optim.Adam(mlp.parameters(), lr=flags.lr)
    #train_envs = create_mixed_logit_environments(mlp, train_envs, flags)
    print_env_info(train_envs, test_envs)

    pretty_print('step', 'train nll', 'train acc', 'rex penalty', 'test acc')

    for step in range(flags.steps):
        for env in train_envs + test_envs:
            assert not torch.isnan(env['images']).any()
            env['logits'] = mlp(env['images'])
            env['nll'] = mean_nll(env['logits'], env['labels'])
            env['acc'] = mean_accuracy(env['logits'], env['labels'])

        train_nll = torch.stack([e['nll'] for e in train_envs]).mean()
        train_acc = torch.stack([e['acc'] for e in train_envs]).mean()
        rex_penalty = get_rex_penalty(train_envs)


        weight_norm = torch.tensor(0.)
        for w in mlp.parameters():
            weight_norm += w.norm().pow(2)

        loss = train_nll.clone()
        loss += flags.l2_regularizer_weight * weight_norm

        if flags.rex_penalty_weight != 0:
            if step >= flags.penalty_anneal_iters:
                loss /= flags.rex_penalty_weight
            loss += rex_penalty


        optimizer.zero_grad()
        loss.backward()
        optimizer.step()

        logger.log('train_nll', train_nll)
        logger.log('train_acc', train_acc)
        logger.log('rex_penalty', rex_penalty)
        logger.log('test_acc', [e['acc'] for e in test_envs])
        logger.log('losses', [e['nll'] for e in train_envs])

        if step % 100 == 0:
            print_stats(step, logger)

    final_train_accs.append(np.mean(logger['train_acc'][-50:]))
    final_test_accs.append(np.mean(logger['test_acc'][-50:]))
    print('Final train acc (mean/std across restarts so far):')
    print(np.mean(final_train_accs), np.std(final_train_accs))
    print('Final test acc (mean/std across restarts so far):')
    print(np.mean(final_test_accs), np.std(final_test_accs))

    logs.append(logger)

    if flags.plot:
        plot(logger)

#if flags.save:
#    save(logs, 'results/%s_%s_%s' % (flags.save, ','.join([str(e) for e in train_env_ids]), ','.join([str(e) for e in test_env_ids])))

Q = Q_CRIC(train_envs)
Q_t = Q_CRIC(test_envs)
print("Q:", Q)
print("Q_t:", Q_t)
