Implement environment singleton to be accessed throughout the code

Load and parse environment file from working dir

Signed-off-by: Joffrey F <joffrey@docker.com>
This commit is contained in:
Joffrey F 2016-03-01 15:49:42 -08:00
commit c69d8a3bd2
13 changed files with 151 additions and 64 deletions

View file

@ -15,7 +15,7 @@ from operator import attrgetter
import yaml
from docker import errors
from .. import mock
from ..helpers import clear_environment
from compose.cli.command import get_project
from compose.container import Container
from compose.project import OneOffFilter
@ -1452,7 +1452,7 @@ class CLITestCase(DockerClientTestCase):
self.assertEqual(len(containers), 1)
self.assertIn("FOO=1", containers[0].get('Config.Env'))
@mock.patch.dict(os.environ)
@clear_environment
def test_home_and_env_var_in_volume_path(self):
os.environ['VOLUME_NAME'] = 'my-volume'
os.environ['HOME'] = '/tmp/home-dir'

View file

@ -1,9 +1,14 @@
from __future__ import absolute_import
from __future__ import unicode_literals
import functools
import os
from . import mock
from compose.config.config import ConfigDetails
from compose.config.config import ConfigFile
from compose.config.config import load
from compose.config.environment import Environment
def build_config(contents, **kwargs):
@ -14,3 +19,11 @@ def build_config_details(contents, working_dir='working_dir', filename='filename
return ConfigDetails(
working_dir,
[ConfigFile(filename, contents)])
def clear_environment(f):
@functools.wraps(f)
def wrapper(self, *args, **kwargs):
Environment.reset()
with mock.patch.dict(os.environ):
f(self, *args, **kwargs)

View file

@ -12,6 +12,7 @@ from six import StringIO
from six import text_type
from .. import mock
from ..helpers import clear_environment
from .testcases import DockerClientTestCase
from .testcases import get_links
from .testcases import pull_busybox
@ -912,7 +913,7 @@ class ServiceTest(DockerClientTestCase):
}.items():
self.assertEqual(env[k], v)
@mock.patch.dict(os.environ)
@clear_environment
def test_resolve_env(self):
os.environ['FILE_DEF'] = 'E1'
os.environ['FILE_DEF_EMPTY'] = 'E2'

View file

@ -11,6 +11,7 @@ import pytest
from .. import mock
from .. import unittest
from ..helpers import build_config
from ..helpers import clear_environment
from compose.cli.command import get_project
from compose.cli.command import get_project_name
from compose.cli.docopt_command import NoSuchCommand
@ -43,11 +44,11 @@ class CLITestCase(unittest.TestCase):
project_name = get_project_name(None, project_name=name)
self.assertEquals('explicitprojectname', project_name)
@clear_environment
def test_project_name_from_environment_new_var(self):
name = 'namefromenv'
with mock.patch.dict(os.environ):
os.environ['COMPOSE_PROJECT_NAME'] = name
project_name = get_project_name(None)
os.environ['COMPOSE_PROJECT_NAME'] = name
project_name = get_project_name(None)
self.assertEquals(project_name, name)
def test_project_name_with_empty_environment_var(self):

View file

@ -23,6 +23,7 @@ from compose.config.types import VolumeSpec
from compose.const import IS_WINDOWS_PLATFORM
from tests import mock
from tests import unittest
from tests.helpers import clear_environment
DEFAULT_VERSION = V2_0
@ -1581,7 +1582,7 @@ class PortsTest(unittest.TestCase):
class InterpolationTest(unittest.TestCase):
@mock.patch.dict(os.environ)
@clear_environment
def test_config_file_with_environment_variable(self):
os.environ.update(
IMAGE="busybox",
@ -1604,7 +1605,7 @@ class InterpolationTest(unittest.TestCase):
}
])
@mock.patch.dict(os.environ)
@clear_environment
def test_unset_variable_produces_warning(self):
os.environ.pop('FOO', None)
os.environ.pop('BAR', None)
@ -1628,7 +1629,7 @@ class InterpolationTest(unittest.TestCase):
self.assertIn('BAR', warnings[0])
self.assertIn('FOO', warnings[1])
@mock.patch.dict(os.environ)
@clear_environment
def test_invalid_interpolation(self):
with self.assertRaises(config.ConfigurationError) as cm:
config.load(
@ -1667,7 +1668,7 @@ class VolumeConfigTest(unittest.TestCase):
d = make_service_dict('foo', {'build': '.', 'volumes': ['/data']}, working_dir='.')
self.assertEqual(d['volumes'], ['/data'])
@mock.patch.dict(os.environ)
@clear_environment
def test_volume_binding_with_environment_variable(self):
os.environ['VOLUME_PATH'] = '/host/path'
@ -1681,7 +1682,7 @@ class VolumeConfigTest(unittest.TestCase):
self.assertEqual(d['volumes'], [VolumeSpec.parse('/host/path:/container/path')])
@pytest.mark.skipif(IS_WINDOWS_PLATFORM, reason='posix paths')
@mock.patch.dict(os.environ)
@clear_environment
def test_volume_binding_with_home(self):
os.environ['HOME'] = '/home/user'
d = make_service_dict('foo', {'build': '.', 'volumes': ['~:/container/path']}, working_dir='.')
@ -1739,7 +1740,7 @@ class VolumeConfigTest(unittest.TestCase):
working_dir='c:\\Users\\me\\myproject')
self.assertEqual(d['volumes'], ['c:\\Users\\me\\otherproject:/data'])
@mock.patch.dict(os.environ)
@clear_environment
def test_home_directory_with_driver_does_not_expand(self):
os.environ['NAME'] = 'surprise!'
d = make_service_dict('foo', {
@ -2025,7 +2026,7 @@ class EnvTest(unittest.TestCase):
def test_parse_environment_empty(self):
self.assertEqual(config.parse_environment(None), {})
@mock.patch.dict(os.environ)
@clear_environment
def test_resolve_environment(self):
os.environ['FILE_DEF'] = 'E1'
os.environ['FILE_DEF_EMPTY'] = 'E2'
@ -2072,7 +2073,7 @@ class EnvTest(unittest.TestCase):
assert 'Couldn\'t find env file' in exc.exconly()
assert 'nonexistent.env' in exc.exconly()
@mock.patch.dict(os.environ)
@clear_environment
def test_resolve_environment_from_env_file_with_empty_values(self):
os.environ['FILE_DEF'] = 'E1'
os.environ['FILE_DEF_EMPTY'] = 'E2'
@ -2087,7 +2088,7 @@ class EnvTest(unittest.TestCase):
},
)
@mock.patch.dict(os.environ)
@clear_environment
def test_resolve_build_args(self):
os.environ['env_arg'] = 'value2'
@ -2106,7 +2107,7 @@ class EnvTest(unittest.TestCase):
)
@pytest.mark.xfail(IS_WINDOWS_PLATFORM, reason='paths use slash')
@mock.patch.dict(os.environ)
@clear_environment
def test_resolve_path(self):
os.environ['HOSTENV'] = '/tmp'
os.environ['CONTAINERENV'] = '/host/tmp'
@ -2393,7 +2394,7 @@ class ExtendsTest(unittest.TestCase):
assert 'net: container' in excinfo.exconly()
assert 'cannot be extended' in excinfo.exconly()
@mock.patch.dict(os.environ)
@clear_environment
def test_load_config_runs_interpolation_in_extended_service(self):
os.environ.update(HOSTNAME_VALUE="penguin")
expected_interpolated_value = "host-penguin"
@ -2465,6 +2466,7 @@ class ExtendsTest(unittest.TestCase):
},
]))
@clear_environment
def test_extends_with_environment_and_env_files(self):
tmpdir = py.test.ensuretemp('test_extends_with_environment')
self.addCleanup(tmpdir.remove)
@ -2520,12 +2522,12 @@ class ExtendsTest(unittest.TestCase):
},
},
]
with mock.patch.dict(os.environ):
os.environ['SECRET'] = 'secret'
os.environ['THING'] = 'thing'
os.environ['COMMON_ENV_FILE'] = 'secret'
os.environ['TOP_ENV_FILE'] = 'secret'
config = load_from_filename(str(tmpdir.join('docker-compose.yml')))
os.environ['SECRET'] = 'secret'
os.environ['THING'] = 'thing'
os.environ['COMMON_ENV_FILE'] = 'secret'
os.environ['TOP_ENV_FILE'] = 'secret'
config = load_from_filename(str(tmpdir.join('docker-compose.yml')))
assert config == expected

View file

@ -6,12 +6,14 @@ import os
import mock
import pytest
from compose.config.environment import Environment
from compose.config.interpolation import interpolate_environment_variables
@pytest.yield_fixture
def mock_env():
with mock.patch.dict(os.environ):
Environment.reset()
os.environ['USER'] = 'jenny'
os.environ['FOO'] = 'bar'
yield

View file

@ -3,7 +3,7 @@ from __future__ import unicode_literals
import unittest
from compose.config.interpolation import BlankDefaultDict as bddict
from compose.config.environment import BlankDefaultDict as bddict
from compose.config.interpolation import interpolate
from compose.config.interpolation import InvalidInterpolation