"""
cogactvla.py

"""

from __future__ import annotations

from functools import partial
from pathlib import Path
from typing import Callable, Dict, List, Optional, Type, Union, Tuple, Any, Sequence
from copy import deepcopy
from dataclasses import dataclass

import torch
import torch.nn as nn
import numpy as np
from PIL import Image
from torch.distributed.fsdp.wrap import _module_wrap_policy, _or_policy, transformer_auto_wrap_policy
from torch.nn.utils.rnn import pad_sequence
from transformers.modeling_outputs import CausalLMOutputWithPast

from prismatic.models.vlms.prismatic import PrismaticVLM
from prismatic.overwatch import initialize_overwatch
from prismatic.util.nn_utils import FusedMLPProjector, LinearProjector, MLPProjector
from transformers import AutoTokenizer, AutoProcessor, AutoModel
from transformers.models.qwen2.modeling_qwen2 import Qwen2DecoderLayer
from transformers.models.llama.modeling_llama import LlamaDecoderLayer
from transformers.models.siglip.modeling_siglip import SiglipEncoderLayer
import transformers
from packaging import version

import time
import timm
import random

from peft import get_peft_model, LoraConfig, TaskType
import torch.nn.functional as F


import math
from transformers import (
    MODEL_MAPPING,
    AutoConfig,
    AutoModelForCausalLM,
    SchedulerType,
    get_scheduler,
)

from torch.distributed.algorithms._checkpoint.checkpoint_wrapper import (
    CheckpointImpl,
    apply_activation_checkpointing,
    checkpoint_wrapper,
)

from vla.eagle_utils import EagleProcessor, extract_decoder_hidden_states, cleanup_xlora_pre_hooks, model_forward
from types import SimpleNamespace
from transformers import StoppingCriteria, StoppingCriteriaList

# Initialize Overwatch =>> Wraps `logging.Logger`
overwatch = initialize_overwatch(__name__)

# HuggingFace Default / LLaMa-2 IGNORE_INDEX (for labels)
IGNORE_INDEX = -100

# This system message is JUST a placeholder
DEFAULT_SYSTEM_MESSAGE = "You are a helpful assistant."

from .action_head import ActionModel

class InstructVLA(nn.Module):
    def __init__(
        self,
        vlm: AutoModel,
        processor: AutoProcessor = None,
        tokenizer: AutoTokenizer = None,
        token_size: int = 4096,
        action_dim: int = 7,
        future_action_window_size: int = 15,
        past_action_window_size: int = 1,
        norm_stats: Dict[str, Dict[str, Dict[str, Dict[str, List[float]]]]] = None,
        config_json = None,
        meta_token_ids = None,
        stage = "stage1",
        **kwargs,
    ) -> None:
        super().__init__()
        self.action_model = ActionModel(token_size,past_action_window_size,future_action_window_size,action_dim)
        
        self.vlm = vlm
        self.processor = processor
        self.tokenizer = tokenizer
        self.config_json = config_json
        self.token_size = token_size
        self.future_action_window_size = future_action_window_size
        self.past_action_window_size = past_action_window_size

        self.all_module_keys = ['action_model']
        for module_keys in ["vision_model", "language_model", "mlp1"]:
            self.all_module_keys.append("vlm." + module_keys)

        # Diffusion head is always trainable
        self._trainable_module_keys = ['action_model']

        keys = []
        for module_keys in ["vision_model", "language_model", "mlp1"]:
            keys.append("vlm." + module_keys)
        keys += self._trainable_module_keys
        self.trainable_module_keys = keys

        self.norm_stats = norm_stats
        self.vlm.language_model.forward = model_forward(self.vlm.language_model)
        self.vlm.language_model.transformer_layer_cls = Qwen2DecoderLayer
        

        self.vlm.neftune_alpha = None
        self.action_dim = action_dim
        self.meta_token_ids = meta_token_ids
        self.min_meta_token = self.meta_token_ids[0]
        self.max_meta_token = self.meta_token_ids[-1]

        # Freeze all parameters in the model
        if stage == "stage1":
            overwatch.info("Train the model in stage 1 with lora and learnable embeddings")
            for param in self.vlm.parameters():
                param.requires_grad = False

            lora_config = LoraConfig(
                    r=128,
                    lora_alpha=256,
                    target_modules=['self_attn.q_proj', 'self_attn.k_proj', 'self_attn.v_proj', 
                                    'self_attn.o_proj', 'mlp.down_proj', 'mlp.up_proj'],  # adjust based on model architecture
                    lora_dropout=0.05,
                    bias="none",
                    task_type=TaskType.CAUSAL_LM  # assuming a causal language model
                )
            
            # Unfreeze only the new tokens' embeddings
            
            self.vlm.language_model = get_peft_model(self.vlm.language_model, lora_config)

            new_token_ids = self.tokenizer.convert_tokens_to_ids(self.tokenizer.new_tokens)  # Convert new tokens to their respective ids
            new_token_ids = torch.tensor(new_token_ids, device=self.vlm.language_model.base_model.model.lm_head.weight.device)

            # we unfreeze the embed and lm_head, but not all parameters are finetuned, the trainable weight are less than the reported weight
            self.vlm.language_model.base_model.model.lm_head.weight.requires_grad = True
            self.vlm.language_model.base_model.model.model.embed_tokens.weight.requires_grad = True

            def mask_old_token_grad(grad: torch.Tensor):
                mask = torch.ones(grad.shape[0], dtype=torch.bool, device=grad.device)
                mask[new_token_ids] = False 
                grad[mask] = 0
                return grad

            self.vlm.language_model.base_model.model.lm_head.weight.register_hook(mask_old_token_grad)
            self.vlm.language_model.base_model.model.model.embed_tokens.weight.register_hook(mask_old_token_grad)


        elif stage == "stage2":
            overwatch.info("Train the model in stage 2 with X-LoRA")

            # We defaultly use the dense method, the inital scale of each adapter is 1/num_adapter
            # The language expert is initaled from zero
            from peft import XLoraConfig

            empty_language_adapter = os.path.abspath(
                os.path.join(os.path.dirname(__file__), "..", "ckpt", "empty_language_adapter")
            )

            lora_config = XLoraConfig(
                task_type="CAUSAL_LM",
                hidden_size=token_size,
                xlora_depth=4,
                xlora_size=128,
                adapters={
                    "0": empty_language_adapter,
                    "1": empty_language_adapter, # Must be replaced if you want to train the stage-2 model !
                },
            )

            overwatch.warning(
                f"If you are initializing stage-2 training from a stage-1 model, "
                f"you **must** replace **one of** the above empty_language_adapter with "
                f"the unloaded LoRA module from your stage-1 model. "
                f"Because the unloaded stage-1.pt does not contain the pretrained action "
                f"LoRA expert weights."
            )

            self.vlm.language_model = get_peft_model(self.vlm.language_model, lora_config)

            new_token_ids = self.tokenizer.convert_tokens_to_ids(self.tokenizer.new_tokens)

            new_token_ids = torch.tensor(new_token_ids, device=self.vlm.language_model.base_model.model.lm_head.weight.device)
            # we unfreeze the embed and lm_head, but not all parameters are finetuned, the trainable weight are less than the reported weight
            self.vlm.language_model.base_model.model.lm_head.weight.requires_grad = True
            self.vlm.language_model.base_model.model.model.embed_tokens.weight.requires_grad = True

            def mask_old_token_grad(grad: torch.Tensor):
                mask = torch.ones(grad.shape[0], dtype=torch.bool, device=grad.device)
                mask[new_token_ids] = False 
                grad[mask] = 0
                return grad

            self.vlm.language_model.base_model.model.lm_head.weight.register_hook(mask_old_token_grad)
            self.vlm.language_model.base_model.model.model.embed_tokens.weight.register_hook(mask_old_token_grad)

            for name, param in self.vlm.language_model.base_model.lora_model.named_parameters():
                if "lora_" in name:
                    param.requires_grad = True
        self.stage = stage

        self.vlm.language_model.print_trainable_parameters()

        self.vlm.language_model.transformer_layer_cls = Qwen2DecoderLayer

        for name, param in self.named_parameters():
            param.data = param.data.to(torch.float32)
            if param.grad is not None:
                param.grad.data = param.grad.data.to(torch.float32)


        class StoppingCriteriaSub(StoppingCriteria):
            def __init__(self,tokenizer, stops = [], encounters=1):
                super().__init__()
                self.stops = [stop.to("cuda") for stop in stops]
                self.tokenizer = tokenizer

            def __call__(self, input_ids: torch.LongTensor, scores: torch.FloatTensor):
                last_token = input_ids[0][-1]
                for stop in self.stops:
                    if self.tokenizer.decode(stop) == self.tokenizer.decode(last_token):
                        return True
                return False

        stop_words = ["<new_token_0>"]
        stop_words_ids = [self.tokenizer(stop_word, return_tensors='pt', add_special_tokens=False)['input_ids'].squeeze() for stop_word in stop_words]
        self.stopping_criteria = StoppingCriteriaList([StoppingCriteriaSub(stops=stop_words_ids, tokenizer = self.tokenizer)])

        self._fix_system1 = False

        # for inference
        self.last_response = None
        self.run_index = 0
        self.latent = None
    
    @property
    def fix_system1(self):
        return self._fix_system1

    @fix_system1.setter
    def fix_system1(self, value: bool):
        overwatch.warning(f"fix_system1 is being updated to: {value}")
        if value:
            for name, param in self.action_model.named_parameters():
                param.requires_grad = False
        else:
            for name, param in self.action_model.named_parameters():
                param.requires_grad = True

        self._fix_system1 = value

    @property
    def llm_backbone(self):
        return self.vlm.language_model
    
    @property
    def vision_backbone(self):
        return self.vlm.vision_model
    
    def freeze_backbones(self, stage):
        self.vlm.freeze_backbones(stage)
    
    @torch.inference_mode()
    def chat(self, *args, **kwargs):
        # chat method from eagle vlm
        autocast_dtype = torch.bfloat16
        with torch.autocast("cuda", dtype=autocast_dtype, enabled=True):
            ret = self.vlm.chat(*args, **kwargs)
        return ret

    def forward(
        self,
        input_ids: Optional[torch.LongTensor] = None,
        attention_mask: Optional[torch.Tensor] = None,
        pixel_values: Optional[torch.FloatTensor] = None,
        system1_pixel_values: Optional[torch.FloatTensor] = None,
        labels: Optional[torch.LongTensor] = None,
        actions: Optional[torch.FloatTensor] = None,
        inputs_embeds: Optional[torch.FloatTensor] = None,
        past_key_values: Optional[List[torch.FloatTensor]] = None,
        use_cache: Optional[bool] = None,
        output_attentions: Optional[bool] = None,
        output_hidden_states: Optional[bool] = None,
        return_dict: Optional[bool] = None,
        per_device_batch_size: int = 16,
        action_masks = None,
        image_flags = None,
        sampling_type = None,
        t = None,
        train_idx = 0,
        **kwargs,
    ) -> Tuple:
        if actions is None:
            # standard vlm forward
            # from IPython import embed;embed()
            per_device_batch_size = input_ids.shape[0]
            output: CausalLMOutputWithPast = self.vlm(
                input_ids=input_ids,
                attention_mask=attention_mask,
                pixel_values=pixel_values,
                labels=labels,
                image_flags=torch.ones((input_ids.shape[0],1)).to( device=input_ids.device) if image_flags is None else image_flags,
                past_key_values=past_key_values,
                use_cache=use_cache,
                output_attentions=output_attentions,
                output_hidden_states = output_hidden_states,
                return_dict=return_dict,
                fast_loss_cal=True,
                **kwargs,
            )
            return output.loss, output
        else:

            assert per_device_batch_size == actions.shape[0]

            output: CausalLMOutputWithPast = self.vlm(
                input_ids=input_ids,
                attention_mask=attention_mask,
                pixel_values=pixel_values,
                labels=labels,
                # inputs_embeds=inputs_embeds,
                image_flags=torch.ones((input_ids.shape[0],1)).to( device=input_ids.device) if image_flags is None else image_flags,
                past_key_values=past_key_values,
                use_cache=use_cache,
                output_attentions=output_attentions,
                output_hidden_states=output_hidden_states,
                return_dict=return_dict,
                fast_loss_cal=True
            )

            # extract the last hidden state
            last_hidden_states = output.hidden_states[-1]
            
            # extract the latent action
            meta_feature_mask = (input_ids >= self.min_meta_token) & (input_ids <= self.max_meta_token)
            meta_feature = last_hidden_states[torch.where(meta_feature_mask==1)].view(meta_feature_mask.size(0),-1 , last_hidden_states.shape[-1])

            # actions_history = actions[:,0:self.past_action_window_size,:]
            actions_future = actions[:, -(self.future_action_window_size+1):, :]
            _, _, action_dim = actions_future.shape
            
            loss = self.action_model( latent_action = meta_feature,
                        pixel_values = dict(dino = system1_pixel_values),
                        actions = actions_future,
                        t = t,
                    )

            return loss + output.loss, output.loss

    def get_fsdp_wrapping_policy(self) -> Callable:
        """Return an FSDP _or_policy over the policies returned by each individual backbone (and our VLM policy)."""

        # Get Prismatic Wrapping Policy =>> just a module wrapping policy around `self.projector` and DiT

        vit_wrap_policy = partial(transformer_auto_wrap_policy, transformer_layer_cls={SiglipEncoderLayer})
        transformer_block_policy_1 = partial(transformer_auto_wrap_policy, transformer_layer_cls={Qwen2DecoderLayer})
        transformer_block_policy_2 = partial(transformer_auto_wrap_policy, transformer_layer_cls={LlamaDecoderLayer})

        from transformers.models.llama.modeling_llama import LlamaMLP
        from transformers.models.qwen2.modeling_qwen2 import Qwen2MLP

        prismatic_fsdp_wrapping_policy = partial(
            _module_wrap_policy,
            module_classes={LinearProjector, MLPProjector, FusedMLPProjector, Qwen2MLP, LlamaMLP},
        )

        # Return union (_or_) over constituent policies
        #   => Note: there is *not* a fall-through policy; any module that isn't covered by the above constituents will
        #            automatically be folded into the root VLM FSDP instance.
        return partial(
            _or_policy,
            policies=[
                vit_wrap_policy,
                transformer_block_policy_1,
                transformer_block_policy_2,
                prismatic_fsdp_wrapping_policy,
            ],
        )

    @classmethod
    def from_pretrained(
        cls,
        pretrained_checkpoint: Path,
        freeze_weights: bool = True,
        action_dim: int = 7,
        future_action_window_size: int = 15,
        past_action_window_size: int = 5,
        norm_stats = None,
        stage = "stage1",
        num_of_meta_query = 64,
        **kwargs,
    ) -> InstructVLA:

        llm_backbone_id = os.path.abspath(
                os.path.join(os.path.dirname(__file__), "..", "ckpt", "Eagle2-2B")
            )

        # Load VLM backbone, borrowed from PrismaticVLM
        vlm = AutoModel.from_pretrained(llm_backbone_id,
                                        attn_implementation="flash_attention_2",
                                        trust_remote_code=True)

        processor = EagleProcessor(
            llm_backbone_id,
            max_input_tiles=1,
            model_spec=SimpleNamespace(
                num_image_token = 256,
                template = "qwen2-chat"
            ),
        )

        tokenizer = AutoTokenizer.from_pretrained(llm_backbone_id, 
                                                  use_fast=True,
                                                  trust_remote_code=True)

        new_tokens = ['<new_token_{}>'.format(i) for i in range(num_of_meta_query)]  # Create 256 new token names
        overwatch.info(f"add {len(new_tokens)} latent action tokens")
        tokenizer.add_tokens(new_tokens)  # Add them to the tokenizer
        tokenizer.new_tokens = new_tokens
        processor.tokenizer = tokenizer

        # Resize the model's token embeddings to match the new vocabulary size
        try:  # Skip mean resize feature in new version of transformers
            vlm.language_model.resize_token_embeddings(len(tokenizer), mean_resizing = False)
        except Exception as e:
            vlm.language_model.resize_token_embeddings(len(tokenizer))

        # Freeze all parameters in the model
        for param in vlm.parameters():
            param.requires_grad = False

        # Unfreeze only the new tokens' embeddings
        new_token_ids = tokenizer.convert_tokens_to_ids(new_tokens)  # Convert new tokens to their respective ids

        vlm.img_context_token_id = processor.get_img_context_token()
        assert vlm.template == processor.model_spec.template
        assert vlm.num_image_token == processor.model_spec.num_image_token

        model_state_dict = torch.load(pretrained_checkpoint, map_location="cpu")["model"]
        is_a_lora_model = any('lora' in i for i in  model_state_dict['language_model'])
        if not is_a_lora_model:
            overwatch.warning("No LoRA parameters in the checkpoint, so we load weight before init LoRA")
            vlm.language_model.load_state_dict(model_state_dict["language_model"])

        # Initialize InstructVLA
        instruct_vla = InstructVLA(vlm,
                        processor = processor,
                        tokenizer = tokenizer,
                        token_size = vlm.config.llm_config.hidden_size,
                        action_dim = action_dim,
                        future_action_window_size = future_action_window_size,
                        past_action_window_size = past_action_window_size,
                        norm_stats = norm_stats,
                        meta_token_ids = new_token_ids,
                        stage = stage,
                        )


        assert (
            "mlp1" in model_state_dict and "language_model" in model_state_dict and "vision_model" in model_state_dict
        ), "PrismaticVLM `from_pretrained` expects checkpoint with keys for `projector`, `language_model` AND `vision_model`"

        instruct_vla.vlm.mlp1.load_state_dict(model_state_dict["mlp1"])
        instruct_vla.vlm.vision_model.load_state_dict(model_state_dict["vision_model"],strict=False) # The ckpt contains extra head parameters that are not use the previous transformers, but recently, the head is completely removed

        if is_a_lora_model:
            overwatch.warning("LoRA parameters in the checkpoint, so we load weight after init LoRA")
            instruct_vla.vlm.language_model.load_state_dict(model_state_dict["language_model"])
        

        # Load ActionModel from Checkpoint
        if "action_model" in model_state_dict:
            instruct_vla.action_model.load_state_dict(model_state_dict["action_model"], strict=False)
        else:
            overwatch.warning("No Action Model found in the pretrained checkpoint. Initializing a new one.")
        return instruct_vla       

    @torch.inference_mode()
    def predict_action(
        self, 
        image: Image, 
        instruction: str, 
        unnorm_key: Optional[str] = None, 
        use_generate = False,
        cache_latent = False,
        **kwargs: str
    ) -> np.ndarray:
        """
        Core function for VLA inference; maps input image and task instruction to continuous action.

        @param image: PIL Image as [height, width, 3]
        @param instruction: Task instruction string
        @param unnorm_key: Optional dataset name for retrieving un-normalizing statistics; if None, checks that model
                           was trained only on a single dataset, and retrieves those statistics.

        @return Unnormalized (continuous) action vector --> end-effector deltas.
        """
        # Build VLA Prompt
    
        autocast_dtype = torch.bfloat16
        pixel_values = None

        # Prepare Inputs
        if use_generate and self.stage=="stage2":
            if self.last_response is None or self.run_index % 20 == 0:
                prompt = [
                    {"role": "system", "content": DEFAULT_SYSTEM_MESSAGE},
                    {
                        "role": "user",
                        "content": f"What action should the robot take to {instruction}? First answer my question.",
                        "image": [{'np_array': np.asarray(image)}],
                    },
                    {
                        "role": "assistant", 
                        "content": ""
                    }
                ]
                inputs = self.processor.prepare_input({"prompt": prompt})
                input_ids = inputs['input_ids'].to(self.vlm.device)
                pixel_values = inputs['pixel_values']

                pixel_values = pixel_values.to(self.vlm.device, dtype=autocast_dtype)
                
                with torch.autocast("cuda", dtype=autocast_dtype, enabled=True):
                    attention_mask = input_ids.ne(-10)
                    output: CausalLMOutputWithPast = self.vlm.generate(
                        input_ids=input_ids,
                        attention_mask=attention_mask,
                        pixel_values=pixel_values,
                        max_new_tokens=50,
                        # fast_loss_cal=False,
                        output_hidden_states=False,
                        return_dict_in_generate=False,
                        stopping_criteria=self.stopping_criteria # to accelerate primitive
                    )

                # Extract cognition feature
                response = self.tokenizer.decode(output[0]).replace("<new_token_0>","")
                print("====== Reasoning ======= >" + response)
                self.last_response = response
            else:
                response = self.last_response

            prompt = [
                {"role": "system", "content": DEFAULT_SYSTEM_MESSAGE},
                {
                    "role": "user",
                    "content": f"What action should the robot take to {instruction}? First answer my question.",
                    "image": [{'np_array': np.asarray(image)}],
                },
                {
                    "role": "assistant", 
                    "content": response + " " + "".join(self.processor.tokenizer.new_tokens)
                }
            ]
        else:
            prompt = [
                {"role": "system", "content": DEFAULT_SYSTEM_MESSAGE},
                {
                    "role": "user",
                    "content": f"What action should the robot take to {instruction}?",
                    "image": [{'np_array': np.asarray(image)}],
                },
                {
                    "role": "assistant", 
                    "content": "".join(self.processor.tokenizer.new_tokens)
                }
            ]
        inputs = self.processor.preprocess_inputs_and_labels({"prompt": prompt})
        input_ids = inputs['input_ids'].to(self.vlm.device).unsqueeze(0)
        if pixel_values is None:
            # Preprocess Image
            pixel_values = inputs['pixel_values']
            pixel_values = pixel_values.to(self.vlm.device, dtype=autocast_dtype)

        # Generate cognition feature through vlm
        if cache_latent is False or (cache_latent is True and (self.run_index % 2 == 0 or self.latent is None)):
            with torch.autocast("cuda", dtype=autocast_dtype, enabled=True):
                attention_mask = input_ids.ne(-10)
                output: CausalLMOutputWithPast = self.vlm(
                    input_ids=input_ids,
                    attention_mask=attention_mask,
                    pixel_values=pixel_values,
                    image_flags=torch.ones((input_ids.shape[0],1)).to( device=input_ids.device),
                    past_key_values=None,
                    use_cache=None,
                    output_attentions=None,
                    output_hidden_states=True,
                )

            # Extract cognition feature
            last_hidden_states = output.hidden_states[-1]
            self.latent = last_hidden_states.detach()
        else:
            last_hidden_states = self.latent
            
        meta_feature_mask = (input_ids >= self.min_meta_token) & (input_ids <= self.max_meta_token)
        meta_feature = last_hidden_states[torch.where(meta_feature_mask==1)].view(meta_feature_mask.size(0),-1 , last_hidden_states.shape[-1])

        BS, step, dim = meta_feature.shape

        sys1_pixel_values = dict(dino = self.action_model.default_dino_transform(image).unsqueeze(0).to(self.vlm.device))
        with torch.autocast("cuda", dtype=autocast_dtype, enabled=True):
            samples = self.action_model.sampling(   latent_action = meta_feature,
                                                    pixel_values = sys1_pixel_values,
                                                    )
        normalized_actions = samples[0].float().cpu().numpy()

        # Un-normalize Actions        
        action_norm_stats = self.get_action_stats(unnorm_key)
        mask = action_norm_stats.get("mask", np.ones_like(action_norm_stats["q01"], dtype=bool))
        action_high, action_low = np.array(action_norm_stats["q99"]), np.array(action_norm_stats["q01"])
        normalized_actions = np.clip(normalized_actions, -1, 1)
        normalized_actions[:, 6] = np.where(normalized_actions[:, 6] < 0.5, 0, 1) 
        actions = np.where(
            mask,
            0.5 * (normalized_actions + 1) * (action_high - action_low) + action_low,
            normalized_actions,
        )
        self.run_index += 1
        cleanup_xlora_pre_hooks(self.vlm.language_model)
        return actions, normalized_actions, meta_feature

    @staticmethod
    def _check_unnorm_key(norm_stats, unnorm_key):
        if unnorm_key is None:
            assert len(norm_stats) == 1, (
                f"Your model was trained on more than one dataset, "
                f"please pass a `unnorm_key` from the following options to choose the statistics "
                f"used for un-normalizing actions: {norm_stats.keys()}"
            )
            unnorm_key = next(iter(norm_stats.keys()))

        assert unnorm_key in norm_stats, (
            f"The `unnorm_key` you chose is not in the set of available dataset statistics, "
            f"please choose from: {norm_stats.keys()}"
        )
        return unnorm_key

    def get_action_dim(self, unnorm_key=None):
        """Dimensionality of the policy's action space."""
        unnorm_key = self._check_unnorm_key(self.norm_stats, unnorm_key)
        return len(self.norm_stats[unnorm_key]["action"]["q01"])

    def get_action_stats(self, unnorm_key=None):
        """Dimensionality of the policy's action space."""
        unnorm_key = self._check_unnorm_key(self.norm_stats, unnorm_key)
        return self.norm_stats[unnorm_key]["action"]



from pathlib import Path
from typing import Tuple, Type

from torch.utils.data import Dataset
from transformers import PreTrainedTokenizerBase

from prismatic.vla.action_tokenizer import ActionTokenizer
from prismatic.vla.datasets import EpisodicRLDSDataset, RLDSDataset
from transformers import AutoTokenizer, AutoProcessor
from huggingface_hub import HfFileSystem, hf_hub_download
import os
import json

@dataclass
class RLDSBatchTransform:
    action_tokenizer: ActionTokenizer
    base_tokenizer: PreTrainedTokenizerBase
    processor: AutoProcessor
    image_processor: AutoProcessor
    stage: str = "stage1"
    disable_instruction: bool = False

    caption_prompts = [
        "Describe what’s on the table. Don’t mention the robot arm.",
        "What objects are in the scene? Ignore the robot arm.",
        "Tell me what you see on the table, not the robot.",
        "Describe the items and their positions, but skip the robot.",
        "Look at the table and describe it. Don’t include the arm.",
        "Only talk about the objects, not the machine.",
        "Give a short description of the scene, without the robot.",
        "Describe the setup on the table. Leave out the robotic arm.",
        "Focus on the objects and environment. Ignore the robot.",
    ]

    def __call__(self, rlds_batch: Dict[str, Any]) -> Dict[str, Any]:
        """Converts a RLDS batch to the format expected by the OpenVLA collator/models."""
        # dataset_name, action = rlds_batch["dataset_name"], rlds_batch["action"][0]
        
        # For future action predictions
        if rlds_batch["action"].shape[0] > 1:
            dataset_name, action = rlds_batch["dataset_name"], rlds_batch["action"]
        else:
            dataset_name, action = rlds_batch["dataset_name"], rlds_batch["action"][0]

        img = rlds_batch["observation"]["image_primary"]

        lang = rlds_batch["task"]["language_instruction"].decode().lower()

        pixel_values = []
        system1_pixel_values = []
        input_ids = []
        labels = []
        # print(rlds_batch)

        anno = json.loads(rlds_batch["reasonings"].decode())
        move_primitive = anno["move_primitive"]

        # begin =============== prepare inputs for VLM ===================
        last_image = [{'np_array': img[-1]}] # always use the last(current) image for VLM

        if self.stage == 'stage1' or self.disable_instruction:
            if random.random() < 0.2 and move_primitive is not None:
                action_prompt = f"What action should the robot take to {lang}? Give both move primitive and action."
                assistant_content = f"{move_primitive} " + "".join(self.processor.tokenizer.new_tokens)
            else:
                action_prompt = f"What action should the robot take to {lang}?"
                assistant_content = "".join(self.processor.tokenizer.new_tokens)

            prompt = [
                {"role": "system", "content": DEFAULT_SYSTEM_MESSAGE},
                {
                    "role": "user",
                    "content": action_prompt,
                    "image": last_image,
                },
                {
                    "role": "assistant", 
                    "content": assistant_content
                }
            ]
        elif self.stage == 'stage2':

            instruction_prob = random.random()
            primitive_prob = random.random()

            has_QA_or_cap = False
            has_CC = False
            has_CR = False

            if anno['alt_instruction'] is not None:
                has_QA_or_cap = True
                if  "CR" in anno['alt_instruction'] and len(anno['alt_instruction']["CR"]): has_CR=True
                if  "CC" in anno['alt_instruction'] and len(anno['alt_instruction']["CC"]): has_CC=True

            try:
                if 0.4 <= instruction_prob < 0.6 and has_QA_or_cap:
                    all_optional_QA = [ 
                        dict(
                            question = random.sample(self.caption_prompts, 1)[0],
                            answer = anno['alt_instruction']['Caption']
                        )
                    ] + anno['alt_instruction']['QA']
                    curr_QA = random.sample(all_optional_QA, 1)[0]

                    prompt = [
                        {"role": "system", "content": DEFAULT_SYSTEM_MESSAGE},
                        {
                            "role": "user",
                            "content": curr_QA['question'],
                            "image": last_image,
                        },
                        {
                            "role": "assistant", 
                            "content": curr_QA["answer"]
                        },
                        {
                            "role": "user",
                            "content": f"What action should the robot take to {lang}?",
                        },
                        {
                            "role": "assistant", 
                            "content": "".join(self.processor.tokenizer.new_tokens)
                        },
                    ]

                elif 0.6 <= instruction_prob < 0.8 and has_CC:
                    curr_QA = random.sample(anno['alt_instruction']["CC"], 1)[0]
                    prompt = [
                        {"role": "system", "content": DEFAULT_SYSTEM_MESSAGE},
                        {
                            "role": "user",
                            "content": f"What action should the robot take to {curr_QA['question']}? First answer my question.",
                            "image": last_image,
                        },
                        {
                            "role": "assistant", 
                            "content": curr_QA["answer"] + " " + "".join(self.processor.tokenizer.new_tokens)
                        }
                    ]

                elif 0.8 <= instruction_prob and has_CR:
                    curr_QA = random.sample(anno['alt_instruction']["CR"], 1)[0]
                    prompt = [
                        {"role": "system", "content": DEFAULT_SYSTEM_MESSAGE},
                        {
                            "role": "user",
                            "content": f"What action should the robot take to {curr_QA['question']}? First answer my question.",
                            "image": last_image,
                        },
                        {
                            "role": "assistant", 
                            "content": curr_QA["answer"] + " " + "".join(self.processor.tokenizer.new_tokens)
                        }
                    ]
                else:
                    raise ValueError()

            except Exception as e:
                if primitive_prob < 0.2:
                    action_prompt = f"What action should the robot take to {lang}? Give both move primitive and action."
                    assistant_content = f"{move_primitive} " + "".join(self.processor.tokenizer.new_tokens)
                else:
                    action_prompt = f"What action should the robot take to {lang}?"
                    assistant_content = "".join(self.processor.tokenizer.new_tokens)

                prompt = [
                    {"role": "system", "content": DEFAULT_SYSTEM_MESSAGE},
                    {
                        "role": "user",
                        "content": action_prompt,
                        "image": last_image,
                    },
                    {
                        "role": "assistant", 
                        "content": assistant_content
                    }
                ]
        else:
            raise NotImplementedError(f'which stage ???')

        inputs = self.processor.preprocess_inputs_and_labels({"prompt": prompt})

        pixel_values.append(inputs['pixel_values'])
        img_array = np.squeeze(img[-1]).astype(np.uint8)
        pil_img = Image.fromarray(img_array)

        input_ids.append(inputs['input_ids'].unsqueeze(0))
        labels.append(inputs['labels'].unsqueeze(0))

        # end =============== prepare inputs for VLM ===================

        # begin ============= prepare inputs for action expert ===============

        for i in img:
            img_array = np.squeeze(i).astype(np.uint8)
            pil_img = Image.fromarray(img_array)
            system1_pixel_values.append(self.image_processor(pil_img))

        # end =============== prepare inputs for action expert ===============

        pixel_values = torch.stack(pixel_values)
        system1_pixel_values = torch.stack(system1_pixel_values)
        input_ids = torch.cat(input_ids, dim=0)
        labels = torch.cat(labels, dim=0)

        if rlds_batch["action"].shape[0] > 1:
            action = torch.tensor(action, dtype=torch.float32)
            action_mask = None
            if "action_mask" in rlds_batch:
                action_mask = torch.tensor(rlds_batch["action_mask"], dtype=torch.bool)

        return dict(pixel_values=pixel_values, input_ids=input_ids, labels=labels, 
                    dataset_name=dataset_name, actions=action, action_masks=action_mask, 
                    episode_idx=rlds_batch["idx"], frame_idx=rlds_batch['frame_idx'],
                    system1_pixel_values=system1_pixel_values,
                    )

from torch.utils.data import DataLoader, Dataset, DistributedSampler, IterableDataset
from itertools import cycle

@dataclass
class PaddedCollatorForActionPrediction:
    model_max_length: int
    pad_token_id: int
    padding_side: str = "right"
    pixel_values_dtype: torch.dtype = torch.float32
    mm_dataloader: DataLoader = None

    def __call__(self, instances: Sequence[Dict[str, torch.Tensor]]) -> Dict[str, torch.Tensor]:

        batch_pixel_values   = [inst["pixel_values"]            for inst in instances]  # List[Tensor], each [M_i, C, H, W]
        system1_pixel_values = [inst["system1_pixel_values"]    for inst in instances]  # List[Tensor], each [M_i, C, H, W]
        batch_input_ids      = [inst["input_ids"]               for inst in instances]  # List[Tensor], each [M_i, seq_len]
        batch_labels         = [inst["labels"]                  for inst in instances]  


        assert self.padding_side == "right", f"Invalid Tokenizer `padding_side={self.padding_side}`, must be 'right'"
        all_input_ids = []
        all_labels    = []
        for i in range(len(batch_input_ids)):
            for row_ids in batch_input_ids[i]:   # shape (seq_len_i,)
                all_input_ids.append(row_ids)
            for row_lbl in batch_labels[i]:      # shape (seq_len_i,)
                all_labels.append(row_lbl)

        input_ids = pad_sequence(all_input_ids, batch_first=True, padding_value=self.pad_token_id)
        labels    = pad_sequence(all_labels,    batch_first=True, padding_value=IGNORE_INDEX)

        # truncate if needed: model_max_length
        if input_ids.shape[1] > self.model_max_length:
            input_ids = input_ids[:, : self.model_max_length]
            labels    = labels[:, : self.model_max_length]

        attention_mask = input_ids.ne(self.pad_token_id)
        system1_pixel_values = torch.cat(system1_pixel_values, dim=0).to(self.pixel_values_dtype).squeeze(1)
        pixel_values = torch.cat(batch_pixel_values, dim=0).to(self.pixel_values_dtype).squeeze(1)

        if "dataset_name" in instances[0]:
            dataset_names = [instance["dataset_name"] for instance in instances]
        else:
            dataset_names = None

        # Adding continuous actions and batch processing.
        actions = [instance["actions"] for instance in instances]
        actions = torch.stack(actions)
        action_masks = [instance["action_masks"] for instance in instances]
        action_masks = torch.stack(action_masks)

        output = dict(
            pixel_values=pixel_values,
            system1_pixel_values=system1_pixel_values,
            input_ids=input_ids,
            attention_mask=attention_mask,
            labels=labels,
            actions=actions,
            action_masks=action_masks,
            episode_idx=[instance['episode_idx'] for instance in instances] if 'episode_idx' in instances[0] else None,
            frame_idx=[instance['frame_idx'] for instance in instances] if 'frame_idx' in instances[0] else None,
        )
        if dataset_names is not None:
            output["dataset_names"] = dataset_names

        if self.mm_dataloader is not None:
            # co-training VLM
            mm_batch = next(self.mm_dataloader)
        else:
            mm_batch = None
        return output, mm_batch

def get_vla_dataset_and_collator(
    data_root_dir: Path,
    data_mix: str,
    tokenizer: PreTrainedTokenizerBase,
    processor: AutoProcessor,
    image_processor: AutoProcessor,
    default_image_resolution: Tuple[int, int, int],
    padding_side: str = "right",
    shuffle_buffer_size: int = 100_000,
    train: bool = True,
    episodic: bool = False,
    image_aug: bool = False,
    future_action_window_size: int = 0,
    past_action_window_size: int = 1,         # Concatenated `past_action_window_size-1' actions and the current action for the input
    load_all_data_for_training: bool = True,  # Load all data for training, or only a subset
    base_action_tokenizer: PreTrainedTokenizerBase = None,
    mm_dataset = None,
    mm_collator = None,
    stage = "stage1",
    disable_instruction = False,
) -> Tuple[Dataset, ActionTokenizer, PaddedCollatorForActionPrediction]:
    """Initialize RLDS Dataset (wraps TFDS), ActionTokenizer, and initialize transform/collation functions."""
    if base_action_tokenizer is None:
        action_tokenizer = None
    else:
        action_tokenizer = ActionTokenizer(base_action_tokenizer)
    # action_tokenizer = ActionTokenizer(tokenizer)
    if mm_dataset is not None:
        sampler = DistributedSampler(
            mm_dataset,
            num_replicas=overwatch.world_size(),
            rank=overwatch.rank(),
            shuffle=True,
            seed=42,
            drop_last=True,
        )

        mm_dataloader = DataLoader(
            mm_dataset,
            batch_size=2,
            sampler=sampler,
            collate_fn=mm_collator,
            num_workers=4,
        )
    mm_iter = iter(mm_dataloader) if mm_dataset is not None else None
    mm_iter = cycle(mm_iter) if mm_iter is not None else None # make it infinite

    batch_transform = RLDSBatchTransform(
        action_tokenizer, tokenizer, processor, image_processor, stage, disable_instruction
    )
    collator = PaddedCollatorForActionPrediction(
        tokenizer.model_max_length, tokenizer.pad_token_id, padding_side=padding_side, mm_dataloader = mm_iter
    )

    # Build RLDS Iterable Dataset
    cls = RLDSDataset if not episodic else EpisodicRLDSDataset
    dataset = cls(
        data_root_dir,
        data_mix,
        batch_transform,
        resize_resolution=default_image_resolution[1:],
        shuffle_buffer_size=shuffle_buffer_size,
        train=train,
        future_action_window_size=future_action_window_size,
        past_action_window_size=past_action_window_size,
        image_aug=image_aug,
        load_all_data_for_training=load_all_data_for_training,
    )

    return dataset, action_tokenizer, collator


# === Load Pretrained Model ===
def load(
    model_id_or_path: Union[str, Path],
    cache_dir: Optional[Union[str, Path]] = None,
    load_for_training: bool = False,
    llm_backbone_id=None,
    stage = 'stage1',
    num_of_meta_query = 64,
) -> PrismaticVLM:

    if llm_backbone_id is None:
        llm_backbone_id = os.path.abspath(
            os.path.join(os.path.dirname(__file__), "..", "ckpt", "Eagle2-2B")
        )

    """Loads a pretrained PrismaticVLM from either local disk or the HuggingFace Hub."""
    if os.path.isdir(model_id_or_path):
        overwatch.info(f"Loading from local path `{(run_dir := Path(model_id_or_path))}`")

        # Get paths for `config.json` and pretrained checkpoint
        config_json, checkpoint_pt = run_dir / "config.json", run_dir / "checkpoints" / "latest-checkpoint.pt"
        assert config_json.exists(), f"Missing `config.json` for `{run_dir = }`"

    tokenizer = AutoTokenizer.from_pretrained(llm_backbone_id, use_fast=True)

    processor = EagleProcessor(
        llm_backbone_id,
        max_input_tiles=1,
        model_spec=SimpleNamespace(
            num_image_token = 256,
            template = "qwen2-chat"
        ),
    )

    vlm = AutoModel.from_pretrained(
        llm_backbone_id,
        attn_implementation="flash_attention_2",
        trust_remote_code=True
        )

    new_tokens = ['<new_token_{}>'.format(i) for i in range(num_of_meta_query)]  # Create 256 new token names
    overwatch.info(f"add {len(new_tokens)} latent action tokens")
    tokenizer.add_tokens(new_tokens)  # Add them to the tokenizer
    tokenizer.new_tokens = new_tokens
    processor.tokenizer = tokenizer

    # Resize the model's token embeddings to match the new vocabulary size
    try:  # Skip mean resize feature in new version of transformers
        vlm.language_model.resize_token_embeddings(len(tokenizer), mean_resizing = False)
    except Exception as e:
        vlm.language_model.resize_token_embeddings(len(tokenizer))

    vlm.img_context_token_id = processor.get_img_context_token()
    assert vlm.template == processor.model_spec.template
    assert vlm.num_image_token == processor.model_spec.num_image_token

    new_token_ids = tokenizer.convert_tokens_to_ids(new_tokens)

    # Load VLM using `from_pretrained` (clobbers HF syntax... eventually should reconcile)
    overwatch.info(f"Loading VLM [bold blue]{llm_backbone_id}[/] from Checkpoint")
    vla = InstructVLA(
        vlm = vlm,
        config_json = config_json,
        tokenizer = tokenizer,
        processor = processor,
        token_size= vlm.config.llm_config.hidden_size,
        meta_token_ids = new_token_ids,
        stage = stage
    )

    return vla

# === Load Pretrained VLA Model ===
def load_vla(
    model_id_or_path: Union[str, Path],
    cache_dir: Optional[Union[str, Path]] = None,
    load_for_training: bool = False,
    model_type: str = "pretrained",
    **kwargs,
) -> InstructVLA:
    """Loads a pretrained InstructVLA from either local disk or the HuggingFace Hub."""

    # TODO (siddk, moojink) :: Unify semantics with `load()` above; right now, `load_vla()` assumes path points to
    #   checkpoint `.pt` file, rather than the top-level run directory!
    if os.path.isfile(model_id_or_path):
        overwatch.info(f"Loading from local checkpoint path `{(checkpoint_pt := Path(model_id_or_path))}`")

        # [Validate] Checkpoint Path should look like `.../<RUN_ID>/checkpoints/<CHECKPOINT_PATH>.pt`
        assert (checkpoint_pt.suffix == ".pt") and (checkpoint_pt.parent.name == "checkpoints"), "Invalid checkpoint!"
        run_dir = checkpoint_pt.parents[1]

        # Get paths for `config.json`, `dataset_statistics.json` and pretrained checkpoint
        config_json, dataset_statistics_json = run_dir / "config.json", run_dir / "dataset_statistics.json"
        assert config_json.exists(), f"Missing `config.json` for `{run_dir = }`"
        assert dataset_statistics_json.exists(), f"Missing `dataset_statistics.json` for `{run_dir = }`"

    # Otherwise =>> try looking for a match on `model_id_or_path` on the HF Hub (`model_id_or_path`)
    else:
        # Search HF Hub Repo via fsspec API
        overwatch.info(f"Checking HF for `{(hf_path := str(Path(model_id_or_path)))}`")
        if not (tmpfs := HfFileSystem()).exists(hf_path):
            raise ValueError(f"Couldn't find valid HF Hub Path `{hf_path = }`")

        valid_ckpts = tmpfs.glob(f"{hf_path}/checkpoints/*.pt")
        if (len(valid_ckpts) == 0) or (len(valid_ckpts) != 1):
            raise ValueError(f"Couldn't find a valid checkpoint to load from HF Hub Path `{hf_path}/checkpoints/")

        target_ckpt = Path(valid_ckpts[-1]).name
        model_id_or_path = str(model_id_or_path)  # Convert to string for HF Hub API
        overwatch.info(f"Downloading Model `{model_id_or_path}` Config & Checkpoint `{target_ckpt}`")
        with overwatch.local_zero_first():
            # relpath = Path(model_type) / model_id_or_path
            config_json = hf_hub_download(
                repo_id=model_id_or_path, filename=f"{('config.json')!s}", cache_dir=cache_dir
            )
            dataset_statistics_json = hf_hub_download(
                repo_id=model_id_or_path, filename=f"{('dataset_statistics.json')!s}", cache_dir=cache_dir
            )
            checkpoint_pt = hf_hub_download(
                repo_id=model_id_or_path, filename=f"{(Path('checkpoints') / target_ckpt)!s}", cache_dir=cache_dir
            )

    # Load VLA Config (and corresponding base VLM `ModelConfig`) from `config.json`
    with open(dataset_statistics_json, "r") as f:
        norm_stats = json.load(f)
    with open(config_json, "r") as f:
        config = json.load(f)
    num_of_meta_query, num_of_meta_query_from_config = None, None
    if 'num_of_meta_query' in kwargs:
        num_of_meta_query = kwargs.pop("num_of_meta_query")
    if 'num_of_meta_query' in config:
        num_of_meta_query_from_config = config.pop("num_of_meta_query")

    if num_of_meta_query is not None and num_of_meta_query_from_config is not None:
        assert num_of_meta_query == num_of_meta_query_from_config, f'you need {num_of_meta_query} meta queries, but the checkpoint is trained with {num_of_meta_query} meta queries.'
    elif num_of_meta_query is None and num_of_meta_query_from_config is None:
        num_of_meta_query = 64
        overwatch.info(f"not specify the number of meta query, the default value is 64 !!")

    if num_of_meta_query_from_config is not None:
        num_of_meta_query = num_of_meta_query_from_config

    if 'stage' in config:
        training_stage_from_config = config.pop("stage")
    if 'stage' in kwargs:
        _ = kwargs.pop("stage")
    
    vla = InstructVLA.from_pretrained(
        checkpoint_pt,
        freeze_weights=not load_for_training,
        norm_stats = norm_stats,
        num_of_meta_query=num_of_meta_query,
        stage = training_stage_from_config,
        **kwargs,
    )

    return vla


