From acf123d2f826b3fe2691a34c98363e94fa75b467 Mon Sep 17 00:00:00 2001
From: Chris Behrens <cbehrens@codestud.com>
Date: Wed, 3 Jul 2013 22:37:52 +0000
Subject: [PATCH] Add object (de)serialization support to cells

This adjusts the cells rpcapi to use the Object serializer class.
This also makes the cells messaging layer, which handles formatting of
cell<->cell messages, serialize objects correctly when contained as
arguments to the cells messaging methods.

Related to blueprint unified-object-model

Change-Id: Iec328827819f25a16d1fd901f4d6c896525f0bb6
---
 nova/cells/messaging.py                   | 16 ++++++++-
 nova/cells/rpc_driver.py                  |  3 ++
 nova/cells/rpcapi.py                      |  2 ++
 nova/tests/cells/test_cells_messaging.py  | 44 +++++++++++++++++++++++
 nova/tests/cells/test_cells_rpc_driver.py |  8 ++---
 5 files changed, 68 insertions(+), 5 deletions(-)

diff --git a/nova/cells/messaging.py b/nova/cells/messaging.py
index cacb4b8e36ab..5f672f007487 100644
--- a/nova/cells/messaging.py
+++ b/nova/cells/messaging.py
@@ -36,6 +36,7 @@ from nova.consoleauth import rpcapi as consoleauth_rpcapi
 from nova import context
 from nova.db import base
 from nova import exception
+from nova.objects import base as objects_base
 from nova.objects import instance as instance_obj
 from nova.openstack.common import excutils
 from nova.openstack.common import importutils
@@ -163,6 +164,7 @@ class _BaseMessage(object):
         # Each sub-class should set this when the message is inited
         self.next_hops = []
         self.resp_queue = None
+        self.serializer = objects_base.NovaObjectSerializer()
 
     def __repr__(self):
         _dict = self._to_dict()
@@ -305,6 +307,11 @@ class _BaseMessage(object):
         _dict = self._to_dict()
         # Convert context to dict.
         _dict['ctxt'] = _dict['ctxt'].to_dict()
+        # NOTE(comstud): 'method_kwargs' needs special serialization
+        # because it may contain objects.
+        method_kwargs = _dict['method_kwargs']
+        for k, v in method_kwargs.items():
+            method_kwargs[k] = self.serializer.serialize_entity(self.ctxt, v)
         return jsonutils.dumps(_dict)
 
     def source_is_us(self):
@@ -1065,6 +1072,7 @@ class MessageRunner(object):
         self.our_name = CONF.cells.name
         for msg_type, cls in _CELL_MESSAGE_TYPE_TO_METHODS_CLS.iteritems():
             self.methods_by_type[msg_type] = cls(self)
+        self.serializer = objects_base.NovaObjectSerializer()
 
     def _process_message_locally(self, message):
         """Message processing will call this when its determined that
@@ -1124,10 +1132,16 @@ class MessageRunner(object):
         another cell.
         """
         message_dict = jsonutils.loads(json_message)
-        message_type = message_dict.pop('message_type')
         # Need to convert context back.
         ctxt = message_dict['ctxt']
         message_dict['ctxt'] = context.RequestContext.from_dict(ctxt)
+        # NOTE(comstud): We also need to re-serialize any objects that
+        # exist in 'method_kwargs'.
+        method_kwargs = message_dict['method_kwargs']
+        for k, v in method_kwargs.items():
+            method_kwargs[k] = self.serializer.deserialize_entity(
+                    message_dict['ctxt'], v)
+        message_type = message_dict.pop('message_type')
         message_cls = _CELL_MESSAGE_TYPE_TO_MESSAGE_CLS[message_type]
         return message_cls(self, **message_dict)
 
diff --git a/nova/cells/rpc_driver.py b/nova/cells/rpc_driver.py
index 6b11e0f058cc..8ef027c30920 100644
--- a/nova/cells/rpc_driver.py
+++ b/nova/cells/rpc_driver.py
@@ -82,6 +82,9 @@ class CellsRPCDriver(driver.BaseCellsDriver):
         """
         topic_base = CONF.cells.rpc_driver_queue_base
         proxy_manager = InterCellRPCDispatcher(msg_runner)
+        # NOTE(comstud): We do not need to use the object serializer
+        # on this because object serialization is taken care for us in
+        # the messaging module.
         dispatcher = rpc_dispatcher.RpcDispatcher([proxy_manager])
         for msg_type in msg_runner.get_message_types():
             topic = '%s.%s' % (topic_base, msg_type)
diff --git a/nova/cells/rpcapi.py b/nova/cells/rpcapi.py
index 6a54800731ae..5a9ac99fc5f3 100644
--- a/nova/cells/rpcapi.py
+++ b/nova/cells/rpcapi.py
@@ -25,6 +25,7 @@ messging module.
 from oslo.config import cfg
 
 from nova import exception
+from nova.objects import base as objects_base
 from nova.openstack.common import jsonutils
 from nova.openstack.common import log as logging
 from nova.openstack.common.rpc import proxy as rpc_proxy
@@ -77,6 +78,7 @@ class CellsAPI(rpc_proxy.RpcProxy):
                                                CONF.upgrade_levels.cells)
         super(CellsAPI, self).__init__(topic=CONF.cells.topic,
                 default_version=self.BASE_RPC_API_VERSION,
+                serializer=objects_base.NovaObjectSerializer(),
                 version_cap=version_cap)
 
     def cast_compute_api_method(self, ctxt, cell_name, method,
diff --git a/nova/tests/cells/test_cells_messaging.py b/nova/tests/cells/test_cells_messaging.py
index 734fecf0e2e0..3c8508367746 100644
--- a/nova/tests/cells/test_cells_messaging.py
+++ b/nova/tests/cells/test_cells_messaging.py
@@ -23,6 +23,7 @@ from nova.compute import vm_states
 from nova import context
 from nova import db
 from nova import exception
+from nova.objects import base as objects_base
 from nova.objects import instance as instance_obj
 from nova.openstack.common import rpc
 from nova.openstack.common import timeutils
@@ -260,6 +261,49 @@ class CellsMessageClassesTestCase(test.TestCase):
         self.assertEqual(method_kwargs, call_info['kwargs'])
         self.assertEqual(target_cell, call_info['routing_path'])
 
+    def test_child_targeted_message_with_object(self):
+        target_cell = 'api-cell!child-cell1'
+        method = 'our_fake_method'
+        direction = 'down'
+
+        call_info = {}
+
+        class CellsMsgingTestObject(objects_base.NovaObject):
+            """Test object.  We just need 1 field in order to test
+            that this gets serialized properly.
+            """
+            fields = {'test': str}
+
+        test_obj = CellsMsgingTestObject()
+        test_obj.test = 'meow'
+
+        method_kwargs = dict(obj=test_obj, arg1=1, arg2=2)
+
+        def our_fake_method(message, **kwargs):
+            call_info['context'] = message.ctxt
+            call_info['routing_path'] = message.routing_path
+            call_info['kwargs'] = kwargs
+
+        fakes.stub_tgt_method(self, 'child-cell1', 'our_fake_method',
+                our_fake_method)
+
+        tgt_message = messaging._TargetedMessage(self.msg_runner,
+                                                  self.ctxt, method,
+                                                  method_kwargs, direction,
+                                                  target_cell)
+        tgt_message.process()
+
+        self.assertEqual(self.ctxt, call_info['context'])
+        self.assertEqual(target_cell, call_info['routing_path'])
+        self.assertEqual(3, len(call_info['kwargs']))
+        self.assertEqual(1, call_info['kwargs']['arg1'])
+        self.assertEqual(2, call_info['kwargs']['arg2'])
+        # Verify we get a new object with what we expect.
+        obj = call_info['kwargs']['obj']
+        self.assertTrue(isinstance(obj, CellsMsgingTestObject))
+        self.assertNotEqual(id(test_obj), id(obj))
+        self.assertEqual(test_obj.test, obj.test)
+
     def test_grandchild_targeted_message(self):
         target_cell = 'api-cell!child-cell2!grandchild-cell1'
         method = 'our_fake_method'
diff --git a/nova/tests/cells/test_cells_rpc_driver.py b/nova/tests/cells/test_cells_rpc_driver.py
index e5e847c12290..c095a1ad2607 100644
--- a/nova/tests/cells/test_cells_rpc_driver.py
+++ b/nova/tests/cells/test_cells_rpc_driver.py
@@ -101,7 +101,7 @@ class CellsRPCDriverTestCase(test.TestCase):
         msg_runner = fakes.get_message_runner('api-cell')
         cell_state = fakes.get_cell_state('api-cell', 'child-cell2')
         message = messaging._TargetedMessage(msg_runner,
-                self.ctxt, 'fake', 'fake', 'down', cell_state, fanout=False)
+                self.ctxt, 'fake', {}, 'down', cell_state, fanout=False)
 
         call_info = {}
 
@@ -139,7 +139,7 @@ class CellsRPCDriverTestCase(test.TestCase):
         msg_runner = fakes.get_message_runner('api-cell')
         cell_state = fakes.get_cell_state('api-cell', 'child-cell2')
         message = messaging._TargetedMessage(msg_runner,
-                self.ctxt, 'fake', 'fake', 'down', cell_state, fanout=True)
+                self.ctxt, 'fake', {}, 'down', cell_state, fanout=True)
 
         call_info = {}
 
@@ -179,7 +179,7 @@ class CellsRPCDriverTestCase(test.TestCase):
         msg_runner = fakes.get_message_runner('api-cell')
         cell_state = fakes.get_cell_state('api-cell', 'child-cell2')
         message = messaging._BroadcastMessage(msg_runner,
-                self.ctxt, 'fake', 'fake', 'down', fanout=True)
+                self.ctxt, 'fake', {}, 'down', fanout=True)
         message.message_type = 'fake-message-type'
 
         call_info = {}
@@ -198,7 +198,7 @@ class CellsRPCDriverTestCase(test.TestCase):
         msg_runner = fakes.get_message_runner('api-cell')
         dispatcher = rpc_driver.InterCellRPCDispatcher(msg_runner)
         message = messaging._BroadcastMessage(msg_runner,
-                self.ctxt, 'fake', 'fake', 'down', fanout=True)
+                self.ctxt, 'fake', {}, 'down', fanout=True)
 
         call_info = {}