mirror of https://github.com/Xhofe/alist
259 lines
6.6 KiB
Go
259 lines
6.6 KiB
Go
package url_tree
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"github.com/alist-org/alist/v3/internal/op"
|
|
stdpath "path"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/alist-org/alist/v3/internal/driver"
|
|
"github.com/alist-org/alist/v3/internal/errs"
|
|
"github.com/alist-org/alist/v3/internal/model"
|
|
"github.com/alist-org/alist/v3/pkg/utils"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
type Urls struct {
|
|
model.Storage
|
|
Addition
|
|
root *Node
|
|
mutex sync.RWMutex
|
|
}
|
|
|
|
func (d *Urls) Config() driver.Config {
|
|
return config
|
|
}
|
|
|
|
func (d *Urls) GetAddition() driver.Additional {
|
|
return &d.Addition
|
|
}
|
|
|
|
func (d *Urls) Init(ctx context.Context) error {
|
|
node, err := BuildTree(d.UrlStructure, d.HeadSize)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
node.calSize()
|
|
d.root = node
|
|
return nil
|
|
}
|
|
|
|
func (d *Urls) Drop(ctx context.Context) error {
|
|
return nil
|
|
}
|
|
|
|
func (d *Urls) Get(ctx context.Context, path string) (model.Obj, error) {
|
|
d.mutex.RLock()
|
|
defer d.mutex.RUnlock()
|
|
node := GetNodeFromRootByPath(d.root, path)
|
|
return nodeToObj(node, path)
|
|
}
|
|
|
|
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())
|
|
log.Debugf("path: %s, node: %+v", dir.GetPath(), node)
|
|
if node == nil {
|
|
return nil, errs.ObjectNotFound
|
|
}
|
|
if node.isFile() {
|
|
return nil, errs.NotFolder
|
|
}
|
|
return utils.SliceConvert(node.Children, func(node *Node) (model.Obj, error) {
|
|
return nodeToObj(node, stdpath.Join(dir.GetPath(), node.Name))
|
|
})
|
|
}
|
|
|
|
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())
|
|
log.Debugf("path: %s, node: %+v", file.GetPath(), node)
|
|
if node == nil {
|
|
return nil, errs.ObjectNotFound
|
|
}
|
|
if node.isFile() {
|
|
return &model.Link{
|
|
URL: node.Url,
|
|
}, nil
|
|
}
|
|
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) {
|
|
// return nil, errs.NotSupport
|
|
//}
|
|
|
|
var _ driver.Driver = (*Urls)(nil)
|