2
2
mirror of https://github.com/octoleo/restic.git synced 2025-01-11 10:18:10 +00:00

Fix s3 backend, add more tests

This commit is contained in:
Alexander Neumann 2016-01-02 14:38:45 +01:00
parent 0237b0d972
commit 5722ccfcda
3 changed files with 118 additions and 40 deletions

View File

@ -5,9 +5,12 @@ import (
"fmt" "fmt"
"io" "io"
"io/ioutil" "io/ioutil"
"math/rand"
"sort" "sort"
"testing" "testing"
crand "crypto/rand"
"github.com/restic/restic/backend" "github.com/restic/restic/backend"
. "github.com/restic/restic/test" . "github.com/restic/restic/test"
) )
@ -37,6 +40,70 @@ func testBackendConfig(b backend.Backend, t *testing.T) {
} }
} }
func testGetReader(b backend.Backend, t testing.TB) {
length := rand.Intn(1<<23) + 2000
data := make([]byte, length)
_, err := io.ReadFull(crand.Reader, data)
OK(t, err)
blob, err := b.Create()
OK(t, err)
id := backend.Hash(data)
_, err = blob.Write([]byte(data))
OK(t, err)
OK(t, blob.Finalize(backend.Data, id.String()))
for i := 0; i < 500; i++ {
l := rand.Intn(length + 2000)
o := rand.Intn(length + 2000)
d := data
if o < len(d) {
d = d[o:]
} else {
o = len(d)
d = d[:0]
}
if l > 0 && l < len(d) {
d = d[:l]
}
rd, err := b.GetReader(backend.Data, id.String(), uint(o), uint(l))
OK(t, err)
buf, err := ioutil.ReadAll(rd)
OK(t, err)
if !bytes.Equal(buf, d) {
t.Fatalf("data not equal")
}
}
OK(t, b.Remove(backend.Data, id.String()))
}
func store(t testing.TB, b backend.Backend, tpe backend.Type, data []byte) {
id := backend.Hash(data)
blob, err := b.Create()
OK(t, err)
_, err = blob.Write([]byte(data))
OK(t, err)
OK(t, blob.Finalize(tpe, id.String()))
}
func read(t testing.TB, rd io.Reader, expectedData []byte) {
buf, err := ioutil.ReadAll(rd)
OK(t, err)
if expectedData != nil {
Equals(t, expectedData, buf)
}
}
func testBackend(b backend.Backend, t *testing.T) { func testBackend(b backend.Backend, t *testing.T) {
testBackendConfig(b, t) testBackendConfig(b, t)
@ -70,41 +137,34 @@ func testBackend(b backend.Backend, t *testing.T) {
// add files // add files
for _, test := range TestStrings { for _, test := range TestStrings {
// store string in backend store(t, b, tpe, []byte(test.data))
blob, err := b.Create()
OK(t, err)
_, err = blob.Write([]byte(test.data)) // test Get()
OK(t, err)
OK(t, blob.Finalize(tpe, test.id))
// try to get it out again
rd, err := b.Get(tpe, test.id) rd, err := b.Get(tpe, test.id)
OK(t, err) OK(t, err)
Assert(t, rd != nil, "Get() returned nil") Assert(t, rd != nil, "Get() returned nil")
// try to read it out again read(t, rd, []byte(test.data))
reader, err := b.GetReader(tpe, test.id, 0, uint(len(test.data))) OK(t, rd.Close())
// test GetReader()
rd, err = b.GetReader(tpe, test.id, 0, uint(len(test.data)))
OK(t, err) OK(t, err)
Assert(t, reader != nil, "GetReader() returned nil") Assert(t, rd != nil, "GetReader() returned nil")
bytes := make([]byte, len(test.data))
reader.Read(bytes) read(t, rd, []byte(test.data))
Assert(t, test.data == string(bytes), "Read() returned different content") OK(t, rd.Close())
// try to read it out with an offset and a length // try to read it out with an offset and a length
readerOffLen, err := b.GetReader(tpe, test.id, 1, uint(len(test.data)-2)) start := 1
end := len(test.data) - 2
length := end - start
rd, err = b.GetReader(tpe, test.id, uint(start), uint(length))
OK(t, err) OK(t, err)
Assert(t, readerOffLen != nil, "GetReader() returned nil") Assert(t, rd != nil, "GetReader() returned nil")
bytesOffLen := make([]byte, len(test.data)-2)
readerOffLen.Read(bytesOffLen)
Assert(t, test.data[1:len(test.data)-1] == string(bytesOffLen), "Read() with offset and length returned different content")
buf, err := ioutil.ReadAll(rd) read(t, rd, []byte(test.data[start:end]))
OK(t, err) OK(t, rd.Close())
Equals(t, test.data, string(buf))
// compare content
Equals(t, test.data, string(buf))
} }
// test adding the first file again // test adding the first file again
@ -161,7 +221,6 @@ func testBackend(b backend.Backend, t *testing.T) {
found, err := b.Test(tpe, id.String()) found, err := b.Test(tpe, id.String())
OK(t, err) OK(t, err)
Assert(t, found, fmt.Sprintf("id %q was not found before removal", id))
OK(t, b.Remove(tpe, id.String())) OK(t, b.Remove(tpe, id.String()))
@ -170,6 +229,7 @@ func testBackend(b backend.Backend, t *testing.T) {
Assert(t, !found, fmt.Sprintf("id %q not found after removal", id)) Assert(t, !found, fmt.Sprintf("id %q not found after removal", id))
} }
} }
} }
testGetReader(b, t)
} }

View File

@ -3,6 +3,7 @@ package s3
import ( import (
"bytes" "bytes"
"errors" "errors"
"fmt"
"io" "io"
"strings" "strings"
@ -116,14 +117,22 @@ func (bb *s3Blob) Finalize(t backend.Type, name string) error {
return errors.New("key already exists") return errors.New("key already exists")
} }
expectedBytes := bb.buf.Len()
<-bb.b.connChan <-bb.b.connChan
_, err = bb.b.client.PutObject(bb.b.bucketname, path, bb.buf, int64(bb.buf.Len()), "binary/octet-stream") n, err := bb.b.client.PutObject(bb.b.bucketname, path, bb.buf, int64(bb.buf.Len()), "binary/octet-stream")
bb.b.connChan <- struct{}{} bb.b.connChan <- struct{}{}
bb.buf.Reset()
debug.Log("s3.Finalize", "finalized %v -> err %v", path, err) debug.Log("s3.Finalize", "finalized %v -> n %v, err %v", path, n, err)
if err != nil {
return err
}
return err if n != int64(expectedBytes) {
return errors.New("could not store all bytes")
}
return nil
} }
// Create creates a new Blob. The data is available only after Finalize() // Create creates a new Blob. The data is available only after Finalize()
@ -160,24 +169,26 @@ func (be *S3Backend) GetReader(t backend.Type, name string, offset, length uint)
l, o := int64(length), int64(offset) l, o := int64(length), int64(offset)
if l == 0 { if l == 0 {
l = stat.Size - o l = stat.Size
} }
if l > stat.Size-o { if o > stat.Size {
return nil, fmt.Errorf("offset beyond end of file (%v > %v)", o, stat.Size)
}
if o+l > stat.Size {
l = stat.Size - o l = stat.Size - o
} }
debug.Log("s3.GetReader", "%v %v, o %v l %v", t, name, o, l) debug.Log("s3.GetReader", "%v %v, o %v l %v", t, name, o, l)
buf := make([]byte, l) var r io.Reader
n, err := rd.ReadAt(buf, o) r = &ContinuousReader{R: rd, Offset: o}
debug.Log("s3.GetReader", " -> n %v err %v", n, err) if length > 0 {
if err == io.EOF && int64(n) == l { r = io.LimitReader(r, int64(length))
debug.Log("s3.GetReader", " ignoring EOF error")
err = nil
} }
return backend.ReadCloser(bytes.NewReader(buf[:n])), err return backend.ReadCloser(r), nil
} }
// Test returns true if a blob of the given type and name exists in the backend. // Test returns true if a blob of the given type and name exists in the backend.

7
backend/s3/s3_test.go Normal file
View File

@ -0,0 +1,7 @@
package s3
import "testing"
func TestGetReader(t *testing.T) {
}