from importlib.metadata import version
import warnings
import transformers
import transformers.models.mistral.modeling_mistral

from methods.adakv.adaptive_snapkv.adaptive_mistral_hijack import adaptive_mistral_flash_attn2_forward,adaptive_MistralModel_forward
from methods.adakv.adaptive_snapkv.adaptive_mistral_hijack import prepare_inputs_for_generation_mistral as ada_prepare_inputs_for_generation_mistral

from methods.adakv.adaptive_snapkv.adaptive_llama_hijack import adaptive_llama_flash_attn2_forward,adaptive_LlamaModel_forward
from methods.adakv.adaptive_snapkv.adaptive_llama_hijack import prepare_inputs_for_generation_llama as ada_prepare_inputs_for_generation_llama


from methods.adakv.adaptive_snapkv.adaptive_qwen_hijack import adaptive_qwen2_flash_attn2_forward,adaptive_qwen2_model_forward
# def check_version():
#     try:
#         transformers_version = version("transformers")
#     except Exception as e:
#         print(f"Transformers not installed: {e}")
#     version_list = ['4.43']
#     warning_flag = True
#     for x in version_list:
#         if x in transformers_version:
#             warning_flag = False
#             break
#     if warning_flag:
#         warnings.warn(f"Transformers version {transformers_version} might not be compatible with SnapKV. SnapKV is tested with Transformers version {version_list}.")


def replace_mistral():
    # check_version()
    # transformers.models.mistral.modeling_mistral.MistralForCausalLM.prepare_inputs_for_generation = ada_prepare_inputs_for_generation_mistral
    transformers.models.mistral.modeling_mistral.MistralFlashAttention2.forward = adaptive_mistral_flash_attn2_forward
    transformers.models.mistral.modeling_mistral.MistralModel.forward = adaptive_MistralModel_forward

def replace_llama():
    # check_version()
    # transformers.models.llama.modeling_llama.LlamaForCausalLM.prepare_inputs_for_generation = ada_prepare_inputs_for_generation_llama
    transformers.models.llama.modeling_llama.LlamaFlashAttention2.forward = adaptive_llama_flash_attn2_forward
    transformers.models.llama.modeling_llama.LlamaModel.forward = adaptive_LlamaModel_forward

def replace_qwen2():
    # check_version()

    # transformers.models.qwen.modeling_qwen.QwenForCausalLM.prepare_inputs_for_generation = None
    transformers.models.qwen2.modeling_qwen2.Qwen2Model.forward = adaptive_qwen2_model_forward
    transformers.models.qwen2.modeling_qwen2.Qwen2FlashAttention2.forward  = adaptive_qwen2_flash_attn2_forward