accounting: add context.Context #3257 #4685

This commit is contained in:
Nick Craig-Wood 2020-11-05 16:59:59 +00:00
parent e3fe31f7cb
commit 1fb6ad700f
29 changed files with 138 additions and 109 deletions

View file

@ -19,7 +19,7 @@ var (
) )
func prepare(t *testing.T, root string) { func prepare(t *testing.T, root string) {
config.LoadConfig() config.LoadConfig(context.Background())
// Configure the remote // Configure the remote
config.FileSet(remoteName, "type", "alias") config.FileSet(remoteName, "type", "alias")

View file

@ -1182,7 +1182,7 @@ func (f *Fs) purge(ctx context.Context, dir string, oldOnly bool) error {
tr := accounting.Stats(ctx).NewCheckingTransfer(oi) tr := accounting.Stats(ctx).NewCheckingTransfer(oi)
err = f.deleteByID(ctx, object.ID, object.Name) err = f.deleteByID(ctx, object.ID, object.Name)
checkErr(err) checkErr(err)
tr.Done(err) tr.Done(ctx, err)
} }
}() }()
} }
@ -1210,7 +1210,7 @@ func (f *Fs) purge(ctx context.Context, dir string, oldOnly bool) error {
toBeDeleted <- object toBeDeleted <- object
} }
last = remote last = remote
tr.Done(nil) tr.Done(ctx, nil)
} }
return nil return nil
})) }))

View file

@ -47,7 +47,7 @@ func prepareServer(t *testing.T) (configmap.Simple, func()) {
ts := httptest.NewServer(handler) ts := httptest.NewServer(handler)
// Configure the remote // Configure the remote
config.LoadConfig() config.LoadConfig(context.Background())
// fs.Config.LogLevel = fs.LogLevelDebug // fs.Config.LogLevel = fs.LogLevelDebug
// fs.Config.DumpHeaders = true // fs.Config.DumpHeaders = true
// fs.Config.DumpBodies = true // fs.Config.DumpBodies = true

View file

@ -41,7 +41,7 @@ func startServer(t *testing.T, f fs.Fs) {
} }
func TestInit(t *testing.T) { func TestInit(t *testing.T) {
config.LoadConfig() config.LoadConfig(context.Background())
f, err := fs.NewFs(context.Background(), "testdata/files") f, err := fs.NewFs(context.Background(), "testdata/files")
l, _ := f.List(context.Background(), "") l, _ := f.List(context.Background(), "")

View file

@ -294,7 +294,7 @@ func (d *Driver) ListDir(path string, callback func(ftp.FileInfo) error) (err er
// Account the transfer // Account the transfer
tr := accounting.GlobalStats().NewTransferRemoteSize(path, node.Size()) tr := accounting.GlobalStats().NewTransferRemoteSize(path, node.Size())
defer func() { defer func() {
tr.Done(err) tr.Done(d.s.ctx, err)
}() }()
for _, file := range dirEntries { for _, file := range dirEntries {
@ -392,7 +392,7 @@ func (d *Driver) GetFile(path string, offset int64) (size int64, fr io.ReadClose
// Account the transfer // Account the transfer
tr := accounting.GlobalStats().NewTransferRemoteSize(path, node.Size()) tr := accounting.GlobalStats().NewTransferRemoteSize(path, node.Size())
defer tr.Done(nil) defer tr.Done(d.s.ctx, nil)
return node.Size(), handle, nil return node.Size(), handle, nil
} }

View file

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

View file

@ -60,7 +60,7 @@ var (
func TestInit(t *testing.T) { func TestInit(t *testing.T) {
// Configure the remote // Configure the remote
config.LoadConfig() config.LoadConfig(context.Background())
// fs.Config.LogLevel = fs.LogLevelDebug // fs.Config.LogLevel = fs.LogLevelDebug
// fs.Config.DumpHeaders = true // fs.Config.DumpHeaders = true
// fs.Config.DumpBodies = true // fs.Config.DumpBodies = true

View file

@ -225,7 +225,7 @@ const (
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(r.Context()).NewTransferRemoteSize(d.DirRemote, -1) tr := accounting.Stats(r.Context()).NewTransferRemoteSize(d.DirRemote, -1)
defer tr.Done(nil) defer tr.Done(r.Context(), nil)
fs.Infof(d.DirRemote, "%s: Serving directory", r.RemoteAddr) fs.Infof(d.DirRemote, "%s: Serving directory", r.RemoteAddr)

View file

@ -77,7 +77,7 @@ func Object(w http.ResponseWriter, r *http.Request, o fs.Object) {
} }
tr := accounting.Stats(r.Context()).NewTransfer(o) tr := accounting.Stats(r.Context()).NewTransfer(o)
defer func() { defer func() {
tr.Done(err) tr.Done(r.Context(), err)
}() }()
in := tr.Account(r.Context(), file) // account the transfer (no buffering) in := tr.Account(r.Context(), file) // account the transfer (no buffering)

View file

@ -28,8 +28,9 @@ var (
) )
func TestNewAccountSizeName(t *testing.T) { func TestNewAccountSizeName(t *testing.T) {
ctx := context.Background()
in := ioutil.NopCloser(bytes.NewBuffer([]byte{1})) in := ioutil.NopCloser(bytes.NewBuffer([]byte{1}))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(context.Background(), stats, in, 1, "test") acc := newAccountSizeName(context.Background(), stats, in, 1, "test")
assert.Equal(t, in, acc.in) assert.Equal(t, in, acc.in)
assert.Equal(t, acc, stats.inProgress.get("test")) assert.Equal(t, acc, stats.inProgress.get("test"))
@ -42,10 +43,11 @@ func TestNewAccountSizeName(t *testing.T) {
} }
func TestAccountWithBuffer(t *testing.T) { func TestAccountWithBuffer(t *testing.T) {
ctx := context.Background()
in := ioutil.NopCloser(bytes.NewBuffer([]byte{1})) in := ioutil.NopCloser(bytes.NewBuffer([]byte{1}))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(context.Background(), stats, in, -1, "test") acc := newAccountSizeName(ctx, stats, in, -1, "test")
assert.False(t, acc.HasBuffer()) assert.False(t, acc.HasBuffer())
acc.WithBuffer() acc.WithBuffer()
assert.True(t, acc.HasBuffer()) assert.True(t, acc.HasBuffer())
@ -54,7 +56,7 @@ func TestAccountWithBuffer(t *testing.T) {
require.True(t, ok) require.True(t, ok)
assert.NoError(t, acc.Close()) assert.NoError(t, acc.Close())
acc = newAccountSizeName(context.Background(), stats, in, 1, "test") acc = newAccountSizeName(ctx, stats, in, 1, "test")
acc.WithBuffer() acc.WithBuffer()
// should not have a buffer for a small size // should not have a buffer for a small size
_, ok = acc.in.(*asyncreader.AsyncReader) _, ok = acc.in.(*asyncreader.AsyncReader)
@ -63,11 +65,12 @@ func TestAccountWithBuffer(t *testing.T) {
} }
func TestAccountGetUpdateReader(t *testing.T) { func TestAccountGetUpdateReader(t *testing.T) {
ctx := context.Background()
test := func(doClose bool) func(t *testing.T) { test := func(doClose bool) func(t *testing.T) {
return func(t *testing.T) { return func(t *testing.T) {
in := ioutil.NopCloser(bytes.NewBuffer([]byte{1})) in := ioutil.NopCloser(bytes.NewBuffer([]byte{1}))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(context.Background(), stats, in, 1, "test") acc := newAccountSizeName(ctx, stats, in, 1, "test")
assert.Equal(t, in, acc.GetReader()) assert.Equal(t, in, acc.GetReader())
assert.Equal(t, acc, stats.inProgress.get("test")) assert.Equal(t, acc, stats.inProgress.get("test"))
@ -78,7 +81,7 @@ func TestAccountGetUpdateReader(t *testing.T) {
} }
in2 := ioutil.NopCloser(bytes.NewBuffer([]byte{1})) in2 := ioutil.NopCloser(bytes.NewBuffer([]byte{1}))
acc.UpdateReader(context.Background(), in2) acc.UpdateReader(ctx, in2)
assert.Equal(t, in2, acc.GetReader()) assert.Equal(t, in2, acc.GetReader())
assert.Equal(t, acc, stats.inProgress.get("test")) assert.Equal(t, acc, stats.inProgress.get("test"))
@ -91,9 +94,10 @@ func TestAccountGetUpdateReader(t *testing.T) {
} }
func TestAccountRead(t *testing.T) { func TestAccountRead(t *testing.T) {
ctx := context.Background()
in := ioutil.NopCloser(bytes.NewBuffer([]byte{1, 2, 3})) in := ioutil.NopCloser(bytes.NewBuffer([]byte{1, 2, 3}))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(context.Background(), stats, in, 1, "test") acc := newAccountSizeName(ctx, stats, in, 1, "test")
assert.True(t, acc.values.start.IsZero()) assert.True(t, acc.values.start.IsZero())
acc.values.mu.Lock() acc.values.mu.Lock()
@ -128,13 +132,14 @@ func TestAccountRead(t *testing.T) {
} }
func testAccountWriteTo(t *testing.T, withBuffer bool) { func testAccountWriteTo(t *testing.T, withBuffer bool) {
ctx := context.Background()
buf := make([]byte, 2*asyncreader.BufferSize+1) buf := make([]byte, 2*asyncreader.BufferSize+1)
for i := range buf { for i := range buf {
buf[i] = byte(i % 251) buf[i] = byte(i % 251)
} }
in := ioutil.NopCloser(bytes.NewBuffer(buf)) in := ioutil.NopCloser(bytes.NewBuffer(buf))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(context.Background(), stats, in, int64(len(buf)), "test") acc := newAccountSizeName(ctx, stats, in, int64(len(buf)), "test")
if withBuffer { if withBuffer {
acc = acc.WithBuffer() acc = acc.WithBuffer()
} }
@ -172,9 +177,10 @@ func TestAccountWriteToWithBuffer(t *testing.T) {
} }
func TestAccountString(t *testing.T) { func TestAccountString(t *testing.T) {
ctx := context.Background()
in := ioutil.NopCloser(bytes.NewBuffer([]byte{1, 2, 3})) in := ioutil.NopCloser(bytes.NewBuffer([]byte{1, 2, 3}))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(context.Background(), stats, in, 3, "test") acc := newAccountSizeName(ctx, stats, in, 3, "test")
// FIXME not an exhaustive test! // FIXME not an exhaustive test!
@ -192,9 +198,10 @@ func TestAccountString(t *testing.T) {
// Test the Accounter interface methods on Account and accountStream // Test the Accounter interface methods on Account and accountStream
func TestAccountAccounter(t *testing.T) { func TestAccountAccounter(t *testing.T) {
ctx := context.Background()
in := ioutil.NopCloser(bytes.NewBuffer([]byte{1, 2, 3})) in := ioutil.NopCloser(bytes.NewBuffer([]byte{1, 2, 3}))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(context.Background(), stats, in, 3, "test") acc := newAccountSizeName(ctx, stats, in, 3, "test")
assert.True(t, in == acc.OldStream()) assert.True(t, in == acc.OldStream())
@ -250,6 +257,7 @@ func TestAccountAccounter(t *testing.T) {
} }
func TestAccountMaxTransfer(t *testing.T) { func TestAccountMaxTransfer(t *testing.T) {
ctx := context.Background()
old := fs.Config.MaxTransfer old := fs.Config.MaxTransfer
oldMode := fs.Config.CutoffMode oldMode := fs.Config.CutoffMode
@ -260,8 +268,8 @@ func TestAccountMaxTransfer(t *testing.T) {
}() }()
in := ioutil.NopCloser(bytes.NewBuffer(make([]byte, 100))) in := ioutil.NopCloser(bytes.NewBuffer(make([]byte, 100)))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(context.Background(), stats, in, 1, "test") acc := newAccountSizeName(ctx, stats, in, 1, "test")
var b = make([]byte, 10) var b = make([]byte, 10)
@ -277,8 +285,8 @@ func TestAccountMaxTransfer(t *testing.T) {
assert.True(t, fserrors.IsFatalError(err)) assert.True(t, fserrors.IsFatalError(err))
fs.Config.CutoffMode = fs.CutoffModeSoft fs.Config.CutoffMode = fs.CutoffModeSoft
stats = NewStats() stats = NewStats(ctx)
acc = newAccountSizeName(context.Background(), stats, in, 1, "test") acc = newAccountSizeName(ctx, stats, in, 1, "test")
n, err = acc.Read(b) n, err = acc.Read(b)
assert.Equal(t, 10, n) assert.Equal(t, 10, n)
@ -292,6 +300,7 @@ func TestAccountMaxTransfer(t *testing.T) {
} }
func TestAccountMaxTransferWriteTo(t *testing.T) { func TestAccountMaxTransferWriteTo(t *testing.T) {
ctx := context.Background()
old := fs.Config.MaxTransfer old := fs.Config.MaxTransfer
oldMode := fs.Config.CutoffMode oldMode := fs.Config.CutoffMode
@ -302,8 +311,8 @@ func TestAccountMaxTransferWriteTo(t *testing.T) {
}() }()
in := ioutil.NopCloser(readers.NewPatternReader(1024)) in := ioutil.NopCloser(readers.NewPatternReader(1024))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(context.Background(), stats, in, 1, "test") acc := newAccountSizeName(ctx, stats, in, 1, "test")
var b bytes.Buffer var b bytes.Buffer
@ -313,9 +322,10 @@ func TestAccountMaxTransferWriteTo(t *testing.T) {
} }
func TestAccountReadCtx(t *testing.T) { func TestAccountReadCtx(t *testing.T) {
ctx, cancel := context.WithCancel(context.Background()) ctx := context.Background()
ctx, cancel := context.WithCancel(ctx)
in := ioutil.NopCloser(bytes.NewBuffer(make([]byte, 100))) in := ioutil.NopCloser(bytes.NewBuffer(make([]byte, 100)))
stats := NewStats() stats := NewStats(ctx)
acc := newAccountSizeName(ctx, stats, in, 1, "test") acc := newAccountSizeName(ctx, stats, in, 1, "test")
var b = make([]byte, 10) var b = make([]byte, 10)

View file

@ -1,6 +1,7 @@
package accounting package accounting
import ( import (
"context"
"sync" "sync"
"github.com/rclone/rclone/fs" "github.com/rclone/rclone/fs"
@ -13,7 +14,7 @@ type inProgress struct {
} }
// newInProgress makes a new inProgress object // newInProgress makes a new inProgress object
func newInProgress() *inProgress { func newInProgress(ctx context.Context) *inProgress {
return &inProgress{ return &inProgress{
m: make(map[string]*Account, fs.Config.Transfers), m: make(map[string]*Account, fs.Config.Transfers),
} }

View file

@ -1,6 +1,8 @@
package accounting package accounting
import ( import (
"context"
"github.com/prometheus/client_golang/prometheus" "github.com/prometheus/client_golang/prometheus"
) )
@ -8,6 +10,7 @@ var namespace = "rclone_"
// RcloneCollector is a Prometheus collector for Rclone // RcloneCollector is a Prometheus collector for Rclone
type RcloneCollector struct { type RcloneCollector struct {
ctx context.Context
bytesTransferred *prometheus.Desc bytesTransferred *prometheus.Desc
transferSpeed *prometheus.Desc transferSpeed *prometheus.Desc
numOfErrors *prometheus.Desc numOfErrors *prometheus.Desc
@ -21,8 +24,9 @@ type RcloneCollector struct {
} }
// NewRcloneCollector make a new RcloneCollector // NewRcloneCollector make a new RcloneCollector
func NewRcloneCollector() *RcloneCollector { func NewRcloneCollector(ctx context.Context) *RcloneCollector {
return &RcloneCollector{ return &RcloneCollector{
ctx: ctx,
bytesTransferred: prometheus.NewDesc(namespace+"bytes_transferred_total", bytesTransferred: prometheus.NewDesc(namespace+"bytes_transferred_total",
"Total transferred bytes since the start of the Rclone process", "Total transferred bytes since the start of the Rclone process",
nil, nil, nil, nil,
@ -82,7 +86,7 @@ func (c *RcloneCollector) Describe(ch chan<- *prometheus.Desc) {
// Collect is part of the Collector interface: https://godoc.org/github.com/prometheus/client_golang/prometheus#Collector // Collect is part of the Collector interface: https://godoc.org/github.com/prometheus/client_golang/prometheus#Collector
func (c *RcloneCollector) Collect(ch chan<- prometheus.Metric) { func (c *RcloneCollector) Collect(ch chan<- prometheus.Metric) {
s := groups.sum() s := groups.sum(c.ctx)
s.mu.RLock() s.mu.RLock()
ch <- prometheus.MustNewConstMetric(c.bytesTransferred, prometheus.CounterValue, float64(s.bytes)) ch <- prometheus.MustNewConstMetric(c.bytesTransferred, prometheus.CounterValue, float64(s.bytes))

View file

@ -2,6 +2,7 @@ package accounting
import ( import (
"bytes" "bytes"
"context"
"fmt" "fmt"
"sort" "sort"
"strings" "strings"
@ -22,6 +23,7 @@ var startTime = time.Now()
// StatsInfo accounts all transfers // StatsInfo accounts all transfers
type StatsInfo struct { type StatsInfo struct {
mu sync.RWMutex mu sync.RWMutex
ctx context.Context
bytes int64 bytes int64
errors int64 errors int64
lastError error lastError error
@ -49,11 +51,12 @@ type StatsInfo struct {
} }
// NewStats creates an initialised StatsInfo // NewStats creates an initialised StatsInfo
func NewStats() *StatsInfo { func NewStats(ctx context.Context) *StatsInfo {
return &StatsInfo{ return &StatsInfo{
ctx: ctx,
checking: newTransferMap(fs.Config.Checkers, "checking"), checking: newTransferMap(fs.Config.Checkers, "checking"),
transferring: newTransferMap(fs.Config.Transfers, "transferring"), transferring: newTransferMap(fs.Config.Transfers, "transferring"),
inProgress: newInProgress(), inProgress: newInProgress(ctx),
} }
} }
@ -332,10 +335,10 @@ func (s *StatsInfo) String() string {
// 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.String(s.inProgress, s.transferring)) _, _ = fmt.Fprintf(buf, "Checking:\n%s\n", s.checking.String(s.ctx, s.inProgress, s.transferring))
} }
if !s.transferring.empty() { if !s.transferring.empty() {
_, _ = fmt.Fprintf(buf, "Transferring:\n%s\n", s.transferring.String(s.inProgress, nil)) _, _ = fmt.Fprintf(buf, "Transferring:\n%s\n", s.transferring.String(s.ctx, s.inProgress, nil))
} }
} }

View file

@ -59,10 +59,10 @@ func rcRemoteStats(ctx context.Context, in rc.Params) (rc.Params, error) {
return rc.Params{}, err return rc.Params{}, err
} }
if group != "" { if group != "" {
return StatsGroup(group).RemoteStats() return StatsGroup(ctx, group).RemoteStats()
} }
return groups.sum().RemoteStats() return groups.sum(ctx).RemoteStats()
} }
func init() { func init() {
@ -129,9 +129,9 @@ func rcTransferredStats(ctx context.Context, in rc.Params) (rc.Params, error) {
out := make(rc.Params) out := make(rc.Params)
if group != "" { if group != "" {
out["transferred"] = StatsGroup(group).Transferred() out["transferred"] = StatsGroup(ctx, group).Transferred()
} else { } else {
out["transferred"] = groups.sum().Transferred() out["transferred"] = groups.sum(ctx).Transferred()
} }
return out, nil return out, nil
@ -265,28 +265,28 @@ func Stats(ctx context.Context) *StatsInfo {
if !ok { if !ok {
return GlobalStats() return GlobalStats()
} }
return StatsGroup(group) return StatsGroup(ctx, group)
} }
// StatsGroup gets stats by group name. // StatsGroup gets stats by group name.
func StatsGroup(group string) *StatsInfo { func StatsGroup(ctx context.Context, group string) *StatsInfo {
stats := groups.get(group) stats := groups.get(group)
if stats == nil { if stats == nil {
return NewStatsGroup(group) return NewStatsGroup(ctx, group)
} }
return stats return stats
} }
// GlobalStats returns special stats used for global accounting. // GlobalStats returns special stats used for global accounting.
func GlobalStats() *StatsInfo { func GlobalStats() *StatsInfo {
return StatsGroup(globalStats) return StatsGroup(context.Background(), globalStats)
} }
// NewStatsGroup creates new stats under named group. // NewStatsGroup creates new stats under named group.
func NewStatsGroup(group string) *StatsInfo { func NewStatsGroup(ctx context.Context, group string) *StatsInfo {
stats := NewStats() stats := NewStats(ctx)
stats.group = group stats.group = group
groups.set(group, stats) groups.set(ctx, group, stats)
return stats return stats
} }
@ -305,7 +305,7 @@ func newStatsGroups() *statsGroups {
} }
// set marks the stats as belonging to a group // set marks the stats as belonging to a group
func (sg *statsGroups) set(group string, stats *StatsInfo) { func (sg *statsGroups) set(ctx context.Context, group string, stats *StatsInfo) {
sg.mu.Lock() sg.mu.Lock()
defer sg.mu.Unlock() defer sg.mu.Unlock()
@ -343,11 +343,11 @@ func (sg *statsGroups) names() []string {
} }
// sum returns aggregate stats that contains summation of all groups. // sum returns aggregate stats that contains summation of all groups.
func (sg *statsGroups) sum() *StatsInfo { func (sg *statsGroups) sum(ctx context.Context) *StatsInfo {
sg.mu.Lock() sg.mu.Lock()
defer sg.mu.Unlock() defer sg.mu.Unlock()
sum := NewStats() sum := NewStats(ctx)
for _, stats := range sg.m { for _, stats := range sg.m {
stats.mu.RLock() stats.mu.RLock()
{ {

View file

@ -1,6 +1,7 @@
package accounting package accounting
import ( import (
"context"
"fmt" "fmt"
"runtime" "runtime"
"testing" "testing"
@ -11,6 +12,7 @@ import (
) )
func TestStatsGroupOperations(t *testing.T) { func TestStatsGroupOperations(t *testing.T) {
ctx := context.Background()
t.Run("empty group returns nil", func(t *testing.T) { t.Run("empty group returns nil", func(t *testing.T) {
t.Parallel() t.Parallel()
@ -20,10 +22,10 @@ func TestStatsGroupOperations(t *testing.T) {
t.Run("set assigns stats to group", func(t *testing.T) { t.Run("set assigns stats to group", func(t *testing.T) {
t.Parallel() t.Parallel()
stats := NewStats() stats := NewStats(ctx)
sg := newStatsGroups() sg := newStatsGroups()
sg.set("test", stats) sg.set(ctx, "test", stats)
sg.set("test1", stats) sg.set(ctx, "test1", stats)
if len(sg.m) != len(sg.names()) || len(sg.m) != 2 { if len(sg.m) != len(sg.names()) || len(sg.m) != 2 {
t.Fatalf("Expected two stats got %d, %d", len(sg.m), len(sg.order)) t.Fatalf("Expected two stats got %d, %d", len(sg.m), len(sg.order))
} }
@ -31,10 +33,10 @@ func TestStatsGroupOperations(t *testing.T) {
t.Run("get returns correct group", func(t *testing.T) { t.Run("get returns correct group", func(t *testing.T) {
t.Parallel() t.Parallel()
stats := NewStats() stats := NewStats(ctx)
sg := newStatsGroups() sg := newStatsGroups()
sg.set("test", stats) sg.set(ctx, "test", stats)
sg.set("test1", stats) sg.set(ctx, "test1", stats)
got := sg.get("test") got := sg.get("test")
if got != stats { if got != stats {
t.Fatal("get returns incorrect stats") t.Fatal("get returns incorrect stats")
@ -43,20 +45,20 @@ func TestStatsGroupOperations(t *testing.T) {
t.Run("sum returns correct values", func(t *testing.T) { t.Run("sum returns correct values", func(t *testing.T) {
t.Parallel() t.Parallel()
stats1 := NewStats() stats1 := NewStats(ctx)
stats1.bytes = 5 stats1.bytes = 5
stats1.errors = 6 stats1.errors = 6
stats1.oldDuration = time.Second stats1.oldDuration = time.Second
stats1.oldTimeRanges = []timeRange{{time.Now(), time.Now().Add(time.Second)}} stats1.oldTimeRanges = []timeRange{{time.Now(), time.Now().Add(time.Second)}}
stats2 := NewStats() stats2 := NewStats(ctx)
stats2.bytes = 10 stats2.bytes = 10
stats2.errors = 12 stats2.errors = 12
stats2.oldDuration = 2 * time.Second stats2.oldDuration = 2 * time.Second
stats2.oldTimeRanges = []timeRange{{time.Now(), time.Now().Add(2 * time.Second)}} stats2.oldTimeRanges = []timeRange{{time.Now(), time.Now().Add(2 * time.Second)}}
sg := newStatsGroups() sg := newStatsGroups()
sg.set("test1", stats1) sg.set(ctx, "test1", stats1)
sg.set("test2", stats2) sg.set(ctx, "test2", stats2)
sum := sg.sum() sum := sg.sum(ctx)
assert.Equal(t, stats1.bytes+stats2.bytes, sum.bytes) assert.Equal(t, stats1.bytes+stats2.bytes, sum.bytes)
assert.Equal(t, stats1.errors+stats2.errors, sum.errors) assert.Equal(t, stats1.errors+stats2.errors, sum.errors)
assert.Equal(t, stats1.oldDuration+stats2.oldDuration, sum.oldDuration) assert.Equal(t, stats1.oldDuration+stats2.oldDuration, sum.oldDuration)
@ -70,10 +72,10 @@ func TestStatsGroupOperations(t *testing.T) {
t.Run("delete removes stats", func(t *testing.T) { t.Run("delete removes stats", func(t *testing.T) {
t.Parallel() t.Parallel()
stats := NewStats() stats := NewStats(ctx)
sg := newStatsGroups() sg := newStatsGroups()
sg.set("test", stats) sg.set(ctx, "test", stats)
sg.set("test1", stats) sg.set(ctx, "test1", stats)
sg.delete("test1") sg.delete("test1")
if sg.get("test1") != nil { if sg.get("test1") != nil {
t.Fatal("stats not deleted") t.Fatal("stats not deleted")
@ -95,7 +97,7 @@ func TestStatsGroupOperations(t *testing.T) {
runtime.ReadMemStats(&start) runtime.ReadMemStats(&start)
for i := 0; i < count; i++ { for i := 0; i < count; i++ {
sg.set(fmt.Sprintf("test-%d", i), NewStats()) sg.set(ctx, fmt.Sprintf("test-%d", i), NewStats(ctx))
} }
for i := 0; i < count; i++ { for i := 0; i < count; i++ {

View file

@ -1,6 +1,7 @@
package accounting package accounting
import ( import (
"context"
"fmt" "fmt"
"io" "io"
"testing" "testing"
@ -67,7 +68,8 @@ func TestPercentage(t *testing.T) {
} }
func TestStatsError(t *testing.T) { func TestStatsError(t *testing.T) {
s := NewStats() ctx := context.Background()
s := NewStats(ctx)
assert.Equal(t, int64(0), s.GetErrors()) assert.Equal(t, int64(0), s.GetErrors())
assert.False(t, s.HadFatalError()) assert.False(t, s.HadFatalError())
assert.False(t, s.HadRetryError()) assert.False(t, s.HadRetryError())
@ -132,6 +134,7 @@ func TestStatsError(t *testing.T) {
} }
func TestStatsTotalDuration(t *testing.T) { func TestStatsTotalDuration(t *testing.T) {
ctx := context.Background()
startTime := time.Now() startTime := time.Now()
time1 := startTime.Add(-40 * time.Second) time1 := startTime.Add(-40 * time.Second)
time2 := time1.Add(10 * time.Second) time2 := time1.Add(10 * time.Second)
@ -139,7 +142,7 @@ func TestStatsTotalDuration(t *testing.T) {
time4 := time3.Add(10 * time.Second) time4 := time3.Add(10 * time.Second)
t.Run("Single completed transfer", func(t *testing.T) { t.Run("Single completed transfer", func(t *testing.T) {
s := NewStats() s := NewStats(ctx)
tr1 := &Transfer{ tr1 := &Transfer{
startedAt: time1, startedAt: time1,
completedAt: time2, completedAt: time2,
@ -158,7 +161,7 @@ func TestStatsTotalDuration(t *testing.T) {
}) })
t.Run("Single uncompleted transfer", func(t *testing.T) { t.Run("Single uncompleted transfer", func(t *testing.T) {
s := NewStats() s := NewStats(ctx)
tr1 := &Transfer{ tr1 := &Transfer{
startedAt: time1, startedAt: time1,
} }
@ -174,7 +177,7 @@ func TestStatsTotalDuration(t *testing.T) {
}) })
t.Run("Overlapping without ending", func(t *testing.T) { t.Run("Overlapping without ending", func(t *testing.T) {
s := NewStats() s := NewStats(ctx)
tr1 := &Transfer{ tr1 := &Transfer{
startedAt: time2, startedAt: time2,
completedAt: time3, completedAt: time3,
@ -218,7 +221,7 @@ func TestStatsTotalDuration(t *testing.T) {
}) })
t.Run("Mixed completed and uncompleted transfers", func(t *testing.T) { t.Run("Mixed completed and uncompleted transfers", func(t *testing.T) {
s := NewStats() s := NewStats(ctx)
s.AddTransfer(&Transfer{ s.AddTransfer(&Transfer{
startedAt: time1, startedAt: time1,
completedAt: time2, completedAt: time2,
@ -382,6 +385,7 @@ func TestTimeRangeDuration(t *testing.T) {
} }
func TestPruneTransfers(t *testing.T) { func TestPruneTransfers(t *testing.T) {
ctx := context.Background()
for _, test := range []struct { for _, test := range []struct {
Name string Name string
Transfers int Transfers int
@ -406,7 +410,7 @@ func TestPruneTransfers(t *testing.T) {
MaxCompletedTransfers = test.Limit MaxCompletedTransfers = test.Limit
defer func() { MaxCompletedTransfers = prevLimit }() defer func() { MaxCompletedTransfers = prevLimit }()
s := NewStats() s := NewStats(ctx)
for i := int64(1); i <= int64(test.Transfers); i++ { for i := int64(1); i <= int64(test.Transfers); i++ {
s.AddTransfer(&Transfer{ s.AddTransfer(&Transfer{
startedAt: time.Unix(i, 0), startedAt: time.Unix(i, 0),

View file

@ -35,7 +35,7 @@ func newTokenBucket(bandwidth fs.SizeSuffix) *rate.Limiter {
} }
// StartTokenBucket starts the token bucket if necessary // StartTokenBucket starts the token bucket if necessary
func StartTokenBucket() { func StartTokenBucket(ctx context.Context) {
currLimitMu.Lock() currLimitMu.Lock()
currLimit := fs.Config.BwLimit.LimitAt(time.Now()) currLimit := fs.Config.BwLimit.LimitAt(time.Now())
currLimitMu.Unlock() currLimitMu.Unlock()
@ -51,7 +51,7 @@ func StartTokenBucket() {
} }
// StartTokenTicker creates a ticker to update the bandwidth limiter every minute. // StartTokenTicker creates a ticker to update the bandwidth limiter every minute.
func StartTokenTicker() { func StartTokenTicker(ctx context.Context) {
// If the timetable has a single entry or was not specified, we don't need // If the timetable has a single entry or was not specified, we don't need
// a ticker to update the bandwidth. // a ticker to update the bandwidth.
if len(fs.Config.BwLimit) <= 1 { if len(fs.Config.BwLimit) <= 1 {

View file

@ -86,7 +86,7 @@ func newTransferRemoteSize(stats *StatsInfo, remote string, size int64, checking
// Done ends the transfer. // Done ends the transfer.
// Must be called after transfer is finished to run proper cleanups. // Must be called after transfer is finished to run proper cleanups.
func (tr *Transfer) Done(err error) { func (tr *Transfer) Done(ctx context.Context, err error) {
if err != nil { if err != nil {
err = tr.stats.Error(err) err = tr.stats.Error(err)
@ -123,7 +123,7 @@ func (tr *Transfer) Done(err error) {
} }
// Reset allows to switch the Account to another transfer method. // Reset allows to switch the Account to another transfer method.
func (tr *Transfer) Reset() { func (tr *Transfer) Reset(ctx context.Context) {
tr.mu.RLock() tr.mu.RLock()
acc := tr.acc acc := tr.acc
tr.acc = nil tr.acc = nil

View file

@ -1,6 +1,7 @@
package accounting package accounting
import ( import (
"context"
"fmt" "fmt"
"sort" "sort"
"strings" "strings"
@ -88,7 +89,7 @@ func (tm *transferMap) _sortedSlice() []*Transfer {
// String returns string representation of map items excluding any in // String returns string representation of map items excluding any in
// exclude (if set). // exclude (if set).
func (tm *transferMap) String(progress *inProgress, exclude *transferMap) string { func (tm *transferMap) String(ctx context.Context, progress *inProgress, exclude *transferMap) string {
tm.mu.RLock() tm.mu.RLock()
defer tm.mu.RUnlock() defer tm.mu.RUnlock()
stringList := make([]string, 0, len(tm.items)) stringList := make([]string, 0, len(tm.items))

View file

@ -4,6 +4,7 @@ package config
import ( import (
"bufio" "bufio"
"bytes" "bytes"
"context"
"crypto/rand" "crypto/rand"
"crypto/sha256" "crypto/sha256"
"encoding/base64" "encoding/base64"
@ -113,7 +114,7 @@ func init() {
func getConfigData() *goconfig.ConfigFile { func getConfigData() *goconfig.ConfigFile {
if configFile == nil { if configFile == nil {
LoadConfig() LoadConfig(context.Background())
} }
return configFile return configFile
} }
@ -212,7 +213,7 @@ func makeConfigPath() string {
} }
// LoadConfig loads the config file // LoadConfig loads the config file
func LoadConfig() { func LoadConfig(ctx context.Context) {
// Set RCLONE_CONFIG_DIR for backend config and subprocesses // Set RCLONE_CONFIG_DIR for backend config and subprocesses
_ = os.Setenv("RCLONE_CONFIG_DIR", filepath.Dir(ConfigPath)) _ = os.Setenv("RCLONE_CONFIG_DIR", filepath.Dir(ConfigPath))
@ -229,10 +230,10 @@ func LoadConfig() {
} }
// Start the token bucket limiter // Start the token bucket limiter
accounting.StartTokenBucket() accounting.StartTokenBucket(ctx)
// Start the bandwidth update ticker // Start the bandwidth update ticker
accounting.StartTokenTicker() accounting.StartTokenTicker(ctx)
// Start the transactions per second limiter // Start the transactions per second limiter
fshttp.StartHTTPTokenBucket() fshttp.StartHTTPTokenBucket()

View file

@ -2,6 +2,7 @@ package config
import ( import (
"bytes" "bytes"
"context"
"fmt" "fmt"
"io/ioutil" "io/ioutil"
"os" "os"
@ -15,6 +16,7 @@ import (
) )
func testConfigFile(t *testing.T, configFileName string) func() { func testConfigFile(t *testing.T, configFileName string) func() {
ctx := context.Background()
configKey = nil // reset password configKey = nil // reset password
_ = os.Unsetenv("_RCLONE_CONFIG_KEY_FILE") _ = os.Unsetenv("_RCLONE_CONFIG_KEY_FILE")
_ = os.Unsetenv("RCLONE_CONFIG_PASS") _ = os.Unsetenv("RCLONE_CONFIG_PASS")
@ -36,7 +38,7 @@ func testConfigFile(t *testing.T, configFileName string) func() {
fs.Config = &fs.ConfigInfo{} fs.Config = &fs.ConfigInfo{}
configFile = nil configFile = nil
LoadConfig() LoadConfig(ctx)
assert.Equal(t, []string{}, getConfigData().GetSectionList()) assert.Equal(t, []string{}, getConfigData().GetSectionList())
// Fake a remote // Fake a remote

View file

@ -116,7 +116,7 @@ func (c *checkMarch) SrcOnly(src fs.DirEntry) (recurse bool) {
func (c *checkMarch) checkIdentical(ctx context.Context, dst, src fs.Object) (differ bool, noHash bool, err error) { func (c *checkMarch) checkIdentical(ctx context.Context, dst, src fs.Object) (differ bool, noHash bool, err error) {
tr := accounting.Stats(ctx).NewCheckingTransfer(src) tr := accounting.Stats(ctx).NewCheckingTransfer(src)
defer func() { defer func() {
tr.Done(err) tr.Done(ctx, err)
}() }()
if sizeDiffers(src, dst) { if sizeDiffers(src, dst) {
err = errors.Errorf("Sizes differ") err = errors.Errorf("Sizes differ")
@ -323,7 +323,7 @@ func checkIdenticalDownload(ctx context.Context, dst, src fs.Object) (differ boo
} }
tr1 := accounting.Stats(ctx).NewTransfer(dst) tr1 := accounting.Stats(ctx).NewTransfer(dst)
defer func() { defer func() {
tr1.Done(nil) // error handling is done by the caller tr1.Done(ctx, nil) // error handling is done by the caller
}() }()
in1 = tr1.Account(ctx, in1).WithBuffer() // account and buffer the transfer in1 = tr1.Account(ctx, in1).WithBuffer() // account and buffer the transfer
@ -333,7 +333,7 @@ func checkIdenticalDownload(ctx context.Context, dst, src fs.Object) (differ boo
} }
tr2 := accounting.Stats(ctx).NewTransfer(dst) tr2 := accounting.Stats(ctx).NewTransfer(dst)
defer func() { defer func() {
tr2.Done(nil) // error handling is done by the caller tr2.Done(ctx, nil) // error handling is done by the caller
}() }()
in2 = tr2.Account(ctx, in2).WithBuffer() // account and buffer the transfer in2 = tr2.Account(ctx, in2).WithBuffer() // account and buffer the transfer

View file

@ -132,7 +132,7 @@ func TestMultithreadCopy(t *testing.T) {
tr := accounting.GlobalStats().NewTransfer(src) tr := accounting.GlobalStats().NewTransfer(src)
defer func() { defer func() {
tr.Done(err) tr.Done(ctx, err)
}() }()
dst, err := multiThreadCopy(ctx, r.Flocal, "file1", src, 2, tr) dst, err := multiThreadCopy(ctx, r.Flocal, "file1", src, 2, tr)
require.NoError(t, err) require.NoError(t, err)

View file

@ -359,7 +359,7 @@ func CommonHash(fa, fb fs.Info) (hash.Type, *fs.HashesOption) {
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(ctx).NewTransfer(src) tr := accounting.Stats(ctx).NewTransfer(src)
defer func() { defer func() {
tr.Done(err) tr.Done(ctx, err)
}() }()
newDst = dst newDst = dst
if SkipDestructive(ctx, src, "copy") { if SkipDestructive(ctx, src, "copy") {
@ -401,7 +401,7 @@ func Copy(ctx context.Context, f fs.Fs, dst fs.Object, remote string, src fs.Obj
_ = in.Close() _ = in.Close()
} }
if err == fs.ErrorCantCopy { if err == fs.ErrorCantCopy {
tr.Reset() // skip incomplete accounting - will be overwritten by the manual copy below tr.Reset(ctx) // skip incomplete accounting - will be overwritten by the manual copy below
} }
} else { } else {
err = fs.ErrorCantCopy err = fs.ErrorCantCopy
@ -478,7 +478,7 @@ func Copy(ctx context.Context, f fs.Fs, dst fs.Object, remote string, src fs.Obj
// Retry if err returned a retry error // Retry if err returned a retry error
if fserrors.IsRetryError(err) || fserrors.ShouldRetry(err) { if fserrors.IsRetryError(err) || fserrors.ShouldRetry(err) {
fs.Debugf(src, "Received error: %v - low level retry %d/%d", err, tries, maxTries) fs.Debugf(src, "Received error: %v - low level retry %d/%d", err, tries, maxTries)
tr.Reset() // skip incomplete accounting - will be overwritten by retry tr.Reset(ctx) // skip incomplete accounting - will be overwritten by retry
continue continue
} }
// otherwise finish // otherwise finish
@ -550,7 +550,7 @@ func Move(ctx context.Context, fdst fs.Fs, dst fs.Object, remote string, src fs.
if err == nil { if err == nil {
accounting.Stats(ctx).Renames(1) accounting.Stats(ctx).Renames(1)
} }
tr.Done(err) tr.Done(ctx, err)
}() }()
newDst = dst newDst = dst
if SkipDestructive(ctx, src, "move") { if SkipDestructive(ctx, src, "move") {
@ -627,7 +627,7 @@ func SuffixName(remote string) string {
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) {
tr := accounting.Stats(ctx).NewCheckingTransfer(dst) tr := accounting.Stats(ctx).NewCheckingTransfer(dst)
defer func() { defer func() {
tr.Done(err) tr.Done(ctx, err)
}() }()
numDeletes := accounting.Stats(ctx).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 {
@ -817,7 +817,7 @@ 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) {
tr := accounting.Stats(ctx).NewCheckingTransfer(o) tr := accounting.Stats(ctx).NewCheckingTransfer(o)
defer func() { defer func() {
tr.Done(nil) tr.Done(ctx, nil)
}() }()
modTime := o.ModTime(ctx) modTime := o.ModTime(ctx)
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())
@ -850,7 +850,7 @@ func hashSum(ctx context.Context, ht hash.Type, o fs.Object) (string, error) {
var err error var err error
tr := accounting.Stats(ctx).NewCheckingTransfer(o) tr := accounting.Stats(ctx).NewCheckingTransfer(o)
defer func() { defer func() {
tr.Done(err) tr.Done(ctx, err)
}() }()
sum, err := o.Hash(ctx, ht) sum, err := o.Hash(ctx, ht)
if err == hash.ErrUnsupported { if err == hash.ErrUnsupported {
@ -1058,7 +1058,7 @@ func Cat(ctx context.Context, f fs.Fs, w io.Writer, offset, count int64) error {
var err error var err error
tr := accounting.Stats(ctx).NewTransfer(o) tr := accounting.Stats(ctx).NewTransfer(o)
defer func() { defer func() {
tr.Done(err) tr.Done(ctx, err)
}() }()
opt := fs.RangeOption{Start: offset, End: -1} opt := fs.RangeOption{Start: offset, End: -1}
size := o.Size() size := o.Size()
@ -1100,7 +1100,7 @@ func Cat(ctx context.Context, f fs.Fs, w io.Writer, offset, count int64) error {
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(ctx).NewTransferRemoteSize(dstFileName, -1) tr := accounting.Stats(ctx).NewTransferRemoteSize(dstFileName, -1)
defer func() { defer func() {
tr.Done(err) tr.Done(ctx, err)
}() }()
in = tr.Account(ctx, in).WithBuffer() in = tr.Account(ctx, in).WithBuffer()
@ -1447,7 +1447,7 @@ func RcatSize(ctx context.Context, fdst fs.Fs, dstFileName string, in io.ReadClo
// Size known use Put // Size known use Put
tr := accounting.Stats(ctx).NewTransferRemoteSize(dstFileName, size) tr := accounting.Stats(ctx).NewTransferRemoteSize(dstFileName, size)
defer func() { defer func() {
tr.Done(err) tr.Done(ctx, err)
}() }()
body := ioutil.NopCloser(in) // we let the server close the body body := ioutil.NopCloser(in) // we let the server close the body
in := tr.Account(ctx, body) // account the transfer (no buffering) in := tr.Account(ctx, body) // account the transfer (no buffering)
@ -1624,7 +1624,7 @@ func moveOrCopyFile(ctx context.Context, fdst fs.Fs, fsrc fs.Fs, dstFileName str
} }
tr := accounting.Stats(ctx).NewTransfer(srcObj) tr := accounting.Stats(ctx).NewTransfer(srcObj)
defer func() { defer func() {
tr.Done(err) tr.Done(ctx, err)
}() }()
tmpObj, err := Op(ctx, fdst, nil, tmpObjName, srcObj) tmpObj, err := Op(ctx, fdst, nil, tmpObjName, srcObj)
if err != nil { if err != nil {
@ -1673,7 +1673,7 @@ func moveOrCopyFile(ctx context.Context, fdst fs.Fs, fsrc fs.Fs, dstFileName str
if !cp { if !cp {
err = DeleteFile(ctx, srcObj) err = DeleteFile(ctx, srcObj)
} }
tr.Done(err) tr.Done(ctx, err)
} }
return err return err
} }

View file

@ -42,7 +42,7 @@ var promHandler http.Handler
var onlyOnceWarningAllowOrigin sync.Once var onlyOnceWarningAllowOrigin sync.Once
func init() { func init() {
rcloneCollector := accounting.NewRcloneCollector() rcloneCollector := accounting.NewRcloneCollector(context.Background())
prometheus.MustRegister(rcloneCollector) prometheus.MustRegister(rcloneCollector)
promHandler = promhttp.Handler() promHandler = promhttp.Handler()
} }

View file

@ -344,7 +344,7 @@ func (s *syncCopyMove) pairChecker(in *pipe, out *pipe, fraction int, wg *sync.W
} }
} }
} }
tr.Done(err) tr.Done(s.ctx, err)
} }
} }
@ -749,7 +749,7 @@ func (s *syncCopyMove) makeRenameMap() {
s.pushRenameMap(hash, obj) s.pushRenameMap(hash, obj)
} }
tr.Done(nil) tr.Done(s.ctx, nil)
} }
} }
}() }()

View file

@ -58,6 +58,7 @@ func init() {
// Initialise rclone for testing // Initialise rclone for testing
func Initialise() { func Initialise() {
ctx := context.Background()
// Never ask for passwords, fail instead. // Never ask for passwords, fail instead.
// If your local config is encrypted set environment variable // If your local config is encrypted set environment variable
// "RCLONE_CONFIG_PASS=hunter2" (or your password) // "RCLONE_CONFIG_PASS=hunter2" (or your password)
@ -68,7 +69,7 @@ func Initialise() {
if envConfig := os.Getenv("RCLONE_CONFIG"); envConfig != "" { if envConfig := os.Getenv("RCLONE_CONFIG"); envConfig != "" {
config.ConfigPath = envConfig config.ConfigPath = envConfig
} }
config.LoadConfig() config.LoadConfig(ctx)
if *Verbose { if *Verbose {
fs.Config.LogLevel = fs.LogLevelDebug fs.Config.LogLevel = fs.LogLevelDebug
} }

View file

@ -17,7 +17,7 @@ import (
// ReadFileHandle is an open for read file handle on a File // ReadFileHandle is an open for read file handle on a File
type ReadFileHandle struct { type ReadFileHandle struct {
baseHandle baseHandle
done func(err error) done func(ctx context.Context, err error)
mu sync.Mutex mu sync.Mutex
cond *sync.Cond // cond lock for out of sequence reads cond *sync.Cond // cond lock for out of sequence reads
closed bool // set if handle has been closed closed bool // set if handle has been closed
@ -414,7 +414,7 @@ func (fh *ReadFileHandle) close() error {
if fh.opened { if fh.opened {
var err error var err error
defer func() { defer func() {
fh.done(err) fh.done(context.TODO(), err)
}() }()
// Close first so that we have hashes // Close first so that we have hashes
err = fh.r.Close() err = fh.r.Close()

View file

@ -553,7 +553,7 @@ func (dl *downloader) close(inErr error) (err error) {
dl.in = nil dl.in = nil
} }
if dl.tr != nil { if dl.tr != nil {
dl.tr.Done(inErr) dl.tr.Done(dl.dls.ctx, inErr)
dl.tr = nil dl.tr = nil
} }
dl._closed = true dl._closed = true