背景:
我有机器对机器通信系统,客户端不时向 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 进程忙于等待客户端断开或使用套接字。