Init V4 community edition (#2265)
* Init V4 community edition * Init V4 community edition
This commit is contained in:
172
pkg/filemanager/fs/dbfs/my_navigator.go
Normal file
172
pkg/filemanager/fs/dbfs/my_navigator.go
Normal file
@@ -0,0 +1,172 @@
|
||||
package dbfs
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"github.com/cloudreve/Cloudreve/v4/ent"
|
||||
"github.com/cloudreve/Cloudreve/v4/inventory"
|
||||
"github.com/cloudreve/Cloudreve/v4/pkg/boolset"
|
||||
"github.com/cloudreve/Cloudreve/v4/pkg/cache"
|
||||
"github.com/cloudreve/Cloudreve/v4/pkg/filemanager/fs"
|
||||
"github.com/cloudreve/Cloudreve/v4/pkg/hashid"
|
||||
"github.com/cloudreve/Cloudreve/v4/pkg/logging"
|
||||
"github.com/cloudreve/Cloudreve/v4/pkg/setting"
|
||||
)
|
||||
|
||||
var myNavigatorCapability = &boolset.BooleanSet{}
|
||||
|
||||
// NewMyNavigator creates a navigator for user's "my" file system.
|
||||
func NewMyNavigator(u *ent.User, fileClient inventory.FileClient, userClient inventory.UserClient, l logging.Logger,
|
||||
config *setting.DBFS, hasher hashid.Encoder) Navigator {
|
||||
return &myNavigator{
|
||||
user: u,
|
||||
l: l,
|
||||
fileClient: fileClient,
|
||||
userClient: userClient,
|
||||
config: config,
|
||||
baseNavigator: newBaseNavigator(fileClient, defaultFilter, u, hasher, config),
|
||||
}
|
||||
}
|
||||
|
||||
type myNavigator struct {
|
||||
l logging.Logger
|
||||
user *ent.User
|
||||
fileClient inventory.FileClient
|
||||
userClient inventory.UserClient
|
||||
|
||||
config *setting.DBFS
|
||||
*baseNavigator
|
||||
root *File
|
||||
disableRecycle bool
|
||||
persist func()
|
||||
}
|
||||
|
||||
func (n *myNavigator) Recycle() {
|
||||
if n.persist != nil {
|
||||
n.persist()
|
||||
n.persist = nil
|
||||
}
|
||||
if n.root != nil && !n.disableRecycle {
|
||||
n.root.Recycle()
|
||||
}
|
||||
}
|
||||
|
||||
func (n *myNavigator) PersistState(kv cache.Driver, key string) {
|
||||
n.disableRecycle = true
|
||||
n.persist = func() {
|
||||
kv.Set(key, n.root, ContextHintTTL)
|
||||
}
|
||||
}
|
||||
|
||||
func (n *myNavigator) RestoreState(s State) error {
|
||||
n.disableRecycle = true
|
||||
if state, ok := s.(*File); ok {
|
||||
n.root = state
|
||||
return nil
|
||||
}
|
||||
|
||||
return fmt.Errorf("invalid state type: %T", s)
|
||||
}
|
||||
|
||||
func (n *myNavigator) To(ctx context.Context, path *fs.URI) (*File, error) {
|
||||
if n.root == nil {
|
||||
// Anonymous user does not have a root folder.
|
||||
if inventory.IsAnonymousUser(n.user) {
|
||||
return nil, ErrLoginRequired
|
||||
}
|
||||
|
||||
fsUid, err := n.hasher.Decode(path.ID(hashid.EncodeUserID(n.hasher, n.user.ID)), hashid.UserID)
|
||||
if err != nil {
|
||||
return nil, fs.ErrPathNotExist.WithError(fmt.Errorf("invalid user id"))
|
||||
}
|
||||
if fsUid != n.user.ID {
|
||||
return nil, ErrPermissionDenied
|
||||
}
|
||||
|
||||
targetUser, err := n.userClient.GetLoginUserByID(ctx, fsUid)
|
||||
if err != nil {
|
||||
return nil, fs.ErrPathNotExist.WithError(fmt.Errorf("user not found: %w", err))
|
||||
}
|
||||
|
||||
rootFile, err := n.fileClient.Root(ctx, targetUser)
|
||||
if err != nil {
|
||||
n.l.Info("User's root folder not found: %s, will initialize it.", err)
|
||||
return nil, ErrFsNotInitialized
|
||||
}
|
||||
|
||||
n.root = newFile(nil, rootFile)
|
||||
rootPath := path.Root()
|
||||
n.root.Path[pathIndexRoot], n.root.Path[pathIndexUser] = rootPath, rootPath
|
||||
n.root.OwnerModel = targetUser
|
||||
n.root.IsUserRoot = true
|
||||
n.root.CapabilitiesBs = n.Capabilities(false).Capability
|
||||
}
|
||||
|
||||
current, lastAncestor := n.root, n.root
|
||||
elements := path.Elements()
|
||||
var err error
|
||||
for index, element := range elements {
|
||||
lastAncestor = current
|
||||
current, err = n.walkNext(ctx, current, element, index == len(elements)-1)
|
||||
if err != nil {
|
||||
return lastAncestor, fmt.Errorf("failed to walk into %q: %w", element, err)
|
||||
}
|
||||
}
|
||||
|
||||
return current, nil
|
||||
}
|
||||
|
||||
func (n *myNavigator) Children(ctx context.Context, parent *File, args *ListArgs) (*ListResult, error) {
|
||||
return n.baseNavigator.children(ctx, parent, args)
|
||||
}
|
||||
|
||||
func (n *myNavigator) walkNext(ctx context.Context, root *File, next string, isLeaf bool) (*File, error) {
|
||||
return n.baseNavigator.walkNext(ctx, root, next, isLeaf)
|
||||
}
|
||||
|
||||
func (n *myNavigator) Capabilities(isSearching bool) *fs.NavigatorProps {
|
||||
res := &fs.NavigatorProps{
|
||||
Capability: myNavigatorCapability,
|
||||
OrderDirectionOptions: fullOrderDirectionOption,
|
||||
OrderByOptions: fullOrderByOption,
|
||||
MaxPageSize: n.config.MaxPageSize,
|
||||
}
|
||||
if isSearching {
|
||||
res.OrderByOptions = nil
|
||||
res.OrderDirectionOptions = nil
|
||||
}
|
||||
|
||||
return res
|
||||
}
|
||||
|
||||
func (n *myNavigator) Walk(ctx context.Context, levelFiles []*File, limit, depth int, f WalkFunc) error {
|
||||
return n.baseNavigator.walk(ctx, levelFiles, limit, depth, f)
|
||||
}
|
||||
|
||||
func (n *myNavigator) FollowTx(ctx context.Context) (func(), error) {
|
||||
if _, ok := ctx.Value(inventory.TxCtx{}).(*inventory.Tx); !ok {
|
||||
return nil, fmt.Errorf("navigator: no inherited transaction found in context")
|
||||
}
|
||||
newFileClient, _, _, err := inventory.WithTx(ctx, n.fileClient)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
newUserClient, _, _, err := inventory.WithTx(ctx, n.userClient)
|
||||
|
||||
oldFileClient, oldUserClient := n.fileClient, n.userClient
|
||||
revert := func() {
|
||||
n.fileClient = oldFileClient
|
||||
n.userClient = oldUserClient
|
||||
n.baseNavigator.fileClient = oldFileClient
|
||||
}
|
||||
|
||||
n.fileClient = newFileClient
|
||||
n.userClient = newUserClient
|
||||
n.baseNavigator.fileClient = newFileClient
|
||||
return revert, nil
|
||||
}
|
||||
|
||||
func (n *myNavigator) ExecuteHook(ctx context.Context, hookType fs.HookType, file *File) error {
|
||||
return nil
|
||||
}
|
||||
Reference in New Issue
Block a user