| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927 | # encoding: utf-8from __future__ import absolute_importfrom __future__ import unicode_literalsimport datetimeimport osimport tempfileimport dockerimport pytestfrom docker.errors import NotFoundfrom .. import mockfrom .. import unittestfrom ..helpers import BUSYBOX_IMAGE_WITH_TAGfrom compose.config import ConfigurationErrorfrom compose.config.config import Configfrom compose.config.types import VolumeFromSpecfrom compose.const import COMPOSEFILE_V1 as V1from compose.const import COMPOSEFILE_V2_0 as V2_0from compose.const import COMPOSEFILE_V2_4 as V2_4from compose.const import COMPOSEFILE_V3_7 as V3_7from compose.const import DEFAULT_TIMEOUTfrom compose.const import LABEL_SERVICEfrom compose.container import Containerfrom compose.errors import OperationFailedErrorfrom compose.project import get_secretsfrom compose.project import NoSuchServicefrom compose.project import Projectfrom compose.project import ProjectErrorfrom compose.service import ImageTypefrom compose.service import Serviceclass ProjectTest(unittest.TestCase):    def setUp(self):        self.mock_client = mock.create_autospec(docker.APIClient)        self.mock_client._general_configs = {}        self.mock_client.api_version = docker.constants.DEFAULT_DOCKER_API_VERSION    def test_from_config_v1(self):        config = Config(            version=V1,            services=[                {                    'name': 'web',                    'image': BUSYBOX_IMAGE_WITH_TAG,                },                {                    'name': 'db',                    'image': BUSYBOX_IMAGE_WITH_TAG,                },            ],            networks=None,            volumes=None,            secrets=None,            configs=None,        )        project = Project.from_config(            name='composetest',            config_data=config,            client=None,        )        assert len(project.services) == 2        assert project.get_service('web').name == 'web'        assert project.get_service('web').options['image'] == BUSYBOX_IMAGE_WITH_TAG        assert project.get_service('db').name == 'db'        assert project.get_service('db').options['image'] == BUSYBOX_IMAGE_WITH_TAG        assert not project.networks.use_networking    @mock.patch('compose.network.Network.true_name', lambda n: n.full_name)    def test_from_config_v2(self):        config = Config(            version=V2_0,            services=[                {                    'name': 'web',                    'image': BUSYBOX_IMAGE_WITH_TAG,                },                {                    'name': 'db',                    'image': BUSYBOX_IMAGE_WITH_TAG,                },            ],            networks=None,            volumes=None,            secrets=None,            configs=None,        )        project = Project.from_config('composetest', config, None)        assert len(project.services) == 2        assert project.networks.use_networking    def test_get_service(self):        web = Service(            project='composetest',            name='web',            client=None,            image=BUSYBOX_IMAGE_WITH_TAG,        )        project = Project('test', [web], None)        assert project.get_service('web') == web    def test_get_services_returns_all_services_without_args(self):        web = Service(            project='composetest',            name='web',            image='foo',        )        console = Service(            project='composetest',            name='console',            image='foo',        )        project = Project('test', [web, console], None)        assert project.get_services() == [web, console]    def test_get_services_returns_listed_services_with_args(self):        web = Service(            project='composetest',            name='web',            image='foo',        )        console = Service(            project='composetest',            name='console',            image='foo',        )        project = Project('test', [web, console], None)        assert project.get_services(['console']) == [console]    def test_get_services_with_include_links(self):        db = Service(            project='composetest',            name='db',            image='foo',        )        web = Service(            project='composetest',            name='web',            image='foo',            links=[(db, 'database')]        )        cache = Service(            project='composetest',            name='cache',            image='foo'        )        console = Service(            project='composetest',            name='console',            image='foo',            links=[(web, 'web')]        )        project = Project('test', [web, db, cache, console], None)        assert project.get_services(['console'], include_deps=True) == [db, web, console]    def test_get_services_removes_duplicates_following_links(self):        db = Service(            project='composetest',            name='db',            image='foo',        )        web = Service(            project='composetest',            name='web',            image='foo',            links=[(db, 'database')]        )        project = Project('test', [web, db], None)        assert project.get_services(['web', 'db'], include_deps=True) == [db, web]    def test_use_volumes_from_container(self):        container_id = 'aabbccddee'        container_dict = dict(Name='aaa', Id=container_id)        self.mock_client.inspect_container.return_value = container_dict        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V2_0,                services=[{                    'name': 'test',                    'image': BUSYBOX_IMAGE_WITH_TAG,                    'volumes_from': [VolumeFromSpec('aaa', 'rw', 'container')]                }],                networks=None,                volumes=None,                secrets=None,                configs=None,            ),        )        assert project.get_service('test')._get_volumes_from() == [container_id + ":rw"]    def test_use_volumes_from_service_no_container(self):        container_name = 'test_vol_1'        self.mock_client.containers.return_value = [            {                "Name": container_name,                "Names": [container_name],                "Id": container_name,                "Image": BUSYBOX_IMAGE_WITH_TAG            }        ]        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V2_0,                services=[                    {                        'name': 'vol',                        'image': BUSYBOX_IMAGE_WITH_TAG                    },                    {                        'name': 'test',                        'image': BUSYBOX_IMAGE_WITH_TAG,                        'volumes_from': [VolumeFromSpec('vol', 'rw', 'service')]                    }                ],                networks=None,                volumes=None,                secrets=None,                configs=None,            ),        )        assert project.get_service('test')._get_volumes_from() == [container_name + ":rw"]    @mock.patch('compose.network.Network.true_name', lambda n: n.full_name)    def test_use_volumes_from_service_container(self):        container_ids = ['aabbccddee', '12345']        project = Project.from_config(            name='test',            client=None,            config_data=Config(                version=V2_0,                services=[                    {                        'name': 'vol',                        'image': BUSYBOX_IMAGE_WITH_TAG                    },                    {                        'name': 'test',                        'image': BUSYBOX_IMAGE_WITH_TAG,                        'volumes_from': [VolumeFromSpec('vol', 'rw', 'service')]                    }                ],                networks=None,                volumes=None,                secrets=None,                configs=None,            ),        )        with mock.patch.object(Service, 'containers') as mock_return:            mock_return.return_value = [                mock.Mock(id=container_id, spec=Container)                for container_id in container_ids]            assert (                project.get_service('test')._get_volumes_from() ==                [container_ids[0] + ':rw']            )    def test_events_legacy(self):        services = [Service(name='web'), Service(name='db')]        project = Project('test', services, self.mock_client)        self.mock_client.api_version = '1.21'        self.mock_client.events.return_value = iter([            {                'status': 'create',                'from': 'example/image',                'id': 'abcde',                'time': 1420092061,                'timeNano': 14200920610000002000,            },            {                'status': 'attach',                'from': 'example/image',                'id': 'abcde',                'time': 1420092061,                'timeNano': 14200920610000003000,            },            {                'status': 'create',                'from': 'example/other',                'id': 'bdbdbd',                'time': 1420092061,                'timeNano': 14200920610000005000,            },            {                'status': 'create',                'from': 'example/db',                'id': 'ababa',                'time': 1420092061,                'timeNano': 14200920610000004000,            },            {                'status': 'destroy',                'from': 'example/db',                'id': 'eeeee',                'time': 1420092061,                'timeNano': 14200920610000004000,            },        ])        def dt_with_microseconds(dt, us):            return datetime.datetime.fromtimestamp(dt).replace(microsecond=us)        def get_container(cid):            if cid == 'eeeee':                raise NotFound(None, None, "oops")            if cid == 'abcde':                name = 'web'                labels = {LABEL_SERVICE: name}            elif cid == 'ababa':                name = 'db'                labels = {LABEL_SERVICE: name}            else:                labels = {}                name = ''            return {                'Id': cid,                'Config': {'Labels': labels},                'Name': '/project_%s_1' % name,            }        self.mock_client.inspect_container.side_effect = get_container        events = project.events()        events_list = list(events)        # Assert the return value is a generator        assert not list(events)        assert events_list == [            {                'type': 'container',                'service': 'web',                'action': 'create',                'id': 'abcde',                'attributes': {                    'name': 'project_web_1',                    'image': 'example/image',                },                'time': dt_with_microseconds(1420092061, 2),                'container': Container(None, {'Id': 'abcde'}),            },            {                'type': 'container',                'service': 'web',                'action': 'attach',                'id': 'abcde',                'attributes': {                    'name': 'project_web_1',                    'image': 'example/image',                },                'time': dt_with_microseconds(1420092061, 3),                'container': Container(None, {'Id': 'abcde'}),            },            {                'type': 'container',                'service': 'db',                'action': 'create',                'id': 'ababa',                'attributes': {                    'name': 'project_db_1',                    'image': 'example/db',                },                'time': dt_with_microseconds(1420092061, 4),                'container': Container(None, {'Id': 'ababa'}),            },        ]    def test_events(self):        services = [Service(name='web'), Service(name='db')]        project = Project('test', services, self.mock_client)        self.mock_client.api_version = '1.35'        self.mock_client.events.return_value = iter([            {                'status': 'create',                'from': 'example/image',                'Type': 'container',                'Actor': {                    'ID': 'abcde',                    'Attributes': {                        'com.docker.compose.project': 'test',                        'com.docker.compose.service': 'web',                        'image': 'example/image',                        'name': 'test_web_1',                    }                },                'id': 'abcde',                'time': 1420092061,                'timeNano': 14200920610000002000,            },            {                'status': 'attach',                'from': 'example/image',                'Type': 'container',                'Actor': {                    'ID': 'abcde',                    'Attributes': {                        'com.docker.compose.project': 'test',                        'com.docker.compose.service': 'web',                        'image': 'example/image',                        'name': 'test_web_1',                    }                },                'id': 'abcde',                'time': 1420092061,                'timeNano': 14200920610000003000,            },            {                'status': 'create',                'from': 'example/other',                'Type': 'container',                'Actor': {                    'ID': 'bdbdbd',                    'Attributes': {                        'image': 'example/other',                        'name': 'shrewd_einstein',                    }                },                'id': 'bdbdbd',                'time': 1420092061,                'timeNano': 14200920610000005000,            },            {                'status': 'create',                'from': 'example/db',                'Type': 'container',                'Actor': {                    'ID': 'ababa',                    'Attributes': {                        'com.docker.compose.project': 'test',                        'com.docker.compose.service': 'db',                        'image': 'example/db',                        'name': 'test_db_1',                    }                },                'id': 'ababa',                'time': 1420092061,                'timeNano': 14200920610000004000,            },            {                'status': 'destroy',                'from': 'example/db',                'Type': 'container',                'Actor': {                    'ID': 'eeeee',                    'Attributes': {                        'com.docker.compose.project': 'test',                        'com.docker.compose.service': 'db',                        'image': 'example/db',                        'name': 'test_db_1',                    }                },                'id': 'eeeee',                'time': 1420092061,                'timeNano': 14200920610000004000,            },        ])        def dt_with_microseconds(dt, us):            return datetime.datetime.fromtimestamp(dt).replace(microsecond=us)        def get_container(cid):            if cid == 'eeeee':                raise NotFound(None, None, "oops")            if cid == 'abcde':                name = 'web'                labels = {LABEL_SERVICE: name}            elif cid == 'ababa':                name = 'db'                labels = {LABEL_SERVICE: name}            else:                labels = {}                name = ''            return {                'Id': cid,                'Config': {'Labels': labels},                'Name': '/project_%s_1' % name,            }        self.mock_client.inspect_container.side_effect = get_container        events = project.events()        events_list = list(events)        # Assert the return value is a generator        assert not list(events)        assert events_list == [            {                'type': 'container',                'service': 'web',                'action': 'create',                'id': 'abcde',                'attributes': {                    'name': 'test_web_1',                    'image': 'example/image',                },                'time': dt_with_microseconds(1420092061, 2),                'container': Container(None, get_container('abcde')),            },            {                'type': 'container',                'service': 'web',                'action': 'attach',                'id': 'abcde',                'attributes': {                    'name': 'test_web_1',                    'image': 'example/image',                },                'time': dt_with_microseconds(1420092061, 3),                'container': Container(None, get_container('abcde')),            },            {                'type': 'container',                'service': 'db',                'action': 'create',                'id': 'ababa',                'attributes': {                    'name': 'test_db_1',                    'image': 'example/db',                },                'time': dt_with_microseconds(1420092061, 4),                'container': Container(None, get_container('ababa')),            },            {                'type': 'container',                'service': 'db',                'action': 'destroy',                'id': 'eeeee',                'attributes': {                    'name': 'test_db_1',                    'image': 'example/db',                },                'time': dt_with_microseconds(1420092061, 4),                'container': None,            },        ]    def test_net_unset(self):        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V1,                services=[                    {                        'name': 'test',                        'image': BUSYBOX_IMAGE_WITH_TAG,                    }                ],                networks=None,                volumes=None,                secrets=None,                configs=None,            ),        )        service = project.get_service('test')        assert service.network_mode.id is None        assert 'NetworkMode' not in service._get_container_host_config({})    def test_use_net_from_container(self):        container_id = 'aabbccddee'        container_dict = dict(Name='aaa', Id=container_id)        self.mock_client.inspect_container.return_value = container_dict        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V2_0,                services=[                    {                        'name': 'test',                        'image': BUSYBOX_IMAGE_WITH_TAG,                        'network_mode': 'container:aaa'                    },                ],                networks=None,                volumes=None,                secrets=None,                configs=None,            ),        )        service = project.get_service('test')        assert service.network_mode.mode == 'container:' + container_id    def test_use_net_from_service(self):        container_name = 'test_aaa_1'        self.mock_client.containers.return_value = [            {                "Name": container_name,                "Names": [container_name],                "Id": container_name,                "Image": BUSYBOX_IMAGE_WITH_TAG            }        ]        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V2_0,                services=[                    {                        'name': 'aaa',                        'image': BUSYBOX_IMAGE_WITH_TAG                    },                    {                        'name': 'test',                        'image': BUSYBOX_IMAGE_WITH_TAG,                        'network_mode': 'service:aaa'                    },                ],                networks=None,                volumes=None,                secrets=None,                configs=None,            ),        )        service = project.get_service('test')        assert service.network_mode.mode == 'container:' + container_name    def test_uses_default_network_true(self):        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V2_0,                services=[                    {                        'name': 'foo',                        'image': BUSYBOX_IMAGE_WITH_TAG                    },                ],                networks=None,                volumes=None,                secrets=None,                configs=None,            ),        )        assert 'default' in project.networks.networks    def test_uses_default_network_false(self):        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V2_0,                services=[                    {                        'name': 'foo',                        'image': BUSYBOX_IMAGE_WITH_TAG,                        'networks': {'custom': None}                    },                ],                networks={'custom': {}},                volumes=None,                secrets=None,                configs=None,            ),        )        assert 'default' not in project.networks.networks    def test_container_without_name(self):        self.mock_client.containers.return_value = [            {'Image': BUSYBOX_IMAGE_WITH_TAG, 'Id': '1', 'Name': '1'},            {'Image': BUSYBOX_IMAGE_WITH_TAG, 'Id': '2', 'Name': None},            {'Image': BUSYBOX_IMAGE_WITH_TAG, 'Id': '3'},        ]        self.mock_client.inspect_container.return_value = {            'Id': '1',            'Config': {                'Labels': {                    LABEL_SERVICE: 'web',                },            },        }        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V2_0,                services=[{                    'name': 'web',                    'image': BUSYBOX_IMAGE_WITH_TAG,                }],                networks=None,                volumes=None,                secrets=None,                configs=None,            ),        )        assert [c.id for c in project.containers()] == ['1']    def test_down_with_no_resources(self):        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V2_0,                services=[{                    'name': 'web',                    'image': BUSYBOX_IMAGE_WITH_TAG,                }],                networks={'default': {}},                volumes={'data': {}},                secrets=None,                configs=None,            ),        )        self.mock_client.remove_network.side_effect = NotFound(None, None, 'oops')        self.mock_client.remove_volume.side_effect = NotFound(None, None, 'oops')        project.down(ImageType.all, True)        self.mock_client.remove_image.assert_called_once_with(BUSYBOX_IMAGE_WITH_TAG)    def test_no_warning_on_stop(self):        self.mock_client.info.return_value = {'Swarm': {'LocalNodeState': 'active'}}        project = Project('composetest', [], self.mock_client)        with mock.patch('compose.project.log') as fake_log:            project.stop()            assert fake_log.warn.call_count == 0    def test_no_warning_in_normal_mode(self):        self.mock_client.info.return_value = {'Swarm': {'LocalNodeState': 'inactive'}}        project = Project('composetest', [], self.mock_client)        with mock.patch('compose.project.log') as fake_log:            project.up()            assert fake_log.warn.call_count == 0    def test_no_warning_with_no_swarm_info(self):        self.mock_client.info.return_value = {}        project = Project('composetest', [], self.mock_client)        with mock.patch('compose.project.log') as fake_log:            project.up()            assert fake_log.warn.call_count == 0    def test_no_such_service_unicode(self):        assert NoSuchService('十六夜 咲夜'.encode('utf-8')).msg == 'No such service: 十六夜 咲夜'        assert NoSuchService('十六夜 咲夜').msg == 'No such service: 十六夜 咲夜'    def test_project_platform_value(self):        service_config = {            'name': 'web',            'image': BUSYBOX_IMAGE_WITH_TAG,        }        config_data = Config(            version=V2_4, services=[service_config], networks={}, volumes={}, secrets=None, configs=None        )        project = Project.from_config(name='test', client=self.mock_client, config_data=config_data)        assert project.get_service('web').platform is None        project = Project.from_config(            name='test', client=self.mock_client, config_data=config_data, default_platform='windows'        )        assert project.get_service('web').platform == 'windows'        service_config['platform'] = 'linux/s390x'        project = Project.from_config(name='test', client=self.mock_client, config_data=config_data)        assert project.get_service('web').platform == 'linux/s390x'        project = Project.from_config(            name='test', client=self.mock_client, config_data=config_data, default_platform='windows'        )        assert project.get_service('web').platform == 'linux/s390x'    def test_build_container_operation_with_timeout_func_does_not_mutate_options_with_timeout(self):        config_data = Config(            version=V3_7,            services=[                {'name': 'web', 'image': BUSYBOX_IMAGE_WITH_TAG},                {'name': 'db', 'image': BUSYBOX_IMAGE_WITH_TAG, 'stop_grace_period': '1s'},            ],            networks={}, volumes={}, secrets=None, configs=None,        )        project = Project.from_config(name='test', client=self.mock_client, config_data=config_data)        stop_op = project.build_container_operation_with_timeout_func('stop', options={})        web_container = mock.create_autospec(Container, service='web')        db_container = mock.create_autospec(Container, service='db')        # `stop_grace_period` is not set to 'web' service,        # then it is stopped with the default timeout.        stop_op(web_container)        web_container.stop.assert_called_once_with(timeout=DEFAULT_TIMEOUT)        # `stop_grace_period` is set to 'db' service,        # then it is stopped with the specified timeout and        # the value is not overridden by the previous function call.        stop_op(db_container)        db_container.stop.assert_called_once_with(timeout=1)    @mock.patch('compose.parallel.ParallelStreamWriter._write_noansi')    def test_error_parallel_pull(self, mock_write):        project = Project.from_config(            name='test',            client=self.mock_client,            config_data=Config(                version=V2_0,                services=[{                    'name': 'web',                    'image': BUSYBOX_IMAGE_WITH_TAG,                }],                networks=None,                volumes=None,                secrets=None,                configs=None,            ),        )        self.mock_client.pull.side_effect = OperationFailedError('pull error')        with pytest.raises(ProjectError):            project.pull(parallel_pull=True)        self.mock_client.pull.side_effect = OperationFailedError(b'pull error')        with pytest.raises(ProjectError):            project.pull(parallel_pull=True)    def test_avoid_multiple_push(self):        service_config_latest = {'image': 'busybox:latest', 'build': '.'}        service_config_default = {'image': 'busybox', 'build': '.'}        service_config_sha = {            'image': 'busybox@sha256:38a203e1986cf79639cfb9b2e1d6e773de84002feea2d4eb006b52004ee8502d',            'build': '.'        }        svc1 = Service('busy1', **service_config_latest)        svc1_1 = Service('busy11', **service_config_latest)        svc2 = Service('busy2', **service_config_default)        svc2_1 = Service('busy21', **service_config_default)        svc3 = Service('busy3', **service_config_sha)        svc3_1 = Service('busy31', **service_config_sha)        project = Project(            'composetest', [svc1, svc1_1, svc2, svc2_1, svc3, svc3_1], self.mock_client        )        with mock.patch('compose.service.Service.push') as fake_push:            project.push()            assert fake_push.call_count == 2    def test_get_secrets_no_secret_def(self):        service = 'foo'        secret_source = 'bar'        secret_defs = mock.Mock()        secret_defs.get.return_value = None        secret = mock.Mock(source=secret_source)        with self.assertRaises(ConfigurationError):            get_secrets(service, [secret], secret_defs)    def test_get_secrets_external_warning(self):        service = 'foo'        secret_source = 'bar'        secret_def = mock.Mock()        secret_def.get.return_value = True        secret_defs = mock.Mock()        secret_defs.get.side_effect = secret_def        secret = mock.Mock(source=secret_source)        with mock.patch('compose.project.log') as mock_log:            get_secrets(service, [secret], secret_defs)        mock_log.warning.assert_called_with("Service \"{service}\" uses secret \"{secret}\" "                                            "which is external. External secrets are not available"                                            " to containers created by docker-compose."                                            .format(service=service, secret=secret_source))    def test_get_secrets_uid_gid_mode_warning(self):        service = 'foo'        secret_source = 'bar'        _, filename_path = tempfile.mkstemp()        self.addCleanup(os.remove, filename_path)        def mock_get(key):            return {'external': False, 'file': filename_path}[key]        secret_def = mock.MagicMock()        secret_def.get = mock.MagicMock(side_effect=mock_get)        secret_defs = mock.Mock()        secret_defs.get.return_value = secret_def        secret = mock.Mock(uid=True, gid=True, mode=True, source=secret_source)        with mock.patch('compose.project.log') as mock_log:            get_secrets(service, [secret], secret_defs)        mock_log.warning.assert_called_with("Service \"{service}\" uses secret \"{secret}\" with uid, "                                            "gid, or mode. These fields are not supported by this "                                            "implementation of the Compose file"                                            .format(service=service, secret=secret_source))    def test_get_secrets_secret_file_warning(self):        service = 'foo'        secret_source = 'bar'        not_a_path = 'NOT_A_PATH'        def mock_get(key):            return {'external': False, 'file': not_a_path}[key]        secret_def = mock.MagicMock()        secret_def.get = mock.MagicMock(side_effect=mock_get)        secret_defs = mock.Mock()        secret_defs.get.return_value = secret_def        secret = mock.Mock(uid=False, gid=False, mode=False, source=secret_source)        with mock.patch('compose.project.log') as mock_log:            get_secrets(service, [secret], secret_defs)        mock_log.warning.assert_called_with("Service \"{service}\" uses an undefined secret file "                                            "\"{secret_file}\", the following file should be created "                                            "\"{secret_file}\""                                            .format(service=service, secret_file=not_a_path))
 |