fstest: factor chunked copy tests from b2 and use them in s3 and oos
This commit is contained in:
parent
d5d28a7513
commit
c27977d4d5
8 changed files with 133 additions and 45 deletions
|
@ -455,6 +455,14 @@ func (f *Fs) setUploadCutoff(cs fs.SizeSuffix) (old fs.SizeSuffix, err error) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (f *Fs) setCopyCutoff(cs fs.SizeSuffix) (old fs.SizeSuffix, err error) {
|
||||||
|
err = checkUploadChunkSize(cs)
|
||||||
|
if err == nil {
|
||||||
|
old, f.opt.CopyCutoff = f.opt.CopyCutoff, cs
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// setRoot changes the root of the Fs
|
// setRoot changes the root of the Fs
|
||||||
func (f *Fs) setRoot(root string) {
|
func (f *Fs) setRoot(root string) {
|
||||||
f.root = parsePath(root)
|
f.root = parsePath(root)
|
||||||
|
|
|
@ -178,48 +178,6 @@ func TestParseTimeString(t *testing.T) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// The integration tests do a reasonable job of testing the normal
|
|
||||||
// copy but don't test the chunked copy.
|
|
||||||
func (f *Fs) InternalTestChunkedCopy(t *testing.T) {
|
|
||||||
ctx := context.Background()
|
|
||||||
|
|
||||||
contents := random.String(8 * 1024 * 1024)
|
|
||||||
item := fstest.NewItem("chunked-copy", contents, fstest.Time("2001-05-06T04:05:06.499999999Z"))
|
|
||||||
src := fstests.PutTestContents(ctx, t, f, &item, contents, true)
|
|
||||||
defer func() {
|
|
||||||
assert.NoError(t, src.Remove(ctx))
|
|
||||||
}()
|
|
||||||
|
|
||||||
var itemCopy = item
|
|
||||||
itemCopy.Path += ".copy"
|
|
||||||
|
|
||||||
// Set copy cutoff to mininum value so we make chunks
|
|
||||||
origCutoff := f.opt.CopyCutoff
|
|
||||||
f.opt.CopyCutoff = minChunkSize
|
|
||||||
defer func() {
|
|
||||||
f.opt.CopyCutoff = origCutoff
|
|
||||||
}()
|
|
||||||
|
|
||||||
// Do the copy
|
|
||||||
dst, err := f.Copy(ctx, src, itemCopy.Path)
|
|
||||||
require.NoError(t, err)
|
|
||||||
defer func() {
|
|
||||||
assert.NoError(t, dst.Remove(ctx))
|
|
||||||
}()
|
|
||||||
|
|
||||||
// Check size
|
|
||||||
assert.Equal(t, src.Size(), dst.Size())
|
|
||||||
|
|
||||||
// Check modtime
|
|
||||||
srcModTime := src.ModTime(ctx)
|
|
||||||
dstModTime := dst.ModTime(ctx)
|
|
||||||
assert.True(t, srcModTime.Equal(dstModTime))
|
|
||||||
|
|
||||||
// Make sure contents are correct
|
|
||||||
gotContents := fstests.ReadObject(ctx, t, dst, -1)
|
|
||||||
assert.Equal(t, contents, gotContents)
|
|
||||||
}
|
|
||||||
|
|
||||||
// The integration tests do a reasonable job of testing the normal
|
// The integration tests do a reasonable job of testing the normal
|
||||||
// streaming upload but don't test the chunked streaming upload.
|
// streaming upload but don't test the chunked streaming upload.
|
||||||
func (f *Fs) InternalTestChunkedStreamingUpload(t *testing.T, size int) {
|
func (f *Fs) InternalTestChunkedStreamingUpload(t *testing.T, size int) {
|
||||||
|
@ -259,7 +217,6 @@ func (f *Fs) InternalTestChunkedStreamingUpload(t *testing.T, size int) {
|
||||||
|
|
||||||
// -run TestIntegration/FsMkdir/FsPutFiles/Internal
|
// -run TestIntegration/FsMkdir/FsPutFiles/Internal
|
||||||
func (f *Fs) InternalTest(t *testing.T) {
|
func (f *Fs) InternalTest(t *testing.T) {
|
||||||
t.Run("ChunkedCopy", f.InternalTestChunkedCopy)
|
|
||||||
for _, size := range []fs.SizeSuffix{
|
for _, size := range []fs.SizeSuffix{
|
||||||
minChunkSize - 1,
|
minChunkSize - 1,
|
||||||
minChunkSize,
|
minChunkSize,
|
||||||
|
|
|
@ -28,7 +28,12 @@ func (f *Fs) SetUploadCutoff(cs fs.SizeSuffix) (fs.SizeSuffix, error) {
|
||||||
return f.setUploadCutoff(cs)
|
return f.setUploadCutoff(cs)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (f *Fs) SetCopyCutoff(cs fs.SizeSuffix) (fs.SizeSuffix, error) {
|
||||||
|
return f.setCopyCutoff(cs)
|
||||||
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
_ fstests.SetUploadChunkSizer = (*Fs)(nil)
|
_ fstests.SetUploadChunkSizer = (*Fs)(nil)
|
||||||
_ fstests.SetUploadCutoffer = (*Fs)(nil)
|
_ fstests.SetUploadCutoffer = (*Fs)(nil)
|
||||||
|
_ fstests.SetCopyCutoffer = (*Fs)(nil)
|
||||||
)
|
)
|
||||||
|
|
|
@ -138,6 +138,14 @@ func (f *Fs) setUploadCutoff(cs fs.SizeSuffix) (old fs.SizeSuffix, err error) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (f *Fs) setCopyCutoff(cs fs.SizeSuffix) (old fs.SizeSuffix, err error) {
|
||||||
|
err = checkUploadChunkSize(cs)
|
||||||
|
if err == nil {
|
||||||
|
old, f.opt.CopyCutoff = f.opt.CopyCutoff, cs
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// ------------------------------------------------------------
|
// ------------------------------------------------------------
|
||||||
// Implement backed that represents a remote object storage server
|
// Implement backed that represents a remote object storage server
|
||||||
// Fs is the interface a cloud storage system must provide
|
// Fs is the interface a cloud storage system must provide
|
||||||
|
|
|
@ -30,4 +30,12 @@ func (f *Fs) SetUploadCutoff(cs fs.SizeSuffix) (fs.SizeSuffix, error) {
|
||||||
return f.setUploadCutoff(cs)
|
return f.setUploadCutoff(cs)
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ fstests.SetUploadChunkSizer = (*Fs)(nil)
|
func (f *Fs) SetCopyCutoff(cs fs.SizeSuffix) (fs.SizeSuffix, error) {
|
||||||
|
return f.setCopyCutoff(cs)
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
_ fstests.SetUploadChunkSizer = (*Fs)(nil)
|
||||||
|
_ fstests.SetUploadCutoffer = (*Fs)(nil)
|
||||||
|
_ fstests.SetCopyCutoffer = (*Fs)(nil)
|
||||||
|
)
|
||||||
|
|
|
@ -3011,6 +3011,14 @@ func (f *Fs) setUploadCutoff(cs fs.SizeSuffix) (old fs.SizeSuffix, err error) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (f *Fs) setCopyCutoff(cs fs.SizeSuffix) (old fs.SizeSuffix, err error) {
|
||||||
|
err = checkUploadChunkSize(cs)
|
||||||
|
if err == nil {
|
||||||
|
old, f.opt.CopyCutoff = f.opt.CopyCutoff, cs
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// setEndpointValueForIDriveE2 gets user region endpoint against the Access Key details by calling the API
|
// setEndpointValueForIDriveE2 gets user region endpoint against the Access Key details by calling the API
|
||||||
func setEndpointValueForIDriveE2(m configmap.Mapper) (err error) {
|
func setEndpointValueForIDriveE2(m configmap.Mapper) (err error) {
|
||||||
value, ok := m.Get(fs.ConfigProvider)
|
value, ok := m.Get(fs.ConfigProvider)
|
||||||
|
|
|
@ -47,4 +47,12 @@ func (f *Fs) SetUploadCutoff(cs fs.SizeSuffix) (fs.SizeSuffix, error) {
|
||||||
return f.setUploadCutoff(cs)
|
return f.setUploadCutoff(cs)
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ fstests.SetUploadChunkSizer = (*Fs)(nil)
|
func (f *Fs) SetCopyCutoff(cs fs.SizeSuffix) (fs.SizeSuffix, error) {
|
||||||
|
return f.setCopyCutoff(cs)
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
_ fstests.SetUploadChunkSizer = (*Fs)(nil)
|
||||||
|
_ fstests.SetUploadCutoffer = (*Fs)(nil)
|
||||||
|
_ fstests.SetCopyCutoffer = (*Fs)(nil)
|
||||||
|
)
|
||||||
|
|
|
@ -78,6 +78,13 @@ type SetUploadCutoffer interface {
|
||||||
SetUploadCutoff(fs.SizeSuffix) (fs.SizeSuffix, error)
|
SetUploadCutoff(fs.SizeSuffix) (fs.SizeSuffix, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SetCopyCutoffer is a test only interface to change the copy cutoff size at runtime
|
||||||
|
type SetCopyCutoffer interface {
|
||||||
|
// Change the configured CopyCutoff.
|
||||||
|
// Will only be called while no transfer is in progress.
|
||||||
|
SetCopyCutoff(fs.SizeSuffix) (fs.SizeSuffix, error)
|
||||||
|
}
|
||||||
|
|
||||||
// NextPowerOfTwo returns the current or next bigger power of two.
|
// NextPowerOfTwo returns the current or next bigger power of two.
|
||||||
// All values less or equal 0 will return 0
|
// All values less or equal 0 will return 0
|
||||||
func NextPowerOfTwo(i fs.SizeSuffix) fs.SizeSuffix {
|
func NextPowerOfTwo(i fs.SizeSuffix) fs.SizeSuffix {
|
||||||
|
@ -2096,6 +2103,85 @@ func Run(t *testing.T, opt *Opt) {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// Copy files with chunked copy if available
|
||||||
|
t.Run("FsCopyChunked", func(t *testing.T) {
|
||||||
|
skipIfNotOk(t)
|
||||||
|
if testing.Short() {
|
||||||
|
t.Skip("not running with -short")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check have Copy
|
||||||
|
doCopy := f.Features().Copy
|
||||||
|
if doCopy == nil {
|
||||||
|
t.Skip("FS has no Copier interface")
|
||||||
|
}
|
||||||
|
|
||||||
|
if opt.ChunkedUpload.Skip {
|
||||||
|
t.Skip("skipping as ChunkedUpload.Skip is set")
|
||||||
|
}
|
||||||
|
|
||||||
|
do, _ := f.(SetCopyCutoffer)
|
||||||
|
if do == nil {
|
||||||
|
t.Skipf("%T does not implement SetCopyCutoff", f)
|
||||||
|
}
|
||||||
|
|
||||||
|
minChunkSize := opt.ChunkedUpload.MinChunkSize
|
||||||
|
if minChunkSize < 100 {
|
||||||
|
minChunkSize = 100
|
||||||
|
}
|
||||||
|
if opt.ChunkedUpload.CeilChunkSize != nil {
|
||||||
|
minChunkSize = opt.ChunkedUpload.CeilChunkSize(minChunkSize)
|
||||||
|
}
|
||||||
|
|
||||||
|
chunkSizes := fs.SizeSuffixList{
|
||||||
|
minChunkSize,
|
||||||
|
minChunkSize + 1,
|
||||||
|
2*minChunkSize - 1,
|
||||||
|
2 * minChunkSize,
|
||||||
|
2*minChunkSize + 1,
|
||||||
|
}
|
||||||
|
for _, chunkSize := range chunkSizes {
|
||||||
|
t.Run(fmt.Sprintf("%d", chunkSize), func(t *testing.T) {
|
||||||
|
contents := random.String(int(chunkSize))
|
||||||
|
item := fstest.NewItem("chunked-copy", contents, fstest.Time("2001-05-06T04:05:06.499999999Z"))
|
||||||
|
src := PutTestContents(ctx, t, f, &item, contents, true)
|
||||||
|
defer func() {
|
||||||
|
assert.NoError(t, src.Remove(ctx))
|
||||||
|
}()
|
||||||
|
|
||||||
|
var itemCopy = item
|
||||||
|
itemCopy.Path += ".copy"
|
||||||
|
|
||||||
|
// Set copy cutoff to mininum value so we make chunks
|
||||||
|
origCutoff, err := do.SetCopyCutoff(minChunkSize)
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer func() {
|
||||||
|
_, err = do.SetCopyCutoff(origCutoff)
|
||||||
|
require.NoError(t, err)
|
||||||
|
}()
|
||||||
|
|
||||||
|
// Do the copy
|
||||||
|
dst, err := doCopy(ctx, src, itemCopy.Path)
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer func() {
|
||||||
|
assert.NoError(t, dst.Remove(ctx))
|
||||||
|
}()
|
||||||
|
|
||||||
|
// Check size
|
||||||
|
assert.Equal(t, src.Size(), dst.Size())
|
||||||
|
|
||||||
|
// Check modtime
|
||||||
|
srcModTime := src.ModTime(ctx)
|
||||||
|
dstModTime := dst.ModTime(ctx)
|
||||||
|
assert.True(t, srcModTime.Equal(dstModTime))
|
||||||
|
|
||||||
|
// Make sure contents are correct
|
||||||
|
gotContents := ReadObject(ctx, t, dst, -1)
|
||||||
|
assert.Equal(t, contents, gotContents)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
// TestFsUploadUnknownSize ensures Fs.Put() and Object.Update() don't panic when
|
// TestFsUploadUnknownSize ensures Fs.Put() and Object.Update() don't panic when
|
||||||
// src.Size() == -1
|
// src.Size() == -1
|
||||||
//
|
//
|
||||||
|
|
Loading…
Reference in a new issue