feat: handle symlink copying (#39)
							parent
							
								
									af2c5048d2
								
							
						
					
					
						commit
						6122395019
					
				|  | @ -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 | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -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) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | @ -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", | ||||||
|  |  | ||||||
|  | @ -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", | ||||||
|  |  | ||||||
|  | @ -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 | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 Laurynas Gadliauskas
						Laurynas Gadliauskas