Files
urldb/common/quark_pan.go

1043 lines
27 KiB
Go
Raw Normal View History

2025-07-13 09:04:50 +08:00
package pan
2025-07-13 08:02:55 +08:00
import (
"encoding/json"
"fmt"
"log"
2025-08-12 00:56:38 +08:00
"math/rand"
2025-08-13 23:30:42 +08:00
"regexp"
2025-07-24 01:05:46 +08:00
"strconv"
2025-07-13 08:02:55 +08:00
"strings"
2025-07-13 09:04:50 +08:00
"sync"
2025-07-13 08:02:55 +08:00
"time"
2025-08-11 01:34:07 +08:00
2025-08-13 23:30:42 +08:00
commonutils "github.com/ctwj/urldb/common/utils"
"github.com/ctwj/urldb/db"
"github.com/ctwj/urldb/db/entity"
"github.com/ctwj/urldb/db/repo"
2025-08-11 01:34:07 +08:00
"github.com/ctwj/urldb/utils"
2025-07-13 08:02:55 +08:00
)
// QuarkPanService 夸克网盘服务
type QuarkPanService struct {
*BasePanService
2025-07-13 09:04:50 +08:00
configMutex sync.RWMutex // 保护配置的读写锁
2025-07-13 08:02:55 +08:00
}
2025-08-14 00:05:35 +08:00
// 全局配置缓存刷新信号
var configRefreshChan = make(chan bool, 1)
2025-07-13 09:04:50 +08:00
// 单例相关变量
var (
2025-08-14 00:05:35 +08:00
systemConfigRepo repo.SystemConfigRepository
systemConfigOnce sync.Once
2025-07-13 09:04:50 +08:00
)
// NewQuarkPanService 创建夸克网盘服务(单例模式)
2025-07-13 08:02:55 +08:00
func NewQuarkPanService(config *PanConfig) *QuarkPanService {
2025-09-03 16:49:07 +08:00
quarkInstance := &QuarkPanService{
BasePanService: NewBasePanService(config),
}
// 设置夸克网盘的默认请求头
quarkInstance.SetHeaders(map[string]string{
"Accept": "application/json, text/plain, */*",
"Accept-Language": "zh-CN,zh;q=0.9",
"Content-Type": "application/json;charset=UTF-8",
"Sec-Ch-Ua": `"Chromium";v="122", "Not(A:Brand";v="24", "Google Chrome";v="122"`,
"Sec-Ch-Ua-Mobile": "?0",
"Sec-Ch-Ua-Platform": `"Windows"`,
"Sec-Fetch-Dest": "empty",
"Sec-Fetch-Mode": "cors",
"Sec-Fetch-Site": "same-site",
"Referer": "https://pan.quark.cn/",
"Referrer-Policy": "strict-origin-when-cross-origin",
"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36",
"Cookie": config.Cookie,
2025-07-13 08:02:55 +08:00
})
2025-07-13 09:04:50 +08:00
// 更新配置
quarkInstance.UpdateConfig(config)
return quarkInstance
}
// GetQuarkInstance 获取夸克网盘服务单例实例
func GetQuarkInstance() *QuarkPanService {
return NewQuarkPanService(nil)
}
// UpdateConfig 更新配置(线程安全)
func (q *QuarkPanService) UpdateConfig(config *PanConfig) {
if config == nil {
return
}
q.configMutex.Lock()
defer q.configMutex.Unlock()
q.config = config
2025-07-24 01:05:46 +08:00
// 设置Cookie到header
if config.Cookie != "" {
q.SetHeader("Cookie", config.Cookie)
}
}
// SetCookie 设置Cookie
func (q *QuarkPanService) SetCookie(cookie string) {
q.SetHeader("Cookie", cookie)
q.configMutex.Lock()
if q.config != nil {
q.config.Cookie = cookie
}
q.configMutex.Unlock()
}
// GetCookie 获取当前Cookie
func (q *QuarkPanService) GetCookie() string {
return q.GetHeader("Cookie")
2025-07-13 08:02:55 +08:00
}
// GetServiceType 获取服务类型
func (q *QuarkPanService) GetServiceType() ServiceType {
return Quark
}
// Transfer 转存分享链接
func (q *QuarkPanService) Transfer(shareID string) (*TransferResult, error) {
2025-07-13 09:04:50 +08:00
// 读取配置(线程安全)
q.configMutex.RLock()
config := q.config
q.configMutex.RUnlock()
2025-07-13 08:02:55 +08:00
log.Printf("开始处理夸克分享: %s", shareID)
// 获取stoken
var stoken string
2025-07-13 09:04:50 +08:00
if config.Stoken == "" {
2025-07-13 08:02:55 +08:00
stokenResult, err := q.getStoken(shareID)
if err != nil {
return ErrorResult(fmt.Sprintf("获取stoken失败: %v", err)), nil
}
stoken = strings.ReplaceAll(stokenResult.Stoken, " ", "+")
} else {
2025-07-13 09:04:50 +08:00
stoken = strings.ReplaceAll(config.Stoken, " ", "+")
2025-07-13 08:02:55 +08:00
}
// 获取分享详情
shareResult, err := q.getShare(shareID, stoken)
if err != nil {
return ErrorResult(fmt.Sprintf("获取分享详情失败: %v", err)), nil
}
2025-07-13 09:57:36 +08:00
if config.IsType == 1 {
// 遍历 资源目录结构
for _, item := range shareResult.List {
// 获取文件信息
fileList, err := q.getDirFile(item.Fid)
if err != nil {
log.Printf("获取目录文件失败: %v", err)
continue
}
// 处理文件列表
if fileList != nil {
log.Printf("目录 %s 包含 %d 个文件/文件夹", item.Fid, len(fileList))
// 遍历所有文件,可以在这里添加具体的处理逻辑
for _, file := range fileList {
if fileName, ok := file["file_name"].(string); ok {
if fileType, ok := file["file_type"].(float64); ok {
fileTypeStr := "文件"
if fileType == 1 {
fileTypeStr = "目录"
}
log.Printf(" - %s (%s)", fileName, fileTypeStr)
}
}
}
}
}
// 直接返回资源信息
return SuccessResult("检验成功", map[string]interface{}{
"title": shareResult.Share.Title,
"shareUrl": config.URL,
}), nil
}
2025-07-13 08:02:55 +08:00
// 提取文件信息
fidList := make([]string, 0)
fidTokenList := make([]string, 0)
title := shareResult.Share.Title
for _, item := range shareResult.List {
fidList = append(fidList, item.Fid)
fidTokenList = append(fidTokenList, item.ShareFidToken)
}
// 转存资源
saveResult, err := q.getShareSave(shareID, stoken, fidList, fidTokenList)
if err != nil {
return ErrorResult(fmt.Sprintf("转存失败: %v", err)), nil
}
taskID := saveResult.TaskID
// 等待转存完成
myData, err := q.waitForTask(taskID)
if err != nil {
return ErrorResult(fmt.Sprintf("等待转存完成失败: %v", err)), nil
}
// 删除广告文件(如果有配置)
if err := q.deleteAdFiles(myData.SaveAs.SaveAsTopFids[0]); err != nil {
log.Printf("删除广告文件失败: %v", err)
}
2025-08-12 00:56:38 +08:00
// 添加个人自定义广告
if err := q.addAd(myData.SaveAs.SaveAsTopFids[0]); err != nil {
log.Printf("添加广告文件失败: %v", err)
}
2025-07-13 08:02:55 +08:00
// 分享资源
shareBtnResult, err := q.getShareBtn(myData.SaveAs.SaveAsTopFids, title)
if err != nil {
return ErrorResult(fmt.Sprintf("分享失败: %v", err)), nil
}
// 等待分享完成
shareTaskResult, err := q.waitForTask(shareBtnResult.TaskID)
if err != nil {
return ErrorResult(fmt.Sprintf("等待分享完成失败: %v", err)), nil
}
// 获取分享密码
passwordResult, err := q.getSharePassword(shareTaskResult.ShareID)
if err != nil {
return ErrorResult(fmt.Sprintf("获取分享密码失败: %v", err)), nil
}
// 确定fid
var fid string
if len(myData.SaveAs.SaveAsTopFids) > 1 {
fid = strings.Join(myData.SaveAs.SaveAsTopFids, ",")
} else {
fid = passwordResult.FirstFile.Fid
}
return SuccessResult("转存成功", map[string]interface{}{
"shareUrl": passwordResult.ShareURL,
"title": passwordResult.ShareTitle,
"fid": fid,
"code": passwordResult.Code,
}), nil
}
// GetFiles 获取文件列表
func (q *QuarkPanService) GetFiles(pdirFid string) (*TransferResult, error) {
if pdirFid == "" {
pdirFid = "0"
}
queryParams := map[string]string{
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
"pdir_fid": pdirFid,
"_page": "1",
"_size": "50",
"_fetch_total": "1",
"_fetch_sub_dirs": "0",
"_sort": "file_type:asc,updated_at:desc",
}
data, err := q.HTTPGet("https://drive-pc.quark.cn/1/clouddrive/file/sort", queryParams)
if err != nil {
return ErrorResult(fmt.Sprintf("获取文件列表失败: %v", err)), nil
}
var response struct {
Status int `json:"status"`
Message string `json:"message"`
Data struct {
List []interface{} `json:"list"`
} `json:"data"`
}
if err := json.Unmarshal(data, &response); err != nil {
return ErrorResult("解析响应失败"), nil
}
if response.Status != 200 {
message := response.Message
if message == "require login [guest]" {
message = "夸克未登录请检查cookie"
}
return ErrorResult(message), nil
}
return SuccessResult("获取成功", response.Data.List), nil
}
// DeleteFiles 删除文件
func (q *QuarkPanService) DeleteFiles(fileList []string) (*TransferResult, error) {
if len(fileList) == 0 {
return ErrorResult("文件列表为空"), nil
}
// 逐个删除文件,确保每个删除操作都完成
for _, fileID := range fileList {
err := q.deleteSingleFile(fileID)
if err != nil {
log.Printf("删除文件 %s 失败: %v", fileID, err)
return ErrorResult(fmt.Sprintf("删除文件 %s 失败: %v", fileID, err)), nil
}
}
return SuccessResult("删除成功", nil), nil
}
// deleteSingleFile 删除单个文件
func (q *QuarkPanService) deleteSingleFile(fileID string) error {
log.Printf("正在删除文件: %s", fileID)
2025-07-13 08:02:55 +08:00
data := map[string]interface{}{
"action_type": 2,
"filelist": []string{fileID},
"exclude_fids": []string{},
2025-07-13 08:02:55 +08:00
}
queryParams := map[string]string{
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
}
respData, err := q.HTTPPost("https://drive-pc.quark.cn/1/clouddrive/file/delete", data, queryParams)
2025-07-13 08:02:55 +08:00
if err != nil {
return fmt.Errorf("删除文件请求失败: %v", err)
2025-07-13 08:02:55 +08:00
}
// 解析响应
var response struct {
Status int `json:"status"`
Message string `json:"message"`
Data struct {
TaskID string `json:"task_id"`
} `json:"data"`
}
if err := json.Unmarshal(respData, &response); err != nil {
return fmt.Errorf("解析删除响应失败: %v", err)
}
if response.Status != 200 {
return fmt.Errorf("删除文件失败: %s", response.Message)
}
// 如果有任务ID等待任务完成
if response.Data.TaskID != "" {
log.Printf("删除文件任务ID: %s", response.Data.TaskID)
_, err := q.waitForTask(response.Data.TaskID)
if err != nil {
return fmt.Errorf("等待删除任务完成失败: %v", err)
}
log.Printf("文件 %s 删除完成", fileID)
} else {
log.Printf("文件 %s 删除完成无任务ID", fileID)
}
return nil
2025-07-13 08:02:55 +08:00
}
// getStoken 获取stoken
func (q *QuarkPanService) getStoken(shareID string) (*StokenResult, error) {
data := map[string]interface{}{
"passcode": "",
"pwd_id": shareID,
}
queryParams := map[string]string{
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
}
respData, err := q.HTTPPost("https://drive-pc.quark.cn/1/clouddrive/share/sharepage/token", data, queryParams)
if err != nil {
return nil, err
}
var response struct {
Status int `json:"status"`
Message string `json:"message"`
Data StokenResult `json:"data"`
}
if err := json.Unmarshal(respData, &response); err != nil {
return nil, err
}
if response.Status != 200 {
return nil, fmt.Errorf(response.Message)
}
return &response.Data, nil
}
// getShare 获取分享详情
func (q *QuarkPanService) getShare(shareID, stoken string) (*ShareResult, error) {
queryParams := map[string]string{
2025-07-13 09:57:36 +08:00
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
"pwd_id": shareID,
"stoken": stoken,
"pdir_fid": "0",
"force": "0",
"_page": "1",
"_size": "100",
"_fetch_banner": "1",
"_fetch_share": "1",
"_fetch_total": "1",
"_sort": "file_type:asc,updated_at:desc",
}
respData, err := q.HTTPGet("https://drive-pc.quark.cn/1/clouddrive/share/sharepage/detail", queryParams)
2025-07-13 08:02:55 +08:00
if err != nil {
return nil, err
}
var response struct {
Status int `json:"status"`
Message string `json:"message"`
Data ShareResult `json:"data"`
}
if err := json.Unmarshal(respData, &response); err != nil {
return nil, err
}
if response.Status != 200 {
return nil, fmt.Errorf(response.Message)
}
return &response.Data, nil
}
// getShareSave 转存分享
func (q *QuarkPanService) getShareSave(shareID, stoken string, fidList, fidTokenList []string) (*SaveResult, error) {
return q.getShareSaveToDir(shareID, stoken, fidList, fidTokenList, "0")
}
// getShareSaveToDir 转存分享到指定目录
func (q *QuarkPanService) getShareSaveToDir(shareID, stoken string, fidList, fidTokenList []string, toPdirFid string) (*SaveResult, error) {
2025-07-13 08:02:55 +08:00
data := map[string]interface{}{
"pwd_id": shareID,
"stoken": stoken,
"fid_list": fidList,
"fid_token_list": fidTokenList,
"to_pdir_fid": toPdirFid, // 存储到指定目录
2025-07-13 08:02:55 +08:00
}
queryParams := map[string]string{
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
}
respData, err := q.HTTPPost("https://drive-pc.quark.cn/1/clouddrive/share/sharepage/save", data, queryParams)
if err != nil {
return nil, err
}
var response struct {
Status int `json:"status"`
Message string `json:"message"`
Data SaveResult `json:"data"`
}
if err := json.Unmarshal(respData, &response); err != nil {
return nil, err
}
if response.Status != 200 {
return nil, fmt.Errorf(response.Message)
}
return &response.Data, nil
}
2025-07-24 01:05:46 +08:00
// 生成指定长度的时间戳
func (q *QuarkPanService) generateTimestamp(length int) int64 {
2025-08-11 01:34:07 +08:00
timestamp := utils.GetCurrentTime().UnixNano() / int64(time.Millisecond)
2025-07-24 01:05:46 +08:00
timestampStr := strconv.FormatInt(timestamp, 10)
if len(timestampStr) > length {
timestampStr = timestampStr[:length]
}
timestamp, _ = strconv.ParseInt(timestampStr, 10, 64)
return timestamp
}
2025-07-13 08:02:55 +08:00
// getShareBtn 分享按钮
func (q *QuarkPanService) getShareBtn(fidList []string, title string) (*ShareBtnResult, error) {
data := map[string]interface{}{
"fid_list": fidList,
"title": title,
2025-07-24 01:05:46 +08:00
"url_type": 1,
2025-07-13 08:02:55 +08:00
"expired_type": 1, // 永久分享
}
queryParams := map[string]string{
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
}
2025-07-24 01:05:46 +08:00
respData, err := q.HTTPPost("https://drive-pc.quark.cn/1/clouddrive/share", data, queryParams)
2025-07-13 08:02:55 +08:00
if err != nil {
return nil, err
}
var response struct {
Status int `json:"status"`
Message string `json:"message"`
Data ShareBtnResult `json:"data"`
}
if err := json.Unmarshal(respData, &response); err != nil {
return nil, err
}
if response.Status != 200 {
return nil, fmt.Errorf(response.Message)
}
return &response.Data, nil
}
// getShareTask 获取分享任务状态
func (q *QuarkPanService) getShareTask(taskID string, retryIndex int) (*TaskResult, error) {
queryParams := map[string]string{
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
"task_id": taskID,
"retry_index": fmt.Sprintf("%d", retryIndex),
2025-07-24 01:05:46 +08:00
"__dt": "21192",
"__t": fmt.Sprintf("%d", q.generateTimestamp(13)),
2025-07-13 08:02:55 +08:00
}
2025-07-24 01:05:46 +08:00
respData, err := q.HTTPGet("https://drive-pc.quark.cn/1/clouddrive/task", queryParams)
2025-07-13 08:02:55 +08:00
if err != nil {
return nil, err
}
var response struct {
Status int `json:"status"`
Message string `json:"message"`
Data TaskResult `json:"data"`
}
if err := json.Unmarshal(respData, &response); err != nil {
return nil, err
}
if response.Status != 200 {
return nil, fmt.Errorf(response.Message)
}
return &response.Data, nil
}
// getSharePassword 获取分享密码
func (q *QuarkPanService) getSharePassword(shareID string) (*PasswordResult, error) {
queryParams := map[string]string{
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
}
2025-07-24 01:05:46 +08:00
data := map[string]interface{}{
"share_id": shareID,
}
respData, err := q.HTTPPost("https://drive-pc.quark.cn/1/clouddrive/share/password", data, queryParams)
2025-07-13 08:02:55 +08:00
if err != nil {
return nil, err
}
var response struct {
Status int `json:"status"`
Message string `json:"message"`
Data PasswordResult `json:"data"`
}
if err := json.Unmarshal(respData, &response); err != nil {
return nil, err
}
if response.Status != 200 {
return nil, fmt.Errorf(response.Message)
}
return &response.Data, nil
}
// waitForTask 等待任务完成
func (q *QuarkPanService) waitForTask(taskID string) (*TaskResult, error) {
maxRetries := 50
retryDelay := 2 * time.Second
for retryIndex := 0; retryIndex < maxRetries; retryIndex++ {
result, err := q.getShareTask(taskID, retryIndex)
if err != nil {
if strings.Contains(err.Error(), "capacity limit[{0}]") {
return nil, fmt.Errorf("容量不足")
}
return nil, err
}
if result.Status == 2 { // 任务完成
return result, nil
}
time.Sleep(retryDelay)
}
return nil, fmt.Errorf("任务超时")
}
// deleteAdFiles 删除广告文件
func (q *QuarkPanService) deleteAdFiles(pdirFid string) error {
2025-08-12 00:56:38 +08:00
log.Printf("开始删除广告文件目录ID: %s", pdirFid)
// 获取目录文件列表
fileList, err := q.getDirFile(pdirFid)
if err != nil {
log.Printf("获取目录文件失败: %v", err)
return err
}
if fileList == nil || len(fileList) == 0 {
log.Printf("目录为空,无需删除广告文件")
return nil
}
// 删除包含广告关键词的文件
for _, file := range fileList {
if fileName, ok := file["file_name"].(string); ok {
log.Printf("检查文件: %s", fileName)
if q.containsAdKeywords(fileName) {
if fid, ok := file["fid"].(string); ok {
log.Printf("删除广告文件: %s (FID: %s)", fileName, fid)
_, err := q.DeleteFiles([]string{fid})
if err != nil {
log.Printf("删除广告文件失败: %v", err)
} else {
log.Printf("成功删除广告文件: %s", fileName)
}
}
}
}
}
return nil
}
// containsAdKeywords 检查文件名是否包含广告关键词
func (q *QuarkPanService) containsAdKeywords(filename string) bool {
2025-08-13 23:30:42 +08:00
// 从系统配置中获取广告关键词
adKeywordsStr, err := q.getSystemConfigValue(entity.ConfigKeyAdKeywords)
if err != nil {
log.Printf("获取广告关键词配置失败: %v", err)
return false
}
2025-08-12 00:56:38 +08:00
2025-08-13 23:30:42 +08:00
// 如果配置为空返回false
if adKeywordsStr == "" {
return false
}
2025-08-12 00:56:38 +08:00
2025-08-13 23:30:42 +08:00
// 按逗号分割关键词(支持中文和英文逗号)
adKeywords := q.splitKeywords(adKeywordsStr)
2025-08-12 00:56:38 +08:00
return q.checkKeywordsInFilename(filename, adKeywords)
}
// checkKeywordsInFilename 检查文件名是否包含指定关键词
func (q *QuarkPanService) checkKeywordsInFilename(filename string, keywords []string) bool {
// 转为小写进行比较
lowercaseFilename := strings.ToLower(filename)
for _, keyword := range keywords {
if strings.Contains(lowercaseFilename, strings.ToLower(keyword)) {
log.Printf("文件 %s 包含广告关键词: %s", filename, keyword)
return true
}
}
return false
}
2025-08-13 23:30:42 +08:00
// getSystemConfigValue 获取系统配置值
func (q *QuarkPanService) getSystemConfigValue(key string) (string, error) {
2025-08-14 00:05:35 +08:00
// 检查是否需要刷新缓存
select {
case <-configRefreshChan:
// 收到刷新信号,清空缓存
systemConfigOnce.Do(func() {
systemConfigRepo = repo.NewSystemConfigRepository(db.DB)
})
systemConfigRepo.ClearConfigCache()
default:
// 没有刷新信号,继续使用缓存
}
// 使用单例模式获取系统配置仓库
systemConfigOnce.Do(func() {
systemConfigRepo = repo.NewSystemConfigRepository(db.DB)
})
2025-08-13 23:30:42 +08:00
return systemConfigRepo.GetConfigValue(key)
}
2025-08-14 00:05:35 +08:00
// refreshSystemConfigCache 刷新系统配置缓存
func (q *QuarkPanService) refreshSystemConfigCache() {
systemConfigOnce.Do(func() {
systemConfigRepo = repo.NewSystemConfigRepository(db.DB)
})
systemConfigRepo.ClearConfigCache()
}
// RefreshSystemConfigCache 全局刷新系统配置缓存(供外部调用)
func RefreshSystemConfigCache() {
select {
case configRefreshChan <- true:
// 发送刷新信号
default:
// 通道已满,忽略
}
}
2025-08-13 23:30:42 +08:00
// splitKeywords 按逗号分割关键词(支持中文和英文逗号)
func (q *QuarkPanService) splitKeywords(keywordsStr string) []string {
if keywordsStr == "" {
return []string{}
}
// 使用正则表达式同时匹配中英文逗号
re := regexp.MustCompile(`[,]`)
parts := re.Split(keywordsStr, -1)
var result []string
for _, part := range parts {
// 去除首尾空格
trimmed := strings.TrimSpace(part)
if trimmed != "" {
result = append(result, trimmed)
}
}
return result
}
// splitAdURLs 按换行符分割广告URL列表
func (q *QuarkPanService) splitAdURLs(autoInsertAdStr string) []string {
if autoInsertAdStr == "" {
return []string{}
}
// 按换行符分割
lines := strings.Split(autoInsertAdStr, "\n")
var result []string
for _, line := range lines {
// 去除首尾空格
trimmed := strings.TrimSpace(line)
if trimmed != "" {
result = append(result, trimmed)
}
}
return result
}
// extractAdFileIDs 从广告URL列表中提取文件ID
func (q *QuarkPanService) extractAdFileIDs(adURLs []string) []string {
var result []string
for _, url := range adURLs {
// 使用 ExtractShareIdString 提取分享ID
shareID, _ := commonutils.ExtractShareIdString(url)
if shareID != "" {
result = append(result, shareID)
}
}
return result
}
2025-08-12 00:56:38 +08:00
// addAd 添加个人自定义广告
func (q *QuarkPanService) addAd(dirID string) error {
log.Printf("开始添加个人自定义广告到目录: %s", dirID)
2025-08-13 23:30:42 +08:00
// 从系统配置中获取自动插入广告内容
autoInsertAdStr, err := q.getSystemConfigValue(entity.ConfigKeyAutoInsertAd)
if err != nil {
log.Printf("获取自动插入广告配置失败: %v", err)
return err
}
// 如果配置为空,跳过广告插入
if autoInsertAdStr == "" {
log.Printf("没有配置自动插入广告,跳过广告插入")
return nil
}
// 按换行符分割广告URL列表
adURLs := q.splitAdURLs(autoInsertAdStr)
if len(adURLs) == 0 {
log.Printf("没有有效的广告URL跳过广告插入")
return nil
2025-08-12 00:56:38 +08:00
}
2025-08-13 23:30:42 +08:00
// 提取广告文件ID列表
adFileIDs := q.extractAdFileIDs(adURLs)
2025-08-12 00:56:38 +08:00
if len(adFileIDs) == 0 {
2025-08-13 23:30:42 +08:00
log.Printf("没有有效的广告文件ID跳过广告插入")
2025-08-12 00:56:38 +08:00
return nil
}
// 随机选择一个广告文件
rand.Seed(utils.GetCurrentTimestampNano())
2025-08-12 00:56:38 +08:00
selectedAdID := adFileIDs[rand.Intn(len(adFileIDs))]
log.Printf("选择广告文件ID: %s", selectedAdID)
// 获取广告文件的stoken
stokenResult, err := q.getStoken(selectedAdID)
if err != nil {
log.Printf("获取广告文件stoken失败: %v", err)
return err
}
// 获取广告文件详情
adDetail, err := q.getShare(selectedAdID, stokenResult.Stoken)
if err != nil {
log.Printf("获取广告文件详情失败: %v", err)
return err
}
if len(adDetail.List) == 0 {
log.Printf("广告文件详情为空")
return fmt.Errorf("广告文件详情为空")
}
// 获取第一个广告文件的信息
adFile := adDetail.List[0]
fid := adFile.Fid
shareFidToken := adFile.ShareFidToken
// 保存广告文件到目标目录
saveResult, err := q.getShareSaveToDir(selectedAdID, stokenResult.Stoken, []string{fid}, []string{shareFidToken}, dirID)
2025-08-12 00:56:38 +08:00
if err != nil {
log.Printf("保存广告文件失败: %v", err)
return err
}
// 等待保存完成
_, err = q.waitForTask(saveResult.TaskID)
if err != nil {
log.Printf("等待广告文件保存完成失败: %v", err)
return err
}
log.Printf("广告文件添加成功")
2025-07-13 08:02:55 +08:00
return nil
}
2025-07-13 09:57:36 +08:00
// getDirFile 获取指定文件夹的文件列表
func (q *QuarkPanService) getDirFile(pdirFid string) ([]map[string]interface{}, error) {
log.Printf("正在遍历父文件夹: %s", pdirFid)
queryParams := map[string]string{
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
"pdir_fid": pdirFid,
"_page": "1",
"_size": "50",
"_fetch_total": "1",
"_fetch_sub_dirs": "0",
"_sort": "updated_at:desc",
}
respData, err := q.HTTPGet("https://drive-pc.quark.cn/1/clouddrive/file/sort", queryParams)
if err != nil {
log.Printf("获取目录文件失败: %v", err)
return nil, err
}
var response struct {
Status int `json:"status"`
Message string `json:"message"`
Data struct {
List []map[string]interface{} `json:"list"`
} `json:"data"`
}
if err := json.Unmarshal(respData, &response); err != nil {
log.Printf("解析目录文件响应失败: %v", err)
return nil, err
}
if response.Status != 200 {
return nil, fmt.Errorf(response.Message)
}
// 直接返回文件列表,不递归处理子目录(与参考代码保持一致)
return response.Data.List, nil
2025-07-13 09:57:36 +08:00
}
2025-07-13 08:02:55 +08:00
// 定义各种结果结构体
type StokenResult struct {
Stoken string `json:"stoken"`
Title string `json:"title"`
}
type ShareResult struct {
Share struct {
Title string `json:"title"`
} `json:"share"`
List []struct {
Fid string `json:"fid"`
ShareFidToken string `json:"share_fid_token"`
} `json:"list"`
}
type SaveResult struct {
TaskID string `json:"task_id"`
}
type ShareBtnResult struct {
TaskID string `json:"task_id"`
}
type TaskResult struct {
Status int `json:"status"`
ShareID string `json:"share_id"`
SaveAs struct {
SaveAsTopFids []string `json:"save_as_top_fids"`
} `json:"save_as"`
}
type PasswordResult struct {
ShareURL string `json:"share_url"`
ShareTitle string `json:"share_title"`
Code string `json:"code"`
FirstFile struct {
Fid string `json:"fid"`
} `json:"first_file"`
}
2025-07-17 01:43:22 +08:00
// GetUserInfo 获取用户信息
2025-09-03 23:39:26 +08:00
func (q *QuarkPanService) GetUserInfo(cookie *string) (*UserInfo, error) {
2025-07-17 01:43:22 +08:00
// 临时设置cookie
originalCookie := q.GetHeader("Cookie")
2025-09-03 23:39:26 +08:00
q.SetHeader("Cookie", *cookie)
2025-07-17 01:43:22 +08:00
defer q.SetHeader("Cookie", originalCookie) // 恢复原始cookie
// 获取用户基本信息
queryParams := map[string]string{
"platform": "pc",
"fr": "pc",
}
data, err := q.HTTPGet("https://pan.quark.cn/account/info", queryParams)
if err != nil {
return nil, fmt.Errorf("获取用户信息失败: %v", err)
}
var response struct {
Success bool `json:"success"`
Code string `json:"code"`
Data struct {
Nickname string `json:"nickname"`
AvatarUri string `json:"avatarUri"`
Mobilekps string `json:"mobilekps"`
Config struct{} `json:"config"`
} `json:"data"`
}
if err := json.Unmarshal(data, &response); err != nil {
return nil, fmt.Errorf("解析用户信息失败: %v", err)
}
if !response.Success || response.Code != "OK" {
return nil, fmt.Errorf("获取用户信息失败: API返回错误")
}
// 获取用户详细信息(容量和会员信息)
queryParams1 := map[string]string{
"pr": "ucpro",
"fr": "pc",
"uc_param_str": "",
"fetch_subscribe": "true",
"_ch": "home",
"fetch_identity": "true",
}
data1, err := q.HTTPGet("https://drive-pc.quark.cn/1/clouddrive/member", queryParams1)
if err != nil {
return nil, fmt.Errorf("获取用户详细信息失败: %v", err)
}
var memberResponse struct {
Status int `json:"status"`
Code int `json:"code"`
Message string `json:"message"`
Data struct {
2025-07-17 02:09:24 +08:00
TotalCapacity int64 `json:"total_capacity"`
UseCapacity int64 `json:"use_capacity"`
MemberType string `json:"member_type"`
2025-07-17 01:43:22 +08:00
} `json:"data"`
}
if err := json.Unmarshal(data1, &memberResponse); err != nil {
return nil, fmt.Errorf("解析用户详细信息失败: %v", err)
}
if memberResponse.Status != 200 || memberResponse.Code != 0 {
return nil, fmt.Errorf("获取用户详细信息失败: %s", memberResponse.Message)
}
// 判断VIP状态
vipStatus := memberResponse.Data.MemberType != "NORMAL"
return &UserInfo{
Username: response.Data.Nickname,
VIPStatus: vipStatus,
2025-07-17 02:09:24 +08:00
UsedSpace: memberResponse.Data.UseCapacity,
2025-07-17 01:43:22 +08:00
TotalSpace: memberResponse.Data.TotalCapacity,
ServiceType: "quark",
}, nil
}
func (xq *QuarkPanService) SetCKSRepository(cksRepo repo.CksRepository, entity entity.Cks) {
2025-09-03 00:48:10 +08:00
}
2025-07-17 01:43:22 +08:00
// formatBytes 格式化字节数为可读格式
func formatBytes(bytes int64) string {
const unit = 1024
if bytes < unit {
return fmt.Sprintf("%d B", bytes)
}
div, exp := int64(unit), 0
for n := bytes / unit; n >= unit; n /= unit {
div *= unit
exp++
}
return fmt.Sprintf("%.1f %cB", float64(bytes)/float64(div), "KMGTPE"[exp])
}