From b5bf42bd010e340905604d69c56e0e6c60f919ec Mon Sep 17 00:00:00 2001 From: Naadir Jeewa Date: Wed, 16 Sep 2020 13:19:32 +0100 Subject: [PATCH] Refactor of testing to improve developer experience Includes addition of kubetest and conformance testing Signed-off-by: Naadir Jeewa --- .gitignore | 2 + Makefile | 236 ++++++++------- common.mk | 65 ++++ docs/book/src/developer/e2e.md | 58 ++-- hack/tools/Makefile | 115 +++++++ hack/tools/tools.mk | 12 + scripts/ci-conformance.sh | 64 ++++ scripts/ci-e2e.sh | 2 +- test/e2e/Makefile | 65 ---- test/e2e/common.go | 58 +--- test/e2e/config/docker-ci.yaml | 8 + test/e2e/config/docker-dev.yaml | 8 + test/e2e/conformance/ci_artifacts.go | 137 +++++++++ test/e2e/conformance/ci_artifacts_test.go | 39 +++ test/e2e/conformance/conformance.go | 139 +++++++++ .../e2e/conformance/conformance_suite_test.go | 179 +++++++++++ test/e2e/conformance/conformance_test.go | 39 +++ .../cluster-template-ci.yaml | 2 +- .../cluster-template-kcp-adoption.yaml | 2 +- .../cluster-template.yaml | 2 +- .../platform-kustomization.yaml | 19 ++ test/e2e/data/kubetest/conformance-fast.yaml | 8 + test/e2e/data/kubetest/conformance.yaml | 7 + test/e2e/e2e_suite_test.go | 162 ++++------ test/e2e/internal/setup/setup.go | 230 ++++++++++++++ test/e2e/kcp_adoption.go | 24 +- test/e2e/kcp_upgrade.go | 24 +- test/e2e/kcp_upgrade_ci_artifacts.go | 216 +++++++++++++ test/e2e/kcp_upgrade_ci_artifacts_test.go | 39 +++ test/e2e/md_upgrades.go | 24 +- test/e2e/mhc_remediations.go | 24 +- test/e2e/quick_start.go | 24 +- test/e2e/self_hosted.go | 26 +- test/framework/alltypes_helpers.go | 16 + test/framework/bootstrap/kind_provider.go | 2 +- test/framework/bootstrap/kind_util.go | 2 +- test/framework/cluster_helpers.go | 6 +- test/framework/cluster_proxy.go | 2 +- test/framework/clusterctl/client.go | 2 +- .../clusterctl/clusterctl_helpers.go | 2 +- test/framework/clusterctl/e2e_config.go | 13 +- test/framework/config.go | 14 + test/framework/controlplane_helpers.go | 6 +- test/framework/convenience.go | 30 ++ test/framework/deployment_helpers.go | 2 +- test/framework/deprecated.go | 2 +- .../log/log.go => ginkgoextensions/output.go} | 18 +- .../kubernetesversions/bindata.go} | 12 +- .../data/debian_injection_script.envsubst.sh | 106 +++++++ .../data/kustomization.yaml | 8 + test/framework/kubernetesversions/template.go | 177 +++++++++++ test/framework/kubernetesversions/versions.go | 78 +++++ .../zz_generated.bindata.go | 285 ++++++++++++++++++ test/framework/kubetest/run.go | 211 +++++++++++++ test/framework/kubetest/setup.go | 63 ++++ test/framework/log/log.go | 167 ++++++++++ test/framework/machine_helpers.go | 2 +- test/framework/machinedeployment_helpers.go | 2 +- test/framework/namespace_helpers.go | 2 +- test/framework/pod_helpers.go | 1 + test/framework/suite_helpers.go | 46 +++ test/infrastructure/docker/.gitignore | 2 + test/infrastructure/docker/Makefile | 64 ++-- .../api/v1alpha3/dockermachine_types.go | 5 + ...cture.cluster.x-k8s.io_dockermachines.yaml | 5 + ...uster.x-k8s.io_dockermachinetemplates.yaml | 5 + .../controllers/dockermachine_controller.go | 25 +- versions.mk | 16 + 68 files changed, 3007 insertions(+), 451 deletions(-) create mode 100644 common.mk create mode 100644 hack/tools/Makefile create mode 100644 hack/tools/tools.mk create mode 100644 scripts/ci-conformance.sh delete mode 100644 test/e2e/Makefile create mode 100644 test/e2e/conformance/ci_artifacts.go create mode 100644 test/e2e/conformance/ci_artifacts_test.go create mode 100644 test/e2e/conformance/conformance.go create mode 100644 test/e2e/conformance/conformance_suite_test.go create mode 100644 test/e2e/conformance/conformance_test.go create mode 100644 test/e2e/data/infrastructure-docker/platform-kustomization.yaml create mode 100644 test/e2e/data/kubetest/conformance-fast.yaml create mode 100644 test/e2e/data/kubetest/conformance.yaml create mode 100644 test/e2e/internal/setup/setup.go create mode 100644 test/e2e/kcp_upgrade_ci_artifacts.go create mode 100644 test/e2e/kcp_upgrade_ci_artifacts_test.go rename test/framework/{internal/log/log.go => ginkgoextensions/output.go} (65%) rename test/{e2e/internal/log/log.go => framework/kubernetesversions/bindata.go} (67%) create mode 100644 test/framework/kubernetesversions/data/debian_injection_script.envsubst.sh create mode 100644 test/framework/kubernetesversions/data/kustomization.yaml create mode 100644 test/framework/kubernetesversions/template.go create mode 100644 test/framework/kubernetesversions/versions.go create mode 100644 test/framework/kubernetesversions/zz_generated.bindata.go create mode 100644 test/framework/kubetest/run.go create mode 100644 test/framework/kubetest/setup.go create mode 100644 test/framework/log/log.go create mode 100644 test/framework/suite_helpers.go create mode 100644 versions.mk diff --git a/.gitignore b/.gitignore index f86f8b5418d2..50698281f676 100644 --- a/.gitignore +++ b/.gitignore @@ -63,3 +63,5 @@ clusterctl-settings.json # test results _artifacts + +*.sentinel diff --git a/Makefile b/Makefile index d573a9523d97..16a3b21ec256 100644 --- a/Makefile +++ b/Makefile @@ -16,53 +16,22 @@ # https://suva.sh/posts/well-documented-makefiles # Ensure Make is run with bash shell as some syntax below is bash-specific -SHELL:=/usr/bin/env bash -.DEFAULT_GOAL:=help - -# Use GOPROXY environment variable if set -GOPROXY := $(shell go env GOPROXY) -ifeq ($(GOPROXY),) -GOPROXY := https://proxy.golang.org -endif -export GOPROXY - -# Active module mode, as we use go modules to manage dependencies -export GO111MODULE=on +ROOT_DIR_RELATIVE := . +include $(ROOT_DIR_RELATIVE)/common.mk +include $(ROOT_DIR_RELATIVE)/hack/tools/tools.mk # Default timeout for starting/stopping the Kubebuilder test control plane export KUBEBUILDER_CONTROLPLANE_START_TIMEOUT ?=60s export KUBEBUILDER_CONTROLPLANE_STOP_TIMEOUT ?=60s -# This option is for running docker manifest command -export DOCKER_CLI_EXPERIMENTAL := enabled - # Directories. EXP_DIR := exp -TOOLS_DIR := hack/tools -TOOLS_BIN_DIR := $(TOOLS_DIR)/bin BIN_DIR := bin E2E_FRAMEWORK_DIR := test/framework CAPD_DIR := test/infrastructure/docker -RELEASE_NOTES_BIN := bin/release-notes -RELEASE_NOTES := $(TOOLS_DIR)/$(RELEASE_NOTES_BIN) -LINK_CHECKER_BIN := bin/liche -LINK_CHECKER := $(TOOLS_DIR)/$(LINK_CHECKER_BIN) -GO_APIDIFF_BIN := bin/go-apidiff -GO_APIDIFF := $(TOOLS_DIR)/$(GO_APIDIFF_BIN) -ENVSUBST_BIN := bin/envsubst -ENVSUBST := $(TOOLS_DIR)/$(ENVSUBST_BIN) - -# Binaries. -# Need to use abspath so we can invoke these from subdirectories -KUSTOMIZE := $(abspath $(TOOLS_BIN_DIR)/kustomize) -CONTROLLER_GEN := $(abspath $(TOOLS_BIN_DIR)/controller-gen) -GOLANGCI_LINT := $(abspath $(TOOLS_BIN_DIR)/golangci-lint) -CONVERSION_GEN := $(abspath $(TOOLS_BIN_DIR)/conversion-gen) -ENVSUBST := $(abspath $(TOOLS_BIN_DIR)/envsubst) # Bindata. -GOBINDATA := $(abspath $(TOOLS_BIN_DIR)/go-bindata) GOBINDATA_CLUSTERCTL_DIR := cmd/clusterctl/config CLOUDINIT_PKG_DIR := bootstrap/kubeadm/internal/cloudinit CLOUDINIT_GENERATED := $(CLOUDINIT_PKG_DIR)/zz_generated.bindata.go @@ -92,43 +61,68 @@ ALL_ARCH = amd64 arm arm64 ppc64le s390x # Allow overriding the imagePullPolicy PULL_POLICY ?= Always -# Hosts running SELinux need :z added to volume mounts -SELINUX_ENABLED := $(shell cat /sys/fs/selinux/enforce 2> /dev/null || echo 0) - -ifeq ($(SELINUX_ENABLED),1) - DOCKER_VOL_OPTS?=:z -endif - # Set build time variables including version details LDFLAGS := $(shell hack/version.sh) all: test managers clusterctl -help: ## Display this help - @awk 'BEGIN {FS = ":.*##"; printf "\nUsage:\n make \033[36m\033[0m\n"} /^[0-9A-Za-z_-]+:.*?##/ { printf " \033[36m%-45s\033[0m %s\n", $$1, $$2 } /^##@/ { printf "\n\033[1m%s\033[0m\n", substr($$0, 5) } ' $(MAKEFILE_LIST) - ## -------------------------------------- ## Testing ## -------------------------------------- +TEST_ARGS ?= +GOTEST_OPTS ?= +GOTEST_CMD ?= go + .PHONY: test -test: ## Run tests. - source ./scripts/fetch_ext_bins.sh; fetch_tools; setup_envs; go test -v ./... $(TEST_ARGS) +test: ## Run tests. Parameters: GOTEST_CMD (command to use for testing. default="go"), TEST_ARGS (args to pass to the test package. default="") + source ./scripts/fetch_ext_bins.sh + fetch_tools + setup_envs + $(GOTEST_CMD) test -v $(GOTEST_OPTS) ./... $(TEST_ARGS) .PHONY: test-cover -test-cover: ## Run tests with code coverage and code generate reports - source ./scripts/fetch_ext_bins.sh; fetch_tools; setup_envs; go test -v -coverprofile=out/coverage.out ./... $(TEST_ARGS) +test-cover: ## Run tests with code coverage and code generate reports. See test for accepted parameters. + $(MAKE) test GOTEST_OPTS="-coverprofile=out/coverage.out" TEST_ARGS=$(TEST_ARGS) GOTEST_CMD=$(GOTEST_CMD) go tool cover -func=out/coverage.out -o out/coverage.txt go tool cover -html=out/coverage.out -o out/coverage.html -.PHONY: docker-build-e2e -docker-build-e2e: ## Rebuild all Cluster API provider images to be used in the e2e tests - make docker-build REGISTRY=gcr.io/k8s-staging-cluster-api PULL_POLICY=IfNotPresent - $(MAKE) -C test/infrastructure/docker docker-build REGISTRY=gcr.io/k8s-staging-cluster-api +GINKGO_NODES ?= 1 +GINKGO_NOCOLOR ?= false +GINKGO_ARGS ?= +ARTIFACTS ?= $(abspath _artifacts) +E2E_CONF_FILE ?= config/docker-dev.yaml +SKIP_RESOURCE_CLEANUP ?= false +USE_EXISTING_CLUSTER ?= false +GINKGO_FOCUS ?= +export GINKGO_NODES GINKGO_NOCOLOR GINKGO_ARGS ARTIFACTS E2E_CONF_FILE SKIP_RESOURCE_CLEANUP USE_EXISTING_CLUSTER GINKGO_FOCUS .PHONY: test-e2e -test-e2e: ## Run the e2e tests - $(MAKE) -C test/e2e run +test-e2e: $(GINKGO) ## Run the e2e tests. Parameters: GINKGO_NODES (number of parallel Ginkgo runners. default=1), GINKGO_NOCOLOR (whether or not to use color output for Ginkgo. default=false), ARTIFACTS (where to save output. default=_artifacts), E2E_CONF_FILE (clusterctl framework e2e conf file location. default=config/docker-dev.yaml), SKIP_RESOURCE_CLEANUP (default=false), USE_EXISTING_CLUSTER (use current kubectl context, default=false) + $(MAKE) test-e2e-run TEST_DIR=test/e2e + +.PHONY: test-conformance +test-conformance: $(GINKGO) $(KUSTOMIZE) ## Run e2e conformance tests. See test-e2e for accepted parameters. + $(MAKE) test-e2e-run TEST_DIR=test/e2e/conformance + +.PHONY: test-e2e-run +test-e2e-run: docker-build-e2e + $(GINKGO) -v -trace \ + -tags=e2e -focus="$(GINKGO_FOCUS)" \ + -nodes=$(GINKGO_NODES) \ + --noColor=$(GINKGO_NOCOLOR) \ + $(GINKGO_ARGS) $(TEST_DIR) \ + -- \ + -e2e.artifacts-folder="$(ARTIFACTS)" \ + -e2e.config="$(E2E_CONF_FILE)" \ + -e2e.skip-resource-cleanup=$(SKIP_RESOURCE_CLEANUP) \ + -e2e.use-existing-cluster=$(USE_EXISTING_CLUSTER) + +.PHONY: pull-cert-manager +pull-cert-manager: ## Use to cache cert manager images on the top-level Docker + docker pull quay.io/jetstack/cert-manager-cainjector:$(CERT_MANAGER_VERSION) + docker pull quay.io/jetstack/cert-manager-webhook:$(CERT_MANAGER_VERSION) + docker pull quay.io/jetstack/cert-manager-controller:$(CERT_MANAGER_VERSION) ## -------------------------------------- ## Binaries @@ -156,57 +150,38 @@ managers: ## Build all managers clusterctl: ## Build clusterctl binary go build -ldflags "$(LDFLAGS)" -o bin/clusterctl sigs.k8s.io/cluster-api/cmd/clusterctl -$(KUSTOMIZE): $(TOOLS_DIR)/go.mod # Build kustomize from tools folder. - cd $(TOOLS_DIR); go build -tags=tools -o $(BIN_DIR)/kustomize sigs.k8s.io/kustomize/kustomize/v3 - -$(CONTROLLER_GEN): $(TOOLS_DIR)/go.mod # Build controller-gen from tools folder. - cd $(TOOLS_DIR); go build -tags=tools -o $(BIN_DIR)/controller-gen sigs.k8s.io/controller-tools/cmd/controller-gen - -$(GOLANGCI_LINT): $(TOOLS_DIR)/go.mod # Build golangci-lint from tools folder. - cd $(TOOLS_DIR); go build -tags=tools -o $(BIN_DIR)/golangci-lint github.com/golangci/golangci-lint/cmd/golangci-lint - -$(CONVERSION_GEN): $(TOOLS_DIR)/go.mod - cd $(TOOLS_DIR); go build -tags=tools -o $(BIN_DIR)/conversion-gen k8s.io/code-generator/cmd/conversion-gen - -$(GOBINDATA): $(TOOLS_DIR)/go.mod # Build go-bindata from tools folder. - cd $(TOOLS_DIR); go build -tags=tools -o $(BIN_DIR)/go-bindata github.com/go-bindata/go-bindata/go-bindata - -$(RELEASE_NOTES): $(TOOLS_DIR)/go.mod - cd $(TOOLS_DIR) && go build -tags=tools -o $(RELEASE_NOTES_BIN) ./release - -$(LINK_CHECKER): $(TOOLS_DIR)/go.mod - cd $(TOOLS_DIR) && go build -tags=tools -o $(LINK_CHECKER_BIN) github.com/raviqqe/liche - -$(GO_APIDIFF): $(TOOLS_DIR)/go.mod - cd $(TOOLS_DIR) && go build -tags=tools -o $(GO_APIDIFF_BIN) github.com/joelanford/go-apidiff - -$(ENVSUBST): $(TOOLS_DIR)/go.mod - cd $(TOOLS_DIR) && go build -tags=tools -o $(ENVSUBST_BIN) github.com/drone/envsubst/cmd/envsubst - -envsubst: $(ENVSUBST) ## Build a local copy of envsubst. -kustomize: $(KUSTOMIZE) ## Build a local copy of kustomize. - .PHONY: e2e-framework e2e-framework: ## Builds the CAPI e2e framework - cd $(E2E_FRAMEWORK_DIR); go build ./... + go build ./test/framework/... ## -------------------------------------- ## Linting ## -------------------------------------- +LINT_ARGS ?= + .PHONY: lint lint-full lint: $(GOLANGCI_LINT) ## Lint codebase - $(GOLANGCI_LINT) run -v - cd $(E2E_FRAMEWORK_DIR); $(GOLANGCI_LINT) run -v - cd $(CAPD_DIR); $(GOLANGCI_LINT) run -v + $(GOLANGCI_LINT) run -v $(LINT_ARGS) + $(MAKE) -C $(CAPD_DIR) lint LINT_ARGS=$(LINT_ARGS) -lint-full: $(GOLANGCI_LINT) ## Run slower linters to detect possible issues - $(GOLANGCI_LINT) run -v --fast=false - cd $(E2E_FRAMEWORK_DIR); $(GOLANGCI_LINT) run -v --fast=false - cd $(CAPD_DIR); $(GOLANGCI_LINT) run -v --fast=false +lint-full: ## Run slower linters to detect possible issues + $(MAKE) lint LINT_ARGS=--fast=false -apidiff: $(GO_APIDIFF) ## Check for API differences - $(GO_APIDIFF) $(shell git rev-parse origin/master) --print-compatible +REMOTE ?=upstream + +apidiff: $(GO_APIDIFF) ## Check for API differences during development. Parameters: REMOTE (which git remote to use. Default: upstream) + @if !(git diff --quiet HEAD); then \ + git diff; \ + echo "commit changes first"; exit 1; \ + fi + + HEAD_COMMIT=$$(git rev-parse $(REMOTE)/master); \ + TMP_DIR=$$(mktemp -d -t apidiff-XXXXXXXXXX); \ + git clone . $${TMP_DIR}; \ + cd $${TMP_DIR}; \ + $(abspath $(GO_APIDIFF)) $${HEAD_COMMIT} --print-compatible; \ + rm -rf $${TMP_DIR} ## -------------------------------------- ## Generate / Manifests @@ -217,10 +192,11 @@ generate: ## Generate code $(MAKE) generate-manifests $(MAKE) generate-go $(MAKE) generate-bindata - $(MAKE) -C test/infrastructure/docker generate + $(MAKE) -C $(CAPD_DIR) generate .PHONY: generate-go -generate-go: ## Runs Go related generate targets +generate-go: $(GOBINDATA) ## Runs Go related generate targets + go generate ./... $(MAKE) generate-go-core $(MAKE) generate-go-kubeadm-bootstrap $(MAKE) generate-go-kubeadm-control-plane @@ -327,7 +303,7 @@ generate-kubeadm-control-plane-manifests: $(CONTROLLER_GEN) ## Generate manifest .PHONY: modules modules: ## Runs go mod to ensure modules are up to date. go mod tidy - cd $(TOOLS_DIR); go mod tidy + $(MAKE) -C $(TOOLS_DIR) modules $(MAKE) -C $(CAPD_DIR) modules ## -------------------------------------- @@ -335,28 +311,37 @@ modules: ## Runs go mod to ensure modules are up to date. ## -------------------------------------- .PHONY: docker-build -docker-build: ## Build the docker images for controller managers - $(MAKE) ARCH=$(ARCH) docker-build-core - $(MAKE) ARCH=$(ARCH) docker-build-kubeadm-bootstrap - $(MAKE) ARCH=$(ARCH) docker-build-kubeadm-control-plane +docker-build: docker-build-core docker-build-kubeadm-bootstrap docker-build-kubeadm-control-plane ## Build the docker images for controller managers .PHONY: docker-build-core -docker-build-core: ## Build the docker image for core controller manager +docker-build-core: .docker-build-core.sentinel ## Build the docker image for core controller manager + +CORE_SRCS := $(call rwildcard,controllers,*.*) $(call rwildcard,cmd,*.*) $(call rwildcard,feature,*.*) $(call rwildcard,exp,*.*) $(call rwildcard,api,*.*) $(call rwildcard,third_party,*.*) go.mod go.sum + +.docker-build-core.sentinel: $(CORE_SRCS) DOCKER_BUILDKIT=1 docker build --build-arg goproxy=$(GOPROXY) --build-arg ARCH=$(ARCH) --build-arg ldflags="$(LDFLAGS)" . -t $(CONTROLLER_IMG)-$(ARCH):$(TAG) - $(MAKE) set-manifest-image MANIFEST_IMG=$(CONTROLLER_IMG)-$(ARCH) MANIFEST_TAG=$(TAG) TARGET_RESOURCE="./config/manager/manager_image_patch.yaml" - $(MAKE) set-manifest-pull-policy TARGET_RESOURCE="./config/manager/manager_pull_policy.yaml" + touch $@ .PHONY: docker-build-kubeadm-bootstrap -docker-build-kubeadm-bootstrap: ## Build the docker image for kubeadm bootstrap controller manager +docker-build-kubeadm-bootstrap: bootstrap/kubeadm/.docker-build-kubeadm-bootstrap.sentinel ## Build the docker image for kubeadm bootstrap controller manager + +KUBEADM_BOOTSTRAP_SRCS := $(CORE_SRCS) $(call rwildcard,bootstrap,*.*) +bootstrap/kubeadm/.docker-build-kubeadm-bootstrap.sentinel: $(KUBEADM_BOOTSTRAP_SRCS) DOCKER_BUILDKIT=1 docker build --build-arg goproxy=$(GOPROXY) --build-arg ARCH=$(ARCH) --build-arg package=./bootstrap/kubeadm --build-arg ldflags="$(LDFLAGS)" . -t $(KUBEADM_BOOTSTRAP_CONTROLLER_IMG)-$(ARCH):$(TAG) - $(MAKE) set-manifest-image MANIFEST_IMG=$(KUBEADM_BOOTSTRAP_CONTROLLER_IMG)-$(ARCH) MANIFEST_TAG=$(TAG) TARGET_RESOURCE="./bootstrap/kubeadm/config/manager/manager_image_patch.yaml" - $(MAKE) set-manifest-pull-policy TARGET_RESOURCE="./bootstrap/kubeadm/config/manager/manager_pull_policy.yaml" + touch $@ .PHONY: docker-build-kubeadm-control-plane -docker-build-kubeadm-control-plane: ## Build the docker image for kubeadm control plane controller manager +docker-build-kubeadm-control-plane: controlplane/kubeadm/.docker-build-kubeadm-control-plane.sentinel ## Build the docker image for kubeadm control plane controller manager + +KUBEADM_CONTROLPLANE_SRCS := $(KUBEADM_BOOTSTRAP_SRCS) $(call rwildcard,controlplane,*.*) +controlplane/kubeadm/.docker-build-kubeadm-control-plane.sentinel: $(KUBEADM_CONTROLPLANE_SRCS) DOCKER_BUILDKIT=1 docker build --build-arg goproxy=$(GOPROXY) --build-arg ARCH=$(ARCH) --build-arg package=./controlplane/kubeadm --build-arg ldflags="$(LDFLAGS)" . -t $(KUBEADM_CONTROL_PLANE_CONTROLLER_IMG)-$(ARCH):$(TAG) - $(MAKE) set-manifest-image MANIFEST_IMG=$(KUBEADM_CONTROL_PLANE_CONTROLLER_IMG)-$(ARCH) MANIFEST_TAG=$(TAG) TARGET_RESOURCE="./controlplane/kubeadm/config/manager/manager_image_patch.yaml" - $(MAKE) set-manifest-pull-policy TARGET_RESOURCE="./controlplane/kubeadm/config/manager/manager_pull_policy.yaml" + touch $@ + +.PHONY: docker-build-e2e +docker-build-e2e: ## Rebuild all Cluster API provider images to be used for e2e testing + $(MAKE) docker-build REGISTRY=gcr.io/k8s-staging-cluster-api PULL_POLICY=IfNotPresent + $(MAKE) -C $(CAPD_DIR) docker-build REGISTRY=gcr.io/k8s-staging-cluster-api .PHONY: docker-push docker-push: ## Push the docker images @@ -454,10 +439,10 @@ release: clean-release ## Builds and push container images using the latest git $(MAKE) set-manifest-pull-policy PULL_POLICY=IfNotPresent TARGET_RESOURCE="./controlplane/kubeadm/config/manager/manager_pull_policy.yaml" $(MAKE) release-manifests # Release CAPD components and add them to the release dir - $(MAKE) -C test/infrastructure/docker/ release - cp test/infrastructure/docker/out/infrastructure-components.yaml $(RELEASE_DIR)/infrastructure-components-development.yaml + $(MAKE) -C $(CAPD_DIR) release + cp $(CAPD_DIR)/out/infrastructure-components.yaml $(RELEASE_DIR)/infrastructure-components-development.yaml # Adds CAPD templates - cp test/infrastructure/docker/templates/* $(RELEASE_DIR)/ + cp $(CAPD_DIR)/templates/* $(RELEASE_DIR)/ .PHONY: release-manifests release-manifests: $(RELEASE_DIR) $(KUSTOMIZE) ## Builds the manifests to publish with a release @@ -529,6 +514,20 @@ docker-build-example-provider: ## Build the docker image for example provider clean: ## Remove all generated files $(MAKE) clean-bin $(MAKE) clean-book + $(MAKE) clean-artifacts + $(MAKE) clean-envtest + $(MAKE) clean-sentinels + +.PHONY: clean-sentinels +clean-sentinels: ## Delete docker build sentinels + rm -f .docker-build-core.sentinel + rm -f bootstrap/kubeadm/.docker-build-kubeadm-bootstrap.sentinel + rm -f controlplane/kubeadm/.docker-build-kubeadm-control-plane.sentinel + $(MAKE) -C $(CAPD_DIR) clean-sentinels + +.PHONY: clean-artifacts +clean-artifacts: ## Remove all test artifacts + rm -rf _artifacts .PHONY: clean-bin clean-bin: ## Remove all generated binaries @@ -547,11 +546,15 @@ clean-book: ## Remove all generated GitBook files clean-bindata: ## Remove bindata generated folder rm -rf $(GOBINDATA_CLUSTERCTL_DIR)/manifest -.PHONY: clean-manifests ## Reset manifests in config directories back to master -clean-manifests: +.PHONY: clean-manifests +clean-manifests: ## Reset manifests in config directories back to master @read -p "WARNING: This will reset all config directories to local master. Press [ENTER] to continue." git checkout master config bootstrap/kubeadm/config controlplane/kubeadm/config test/infrastructure/docker/config +.PHONY: clean-envtest +clean-envtest: ## Remove kubebuilder envtest binaries and tars + -rm -rf /tmp/kubebuilder* + .PHONY: format-tiltfile format-tiltfile: ## Format Tiltfile ./hack/verify-starlark.sh fix @@ -588,12 +591,13 @@ verify-gen: generate .PHONY: verify-docker-provider verify-docker-provider: @echo "Verifying CAPD" - cd $(CAPD_DIR); $(MAKE) verify + $(MAKE) -C $(CAPD_DIR) verify .PHONY: verify-book-links verify-book-links: $(LINK_CHECKER) # Ignore localhost links and set concurrency to a reasonable number $(LINK_CHECKER) -r docs/book -x "^https?://" -c 10 + ## -------------------------------------- ## Others / Utilities ## -------------------------------------- diff --git a/common.mk b/common.mk new file mode 100644 index 000000000000..59e63bbdcd49 --- /dev/null +++ b/common.mk @@ -0,0 +1,65 @@ +# Copyright 2020 The Kubernetes Authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +include $(ROOT_DIR_RELATIVE)/versions.mk + +# Ensure Make is run with bash shell as some syntax below is bash-specific +SHELL:=bash +.ONESHELL: +.SHELLFLAGS := -eu -o pipefail -c +.DELETE_ON_ERROR: +MAKEFLAGS += --warn-undefined-variables +MAKEFLAGS += --no-builtin-rules + +TOOLS_DIR := $(ROOT_DIR_RELATIVE)/hack/tools +TOOLS_DIR_DEPS := $(TOOLS_DIR)/go.sum $(TOOLS_DIR)/go.mod $(TOOLS_DIR)/Makefile +TOOLS_BIN_DIR := $(TOOLS_DIR)/bin + +PATH := $(abspath $(TOOLS_BIN_DIR)):$(PATH) +export PATH + +UID := $(shell id -u) +GID := $(shell id -g) + +rwildcard=$(foreach d,$(wildcard $(1:=/*)),$(call rwildcard,$d,$2) $(filter $(subst *,%,$2),$d)) + +# Hosts running SELinux need :z added to volume mounts +SELINUX_ENABLED := $(shell cat /sys/fs/selinux/enforce 2> /dev/null || echo 0) + +ifeq ($(SELINUX_ENABLED),1) + DOCKER_VOL_OPTS?=:z +endif + +# This option is for running docker manifest command +export DOCKER_CLI_EXPERIMENTAL := enabled + +# Use GOPROXY environment variable if set +GOPROXY := $(shell go env GOPROXY) +ifeq ($(GOPROXY),) +GOPROXY := https://proxy.golang.org +endif +export GOPROXY + + +$(TOOLS_BIN_DIR)/%: $(TOOLS_DIR_DEPS) + make -C $(TOOLS_DIR) $(@:hack/tools/%=%) + +## -------------------------------------- +## Help +## -------------------------------------- + +.DEFAULT_GOAL:=help + +help: ## Display this help + @awk 'BEGIN {FS = ":.*##"; printf "\nUsage:\n make \033[36m\033[0m\n"} /^[a-zA-Z0-9_-]+:.*?##/ { printf " \033[36m%-15s\033[0m %s\n", $$1, $$2 } /^##@/ { printf "\n\033[1m%s\033[0m\n", substr($$0, 5) } ' $(MAKEFILE_LIST) diff --git a/docs/book/src/developer/e2e.md b/docs/book/src/developer/e2e.md index ea33db4fd032..4f0563f2eb1a 100644 --- a/docs/book/src/developer/e2e.md +++ b/docs/book/src/developer/e2e.md @@ -9,20 +9,20 @@ Following guidelines should be followed when developing E2E tests: - Define test spec reflecting real user workflow, e.g. [Cluster API quick start]. - Unless you are testing provider specific features, ensure your test can run with different infrastructure providers (see [Writing Portable Tests](#writing-portable-e2e-tests)). - + The [Cluster API test framework] provides you a set of helpers method for getting your test in place -quickly; the [test E2E package] provide examples of how this can be achieved and reusable +quickly; the [test E2E package] provide examples of how this can be achieved and reusable test specs for the most common Cluster API use cases. ## Prerequisites Each E2E test requires a set of artifacts to be available: -- Binaries & docker images for Kubernetes, CNI, CRI & CSI +- Binaries & docker images for Kubernetes, CNI, CRI & CSI - Manifests & docker images for the Cluster API core components - Manifests & docker images for the Cluster API infrastructure provider; in most cases also machine images are required (AMI, OVA etc.) -- Credentials for the target infrastructure provider +- Credentials for the target infrastructure provider - Other support tools (e.g. kustomize, gsutil etc.) The Cluster API test framework provides support for building and retrieving the manifest @@ -30,12 +30,12 @@ files for Cluster API core components and for the Cluster API infrastructure pro (see [Setup](#setup)) For the remaining tasks you can find examples of -how this can be implemented e.g. in [CAPA E2E tests] and [CAPG E2E tests]. +how this can be implemented e.g. in [CAPA E2E tests] and [CAPG E2E tests]. ## Setup -In order to run E2E tests it is required to create a Kubernetes cluster with a -complete set of Cluster API providers installed. Setting up those elements is +In order to run E2E tests it is required to create a Kubernetes cluster with a +complete set of Cluster API providers installed. Setting up those elements is usually implemented in a `BeforeSuite` function, and it consists of two steps: - Defining an E2E config file @@ -48,7 +48,7 @@ setting up a management cluster. Using the config file it is possible to: -- Define the list of providers to be installed in the management cluster. Most notably, +- Define the list of providers to be installed in the management cluster. Most notably, for each provider it is possible to define: - One or more versions of the providers manifest (built from the sources, or pulled from a remote location). @@ -68,14 +68,14 @@ An [example E2E config file] can be found here.

Deprecated E2E config file format

The [Cluster API test framework] includes also a [deprecated E2E config file] implementation, -that was used before the introduction of clusterctl. This might be removed in future releases +that was used before the introduction of clusterctl. This might be removed in future releases of the test framework. ### Creating the management cluster and installing providers -In order to run Cluster API E2E tests, you need a Kubernetes cluster; the [NewKindClusterProvider] gives you a +In order to run Cluster API E2E tests, you need a Kubernetes cluster; the [NewKindClusterProvider] gives you a type that can be used to create a local kind cluster and pre-load images into it, but also existing clusters can be used if available. @@ -92,7 +92,7 @@ This method:

Deprecated InitManagementCluster method

The [Cluster API test framework] includes also a [deprecated InitManagementCluster method] implementation, -that was used before the introduction of clusterctl. This might be removed in future releases +that was used before the introduction of clusterctl. This might be removed in future releases of the test framework. @@ -109,14 +109,14 @@ A typical test spec is a sequence of: ### Creating Namespaces The [CreateNamespaceAndWatchEvents method] provides a convenient way to create a namespace and setup -watches for capturing namespaces events +watches for capturing namespaces events ### Creating objects There are two possible approaches for creating objects in the management cluster: - Create object by object: create the `Cluster` object, then `AwsCluster`, `Machines`, `AwsMachines` etc. -- Apply a `cluster-templates.yaml` file thus creating all the objects this file contains. +- Apply a `cluster-templates.yaml` file thus creating all the objects this file contains. The first approaches leverage on the [controller-runtime Client] and gives you full control, but it comes with some drawbacks as well, because this method does not reflect directly real user workflows, and most importantly, @@ -124,7 +124,7 @@ the resulting tests are not as reusable with other infrastructure providers. (Se We recommend using the [ClusterTemplate method] and the [Apply method] for creating objects in the cluster. This methods mimics the recommended user workflows, and it is based on `cluster-templates.yaml` files that can be -provided via the [E2E config file], and thus easily swappable when changing the target infrastructure provider. +provided via the [E2E config file], and thus easily swappable when changing the target infrastructure provider.