Interpolate environment variables

Signed-off-by: Aanand Prasad <aanand.prasad@gmail.com>
This commit is contained in:
Aanand Prasad 2015-07-24 15:58:18 +01:00
parent 31ac3ce22a
commit 8b5bd945d0
9 changed files with 235 additions and 35 deletions

View File

@ -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

View File

@ -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

View File

@ -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](/)

View File

@ -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}"

View File

@ -0,0 +1,5 @@
test:
image: busybox
command: top
volumes:
- "~/${VOLUME_NAME}:/container-path"

View File

@ -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)

View File

@ -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()

View File

@ -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']))

View File

@ -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()))