如何用PHP来实现一个动态Web服务器


Posted in PHP onJuly 29, 2015

要是现实一个 web 服务器,那么就需要大概了解 web 服务器的运行原理。先从静态的文本服务器开始,以访问 web 服务器的1.html为例

1.客户端通过发送一个 http 请求到服务器,如果服务器监听的端口号是9002,那么在本机自身测试访问的地址就是http://localhost:9002/1.html。

2.服务器监听着9002端口,那么在收到请求了请求之后,就能从 http head 头中获取到请求里需要访问的 uri 资源在web 目录中的位置。

3.服务器读取需要访问的资源文件,然后填充到 http 的实体中返回给客户端。

示意图如下:

如何用PHP来实现一个动态Web服务器

<?php
class web_config {
  // 监听的端口号
  const PORT = 9003;
  // 项目根目录
  const WEB_ROOT = "/Users/zhoumengkang/Documents/html";
}


class server {
  private $ip;
  private $port;
  public function __construct($ip, $port) {
    $this->ip = $ip;
    $this->port = $port;
    $this->await();
  }
  private function await() {
    $sock = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
    if ($sock < 0) {
      echo "Error:" . socket_strerror(socket_last_error()) . "\n";
    }
    $ret = socket_bind($sock, $this->ip, $this->port);
    if (!$ret) {
      echo "BIND FAILED:" . socket_strerror(socket_last_error()) . "\n";
      exit;
    }
    echo "OK\n";
    $ret = socket_listen($sock);
    if ($ret < 0) {
      echo "LISTEN FAILED:" . socket_strerror(socket_last_error()) . "\n";
    }
    do {
      $new_sock = null;
      try {
        $new_sock = socket_accept($sock);
      } catch (Exception $e) {
        echo $e->getMessage();
        echo "ACCEPT FAILED:" . socket_strerror(socket_last_error()) . "\n";
      }
      try {
        $request_string = socket_read($new_sock, 1024);
        $response = $this->output($request_string);
        socket_write($new_sock, $response);
        socket_close($new_sock);
      } catch (Exception $e) {
        echo $e->getMessage();
        echo "READ FAILED:" . socket_strerror(socket_last_error()) . "\n";
      }
    } while (TRUE);
  }
  /**
   * @param $request_string
   * @return string
   */
  private function output($request_string){
    // 静态 GET /1.html HTTP/1.1 ...
    $request_array = explode(" ",$request_string);
    if(count($request_array) < 2){
      return $this->not_found();
    }
    $uri = $request_array[1];
    $filename = web_config::WEB_ROOT . $uri;
    echo "request:".$filename."\n";
    // 静态文件的处理
    if (file_exists($filename)) {
      return $this->add_header(file_get_contents($filename));
    } else {
      return $this->not_found();
    }
  }
  /**
   * 404 返回
   * @return string
   */
  private function not_found(){
    $content = "

<h1>File Not Found </h1>

";
    return "HTTP/1.1 404 File Not Found\r\nContent-Type: text/html\r\nContent-Length: ".strlen($content)."\r\n\r\n".$content;
  }
  /**
   * 加上头信息
   * @param $string
   * @return string
   */
  private function add_header($string){
    return "HTTP/1.1 200 OK\r\nContent-Length: ".strlen($string)."\r\nServer: mengkang\r\n\r\n".$string;
  }
}
$server = new server("127.0.0.1", web_config::PORT);

如上代码所述,只要在终端执行该文件,那么一个静态的 web 服务器就启动啦。

下图为我访问我 web 目录下的1.jpg文件的截图

如何用PHP来实现一个动态Web服务器

简单的静态 web 服务器已经完成了,下面的问题就是怎么让其支持动态内容的输出了。是不是只需要在 web 服务器内部执行完某个程序之后,把得到的结果返回给客户端就行呢?但是这样 web 服务器的代码就和业务代码耦合在一起了,怎么解决一个 web 服务器,可以运用在各个业务场景下呢?

CGI 的出现解决了这一问题。那么 CGI 是什么呢?下面这段话复制的:

CGI是外部应用程序(CGI程序)与Web服务器之间的接口标准,是在CGI程序和Web服务器之间传递信息的规程。CGI规范允许Web服务器执行外部程序,并将它们的输出发送给Web浏览器,CGI将Web的一组简单的静态超媒体文档变成一个完整的新的交互式媒体。

好晕,举个具体的例子,比如我们在使用的 PHP 的全局变量$_SERVER['QUERY_STRING']就是 Web 服务器通过 CGI 协议之上,传递过来的。例如在 Nginx 中,也许你记得这样的 fastcgi 配置

fastcgi_param  QUERY_STRING       $query_string;

没错 nginx 把自己的全局变量$query_string传递给了 fastcgi_param 的环境变量中。

下面我们也以CGI的QUERY_STRING作为桥梁,将客户端请求的 uri 中的信息传递到 cgi 程序中去。通过putenv的方式把QUERY_STRING存入该请求的环境变量中。

我们约定 Web 服务器中访问的资源是.cgi后缀则表示是动态访问,这一点有点儿类似于 nginx 里配置 location 来寻找 php 脚本程序一样。都是一种检查是否应该请求 cgi 程序的规则。为了和 Web 服务器区别开来,我用 C 写了一个查询用户信息的 cgi 程序,根据用户 id 查询用户资料。

大致的访问逻辑如下图

如何用PHP来实现一个动态Web服务器

演示代码地址:https://github.com/zhoumengkang/php/tree/master/php-webserver/dynamic

如果要运行该 demo 需要做如下操作

1.修改config.php里的项目根目录WEB_ROOT

2.编译cgi-demo\user.c,编译命令gcc -o user.cgi user.c,然后将user.cgi文件放入你配置的项目根目录下面

3.在终端执行php start.php ,这样该 web 服务器就启动了

4.通过 http://localhost:9003/user.cgi?id=1 就可以访问看到如下效果了

如何用PHP来实现一个动态Web服务器

其实只是在静态服务器的基础上做了一些 cgi 的判断是请求的转发处理,把github 上的三个文件的代码合并到一个文件里方便大家观看

<?php
class web_config {

  // 监听的端口号
  const PORT = 9003;

  // 项目根目录
  const WEB_ROOT = "/Users/zhoumengkang/Documents/html";

  // 系统支持的 cgi 程序的文件扩展名
  const CGI_EXTENSION = "cgi";

}

class server {
  private $ip;
  private $port;
  public function __construct($ip, $port) {
    $this->ip = $ip;
    $this->port = $port;
    $this->await();
  }

  private function await() {
    $sock = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
    if ($sock < 0) {
      echo "Error:" . socket_strerror(socket_last_error()) . "\n";
    }

    $ret = socket_bind($sock, $this->ip, $this->port);
    if (!$ret) {
      echo "BIND FAILED:" . socket_strerror(socket_last_error()) . "\n";
      exit;
    }
    echo "OK\n";

    $ret = socket_listen($sock);
    if ($ret < 0) {
      echo "LISTEN FAILED:" . socket_strerror(socket_last_error()) . "\n";
    }

    do {
      $new_sock = null;
      try {
        $new_sock = socket_accept($sock);
      } catch (Exception $e) {
        echo $e->getMessage();
        echo "ACCEPT FAILED:" . socket_strerror(socket_last_error()) . "\n";
      }
      try {
        $request_string = socket_read($new_sock, 1024);
        $response = $this->output($request_string);
        socket_write($new_sock, $response);
        socket_close($new_sock);
      } catch (Exception $e) {
        echo $e->getMessage();
        echo "READ FAILED:" . socket_strerror(socket_last_error()) . "\n";
      }
    } while (TRUE);
  }
  /**
   * @param $request_string
   * @return string
   */
  private function output($request_string){
    // 静态 GET /1.html HTTP/1.1 ...
    // 动态 GET /user.cgi?id=1 HTTP/1.1 ...
    $request_array = explode(" ",$request_string);
    if(count($request_array) < 2){
      return "";
    }
    $uri = $request_array[1];
    echo "request:".web_config::WEB_ROOT . $uri."\n";
    $query_string = null;
    if ($uri == "/favicon.ico") {
      return "";
    }
    if (strpos($uri,"?")) {
      $uriArr = explode("?", $uri);
      $uri = $uriArr[0];
      $query_string = isset($uriArr[1]) ? $uriArr[1] : null;
    }
    $filename = web_config::WEB_ROOT . $uri;
    if ($this->cgi_check($uri)) {

      $this->set_env($query_string);
      $handle = popen(web_config::WEB_ROOT.$uri, "r");
      $read = stream_get_contents($handle);
      pclose($handle);
      return $this->add_header($read);
    }
    // 静态文件的处理
    if (file_exists($filename)) {
      return $this->add_header(file_get_contents($filename));
    } else {
      return $this->not_found();
    }
  }
  /**
   * 设置环境变量 给 cgi 程序使用
   * @param $query_string
   * @return bool
   */
  private function set_env($query_string){
    if($query_string == null){
      return false;
    }
    if (strpos($query_string, "=")) {
      putenv("QUERY_STRING=".$query_string);
    }
  }
  /**
   * 判断请求的 uri 是否是合法的 cgi 资源
   * @param $uri
   * @return bool
   */
  private function cgi_check($uri){
    $info = pathinfo($uri);
    $extension = isset($info["extension"]) ? $info["extension"] : null;
    if( $extension && in_array($extension,explode(",",web_config::CGI_EXTENSION))){
      return true;
    }
    return false;
  }
  /**
   * 404 返回
   * @return string
   */
  private function not_found(){
    $content = "<h1>File Not Found </h1>";
    return "HTTP/1.1 404 File Not Found\r\nContent-Type: text/html\r\nContent-Length: ".strlen($content)."\r\n\r\n".$content;
  }
  /**
   * 加上头信息
   * @param $string
   * @return string
   */
  private function add_header($string){
    return "HTTP/1.1 200 OK\r\nContent-Length: ".strlen($string)."\r\nServer: mengkang\r\n\r\n".$string;
  }
}

$server = new server("127.0.0.1", web_config::PORT);

以上就是PHP实现一个动态Web服务器的全部实现过程,希望对大家的学习有所帮助。

PHP 相关文章推荐
处理单名多值表单的详解
Jun 08 PHP
深入解析PHP垃圾回收机制对内存泄露的处理
Jun 14 PHP
xss防御之php利用httponly防xss攻击
Mar 21 PHP
PHP采用get获取url汉字出现乱码的解决方法
Nov 13 PHP
PHP中curl_setopt函数用法实例分析
Apr 16 PHP
php生成静态html页面的方法(2种方法)
Sep 14 PHP
求帮忙修改个php curl模拟post请求内容后并下载文件的解决思路
Sep 20 PHP
php简单获取复选框值的方法
May 11 PHP
PHP页面输出搜索后跳转下一页的处理方法
Sep 30 PHP
Yii2实现跨mysql数据库关联查询排序功能代码
Feb 10 PHP
PHP __call()方法实现委托示例
May 20 PHP
详解PHP服务器如何在有限的资源里最大提升并发能力
May 25 PHP
php实现过滤字符串中的中文和数字实例
Jul 29 #PHP
php实现XSS安全过滤的方法
Jul 29 #PHP
php检查字符串中是否有外链的方法
Jul 29 #PHP
php数组比较实现查找连续数的方法
Jul 29 #PHP
PHP实现XML与数据格式进行转换类实例
Jul 29 #PHP
PHP获取某个月最大天数(最后一天)的方法
Jul 29 #PHP
discuz图片顺序混乱解决方案
Jul 29 #PHP
You might like
frename PHP 灵活文件命名函数 frename
2009/09/09 PHP
PHP学习笔记 (1) 环境配置与代码调试
2011/06/19 PHP
PHP 自定义错误处理函数的使用详解
2013/05/10 PHP
php json与xml序列化/反序列化
2013/10/28 PHP
smarty内置函数capture用法分析
2015/01/22 PHP
php 实现301重定向跳转实例代码
2016/07/18 PHP
javascritp实现input输入框相关限制用法
2007/06/29 Javascript
javascript 数据类型转换(parseInt,parseFloat)
2010/07/20 Javascript
jquery中的事件处理详细介绍
2013/06/24 Javascript
JQuery Highcharts 动态生成图表的方法
2013/11/15 Javascript
JavaScript 里的类数组对象
2015/04/08 Javascript
JS实现兼容性好,自动置顶的淘宝悬浮工具栏效果
2015/09/18 Javascript
element-ui 限制日期选择的方法(datepicker)
2018/05/16 Javascript
微信小程序登录数据解密及状态维持实例详解
2019/05/06 Javascript
Swiper.js实现移动端元素左右滑动
2019/09/08 Javascript
c++生成dll使用python调用dll的方法
2014/01/20 Python
Python xlrd读取excel日期类型的2种方法
2015/04/28 Python
Python导入模块时遇到的错误分析
2017/08/30 Python
对python3标准库httpclient的使用详解
2018/12/18 Python
Python字符串内置函数功能与用法总结
2019/04/16 Python
使用python获取(宜宾市地震信息)地震信息
2019/06/20 Python
python中PS 图像调整算法原理之亮度调整
2019/06/28 Python
Django logging配置及使用详解
2019/07/23 Python
django 控制页面跳转的例子
2019/08/06 Python
在python中做正态性检验示例
2019/12/09 Python
学生如何注册Pycharm专业版以及pycharm的安装
2020/09/24 Python
详解anaconda安装步骤
2020/11/23 Python
Python实现迪杰斯特拉算法并生成最短路径的示例代码
2020/12/01 Python
matplotlib之多边形选区(PolygonSelector)的使用
2021/02/24 Python
小学生运动会报道稿
2014/09/12 职场文书
专题民主生活会对照检查材料思想汇报
2014/09/29 职场文书
2014年驾驶员工作总结
2014/11/18 职场文书
2014年局领导班子自身建设情况汇报
2014/11/21 职场文书
2014年学校体育工作总结
2014/12/08 职场文书
mysql查询结果实现多列拼接查询
2022/04/03 MySQL
进阶篇之linux环境下安装MySQL数据库
2022/04/09 MySQL