O serializador e deserializador definitivo para PHP
Constructo é uma poderosa biblioteca PHP que fornece capacidades avançadas de serialização e deserialização para objetos PHP. Ela permite conversão perfeita entre objetos e arrays/JSON, com suporte para estruturas aninhadas complexas, conversão de tipos, validação e formatação personalizada.
- Conversão Bidirecional: Serialize objetos para arrays/JSON e deserialize de volta para objetos tipados
- Segurança de Tipos: Suporte completo ao sistema de tipos do PHP 8.3+ incluindo union types, backed enums e propriedades readonly
- Mapeamento Inteligente: Mapeamento automático de propriedades com conversão de snake_case para camelCase
- Formatadores Personalizados: Sistema de formatação extensível para transformações de dados customizadas
- Objetos Aninhados: Manipule hierarquias de objetos complexas e coleções perfeitamente
- Tratamento de Erros: Relatório de erros abrangente com contexto detalhado
- Validação: Validação integrada com suporte a atributos personalizados
- Manipulação de Data/Hora: Análise e formatação inteligente de DateTime
- Coleções: Suporte de primeira classe para coleções tipadas
- Injeção de Dependência: Resolução automática de dependência para construção de objetos
Instale o Constructo via Composer:
composer require devitools/constructo
- PHP 8.3 ou superior
- ext-json
<?php
# ...
use Constructo\Core\Serialize\Builder;
use Constructo\Support\Set;
use Constructo\Type\Timestamp;
// Defina sua entidade informando os valores das propriedades no construtor
readonly class User
{
public function __construct(
public int $id,
public string $name,
public Timestamp $birthDate,
public bool $isActive = true,
public array $tags = [],
) {}
}
// Monte um set com os dados (de JSON, banco de dados, etc.)
$set = Set::createFrom([
'id' => 1,
'name' => 'João Silva',
'birth_date' => '1981-08-13',
'is_active' => true,
'tags' => ['nice', 'welcome'],
]);
// Crie um novo builder e use-o para construir o objeto
$user = (new Builder())->build(User::class, $set);
echo "Usuário: \n";
echo sprintf(" ID: %s\n", $user->id);
echo sprintf(" Nome: %s\n", $user->name);
echo sprintf(" Ativo: %s\n", $user->isActive);
echo sprintf(" Tags: %s\n", implode(', ', $user->tags));
echo sprintf(" Data de Nascimento: %s\n", $user->birthDate->format('Y-m-d'));
<?php
use Constructo\Core\Deserialize\Demolisher;
// Crie uma instância
$user = new User(1, 'João Silva', '[email protected]', true, ['admin', 'usuario']);
// Serialize para objeto/array
$demolisher = new Demolisher();
$data = $demolisher->demolish($user);
echo json_encode($data);
// Saída: {"id":1,"name":"João Silva","email":"[email protected]","isActive":true,"tags":["admin","usuario"]}
Constructo suporta formatadores personalizados para transformação de dados durante a deserialização:
<?php
use Constructo\Core\Serialize\Builder;
// Formatador personalizado para arrays
class ArrayFormatter
{
public function __invoke($value)
{
return is_string($value) ? json_decode($value, true) : $value;
}
}
// Use com Builder
$builder = new Builder(formatters: [
'array' => new ArrayFormatter(),
]);
$data = [
'id' => 1,
'name' => 'Maria Santos',
'tags' => '["desenvolvedor", "php"]' // String JSON será convertida para array
];
$user = $builder->build(User::class, Set::createFrom($data));
echo implode(', ', $user->tags); // "desenvolvedor, php"
<?php
class Address extends Entity
{
public function __construct(
public readonly string $street,
public readonly string $city,
public readonly string $country
) {}
}
class User extends Entity
{
public function __construct(
public readonly int $id,
public readonly string $name,
public readonly Address $address,
public readonly ?DateTime $createdAt = null
) {}
}
// Dados aninhados
$data = [
'id' => 1,
'name' => 'João Silva',
'address' => [
'street' => 'Rua Principal, 123',
'city' => 'São Paulo',
'country' => 'Brasil'
],
'created_at' => '2023-01-15T10:30:00+00:00'
];
$builder = new Builder();
$user = $builder->build(User::class, Set::createFrom($data));
echo $user->address->city; // "São Paulo"
echo $user->createdAt->format('d/m/Y'); // "15/01/2023"
<?php
enum Status: string
{
case ACTIVE = 'ativo';
case INACTIVE = 'inativo';
case PENDING = 'pendente';
}
class Order extends Entity
{
public function __construct(
public readonly int $id,
public readonly Status $status,
public readonly float $amount
) {}
}
$data = [
'id' => 1,
'status' => 'ativo', // String será convertida para enum
'amount' => 99.99
];
$builder = new Builder();
$order = $builder->build(Order::class, Set::createFrom($data));
echo $order->status->value; // "ativo"
Quando a deserialização falha, o Constructo fornece informações detalhadas de erro:
<?php
use Constructo\Support\Datum;
use Constructo\Exception\AdapterException;
try {
$result = $builder->build(User::class, Set::createFrom($invalidData));
} catch (AdapterException $e) {
// Crie um objeto Datum com detalhes do erro
$datum = new Datum($e, $invalidData);
$errorData = $datum->export();
// Contém dados originais mais '@error' com detalhes da exceção
}
<?php
use Constructo\Core\Deserialize\Demolisher;
// Formatador de string personalizado
$stringFormatter = fn($value) => sprintf('[%s]', $value);
// Use com Demolisher
$demolisher = new Demolisher(formatters: [
'string' => $stringFormatter,
]);
$user = new User(1, 'Ana Costa', '[email protected]');
$data = $demolisher->demolish($user);
echo $data->name; // "[Ana Costa]"
<?php
use Constructo\Contract\Collectable;
use Constructo\Type\Collection;
class UserCollection extends Collection implements Collectable
{
protected function getItemClass(): string
{
return User::class;
}
}
// Serialize coleção
$collection = new UserCollection();
$collection->push($user1);
$collection->push($user2);
$demolisher = new Demolisher();
$arrayData = $demolisher->demolishCollection($collection);
Constructo inclui várias funções utilitárias para operações comuns:
<?php
use function Constructo\Json\decode;
use function Constructo\Json\encode;
$array = decode('{"name":"João","age":30}');
$json = encode(['name' => 'João', 'age' => 30]);
<?php
use function Constructo\Cast\arrayify;
use function Constructo\Cast\stringify;
$array = arrayify($data); // Converte para array com segurança
$string = stringify($value); // Converte para string com segurança
<?php
use function Constructo\Util\extractString;
use function Constructo\Util\extractInt;
use function Constructo\Util\extractBool;
use function Constructo\Util\extractArray;
$name = extractString($data, 'name', 'padrão');
$age = extractInt($data, 'age', 0);
$active = extractBool($data, 'is_active', false);
$tags = extractArray($data, 'tags', []);
Constructo fornece utilitários de teste para facilitar os testes:
<?php
use Constructo\Testing\BuilderExtension;
use Constructo\Testing\MakeExtension;
use PHPUnit\Framework\TestCase;
class MyTest extends TestCase
{
use BuilderExtension, MakeExtension;
public function testSerialization(): void
{
$user = $this->builder()->build(User::class, Set::createFrom($data));
$this->assertInstanceOf(User::class, $user);
}
}
Estenda a classe Entity
para obter suporte automático de serialização:
<?php
use Constructo\Support\Entity;
class MyEntity extends Entity
{
// Automaticamente implementa Exportable e JsonSerializable
// Fornece método export() que retorna objeto com todas as propriedades públicas
}
A classe Set
é usada para gerenciar coleções de dados com segurança de tipos:
<?php
use Constructo\Support\Set;
$set = Set::createFrom(['key' => 'valor']);
$set = new Set(['key' => 'valor']);
$value = $set->get('key', 'padrão');
$array = $set->toArray();
Para manipular valores individuais com validação e transformação:
<?php
use Constructo\Support\Value;
$value = new Value('alguns dados');
// Fornece vários métodos para manipulação e validação de valores
Constructo pode gerar schemas para seus objetos:
<?php
use Constructo\Factory\SchemaFactory;
use Constructo\Factory\DefaultSpecsFactory;
$schemaFactory = new SchemaFactory(new DefaultSpecsFactory());
$schema = $schemaFactory->make();
Capacidades avançadas de reflexão para introspecção de objetos:
<?php
use Constructo\Support\Reflective\Engine;
use Constructo\Factory\ReflectorFactory;
$reflectorFactory = new ReflectorFactory();
$reflector = $reflectorFactory->make();
Suporte integrado de cache para melhor performance:
<?php
use Constructo\Support\Cache;
$cache = new Cache();
// Fornece mecanismos de cache para dados de reflexão e schemas
Contribuições são bem-vindas! Sinta-se à vontade para enviar um Pull Request. Para mudanças importantes, abra primeiro uma issue para discutir o que você gostaria de alterar.
- Clone o repositório
- Instale as dependências:
composer install
- Execute os testes:
composer test
- Execute o linting:
composer lint:phpcs
- Execute análise estática:
composer lint:phpstan
O projeto usa várias ferramentas de qualidade de código:
- PHPUnit para testes
- PHPStan para análise estática
- PHP_CodeSniffer para estilo de código
- PHPMD para detecção de bagunça
- Psalm para análise estática adicional
- Rector para modernização de código
Este projeto está licenciado sob a Licença MIT - veja o arquivo LICENSE para detalhes.
Constructo é desenvolvido e mantido pela Devitools. Nós nos especializamos em criar ferramentas de desenvolvimento poderosas e bibliotecas para aplicações web modernas.
Para mais informações e exemplos de uso avançado, visite nossa documentação em devi.tools/constructo.