2014-04-27 22:00:15 +00:00
package main
import (
2016-04-01 11:50:45 +00:00
"bufio"
2018-05-01 12:40:52 +00:00
"bytes"
2018-04-22 09:57:20 +00:00
"context"
2018-05-01 12:40:52 +00:00
"io/ioutil"
2014-09-23 20:39:12 +00:00
"os"
2018-04-22 09:57:20 +00:00
"strconv"
2016-04-16 20:04:29 +00:00
"strings"
"time"
2016-09-17 10:36:05 +00:00
"github.com/spf13/cobra"
2018-04-22 09:57:20 +00:00
tomb "gopkg.in/tomb.v2"
2015-04-25 17:20:41 +00:00
2017-07-23 12:21:03 +00:00
"github.com/restic/restic/internal/archiver"
"github.com/restic/restic/internal/debug"
"github.com/restic/restic/internal/errors"
"github.com/restic/restic/internal/fs"
2018-04-22 09:57:20 +00:00
"github.com/restic/restic/internal/repository"
2017-07-24 15:42:25 +00:00
"github.com/restic/restic/internal/restic"
2018-05-01 12:40:52 +00:00
"github.com/restic/restic/internal/textfile"
2018-04-22 09:57:20 +00:00
"github.com/restic/restic/internal/ui"
"github.com/restic/restic/internal/ui/termstatus"
2016-09-17 10:36:05 +00:00
)
2015-04-25 17:20:41 +00:00
2016-09-17 10:36:05 +00:00
var cmdBackup = & cobra . Command {
Use : "backup [flags] FILE/DIR [FILE/DIR] ..." ,
2017-09-11 16:32:44 +00:00
Short : "Create a new backup of files and/or directories" ,
2016-09-17 10:36:05 +00:00
Long : `
The "backup" command creates a new snapshot and saves the files and directories
given as the arguments .
` ,
2017-08-05 09:54:59 +00:00
PreRun : func ( cmd * cobra . Command , args [ ] string ) {
if backupOptions . Hostname == "" {
hostname , err := os . Hostname ( )
if err != nil {
debug . Log ( "os.Hostname() returned err: %v" , err )
return
}
backupOptions . Hostname = hostname
}
} ,
2017-08-06 19:02:16 +00:00
DisableAutoGenTag : true ,
2016-09-17 10:36:05 +00:00
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
2017-02-06 11:43:44 +00:00
if backupOptions . Stdin && backupOptions . FilesFrom == "-" {
return errors . Fatal ( "cannot use both `--stdin` and `--files-from -`" )
}
2018-04-22 09:57:20 +00:00
var t tomb . Tomb
2018-05-02 19:24:18 +00:00
term := termstatus . New ( globalOptions . stdout , globalOptions . stderr , globalOptions . Quiet )
2018-04-22 09:57:20 +00:00
t . Go ( func ( ) error { term . Run ( t . Context ( globalOptions . ctx ) ) ; return nil } )
2015-04-25 17:20:41 +00:00
2018-04-22 09:57:20 +00:00
err := runBackup ( backupOptions , globalOptions , term , args )
if err != nil {
return err
}
t . Kill ( nil )
return t . Wait ( )
2016-09-17 10:36:05 +00:00
} ,
2015-04-25 17:20:41 +00:00
}
2016-09-17 10:36:05 +00:00
// BackupOptions bundles all options for the backup command.
type BackupOptions struct {
2017-08-19 20:44:18 +00:00
Parent string
Force bool
Excludes [ ] string
ExcludeFiles [ ] string
ExcludeOtherFS bool
2017-09-09 19:24:29 +00:00
ExcludeIfPresent [ ] string
2017-08-19 20:44:18 +00:00
ExcludeCaches bool
Stdin bool
StdinFilename string
Tags [ ] string
Hostname string
FilesFrom string
2017-08-29 16:29:46 +00:00
TimeStamp string
2017-11-28 20:31:35 +00:00
WithAtime bool
2015-04-25 17:20:41 +00:00
}
2016-09-17 10:36:05 +00:00
var backupOptions BackupOptions
2015-01-04 17:23:00 +00:00
2016-09-17 10:36:05 +00:00
func init ( ) {
cmdRoot . AddCommand ( cmdBackup )
f := cmdBackup . Flags ( )
f . StringVar ( & backupOptions . Parent , "parent" , "" , "use this parent snapshot (default: last snapshot in the repo that has the same target files/directories)" )
2017-02-13 15:02:47 +00:00
f . BoolVarP ( & backupOptions . Force , "force" , "f" , false , ` force re-reading the target files/directories (overrides the "parent" flag) ` )
2017-07-07 01:19:06 +00:00
f . StringArrayVarP ( & backupOptions . Excludes , "exclude" , "e" , nil , "exclude a `pattern` (can be specified multiple times)" )
f . StringArrayVar ( & backupOptions . ExcludeFiles , "exclude-file" , nil , "read exclude patterns from a `file` (can be specified multiple times)" )
2017-02-13 15:02:47 +00:00
f . BoolVarP ( & backupOptions . ExcludeOtherFS , "one-file-system" , "x" , false , "exclude other file systems" )
2017-09-09 19:24:29 +00:00
f . StringArrayVar ( & backupOptions . ExcludeIfPresent , "exclude-if-present" , nil , "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)" )
2017-08-19 20:44:18 +00:00
f . BoolVar ( & backupOptions . ExcludeCaches , "exclude-caches" , false , ` excludes cache directories that are marked with a CACHEDIR.TAG file ` )
2016-09-17 10:36:05 +00:00
f . BoolVar ( & backupOptions . Stdin , "stdin" , false , "read backup from stdin" )
2017-02-08 21:37:02 +00:00
f . StringVar ( & backupOptions . StdinFilename , "stdin-filename" , "stdin" , "file name to use when reading from stdin" )
2017-07-07 01:19:06 +00:00
f . StringArrayVar ( & backupOptions . Tags , "tag" , nil , "add a `tag` for the new snapshot (can be specified multiple times)" )
2017-09-22 12:29:04 +00:00
f . StringVar ( & backupOptions . Hostname , "hostname" , "" , "set the `hostname` for the snapshot manually. To prevent an expensive rescan use the \"parent\" flag" )
2016-11-03 17:48:07 +00:00
f . StringVar ( & backupOptions . FilesFrom , "files-from" , "" , "read the files to backup from file (can be combined with file args)" )
2017-08-29 16:29:46 +00:00
f . StringVar ( & backupOptions . TimeStamp , "time" , "" , "time of the backup (ex. '2012-11-01 22:08:41') (default: now)" )
2017-11-28 20:31:35 +00:00
f . BoolVar ( & backupOptions . WithAtime , "with-atime" , false , "store the atime for all files and directories" )
2014-12-07 15:30:52 +00:00
}
2015-07-06 21:02:16 +00:00
// filterExisting returns a slice of all existing items, or an error if no
// items exist at all.
func filterExisting ( items [ ] string ) ( result [ ] string , err error ) {
for _ , item := range items {
Fix 567 (#570)
* Patch for https://github.com/restic/restic/issues/567
Backup also files on windows with longer pathnames than 255 chars (e.g. from node).
as fd0 says "So, as far as I can see, we need to have custom methods for all functions that accept a path, so that on Windows we can substitute the normal (possibly relative) path used within restic by an (absolute) UNC path, and only then call the underlying functions like os.Stat(), os.Lstat(), os.Open() and so on.
I've already thought about adding a generic abstraction for the file system (so we can mock this easier in tests), and this looks like a good opportunity to build it."
* fixed building tests
* Restructured patches
Add Wrapper for filepath.Walk
* using \\?\ requires absolute pathes to be used.
Now all tests run
* used gofmt on the code
* Restructured Code. No patches dir, integrate the file functions into restic/fs/
There is still an issue, because restic.fs.Open has a different api the os.Open, which returns the result of OpenFile, but takes only a string
* Changed the last os.Open() calls to fs.Open() after extending the File interface
* fixed name-clash of restic.fs and fuse.fs detected by travis
* fixed fmt with gofmt
* c&p failure: removed fixpath() call.
* missing include
* fixed includes in linux variant
* Fix for Linux. Fd() is required on File interface
* done gofmt
2016-08-15 19:59:13 +00:00
_ , err := fs . Lstat ( item )
2016-08-29 17:18:57 +00:00
if err != nil && os . IsNotExist ( errors . Cause ( err ) ) {
2017-09-09 19:12:41 +00:00
Warnf ( "%v does not exist, skipping\n" , item )
2015-07-06 21:02:16 +00:00
continue
}
result = append ( result , item )
}
if len ( result ) == 0 {
2016-09-01 20:17:37 +00:00
return nil , errors . Fatal ( "all target directories/files do not exist" )
2015-07-06 21:02:16 +00:00
}
return
}
2018-04-22 09:57:20 +00:00
// readFromFile will read all lines from the given filename and return them as
// a string array, if filename is empty readFromFile returns and empty string
// array. If filename is a dash (-), readFromFile will read the lines from the
// standard input.
2016-11-03 17:48:07 +00:00
func readLinesFromFile ( filename string ) ( [ ] string , error ) {
if filename == "" {
return nil , nil
}
2018-05-01 12:40:52 +00:00
var (
data [ ] byte
err error
)
if filename == "-" {
data , err = ioutil . ReadAll ( os . Stdin )
} else {
data , err = textfile . Read ( filename )
}
if err != nil {
return nil , err
2016-11-03 17:48:07 +00:00
}
var lines [ ] string
2018-05-01 12:40:52 +00:00
scanner := bufio . NewScanner ( bytes . NewReader ( data ) )
2016-11-03 17:48:07 +00:00
for scanner . Scan ( ) {
2018-02-11 19:56:11 +00:00
line := strings . TrimSpace ( scanner . Text ( ) )
2017-10-19 14:48:22 +00:00
// ignore empty lines
2017-02-27 18:42:00 +00:00
if line == "" {
continue
}
2017-10-19 14:52:06 +00:00
// strip comments
if strings . HasPrefix ( line , "#" ) {
2017-10-19 14:48:22 +00:00
continue
}
2017-02-27 18:42:00 +00:00
lines = append ( lines , line )
2016-11-03 17:48:07 +00:00
}
if err := scanner . Err ( ) ; err != nil {
return nil , err
}
return lines , nil
}
2018-04-22 09:57:20 +00:00
// Check returns an error when an invalid combination of options was set.
func ( opts BackupOptions ) Check ( gopts GlobalOptions , args [ ] string ) error {
2017-07-24 21:15:31 +00:00
if opts . FilesFrom == "-" && 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" )
2017-02-06 11:43:44 +00:00
}
2018-04-22 09:57:20 +00:00
if opts . Stdin {
if opts . FilesFrom != "" {
return errors . Fatal ( "--stdin and --files-from cannot be used together" )
2015-03-02 13:48:47 +00:00
}
2018-04-22 09:57:20 +00:00
if len ( args ) > 0 {
return errors . Fatal ( "--stdin was specified and files/dirs were listed as arguments" )
}
2015-07-06 21:02:16 +00:00
}
2018-04-22 09:57:20 +00:00
return nil
}
2017-09-10 13:13:40 +00:00
2018-04-22 09:57:20 +00:00
// collectRejectFuncs returns a list of all functions which may reject data
// from being saved in a snapshot
func collectRejectFuncs ( opts BackupOptions , repo * repository . Repository , targets [ ] string ) ( fs [ ] RejectFunc , err error ) {
2016-09-18 15:10:33 +00:00
// allowed devices
2016-09-17 10:36:05 +00:00
if opts . ExcludeOtherFS {
2018-04-22 09:57:20 +00:00
f , err := rejectByDevice ( targets )
2017-09-10 13:13:40 +00:00
if err != nil {
2018-04-22 09:57:20 +00:00
return nil , err
2017-09-10 13:13:40 +00:00
}
2018-04-22 09:57:20 +00:00
fs = append ( fs , f )
}
// exclude restic cache
if repo . Cache != nil {
f , err := rejectResticCache ( repo )
if err != nil {
return nil , err
}
fs = append ( fs , f )
2017-09-10 13:13:40 +00:00
}
// add patterns from file
if len ( opts . ExcludeFiles ) > 0 {
opts . Excludes = append ( opts . Excludes , readExcludePatternsFromFiles ( opts . ExcludeFiles ) ... )
}
if len ( opts . Excludes ) > 0 {
2018-04-22 09:57:20 +00:00
fs = append ( fs , rejectByPattern ( opts . Excludes ) )
2017-09-10 13:13:40 +00:00
}
if opts . ExcludeCaches {
opts . ExcludeIfPresent = append ( opts . ExcludeIfPresent , "CACHEDIR.TAG:Signature: 8a477f597d28d172789f06886806bc55" )
}
for _ , spec := range opts . ExcludeIfPresent {
2017-11-27 16:30:53 +00:00
f , err := rejectIfPresent ( spec )
2016-09-18 15:10:33 +00:00
if err != nil {
2018-04-22 09:57:20 +00:00
return nil , err
2016-09-18 15:10:33 +00:00
}
2017-09-10 13:13:40 +00:00
2018-04-22 09:57:20 +00:00
fs = append ( fs , f )
2016-09-18 15:10:33 +00:00
}
2018-04-22 09:57:20 +00:00
return fs , nil
}
2014-04-27 22:00:15 +00:00
2018-04-22 09:57:20 +00:00
// readExcludePatternsFromFiles reads all exclude files and returns the list of
// exclude patterns.
func readExcludePatternsFromFiles ( excludeFiles [ ] string ) [ ] string {
var excludes [ ] string
for _ , filename := range excludeFiles {
err := func ( ) ( err error ) {
2018-05-01 12:40:52 +00:00
data , err := textfile . Read ( filename )
2018-04-22 09:57:20 +00:00
if err != nil {
return err
}
2015-06-27 12:40:18 +00:00
2018-05-01 12:40:52 +00:00
scanner := bufio . NewScanner ( bytes . NewReader ( data ) )
2018-04-22 09:57:20 +00:00
for scanner . Scan ( ) {
line := strings . TrimSpace ( scanner . Text ( ) )
// ignore empty lines
if line == "" {
continue
}
// strip comments
if strings . HasPrefix ( line , "#" ) {
continue
}
line = os . ExpandEnv ( line )
excludes = append ( excludes , line )
}
return scanner . Err ( )
} ( )
2017-09-11 19:37:10 +00:00
if err != nil {
2018-04-22 09:57:20 +00:00
Warnf ( "error reading exclude patterns: %v:" , err )
return nil
2017-09-11 19:37:10 +00:00
}
2018-04-22 09:57:20 +00:00
}
return excludes
}
2017-09-11 19:37:10 +00:00
2018-04-22 09:57:20 +00:00
// collectTargets returns a list of target files/dirs from several sources.
func collectTargets ( opts BackupOptions , args [ ] string ) ( targets [ ] string , err error ) {
if opts . Stdin {
return nil , nil
2017-09-11 19:37:10 +00:00
}
2018-04-22 09:57:20 +00:00
fromfile , err := readLinesFromFile ( opts . FilesFrom )
2015-04-26 15:44:38 +00:00
if err != nil {
2018-04-22 09:57:20 +00:00
return nil , err
}
// merge files from files-from into normal args so we can reuse the normal
// args checks and have the ability to use both files-from and args at the
// same time
args = append ( args , fromfile ... )
if len ( args ) == 0 && ! opts . Stdin {
return nil , errors . Fatal ( "nothing to backup, please specify target files/dirs" )
2015-04-26 15:44:38 +00:00
}
2018-04-22 09:57:20 +00:00
targets = args
targets , err = filterExisting ( targets )
if err != nil {
return nil , err
}
return targets , nil
}
2014-11-30 21:34:21 +00:00
2018-04-22 09:57:20 +00:00
// parent returns the ID of the parent snapshot. If there is none, nil is
// returned.
func findParentSnapshot ( ctx context . Context , repo restic . Repository , opts BackupOptions , targets [ ] string ) ( parentID * restic . ID , err error ) {
2015-04-03 19:18:09 +00:00
// Force using a parent
2016-09-17 10:36:05 +00:00
if ! opts . Force && opts . Parent != "" {
id , err := restic . FindSnapshot ( repo , opts . Parent )
2014-11-30 21:34:21 +00:00
if err != nil {
2018-04-22 09:57:20 +00:00
return nil , errors . Fatalf ( "invalid id %q: %v" , opts . Parent , err )
2014-11-30 21:34:21 +00:00
}
2018-04-22 09:57:20 +00:00
parentID = & id
2014-11-30 21:34:21 +00:00
}
2014-04-27 22:00:15 +00:00
2015-04-03 19:18:09 +00:00
// Find last snapshot to set it as parent, if not already set
2018-04-22 09:57:20 +00:00
if ! opts . Force && parentID == nil {
id , err := restic . FindLatestSnapshot ( ctx , repo , targets , [ ] restic . TagList { } , opts . Hostname )
2015-07-25 15:05:45 +00:00
if err == nil {
2018-04-22 09:57:20 +00:00
parentID = & id
2016-04-27 16:36:48 +00:00
} else if err != restic . ErrNoSnapshotFound {
2018-04-22 09:57:20 +00:00
return nil , err
2015-04-03 19:18:09 +00:00
}
}
2018-04-22 09:57:20 +00:00
return parentID , nil
}
func runBackup ( opts BackupOptions , gopts GlobalOptions , term * termstatus . Terminal , args [ ] string ) error {
err := opts . Check ( gopts , args )
if err != nil {
return err
2015-07-25 16:09:38 +00:00
}
2018-04-22 09:57:20 +00:00
targets , err := collectTargets ( opts , args )
if err != nil {
return err
}
2014-11-16 20:29:11 +00:00
2018-06-01 16:41:44 +00:00
timeStamp := time . Now ( )
if opts . TimeStamp != "" {
timeStamp , err = time . Parse ( TimeFormat , opts . TimeStamp )
if err != nil {
return errors . Fatalf ( "error in time option: %v\n" , err )
}
}
2018-04-22 09:57:20 +00:00
var t tomb . Tomb
p := ui . NewBackup ( term , gopts . verbosity )
// use the terminal for stdout/stderr
prevStdout , prevStderr := gopts . stdout , gopts . stderr
defer func ( ) {
gopts . stdout , gopts . stderr = prevStdout , prevStderr
} ( )
gopts . stdout , gopts . stderr = p . Stdout ( ) , p . Stderr ( )
if s , ok := os . LookupEnv ( "RESTIC_PROGRESS_FPS" ) ; ok {
fps , err := strconv . Atoi ( s )
if err == nil && fps >= 1 {
if fps > 60 {
fps = 60
2017-09-09 19:24:29 +00:00
}
2018-04-22 09:57:20 +00:00
p . MinUpdatePause = time . Second / time . Duration ( fps )
2017-08-19 20:44:18 +00:00
}
2015-07-19 22:13:39 +00:00
}
2014-11-16 20:29:11 +00:00
2018-04-22 09:57:20 +00:00
t . Go ( func ( ) error { return p . Run ( t . Context ( gopts . ctx ) ) } )
p . V ( "open repository" )
repo , err := OpenRepository ( gopts )
if err != nil {
return err
}
p . V ( "lock repository" )
lock , err := lockRepo ( repo )
defer unlockRepo ( lock )
if err != nil {
return err
}
// rejectFuncs collect functions that can reject items from the backup
rejectFuncs , err := collectRejectFuncs ( opts , repo , targets )
if err != nil {
return err
2015-07-19 22:13:39 +00:00
}
2014-11-16 20:29:11 +00:00
2018-04-22 09:57:20 +00:00
p . V ( "load index files" )
err = repo . LoadIndex ( gopts . ctx )
if err != nil {
return err
}
2014-11-23 11:05:43 +00:00
2018-04-22 09:57:20 +00:00
parentSnapshotID , err := findParentSnapshot ( gopts . ctx , repo , opts , targets )
if err != nil {
return err
}
if parentSnapshotID != nil {
p . V ( "using parent snapshot %v\n" , parentSnapshotID . Str ( ) )
}
selectFilter := func ( item string , fi os . FileInfo ) bool {
for _ , reject := range rejectFuncs {
if reject ( item , fi ) {
return false
}
}
return true
2014-11-16 20:29:11 +00:00
}
2018-04-22 09:57:20 +00:00
var targetFS fs . FS = fs . Local { }
if opts . Stdin {
p . V ( "read data from stdin" )
targetFS = & fs . Reader {
ModTime : timeStamp ,
Name : opts . StdinFilename ,
Mode : 0644 ,
ReadCloser : os . Stdin ,
}
targets = [ ] string { opts . StdinFilename }
2014-11-16 20:29:11 +00:00
}
2018-04-22 09:57:20 +00:00
sc := archiver . NewScanner ( targetFS )
sc . Select = selectFilter
sc . Error = p . ScannerError
sc . Result = p . ReportTotal
2014-04-27 22:00:15 +00:00
2018-05-01 20:02:48 +00:00
p . V ( "start scan on %v" , targets )
2018-04-22 09:57:20 +00:00
t . Go ( func ( ) error { return sc . Scan ( t . Context ( gopts . ctx ) , targets ) } )
2017-08-01 17:20:09 +00:00
2018-04-22 09:57:20 +00:00
arch := archiver . New ( repo , targetFS , archiver . Options { } )
arch . Select = selectFilter
arch . WithAtime = opts . WithAtime
arch . Error = p . Error
arch . CompleteItem = p . CompleteItemFn
arch . StartFile = p . StartFile
arch . CompleteBlob = p . CompleteBlob
2017-08-01 17:20:09 +00:00
2018-04-22 09:57:20 +00:00
if parentSnapshotID == nil {
parentSnapshotID = & restic . ID { }
}
2017-08-01 17:20:09 +00:00
2018-04-22 09:57:20 +00:00
snapshotOpts := archiver . SnapshotOptions {
Excludes : opts . Excludes ,
Tags : opts . Tags ,
Time : timeStamp ,
Hostname : opts . Hostname ,
ParentSnapshot : * parentSnapshotID ,
}
2017-08-01 17:20:09 +00:00
2018-04-22 09:57:20 +00:00
uploader := archiver . IndexUploader {
Repository : repo ,
Start : func ( ) {
p . VV ( "uploading intermediate index" )
} ,
Complete : func ( id restic . ID ) {
p . V ( "uploaded intermediate index %v" , id . Str ( ) )
} ,
}
2017-08-01 17:20:09 +00:00
2018-04-22 09:57:20 +00:00
t . Go ( func ( ) error {
return uploader . Upload ( gopts . ctx , t . Context ( gopts . ctx ) , 30 * time . Second )
} )
2018-05-01 20:02:48 +00:00
p . V ( "start backup on %v" , targets )
2018-04-22 09:57:20 +00:00
_ , id , err := arch . Snapshot ( gopts . ctx , targets , snapshotOpts )
if err != nil {
2018-05-12 19:40:31 +00:00
return errors . Fatalf ( "unable to save snapshot: %v" , err )
2017-08-01 17:20:09 +00:00
}
2018-04-22 09:57:20 +00:00
p . Finish ( )
p . P ( "snapshot %s saved\n" , id . Str ( ) )
// cleanly shutdown all running goroutines
t . Kill ( nil )
// let's see if one returned an error
err = t . Wait ( )
if err != nil {
return err
}
return nil
2017-08-01 17:20:09 +00:00
}