| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417 |
- /*
- Copyright 2020 Docker Compose CLI 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.
- */
- package e2e
- import (
- "net/http"
- "runtime"
- "strings"
- "testing"
- "time"
- "github.com/stretchr/testify/require"
- "gotest.tools/v3/assert"
- "gotest.tools/v3/icmd"
- )
- func TestLocalComposeBuild(t *testing.T) {
- c := NewParallelCLI(t)
- t.Run("build named and unnamed images", func(t *testing.T) {
- // ensure local test run does not reuse previously build image
- c.RunDockerOrExitError(t, "rmi", "build-test-nginx")
- c.RunDockerOrExitError(t, "rmi", "custom-nginx")
- res := c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test", "build")
- res.Assert(t, icmd.Expected{Out: "COPY static /usr/share/nginx/html"})
- c.RunDockerCmd(t, "image", "inspect", "build-test-nginx")
- c.RunDockerCmd(t, "image", "inspect", "custom-nginx")
- })
- t.Run("build with build-arg", func(t *testing.T) {
- // ensure local test run does not reuse previously build image
- c.RunDockerOrExitError(t, "rmi", "build-test-nginx")
- c.RunDockerOrExitError(t, "rmi", "custom-nginx")
- c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test", "build", "--build-arg", "FOO=BAR")
- res := c.RunDockerCmd(t, "image", "inspect", "build-test-nginx")
- res.Assert(t, icmd.Expected{Out: `"FOO": "BAR"`})
- })
- t.Run("build with build-arg set by env", func(t *testing.T) {
- // ensure local test run does not reuse previously build image
- c.RunDockerOrExitError(t, "rmi", "build-test-nginx")
- c.RunDockerOrExitError(t, "rmi", "custom-nginx")
- icmd.RunCmd(c.NewDockerComposeCmd(t,
- "--project-directory",
- "fixtures/build-test",
- "build",
- "--build-arg",
- "FOO"),
- func(cmd *icmd.Cmd) {
- cmd.Env = append(cmd.Env, "FOO=BAR")
- })
- res := c.RunDockerCmd(t, "image", "inspect", "build-test-nginx")
- res.Assert(t, icmd.Expected{Out: `"FOO": "BAR"`})
- })
- t.Run("build with multiple build-args ", func(t *testing.T) {
- // ensure local test run does not reuse previously build image
- c.RunDockerOrExitError(t, "rmi", "-f", "multi-args-multiargs")
- cmd := c.NewDockerComposeCmd(t, "--project-directory", "fixtures/build-test/multi-args", "build")
- icmd.RunCmd(cmd, func(cmd *icmd.Cmd) {
- cmd.Env = append(cmd.Env, "DOCKER_BUILDKIT=0")
- })
- res := c.RunDockerCmd(t, "image", "inspect", "multi-args-multiargs")
- res.Assert(t, icmd.Expected{Out: `"RESULT": "SUCCESS"`})
- })
- t.Run("build as part of up", func(t *testing.T) {
- c.RunDockerOrExitError(t, "rmi", "build-test-nginx")
- c.RunDockerOrExitError(t, "rmi", "custom-nginx")
- res := c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test", "up", "-d")
- t.Cleanup(func() {
- c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test", "down")
- })
- res.Assert(t, icmd.Expected{Out: "COPY static /usr/share/nginx/html"})
- res.Assert(t, icmd.Expected{Out: "COPY static2 /usr/share/nginx/html"})
- output := HTTPGetWithRetry(t, "http://localhost:8070", http.StatusOK, 2*time.Second, 20*time.Second)
- assert.Assert(t, strings.Contains(output, "Hello from Nginx container"))
- c.RunDockerCmd(t, "image", "inspect", "build-test-nginx")
- c.RunDockerCmd(t, "image", "inspect", "custom-nginx")
- })
- t.Run("no rebuild when up again", func(t *testing.T) {
- res := c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test", "up", "-d")
- assert.Assert(t, !strings.Contains(res.Stdout(), "COPY static"), res.Stdout())
- })
- t.Run("rebuild when up --build", func(t *testing.T) {
- res := c.RunDockerComposeCmd(t, "--workdir", "fixtures/build-test", "up", "-d", "--build")
- res.Assert(t, icmd.Expected{Out: "COPY static /usr/share/nginx/html"})
- res.Assert(t, icmd.Expected{Out: "COPY static2 /usr/share/nginx/html"})
- })
- t.Run("cleanup build project", func(t *testing.T) {
- c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test", "down")
- c.RunDockerCmd(t, "rmi", "build-test-nginx")
- c.RunDockerCmd(t, "rmi", "custom-nginx")
- })
- }
- func TestBuildSSH(t *testing.T) {
- if runtime.GOOS == "windows" {
- t.Skip("Running on Windows. Skipping...")
- }
- c := NewParallelCLI(t)
- t.Run("build failed with ssh default value", func(t *testing.T) {
- res := c.RunDockerComposeCmdNoCheck(t, "--project-directory", "fixtures/build-test", "build", "--ssh", "")
- res.Assert(t, icmd.Expected{
- ExitCode: 1,
- Err: "invalid empty ssh agent socket: make sure SSH_AUTH_SOCK is set",
- })
- })
- t.Run("build succeed with ssh from Compose file", func(t *testing.T) {
- c.RunDockerOrExitError(t, "rmi", "build-test-ssh")
- c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test/ssh", "build")
- c.RunDockerCmd(t, "image", "inspect", "build-test-ssh")
- })
- t.Run("build succeed with ssh from CLI", func(t *testing.T) {
- c.RunDockerOrExitError(t, "rmi", "build-test-ssh")
- c.RunDockerComposeCmd(t, "-f", "fixtures/build-test/ssh/compose-without-ssh.yaml", "--project-directory",
- "fixtures/build-test/ssh", "build", "--no-cache", "--ssh", "fake-ssh=./fixtures/build-test/ssh/fake_rsa")
- c.RunDockerCmd(t, "image", "inspect", "build-test-ssh")
- })
- t.Run("build failed with wrong ssh key id from CLI", func(t *testing.T) {
- c.RunDockerOrExitError(t, "rmi", "build-test-ssh")
- res := c.RunDockerComposeCmdNoCheck(t, "-f", "fixtures/build-test/ssh/compose-without-ssh.yaml",
- "--project-directory", "fixtures/build-test/ssh", "build", "--no-cache", "--ssh",
- "wrong-ssh=./fixtures/build-test/ssh/fake_rsa")
- res.Assert(t, icmd.Expected{
- ExitCode: 17,
- Err: "failed to solve: rpc error: code = Unknown desc = unset ssh forward key fake-ssh",
- })
- })
- t.Run("build succeed as part of up with ssh from Compose file", func(t *testing.T) {
- c.RunDockerOrExitError(t, "rmi", "build-test-ssh")
- c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test/ssh", "up", "-d", "--build")
- t.Cleanup(func() {
- c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test/ssh", "down")
- })
- c.RunDockerCmd(t, "image", "inspect", "build-test-ssh")
- })
- }
- func TestBuildSecrets(t *testing.T) {
- if runtime.GOOS == "windows" {
- t.Skip("skipping test on windows")
- }
- c := NewParallelCLI(t)
- t.Run("build with secrets", func(t *testing.T) {
- // ensure local test run does not reuse previously build image
- c.RunDockerOrExitError(t, "rmi", "build-test-secret")
- cmd := c.NewDockerComposeCmd(t, "--project-directory", "fixtures/build-test/secrets", "build")
- res := icmd.RunCmd(cmd, func(cmd *icmd.Cmd) {
- cmd.Env = append(cmd.Env, "SOME_SECRET=bar")
- })
- res.Assert(t, icmd.Success)
- })
- }
- func TestBuildTags(t *testing.T) {
- c := NewParallelCLI(t)
- t.Run("build with tags", func(t *testing.T) {
- // ensure local test run does not reuse previously build image
- c.RunDockerOrExitError(t, "rmi", "build-test-tags")
- c.RunDockerComposeCmd(t, "--project-directory", "./fixtures/build-test/tags", "build", "--no-cache")
- res := c.RunDockerCmd(t, "image", "inspect", "build-test-tags")
- expectedOutput := `"RepoTags": [
- "docker/build-test-tags:1.0.0",
- "build-test-tags:latest",
- "other-image-name:v1.0.0"
- ],
- `
- res.Assert(t, icmd.Expected{Out: expectedOutput})
- })
- }
- func TestBuildImageDependencies(t *testing.T) {
- doTest := func(t *testing.T, cli *CLI) {
- resetState := func() {
- cli.RunDockerComposeCmd(t, "down", "--rmi=all", "-t=0")
- res := cli.RunDockerOrExitError(t, "image", "rm", "build-dependencies-service")
- if res.Error != nil {
- require.Contains(t, res.Stderr(), `Error: No such image: build-dependencies-service`)
- }
- }
- resetState()
- t.Cleanup(resetState)
- // the image should NOT exist now
- res := cli.RunDockerOrExitError(t, "image", "inspect", "build-dependencies-service")
- res.Assert(t, icmd.Expected{
- ExitCode: 1,
- Err: "Error: No such image: build-dependencies-service",
- })
- res = cli.RunDockerComposeCmd(t, "build")
- t.Log(res.Combined())
- res = cli.RunDockerCmd(t,
- "image", "inspect", "--format={{ index .RepoTags 0 }}",
- "build-dependencies-service")
- res.Assert(t, icmd.Expected{Out: "build-dependencies-service:latest"})
- res = cli.RunDockerComposeCmd(t, "down", "-t0", "--rmi=all", "--remove-orphans")
- t.Log(res.Combined())
- res = cli.RunDockerOrExitError(t, "image", "inspect", "build-dependencies-service")
- res.Assert(t, icmd.Expected{
- ExitCode: 1,
- Err: "Error: No such image: build-dependencies-service",
- })
- }
- t.Run("ClassicBuilder", func(t *testing.T) {
- cli := NewParallelCLI(t, WithEnv(
- "DOCKER_BUILDKIT=0",
- "COMPOSE_FILE=./fixtures/build-dependencies/compose.yaml",
- ))
- doTest(t, cli)
- })
- t.Run("BuildKit", func(t *testing.T) {
- t.Skip("See https://github.com/docker/compose/issues/9232")
- })
- }
- func TestBuildPlatformsWithCorrectBuildxConfig(t *testing.T) {
- if runtime.GOOS == "windows" {
- t.Skip("Running on Windows. Skipping...")
- }
- c := NewParallelCLI(t)
- // declare builder
- result := c.RunDockerCmd(t, "buildx", "create", "--name", "build-platform", "--use", "--bootstrap")
- assert.NilError(t, result.Error)
- t.Cleanup(func() {
- c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test/platforms", "down")
- _ = c.RunDockerCmd(t, "buildx", "rm", "-f", "build-platform")
- })
- t.Run("platform not supported by builder", func(t *testing.T) {
- res := c.RunDockerComposeCmdNoCheck(t, "--project-directory", "fixtures/build-test/platforms",
- "-f", "fixtures/build-test/platforms/compose-unsupported-platform.yml", "build")
- res.Assert(t, icmd.Expected{
- ExitCode: 17,
- Err: "failed to solve: alpine: no match for platform in",
- })
- })
- t.Run("multi-arch build ok", func(t *testing.T) {
- res := c.RunDockerComposeCmdNoCheck(t, "--project-directory", "fixtures/build-test/platforms", "build")
- assert.NilError(t, res.Error, res.Stderr())
- res.Assert(t, icmd.Expected{Out: "I am building for linux/arm64"})
- res.Assert(t, icmd.Expected{Out: "I am building for linux/amd64"})
- })
- t.Run("multi-arch multi service builds ok", func(t *testing.T) {
- res := c.RunDockerComposeCmdNoCheck(t, "--project-directory", "fixtures/build-test/platforms",
- "-f", "fixtures/build-test/platforms/compose-multiple-platform-builds.yaml", "build")
- assert.NilError(t, res.Error, res.Stderr())
- res.Assert(t, icmd.Expected{Out: "I'm Service A and I am building for linux/arm64"})
- res.Assert(t, icmd.Expected{Out: "I'm Service A and I am building for linux/amd64"})
- res.Assert(t, icmd.Expected{Out: "I'm Service B and I am building for linux/arm64"})
- res.Assert(t, icmd.Expected{Out: "I'm Service B and I am building for linux/amd64"})
- res.Assert(t, icmd.Expected{Out: "I'm Service C and I am building for linux/arm64"})
- res.Assert(t, icmd.Expected{Out: "I'm Service C and I am building for linux/amd64"})
- })
- t.Run("multi-arch up --build", func(t *testing.T) {
- res := c.RunDockerComposeCmdNoCheck(t, "--project-directory", "fixtures/build-test/platforms", "up", "--build")
- assert.NilError(t, res.Error, res.Stderr())
- res.Assert(t, icmd.Expected{Out: "platforms-platforms-1 exited with code 0"})
- })
- t.Run("use DOCKER_DEFAULT_PLATFORM value when up --build", func(t *testing.T) {
- cmd := c.NewDockerComposeCmd(t, "--project-directory", "fixtures/build-test/platforms", "up", "--build")
- res := icmd.RunCmd(cmd, func(cmd *icmd.Cmd) {
- cmd.Env = append(cmd.Env, "DOCKER_DEFAULT_PLATFORM=linux/amd64")
- })
- assert.NilError(t, res.Error, res.Stderr())
- res.Assert(t, icmd.Expected{Out: "I am building for linux/amd64"})
- assert.Assert(t, !strings.Contains(res.Stdout(), "I am building for linux/arm64"))
- })
- t.Run("use service platform value when no build platforms defined ", func(t *testing.T) {
- res := c.RunDockerComposeCmdNoCheck(t, "--project-directory", "fixtures/build-test/platforms",
- "-f", "fixtures/build-test/platforms/compose-service-platform-and-no-build-platforms.yaml", "build")
- assert.NilError(t, res.Error, res.Stderr())
- res.Assert(t, icmd.Expected{Out: "I am building for linux/386"})
- })
- }
- func TestBuildPrivileged(t *testing.T) {
- c := NewParallelCLI(t)
- // declare builder
- result := c.RunDockerCmd(t, "buildx", "create", "--name", "build-privileged", "--use", "--bootstrap", "--buildkitd-flags",
- `'--allow-insecure-entitlement=security.insecure'`)
- assert.NilError(t, result.Error)
- t.Cleanup(func() {
- c.RunDockerComposeCmd(t, "--project-directory", "fixtures/build-test/privileged", "down")
- _ = c.RunDockerCmd(t, "buildx", "rm", "-f", "build-privileged")
- })
- t.Run("use build privileged mode to run insecure build command", func(t *testing.T) {
- res := c.RunDockerComposeCmdNoCheck(t, "--project-directory", "fixtures/build-test/privileged", "build")
- assert.NilError(t, res.Error, res.Stderr())
- res.Assert(t, icmd.Expected{Out: "CapEff:\t0000003fffffffff"})
- })
- }
- func TestBuildPlatformsStandardErrors(t *testing.T) {
- c := NewParallelCLI(t)
- t.Run("no platform support with Classic Builder", func(t *testing.T) {
- cmd := c.NewDockerComposeCmd(t, "--project-directory", "fixtures/build-test/platforms", "build")
- res := icmd.RunCmd(cmd, func(cmd *icmd.Cmd) {
- cmd.Env = append(cmd.Env, "DOCKER_BUILDKIT=0")
- })
- res.Assert(t, icmd.Expected{
- ExitCode: 1,
- Err: "this builder doesn't support multi-arch build, set DOCKER_BUILDKIT=1 to use multi-arch builder",
- })
- })
- t.Run("builder does not support multi-arch", func(t *testing.T) {
- res := c.RunDockerComposeCmdNoCheck(t, "--project-directory", "fixtures/build-test/platforms", "build")
- res.Assert(t, icmd.Expected{
- ExitCode: 17,
- Err: `multiple platforms feature is currently not supported for docker driver. Please switch to a different driver (eg. "docker buildx create --use")`,
- })
- })
- t.Run("service platform not defined in platforms build section", func(t *testing.T) {
- res := c.RunDockerComposeCmdNoCheck(t, "--project-directory", "fixtures/build-test/platforms",
- "-f", "fixtures/build-test/platforms/compose-service-platform-not-in-build-platforms.yaml", "build")
- res.Assert(t, icmd.Expected{
- ExitCode: 1,
- Err: `service.platform "linux/riscv64" should be part of the service.build.platforms: ["linux/amd64" "linux/arm64"]`,
- })
- })
- t.Run("DOCKER_DEFAULT_PLATFORM value not defined in platforms build section", func(t *testing.T) {
- cmd := c.NewDockerComposeCmd(t, "--project-directory", "fixtures/build-test/platforms", "build")
- res := icmd.RunCmd(cmd, func(cmd *icmd.Cmd) {
- cmd.Env = append(cmd.Env, "DOCKER_DEFAULT_PLATFORM=windows/amd64")
- })
- res.Assert(t, icmd.Expected{
- ExitCode: 1,
- Err: `DOCKER_DEFAULT_PLATFORM "windows/amd64" value should be part of the service.build.platforms: ["linux/amd64" "linux/arm64"]`,
- })
- })
- t.Run("no privileged support with Classic Builder", func(t *testing.T) {
- cmd := c.NewDockerComposeCmd(t, "--project-directory", "fixtures/build-test/privileged", "build")
- res := icmd.RunCmd(cmd, func(cmd *icmd.Cmd) {
- cmd.Env = append(cmd.Env, "DOCKER_BUILDKIT=0")
- })
- res.Assert(t, icmd.Expected{
- ExitCode: 1,
- Err: "this builder doesn't support privileged mode, set DOCKER_BUILDKIT=1 to use builder supporting privileged mode",
- })
- })
- }
|