一次nginx 504 Gateway Time-out錯誤排查、解決記錄
記一次莫名其妙的網(wǎng)站失去響應(yīng)排查。之前網(wǎng)站一直是使用nginx做代理后端的apache運(yùn)行php來提供服務(wù)。apache經(jīng)常會不定期不定時間的出現(xiàn)不能服務(wù)失去響應(yīng),然后nginx出現(xiàn)"504 Gateway Time-out"
查看錯誤日志也看不到任何東西,以為是apache的bug(其實不是,下面會說原因)。
也許年齡大了人就不愛折騰,愿意保持原狀不動,使用監(jiān)控工具,每次收到報警后都重新啟動apache勉強(qiáng)維持著。終于有一天我煩了,不就是處理php嗎,我不用apache總行了吧,一怒之下使用源安裝php-fpm轉(zhuǎn)移到php-fpm來運(yùn)行php。安裝php并不麻煩,使用源安裝還是很順利的,唯一需要做的就是設(shè)置php worker工作進(jìn)程的日志輸出php錯誤日志。
一切準(zhǔn)備就緒后把原來的proxy_pass換成fastcgipass就可以了。
upstream apachephp {
server www.dbjr.com.cn:8080; #Apache1
}
....
proxy_pass http://apachephp;
替換成成
upstream php {
server 127.0.0.1:9000;
}
...
fastcgi_pass php;
就可以把a(bǔ)pache上跑的php遷移到php-fpm上來跑。
原以為這樣就可以高枕無憂了,遷移完成是也確實沒什么問題,但是如果你不去分析問題的根本原因在哪。問題還是會找上門來,第二天nginx又報了504的gateway timeout。這回沒apache什么事了吧,apache總算撇清了關(guān)系。
那應(yīng)該還是在nginx和php-fpm身上,查看nginx的錯誤日志,可以看到
[error] 6695#0: *168438 upstream timed out (110: Connection timed out) while reading response header from upstream,
...
request: "GET /kd/open.php?company=chinapost&number=PA24977020344 HTTP/1.1", upstream: "fastcgi://127.0.0.1:9000", host: "www.dbjr.com.cn"
看到這里基本上就排除了nginx嫌疑,nginx是在等待php處理"GET /kd/open.php?company=chinapost&number=PA24977020344 HTTP/1.1"超時退出了。
馬上重啟php-fpm,問題沒有了,網(wǎng)站可以訪問了。
再次訪問該頁面,依然沒有響應(yīng),但同時訪問別的頁面正常,該頁面刷新幾次后,整個網(wǎng)站都是bad gateway timeout了。
問題就縮小到這個php腳本上了。
netstat -napo |grep "php5-fpm" | wc -l
查看php工作進(jìn)程已經(jīng)達(dá)到了配置文件里的上限10,有種感覺就是大家都被open.php這個腳本卡住了。
這個腳本是干什么的呢?這個腳本就是采集快遞信息的,里面用到了php_curl。
PHP腳本如果執(zhí)行時間超過php.ini中的配置項max_execution_time不出結(jié)果就會強(qiáng)制退出。
查看了php.ini中max_execution_time確實配了,值為30。
萬能google派上用場了,經(jīng)過不斷google后得到下面這句話
set_time_limit()函數(shù)和配置指令max_execution_time只影響腳本本身執(zhí)行的時間。任何發(fā)生在諸如使用system()的系統(tǒng)調(diào)用,流操作,數(shù)據(jù)庫操作等的腳本執(zhí)行的最大時間不包括其中,當(dāng)該腳本已運(yùn)行。
就是說如果腳本中執(zhí)行了其它操作的時間是不計在腳本運(yùn)行時間當(dāng)中的,如果你沒設(shè)置超時,那么php就會一直等待調(diào)用的結(jié)果。
查看open.php源文件一看,果然沒有設(shè)置curl的超時時間。
增加如下兩行,重新刷新,后問題解決了。
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10); //timeout on connect
curl_setopt($ch, CURLOPT_TIMEOUT, 10); //timeout on response
當(dāng)然,除了這種方法外,php-fpm里也提供參數(shù)供我們強(qiáng)制殺死長時間無結(jié)果的進(jìn)程,只是該參數(shù)默認(rèn)沒打開。
php-fpm的配置文件里可以設(shè)置一個參數(shù)request_terminate_timeout,請求終止的超時時間,當(dāng)請求執(zhí)行超過這個時間就會被kill。
同時它還有個參數(shù)request_slowlog_timeout,用來記錄慢請求日志的。
命令行運(yùn)行php的話,可以使用這段代碼
$real_execution_time_limit = 60; //時間限制
if (pcntl_fork())
{
// some long time code which should be
// terminated after $real_execution_time_limit seconds passed if it's not
// finished by that time
}
else
{
sleep($real_execution_time_limit);
posix_kill(posix_getppid(), SIGKILL);
}
相關(guān)文章
nginx 負(fù)載均衡的三種參數(shù)設(shè)置
這篇文章主要介紹了nginx 負(fù)載均衡的三種參數(shù)設(shè)置,需要的朋友可以參考下2017-07-07配置Nginx出現(xiàn)403(Forbidden)靜態(tài)文件加載不出來的解決方法
本文主要介紹了配置Nginx出現(xiàn)403(Forbidden)靜態(tài)文件加載不出來的解決方法,文中通過圖文介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-12-12nginx配置ssl實現(xiàn)https訪問的步驟(適合新手)
這篇文章主要給大家介紹了關(guān)于nginx配置ssl實現(xiàn)https訪問的相關(guān)資料,這個教程非常適合新手小白,文中通過示例代碼將實現(xiàn)的方法一步步介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來一起看看吧2018-12-12Nginx對網(wǎng)段內(nèi)ip的連接數(shù)限流配置詳解
這篇文章主要介紹了Nginx對網(wǎng)段內(nèi)ip的連接數(shù)限流配置詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03使用?nginx?搭建代理服務(wù)器(正向代理?https?網(wǎng)站)的詳細(xì)步驟
這篇文章主要介紹了使用?nginx?搭建代理服務(wù)器(正向代理?https?網(wǎng)站)指南的相關(guān)操作,本文給大家介紹的非常詳細(xì),感興趣的朋友跟隨小編一起看看吧2024-08-08