adaptor.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. package aws
  2. import (
  3. "io"
  4. "net/http"
  5. "github.com/QuantumNous/new-api/common"
  6. "github.com/QuantumNous/new-api/dto"
  7. "github.com/QuantumNous/new-api/relay/channel/claude"
  8. relaycommon "github.com/QuantumNous/new-api/relay/common"
  9. "github.com/QuantumNous/new-api/types"
  10. "github.com/aws/aws-sdk-go-v2/aws"
  11. "github.com/aws/aws-sdk-go-v2/service/bedrockruntime"
  12. "github.com/pkg/errors"
  13. "github.com/gin-gonic/gin"
  14. )
  15. const (
  16. RequestModeCompletion = 1
  17. RequestModeMessage = 2
  18. )
  19. type Adaptor struct {
  20. AwsClient *bedrockruntime.Client
  21. AwsModelId string
  22. AwsReq any
  23. IsNova bool
  24. }
  25. func (a *Adaptor) ConvertGeminiRequest(*gin.Context, *relaycommon.RelayInfo, *dto.GeminiChatRequest) (any, error) {
  26. //TODO implement me
  27. return nil, errors.New("not implemented")
  28. }
  29. func (a *Adaptor) ConvertClaudeRequest(c *gin.Context, info *relaycommon.RelayInfo, request *dto.ClaudeRequest) (any, error) {
  30. return request, nil
  31. }
  32. func (a *Adaptor) ConvertAudioRequest(c *gin.Context, info *relaycommon.RelayInfo, request dto.AudioRequest) (io.Reader, error) {
  33. //TODO implement me
  34. return nil, errors.New("not implemented")
  35. }
  36. func (a *Adaptor) ConvertImageRequest(c *gin.Context, info *relaycommon.RelayInfo, request dto.ImageRequest) (any, error) {
  37. //TODO implement me
  38. return nil, errors.New("not implemented")
  39. }
  40. func (a *Adaptor) Init(info *relaycommon.RelayInfo) {
  41. }
  42. func (a *Adaptor) GetRequestURL(info *relaycommon.RelayInfo) (string, error) {
  43. return "", nil
  44. }
  45. func (a *Adaptor) SetupRequestHeader(c *gin.Context, req *http.Header, info *relaycommon.RelayInfo) error {
  46. claude.CommonClaudeHeadersOperation(c, req, info)
  47. return nil
  48. }
  49. func (a *Adaptor) ConvertOpenAIRequest(c *gin.Context, info *relaycommon.RelayInfo, request *dto.GeneralOpenAIRequest) (any, error) {
  50. if request == nil {
  51. return nil, errors.New("request is nil")
  52. }
  53. // 检查是否为Nova模型
  54. if isNovaModel(request.Model) {
  55. novaReq := convertToNovaRequest(request)
  56. a.IsNova = true
  57. return novaReq, nil
  58. }
  59. // 原有的Claude模型处理逻辑
  60. var claudeReq *dto.ClaudeRequest
  61. var err error
  62. claudeReq, err = claude.RequestOpenAI2ClaudeMessage(c, *request)
  63. if err != nil {
  64. return nil, err
  65. }
  66. return claudeReq, err
  67. }
  68. func (a *Adaptor) ConvertRerankRequest(c *gin.Context, relayMode int, request dto.RerankRequest) (any, error) {
  69. return nil, nil
  70. }
  71. func (a *Adaptor) ConvertEmbeddingRequest(c *gin.Context, info *relaycommon.RelayInfo, request dto.EmbeddingRequest) (any, error) {
  72. //TODO implement me
  73. return nil, errors.New("not implemented")
  74. }
  75. func (a *Adaptor) ConvertOpenAIResponsesRequest(c *gin.Context, info *relaycommon.RelayInfo, request dto.OpenAIResponsesRequest) (any, error) {
  76. // TODO implement me
  77. return nil, errors.New("not implemented")
  78. }
  79. func (a *Adaptor) DoRequest(c *gin.Context, info *relaycommon.RelayInfo, requestBody io.Reader) (any, error) {
  80. awsCli, err := newAwsClient(c, info)
  81. if err != nil {
  82. return nil, types.NewError(err, types.ErrorCodeChannelAwsClientError)
  83. }
  84. a.AwsClient = awsCli
  85. awsModelId := awsModelID(info.UpstreamModelName)
  86. awsRegionPrefix := awsRegionPrefix(awsCli.Options().Region)
  87. canCrossRegion := awsModelCanCrossRegion(awsModelId, awsRegionPrefix)
  88. if canCrossRegion {
  89. awsModelId = awsModelCrossRegion(awsModelId, awsRegionPrefix)
  90. }
  91. if isNovaModel(awsModelId) {
  92. var novaReq *NovaRequest
  93. err = common.DecodeJson(requestBody, &novaReq)
  94. if err != nil {
  95. return nil, types.NewError(errors.Wrap(err, "decode nova request fail"), types.ErrorCodeBadRequestBody)
  96. }
  97. // 使用InvokeModel API,但使用Nova格式的请求体
  98. awsReq := &bedrockruntime.InvokeModelInput{
  99. ModelId: aws.String(awsModelId),
  100. Accept: aws.String("application/json"),
  101. ContentType: aws.String("application/json"),
  102. }
  103. reqBody, err := common.Marshal(novaReq)
  104. if err != nil {
  105. return nil, types.NewError(errors.Wrap(err, "marshal nova request"), types.ErrorCodeBadResponseBody)
  106. }
  107. awsReq.Body = reqBody
  108. return nil, nil
  109. } else {
  110. awsClaudeReq, err := formatRequest(requestBody)
  111. if err != nil {
  112. return nil, types.NewError(errors.Wrap(err, "format aws request fail"), types.ErrorCodeBadRequestBody)
  113. }
  114. if info.IsStream {
  115. awsReq := &bedrockruntime.InvokeModelWithResponseStreamInput{
  116. ModelId: aws.String(awsModelId),
  117. Accept: aws.String("application/json"),
  118. ContentType: aws.String("application/json"),
  119. }
  120. awsReq.Body, err = common.Marshal(awsClaudeReq)
  121. if err != nil {
  122. return nil, types.NewError(errors.Wrap(err, "marshal aws request fail"), types.ErrorCodeBadRequestBody)
  123. }
  124. a.AwsReq = awsReq
  125. return nil, nil
  126. } else {
  127. awsReq := &bedrockruntime.InvokeModelInput{
  128. ModelId: aws.String(awsModelId),
  129. Accept: aws.String("application/json"),
  130. ContentType: aws.String("application/json"),
  131. }
  132. awsReq.Body, err = common.Marshal(awsClaudeReq)
  133. if err != nil {
  134. return nil, types.NewError(errors.Wrap(err, "marshal aws request fail"), types.ErrorCodeBadRequestBody)
  135. }
  136. a.AwsReq = awsReq
  137. return nil, nil
  138. }
  139. }
  140. }
  141. func (a *Adaptor) DoResponse(c *gin.Context, resp *http.Response, info *relaycommon.RelayInfo) (usage any, err *types.NewAPIError) {
  142. if a.IsNova {
  143. err, usage = handleNovaRequest(c, info, a)
  144. } else {
  145. if info.IsStream {
  146. err, usage = awsStreamHandler(c, info, a)
  147. } else {
  148. err, usage = awsHandler(c, info, a)
  149. }
  150. }
  151. return
  152. }
  153. func (a *Adaptor) GetModelList() (models []string) {
  154. for n := range awsModelIDMap {
  155. models = append(models, n)
  156. }
  157. return
  158. }
  159. func (a *Adaptor) GetChannelName() string {
  160. return ChannelName
  161. }