Cloudreve/pkg/filemanager/fs/dbfs/sharewithme_navigator.go

152 lines
4.2 KiB
Go

package dbfs
import (
"context"
"errors"
"fmt"
"github.com/cloudreve/Cloudreve/v4/application/constants"
"github.com/cloudreve/Cloudreve/v4/ent"
"github.com/cloudreve/Cloudreve/v4/inventory"
"github.com/cloudreve/Cloudreve/v4/inventory/types"
"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 sharedWithMeNavigatorCapability = &boolset.BooleanSet{}
// NewSharedWithMeNavigator creates a navigator for user's "shared with me" file system.
func NewSharedWithMeNavigator(u *ent.User, fileClient inventory.FileClient, l logging.Logger,
config *setting.DBFS, hasher hashid.Encoder) Navigator {
n := &sharedWithMeNavigator{
user: u,
l: l,
fileClient: fileClient,
config: config,
hasher: hasher,
}
n.baseNavigator = newBaseNavigator(fileClient, defaultFilter, u, hasher, config)
return n
}
type sharedWithMeNavigator struct {
l logging.Logger
user *ent.User
fileClient inventory.FileClient
config *setting.DBFS
hasher hashid.Encoder
root *File
*baseNavigator
}
func (t *sharedWithMeNavigator) Recycle() {
}
func (n *sharedWithMeNavigator) PersistState(kv cache.Driver, key string) {
}
func (n *sharedWithMeNavigator) RestoreState(s State) error {
return nil
}
func (t *sharedWithMeNavigator) To(ctx context.Context, path *fs.URI) (*File, error) {
// Anonymous user does not have a trash folder.
if inventory.IsAnonymousUser(t.user) {
return nil, ErrLoginRequired
}
elements := path.Elements()
if len(elements) > 0 {
// Shared with me folder is a flatten tree, only root can be accessed.
return nil, fs.ErrPathNotExist.WithError(fmt.Errorf("invalid Path %q", path))
}
if t.root == nil {
rootFile, err := t.fileClient.Root(ctx, t.user)
if err != nil {
t.l.Info("User's root folder not found: %s, will initialize it.", err)
return nil, ErrFsNotInitialized
}
t.root = newFile(nil, rootFile)
rootPath := newSharedWithMeUri("")
t.root.Path[pathIndexRoot], t.root.Path[pathIndexUser] = rootPath, rootPath
t.root.OwnerModel = t.user
t.root.IsUserRoot = true
t.root.CapabilitiesBs = t.Capabilities(false).Capability
}
return t.root, nil
}
func (t *sharedWithMeNavigator) Children(ctx context.Context, parent *File, args *ListArgs) (*ListResult, error) {
args.SharedWithMe = true
res, err := t.baseNavigator.children(ctx, nil, args)
if err != nil {
return nil, err
}
// Adding user uri for each file.
for i := 0; i < len(res.Files); i++ {
res.Files[i].Path[pathIndexUser] = newSharedWithMeUri(hashid.EncodeFileID(t.hasher, res.Files[i].Model.ID))
}
return res, nil
}
func (t *sharedWithMeNavigator) Capabilities(isSearching bool) *fs.NavigatorProps {
res := &fs.NavigatorProps{
Capability: sharedWithMeNavigatorCapability,
OrderDirectionOptions: fullOrderDirectionOption,
OrderByOptions: fullOrderByOption,
MaxPageSize: t.config.MaxPageSize,
}
if isSearching {
res.OrderByOptions = searchLimitedOrderByOption
}
return res
}
func (t *sharedWithMeNavigator) Walk(ctx context.Context, levelFiles []*File, limit, depth int, f WalkFunc) error {
return errors.New("not implemented")
}
func (n *sharedWithMeNavigator) 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
}
oldFileClient := n.fileClient
revert := func() {
n.fileClient = oldFileClient
n.baseNavigator.fileClient = oldFileClient
}
n.fileClient = newFileClient
n.baseNavigator.fileClient = newFileClient
return revert, nil
}
func (n *sharedWithMeNavigator) ExecuteHook(ctx context.Context, hookType fs.HookType, file *File) error {
return nil
}
func (n *sharedWithMeNavigator) GetView(ctx context.Context, file *File) *types.ExplorerView {
if view, ok := n.user.Settings.FsViewMap[string(constants.FileSystemSharedWithMe)]; ok {
return &view
}
return defaultView
}