From 7d71bad4eb626c90af16cd6d1d6bddf480eee263 Mon Sep 17 00:00:00 2001 From: Alexander Neumann Date: Sat, 10 Dec 2016 16:36:58 +0100 Subject: [PATCH 1/3] Test if modified files are correctly saved --- src/cmds/restic/integration_test.go | 46 +++++++++++++++++++++++++++++ src/restic/archiver/archiver.go | 2 ++ 2 files changed, 48 insertions(+) diff --git a/src/cmds/restic/integration_test.go b/src/cmds/restic/integration_test.go index 0987a47ad..d67842682 100644 --- a/src/cmds/restic/integration_test.go +++ b/src/cmds/restic/integration_test.go @@ -343,6 +343,52 @@ func TestBackupMissingFile2(t *testing.T) { }) } +func TestBackupChangedFile(t *testing.T) { + withTestEnvironment(t, func(env *testEnvironment, gopts GlobalOptions) { + datafile := filepath.Join("testdata", "backup-data.tar.gz") + fd, err := os.Open(datafile) + if os.IsNotExist(errors.Cause(err)) { + t.Skipf("unable to find data file %q, skipping", datafile) + return + } + OK(t, err) + OK(t, fd.Close()) + + SetupTarTestFixture(t, env.testdata, datafile) + + testRunInit(t, gopts) + + globalOptions.stderr = ioutil.Discard + defer func() { + globalOptions.stderr = os.Stderr + }() + + modFile := filepath.Join(env.testdata, "0", "0", "6", "18") + + ranHook := false + debug.Hook("archiver.SaveFile", func(context interface{}) { + pathname := context.(string) + + if pathname != modFile { + return + } + + t.Logf("in hook, modifying test file %v", modFile) + ranHook = true + + OK(t, ioutil.WriteFile(modFile, []byte("modified"), 0600)) + }) + + opts := BackupOptions{} + + testRunBackup(t, []string{env.testdata}, opts, gopts) + testRunCheck(t, gopts) + + Assert(t, ranHook, "hook did not run") + debug.RemoveHook("archiver.SaveFile") + }) +} + func TestBackupDirectoryError(t *testing.T) { withTestEnvironment(t, func(env *testEnvironment, gopts GlobalOptions) { datafile := filepath.Join("testdata", "backup-data.tar.gz") diff --git a/src/restic/archiver/archiver.go b/src/restic/archiver/archiver.go index 2423dc5aa..5d606176c 100644 --- a/src/restic/archiver/archiver.go +++ b/src/restic/archiver/archiver.go @@ -214,6 +214,8 @@ func (arch *Archiver) SaveFile(p *restic.Progress, node *restic.Node) error { return errors.Wrap(err, "Open") } + debug.RunHook("archiver.SaveFile", node.Path) + node, err = arch.reloadFileIfChanged(node, file) if err != nil { return err From 3fcbb4ac2533139c7f3216cf04b053fb73611d4c Mon Sep 17 00:00:00 2001 From: Alexander Neumann Date: Sat, 10 Dec 2016 16:54:20 +0100 Subject: [PATCH 2/3] Use new Node if file has changed Closes #604 --- src/restic/archiver/archiver.go | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/src/restic/archiver/archiver.go b/src/restic/archiver/archiver.go index 5d606176c..c5a473295 100644 --- a/src/restic/archiver/archiver.go +++ b/src/restic/archiver/archiver.go @@ -207,18 +207,18 @@ func updateNodeContent(node *restic.Node, results []saveResult) error { // SaveFile stores the content of the file on the backend as a Blob by calling // Save for each chunk. -func (arch *Archiver) SaveFile(p *restic.Progress, node *restic.Node) error { +func (arch *Archiver) SaveFile(p *restic.Progress, node *restic.Node) (*restic.Node, error) { file, err := fs.Open(node.Path) defer file.Close() if err != nil { - return errors.Wrap(err, "Open") + return nil, errors.Wrap(err, "Open") } debug.RunHook("archiver.SaveFile", node.Path) node, err = arch.reloadFileIfChanged(node, file) if err != nil { - return err + return nil, err } chnker := chunker.New(file, arch.repo.Config().ChunkerPolynomial) @@ -231,7 +231,7 @@ func (arch *Archiver) SaveFile(p *restic.Progress, node *restic.Node) error { } if err != nil { - return errors.Wrap(err, "chunker.Next") + return nil, errors.Wrap(err, "chunker.Next") } resCh := make(chan saveResult, 1) @@ -241,11 +241,11 @@ func (arch *Archiver) SaveFile(p *restic.Progress, node *restic.Node) error { results, err := waitForResults(resultChannels) if err != nil { - return err + return nil, err } - err = updateNodeContent(node, results) - return err + + return node, err } func (arch *Archiver) fileWorker(wg *sync.WaitGroup, p *restic.Progress, done <-chan struct{}, entCh <-chan pipe.Entry) { @@ -309,7 +309,7 @@ func (arch *Archiver) fileWorker(wg *sync.WaitGroup, p *restic.Progress, done <- // otherwise read file normally if node.Type == "file" && len(node.Content) == 0 { debug.Log(" read and save %v, content: %v", e.Path(), node.Content) - err = arch.SaveFile(p, node) + node, err = arch.SaveFile(p, node) if err != nil { // TODO: integrate error reporting fmt.Fprintf(os.Stderr, "error for %v: %v\n", node.Path, err) From e6a40af06d979cf2ad36f861b002ae3559f8e35f Mon Sep 17 00:00:00 2001 From: Alexander Neumann Date: Sat, 10 Dec 2016 17:14:13 +0100 Subject: [PATCH 3/3] Treat changed files as a warning, not an error --- src/cmds/restic/cmd_backup.go | 5 ++--- src/restic/archiver/archiver.go | 13 ++++++------- 2 files changed, 8 insertions(+), 10 deletions(-) diff --git a/src/cmds/restic/cmd_backup.go b/src/cmds/restic/cmd_backup.go index ecdc23537..aa3f3ccb7 100644 --- a/src/cmds/restic/cmd_backup.go +++ b/src/cmds/restic/cmd_backup.go @@ -431,10 +431,9 @@ func runBackup(opts BackupOptions, gopts GlobalOptions, args []string) error { arch.Excludes = opts.Excludes arch.SelectFilter = selectFilter - arch.Error = func(dir string, fi os.FileInfo, err error) error { + arch.Warn = func(dir string, fi os.FileInfo, err error) { // TODO: make ignoring errors configurable - Warnf("%s\rerror for %s: %v\n", ClearLine(), dir, err) - return nil + Warnf("%s\rwarning for %s: %v\n", ClearLine(), dir, err) } _, id, err := arch.Snapshot(newArchiveProgress(gopts, stat), target, opts.Tags, parentSnapshotID) diff --git a/src/restic/archiver/archiver.go b/src/restic/archiver/archiver.go index c5a473295..b1912f806 100644 --- a/src/restic/archiver/archiver.go +++ b/src/restic/archiver/archiver.go @@ -26,7 +26,9 @@ const ( maxConcurrency = 10 ) -var archiverAbortOnAllErrors = func(str string, fi os.FileInfo, err error) error { return err } +var archiverPrintWarnings = func(path string, fi os.FileInfo, err error) { + fmt.Fprintf(os.Stderr, "warning for %v: %v", path, err) +} var archiverAllowAllFiles = func(string, os.FileInfo) bool { return true } // Archiver is used to backup a set of directories. @@ -39,7 +41,7 @@ type Archiver struct { blobToken chan struct{} - Error func(dir string, fi os.FileInfo, err error) error + Warn func(dir string, fi os.FileInfo, err error) SelectFilter pipe.SelectFunc Excludes []string } @@ -61,7 +63,7 @@ func New(repo restic.Repository) *Archiver { arch.blobToken <- struct{}{} } - arch.Error = archiverAbortOnAllErrors + arch.Warn = archiverPrintWarnings arch.SelectFilter = archiverAllowAllFiles return arch @@ -135,10 +137,7 @@ func (arch *Archiver) reloadFileIfChanged(node *restic.Node, file fs.File) (*res return node, nil } - err = arch.Error(node.Path, fi, errors.New("file has changed")) - if err != nil { - return nil, err - } + arch.Warn(node.Path, fi, errors.New("file has changed")) node, err = restic.NodeFromFileInfo(node.Path, fi) if err != nil {