forked from TrueCloudLab/restic
Introduce CreateBlob() method for backend
This commit is contained in:
parent
f8f8107d55
commit
35636a9d92
5 changed files with 168 additions and 1 deletions
|
@ -23,6 +23,11 @@ var (
|
||||||
ErrAlreadyPresent = errors.New("blob is already present in backend")
|
ErrAlreadyPresent = errors.New("blob is already present in backend")
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type Blob interface {
|
||||||
|
io.WriteCloser
|
||||||
|
ID() (ID, error)
|
||||||
|
}
|
||||||
|
|
||||||
type Lister interface {
|
type Lister interface {
|
||||||
List(Type) (IDs, error)
|
List(Type) (IDs, error)
|
||||||
}
|
}
|
||||||
|
@ -35,6 +40,7 @@ type Getter interface {
|
||||||
type Creater interface {
|
type Creater interface {
|
||||||
Create(Type, []byte) (ID, error)
|
Create(Type, []byte) (ID, error)
|
||||||
CreateFrom(Type, io.Reader) (ID, error)
|
CreateFrom(Type, io.Reader) (ID, error)
|
||||||
|
CreateBlob(Type) (Blob, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type Tester interface {
|
type Tester interface {
|
||||||
|
|
|
@ -3,6 +3,7 @@ package backend
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"hash"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
|
@ -272,6 +273,79 @@ func (b *Local) CreateFrom(t Type, rd io.Reader) (ID, error) {
|
||||||
return id, nil
|
return id, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type localBlob struct {
|
||||||
|
f *os.File
|
||||||
|
h hash.Hash
|
||||||
|
tw io.Writer
|
||||||
|
backend *Local
|
||||||
|
tpe Type
|
||||||
|
id ID
|
||||||
|
}
|
||||||
|
|
||||||
|
func (lb *localBlob) Close() error {
|
||||||
|
err := lb.f.Close()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// get ID
|
||||||
|
lb.id = ID(lb.h.Sum(nil))
|
||||||
|
|
||||||
|
// check for duplicate ID
|
||||||
|
res, err := lb.backend.Test(lb.tpe, lb.id)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("testing presence of ID %v failed: %v", lb.id, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if res {
|
||||||
|
return ErrAlreadyPresent
|
||||||
|
}
|
||||||
|
|
||||||
|
// rename file
|
||||||
|
err = lb.backend.renameFile(lb.f, lb.tpe, lb.id)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (lb *localBlob) Write(p []byte) (int, error) {
|
||||||
|
return lb.tw.Write(p)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (lb *localBlob) ID() (ID, error) {
|
||||||
|
if lb.id == nil {
|
||||||
|
return nil, errors.New("blob is not closed, ID unavailable")
|
||||||
|
}
|
||||||
|
|
||||||
|
return lb.id, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create creates a new blob of type t. Blob implements io.WriteCloser. Once
|
||||||
|
// Close() has been called, ID() can be used to retrieve the ID. If the blob is
|
||||||
|
// already present, Close() returns ErrAlreadyPresent.
|
||||||
|
func (b *Local) CreateBlob(t Type) (Blob, error) {
|
||||||
|
// TODO: make sure that tempfile is removed upon error
|
||||||
|
|
||||||
|
// create tempfile in backend
|
||||||
|
file, err := b.tempFile()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
h := newHash()
|
||||||
|
blob := localBlob{
|
||||||
|
h: h,
|
||||||
|
tw: io.MultiWriter(h, file),
|
||||||
|
f: file,
|
||||||
|
backend: b,
|
||||||
|
tpe: t,
|
||||||
|
}
|
||||||
|
|
||||||
|
return &blob, nil
|
||||||
|
}
|
||||||
|
|
||||||
// Construct path for given Type and ID.
|
// Construct path for given Type and ID.
|
||||||
func (b *Local) filename(t Type, id ID) string {
|
func (b *Local) filename(t Type, id ID) string {
|
||||||
return filepath.Join(b.dirname(t, id), id.String())
|
return filepath.Join(b.dirname(t, id), id.String())
|
||||||
|
|
|
@ -69,7 +69,14 @@ 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 string in backend
|
||||||
id, err := b.Create(tpe, []byte(test.data))
|
blob, err := b.CreateBlob(tpe)
|
||||||
|
ok(t, err)
|
||||||
|
|
||||||
|
_, err = blob.Write([]byte(test.data))
|
||||||
|
ok(t, err)
|
||||||
|
ok(t, blob.Close())
|
||||||
|
|
||||||
|
id, err := blob.ID()
|
||||||
ok(t, err)
|
ok(t, err)
|
||||||
|
|
||||||
equals(t, test.id, id.String())
|
equals(t, test.id, id.String())
|
||||||
|
|
|
@ -5,6 +5,7 @@ import (
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"hash"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
"log"
|
||||||
|
@ -380,6 +381,81 @@ func (r *SFTP) CreateFrom(t Type, rd io.Reader) (ID, error) {
|
||||||
return id, nil
|
return id, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type sftpBlob struct {
|
||||||
|
f *sftp.File
|
||||||
|
name string
|
||||||
|
h hash.Hash
|
||||||
|
tw io.Writer
|
||||||
|
backend *SFTP
|
||||||
|
tpe Type
|
||||||
|
id ID
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sb *sftpBlob) Close() error {
|
||||||
|
err := sb.f.Close()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// get ID
|
||||||
|
sb.id = ID(sb.h.Sum(nil))
|
||||||
|
|
||||||
|
// check for duplicate ID
|
||||||
|
res, err := sb.backend.Test(sb.tpe, sb.id)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("testing presence of ID %v failed: %v", sb.id, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if res {
|
||||||
|
return ErrAlreadyPresent
|
||||||
|
}
|
||||||
|
|
||||||
|
// rename file
|
||||||
|
err = sb.backend.renameFile(sb.name, sb.tpe, sb.id)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sb *sftpBlob) Write(p []byte) (int, error) {
|
||||||
|
return sb.tw.Write(p)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sb *sftpBlob) ID() (ID, error) {
|
||||||
|
if sb.id == nil {
|
||||||
|
return nil, errors.New("blob is not closed, ID unavailable")
|
||||||
|
}
|
||||||
|
|
||||||
|
return sb.id, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create creates a new blob of type t. Blob implements io.WriteCloser. Once
|
||||||
|
// Close() has been called, ID() can be used to retrieve the ID. If the blob is
|
||||||
|
// already present, Close() returns ErrAlreadyPresent.
|
||||||
|
func (r *SFTP) CreateBlob(t Type) (Blob, error) {
|
||||||
|
// TODO: make sure that tempfile is removed upon error
|
||||||
|
|
||||||
|
// create tempfile in backend
|
||||||
|
filename, file, err := r.tempFile()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
h := newHash()
|
||||||
|
blob := sftpBlob{
|
||||||
|
h: h,
|
||||||
|
tw: io.MultiWriter(h, file),
|
||||||
|
f: file,
|
||||||
|
name: filename,
|
||||||
|
backend: r,
|
||||||
|
tpe: t,
|
||||||
|
}
|
||||||
|
|
||||||
|
return &blob, nil
|
||||||
|
}
|
||||||
|
|
||||||
// Construct path for given Type and ID.
|
// Construct path for given Type and ID.
|
||||||
func (r *SFTP) filename(t Type, id ID) string {
|
func (r *SFTP) filename(t Type, id ID) string {
|
||||||
return filepath.Join(r.dirname(t, id), id.String())
|
return filepath.Join(r.dirname(t, id), id.String())
|
||||||
|
|
|
@ -330,6 +330,10 @@ func (s Server) CreateFrom(t backend.Type, r io.Reader) (backend.ID, error) {
|
||||||
return s.be.CreateFrom(t, r)
|
return s.be.CreateFrom(t, r)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s Server) CreateBlob(t backend.Type) (backend.Blob, error) {
|
||||||
|
return s.be.CreateBlob(t)
|
||||||
|
}
|
||||||
|
|
||||||
func (s Server) Test(t backend.Type, id backend.ID) (bool, error) {
|
func (s Server) Test(t backend.Type, id backend.ID) (bool, error) {
|
||||||
return s.be.Test(t, id)
|
return s.be.Test(t, id)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue