#
# Perform classification on CUB Birds 200-2011.
#

import os
import sys
import numpy as np
import argparse
import torch
import torch.nn as nn
import torch.nn.functional as F
import torch.optim as optim
from   torchvision import datasets, transforms

import helper
sys.path.append("models/cub/")
import resnet
    

################################################################################
# Training epoch.
################################################################################

#
# Main training function for training using polar protoypical networks,
# softmax cross-entropy, and squared loss.
#
# Works for both classification and regression.
#
# model (object)      - Network.
# device (torch)      - Torch device, e.g. CUDA or CPU.
# trainloader (torch) - Training data.
# optimizer (torch)   - Type of optimizer.
# f_loss (torch)      - Loss function.
# approach (str)      - ce, polar, euclidean.
# epoch (int)         - Epoch iteration.
#
def main_train(model, device, trainloader, optimizer, f_loss, approach, epoch):
    # Set mode to training.
    model.train()
    avgloss, avglosscount = 0., 0.
    
    # Go over all batches.
    for bidx, (data, target) in enumerate(trainloader):
        # Data to device.
        if approach == "polar":
            target = model.polars[target]
        #target = target.cuda(async=True)
        data   = torch.autograd.Variable(data).cuda()
        target = torch.autograd.Variable(target).cuda()
        
        # Compute outputs and losses.
        output = model(data)
        if approach == "polar":
            #loss = (1 - f_loss(output, target)).sum()
            loss = (1 - f_loss(output, target)).pow(2).sum()
        elif approach == "ce":
            loss = f_loss(output, target)
        
        # Backpropagation.
        optimizer.zero_grad()
        loss.backward()
        optimizer.step()
        
        # Update loss.
        avgloss += loss.item()
        avglosscount += 1.
        newloss = avgloss / avglosscount
        
        # Print updates.
        print "Training epoch %d: loss %8.4f - %.0f\r" \
                %(epoch, newloss, 100.*(bidx+1)/len(trainloader)),
        sys.stdout.flush()
    print

################################################################################
# Testing epoch.
################################################################################

#
# Main test function using polar protoypical networks,
# softmax cross-entropy, and squared loss.
#
# Works for both classification and regression.
#
# model (object)             - Network.
# device (torch)             - Torch device, e.g. CUDA or CPU.
# testloader (torch)         - Test data.
# approach (str)             - ce, polar, euclidean.
# classpolars (torch Tensor) - Polar locations per class.
#
def main_test(model, device, testloader, approach):
    # Set model to evaluation and initialize accuracy and cosine similarity.
    model.eval()
    cos = nn.CosineSimilarity(eps=1e-9)
    acc = 0
    
    # Go over all batches.
    with torch.no_grad():
        for data, target in testloader:
            # Data to device.
            data = torch.autograd.Variable(data).cuda()
            target = target.cuda(async=True)
            target = torch.autograd.Variable(target)
            
            # Forward.
            output = model(data).float()

            # For polar, obtain class scores.
            if approach == "polar":
                output = model.predict(output).float()
                
            pred = output.max(1, keepdim=True)[1]
            acc += pred.eq(target.view_as(pred)).sum().item()
    
    # Print results.
    testlen = len(testloader.dataset)
    print "Testing: classification accuracy: %d/%d - %.3f" \
            %(acc, testlen, 100. * acc / testlen)
    return acc / float(testlen)

################################################################################
# Main entry point of the script.
################################################################################

#
# Parse all user arguments.
#
def parse_args():
    parser = argparse.ArgumentParser(description="CUB-200-2011 classification")
    parser.add_argument("-a", dest="approach", default="polar", type=str)
    
    parser.add_argument("--datadir", dest="datadir", default="data/", type=str)
    parser.add_argument("--resdir", dest="resdir", default="results/classification/", type=str)
    parser.add_argument("--polarfile", dest="polarfile", default="", type=str)

    parser.add_argument("-n", dest="network", default="resnet32", type=str)
    parser.add_argument("-r", dest="optimizer", default="sgd", type=str)
    parser.add_argument("-l", dest="learning_rate", default=0.01, type=float)
    parser.add_argument("-m", dest="momentum", default=0.9, type=float)
    parser.add_argument("-c", dest="decay", default=0.0001, type=float)
    parser.add_argument("-s", dest="batch_size", default=64, type=int)
    parser.add_argument("-e", dest="epochs", default=250, type=int)
    parser.add_argument("--drop1", dest="drop1", default=100, type=int)
    parser.add_argument("--drop2", dest="drop2", default=200, type=int)
    parser.add_argument("--seed", dest="seed", default=100, type=int)
    
    parser.add_argument("--dosave", dest="dosave", default=1, type=int)
    args = parser.parse_args()
    return args

#
# Main entry point of the script.
#
if __name__ == "__main__":
    # Parse user parameters and set device.
    args     = parse_args()
    os.environ["CUDA_VISIBLE_DEVICES"] = "0"
    device   = torch.device("cuda")
    kwargs   = {'num_workers': 64, 'pin_memory': True}
    approach = args.approach
    args.dataset = "cub"

    # Set the random seeds.
    seed = args.seed
    torch.manual_seed(seed)
    torch.cuda.manual_seed(seed)
    torch.cuda.manual_seed_all(seed)
    np.random.seed(seed)
    
    # Load data.
    batch_size = args.batch_size
    trainloader, testloader = helper.load_cub(args.datadir, \
            batch_size, kwargs)
    nr_classes = 200

    # Load the polars and update the trainy labels.
    if approach == "polar":
        classpolars = torch.from_numpy(np.load(args.polarfile)).float()
        args.output_dims = int(args.polarfile.split("/")[-1].split("-")[1][:-1])
    else:
        classpolars = None
        args.output_dims = nr_classes
    
    # Load the model.
    if args.network == "resnet32":
        model = resnet.ResNet34(args.output_dims, classpolars)
    model = model.to(device)
    
    # Load the optimizer.
    optimizer = helper.get_optimizer(args.optimizer, model.parameters(), \
            args.learning_rate, args.momentum, args.decay)
    
    # Initialize the loss functions.
    if approach == "polar":
        f_loss = nn.CosineSimilarity(eps=1e-9).cuda()
    else:
        f_loss = nn.CrossEntropyLoss().cuda()

    # Main loop.
    testscores = []
    learning_rate = args.learning_rate
    for i in xrange(args.epochs):
        print "---"
        # Learning rate decay.
        if i in [args.drop1, args.drop2]:
            learning_rate *= 0.1
            for param_group in optimizer.param_groups:
                param_group['lr'] = learning_rate
        
        # Train and test.
        main_train(model, device, trainloader, optimizer, f_loss, approach, i)
        if i % 10 == 0 or i == args.epochs - 1:
            t = main_test(model, device, testloader, approach)
            testscores.append([i,t])
    
    # Store the test scores.
    if args.dosave == 1:
        testscores = np.array(testscores)
        if approach == "ce":
            helper.store_results_ce(args, testscores, "data-acc.txt")
        elif approach == "polar":
            helper.store_results_polar(args, testscores, "data-acc.txt")
