Skip to content

Commit aead1a5

Browse files
committed
Split metadata and AST representation
1 parent 2455e99 commit aead1a5

File tree

5 files changed

+119
-67
lines changed

5 files changed

+119
-67
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\Mapper\Mapping\Introspection;
6+
7+
use TypeLang\Mapper\Mapping\Metadata\ClassMetadata;
8+
use TypeLang\Mapper\Mapping\Metadata\PropertyMetadata;
9+
use TypeLang\Mapper\Runtime\Context;
10+
use TypeLang\Parser\Node\Stmt\NamedTypeNode;
11+
use TypeLang\Parser\Node\Stmt\Shape\FieldNode;
12+
use TypeLang\Parser\Node\Stmt\Shape\FieldsListNode;
13+
use TypeLang\Parser\Node\Stmt\Shape\NamedFieldNode;
14+
use TypeLang\Parser\Node\Stmt\TypeStatement;
15+
16+
final class ClassIntrospection implements IntrospectionInterface
17+
{
18+
public function __construct(
19+
/**
20+
* @var ClassMetadata<object>
21+
*/
22+
private readonly ClassMetadata $metadata,
23+
) {}
24+
25+
public function getTypeStatement(Context $context): TypeStatement
26+
{
27+
if (!$context->isDetailedTypes()) {
28+
return new NamedTypeNode($this->metadata->getName());
29+
}
30+
31+
$fields = [];
32+
33+
foreach ($this->metadata->getProperties() as $property) {
34+
$fields[] = $this->getFieldNode($property, $context);
35+
}
36+
37+
if ($fields === []) {
38+
return new NamedTypeNode($this->metadata->getName());
39+
}
40+
41+
return new NamedTypeNode($this->metadata->getName(), fields: new FieldsListNode($fields));
42+
}
43+
44+
private function getFieldNode(PropertyMetadata $metadata, Context $context): FieldNode
45+
{
46+
$name = $metadata->getName();
47+
48+
if ($context->isDenormalization()) {
49+
$name = $metadata->getExportName();
50+
}
51+
52+
return new NamedFieldNode(
53+
key: $name,
54+
of: $metadata->getTypeStatement($context),
55+
optional: $metadata->hasDefaultValue(),
56+
);
57+
}
58+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\Mapper\Mapping\Introspection;
6+
7+
use TypeLang\Mapper\Runtime\Context;
8+
use TypeLang\Parser\Node\Stmt\TypeStatement;
9+
10+
interface IntrospectionInterface
11+
{
12+
/**
13+
* Gets an abstract syntax tree (AST) representation of the structure.
14+
*/
15+
public function getTypeStatement(Context $context): TypeStatement;
16+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\Mapper\Mapping\Introspection;
6+
7+
use TypeLang\Mapper\Mapping\Metadata\PropertyMetadata;
8+
use TypeLang\Mapper\Runtime\Context;
9+
use TypeLang\Parser\Node\Stmt\NamedTypeNode;
10+
use TypeLang\Parser\Node\Stmt\TypeStatement;
11+
12+
final class PropertyIntrospection implements IntrospectionInterface
13+
{
14+
public function __construct(
15+
private readonly PropertyMetadata $metadata,
16+
) {}
17+
18+
public function getTypeStatement(Context $context): TypeStatement
19+
{
20+
$info = $this->metadata->findTypeInfo();
21+
22+
if ($info === null) {
23+
return new NamedTypeNode('mixed');
24+
}
25+
26+
$statement = clone $info->getTypeStatement();
27+
28+
if ($context->isDetailedTypes() || !$statement instanceof NamedTypeNode) {
29+
return $statement;
30+
}
31+
32+
return new NamedTypeNode($statement->name);
33+
}
34+
}

src/Mapping/Metadata/ClassMetadata.php

+5-24
Original file line numberDiff line numberDiff line change
@@ -4,17 +4,17 @@
44

55
namespace TypeLang\Mapper\Mapping\Metadata;
66

7+
use TypeLang\Mapper\Mapping\Introspection\ClassIntrospection;
8+
use TypeLang\Mapper\Mapping\Introspection\IntrospectionInterface;
79
use TypeLang\Mapper\Runtime\Context;
8-
use TypeLang\Parser\Node\Stmt\NamedTypeNode;
9-
use TypeLang\Parser\Node\Stmt\Shape\FieldsListNode;
1010
use TypeLang\Parser\Node\Stmt\TypeStatement;
1111

1212
/**
1313
* Represents an abstraction over general information about a class.
1414
*
1515
* @template T of object
1616
*/
17-
final class ClassMetadata extends Metadata
17+
final class ClassMetadata extends Metadata implements IntrospectionInterface
1818
{
1919
/**
2020
* Contains a list of class fields available for
@@ -66,27 +66,8 @@ public function __construct(
6666
*/
6767
public function getTypeStatement(Context $context): TypeStatement
6868
{
69-
if (!$context->isDetailedTypes()) {
70-
return new NamedTypeNode($this->getName());
71-
}
72-
73-
$fields = [];
74-
75-
foreach ($this->getProperties() as $property) {
76-
$field = $property->getFieldNode($context);
77-
78-
if ($field === null) {
79-
continue;
80-
}
81-
82-
$fields[] = $field;
83-
}
84-
85-
if ($fields === []) {
86-
return new NamedTypeNode($this->getName());
87-
}
88-
89-
return new NamedTypeNode($this->getName(), fields: new FieldsListNode($fields));
69+
return (new ClassIntrospection($this))
70+
->getTypeStatement($context);
9071
}
9172

9273
/**

src/Mapping/Metadata/PropertyMetadata.php

+6-43
Original file line numberDiff line numberDiff line change
@@ -4,12 +4,12 @@
44

55
namespace TypeLang\Mapper\Mapping\Metadata;
66

7+
use TypeLang\Mapper\Mapping\Introspection\IntrospectionInterface;
8+
use TypeLang\Mapper\Mapping\Introspection\PropertyIntrospection;
79
use TypeLang\Mapper\Runtime\Context;
8-
use TypeLang\Parser\Node\Stmt\NamedTypeNode;
9-
use TypeLang\Parser\Node\Stmt\Shape\NamedFieldNode;
1010
use TypeLang\Parser\Node\Stmt\TypeStatement;
1111

12-
final class PropertyMetadata extends Metadata
12+
final class PropertyMetadata extends Metadata implements IntrospectionInterface
1313
{
1414
/**
1515
* @var non-empty-string
@@ -43,47 +43,10 @@ public function __construct(
4343
*
4444
* @codeCoverageIgnore
4545
*/
46-
public function getTypeStatement(Context $context): ?TypeStatement
46+
public function getTypeStatement(Context $context): TypeStatement
4747
{
48-
$info = $this->findTypeInfo();
49-
50-
if ($info === null) {
51-
return null;
52-
}
53-
54-
$statement = clone $info->getTypeStatement();
55-
56-
if ($context->isDetailedTypes() || !$statement instanceof NamedTypeNode) {
57-
return $statement;
58-
}
59-
60-
return new NamedTypeNode($statement->name);
61-
}
62-
63-
/**
64-
* Dynamically creates AST field representation.
65-
*
66-
* @codeCoverageIgnore
67-
*/
68-
public function getFieldNode(Context $context): ?NamedFieldNode
69-
{
70-
$statement = $this->getTypeStatement($context);
71-
72-
if ($statement === null) {
73-
return null;
74-
}
75-
76-
$name = $this->getName();
77-
78-
if ($context->isDenormalization()) {
79-
$name = $this->getExportName();
80-
}
81-
82-
return new NamedFieldNode(
83-
key: $name,
84-
of: $statement,
85-
optional: $this->hasDefaultValue(),
86-
);
48+
return (new PropertyIntrospection($this))
49+
->getTypeStatement($context);
8750
}
8851

8952
/**

0 commit comments

Comments
 (0)