Obsolete: see Typed Entity in drupal.org.
Use Typed Entity as a namespace for your business logic. It provides a scope to place your business logic, and help you keep your global scope clean of myriads of small functions.
This module provides a simple way to treat you existing entities like typed objects. This will allow you to have a more maintainable and easier to debug codebase.
You have modelled your content using Drupal's entities and Field API, but now
you are stuck with stdClass
and field_get_items
. You could go further and
use Entity Metadata Wrapper
to have a more comprehensive entity interaction.
That's fine so far.
Now imagine that you need to have some custom business logic that applies only
to node > article
. You could create your custom module and have your custom
function:
/**
* Gets the aspect ratio of the field_image attached to a node article.
*
* @param object $node
* The node.
*
* @return array
* Array with width and height.
*/
function custom_module_article_image_ratio($node) {
// Do some computation.
return array('width' => $width, 'height' => $height);
}
As the project grows this is hard to maintain and can get pretty wild. Imagine
custom_module_article_related_article_image_ratio
, and the several of
variations –even those that don't relate to image ratio. Gasp!–.
Create a typed entity in your custom module under src/TypedEntity
and make it implement TypedEntityInterface
. Basically you just have to follow the example.
For our example above we would create TypedNodeArticle
and then another class called TypedFileImage
. TypedFileImage
would have a method to get the aspect ratio:
<?php
/**
* @file
* Contains \Drupal\custom_module\TypedEntity\TypedFileImage.
*/
namespace Drupal\custom_module\TypedEntity;
use Drupal\typed_entity\TypedEntity\TypedEntity;
class TypedFileImage extends TypedEntity implements TypedFileImageInterface {
/**
* Gets the aspect ratio for the underlying entity.
*
* @return array
* The width and height.
*/
public function getAspectRatio() {
$file = $this->getEntity();
// Do some computation on $file.
return array('width' => $width, 'height' => $height);
}
}
Now you can do things like:
// Use the following comment to hint your IDE about the object type. That will
// enable IDE autocomplete.
/** @var \Drupal\typed_entity_example\TypedEntity\Node\Article $typed_node */
$typed_article = TypedEntityManager::create('node', $node);
// custom_module_article_image_ratio(…)
$ratio = $typed_article
->getImage()
->getAspectRatio();
// custom_module_article_related_article_image_ratio(…)
$ratio = $typed_article
->getRelatedArticle()
->getImage()
->getAspectRatio();
To declare your typed entities you just have to create the class following the name convention:
- Typed<EntityTypeCamelCase><BundleCamelCase> (Ex:
TypedNodeArticle
). - Typed<EntityTypeCamelCase> (Ex:
TypedUser
,TypedNode
).
That class will be discovered and returned by the factory function in
TypedEntityManager
. Use:
$typed_entity = TypedEntityManager::create($entity_type, $entity);
If you don't like the naming convention or you prefer to use something else, you can provide the class for your entity. For that you will have to implement hook_typed_entity_registry_info
. See the example.
/**
* Implements hook_typed_entity_registry_info().
*/
function custom_module_typed_entity_registry_info() {
$items['user'] = array(
'entity_type' => 'user',
'class' => '\\Drupal\\custom_module\\Foo\\Bar\\User',
);
$items['image'] = array(
'entity_type' => 'file',
'bundle' => 'image',
'class' => '\\Drupal\\custom_module\\File\\Image',
);
return $items;
}
This module uses the PHP magic methods to allow you to do things like:
$typed_entity = TypedEntityManager::create($entity_type, $entity);
// Access the uuid property on the underlying entity.
print $typed_entity->uuid;
// Set properties on the entity.
$typed_entity->status = NODE_PUBLISHED;
// Execute entity methods.
print $typed_entity->myEntityMethod('arg1', 2);
This has the benefit (over doing $typed_entity->getEntity()->myEntityMethod('arg1', 2)
)
that you can then pass the $typed_entity
in place of a node to functions that
access the node properties or call custom methods on the entity.
Besides the benefits of OOP you can have more structured Drupal entities with clearer relationships. Stop passing ($entity_type, $entity) around, an entity should be able to know its own type and bundle.
Care to add tests? You can even have unit testing on your custom business logic (make sure those computations on the aspect ratio return the expected values).
Check out the unit test example.
This module uses unit testing as an example of how you should test your custom business logic. Sometimes your custom logic contains calls to the drupal api that is not loaded for unit testing. To work around that you can use the Service Container module.
This module needs an extra patch to do this, so you will have to patch: