curl_errno возвращает 0 вместо 6
$data_struct = array();
$data_struct[]['opts'] = array( CURLOPT_URL => 'http://www.yahoo.com/', CURLOPT_RETURNTRANSFER => true);
$data_struct[]['opts'] = array( CURLOPT_URL => 'http://www.google.com/', CURLOPT_RETURNTRANSFER => true);
$data_struct[]['opts'] = array( CURLOPT_URL => 'http://404.php.net/', CURLOPT_RETURNTRANSFER => true);
//create the multiple cURL handle
$mh = curl_multi_init();
// create and add handles to data structure
foreach ($data_struct as $i => $data){
$data_struct[$i]['handle'] = curl_init();
curl_setopt_array($data_struct[$i]['handle'], $data_struct[$i]['opts']);
curl_multi_add_handle($mh, $data_struct[$i]['handle']);
}
$active = null;
//execute the handles
do {
$mrc = curl_multi_exec($mh, $active);
} while ($mrc == CURLM_CALL_MULTI_PERFORM);
while ($active && $mrc == CURLM_OK) {
if (curl_multi_select($mh) != -1) {
do {
$mrc = curl_multi_exec($mh, $active);
} while ($mrc == CURLM_CALL_MULTI_PERFORM);
}
}
foreach ($data_struct as $i => $data){
$row = array();
$row['httpcode'] = curl_getinfo($data_struct[$i]['handle'] , CURLINFO_HTTP_CODE);
$row['url'] = $data['opts'][CURLOPT_URL];
$row['error'] = curl_error($data_struct[$i]['handle']);
$row['errorno'] = curl_errno($data_struct[$i]['handle']);
print_r($row);
echo "\n";
curl_multi_remove_handle($mh, $data_struct[$i]['handle']);
curl_close($data_struct[$i]['handle']);
}
выход
Array
(
[httpcode] => 302
[url] => http://www.yahoo.com/
[error] =>
[errorno] => 0
)
Array
(
[httpcode] => 302
[url] => http://www.google.com/
[error] =>
[errorno] => 0
)
Array
(
[httpcode] => 0
[url] => http://404.php.net/
[error] => Couldn't resolve host '404.php.net'
[errorno] => 0
)
В той же ситуации multi curl возвращает 0 с функцией curl_errno, несмотря на то, что curl_error возвращает сообщение об ошибке.
2 ответа
Решение
При условии, что у вас PHP 5.2.x или выше (новее лучше, здесь), я бы предложил использовать curl_multi_info_read()
, в этом случае. (См. Документацию curl_multi_info_read().)
Вы можете увидеть вывод curl_multi_info_read()
поместив его ближе к концу вашего кода (после вашего echo "\n";
строка) вот так:
var_dump(curl_multi_info_read($mh));
Это, кажется, дает правильный errno (в частности, "result" => 6
):
Array
(
[httpcode] => 0
[url] => http://404.php.net/
[error] => Couldn't resolve host '404.php.net'
[errorno] => 0
)
array(3) {
["msg"]=>
int(1)
["result"]=>
int(6)
["handle"]=>
resource(7) of type (curl)
}
Примечание: проверено на установке PHP 5.2.8.
Коды Массив:
$codes_texts = [
0 => 'OK',
1 => 'UNSUPPORTED_PROTOCOL',
2 => 'FAILED_INIT',
3 => 'URL_MALFORMAT',
4 => 'URL_MALFORMAT_USER',
5 => 'COULDNT_RESOLVE_PROXY',
6 => 'COULDNT_RESOLVE_HOST',
7 => 'COULDNT_CONNECT',
8 => 'FTP_WEIRD_SERVER_REPLY',
9 => 'FTP_ACCESS_DENIED',
10 => 'FTP_USER_PASSWORD_INCORRECT',
11 => 'FTP_WEIRD_PASS_REPLY',
12 => 'FTP_WEIRD_USER_REPLY',
13 => 'FTP_WEIRD_PASV_REPLY',
14 => 'FTP_WEIRD_227_FORMAT',
15 => 'FTP_CANT_GET_HOST',
16 => 'FTP_CANT_RECONNECT',
17 => 'FTP_COULDNT_SET_BINARY',
18 => 'PARTIAL_FILE',
19 => 'FTP_COULDNT_RETR_FILE',
20 => 'FTP_WRITE_ERROR',
21 => 'FTP_QUOTE_ERROR',
22 => 'HTTP_NOT_FOUND',
23 => 'WRITE_ERROR',
24 => 'MALFORMAT_USER',
25 => 'FTP_COULDNT_STOR_FILE',
26 => 'READ_ERROR',
27 => 'OUT_OF_MEMORY',
28 => 'OPERATION_TIMEOUTED',
29 => 'FTP_COULDNT_SET_ASCII',
30 => 'FTP_PORT_FAILED',
31 => 'FTP_COULDNT_USE_REST',
32 => 'FTP_COULDNT_GET_SIZE',
33 => 'HTTP_RANGE_ERROR',
34 => 'HTTP_POST_ERROR',
35 => 'SSL_CONNECT_ERROR',
36 => 'FTP_BAD_DOWNLOAD_RESUME',
37 => 'FILE_COULDNT_READ_FILE',
38 => 'LDAP_CANNOT_BIND',
39 => 'LDAP_SEARCH_FAILED',
40 => 'LIBRARY_NOT_FOUND',
41 => 'FUNCTION_NOT_FOUND',
42 => 'ABORTED_BY_CALLBACK',
43 => 'BAD_FUNCTION_ARGUMENT',
44 => 'BAD_CALLING_ORDER',
45 => 'HTTP_PORT_FAILED',
46 => 'BAD_PASSWORD_ENTERED',
47 => 'TOO_MANY_REDIRECTS',
48 => 'UNKNOWN_TELNET_OPTION',
49 => 'TELNET_OPTION_SYNTAX',
50 => 'OBSOLETE',
51 => 'SSL_PEER_CERTIFICATE',
52 => 'GOT_NOTHING',
53 => 'SSL_ENGINE_NOTFOUND',
54 => 'SSL_ENGINE_SETFAILED',
55 => 'SEND_ERROR',
56 => 'RECV_ERROR',
57 => 'SHARE_IN_USE',
58 => 'SSL_CERTPROBLEM',
59 => 'SSL_CIPHER',
60 => 'SSL_CACERT',
61 => 'BAD_CONTENT_ENCODING',
62 => 'LDAP_INVALID_URL',
63 => 'FILESIZE_EXCEEDED',
64 => 'FTP_SSL_FAILED',
];
Авторсценария:
$urls = array(
"http://php.net/",
"http://404.php.net/",
"httpnot://php.net/",
);
$mh = curl_multi_init();
$codes = [];
$result = [];
$connects = [];
foreach ($urls as $i => $url) {
$connects[$i] = curl_init($url);
curl_setopt($connects[$i], CURLOPT_RETURNTRANSFER, 1);
curl_multi_add_handle($mh, $connects[$i]);
}
do {
$status = curl_multi_exec($mh, $active);
$info = curl_multi_info_read($mh, $count);
if ($info !== false) {
$codes[(int)$info['handle']] = (int)$info['result'];
}
}
while ($status === CURLM_CALL_MULTI_PERFORM || $active || $count);
foreach ($connects as $i => $connect) {
$result[$i] = [
'url' => $urls[$i],
'code' => $codes[(int)$connect], // 0 - 64 = curl_errno
'code_text' => $codes_texts[$codes[(int)$connect]],
'body_len' => strlen(curl_multi_getcontent($connect)),
];
curl_close($connect);
}
curl_multi_close($mh);
echo "<pre>" . var_export($result, 1) . "</pre>";
Результат:
array (
0 =>
array (
'url' => 'http://php.net/',
'code' => 0,
'code_text' => 'OK',
'body_len' => 34462,
),
1 =>
array (
'url' => 'http://404.php.net/',
'code' => 6,
'code_text' => 'COULDNT_RESOLVE_HOST',
'body_len' => 0,
),
2 =>
array (
'url' => 'httpnot://php.net/',
'code' => 1,
'code_text' => 'UNSUPPORTED_PROTOCOL',
'body_len' => 0,
),
)