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请求就诞生了,代码如下:
* @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('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 = 'error', 'error' => "$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( '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.phprm.com')))); /* e.g: socket_post('http://www.phprm.com', array('name='=>'qiufeng','password'=>md5('phprm.com'))); */ /* End of file socket_helper.php */
实际上,当socket通道打开时,我们传的COOKIE是正确的话,(截图运行的php代码来自上边,运行后返回的网页出现了我的用户名,说明对方网站已经承认我已经登录了)咱就可以干N多事情,比如刷帖,刷回复等,你们懂的,对吧?
好了上面还不够有说服力我们来看一个php socket实现图片上传,这个代码有两点要注意.
一是他是http的post 请求;
二是表单上传协议,
下面的请求串适合任何语言,代码如下:
$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 = ''; // and attach the file $data .= "--$boundary\r\n"; $content_file = file_get_contents('aaa.jpg'); $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);
本文地址:
转载随意,但请附上文章地址:-)