Serialization of any python objects using a Factory Pattern
Project description
Serializall is meant to serialize any python objects. It incorporates python built-in objects serialization and can also serialize numpy arrays.
It is based on a Factory pattern allowing to register objects that can be serialized. To do so objects should inherit a MixIn base class exposing a mandatory interface with two methods namely serialize and deserialize implementing the serialization using builtins (see example).
It is currently used in projects such as PyMoDAQ and pyqtgraph.
Usage
The serialization is easy to perform using the SerializableFactory object and its two API methods:
get_apply_serializer
get_apply_deserializer
Example with a string object:
>>> from serializall.factory import SerializableFactory
>>> ser_factory = SerializableFactory()
>>> astring = 'a string to serialize'
>>> a_serialized_string = ser_factory.get_apply_serializer(astring)
>>> print(a_serialized_string)
b'\x00\x00\x00\x03str\x00\x00\x00\x15a string to serialize'
>>> print(ser_factory.get_apply_deserializer(a_serialized_string))
'a string to serialize'
From this example, one see (with good eyes) the mechanism used in the serialization. First the object type is serialized as a string passing first the length of the string on four bytes (here 3), then the binary string itself (here the str characters), followed by the length of the object on four bytes (here 15 characters) character of the object: a string to serialize and finally the object itself
Example with a list object:
List can contain any builtins or objects registered in the SerializableFactory
>>> from serializall.factory import SerializableFactory
>>> import numpy as np
>>> ser_factory = SerializableFactory()
>>> a_list = ['hjk', 23, 34.7, np.array([1, 2, 3])]
>>> a_serialized_list = ser_factory.get_apply_serializer(a_list)
>>> print(a_serialized_list)
b'\x00\x00\x00\x04list\x00\x00\x00\x04\x00\x00\x00\x03str\x00\x00\x00\x03hjk\x00\x00\x00\x03int\x00\x00\x00\x03<i8
\x00\x00\x00\x08\x17\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x05float\x00\x00\x00\x03<f8\x00\x00\x00\x08\x9a\x99\x99
\x99\x99YA@\x00\x00\x00\x07ndarray\x00\x00\x00\x03<i8\x00\x00\x00\x18\x00\x00\x00\x01\x00\x00\x00\x03\x01\x00\x00\x00
\x00\x00\x00\x00\x02\x00\x00\x00\x00\x00\x00\x00\x03\x00\x00\x00\x00\x00\x00\x00'
>>> print(ser_factory.get_apply_deserializer(a_serialized_list))
['hjk', 23, 34.7, array([1, 2, 3])]
From this example, one see (with good eyes) the mechanism used in the serialization. First the object type is serialized as a string passing first the length of the string on four bytes, then the binary string (here the list characters), followed by the elements of the list: a str (hjk characters), an int (serialized itself using numpy buffering), then a float and finally a ndarray. During the deserialization steps, the type of the next element to be deserialized will be inferred from the string characters and the right deserialization method fetch from the factory using the retrieved type as a key
By systemizing this approach: length of the type, type as a string, length of the serialized object itself and finally serialized object, the mechanism is stable and the factory will work seamlessly independently of the particular object serialization (handles in both serialize and deserialize methods (see below).
Implementation example:
Let’s say we have a custom object defined as below, a kind of DataClass only storing two attributes:
class MyObject():
def __init__(self, a: int, b: str):
self.a = a
self.b = b
The first thing to do is to force the SerializableBase interface using inheritance and to register the object in the factory using the decorator SerializableFactory.register_decorator on the class itself.
from serializall.factory import SerializableFactory, SerializableBase
@SerializableFactory.register_decorator()
class MyObject(SerializableBase):
def __init__(self, a: int, b: str):
super().__init()
self.a = a
self.b = b
@staticmethod
def serialize(param: 'MyObject') -> bytes:
""" Implement the MyObject type serialization """
pass
@staticmethod
def deserialize(bytes_str: bytes) -> tuple['MyObject', bytes]:
""" Implements deserialization into a MyObject object from bytes """
pass
As you can see the two methods are not completely symmetric as the deserialize method returns a tuple containing an instance ot the object and eventual remaining bytes. This is important in the case where your object has been, for instance, part of a list object. The remaining bytes will contains the eventual next objects of the list to deserialize.
It is then up to you to do the actual serialization/deserialization using the builtins methods already present in the factory. For instance:
from serializall.factory import SerializableFactory, SerializableBase
ser_factory = SerializableFactory()
class MyObject(SerializableBase):
def __init__(self, a: int, b: str):
super().__init()
self.a = a
self.b = b
@staticmethod
def serialize(param: 'MyObject') -> bytes:
""" Implement the MyObject type serialization """
bytes_string = b''
bytes_string += ser_factory.get_apply_serializer(param.a)
bytes_string += ser_factory.get_apply_serializer(param.b)
return bytes_string
@staticmethod
def deserialize(bytes_str: bytes) -> tuple['MyObject', bytes]:
""" Implements deserialization into a MyObject object from bytes """
a, remaining_bytes = ser_factory.get_apply_deserializer(bytes_str, only_object=False)
b, remaining_bytes = ser_factory.get_apply_deserializer(remaining_bytes, only_object=False)
return MyObject(a, b), remaining_bytes
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file serializall-0.0.0.tar.gz.
File metadata
- Download URL: serializall-0.0.0.tar.gz
- Upload date:
- Size: 9.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: python-httpx/0.28.1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
84ef7cdf6c8eda9731638de012a2a1329868ac58b43b57ac865100b292088344
|
|
| MD5 |
b5e8f063fffe23a366eef09eec9d1d4d
|
|
| BLAKE2b-256 |
15d859ef5a1c24b5c053ae925528524413b377a389177f0710e7fc11868f0e92
|
File details
Details for the file serializall-0.0.0-py3-none-any.whl.
File metadata
- Download URL: serializall-0.0.0-py3-none-any.whl
- Upload date:
- Size: 10.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: python-httpx/0.28.1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a94e74a91926b5fb32b0e3d5f416b1edac4cc11e043b168719e7a65c58aa1e9c
|
|
| MD5 |
e2c07c6bf775a1dca63808f4aa7b7e28
|
|
| BLAKE2b-256 |
b14030e7af0aeb34e596e2cadb5a2e011807f51966d7e4f933e14894e810c681
|