背景:項目中需要在抓取紛享銷客CRM圖片上傳到OSS,調用OssClient.php時,容易發生解析超時(多重試幾次就ok)。
專注于為中小企業提供成都網站建設、成都網站制作服務,電腦端+手機端+微信端的三站合一,更高效的管理,為中小企業羅湖免費做網站提供優質的服務。我們立足成都,凝聚了一批互聯網行業人才,有力地推動了千余家企業的穩健成長,幫助中小企業通過網站建設實現規模擴充和轉變。
錯誤提示:
[2019-04-08 19:41:01] lumen.DEBUG: 出錯文件:/home/zrj/www/admin/yundou-admin/vendor/aliyuncs/oss-sdk-php/src/OSS/OssClient.php
[2019-04-08 19:41:01] lumen.DEBUG: 出錯編碼:0
[2019-04-08 19:41:01] lumen.DEBUG: 出錯行號:2187
[2019-04-08 19:41:01] lumen.DEBUG: 出錯信息:RequestCoreException: cURL resource: Resource id #371; cURL error: Resolving timed out after 10521 milliseconds (28)
Resolving timed out after 10521 milliseconds (28)
解析超時
源碼分析:
try {
$ossClient = new OssClient(self::$accessKeyId, self::$accessKeySecret, self::$endpoint);
$ossClient->uploadFile(self::$bucket, $ossFileName, $localhostFileName);//上傳文件
$ossClient->putBucketAcl(self::$bucket, OssClient::OSS_ACL_TYPE_PUBLIC_READ);
} catch (OssException $e) {
self::debugException($e);
throw new \Exception("上傳oss失敗:" . $e->getMessage(), $e->getCode());
}
1.實例化OssClient客戶端后,調用uploadFile方法。
2.uploadFile調用了auth,驗證并且執行請求,按照OSS Api協議,執行操作。
/**
* 上傳本地文件
*
* @param string $bucket bucket名稱
* @param string $object object名稱
* @param string $file 本地文件路徑
* @param array $options
* @return null
* @throws OssException
*/
public function uploadFile($bucket, $object, $file, $options = NULL)
{
......
$response = $this->auth($options);
$result = new PutSetDeleteResult($response);
return $result->getData();
}
3.auth中調用RequestCore類創建請求
/**
* 驗證并且執行請求,按照OSS Api協議,執行操作
*
* @param array $options
* @return ResponseCore
* @throws OssException
* @throws RequestCore_Exception
*/
private function auth($options)
{
......
//創建請求
$request = new RequestCore($this->requestUrl, $this->requestProxy);
$request->set_useragent($this->generateUserAgent());
......
try {
$request->send_request();
} catch (RequestCore_Exception $e) {
throw(new OssException('RequestCoreException: ' . $e->getMessage()));
}
}
4.OSS\Http\RequestCore類中send_request方法通過CURL發送請求(調用了prep_request準備請求方法)
/**
* Sends the request, calling necessary utility functions to update built-in properties.
*
* @param boolean $parse (Optional) Whether to parse the response with ResponseCore or not.
* @return string The resulting unparsed data from the request.
*/
public function send_request($parse = false)
{
set_time_limit(0);
$curl_handle = $this->prep_request();
$this->response = curl_exec($curl_handle);
if ($this->response === false) {
throw new RequestCore_Exception('cURL resource: ' . (string)$curl_handle . '; cURL error: ' . curl_error($curl_handle) . ' (' . curl_errno($curl_handle) . ')');
}
$parsed_response = $this->process_response($curl_handle, $this->response);
curl_close($curl_handle);
if ($parse) {
return $parsed_response;
}
return $this->response;
}
5.最終可以推導出問題出在RequestCore中的CURL。
解決方案:
curl有一個CURLOPT_IPRESOLVE選項,作用是指定使用那種IP協議(IPV4/IPV6)。
CURLOPT_IPRESOLVE - specify which IP protocol version to use
選項:
CURL_IPRESOLVE_WHATEVER
Default, resolves addresses to all IP versions that your system allows.
CURL_IPRESOLVE_V4
Resolve to IPv4 addresses.
CURL_IPRESOLVE_V6
Resolve to IPv6 addresses.
這里要注意:默認值(curl沒有主動設置CURLOPT_IPRESOLVE選項時,則系統會使用默認值)。
Default, resolves addresses to all IP versions that your system allows.
含義為:默認使用服務器系統允許的所有IP版本來解析地址。
而服務器系統可能存在同時支持IPV4/IPV6,CURL會挨個去試。當本地網絡不支持其中任何一種IP版本協議時,會出現超時。
所以,最終的解決方案為顯式指定CURL的CURLOPT_IPRESOLVE選項。
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
參考官方CURLOPT_IPRESOLVE
網站欄目:CURL解析超時的解決方案
轉載來于:http://m.newbst.com/article0/gejgio.html
成都網站建設公司_創新互聯,為您提供移動網站建設、面包屑導航、電子商務、Google、全網營銷推廣、關鍵詞優化
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯