"""OpenAI 请求-响应模型定义""" from fastapi import HTTPException from typing import Optional, List, Union, Dict, Any from pydantic import BaseModel, Field, field_validator from app.models.grok_models import Models class OpenAIChatRequest(BaseModel): """OpenAI聊天请求""" model: str = Field(..., description="模型名称", min_length=1) messages: List[Dict[str, Any]] = Field(..., description="消息列表", min_length=1) stream: bool = Field(False, description="流式响应") temperature: Optional[float] = Field(0.7, ge=0, le=2, description="采样温度") max_tokens: Optional[int] = Field(None, ge=1, le=100000, description="最大Token数") top_p: Optional[float] = Field(1.0, ge=0, le=1, description="采样参数") @classmethod @field_validator('messages') def validate_messages(cls, v): """验证消息格式""" if not v: raise HTTPException(status_code=400, detail="消息列表不能为空") for msg in v: if not isinstance(msg, dict): raise HTTPException(status_code=400, detail="每个消息必须是字典") if 'role' not in msg: raise HTTPException(status_code=400, detail="消息缺少 'role' 字段") if 'content' not in msg: raise HTTPException(status_code=400, detail="消息缺少 'content' 字段") if msg['role'] not in ['system', 'user', 'assistant']: raise HTTPException( status_code=400, detail=f"无效角色 '{msg['role']}', 必须是 system/user/assistant" ) return v @classmethod @field_validator('model') def validate_model(cls, v): """验证模型名称""" if not Models.is_valid_model(v): supported = Models.get_all_model_names() raise HTTPException( status_code=400, detail=f"不支持的模型 '{v}', 支持: {', '.join(supported)}" ) return v class OpenAIChatCompletionMessage(BaseModel): """聊天完成消息""" role: str = Field(..., description="角色") content: str = Field(..., description="内容") reference_id: Optional[str] = Field(default=None, description="参考ID") annotations: Optional[List[str]] = Field(default=None, description="注释") class OpenAIChatCompletionChoice(BaseModel): """聊天完成选项""" index: int = Field(..., description="索引") message: OpenAIChatCompletionMessage = Field(..., description="消息") logprobs: Optional[float] = Field(default=None, description="对数概率") finish_reason: str = Field(default="stop", description="完成原因") class OpenAIChatCompletionResponse(BaseModel): """聊天完成响应""" id: str = Field(..., description="响应ID") object: str = Field("chat.completion", description="对象类型") created: int = Field(..., description="创建时间戳") model: str = Field(..., description="模型") choices: List[OpenAIChatCompletionChoice] = Field(..., description="选项") usage: Optional[Dict[str, Any]] = Field(None, description="令牌使用") class OpenAIChatCompletionChunkMessage(BaseModel): """流式消息片段""" role: str = Field(..., description="角色") content: str = Field(..., description="内容") class OpenAIChatCompletionChunkChoice(BaseModel): """流式选项""" index: int = Field(..., description="索引") delta: Optional[Union[Dict[str, Any], OpenAIChatCompletionChunkMessage]] = Field( None, description="Delta数据" ) finish_reason: Optional[str] = Field(None, description="完成原因") class OpenAIChatCompletionChunkResponse(BaseModel): """流式聊天响应""" id: str = Field(..., description="响应ID") object: str = Field(default="chat.completion.chunk", description="对象类型") created: int = Field(..., description="创建时间戳") model: str = Field(..., description="模型") system_fingerprint: Optional[str] = Field(default=None, description="系统指纹") choices: List[OpenAIChatCompletionChunkChoice] = Field(..., description="选项")