实用 | 10分钟教你通过网页点灯

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

 [[414593]]

大家好,我是ZhengN。

上一篇我们已经基于我们的开发板搭好了一个基于boa的web服务器:10分钟教你搭建一个嵌入式web服务器。这一篇接着分享如何使用网页来点灯。

网页点灯的整个流程大致为:

从上到下,前面两部分我们就是我们上一篇做的实验,已经打通了。而第三部分就是本篇笔记的重点,我们要点灯,自然需要写一个程序来解析数据及进行点灯操作,即这里的CGI程序。

什么是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.  
  3. int main() 
  4.  { 
  5.     printf("Content-type: text/html\n\n") ;  // 这一句是必须的,设定输出到HTML 
  6.     printf("Hello CGI\n") ; 
  7.     return 0; 

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

  1. arm-linux-gnueabihf-gcc test.c -o test.cgi 

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

  1. scp test.cgi root@192.168.1.8:/www/cgi-bin 

修改权限;

  1. chmod 777 test.cgi 

浏览器进行访问:

  1. http://192.168.1.8/www/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/中,通过浏览器访问:

  1. 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.  
  11. int main(int argc, const char *argv[]) 
  12.  int led_num,led_state; 
  13.  char *data; 
  14.  
  15.     printf("Content-type: text/html;charset=utf-8\n\n"); 
  16.     printf("<html>\n"); 
  17.     printf("<head><title>cgi control led web</title></head>\n"); 
  18.     printf("<body>\n"); 
  19.     printf("<p>led is setted successful! you can watch the led's change</p>\n"); 
  20.     printf("<p><a herf=led.html><button>get back</button></a></p>\n"); 
  21.     printf("</body>\n"); 
  22.  
  23.     data = getenv("QUERY_STRING");   //getenv()读取环境变量的当前值的函数  
  24.  
  25.     if(sscanf(data, "led_num=%d&led_state=%d", &led_num, &led_state) != 2) 
  26.     {   //利用sscnaf()函数的特点将环境变量分别提取出led_num和led_state这两个值 
  27.         printf("<p>please input right");  
  28.         printf("</p>"); 
  29.     }  
  30.     printf("<p>led_num = %d,led_state =  %d</p>",  led_num,  led_state); 
  31.     if(led_num < 2 || led_num > 5)  
  32.     {  
  33.         printf("<p>Please input 2<=led_num<=5!");  
  34.         printf("</p>"); 
  35.     }  
  36.     if(led_state>1) 
  37.     { 
  38.         printf("<p>Please input 0<=led_state<=1!");  
  39.         printf("</p>");  
  40.     } 
  41.  
  42.     printf("</html>\n"); 
  43.  
  44.  return 0; 

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

  1. 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 

「第②个修改点:」

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

修改为:

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

「第③个修改点:」

  1. gcc capture.o -o capture ${LIBS} 

修改为:

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

「第④个修改点:」

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

修改为:

  1. 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.  
  3. // cgic程序以cgiMain作为入口点, cgic的函数库会自动把cgiMain连接到相应的main()上去 
  4. int cgiMain()  
  5. {  
  6.  char led_num[10];   
  7.  char led_state[10];  
  8.  cgiFormString("led_num",  led_num, 10);    // 从表单中的led_num字段获取值存入到led_num 
  9.  cgiFormString("led_state",  led_state, 10);// 从表单中的led_state字段获取值存入到led_state 
  10.  cgiHeaderContentType("text/html");         // 设定输出的内容格式 这里我们要输出HTML 
  11.  fprintf(cgiOut,"<title>LED Test</title>");    
  12.  fprintf(cgiOut,"<p>recv from arm:</p>"); 
  13.  fprintf(cgiOut,"led_num: %s", led_num); 
  14.  fprintf(cgiOut,"<br>");                     
  15.  fprintf(cgiOut,"led_state: %s", led_state); 
  16.  return 0; 

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

浏览器访问:

  1. http://192.168.1.8/led.html 

led cgi程序可以正确地解析led_num及led_state。有了这两个信息,我们就可以愉快地点灯了,这里没有加上点灯相关地操作,大家可以自己加上。

以上就是本次关于网页点灯的实战笔记,如有错误,欢迎指出,谢谢!

本文转载自微信公众号「嵌入式大杂烩」,可以通过以下二维码关注。转载本文请联系嵌入式大杂烩公众号。

 

责任编辑:武晓燕 来源: 嵌入式大杂烩
相关推荐

2019-08-09 10:33:36

开发技能代码

2021-07-15 06:43:11

Bash调试脚本

2013-09-13 14:08:01

2021-04-23 09:50:41

topLinux命令

2014-08-08 09:30:04

android scrollview

2020-10-13 18:22:58

DevOps工具开发

2017-10-11 15:17:42

sklearn机器学习pandas

2018-08-30 14:31:28

Linux磁盘LVM

2018-11-28 11:20:53

Python函数式编程编程语言

2022-08-29 15:02:53

远程桌面

2020-10-29 08:28:42

Java NIO异步非阻塞

2021-05-28 07:38:20

内存溢出场景

2021-05-17 20:13:50

数仓操作型数据库

2020-01-22 16:40:48

Java开发代码

2009-10-09 14:45:29

VB程序

2020-07-14 20:03:55

Windows 10Windows微软

2021-07-29 23:29:55

web服务器开发

2019-07-18 16:32:06

Python函数数据

2021-10-27 05:47:53

通信协议协议网络技术

2018-02-01 14:15:00

Python函数
点赞
收藏

51CTO技术栈公众号