fsocket模拟GET/POST提交

//fsocket模拟post提交
$url = "http://localhost/test2.php?site=nowamagic.net";
print_r(parse_url($url));

//echo $query;
sock_get($url,"user=gonn");
//sock_get($url, $query);

//fsocket模拟get提交
function sock_get($url, $query)
{
	$data = array(
	'foo'=>'bar', 
	'baz'=>'boom', 
	'site'=>'www.nowamagic.net', 
	'name'=>'nowa magic'); 
	
	$query_str = http_build_query($data);
	
    $info = parse_url($url);
    $fp = fsockopen($info["host"], 80, $errno, $errstr, 3);
    //$head = "GET ".$info['path']."?".$info["query"]." HTTP/1.0\r\n";
	$head = "GET ".$info['path']."?".$query_str." HTTP/1.0\r\n";
    $head .= "Host: ".$info['host']."\r\n";
    $head .= "\r\n";
    $write = fputs($fp, $head);
    while (!feof($fp))
    {
        $line = fread($fp,4096);
        echo $line;
    }
}

sock_post($url,"user=gonn");

function sock_post($url, $query)
{	
    $info = parse_url($url);
    $fp = fsockopen($info["host"], 80, $errno, $errstr, 3);
    $head = "POST ".$info['path']."?".$info["query"]." HTTP/1.0\r\n";
    $head .= "Host: ".$info['host']."\r\n";
    $head .= "Referer: http://".$info['host'].$info['path']."\r\n";
    $head .= "Content-type: application/x-www-form-urlencoded\r\n";
    $head .= "Content-Length: ".strlen(trim($query))."\r\n";
    $head .= "\r\n";
    $head .= trim($query);
    $write = fputs($fp, $head);
    while (!feof($fp))
    {
        $line = fread($fp,4096);
        echo $line;
    }
}
接收页面 test2.php 的代码为:

$data = $_REQUEST;

echo '<pre>';
print_r( $data );
echo '</pre>';

在这段代码里面,调用后忽略执行结果直接返回,可用于php异步执行。

在nginx服务器上有一个比较诡异的情况就是有时候无法调用异步的脚本。

查阅相关资料后,是nginx 499 的问题。

其中解决方案有以下,经过对每一个方案的验证最终得出结果:

1、NGINX 499

查看 NGINX access log,发现这样的请求会以 499(Client Closed Request)记录。确定问题是因为:客户端主动端口请求连接时,NGINX 不会将该请求代理给上游服务(FastCGI PHP 进程),这个时候 access log 中会以 499 记录这个请求。

要解决这个问题需要将 NGINX FastCGI 忽略客户端中断配置打开:

fastcgi_ignore_client_abort on;

这样无论客户端是否断开,都会将这个请求代理给上游,并且会记录上游服务处理后的返回状态。

2、NGINX 线程原因

将nginx的worker_processes 由之前的auto修改为2(我的是单核服务器)

3、NGINX 499

nginx对499的定义是”client has closed connection”,并且在这些情况下会返回这个状态码:

upstream 在收到读写事件处理之前时发现连接不可用。

server处理请求未结束,而client提前关闭了连接。

upstream出错,执行next_upstream时发现连接不可用。

一个不安全的做法是在fclose之前,让当前的进程先睡眠一段时间;我这里设置为10毫秒,这10毫秒的延迟对我完成整个请求的影响不大,同时我也认为nginx一定能在10毫米内把请求转到fastcgi去执行。这个时间间隔很难把握,不能保证php一定有执行到。

这种方式并不是真正的异步,只是很取巧的强制关闭连接而不等待服务器端响应。所以在Laruence的那2篇文章中,有2个问题:

PHP使用fsock不能叫做异步,只是伪异步。

fwrite之后马上执行fclose,nginx会直接返回499

由于我的代码上面usleep为1000,初步估计是时间不够导致没发出去就close了,所以调整为20000。并进行最后测试。

结果晚点更新。

更新结果:测试2天结果显示正常,的确是usleep数值过小的问题。

发表评论

电子邮件地址不会被公开。

This site uses Akismet to reduce spam. Learn how your comment data is processed.