diff --git a/.changes/unreleased/BUG FIXES-20240115-101358.yaml b/.changes/unreleased/BUG FIXES-20240115-101358.yaml new file mode 100644 index 00000000..9571d833 --- /dev/null +++ b/.changes/unreleased/BUG FIXES-20240115-101358.yaml @@ -0,0 +1,7 @@ +kind: BUG FIXES +body: 'data-source/archive_file: Prevent error when generating archive from source + containing symbolically linked directories, and `exclude_symlink_directories` + is set to true' +time: 2024-01-15T10:13:58.177253Z +custom: + Issue: "298" diff --git a/.changes/unreleased/BUG FIXES-20240115-101510.yaml b/.changes/unreleased/BUG FIXES-20240115-101510.yaml new file mode 100644 index 00000000..96259e8b --- /dev/null +++ b/.changes/unreleased/BUG FIXES-20240115-101510.yaml @@ -0,0 +1,6 @@ +kind: BUG FIXES +body: 'resource/archive_file: Prevent error when generating archive from source containing + symbolically linked directories, and `exclude_symlink_directories` is set to true' +time: 2024-01-15T10:15:10.869072Z +custom: + Issue: "298" diff --git a/.changes/unreleased/BUG FIXES-20240117-101851.yaml b/.changes/unreleased/BUG FIXES-20240117-101851.yaml new file mode 100644 index 00000000..99e4f247 --- /dev/null +++ b/.changes/unreleased/BUG FIXES-20240117-101851.yaml @@ -0,0 +1,5 @@ +kind: BUG FIXES +body: 'resource/archive_file: Return error when generated archive would be empty' +time: 2024-01-17T10:18:51.941981Z +custom: + Issue: "298" diff --git a/.changes/unreleased/BUG FIXES-20240117-101923.yaml b/.changes/unreleased/BUG FIXES-20240117-101923.yaml new file mode 100644 index 00000000..c51c97e1 --- /dev/null +++ b/.changes/unreleased/BUG FIXES-20240117-101923.yaml @@ -0,0 +1,5 @@ +kind: BUG FIXES +body: 'data-source/archive_file: Return error when generated archive would be empty' +time: 2024-01-17T10:19:23.907477Z +custom: + Issue: "298" diff --git a/internal/provider/data_source_archive_file_test.go b/internal/provider/data_source_archive_file_test.go index 8ec32870..3a78a59d 100644 --- a/internal/provider/data_source_archive_file_test.go +++ b/internal/provider/data_source_archive_file_test.go @@ -960,8 +960,8 @@ func TestAccArchiveFile_SymlinkFile_Absolute_ExcludeSymlinkDirectories(t *testin }) } -// TestAccArchiveFile_SymlinkDirectory_Relative_ExcludeSymlinkDirectories verifies that an error is generated when -// trying to use a symlink to a directory. +// TestAccArchiveFile_SymlinkDirectory_Relative_ExcludeSymlinkDirectories verifies that an empty archive +// is generated when trying to archive a directory which only contains a symlink to a directory. func TestAccArchiveFile_SymlinkDirectory_Relative_ExcludeSymlinkDirectories(t *testing.T) { td := t.TempDir() @@ -980,14 +980,14 @@ func TestAccArchiveFile_SymlinkDirectory_Relative_ExcludeSymlinkDirectories(t *t exclude_symlink_directories = true } `, filepath.ToSlash("test-fixtures/test-symlink-dir"), filepath.ToSlash(f)), - ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: error reading file for`), + ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: archive has not been\ncreated as it would be empty`), }, }, }) } -// TestAccArchiveFile_SymlinkDirectory_Absolute_ExcludeSymlinkDirectories verifies that an error is generated when -// trying to use a symlink to a directory. +// TestAccArchiveFile_SymlinkDirectory_Absolute_ExcludeSymlinkDirectories verifies that an empty archive +// is generated when trying to archive a directory which only contains a symlink to a directory. func TestAccArchiveFile_SymlinkDirectory_Absolute_ExcludeSymlinkDirectories(t *testing.T) { td := t.TempDir() @@ -1011,7 +1011,7 @@ func TestAccArchiveFile_SymlinkDirectory_Absolute_ExcludeSymlinkDirectories(t *t exclude_symlink_directories = true } `, filepath.ToSlash(symlinkDirWithRegularFilesAbs), filepath.ToSlash(f)), - ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: error reading file for`), + ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: archive has not been\ncreated as it would be empty`), }, }, }) @@ -1181,8 +1181,8 @@ func TestAccArchiveFile_SymlinkDirectoryWithSymlinkFile_Absolute_ExcludeSymlinkD }) } -// TestAccArchiveFile_DirectoryWithSymlinkDirectory_Relative_ExcludeSymlinkDirectories verifies that an error is -// generated when trying to a directory which contains a symlink to a directory. +// TestAccArchiveFile_DirectoryWithSymlinkDirectory_Relative_ExcludeSymlinkDirectories verifies that an empty archive +// is generated when trying to archive a directory which only contains a symlink to a directory. func TestAccArchiveFile_DirectoryWithSymlinkDirectory_Relative_ExcludeSymlinkDirectories(t *testing.T) { td := t.TempDir() @@ -1197,18 +1197,17 @@ func TestAccArchiveFile_DirectoryWithSymlinkDirectory_Relative_ExcludeSymlinkDir type = "zip" source_dir = "%s" output_path = "%s" - output_file_mode = "0666" exclude_symlink_directories = true } `, filepath.ToSlash("test-fixtures/test-dir-with-symlink-dir"), filepath.ToSlash(f)), - ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: error reading file for`), + ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: archive has not been\ncreated as it would be empty`), }, }, }) } -// TestAccArchiveFile_IncludeDirectoryWithSymlinkDirectory_Absolute_ExcludeSymlinkDirectories verifies that an error is -// generated when trying to a directory which contains a symlink to a directory. +// TestAccArchiveFile_IncludeDirectoryWithSymlinkDirectory_Absolute_ExcludeSymlinkDirectories verifies that an empty archive +// is generated when trying to archive a directory which only contains a symlink to a directory. func TestAccArchiveFile_IncludeDirectoryWithSymlinkDirectory_Absolute_ExcludeSymlinkDirectories(t *testing.T) { td := t.TempDir() @@ -1228,23 +1227,24 @@ func TestAccArchiveFile_IncludeDirectoryWithSymlinkDirectory_Absolute_ExcludeSym type = "zip" source_dir = "%s" output_path = "%s" - output_file_mode = "0666" exclude_symlink_directories = true } `, filepath.ToSlash(symlinkDirInRegularDirAbs), filepath.ToSlash(f)), - ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: error reading file for`), + ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: archive has not been\ncreated as it would be empty`), }, }, }) } -// TestAccArchiveFile_Multiple_Relative_ExcludeSymlinkDirectories verifies that an error is -// generated when trying to a directory which contains a symlink to a directory. +// TestAccArchiveFile_Multiple_Relative_ExcludeSymlinkDirectories verifies that +// symlinked directories are excluded. func TestAccArchiveFile_Multiple_Relative_ExcludeSymlinkDirectories(t *testing.T) { td := t.TempDir() f := filepath.Join(td, "zip_file_acc_test.zip") + var fileSize string + r.ParallelTest(t, r.TestCase{ ProtoV5ProviderFactories: protoV5ProviderFactories(), Steps: []r.TestStep{ @@ -1258,14 +1258,32 @@ func TestAccArchiveFile_Multiple_Relative_ExcludeSymlinkDirectories(t *testing.T exclude_symlink_directories = true } `, filepath.ToSlash("test-fixtures"), filepath.ToSlash(f)), - ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: error reading file for`), + Check: r.ComposeTestCheckFunc( + testAccArchiveFileSize(f, &fileSize), + r.TestCheckResourceAttrPtr("data.archive_file.foo", "output_size", &fileSize), + r.TestCheckResourceAttrWith("data.archive_file.foo", "output_path", func(value string) error { + ensureContents(t, value, map[string][]byte{ + "test-dir/test-dir1/file1.txt": []byte("This is file 1"), + "test-dir/test-dir1/file2.txt": []byte("This is file 2"), + "test-dir/test-dir1/file3.txt": []byte("This is file 3"), + "test-dir/test-dir2/file1.txt": []byte("This is file 1"), + "test-dir/test-dir2/file2.txt": []byte("This is file 2"), + "test-dir/test-dir2/file3.txt": []byte("This is file 3"), + "test-dir/test-file.txt": []byte("This is test content"), + "test-dir-with-symlink-file/test-file.txt": []byte("This is test content"), + "test-dir-with-symlink-file/test-symlink.txt": []byte("This is test content"), + }) + ensureFileMode(t, value, "0666") + return nil + }), + ), }, }, }) } -// TestAccArchiveFile_Multiple_Absolute_ExcludeSymlinkDirectories verifies that an error is -// generated when trying to a directory which contains a symlink to a directory. +// TestAccArchiveFile_Multiple_Relative_ExcludeSymlinkDirectories verifies that +// symlinked directories are excluded. func TestAccArchiveFile_Multiple_Absolute_ExcludeSymlinkDirectories(t *testing.T) { td := t.TempDir() @@ -1276,6 +1294,8 @@ func TestAccArchiveFile_Multiple_Absolute_ExcludeSymlinkDirectories(t *testing.T t.Fatal(err) } + var fileSize string + r.ParallelTest(t, r.TestCase{ ProtoV5ProviderFactories: protoV5ProviderFactories(), Steps: []r.TestStep{ @@ -1289,7 +1309,25 @@ func TestAccArchiveFile_Multiple_Absolute_ExcludeSymlinkDirectories(t *testing.T exclude_symlink_directories = true } `, filepath.ToSlash(multipleDirsAndFilesAbs), filepath.ToSlash(f)), - ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: error reading file for`), + Check: r.ComposeTestCheckFunc( + testAccArchiveFileSize(f, &fileSize), + r.TestCheckResourceAttrPtr("data.archive_file.foo", "output_size", &fileSize), + r.TestCheckResourceAttrWith("data.archive_file.foo", "output_path", func(value string) error { + ensureContents(t, value, map[string][]byte{ + "test-dir/test-dir1/file1.txt": []byte("This is file 1"), + "test-dir/test-dir1/file2.txt": []byte("This is file 2"), + "test-dir/test-dir1/file3.txt": []byte("This is file 3"), + "test-dir/test-dir2/file1.txt": []byte("This is file 1"), + "test-dir/test-dir2/file2.txt": []byte("This is file 2"), + "test-dir/test-dir2/file3.txt": []byte("This is file 3"), + "test-dir/test-file.txt": []byte("This is test content"), + "test-dir-with-symlink-file/test-file.txt": []byte("This is test content"), + "test-dir-with-symlink-file/test-symlink.txt": []byte("This is test content"), + }) + ensureFileMode(t, value, "0666") + return nil + }), + ), }, }, }) diff --git a/internal/provider/resource_archive_file_test.go b/internal/provider/resource_archive_file_test.go index 9e9ce08e..84f6c71d 100644 --- a/internal/provider/resource_archive_file_test.go +++ b/internal/provider/resource_archive_file_test.go @@ -1041,6 +1041,379 @@ func TestResource_ArchiveFile_SymlinkFile_Absolute_ExcludeSymlinkDirectories(t * }) } +// TestResource_ArchiveFile_SymlinkDirectory_Relative_ExcludeSymlinkDirectories verifies that an empty archive +// is generated when trying to archive a directory which only contains a symlink to a directory. +func TestResource_ArchiveFile_SymlinkDirectory_Relative_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_dir = "%s" + output_path = "%s" + output_file_mode = "0666" + exclude_symlink_directories = true + } + `, filepath.ToSlash("test-fixtures/test-symlink-dir"), filepath.ToSlash(f)), + ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: archive has not been\ncreated as it would be empty`), + }, + }, + }) +} + +// TestResource_ArchiveFile_SymlinkDirectory_Absolute_ExcludeSymlinkDirectories verifies that an empty archive +// is generated when trying to archive a directory which only contains a symlink to a directory. +func TestResource_ArchiveFile_SymlinkDirectory_Absolute_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + symlinkDirWithRegularFilesAbs, err := filepath.Abs("test-fixtures/test-symlink-dir") + if err != nil { + t.Fatal(err) + } + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_dir = "%s" + output_path = "%s" + output_file_mode = "0666" + exclude_symlink_directories = true + } + `, filepath.ToSlash(symlinkDirWithRegularFilesAbs), filepath.ToSlash(f)), + ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: archive has not been\ncreated as it would be empty`), + }, + }, + }) +} + +// TestResource_ArchiveFile_DirectoryWithSymlinkFile_Relative_ExcludeSymlinkDirectories verifies that a relative path to a +// directory containing a symlink file generates an archive which includes the files in the directory. +func TestResource_ArchiveFile_DirectoryWithSymlinkFile_Relative_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + var fileSize string + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_dir = "%s" + output_path = "%s" + output_file_mode = "0666" + exclude_symlink_directories = true + } + `, filepath.ToSlash("test-fixtures/test-dir-with-symlink-file"), filepath.ToSlash(f)), + Check: r.ComposeTestCheckFunc( + testAccArchiveFileSize(f, &fileSize), + r.TestCheckResourceAttrPtr("archive_file.foo", "output_size", &fileSize), + r.TestCheckResourceAttrWith("archive_file.foo", "output_path", func(value string) error { + ensureContents(t, value, map[string][]byte{ + "test-file.txt": []byte(`This is test content`), + "test-symlink.txt": []byte(`This is test content`), + }) + ensureFileMode(t, value, "0666") + return nil + }), + ), + }, + }, + }) +} + +// TestResource_ArchiveFile_DirectoryWithSymlinkFile_Absolute_ExcludeSymlinkDirectories verifies that an absolute path to a +// directory containing a symlink file generates an archive which includes the files in the directory. +func TestResource_ArchiveFile_DirectoryWithSymlinkFile_Absolute_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + symlinkDirWithSymlinkFilesAbs, err := filepath.Abs("test-fixtures/test-dir-with-symlink-file") + if err != nil { + t.Fatal(err) + } + + var fileSize string + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_dir = "%s" + output_path = "%s" + output_file_mode = "0666" + exclude_symlink_directories = true + } + `, filepath.ToSlash(symlinkDirWithSymlinkFilesAbs), filepath.ToSlash(f)), + Check: r.ComposeTestCheckFunc( + testAccArchiveFileSize(f, &fileSize), + r.TestCheckResourceAttrPtr("archive_file.foo", "output_size", &fileSize), + r.TestCheckResourceAttrWith("archive_file.foo", "output_path", func(value string) error { + ensureContents(t, value, map[string][]byte{ + "test-file.txt": []byte(`This is test content`), + "test-symlink.txt": []byte(`This is test content`), + }) + ensureFileMode(t, value, "0666") + return nil + }), + ), + }, + }, + }) +} + +// TestResource_ArchiveFile_SymlinkDirectoryWithSymlinkFile_Relative_ExcludeSymlinkDirectories verifies that a relative path +// to a symlink file in a symlink directory generates an archive which includes the files in the directory. +func TestResource_ArchiveFile_SymlinkDirectoryWithSymlinkFile_Relative_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + var fileSize string + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_file = "%s" + output_path = "%s" + output_file_mode = "0666" + exclude_symlink_directories = true + } + `, filepath.ToSlash("test-fixtures/test-symlink-dir-with-symlink-file/test-symlink.txt"), filepath.ToSlash(f)), + Check: r.ComposeTestCheckFunc( + testAccArchiveFileSize(f, &fileSize), + r.TestCheckResourceAttrPtr("archive_file.foo", "output_size", &fileSize), + r.TestCheckResourceAttrWith("archive_file.foo", "output_path", func(value string) error { + ensureContents(t, value, map[string][]byte{ + "test-symlink.txt": []byte(`This is test content`), + }) + ensureFileMode(t, value, "0666") + return nil + }), + ), + }, + }, + }) +} + +// TestResource_ArchiveFile_SymlinkDirectoryWithSymlinkFile_Absolute_ExcludeSymlinkDirectories verifies that an absolute path +// to a symlink file in a symlink directory generates an archive which includes the files in the directory. +func TestResource_ArchiveFile_SymlinkDirectoryWithSymlinkFile_Absolute_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + symlinkFileInSymlinkDirAbs, err := filepath.Abs("test-fixtures/test-symlink-dir-with-symlink-file/test-symlink.txt") + if err != nil { + t.Fatal(err) + } + + var fileSize string + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_file = "%s" + output_path = "%s" + output_file_mode = "0666" + exclude_symlink_directories = true + } + `, filepath.ToSlash(symlinkFileInSymlinkDirAbs), filepath.ToSlash(f)), + Check: r.ComposeTestCheckFunc( + testAccArchiveFileSize(f, &fileSize), + r.TestCheckResourceAttrPtr("archive_file.foo", "output_size", &fileSize), + r.TestCheckResourceAttrWith("archive_file.foo", "output_path", func(value string) error { + ensureContents(t, value, map[string][]byte{ + "test-symlink.txt": []byte(`This is test content`), + }) + ensureFileMode(t, value, "0666") + return nil + }), + ), + }, + }, + }) +} + +// TestResource_ArchiveFile_DirectoryWithSymlinkDirectory_Relative_ExcludeSymlinkDirectories verifies that an empty archive +// is generated when trying to archive a directory which only contains a symlink to a directory. +func TestResource_ArchiveFile_DirectoryWithSymlinkDirectory_Relative_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_dir = "%s" + output_path = "%s" + exclude_symlink_directories = true + } + `, filepath.ToSlash("test-fixtures/test-dir-with-symlink-dir"), filepath.ToSlash(f)), + ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: archive has not been\ncreated as it would be empty`), + }, + }, + }) +} + +// TestResource_ArchiveFile_IncludeDirectoryWithSymlinkDirectory_Absolute_ExcludeSymlinkDirectories verifies that an empty archive +// is generated when trying to archive a directory which only contains a symlink to a directory. +func TestResource_ArchiveFile_IncludeDirectoryWithSymlinkDirectory_Absolute_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + symlinkDirInRegularDirAbs, err := filepath.Abs("test-fixtures/test-dir-with-symlink-dir") + if err != nil { + t.Fatal(err) + } + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_dir = "%s" + output_path = "%s" + exclude_symlink_directories = true + } + `, filepath.ToSlash(symlinkDirInRegularDirAbs), filepath.ToSlash(f)), + ExpectError: regexp.MustCompile(`.*error creating archive: error archiving directory: archive has not been\ncreated as it would be empty`), + }, + }, + }) +} + +// TestResource_ArchiveFile_Multiple_Relative_ExcludeSymlinkDirectories verifies that +// symlinked directories are excluded. +func TestResource_ArchiveFile_Multiple_Relative_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + var fileSize string + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_dir = "%s" + output_path = "%s" + output_file_mode = "0666" + exclude_symlink_directories = true + } + `, filepath.ToSlash("test-fixtures"), filepath.ToSlash(f)), + Check: r.ComposeTestCheckFunc( + testAccArchiveFileSize(f, &fileSize), + r.TestCheckResourceAttrPtr("archive_file.foo", "output_size", &fileSize), + r.TestCheckResourceAttrWith("archive_file.foo", "output_path", func(value string) error { + ensureContents(t, value, map[string][]byte{ + "test-dir/test-dir1/file1.txt": []byte("This is file 1"), + "test-dir/test-dir1/file2.txt": []byte("This is file 2"), + "test-dir/test-dir1/file3.txt": []byte("This is file 3"), + "test-dir/test-dir2/file1.txt": []byte("This is file 1"), + "test-dir/test-dir2/file2.txt": []byte("This is file 2"), + "test-dir/test-dir2/file3.txt": []byte("This is file 3"), + "test-dir/test-file.txt": []byte("This is test content"), + "test-dir-with-symlink-file/test-file.txt": []byte("This is test content"), + "test-dir-with-symlink-file/test-symlink.txt": []byte("This is test content"), + }) + ensureFileMode(t, value, "0666") + return nil + }), + ), + }, + }, + }) +} + +// TestResource_ArchiveFile_Multiple_Relative_ExcludeSymlinkDirectories verifies that +// symlinked directories are excluded. +func TestResource_ArchiveFile_Multiple_Absolute_ExcludeSymlinkDirectories(t *testing.T) { + td := t.TempDir() + + f := filepath.Join(td, "zip_file_acc_test.zip") + + multipleDirsAndFilesAbs, err := filepath.Abs("test-fixtures") + if err != nil { + t.Fatal(err) + } + + var fileSize string + + r.ParallelTest(t, r.TestCase{ + ProtoV5ProviderFactories: protoV5ProviderFactories(), + Steps: []r.TestStep{ + { + Config: fmt.Sprintf(` + resource "archive_file" "foo" { + type = "zip" + source_dir = "%s" + output_path = "%s" + output_file_mode = "0666" + exclude_symlink_directories = true + } + `, filepath.ToSlash(multipleDirsAndFilesAbs), filepath.ToSlash(f)), + Check: r.ComposeTestCheckFunc( + testAccArchiveFileSize(f, &fileSize), + r.TestCheckResourceAttrPtr("archive_file.foo", "output_size", &fileSize), + r.TestCheckResourceAttrWith("archive_file.foo", "output_path", func(value string) error { + ensureContents(t, value, map[string][]byte{ + "test-dir/test-dir1/file1.txt": []byte("This is file 1"), + "test-dir/test-dir1/file2.txt": []byte("This is file 2"), + "test-dir/test-dir1/file3.txt": []byte("This is file 3"), + "test-dir/test-dir2/file1.txt": []byte("This is file 1"), + "test-dir/test-dir2/file2.txt": []byte("This is file 2"), + "test-dir/test-dir2/file3.txt": []byte("This is file 3"), + "test-dir/test-file.txt": []byte("This is test content"), + "test-dir-with-symlink-file/test-file.txt": []byte("This is test content"), + "test-dir-with-symlink-file/test-symlink.txt": []byte("This is test content"), + }) + ensureFileMode(t, value, "0666") + return nil + }), + ), + }, + }, + }) +} + func alterFileContents(content, path string) { f, err := os.Create(path) if err != nil { diff --git a/internal/provider/zip_archiver.go b/internal/provider/zip_archiver.go index 9b149529..d4c00121 100644 --- a/internal/provider/zip_archiver.go +++ b/internal/provider/zip_archiver.go @@ -107,15 +107,28 @@ func (a *ZipArchiver) ArchiveDir(indirname string, opts ArchiveDirOpts) error { opts.Excludes[i] = filepath.FromSlash(opts.Excludes[i]) } + // Determine whether an empty archive would be generated. + isArchiveEmpty := true + + err = filepath.Walk(indirname, a.createWalkFunc("", indirname, opts, &isArchiveEmpty, true)) + if err != nil { + return err + } + + // Return an error if an empty archive would be generated. + if isArchiveEmpty { + return fmt.Errorf("archive has not been created as it would be empty") + } + if err := a.open(); err != nil { return err } defer a.close() - return filepath.Walk(indirname, a.createWalkFunc("", indirname, opts)) + return filepath.Walk(indirname, a.createWalkFunc("", indirname, opts, &isArchiveEmpty, false)) } -func (a *ZipArchiver) createWalkFunc(basePath string, indirname string, opts ArchiveDirOpts) func(path string, info os.FileInfo, err error) error { +func (a *ZipArchiver) createWalkFunc(basePath, indirname string, opts ArchiveDirOpts, isArchiveEmpty *bool, dryRun bool) func(path string, info os.FileInfo, err error) error { return func(path string, info os.FileInfo, err error) error { if err != nil { return fmt.Errorf("error encountered during file walk: %s", err) @@ -146,23 +159,31 @@ func (a *ZipArchiver) createWalkFunc(basePath string, indirname string, opts Arc } if info.Mode()&os.ModeSymlink == os.ModeSymlink { - if !opts.ExcludeSymlinkDirectories { - realPath, err := filepath.EvalSymlinks(path) - if err != nil { - return err - } + realPath, err := filepath.EvalSymlinks(path) + if err != nil { + return err + } - realInfo, err := os.Stat(realPath) - if err != nil { - return err - } + realInfo, err := os.Stat(realPath) + if err != nil { + return err + } - if realInfo.IsDir() { - return filepath.Walk(realPath, a.createWalkFunc(archivePath, realPath, opts)) + if realInfo.IsDir() { + if !opts.ExcludeSymlinkDirectories { + return filepath.Walk(realPath, a.createWalkFunc(archivePath, realPath, opts, isArchiveEmpty, dryRun)) + } else { + return filepath.SkipDir } - - info = realInfo } + + info = realInfo + } + + *isArchiveEmpty = false + + if dryRun { + return nil } fh, err := zip.FileInfoHeader(info) diff --git a/internal/provider/zip_archiver_test.go b/internal/provider/zip_archiver_test.go index 3d45f324..a46de0d8 100644 --- a/internal/provider/zip_archiver_test.go +++ b/internal/provider/zip_archiver_test.go @@ -9,7 +9,6 @@ import ( "io" "os" "path/filepath" - "regexp" "strconv" "testing" "time" @@ -218,11 +217,8 @@ func TestZipArchiver_Dir_ExcludeSymlinkDirectories(t *testing.T) { ExcludeSymlinkDirectories: true, }) - regex := regexp.MustCompile(`error reading file for archival: read test-fixtures(\/|\\)test-dir-with-symlink-dir(\/|\\)test-symlink-dir: `) - found := regex.Match([]byte(err.Error())) - - if !found { - t.Fatalf("expedted error to match %q, got: %s", regex.String(), err.Error()) + if err != nil { + t.Errorf("expected no error: %s", err) } } @@ -270,11 +266,8 @@ func TestZipArchiver_Dir_Exclude_ExcludeSymlinkDirectories(t *testing.T) { ExcludeSymlinkDirectories: true, }) - regex := regexp.MustCompile(`error reading file for archival: read test-fixtures(\/|\\)test-dir-with-symlink-dir(\/|\\)test-symlink-dir: `) - found := regex.Match([]byte(err.Error())) - - if !found { - t.Fatalf("expedted error to match %q, got: %s", regex.String(), err.Error()) + if err != nil { + t.Errorf("expected no error: %s", err) } }