|
|
|
package aria2
|
|
|
|
|
|
|
|
import (
|
|
|
|
model "github.com/cloudreve/Cloudreve/v3/models"
|
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/aria2"
|
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/aria2/common"
|
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/aria2/monitor"
|
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/cluster"
|
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/filesystem"
|
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/mq"
|
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/serializer"
|
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/util"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
)
|
|
|
|
|
|
|
|
// AddURLService 添加URL离线下载服务
|
|
|
|
type BatchAddURLService struct {
|
|
|
|
URLs []string `json:"url" binding:"required"`
|
|
|
|
Dst string `json:"dst" binding:"required,min=1"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add 主机批量创建新的链接离线下载任务
|
|
|
|
func (service *BatchAddURLService) Add(c *gin.Context, taskType int) serializer.Response {
|
|
|
|
// 创建文件系统
|
|
|
|
fs, err := filesystem.NewFileSystemFromContext(c)
|
|
|
|
if err != nil {
|
|
|
|
return serializer.Err(serializer.CodeCreateFSError, "", err)
|
|
|
|
}
|
|
|
|
defer fs.Recycle()
|
|
|
|
|
|
|
|
// 检查用户组权限
|
|
|
|
if !fs.User.Group.OptionsSerialized.Aria2 {
|
|
|
|
return serializer.Err(serializer.CodeGroupNotAllowed, "", nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 存放目录是否存在
|
|
|
|
if exist, _ := fs.IsPathExist(service.Dst); !exist {
|
|
|
|
return serializer.Err(serializer.CodeParentNotExist, "", nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 检查批量任务数量
|
|
|
|
limit := fs.User.Group.OptionsSerialized.Aria2BatchSize
|
|
|
|
if limit > 0 && len(service.URLs) > limit {
|
|
|
|
return serializer.Err(serializer.CodeBatchAria2Size, "", nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
res := make([]serializer.Response, 0, len(service.URLs))
|
|
|
|
for _, target := range service.URLs {
|
|
|
|
subService := &AddURLService{
|
|
|
|
URL: target,
|
|
|
|
Dst: service.Dst,
|
|
|
|
}
|
|
|
|
|
|
|
|
addRes := subService.Add(c, fs, taskType)
|
|
|
|
res = append(res, addRes)
|
|
|
|
}
|
|
|
|
|
|
|
|
return serializer.Response{Data: res}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddURLService 添加URL离线下载服务
|
|
|
|
type AddURLService struct {
|
|
|
|
URL string `json:"url" binding:"required"`
|
|
|
|
Dst string `json:"dst" binding:"required,min=1"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add 主机创建新的链接离线下载任务
|
|
|
|
func (service *AddURLService) Add(c *gin.Context, fs *filesystem.FileSystem, taskType int) serializer.Response {
|
|
|
|
if fs == nil {
|
|
|
|
var err error
|
|
|
|
// 创建文件系统
|
|
|
|
fs, err = filesystem.NewFileSystemFromContext(c)
|
|
|
|
if err != nil {
|
|
|
|
return serializer.Err(serializer.CodeCreateFSError, "", err)
|
|
|
|
}
|
|
|
|
defer fs.Recycle()
|
|
|
|
|
|
|
|
// 检查用户组权限
|
|
|
|
if !fs.User.Group.OptionsSerialized.Aria2 {
|
|
|
|
return serializer.Err(serializer.CodeGroupNotAllowed, "", nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 存放目录是否存在
|
|
|
|
if exist, _ := fs.IsPathExist(service.Dst); !exist {
|
|
|
|
return serializer.Err(serializer.CodeParentNotExist, "", nil)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
downloads := model.GetDownloadsByStatusAndUser(0, fs.User.ID, common.Downloading, common.Paused, common.Ready)
|
|
|
|
limit := fs.User.Group.OptionsSerialized.Aria2BatchSize
|
|
|
|
if limit > 0 && len(downloads)+1 > limit {
|
|
|
|
return serializer.Err(serializer.CodeBatchAria2Size, "", nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 创建任务
|
|
|
|
task := &model.Download{
|
|
|
|
Status: common.Ready,
|
|
|
|
Type: taskType,
|
|
|
|
Dst: service.Dst,
|
|
|
|
UserID: fs.User.ID,
|
|
|
|
Source: service.URL,
|
|
|
|
}
|
|
|
|
|
|
|
|
// 获取 Aria2 负载均衡器
|
|
|
|
lb := aria2.GetLoadBalancer()
|
|
|
|
|
|
|
|
// 获取 Aria2 实例
|
|
|
|
err, node := cluster.Default.BalanceNodeByFeature("aria2", lb)
|
|
|
|
if err != nil {
|
|
|
|
return serializer.Err(serializer.CodeInternalSetting, "Failed to get Aria2 instance", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 创建任务
|
|
|
|
gid, err := node.GetAria2Instance().CreateTask(task, fs.User.Group.OptionsSerialized.Aria2Options)
|
|
|
|
if err != nil {
|
|
|
|
return serializer.Err(serializer.CodeCreateTaskError, "", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
task.GID = gid
|
|
|
|
task.NodeID = node.ID()
|
|
|
|
_, err = task.Create()
|
|
|
|
if err != nil {
|
|
|
|
return serializer.DBErr("Failed to create task record", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 创建任务监控
|
|
|
|
monitor.NewMonitor(task, cluster.Default, mq.GlobalMQ)
|
|
|
|
|
|
|
|
return serializer.Response{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add 从机创建新的链接离线下载任务
|
|
|
|
func Add(c *gin.Context, service *serializer.SlaveAria2Call) serializer.Response {
|
|
|
|
caller, _ := c.Get("MasterAria2Instance")
|
|
|
|
|
|
|
|
// 创建任务
|
|
|
|
gid, err := caller.(common.Aria2).CreateTask(service.Task, service.GroupOptions)
|
|
|
|
if err != nil {
|
|
|
|
return serializer.Err(serializer.CodeInternalSetting, "Failed to create aria2 task", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 创建事件通知回调
|
|
|
|
siteID, _ := c.Get("MasterSiteID")
|
|
|
|
mq.GlobalMQ.SubscribeCallback(gid, func(message mq.Message) {
|
|
|
|
if err := cluster.DefaultController.SendNotification(siteID.(string), message.TriggeredBy, message); err != nil {
|
|
|
|
util.Log().Warning("Failed to send remote download task status change notifications: %s", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return serializer.Response{Data: gid}
|
|
|
|
}
|