CGI(通用网关接口)

公共网关接口

CGI(Common Gateway Interface) 是WWW技术中最重要的技术之一,有着不可替代的重要地位。CGI是外部应用程序(CGI程序)与Web服务器之间的接口标准,是在CGI程序和Web服务器之间传递信息的规程。CGI规范允许Web服务器执行外部程序,并将它们的输出发送给Web浏览器,CGI将Web的一组简单的静态超媒体文档变成一个完整的新的交互式媒体。

Common Gateway Interface,简称CGI。在物理上是一段程序,运行在服务器上,提供同客户端HTML页面的接口。这样说大概还不好理解。那么我们看一个实际例子:现在的个人主页上大部分都有一个留言本。留言本的工作是这样的:先由用户在客户端输入一些信息,如名字之类的东西。接着用户按一下“留言”(到目前为止工作都在客户端),浏览器把这些信息传送到服务器的CGI目录下特定的cgi程序中,于是cgi程序在服务器上按照预定的方法进行处理。在本例中就是把用户提交的信息存入指定的文件中。然后cgi程序客户端发送一个信息,表示请求的任务已经结束。此时用户在浏览器里将看到“留言结束”的字样。整个过程结束。

功能

绝大多数的CGI程序被用来解释处理来自表单的输入信息,并在服务器产生相应的处理,或将相应的信息反馈给浏览器。CGI程序使网页具有交互功能。

处理步骤

⑴通过Internet把用户请求送到web服务器

⑵web服务器接收用户请求并交给CGI程序处理。

⑶CGI程序把处理结果传送给web服务器

⑷web服务器把结果送回到用户。

服务器配置

CGI程序不是放在服务器上就能顺利运行,如果要想使其在服务器上顺利的运行并准确的处理用户的请求,则须对所使用的服务器进行必要的设置。

配置:根据所使用的服务器类型以及它的设置把CGI程序放在某一特定的目录中或使其带有特定的扩展名。

⑴CREN格式服务器的配置:

编辑CREN格式服务器的配置文件(通常为/etc/httpd.conf)在文件中加入:Exec cgi-bin/*/home/www/cgi-bin/*.exec。命令中出现的第一个参数cgi-bin/*指出了在URL中出现的目录名字,并表示它出现在系统主机后的第一个目录中,如:http://edgar.stern.nyn.***/cgi-bin/。命令中的第二个参数表示CGI程序目录放在系统中的真实路径。

CGI目录除了可以跟网络文件放在同一目录中,也可以放在系统的其它目录中,但必须保证在你的系统中也具有同样的目录。在对服务器完成设置后,须重新启动服务器(除非HTTP服务器是用inetd启动的)。

⑵NCSA格式服务器的配置

在NCSA格式服务器上有两种方法进行设置:

①在srm.conf文件(通常在conf目录下)中加入:Script Alias/cgi-bin/cgi-bin/。Script Alias命令指出某一目录下的文件是可执行程序,且这个命令是用来执行这些程序的;此命令的两个参数与CERN格式服务器中的Exec命令的参数的含意一样。

②在srm.conf文件加入:Add type application/x-httpd-cgi.cgi。此命令表示在服务器上增加了一种新的文件类型,其后第一个参数为CGI程序的MIME类型,第二个参数是文件的扩展名,表示以这一扩展名为扩展名的文件是CGI程序。

在用上述方法之一设置服务器后,都得重新启动服务器(除非HTTP服务器是用inetd启动的)。

编写语言

CGI可以用任何一种语言编写,只要这种语言具有标准输入、输出和环境变量。对初学者来说,最好选用易于归档和能有效表示大量数据结构的语言,例如UNIX环境中:

· Perl (Practical Extraction and Report Language)

· Bourne Shell或者Tcl (Tool Command Language)

· PHP(Hypertext Preprocessor))

由于C语言有较强的平台无关性,所以也是编写CGI程序的首选。

Windows环境中:

· C和C++

由于Internet上大部分服务器使用的是UNIX操作系统,且几乎任一UNIX操作系统中都有Bourne Shell,因而后面讲述的例子中大部分是用Bourne Shell编写的。

最终Perl由于其跨操作系统、易于修改的特性成为了CGI的主流编写语言,以至于一般的“cgi程序”就是Perl程序。

环境变量列表

SERVER_NAME:运行CGI序为机器名或IP地址。

SERVER_INTERFACE:WWW服务器的类型,如:CERN型或NCSA型。

SERVER_PROTOCOL:通信协议,应当是HTTP/1.0。

SERVER_PORT:TCP端口,一般说来web端口是80。

HTTP_ACCEPT:HTTP定义的浏览器能够接受的数据类型

HTTP_REFERER:发送表单的文件URL。(并非所有的浏览器都传送这一变量

HTTP_USER-AGENT:发送表单浏览器的有关信息。

GETWAY_INTERFACE:CGI程序的版本,在UNIX下为 CGI/1.1。

PATH_TRANSLATED:PATH_INFO中包含的实际路径名。

PATH_INFO:浏览器用GET方式发送数据时的附加路径。

SCRIPT_NAME:CGI程序的路径名。

QUERY_STRING:表单输入的数据,URL中问号后的内容。

REMOTE_HOST:发送程序主机名,不能确定该值。

REMOTE_ADDR:发送程序的机器的IP地址。

REMOTE_USER:发送程序的人名。

CONTENT_TYPE:POST发送,一般为application/xwww-form-urlencoded。

CONTENT_LENGTH:POST方法输入的数据的字节数。

搭建C语言CGI和Apache服务器的开发环境

步骤如下:

首先,需要用到的这些工具和代码:

  • C语言编译器;

  • Apache服务器,我用的是USBWebSever中包含的Apache服务器(下载地址) ,这是个AMP服务器套装,不用安装即可使用,而本地安装的Apche服务器也可以使用;

接着,编译C语言的cgi程序。

源码如下:C CGI Example

然后,配置和启动Apache服务器。

对USBWebSever,对settings目录下的httpd.conf如下内容进行修改,如下:

ScriptAlias /cgi-bin/ "{rootdir}/cgi-bin/"

<Directory "{rootdir}/cgi-bin">
#   AllowOverride All
#   Options None
    AllowOverride None
    Options ExecCGI
    Order allow,deny
    Allow from all
</Directory>
AddHandler cgi-script .exe .pl .cgi

修改好以后,双击USBWebSever.exe就可以启动Apache服务器了。

最后把刚才生成的cgi程序(.exe文件),复制放到上文中提到的/cgi-bin/目录下,文件名最好改成index.cgi这样的形式。对于USBWebSever,cgi-bin目录应该是root目录下的cgi-bin目录(如果没有要新建一个),不是和USBWebSever在同一目录下的cgi-bin目录。目录结构如下:

打开浏览器输入http://127.0.0.1:8080/cgi-bin/index.cgi,即可访问cgi程序。

C CGI Example

index.html

<head>
  <meta http-equiv="Content-Type" content="text/html;charset=utf-8" />
  <body>
    <form action="/cgi-bin/mult.cgi">
        <p>CGI表单处理:Get方法演示:输入乘数和被乘数,计算结果</p>
            M : <input name="m" size="5">
            N : <input name="n" size="5">
        <br><input type="submit" value="确定"></input></br>

    </form>

    <form action="/cgi-bin/collect.cgi" method="POST" >
        <p>CGI表单处理:Post方法演示:请输入您的留言(最多80个字符):
        <br><input name="data" size="60" maxlength="80"></br>
        <input type="SUBMIT" value="确定">
    </form> 

    <form action="/cgi-bin/viewdata.cgi">
        <p><input type="SUBMIT" value="察看留言">
    </form>
  </body>
</head>

mult.cgi

#include <stdio.h>
#include <stdlib.h>
int main(void)
{
    char *data;
    long m, n;
    printf("%s", "Content-Type:text/html\n\n");
    printf("<html>");
    printf("<head><title>乘法结果</title>");
    printf("<h3>乘法结果</h3> ");
    printf("</head><body>");
    data = getenv("QUERY_STRING");
    if (data == NULL)
        printf("<p>错误!数据没有被输入或者数据传输有问题");
    else if (sscanf(data, "m=%ld&n=%ld", &m, &n) != 2)
        printf("<p>错误!输入数据非法。表单中输入的必须是数字。");
    else
        printf("<p>%ld * %ld = %ld。", m, n, m * n);
    printf("</body></html>");
    return 0;
}

运行示例:

collect.cgi

#include <stdio.h>
#include <stdlib.h>
#define MAXLEN 80
#define EXTRA 5
 4个字节留给字段的名字"data", 1个字节留给"="
#define MAXINPUT MAXLEN+EXTRA+2
 1个字节留给换行符,还有一个留给后面的NULL
#define DATAFILE "../data/data.txt"
 要被添加数据的文件
void decode(char *src, char *last, char *dest)
{
    for (; src != last; src++, dest++)
        if (*src == ‘+‘)
            *dest = ‘ ‘;
        else if (*src == ‘%‘)
        {
            int code;
            if (sscanf(src + 1, "%2x", &code) != 1)
                code = ‘?‘;
            *dest = code;
            src += 2;
        }
        else
            *dest = *src;
    *dest = ‘\n‘;
    *++dest = ‘\0‘;
}
int main(void)
{
    char *lenstr;
    char input[MAXINPUT], data[MAXINPUT];
    long len;
    printf("%s", "Content-Type:text/html\n\n");
    printf("<html><head><title>Response</title>");
    printf("</head><body>");
    lenstr = getenv("CONTENT_LENGTH");
    if (lenstr == NULL || sscanf(lenstr, "%ld", &len) != 1 || len> MAXLEN)
        printf("<p>表单提交错误");
    else
    {
        FILE *f;
        fgets(input, len + 1, stdin);
        decode(input + EXTRA, input + len, data);
        f = fopen(DATAFILE, "a");
        if (f == NULL)
            printf("<p>对不起,意外错误导致系统无法保存你的留言");
        else{
            fputs(data, f);
            printf("<p>非常感谢,您的留言已被系统接收");
        }

        fclose(f);

    }

    printf("</body></html>");
    return 0;
}

运行示例:

viewdata.cgi

#include <stdio.h>
#include <stdlib.h>
#define DATAFILE "../data/data.txt"
int main(void)
{
    FILE *f = fopen(DATAFILE, "r");
    int ch;
    if (f == NULL)
    {
        printf("%s", "Content-Type:text/html\n\n");
        printf("<P><EM>意外错误,无法打开文件</EM>");
    }
    else
    {
        printf("%s", "Content-Type:text/plain\n\n");
        while ((ch = getc(f)) != EOF)
            putchar(ch);
        fclose(f);
    }

    return 0;
}

运行示例:

时间: 2024-12-10 18:06:16

CGI(通用网关接口)的相关文章

(一)CGI (通用网关接口) 简介

CGI (通用网关接口)公共网关接口(Common Gateway Interface,CGI)是Web 服务器运行时外部程序的规范,按CGI 编写的程序可以扩展服务器功能.CGI 应用程序能与浏览器进行交互,还可通过数据API与数据库服务器等外部数据源进行通信,从数据库服务器中获取数据.格式化为HTML文档后,发送给浏览器,也可以将从浏览器获得的数据放到数据库中.几乎所有服务器都支持CGI,可用任何语言编写CGI,包括流行的C.C ++.VB 和Delphi 等.CGI分为标准CGI和间接CG

wemall app商城源码Android之通用通知接口demo

wemall-mobile是基于WeMall的Android app商城,只需要在原商城目录下上传接口文件即可完成服务端的配置,客户端可定制修改.本文分享Native(原生)支付模式一demo,供技术员参考学习. wemall官网地址:http://www.wemallshop.com 支付完成后,微信会把相关支付和用户信息发送到商户设定的通知URL,商户接收回调信息后,根据需要设定相应的处理流程. 这里举例使用log文件形式记录回调信息. notify_url.php <?php includ

金山WPS、微软Office EXCEL表格通用C++接口

金山WPS.微软Office EXCEL表格通用C++接口 之前有个小项目在win32框架下,有个导出报表的小需求excel,小小的整理了下几个接口.最近闲着了解了下VAB宏,扩展了几个实用接口,适用于金山wps办公软件,至于微软的excel版本是否能够忽略,这个还真没试过,理论上某些接口是可以的(只要VAB宏没变,这些信息是从一些专业的VAB群中得来,感谢群里的兄弟支持).源码往下拉,供学习交流之用.后边有时间再做word的. 描述: 对Excel常用操作封装类,适用于Microexcel.W

中国电信翼支付网关接口接入

最近在做中国电信的翼支付网关接口的接入,正好拿Java练练手.到目前为止,唯一不太适应的就是自己的Java积累几乎为0,什么都要重头写起,不像C#有这么多年的沉淀,可以随手拿来用.   废话先不多说.这个支付接口就和支付宝差不多,无非就是把一些必要的数据参数,POST到指定的接口地址,然后接收它返回的支付结果.接口没有什么复杂的逻辑处理,所以可以直接采用JSP+Servlet+JDBC来做简单处理.对于WEB支付请求的发起,可以从JSP把用户输入或选择的相关数据传到Servlet,然后在Serv

STM32W108无线射频模块通用IO接口应用实例

STM32W108无线射频模块通用IO接口应用实例 本实例编写STM32W108的GPIO测试程序,通过控制GPIO引脚,实现对LED灯的控制. 开发环境与硬件说明 硬件:STM32W108无线开发板,5V 1A电源,J-Link烧写器,PC机等. 软件:IAR Embedded Workbench for ARM开发软件. 硬件连接的原理图如图6.15所示:通过STM32W108无线模块的GPIO引脚PA3控制LED1,GPIO引脚PC2控制LED2. 硬件规划:其中用到的控制管脚为GPIO_

七、springboot整合Spring-data-jpa(二)之通用DAO接口与添加自定义方法

@NoRepositoryBean:Spring Data Jpa在启动时就不会去实例化BaseRepository这个接口 1.通用接口: import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.data.jpa.repository.JpaSpecificationExecutor; import org.springframework.data.repository.N

python调用tushare的pro_bar通用行情接口

接口名称:pro_bar 更新时间:股票和指数通常在15点-17点之间,数字货币实时更新,具体请参考各接口文档明细. 描述:目前整合了股票(未复权.前复权.后复权).指数.数字货币.ETF基金.期货.期权的行情数据,未来还将整合包括外汇在内的所有交易行情数据,同时提供分钟数据. 其它:由于本接口是集成接口,在SDK层做了一些逻辑处理,目前暂时没法用http的方式调取通用行情接口.用户可以访问Tushare的Github,查看源代码完成类似功能. 注:tushare包下载和初始化教程,请查阅我之前

SpringBoot实现通用的接口参数校验

本文介绍基于Spring Boot和JDK8编写一个AOP,结合自定义注解实现通用的接口参数校验. 缘由 目前参数校验常用的方法是在实体类上添加注解,但对于不同的方法,所应用的校验规则也是不一样的,例如有一个AccountVO实体: publicclassAccountVO{privateStringname;//姓名privateIntegerage;//年龄} 假设存在这样一个业务:用户注册时需要填写姓名和年龄,用户登陆时只需要填写姓名就可以了.那么把校验规则加在实体类上显然就不合适了. 所

CacheManager:–个通用缓存接口抽象类库

CacheManager是–个缓存通用接口抽象类库,它支持各种高速缓存提供者,例如Memcache,Redis,并且有许多先进的功能特性.具体可以访问官方网站  http://cachemanager.net. CacheManager设计目标就是简化程序员对各种复杂缓存场景的处理,通过CacheManager只需要几行的代码就可以支持多层的缓存,从进程内缓存到分布式的缓存.通过CacheManager可以很容易在项目中更改缓存策略,它还提供一些更有价值的特性,例如高速缓存同步,并发更新,事件通