service_test.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  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_start_container_passes_through_options(self):
  107. db = self.create_service('db')
  108. db.start_container(environment={'FOO': 'BAR'})
  109. self.assertEqual(db.containers()[0].environment['FOO'], 'BAR')
  110. def test_start_container_inherits_options_from_constructor(self):
  111. db = self.create_service('db', environment={'FOO': 'BAR'})
  112. db.start_container()
  113. self.assertEqual(db.containers()[0].environment['FOO'], 'BAR')
  114. def test_start_container_creates_links(self):
  115. db = self.create_service('db')
  116. web = self.create_service('web', links=[(db, None)])
  117. db.start_container()
  118. web.start_container()
  119. self.assertIn('figtest_db_1', web.containers()[0].links())
  120. self.assertIn('db_1', web.containers()[0].links())
  121. def test_start_container_creates_links_with_names(self):
  122. db = self.create_service('db')
  123. web = self.create_service('web', links=[(db, 'custom_link_name')])
  124. db.start_container()
  125. web.start_container()
  126. self.assertIn('custom_link_name', web.containers()[0].links())
  127. def test_start_normal_container_does_not_create_links_to_its_own_service(self):
  128. db = self.create_service('db')
  129. c1 = db.start_container()
  130. c2 = db.start_container()
  131. self.assertNotIn(c1.name, c2.links())
  132. def test_start_one_off_container_creates_links_to_its_own_service(self):
  133. db = self.create_service('db')
  134. c1 = db.start_container()
  135. c2 = db.start_container(one_off=True)
  136. self.assertIn(c1.name, c2.links())
  137. def test_start_container_builds_images(self):
  138. service = Service(
  139. name='test',
  140. client=self.client,
  141. build='tests/fixtures/simple-dockerfile',
  142. project='figtest',
  143. )
  144. container = service.start_container()
  145. container.wait()
  146. self.assertIn('success', container.logs())
  147. self.assertEqual(len(self.client.images(name='figtest_test')), 1)
  148. def test_start_container_uses_tagged_image_if_it_exists(self):
  149. self.client.build('tests/fixtures/simple-dockerfile', tag='figtest_test')
  150. service = Service(
  151. name='test',
  152. client=self.client,
  153. build='this/does/not/exist/and/will/throw/error',
  154. project='figtest',
  155. )
  156. container = service.start_container()
  157. container.wait()
  158. self.assertIn('success', container.logs())
  159. def test_start_container_creates_ports(self):
  160. service = self.create_service('web', ports=[8000])
  161. container = service.start_container().inspect()
  162. self.assertEqual(list(container['NetworkSettings']['Ports'].keys()), ['8000/tcp'])
  163. self.assertNotEqual(container['NetworkSettings']['Ports']['8000/tcp'][0]['HostPort'], '8000')
  164. def test_start_container_stays_unpriviliged(self):
  165. service = self.create_service('web')
  166. container = service.start_container().inspect()
  167. self.assertEqual(container['HostConfig']['Privileged'], False)
  168. def test_start_container_becomes_priviliged(self):
  169. service = self.create_service('web', privileged = True)
  170. container = service.start_container().inspect()
  171. self.assertEqual(container['HostConfig']['Privileged'], True)
  172. def test_expose_does_not_publish_ports(self):
  173. service = self.create_service('web', expose=[8000])
  174. container = service.start_container().inspect()
  175. self.assertEqual(container['NetworkSettings']['Ports'], {'8000/tcp': None})
  176. def test_start_container_creates_port_with_explicit_protocol(self):
  177. service = self.create_service('web', ports=['8000/udp'])
  178. container = service.start_container().inspect()
  179. self.assertEqual(list(container['NetworkSettings']['Ports'].keys()), ['8000/udp'])
  180. def test_start_container_creates_fixed_external_ports(self):
  181. service = self.create_service('web', ports=['8000:8000'])
  182. container = service.start_container().inspect()
  183. self.assertIn('8000/tcp', container['NetworkSettings']['Ports'])
  184. self.assertEqual(container['NetworkSettings']['Ports']['8000/tcp'][0]['HostPort'], '8000')
  185. def test_start_container_creates_fixed_external_ports_when_it_is_different_to_internal_port(self):
  186. service = self.create_service('web', ports=['8001:8000'])
  187. container = service.start_container().inspect()
  188. self.assertIn('8000/tcp', container['NetworkSettings']['Ports'])
  189. self.assertEqual(container['NetworkSettings']['Ports']['8000/tcp'][0]['HostPort'], '8001')
  190. def test_scale(self):
  191. service = self.create_service('web')
  192. service.scale(1)
  193. self.assertEqual(len(service.containers()), 1)
  194. service.scale(3)
  195. self.assertEqual(len(service.containers()), 3)
  196. service.scale(1)
  197. self.assertEqual(len(service.containers()), 1)
  198. service.scale(0)
  199. self.assertEqual(len(service.containers()), 0)
  200. def test_scale_on_service_that_cannot_be_scaled(self):
  201. service = self.create_service('web', ports=['8000:8000'])
  202. self.assertRaises(CannotBeScaledError, lambda: service.scale(1))
  203. def test_scale_sets_ports(self):
  204. service = self.create_service('web', ports=['8000'])
  205. service.scale(2)
  206. containers = service.containers()
  207. self.assertEqual(len(containers), 2)
  208. for container in containers:
  209. self.assertEqual(list(container.inspect()['HostConfig']['PortBindings'].keys()), ['8000/tcp'])