mirror of https://github.com/docker/compose.git
Interpolate environment variables
Signed-off-by: Aanand Prasad <aanand.prasad@gmail.com>
This commit is contained in:
parent
31ac3ce22a
commit
8b5bd945d0
|
@ -8,6 +8,7 @@ import six
|
||||||
|
|
||||||
from compose.cli.utils import find_candidates_in_parent_dirs
|
from compose.cli.utils import find_candidates_in_parent_dirs
|
||||||
|
|
||||||
|
from .interpolation import interpolate_environment_variables
|
||||||
from .errors import (
|
from .errors import (
|
||||||
ConfigurationError,
|
ConfigurationError,
|
||||||
CircularReference,
|
CircularReference,
|
||||||
|
@ -132,11 +133,11 @@ def get_config_path(base_dir):
|
||||||
|
|
||||||
def load(config_details):
|
def load(config_details):
|
||||||
dictionary, working_dir, filename = config_details
|
dictionary, working_dir, filename = config_details
|
||||||
|
dictionary = interpolate_environment_variables(dictionary)
|
||||||
|
|
||||||
service_dicts = []
|
service_dicts = []
|
||||||
|
|
||||||
for service_name, service_dict in list(dictionary.items()):
|
for service_name, service_dict in list(dictionary.items()):
|
||||||
if not isinstance(service_dict, dict):
|
|
||||||
raise ConfigurationError('Service "%s" doesn\'t have any configuration options. All top level keys in your docker-compose.yml must map to a dictionary of configuration options.' % service_name)
|
|
||||||
loader = ServiceLoader(working_dir=working_dir, filename=filename)
|
loader = ServiceLoader(working_dir=working_dir, filename=filename)
|
||||||
service_dict = loader.make_service_dict(service_name, service_dict)
|
service_dict = loader.make_service_dict(service_name, service_dict)
|
||||||
validate_paths(service_dict)
|
validate_paths(service_dict)
|
||||||
|
@ -429,9 +430,9 @@ def resolve_volume_paths(volumes, working_dir=None):
|
||||||
|
|
||||||
def resolve_volume_path(volume, working_dir):
|
def resolve_volume_path(volume, working_dir):
|
||||||
container_path, host_path = split_path_mapping(volume)
|
container_path, host_path = split_path_mapping(volume)
|
||||||
container_path = os.path.expanduser(os.path.expandvars(container_path))
|
container_path = os.path.expanduser(container_path)
|
||||||
if host_path is not None:
|
if host_path is not None:
|
||||||
host_path = os.path.expanduser(os.path.expandvars(host_path))
|
host_path = os.path.expanduser(host_path)
|
||||||
return "%s:%s" % (expand_path(working_dir, host_path), container_path)
|
return "%s:%s" % (expand_path(working_dir, host_path), container_path)
|
||||||
else:
|
else:
|
||||||
return container_path
|
return container_path
|
||||||
|
|
|
@ -0,0 +1,69 @@
|
||||||
|
import os
|
||||||
|
from string import Template
|
||||||
|
from collections import defaultdict
|
||||||
|
|
||||||
|
import six
|
||||||
|
|
||||||
|
from .errors import ConfigurationError
|
||||||
|
|
||||||
|
|
||||||
|
def interpolate_environment_variables(config):
|
||||||
|
return dict(
|
||||||
|
(service_name, process_service(service_name, service_dict))
|
||||||
|
for (service_name, service_dict) in config.items()
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def process_service(service_name, service_dict):
|
||||||
|
if not isinstance(service_dict, dict):
|
||||||
|
raise ConfigurationError(
|
||||||
|
'Service "%s" doesn\'t have any configuration options. '
|
||||||
|
'All top level keys in your docker-compose.yml must map '
|
||||||
|
'to a dictionary of configuration options.' % service_name
|
||||||
|
)
|
||||||
|
|
||||||
|
return dict(
|
||||||
|
(key, interpolate_value(service_name, key, val))
|
||||||
|
for (key, val) in service_dict.items()
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def interpolate_value(service_name, config_key, value):
|
||||||
|
try:
|
||||||
|
return recursive_interpolate(value)
|
||||||
|
except InvalidInterpolation as e:
|
||||||
|
raise ConfigurationError(
|
||||||
|
'Invalid interpolation format for "{config_key}" option '
|
||||||
|
'in service "{service_name}": "{string}"'
|
||||||
|
.format(
|
||||||
|
config_key=config_key,
|
||||||
|
service_name=service_name,
|
||||||
|
string=e.string,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def recursive_interpolate(obj):
|
||||||
|
if isinstance(obj, six.string_types):
|
||||||
|
return interpolate(obj, os.environ)
|
||||||
|
elif isinstance(obj, dict):
|
||||||
|
return dict(
|
||||||
|
(key, recursive_interpolate(val))
|
||||||
|
for (key, val) in obj.items()
|
||||||
|
)
|
||||||
|
elif isinstance(obj, list):
|
||||||
|
return map(recursive_interpolate, obj)
|
||||||
|
else:
|
||||||
|
return obj
|
||||||
|
|
||||||
|
|
||||||
|
def interpolate(string, mapping):
|
||||||
|
try:
|
||||||
|
return Template(string).substitute(defaultdict(lambda: "", mapping))
|
||||||
|
except ValueError:
|
||||||
|
raise InvalidInterpolation(string)
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidInterpolation(Exception):
|
||||||
|
def __init__(self, string):
|
||||||
|
self.string = string
|
31
docs/yml.md
31
docs/yml.md
|
@ -19,6 +19,10 @@ As with `docker run`, options specified in the Dockerfile (e.g., `CMD`,
|
||||||
`EXPOSE`, `VOLUME`, `ENV`) are respected by default - you don't need to
|
`EXPOSE`, `VOLUME`, `ENV`) are respected by default - you don't need to
|
||||||
specify them again in `docker-compose.yml`.
|
specify them again in `docker-compose.yml`.
|
||||||
|
|
||||||
|
Values for configuration options can contain environment variables, e.g.
|
||||||
|
`image: postgres:${POSTGRES_VERSION}`. For more details, see the section on
|
||||||
|
[variable substitution](#variable-substitution).
|
||||||
|
|
||||||
### image
|
### image
|
||||||
|
|
||||||
Tag or partial image ID. Can be local or remote - Compose will attempt to
|
Tag or partial image ID. Can be local or remote - Compose will attempt to
|
||||||
|
@ -369,6 +373,33 @@ Each of these is a single value, analogous to its
|
||||||
volume_driver: mydriver
|
volume_driver: mydriver
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Variable substitution
|
||||||
|
|
||||||
|
Your configuration options can contain environment variables. Compose uses the
|
||||||
|
variable values from the shell environment in which `docker-compose` is run. For
|
||||||
|
example, suppose the shell contains `POSTGRES_VERSION=9.3` and you supply this
|
||||||
|
configuration:
|
||||||
|
|
||||||
|
db:
|
||||||
|
image: "postgres:${POSTGRES_VERSION}"
|
||||||
|
|
||||||
|
When you run `docker-compose up` with this configuration, Compose looks for the
|
||||||
|
`POSTGRES_VERSION` environment variable in the shell and substitutes its value
|
||||||
|
in. For this example, Compose resolves the `image` to `postgres:9.3` before
|
||||||
|
running the configuration.
|
||||||
|
|
||||||
|
If an environment variable is not set, Compose substitutes with an empty
|
||||||
|
string. In the example above, if `POSTGRES_VERSION` is not set, the value for
|
||||||
|
the `image` option is `postgres:`.
|
||||||
|
|
||||||
|
Both `$VARIABLE` and `${VARIABLE}` syntax are supported. Extended shell-style
|
||||||
|
features, such as `${VARIABLE-default}` and `${VARIABLE/foo/bar}`, are not
|
||||||
|
supported.
|
||||||
|
|
||||||
|
If you need to put a literal dollar sign in a configuration value, use a double
|
||||||
|
dollar sign (`$$`).
|
||||||
|
|
||||||
|
|
||||||
## Compose documentation
|
## Compose documentation
|
||||||
|
|
||||||
- [User guide](/)
|
- [User guide](/)
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
web:
|
||||||
|
# unbracketed name
|
||||||
|
image: $IMAGE
|
||||||
|
|
||||||
|
# array element
|
||||||
|
ports:
|
||||||
|
- "${HOST_PORT}:8000"
|
||||||
|
|
||||||
|
# dictionary item value
|
||||||
|
labels:
|
||||||
|
mylabel: "${LABEL_VALUE}"
|
||||||
|
|
||||||
|
# unset value
|
||||||
|
hostname: "host-${UNSET_VALUE}"
|
||||||
|
|
||||||
|
# escaped interpolation
|
||||||
|
command: "$${ESCAPED}"
|
|
@ -0,0 +1,5 @@
|
||||||
|
test:
|
||||||
|
image: busybox
|
||||||
|
command: top
|
||||||
|
volumes:
|
||||||
|
- "~/${VOLUME_NAME}:/container-path"
|
|
@ -488,6 +488,21 @@ class CLITestCase(DockerClientTestCase):
|
||||||
self.assertEqual(len(containers), 1)
|
self.assertEqual(len(containers), 1)
|
||||||
self.assertIn("FOO=1", containers[0].get('Config.Env'))
|
self.assertIn("FOO=1", containers[0].get('Config.Env'))
|
||||||
|
|
||||||
|
@patch.dict(os.environ)
|
||||||
|
def test_home_and_env_var_in_volume_path(self):
|
||||||
|
os.environ['VOLUME_NAME'] = 'my-volume'
|
||||||
|
os.environ['HOME'] = '/tmp/home-dir'
|
||||||
|
expected_host_path = os.path.join(os.environ['HOME'], os.environ['VOLUME_NAME'])
|
||||||
|
|
||||||
|
self.command.base_dir = 'tests/fixtures/volume-path-interpolation'
|
||||||
|
self.command.dispatch(['up', '-d'], None)
|
||||||
|
|
||||||
|
container = self.project.containers(stopped=True)[0]
|
||||||
|
actual_host_path = container.get('Volumes')['/container-path']
|
||||||
|
components = actual_host_path.split('/')
|
||||||
|
self.assertTrue(components[-2:] == ['home-dir', 'my-volume'],
|
||||||
|
msg="Last two components differ: %s, %s" % (actual_host_path, expected_host_path))
|
||||||
|
|
||||||
def test_up_with_extends(self):
|
def test_up_with_extends(self):
|
||||||
self.command.base_dir = 'tests/fixtures/extends'
|
self.command.base_dir = 'tests/fixtures/extends'
|
||||||
self.command.dispatch(['up', '-d'], None)
|
self.command.dispatch(['up', '-d'], None)
|
||||||
|
|
|
@ -273,24 +273,6 @@ class ServiceTest(DockerClientTestCase):
|
||||||
|
|
||||||
self.assertEqual(service.containers(stopped=False), [new_container])
|
self.assertEqual(service.containers(stopped=False), [new_container])
|
||||||
|
|
||||||
@patch.dict(os.environ)
|
|
||||||
def test_create_container_with_home_and_env_var_in_volume_path(self):
|
|
||||||
os.environ['VOLUME_NAME'] = 'my-volume'
|
|
||||||
os.environ['HOME'] = '/tmp/home-dir'
|
|
||||||
expected_host_path = os.path.join(os.environ['HOME'], os.environ['VOLUME_NAME'])
|
|
||||||
|
|
||||||
host_path = '~/${VOLUME_NAME}'
|
|
||||||
container_path = '/container-path'
|
|
||||||
|
|
||||||
service = self.create_service('db', volumes=['%s:%s' % (host_path, container_path)])
|
|
||||||
container = service.create_container()
|
|
||||||
service.start_container(container)
|
|
||||||
|
|
||||||
actual_host_path = container.get('Volumes')[container_path]
|
|
||||||
components = actual_host_path.split('/')
|
|
||||||
self.assertTrue(components[-2:] == ['home-dir', 'my-volume'],
|
|
||||||
msg="Last two components differ: %s, %s" % (actual_host_path, expected_host_path))
|
|
||||||
|
|
||||||
def test_create_container_with_volumes_from(self):
|
def test_create_container_with_volumes_from(self):
|
||||||
volume_service = self.create_service('data')
|
volume_service = self.create_service('data')
|
||||||
volume_container_1 = volume_service.create_container()
|
volume_container_1 = volume_service.create_container()
|
||||||
|
|
|
@ -59,11 +59,56 @@ class ConfigTest(unittest.TestCase):
|
||||||
make_service_dict('foo', {'ports': ['8000']}, 'tests/')
|
make_service_dict('foo', {'ports': ['8000']}, 'tests/')
|
||||||
|
|
||||||
|
|
||||||
class VolumePathTest(unittest.TestCase):
|
class InterpolationTest(unittest.TestCase):
|
||||||
@mock.patch.dict(os.environ)
|
@mock.patch.dict(os.environ)
|
||||||
def test_volume_binding_with_environ(self):
|
def test_config_file_with_environment_variable(self):
|
||||||
|
os.environ.update(
|
||||||
|
IMAGE="busybox",
|
||||||
|
HOST_PORT="80",
|
||||||
|
LABEL_VALUE="myvalue",
|
||||||
|
)
|
||||||
|
|
||||||
|
service_dicts = config.load(
|
||||||
|
config.find('tests/fixtures/environment-interpolation', None),
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(service_dicts, [
|
||||||
|
{
|
||||||
|
'name': 'web',
|
||||||
|
'image': 'busybox',
|
||||||
|
'ports': ['80:8000'],
|
||||||
|
'labels': {'mylabel': 'myvalue'},
|
||||||
|
'hostname': 'host-',
|
||||||
|
'command': '${ESCAPED}',
|
||||||
|
}
|
||||||
|
])
|
||||||
|
|
||||||
|
@mock.patch.dict(os.environ)
|
||||||
|
def test_invalid_interpolation(self):
|
||||||
|
with self.assertRaises(config.ConfigurationError) as cm:
|
||||||
|
config.load(
|
||||||
|
config.ConfigDetails(
|
||||||
|
{'web': {'image': '${'}},
|
||||||
|
'working_dir',
|
||||||
|
'filename.yml'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertIn('Invalid', cm.exception.msg)
|
||||||
|
self.assertIn('for "image" option', cm.exception.msg)
|
||||||
|
self.assertIn('in service "web"', cm.exception.msg)
|
||||||
|
self.assertIn('"${"', cm.exception.msg)
|
||||||
|
|
||||||
|
@mock.patch.dict(os.environ)
|
||||||
|
def test_volume_binding_with_environment_variable(self):
|
||||||
os.environ['VOLUME_PATH'] = '/host/path'
|
os.environ['VOLUME_PATH'] = '/host/path'
|
||||||
d = make_service_dict('foo', {'volumes': ['${VOLUME_PATH}:/container/path']}, working_dir='.')
|
d = config.load(
|
||||||
|
config.ConfigDetails(
|
||||||
|
config={'foo': {'volumes': ['${VOLUME_PATH}:/container/path']}},
|
||||||
|
working_dir='.',
|
||||||
|
filename=None,
|
||||||
|
)
|
||||||
|
)[0]
|
||||||
self.assertEqual(d['volumes'], ['/host/path:/container/path'])
|
self.assertEqual(d['volumes'], ['/host/path:/container/path'])
|
||||||
|
|
||||||
@mock.patch.dict(os.environ)
|
@mock.patch.dict(os.environ)
|
||||||
|
@ -400,18 +445,22 @@ class EnvTest(unittest.TestCase):
|
||||||
os.environ['HOSTENV'] = '/tmp'
|
os.environ['HOSTENV'] = '/tmp'
|
||||||
os.environ['CONTAINERENV'] = '/host/tmp'
|
os.environ['CONTAINERENV'] = '/host/tmp'
|
||||||
|
|
||||||
service_dict = make_service_dict(
|
service_dict = config.load(
|
||||||
'foo',
|
config.ConfigDetails(
|
||||||
{'volumes': ['$HOSTENV:$CONTAINERENV']},
|
config={'foo': {'volumes': ['$HOSTENV:$CONTAINERENV']}},
|
||||||
working_dir="tests/fixtures/env"
|
working_dir="tests/fixtures/env",
|
||||||
|
filename=None,
|
||||||
)
|
)
|
||||||
|
)[0]
|
||||||
self.assertEqual(set(service_dict['volumes']), set(['/tmp:/host/tmp']))
|
self.assertEqual(set(service_dict['volumes']), set(['/tmp:/host/tmp']))
|
||||||
|
|
||||||
service_dict = make_service_dict(
|
service_dict = config.load(
|
||||||
'foo',
|
config.ConfigDetails(
|
||||||
{'volumes': ['/opt${HOSTENV}:/opt${CONTAINERENV}']},
|
config={'foo': {'volumes': ['/opt${HOSTENV}:/opt${CONTAINERENV}']}},
|
||||||
working_dir="tests/fixtures/env"
|
working_dir="tests/fixtures/env",
|
||||||
|
filename=None,
|
||||||
)
|
)
|
||||||
|
)[0]
|
||||||
self.assertEqual(set(service_dict['volumes']), set(['/opt/tmp:/opt/host/tmp']))
|
self.assertEqual(set(service_dict['volumes']), set(['/opt/tmp:/opt/host/tmp']))
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
from compose.config.interpolation import interpolate, InvalidInterpolation
|
||||||
|
|
||||||
|
|
||||||
|
class InterpolationTest(unittest.TestCase):
|
||||||
|
def test_valid_interpolations(self):
|
||||||
|
self.assertEqual(interpolate('$foo', dict(foo='hi')), 'hi')
|
||||||
|
self.assertEqual(interpolate('${foo}', dict(foo='hi')), 'hi')
|
||||||
|
|
||||||
|
self.assertEqual(interpolate('${subject} love you', dict(subject='i')), 'i love you')
|
||||||
|
self.assertEqual(interpolate('i ${verb} you', dict(verb='love')), 'i love you')
|
||||||
|
self.assertEqual(interpolate('i love ${object}', dict(object='you')), 'i love you')
|
||||||
|
|
||||||
|
def test_empty_value(self):
|
||||||
|
self.assertEqual(interpolate('${foo}', dict(foo='')), '')
|
||||||
|
|
||||||
|
def test_unset_value(self):
|
||||||
|
self.assertEqual(interpolate('${foo}', dict()), '')
|
||||||
|
|
||||||
|
def test_escaped_interpolation(self):
|
||||||
|
self.assertEqual(interpolate('$${foo}', dict(foo='hi')), '${foo}')
|
||||||
|
|
||||||
|
def test_invalid_strings(self):
|
||||||
|
self.assertRaises(InvalidInterpolation, lambda: interpolate('${', dict()))
|
||||||
|
self.assertRaises(InvalidInterpolation, lambda: interpolate('$}', dict()))
|
||||||
|
self.assertRaises(InvalidInterpolation, lambda: interpolate('${}', dict()))
|
||||||
|
self.assertRaises(InvalidInterpolation, lambda: interpolate('${ }', dict()))
|
||||||
|
self.assertRaises(InvalidInterpolation, lambda: interpolate('${ foo}', dict()))
|
||||||
|
self.assertRaises(InvalidInterpolation, lambda: interpolate('${foo }', dict()))
|
||||||
|
self.assertRaises(InvalidInterpolation, lambda: interpolate('${foo!}', dict()))
|
Loading…
Reference in New Issue