interpolation_test.py 14 KB

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