import threading
import time
from collections import deque
from typing import Deque, List, Optional, Union

import torch

from vllm.distributed.kv_transfer.kv_lookup_buffer.base import (
    KVLookupBufferBase)
from vllm.distributed.kv_transfer.kv_pipe.base import KVPipeBase
from vllm.distributed.kv_transfer.kv_pipe.torch_distributed_pipe import TorchDistributedPipe

from concurrent.futures import Future

from vllm.logger import init_logger

from queue import SimpleQueue, Empty

logger = init_logger(__name__)


class SimpleKVLookupBuffer(KVLookupBufferBase):

    def __init__(self, signal_pipe: KVPipeBase, data_pipe: KVPipeBase,
                 buffer_size_thresh: int):
        """
        signal_pipe: on CPU 
        
        NOTE: on-device recv will block all threads in the process, making the 
        KV cache producer unable to listen to new request while transmitting 
        KV cache. Luckily CPU recv only blocks the current thread so we use 
        CPU recv to listen to new request.
        
        data_pipe: on device (e.g. GPU)
        """

        self.buffer: Deque[List[torch.Tensor]] = deque()

        self.buffer_size = 0
        self.buffer_size_threshold = buffer_size_thresh
        self.signal_pipe = signal_pipe
        self.data_pipe = data_pipe
        self.request_handling_thread: Optional[threading.Thread] = None

        self.normal_signal = torch.tensor([0])
        self.end_signal = None

        self.insert_queue: SimpleQueue[List[torch.Tensor]] = SimpleQueue()
        self.sent_size_queue: SimpleQueue[int] = SimpleQueue()

    def _matches(self, tokens_roi_sender: List[torch.Tensor],
                 tokens_roi_recver: List[torch.Tensor]):

        # tokens_roi_sender: tokens and roi of the producer (in the buffer)
        # tokens_roi_recver: tokens and roi of the consumer (query)

        tokens_sender = tokens_roi_sender[0]
        tokens_recver = tokens_roi_recver[0]

        if tokens_recver is None:
            # consumer sends an empty request
            # semantics: DROP SELECT * LIMIT 1
            # so any of the data in the buffer can be drop-selected
            return True

        # simple common prefix matching
        min_length = min(len(tokens_sender), len(tokens_recver))
        if torch.allclose(tokens_sender[:min_length],
                          tokens_recver[:min_length]):
            return min_length

        return 0

    def _send_tensor_and_dec_size(self,
                                  tensor: Optional[torch.Tensor]) -> None:

        assert tensor is not None, "Use self.data_pipe.send(None) instead"
        self.sent_size_queue.put(tensor.element_size() * tensor.numel())
        self.data_pipe.send_tensor(tensor)

    def _send_tensor_list_and_dec_size(
            self,
            tensor_list: List[Optional[torch.Tensor]]
        ) -> None:
        tensor_size = 0
        for tensor in tensor_list:
            if tensor is not None:
                tensor_size += tensor.element_size() * tensor.numel()
        self.sent_size_queue.put(tensor_size)
        self.data_pipe.send_tensor(tensor_list)

    def _get_element_size(self, data: Optional[Union[List, torch.Tensor]]):

        if isinstance(data, torch.Tensor):
            return data.element_size() * data.numel()
        if not data:
            # cannot perform `not data` on a tensor
            # so this check needs to go after the check above
            return 0

        raise AssertionError("Unknown data type %s" % type(data))

    def _add_to_buffer(self, input_tokens: torch.Tensor, roi: torch.Tensor,
                       key: torch.Tensor, value: torch.Tensor,
                       hidden: torch.Tensor):

        if isinstance(input_tokens, torch.Tensor):
            input_tokens = input_tokens.clone()
        # ignore roi
        #if isinstance(key, torch.Tensor):
        #    key = key.clone()
        #if isinstance(value, torch.Tensor):
        #    value = value.clone()
        if isinstance(hidden, torch.Tensor):
            hidden = hidden.clone()

        buffer_item = [input_tokens, key, value, hidden]

        item_size = sum(self._get_element_size(item) for item in buffer_item)

        self.buffer_size += item_size
        self.insert_queue.put(buffer_item)

    def _is_end_signal(self, signal):
        return signal is None

    def drop_select_handler(self):
        try:

            while True:
                signal = self.signal_pipe.recv_tensor()
                if self._is_end_signal(signal):
                    logger.info("Received end signal!")
                    break

                input_tokens = self.data_pipe.recv_tensor()

                tokens_roi_recver = [input_tokens]

                matched_length = 0

                # move buffer item from insert queue to buffer
                try:
                    while True:
                        self.buffer.append(self.insert_queue.get_nowait())
                except Empty:
                    pass

                # perform input tokens and roi matching
                # FIXME: this matching is O(n), ideally it should be O(1)
                # but this buffer size won't (and shouldn't) be too large so
                # the fix is not urgent.

                for _ in range(len(self.buffer)):

                    temp_length = self._matches(self.buffer[0],
                                                tokens_roi_recver)
                    if temp_length > 0:
                        matched_length = temp_length
                        break
                    # rotate the element we just accessed to the end
                    self.buffer.rotate(-1)

                if matched_length > 0:
                    # need to clone the tensor
                    # in case the tensor is freed before sending finishes
                    matched_item = self.buffer.popleft()
                    self._send_tensor_list_and_dec_size(matched_item)

                else:
                    # no match, just send None
                    for _ in range(4):
                        self.data_pipe.send_tensor(None)

        except RuntimeError as e:
            if 'Connection closed by peer' not in str(e):
                raise e

        logger.debug("Closing drop_select_handler")

    def drop_select(
            self, input_tokens: Optional[torch.Tensor],
            roi: Optional[torch.Tensor]) -> List[Optional[torch.Tensor]]:

        assert self.request_handling_thread is None, \
            "drop_select should be called by the KV cache consumer "\
            "(e.g. the decode vLLM instance)"

        if isinstance(input_tokens, torch.Tensor):
            input_tokens = input_tokens.clone()
        #ignore roi

        self.signal_pipe.send_tensor(self.normal_signal)
        self.data_pipe.send_tensor(input_tokens)

        input_tokens = self.data_pipe.recv_tensor()
        key = self.data_pipe.recv_tensor()
        value = self.data_pipe.recv_tensor()
        hidden = self.data_pipe.recv_tensor()

        return [input_tokens, key, value, hidden]
    
    def async_drop_select(
            self, input_tokens: Optional[torch.Tensor],
            ) -> Future[List[torch.Tensor]]:

        assert self.request_handling_thread is None, \
            "drop_select should be called by the KV cache consumer "\
            "(e.g. the decode vLLM instance)"
        
        assert isinstance(self.data_pipe, TorchDistributedPipe)

        if isinstance(input_tokens, torch.Tensor):
            input_tokens = input_tokens.clone()

        self.signal_pipe.send_tensor(self.normal_signal)

        return self.data_pipe.irecv_match_tensor(input_tokens)
        # self.data_pipe.send_tensor(input_tokens)
        # return self.data_pipe.irecv_tensor(4)
    
    def async_drop_select_get_result(self, fut: Future[List[torch.Tensor]]):
        assert isinstance(self.data_pipe, TorchDistributedPipe)
        return self.data_pipe.irecv_get_result(fut)

    def full_handler(self):
        time.sleep(0.001)

    def insert(self, input_tokens: torch.Tensor, roi: torch.Tensor,
               key: torch.Tensor, value: torch.Tensor,
               hidden: torch.Tensor) -> None:
        
        try:
            while True:
                self.buffer_size -= self.sent_size_queue.get_nowait()
        except Empty:
            pass

        buffer_item = [input_tokens, key, value, hidden]
        item_size = sum(self._get_element_size(item) for item in buffer_item)

        if self.buffer_size + item_size > self.buffer_size_threshold:
            # log outside the while loop to avoid this message being logged
            # repeatedly.
            logger.debug("KV transfer buffer is full. Handling...")
        while self.buffer_size + item_size > self.buffer_size_threshold:
            self.buffer_size -= self.sent_size_queue.get()

        self._add_to_buffer(input_tokens, roi, key, value, hidden)

        # when calling the insert, the current process is a sender
        # need to launch the request handler and start listening to request.
        if self.request_handling_thread is None:
            self.request_handling_thread = threading.Thread(
                target=self.drop_select_handler)
            self.request_handling_thread.start()

    def close(self):

        if hasattr(self, "request_handling_thread"
                   ) and self.request_handling_thread is not None:
            self.request_handling_thread.join()

        else:
            # TODO: have a explicit close signal and have a explicit way to
            # check if it's requester
            self.signal_pipe.send_tensor(self.end_signal)
