以前模拟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);
|