accounting: isolate stats to groups

Introduce stats groups that will isolate accounting for logically
different transferring operations. That way multiple accounting
operations can be done in parallel without interfering with each other
stats.

Using groups is optional. There is dedicated global stats that will be
used by default if no group is specified. This is operating mode for CLI
usage which is just fire and forget operation.

For running rclone as rc http server each request will create it's own
group. Also there is an option to specify your own group.
This commit is contained in:
Aleksandar Jankovic 2019-07-18 12:13:54 +02:00 committed by Nick Craig-Wood
parent be0464f5f1
commit 8243ff8bc8
25 changed files with 551 additions and 281 deletions

View file

@ -1084,16 +1084,16 @@ func (f *Fs) purge(ctx context.Context, oldOnly bool) error {
go func() { go func() {
defer wg.Done() defer wg.Done()
for object := range toBeDeleted { for object := range toBeDeleted {
accounting.Stats.Checking(object.Name) accounting.Stats(ctx).Checking(object.Name)
checkErr(f.deleteByID(object.ID, object.Name)) checkErr(f.deleteByID(object.ID, object.Name))
accounting.Stats.DoneChecking(object.Name) accounting.Stats(ctx).DoneChecking(object.Name)
} }
}() }()
} }
last := "" last := ""
checkErr(f.list(ctx, "", true, "", 0, true, func(remote string, object *api.File, isDirectory bool) error { checkErr(f.list(ctx, "", true, "", 0, true, func(remote string, object *api.File, isDirectory bool) error {
if !isDirectory { if !isDirectory {
accounting.Stats.Checking(remote) accounting.Stats(ctx).Checking(remote)
if oldOnly && last != remote { if oldOnly && last != remote {
if object.Action == "hide" { if object.Action == "hide" {
fs.Debugf(remote, "Deleting current version (id %q) as it is a hide marker", object.ID) fs.Debugf(remote, "Deleting current version (id %q) as it is a hide marker", object.ID)
@ -1109,7 +1109,7 @@ func (f *Fs) purge(ctx context.Context, oldOnly bool) error {
toBeDeleted <- object toBeDeleted <- object
} }
last = remote last = remote
accounting.Stats.DoneChecking(remote) accounting.Stats(ctx).DoneChecking(remote)
} }
return nil return nil
})) }))

View file

@ -359,7 +359,7 @@ func (f *Fs) List(ctx context.Context, dir string) (entries fs.DirEntries, err e
err = errors.Wrapf(err, "failed to open directory %q", dir) err = errors.Wrapf(err, "failed to open directory %q", dir)
fs.Errorf(dir, "%v", err) fs.Errorf(dir, "%v", err)
if isPerm { if isPerm {
accounting.Stats.Error(fserrors.NoRetryError(err)) accounting.Stats(ctx).Error(fserrors.NoRetryError(err))
err = nil // ignore error but fail sync err = nil // ignore error but fail sync
} }
return nil, err return nil, err
@ -395,7 +395,7 @@ func (f *Fs) List(ctx context.Context, dir string) (entries fs.DirEntries, err e
if fierr != nil { if fierr != nil {
err = errors.Wrapf(err, "failed to read directory %q", namepath) err = errors.Wrapf(err, "failed to read directory %q", namepath)
fs.Errorf(dir, "%v", fierr) fs.Errorf(dir, "%v", fierr)
accounting.Stats.Error(fserrors.NoRetryError(fierr)) // fail the sync accounting.Stats(ctx).Error(fserrors.NoRetryError(fierr)) // fail the sync
continue continue
} }
fis = append(fis, fi) fis = append(fis, fi)
@ -418,7 +418,7 @@ func (f *Fs) List(ctx context.Context, dir string) (entries fs.DirEntries, err e
// Skip bad symlinks // Skip bad symlinks
err = fserrors.NoRetryError(errors.Wrap(err, "symlink")) err = fserrors.NoRetryError(errors.Wrap(err, "symlink"))
fs.Errorf(newRemote, "Listing error: %v", err) fs.Errorf(newRemote, "Listing error: %v", err)
accounting.Stats.Error(err) accounting.Stats(ctx).Error(err)
continue continue
} }
if err != nil { if err != nil {

View file

@ -232,25 +232,25 @@ func Run(Retry bool, showStats bool, cmd *cobra.Command, f func() error) {
for try := 1; try <= *retries; try++ { for try := 1; try <= *retries; try++ {
err = f() err = f()
fs.CountError(err) fs.CountError(err)
lastErr := accounting.Stats.GetLastError() lastErr := accounting.GlobalStats().GetLastError()
if err == nil { if err == nil {
err = lastErr err = lastErr
} }
if !Retry || !accounting.Stats.Errored() { if !Retry || !accounting.GlobalStats().Errored() {
if try > 1 { if try > 1 {
fs.Errorf(nil, "Attempt %d/%d succeeded", try, *retries) fs.Errorf(nil, "Attempt %d/%d succeeded", try, *retries)
} }
break break
} }
if accounting.Stats.HadFatalError() { if accounting.GlobalStats().HadFatalError() {
fs.Errorf(nil, "Fatal error received - not attempting retries") fs.Errorf(nil, "Fatal error received - not attempting retries")
break break
} }
if accounting.Stats.Errored() && !accounting.Stats.HadRetryError() { if accounting.GlobalStats().Errored() && !accounting.GlobalStats().HadRetryError() {
fs.Errorf(nil, "Can't retry this error - not attempting retries") fs.Errorf(nil, "Can't retry this error - not attempting retries")
break break
} }
if retryAfter := accounting.Stats.RetryAfter(); !retryAfter.IsZero() { if retryAfter := accounting.GlobalStats().RetryAfter(); !retryAfter.IsZero() {
d := retryAfter.Sub(time.Now()) d := retryAfter.Sub(time.Now())
if d > 0 { if d > 0 {
fs.Logf(nil, "Received retry after error - sleeping until %s (%v)", retryAfter.Format(time.RFC3339Nano), d) fs.Logf(nil, "Received retry after error - sleeping until %s (%v)", retryAfter.Format(time.RFC3339Nano), d)
@ -258,12 +258,12 @@ func Run(Retry bool, showStats bool, cmd *cobra.Command, f func() error) {
} }
} }
if lastErr != nil { if lastErr != nil {
fs.Errorf(nil, "Attempt %d/%d failed with %d errors and: %v", try, *retries, accounting.Stats.GetErrors(), lastErr) fs.Errorf(nil, "Attempt %d/%d failed with %d errors and: %v", try, *retries, accounting.GlobalStats().GetErrors(), lastErr)
} else { } else {
fs.Errorf(nil, "Attempt %d/%d failed with %d errors", try, *retries, accounting.Stats.GetErrors()) fs.Errorf(nil, "Attempt %d/%d failed with %d errors", try, *retries, accounting.GlobalStats().GetErrors())
} }
if try < *retries { if try < *retries {
accounting.Stats.ResetErrors() accounting.GlobalStats().ResetErrors()
} }
if *retriesInterval > 0 { if *retriesInterval > 0 {
time.Sleep(*retriesInterval) time.Sleep(*retriesInterval)
@ -271,7 +271,7 @@ func Run(Retry bool, showStats bool, cmd *cobra.Command, f func() error) {
} }
stopStats() stopStats()
if err != nil { if err != nil {
nerrs := accounting.Stats.GetErrors() nerrs := accounting.GlobalStats().GetErrors()
if nerrs <= 1 { if nerrs <= 1 {
log.Printf("Failed to %s: %v", cmd.Name(), err) log.Printf("Failed to %s: %v", cmd.Name(), err)
} else { } else {
@ -279,8 +279,8 @@ func Run(Retry bool, showStats bool, cmd *cobra.Command, f func() error) {
} }
resolveExitCode(err) resolveExitCode(err)
} }
if showStats && (accounting.Stats.Errored() || *statsInterval > 0) { if showStats && (accounting.GlobalStats().Errored() || *statsInterval > 0) {
accounting.Stats.Log() accounting.GlobalStats().Log()
} }
fs.Debugf(nil, "%d go routines active\n", runtime.NumGoroutine()) fs.Debugf(nil, "%d go routines active\n", runtime.NumGoroutine())
@ -303,8 +303,8 @@ func Run(Retry bool, showStats bool, cmd *cobra.Command, f func() error) {
} }
} }
if accounting.Stats.Errored() { if accounting.GlobalStats().Errored() {
resolveExitCode(accounting.Stats.GetLastError()) resolveExitCode(accounting.GlobalStats().GetLastError())
} }
} }
@ -337,7 +337,7 @@ func StartStats() func() {
for { for {
select { select {
case <-ticker.C: case <-ticker.C:
accounting.Stats.Log() accounting.GlobalStats().Log()
case <-stopStats: case <-stopStats:
ticker.Stop() ticker.Stop()
return return

View file

@ -93,7 +93,7 @@ func printProgress(logMessage string) {
w, h = 80, 25 w, h = 80, 25
} }
_ = h _ = h
stats := strings.TrimSpace(accounting.Stats.String()) stats := strings.TrimSpace(accounting.GlobalStats().String())
logMessage = strings.TrimSpace(logMessage) logMessage = strings.TrimSpace(logMessage)
out := func(s string) { out := func(s string) {

View file

@ -214,7 +214,7 @@ func (d *Driver) ListDir(path string, callback func(ftp.FileInfo) error) (err er
} }
// Account the transfer // Account the transfer
tr := accounting.Stats.NewTransferRemoteSize(path, node.Size()) tr := accounting.GlobalStats().NewTransferRemoteSize(path, node.Size())
defer func() { defer func() {
tr.Done(err) tr.Done(err)
}() }()
@ -313,7 +313,7 @@ func (d *Driver) GetFile(path string, offset int64) (size int64, fr io.ReadClose
} }
// Account the transfer // Account the transfer
tr := accounting.Stats.NewTransferRemoteSize(path, node.Size()) tr := accounting.GlobalStats().NewTransferRemoteSize(path, node.Size())
defer tr.Done(nil) defer tr.Done(nil)
return node.Size(), handle, nil return node.Size(), handle, nil

View file

@ -187,7 +187,7 @@ func (s *server) serveFile(w http.ResponseWriter, r *http.Request, remote string
}() }()
// Account the transfer // Account the transfer
tr := accounting.Stats.NewTransfer(obj) tr := accounting.Stats(r.Context()).NewTransfer(obj)
defer tr.Done(nil) defer tr.Done(nil)
// FIXME in = fs.NewAccount(in, obj).WithBuffer() // account the transfer // FIXME in = fs.NewAccount(in, obj).WithBuffer() // account the transfer

View file

@ -75,7 +75,7 @@ func Error(what interface{}, w http.ResponseWriter, text string, err error) {
// Serve serves a directory // Serve serves a directory
func (d *Directory) Serve(w http.ResponseWriter, r *http.Request) { func (d *Directory) Serve(w http.ResponseWriter, r *http.Request) {
// Account the transfer // Account the transfer
tr := accounting.Stats.NewTransferRemoteSize(d.DirRemote, -1) tr := accounting.Stats(r.Context()).NewTransferRemoteSize(d.DirRemote, -1)
defer tr.Done(nil) defer tr.Done(nil)
fs.Infof(d.DirRemote, "%s: Serving directory", r.RemoteAddr) fs.Infof(d.DirRemote, "%s: Serving directory", r.RemoteAddr)

View file

@ -75,7 +75,7 @@ func Object(w http.ResponseWriter, r *http.Request, o fs.Object) {
http.Error(w, http.StatusText(http.StatusNotFound), http.StatusNotFound) http.Error(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
return return
} }
tr := accounting.Stats.NewTransfer(o) tr := accounting.Stats(r.Context()).NewTransfer(o)
defer func() { defer func() {
tr.Done(err) tr.Done(err)
}() }()

View file

@ -271,7 +271,7 @@ func (s *server) postObject(w http.ResponseWriter, r *http.Request, remote strin
_, err := operations.RcatSize(r.Context(), s.f, remote, r.Body, r.ContentLength, time.Now()) _, err := operations.RcatSize(r.Context(), s.f, remote, r.Body, r.ContentLength, time.Now())
if err != nil { if err != nil {
accounting.Stats.Error(err) accounting.Stats(r.Context()).Error(err)
fs.Errorf(remote, "Post request rcat error: %v", err) fs.Errorf(remote, "Post request rcat error: %v", err)
http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError) http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)

View file

@ -39,3 +39,14 @@ func (ip *inProgress) get(name string) *Account {
defer ip.mu.Unlock() defer ip.mu.Unlock()
return ip.m[name] return ip.m[name]
} }
// merge adds items from another inProgress
func (ip *inProgress) merge(m *inProgress) {
ip.mu.Lock()
defer ip.mu.Unlock()
m.mu.Lock()
defer m.mu.Unlock()
for key, val := range m.m {
ip.m[key] = val
}
}

View file

@ -2,7 +2,6 @@ package accounting
import ( import (
"bytes" "bytes"
"context"
"fmt" "fmt"
"sort" "sort"
"strings" "strings"
@ -14,60 +13,6 @@ import (
"github.com/ncw/rclone/fs/rc" "github.com/ncw/rclone/fs/rc"
) )
var (
// Stats is global statistics counter
Stats = NewStats()
)
func init() {
// Set the function pointer up in fs
fs.CountError = Stats.Error
rc.Add(rc.Call{
Path: "core/stats",
Fn: Stats.RemoteStats,
Title: "Returns stats about current transfers.",
Help: `
This returns all available stats
rclone rc core/stats
Returns the following values:
` + "```" + `
{
"speed": average speed in bytes/sec since start of the process,
"bytes": total transferred bytes since the start of the process,
"errors": number of errors,
"fatalError": whether there has been at least one FatalError,
"retryError": whether there has been at least one non-NoRetryError,
"checks": number of checked files,
"transfers": number of transferred files,
"deletes" : number of deleted files,
"elapsedTime": time in seconds since the start of the process,
"lastError": last occurred error,
"transferring": an array of currently active file transfers:
[
{
"bytes": total transferred bytes for this file,
"eta": estimated time in seconds until file transfer completion
"name": name of the file,
"percentage": progress of the file transfer in percent,
"speed": speed in bytes/sec,
"speedAvg": speed in bytes/sec as an exponentially weighted moving average,
"size": size of the file in bytes
}
],
"checking": an array of names of currently active file checks
[]
}
` + "```" + `
Values for "transferring", "checking" and "lastError" are only assigned if data is available.
The value for "eta" is null if an eta cannot be determined.
`,
})
}
// StatsInfo accounts all transfers // StatsInfo accounts all transfers
type StatsInfo struct { type StatsInfo struct {
mu sync.RWMutex mu sync.RWMutex
@ -102,7 +47,7 @@ func NewStats() *StatsInfo {
} }
// RemoteStats returns stats for rc // RemoteStats returns stats for rc
func (s *StatsInfo) RemoteStats(ctx context.Context, in rc.Params) (out rc.Params, err error) { func (s *StatsInfo) RemoteStats() (out rc.Params, err error) {
out = make(rc.Params) out = make(rc.Params)
s.mu.RLock() s.mu.RLock()
dt := s.totalDuration() dt := s.totalDuration()
@ -237,7 +182,7 @@ func (s *StatsInfo) String() string {
// checking and transferring have their own locking so read // checking and transferring have their own locking so read
// here before lock to prevent deadlock on GetBytes // here before lock to prevent deadlock on GetBytes
transferring, checking := s.transferring.count(), s.checking.count() transferring, checking := s.transferring.count(), s.checking.count()
transferringBytesDone, transferringBytesTotal := s.transferring.progress() transferringBytesDone, transferringBytesTotal := s.transferring.progress(s)
s.mu.RLock() s.mu.RLock()
@ -325,10 +270,10 @@ Elapsed time: %10v
// Add per transfer stats if required // Add per transfer stats if required
if !fs.Config.StatsOneLine { if !fs.Config.StatsOneLine {
if !s.checking.empty() { if !s.checking.empty() {
_, _ = fmt.Fprintf(buf, "Checking:\n%s\n", s.checking) _, _ = fmt.Fprintf(buf, "Checking:\n%s\n", s.checking.String(s.inProgress))
} }
if !s.transferring.empty() { if !s.transferring.empty() {
_, _ = fmt.Fprintf(buf, "Transferring:\n%s\n", s.transferring) _, _ = fmt.Fprintf(buf, "Transferring:\n%s\n", s.transferring.String(s.inProgress))
} }
} }

View file

@ -0,0 +1,189 @@
package accounting
import (
"context"
"sync"
"github.com/ncw/rclone/fs/rc"
"github.com/ncw/rclone/fs"
)
const globalStats = "global_stats"
var groups *statsGroups
func remoteStats(ctx context.Context, in rc.Params) (rc.Params, error) {
// Check to see if we should filter by group.
group, err := in.GetString("group")
if rc.NotErrParamNotFound(err) {
return rc.Params{}, err
}
if group != "" {
return StatsGroup(group).RemoteStats()
}
return groups.sum().RemoteStats()
}
func init() {
// Init stats container
groups = newStatsGroups()
// Set the function pointer up in fs
fs.CountError = GlobalStats().Error
rc.Add(rc.Call{
Path: "core/stats",
Fn: remoteStats,
Title: "Returns stats about current transfers.",
Help: `
This returns all available stats
rclone rc core/stats
Returns the following values:
` + "```" + `
{
"speed": average speed in bytes/sec since start of the process,
"bytes": total transferred bytes since the start of the process,
"errors": number of errors,
"fatalError": whether there has been at least one FatalError,
"retryError": whether there has been at least one non-NoRetryError,
"checks": number of checked files,
"transfers": number of transferred files,
"deletes" : number of deleted files,
"elapsedTime": time in seconds since the start of the process,
"lastError": last occurred error,
"transferring": an array of currently active file transfers:
[
{
"bytes": total transferred bytes for this file,
"eta": estimated time in seconds until file transfer completion
"name": name of the file,
"percentage": progress of the file transfer in percent,
"speed": speed in bytes/sec,
"speedAvg": speed in bytes/sec as an exponentially weighted moving average,
"size": size of the file in bytes
}
],
"checking": an array of names of currently active file checks
[]
}
` + "```" + `
Values for "transferring", "checking" and "lastError" are only assigned if data is available.
The value for "eta" is null if an eta cannot be determined.
`,
})
}
type statsGroupCtx int64
const statsGroupKey statsGroupCtx = 1
// WithStatsGroup returns copy of the parent context with assigned group.
func WithStatsGroup(parent context.Context, group string) context.Context {
return context.WithValue(parent, statsGroupKey, group)
}
// StatsGroupFromContext returns group from the context if it's available.
// Returns false if group is empty.
func StatsGroupFromContext(ctx context.Context) (string, bool) {
statsGroup, ok := ctx.Value(statsGroupKey).(string)
if statsGroup == "" {
ok = false
}
return statsGroup, ok
}
// Stats gets stats by extracting group from context.
func Stats(ctx context.Context) *StatsInfo {
group, ok := StatsGroupFromContext(ctx)
if !ok {
return GlobalStats()
}
return StatsGroup(group)
}
// StatsGroup gets stats by group name.
func StatsGroup(group string) *StatsInfo {
stats := groups.get(group)
if stats == nil {
return NewStatsGroup(group)
}
return stats
}
// GlobalStats returns special stats used for global accounting.
func GlobalStats() *StatsInfo {
return StatsGroup(globalStats)
}
// NewStatsGroup creates new stats under named group.
func NewStatsGroup(group string) *StatsInfo {
stats := NewStats()
groups.set(group, stats)
return stats
}
// statsGroups holds a synchronized map of stats
type statsGroups struct {
mu sync.Mutex
m map[string]*StatsInfo
}
// newStatsGroups makes a new statsGroups object
func newStatsGroups() *statsGroups {
return &statsGroups{
m: make(map[string]*StatsInfo),
}
}
// set marks the stats as belonging to a group
func (sg *statsGroups) set(group string, acc *StatsInfo) {
sg.mu.Lock()
defer sg.mu.Unlock()
sg.m[group] = acc
}
// clear discards reference to group
func (sg *statsGroups) clear(group string) {
sg.mu.Lock()
defer sg.mu.Unlock()
delete(sg.m, group)
}
// get gets the stats for group, or nil if not found
func (sg *statsGroups) get(group string) *StatsInfo {
sg.mu.Lock()
defer sg.mu.Unlock()
stats, ok := sg.m[group]
if !ok {
return nil
}
return stats
}
// get gets the stats for group, or nil if not found
func (sg *statsGroups) sum() *StatsInfo {
sg.mu.Lock()
defer sg.mu.Unlock()
sum := NewStats()
for _, stats := range sg.m {
sum.bytes += stats.bytes
sum.errors += stats.errors
sum.fatalError = sum.fatalError || stats.fatalError
sum.retryError = sum.retryError || stats.retryError
sum.checks += stats.checks
sum.transfers += stats.transfers
sum.deletes += stats.deletes
sum.checking.merge(stats.checking)
sum.transferring.merge(stats.transferring)
sum.inProgress.merge(stats.inProgress)
if sum.lastError == nil && stats.lastError != nil {
sum.lastError = stats.lastError
}
}
return sum
}

View file

@ -38,6 +38,17 @@ func (ss *stringSet) del(remote string) {
ss.mu.Unlock() ss.mu.Unlock()
} }
// merge adds items from another set
func (ss *stringSet) merge(m *stringSet) {
ss.mu.Lock()
m.mu.Lock()
for item := range m.items {
ss.items[item] = struct{}{}
}
m.mu.Unlock()
ss.mu.Unlock()
}
// empty returns whether the set has any items // empty returns whether the set has any items
func (ss *stringSet) empty() bool { func (ss *stringSet) empty() bool {
ss.mu.RLock() ss.mu.RLock()
@ -52,14 +63,14 @@ func (ss *stringSet) count() int {
return len(ss.items) return len(ss.items)
} }
// Strings returns all the strings in the stringSet // String returns string representation of set items.
func (ss *stringSet) Strings() []string { func (ss *stringSet) String(progress *inProgress) string {
ss.mu.RLock() ss.mu.RLock()
defer ss.mu.RUnlock() defer ss.mu.RUnlock()
strings := make([]string, 0, len(ss.items)) strngs := make([]string, 0, len(ss.items))
for name := range ss.items { for name := range ss.items {
var out string var out string
if acc := Stats.inProgress.get(name); acc != nil { if acc := progress.get(name); acc != nil {
out = acc.String() out = acc.String()
} else { } else {
out = fmt.Sprintf("%*s: %s", out = fmt.Sprintf("%*s: %s",
@ -68,24 +79,19 @@ func (ss *stringSet) Strings() []string {
ss.name, ss.name,
) )
} }
strings = append(strings, " * "+out) strngs = append(strngs, " * "+out)
} }
sorted := sort.StringSlice(strings) sorted := sort.StringSlice(strngs)
sorted.Sort() sorted.Sort()
return sorted return strings.Join(sorted, "\n")
}
// String returns all the file names in the stringSet joined by newline
func (ss *stringSet) String() string {
return strings.Join(ss.Strings(), "\n")
} }
// progress returns total bytes read as well as the size. // progress returns total bytes read as well as the size.
func (ss *stringSet) progress() (totalBytes, totalSize int64) { func (ss *stringSet) progress(stats *StatsInfo) (totalBytes, totalSize int64) {
ss.mu.RLock() ss.mu.RLock()
defer ss.mu.RUnlock() defer ss.mu.RUnlock()
for name := range ss.items { for name := range ss.items {
if acc := Stats.inProgress.get(name); acc != nil { if acc := stats.inProgress.get(name); acc != nil {
bytes, size := acc.progress() bytes, size := acc.progress()
if size >= 0 && bytes >= 0 { if size >= 0 && bytes >= 0 {
totalBytes += bytes totalBytes += bytes

View file

@ -60,8 +60,8 @@ func TestMultithreadCopy(t *testing.T) {
src, err := r.Fremote.NewObject(context.Background(), "file1") src, err := r.Fremote.NewObject(context.Background(), "file1")
require.NoError(t, err) require.NoError(t, err)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
tr := accounting.Stats.NewTransfer(src) tr := accounting.GlobalStats().NewTransfer(src)
defer func() { defer func() {
tr.Done(err) tr.Done(err)

View file

@ -251,7 +251,7 @@ var _ fs.MimeTyper = (*overrideRemoteObject)(nil)
// It returns the destination object if possible. Note that this may // It returns the destination object if possible. Note that this may
// be nil. // be nil.
func Copy(ctx context.Context, f fs.Fs, dst fs.Object, remote string, src fs.Object) (newDst fs.Object, err error) { func Copy(ctx context.Context, f fs.Fs, dst fs.Object, remote string, src fs.Object) (newDst fs.Object, err error) {
tr := accounting.Stats.NewTransfer(src) tr := accounting.Stats(ctx).NewTransfer(src)
defer func() { defer func() {
tr.Done(err) tr.Done(err)
}() }()
@ -281,13 +281,13 @@ func Copy(ctx context.Context, f fs.Fs, dst fs.Object, remote string, src fs.Obj
actionTaken = "Copied (server side copy)" actionTaken = "Copied (server side copy)"
if doCopy := f.Features().Copy; doCopy != nil && (SameConfig(src.Fs(), f) || (SameRemoteType(src.Fs(), f) && f.Features().ServerSideAcrossConfigs)) { if doCopy := f.Features().Copy; doCopy != nil && (SameConfig(src.Fs(), f) || (SameRemoteType(src.Fs(), f) && f.Features().ServerSideAcrossConfigs)) {
// Check transfer limit for server side copies // Check transfer limit for server side copies
if fs.Config.MaxTransfer >= 0 && accounting.Stats.GetBytes() >= int64(fs.Config.MaxTransfer) { if fs.Config.MaxTransfer >= 0 && accounting.Stats(ctx).GetBytes() >= int64(fs.Config.MaxTransfer) {
return nil, accounting.ErrorMaxTransferLimitReached return nil, accounting.ErrorMaxTransferLimitReached
} }
newDst, err = doCopy(ctx, src, remote) newDst, err = doCopy(ctx, src, remote)
if err == nil { if err == nil {
dst = newDst dst = newDst
accounting.Stats.Bytes(dst.Size()) // account the bytes for the server side transfer accounting.Stats(ctx).Bytes(dst.Size()) // account the bytes for the server side transfer
} }
} else { } else {
err = fs.ErrorCantCopy err = fs.ErrorCantCopy
@ -428,9 +428,9 @@ func SameObject(src, dst fs.Object) bool {
// It returns the destination object if possible. Note that this may // It returns the destination object if possible. Note that this may
// be nil. // be nil.
func Move(ctx context.Context, fdst fs.Fs, dst fs.Object, remote string, src fs.Object) (newDst fs.Object, err error) { func Move(ctx context.Context, fdst fs.Fs, dst fs.Object, remote string, src fs.Object) (newDst fs.Object, err error) {
accounting.Stats.Checking(src.Remote()) accounting.Stats(ctx).Checking(src.Remote())
defer func() { defer func() {
accounting.Stats.DoneChecking(src.Remote()) accounting.Stats(ctx).DoneChecking(src.Remote())
}() }()
newDst = dst newDst = dst
if fs.Config.DryRun { if fs.Config.DryRun {
@ -501,8 +501,8 @@ func SuffixName(remote string) string {
// If backupDir is set then it moves the file to there instead of // If backupDir is set then it moves the file to there instead of
// deleting // deleting
func DeleteFileWithBackupDir(ctx context.Context, dst fs.Object, backupDir fs.Fs) (err error) { func DeleteFileWithBackupDir(ctx context.Context, dst fs.Object, backupDir fs.Fs) (err error) {
accounting.Stats.Checking(dst.Remote()) accounting.Stats(ctx).Checking(dst.Remote())
numDeletes := accounting.Stats.Deletes(1) numDeletes := accounting.Stats(ctx).Deletes(1)
if fs.Config.MaxDelete != -1 && numDeletes > fs.Config.MaxDelete { if fs.Config.MaxDelete != -1 && numDeletes > fs.Config.MaxDelete {
return fserrors.FatalError(errors.New("--max-delete threshold reached")) return fserrors.FatalError(errors.New("--max-delete threshold reached"))
} }
@ -523,7 +523,7 @@ func DeleteFileWithBackupDir(ctx context.Context, dst fs.Object, backupDir fs.Fs
} else if !fs.Config.DryRun { } else if !fs.Config.DryRun {
fs.Infof(dst, actioned) fs.Infof(dst, actioned)
} }
accounting.Stats.DoneChecking(dst.Remote()) accounting.Stats(ctx).DoneChecking(dst.Remote())
return err return err
} }
@ -709,8 +709,8 @@ func (c *checkMarch) SrcOnly(src fs.DirEntry) (recurse bool) {
// check to see if two objects are identical using the check function // check to see if two objects are identical using the check function
func (c *checkMarch) checkIdentical(ctx context.Context, dst, src fs.Object) (differ bool, noHash bool) { func (c *checkMarch) checkIdentical(ctx context.Context, dst, src fs.Object) (differ bool, noHash bool) {
accounting.Stats.Checking(src.Remote()) accounting.Stats(ctx).Checking(src.Remote())
defer accounting.Stats.DoneChecking(src.Remote()) defer accounting.Stats(ctx).DoneChecking(src.Remote())
if sizeDiffers(src, dst) { if sizeDiffers(src, dst) {
err := errors.Errorf("Sizes differ") err := errors.Errorf("Sizes differ")
fs.Errorf(src, "%v", err) fs.Errorf(src, "%v", err)
@ -797,7 +797,7 @@ func CheckFn(ctx context.Context, fdst, fsrc fs.Fs, check checkFn, oneway bool)
fs.Logf(fsrc, "%d files missing", c.srcFilesMissing) fs.Logf(fsrc, "%d files missing", c.srcFilesMissing)
} }
fs.Logf(fdst, "%d differences found", accounting.Stats.GetErrors()) fs.Logf(fdst, "%d differences found", accounting.Stats(ctx).GetErrors())
if c.noHashes > 0 { if c.noHashes > 0 {
fs.Logf(fdst, "%d hashes could not be checked", c.noHashes) fs.Logf(fdst, "%d hashes could not be checked", c.noHashes)
} }
@ -854,7 +854,7 @@ func CheckIdentical(ctx context.Context, dst, src fs.Object) (differ bool, err e
if err != nil { if err != nil {
return true, errors.Wrapf(err, "failed to open %q", dst) return true, errors.Wrapf(err, "failed to open %q", dst)
} }
tr1 := accounting.Stats.NewTransfer(dst) tr1 := accounting.Stats(ctx).NewTransfer(dst)
defer func() { defer func() {
tr1.Done(err) tr1.Done(err)
}() }()
@ -864,7 +864,7 @@ func CheckIdentical(ctx context.Context, dst, src fs.Object) (differ bool, err e
if err != nil { if err != nil {
return true, errors.Wrapf(err, "failed to open %q", src) return true, errors.Wrapf(err, "failed to open %q", src)
} }
tr2 := accounting.Stats.NewTransfer(dst) tr2 := accounting.Stats(ctx).NewTransfer(dst)
defer func() { defer func() {
tr2.Done(err) tr2.Done(err)
}() }()
@ -930,9 +930,9 @@ func List(ctx context.Context, f fs.Fs, w io.Writer) error {
// Lists in parallel which may get them out of order // Lists in parallel which may get them out of order
func ListLong(ctx context.Context, f fs.Fs, w io.Writer) error { func ListLong(ctx context.Context, f fs.Fs, w io.Writer) error {
return ListFn(ctx, f, func(o fs.Object) { return ListFn(ctx, f, func(o fs.Object) {
accounting.Stats.Checking(o.Remote()) accounting.Stats(ctx).Checking(o.Remote())
modTime := o.ModTime(ctx) modTime := o.ModTime(ctx)
accounting.Stats.DoneChecking(o.Remote()) accounting.Stats(ctx).DoneChecking(o.Remote())
syncFprintf(w, "%9d %s %s\n", o.Size(), modTime.Local().Format("2006-01-02 15:04:05.000000000"), o.Remote()) syncFprintf(w, "%9d %s %s\n", o.Size(), modTime.Local().Format("2006-01-02 15:04:05.000000000"), o.Remote())
}) })
} }
@ -968,9 +968,9 @@ func DropboxHashSum(ctx context.Context, f fs.Fs, w io.Writer) error {
// hashSum returns the human readable hash for ht passed in. This may // hashSum returns the human readable hash for ht passed in. This may
// be UNSUPPORTED or ERROR. // be UNSUPPORTED or ERROR.
func hashSum(ctx context.Context, ht hash.Type, o fs.Object) string { func hashSum(ctx context.Context, ht hash.Type, o fs.Object) string {
accounting.Stats.Checking(o.Remote()) accounting.Stats(ctx).Checking(o.Remote())
sum, err := o.Hash(ctx, ht) sum, err := o.Hash(ctx, ht)
accounting.Stats.DoneChecking(o.Remote()) accounting.Stats(ctx).DoneChecking(o.Remote())
if err == hash.ErrUnsupported { if err == hash.ErrUnsupported {
sum = "UNSUPPORTED" sum = "UNSUPPORTED"
} else if err != nil { } else if err != nil {
@ -1167,7 +1167,7 @@ func Cat(ctx context.Context, f fs.Fs, w io.Writer, offset, count int64) error {
var mu sync.Mutex var mu sync.Mutex
return ListFn(ctx, f, func(o fs.Object) { return ListFn(ctx, f, func(o fs.Object) {
var err error var err error
tr := accounting.Stats.NewTransfer(o) tr := accounting.Stats(ctx).NewTransfer(o)
defer func() { defer func() {
tr.Done(err) tr.Done(err)
}() }()
@ -1206,7 +1206,7 @@ func Cat(ctx context.Context, f fs.Fs, w io.Writer, offset, count int64) error {
// Rcat reads data from the Reader until EOF and uploads it to a file on remote // Rcat reads data from the Reader until EOF and uploads it to a file on remote
func Rcat(ctx context.Context, fdst fs.Fs, dstFileName string, in io.ReadCloser, modTime time.Time) (dst fs.Object, err error) { func Rcat(ctx context.Context, fdst fs.Fs, dstFileName string, in io.ReadCloser, modTime time.Time) (dst fs.Object, err error) {
tr := accounting.Stats.NewTransferRemoteSize(dstFileName, -1) tr := accounting.Stats(ctx).NewTransferRemoteSize(dstFileName, -1)
defer func() { defer func() {
tr.Done(err) tr.Done(err)
}() }()
@ -1527,7 +1527,7 @@ func RcatSize(ctx context.Context, fdst fs.Fs, dstFileName string, in io.ReadClo
if size >= 0 { if size >= 0 {
var err error var err error
// Size known use Put // Size known use Put
tr := accounting.Stats.NewTransferRemoteSize(dstFileName, size) tr := accounting.Stats(ctx).NewTransferRemoteSize(dstFileName, size)
defer func() { defer func() {
tr.Done(err) tr.Done(err)
}() }()
@ -1664,7 +1664,7 @@ func moveOrCopyFile(ctx context.Context, fdst fs.Fs, fsrc fs.Fs, dstFileName str
} }
return errors.Wrap(err, "error while attempting to move file to a temporary location") return errors.Wrap(err, "error while attempting to move file to a temporary location")
} }
tr := accounting.Stats.NewTransfer(srcObj) tr := accounting.Stats(ctx).NewTransfer(srcObj)
defer func() { defer func() {
tr.Done(err) tr.Done(err)
}() }()
@ -1711,11 +1711,11 @@ func moveOrCopyFile(ctx context.Context, fdst fs.Fs, fsrc fs.Fs, dstFileName str
_, err = Op(ctx, fdst, dstObj, dstFileName, srcObj) _, err = Op(ctx, fdst, dstObj, dstFileName, srcObj)
} else { } else {
accounting.Stats.Checking(srcFileName) accounting.Stats(ctx).Checking(srcFileName)
if !cp { if !cp {
err = DeleteFile(ctx, srcObj) err = DeleteFile(ctx, srcObj)
} }
defer accounting.Stats.DoneChecking(srcFileName) defer accounting.Stats(ctx).DoneChecking(srcFileName)
} }
return err return err
} }

View file

@ -14,7 +14,7 @@
// fstest.CheckItems() before use. This make sure the directory // fstest.CheckItems() before use. This make sure the directory
// listing is now consistent and stops cascading errors. // listing is now consistent and stops cascading errors.
// //
// Call accounting.Stats.ResetCounters() before every fs.Sync() as it // Call accounting.GlobalStats().ResetCounters() before every fs.Sync() as it
// uses the error count internally. // uses the error count internally.
package operations_test package operations_test
@ -315,15 +315,15 @@ func testCheck(t *testing.T, checkFunction func(ctx context.Context, fdst, fsrc
check := func(i int, wantErrors int64, wantChecks int64, oneway bool) { check := func(i int, wantErrors int64, wantChecks int64, oneway bool) {
fs.Debugf(r.Fremote, "%d: Starting check test", i) fs.Debugf(r.Fremote, "%d: Starting check test", i)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
var buf bytes.Buffer var buf bytes.Buffer
log.SetOutput(&buf) log.SetOutput(&buf)
defer func() { defer func() {
log.SetOutput(os.Stderr) log.SetOutput(os.Stderr)
}() }()
err := checkFunction(context.Background(), r.Fremote, r.Flocal, oneway) err := checkFunction(context.Background(), r.Fremote, r.Flocal, oneway)
gotErrors := accounting.Stats.GetErrors() gotErrors := accounting.GlobalStats().GetErrors()
gotChecks := accounting.Stats.GetChecks() gotChecks := accounting.GlobalStats().GetChecks()
if wantErrors == 0 && err != nil { if wantErrors == 0 && err != nil {
t.Errorf("%d: Got error when not expecting one: %v", i, err) t.Errorf("%d: Got error when not expecting one: %v", i, err)
} }

View file

@ -1,13 +1,18 @@
// Manage background jobs that the rc is running // Manage background jobs that the rc is running
package rc package jobs
import ( import (
"context" "context"
"fmt"
"sync" "sync"
"sync/atomic" "sync/atomic"
"time" "time"
"github.com/ncw/rclone/fs/rc"
"github.com/ncw/rclone/fs/accounting"
"github.com/ncw/rclone/fs" "github.com/ncw/rclone/fs"
"github.com/pkg/errors" "github.com/pkg/errors"
) )
@ -16,13 +21,14 @@ import (
type Job struct { type Job struct {
mu sync.Mutex mu sync.Mutex
ID int64 `json:"id"` ID int64 `json:"id"`
Group string `json:"group"`
StartTime time.Time `json:"startTime"` StartTime time.Time `json:"startTime"`
EndTime time.Time `json:"endTime"` EndTime time.Time `json:"endTime"`
Error string `json:"error"` Error string `json:"error"`
Finished bool `json:"finished"` Finished bool `json:"finished"`
Success bool `json:"success"` Success bool `json:"success"`
Duration float64 `json:"duration"` Duration float64 `json:"duration"`
Output Params `json:"output"` Output rc.Params `json:"output"`
Stop func() `json:"-"` Stop func() `json:"-"`
} }
@ -96,11 +102,11 @@ func (jobs *Jobs) Get(ID int64) *Job {
} }
// mark the job as finished // mark the job as finished
func (job *Job) finish(out Params, err error) { func (job *Job) finish(out rc.Params, err error) {
job.mu.Lock() job.mu.Lock()
job.EndTime = time.Now() job.EndTime = time.Now()
if out == nil { if out == nil {
out = make(Params) out = make(rc.Params)
} }
job.Output = out job.Output = out
job.Duration = job.EndTime.Sub(job.StartTime).Seconds() job.Duration = job.EndTime.Sub(job.StartTime).Seconds()
@ -117,7 +123,7 @@ func (job *Job) finish(out Params, err error) {
} }
// run the job until completion writing the return status // run the job until completion writing the return status
func (job *Job) run(ctx context.Context, fn Func, in Params) { func (job *Job) run(ctx context.Context, fn rc.Func, in rc.Params) {
defer func() { defer func() {
if r := recover(); r != nil { if r := recover(); r != nil {
job.finish(nil, errors.Errorf("panic received: %v", r)) job.finish(nil, errors.Errorf("panic received: %v", r))
@ -126,37 +132,93 @@ func (job *Job) run(ctx context.Context, fn Func, in Params) {
job.finish(fn(ctx, in)) job.finish(fn(ctx, in))
} }
// NewJob start a new Job off func getGroup(in rc.Params) string {
func (jobs *Jobs) NewJob(fn Func, in Params) *Job { // Check to see if the group is set
ctx, cancel := context.WithCancel(context.Background()) group, err := in.GetString("_group")
if rc.NotErrParamNotFound(err) {
fs.Errorf(nil, "Can't get _group param %+v", err)
}
delete(in, "_group")
return group
}
// NewAsyncJob start a new asynchronous Job off
func (jobs *Jobs) NewAsyncJob(fn rc.Func, in rc.Params) *Job {
id := atomic.AddInt64(&jobID, 1)
group := getGroup(in)
if group == "" {
group = fmt.Sprintf("job/%d", id)
}
ctx := accounting.WithStatsGroup(context.Background(), group)
ctx, cancel := context.WithCancel(ctx)
stop := func() { stop := func() {
cancel() cancel()
// Wait for cancel to propagate before returning. // Wait for cancel to propagate before returning.
<-ctx.Done() <-ctx.Done()
} }
job := &Job{ job := &Job{
ID: atomic.AddInt64(&jobID, 1), ID: id,
Group: group,
StartTime: time.Now(), StartTime: time.Now(),
Stop: stop, Stop: stop,
} }
go job.run(ctx, fn, in)
jobs.mu.Lock() jobs.mu.Lock()
jobs.jobs[job.ID] = job jobs.jobs[job.ID] = job
jobs.mu.Unlock() jobs.mu.Unlock()
go job.run(ctx, fn, in)
return job return job
} }
// StartJob starts a new job and returns a Param suitable for output // NewSyncJob start a new synchronous Job off
func StartJob(fn Func, in Params) (Params, error) { func (jobs *Jobs) NewSyncJob(ctx context.Context, in rc.Params) (*Job, context.Context) {
job := running.NewJob(fn, in) id := atomic.AddInt64(&jobID, 1)
out := make(Params) group := getGroup(in)
if group == "" {
group = fmt.Sprintf("job/%d", id)
}
ctxG := accounting.WithStatsGroup(ctx, fmt.Sprintf("job/%d", id))
ctx, cancel := context.WithCancel(ctxG)
stop := func() {
cancel()
// Wait for cancel to propagate before returning.
<-ctx.Done()
}
job := &Job{
ID: id,
Group: group,
StartTime: time.Now(),
Stop: stop,
}
jobs.mu.Lock()
jobs.jobs[job.ID] = job
jobs.mu.Unlock()
return job, ctx
}
// StartAsyncJob starts a new job asynchronously and returns a Param suitable
// for output.
func StartAsyncJob(fn rc.Func, in rc.Params) (rc.Params, error) {
job := running.NewAsyncJob(fn, in)
out := make(rc.Params)
out["jobid"] = job.ID out["jobid"] = job.ID
return out, nil return out, nil
} }
// ExecuteJob executes new job synchronously and returns a Param suitable for
// output.
func ExecuteJob(ctx context.Context, fn rc.Func, in rc.Params) (rc.Params, int64, error) {
job, ctx := running.NewSyncJob(ctx, in)
job.run(ctx, fn, in)
var err error
if !job.Success {
err = errors.New(job.Error)
}
return job.Output, job.ID, err
}
func init() { func init() {
Add(Call{ rc.Add(rc.Call{
Path: "job/status", Path: "job/status",
Fn: rcJobStatus, Fn: rcJobStatus,
Title: "Reads the status of the job ID", Title: "Reads the status of the job ID",
@ -179,7 +241,7 @@ Results
} }
// Returns the status of a job // Returns the status of a job
func rcJobStatus(ctx context.Context, in Params) (out Params, err error) { func rcJobStatus(ctx context.Context, in rc.Params) (out rc.Params, err error) {
jobID, err := in.GetInt64("jobid") jobID, err := in.GetInt64("jobid")
if err != nil { if err != nil {
return nil, err return nil, err
@ -190,8 +252,8 @@ func rcJobStatus(ctx context.Context, in Params) (out Params, err error) {
} }
job.mu.Lock() job.mu.Lock()
defer job.mu.Unlock() defer job.mu.Unlock()
out = make(Params) out = make(rc.Params)
err = Reshape(&out, job) err = rc.Reshape(&out, job)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "reshape failed in job status") return nil, errors.Wrap(err, "reshape failed in job status")
} }
@ -199,7 +261,7 @@ func rcJobStatus(ctx context.Context, in Params) (out Params, err error) {
} }
func init() { func init() {
Add(Call{ rc.Add(rc.Call{
Path: "job/list", Path: "job/list",
Fn: rcJobList, Fn: rcJobList,
Title: "Lists the IDs of the running jobs", Title: "Lists the IDs of the running jobs",
@ -212,14 +274,14 @@ Results
} }
// Returns list of job ids. // Returns list of job ids.
func rcJobList(ctx context.Context, in Params) (out Params, err error) { func rcJobList(ctx context.Context, in rc.Params) (out rc.Params, err error) {
out = make(Params) out = make(rc.Params)
out["jobids"] = running.IDs() out["jobids"] = running.IDs()
return out, nil return out, nil
} }
func init() { func init() {
Add(Call{ rc.Add(rc.Call{
Path: "job/stop", Path: "job/stop",
Fn: rcJobStop, Fn: rcJobStop,
Title: "Stop the running job", Title: "Stop the running job",
@ -230,7 +292,7 @@ func init() {
} }
// Stops the running job. // Stops the running job.
func rcJobStop(ctx context.Context, in Params) (out Params, err error) { func rcJobStop(ctx context.Context, in rc.Params) (out rc.Params, err error) {
jobID, err := in.GetInt64("jobid") jobID, err := in.GetInt64("jobid")
if err != nil { if err != nil {
return nil, err return nil, err
@ -241,7 +303,7 @@ func rcJobStop(ctx context.Context, in Params) (out Params, err error) {
} }
job.mu.Lock() job.mu.Lock()
defer job.mu.Unlock() defer job.mu.Unlock()
out = make(Params) out = make(rc.Params)
job.Stop() job.Stop()
return out, nil return out, nil
} }

View file

@ -1,4 +1,4 @@
package rc package jobs
import ( import (
"context" "context"
@ -7,6 +7,7 @@ import (
"time" "time"
"github.com/ncw/rclone/fs" "github.com/ncw/rclone/fs"
"github.com/ncw/rclone/fs/rc"
"github.com/pkg/errors" "github.com/pkg/errors"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
@ -36,10 +37,10 @@ func TestJobsExpire(t *testing.T) {
jobs := newJobs() jobs := newJobs()
jobs.expireInterval = time.Millisecond jobs.expireInterval = time.Millisecond
assert.Equal(t, false, jobs.expireRunning) assert.Equal(t, false, jobs.expireRunning)
job := jobs.NewJob(func(ctx context.Context, in Params) (Params, error) { job := jobs.NewAsyncJob(func(ctx context.Context, in rc.Params) (rc.Params, error) {
defer close(wait) defer close(wait)
return in, nil return in, nil
}, Params{}) }, rc.Params{})
<-wait <-wait
assert.Equal(t, 1, len(jobs.jobs)) assert.Equal(t, 1, len(jobs.jobs))
jobs.Expire() jobs.Expire()
@ -57,14 +58,14 @@ func TestJobsExpire(t *testing.T) {
jobs.mu.Unlock() jobs.mu.Unlock()
} }
var noopFn = func(ctx context.Context, in Params) (Params, error) { var noopFn = func(ctx context.Context, in rc.Params) (rc.Params, error) {
return nil, nil return nil, nil
} }
func TestJobsIDs(t *testing.T) { func TestJobsIDs(t *testing.T) {
jobs := newJobs() jobs := newJobs()
job1 := jobs.NewJob(noopFn, Params{}) job1 := jobs.NewAsyncJob(noopFn, rc.Params{})
job2 := jobs.NewJob(noopFn, Params{}) job2 := jobs.NewAsyncJob(noopFn, rc.Params{})
wantIDs := []int64{job1.ID, job2.ID} wantIDs := []int64{job1.ID, job2.ID}
gotIDs := jobs.IDs() gotIDs := jobs.IDs()
require.Equal(t, 2, len(gotIDs)) require.Equal(t, 2, len(gotIDs))
@ -76,17 +77,22 @@ func TestJobsIDs(t *testing.T) {
func TestJobsGet(t *testing.T) { func TestJobsGet(t *testing.T) {
jobs := newJobs() jobs := newJobs()
job := jobs.NewJob(noopFn, Params{}) job := jobs.NewAsyncJob(noopFn, rc.Params{})
assert.Equal(t, job, jobs.Get(job.ID)) assert.Equal(t, job, jobs.Get(job.ID))
assert.Nil(t, jobs.Get(123123123123)) assert.Nil(t, jobs.Get(123123123123))
} }
var longFn = func(ctx context.Context, in Params) (Params, error) { var longFn = func(ctx context.Context, in rc.Params) (rc.Params, error) {
time.Sleep(1 * time.Hour) time.Sleep(1 * time.Hour)
return nil, nil return nil, nil
} }
var ctxFn = func(ctx context.Context, in Params) (Params, error) { var shortFn = func(ctx context.Context, in rc.Params) (rc.Params, error) {
time.Sleep(time.Millisecond)
return nil, nil
}
var ctxFn = func(ctx context.Context, in rc.Params) (rc.Params, error) {
select { select {
case <-ctx.Done(): case <-ctx.Done():
return nil, ctx.Err() return nil, ctx.Err()
@ -105,17 +111,17 @@ func sleepJob() {
func TestJobFinish(t *testing.T) { func TestJobFinish(t *testing.T) {
jobs := newJobs() jobs := newJobs()
job := jobs.NewJob(longFn, Params{}) job := jobs.NewAsyncJob(longFn, rc.Params{})
sleepJob() sleepJob()
assert.Equal(t, true, job.EndTime.IsZero()) assert.Equal(t, true, job.EndTime.IsZero())
assert.Equal(t, Params(nil), job.Output) assert.Equal(t, rc.Params(nil), job.Output)
assert.Equal(t, 0.0, job.Duration) assert.Equal(t, 0.0, job.Duration)
assert.Equal(t, "", job.Error) assert.Equal(t, "", job.Error)
assert.Equal(t, false, job.Success) assert.Equal(t, false, job.Success)
assert.Equal(t, false, job.Finished) assert.Equal(t, false, job.Finished)
wantOut := Params{"a": 1} wantOut := rc.Params{"a": 1}
job.finish(wantOut, nil) job.finish(wantOut, nil)
assert.Equal(t, false, job.EndTime.IsZero()) assert.Equal(t, false, job.EndTime.IsZero())
@ -125,18 +131,18 @@ func TestJobFinish(t *testing.T) {
assert.Equal(t, true, job.Success) assert.Equal(t, true, job.Success)
assert.Equal(t, true, job.Finished) assert.Equal(t, true, job.Finished)
job = jobs.NewJob(longFn, Params{}) job = jobs.NewAsyncJob(longFn, rc.Params{})
sleepJob() sleepJob()
job.finish(nil, nil) job.finish(nil, nil)
assert.Equal(t, false, job.EndTime.IsZero()) assert.Equal(t, false, job.EndTime.IsZero())
assert.Equal(t, Params{}, job.Output) assert.Equal(t, rc.Params{}, job.Output)
assert.True(t, job.Duration >= floatSleepTime) assert.True(t, job.Duration >= floatSleepTime)
assert.Equal(t, "", job.Error) assert.Equal(t, "", job.Error)
assert.Equal(t, true, job.Success) assert.Equal(t, true, job.Success)
assert.Equal(t, true, job.Finished) assert.Equal(t, true, job.Finished)
job = jobs.NewJob(longFn, Params{}) job = jobs.NewAsyncJob(longFn, rc.Params{})
sleepJob() sleepJob()
job.finish(wantOut, errors.New("potato")) job.finish(wantOut, errors.New("potato"))
@ -152,14 +158,14 @@ func TestJobFinish(t *testing.T) {
// part of NewJob, now just test the panic catching // part of NewJob, now just test the panic catching
func TestJobRunPanic(t *testing.T) { func TestJobRunPanic(t *testing.T) {
wait := make(chan struct{}) wait := make(chan struct{})
boom := func(ctx context.Context, in Params) (Params, error) { boom := func(ctx context.Context, in rc.Params) (rc.Params, error) {
sleepJob() sleepJob()
defer close(wait) defer close(wait)
panic("boom") panic("boom")
} }
jobs := newJobs() jobs := newJobs()
job := jobs.NewJob(boom, Params{}) job := jobs.NewAsyncJob(boom, rc.Params{})
<-wait <-wait
runtime.Gosched() // yield to make sure job is updated runtime.Gosched() // yield to make sure job is updated
@ -176,7 +182,7 @@ func TestJobRunPanic(t *testing.T) {
job.mu.Lock() job.mu.Lock()
assert.Equal(t, false, job.EndTime.IsZero()) assert.Equal(t, false, job.EndTime.IsZero())
assert.Equal(t, Params{}, job.Output) assert.Equal(t, rc.Params{}, job.Output)
assert.True(t, job.Duration >= floatSleepTime) assert.True(t, job.Duration >= floatSleepTime)
assert.Equal(t, "panic received: boom", job.Error) assert.Equal(t, "panic received: boom", job.Error)
assert.Equal(t, false, job.Success) assert.Equal(t, false, job.Success)
@ -187,7 +193,7 @@ func TestJobRunPanic(t *testing.T) {
func TestJobsNewJob(t *testing.T) { func TestJobsNewJob(t *testing.T) {
jobID = 0 jobID = 0
jobs := newJobs() jobs := newJobs()
job := jobs.NewJob(noopFn, Params{}) job := jobs.NewAsyncJob(noopFn, rc.Params{})
assert.Equal(t, int64(1), job.ID) assert.Equal(t, int64(1), job.ID)
assert.Equal(t, job, jobs.Get(1)) assert.Equal(t, job, jobs.Get(1))
assert.NotEmpty(t, job.Stop) assert.NotEmpty(t, job.Stop)
@ -195,19 +201,26 @@ func TestJobsNewJob(t *testing.T) {
func TestStartJob(t *testing.T) { func TestStartJob(t *testing.T) {
jobID = 0 jobID = 0
out, err := StartJob(longFn, Params{}) out, err := StartAsyncJob(longFn, rc.Params{})
assert.NoError(t, err) assert.NoError(t, err)
assert.Equal(t, Params{"jobid": int64(1)}, out) assert.Equal(t, rc.Params{"jobid": int64(1)}, out)
}
func TestExecuteJob(t *testing.T) {
jobID = 0
_, id, err := ExecuteJob(context.Background(), shortFn, rc.Params{})
assert.NoError(t, err)
assert.Equal(t, int64(1), id)
} }
func TestRcJobStatus(t *testing.T) { func TestRcJobStatus(t *testing.T) {
jobID = 0 jobID = 0
_, err := StartJob(longFn, Params{}) _, err := StartAsyncJob(longFn, rc.Params{})
assert.NoError(t, err) assert.NoError(t, err)
call := Calls.Get("job/status") call := rc.Calls.Get("job/status")
assert.NotNil(t, call) assert.NotNil(t, call)
in := Params{"jobid": 1} in := rc.Params{"jobid": 1}
out, err := call.Fn(context.Background(), in) out, err := call.Fn(context.Background(), in)
require.NoError(t, err) require.NoError(t, err)
require.NotNil(t, out) require.NotNil(t, out)
@ -216,12 +229,12 @@ func TestRcJobStatus(t *testing.T) {
assert.Equal(t, false, out["finished"]) assert.Equal(t, false, out["finished"])
assert.Equal(t, false, out["success"]) assert.Equal(t, false, out["success"])
in = Params{"jobid": 123123123} in = rc.Params{"jobid": 123123123}
_, err = call.Fn(context.Background(), in) _, err = call.Fn(context.Background(), in)
require.Error(t, err) require.Error(t, err)
assert.Contains(t, err.Error(), "job not found") assert.Contains(t, err.Error(), "job not found")
in = Params{"jobidx": 123123123} in = rc.Params{"jobidx": 123123123}
_, err = call.Fn(context.Background(), in) _, err = call.Fn(context.Background(), in)
require.Error(t, err) require.Error(t, err)
assert.Contains(t, err.Error(), "Didn't find key") assert.Contains(t, err.Error(), "Didn't find key")
@ -229,45 +242,88 @@ func TestRcJobStatus(t *testing.T) {
func TestRcJobList(t *testing.T) { func TestRcJobList(t *testing.T) {
jobID = 0 jobID = 0
_, err := StartJob(longFn, Params{}) _, err := StartAsyncJob(longFn, rc.Params{})
assert.NoError(t, err) assert.NoError(t, err)
call := Calls.Get("job/list") call := rc.Calls.Get("job/list")
assert.NotNil(t, call) assert.NotNil(t, call)
in := Params{} in := rc.Params{}
out, err := call.Fn(context.Background(), in) out, err := call.Fn(context.Background(), in)
require.NoError(t, err) require.NoError(t, err)
require.NotNil(t, out) require.NotNil(t, out)
assert.Equal(t, Params{"jobids": []int64{1}}, out) assert.Equal(t, rc.Params{"jobids": []int64{1}}, out)
} }
func TestRcJobStop(t *testing.T) { func TestRcAsyncJobStop(t *testing.T) {
jobID = 0 jobID = 0
_, err := StartJob(ctxFn, Params{}) _, err := StartAsyncJob(ctxFn, rc.Params{})
assert.NoError(t, err) assert.NoError(t, err)
call := Calls.Get("job/stop") call := rc.Calls.Get("job/stop")
assert.NotNil(t, call) assert.NotNil(t, call)
in := Params{"jobid": 1} in := rc.Params{"jobid": 1}
out, err := call.Fn(context.Background(), in) out, err := call.Fn(context.Background(), in)
require.NoError(t, err) require.NoError(t, err)
require.Empty(t, out) require.Empty(t, out)
in = Params{"jobid": 123123123} in = rc.Params{"jobid": 123123123}
_, err = call.Fn(context.Background(), in) _, err = call.Fn(context.Background(), in)
require.Error(t, err) require.Error(t, err)
assert.Contains(t, err.Error(), "job not found") assert.Contains(t, err.Error(), "job not found")
in = Params{"jobidx": 123123123} in = rc.Params{"jobidx": 123123123}
_, err = call.Fn(context.Background(), in) _, err = call.Fn(context.Background(), in)
require.Error(t, err) require.Error(t, err)
assert.Contains(t, err.Error(), "Didn't find key") assert.Contains(t, err.Error(), "Didn't find key")
time.Sleep(10 * time.Millisecond) time.Sleep(10 * time.Millisecond)
call = Calls.Get("job/status") call = rc.Calls.Get("job/status")
assert.NotNil(t, call) assert.NotNil(t, call)
in = Params{"jobid": 1} in = rc.Params{"jobid": 1}
out, err = call.Fn(context.Background(), in)
require.NoError(t, err)
require.NotNil(t, out)
assert.Equal(t, float64(1), out["id"])
assert.Equal(t, "context canceled", out["error"])
assert.Equal(t, true, out["finished"])
assert.Equal(t, false, out["success"])
}
func TestRcSyncJobStop(t *testing.T) {
ctx, cancel := context.WithCancel(context.Background())
go func() {
jobID = 0
_, id, err := ExecuteJob(ctx, ctxFn, rc.Params{})
assert.Error(t, err)
assert.Equal(t, int64(1), id)
}()
time.Sleep(10 * time.Millisecond)
call := rc.Calls.Get("job/stop")
assert.NotNil(t, call)
in := rc.Params{"jobid": 1}
out, err := call.Fn(context.Background(), in)
require.NoError(t, err)
require.Empty(t, out)
in = rc.Params{"jobid": 123123123}
_, err = call.Fn(context.Background(), in)
require.Error(t, err)
assert.Contains(t, err.Error(), "job not found")
in = rc.Params{"jobidx": 123123123}
_, err = call.Fn(context.Background(), in)
require.Error(t, err)
assert.Contains(t, err.Error(), "Didn't find key")
cancel()
time.Sleep(10 * time.Millisecond)
call = rc.Calls.Get("job/status")
assert.NotNil(t, call)
in = rc.Params{"jobid": 1}
out, err = call.Fn(context.Background(), in) out, err = call.Fn(context.Background(), in)
require.NoError(t, err) require.NoError(t, err)
require.NotNil(t, out) require.NotNil(t, out)

View file

@ -3,6 +3,8 @@ package rcserver
import ( import (
"encoding/json" "encoding/json"
"flag"
"fmt"
"mime" "mime"
"net/http" "net/http"
"net/url" "net/url"
@ -10,6 +12,10 @@ import (
"sort" "sort"
"strings" "strings"
"github.com/skratchdot/open-golang/open"
"github.com/ncw/rclone/fs/rc/jobs"
"github.com/ncw/rclone/cmd/serve/httplib" "github.com/ncw/rclone/cmd/serve/httplib"
"github.com/ncw/rclone/cmd/serve/httplib/serve" "github.com/ncw/rclone/cmd/serve/httplib/serve"
"github.com/ncw/rclone/fs" "github.com/ncw/rclone/fs"
@ -18,7 +24,6 @@ import (
"github.com/ncw/rclone/fs/list" "github.com/ncw/rclone/fs/list"
"github.com/ncw/rclone/fs/rc" "github.com/ncw/rclone/fs/rc"
"github.com/pkg/errors" "github.com/pkg/errors"
"github.com/skratchdot/open-golang/open"
) )
// Start the remote control server if configured // Start the remote control server if configured
@ -79,7 +84,10 @@ func (s *Server) Serve() error {
if user != "" || pass != "" { if user != "" || pass != "" {
openURL.User = url.UserPassword(user, pass) openURL.User = url.UserPassword(user, pass)
} }
_ = open.Start(openURL.String()) // Don't open browser if serving in testing environment.
if flag.Lookup("test.v") == nil {
_ = open.Start(openURL.String())
}
} }
return nil return nil
} }
@ -181,15 +189,16 @@ func (s *Server) handlePost(w http.ResponseWriter, r *http.Request, path string)
writeError(path, in, w, err, http.StatusBadRequest) writeError(path, in, w, err, http.StatusBadRequest)
return return
} }
delete(in, "_async") // remove the async parameter after parsing so vfs operations don't get confused delete(in, "_async") // remove the async parameter after parsing so vfs operations don't get confused
fs.Debugf(nil, "rc: %q: with parameters %+v", path, in) fs.Debugf(nil, "rc: %q: with parameters %+v", path, in)
var out rc.Params var out rc.Params
if isAsync { if isAsync {
out, err = rc.StartJob(call.Fn, in) out, err = jobs.StartAsyncJob(call.Fn, in)
} else { } else {
out, err = call.Fn(r.Context(), in) var jobID int64
out, jobID, err = jobs.ExecuteJob(r.Context(), call.Fn, in)
w.Header().Add("x-rclone-jobid", fmt.Sprintf("%d", jobID))
} }
if err != nil { if err != nil {
writeError(path, in, w, err, http.StatusInternalServerError) writeError(path, in, w, err, http.StatusInternalServerError)

View file

@ -611,10 +611,7 @@ func TestRCAsync(t *testing.T) {
ContentType: "application/json", ContentType: "application/json",
Body: `{ "_async":true }`, Body: `{ "_async":true }`,
Status: http.StatusOK, Status: http.StatusOK,
Expected: `{ Contains: regexp.MustCompile(`(?s)\{.*\"jobid\":.*\}`),
"jobid": 1
}
`,
}, { }, {
Name: "bad", Name: "bad",
URL: "rc/noop", URL: "rc/noop",

View file

@ -82,12 +82,12 @@ func newSyncCopyMove(ctx context.Context, fdst, fsrc fs.Fs, deleteMode fs.Delete
dstEmptyDirs: make(map[string]fs.DirEntry), dstEmptyDirs: make(map[string]fs.DirEntry),
srcEmptyDirs: make(map[string]fs.DirEntry), srcEmptyDirs: make(map[string]fs.DirEntry),
noTraverse: fs.Config.NoTraverse, noTraverse: fs.Config.NoTraverse,
toBeChecked: newPipe(accounting.Stats.SetCheckQueue, fs.Config.MaxBacklog), toBeChecked: newPipe(accounting.Stats(ctx).SetCheckQueue, fs.Config.MaxBacklog),
toBeUploaded: newPipe(accounting.Stats.SetTransferQueue, fs.Config.MaxBacklog), toBeUploaded: newPipe(accounting.Stats(ctx).SetTransferQueue, fs.Config.MaxBacklog),
deleteFilesCh: make(chan fs.Object, fs.Config.Checkers), deleteFilesCh: make(chan fs.Object, fs.Config.Checkers),
trackRenames: fs.Config.TrackRenames, trackRenames: fs.Config.TrackRenames,
commonHash: fsrc.Hashes().Overlap(fdst.Hashes()).GetOne(), commonHash: fsrc.Hashes().Overlap(fdst.Hashes()).GetOne(),
toBeRenamed: newPipe(accounting.Stats.SetRenameQueue, fs.Config.MaxBacklog), toBeRenamed: newPipe(accounting.Stats(ctx).SetRenameQueue, fs.Config.MaxBacklog),
trackRenamesCh: make(chan fs.Object, fs.Config.Checkers), trackRenamesCh: make(chan fs.Object, fs.Config.Checkers),
} }
s.ctx, s.cancel = context.WithCancel(ctx) s.ctx, s.cancel = context.WithCancel(ctx)
@ -215,7 +215,7 @@ func (s *syncCopyMove) pairChecker(in *pipe, out *pipe, wg *sync.WaitGroup) {
return return
} }
src := pair.Src src := pair.Src
accounting.Stats.Checking(src.Remote()) accounting.Stats(s.ctx).Checking(src.Remote())
// Check to see if can store this // Check to see if can store this
if src.Storable() { if src.Storable() {
NoNeedTransfer, err := operations.CompareOrCopyDest(s.ctx, s.fdst, pair.Dst, pair.Src, s.compareCopyDest, s.backupDir) NoNeedTransfer, err := operations.CompareOrCopyDest(s.ctx, s.fdst, pair.Dst, pair.Src, s.compareCopyDest, s.backupDir)
@ -256,7 +256,7 @@ func (s *syncCopyMove) pairChecker(in *pipe, out *pipe, wg *sync.WaitGroup) {
} }
} }
} }
accounting.Stats.DoneChecking(src.Remote()) accounting.Stats(s.ctx).DoneChecking(src.Remote())
} }
} }
@ -401,7 +401,7 @@ func (s *syncCopyMove) stopDeleters() {
// checkSrcMap is clear then it assumes that the any source files that // checkSrcMap is clear then it assumes that the any source files that
// have been found have been removed from dstFiles already. // have been found have been removed from dstFiles already.
func (s *syncCopyMove) deleteFiles(checkSrcMap bool) error { func (s *syncCopyMove) deleteFiles(checkSrcMap bool) error {
if accounting.Stats.Errored() && !fs.Config.IgnoreErrors { if accounting.Stats(s.ctx).Errored() && !fs.Config.IgnoreErrors {
fs.Errorf(s.fdst, "%v", fs.ErrorNotDeleting) fs.Errorf(s.fdst, "%v", fs.ErrorNotDeleting)
return fs.ErrorNotDeleting return fs.ErrorNotDeleting
} }
@ -437,7 +437,7 @@ func deleteEmptyDirectories(ctx context.Context, f fs.Fs, entriesMap map[string]
if len(entriesMap) == 0 { if len(entriesMap) == 0 {
return nil return nil
} }
if accounting.Stats.Errored() && !fs.Config.IgnoreErrors { if accounting.Stats(ctx).Errored() && !fs.Config.IgnoreErrors {
fs.Errorf(f, "%v", fs.ErrorNotDeletingDirs) fs.Errorf(f, "%v", fs.ErrorNotDeletingDirs)
return fs.ErrorNotDeletingDirs return fs.ErrorNotDeletingDirs
} }
@ -497,8 +497,8 @@ func copyEmptyDirectories(ctx context.Context, f fs.Fs, entries map[string]fs.Di
} }
} }
if accounting.Stats.Errored() { if accounting.Stats(ctx).Errored() {
fs.Debugf(f, "failed to copy %d directories", accounting.Stats.GetErrors()) fs.Debugf(f, "failed to copy %d directories", accounting.Stats(ctx).GetErrors())
} }
if okCount > 0 { if okCount > 0 {
@ -587,12 +587,12 @@ func (s *syncCopyMove) makeRenameMap() {
for obj := range in { for obj := range in {
// only create hash for dst fs.Object if its size could match // only create hash for dst fs.Object if its size could match
if _, found := possibleSizes[obj.Size()]; found { if _, found := possibleSizes[obj.Size()]; found {
accounting.Stats.Checking(obj.Remote()) accounting.Stats(s.ctx).Checking(obj.Remote())
hash := s.renameHash(obj) hash := s.renameHash(obj)
if hash != "" { if hash != "" {
s.pushRenameMap(hash, obj) s.pushRenameMap(hash, obj)
} }
accounting.Stats.DoneChecking(obj.Remote()) accounting.Stats(s.ctx).DoneChecking(obj.Remote())
} }
} }
}() }()

View file

@ -105,7 +105,7 @@ func TestSyncNoTraverse(t *testing.T) {
file1 := r.WriteFile("sub dir/hello world", "hello world", t1) file1 := r.WriteFile("sub dir/hello world", "hello world", t1)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
@ -306,24 +306,24 @@ func TestSyncBasedOnCheckSum(t *testing.T) {
file1 := r.WriteFile("check sum", "-", t1) file1 := r.WriteFile("check sum", "-", t1)
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// We should have transferred exactly one file. // We should have transferred exactly one file.
assert.Equal(t, int64(1), accounting.Stats.GetTransfers()) assert.Equal(t, int64(1), accounting.GlobalStats().GetTransfers())
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
// Change last modified date only // Change last modified date only
file2 := r.WriteFile("check sum", "-", t2) file2 := r.WriteFile("check sum", "-", t2)
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), r.Fremote, r.Flocal, false) err = Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// We should have transferred no files // We should have transferred no files
assert.Equal(t, int64(0), accounting.Stats.GetTransfers()) assert.Equal(t, int64(0), accounting.GlobalStats().GetTransfers())
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
} }
@ -340,24 +340,24 @@ func TestSyncSizeOnly(t *testing.T) {
file1 := r.WriteFile("sizeonly", "potato", t1) file1 := r.WriteFile("sizeonly", "potato", t1)
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// We should have transferred exactly one file. // We should have transferred exactly one file.
assert.Equal(t, int64(1), accounting.Stats.GetTransfers()) assert.Equal(t, int64(1), accounting.GlobalStats().GetTransfers())
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
// Update mtime, md5sum but not length of file // Update mtime, md5sum but not length of file
file2 := r.WriteFile("sizeonly", "POTATO", t2) file2 := r.WriteFile("sizeonly", "POTATO", t2)
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), r.Fremote, r.Flocal, false) err = Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// We should have transferred no files // We should have transferred no files
assert.Equal(t, int64(0), accounting.Stats.GetTransfers()) assert.Equal(t, int64(0), accounting.GlobalStats().GetTransfers())
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
} }
@ -374,24 +374,24 @@ func TestSyncIgnoreSize(t *testing.T) {
file1 := r.WriteFile("ignore-size", "contents", t1) file1 := r.WriteFile("ignore-size", "contents", t1)
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// We should have transferred exactly one file. // We should have transferred exactly one file.
assert.Equal(t, int64(1), accounting.Stats.GetTransfers()) assert.Equal(t, int64(1), accounting.GlobalStats().GetTransfers())
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
// Update size but not date of file // Update size but not date of file
file2 := r.WriteFile("ignore-size", "longer contents but same date", t1) file2 := r.WriteFile("ignore-size", "longer contents but same date", t1)
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), r.Fremote, r.Flocal, false) err = Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// We should have transferred no files // We should have transferred no files
assert.Equal(t, int64(0), accounting.Stats.GetTransfers()) assert.Equal(t, int64(0), accounting.GlobalStats().GetTransfers())
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
} }
@ -402,24 +402,24 @@ func TestSyncIgnoreTimes(t *testing.T) {
file1 := r.WriteBoth(context.Background(), "existing", "potato", t1) file1 := r.WriteBoth(context.Background(), "existing", "potato", t1)
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// We should have transferred exactly 0 files because the // We should have transferred exactly 0 files because the
// files were identical. // files were identical.
assert.Equal(t, int64(0), accounting.Stats.GetTransfers()) assert.Equal(t, int64(0), accounting.GlobalStats().GetTransfers())
fs.Config.IgnoreTimes = true fs.Config.IgnoreTimes = true
defer func() { fs.Config.IgnoreTimes = false }() defer func() { fs.Config.IgnoreTimes = false }()
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), r.Fremote, r.Flocal, false) err = Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// We should have transferred exactly one file even though the // We should have transferred exactly one file even though the
// files were identical. // files were identical.
assert.Equal(t, int64(1), accounting.Stats.GetTransfers()) assert.Equal(t, int64(1), accounting.GlobalStats().GetTransfers())
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
@ -433,7 +433,7 @@ func TestSyncIgnoreExisting(t *testing.T) {
fs.Config.IgnoreExisting = true fs.Config.IgnoreExisting = true
defer func() { fs.Config.IgnoreExisting = false }() defer func() { fs.Config.IgnoreExisting = false }()
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
@ -441,7 +441,7 @@ func TestSyncIgnoreExisting(t *testing.T) {
// Change everything // Change everything
r.WriteFile("existing", "newpotatoes", t2) r.WriteFile("existing", "newpotatoes", t2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), r.Fremote, r.Flocal, false) err = Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// Items should not change // Items should not change
@ -488,7 +488,7 @@ func TestSyncIgnoreErrors(t *testing.T) {
fs.GetModifyWindow(r.Fremote), fs.GetModifyWindow(r.Fremote),
) )
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
fs.CountError(errors.New("boom")) fs.CountError(errors.New("boom"))
assert.NoError(t, Sync(context.Background(), r.Fremote, r.Flocal, false)) assert.NoError(t, Sync(context.Background(), r.Fremote, r.Flocal, false))
@ -532,7 +532,7 @@ func TestSyncAfterChangingModtimeOnly(t *testing.T) {
fs.Config.DryRun = true fs.Config.DryRun = true
defer func() { fs.Config.DryRun = false }() defer func() { fs.Config.DryRun = false }()
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
@ -541,7 +541,7 @@ func TestSyncAfterChangingModtimeOnly(t *testing.T) {
fs.Config.DryRun = false fs.Config.DryRun = false
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), r.Fremote, r.Flocal, false) err = Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
@ -569,7 +569,7 @@ func TestSyncAfterChangingModtimeOnlyWithNoUpdateModTime(t *testing.T) {
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
fstest.CheckItems(t, r.Fremote, file2) fstest.CheckItems(t, r.Fremote, file2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
@ -590,7 +590,7 @@ func TestSyncDoesntUpdateModtime(t *testing.T) {
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
fstest.CheckItems(t, r.Fremote, file2) fstest.CheckItems(t, r.Fremote, file2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
@ -598,7 +598,7 @@ func TestSyncDoesntUpdateModtime(t *testing.T) {
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
// We should have transferred exactly one file, not set the mod time // We should have transferred exactly one file, not set the mod time
assert.Equal(t, int64(1), accounting.Stats.GetTransfers()) assert.Equal(t, int64(1), accounting.GlobalStats().GetTransfers())
} }
func TestSyncAfterAddingAFile(t *testing.T) { func TestSyncAfterAddingAFile(t *testing.T) {
@ -610,7 +610,7 @@ func TestSyncAfterAddingAFile(t *testing.T) {
fstest.CheckItems(t, r.Flocal, file1, file2) fstest.CheckItems(t, r.Flocal, file1, file2)
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Flocal, file1, file2) fstest.CheckItems(t, r.Flocal, file1, file2)
@ -625,7 +625,7 @@ func TestSyncAfterChangingFilesSizeOnly(t *testing.T) {
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
@ -648,7 +648,7 @@ func TestSyncAfterChangingContentsOnly(t *testing.T) {
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
@ -664,7 +664,7 @@ func TestSyncAfterRemovingAFileAndAddingAFileDryRun(t *testing.T) {
file3 := r.WriteBoth(context.Background(), "empty space", "-", t2) file3 := r.WriteBoth(context.Background(), "empty space", "-", t2)
fs.Config.DryRun = true fs.Config.DryRun = true
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
fs.Config.DryRun = false fs.Config.DryRun = false
require.NoError(t, err) require.NoError(t, err)
@ -683,7 +683,7 @@ func TestSyncAfterRemovingAFileAndAddingAFile(t *testing.T) {
fstest.CheckItems(t, r.Fremote, file2, file3) fstest.CheckItems(t, r.Fremote, file2, file3)
fstest.CheckItems(t, r.Flocal, file1, file3) fstest.CheckItems(t, r.Flocal, file1, file3)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Flocal, file1, file3) fstest.CheckItems(t, r.Flocal, file1, file3)
@ -729,7 +729,7 @@ func TestSyncAfterRemovingAFileAndAddingAFileSubDir(t *testing.T) {
fs.GetModifyWindow(r.Fremote), fs.GetModifyWindow(r.Fremote),
) )
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
@ -798,7 +798,7 @@ func TestSyncAfterRemovingAFileAndAddingAFileSubDirWithErrors(t *testing.T) {
fs.GetModifyWindow(r.Fremote), fs.GetModifyWindow(r.Fremote),
) )
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
fs.CountError(errors.New("boom")) fs.CountError(errors.New("boom"))
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
assert.Equal(t, fs.ErrorNotDeleting, err) assert.Equal(t, fs.ErrorNotDeleting, err)
@ -876,7 +876,7 @@ func TestCopyDeleteBefore(t *testing.T) {
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := CopyDir(context.Background(), r.Fremote, r.Flocal, false) err := CopyDir(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
@ -899,14 +899,14 @@ func TestSyncWithExclude(t *testing.T) {
filter.Active.Opt.MaxSize = -1 filter.Active.Opt.MaxSize = -1
}() }()
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Fremote, file2, file1) fstest.CheckItems(t, r.Fremote, file2, file1)
// Now sync the other way round and check enormous doesn't get // Now sync the other way round and check enormous doesn't get
// deleted as it is excluded from the sync // deleted as it is excluded from the sync
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), r.Flocal, r.Fremote, false) err = Sync(context.Background(), r.Flocal, r.Fremote, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Flocal, file2, file1, file3) fstest.CheckItems(t, r.Flocal, file2, file1, file3)
@ -929,14 +929,14 @@ func TestSyncWithExcludeAndDeleteExcluded(t *testing.T) {
filter.Active.Opt.DeleteExcluded = false filter.Active.Opt.DeleteExcluded = false
}() }()
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Fremote, file2) fstest.CheckItems(t, r.Fremote, file2)
// Check sync the other way round to make sure enormous gets // Check sync the other way round to make sure enormous gets
// deleted even though it is excluded // deleted even though it is excluded
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), r.Flocal, r.Fremote, false) err = Sync(context.Background(), r.Flocal, r.Fremote, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
@ -971,7 +971,7 @@ func TestSyncWithUpdateOlder(t *testing.T) {
fs.Config.ModifyWindow = oldModifyWindow fs.Config.ModifyWindow = oldModifyWindow
}() }()
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Fremote, oneO, twoF, threeO, fourF, fiveF) fstest.CheckItems(t, r.Fremote, oneO, twoF, threeO, fourF, fiveF)
@ -995,7 +995,7 @@ func TestSyncWithTrackRenames(t *testing.T) {
f1 := r.WriteFile("potato", "Potato Content", t1) f1 := r.WriteFile("potato", "Potato Content", t1)
f2 := r.WriteFile("yam", "Yam Content", t2) f2 := r.WriteFile("yam", "Yam Content", t2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
require.NoError(t, Sync(context.Background(), r.Fremote, r.Flocal, false)) require.NoError(t, Sync(context.Background(), r.Fremote, r.Flocal, false))
fstest.CheckItems(t, r.Fremote, f1, f2) fstest.CheckItems(t, r.Fremote, f1, f2)
@ -1004,7 +1004,7 @@ func TestSyncWithTrackRenames(t *testing.T) {
// Now rename locally. // Now rename locally.
f2 = r.RenameFile(f2, "yaml") f2 = r.RenameFile(f2, "yaml")
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
require.NoError(t, Sync(context.Background(), r.Fremote, r.Flocal, false)) require.NoError(t, Sync(context.Background(), r.Fremote, r.Flocal, false))
fstest.CheckItems(t, r.Fremote, f1, f2) fstest.CheckItems(t, r.Fremote, f1, f2)
@ -1012,15 +1012,15 @@ func TestSyncWithTrackRenames(t *testing.T) {
if canTrackRenames { if canTrackRenames {
if r.Fremote.Features().Move == nil || r.Fremote.Name() == "TestUnion" { // union remote can Move but returns CantMove error if r.Fremote.Features().Move == nil || r.Fremote.Name() == "TestUnion" { // union remote can Move but returns CantMove error
// If no server side Move, we are falling back to Copy + Delete // If no server side Move, we are falling back to Copy + Delete
assert.Equal(t, int64(1), accounting.Stats.GetTransfers()) // 1 copy assert.Equal(t, int64(1), accounting.GlobalStats().GetTransfers()) // 1 copy
assert.Equal(t, int64(4), accounting.Stats.GetChecks()) // 2 file checks + 1 move + 1 delete assert.Equal(t, int64(4), accounting.GlobalStats().GetChecks()) // 2 file checks + 1 move + 1 delete
} else { } else {
assert.Equal(t, int64(0), accounting.Stats.GetTransfers()) // 0 copy assert.Equal(t, int64(0), accounting.GlobalStats().GetTransfers()) // 0 copy
assert.Equal(t, int64(3), accounting.Stats.GetChecks()) // 2 file checks + 1 move assert.Equal(t, int64(3), accounting.GlobalStats().GetChecks()) // 2 file checks + 1 move
} }
} else { } else {
assert.Equal(t, int64(2), accounting.Stats.GetChecks()) // 2 file checks assert.Equal(t, int64(2), accounting.GlobalStats().GetChecks()) // 2 file checks
assert.Equal(t, int64(1), accounting.Stats.GetTransfers()) // 0 copy assert.Equal(t, int64(1), accounting.GlobalStats().GetTransfers()) // 0 copy
} }
} }
@ -1049,7 +1049,7 @@ func testServerSideMove(t *testing.T, r *fstest.Run, withFilter, testDeleteEmpty
fstest.CheckItems(t, FremoteMove, file2, file3) fstest.CheckItems(t, FremoteMove, file2, file3)
// Do server side move // Do server side move
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = MoveDir(context.Background(), FremoteMove, r.Fremote, testDeleteEmptyDirs, false) err = MoveDir(context.Background(), FremoteMove, r.Fremote, testDeleteEmptyDirs, false)
require.NoError(t, err) require.NoError(t, err)
@ -1076,7 +1076,7 @@ func testServerSideMove(t *testing.T, r *fstest.Run, withFilter, testDeleteEmpty
} }
// Move it back to a new empty remote, dst does not exist this time // Move it back to a new empty remote, dst does not exist this time
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = MoveDir(context.Background(), FremoteMove2, FremoteMove, testDeleteEmptyDirs, false) err = MoveDir(context.Background(), FremoteMove2, FremoteMove, testDeleteEmptyDirs, false)
require.NoError(t, err) require.NoError(t, err)
@ -1439,7 +1439,7 @@ func testSyncBackupDir(t *testing.T, suffix string, suffixKeepExtension bool) {
fdst, err := fs.NewFs(r.FremoteName + "/dst") fdst, err := fs.NewFs(r.FremoteName + "/dst")
require.NoError(t, err) require.NoError(t, err)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), fdst, r.Flocal, false) err = Sync(context.Background(), fdst, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
@ -1464,7 +1464,7 @@ func testSyncBackupDir(t *testing.T, suffix string, suffixKeepExtension bool) {
// This should delete three and overwrite one again, checking // This should delete three and overwrite one again, checking
// the files got overwritten correctly in backup-dir // the files got overwritten correctly in backup-dir
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), fdst, r.Flocal, false) err = Sync(context.Background(), fdst, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
@ -1518,7 +1518,7 @@ func testSyncSuffix(t *testing.T, suffix string, suffixKeepExtension bool) {
fdst, err := fs.NewFs(r.FremoteName + "/dst") fdst, err := fs.NewFs(r.FremoteName + "/dst")
require.NoError(t, err) require.NoError(t, err)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = operations.CopyFile(context.Background(), fdst, r.Flocal, "one", "one") err = operations.CopyFile(context.Background(), fdst, r.Flocal, "one", "one")
require.NoError(t, err) require.NoError(t, err)
err = operations.CopyFile(context.Background(), fdst, r.Flocal, "two", "two") err = operations.CopyFile(context.Background(), fdst, r.Flocal, "two", "two")
@ -1548,7 +1548,7 @@ func testSyncSuffix(t *testing.T, suffix string, suffixKeepExtension bool) {
// This should delete three and overwrite one again, checking // This should delete three and overwrite one again, checking
// the files got overwritten correctly in backup-dir // the files got overwritten correctly in backup-dir
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = operations.CopyFile(context.Background(), fdst, r.Flocal, "one", "one") err = operations.CopyFile(context.Background(), fdst, r.Flocal, "one", "one")
require.NoError(t, err) require.NoError(t, err)
err = operations.CopyFile(context.Background(), fdst, r.Flocal, "two", "two") err = operations.CopyFile(context.Background(), fdst, r.Flocal, "two", "two")
@ -1594,13 +1594,13 @@ func TestSyncUTFNorm(t *testing.T) {
file2 := r.WriteObject(context.Background(), Encoding2, "This is a old test", t2) file2 := r.WriteObject(context.Background(), Encoding2, "This is a old test", t2)
fstest.CheckItems(t, r.Fremote, file2) fstest.CheckItems(t, r.Fremote, file2)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
// We should have transferred exactly one file, but kept the // We should have transferred exactly one file, but kept the
// normalized state of the file. // normalized state of the file.
assert.Equal(t, int64(1), accounting.Stats.GetTransfers()) assert.Equal(t, int64(1), accounting.GlobalStats().GetTransfers())
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
file1.Path = file2.Path file1.Path = file2.Path
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
@ -1620,7 +1620,7 @@ func TestSyncImmutable(t *testing.T) {
fstest.CheckItems(t, r.Fremote) fstest.CheckItems(t, r.Fremote)
// Should succeed // Should succeed
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
@ -1632,7 +1632,7 @@ func TestSyncImmutable(t *testing.T) {
fstest.CheckItems(t, r.Fremote, file1) fstest.CheckItems(t, r.Fremote, file1)
// Should fail with ErrorImmutableModified and not modify local or remote files // Should fail with ErrorImmutableModified and not modify local or remote files
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err = Sync(context.Background(), r.Fremote, r.Flocal, false) err = Sync(context.Background(), r.Fremote, r.Flocal, false)
assert.EqualError(t, err, fs.ErrorImmutableModified.Error()) assert.EqualError(t, err, fs.ErrorImmutableModified.Error())
fstest.CheckItems(t, r.Flocal, file2) fstest.CheckItems(t, r.Flocal, file2)
@ -1659,7 +1659,7 @@ func TestSyncIgnoreCase(t *testing.T) {
fstest.CheckItems(t, r.Fremote, file2) fstest.CheckItems(t, r.Fremote, file2)
// Should not copy files that are differently-cased but otherwise identical // Should not copy files that are differently-cased but otherwise identical
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
require.NoError(t, err) require.NoError(t, err)
fstest.CheckItems(t, r.Flocal, file1) fstest.CheckItems(t, r.Flocal, file1)
@ -1694,7 +1694,7 @@ func TestAbort(t *testing.T) {
fstest.CheckItems(t, r.Flocal, file1, file2, file3) fstest.CheckItems(t, r.Flocal, file1, file2, file3)
fstest.CheckItems(t, r.Fremote) fstest.CheckItems(t, r.Fremote)
accounting.Stats.ResetCounters() accounting.GlobalStats().ResetCounters()
err := Sync(context.Background(), r.Fremote, r.Flocal, false) err := Sync(context.Background(), r.Fremote, r.Flocal, false)
assert.Equal(t, accounting.ErrorMaxTransferLimitReached, err) assert.Equal(t, accounting.ErrorMaxTransferLimitReached, err)

View file

@ -274,7 +274,8 @@ func CheckListingWithPrecision(t *testing.T, f fs.Fs, items []Item, expectedDirs
expectedDirs = filterEmptyDirs(t, items, expectedDirs) expectedDirs = filterEmptyDirs(t, items, expectedDirs)
} }
is := NewItems(items) is := NewItems(items)
oldErrors := accounting.Stats.GetErrors() ctx := context.Background()
oldErrors := accounting.Stats(ctx).GetErrors()
var objs []fs.Object var objs []fs.Object
var dirs []fs.Directory var dirs []fs.Directory
var err error var err error
@ -283,7 +284,6 @@ func CheckListingWithPrecision(t *testing.T, f fs.Fs, items []Item, expectedDirs
wantListing1, wantListing2 := makeListingFromItems(items) wantListing1, wantListing2 := makeListingFromItems(items)
gotListing := "<unset>" gotListing := "<unset>"
listingOK := false listingOK := false
ctx := context.Background()
for i := 1; i <= retries; i++ { for i := 1; i <= retries; i++ {
objs, dirs, err = walk.GetAll(ctx, f, "", true, -1) objs, dirs, err = walk.GetAll(ctx, f, "", true, -1)
if err != nil && err != fs.ErrorDirNotFound { if err != nil && err != fs.ErrorDirNotFound {
@ -317,8 +317,8 @@ func CheckListingWithPrecision(t *testing.T, f fs.Fs, items []Item, expectedDirs
} }
is.Done(t) is.Done(t)
// Don't notice an error when listing an empty directory // Don't notice an error when listing an empty directory
if len(items) == 0 && oldErrors == 0 && accounting.Stats.GetErrors() == 1 { if len(items) == 0 && oldErrors == 0 && accounting.Stats(ctx).GetErrors() == 1 {
accounting.Stats.ResetErrors() accounting.Stats(ctx).ResetErrors()
} }
// Check the directories // Check the directories
if expectedDirs != nil { if expectedDirs != nil {

View file

@ -71,7 +71,7 @@ func (fh *ReadFileHandle) openPending() (err error) {
if err != nil { if err != nil {
return err return err
} }
tr := accounting.Stats.NewTransfer(o) tr := accounting.GlobalStats().NewTransfer(o)
fh.done = tr.Done fh.done = tr.Done
fh.r = tr.Account(r).WithBuffer() // account the transfer fh.r = tr.Account(r).WithBuffer() // account the transfer
fh.opened = true fh.opened = true

View file

@ -10,7 +10,6 @@ import (
"sync" "sync"
"github.com/ncw/rclone/fs" "github.com/ncw/rclone/fs"
"github.com/ncw/rclone/fs/accounting"
"github.com/ncw/rclone/fs/log" "github.com/ncw/rclone/fs/log"
"github.com/ncw/rclone/fs/operations" "github.com/ncw/rclone/fs/operations"
"github.com/ncw/rclone/lib/file" "github.com/ncw/rclone/lib/file"
@ -87,10 +86,6 @@ func newRWFileHandle(d *Dir, f *File, remote string, flags int) (fh *RWFileHandl
// copy an object to or from the remote while accounting for it // copy an object to or from the remote while accounting for it
func copyObj(f fs.Fs, dst fs.Object, remote string, src fs.Object) (newDst fs.Object, err error) { func copyObj(f fs.Fs, dst fs.Object, remote string, src fs.Object) (newDst fs.Object, err error) {
if operations.NeedTransfer(context.TODO(), dst, src) { if operations.NeedTransfer(context.TODO(), dst, src) {
tr := accounting.Stats.NewTransfer(src)
defer func() {
tr.Done(err)
}()
newDst, err = operations.Copy(context.TODO(), f, dst, remote, src) newDst, err = operations.Copy(context.TODO(), f, dst, remote, src)
} else { } else {
newDst = dst newDst = dst