interpolation_test.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416
  1. # encoding: utf-8
  2. from __future__ import absolute_import
  3. from __future__ import unicode_literals
  4. import pytest
  5. from compose.config.environment import Environment
  6. from compose.config.errors import ConfigurationError
  7. from compose.config.interpolation import interpolate_environment_variables
  8. from compose.config.interpolation import Interpolator
  9. from compose.config.interpolation import InvalidInterpolation
  10. from compose.config.interpolation import TemplateWithDefaults
  11. from compose.config.interpolation import UnsetRequiredSubstitution
  12. from compose.const import COMPOSEFILE_V2_0 as V2_0
  13. from compose.const import COMPOSEFILE_V2_3 as V2_3
  14. from compose.const import COMPOSEFILE_V3_4 as V3_4
  15. @pytest.fixture
  16. def mock_env():
  17. return Environment({
  18. 'USER': 'jenny',
  19. 'FOO': 'bar',
  20. 'TRUE': 'True',
  21. 'FALSE': 'OFF',
  22. 'POSINT': '50',
  23. 'NEGINT': '-200',
  24. 'FLOAT': '0.145',
  25. 'MODE': '0600',
  26. })
  27. @pytest.fixture
  28. def variable_mapping():
  29. return Environment({'FOO': 'first', 'BAR': ''})
  30. @pytest.fixture
  31. def defaults_interpolator(variable_mapping):
  32. return Interpolator(TemplateWithDefaults, variable_mapping).interpolate
  33. def test_interpolate_environment_variables_in_services(mock_env):
  34. services = {
  35. 'servicea': {
  36. 'image': 'example:${USER}',
  37. 'volumes': ['$FOO:/target'],
  38. 'logging': {
  39. 'driver': '${FOO}',
  40. 'options': {
  41. 'user': '$USER',
  42. }
  43. }
  44. }
  45. }
  46. expected = {
  47. 'servicea': {
  48. 'image': 'example:jenny',
  49. 'volumes': ['bar:/target'],
  50. 'logging': {
  51. 'driver': 'bar',
  52. 'options': {
  53. 'user': 'jenny',
  54. }
  55. }
  56. }
  57. }
  58. value = interpolate_environment_variables(V2_0, services, 'service', mock_env)
  59. assert value == expected
  60. def test_interpolate_environment_variables_in_volumes(mock_env):
  61. volumes = {
  62. 'data': {
  63. 'driver': '$FOO',
  64. 'driver_opts': {
  65. 'max': 2,
  66. 'user': '${USER}'
  67. }
  68. },
  69. 'other': None,
  70. }
  71. expected = {
  72. 'data': {
  73. 'driver': 'bar',
  74. 'driver_opts': {
  75. 'max': 2,
  76. 'user': 'jenny'
  77. }
  78. },
  79. 'other': {},
  80. }
  81. value = interpolate_environment_variables(V2_0, volumes, 'volume', mock_env)
  82. assert value == expected
  83. def test_interpolate_environment_variables_in_secrets(mock_env):
  84. secrets = {
  85. 'secretservice': {
  86. 'file': '$FOO',
  87. 'labels': {
  88. 'max': 2,
  89. 'user': '${USER}'
  90. }
  91. },
  92. 'other': None,
  93. }
  94. expected = {
  95. 'secretservice': {
  96. 'file': 'bar',
  97. 'labels': {
  98. 'max': 2,
  99. 'user': 'jenny'
  100. }
  101. },
  102. 'other': {},
  103. }
  104. value = interpolate_environment_variables(V3_4, secrets, 'secret', mock_env)
  105. assert value == expected
  106. def test_interpolate_environment_services_convert_types_v2(mock_env):
  107. entry = {
  108. 'service1': {
  109. 'blkio_config': {
  110. 'weight': '${POSINT}',
  111. 'weight_device': [{'file': '/dev/sda1', 'weight': '${POSINT}'}]
  112. },
  113. 'cpus': '${FLOAT}',
  114. 'cpu_count': '$POSINT',
  115. 'healthcheck': {
  116. 'retries': '${POSINT:-3}',
  117. 'disable': '${FALSE}',
  118. 'command': 'true'
  119. },
  120. 'mem_swappiness': '${DEFAULT:-127}',
  121. 'oom_score_adj': '${NEGINT}',
  122. 'scale': '${POSINT}',
  123. 'ulimits': {
  124. 'nproc': '${POSINT}',
  125. 'nofile': {
  126. 'soft': '${POSINT}',
  127. 'hard': '${DEFAULT:-40000}'
  128. },
  129. },
  130. 'privileged': '${TRUE}',
  131. 'read_only': '${DEFAULT:-no}',
  132. 'tty': '${DEFAULT:-N}',
  133. 'stdin_open': '${DEFAULT-on}',
  134. }
  135. }
  136. expected = {
  137. 'service1': {
  138. 'blkio_config': {
  139. 'weight': 50,
  140. 'weight_device': [{'file': '/dev/sda1', 'weight': 50}]
  141. },
  142. 'cpus': 0.145,
  143. 'cpu_count': 50,
  144. 'healthcheck': {
  145. 'retries': 50,
  146. 'disable': False,
  147. 'command': 'true'
  148. },
  149. 'mem_swappiness': 127,
  150. 'oom_score_adj': -200,
  151. 'scale': 50,
  152. 'ulimits': {
  153. 'nproc': 50,
  154. 'nofile': {
  155. 'soft': 50,
  156. 'hard': 40000
  157. },
  158. },
  159. 'privileged': True,
  160. 'read_only': False,
  161. 'tty': False,
  162. 'stdin_open': True,
  163. }
  164. }
  165. value = interpolate_environment_variables(V2_3, entry, 'service', mock_env)
  166. assert value == expected
  167. def test_interpolate_environment_services_convert_types_v3(mock_env):
  168. entry = {
  169. 'service1': {
  170. 'healthcheck': {
  171. 'retries': '${POSINT:-3}',
  172. 'disable': '${FALSE}',
  173. 'command': 'true'
  174. },
  175. 'ulimits': {
  176. 'nproc': '${POSINT}',
  177. 'nofile': {
  178. 'soft': '${POSINT}',
  179. 'hard': '${DEFAULT:-40000}'
  180. },
  181. },
  182. 'privileged': '${TRUE}',
  183. 'read_only': '${DEFAULT:-no}',
  184. 'tty': '${DEFAULT:-N}',
  185. 'stdin_open': '${DEFAULT-on}',
  186. 'deploy': {
  187. 'update_config': {
  188. 'parallelism': '${DEFAULT:-2}',
  189. 'max_failure_ratio': '${FLOAT}',
  190. },
  191. 'restart_policy': {
  192. 'max_attempts': '$POSINT',
  193. },
  194. 'replicas': '${DEFAULT-3}'
  195. },
  196. 'ports': [{'target': '${POSINT}', 'published': '${DEFAULT:-5000}'}],
  197. 'configs': [{'mode': '${MODE}', 'source': 'config1'}],
  198. 'secrets': [{'mode': '${MODE}', 'source': 'secret1'}],
  199. }
  200. }
  201. expected = {
  202. 'service1': {
  203. 'healthcheck': {
  204. 'retries': 50,
  205. 'disable': False,
  206. 'command': 'true'
  207. },
  208. 'ulimits': {
  209. 'nproc': 50,
  210. 'nofile': {
  211. 'soft': 50,
  212. 'hard': 40000
  213. },
  214. },
  215. 'privileged': True,
  216. 'read_only': False,
  217. 'tty': False,
  218. 'stdin_open': True,
  219. 'deploy': {
  220. 'update_config': {
  221. 'parallelism': 2,
  222. 'max_failure_ratio': 0.145,
  223. },
  224. 'restart_policy': {
  225. 'max_attempts': 50,
  226. },
  227. 'replicas': 3
  228. },
  229. 'ports': [{'target': 50, 'published': 5000}],
  230. 'configs': [{'mode': 0o600, 'source': 'config1'}],
  231. 'secrets': [{'mode': 0o600, 'source': 'secret1'}],
  232. }
  233. }
  234. value = interpolate_environment_variables(V3_4, entry, 'service', mock_env)
  235. assert value == expected
  236. def test_interpolate_environment_services_convert_types_invalid(mock_env):
  237. entry = {'service1': {'privileged': '${POSINT}'}}
  238. with pytest.raises(ConfigurationError) as exc:
  239. interpolate_environment_variables(V2_3, entry, 'service', mock_env)
  240. assert 'Error while attempting to convert service.service1.privileged to '\
  241. 'appropriate type: "50" is not a valid boolean value' in exc.exconly()
  242. entry = {'service1': {'cpus': '${TRUE}'}}
  243. with pytest.raises(ConfigurationError) as exc:
  244. interpolate_environment_variables(V2_3, entry, 'service', mock_env)
  245. assert 'Error while attempting to convert service.service1.cpus to '\
  246. 'appropriate type: "True" is not a valid float' in exc.exconly()
  247. entry = {'service1': {'ulimits': {'nproc': '${FLOAT}'}}}
  248. with pytest.raises(ConfigurationError) as exc:
  249. interpolate_environment_variables(V2_3, entry, 'service', mock_env)
  250. assert 'Error while attempting to convert service.service1.ulimits.nproc to '\
  251. 'appropriate type: "0.145" is not a valid integer' in exc.exconly()
  252. def test_interpolate_environment_network_convert_types(mock_env):
  253. entry = {
  254. 'network1': {
  255. 'external': '${FALSE}',
  256. 'attachable': '${TRUE}',
  257. 'internal': '${DEFAULT:-false}'
  258. }
  259. }
  260. expected = {
  261. 'network1': {
  262. 'external': False,
  263. 'attachable': True,
  264. 'internal': False,
  265. }
  266. }
  267. value = interpolate_environment_variables(V3_4, entry, 'network', mock_env)
  268. assert value == expected
  269. def test_interpolate_environment_external_resource_convert_types(mock_env):
  270. entry = {
  271. 'resource1': {
  272. 'external': '${TRUE}',
  273. }
  274. }
  275. expected = {
  276. 'resource1': {
  277. 'external': True,
  278. }
  279. }
  280. value = interpolate_environment_variables(V3_4, entry, 'network', mock_env)
  281. assert value == expected
  282. value = interpolate_environment_variables(V3_4, entry, 'volume', mock_env)
  283. assert value == expected
  284. value = interpolate_environment_variables(V3_4, entry, 'secret', mock_env)
  285. assert value == expected
  286. value = interpolate_environment_variables(V3_4, entry, 'config', mock_env)
  287. assert value == expected
  288. def test_escaped_interpolation(defaults_interpolator):
  289. assert defaults_interpolator('$${foo}') == '${foo}'
  290. def test_invalid_interpolation(defaults_interpolator):
  291. with pytest.raises(InvalidInterpolation):
  292. defaults_interpolator('${')
  293. with pytest.raises(InvalidInterpolation):
  294. defaults_interpolator('$}')
  295. with pytest.raises(InvalidInterpolation):
  296. defaults_interpolator('${}')
  297. with pytest.raises(InvalidInterpolation):
  298. defaults_interpolator('${ }')
  299. with pytest.raises(InvalidInterpolation):
  300. defaults_interpolator('${ foo}')
  301. with pytest.raises(InvalidInterpolation):
  302. defaults_interpolator('${foo }')
  303. with pytest.raises(InvalidInterpolation):
  304. defaults_interpolator('${foo!}')
  305. def test_interpolate_missing_no_default(defaults_interpolator):
  306. assert defaults_interpolator("This ${missing} var") == "This var"
  307. assert defaults_interpolator("This ${BAR} var") == "This var"
  308. def test_interpolate_with_value(defaults_interpolator):
  309. assert defaults_interpolator("This $FOO var") == "This first var"
  310. assert defaults_interpolator("This ${FOO} var") == "This first var"
  311. def test_interpolate_missing_with_default(defaults_interpolator):
  312. assert defaults_interpolator("ok ${missing:-def}") == "ok def"
  313. assert defaults_interpolator("ok ${missing-def}") == "ok def"
  314. def test_interpolate_with_empty_and_default_value(defaults_interpolator):
  315. assert defaults_interpolator("ok ${BAR:-def}") == "ok def"
  316. assert defaults_interpolator("ok ${BAR-def}") == "ok "
  317. def test_interpolate_mandatory_values(defaults_interpolator):
  318. assert defaults_interpolator("ok ${FOO:?bar}") == "ok first"
  319. assert defaults_interpolator("ok ${FOO?bar}") == "ok first"
  320. assert defaults_interpolator("ok ${BAR?bar}") == "ok "
  321. with pytest.raises(UnsetRequiredSubstitution) as e:
  322. defaults_interpolator("not ok ${BAR:?high bar}")
  323. assert e.value.err == 'high bar'
  324. with pytest.raises(UnsetRequiredSubstitution) as e:
  325. defaults_interpolator("not ok ${BAZ?dropped the bazz}")
  326. assert e.value.err == 'dropped the bazz'
  327. def test_interpolate_mandatory_no_err_msg(defaults_interpolator):
  328. with pytest.raises(UnsetRequiredSubstitution) as e:
  329. defaults_interpolator("not ok ${BAZ?}")
  330. assert e.value.err == ''
  331. def test_interpolate_mixed_separators(defaults_interpolator):
  332. assert defaults_interpolator("ok ${BAR:-/non:-alphanumeric}") == "ok /non:-alphanumeric"
  333. assert defaults_interpolator("ok ${BAR:-:?wwegegr??:?}") == "ok :?wwegegr??:?"
  334. assert defaults_interpolator("ok ${BAR-:-hello}") == 'ok '
  335. with pytest.raises(UnsetRequiredSubstitution) as e:
  336. defaults_interpolator("not ok ${BAR:?xazz:-redf}")
  337. assert e.value.err == 'xazz:-redf'
  338. assert defaults_interpolator("ok ${BAR?...:?bar}") == "ok "
  339. def test_unbraced_separators(defaults_interpolator):
  340. assert defaults_interpolator("ok $FOO:-bar") == "ok first:-bar"
  341. assert defaults_interpolator("ok $BAZ?error") == "ok ?error"
  342. def test_interpolate_unicode_values():
  343. variable_mapping = {
  344. 'FOO': '十六夜 咲夜'.encode('utf-8'),
  345. 'BAR': '十六夜 咲夜'
  346. }
  347. interpol = Interpolator(TemplateWithDefaults, variable_mapping).interpolate
  348. interpol("$FOO") == '十六夜 咲夜'
  349. interpol("${BAR}") == '十六夜 咲夜'