| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396 | package webdavfsimport (	"errors"	"io"	"log"	"os"	"path/filepath"	"regexp"	"strings"	"time"	"github.com/studio-b12/gowebdav")/*	WebDAV Client	This script is design as a wrapper of the studio-b12/gowebdav module	that allow access to webdav network drive in ArozOS and allow arozos	cross-mounting each others*/type WebDAVFileSystem struct {	UUID      string	Hierarchy string	root      string	user      string	c         *gowebdav.Client}func NewWebDAVMount(UUID string, Hierarchy string, root string, user string, password string) (*WebDAVFileSystem, error) {	//Connect to webdav server	c := gowebdav.NewClient(root, user, password)	err := c.Connect()	if err != nil {		log.Println("[WebDAV FS] Unable to connect to remote: ", err.Error())		return nil, err	} else {		log.Println("[WebDAV FS] Connected to remote: " + root)	}	return &WebDAVFileSystem{		UUID:      UUID,		Hierarchy: Hierarchy,		c:         c,		root:      root,		user:      user,	}, nil}func (e WebDAVFileSystem) Chmod(filename string, mode os.FileMode) error {	return errors.New("filesystem type not supported")}func (e WebDAVFileSystem) Chown(filename string, uid int, gid int) error {	return errors.New("filesystem type not supported")}func (e WebDAVFileSystem) Chtimes(filename string, atime time.Time, mtime time.Time) error {	return errors.New("filesystem type not supported")}func (e WebDAVFileSystem) Create(filename string) (*os.File, error) {	return nil, errors.New("filesystem type not supported")}func (e WebDAVFileSystem) Mkdir(filename string, mode os.FileMode) error {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	return e.c.Mkdir(filename, mode)}func (e WebDAVFileSystem) MkdirAll(filename string, mode os.FileMode) error {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	return e.c.MkdirAll(filename, mode)}func (e WebDAVFileSystem) Name() string {	return ""}func (e WebDAVFileSystem) Open(filename string) (*os.File, error) {	return nil, errors.New("filesystem type not supported")}func (e WebDAVFileSystem) OpenFile(filename string, flag int, perm os.FileMode) (*os.File, error) {	//Buffer the target file to memory	//To be implement: Wait for Golang's fs.File.Write function to be released	//f := bufffs.New(filename)	//return f, nil	return nil, errors.New("filesystem type not supported")}func (e WebDAVFileSystem) Remove(filename string) error {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	return e.c.Remove(filename)}func (e WebDAVFileSystem) RemoveAll(filename string) error {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	return e.c.RemoveAll(filename)}func (e WebDAVFileSystem) Rename(oldname, newname string) error {	oldname = filterFilepath(filepath.ToSlash(filepath.Clean(oldname)))	newname = filterFilepath(filepath.ToSlash(filepath.Clean(newname)))	err := e.c.Rename(oldname, newname, true)	if err != nil {		//Unable to rename due to reverse proxy issue. Use Copy and Delete		f, err := e.c.ReadStream(oldname)		if err != nil {			return err		}		err = e.c.WriteStream(newname, f, 0775)		if err != nil {			return err		}		f.Close()		e.c.RemoveAll(oldname)	}	return nil}func (e WebDAVFileSystem) Stat(filename string) (os.FileInfo, error) {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	return e.c.Stat(filename)}func (e WebDAVFileSystem) VirtualPathToRealPath(subpath string, username string) (string, error) {	subpath = filterFilepath(filepath.ToSlash(filepath.Clean(subpath)))	if strings.HasPrefix(subpath, e.UUID+":") {		//This is full virtual path. Trim the uuid and correct the subpath		subpath = strings.TrimPrefix(subpath, e.UUID+":")	}	if e.Hierarchy == "user" {		return filepath.ToSlash(filepath.Clean(filepath.Join("users", username, subpath))), nil	} else if e.Hierarchy == "public" {		return filepath.ToSlash(filepath.Clean(subpath)), nil	}	return "", errors.New("unsupported filesystem hierarchy")}func (e WebDAVFileSystem) RealPathToVirtualPath(rpath string, username string) (string, error) {	rpath = filterFilepath(filepath.ToSlash(filepath.Clean(rpath)))	if e.Hierarchy == "user" && strings.HasPrefix(rpath, "/users/"+username) {		rpath = strings.TrimPrefix(rpath, "/users/"+username)	}	return e.UUID + ":" + filepath.ToSlash(rpath), nil}func (e WebDAVFileSystem) FileExists(filename string) bool {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	_, err := e.c.Stat(filename)	if os.IsNotExist(err) || err != nil {		return false	}	return true}func (e WebDAVFileSystem) IsDir(filename string) bool {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	s, err := e.c.Stat(filename)	if err != nil {		return false	}	return s.IsDir()}//Notes: This is not actual Glob function. This just emulate Glob using ReadDir with max depth 1 layerfunc (e WebDAVFileSystem) Glob(wildcard string) ([]string, error) {	wildcard = filepath.ToSlash(filepath.Clean(wildcard))	if !strings.HasPrefix(wildcard, "/") {		//Handle case for listing root, "*"		wildcard = "/" + wildcard	}	chunks := strings.Split(strings.TrimPrefix(wildcard, "/"), "/")	results, err := e.globpath("/", chunks, 0)	return results, err	/*		//Get the longest path without *		chunksWithoutStar := []string{}		chunks := strings.Split(wildcard, "/")		for _, chunk := range chunks {			if !strings.Contains(chunk, "*") {				chunksWithoutStar = append(chunksWithoutStar, chunk)			} else {				//Cut off				break			}		}		if strings.Count(wildcard, "*") <= 1 && strings.Contains(chunks[len(chunks)-1], "*") {			//Fast Glob			fileInfos, err := e.c.ReadDir(filterFilepath(filepath.ToSlash(filepath.Clean(filepath.Dir(wildcard)))))			if err != nil {				return []string{}, err			}			validFiles := []string{}			matchingRule := wildCardToRegexp(wildcard)			for _, thisFileInfo := range fileInfos {				thisFileFullpath := filepath.ToSlash(filepath.Join(filepath.Dir(wildcard), thisFileInfo.Name()))				match, _ := regexp.MatchString(matchingRule, thisFileFullpath)				if match {					validFiles = append(validFiles, thisFileFullpath)				}			}			return validFiles, nil		} else {			//Slow Glob			walkRoot := strings.Join(chunksWithoutStar, "/")			if !strings.HasPrefix(walkRoot, "/") {				walkRoot = "/" + walkRoot			}			allFiles := []string{}			e.Walk(walkRoot, func(path string, info fs.FileInfo, err error) error {				allFiles = append(allFiles, path)				return nil			})			validFiles := []string{}			matchingRule := wildCardToRegexp(wildcard) + "$"			for _, thisFilepath := range allFiles {				match, _ := regexp.MatchString(matchingRule, thisFilepath)				if match {					validFiles = append(validFiles, thisFilepath)				}			}			return validFiles, nil		}	*/}func (e WebDAVFileSystem) GetFileSize(filename string) int64 {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	s, err := e.Stat(filename)	if err != nil {		log.Println(err)		return 0	}	return s.Size()}func (e WebDAVFileSystem) GetModTime(filename string) (int64, error) {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	s, err := e.Stat(filename)	if err != nil {		return 0, err	}	return s.ModTime().Unix(), nil}func (e WebDAVFileSystem) WriteFile(filename string, content []byte, mode os.FileMode) error {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	return e.c.Write(filename, content, mode)}func (e WebDAVFileSystem) ReadFile(filename string) ([]byte, error) {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	bytes, err := e.c.Read(filename)	if err != nil {		return []byte(""), err	}	return bytes, nil}func (e WebDAVFileSystem) WriteStream(filename string, stream io.Reader, mode os.FileMode) error {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	return e.c.WriteStream(filename, stream, mode)}func (e WebDAVFileSystem) ReadStream(filename string) (io.ReadCloser, error) {	filename = filterFilepath(filepath.ToSlash(filepath.Clean(filename)))	return e.c.ReadStream(filename)}func (e WebDAVFileSystem) Walk(rootpath string, walkFn filepath.WalkFunc) error {	rootpath = filepath.ToSlash(filepath.Clean(rootpath))	rootStat, err := e.Stat(rootpath)	err = walkFn(rootpath, rootStat, err)	if err != nil {		return err	}	return e.walk(rootpath, walkFn)}func (e WebDAVFileSystem) Close() error {	return nil}/*	Helper Functions*/func (e WebDAVFileSystem) walk(thisPath string, walkFun filepath.WalkFunc) error {	files, err := e.c.ReadDir(thisPath)	if err != nil {		return err	}	for _, file := range files {		thisFileFullPath := filepath.ToSlash(filepath.Join(thisPath, file.Name()))		if file.IsDir() {			err = walkFun(thisFileFullPath, file, nil)			if err != nil {				return err			}			err = e.walk(thisFileFullPath, walkFun)			if err != nil {				return err			}		} else {			err = walkFun(thisFileFullPath, file, nil)			if err != nil {				return err			}		}	}	return nil}func (e WebDAVFileSystem) globpath(currentPath string, pathSegments []string, depth int) ([]string, error) {	const pathSeparatorsLimit = 1000	if depth == pathSeparatorsLimit {		return nil, errors.New("bad pattern")	}	// Check pattern is well-formed.	if _, err := regexp.MatchString(wildCardToRegexp(strings.Join(pathSegments, "/")), ""); err != nil {		return nil, err	}	if len(pathSegments) == 0 {		//Reaching the bottom		return []string{}, nil	}	thisSegment := pathSegments[0]	if strings.Contains(thisSegment, "*") {		//Search for matching		matchPattern := currentPath + thisSegment		files, err := e.c.ReadDir(currentPath)		if err != nil {			return []string{}, nil		}		//Check which file in the currentPath matches the wildcard		matchedSubpaths := []string{}		for _, file := range files {			thisPath := currentPath + file.Name()			match, _ := regexp.MatchString(wildCardToRegexp(matchPattern), thisPath)			if match {				if file.IsDir() {					matchedSubpaths = append(matchedSubpaths, thisPath+"/")				} else {					matchedSubpaths = append(matchedSubpaths, thisPath)				}			}		}		if len(pathSegments[1:]) == 0 {			return matchedSubpaths, nil		}		//For each of the subpaths, do a globpath		matchingFilenames := []string{}		for _, subpath := range matchedSubpaths {			thisMatchedNames, _ := e.globpath(subpath, pathSegments[1:], depth+1)			matchingFilenames = append(matchingFilenames, thisMatchedNames...)		}		return matchingFilenames, nil	} else {		//Check folder exists		if e.FileExists(currentPath+thisSegment) && e.IsDir(currentPath+thisSegment) {			return e.globpath(currentPath+thisSegment+"/", pathSegments[1:], depth+1)		} else {			//Not matching			return []string{}, nil		}	}}func filterFilepath(rawpath string) string {	rawpath = strings.TrimSpace(rawpath)	if strings.HasPrefix(rawpath, "./") {		return rawpath[1:]	} else if rawpath == "." || rawpath == "" {		return "/"	}	return rawpath}func wildCardToRegexp(pattern string) string {	var result strings.Builder	for i, literal := range strings.Split(pattern, "*") {		// Replace * with .*		if i > 0 {			result.WriteString(".*")		}		// Quote any regular expression meta characters in the		// literal text.		result.WriteString(regexp.QuoteMeta(literal))	}	return result.String()}
 |