什么是CGI?

CGI即通用网关接口(Common Gateway Interface),是一个Web服务器主机提供信息服务的标准接口。通过CGI接口,Web服务器就能够获取客户端提交的信息,转交给服务器端的CGI程序进行处理,最后返回结果给客户端。

根据CGI标准,编写外部扩展应用程序,可以对客户端浏览器输入的数据进行处理,完成客户端与服务器的交互操作。CGI规范定义了Web服务器如何向扩展应用程序发送消息,在收到扩展应用程序的信息后又如何进行处理等内容。

一般情况下,服务器和CGI程序之间是通过标准输入输出来进行数据传递的,CGI程序中的标准输出stdout是经过重定义了的,它并没有在服务器上产生任何的输出内容,而是被重定向到客户浏览器。

即我们的CGI程序是在我们的板子服务器里运行,但实际调用printf输出信息时并不会输出到我们的板子终端,而是输出到客户端浏览器。

CGI程序小例子

CGI程序可以使用多种语言来实现,这里我们选用我们比较熟悉的C语言来实现。先来看一个简单的示例:

「test.c:」

  1. #include <stdio.h>

  2. int main()

  3.  {

  4.     printf("Content-type: text/html\n\n") ;  // 这一句是必须的,设定输出到HTML

  5.     printf("Hello CGI\n") ;

  6.     return 0;

  7. }

利用交叉编译工具,编译上述文件:

arm-linux-xxxxx-gcc test.c -o test.cgi

把test.cgi程序传到开发板上的/www/cgi-bin目录

scp test.cgi root@192.168.31.12:/www/cgi-bin

修改权限;

chmod 777 test.cgi

浏览器进行访问:

http://192.168.31.12/cgi-bin/test.cgi

已经知道了CGI程序是什么了。下面我们来进行网页点灯。首先,需要说明的是,我们有两种方式来编写CGI程序。一种是借助环境变量来获取相关信息;另一种是使用CGIC库。

方法一:

对于CGI程序来说,它继承了系统的环境变量。CGI环境变量在CGI程序启动时初始化,在结束时销毁。当一个CGI程序不是被服务器调用时,它的环境变量几乎是系统环境变量的复制。当这个CGI程序被服务器调用时,它的环境变量就会多了以下关于服务器、客户端、CGI传输过程等项目。如:

方法二:

用C语言写CGI程序还可以有比较简单的方式,那就是我们可以借助使用第三方库CGIC(CGIC是一个功能比较强大的支持CGI开发的标准C库,并支持Linux, Unix 和Windows等多操作系统)来编写,省去了必须自己去遵循CGI规范来编码的痛苦。

CGIC库下载地址:

https://github.com/boutell/cgic

首先,我们需要点灯,自然需要先设计一个简单的led.html网页。我们要通过网页控制开发板上的led,需要两条信息:led的序号及led的状态。在网上找到了现成的html代码,稍微修改一下就直接拿来用了(文末给出参考的博客)。

led.html:

  1. <html xmlns="http://www.w3.org/1999/xhtml">

  2.  <head>

  3.  <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />    

  4.  <title>led control</title>

  5.  </head>

  6.  

  7.  <body>

  8.  <h1 align="center">基于ARM的web控制LED灯</h1>

  9.   <!--新建一个表单,动作链接到开发板的/cgi-bin/led.cgi,采用的方法为GET--> 

  10.   <form action="/cgi-bin/led.cgi" method="get">  

  11.    <p align="center">Web端的led的控制测试</p>

  12.    <p align="center">请输入需要控制的led <input type="text" name="led_num"/></p>

  13.    <p align="center">请输入控制led的动作 <input type="text" name="led_state"/></p>

  14.    <p align="center"><input type="submit" value="sure"/>        

  15.          <input type="reset" value="back"/>

  16.    </p>

  17.   </form>

  18.  </body>

  19. </html>

把led.html放到我们开发板上的/www/中,通过浏览器访问:

http://192.168.1.8/led.html

实际效果如:

下面我们编写我们的CGI程序。

「方法一:借助环境变量来获取相关信息。」

  1. #include<stdio.h>

  2. #include <stdlib.h>

  3. #include <sys/types.h>

  4. #include <sys/stat.h>

  5. #include <fcntl.h>

  6. #include <unistd.h>

  7. #include <sys/ioctl.h>

  8. #include <linux/limits.h>

  9. #include <errno.h>

  10. int main(int argc, const char *argv[])

  11. {

  12.  int led_num,led_state;

  13.  char *data;

  14.     printf("Content-type: text/html;charset=utf-8\n\n");

  15.     printf("<html>\n");

  16.     printf("<head><title>cgi control led web</title></head>\n");

  17.     printf("<body>\n");

  18.     printf("<p>led is setted successful! you can watch the led's change</p>\n");

  19.     printf("<p><a herf=led.html><button>get back</button></a></p>\n");

  20.     printf("</body>\n");

  21.     data = getenv("QUERY_STRING");   //getenv()读取环境变量的当前值的函数 

  22.     if(sscanf(data, "led_num=%d&led_state=%d", &led_num, &led_state) != 2)

  23.     {   //利用sscnaf()函数的特点将环境变量分别提取出led_num和led_state这两个值

  24.         printf("<p>please input right"); 

  25.         printf("</p>");

  26.     } 

  27.     printf("<p>led_num = %d,led_state =  %d</p>",  led_num,  led_state);

  28.     if(led_num < 2 || led_num > 5) 

  29.     { 

  30.         printf("<p>Please input 2<=led_num<=5!"); 

  31.         printf("</p>");

  32.     } 

  33.     if(led_state>1)

  34.     {

  35.         printf("<p>Please input 0<=led_state<=1!"); 

  36.         printf("</p>"); 

  37.     }

  38.     printf("</html>\n");

  39.  return 0;

  40. }

这里的getenv函数就是获取环境变量当前值的函数,其中,各环境变量的意义可以参考上面的表。这里的QUERY_STRING环境变量就是采用GET时所传输的信息。在这个例子中就是:

led_num=3&led_state=1

上面的CGI程序把收到网页发送过来的led_num及led_state的至再使用printf返回至网页中显示。从网页中也可以看到相关信息,整个流程也就打通了。

「方法二:使用CGIC库。」

第二种方法我们使用CGIC库来编写我们的CGI程序。首先我们需要下载CGIC库,下载地址如:

https://github.com/boutell/cgic

下载得到:

整个包的内容不多,就几个文件。其中:

  • cgic.c   函数库。

  • capture.c  一个很简单的CGI例子。仅仅输出两行提示文字。

  • cgictest.c 一个演示读取form表单数据的CGI例子。

因为我们的CGI程序要运行在我们的arm板中,当然要交叉编译,首先需要修改Makefile文件,需要修改如下几处内容:

「第①个修改点:」

  1. CFLAGS=-g -Wall

  2. CC=gcc

  3. AR=ar

  4. RANLIB=ranlib

  5. LIBS=-L./ -lcgic

修改为:

  1. CFLAGS=-g -Wall

  2. CC=arm-linux-gnueabihf-gcc

  3. AR=arm-linux-gnueabihf-ar

  4. RANLIB=arm-linux-gnueabihf-ranlib

  5. LIBS=-L./ -lcgic

「第②个修改点:」

gcc cgictest.o -o cgictest.cgi ${LIBS}

修改为:

arm-linux-gnueabihf-gcc cgictest.o -o cgictest.cgi ${LIBS}

「第③个修改点:」

gcc capture.o -o capture ${LIBS}

修改为:

arm-linux-gnueabihf-gcc cgictest.o -o cgictest.cgi ${LIBS} capture.o -o capture ${LIBS}

「第④个修改点:」

gcc -D UNIT_TEST=1 cgic.c -o cgicunittest

修改为:

arm-linux-gnueabihf-gcc -D UNIT_TEST=1 cgic.c -o cgicunittest

这里我的编译器是arm-linux-gnueabihf-gcc,大家需根据自己的实际环境进行修改。

make编译得到cgictest.cgi及capture程序,大家可以自行传到板子上进行测试。下面我们在这个文件夹下新曾一个led.c文件,用于编写我们的leg cgi程序:

  1. #include "cgic.h"

  2. // cgic程序以cgiMain作为入口点, cgic的函数库会自动把cgiMain连接到相应的main()上去

  3. int cgiMain() 

  4.  char led_num[10];  

  5.  char led_state[10]; 

  6.  cgiFormString("led_num",  led_num, 10);    // 从表单中的led_num字段获取值存入到led_num

  7.  cgiFormString("led_state",  led_state, 10);// 从表单中的led_state字段获取值存入到led_state

  8.  cgiHeaderContentType("text/html");         // 设定输出的内容格式 这里我们要输出HTML

  9.  fprintf(cgiOut,"<title>LED Test</title>");   

  10.  fprintf(cgiOut,"<p>recv from arm:</p>");

  11.  fprintf(cgiOut,"led_num: %s", led_num);

  12.  fprintf(cgiOut,"<br>");                    

  13.  fprintf(cgiOut,"led_state: %s", led_state);

  14.  return 0;

  15. }

修改Makefile,把我们的led.c也加入编译。编译得到led.cgi,传入到板子中的/www/cgi-bin文件夹下:

浏览器访问:

http://192.168.1.8/led.html

led cgi程序可以正确地解析led_num及led_state。

这个目录架构可以根据目标开发板上的文件系统的实际情况进行调整,具体的调整方法就是通过去修改boa.conf配置文件去实现的。

boa.conf配置文件

#boa服务器监听客户端所用的端口
Port 80

#服务器绑定的IP地址,注释掉表示绑定到INADDR_ANY,适配服务器所有的IP
#Listen 192.68.0.5

#服务器运行的用户和组
#User 0 表示root用户
User 0
#Group o 表示root组
Group 0

#当服务器发生问题时发送报警的email地址
#ServerAdmin root@localhost

#错误日志文件存储的位置,可以根据部署在目标开发板的实际位置来调整。
ErrorLog /var/log/boa/error_log
#访问日志文件的位置,可以根据部署在目标开发板的实际位置来调整。
AccessLog /var/log/boa/access_log

#是否使用本地时间。如果没注释掉,则使用本地时间。注释掉则使用UTC时间
#UseLocaltime

#是否记录CGI运行信息
#VerboseCGILogs

#服务器名字
ServerName www.your.org.here

#是否启动虚拟主机功能,即设备可以有多个网络接口,每个接口都可以拥有一个虚拟的Web服务器
#VirtualHost

#非常重要,boa服务器的根目录,也就是通过浏览器输入的服务器IP地址所能访问到的目录,html文件,cgi程序都部署在这个目录下
DocumentRoot /var/www

#如果收到一个用户请求的话,在用户主目录后再增加的目录名
UserDir public_html

#HTML目录索引的文件名
DirectoryIndex index.html

#一个连接所允许的HTTP持续作用请求最大数目
KeepAliveMax 1000

#HTTP持续作用中服务器在两次请求之间等待的时间数,以秒为单位,超时将关闭连接
KeepAliveTimeout 10

#指明mime.types文件位置
MimeTypes /etc/mime.types

#文件扩展名没有或未知的话,使用的缺省MIME类型
DefaultType text/plain

#提供CGI程序的PATH环境变量值
CGIPath /bin:/usr/bin:/usr/local/bin

#为路径加上别名
Alias /doc /usr/doc

#指明CGI脚本的虚拟路径对应的实际路径
ScriptAlias /cgi-bin/ /var/www/cgi-bin/

通过网页控制嵌入式设备相关推荐

  1. 网页里如何嵌入服务器控制,在嵌入式设备中实现Web动态服务与Web控制的实现思路...

    随着网络技术的不断发展,嵌入式系统将不断地和网络相结合.嵌入式Web技术是计算机领域研究的热点,其优点是开发成本低.通用性强,能运行在8位或16 位MCU环境中,其丰富的Web用户图形界面使得嵌入式设 ...

  2. esp8266烧录Html文件,实现内置网页控制设备!

    代码地址如下: http://www.demodashi.com/demo/14321.html 一.前言: 这个月也快结束了,时间真快,我服务器知识自学依然在路途中,这几天听到热点网页配置esp82 ...

  3. esp32语音控制_乐鑫发布针对物联网嵌入式设备AI语音麦克风阵列开发板

    乐鑫信息科技(股票代码:688018)发布 AI 语音麦克风阵列开发板 ESP32-Korvo .这是一款针对物联网嵌入式设备的 AI 语音开发板,基于乐鑫的旗舰芯片 ESP32,搭载多麦克风阵列,能 ...

  4. Esp8266 进阶之路32【高级篇】当esp8266遇到 Html,该怎么内置网页控制设备,理清内置网页的实现过程,实现无需路由器手机也可以控制esp8266。(附带固件)

    本系列博客学习由非官方人员 半颗心脏 潜心所力所写,仅仅做个人技术交流分享,不做任何商业用途.如有不对之处,请留言,本人及时更改. 1. Esp8266之 搭建开发环境,开始一个"hello ...

  5. 嵌入式设备web服务器比较

    现在在嵌入式设备中所使用的web服务器主要有:boa.thttpd.mini_httpd.shttpd.lighttpd.goaheand.appweb和apache等. Boa 1.介绍 Boa诞生 ...

  6. 嵌入式设备的容器化App

                                                                                                        ...

  7. 嵌入式设备web服务器

    操作系统:ubuntu10.04 前言:     为了提高对设备的易操作性,很多设备中提供pc机直接通过浏览器操作设备的功能.这就需要在设备中实现web服务器.     现在在嵌入式设备中所使用的we ...

  8. CubeMX配置STM32实现httpd服务器CGI功能并使用网页控制STM32单片机(四)

    CubeMX配置STM32实现httpd服务器CGI功能并使用网页控制STM32单片机 引言 CubeMX配置HTTPD的CGI功能 实验过程 发现的问题 总结 引言 在前三篇文章中自己介绍了如何配置 ...

  9. 什么是Android - 嵌入式设备编程的历史

    -第一章 暂时可以这样说,传统的桌面 应用程序 开发者已经被惯坏了.这个不是说桌面应用程序开发比其他开发很简单.总之作为桌面应用程序开发者,我们已经有能力按照我们的想法创造出各种应用程序.包括我自己, ...

最新文章

  1. 现半透明的popupwindow
  2. VTK:图片之ImageMandelbrotSource
  3. Zabbix Maintenance 维护周期
  4. win7系统0x0000001a蓝屏代码怎么办
  5. 分布式缓存服务是什么?
  6. 海园帮忙写的JQUERY功能,实现了我们想要的,我觉得有点屌哟~~
  7. 数据中心交换机芯片学习总结
  8. 自然语言处理(3)——形式语言与自动机
  9. JanusGraph概述
  10. MySQL统计每月数量并计算同比增长率
  11. TCP和UDP对比的优势和劣势
  12. 一网打尽寄存器、SRAM、DRAM、主存、磁盘、闪存、固态硬盘SSD
  13. 伺服驱动器原点回归有一点让人很迷惑
  14. Access数据类型和SQL数据类型
  15. java中英文切换_中英文切换
  16. Drop Shipment PO以及Replenishment PO有何异同?
  17. 2021年 第一个月的福利 程序员免费资源!
  18. 2015华为校园招聘机试题<一>
  19. 学计算机加数模社团,数学建模社团简介
  20. 储能系统下垂控制,输出电流按虚拟电阻比例分配,并补偿有下垂系数带来的母线压降

热门文章

  1. nginx 过滤ip
  2. 天才小毒妃 第907章 底气,狮子大开口
  3. flex effect
  4. DIVA靶场测试APP客户端不规范项(一)
  5. 语义分割介绍和FCN
  6. 解决微星主板点击睡眠后电脑死机
  7. [分享] Hyper-V 安装Win7激活后黑屏
  8. DataFrame-索引与切片
  9. 学习用Photoshop来设计简单的网页
  10. 【随手记】Oracle存储过程报错 Compilation errors for PACKAGE BODY