Allow --files-from multiple times
Fixes https://github.com/restic/restic/issues/2085
This commit is contained in:
parent
c8fc72364a
commit
09e9b74cbd
4 changed files with 43 additions and 23 deletions
10
changelog/unreleased/pull-2086
Normal file
10
changelog/unreleased/pull-2086
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
Enhancement: Allow --files-from to be specified multiple times
|
||||||
|
|
||||||
|
This change should help with managing backup sources as source paths
|
||||||
|
can be specified in multiple files.
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
restic backup --files-from afile --files-from anotherfile ..
|
||||||
|
|
||||||
|
https://github.com/restic/restic/pull/2086
|
||||||
|
https://github.com/restic/restic/issues/2085
|
|
@ -45,8 +45,12 @@ given as the arguments.
|
||||||
},
|
},
|
||||||
DisableAutoGenTag: true,
|
DisableAutoGenTag: true,
|
||||||
RunE: func(cmd *cobra.Command, args []string) error {
|
RunE: func(cmd *cobra.Command, args []string) error {
|
||||||
if backupOptions.Stdin && backupOptions.FilesFrom == "-" {
|
if backupOptions.Stdin {
|
||||||
return errors.Fatal("cannot use both `--stdin` and `--files-from -`")
|
for _, filename := range backupOptions.FilesFrom {
|
||||||
|
if filename == "-" {
|
||||||
|
return errors.Fatal("cannot use both `--stdin` and `--files-from -`")
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var t tomb.Tomb
|
var t tomb.Tomb
|
||||||
|
@ -75,7 +79,7 @@ type BackupOptions struct {
|
||||||
StdinFilename string
|
StdinFilename string
|
||||||
Tags []string
|
Tags []string
|
||||||
Host string
|
Host string
|
||||||
FilesFrom string
|
FilesFrom []string
|
||||||
TimeStamp string
|
TimeStamp string
|
||||||
WithAtime bool
|
WithAtime bool
|
||||||
}
|
}
|
||||||
|
@ -101,7 +105,7 @@ func init() {
|
||||||
f.StringVar(&backupOptions.Host, "hostname", "", "set the `hostname` for the snapshot manually")
|
f.StringVar(&backupOptions.Host, "hostname", "", "set the `hostname` for the snapshot manually")
|
||||||
f.MarkDeprecated("hostname", "use --host")
|
f.MarkDeprecated("hostname", "use --host")
|
||||||
|
|
||||||
f.StringVar(&backupOptions.FilesFrom, "files-from", "", "read the files to backup from file (can be combined with file args)")
|
f.StringArrayVar(&backupOptions.FilesFrom, "files-from", nil, "read the files to backup from file (can be combined with file args/can be specified multiple times)")
|
||||||
f.StringVar(&backupOptions.TimeStamp, "time", "", "time of the backup (ex. '2012-11-01 22:08:41') (default: now)")
|
f.StringVar(&backupOptions.TimeStamp, "time", "", "time of the backup (ex. '2012-11-01 22:08:41') (default: now)")
|
||||||
f.BoolVar(&backupOptions.WithAtime, "with-atime", false, "store the atime for all files and directories")
|
f.BoolVar(&backupOptions.WithAtime, "with-atime", false, "store the atime for all files and directories")
|
||||||
}
|
}
|
||||||
|
@ -175,12 +179,16 @@ func readLinesFromFile(filename string) ([]string, error) {
|
||||||
|
|
||||||
// Check returns an error when an invalid combination of options was set.
|
// Check returns an error when an invalid combination of options was set.
|
||||||
func (opts BackupOptions) Check(gopts GlobalOptions, args []string) error {
|
func (opts BackupOptions) Check(gopts GlobalOptions, args []string) error {
|
||||||
if opts.FilesFrom == "-" && gopts.password == "" {
|
if gopts.password == "" {
|
||||||
return errors.Fatal("unable to read password from stdin when data is to be read from stdin, use --password-file or $RESTIC_PASSWORD")
|
for _, filename := range opts.FilesFrom {
|
||||||
|
if filename == "-" {
|
||||||
|
return errors.Fatal("unable to read password from stdin when data is to be read from stdin, use --password-file or $RESTIC_PASSWORD")
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if opts.Stdin {
|
if opts.Stdin {
|
||||||
if opts.FilesFrom != "" {
|
if len(opts.FilesFrom) > 0 {
|
||||||
return errors.Fatal("--stdin and --files-from cannot be used together")
|
return errors.Fatal("--stdin and --files-from cannot be used together")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -302,23 +310,25 @@ func collectTargets(opts BackupOptions, args []string) (targets []string, err er
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
fromfile, err := readLinesFromFile(opts.FilesFrom)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// expand wildcards
|
|
||||||
var lines []string
|
var lines []string
|
||||||
for _, line := range fromfile {
|
for _, file := range opts.FilesFrom {
|
||||||
var expanded []string
|
fromfile, err := readLinesFromFile(file)
|
||||||
expanded, err := filepath.Glob(line)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.WithMessage(err, fmt.Sprintf("pattern: %s", line))
|
return nil, err
|
||||||
}
|
}
|
||||||
if len(expanded) == 0 {
|
|
||||||
Warnf("pattern %q does not match any files, skipping\n", line)
|
// expand wildcards
|
||||||
|
for _, line := range fromfile {
|
||||||
|
var expanded []string
|
||||||
|
expanded, err := filepath.Glob(line)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.WithMessage(err, fmt.Sprintf("pattern: %s", line))
|
||||||
|
}
|
||||||
|
if len(expanded) == 0 {
|
||||||
|
Warnf("pattern %q does not match any files, skipping\n", line)
|
||||||
|
}
|
||||||
|
lines = append(lines, expanded...)
|
||||||
}
|
}
|
||||||
lines = append(lines, expanded...)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// merge files from files-from into normal args so we can reuse the normal
|
// merge files from files-from into normal args so we can reuse the normal
|
||||||
|
|
|
@ -205,8 +205,8 @@ backup ``/sys`` or ``/dev`` on a Linux system:
|
||||||
$ restic -r /srv/restic-repo backup --one-file-system /
|
$ restic -r /srv/restic-repo backup --one-file-system /
|
||||||
|
|
||||||
By using the ``--files-from`` option you can read the files you want to
|
By using the ``--files-from`` option you can read the files you want to
|
||||||
backup from a file. This is especially useful if a lot of files have to
|
backup from one or more files. This is especially useful if a lot of files have
|
||||||
be backed up that are not in the same folder or are maybe pre-filtered
|
to be backed up that are not in the same folder or are maybe pre-filtered
|
||||||
by other software.
|
by other software.
|
||||||
|
|
||||||
For example maybe you want to backup files which have a name that matches a
|
For example maybe you want to backup files which have a name that matches a
|
||||||
|
|
|
@ -80,7 +80,7 @@ command:
|
||||||
--exclude-caches excludes cache directories that are marked with a CACHEDIR.TAG file
|
--exclude-caches excludes cache directories that are marked with a CACHEDIR.TAG file
|
||||||
--exclude-file file read exclude patterns from a file (can be specified multiple times)
|
--exclude-file file read exclude patterns from a file (can be specified multiple times)
|
||||||
--exclude-if-present stringArray takes filename[:header], exclude contents of directories containing filename (except filename itself) if header of that file is as provided (can be specified multiple times)
|
--exclude-if-present stringArray takes filename[:header], exclude contents of directories containing filename (except filename itself) if header of that file is as provided (can be specified multiple times)
|
||||||
--files-from string read the files to backup from file (can be combined with file args)
|
--files-from string read the files to backup from file (can be combined with file args/can be specified multiple times)
|
||||||
-f, --force force re-reading the target files/directories (overrides the "parent" flag)
|
-f, --force force re-reading the target files/directories (overrides the "parent" flag)
|
||||||
-h, --help help for backup
|
-h, --help help for backup
|
||||||
--hostname hostname set the hostname for the snapshot manually. To prevent an expensive rescan use the "parent" flag
|
--hostname hostname set the hostname for the snapshot manually. To prevent an expensive rescan use the "parent" flag
|
||||||
|
|
Loading…
Reference in a new issue