-
Notifications
You must be signed in to change notification settings - Fork 3.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
111151: roachtest: clean up artifacts zipping code r=RaduBerinde a=RaduBerinde This change rewrites the artifacts zipping code and adds a test. Instead of doing the walk twice (with the second one restricted to top-level entries), we first list and filter the directory entries. This will make it easy to reuse the code to first move go coverage artifacts to their own archive. Epic: none Release note: None Co-authored-by: Radu Berinde <[email protected]>
- Loading branch information
Showing
4 changed files
with
195 additions
and
72 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,98 @@ | ||
// Copyright 2023 The Cockroach Authors. | ||
// | ||
// Use of this software is governed by the Business Source License | ||
// included in the file licenses/BSL.txt. | ||
// | ||
// As of the Change Date specified in that file, in accordance with | ||
// the Business Source License, use of this software will be governed | ||
// by the Apache License, Version 2.0, included in the file | ||
// licenses/APL.txt. | ||
|
||
package main | ||
|
||
import ( | ||
"archive/zip" | ||
"io" | ||
"os" | ||
"path/filepath" | ||
"sort" | ||
) | ||
|
||
// moveToZipArchive creates a zip archive inside rootPath with the files/dirs | ||
// given as relative paths, after which the given files/dirs are deleted. | ||
// | ||
// See filterDirEntries for a convenient way to get a list of files/dirs. | ||
func moveToZipArchive(archiveName string, rootPath string, relPaths ...string) error { | ||
f, err := os.Create(filepath.Join(rootPath, archiveName)) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
z := zip.NewWriter(f) | ||
for _, relPath := range relPaths { | ||
// Walk the given path. | ||
if err := filepath.Walk(filepath.Join(rootPath, relPath), func(path string, info os.FileInfo, err error) error { | ||
if err != nil { | ||
return err | ||
} | ||
if info.IsDir() { | ||
// Let Walk recurse inside. | ||
return nil | ||
} | ||
relPath, err := filepath.Rel(rootPath, path) | ||
if err != nil { | ||
return err | ||
} | ||
w, err := z.Create(relPath) | ||
if err != nil { | ||
return err | ||
} | ||
r, err := os.Open(path) | ||
if err != nil { | ||
return err | ||
} | ||
if _, err := io.Copy(w, r); err != nil { | ||
_ = r.Close() | ||
return err | ||
} | ||
return r.Close() | ||
}); err != nil { | ||
_ = f.Close() | ||
return err | ||
} | ||
} | ||
|
||
if err := z.Close(); err != nil { | ||
_ = f.Close() | ||
return err | ||
} | ||
if err := f.Close(); err != nil { | ||
return err | ||
} | ||
|
||
// Now that the zip file is there, remove all the files that went into it. | ||
for _, relPath := range relPaths { | ||
if err := os.RemoveAll(filepath.Join(rootPath, relPath)); err != nil { | ||
return err | ||
} | ||
} | ||
return nil | ||
} | ||
|
||
// filterDirEntries lists the given directory, runs a filter function on each | ||
// entry, and returns the base names of those which passed the filter. | ||
func filterDirEntries( | ||
path string, filter func(entry os.DirEntry) bool, | ||
) (baseNames []string, _ error) { | ||
entries, err := os.ReadDir(path) | ||
if err != nil { | ||
return nil, err | ||
} | ||
for _, e := range entries { | ||
if filter(e) { | ||
baseNames = append(baseNames, e.Name()) | ||
} | ||
} | ||
sort.Strings(baseNames) | ||
return baseNames, nil | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,84 @@ | ||
// Copyright 2023 The Cockroach Authors. | ||
// | ||
// Use of this software is governed by the Business Source License | ||
// included in the file licenses/BSL.txt. | ||
// | ||
// As of the Change Date specified in that file, in accordance with | ||
// the Business Source License, use of this software will be governed | ||
// by the Apache License, Version 2.0, included in the file | ||
// licenses/APL.txt. | ||
|
||
package main | ||
|
||
import ( | ||
"archive/zip" | ||
"os" | ||
"path/filepath" | ||
"strings" | ||
"testing" | ||
|
||
"github.com/stretchr/testify/require" | ||
) | ||
|
||
func TestMoveToZipArchive(t *testing.T) { | ||
dir := t.TempDir() | ||
p := func(elems ...string) string { | ||
return filepath.Join(append([]string{dir}, elems...)...) | ||
} | ||
require.NoError(t, os.WriteFile(p("a1"), []byte("foo"), 0777)) | ||
require.NoError(t, os.WriteFile(p("a2"), []byte("foo"), 0777)) | ||
require.NoError(t, os.WriteFile(p("a3"), []byte("foo"), 0777)) | ||
require.NoError(t, os.Mkdir(p("dir1"), 0777)) | ||
require.NoError(t, os.WriteFile(p("dir1", "file1"), []byte("foo"), 0777)) | ||
require.NoError(t, os.WriteFile(p("dir1", "file2"), []byte("foo"), 0777)) | ||
require.NoError(t, os.Mkdir(p("dir2"), 0777)) | ||
require.NoError(t, os.WriteFile(p("dir2", "file1"), []byte("foo"), 0777)) | ||
require.NoError(t, os.WriteFile(p("dir2", "file2"), []byte("foo"), 0777)) | ||
|
||
// expectLs checks the current directory listing of dir. | ||
expectLs := func(expected ...string) { | ||
t.Helper() | ||
var actual []string | ||
require.NoError(t, filepath.Walk(dir, func(path string, info os.FileInfo, err error) error { | ||
require.NoError(t, err) | ||
if !info.IsDir() { | ||
rel, err := filepath.Rel(dir, path) | ||
require.NoError(t, err) | ||
actual = append(actual, rel) | ||
} | ||
return nil | ||
})) | ||
require.Equal(t, expected, actual) | ||
} | ||
// expectZip checks the files contained in the given archive; paths must use | ||
// slashes. | ||
expectZip := func(archiveName string, expected ...string) { | ||
r, err := zip.OpenReader(p(archiveName)) | ||
require.NoError(t, err) | ||
var actual []string | ||
for _, f := range r.File { | ||
actual = append(actual, f.Name) | ||
} | ||
require.Equal(t, expected, actual) | ||
require.NoError(t, r.Close()) | ||
} | ||
j := filepath.Join | ||
expectLs("a1", "a2", "a3", j("dir1", "file1"), j("dir1", "file2"), j("dir2", "file1"), j("dir2", "file2")) | ||
|
||
list, err := filterDirEntries(dir, func(entry os.DirEntry) bool { | ||
return entry.Name() != "a2" && entry.Name() != "dir2" | ||
}) | ||
require.NoError(t, err) | ||
require.Equal(t, []string{"a1", "a3", "dir1"}, list) | ||
require.NoError(t, moveToZipArchive("first.zip", dir, list...)) | ||
expectZip("first.zip", "a1", "a3", "dir1/file1", "dir1/file2") | ||
expectLs("a2", j("dir2", "file1"), j("dir2", "file2"), "first.zip") | ||
|
||
list, err = filterDirEntries(dir, func(entry os.DirEntry) bool { | ||
return !strings.HasSuffix(entry.Name(), ".zip") | ||
}) | ||
require.NoError(t, err) | ||
require.NoError(t, moveToZipArchive("second.zip", dir, list...)) | ||
expectZip("second.zip", "a2", "dir2/file1", "dir2/file2") | ||
expectLs("first.zip", "second.zip") | ||
} |