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) } }