This directory contains Crossplane's E2E tests. These tests exercise Crossplane features by:
- Creating a
kind
cluster. - Deploying Crossplane, using its Helm chart.
- Applying a series of YAML manifests.
- Testing whether the resulting resources work as expected.
These tests focus on core Crossplane features - i.e. the package manager,
Composition, etc. They should not take dependencies on 'real' Crossplane
providers or external systems. Instead they use 'fake' providers like
provider-nop
and provider-dummy
.
All Crossplane features must be exercised by these tests, as well as unit tests.
Run earthly -P +e2e
to run E2E tests.
This compiles Crossplane and an E2E test binary. It then uses the test binary to
run the base test suite. Use the FLAGS
to pass flags to the test binary. For
example:
# Some functions that setup the test environment (e.g. kind) use the klog logger
# The -v flag controls the verbosity of klog. Use -v=4 for debug logging.
earthly -P +e2e --FLAGS="-v=4"
# To run only a specific test, match it by regular expression
earthly -P +e2e --FLAGS="-test.run ^TestConfiguration"
# To test features with certain labels, use the labels flag
earthly -P +e2e --FLAGS="-labels area=apiextensions"
# To test a specific feature, use the feature flag
earthly -P +e2e --FLAGS="-feature=ConfigurationWithDependency"
# Stop immediately on first test failure, and leave the kind cluster to debug.
earthly -i -P +e2e --FLAGS="-test.failfast -destroy-kind-cluster=false"
# Run a specific test suite.
earthly -P +e2e --FLAGS="-test.v -test-suite=composition-webhook-schema-validation"
Earthly runs e2e tests in a buildkit container, which is not directly accessible
from host via regular docker ps
and docker exec
commands. To access the
cluster, you can use the following commands:
- Make sure you have started the tests with the
-i
flag. For example:
earthly -i -P +e2e --FLAGS="-v=4"
- Get the container ID of the buildkit container
$ docker ps | grep earthly/buildkitd
$ export EARTHLY_CONTAINER_ID=<container_id>
- Find the id of the runc container running in earthly
$ docker exec $EARTHLY_CONTAINER_ID buildkit-runc list
# Export the container id of the runc container
$ export RUNC_CONTAINER_ID=<container_id>
In case you have only one earthly job running, you can use the following shortcut to get the runc container id:
$ export RUNC_CONTAINER_ID=$(docker exec $EARTHLY_CONTAINER_ID buildkit-runc list -q)
- Exec into the runc container and access the cluster
# Exec into the runc container
$ docker exec -ti $EARTHLY_CONTAINER_ID buildkit-runc exec -t $RUNC_CONTAINER_ID sh
# See the kind cluster
$ kind get clusters
crossplane-e2e-7eda80e167ed36325
# Install kubectl
$ apk add kubectl
# Now you can use kubectl to access the cluster
earthly -P +e2e
runs all defined E2E tests serially. Tests do not run in
parallel. This is because all tests run against the same API server and
Crossplane has a lot of cluster-scoped state - XRDs, Providers, Compositions,
etc. It's easier and less error-prone to write tests when you don't have to
worry about one test potentially conflicting with another - for example by
installing the same provider another test would install.
The CI GitHub workflow uses a matrix strategy to run multiple jobs in parallel, each running a test suite, see the dedicated section for more details.
We are currently splitting the tests to be able to run all basic tests against the default installation of Crossplane, and for each alpha feature covered we run all basic tests plus the feature specific ones against a Crossplane installation with the feature enabled.
In the future we could think of improving parallelism by, for example, adding the area to the matrix or spinning multiple kind clusters for each job.
In order to be able to run specific subsets of tests, we introduced the concept
of test suites. To run a specific test suite use the -test-suite
flag.
A test suite is currently defined by:
- A key to be used as value of the
-test-suite
flag. - A set of labels that will be used to filter the tests to run, which will be added to the user-provided ones, preserving the latter ones in case of key conflicts.
- A list of Helm install options defining the initial Crossplane installation for the given test suite, which will be used to install Crossplane before running the tests. E.g. adding flags to enable alpha features and feature specific flags.
- A list of additional setup steps to be run before installing Crossplane and running the tests. E.g. Loading additional images into the cluster.
- Whether the suite should include the default suite or not, meaning that install options will be added to the default ones if not explicitly specified not to do so.
Test suites enable use cases such as installing Crossplane with a specific alpha feature enabled and running all the basic tests, plus the ones specific to that feature, to make sure the feature is not breaking any default behavior.
In case a test needs a specific Crossplane configuration, it must still take
care of upgrading the installation to the desired configuration, but should then
use environment.GetSelectedSuiteInstallOpts
to retrieve at runtime the baseline
installation options to be sure to restore the previous state. This allows tests
to run against any suite if needed.
Test suites can be combined with labels to run a subset of tests, e.g.
splitting by area, or with the usual Go -run <regexp>
flag to run only
specific tests, in such case, suite labels will be ignored altogether.
We're still learning what the best way to arrange E2E tests is. It's okay for this pattern to change if it's not working well, but please discuss first!
We try to follow this pattern when adding a new test:
- Define a single feature per Test function, if possible.
- Setup a directory of plain YAML manifests per test - i.e. test fixtures - at
e2e/manifests/<area>/<test>
, usually with asetup
sub-folder containing resources to be deployed at setup phase and cleaned up during the teardown. Try to avoid reusing other feature's fixtures, as this would introduce hidden dependencies between tests. - Try reusing existing helpers as much as possible, see package
github.com/crossplane/crossplane/test/e2e/funcs
, or add new ones there if needed. - Prefer using the Fluent APIs to define features
(
features.NewWithDescription(...).WithSetup(...).AssessWithDescription(...).WithTeardown(...).Feature()
).features.Table
should be used only to define multiple self-contained assessments to be run sequentially, but without assuming any ordering among them, similarly to the usual table driven style we adopt for unit testing.
- Prefer the usage of
WithSetup
andWithTeardown
to their unnamed counterparts (Setup
andTeardown
) to define the setup and teardown phases of a feature, as they allow to provide a description. - Use short but explicative
CamelCase
sentences as descriptions for everything used to define the name of tests/subtests, e.g.features.NewWithDescription("CrossplaneUpgrade", ...)
WithSetup("InstallProviderNop", ...)
,AssessWithDescription("ProviderNopIsInstalled", ...)
,WithTeardown("UninstallProviderNop", ...)
. - Use the
Setup
andTeardown
phases to define respectively actions that are not strictly part of the feature being tested, but are needed to make it work, and actions that are needed to clean up the environment after the test has run. - Use
Assess
steps to define the steps required to exercise the actual feature at hand. - Use
Assess
steps to define both conditions that should hold and actions that should be performed. In the former case use active descriptions, e.g.InstallProviderNop
, while in the latter use passive descriptions, e.g.ProviderNopIsInstalled
. - Try to group actions and the checks of what you have done
Assess
step in a single step with an active description if possible, to avoid having twice the steps and making it explicit that we are checking the action executed by the previous function. e.g."UpgradeProvider"
should both upgrade the provider and check that it becomes healthy within a reasonable time. - Avoid using the available context to pass data between steps, as it makes it harder to understand the flow of the test and could lead to data races if not handled properly.
- Keep in mind that all
Setup
andTeardown
steps, wherever are defined are always going to be executed respectively before and after all theAssess
steps defined, so you can defineTeardowns
immediately after the step that defined the resource to be deleted as a sort ofdefer
statement. Same applies toSetup
steps which could actually be located immediately before the step requiring them. But be careful with this, as a non-linear narrative is going to be easier to follow, so if possible stick to all Setups at the beginning and all Teardowns at the end of the feature. - Features can be assigned labels, to allow dicing and slicing the test suite, see below for more details about available labels, but overall try to define all the labels that could be useful to select the test in the future and make sure it's actually being selected when run in CI.
Here an example of a test following the above guidelines:
package e2e
import "sigs.k8s.io/e2e-framework/pkg/features"
// ...
// TestSomeFeature ...
func TestSomeFeature(t *testing.T) {
manifests := "test/e2e/manifests/pkg/some-area/some-feature"
namespace := "some-namespace"
// ... other variables or constants ...
environment.Test(t,
features.NewWithDescription(t.Name(), ...).
WithLabel(LabelArea, ...).
WithLabel(LabelSize, ...).
WithLabel(config.LabelTestSuite, config.TestSuiteDefault).
// ...
WithSetup("ReadyPrerequisites", ...).
// ... other setup steps ...
AssessWithDescription("DoSomething", ...).
AssessWithDescription("SomethingElseIsInSomeState", ...).
// ... other assess steps ...
WithTeardown("DeleteCreatedResources", ...).
// ... other teardown steps ...
Feature(),
)
}
// ...
Features are grouped into broad feature areas - e.g. TestComposition
in
composition_test.go
. Features pertaining to Composition should be added to
TestComposition
.
Some tests may involve updating an existing test - for example you might test a
new kind of transform by updating the composition/patch-and-transform
manifests and adding a new assessment or two to the pandt
features.Table
.
Other, larger tests may involve creating a new directory of manifests and a new
features.Table
. Every features.Table
must be passed to environment.Test
to
be run.
When you pass a feature to environment.Test
you can add arbitrary labels that
may be used to filter which tests are run. Common labels include:
area
: The area of Crossplane being tested -pkg
,apiextensions
, etc.size
:small
if the test completes in under a minute, otherwiselarge
.
If you add a new area
label, be sure to add it to the matrix strategy of the
e2e-tests job in the CI GitHub workflow. We run E2E tests for each area
of Crossplane in parallel.
When adding a test:
- Use commentary to explain what your tests should do.
- Use brief
CamelCase
test names, not descriptive sentences. - Avoid adding complex logic in the
e2e
package. - Implement new test logic as a
features.Func
in thee2e/funcs
package.
The goals of these tests are to:
- Simulate real-world use to exercise all of Crossplane's functionality.
- Run reliably, with no flaky tests.
- Converge on requiring mostly updated manifests - and little Go - to add tests.
- Use tools and patterns that are familiar and unsurprising to contributors.
The following design principals help achieve these goals:
- Use familiar Kubernetes YAML manifests as test fixtures.
- Provide a library of reusable test functions (e.g.
funcs.ResourcesCreatedIn
). - Remain as close to idiomatic Go stdlib
testing
tests as possible. - Use common, idiomatic Kubernetes ecosystem tooling -
e2e-framework
. - Implement as much logic in Go as possible - avoid scripts or shelling out.
Refer to the E2E one-pager for more context.