Gemini / func.py
Moonfanz's picture
Upload 6 files
8edba3e verified
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