feat: handle symlink copying (#39)

pull/3756/head
Laurynas Gadliauskas 2023-03-24 13:30:26 +02:00 committed by GitHub
parent af2c5048d2
commit 6122395019
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 52 additions and 21 deletions

View File

@ -3,12 +3,14 @@ package fileutils
import ( import (
"os" "os"
"path" "path"
"path/filepath"
"strings"
"github.com/spf13/afero" "github.com/spf13/afero"
) )
// Copy copies a file or folder from one place to another. // Copy copies a file or folder from one place to another.
func Copy(fs afero.Fs, src, dst string) error { func Copy(fs afero.Fs, src, dst, scope string) error {
if src = path.Clean("/" + src); src == "" { if src = path.Clean("/" + src); src == "" {
return os.ErrNotExist return os.ErrNotExist
} }
@ -31,9 +33,34 @@ func Copy(fs afero.Fs, src, dst string) error {
return err return err
} }
if info.IsDir() { switch info.Mode() & os.ModeType {
return CopyDir(fs, src, dst) case os.ModeDir:
return CopyDir(fs, src, dst, scope)
case os.ModeSymlink:
return CopySymLink(fs, src, dst, scope)
default:
return CopyFile(fs, src, dst)
} }
}
return CopyFile(fs, src, dst)
func CopySymLink(fs afero.Fs, source, dest, scope string) error {
if reader, ok := fs.(afero.LinkReader); ok {
link, err := reader.ReadlinkIfPossible(source)
if err != nil {
return err
}
if filepath.IsAbs(link) {
link = strings.TrimPrefix(link, scope)
link = filepath.Join(string(os.PathSeparator), link)
} else {
link = filepath.Clean(filepath.Join(filepath.Dir(source), link))
}
if linker, ok := fs.(afero.Linker); ok {
return linker.SymlinkIfPossible(link, dest)
}
return nil
}
return nil
} }

View File

@ -2,6 +2,7 @@ package fileutils
import ( import (
"errors" "errors"
"os"
"path/filepath" "path/filepath"
"github.com/spf13/afero" "github.com/spf13/afero"
@ -10,7 +11,7 @@ import (
// CopyDir copies a directory from source to dest and all // CopyDir copies a directory from source to dest and all
// of its sub-directories. It doesn't stop if it finds an error // of its sub-directories. It doesn't stop if it finds an error
// during the copy. Returns an error if any. // during the copy. Returns an error if any.
func CopyDir(fs afero.Fs, source, dest string) error { func CopyDir(fs afero.Fs, source, dest, scope string) error {
// Get properties of source. // Get properties of source.
srcinfo, err := fs.Stat(source) srcinfo, err := fs.Stat(source)
if err != nil { if err != nil {
@ -35,16 +36,19 @@ func CopyDir(fs afero.Fs, source, dest string) error {
fsource := source + "/" + obj.Name() fsource := source + "/" + obj.Name()
fdest := dest + "/" + obj.Name() fdest := dest + "/" + obj.Name()
if obj.IsDir() { switch obj.Mode() & os.ModeType {
case os.ModeDir:
// Create sub-directories, recursively. // Create sub-directories, recursively.
err = CopyDir(fs, fsource, fdest) if err := CopyDir(fs, fsource, fdest, scope); err != nil {
if err != nil {
errs = append(errs, err) errs = append(errs, err)
} }
} else { case os.ModeSymlink:
if err := CopySymLink(fs, fsource, fdest, scope); err != nil {
return err
}
default:
// Perform the file copy. // Perform the file copy.
err = CopyFile(fs, fsource, fdest) if err := CopyFile(fs, fsource, fdest); err != nil {
if err != nil {
errs = append(errs, err) errs = append(errs, err)
} }
} }

View File

@ -8,7 +8,7 @@
"name": "filebrowser-frontend", "name": "filebrowser-frontend",
"version": "2.0.0", "version": "2.0.0",
"dependencies": { "dependencies": {
"ace-builds": "^1.4.7", "ace-builds": "^1.5.1",
"clipboard": "^2.0.4", "clipboard": "^2.0.4",
"core-js": "^3.9.1", "core-js": "^3.9.1",
"css-vars-ponyfill": "^2.4.3", "css-vars-ponyfill": "^2.4.3",
@ -2397,9 +2397,9 @@
} }
}, },
"node_modules/ace-builds": { "node_modules/ace-builds": {
"version": "1.4.12", "version": "1.15.3",
"resolved": "https://registry.npmjs.org/ace-builds/-/ace-builds-1.4.12.tgz", "resolved": "https://registry.npmjs.org/ace-builds/-/ace-builds-1.15.3.tgz",
"integrity": "sha512-G+chJctFPiiLGvs3+/Mly3apXTcfgE45dT5yp12BcWZ1kUs+gm0qd3/fv4gsz6fVag4mM0moHVpjHDIgph6Psg==" "integrity": "sha512-hq8+4DfQcUYcUyAF3vF7UoGFXwNxXST5A2IdarUOp9/Xg1thWTfxusPI2HAlTvXRTVjLDQOj9O34uPoTehEs0A=="
}, },
"node_modules/acorn": { "node_modules/acorn": {
"version": "7.4.1", "version": "7.4.1",
@ -17408,9 +17408,9 @@
} }
}, },
"ace-builds": { "ace-builds": {
"version": "1.4.12", "version": "1.15.3",
"resolved": "https://registry.npmjs.org/ace-builds/-/ace-builds-1.4.12.tgz", "resolved": "https://registry.npmjs.org/ace-builds/-/ace-builds-1.15.3.tgz",
"integrity": "sha512-G+chJctFPiiLGvs3+/Mly3apXTcfgE45dT5yp12BcWZ1kUs+gm0qd3/fv4gsz6fVag4mM0moHVpjHDIgph6Psg==" "integrity": "sha512-hq8+4DfQcUYcUyAF3vF7UoGFXwNxXST5A2IdarUOp9/Xg1thWTfxusPI2HAlTvXRTVjLDQOj9O34uPoTehEs0A=="
}, },
"acorn": { "acorn": {
"version": "7.4.1", "version": "7.4.1",

View File

@ -10,7 +10,7 @@
"watch": "find ./dist -maxdepth 1 -mindepth 1 ! -name '.gitignore' -exec rm -r {} + && vue-cli-service build --watch --no-clean" "watch": "find ./dist -maxdepth 1 -mindepth 1 ! -name '.gitignore' -exec rm -r {} + && vue-cli-service build --watch --no-clean"
}, },
"dependencies": { "dependencies": {
"ace-builds": "^1.4.7", "ace-builds": "^1.5.1",
"clipboard": "^2.0.4", "clipboard": "^2.0.4",
"core-js": "^3.9.1", "core-js": "^3.9.1",
"css-vars-ponyfill": "^2.4.3", "css-vars-ponyfill": "^2.4.3",

View File

@ -438,7 +438,7 @@ func patchAction(ctx context.Context, action, src, dst string, d *data, fileCach
return errors.ErrPermissionDenied return errors.ErrPermissionDenied
} }
return fileutils.Copy(d.user.Fs, src, dst) return fileutils.Copy(d.user.Fs, src, dst, d.server.Root)
case "rename": case "rename":
if !d.user.Perm.Rename { if !d.user.Perm.Rename {
return errors.ErrPermissionDenied return errors.ErrPermissionDenied