Annotations¶
Overview¶
Phalcon has introduced the first annotations parser component written in C for PHP. The Phalcon\Annotations namespace encompasses general-purpose components that provide an easy way to parse and cache annotations in PHP applications.
Usage¶
Annotations are extracted from docblocks in classes, methods, and properties. An annotation can be placed at any position in the docblock:
<?php
/**
* This is the class description
*
* @AmazingClass(true)
*/
class Example
{
/**
* This is a property with a special feature
*
* @SpecialFeature
*/
protected $someProperty;
/**
* This is a method
*
* @SpecialFeature
*/
public function someMethod()
{
// ...
}
}
An annotation has the following syntax:
Additionally, an annotation can be placed at any part of a docblock:
<?php
/**
* This is a property with a special feature
*
* @SpecialFeature
*
* More comments
*
* @AnotherSpecialFeature(true)
*/
While the parser is highly flexible, it is recommended for code maintainability and understanding to place annotations at the end of the docblock:
<?php
/**
* This is a property with a special feature
* More comments
*
* @SpecialFeature({someParameter='the value', false})
* @AnotherSpecialFeature(true)
An example for a model is:
<?php
use Phalcon\Mvc\Model;
/**
* Customers
*
* Represents a customer record
*
* @Source('co_customers');
* @HasMany("cst_id", "Invoices", "inv_cst_id")
*/
class Customers extends Model
{
/**
* @Primary
* @Identity
* @Column(type="integer", nullable=false, column="cst_id")
*/
public $id;
/**
* @Column(type="string", nullable=false, column="cst_name_first")
*/
public $nameFirst;
/**
* @Column(type="string", nullable=false, column="cst_name_last")
*/
public $nameLast;
}
Types¶
Annotations may or may not have parameters. A parameter could be a simple literal (strings, number, boolean, null), an array, a hashed list, or another annotation:
Simple Annotation
Annotation with parameters
/**
* @SomeAnnotation(first='hello', second='world', third=1)
* @SomeAnnotation(first: 'hello', second: 'world', third: 1)
Annotation with named parameters
Passing an array
/**
* @SomeAnnotation({first=1, second=2, third=3})
* @SomeAnnotation({'first'=1, 'second'=2, 'third'=3})
* @SomeAnnotation({'first': 1, 'second': 2, 'third': 3})
* @SomeAnnotation(['first': 1, 'second': 2, 'third': 3])
Passing a hash as a parameter
/**
* @SomeAnnotation({'name'='SomeName', 'other'={
* 'foo1': 'bar1', 'foo2': 'bar2', {1, 2, 3},
* }})
Nested arrays/hashes
Nested Annotations
Adapters¶
This component employs adapters to cache or not cache the parsed and processed annotations, thereby improving performance:
| Adapter | Description |
|---|---|
| Phalcon\Annotations\Adapter\Apcu | Use APCu to store parsed and processed annotations (production) |
| Phalcon\Annotations\Adapter\Memory | Use memory to store annotations (development) |
| Phalcon\Annotations\Adapter\Stream | Use a file stream to store annotations. Must be used with a byte-code cache. |
Apcu¶
Phalcon\Annotations\Adapter\Apcu stores the parsed and processed annotations using the APCu cache. This adapter is suitable for production systems. However, once the web server restarts, the cache will be cleared and will have to be rebuilt. The adapter accepts two parameters in the constructor's options array:
prefix- the prefix for the key storedlifetime- the cache lifetime
<?php
use Phalcon\Annotations\Adapter\Apcu;
$adapter = new Apcu(
[
'prefix' => 'my-prefix',
'lifetime' => 3600,
]
);
Internally, the adapter stores data prefixing every key with _PHAN. This setting cannot be changed. It, however, gives you the option to scan APCu for keys that are prefixed with _PHAN and clear them if needed.
<?php
use APCuIterator;
$result = true;
$pattern = "/^_PHAN/";
$iterator = new APCuIterator($pattern);
if (true === is_object($iterator)) {
return false;
}
foreach ($iterator as $item) {
if (true !== apcu_delete($item["key"])) {
$result = false;
}
}
return $result;
Memory¶
Phalcon\Annotations\Adapter\Memory stores the parsed and processed annotations in memory. This adapter is suitable for development systems. The cache is rebuilt on every request, and therefore can immediately reflect changes while developing your application.
Stream¶
Phalcon\Annotations\Adapter\Stream stores the parsed and processed annotations in a file on the server. This adapter can be used in production systems, but it will increase the I/O since for every request the annotations cache files will need to be read from the file system. The adapter accepts one parameter in the constructor's $options array:
annotationsDir- the directory to store the annotations cache
<?php
use Phalcon\Annotations\Adapter\Stream;
$adapter = new Stream(
[
'annotationsDir' => '/app/storage/cache/annotations',
]
);
If there is a problem with storing the data in the folder due to permissions or any other reason, a Phalcon\Annotations\Exception will be thrown.
Custom¶
Phalcon\Annotations\Adapter\AdapterInterface is available
Limiting the In-Memory Cache¶
Each adapter caches parsed Phalcon\Annotations\Reflection results keyed by class name. The cache lives for the adapter instance lifetime and is bounded in practice by the number of annotated classes in the application.
For long-running processes that load classes dynamically (test runners, code generators, multi-tenant workers) call setAnnotationsLimit() to clear the cache when adding a new class would exceed the cap; the cache repopulates lazily on subsequent reads.
<?php
use Phalcon\Annotations\Adapter\Memory;
$adapter = new Memory();
$adapter->setAnnotationsLimit(500);
The default value 0 preserves the original unbounded behavior. getAnnotationsLimit() returns the current cap. The cap applies uniformly to every adapter (Apcu, Memory, Stream, custom) because the methods live on Phalcon\Annotations\Adapter\AbstractAdapter.
Examples¶
Controller-based Access¶
You can use annotations to define which areas are controlled by the ACL. This can be achieved by registering a plugin in the events manager listening to the beforeExecuteRoute event, or simply by implementing the method in your base controller.
First, set the annotations manager in your DI container:
<?php
use Phalcon\Di\FactoryDefault;
use Phalcon\Annotations\Adapter\Apcu;
$container = new FactoryDefault();
$container->set(
'annotations',
function () {
return new Apcu(
[
'lifetime' => 86400
]
);
}
);
Now, in the base controller, implement the beforeExecuteRoute method:
<?php
namespace MyApp\Controllers;
use Phalcon\Annotations\Adapter\Apcu;
use Phalcon\Events\Event;
use Phalcon\Mvc\Dispatcher;
use Phalcon\Mvc\Controller;
use MyApp\Components\Auth;
/**
* @property Apcu $annotations
* @property Auth $auth
*/
class BaseController extends Controller
{
/**
* @param Event $event
* @param Dispatcher $dispatcher
*
* @return bool
*/
public function beforeExecuteRoute(
Dispatcher $dispatcher
) {
$controllerName = $dispatcher->getControllerClass();
$annotations = $this
->annotations
->get($controllerName)
;
$exists = $annotations
->getClassAnnotations()
->has('Private')
;
if (!$exists) {
return true;
}
if ($this->auth->isLoggedIn()) {
return true;
}
$dispatcher->forward(
[
'controller' => 'session',
'action' => 'login',
]
);
return false;
}
}
In your controllers, specify:
<?php
namespace MyApp\Controllers;
use MyApp\Controllers\BaseController;
/**
* @Private(true)
*/
class Invoices extends BaseController
{
public function indexAction()
{
}
}
Group-based Access¶
You might want to expand on the above and offer more granular access control for your application. For this, also use the beforeExecuteRoute in the controller but add the access metadata on each action. If you need a specific controller to be "locked," you can also use the initialize method.
First, set the annotations manager in your DI container:
<?php
use Phalcon\Di\FactoryDefault;
use Phalcon\Annotations\Adapter\Apcu;
$container = a FactoryDefault();
$container->set(
'annotations',
function () {
return new Apcu(
[
'lifetime' => 86400
]
);
}
);
Now, in the base controller, implement the beforeExecuteRoute method:
<?php
namespace MyApp\Controllers;
use Phalcon\Annotations\Adapter\Apcu;
use Phalcon\Events\Event;
use Phalcon\Mvc\Dispatcher;
use Phalcon\Mvc\Controller;
use MyApp\Components\Auth;
/**
* @property Apcu $annotations
* @property Auth $auth
*/
class BaseController extends Controller
{
/**
* @param Event $event
* @param Dispatcher $dispatcher
*
* @return bool
*/
public function beforeExecuteRoute(
Dispatcher $dispatcher
) {
$controllerName = $dispatcher->getControllerClass();
$actionName = $dispatcher->getActionName() . 'Action';
$data = $this
->annotations
->getMethod($controllerName, $actionName)
;
$access = $data->get('Access');
$aclGroups = $access->getArguments();
$user = $this->acl->getUser();
$groups = $user->getRelated('groups');
$userGroups = [];
foreach ($groups as $group) {
$userGroups[] = $group->grp_name;
}
$allowed = array_intersect($userGroups, $aclGroups);
$allowed = (count($allowed) > 0);
if ($allowed) {
return true;
}
$dispatcher->forward(
[
'controller' => 'session',
'action' => 'login',
]
);
return false;
}
}
In your controllers:
<?php
namespace MyApp\Controllers;
use MyApp\Controllers\BaseController;
/**
* @Private(true)
*/
class Invoices extends BaseController
{
/**
* @Access(
* 'Administrators',
* 'Accounting',
* 'Users',
* 'Guests'
* )
*/
public function indexAction()
{
}
/**
* @Access(
* 'Administrators',
* 'Accounting',
* )
*/
public function listAction()
{
}
/**
* @Access(
* 'Administrators',
* 'Accounting',
* )
*/
public function viewAction()
{
}
}
Additional Resources¶
Exceptions¶
Any exceptions thrown in the Phalcon\Annotations namespace will be of type Phalcon\Annotations\Exception. You can use these exceptions to selectively catch exceptions thrown only from this component.
<?php
use Phalcon\Annotations\Adapter\Memory;
use Phalcon\Annotations\Exception;
use Phalcon\Mvc\Controller;
class IndexController extends Controller
{
public function index()
{
try {
$adapter = new Memory();
$reflector = $adapter->get('Invoices');
$annotations = $reflector->getClassAnnotations();
foreach ($annotations as $annotation) {
echo $annotation->getExpression('unknown-expression');
}
} catch (Exception $ex) {
echo $ex->getMessage();
}
}
}
Granular Exceptions¶
As of 5.14 the component raises granular subclasses so callers can catch a specific failure mode. Three subclasses extend Phalcon\Annotations\Exception; one keeps its original SPL parent (RuntimeException) because the historical throw site used that type.
| Class | Parent | Thrown when |
|---|---|---|
Phalcon\Annotations\Exceptions\AnnotationNotFound | Phalcon\Annotations\Exception | A named annotation is requested from a Collection that does not contain it. |
Phalcon\Annotations\Exceptions\AnnotationsDirectoryNotWritable | Phalcon\Annotations\Exception | The Stream adapter fails to write a serialized annotation file. |
Phalcon\Annotations\Exceptions\CannotReadAnnotationData | RuntimeException | The Stream adapter cannot read a serialized annotation file. |
Phalcon\Annotations\Exceptions\UnknownAnnotationExpression | Phalcon\Annotations\Exception | An annotation expression has an unrecognized AST type. |