Skip to content

Commit b5dc333

Browse files
authored
Merge pull request #343 from phpDocumentor/poc-phpstan-integration
PhpStan based tag parsing
2 parents 203354b + 1f95f3b commit b5dc333

38 files changed

+1420
-76
lines changed

composer.json

+3-2
Original file line numberDiff line numberDiff line change
@@ -15,10 +15,11 @@
1515
],
1616
"require": {
1717
"php": "^7.2 || ^8.0",
18-
"phpdocumentor/type-resolver": "^1.3",
18+
"phpdocumentor/type-resolver": "1.x-dev@dev",
1919
"webmozart/assert": "^1.9.1",
2020
"phpdocumentor/reflection-common": "^2.2",
21-
"ext-filter": "*"
21+
"ext-filter": "*",
22+
"phpstan/phpdoc-parser": "^1.7"
2223
},
2324
"require-dev": {
2425
"mockery/mockery": "~1.3.5",

composer.lock

+66-11
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

phpcs.xml.dist

+1-4
Original file line numberDiff line numberDiff line change
@@ -10,13 +10,10 @@
1010

1111
<!-- Set the minimum PHP version for PHPCompatibility.
1212
This should be kept in sync with the requirements in the composer.json file. -->
13-
<config name="testVersion" value="7.2-"/>
13+
<config name="testVersion" value="7.4-"/>
1414

1515
<rule ref="phpDocumentor">
1616
<exclude name="SlevomatCodingStandard.Exceptions.ReferenceThrowableOnly.ReferencedGeneralException" />
17-
18-
<!-- Property type declarations are a PHP 7.4 feature. -->
19-
<exclude name="SlevomatCodingStandard.TypeHints.PropertyTypeHint.MissingNativeTypeHint"/>
2017
</rule>
2118

2219
<rule ref="SlevomatCodingStandard.Classes.SuperfluousAbstractClassNaming.SuperfluousPrefix">

phpunit.xml.dist

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,8 @@
44
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
55
xsi:noNamespaceSchemaLocation="https://schema.phpunit.de/8.0/phpunit.xsd"
66
colors="true"
7-
convertDeprecationsToExceptions="true"
8-
beStrictAboutOutputDuringTests="true"
7+
convertDeprecationsToExceptions="false"
8+
beStrictAboutOutputDuringTests="false"
99
forceCoversAnnotation="true"
1010
verbose="true"
1111
bootstrap="vendor/autoload.php"

src/DocBlock/DescriptionFactory.php

+3-2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313

1414
namespace phpDocumentor\Reflection\DocBlock;
1515

16+
use phpDocumentor\Reflection\DocBlock\Tags\Factory\Factory;
1617
use phpDocumentor\Reflection\Types\Context as TypeContext;
1718
use phpDocumentor\Reflection\Utils;
1819

@@ -47,13 +48,13 @@
4748
*/
4849
class DescriptionFactory
4950
{
50-
/** @var TagFactory */
51+
/** @var Factory */
5152
private $tagFactory;
5253

5354
/**
5455
* Initializes this factory with the means to construct (inline) tags.
5556
*/
56-
public function __construct(TagFactory $tagFactory)
57+
public function __construct(Factory $tagFactory)
5758
{
5859
$this->tagFactory = $tagFactory;
5960
}

src/DocBlock/StandardTagFactory.php

+30-13
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
use phpDocumentor\Reflection\DocBlock\Tags\Author;
1818
use phpDocumentor\Reflection\DocBlock\Tags\Covers;
1919
use phpDocumentor\Reflection\DocBlock\Tags\Deprecated;
20+
use phpDocumentor\Reflection\DocBlock\Tags\Factory\Factory;
2021
use phpDocumentor\Reflection\DocBlock\Tags\Generic;
2122
use phpDocumentor\Reflection\DocBlock\Tags\InvalidTag;
2223
use phpDocumentor\Reflection\DocBlock\Tags\Link as LinkTag;
@@ -40,12 +41,15 @@
4041
use ReflectionParameter;
4142
use Webmozart\Assert\Assert;
4243

44+
use function array_key_exists;
4345
use function array_merge;
4446
use function array_slice;
4547
use function call_user_func_array;
4648
use function count;
4749
use function get_class;
50+
use function is_object;
4851
use function preg_match;
52+
use function sprintf;
4953
use function strpos;
5054
use function trim;
5155

@@ -72,7 +76,7 @@ final class StandardTagFactory implements TagFactory
7276
public const REGEX_TAGNAME = '[\w\-\_\\\\:]+';
7377

7478
/**
75-
* @var array<class-string<Tag>> An array with a tag as a key, and an
79+
* @var array<class-string<Tag>|Factory> An array with a tag as a key, and an
7680
* FQCN to a class that handles it as an array value.
7781
*/
7882
private $tagHandlerMappings = [
@@ -162,18 +166,25 @@ public function addService(object $service, ?string $alias = null): void
162166
$this->serviceLocator[$alias ?: get_class($service)] = $service;
163167
}
164168

165-
public function registerTagHandler(string $tagName, string $handler): void
169+
/** {@inheritDoc} */
170+
public function registerTagHandler(string $tagName, $handler): void
166171
{
167172
Assert::stringNotEmpty($tagName);
168-
Assert::classExists($handler);
169-
Assert::implementsInterface($handler, Tag::class);
170-
171173
if (strpos($tagName, '\\') && $tagName[0] !== '\\') {
172174
throw new InvalidArgumentException(
173175
'A namespaced tag must have a leading backslash as it must be fully qualified'
174176
);
175177
}
176178

179+
if (is_object($handler)) {
180+
Assert::isInstanceOf($handler, Factory::class);
181+
$this->tagHandlerMappings[$tagName] = $handler;
182+
183+
return;
184+
}
185+
186+
Assert::classExists($handler);
187+
Assert::implementsInterface($handler, Tag::class);
177188
$this->tagHandlerMappings[$tagName] = $handler;
178189
}
179190

@@ -210,6 +221,10 @@ private function createTag(string $body, string $name, TypeContext $context): Ta
210221
$this->getServiceLocatorWithDynamicParameters($context, $name, $body)
211222
);
212223

224+
if (array_key_exists('tagLine', $arguments)) {
225+
$arguments['tagLine'] = sprintf('@%s %s', $name, $body);
226+
}
227+
213228
try {
214229
$callable = [$handlerClassName, 'create'];
215230
Assert::isCallable($callable);
@@ -225,9 +240,9 @@ private function createTag(string $body, string $name, TypeContext $context): Ta
225240
/**
226241
* Determines the Fully Qualified Class Name of the Factory or Tag (containing a Factory Method `create`).
227242
*
228-
* @return class-string<Tag>
243+
* @return class-string<Tag>|Factory
229244
*/
230-
private function findHandlerClassName(string $tagName, TypeContext $context): string
245+
private function findHandlerClassName(string $tagName, TypeContext $context)
231246
{
232247
$handlerClassName = Generic::class;
233248
if (isset($this->tagHandlerMappings[$tagName])) {
@@ -268,18 +283,18 @@ private function getArgumentsForParametersFromWiring(array $parameters, array $l
268283
}
269284
}
270285

286+
$parameterName = $parameter->getName();
271287
if (isset($locator[$typeHint])) {
272-
$arguments[] = $locator[$typeHint];
288+
$arguments[$parameterName] = $locator[$typeHint];
273289
continue;
274290
}
275291

276-
$parameterName = $parameter->getName();
277292
if (isset($locator[$parameterName])) {
278-
$arguments[] = $locator[$parameterName];
293+
$arguments[$parameterName] = $locator[$parameterName];
279294
continue;
280295
}
281296

282-
$arguments[] = null;
297+
$arguments[$parameterName] = null;
283298
}
284299

285300
return $arguments;
@@ -289,12 +304,14 @@ private function getArgumentsForParametersFromWiring(array $parameters, array $l
289304
* Retrieves a series of ReflectionParameter objects for the static 'create' method of the given
290305
* tag handler class name.
291306
*
292-
* @param class-string $handlerClassName
307+
* @param class-string|Factory $handler
293308
*
294309
* @return ReflectionParameter[]
295310
*/
296-
private function fetchParametersForHandlerFactoryMethod(string $handlerClassName): array
311+
private function fetchParametersForHandlerFactoryMethod($handler): array
297312
{
313+
$handlerClassName = is_object($handler) ? get_class($handler) : $handler;
314+
298315
if (!isset($this->tagHandlerParameterCache[$handlerClassName])) {
299316
$methodReflection = new ReflectionMethod($handlerClassName, 'create');
300317
$this->tagHandlerParameterCache[$handlerClassName] = $methodReflection->getParameters();

src/DocBlock/TagFactory.php

+4-15
Original file line numberDiff line numberDiff line change
@@ -14,9 +14,9 @@
1414
namespace phpDocumentor\Reflection\DocBlock;
1515

1616
use InvalidArgumentException;
17-
use phpDocumentor\Reflection\Types\Context as TypeContext;
17+
use phpDocumentor\Reflection\DocBlock\Tags\Factory\Factory;
1818

19-
interface TagFactory
19+
interface TagFactory extends Factory
2020
{
2121
/**
2222
* Adds a parameter to the service locator that can be injected in a tag's factory method.
@@ -40,17 +40,6 @@ interface TagFactory
4040
*/
4141
public function addParameter(string $name, $value): void;
4242

43-
/**
44-
* Factory method responsible for instantiating the correct sub type.
45-
*
46-
* @param string $tagLine The text for this tag, including description.
47-
*
48-
* @return Tag A new tag object.
49-
*
50-
* @throws InvalidArgumentException If an invalid tag line was presented.
51-
*/
52-
public function create(string $tagLine, ?TypeContext $context = null): Tag;
53-
5443
/**
5544
* Registers a service with the Service Locator using the FQCN of the class or the alias, if provided.
5645
*
@@ -71,7 +60,7 @@ public function addService(object $service): void;
7160
*
7261
* @param string $tagName Name of tag to register a handler for. When registering a namespaced
7362
* tag, the full name, along with a prefixing slash MUST be provided.
74-
* @param class-string<Tag> $handler FQCN of handler.
63+
* @param class-string<Tag>|Factory $handler FQCN of handler.
7564
*
7665
* @throws InvalidArgumentException If the tag name is not a string.
7766
* @throws InvalidArgumentException If the tag name is namespaced (contains backslashes) but
@@ -80,5 +69,5 @@ public function addService(object $service): void;
8069
* @throws InvalidArgumentException If the handler is not an existing class.
8170
* @throws InvalidArgumentException If the handler does not implement the {@see Tag} interface.
8271
*/
83-
public function registerTagHandler(string $tagName, string $handler): void;
72+
public function registerTagHandler(string $tagName, $handler): void;
8473
}

0 commit comments

Comments
 (0)