import sys
import random
import numpy as np
import chainer
import chainer.functions as F
import chainer.links as L
from chainer.datasets import split_dataset_random, tuple_dataset
from chainer import training, iterators, serializers
from chainer.backends import cuda
from chainer.training import extensions
from chainer.reporter import report
from chainer import static_graph
from estimator import RecoNet
from sigmoid_cross_entropy_util import sigmoid_cross_entropy
import argparse


def encode_acc(a):
    if a < -0.01:
        return 0
    if a > 0.01:
        return 2
    return 1


def encode_om(w):
    if w < -0.2:
        return 0
    if w < -0.02:
        return 1
    if w > 0.2:
        return 4
    if w > 0.02:
        return 3
    return 2


def encode_action(x):
    return encode_acc(x[0]) * 5 + encode_om(x[1])


if __name__ == '__main__':
    parser = argparse.ArgumentParser(
        prog='car_trainer.py',
        description='train threat network',
        add_help=True
    )
    parser.add_argument('--gpu', dest='gpu', action='store_true')
    parser.add_argument('-n', dest='n_epoch', type=int, default=25)
    parser.add_argument('--batch', dest='batch', type=int, default=512)
    parser.add_argument('--lr', dest='lr', type=float, default=1e-3)
    args = parser.parse_args()

    xp = cuda.cupy if args.gpu else np
    device = 0 if args.gpu else -1
    train_val = np.load('data/jam_car_threat.npy')
    print('data size :', len(train_val))

    action = train_val[:, 5:7]
    idx = np.apply_along_axis(encode_action, 1, action).reshape(-1, 1)
    teacher = np.concatenate((idx, train_val[:, -1:]), axis=1)

    if args.gpu:
        train_val = xp.asarray(train_val)
        teacher = xp.asarray(teacher)

    train_val = tuple_dataset.TupleDataset(
        train_val[:, :5], teacher)
    n_train = int(len(train_val) * 0.8)
    train, val = split_dataset_random(train_val, n_train, seed=0)

    train_iter = iterators.SerialIterator(train, args.batch)
    valid_iter = iterators.SerialIterator(
        val, args.batch, repeat=False, shuffle=False)

    net = RecoNet()
    model = L.Classifier(net, lossfun=sigmoid_cross_entropy)
    model.compute_accuracy = False

    if args.gpu:
        model.to_gpu(0)

    optimizer = chainer.optimizers.Adam(alpha=args.lr)
    optimizer.setup(model)

    updater = training.StandardUpdater(train_iter, optimizer, device=device)

    trainer = training.Trainer(updater, (args.n_epoch, 'epoch'), out='results')

    trainer.extend(extensions.LogReport())
    trainer.extend(extensions.Evaluator(
        valid_iter, model, device=device), name='val')
    trainer.extend(extensions.PrintReport(
        ['epoch', 'main/loss', 'val/main/loss', 'elapsed_time']))
    trainer.extend(extensions.PlotReport(
        ['main/loss', 'val/main/loss'], x_key='epoch', file_name='loss.png'))
    trainer.extend(extensions.dump_graph('main/loss'))

    trainer.run()

    chainer.serializers.save_npz('jam/car-threat.model', net)
