convergence_test.go 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. /*
  2. Copyright 2020 Docker Compose CLI authors
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package compose
  14. import (
  15. "context"
  16. "fmt"
  17. "strings"
  18. "testing"
  19. "github.com/compose-spec/compose-go/v2/types"
  20. moby "github.com/docker/docker/api/types"
  21. containerType "github.com/docker/docker/api/types/container"
  22. "github.com/docker/docker/api/types/filters"
  23. "go.uber.org/mock/gomock"
  24. "gotest.tools/v3/assert"
  25. "github.com/docker/compose/v2/pkg/api"
  26. "github.com/docker/compose/v2/pkg/mocks"
  27. )
  28. func TestContainerName(t *testing.T) {
  29. s := types.ServiceConfig{
  30. Name: "testservicename",
  31. ContainerName: "testcontainername",
  32. Scale: intPtr(1),
  33. Deploy: &types.DeployConfig{},
  34. }
  35. ret, err := getScale(s)
  36. assert.NilError(t, err)
  37. assert.Equal(t, ret, *s.Scale)
  38. s.Scale = intPtr(0)
  39. ret, err = getScale(s)
  40. assert.NilError(t, err)
  41. assert.Equal(t, ret, *s.Scale)
  42. s.Scale = intPtr(2)
  43. _, err = getScale(s)
  44. assert.Error(t, err, fmt.Sprintf(doubledContainerNameWarning, s.Name, s.ContainerName))
  45. }
  46. func intPtr(i int) *int {
  47. return &i
  48. }
  49. func TestServiceLinks(t *testing.T) {
  50. const dbContainerName = "/" + testProject + "-db-1"
  51. const webContainerName = "/" + testProject + "-web-1"
  52. s := types.ServiceConfig{
  53. Name: "web",
  54. Scale: intPtr(1),
  55. }
  56. containerListOptions := containerType.ListOptions{
  57. Filters: filters.NewArgs(
  58. projectFilter(testProject),
  59. serviceFilter("db"),
  60. oneOffFilter(false),
  61. hasConfigHashLabel(),
  62. ),
  63. All: true,
  64. }
  65. t.Run("service links default", func(t *testing.T) {
  66. mockCtrl := gomock.NewController(t)
  67. defer mockCtrl.Finish()
  68. apiClient := mocks.NewMockAPIClient(mockCtrl)
  69. cli := mocks.NewMockCli(mockCtrl)
  70. tested := composeService{
  71. dockerCli: cli,
  72. }
  73. cli.EXPECT().Client().Return(apiClient).AnyTimes()
  74. s.Links = []string{"db"}
  75. c := testContainer("db", dbContainerName, false)
  76. apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptions).Return([]moby.Container{c}, nil)
  77. links, err := tested.getLinks(context.Background(), testProject, s, 1)
  78. assert.NilError(t, err)
  79. assert.Equal(t, len(links), 3)
  80. assert.Equal(t, links[0], "testProject-db-1:db")
  81. assert.Equal(t, links[1], "testProject-db-1:db-1")
  82. assert.Equal(t, links[2], "testProject-db-1:testProject-db-1")
  83. })
  84. t.Run("service links", func(t *testing.T) {
  85. mockCtrl := gomock.NewController(t)
  86. defer mockCtrl.Finish()
  87. apiClient := mocks.NewMockAPIClient(mockCtrl)
  88. cli := mocks.NewMockCli(mockCtrl)
  89. tested := composeService{
  90. dockerCli: cli,
  91. }
  92. cli.EXPECT().Client().Return(apiClient).AnyTimes()
  93. s.Links = []string{"db:db"}
  94. c := testContainer("db", dbContainerName, false)
  95. apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptions).Return([]moby.Container{c}, nil)
  96. links, err := tested.getLinks(context.Background(), testProject, s, 1)
  97. assert.NilError(t, err)
  98. assert.Equal(t, len(links), 3)
  99. assert.Equal(t, links[0], "testProject-db-1:db")
  100. assert.Equal(t, links[1], "testProject-db-1:db-1")
  101. assert.Equal(t, links[2], "testProject-db-1:testProject-db-1")
  102. })
  103. t.Run("service links name", func(t *testing.T) {
  104. mockCtrl := gomock.NewController(t)
  105. defer mockCtrl.Finish()
  106. apiClient := mocks.NewMockAPIClient(mockCtrl)
  107. cli := mocks.NewMockCli(mockCtrl)
  108. tested := composeService{
  109. dockerCli: cli,
  110. }
  111. cli.EXPECT().Client().Return(apiClient).AnyTimes()
  112. s.Links = []string{"db:dbname"}
  113. c := testContainer("db", dbContainerName, false)
  114. apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptions).Return([]moby.Container{c}, nil)
  115. links, err := tested.getLinks(context.Background(), testProject, s, 1)
  116. assert.NilError(t, err)
  117. assert.Equal(t, len(links), 3)
  118. assert.Equal(t, links[0], "testProject-db-1:dbname")
  119. assert.Equal(t, links[1], "testProject-db-1:db-1")
  120. assert.Equal(t, links[2], "testProject-db-1:testProject-db-1")
  121. })
  122. t.Run("service links external links", func(t *testing.T) {
  123. mockCtrl := gomock.NewController(t)
  124. defer mockCtrl.Finish()
  125. apiClient := mocks.NewMockAPIClient(mockCtrl)
  126. cli := mocks.NewMockCli(mockCtrl)
  127. tested := composeService{
  128. dockerCli: cli,
  129. }
  130. cli.EXPECT().Client().Return(apiClient).AnyTimes()
  131. s.Links = []string{"db:dbname"}
  132. s.ExternalLinks = []string{"db1:db2"}
  133. c := testContainer("db", dbContainerName, false)
  134. apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptions).Return([]moby.Container{c}, nil)
  135. links, err := tested.getLinks(context.Background(), testProject, s, 1)
  136. assert.NilError(t, err)
  137. assert.Equal(t, len(links), 4)
  138. assert.Equal(t, links[0], "testProject-db-1:dbname")
  139. assert.Equal(t, links[1], "testProject-db-1:db-1")
  140. assert.Equal(t, links[2], "testProject-db-1:testProject-db-1")
  141. // ExternalLink
  142. assert.Equal(t, links[3], "db1:db2")
  143. })
  144. t.Run("service links itself oneoff", func(t *testing.T) {
  145. mockCtrl := gomock.NewController(t)
  146. defer mockCtrl.Finish()
  147. apiClient := mocks.NewMockAPIClient(mockCtrl)
  148. cli := mocks.NewMockCli(mockCtrl)
  149. tested := composeService{
  150. dockerCli: cli,
  151. }
  152. cli.EXPECT().Client().Return(apiClient).AnyTimes()
  153. s.Links = []string{}
  154. s.ExternalLinks = []string{}
  155. s.Labels = s.Labels.Add(api.OneoffLabel, "True")
  156. c := testContainer("web", webContainerName, true)
  157. containerListOptionsOneOff := containerType.ListOptions{
  158. Filters: filters.NewArgs(
  159. projectFilter(testProject),
  160. serviceFilter("web"),
  161. oneOffFilter(false),
  162. hasConfigHashLabel(),
  163. ),
  164. All: true,
  165. }
  166. apiClient.EXPECT().ContainerList(gomock.Any(), containerListOptionsOneOff).Return([]moby.Container{c}, nil)
  167. links, err := tested.getLinks(context.Background(), testProject, s, 1)
  168. assert.NilError(t, err)
  169. assert.Equal(t, len(links), 3)
  170. assert.Equal(t, links[0], "testProject-web-1:web")
  171. assert.Equal(t, links[1], "testProject-web-1:web-1")
  172. assert.Equal(t, links[2], "testProject-web-1:testProject-web-1")
  173. })
  174. }
  175. func TestWaitDependencies(t *testing.T) {
  176. mockCtrl := gomock.NewController(t)
  177. defer mockCtrl.Finish()
  178. apiClient := mocks.NewMockAPIClient(mockCtrl)
  179. cli := mocks.NewMockCli(mockCtrl)
  180. tested := composeService{
  181. dockerCli: cli,
  182. }
  183. cli.EXPECT().Client().Return(apiClient).AnyTimes()
  184. t.Run("should skip dependencies with scale 0", func(t *testing.T) {
  185. dbService := types.ServiceConfig{Name: "db", Scale: intPtr(0)}
  186. redisService := types.ServiceConfig{Name: "redis", Scale: intPtr(0)}
  187. project := types.Project{Name: strings.ToLower(testProject), Services: types.Services{
  188. "db": dbService,
  189. "redis": redisService,
  190. }}
  191. dependencies := types.DependsOnConfig{
  192. "db": {Condition: ServiceConditionRunningOrHealthy},
  193. "redis": {Condition: ServiceConditionRunningOrHealthy},
  194. }
  195. assert.NilError(t, tested.waitDependencies(context.Background(), &project, "", dependencies, nil))
  196. })
  197. t.Run("should skip dependencies with condition service_started", func(t *testing.T) {
  198. dbService := types.ServiceConfig{Name: "db", Scale: intPtr(1)}
  199. redisService := types.ServiceConfig{Name: "redis", Scale: intPtr(1)}
  200. project := types.Project{Name: strings.ToLower(testProject), Services: types.Services{
  201. "db": dbService,
  202. "redis": redisService,
  203. }}
  204. dependencies := types.DependsOnConfig{
  205. "db": {Condition: types.ServiceConditionStarted, Required: true},
  206. "redis": {Condition: types.ServiceConditionStarted, Required: true},
  207. }
  208. assert.NilError(t, tested.waitDependencies(context.Background(), &project, "", dependencies, nil))
  209. })
  210. }