config_test.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506
  1. import os
  2. import mock
  3. from .. import unittest
  4. from compose import config
  5. class ConfigTest(unittest.TestCase):
  6. def test_from_dictionary(self):
  7. service_dicts = config.from_dictionary({
  8. 'foo': {'image': 'busybox'},
  9. 'bar': {'environment': ['FOO=1']},
  10. })
  11. self.assertEqual(
  12. sorted(service_dicts, key=lambda d: d['name']),
  13. sorted([
  14. {
  15. 'name': 'bar',
  16. 'environment': {'FOO': '1'},
  17. },
  18. {
  19. 'name': 'foo',
  20. 'image': 'busybox',
  21. }
  22. ])
  23. )
  24. def test_from_dictionary_throws_error_when_not_dict(self):
  25. with self.assertRaises(config.ConfigurationError):
  26. config.from_dictionary({
  27. 'web': 'busybox:latest',
  28. })
  29. def test_config_validation(self):
  30. self.assertRaises(
  31. config.ConfigurationError,
  32. lambda: config.make_service_dict('foo', {'port': ['8000']})
  33. )
  34. config.make_service_dict('foo', {'ports': ['8000']})
  35. class VolumePathTest(unittest.TestCase):
  36. @mock.patch.dict(os.environ)
  37. def test_volume_binding_with_environ(self):
  38. os.environ['VOLUME_PATH'] = '/host/path'
  39. d = config.make_service_dict('foo', {'volumes': ['${VOLUME_PATH}:/container/path']}, working_dir='.')
  40. self.assertEqual(d['volumes'], ['/host/path:/container/path'])
  41. @mock.patch.dict(os.environ)
  42. def test_volume_binding_with_home(self):
  43. os.environ['HOME'] = '/home/user'
  44. d = config.make_service_dict('foo', {'volumes': ['~:/container/path']}, working_dir='.')
  45. self.assertEqual(d['volumes'], ['/home/user:/container/path'])
  46. class MergePathMappingTest(object):
  47. def config_name(self):
  48. return ""
  49. def test_empty(self):
  50. service_dict = config.merge_service_dicts({}, {})
  51. self.assertNotIn(self.config_name(), service_dict)
  52. def test_no_override(self):
  53. service_dict = config.merge_service_dicts(
  54. {self.config_name(): ['/foo:/code', '/data']},
  55. {},
  56. )
  57. self.assertEqual(set(service_dict[self.config_name()]), set(['/foo:/code', '/data']))
  58. def test_no_base(self):
  59. service_dict = config.merge_service_dicts(
  60. {},
  61. {self.config_name(): ['/bar:/code']},
  62. )
  63. self.assertEqual(set(service_dict[self.config_name()]), set(['/bar:/code']))
  64. def test_override_explicit_path(self):
  65. service_dict = config.merge_service_dicts(
  66. {self.config_name(): ['/foo:/code', '/data']},
  67. {self.config_name(): ['/bar:/code']},
  68. )
  69. self.assertEqual(set(service_dict[self.config_name()]), set(['/bar:/code', '/data']))
  70. def test_add_explicit_path(self):
  71. service_dict = config.merge_service_dicts(
  72. {self.config_name(): ['/foo:/code', '/data']},
  73. {self.config_name(): ['/bar:/code', '/quux:/data']},
  74. )
  75. self.assertEqual(set(service_dict[self.config_name()]), set(['/bar:/code', '/quux:/data']))
  76. def test_remove_explicit_path(self):
  77. service_dict = config.merge_service_dicts(
  78. {self.config_name(): ['/foo:/code', '/quux:/data']},
  79. {self.config_name(): ['/bar:/code', '/data']},
  80. )
  81. self.assertEqual(set(service_dict[self.config_name()]), set(['/bar:/code', '/data']))
  82. class MergeVolumesTest(unittest.TestCase, MergePathMappingTest):
  83. def config_name(self):
  84. return 'volumes'
  85. class MergeDevicesTest(unittest.TestCase, MergePathMappingTest):
  86. def config_name(self):
  87. return 'devices'
  88. class BuildOrImageMergeTest(unittest.TestCase):
  89. def test_merge_build_or_image_no_override(self):
  90. self.assertEqual(
  91. config.merge_service_dicts({'build': '.'}, {}),
  92. {'build': '.'},
  93. )
  94. self.assertEqual(
  95. config.merge_service_dicts({'image': 'redis'}, {}),
  96. {'image': 'redis'},
  97. )
  98. def test_merge_build_or_image_override_with_same(self):
  99. self.assertEqual(
  100. config.merge_service_dicts({'build': '.'}, {'build': './web'}),
  101. {'build': './web'},
  102. )
  103. self.assertEqual(
  104. config.merge_service_dicts({'image': 'redis'}, {'image': 'postgres'}),
  105. {'image': 'postgres'},
  106. )
  107. def test_merge_build_or_image_override_with_other(self):
  108. self.assertEqual(
  109. config.merge_service_dicts({'build': '.'}, {'image': 'redis'}),
  110. {'image': 'redis'}
  111. )
  112. self.assertEqual(
  113. config.merge_service_dicts({'image': 'redis'}, {'build': '.'}),
  114. {'build': '.'}
  115. )
  116. class MergeListsTest(unittest.TestCase):
  117. def test_empty(self):
  118. service_dict = config.merge_service_dicts({}, {})
  119. self.assertNotIn('ports', service_dict)
  120. def test_no_override(self):
  121. service_dict = config.merge_service_dicts(
  122. {'ports': ['10:8000', '9000']},
  123. {},
  124. )
  125. self.assertEqual(set(service_dict['ports']), set(['10:8000', '9000']))
  126. def test_no_base(self):
  127. service_dict = config.merge_service_dicts(
  128. {},
  129. {'ports': ['10:8000', '9000']},
  130. )
  131. self.assertEqual(set(service_dict['ports']), set(['10:8000', '9000']))
  132. def test_add_item(self):
  133. service_dict = config.merge_service_dicts(
  134. {'ports': ['10:8000', '9000']},
  135. {'ports': ['20:8000']},
  136. )
  137. self.assertEqual(set(service_dict['ports']), set(['10:8000', '9000', '20:8000']))
  138. class MergeStringsOrListsTest(unittest.TestCase):
  139. def test_no_override(self):
  140. service_dict = config.merge_service_dicts(
  141. {'dns': '8.8.8.8'},
  142. {},
  143. )
  144. self.assertEqual(set(service_dict['dns']), set(['8.8.8.8']))
  145. def test_no_base(self):
  146. service_dict = config.merge_service_dicts(
  147. {},
  148. {'dns': '8.8.8.8'},
  149. )
  150. self.assertEqual(set(service_dict['dns']), set(['8.8.8.8']))
  151. def test_add_string(self):
  152. service_dict = config.merge_service_dicts(
  153. {'dns': ['8.8.8.8']},
  154. {'dns': '9.9.9.9'},
  155. )
  156. self.assertEqual(set(service_dict['dns']), set(['8.8.8.8', '9.9.9.9']))
  157. def test_add_list(self):
  158. service_dict = config.merge_service_dicts(
  159. {'dns': '8.8.8.8'},
  160. {'dns': ['9.9.9.9']},
  161. )
  162. self.assertEqual(set(service_dict['dns']), set(['8.8.8.8', '9.9.9.9']))
  163. class MergeLabelsTest(unittest.TestCase):
  164. def test_empty(self):
  165. service_dict = config.merge_service_dicts({}, {})
  166. self.assertNotIn('labels', service_dict)
  167. def test_no_override(self):
  168. service_dict = config.merge_service_dicts(
  169. config.make_service_dict('foo', {'labels': ['foo=1', 'bar']}),
  170. config.make_service_dict('foo', {}),
  171. )
  172. self.assertEqual(service_dict['labels'], {'foo': '1', 'bar': ''})
  173. def test_no_base(self):
  174. service_dict = config.merge_service_dicts(
  175. config.make_service_dict('foo', {}),
  176. config.make_service_dict('foo', {'labels': ['foo=2']}),
  177. )
  178. self.assertEqual(service_dict['labels'], {'foo': '2'})
  179. def test_override_explicit_value(self):
  180. service_dict = config.merge_service_dicts(
  181. config.make_service_dict('foo', {'labels': ['foo=1', 'bar']}),
  182. config.make_service_dict('foo', {'labels': ['foo=2']}),
  183. )
  184. self.assertEqual(service_dict['labels'], {'foo': '2', 'bar': ''})
  185. def test_add_explicit_value(self):
  186. service_dict = config.merge_service_dicts(
  187. config.make_service_dict('foo', {'labels': ['foo=1', 'bar']}),
  188. config.make_service_dict('foo', {'labels': ['bar=2']}),
  189. )
  190. self.assertEqual(service_dict['labels'], {'foo': '1', 'bar': '2'})
  191. def test_remove_explicit_value(self):
  192. service_dict = config.merge_service_dicts(
  193. config.make_service_dict('foo', {'labels': ['foo=1', 'bar=2']}),
  194. config.make_service_dict('foo', {'labels': ['bar']}),
  195. )
  196. self.assertEqual(service_dict['labels'], {'foo': '1', 'bar': ''})
  197. class EnvTest(unittest.TestCase):
  198. def test_parse_environment_as_list(self):
  199. environment = [
  200. 'NORMAL=F1',
  201. 'CONTAINS_EQUALS=F=2',
  202. 'TRAILING_EQUALS=',
  203. ]
  204. self.assertEqual(
  205. config.parse_environment(environment),
  206. {'NORMAL': 'F1', 'CONTAINS_EQUALS': 'F=2', 'TRAILING_EQUALS': ''},
  207. )
  208. def test_parse_environment_as_dict(self):
  209. environment = {
  210. 'NORMAL': 'F1',
  211. 'CONTAINS_EQUALS': 'F=2',
  212. 'TRAILING_EQUALS': None,
  213. }
  214. self.assertEqual(config.parse_environment(environment), environment)
  215. def test_parse_environment_invalid(self):
  216. with self.assertRaises(config.ConfigurationError):
  217. config.parse_environment('a=b')
  218. def test_parse_environment_empty(self):
  219. self.assertEqual(config.parse_environment(None), {})
  220. @mock.patch.dict(os.environ)
  221. def test_resolve_environment(self):
  222. os.environ['FILE_DEF'] = 'E1'
  223. os.environ['FILE_DEF_EMPTY'] = 'E2'
  224. os.environ['ENV_DEF'] = 'E3'
  225. service_dict = config.make_service_dict(
  226. 'foo', {
  227. 'environment': {
  228. 'FILE_DEF': 'F1',
  229. 'FILE_DEF_EMPTY': '',
  230. 'ENV_DEF': None,
  231. 'NO_DEF': None
  232. },
  233. },
  234. )
  235. self.assertEqual(
  236. service_dict['environment'],
  237. {'FILE_DEF': 'F1', 'FILE_DEF_EMPTY': '', 'ENV_DEF': 'E3', 'NO_DEF': ''},
  238. )
  239. def test_env_from_file(self):
  240. service_dict = config.make_service_dict(
  241. 'foo',
  242. {'env_file': 'one.env'},
  243. 'tests/fixtures/env',
  244. )
  245. self.assertEqual(
  246. service_dict['environment'],
  247. {'ONE': '2', 'TWO': '1', 'THREE': '3', 'FOO': 'bar'},
  248. )
  249. def test_env_from_multiple_files(self):
  250. service_dict = config.make_service_dict(
  251. 'foo',
  252. {'env_file': ['one.env', 'two.env']},
  253. 'tests/fixtures/env',
  254. )
  255. self.assertEqual(
  256. service_dict['environment'],
  257. {'ONE': '2', 'TWO': '1', 'THREE': '3', 'FOO': 'baz', 'DOO': 'dah'},
  258. )
  259. def test_env_nonexistent_file(self):
  260. options = {'env_file': 'nonexistent.env'}
  261. self.assertRaises(
  262. config.ConfigurationError,
  263. lambda: config.make_service_dict('foo', options, 'tests/fixtures/env'),
  264. )
  265. @mock.patch.dict(os.environ)
  266. def test_resolve_environment_from_file(self):
  267. os.environ['FILE_DEF'] = 'E1'
  268. os.environ['FILE_DEF_EMPTY'] = 'E2'
  269. os.environ['ENV_DEF'] = 'E3'
  270. service_dict = config.make_service_dict(
  271. 'foo',
  272. {'env_file': 'resolve.env'},
  273. 'tests/fixtures/env',
  274. )
  275. self.assertEqual(
  276. service_dict['environment'],
  277. {'FILE_DEF': 'F1', 'FILE_DEF_EMPTY': '', 'ENV_DEF': 'E3', 'NO_DEF': ''},
  278. )
  279. class ExtendsTest(unittest.TestCase):
  280. def test_extends(self):
  281. service_dicts = config.load('tests/fixtures/extends/docker-compose.yml')
  282. service_dicts = sorted(
  283. service_dicts,
  284. key=lambda sd: sd['name'],
  285. )
  286. self.assertEqual(service_dicts, [
  287. {
  288. 'name': 'mydb',
  289. 'image': 'busybox',
  290. 'command': 'top',
  291. },
  292. {
  293. 'name': 'myweb',
  294. 'image': 'busybox',
  295. 'command': 'top',
  296. 'links': ['mydb:db'],
  297. 'environment': {
  298. "FOO": "1",
  299. "BAR": "2",
  300. "BAZ": "2",
  301. },
  302. }
  303. ])
  304. def test_nested(self):
  305. service_dicts = config.load('tests/fixtures/extends/nested.yml')
  306. self.assertEqual(service_dicts, [
  307. {
  308. 'name': 'myweb',
  309. 'image': 'busybox',
  310. 'command': '/bin/true',
  311. 'environment': {
  312. "FOO": "2",
  313. "BAR": "2",
  314. },
  315. },
  316. ])
  317. def test_circular(self):
  318. try:
  319. config.load('tests/fixtures/extends/circle-1.yml')
  320. raise Exception("Expected config.CircularReference to be raised")
  321. except config.CircularReference as e:
  322. self.assertEqual(
  323. [(os.path.basename(filename), service_name) for (filename, service_name) in e.trail],
  324. [
  325. ('circle-1.yml', 'web'),
  326. ('circle-2.yml', 'web'),
  327. ('circle-1.yml', 'web'),
  328. ],
  329. )
  330. def test_extends_validation(self):
  331. dictionary = {'extends': None}
  332. def load_config():
  333. return config.make_service_dict('myweb', dictionary, working_dir='tests/fixtures/extends')
  334. self.assertRaisesRegexp(config.ConfigurationError, 'dictionary', load_config)
  335. dictionary['extends'] = {}
  336. self.assertRaises(config.ConfigurationError, load_config)
  337. dictionary['extends']['file'] = 'common.yml'
  338. self.assertRaisesRegexp(config.ConfigurationError, 'service', load_config)
  339. dictionary['extends']['service'] = 'web'
  340. self.assertIsInstance(load_config(), dict)
  341. dictionary['extends']['what'] = 'is this'
  342. self.assertRaisesRegexp(config.ConfigurationError, 'what', load_config)
  343. def test_blacklisted_options(self):
  344. def load_config():
  345. return config.make_service_dict('myweb', {
  346. 'extends': {
  347. 'file': 'whatever',
  348. 'service': 'web',
  349. }
  350. }, '.')
  351. with self.assertRaisesRegexp(config.ConfigurationError, 'links'):
  352. other_config = {'web': {'links': ['db']}}
  353. with mock.patch.object(config, 'load_yaml', return_value=other_config):
  354. print load_config()
  355. with self.assertRaisesRegexp(config.ConfigurationError, 'volumes_from'):
  356. other_config = {'web': {'volumes_from': ['db']}}
  357. with mock.patch.object(config, 'load_yaml', return_value=other_config):
  358. print load_config()
  359. with self.assertRaisesRegexp(config.ConfigurationError, 'net'):
  360. other_config = {'web': {'net': 'container:db'}}
  361. with mock.patch.object(config, 'load_yaml', return_value=other_config):
  362. print load_config()
  363. other_config = {'web': {'net': 'host'}}
  364. with mock.patch.object(config, 'load_yaml', return_value=other_config):
  365. print load_config()
  366. def test_volume_path(self):
  367. dicts = config.load('tests/fixtures/volume-path/docker-compose.yml')
  368. paths = [
  369. '%s:/foo' % os.path.abspath('tests/fixtures/volume-path/common/foo'),
  370. '%s:/bar' % os.path.abspath('tests/fixtures/volume-path/bar'),
  371. ]
  372. self.assertEqual(set(dicts[0]['volumes']), set(paths))
  373. def test_parent_build_path_dne(self):
  374. child = config.load('tests/fixtures/extends/nonexistent-path-child.yml')
  375. self.assertEqual(child, [
  376. {
  377. 'name': 'dnechild',
  378. 'image': 'busybox',
  379. 'command': '/bin/true',
  380. 'environment': {
  381. "FOO": "1",
  382. "BAR": "2",
  383. },
  384. },
  385. ])
  386. class BuildPathTest(unittest.TestCase):
  387. def setUp(self):
  388. self.abs_context_path = os.path.join(os.getcwd(), 'tests/fixtures/build-ctx')
  389. def test_nonexistent_path(self):
  390. options = {'build': 'nonexistent.path'}
  391. self.assertRaises(
  392. config.ConfigurationError,
  393. lambda: config.from_dictionary({
  394. 'foo': options,
  395. 'working_dir': 'tests/fixtures/build-path'
  396. })
  397. )
  398. def test_relative_path(self):
  399. relative_build_path = '../build-ctx/'
  400. service_dict = config.make_service_dict(
  401. 'relpath',
  402. {'build': relative_build_path},
  403. working_dir='tests/fixtures/build-path'
  404. )
  405. self.assertEquals(service_dict['build'], self.abs_context_path)
  406. def test_absolute_path(self):
  407. service_dict = config.make_service_dict(
  408. 'abspath',
  409. {'build': self.abs_context_path},
  410. working_dir='tests/fixtures/build-path'
  411. )
  412. self.assertEquals(service_dict['build'], self.abs_context_path)
  413. def test_from_file(self):
  414. service_dict = config.load('tests/fixtures/build-path/docker-compose.yml')
  415. self.assertEquals(service_dict, [{'name': 'foo', 'build': self.abs_context_path}])