forked from TrueCloudLab/restic
ea59896bd6
Fixes restic#719 If the option is passed, restic will wait the specified duration of time and retry locking the repo every 10 seconds (or more often if the total timeout is relatively small). - Play nice with json output - Reduce wait time in lock tests - Rework timeout last attempt - Reduce test wait time to 0.1s - Use exponential back off for the retry lock - Don't pass gopts to lockRepo functions - Use global variable for retry sleep setup - Exit retry lock on cancel - Better wording for flag help - Reorder debug statement - Refactor tests - Lower max sleep time to 1m - Test that we cancel/timeout in time - Use non blocking sleep function - Refactor into minDuration func Co-authored-by: Julian Brost <julian@0x4a42.net>
175 lines
4.6 KiB
Go
175 lines
4.6 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
|
|
"github.com/restic/restic/internal/debug"
|
|
"github.com/restic/restic/internal/dump"
|
|
"github.com/restic/restic/internal/errors"
|
|
"github.com/restic/restic/internal/restic"
|
|
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
var cmdDump = &cobra.Command{
|
|
Use: "dump [flags] snapshotID file",
|
|
Short: "Print a backed-up file to stdout",
|
|
Long: `
|
|
The "dump" command extracts files from a snapshot from the repository. If a
|
|
single file is selected, it prints its contents to stdout. Folders are output
|
|
as a tar (default) or zip file containing the contents of the specified folder.
|
|
Pass "/" as file name to dump the whole snapshot as an archive file.
|
|
|
|
The special snapshot "latest" can be used to use the latest snapshot in the
|
|
repository.
|
|
|
|
EXIT STATUS
|
|
===========
|
|
|
|
Exit status is 0 if the command was successful, and non-zero if there was any error.
|
|
`,
|
|
DisableAutoGenTag: true,
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
return runDump(cmd.Context(), dumpOptions, globalOptions, args)
|
|
},
|
|
}
|
|
|
|
// DumpOptions collects all options for the dump command.
|
|
type DumpOptions struct {
|
|
restic.SnapshotFilter
|
|
Archive string
|
|
}
|
|
|
|
var dumpOptions DumpOptions
|
|
|
|
func init() {
|
|
cmdRoot.AddCommand(cmdDump)
|
|
|
|
flags := cmdDump.Flags()
|
|
initSingleSnapshotFilter(flags, &dumpOptions.SnapshotFilter)
|
|
flags.StringVarP(&dumpOptions.Archive, "archive", "a", "tar", "set archive `format` as \"tar\" or \"zip\"")
|
|
}
|
|
|
|
func splitPath(p string) []string {
|
|
d, f := path.Split(p)
|
|
if d == "" || d == "/" {
|
|
return []string{f}
|
|
}
|
|
s := splitPath(path.Join("/", d))
|
|
return append(s, f)
|
|
}
|
|
|
|
func printFromTree(ctx context.Context, tree *restic.Tree, repo restic.Repository, prefix string, pathComponents []string, d *dump.Dumper) error {
|
|
// If we print / we need to assume that there are multiple nodes at that
|
|
// level in the tree.
|
|
if pathComponents[0] == "" {
|
|
if err := checkStdoutArchive(); err != nil {
|
|
return err
|
|
}
|
|
return d.DumpTree(ctx, tree, "/")
|
|
}
|
|
|
|
item := filepath.Join(prefix, pathComponents[0])
|
|
l := len(pathComponents)
|
|
for _, node := range tree.Nodes {
|
|
// If dumping something in the highest level it will just take the
|
|
// first item it finds and dump that according to the switch case below.
|
|
if node.Name == pathComponents[0] {
|
|
switch {
|
|
case l == 1 && dump.IsFile(node):
|
|
return d.WriteNode(ctx, node)
|
|
case l > 1 && dump.IsDir(node):
|
|
subtree, err := restic.LoadTree(ctx, repo, *node.Subtree)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "cannot load subtree for %q", item)
|
|
}
|
|
return printFromTree(ctx, subtree, repo, item, pathComponents[1:], d)
|
|
case dump.IsDir(node):
|
|
if err := checkStdoutArchive(); err != nil {
|
|
return err
|
|
}
|
|
subtree, err := restic.LoadTree(ctx, repo, *node.Subtree)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return d.DumpTree(ctx, subtree, item)
|
|
case l > 1:
|
|
return fmt.Errorf("%q should be a dir, but is a %q", item, node.Type)
|
|
case !dump.IsFile(node):
|
|
return fmt.Errorf("%q should be a file, but is a %q", item, node.Type)
|
|
}
|
|
}
|
|
}
|
|
return fmt.Errorf("path %q not found in snapshot", item)
|
|
}
|
|
|
|
func runDump(ctx context.Context, opts DumpOptions, gopts GlobalOptions, args []string) error {
|
|
if len(args) != 2 {
|
|
return errors.Fatal("no file and no snapshot ID specified")
|
|
}
|
|
|
|
switch opts.Archive {
|
|
case "tar", "zip":
|
|
default:
|
|
return fmt.Errorf("unknown archive format %q", opts.Archive)
|
|
}
|
|
|
|
snapshotIDString := args[0]
|
|
pathToPrint := args[1]
|
|
|
|
debug.Log("dump file %q from %q", pathToPrint, snapshotIDString)
|
|
|
|
splittedPath := splitPath(path.Clean(pathToPrint))
|
|
|
|
repo, err := OpenRepository(ctx, gopts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if !gopts.NoLock {
|
|
var lock *restic.Lock
|
|
lock, ctx, err = lockRepo(ctx, repo, gopts.RetryLock, gopts.JSON)
|
|
defer unlockRepo(lock)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
sn, err := (&restic.SnapshotFilter{
|
|
Hosts: opts.Hosts,
|
|
Paths: opts.Paths,
|
|
Tags: opts.Tags,
|
|
}).FindLatest(ctx, repo.Backend(), repo, snapshotIDString)
|
|
if err != nil {
|
|
return errors.Fatalf("failed to find snapshot: %v", err)
|
|
}
|
|
|
|
err = repo.LoadIndex(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
tree, err := restic.LoadTree(ctx, repo, *sn.Tree)
|
|
if err != nil {
|
|
return errors.Fatalf("loading tree for snapshot %q failed: %v", snapshotIDString, err)
|
|
}
|
|
|
|
d := dump.New(opts.Archive, repo, os.Stdout)
|
|
err = printFromTree(ctx, tree, repo, "/", splittedPath, d)
|
|
if err != nil {
|
|
return errors.Fatalf("cannot dump file: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func checkStdoutArchive() error {
|
|
if stdoutIsTerminal() {
|
|
return fmt.Errorf("stdout is the terminal, please redirect output")
|
|
}
|
|
return nil
|
|
}
|