mirror of
https://github.com/QuantumNous/new-api.git
synced 2026-03-30 09:35:21 +00:00
* feat: openai response /v1/response/compact * feat: /v1/response/compact bill * feat: /v1/response/compact * feat: /v1/responses/compact -> codex channel * feat: /v1/responses/compact -> codex channel * feat: /v1/responses/compact -> codex channel * feat: codex channel default models * feat: compact model price * feat: /v1/responses/comapct test
177 lines
5.6 KiB
Go
177 lines
5.6 KiB
Go
package codex
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
|
|
"github.com/QuantumNous/new-api/common"
|
|
"github.com/QuantumNous/new-api/dto"
|
|
"github.com/QuantumNous/new-api/relay/channel"
|
|
"github.com/QuantumNous/new-api/relay/channel/openai"
|
|
relaycommon "github.com/QuantumNous/new-api/relay/common"
|
|
relayconstant "github.com/QuantumNous/new-api/relay/constant"
|
|
"github.com/QuantumNous/new-api/types"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
)
|
|
|
|
type Adaptor struct {
|
|
}
|
|
|
|
func (a *Adaptor) ConvertGeminiRequest(c *gin.Context, info *relaycommon.RelayInfo, request *dto.GeminiChatRequest) (any, error) {
|
|
return nil, errors.New("codex channel: endpoint not supported")
|
|
}
|
|
|
|
func (a *Adaptor) ConvertClaudeRequest(*gin.Context, *relaycommon.RelayInfo, *dto.ClaudeRequest) (any, error) {
|
|
return nil, errors.New("codex channel: endpoint not supported")
|
|
}
|
|
|
|
func (a *Adaptor) ConvertAudioRequest(c *gin.Context, info *relaycommon.RelayInfo, request dto.AudioRequest) (io.Reader, error) {
|
|
return nil, errors.New("codex channel: endpoint not supported")
|
|
}
|
|
|
|
func (a *Adaptor) ConvertImageRequest(c *gin.Context, info *relaycommon.RelayInfo, request dto.ImageRequest) (any, error) {
|
|
return nil, errors.New("codex channel: endpoint not supported")
|
|
}
|
|
|
|
func (a *Adaptor) Init(info *relaycommon.RelayInfo) {
|
|
}
|
|
|
|
func (a *Adaptor) ConvertOpenAIRequest(c *gin.Context, info *relaycommon.RelayInfo, request *dto.GeneralOpenAIRequest) (any, error) {
|
|
return nil, errors.New("codex channel: endpoint not supported")
|
|
}
|
|
|
|
func (a *Adaptor) ConvertRerankRequest(c *gin.Context, relayMode int, request dto.RerankRequest) (any, error) {
|
|
return nil, errors.New("codex channel: endpoint not supported")
|
|
}
|
|
|
|
func (a *Adaptor) ConvertEmbeddingRequest(c *gin.Context, info *relaycommon.RelayInfo, request dto.EmbeddingRequest) (any, error) {
|
|
return nil, errors.New("codex channel: endpoint not supported")
|
|
}
|
|
|
|
func (a *Adaptor) ConvertOpenAIResponsesRequest(c *gin.Context, info *relaycommon.RelayInfo, request dto.OpenAIResponsesRequest) (any, error) {
|
|
isCompact := info != nil && info.RelayMode == relayconstant.RelayModeResponsesCompact
|
|
|
|
if info != nil && info.ChannelSetting.SystemPrompt != "" {
|
|
systemPrompt := info.ChannelSetting.SystemPrompt
|
|
|
|
if len(request.Instructions) == 0 {
|
|
if b, err := common.Marshal(systemPrompt); err == nil {
|
|
request.Instructions = b
|
|
} else {
|
|
return nil, err
|
|
}
|
|
} else if info.ChannelSetting.SystemPromptOverride {
|
|
var existing string
|
|
if err := common.Unmarshal(request.Instructions, &existing); err == nil {
|
|
existing = strings.TrimSpace(existing)
|
|
if existing == "" {
|
|
if b, err := common.Marshal(systemPrompt); err == nil {
|
|
request.Instructions = b
|
|
} else {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
if b, err := common.Marshal(systemPrompt + "\n" + existing); err == nil {
|
|
request.Instructions = b
|
|
} else {
|
|
return nil, err
|
|
}
|
|
}
|
|
} else {
|
|
if b, err := common.Marshal(systemPrompt); err == nil {
|
|
request.Instructions = b
|
|
} else {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if isCompact {
|
|
return request, nil
|
|
}
|
|
// codex: store must be false
|
|
request.Store = json.RawMessage("false")
|
|
// rm max_output_tokens
|
|
request.MaxOutputTokens = 0
|
|
request.Temperature = nil
|
|
return request, nil
|
|
}
|
|
|
|
func (a *Adaptor) DoRequest(c *gin.Context, info *relaycommon.RelayInfo, requestBody io.Reader) (any, error) {
|
|
return channel.DoApiRequest(a, c, info, requestBody)
|
|
}
|
|
|
|
func (a *Adaptor) DoResponse(c *gin.Context, resp *http.Response, info *relaycommon.RelayInfo) (usage any, err *types.NewAPIError) {
|
|
if info.RelayMode != relayconstant.RelayModeResponses && info.RelayMode != relayconstant.RelayModeResponsesCompact {
|
|
return nil, types.NewError(errors.New("codex channel: endpoint not supported"), types.ErrorCodeInvalidRequest)
|
|
}
|
|
|
|
if info.RelayMode == relayconstant.RelayModeResponsesCompact {
|
|
return openai.OaiResponsesCompactionHandler(c, resp)
|
|
}
|
|
|
|
if info.IsStream {
|
|
return openai.OaiResponsesStreamHandler(c, info, resp)
|
|
}
|
|
return openai.OaiResponsesHandler(c, info, resp)
|
|
}
|
|
|
|
func (a *Adaptor) GetModelList() []string {
|
|
return ModelList
|
|
}
|
|
|
|
func (a *Adaptor) GetChannelName() string {
|
|
return ChannelName
|
|
}
|
|
|
|
func (a *Adaptor) GetRequestURL(info *relaycommon.RelayInfo) (string, error) {
|
|
if info.RelayMode != relayconstant.RelayModeResponses && info.RelayMode != relayconstant.RelayModeResponsesCompact {
|
|
return "", errors.New("codex channel: only /v1/responses and /v1/responses/compact are supported")
|
|
}
|
|
path := "/backend-api/codex/responses"
|
|
if info.RelayMode == relayconstant.RelayModeResponsesCompact {
|
|
path = "/backend-api/codex/responses/compact"
|
|
}
|
|
return relaycommon.GetFullRequestURL(info.ChannelBaseUrl, path, info.ChannelType), nil
|
|
}
|
|
|
|
func (a *Adaptor) SetupRequestHeader(c *gin.Context, req *http.Header, info *relaycommon.RelayInfo) error {
|
|
channel.SetupApiRequestHeader(info, c, req)
|
|
|
|
key := strings.TrimSpace(info.ApiKey)
|
|
if !strings.HasPrefix(key, "{") {
|
|
return errors.New("codex channel: key must be a JSON object")
|
|
}
|
|
|
|
oauthKey, err := ParseOAuthKey(key)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
accessToken := strings.TrimSpace(oauthKey.AccessToken)
|
|
accountID := strings.TrimSpace(oauthKey.AccountID)
|
|
|
|
if accessToken == "" {
|
|
return errors.New("codex channel: access_token is required")
|
|
}
|
|
if accountID == "" {
|
|
return errors.New("codex channel: account_id is required")
|
|
}
|
|
|
|
req.Set("Authorization", "Bearer "+accessToken)
|
|
req.Set("chatgpt-account-id", accountID)
|
|
|
|
if req.Get("OpenAI-Beta") == "" {
|
|
req.Set("OpenAI-Beta", "responses=experimental")
|
|
}
|
|
if req.Get("originator") == "" {
|
|
req.Set("originator", "codex_cli_rs")
|
|
}
|
|
|
|
return nil
|
|
}
|