2016-08-31 22:39:36 +02:00
|
|
|
package archiver
|
2016-05-08 21:50:27 +02:00
|
|
|
|
|
|
|
import (
|
2017-06-04 11:16:55 +02:00
|
|
|
"context"
|
2016-05-08 21:50:27 +02:00
|
|
|
"io"
|
|
|
|
"time"
|
|
|
|
|
2017-07-23 14:21:03 +02:00
|
|
|
"github.com/restic/restic/internal/debug"
|
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/errors"
|
2016-09-03 20:11:10 +02:00
|
|
|
|
|
|
|
"github.com/restic/chunker"
|
2016-05-08 21:50:27 +02:00
|
|
|
)
|
|
|
|
|
2017-03-02 15:45:35 +01:00
|
|
|
// Reader allows saving a stream of data to the repository.
|
|
|
|
type Reader struct {
|
|
|
|
restic.Repository
|
|
|
|
|
|
|
|
Tags []string
|
|
|
|
Hostname string
|
|
|
|
}
|
|
|
|
|
|
|
|
// Archive reads data from the reader and saves it to the repo.
|
2017-06-04 11:16:55 +02:00
|
|
|
func (r *Reader) Archive(ctx context.Context, name string, rd io.Reader, p *restic.Progress) (*restic.Snapshot, restic.ID, error) {
|
2017-02-08 22:37:02 +01:00
|
|
|
if name == "" {
|
|
|
|
return nil, restic.ID{}, errors.New("no filename given")
|
|
|
|
}
|
|
|
|
|
2016-09-27 22:35:08 +02:00
|
|
|
debug.Log("start archiving %s", name)
|
2017-08-29 18:29:46 +02:00
|
|
|
sn, err := restic.NewSnapshot([]string{name}, r.Tags, r.Hostname, time.Now())
|
2016-05-08 21:50:27 +02:00
|
|
|
if err != nil {
|
2016-08-31 22:39:36 +02:00
|
|
|
return nil, restic.ID{}, err
|
2016-05-08 21:50:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
p.Start()
|
|
|
|
defer p.Done()
|
|
|
|
|
2017-03-02 15:45:35 +01:00
|
|
|
repo := r.Repository
|
2016-08-31 22:39:36 +02:00
|
|
|
chnker := chunker.New(rd, repo.Config().ChunkerPolynomial)
|
2016-05-08 21:50:27 +02:00
|
|
|
|
2017-02-04 16:38:33 +01:00
|
|
|
ids := restic.IDs{}
|
2016-05-08 22:21:16 +02:00
|
|
|
var fileSize uint64
|
2016-05-08 21:50:27 +02:00
|
|
|
|
|
|
|
for {
|
|
|
|
chunk, err := chnker.Next(getBuf())
|
2016-08-29 19:18:57 +02:00
|
|
|
if errors.Cause(err) == io.EOF {
|
2016-05-08 21:50:27 +02:00
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2016-08-31 22:39:36 +02:00
|
|
|
return nil, restic.ID{}, errors.Wrap(err, "chunker.Next()")
|
2016-05-08 21:50:27 +02:00
|
|
|
}
|
|
|
|
|
2016-08-31 22:39:36 +02:00
|
|
|
id := restic.Hash(chunk.Data)
|
2016-05-08 21:50:27 +02:00
|
|
|
|
2016-08-31 22:39:36 +02:00
|
|
|
if !repo.Index().Has(id, restic.DataBlob) {
|
2017-06-04 11:16:55 +02:00
|
|
|
_, err := repo.SaveBlob(ctx, restic.DataBlob, chunk.Data, id)
|
2016-05-08 21:50:27 +02:00
|
|
|
if err != nil {
|
2016-08-31 22:39:36 +02:00
|
|
|
return nil, restic.ID{}, err
|
2016-05-08 21:50:27 +02:00
|
|
|
}
|
2018-01-25 20:49:41 +01:00
|
|
|
debug.Log("saved blob %v (%d bytes)\n", id, chunk.Length)
|
2016-05-08 21:50:27 +02:00
|
|
|
} else {
|
2018-01-25 20:49:41 +01:00
|
|
|
debug.Log("blob %v already saved in the repo\n", id)
|
2016-05-08 21:50:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
freeBuf(chunk.Data)
|
|
|
|
|
|
|
|
ids = append(ids, id)
|
|
|
|
|
2016-08-31 22:39:36 +02:00
|
|
|
p.Report(restic.Stat{Bytes: uint64(chunk.Length)})
|
2016-05-08 22:21:16 +02:00
|
|
|
fileSize += uint64(chunk.Length)
|
2016-05-08 21:50:27 +02:00
|
|
|
}
|
|
|
|
|
2016-08-31 22:39:36 +02:00
|
|
|
tree := &restic.Tree{
|
|
|
|
Nodes: []*restic.Node{
|
2017-05-17 01:28:39 +02:00
|
|
|
{
|
2016-05-08 21:50:27 +02:00
|
|
|
Name: name,
|
|
|
|
AccessTime: time.Now(),
|
|
|
|
ModTime: time.Now(),
|
2016-09-01 21:20:03 +02:00
|
|
|
Type: "file",
|
2016-05-08 21:50:27 +02:00
|
|
|
Mode: 0644,
|
2016-05-08 22:21:16 +02:00
|
|
|
Size: fileSize,
|
2016-05-08 21:50:27 +02:00
|
|
|
UID: sn.UID,
|
|
|
|
GID: sn.GID,
|
|
|
|
User: sn.Username,
|
|
|
|
Content: ids,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2017-06-04 11:16:55 +02:00
|
|
|
treeID, err := repo.SaveTree(ctx, tree)
|
2016-05-08 21:50:27 +02:00
|
|
|
if err != nil {
|
2016-08-31 22:39:36 +02:00
|
|
|
return nil, restic.ID{}, err
|
2016-05-08 21:50:27 +02:00
|
|
|
}
|
|
|
|
sn.Tree = &treeID
|
2018-01-25 20:49:41 +01:00
|
|
|
debug.Log("tree saved as %v", treeID)
|
2016-05-08 21:50:27 +02:00
|
|
|
|
2017-06-04 11:16:55 +02:00
|
|
|
id, err := repo.SaveJSONUnpacked(ctx, restic.SnapshotFile, sn)
|
2016-05-08 21:50:27 +02:00
|
|
|
if err != nil {
|
2016-08-31 22:39:36 +02:00
|
|
|
return nil, restic.ID{}, err
|
2016-05-08 21:50:27 +02:00
|
|
|
}
|
|
|
|
|
2018-01-25 20:49:41 +01:00
|
|
|
debug.Log("snapshot saved as %v", id)
|
2016-05-08 21:50:27 +02:00
|
|
|
|
2017-11-22 06:27:29 -05:00
|
|
|
err = repo.Flush(ctx)
|
2016-05-08 21:50:27 +02:00
|
|
|
if err != nil {
|
2016-08-31 22:39:36 +02:00
|
|
|
return nil, restic.ID{}, err
|
2016-05-08 21:50:27 +02:00
|
|
|
}
|
|
|
|
|
2017-06-04 11:16:55 +02:00
|
|
|
err = repo.SaveIndex(ctx)
|
2016-05-08 21:50:27 +02:00
|
|
|
if err != nil {
|
2016-08-31 22:39:36 +02:00
|
|
|
return nil, restic.ID{}, err
|
2016-05-08 21:50:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return sn, id, nil
|
|
|
|
}
|