2017-08-18 23:56:12 +02:00
|
|
|
<?php declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace PhpParser;
|
|
|
|
|
2022-08-28 22:57:06 +02:00
|
|
|
class JsonDecoder {
|
2022-09-11 22:33:18 +02:00
|
|
|
/** @var \ReflectionClass<Node>[] Node type to reflection class map */
|
2023-08-16 21:18:30 +02:00
|
|
|
private array $reflectionClassCache;
|
2017-08-18 23:56:12 +02:00
|
|
|
|
2022-09-11 17:51:59 +02:00
|
|
|
/** @return mixed */
|
2017-08-18 23:56:12 +02:00
|
|
|
public function decode(string $json) {
|
|
|
|
$value = json_decode($json, true);
|
|
|
|
if (json_last_error()) {
|
|
|
|
throw new \RuntimeException('JSON decoding error: ' . json_last_error_msg());
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->decodeRecursive($value);
|
|
|
|
}
|
|
|
|
|
2022-09-11 17:51:59 +02:00
|
|
|
/**
|
|
|
|
* @param mixed $value
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-08-18 23:56:12 +02:00
|
|
|
private function decodeRecursive($value) {
|
|
|
|
if (\is_array($value)) {
|
|
|
|
if (isset($value['nodeType'])) {
|
|
|
|
if ($value['nodeType'] === 'Comment' || $value['nodeType'] === 'Comment_Doc') {
|
|
|
|
return $this->decodeComment($value);
|
|
|
|
}
|
|
|
|
return $this->decodeNode($value);
|
|
|
|
}
|
|
|
|
return $this->decodeArray($value);
|
|
|
|
}
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
|
2022-08-28 22:57:06 +02:00
|
|
|
private function decodeArray(array $array): array {
|
2017-08-18 23:56:12 +02:00
|
|
|
$decodedArray = [];
|
|
|
|
foreach ($array as $key => $value) {
|
|
|
|
$decodedArray[$key] = $this->decodeRecursive($value);
|
|
|
|
}
|
|
|
|
return $decodedArray;
|
|
|
|
}
|
|
|
|
|
2022-08-28 22:57:06 +02:00
|
|
|
private function decodeNode(array $value): Node {
|
2017-08-18 23:56:12 +02:00
|
|
|
$nodeType = $value['nodeType'];
|
|
|
|
if (!\is_string($nodeType)) {
|
|
|
|
throw new \RuntimeException('Node type must be a string');
|
|
|
|
}
|
|
|
|
|
|
|
|
$reflectionClass = $this->reflectionClassFromNodeType($nodeType);
|
|
|
|
$node = $reflectionClass->newInstanceWithoutConstructor();
|
|
|
|
|
|
|
|
if (isset($value['attributes'])) {
|
|
|
|
if (!\is_array($value['attributes'])) {
|
|
|
|
throw new \RuntimeException('Attributes must be an array');
|
|
|
|
}
|
|
|
|
|
|
|
|
$node->setAttributes($this->decodeArray($value['attributes']));
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($value as $name => $subNode) {
|
|
|
|
if ($name === 'nodeType' || $name === 'attributes') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$node->$name = $this->decodeRecursive($subNode);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $node;
|
|
|
|
}
|
|
|
|
|
2022-08-28 22:57:06 +02:00
|
|
|
private function decodeComment(array $value): Comment {
|
2017-08-18 23:56:12 +02:00
|
|
|
$className = $value['nodeType'] === 'Comment' ? Comment::class : Comment\Doc::class;
|
|
|
|
if (!isset($value['text'])) {
|
|
|
|
throw new \RuntimeException('Comment must have text');
|
|
|
|
}
|
|
|
|
|
2017-11-04 17:45:14 +01:00
|
|
|
return new $className(
|
2020-02-09 16:53:46 +01:00
|
|
|
$value['text'],
|
|
|
|
$value['line'] ?? -1, $value['filePos'] ?? -1, $value['tokenPos'] ?? -1,
|
2020-02-09 17:15:59 +01:00
|
|
|
$value['endLine'] ?? -1, $value['endFilePos'] ?? -1, $value['endTokenPos'] ?? -1
|
2017-11-04 17:45:14 +01:00
|
|
|
);
|
2017-08-18 23:56:12 +02:00
|
|
|
}
|
|
|
|
|
2022-09-11 22:33:18 +02:00
|
|
|
/** @return \ReflectionClass<Node> */
|
2022-08-28 22:57:06 +02:00
|
|
|
private function reflectionClassFromNodeType(string $nodeType): \ReflectionClass {
|
2017-08-18 23:56:12 +02:00
|
|
|
if (!isset($this->reflectionClassCache[$nodeType])) {
|
|
|
|
$className = $this->classNameFromNodeType($nodeType);
|
|
|
|
$this->reflectionClassCache[$nodeType] = new \ReflectionClass($className);
|
|
|
|
}
|
|
|
|
return $this->reflectionClassCache[$nodeType];
|
|
|
|
}
|
|
|
|
|
2022-09-17 20:33:04 +02:00
|
|
|
/** @return class-string<Node> */
|
2022-08-28 22:57:06 +02:00
|
|
|
private function classNameFromNodeType(string $nodeType): string {
|
2017-08-18 23:56:12 +02:00
|
|
|
$className = 'PhpParser\\Node\\' . strtr($nodeType, '_', '\\');
|
|
|
|
if (class_exists($className)) {
|
|
|
|
return $className;
|
|
|
|
}
|
|
|
|
|
|
|
|
$className .= '_';
|
|
|
|
if (class_exists($className)) {
|
|
|
|
return $className;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new \RuntimeException("Unknown node type \"$nodeType\"");
|
|
|
|
}
|
2018-01-10 15:04:06 -02:00
|
|
|
}
|