1
mirror of https://github.com/rclone/rclone synced 2025-01-14 18:27:30 +01:00

fstest: factor out retry logic from put code and make testPut return the object too

This commit is contained in:
Nick Craig-Wood 2018-11-15 17:15:56 +00:00
parent 084fe38922
commit 762561f88e

View File

@ -125,60 +125,65 @@ func findObject(t *testing.T, f fs.Fs, Name string) fs.Object {
return obj return obj
} }
// testPut puts file to the remote // retry f() until no retriable error
func testPut(t *testing.T, f fs.Fs, file *fstest.Item) string { func retry(t *testing.T, what string, f func() error) {
tries := 1
const maxTries = 10 const maxTries = 10
again: var err error
contents := fstest.RandomString(100) for tries := 1; tries <= maxTries; tries++ {
buf := bytes.NewBufferString(contents) err = f()
hash := hash.NewMultiHasher() // exit if no error, or error is not retriable
in := io.TeeReader(buf, hash) if err == nil || !fserrors.IsRetryError(err) {
break
file.Size = int64(buf.Len())
obji := object.NewStaticObjectInfo(file.Path, file.ModTime, file.Size, true, nil, nil)
obj, err := f.Put(in, obji)
if err != nil {
// Retry if err returned a retry error
if fserrors.IsRetryError(err) && tries < maxTries {
t.Logf("Put error: %v - low level retry %d/%d", err, tries, maxTries)
time.Sleep(2 * time.Second)
tries++
goto again
} }
require.NoError(t, err, fmt.Sprintf("Put error: %v", err)) t.Logf("%s error: %v - low level retry %d/%d", what, err, tries, maxTries)
time.Sleep(2 * time.Second)
} }
file.Hashes = hash.Sums() require.NoError(t, err, what)
}
// testPut puts file to the remote
func testPut(t *testing.T, f fs.Fs, file *fstest.Item) (string, fs.Object) {
var (
err error
obj fs.Object
uploadHash *hash.MultiHasher
contents string
)
retry(t, "Put", func() error {
contents = fstest.RandomString(100)
buf := bytes.NewBufferString(contents)
uploadHash = hash.NewMultiHasher()
in := io.TeeReader(buf, uploadHash)
file.Size = int64(buf.Len())
obji := object.NewStaticObjectInfo(file.Path, file.ModTime, file.Size, true, nil, nil)
obj, err = f.Put(in, obji)
return err
})
file.Hashes = uploadHash.Sums()
file.Check(t, obj, f.Precision()) file.Check(t, obj, f.Precision())
// Re-read the object and check again // Re-read the object and check again
obj = findObject(t, f, file.Path) obj = findObject(t, f, file.Path)
file.Check(t, obj, f.Precision()) file.Check(t, obj, f.Precision())
return contents return contents, obj
} }
// testPutLarge puts file to the remote, checks it and removes it on success. // testPutLarge puts file to the remote, checks it and removes it on success.
func testPutLarge(t *testing.T, f fs.Fs, file *fstest.Item) { func testPutLarge(t *testing.T, f fs.Fs, file *fstest.Item) {
tries := 1 var (
const maxTries = 10 err error
again: obj fs.Object
r := readers.NewPatternReader(file.Size) uploadHash *hash.MultiHasher
uploadHash := hash.NewMultiHasher() )
in := io.TeeReader(r, uploadHash) retry(t, "PutLarge", func() error {
r := readers.NewPatternReader(file.Size)
uploadHash = hash.NewMultiHasher()
in := io.TeeReader(r, uploadHash)
obji := object.NewStaticObjectInfo(file.Path, file.ModTime, file.Size, true, nil, nil) obji := object.NewStaticObjectInfo(file.Path, file.ModTime, file.Size, true, nil, nil)
obj, err := f.Put(in, obji) obj, err = f.Put(in, obji)
if err != nil { return err
// Retry if err returned a retry error })
if fserrors.IsRetryError(err) && tries < maxTries {
t.Logf("Put error: %v - low level retry %d/%d", err, tries, maxTries)
time.Sleep(2 * time.Second)
tries++
goto again
}
require.NoError(t, err, fmt.Sprintf("Put error: %v", err))
}
file.Hashes = uploadHash.Sums() file.Hashes = uploadHash.Sums()
file.Check(t, obj, f.Precision()) file.Check(t, obj, f.Precision())
@ -496,7 +501,7 @@ func Run(t *testing.T, opt *Opt) {
// TestFsPutFile1 tests putting a file // TestFsPutFile1 tests putting a file
t.Run("TestFsPutFile1", func(t *testing.T) { t.Run("TestFsPutFile1", func(t *testing.T) {
skipIfNotOk(t) skipIfNotOk(t)
file1Contents = testPut(t, remote, &file1) file1Contents, _ = testPut(t, remote, &file1)
}) })
// TestFsPutError tests uploading a file where there is an error // TestFsPutError tests uploading a file where there is an error
@ -534,7 +539,7 @@ func Run(t *testing.T, opt *Opt) {
// TestFsUpdateFile1 tests updating file1 with new contents // TestFsUpdateFile1 tests updating file1 with new contents
t.Run("TestFsUpdateFile1", func(t *testing.T) { t.Run("TestFsUpdateFile1", func(t *testing.T) {
skipIfNotOk(t) skipIfNotOk(t)
file1Contents = testPut(t, remote, &file1) file1Contents, _ = testPut(t, remote, &file1)
// Note that the next test will check there are no duplicated file names // Note that the next test will check there are no duplicated file names
}) })
@ -1335,30 +1340,24 @@ func Run(t *testing.T, opt *Opt) {
Size: -1, // use unknown size during upload Size: -1, // use unknown size during upload
} }
tries := 1 var (
const maxTries = 10 err error
again: obj fs.Object
contentSize := 100 uploadHash *hash.MultiHasher
contents := fstest.RandomString(contentSize) contentSize = 100
buf := bytes.NewBufferString(contents) )
hash := hash.NewMultiHasher() retry(t, "PutStream", func() error {
in := io.TeeReader(buf, hash) contents := fstest.RandomString(contentSize)
buf := bytes.NewBufferString(contents)
uploadHash = hash.NewMultiHasher()
in := io.TeeReader(buf, uploadHash)
file.Size = -1 file.Size = -1
obji := object.NewStaticObjectInfo(file.Path, file.ModTime, file.Size, true, nil, nil) obji := object.NewStaticObjectInfo(file.Path, file.ModTime, file.Size, true, nil, nil)
obj, err := remote.Features().PutStream(in, obji) obj, err = remote.Features().PutStream(in, obji)
if err != nil { return err
// Retry if err returned a retry error })
if fserrors.IsRetryError(err) && tries < maxTries { file.Hashes = uploadHash.Sums()
t.Logf("Put error: %v - low level retry %d/%d", err, tries, maxTries)
time.Sleep(2 * time.Second)
tries++
goto again
}
require.NoError(t, err, fmt.Sprintf("PutStream Unknown Length error: %v", err))
}
file.Hashes = hash.Sums()
file.Size = int64(contentSize) // use correct size when checking file.Size = int64(contentSize) // use correct size when checking
file.Check(t, obj, remote.Precision()) file.Check(t, obj, remote.Precision())
// Re-read the object and check again // Re-read the object and check again