# 项目创建时间：2024/9/18 19:26

import os
import argparse
from os.path import join as opj
import datetime
from importlib import import_module
from omegaconf import OmegaConf
import pytorch_lightning as pl
from pytorch_lightning.loggers import TensorBoardLogger
from pytorch_lightning.callbacks import ModelCheckpoint
from torch.utils.data import DataLoader, ConcatDataset
from pytorch_lightning import seed_everything

from util.logger import ImageLogger
from util.model_init import create_model, load_state_dict
from util.utils import save_args



#-------------------------，注意这里使用的是哪一个Ss_Diffusion
from ldm.models.diffusion.Ss_Diffusion2 import QTaskDiffusion



def build_args():
    parser = argparse.ArgumentParser()
    parser.add_argument("--config_name", type=str, default=None)
    parser.add_argument("--data_root_dir", type=str, default="./DATA/VITON-HD")
    parser.add_argument("--category", type=str, default=None, choices=["upper", "lower_body", "dresses"])
    #parser.add_argument("--vae_load_path", type=str, default="./ckpts/VITONHD_VAE_finetuning.ckpt")
    parser.add_argument("--batch_size", "-bs", type=int, default=84) # 32
    #parser.add_argument("--transform_size", default=None, nargs="+",
    #                    choices=["crop", "hflip", "shiftscale", "shiftscale2", "shiftscale3", "resize"])
    #parser.add_argument("--transform_color", default=None, nargs="+",
    #                    choices=["hsv", "bright_contrast", "colorjitter", "resize"])
    #parser.add_argument("--use_atv_loss", action="store_true")
    parser.add_argument("--valid_epoch_freq", type=int, default=5) # 每隔多少个轮次进行一次验证,默认20
    parser.add_argument("--save_every_n_epochs", type=int, default=20)
    parser.add_argument("--max_epochs", type=int, default=100) # 原1000
    parser.add_argument("--save_root_dir", type=str, default="./logs")
    parser.add_argument("--save_name", type=str, default="dresscode_5_tryon_from219_119")
    parser.add_argument("--only_tryoff_training", action="store_true")
    parser.add_argument("--vitonhd_training", action="store_true")

    parser.add_argument("--use_validation", action="store_false")
    parser.add_argument("--resume_path", type=str, default=None) #----------
    parser.add_argument("--accum_iter", type=int, default=1)  #累积多少批次更新梯度
    parser.add_argument("--img_H", type=int, default=512)
    parser.add_argument("--img_W", type=int, default=512) # 384
    parser.add_argument("--logger_freq", type=int, default=500) # 1000 记录日志信息频率，其实就是log_image隔多少次sample一下
    parser.add_argument("--learning_rate", type=float, default=1e-5) # 0-50采用的原1e-4
    #parser.add_argument("--sd_unlocked", action="store_true")  # ？？？
    #parser.add_argument("--all_unlocked", action="store_true") # ？？？？
    #parser.add_argument("--only_mid_control", action="store_true") # ？？？
    parser.add_argument("--precision", type=int, default=16)  
    parser.add_argument("--num_sanity_val_steps", type=int, default=1)  # ？？
    parser.add_argument("-s","--seed",type=int,default=23,help="seed for seed_everything")
    #parser.add_argument("--pbe_train_mode", action="store_true")

    #parser.add_argument("--lambda_simple", type=float, default=1.0) ？？？
    #parser.add_argument("--control_scales", nargs="+", type=float, default=None)  ？？？
    #parser.add_argument("--imageclip_trainable", action="store_false")
    parser.add_argument("--no_strict_load", action="store_true") # 模型是否严格加载checkpoint权重

    args = parser.parse_args()
    # todo- 这里改了
    args.config_path = opj("configs", "configs5.yaml")  # 将vitonhd 和 dresscode两个数据集构成不同的config文件，为什么
    args.n_gpus = len(os.environ["CUDA_VISIBLE_DEVICES"].split(",")) #todo-用gpu时打开
    args.devices = [i for i in range(args.n_gpus)] # todo-用gpu时打开
    print(f"device : {args.devices}, args.n_gpus:{args.n_gpus}")
    args.strategy = "auto"
    #args.sd_locked = not args.sd_unlocked ？？？？
    args.no_validation = not args.use_validation

    args.valid_real_dir = opj(args.data_root_dir, "test", "image") #---------？？？

    args.save_dir = opj(args.save_root_dir, f"{datetime.datetime.now().strftime('%Y%m%d')}_" + args.save_name)
    args.img_save_dir = opj(args.save_dir, "images")
    args.model_save_dir = opj(args.save_dir, "models")
    args.tb_save_dir = opj(args.save_dir, "tb")
    args.valid_img_save_dir = opj(args.save_dir, "validation_sampled_images")
    args.args_save_path = opj(args.save_dir, "args.json")
    args.config_save_path = opj(args.save_dir, "config.yaml")
    os.makedirs(args.img_save_dir, exist_ok=True)
    os.makedirs(args.model_save_dir, exist_ok=True)
    os.makedirs(args.tb_save_dir, exist_ok=True)
    os.makedirs(args.valid_img_save_dir, exist_ok=True)

    return args



def build_config(args, config_path=None):
    if config_path is None:
        config_path = args.config_path
    config = OmegaConf.load(config_path)
    #config.model.params.setdefault("use_VAEdownsample", False)
    #config.model.params.setdefault("use_imageCLIP", False)
    #config.model.params.setdefault("use_lastzc", False)
    #config.model.params.setdefault("use_pbe_weight", False)
    if args is not None:
        for k, v in vars(args).items():
            config.model.params.setdefault(k, v)
    """
    #--------------
    if not config.model.params.get("validation_config", None):
        config.model.params.validation_config = OmegaConf.create()
    config.model.params.validation_config.ddim_steps = config.model.params.validation_config.get("ddim_steps", 50)
    config.model.params.validation_config.eta = config.model.params.validation_config.get("eta", 0.0)
    config.model.params.validation_config.scale = config.model.params.validation_config.get("scale", 1.0)
    #--------------
    """
    if args is not None:
        #config.model.params.unet_config.params.use_atv_loss = args.use_atv_loss
        config.model.params.validation_config.img_save_dir = args.valid_img_save_dir
        config.model.params.validation_config.real_dir = args.valid_real_dir

        #if args.use_atv_loss:
        #    config.model.params.use_attn_mask = True
    return config


def main_worker(args):
    config = build_config(args)
    OmegaConf.save(config, args.config_save_path)
    seed_everything(args.seed)

    model = create_model(args.config_path, config=config).cpu()  # 模型还是从args.config_path进行加载

    if args.resume_path is not None:
        if not args.no_strict_load:
            missing,unexpected = model.load_state_dict(load_state_dict(args.resume_path, location="cpu"))
        else:
            missing,unexpected =model.load_state_dict(load_state_dict(args.resume_path, location="cpu"), strict=False)
    elif config.resume_path is not None:
        print(f"resume_path:{config.resume_path}")
        if not args.no_strict_load:
            missing,unexpected = model.load_state_dict(load_state_dict(config.resume_path, location="cpu"))
        else:
            missing,unexpected = model.load_state_dict(load_state_dict(config.resume_path, location="cpu"), strict=False)
    
    print(f"Restored from {args.config_path} with {len(missing)} missing and {len(unexpected)} unexpected keys")


    # finetuned vae load
    #if args.vae_load_path is not None:
    #    state_dict = load_state_dict(args.vae_load_path, location="cpu")
    #    new_state_dict = {}
    #    for k, v in state_dict.items():
    #        if "loss." not in k:
    #            new_state_dict[k] = v.clone()
    #    model.first_stage_model.load_state_dict(new_state_dict)

    model.learning_rate = args.learning_rate
    ## print(f"learning rate:{model.learning_rate}")
    #model.sd_locked = args.sd_locked
    #model.only_mid_control = args.only_mid_control
    #-----------------
    
    #model.cuda()
    """
    all_params_on_cuda = True
    for name, param in model.named_parameters():
        if param.device.type != 'cuda':
            all_params_on_cuda = False
            print(f"Parameter '{name}' is not on CUDA: {param.device}")

    if all_params_on_cuda:
        print("All parameters are on CUDA.")
    else:
        print("Some parameters are not on CUDA.")
    """
    #print(f"模型及参数初始化完成, model.device={model.device}")
    if args.only_tryoff_training:
        from datasets.cp_cloth_mask_free_inference import CPDataset
        from datasets.dc_dataset5_cloth import DCDataset
    else:
        from datasets.cp_dataset5 import CPDataset
        from datasets.dc_dataset5 import DCDataset

    if args.vitonhd_training:
        train_dataset = CPDataset(
            dataroot=args.data_root_dir,
            image_size = 512,
            mode='train',
            unpaired=False, # False
            #caption_folder='captions.json'
            
        )
        valid_paired_dataset = CPDataset(
            dataroot=args.data_root_dir,
            image_size = 512,
            mode='test',
            unpaired=False, # False
            #caption_folder='captions.json'
        )
        valid_unpaired_dataset =CPDataset(
            dataroot=args.data_root_dir,
            image_size = 512,
            mode='test' ,
            unpaired=True,
            #caption_folder='captions.json'
        )
    else:
        train_dataset = DCDataset(
            dataroot_path=args.data_root_dir,
            phase='train',
            order= 'paired',
            category= args.category,
            size=512
        )

        
        valid_paired_dataset = DCDataset(
            dataroot_path=args.data_root_dir,
            phase='test',
            order= 'paired',
            category= args.category,
            size=512
        )
        valid_unpaired_dataset = DCDataset(
            dataroot_path=args.data_root_dir,
            phase='test',
            order= 'unpaired',
            category=args.category,
            size=512
        )
        
    

    train_dataloader = DataLoader(
        train_dataset,
        num_workers=4, 
        batch_size= max(args.batch_size // args.n_gpus, 1),#max(args.batch_size // args.n_gpus, 1),#max(args.batch_size // args.n_gpus, 1),
        shuffle=True, # 原true
        pin_memory=True # 原true
    )
    valid_paired_dataloader = DataLoader(
        valid_paired_dataset,
        num_workers=4,
        batch_size= max(args.batch_size // args.n_gpus, 1),#max(args.batch_size // args.n_gpus, 1),#max(args.batch_size // args.n_gpus, 1),
        shuffle=False,
        pin_memory=False
    )
    valid_unpaired_dataloader = DataLoader(
        valid_unpaired_dataset,
        num_workers=4, # 设置为4
        batch_size= max(args.batch_size // args.n_gpus, 1),#max(args.batch_size // args.n_gpus, 1),#max(args.batch_size // args.n_gpus, 1),
        shuffle=False,
        pin_memory=False
    )
    print("数据加载完成")
    #### trainer >>>>
    img_logger = ImageLogger(
        batch_frequency=args.logger_freq,
        save_dir=args.img_save_dir,
        #log_images_kwargs=config.get("log_images_kwargs", None)
    )
    tb_logger = TensorBoardLogger(args.tb_save_dir) # 创建一个 TensorBoard 日志记录器
    cp_callback = ModelCheckpoint( # ModelCheckpoint 是 PyTorch Lightning 中的一个回调类，用于在训练过程中自动保存模型的权重
        dirpath=args.model_save_dir,
        filename="[Train]_[{epoch}]_[{train_loss_epoch:.05f}]",
        save_top_k=-1,
        every_n_epochs=args.save_every_n_epochs,
        save_last=False,
        save_on_train_epoch_end=True
    )

    trainer = pl.Trainer(
        strategy='ddp_find_unused_parameters_true',
        precision=args.precision,
        #precision="16",
        callbacks=[img_logger, cp_callback],
        logger=tb_logger,
        devices= args.devices, #args.devices, # 这里是gpu的设备号
        accelerator='gpu',#"gpu", # todo-原来是gpu
        #strategy="ddp", #"ddp", # 分布式数据并行（Distributed Data Parallel）策略进行训练
        max_epochs=args.max_epochs,
        accumulate_grad_batches=args.accum_iter, # 累积梯度
        check_val_every_n_epoch=args.valid_epoch_freq, # 每隔多少个轮次进行一次验证。
        num_sanity_val_steps=args.num_sanity_val_steps # 在训练开始前进行多少个验证步骤
    )
    #### trainer <<<<

    if not args.no_validation:
        print("Use validation!")
        trainer.fit(model, train_dataloader, [valid_paired_dataloader, valid_unpaired_dataloader])
    else:
        print("Do not use validation!")
        trainer.fit(model, train_dataloader)


if __name__ == "__main__":
    args = build_args()
    print(args)
    save_args(args, args.args_save_path)  # 保存命令行参数成yaml格式
    main_worker(args)
    print("Done")