在我們的業(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ù)。