Nginx 中 502 和 504 錯誤

Nginx 中 502 和 504 錯誤

本文轉自: https://segmentfault.com/a/1190000002686153

在使用Nginx時,經常會碰到 502 Bad Gateway 和 504 Gateway Time-out 錯誤,下面以 Nginx+PHP-FPM 來分析下這兩種常見錯誤的原因和解決方案。

1. 502 Bad Gateway 錯誤

在 php.ini 和 php-fpm.conf 中分別有這樣兩個配置項:max_execution_time 和 request_terminate_timeout。

這兩項都是用來配置一個 PHP 腳本的最大執行時間的。當超過這個時間時,PHP-FPM不只會終止腳本的執行,還會終止執行腳本的Worker進程。所以Nginx會發現與自己通信的連接斷掉了,就會返回給客戶端502錯誤。

以PHP-FPM的 request_terminate_timeout=30 秒時為例,報 502 Bad Gateway 錯誤的具體信息如下:

1)Nginx錯誤訪問日志:

2013/09/19 01:09:00 [error] 27600#0: *78887 recv() failed (104: Connection reset by peer) while reading response header from upstream, 
client: 192.168.1.101, server: test.com, request: "POST /index.php HTTP/1.1", upstream: "fastcgi://unix:/dev/shm/php-fcgi.sock:", 
host: "test.com", referrer: "http://test.com/index.php"

2)PHP-FPM報錯日志:

WARNING: child 25708 exited on signal 15 (SIGTERM) after 21008.883410 seconds from start
所以只需將這兩項的值調大一些就可以讓PHP腳本不會因為執行時間長而被終止了。request_terminate_timeout 可以覆蓋 max_execution_time,所以如果不想改全局的php.ini,那只改PHP-FPM的配置就可以了。

此外要注意的是Nginx的upstream模塊中的 max_fail 和 fail_timeout 兩項。有時Nginx與上游服務器(如Tomcat、FastCGI)的通信只是偶然斷掉了,但 max_fail 如果設置的比較小的話,那么在接下來的 fail_timeout時間內,Nginx都會認為上游服務器掛掉了,都會返回502錯誤。
所以可以將 max_fail 調大一些,將 fail_timeout 調小一些。

2. 504 Gateway Time-out 錯誤

PHP-FPM設置的腳本最大執行時間已經夠長了,但執行耗時PHP腳本時,發現Nginx報錯從502變為504了。這是為什么呢?

因為我們修改的只是PHP的配置,Nginx中也有關于與上游服務器通信超時時間的配置factcgi_connect/read/send_timeout。

以Nginx超時時間為90秒,PHP-FPM超時時間為300秒為例,報504 Gateway Timeout錯誤時的Nginx錯誤訪問日志如下:

 2013/09/19 00:55:51 [error] 27600#0: *78877 upstream timed out (110: Connection timed out) while reading response header from upstream, 
 client: 192.168.1.101, server: test.com, request: "POST /index.php HTTP/1.1", upstream: "fastcgi://unix:/dev/shm/php-fcgi.sock:", 
 host: "test.com", referrer: "http://test.com/index.php"

調高這三項的值(主要是read和send兩項,默認不配置的話Nginx會將超時時間設為60秒)之后,504錯誤也解決了。

而且這三項配置可以配置在http、server級別,也可以配置在location級別。擔心影響其他應用的話,就配置在自己應用的location中吧。

要注意的是 factcgi_connect/read/send_timeout 是對 FastCGI 生效的,而 proxy_connect/read/send_timeout 是對 proxy_pass 生效的。

配置舉例:

location ~ \.php$ {
    root                    /home/cdai/test.com;
    include                 fastcgi_params;
    fastcgi_connect_timeout      180;
    fastcgi_read_timeout            600;
    fastcgi_send_timeout            600;
    fastcgi_pass            unix:/dev/shm/php-fcgi.sock;
    fastcgi_index           index.php;
    fastcgi_param          SCRIPT_FILENAME         /home/cdai/test.com$fastcgi_script_name;
}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • 小強性能測試2016年11月6號開課。小強python(接口+selenium+Appium)全棧自動化測試班20...
    測試幫日記閱讀 5,560評論 0 2
  • 在實際的開發和應用中偶發的502,504讓人頭痛,下面轉發一個寫的比較全面的。 PHP-fpm PHP-FPM是一...
    daos閱讀 2,405評論 2 18
  • 一、502和504錯誤提示說明: Nginx 502 Bad Gateway的意思是請求PHP-CGI已經執行,但...
    meng_philip123閱讀 5,295評論 0 3
  • Nginx簡介 解決基于進程模型產生的C10K問題,請求時即使無狀態連接如web服務都無法達到并發響應量級一萬的現...
    魏鎮坪閱讀 2,079評論 0 9
  • 當我們不能買自己喜歡的東西時,我們的內心是貧瘠的,覺得這個世界也是貧瘠的。但當我們選擇買自己喜歡的東西,而不是看價...
    愛所有的一切閱讀 700評論 0 0