38、Ruby CGI 编程

Ruby 不仅可以编写自己的 SMTP 服务器,FTP程序,或 Web 服务器,而且还可以开发 CGI 程序

接下来,我们将花几节时间来学习 Ruby 的 CGI 开发

网页浏览过程

为了更好的了解 CGI 是如何工作的,我们可以从在网页上点击一个链接或 URL 来了解网页请求响应流程:

1、 打开浏览器访问URL并连接到HTTPweb服务器;
2、 WEB服务器接收到请求信息后会解析URL,并查找访问的文件在服务器上是否存在,如果存在返回文件的内容,否则返回错误信息;
3、 浏览器从服务器上接收信息,并显示接收的文件或者错误信息;

CGI程序可以是 Ruby 脚本,Python 脚本,PERL 脚本,SHELL 脚本,C 或者 C++ 程序等

CGI 架构图

 

Web服务器支持及配置

在开发CGI 程序前,我们先要配置 WEB 服务器支持 CGI 及 CGI 的处理程序

Apache 支持 CGI 配置:

设置好CGI目录:

ScriptAlias /cgi-bin/ /var/www/cgi-bin/

所有的HTTP 服务器执行 CGI 程序都保存在一个预先配置的目录。 这个目录被称为 CGI 目录,并按照惯例,它被命名为 /var/www/cgi-bin 目录

CGI程序的扩展名为 .cgi,Ruby 也可以使用 .rb 扩展名

默认情况下,Linux 服务器配置运行的 cgi-bin 目录中为 /var/www

如果想指定其它运行 CGI 脚本的目录,可以修改 httpd.conf 配置文件

<Directory "/var/www/cgi-bin">
   AllowOverride None
   Options +ExecCGI
   Order allow,deny
   Allow from all
</Directory>

在AddHandler 中添加 .rb 后缀,这样我们就可以访问 .rb 结尾的 Ruby 脚本文件

AddHandler cgi-script .cgi .pl .rb

编写 CGI 脚本

一个简单的 Ruby CGI 脚本如下所示

hello.cgi

# !/usr/bin/ruby -w
# -*- encoding:utf-8 -*-
# filename: hello.cgi
# author: DDKK.COM 弟弟快看,程序员编程资料站(www.ddkk.com)
# Copyright © 2015-2065 www.ddkk.com. All rights reserved.
puts "Content-type: text/html\n\n"
puts "<html><body>Hello World</body></html>"

将以上代码保持到 hello.cgi 文件中,上传到服务器并赋予足够权限,即可作为 CGI 脚本执行

假设我们站点的地址为: http://localhost:8080/ 那么可以通过 http://localhost:8080/hello.cgi 访问 CGI 程序,输出结果为: "Hello World"

浏览器访问该网址后,Web 服务器会在站点目录下找到 hello.cgi 文件,然后通过 Ruby 解析器来解析脚本代码并访问 HTML 文档

使用 cgi.rb

Ruby 可以调用 CGI 库来编写更复杂的 CGI 脚本

print_header.cgi

# !/usr/bin/ruby -w
# -*- encoding:utf-8 -*-
# filename: print_header.cgi
# author: DDKK.COM 弟弟快看,程序员编程资料站(www.ddkk.com)
# Copyright © 2015-2065 www.ddkk.com. All rights reserved.
require 'cgi'

cgi = CGI.new
puts cgi.header
puts "<html><body>Hello World</body></html>"

上面的代码,创建了 CGI 对象并打印头部信息

表单处理

CGI程序可以通过两种方式获取表单提交的数据

例如URL: /cgi-bin/form_get.cgi?FirstName=Li&LastName=Hong

可以使用 CGI#[] 来直接获取参数 FirstName 和 LastName

# !/usr/bin/ruby -w
# -*- encoding:utf-8 -*-
# filename: form_get.cgi
# author: DDKK.COM 弟弟快看,程序员编程资料站(www.ddkk.com)
# Copyright © 2015-2065 www.ddkk.com. All rights reserved.

require 'cgi'
cgi = CGI.new
cgi['FirstName'] # =>  ["Li"]
cgi['LastName']  # =>  ["Hong"]

另外一种获取表单数据的方法:

# !/usr/bin/ruby -w
# -*- encoding:utf-8 -*-
# filename: form_get.cgi
# author: DDKK.COM 弟弟快看,程序员编程资料站(www.ddkk.com)
# Copyright © 2015-2065 www.ddkk.com. All rights reserved.

require 'cgi'
cgi = CGI.new
h = cgi.params  # =>  {"FirstName"=>["Li"],"LastName"=>["Hong"]}
h['FirstName']  # =>  ["Li"]
h['LastName']   # =>  ["Hong"]

以下代码用于检索所有的表单键值:

# !/usr/bin/ruby -w
# -*- encoding:utf-8 -*-
# filename: form_get.rb
# author: DDKK.COM 弟弟快看,程序员编程资料站(www.ddkk.com)
# Copyright © 2015-2065 www.ddkk.com. All rights reserved.

require 'cgi'
cgi = CGI.new
cgi.keys         # =>  ["FirstName", "LastName"]

如果表单包含了多个相同名称的字段,则该相同字段的值将保存在数组中

以下范例中,指定表单中三个相同的字段 "name",值分别为 "XiaoHong", "XiaoMing" 和 "XiaoZhang"

#!/usr/bin/ruby

require 'cgi'
cgi = CGI.new
cgi['name']        # => "XiaoHong"
cgi.params['name'] # => ["XiaoHong", "XiaoMing", "XiaoZhang"]
cgi.keys           # => ["name"]
cgi.params         # => {"name"=>["XiaoHong", "XiaoMing", "XiaoZhang"]}

注意: Ruby 会自动判断 GET 和 POST 方法,所以无需对两种方法区别对待

下面是相关的 HTML 代码

form_get.html

<html>
<body>
<form method="POST" action="/form_get.cgi">
<p>First Name :<input type="text" name="FirstName" value="" /></p>
<p>Last Name :<input type="text" name="LastName" value="" /></p>
<p><input type="submit" value="Submit Data" /></p>
</form>
</body>
</html>

创建 Form 表单和 HTML

Ruby CGI 模块包含了大量的方法来创建 HTML,每个HTML标签都有相对应的方法。

在使用这些方法前,必须通过 CGI.new 来创建 CGI 对象

为了使标签的嵌套更加的简单,这些方法将内容作为了代码块,代码块将返回字符串作为标签的内容

# !/usr/bin/ruby -w
# -*- encoding:utf-8 -*-
# filename: main.rb
# author: DDKK.COM 弟弟快看,程序员编程资料站(www.ddkk.com)
# Copyright © 2015-2065 www.ddkk.com. All rights reserved.
require "cgi"
cgi = CGI.new("html5")
cgi.out{
   cgi.html{
      cgi.head{ "\n"+cgi.title{"Hello World"} } +
      cgi.body{ "\n"+
         cgi.form{"\n"+
            cgi.hr +
            cgi.h1 { "A Form: " } + "\n"+
            cgi.textarea("get_text") +"\n"+
            cgi.br +
            cgi.submit
         }
      }
   }
}

字符串转义

处理URL 中的参数或者 HTML 表单数据时,需要对指定的特殊字符进行转义,如:引号("),反斜杠(/)

Ruby CGI 对象提供了 CGI.escape 和 CGI.unescape 方法来处理特殊字符

# !/usr/bin/ruby -w
# -*- encoding:utf-8 -*-
# filename: main.rb
# author: DDKK.COM 弟弟快看,程序员编程资料站(www.ddkk.com)
# Copyright © 2015-2065 www.ddkk.com. All rights reserved.

require 'cgi'
puts CGI.escape("Li XiaoMing /A Sweet & Sour Girl")

运行以上范例,输出结果如下

$ ruby main.rb
Li+XiaoMing+%2FA+Sweet+%26+Sour+Girl

另一组范例

# !/usr/bin/ruby -w
# -*- encoding:utf-8 -*-
# filename: main.rb
# author: DDKK.COM 弟弟快看,程序员编程资料站(www.ddkk.com)
# Copyright © 2015-2065 www.ddkk.com. All rights reserved.
require 'cgi'
puts CGI.escapeHTML("<h1>Li XiaoMing /A Sweet & Sour Girl</h1>")

运行以上脚本,输出结果如下

$ ruby main.rb
<h1>Li XiaoMing /A Sweet & Sour Girl</h1>

CGI 类中常用的方法

如果想知道 Ruby 中完整的 CGI 类的方法,请移步 Ruby CGI

Cookies 和 Sessions

  • Ruby CGI Cookies - 如何处理 CGI Cookies
  • Ruby CGI Sessions - 如何处理 CGI sessions