Назад | Перейти на главную страницу

Передача переменной Apache в директиву SetHandler

Я пытаюсь иметь динамически определяемый сокет Unix используется для разных отдельных «приложений» PHP (разные сокеты приводят к разным PHP-FPM пулов) на моем сервере Apache (2.4.18) (используя mod_proxy_fcgi):

<DirectoryMatch "/home/apps/app_(?<appname>[a-zA-Z]+)">

    <FilesMatch \.php$>
        SetHandler "proxy:unix://var/run/app_%{env:MATCH_APPNAME}.sock|fcgi://localhost:42001"
    </FilesMatch>

</DirectoryMatch>

К сожалению Apache, похоже, не распознает / не оценивает переменную определяется совпадением регулярного выражения в <DirectoryMatch> директива при использовании с SetHandler директива.

В error.log говорит это:

[proxy:error] (2)No such file or directory: AH02454: FCGI: attempt to connect to Unix domain socket /run/app_%{MATCH_APPNAME}e.sock (*) failed

Как можно передать переменную в SetHandler директива? Мой PHP-FPM работает правильно, когда я использую «жесткий» путь к UDS (например, "proxy:unix://var/run/app_someappname.sock|fcgi://localhost:42001", но при использовании в нем переменной возникает проблема.


Я также попытался построить путь, используя несколько объединенных вместе переменных:

<FilesMatch \.php$>
    Define one "proxy:unix://var/run/app_"
    Define two ".sock|fcgi://localhost:42001"
    Define final ${one}%{MATCH_APPNAME}e${two}
    Header set HANDLER_PATH ${final}
</FilesMatch>

В этом случае источником PHP является не обрабатывается PHP, но заголовок ответа HTTP для URL http://127.0.0.1/apps/someappname/index.php (правильно) содержит: HANDLER_PATH: proxy:unix://var/run/app_someappname.sock|fcgi://localhost:42001

Но когда я прохожу мимо ${final} переменная для SetHandler, сервер возвращает 503 Service Unavailable и error.log говорит:

*: using default reverse proxy worker for unix://var/run/app_%{MATCH_APPNAME}e.sock|fcgi://localhost:42001/home/apps/app_someappname/www/index.php (no keepalive)
*: rewrite of url due to UDS(/run/app_%{MATCH_APPNAME}e.sock): fcgi://localhost:42001/home/apps/app_someappname/www/index.php (proxy:fcgi://localhost:42001/home/apps/app_someappname/www/index.php)
AH01143: Running scheme unix handler (attempt 0)
AH01076: url: fcgi://localhost:42001/home/apps/app_someappname/www/index.php proxyname: (null) proxyport: 0
AH01078: serving URL fcgi://localhost:42001/home/apps/app_someappname/www/index.php
AH00942: FCGI: has acquired connection for (*)
AH00944: connecting fcgi://localhost:42001/home/apps/app_someappname/www/index.php to localhost:42001
AH02545: fcgi: has determined UDS as /run/app_%{MATCH_APPNAME}e.sock
AH00947: connected /home/apps/app_someappname/www/index.php to httpd-UDS:0
(2)No such file or directory: AH02454: FCGI: attempt to connect to Unix domain socket /run/app_%{MATCH_APPNAME}e.sock (*) failed
AH01079: failed to make connection to backend: httpd-UDS
proxy_util.c(2175): AH00943: FCGI: has released connection for (*)

Что меня поражает, так это то, что когда путь носка помещается в HTTP-заголовок, он правильно разрешается как proxy:unix://var/run/app_someappname.sock, но когда переменная передается в SetHandler, это внезапно становится unix://var/run/app_%{MATCH_APPNAME}e.sock с буквальным %{MATCH_APPNAME}e, например, если бы переменная была разрешена SetHandler сам (и неправильно) ...

Как бы это сделать? Это вообще возможно? Думаю, должно быть.

Вы пробовали RewriteRule с [H= вместо того SetHandler?

RewriteRule будет иметь преимущество оценки аргументов, когда оно выполняется, а не при запуске (типичные директивы не интерполируют переменные для каждого запроса, когда они фактически выполняются)