[837] | 1 | from __future__ import annotations
|
---|
[489] | 2 | from abc import ABC
|
---|
| 3 | from datetime import datetime
|
---|
| 4 | from decimal import Decimal
|
---|
| 5 | import json
|
---|
| 6 | import re
|
---|
| 7 | import sys, traceback
|
---|
[1283] | 8 | from typing import Dict, List, Set, Any, Union, Optional, Collection, Type
|
---|
[489] | 9 | import unittest
|
---|
| 10 | from uuid import uuid4, UUID
|
---|
| 11 |
|
---|
| 12 | from pyson.Deserializer import Deserializer
|
---|
[570] | 13 | from pyson.Serializer import Serializer
|
---|
| 14 | from pyson.JsonSerialize import JsonSerialize
|
---|
[489] | 15 | from pyson.JsonDeserialize import JsonDeserialize
|
---|
| 16 | from pyson.JsonGetter import JsonGetter
|
---|
| 17 | from pyson.JsonSubTypes import JsonSubTypes
|
---|
| 18 | from pyson.JsonTypeInfo import Id, As
|
---|
| 19 | from pyson.JsonTypeInfo import JsonTypeInfo
|
---|
| 20 | from pyson.JsonValue import JsonValue
|
---|
| 21 | from pyson.ObjectMapper import ObjectMapper
|
---|
[568] | 22 | from pickle import NONE
|
---|
[571] | 23 | from test.MyDeserializer import ValueDeserializer2
|
---|
[938] | 24 | from cgitb import reset
|
---|
[489] | 25 |
|
---|
[971] | 26 | class 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] | 43 | class 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 "$"
|
---|
| 52 | class 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] | 58 | class 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] | 73 | class 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] | 89 | class 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)
|
---|
| 103 | class Simple3 (Simple):
|
---|
| 104 | pass
|
---|
[967] | 105 |
|
---|
[489] | 106 |
|
---|
[570] | 107 | class MyList:
|
---|
| 108 | def __init__(self, data: List[Simple] ):
|
---|
| 109 | self.data=data
|
---|
| 110 | def getData(self)->List[Simple]:
|
---|
| 111 | return self.data
|
---|
| 112 |
|
---|
| 113 | class 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 |
|
---|
| 125 | class 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] | 137 | class 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)
|
---|
| 151 | class 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 |
|
---|
| 165 | class A(Root):
|
---|
[835] | 166 | pass
|
---|
[827] | 167 |
|
---|
[836] | 168 | class B(Root):
|
---|
[835] | 169 | pass
|
---|
| 170 |
|
---|
[938] | 171 | class 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] | 184 | class 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)
|
---|
| 192 | class 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] | 202 | class 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)
|
---|
| 213 | class RootClass(SuperSimple):
|
---|
| 214 | pass
|
---|
| 215 |
|
---|
| 216 | @JsonSubTypes(['test.DeserializerTest.SubSubClass'])
|
---|
| 217 | class SubClass(RootClass):
|
---|
| 218 | pass
|
---|
| 219 |
|
---|
| 220 | # INDIRECTLY inherits from RootClass.
|
---|
[1360] | 221 | class SubSubClass(SubClass):
|
---|
[1358] | 222 | pass
|
---|
| 223 |
|
---|
[1364] | 224 |
|
---|
| 225 | @JsonTypeInfo(use=Id.NAME, include=As.WRAPPER_OBJECT)
|
---|
| 226 | class 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] | 239 | class 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 |
|
---|
[1492] | 385 | def testDeserializeStandardDictDecimal(self):
|
---|
| 386 | val:Dict = self.pyson.parse(json.loads("{\"a\":0.3,\"b\":{\"x\":3},\"c\":[1,2,3]}", parse_float=lambda x:Decimal(x)),Dict)
|
---|
| 387 |
|
---|
[971] | 388 |
|
---|
[1283] | 389 | def testDeserializeClass(self):
|
---|
[1285] | 390 | clazz:Type[Root] = self.pyson.parse("test.DeserializerTest$C", Type[Root])
|
---|
[1283] | 391 | self.assertEquals(C, clazz)
|
---|
[971] | 392 |
|
---|
[1283] | 393 | def testSerializeClass(self):
|
---|
| 394 | js = self.pyson.toJson(C)
|
---|
| 395 | print("serialized Class<C>:"+json.dumps(js))
|
---|
[1285] | 396 | self.assertEquals("test.DeserializerTest$C", js)
|
---|
[1283] | 397 |
|
---|
[971] | 398 |
|
---|
[1284] | 399 | def testDeserializeListOfClass(self):
|
---|
[1285] | 400 | clazzlist:List[Type[Root]] = self.pyson.parse(["test.DeserializerTest$C","test.DeserializerTest$A"], List[Type[Root]])
|
---|
[1284] | 401 | self.assertEquals([C,A], clazzlist)
|
---|
[971] | 402 |
|
---|
[1358] | 403 |
|
---|
| 404 | def testSerializeSubSubClass(self):
|
---|
| 405 | js = self.pyson.toJson(SubSubClass(22))
|
---|
| 406 | print("serialized SuperSimple:"+json.dumps(js))
|
---|
| 407 | self.assertEquals({'SubSubClass': {'a': 22}}, js)
|
---|
[1284] | 408 |
|
---|
[1358] | 409 |
|
---|
| 410 | def testDeserializeSubSubClass(self):
|
---|
| 411 | js=self.pyson.parse({'SubSubClass': {'a': 22}}, RootClass)
|
---|
| 412 | self.assertEquals(SubSubClass(22), js)
|
---|
[1359] | 413 |
|
---|
[1358] | 414 |
|
---|
[1364] | 415 | def testSerializeBothTypeInfoAndValue(self):
|
---|
| 416 | js = self.pyson.toJson(BothTypeInfoAndValue("ok"))
|
---|
| 417 | print("serialized BothTypeInfoAndValue:"+json.dumps(js))
|
---|
| 418 | self.assertEquals({'BothTypeInfoAndValue': "ok"}, js)
|
---|
| 419 |
|
---|
| 420 | def testDeserializeBothTypeInfoAndValue(self):
|
---|
| 421 | js = self.pyson.parse({'BothTypeInfoAndValue': "ok"}, BothTypeInfoAndValue)
|
---|
| 422 | self.assertEquals(BothTypeInfoAndValue("ok"), js)
|
---|
| 423 |
|
---|
| 424 |
|
---|