2026-02-25 10:15:38 +08:00
|
|
|
|
package service
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"bytes"
|
|
|
|
|
|
"context"
|
|
|
|
|
|
"encoding/json"
|
|
|
|
|
|
"errors"
|
|
|
|
|
|
"fmt"
|
|
|
|
|
|
"io"
|
|
|
|
|
|
"log"
|
|
|
|
|
|
"net/http"
|
|
|
|
|
|
"strings"
|
|
|
|
|
|
"sync"
|
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
|
|
"github.com/DouDOU-start/go-sora2api/sora"
|
|
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/config"
|
2026-02-28 15:01:20 +08:00
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/pkg/logger"
|
2026-02-25 10:15:38 +08:00
|
|
|
|
openaioauth "github.com/Wei-Shaw/sub2api/internal/pkg/openai"
|
|
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/util/logredact"
|
|
|
|
|
|
"github.com/tidwall/gjson"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
// SoraSDKClient 基于 go-sora2api SDK 的 Sora 客户端实现。
|
|
|
|
|
|
// 它实现了 SoraClient 接口,用 SDK 替代原有的自建 HTTP/PoW/TLS 指纹逻辑。
|
|
|
|
|
|
type SoraSDKClient struct {
|
|
|
|
|
|
cfg *config.Config
|
|
|
|
|
|
httpUpstream HTTPUpstream
|
|
|
|
|
|
tokenProvider *OpenAITokenProvider
|
|
|
|
|
|
accountRepo AccountRepository
|
|
|
|
|
|
soraAccountRepo SoraAccountRepository
|
|
|
|
|
|
|
|
|
|
|
|
// 每个 proxyURL 对应一个 SDK 客户端实例
|
|
|
|
|
|
sdkClients sync.Map // key: proxyURL (string), value: *sora.Client
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// NewSoraSDKClient 创建基于 SDK 的 Sora 客户端
|
|
|
|
|
|
func NewSoraSDKClient(cfg *config.Config, httpUpstream HTTPUpstream, tokenProvider *OpenAITokenProvider) *SoraSDKClient {
|
|
|
|
|
|
return &SoraSDKClient{
|
|
|
|
|
|
cfg: cfg,
|
|
|
|
|
|
httpUpstream: httpUpstream,
|
|
|
|
|
|
tokenProvider: tokenProvider,
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// SetAccountRepositories 设置账号和 Sora 扩展仓库(用于 token 持久化)
|
|
|
|
|
|
func (c *SoraSDKClient) SetAccountRepositories(accountRepo AccountRepository, soraAccountRepo SoraAccountRepository) {
|
|
|
|
|
|
if c == nil {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
c.accountRepo = accountRepo
|
|
|
|
|
|
c.soraAccountRepo = soraAccountRepo
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Enabled 判断是否启用 Sora
|
|
|
|
|
|
func (c *SoraSDKClient) Enabled() bool {
|
|
|
|
|
|
if c == nil || c.cfg == nil {
|
|
|
|
|
|
return false
|
|
|
|
|
|
}
|
|
|
|
|
|
return strings.TrimSpace(c.cfg.Sora.Client.BaseURL) != ""
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// PreflightCheck 在创建任务前执行账号能力预检。
|
|
|
|
|
|
// 当前仅对视频模型执行预检,用于提前识别额度耗尽或能力缺失。
|
|
|
|
|
|
func (c *SoraSDKClient) PreflightCheck(ctx context.Context, account *Account, requestedModel string, modelCfg SoraModelConfig) error {
|
|
|
|
|
|
if modelCfg.Type != "video" {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
balance, err := sdkClient.GetCreditBalance(ctx, token)
|
|
|
|
|
|
if err != nil {
|
2026-02-28 15:01:20 +08:00
|
|
|
|
accountID := int64(0)
|
|
|
|
|
|
if account != nil {
|
|
|
|
|
|
accountID = account.ID
|
|
|
|
|
|
}
|
|
|
|
|
|
logger.LegacyPrintf(
|
|
|
|
|
|
"service.sora_sdk",
|
|
|
|
|
|
"[PreflightCheckRawError] account_id=%d model=%s op=get_credit_balance raw_err=%s",
|
|
|
|
|
|
accountID,
|
|
|
|
|
|
requestedModel,
|
|
|
|
|
|
logredact.RedactText(err.Error()),
|
|
|
|
|
|
)
|
2026-02-25 10:15:38 +08:00
|
|
|
|
return &SoraUpstreamError{
|
|
|
|
|
|
StatusCode: http.StatusForbidden,
|
|
|
|
|
|
Message: "当前账号未开通 Sora2 能力或无可用配额",
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if balance.RateLimitReached || balance.RemainingCount <= 0 {
|
|
|
|
|
|
msg := "当前账号 Sora2 可用配额不足"
|
|
|
|
|
|
if requestedModel != "" {
|
|
|
|
|
|
msg = fmt.Sprintf("当前账号 %s 可用配额不足", requestedModel)
|
|
|
|
|
|
}
|
|
|
|
|
|
return &SoraUpstreamError{
|
|
|
|
|
|
StatusCode: http.StatusTooManyRequests,
|
|
|
|
|
|
Message: msg,
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) UploadImage(ctx context.Context, account *Account, data []byte, filename string) (string, error) {
|
|
|
|
|
|
if len(data) == 0 {
|
|
|
|
|
|
return "", errors.New("empty image data")
|
|
|
|
|
|
}
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
if filename == "" {
|
|
|
|
|
|
filename = "image.png"
|
|
|
|
|
|
}
|
|
|
|
|
|
mediaID, err := sdkClient.UploadImage(ctx, token, data, filename)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return mediaID, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) CreateImageTask(ctx context.Context, account *Account, req SoraImageRequest) (string, error) {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sentinel, err := sdkClient.GenerateSentinelToken(ctx, token)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
var taskID string
|
|
|
|
|
|
if strings.TrimSpace(req.MediaID) != "" {
|
|
|
|
|
|
taskID, err = sdkClient.CreateImageTaskWithImage(ctx, token, sentinel, req.Prompt, req.Width, req.Height, req.MediaID)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
taskID, err = sdkClient.CreateImageTask(ctx, token, sentinel, req.Prompt, req.Width, req.Height)
|
|
|
|
|
|
}
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return taskID, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) CreateVideoTask(ctx context.Context, account *Account, req SoraVideoRequest) (string, error) {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sentinel, err := sdkClient.GenerateSentinelToken(ctx, token)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
orientation := req.Orientation
|
|
|
|
|
|
if orientation == "" {
|
|
|
|
|
|
orientation = "landscape"
|
|
|
|
|
|
}
|
|
|
|
|
|
nFrames := req.Frames
|
|
|
|
|
|
if nFrames <= 0 {
|
|
|
|
|
|
nFrames = 450
|
|
|
|
|
|
}
|
|
|
|
|
|
model := req.Model
|
|
|
|
|
|
if model == "" {
|
|
|
|
|
|
model = "sy_8"
|
|
|
|
|
|
}
|
|
|
|
|
|
size := req.Size
|
|
|
|
|
|
if size == "" {
|
|
|
|
|
|
size = "small"
|
|
|
|
|
|
}
|
2026-02-28 15:01:20 +08:00
|
|
|
|
videoCount := req.VideoCount
|
|
|
|
|
|
if videoCount <= 0 {
|
|
|
|
|
|
videoCount = 1
|
|
|
|
|
|
}
|
|
|
|
|
|
if videoCount > 3 {
|
|
|
|
|
|
videoCount = 3
|
|
|
|
|
|
}
|
2026-02-25 10:15:38 +08:00
|
|
|
|
|
|
|
|
|
|
// Remix 模式
|
|
|
|
|
|
if strings.TrimSpace(req.RemixTargetID) != "" {
|
2026-02-28 15:01:20 +08:00
|
|
|
|
if videoCount > 1 {
|
|
|
|
|
|
accountID := int64(0)
|
|
|
|
|
|
if account != nil {
|
|
|
|
|
|
accountID = account.ID
|
|
|
|
|
|
}
|
|
|
|
|
|
c.debugLogf("video_count_ignored_for_remix account_id=%d count=%d", accountID, videoCount)
|
|
|
|
|
|
}
|
2026-02-25 10:15:38 +08:00
|
|
|
|
styleID := "" // SDK ExtractStyle 可从 prompt 中提取
|
|
|
|
|
|
taskID, err := sdkClient.RemixVideo(ctx, token, sentinel, req.RemixTargetID, req.Prompt, orientation, nFrames, styleID)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return taskID, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 普通视频(文生视频或图生视频)
|
2026-02-28 15:01:20 +08:00
|
|
|
|
var taskID string
|
|
|
|
|
|
if videoCount <= 1 {
|
|
|
|
|
|
taskID, err = sdkClient.CreateVideoTaskWithOptions(ctx, token, sentinel, req.Prompt, orientation, nFrames, model, size, req.MediaID, "")
|
|
|
|
|
|
} else {
|
|
|
|
|
|
taskID, err = c.createVideoTaskWithVariants(ctx, account, token, sentinel, req.Prompt, orientation, nFrames, model, size, req.MediaID, videoCount)
|
|
|
|
|
|
}
|
2026-02-25 10:15:38 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return taskID, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-28 15:01:20 +08:00
|
|
|
|
func (c *SoraSDKClient) createVideoTaskWithVariants(
|
|
|
|
|
|
ctx context.Context,
|
|
|
|
|
|
account *Account,
|
|
|
|
|
|
accessToken string,
|
|
|
|
|
|
sentinelToken string,
|
|
|
|
|
|
prompt string,
|
|
|
|
|
|
orientation string,
|
|
|
|
|
|
nFrames int,
|
|
|
|
|
|
model string,
|
|
|
|
|
|
size string,
|
|
|
|
|
|
mediaID string,
|
|
|
|
|
|
videoCount int,
|
|
|
|
|
|
) (string, error) {
|
|
|
|
|
|
inpaintItems := make([]any, 0, 1)
|
|
|
|
|
|
if strings.TrimSpace(mediaID) != "" {
|
|
|
|
|
|
inpaintItems = append(inpaintItems, map[string]any{
|
|
|
|
|
|
"kind": "upload",
|
|
|
|
|
|
"upload_id": mediaID,
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
payload := map[string]any{
|
|
|
|
|
|
"kind": "video",
|
|
|
|
|
|
"prompt": prompt,
|
|
|
|
|
|
"orientation": orientation,
|
|
|
|
|
|
"size": size,
|
|
|
|
|
|
"n_frames": nFrames,
|
|
|
|
|
|
"n_variants": videoCount,
|
|
|
|
|
|
"model": model,
|
|
|
|
|
|
"inpaint_items": inpaintItems,
|
|
|
|
|
|
"style_id": nil,
|
|
|
|
|
|
}
|
|
|
|
|
|
raw, err := c.doSoraBackendJSON(ctx, account, http.MethodPost, "/nf/create", accessToken, sentinelToken, payload)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
taskID := strings.TrimSpace(gjson.GetBytes(raw, "id").String())
|
|
|
|
|
|
if taskID == "" {
|
|
|
|
|
|
return "", errors.New("create video task response missing id")
|
|
|
|
|
|
}
|
|
|
|
|
|
return taskID, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-25 10:15:38 +08:00
|
|
|
|
func (c *SoraSDKClient) CreateStoryboardTask(ctx context.Context, account *Account, req SoraStoryboardRequest) (string, error) {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sentinel, err := sdkClient.GenerateSentinelToken(ctx, token)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
orientation := req.Orientation
|
|
|
|
|
|
if orientation == "" {
|
|
|
|
|
|
orientation = "landscape"
|
|
|
|
|
|
}
|
|
|
|
|
|
nFrames := req.Frames
|
|
|
|
|
|
if nFrames <= 0 {
|
|
|
|
|
|
nFrames = 450
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
taskID, err := sdkClient.CreateStoryboardTask(ctx, token, sentinel, req.Prompt, orientation, nFrames, req.MediaID, "")
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return taskID, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) UploadCharacterVideo(ctx context.Context, account *Account, data []byte) (string, error) {
|
|
|
|
|
|
if len(data) == 0 {
|
|
|
|
|
|
return "", errors.New("empty video data")
|
|
|
|
|
|
}
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
cameoID, err := sdkClient.UploadCharacterVideo(ctx, token, data)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return cameoID, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) GetCameoStatus(ctx context.Context, account *Account, cameoID string) (*SoraCameoStatus, error) {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
status, err := sdkClient.GetCameoStatus(ctx, token, cameoID)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return &SoraCameoStatus{
|
|
|
|
|
|
Status: status.Status,
|
|
|
|
|
|
DisplayNameHint: status.DisplayNameHint,
|
|
|
|
|
|
UsernameHint: status.UsernameHint,
|
|
|
|
|
|
ProfileAssetURL: status.ProfileAssetURL,
|
|
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) DownloadCharacterImage(ctx context.Context, account *Account, imageURL string) ([]byte, error) {
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
data, err := sdkClient.DownloadCharacterImage(ctx, imageURL)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return data, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) UploadCharacterImage(ctx context.Context, account *Account, data []byte) (string, error) {
|
|
|
|
|
|
if len(data) == 0 {
|
|
|
|
|
|
return "", errors.New("empty character image")
|
|
|
|
|
|
}
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
assetPointer, err := sdkClient.UploadCharacterImage(ctx, token, data)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return assetPointer, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) FinalizeCharacter(ctx context.Context, account *Account, req SoraCharacterFinalizeRequest) (string, error) {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
characterID, err := sdkClient.FinalizeCharacter(ctx, token, req.CameoID, req.Username, req.DisplayName, req.ProfileAssetPointer)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return characterID, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) SetCharacterPublic(ctx context.Context, account *Account, cameoID string) error {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
if err := sdkClient.SetCharacterPublic(ctx, token, cameoID); err != nil {
|
|
|
|
|
|
return c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) DeleteCharacter(ctx context.Context, account *Account, characterID string) error {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
if err := sdkClient.DeleteCharacter(ctx, token, characterID); err != nil {
|
|
|
|
|
|
return c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) PostVideoForWatermarkFree(ctx context.Context, account *Account, generationID string) (string, error) {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sentinel, err := sdkClient.GenerateSentinelToken(ctx, token)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
postID, err := sdkClient.PublishVideo(ctx, token, sentinel, generationID)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return postID, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) DeletePost(ctx context.Context, account *Account, postID string) error {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
if err := sdkClient.DeletePost(ctx, token, postID); err != nil {
|
|
|
|
|
|
return c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GetWatermarkFreeURLCustom 使用自定义第三方解析服务获取去水印链接。
|
|
|
|
|
|
// SDK 不涉及此功能,保留自建实现。
|
|
|
|
|
|
func (c *SoraSDKClient) GetWatermarkFreeURLCustom(ctx context.Context, account *Account, parseURL, parseToken, postID string) (string, error) {
|
|
|
|
|
|
parseURL = strings.TrimRight(strings.TrimSpace(parseURL), "/")
|
|
|
|
|
|
if parseURL == "" {
|
|
|
|
|
|
return "", errors.New("custom parse url is required")
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(parseToken) == "" {
|
|
|
|
|
|
return "", errors.New("custom parse token is required")
|
|
|
|
|
|
}
|
|
|
|
|
|
shareURL := "https://sora.chatgpt.com/p/" + strings.TrimSpace(postID)
|
|
|
|
|
|
payload := map[string]any{
|
|
|
|
|
|
"url": shareURL,
|
|
|
|
|
|
"token": strings.TrimSpace(parseToken),
|
|
|
|
|
|
}
|
|
|
|
|
|
body, err := json.Marshal(payload)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodPost, parseURL+"/get-sora-link", bytes.NewReader(body))
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
|
|
|
|
|
|
|
|
proxyURL := c.resolveProxyURL(account)
|
|
|
|
|
|
accountID := int64(0)
|
|
|
|
|
|
accountConcurrency := 0
|
|
|
|
|
|
if account != nil {
|
|
|
|
|
|
accountID = account.ID
|
|
|
|
|
|
accountConcurrency = account.Concurrency
|
|
|
|
|
|
}
|
|
|
|
|
|
var resp *http.Response
|
|
|
|
|
|
if c.httpUpstream != nil {
|
|
|
|
|
|
resp, err = c.httpUpstream.Do(req, proxyURL, accountID, accountConcurrency)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
resp, err = http.DefaultClient.Do(req)
|
|
|
|
|
|
}
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
defer func() { _ = resp.Body.Close() }()
|
|
|
|
|
|
raw, err := io.ReadAll(io.LimitReader(resp.Body, 4<<20))
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
|
|
return "", fmt.Errorf("custom parse failed: %d %s", resp.StatusCode, truncateForLog(raw, 256))
|
|
|
|
|
|
}
|
|
|
|
|
|
downloadLink := strings.TrimSpace(gjson.GetBytes(raw, "download_link").String())
|
|
|
|
|
|
if downloadLink == "" {
|
|
|
|
|
|
return "", errors.New("custom parse response missing download_link")
|
|
|
|
|
|
}
|
|
|
|
|
|
return downloadLink, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) EnhancePrompt(ctx context.Context, account *Account, prompt, expansionLevel string, durationS int) (string, error) {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(expansionLevel) == "" {
|
|
|
|
|
|
expansionLevel = "medium"
|
|
|
|
|
|
}
|
|
|
|
|
|
if durationS <= 0 {
|
|
|
|
|
|
durationS = 10
|
|
|
|
|
|
}
|
|
|
|
|
|
enhanced, err := sdkClient.EnhancePrompt(ctx, token, prompt, expansionLevel, durationS)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", c.wrapSDKError(err, account)
|
|
|
|
|
|
}
|
|
|
|
|
|
return enhanced, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) GetImageTask(ctx context.Context, account *Account, taskID string) (*SoraImageTaskStatus, error) {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
result := sdkClient.QueryImageTaskOnce(ctx, token, taskID, time.Now().Add(-10*time.Second))
|
|
|
|
|
|
if result.Err != nil {
|
|
|
|
|
|
return &SoraImageTaskStatus{
|
|
|
|
|
|
ID: taskID,
|
|
|
|
|
|
Status: "failed",
|
|
|
|
|
|
ErrorMsg: result.Err.Error(),
|
|
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
if result.Done && result.ImageURL != "" {
|
|
|
|
|
|
return &SoraImageTaskStatus{
|
|
|
|
|
|
ID: taskID,
|
|
|
|
|
|
Status: "succeeded",
|
|
|
|
|
|
URLs: []string{result.ImageURL},
|
|
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
status := result.Progress.Status
|
|
|
|
|
|
if status == "" {
|
|
|
|
|
|
status = "processing"
|
|
|
|
|
|
}
|
|
|
|
|
|
return &SoraImageTaskStatus{
|
|
|
|
|
|
ID: taskID,
|
|
|
|
|
|
Status: status,
|
|
|
|
|
|
ProgressPct: float64(result.Progress.Percent) / 100.0,
|
|
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) GetVideoTask(ctx context.Context, account *Account, taskID string) (*SoraVideoTaskStatus, error) {
|
|
|
|
|
|
token, err := c.getAccessToken(ctx, account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 先查询 pending 列表
|
|
|
|
|
|
result := sdkClient.QueryVideoTaskOnce(ctx, token, taskID, time.Now().Add(-10*time.Second), 0)
|
|
|
|
|
|
if result.Err != nil {
|
|
|
|
|
|
return &SoraVideoTaskStatus{
|
|
|
|
|
|
ID: taskID,
|
|
|
|
|
|
Status: "failed",
|
|
|
|
|
|
ErrorMsg: result.Err.Error(),
|
|
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
if !result.Done {
|
|
|
|
|
|
return &SoraVideoTaskStatus{
|
|
|
|
|
|
ID: taskID,
|
|
|
|
|
|
Status: result.Progress.Status,
|
|
|
|
|
|
ProgressPct: result.Progress.Percent,
|
|
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 任务不在 pending 中,查询 drafts 获取下载链接
|
2026-02-28 15:01:20 +08:00
|
|
|
|
downloadURLs, err := c.getVideoTaskDownloadURLs(ctx, account, token, taskID)
|
2026-02-25 10:15:38 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
errMsg := err.Error()
|
|
|
|
|
|
if strings.Contains(errMsg, "内容违规") || strings.Contains(errMsg, "Content violates") {
|
|
|
|
|
|
return &SoraVideoTaskStatus{
|
|
|
|
|
|
ID: taskID,
|
|
|
|
|
|
Status: "failed",
|
|
|
|
|
|
ErrorMsg: errMsg,
|
|
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
// 可能还在处理中
|
|
|
|
|
|
return &SoraVideoTaskStatus{
|
|
|
|
|
|
ID: taskID,
|
|
|
|
|
|
Status: "processing",
|
|
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
2026-02-28 15:01:20 +08:00
|
|
|
|
if len(downloadURLs) == 0 {
|
|
|
|
|
|
return &SoraVideoTaskStatus{
|
|
|
|
|
|
ID: taskID,
|
|
|
|
|
|
Status: "processing",
|
|
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
2026-02-25 10:15:38 +08:00
|
|
|
|
return &SoraVideoTaskStatus{
|
|
|
|
|
|
ID: taskID,
|
|
|
|
|
|
Status: "completed",
|
2026-02-28 15:01:20 +08:00
|
|
|
|
URLs: downloadURLs,
|
2026-02-25 10:15:38 +08:00
|
|
|
|
}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-28 15:01:20 +08:00
|
|
|
|
func (c *SoraSDKClient) getVideoTaskDownloadURLs(ctx context.Context, account *Account, accessToken, taskID string) ([]string, error) {
|
|
|
|
|
|
raw, err := c.doSoraBackendJSON(ctx, account, http.MethodGet, "/project_y/profile/drafts?limit=30", accessToken, "", nil)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
items := gjson.GetBytes(raw, "items")
|
|
|
|
|
|
if !items.Exists() || !items.IsArray() {
|
|
|
|
|
|
return nil, fmt.Errorf("drafts response missing items for task %s", taskID)
|
|
|
|
|
|
}
|
|
|
|
|
|
urlSet := make(map[string]struct{}, 4)
|
|
|
|
|
|
urls := make([]string, 0, 4)
|
|
|
|
|
|
items.ForEach(func(_, item gjson.Result) bool {
|
|
|
|
|
|
if strings.TrimSpace(item.Get("task_id").String()) != taskID {
|
|
|
|
|
|
return true
|
|
|
|
|
|
}
|
|
|
|
|
|
kind := strings.TrimSpace(item.Get("kind").String())
|
|
|
|
|
|
reason := strings.TrimSpace(item.Get("reason_str").String())
|
|
|
|
|
|
markdownReason := strings.TrimSpace(item.Get("markdown_reason_str").String())
|
|
|
|
|
|
if kind == "sora_content_violation" || reason != "" || markdownReason != "" {
|
|
|
|
|
|
if reason == "" {
|
|
|
|
|
|
reason = markdownReason
|
|
|
|
|
|
}
|
|
|
|
|
|
if reason == "" {
|
|
|
|
|
|
reason = "内容违规"
|
|
|
|
|
|
}
|
|
|
|
|
|
err = fmt.Errorf("内容违规: %s", reason)
|
|
|
|
|
|
return false
|
|
|
|
|
|
}
|
|
|
|
|
|
url := strings.TrimSpace(item.Get("downloadable_url").String())
|
|
|
|
|
|
if url == "" {
|
|
|
|
|
|
url = strings.TrimSpace(item.Get("url").String())
|
|
|
|
|
|
}
|
|
|
|
|
|
if url == "" {
|
|
|
|
|
|
return true
|
|
|
|
|
|
}
|
|
|
|
|
|
if _, exists := urlSet[url]; exists {
|
|
|
|
|
|
return true
|
|
|
|
|
|
}
|
|
|
|
|
|
urlSet[url] = struct{}{}
|
|
|
|
|
|
urls = append(urls, url)
|
|
|
|
|
|
return true
|
|
|
|
|
|
})
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
if len(urls) > 0 {
|
|
|
|
|
|
return urls, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 兼容旧 SDK 的兜底逻辑
|
|
|
|
|
|
sdkClient, sdkErr := c.getSDKClient(account)
|
|
|
|
|
|
if sdkErr != nil {
|
|
|
|
|
|
return nil, sdkErr
|
|
|
|
|
|
}
|
|
|
|
|
|
downloadURL, sdkErr := sdkClient.GetDownloadURL(ctx, accessToken, taskID)
|
|
|
|
|
|
if sdkErr != nil {
|
|
|
|
|
|
return nil, sdkErr
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(downloadURL) == "" {
|
|
|
|
|
|
return nil, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
return []string{downloadURL}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) doSoraBackendJSON(
|
|
|
|
|
|
ctx context.Context,
|
|
|
|
|
|
account *Account,
|
|
|
|
|
|
method string,
|
|
|
|
|
|
path string,
|
|
|
|
|
|
accessToken string,
|
|
|
|
|
|
sentinelToken string,
|
|
|
|
|
|
payload map[string]any,
|
|
|
|
|
|
) ([]byte, error) {
|
|
|
|
|
|
endpoint := "https://sora.chatgpt.com/backend" + path
|
|
|
|
|
|
var body io.Reader
|
|
|
|
|
|
if payload != nil {
|
|
|
|
|
|
raw, err := json.Marshal(payload)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
body = bytes.NewReader(raw)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, method, endpoint, body)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
req.Header.Set("Authorization", "Bearer "+accessToken)
|
|
|
|
|
|
req.Header.Set("Accept", "application/json, text/plain, */*")
|
|
|
|
|
|
req.Header.Set("Origin", "https://sora.chatgpt.com")
|
|
|
|
|
|
req.Header.Set("Referer", "https://sora.chatgpt.com/")
|
|
|
|
|
|
req.Header.Set("User-Agent", "Sora/1.2026.007 (Android 15; 24122RKC7C; build 2600700)")
|
|
|
|
|
|
if payload != nil {
|
|
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(sentinelToken) != "" {
|
|
|
|
|
|
req.Header.Set("openai-sentinel-token", sentinelToken)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
proxyURL := c.resolveProxyURL(account)
|
|
|
|
|
|
accountID := int64(0)
|
|
|
|
|
|
accountConcurrency := 0
|
|
|
|
|
|
if account != nil {
|
|
|
|
|
|
accountID = account.ID
|
|
|
|
|
|
accountConcurrency = account.Concurrency
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var resp *http.Response
|
|
|
|
|
|
if c.httpUpstream != nil {
|
|
|
|
|
|
resp, err = c.httpUpstream.Do(req, proxyURL, accountID, accountConcurrency)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
resp, err = http.DefaultClient.Do(req)
|
|
|
|
|
|
}
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
defer func() { _ = resp.Body.Close() }()
|
|
|
|
|
|
|
|
|
|
|
|
raw, err := io.ReadAll(io.LimitReader(resp.Body, 4<<20))
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK && resp.StatusCode != http.StatusCreated {
|
|
|
|
|
|
return nil, fmt.Errorf("HTTP %d: %s", resp.StatusCode, truncateForLog(raw, 256))
|
|
|
|
|
|
}
|
|
|
|
|
|
return raw, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-25 10:15:38 +08:00
|
|
|
|
// --- 内部方法 ---
|
|
|
|
|
|
|
|
|
|
|
|
// getSDKClient 获取或创建指定代理的 SDK 客户端实例
|
|
|
|
|
|
func (c *SoraSDKClient) getSDKClient(account *Account) (*sora.Client, error) {
|
|
|
|
|
|
proxyURL := c.resolveProxyURL(account)
|
|
|
|
|
|
if v, ok := c.sdkClients.Load(proxyURL); ok {
|
2026-02-25 11:43:08 +08:00
|
|
|
|
if cli, ok2 := v.(*sora.Client); ok2 {
|
|
|
|
|
|
return cli, nil
|
|
|
|
|
|
}
|
2026-02-25 10:15:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
client, err := sora.New(proxyURL)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("创建 Sora SDK 客户端失败: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
actual, _ := c.sdkClients.LoadOrStore(proxyURL, client)
|
2026-02-25 11:43:08 +08:00
|
|
|
|
if cli, ok := actual.(*sora.Client); ok {
|
|
|
|
|
|
return cli, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
return client, nil
|
2026-02-25 10:15:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) resolveProxyURL(account *Account) string {
|
|
|
|
|
|
if account == nil || account.ProxyID == nil || account.Proxy == nil {
|
|
|
|
|
|
return ""
|
|
|
|
|
|
}
|
|
|
|
|
|
return strings.TrimSpace(account.Proxy.URL())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// getAccessToken 获取账号的 access_token,支持多种 token 来源和自动刷新。
|
|
|
|
|
|
// 此方法保留了原 SoraDirectClient 的 token 管理逻辑。
|
|
|
|
|
|
func (c *SoraSDKClient) getAccessToken(ctx context.Context, account *Account) (string, error) {
|
|
|
|
|
|
if account == nil {
|
|
|
|
|
|
return "", errors.New("account is nil")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 优先尝试 OpenAI Token Provider
|
|
|
|
|
|
allowProvider := c.allowOpenAITokenProvider(account)
|
|
|
|
|
|
var providerErr error
|
|
|
|
|
|
if allowProvider && c.tokenProvider != nil {
|
|
|
|
|
|
token, err := c.tokenProvider.GetAccessToken(ctx, account)
|
|
|
|
|
|
if err == nil && strings.TrimSpace(token) != "" {
|
|
|
|
|
|
c.debugLogf("token_selected account_id=%d source=openai_token_provider", account.ID)
|
|
|
|
|
|
return token, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
providerErr = err
|
|
|
|
|
|
if err != nil && c.debugEnabled() {
|
|
|
|
|
|
c.debugLogf("token_provider_failed account_id=%d err=%s", account.ID, logredact.RedactText(err.Error()))
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 尝试直接使用 credentials 中的 access_token
|
|
|
|
|
|
token := strings.TrimSpace(account.GetCredential("access_token"))
|
|
|
|
|
|
if token != "" {
|
|
|
|
|
|
expiresAt := account.GetCredentialAsTime("expires_at")
|
|
|
|
|
|
if expiresAt != nil && time.Until(*expiresAt) <= 2*time.Minute {
|
|
|
|
|
|
refreshed, refreshErr := c.recoverAccessToken(ctx, account, "access_token_expiring")
|
|
|
|
|
|
if refreshErr == nil && strings.TrimSpace(refreshed) != "" {
|
|
|
|
|
|
return refreshed, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return token, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 尝试通过 session_token 或 refresh_token 恢复
|
|
|
|
|
|
recovered, recoverErr := c.recoverAccessToken(ctx, account, "access_token_missing")
|
|
|
|
|
|
if recoverErr == nil && strings.TrimSpace(recovered) != "" {
|
|
|
|
|
|
return recovered, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
if providerErr != nil {
|
|
|
|
|
|
return "", providerErr
|
|
|
|
|
|
}
|
|
|
|
|
|
return "", errors.New("access_token not found")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// recoverAccessToken 通过 session_token 或 refresh_token 恢复 access_token
|
|
|
|
|
|
func (c *SoraSDKClient) recoverAccessToken(ctx context.Context, account *Account, reason string) (string, error) {
|
|
|
|
|
|
if account == nil {
|
|
|
|
|
|
return "", errors.New("account is nil")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 先尝试 session_token
|
|
|
|
|
|
if sessionToken := strings.TrimSpace(account.GetCredential("session_token")); sessionToken != "" {
|
|
|
|
|
|
accessToken, expiresAt, err := c.exchangeSessionToken(ctx, account, sessionToken)
|
|
|
|
|
|
if err == nil && strings.TrimSpace(accessToken) != "" {
|
|
|
|
|
|
c.applyRecoveredToken(ctx, account, accessToken, "", expiresAt, sessionToken)
|
|
|
|
|
|
return accessToken, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 再尝试 refresh_token
|
|
|
|
|
|
refreshToken := strings.TrimSpace(account.GetCredential("refresh_token"))
|
|
|
|
|
|
if refreshToken == "" {
|
|
|
|
|
|
return "", errors.New("session_token/refresh_token not found")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sdkClient, err := c.getSDKClient(account)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 尝试多个 client_id
|
|
|
|
|
|
clientIDs := []string{
|
|
|
|
|
|
strings.TrimSpace(account.GetCredential("client_id")),
|
|
|
|
|
|
openaioauth.SoraClientID,
|
|
|
|
|
|
openaioauth.ClientID,
|
|
|
|
|
|
}
|
|
|
|
|
|
tried := make(map[string]struct{}, len(clientIDs))
|
|
|
|
|
|
var lastErr error
|
|
|
|
|
|
|
|
|
|
|
|
for _, clientID := range clientIDs {
|
|
|
|
|
|
if clientID == "" {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
if _, ok := tried[clientID]; ok {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
tried[clientID] = struct{}{}
|
|
|
|
|
|
|
|
|
|
|
|
newAccess, newRefresh, refreshErr := sdkClient.RefreshAccessToken(ctx, refreshToken, clientID)
|
|
|
|
|
|
if refreshErr != nil {
|
|
|
|
|
|
lastErr = refreshErr
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(newAccess) == "" {
|
|
|
|
|
|
lastErr = errors.New("refreshed access_token is empty")
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
c.applyRecoveredToken(ctx, account, newAccess, newRefresh, "", "")
|
|
|
|
|
|
return newAccess, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if lastErr != nil {
|
|
|
|
|
|
return "", lastErr
|
|
|
|
|
|
}
|
|
|
|
|
|
return "", errors.New("no available client_id for refresh_token exchange")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// exchangeSessionToken 通过 session_token 换取 access_token
|
|
|
|
|
|
func (c *SoraSDKClient) exchangeSessionToken(ctx context.Context, account *Account, sessionToken string) (string, string, error) {
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, "https://sora.chatgpt.com/api/auth/session", nil)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
req.Header.Set("Cookie", "__Secure-next-auth.session-token="+sessionToken)
|
|
|
|
|
|
req.Header.Set("Accept", "application/json")
|
|
|
|
|
|
req.Header.Set("Origin", "https://sora.chatgpt.com")
|
|
|
|
|
|
req.Header.Set("Referer", "https://sora.chatgpt.com/")
|
|
|
|
|
|
req.Header.Set("User-Agent", "Sora/1.2026.007 (Android 15; 24122RKC7C; build 2600700)")
|
|
|
|
|
|
|
|
|
|
|
|
proxyURL := c.resolveProxyURL(account)
|
|
|
|
|
|
accountID := int64(0)
|
|
|
|
|
|
accountConcurrency := 0
|
|
|
|
|
|
if account != nil {
|
|
|
|
|
|
accountID = account.ID
|
|
|
|
|
|
accountConcurrency = account.Concurrency
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var resp *http.Response
|
|
|
|
|
|
if c.httpUpstream != nil {
|
|
|
|
|
|
resp, err = c.httpUpstream.Do(req, proxyURL, accountID, accountConcurrency)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
resp, err = http.DefaultClient.Do(req)
|
|
|
|
|
|
}
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
defer func() { _ = resp.Body.Close() }()
|
|
|
|
|
|
body, err := io.ReadAll(io.LimitReader(resp.Body, 2<<20))
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return "", "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
|
|
return "", "", fmt.Errorf("session exchange failed: %d", resp.StatusCode)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
accessToken := strings.TrimSpace(gjson.GetBytes(body, "accessToken").String())
|
|
|
|
|
|
if accessToken == "" {
|
|
|
|
|
|
return "", "", errors.New("session exchange missing accessToken")
|
|
|
|
|
|
}
|
|
|
|
|
|
expiresAt := strings.TrimSpace(gjson.GetBytes(body, "expires").String())
|
|
|
|
|
|
return accessToken, expiresAt, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// applyRecoveredToken 将恢复的 token 写入账号内存和数据库
|
|
|
|
|
|
func (c *SoraSDKClient) applyRecoveredToken(ctx context.Context, account *Account, accessToken, refreshToken, expiresAt, sessionToken string) {
|
|
|
|
|
|
if account == nil {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
if account.Credentials == nil {
|
|
|
|
|
|
account.Credentials = make(map[string]any)
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(accessToken) != "" {
|
|
|
|
|
|
account.Credentials["access_token"] = accessToken
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(refreshToken) != "" {
|
|
|
|
|
|
account.Credentials["refresh_token"] = refreshToken
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(expiresAt) != "" {
|
|
|
|
|
|
account.Credentials["expires_at"] = expiresAt
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(sessionToken) != "" {
|
|
|
|
|
|
account.Credentials["session_token"] = sessionToken
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if c.accountRepo != nil {
|
|
|
|
|
|
if err := c.accountRepo.Update(ctx, account); err != nil && c.debugEnabled() {
|
|
|
|
|
|
c.debugLogf("persist_recovered_token_failed account_id=%d err=%s", account.ID, logredact.RedactText(err.Error()))
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
c.updateSoraAccountExtension(ctx, account, accessToken, refreshToken, sessionToken)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) updateSoraAccountExtension(ctx context.Context, account *Account, accessToken, refreshToken, sessionToken string) {
|
|
|
|
|
|
if c == nil || c.soraAccountRepo == nil || account == nil || account.ID <= 0 {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
updates := make(map[string]any)
|
|
|
|
|
|
if strings.TrimSpace(accessToken) != "" && strings.TrimSpace(refreshToken) != "" {
|
|
|
|
|
|
updates["access_token"] = accessToken
|
|
|
|
|
|
updates["refresh_token"] = refreshToken
|
|
|
|
|
|
}
|
|
|
|
|
|
if strings.TrimSpace(sessionToken) != "" {
|
|
|
|
|
|
updates["session_token"] = sessionToken
|
|
|
|
|
|
}
|
|
|
|
|
|
if len(updates) == 0 {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
if err := c.soraAccountRepo.Upsert(ctx, account.ID, updates); err != nil && c.debugEnabled() {
|
|
|
|
|
|
c.debugLogf("persist_sora_extension_failed account_id=%d err=%s", account.ID, logredact.RedactText(err.Error()))
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) allowOpenAITokenProvider(account *Account) bool {
|
|
|
|
|
|
if c == nil || c.tokenProvider == nil {
|
|
|
|
|
|
return false
|
|
|
|
|
|
}
|
|
|
|
|
|
if account != nil && account.Platform == PlatformSora {
|
|
|
|
|
|
return c.cfg != nil && c.cfg.Sora.Client.UseOpenAITokenProvider
|
|
|
|
|
|
}
|
|
|
|
|
|
return true
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// wrapSDKError 将 SDK 错误包装为 SoraUpstreamError
|
|
|
|
|
|
func (c *SoraSDKClient) wrapSDKError(err error, account *Account) error {
|
|
|
|
|
|
if err == nil {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
msg := err.Error()
|
|
|
|
|
|
statusCode := http.StatusBadGateway
|
|
|
|
|
|
if strings.Contains(msg, "HTTP 401") || strings.Contains(msg, "HTTP 403") {
|
|
|
|
|
|
statusCode = http.StatusUnauthorized
|
|
|
|
|
|
} else if strings.Contains(msg, "HTTP 429") {
|
|
|
|
|
|
statusCode = http.StatusTooManyRequests
|
|
|
|
|
|
} else if strings.Contains(msg, "HTTP 404") {
|
|
|
|
|
|
statusCode = http.StatusNotFound
|
|
|
|
|
|
}
|
2026-02-28 15:01:20 +08:00
|
|
|
|
accountID := int64(0)
|
|
|
|
|
|
if account != nil {
|
|
|
|
|
|
accountID = account.ID
|
|
|
|
|
|
}
|
|
|
|
|
|
logger.LegacyPrintf(
|
|
|
|
|
|
"service.sora_sdk",
|
|
|
|
|
|
"[WrapSDKError] account_id=%d mapped_status=%d raw_err=%s",
|
|
|
|
|
|
accountID,
|
|
|
|
|
|
statusCode,
|
|
|
|
|
|
logredact.RedactText(msg),
|
|
|
|
|
|
)
|
2026-02-25 10:15:38 +08:00
|
|
|
|
return &SoraUpstreamError{
|
|
|
|
|
|
StatusCode: statusCode,
|
|
|
|
|
|
Message: msg,
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) debugEnabled() bool {
|
|
|
|
|
|
return c != nil && c.cfg != nil && c.cfg.Sora.Client.Debug
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (c *SoraSDKClient) debugLogf(format string, args ...any) {
|
|
|
|
|
|
if c.debugEnabled() {
|
|
|
|
|
|
log.Printf("[SoraSDK] "+format, args...)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|