service_test.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. from __future__ import unicode_literals
  2. from __future__ import absolute_import
  3. from fig import Service
  4. from fig.service import CannotBeScaledError
  5. from fig.packages.docker.errors import APIError
  6. from .testcases import DockerClientTestCase
  7. class ServiceTest(DockerClientTestCase):
  8. def test_containers(self):
  9. foo = self.create_service('foo')
  10. bar = self.create_service('bar')
  11. foo.start_container()
  12. self.assertEqual(len(foo.containers()), 1)
  13. self.assertEqual(foo.containers()[0].name, 'figtest_foo_1')
  14. self.assertEqual(len(bar.containers()), 0)
  15. bar.start_container()
  16. bar.start_container()
  17. self.assertEqual(len(foo.containers()), 1)
  18. self.assertEqual(len(bar.containers()), 2)
  19. names = [c.name for c in bar.containers()]
  20. self.assertIn('figtest_bar_1', names)
  21. self.assertIn('figtest_bar_2', names)
  22. def test_containers_one_off(self):
  23. db = self.create_service('db')
  24. container = db.create_container(one_off=True)
  25. self.assertEqual(db.containers(stopped=True), [])
  26. self.assertEqual(db.containers(one_off=True, stopped=True), [container])
  27. def test_project_is_added_to_container_name(self):
  28. service = self.create_service('web')
  29. service.start_container()
  30. self.assertEqual(service.containers()[0].name, 'figtest_web_1')
  31. def test_start_stop(self):
  32. service = self.create_service('scalingtest')
  33. self.assertEqual(len(service.containers(stopped=True)), 0)
  34. service.create_container()
  35. self.assertEqual(len(service.containers()), 0)
  36. self.assertEqual(len(service.containers(stopped=True)), 1)
  37. service.start()
  38. self.assertEqual(len(service.containers()), 1)
  39. self.assertEqual(len(service.containers(stopped=True)), 1)
  40. service.stop(timeout=1)
  41. self.assertEqual(len(service.containers()), 0)
  42. self.assertEqual(len(service.containers(stopped=True)), 1)
  43. service.stop(timeout=1)
  44. self.assertEqual(len(service.containers()), 0)
  45. self.assertEqual(len(service.containers(stopped=True)), 1)
  46. def test_kill_remove(self):
  47. service = self.create_service('scalingtest')
  48. service.start_container()
  49. self.assertEqual(len(service.containers()), 1)
  50. service.remove_stopped()
  51. self.assertEqual(len(service.containers()), 1)
  52. service.kill()
  53. self.assertEqual(len(service.containers()), 0)
  54. self.assertEqual(len(service.containers(stopped=True)), 1)
  55. service.remove_stopped()
  56. self.assertEqual(len(service.containers(stopped=True)), 0)
  57. def test_create_container_with_one_off(self):
  58. db = self.create_service('db')
  59. container = db.create_container(one_off=True)
  60. self.assertEqual(container.name, 'figtest_db_run_1')
  61. def test_create_container_with_one_off_when_existing_container_is_running(self):
  62. db = self.create_service('db')
  63. db.start()
  64. container = db.create_container(one_off=True)
  65. self.assertEqual(container.name, 'figtest_db_run_1')
  66. def test_create_container_with_unspecified_volume(self):
  67. service = self.create_service('db', volumes=['/var/db'])
  68. container = service.create_container()
  69. service.start_container(container)
  70. self.assertIn('/var/db', container.inspect()['Volumes'])
  71. def test_create_container_with_specified_volume(self):
  72. service = self.create_service('db', volumes=['/tmp:/host-tmp'])
  73. container = service.create_container()
  74. service.start_container(container)
  75. self.assertIn('/host-tmp', container.inspect()['Volumes'])
  76. def test_recreate_containers(self):
  77. service = self.create_service(
  78. 'db',
  79. environment={'FOO': '1'},
  80. volumes=['/var/db'],
  81. entrypoint=['ps'],
  82. command=['ax']
  83. )
  84. old_container = service.create_container()
  85. self.assertEqual(old_container.dictionary['Config']['Entrypoint'], ['ps'])
  86. self.assertEqual(old_container.dictionary['Config']['Cmd'], ['ax'])
  87. self.assertIn('FOO=1', old_container.dictionary['Config']['Env'])
  88. self.assertEqual(old_container.name, 'figtest_db_1')
  89. service.start_container(old_container)
  90. volume_path = old_container.inspect()['Volumes']['/var/db']
  91. num_containers_before = len(self.client.containers(all=True))
  92. service.options['environment']['FOO'] = '2'
  93. tuples = service.recreate_containers()
  94. self.assertEqual(len(tuples), 1)
  95. intermediate_container = tuples[0][0]
  96. new_container = tuples[0][1]
  97. self.assertEqual(intermediate_container.dictionary['Config']['Entrypoint'], ['echo'])
  98. self.assertEqual(new_container.dictionary['Config']['Entrypoint'], ['ps'])
  99. self.assertEqual(new_container.dictionary['Config']['Cmd'], ['ax'])
  100. self.assertIn('FOO=2', new_container.dictionary['Config']['Env'])
  101. self.assertEqual(new_container.name, 'figtest_db_1')
  102. self.assertEqual(new_container.inspect()['Volumes']['/var/db'], volume_path)
  103. self.assertEqual(len(self.client.containers(all=True)), num_containers_before)
  104. self.assertNotEqual(old_container.id, new_container.id)
  105. self.assertRaises(APIError, lambda: self.client.inspect_container(intermediate_container.id))
  106. def test_recreate_containers_with_keep_old_running(self):
  107. service = self.create_service(
  108. 'db',
  109. environment={'FOO': '1'},
  110. volumes=['/var/db'],
  111. entrypoint=['ps'],
  112. command=['ax']
  113. )
  114. old_container = service.create_container()
  115. service.start_container(old_container)
  116. num_containers_before = len(self.client.containers(all=True))
  117. tuples = service.recreate_containers(keep_old=True)
  118. self.assertEqual(len(tuples), 1)
  119. intermediate_container = tuples[0][0]
  120. new_container = tuples[0][1]
  121. self.assertIsNone(intermediate_container)
  122. self.assertEqual(len(self.client.containers(all=True)), num_containers_before)
  123. self.assertEqual(old_container.id, new_container.id)
  124. def test_recreate_containers_with_keep_old_stopped(self):
  125. service = self.create_service(
  126. 'db',
  127. environment={'FOO': '1'},
  128. volumes=['/var/db'],
  129. entrypoint=['ps'],
  130. command=['ax']
  131. )
  132. old_container = service.create_container()
  133. old_container.stop()
  134. num_containers_before = len(self.client.containers(all=True))
  135. tuples = service.recreate_containers(keep_old=True)
  136. self.assertEqual(len(tuples), 1)
  137. intermediate_container = tuples[0][0]
  138. new_container = tuples[0][1]
  139. self.assertIsNone(intermediate_container)
  140. self.assertEqual(len(self.client.containers(all=True)), num_containers_before)
  141. self.assertEqual(old_container.id, new_container.id)
  142. def test_start_container_passes_through_options(self):
  143. db = self.create_service('db')
  144. db.start_container(environment={'FOO': 'BAR'})
  145. self.assertEqual(db.containers()[0].environment['FOO'], 'BAR')
  146. def test_start_container_inherits_options_from_constructor(self):
  147. db = self.create_service('db', environment={'FOO': 'BAR'})
  148. db.start_container()
  149. self.assertEqual(db.containers()[0].environment['FOO'], 'BAR')
  150. def test_start_container_creates_links(self):
  151. db = self.create_service('db')
  152. web = self.create_service('web', links=[(db, None)])
  153. db.start_container()
  154. web.start_container()
  155. self.assertIn('figtest_db_1', web.containers()[0].links())
  156. self.assertIn('db_1', web.containers()[0].links())
  157. def test_start_container_creates_links_with_names(self):
  158. db = self.create_service('db')
  159. web = self.create_service('web', links=[(db, 'custom_link_name')])
  160. db.start_container()
  161. web.start_container()
  162. self.assertIn('custom_link_name', web.containers()[0].links())
  163. def test_start_normal_container_does_not_create_links_to_its_own_service(self):
  164. db = self.create_service('db')
  165. c1 = db.start_container()
  166. c2 = db.start_container()
  167. self.assertNotIn(c1.name, c2.links())
  168. def test_start_one_off_container_creates_links_to_its_own_service(self):
  169. db = self.create_service('db')
  170. c1 = db.start_container()
  171. c2 = db.start_container(one_off=True)
  172. self.assertIn(c1.name, c2.links())
  173. def test_start_container_builds_images(self):
  174. service = Service(
  175. name='test',
  176. client=self.client,
  177. build='tests/fixtures/simple-dockerfile',
  178. project='figtest',
  179. )
  180. container = service.start_container()
  181. container.wait()
  182. self.assertIn('success', container.logs())
  183. self.assertEqual(len(self.client.images(name='figtest_test')), 1)
  184. def test_start_container_uses_tagged_image_if_it_exists(self):
  185. self.client.build('tests/fixtures/simple-dockerfile', tag='figtest_test')
  186. service = Service(
  187. name='test',
  188. client=self.client,
  189. build='this/does/not/exist/and/will/throw/error',
  190. project='figtest',
  191. )
  192. container = service.start_container()
  193. container.wait()
  194. self.assertIn('success', container.logs())
  195. def test_start_container_creates_ports(self):
  196. service = self.create_service('web', ports=[8000])
  197. container = service.start_container().inspect()
  198. self.assertEqual(list(container['NetworkSettings']['Ports'].keys()), ['8000/tcp'])
  199. self.assertNotEqual(container['NetworkSettings']['Ports']['8000/tcp'][0]['HostPort'], '8000')
  200. def test_start_container_stays_unpriviliged(self):
  201. service = self.create_service('web')
  202. container = service.start_container().inspect()
  203. self.assertEqual(container['HostConfig']['Privileged'], False)
  204. def test_start_container_becomes_priviliged(self):
  205. service = self.create_service('web', privileged = True)
  206. container = service.start_container().inspect()
  207. self.assertEqual(container['HostConfig']['Privileged'], True)
  208. def test_expose_does_not_publish_ports(self):
  209. service = self.create_service('web', expose=[8000])
  210. container = service.start_container().inspect()
  211. self.assertEqual(container['NetworkSettings']['Ports'], {'8000/tcp': None})
  212. def test_start_container_creates_port_with_explicit_protocol(self):
  213. service = self.create_service('web', ports=['8000/udp'])
  214. container = service.start_container().inspect()
  215. self.assertEqual(list(container['NetworkSettings']['Ports'].keys()), ['8000/udp'])
  216. def test_start_container_creates_fixed_external_ports(self):
  217. service = self.create_service('web', ports=['8000:8000'])
  218. container = service.start_container().inspect()
  219. self.assertIn('8000/tcp', container['NetworkSettings']['Ports'])
  220. self.assertEqual(container['NetworkSettings']['Ports']['8000/tcp'][0]['HostPort'], '8000')
  221. def test_start_container_creates_fixed_external_ports_when_it_is_different_to_internal_port(self):
  222. service = self.create_service('web', ports=['8001:8000'])
  223. container = service.start_container().inspect()
  224. self.assertIn('8000/tcp', container['NetworkSettings']['Ports'])
  225. self.assertEqual(container['NetworkSettings']['Ports']['8000/tcp'][0]['HostPort'], '8001')
  226. def test_scale(self):
  227. service = self.create_service('web')
  228. service.scale(1)
  229. self.assertEqual(len(service.containers()), 1)
  230. service.scale(3)
  231. self.assertEqual(len(service.containers()), 3)
  232. service.scale(1)
  233. self.assertEqual(len(service.containers()), 1)
  234. service.scale(0)
  235. self.assertEqual(len(service.containers()), 0)
  236. def test_scale_on_service_that_cannot_be_scaled(self):
  237. service = self.create_service('web', ports=['8000:8000'])
  238. self.assertRaises(CannotBeScaledError, lambda: service.scale(1))
  239. def test_scale_sets_ports(self):
  240. service = self.create_service('web', ports=['8000'])
  241. service.scale(2)
  242. containers = service.containers()
  243. self.assertEqual(len(containers), 2)
  244. for container in containers:
  245. self.assertEqual(list(container.inspect()['HostConfig']['PortBindings'].keys()), ['8000/tcp'])