How to implement your own fields inclusion rules with JMS Serializer
A common good practice in REST API is to allow clients to specify a list of fields the server has to respond in the resources representation, allowing lighter responses and more efficient bandwidth usage. This is even a recommendation in JSON API, so today I’ll show you how you can implement this inside your PHP projets using the JMS Serializer (via Symfony2 or not).
Section intitulée introducing-exclusionstrategyIntroducing ExclusionStrategy
We gonna dig in the code a little, and see how to use the ExclusionStrategy. You may not know, but you’ve surely already used it, as it’s used by the @Groups and @Version annotations. They are driven by the same interface: ExclusionStrategyInterface.
interface ExclusionStrategyInterface
{
public function shouldSkipClass(ClassMetadata $metadata, Context $context);
public function shouldSkipProperty(PropertyMetadata $property, Context $context);
}
The two methods names speak for themselves:
- shouldSkipClass: Whether the class should be skipped;
- shouldSkipProperty: Whether the property should be skipped.
As you can see the first argument is a [Class|Property]Metadata from the great Metadata library, so you get all the annotations and values from the currently targeted class or property.
Section intitulée fields-white-list-the-right-wayFields white list the right way
By default, our resources must have a view with all the allowed fields, this is the role of Groups or Exclude, you can chose the exclusion strategy of your choice:
class Pony
{
/**
* @Serializer\Groups({"Default"})
*/
private $id;
/**
* @Serializer\Groups({"Default", "MyCustomViewName"})
*/
protected $title;
/**
* @Serializer\Groups({"Default"})
*/
protected $body;
/**
* @Serializer\Groups({"Admin"})
*/
protected $viewCount;
Then comes the Fields exclusion strategy. We need our controllers to tell the Serializer “Hey bro, please only include id and body in the response, the client only asked for those”. Holding this information is the role of the SerializationContext, a class knowing where we are, which format, what are the exclusion strategies, should we serialize null values…
Let’s call the serializer with a custom context:
$context = new SerializationContext();
$groups[] = 'Default';
$context->setGroups($groups);
$serializer->serialize(new Pony(), 'json', $context);
With this code my Pony id, title and body fields will be serialized. This is our API default view and we now want to filter by a end-user field list.
We can add our new exclusion strategy on the context:
$fieldList = ['id', 'title'];
$context->addExclusionStrategy(
new FieldsListExclusionStrategy($fieldList)
);
The code is pretty simple:
namespace Acme\Bundle\ApiBundle\Serializer\Exclusion;
use JMS\Serializer\Exclusion\ExclusionStrategyInterface;
use JMS\Serializer\Metadata\ClassMetadata;
use JMS\Serializer\Metadata\PropertyMetadata;
use JMS\Serializer\Context;
class FieldsListExclusionStrategy implements ExclusionStrategyInterface
{
private $fields = array();
public function __construct(array $fields)
{
$this->fields = $fields;
}
/**
* {@inheritDoc}
*/
public function shouldSkipClass(ClassMetadata $metadata, Context $navigatorContext)
{
return false;
}
/**
* {@inheritDoc}
*/
public function shouldSkipProperty(PropertyMetadata $property, Context $navigatorContext)
{
if (empty($this->fields)) {
return false;
}
$name = $property->serializedName ?: $property->name;
return !in_array($name, $this->fields);
}
}
This is a very straightforward example, on a real world API, depth handling should also be considered because we always want first level node, whatever their names are, so we can add something like this:
// Keep first level
if ($navigatorContext->getDepth() == 1) {
return false;
}
That’s it! Our JSON representation is now only displaying id and title, and as this is an exclusion, there is no way to add a field already excluded by another strategy.
Section intitulée you-can-do-moreYou can do more
Implementing the whole JSON API specification (with the fields[TYPE] notation) would be more related on your model and choices so we do not cover it here. ExclusionStrategy are a powerful tool and you can do a lot of logic in it, adding your own annotations, making it a service…
Hope you liked this quick article, happy coding!
Commentaires et discussions
Ces clients ont profité de notre expertise
La société AramisAuto a fait appel à JoliCode pour développer au forfait leur plateforme B2B. L’objectif était de développer une nouvelle offre à destination des professionnels ; déjà testé commercialement, pro.aramisauto.com est la concrétisation de 3 mois de développement. Le service est indépendant de l’infrastructure existante grâce à la mise en…
Dans le cadre du renouveau de sa stratégie digitale, Orpi France a fait appel à JoliCode afin de diriger la refonte du site Web orpi.com et l’intégration de nombreux nouveaux services. Pour effectuer cette migration, nous nous sommes appuyés sur une architecture en microservices à l’aide de PHP, Symfony, RabbitMQ, Elasticsearch et Docker.
Pour améliorer les performances et la pertinence des recherches sur le site e-commerce, JoliCode a réalisé un audit approfondi du moteur Elasticsearch existant. Nous avons optimisé les processus d’indexation, réduisant considérablement les temps nécessaires tout en minimisant les requêtes inutiles. Nous avons également ajusté les analyses pour mieux…