Perl的普及与互联网的蓬勃发展有直接的关系。在互联网发展的早期,人们发现仅仅使用静态的HTML文档不能生成有效的交互式环境,于是引进了公用网关接口(CGI)的概念。Perl强大的功能和容易扩充的特性使得它成为开发CGI应用最自然的选择,并由此迅速地成为CGI脚本的首选语言。CGI本身并非十全十美。但由于得到了众多开发商的青睐,CGI的应用至今仍然十分广泛,而且没有迹象表明在近期会“退休”。
CGI::XMLApplication提供了一个基于XML、可以作为传统CGI脚本的模块。典型的CGI::XMLApplication脚本包括三部分:一个很小的提供对该应用程序访问支持的可执行脚本、实现各种管理者方法的逻辑模块、根据应用状态可能有一个或多个XSLT样式表,XSLT样式表能够将模块返回的结果转化成浏览器可以向用户显示的格式。
下面我们通过例子来简要地介绍CGI::XMLApplication的应用。
例1:CGI XSLT网关
CGI::XMLApplication假定,参与一个项目的设计和开发人员使用XSLT样式表分离应用的逻辑和表示,这样可以使这种分离显得非常直接,也不会对项目带来影响。开发人员只要能够使setStylesheet返回符合当前应用状态的XSLT样式表的位置即可。应用建立的DOM树的转换、XSLT参数向转换引擎的传递、转换后内容向浏览器的传输对用户而言都是透明的。
为了重点说明这种分离,我们的第一个例子不是传统意义上的Web应用,而是一个通用的XSLT网关,它可以添加到服务器的cgi-bin中,将整个XML内容的目录树转化为符合请求的浏览器的格式,而这一切对于用户、样式表和文档的作者而言也都是透明的。
第一步是建立连接客户端的请求和应用的CGI脚本。我们希望XML文档能够方便地通过URL浏览,并使创建这些文档间的超链接非常直观。因此,我们将创建一个没有扩展名的CGI脚本,以便将它作为URL路径中的一个节点,节点右边的所有内容将在包含XML内容的虚拟文档环境中进行解释。在这种情况下,我们将CGI称作是样式表选择者。
use strict;
use lib '/path/to/secure/webapp/libs';
use XSLGateway;
use CGI qw(:standard);my $q = CGI-new();
my %context = ();
my $gateway_name = 'stylechooser';
在加载合适的模块和设置一些在整个脚本范围内有效的变量后,我们开始向被传递给处理该应用逻辑的类的%context中添加一些域。在这个应用软件中,我们只传输要求的指向脚本文件路径右边的URL(REQUEST条目)和包含有存储在查询参数style中的数据的STYLE关健字。
$context{REQUEST} = $q-url(-path = 1);
$context{REQUEST} =~ s/^$gateway_name/?//;
$context{REQUEST} ||= 'index.xml';
$context{STYLE} = $q-param('style') if $q-param('style');
最后,我们创建了XSLGateway逻辑类的一个实例,并通过调用其run方法处理请求,将%context作为唯一的参数。
my $app = XSLGateway-new();
$app-run(%context);
CGI脚本就完成了。下面我们创建完成大部分工作的XSLGateway模块:
package XSLGateway;
use strict;
use vars qw(@ISA);
use CGI::XMLApplication;
use XML::LibXML;
@ISA = qw(CGI::XMLApplication);
象我在简介中提到的那样,CGI::XMLApplication通过事件调用起作用:应用程序类中一个给定的方法的执行依赖于一个指定域的输入(一般情况下是用来提交表格的按钮的名字。),必须执行二种调用方法:selectStylesheet和requestDOM方法。
selectStylesheet返回有关的XSLT样式表的全文件系统路径。为了简单起见,我们假定样式表将保存在一个单一的目录中。我们可以通过$context-{STYLE}域提供其他的样式表,从而增加系统的灵活性。
sub selectStylesheet {
my $self = shift;
my $context = shift;
my $style = $context-{STYLE} || 'default';
my $style_path = '/opt/www/htdocs/stylesheets/';
return $style_path . $style . '.xsl';
}
下一步,我们需要创建requestDOM方法,该方法将返回被传输的XML文档的XML::LibXML DOM表达式。由于我们的网关只适用于静态文件,我们需要使用XML::LibXML对文档进行解析,并返回结果树。
sub requestDOM {
my $self = shift;
my $context = shift;
my $xml_file = $context-{REQUEST} || 'index.xml';
my $doc_path = '/opt/www/htdocs/xmldocs/';
my $requested_doc = $doc_path . $xml_file;
my $parser = XML::LibXML-new;
my $doc = $parser-parse_file($requested_doc);
return $doc;
}
至此,我们的CGI脚本已经可以安全地在服务器的cgi-bin目录中安全地运行了,并在一些适当的目录中上载一些XML文档和一个或二个XSLT样式表。下面我们就可以开始检验我们的工作成果了。对http://localhost/cgi-bin/stylechooser/mydocs/somefile.xml的请求将会使互联网服务器从/opt/www/htdocs/xmldocs/
# 事件的注册和事件调用
sub registerEvents {
return qw( order_confirm order_send );
}
sub event_order_confirm {
my ($self, $context) = @_;
$context-{SCREENSTYLE} = 'order_confirm.xsl';
}
sub event_order_send {
my ($self, $context) = @_;
$context-{SCREENSTYLE} = 'order_send.xsl';
}
如果没有请求执行其他的事件,则缺省地执行event_default。在本例中,我们只使用它将SCREENSTYLE域设定为一个合适的值。
sub event_default {
my ($self, $context) = @_;
$context-{SCREENSTYLE} = 'order_default.xsl';
}
每次请求都会执行event_init方法,而且总是在其他方法之前执行它,这使得它非常适合对应用中被其他事件使用的部分进行初始化。在本例中,我们使用它返回利用fetch_recordset()方法从数据库中获取的产品信息的、最初的DOM树。
sub event_init {
my ($self, $context) = @_;
$context-{DOMTREE} = $self-fetch_recordset();
}
state-handler方法完成后,我们需要执行必需的selectStylesheet和requestDOM方法。
与在第一个例子中一样,我们假设所有的应用的样式表都存储在服务器上相同的目录中。我们所需要作的是返回$context-{SCREENSTYLE}的值所指定的路线,并添加到末尾。
# app config and helpers
sub selectStylesheet {
my ($self, $context) = @_;
my $style = $context-{SCREENSTYLE};
my $style_path = '/opt/www/htdocs/stylesheets/cart/';
return $style_path . $style;
}
在研究requestDOM处理程序之前,我们先来详细地研究fetch_recordset helper方法。
需要记住的是,我们要做的工作是从一个关系数据库中选择所订购产品的有关信息,但传递给XSLT处理器的数据必须是DOM树。在本例中,我们不通过编程的方法,而是利用XML::Generator::DBI,它能够从执行SQL SELECT语句得到的数据中生成SAX数据。创建要求的DOM树就是建立XML::LibXML::SAX::Builder(它从SAX事件中创建XML::LibXML DOM树)的实例。