source: pyson/test/DeserializerTest.py@ 1428

Last change on this file since 1428 was 1365, checked in by wouter, 6 weeks ago

#422 fixed order of handling @JsonValue and @JsonTypeInfo in parse

File size: 13.0 KB
RevLine 
[837]1from __future__ import annotations
[489]2from abc import ABC
3from datetime import datetime
4from decimal import Decimal
5import json
6import re
7import sys, traceback
[1283]8from typing import Dict, List, Set, Any, Union, Optional, Collection, Type
[489]9import unittest
10from uuid import uuid4, UUID
11
12from pyson.Deserializer import Deserializer
[570]13from pyson.Serializer import Serializer
14from pyson.JsonSerialize import JsonSerialize
[489]15from pyson.JsonDeserialize import JsonDeserialize
16from pyson.JsonGetter import JsonGetter
17from pyson.JsonSubTypes import JsonSubTypes
18from pyson.JsonTypeInfo import Id, As
19from pyson.JsonTypeInfo import JsonTypeInfo
20from pyson.JsonValue import JsonValue
21from pyson.ObjectMapper import ObjectMapper
[568]22from pickle import NONE
[571]23from test.MyDeserializer import ValueDeserializer2
[938]24from cgitb import reset
[489]25
[971]26class SuperSimple:
27 def __init__(self, a:Optional[int]):
28 self.a=a
29 def __eq__(self, other):
30 return isinstance(other, self.__class__) and \
31 self.a==other.a
32 def getA(self):
33 return self.a
34 def __repr__(self):
35 return "SuperSimple::"+str(self.a)
36 def __hash__(self):
37 return hash(self.a)
38
39
40
[568]41# deserializer DROPS FIRST CHAR from string and assumes rest is an int.
42# Then returns Simple(int)
[490]43class ValueDeserializer(Deserializer):
44 def __hash__(self):
45 return hash(self.geta())
46 def deserialize(self, data:object, clas: object)-> object:
47 if type(data)!=str:
48 raise ValueError("Expected str starting with '$', got "+str(data))
49 return Simple(int(data[1:]))
[489]50
[570]51# serializes Simple object, just prefixing its value (as string) with "$"
52class ValueSerializer(Serializer):
53 def serialize(self, obj:object)-> object:
54 if not isinstance(obj, Simple):
55 raise ValueError("Expected Dimple object")
56 return "$" + str(obj.geta())
[490]57
[570]58class Basic:
59 def __init__(self, v:float):
60 self._v=v
61 def __eq__(self, other):
62 return isinstance(other, self.__class__) and \
63 self._v==other._v
64 def getV(self):
65 return self._v
66 def __repr__(self):
67 return "Basic:"+str(self._v)
68 def __hash__(self):
69 return hash(self._v)
70
[571]71@JsonDeserialize(ValueDeserializer)
[572]72@JsonSerialize(ValueSerializer)
[489]73class Simple:
74 def __init__(self, a:int):
75 self._a=a
76 def geta(self)->int:
77 return self._a
78 def __eq__(self, other):
79 return isinstance(other, self.__class__) and \
80 self._a==other._a
[570]81 def __repr__(self):
82 return "Simple:"+str(self._a)
83 def __hash__(self):
84 return hash(self._a)
[489]85
86
87
[571]88@JsonDeserialize(ValueDeserializer2)
[490]89class Simple2:
90 def __init__(self, a:int):
91 self._a=a
92 def geta(self)->int:
93 return self._a
94 def __eq__(self, other):
95 return isinstance(other, self.__class__) and \
96 self._a==other._a
[570]97 def __repr__(self):
[490]98 return self._name+","+str(self._a)
[568]99
100#None cancels out the existing deserializer.
101# parsing with str should now fail.
102@JsonDeserialize(None)
103class Simple3 (Simple):
104 pass
[967]105
[489]106
[570]107class MyList:
108 def __init__(self, data: List[Simple] ):
109 self.data=data
110 def getData(self)->List[Simple]:
111 return self.data
112
113class BasicDict:
114 def __init__(self, data: Dict[Basic,Basic] ):
115 self.data=data
116 def getData(self)->Dict[Basic,Basic]:
117 return self.data
118 def __repr__(self):
119 return "BasicDict:"+str(self.data)
[568]120 def __eq__(self, other):
121 return isinstance(other, self.__class__) and \
[570]122 self.data==other.data
[490]123
[570]124
125class SimpleDict:
126 def __init__(self, data: Dict[Simple,Simple] ):
127 self.data=data
128 def getData(self)->Dict[Simple,Simple]:
129 return self.data
130 def __eq__(self, other):
131 return isinstance(other, self.__class__) and \
132 self.data==other.data
133 def __repr__(self):
134 return "SimpleDict:"+str(self.data)
[490]135
[570]136
[827]137class MyOptionalString:
138 def __init__(self, s:Optional[str]):
[829]139 self.data:Optional[str] = s
140 def getData(self)->Optional[str]:
[827]141 return self.data
142 def __eq__(self, other):
143 return isinstance(other, self.__class__) and \
144 self.data==other.data
145 def __repr__(self):
[829]146 return "MyOptionalString:"+str(self.data)
[835]147
[836]148
[938]149@JsonSubTypes(['test.DeserializerTest.A','test.DeserializerTest.B','test.DeserializerTest.C'])
[836]150@JsonTypeInfo(use=Id.NAME, include=As.WRAPPER_OBJECT)
151class Root:
152 def __init__(self, data: int ):
153 self.data:int=data
154 def getData(self)->int:
155 return self.data
156 def __eq__(self, other):
157 return isinstance(other, self.__class__) and \
158 self.data==other.data
159 def __repr__(self):
160 return type(self).__name__ + "/" + str(self.data)
161 def __hash__(self):
162 return 0
163
164
165class A(Root):
[835]166 pass
[827]167
[836]168class B(Root):
[835]169 pass
170
[938]171class C(Root):
172 def __init__(self, data:int, name: str):
173 super().__init__(data)
174 self.name = name
175 def __eq__(self, other):
176 return isinstance(other, self.__class__) and \
177 self.data==other.data and self.name==other.name
178 def __repr__(self):
179 return super().__repr__()+","+self.name
[836]180
181
[938]182
183
[967]184class BaseValueDeserializer (Deserializer):
185 def deserialize(self, data:object, clas: object) -> 'BaseValue':
186 if isinstance(data,str):
187 return BaseDiscreteValue(data)
188 raise ValueError("Expected number or double quoted string but found " + str(data)
189 + " of type " + str(type(data)))
190
191@JsonDeserialize(using=BaseValueDeserializer)
192class BaseValue(ABC):
193 def __init__(self, value):
194 self._value = value;
195
196 @JsonValue()
197 def getValue(self) -> str:
198 return self._value;
199
[968]200#disable parent deserializer. This is essential #331
201#@JsonDeserialize(using=None)
[967]202class BaseDiscreteValue(BaseValue):
203 def __init__(self, value:str):
204 self.__value=value
[968]205 @JsonValue()
206 def getValue(self)->str:
207 return self.__value
[967]208
209
[1358]210
211@JsonSubTypes(['test.DeserializerTest.SubClass'])
212@JsonTypeInfo(use=Id.NAME, include=As.WRAPPER_OBJECT)
213class RootClass(SuperSimple):
214 pass
215
216@JsonSubTypes(['test.DeserializerTest.SubSubClass'])
217class SubClass(RootClass):
218 pass
219
220# INDIRECTLY inherits from RootClass.
[1360]221class SubSubClass(SubClass):
[1358]222 pass
223
[1364]224
225@JsonTypeInfo(use=Id.NAME, include=As.WRAPPER_OBJECT)
226class BothTypeInfoAndValue:
227 def __init__(self, val:str):
228 self.__value=val
229 @JsonValue()
230 def getVal(self)->str:
231 return self.__value
[1365]232 def __eq__(self, other):
233 return isinstance(other, self.__class__) and \
234 self.__value==other.__value
235 def __hash__(self):
236 return hash(self.__value)
237
238
[489]239class DeserializerTest(unittest.TestCase):
240 '''
241 Test a lot of back-and-forth cases.
242 FIXME Can we make this a parameterized test?
243 '''
244 pyson=ObjectMapper()
[836]245 mixedList:List[Root] = [A(1), B(1)]
246 mixedlistjson:List = [{'A': {'data': 1}}, {'B': {'data': 1}}]
[489]247
[836]248 mixedDict:Dict[Root,int] = {A(1):1, B(1):2}
249 mixedDictJson:dict = {'{"A": {"data": 1}}': 1, '{"B": {"data": 1}}': 2}
[938]250 MyC:C = C(3, "MyC")
[836]251
[489]252 def testDeserialize(self):
253 objson= "$12"
[490]254 self.assertEqual(Simple(12), self.pyson.parse(objson, Simple))
[489]255
[490]256 def testExternalDeserialize2(self):
257 objson= "$13"
258 self.assertEqual(13, self.pyson.parse(objson, Simple2))
[568]259
260
261 def testExternalDeserialize3(self):
262 objson= "$13"
[569]263 self.assertRaises(ValueError, lambda:self.pyson.parse(objson, Simple3))
[568]264
[570]265 def testDeserializeMyList(self):
266 print(self.pyson.toJson(MyList([12,13])))
267
268 # the json we provide is NOT the proper json but a STRING.
269 # This triggers a fallback mechanism that tries to parse the string as json.
270 objson={"data": ["$12", "$13"]}
271 res = self.pyson.parse(objson, MyList)
272 print(res)
273 self.assertEqual([Simple(12),Simple(13)], res.data)
274
[748]275 def testDeserializeCollection(self):
276 objson=[1,2,3]
277 # this checks that pyson can parse a list as Collection.
278 # The result is a simple list like in java/jackson.
[749]279 # Collection without typing info means Collectino[Any]
[748]280 res = self.pyson.parse(objson, Collection)
281 print(res)
282 self.assertEqual([1,2,3], res)
[570]283
[748]284
[570]285 def testSerializeBasicDict(self):
286 '''
287 Basic object keys. Special (de)serializer should kick in #190
288 '''
289 d= BasicDict( { Basic(1.):Basic(2.), Basic(3.): Basic(4.) } )
290 objson={"data": {"{\"v\": 1.0}": {"v": 2.0}, "{\"v\": 3.0}": {"v": 4.0}}}
291 dump=json.dumps(self.pyson.toJson(d));
292 print(dump)
293 # self.assertEqual(objson, dump);
294 res=self.pyson.parse(objson, BasicDict)
295 print("res="+str(res))
296 self.assertEqual(d, res)
297
298
299 def testSerializeSimpleDictCustomSerializer(self):
300 d= SimpleDict( { Simple(1):Simple(2), Simple(3): Simple(4) } )
301
302 # The keys need extra quotes, they are deserialied by
303 # our special deserializer that parses the string as json,
304 # and json requires double quotes around its strings.
305 objson = {"data": {'"$1"': "$2", '"$3"': "$4"}}
306
307 obj=self.pyson.toJson(d)
308 print(json.dumps(obj))
309 self.assertEqual(objson, obj)
310
311 res = self.pyson.parse(objson, SimpleDict)
312 print("res="+str(res))
313 self.assertEqual(d, res)
[827]314
315 def testDeserializeOptString(self):
[828]316 objson:dict={'s':None}
[827]317 res = self.pyson.parse(objson, MyOptionalString)
318 print(res)
[829]319 self.assertEqual(MyOptionalString(None), res)
[570]320
[828]321 objson={'s':"something"}
[827]322 res = self.pyson.parse(objson, MyOptionalString)
323 print(res)
[829]324 self.assertEqual(MyOptionalString("something"), res)
[827]325
[835]326 def testSerializeMixedList(self):
327 # see #296
[836]328 res=self.pyson.toJson(self.mixedList)
329 print(res)
330 self.assertEqual(self.mixedlistjson, res)
[835]331
[836]332 def testDeserializeMixedList(self):
333 # see #296
334 res=self.pyson.parse(self.mixedlistjson, List[Root])
335 print(res)
336 self.assertEqual(self.mixedList,res)
337
[837]338 def testDeserializeBadList(self):
339 # see #298. This SHOULD fail because B is not subtype of A
340 self.assertRaises(ValueError, lambda:self.pyson.parse(self.mixedlistjson, List[A]))
341
[836]342
343 def testSerializeMixedDict(self):
344 # see #296
345 res=self.pyson.toJson(self.mixedDict)
346 print(res)
347 self.assertEqual(self.mixedDictJson, res)
348
349 def testDeserializeMixedDict(self):
350 # see #296
351 print("testDeserializeMixedDict")
352 res=self.pyson.parse(self.mixedDictJson, Dict[Root,int])
353 print(res)
354 self.assertEqual(self.mixedDict,res)
355
[938]356 def testDeserializeC(self):
357 # see #298. Sub-class should not expect
358 res=self.pyson.parse({'C':{'data':3, 'name':'MyC' }}, Root)
359 print(reset)
360 self.assertEqual(self.MyC, res)
361 # even if you serialize as C, you still need the 'C' wrapper
362 res=self.pyson.parse({'C':{'data':3, 'name':'MyC' }}, C)
363 self.assertEqual(self.MyC, res)
364
[967]365 def testDeserializeBaseValue(self):
366 # see #331
367 res=self.pyson.parse({'yes': 1, 'no': 0}, Dict[BaseDiscreteValue, int])
368 print(res)
369
[971]370
371 def testDeserializeSuperSimple(self):
372 res=self.pyson.parse({'a':1 } , SuperSimple)
373 print(res)
374 self.assertEqual(SuperSimple(1), res)
375
376
377 def testDeserializeMissingValue(self):
378 res=self.pyson.parse({ } , SuperSimple) # missing value for a
379 print(res)
380 self.assertEqual(SuperSimple(None), res)
381
[1214]382 def testDeserializeStandardDict(self):
383 val:Dict = self.pyson.parse(json.loads("{\"a\":0.3,\"b\":{\"x\":3},\"c\":[1,2,3]}"),Dict)
[971]384
385
[1283]386 def testDeserializeClass(self):
[1285]387 clazz:Type[Root] = self.pyson.parse("test.DeserializerTest$C", Type[Root])
[1283]388 self.assertEquals(C, clazz)
[971]389
[1283]390 def testSerializeClass(self):
391 js = self.pyson.toJson(C)
392 print("serialized Class<C>:"+json.dumps(js))
[1285]393 self.assertEquals("test.DeserializerTest$C", js)
[1283]394
[971]395
[1284]396 def testDeserializeListOfClass(self):
[1285]397 clazzlist:List[Type[Root]] = self.pyson.parse(["test.DeserializerTest$C","test.DeserializerTest$A"], List[Type[Root]])
[1284]398 self.assertEquals([C,A], clazzlist)
[971]399
[1358]400
401 def testSerializeSubSubClass(self):
402 js = self.pyson.toJson(SubSubClass(22))
403 print("serialized SuperSimple:"+json.dumps(js))
404 self.assertEquals({'SubSubClass': {'a': 22}}, js)
[1284]405
[1358]406
407 def testDeserializeSubSubClass(self):
408 js=self.pyson.parse({'SubSubClass': {'a': 22}}, RootClass)
409 self.assertEquals(SubSubClass(22), js)
[1359]410
[1358]411
[1364]412 def testSerializeBothTypeInfoAndValue(self):
413 js = self.pyson.toJson(BothTypeInfoAndValue("ok"))
414 print("serialized BothTypeInfoAndValue:"+json.dumps(js))
415 self.assertEquals({'BothTypeInfoAndValue': "ok"}, js)
416
417 def testDeserializeBothTypeInfoAndValue(self):
418 js = self.pyson.parse({'BothTypeInfoAndValue': "ok"}, BothTypeInfoAndValue)
419 self.assertEquals(BothTypeInfoAndValue("ok"), js)
420
421
Note: See TracBrowser for help on using the repository browser.