mirror of
https://github.com/rclone/rclone.git
synced 2025-01-23 10:23:17 +08:00
fd95511091
Before this change, bisync needed to build a full listing for Path1, then a full listing for Path2, then compare them -- and each of those tasks needed to finish before the next one could start. In addition to being slow and inefficient, it also caused real problems if a file changed between the time bisync checked it on Path1 and the time it checked the corresponding file on Path2. This change solves these problems by listing both paths concurrently, using the same March infrastructure that check and sync use to traverse two directories in lock-step, optimized by Go's robust concurrency support. Listings should now be much faster, and any given path is now checked nearly-instantaneously on both sides, minimizing room for error. Further discussion: https://forum.rclone.org/t/bisync-bugs-and-feature-requests/37636#:~:text=4.%20Listings%20should%20alternate%20between%20paths%20to%20minimize%20errors
137 lines
2.8 KiB
Go
137 lines
2.8 KiB
Go
// Package bilib provides common stuff for bisync and bisync_test
|
|
// Here it's got local file/directory helpers (nice to have in lib/file)
|
|
package bilib
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"regexp"
|
|
"runtime"
|
|
)
|
|
|
|
// PermSecure is a Unix permission for a file accessible only by its owner
|
|
const PermSecure = 0600
|
|
|
|
var (
|
|
regexLocalPath = regexp.MustCompile(`^[./\\]`)
|
|
regexWindowsPath = regexp.MustCompile(`^[a-zA-Z]:`)
|
|
regexRemotePath = regexp.MustCompile(`^[a-zA-Z_][a-zA-Z0-9_-]*:`)
|
|
)
|
|
|
|
// IsLocalPath returns true if its argument is a non-remote path.
|
|
// Empty string or a relative path will be considered local.
|
|
// Note: `c:dir` will be considered local on Windows but remote on Linux.
|
|
func IsLocalPath(path string) bool {
|
|
if path == "" || regexLocalPath.MatchString(path) {
|
|
return true
|
|
}
|
|
if runtime.GOOS == "windows" && regexWindowsPath.MatchString(path) {
|
|
return true
|
|
}
|
|
return !regexRemotePath.MatchString(path)
|
|
}
|
|
|
|
// FileExists returns true if the local file exists
|
|
func FileExists(file string) bool {
|
|
_, err := os.Stat(file)
|
|
return !os.IsNotExist(err)
|
|
}
|
|
|
|
// CopyFileIfExists is like CopyFile but does not fail if source does not exist
|
|
func CopyFileIfExists(srcFile, dstFile string) error {
|
|
if !FileExists(srcFile) {
|
|
return nil
|
|
}
|
|
return CopyFile(srcFile, dstFile)
|
|
}
|
|
|
|
// CopyFile copies a local file
|
|
func CopyFile(src, dst string) (err error) {
|
|
var (
|
|
rd io.ReadCloser
|
|
wr io.WriteCloser
|
|
info os.FileInfo
|
|
)
|
|
if info, err = os.Stat(src); err != nil {
|
|
return
|
|
}
|
|
if rd, err = os.Open(src); err != nil {
|
|
return
|
|
}
|
|
defer func() {
|
|
_ = rd.Close()
|
|
}()
|
|
if wr, err = os.Create(dst); err != nil {
|
|
return
|
|
}
|
|
_, err = io.Copy(wr, rd)
|
|
if e := wr.Close(); err == nil {
|
|
err = e
|
|
}
|
|
if e := os.Chmod(dst, info.Mode()); err == nil {
|
|
err = e
|
|
}
|
|
if e := os.Chtimes(dst, info.ModTime(), info.ModTime()); err == nil {
|
|
err = e
|
|
}
|
|
return
|
|
}
|
|
|
|
// CopyDir copies a local directory
|
|
func CopyDir(src string, dst string) (err error) {
|
|
src = filepath.Clean(src)
|
|
dst = filepath.Clean(dst)
|
|
|
|
si, err := os.Stat(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !si.IsDir() {
|
|
return fmt.Errorf("source is not a directory")
|
|
}
|
|
|
|
_, err = os.Stat(dst)
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return
|
|
}
|
|
if err == nil {
|
|
return fmt.Errorf("destination already exists")
|
|
}
|
|
|
|
err = os.MkdirAll(dst, si.Mode())
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
entries, err := os.ReadDir(src)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
for _, entry := range entries {
|
|
srcPath := filepath.Join(src, entry.Name())
|
|
dstPath := filepath.Join(dst, entry.Name())
|
|
|
|
if entry.IsDir() {
|
|
err = CopyDir(srcPath, dstPath)
|
|
if err != nil {
|
|
return
|
|
}
|
|
} else {
|
|
// Skip symlinks.
|
|
if entry.Type()&os.ModeSymlink != 0 {
|
|
continue
|
|
}
|
|
|
|
err = CopyFile(srcPath, dstPath)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|