123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 |
- package bokofs
- import (
- "context"
- "fmt"
- "os"
- "path/filepath"
- "strings"
- "time"
- "golang.org/x/net/webdav"
- "imuslab.com/bokofs/bokofsd/mod/bokofs/bokoworker"
- )
- type RootRouter struct {
- pathPrefix string
- parent *Server
- }
- // InitiateRootRouter create and prepare a virtual root file system for
- // this bokoFS instance
- func (s *Server) InitiateRootRouter() error {
- s.RootRouter = &RootRouter{
- pathPrefix: s.prefix,
- parent: s,
- }
- return nil
- }
- // fixpath fix the path to be relative to the root path of this router
- func (r *RootRouter) fixpath(name string) string {
- if name == r.pathPrefix || name == "" {
- return "/"
- }
- //Trim off the prefix path
- name = strings.TrimPrefix(name, r.pathPrefix)
- if !strings.HasPrefix(name, "/") {
- name = "/" + name
- }
- return name
- }
- func (r *RootRouter) GetRootDir(name string) string {
- if name == "" || name == "/" {
- return "/"
- }
- name = filepath.ToSlash(filepath.Clean(name))
- pathChunks := strings.Split(name, "/")
- reqRootPath := "/" + pathChunks[1]
- fmt.Println("Requesting Root Path: ", reqRootPath)
- name = strings.TrimSuffix(reqRootPath, "/")
- return name
- }
- func (r *RootRouter) GetWorkerByPath(name string) (*bokoworker.Worker, error) {
- reqRootPath := r.GetRootDir(name)
- targetWorker, ok := r.parent.LoadedWorkers.Load(reqRootPath)
- if !ok {
- return nil, os.ErrNotExist
- }
- return targetWorker.(*bokoworker.Worker), nil
- }
- func (r *RootRouter) Mkdir(ctx context.Context, name string, perm os.FileMode) error {
- // Implement the Mkdir method
- name = r.fixpath(name)
- fmt.Println("Mkdir called to " + name)
- return nil
- }
- func (r *RootRouter) OpenFile(ctx context.Context, name string, flag int, perm os.FileMode) (webdav.File, error) {
- // Implement the OpenFile method
- name = r.fixpath(name)
- fmt.Println("OpenFile called to " + name)
- if filepath.ToSlash(filepath.Base(name)) == "/" {
- //Request to the vObject base path
- thisVirtualObject := r.newVirtualObject(&vObjectProperties{
- name: name,
- size: 0,
- mode: os.ModeDir,
- modTime: time.Now(),
- isDir: true,
- })
- return thisVirtualObject, nil
- }
- targetWorker, err := r.GetWorkerByPath(name)
- if err != nil {
- return nil, err
- }
- return targetWorker.Filesystem.OpenFile(ctx, name, flag, perm)
- }
- func (r *RootRouter) RemoveAll(ctx context.Context, name string) error {
- // Implement the RemoveAll method
- name = r.fixpath(name)
- fmt.Println("RemoveAll called to " + name)
- return nil
- }
- func (r *RootRouter) Rename(ctx context.Context, oldName, newName string) error {
- // Implement the Rename method
- oldName = r.fixpath(oldName)
- newName = r.fixpath(newName)
- fmt.Println("Rename called from " + oldName + " to " + newName)
- return nil
- }
- func (r *RootRouter) Stat(ctx context.Context, name string) (os.FileInfo, error) {
- // Implement the Stat method
- name = r.fixpath(name)
- fmt.Println("Stat called to " + name)
- if filepath.ToSlash(filepath.Base(name)) == "/" {
- //Create an emulated file system to serve the mounted workers
- thisVirtualObject := r.newVirtualObject(&vObjectProperties{
- name: name,
- size: 0,
- mode: os.ModeDir,
- modTime: time.Now(),
- isDir: true,
- })
- thisVirtualObjectFileInfo := thisVirtualObject.GetFileInfo()
- return thisVirtualObjectFileInfo, nil
- }
- //Load the target worker from the path
- targetWorker, err := r.GetWorkerByPath(name)
- fmt.Println("Target Worker: ", targetWorker, name)
- if err != nil {
- return nil, err
- }
- return targetWorker.Filesystem.Stat(ctx, name)
- }
- // Ensure RootRouter implements the FileSystem interface
- var _ webdav.FileSystem = (*RootRouter)(nil)
|