欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

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

程序员文章站 2022-03-10 15:08:13
从零开始一个http服务器 模拟cgi(五) 代码地址 : https://github.com/flamedancer/cserver git checkout step5 运行: make clean && make && ./myserver.out 测试 浏览器打开 http://127.0 ......

从零开始一个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 打开 可以看到实时的时间