Executando Symfony 5 com proxy reverso no subdiretório

Aug 21 2020

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

4 Jimmix Sep 29 2020 at 11:35

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
1 Johni Oct 05 2020 at 19:55

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()
            );
        }
    }

}