-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
0f3027f
commit 232495a
Showing
17 changed files
with
623 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
root = true | ||
|
||
[*] | ||
charset = utf-8 | ||
indent_size = 4 | ||
indent_style = space | ||
end_of_line = lf | ||
insert_final_newline = true | ||
trim_trailing_whitespace = true | ||
|
||
[*.{yml,yaml}] | ||
indent_size = 2 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,42 @@ | ||
name: "Tests" | ||
|
||
on: | ||
push: | ||
branches: | ||
- 'main' | ||
pull_request: | ||
|
||
permissions: | ||
contents: read | ||
|
||
jobs: | ||
tests: | ||
name: "Package Tests" | ||
runs-on: ubuntu-20.04 | ||
continue-on-error: false | ||
|
||
steps: | ||
- name: "Checkout" | ||
uses: "actions/checkout@v4" | ||
with: | ||
ref: ${{ github.event.pull_request.head.ref }} | ||
fetch-depth: 100 | ||
|
||
- name: "Install PHP" | ||
uses: "shivammathur/setup-php@v2" | ||
with: | ||
ini-values: "memory_limit=-1,display_errors=1" | ||
php-version: "8.3" | ||
coverage: "pcov" | ||
|
||
- name: "Install Dependencies" | ||
run: "composer install" | ||
|
||
- name: "Run Tests" | ||
run: "vendor/bin/phpunit --coverage-php /tmp/${{ github.sha }}_coverage.cov" | ||
|
||
- uses: "actions/upload-artifact@v4" | ||
with: | ||
name: "tests_coverage" | ||
path: "/tmp/${{ github.sha }}_coverage.cov" | ||
retention-days: 1 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
{ | ||
"name": "andreypostal/json-handler-php", | ||
"description": "Just a light and simple JSON helper that will make it easy for you to deal with json and objects", | ||
"type": "library", | ||
"license": "MIT", | ||
"authors": [ | ||
{ | ||
"name": "Andrey Postal", | ||
"email": "andreypostal@gmail.com" | ||
} | ||
], | ||
"minimum-stability": "dev", | ||
"autoload": { | ||
"psr-4": { | ||
"Andrey\\JsonHandler\\": "src/" | ||
} | ||
}, | ||
"require-dev": { | ||
"phpunit/phpunit": "^11.4@dev" | ||
}, | ||
"scripts": { | ||
"test": "phpunit" | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
<?xml version="1.0" encoding="UTF-8"?> | ||
|
||
<phpunit | ||
backupGlobals="false" | ||
backupStaticProperties="false" | ||
defaultTestSuite="Test Suite" | ||
colors="true" | ||
processIsolation= "false" | ||
stopOnFailure="false" | ||
bootstrap = "tests/bootstrap.php" | ||
> | ||
<testsuites> | ||
<testsuite name="Test Suite"> | ||
<directory>tests</directory> | ||
<exclude>src/</exclude> | ||
</testsuite> | ||
</testsuites> | ||
|
||
<coverage/> | ||
|
||
<source> | ||
<include> | ||
<directory suffix=".php">src/</directory> | ||
</include> | ||
</source> | ||
</phpunit> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,27 @@ | ||
<?php | ||
|
||
namespace Andrey\JsonHandler; | ||
|
||
use JsonException; | ||
|
||
final class JsonHandler | ||
{ | ||
use JsonSerializerTrait; | ||
use JsonHydratorTrait; | ||
|
||
/** | ||
* @throws JsonException | ||
*/ | ||
public static function Decode(string $json): array | ||
{ | ||
return json_decode($json, true, 512, JSON_THROW_ON_ERROR); | ||
} | ||
|
||
/** | ||
* @throws JsonException | ||
*/ | ||
public static function Encode(array $jsonArr): string | ||
{ | ||
return json_encode($jsonArr, JSON_THROW_ON_ERROR); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,92 @@ | ||
<?php | ||
namespace Andrey\JsonHandler; | ||
|
||
use InvalidArgumentException; | ||
use JsonException; | ||
use LogicException; | ||
use ReflectionClass; | ||
use ReflectionProperty; | ||
|
||
trait JsonHydratorTrait | ||
{ | ||
/** | ||
* @throws JsonException | ||
*/ | ||
public function hydrateObjectImmutable(string|array $json, object $obj): object | ||
{ | ||
return $this->hydrateObject($json, clone $obj); | ||
} | ||
|
||
/** | ||
* @throws JsonException | ||
*/ | ||
public function hydrateObject(string|array $json, object $obj): object | ||
{ | ||
$jsonArr = is_string($json) ? JsonHandler::Decode($json) : $json; | ||
$reflectionClass = new ReflectionClass($obj); | ||
$data = $this->processClass($reflectionClass, $jsonArr); | ||
if ($reflectionClass->hasMethod('hydrate')) { | ||
$obj->hydrate($data); | ||
} else { | ||
foreach ($data as $key => $value) { | ||
$obj->{$key} = $value; | ||
} | ||
} | ||
return $obj; | ||
} | ||
|
||
/** | ||
* @throws JsonException | ||
*/ | ||
private function processClass(ReflectionClass $class, array $jsonArr): array | ||
{ | ||
$output = []; | ||
$properties = $class->getProperties(); | ||
foreach ($properties as $property) { | ||
$output[$property->getName()] = $this->processProperty($property, $jsonArr); | ||
} | ||
return $output; | ||
} | ||
|
||
/** | ||
* @throws JsonException | ||
*/ | ||
private function processProperty(ReflectionProperty $property, array $jsonArr): mixed | ||
{ | ||
$attributes = $property->getAttributes(JsonItemAttribute::class); | ||
$attr = $attributes[0] ?? null; | ||
if ($attr === null) { | ||
return null; | ||
} | ||
|
||
/** @var JsonItemAttribute $item */ | ||
$item = $attr->newInstance(); | ||
$key = $item->key ?? $property->getName(); | ||
if ($item->required && !array_key_exists($key, $jsonArr)) { | ||
throw new InvalidArgumentException(sprintf('required item <%s> not found', $key)); | ||
} | ||
|
||
if ($property->getType()?->isBuiltin()) { | ||
if ($item->type !== null && $property->getType()?->getName() === 'array') { | ||
$output = []; | ||
$classExists = class_exists($item->type); | ||
foreach ($jsonArr[$key] ?? [] as $k => $v) { | ||
$value = $v; | ||
if ($classExists) { | ||
$value = $this->hydrateObject($v, new $item->type); | ||
} elseif (gettype($v) !== $item->type) { | ||
throw new LogicException(sprintf('expected array with items of type <%s> but found <%s>', $item->type, gettype($v))); | ||
} | ||
$output[$k] = $value; | ||
} | ||
return $output; | ||
} | ||
return $jsonArr[$key] ?? $property->getDefaultValue(); | ||
} | ||
|
||
return $this->hydrateObject( | ||
$jsonArr[$key], | ||
new ($property->getType()?->getName())(), | ||
); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
<?php | ||
namespace Andrey\JsonHandler; | ||
|
||
use Attribute; | ||
|
||
#[Attribute(Attribute::TARGET_PROPERTY)] | ||
class JsonItemAttribute | ||
{ | ||
public function __construct(public ?string $key = null, public bool $required = false, public ?string $type = null) {} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
<?php | ||
namespace Andrey\JsonHandler; | ||
|
||
use ReflectionClass; | ||
|
||
trait JsonSerializerTrait | ||
{ | ||
public function serialize(object $obj): array | ||
{ | ||
$class = new ReflectionClass($obj); | ||
$output = []; | ||
$properties = $class->getProperties(); | ||
foreach ($properties as $property) { | ||
$attributes = $property->getAttributes(JsonItemAttribute::class); | ||
$attr = $attributes[0] ?? null; | ||
if ($attr === null) { | ||
continue; | ||
} | ||
/** @var JsonItemAttribute $item */ | ||
$item = $attr->newInstance(); | ||
$key = $item->key ?? $property->name; | ||
|
||
if ($property->getType()?->isBuiltin()) { | ||
$output[$key] = $property->getValue($obj); | ||
continue; | ||
} | ||
$output[$key] = $this->serialize($property->getValue($obj)); | ||
} | ||
return $output; | ||
} | ||
} |
Oops, something went wrong.