PHP socket模拟POST请求实例

作者:袖梨 2022-06-24

以前模拟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=111com.net   //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请求就诞生了。

 代码如下 复制代码

/**
 * SOCKET扩展函数
 * @copyright (c) 2013
 * @author Qiufeng
 * @link http://www.111com.net
 * @version 1.0
 */
 
/**
 * Post Request
 *
 * @param string $url 
 * @param array $data
 * @param string $referer
 * @return array
 */
if ( ! function_exists('socket_post'))
{
 function socket_post($url, $data, $referer='')
 {
  if( ! is_array($data))
  {
   return;
  }
 
  $data = http_build_query($data);
  $url = parse_url($url);
 
  if ( ! isset($url['scheme']) || $url['scheme'] != 'http')
  {
   die('Error: Only HTTP request are supported !');
  }
 
  $host = $url['host'];
  $path = isset($url['path']) ? $url['path'] : '/';
 
  // 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 = << 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: closern
{$data}
HEADER;
   fwrite($fp, $POST);
   $result = '';
   while(!feof($fp))
   {
    // receive the results of the request
    $result .= fread($fp, 512);
   }
  }
  else
  {
   return array(
     'status' => 'error',
     'error' => "$errstr ($errno)"
     );
  }
 
  // close the socket connection:
  fclose($fp);
 
  // split the result header from the content
  $result = explode("rnrn", $result, 2);
 
  // return as structured array:
  return array(
    'status' => 'ok',
    'header' => isset($result[0]) ? $result[0] : '',
    'content' => isset($result[1]) ? $result[1] : ''
    );
 }
}
 
print_r(socket_post('http://www.111com.net/', array('name='=>'qiufeng','password'=>md5('www.111com.net'))));
/* e.g: socket_post('http://www.111com.net', array('name='=>'qiufeng','password'=>md5('111com.net'))); */
/* End of file socket_helper.php */

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

好了上面还不够有说服力我们来看一个php socket实现图片上传

这个代码有两点要注意

一是他是http的post 请求;

二是表单上传协议,

下的请求 串适合任何语言.

 代码如下 复制代码

   
        $remote_server = "111com.net";
 
        $boundary = "---------------------".substr(md5(rand(0,32000)),0,10);
        
        // Build the header
        $header = "POST /api.php?action=twupload HTTP/1.0rn";
        $header .= "Host: {$remote_server}rn";
        $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.""rn";
            $data .= "rn".$value."rn";
            $data .="--$boundaryrn";
        }
        */
        $file_name = "aaa.jpg";
        $content_type = "image/jpg";
 
        $data = '';
        // and attach the file
        $data .= "--$boundaryrn";
 
        $content_file = file_get_contents('aaa.jpg');
        $data .="Content-Disposition: form-data; name="userfile"; filename="$file_name"rn";
        $data .= "Content-Type: $content_typernrn";
        $data .= "".$content_file."rn";
        $data .="--$boundary--rn";
 
       $header .= "Content-length: " . strlen($data) . "rnrn";
                 // Open the connection
 
 
        $fp = fsockopen($remote_server, 80);
        // then just
        fputs($fp, $header.$data);
        // reader
 
    while (!feof($fp)) {
        echo fgets($fp, 128);
    }
 
fclose($fp);

相关文章

精彩推荐