修复Windows 8 apache中的curl_exec挂起问题


fixing curl_exec hangs in Windows 8 apache

我已经研究和试验这个问题一段时间了,还没有找到可行的解决方案,所以我认为是时候寻求帮助了。

curl_exec有问题,但仅在特定的服务器上。这里有一些背景,首先:

  • CPU:Intel Core I7
  • 内存:64GB
  • 操作系统:Windows 8.0
  • 服务器:Apache 2.4.4 x86 TS
  • PHP版本:5.5.1 x86 TS w/xDebug 2.2.3
  • cURL版本:7.30.0

显示问题的PHP代码:

$input_vars = (!empty($_POST)) ? filter_input_array(INPUT_POST) : array();
$url = 'http://192.168.1.100/geekcavecreations/Morti/chatbot/conversation_start.php';
$qs = '?';
foreach ($input_vars as $key => $value)
{
  $qs .= "$key=$value&";
}
$qs= rtrim($qs, '&');
$url .= $qs;
$bot_id = $input_vars['bot_id'];
$options = array(
    CURLOPT_USERAGENT => 'Program O XML API',
    CURLOPT_RETURNTRANSFER => true,
    //CURLOPT_POST => 1,
    CURLOPT_MAXREDIRS => 5,
    CURLOPT_CONNECTTIMEOUT => 5,
    CURLOPT_TIMEOUT => 5,
);
$ch = curl_init($url);
curl_setopt_array($ch, $options);
//curl_setopt($ch, CURLOPT_POSTFIELDS, $input_vars);
$data = curl_exec($ch);
$debug = curl_getinfo($ch);
curl_close($ch);
echo '<pre>Data = ', htmlentities($data), '</pre><br>';
var_dump($debug);

可以看出,我在GET和POST中都尝试过这种方法,并且都给出了相同的结果。上面列出的超时选项在那里,这样脚本就不会无限期地运行。在我停止Apache服务以停止挂起之前,我已经挂起了3个多小时(只是在浏览器中取消是不行的)。脚本的输出如下:

array (size=26)
  'url' => string 'ht tp://192.168.1.100/geekcavecreations/Morti/chatbot/conversation_start.php?say=hello&bot_id=1&convo_id=78a9s39gut34lurq055in5s6r4&format=xml' (length=141)
  'content_type' => null
  'http_code' => int 0
  'header_size' => int 0
  'request_size' => int 203
  'filetime' => int -1
  'ssl_verify_result' => int 0
  'redirect_count' => int 0
  'total_time' => float 5
  'namelookup_time' => float 0
  'connect_time' => float 0
  'pretransfer_time' => float 0
  'size_upload' => float 0
  'size_download' => float 0
  'speed_download' => float 0
  'speed_upload' => float 0
  'download_content_length' => float -1
  'upload_content_length' => float 0
  'starttransfer_time' => float 0
  'redirect_time' => float 0
  'redirect_url' => string '' (length=0)
  'primary_ip' => string '192.168.1.100' (length=13)
  'certinfo' => 
    array (size=0)
      empty
  'primary_port' => int 80
  'local_ip' => string '192.168.1.100' (length=13)
  'local_port' => int 2546
Data = 

(抱歉,找不到更好的格式化方法)

在同一台计算机上还有几个虚拟机,每个虚拟机都有不同的OS/Server/PHP版本,并且都具有完全相同的物理文档根,该根位于主机上。这些机器的范围从Windows 7/IIS到Cents/Apache 2.2,以及其他组合,所有这些机器无一例外地运行相同的脚本,并输出预期的XML文档。如果我只在web浏览器中运行URL,输出如下:

<?xml version="1.0"?>
<program_o>
  <version>2.3.0</version>
  <status><success>1</success></status>
  <bot_id>1</bot_id>
  <bot_name>Morti</bot_name>
  <user_id>1</user_id>
  <user_name>Seeker</user_name>
  <chat>
    <line>
      <input>hello</input>
      <response>And a good failed to you, undefined. How are you?</response>
    </line>
  </chat>
</program_o>

我还获得了上面的XML输出并将其保存到一个文件中,并让有问题的脚本对保存的XML文件的URL执行cURL调用,并且该脚本在这一点上工作没有问题,所以我还创建了一个模型脚本,该脚本只创建一个SimpleXMLElement对象,填充一些新标记,然后echo是所创建对象的asXML()输出(本质上是conversation_start.php所做的,但远没有那么复杂),我也遇到了同样的问题。模型脚本的代码如下:

$xml = new SimpleXMLElement('<program_o></program_o>');
$xml->addChild('version', '2.3.0');
$status = $xml->addChild('status');
$status->addChild('success', '1');
$xml->addChild('bot_id', '1');
$xml->addChild('bot_name', 'Morti');
$xml->addChild('user_id', '1');
$xml->addChild('user_name', 'Seeker');
$chat = $xml->addChild('chat');
$line = $chat->addChild('line');
$line->addChild('input', 'hello');
$line->addChild('response', 'And a good failed to you, undefined. How are you?');
$output = $xml->asXML();
header('Content-type: text/xml');
exit($output);

我已经无计可施了。我已经更改了PHP版本、Apache版本,尝试了无数关于cURL冻结的建议,比如在这里、这里和这里,以及其他几十个问题。

现在我已经写了一本书来阐述我的问题,我不得不问:我如何才能让cURL不挂在Windows8平台上?

好吧,我似乎终于找到了问题的根源。当您对执行调用的脚本所在的同一服务器执行cURL调用时,如果"调用者"answers"被调用者"脚本都试图使用相同的会话id,则会出现死锁,导致两个脚本都等待另一个脚本释放会话。我最后添加了一个测试,看看是否已经有会话id在使用,如果已经有,调用脚本就不会启动会话。如果没有会话id,那么调用者启动会话,获得会话id,然后销毁会话,这允许"被调用者"脚本不受限制地访问所述会话,从而消除死锁情况。下面是我用来做这件事的代码:

$convo_id = (isset ($request_vars['convo_id'])) ? $request_vars['convo_id'] : get_convo_id();
// do stuff here
function get_convo_id()
{
  session_name('Program O XML GUI');
  session_start();
  $convo_id = session_id();
  session_destroy();
  return $convo_id;
}

使用这种方法,一切都能按预期进行。我真诚地希望这对今后的其他人有用。