Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/flix-tech/avro-serde-php
Avro Serialisation/Deserialisation (SerDe) library for PHP 8.1+ with a Symfony Serializer integration
https://github.com/flix-tech/avro-serde-php
avro avro-format avro-schema confluent confluent-platform deserialization php serde serialization symfony symfony-serializer
Last synced: 5 days ago
JSON representation
Avro Serialisation/Deserialisation (SerDe) library for PHP 8.1+ with a Symfony Serializer integration
- Host: GitHub
- URL: https://github.com/flix-tech/avro-serde-php
- Owner: flix-tech
- License: mit
- Created: 2018-09-01T11:48:58.000Z (over 6 years ago)
- Default Branch: master
- Last Pushed: 2024-12-25T04:59:35.000Z (30 days ago)
- Last Synced: 2025-01-11T23:03:59.834Z (12 days ago)
- Topics: avro, avro-format, avro-schema, confluent, confluent-platform, deserialization, php, serde, serialization, symfony, symfony-serializer
- Language: PHP
- Homepage: https://www.flix.tech/
- Size: 205 KB
- Stars: 64
- Watchers: 8
- Forks: 37
- Open Issues: 16
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Avro SerDe for PHP 8.1+
[![php-confluent-serde Actions Status](https://github.com/flix-tech/avro-serde-php/workflows/php-confluent-serde/badge.svg?branch=master)](https://github.com/flix-tech/avro-serde-php/actions)
[![Maintainability](https://api.codeclimate.com/v1/badges/7500470a6812cf5a1ad5/maintainability)](https://codeclimate.com/github/flix-tech/avro-serde-php/maintainability)
[![Test Coverage](https://api.codeclimate.com/v1/badges/7500470a6812cf5a1ad5/test_coverage)](https://codeclimate.com/github/flix-tech/avro-serde-php/test_coverage)
[![Latest Stable Version](https://poser.pugx.org/flix-tech/avro-serde-php/version)](https://packagist.org/packages/flix-tech/avro-serde-php)
[![Total Downloads](https://poser.pugx.org/flix-tech/avro-serde-php/downloads)](https://packagist.org/packages/flix-tech/avro-serde-php)
[![License](https://poser.pugx.org/flix-tech/avro-serde-php/license)](https://packagist.org/packages/flix-tech/avro-serde-php)## Motivation
When serializing and deserializing messages using the [Avro](http://avro.apache.org/docs/current/) serialization format,
especially when integrating with the [Confluent Platform](https://docs.confluent.io/current/avro.html), you want to make
sure that schemas are evolved in a way that downstream consumers are not affected.Hence, [Confluent](https://www.confluent.io/) developed the
[Schema Registry](https://docs.confluent.io/current/schema-registry/docs/index.html) which has the responsibility to
validate a given schema evolution against a configurable compatibility policy.Unfortunately Confluent is not providing an official Avro SerDe package for PHP. This library aims to provide an Avro
SerDe library for PHP that implements the
[Confluent wire format](https://docs.confluent.io/current/schema-registry/docs/serializer-formatter.html#wire-format) and
integrates FlixTech's [Schema Registry Client](https://github.com/flix-tech/schema-registry-php-client).## Installation
This library is using the [composer package manager](https://getcomposer.org/) for PHP.
```bash
composer require 'flix-tech/avro-serde-php:^2.2'
```## Quickstart
> **NOTE**
>
> You should **always** use a cached schema registry client, since otherwise you'd make an HTTP request for every
> message serialized or deserialized.### 1. Create a cached Schema Registry client
See the [Schema Registry client documentation on caching](https://github.com/flix-tech/schema-registry-php-client#caching)
for more detailed information.```php
'registry.example.com'])
),
new AvroObjectCacheAdapter()
);
```### 2. Build the `RecordSerializer` instance
The `RecordSerializer` is the main way you interact with this library. It provides the `encodeRecord` and
`decodeMessage` methods for SerDe operations.```php
false,
// If you want to auto-register missing subjects set this to true
RecordSerializer::OPTION_REGISTER_MISSING_SUBJECTS => false,
]
);
```### 3. Encoding records
This is a simple example on how you can use the `RecordSerializer` to encode messages in the Confluent Avro wire format.
```php
encodeRecord($subject, $avroSchema, $record);
// Send this over the wire...
```### 4. Decoding messages
This is a simple example on how you can use the `RecordSerializer` to decode messages.
```php
decodeMessage($encodedBinaryAvro);echo $record; // 'Test message'
```## Schema Resolvers
Schema Resolvers are responsible to know which Avro schema belongs to which type of record. This is especially useful
if you want to manage your Avro schemas in separate files. Schema Resolvers enable you to integrate with whatever schema
management concept you may have outside of the scope of this library.Schema Resolvers take a `$record` of any type and try to resolve a matching [`AvroSchema`]() instance for it.
### FileResolver
In even moderately complicated applications you want to manage your schemas within the VCS, most probably as `.avsc`
files. These files contain JSON that is describing the Avro schema.The resolver takes a `$baseDir` in which you want to manage the files and an inflector `callable`, which is a simple
function that takes the record as first parameter, and a second boolean `$isKey` parameter indicating if the inflection
is targeting a key schema.```php
valueSchemaFor($record); // This will load from $baseDir . '/' . MyNamespace.MyRecord.avsc
$resolver->keySchemaFor($record); // This will load from $baseDir . '/' . MyNamespace.MyRecord.key.avsc
```### CallableResolver
This is the simplest but also most flexible resolver. It just takes two `callables` that are responsible to fetch either
value- or key-schemas respectively. A key schema resolver is optional.```php
'bar' ];$schema = $resolver->valueSchemaFor($record);
Assert::assertEquals($schema, $valueSchema);
```### DefinitionInterfaceResolver
This library also provides a [`HasSchemaDefinitionInterface`](src/Objects/HasSchemaDefinitionInterface.php)
that exposes two static methods:* `HasSchemaDefinitionInterface::valueSchemaJson` returns the schema definition for the value as JSON string
* `HasSchemaDefinitionInterface::keySchemaJson` returns either `NULL` or the schema definition for the key as JSON
string.The `DefinitionInterfaceResolver` checks if a given record implements that interface (if not it will throw an
`InvalidArgumentException`) and resolves the schemas via the static methods.```php
valueSchemaFor($record); // Will resolve from $record::valueSchemaJson();
$resolver->keySchemaFor($record); // Will resolve from $record::keySchemaJson();
```### ChainResolver
The chain resolver is a useful tool for composing multiple resolvers. The first resolver to be able to resolve a schema
will win. If none of the resolvers in the chain is able to determine a schema, an `InvalidArgumentException` is thrown.```php
'bar'];/** @var \FlixTech\AvroSerializer\Objects\SchemaResolvers\FileResolver $fileResolver */
/** @var \FlixTech\AvroSerializer\Objects\SchemaResolvers\CallableResolver $callableResolver */$resolver = new ChainResolver($fileResolver, $callableResolver);
// or new ChainResolver(...[$fileResolver, $callableResolver]);$resolver->valueSchemaFor($record); // Will resolve $fileResolver, then $callableResolver
$resolver->keySchemaFor($record); // Will resolve $fileResolver, then $callableResolver
```## Symfony Serializer Integration
This library provides integrations with the [Symfony Serializer component](https://symfony.com/doc/master/components/serializer.html) from version 6.4 and above.
```php
name = $name;
$this->age = $age;
}public function getName(): string
{
return $this->name;
}public function setName(string $name): void
{
$this->name = $name;
}public function getAge(): int
{
return $this->age;
}public function setAge(int $age): void
{
$this->age = $age;
}
}$recordSerializer = DefaultRecordSerializerFactory::get(
getenv('SCHEMA_REGISTRY_HOST')
);$avroSchemaJson = '{
"type": "record",
"name": "user",
"fields": [
{"name": "name", "type": "string"},
{"name": "age", "type": "int"}
]
}';$user = new User('Thomas', 38);
$normalizer = new GetSetMethodNormalizer();
$encoder = new AvroSerDeEncoder($recordSerializer);$symfonySerializer = new Serializer([$normalizer], [$encoder]);
$serialized = $symfonySerializer->serialize(
$user,
AvroSerDeEncoder::FORMAT_AVRO,
[
AvroSerDeEncoder::CONTEXT_ENCODE_SUBJECT => 'users-value',
AvroSerDeEncoder::CONTEXT_ENCODE_WRITERS_SCHEMA => AvroSchema::parse($avroSchemaJson),
]
);$deserializedUser = $symfonySerializer->deserialize(
$serialized,
User::class,
AvroSerDeEncoder::FORMAT_AVRO
);Assert::assertEquals($deserializedUser, $user);
```
### Name converter
Sometimes your property names may differ from the names of the fields in your schema. One option
to solve this is by using [custom Serializer annotations](https://symfony.com/doc/current/components/serializer.html#configure-name-conversion-using-metadata). However, if you're using the annotations
provided by this library,
you may use our [name converter](integrations/Symfony/Serializer/NameConverter/AvroNameConverter.php)
that parses these annotations and maps between the schema field names and the property names.```php
name('object')
->namespace('org.acme')
->doc('A test object')
->aliases(['stdClass', 'array'])
->field('name', Schema::string(), FieldOption::doc('Name of the object'), FieldOption::orderDesc())
->field('answer', Schema::int(), FieldOption::default(42), FieldOption::orderAsc(), FieldOption::aliases('wrong', 'correct'))
->field('ignore', Schema::boolean(), FieldOption::orderIgnore())
->parse();
```## Schema generator
Besides providing a fluent api for defining schemas, we also provide means of generating schema from
class metadata (annotations). For this to work, you have to install the `doctrine/annotations` package.```php
firstName = $firstName;
$this->lastName = $lastName;
$this->age = $age;
}public function getFirstName(): string
{
return $this->firstName;
}public function getLastName(): string
{
return $this->lastName;
}public function getAge(): int
{
return $this->age;
}
}$generator = DefaultSchemaGeneratorFactory::get();
$schema = $generator->generate(User::class);
$avroSchema = $schema->parse();
```Further examples on the possible annotations can be seen in the [test case](test/Objects/Schema/Generation/SchemaGeneratorTest.php).
## Examples
This library provides a few executable examples in the [examples](examples) folder. You should have a look to get an
understanding how this library works.