Skip to main content

Simple library for de/serializing binary data

Project description

binpi

binpi aims to provide a simple interface for serializing and deserializing binary file formats.

Usage:

import binpi

class Data:
    ...

class FileHeader:
    prop1 = binpi.Int()
    prop2 = binpi.Short()
    prop3 = binpi.Byte()
    is_compressed = binpi.Boolean()
    float_prop = binpi.Float()
    some_data = binpi.ByteArray(size="prop1")
    other_data = binpi.List(Data, size="prop3")
    sub_struct = binpi.WrapType(Data)
    children_count = binpi.Int()
    children = binpi.List(binpi.RecursiveType(), size="children_count")
    
# deserializing    
header_data = binpi.deserialize(FileHeader, binpi.FileReader("./some_path"), endianness=binpi.LITTLE_ENDIAN)

# modify
header_data.prop2 = 200

# serializing 
writer = binpi.serialize(header_data, binpi.FileWriter("./another_path"), endianness=binpi.LITTLE_ENDIAN)

For more complex examples, check ./examples/

How to install:

pip install binpi

Supported Types:

  • Int, UInt, Short, UShort, Byte, UByte, Float, Double
  • IntEnumType
  • List, String, ByteArray
  • Boolean
  • RecursiveType (for cases where the structure contains list of substructures of the same type, check the advanced_structure example)
  • WrapType (for subtypes, check the simple_image_archive_format example)
  • All the types above support LE/BE

Comparing with other (de)serializing libraries

  • pickle - should be used for completely different use-cases than binpi, which is just simple deserializing of python objects, without having to care about its structure.
  • struct - anything binpi does can be implemented using struct, but binpi provides simpler interface for defining data structure, for the cost of performance.
  • origami - origami might be a better choice for (de)serializing fixed size data, but it doesn't provide (de)serializing of dynamically sized data, out of the box.
  • bstruct - same as origami
  • construct - probably the most comparable library to binpi, has even more feature, but instead of binpi, the data structures and output is represented using dictionaries

Interface

Serializing

def serialize(
    value,           # value to be serialized
    writer: Writer,  # the output writer
    first=None,      # first field to serialize
    last=None        # last field to serialize
) -> None: ...

class Writer(Protocol):
    """ writer can be anything that implements method write_bytes """
    def write_bytes(self, data: bytes) -> None: ...

binpi contains FileWriter and BufferWriter

Deserializing

def deserialize(
    class_: type,    # type of the target object 
    reader: Reader,  # the input reader
    first=None,      # first field to serialize
    last=None        # last field to serialize
) -> None: ...

class Reader(Protocol):
    """ reader can be anything that implements method read_bytes """
    def read_bytes(self, n: int) -> bytes: ...

binpi contains FileReader and BufferReader

Extending with custom types

To create your own custom (de)serializable type, you have to just create a new child class of SerializableType that implements load_from_bytes and write_from_value

import typing, binpi, struct

class CustomDoubledInt(binpi.SerializableType):
    def load_from_bytes(self, deserializer: binpi.Deserializer, instance, *args, **kwargs):
        return struct.unpack("<i", deserializer.reader.read_bytes(4))[0] * 2

    def write_from_value(self, serializer: binpi.Serializer, value, parent_instance, *args, **kwargs):
        serializer.writer.write_bytes(struct.pack("<i", value // 2))

""" In case we want to have functional typechecking """
CustomDoubleInt: typing.Callable[..., int]

TODO:

  • Tests
  • Performance benchmarks

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

binpi-0.1.21.tar.gz (9.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

binpi-0.1.21-py3-none-any.whl (11.1 kB view details)

Uploaded Python 3

File details

Details for the file binpi-0.1.21.tar.gz.

File metadata

  • Download URL: binpi-0.1.21.tar.gz
  • Upload date:
  • Size: 9.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.9.23

File hashes

Hashes for binpi-0.1.21.tar.gz
Algorithm Hash digest
SHA256 8c5d225db609fc4ce284ebfbe6105b57f59ed6522503c5e0d368db42fa9be01b
MD5 33b9fa9e5a3a7e0ed65fdfb54f01bbc3
BLAKE2b-256 e9ccc61cf17925c6c24092f7b255d85bb381af85a0533d352df2d3ae3d648857

See more details on using hashes here.

File details

Details for the file binpi-0.1.21-py3-none-any.whl.

File metadata

  • Download URL: binpi-0.1.21-py3-none-any.whl
  • Upload date:
  • Size: 11.1 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.9.23

File hashes

Hashes for binpi-0.1.21-py3-none-any.whl
Algorithm Hash digest
SHA256 b36a71b14e71a7ebbcf095d7b818856ae1f16386c58c671c709659c67d861e50
MD5 13566f52588e3ee2468a1fb2300841a6
BLAKE2b-256 bcdd8bfaaa9b27ecb8557979cb5fd46798ba059fd8e580c0199d774862611465

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page