-
Notifications
You must be signed in to change notification settings - Fork 51
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
ci: Parallelize change detector #1871
Changes from 21 commits
605f428
de1a1eb
5b389cb
f6bb25b
dc5aaf8
6ab7371
8195242
01afa55
4b9c36e
9ec616a
3b84d9f
536f6fe
b312303
6039650
f6af08b
eb80d80
97396de
08a1186
fc6cbcd
cfc1182
de3e575
b01d301
3626037
cf6267f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
# Parallel change detector | ||
|
||
This is is not a breaking change. The change detector has been updated to allow for parallel test runs. There were changes to environment variables and test setup that makes the previous version of the change detector incompatible with this version. |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
# Change Detector | ||
|
||
The change detector is used to detect data format changes between versions of DefraDB. | ||
|
||
## How it works | ||
|
||
The tests run using a `source` and `target` branch of DefraDB. Each branch is cloned into a temporary directory and dependencies are installed. | ||
|
||
The test runner executes all of the common test packages available in the `source` and `target` tests directory. | ||
|
||
For each test package execution the following steps occur: | ||
|
||
- Create a temporary data directory. This is used to share data between `source` and `target`. | ||
- Run the `source` version in setup only mode. This creates test fixtures in the shared data directory. | ||
- Run the `target` version in change detector mode. This skips the setup and executes the tests using the shared data directory. |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,205 @@ | ||
// Copyright 2023 Democratized Data Foundation | ||
// | ||
// 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. | ||
|
||
//go:build change_detector | ||
|
||
package change_detector | ||
|
||
import ( | ||
"fmt" | ||
"io/fs" | ||
"os" | ||
"os/exec" | ||
"path" | ||
"path/filepath" | ||
"strings" | ||
"testing" | ||
|
||
"github.com/stretchr/testify/require" | ||
) | ||
|
||
func TestChanges(t *testing.T) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. praise: I really like this approach, it is simpler and neater than some kind of file lock IMO. |
||
sourceRepoDir := t.TempDir() | ||
execClone(t, sourceRepoDir, Repository, SourceBranch) | ||
execMakeDeps(t, sourceRepoDir) | ||
|
||
var targetRepoDir string | ||
if TargetBranch == "" { | ||
// default to the local branch | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. question: This seems odd and a bit dangerous no? If I understand correctly this will now default to testing against itself, which is essentially the same as running This does seem like an easy way to accidentally not run the change detector at all. EDIT: I misunderstood, we now have source and target branches, and the new source branch is what the old target used to be. Ignore this comment :) |
||
out, err := exec.Command("git", "rev-parse", "--show-toplevel").Output() | ||
require.NoError(t, err, string(out)) | ||
targetRepoDir = strings.TrimSpace(string(out)) | ||
} else { | ||
// check out the target branch | ||
targetRepoDir = t.TempDir() | ||
execClone(t, targetRepoDir, Repository, TargetBranch) | ||
execMakeDeps(t, targetRepoDir) | ||
} | ||
|
||
if checkIfDatabaseFormatChangesAreDocumented(t, sourceRepoDir, targetRepoDir) { | ||
t.Skip("skipping test with documented database format changes") | ||
} | ||
|
||
targetRepoTestDir := filepath.Join(targetRepoDir, "tests", "integration") | ||
targetRepoPkgList := execList(t, targetRepoTestDir) | ||
|
||
sourceRepoTestDir := filepath.Join(sourceRepoDir, "tests", "integration") | ||
sourceRepoPkgList := execList(t, sourceRepoTestDir) | ||
|
||
sourceRepoPkgMap := make(map[string]bool) | ||
for _, pkg := range sourceRepoPkgList { | ||
sourceRepoPkgMap[pkg] = true | ||
} | ||
|
||
for _, pkg := range targetRepoPkgList { | ||
pkgName := strings.TrimPrefix(pkg, "github.com/sourcenetwork/defradb/") | ||
t.Run(pkgName, func(t *testing.T) { | ||
if pkg == "" || !sourceRepoPkgMap[pkg] { | ||
t.Skip("skipping unknown or new test package") | ||
} | ||
|
||
t.Parallel() | ||
dataDir := t.TempDir() | ||
|
||
sourceTestPkg := filepath.Join(sourceRepoDir, pkgName) | ||
execTest(t, sourceTestPkg, dataDir, true) | ||
|
||
targetTestPkg := filepath.Join(targetRepoDir, pkgName) | ||
execTest(t, targetTestPkg, dataDir, false) | ||
}) | ||
} | ||
} | ||
|
||
// execList returns a list of all packages in the given directory. | ||
func execList(t *testing.T, dir string) []string { | ||
cmd := exec.Command("go", "list", "./...") | ||
cmd.Dir = dir | ||
|
||
out, err := cmd.Output() | ||
require.NoError(t, err, string(out)) | ||
|
||
return strings.Split(string(out), "\n") | ||
} | ||
|
||
// execTest runs the tests in the given directory and sets the data | ||
// directory and setup only environment variables. | ||
func execTest(t *testing.T, dir, dataDir string, setupOnly bool) { | ||
cmd := exec.Command("go", "test", ".", "-count", "1", "-v") | ||
cmd.Dir = dir | ||
cmd.Env = append( | ||
os.Environ(), | ||
fmt.Sprintf("%s=%s", enableEnvName, "true"), | ||
fmt.Sprintf("%s=%s", rootDataDirEnvName, dataDir), | ||
) | ||
|
||
if setupOnly { | ||
cmd.Env = append(cmd.Env, fmt.Sprintf("%s=%s", setupOnlyEnvName, "true")) | ||
} | ||
|
||
out, err := cmd.Output() | ||
require.NoError(t, err, string(out)) | ||
} | ||
|
||
// execClone clones the repo from the given url and branch into the directory. | ||
func execClone(t *testing.T, dir, url, branch string) { | ||
cmd := exec.Command( | ||
"git", | ||
"clone", | ||
"--single-branch", | ||
"--branch", branch, | ||
"--depth", "1", | ||
url, | ||
dir, | ||
) | ||
|
||
out, err := cmd.Output() | ||
require.NoError(t, err, string(out)) | ||
} | ||
|
||
// execMakeDeps runs make:deps in the given directory. | ||
func execMakeDeps(t *testing.T, dir string) { | ||
cmd := exec.Command("make", "deps:lens") | ||
cmd.Dir = dir | ||
|
||
out, err := cmd.Output() | ||
require.NoError(t, err, string(out)) | ||
} | ||
|
||
func checkIfDatabaseFormatChangesAreDocumented(t *testing.T, sourceDir, targetDir string) bool { | ||
sourceChanges, ok := getDatabaseFormatDocumentation(t, sourceDir, false) | ||
require.True(t, ok, "Documentation directory not found") | ||
|
||
changes := make(map[string]struct{}, len(sourceChanges)) | ||
for _, f := range sourceChanges { | ||
// Note: we assume flat directory for now - sub directories are not expanded | ||
changes[f.Name()] = struct{}{} | ||
} | ||
|
||
targetChanges, ok := getDatabaseFormatDocumentation(t, targetDir, true) | ||
require.True(t, ok, "Documentation directory not found") | ||
|
||
for _, f := range targetChanges { | ||
if _, isChangeOld := changes[f.Name()]; !isChangeOld { | ||
// If there is a new file in the directory then the change | ||
// has been documented and the test should pass | ||
return true | ||
} | ||
} | ||
|
||
return false | ||
} | ||
|
||
func getDatabaseFormatDocumentation(t *testing.T, startPath string, allowDescend bool) ([]fs.DirEntry, bool) { | ||
startInfo, err := os.Stat(startPath) | ||
require.NoError(t, err) | ||
|
||
var currentDirectory string | ||
if startInfo.IsDir() { | ||
currentDirectory = startPath | ||
} else { | ||
currentDirectory = path.Dir(startPath) | ||
} | ||
|
||
for { | ||
directoryContents, err := os.ReadDir(currentDirectory) | ||
require.NoError(t, err) | ||
|
||
for _, directoryItem := range directoryContents { | ||
directoryItemPath := path.Join(currentDirectory, directoryItem.Name()) | ||
if directoryItem.Name() == documentationDirectoryName { | ||
probableFormatChangeDirectoryContents, err := os.ReadDir(directoryItemPath) | ||
require.NoError(t, err) | ||
|
||
for _, possibleDocumentationItem := range probableFormatChangeDirectoryContents { | ||
if path.Ext(possibleDocumentationItem.Name()) == ".md" { | ||
// If the directory's name matches the expected, and contains .md files | ||
// we assume it is the documentation directory | ||
return probableFormatChangeDirectoryContents, true | ||
} | ||
} | ||
} else { | ||
if directoryItem.IsDir() { | ||
childContents, directoryFound := getDatabaseFormatDocumentation(t, directoryItemPath, false) | ||
if directoryFound { | ||
return childContents, true | ||
} | ||
} | ||
} | ||
} | ||
|
||
if allowDescend { | ||
// If not found in this directory, continue down the path | ||
currentDirectory = path.Dir(currentDirectory) | ||
require.True(t, currentDirectory != "." && currentDirectory != "/") | ||
} else { | ||
return []fs.DirEntry{}, false | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,102 @@ | ||
// Copyright 2023 Democratized Data Foundation | ||
// | ||
// 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 change_detector | ||
|
||
import ( | ||
"os" | ||
"path" | ||
"strconv" | ||
"testing" | ||
|
||
"github.com/stretchr/testify/require" | ||
) | ||
|
||
var ( | ||
// Enabled is true when the change detector is running. | ||
Enabled bool | ||
// SetupOnly is true when the change detector is running in setup mode. | ||
SetupOnly bool | ||
// Repository is the url of the repository to run change detector on. | ||
Repository string | ||
// SourceBranch is the name of the source branch to run change detector on. | ||
SourceBranch string | ||
// TargetBranch is the name of the target branch to run change detector on. | ||
TargetBranch string | ||
// rootDatabaseDir is the shared database directory for running tests. | ||
rootDatabaseDir string | ||
// previousTestCaseTestName is the name of the previous test. | ||
previousTestCaseTestName string | ||
) | ||
|
||
const ( | ||
repositoryEnvName = "DEFRA_CHANGE_DETECTOR_REPOSITORY" | ||
sourceBranchEnvName = "DEFRA_CHANGE_DETECTOR_SOURCE_BRANCH" | ||
targetBranchEnvName = "DEFRA_CHANGE_DETECTOR_TARGET_BRANCH" | ||
setupOnlyEnvName = "DEFRA_CHANGE_DETECTOR_SETUP_ONLY" | ||
rootDataDirEnvName = "DEFRA_CHANGE_DETECTOR_ROOT_DATA_DIR" | ||
enableEnvName = "DEFRA_CHANGE_DETECTOR_ENABLE" | ||
) | ||
|
||
const ( | ||
defaultRepository = "https://github.com/sourcenetwork/defradb.git" | ||
defaultSourceBranch = "develop" | ||
documentationDirectoryName = "data_format_changes" | ||
) | ||
|
||
func init() { | ||
Enabled, _ = strconv.ParseBool(os.Getenv(enableEnvName)) | ||
SetupOnly, _ = strconv.ParseBool(os.Getenv(setupOnlyEnvName)) | ||
TargetBranch = os.Getenv(targetBranchEnvName) | ||
rootDatabaseDir = os.Getenv(rootDataDirEnvName) | ||
|
||
if value, ok := os.LookupEnv(repositoryEnvName); ok { | ||
Repository = value | ||
} else { | ||
Repository = defaultRepository | ||
} | ||
|
||
if value, ok := os.LookupEnv(sourceBranchEnvName); ok { | ||
SourceBranch = value | ||
} else { | ||
SourceBranch = defaultSourceBranch | ||
} | ||
} | ||
|
||
// DatabaseDir returns the database directory for change detector test. | ||
func DatabaseDir(t testing.TB) string { | ||
return path.Join(rootDatabaseDir, t.Name()) | ||
} | ||
|
||
// PreTestChecks skips any test that can't be run by the change detector. | ||
func PreTestChecks(t *testing.T, collectionNames []string) { | ||
if !Enabled { | ||
return | ||
} | ||
|
||
if previousTestCaseTestName == t.Name() { | ||
t.Skip("skipping duplicate test") | ||
} | ||
previousTestCaseTestName = t.Name() | ||
|
||
if len(collectionNames) == 0 { | ||
t.Skip("skipping test with no collections") | ||
} | ||
|
||
if SetupOnly { | ||
return | ||
} | ||
|
||
_, err := os.Stat(DatabaseDir(t)) | ||
if os.IsNotExist(err) { | ||
t.Skip("skipping new test package") | ||
} | ||
require.NoError(t, err) | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
suggestion(non-blocking, minor): Whilst I like the approach, I do dislike that this test will be executed by default with
go test ./...
, I think it might be nicer if we exclude this test by default, perhaps using a build flag. This would however complicate the github action matrix stuff.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added build tag here and updated Makefile to include
--tags change_detector
.