func (cmd CmdRestore) Execute(args []string) error { if len(args) != 1 { return fmt.Errorf("wrong number of arguments, Usage: %s", cmd.Usage()) } if cmd.Target == "" { return errors.New("please specify a directory to restore to (--target)") } if len(cmd.Exclude) > 0 && len(cmd.Include) > 0 { return errors.New("exclude and include patterns are mutually exclusive") } snapshotIDString := args[0] debug.Log("restore", "restore %v to %v", snapshotIDString, cmd.Target) repo, err := cmd.global.OpenRepository() if err != nil { return err } lock, err := lockRepo(repo) defer unlockRepo(lock) if err != nil { return err } err = repo.LoadIndex() if err != nil { return err } id, err := restic.FindSnapshot(repo, snapshotIDString) if err != nil { cmd.global.Exitf(1, "invalid id %q: %v", snapshotIDString, err) } res, err := restic.NewRestorer(repo, id) if err != nil { cmd.global.Exitf(2, "creating restorer failed: %v\n", err) } res.Error = func(dir string, node *restic.Node, err error) error { cmd.global.Warnf("error for %s: %+v\n", dir, err) return err } selectExcludeFilter := func(item string, dstpath string, node *restic.Node) bool { matched, err := filter.List(cmd.Exclude, item) if err != nil { cmd.global.Warnf("error for exclude pattern: %v", err) } return !matched } selectIncludeFilter := func(item string, dstpath string, node *restic.Node) bool { matched, err := filter.List(cmd.Include, item) if err != nil { cmd.global.Warnf("error for include pattern: %v", err) } return matched } if len(cmd.Exclude) > 0 { res.SelectFilter = selectExcludeFilter } else if len(cmd.Include) > 0 { res.SelectFilter = selectIncludeFilter } cmd.global.Verbosef("restoring %s to %s\n", res.Snapshot(), cmd.Target) err = res.RestoreTo(cmd.Target) if err != nil { return err } return nil }
func (cmd CmdRestore) Execute(args []string) error { if len(args) < 2 || len(args) > 3 { return fmt.Errorf("wrong number of arguments, Usage: %s", cmd.Usage()) } repo, err := cmd.global.OpenRepository() if err != nil { return err } lock, err := lockRepo(repo) defer unlockRepo(lock) if err != nil { return err } err = repo.LoadIndex() if err != nil { return err } id, err := restic.FindSnapshot(repo, args[0]) if err != nil { cmd.global.Exitf(1, "invalid id %q: %v", args[0], err) } target := args[1] // create restorer res, err := restic.NewRestorer(repo, id) if err != nil { cmd.global.Exitf(2, "creating restorer failed: %v\n", err) } res.Error = func(dir string, node *restic.Node, err error) error { cmd.global.Warnf("error for %s: %+v\n", dir, err) // if node.Type == "dir" { // if e, ok := err.(*os.PathError); ok { // if errn, ok := e.Err.(syscall.Errno); ok { // if errn == syscall.EEXIST { // fmt.Printf("ignoring already existing directory %s\n", dir) // return nil // } // } // } // } return err } // TODO: a filter against the full path sucks as filepath.Match doesn't match // directory separators on '*'. still, it's better than nothing. if len(args) > 2 { res.Filter = func(item string, dstpath string, node *restic.Node) bool { matched, err := filepath.Match(item, args[2]) if err != nil { panic(err) } return matched } } cmd.global.Verbosef("restoring %s to %s\n", res.Snapshot(), target) err = res.RestoreTo(target) if err != nil { return err } return nil }