mirror of
https://github.com/docker/compose.git
synced 2025-10-24 08:43:49 +02:00
ci: merge Go coverage reports before upload (#10666)
Attempting to fix the state of codecov action checks right now, which are behaving very erratically. Using the new functionality in Go 1.20 to merge multiple reports, so now the unit & E2E coverage data reports are stored as artifacts and then downloaded, merged, and finally uploaded to codecov as a new job. Additionally, add a `codecov.yml` config and try to turn down the aggressiveness of it for CI checks. Signed-off-by: Milas Bowman <milas.bowman@docker.com>
This commit is contained in:
parent
32cf776ecd
commit
e63ab14b1e
80
.github/workflows/ci.yml
vendored
80
.github/workflows/ci.yml
vendored
@ -19,7 +19,6 @@ on:
|
|||||||
default: "false"
|
default: "false"
|
||||||
|
|
||||||
env:
|
env:
|
||||||
DESTDIR: "./bin"
|
|
||||||
DOCKER_CLI_VERSION: "20.10.17"
|
DOCKER_CLI_VERSION: "20.10.17"
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
@ -103,7 +102,7 @@ jobs:
|
|||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: compose
|
name: compose
|
||||||
path: ${{ env.DESTDIR }}/*
|
path: ./bin/release/*
|
||||||
if-no-files-found: error
|
if-no-files-found: error
|
||||||
|
|
||||||
test:
|
test:
|
||||||
@ -124,13 +123,15 @@ jobs:
|
|||||||
*.cache-from=type=gha,scope=test
|
*.cache-from=type=gha,scope=test
|
||||||
*.cache-to=type=gha,scope=test
|
*.cache-to=type=gha,scope=test
|
||||||
-
|
-
|
||||||
name: Upload coverage to Codecov
|
name: Gather coverage data
|
||||||
uses: codecov/codecov-action@v3
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: coverage-data-unit
|
||||||
|
path: bin/coverage/unit/
|
||||||
|
if-no-files-found: error
|
||||||
|
|
||||||
e2e:
|
e2e:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
env:
|
|
||||||
DESTDIR: "./bin/build"
|
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
@ -179,11 +180,17 @@ jobs:
|
|||||||
name: Test plugin mode
|
name: Test plugin mode
|
||||||
if: ${{ matrix.mode == 'plugin' }}
|
if: ${{ matrix.mode == 'plugin' }}
|
||||||
run: |
|
run: |
|
||||||
rm -rf ./covdatafiles
|
rm -rf ./bin/coverage/e2e
|
||||||
mkdir ./covdatafiles
|
mkdir -p ./bin/coverage/e2e
|
||||||
make e2e-compose GOCOVERDIR=covdatafiles
|
make e2e-compose GOCOVERDIR=bin/coverage/e2e TEST_FLAGS="-v"
|
||||||
go tool covdata textfmt -i=covdatafiles -o=coverage.out
|
-
|
||||||
|
name: Gather coverage data
|
||||||
|
if: ${{ matrix.mode == 'plugin' }}
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: coverage-data-e2e
|
||||||
|
path: bin/coverage/e2e/
|
||||||
|
if-no-files-found: error
|
||||||
-
|
-
|
||||||
name: Test standalone mode
|
name: Test standalone mode
|
||||||
if: ${{ matrix.mode == 'standalone' }}
|
if: ${{ matrix.mode == 'standalone' }}
|
||||||
@ -196,9 +203,44 @@ jobs:
|
|||||||
if: ${{ matrix.mode == 'cucumber'}}
|
if: ${{ matrix.mode == 'cucumber'}}
|
||||||
run: |
|
run: |
|
||||||
make test-cucumber
|
make test-cucumber
|
||||||
-
|
|
||||||
name: Upload coverage to Codecov
|
coverage:
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
needs:
|
||||||
|
- test
|
||||||
|
- e2e
|
||||||
|
steps:
|
||||||
|
# codecov won't process the report without the source code available
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: Set up Go
|
||||||
|
uses: actions/setup-go@v4
|
||||||
|
with:
|
||||||
|
go-version-file: 'go.mod'
|
||||||
|
check-latest: true
|
||||||
|
- name: Download unit test coverage
|
||||||
|
uses: actions/download-artifact@v3
|
||||||
|
with:
|
||||||
|
name: coverage-data-unit
|
||||||
|
path: coverage/unit
|
||||||
|
- name: Download E2E test coverage
|
||||||
|
uses: actions/download-artifact@v3
|
||||||
|
with:
|
||||||
|
name: coverage-data-e2e
|
||||||
|
path: coverage/e2e
|
||||||
|
- name: Merge coverage reports
|
||||||
|
run: |
|
||||||
|
go tool covdata textfmt -i=./coverage/unit,./coverage/e2e -o ./coverage.txt
|
||||||
|
- name: Store coverage report in GitHub Actions
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: go-covdata-txt
|
||||||
|
path: ./coverage.txt
|
||||||
|
if-no-files-found: error
|
||||||
|
- name: Upload coverage to Codecov
|
||||||
uses: codecov/codecov-action@v3
|
uses: codecov/codecov-action@v3
|
||||||
|
with:
|
||||||
|
files: ./coverage.txt
|
||||||
|
|
||||||
release:
|
release:
|
||||||
permissions:
|
permissions:
|
||||||
@ -216,10 +258,10 @@ jobs:
|
|||||||
uses: actions/download-artifact@v3
|
uses: actions/download-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: compose
|
name: compose
|
||||||
path: ${{ env.DESTDIR }}
|
path: bin/release
|
||||||
-
|
-
|
||||||
name: Create checksums
|
name: Create checksums
|
||||||
working-directory: ${{ env.DESTDIR }}
|
working-directory: bin/release
|
||||||
run: |
|
run: |
|
||||||
find . -type f -print0 | sort -z | xargs -r0 shasum -a 256 -b | sed 's# \*\./# *#' > $RUNNER_TEMP/checksums.txt
|
find . -type f -print0 | sort -z | xargs -r0 shasum -a 256 -b | sed 's# \*\./# *#' > $RUNNER_TEMP/checksums.txt
|
||||||
shasum -a 256 -U -c $RUNNER_TEMP/checksums.txt
|
shasum -a 256 -U -c $RUNNER_TEMP/checksums.txt
|
||||||
@ -227,21 +269,21 @@ jobs:
|
|||||||
cat checksums.txt | while read sum file; do echo "$sum $file" > ${file#\*}.sha256; done
|
cat checksums.txt | while read sum file; do echo "$sum $file" > ${file#\*}.sha256; done
|
||||||
-
|
-
|
||||||
name: License
|
name: License
|
||||||
run: cp packaging/* ${{ env.DESTDIR }}/
|
run: cp packaging/* bin/release/
|
||||||
-
|
-
|
||||||
name: List artifacts
|
name: List artifacts
|
||||||
run: |
|
run: |
|
||||||
tree -nh ${{ env.DESTDIR }}
|
tree -nh bin/release
|
||||||
-
|
-
|
||||||
name: Check artifacts
|
name: Check artifacts
|
||||||
run: |
|
run: |
|
||||||
find ${{ env.DESTDIR }} -type f -exec file -e ascii -- {} +
|
find bin/release -type f -exec file -e ascii -- {} +
|
||||||
-
|
-
|
||||||
name: GitHub Release
|
name: GitHub Release
|
||||||
if: startsWith(github.ref, 'refs/tags/v')
|
if: startsWith(github.ref, 'refs/tags/v')
|
||||||
uses: ncipollo/release-action@58ae73b360456532aafd58ee170c045abbeaee37 # v1.10.0
|
uses: ncipollo/release-action@58ae73b360456532aafd58ee170c045abbeaee37 # v1.10.0
|
||||||
with:
|
with:
|
||||||
artifacts: ${{ env.DESTDIR }}/*
|
artifacts: bin/release/*
|
||||||
generateReleaseNotes: true
|
generateReleaseNotes: true
|
||||||
draft: true
|
draft: true
|
||||||
token: ${{ secrets.GITHUB_TOKEN }}
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
16
Dockerfile
16
Dockerfile
@ -84,8 +84,8 @@ RUN --mount=type=bind,target=. \
|
|||||||
--mount=type=bind,from=osxcross,src=/osxsdk,target=/xx-sdk \
|
--mount=type=bind,from=osxcross,src=/osxsdk,target=/xx-sdk \
|
||||||
xx-go --wrap && \
|
xx-go --wrap && \
|
||||||
if [ "$(xx-info os)" == "darwin" ]; then export CGO_ENABLED=1; fi && \
|
if [ "$(xx-info os)" == "darwin" ]; then export CGO_ENABLED=1; fi && \
|
||||||
make build GO_BUILDTAGS="$BUILD_TAGS" DESTDIR=/usr/bin && \
|
make build GO_BUILDTAGS="$BUILD_TAGS" DESTDIR=/out && \
|
||||||
xx-verify --static /usr/bin/docker-compose
|
xx-verify --static /out/docker-compose
|
||||||
|
|
||||||
FROM build-base AS lint
|
FROM build-base AS lint
|
||||||
ARG BUILD_TAGS
|
ARG BUILD_TAGS
|
||||||
@ -100,11 +100,13 @@ ARG BUILD_TAGS
|
|||||||
RUN --mount=type=bind,target=. \
|
RUN --mount=type=bind,target=. \
|
||||||
--mount=type=cache,target=/root/.cache \
|
--mount=type=cache,target=/root/.cache \
|
||||||
--mount=type=cache,target=/go/pkg/mod \
|
--mount=type=cache,target=/go/pkg/mod \
|
||||||
go test -tags "$BUILD_TAGS" -v -coverprofile=/tmp/coverage.txt -covermode=atomic $(go list $(TAGS) ./... | grep -vE 'e2e') && \
|
rm -rf /tmp/coverage && \
|
||||||
go tool cover -func=/tmp/coverage.txt
|
mkdir -p /tmp/coverage && \
|
||||||
|
go test -tags "$BUILD_TAGS" -v -cover -covermode=atomic $(go list $(TAGS) ./... | grep -vE 'e2e') -args -test.gocoverdir="/tmp/coverage" && \
|
||||||
|
go tool covdata percent -i=/tmp/coverage
|
||||||
|
|
||||||
FROM scratch AS test-coverage
|
FROM scratch AS test-coverage
|
||||||
COPY --from=test /tmp/coverage.txt /coverage.txt
|
COPY --from=test --link /tmp/coverage /
|
||||||
|
|
||||||
FROM base AS license-set
|
FROM base AS license-set
|
||||||
ARG LICENSE_FILES
|
ARG LICENSE_FILES
|
||||||
@ -162,11 +164,11 @@ RUN --mount=target=/context \
|
|||||||
EOT
|
EOT
|
||||||
|
|
||||||
FROM scratch AS binary-unix
|
FROM scratch AS binary-unix
|
||||||
COPY --link --from=build /usr/bin/docker-compose /
|
COPY --link --from=build /out/docker-compose /
|
||||||
FROM binary-unix AS binary-darwin
|
FROM binary-unix AS binary-darwin
|
||||||
FROM binary-unix AS binary-linux
|
FROM binary-unix AS binary-linux
|
||||||
FROM scratch AS binary-windows
|
FROM scratch AS binary-windows
|
||||||
COPY --link --from=build /usr/bin/docker-compose /docker-compose.exe
|
COPY --link --from=build /out/docker-compose /docker-compose.exe
|
||||||
FROM binary-$TARGETOS AS binary
|
FROM binary-$TARGETOS AS binary
|
||||||
# enable scanning for this stage
|
# enable scanning for this stage
|
||||||
ARG BUILDKIT_SBOM_SCAN_STAGE=true
|
ARG BUILDKIT_SBOM_SCAN_STAGE=true
|
||||||
|
28
Makefile
28
Makefile
@ -25,22 +25,10 @@ else
|
|||||||
DETECTED_OS = $(shell uname -s)
|
DETECTED_OS = $(shell uname -s)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
ifeq ($(DETECTED_OS),Linux)
|
|
||||||
MOBY_DOCKER=/usr/bin/docker
|
|
||||||
endif
|
|
||||||
ifeq ($(DETECTED_OS),Darwin)
|
|
||||||
MOBY_DOCKER=/Applications/Docker.app/Contents/Resources/bin/docker
|
|
||||||
endif
|
|
||||||
ifeq ($(DETECTED_OS),Windows)
|
ifeq ($(DETECTED_OS),Windows)
|
||||||
BINARY_EXT=.exe
|
BINARY_EXT=.exe
|
||||||
endif
|
endif
|
||||||
|
|
||||||
TEST_COVERAGE_FLAGS = -coverprofile=coverage.out -covermode=atomic
|
|
||||||
ifneq ($(DETECTED_OS),Windows)
|
|
||||||
# go race detector requires gcc on Windows so not used by default
|
|
||||||
# https://github.com/golang/go/issues/27089
|
|
||||||
TEST_COVERAGE_FLAGS += -race
|
|
||||||
endif
|
|
||||||
BUILD_FLAGS?=
|
BUILD_FLAGS?=
|
||||||
TEST_FLAGS?=
|
TEST_FLAGS?=
|
||||||
E2E_TEST?=
|
E2E_TEST?=
|
||||||
@ -50,13 +38,23 @@ else
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
BUILDX_CMD ?= docker buildx
|
BUILDX_CMD ?= docker buildx
|
||||||
DESTDIR ?= ./bin/build
|
|
||||||
|
# DESTDIR overrides the output path for binaries and other artifacts
|
||||||
|
# this is used by docker/docker-ce-packaging for the apt/rpm builds,
|
||||||
|
# so it's important that the resulting binary ends up EXACTLY at the
|
||||||
|
# path $DESTDIR/docker-compose when specified.
|
||||||
|
#
|
||||||
|
# See https://github.com/docker/docker-ce-packaging/blob/e43fbd37e48fde49d907b9195f23b13537521b94/rpm/SPECS/docker-compose-plugin.spec#L47
|
||||||
|
#
|
||||||
|
# By default, all artifacts go to subdirectories under ./bin/ in the
|
||||||
|
# repo root, e.g. ./bin/build, ./bin/coverage, ./bin/release.
|
||||||
|
DESTDIR ?=
|
||||||
|
|
||||||
all: build
|
all: build
|
||||||
|
|
||||||
.PHONY: build ## Build the compose cli-plugin
|
.PHONY: build ## Build the compose cli-plugin
|
||||||
build:
|
build:
|
||||||
GO111MODULE=on go build $(BUILD_FLAGS) -trimpath -tags "$(GO_BUILDTAGS)" -ldflags "$(GO_LDFLAGS)" -o "$(DESTDIR)/docker-compose$(BINARY_EXT)" ./cmd
|
GO111MODULE=on go build $(BUILD_FLAGS) -trimpath -tags "$(GO_BUILDTAGS)" -ldflags "$(GO_LDFLAGS)" -o "$(or $(DESTDIR),./bin/build)/docker-compose$(BINARY_EXT)" ./cmd
|
||||||
|
|
||||||
.PHONY: binary
|
.PHONY: binary
|
||||||
binary:
|
binary:
|
||||||
@ -69,7 +67,7 @@ binary-with-coverage:
|
|||||||
.PHONY: install
|
.PHONY: install
|
||||||
install: binary
|
install: binary
|
||||||
mkdir -p ~/.docker/cli-plugins
|
mkdir -p ~/.docker/cli-plugins
|
||||||
install bin/build/docker-compose ~/.docker/cli-plugins/docker-compose
|
install $(or $(DESTDIR),./bin/build)/docker-compose ~/.docker/cli-plugins/docker-compose
|
||||||
|
|
||||||
.PHONY: e2e-compose
|
.PHONY: e2e-compose
|
||||||
e2e-compose: ## Run end to end local tests in plugin mode. Set E2E_TEST=TestName to run a single test
|
e2e-compose: ## Run end to end local tests in plugin mode. Set E2E_TEST=TestName to run a single test
|
||||||
|
21
codecov.yml
Normal file
21
codecov.yml
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
coverage:
|
||||||
|
status:
|
||||||
|
project:
|
||||||
|
default:
|
||||||
|
informational: true
|
||||||
|
target: auto
|
||||||
|
threshold: 2%
|
||||||
|
patch:
|
||||||
|
default:
|
||||||
|
informational: true
|
||||||
|
|
||||||
|
comment:
|
||||||
|
require_changes: true
|
||||||
|
|
||||||
|
ignore:
|
||||||
|
- "packaging"
|
||||||
|
- "docs"
|
||||||
|
- "bin"
|
||||||
|
- "e2e"
|
||||||
|
- "pkg/e2e"
|
||||||
|
- "**/*_test.go"
|
@ -25,13 +25,16 @@ variable "DOCS_FORMATS" {
|
|||||||
default = "md,yaml"
|
default = "md,yaml"
|
||||||
}
|
}
|
||||||
|
|
||||||
# Defines the output folder
|
# Defines the output folder to override the default behavior.
|
||||||
|
# See Makefile for details, this is generally only useful for
|
||||||
|
# the packaging scripts and care should be taken to not break
|
||||||
|
# them.
|
||||||
variable "DESTDIR" {
|
variable "DESTDIR" {
|
||||||
default = ""
|
default = ""
|
||||||
}
|
}
|
||||||
function "bindir" {
|
function "outdir" {
|
||||||
params = [defaultdir]
|
params = [defaultdir]
|
||||||
result = DESTDIR != "" ? DESTDIR : "./bin/${defaultdir}"
|
result = DESTDIR != "" ? DESTDIR : "${defaultdir}"
|
||||||
}
|
}
|
||||||
|
|
||||||
# Special target: https://github.com/docker/metadata-action#bake-definition
|
# Special target: https://github.com/docker/metadata-action#bake-definition
|
||||||
@ -84,23 +87,23 @@ target "vendor-update" {
|
|||||||
target "test" {
|
target "test" {
|
||||||
inherits = ["_common"]
|
inherits = ["_common"]
|
||||||
target = "test-coverage"
|
target = "test-coverage"
|
||||||
output = [bindir("coverage")]
|
output = [outdir("./bin/coverage/unit")]
|
||||||
}
|
}
|
||||||
|
|
||||||
target "binary-with-coverage" {
|
target "binary-with-coverage" {
|
||||||
inherits = ["_common"]
|
inherits = ["_common"]
|
||||||
target = "binary"
|
target = "binary"
|
||||||
args = {
|
args = {
|
||||||
BUILD_FLAGS = "-cover"
|
BUILD_FLAGS = "-cover -covermode=atomic"
|
||||||
}
|
}
|
||||||
output = [bindir("build")]
|
output = [outdir("./bin/build")]
|
||||||
platforms = ["local"]
|
platforms = ["local"]
|
||||||
}
|
}
|
||||||
|
|
||||||
target "binary" {
|
target "binary" {
|
||||||
inherits = ["_common"]
|
inherits = ["_common"]
|
||||||
target = "binary"
|
target = "binary"
|
||||||
output = [bindir("build")]
|
output = [outdir("./bin/build")]
|
||||||
platforms = ["local"]
|
platforms = ["local"]
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -124,7 +127,7 @@ target "binary-cross" {
|
|||||||
target "release" {
|
target "release" {
|
||||||
inherits = ["binary-cross"]
|
inherits = ["binary-cross"]
|
||||||
target = "release"
|
target = "release"
|
||||||
output = [bindir("release")]
|
output = [outdir("./bin/release")]
|
||||||
}
|
}
|
||||||
|
|
||||||
target "docs-validate" {
|
target "docs-validate" {
|
||||||
|
@ -20,6 +20,7 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"io/fs"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
@ -134,7 +135,7 @@ func initializePlugins(t testing.TB, configDir string) {
|
|||||||
require.NoError(t, os.MkdirAll(filepath.Join(configDir, "cli-plugins"), 0o755),
|
require.NoError(t, os.MkdirAll(filepath.Join(configDir, "cli-plugins"), 0o755),
|
||||||
"Failed to create cli-plugins directory")
|
"Failed to create cli-plugins directory")
|
||||||
composePlugin, err := findExecutable(DockerComposeExecutableName)
|
composePlugin, err := findExecutable(DockerComposeExecutableName)
|
||||||
if os.IsNotExist(err) {
|
if errors.Is(err, fs.ErrNotExist) {
|
||||||
t.Logf("WARNING: docker-compose cli-plugin not found")
|
t.Logf("WARNING: docker-compose cli-plugin not found")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -161,20 +162,21 @@ func dirContents(dir string) []string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func findExecutable(executableName string) (string, error) {
|
func findExecutable(executableName string) (string, error) {
|
||||||
_, filename, _, _ := runtime.Caller(0)
|
bin := os.Getenv("COMPOSE_E2E_BIN_PATH")
|
||||||
root := filepath.Join(filepath.Dir(filename), "..", "..")
|
if bin == "" {
|
||||||
buildPath := filepath.Join(root, "bin", "build")
|
_, filename, _, _ := runtime.Caller(0)
|
||||||
|
buildPath := filepath.Join(filepath.Dir(filename), "..", "..", "bin", "build")
|
||||||
bin, err := filepath.Abs(filepath.Join(buildPath, executableName))
|
var err error
|
||||||
if err != nil {
|
bin, err = filepath.Abs(filepath.Join(buildPath, executableName))
|
||||||
return "", err
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, err := os.Stat(bin); err == nil {
|
if _, err := os.Stat(bin); err == nil {
|
||||||
return bin, nil
|
return bin, nil
|
||||||
}
|
}
|
||||||
|
return "", fmt.Errorf("looking for %q: %w", bin, fs.ErrNotExist)
|
||||||
return "", errors.Wrap(os.ErrNotExist, "executable not found")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func findPluginExecutable(pluginExecutableName string) (string, error) {
|
func findPluginExecutable(pluginExecutableName string) (string, error) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user