[Zope-Checkins] SVN: Zope/trunk/lib/python/ Made DateTime.DateTime marshallable via XML-RPC.

Stefan H. Holek cvs-admin at zope.org
Fri Jun 16 11:55:53 EDT 2006


Log message for revision 68692:
  Made DateTime.DateTime marshallable via XML-RPC.
  Fixes http://www.zope.org/Collectors/Zope/2109
  

Changed:
  U   Zope/trunk/lib/python/DateTime/DateTime.py
  U   Zope/trunk/lib/python/ZPublisher/tests/test_xmlrpc.py
  U   Zope/trunk/lib/python/ZPublisher/xmlrpc.py

-=-
Modified: Zope/trunk/lib/python/DateTime/DateTime.py
===================================================================
--- Zope/trunk/lib/python/DateTime/DateTime.py	2006-06-16 15:54:47 UTC (rev 68691)
+++ Zope/trunk/lib/python/DateTime/DateTime.py	2006-06-16 15:55:50 UTC (rev 68692)
@@ -1797,7 +1797,15 @@
         d1 = (( d4 - L) % 365) + L
         return d1/7 + 1
 
+    def encode(self, out):
+        """
+        Encode value for XML-RPC
+        """
+        out.write('<value><dateTime.iso8601>')
+        out.write(self.ISO8601())
+        out.write('</dateTime.iso8601></value>\n')
 
+
 class strftimeFormatter:
 
     def __init__(self, dt, format):

Modified: Zope/trunk/lib/python/ZPublisher/tests/test_xmlrpc.py
===================================================================
--- Zope/trunk/lib/python/ZPublisher/tests/test_xmlrpc.py	2006-06-16 15:54:47 UTC (rev 68691)
+++ Zope/trunk/lib/python/ZPublisher/tests/test_xmlrpc.py	2006-06-16 15:55:50 UTC (rev 68692)
@@ -1,4 +1,5 @@
 import unittest
+from DateTime import DateTime
 
 class FauxResponse:
 
@@ -12,6 +13,9 @@
     def setHeader(self, name, value):
         self._headers[name] = value
 
+    def setStatus(self, status):
+        self._status = status
+
 class FauxInstance:
     def __init__(self, **kw):
         self.__dict__.update(kw)
@@ -55,7 +59,139 @@
         data, method = xmlrpclib.loads(faux._body)
         self.assert_(data[0]['public'] is None)
 
+    def test_instance(self):
+        # Instances are turned into dicts with their private
+        # attributes removed.
+        import xmlrpclib
+        body = FauxInstance(_secret='abc', public='def')
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0]
+        self.assertEqual(data, {'public': 'def'})
 
+    def test_instanceattribute(self):
+        # While the removal of private ('_') attributes works fine for the
+        # top-level instance, how about attributes that are themselves
+        # instances?
+        import xmlrpclib
+        body = FauxInstance(public=FauxInstance(_secret='abc', public='def'))
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0]['public']
+        self.assertEqual(data, {'public': 'def'})
+
+    def test_instanceattribute_recursive(self):
+        # Instance "flattening" should work recursively, ad infinitum
+        import xmlrpclib
+        body = FauxInstance(public=FauxInstance(public=FauxInstance(_secret='abc', public='def')))
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0]['public']['public']
+        self.assertEqual(data, {'public': 'def'})
+
+    def test_instance_in_list(self):
+        # Instances are turned into dicts with their private
+        # attributes removed, even when embedded in another
+        # data structure.
+        import xmlrpclib
+        body = [FauxInstance(_secret='abc', public='def')]
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0][0]
+        self.assertEqual(data, {'public': 'def'})
+
+    def test_instance_in_dict(self):
+        # Instances are turned into dicts with their private
+        # attributes removed, even when embedded in another
+        # data structure.
+        import xmlrpclib
+        body = {'faux': FauxInstance(_secret='abc', public='def')}
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0]['faux']
+        self.assertEqual(data, {'public': 'def'})
+
+    def test_zopedatetimeinstance(self):
+        # DateTime instance at top-level
+        import xmlrpclib
+        body = DateTime('2006-05-24 07:00:00 GMT+0')
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0]
+        self.failUnless(isinstance(data, xmlrpclib.DateTime))
+        self.assertEqual(data.value, u'2006-05-24T07:00:00+00:00')
+
+    def test_zopedatetimeattribute(self):
+        # DateTime instance as attribute
+        import xmlrpclib
+        body = FauxInstance(public=DateTime('2006-05-24 07:00:00 GMT+0'))
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0]['public']
+        self.failUnless(isinstance(data, xmlrpclib.DateTime))
+        self.assertEqual(data.value, u'2006-05-24T07:00:00+00:00')
+
+    def test_zopedatetimeattribute_recursive(self):
+        # DateTime encoding should work recursively
+        import xmlrpclib
+        body = FauxInstance(public=FauxInstance(public=DateTime('2006-05-24 07:00:00 GMT+0')))
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0]['public']['public']
+        self.failUnless(isinstance(data, xmlrpclib.DateTime))
+        self.assertEqual(data.value, u'2006-05-24T07:00:00+00:00')
+
+    def test_zopedatetimeinstance_in_list(self):
+        # DateTime instance embedded in a list
+        import xmlrpclib
+        body = [DateTime('2006-05-24 07:00:00 GMT+0')]
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0][0]
+        self.failUnless(isinstance(data, xmlrpclib.DateTime))
+        self.assertEqual(data.value, u'2006-05-24T07:00:00+00:00')
+
+    def test_zopedatetimeinstance_in_dict(self):
+        # DateTime instance embedded in a dict
+        import xmlrpclib
+        body = {'date': DateTime('2006-05-24 07:00:00 GMT+0')}
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        data, method = xmlrpclib.loads(faux._body)
+        data = data[0]['date']
+        self.failUnless(isinstance(data, xmlrpclib.DateTime))
+        self.assertEqual(data.value, u'2006-05-24T07:00:00+00:00')
+
+    def test_functionattribute(self):
+        # Cannot marshal functions or methods, obviously
+        import xmlrpclib
+        def foo(): pass
+        body = FauxInstance(public=foo)
+        faux = FauxResponse()
+        response = self._makeOne(faux)
+        response.setBody(body)
+        self.assertRaises(xmlrpclib.Fault, xmlrpclib.loads, faux._body)
+
+
 def test_suite():
     return unittest.TestSuite((unittest.makeSuite(XMLRPCResponseTests),))
 

Modified: Zope/trunk/lib/python/ZPublisher/xmlrpc.py
===================================================================
--- Zope/trunk/lib/python/ZPublisher/xmlrpc.py	2006-06-16 15:54:47 UTC (rev 68691)
+++ Zope/trunk/lib/python/ZPublisher/xmlrpc.py	2006-06-16 15:55:50 UTC (rev 68692)
@@ -25,7 +25,29 @@
 import xmlrpclib
 
 from zExceptions import Unauthorized
+from ZODB.POSException import ConflictError
 
+# Make DateTime.DateTime marshallable via XML-RPC
+# http://www.zope.org/Collectors/Zope/2109
+from DateTime import DateTime
+WRAPPERS = xmlrpclib.WRAPPERS + (DateTime,)
+
+def dump_instance(self, value, write):
+    # Check for special wrappers
+    if value.__class__ in WRAPPERS:
+        self.write = write
+        value.encode(self)
+        del self.write
+    else:
+        # Store instance attributes as a struct (really?).
+        # We want to avoid disclosing private attributes.
+        # Private attributes are by convention named with
+        # a leading underscore character.
+        value = dict([(k, v) for (k, v) in value.__dict__.items() if k[0] != '_'])
+        self.dump_struct(value, write)
+
+xmlrpclib.Marshaller.dispatch[types.InstanceType] = dump_instance
+
 def parse_input(data):
     """Parse input data and return a method path and argument tuple
 
@@ -100,16 +122,6 @@
             # Convert Fault object to XML-RPC response.
             body=xmlrpclib.dumps(body, methodresponse=1, allow_none=True)
         else:
-            if type(body) == types.InstanceType:
-                # Avoid disclosing private members. Private members are
-                # by convention named with a leading underscore char.
-                orig = body.__dict__
-                dict = {}
-                for key in orig.keys():
-                    if key[:1] != '_':
-                        dict[key] = orig[key]
-                body = dict
-
             # Marshall our body as an XML-RPC response. Strings will be sent
             # strings, integers as integers, etc. We do *not* convert
             # everything to a string first.
@@ -119,6 +131,8 @@
             try:
                 body = xmlrpclib.dumps(
                     (body,), methodresponse=1, allow_none=True)
+            except ConflictError:
+                raise
             except:
                 self.exception()
                 return
@@ -165,6 +179,8 @@
                 f=Fault(-1, 'Unexpected Zope exception: %s' % value)
             else:
                 f=Fault(-2, 'Unexpected Zope error value: %s' % value)
+        except ConflictError:
+            raise
         except:
             f=Fault(-3, "Unknown Zope fault type")
 



More information about the Zope-Checkins mailing list