from __future__ import annotations

import base64
import logging
import math
import os
import sys
import time
import warnings
from functools import lru_cache
from io import BytesIO
import re

import requests
import torch
import torchvision
from packaging import version
from PIL import Image
from torchvision import io, transforms
from torchvision.transforms import InterpolationMode
from typing import Optional
import random

logger = logging.getLogger(__name__)

IMAGE_FACTOR = 28
MIN_PIXELS = 4 * 28 * 28
MAX_PIXELS = 16384 * 28 * 28
MAX_RATIO = 200

VIDEO_MIN_PIXELS = 128 * 28 * 28
VIDEO_MAX_PIXELS = 768 * 28 * 28
FRAME_FACTOR = 2
FPS = 2.0
FPS_MIN_FRAMES = 128
FPS_MAX_FRAMES = 1024

# Set the maximum number of video token inputs.
# Here, 128K represents the maximum number of input tokens for the VLLM model.
# Remember to adjust it according to your own configuration.
VIDEO_TOTAL_PIXELS = int(float(os.environ.get('VIDEO_MAX_PIXELS', 128000 * 28 * 28)))
logger.info(f"set VIDEO_TOTAL_PIXELS: {VIDEO_TOTAL_PIXELS}")


def round_by_factor(number: int, factor: int) -> int:
    """Returns the closest integer to 'number' that is divisible by 'factor'."""
    return round(number / factor) * factor


def ceil_by_factor(number: int, factor: int) -> int:
    """Returns the smallest integer greater than or equal to 'number' that is divisible by 'factor'."""
    return math.ceil(number / factor) * factor


def floor_by_factor(number: int, factor: int) -> int:
    """Returns the largest integer less than or equal to 'number' that is divisible by 'factor'."""
    return math.floor(number / factor) * factor


def smart_resize(
    height: int, width: int, factor: int = IMAGE_FACTOR, min_pixels: int = MIN_PIXELS, max_pixels: int = MAX_PIXELS
) -> tuple[int, int]:
    """
    Rescales the image so that the following conditions are met:

    1. Both dimensions (height and width) are divisible by 'factor'.

    2. The total number of pixels is within the range ['min_pixels', 'max_pixels'].

    3. The aspect ratio of the image is maintained as closely as possible.
    """
    if max(height, width) / min(height, width) > MAX_RATIO:
        raise ValueError(
            f"absolute aspect ratio must be smaller than {MAX_RATIO}, got {max(height, width) / min(height, width)}"
        )
    h_bar = max(factor, round_by_factor(height, factor))
    w_bar = max(factor, round_by_factor(width, factor))
    if h_bar * w_bar > max_pixels:
        beta = math.sqrt((height * width) / max_pixels)
        h_bar = floor_by_factor(height / beta, factor)
        w_bar = floor_by_factor(width / beta, factor)
    elif h_bar * w_bar < min_pixels:
        beta = math.sqrt(min_pixels / (height * width))
        h_bar = ceil_by_factor(height * beta, factor)
        w_bar = ceil_by_factor(width * beta, factor)
    return h_bar, w_bar


def to_rgb(pil_image: Image.Image) -> Image.Image:
      if pil_image.mode == 'RGBA':
          white_background = Image.new("RGB", pil_image.size, (255, 255, 255))
          white_background.paste(pil_image, mask=pil_image.split()[3])  # Use alpha channel as mask
          return white_background
      else:
          return pil_image.convert("RGB")


def fetch_image(ele: dict[str, str | Image.Image], size_factor: int = IMAGE_FACTOR) -> Image.Image:
    if "image" in ele:
        image = ele["image"]
    else:
        image = ele["image_url"]
    image_obj = None
    if isinstance(image, Image.Image):
        image_obj = image
    elif image.startswith("http://") or image.startswith("https://"):
        response = requests.get(image, stream=True)
        image_obj = Image.open(BytesIO(response.content))
    elif image.startswith("file://"):
        image_obj = Image.open(image[7:])
    elif image.startswith("data:image"):
        if "base64," in image:
            _, base64_data = image.split("base64,", 1)
            data = base64.b64decode(base64_data)
            image_obj = Image.open(BytesIO(data))
    else:
        image_obj = Image.open(image)
    if image_obj is None:
        raise ValueError(f"Unrecognized image input, support local path, http url, base64 and PIL.Image, got {image}")
    image = to_rgb(image_obj)
    ## resize
    if "resized_height" in ele and "resized_width" in ele:
        resized_height, resized_width = smart_resize(
            ele["resized_height"],
            ele["resized_width"],
            factor=size_factor,
        )
    else:
        width, height = image.size
        min_pixels = ele.get("min_pixels", MIN_PIXELS)
        max_pixels = ele.get("max_pixels", MAX_PIXELS)
        resized_height, resized_width = smart_resize(
            height,
            width,
            factor=size_factor,
            min_pixels=min_pixels,
            max_pixels=max_pixels,
        )
    image = image.resize((resized_width, resized_height))

    return image


def smart_nframes(
    ele: dict,
    total_frames: int,
    video_fps: int | float,
) -> int:
    """calculate the number of frames for video used for model inputs.

    Args:
        ele (dict): a dict contains the configuration of video.
            support either `fps` or `nframes`:
                - nframes: the number of frames to extract for model inputs.
                - fps: the fps to extract frames for model inputs.
                    - min_frames: the minimum number of frames of the video, only used when fps is provided.
                    - max_frames: the maximum number of frames of the video, only used when fps is provided.
        total_frames (int): the original total number of frames of the video.
        video_fps (int | float): the original fps of the video.

    Raises:
        ValueError: nframes should in interval [FRAME_FACTOR, total_frames].

    Returns:
        int: the number of frames for video used for model inputs.
    """
    assert not ("fps" in ele and "nframes" in ele), "Only accept either `fps` or `nframes`"
    if "nframes" in ele:
        nframes = min(round_by_factor(ele["nframes"], FRAME_FACTOR), total_frames)
    else:
        fps = ele.get("fps", FPS)
        min_frames = ceil_by_factor(ele.get("min_frames", FPS_MIN_FRAMES), FRAME_FACTOR)
        max_frames = floor_by_factor(ele.get("max_frames", min(FPS_MAX_FRAMES, total_frames)), FRAME_FACTOR)
        nframes = total_frames / video_fps * fps
        if nframes > total_frames:
            logger.warning(f"smart_nframes: nframes[{nframes}] > total_frames[{total_frames}]")
        nframes = min(min(max(nframes, min_frames), max_frames), total_frames)
        nframes = floor_by_factor(nframes, FRAME_FACTOR)
    if not (FRAME_FACTOR <= nframes and nframes <= total_frames):
        raise ValueError(f"nframes should in interval [{FRAME_FACTOR}, {total_frames}], but got {nframes}.")
    return nframes


def _read_video_torchvision(
    ele: dict,
) -> (torch.Tensor, float):
    raise NotADirectoryError
    """read video using torchvision.io.read_video

    Args:
        ele (dict): a dict contains the configuration of video.
        support keys:
            - video: the path of video. support "file://", "http://", "https://" and local path.
            - video_start: the start time of video.
            - video_end: the end time of video.
    Returns:
        torch.Tensor: the video tensor with shape (T, C, H, W).
    """
    video_path = ele["video"]
    if version.parse(torchvision.__version__) < version.parse("0.19.0"):
        if "http://" in video_path or "https://" in video_path:
            warnings.warn("torchvision < 0.19.0 does not support http/https video path, please upgrade to 0.19.0.")
        if "file://" in video_path:
            video_path = video_path[7:]
    st = time.time()
    video, audio, info = io.read_video(
        video_path,
        start_pts=ele.get("video_start", 0.0),
        end_pts=ele.get("video_end", None),
        pts_unit="sec",
        output_format="TCHW",
    )
    total_frames, video_fps = video.size(0), info["video_fps"]
    logger.info(f"torchvision:  {video_path=}, {total_frames=}, {video_fps=}, time={time.time() - st:.3f}s")
    nframes = smart_nframes(ele, total_frames=total_frames, video_fps=video_fps)
    idx = torch.linspace(0, total_frames - 1, nframes).round().long()
    sample_fps = nframes / max(total_frames, 1e-6) * video_fps
    video = video[idx]
    return video, sample_fps


def is_decord_available() -> bool:
    import importlib.util

    return importlib.util.find_spec("decord") is not None


def _read_video_decord(
    ele: dict,
) -> (torch.Tensor, float):
    """read video using decord.VideoReader

    Args:
        ele (dict): a dict contains the configuration of video.
        support keys:
            - video: the path of video. support "file://", "http://", "https://" and local path.
            - video_start: the start time of video.
            - video_end: the end time of video.
    Returns:
        torch.Tensor: the video tensor with shape (T, C, H, W).
    """
    import decord
    video_path = ele["video"]
    st = time.time()

    if 's3://' in video_path:
        video_bytes = client.get(video_path)
        if video_bytes is None or len(video_bytes) == 0:
            raise ValueError(f"Can't read byte from {video_path}!")
        byteio = BytesIO(video_bytes)
        vr = decord.VideoReader(byteio, num_threads=1)
    else:
        byteio = None
        vr = decord.VideoReader(video_path)

    # TODO: support start_pts and end_pts
    if 'video_start' in ele or 'video_end' in ele:
        raise NotImplementedError("not support start_pts and end_pts in decord for now.")
    total_frames, video_fps = len(vr), vr.get_avg_fps()
    logger.info(f"decord:  {video_path=}, {total_frames=}, {video_fps=}, time={time.time() - st:.3f}s")
    nframes = smart_nframes(ele, total_frames=total_frames, video_fps=video_fps)
    idx = torch.linspace(0, total_frames - 1, nframes).round().long().tolist()
    video = vr.get_batch(idx).asnumpy()
    video = torch.tensor(video).permute(0, 3, 1, 2)  # Convert to TCHW format
    sample_fps = nframes / max(total_frames, 1e-6) * video_fps

    vr.seek(0)

    if byteio != None:
        byteio.close()
    return video, sample_fps


VIDEO_READER_BACKENDS = {
    "decord": _read_video_decord,
    "torchvision": _read_video_torchvision,
}

FORCE_QWENVL_VIDEO_READER = os.getenv("FORCE_QWENVL_VIDEO_READER", None)


@lru_cache(maxsize=1)
def get_video_reader_backend() -> str:
    if FORCE_QWENVL_VIDEO_READER is not None:
        video_reader_backend = FORCE_QWENVL_VIDEO_READER
    elif is_decord_available():
        video_reader_backend = "decord"
    else:
        video_reader_backend = "torchvision"
    print(f"qwen-vl-utils using {video_reader_backend} to read video.", file=sys.stderr)
    return video_reader_backend

def resize_video(video, sample_fps, total_pixels=VIDEO_TOTAL_PIXELS, min_pixels=16*28*28, image_factor: int = 28):
    maximum_frames = int(total_pixels / (min_pixels * 1.05) * 2)
    if video.shape[0] > maximum_frames:
        frame_idx = torch.linspace(0, video.shape[0] - 1, maximum_frames).round().long().tolist()
        sample_fps = maximum_frames / video.shape[0] * sample_fps
        video = video[frame_idx]

    nframes, _, height, width = video.shape
    
    max_pixels = max(min(768 * 28 * 28, total_pixels / nframes * 2.0), int(min_pixels * 1.05))
    max_pixels = min(total_pixels, max_pixels)
    resized_height, resized_width = smart_resize(
        height,
        width,
        factor=image_factor,
        min_pixels=min_pixels,
        max_pixels=max_pixels,
    )
    video = transforms.functional.resize(
        video,
        [resized_height, resized_width],
        interpolation=InterpolationMode.BICUBIC,
        antialias=True,
    ).float()
    return video, sample_fps

def fetch_video(ele: dict, image_factor: int = IMAGE_FACTOR, return_video_sample_fps: bool = False, resize: bool = True, maximum_frames=None) -> torch.Tensor | list[Image.Image]:
    if isinstance(ele["video"], str):
        video_reader_backend = get_video_reader_backend()
        try:
            video, sample_fps = VIDEO_READER_BACKENDS[video_reader_backend](ele)
        except Exception as e:
            logger.warning(f"read error {ele}, {e}")
            video = torch.zeros(10, 3, 224, 224)
            sample_fps = 1.0
        
        if "timestamps" in ele and ele["timestamps"] is not None:
            timestamps = ele["timestamps"]
            original_fps = sample_fps # Store original FPS for frame calculation
            nframes_original = video.shape[0]

            select_frames = []

            try:
                for start_ts, end_ts in timestamps:
                    start_frame = int(torch.floor(torch.tensor(start_ts * original_fps)).item())
                    end_frame = int(torch.ceil(torch.tensor(end_ts * original_fps)).item())

                    # Ensure frame indices are within bounds
                    start_frame = max(0, start_frame)
                    end_frame = min(nframes_original, end_frame) # Use nframes_original

                    if start_frame < end_frame:
                        select_frames.extend(list(range(start_frame, end_frame)))
                    else:
                        logger.warning(f"Invalid timestamp range [{start_ts}, {end_ts}] resulting in start_frame {start_frame} >= end_frame {end_frame}. Skipping this segment.")
            except:
                select_frames = []
            
            if select_frames:
                select_frames = sorted(list(set(select_frames)))

                if len(select_frames) > 256:
                    frame_idx = torch.linspace(0, len(select_frames) - 1, 256).round().long().tolist()
                    select_frames = [select_frames[i] for i in frame_idx]
                
                video = video[select_frames]
                nframes = video.shape[0] # Update nframes after cropping
                logger.info(f"Video cropped to {nframes} frames based on timestamps.")
            else:
                logger.warning("No valid video segments found after applying timestamps. Returning original video.")

        if not resize:
            return video, sample_fps
        
        total_pixels = ele.get("total_pixels", VIDEO_TOTAL_PIXELS)
        min_pixels = ele.get("min_pixels", VIDEO_MIN_PIXELS)
        maximum_frames = int(total_pixels / (min_pixels * 1.05) * 2) if maximum_frames is None else maximum_frames
        if video.shape[0] > maximum_frames:
            frame_idx = torch.linspace(0, video.shape[0] - 1, maximum_frames).round().long().tolist()
            sample_fps = maximum_frames / video.shape[0] * sample_fps
            video = video[frame_idx]

        nframes, _, height, width = video.shape
        
        max_pixels = max(min(VIDEO_MAX_PIXELS, total_pixels / nframes * FRAME_FACTOR), int(min_pixels * 1.05))
        max_pixels_supposed = ele.get("max_pixels", max_pixels)
        if max_pixels_supposed > max_pixels:
            logger.warning(f"The given max_pixels[{max_pixels_supposed}] exceeds limit[{max_pixels}].")
        max_pixels = min(max_pixels_supposed, max_pixels)
        if "resized_height" in ele and "resized_width" in ele:
            resized_height, resized_width = smart_resize(
                ele["resized_height"],
                ele["resized_width"],
                factor=image_factor,
            )
        else:
            resized_height, resized_width = smart_resize(
                height,
                width,
                factor=image_factor,
                min_pixels=min_pixels,
                max_pixels=max_pixels,
            )
        video = transforms.functional.resize(
            video,
            [resized_height, resized_width],
            interpolation=InterpolationMode.BICUBIC,
            antialias=True,
        ).float()
        if return_video_sample_fps:
            return video, sample_fps
        return video
    else:
        assert isinstance(ele["video"], (list, tuple))
        process_info = ele.copy()
        process_info.pop("type", None)
        process_info.pop("video", None)
        images = [
            fetch_image({"image": video_element, **process_info}, size_factor=image_factor)
            for video_element in ele["video"]
        ]
        nframes = ceil_by_factor(len(images), FRAME_FACTOR)
        if len(images) < nframes:
            images.extend([images[-1]] * (nframes - len(images)))
        if return_video_sample_fps:
            return images, process_info.pop("fps", 2.0)
        return images

def extract_vision_info(conversations: list[dict] | list[list[dict]]) -> list[dict]:
    vision_infos = []
    if isinstance(conversations[0], dict):
        conversations = [conversations]
    for conversation in conversations:
        for message in conversation:
            if isinstance(message["content"], list):
                for ele in message["content"]:
                    if (
                        "image" in ele
                        or "image_url" in ele
                        or "video" in ele
                        or ele["type"] in ("image", "image_url", "video")
                    ):
                        vision_infos.append(ele)
    return vision_infos


def process_vision_info(
    conversations: list[dict] | list[list[dict]],
    return_video_kwargs: bool = False,
    client=None,
    maximum_frames=None
) -> tuple[list[Image.Image] | None, list[torch.Tensor | list[Image.Image]] | None, Optional[dict]]:

    vision_infos = extract_vision_info(conversations)
    ## Read images or videos
    image_inputs = []
    video_inputs = []
    video_sample_fps_list = []
    for vision_info in vision_infos:
        if "image" in vision_info or "image_url" in vision_info:
            image_inputs.append(fetch_image(vision_info))
        elif "video" in vision_info:
            video_input, video_sample_fps = fetch_video(vision_info, return_video_sample_fps=True, maximum_frames=maximum_frames)
            video_sample_fps_list.append(video_sample_fps)
            video_inputs.append(video_input)
        else:
            raise ValueError("image, image_url or video should in content.")
    if len(image_inputs) == 0:
        image_inputs = None
    if len(video_inputs) == 0:
        video_inputs = None
    if return_video_kwargs:
        return image_inputs, video_inputs, {'fps': video_sample_fps_list}
    return image_inputs, video_inputs

def read_subtitle(filename):
    """
    Reads an SRT file, extracts all subtitles with their timestamps,
    cleans the text by removing HTML tags and bracketed captions,
    and returns a dictionary of all subtitles.
    
    Args:
        filename (str): The path to the SRT file.
    
    Returns:
        dict: A dictionary where keys are the start times in seconds (float)
              and values are the cleaned subtitle texts (str).
    """
    subtitles = {}
    current_subtitle_text = []
    current_start_time = None

    def clean_text(text):
        """Removes HTML-like tags and bracketed text from a string."""
        # Remove HTML-like tags
        clean_html = re.compile('<.*?>')
        text = re.sub(clean_html, '', text)
        # Remove bracketed text like [Music]
        clean_brackets = re.compile(r'\[.*?\]')
        return re.sub(clean_brackets, '', text).strip()

    with open(filename, 'r', encoding='utf-8') as f:
        for line in f:
            line = line.strip()

            if not line:
                continue

            if '-->' in line:
                # Process the previous subtitle block
                if current_subtitle_text and current_start_time is not None:
                    subtitle_text = ' '.join(current_subtitle_text).strip()
                    cleaned_text = clean_text(subtitle_text)
                    if cleaned_text:
                        subtitles[current_start_time] = cleaned_text
                
                time_range = line.split('-->')
                start_time_str = time_range[0].strip()
                
                try:
                    h, m, s, _ = map(int, start_time_str.replace(',', ':').split(':'))
                    current_start_time = int(h) * 3600 + int(m) * 60 + int(s)
                except ValueError:
                    current_start_time = None
                
                current_subtitle_text = []
            elif not re.match(r'^\d+$', line) and current_start_time is not None:
                current_subtitle_text.append(line)

    # Process the last subtitle block
    if current_subtitle_text and current_start_time is not None:
        subtitle_text = ' '.join(current_subtitle_text).strip()
        cleaned_text = clean_text(subtitle_text)
        if cleaned_text:
            subtitles[current_start_time] = cleaned_text
    
    subtitles = group_subtitles_by_interval(subtitles, 5)

    return subtitles

def group_subtitles_by_interval(all_subtitles, interval_seconds=5):
    """
    Groups subtitles from a dictionary into time intervals and removes duplicates.
    
    Args:
        all_subtitles (dict): A dictionary of all subtitles with timestamps.
        interval_seconds (int): The time interval in seconds to group subtitles.
    
    Returns:
        dict: A dictionary where keys are the start times of each interval
              and values are the combined unique subtitle texts (str).
    """
    combined_subtitles = {}
    unique_combined_subtitles = set()
    
    if not all_subtitles:
        return {}

    sorted_times = sorted(all_subtitles.keys())
    
    current_group_start_time = (sorted_times[0] // interval_seconds) * interval_seconds
    current_group_text = []

    for time in sorted_times:
        if time < current_group_start_time + interval_seconds:
            if all_subtitles[time]:
                current_group_text.append(all_subtitles[time])
        else:
            if current_group_text:
                combined_text = ' '.join(current_group_text).strip()
                if combined_text and combined_text not in unique_combined_subtitles:
                    combined_subtitles[current_group_start_time] = combined_text
                    unique_combined_subtitles.add(combined_text)

            current_group_start_time = (time // interval_seconds) * interval_seconds
            current_group_text = []
            if all_subtitles[time]:
                current_group_text.append(all_subtitles[time])
    
    if current_group_text:
        combined_text = ' '.join(current_group_text).strip()
        if combined_text and combined_text not in unique_combined_subtitles:
            combined_subtitles[current_group_start_time] = combined_text

    return dict(sorted(combined_subtitles.items()))

def confidence_score(sequence_scores, generated_tokens_seq, processor, topk=20):
    probs = torch.softmax(sequence_scores, dim=-1)
    topk_probs, _ = torch.topk(probs, topk, dim=-1)
    token_confidence = -torch.log(topk_probs).mean(dim=-1)
    answer_token_index = -1
    answer_token_id = None
    glue_start_index = -1
    end_token_index = -1
    for i, tok in enumerate(generated_tokens_seq):
        if generated_tokens_seq[i] == 27 and generated_tokens_seq[i+1] == 9217:
            answer_token_index = i + 2
            answer_token_id = generated_tokens_seq[i+2]
        if generated_tokens_seq[i] == 29  and generated_tokens_seq[i+1] == 9697:
            glue_start_index = i + 2
        if generated_tokens_seq[i] == 151645:
            end_token_index = i
            break
        if len(generated_tokens_seq) == i + 2:
            break
    answer_confidence, glue_confidence = 0.0, 0.0
    if answer_token_index != -1:
        if answer_token_id is not None:
            answer_confidence = probs[answer_token_index][answer_token_id].item()
        else:
            answer_confidence = 0.0
    if glue_start_index != -1:
        glue_end_index = end_token_index - 5
        glue_confidence = token_confidence[glue_start_index : glue_end_index].mean().item()
    return answer_confidence, glue_confidence