|  | @@ -7,14 +7,35 @@ from string import Template
 | 
	
		
			
				|  |  |  import six
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  from .errors import ConfigurationError
 | 
	
		
			
				|  |  | +from compose.const import COMPOSEFILE_V1 as V1
 | 
	
		
			
				|  |  | +from compose.const import COMPOSEFILE_V2_0 as V2_0
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  log = logging.getLogger(__name__)
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -def interpolate_environment_variables(config, section, environment):
 | 
	
		
			
				|  |  | +class Interpolator(object):
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    def __init__(self, templater, mapping):
 | 
	
		
			
				|  |  | +        self.templater = templater
 | 
	
		
			
				|  |  | +        self.mapping = mapping
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    def interpolate(self, string):
 | 
	
		
			
				|  |  | +        try:
 | 
	
		
			
				|  |  | +            return self.templater(string).substitute(self.mapping)
 | 
	
		
			
				|  |  | +        except ValueError:
 | 
	
		
			
				|  |  | +            raise InvalidInterpolation(string)
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +def interpolate_environment_variables(version, config, section, environment):
 | 
	
		
			
				|  |  | +    if version in (V2_0, V1):
 | 
	
		
			
				|  |  | +        interpolator = Interpolator(Template, environment)
 | 
	
		
			
				|  |  | +    else:
 | 
	
		
			
				|  |  | +        interpolator = Interpolator(TemplateWithDefaults, environment)
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      def process_item(name, config_dict):
 | 
	
		
			
				|  |  |          return dict(
 | 
	
		
			
				|  |  | -            (key, interpolate_value(name, key, val, section, environment))
 | 
	
		
			
				|  |  | +            (key, interpolate_value(name, key, val, section, interpolator))
 | 
	
		
			
				|  |  |              for key, val in (config_dict or {}).items()
 | 
	
		
			
				|  |  |          )
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -24,9 +45,9 @@ def interpolate_environment_variables(config, section, environment):
 | 
	
		
			
				|  |  |      )
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -def interpolate_value(name, config_key, value, section, mapping):
 | 
	
		
			
				|  |  | +def interpolate_value(name, config_key, value, section, interpolator):
 | 
	
		
			
				|  |  |      try:
 | 
	
		
			
				|  |  | -        return recursive_interpolate(value, mapping)
 | 
	
		
			
				|  |  | +        return recursive_interpolate(value, interpolator)
 | 
	
		
			
				|  |  |      except InvalidInterpolation as e:
 | 
	
		
			
				|  |  |          raise ConfigurationError(
 | 
	
		
			
				|  |  |              'Invalid interpolation format for "{config_key}" option '
 | 
	
	
		
			
				|  | @@ -37,25 +58,44 @@ def interpolate_value(name, config_key, value, section, mapping):
 | 
	
		
			
				|  |  |                  string=e.string))
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -def recursive_interpolate(obj, mapping):
 | 
	
		
			
				|  |  | +def recursive_interpolate(obj, interpolator):
 | 
	
		
			
				|  |  |      if isinstance(obj, six.string_types):
 | 
	
		
			
				|  |  | -        return interpolate(obj, mapping)
 | 
	
		
			
				|  |  | -    elif isinstance(obj, dict):
 | 
	
		
			
				|  |  | +        return interpolator.interpolate(obj)
 | 
	
		
			
				|  |  | +    if isinstance(obj, dict):
 | 
	
		
			
				|  |  |          return dict(
 | 
	
		
			
				|  |  | -            (key, recursive_interpolate(val, mapping))
 | 
	
		
			
				|  |  | +            (key, recursive_interpolate(val, interpolator))
 | 
	
		
			
				|  |  |              for (key, val) in obj.items()
 | 
	
		
			
				|  |  |          )
 | 
	
		
			
				|  |  | -    elif isinstance(obj, list):
 | 
	
		
			
				|  |  | -        return [recursive_interpolate(val, mapping) for val in obj]
 | 
	
		
			
				|  |  | -    else:
 | 
	
		
			
				|  |  | -        return obj
 | 
	
		
			
				|  |  | +    if isinstance(obj, list):
 | 
	
		
			
				|  |  | +        return [recursive_interpolate(val, interpolator) for val in obj]
 | 
	
		
			
				|  |  | +    return obj
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -def interpolate(string, mapping):
 | 
	
		
			
				|  |  | -    try:
 | 
	
		
			
				|  |  | -        return Template(string).substitute(mapping)
 | 
	
		
			
				|  |  | -    except ValueError:
 | 
	
		
			
				|  |  | -        raise InvalidInterpolation(string)
 | 
	
		
			
				|  |  | +class TemplateWithDefaults(Template):
 | 
	
		
			
				|  |  | +    idpattern = r'[_a-z][_a-z0-9]*(?::?-[_a-z0-9]+)?'
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    # Modified from python2.7/string.py
 | 
	
		
			
				|  |  | +    def substitute(self, mapping):
 | 
	
		
			
				|  |  | +        # Helper function for .sub()
 | 
	
		
			
				|  |  | +        def convert(mo):
 | 
	
		
			
				|  |  | +            # Check the most common path first.
 | 
	
		
			
				|  |  | +            named = mo.group('named') or mo.group('braced')
 | 
	
		
			
				|  |  | +            if named is not None:
 | 
	
		
			
				|  |  | +                if ':-' in named:
 | 
	
		
			
				|  |  | +                    var, _, default = named.partition(':-')
 | 
	
		
			
				|  |  | +                    return mapping.get(var) or default
 | 
	
		
			
				|  |  | +                if '-' in named:
 | 
	
		
			
				|  |  | +                    var, _, default = named.partition('-')
 | 
	
		
			
				|  |  | +                    return mapping.get(var, default)
 | 
	
		
			
				|  |  | +                val = mapping[named]
 | 
	
		
			
				|  |  | +                return '%s' % (val,)
 | 
	
		
			
				|  |  | +            if mo.group('escaped') is not None:
 | 
	
		
			
				|  |  | +                return self.delimiter
 | 
	
		
			
				|  |  | +            if mo.group('invalid') is not None:
 | 
	
		
			
				|  |  | +                self._invalid(mo)
 | 
	
		
			
				|  |  | +            raise ValueError('Unrecognized named group in pattern',
 | 
	
		
			
				|  |  | +                             self.pattern)
 | 
	
		
			
				|  |  | +        return self.pattern.sub(convert, self.template)
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  class InvalidInterpolation(Exception):
 |