interpolation_test.py 14 KB

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