Spaces:
Runtime error
Runtime error
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) | |
} | |
} | |