From e2bf773f3b607f8a500a7aa034f7e12fb812eee3 Mon Sep 17 00:00:00 2001 From: Gokcehan Date: Fri, 3 Jul 2020 21:42:13 +0300 Subject: [PATCH] fallback to copy-del for cross-device linking Related #151 --- nav.go | 46 +++++++++++++++++++++++++++++++++++++++++++--- 1 file changed, 43 insertions(+), 3 deletions(-) diff --git a/nav.go b/nav.go index 28eeb68..121cd4b 100644 --- a/nav.go +++ b/nav.go @@ -12,6 +12,7 @@ import ( "sort" "strconv" "strings" + "syscall" "time" times "gopkg.in/djherbis/times.v1" @@ -743,9 +744,48 @@ func (nav *nav) moveAsync(ui *ui, srcs []string, dstDir string) { } if err := os.Rename(src, dst); err != nil { - errCount++ - echo.args[0] = fmt.Sprintf("[%d] %s", errCount, err) - ui.exprChan <- echo + if err.(*os.LinkError).Err.(syscall.Errno) == syscall.EXDEV { + total, err := copySize([]string{src}) + if err != nil { + echo.args[0] = err.Error() + ui.exprChan <- echo + continue + } + + nav.copyTotalChan <- total + + nums, errs := copyAll([]string{src}, dstDir) + + oldCount := errCount + loop: + for { + select { + case n := <-nums: + nav.copyBytesChan <- n + case err, ok := <-errs: + if !ok { + break loop + } + errCount++ + echo.args[0] = fmt.Sprintf("[%d] %s", errCount, err) + ui.exprChan <- echo + } + } + + nav.copyTotalChan <- -total + + if errCount == oldCount { + if err := os.RemoveAll(src); err != nil { + errCount++ + echo.args[0] = fmt.Sprintf("[%d] %s", errCount, err) + ui.exprChan <- echo + } + } + } else { + errCount++ + echo.args[0] = fmt.Sprintf("[%d] %s", errCount, err) + ui.exprChan <- echo + } } }