2021-10-10 10:48:53 +08:00
|
|
|
|
// Copyright (c) 2020 tickstep.
|
|
|
|
|
//
|
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
|
//
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
//
|
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
|
// limitations under the License.
|
|
|
|
|
package panupload
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
2022-12-19 19:41:13 +08:00
|
|
|
|
"github.com/tickstep/aliyunpan/internal/log"
|
2022-04-20 23:16:04 +08:00
|
|
|
|
"github.com/tickstep/aliyunpan/internal/plugins"
|
2022-01-16 14:21:20 +08:00
|
|
|
|
"github.com/tickstep/library-go/logger"
|
2021-10-24 22:03:47 +08:00
|
|
|
|
"os"
|
2021-10-10 10:48:53 +08:00
|
|
|
|
"path"
|
|
|
|
|
"path/filepath"
|
|
|
|
|
"strings"
|
|
|
|
|
"sync"
|
|
|
|
|
"time"
|
|
|
|
|
|
2022-01-15 15:33:35 +08:00
|
|
|
|
"github.com/tickstep/aliyunpan/internal/utils"
|
|
|
|
|
"github.com/tickstep/library-go/requester/rio/speeds"
|
|
|
|
|
|
2021-10-10 10:48:53 +08:00
|
|
|
|
"github.com/tickstep/aliyunpan-api/aliyunpan"
|
|
|
|
|
"github.com/tickstep/aliyunpan-api/aliyunpan/apierror"
|
|
|
|
|
"github.com/tickstep/aliyunpan/internal/config"
|
|
|
|
|
"github.com/tickstep/aliyunpan/internal/file/uploader"
|
|
|
|
|
"github.com/tickstep/aliyunpan/internal/functions"
|
|
|
|
|
"github.com/tickstep/aliyunpan/internal/localfile"
|
|
|
|
|
"github.com/tickstep/aliyunpan/internal/taskframework"
|
|
|
|
|
"github.com/tickstep/library-go/converter"
|
|
|
|
|
"github.com/tickstep/library-go/requester/rio"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type (
|
|
|
|
|
// StepUpload 上传步骤
|
|
|
|
|
StepUpload int
|
|
|
|
|
|
|
|
|
|
// UploadTaskUnit 上传的任务单元
|
|
|
|
|
UploadTaskUnit struct {
|
|
|
|
|
LocalFileChecksum *localfile.LocalFileEntity // 要上传的本地文件详情
|
|
|
|
|
Step StepUpload
|
|
|
|
|
SavePath string // 保存路径
|
|
|
|
|
DriveId string // 网盘ID,例如:文件网盘,相册网盘
|
|
|
|
|
FolderCreateMutex *sync.Mutex
|
|
|
|
|
|
|
|
|
|
PanClient *aliyunpan.PanClient
|
|
|
|
|
UploadingDatabase *UploadingDatabase // 数据库
|
|
|
|
|
Parallel int
|
2022-01-15 15:33:35 +08:00
|
|
|
|
NoRapidUpload bool // 禁用秒传,无需计算SHA1,直接上传
|
2021-10-10 10:48:53 +08:00
|
|
|
|
BlockSize int64 // 分片大小
|
|
|
|
|
|
|
|
|
|
UploadStatistic *UploadStatistic
|
|
|
|
|
|
|
|
|
|
taskInfo *taskframework.TaskInfo
|
|
|
|
|
panDir string
|
|
|
|
|
panFile string
|
|
|
|
|
state *uploader.InstanceState
|
|
|
|
|
|
|
|
|
|
ShowProgress bool
|
|
|
|
|
IsOverwrite bool // 覆盖已存在的文件,如果同名文件已存在则移到回收站里
|
2021-11-11 23:24:56 +08:00
|
|
|
|
|
|
|
|
|
// 是否使用内置链接
|
|
|
|
|
UseInternalUrl bool
|
2022-01-07 23:41:22 +08:00
|
|
|
|
|
|
|
|
|
// 全局速度统计
|
2022-01-15 15:33:35 +08:00
|
|
|
|
GlobalSpeedsStat *speeds.Speeds
|
2022-12-19 19:41:13 +08:00
|
|
|
|
|
|
|
|
|
// 上传文件记录器
|
|
|
|
|
FileRecorder *log.FileRecorder
|
2021-10-10 10:48:53 +08:00
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const (
|
|
|
|
|
// StepUploadInit 初始化步骤
|
|
|
|
|
StepUploadInit StepUpload = iota
|
|
|
|
|
// 上传前准备,创建上传任务
|
|
|
|
|
StepUploadPrepareUpload
|
|
|
|
|
// StepUploadRapidUpload 秒传步骤
|
|
|
|
|
StepUploadRapidUpload
|
|
|
|
|
// StepUploadUpload 正常上传步骤
|
|
|
|
|
StepUploadUpload
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const (
|
|
|
|
|
StrUploadFailed = "上传文件失败"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func (utu *UploadTaskUnit) SetTaskInfo(taskInfo *taskframework.TaskInfo) {
|
|
|
|
|
utu.taskInfo = taskInfo
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// prepareFile 解析文件阶段
|
|
|
|
|
func (utu *UploadTaskUnit) prepareFile() {
|
|
|
|
|
// 解析文件保存路径
|
|
|
|
|
var (
|
|
|
|
|
panDir, panFile = path.Split(utu.SavePath)
|
|
|
|
|
)
|
|
|
|
|
utu.panDir = path.Clean(panDir)
|
|
|
|
|
utu.panFile = panFile
|
|
|
|
|
|
|
|
|
|
// 检测断点续传
|
|
|
|
|
utu.state = utu.UploadingDatabase.Search(&utu.LocalFileChecksum.LocalFileMeta)
|
|
|
|
|
if utu.state != nil || utu.LocalFileChecksum.LocalFileMeta.UploadOpEntity != nil { // 读取到了上一次上传task请求的fileId
|
|
|
|
|
utu.Step = StepUploadUpload
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if utu.LocalFileChecksum.UploadOpEntity == nil {
|
|
|
|
|
utu.Step = StepUploadPrepareUpload
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if utu.NoRapidUpload {
|
|
|
|
|
utu.Step = StepUploadUpload
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-20 11:54:20 +08:00
|
|
|
|
//if utu.LocalFileChecksum.Length > MaxRapidUploadSize {
|
|
|
|
|
// fmt.Printf("[%s] 文件超过20GB, 无法使用秒传功能, 跳过秒传...\n", utu.taskInfo.Id())
|
|
|
|
|
// utu.Step = StepUploadUpload
|
|
|
|
|
// return
|
|
|
|
|
//}
|
2021-10-10 10:48:53 +08:00
|
|
|
|
// 下一步: 秒传
|
|
|
|
|
utu.Step = StepUploadRapidUpload
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// rapidUpload 执行秒传
|
|
|
|
|
func (utu *UploadTaskUnit) rapidUpload() (isContinue bool, result *taskframework.TaskUnitRunResult) {
|
|
|
|
|
utu.Step = StepUploadRapidUpload
|
|
|
|
|
|
|
|
|
|
// 是否可以秒传
|
|
|
|
|
result = &taskframework.TaskUnitRunResult{}
|
2022-01-15 15:33:35 +08:00
|
|
|
|
fmt.Printf("[%s] %s 检测秒传中, 请稍候...\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"))
|
2021-10-10 10:48:53 +08:00
|
|
|
|
if utu.LocalFileChecksum.UploadOpEntity.RapidUpload {
|
2022-01-15 15:33:35 +08:00
|
|
|
|
fmt.Printf("[%s] %s 秒传成功, 保存到网盘路径: %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), utu.SavePath)
|
2021-10-10 10:48:53 +08:00
|
|
|
|
result.Succeed = true
|
|
|
|
|
return false, result
|
|
|
|
|
} else {
|
2022-01-15 15:33:35 +08:00
|
|
|
|
fmt.Printf("[%s] %s 秒传失败,开始正常上传文件\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"))
|
2021-10-10 10:48:53 +08:00
|
|
|
|
result.Succeed = false
|
|
|
|
|
result.ResultMessage = "文件未曾上传,无法秒传"
|
|
|
|
|
return true, result
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// upload 上传文件
|
|
|
|
|
func (utu *UploadTaskUnit) upload() (result *taskframework.TaskUnitRunResult) {
|
|
|
|
|
utu.Step = StepUploadUpload
|
|
|
|
|
|
|
|
|
|
// 创建分片上传器
|
|
|
|
|
// 阿里云盘默认就是分片上传,每一个分片对应一个part_info
|
|
|
|
|
// 但是不支持分片同时上传,必须单线程,并且按照顺序从1开始一个一个上传
|
|
|
|
|
muer := uploader.NewMultiUploader(
|
2021-11-11 23:24:56 +08:00
|
|
|
|
NewPanUpload(utu.PanClient, utu.SavePath, utu.DriveId, utu.LocalFileChecksum.UploadOpEntity, utu.UseInternalUrl),
|
2021-10-10 10:48:53 +08:00
|
|
|
|
rio.NewFileReaderAtLen64(utu.LocalFileChecksum.GetFile()), &uploader.MultiUploaderConfig{
|
|
|
|
|
Parallel: utu.Parallel,
|
|
|
|
|
BlockSize: utu.BlockSize,
|
|
|
|
|
MaxRate: config.Config.MaxUploadRate,
|
2022-01-07 23:41:22 +08:00
|
|
|
|
}, utu.LocalFileChecksum.UploadOpEntity, utu.GlobalSpeedsStat)
|
2021-10-10 10:48:53 +08:00
|
|
|
|
|
|
|
|
|
// 设置断点续传
|
|
|
|
|
if utu.state != nil {
|
|
|
|
|
muer.SetInstanceState(utu.state)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
muer.OnUploadStatusEvent(func(status uploader.Status, updateChan <-chan struct{}) {
|
|
|
|
|
select {
|
|
|
|
|
case <-updateChan:
|
|
|
|
|
utu.UploadingDatabase.UpdateUploading(&utu.LocalFileChecksum.LocalFileMeta, muer.InstanceState())
|
|
|
|
|
utu.UploadingDatabase.Save()
|
|
|
|
|
default:
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if utu.ShowProgress {
|
2022-01-15 19:51:51 +08:00
|
|
|
|
fmt.Printf("\r[%s] ↑ %s/%s %s/s(%s/s) in %s ............", utu.taskInfo.Id(),
|
2021-10-10 10:48:53 +08:00
|
|
|
|
converter.ConvertFileSize(status.Uploaded(), 2),
|
|
|
|
|
converter.ConvertFileSize(status.TotalSize(), 2),
|
|
|
|
|
converter.ConvertFileSize(status.SpeedsPerSecond(), 2),
|
2022-01-07 23:41:22 +08:00
|
|
|
|
converter.ConvertFileSize(utu.GlobalSpeedsStat.GetSpeeds(), 2),
|
2021-10-10 10:48:53 +08:00
|
|
|
|
status.TimeElapsed(),
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// result
|
|
|
|
|
result = &taskframework.TaskUnitRunResult{}
|
|
|
|
|
muer.OnSuccess(func() {
|
|
|
|
|
fmt.Printf("\n")
|
2022-01-15 15:33:35 +08:00
|
|
|
|
fmt.Printf("[%s] %s 上传文件成功, 保存到网盘路径: %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), utu.SavePath)
|
2021-10-10 10:48:53 +08:00
|
|
|
|
// 统计
|
|
|
|
|
utu.UploadStatistic.AddTotalSize(utu.LocalFileChecksum.Length)
|
|
|
|
|
utu.UploadingDatabase.Delete(&utu.LocalFileChecksum.LocalFileMeta) // 删除
|
|
|
|
|
utu.UploadingDatabase.Save()
|
|
|
|
|
result.Succeed = true
|
|
|
|
|
})
|
|
|
|
|
muer.OnError(func(err error) {
|
|
|
|
|
apiError, ok := err.(*apierror.ApiError)
|
|
|
|
|
if !ok {
|
|
|
|
|
// 未知错误类型 (非预期的)
|
|
|
|
|
// 不重试
|
|
|
|
|
result.ResultMessage = "上传文件错误"
|
|
|
|
|
result.Err = err
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 默认需要重试
|
|
|
|
|
result.NeedRetry = true
|
|
|
|
|
|
|
|
|
|
switch apiError.ErrCode() {
|
|
|
|
|
default:
|
|
|
|
|
result.ResultMessage = StrUploadFailed
|
|
|
|
|
result.NeedRetry = false
|
|
|
|
|
result.Err = apiError
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
})
|
2022-06-16 19:58:49 +08:00
|
|
|
|
er := muer.Execute()
|
|
|
|
|
if er != nil {
|
|
|
|
|
result.ResultMessage = StrUploadFailed
|
|
|
|
|
result.NeedRetry = true
|
|
|
|
|
result.Err = er
|
|
|
|
|
}
|
2021-10-10 10:48:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (utu *UploadTaskUnit) OnRetry(lastRunResult *taskframework.TaskUnitRunResult) {
|
|
|
|
|
// 输出错误信息
|
|
|
|
|
if lastRunResult.Err == nil {
|
|
|
|
|
// result中不包含Err, 忽略输出
|
|
|
|
|
fmt.Printf("[%s] %s, 重试 %d/%d\n", utu.taskInfo.Id(), lastRunResult.ResultMessage, utu.taskInfo.Retry(), utu.taskInfo.MaxRetry())
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
fmt.Printf("[%s] %s, %s, 重试 %d/%d\n", utu.taskInfo.Id(), lastRunResult.ResultMessage, lastRunResult.Err, utu.taskInfo.Retry(), utu.taskInfo.MaxRetry())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (utu *UploadTaskUnit) OnSuccess(lastRunResult *taskframework.TaskUnitRunResult) {
|
2022-04-20 23:16:04 +08:00
|
|
|
|
// 执行插件
|
|
|
|
|
utu.pluginCallback("success")
|
2022-12-19 19:41:13 +08:00
|
|
|
|
|
|
|
|
|
// 上传文件数据记录
|
|
|
|
|
if config.Config.FileRecordConfig == "1" {
|
|
|
|
|
utu.FileRecorder.Append(&log.FileRecordItem{
|
|
|
|
|
Status: "成功",
|
|
|
|
|
TimeStr: utils.NowTimeStr(),
|
|
|
|
|
FileSize: utu.LocalFileChecksum.LocalFileMeta.Length,
|
|
|
|
|
FilePath: utu.LocalFileChecksum.Path.LogicPath,
|
|
|
|
|
})
|
|
|
|
|
}
|
2021-10-10 10:48:53 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (utu *UploadTaskUnit) OnFailed(lastRunResult *taskframework.TaskUnitRunResult) {
|
|
|
|
|
// 失败
|
2022-04-20 23:16:04 +08:00
|
|
|
|
utu.pluginCallback("fail")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (utu *UploadTaskUnit) pluginCallback(result string) {
|
2022-06-16 14:38:45 +08:00
|
|
|
|
if utu.LocalFileChecksum == nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
2022-04-20 23:16:04 +08:00
|
|
|
|
pluginManger := plugins.NewPluginManager(config.GetPluginDir())
|
|
|
|
|
plugin, _ := pluginManger.GetPlugin()
|
2022-06-28 15:10:54 +08:00
|
|
|
|
_, fileName := filepath.Split(utu.LocalFileChecksum.Path.LogicPath)
|
2022-04-20 23:16:04 +08:00
|
|
|
|
pluginParam := &plugins.UploadFileFinishParams{
|
2022-06-28 15:10:54 +08:00
|
|
|
|
LocalFilePath: utu.LocalFileChecksum.Path.LogicPath,
|
2022-04-20 23:16:04 +08:00
|
|
|
|
LocalFileName: fileName,
|
|
|
|
|
LocalFileSize: utu.LocalFileChecksum.LocalFileMeta.Length,
|
|
|
|
|
LocalFileType: "file",
|
|
|
|
|
LocalFileUpdatedAt: time.Unix(utu.LocalFileChecksum.LocalFileMeta.ModTime, 0).Format("2006-01-02 15:04:05"),
|
|
|
|
|
LocalFileSha1: utu.LocalFileChecksum.LocalFileMeta.SHA1,
|
|
|
|
|
UploadResult: result,
|
|
|
|
|
DriveId: utu.DriveId,
|
|
|
|
|
DriveFilePath: utu.panDir + "/" + utu.panFile,
|
|
|
|
|
}
|
|
|
|
|
if er := plugin.UploadFileFinishCallback(plugins.GetContext(config.Config.ActiveUser()), pluginParam); er != nil {
|
|
|
|
|
logger.Verboseln("插件UploadFileFinishCallback调用失败: {}", er)
|
|
|
|
|
} else {
|
|
|
|
|
logger.Verboseln("插件UploadFileFinishCallback调用成功")
|
|
|
|
|
}
|
2021-10-10 10:48:53 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (utu *UploadTaskUnit) OnComplete(lastRunResult *taskframework.TaskUnitRunResult) {
|
2022-04-20 23:16:04 +08:00
|
|
|
|
// 任务结束,可能成功也可能失败
|
2021-10-10 10:48:53 +08:00
|
|
|
|
}
|
2022-04-21 16:30:12 +08:00
|
|
|
|
func (utu *UploadTaskUnit) OnCancel(lastRunResult *taskframework.TaskUnitRunResult) {
|
2021-10-10 10:48:53 +08:00
|
|
|
|
|
2022-04-21 16:30:12 +08:00
|
|
|
|
}
|
2021-10-10 10:48:53 +08:00
|
|
|
|
func (utu *UploadTaskUnit) RetryWait() time.Duration {
|
|
|
|
|
return functions.RetryWait(utu.taskInfo.Retry())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (utu *UploadTaskUnit) Run() (result *taskframework.TaskUnitRunResult) {
|
|
|
|
|
err := utu.LocalFileChecksum.OpenPath()
|
|
|
|
|
if err != nil {
|
|
|
|
|
fmt.Printf("[%s] 文件不可读, 错误信息: %s, 跳过...\n", utu.taskInfo.Id(), err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
defer utu.LocalFileChecksum.Close() // 关闭文件
|
|
|
|
|
|
|
|
|
|
timeStart := time.Now()
|
|
|
|
|
result = &taskframework.TaskUnitRunResult{}
|
|
|
|
|
|
2022-06-28 15:10:54 +08:00
|
|
|
|
fmt.Printf("[%s] %s 准备上传: %s => %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), utu.LocalFileChecksum.Path.LogicPath, utu.SavePath)
|
2021-10-10 10:48:53 +08:00
|
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
|
var msg string
|
|
|
|
|
if result.Err != nil {
|
|
|
|
|
msg = "失败!" + result.ResultMessage + "," + result.Err.Error()
|
|
|
|
|
} else if result.Succeed {
|
|
|
|
|
msg = "成功!" + result.ResultMessage
|
|
|
|
|
} else {
|
|
|
|
|
msg = result.ResultMessage
|
|
|
|
|
}
|
2022-01-15 16:14:23 +08:00
|
|
|
|
fmt.Printf("[%s] %s 文件上传结果: %s 耗时 %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), msg, utils.ConvertTime(time.Now().Sub(timeStart)))
|
2021-10-10 10:48:53 +08:00
|
|
|
|
}()
|
|
|
|
|
// 准备文件
|
|
|
|
|
utu.prepareFile()
|
2022-01-16 14:21:20 +08:00
|
|
|
|
logger.Verbosef("[%s] %s 准备结束, 准备耗时 %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), utils.ConvertTime(time.Now().Sub(timeStart)))
|
2021-10-10 10:48:53 +08:00
|
|
|
|
|
|
|
|
|
var apierr *apierror.ApiError
|
|
|
|
|
var rs *aliyunpan.MkdirResult
|
|
|
|
|
var appCreateUploadFileParam *aliyunpan.CreateFileUploadParam
|
2022-01-15 15:33:35 +08:00
|
|
|
|
var sha1Str string
|
|
|
|
|
var contentHashName string
|
|
|
|
|
var checkNameMode string
|
2021-10-10 10:48:53 +08:00
|
|
|
|
var saveFilePath string
|
|
|
|
|
var uploadOpEntity *aliyunpan.CreateFileUploadResult
|
2021-10-24 22:03:47 +08:00
|
|
|
|
var proofCode = ""
|
|
|
|
|
var localFileInfo os.FileInfo
|
|
|
|
|
var localFile *os.File
|
2022-12-13 16:32:08 +08:00
|
|
|
|
var newBlockSize int64
|
2021-10-10 10:48:53 +08:00
|
|
|
|
|
|
|
|
|
switch utu.Step {
|
|
|
|
|
case StepUploadPrepareUpload:
|
|
|
|
|
goto StepUploadPrepareUpload
|
|
|
|
|
case StepUploadRapidUpload:
|
|
|
|
|
goto stepUploadRapidUpload
|
|
|
|
|
case StepUploadUpload:
|
|
|
|
|
goto stepUploadUpload
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
StepUploadPrepareUpload:
|
2022-01-12 11:34:07 +08:00
|
|
|
|
// 创建上传任务
|
|
|
|
|
// 创建云盘文件夹
|
2021-10-10 10:48:53 +08:00
|
|
|
|
saveFilePath = path.Dir(utu.SavePath)
|
|
|
|
|
if saveFilePath != "/" {
|
2023-01-03 10:57:08 +08:00
|
|
|
|
utu.FolderCreateMutex.Lock()
|
2022-01-15 15:33:35 +08:00
|
|
|
|
fmt.Printf("[%s] %s 正在检测和创建云盘文件夹: %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), saveFilePath)
|
2022-06-20 17:28:25 +08:00
|
|
|
|
fe, apierr1 := utu.PanClient.FileInfoByPath(utu.DriveId, saveFilePath)
|
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
|
if apierr1 != nil && apierr1.Code == apierror.ApiCodeFileNotFoundCode {
|
2022-01-16 14:21:20 +08:00
|
|
|
|
logger.Verbosef("[%s] %s 创建云盘文件夹: %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), saveFilePath)
|
2022-01-15 17:17:03 +08:00
|
|
|
|
// rs, apierr = utu.PanClient.MkdirRecursive(utu.DriveId, "", "", 0, strings.Split(path.Clean(saveFilePath), "/"))
|
|
|
|
|
// 可以直接创建的,不用循环创建
|
|
|
|
|
rs, apierr = utu.PanClient.Mkdir(utu.DriveId, "root", saveFilePath)
|
2021-10-10 10:48:53 +08:00
|
|
|
|
if apierr != nil || rs.FileId == "" {
|
|
|
|
|
result.Err = apierr
|
|
|
|
|
result.ResultMessage = "创建云盘文件夹失败"
|
2023-01-03 10:57:08 +08:00
|
|
|
|
utu.FolderCreateMutex.Unlock()
|
2021-10-10 10:48:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
2022-06-20 17:28:25 +08:00
|
|
|
|
logger.Verbosef("[%s] %s 创建云盘文件夹成功\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"))
|
|
|
|
|
} else {
|
|
|
|
|
rs = &aliyunpan.MkdirResult{}
|
|
|
|
|
rs.FileId = fe.FileId
|
2021-10-10 10:48:53 +08:00
|
|
|
|
}
|
2023-01-03 10:57:08 +08:00
|
|
|
|
utu.FolderCreateMutex.Unlock()
|
2021-10-10 10:48:53 +08:00
|
|
|
|
} else {
|
|
|
|
|
rs = &aliyunpan.MkdirResult{}
|
|
|
|
|
rs.FileId = ""
|
|
|
|
|
}
|
2022-06-20 17:28:25 +08:00
|
|
|
|
time.Sleep(time.Duration(2) * time.Second)
|
2021-10-10 10:48:53 +08:00
|
|
|
|
|
2022-01-12 11:34:07 +08:00
|
|
|
|
sha1Str = ""
|
|
|
|
|
proofCode = ""
|
|
|
|
|
contentHashName = "sha1"
|
|
|
|
|
checkNameMode = "auto_rename"
|
|
|
|
|
if !utu.NoRapidUpload {
|
|
|
|
|
// 计算文件SHA1
|
2022-06-28 15:10:54 +08:00
|
|
|
|
fmt.Printf("[%s] %s 正在计算文件SHA1: %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), utu.LocalFileChecksum.Path.LogicPath)
|
2022-01-12 11:34:07 +08:00
|
|
|
|
utu.LocalFileChecksum.Sum(localfile.CHECKSUM_SHA1)
|
|
|
|
|
sha1Str = utu.LocalFileChecksum.SHA1
|
|
|
|
|
if utu.LocalFileChecksum.Length == 0 {
|
|
|
|
|
sha1Str = aliyunpan.DefaultZeroSizeFileContentHash
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// proof code
|
2022-06-28 15:10:54 +08:00
|
|
|
|
localFile, _ = os.Open(utu.LocalFileChecksum.Path.RealPath)
|
2022-01-15 15:33:35 +08:00
|
|
|
|
localFileInfo, _ = localFile.Stat()
|
2022-01-12 11:34:07 +08:00
|
|
|
|
proofCode = aliyunpan.CalcProofCode(utu.PanClient.GetAccessToken(), rio.NewFileReaderAtLen64(localFile), localFileInfo.Size())
|
|
|
|
|
localFile.Close()
|
|
|
|
|
} else {
|
2022-01-15 15:33:35 +08:00
|
|
|
|
fmt.Printf("[%s] %s 已经禁用秒传检测,直接上传\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"))
|
2022-01-12 11:34:07 +08:00
|
|
|
|
contentHashName = "none"
|
|
|
|
|
checkNameMode = "auto_rename"
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-10 10:48:53 +08:00
|
|
|
|
if utu.IsOverwrite {
|
|
|
|
|
// 标记覆盖旧同名文件
|
|
|
|
|
// 检查同名文件是否存在
|
|
|
|
|
efi, apierr := utu.PanClient.FileInfoByPath(utu.DriveId, utu.SavePath)
|
|
|
|
|
if apierr != nil && apierr.Code != apierror.ApiCodeFileNotFoundCode {
|
|
|
|
|
result.Err = apierr
|
|
|
|
|
result.ResultMessage = "检测同名文件失败"
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if efi != nil && efi.FileId != "" {
|
2022-01-12 11:34:07 +08:00
|
|
|
|
if strings.ToUpper(efi.ContentHash) == strings.ToUpper(sha1Str) {
|
2021-10-10 10:48:53 +08:00
|
|
|
|
result.Succeed = true
|
|
|
|
|
result.Extra = efi
|
2022-01-15 15:33:35 +08:00
|
|
|
|
fmt.Printf("[%s] %s 检测到同名文件,文件内容完全一致,无需重复上传: %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), utu.SavePath)
|
2021-10-10 10:48:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
// existed, delete it
|
|
|
|
|
var fileDeleteResult []*aliyunpan.FileBatchActionResult
|
|
|
|
|
var err *apierror.ApiError
|
2022-01-15 15:33:35 +08:00
|
|
|
|
fileDeleteResult, err = utu.PanClient.FileDelete([]*aliyunpan.FileBatchActionParam{{DriveId: efi.DriveId, FileId: efi.FileId}})
|
2021-10-10 10:48:53 +08:00
|
|
|
|
if err != nil || len(fileDeleteResult) == 0 {
|
|
|
|
|
result.Err = err
|
|
|
|
|
result.ResultMessage = "无法删除文件,请稍后重试"
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
time.Sleep(time.Duration(500) * time.Millisecond)
|
2022-01-15 15:33:35 +08:00
|
|
|
|
fmt.Printf("[%s] %s 检测到同名文件,已移动到回收站: %s\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"), utu.SavePath)
|
2021-10-10 10:48:53 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-13 16:32:08 +08:00
|
|
|
|
// 自动调整BlockSize大小
|
|
|
|
|
newBlockSize = utils.ResizeUploadBlockSize(utu.LocalFileChecksum.Length, utu.BlockSize)
|
|
|
|
|
if newBlockSize != utu.BlockSize {
|
|
|
|
|
logger.Verboseln("resize upload block size to: " + converter.ConvertFileSize(newBlockSize, 2))
|
|
|
|
|
utu.BlockSize = newBlockSize
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-12 11:34:07 +08:00
|
|
|
|
// 创建上传任务
|
2021-10-10 10:48:53 +08:00
|
|
|
|
appCreateUploadFileParam = &aliyunpan.CreateFileUploadParam{
|
2022-01-15 15:33:35 +08:00
|
|
|
|
DriveId: utu.DriveId,
|
2022-04-30 07:18:13 +08:00
|
|
|
|
Name: filepath.Base(utu.SavePath),
|
2022-01-15 15:33:35 +08:00
|
|
|
|
Size: utu.LocalFileChecksum.Length,
|
|
|
|
|
ContentHash: sha1Str,
|
2022-01-12 11:34:07 +08:00
|
|
|
|
ContentHashName: contentHashName,
|
2022-01-15 15:33:35 +08:00
|
|
|
|
CheckNameMode: checkNameMode,
|
|
|
|
|
ParentFileId: rs.FileId,
|
|
|
|
|
BlockSize: utu.BlockSize,
|
|
|
|
|
ProofCode: proofCode,
|
|
|
|
|
ProofVersion: "v1",
|
2021-10-10 10:48:53 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uploadOpEntity, apierr = utu.PanClient.CreateUploadFile(appCreateUploadFileParam)
|
|
|
|
|
if apierr != nil {
|
|
|
|
|
result.Err = apierr
|
|
|
|
|
result.ResultMessage = "创建上传任务失败:" + apierr.Error()
|
2022-12-06 19:48:31 +08:00
|
|
|
|
if apierr.Code == apierror.ApiCodeTooManyRequests || apierr.Code == apierror.ApiCodeBadGateway {
|
|
|
|
|
logger.Verboseln("create upload file error: " + result.ResultMessage)
|
|
|
|
|
// 重试
|
|
|
|
|
result.NeedRetry = true
|
|
|
|
|
}
|
2021-10-10 10:48:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
utu.LocalFileChecksum.UploadOpEntity = uploadOpEntity
|
|
|
|
|
utu.LocalFileChecksum.ParentFolderId = rs.FileId
|
|
|
|
|
|
|
|
|
|
stepUploadRapidUpload:
|
|
|
|
|
// 秒传
|
|
|
|
|
if !utu.NoRapidUpload {
|
|
|
|
|
isContinue, rapidUploadResult := utu.rapidUpload()
|
|
|
|
|
if !isContinue {
|
|
|
|
|
// 秒传成功, 返回秒传的结果
|
|
|
|
|
return rapidUploadResult
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
stepUploadUpload:
|
|
|
|
|
// 正常上传流程
|
|
|
|
|
uploadResult := utu.upload()
|
2022-06-16 19:58:49 +08:00
|
|
|
|
if uploadResult != nil && uploadResult.Err != nil {
|
2022-12-24 17:04:38 +08:00
|
|
|
|
if uploadResult.Err == uploader.UploadPartNotSeq {
|
2022-06-16 19:58:49 +08:00
|
|
|
|
fmt.Printf("[%s] %s 文件分片上传顺序错误,开始重新上传文件\n", utu.taskInfo.Id(), time.Now().Format("2006-01-02 15:04:06"))
|
|
|
|
|
// 需要重新从0开始上传
|
|
|
|
|
uploadResult = nil
|
|
|
|
|
utu.LocalFileChecksum.UploadOpEntity = nil
|
|
|
|
|
utu.state = nil
|
|
|
|
|
goto StepUploadPrepareUpload
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-10-10 10:48:53 +08:00
|
|
|
|
return uploadResult
|
|
|
|
|
}
|