create_test.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486
  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 only primary network in EndpointsConfig for API < 1.44", func(t *testing.T) {
  252. service := composetypes.ServiceConfig{
  253. Name: "myService",
  254. Networks: map[string]*composetypes.ServiceNetworkConfig{
  255. "myNetwork1": {Priority: 10},
  256. "myNetwork2": {Priority: 1000},
  257. },
  258. }
  259. project := composetypes.Project{
  260. Name: "myProject",
  261. Services: composetypes.Services{"myService": service},
  262. Networks: composetypes.Networks(map[string]composetypes.NetworkConfig{
  263. "myNetwork1": {Name: "myProject_myNetwork1"},
  264. "myNetwork2": {Name: "myProject_myNetwork2"},
  265. }),
  266. }
  267. networkMode, networkConfig, err := defaultNetworkSettings(&project, service, 1, nil, true, "1.43")
  268. assert.NilError(t, err)
  269. assert.Equal(t, string(networkMode), "myProject_myNetwork2")
  270. assert.Check(t, cmp.Len(networkConfig.EndpointsConfig, 1))
  271. assert.Check(t, cmp.Contains(networkConfig.EndpointsConfig, "myProject_myNetwork2"))
  272. })
  273. t.Run("returns defined network mode if explicitly set", func(t *testing.T) {
  274. service := composetypes.ServiceConfig{
  275. Name: "myService",
  276. NetworkMode: "host",
  277. }
  278. project := composetypes.Project{
  279. Name: "myProject",
  280. Services: composetypes.Services{"myService": service},
  281. Networks: composetypes.Networks(map[string]composetypes.NetworkConfig{
  282. "default": {
  283. Name: "myProject_default",
  284. },
  285. }),
  286. }
  287. networkMode, networkConfig, err := defaultNetworkSettings(&project, service, 1, nil, true, "1.44")
  288. assert.NilError(t, err)
  289. assert.Equal(t, string(networkMode), "host")
  290. assert.Check(t, cmp.Nil(networkConfig))
  291. })
  292. }
  293. func TestCreateEndpointSettings(t *testing.T) {
  294. eps, err := createEndpointSettings(&composetypes.Project{
  295. Name: "projName",
  296. }, composetypes.ServiceConfig{
  297. Name: "serviceName",
  298. ContainerName: "containerName",
  299. Networks: map[string]*composetypes.ServiceNetworkConfig{
  300. "netName": {
  301. Priority: 100,
  302. Aliases: []string{"alias1", "alias2"},
  303. Ipv4Address: "10.16.17.18",
  304. Ipv6Address: "fdb4:7a7f:373a:3f0c::42",
  305. LinkLocalIPs: []string{"169.254.10.20"},
  306. MacAddress: "02:00:00:00:00:01",
  307. DriverOpts: composetypes.Options{
  308. "driverOpt1": "optval1",
  309. "driverOpt2": "optval2",
  310. },
  311. },
  312. },
  313. }, 0, "netName", []string{"link1", "link2"}, true)
  314. assert.NilError(t, err)
  315. macAddr, _ := net.ParseMAC("02:00:00:00:00:01")
  316. assert.Check(t, cmp.DeepEqual(eps, &network.EndpointSettings{
  317. IPAMConfig: &network.EndpointIPAMConfig{
  318. IPv4Address: netip.MustParseAddr("10.16.17.18").Unmap(),
  319. IPv6Address: netip.MustParseAddr("fdb4:7a7f:373a:3f0c::42"),
  320. LinkLocalIPs: []netip.Addr{netip.MustParseAddr("169.254.10.20").Unmap()},
  321. },
  322. Links: []string{"link1", "link2"},
  323. Aliases: []string{"containerName", "serviceName", "alias1", "alias2"},
  324. MacAddress: network.HardwareAddr(macAddr),
  325. DriverOpts: map[string]string{
  326. "driverOpt1": "optval1",
  327. "driverOpt2": "optval2",
  328. },
  329. // FIXME(robmry) - IPAddress and IPv6Gateway are "operational data" fields...
  330. // - The IPv6 address here is the container's address, not the gateway.
  331. // - Both fields will be cleared by the daemon, but they could be removed from
  332. // the request.
  333. IPAddress: netip.MustParseAddr("10.16.17.18").Unmap(),
  334. IPv6Gateway: netip.MustParseAddr("fdb4:7a7f:373a:3f0c::42"),
  335. }, cmpopts.EquateComparable(netip.Addr{})))
  336. }
  337. func Test_buildContainerVolumes(t *testing.T) {
  338. pwd, err := os.Getwd()
  339. assert.NilError(t, err)
  340. tests := []struct {
  341. name string
  342. yaml string
  343. binds []string
  344. mounts []mountTypes.Mount
  345. }{
  346. {
  347. name: "bind mount local path",
  348. yaml: `
  349. services:
  350. test:
  351. volumes:
  352. - ./data:/data
  353. `,
  354. binds: []string{filepath.Join(pwd, "data") + ":/data:rw"},
  355. mounts: nil,
  356. },
  357. {
  358. name: "bind mount, not create host path",
  359. yaml: `
  360. services:
  361. test:
  362. volumes:
  363. - type: bind
  364. source: ./data
  365. target: /data
  366. bind:
  367. create_host_path: false
  368. `,
  369. binds: nil,
  370. mounts: []mountTypes.Mount{
  371. {
  372. Type: "bind",
  373. Source: filepath.Join(pwd, "data"),
  374. Target: "/data",
  375. BindOptions: &mountTypes.BindOptions{CreateMountpoint: false},
  376. },
  377. },
  378. },
  379. {
  380. name: "mount volume",
  381. yaml: `
  382. services:
  383. test:
  384. volumes:
  385. - data:/data
  386. volumes:
  387. data:
  388. name: my_volume
  389. `,
  390. binds: []string{"my_volume:/data:rw"},
  391. mounts: nil,
  392. },
  393. {
  394. name: "mount volume, readonly",
  395. yaml: `
  396. services:
  397. test:
  398. volumes:
  399. - data:/data:ro
  400. volumes:
  401. data:
  402. name: my_volume
  403. `,
  404. binds: []string{"my_volume:/data:ro"},
  405. mounts: nil,
  406. },
  407. {
  408. name: "mount volume subpath",
  409. yaml: `
  410. services:
  411. test:
  412. volumes:
  413. - type: volume
  414. source: data
  415. target: /data
  416. volume:
  417. subpath: test/
  418. volumes:
  419. data:
  420. name: my_volume
  421. `,
  422. binds: nil,
  423. mounts: []mountTypes.Mount{
  424. {
  425. Type: "volume",
  426. Source: "my_volume",
  427. Target: "/data",
  428. VolumeOptions: &mountTypes.VolumeOptions{Subpath: "test/"},
  429. },
  430. },
  431. },
  432. }
  433. for _, tt := range tests {
  434. t.Run(tt.name, func(t *testing.T) {
  435. p, err := composeloader.LoadWithContext(t.Context(), composetypes.ConfigDetails{
  436. ConfigFiles: []composetypes.ConfigFile{
  437. {
  438. Filename: "test",
  439. Content: []byte(tt.yaml),
  440. },
  441. },
  442. }, func(options *composeloader.Options) {
  443. options.SkipValidation = true
  444. options.SkipConsistencyCheck = true
  445. })
  446. assert.NilError(t, err)
  447. s := &composeService{}
  448. binds, mounts, err := s.buildContainerVolumes(t.Context(), *p, p.Services["test"], nil)
  449. assert.NilError(t, err)
  450. assert.DeepEqual(t, tt.binds, binds)
  451. assert.DeepEqual(t, tt.mounts, mounts)
  452. })
  453. }
  454. }