Documentação do Symfony - versão 3.4
Renderizada do repositório symfony-docs-pt-BR no Github
Na conclusão da segunda parte deste livro falei sobre um grande benefício
do uso dos componentes do Symfony: a interoperabilidade entre todos os
frameworks e aplicações que os utilizam. Vamos dar um grande passo neste
sentido fazendo o nosso framework implementar a HttpKernelInterface
:
namespace Symfony\Component\HttpKernel;
interface HttpKernelInterface
{
/**
* @return Response A Response instance
*/
function handle(Request $request, $type = self::MASTER_REQUEST, $catch = true);
}
HttpKernelInterface
é, provavelmente, o trecho de código mais importante no
componente HttpKernel, sem brincadeira. Frameworks e aplicações que implementam
essa interface são completamente interoperáveis. Além disso, vários recursos
incríveis virão com ela, de graça.
Atualize seu framework para que ele implemente essa interface:
// example.com/src/Framework.php
// ...
use Symfony\Component\HttpKernel\HttpKernelInterface;
class Framework implements HttpKernelInterface
{
// ...
public function handle(Request $request, $type = HttpKernelInterface::MASTER_REQUEST, $catch = true)
{
// ...
}
}
Mesmo essa mudança parecendo trivial, ela nos traz muito! Vamos falar sobre uma das funcionalidades mais impressionantes: suporte à HTTP caching transparente.
A classe HttpCache
implementa um proxy reverso cheio de recursos, escrito em
PHP; ele implementa a HttpKernelInterface
e envolve uma outra
instância HttpKernelInterface
:
// example.com/web/front.php
$framework = new Simplex\Framework($dispatcher, $matcher, $resolver);
$framework = new HttpKernel\HttpCache\HttpCache($framework, new HttpKernel\HttpCache\Store(__DIR__.'/../cache'));
$framework->handle($request)->send();
Isso é tudo que precisa para adicionar suporte a cache HTTP no nosso framework. Não é surpreendente?
A configuração do cache precisa ser feita através de cabeçalhos HTTP cache. Por exemplo,
para armazenar em cache uma resposta por 10 segundos, use o método Response::setTtl()
:
// example.com/src/Calendar/Controller/LeapYearController.php
public function indexAction(Request $request, $year)
{
$leapyear = new LeapYear();
if ($leapyear->isLeapYear($year)) {
$response = new Response('Yep, this is a leap year!');
} else {
$response = new Response('Nope, this is not a leap year.');
}
$response->setTtl(10);
return $response;
}
Tip
Se, como eu, você está executando o framework a partir da linha de comando
simulando pedidos (Request::create('/is_leap_year/2012')
), você pode
depurar facilmente instâncias Response
através do dump de sua representação string
(echo $response;
) já que, ele exibe todos os cabeçalhos bem como o conteúdo da
resposta.
Para validar se está funcionando corretamente, adicione um número aleatório ao conteúdo da resposta e verifique se o número só muda a cada 10 segundos:
$response = new Response('Yep, this is a leap year! '.rand());
Note
Ao implantar em seu ambiente de produção, continue usando o proxy reverso do Symfony (ótimo para hospedagem compartilhada) ou, ainda melhor, mude para um proxy reverso mais eficiente como o Varnish.
O uso de cabeçalhos HTTP Cache para gerenciar o cache da sua aplicação é muito poderoso e
permite ajustar finamente a sua estratégia de cache, pois, você pode usar tanto o
modelo expiration
quanto o validation
da especificação HTTP. Se não estiver
confortável com esses conceitos, leia o capítulo HTTP caching na documentação
do Symfony.
A classe Response
contém muitos outros métodos que permitem configurar o
cache HTTP com muita facilidade. Um dos mais poderosos é o setCache()
uma vez que ele
abstrai as estratégias mais utilizadas de cache em um simples array:
$date = date_create_from_format('Y-m-d H:i:s', '2005-10-15 10:00:00');
$response->setCache(array(
'public' => true,
'etag' => 'abcde',
'last_modified' => $date,
'max_age' => 10,
's_maxage' => 10,
));
// it is equivalent to the following code
$response->setPublic();
$response->setEtag('abcde');
$response->setLastModified($date);
$response->setMaxAge(10);
$response->setSharedMaxAge(10);
Ao usar o modelo de validação, o método isNotModified()
permite que você
facilmente corte o tempo de resposta gerando a mesma o mais cedo
possível:
$response->setETag('whatever_you_compute_as_an_etag');
if ($response->isNotModified($request)) {
return $response;
}
$response->setContent('The computed content of the response');
return $response;
Usar o cache HTTP é ótimo, mas, e se você não pode armazenar em cache a página inteira? E se você pode armazenar em cache tudo, menos algumas barras laterais que são mais dinâmicas que o resto do conteúdo? Temos o Edge Side Includes (ESI) para nos socorrer! Em vez de gerar todo o conteúdo de uma só vez, o ESI permite que você marque uma região de uma página como sendo o conteúdo da chamada de um sub-pedido:
Este é o conteúdo de sua página
2012 é um ano bissexto? <esi:include src="/leapyear/2012" />
Algum outro conteúdo
Para as tags ESI serem suportadas pelo HttpCache, você precisa passar uma instância da
classe ESI
. A classe ESI
automaticamente realiza o parse das tags ESI e faz
sub-pedidos para convertê-las em seu conteúdo apropriado:
$framework = new HttpKernel\HttpCache\HttpCache(
$framework,
new HttpKernel\HttpCache\Store(__DIR__.'/../cache'),
new HttpKernel\HttpCache\ESI()
);
Note
Para o ESI funcionar, você precisa usar um proxy reverso que o suporte, como a implementação do Symfony. Varnish é a melhor alternativa e é Open-Source.
Ao utilizar estratégias complexas de cache HTTP e/ou muitas include tags ESI, pode ser difícil entender por que e quando um recurso deve ser armazenado em cache ou não. Para depurar facilmente, você pode ativar o modo de depuração:
$framework = new HttpCache($framework, new Store(__DIR__.'/../cache'), new ESI(), array('debug' => true));
O modo de depuração acrescenta um cabeçalho X-Symfony-Cache
em cada resposta que
descreve o que a camada de cache fez:
1 2 3 | X-Symfony-Cache: GET /is_leap_year/2012: stale, invalid, store
X-Symfony-Cache: GET /is_leap_year/2012: fresh
|
O HttpCache tem muitas funcionalidades, uma delas é o suporte as
extensões HTTP Cache-Control stale-while-revalidate
e stale-if-error
como definidas no RFC 5861.
Com a adição de uma única interface, o nosso framework agora pode se beneficiar de muitas funcionalidades incorporadas no componente HttpKernel; sendo o cache HTTP apenas uma delas, porém importante, pois ele pode fazer a suas aplicações voarem!