# Copyright 2025 ZTE Corporation.
# All Rights Reserved.
#
#    Licensed under the Apache License, Version 2.0 (the "License"); you may
#    not use this file except in compliance with the License. You may obtain
#    a copy of the License at
#
#         http://www.apache.org/licenses/LICENSE-2.0
#
#    Unless required by applicable law or agreed to in writing, software
#    distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
#    WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
#    License for the specific language governing permissions and limitations
#    under the License.

import os
import base64
import asyncio
from urllib.parse import urlparse
import requests
from tqdm import tqdm
from openai import OpenAI
from configs import logger


def get_filename_from_url(url):
    """Extracts the filename from a given URL."""
    parsed_url = urlparse(url)
    return os.path.basename(parsed_url.path)

def download_file(url, dest_path):
    """Downloads a file from a URL to a destination path with a progress bar."""
    chunk_size = 1024
    if os.path.exists(dest_path):
        logger.warning(f"Warning: File already exists, overwriting: {dest_path}")
        os.remove(dest_path)

    with requests.get(url, stream=True) as response:
        response.raise_for_status()
        total_size = int(response.headers.get('Content-Length', 0))

        with open(dest_path, 'wb') as f, tqdm(
                total=total_size,
                unit='B',
                unit_scale=True,
                unit_divisor=1024,
                desc=os.path.basename(dest_path)
        ) as bar:
            for chunk in response.iter_content(chunk_size=chunk_size):
                if chunk:
                    f.write(chunk)
                    bar.update(len(chunk))

    logger.info(f"Download complete: {dest_path}")
    return dest_path

def is_youtube_url(url):
    """Checks if a URL is a YouTube URL. (Placeholder implementation)"""
    return "youtube.com" in url or "youtu.be" in url

def download_youtube_audio_and_subs(url, output_dir):
    """
    Downloads video from YouTube. (Placeholder implementation)
    This function should be implemented to handle YouTube downloads.
    """
    print(f"Downloading YouTube video from: {url} to {output_dir}")
    # You would use a library like pytube or yt-dlp here.
    # For this example, we'll just simulate a downloaded file path.
    file_name = f"{get_filename_from_url(url)}.mp4"
    return os.path.join(output_dir, file_name)


class MediaAnalysisTool:
    """
    A unified tool to analyze images and videos using OpenAI's multimodal APIs.
    """
    def __init__(self, client, model_name, stream=False, WORKSPACE_PATH=None):
        self.WORKSPACE_PATH = WORKSPACE_PATH or os.getcwd()
        self.client = client
        self.model_name = model_name
        self.stream = stream

    name: str = "Media Analysis Tool"
    description: str = (
        "This tool uses OpenAI's multimodal APIs to describe the contents of an image or a video."
    )

    def _encode_media(self, file_path: str) -> str:
        """Encodes a media file (image or video) into a Base64 string."""
        with open(file_path, "rb") as media_file:
            return base64.b64encode(media_file.read()).decode("utf-8")

    def _prepare_media_data_url(self, media_path: str, media_type: str) -> str:
        """
        Handles local files or downloads from a URL, then returns a Base64 data URL.
        """
        if media_path.startswith('http://') or media_path.startswith('https://'):
            workspace_path = self.WORKSPACE_PATH
            absolute_path = ''
            
            if media_type == 'video' and is_youtube_url(media_path):
                # Specific handling for YouTube URLs
                absolute_path = download_youtube_audio_and_subs(url=media_path, output_dir=workspace_path)
            else:
                # Standard file download
                file_name = get_filename_from_url(media_path)
                absolute_path = os.path.join(workspace_path, file_name)
                download_file(media_path, absolute_path)
        else:
            # Path is a local file
            absolute_path = media_path

        base64_media = self._encode_media(absolute_path)

        if media_type == 'image':
            # Format for images (e.g., image/png, image/jpeg)
            return f"data:image/png;base64,{base64_media}"
        elif media_type == 'video':
            # Format for videos (MIME type is often omitted in this context)
            return f"data:;base64,{base64_media}"
        else:
            raise ValueError("Unsupported media_type specified.")

    async def _analyze_media(self, media_path: str, prompt: str, media_type: str) -> str:
        """
        Core asynchronous method to send a media analysis request to the OpenAI API.
        """
        media_data_url = self._prepare_media_data_url(media_path, media_type)
        
        # Dynamically create the content payload based on media type
        content_key = f"{media_type}_url"
        user_content = [
            {
                "type": content_key,
                content_key: {"url": media_data_url},
            },
            {"type": "text", "text": prompt},
        ]

        completion = self.client.chat.completions.create(
            model=self.model_name,
            messages=[{
                "role": "user",
                "content": user_content,
            }],
            stream=self.stream
        )

        return completion.choices[0].message.content


    def ask_question_about_image(self, image_path_url: str, task_prompt: str) -> str:
        """
        Public method to analyze an image.
        
        Args:
            image_path_url: Path or URL to the image file.
            task_prompt: The question to ask about the image.
        
        Returns:
            The text response from the model.
        """
        print(f"Using Tool: {self.name} for Image Analysis")
        return asyncio.run(self._analyze_media(image_path_url, task_prompt, 'image'))

    def ask_question_about_video(self, video_path_url: str, question: str) -> str:
        """
        Public method to analyze a video.
        
        Args:
            video_path_url: Path or URL to the video file.
            question: The question to ask about the video.
        
        Returns:
            The text response from the model.
        """
        print(f"Using Tool: {self.name} for Video Analysis")
        return asyncio.run(self._analyze_media(video_path_url, question, 'video'))