Skip to content

Optimizations and benchmark

apischema is (a lot) faster than its known alternatives, thanks to advanced optimizations.

benchmark chart benchmark chart

Note

Chart is truncated to a relative performance of 20x slower. Benchmark results are detailed in the results table.

Precomputed (de)serialization methods

apischema precomputes (de)serialization methods depending on the (de)serialized type (and other parameters); type annotations processing is done in the precomputation. Methods are then cached using functools.lru_cache, so deserialize and serialize don't recompute them every time.

Note

The cache is automatically reset when global settings are modified, because it impacts the generated methods.

However, if lru_cache is fast, using the methods directly is faster, so apischema provides apischema.deserialization_method and apischema.serialization_method. These functions share the same parameters than deserialize/serialize, except the data/object parameter to (de)serialize. Using the computed methods directly can increase performances by 10%.

from dataclasses import dataclass

from apischema import deserialization_method, serialization_method


@dataclass
class Foo:
    bar: int


deserialize_foo = deserialization_method(Foo)
serialize_foo = serialization_method(Foo)

assert deserialize_foo({"bar": 0}) == Foo(0)
assert serialize_foo(Foo(0)) == {"bar": 0}

Warning

Methods computed before settings modification will not be updated and use the old settings. Be careful to set your settings first.

Avoid unnecessary copies

As an example, when a list of integers is deserialized, json.load already return a list of integers. The loaded data can thus be "reused", and the deserialization just become a validation step. The same principle applies to serialization.

It's controlled by the settings apischema.settings.deserialization.no_copy/apischema.settings.serialization.no_copy, or no_copy parameter of deserialize/serialize methods. Default behavior is to avoid these unnecessary copies, i.e. no_copy=False.

from timeit import timeit

from apischema import deserialize

ints = list(range(100))

assert deserialize(list[int], ints, no_copy=True) is ints  # default
assert deserialize(list[int], ints, no_copy=False) is not ints

print(timeit("deserialize(list[int], ints, no_copy=True)", globals=globals()))
# 8.596703557006549
print(timeit("deserialize(list[int], ints, no_copy=False)", globals=globals()))
# 9.365363762015477

Serialization passthrough

JSON serialization libraries expect primitive data types (dict/list/str/etc.). A non-negligible part of objects to be serialized are primitive.

When type checking is disabled (this is default), objects annotated with primitive types doesn't need to be transformed or checked; apischema can simply "pass through" them, and it will result into an identity serialization method, just returning its argument.

Container types like list or dict are passed through only when the contained types are passed through too (and when no_copy=True)

from apischema import serialize

ints = list(range(5))
assert serialize(list[int], ints) is ints

Note

Enum subclasses which also inherit str/int are also passed through

Passthrough options

Some JSON serialization libraries natively support types like UUID or datetime, sometimes with a faster implementation than the apischema one — orjson, written in Rust, is a good example.

To take advantage of that, apischema provides apischema.PassThroughOptions class to specify which type should be passed through, whether they are supported natively by JSON libraries (or handled in a default fallback).

apischema.serialization_default can be used as default fallback in combination to PassThroughOptions. It has to be instantiated with the same kwargs parameters (aliaser, etc.) than serialization_method.

from collections.abc import Collection
from uuid import UUID, uuid4

from apischema import PassThroughOptions, serialization_method

uuids_method = serialization_method(
    Collection[UUID], pass_through=PassThroughOptions(collections=True, types={UUID})
)
uuids = [uuid4() for _ in range(5)]
assert uuids_method(uuids) is uuids

Important

Passthrough optimization is a lot diminished with check_type=False.

PassThroughOptions has the following parameters:

any — pass through Any

collections — pass through collections

Standard collections list, tuple and dict are natively handled by JSON libraries, but set, for example, isn't. Moreover, standard abstract collections like Collection or Mapping, which are used a lot, are not guaranteed to have their runtime type supported (having a set annotated with Collection for instance).

But, most of the time, collections runtime types are list/dict, so others can be handled in default fallback.

Note

Set-like type will not be passed through.

dataclasses - pass through dataclasses

Some JSON libraries, like orjson, support dataclasses natively. However, because apischema has a lot of specific features (aliasing, flatten fields, conditional skipping, fields ordering, etc.), only dataclasses with none of these features, and only passed through fields, will be passed through too.

enums — pass through enum.Enum subclasses

tuple — pass through tuple

Even if tuple is often supported by JSON serializers, if this options is not enabled, tuples will be serialized as lists. It also allows easier test writing for example.

Note

collections=True implies tuple=True;

types — pass through arbitrary types

Either a collection of types, or a predicate to determine if type has to be passed through.

Binary compilation using Cython

apischema use Cython in order to compile critical parts of the code, i.e. the (de)serialization methods.

However, apischema remains a pure Python library — it can work without binary modules. Cython source files (.pyx) are in fact generated from Python modules. It allows notably keeping the code simple, by adding switch-case optimization to replace dynamic dispatch, avoiding big chains of elif in Python code.

Note

Compilation is disabled when using PyPy, because it's even faster with the bare Python code. That's another interest of generating .pyx files: keeping Python source for PyPy.

Override dataclass constructors

Warning

This feature is still experimental and disabled by default. Test carefully its impact on your code before enable it in production.

Dataclass constructors calls is the slowest part of the deserialization, about 50% of its runtime! They are indeed pure Python functions and cannot be compiled.

In case of "normal" dataclass (no __slots__, __post_init__, or __init__/__new__/__setattr__ overriding), apischema can override the constructor with a compilable code.

This feature can be toggled on/off globally using apischema.settings.deserialization.override_dataclass_constructors

Discriminator

OpenAPI discriminator allows making union deserialization time more homogeneous.

from dataclasses import dataclass
from timeit import timeit
from typing import Annotated, Union

from apischema import deserialization_method, discriminator


@dataclass
class Cat:
    love_dog: bool = False


@dataclass
class Dog:
    love_cat: bool = False


Pet = Union[Cat, Dog]
DiscriminatedPet = Annotated[Pet, discriminator("type", {"dog": Dog})]

deserialize_union = deserialization_method(Union[Cat, Dog])
deserialize_discriminated = deserialization_method(
    Annotated[Union[Cat, Dog], discriminator("type")]
)
##### Without discrimininator
print(timeit('deserialize_union({"love_dog": False})', globals=globals()))
# Cat: 0.760085788
print(timeit('deserialize_union({"love_cat": False})', globals=globals()))
# Dog: 3.078876515 ≈ x4
##### With discriminator
print(timeit('deserialize_discriminated({"type": "Cat"})', globals=globals()))
# Cat: 1.244204702
print(timeit('deserialize_discriminated({"type": "Dog"})', globals=globals()))
# Dog: 1.234058598 ≈ same

Note

As you can notice in the example, discriminator brings its own additional cost, but it's completely worth it.

Benchmark

Benchmark code is located benchmark directory or apischema repository. Performances are measured on two datasets: a simple, a more complex one.

Benchmark is run by Github Actions workflow on ubuntu-latest with Python 3.10.

Results are given relatively to the fastest library, i.e. apischema; simple and complex results are detailed in the table, displayed result is the mean of both.

Relative execution time (lower is better)

library version deserialization serialization
apischema 0.17.5 / /
mashumaro 2.11 x2.1 (2.5/1.7) x2.1 (2.2/1.9)
cattrs 1.10.0 x3.0 (4.0/2.0) x3.8 (6.0/1.7)
typical 2.8.0 x6.4 (6.9/5.9) x8.1 (11.7/4.6)
pydantic 1.9.0 x9.0 (10.4/7.6) x25.5 (34.0/17.1)
marshmallow 3.14.1 x24.3 (28.3/20.4) x22.7 (29.0/16.3)
typedload 2.15 x11.7 (15.4/8.0) x59.0 (68.2/49.7)

Note

Benchmark use binary optimization, but even running as a pure Python library, apischema still performs better than almost all of the competitors.