github-actions[bot]
Update from GitHub Actions
c5eab62
package apiserver
import (
"monica-proxy/internal/middleware"
"monica-proxy/internal/monica"
"monica-proxy/internal/types"
"net/http"
"github.com/labstack/echo/v4"
"github.com/sashabaranov/go-openai"
)
// RegisterRoutes 注册 Echo 路由
func RegisterRoutes(e *echo.Echo) {
// 添加Bearer Token认证中间件
e.Use(middleware.BearerAuth())
// ChatGPT 风格的请求转发到 /v1/chat/completions
e.POST("/v1/chat/completions", handleChatCompletion)
// 获取支持的模型列表
e.GET("/v1/models", handleListModels)
}
// handleChatCompletion 接收 ChatGPT 形式的对话请求并转发给 Monica
func handleChatCompletion(c echo.Context) error {
var req openai.ChatCompletionRequest
if err := c.Bind(&req); err != nil {
return c.JSON(http.StatusBadRequest, map[string]interface{}{
"error": "Invalid request payload",
})
}
// 检查请求是否包含消息
if len(req.Messages) == 0 {
return c.JSON(http.StatusBadRequest, map[string]interface{}{
"error": "No messages found",
})
}
// marshalIndent, err := json.MarshalIndent(req, "", " ")
// if err != nil {
// return err
// }
// log.Printf("Received completion request: \n%s\n", marshalIndent)
// 将 ChatGPTRequest 转换为 MonicaRequest
monicaReq, err := types.ChatGPTToMonica(req)
if err != nil {
return c.JSON(http.StatusInternalServerError, map[string]interface{}{
"error": err.Error(),
})
}
// 调用 Monica 并获取 SSE Stream
stream, err := monica.SendMonicaRequest(c.Request().Context(), monicaReq)
if err != nil {
return c.JSON(http.StatusInternalServerError, map[string]interface{}{
"error": err.Error(),
})
}
// Resty 不会自动关闭 Body,需要我们自己来处理
defer stream.RawBody().Close()
// 根据 stream 参数决定是否使用流式响应
if req.Stream {
// 使用流式响应
c.Response().Header().Set(echo.HeaderContentType, "text/event-stream")
c.Response().Header().Set("Cache-Control", "no-cache")
c.Response().Header().Set("Transfer-Encoding", "chunked")
c.Response().WriteHeader(http.StatusOK)
// 将 Monica 的 SSE 数据逐行读出,再以 SSE 格式返回给调用方
if err := monica.StreamMonicaSSEToClient(req.Model, c.Response().Writer, stream.RawBody()); err != nil {
return err
}
} else {
// 使用非流式响应
c.Response().Header().Set(echo.HeaderContentType, "application/json")
// 收集所有的 SSE 数据并转换为完整的响应
response, err := monica.CollectMonicaSSEToCompletion(req.Model, stream.RawBody())
if err != nil {
return c.JSON(http.StatusInternalServerError, map[string]interface{}{
"error": err.Error(),
})
}
// 返回完整的响应
return c.JSON(http.StatusOK, response)
}
return nil
}
// handleListModels 返回支持的模型列表
func handleListModels(c echo.Context) error {
models := types.GetSupportedModels()
return c.JSON(http.StatusOK, models)
}