import argparse
import os
os.environ['EQX_ON_ERROR'] = 'nan'
import jax
import jax.numpy as jnp
import numpy as np
import equinox as eqx
import optax as optx
from PHIROM.modules.models import NodeROM, DecoderArchEnum
from PHIROM.pde.data_utils import NumpyLoader, JaxLoader
from functools import partial
from jaxtyping import PRNGKeyArray, Array
from PHIROM.training.train import NodeTrainingModeEnum, IrregularPhiROMTrainer
from PHIROM.training.baseline import IrregularDINoTrainer
from PHIROM.utils.serial import make_CROMOffline, save_model, load_model
from PHIROM.training.callbacks import CheckpointCallback, NODEUnrollingEvaluationCallback
from pathlib import Path
from datetime import datetime
from PHIROM.pde.navier_turbulence_irregular import *
from torch.utils.data import DataLoader
from PHIROM.utils.experiment_utils import *

parser = argparse.ArgumentParser()
parser.add_argument("--latent_dim", type=int, default=100)
parser.add_argument("--width", type=int, default=64)
parser.add_argument("--activation", type=str, default="sin")
parser.add_argument("--node_activation", type=str, default="swish")
parser.add_argument("--node_width", type=int, default=64)
parser.add_argument("--epochs", type=int, default=20000)
parser.add_argument("--dataset", type=str, default="ns_turbulence_irregular1024_64x64_ins=5")
parser.add_argument("--prefix", type=str, default="")
parser.add_argument("--seed", type=int, default=101)
parser.add_argument("--loss", type=str, default="nmse")
parser.add_argument("--ode_solver", type=str, default="bosh3", choices=["bosh3", "dopri5", "euler"])
parser.add_argument("--adaptive", action=argparse.BooleanOptionalAction, default=False)
parser.add_argument("--max_ode_steps", type=int, default=None)
parser.add_argument("--dino", action=argparse.BooleanOptionalAction, default=False, help="Train Data-Driven only (DINo)")
parser.add_argument("--gamma", type=float, default=1.0)
parser.add_argument("--gamma_decay_rate", type=float, default=0.99)
parser.add_argument("--gamma_epochs", type=int, default=10, help="Scheduling gamma decay epochs")
parser.add_argument("--final_gamma", type=float, default=0.0)
parser.add_argument("--final_lr", type=float, default=1e-6)
parser.add_argument("--decay_steps", type=int, default=50)
parser.add_argument("--decay_rate", type=float, default=0.985)
parser.add_argument("--num_samples", type=int, default=10)
parser.add_argument("--autodecoder", action=argparse.BooleanOptionalAction, default=True, help="Use autodecoder")
parser.add_argument("--max_step", type=int, default=20)
parser.add_argument("--evolve_start", type=int, default=0)
parser.add_argument("--decoder_arch", type=str, default="hyper", choices=['mlp', 'hyper'])
parser.add_argument("--node_arch", type=str, default="mlp", choices=['mlp', 'hyper_concat'])
parser.add_argument("--node_training_mode", type=str, default=str(NodeTrainingModeEnum.JACOBIAN_INVERSE), 
                    choices=[NodeTrainingModeEnum.JACOBIAN_INVERSE, NodeTrainingModeEnum.JACOBIAN_PSI, NodeTrainingModeEnum.ZERO, NodeTrainingModeEnum.LABELS])
parser.add_argument("--batch_size", type=int, default=128)
parser.add_argument("--learning_rate_decoder", type=float, default=1e-2)
parser.add_argument("--learning_rate_node", type=float, default=-1)
parser.add_argument("--learning_rate_latent", type=float, default=-1)
parser.add_argument("--normalize", action=argparse.BooleanOptionalAction, default=True)
parser.add_argument("--pinn", action=argparse.BooleanOptionalAction, default=False)
parser.add_argument("--loss_lambda", type=float, default=0.8)

args = parser.parse_args()
latent_dim = args.latent_dim
width = args.width
activation = args.activation
node_activation = args.node_activation
epochs = args.epochs
dataset_name = args.dataset
prefix = args.prefix
seed = args.seed
loss = args.loss
DINO = args.dino
final_lr = args.final_lr
decay_steps = args.decay_steps
decay_rate = args.decay_rate
gamma = args.gamma
gamma_decay_rate = args.gamma_decay_rate
gamma_epochs = args.gamma_epochs
final_gamma = args.final_gamma
num_samples = args.num_samples
autodecoder = args.autodecoder
max_step = args.max_step
evolve_start = args.evolve_start
max_split = 0
split_start = 0
arch = args.decoder_arch
ode_solver = args.ode_solver
adaptive = args.adaptive
max_ode_steps = args.max_ode_steps
node_activation = args.node_activation
node_width = args.node_width
node_arch = args.node_arch
batch_size = args.batch_size
learning_rate_decoder = args.learning_rate_decoder
learning_rate_node = args.learning_rate_node
learning_rate_latent = args.learning_rate_latent
normalize = args.normalize
node_training_mode = args.node_training_mode
loss_lambda = args.loss_lambda

paramed = False
path = f'data/{dataset_name}.h5'

if paramed:
    param_dim = 1
else:
    param_dim = 0
    if node_arch == "hyper" or node_arch == "hyperV2":
        raise ValueError("Hyper node not supported for non-parametric datasets")

if autodecoder:
    if not DINO:
        dataset_train = IrregularDecayingTurbulenceDatasetTorch(path, max_step, indices=(0, num_samples), paramed=paramed)
    else:
        dataset_train = IrregularDecayingTurbulenceTrajDatasetTorch(path, max_step, indices=(0, num_samples), paramed=paramed, return_regular=True)
    dataset_validation = IrregularDecayingTurbulenceTrajDatasetTorch(path, max_step*2, indices=(256, 256 + 16), paramed=paramed)
    subdataset_train = IrregularDecayingTurbulenceTrajDatasetTorch(path, max_step*2, indices=(0, 16), paramed=paramed)
else:
    raise NotImplementedError("AE Not implemented")

loader_train = DataLoader(dataset_train, batch_size=batch_size, shuffle=True)
decay_steps = len(loader_train) * decay_steps

print(f"Training on {dataset_name} dataset - num batches: {len(loader_train)} - num samples: {num_samples} - max step: {max_step}")
print(f"Decay every {decay_steps} steps with rate {decay_rate}")

if paramed:
    MEAN_NODE_ARGS = dataset_train.node_args.mean(axis=0).numpy()
    STD_NODE_ARGS = dataset_train.node_args.std(axis=0).numpy()
else:
    MEAN_NODE_ARGS = None
    STD_NODE_ARGS = None

path, name = get_path_and_name(args)

MEAN, STD = dataset_train.compute_mean_std_fields()
nx = dataset_train.x.shape[0]
ny = dataset_train.y.shape[0]

hyperparams = {"latent_dim": latent_dim, "num_sensors": nx * ny, "field_dim": 2, 
               "spatial_dim": 2, "mean_field": MEAN if normalize else None, "std_field": STD if normalize else None,
               "activation": activation, "node_kwargs": {
                   "node_arch": node_arch,
                   "activation": node_activation,
                   "depth": 4,
                   "width": node_width,
                   "param_size": param_dim,
                   "solver": ode_solver,
                   "adaptive": adaptive,
                   "max_steps": max_ode_steps,
                   "mean_params": MEAN_NODE_ARGS,
                   "std_params": STD_NODE_ARGS
               }}

if arch == "mlp":
    arch = DecoderArchEnum.MLP
    hyperparams["width_scale"] = width
    hyperparams["decoder_arch"] = arch
elif arch == "hyper":
    arch = DecoderArchEnum.HYPER
    hyperparams["decoder_arch"] = arch
    hyperparams["width"] = width
    hyperparams['n_layers'] = 3
    hyperparams['std_coords'] = True
    hyperparams['input_scale'] = max(dataset_train.length)

if activation in ["tanh", "relu", "swish", "elu", 'softplus']:
    mean_x, std_x = dataset_train.compute_mean_std_coords()
    hyperparams["mean_x"] = mean_x
    hyperparams["std_x"] = std_x
elif activation == "sin":
    min_x, max_x = dataset_train.compute_min_max_coords()
    print(min_x, max_x)
    hyperparams["min_x"] = min_x
    hyperparams["max_x"] = max_x

key = jax.random.PRNGKey(seed)
key, subkey = jax.random.split(key)

model, model_state = eqx.nn.make_with_state(NodeROM)(**hyperparams, key=subkey)

path_experiment = os.path.join("NODE_experiments", path, name)
path_checkpoint = os.path.join(path_experiment, "checkpoints")
Path(path_experiment).mkdir(parents=True, exist_ok=True)

callbacks = [CheckpointCallback(path_checkpoint, name, hyperparams, True, 200),
             NODEUnrollingEvaluationCallback(dataset_validation, max_step, max_step*2, 100, plot_results=False, plot_dir=path_experiment, dict_key_prefix="validation_unrolling", batch_size=4),
             NODEUnrollingEvaluationCallback(subdataset_train, max_step, max_step*2, 100, plot_results=True, plot_dir=path_experiment, dict_key_prefix="train_unrolling", batch_size=4)]


key, subkey = jax.random.split(key)
if not DINO and node_training_mode != NodeTrainingModeEnum.ZERO:
    evolve_fn = cfd_residual_builder(dataset_train.cfd_grid)
else:
    evolve_fn = None

if DINO:
    scheduler = optx.schedules.exponential_decay(learning_rate_decoder, decay_steps, decay_rate, end_value=final_lr, staircase=True) if decay_rate < 1.0 else learning_rate_decoder
    optimizer =optx.adam(scheduler)
    scheduler_node = optx.schedules.exponential_decay(learning_rate_node, decay_steps, decay_rate, end_value=final_lr, staircase=True) if decay_rate < 1.0 else learning_rate_node
    optimizer_node = optx.adam(scheduler_node)
    scheduler_latent = optx.schedules.exponential_decay(learning_rate_latent, decay_steps, decay_rate, end_value=final_lr, staircase=True) if decay_rate < 1.0 else learning_rate_latent
    assert learning_rate_latent > 0, "Learning rate for latent variable must be positive"
    optimizer_latent = optx.adam(scheduler_latent)
    trainer = IrregularDINoTrainer(model=model, model_state=model_state, optimizer=optimizer, optimizer_node=optimizer_node, optimizer_latent=optimizer_latent, 
                                 loss=loss, evolve_fn=evolve_fn, evolve_start=evolve_start, max_evolve_split=max_split, split_start=split_start, random_split=False, 
                                 num_trajectories=num_samples, num_time_steps=max_step, latent_dim=latent_dim, callbacks=callbacks, gamma=gamma, 
                                 gamma_decay_rate=gamma_decay_rate, gamma_decay_epochs=gamma_epochs, final_gamma=final_gamma, key=subkey)

elif not DINO:
    scheduler = optx.schedules.exponential_decay(learning_rate_decoder, decay_steps, decay_rate, end_value=final_lr, staircase=True) if decay_rate < 1.0 else learning_rate_decoder
    print(scheduler)
    optimizer = optx.adamw(scheduler)
    if learning_rate_node > 0:
        scheduler_node = optx.schedules.exponential_decay(learning_rate_node, decay_steps, decay_rate, end_value=final_lr, staircase=True) if decay_rate < 1.0 else learning_rate_node
        optimizer_node = optx.adamw(scheduler_node)
    else:
        optimizer_node = None
    if learning_rate_latent > 0:
        scheduler_latent = optx.schedules.exponential_decay(learning_rate_latent, decay_steps, decay_rate, end_value=final_lr, staircase=True) if decay_rate < 1.0 else learning_rate_latent
        optimizer_latent = optx.adamw(scheduler_latent)
    else:
        optimizer_latent = None
    trainer = IrregularPhiROMTrainer(model=model, model_state=model_state, optimizer=optimizer, optimizer_node=optimizer_node, optimizer_latent=optimizer_latent, 
                                        node_training_mode=node_training_mode, loss=loss, evolve_fn=evolve_fn, evolve_start=evolve_start, num_trajectories=num_samples, 
                                        num_time_steps=max_step, latent_dim=latent_dim, callbacks=callbacks, gamma=gamma, key=subkey, loss_lambda=loss_lambda)   
model, model_state, opt_state, history = trainer.fit(loader_train, epochs=epochs, warm_start=True)

save_model(os.path.join(path_experiment, "model.eqx"), hyperparams, model, model_state)
history["loss_reconstruction"] = np.array(history["loss_reconstruction"])
history["loss_time_stepping"] = np.array(history["loss_time_stepping"])
np.savez(os.path.join(path_experiment, "history.npz"), **history)

if autodecoder:
    l = np.array(trainer.latent_memory)
    np.save(os.path.join(path_experiment, "latent_memory.npy"), l)

