loader_test.go 20 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043
  1. package loader
  2. import (
  3. "fmt"
  4. "io/ioutil"
  5. "os"
  6. "sort"
  7. "testing"
  8. "time"
  9. "github.com/docker/docker/cli/compose/types"
  10. "github.com/stretchr/testify/assert"
  11. )
  12. func buildConfigDetails(source types.Dict) types.ConfigDetails {
  13. workingDir, err := os.Getwd()
  14. if err != nil {
  15. panic(err)
  16. }
  17. return types.ConfigDetails{
  18. WorkingDir: workingDir,
  19. ConfigFiles: []types.ConfigFile{
  20. {Filename: "filename.yml", Config: source},
  21. },
  22. Environment: nil,
  23. }
  24. }
  25. var sampleYAML = `
  26. version: "3"
  27. services:
  28. foo:
  29. image: busybox
  30. networks:
  31. with_me:
  32. bar:
  33. image: busybox
  34. environment:
  35. - FOO=1
  36. networks:
  37. - with_ipam
  38. volumes:
  39. hello:
  40. driver: default
  41. driver_opts:
  42. beep: boop
  43. networks:
  44. default:
  45. driver: bridge
  46. driver_opts:
  47. beep: boop
  48. with_ipam:
  49. ipam:
  50. driver: default
  51. config:
  52. - subnet: 172.28.0.0/16
  53. `
  54. var sampleDict = types.Dict{
  55. "version": "3",
  56. "services": types.Dict{
  57. "foo": types.Dict{
  58. "image": "busybox",
  59. "networks": types.Dict{"with_me": nil},
  60. },
  61. "bar": types.Dict{
  62. "image": "busybox",
  63. "environment": []interface{}{"FOO=1"},
  64. "networks": []interface{}{"with_ipam"},
  65. },
  66. },
  67. "volumes": types.Dict{
  68. "hello": types.Dict{
  69. "driver": "default",
  70. "driver_opts": types.Dict{
  71. "beep": "boop",
  72. },
  73. },
  74. },
  75. "networks": types.Dict{
  76. "default": types.Dict{
  77. "driver": "bridge",
  78. "driver_opts": types.Dict{
  79. "beep": "boop",
  80. },
  81. },
  82. "with_ipam": types.Dict{
  83. "ipam": types.Dict{
  84. "driver": "default",
  85. "config": []interface{}{
  86. types.Dict{
  87. "subnet": "172.28.0.0/16",
  88. },
  89. },
  90. },
  91. },
  92. },
  93. }
  94. var sampleConfig = types.Config{
  95. Services: []types.ServiceConfig{
  96. {
  97. Name: "foo",
  98. Image: "busybox",
  99. Environment: map[string]string{},
  100. Networks: map[string]*types.ServiceNetworkConfig{
  101. "with_me": nil,
  102. },
  103. },
  104. {
  105. Name: "bar",
  106. Image: "busybox",
  107. Environment: map[string]string{"FOO": "1"},
  108. Networks: map[string]*types.ServiceNetworkConfig{
  109. "with_ipam": nil,
  110. },
  111. },
  112. },
  113. Networks: map[string]types.NetworkConfig{
  114. "default": {
  115. Driver: "bridge",
  116. DriverOpts: map[string]string{
  117. "beep": "boop",
  118. },
  119. },
  120. "with_ipam": {
  121. Ipam: types.IPAMConfig{
  122. Driver: "default",
  123. Config: []*types.IPAMPool{
  124. {
  125. Subnet: "172.28.0.0/16",
  126. },
  127. },
  128. },
  129. },
  130. },
  131. Volumes: map[string]types.VolumeConfig{
  132. "hello": {
  133. Driver: "default",
  134. DriverOpts: map[string]string{
  135. "beep": "boop",
  136. },
  137. },
  138. },
  139. }
  140. func TestParseYAML(t *testing.T) {
  141. dict, err := ParseYAML([]byte(sampleYAML))
  142. if !assert.NoError(t, err) {
  143. return
  144. }
  145. assert.Equal(t, sampleDict, dict)
  146. }
  147. func TestLoad(t *testing.T) {
  148. actual, err := Load(buildConfigDetails(sampleDict))
  149. if !assert.NoError(t, err) {
  150. return
  151. }
  152. assert.Equal(t, serviceSort(sampleConfig.Services), serviceSort(actual.Services))
  153. assert.Equal(t, sampleConfig.Networks, actual.Networks)
  154. assert.Equal(t, sampleConfig.Volumes, actual.Volumes)
  155. }
  156. func TestLoadV31(t *testing.T) {
  157. actual, err := loadYAML(`
  158. version: "3.1"
  159. services:
  160. foo:
  161. image: busybox
  162. secrets: [super]
  163. secrets:
  164. super:
  165. external: true
  166. `)
  167. if !assert.NoError(t, err) {
  168. return
  169. }
  170. assert.Equal(t, len(actual.Services), 1)
  171. assert.Equal(t, len(actual.Secrets), 1)
  172. }
  173. func TestParseAndLoad(t *testing.T) {
  174. actual, err := loadYAML(sampleYAML)
  175. if !assert.NoError(t, err) {
  176. return
  177. }
  178. assert.Equal(t, serviceSort(sampleConfig.Services), serviceSort(actual.Services))
  179. assert.Equal(t, sampleConfig.Networks, actual.Networks)
  180. assert.Equal(t, sampleConfig.Volumes, actual.Volumes)
  181. }
  182. func TestInvalidTopLevelObjectType(t *testing.T) {
  183. _, err := loadYAML("1")
  184. assert.Error(t, err)
  185. assert.Contains(t, err.Error(), "Top-level object must be a mapping")
  186. _, err = loadYAML("\"hello\"")
  187. assert.Error(t, err)
  188. assert.Contains(t, err.Error(), "Top-level object must be a mapping")
  189. _, err = loadYAML("[\"hello\"]")
  190. assert.Error(t, err)
  191. assert.Contains(t, err.Error(), "Top-level object must be a mapping")
  192. }
  193. func TestNonStringKeys(t *testing.T) {
  194. _, err := loadYAML(`
  195. version: "3"
  196. 123:
  197. foo:
  198. image: busybox
  199. `)
  200. assert.Error(t, err)
  201. assert.Contains(t, err.Error(), "Non-string key at top level: 123")
  202. _, err = loadYAML(`
  203. version: "3"
  204. services:
  205. foo:
  206. image: busybox
  207. 123:
  208. image: busybox
  209. `)
  210. assert.Error(t, err)
  211. assert.Contains(t, err.Error(), "Non-string key in services: 123")
  212. _, err = loadYAML(`
  213. version: "3"
  214. services:
  215. foo:
  216. image: busybox
  217. networks:
  218. default:
  219. ipam:
  220. config:
  221. - 123: oh dear
  222. `)
  223. assert.Error(t, err)
  224. assert.Contains(t, err.Error(), "Non-string key in networks.default.ipam.config[0]: 123")
  225. _, err = loadYAML(`
  226. version: "3"
  227. services:
  228. dict-env:
  229. image: busybox
  230. environment:
  231. 1: FOO
  232. `)
  233. assert.Error(t, err)
  234. assert.Contains(t, err.Error(), "Non-string key in services.dict-env.environment: 1")
  235. }
  236. func TestSupportedVersion(t *testing.T) {
  237. _, err := loadYAML(`
  238. version: "3"
  239. services:
  240. foo:
  241. image: busybox
  242. `)
  243. assert.NoError(t, err)
  244. _, err = loadYAML(`
  245. version: "3.0"
  246. services:
  247. foo:
  248. image: busybox
  249. `)
  250. assert.NoError(t, err)
  251. }
  252. func TestUnsupportedVersion(t *testing.T) {
  253. _, err := loadYAML(`
  254. version: "2"
  255. services:
  256. foo:
  257. image: busybox
  258. `)
  259. assert.Error(t, err)
  260. assert.Contains(t, err.Error(), "version")
  261. _, err = loadYAML(`
  262. version: "2.0"
  263. services:
  264. foo:
  265. image: busybox
  266. `)
  267. assert.Error(t, err)
  268. assert.Contains(t, err.Error(), "version")
  269. }
  270. func TestInvalidVersion(t *testing.T) {
  271. _, err := loadYAML(`
  272. version: 3
  273. services:
  274. foo:
  275. image: busybox
  276. `)
  277. assert.Error(t, err)
  278. assert.Contains(t, err.Error(), "version must be a string")
  279. }
  280. func TestV1Unsupported(t *testing.T) {
  281. _, err := loadYAML(`
  282. foo:
  283. image: busybox
  284. `)
  285. assert.Error(t, err)
  286. }
  287. func TestNonMappingObject(t *testing.T) {
  288. _, err := loadYAML(`
  289. version: "3"
  290. services:
  291. - foo:
  292. image: busybox
  293. `)
  294. assert.Error(t, err)
  295. assert.Contains(t, err.Error(), "services must be a mapping")
  296. _, err = loadYAML(`
  297. version: "3"
  298. services:
  299. foo: busybox
  300. `)
  301. assert.Error(t, err)
  302. assert.Contains(t, err.Error(), "services.foo must be a mapping")
  303. _, err = loadYAML(`
  304. version: "3"
  305. networks:
  306. - default:
  307. driver: bridge
  308. `)
  309. assert.Error(t, err)
  310. assert.Contains(t, err.Error(), "networks must be a mapping")
  311. _, err = loadYAML(`
  312. version: "3"
  313. networks:
  314. default: bridge
  315. `)
  316. assert.Error(t, err)
  317. assert.Contains(t, err.Error(), "networks.default must be a mapping")
  318. _, err = loadYAML(`
  319. version: "3"
  320. volumes:
  321. - data:
  322. driver: local
  323. `)
  324. assert.Error(t, err)
  325. assert.Contains(t, err.Error(), "volumes must be a mapping")
  326. _, err = loadYAML(`
  327. version: "3"
  328. volumes:
  329. data: local
  330. `)
  331. assert.Error(t, err)
  332. assert.Contains(t, err.Error(), "volumes.data must be a mapping")
  333. }
  334. func TestNonStringImage(t *testing.T) {
  335. _, err := loadYAML(`
  336. version: "3"
  337. services:
  338. foo:
  339. image: ["busybox", "latest"]
  340. `)
  341. assert.Error(t, err)
  342. assert.Contains(t, err.Error(), "services.foo.image must be a string")
  343. }
  344. func TestValidEnvironment(t *testing.T) {
  345. config, err := loadYAML(`
  346. version: "3"
  347. services:
  348. dict-env:
  349. image: busybox
  350. environment:
  351. FOO: "1"
  352. BAR: 2
  353. BAZ: 2.5
  354. QUUX:
  355. list-env:
  356. image: busybox
  357. environment:
  358. - FOO=1
  359. - BAR=2
  360. - BAZ=2.5
  361. - QUUX=
  362. `)
  363. assert.NoError(t, err)
  364. expected := types.MappingWithEquals{
  365. "FOO": "1",
  366. "BAR": "2",
  367. "BAZ": "2.5",
  368. "QUUX": "",
  369. }
  370. assert.Equal(t, 2, len(config.Services))
  371. for _, service := range config.Services {
  372. assert.Equal(t, expected, service.Environment)
  373. }
  374. }
  375. func TestInvalidEnvironmentValue(t *testing.T) {
  376. _, err := loadYAML(`
  377. version: "3"
  378. services:
  379. dict-env:
  380. image: busybox
  381. environment:
  382. FOO: ["1"]
  383. `)
  384. assert.Error(t, err)
  385. assert.Contains(t, err.Error(), "services.dict-env.environment.FOO must be a string, number or null")
  386. }
  387. func TestInvalidEnvironmentObject(t *testing.T) {
  388. _, err := loadYAML(`
  389. version: "3"
  390. services:
  391. dict-env:
  392. image: busybox
  393. environment: "FOO=1"
  394. `)
  395. assert.Error(t, err)
  396. assert.Contains(t, err.Error(), "services.dict-env.environment must be a mapping")
  397. }
  398. func TestEnvironmentInterpolation(t *testing.T) {
  399. config, err := loadYAML(`
  400. version: "3"
  401. services:
  402. test:
  403. image: busybox
  404. labels:
  405. - home1=$HOME
  406. - home2=${HOME}
  407. - nonexistent=$NONEXISTENT
  408. - default=${NONEXISTENT-default}
  409. networks:
  410. test:
  411. driver: $HOME
  412. volumes:
  413. test:
  414. driver: $HOME
  415. `)
  416. assert.NoError(t, err)
  417. home := os.Getenv("HOME")
  418. expectedLabels := types.MappingWithEquals{
  419. "home1": home,
  420. "home2": home,
  421. "nonexistent": "",
  422. "default": "default",
  423. }
  424. assert.Equal(t, expectedLabels, config.Services[0].Labels)
  425. assert.Equal(t, home, config.Networks["test"].Driver)
  426. assert.Equal(t, home, config.Volumes["test"].Driver)
  427. }
  428. func TestUnsupportedProperties(t *testing.T) {
  429. dict, err := ParseYAML([]byte(`
  430. version: "3"
  431. services:
  432. web:
  433. image: web
  434. build: ./web
  435. links:
  436. - bar
  437. db:
  438. image: db
  439. build: ./db
  440. `))
  441. assert.NoError(t, err)
  442. configDetails := buildConfigDetails(dict)
  443. _, err = Load(configDetails)
  444. assert.NoError(t, err)
  445. unsupported := GetUnsupportedProperties(configDetails)
  446. assert.Equal(t, []string{"build", "links"}, unsupported)
  447. }
  448. func TestDeprecatedProperties(t *testing.T) {
  449. dict, err := ParseYAML([]byte(`
  450. version: "3"
  451. services:
  452. web:
  453. image: web
  454. container_name: web
  455. db:
  456. image: db
  457. container_name: db
  458. expose: ["5434"]
  459. `))
  460. assert.NoError(t, err)
  461. configDetails := buildConfigDetails(dict)
  462. _, err = Load(configDetails)
  463. assert.NoError(t, err)
  464. deprecated := GetDeprecatedProperties(configDetails)
  465. assert.Equal(t, 2, len(deprecated))
  466. assert.Contains(t, deprecated, "container_name")
  467. assert.Contains(t, deprecated, "expose")
  468. }
  469. func TestForbiddenProperties(t *testing.T) {
  470. _, err := loadYAML(`
  471. version: "3"
  472. services:
  473. foo:
  474. image: busybox
  475. volumes:
  476. - /data
  477. volume_driver: some-driver
  478. bar:
  479. extends:
  480. service: foo
  481. `)
  482. assert.Error(t, err)
  483. assert.IsType(t, &ForbiddenPropertiesError{}, err)
  484. fmt.Println(err)
  485. forbidden := err.(*ForbiddenPropertiesError).Properties
  486. assert.Equal(t, 2, len(forbidden))
  487. assert.Contains(t, forbidden, "volume_driver")
  488. assert.Contains(t, forbidden, "extends")
  489. }
  490. func durationPtr(value time.Duration) *time.Duration {
  491. return &value
  492. }
  493. func int64Ptr(value int64) *int64 {
  494. return &value
  495. }
  496. func uint64Ptr(value uint64) *uint64 {
  497. return &value
  498. }
  499. func TestFullExample(t *testing.T) {
  500. bytes, err := ioutil.ReadFile("full-example.yml")
  501. assert.NoError(t, err)
  502. config, err := loadYAML(string(bytes))
  503. if !assert.NoError(t, err) {
  504. return
  505. }
  506. workingDir, err := os.Getwd()
  507. assert.NoError(t, err)
  508. homeDir := os.Getenv("HOME")
  509. stopGracePeriod := time.Duration(20 * time.Second)
  510. expectedServiceConfig := types.ServiceConfig{
  511. Name: "foo",
  512. CapAdd: []string{"ALL"},
  513. CapDrop: []string{"NET_ADMIN", "SYS_ADMIN"},
  514. CgroupParent: "m-executor-abcd",
  515. Command: []string{"bundle", "exec", "thin", "-p", "3000"},
  516. ContainerName: "my-web-container",
  517. DependsOn: []string{"db", "redis"},
  518. Deploy: types.DeployConfig{
  519. Mode: "replicated",
  520. Replicas: uint64Ptr(6),
  521. Labels: map[string]string{"FOO": "BAR"},
  522. UpdateConfig: &types.UpdateConfig{
  523. Parallelism: uint64Ptr(3),
  524. Delay: time.Duration(10 * time.Second),
  525. FailureAction: "continue",
  526. Monitor: time.Duration(60 * time.Second),
  527. MaxFailureRatio: 0.3,
  528. },
  529. Resources: types.Resources{
  530. Limits: &types.Resource{
  531. NanoCPUs: "0.001",
  532. MemoryBytes: 50 * 1024 * 1024,
  533. },
  534. Reservations: &types.Resource{
  535. NanoCPUs: "0.0001",
  536. MemoryBytes: 20 * 1024 * 1024,
  537. },
  538. },
  539. RestartPolicy: &types.RestartPolicy{
  540. Condition: "on_failure",
  541. Delay: durationPtr(5 * time.Second),
  542. MaxAttempts: uint64Ptr(3),
  543. Window: durationPtr(2 * time.Minute),
  544. },
  545. Placement: types.Placement{
  546. Constraints: []string{"node=foo"},
  547. },
  548. },
  549. Devices: []string{"/dev/ttyUSB0:/dev/ttyUSB0"},
  550. DNS: []string{"8.8.8.8", "9.9.9.9"},
  551. DNSSearch: []string{"dc1.example.com", "dc2.example.com"},
  552. DomainName: "foo.com",
  553. Entrypoint: []string{"/code/entrypoint.sh", "-p", "3000"},
  554. Environment: map[string]string{
  555. "RACK_ENV": "development",
  556. "SHOW": "true",
  557. "SESSION_SECRET": "",
  558. "FOO": "1",
  559. "BAR": "2",
  560. "BAZ": "3",
  561. },
  562. EnvFile: []string{
  563. "./example1.env",
  564. "./example2.env",
  565. },
  566. Expose: []string{"3000", "8000"},
  567. ExternalLinks: []string{
  568. "redis_1",
  569. "project_db_1:mysql",
  570. "project_db_1:postgresql",
  571. },
  572. ExtraHosts: map[string]string{
  573. "otherhost": "50.31.209.229",
  574. "somehost": "162.242.195.82",
  575. },
  576. HealthCheck: &types.HealthCheckConfig{
  577. Test: types.HealthCheckTest([]string{"CMD-SHELL", "echo \"hello world\""}),
  578. Interval: "10s",
  579. Timeout: "1s",
  580. Retries: uint64Ptr(5),
  581. },
  582. Hostname: "foo",
  583. Image: "redis",
  584. Ipc: "host",
  585. Labels: map[string]string{
  586. "com.example.description": "Accounting webapp",
  587. "com.example.number": "42",
  588. "com.example.empty-label": "",
  589. },
  590. Links: []string{
  591. "db",
  592. "db:database",
  593. "redis",
  594. },
  595. Logging: &types.LoggingConfig{
  596. Driver: "syslog",
  597. Options: map[string]string{
  598. "syslog-address": "tcp://192.168.0.42:123",
  599. },
  600. },
  601. MacAddress: "02:42:ac:11:65:43",
  602. NetworkMode: "container:0cfeab0f748b9a743dc3da582046357c6ef497631c1a016d28d2bf9b4f899f7b",
  603. Networks: map[string]*types.ServiceNetworkConfig{
  604. "some-network": {
  605. Aliases: []string{"alias1", "alias3"},
  606. Ipv4Address: "",
  607. Ipv6Address: "",
  608. },
  609. "other-network": {
  610. Ipv4Address: "172.16.238.10",
  611. Ipv6Address: "2001:3984:3989::10",
  612. },
  613. "other-other-network": nil,
  614. },
  615. Pid: "host",
  616. Ports: []types.ServicePortConfig{
  617. //"3000",
  618. {
  619. Mode: "ingress",
  620. Target: 3000,
  621. Protocol: "tcp",
  622. },
  623. //"3000-3005",
  624. {
  625. Mode: "ingress",
  626. Target: 3000,
  627. Protocol: "tcp",
  628. },
  629. {
  630. Mode: "ingress",
  631. Target: 3001,
  632. Protocol: "tcp",
  633. },
  634. {
  635. Mode: "ingress",
  636. Target: 3002,
  637. Protocol: "tcp",
  638. },
  639. {
  640. Mode: "ingress",
  641. Target: 3003,
  642. Protocol: "tcp",
  643. },
  644. {
  645. Mode: "ingress",
  646. Target: 3004,
  647. Protocol: "tcp",
  648. },
  649. {
  650. Mode: "ingress",
  651. Target: 3005,
  652. Protocol: "tcp",
  653. },
  654. //"8000:8000",
  655. {
  656. Mode: "ingress",
  657. Target: 8000,
  658. Published: 8000,
  659. Protocol: "tcp",
  660. },
  661. //"9090-9091:8080-8081",
  662. {
  663. Mode: "ingress",
  664. Target: 8080,
  665. Published: 9090,
  666. Protocol: "tcp",
  667. },
  668. {
  669. Mode: "ingress",
  670. Target: 8081,
  671. Published: 9091,
  672. Protocol: "tcp",
  673. },
  674. //"49100:22",
  675. {
  676. Mode: "ingress",
  677. Target: 22,
  678. Published: 49100,
  679. Protocol: "tcp",
  680. },
  681. //"127.0.0.1:8001:8001",
  682. {
  683. Mode: "ingress",
  684. Target: 8001,
  685. Published: 8001,
  686. Protocol: "tcp",
  687. },
  688. //"127.0.0.1:5000-5010:5000-5010",
  689. {
  690. Mode: "ingress",
  691. Target: 5000,
  692. Published: 5000,
  693. Protocol: "tcp",
  694. },
  695. {
  696. Mode: "ingress",
  697. Target: 5001,
  698. Published: 5001,
  699. Protocol: "tcp",
  700. },
  701. {
  702. Mode: "ingress",
  703. Target: 5002,
  704. Published: 5002,
  705. Protocol: "tcp",
  706. },
  707. {
  708. Mode: "ingress",
  709. Target: 5003,
  710. Published: 5003,
  711. Protocol: "tcp",
  712. },
  713. {
  714. Mode: "ingress",
  715. Target: 5004,
  716. Published: 5004,
  717. Protocol: "tcp",
  718. },
  719. {
  720. Mode: "ingress",
  721. Target: 5005,
  722. Published: 5005,
  723. Protocol: "tcp",
  724. },
  725. {
  726. Mode: "ingress",
  727. Target: 5006,
  728. Published: 5006,
  729. Protocol: "tcp",
  730. },
  731. {
  732. Mode: "ingress",
  733. Target: 5007,
  734. Published: 5007,
  735. Protocol: "tcp",
  736. },
  737. {
  738. Mode: "ingress",
  739. Target: 5008,
  740. Published: 5008,
  741. Protocol: "tcp",
  742. },
  743. {
  744. Mode: "ingress",
  745. Target: 5009,
  746. Published: 5009,
  747. Protocol: "tcp",
  748. },
  749. {
  750. Mode: "ingress",
  751. Target: 5010,
  752. Published: 5010,
  753. Protocol: "tcp",
  754. },
  755. },
  756. Privileged: true,
  757. ReadOnly: true,
  758. Restart: "always",
  759. SecurityOpt: []string{
  760. "label=level:s0:c100,c200",
  761. "label=type:svirt_apache_t",
  762. },
  763. StdinOpen: true,
  764. StopSignal: "SIGUSR1",
  765. StopGracePeriod: &stopGracePeriod,
  766. Tmpfs: []string{"/run", "/tmp"},
  767. Tty: true,
  768. Ulimits: map[string]*types.UlimitsConfig{
  769. "nproc": {
  770. Single: 65535,
  771. },
  772. "nofile": {
  773. Soft: 20000,
  774. Hard: 40000,
  775. },
  776. },
  777. User: "someone",
  778. Volumes: []string{
  779. "/var/lib/mysql",
  780. "/opt/data:/var/lib/mysql",
  781. fmt.Sprintf("%s:/code", workingDir),
  782. fmt.Sprintf("%s/static:/var/www/html", workingDir),
  783. fmt.Sprintf("%s/configs:/etc/configs/:ro", homeDir),
  784. "datavolume:/var/lib/mysql",
  785. },
  786. WorkingDir: "/code",
  787. }
  788. assert.Equal(t, []types.ServiceConfig{expectedServiceConfig}, config.Services)
  789. expectedNetworkConfig := map[string]types.NetworkConfig{
  790. "some-network": {},
  791. "other-network": {
  792. Driver: "overlay",
  793. DriverOpts: map[string]string{
  794. "foo": "bar",
  795. "baz": "1",
  796. },
  797. Ipam: types.IPAMConfig{
  798. Driver: "overlay",
  799. Config: []*types.IPAMPool{
  800. {Subnet: "172.16.238.0/24"},
  801. {Subnet: "2001:3984:3989::/64"},
  802. },
  803. },
  804. },
  805. "external-network": {
  806. External: types.External{
  807. Name: "external-network",
  808. External: true,
  809. },
  810. },
  811. "other-external-network": {
  812. External: types.External{
  813. Name: "my-cool-network",
  814. External: true,
  815. },
  816. },
  817. }
  818. assert.Equal(t, expectedNetworkConfig, config.Networks)
  819. expectedVolumeConfig := map[string]types.VolumeConfig{
  820. "some-volume": {},
  821. "other-volume": {
  822. Driver: "flocker",
  823. DriverOpts: map[string]string{
  824. "foo": "bar",
  825. "baz": "1",
  826. },
  827. },
  828. "external-volume": {
  829. External: types.External{
  830. Name: "external-volume",
  831. External: true,
  832. },
  833. },
  834. "other-external-volume": {
  835. External: types.External{
  836. Name: "my-cool-volume",
  837. External: true,
  838. },
  839. },
  840. }
  841. assert.Equal(t, expectedVolumeConfig, config.Volumes)
  842. }
  843. func loadYAML(yaml string) (*types.Config, error) {
  844. dict, err := ParseYAML([]byte(yaml))
  845. if err != nil {
  846. return nil, err
  847. }
  848. return Load(buildConfigDetails(dict))
  849. }
  850. func serviceSort(services []types.ServiceConfig) []types.ServiceConfig {
  851. sort.Sort(servicesByName(services))
  852. return services
  853. }
  854. type servicesByName []types.ServiceConfig
  855. func (sbn servicesByName) Len() int { return len(sbn) }
  856. func (sbn servicesByName) Swap(i, j int) { sbn[i], sbn[j] = sbn[j], sbn[i] }
  857. func (sbn servicesByName) Less(i, j int) bool { return sbn[i].Name < sbn[j].Name }
  858. func TestLoadAttachableNetwork(t *testing.T) {
  859. config, err := loadYAML(`
  860. version: "3.1"
  861. networks:
  862. mynet1:
  863. driver: overlay
  864. attachable: true
  865. mynet2:
  866. driver: bridge
  867. `)
  868. assert.NoError(t, err)
  869. expected := map[string]types.NetworkConfig{
  870. "mynet1": {
  871. Driver: "overlay",
  872. Attachable: true,
  873. },
  874. "mynet2": {
  875. Driver: "bridge",
  876. Attachable: false,
  877. },
  878. }
  879. assert.Equal(t, expected, config.Networks)
  880. }
  881. func TestLoadExpandedPortFormat(t *testing.T) {
  882. config, err := loadYAML(`
  883. version: "3.1"
  884. services:
  885. web:
  886. image: busybox
  887. ports:
  888. - "80-82:8080-8082"
  889. - "90-92:8090-8092/udp"
  890. - "85:8500"
  891. - 8600
  892. - protocol: udp
  893. target: 53
  894. published: 10053
  895. - mode: host
  896. target: 22
  897. published: 10022
  898. `)
  899. assert.NoError(t, err)
  900. expected := []types.ServicePortConfig{
  901. {
  902. Mode: "ingress",
  903. Target: 8080,
  904. Published: 80,
  905. Protocol: "tcp",
  906. },
  907. {
  908. Mode: "ingress",
  909. Target: 8081,
  910. Published: 81,
  911. Protocol: "tcp",
  912. },
  913. {
  914. Mode: "ingress",
  915. Target: 8082,
  916. Published: 82,
  917. Protocol: "tcp",
  918. },
  919. {
  920. Mode: "ingress",
  921. Target: 8090,
  922. Published: 90,
  923. Protocol: "udp",
  924. },
  925. {
  926. Mode: "ingress",
  927. Target: 8091,
  928. Published: 91,
  929. Protocol: "udp",
  930. },
  931. {
  932. Mode: "ingress",
  933. Target: 8092,
  934. Published: 92,
  935. Protocol: "udp",
  936. },
  937. {
  938. Mode: "ingress",
  939. Target: 8500,
  940. Published: 85,
  941. Protocol: "tcp",
  942. },
  943. {
  944. Mode: "ingress",
  945. Target: 8600,
  946. Published: 0,
  947. Protocol: "tcp",
  948. },
  949. {
  950. Target: 53,
  951. Published: 10053,
  952. Protocol: "udp",
  953. },
  954. {
  955. Mode: "host",
  956. Target: 22,
  957. Published: 10022,
  958. },
  959. }
  960. assert.Equal(t, 1, len(config.Services))
  961. assert.Equal(t, expected, config.Services[0].Ports)
  962. }