
Prevents the copy method from calling discard on the writer when the reader is not seekable. Instead, the copy method will discard up to the offset. Truncate is a more expensive operation since any bytes that are truncated already have their hash calculated and are stored on disk in the backend. Re-writing bytes which were truncated requires transfering the data over GRPC again and re-computing the hash up to the point of truncation. Signed-off-by: Derek McGowan <derek@mcgstyle.net>
113 lines
2.4 KiB
Go
113 lines
2.4 KiB
Go
package content
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"io"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/containerd/containerd/errdefs"
|
|
"github.com/opencontainers/go-digest"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
type copySource struct {
|
|
reader io.Reader
|
|
size int64
|
|
digest digest.Digest
|
|
}
|
|
|
|
func TestCopy(t *testing.T) {
|
|
defaultSource := newCopySource("this is the source to copy")
|
|
|
|
var testcases = []struct {
|
|
name string
|
|
source copySource
|
|
writer fakeWriter
|
|
expected string
|
|
}{
|
|
{
|
|
name: "copy no offset",
|
|
source: defaultSource,
|
|
writer: fakeWriter{},
|
|
expected: "this is the source to copy",
|
|
},
|
|
{
|
|
name: "copy with offset from seeker",
|
|
source: defaultSource,
|
|
writer: fakeWriter{status: Status{Offset: 8}},
|
|
expected: "the source to copy",
|
|
},
|
|
{
|
|
name: "copy with offset from unseekable source",
|
|
source: copySource{reader: bytes.NewBufferString("foobar"), size: 6},
|
|
writer: fakeWriter{status: Status{Offset: 3}},
|
|
expected: "bar",
|
|
},
|
|
{
|
|
name: "commit already exists",
|
|
source: defaultSource,
|
|
writer: fakeWriter{commitFunc: func() error {
|
|
return errdefs.ErrAlreadyExists
|
|
}},
|
|
},
|
|
}
|
|
|
|
for _, testcase := range testcases {
|
|
t.Run(testcase.name, func(t *testing.T) {
|
|
err := Copy(context.Background(),
|
|
&testcase.writer,
|
|
testcase.source.reader,
|
|
testcase.source.size,
|
|
testcase.source.digest)
|
|
|
|
require.NoError(t, err)
|
|
assert.Equal(t, testcase.source.digest, testcase.writer.commitedDigest)
|
|
assert.Equal(t, testcase.expected, testcase.writer.String())
|
|
})
|
|
}
|
|
}
|
|
|
|
func newCopySource(raw string) copySource {
|
|
return copySource{
|
|
reader: strings.NewReader(raw),
|
|
size: int64(len(raw)),
|
|
digest: digest.FromBytes([]byte(raw)),
|
|
}
|
|
}
|
|
|
|
type fakeWriter struct {
|
|
bytes.Buffer
|
|
commitedDigest digest.Digest
|
|
status Status
|
|
commitFunc func() error
|
|
}
|
|
|
|
func (f *fakeWriter) Close() error {
|
|
f.Buffer.Reset()
|
|
return nil
|
|
}
|
|
|
|
func (f *fakeWriter) Commit(ctx context.Context, size int64, expected digest.Digest, opts ...Opt) error {
|
|
f.commitedDigest = expected
|
|
if f.commitFunc == nil {
|
|
return nil
|
|
}
|
|
return f.commitFunc()
|
|
}
|
|
|
|
func (f *fakeWriter) Digest() digest.Digest {
|
|
return f.commitedDigest
|
|
}
|
|
|
|
func (f *fakeWriter) Status() (Status, error) {
|
|
return f.status, nil
|
|
}
|
|
|
|
func (f *fakeWriter) Truncate(size int64) error {
|
|
f.Buffer.Truncate(int(size))
|
|
return nil
|
|
}
|