Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
alist-org
GitHub Repository: alist-org/alist
Path: blob/main/drivers/189pc/driver.go
1986 views
1
package _189pc
2
3
import (
4
"context"
5
"fmt"
6
"net/http"
7
"strconv"
8
"strings"
9
"time"
10
11
"github.com/alist-org/alist/v3/drivers/base"
12
"github.com/alist-org/alist/v3/internal/driver"
13
"github.com/alist-org/alist/v3/internal/errs"
14
"github.com/alist-org/alist/v3/internal/model"
15
"github.com/alist-org/alist/v3/pkg/utils"
16
"github.com/go-resty/resty/v2"
17
"github.com/google/uuid"
18
)
19
20
type Cloud189PC struct {
21
model.Storage
22
Addition
23
24
identity string
25
26
client *resty.Client
27
28
loginParam *LoginParam
29
tokenInfo *AppSessionResp
30
31
uploadThread int
32
33
familyTransferFolder *Cloud189Folder
34
cleanFamilyTransferFile func()
35
36
storageConfig driver.Config
37
ref *Cloud189PC
38
}
39
40
func (y *Cloud189PC) Config() driver.Config {
41
if y.storageConfig.Name == "" {
42
y.storageConfig = config
43
}
44
return y.storageConfig
45
}
46
47
func (y *Cloud189PC) GetAddition() driver.Additional {
48
return &y.Addition
49
}
50
51
func (y *Cloud189PC) Init(ctx context.Context) (err error) {
52
y.storageConfig = config
53
if y.isFamily() {
54
// 兼容旧上传接口
55
if y.Addition.RapidUpload || y.Addition.UploadMethod == "old" {
56
y.storageConfig.NoOverwriteUpload = true
57
}
58
} else {
59
// 家庭云转存,不支持覆盖上传
60
if y.Addition.FamilyTransfer {
61
y.storageConfig.NoOverwriteUpload = true
62
}
63
}
64
// 处理个人云和家庭云参数
65
if y.isFamily() && y.RootFolderID == "-11" {
66
y.RootFolderID = ""
67
}
68
if !y.isFamily() && y.RootFolderID == "" {
69
y.RootFolderID = "-11"
70
}
71
72
// 限制上传线程数
73
y.uploadThread, _ = strconv.Atoi(y.UploadThread)
74
if y.uploadThread < 1 || y.uploadThread > 32 {
75
y.uploadThread, y.UploadThread = 3, "3"
76
}
77
78
if y.ref == nil {
79
// 初始化请求客户端
80
if y.client == nil {
81
y.client = base.NewRestyClient().SetHeaders(map[string]string{
82
"Accept": "application/json;charset=UTF-8",
83
"Referer": WEB_URL,
84
})
85
}
86
87
// 避免重复登陆
88
identity := utils.GetMD5EncodeStr(y.Username + y.Password)
89
if !y.isLogin() || y.identity != identity {
90
y.identity = identity
91
if err = y.login(); err != nil {
92
return
93
}
94
}
95
}
96
97
// 处理家庭云ID
98
if y.FamilyID == "" {
99
if y.FamilyID, err = y.getFamilyID(); err != nil {
100
return err
101
}
102
}
103
104
// 创建中转文件夹
105
if y.FamilyTransfer {
106
if err := y.createFamilyTransferFolder(); err != nil {
107
return err
108
}
109
}
110
111
// 清理转存文件节流
112
y.cleanFamilyTransferFile = utils.NewThrottle2(time.Minute, func() {
113
if err := y.cleanFamilyTransfer(context.TODO()); err != nil {
114
utils.Log.Errorf("cleanFamilyTransferFolderError:%s", err)
115
}
116
})
117
return
118
}
119
120
func (d *Cloud189PC) InitReference(storage driver.Driver) error {
121
refStorage, ok := storage.(*Cloud189PC)
122
if ok {
123
d.ref = refStorage
124
return nil
125
}
126
return errs.NotSupport
127
}
128
129
func (y *Cloud189PC) Drop(ctx context.Context) error {
130
y.ref = nil
131
return nil
132
}
133
134
func (y *Cloud189PC) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) {
135
return y.getFiles(ctx, dir.GetID(), y.isFamily())
136
}
137
138
func (y *Cloud189PC) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) {
139
var downloadUrl struct {
140
URL string `json:"fileDownloadUrl"`
141
}
142
143
isFamily := y.isFamily()
144
fullUrl := API_URL
145
if isFamily {
146
fullUrl += "/family/file"
147
}
148
fullUrl += "/getFileDownloadUrl.action"
149
150
_, err := y.get(fullUrl, func(r *resty.Request) {
151
r.SetContext(ctx)
152
r.SetQueryParam("fileId", file.GetID())
153
if isFamily {
154
r.SetQueryParams(map[string]string{
155
"familyId": y.FamilyID,
156
})
157
} else {
158
r.SetQueryParams(map[string]string{
159
"dt": "3",
160
"flag": "1",
161
})
162
}
163
}, &downloadUrl, isFamily)
164
if err != nil {
165
return nil, err
166
}
167
168
// 重定向获取真实链接
169
downloadUrl.URL = strings.Replace(strings.ReplaceAll(downloadUrl.URL, "&amp;", "&"), "http://", "https://", 1)
170
res, err := base.NoRedirectClient.R().SetContext(ctx).SetDoNotParseResponse(true).Get(downloadUrl.URL)
171
if err != nil {
172
return nil, err
173
}
174
defer res.RawBody().Close()
175
if res.StatusCode() == 302 {
176
downloadUrl.URL = res.Header().Get("location")
177
}
178
179
like := &model.Link{
180
URL: downloadUrl.URL,
181
Header: http.Header{
182
"User-Agent": []string{base.UserAgent},
183
},
184
}
185
/*
186
// 获取链接有效时常
187
strs := regexp.MustCompile(`(?i)expire[^=]*=([0-9]*)`).FindStringSubmatch(downloadUrl.URL)
188
if len(strs) == 2 {
189
timestamp, err := strconv.ParseInt(strs[1], 10, 64)
190
if err == nil {
191
expired := time.Duration(timestamp-time.Now().Unix()) * time.Second
192
like.Expiration = &expired
193
}
194
}
195
*/
196
return like, nil
197
}
198
199
func (y *Cloud189PC) MakeDir(ctx context.Context, parentDir model.Obj, dirName string) (model.Obj, error) {
200
isFamily := y.isFamily()
201
fullUrl := API_URL
202
if isFamily {
203
fullUrl += "/family/file"
204
}
205
fullUrl += "/createFolder.action"
206
207
var newFolder Cloud189Folder
208
safeName := y.sanitizeName(dirName)
209
_, err := y.post(fullUrl, func(req *resty.Request) {
210
req.SetContext(ctx)
211
req.SetQueryParams(map[string]string{
212
"folderName": safeName,
213
"relativePath": "",
214
})
215
if isFamily {
216
req.SetQueryParams(map[string]string{
217
"familyId": y.FamilyID,
218
"parentId": parentDir.GetID(),
219
})
220
} else {
221
req.SetQueryParams(map[string]string{
222
"parentFolderId": parentDir.GetID(),
223
})
224
}
225
}, &newFolder, isFamily)
226
if err != nil {
227
return nil, err
228
}
229
newFolder.Name = safeName
230
return &newFolder, nil
231
}
232
233
func (y *Cloud189PC) Move(ctx context.Context, srcObj, dstDir model.Obj) (model.Obj, error) {
234
isFamily := y.isFamily()
235
other := map[string]string{"targetFileName": dstDir.GetName()}
236
237
resp, err := y.CreateBatchTask("MOVE", IF(isFamily, y.FamilyID, ""), dstDir.GetID(), other, BatchTaskInfo{
238
FileId: srcObj.GetID(),
239
FileName: srcObj.GetName(),
240
IsFolder: BoolToNumber(srcObj.IsDir()),
241
})
242
if err != nil {
243
return nil, err
244
}
245
if err = y.WaitBatchTask("MOVE", resp.TaskID, time.Millisecond*400); err != nil {
246
return nil, err
247
}
248
return srcObj, nil
249
}
250
251
func (y *Cloud189PC) Rename(ctx context.Context, srcObj model.Obj, newName string) (model.Obj, error) {
252
isFamily := y.isFamily()
253
queryParam := make(map[string]string)
254
fullUrl := API_URL
255
method := http.MethodPost
256
if isFamily {
257
fullUrl += "/family/file"
258
method = http.MethodGet
259
queryParam["familyId"] = y.FamilyID
260
}
261
262
var newObj model.Obj
263
safeName := y.sanitizeName(newName)
264
switch f := srcObj.(type) {
265
case *Cloud189File:
266
fullUrl += "/renameFile.action"
267
queryParam["fileId"] = srcObj.GetID()
268
queryParam["destFileName"] = safeName
269
newObj = &Cloud189File{Icon: f.Icon} // 复用预览
270
case *Cloud189Folder:
271
fullUrl += "/renameFolder.action"
272
queryParam["folderId"] = srcObj.GetID()
273
queryParam["destFolderName"] = safeName
274
newObj = &Cloud189Folder{}
275
default:
276
return nil, errs.NotSupport
277
}
278
279
switch obj := newObj.(type) {
280
case *Cloud189File:
281
obj.Name = safeName
282
case *Cloud189Folder:
283
obj.Name = safeName
284
}
285
286
_, err := y.request(fullUrl, method, func(req *resty.Request) {
287
req.SetContext(ctx).SetQueryParams(queryParam)
288
}, nil, newObj, isFamily)
289
if err != nil {
290
return nil, err
291
}
292
return newObj, nil
293
}
294
295
func (y *Cloud189PC) Copy(ctx context.Context, srcObj, dstDir model.Obj) error {
296
isFamily := y.isFamily()
297
other := map[string]string{"targetFileName": dstDir.GetName()}
298
299
resp, err := y.CreateBatchTask("COPY", IF(isFamily, y.FamilyID, ""), dstDir.GetID(), other, BatchTaskInfo{
300
FileId: srcObj.GetID(),
301
FileName: srcObj.GetName(),
302
IsFolder: BoolToNumber(srcObj.IsDir()),
303
})
304
305
if err != nil {
306
return err
307
}
308
return y.WaitBatchTask("COPY", resp.TaskID, time.Second)
309
}
310
311
func (y *Cloud189PC) Remove(ctx context.Context, obj model.Obj) error {
312
isFamily := y.isFamily()
313
314
resp, err := y.CreateBatchTask("DELETE", IF(isFamily, y.FamilyID, ""), "", nil, BatchTaskInfo{
315
FileId: obj.GetID(),
316
FileName: obj.GetName(),
317
IsFolder: BoolToNumber(obj.IsDir()),
318
})
319
if err != nil {
320
return err
321
}
322
// 批量任务数量限制,过快会导致无法删除
323
return y.WaitBatchTask("DELETE", resp.TaskID, time.Millisecond*200)
324
}
325
326
func (y *Cloud189PC) Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up driver.UpdateProgress) (newObj model.Obj, err error) {
327
overwrite := true
328
isFamily := y.isFamily()
329
330
// 响应时间长,按需启用
331
if y.Addition.RapidUpload && !stream.IsForceStreamUpload() {
332
if newObj, err := y.RapidUpload(ctx, dstDir, stream, isFamily, overwrite); err == nil {
333
return newObj, nil
334
}
335
}
336
337
uploadMethod := y.UploadMethod
338
if stream.IsForceStreamUpload() {
339
uploadMethod = "stream"
340
}
341
342
// 旧版上传家庭云也有限制
343
if uploadMethod == "old" {
344
return y.OldUpload(ctx, dstDir, stream, up, isFamily, overwrite)
345
}
346
347
// 开启家庭云转存
348
if !isFamily && y.FamilyTransfer {
349
// 修改上传目标为家庭云文件夹
350
transferDstDir := dstDir
351
dstDir = y.familyTransferFolder
352
353
// 使用临时文件名
354
srcName := stream.GetName()
355
stream = &WrapFileStreamer{
356
FileStreamer: stream,
357
Name: fmt.Sprintf("0%s.transfer", uuid.NewString()),
358
}
359
360
// 使用家庭云上传
361
isFamily = true
362
overwrite = false
363
364
defer func() {
365
if newObj != nil {
366
// 转存家庭云文件到个人云
367
err = y.SaveFamilyFileToPersonCloud(context.TODO(), y.FamilyID, newObj, transferDstDir, true)
368
// 删除家庭云源文件
369
go y.Delete(context.TODO(), y.FamilyID, newObj)
370
// 批量任务有概率删不掉
371
go y.cleanFamilyTransferFile()
372
// 转存失败返回错误
373
if err != nil {
374
return
375
}
376
377
// 查找转存文件
378
var file *Cloud189File
379
file, err = y.findFileByName(context.TODO(), newObj.GetName(), transferDstDir.GetID(), false)
380
if err != nil {
381
if err == errs.ObjectNotFound {
382
err = fmt.Errorf("unknown error: No transfer file obtained %s", newObj.GetName())
383
}
384
return
385
}
386
387
// 重命名转存文件
388
newObj, err = y.Rename(context.TODO(), file, srcName)
389
if err != nil {
390
// 重命名失败删除源文件
391
_ = y.Delete(context.TODO(), "", file)
392
}
393
return
394
}
395
}()
396
}
397
398
switch uploadMethod {
399
case "rapid":
400
return y.FastUpload(ctx, dstDir, stream, up, isFamily, overwrite)
401
case "stream":
402
if stream.GetSize() == 0 {
403
return y.FastUpload(ctx, dstDir, stream, up, isFamily, overwrite)
404
}
405
fallthrough
406
default:
407
return y.StreamUpload(ctx, dstDir, stream, up, isFamily, overwrite)
408
}
409
}
410
411