用户登录
用户注册

分享至

从零开始一个http服务器-模拟cgi(五)

  • 作者: 腰力很好但没用
  • 来源: 51数据库
  • 2021-11-21

从零开始一个http服务器-模拟cgi(五)

代码地址 :
git checkout step5
运行:
make clean && make && ./myserver.out
测试
浏览器打开

模拟cgi:用外部程序来优化 动态 response

  • cgi解释
  • 调用外部程序

cgi解释

上一节中,我们确实是实现了动态的response:我们不需要修改我们的代码,也不需要中断server,只需要修改我们的html页面文件就可以实时的更改返回内容。

但是当需要返回更加灵活的内容,比如当前时间的时候,我们不可能每隔一秒钟就去改下页面文件。这时候就可以借助外部的程序,比如shell,来为我们生产返回内容。这个外部程序,就类似于常说的cgi程序。所谓cgi,维基百科是这么解释:通用网关接口common gateway interface/cgi)是一种重要的互联网技术,可以让一个客户端,从网页浏览器向执行在网络服务器上的程序请求数据。cgi描述了服务器和请求处理程序之间传输数据的一种标准。

这是说cgi是一种标准,只要服务器 和 外部的程序 都实现了这个标准,就可以相互通信。

我们这比较粗鲁一点,我们不管正规的cgi标准是什么,我们定义一个我们自己简单粗暴的标准,这个标准只有两条:

  1. 外部的程序 可以产生 标准输出
  2. 服务器可以获得外部的程序产生的标准输出

这样的话我们就可以把外部的程序的标准输出直接作为response的body。

调用外部程序

执行外部程序的核心函数为popen ,它会开启一个新进程执行传入的外部命令,返回一个管道文件流,读取这个管道文件流就可以读取到外部命令的输出。管道文件流需要用pclose关闭,而不是fclose。

void docgi(char * filepath, struct http_response * response) {
    char filename[100];
    char cmd[100];
    sprintf(filename, "cgi/%s", filepath + strlen(action_url + 1));
    sprintf(cmd, "%s 2>&1", filename);

    file *fstream = null;
    if (access(filename, f_ok) == -1 || null == (fstream = popen(cmd, "r"))) {
        // file doesn't exist or file cannot be exec
        setresponsemsg(response, errormsg);
        return;
    }

    response->body = (char *)malloc((5000));
    int len = 0;
    char *buff = response->body;
    do {
        buff += len;
        len = fread(buff, 1024, 1, fstream);
        // printf("%d\n", len);

    } while (len);
    pclose(fstream);
    response->body_size = strlen(response->body);
    struct item *item2 = newitem(
        "content-type",
        "text/html; charset=utf-8");
    mappush(response->headers, item2);

    return;
}

建一个目录名为cgi,把我们写的外部程序放到这个目录,例如我们写个 cgi/show_date 程序:

#! /bin/bash
echo $(date)

make clean && make && ./myserver.out 打开 可以看到实时的时间

软件
前端设计
程序设计
Java相关