# Copyright 2024 Luping Liu
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.


import os
import sys
import math
import json
import inspect
import random
import itertools
import argparse
import platform
import numpy as np
import wandb.util
from tqdm.auto import tqdm
from PIL import Image
import matplotlib.pyplot as plt

import torch
import torch.nn.functional as F
import torch.utils.checkpoint
from torch.utils.data import Dataset
from torchvision import transforms
from accelerate import Accelerator
from diffusers.optimization import get_scheduler
from diffusers import AutoencoderKL, DDPMScheduler, UNet2DConditionModel
from transformers import AutoTokenizer, T5EncoderModel, CLIPTextModel

from modules.lora import inject_trainable_lora_extended, save_lora_weight
from modules.lora import monkeypatch_or_replace_lora_extended, collapse_lora, monkeypatch_remove_lora
from modules.adapters import TextAdapter
from tools import load_dataset, collate_fn, caption2embed, sample_images
from tools import get_predicted_original, DDIMSolver


# Arguments
def parse_args():
    parser = argparse.ArgumentParser(description="LongAlign Training")
    parser.add_argument("--pretrained_decoder", type=str, default="stabilityai/stable-diffusion-2-1")
    parser.add_argument("--mixed_precision", type=str, default="no")
    parser.add_argument("--token_length", type=int, default=77)
    parser.add_argument("--resolution", type=int, default=512)
    parser.add_argument("--lora_rank", type=int, default=32)
    parser.add_argument("--loss_type", type=str, default="l2")
    parser.add_argument("--batch_size", type=int, default=1)
    parser.add_argument("--accumulation_steps", type=int, default=1)
    parser.add_argument("--learning_rate", type=float, default=1e-5)
    parser.add_argument("--gradient_checkpointing", action="store_true")
    parser.add_argument("--max_train_epochs", type=int, default=1)
    parser.add_argument("--validation_steps", type=int, default=500)
    parser.add_argument("--save_steps", type=int, default=1250)
    parser.add_argument("--local_rank", type=int, default=-1)
    parser.add_argument("--output_dir", type=str, default="")
    parser.add_argument("--ckpt_dir", type=str, default="")
    parser.add_argument("--resume", action="store_true")
    parser.add_argument("--debug", action="store_true")
    args = parser.parse_args()

    env_local_rank = int(os.environ.get("LOCAL_RANK", -1))
    if env_local_rank != -1 and env_local_rank != args.local_rank:
        args.local_rank = env_local_rank
    return args


def main(args):
    accelerator = Accelerator(
        mixed_precision=args.mixed_precision,
        gradient_accumulation_steps=args.accumulation_steps,
        log_with='wandb' if not args.debug else None,
    )
    if accelerator.is_main_process and not args.debug:
        os.makedirs(args.output_dir, exist_ok=True)

    weight_dtype = torch.float32
    if args.mixed_precision == "fp16":
        weight_dtype = torch.float16
    elif args.mixed_precision == "bf16":
        weight_dtype = torch.bfloat16

    # Blocks to inject LoRA
    # note: result without CrossAttention is poor
    VIS_REPLACE_MODULES = {"ResnetBlock2D", "CrossAttention", "Attention", "GEGLU"}

    # Modules of T2I diffusion models
    vae = AutoencoderKL.from_pretrained(args.pretrained_decoder, subfolder="vae", torch_dtype=weight_dtype)
    vis = UNet2DConditionModel.from_pretrained(args.pretrained_decoder, subfolder="unet", torch_dtype=weight_dtype)

    tokenizer_clip = AutoTokenizer.from_pretrained(args.pretrained_decoder, subfolder="tokenizer",
                                                   torch_dtype=weight_dtype, use_fast=False)
    text_encoder_clip = CLIPTextModel.from_pretrained(args.pretrained_decoder, subfolder="text_encoder",
                                                      torch_dtype=weight_dtype)
    tokenizer_t5 = AutoTokenizer.from_pretrained("google-t5/t5-large", torch_dtype=weight_dtype,
                                                 model_max_length=512)
    text_encoder_t5 = T5EncoderModel.from_pretrained("google-t5/t5-large", torch_dtype=weight_dtype)

    # download from https://huggingface.co/shihaozhao/LaVi-Bridge/tree/main/t5_unet/adapter --> ./model/LaVi-Bridge
    adapter = TextAdapter.from_pretrained('./model/LaVi-Bridge')

    noise_scheduler = DDPMScheduler.from_pretrained(args.pretrained_decoder, subfolder="scheduler",
                                                    torch_dtype=weight_dtype)
    alpha_schedule = torch.sqrt(noise_scheduler.alphas_cumprod)
    sigma_schedule = torch.sqrt(1 - noise_scheduler.alphas_cumprod)
    args.num_ddim_timesteps = 50
    solver = DDIMSolver(noise_scheduler.alphas_cumprod.numpy(),
                        timesteps=noise_scheduler.config.num_train_timesteps,
                        ddim_timesteps=args.num_ddim_timesteps)

    if args.ckpt_dir != "":
        args.ckpt_dir = args.ckpt_dir.split(", ")

        for ckpt_dir in args.ckpt_dir:
            # LoRA
            monkeypatch_or_replace_lora_extended(
                vis,
                torch.load(os.path.join(ckpt_dir, f"lora_vis.pt"), map_location="cpu"),
                r=32,
                target_replace_module=VIS_REPLACE_MODULES,
            )

            # merge LoRA
            collapse_lora(vis, VIS_REPLACE_MODULES)
            monkeypatch_remove_lora(vis)

    vae.requires_grad_(False)
    vis.requires_grad_(False)
    text_encoder_clip.requires_grad_(False)
    text_encoder_t5.requires_grad_(False)

    # LoRA injection
    vis_lora_params, _ = inject_trainable_lora_extended(
        vis,
        r=args.lora_rank,
        target_replace_module=VIS_REPLACE_MODULES,
    )

    if args.gradient_checkpointing:
        vis.enable_gradient_checkpointing()

    # Dataset and dataloader
    train_dataset, args.validation_prompts = load_dataset(args)
    train_dataloader = torch.utils.data.DataLoader(train_dataset, batch_size=args.batch_size, prefetch_factor=3,
                                                   num_workers=6 if not args.debug else 6, collate_fn=collate_fn,
                                                   shuffle=True, drop_last=True, pin_memory=True)
    args.max_train_steps = len(train_dataset) * args.max_train_epochs // \
                           (args.batch_size * accelerator.num_processes * accelerator.gradient_accumulation_steps)

    # Optimizer and scheduler
    optimizer_class = torch.optim.AdamW
    params_to_optimize = ([
        {"params": itertools.chain(*vis_lora_params)},
        {"params": adapter.parameters()},
    ])
    # todo check 1e-6 and 1e-8 optim_eps
    optimizer = optimizer_class(params_to_optimize, lr=args.learning_rate, betas=(0.9, 0.999),
                                weight_decay=1e-2, eps=1e-8)
    lr_scheduler = get_scheduler("constant_with_warmup", optimizer=optimizer,
                                 num_warmup_steps=2000 * accelerator.num_processes,
                                 num_training_steps=args.max_train_steps * accelerator.num_processes)

    vae.to(accelerator.device, weight_dtype)
    text_encoder_clip.to(accelerator.device, weight_dtype)
    text_encoder_t5.to(accelerator.device, weight_dtype)
    alpha_schedule = alpha_schedule.to(accelerator.device, weight_dtype)
    sigma_schedule = sigma_schedule.to(accelerator.device, weight_dtype)
    get_predicted_original_sample = lambda model_output, timesteps, sample: get_predicted_original(
        model_output, timesteps, sample, noise_scheduler.config.prediction_type, alpha_schedule, sigma_schedule)
    solver = solver.to(accelerator.device, weight_dtype)
    vae.eval()
    text_encoder_clip.eval()
    text_encoder_t5.eval()
    vis, adapter, optimizer, train_dataloader, lr_scheduler = accelerator.prepare(
        vis, adapter, optimizer, train_dataloader, lr_scheduler,)

    caption2embed_simple = lambda captions: caption2embed(captions, [tokenizer_clip, tokenizer_t5], [text_encoder_clip, text_encoder_t5],
                                                          args, accelerator.device, weight_dtype)

    if not os.path.exists(f"{args.output_dir}/latest_status"):
        args.resume = False
    if args.resume:
        accelerator.load_state(f"{args.output_dir}/latest_status")
        resume_json = json.load(open(f"{args.output_dir}/latest_status/resume.json", 'r'))
        global_step = resume_json['global_step']
        last_save = global_step // accelerator.gradient_accumulation_steps
        num_train_epochs = args.max_train_epochs - global_step // len(train_dataset)
        wandb_id = resume_json['wandb_id']
    else:
        global_step = last_save = 0
        num_train_epochs = args.max_train_epochs
        wandb_id = wandb.util.generate_id()
        resume_json = {'global_step': global_step, 'wandb_id': wandb_id}

    # Log
    if accelerator.is_main_process:
        if not args.debug:
            tracker_config = dict(vars(args))
            accelerator.init_trackers(project_name='dense-final-sea', config=tracker_config,
                                      init_kwargs={'wandb': {'name': args.output_dir.split('/')[-1],
                                                             'resume': args.resume, 'id': wandb_id}, })

            adapter.eval()
            vis.eval()
            sample_images(vae, adapter, caption2embed_simple, vis, args, accelerator, weight_dtype, global_step=0)

        print(f"Comuting node = {platform.node()}")
        print(f"Num examples = {len(train_dataset)}")
        print(f"Total batch size = {args.batch_size * accelerator.num_processes * accelerator.gradient_accumulation_steps}")
        print(f"Num Epochs = {num_train_epochs}")
        print(f"Total optimization steps = {args.max_train_steps}")

    accelerator.wait_for_everyone()

    progress_bar = tqdm(range(args.max_train_steps), disable=not accelerator.is_local_main_process, dynamic_ncols=True)
    progress_bar.set_description("Steps")
    if args.resume:
        progress_bar.update(global_step // accelerator.gradient_accumulation_steps)

    # Training
    for _ in range(num_train_epochs):
        adapter.train()
        vis.train()

        for _, batch in enumerate(train_dataloader):
            with accelerator.accumulate(vis, adapter):
                with torch.no_grad():
                    # Latent preparation
                    latents = []
                    batch['pixel_values'] = batch['pixel_values'].to(weight_dtype)
                    for i in range(0, batch['pixel_values'].shape[0], 8):
                        latents.append(vae.encode(batch['pixel_values'][i: i + 8]).latent_dist.sample())
                    latents = torch.cat(latents, dim=0)
                    # latents = vae.encode(batch['pixel_values'].to(weight_dtype)).latent_dist.sample()
                    latents = latents * vae.config.scaling_factor
                    bsz = latents.shape[0]

                    start_timesteps = torch.randint(0, noise_scheduler.config.num_train_timesteps, (latents.shape[0],),
                                                    device=latents.device).long()
                    # timesteps = start_timesteps - solver.step_ratio
                    # timesteps = torch.where(timesteps < 0, torch.zeros_like(timesteps), timesteps)

                    noise = torch.randn_like(latents)
                    noisy_model_input = noise_scheduler.add_noise(latents, noise, start_timesteps)
                    # noisy_model_input_pre = noise_scheduler.add_noise(latents, noise, timesteps)

                    # 6. Prepare prompt embeds and unet_added_conditions
                    # captions = [c if random.random() > 0.1 else '' for c in batch['caption']]
                    captions = batch['caption']
                    # encoder_hidden_states_pre = text_encoder(batch['input_ids'])[0]
                    encoder_hidden_states_pre = caption2embed_simple(captions)
                    encoder_hidden_states_clip = encoder_hidden_states_pre['encoder_hidden_states_clip_concat']
                    encoder_hidden_states_t5 = encoder_hidden_states_pre["encoder_hidden_states_t5"]

                # import pdb; pdb.set_trace()
                encoder_hidden_states_t5 = adapter(encoder_hidden_states_t5).sample
                encoder_hidden_states_ct5 = torch.cat([encoder_hidden_states_clip, encoder_hidden_states_t5], dim=1)
                # 7. Get online LCM prediction on z_{t_{n + k}} (noisy_model_input), w, c, t_{n + k} (start_timesteps)
                noise_pred = vis(noisy_model_input, start_timesteps, encoder_hidden_states_ct5).sample
                # pred_x_0_s = get_predicted_original_sample(noise_pred, start_timesteps, noisy_model_input)
                # model_pred = c_skip_start * noisy_model_input + c_out_start * pred_x_0

                # Optimization
                if args.loss_type == "huber":
                    args.huber_c = 0.001
                    loss = torch.sqrt((noise_pred.float() - noise.float()) ** 2 + args.huber_c ** 2).mean() - args.huber_c
                    # note: cannot directly combine with mse_loss of noise
                    # loss_1 = F.mse_loss(noise_pred.float(), noise.float(), reduction="mean")
                else:
                    loss = F.mse_loss(noise_pred.float(), noise.float(), reduction="mean")
                # if args.pretrained_decoder == "stabilityai/stable-diffusion-2-1":
                #     loss = loss * 0.2
                accelerator.backward(loss)
                if accelerator.sync_gradients:
                    params_to_clip = (itertools.chain(vis.parameters(), adapter.parameters()))
                    accelerator.clip_grad_norm_(params_to_clip, 0.3)
                optimizer.step()
                lr_scheduler.step()
                global_step += 1
                optimizer.zero_grad()

            # Saving
            if accelerator.sync_gradients:
                progress_bar.update(1)
                global_step_ = global_step // accelerator.gradient_accumulation_steps

                if accelerator.is_main_process:  # accelerator.sync_gradients and
                    if global_step_ % 5 == 0:
                        accelerator.log({"train_loss": loss.detach().item(), "lr": lr_scheduler.get_last_lr()[0]},
                                        step=global_step_)

                    if global_step_ % args.validation_steps == 0:
                        adapter.eval()
                        vis.eval()
                        sample_images(vae, adapter, caption2embed_simple, vis, args, accelerator, weight_dtype,
                                      global_step_)
                        adapter.train()
                        vis.train()

                    if global_step_ % 500 == 0:
                        # training status
                        accelerator.save_state(f"{args.output_dir}/latest_status")
                        resume_json['global_step'] = global_step
                        json.dump(resume_json, open(f"{args.output_dir}/latest_status/resume.json", 'w'))

                    if global_step_ - last_save >= args.save_steps or global_step_ >= args.max_train_steps:
                        accepts_keep_fp32_wrapper = "keep_fp32_wrapper" in set(
                            inspect.signature(accelerator.unwrap_model).parameters.keys())
                        extra_args = ({"keep_fp32_wrapper": True} if accepts_keep_fp32_wrapper else {})
                        save_lora_weight(
                            accelerator.unwrap_model(vis, **extra_args),
                            f"{args.output_dir}/s{global_step_}_lora_vis.pt",
                            target_replace_module=VIS_REPLACE_MODULES,
                        )
                        accelerator.unwrap_model(adapter, **extra_args).save_pretrained(
                            f"{args.output_dir}/s{global_step_}_adapter")
                        last_save = global_step_

                        # training status
                        accelerator.save_state(f"{args.output_dir}/latest_status")
                        resume_json['global_step'] = global_step
                        json.dump(resume_json, open(f"{args.output_dir}/latest_status/resume.json", 'w'))

                    logs = {"loss": loss.detach().item(), "lr": lr_scheduler.get_last_lr()[0]}
                    progress_bar.set_postfix(**logs)
                    # accelerator.log(logs, step=global_step)

                if global_step_ >= args.max_train_steps:
                    accelerator.wait_for_everyone()
                    accelerator.end_training()

    accelerator.wait_for_everyone()
    accelerator.end_training()


if __name__ == "__main__":
    args = parse_args()
    main(args)
