processTags($object, $tags); } public function convert($data, array $tags = []): array { if (!is_object($data) && !is_iterable($data)) { throw new \InvalidArgumentException(); } return is_iterable($data) ? $this->convertArray($data, $tags) : $this->convertObject($data, $tags); } public function convertArray(iterable $items, array $tags = []): array { $result = []; foreach ($items as $item) { if (!is_object($item)) { throw new \InvalidArgumentException(); } $result[] = $this->convertObject($item, $tags); } return $result; } public function convertObject(object $object, array $tags = []): array { $tags = empty($tags) ? ['default'] : $tags; $schema = $this->grabSchema($object, $tags); return $this->convertObjectByMap($object, $schema->map, $tags); } public function convertObjectByMap(object $object, array $map, array $tags = []): array { $data = []; $reflectionObject = new \ReflectionObject($object); foreach ($map as $key => $propertyInfo) { $sourceName = is_array($propertyInfo) ? $key : $propertyInfo; if (!method_exists($object, $sourceName) && !property_exists($object, $sourceName)) { continue; } $value = $this->grabValue($object, $sourceName, $tags); // Если нет карты, то не сериализуем. if (is_iterable($value)) { $childMap = is_array($propertyInfo) ? $propertyInfo : null; $value = $this->convertArrayByMap($value, $childMap, $tags); } elseif (is_object($value) && is_array($propertyInfo)) { $value = $this->convertObjectByMap($value, $propertyInfo, $tags); } elseif (!is_scalar($value) && null !== $value) { continue; } $this->processIterationTransformers($object, $sourceName, $value, $tags); $dataPath = $this->grabIterationDataPath($object, $sourceName, $tags); $data[$dataPath] = $value; } $this->processClassTransformers($reflectionObject, $data, $tags); return $data; } public function convertArrayByMap($data, ?array $map, array $tags = []): ?array { // $isAssociative = count($data) && !array_key_exists(0, $data); $tempValue = []; foreach ($data as $key => $item) { if (is_scalar($item)) { $tempValue[$key] = $item; continue; } if (is_iterable($item) && $map) { $tempValue[$key] = $this->convertArrayByMap($item, $map, $tags); continue; } if (is_object($item) && $map) { $tempValue[$key] = $this->convertObjectByMap($item, $map, $tags); continue; } } return $tempValue; } protected function grabValue(object $object, $sourceName, array $tags) { if (method_exists($object, $sourceName)) { $reflectionSource = new ReflectionMethod($object, $sourceName); return $this->getMethodValue($object, $reflectionSource); } elseif (property_exists($object, $sourceName)) { $reflectionSource = new ReflectionProperty($object, $sourceName); $propertyExtractor = new PropertyExtractor($object::class, $sourceName); /** @var PropertyPath $propertyPath */ while ($propertyPath = $propertyExtractor->fetch(PropertyPath::class)) { $filteredTags = array_diff($tags, $propertyPath->tags); if (count($filteredTags) === count($tags)) { continue; } return $this->getValueByPath($object, $propertyPath->path); } return $this->getValue($object, $reflectionSource); } return null; } public function processIterationTransformers(object $object, string $sourceName, &$value, array $tags): void { if (method_exists($object, $sourceName)) { $reflectionSource = new ReflectionMethod($object, $sourceName); $this->processMethodTransformers($reflectionSource, $value, $tags); } elseif (property_exists($object, $sourceName)) { $reflectionSource = new ReflectionProperty($object, $sourceName); $this->processTransformers($reflectionSource, $value, $tags); } } public function grabIterationDataPath(object $object, string $sourceName, array $tags): string { if (method_exists($object, $sourceName)) { $reflectionSource = new ReflectionMethod($object, $sourceName); $dataPath = $this->grabMethodDataPath($reflectionSource, $tags); } elseif (property_exists($object, $sourceName)) { $reflectionSource = new ReflectionProperty($object, $sourceName); $dataPath = $this->grabDataPath($reflectionSource, $tags); } return $dataPath ?? $sourceName; } /** * Получаем теги для обработки */ protected function processTags(object $object, array $tags): array { $classExtractor = new ClassExtractor($object::class); /** @var HandleTags $tagsMeta */ if ($tagsMeta = $classExtractor->fetch(HandleTags::class)) { if (method_exists($object, $tagsMeta->method)) { $reflectionMethod = new ReflectionMethod($object, $tagsMeta->method); if (!$reflectionMethod->isPublic()) { $reflectionMethod->setAccessible(true); } $methodTags = $reflectionMethod->invoke($object, ...[$tags]); if (!$reflectionMethod->isPublic()) { $reflectionMethod->setAccessible(false); } return $methodTags; } } return $tags; } /** * Трнансформируем на уровне класса */ protected function processClassTransformers(\ReflectionObject $object, &$data, array $tags): void { $className = $object->getName(); $classExtractor = new ClassExtractor($className); /** @var Meta $transformMeta */ while ($transformMeta = $classExtractor->fetch(Meta::class)) { $transformMeta->returnType = $className; $filteredTags = array_diff($tags, $transformMeta->tags); if (count($filteredTags) === count($tags)) { continue; } $transformer = $this->grabTransformer($transformMeta); $transformer->transform($data, $transformMeta); } } /** * Трнансформируем на уровне свойст объекта */ protected function processTransformers(\ReflectionProperty $property, &$data, array $tags): void { $propertyName = $property->getName(); $propertyExtractor = new PropertyExtractor($property->class, $propertyName); /** @var Meta $transformMeta */ while ($transformMeta = $propertyExtractor->fetch(Meta::class)) { $filteredTags = array_diff($tags, $transformMeta->tags); if (count($filteredTags) === count($tags)) { continue; } /** @var \ReflectionNamedType $reflectionPropertyType */ $reflectionPropertyType = $property->getType(); $propertyType = $reflectionPropertyType->getName(); $transformMeta->returnType = $propertyType; $transformMeta->allowsNull = $reflectionPropertyType->allowsNull(); $transformer = $this->grabTransformer($transformMeta); $transformer->transform($data, $transformMeta); } } protected function processMethodTransformers(ReflectionMethod $method, &$data, array $tags): void { $methodName = $method->getName(); $methodExtractor = new MethodExtractor($method->class, $methodName); /** @var Meta $transformMeta */ while ($transformMeta = $methodExtractor->fetch(Meta::class)) { $filteredTags = array_diff($tags, $transformMeta->tags); if (count($filteredTags) === count($tags)) { continue; } /** @var \ReflectionNamedType $reflectionMethodType */ $reflectionMethodType = $method->getReturnType(); $methodType = $reflectionMethodType->getName(); $transformMeta->returnType = $methodType; $transformMeta->allowsNull = $reflectionMethodType->allowsNull(); $transformer = $this->grabTransformer($transformMeta); $transformer->transform($data, $transformMeta); } } protected function grabTransformer(Meta $meta): TransformerInterface { $storage = TransformerResolverStorage::getInstance(); $resolver = $storage->get($meta::TYPE); return $resolver->resolve($meta); } private function getValueByPath($data, string $path) { $parts = explode('.', $path); $length = count($parts); $i = 1; foreach ($parts as $part) { // Если получили скалярное значение но прошли не весь путь, то вернем null if (is_scalar($data) && $i < $length) { return null; } // Если объекс реализует ArrayAccess, то получаем значение и идем дальше if (is_object($data) && $data instanceof \ArrayAccess) { $data = $data[$part] ?? null; continue; } // Если объект, то достаем значение if (is_object($data)) { if (!property_exists($data, $part)) { return null; } $property = new ReflectionProperty($data, $part); $data = $this->getValue($data, $property); continue; } // Если массив, то достаем занчение и идем дальше if (is_array($data)) { $data = $data[$part] ?? null; continue; } $i++; } return $data; } private function getValue(object $object, \ReflectionProperty $property) { if (!$property->isPublic()) { $property->setAccessible(true); } if (!$property->isInitialized($object)) { return null; } $value = $property->getValue($object); if (!$property->isPublic()) { $property->setAccessible(false); } return $value; } private function getMethodValue(object $object, ReflectionMethod $method) { if (!$method->isPublic()) { $method->setAccessible(true); } $value = $method->invoke($object); if (!$method->isPublic()) { $method->setAccessible(false); } return $value; } private function grabSchema(object $object, array $tags): ?Schema { $classExtractor = new ClassExtractor($object::class); /** @var Schema $schema */ while ($schema = $classExtractor->fetch(Schema::class)) { $filteredTags = array_diff($tags, $schema->tags); if (count($filteredTags) === count($tags)) { continue; } return $schema; } return null; } private function grabDataPath(\ReflectionProperty $property, array $tags): ?string { $propertyName = $property->getName(); $propertyExtractor = new PropertyExtractor($property->class, $propertyName); /** @var DataPath $schema */ while ($dataPath = $propertyExtractor->fetch(DataPath::class)) { $filteredTags = array_diff($tags, $dataPath->tags); if (count($filteredTags) === count($tags)) { continue; } return $dataPath->path; } return null; } private function grabMethodDataPath(ReflectionMethod $method, array $tags): ?string { $methodName = $method->getName(); $methodExtractor = new MethodExtractor($method->class, $methodName); /** @var DataPath $schema */ while ($dataPath = $methodExtractor->fetch(DataPath::class)) { $filteredTags = array_diff($tags, $dataPath->tags); if (count($filteredTags) === count($tags)) { continue; } return $dataPath->path; } return null; } }