PHP socket模拟POST请求实例
我们用到最多的模拟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=111cn.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请求就诞生了,代码如下:
- <?php
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- 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'] : '/';
-
-
- $fp = fsockopen($host, 80, $errno, $errstr, 30);
-
- if ($fp)
- {
-
- $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: closern
- {$data}
- HEADER;
- fwrite($fp, $POST);
- $result = '';
- while(!feof($fp))
- {
-
- $result .= fread($fp, 512);
- }
- }
- else
- {
- return array(
- 'status' => 'error',
- 'error' => "$errstr ($errno)"
- );
- }
-
-
- fclose($fp);
-
-
- $result = explode("rnrn", $result, 2);
-
-
- return array(
- 'status' => 'ok',
- 'header' => isset($result[0]) ? $result[0] : '',
- 'content' => isset($result[1]) ? $result[1] : ''
- );
- }
- }
-
- print_r(socket_post('http://hzhuti.com/', array('name='=>'qiufeng','password'=>md5('www.111cn.net'))));
-
-
实际上,当socket通道打开时,我们传的COOKIE是正确的话,(截图运行的php代码来自上边,运行后返回的网页出现了我的用户名,说明对方网站已经承认我已经登录了)咱就可以干N多事情,比如刷帖,刷回复等,你们懂的,对吧?
好了上面还不够有说服力我们来看一个php socket实现图片上传,这个代码有两点要注意.
一是他是http的post 请求;
二是表单上传协议,
下面的请求串适合任何语言,代码如下:
- <?php
-
- $remote_server = "phpfensi.com";
-
- $boundary = "---------------------".substr(md5(rand(0,32000)),0,10);
-
-
- $header = "POST /api.php?action=twupload HTTP/1.0rn";
- $header .= "Host: {$remote_server}rn";
- $header .= "Content-type: multipart/form-data, boundary=$boundaryrn";
-
-
-
-
-
-
-
-
-
-
- $file_name = "aaa.jpg";
- $content_type = "image/jpg";
-
- $data = '';
-
- $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";
-
- $fp = fsockopen($remote_server, 80);
-
- fputs($fp, $header.$data);
-
-
- while (!feof($fp)) {
- echo fgets($fp, 128);
- }
-
- fclose($fp);