From f71e2b70138021c54af25c2541147ceda1a1a6fe Mon Sep 17 00:00:00 2001 From: Fritz Duchardt Date: Sun, 3 Dec 2023 10:50:47 +0100 Subject: [PATCH] fix: stop deletion of contents if directory paths overlap --- .../some-content/some-file.txt | 0 ...ir.yml => vendir-content-path-overlap.yml} | 0 .../vendir-directory-path-overlap-1.yml | 13 +++++ .../vendir-directory-path-overlap-2.yml | 15 ++++++ examples/overlapping-dir/vendir.lock.yml | 13 +++++ .../overlapping-dir/vendor/dir3/some-file.txt | 0 pkg/vendir/cmd/sync.go | 34 ++++++++++++- pkg/vendir/directory/directory.go | 12 +---- ...o => overlapping_content_path_err_test.go} | 4 +- test/e2e/overlapping_dir_path_test.go | 50 +++++++++++++++++++ 10 files changed, 126 insertions(+), 15 deletions(-) create mode 100644 examples/overlapping-dir/some-content/some-file.txt rename examples/overlapping-dir/{vendir.yml => vendir-content-path-overlap.yml} (100%) create mode 100644 examples/overlapping-dir/vendir-directory-path-overlap-1.yml create mode 100644 examples/overlapping-dir/vendir-directory-path-overlap-2.yml create mode 100644 examples/overlapping-dir/vendir.lock.yml create mode 100644 examples/overlapping-dir/vendor/dir3/some-file.txt rename test/e2e/{overlapping_dir_err_test.go => overlapping_content_path_err_test.go} (69%) create mode 100644 test/e2e/overlapping_dir_path_test.go diff --git a/examples/overlapping-dir/some-content/some-file.txt b/examples/overlapping-dir/some-content/some-file.txt new file mode 100644 index 00000000..e69de29b diff --git a/examples/overlapping-dir/vendir.yml b/examples/overlapping-dir/vendir-content-path-overlap.yml similarity index 100% rename from examples/overlapping-dir/vendir.yml rename to examples/overlapping-dir/vendir-content-path-overlap.yml diff --git a/examples/overlapping-dir/vendir-directory-path-overlap-1.yml b/examples/overlapping-dir/vendir-directory-path-overlap-1.yml new file mode 100644 index 00000000..d9817dfd --- /dev/null +++ b/examples/overlapping-dir/vendir-directory-path-overlap-1.yml @@ -0,0 +1,13 @@ +apiVersion: vendir.k14s.io/v1alpha1 +kind: Config +directories: + - path: vendor + contents: + - path: dir1 + directory: + path: ./some-content + - path: vendor + contents: + - path: dir2 + directory: + path: ./some-content diff --git a/examples/overlapping-dir/vendir-directory-path-overlap-2.yml b/examples/overlapping-dir/vendir-directory-path-overlap-2.yml new file mode 100644 index 00000000..6cf29f51 --- /dev/null +++ b/examples/overlapping-dir/vendir-directory-path-overlap-2.yml @@ -0,0 +1,15 @@ +apiVersion: vendir.k14s.io/v1alpha1 +kind: Config +directories: + - path: vendor + contents: + - path: dir1 + lazy: true + directory: + path: ./some-content + - path: vendor + contents: + - path: dir3 + lazy: true + directory: + path: ./some-content diff --git a/examples/overlapping-dir/vendir.lock.yml b/examples/overlapping-dir/vendir.lock.yml new file mode 100644 index 00000000..91db41d1 --- /dev/null +++ b/examples/overlapping-dir/vendir.lock.yml @@ -0,0 +1,13 @@ +apiVersion: vendir.k14s.io/v1alpha1 +directories: +- contents: + - configDigest: d0aef6c75d30608bf40883a6c52adcea0e8afc62c23513e0462a059390ea9b6b + directory: {} + path: dir1 + path: vendor +- contents: + - configDigest: 147cb489ce06d7fb463ed370dd6c0f41b237af66a1260fecfbd82e5c33bf6c4a + directory: {} + path: dir3 + path: vendor +kind: LockConfig diff --git a/examples/overlapping-dir/vendor/dir3/some-file.txt b/examples/overlapping-dir/vendor/dir3/some-file.txt new file mode 100644 index 00000000..e69de29b diff --git a/pkg/vendir/cmd/sync.go b/pkg/vendir/cmd/sync.go index b2d8d757..0a5cd9e3 100644 --- a/pkg/vendir/cmd/sync.go +++ b/pkg/vendir/cmd/sync.go @@ -140,10 +140,8 @@ func (o *SyncOptions) Run() error { HelmBinary: os.Getenv("VENDIR_HELM_BINARY"), Cache: cache, Lazy: o.Lazy, - Partial: len(dirs) > 0, } newLockConfig := ctlconf.NewLockConfig() - for _, dirConf := range conf.Directories { // error safe to ignore, since lock file might not exist dirExistingLockConf, _ := existingLockConfig.FindDirectory(dirConf.Path) @@ -178,6 +176,38 @@ func (o *SyncOptions) Run() error { } } + // Clean old dirs from final output dir + // Iterate over directory paths + for _, dir := range newLockConfig.Directories { + // Walk through file system dirs and match with lock file config + err := filepath.WalkDir(dir.Path, func(filePath string, info os.DirEntry, err error) error { + if err != nil { + return err + } + for _, dir := range newLockConfig.Directories { + for _, content := range dir.Contents { + // if file system path found in config file then skip + if filepath.Join(dir.Path, content.Path) == filePath { + return filepath.SkipDir + } + // if file system path is a parent directory of config file then continue iterating + if strings.HasPrefix(filepath.Join(dir.Path, content.Path), filePath+"/") { + return nil + } + } + } + // if file system path not found in config file then delete + err = os.RemoveAll(filePath) + if err != nil { + return err + } + return filepath.SkipDir + }) + if err != nil { + return fmt.Errorf("Error while cleaning old directories: %s", err) + } + } + newLockConfigBs, err := newLockConfig.AsBytes() if err != nil { return err diff --git a/pkg/vendir/directory/directory.go b/pkg/vendir/directory/directory.go index 8f580cf5..2afa6058 100644 --- a/pkg/vendir/directory/directory.go +++ b/pkg/vendir/directory/directory.go @@ -42,7 +42,6 @@ type SyncOpts struct { HelmBinary string Cache ctlcache.Cache Lazy bool - Partial bool } func createConfigDigest(contents ctlconf.DirectoryContents) (string, error) { @@ -258,16 +257,7 @@ func (d *Directory) Sync(syncOpts SyncOpts) (ctlconf.LockDirectory, error) { lockConfig.Contents = append(lockConfig.Contents, lockDirContents) - if syncOpts.Partial { - err = stagingDir.PartialRepace(contents.Path, filepath.Join(d.opts.Path, contents.Path)) - if err != nil { - return lockConfig, err - } - } - } - - if !syncOpts.Partial { - err = stagingDir.Replace(d.opts.Path) + err = stagingDir.PartialRepace(contents.Path, filepath.Join(d.opts.Path, contents.Path)) if err != nil { return lockConfig, err } diff --git a/test/e2e/overlapping_dir_err_test.go b/test/e2e/overlapping_content_path_err_test.go similarity index 69% rename from test/e2e/overlapping_dir_err_test.go rename to test/e2e/overlapping_content_path_err_test.go index 16a9fb97..b15cb44c 100644 --- a/test/e2e/overlapping_dir_err_test.go +++ b/test/e2e/overlapping_content_path_err_test.go @@ -8,13 +8,13 @@ import ( "testing" ) -func TestOverlappingDirErr(t *testing.T) { +func TestOverlappingContentPathErr(t *testing.T) { env := BuildEnv(t) vendir := Vendir{t, env.BinaryPath, Logger{}} path := "../../examples/overlapping-dir" - _, err := vendir.RunWithOpts([]string{"sync"}, RunOpts{Dir: path, AllowError: true}) + _, err := vendir.RunWithOpts([]string{"sync", "-f", "vendir-content-path-overlap.yml"}, RunOpts{Dir: path, AllowError: true}) if err == nil { t.Fatalf("Expected err") } diff --git a/test/e2e/overlapping_dir_path_test.go b/test/e2e/overlapping_dir_path_test.go new file mode 100644 index 00000000..3b7eff53 --- /dev/null +++ b/test/e2e/overlapping_dir_path_test.go @@ -0,0 +1,50 @@ +// Copyright 2020 VMware, Inc. +// SPDX-License-Identifier: Apache-2.0 + +package e2e + +import ( + "github.com/stretchr/testify/require" + "os" + "path/filepath" + "testing" +) + +func TestOverlappingDirPath(t *testing.T) { + env := BuildEnv(t) + vendir := Vendir{t, env.BinaryPath, Logger{}} + + path := "../../examples/overlapping-dir" + + _, err := vendir.RunWithOpts([]string{"sync", "-f", "vendir-directory-path-overlap-1.yml"}, RunOpts{Dir: path, AllowError: true}) + require.NoError(t, err) + + expectedOutputDirs := []string{ + "dir1", + "dir2", + } + + filepath.WalkDir(filepath.Join(path, "vendor"), func(path string, d os.DirEntry, err error) error { + if d.Name() == "vendor" { + return nil + } + require.Contains(t, expectedOutputDirs, d.Name()) + return filepath.SkipAll + }) + + _, err = vendir.RunWithOpts([]string{"sync", "-f", "vendir-directory-path-overlap-2.yml"}, RunOpts{Dir: path, AllowError: true}) + require.NoError(t, err) + + expectedOutputDirs = []string{ + "dir1", + "dir3", + } + + filepath.WalkDir(filepath.Join(path, "vendor"), func(path string, d os.DirEntry, err error) error { + if d.Name() == "vendor" { + return nil + } + require.Contains(t, expectedOutputDirs, d.Name()) + return filepath.SkipAll + }) +}