运行 Nginx 反向代理背后的 Apache(带 .htaccess)应用程序

Running Apache (with .htaccess) App Behind Nginx Rerverse Proxy

我最近开始尝试对我的服务进行 Docker 化,并且我已经到了对已经构建了映像的所有内容进行 Docker 化的地步。现在我正在尝试为 facileManager (FM) 构建一个图像,它还没有。我大部分时间都在使用它,但是当 运行 在 Nginx 后面使用它时我遇到了问题。 FM 通常是一个 apache-php 应用程序,不包括 Nginx 的安装说明。我注意到我的 container/image 是,当我通过已发布的端口直接连接到它时它工作正常,但如果我尝试通过 Nginx 连接到它,它会出错并抱怨 .htaccess 文件不工作。我不是 Apache 或 Nginx 的专家,所以我进行了谷歌搜索,但除了 Wordpress 的“漂亮网址”存在类似问题外,我没有想出太多,所以我希望这里有人能伸出援手。

首先是应用程序的 Github 存储库:https://github.com/WillyXJ/facileManager/tree/ea159f5f6112727de8422c552aa05b6682aa4d79/server

.htaccess 文件具体为:

<IfModule mod_headers.c>
    <FilesMatch "\.(js|css|txt)$">
        Header set Cache-Control "max-age=7200"
    </FilesMatch>
    <FilesMatch "\.(jpe?g|png|gif|ico)$">
        Header set Cache-Control "max-age=2592000"
    </FilesMatch>
</IfModule>

<IfModule mod_rewrite.c>
RewriteEngine On

RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule . index.php [L]
</IfModule>

我得到的确切错误来自这里: https://github.com/WillyXJ/facileManager/blob/ea159f5f6112727de8422c552aa05b6682aa4d79/server/fm-includes/init.php#L153

if (!defined('INSTALL')) {
        if (@dns_get_record($_SERVER['SERVER_NAME'], DNS_A + DNS_AAAA)) {
            $test_output = getPostData($GLOBALS['FM_URL'] . 'admin-accounts.php?verify', array('module_type' => 'CLIENT'));
            $test_output = isSerialized($test_output) ? unserialize($test_output) : $test_output;
            if (strpos($test_output, 'Account is not found.') === false) {
                $message = sprintf(_('The required .htaccess file appears to not work with your Apache configuration which is required by %1s. '
                        . 'AllowOverride None in your configuration may be blocking the use of .htaccess or %s is not resolvable.'),
                        $fm_name, $_SERVER['SERVER_NAME']);
                if ($single_check) {
                    bailOut($message);
                } else {
                    $requirement_check .= displayProgress(_('Test Rewrites'), false, 'display', $message);
                    $error = true;
                }
            } else {
                if (!$single_check) $requirement_check .= displayProgress(_('Test Rewrites'), true, 'display');
            }
        }
    }

Nginx 配置:

server {
        listen 80;

        server_tokens off;

        location /.well-known/acme-challenge/ {
                root /var/www/certbot;
        }

        location / {
                return 301 https://$host$request_uri;
        }
}

server {
        listen 443 ssl;

        server_name bound.example.com;

        ssl_certificate /etc/nginx/ssl/live/bound.example.com/fullchain.pem;
        ssl_certificate_key /etc/nginx/ssl/live/bound.example.com/privkey.pem;
        include /etc/nginx/ssl/options-ssl-nginx.conf;
        ssl_dhparam /etc/nginx/ssl/ssl-dhparams.pem;

        location / {
                proxy_pass http://FM.;     <<< Docker service name
                proxy_http_version 1.1;
                proxy_set_header Upgrade $http_upgrade;
                proxy_set_header Connection "upgrade";
                proxy_set_header Host $host;
                proxy_set_header X-Real-IP $remote_addr;
                proxy_set_header X-Forward-For $proxy_add_x_forwarded_for;
        }
}

我的Dockerfile/commands运行:https://github.com/MeCJay12/facileManager-docker

FROM php:7.4-apache

ENV TZ=UTC
ENV Version=4.2.0
ARG DEBIAN_FRONTEND=noninteractive
WORKDIR /src

RUN apt-get update \
    && apt-get -qqy install wget libldb-dev libldap2-dev tzdata \
    && wget http://www.facilemanager.com/download/facilemanager-complete-$Version.tar.gz \
    && tar -xvf facilemanager-complete-$Version.tar.gz \
    && mv facileManager/server/* /var/www/html/

RUN ln -s /usr/lib/x86_64-linux-gnu/libldap.so /usr/lib/libldap.so \
    && docker-php-ext-install mysqli ldap \
    && a2enmod rewrite dump_io

COPY php.ini /usr/local/etc/php/php.ini
RUN rm -r /src

奖金问题:直接通过已发布的端口访问此 container/image 时,图像全部损坏。我认为它是相关的,因为 .htaccess 文件包含对图像文件的引用。

在此先感谢您的帮助!

点数:

  • 在您的代理通行证中包含 $request_uri
  • 在您的代理位置块中提供解析器
  • 在 Docker 的网络堆栈中为您的容器声明一个条目
  • 在您的服务名称中使用全部小写字母

下面是我用来反向代理到 Ubiquiti Unifi 容器的配置文件。我所有的 certbot 都是在场外处理的,所以我不需要在这里考虑。如果您比较我们的位置块,问题可能会立即变得明显,但为了清楚起见,我会进行解释。

您需要查看的是您的 Proxy Pass 指令。这当然是魔法代理发生的地方。我注意到您没有包含 $request_uri,因此 nginx 收到的任何 bound.example.com/testpage1 请求,都会向上游 apache 服务器发送 bound.example.com 请求。当然,如果您需要包括一个端口,就像我在这里所做的那样 8443,这里也是这样做的地方。

如果包含此变量,应该可以解决您的问题。

以下内容没有回答您的问题,但我想我也会将其作为一些有用的信息包括在内。

此外,我只想指出我已经包含了一个解析器。 IP 地址 127.0.0.11 指向 Docker's internal DNS resolver. 可能您不需要包括它,但是我自己这样做是为了确保我没有遇到奇怪的问题。 最后,我只想建议您考虑升级您的 SSL 设置,以确保您免受来自较弱 SSL / TLS 版本的攻击。

我希望将变量 $request_uri 添加到您的代理传递指令中即可让您的网站正常运行。

server {
    listen       80;
    server_name  unifi.localdomain;

    location / {
        return 301 https://$host$request_uri;
    }
}

server {
    listen                     443 ssl;
    server_name                unifi.localdomain;

    add_header                 Strict-Transport-Security "max-age=31536000; includeSubDomains; preload;";

    ssl_protocols              TLSv1.3 TLSv1.2;
    ssl_session_cache          builtin:1000  shared:SSL:10m;
    ssl_session_timeout        1d;
    ssl_ciphers                ALL:!RSA:!CAMELLIA:!aNULL:!eNULL:!LOW:!3DES:!MD5:!EXP:!PSK:!SRP:!DSS:!RC4:!SHA1:!SHA256:!SHA384;
    ssl_prefer_server_ciphers  on;
    ssl_stapling               on;
    ssl_stapling_verify        on;
    resolver                   8.8.8.8 8.8.4.4 valid=300s;
    resolver_timeout           5s;
    ssl_certificate            /etc/nginx/certs/fullchain.pem;
    ssl_certificate_key        /etc/nginx/certs/privkey.pem;
    ssl_dhparam                /etc/nginx/certs/dhparam.pem;

    location / {
        resolver          127.0.0.11;
        proxy_pass        https://unifi:8443$request_uri;
        proxy_set_header  X-Real-IP  $remote_addr;
        proxy_set_header  X-Forwarded-For $remote_addr;
        proxy_set_header  Host $host;
        proxy_set_header  Upgrade $http_upgrade;
        proxy_set_header  Connection "Upgrade";
    }
}

另外

除此之外,我有机会在我自己的环境中对此进行了测试,并且似乎能够使其正常工作(尽管我没有完成安装过程)。我发现如果我没有在 docker 网络堆栈上为此容器声明网络地址,那么 docker 解析器中将没有 DNS 条目。我使用了以下 docker 配置;

version: "3.4"

services:
  fm:
    build: ./boundexample
    container_name: "fm"
    networks:
      primary:
        ipv4_address: "172.19.0.11"
  nginx:
    image: "nginx:stable"
    container_name: "nginx"
    restart: "unless-stopped"
    environment:
      TZ: "Australia/Perth"
    networks:
      primary:
        ipv4_address: "172.19.0.2"
    ports:
      - "10.0.0.4:80:80/tcp"
      - "10.0.0.4:443:443/tcp"
networks:
  primary:
    driver: bridge
    driver_opts:
      parent: enp3s0.2
    ipam:
      config:
        - subnet: "172.19.0.0/24"

其中目录 boundexample 指向 local copy of your github repository。我的 nginx 配置与您提供的相同,只是我针对自己的 SSL 进行了修改,并在我的位置块中使用了以下内容;

resolver          127.0.0.11;
proxy_pass http://fm$request_uri;

也有可能在您的 docker 配置中使用大写字母 FM 可能会给您带来问题,我自己无法使用大写字母,因为 docker-compose 不允许。

https://github.com/WillyXJ/facileManager/issues/491 中找到了解决方案。需要删除 webapp 中运行不正常的检查。