Executando Symfony 5 com proxy reverso no subdiretório
Eu gosto de executar um aplicativo Symfony 5 atrás de um proxy reverso que fornece o seguinte endpoint:
https://my.domain/service1/
A configuração do proxy é basicamente esta:
ProxyPass /marketsy/ http://internal.service1/
No servidor ao qual o proxy reverso está se conectando, eu uso a seguinte regra do apache para servir meu aplicativo Symfony:
<VirtualHost *:80>
ServerName internal.service1
DocumentRoot /webroot/service1/public
<FilesMatch \.php$> SetHandler proxy:unix:/run/php/php7.2-fpm-ui.sock|fcgi://localhost SetEnvIfNoCase ^Authorization$ "(.+)" HTTP_AUTHORIZATION=$1
SetEnv HTTP_X_FORWARDED_PROTO "https"
</FilesMatch>
<Directory /webroot/service1/public>
AllowOverride None
Require all granted
FallbackResource /index.php
</Directory>
<Directory /webroot/service1/public/bundles>
FallbackResource disabled
</Directory>
</VirtualHost>
O aplicativo em si é reqachable, mas Symfony não consegue lidar com o prefixo de caminho "service1".
Por exemplo, ele tenta acessar o profiler em https://my.domain/_wdt/8e3926 ao invés de https://my.domain/service1/_wdt/8e3926 e ao lado da rota raiz todo o roteamento não está funcionando:
Por exemplo: Quando tento acessar https://my.domain/service1/my/page eu serei redirecionado para https://my.domain/my/page
Agora minha pergunta é, como posso configurar o Symfony para saber sobre o prefixo de caminho "service1" ao gerar urls ets.
Respostas
Maneira adequada de fazer isso (exemplo):
Crio src/Controller/BarController.php
<?php
namespace App\Controller;
use Symfony\Component\HttpFoundation\Response;
class BarController
{
public function index()
{
return new Response('<p>Bar controler response</p>');
}
}
e src/Controller/FooController.php
<?php
namespace App\Controller;
use Symfony\Component\HttpFoundation\Response;
class FooController
{
public function index()
{
return new Response('<p>Foo controler response</p>');
}
}
Crio config/routes/prefix-routes.yaml
index:
path: /
controller: App\Controller\DefaultController::index
bar:
path: /bar
controller: App\Controller\BarController::index
foo:
path: /foo
controller: App\Controller\FooController::index
e editar roteamento config/routes.yaml- exclua seu conteúdo e apenas coloque:
prefixed:
resource: "routes/prefix-routes.yaml"
prefix: service1
todos os controladores estão agora disponíveis em urls:
http://localhost/service1/ for DefaultController.php
http://localhost/service1/bar for BarController.php
http://localhost/service1/foo for FooController.php
Se você quiser que seu criador de perfil funcione com service1prefixo também, edite config/routes/dev/web_profiler.yamldesta forma:
web_profiler_wdt:
resource: '@WebProfilerBundle/Resources/config/routing/wdt.xml'
prefix: service1/_wdt
web_profiler_profiler:
resource: '@WebProfilerBundle/Resources/config/routing/profiler.xml'
prefix: service1/_profiler
agora eles devem estar disponíveis em:
http://localhost/service1/_wdt... for wdt
http://localhost/service1/_profiler for profiler
Adicionar prefixo para anotações:
Criar controlador src/Controller/AnnoController.php:
<?php
namespace App\Controller;
use Symfony\Component\Routing\Annotation\Route;
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
use Symfony\Component\HttpFoundation\Response;
class AnnoController extends AbstractController
{
/**
* @Route("/anno", name="anno")
*/
public function index()
{
return new Response('<p>Anno controler response</p>');
}
}
editar config/routes/annotations.yamle adicionar prefix: service1:
controllers:
resource: ../../src/Controller/
type: annotation
prefix: service1
kernel:
resource: ../../src/Kernel.php
type: annotation
Agora o prefixo é adicionado às rotas feitas por meio de anotação:
http://localhost/service1/anno for AnnoController.php
Algumas referências:
Symfony Routing Prefix
Chaves de configuração de roteamento do Symfony
Adicionar prefixo solução alternativa rápida e suja para adicionar prefixo service1a todo o roteamento (não recomendado).
Em vez de alterar o roteamento como acima, basta editar src/Kernel.php protected function configureRoutes
e altere todas as $routes->importlinhas adicionando ->prefix('service1')no final para que fique assim:
protected function configureRoutes(RoutingConfigurator $routes): void
{
$routes->import('../config/{routes}/'.$this->environment.'/*.yaml')->prefix('service1');
$routes->import('../config/{routes}/*.yaml')->prefix('service1'); if (is_file(\dirname(__DIR__).'/config/routes.yaml')) { $routes->import('../config/{routes}.yaml')->prefix('service1');
} elseif (is_file($path = \dirname(__DIR__).'/config/routes.php')) { (require $path)($routes->withPath($path), $this);
}
}
todos os controladores estão agora disponíveis em urls:
http://localhost/service1/ for DefaultController.php
http://localhost/service1/bar for BarController.php
http://localhost/service1/foo for FooController.php
bem como o criador de perfil:
http://localhost/service1/_wdt... for wdt
http://localhost/service1/_profiler for profiler
Além da solução fornecida por @Jimmix, tive que escrever um assinante para adicionar um prefixo ao meu caminho de solicitação:
<?php namespace My\Bundle\EventSubscriber;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\HttpKernel\Event\RequestEvent;
use Symfony\Component\HttpKernel\KernelEvents;
class AppPrefixSubscriber implements EventSubscriberInterface {
/** @var string */
private $appPrefix; public function __construct(?string $appPrefix) {
$this->appPrefix = $appPrefix;
}
/**
* Returns events to subscribe to
*
* @return array
*/
public static function getSubscribedEvents() {
return [
KernelEvents::REQUEST => [
['onKernelRequest', 3000]
]
];
}
/**
* Adds base url to request based on environment var
*
* @param RequestEvent $event */ public function onKernelRequest(RequestEvent $event) {
if (!$event->isMasterRequest()) { return; } if ($this->appPrefix) {
$request = $event->getRequest();
$newUri = $this->appPrefix .
$request->server->get('REQUEST_URI'); $event->getRequest()->server->set('REQUEST_URI', $newUri); $request->initialize(
$request->query->all(), $request->request->all(),
$request->attributes->all(), $request->cookies->all(),
$request->files->all(), $request->server->all(),
$request->getContent()
);
}
}
}