service_test.py 11 KB

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