create_test.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  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. "net"
  16. "net/netip"
  17. "os"
  18. "path/filepath"
  19. "sort"
  20. "testing"
  21. composeloader "github.com/compose-spec/compose-go/v2/loader"
  22. composetypes "github.com/compose-spec/compose-go/v2/types"
  23. "github.com/google/go-cmp/cmp/cmpopts"
  24. "github.com/moby/moby/api/types/container"
  25. mountTypes "github.com/moby/moby/api/types/mount"
  26. "github.com/moby/moby/api/types/network"
  27. "github.com/moby/moby/client"
  28. "go.uber.org/mock/gomock"
  29. "gotest.tools/v3/assert"
  30. "gotest.tools/v3/assert/cmp"
  31. "github.com/docker/compose/v5/pkg/api"
  32. )
  33. func TestBuildBindMount(t *testing.T) {
  34. project := composetypes.Project{}
  35. volume := composetypes.ServiceVolumeConfig{
  36. Type: composetypes.VolumeTypeBind,
  37. Source: "",
  38. Target: "/data",
  39. }
  40. mount, err := buildMount(project, volume)
  41. assert.NilError(t, err)
  42. assert.Assert(t, filepath.IsAbs(mount.Source))
  43. _, err = os.Stat(mount.Source)
  44. assert.NilError(t, err)
  45. assert.Equal(t, mount.Type, mountTypes.TypeBind)
  46. }
  47. func TestBuildNamedPipeMount(t *testing.T) {
  48. project := composetypes.Project{}
  49. volume := composetypes.ServiceVolumeConfig{
  50. Type: composetypes.VolumeTypeNamedPipe,
  51. Source: "\\\\.\\pipe\\docker_engine_windows",
  52. Target: "\\\\.\\pipe\\docker_engine",
  53. }
  54. mount, err := buildMount(project, volume)
  55. assert.NilError(t, err)
  56. assert.Equal(t, mount.Type, mountTypes.TypeNamedPipe)
  57. }
  58. func TestBuildVolumeMount(t *testing.T) {
  59. project := composetypes.Project{
  60. Name: "myProject",
  61. Volumes: composetypes.Volumes(map[string]composetypes.VolumeConfig{
  62. "myVolume": {
  63. Name: "myProject_myVolume",
  64. },
  65. }),
  66. }
  67. volume := composetypes.ServiceVolumeConfig{
  68. Type: composetypes.VolumeTypeVolume,
  69. Source: "myVolume",
  70. Target: "/data",
  71. }
  72. mount, err := buildMount(project, volume)
  73. assert.NilError(t, err)
  74. assert.Equal(t, mount.Source, "myProject_myVolume")
  75. assert.Equal(t, mount.Type, mountTypes.TypeVolume)
  76. }
  77. func TestServiceImageName(t *testing.T) {
  78. assert.Equal(t, api.GetImageNameOrDefault(composetypes.ServiceConfig{Image: "myImage"}, "myProject"), "myImage")
  79. assert.Equal(t, api.GetImageNameOrDefault(composetypes.ServiceConfig{Name: "aService"}, "myProject"), "myProject-aService")
  80. }
  81. func TestPrepareNetworkLabels(t *testing.T) {
  82. project := composetypes.Project{
  83. Name: "myProject",
  84. Networks: composetypes.Networks(map[string]composetypes.NetworkConfig{"skynet": {}}),
  85. }
  86. prepareNetworks(&project)
  87. assert.DeepEqual(t, project.Networks["skynet"].CustomLabels, composetypes.Labels(map[string]string{
  88. "com.docker.compose.network": "skynet",
  89. "com.docker.compose.project": "myProject",
  90. "com.docker.compose.version": api.ComposeVersion,
  91. }))
  92. }
  93. func TestBuildContainerMountOptions(t *testing.T) {
  94. project := composetypes.Project{
  95. Name: "myProject",
  96. Services: composetypes.Services{
  97. "myService": {
  98. Name: "myService",
  99. Volumes: []composetypes.ServiceVolumeConfig{
  100. {
  101. Type: composetypes.VolumeTypeVolume,
  102. Target: "/var/myvolume1",
  103. },
  104. {
  105. Type: composetypes.VolumeTypeVolume,
  106. Target: "/var/myvolume2",
  107. },
  108. {
  109. Type: composetypes.VolumeTypeVolume,
  110. Source: "myVolume3",
  111. Target: "/var/myvolume3",
  112. Volume: &composetypes.ServiceVolumeVolume{
  113. Subpath: "etc",
  114. },
  115. },
  116. {
  117. Type: composetypes.VolumeTypeNamedPipe,
  118. Source: "\\\\.\\pipe\\docker_engine_windows",
  119. Target: "\\\\.\\pipe\\docker_engine",
  120. },
  121. },
  122. },
  123. },
  124. Volumes: composetypes.Volumes(map[string]composetypes.VolumeConfig{
  125. "myVolume1": {
  126. Name: "myProject_myVolume1",
  127. },
  128. "myVolume2": {
  129. Name: "myProject_myVolume2",
  130. },
  131. }),
  132. }
  133. inherit := &container.Summary{
  134. Mounts: []container.MountPoint{
  135. {
  136. Type: composetypes.VolumeTypeVolume,
  137. Destination: "/var/myvolume1",
  138. },
  139. {
  140. Type: composetypes.VolumeTypeVolume,
  141. Destination: "/var/myvolume2",
  142. },
  143. },
  144. }
  145. mockCtrl := gomock.NewController(t)
  146. defer mockCtrl.Finish()
  147. mock, cli := prepareMocks(mockCtrl)
  148. s := composeService{
  149. dockerCli: cli,
  150. }
  151. mock.EXPECT().ImageInspect(gomock.Any(), "myProject-myService").AnyTimes().Return(client.ImageInspectResult{}, nil)
  152. mounts, err := s.buildContainerMountOptions(t.Context(), project, project.Services["myService"], inherit)
  153. sort.Slice(mounts, func(i, j int) bool {
  154. return mounts[i].Target < mounts[j].Target
  155. })
  156. assert.NilError(t, err)
  157. assert.Assert(t, len(mounts) == 4)
  158. assert.Equal(t, mounts[0].Target, "/var/myvolume1")
  159. assert.Equal(t, mounts[1].Target, "/var/myvolume2")
  160. assert.Equal(t, mounts[2].Target, "/var/myvolume3")
  161. assert.Equal(t, mounts[2].VolumeOptions.Subpath, "etc")
  162. assert.Equal(t, mounts[3].Target, "\\\\.\\pipe\\docker_engine")
  163. mounts, err = s.buildContainerMountOptions(t.Context(), project, project.Services["myService"], inherit)
  164. sort.Slice(mounts, func(i, j int) bool {
  165. return mounts[i].Target < mounts[j].Target
  166. })
  167. assert.NilError(t, err)
  168. assert.Assert(t, len(mounts) == 4)
  169. assert.Equal(t, mounts[0].Target, "/var/myvolume1")
  170. assert.Equal(t, mounts[1].Target, "/var/myvolume2")
  171. assert.Equal(t, mounts[2].Target, "/var/myvolume3")
  172. assert.Equal(t, mounts[2].VolumeOptions.Subpath, "etc")
  173. assert.Equal(t, mounts[3].Target, "\\\\.\\pipe\\docker_engine")
  174. }
  175. func TestDefaultNetworkSettings(t *testing.T) {
  176. t.Run("returns the network with the highest priority as primary when service has multiple networks", func(t *testing.T) {
  177. service := composetypes.ServiceConfig{
  178. Name: "myService",
  179. Networks: map[string]*composetypes.ServiceNetworkConfig{
  180. "myNetwork1": {
  181. Priority: 10,
  182. },
  183. "myNetwork2": {
  184. Priority: 1000,
  185. },
  186. },
  187. }
  188. project := composetypes.Project{
  189. Name: "myProject",
  190. Services: composetypes.Services{
  191. "myService": service,
  192. },
  193. Networks: composetypes.Networks(map[string]composetypes.NetworkConfig{
  194. "myNetwork1": {
  195. Name: "myProject_myNetwork1",
  196. },
  197. "myNetwork2": {
  198. Name: "myProject_myNetwork2",
  199. },
  200. }),
  201. }
  202. networkMode, networkConfig, err := defaultNetworkSettings(&project, service, 1, nil, true, "1.44")
  203. assert.NilError(t, err)
  204. assert.Equal(t, string(networkMode), "myProject_myNetwork2")
  205. assert.Check(t, cmp.Len(networkConfig.EndpointsConfig, 2))
  206. assert.Check(t, cmp.Contains(networkConfig.EndpointsConfig, "myProject_myNetwork1"))
  207. assert.Check(t, cmp.Contains(networkConfig.EndpointsConfig, "myProject_myNetwork2"))
  208. })
  209. t.Run("returns default network when service has no networks", func(t *testing.T) {
  210. service := composetypes.ServiceConfig{
  211. Name: "myService",
  212. }
  213. project := composetypes.Project{
  214. Name: "myProject",
  215. Services: composetypes.Services{
  216. "myService": service,
  217. },
  218. Networks: composetypes.Networks(map[string]composetypes.NetworkConfig{
  219. "myNetwork1": {
  220. Name: "myProject_myNetwork1",
  221. },
  222. "myNetwork2": {
  223. Name: "myProject_myNetwork2",
  224. },
  225. "default": {
  226. Name: "myProject_default",
  227. },
  228. }),
  229. }
  230. networkMode, networkConfig, err := defaultNetworkSettings(&project, service, 1, nil, true, "1.44")
  231. assert.NilError(t, err)
  232. assert.Equal(t, string(networkMode), "myProject_default")
  233. assert.Check(t, cmp.Len(networkConfig.EndpointsConfig, 1))
  234. assert.Check(t, cmp.Contains(networkConfig.EndpointsConfig, "myProject_default"))
  235. })
  236. t.Run("returns none if project has no networks", func(t *testing.T) {
  237. service := composetypes.ServiceConfig{
  238. Name: "myService",
  239. }
  240. project := composetypes.Project{
  241. Name: "myProject",
  242. Services: composetypes.Services{
  243. "myService": service,
  244. },
  245. }
  246. networkMode, networkConfig, err := defaultNetworkSettings(&project, service, 1, nil, true, "1.44")
  247. assert.NilError(t, err)
  248. assert.Equal(t, string(networkMode), "none")
  249. assert.Check(t, cmp.Nil(networkConfig))
  250. })
  251. t.Run("returns defined network mode if explicitly set", func(t *testing.T) {
  252. service := composetypes.ServiceConfig{
  253. Name: "myService",
  254. NetworkMode: "host",
  255. }
  256. project := composetypes.Project{
  257. Name: "myProject",
  258. Services: composetypes.Services{"myService": service},
  259. Networks: composetypes.Networks(map[string]composetypes.NetworkConfig{
  260. "default": {
  261. Name: "myProject_default",
  262. },
  263. }),
  264. }
  265. networkMode, networkConfig, err := defaultNetworkSettings(&project, service, 1, nil, true, "1.44")
  266. assert.NilError(t, err)
  267. assert.Equal(t, string(networkMode), "host")
  268. assert.Check(t, cmp.Nil(networkConfig))
  269. })
  270. }
  271. func TestCreateEndpointSettings(t *testing.T) {
  272. eps, err := createEndpointSettings(&composetypes.Project{
  273. Name: "projName",
  274. }, composetypes.ServiceConfig{
  275. Name: "serviceName",
  276. ContainerName: "containerName",
  277. Networks: map[string]*composetypes.ServiceNetworkConfig{
  278. "netName": {
  279. Priority: 100,
  280. Aliases: []string{"alias1", "alias2"},
  281. Ipv4Address: "10.16.17.18",
  282. Ipv6Address: "fdb4:7a7f:373a:3f0c::42",
  283. LinkLocalIPs: []string{"169.254.10.20"},
  284. MacAddress: "02:00:00:00:00:01",
  285. DriverOpts: composetypes.Options{
  286. "driverOpt1": "optval1",
  287. "driverOpt2": "optval2",
  288. },
  289. },
  290. },
  291. }, 0, "netName", []string{"link1", "link2"}, true)
  292. assert.NilError(t, err)
  293. macAddr, _ := net.ParseMAC("02:00:00:00:00:01")
  294. assert.Check(t, cmp.DeepEqual(eps, &network.EndpointSettings{
  295. IPAMConfig: &network.EndpointIPAMConfig{
  296. IPv4Address: netip.MustParseAddr("10.16.17.18").Unmap(),
  297. IPv6Address: netip.MustParseAddr("fdb4:7a7f:373a:3f0c::42"),
  298. LinkLocalIPs: []netip.Addr{netip.MustParseAddr("169.254.10.20").Unmap()},
  299. },
  300. Links: []string{"link1", "link2"},
  301. Aliases: []string{"containerName", "serviceName", "alias1", "alias2"},
  302. MacAddress: network.HardwareAddr(macAddr),
  303. DriverOpts: map[string]string{
  304. "driverOpt1": "optval1",
  305. "driverOpt2": "optval2",
  306. },
  307. // FIXME(robmry) - IPAddress and IPv6Gateway are "operational data" fields...
  308. // - The IPv6 address here is the container's address, not the gateway.
  309. // - Both fields will be cleared by the daemon, but they could be removed from
  310. // the request.
  311. IPAddress: netip.MustParseAddr("10.16.17.18").Unmap(),
  312. IPv6Gateway: netip.MustParseAddr("fdb4:7a7f:373a:3f0c::42"),
  313. }, cmpopts.EquateComparable(netip.Addr{})))
  314. }
  315. func Test_buildContainerVolumes(t *testing.T) {
  316. pwd, err := os.Getwd()
  317. assert.NilError(t, err)
  318. tests := []struct {
  319. name string
  320. yaml string
  321. binds []string
  322. mounts []mountTypes.Mount
  323. }{
  324. {
  325. name: "bind mount local path",
  326. yaml: `
  327. services:
  328. test:
  329. volumes:
  330. - ./data:/data
  331. `,
  332. binds: []string{filepath.Join(pwd, "data") + ":/data:rw"},
  333. mounts: nil,
  334. },
  335. {
  336. name: "bind mount, not create host path",
  337. yaml: `
  338. services:
  339. test:
  340. volumes:
  341. - type: bind
  342. source: ./data
  343. target: /data
  344. bind:
  345. create_host_path: false
  346. `,
  347. binds: nil,
  348. mounts: []mountTypes.Mount{
  349. {
  350. Type: "bind",
  351. Source: filepath.Join(pwd, "data"),
  352. Target: "/data",
  353. BindOptions: &mountTypes.BindOptions{CreateMountpoint: false},
  354. },
  355. },
  356. },
  357. {
  358. name: "mount volume",
  359. yaml: `
  360. services:
  361. test:
  362. volumes:
  363. - data:/data
  364. volumes:
  365. data:
  366. name: my_volume
  367. `,
  368. binds: []string{"my_volume:/data:rw"},
  369. mounts: nil,
  370. },
  371. {
  372. name: "mount volume, readonly",
  373. yaml: `
  374. services:
  375. test:
  376. volumes:
  377. - data:/data:ro
  378. volumes:
  379. data:
  380. name: my_volume
  381. `,
  382. binds: []string{"my_volume:/data:ro"},
  383. mounts: nil,
  384. },
  385. {
  386. name: "mount volume subpath",
  387. yaml: `
  388. services:
  389. test:
  390. volumes:
  391. - type: volume
  392. source: data
  393. target: /data
  394. volume:
  395. subpath: test/
  396. volumes:
  397. data:
  398. name: my_volume
  399. `,
  400. binds: nil,
  401. mounts: []mountTypes.Mount{
  402. {
  403. Type: "volume",
  404. Source: "my_volume",
  405. Target: "/data",
  406. VolumeOptions: &mountTypes.VolumeOptions{Subpath: "test/"},
  407. },
  408. },
  409. },
  410. }
  411. for _, tt := range tests {
  412. t.Run(tt.name, func(t *testing.T) {
  413. p, err := composeloader.LoadWithContext(t.Context(), composetypes.ConfigDetails{
  414. ConfigFiles: []composetypes.ConfigFile{
  415. {
  416. Filename: "test",
  417. Content: []byte(tt.yaml),
  418. },
  419. },
  420. }, func(options *composeloader.Options) {
  421. options.SkipValidation = true
  422. options.SkipConsistencyCheck = true
  423. })
  424. assert.NilError(t, err)
  425. s := &composeService{}
  426. binds, mounts, err := s.buildContainerVolumes(t.Context(), *p, p.Services["test"], nil)
  427. assert.NilError(t, err)
  428. assert.DeepEqual(t, tt.binds, binds)
  429. assert.DeepEqual(t, tt.mounts, mounts)
  430. })
  431. }
  432. }