| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436 |
- /*
- 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 compose
- import (
- "context"
- "fmt"
- "strings"
- "testing"
- "github.com/compose-spec/compose-go/v2/types"
- "github.com/docker/cli/cli/config/configfile"
- moby "github.com/docker/docker/api/types"
- containerType "github.com/docker/docker/api/types/container"
- "github.com/docker/docker/api/types/filters"
- "github.com/docker/docker/api/types/network"
- "github.com/docker/go-connections/nat"
- "go.uber.org/mock/gomock"
- "gotest.tools/v3/assert"
- "github.com/docker/compose/v2/pkg/api"
- "github.com/docker/compose/v2/pkg/mocks"
- "github.com/docker/compose/v2/pkg/progress"
- )
- func TestContainerName(t *testing.T) {
- s := types.ServiceConfig{
- Name: "testservicename",
- ContainerName: "testcontainername",
- Scale: intPtr(1),
- Deploy: &types.DeployConfig{},
- }
- ret, err := getScale(s)
- assert.NilError(t, err)
- assert.Equal(t, ret, *s.Scale)
- s.Scale = intPtr(0)
- ret, err = getScale(s)
- assert.NilError(t, err)
- assert.Equal(t, ret, *s.Scale)
- s.Scale = intPtr(2)
- _, err = getScale(s)
- assert.Error(t, err, fmt.Sprintf(doubledContainerNameWarning, s.Name, s.ContainerName))
- }
- func intPtr(i int) *int {
- return &i
- }
- func TestServiceLinks(t *testing.T) {
- const dbContainerName = "/" + testProject + "-db-1"
- const webContainerName = "/" + testProject + "-web-1"
- s := types.ServiceConfig{
- Name: "web",
- Scale: intPtr(1),
- }
- containerListOptions := containerType.ListOptions{
- Filters: filters.NewArgs(
- projectFilter(testProject),
- serviceFilter("db"),
- oneOffFilter(false),
- hasConfigHashLabel(),
- ),
- All: true,
- }
- t.Run("service links default", func(t *testing.T) {
- mockCtrl := gomock.NewController(t)
- defer mockCtrl.Finish()
- apiClient := mocks.NewMockAPIClient(mockCtrl)
- cli := mocks.NewMockCli(mockCtrl)
- tested := composeService{
- dockerCli: cli,
- }
- cli.EXPECT().Client().Return(apiClient).AnyTimes()
- s.Links = []string{"db"}
- c := testContainer("db", dbContainerName, false)
- apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptions).Return([]moby.Container{c}, nil)
- links, err := tested.getLinks(context.Background(), testProject, s, 1)
- assert.NilError(t, err)
- assert.Equal(t, len(links), 3)
- assert.Equal(t, links[0], "testProject-db-1:db")
- assert.Equal(t, links[1], "testProject-db-1:db-1")
- assert.Equal(t, links[2], "testProject-db-1:testProject-db-1")
- })
- t.Run("service links", func(t *testing.T) {
- mockCtrl := gomock.NewController(t)
- defer mockCtrl.Finish()
- apiClient := mocks.NewMockAPIClient(mockCtrl)
- cli := mocks.NewMockCli(mockCtrl)
- tested := composeService{
- dockerCli: cli,
- }
- cli.EXPECT().Client().Return(apiClient).AnyTimes()
- s.Links = []string{"db:db"}
- c := testContainer("db", dbContainerName, false)
- apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptions).Return([]moby.Container{c}, nil)
- links, err := tested.getLinks(context.Background(), testProject, s, 1)
- assert.NilError(t, err)
- assert.Equal(t, len(links), 3)
- assert.Equal(t, links[0], "testProject-db-1:db")
- assert.Equal(t, links[1], "testProject-db-1:db-1")
- assert.Equal(t, links[2], "testProject-db-1:testProject-db-1")
- })
- t.Run("service links name", func(t *testing.T) {
- mockCtrl := gomock.NewController(t)
- defer mockCtrl.Finish()
- apiClient := mocks.NewMockAPIClient(mockCtrl)
- cli := mocks.NewMockCli(mockCtrl)
- tested := composeService{
- dockerCli: cli,
- }
- cli.EXPECT().Client().Return(apiClient).AnyTimes()
- s.Links = []string{"db:dbname"}
- c := testContainer("db", dbContainerName, false)
- apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptions).Return([]moby.Container{c}, nil)
- links, err := tested.getLinks(context.Background(), testProject, s, 1)
- assert.NilError(t, err)
- assert.Equal(t, len(links), 3)
- assert.Equal(t, links[0], "testProject-db-1:dbname")
- assert.Equal(t, links[1], "testProject-db-1:db-1")
- assert.Equal(t, links[2], "testProject-db-1:testProject-db-1")
- })
- t.Run("service links external links", func(t *testing.T) {
- mockCtrl := gomock.NewController(t)
- defer mockCtrl.Finish()
- apiClient := mocks.NewMockAPIClient(mockCtrl)
- cli := mocks.NewMockCli(mockCtrl)
- tested := composeService{
- dockerCli: cli,
- }
- cli.EXPECT().Client().Return(apiClient).AnyTimes()
- s.Links = []string{"db:dbname"}
- s.ExternalLinks = []string{"db1:db2"}
- c := testContainer("db", dbContainerName, false)
- apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptions).Return([]moby.Container{c}, nil)
- links, err := tested.getLinks(context.Background(), testProject, s, 1)
- assert.NilError(t, err)
- assert.Equal(t, len(links), 4)
- assert.Equal(t, links[0], "testProject-db-1:dbname")
- assert.Equal(t, links[1], "testProject-db-1:db-1")
- assert.Equal(t, links[2], "testProject-db-1:testProject-db-1")
- // ExternalLink
- assert.Equal(t, links[3], "db1:db2")
- })
- t.Run("service links itself oneoff", func(t *testing.T) {
- mockCtrl := gomock.NewController(t)
- defer mockCtrl.Finish()
- apiClient := mocks.NewMockAPIClient(mockCtrl)
- cli := mocks.NewMockCli(mockCtrl)
- tested := composeService{
- dockerCli: cli,
- }
- cli.EXPECT().Client().Return(apiClient).AnyTimes()
- s.Links = []string{}
- s.ExternalLinks = []string{}
- s.Labels = s.Labels.Add(api.OneoffLabel, "True")
- c := testContainer("web", webContainerName, true)
- containerListOptionsOneOff := containerType.ListOptions{
- Filters: filters.NewArgs(
- projectFilter(testProject),
- serviceFilter("web"),
- oneOffFilter(false),
- hasConfigHashLabel(),
- ),
- All: true,
- }
- apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptionsOneOff).Return([]moby.Container{c}, nil)
- links, err := tested.getLinks(context.Background(), testProject, s, 1)
- assert.NilError(t, err)
- assert.Equal(t, len(links), 3)
- assert.Equal(t, links[0], "testProject-web-1:web")
- assert.Equal(t, links[1], "testProject-web-1:web-1")
- assert.Equal(t, links[2], "testProject-web-1:testProject-web-1")
- })
- }
- func TestWaitDependencies(t *testing.T) {
- mockCtrl := gomock.NewController(t)
- defer mockCtrl.Finish()
- apiClient := mocks.NewMockAPIClient(mockCtrl)
- cli := mocks.NewMockCli(mockCtrl)
- tested := composeService{
- dockerCli: cli,
- }
- cli.EXPECT().Client().Return(apiClient).AnyTimes()
- t.Run("should skip dependencies with scale 0", func(t *testing.T) {
- dbService := types.ServiceConfig{Name: "db", Scale: intPtr(0)}
- redisService := types.ServiceConfig{Name: "redis", Scale: intPtr(0)}
- project := types.Project{Name: strings.ToLower(testProject), Services: types.Services{
- "db": dbService,
- "redis": redisService,
- }}
- dependencies := types.DependsOnConfig{
- "db": {Condition: ServiceConditionRunningOrHealthy},
- "redis": {Condition: ServiceConditionRunningOrHealthy},
- }
- assert.NilError(t, tested.waitDependencies(context.Background(), &project, "", dependencies, nil, 0))
- })
- t.Run("should skip dependencies with condition service_started", func(t *testing.T) {
- dbService := types.ServiceConfig{Name: "db", Scale: intPtr(1)}
- redisService := types.ServiceConfig{Name: "redis", Scale: intPtr(1)}
- project := types.Project{Name: strings.ToLower(testProject), Services: types.Services{
- "db": dbService,
- "redis": redisService,
- }}
- dependencies := types.DependsOnConfig{
- "db": {Condition: types.ServiceConditionStarted, Required: true},
- "redis": {Condition: types.ServiceConditionStarted, Required: true},
- }
- assert.NilError(t, tested.waitDependencies(context.Background(), &project, "", dependencies, nil, 0))
- })
- }
- func TestCreateMobyContainer(t *testing.T) {
- t.Run("connects container networks one by one if API <1.44", func(t *testing.T) {
- mockCtrl := gomock.NewController(t)
- defer mockCtrl.Finish()
- apiClient := mocks.NewMockAPIClient(mockCtrl)
- cli := mocks.NewMockCli(mockCtrl)
- tested := composeService{
- dockerCli: cli,
- }
- cli.EXPECT().Client().Return(apiClient).AnyTimes()
- cli.EXPECT().ConfigFile().Return(&configfile.ConfigFile{}).AnyTimes()
- apiClient.EXPECT().DaemonHost().Return("").AnyTimes()
- apiClient.EXPECT().ImageInspectWithRaw(gomock.Any(), gomock.Any()).Return(moby.ImageInspect{}, nil, nil).AnyTimes()
- // force `RuntimeVersion` to fetch again
- runtimeVersion = runtimeVersionCache{}
- apiClient.EXPECT().ServerVersion(gomock.Any()).Return(moby.Version{
- APIVersion: "1.43",
- }, nil).AnyTimes()
- service := types.ServiceConfig{
- Name: "test",
- Networks: map[string]*types.ServiceNetworkConfig{
- "a": {
- Priority: 10,
- },
- "b": {
- Priority: 100,
- },
- },
- }
- project := types.Project{
- Name: "bork",
- Services: types.Services{
- "test": service,
- },
- Networks: types.Networks{
- "a": types.NetworkConfig{
- Name: "a-moby-name",
- },
- "b": types.NetworkConfig{
- Name: "b-moby-name",
- },
- },
- }
- var falseBool bool
- apiClient.EXPECT().ContainerCreate(gomock.Any(), gomock.Any(), gomock.Eq(
- &containerType.HostConfig{
- PortBindings: nat.PortMap{},
- ExtraHosts: []string{},
- Tmpfs: map[string]string{},
- Resources: containerType.Resources{
- OomKillDisable: &falseBool,
- },
- NetworkMode: "b-moby-name",
- }), gomock.Eq(
- &network.NetworkingConfig{
- EndpointsConfig: map[string]*network.EndpointSettings{
- "b-moby-name": {
- IPAMConfig: &network.EndpointIPAMConfig{},
- Aliases: []string{"bork-test-0"},
- },
- },
- }), gomock.Any(), gomock.Any()).Times(1).Return(
- containerType.CreateResponse{
- ID: "an-id",
- }, nil)
- apiClient.EXPECT().ContainerInspect(gomock.Any(), gomock.Eq("an-id")).Times(1).Return(
- moby.ContainerJSON{
- ContainerJSONBase: &moby.ContainerJSONBase{
- ID: "an-id",
- Name: "a-name",
- },
- Config: &containerType.Config{},
- NetworkSettings: &moby.NetworkSettings{},
- }, nil)
- apiClient.EXPECT().NetworkConnect(gomock.Any(), "a-moby-name", "an-id", gomock.Eq(
- &network.EndpointSettings{
- IPAMConfig: &network.EndpointIPAMConfig{},
- Aliases: []string{"bork-test-0"},
- }))
- _, err := tested.createMobyContainer(context.Background(), &project, service, "test", 0, nil, createOptions{
- Labels: make(types.Labels),
- }, progress.ContextWriter(context.TODO()))
- assert.NilError(t, err)
- })
- t.Run("includes all container networks in ContainerCreate call if API >=1.44", func(t *testing.T) {
- mockCtrl := gomock.NewController(t)
- defer mockCtrl.Finish()
- apiClient := mocks.NewMockAPIClient(mockCtrl)
- cli := mocks.NewMockCli(mockCtrl)
- tested := composeService{
- dockerCli: cli,
- }
- cli.EXPECT().Client().Return(apiClient).AnyTimes()
- cli.EXPECT().ConfigFile().Return(&configfile.ConfigFile{}).AnyTimes()
- apiClient.EXPECT().DaemonHost().Return("").AnyTimes()
- apiClient.EXPECT().ImageInspectWithRaw(gomock.Any(), gomock.Any()).Return(moby.ImageInspect{}, nil, nil).AnyTimes()
- // force `RuntimeVersion` to fetch fresh version
- runtimeVersion = runtimeVersionCache{}
- apiClient.EXPECT().ServerVersion(gomock.Any()).Return(moby.Version{
- APIVersion: "1.44",
- }, nil).AnyTimes()
- service := types.ServiceConfig{
- Name: "test",
- Networks: map[string]*types.ServiceNetworkConfig{
- "a": {
- Priority: 10,
- },
- "b": {
- Priority: 100,
- },
- },
- }
- project := types.Project{
- Name: "bork",
- Services: types.Services{
- "test": service,
- },
- Networks: types.Networks{
- "a": types.NetworkConfig{
- Name: "a-moby-name",
- },
- "b": types.NetworkConfig{
- Name: "b-moby-name",
- },
- },
- }
- var falseBool bool
- apiClient.EXPECT().ContainerCreate(gomock.Any(), gomock.Any(), gomock.Eq(
- &containerType.HostConfig{
- PortBindings: nat.PortMap{},
- ExtraHosts: []string{},
- Tmpfs: map[string]string{},
- Resources: containerType.Resources{
- OomKillDisable: &falseBool,
- },
- NetworkMode: "b-moby-name",
- }), gomock.Eq(
- &network.NetworkingConfig{
- EndpointsConfig: map[string]*network.EndpointSettings{
- "a-moby-name": {
- IPAMConfig: &network.EndpointIPAMConfig{},
- Aliases: []string{"bork-test-0"},
- },
- "b-moby-name": {
- IPAMConfig: &network.EndpointIPAMConfig{},
- Aliases: []string{"bork-test-0"},
- },
- },
- }), gomock.Any(), gomock.Any()).Times(1).Return(
- containerType.CreateResponse{
- ID: "an-id",
- }, nil)
- apiClient.EXPECT().ContainerInspect(gomock.Any(), gomock.Eq("an-id")).Times(1).Return(
- moby.ContainerJSON{
- ContainerJSONBase: &moby.ContainerJSONBase{
- ID: "an-id",
- Name: "a-name",
- },
- Config: &containerType.Config{},
- NetworkSettings: &moby.NetworkSettings{},
- }, nil)
- _, err := tested.createMobyContainer(context.Background(), &project, service, "test", 0, nil, createOptions{
- Labels: make(types.Labels),
- }, progress.ContextWriter(context.TODO()))
- assert.NilError(t, err)
- })
- }
|