Skip to content

Commit 2e8e6b7

Browse files
committed
Add tag factory support
1 parent c82d969 commit 2e8e6b7

8 files changed

+91
-38
lines changed

src/Factory.php

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\PHPDoc;
6+
7+
use TypeLang\PHPDoc\Parser\Description\DescriptionParserInterface;
8+
use TypeLang\PHPDoc\Tag\Tag;
9+
10+
final class Factory implements MutableFactoryInterface
11+
{
12+
/**
13+
* @param array<non-empty-string, FactoryInterface> $factories
14+
*/
15+
public function __construct(
16+
private array $factories = [],
17+
) {}
18+
19+
public function add(array|string $tags, FactoryInterface $delegate): void
20+
{
21+
foreach ((array)$tags as $tag) {
22+
$this->factories[$tag] = $delegate;
23+
}
24+
}
25+
26+
public function create(string $name, string $content, DescriptionParserInterface $descriptions): Tag
27+
{
28+
if (($delegate = $this->factories[$name]) !== null) {
29+
return $delegate->create($name, $content, $descriptions);
30+
}
31+
32+
return new Tag($name, $descriptions->parse($content));
33+
}
34+
}

src/FactoryInterface.php

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\PHPDoc;
6+
7+
use TypeLang\PHPDoc\Parser\Description\DescriptionParserInterface;
8+
use TypeLang\PHPDoc\Tag\Tag;
9+
10+
interface FactoryInterface
11+
{
12+
/**
13+
* Returns a tag object with the specified name and description.
14+
*
15+
* @param non-empty-string $name
16+
*/
17+
public function create(string $name, string $content, DescriptionParserInterface $descriptions): Tag;
18+
}

src/MutableFactoryInterface.php

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\PHPDoc;
6+
7+
interface MutableFactoryInterface extends FactoryInterface
8+
{
9+
public function add(string|array $tags, FactoryInterface $delegate): void;
10+
}

src/Parser.php

+13-5
Original file line numberDiff line numberDiff line change
@@ -21,11 +21,19 @@
2121
*/
2222
class Parser implements ParserInterface
2323
{
24+
private readonly CommentParserInterface $comments;
25+
26+
private readonly DescriptionParserInterface $descriptions;
27+
28+
private readonly TagParserInterface $tags;
29+
2430
public function __construct(
25-
private readonly CommentParserInterface $comments = new RegexCommentParser(),
26-
private readonly DescriptionParserInterface $descriptions = new SprintfDescriptionReader(),
27-
private readonly TagParserInterface $tags = new TagParser(),
28-
) {}
31+
FactoryInterface $tags = new Factory(),
32+
) {
33+
$this->tags = new TagParser($tags);
34+
$this->descriptions = new SprintfDescriptionReader($this->tags);
35+
$this->comments = new RegexCommentParser();
36+
}
2937

3038
/**
3139
* @throws RuntimeExceptionInterface
@@ -75,7 +83,7 @@ private function analyze(string $docblock): \Generator
7583
foreach ($blocks->getReturn() as $block) {
7684
try {
7785
if ($description === null) {
78-
$description = $this->descriptions->parse($block, $this->tags);
86+
$description = $this->descriptions->parse($block);
7987
} else {
8088
$tags[] = $this->tags->parse($block, $this->descriptions);
8189
}

src/Parser/Description/DescriptionParser.php

+5-10
Original file line numberDiff line numberDiff line change
@@ -9,16 +9,11 @@
99

1010
abstract class DescriptionParser implements DescriptionParserInterface
1111
{
12-
public function parse(string $description, TagParserInterface $parser = null): Description
13-
{
14-
if ($parser === null || $description === '') {
15-
return new Description($description);
16-
}
17-
18-
return $this->doParseDescription($description, $parser);
19-
}
12+
public function __construct(
13+
private readonly TagParserInterface $tags,
14+
) {}
2015

21-
private function doParseDescription(string $description, TagParserInterface $parser): Description
16+
public function parse(string $description): Description
2217
{
2318
$tags = [];
2419
$tagIdentifier = 0;
@@ -32,7 +27,7 @@ private function doParseDescription(string $description, TagParserInterface $par
3227

3328
if (\str_starts_with($chunk, '@')) {
3429
try {
35-
$tags[] = $parser->parse($chunk, $this);
30+
$tags[] = $this->tags->parse($chunk, $this);
3631
$result .= $this->createDescriptionChunkPlaceholder(++$tagIdentifier);
3732
} catch (\Throwable) {
3833
$result .= "{{$chunk}}";

src/Parser/Description/DescriptionParserInterface.php

+1-2
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,6 @@
44

55
namespace TypeLang\PHPDoc\Parser\Description;
66

7-
use TypeLang\PHPDoc\Parser\Tag\TagParserInterface;
87
use TypeLang\PHPDoc\Tag\Description;
98

109
interface DescriptionParserInterface
@@ -26,5 +25,5 @@ interface DescriptionParserInterface
2625
* // }
2726
* ```
2827
*/
29-
public function parse(string $description, TagParserInterface $parser = null): Description;
28+
public function parse(string $description): Description;
3029
}

src/Parser/Tag/TagParser.php

+9-20
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
namespace TypeLang\PHPDoc\Parser\Tag;
66

77
use TypeLang\PHPDoc\Exception\InvalidTagNameException;
8+
use TypeLang\PHPDoc\FactoryInterface;
89
use TypeLang\PHPDoc\Parser\Description\DescriptionParserInterface;
910
use TypeLang\PHPDoc\Tag\Tag;
1011

@@ -15,6 +16,10 @@ final class TagParser implements TagParserInterface
1516
*/
1617
private const PATTERN_TAG = '\G@[a-zA-Z_\x80-\xff\\\][\w\x80-\xff\-:\\\]*';
1718

19+
public function __construct(
20+
private readonly FactoryInterface $tags,
21+
) {}
22+
1823
/**
1924
* Read tag name from passed content.
2025
*
@@ -54,33 +59,17 @@ private function getTagName(string $content): string
5459
}
5560

5661
/**
57-
* @return array{non-empty-string, string}
5862
* @throws InvalidTagNameException
5963
*/
60-
private function getTagParts(string $content): array
64+
public function parse(string $tag, DescriptionParserInterface $parser): Tag
6165
{
62-
$name = $this->getTagName($content);
66+
$name = $this->getTagName($tag);
6367
/** @var non-empty-string $name */
6468
$name = \substr($name, 1);
6569

66-
$content = \substr($content, \strlen($name) + 1);
70+
$content = \substr($tag, \strlen($name) + 1);
6771
$content = \ltrim($content);
6872

69-
return [$name, $content];
70-
}
71-
72-
/**
73-
* @throws InvalidTagNameException
74-
*/
75-
public function parse(string $tag, DescriptionParserInterface $parser = null): Tag
76-
{
77-
// Tag name like ["var", "example"] extracted from "@var example"
78-
[$name, $content] = $this->getTagParts($tag);
79-
80-
if ($parser !== null) {
81-
$content = $parser->parse($content, $this);
82-
}
83-
84-
return new Tag($name, $content);
73+
return $this->tags->create($name, $content, $parser);
8574
}
8675
}

src/Parser/Tag/TagParserInterface.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,5 +24,5 @@ interface TagParserInterface
2424
* // }
2525
* ```
2626
*/
27-
public function parse(string $tag, DescriptionParserInterface $parser = null): Tag;
27+
public function parse(string $tag, DescriptionParserInterface $parser): Tag;
2828
}

0 commit comments

Comments
 (0)