Skip to main content

Apache Fory™ is a blazingly fast multi-language serialization framework powered by jit and zero-copy

Project description

Apache Fory™ Python

Build Status PyPI Python Versions License Slack Channel X

Apache Fory™ is a blazing fast multi-language serialization framework powered by JIT compilation and zero-copy techniques, providing up to ultra-fast performance while maintaining ease of use and safety.

pyfory provides the Python implementation of Apache Fory™, offering both high-performance object serialization and advanced row-format capabilities for data processing tasks.

Key Features

Flexible Serialization Modes

  • Xlang mode: Default cross-language wire format with compatible schema evolution
  • Python native mode: Same-language mode and drop-in replacement for pickle/cloudpickle
  • Row Format: Zero-copy row format for analytics workloads

Versatile Serialization Features

  • Shared/circular reference support for complex object graphs in both Python native and xlang modes
  • Polymorphism support for customized types with automatic type dispatching
  • Schema evolution support for backward/forward compatibility when using dataclasses in xlang mode
  • Out-of-band buffer support for zero-copy serialization of large data structures like NumPy arrays and Pandas DataFrames, compatible with pickle protocol 5
  • Reduced-precision xlang types use reserved pyfory.Float16 and pyfory.BFloat16 annotations and native Python float values; dense array payloads use public wrappers such as Float16Array and BFloat16Array

Blazing Fast Performance

  • Extremely fast performance compared to other serialization frameworks
  • Runtime code generation and Cython-accelerated core implementation for optimal performance

Compact Data Size

  • Compact object graph protocol with minimal space overhead—up to 3× size reduction compared to pickle/cloudpickle
  • Meta packing and sharing to minimize type forward/backward compatibility space overhead

Security & Safety

  • Strict mode prevents deserialization of untrusted types by type registration and checks.
  • Reference tracking for handling circular references safely

Installation

Basic Installation

Install pyfory using pip:

pip install pyfory

Optional Dependencies

# Install with row format support (requires Apache Arrow)
pip install pyfory[format]

# Install from source for development
git clone https://github.com/apache/fory.git
cd fory/python
pip install -e ".[dev,format]"

Requirements

  • Python: 3.8 or higher
  • OS: Linux, macOS, Windows

Python Native Serialization

pyfory provides a Python native mode for Python-only payloads. It is optimized for Python's type system and offers the same object surface as pickle/cloudpickle, but with significantly better performance, smaller data size, and enhanced security features.

The binary protocol and API are similar to Fory's xlang mode, but Python native mode can serialize any Python object—including global functions, local functions, lambdas, local classes and types with customized serialization using __getstate__/__reduce__/__reduce_ex__, which are not allowed in xlang mode.

To use Python native mode, create Fory with xlang=False. Use this mode when replacing pickle or cloudpickle for pure Python applications:

import pyfory
fory = pyfory.Fory(xlang=False, ref=False, strict=True)

Xlang Object Serialization

Basic Object Serialization

Serialize and deserialize Python objects with a simple API. This example shows serializing a dictionary with mixed types:

import pyfory

# Create an xlang Fory instance.
fory = pyfory.Fory(xlang=True)

# Serialize xlang-compatible values
data = fory.dumps({"name": "Alice", "age": 30, "scores": [95, 87, 92]})

# Deserialize back to Python object
obj = fory.loads(data)
print(obj)  # {'name': 'Alice', 'age': 30, 'scores': [95, 87, 92]}

Note: dumps()/loads() are aliases for serialize()/deserialize(). Both APIs are identical, use whichever feels more intuitive.

Custom Class Serialization

Fory automatically handles dataclasses and custom types. Register your class once, then serialize instances seamlessly:

import pyfory
from dataclasses import dataclass
from typing import List, Dict

@dataclass
class Person:
    name: str
    age: pyfory.Int32
    scores: List[pyfory.Int32]
    metadata: Dict[str, str]

fory = pyfory.Fory(xlang=True, ref=True)
fory.register(Person, typename="example.Person")
person = Person("Bob", 25, [88, 92, 85], {"team": "engineering"})
data = fory.serialize(person)
result = fory.deserialize(data)
print(result)  # Person(name='Bob', age=25, ...)

Drop-in Replacement for Pickle/Cloudpickle

pyfory can serialize any Python object with the following configuration:

  • For circular references: Set ref=True to enable reference tracking
  • For functions/classes: Set strict=False to allow deserialization of dynamic types

Security Warning: When strict=False, Fory will deserialize arbitrary types, which can pose security risks if data comes from untrusted sources. Only use strict=False in controlled environments where you trust the data source completely. If you do need to use strict=False, please configure a DeserializationPolicy when creating fory using policy=your_policy to controlling deserialization behavior.

Common Usage

Serialize common Python objects including dicts, lists, and custom classes without any registration:

import pyfory

# Create Fory instance
fory = pyfory.Fory(xlang=False, ref=True, strict=False)

# serialize common Python objects
data = fory.dumps({"name": "Alice", "age": 30, "scores": [95, 87, 92]})
print(fory.loads(data))

# serialize custom objects
from dataclasses import dataclass

@dataclass
class Person:
    name: str
    age: int

person = Person("Bob", 25)
data = fory.dumps(person)
print(fory.loads(data))  # Person(name='Bob', age=25)

Serialize Global Functions

Capture and get functions defined at module level. Fory deserialize and return same function object:

import pyfory

# Create Fory instance
fory = pyfory.Fory(xlang=False, ref=True, strict=False)

# serialize global functions
def my_global_function(x):
    return 10 * x

data = fory.dumps(my_global_function)
print(fory.loads(data)(10))  # 100

Serialize Local Functions/Lambdas

Serialize functions with closures and lambda expressions. Fory captures the closure variables automatically:

import pyfory

# Create Fory instance
fory = pyfory.Fory(xlang=False, ref=True, strict=False)

# serialize local functions with closures
def my_function():
    local_var = 10
    def local_func(x):
        return x * local_var
    return local_func

data = fory.dumps(my_function())
print(fory.loads(data)(10))  # 100

# serialize lambdas
data = fory.dumps(lambda x: 10 * x)
print(fory.loads(data)(10))  # 100

Serialize Global Classes/Methods

Serialize class objects, instance methods, class methods, and static methods. All method types are supported:

from dataclasses import dataclass
import pyfory
fory = pyfory.Fory(xlang=False, ref=True, strict=False)

# serialize global class
@dataclass
class Person:
    name: str
    age: int

    def f(self, x):
        return self.age * x

    @classmethod
    def g(cls, x):
        return 10 * x

    @staticmethod
    def h(x):
        return 10 * x

print(fory.loads(fory.dumps(Person))("Bob", 25))  # Person(name='Bob', age=25)
# serialize global class instance method
print(fory.loads(fory.dumps(Person("Bob", 20).f))(10))  # 200
# serialize global class class method
print(fory.loads(fory.dumps(Person.g))(10))  # 100
# serialize global class static method
print(fory.loads(fory.dumps(Person.h))(10))  # 100

Serialize Local Classes/Methods

Serialize classes defined inside functions along with their methods. Useful for dynamic class creation:

from dataclasses import dataclass
import pyfory
fory = pyfory.Fory(xlang=False, ref=True, strict=False)

def create_local_class():
    class LocalClass:
        def f(self, x):
            return 10 * x

        @classmethod
        def g(cls, x):
            return 10 * x

        @staticmethod
        def h(x):
            return 10 * x
    return LocalClass

# serialize local class
data = fory.dumps(create_local_class())
print(fory.loads(data)().f(10))  # 100

# serialize local class instance method
data = fory.dumps(create_local_class()().f)
print(fory.loads(data)(10))  # 100

# serialize local class method
data = fory.dumps(create_local_class().g)
print(fory.loads(data)(10))  # 100

# serialize local class static method
data = fory.dumps(create_local_class().h)
print(fory.loads(data)(10))  # 100

Out-of-Band Buffer Serialization

Fory supports pickle5-compatible out-of-band buffer serialization for efficient zero-copy handling of large data structures. This is particularly useful for NumPy arrays, Pandas DataFrames, and other objects with large memory footprints.

Out-of-band serialization separates metadata from the actual data buffers, allowing for:

  • Zero-copy transfers when sending data over networks or IPC using memoryview
  • Improved performance for large datasets
  • Pickle5 compatibility using pickle.PickleBuffer
  • Flexible stream support - write to any writable object (files, BytesIO, sockets, etc.)

Basic Out-of-Band Serialization

import pyfory
import numpy as np

fory = pyfory.Fory(xlang=False, ref=False, strict=False)

# Large numpy array
array = np.arange(10000, dtype=np.float64)

# Serialize with out-of-band buffers
buffer_objects = []
serialized_data = fory.serialize(array, buffer_callback=buffer_objects.append)

# Convert buffer objects to memoryview for zero-copy transmission
# For contiguous buffers (bytes, numpy arrays), this is zero-copy
# For non-contiguous data, a copy may be created to ensure contiguity
buffers = [obj.getbuffer() for obj in buffer_objects]

# Deserialize with out-of-band buffers (accepts memoryview, bytes, or Buffer)
deserialized_array = fory.deserialize(serialized_data, buffers=buffers)

assert np.array_equal(array, deserialized_array)

Out-of-Band with Pandas DataFrames

import pyfory
import pandas as pd
import numpy as np

fory = pyfory.Fory(xlang=False, ref=False, strict=False)

# Create a DataFrame with numeric columns
df = pd.DataFrame({
    'a': np.arange(1000, dtype=np.float64),
    'b': np.arange(1000, dtype=np.int64),
    'c': ['text'] * 1000
})

# Serialize with out-of-band buffers
buffer_objects = []
serialized_data = fory.serialize(df, buffer_callback=buffer_objects.append)
buffers = [obj.getbuffer() for obj in buffer_objects]

# Deserialize
deserialized_df = fory.deserialize(serialized_data, buffers=buffers)

assert df.equals(deserialized_df)

Selective Out-of-Band Serialization

You can control which buffers go out-of-band by providing a callback that returns True to keep data in-band or False (and appending to a list) to send it out-of-band:

import pyfory
import numpy as np

fory = pyfory.Fory(xlang=False, ref=True, strict=False)

arr1 = np.arange(1000, dtype=np.float64)
arr2 = np.arange(2000, dtype=np.float64)
data = [arr1, arr2]

buffer_objects = []
counter = 0

def selective_callback(buffer_object):
    global counter
    counter += 1
    # Only send even-numbered buffers out-of-band
    if counter % 2 == 0:
        buffer_objects.append(buffer_object)
        return False  # Out-of-band
    return True  # In-band

serialized = fory.serialize(data, buffer_callback=selective_callback)
buffers = [obj.getbuffer() for obj in buffer_objects]
deserialized = fory.deserialize(serialized, buffers=buffers)

Pickle5 Compatibility

Fory's out-of-band serialization is fully compatible with pickle protocol 5. When objects implement __reduce_ex__(protocol), Fory automatically uses protocol 5 to enable pickle.PickleBuffer support:

import pyfory
import pickle

fory = pyfory.Fory(xlang=False, ref=False, strict=False)

# PickleBuffer objects are automatically supported
data = b"Large binary data"
pickle_buffer = pickle.PickleBuffer(data)

# Serialize with buffer callback for out-of-band handling
buffer_objects = []
serialized = fory.serialize(pickle_buffer, buffer_callback=buffer_objects.append)
buffers = [obj.getbuffer() for obj in buffer_objects]

# Deserialize with buffers
deserialized = fory.deserialize(serialized, buffers=buffers)
assert bytes(deserialized.raw()) == data

Writing Buffers to Different Streams

The BufferObject.write_to() method accepts any writable stream object, making it flexible for various use cases:

import pyfory
import numpy as np
import io

fory = pyfory.Fory(xlang=False, ref=False, strict=False)

array = np.arange(1000, dtype=np.float64)

# Collect out-of-band buffers
buffer_objects = []
serialized = fory.serialize(array, buffer_callback=buffer_objects.append)

# Write to different stream types
for buffer_obj in buffer_objects:
    # Write to BytesIO (in-memory stream)
    bytes_stream = io.BytesIO()
    buffer_obj.write_to(bytes_stream)

    # Write to file
    with open('/tmp/buffer_data.bin', 'wb') as f:
        buffer_obj.write_to(f)

    # Get zero-copy memoryview (for contiguous buffers)
    mv = buffer_obj.getbuffer()
    assert isinstance(mv, memoryview)

Note: For contiguous memory buffers (like bytes, numpy arrays), getbuffer() returns a zero-copy memoryview. For non-contiguous data, a copy may be created to ensure contiguity.

Cross-Language Object Graph Serialization

pyfory supports cross-language object graph serialization, allowing you to serialize data in Python and deserialize it in Java, Go, Rust, or other supported languages.

The binary protocol and API are similar to pyfory's Python native mode, but Python native mode can serialize any Python object—including global functions, local functions, lambdas, local classes, and types with customized serialization using __getstate__/__reduce__/__reduce_ex__, which are not allowed in xlang mode.

Xlang mode is the default. Set xlang=True explicitly in cross-language examples so the mode choice is visible:

import pyfory
fory = pyfory.Fory(xlang=True, ref=False, strict=True)

Cross-Language Serialization

Serialize data in Python and deserialize it in Java, Go, Rust, or other supported languages. Both sides must register the same type with matching names:

Python (Serializer)

from dataclasses import dataclass
import pyfory

# Xlang mode for interoperability
f = pyfory.Fory(xlang=True, ref=True)

# Register type for cross-language compatibility
@dataclass
class Person:
    name: str
    age: pyfory.Int32

f.register(Person, typename="example.Person")

person = Person("Charlie", 35)
binary_data = f.serialize(person)
# binary_data can now be sent to Java, Go, etc.

Nested collection annotations are declared schema and are honored in both pure Python and Cython modes.

Java (Deserializer)

import org.apache.fory.*;

public class Person {
    public String name;
    public int age;
}

Fory fory = Fory.builder()
    .withXlang(true)
    .withRefTracking(true)
    .build();

fory.register(Person.class, "example.Person");
Person person = (Person) fory.deserialize(binaryData);

Row Format - Zero-Copy Processing

Apache Fory™ provides a random-access row format that enables reading nested fields from binary data without full deserialization. This drastically reduces overhead when working with large objects where only partial data access is needed. The format also supports memory-mapped files for ultra-low memory footprint.

Basic Row Format Usage

Encode objects to row format for random access without full deserialization. Ideal for large datasets:

Python

import pyfory
import pyarrow as pa
from dataclasses import dataclass
from typing import List, Dict

@dataclass
class Bar:
    f1: str
    f2: List[pa.int64]

@dataclass
class Foo:
    f1: pa.int32
    f2: List[pa.int32]
    f3: Dict[str, pa.int32]
    f4: List[Bar]

# Create encoder for row format
encoder = pyfory.encoder(Foo)

# Create large dataset
foo = Foo(
    f1=10,
    f2=list(range(1_000_000)),
    f3={f"k{i}": i for i in range(1_000_000)},
    f4=[Bar(f1=f"s{i}", f2=list(range(10))) for i in range(1_000_000)]
)

# Encode to row format
binary: bytes = encoder.to_row(foo).to_bytes()

# Zero-copy access - no full deserialization needed!
foo_row = pyfory.RowData(encoder.schema, binary)
print(foo_row.f2[100000])              # Access 100,000th element directly
print(foo_row.f4[100000].f1)           # Access nested field directly
print(foo_row.f4[200000].f2[5])        # Access deeply nested field directly

Cross-Language Compatibility

Row format works across languages. Here's the same data structure accessed in Java:

Java

public class Bar {
  String f1;
  List<Long> f2;
}

public class Foo {
  int f1;
  List<Integer> f2;
  Map<String, Integer> f3;
  List<Bar> f4;
}

RowEncoder<Foo> encoder = Encoders.bean(Foo.class);

// Create large dataset
Foo foo = new Foo();
foo.f1 = 10;
foo.f2 = IntStream.range(0, 1_000_000).boxed().collect(Collectors.toList());
foo.f3 = IntStream.range(0, 1_000_000).boxed().collect(Collectors.toMap(i -> "k" + i, i -> i));
List<Bar> bars = new ArrayList<>(1_000_000);
for (int i = 0; i < 1_000_000; i++) {
  Bar bar = new Bar();
  bar.f1 = "s" + i;
  bar.f2 = LongStream.range(0, 10).boxed().collect(Collectors.toList());
  bars.add(bar);
}
foo.f4 = bars;

// Encode to row format (cross-language compatible with Python)
BinaryRow binaryRow = encoder.toRow(foo);

// Zero-copy random access without full deserialization
BinaryArray f2Array = binaryRow.getArray(1);              // Access f2 list
BinaryArray f4Array = binaryRow.getArray(3);              // Access f4 list
BinaryRow bar10 = f4Array.getStruct(10);                  // Access 11th Bar
long value = bar10.getArray(1).getInt64(5);               // Access 6th element of bar.f2

// Partial deserialization - only deserialize what you need
RowEncoder<Bar> barEncoder = Encoders.bean(Bar.class);
Bar bar1 = barEncoder.fromRow(f4Array.getStruct(10));     // Deserialize 11th Bar only
Bar bar2 = barEncoder.fromRow(f4Array.getStruct(20));     // Deserialize 21st Bar only

// Full deserialization when needed
Foo newFoo = encoder.fromRow(binaryRow);

C++

And in C++ with compile-time type information:

#include "fory/encoder/row_encoder.h"
#include "fory/row/writer.h"

struct Bar {
  std::string f1;
  std::vector<int64_t> f2;
  FORY_STRUCT(Bar, f1, f2);
};

struct Foo {
  int32_t f1;
  std::vector<int32_t> f2;
  std::map<std::string, int32_t> f3;
  std::vector<Bar> f4;
  FORY_STRUCT(Foo, f1, f2, f3, f4);
};

// Create large dataset
Foo foo;
foo.f1 = 10;
for (int i = 0; i < 1000000; i++) {
  foo.f2.push_back(i);
  foo.f3["k" + std::to_string(i)] = i;
}
for (int i = 0; i < 1000000; i++) {
  Bar bar;
  bar.f1 = "s" + std::to_string(i);
  for (int j = 0; j < 10; j++) {
    bar.f2.push_back(j);
  }
  foo.f4.push_back(bar);
}

// Encode to row format (cross-language compatible with Python/Java)
fory::row::encoder::RowEncoder<Foo> encoder;
encoder.encode(foo);
auto row = encoder.get_writer().to_row();

// Zero-copy random access without full deserialization
auto f2_array = row->get_array(1);                   // Access f2 list
auto f4_array = row->get_array(3);                   // Access f4 list
auto bar10 = f4_array->get_struct(10);               // Access 11th Bar
int64_t value = bar10->get_array(1)->get_int64(5);   // Access 6th element of bar.f2
std::string str = bar10->get_string(0);              // Access bar.f1

Key Benefits

  • Zero-Copy Access: Read nested fields without deserializing the entire object
  • Memory Efficiency: Memory-map large datasets directly from disk
  • Cross-Language: Binary format is compatible between Python, Java, and other Fory implementations
  • Partial Deserialization: Deserialize only the specific elements you need
  • High Performance: Skip unnecessary data parsing for analytics and big data workloads

Core API Reference

Fory Class

The main serialization interface:

class Fory:
    def __init__(
        self,
        xlang: bool = True,
        ref: bool = False,
        strict: bool = True,
        compatible: bool | None = None,
        max_depth: int = 50
    )

ThreadSafeFory Class

Thread-safe serialization interface using thread-local storage:

class ThreadSafeFory:
    def __init__(
        self,
        xlang: bool = True,
        ref: bool = False,
        strict: bool = True,
        compatible: bool | None = None,
        max_depth: int = 50
    )

ThreadSafeFory provides thread-safe serialization by maintaining a pool of Fory instances protected by a lock. When a thread needs to serialize/deserialize, it gets an instance from the pool, uses it, and returns it. All type registrations must be done before any serialization to ensure consistency across all instances.

Thread Safety Example:

import pyfory
import threading
from dataclasses import dataclass

@dataclass
class Person:
    name: str
    age: int

# Create thread-safe Fory instance
fory = pyfory.ThreadSafeFory(xlang=False, ref=True)
fory.register(Person)

# Use in multiple threads safely
def serialize_in_thread(thread_id):
    person = Person(name=f"User{thread_id}", age=25 + thread_id)
    data = fory.serialize(person)
    result = fory.deserialize(data)
    print(f"Thread {thread_id}: {result}")

threads = [threading.Thread(target=serialize_in_thread, args=(i,)) for i in range(10)]
for t in threads: t.start()
for t in threads: t.join()

Key Features:

  • Instance Pool: Maintains a pool of Fory instances protected by a lock for thread safety
  • Shared Configuration: All registrations must be done upfront and are applied to all instances
  • Same API: Drop-in replacement for Fory class with identical methods
  • Registration Safety: Prevents registration after first use to ensure consistency

When to Use:

  • Multi-threaded Applications: Web servers, concurrent workers, parallel processing
  • Shared Fory Instances: When multiple threads need to serialize/deserialize data
  • Thread Pools: Applications using thread pools or concurrent.futures

Parameters:

  • xlang (bool, default=True): Use xlang mode. Set False for Python native mode supporting Python-specific objects.
  • ref (bool, default=False): Enable reference tracking for shared/circular references. Disable for better performance if your data has no shared references.
  • strict (bool, default=True): Require type registration for security. Highly recommended for production. Only disable in trusted environments.
  • compatible (bool, default follows xlang): Enable schema evolution. Xlang mode defaults to compatible mode; native mode defaults to schema-consistent mode.
  • max_depth (int, default=50): Maximum deserialization depth for security, preventing stack overflow attacks.

Key Methods:

# Serialization (serialize/deserialize are identical to dumps/loads)
data: bytes = fory.serialize(obj)
obj = fory.deserialize(data)

# Alternative API (aliases)
data: bytes = fory.dumps(obj)
obj = fory.loads(data)

# Type registration by id
fory.register(MyClass, type_id=123)
fory.register(MyClass, type_id=123, serializer=custom_serializer)

# Type registration by name
fory.register(MyClass, typename="my.package.MyClass")
fory.register(MyClass, typename="my.package.MyClass", serializer=custom_serializer)

Xlang And Native Mode Comparison

Feature Native mode (xlang=False) Xlang mode (default)
Use case Pure Python applications Multi-language systems
Compatibility Python only Java, Go, Rust, C++, JavaScript, etc.
Supported types Python object surface Cross-language compatible types
Functions/lambdas Supported with trusted dynamic deserialization Not allowed
Local classes Supported with trusted dynamic deserialization Not allowed
Dynamic classes Supported with trusted dynamic deserialization Not allowed
Schema mode default Schema-consistent Compatible

Native Mode (xlang=False)

Python native mode supports Python-specific objects including functions, classes, and closures. Use it for Python-only applications:

import pyfory

# Python native mode
fory = pyfory.Fory(xlang=False, ref=True, strict=False)

# Supports ALL Python objects:
data = fory.dumps({
    'function': lambda x: x * 2,        # Functions and lambdas
    'class': type('Dynamic', (), {}),    # Dynamic classes
    'method': str.upper,                # Methods
    'nested': {'circular_ref': None}    # Circular references (when ref=True)
})

# Drop-in replacement for pickle/cloudpickle
import pickle
obj = [1, 2, {"nested": [3, 4]}]
assert fory.loads(fory.dumps(obj)) == pickle.loads(pickle.dumps(obj))

# Significantly faster and more compact than pickle
import timeit
obj = {f"key{i}": f"value{i}" for i in range(10000)}
print(f"Fory: {timeit.timeit(lambda: fory.dumps(obj), number=1000):.3f}s")
print(f"Pickle: {timeit.timeit(lambda: pickle.dumps(obj), number=1000):.3f}s")

Xlang Mode

Xlang mode restricts types to those compatible across all Fory implementations. Use it for multi-language systems:

import pyfory

f = pyfory.Fory(xlang=True, ref=True)

# Only supports cross-language compatible types
f.register(MyDataClass, typename="com.example.MyDataClass")

# Data can be read by Java, Go, Rust, etc.
data = f.serialize(MyDataClass(field1="value", field2=42))

Advanced Features

Reference Tracking & Circular References

Handle shared references and circular dependencies safely. Set ref=True to deduplicate objects:

import pyfory

f = pyfory.Fory(xlang=False, ref=True)  # Enable reference tracking

# Handle circular references safely
class Node:
    def __init__(self, value):
        self.value = value
        self.children = []
        self.parent = None

root = Node("root")
child = Node("child")
child.parent = root  # Circular reference
root.children.append(child)

# Serializes without infinite recursion
data = f.serialize(root)
result = f.deserialize(data)
assert result.children[0].parent is result  # Reference preserved

Type Registration

In strict mode, only registered types can be deserialized. This prevents arbitrary code execution:

import pyfory

# Strict mode (recommended for production)
f = pyfory.Fory(xlang=False, strict=True)

class SafeClass:
    def __init__(self, data):
        self.data = data

# Must register types in strict mode
f.register(SafeClass, typename="com.example.SafeClass")

# Now serialization works
obj = SafeClass("safe data")
data = f.serialize(obj)
result = f.deserialize(data)

# Unregistered types will raise an exception
class UnsafeClass:
    pass

# This will fail in strict mode
try:
    f.serialize(UnsafeClass())
except Exception as e:
    print("Security protection activated!")

Custom Serializers

Implement custom serialization logic for specialized types with a single write/read API:

import pyfory
from pyfory.serializer import Serializer
from dataclasses import dataclass

@dataclass
class Foo:
    f1: int
    f2: str

class FooSerializer(Serializer):
    def __init__(self, fory, cls):
        super().__init__(fory, cls)

    def write(self, buffer, obj: Foo):
        # Custom serialization logic
        buffer.write_varint32(obj.f1)
        buffer.write_string(obj.f2)

    def read(self, buffer):
        # Custom deserialization logic
        f1 = buffer.read_varint32()
        f2 = buffer.read_string()
        return Foo(f1, f2)

f = pyfory.Fory(xlang=False)
f.register(Foo, type_id=100, serializer=FooSerializer(f, Foo))

# Now Foo uses your custom serializer
data = f.dumps(Foo(42, "hello"))
result = f.loads(data)
print(result)  # Foo(f1=42, f2='hello')

Numpy & Scientific Computing

Fory natively supports numpy arrays with optimized serialization. Large arrays use zero-copy when possible:

import pyfory
import numpy as np

f = pyfory.Fory(xlang=False)

# Numpy arrays are supported natively
arrays = {
    'matrix': np.random.rand(1000, 1000),
    'vector': np.arange(10000),
    'bool_mask': np.random.choice([True, False], size=5000)
}

data = f.serialize(arrays)
result = f.deserialize(data)

# Zero-copy for compatible array types
assert np.array_equal(arrays['matrix'], result['matrix'])

Best Practices

Production Configuration

Use these recommended settings to balance security, performance, and functionality in production:

import pyfory

# Recommended settings for production
fory = pyfory.Fory(
    xlang=False,        # Native mode for Python-only traffic
    ref=False,           # Enable if you have shared/circular references
    strict=True,        # CRITICAL: Always True in production
    compatible=False,   # Native mode defaults to schema-consistent payloads
    max_depth=20       # Adjust based on your data structure depth
)

# Register all types upfront
fory.register(UserModel, type_id=100)
fory.register(OrderModel, type_id=101)
fory.register(ProductModel, type_id=102)

Performance Tips

Optimize serialization speed and memory usage with these guidelines:

  1. Disable ref=True if not needed: Reference tracking has overhead
  2. Use type_id instead of typename: Integer IDs are faster than string names
  3. Reuse Fory instances: Create once, use many times
  4. Enable Cython: Make sure ENABLE_FORY_CYTHON_SERIALIZATION=1, should be enabled by default
  5. Use row format for large arrays: Zero-copy access for analytics
# Good: Reuse instance
fory = pyfory.Fory(xlang=False)
for obj in objects:
    data = fory.dumps(obj)

# Bad: Create new instance each time
for obj in objects:
    fory = pyfory.Fory(xlang=False)  # Wasteful!
    data = fory.dumps(obj)

Type Registration Patterns

Choose the right registration approach for your use case:

# Pattern 1: Simple registration
fory.register(MyClass, type_id=100)

# Pattern 2: Cross-language with typename
fory.register(MyClass, typename="com.example.MyClass")

# Pattern 3: With custom serializer
fory.register(MyClass, type_id=100, serializer=MySerializer(fory, MyClass))

# Pattern 4: Batch registration
type_id = 100
for model_class in [User, Order, Product, Invoice]:
    fory.register(model_class, type_id=type_id)
    type_id += 1

Error Handling

Handle common serialization errors gracefully. Catch specific exceptions for better error recovery:

import pyfory
from pyfory.error import TypeUnregisteredError, TypeNotCompatibleError

fory = pyfory.Fory(strict=True)

try:
    data = fory.dumps(my_object)
except TypeUnregisteredError as e:
    print(f"Type not registered: {e}")
    # Register the type and retry
    fory.register(type(my_object), type_id=100)
    data = fory.dumps(my_object)
except Exception as e:
    print(f"Serialization failed: {e}")

try:
    obj = fory.loads(data)
except TypeNotCompatibleError as e:
    print(f"Schema mismatch: {e}")
    # Handle version mismatch
except Exception as e:
    print(f"Deserialization failed: {e}")

Security Best Practices

Production Configuration

Never disable strict=True in production unless your environment is completely trusted:

import pyfory

# Recommended production settings
f = pyfory.Fory(
    ref=True,      # Handle circular references
    strict=True,   # IMPORTANT: Prevent malicious data
    max_depth=100  # Prevent deep recursion attacks
)

# Explicitly register allowed types
f.register(UserModel, type_id=100)
f.register(OrderModel, type_id=101)
# Never set strict=False in production with untrusted data!

Development vs Production

Use environment variables to switch between development and production configurations:

import pyfory
import os

# Development configuration
if os.getenv('ENV') == 'development':
    fory = pyfory.Fory(
        xlang=False,
        ref=True,
        strict=False,    # Allow any type for development
        max_depth=1000   # Higher limit for development
    )
else:
    # Production configuration (security hardened)
    fory = pyfory.Fory(
        ref=True,
        strict=True,     # CRITICAL: Require registration
        max_depth=100    # Reasonable limit
    )
    # Register only known safe types
    for idx, model_class in enumerate([UserModel, ProductModel, OrderModel]):
        fory.register(model_class, type_id=100 + idx)

DeserializationPolicy

When strict=False is necessary (e.g., deserializing functions/lambdas), use DeserializationPolicy to implement fine-grained security controls during deserialization. This provides protection similar to pickle.Unpickler.find_class() but with more comprehensive hooks.

Why use DeserializationPolicy?

  • Block dangerous classes/modules (e.g., subprocess.Popen)
  • Intercept and validate __reduce__ callables before invocation
  • Sanitize sensitive data during __setstate__
  • Replace or reject deserialized objects based on custom rules

Example: Blocking Dangerous Classes

import pyfory
from pyfory import DeserializationPolicy

dangerous_modules = {'subprocess', 'os', '__builtin__'}

class SafeDeserializationPolicy(DeserializationPolicy):
    """Block potentially dangerous classes during deserialization."""

    def validate_class(self, cls, is_local, **kwargs):
        # Block dangerous modules
        if cls.__module__ in dangerous_modules:
            raise ValueError(f"Blocked dangerous class: {cls.__module__}.{cls.__name__}")

    def intercept_reduce_call(self, callable_obj, args, **kwargs):
        # Block specific callable invocations during __reduce__
        if getattr(callable_obj, '__name__', "") == 'Popen':
            raise ValueError("Blocked attempt to invoke subprocess.Popen")
        return None

    def intercept_setstate(self, obj, state, **kwargs):
        # Sanitize sensitive data
        if isinstance(state, dict) and 'password' in state:
            state['password'] = '***REDACTED***'
        return None

# Create Fory with custom security policy
policy = SafeDeserializationPolicy()
fory = pyfory.Fory(xlang=False, ref=True, strict=False, policy=policy)

# Now deserialization is protected by your custom policy
data = fory.serialize(my_object)
result = fory.deserialize(data)  # Policy hooks will be invoked

Available Policy Hooks:

  • Reference validation hooks reject by raising exceptions and otherwise leave deserialized references unchanged.
  • validate_class(cls, is_local) - Validate/block class types during deserialization
  • validate_module(module_name, is_local) - Validate/block module imports
  • validate_function(func, is_local) - Validate/block function references
  • validate_method(method, is_local) - Validate/block method references
  • intercept_reduce_call(callable_obj, args) - Intercept __reduce__ invocations
  • inspect_reduced_object(obj) - Inspect/replace objects created via __reduce__
  • intercept_setstate(obj, state) - Sanitize state before __setstate__
  • authorize_instantiation(cls, args, kwargs) - Control class instantiation

See also: pyfory/policy.py contains detailed documentation and examples for each hook.

Troubleshooting

Common Issues

Q: ImportError with format features

# A: Install Row format support
pip install pyfory[format]

# Or install from source with format support
pip install -e ".[format]"

Q: Slow serialization performance

# A: Check if Cython acceleration is enabled
import pyfory
print(pyfory.ENABLE_FORY_CYTHON_SERIALIZATION)  # Should be True

# If False, Cython extension may not be compiled correctly
# Reinstall with: pip install --force-reinstall --no-cache-dir pyfory

# For debugging, you can disable the Cython implementation before importing
import os
os.environ['ENABLE_FORY_CYTHON_SERIALIZATION'] = '0'
import pyfory  # Now uses the pure Python implementation

Q: Cross-language compatibility issues

# A: Use explicit type registration with consistent naming
f = pyfory.Fory(xlang=True)
f.register(MyClass, typename="com.package.MyClass")  # Use same name in all languages

Q: Circular reference errors or duplicate data

Registered xlang schema objects and Python native objects both require reference tracking when object identity or cycles matter:

# A: Enable reference tracking for registered schema objects
f = pyfory.Fory(ref=True)

For arbitrary Python object graphs with circular references, use Python native mode:

f = pyfory.Fory(xlang=False, ref=True, strict=False)

# Example with circular reference
class Node:
    def __init__(self, value):
        self.value = value
        self.next = None

node1 = Node(1)
node2 = Node(2)
node1.next = node2
node2.next = node1  # Circular reference

data = f.dumps(node1)
result = f.loads(data)
assert result.next.next is result  # Circular reference preserved

Debug Mode

# Set environment variable BEFORE importing pyfory to disable Cython for debugging
import os
os.environ['ENABLE_FORY_CYTHON_SERIALIZATION'] = '0'
import pyfory  # Now uses pure Python implementation

# This is useful for:
# 1. Debugging protocol issues
# 2. Understanding serialization behavior
# 3. Development without recompiling Cython

Q: Schema evolution not working

# A: Xlang mode defaults to compatible schema evolution.
f = pyfory.Fory(xlang=True)

# Version 1: Original class
@dataclass
class User:
    name: str
    age: int

f.register(User, typename="User")
data = f.dumps(User("Alice", 30))

# Version 2: Add new field (backward compatible)
@dataclass
class User:
    name: str
    age: int
    email: str = "unknown@example.com"  # New field with default

# Can still deserialize old data
user = f.loads(data)
print(user.email)  # "unknown@example.com"

Q: Type registration errors in strict mode

# A: Register all custom types before serialization
f = pyfory.Fory(strict=True)

# Must register before use
f.register(MyClass, type_id=100)
f.register(AnotherClass, type_id=101)

# Or disable strict mode (NOT recommended for production)
f = pyfory.Fory(strict=False)  # Use only in trusted environments

Contributing

Apache Fory™ is an open-source project under the Apache Software Foundation. We welcome all forms of contributions:

How to Contribute

  1. Report Issues: Found a bug? Open an issue
  2. Suggest Features: Have an idea? Start a discussion
  3. Improve Docs: Documentation improvements are always welcome
  4. Submit Code: See our Contributing Guide

For Contributors: See CONTRIBUTING.md for comprehensive development setup instructions

License

Apache License 2.0. See LICENSE for details.


Apache Fory™ - Blazing fast, secure, and versatile serialization for modern applications.

Links

Community

We welcome contributions! Whether it's bug reports, feature requests, documentation improvements, or code contributions, we appreciate your help.

Project details


Download files

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

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distributions

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

pyfory-1.1.0-cp313-cp313-win_amd64.whl (1.1 MB view details)

Uploaded CPython 3.13Windows x86-64

pyfory-1.1.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.whl (1.5 MB view details)

Uploaded CPython 3.13manylinux: glibc 2.17+ x86-64

pyfory-1.1.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.whl (1.4 MB view details)

Uploaded CPython 3.13manylinux: glibc 2.17+ ARM64

pyfory-1.1.0-cp313-cp313-macosx_11_0_universal2.whl (2.4 MB view details)

Uploaded CPython 3.13macOS 11.0+ universal2 (ARM64, x86-64)

pyfory-1.1.0-cp312-cp312-win_amd64.whl (1.1 MB view details)

Uploaded CPython 3.12Windows x86-64

pyfory-1.1.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl (1.5 MB view details)

Uploaded CPython 3.12manylinux: glibc 2.17+ x86-64

pyfory-1.1.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl (1.4 MB view details)

Uploaded CPython 3.12manylinux: glibc 2.17+ ARM64

pyfory-1.1.0-cp312-cp312-macosx_11_0_universal2.whl (2.4 MB view details)

Uploaded CPython 3.12macOS 11.0+ universal2 (ARM64, x86-64)

pyfory-1.1.0-cp311-cp311-win_amd64.whl (1.1 MB view details)

Uploaded CPython 3.11Windows x86-64

pyfory-1.1.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl (1.5 MB view details)

Uploaded CPython 3.11manylinux: glibc 2.17+ x86-64

pyfory-1.1.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl (1.4 MB view details)

Uploaded CPython 3.11manylinux: glibc 2.17+ ARM64

pyfory-1.1.0-cp311-cp311-macosx_11_0_universal2.whl (2.4 MB view details)

Uploaded CPython 3.11macOS 11.0+ universal2 (ARM64, x86-64)

pyfory-1.1.0-cp310-cp310-win_amd64.whl (1.1 MB view details)

Uploaded CPython 3.10Windows x86-64

pyfory-1.1.0-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.whl (1.5 MB view details)

Uploaded CPython 3.10manylinux: glibc 2.17+ x86-64

pyfory-1.1.0-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.whl (1.4 MB view details)

Uploaded CPython 3.10manylinux: glibc 2.17+ ARM64

pyfory-1.1.0-cp310-cp310-macosx_11_0_universal2.whl (2.4 MB view details)

Uploaded CPython 3.10macOS 11.0+ universal2 (ARM64, x86-64)

pyfory-1.1.0-cp39-cp39-win_amd64.whl (1.1 MB view details)

Uploaded CPython 3.9Windows x86-64

pyfory-1.1.0-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.whl (1.5 MB view details)

Uploaded CPython 3.9manylinux: glibc 2.17+ x86-64

pyfory-1.1.0-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.whl (1.4 MB view details)

Uploaded CPython 3.9manylinux: glibc 2.17+ ARM64

pyfory-1.1.0-cp39-cp39-macosx_11_0_universal2.whl (2.4 MB view details)

Uploaded CPython 3.9macOS 11.0+ universal2 (ARM64, x86-64)

pyfory-1.1.0-cp38-cp38-win_amd64.whl (1.1 MB view details)

Uploaded CPython 3.8Windows x86-64

pyfory-1.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (1.5 MB view details)

Uploaded CPython 3.8manylinux: glibc 2.17+ x86-64

pyfory-1.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl (1.4 MB view details)

Uploaded CPython 3.8manylinux: glibc 2.17+ ARM64

pyfory-1.1.0-cp38-cp38-macosx_11_0_universal2.whl (2.4 MB view details)

Uploaded CPython 3.8macOS 11.0+ universal2 (ARM64, x86-64)

File details

Details for the file pyfory-1.1.0-cp313-cp313-win_amd64.whl.

File metadata

  • Download URL: pyfory-1.1.0-cp313-cp313-win_amd64.whl
  • Upload date:
  • Size: 1.1 MB
  • Tags: CPython 3.13, Windows x86-64
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pyfory-1.1.0-cp313-cp313-win_amd64.whl
Algorithm Hash digest
SHA256 6f6bdf487d7b882854a7cede62ea305c18dc677f1d2dcf7f0320e3bc4a5d80e9
MD5 eee7c04af2e92ff3637da8b6570a855d
BLAKE2b-256 df5d46d656ca66514d008e03ead161afefd40028b01c43ac4ff41dafc4c9ab30

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp313-cp313-win_amd64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.whl
Algorithm Hash digest
SHA256 9d7f817216580e8e5503e8066f3f90f2d42a8400bf83eba968fcf97f0ab43dd9
MD5 04a574bd80f7071f8e189c80bc5d5d71
BLAKE2b-256 36b632efbea0d181f9394fcea265b2897b8cb2c976d2247dda7ac214ebae3efb

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.whl
Algorithm Hash digest
SHA256 99083b7913352e5766aebfe9b02bbe4cae69eb439fdbc408896caf1d948e423f
MD5 0aafe4f8618a40cb145f677d9c69d7a2
BLAKE2b-256 0aba00e0cfc732c0efa1301febd6424815abe0bc8b7f19ba148352f4ad96e201

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp313-cp313-macosx_11_0_universal2.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp313-cp313-macosx_11_0_universal2.whl
Algorithm Hash digest
SHA256 62b7091072b451fd82ee3ee6a2193685a15c62aacfac0e356b4632d909ec1513
MD5 8b59c64530fde22c21e5a1e46e67fbfd
BLAKE2b-256 a24a330bc2f2e3e024a3db487e7865287eb79b792fc140b20f6e8bdb7e4f77ba

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp313-cp313-macosx_11_0_universal2.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp312-cp312-win_amd64.whl.

File metadata

  • Download URL: pyfory-1.1.0-cp312-cp312-win_amd64.whl
  • Upload date:
  • Size: 1.1 MB
  • Tags: CPython 3.12, Windows x86-64
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pyfory-1.1.0-cp312-cp312-win_amd64.whl
Algorithm Hash digest
SHA256 12ca6edee68bf6af860b3ea4a75b76841ede880ba7704e223c153b963d4c2b9e
MD5 482cb7e0df241dbdaedeba988929a143
BLAKE2b-256 040e76d2e8e66d4e2f5d0c091793fc893fd035a367c95b6abdb32e5d8e342bff

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp312-cp312-win_amd64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl
Algorithm Hash digest
SHA256 61dac513732302d5a32e96f26bfaf0b7a1bfe1f59119b6dfcde136054ee69213
MD5 30b43ddedfb59e447c3e37607e069454
BLAKE2b-256 53a186b894551756fb32a8e769f8a3bf1250e6e1450e0912f6f57886a8821afc

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl
Algorithm Hash digest
SHA256 08bfebbb82303a31b2419aca59b4d6facf8bc7385d47b416365631bd6417a8d1
MD5 06f0d0c60a785e73ab86ce6541bfaa81
BLAKE2b-256 3dbadb7aa6857626993dad7f682f10040944395fb78cae5a235e4a7789feac73

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp312-cp312-macosx_11_0_universal2.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp312-cp312-macosx_11_0_universal2.whl
Algorithm Hash digest
SHA256 04c10eb77bf5afcbce20c3c9fa0e28ec6ba38b1b3ae0aca71ba6185f967f4727
MD5 9ad0f8e75170640837e1f974d43d5ef9
BLAKE2b-256 75e8510ca6cb4cdf124bea03ffc49fddeed38bc380f9e92de3e9d0bf958fac2d

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp312-cp312-macosx_11_0_universal2.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp311-cp311-win_amd64.whl.

File metadata

  • Download URL: pyfory-1.1.0-cp311-cp311-win_amd64.whl
  • Upload date:
  • Size: 1.1 MB
  • Tags: CPython 3.11, Windows x86-64
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pyfory-1.1.0-cp311-cp311-win_amd64.whl
Algorithm Hash digest
SHA256 9bea3fb65a01a9621cb8beb33fbf217dd730081d207220601c4d73392e0a0477
MD5 28d813e1e06fbfd8ac8fbebb724ecd2f
BLAKE2b-256 bd27cbb09989c24e9caa54382afb2428335806cc58a734dcb6e77bd9b32c2a68

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp311-cp311-win_amd64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl
Algorithm Hash digest
SHA256 f34f702c0bc63bdc303a55111e21578ffc2c34f8bb0738c62b42254751a64749
MD5 ad9131aa604b805f6952b54cc1a9c0e9
BLAKE2b-256 145f202b7862a615c28674f62590ee3081ec9b320cced4af2b9c9ce2d0db9450

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl
Algorithm Hash digest
SHA256 0ea09b0db8c99a8fda6c0640e92c429cca1e08dd8cd11bfb235b5114ca84efd0
MD5 315c1c5bf10686b148e7871b23c6ec39
BLAKE2b-256 daadb174bfc6e0e118d90cd3ef91cc42b662d6febeda4ad1f91f20b4299a976f

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp311-cp311-macosx_11_0_universal2.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp311-cp311-macosx_11_0_universal2.whl
Algorithm Hash digest
SHA256 1623355353f8170f270115e593576fceaea17b781ed9adf308fe309576155486
MD5 f4632de6add51b3989b1afc5b3923f01
BLAKE2b-256 164a9ce2bcd3ab9a469ff1b8c260be222228eb2c845b17936ac900109edc71e6

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp311-cp311-macosx_11_0_universal2.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp310-cp310-win_amd64.whl.

File metadata

  • Download URL: pyfory-1.1.0-cp310-cp310-win_amd64.whl
  • Upload date:
  • Size: 1.1 MB
  • Tags: CPython 3.10, Windows x86-64
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pyfory-1.1.0-cp310-cp310-win_amd64.whl
Algorithm Hash digest
SHA256 a3eedcb48fbdf46f8dc311ca403e433fddfe6e5782afd96cff6058e356a8cb7e
MD5 0679d10ed8c3090c5928082fb202ab06
BLAKE2b-256 1311851a36a50abe360246622c736ec9e228d0d8fe2c92309151679cab3fef37

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp310-cp310-win_amd64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.whl
Algorithm Hash digest
SHA256 0b8cdcbe1b6570402e1277612c702fe6045a9c445e728d81479be74218d1a5aa
MD5 e291f255b9b6c8b7656917b9fa63b573
BLAKE2b-256 5f838e9be10b1a591d42bf4ba47c138aa8cebf1f46208364d715167a77f723db

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.whl
Algorithm Hash digest
SHA256 a915ed8cd8d4493630fb38495be5b9419afe9bc637901bb1b19a10e3ab9176c7
MD5 f3c64cabdf019c51d2544e1b3ea959ac
BLAKE2b-256 e7e96ad59d97ee2ec273cd243f1c051094e570c710c695bfc110619a22cfc5c0

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp310-cp310-macosx_11_0_universal2.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp310-cp310-macosx_11_0_universal2.whl
Algorithm Hash digest
SHA256 3dddf2d87800deca4b08fdccfa7d2a2b12c5de88f72fc19ef796ecbca3a094bb
MD5 4d7e32634d8bb44d051215ce5e0a4463
BLAKE2b-256 03f403a3ae607a77f20f4ef8de7b6774fda759cb65e85eae98c9de9ed5e47d8d

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp310-cp310-macosx_11_0_universal2.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp39-cp39-win_amd64.whl.

File metadata

  • Download URL: pyfory-1.1.0-cp39-cp39-win_amd64.whl
  • Upload date:
  • Size: 1.1 MB
  • Tags: CPython 3.9, Windows x86-64
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pyfory-1.1.0-cp39-cp39-win_amd64.whl
Algorithm Hash digest
SHA256 ad94694679b6b289dc3ba2b44e21adc90cfce4cbd3f98576d139383f521a5a85
MD5 ad8eb4698103fc130858d1a73235b21a
BLAKE2b-256 d03474f1b3693eaf99ac295098f42208502d0e4ea527a43f1bfe784bd1978c3d

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp39-cp39-win_amd64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.whl
Algorithm Hash digest
SHA256 dbf0234d9f5658c4d96fba6fe554818acc4152aeeb3373d15210f039cbec2c5f
MD5 0968dd840dbfd416952db6e683301a70
BLAKE2b-256 6dbc30d9f1ce3e3ef9bb15f0457ad54e32ed641839c654a495094c7828439c70

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.whl
Algorithm Hash digest
SHA256 9df0a0a72bbc534afc6b269c5f3eda8076ec60bb820b406134f7ee820124cd61
MD5 b73b77f6fd651a0e20d2c8d03f0cd007
BLAKE2b-256 b7a499dc130e0552e421df935127403698a06818a5e4cfcdfb85ed69e5598cfd

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp39-cp39-macosx_11_0_universal2.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp39-cp39-macosx_11_0_universal2.whl
Algorithm Hash digest
SHA256 ae1367dde2875730c5393a3914548b4fe4ae3734fba75c69ba88fe26061e9e54
MD5 eccad28fd0cb3ad3196c696f965971a1
BLAKE2b-256 654c2305689117dc694062187c90a279deadb9644e7bc185f36f5c52dda28c93

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp39-cp39-macosx_11_0_universal2.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp38-cp38-win_amd64.whl.

File metadata

  • Download URL: pyfory-1.1.0-cp38-cp38-win_amd64.whl
  • Upload date:
  • Size: 1.1 MB
  • Tags: CPython 3.8, Windows x86-64
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pyfory-1.1.0-cp38-cp38-win_amd64.whl
Algorithm Hash digest
SHA256 1d1f8b0a0bca7f036c310866814d057f02a0f740d9960d3cb919130f9a6ebba9
MD5 19e40f8dd536038718a56a9b819d33c1
BLAKE2b-256 483639277f480316aac1fd93b5196286e414ca5b4a43ffa57bfe499398afd44d

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp38-cp38-win_amd64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 fb2510b6001117b34af870de379e230d439c5c0785fee53ad9d07df0cbc65b31
MD5 fb907f2ede9ccf169deaa4f22be50588
BLAKE2b-256 26d8c8c6644189bfced1e545475134af3744d2ff8346ef6a0676cde2721cd842

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl
Algorithm Hash digest
SHA256 202b5d91c245ffae523f837e9180f3ece345a63bbb1132c7623a7d077a9ed2b4
MD5 8a083f42da5ba3c41430e14cd06170d6
BLAKE2b-256 2a696f9d32461243cc71167e0fb258efb26ec30c2071eee05dfe69939633a426

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyfory-1.1.0-cp38-cp38-macosx_11_0_universal2.whl.

File metadata

File hashes

Hashes for pyfory-1.1.0-cp38-cp38-macosx_11_0_universal2.whl
Algorithm Hash digest
SHA256 d0ef1e08f060cd927ae9e7222ec59a5391ddac3a59f3ad665f0ecc5fceba0f18
MD5 b77cff5679e972c66bd57c5e1a066b25
BLAKE2b-256 11b3bf03b25da87594f4ca75986d861e02c7ea7aad0ad832f96adabdf7a73611

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyfory-1.1.0-cp38-cp38-macosx_11_0_universal2.whl:

Publisher: release-python.yaml on apache/fory

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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