feat: seeding status for aria2 download tasks (#1422)
* feat: add aria2 seeding * fix: move RecycleTaskType to the bottom * refactor: refactor recycle aria2 temp filepull/1491/head
parent
846438e3af
commit
b1685d2863
@ -1 +1 @@
|
|||||||
Subproject commit a1028e7e0ae96be4bb67d8c117cf39e07c207473
|
Subproject commit 02d93206cc5b943c34b5f5ac86c23dd96f5ef603
|
@ -0,0 +1,130 @@
|
|||||||
|
package task
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
|
||||||
|
model "github.com/cloudreve/Cloudreve/v3/models"
|
||||||
|
"github.com/cloudreve/Cloudreve/v3/pkg/cluster"
|
||||||
|
"github.com/cloudreve/Cloudreve/v3/pkg/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
// RecycleTask 文件回收任务
|
||||||
|
type RecycleTask struct {
|
||||||
|
User *model.User
|
||||||
|
TaskModel *model.Task
|
||||||
|
TaskProps RecycleProps
|
||||||
|
Err *JobError
|
||||||
|
}
|
||||||
|
|
||||||
|
// RecycleProps 回收任务属性
|
||||||
|
type RecycleProps struct {
|
||||||
|
// 下载任务 GID
|
||||||
|
DownloadGID string `json:"download_gid"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Props 获取任务属性
|
||||||
|
func (job *RecycleTask) Props() string {
|
||||||
|
res, _ := json.Marshal(job.TaskProps)
|
||||||
|
return string(res)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Type 获取任务状态
|
||||||
|
func (job *RecycleTask) Type() int {
|
||||||
|
return RecycleTaskType
|
||||||
|
}
|
||||||
|
|
||||||
|
// Creator 获取创建者ID
|
||||||
|
func (job *RecycleTask) Creator() uint {
|
||||||
|
return job.User.ID
|
||||||
|
}
|
||||||
|
|
||||||
|
// Model 获取任务的数据库模型
|
||||||
|
func (job *RecycleTask) Model() *model.Task {
|
||||||
|
return job.TaskModel
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetStatus 设定状态
|
||||||
|
func (job *RecycleTask) SetStatus(status int) {
|
||||||
|
job.TaskModel.SetStatus(status)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetError 设定任务失败信息
|
||||||
|
func (job *RecycleTask) SetError(err *JobError) {
|
||||||
|
job.Err = err
|
||||||
|
res, _ := json.Marshal(job.Err)
|
||||||
|
job.TaskModel.SetError(string(res))
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetErrorMsg 设定任务失败信息
|
||||||
|
func (job *RecycleTask) SetErrorMsg(msg string, err error) {
|
||||||
|
jobErr := &JobError{Msg: msg}
|
||||||
|
if err != nil {
|
||||||
|
jobErr.Error = err.Error()
|
||||||
|
}
|
||||||
|
job.SetError(jobErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetError 返回任务失败信息
|
||||||
|
func (job *RecycleTask) GetError() *JobError {
|
||||||
|
return job.Err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do 开始执行任务
|
||||||
|
func (job *RecycleTask) Do() {
|
||||||
|
download, err := model.GetDownloadByGid(job.TaskProps.DownloadGID, job.User.ID)
|
||||||
|
if err != nil {
|
||||||
|
util.Log().Warning("回收任务 %d 找不到下载记录", job.TaskModel.ID)
|
||||||
|
job.SetErrorMsg("无法找到下载任务", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
nodeID := download.GetNodeID()
|
||||||
|
node := cluster.Default.GetNodeByID(nodeID)
|
||||||
|
if node == nil {
|
||||||
|
util.Log().Warning("回收任务 %d 找不到节点", job.TaskModel.ID)
|
||||||
|
job.SetErrorMsg("从机节点不可用", nil)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
err = node.GetAria2Instance().DeleteTempFile(download)
|
||||||
|
if err != nil {
|
||||||
|
util.Log().Warning("无法删除中转临时目录[%s], %s", download.Parent, err)
|
||||||
|
job.SetErrorMsg("文件回收失败", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewRecycleTask 新建回收任务
|
||||||
|
func NewRecycleTask(download *model.Download) (Job, error) {
|
||||||
|
newTask := &RecycleTask{
|
||||||
|
User: download.GetOwner(),
|
||||||
|
TaskProps: RecycleProps{
|
||||||
|
DownloadGID: download.GID,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
record, err := Record(newTask)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
newTask.TaskModel = record
|
||||||
|
|
||||||
|
return newTask, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewRecycleTaskFromModel 从数据库记录中恢复回收任务
|
||||||
|
func NewRecycleTaskFromModel(task *model.Task) (Job, error) {
|
||||||
|
user, err := model.GetActiveUserByID(task.UserID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
newTask := &RecycleTask{
|
||||||
|
User: &user,
|
||||||
|
TaskModel: task,
|
||||||
|
}
|
||||||
|
|
||||||
|
err = json.Unmarshal([]byte(task.Props), &newTask.TaskProps)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return newTask, nil
|
||||||
|
}
|
@ -0,0 +1,117 @@
|
|||||||
|
package task
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/DATA-DOG/go-sqlmock"
|
||||||
|
model "github.com/cloudreve/Cloudreve/v3/models"
|
||||||
|
"github.com/jinzhu/gorm"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestRecycleTask_Props(t *testing.T) {
|
||||||
|
asserts := assert.New(t)
|
||||||
|
task := &RecycleTask{
|
||||||
|
User: &model.User{},
|
||||||
|
}
|
||||||
|
asserts.NotEmpty(task.Props())
|
||||||
|
asserts.Equal(RecycleTaskType, task.Type())
|
||||||
|
asserts.EqualValues(0, task.Creator())
|
||||||
|
asserts.Nil(task.Model())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRecycleTask_SetStatus(t *testing.T) {
|
||||||
|
asserts := assert.New(t)
|
||||||
|
task := &RecycleTask{
|
||||||
|
User: &model.User{},
|
||||||
|
TaskModel: &model.Task{
|
||||||
|
Model: gorm.Model{ID: 1},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
mock.ExpectBegin()
|
||||||
|
mock.ExpectExec("UPDATE(.+)").WillReturnResult(sqlmock.NewResult(1, 1))
|
||||||
|
mock.ExpectCommit()
|
||||||
|
task.SetStatus(3)
|
||||||
|
asserts.NoError(mock.ExpectationsWereMet())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRecycleTask_SetError(t *testing.T) {
|
||||||
|
asserts := assert.New(t)
|
||||||
|
task := &RecycleTask{
|
||||||
|
User: &model.User{},
|
||||||
|
TaskModel: &model.Task{
|
||||||
|
Model: gorm.Model{ID: 1},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
mock.ExpectBegin()
|
||||||
|
mock.ExpectExec("UPDATE(.+)").WillReturnResult(sqlmock.NewResult(1, 1))
|
||||||
|
mock.ExpectCommit()
|
||||||
|
|
||||||
|
task.SetErrorMsg("error", nil)
|
||||||
|
asserts.NoError(mock.ExpectationsWereMet())
|
||||||
|
asserts.Equal("error", task.GetError().Msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewRecycleTask(t *testing.T) {
|
||||||
|
asserts := assert.New(t)
|
||||||
|
|
||||||
|
// 成功
|
||||||
|
{
|
||||||
|
mock.ExpectQuery("SELECT(.+)").WillReturnRows(sqlmock.NewRows([]string{"id"}).AddRow(1))
|
||||||
|
mock.ExpectBegin()
|
||||||
|
mock.ExpectExec("INSERT(.+)").WillReturnResult(sqlmock.NewResult(1, 1))
|
||||||
|
mock.ExpectCommit()
|
||||||
|
job, err := NewRecycleTask(&model.Download{
|
||||||
|
Model: gorm.Model{ID: 1},
|
||||||
|
GID: "test_g_id",
|
||||||
|
Parent: "/",
|
||||||
|
UserID: 1,
|
||||||
|
NodeID: 1,
|
||||||
|
})
|
||||||
|
asserts.NoError(mock.ExpectationsWereMet())
|
||||||
|
asserts.NotNil(job)
|
||||||
|
asserts.NoError(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 失败
|
||||||
|
{
|
||||||
|
mock.ExpectQuery("SELECT(.+)").WillReturnRows(sqlmock.NewRows([]string{"id"}).AddRow(1))
|
||||||
|
mock.ExpectBegin()
|
||||||
|
mock.ExpectExec("INSERT(.+)").WillReturnError(errors.New("error"))
|
||||||
|
mock.ExpectRollback()
|
||||||
|
job, err := NewRecycleTask(&model.Download{
|
||||||
|
Model: gorm.Model{ID: 1},
|
||||||
|
GID: "test_g_id",
|
||||||
|
Parent: "test/not_exist",
|
||||||
|
UserID: 1,
|
||||||
|
NodeID: 1,
|
||||||
|
})
|
||||||
|
asserts.NoError(mock.ExpectationsWereMet())
|
||||||
|
asserts.Nil(job)
|
||||||
|
asserts.Error(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewRecycleTaskFromModel(t *testing.T) {
|
||||||
|
asserts := assert.New(t)
|
||||||
|
|
||||||
|
// 成功
|
||||||
|
{
|
||||||
|
mock.ExpectQuery("SELECT(.+)").WillReturnRows(sqlmock.NewRows([]string{"id"}).AddRow(1))
|
||||||
|
job, err := NewRecycleTaskFromModel(&model.Task{Props: "{}"})
|
||||||
|
asserts.NoError(mock.ExpectationsWereMet())
|
||||||
|
asserts.NoError(err)
|
||||||
|
asserts.NotNil(job)
|
||||||
|
}
|
||||||
|
|
||||||
|
// JSON解析失败
|
||||||
|
{
|
||||||
|
mock.ExpectQuery("SELECT(.+)").WillReturnRows(sqlmock.NewRows([]string{"id"}).AddRow(1))
|
||||||
|
job, err := NewRecycleTaskFromModel(&model.Task{Props: "?"})
|
||||||
|
asserts.NoError(mock.ExpectationsWereMet())
|
||||||
|
asserts.Error(err)
|
||||||
|
asserts.Nil(job)
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue