PHP fsockopen函数详解

PHP fsockopen是一个功能比较强大的函数,我们在这篇文章中将会对这个函数做一个具体的介绍,希望对大家有所帮助,记得以前的B2C网站就是通过这个函数实现前台和订单处理系统的交互。

PHP fsockopen函数说明:

语法:

resource fsockopen ( string $hostname [, int KaTeX parse error: Expected 'EOF', got '&' at position 20: … = -1 [, int &̲errno [, string &$errstr [, float $timeout = ini_get(“default_socket_timeout”) ]]]] )

开启PHP fsockopen这个函数

PHP fsockopen需要 PHP.ini 中 allow_url_fopen 选项开启。

allow_url_fopen = On

参数:

hostname:如果安装了OpenSSL,那么你也许应该在你的主机名地址前面添加访问协议ssl://或者是tls://,从而可以使用基于TCP/IP协议的SSL或者TLS的客户端连接到远程主机。

port:端口号。如果对该参数传一个-1,则表示不使用端口,例如unix://。

errno:如果errno的返回值为0,而且这个函数的返回值为 FALSE ,那么这表明该错误发生在套接字连接(connect())调用之前,导致连接失败的原因最大的可能是初始化套接字的时候发生了错误。

errstr:错误信息将以字符串的信息返回。

timeout:设置连接的时限,单位为秒。

返回值:

fsockopen() 将返回一个文件句柄,之后可以被其他文件类函数调用(例如: fgets() , fgetss() , fwrite() , fclose() 还有 feof() )。如果调用失败,将返回 FALSE 。

php fsockopen使用案例

1、fsockopen 来模拟生成 HTTP 连接

  1. <?php
  2. $fp = fsockopen("127.0.0.1",80,$errno,$errstr,30);
  3. if(!$fp){
  4. echo "$errstr ($errno)<br />\n";
  5. }else{
  6. $out = "GET / HTTP/1.1\r\n";
  7. $out .= "Host: 127.0.0.1\r\n";
  8. $out .= "Connection: Close\r\n\r\n";
  9. fwrite($fp,$out);
  10. $content = '';
  11. while(!feof($fp)){
  12. $content .= fgets($fp,128);
  13. }
  14. echo $content;
  15. fclose($fp);
  16. }
  17. ?>

运行结果:

1.jpg

2、PHP fsockopen模拟POST/GET方法

fsockopen除了像上面实例模拟生成 HTTP 连接之外,还能实现很多功能,比如模拟post 和 get 传送数据的方法。

get :

  1. <?php
  2. $url = "http://localhost/test2.php?site=www.phpfensi.com";
  3. print_r(parse_url($url));// 解析 URL,返回其组成部分
  4. /* get提交 */
  5. sock_get($url,'user=gonn');
  6. // fsocket模拟get提交
  7. function sock_get($url,$query){
  8. $data = array(
  9. 'foo' => 'bar',
  10. 'baz' => 'boom',
  11. 'site' => 'www.tbrer.com',
  12. 'name' => 'nowa magic'
  13. );
  14. $query_str = http_build_query($data);// http_build_query()函数的作用是使用给出的关联(或下标)数组生成一个经过 URL-encode 的请求字符串
  15. $info = parse_url($url);
  16. $fp = fsockopen($info["host"],80,$errno,$errstr,30);
  17. $head = "GET " . $info['path'] . '?' . $query_str . " HTTP/1.0\r\n";
  18. $head .= "Host: " . $info['host'] . "\r\n";
  19. $head .= "\r\n";
  20. $write = fputs($fp,$head);
  21. while(!feof($fp)){
  22. $line = fread($fp,4096);
  23. echo $line;
  24. }
  25. }
  26. ?>

post :

  1. <?php
  2. $url = "http://localhost/test2.php?site=www.phpfensi.com";
  3. print_r(parse_url($url));// 解析 URL,返回其组成部分
  4. /* get提交 */
  5. sock_post($url,'user=gonn');
  6. // fsocket模拟get提交
  7. function sock_post($url,$query){
  8. $info = parse_url($url);
  9. $fp = fsockopen($info["host"],80,$errno,$errstr,30);
  10. $head = "POST " . $info['path'] . "?" . $info["query"] . " HTTP/1.0\r\n";
  11. $head .= "Host: " . $info['host'] . "\r\n";
  12. $head .= "Referer: http://" . $info['host'] . $info['path'] . "\r\n";
  13. $head .= "Content-type: application/x-www-form-urlencoded\r\n";
  14. $head .= "Content-Length: ". strlen(trim($query)) . "\r\n";
  15. $head .= "\r\n";
  16. $head .= trim($query);
  17. $write = fputs($fp,$head);
  18. while(!feof($fp)){
  19. $line = fread($fp,4096);
  20. echo $line;
  21. }
  22. }
  23. ?>

接收页面 test2.php 的代码为:

  1. <?php
  2. $data = $_REQUEST;
  3. echo '<pre>';
  4. print_r($data);
  5. echo '</pre>';
  6. ?>

3、fsockopen以Socket方式模拟HTTP下载文件

  1. <?php
  2. /*
  3. * Socket 模拟HTTP协议传输文件
  4. * Http是应用层协议使用80端口
  5. */
  6. $hostname = '127.0.0.1';
  7. $port = '80';
  8. // 建立连接
  9. $fp = fsockopen($hostname,$port,$errno,$errstr);
  10. stream_set_blocking($fp,true);
  11. if(!$fp){
  12. echo "$errno : $errstr<br />";
  13. }else{
  14. // 发送一个HTTP请求信息头
  15. $request_header = "GET /aaa.txt";
  16. // 起始行
  17. // 头域
  18. $request_header .= "Host: $hostname\n";
  19. // 再一个回车换行表示头信息结束
  20. $request_header .= "\n";
  21. // 发送请求到服务器
  22. fputs($fp,$request_header);
  23. // 接受响应
  24. $fp2 = fopen('aaa.txt','w');
  25. while(!feof($fp)){
  26. $line = fputs($fp2,fgets($fp,128));
  27. echo $line;
  28. }
  29. // 关闭
  30. fclose($fp2);
  31. fclose($fp);
  32. }
  33. ?>

执行程序,你会发现在这个程序文件的同级目录就会出现那个你需要下载的文件了。

这实质上是 Socket 模拟HTTP协议传输文件。同时还要注意一下 PHP 的超时限制,这里设置我 PHP 服务器超时为无限才能正确下载,否则可能下载不全 PHP 程序就停止了。

注意:

bool stream_set_blocking ( resource $stream , int $mode )

为 stream 设置阻塞或者阻塞模。

此函数适用于支持非阻塞模式的任何资源流(常规文件,套接字资源流等)。

参数

stream:资源流。

mode:如果 mode 为0,资源流将会被转换为非阻塞模式;如果是1,资源流将会被转换为阻塞模式。 该参数的设置将会影响到像 fgets() 和 fread() 这样的函数从资源流里读取数据。 在非阻塞模式下,调用 fgets() 总是会立即返回;而在阻塞模式下,将会一直等到从资源流里面获取到数据才能返回。

返回值

成功时返回 TRUE, 或者在失败时返回 FALSE。

4、使用 fsockopen 伪造来路

  1. <?php
  2. $host = "127.0.0.1"; //你要访问的域名
  3. $ip = '127.0.0.1';
  4. $target = "/test2.php"; //你要访问的页面地址
  5. $referer = "http://www.phpfensi.com/"; //伪造来路页面
  6. //$fp = fsockopen($host, 80, $errno, $errstr, 30);
  7. $fp = fsockopen($ip, 80, $errno, $errstr, 30);
  8. if(!$fp)
  9. {
  10. echo "$errstr($errno)<br />\n";
  11. }
  12. else
  13. {
  14. $end = "\r\n";
  15. $out = "GET $target HTTP/1.1$end";
  16. $out .= "Host: $ip$end";
  17. $out .= "Referer: $referer$end";
  18. $out .= "Connection: Close$end";
  19. $out .= "$end";
  20. fwrite($fp, $out);
  21. while(!feof($fp))
  22. {
  23. echo fgets($fp, 1024);
  24. }
  25. fclose($fp);
  26. }
  27. ?>

test2.php 的代码为:

  1. <?php
  2. $data = $_REQUEST;
  3. echo '<pre>';
  4. print_r($data);
  5. echo '</pre>';
  6. ?>

可以看到 HTTP_REFERER 的值为 http://www.phpfensi.com/,即来路已经伪造成功。