import base64
import os
from typing import List, Optional

import anthropic
from anthropic.types import Message, ContentBlock, TextBlockParam, MessageParam
from anthropic.types.image_block_param import Source, ImageBlockParam

from src.llm_messenger.classes.content import Content
from src.llm_messenger.classes.image_content import ImageContent
from src.llm_messenger.classes.llm_messenger import LLMMessenger
from src.llm_messenger.classes.text_content import TextContent


class ClaudeMessenger(LLMMessenger):
    def __init__(
        self,
        api_key: str,
        model_name: str = "claude-3-opus-20240229",
        temperature: float = 0.0,
        log_directory: str = "",
    ):
        super().__init__(model_name, temperature, log_directory)
        self.__client = anthropic.Anthropic(api_key=api_key)
        self._context: Optional[MessageParam] = None  # set type explicitly

    def __encode_image(self, image_path: str) -> str:
        with open(image_path, "rb") as image_file:
            return base64.b64encode(image_file.read()).decode("utf-8")

    def __get_text_content(self, message: str) -> TextBlockParam:
        return TextBlockParam(type="text", text=message)

    def __get_image_content(self, image_path: str) -> ImageBlockParam:
        _, ext = os.path.splitext(image_path)
        ext = ext.replace(".", "")
        ext = "jpeg" if ext == "jpg" else ext
        assert ext in {"jpeg", "png", "gif", "webp"}
        media_type = f"image/{ext}"
        return ImageBlockParam(
            type="image",
            source=Source(
                type="base64",
                media_type=media_type,
                data=self.__encode_image(image_path),
            ),
        )

    def __get_messages_with_context(
        self, contents: List[Content]
    ) -> List[ContentBlock]:
        messages = self.get_context()
        messages.append(
            MessageParam(role="user", content=self.__convert_contents(contents))
        )
        return messages

    def __update_context(self, messages: List[MessageParam], model_response: str):
        if self._context is not None:
            if not self._keep_image_history:
                messages = [
                    MessageParam(
                        role=m.role,
                        content=[c for c in m.content if isinstance(c, TextBlockParam)],
                    )
                    for m in messages
                ]
            messages.append(
                MessageParam(
                    role="assistant",
                    content=[TextBlockParam(type="text", text=model_response)],
                )
            )
            self._context = messages

    def __convert_contents(self, contents: List[Content]) -> List[ContentBlock]:
        converted_contents = []

        for content in contents:
            if isinstance(content, ImageContent):
                converted_contents.append(self.__get_image_content(content.image_path))
            elif isinstance(content, TextContent):
                converted_contents.append(self.__get_text_content(content.text))

        return converted_contents

    def __get_model_response(self, response: Message) -> str:
        return response.content[0].text

    def ask(self, contents: List[Content]) -> str:
        if self._context is None:
            self.log("INFO", [TextContent("Opening new context")])

        self.log("USER", contents)
        messages = self.__get_messages_with_context(contents)

        response = self.__client.messages.create(
            model=self._model_name,
            max_tokens=1024,
            messages=messages,
            temperature=self._temperature,
        )

        model_response = self.__get_model_response(response)
        self.log("ASSISTANT", [TextContent(model_response)])
        self.__update_context(messages, model_response)
        return model_response
