如何在使用 mod_proxy 时正确增加 Apache 的超时时间

如何在使用 mod_proxy 时正确增加 Apache 的超时时间

背景:

我有机器对机器通信系统,客户端不时向 Apache 服务器发送数据。由于使用移动 2G 进行通信,因此通信速度很慢。Apache 使用 mod_wsgi(用于 Django)。

由于客户端在开始通信时不知道实际的数据量,因此客户端和服务器之间使用分块传输编码。由于 mod_wsgi 不支持分块传输编码,因此我使用 mod_proxy,它接收分块并在收到所有数据后将它们作为单个请求传递给 mod_wsgi,并添加内容长度信息。

问题:

我将服务器上的 Ubuntu 从 16.04 更新到 18.04,之后客户端和服务器之间的某些通信开始失败。从 Apache 的 access.log 中我可以看到,服务器对包含“大量”数据的客户端响应 408。大量是指在这种情况下需要大约 60 秒才能传输的数据量。客户端没有任何变化。

我了解到,对于 Apache 2.4(我在服务器的 Ubuntu 18.04 上安装),超时时间为 60 秒,对于早期的 Apache 版本,超时时间为 300 秒。我猜这就是在“大”数据的情况下与新服务器操作系统通信失败的原因。

边注:

我注意到启用 mod_security 可以解决问题(“大”数据时不会出现 408),但是当我禁用 mod_security 并重新启动 Apache 时,408 又回到了画面。security.conf 中没有任何与超时相关的内容。

问题:

如何正确增加TimeOut?

迄今已尝试:

由于 Apache 确实不是我的核心竞争力,我一直在尝试增加某些地方的超时时间(也尝试过 KeepAliveTimeout 尽管我认为这不是正确的方法):

apache.conf 片段:

#
# Timeout: The number of seconds before receives and sends time out.
#
Timeout 300

#
# KeepAlive: Whether or not to allow persistent connections (more than
# one request per connection). Set to "Off" to deactivate.
#
KeepAlive On

#
# MaxKeepAliveRequests: The maximum number of requests to allow
# during a persistent connection. Set to 0 to allow an unlimited amount.
# We recommend you leave this number high, for maximum performance.
#
MaxKeepAliveRequests 100

#
# KeepAliveTimeout: Number of seconds to wait for the next request from the
# same client on the same connection.
#
KeepAliveTimeout 300

example.com.conf 代码片段:

ProxyRequests Off
ProxyTimeout 300

<Proxy http://example.com:81>
        Order deny,allow
        Allow from all
</Proxy>

<VirtualHost *:80>
        SetEnv proxy-sendcl 1
        ProxyPass / http://example.com:81/ connectiontimeout=300 timeout=300
        ProxyTimeout 300
        ProxyPassReverse / http://example.com:81/
        ProxyPreserveHost On
        ProxyVia Full

        <Directory proxy:*>
                Order deny,allow
                Allow from all
        </Directory>
</VirtualHost>

Listen 81

<VirtualHost *:81>
    # The ServerName directive sets the request scheme, hostname and port that
    # the server uses to identify itself. This is used when creating
    # redirection URLs. In the context of virtual hosts, the ServerName
    # specifies what hostname must appear in the request's Host: header to
    # match this virtual host. For the default virtual host (this file) this
    # value is not decisive as it is used as a last resort host regardless.
    # However, you must set it for any further virtual host explicitly.
    ServerName example.com

答案1

您确定 408 错误是由于 Apache 处理请求的时间过长造成的吗?

HTTP 408 是“请求超时”。当客户端连接但在触发超时之前未发送任何数据时,就会发生这种情况。您可以使用 ReqTimeout 模块(https://httpd.apache.org/docs/2.4/mod/mod_reqtimeout.html) 来增加接收请求前的超时时间。请记住,通过增加超时时间,您将有更多的 Apache 进程忙于等待客户端断开或使用套接字。

相关内容