关于curl:curl_exec导致PHP脚本停止执行任何操作

curl_exec causes php script to stop doing anything

当我在一个特定的URL上运行curl时,该站点停止响应,并且不会生成错误,尽管我已经将错误报告设置为on。我试过将curl超时设置为低值,然后它会产生一个错误,所以我知道它不会超时。

我想知道的主要问题是,这怎么可能发生,我怎么知道为什么?

我试图访问的URL是对事实API的调用,我在这里使用的URL

(http://api.factual.com/v2/tables/bi0eJZ/read?api_key=*apikey*&filters={"category":"Automotive","$loc":{"$within":{"$center":[[41,-74],80467.2]}})

当你把它放在浏览器中时工作。如果将纬度和经度本质上更改为任何其他值,PHP脚本将按预期工作。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
error_reporting(E_ALL);
ini_set('display_errors', '2');
$url="http://api.factual.com/v2/tables/bi0eJZ/read?api_key=*apikey*&filters={"category":"Automotive","\$loc":{"\$within":{"\$center":[[41,-74],80467.2]}},"website":{"\$blank":false}}";
Echo"

1"
;

$ch = curl_init($url);
Echo 2;
curl_setopt($ch, CURLOPT_HEADER, 0);
Echo 3;
curl_setopt($ch, CURLOPT_POST, 1);
Echo 4;
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT,15);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_TIMEOUT,30);
Echo 5;
$output = curl_exec($ch) or die("hhtrjrstjsrjt".curl_error($ch));  
Echo 6;
curl_close($ch);
Echo"out:".$output;


看起来PHP配置文件中有一些错误。

要修复错误,必须编辑php.ini文件。

为了显示开发模式中的错误,将error_reporting值更改为E_ALL

然后你必须启用卷曲延伸。要在php.ini中启用它,必须取消对以下行的注释:

1
extension=php_curl.dll

一旦编辑了这些值,不要忘记重新启动Web服务器(apache或nginx)

我也同意我的同事,你应该使用json字符串。

从我的角度来看,代码应该是:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
<?php
ini_set('display_errors', '1');
error_reporting(E_ALL);

$apiKey = '*apikey*';
$filters = '{"category":"Automotive","$loc":{"$within":{"$center":[[41,-74],80467.2]}},"website":{"$blank":false}}';
$params = '?api_key=' . $apiKey . '&filters=' . url_encode($filters);

$url = 'http://api.factual.com/v2/tables/bi0eJZ/read';
$url .= $params;

$ch = curl_init($url);
curl_setopt($ch, CURLOPT_HEADER, 0);
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 15);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_TIMEOUT, 30);
$output = curl_exec($ch) or die("cURL Error" . curl_error($ch));  
curl_close($ch);

echo"out:" . $output;

编辑:

另一种方法是为事实API使用官方的PHP驱动程序:事实API的官方PHP驱动程序

它提供了一个带有curl调试输出和异常调试输出的调试模式。


我以前处理过普通PHP中的curl调用。curl调用是curl类的一部分。正如其他用户所建议的,url_encode($url)函数将为这个特定类准备字符串。如果不运行此函数,则可以传入破坏处理程序的URL(例如,通过使用无效字符或URL语法)。

除此之外,似乎您正试图将JSON对象直接传递到页面的URL中。我不认为在这样的curl调用中使用JSON包是最佳实践。如果这正是您要做的,请参见本页:如何从终端/commandline发布带有curl的JSON数据以测试SpringRest?


您的URL不是url_编码的,因为curl是一个外部应用程序转义是必需的。您的浏览器将自动对url_编码参数,但是您可能正在破坏服务器上的curl,它正在停止。

尝试更改:

1
$url="http://api.factual.com/v2/tables/bi0eJZ/read?api_key=*apikey*&filters={"category":"Automotive","\$loc":{"\$within":{"\$center":[[41,-74],80467.2]}},"website":{"\$blank":false}}";

到:

1
2
3
$url_filters = '{"category":"Automotive","$loc":{"$within":{"$center":[[41,-74],80467.2]}},"website":{"$blank":false}}';

$url="http://api.factual.com/v2/tables/bi0eJZ/read?api_key=*apikey*&filters=".urlencode($url_filters);

但是我有个问题,你的电话对吗?Literal"$loc"的键是正确的吗?

更新以消除对所有单引号不支持变量替换的内容进行反斜杠操作的需要,并允许双引号不转义它们。


在我的例子中,curl失败是由于宿主提供者用liteSpeed替换了apache,liteSpeed在curl请求期间终止了进程。

php.ini设置没有解决这个问题,因为lsphp每次都会在30秒后终止。

我们聊了很久,我确信他们的服务器确实坏了(最终)。

为了让其他可能有类似或相关问题的人更清楚地了解这一点:我的PHP脚本正在运行,并且没有任何错误被记录到PHP中,因为整个PHP进程,包括错误记录器都被Web服务器终止。


为了将来的利益:

  • 将查询参数的urlencode用作筛选参数包含许多对URL不安全/无效的字符

  • 使用curl_getinfo()查看有关http代码和其他有用信息。


  • 此代码仅供测试,请根据我的代码修改您的代码

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    <?php  

    $useragent = 'Mozilla/5.0 (Windows; U; Windows NT 6.1; en-US) AppleWebKit/534.10 (KHTML, like Gecko)
    Chrome/8.0.552.224: Safari/534.10'
    ; // notice this
    $url="http://api.factual.com/v2/tables/bi0eJZ/read?api_key=*apikey*&filters={"category":"Automotive","\$loc":{"\$within":{"\$center":[[41,-74],80467.2]}},"website":{"\$blank":false}}";
    $ch = curl_init(); // notice this

    curl_setopt($ch, CURLOPT_URL, $url); // notice this
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
    curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);

    curl_setopt($ch, CURLOPT_USERAGENT, $useragent);




    $contents = curl_exec($ch);
    echo $contents;
    ?>


    事实上,脚本完全停止执行任何操作,您可以在这里找到它:

    http://php.net/manual/en/function.set-time-limit.php

    摘录:

    set_time_limit()函数和配置指令max_execution_time只影响脚本本身的执行时间。在确定脚本运行的最长时间时,不包括在脚本执行之外发生的任何活动上花费的时间,例如使用System()的系统调用、流操作、数据库查询等。在测量的时间是真实的窗口中,情况并非如此。

    所以基本上,curl会阻塞所有的PHP脚本,基本上唯一实际运行的就是curl,所以如果它永远阻塞,你的站点将不会响应,这就是为什么你需要使用超时的原因…

    至于如何避免它,只需使用超时…