Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
27 changes: 14 additions & 13 deletions .github/workflows/php.yml
Original file line number Diff line number Diff line change
Expand Up @@ -12,24 +12,25 @@ jobs:
runs-on: ubuntu-latest
strategy:
matrix:
php-version: ['7.2', '7.3', '7.4', '8.0', '8.1', '8.2', '8.3', '8.4']
php-version: ['7.2', '7.3', '7.4', '8.0', '8.1', '8.2', '8.3', '8.4', '8.5']
name: PHP ${{ matrix.php-version }}
steps:
- uses: actions/checkout@v6

- name: Set up PHP ${{ matrix.php-version }}
uses: shivammathur/setup-php@v2
with:
php-version: ${{ matrix.php-version }}
extensions: mbstring

- name: Validate composer.json and composer.lock
run: composer validate
- name: Cache Composer packages
id: composer-cache
uses: actions/cache@v5
with:
path: vendor
key: ${{ runner.os }}-php-${{ hashFiles('**/composer.lock') }}
restore-keys: |
${{ runner.os }}-php-

- name: Install suggested dependencies
run: composer require jeremeamia/superclosure opis/closure
- name: Install dependencies
if: steps.composer-cache.outputs.cache-hit != 'true'
run: composer install --prefer-dist --no-progress --no-suggest
run: composer require jeremeamia/superclosure opis/closure --no-update --no-interaction --ansi

- name: Install Composer dependencies
uses: ramsey/composer-install@v4

- name: Run test suite
run: vendor/bin/phpunit
4 changes: 3 additions & 1 deletion phpunit.xml
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,9 @@
colors="true">
<testsuites>
<testsuite name="JsonSerializerTests">
<directory>./tests</directory>
<file phpVersion="8.1.0">./tests/EnumSerializationTest.php</file>
<file>./tests/JsonSerializerTest.php</file>
<directory>./tests/ClosureSerializer</directory>
</testsuite>
</testsuites>
<filter>
Expand Down
14 changes: 9 additions & 5 deletions src/JsonSerializer/JsonSerializer.php
Original file line number Diff line number Diff line change
Expand Up @@ -308,10 +308,10 @@ protected function serializeData($value)
*/
protected function serializeObject($value)
{
if ($this->objectStorage->contains($value)) {
if ($this->objectStorage->offsetExists($value)) {
return [static::CLASS_IDENTIFIER_KEY => '@' . $this->objectStorage[$value]];
}
$this->objectStorage->attach($value, $this->objectMappingIndex++);
$this->objectStorage->offsetSet($value, $this->objectMappingIndex++);

$ref = new ReflectionClass($value);
$className = $ref->getName();
Expand Down Expand Up @@ -402,8 +402,10 @@ protected function extractObjectData($value, $ref, $properties)
foreach ($properties as $property) {
try {
$propRef = $this->getReflectionProperty($ref, $property);
$propRef->setAccessible(true);
if (!$propRef->isInitialized($value)) {
if (PHP_VERSION_ID < 80100) {
$propRef->setAccessible(true);
}
if (PHP_VERSION_ID >= 70400 && !$propRef->isInitialized($value)) {
continue;
}
$data[$property] = $propRef->getValue($value);
Expand Down Expand Up @@ -548,7 +550,9 @@ protected function unserializeObject($value)
foreach ($value as $property => $propertyValue) {
try {
$propRef = $this->getReflectionProperty($ref, $property);
$propRef->setAccessible(true);
if (PHP_VERSION_ID < 80100) {
$propRef->setAccessible(true);
}
$propRef->setValue($obj, $this->unserializeData($propertyValue));
} catch (ReflectionException $e) {
switch ($this->undefinedAttributeMode) {
Expand Down
2 changes: 1 addition & 1 deletion tests/ClosureSerializer/ClosureSerializerManagerTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ class ClosureSerializerManagerTest extends TestCase
{
public function setUp(): void
{
if (! class_exists(\SuperClosure\SerializerInterface::class)) {
if (! interface_exists(\SuperClosure\SerializerInterface::class)) {
$this->markTestSkipped('Missing jeremeamia/superclosure to run this test');
}
}
Expand Down
2 changes: 1 addition & 1 deletion tests/ClosureSerializer/SuperClosureSerializerTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ class SuperClosureSerializerTest extends TestCase
{
public function setUp(): void
{
if (! class_exists(\SuperClosure\SerializerInterface::class)) {
if (! interface_exists(\SuperClosure\SerializerInterface::class)) {
$this->markTestSkipped('Missing jeremeamia/superclosure to run this test');
}
}
Expand Down
145 changes: 145 additions & 0 deletions tests/EnumSerializationTest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,145 @@
<?php

namespace Zumba\JsonSerializer\Test;

use PHPUnit\Framework\TestCase;
use stdClass;
use Zumba\JsonSerializer\JsonSerializer;

final class EnumSerializationTest extends TestCase
{
/**
* Serializer instance
*
* @var JsonSerializer
*/
protected $serializer;

/**
* Test case setup
*
* @before
* @return void
*/
#[\PHPUnit\Framework\Attributes\Before]
public function setUpSerializer()
{
$customObjectSerializerMap['Zumba\\JsonSerializer\\Test\\SupportClasses\\MyType'] = new \Zumba\JsonSerializer\Test\SupportClasses\MyTypeSerializer();
$this->serializer = new JsonSerializer(null, $customObjectSerializerMap);
}

/**
* Test serialization of Enums
*
* @return void
*/
public function testSerializeEnums()
{
$unitEnum = SupportEnums\MyUnitEnum::Hearts;
$expected = '{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyUnitEnum","name":"Hearts"}';
$this->assertSame($expected, $this->serializer->serialize($unitEnum));

$backedEnum = SupportEnums\MyBackedEnum::Hearts;
$expected = '{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyBackedEnum","name":"Hearts","value":"H"}';
$this->assertSame($expected, $this->serializer->serialize($backedEnum));

$intBackedEnum = SupportEnums\MyIntBackedEnum::One;
$expected = '{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyIntBackedEnum","name":"One","value":1}';
$this->assertSame($expected, $this->serializer->serialize($intBackedEnum));
}

/**
* Test serialization of multiple Enums
*
* @return void
*/
public function testSerializeMultipleEnums()
{
$obj = new stdClass();
$obj->enum1 = SupportEnums\MyUnitEnum::Hearts;
$obj->enum2 = SupportEnums\MyBackedEnum::Hearts;
$obj->enum3 = SupportEnums\MyIntBackedEnum::One;
$obj->enum4 = SupportEnums\MyUnitEnum::Hearts;
$obj->enum5 = SupportEnums\MyBackedEnum::Hearts;
$obj->enum6 = SupportEnums\MyIntBackedEnum::One;

$expected = '{"@type":"stdClass","enum1":{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyUnitEnum","name":"Hearts"},"enum2":{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyBackedEnum","name":"Hearts","value":"H"},"enum3":{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyIntBackedEnum","name":"One","value":1},"enum4":{"@type":"@1"},"enum5":{"@type":"@2"},"enum6":{"@type":"@3"}}';
$this->assertSame($expected, $this->serializer->serialize($obj));
}

/**
* Test unserialization of Enums
*
* @return void
*/
public function testUnserializeEnums()
{
$serialized = '{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyUnitEnum","name":"Hearts"}';
$obj = $this->serializer->unserialize($serialized);
$this->assertInstanceOf('Zumba\JsonSerializer\Test\SupportEnums\MyUnitEnum', $obj);
$this->assertSame(SupportEnums\MyUnitEnum::Hearts, $obj);

$serialized = '{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyBackedEnum","name":"Hearts","value":"H"}';
$obj = $this->serializer->unserialize($serialized);
$this->assertInstanceOf('Zumba\JsonSerializer\Test\SupportEnums\MyBackedEnum', $obj);
$this->assertSame(SupportEnums\MyBackedEnum::Hearts, $obj);

$serialized = '{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyIntBackedEnum","name":"Two","value":2}';
$obj = $this->serializer->unserialize($serialized);
$this->assertInstanceOf('Zumba\JsonSerializer\Test\SupportEnums\MyIntBackedEnum', $obj);
$this->assertSame(SupportEnums\MyIntBackedEnum::Two, $obj);
$this->assertSame(SupportEnums\MyIntBackedEnum::Two->value, $obj->value);

// wrong value of BackedEnum is ignored
$serialized = '{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyBackedEnum","name":"Hearts","value":"S"}';
$obj = $this->serializer->unserialize($serialized);
$this->assertInstanceOf('Zumba\JsonSerializer\Test\SupportEnums\MyBackedEnum', $obj);
$this->assertSame(SupportEnums\MyBackedEnum::Hearts, $obj);
$this->assertSame(SupportEnums\MyBackedEnum::Hearts->value, $obj->value);
}

/**
* Test unserialization of multiple Enums
*
* @return void
*/
public function testUnserializeMultipleEnums()
{
$obj = new stdClass();
$obj->enum1 = SupportEnums\MyUnitEnum::Hearts;
$obj->enum2 = SupportEnums\MyBackedEnum::Hearts;
$obj->enum3 = SupportEnums\MyIntBackedEnum::One;
$obj->enum4 = SupportEnums\MyUnitEnum::Hearts;
$obj->enum5 = SupportEnums\MyBackedEnum::Hearts;
$obj->enum6 = SupportEnums\MyIntBackedEnum::One;

$serialized = '{"@type":"stdClass","enum1":{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyUnitEnum","name":"Hearts"},"enum2":{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyBackedEnum","name":"Hearts","value":"H"},"enum3":{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyIntBackedEnum","name":"One","value":1},"enum4":{"@type":"@1"},"enum5":{"@type":"@2"},"enum6":{"@type":"@3"}}';
$actualObj = $this->serializer->unserialize($serialized);
$this->assertInstanceOf('stdClass', $actualObj);
$this->assertEquals($obj, $actualObj);
}

/**
* Test unserialization of wrong UnitEnum
*
* @return void
*/
public function testUnserializeWrongUnitEnum() {
// bad case generate Error
$serialized = '{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyUnitEnum","name":"Circles"}';
$this->expectException(\Error::class);
$this->serializer->unserialize($serialized);
}

/**
* Test unserialization of wrong BackedEnum
*
* @return void
*/
public function testUnserializeWrongBackedEnum() {
// bad case generate Error
$serialized = '{"@type":"Zumba\\\\JsonSerializer\\\\Test\\\\SupportEnums\\\\MyBackedEnum","name":"Circles","value":"C"}';
$this->expectException(\Error::class);
$this->serializer->unserialize($serialized);
}
}
Loading