From cad9cc263b077ba80575beb8732ca22a987e4ec7 Mon Sep 17 00:00:00 2001 From: Lucas Bajolet Date: Fri, 13 Sep 2024 10:06:08 -0400 Subject: [PATCH] packer_test: introduce global compilation queue Compiling plugins was originally intended to be an idempotent operation. This however starts to change as we introduce build customisations, which have the unfortunate side-effect of changing the state of the plugin directory, leading to conflicts between concurrent compilation jobs. Therefore to mitigate this problem, this commit changes how compilation jobs are processed, by introducing a global compilation queue, and processing plugins' compilation one-by-one from this queue. This however makes such requests asynchronous, so test suites that require plugins to be compiled will now have to wait on their completion before they can start their tests. To this effect, we introduce one more convenience function that processes those errors, and automatically fails the test should one compilation job fail for any reason. --- packer_test/common/plugin.go | 99 +++++++++++++++++++++++++++++++++--- packer_test/common/suite.go | 16 ++---- 2 files changed, 96 insertions(+), 19 deletions(-) diff --git a/packer_test/common/plugin.go b/packer_test/common/plugin.go index 90d29647abf..c8adc77ae7e 100644 --- a/packer_test/common/plugin.go +++ b/packer_test/common/plugin.go @@ -2,6 +2,7 @@ package common import ( "fmt" + "log" "os" "os/exec" "path/filepath" @@ -130,6 +131,49 @@ func (ts *PackerTestSuite) GetPluginPath(t *testing.T, version string) string { return path.(string) } +type CompilationResult struct { + Error error + Version string +} + +// Ready processes a series of CompilationResults, as returned by CompilePlugin +// +// If any of the jobs requested failed, the test will fail also. +func Ready(t *testing.T, results []chan CompilationResult) { + for _, res := range results { + jobErr := <-res + empty := CompilationResult{} + if jobErr != empty { + t.Errorf("failed to compile plugin at version %s: %s", jobErr.Version, jobErr.Error) + } + } + + if t.Failed() { + t.Fatalf("some plugins failed to be compiled, see logs for more info") + } +} + +type compilationJob struct { + versionString string + suite *PackerTestSuite + done bool + resultCh chan CompilationResult + customisations []BuildCustomisation +} + +// CompilationJobs keeps a queue of compilation jobs for plugins +// +// This approach allows us to avoid conflicts between compilation jobs. +// Typically building the plugin with different ldflags is safe to perform +// in parallel on the same file set, however customisations tend to be more +// conflictual, as two concurrent compilation jobs may end-up compiling the +// wrong plugin, which may cause some tests to misbehave, or even compilation +// jobs to fail. +// +// The solution to this approach is to have a global queue for every plugin +// compilation to be performed safely. +var CompilationJobs = make(chan compilationJob, 10) + // CompilePlugin builds a tester plugin with the specified version. // // The plugin's code is contained in a subdirectory of this file, and lets us @@ -146,7 +190,49 @@ func (ts *PackerTestSuite) GetPluginPath(t *testing.T, version string) string { // Note: each tester plugin may only be compiled once for a specific version in // a test suite. The version may include core (mandatory), pre-release and // metadata. Unlike Packer core, metadata does matter for the version being built. -func (ts *PackerTestSuite) CompilePlugin(t *testing.T, versionString string, customisations ...BuildCustomisation) { +// +// Note: the compilation will process asynchronously, and should be waited upon +// before tests that use this plugin may proceed. Refer to the `Ready` function +// for doing that. +func (ts *PackerTestSuite) CompilePlugin(versionString string, customisations ...BuildCustomisation) chan CompilationResult { + resultCh := make(chan CompilationResult) + + CompilationJobs <- compilationJob{ + versionString: versionString, + suite: ts, + customisations: customisations, + done: false, + resultCh: resultCh, + } + + return resultCh +} + +func init() { + // Run a processor coroutine for the duration of the test. + // + // It's simpler to have this occurring on the side at all times, without + // trying to manage its lifecycle based on the current amount of queued + // tasks, since this is linked to the test lifecycle, and as it's a single + // coroutine, we can leave it run until the process exits. + go func() { + for job := range CompilationJobs { + log.Printf("compiling plugin on version %s", job.versionString) + err := compilePlugin(job.suite, job.versionString, job.customisations...) + if err != nil { + job.resultCh <- CompilationResult{ + Error: err, + Version: job.versionString, + } + } + close(job.resultCh) + } + }() +} + +// compilePlugin performs the actual compilation procedure for the plugin, and +// registers it to the test suite instance passed as a parameter. +func compilePlugin(ts *PackerTestSuite, versionString string, customisations ...BuildCustomisation) error { // Fail to build plugin if already built. // // Especially with customisations being a thing, relying on cache to get and @@ -154,23 +240,21 @@ func (ts *PackerTestSuite) CompilePlugin(t *testing.T, versionString string, cus // and therefore we cannot rely on it being called twice and producing the // same result, so we forbid it. if _, ok := ts.compiledPlugins.Load(versionString); ok { - t.Fatalf("plugin version %q was already built, use GetTestPlugin instead", versionString) + return fmt.Errorf("plugin version %q was already built, use GetTestPlugin instead", versionString) } v := version.Must(version.NewSemver(versionString)) - t.Logf("Building tester plugin in version %v", v) - testDir, err := currentDir() if err != nil { - t.Fatalf("failed to compile plugin binary: %s", err) + return fmt.Errorf("failed to compile plugin binary: %s", err) } testerPluginDir := filepath.Join(testDir, "plugin_tester") for _, custom := range customisations { err, cleanup := custom(testerPluginDir) if err != nil { - t.Fatalf("failed to prepare plugin workdir: %s", err) + return fmt.Errorf("failed to prepare plugin workdir: %s", err) } defer cleanup() } @@ -180,10 +264,11 @@ func (ts *PackerTestSuite) CompilePlugin(t *testing.T, versionString string, cus compileCommand := exec.Command("go", "build", "-C", testerPluginDir, "-o", outBin, "-ldflags", LDFlags(v), ".") logs, err := compileCommand.CombinedOutput() if err != nil { - t.Fatalf("failed to compile plugin binary: %s\ncompiler logs: %s", err, logs) + return fmt.Errorf("failed to compile plugin binary: %s\ncompiler logs: %s", err, logs) } ts.compiledPlugins.Store(v.String(), outBin) + return nil } type PluginDirSpec struct { diff --git a/packer_test/common/suite.go b/packer_test/common/suite.go index 5c7477ce33f..dad9766e2c7 100644 --- a/packer_test/common/suite.go +++ b/packer_test/common/suite.go @@ -30,22 +30,14 @@ type PackerTestSuite struct { compiledPlugins sync.Map } -func (ts *PackerTestSuite) buildPluginVersion(waitgroup *sync.WaitGroup, versionString string, t *testing.T) { - waitgroup.Add(1) - go func() { - defer waitgroup.Done() - ts.CompilePlugin(t, versionString) - }() -} - +// CompileTestPluginVersions batch compiles a series of plugins func (ts *PackerTestSuite) CompileTestPluginVersions(t *testing.T, versions ...string) { - wg := &sync.WaitGroup{} - + results := []chan CompilationResult{} for _, ver := range versions { - ts.buildPluginVersion(wg, ver, t) + results = append(results, ts.CompilePlugin(ver)) } - wg.Wait() + Ready(t, results) } // SkipNoAcc is a pre-condition that skips the test if the PACKER_ACC environment