Heim >php教程 >php手册 >PHP socket模拟POST请求实例

PHP socket模拟POST请求实例

WBOY
WBOYOriginal
2016-05-25 16:49:491197Durchsuche

我们用到最多的模拟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);


本文地址:

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

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn