mirror of https://github.com/Xhofe/alist
* feat: support url tree writing * fix: meta writable * feat: disable writable via additionpull/7716/head^2
parent
25b4b55ee1
commit
51bcf83511
|
@ -2,7 +2,11 @@ package url_tree
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
|
"github.com/alist-org/alist/v3/internal/op"
|
||||||
stdpath "path"
|
stdpath "path"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
"github.com/alist-org/alist/v3/internal/driver"
|
"github.com/alist-org/alist/v3/internal/driver"
|
||||||
"github.com/alist-org/alist/v3/internal/errs"
|
"github.com/alist-org/alist/v3/internal/errs"
|
||||||
|
@ -14,7 +18,8 @@ import (
|
||||||
type Urls struct {
|
type Urls struct {
|
||||||
model.Storage
|
model.Storage
|
||||||
Addition
|
Addition
|
||||||
root *Node
|
root *Node
|
||||||
|
mutex sync.RWMutex
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *Urls) Config() driver.Config {
|
func (d *Urls) Config() driver.Config {
|
||||||
|
@ -40,11 +45,15 @@ func (d *Urls) Drop(ctx context.Context) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *Urls) Get(ctx context.Context, path string) (model.Obj, error) {
|
func (d *Urls) Get(ctx context.Context, path string) (model.Obj, error) {
|
||||||
|
d.mutex.RLock()
|
||||||
|
defer d.mutex.RUnlock()
|
||||||
node := GetNodeFromRootByPath(d.root, path)
|
node := GetNodeFromRootByPath(d.root, path)
|
||||||
return nodeToObj(node, path)
|
return nodeToObj(node, path)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *Urls) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) {
|
func (d *Urls) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) {
|
||||||
|
d.mutex.RLock()
|
||||||
|
defer d.mutex.RUnlock()
|
||||||
node := GetNodeFromRootByPath(d.root, dir.GetPath())
|
node := GetNodeFromRootByPath(d.root, dir.GetPath())
|
||||||
log.Debugf("path: %s, node: %+v", dir.GetPath(), node)
|
log.Debugf("path: %s, node: %+v", dir.GetPath(), node)
|
||||||
if node == nil {
|
if node == nil {
|
||||||
|
@ -59,6 +68,8 @@ func (d *Urls) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *Urls) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) {
|
func (d *Urls) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) {
|
||||||
|
d.mutex.RLock()
|
||||||
|
defer d.mutex.RUnlock()
|
||||||
node := GetNodeFromRootByPath(d.root, file.GetPath())
|
node := GetNodeFromRootByPath(d.root, file.GetPath())
|
||||||
log.Debugf("path: %s, node: %+v", file.GetPath(), node)
|
log.Debugf("path: %s, node: %+v", file.GetPath(), node)
|
||||||
if node == nil {
|
if node == nil {
|
||||||
|
@ -72,6 +83,174 @@ func (d *Urls) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*
|
||||||
return nil, errs.NotFile
|
return nil, errs.NotFile
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (d *Urls) MakeDir(ctx context.Context, parentDir model.Obj, dirName string) (model.Obj, error) {
|
||||||
|
if !d.Writable {
|
||||||
|
return nil, errs.PermissionDenied
|
||||||
|
}
|
||||||
|
d.mutex.Lock()
|
||||||
|
defer d.mutex.Unlock()
|
||||||
|
node := GetNodeFromRootByPath(d.root, parentDir.GetPath())
|
||||||
|
if node == nil {
|
||||||
|
return nil, errs.ObjectNotFound
|
||||||
|
}
|
||||||
|
if node.isFile() {
|
||||||
|
return nil, errs.NotFolder
|
||||||
|
}
|
||||||
|
dir := &Node{
|
||||||
|
Name: dirName,
|
||||||
|
Level: node.Level + 1,
|
||||||
|
}
|
||||||
|
node.Children = append(node.Children, dir)
|
||||||
|
d.updateStorage()
|
||||||
|
return nodeToObj(dir, stdpath.Join(parentDir.GetPath(), dirName))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Urls) Move(ctx context.Context, srcObj, dstDir model.Obj) (model.Obj, error) {
|
||||||
|
if !d.Writable {
|
||||||
|
return nil, errs.PermissionDenied
|
||||||
|
}
|
||||||
|
if strings.HasPrefix(dstDir.GetPath(), srcObj.GetPath()) {
|
||||||
|
return nil, errors.New("cannot move parent dir to child")
|
||||||
|
}
|
||||||
|
d.mutex.Lock()
|
||||||
|
defer d.mutex.Unlock()
|
||||||
|
dstNode := GetNodeFromRootByPath(d.root, dstDir.GetPath())
|
||||||
|
if dstNode == nil || dstNode.isFile() {
|
||||||
|
return nil, errs.NotFolder
|
||||||
|
}
|
||||||
|
srcDir, srcName := stdpath.Split(srcObj.GetPath())
|
||||||
|
srcParentNode := GetNodeFromRootByPath(d.root, srcDir)
|
||||||
|
if srcParentNode == nil {
|
||||||
|
return nil, errs.ObjectNotFound
|
||||||
|
}
|
||||||
|
newChildren := make([]*Node, 0, len(srcParentNode.Children))
|
||||||
|
var srcNode *Node
|
||||||
|
for _, child := range srcParentNode.Children {
|
||||||
|
if child.Name == srcName {
|
||||||
|
srcNode = child
|
||||||
|
} else {
|
||||||
|
newChildren = append(newChildren, child)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if srcNode == nil {
|
||||||
|
return nil, errs.ObjectNotFound
|
||||||
|
}
|
||||||
|
srcParentNode.Children = newChildren
|
||||||
|
srcNode.setLevel(dstNode.Level + 1)
|
||||||
|
dstNode.Children = append(dstNode.Children, srcNode)
|
||||||
|
d.root.calSize()
|
||||||
|
d.updateStorage()
|
||||||
|
return nodeToObj(srcNode, stdpath.Join(dstDir.GetPath(), srcName))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Urls) Rename(ctx context.Context, srcObj model.Obj, newName string) (model.Obj, error) {
|
||||||
|
if !d.Writable {
|
||||||
|
return nil, errs.PermissionDenied
|
||||||
|
}
|
||||||
|
d.mutex.Lock()
|
||||||
|
defer d.mutex.Unlock()
|
||||||
|
srcNode := GetNodeFromRootByPath(d.root, srcObj.GetPath())
|
||||||
|
if srcNode == nil {
|
||||||
|
return nil, errs.ObjectNotFound
|
||||||
|
}
|
||||||
|
srcNode.Name = newName
|
||||||
|
d.updateStorage()
|
||||||
|
return nodeToObj(srcNode, stdpath.Join(stdpath.Dir(srcObj.GetPath()), newName))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Urls) Copy(ctx context.Context, srcObj, dstDir model.Obj) (model.Obj, error) {
|
||||||
|
if !d.Writable {
|
||||||
|
return nil, errs.PermissionDenied
|
||||||
|
}
|
||||||
|
if strings.HasPrefix(dstDir.GetPath(), srcObj.GetPath()) {
|
||||||
|
return nil, errors.New("cannot copy parent dir to child")
|
||||||
|
}
|
||||||
|
d.mutex.Lock()
|
||||||
|
defer d.mutex.Unlock()
|
||||||
|
dstNode := GetNodeFromRootByPath(d.root, dstDir.GetPath())
|
||||||
|
if dstNode == nil || dstNode.isFile() {
|
||||||
|
return nil, errs.NotFolder
|
||||||
|
}
|
||||||
|
srcNode := GetNodeFromRootByPath(d.root, srcObj.GetPath())
|
||||||
|
if srcNode == nil {
|
||||||
|
return nil, errs.ObjectNotFound
|
||||||
|
}
|
||||||
|
newNode := srcNode.deepCopy(dstNode.Level + 1)
|
||||||
|
dstNode.Children = append(dstNode.Children, newNode)
|
||||||
|
d.root.calSize()
|
||||||
|
d.updateStorage()
|
||||||
|
return nodeToObj(newNode, stdpath.Join(dstDir.GetPath(), stdpath.Base(srcObj.GetPath())))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Urls) Remove(ctx context.Context, obj model.Obj) error {
|
||||||
|
if !d.Writable {
|
||||||
|
return errs.PermissionDenied
|
||||||
|
}
|
||||||
|
d.mutex.Lock()
|
||||||
|
defer d.mutex.Unlock()
|
||||||
|
objDir, objName := stdpath.Split(obj.GetPath())
|
||||||
|
nodeParent := GetNodeFromRootByPath(d.root, objDir)
|
||||||
|
if nodeParent == nil {
|
||||||
|
return errs.ObjectNotFound
|
||||||
|
}
|
||||||
|
newChildren := make([]*Node, 0, len(nodeParent.Children))
|
||||||
|
var deletedObj *Node
|
||||||
|
for _, child := range nodeParent.Children {
|
||||||
|
if child.Name != objName {
|
||||||
|
newChildren = append(newChildren, child)
|
||||||
|
} else {
|
||||||
|
deletedObj = child
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if deletedObj == nil {
|
||||||
|
return errs.ObjectNotFound
|
||||||
|
}
|
||||||
|
nodeParent.Children = newChildren
|
||||||
|
if deletedObj.Size > 0 {
|
||||||
|
d.root.calSize()
|
||||||
|
}
|
||||||
|
d.updateStorage()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Urls) PutURL(ctx context.Context, dstDir model.Obj, name, url string) (model.Obj, error) {
|
||||||
|
if !d.Writable {
|
||||||
|
return nil, errs.PermissionDenied
|
||||||
|
}
|
||||||
|
d.mutex.Lock()
|
||||||
|
defer d.mutex.Unlock()
|
||||||
|
dirNode := GetNodeFromRootByPath(d.root, dstDir.GetPath())
|
||||||
|
if dirNode == nil || dirNode.isFile() {
|
||||||
|
return nil, errs.NotFolder
|
||||||
|
}
|
||||||
|
newNode := &Node{
|
||||||
|
Name: name,
|
||||||
|
Level: dirNode.Level + 1,
|
||||||
|
Url: url,
|
||||||
|
}
|
||||||
|
dirNode.Children = append(dirNode.Children, newNode)
|
||||||
|
if d.HeadSize {
|
||||||
|
size, err := getSizeFromUrl(url)
|
||||||
|
if err != nil {
|
||||||
|
log.Errorf("get size from url error: %s", err)
|
||||||
|
} else {
|
||||||
|
newNode.Size = size
|
||||||
|
d.root.calSize()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
d.updateStorage()
|
||||||
|
return nodeToObj(newNode, stdpath.Join(dstDir.GetPath(), name))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Urls) Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up driver.UpdateProgress) error {
|
||||||
|
return errs.UploadNotSupported
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Urls) updateStorage() {
|
||||||
|
d.UrlStructure = StringifyTree(d.root)
|
||||||
|
op.MustSaveDriverStorage(d)
|
||||||
|
}
|
||||||
|
|
||||||
//func (d *Template) Other(ctx context.Context, args model.OtherArgs) (interface{}, error) {
|
//func (d *Template) Other(ctx context.Context, args model.OtherArgs) (interface{}, error) {
|
||||||
// return nil, errs.NotSupport
|
// return nil, errs.NotSupport
|
||||||
//}
|
//}
|
||||||
|
|
|
@ -12,6 +12,7 @@ type Addition struct {
|
||||||
// define other
|
// define other
|
||||||
UrlStructure string `json:"url_structure" type:"text" required:"true" default:"https://jsd.nn.ci/gh/alist-org/alist/README.md\nhttps://jsd.nn.ci/gh/alist-org/alist/README_cn.md\nfolder:\n CONTRIBUTING.md:1635:https://jsd.nn.ci/gh/alist-org/alist/CONTRIBUTING.md\n CODE_OF_CONDUCT.md:2093:https://jsd.nn.ci/gh/alist-org/alist/CODE_OF_CONDUCT.md" help:"structure:FolderName:\n [FileName:][FileSize:][Modified:]Url"`
|
UrlStructure string `json:"url_structure" type:"text" required:"true" default:"https://jsd.nn.ci/gh/alist-org/alist/README.md\nhttps://jsd.nn.ci/gh/alist-org/alist/README_cn.md\nfolder:\n CONTRIBUTING.md:1635:https://jsd.nn.ci/gh/alist-org/alist/CONTRIBUTING.md\n CODE_OF_CONDUCT.md:2093:https://jsd.nn.ci/gh/alist-org/alist/CODE_OF_CONDUCT.md" help:"structure:FolderName:\n [FileName:][FileSize:][Modified:]Url"`
|
||||||
HeadSize bool `json:"head_size" type:"bool" default:"false" help:"Use head method to get file size, but it may be failed."`
|
HeadSize bool `json:"head_size" type:"bool" default:"false" help:"Use head method to get file size, but it may be failed."`
|
||||||
|
Writable bool `json:"writable" type:"bool" default:"false"`
|
||||||
}
|
}
|
||||||
|
|
||||||
var config = driver.Config{
|
var config = driver.Config{
|
||||||
|
@ -20,7 +21,7 @@ var config = driver.Config{
|
||||||
OnlyLocal: false,
|
OnlyLocal: false,
|
||||||
OnlyProxy: false,
|
OnlyProxy: false,
|
||||||
NoCache: true,
|
NoCache: true,
|
||||||
NoUpload: true,
|
NoUpload: false,
|
||||||
NeedMs: false,
|
NeedMs: false,
|
||||||
DefaultRoot: "",
|
DefaultRoot: "",
|
||||||
CheckStatus: true,
|
CheckStatus: true,
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
package url_tree
|
package url_tree
|
||||||
|
|
||||||
|
import "github.com/alist-org/alist/v3/pkg/utils"
|
||||||
|
|
||||||
// Node is a node in the folder tree
|
// Node is a node in the folder tree
|
||||||
type Node struct {
|
type Node struct {
|
||||||
Url string
|
Url string
|
||||||
|
@ -44,3 +46,19 @@ func (node *Node) calSize() int64 {
|
||||||
node.Size = size
|
node.Size = size
|
||||||
return size
|
return size
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (node *Node) setLevel(level int) {
|
||||||
|
node.Level = level
|
||||||
|
for _, child := range node.Children {
|
||||||
|
child.setLevel(level + 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (node *Node) deepCopy(level int) *Node {
|
||||||
|
ret := *node
|
||||||
|
ret.Level = level
|
||||||
|
ret.Children, _ = utils.SliceConvert(ret.Children, func(child *Node) (*Node, error) {
|
||||||
|
return child.deepCopy(level + 1), nil
|
||||||
|
})
|
||||||
|
return &ret
|
||||||
|
}
|
||||||
|
|
|
@ -153,6 +153,9 @@ func splitPath(path string) []string {
|
||||||
if path == "/" {
|
if path == "/" {
|
||||||
return []string{"root"}
|
return []string{"root"}
|
||||||
}
|
}
|
||||||
|
if strings.HasSuffix(path, "/") {
|
||||||
|
path = path[:len(path)-1]
|
||||||
|
}
|
||||||
parts := strings.Split(path, "/")
|
parts := strings.Split(path, "/")
|
||||||
parts[0] = "root"
|
parts[0] = "root"
|
||||||
return parts
|
return parts
|
||||||
|
@ -190,3 +193,46 @@ func getSizeFromUrl(url string) (int64, error) {
|
||||||
}
|
}
|
||||||
return size, nil
|
return size, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func StringifyTree(node *Node) string {
|
||||||
|
sb := strings.Builder{}
|
||||||
|
if node.Level == -1 {
|
||||||
|
for i, child := range node.Children {
|
||||||
|
sb.WriteString(StringifyTree(child))
|
||||||
|
if i < len(node.Children)-1 {
|
||||||
|
sb.WriteString("\n")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return sb.String()
|
||||||
|
}
|
||||||
|
for i := 0; i < node.Level; i++ {
|
||||||
|
sb.WriteString(" ")
|
||||||
|
}
|
||||||
|
if node.Url == "" {
|
||||||
|
sb.WriteString(node.Name)
|
||||||
|
sb.WriteString(":")
|
||||||
|
for _, child := range node.Children {
|
||||||
|
sb.WriteString("\n")
|
||||||
|
sb.WriteString(StringifyTree(child))
|
||||||
|
}
|
||||||
|
} else if node.Size == 0 && node.Modified == 0 {
|
||||||
|
if stdpath.Base(node.Url) == node.Name {
|
||||||
|
sb.WriteString(node.Url)
|
||||||
|
} else {
|
||||||
|
sb.WriteString(fmt.Sprintf("%s:%s", node.Name, node.Url))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
sb.WriteString(node.Name)
|
||||||
|
sb.WriteString(":")
|
||||||
|
if node.Size != 0 || node.Modified != 0 {
|
||||||
|
sb.WriteString(strconv.FormatInt(node.Size, 10))
|
||||||
|
sb.WriteString(":")
|
||||||
|
}
|
||||||
|
if node.Modified != 0 {
|
||||||
|
sb.WriteString(strconv.FormatInt(node.Modified, 10))
|
||||||
|
sb.WriteString(":")
|
||||||
|
}
|
||||||
|
sb.WriteString(node.Url)
|
||||||
|
}
|
||||||
|
return sb.String()
|
||||||
|
}
|
||||||
|
|
|
@ -80,6 +80,13 @@ type Put interface {
|
||||||
Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up UpdateProgress) error
|
Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up UpdateProgress) error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type PutURL interface {
|
||||||
|
// PutURL directly put a URL into the storage
|
||||||
|
// Applicable to index-based drivers like URL-Tree or drivers that support uploading files as URLs
|
||||||
|
// Called when using SimpleHttp for offline downloading, skipping creating a download task
|
||||||
|
PutURL(ctx context.Context, dstDir model.Obj, name, url string) error
|
||||||
|
}
|
||||||
|
|
||||||
//type WriteResult interface {
|
//type WriteResult interface {
|
||||||
// MkdirResult
|
// MkdirResult
|
||||||
// MoveResult
|
// MoveResult
|
||||||
|
@ -109,6 +116,13 @@ type PutResult interface {
|
||||||
Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up UpdateProgress) (model.Obj, error)
|
Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up UpdateProgress) (model.Obj, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type PutURLResult interface {
|
||||||
|
// PutURL directly put a URL into the storage
|
||||||
|
// Applicable to index-based drivers like URL-Tree or drivers that support uploading files as URLs
|
||||||
|
// Called when using SimpleHttp for offline downloading, skipping creating a download task
|
||||||
|
PutURL(ctx context.Context, dstDir model.Obj, name, url string) (model.Obj, error)
|
||||||
|
}
|
||||||
|
|
||||||
type UpdateProgress func(percentage float64)
|
type UpdateProgress func(percentage float64)
|
||||||
|
|
||||||
type Progress struct {
|
type Progress struct {
|
||||||
|
|
|
@ -2,8 +2,11 @@ package tool
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"github.com/alist-org/alist/v3/internal/driver"
|
||||||
"github.com/alist-org/alist/v3/internal/model"
|
"github.com/alist-org/alist/v3/internal/model"
|
||||||
"github.com/alist-org/alist/v3/internal/task"
|
"github.com/alist-org/alist/v3/internal/task"
|
||||||
|
"net/url"
|
||||||
|
"path"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
||||||
"github.com/alist-org/alist/v3/internal/conf"
|
"github.com/alist-org/alist/v3/internal/conf"
|
||||||
|
@ -30,18 +33,6 @@ type AddURLArgs struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func AddURL(ctx context.Context, args *AddURLArgs) (task.TaskExtensionInfo, error) {
|
func AddURL(ctx context.Context, args *AddURLArgs) (task.TaskExtensionInfo, error) {
|
||||||
// get tool
|
|
||||||
tool, err := Tools.Get(args.Tool)
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.Wrapf(err, "failed get tool")
|
|
||||||
}
|
|
||||||
// check tool is ready
|
|
||||||
if !tool.IsReady() {
|
|
||||||
// try to init tool
|
|
||||||
if _, err := tool.Init(); err != nil {
|
|
||||||
return nil, errors.Wrapf(err, "failed init tool %s", args.Tool)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// check storage
|
// check storage
|
||||||
storage, dstDirActualPath, err := op.GetStorageAndActualPath(args.DstDirPath)
|
storage, dstDirActualPath, err := op.GetStorageAndActualPath(args.DstDirPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -63,6 +54,23 @@ func AddURL(ctx context.Context, args *AddURLArgs) (task.TaskExtensionInfo, erro
|
||||||
return nil, errors.WithStack(errs.NotFolder)
|
return nil, errors.WithStack(errs.NotFolder)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// try putting url
|
||||||
|
if args.Tool == "SimpleHttp" && tryPutUrl(ctx, storage, dstDirActualPath, args.URL) {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// get tool
|
||||||
|
tool, err := Tools.Get(args.Tool)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrapf(err, "failed get tool")
|
||||||
|
}
|
||||||
|
// check tool is ready
|
||||||
|
if !tool.IsReady() {
|
||||||
|
// try to init tool
|
||||||
|
if _, err := tool.Init(); err != nil {
|
||||||
|
return nil, errors.Wrapf(err, "failed init tool %s", args.Tool)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
uid := uuid.NewString()
|
uid := uuid.NewString()
|
||||||
tempDir := filepath.Join(conf.Conf.TempDir, args.Tool, uid)
|
tempDir := filepath.Join(conf.Conf.TempDir, args.Tool, uid)
|
||||||
|
@ -98,3 +106,18 @@ func AddURL(ctx context.Context, args *AddURLArgs) (task.TaskExtensionInfo, erro
|
||||||
DownloadTaskManager.Add(t)
|
DownloadTaskManager.Add(t)
|
||||||
return t, nil
|
return t, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func tryPutUrl(ctx context.Context, storage driver.Driver, dstDirActualPath, urlStr string) bool {
|
||||||
|
_, ok := storage.(driver.PutURL)
|
||||||
|
_, okResult := storage.(driver.PutURLResult)
|
||||||
|
if !ok && !okResult {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
u, err := url.Parse(urlStr)
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
dstName := path.Base(u.Path)
|
||||||
|
err = op.PutURL(ctx, storage, dstDirActualPath, dstName, urlStr)
|
||||||
|
return err == nil
|
||||||
|
}
|
||||||
|
|
|
@ -586,3 +586,43 @@ func Put(ctx context.Context, storage driver.Driver, dstDirPath string, file mod
|
||||||
}
|
}
|
||||||
return errors.WithStack(err)
|
return errors.WithStack(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func PutURL(ctx context.Context, storage driver.Driver, dstDirPath, dstName, url string, lazyCache ...bool) error {
|
||||||
|
if storage.Config().CheckStatus && storage.GetStorage().Status != WORK {
|
||||||
|
return errors.Errorf("storage not init: %s", storage.GetStorage().Status)
|
||||||
|
}
|
||||||
|
dstDirPath = utils.FixAndCleanPath(dstDirPath)
|
||||||
|
_, err := GetUnwrap(ctx, storage, stdpath.Join(dstDirPath, dstName))
|
||||||
|
if err == nil {
|
||||||
|
return errors.New("obj already exists")
|
||||||
|
}
|
||||||
|
err = MakeDir(ctx, storage, dstDirPath)
|
||||||
|
if err != nil {
|
||||||
|
return errors.WithMessagef(err, "failed to put url")
|
||||||
|
}
|
||||||
|
dstDir, err := GetUnwrap(ctx, storage, dstDirPath)
|
||||||
|
if err != nil {
|
||||||
|
return errors.WithMessagef(err, "failed to put url")
|
||||||
|
}
|
||||||
|
switch s := storage.(type) {
|
||||||
|
case driver.PutURLResult:
|
||||||
|
var newObj model.Obj
|
||||||
|
newObj, err = s.PutURL(ctx, dstDir, dstName, url)
|
||||||
|
if err == nil {
|
||||||
|
if newObj != nil {
|
||||||
|
addCacheObj(storage, dstDirPath, model.WrapObjName(newObj))
|
||||||
|
} else if !utils.IsBool(lazyCache...) {
|
||||||
|
ClearCache(storage, dstDirPath)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case driver.PutURL:
|
||||||
|
err = s.PutURL(ctx, dstDir, dstName, url)
|
||||||
|
if err == nil && !utils.IsBool(lazyCache...) {
|
||||||
|
ClearCache(storage, dstDirPath)
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return errs.NotImplement
|
||||||
|
}
|
||||||
|
log.Debugf("put url [%s](%s) done", dstName, url)
|
||||||
|
return errors.WithStack(err)
|
||||||
|
}
|
||||||
|
|
|
@ -145,7 +145,9 @@ func AddOfflineDownload(c *gin.Context) {
|
||||||
common.ErrorResp(c, err, 500)
|
common.ErrorResp(c, err, 500)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
tasks = append(tasks, t)
|
if t != nil {
|
||||||
|
tasks = append(tasks, t)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
common.SuccessResp(c, gin.H{
|
common.SuccessResp(c, gin.H{
|
||||||
"tasks": getTaskInfos(tasks),
|
"tasks": getTaskInfos(tasks),
|
||||||
|
|
Loading…
Reference in New Issue