service_test.py 71 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785
  1. import os
  2. import re
  3. import shutil
  4. import tempfile
  5. from distutils.spawn import find_executable
  6. from io import StringIO
  7. from os import path
  8. import pytest
  9. from docker.errors import APIError
  10. from docker.errors import ImageNotFound
  11. from .. import mock
  12. from ..helpers import BUSYBOX_IMAGE_WITH_TAG
  13. from .testcases import docker_client
  14. from .testcases import DockerClientTestCase
  15. from .testcases import get_links
  16. from .testcases import pull_busybox
  17. from .testcases import SWARM_SKIP_CONTAINERS_ALL
  18. from .testcases import SWARM_SKIP_CPU_SHARES
  19. from compose import __version__
  20. from compose.config.types import MountSpec
  21. from compose.config.types import SecurityOpt
  22. from compose.config.types import VolumeFromSpec
  23. from compose.config.types import VolumeSpec
  24. from compose.const import IS_WINDOWS_PLATFORM
  25. from compose.const import LABEL_CONFIG_HASH
  26. from compose.const import LABEL_CONTAINER_NUMBER
  27. from compose.const import LABEL_ONE_OFF
  28. from compose.const import LABEL_PROJECT
  29. from compose.const import LABEL_SERVICE
  30. from compose.const import LABEL_VERSION
  31. from compose.container import Container
  32. from compose.errors import OperationFailedError
  33. from compose.parallel import ParallelStreamWriter
  34. from compose.project import OneOffFilter
  35. from compose.project import Project
  36. from compose.service import BuildAction
  37. from compose.service import ConvergencePlan
  38. from compose.service import ConvergenceStrategy
  39. from compose.service import IpcMode
  40. from compose.service import NetworkMode
  41. from compose.service import PidMode
  42. from compose.service import Service
  43. from compose.utils import parse_nanoseconds_int
  44. from tests.helpers import create_custom_host_file
  45. from tests.integration.testcases import is_cluster
  46. from tests.integration.testcases import no_cluster
  47. from tests.integration.testcases import v2_1_only
  48. from tests.integration.testcases import v2_2_only
  49. from tests.integration.testcases import v2_3_only
  50. from tests.integration.testcases import v2_only
  51. from tests.integration.testcases import v3_only
  52. def create_and_start_container(service, **override_options):
  53. container = service.create_container(**override_options)
  54. return service.start_container(container)
  55. class ServiceTest(DockerClientTestCase):
  56. def test_containers(self):
  57. foo = self.create_service('foo')
  58. bar = self.create_service('bar')
  59. create_and_start_container(foo)
  60. assert len(foo.containers()) == 1
  61. assert foo.containers()[0].name.startswith('composetest_foo_')
  62. assert len(bar.containers()) == 0
  63. create_and_start_container(bar)
  64. create_and_start_container(bar)
  65. assert len(foo.containers()) == 1
  66. assert len(bar.containers()) == 2
  67. names = [c.name for c in bar.containers()]
  68. assert len(names) == 2
  69. assert all(name.startswith('composetest_bar_') for name in names)
  70. def test_containers_one_off(self):
  71. db = self.create_service('db')
  72. container = db.create_container(one_off=True)
  73. assert db.containers(stopped=True) == []
  74. assert db.containers(one_off=OneOffFilter.only, stopped=True) == [container]
  75. def test_project_is_added_to_container_name(self):
  76. service = self.create_service('web')
  77. create_and_start_container(service)
  78. assert service.containers()[0].name.startswith('composetest_web_')
  79. def test_create_container_with_one_off(self):
  80. db = self.create_service('db')
  81. container = db.create_container(one_off=True)
  82. assert container.name.startswith('composetest_db_run_')
  83. def test_create_container_with_one_off_when_existing_container_is_running(self):
  84. db = self.create_service('db')
  85. db.start()
  86. container = db.create_container(one_off=True)
  87. assert container.name.startswith('composetest_db_run_')
  88. def test_create_container_with_unspecified_volume(self):
  89. service = self.create_service('db', volumes=[VolumeSpec.parse('/var/db')])
  90. container = service.create_container()
  91. service.start_container(container)
  92. assert container.get_mount('/var/db')
  93. def test_create_container_with_volume_driver(self):
  94. service = self.create_service('db', volume_driver='foodriver')
  95. container = service.create_container()
  96. service.start_container(container)
  97. assert 'foodriver' == container.get('HostConfig.VolumeDriver')
  98. @pytest.mark.skipif(SWARM_SKIP_CPU_SHARES, reason='Swarm --cpu-shares bug')
  99. def test_create_container_with_cpu_shares(self):
  100. service = self.create_service('db', cpu_shares=73)
  101. container = service.create_container()
  102. service.start_container(container)
  103. assert container.get('HostConfig.CpuShares') == 73
  104. def test_create_container_with_cpu_quota(self):
  105. service = self.create_service('db', cpu_quota=40000, cpu_period=150000)
  106. container = service.create_container()
  107. container.start()
  108. assert container.get('HostConfig.CpuQuota') == 40000
  109. assert container.get('HostConfig.CpuPeriod') == 150000
  110. @pytest.mark.xfail(raises=OperationFailedError, reason='not supported by kernel')
  111. def test_create_container_with_cpu_rt(self):
  112. service = self.create_service('db', cpu_rt_runtime=40000, cpu_rt_period=150000)
  113. container = service.create_container()
  114. container.start()
  115. assert container.get('HostConfig.CpuRealtimeRuntime') == 40000
  116. assert container.get('HostConfig.CpuRealtimePeriod') == 150000
  117. @v2_2_only()
  118. def test_create_container_with_cpu_count(self):
  119. self.require_api_version('1.25')
  120. service = self.create_service('db', cpu_count=2)
  121. container = service.create_container()
  122. service.start_container(container)
  123. assert container.get('HostConfig.CpuCount') == 2
  124. @v2_2_only()
  125. @pytest.mark.skipif(not IS_WINDOWS_PLATFORM, reason='cpu_percent is not supported for Linux')
  126. def test_create_container_with_cpu_percent(self):
  127. self.require_api_version('1.25')
  128. service = self.create_service('db', cpu_percent=12)
  129. container = service.create_container()
  130. service.start_container(container)
  131. assert container.get('HostConfig.CpuPercent') == 12
  132. @v2_2_only()
  133. def test_create_container_with_cpus(self):
  134. self.require_api_version('1.25')
  135. service = self.create_service('db', cpus=1)
  136. container = service.create_container()
  137. service.start_container(container)
  138. assert container.get('HostConfig.NanoCpus') == 1000000000
  139. def test_create_container_with_shm_size(self):
  140. self.require_api_version('1.22')
  141. service = self.create_service('db', shm_size=67108864)
  142. container = service.create_container()
  143. service.start_container(container)
  144. assert container.get('HostConfig.ShmSize') == 67108864
  145. def test_create_container_with_init_bool(self):
  146. self.require_api_version('1.25')
  147. service = self.create_service('db', init=True)
  148. container = service.create_container()
  149. service.start_container(container)
  150. assert container.get('HostConfig.Init') is True
  151. @pytest.mark.xfail(True, reason='Option has been removed in Engine 17.06.0')
  152. def test_create_container_with_init_path(self):
  153. self.require_api_version('1.25')
  154. docker_init_path = find_executable('docker-init')
  155. service = self.create_service('db', init=docker_init_path)
  156. container = service.create_container()
  157. service.start_container(container)
  158. assert container.get('HostConfig.InitPath') == docker_init_path
  159. @pytest.mark.xfail(True, reason='Some kernels/configs do not support pids_limit')
  160. def test_create_container_with_pids_limit(self):
  161. self.require_api_version('1.23')
  162. service = self.create_service('db', pids_limit=10)
  163. container = service.create_container()
  164. service.start_container(container)
  165. assert container.get('HostConfig.PidsLimit') == 10
  166. def test_create_container_with_extra_hosts_list(self):
  167. extra_hosts = ['somehost:162.242.195.82', 'otherhost:50.31.209.229']
  168. service = self.create_service('db', extra_hosts=extra_hosts)
  169. container = service.create_container()
  170. service.start_container(container)
  171. assert set(container.get('HostConfig.ExtraHosts')) == set(extra_hosts)
  172. def test_create_container_with_extra_hosts_dicts(self):
  173. extra_hosts = {'somehost': '162.242.195.82', 'otherhost': '50.31.209.229'}
  174. extra_hosts_list = ['somehost:162.242.195.82', 'otherhost:50.31.209.229']
  175. service = self.create_service('db', extra_hosts=extra_hosts)
  176. container = service.create_container()
  177. service.start_container(container)
  178. assert set(container.get('HostConfig.ExtraHosts')) == set(extra_hosts_list)
  179. def test_create_container_with_cpu_set(self):
  180. service = self.create_service('db', cpuset='0')
  181. container = service.create_container()
  182. service.start_container(container)
  183. assert container.get('HostConfig.CpusetCpus') == '0'
  184. def test_create_container_with_read_only_root_fs(self):
  185. read_only = True
  186. service = self.create_service('db', read_only=read_only)
  187. container = service.create_container()
  188. service.start_container(container)
  189. assert container.get('HostConfig.ReadonlyRootfs') == read_only
  190. @pytest.mark.xfail(True, reason='Getting "Your kernel does not support '
  191. 'cgroup blkio weight and weight_device" on daemon start '
  192. 'on Linux kernel 5.3.x')
  193. def test_create_container_with_blkio_config(self):
  194. blkio_config = {
  195. 'weight': 300,
  196. 'weight_device': [{'path': '/dev/sda', 'weight': 200}],
  197. 'device_read_bps': [{'path': '/dev/sda', 'rate': 1024 * 1024 * 100}],
  198. 'device_read_iops': [{'path': '/dev/sda', 'rate': 1000}],
  199. 'device_write_bps': [{'path': '/dev/sda', 'rate': 1024 * 1024}],
  200. 'device_write_iops': [{'path': '/dev/sda', 'rate': 800}]
  201. }
  202. service = self.create_service('web', blkio_config=blkio_config)
  203. container = service.create_container()
  204. assert container.get('HostConfig.BlkioWeight') == 300
  205. assert container.get('HostConfig.BlkioWeightDevice') == [{
  206. 'Path': '/dev/sda', 'Weight': 200
  207. }]
  208. assert container.get('HostConfig.BlkioDeviceReadBps') == [{
  209. 'Path': '/dev/sda', 'Rate': 1024 * 1024 * 100
  210. }]
  211. assert container.get('HostConfig.BlkioDeviceWriteBps') == [{
  212. 'Path': '/dev/sda', 'Rate': 1024 * 1024
  213. }]
  214. assert container.get('HostConfig.BlkioDeviceReadIOps') == [{
  215. 'Path': '/dev/sda', 'Rate': 1000
  216. }]
  217. assert container.get('HostConfig.BlkioDeviceWriteIOps') == [{
  218. 'Path': '/dev/sda', 'Rate': 800
  219. }]
  220. def test_create_container_with_security_opt(self):
  221. security_opt = [SecurityOpt.parse('label:disable')]
  222. service = self.create_service('db', security_opt=security_opt)
  223. container = service.create_container()
  224. service.start_container(container)
  225. assert set(container.get('HostConfig.SecurityOpt')) == set([o.repr() for o in security_opt])
  226. @pytest.mark.xfail(True, reason='Not supported on most drivers')
  227. def test_create_container_with_storage_opt(self):
  228. storage_opt = {'size': '1G'}
  229. service = self.create_service('db', storage_opt=storage_opt)
  230. container = service.create_container()
  231. service.start_container(container)
  232. assert container.get('HostConfig.StorageOpt') == storage_opt
  233. def test_create_container_with_oom_kill_disable(self):
  234. self.require_api_version('1.20')
  235. service = self.create_service('db', oom_kill_disable=True)
  236. container = service.create_container()
  237. assert container.get('HostConfig.OomKillDisable') is True
  238. def test_create_container_with_mac_address(self):
  239. service = self.create_service('db', mac_address='02:42:ac:11:65:43')
  240. container = service.create_container()
  241. service.start_container(container)
  242. assert container.inspect()['Config']['MacAddress'] == '02:42:ac:11:65:43'
  243. def test_create_container_with_device_cgroup_rules(self):
  244. service = self.create_service('db', device_cgroup_rules=['c 7:128 rwm'])
  245. container = service.create_container()
  246. assert container.get('HostConfig.DeviceCgroupRules') == ['c 7:128 rwm']
  247. def test_create_container_with_specified_volume(self):
  248. host_path = '/tmp/host-path'
  249. container_path = '/container-path'
  250. service = self.create_service(
  251. 'db',
  252. volumes=[VolumeSpec(host_path, container_path, 'rw')])
  253. container = service.create_container()
  254. service.start_container(container)
  255. assert container.get_mount(container_path)
  256. # Match the last component ("host-path"), because boot2docker symlinks /tmp
  257. actual_host_path = container.get_mount(container_path)['Source']
  258. assert path.basename(actual_host_path) == path.basename(host_path), (
  259. "Last component differs: %s, %s" % (actual_host_path, host_path)
  260. )
  261. @v2_3_only()
  262. def test_create_container_with_host_mount(self):
  263. host_path = '/tmp/host-path'
  264. container_path = '/container-path'
  265. create_custom_host_file(self.client, path.join(host_path, 'a.txt'), 'test')
  266. service = self.create_service(
  267. 'db',
  268. volumes=[
  269. MountSpec(type='bind', source=host_path, target=container_path, read_only=True)
  270. ]
  271. )
  272. container = service.create_container()
  273. service.start_container(container)
  274. mount = container.get_mount(container_path)
  275. assert mount
  276. assert path.basename(mount['Source']) == path.basename(host_path)
  277. assert mount['RW'] is False
  278. @v2_3_only()
  279. def test_create_container_with_tmpfs_mount(self):
  280. container_path = '/container-tmpfs'
  281. service = self.create_service(
  282. 'db',
  283. volumes=[MountSpec(type='tmpfs', target=container_path)]
  284. )
  285. container = service.create_container()
  286. service.start_container(container)
  287. mount = container.get_mount(container_path)
  288. assert mount
  289. assert mount['Type'] == 'tmpfs'
  290. @v2_3_only()
  291. def test_create_container_with_tmpfs_mount_tmpfs_size(self):
  292. container_path = '/container-tmpfs'
  293. service = self.create_service(
  294. 'db',
  295. volumes=[MountSpec(type='tmpfs', target=container_path, tmpfs={'size': 5368709})]
  296. )
  297. container = service.create_container()
  298. service.start_container(container)
  299. mount = container.get_mount(container_path)
  300. assert mount
  301. print(container.dictionary)
  302. assert mount['Type'] == 'tmpfs'
  303. assert container.get('HostConfig.Mounts')[0]['TmpfsOptions'] == {
  304. 'SizeBytes': 5368709
  305. }
  306. @v2_3_only()
  307. def test_create_container_with_volume_mount(self):
  308. container_path = '/container-volume'
  309. volume_name = 'composetest_abcde'
  310. self.client.create_volume(volume_name)
  311. service = self.create_service(
  312. 'db',
  313. volumes=[MountSpec(type='volume', source=volume_name, target=container_path)]
  314. )
  315. container = service.create_container()
  316. service.start_container(container)
  317. mount = container.get_mount(container_path)
  318. assert mount
  319. assert mount['Name'] == volume_name
  320. @v3_only()
  321. def test_create_container_with_legacy_mount(self):
  322. # Ensure mounts are converted to volumes if API version < 1.30
  323. # Needed to support long syntax in the 3.2 format
  324. client = docker_client({}, version='1.25')
  325. container_path = '/container-volume'
  326. volume_name = 'composetest_abcde'
  327. self.client.create_volume(volume_name)
  328. service = Service('db', client=client, volumes=[
  329. MountSpec(type='volume', source=volume_name, target=container_path)
  330. ], image=BUSYBOX_IMAGE_WITH_TAG, command=['top'], project='composetest')
  331. container = service.create_container()
  332. service.start_container(container)
  333. mount = container.get_mount(container_path)
  334. assert mount
  335. assert mount['Name'] == volume_name
  336. @v3_only()
  337. def test_create_container_with_legacy_tmpfs_mount(self):
  338. # Ensure tmpfs mounts are converted to tmpfs entries if API version < 1.30
  339. # Needed to support long syntax in the 3.2 format
  340. client = docker_client({}, version='1.25')
  341. container_path = '/container-tmpfs'
  342. service = Service('db', client=client, volumes=[
  343. MountSpec(type='tmpfs', target=container_path)
  344. ], image=BUSYBOX_IMAGE_WITH_TAG, command=['top'], project='composetest')
  345. container = service.create_container()
  346. service.start_container(container)
  347. mount = container.get_mount(container_path)
  348. assert mount is None
  349. assert container_path in container.get('HostConfig.Tmpfs')
  350. def test_create_container_with_healthcheck_config(self):
  351. one_second = parse_nanoseconds_int('1s')
  352. healthcheck = {
  353. 'test': ['true'],
  354. 'interval': 2 * one_second,
  355. 'timeout': 5 * one_second,
  356. 'retries': 5,
  357. 'start_period': 2 * one_second
  358. }
  359. service = self.create_service('db', healthcheck=healthcheck)
  360. container = service.create_container()
  361. remote_healthcheck = container.get('Config.Healthcheck')
  362. assert remote_healthcheck['Test'] == healthcheck['test']
  363. assert remote_healthcheck['Interval'] == healthcheck['interval']
  364. assert remote_healthcheck['Timeout'] == healthcheck['timeout']
  365. assert remote_healthcheck['Retries'] == healthcheck['retries']
  366. assert remote_healthcheck['StartPeriod'] == healthcheck['start_period']
  367. def test_recreate_preserves_volume_with_trailing_slash(self):
  368. """When the Compose file specifies a trailing slash in the container path, make
  369. sure we copy the volume over when recreating.
  370. """
  371. service = self.create_service('data', volumes=[VolumeSpec.parse('/data/')])
  372. old_container = create_and_start_container(service)
  373. volume_path = old_container.get_mount('/data')['Source']
  374. new_container = service.recreate_container(old_container)
  375. assert new_container.get_mount('/data')['Source'] == volume_path
  376. def test_recreate_volume_to_mount(self):
  377. # https://github.com/docker/compose/issues/6280
  378. service = Service(
  379. project='composetest',
  380. name='db',
  381. client=self.client,
  382. build={'context': 'tests/fixtures/dockerfile-with-volume'},
  383. volumes=[MountSpec.parse({
  384. 'type': 'volume',
  385. 'target': '/data',
  386. })]
  387. )
  388. old_container = create_and_start_container(service)
  389. new_container = service.recreate_container(old_container)
  390. assert new_container.get_mount('/data')['Source']
  391. def test_duplicate_volume_trailing_slash(self):
  392. """
  393. When an image specifies a volume, and the Compose file specifies a host path
  394. but adds a trailing slash, make sure that we don't create duplicate binds.
  395. """
  396. host_path = '/tmp/data'
  397. container_path = '/data'
  398. volumes = [VolumeSpec.parse('{}:{}/'.format(host_path, container_path))]
  399. tmp_container = self.client.create_container(
  400. 'busybox', 'true',
  401. volumes={container_path: {}},
  402. labels={'com.docker.compose.test_image': 'true'},
  403. host_config={}
  404. )
  405. image = self.client.commit(tmp_container)['Id']
  406. service = self.create_service('db', image=image, volumes=volumes)
  407. old_container = create_and_start_container(service)
  408. assert old_container.get('Config.Volumes') == {container_path: {}}
  409. service = self.create_service('db', image=image, volumes=volumes)
  410. new_container = service.recreate_container(old_container)
  411. assert new_container.get('Config.Volumes') == {container_path: {}}
  412. assert service.containers(stopped=False) == [new_container]
  413. def test_create_container_with_volumes_from(self):
  414. volume_service = self.create_service('data')
  415. volume_container_1 = volume_service.create_container()
  416. volume_container_2 = Container.create(
  417. self.client,
  418. image=BUSYBOX_IMAGE_WITH_TAG,
  419. command=["top"],
  420. labels={LABEL_PROJECT: 'composetest'},
  421. host_config={},
  422. environment=['affinity:container=={}'.format(volume_container_1.id)],
  423. )
  424. host_service = self.create_service(
  425. 'host',
  426. volumes_from=[
  427. VolumeFromSpec(volume_service, 'rw', 'service'),
  428. VolumeFromSpec(volume_container_2, 'rw', 'container')
  429. ],
  430. environment=['affinity:container=={}'.format(volume_container_1.id)],
  431. )
  432. host_container = host_service.create_container()
  433. host_service.start_container(host_container)
  434. assert volume_container_1.id + ':rw' in host_container.get('HostConfig.VolumesFrom')
  435. assert volume_container_2.id + ':rw' in host_container.get('HostConfig.VolumesFrom')
  436. def test_execute_convergence_plan_recreate(self):
  437. service = self.create_service(
  438. 'db',
  439. environment={'FOO': '1'},
  440. volumes=[VolumeSpec.parse('/etc')],
  441. entrypoint=['top'],
  442. command=['-d', '1']
  443. )
  444. old_container = service.create_container()
  445. assert old_container.get('Config.Entrypoint') == ['top']
  446. assert old_container.get('Config.Cmd') == ['-d', '1']
  447. assert 'FOO=1' in old_container.get('Config.Env')
  448. assert old_container.name.startswith('composetest_db_')
  449. service.start_container(old_container)
  450. old_container.inspect() # reload volume data
  451. volume_path = old_container.get_mount('/etc')['Source']
  452. num_containers_before = len(self.client.containers(all=True))
  453. service.options['environment']['FOO'] = '2'
  454. new_container, = service.execute_convergence_plan(
  455. ConvergencePlan('recreate', [old_container]))
  456. assert new_container.get('Config.Entrypoint') == ['top']
  457. assert new_container.get('Config.Cmd') == ['-d', '1']
  458. assert 'FOO=2' in new_container.get('Config.Env')
  459. assert new_container.name.startswith('composetest_db_')
  460. assert new_container.get_mount('/etc')['Source'] == volume_path
  461. if not is_cluster(self.client):
  462. assert (
  463. 'affinity:container==%s' % old_container.id in
  464. new_container.get('Config.Env')
  465. )
  466. else:
  467. # In Swarm, the env marker is consumed and the container should be deployed
  468. # on the same node.
  469. assert old_container.get('Node.Name') == new_container.get('Node.Name')
  470. assert len(self.client.containers(all=True)) == num_containers_before
  471. assert old_container.id != new_container.id
  472. with pytest.raises(APIError):
  473. self.client.inspect_container(old_container.id)
  474. def test_execute_convergence_plan_recreate_change_mount_target(self):
  475. service = self.create_service(
  476. 'db',
  477. volumes=[MountSpec(target='/app1', type='volume')],
  478. entrypoint=['top'], command=['-d', '1']
  479. )
  480. old_container = create_and_start_container(service)
  481. assert (
  482. [mount['Destination'] for mount in old_container.get('Mounts')] ==
  483. ['/app1']
  484. )
  485. service.options['volumes'] = [MountSpec(target='/app2', type='volume')]
  486. new_container, = service.execute_convergence_plan(
  487. ConvergencePlan('recreate', [old_container])
  488. )
  489. assert (
  490. [mount['Destination'] for mount in new_container.get('Mounts')] ==
  491. ['/app2']
  492. )
  493. def test_execute_convergence_plan_recreate_twice(self):
  494. service = self.create_service(
  495. 'db',
  496. volumes=[VolumeSpec.parse('/etc')],
  497. entrypoint=['top'],
  498. command=['-d', '1'])
  499. orig_container = service.create_container()
  500. service.start_container(orig_container)
  501. orig_container.inspect() # reload volume data
  502. volume_path = orig_container.get_mount('/etc')['Source']
  503. # Do this twice to reproduce the bug
  504. for _ in range(2):
  505. new_container, = service.execute_convergence_plan(
  506. ConvergencePlan('recreate', [orig_container]))
  507. assert new_container.get_mount('/etc')['Source'] == volume_path
  508. if not is_cluster(self.client):
  509. assert ('affinity:container==%s' % orig_container.id in
  510. new_container.get('Config.Env'))
  511. else:
  512. # In Swarm, the env marker is consumed and the container should be deployed
  513. # on the same node.
  514. assert orig_container.get('Node.Name') == new_container.get('Node.Name')
  515. orig_container = new_container
  516. @v2_3_only()
  517. def test_execute_convergence_plan_recreate_twice_with_mount(self):
  518. service = self.create_service(
  519. 'db',
  520. volumes=[MountSpec(target='/etc', type='volume')],
  521. entrypoint=['top'],
  522. command=['-d', '1']
  523. )
  524. orig_container = service.create_container()
  525. service.start_container(orig_container)
  526. orig_container.inspect() # reload volume data
  527. volume_path = orig_container.get_mount('/etc')['Source']
  528. # Do this twice to reproduce the bug
  529. for _ in range(2):
  530. new_container, = service.execute_convergence_plan(
  531. ConvergencePlan('recreate', [orig_container])
  532. )
  533. assert new_container.get_mount('/etc')['Source'] == volume_path
  534. if not is_cluster(self.client):
  535. assert ('affinity:container==%s' % orig_container.id in
  536. new_container.get('Config.Env'))
  537. else:
  538. # In Swarm, the env marker is consumed and the container should be deployed
  539. # on the same node.
  540. assert orig_container.get('Node.Name') == new_container.get('Node.Name')
  541. orig_container = new_container
  542. def test_execute_convergence_plan_when_containers_are_stopped(self):
  543. service = self.create_service(
  544. 'db',
  545. environment={'FOO': '1'},
  546. volumes=[VolumeSpec.parse('/var/db')],
  547. entrypoint=['top'],
  548. command=['-d', '1']
  549. )
  550. service.create_container()
  551. containers = service.containers(stopped=True)
  552. assert len(containers) == 1
  553. container, = containers
  554. assert not container.is_running
  555. service.execute_convergence_plan(ConvergencePlan('start', [container]))
  556. containers = service.containers()
  557. assert len(containers) == 1
  558. container.inspect()
  559. assert container == containers[0]
  560. assert container.is_running
  561. def test_execute_convergence_plan_with_image_declared_volume(self):
  562. service = Service(
  563. project='composetest',
  564. name='db',
  565. client=self.client,
  566. build={'context': 'tests/fixtures/dockerfile-with-volume'},
  567. )
  568. old_container = create_and_start_container(service)
  569. assert [mount['Destination'] for mount in old_container.get('Mounts')] == ['/data']
  570. volume_path = old_container.get_mount('/data')['Source']
  571. new_container, = service.execute_convergence_plan(
  572. ConvergencePlan('recreate', [old_container]))
  573. assert [mount['Destination'] for mount in new_container.get('Mounts')] == ['/data']
  574. assert new_container.get_mount('/data')['Source'] == volume_path
  575. def test_execute_convergence_plan_with_image_declared_volume_renew(self):
  576. service = Service(
  577. project='composetest',
  578. name='db',
  579. client=self.client,
  580. build={'context': 'tests/fixtures/dockerfile-with-volume'},
  581. )
  582. old_container = create_and_start_container(service)
  583. assert [mount['Destination'] for mount in old_container.get('Mounts')] == ['/data']
  584. volume_path = old_container.get_mount('/data')['Source']
  585. new_container, = service.execute_convergence_plan(
  586. ConvergencePlan('recreate', [old_container]), renew_anonymous_volumes=True
  587. )
  588. assert [mount['Destination'] for mount in new_container.get('Mounts')] == ['/data']
  589. assert new_container.get_mount('/data')['Source'] != volume_path
  590. def test_execute_convergence_plan_when_image_volume_masks_config(self):
  591. service = self.create_service(
  592. 'db',
  593. build={'context': 'tests/fixtures/dockerfile-with-volume'},
  594. )
  595. old_container = create_and_start_container(service)
  596. assert [mount['Destination'] for mount in old_container.get('Mounts')] == ['/data']
  597. volume_path = old_container.get_mount('/data')['Source']
  598. service.options['volumes'] = [VolumeSpec.parse('/tmp:/data')]
  599. with mock.patch('compose.service.log') as mock_log:
  600. new_container, = service.execute_convergence_plan(
  601. ConvergencePlan('recreate', [old_container]))
  602. mock_log.warning.assert_called_once_with(mock.ANY)
  603. _, args, kwargs = mock_log.warning.mock_calls[0]
  604. assert "Service \"db\" is using volume \"/data\" from the previous container" in args[0]
  605. assert [mount['Destination'] for mount in new_container.get('Mounts')] == ['/data']
  606. assert new_container.get_mount('/data')['Source'] == volume_path
  607. def test_execute_convergence_plan_when_host_volume_is_removed(self):
  608. host_path = '/tmp/host-path'
  609. service = self.create_service(
  610. 'db',
  611. build={'context': 'tests/fixtures/dockerfile-with-volume'},
  612. volumes=[VolumeSpec(host_path, '/data', 'rw')])
  613. old_container = create_and_start_container(service)
  614. assert (
  615. [mount['Destination'] for mount in old_container.get('Mounts')] ==
  616. ['/data']
  617. )
  618. service.options['volumes'] = []
  619. with mock.patch('compose.service.log', autospec=True) as mock_log:
  620. new_container, = service.execute_convergence_plan(
  621. ConvergencePlan('recreate', [old_container]))
  622. assert not mock_log.warn.called
  623. assert (
  624. [mount['Destination'] for mount in new_container.get('Mounts')] ==
  625. ['/data']
  626. )
  627. assert new_container.get_mount('/data')['Source'] != host_path
  628. def test_execute_convergence_plan_anonymous_volume_renew(self):
  629. service = self.create_service(
  630. 'db',
  631. image='busybox',
  632. volumes=[VolumeSpec(None, '/data', 'rw')])
  633. old_container = create_and_start_container(service)
  634. assert (
  635. [mount['Destination'] for mount in old_container.get('Mounts')] ==
  636. ['/data']
  637. )
  638. volume_path = old_container.get_mount('/data')['Source']
  639. new_container, = service.execute_convergence_plan(
  640. ConvergencePlan('recreate', [old_container]),
  641. renew_anonymous_volumes=True
  642. )
  643. assert (
  644. [mount['Destination'] for mount in new_container.get('Mounts')] ==
  645. ['/data']
  646. )
  647. assert new_container.get_mount('/data')['Source'] != volume_path
  648. def test_execute_convergence_plan_anonymous_volume_recreate_then_renew(self):
  649. service = self.create_service(
  650. 'db',
  651. image='busybox',
  652. volumes=[VolumeSpec(None, '/data', 'rw')])
  653. old_container = create_and_start_container(service)
  654. assert (
  655. [mount['Destination'] for mount in old_container.get('Mounts')] ==
  656. ['/data']
  657. )
  658. volume_path = old_container.get_mount('/data')['Source']
  659. mid_container, = service.execute_convergence_plan(
  660. ConvergencePlan('recreate', [old_container]),
  661. )
  662. assert (
  663. [mount['Destination'] for mount in mid_container.get('Mounts')] ==
  664. ['/data']
  665. )
  666. assert mid_container.get_mount('/data')['Source'] == volume_path
  667. new_container, = service.execute_convergence_plan(
  668. ConvergencePlan('recreate', [mid_container]),
  669. renew_anonymous_volumes=True
  670. )
  671. assert (
  672. [mount['Destination'] for mount in new_container.get('Mounts')] ==
  673. ['/data']
  674. )
  675. assert new_container.get_mount('/data')['Source'] != volume_path
  676. def test_execute_convergence_plan_without_start(self):
  677. service = self.create_service(
  678. 'db',
  679. build={'context': 'tests/fixtures/dockerfile-with-volume'}
  680. )
  681. containers = service.execute_convergence_plan(ConvergencePlan('create', []), start=False)
  682. service_containers = service.containers(stopped=True)
  683. assert len(service_containers) == 1
  684. assert not service_containers[0].is_running
  685. containers = service.execute_convergence_plan(
  686. ConvergencePlan('recreate', containers),
  687. start=False)
  688. service_containers = service.containers(stopped=True)
  689. assert len(service_containers) == 1
  690. assert not service_containers[0].is_running
  691. service.execute_convergence_plan(ConvergencePlan('start', containers), start=False)
  692. service_containers = service.containers(stopped=True)
  693. assert len(service_containers) == 1
  694. assert not service_containers[0].is_running
  695. def test_execute_convergence_plan_image_with_volume_is_removed(self):
  696. service = self.create_service(
  697. 'db', build={'context': 'tests/fixtures/dockerfile-with-volume'}
  698. )
  699. old_container = create_and_start_container(service)
  700. assert (
  701. [mount['Destination'] for mount in old_container.get('Mounts')] ==
  702. ['/data']
  703. )
  704. volume_path = old_container.get_mount('/data')['Source']
  705. old_container.stop()
  706. self.client.remove_image(service.image(), force=True)
  707. service.ensure_image_exists()
  708. with pytest.raises(ImageNotFound):
  709. service.execute_convergence_plan(
  710. ConvergencePlan('recreate', [old_container])
  711. )
  712. old_container.inspect() # retrieve new name from server
  713. new_container, = service.execute_convergence_plan(
  714. ConvergencePlan('recreate', [old_container]),
  715. reset_container_image=True
  716. )
  717. assert [mount['Destination'] for mount in new_container.get('Mounts')] == ['/data']
  718. assert new_container.get_mount('/data')['Source'] == volume_path
  719. def test_start_container_passes_through_options(self):
  720. db = self.create_service('db')
  721. create_and_start_container(db, environment={'FOO': 'BAR'})
  722. assert db.containers()[0].environment['FOO'] == 'BAR'
  723. def test_start_container_inherits_options_from_constructor(self):
  724. db = self.create_service('db', environment={'FOO': 'BAR'})
  725. create_and_start_container(db)
  726. assert db.containers()[0].environment['FOO'] == 'BAR'
  727. @no_cluster('No legacy links support in Swarm')
  728. def test_start_container_creates_links(self):
  729. db = self.create_service('db')
  730. web = self.create_service('web', links=[(db, None)])
  731. db1 = create_and_start_container(db)
  732. db2 = create_and_start_container(db)
  733. create_and_start_container(web)
  734. assert set(get_links(web.containers()[0])) == set([
  735. db1.name, db1.name_without_project,
  736. db2.name, db2.name_without_project,
  737. 'db'
  738. ])
  739. @no_cluster('No legacy links support in Swarm')
  740. def test_start_container_creates_links_with_names(self):
  741. db = self.create_service('db')
  742. web = self.create_service('web', links=[(db, 'custom_link_name')])
  743. db1 = create_and_start_container(db)
  744. db2 = create_and_start_container(db)
  745. create_and_start_container(web)
  746. assert set(get_links(web.containers()[0])) == set([
  747. db1.name, db1.name_without_project,
  748. db2.name, db2.name_without_project,
  749. 'custom_link_name'
  750. ])
  751. @no_cluster('No legacy links support in Swarm')
  752. def test_start_container_with_external_links(self):
  753. db = self.create_service('db')
  754. db_ctnrs = [create_and_start_container(db) for _ in range(3)]
  755. web = self.create_service(
  756. 'web', external_links=[
  757. db_ctnrs[0].name,
  758. db_ctnrs[1].name,
  759. '{}:db_3'.format(db_ctnrs[2].name)
  760. ]
  761. )
  762. create_and_start_container(web)
  763. assert set(get_links(web.containers()[0])) == set([
  764. db_ctnrs[0].name,
  765. db_ctnrs[1].name,
  766. 'db_3'
  767. ])
  768. @no_cluster('No legacy links support in Swarm')
  769. def test_start_normal_container_does_not_create_links_to_its_own_service(self):
  770. db = self.create_service('db')
  771. create_and_start_container(db)
  772. create_and_start_container(db)
  773. c = create_and_start_container(db)
  774. assert set(get_links(c)) == set([])
  775. @no_cluster('No legacy links support in Swarm')
  776. def test_start_one_off_container_creates_links_to_its_own_service(self):
  777. db = self.create_service('db')
  778. db1 = create_and_start_container(db)
  779. db2 = create_and_start_container(db)
  780. c = create_and_start_container(db, one_off=OneOffFilter.only)
  781. assert set(get_links(c)) == set([
  782. db1.name, db1.name_without_project,
  783. db2.name, db2.name_without_project,
  784. 'db'
  785. ])
  786. def test_start_container_builds_images(self):
  787. service = Service(
  788. name='test',
  789. client=self.client,
  790. build={'context': 'tests/fixtures/simple-dockerfile'},
  791. project='composetest',
  792. )
  793. container = create_and_start_container(service)
  794. container.wait()
  795. assert b'success' in container.logs()
  796. assert len(self.client.images(name='composetest_test')) >= 1
  797. def test_start_container_uses_tagged_image_if_it_exists(self):
  798. self.check_build('tests/fixtures/simple-dockerfile', tag='composetest_test')
  799. service = Service(
  800. name='test',
  801. client=self.client,
  802. build={'context': 'this/does/not/exist/and/will/throw/error'},
  803. project='composetest',
  804. )
  805. container = create_and_start_container(service)
  806. container.wait()
  807. assert b'success' in container.logs()
  808. def test_start_container_creates_ports(self):
  809. service = self.create_service('web', ports=[8000])
  810. container = create_and_start_container(service).inspect()
  811. assert list(container['NetworkSettings']['Ports'].keys()) == ['8000/tcp']
  812. assert container['NetworkSettings']['Ports']['8000/tcp'][0]['HostPort'] != '8000'
  813. def test_build(self):
  814. base_dir = tempfile.mkdtemp()
  815. self.addCleanup(shutil.rmtree, base_dir)
  816. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  817. f.write("FROM busybox\n")
  818. service = self.create_service('web', build={'context': base_dir})
  819. service.build()
  820. self.addCleanup(self.client.remove_image, service.image_name)
  821. assert self.client.inspect_image('composetest_web')
  822. def test_build_cli(self):
  823. base_dir = tempfile.mkdtemp()
  824. self.addCleanup(shutil.rmtree, base_dir)
  825. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  826. f.write("FROM busybox\n")
  827. service = self.create_service('web',
  828. build={'context': base_dir},
  829. environment={
  830. 'COMPOSE_DOCKER_CLI_BUILD': '1',
  831. 'DOCKER_BUILDKIT': '1',
  832. })
  833. service.build(cli=True)
  834. self.addCleanup(self.client.remove_image, service.image_name)
  835. assert self.client.inspect_image('composetest_web')
  836. def test_build_cli_with_build_labels(self):
  837. base_dir = tempfile.mkdtemp()
  838. self.addCleanup(shutil.rmtree, base_dir)
  839. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  840. f.write("FROM busybox\n")
  841. service = self.create_service('web',
  842. build={
  843. 'context': base_dir,
  844. 'labels': {'com.docker.compose.test': 'true'}},
  845. )
  846. service.build(cli=True)
  847. self.addCleanup(self.client.remove_image, service.image_name)
  848. image = self.client.inspect_image('composetest_web')
  849. assert image['Config']['Labels']['com.docker.compose.test']
  850. def test_up_build_cli(self):
  851. base_dir = tempfile.mkdtemp()
  852. self.addCleanup(shutil.rmtree, base_dir)
  853. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  854. f.write("FROM busybox\n")
  855. web = self.create_service('web',
  856. build={'context': base_dir},
  857. environment={
  858. 'COMPOSE_DOCKER_CLI_BUILD': '1',
  859. 'DOCKER_BUILDKIT': '1',
  860. })
  861. project = Project('composetest', [web], self.client)
  862. project.up(do_build=BuildAction.force)
  863. containers = project.containers(['web'])
  864. assert len(containers) == 1
  865. assert containers[0].name.startswith('composetest_web_')
  866. def test_build_non_ascii_filename(self):
  867. base_dir = tempfile.mkdtemp()
  868. self.addCleanup(shutil.rmtree, base_dir)
  869. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  870. f.write("FROM busybox\n")
  871. with open(os.path.join(base_dir.encode('utf8'), b'foo\xE2bar'), 'w') as f:
  872. f.write("hello world\n")
  873. service = self.create_service('web', build={'context': str(base_dir)})
  874. service.build()
  875. self.addCleanup(self.client.remove_image, service.image_name)
  876. assert self.client.inspect_image('composetest_web')
  877. def test_build_with_image_name(self):
  878. base_dir = tempfile.mkdtemp()
  879. self.addCleanup(shutil.rmtree, base_dir)
  880. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  881. f.write("FROM busybox\n")
  882. image_name = 'examples/composetest:latest'
  883. self.addCleanup(self.client.remove_image, image_name)
  884. self.create_service('web', build={'context': base_dir}, image=image_name).build()
  885. assert self.client.inspect_image(image_name)
  886. def test_build_with_git_url(self):
  887. build_url = "https://github.com/dnephin/docker-build-from-url.git"
  888. service = self.create_service('buildwithurl', build={'context': build_url})
  889. self.addCleanup(self.client.remove_image, service.image_name)
  890. service.build()
  891. assert service.image()
  892. def test_build_with_build_args(self):
  893. base_dir = tempfile.mkdtemp()
  894. self.addCleanup(shutil.rmtree, base_dir)
  895. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  896. f.write("FROM busybox\n")
  897. f.write("ARG build_version\n")
  898. f.write("RUN echo ${build_version}\n")
  899. service = self.create_service('buildwithargs',
  900. build={'context': str(base_dir),
  901. 'args': {"build_version": "1"}})
  902. service.build()
  903. self.addCleanup(self.client.remove_image, service.image_name)
  904. assert service.image()
  905. assert "build_version=1" in service.image()['ContainerConfig']['Cmd']
  906. def test_build_with_build_args_override(self):
  907. base_dir = tempfile.mkdtemp()
  908. self.addCleanup(shutil.rmtree, base_dir)
  909. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  910. f.write("FROM busybox\n")
  911. f.write("ARG build_version\n")
  912. f.write("RUN echo ${build_version}\n")
  913. service = self.create_service('buildwithargs',
  914. build={'context': str(base_dir),
  915. 'args': {"build_version": "1"}})
  916. service.build(build_args_override={'build_version': '2'})
  917. self.addCleanup(self.client.remove_image, service.image_name)
  918. assert service.image()
  919. assert "build_version=2" in service.image()['ContainerConfig']['Cmd']
  920. def test_build_with_build_labels(self):
  921. base_dir = tempfile.mkdtemp()
  922. self.addCleanup(shutil.rmtree, base_dir)
  923. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  924. f.write('FROM busybox\n')
  925. service = self.create_service('buildlabels', build={
  926. 'context': str(base_dir),
  927. 'labels': {'com.docker.compose.test': 'true'}
  928. })
  929. service.build()
  930. self.addCleanup(self.client.remove_image, service.image_name)
  931. assert service.image()
  932. assert service.image()['Config']['Labels']['com.docker.compose.test'] == 'true'
  933. @no_cluster('Container networks not on Swarm')
  934. def test_build_with_network(self):
  935. base_dir = tempfile.mkdtemp()
  936. self.addCleanup(shutil.rmtree, base_dir)
  937. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  938. f.write('FROM busybox\n')
  939. f.write('RUN ping -c1 google.local\n')
  940. net_container = self.client.create_container(
  941. 'busybox', 'top', host_config=self.client.create_host_config(
  942. extra_hosts={'google.local': '127.0.0.1'}
  943. ), name='composetest_build_network'
  944. )
  945. self.addCleanup(self.client.remove_container, net_container, force=True)
  946. self.client.start(net_container)
  947. service = self.create_service('buildwithnet', build={
  948. 'context': str(base_dir),
  949. 'network': 'container:{}'.format(net_container['Id'])
  950. })
  951. service.build()
  952. self.addCleanup(self.client.remove_image, service.image_name)
  953. assert service.image()
  954. @v2_3_only()
  955. @no_cluster('Not supported on UCP 2.2.0-beta1') # FIXME: remove once support is added
  956. def test_build_with_target(self):
  957. self.require_api_version('1.30')
  958. base_dir = tempfile.mkdtemp()
  959. self.addCleanup(shutil.rmtree, base_dir)
  960. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  961. f.write('FROM busybox as one\n')
  962. f.write('LABEL com.docker.compose.test=true\n')
  963. f.write('LABEL com.docker.compose.test.target=one\n')
  964. f.write('FROM busybox as two\n')
  965. f.write('LABEL com.docker.compose.test.target=two\n')
  966. service = self.create_service('buildtarget', build={
  967. 'context': str(base_dir),
  968. 'target': 'one'
  969. })
  970. service.build()
  971. assert service.image()
  972. assert service.image()['Config']['Labels']['com.docker.compose.test.target'] == 'one'
  973. @v2_3_only()
  974. def test_build_with_extra_hosts(self):
  975. self.require_api_version('1.27')
  976. base_dir = tempfile.mkdtemp()
  977. self.addCleanup(shutil.rmtree, base_dir)
  978. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  979. f.write('\n'.join([
  980. 'FROM busybox',
  981. 'RUN ping -c1 foobar',
  982. 'RUN ping -c1 baz',
  983. ]))
  984. service = self.create_service('build_extra_hosts', build={
  985. 'context': str(base_dir),
  986. 'extra_hosts': {
  987. 'foobar': '127.0.0.1',
  988. 'baz': '127.0.0.1'
  989. }
  990. })
  991. service.build()
  992. assert service.image()
  993. def test_build_with_gzip(self):
  994. base_dir = tempfile.mkdtemp()
  995. self.addCleanup(shutil.rmtree, base_dir)
  996. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  997. f.write('\n'.join([
  998. 'FROM busybox',
  999. 'COPY . /src',
  1000. 'RUN cat /src/hello.txt'
  1001. ]))
  1002. with open(os.path.join(base_dir, 'hello.txt'), 'w') as f:
  1003. f.write('hello world\n')
  1004. service = self.create_service('build_gzip', build={
  1005. 'context': str(base_dir),
  1006. })
  1007. service.build(gzip=True)
  1008. assert service.image()
  1009. @v2_1_only()
  1010. def test_build_with_isolation(self):
  1011. base_dir = tempfile.mkdtemp()
  1012. self.addCleanup(shutil.rmtree, base_dir)
  1013. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  1014. f.write('FROM busybox\n')
  1015. service = self.create_service('build_isolation', build={
  1016. 'context': str(base_dir),
  1017. 'isolation': 'default',
  1018. })
  1019. service.build()
  1020. assert service.image()
  1021. def test_build_with_illegal_leading_chars(self):
  1022. base_dir = tempfile.mkdtemp()
  1023. self.addCleanup(shutil.rmtree, base_dir)
  1024. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  1025. f.write('FROM busybox\nRUN echo "Embodiment of Scarlet Devil"\n')
  1026. service = Service(
  1027. 'build_leading_slug', client=self.client,
  1028. project='___-composetest', build={
  1029. 'context': str(base_dir)
  1030. }
  1031. )
  1032. assert service.image_name == 'composetest_build_leading_slug'
  1033. service.build()
  1034. assert service.image()
  1035. def test_start_container_stays_unprivileged(self):
  1036. service = self.create_service('web')
  1037. container = create_and_start_container(service).inspect()
  1038. assert container['HostConfig']['Privileged'] is False
  1039. def test_start_container_becomes_privileged(self):
  1040. service = self.create_service('web', privileged=True)
  1041. container = create_and_start_container(service).inspect()
  1042. assert container['HostConfig']['Privileged'] is True
  1043. def test_expose_does_not_publish_ports(self):
  1044. service = self.create_service('web', expose=["8000"])
  1045. container = create_and_start_container(service).inspect()
  1046. assert container['NetworkSettings']['Ports'] == {'8000/tcp': None}
  1047. def test_start_container_creates_port_with_explicit_protocol(self):
  1048. service = self.create_service('web', ports=['8000/udp'])
  1049. container = create_and_start_container(service).inspect()
  1050. assert list(container['NetworkSettings']['Ports'].keys()) == ['8000/udp']
  1051. def test_start_container_creates_fixed_external_ports(self):
  1052. service = self.create_service('web', ports=['8000:8000'])
  1053. container = create_and_start_container(service).inspect()
  1054. assert '8000/tcp' in container['NetworkSettings']['Ports']
  1055. assert container['NetworkSettings']['Ports']['8000/tcp'][0]['HostPort'] == '8000'
  1056. def test_start_container_creates_fixed_external_ports_when_it_is_different_to_internal_port(self):
  1057. service = self.create_service('web', ports=['8001:8000'])
  1058. container = create_and_start_container(service).inspect()
  1059. assert '8000/tcp' in container['NetworkSettings']['Ports']
  1060. assert container['NetworkSettings']['Ports']['8000/tcp'][0]['HostPort'] == '8001'
  1061. def test_port_with_explicit_interface(self):
  1062. service = self.create_service('web', ports=[
  1063. '127.0.0.1:8001:8000',
  1064. '0.0.0.0:9001:9000/udp',
  1065. ])
  1066. container = create_and_start_container(service).inspect()
  1067. assert container['NetworkSettings']['Ports']['8000/tcp'] == [{
  1068. 'HostIp': '127.0.0.1',
  1069. 'HostPort': '8001',
  1070. }]
  1071. assert container['NetworkSettings']['Ports']['9000/udp'][0]['HostPort'] == '9001'
  1072. if not is_cluster(self.client):
  1073. assert container['NetworkSettings']['Ports']['9000/udp'][0]['HostIp'] == '0.0.0.0'
  1074. # self.assertEqual(container['NetworkSettings']['Ports'], {
  1075. # '8000/tcp': [
  1076. # {
  1077. # 'HostIp': '127.0.0.1',
  1078. # 'HostPort': '8001',
  1079. # },
  1080. # ],
  1081. # '9000/udp': [
  1082. # {
  1083. # 'HostIp': '0.0.0.0',
  1084. # 'HostPort': '9001',
  1085. # },
  1086. # ],
  1087. # })
  1088. def test_create_with_image_id(self):
  1089. pull_busybox(self.client)
  1090. image_id = self.client.inspect_image(BUSYBOX_IMAGE_WITH_TAG)['Id'][:12]
  1091. service = self.create_service('foo', image=image_id)
  1092. service.create_container()
  1093. def test_scale(self):
  1094. service = self.create_service('web')
  1095. service.scale(1)
  1096. assert len(service.containers()) == 1
  1097. # Ensure containers don't have stdout or stdin connected
  1098. container = service.containers()[0]
  1099. config = container.inspect()['Config']
  1100. assert not config['AttachStderr']
  1101. assert not config['AttachStdout']
  1102. assert not config['AttachStdin']
  1103. service.scale(3)
  1104. assert len(service.containers()) == 3
  1105. service.scale(1)
  1106. assert len(service.containers()) == 1
  1107. service.scale(0)
  1108. assert len(service.containers()) == 0
  1109. @pytest.mark.skipif(
  1110. SWARM_SKIP_CONTAINERS_ALL,
  1111. reason='Swarm /containers/json bug'
  1112. )
  1113. def test_scale_with_stopped_containers(self):
  1114. """
  1115. Given there are some stopped containers and scale is called with a
  1116. desired number that is the same as the number of stopped containers,
  1117. test that those containers are restarted and not removed/recreated.
  1118. """
  1119. service = self.create_service('web')
  1120. service.create_container(number=1)
  1121. service.create_container(number=2)
  1122. ParallelStreamWriter.instance = None
  1123. with mock.patch('sys.stderr', new_callable=StringIO) as mock_stderr:
  1124. service.scale(2)
  1125. for container in service.containers():
  1126. assert container.is_running
  1127. assert container.number in [1, 2]
  1128. captured_output = mock_stderr.getvalue()
  1129. assert 'Creating' not in captured_output
  1130. assert 'Starting' in captured_output
  1131. def test_scale_with_stopped_containers_and_needing_creation(self):
  1132. """
  1133. Given there are some stopped containers and scale is called with a
  1134. desired number that is greater than the number of stopped containers,
  1135. test that those containers are restarted and required number are created.
  1136. """
  1137. service = self.create_service('web')
  1138. next_number = service._next_container_number()
  1139. service.create_container(number=next_number, quiet=True)
  1140. for container in service.containers():
  1141. assert not container.is_running
  1142. ParallelStreamWriter.instance = None
  1143. with mock.patch('sys.stderr', new_callable=StringIO) as mock_stderr:
  1144. service.scale(2)
  1145. assert len(service.containers()) == 2
  1146. for container in service.containers():
  1147. assert container.is_running
  1148. captured_output = mock_stderr.getvalue()
  1149. assert 'Creating' in captured_output
  1150. assert 'Starting' in captured_output
  1151. def test_scale_with_api_error(self):
  1152. """Test that when scaling if the API returns an error, that error is handled
  1153. and the remaining threads continue.
  1154. """
  1155. service = self.create_service('web')
  1156. next_number = service._next_container_number()
  1157. service.create_container(number=next_number, quiet=True)
  1158. with mock.patch(
  1159. 'compose.container.Container.create',
  1160. side_effect=APIError(
  1161. message="testing",
  1162. response={},
  1163. explanation="Boom")):
  1164. with mock.patch('sys.stderr', new_callable=StringIO) as mock_stderr:
  1165. with pytest.raises(OperationFailedError):
  1166. service.scale(3)
  1167. assert len(service.containers()) == 1
  1168. assert service.containers()[0].is_running
  1169. assert "ERROR: for composetest_web_" in mock_stderr.getvalue()
  1170. assert "Cannot create container for service web: Boom" in mock_stderr.getvalue()
  1171. def test_scale_with_unexpected_exception(self):
  1172. """Test that when scaling if the API returns an error, that is not of type
  1173. APIError, that error is re-raised.
  1174. """
  1175. service = self.create_service('web')
  1176. next_number = service._next_container_number()
  1177. service.create_container(number=next_number, quiet=True)
  1178. with mock.patch(
  1179. 'compose.container.Container.create',
  1180. side_effect=ValueError("BOOM")
  1181. ):
  1182. with pytest.raises(ValueError):
  1183. service.scale(3)
  1184. assert len(service.containers()) == 1
  1185. assert service.containers()[0].is_running
  1186. @mock.patch('compose.service.log')
  1187. def test_scale_with_desired_number_already_achieved(self, mock_log):
  1188. """
  1189. Test that calling scale with a desired number that is equal to the
  1190. number of containers already running results in no change.
  1191. """
  1192. service = self.create_service('web')
  1193. next_number = service._next_container_number()
  1194. container = service.create_container(number=next_number, quiet=True)
  1195. container.start()
  1196. container.inspect()
  1197. assert container.is_running
  1198. assert len(service.containers()) == 1
  1199. service.scale(1)
  1200. assert len(service.containers()) == 1
  1201. container.inspect()
  1202. assert container.is_running
  1203. captured_output = mock_log.info.call_args[0]
  1204. assert 'Desired container number already achieved' in captured_output
  1205. @mock.patch('compose.service.log')
  1206. def test_scale_with_custom_container_name_outputs_warning(self, mock_log):
  1207. """Test that calling scale on a service that has a custom container name
  1208. results in warning output.
  1209. """
  1210. service = self.create_service('app', container_name='custom-container')
  1211. assert service.custom_container_name == 'custom-container'
  1212. with pytest.raises(OperationFailedError):
  1213. service.scale(3)
  1214. captured_output = mock_log.warning.call_args[0][0]
  1215. assert len(service.containers()) == 1
  1216. assert "Remove the custom name to scale the service." in captured_output
  1217. def test_scale_sets_ports(self):
  1218. service = self.create_service('web', ports=['8000'])
  1219. service.scale(2)
  1220. containers = service.containers()
  1221. assert len(containers) == 2
  1222. for container in containers:
  1223. assert list(container.get('HostConfig.PortBindings')) == ['8000/tcp']
  1224. def test_scale_with_immediate_exit(self):
  1225. service = self.create_service('web', image='busybox', command='true')
  1226. service.scale(2)
  1227. assert len(service.containers(stopped=True)) == 2
  1228. def test_network_mode_none(self):
  1229. service = self.create_service('web', network_mode=NetworkMode('none'))
  1230. container = create_and_start_container(service)
  1231. assert container.get('HostConfig.NetworkMode') == 'none'
  1232. def test_network_mode_bridged(self):
  1233. service = self.create_service('web', network_mode=NetworkMode('bridge'))
  1234. container = create_and_start_container(service)
  1235. assert container.get('HostConfig.NetworkMode') == 'bridge'
  1236. def test_network_mode_host(self):
  1237. service = self.create_service('web', network_mode=NetworkMode('host'))
  1238. container = create_and_start_container(service)
  1239. assert container.get('HostConfig.NetworkMode') == 'host'
  1240. def test_pid_mode_none_defined(self):
  1241. service = self.create_service('web', pid_mode=None)
  1242. container = create_and_start_container(service)
  1243. assert container.get('HostConfig.PidMode') == ''
  1244. def test_pid_mode_host(self):
  1245. service = self.create_service('web', pid_mode=PidMode('host'))
  1246. container = create_and_start_container(service)
  1247. assert container.get('HostConfig.PidMode') == 'host'
  1248. def test_ipc_mode_none_defined(self):
  1249. service = self.create_service('web', ipc_mode=None)
  1250. container = create_and_start_container(service)
  1251. print(container.get('HostConfig.IpcMode'))
  1252. assert container.get('HostConfig.IpcMode') == 'shareable'
  1253. def test_ipc_mode_host(self):
  1254. service = self.create_service('web', ipc_mode=IpcMode('host'))
  1255. container = create_and_start_container(service)
  1256. assert container.get('HostConfig.IpcMode') == 'host'
  1257. @v2_1_only()
  1258. def test_userns_mode_none_defined(self):
  1259. service = self.create_service('web', userns_mode=None)
  1260. container = create_and_start_container(service)
  1261. assert container.get('HostConfig.UsernsMode') == ''
  1262. @v2_1_only()
  1263. def test_userns_mode_host(self):
  1264. service = self.create_service('web', userns_mode='host')
  1265. container = create_and_start_container(service)
  1266. assert container.get('HostConfig.UsernsMode') == 'host'
  1267. def test_dns_no_value(self):
  1268. service = self.create_service('web')
  1269. container = create_and_start_container(service)
  1270. assert container.get('HostConfig.Dns') is None
  1271. def test_dns_list(self):
  1272. service = self.create_service('web', dns=['8.8.8.8', '9.9.9.9'])
  1273. container = create_and_start_container(service)
  1274. assert container.get('HostConfig.Dns') == ['8.8.8.8', '9.9.9.9']
  1275. def test_mem_swappiness(self):
  1276. service = self.create_service('web', mem_swappiness=11)
  1277. container = create_and_start_container(service)
  1278. assert container.get('HostConfig.MemorySwappiness') == 11
  1279. def test_mem_reservation(self):
  1280. service = self.create_service('web', mem_reservation='20m')
  1281. container = create_and_start_container(service)
  1282. assert container.get('HostConfig.MemoryReservation') == 20 * 1024 * 1024
  1283. def test_restart_always_value(self):
  1284. service = self.create_service('web', restart={'Name': 'always'})
  1285. container = create_and_start_container(service)
  1286. assert container.get('HostConfig.RestartPolicy.Name') == 'always'
  1287. def test_oom_score_adj_value(self):
  1288. service = self.create_service('web', oom_score_adj=500)
  1289. container = create_and_start_container(service)
  1290. assert container.get('HostConfig.OomScoreAdj') == 500
  1291. def test_group_add_value(self):
  1292. service = self.create_service('web', group_add=["root", "1"])
  1293. container = create_and_start_container(service)
  1294. host_container_groupadd = container.get('HostConfig.GroupAdd')
  1295. assert "root" in host_container_groupadd
  1296. assert "1" in host_container_groupadd
  1297. def test_dns_opt_value(self):
  1298. service = self.create_service('web', dns_opt=["use-vc", "no-tld-query"])
  1299. container = create_and_start_container(service)
  1300. dns_opt = container.get('HostConfig.DnsOptions')
  1301. assert 'use-vc' in dns_opt
  1302. assert 'no-tld-query' in dns_opt
  1303. def test_restart_on_failure_value(self):
  1304. service = self.create_service('web', restart={
  1305. 'Name': 'on-failure',
  1306. 'MaximumRetryCount': 5
  1307. })
  1308. container = create_and_start_container(service)
  1309. assert container.get('HostConfig.RestartPolicy.Name') == 'on-failure'
  1310. assert container.get('HostConfig.RestartPolicy.MaximumRetryCount') == 5
  1311. def test_cap_add_list(self):
  1312. service = self.create_service('web', cap_add=['SYS_ADMIN', 'NET_ADMIN'])
  1313. container = create_and_start_container(service)
  1314. assert container.get('HostConfig.CapAdd') == ['SYS_ADMIN', 'NET_ADMIN']
  1315. def test_cap_drop_list(self):
  1316. service = self.create_service('web', cap_drop=['SYS_ADMIN', 'NET_ADMIN'])
  1317. container = create_and_start_container(service)
  1318. assert container.get('HostConfig.CapDrop') == ['SYS_ADMIN', 'NET_ADMIN']
  1319. def test_dns_search(self):
  1320. service = self.create_service('web', dns_search=['dc1.example.com', 'dc2.example.com'])
  1321. container = create_and_start_container(service)
  1322. assert container.get('HostConfig.DnsSearch') == ['dc1.example.com', 'dc2.example.com']
  1323. @v2_only()
  1324. def test_tmpfs(self):
  1325. service = self.create_service('web', tmpfs=['/run'])
  1326. container = create_and_start_container(service)
  1327. assert container.get('HostConfig.Tmpfs') == {'/run': ''}
  1328. def test_working_dir_param(self):
  1329. service = self.create_service('container', working_dir='/working/dir/sample')
  1330. container = service.create_container()
  1331. assert container.get('Config.WorkingDir') == '/working/dir/sample'
  1332. def test_split_env(self):
  1333. service = self.create_service(
  1334. 'web',
  1335. environment=['NORMAL=F1', 'CONTAINS_EQUALS=F=2', 'TRAILING_EQUALS='])
  1336. env = create_and_start_container(service).environment
  1337. for k, v in {'NORMAL': 'F1', 'CONTAINS_EQUALS': 'F=2', 'TRAILING_EQUALS': ''}.items():
  1338. assert env[k] == v
  1339. def test_env_from_file_combined_with_env(self):
  1340. service = self.create_service(
  1341. 'web',
  1342. environment=['ONE=1', 'TWO=2', 'THREE=3'],
  1343. env_file=['tests/fixtures/env/one.env', 'tests/fixtures/env/two.env'])
  1344. env = create_and_start_container(service).environment
  1345. for k, v in {
  1346. 'ONE': '1',
  1347. 'TWO': '2',
  1348. 'THREE': '3',
  1349. 'FOO': 'baz',
  1350. 'DOO': 'dah'
  1351. }.items():
  1352. assert env[k] == v
  1353. @v3_only()
  1354. def test_build_with_cachefrom(self):
  1355. base_dir = tempfile.mkdtemp()
  1356. self.addCleanup(shutil.rmtree, base_dir)
  1357. with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
  1358. f.write("FROM busybox\n")
  1359. service = self.create_service('cache_from',
  1360. build={'context': base_dir,
  1361. 'cache_from': ['build1']})
  1362. service.build()
  1363. self.addCleanup(self.client.remove_image, service.image_name)
  1364. assert service.image()
  1365. @mock.patch.dict(os.environ)
  1366. def test_resolve_env(self):
  1367. os.environ['FILE_DEF'] = 'E1'
  1368. os.environ['FILE_DEF_EMPTY'] = 'E2'
  1369. os.environ['ENV_DEF'] = 'E3'
  1370. service = self.create_service(
  1371. 'web',
  1372. environment={
  1373. 'FILE_DEF': 'F1',
  1374. 'FILE_DEF_EMPTY': '',
  1375. 'ENV_DEF': None,
  1376. 'NO_DEF': None
  1377. }
  1378. )
  1379. env = create_and_start_container(service).environment
  1380. for k, v in {
  1381. 'FILE_DEF': 'F1',
  1382. 'FILE_DEF_EMPTY': '',
  1383. 'ENV_DEF': 'E3',
  1384. 'NO_DEF': None
  1385. }.items():
  1386. assert env[k] == v
  1387. def test_with_high_enough_api_version_we_get_default_network_mode(self):
  1388. # TODO: remove this test once minimum docker version is 1.8.x
  1389. with mock.patch.object(self.client, '_version', '1.20'):
  1390. service = self.create_service('web')
  1391. service_config = service._get_container_host_config({})
  1392. assert service_config['NetworkMode'] == 'default'
  1393. def test_labels(self):
  1394. labels_dict = {
  1395. 'com.example.description': "Accounting webapp",
  1396. 'com.example.department': "Finance",
  1397. 'com.example.label-with-empty-value': "",
  1398. }
  1399. compose_labels = {
  1400. LABEL_ONE_OFF: 'False',
  1401. LABEL_PROJECT: 'composetest',
  1402. LABEL_SERVICE: 'web',
  1403. LABEL_VERSION: __version__,
  1404. LABEL_CONTAINER_NUMBER: '1'
  1405. }
  1406. expected = dict(labels_dict, **compose_labels)
  1407. service = self.create_service('web', labels=labels_dict)
  1408. ctnr = create_and_start_container(service)
  1409. labels = ctnr.labels.items()
  1410. for pair in expected.items():
  1411. assert pair in labels
  1412. def test_empty_labels(self):
  1413. labels_dict = {'foo': '', 'bar': ''}
  1414. service = self.create_service('web', labels=labels_dict)
  1415. labels = create_and_start_container(service).labels.items()
  1416. for name in labels_dict:
  1417. assert (name, '') in labels
  1418. def test_stop_signal(self):
  1419. stop_signal = 'SIGINT'
  1420. service = self.create_service('web', stop_signal=stop_signal)
  1421. container = create_and_start_container(service)
  1422. assert container.stop_signal == stop_signal
  1423. def test_custom_container_name(self):
  1424. service = self.create_service('web', container_name='my-web-container')
  1425. assert service.custom_container_name == 'my-web-container'
  1426. container = create_and_start_container(service)
  1427. assert container.name == 'my-web-container'
  1428. one_off_container = service.create_container(one_off=True)
  1429. assert one_off_container.name != 'my-web-container'
  1430. @pytest.mark.skipif(True, reason="Broken on 1.11.0 - 17.03.0")
  1431. def test_log_drive_invalid(self):
  1432. service = self.create_service('web', logging={'driver': 'xxx'})
  1433. expected_error_msg = "logger: no log driver named 'xxx' is registered"
  1434. with pytest.raises(APIError) as excinfo:
  1435. create_and_start_container(service)
  1436. assert re.search(expected_error_msg, excinfo.value)
  1437. def test_log_drive_empty_default_jsonfile(self):
  1438. service = self.create_service('web')
  1439. log_config = create_and_start_container(service).log_config
  1440. assert 'json-file' == log_config['Type']
  1441. assert not log_config['Config']
  1442. def test_log_drive_none(self):
  1443. service = self.create_service('web', logging={'driver': 'none'})
  1444. log_config = create_and_start_container(service).log_config
  1445. assert 'none' == log_config['Type']
  1446. assert not log_config['Config']
  1447. def test_devices(self):
  1448. service = self.create_service('web', devices=["/dev/random:/dev/mapped-random"])
  1449. device_config = create_and_start_container(service).get('HostConfig.Devices')
  1450. device_dict = {
  1451. 'PathOnHost': '/dev/random',
  1452. 'CgroupPermissions': 'rwm',
  1453. 'PathInContainer': '/dev/mapped-random'
  1454. }
  1455. assert 1 == len(device_config)
  1456. assert device_dict == device_config[0]
  1457. def test_duplicate_containers(self):
  1458. service = self.create_service('web')
  1459. options = service._get_container_create_options({}, service._next_container_number())
  1460. original = Container.create(service.client, **options)
  1461. assert set(service.containers(stopped=True)) == set([original])
  1462. assert set(service.duplicate_containers()) == set()
  1463. options['name'] = 'temporary_container_name'
  1464. duplicate = Container.create(service.client, **options)
  1465. assert set(service.containers(stopped=True)) == set([original, duplicate])
  1466. assert set(service.duplicate_containers()) == set([duplicate])
  1467. def converge(service, strategy=ConvergenceStrategy.changed):
  1468. """Create a converge plan from a strategy and execute the plan."""
  1469. plan = service.convergence_plan(strategy)
  1470. return service.execute_convergence_plan(plan, timeout=1)
  1471. class ConfigHashTest(DockerClientTestCase):
  1472. def test_no_config_hash_when_one_off(self):
  1473. web = self.create_service('web')
  1474. container = web.create_container(one_off=True)
  1475. assert LABEL_CONFIG_HASH not in container.labels
  1476. def test_no_config_hash_when_overriding_options(self):
  1477. web = self.create_service('web')
  1478. container = web.create_container(environment={'FOO': '1'})
  1479. assert LABEL_CONFIG_HASH not in container.labels
  1480. def test_config_hash_with_custom_labels(self):
  1481. web = self.create_service('web', labels={'foo': '1'})
  1482. container = converge(web)[0]
  1483. assert LABEL_CONFIG_HASH in container.labels
  1484. assert 'foo' in container.labels
  1485. def test_config_hash_sticks_around(self):
  1486. web = self.create_service('web', command=["top"])
  1487. container = converge(web)[0]
  1488. assert LABEL_CONFIG_HASH in container.labels
  1489. web = self.create_service('web', command=["top", "-d", "1"])
  1490. container = converge(web)[0]
  1491. assert LABEL_CONFIG_HASH in container.labels