# MIT License

# Copyright (c) 2024 The HuggingFace Team

# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:

# The above copyright notice and this permission notice shall be included in all
# copies or substantial portions of the Software.

# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.

import asyncio
from dataclasses import dataclass, replace
from typing import Coroutine, Optional

import requests
from huggingface_hub import TextGenerationInputGenerateParameters, TextGenerationInputGrammarType, TextGenerationOutput
from transformers import AutoTokenizer

from lighteval.models.endpoints.endpoint_model import InferenceEndpointModel, ModelInfo
from lighteval.models.model_input import GenerationParameters
from lighteval.utils.imports import NO_TGI_ERROR_MSG, is_tgi_available


if is_tgi_available():
    from text_generation import AsyncClient


BATCH_SIZE = 50


def divide_chunks(array, n):
    # looping till length array
    for i in range(0, len(array), n):
        yield array[i : i + n]


@dataclass
class TGIModelConfig:
    inference_server_address: str
    inference_server_auth: str
    model_id: str
    generation_parameters: GenerationParameters = None

    def __post_init__(self):
        if not self.generation_parameters:
            self.generation_parameters = GenerationParameters()

    @classmethod
    def from_path(cls, path: str) -> "TGIModelConfig":
        """Load configuration for TGI endpoint model from YAML file path.

        Args:
            path (`str`): Path of the model configuration YAML file.

        Returns:
            [`TGIModelConfig`]: Configuration for TGI endpoint model.
        """
        import yaml

        with open(path, "r") as f:
            config = yaml.safe_load(f)["model"]
        return cls(**config["instance"], generation_parameters=GenerationParameters.from_dict(config))


# inherit from InferenceEndpointModel instead of LightevalModel since they both use the same interface, and only overwrite
# the client functions, since they use a different client.
class ModelClient(InferenceEndpointModel):
    _DEFAULT_MAX_LENGTH: int = 4096

    def __init__(self, config: TGIModelConfig) -> None:
        if not is_tgi_available():
            raise ImportError(NO_TGI_ERROR_MSG)
        headers = (
            {} if config.inference_server_auth is None else {"Authorization": f"Bearer {config.inference_server_auth}"}
        )

        self.client = AsyncClient(config.inference_server_address, headers=headers, timeout=240)
        self.generation_parameters = config.generation_parameters
        self.generation_config = TextGenerationInputGenerateParameters(**self.generation_parameters.to_tgi_ie_dict())
        self._max_gen_toks = 256
        self.model_info = requests.get(f"{config.inference_server_address}/info", headers=headers).json()
        if "model_id" not in self.model_info:
            raise ValueError("Error occured when fetching info: " + str(self.model_info))
        if config.model_id:
            self.model_info["model_id"] = config.model_id
        self._tokenizer = AutoTokenizer.from_pretrained(self.model_info["model_id"])
        self._add_special_tokens = True
        self.use_async = True

        model_name = str(self.model_info["model_id"])
        model_sha = self.model_info["model_sha"]
        model_precision = self.model_info["model_dtype"]
        self.model_info = ModelInfo(
            model_name=model_name,
            model_sha=model_sha,
            model_dtype=model_precision,
            model_size=-1,
        )

    def _async_process_request(
        self,
        context: str,
        stop_tokens: list[str],
        max_tokens: int,
        grammar: Optional[TextGenerationInputGrammarType] = None,
    ) -> Coroutine[None, list[TextGenerationOutput], str]:
        # Todo: add an option to launch with conversational instead for chat prompts
        # We create a copy of the current text generation params
        generation_config: TextGenerationInputGenerateParameters = replace(
            self.generation_config,
            stop=stop_tokens,
            max_new_tokens=max_tokens,
            details=True,
            decoder_input_details=True,
            grammar=grammar,
        )

        generated_text = self.client.generate(prompt=context, generation_config=generation_config)

        return generated_text

    def _process_request(self, *args, **kwargs) -> TextGenerationOutput:
        return asyncio.run(self._async_process_request(*args, **kwargs))

    def set_cache_hook(self, cache_hook):
        self.cache_hook = cache_hook

    @property
    def tokenizer(self):
        return self._tokenizer

    @property
    def add_special_tokens(self):
        return self._add_special_tokens

    @property
    def max_length(self) -> int:
        if hasattr(self.tokenizer, "model_max_length"):
            return self.tokenizer.model_max_length
        return ModelClient._DEFAULT_MAX_LENGTH

    @property
    def disable_tqdm(self) -> bool:
        False

    def cleanup(self):
        pass
