ホームページ >php教程 >php手册 >PHP socket模拟POST请求实例

PHP socket模拟POST请求实例

WBOY
WBOYオリジナル
2016-05-25 16:49:491188ブラウズ

我们用到最多的模拟POST请求几乎都是使用php curl来实现了,没考虑到PHP socket也可以实现,今天看到朋友写了一文章,下面我来给大家分享一下PHP socket模拟POST请求实例.

以前模拟post请求俺都用PHP curl扩展实现来着,没想过PHP socket也可以实现,最近翻了下相关资料才发现原来没有那么高深,只是以前一直没有完全理解post的原理和本质而已,其实就是发送给目的程序一个标志为post的协议串如下:

POST /目的程序url HTTP/1.1 

Accept: 接收信息格式 

Referer: url来路 

Accept-Language: 接收语言 

Content-Type: application/x-www-form-urlencoded 

Cookie: 网站cookie,不用俺过多解释,对吧? 

User-Agent: 用户代理,操作系统及版本、CPU 类型、浏览器及版本等信息 

Host: 要发送到的主机地址 

Content-Length: 发送数据的长度 

Pragma: 本地是否存在缓存 

Cache-Control: 是否需要网页缓存 

Connection: 连接状态 

username=fengdingbo&password=phprm.com   //post发送的数据 

我想大家对表单的post方法提交数据应该是最熟悉不过了,例如我们想把用户名和密码发送给某个页面的时候,填写好相应的input框,点击提交按钮,最后把这个表单发送到action程序的就是以上数据,知道了这一点我想就不难了

这时候我们只需要用php的socket打开一个端口,例如80端口,把以上信息利用这个端口发送给目的程序就行了。

我们如何在一个端口上建立一个socket通道呢?在PHP中是如此简单呢!官方给的原型:

resource fsockopen ( string $hostname [, int $port = -1 [, int &$errno [, string &$errstr [, float $timeout = ini_get("default_socket_timeout") ]]]] ) 

下边是人类的理解:

fsockopen(主机名称,端口号,错误号的&变量,错误提示的&变量,超时时间)

主机名称就是你需要发送数据的目的地;

端口号就是这个目的程序会在哪个端口等着你的数据;错误号的&变量,这个是如果建立socket不成功的时候返回的错误编号;

错误提示的&变量,是错误的时候返回的错误提示信息;

超时时间,就是post数据之后如果对方没有回应信息,等待的最长时间。

如果不出意外(你正确的设置fsockopen()函数的参数)的话,一个socket通道现在已经打开了,我们下一步需要做的就是,通过这个打开的通道把post请求协议发给目的程序,这时候可以使用fwrite或者fputs函数中的任意一个,把post的请求格式发给fsockopen()打开的资源句柄,这时候一个伟大的socket模拟的post请求就诞生了,代码如下:

<?php 
/** 
 * SOCKET扩展函数 
 * @copyright (c) 2013 
 * @author Qiufeng <fengdingbo@gmail.com> 
 * @link http://www.phprm.com 
 * @version 1.0 
 */ 
 
/** 
 * Post Request 
 * 
 * @param string http://pic1.phprm.com/2014/01/18/$url.jpg  
 * @param array $data 
 * @param string $referer 
 * @return array 
 */ 
if ( ! function_exists(&#39;socket_post&#39;)) 
{ 
 function socket_post($url, $data, $referer=&#39;&#39;) 
 { 
  if( ! is_array($data)) 
  { 
   return; 
  } 
 
  $data = http_build_query($data); 
  $url = parse_url($url); 
 
  if ( ! isset($url[&#39;scheme&#39;]) || $url[&#39;scheme&#39;] != &#39;http&#39;) 
  { 
   die(&#39;Error: Only HTTP request are supported !&#39;); 
  } 
 
  $host = $url[&#39;host&#39;]; 
  $path = isset($url[&#39;path&#39;]) ? $url[&#39;path&#39;] : &#39;/&#39;; 
 
  // open a socket connection on port 80 - timeout: 30 sec 
  $fp = fsockopen($host, 80, $errno, $errstr, 30); 
 
  if ($fp) 
  { 
   // send the request headers: 
   $length = strlen($data); 
   $POST = <<<HEADER 
POST {$path} HTTP/1.1 
Accept: text/plain, text/html 
Referer: {$referer} 
Accept-Language: zh-CN,zh;q=0.8 
Content-Type: application/x-www-form-urlencodem  
Cookie: token=value; pub_cookietime=2592000; pub_sauth1=value; pub_sauth2=value 
User-Agent: Mozilla/5.0 (X11; Linux i686) AppleWebKit/537.17 (KHTML, like Gecko) Chrome/24.0.1312.56 Safari/537.17 
Host: {$host} 
Content-Length: {$length} 
Pragma: no-cache 
Cache-Control: no-cache 
Connection: close\r\n 
{$data} 
HEADER; 
   fwrite($fp, $POST); 
   $result = &#39;&#39;; 
   while(!feof($fp)) 
   { 
// receive the results of the request 
$result .= fread($fp, 512); 
   } 
  } 
  else 
  { 
   return array( 
 &#39;status&#39; => &#39;error&#39;, 
 &#39;error&#39; => "$errstr ($errno)" 
 ); 
  } 
 
  // close the socket connection: 
  fclose($fp); 
 
  // split the result header from the content 
  $result = explode("\r\n\r\n", $result, 2); 
 
  // return as structured array: 
  return array( 
&#39;status&#39; => &#39;ok&#39;, 
&#39;header&#39; => isset($result[0]) ? $result[0] : &#39;&#39;, 
&#39;content&#39; => isset($result[1]) ? $result[1] : &#39;&#39; 
); 
 } 
} 
 
print_r(socket_post(&#39;http://hzhuti.com/&#39;, array(&#39;name=&#39;=>&#39;qiufeng&#39;,&#39;password&#39;=>md5(&#39;www.phprm.com&#39;)))); 
/* e.g: socket_post(&#39;http://www.phprm.com&#39;, array(&#39;name=&#39;=>&#39;qiufeng&#39;,&#39;password&#39;=>md5(&#39;phprm.com&#39;))); */ 
/* End of file socket_helper.php */

实际上,当socket通道打开时,我们传的COOKIE是正确的话,(截图运行的php代码来自上边,运行后返回的网页出现了我的用户名,说明对方网站已经承认我已经登录了)咱就可以干N多事情,比如刷帖,刷回复等,你们懂的,对吧?

好了上面还不够有说服力我们来看一个php socket实现图片上传,这个代码有两点要注意.

一是他是http的post 请求;

二是表单上传协议,

下面的请求串适合任何语言,代码如下:

<?php  
 
$remote_server = "phprm.com";  
   
$boundary = "---------------------".substr(md5(rand(0,32000)),0,10);  
  
// Build the header  
$header = "POST /api.php?action=twupload HTTP/1.0\r\n";  
$header .= "Host: {$remote_server}\r\n";  
$header .= "Content-type: multipart/form-data, boundary=$boundaryrn";  
   
/*  
// attach post vars  
foreach($_POST AS $index => $value){  
$data .="--$boundaryrn";  
$data .= "Content-Disposition: form-data; name=\"".$index."\"\r\n";  
$data .= "\r\n".$value."\r\n";  
$data .="--$boundary\r\n";  
}  
*/ 
$file_name = "aaa.jpg";  
$content_type = "image/jpg";  
   
$data = &#39;&#39;;  
// and attach the file  
$data .= "--$boundary\r\n";  
   
$content_file = file_get_contents(&#39;aaa.jpg&#39;);  
$data .="Content-Disposition: form-data; name="userfile"; filename="$file_name"\r\n";  
$data .= "Content-Type: $content_typern\r\n";  
$data .= "".$content_file."\r\n";  
$data .="--$boundary--\r\n";  
   
   $header .= "Content-length: " . strlen($data) . "\r\n\r\n";  
 // Open the connection  
$fp = fsockopen($remote_server, 80);  
// then just  
fputs($fp, $header.$data);  
// reader  
   
while (!feof($fp)) {  
echo fgets($fp, 128);  
}  
   
fclose($fp);


本文地址:

转载随意,但请附上文章地址:-)

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。