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