Allow --files-from multiple times

Fixes https://github.com/restic/restic/issues/2085
This commit is contained in:
Joe Groocock 2018-11-12 16:54:37 +00:00 committed by Alexander Neumann
parent c8fc72364a
commit 09e9b74cbd
4 changed files with 43 additions and 23 deletions

View 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

View file

@ -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

View file

@ -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

View file

@ -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