Implements automatic creation of required plugin pages (Community Login, Trainer Registration, Trainer Dashboard) upon plugin activation. This addresses E2E test failures caused by missing pages in the test environment. - Adds activation hook in `hvac-community-events.php` to call `hvac_ce_create_required_pages`. - The callback function checks for existing pages by slug and creates them using `wp_insert_post` if missing. Includes debug logging. Also fixes issues identified during E2E test debugging: - Corrects fatal error in `includes/community/class-login-handler.php` by replacing undefined constant `HVAC_COMMUNITY_EVENTS_PATH` with `HVAC_CE_PLUGIN_DIR`. - Updates `tests/e2e/tests/login.spec.ts` to use the correct selector `#wp-submit` for the login form submit button instead of `button[type="submit"]`. Documentation updates: - Adds `docs/automatic-page-creation-plan.md`. - Updates `README.md` regarding automatic page creation. - Updates Memory Bank files (`decisionLog.md`, `progress.md`, `activeContext.md`). Note: Activation hook logging did not appear during WP-CLI activation, requiring further investigation if page creation issues persist. E2E test confirmation pending.
108 lines
3.4 KiB
PHP
108 lines
3.4 KiB
PHP
<?php declare(strict_types=1);
|
|
|
|
namespace PhpParser;
|
|
|
|
class JsonDecoder {
|
|
/** @var \ReflectionClass<Node>[] Node type to reflection class map */
|
|
private array $reflectionClassCache;
|
|
|
|
/** @return mixed */
|
|
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);
|
|
}
|
|
|
|
/**
|
|
* @param mixed $value
|
|
* @return mixed
|
|
*/
|
|
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;
|
|
}
|
|
|
|
private function decodeArray(array $array): array {
|
|
$decodedArray = [];
|
|
foreach ($array as $key => $value) {
|
|
$decodedArray[$key] = $this->decodeRecursive($value);
|
|
}
|
|
return $decodedArray;
|
|
}
|
|
|
|
private function decodeNode(array $value): Node {
|
|
$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;
|
|
}
|
|
|
|
private function decodeComment(array $value): Comment {
|
|
$className = $value['nodeType'] === 'Comment' ? Comment::class : Comment\Doc::class;
|
|
if (!isset($value['text'])) {
|
|
throw new \RuntimeException('Comment must have text');
|
|
}
|
|
|
|
return new $className(
|
|
$value['text'],
|
|
$value['line'] ?? -1, $value['filePos'] ?? -1, $value['tokenPos'] ?? -1,
|
|
$value['endLine'] ?? -1, $value['endFilePos'] ?? -1, $value['endTokenPos'] ?? -1
|
|
);
|
|
}
|
|
|
|
/** @return \ReflectionClass<Node> */
|
|
private function reflectionClassFromNodeType(string $nodeType): \ReflectionClass {
|
|
if (!isset($this->reflectionClassCache[$nodeType])) {
|
|
$className = $this->classNameFromNodeType($nodeType);
|
|
$this->reflectionClassCache[$nodeType] = new \ReflectionClass($className);
|
|
}
|
|
return $this->reflectionClassCache[$nodeType];
|
|
}
|
|
|
|
/** @return class-string<Node> */
|
|
private function classNameFromNodeType(string $nodeType): string {
|
|
$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\"");
|
|
}
|
|
}
|