php中curl未設(shè)置超時(shí)導(dǎo)致的睡眠進(jìn)程問題

在我們的業(yè)務(wù)中,需要定時(shí)任務(wù)去執(zhí)行php腳本,對(duì)蘋果自動(dòng)續(xù)費(fèi)用戶進(jìn)行續(xù)約。某次發(fā)現(xiàn)有大量的定時(shí)任務(wù)在執(zhí)行,有些開始時(shí)間是半年前,所有進(jìn)程都處于睡眠中。

root     31625  0.0  0.0 108124  1304 ?        Ss   Mar22   0:00 /bin/sh -c /usr/local/php7/bin/php -r 'Crontab::debug("/data/erge_api", "Cron_Apple_Pay_Subscribe");' >> /tmp/cronapplepaysubscribe.log 2>&1
root     31634  0.0  0.0 233520 10064 ?        S    Mar22   0:49 /usr/local/php7/bin/php -r Crontab::debug("/data/erge_api", "Cron_Apple_Pay_Subscribe");

通過lsof命令可以發(fā)現(xiàn),該進(jìn)程打開了兩個(gè)socket,一個(gè)是連接數(shù)據(jù)庫的,一個(gè)是請(qǐng)求蘋果服務(wù)器的。

php     31634 root    3u  IPv4 3973134538      0t0        TCP localhost:63880->localhost:27017 (ESTABLISHED)
php     31634 root    4u  IPv4 3973134589      0t0        TCP 192.168.10.100:17462->17.154.66.159:https (ESTABLISHED)

再通過strace命令,可以發(fā)現(xiàn)進(jìn)程一直在進(jìn)行如下調(diào)用,而且文件描述符就是4,也就是說,進(jìn)程一直輪詢地獲取從蘋果服務(wù)器那邊的數(shù)據(jù)。

restart_syscall(<... resuming interrupted call ...>) = 0
poll([{fd=4, events=POLLIN|POLLPRI|POLLRDNORM|POLLRDBAND}], 1, 0) = 0 (Timeout)
clock_gettime(CLOCK_MONOTONIC, {41212631, 418739828}) = 0
clock_gettime(CLOCK_MONOTONIC, {41212631, 418821856}) = 0
clock_gettime(CLOCK_MONOTONIC, {41212631, 418887370}) = 0

為了確認(rèn)這個(gè)現(xiàn)象,可以在應(yīng)用程序中加上日志。Logger打印日志的時(shí)候還會(huì)加上進(jìn)程ID。

        Logger::crontab($value, 'start', __CLASS__);
        $ch = curl_init($endpoint);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
        curl_setopt($ch, CURLOPT_POST, true);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $postData);
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, 0);
        curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 0);   

        $response = curl_exec($ch);
        $errno = curl_errno($ch);
        $errmsg = curl_error($ch);
        curl_close($ch);
        Logger::crontab($response, "applepay|notify|receiptdata", __CLASS__);

通過日志發(fā)現(xiàn),只有start,沒有notify的記錄,那么問題基本可以確定出現(xiàn)在curl上。curl做了一次http請(qǐng)求,那么有請(qǐng)求就有超時(shí)時(shí)間,查看curl_setopt選項(xiàng),發(fā)現(xiàn)提供了兩個(gè)請(qǐng)求超時(shí)選項(xiàng)的設(shè)置。如果設(shè)置為0,則無限等待,默認(rèn)值沒說。兩個(gè)選項(xiàng)的優(yōu)先級(jí)也沒說。不過這里可以通過設(shè)置CURLOPT_TIMEOUT來限制curl最長執(zhí)行時(shí)間,最后解決問題。

CURLOPT_CONNECTTIMEOUT  在嘗試連接時(shí)等待的秒數(shù)。設(shè)置為0,則無限等待
CURLOPT_TIMEOUT 允許 cURL 函數(shù)執(zhí)行的最長秒數(shù)。
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容