Make RPCFixture support multiple connections

For testing cells, we will need to track the driver instances that
we give out by url. This normally just works with a conventional
oslo.messaging driver, but the fake driver keeps internal data
structures for simulating its bus. If we end up with clients creating
a new instance of the driver in the rpc switching code, we'll never
be able to send messages to services because we'll always have
private/separate data structures.

So, this makes the fixture wrap the transport creation stuff
and unify references by url. In order to make this work, some
retooling of rpc.init() is done, which makes it more in line with
the recent additions we had for wrapping transport initialization
per connection anyway.

For now, a lot of our tests can't handle the possibility of
multiple RPC connections due to them looking at the global
transport_url configuration. So for the moment, even though this
makes the fixture support multiple independent connections, we
collapse any such attempts down to a single connection to the
default broker.

Note: this requires a fix in oslo.messaging 5.14.0

Depends-On: I01b6f5a20ea9752da46a546a908bd38cf11da681
Change-Id: Icb63d7dabd17f3c5633387793f68a8ba20863a7e
This commit is contained in:
Dan Smith 2016-11-10 13:19:32 -08:00
parent 668f2f9631
commit 9dcab12b7c
4 changed files with 51 additions and 21 deletions

View File

@ -71,9 +71,7 @@ TRANSPORT_ALIASES = {
def init(conf): def init(conf):
global TRANSPORT, NOTIFICATION_TRANSPORT, LEGACY_NOTIFIER, NOTIFIER global TRANSPORT, NOTIFICATION_TRANSPORT, LEGACY_NOTIFIER, NOTIFIER
exmods = get_allowed_exmods() exmods = get_allowed_exmods()
TRANSPORT = messaging.get_transport(conf, TRANSPORT = create_transport(get_transport_url())
allowed_remote_exmods=exmods,
aliases=TRANSPORT_ALIASES)
NOTIFICATION_TRANSPORT = messaging.get_notification_transport( NOTIFICATION_TRANSPORT = messaging.get_notification_transport(
conf, allowed_remote_exmods=exmods, aliases=TRANSPORT_ALIASES) conf, allowed_remote_exmods=exmods, aliases=TRANSPORT_ALIASES)
serializer = RequestContextSerializer(JsonPayloadSerializer()) serializer = RequestContextSerializer(JsonPayloadSerializer())

View File

@ -27,6 +27,7 @@ import mock
from oslo_concurrency import lockutils from oslo_concurrency import lockutils
from oslo_config import cfg from oslo_config import cfg
from oslo_db.sqlalchemy import enginefacade from oslo_db.sqlalchemy import enginefacade
import oslo_messaging as messaging
from oslo_messaging import conffixture as messaging_conffixture from oslo_messaging import conffixture as messaging_conffixture
import six import six
@ -495,6 +496,27 @@ class RPCFixture(fixtures.Fixture):
super(RPCFixture, self).__init__() super(RPCFixture, self).__init__()
self.exmods = [] self.exmods = []
self.exmods.extend(exmods) self.exmods.extend(exmods)
self._buses = {}
def _fake_create_transport(self, url):
# FIXME(danms): Right now, collapse all connections
# to a single bus. This is how our tests expect things
# to work. When the tests are fixed, this fixture can
# support simulating multiple independent buses, and this
# hack should be removed.
url = None
# NOTE(danms): This will be called with a non-None url by
# cells-aware code that is requesting to contact something on
# one of the many transports we're multplexing here.
if url not in self._buses:
exmods = rpc.get_allowed_exmods()
self._buses[url] = messaging.get_transport(
CONF,
url=url,
allowed_remote_exmods=exmods,
aliases=rpc.TRANSPORT_ALIASES)
return self._buses[url]
def setUp(self): def setUp(self):
super(RPCFixture, self).setUp() super(RPCFixture, self).setUp()
@ -504,7 +526,15 @@ class RPCFixture(fixtures.Fixture):
self.messaging_conf = messaging_conffixture.ConfFixture(CONF) self.messaging_conf = messaging_conffixture.ConfFixture(CONF)
self.messaging_conf.transport_driver = 'fake' self.messaging_conf.transport_driver = 'fake'
self.useFixture(self.messaging_conf) self.useFixture(self.messaging_conf)
rpc.init(CONF) self.useFixture(fixtures.MonkeyPatch(
'nova.rpc.create_transport', self._fake_create_transport))
# NOTE(danms): Execute the init with get_transport_url() as None,
# instead of the parsed TransportURL(None) so that we can cache
# it as it will be called later if the default is requested by
# one of our mq-switching methods.
with mock.patch('nova.rpc.get_transport_url') as mock_gtu:
mock_gtu.return_value = None
rpc.init(CONF)
class WarningsFixture(fixtures.Fixture): class WarningsFixture(fixtures.Fixture):

View File

@ -56,38 +56,35 @@ class TestRPC(testtools.TestCase):
@mock.patch.object(rpc, 'get_allowed_exmods') @mock.patch.object(rpc, 'get_allowed_exmods')
@mock.patch.object(rpc, 'RequestContextSerializer') @mock.patch.object(rpc, 'RequestContextSerializer')
@mock.patch.object(messaging, 'get_transport')
@mock.patch.object(messaging, 'get_notification_transport') @mock.patch.object(messaging, 'get_notification_transport')
@mock.patch.object(messaging, 'Notifier') @mock.patch.object(messaging, 'Notifier')
def test_init_unversioned(self, mock_notif, mock_noti_trans, mock_trans, def test_init_unversioned(self, mock_notif, mock_noti_trans,
mock_ser, mock_exmods): mock_ser, mock_exmods):
# The expected call to get the legacy notifier will require no new # The expected call to get the legacy notifier will require no new
# kwargs, and we expect the new notifier will need the noop driver # kwargs, and we expect the new notifier will need the noop driver
expected = [{}, {'driver': 'noop'}] expected = [{}, {'driver': 'noop'}]
self._test_init(mock_notif, mock_noti_trans, mock_trans, mock_ser, self._test_init(mock_notif, mock_noti_trans, mock_ser,
mock_exmods, 'unversioned', expected) mock_exmods, 'unversioned', expected)
@mock.patch.object(rpc, 'get_allowed_exmods') @mock.patch.object(rpc, 'get_allowed_exmods')
@mock.patch.object(rpc, 'RequestContextSerializer') @mock.patch.object(rpc, 'RequestContextSerializer')
@mock.patch.object(messaging, 'get_transport')
@mock.patch.object(messaging, 'get_notification_transport') @mock.patch.object(messaging, 'get_notification_transport')
@mock.patch.object(messaging, 'Notifier') @mock.patch.object(messaging, 'Notifier')
def test_init_both(self, mock_notif, mock_noti_trans, mock_trans, def test_init_both(self, mock_notif, mock_noti_trans,
mock_ser, mock_exmods): mock_ser, mock_exmods):
expected = [{}, {'topics': ['versioned_notifications']}] expected = [{}, {'topics': ['versioned_notifications']}]
self._test_init(mock_notif, mock_noti_trans, mock_trans, mock_ser, self._test_init(mock_notif, mock_noti_trans, mock_ser,
mock_exmods, 'both', expected) mock_exmods, 'both', expected)
@mock.patch.object(rpc, 'get_allowed_exmods') @mock.patch.object(rpc, 'get_allowed_exmods')
@mock.patch.object(rpc, 'RequestContextSerializer') @mock.patch.object(rpc, 'RequestContextSerializer')
@mock.patch.object(messaging, 'get_transport')
@mock.patch.object(messaging, 'get_notification_transport') @mock.patch.object(messaging, 'get_notification_transport')
@mock.patch.object(messaging, 'Notifier') @mock.patch.object(messaging, 'Notifier')
def test_init_versioned(self, mock_notif, mock_noti_trans, mock_trans, def test_init_versioned(self, mock_notif, mock_noti_trans,
mock_ser, mock_exmods): mock_ser, mock_exmods):
expected = [{'driver': 'noop'}, expected = [{'driver': 'noop'},
{'topics': ['versioned_notifications']}] {'topics': ['versioned_notifications']}]
self._test_init(mock_notif, mock_noti_trans, mock_trans, mock_ser, self._test_init(mock_notif, mock_noti_trans, mock_ser,
mock_exmods, 'versioned', expected) mock_exmods, 'versioned', expected)
def test_cleanup_transport_null(self): def test_cleanup_transport_null(self):
@ -266,7 +263,7 @@ class TestRPC(testtools.TestCase):
allowed_remote_exmods=exmods, allowed_remote_exmods=exmods,
aliases=rpc.TRANSPORT_ALIASES) aliases=rpc.TRANSPORT_ALIASES)
def _test_init(self, mock_notif, mock_noti_trans, mock_trans, mock_ser, def _test_init(self, mock_notif, mock_noti_trans, mock_ser,
mock_exmods, notif_format, expected_driver_topic_kwargs): mock_exmods, notif_format, expected_driver_topic_kwargs):
legacy_notifier = mock.Mock() legacy_notifier = mock.Mock()
notifier = mock.Mock() notifier = mock.Mock()
@ -275,19 +272,24 @@ class TestRPC(testtools.TestCase):
serializer = mock.Mock() serializer = mock.Mock()
conf = mock.Mock() conf = mock.Mock()
conf.transport_url = None
conf.notification_format = notif_format conf.notification_format = notif_format
mock_exmods.return_value = ['foo'] mock_exmods.return_value = ['foo']
mock_trans.return_value = transport
mock_noti_trans.return_value = notif_transport mock_noti_trans.return_value = notif_transport
mock_ser.return_value = serializer mock_ser.return_value = serializer
mock_notif.side_effect = [legacy_notifier, notifier] mock_notif.side_effect = [legacy_notifier, notifier]
rpc.init(conf) @mock.patch.object(rpc, 'CONF', new=conf)
@mock.patch.object(rpc, 'create_transport')
@mock.patch.object(rpc, 'get_transport_url')
def _test(get_url, create_transport):
create_transport.return_value = transport
rpc.init(conf)
create_transport.assert_called_once_with(get_url.return_value)
mock_exmods.assert_called_once_with() _test()
mock_trans.assert_called_once_with(conf,
allowed_remote_exmods=['foo'], self.assertTrue(mock_exmods.called)
aliases=rpc.TRANSPORT_ALIASES)
self.assertIsNotNone(rpc.TRANSPORT) self.assertIsNotNone(rpc.TRANSPORT)
self.assertIsNotNone(rpc.LEGACY_NOTIFIER) self.assertIsNotNone(rpc.LEGACY_NOTIFIER)
self.assertIsNotNone(rpc.NOTIFIER) self.assertIsNotNone(rpc.NOTIFIER)

View File

@ -43,7 +43,7 @@ oslo.serialization>=1.10.0 # Apache-2.0
oslo.utils>=3.18.0 # Apache-2.0 oslo.utils>=3.18.0 # Apache-2.0
oslo.db!=4.13.1,!=4.13.2,>=4.11.0 # Apache-2.0 oslo.db!=4.13.1,!=4.13.2,>=4.11.0 # Apache-2.0
oslo.rootwrap>=5.0.0 # Apache-2.0 oslo.rootwrap>=5.0.0 # Apache-2.0
oslo.messaging>=5.2.0 # Apache-2.0 oslo.messaging>=5.14.0 # Apache-2.0
oslo.policy>=1.17.0 # Apache-2.0 oslo.policy>=1.17.0 # Apache-2.0
oslo.privsep>=1.9.0 # Apache-2.0 oslo.privsep>=1.9.0 # Apache-2.0
oslo.i18n>=2.1.0 # Apache-2.0 oslo.i18n>=2.1.0 # Apache-2.0