from io import BytesIO
import os
from flask import jsonify
import logging
import json
import re
logger = logging.getLogger(__name__)

ProxyPasswords = os.environ.get('ProxyPasswords').split(',')

def authenticate_request(request):
    auth_header = request.headers.get('Authorization')

    if not auth_header:
        return False, jsonify({'error': 'Authorization header is missing'}), 401
    
    try:
        auth_type, api_key = auth_header.split(' ', 1)
    except ValueError:
        return False, jsonify({'error': 'Invalid Authorization header format'}), 401

    if auth_type.lower() != 'bearer':
        return False, jsonify({'error': 'Authorization type must be Bearer'}), 401

    if api_key not in ProxyPasswords:
        return False, jsonify({'error': 'Unauthorized'}), 401

    return True, None, None

def sanitize_request_data(request_data):
    """
    从请求数据中删除base64编码的数据。

    Args:
        request_data: 包含可能存在base64数据的字典。

    Returns:
        清理后的字典,其中base64数据被替换为"[Base64 Data Omitted]"。
    """
    
    
    def replace_base64(match):
        # 替换base64数据为提示信息
        return '"[Base64 Data Omitted]"'

    
    request_data_str = json.dumps(request_data)

    # 使用正则表达式匹配base64数据,并替换为提示信息
    sanitized_request_data_str = re.sub(
        r'"(data:[^;]+;base64,)[^"]+"',
        replace_base64,
        request_data_str
    )

    return json.loads(sanitized_request_data_str)

def process_messages_for_gemini(messages):
    """
    将通用的对话消息格式转换为Gemini API所需的格式
    这个函数处理消息列表并将其转换为Gemini API兼容的格式。它支持文本、图片和文件内容的处理。
    参数:
        messages (list): 包含对话消息的列表。每条消息应该是一个字典,包含'role'和'content'字段。
            - role: 可以是 'system', 'user' 或 'assistant'
            - content: 可以是字符串或包含多个内容项的列表
    返回:
        tuple: 包含三个元素:
            - gemini_history (list): 转换后的历史消息列表
            - user_message (dict): 最新的用户消息
            - error (tuple or None): 如果有错误,返回错误响应;否则返回None
    错误处理:
        - 检查角色是否有效
        - 验证图片URL格式
        - 验证文件URL格式
    示例消息格式:
        文本消息:
        {
            'role': 'user',
            'content': '你好'
        }
        多模态消息:
        {
            'role': 'user',
            'content': [
                {'type': 'text', 'text': '这是什么图片?'},
                {'type': 'image_url', 'image_url': {'url': 'data:image/jpeg;base64,...'}}
            ]
        }
    """
    gemini_history = []
    errors = []
    for message in messages:
        role = message.get('role')
        content = message.get('content')

        if isinstance(content, str):
            if role == 'system':
                gemini_history.append({"role": "user", "parts": [content]})
            elif role == 'user':
                gemini_history.append({"role": "user", "parts": [content]})
            elif role == 'assistant':
                gemini_history.append({"role": "model", "parts": [content]})
            else:
                errors.append(f"Invalid role: {role}")
        elif isinstance(content, list):
            parts = []
            for item in content:
                if item.get('type') == 'text':
                    parts.append({"text": item.get('text')})  
                elif item.get('type') == 'image_url':
                    image_data = item.get('image_url', {}).get('url', '')
                    if image_data.startswith('data:image/'):

                        try:
                            mime_type, base64_data = image_data.split(';')[0].split(':')[1], image_data.split(',')[1]
                            parts.append({
                                "inline_data": {
                                    "mime_type": mime_type,
                                    "data": base64_data
                                }
                            })
                        except (IndexError, ValueError):
                            errors.append(f"Invalid data URI for image: {image_data}")
                    else:
                        errors.append(f"Invalid image URL format for item: {item}")
                elif item.get('type') == 'file_url':
                    file_data = item.get('file_url', {}).get('url', '')
                    if file_data.startswith('data:'):

                        try:
                            mime_type, base64_data = file_data.split(';')[0].split(':')[1], file_data.split(',')[1]
                            parts.append({
                                "inline_data": {
                                    "mime_type": mime_type,
                                    "data": base64_data
                                }
                            })
                        except (IndexError, ValueError):
                            errors.append(f"Invalid data URI for file: {file_data}")
                    else:
                        errors.append(f"Invalid file URL format for item: {item}")

            if parts: 
                if role in ['user', 'system']:
                    gemini_history.append({"role": "user", "parts": parts})
                elif role == 'assistant':
                    gemini_history.append({"role": "model", "parts": parts})
                else:
                    errors.append(f"Invalid role: {role}")

    if gemini_history:
        user_message = gemini_history[-1]
        gemini_history = gemini_history[:-1]
    else:
        user_message = {"role": "user", "parts": [""]}

    if errors:
        return gemini_history, user_message, (jsonify({'error': errors}), 400)
    else:
        return gemini_history, user_message, None