2020-11-10 00:44:03 +00:00
|
|
|
package dump
|
|
|
|
|
|
|
|
import (
|
|
|
|
"archive/zip"
|
|
|
|
"context"
|
|
|
|
"io"
|
|
|
|
"path/filepath"
|
|
|
|
|
|
|
|
"github.com/restic/restic/internal/errors"
|
|
|
|
"github.com/restic/restic/internal/restic"
|
|
|
|
)
|
|
|
|
|
|
|
|
type zipDumper struct {
|
|
|
|
w *zip.Writer
|
|
|
|
}
|
|
|
|
|
|
|
|
// Statically ensure that zipDumper implements dumper.
|
2020-12-18 23:04:17 +00:00
|
|
|
var _ dumper = zipDumper{}
|
2020-11-10 00:44:03 +00:00
|
|
|
|
|
|
|
// WriteZip will write the contents of the given tree, encoded as a zip to the given destination.
|
|
|
|
func WriteZip(ctx context.Context, repo restic.Repository, tree *restic.Tree, rootPath string, dst io.Writer) error {
|
|
|
|
dmp := zipDumper{w: zip.NewWriter(dst)}
|
|
|
|
|
2020-12-18 23:42:46 +00:00
|
|
|
return writeDump(ctx, repo, tree, rootPath, dmp, dst)
|
|
|
|
}
|
2020-11-10 00:44:03 +00:00
|
|
|
|
2020-12-18 23:42:46 +00:00
|
|
|
func (dmp zipDumper) Close() error {
|
2020-11-10 00:44:03 +00:00
|
|
|
return dmp.w.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (dmp zipDumper) dumpNode(ctx context.Context, node *restic.Node, repo restic.Repository) error {
|
|
|
|
relPath, err := filepath.Rel("/", node.Path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
header := &zip.FileHeader{
|
|
|
|
Name: filepath.ToSlash(relPath),
|
|
|
|
UncompressedSize64: node.Size,
|
|
|
|
Modified: node.ModTime,
|
|
|
|
}
|
|
|
|
header.SetMode(node.Mode)
|
|
|
|
|
|
|
|
if IsDir(node) {
|
|
|
|
header.Name += "/"
|
|
|
|
}
|
|
|
|
|
|
|
|
w, err := dmp.w.CreateHeader(header)
|
|
|
|
if err != nil {
|
2020-12-18 23:04:17 +00:00
|
|
|
return errors.Wrap(err, "ZipHeader")
|
2020-11-10 00:44:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if IsLink(node) {
|
|
|
|
if _, err = w.Write([]byte(node.LinkTarget)); err != nil {
|
|
|
|
return errors.Wrap(err, "Write")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return GetNodeData(ctx, w, repo, node)
|
|
|
|
}
|