This project is not being actively maintained. If you are interested in helping to maintain this project, take a look at the open issues on GitHub and submit pull requests. |
Custom Annotation Classes
If you want to define your own annotations you just have to group them in a namespace and register this namespace
in the AnnotationRegistry. Annotation classes have to contain a class-level docblock with the text @Annotation
:
Inject annotation values
The annotation parser check if the annotation constructor has arguments, if so then we will pass the value array, otherwise will try to inject values into public properties directly:
1 namespace MyCompany\Annotations;
/**
* @Annotation
*
* Some Annotation using a constructor
*/
class Bar
{
private $foo;
public function __construct(array $values)
{
$this->foo = $values['foo'];
}
}
/**
* @Annotation
*
* Some Annotation without a constructor
*/
class Foo
{
public $bar;
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
Annotation Target
@Target
indicates the kinds of class element to which an annotation type is applicable.
Then you could define one or more targets:
CLASS
Allowed in the class docblockPROPERTY
Allowed in the property docblockMETHOD
Allowed in the method docblockALL
Allowed in the class, property and method docblockANNOTATION
Allowed inside other annotations
If the annotations is not allowed in the current context you got an AnnotationException
Attribute types
Annotation parser check the given parameters using the phpdoc annotation @var
,
The data type could be validated using the @var
annotation on the annotation properties
or using the annotations @Attributes
and @Attribute
.
If the data type not match you got an AnnotationException
1 namespace MyCompany\Annotations;
/**
* @Annotation
* @Target({"METHOD","PROPERTY"})
*/
class Bar
{
/** @var mixed */
public $mixed;
/** @var boolean */
public $boolean;
/** @var bool */
public $bool;
/** @var float */
public $float;
/** @var string */
public $string;
/** @var integer */
public $integer;
/** @var array */
public $array;
/** @var SomeAnnotationClass */
public $annotation;
/** @var array<integer> */
public $arrayOfIntegers;
/** @var array<SomeAnnotationClass> */
public $arrayOfAnnotations;
}
/**
* @Annotation
* @Target({"METHOD","PROPERTY"})
* @Attributes({
* @Attribute("stringProperty", type = "string"),
* @Attribute("annotProperty", type = "SomeAnnotationClass"),
* })
*/
class Foo
{
public function __construct(array $values)
{
$this->stringProperty = $values['stringProperty'];
$this->annotProperty = $values['annotProperty'];
}
// some code
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
Annotation Required
@Required
indicates that the field must be specified when the annotation is used.
If it is not used you get an AnnotationException
stating that this value can not be null.
Declaring a required field:
Usage:
Enumerated values
- An annotation property marked with
@Enum
is a field that accept a fixed set of scalar values. - You should use
@Enum
fields any time you need to represent fixed values. - The annotation parser check the given value and throws an
AnnotationException
if the value not match.
Declaring an enumerated property:
Annotation usage:
Constants
The use of constants and class constants are available on the annotations parser.
The following usage are allowed:
1 namespace MyCompany\Entity;
use MyCompany\Annotations\Foo;
use MyCompany\Annotations\Bar;
use MyCompany\Entity\SomeClass;
/**
* @Foo(PHP_EOL)
* @Bar(Bar::FOO)
* @Foo({SomeClass::FOO, SomeClass::BAR})
* @Bar({SomeClass::FOO_KEY = SomeClass::BAR_VALUE})
*/
class User
{
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Be careful with constants and the cache !
The cached reader will not re-evaluate each time an annotation is loaded from cache. When a constant is changed the cache must be cleaned. |
Usage
Using the library API is simple. Using the annotations described in the previous section you can now annotate other classes with your annotations:
Now we can write a script to get the annotations above:
1 $reflClass = new ReflectionClass('MyCompany\Entity\User');
$classAnnotations = $reader->getClassAnnotations($reflClass);
foreach ($classAnnotations AS $annot) {
if ($annot instanceof \MyCompany\Annotations\Foo) {
echo $annot->bar; // prints "foo";
} else if ($annot instanceof \MyCompany\Annotations\Bar) {
echo $annot->foo; // prints "bar";
}
}
2
3
4
5
6
7
8
9
10
You have a complete API for retrieving annotation class instances from a class, property or method docblock:
Reader API
Access all annotations of a class
1 public function getClassAnnotations(\ReflectionClass $class);
Access one annotation of a class
1 public function getClassAnnotation(\ReflectionClass $class, $annotationName);
Access all annotations of a method
1 public function getMethodAnnotations(\ReflectionMethod $method);
Access one annotation of a method
1 public function getMethodAnnotation(\ReflectionMethod $method, $annotationName);
Access all annotations of a property
1 public function getPropertyAnnotations(\ReflectionProperty $property);
Access one annotation of a property
1 public function getPropertyAnnotation(\ReflectionProperty $property, $annotationName);