mirror of https://github.com/Xhofe/alist
362 lines
9.2 KiB
Go
362 lines
9.2 KiB
Go
package _189pc
|
|
|
|
import (
|
|
"container/ring"
|
|
"context"
|
|
"net/http"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/alist-org/alist/v3/drivers/base"
|
|
"github.com/alist-org/alist/v3/internal/driver"
|
|
"github.com/alist-org/alist/v3/internal/errs"
|
|
"github.com/alist-org/alist/v3/internal/model"
|
|
"github.com/alist-org/alist/v3/pkg/utils"
|
|
"github.com/go-resty/resty/v2"
|
|
)
|
|
|
|
type Cloud189PC struct {
|
|
model.Storage
|
|
Addition
|
|
|
|
identity string
|
|
|
|
client *resty.Client
|
|
|
|
loginParam *LoginParam
|
|
tokenInfo *AppSessionResp
|
|
|
|
uploadThread int
|
|
|
|
familyTransferFolder *ring.Ring
|
|
cleanFamilyTransferFile func()
|
|
|
|
storageConfig driver.Config
|
|
}
|
|
|
|
func (y *Cloud189PC) Config() driver.Config {
|
|
if y.storageConfig.Name == "" {
|
|
y.storageConfig = config
|
|
}
|
|
return y.storageConfig
|
|
}
|
|
|
|
func (y *Cloud189PC) GetAddition() driver.Additional {
|
|
return &y.Addition
|
|
}
|
|
|
|
func (y *Cloud189PC) Init(ctx context.Context) (err error) {
|
|
// 兼容旧上传接口
|
|
y.storageConfig.NoOverwriteUpload = y.isFamily() && (y.Addition.RapidUpload || y.Addition.UploadMethod == "old")
|
|
|
|
// 处理个人云和家庭云参数
|
|
if y.isFamily() && y.RootFolderID == "-11" {
|
|
y.RootFolderID = ""
|
|
}
|
|
if !y.isFamily() && y.RootFolderID == "" {
|
|
y.RootFolderID = "-11"
|
|
}
|
|
|
|
// 限制上传线程数
|
|
y.uploadThread, _ = strconv.Atoi(y.UploadThread)
|
|
if y.uploadThread < 1 || y.uploadThread > 32 {
|
|
y.uploadThread, y.UploadThread = 3, "3"
|
|
}
|
|
|
|
// 初始化请求客户端
|
|
if y.client == nil {
|
|
y.client = base.NewRestyClient().SetHeaders(map[string]string{
|
|
"Accept": "application/json;charset=UTF-8",
|
|
"Referer": WEB_URL,
|
|
})
|
|
}
|
|
|
|
// 避免重复登陆
|
|
identity := utils.GetMD5EncodeStr(y.Username + y.Password)
|
|
if !y.isLogin() || y.identity != identity {
|
|
y.identity = identity
|
|
if err = y.login(); err != nil {
|
|
return
|
|
}
|
|
}
|
|
|
|
// 处理家庭云ID
|
|
if y.FamilyID == "" {
|
|
if y.FamilyID, err = y.getFamilyID(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// 创建中转文件夹,防止重名文件
|
|
if y.FamilyTransfer {
|
|
if y.familyTransferFolder, err = y.createFamilyTransferFolder(32); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
y.cleanFamilyTransferFile = utils.NewThrottle2(time.Minute, func() {
|
|
if err := y.cleanFamilyTransfer(context.TODO()); err != nil {
|
|
utils.Log.Errorf("cleanFamilyTransferFolderError:%s", err)
|
|
}
|
|
})
|
|
return
|
|
}
|
|
|
|
func (y *Cloud189PC) Drop(ctx context.Context) error {
|
|
return nil
|
|
}
|
|
|
|
func (y *Cloud189PC) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) {
|
|
return y.getFiles(ctx, dir.GetID(), y.isFamily())
|
|
}
|
|
|
|
func (y *Cloud189PC) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) {
|
|
var downloadUrl struct {
|
|
URL string `json:"fileDownloadUrl"`
|
|
}
|
|
|
|
isFamily := y.isFamily()
|
|
fullUrl := API_URL
|
|
if isFamily {
|
|
fullUrl += "/family/file"
|
|
}
|
|
fullUrl += "/getFileDownloadUrl.action"
|
|
|
|
_, err := y.get(fullUrl, func(r *resty.Request) {
|
|
r.SetContext(ctx)
|
|
r.SetQueryParam("fileId", file.GetID())
|
|
if isFamily {
|
|
r.SetQueryParams(map[string]string{
|
|
"familyId": y.FamilyID,
|
|
})
|
|
} else {
|
|
r.SetQueryParams(map[string]string{
|
|
"dt": "3",
|
|
"flag": "1",
|
|
})
|
|
}
|
|
}, &downloadUrl, isFamily)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// 重定向获取真实链接
|
|
downloadUrl.URL = strings.Replace(strings.ReplaceAll(downloadUrl.URL, "&", "&"), "http://", "https://", 1)
|
|
res, err := base.NoRedirectClient.R().SetContext(ctx).SetDoNotParseResponse(true).Get(downloadUrl.URL)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer res.RawBody().Close()
|
|
if res.StatusCode() == 302 {
|
|
downloadUrl.URL = res.Header().Get("location")
|
|
}
|
|
|
|
like := &model.Link{
|
|
URL: downloadUrl.URL,
|
|
Header: http.Header{
|
|
"User-Agent": []string{base.UserAgent},
|
|
},
|
|
}
|
|
/*
|
|
// 获取链接有效时常
|
|
strs := regexp.MustCompile(`(?i)expire[^=]*=([0-9]*)`).FindStringSubmatch(downloadUrl.URL)
|
|
if len(strs) == 2 {
|
|
timestamp, err := strconv.ParseInt(strs[1], 10, 64)
|
|
if err == nil {
|
|
expired := time.Duration(timestamp-time.Now().Unix()) * time.Second
|
|
like.Expiration = &expired
|
|
}
|
|
}
|
|
*/
|
|
return like, nil
|
|
}
|
|
|
|
func (y *Cloud189PC) MakeDir(ctx context.Context, parentDir model.Obj, dirName string) (model.Obj, error) {
|
|
isFamily := y.isFamily()
|
|
fullUrl := API_URL
|
|
if isFamily {
|
|
fullUrl += "/family/file"
|
|
}
|
|
fullUrl += "/createFolder.action"
|
|
|
|
var newFolder Cloud189Folder
|
|
_, err := y.post(fullUrl, func(req *resty.Request) {
|
|
req.SetContext(ctx)
|
|
req.SetQueryParams(map[string]string{
|
|
"folderName": dirName,
|
|
"relativePath": "",
|
|
})
|
|
if isFamily {
|
|
req.SetQueryParams(map[string]string{
|
|
"familyId": y.FamilyID,
|
|
"parentId": parentDir.GetID(),
|
|
})
|
|
} else {
|
|
req.SetQueryParams(map[string]string{
|
|
"parentFolderId": parentDir.GetID(),
|
|
})
|
|
}
|
|
}, &newFolder, isFamily)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &newFolder, nil
|
|
}
|
|
|
|
func (y *Cloud189PC) Move(ctx context.Context, srcObj, dstDir model.Obj) (model.Obj, error) {
|
|
isFamily := y.isFamily()
|
|
other := map[string]string{"targetFileName": dstDir.GetName()}
|
|
|
|
resp, err := y.CreateBatchTask("MOVE", IF(isFamily, y.FamilyID, ""), dstDir.GetID(), other, BatchTaskInfo{
|
|
FileId: srcObj.GetID(),
|
|
FileName: srcObj.GetName(),
|
|
IsFolder: BoolToNumber(srcObj.IsDir()),
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err = y.WaitBatchTask("MOVE", resp.TaskID, time.Millisecond*400); err != nil {
|
|
return nil, err
|
|
}
|
|
return srcObj, nil
|
|
}
|
|
|
|
func (y *Cloud189PC) Rename(ctx context.Context, srcObj model.Obj, newName string) (model.Obj, error) {
|
|
isFamily := y.isFamily()
|
|
queryParam := make(map[string]string)
|
|
fullUrl := API_URL
|
|
method := http.MethodPost
|
|
if isFamily {
|
|
fullUrl += "/family/file"
|
|
method = http.MethodGet
|
|
queryParam["familyId"] = y.FamilyID
|
|
}
|
|
|
|
var newObj model.Obj
|
|
switch f := srcObj.(type) {
|
|
case *Cloud189File:
|
|
fullUrl += "/renameFile.action"
|
|
queryParam["fileId"] = srcObj.GetID()
|
|
queryParam["destFileName"] = newName
|
|
newObj = &Cloud189File{Icon: f.Icon} // 复用预览
|
|
case *Cloud189Folder:
|
|
fullUrl += "/renameFolder.action"
|
|
queryParam["folderId"] = srcObj.GetID()
|
|
queryParam["destFolderName"] = newName
|
|
newObj = &Cloud189Folder{}
|
|
default:
|
|
return nil, errs.NotSupport
|
|
}
|
|
|
|
_, err := y.request(fullUrl, method, func(req *resty.Request) {
|
|
req.SetContext(ctx).SetQueryParams(queryParam)
|
|
}, nil, newObj, isFamily)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return newObj, nil
|
|
}
|
|
|
|
func (y *Cloud189PC) Copy(ctx context.Context, srcObj, dstDir model.Obj) error {
|
|
isFamily := y.isFamily()
|
|
other := map[string]string{"targetFileName": dstDir.GetName()}
|
|
|
|
resp, err := y.CreateBatchTask("COPY", IF(isFamily, y.FamilyID, ""), dstDir.GetID(), other, BatchTaskInfo{
|
|
FileId: srcObj.GetID(),
|
|
FileName: srcObj.GetName(),
|
|
IsFolder: BoolToNumber(srcObj.IsDir()),
|
|
})
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return y.WaitBatchTask("COPY", resp.TaskID, time.Second)
|
|
}
|
|
|
|
func (y *Cloud189PC) Remove(ctx context.Context, obj model.Obj) error {
|
|
isFamily := y.isFamily()
|
|
|
|
resp, err := y.CreateBatchTask("DELETE", IF(isFamily, y.FamilyID, ""), "", nil, BatchTaskInfo{
|
|
FileId: obj.GetID(),
|
|
FileName: obj.GetName(),
|
|
IsFolder: BoolToNumber(obj.IsDir()),
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// 批量任务数量限制,过快会导致无法删除
|
|
return y.WaitBatchTask("DELETE", resp.TaskID, time.Millisecond*200)
|
|
}
|
|
|
|
func (y *Cloud189PC) Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up driver.UpdateProgress) (newObj model.Obj, err error) {
|
|
overwrite := true
|
|
isFamily := y.isFamily()
|
|
|
|
// 响应时间长,按需启用
|
|
if y.Addition.RapidUpload && !stream.IsForceStreamUpload() {
|
|
if newObj, err := y.RapidUpload(ctx, dstDir, stream, isFamily, overwrite); err == nil {
|
|
return newObj, nil
|
|
}
|
|
}
|
|
|
|
uploadMethod := y.UploadMethod
|
|
if stream.IsForceStreamUpload() {
|
|
uploadMethod = "stream"
|
|
}
|
|
|
|
// 旧版上传家庭云也有限制
|
|
if uploadMethod == "old" {
|
|
return y.OldUpload(ctx, dstDir, stream, up, isFamily, overwrite)
|
|
}
|
|
|
|
// 开启家庭云转存
|
|
if !isFamily && y.FamilyTransfer {
|
|
// 修改上传目标为家庭云文件夹
|
|
transferDstDir := dstDir
|
|
dstDir = (y.familyTransferFolder.Value).(*Cloud189Folder)
|
|
y.familyTransferFolder = y.familyTransferFolder.Next()
|
|
|
|
isFamily = true
|
|
overwrite = false
|
|
|
|
defer func() {
|
|
if newObj != nil {
|
|
// 批量任务有概率删不掉
|
|
y.cleanFamilyTransferFile()
|
|
|
|
// 转存家庭云文件到个人云
|
|
err = y.SaveFamilyFileToPersonCloud(context.TODO(), y.FamilyID, newObj, transferDstDir, true)
|
|
|
|
task := BatchTaskInfo{
|
|
FileId: newObj.GetID(),
|
|
FileName: newObj.GetName(),
|
|
IsFolder: BoolToNumber(newObj.IsDir()),
|
|
}
|
|
|
|
// 删除源文件
|
|
if resp, err := y.CreateBatchTask("DELETE", y.FamilyID, "", nil, task); err == nil {
|
|
y.WaitBatchTask("DELETE", resp.TaskID, time.Second)
|
|
// 永久删除
|
|
if resp, err := y.CreateBatchTask("CLEAR_RECYCLE", y.FamilyID, "", nil, task); err == nil {
|
|
y.WaitBatchTask("CLEAR_RECYCLE", resp.TaskID, time.Second)
|
|
}
|
|
}
|
|
newObj = nil
|
|
}
|
|
}()
|
|
}
|
|
|
|
switch uploadMethod {
|
|
case "rapid":
|
|
return y.FastUpload(ctx, dstDir, stream, up, isFamily, overwrite)
|
|
case "stream":
|
|
if stream.GetSize() == 0 {
|
|
return y.FastUpload(ctx, dstDir, stream, up, isFamily, overwrite)
|
|
}
|
|
fallthrough
|
|
default:
|
|
return y.StreamUpload(ctx, dstDir, stream, up, isFamily, overwrite)
|
|
}
|
|
}
|