2015-11-18 20:20:25 +01:00
|
|
|
package repository
|
|
|
|
|
|
|
|
import (
|
2017-06-04 11:16:55 +02:00
|
|
|
"context"
|
2017-01-22 17:53:00 +01:00
|
|
|
"crypto/sha256"
|
2016-03-06 12:26:25 +01:00
|
|
|
"io"
|
|
|
|
"os"
|
2015-11-18 20:20:25 +01:00
|
|
|
"sync"
|
|
|
|
|
2017-07-23 14:21:03 +02:00
|
|
|
"github.com/restic/restic/internal/errors"
|
|
|
|
"github.com/restic/restic/internal/hashing"
|
2017-07-24 17:42:25 +02:00
|
|
|
"github.com/restic/restic/internal/restic"
|
2017-07-23 14:21:03 +02:00
|
|
|
|
|
|
|
"github.com/restic/restic/internal/crypto"
|
|
|
|
"github.com/restic/restic/internal/debug"
|
|
|
|
"github.com/restic/restic/internal/fs"
|
|
|
|
"github.com/restic/restic/internal/pack"
|
2015-11-18 20:20:25 +01:00
|
|
|
)
|
|
|
|
|
2016-03-05 15:58:39 +01:00
|
|
|
// Saver implements saving data in a backend.
|
|
|
|
type Saver interface {
|
2017-06-04 11:16:55 +02:00
|
|
|
Save(context.Context, restic.Handle, io.Reader) error
|
2016-03-05 15:58:39 +01:00
|
|
|
}
|
|
|
|
|
2017-01-22 17:53:00 +01:00
|
|
|
// Packer holds a pack.Packer together with a hash writer.
|
|
|
|
type Packer struct {
|
|
|
|
*pack.Packer
|
|
|
|
hw *hashing.Writer
|
|
|
|
tmpfile *os.File
|
|
|
|
}
|
|
|
|
|
2015-11-18 20:20:25 +01:00
|
|
|
// packerManager keeps a list of open packs and creates new on demand.
|
|
|
|
type packerManager struct {
|
2017-01-22 17:53:00 +01:00
|
|
|
be Saver
|
|
|
|
key *crypto.Key
|
|
|
|
pm sync.Mutex
|
|
|
|
packers []*Packer
|
2016-03-06 12:26:25 +01:00
|
|
|
|
2016-03-06 13:14:06 +01:00
|
|
|
pool sync.Pool
|
2015-11-18 20:20:25 +01:00
|
|
|
}
|
|
|
|
|
2016-02-22 21:09:21 +01:00
|
|
|
const minPackSize = 4 * 1024 * 1024
|
|
|
|
const maxPackSize = 16 * 1024 * 1024
|
2015-11-18 20:20:25 +01:00
|
|
|
const maxPackers = 200
|
|
|
|
|
2016-03-06 14:20:48 +01:00
|
|
|
// newPackerManager returns an new packer manager which writes temporary files
|
2016-03-06 12:26:25 +01:00
|
|
|
// to a temporary directory
|
2016-03-06 14:20:48 +01:00
|
|
|
func newPackerManager(be Saver, key *crypto.Key) *packerManager {
|
2016-03-06 13:14:06 +01:00
|
|
|
return &packerManager{
|
2016-03-06 12:26:25 +01:00
|
|
|
be: be,
|
|
|
|
key: key,
|
2016-03-06 13:14:06 +01:00
|
|
|
pool: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return make([]byte, (minPackSize+maxPackSize)/2)
|
|
|
|
},
|
|
|
|
},
|
2016-03-06 12:26:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-18 20:20:25 +01:00
|
|
|
// findPacker returns a packer for a new blob of size bytes. Either a new one is
|
|
|
|
// created or one is returned that already has some blobs.
|
2017-01-22 17:53:00 +01:00
|
|
|
func (r *packerManager) findPacker(size uint) (packer *Packer, err error) {
|
2015-11-18 20:20:25 +01:00
|
|
|
r.pm.Lock()
|
|
|
|
defer r.pm.Unlock()
|
|
|
|
|
|
|
|
// search for a suitable packer
|
2017-01-22 17:53:00 +01:00
|
|
|
if len(r.packers) > 0 {
|
2016-09-27 22:35:08 +02:00
|
|
|
debug.Log("searching packer for %d bytes\n", size)
|
2017-01-22 17:53:00 +01:00
|
|
|
for i, p := range r.packers {
|
|
|
|
if p.Packer.Size()+size < maxPackSize {
|
2016-09-27 22:35:08 +02:00
|
|
|
debug.Log("found packer %v", p)
|
2015-11-18 20:20:25 +01:00
|
|
|
// remove from list
|
2017-01-22 17:53:00 +01:00
|
|
|
r.packers = append(r.packers[:i], r.packers[i+1:]...)
|
2015-11-18 20:20:25 +01:00
|
|
|
return p, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// no suitable packer found, return new
|
2016-09-27 22:35:08 +02:00
|
|
|
debug.Log("create new pack for %d bytes", size)
|
2017-05-10 19:48:22 +02:00
|
|
|
tmpfile, err := fs.TempFile("", "restic-temp-pack-")
|
2016-03-06 12:26:25 +01:00
|
|
|
if err != nil {
|
2017-05-10 19:48:22 +02:00
|
|
|
return nil, errors.Wrap(err, "fs.TempFile")
|
2016-03-06 12:26:25 +01:00
|
|
|
}
|
|
|
|
|
2017-01-22 17:53:00 +01:00
|
|
|
hw := hashing.NewWriter(tmpfile, sha256.New())
|
|
|
|
p := pack.NewPacker(r.key, hw)
|
|
|
|
packer = &Packer{
|
|
|
|
Packer: p,
|
|
|
|
hw: hw,
|
|
|
|
tmpfile: tmpfile,
|
|
|
|
}
|
|
|
|
|
|
|
|
return packer, nil
|
2015-11-18 20:20:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// insertPacker appends p to s.packs.
|
2017-01-22 17:53:00 +01:00
|
|
|
func (r *packerManager) insertPacker(p *Packer) {
|
2015-11-18 20:20:25 +01:00
|
|
|
r.pm.Lock()
|
|
|
|
defer r.pm.Unlock()
|
|
|
|
|
2017-01-22 17:53:00 +01:00
|
|
|
r.packers = append(r.packers, p)
|
|
|
|
debug.Log("%d packers\n", len(r.packers))
|
2015-11-18 20:20:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// savePacker stores p in the backend.
|
2017-01-22 17:53:00 +01:00
|
|
|
func (r *Repository) savePacker(p *Packer) error {
|
2017-06-11 10:51:17 +02:00
|
|
|
debug.Log("save packer with %d blobs (%d bytes)\n", p.Packer.Count(), p.Packer.Size())
|
2017-01-22 17:53:00 +01:00
|
|
|
_, err := p.Packer.Finalize()
|
2015-11-18 20:20:25 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2017-01-23 18:45:15 +01:00
|
|
|
_, err = p.tmpfile.Seek(0, 0)
|
2016-03-06 12:26:25 +01:00
|
|
|
if err != nil {
|
2017-01-23 18:45:15 +01:00
|
|
|
return errors.Wrap(err, "Seek")
|
2016-03-06 12:26:25 +01:00
|
|
|
}
|
|
|
|
|
2017-01-22 17:53:00 +01:00
|
|
|
id := restic.IDFromHash(p.hw.Sum(nil))
|
2016-09-01 21:19:30 +02:00
|
|
|
h := restic.Handle{Type: restic.DataFile, Name: id.String()}
|
2016-01-24 19:30:14 +01:00
|
|
|
|
2017-06-04 11:16:55 +02:00
|
|
|
err = r.be.Save(context.TODO(), h, p.tmpfile)
|
2015-11-18 20:20:25 +01:00
|
|
|
if err != nil {
|
2016-09-27 22:35:08 +02:00
|
|
|
debug.Log("Save(%v) error: %v", h, err)
|
2015-11-18 20:20:25 +01:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2016-09-27 22:35:08 +02:00
|
|
|
debug.Log("saved as %v", h)
|
2015-11-18 20:20:25 +01:00
|
|
|
|
2017-01-23 18:45:15 +01:00
|
|
|
err = p.tmpfile.Close()
|
2017-01-22 17:53:00 +01:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "close tempfile")
|
|
|
|
}
|
|
|
|
|
2017-05-10 19:48:22 +02:00
|
|
|
err = fs.RemoveIfExists(p.tmpfile.Name())
|
2016-03-06 13:14:06 +01:00
|
|
|
if err != nil {
|
2016-08-29 22:16:58 +02:00
|
|
|
return errors.Wrap(err, "Remove")
|
2016-03-06 13:14:06 +01:00
|
|
|
}
|
|
|
|
|
2015-11-18 20:20:25 +01:00
|
|
|
// update blobs in the index
|
2017-01-22 17:53:00 +01:00
|
|
|
for _, b := range p.Packer.Blobs() {
|
2016-09-27 22:35:08 +02:00
|
|
|
debug.Log(" updating blob %v to pack %v", b.ID.Str(), id.Str())
|
2017-01-02 14:14:51 +01:00
|
|
|
r.idx.Store(restic.PackedBlob{
|
2016-08-31 22:39:36 +02:00
|
|
|
Blob: restic.Blob{
|
|
|
|
Type: b.Type,
|
|
|
|
ID: b.ID,
|
|
|
|
Offset: b.Offset,
|
|
|
|
Length: uint(b.Length),
|
|
|
|
},
|
2016-01-24 19:30:14 +01:00
|
|
|
PackID: id,
|
2015-11-18 20:20:25 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// countPacker returns the number of open (unfinished) packers.
|
|
|
|
func (r *packerManager) countPacker() int {
|
|
|
|
r.pm.Lock()
|
|
|
|
defer r.pm.Unlock()
|
|
|
|
|
2017-01-22 17:53:00 +01:00
|
|
|
return len(r.packers)
|
2015-11-18 20:20:25 +01:00
|
|
|
}
|