20250222
[О] [ClassMapper::GetDefaults]: Улучшено определение типа. Теперь проверка integer не вызовет ошибку, что ожидается int. [О] [ClassMapper::MapClass]: Теперь идёт проверка свойства на доступность get и set. Свойства с только get и только set пропускаются.
This commit is contained in:
parent
054e6a7cdc
commit
ebfd42a88e
@ -16,11 +16,11 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"require": {
|
"require": {
|
||||||
"php": ">=8.4",
|
"php": "^8.4",
|
||||||
"ext-mbstring": "*"
|
"ext-mbstring": "*"
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
"phpunit/phpunit": ">=11.5.6"
|
"phpunit/phpunit": "^12.0.4"
|
||||||
},
|
},
|
||||||
"autoload": {
|
"autoload": {
|
||||||
"psr-4": {
|
"psr-4": {
|
||||||
|
538
composer.lock
generated
538
composer.lock
generated
File diff suppressed because it is too large
Load Diff
@ -9,6 +9,7 @@ use Exception;
|
|||||||
use ReflectionClass;
|
use ReflectionClass;
|
||||||
use ReflectionException;
|
use ReflectionException;
|
||||||
use stdClass;
|
use stdClass;
|
||||||
|
use Throwable;
|
||||||
use UnitEnum;
|
use UnitEnum;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -53,24 +54,54 @@ final class ClassMapper
|
|||||||
// Задаю массив свойств
|
// Задаю массив свойств
|
||||||
$properties = [];
|
$properties = [];
|
||||||
|
|
||||||
|
// Получаю имя исходного класса
|
||||||
|
$className = get_class($from);
|
||||||
|
|
||||||
|
// Получение всех свойств класса
|
||||||
|
try {
|
||||||
|
$reflection = new ReflectionClass($className);
|
||||||
|
}
|
||||||
|
catch (ReflectionException) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Получение всех свойств класса
|
// Получение всех свойств класса
|
||||||
$reflection = new ReflectionClass(get_class($from));
|
|
||||||
$props = $reflection->getProperties();
|
$props = $reflection->getProperties();
|
||||||
|
|
||||||
var_dump($props);
|
// Создаю экземпляр класса
|
||||||
|
$instance = new $className();
|
||||||
|
|
||||||
// Для каждого свойства
|
// Для каждого свойства
|
||||||
foreach ($props as $prop) {
|
foreach ($props as $prop) {
|
||||||
|
// - получаю имя свойства
|
||||||
$propName = $prop->getName();
|
$propName = $prop->getName();
|
||||||
$value = $from->$propName; // Читаем значение свойства
|
// - получаю значение свойства
|
||||||
|
$value = $from->$propName;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$getterMethod = 'get' . ucfirst($propName);
|
// - получаю тип свойства
|
||||||
$setterMethod = 'set' . ucfirst($propName);
|
$typeOf = gettype($from->$propName);
|
||||||
|
|
||||||
// Проверяем существование геттера и сеттера методами класса
|
// - получаю значение свойства по типу и по умолчанию
|
||||||
if (!(method_exists($from, $getterMethod) && method_exists($from, $setterMethod)))
|
$writeValue = self::GetDefaults($typeOf);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// - проверяем, можно ли записать и прочитать значение
|
||||||
|
// -- пытаюсь установить значение
|
||||||
|
$instance->$propName = $writeValue;
|
||||||
|
// -- пытаюсь прочитать установленное значение
|
||||||
|
$readValue = $instance->$propName;
|
||||||
|
|
||||||
|
// -- и проверяю, что значение совпадают
|
||||||
|
/** @noinspection PhpConditionAlreadyCheckedInspection */
|
||||||
|
if ($readValue !== $writeValue)
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
catch (Throwable) {
|
||||||
|
// - в случае ошибки, понимаем, что свойство доступно или только для чтения, или
|
||||||
|
// только для записи и оно нам не подходит. Поэтому пропускаем его.
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// - если свойство игнорируется
|
// - если свойство игнорируется
|
||||||
if (in_array($propName, $options['ignored']))
|
if (in_array($propName, $options['ignored']))
|
||||||
@ -85,14 +116,12 @@ final class ClassMapper
|
|||||||
// Если не было ошибки, значит свойство имеет и геттер, и сеттер
|
// Если не было ошибки, значит свойство имеет и геттер, и сеттер
|
||||||
$properties[$propName] = $value;
|
$properties[$propName] = $value;
|
||||||
}
|
}
|
||||||
catch (Error $exception) {
|
catch (Error) {
|
||||||
var_dump($exception->getMessage());
|
// - в случае ошибки, понимаем, что свойство нам не подходит. Поэтому пропускаю его.
|
||||||
// Игнорируем исключения, так как нас интересуют только свойства с обоими методами
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var_dump($properties);
|
|
||||||
|
|
||||||
// Для каждого элемента массива
|
// Для каждого элемента массива
|
||||||
foreach ($properties as $name => $value) {
|
foreach ($properties as $name => $value) {
|
||||||
// - если свойство есть в объекте
|
// - если свойство есть в объекте
|
||||||
@ -102,6 +131,26 @@ final class ClassMapper
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Возвращает значение по умолчанию для типа $typeName.
|
||||||
|
*
|
||||||
|
* @param string $typeName Тип
|
||||||
|
*
|
||||||
|
* @return mixed Значение по умолчанию
|
||||||
|
*/
|
||||||
|
public static function GetDefaults (string $typeName): mixed
|
||||||
|
{
|
||||||
|
return match (strtolower($typeName)) {
|
||||||
|
'int', 'integer' => 0,
|
||||||
|
'float', 'double' => 0.0,
|
||||||
|
'bool', 'boolean' => false,
|
||||||
|
'string' => '',
|
||||||
|
'array' => [],
|
||||||
|
'object' => new stdClass(),
|
||||||
|
default => null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Подготавливает значения свойств класса.
|
* Подготавливает значения свойств класса.
|
||||||
*
|
*
|
||||||
@ -347,25 +396,4 @@ final class ClassMapper
|
|||||||
throw new Exception($exception->getMessage());
|
throw new Exception($exception->getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Получает значение по умолчанию для разных типов данных.
|
|
||||||
*
|
|
||||||
* @param string $typeName Имя типа данных.
|
|
||||||
*
|
|
||||||
* @return mixed|null Результат.
|
|
||||||
*/
|
|
||||||
public
|
|
||||||
static function GetDefaults (string $typeName): mixed
|
|
||||||
{
|
|
||||||
return match ($typeName) {
|
|
||||||
'int' => 0,
|
|
||||||
'float' => 0.0,
|
|
||||||
'bool' => false,
|
|
||||||
'string' => '',
|
|
||||||
'array' => [],
|
|
||||||
'object' => new stdClass(),
|
|
||||||
default => null,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
@ -4,6 +4,7 @@ namespace goodboyalex\php_components_pack\tests\classes;
|
|||||||
|
|
||||||
use goodboyalex\php_components_pack\classes\ClassMapper;
|
use goodboyalex\php_components_pack\classes\ClassMapper;
|
||||||
use goodboyalex\php_components_pack\tests\data\A;
|
use goodboyalex\php_components_pack\tests\data\A;
|
||||||
|
use goodboyalex\php_components_pack\tests\data\B;
|
||||||
use PHPUnit\Framework\TestCase;
|
use PHPUnit\Framework\TestCase;
|
||||||
|
|
||||||
class ClassMapperTest extends TestCase
|
class ClassMapperTest extends TestCase
|
||||||
@ -12,15 +13,11 @@ class ClassMapperTest extends TestCase
|
|||||||
{
|
{
|
||||||
$this->PrepareForTest();
|
$this->PrepareForTest();
|
||||||
|
|
||||||
$a = new A();
|
$a = new A('a', 2, true);
|
||||||
$a->a = 'a';
|
|
||||||
$a->b = 2;
|
|
||||||
$a->c = true;
|
|
||||||
|
|
||||||
$b = new B();
|
$b = new B();
|
||||||
ClassMapper::MapClass($a, $b);
|
|
||||||
|
|
||||||
var_dump($b);
|
ClassMapper::MapClass($a, $b);
|
||||||
|
|
||||||
$this->assertEquals('a', $b->a);
|
$this->assertEquals('a', $b->a);
|
||||||
$this->assertEquals(2, $b->b);
|
$this->assertEquals(2, $b->b);
|
||||||
|
@ -1,10 +1,17 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
namespace goodboyalex\php_components_pack\tests\classes;
|
namespace goodboyalex\php_components_pack\tests\data;
|
||||||
|
|
||||||
class B
|
class B
|
||||||
{
|
{
|
||||||
public string $a;
|
public string $a;
|
||||||
public int $b;
|
public int $b;
|
||||||
public string $d;
|
public string $d;
|
||||||
|
|
||||||
|
public function __construct (string $a = "", int $b = 0, string $d = "")
|
||||||
|
{
|
||||||
|
$this->a = $a;
|
||||||
|
$this->b = $b;
|
||||||
|
$this->d = $d;
|
||||||
|
}
|
||||||
}
|
}
|
Loading…
x
Reference in New Issue
Block a user