Skip to content

Commit 72ac67f

Browse files
committed
Add "@Package" and "@subpackage" tags support
1 parent 7d8b3f2 commit 72ac67f

20 files changed

+211
-22
lines changed

README.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -56,7 +56,7 @@ composer require type-lang/phpdoc
5656
- [ ] `@mixin` - TODO
5757
- [x] `@no-named-arguments` - Indicates that argument names may be
5858
changed in the future.
59-
- [ ] `@package` - TODO
59+
- [x] `@package` - Used to categorize _Element(s)_ into logical subdivisions
6060
- [x] `@override` - Mention to see if the method is actually overriding a definition
6161
- [x] `@param` - Used to document a single argument of a function or method
6262
- [ ] `@param-closure-this` - TODO
@@ -78,7 +78,7 @@ composer require type-lang/phpdoc
7878
website or other _Symbol(s)_
7979
- [ ] `@since` - TODO
8080
- [ ] `@source` - TODO
81-
- [ ] `@subpackage` - TODO
81+
- [x] `@subpackage` - Used to categorize _Element(s)_ into logical subdivisions
8282
- [ ] `@suppress` - TODO
8383
- [x] `@template` - Allows classes (and class-like entries), functions and
8484
methods to declare a generic type parameter

src/DocBlock/Tag/MethodTag/MethodTag.php

+3-3
Original file line numberDiff line numberDiff line change
@@ -36,9 +36,9 @@
3636
* with a class or interface.
3737
*
3838
* ```
39-
* "@method" [static] <CallableType> [<description>]
40-
* "@method" [static] <ReturnType> <CallableType> [<description>]
41-
* "@method" [static] <CallableType>: <ReturnType> [<description>]
39+
* "@method" [static] <callable-type> [<description>]
40+
* "@method" [static] <return-type> <callable-type> [<description>]
41+
* "@method" [static] <callable-type>: <return-type> [<description>]
4242
* ```
4343
*/
4444
class MethodTag extends Tag implements OptionalTypeProviderInterface
+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\PHPDoc\DocBlock\Tag\PackageTag;
6+
7+
use TypeLang\PHPDoc\DocBlock\Tag\Tag;
8+
9+
/**
10+
* Used to categorize _Element(s)_ into logical subdivisions.
11+
*
12+
* The "`@package`" tag can be used as a counterpart or supplement to
13+
* Namespaces. Namespaces provide a functional subdivision of _Element(s)_
14+
* where the "`@package`" tag can provide a logical subdivision in which
15+
* way the elements can be grouped with a different hierarchy.
16+
*
17+
* If, across the board, both logical and functional subdivisions
18+
* are equal is it NOT RECOMMENDED to use the "`@package`"
19+
* tag to prevent maintenance overhead.
20+
*
21+
* Each level in the logical hierarchy MUST be separated with a backslash
22+
* (`\`) to be familiar to Namespaces. A hierarchy MAY be of endless depth
23+
* but it is RECOMMENDED to keep the depth at less or equal than six levels.
24+
*
25+
* Please note that the "`@package`" tag applies to different _Element(s)_
26+
* depending where it is defined.
27+
*
28+
* - If the package is defined in the file-level DocBlock then it only applies
29+
* to the following elements in the applicable file:
30+
* - global functions
31+
* - global constants
32+
* - global variables
33+
* - requires and includes
34+
*
35+
* - If the package is defined in a namespace-level or class-level DocBlock
36+
* then the package applies to that namespace, class, trait or interface
37+
* and their contained elements. This means that a function which is
38+
* contained in a namespace with the "`@package`" tag assumes that package.
39+
*
40+
* The "`@package`" tag MUST NOT occur more than once in a PHPDoc.
41+
*
42+
* ```
43+
* "@package" <namespace> [<description>]
44+
* ```
45+
*/
46+
final class PackageTag extends SubPackageTag {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\PHPDoc\DocBlock\Tag\PackageTag;
6+
7+
use TypeLang\Parser\Parser as TypesParser;
8+
use TypeLang\Parser\ParserInterface as TypesParserInterface;
9+
use TypeLang\PHPDoc\DocBlock\Tag\Factory\TagFactoryInterface;
10+
use TypeLang\PHPDoc\Parser\Description\DescriptionParserInterface;
11+
12+
/**
13+
* This class is responsible for creating "`@package`" tags.
14+
*
15+
* See {@see PackageTag} for details about this tag.
16+
*/
17+
final class PackageTagFactory implements TagFactoryInterface
18+
{
19+
private readonly SubPackageTagFactory $factory;
20+
21+
public function __construct(
22+
TypesParserInterface $parser = new TypesParser(tolerant: true),
23+
) {
24+
$this->factory = new SubPackageTagFactory($parser);
25+
}
26+
27+
public function create(string $tag, string $content, DescriptionParserInterface $descriptions): PackageTag
28+
{
29+
$result = $this->factory->create($tag, $content, $descriptions);
30+
31+
return new PackageTag(
32+
name: $result->name,
33+
package: $result->package,
34+
description: $result->description,
35+
);
36+
}
37+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\PHPDoc\DocBlock\Tag\PackageTag;
6+
7+
use TypeLang\Parser\Node\Name;
8+
use TypeLang\PHPDoc\DocBlock\Tag\Tag;
9+
10+
/**
11+
* Used to categorize _Element(s)_ into logical subdivisions.
12+
*
13+
* The "`@subpackage`" tag can be used as a counterpart or supplement to
14+
* Namespaces. Namespaces provide a functional subdivision of _Element(s)_
15+
* where the "`@subpackage`" tag can provide a logical subdivision in
16+
* which way the elements can be grouped with a different hierarchy.
17+
*
18+
* If, across the board, both logical and functional subdivisions
19+
* are equal, it is NOT RECOMMENDED to use the "`@subpackage`"
20+
* tag to prevent maintenance overhead.
21+
*
22+
* The "`@subpackage`" tag MUST only be used in a select set of
23+
* DocBlocks, as is described in the documentation for the "`@package`"
24+
* tag. It MUST also accompany a "`@package`" tag and may occur
25+
* only once per DocBlock.
26+
*
27+
* This tag is considered superseded by the support for multiple levels
28+
* in the "`@package`" tag and is as such considered deprecated.
29+
*
30+
* ```
31+
* ""`@subpackage`"" <namespace> [<description>]
32+
* ```
33+
*/
34+
class SubPackageTag extends Tag
35+
{
36+
public function __construct(
37+
string $name,
38+
public readonly Name $package,
39+
\Stringable|string|null $description = null,
40+
) {
41+
parent::__construct($name, $description);
42+
}
43+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace TypeLang\PHPDoc\DocBlock\Tag\PackageTag;
6+
7+
use TypeLang\Parser\Node\Stmt\NamedTypeNode;
8+
use TypeLang\Parser\Parser as TypesParser;
9+
use TypeLang\Parser\ParserInterface as TypesParserInterface;
10+
use TypeLang\PHPDoc\DocBlock\Tag\Factory\TagFactoryInterface;
11+
use TypeLang\PHPDoc\Parser\Content\Stream;
12+
use TypeLang\PHPDoc\Parser\Content\TypeReader;
13+
use TypeLang\PHPDoc\Parser\Description\DescriptionParserInterface;
14+
15+
/**
16+
* This class is responsible for creating "`@subpackage`" tags.
17+
*
18+
* See {@see SubPackageTag} for details about this tag.
19+
*/
20+
final class SubPackageTagFactory implements TagFactoryInterface
21+
{
22+
public function __construct(
23+
private readonly TypesParserInterface $parser = new TypesParser(tolerant: true),
24+
) {}
25+
26+
public function create(string $tag, string $content, DescriptionParserInterface $descriptions): SubPackageTag
27+
{
28+
$stream = new Stream($tag, $content);
29+
30+
$type = $stream->apply(new TypeReader($this->parser));
31+
32+
if (!$type instanceof NamedTypeNode) {
33+
throw $stream->toException(\sprintf(
34+
'Tag @%s expects the namespace to be defined',
35+
$stream->tag,
36+
));
37+
}
38+
39+
if ($type->arguments !== null) {
40+
throw $stream->toException(\sprintf(
41+
'Tag @%s namespace cannot contain template arguments',
42+
$stream->tag,
43+
));
44+
}
45+
46+
if ($type->fields !== null) {
47+
throw $stream->toException(\sprintf(
48+
'Tag @%s namespace cannot contain shape fields',
49+
$stream->tag,
50+
));
51+
}
52+
53+
return new SubPackageTag(
54+
name: $tag,
55+
package: $type->name,
56+
description: $stream->toOptionalDescription($descriptions),
57+
);
58+
}
59+
}

src/DocBlock/Tag/ParamTag/ParamTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@
2828
* limited to structural elements of type method or function.
2929
*
3030
* ```
31-
* "@param" [<Type>] $<variable> [<description>]
31+
* "@param" [<type>] $<variable> [<description>]
3232
* ```
3333
*/
3434
class ParamTag extends Tag implements

src/DocBlock/Tag/PropertyTag/PropertyReadTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66

77
/**
88
* ```
9-
* "@property-read" [<Type>] $<name> [<description>]
9+
* "@property-read" [<type>] $<name> [<description>]
1010
* ```
1111
*/
1212
class PropertyReadTag extends PropertyTag {}

src/DocBlock/Tag/PropertyTag/PropertyTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@
3434
* - Also {@see PropertyWriteTag} for "`@property-write`" tag implementation.
3535
*
3636
* ```
37-
* "@property" [<Type>] $<name> [<description>]
37+
* "@property" [<type>] $<name> [<description>]
3838
* ```
3939
*/
4040
class PropertyTag extends Tag implements

src/DocBlock/Tag/PropertyTag/PropertyWriteTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66

77
/**
88
* ```
9-
* "@property-write" [<Type>] $<name> [<description>]
9+
* "@property-write" [<type>] $<name> [<description>]
1010
* ```
1111
*/
1212
class PropertyWriteTag extends PropertyTag {}

src/DocBlock/Tag/ReturnTag/ReturnTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@
3030
* structural elements of type method or function.
3131
*
3232
* ```
33-
* "@return" [<Type>] [<description>]
33+
* "@return" [<type>] [<description>]
3434
* ```
3535
*/
3636
class ReturnTag extends Tag implements TypeProviderInterface

src/DocBlock/Tag/TemplateExtendsTag/TemplateExtendsTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66

77
/**
88
* ```
9-
* "@extends" <Type> [<description>]
9+
* "@extends" <type> [<description>]
1010
* ```
1111
*/
1212
class TemplateExtendsTag extends TemplateInheritanceTag {}

src/DocBlock/Tag/TemplateExtendsTag/TemplateImplementsTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66

77
/**
88
* ```
9-
* "@implements" <Type> [<description>]
9+
* "@implements" <type> [<description>]
1010
* ```
1111
*/
1212
class TemplateImplementsTag extends TemplateInheritanceTag {}

src/DocBlock/Tag/TemplateExtendsTag/TemplateUseTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66

77
/**
88
* ```
9-
* "@use" <Type> [<description>]
9+
* "@use" <type> [<description>]
1010
* ```
1111
*/
1212
class TemplateUseTag extends TemplateInheritanceTag {}

src/DocBlock/Tag/TemplateTag/TemplateContravariantTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66

77
/**
88
* ```
9-
* "@template-contravariant" <name> ['of' <Type>] [<description>]
9+
* "@template-contravariant" <name> ['of' <type>] [<description>]
1010
* ```
1111
*/
1212
final class TemplateContravariantTag extends TemplateTag {}

src/DocBlock/Tag/TemplateTag/TemplateCovariantTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66

77
/**
88
* ```
9-
* "@template-covariant" <name> ['of' <Type>] [<description>]
9+
* "@template-covariant" <name> ['of' <type>] [<description>]
1010
* ```
1111
*/
1212
final class TemplateCovariantTag extends TemplateTag {}

src/DocBlock/Tag/TemplateTag/TemplateTag.php

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

1111
/**
1212
* ```
13-
* "@template" <name> ['of' <Type>] [<description>]
13+
* "@template" <name> ['of' <type>] [<description>]
1414
* ```
1515
*/
1616
class TemplateTag extends Tag implements OptionalTypeProviderInterface

src/DocBlock/Tag/ThrowsTag/ThrowsTag.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@
2525
* detailed view is created and the consumer knows for which errors to check.
2626
*
2727
* ```
28-
* "@throws" [<Type>] [<description>]
28+
* "@throws" [<type>] [<description>]
2929
* ```
3030
*/
3131
class ThrowsTag extends Tag implements TypeProviderInterface

src/DocBlock/Tag/VarTag/VarTag.php

+5-5
Original file line numberDiff line numberDiff line change
@@ -26,11 +26,11 @@
2626
* while several items are represented.
2727
*
2828
* ```
29-
* "@var" [<Type>] $<variable> [<description>]
30-
* "@var" [<Type>] ...$<variable> [<description>]
31-
* "@var" [<Type>] &$<variable> [<description>]
32-
* "@var" [<Type>] ...&$<variable> [<description>]
33-
* "@var" [<Type>] &...$<variable> [<description>]
29+
* "@var" [<type>] $<variable> [<description>]
30+
* "@var" [<type>] ...$<variable> [<description>]
31+
* "@var" [<type>] &$<variable> [<description>]
32+
* "@var" [<type>] ...&$<variable> [<description>]
33+
* "@var" [<type>] &...$<variable> [<description>]
3434
* ```
3535
*/
3636
class VarTag extends Tag implements

src/Platform/StandardPlatform.php

+4
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@
1818
use TypeLang\PHPDoc\DocBlock\Tag\MethodTag\MethodTagFactory;
1919
use TypeLang\PHPDoc\DocBlock\Tag\NoNamedArgumentsTag\NoNamedArgumentsTagFactory;
2020
use TypeLang\PHPDoc\DocBlock\Tag\OverrideTag\OverrideTagFactory;
21+
use TypeLang\PHPDoc\DocBlock\Tag\PackageTag\PackageTagFactory;
22+
use TypeLang\PHPDoc\DocBlock\Tag\PackageTag\SubPackageTagFactory;
2123
use TypeLang\PHPDoc\DocBlock\Tag\ParamTag\ParamTagFactory;
2224
use TypeLang\PHPDoc\DocBlock\Tag\PropertyTag\PropertyReadTagFactory;
2325
use TypeLang\PHPDoc\DocBlock\Tag\PropertyTag\PropertyTagFactory;
@@ -54,13 +56,15 @@ protected function load(TypesParserInterface $types): iterable
5456
yield 'link' => new LinkTagFactory();
5557
yield 'method' => new MethodTagFactory($types);
5658
yield 'no-named-arguments' => new NoNamedArgumentsTagFactory();
59+
yield 'package' => new PackageTagFactory($types);
5760
yield 'override' => new OverrideTagFactory();
5861
yield 'param' => new ParamTagFactory($types);
5962
yield 'property' => new PropertyTagFactory($types);
6063
yield 'property-read' => new PropertyReadTagFactory($types);
6164
yield 'property-write' => new PropertyWriteTagFactory($types);
6265
yield 'return' => new ReturnTagFactory($types);
6366
yield 'see' => new SeeTagFactory($types);
67+
yield 'subpackage' => new SubPackageTagFactory($types);
6468
yield 'template' => new TemplateTagFactory($types);
6569
yield 'template-contravariant' => new TemplateContravariantTagFactory($types);
6670
yield 'template-covariant' => new TemplateCovariantTagFactory($types);

0 commit comments

Comments
 (0)