4.3.4 server对象的错误处理
asp没有错误处理机制一直受到批评。
在vbscript中,有一个on error resume next语句,它使脚本解释器忽略运行期错误并继续脚本代码的执行。接着该脚本可以检查err.number属性的值,判别是否出现了错误。如果出现错误,返回一个非零值。在asp 3.0中,也可以使用on error goto 0“转回到”缺省的错误处理。在asp 2.0中实际也进行这种处理,但是没有相应文档说明。
在jscript中,有一个新的错误处理功能:c语言风格的try和catch语句。然而所有的这些错误处理技术都不是由asp或iis实现的,而是由asp使用的脚本引擎实现的。
第7章专门讨论脚本和脚本引擎涉及到的调试和错误处理技术。
同时,asp和iis的开发小组已经增加了一个新的功能,用于在asp网页中进行错误处理。这分为两个部分:iis错误页面的配置及使用asp的一个新的方法和对象。
1. server对象的getlasterror方法
在asp 3.0中,server对象有一个名为getlasterror的新方法。与vbscript的err对象不同,不能为查看是否出现了错误而随时调用该方法,只能在一个asp定制的错误网页中使用。如果像对err对象进行操作那样,通过关闭缺省的错误处理(用on error resume next语句)来使用,则getlasterror方法不能访问错误的详细数据。
getlasterror方法要做的事情是提供更多的有关错误源和错误原因的信息。getlasterror方法创建并返回一个对象的引用,该对象是一个名为asperror的新对象。这个对象具有一系列的属性,这些属性返回有关在getlasterror方法调用之前出现的最新错误的信息。
2. asperror对象的属性
asperror对象提供了九个属性说明所出现的错误的性质和错误源,并返回引发错误的实际代码,其属性及说明如表4-4所示:
表4-4 asperror对象的属性及说明
属 性
说 明
aspcode
整型。由asp/iis产生的错误号,例如0x800a009
aspdescription
字符串型。如果这个错误是与asp相关的错误,这个属性是错误的详细说明
category
字符串型。错误源,即asp内部脚本语言、或一个对象
column
整型。产生错误的文件中的字符位置
description
字符串型。错误的简短说明
file
字符串型。错误出现时正在处理的文件的名称
line
整型。产生错误的文件中的行号
number
整型。一个标准的com错误代码
source
字符串型。引发错误的行的实际代码
3. 配置“单个网页”错误处理
在iis中“不可思议”地出现一个错误(例如404 not found)时,页面看起来像是从服务器返回给客户端的一个错误信息页面,但实际上并不是这样。它们是普通的html网页,在对一个错误进行响应时被下载并且发送给客户端。这些网页通常称为定制的错误网页(custom error page)。
然而,错误网页作为iis的缺省安装部分,可根据要求定制。事实上,也可以在iis的早期版本中建立定制的错误网页。
在iis 4.0中,可以为每种不同类型的http协议或服务器错误指定一个定制的错误网页,为服务器上任意的web网站中的每个目录建立一个定制的错误信息网页。
(1) iis缺省的错误网页
由iis提供的缺省错误页面放在web服务器的winnt\help目录中。在windows 2000中的iis 5.0的环境下,该页面放在winnt\help\iishelp\common目录下,如图4-17所示:
图4-17 缺省的错误页面位置
可在浏览器中打开这些文件查看结果,或者在文本编辑器中查看html源程序和脚本代码。当一个404错误出现时,使用的页面是404b.htm,这个文件包含一个客户端脚本代码部分,它获得当前文档的url(从document对象的url属性中检索)并在该页面中显示:
<tr>
<td width="400" colspan="2"> <font style="color:000000; font: 9pt/11pt 宋体">您正在搜索的网页可能已经删除、更名或暂时不可用。</font></td>
</tr>
<tr>
<td width="400" colspan="2"> <font style="color:000000; font: 9pt/11pt 宋体">
<hr color="#c0c0c0" noshade>
<p>请尝试下列操作:</p>
<ul>
<li>如果您在“地址”栏中键入了网页地址,请检查其拼写是否正确。<br>
</li>
<li>打开 <script>
<!–
if (!((window.navigator.useragent.indexof("msie") > 0) && (window.navigator.appversion.charat(0) == "2")))
{
homepage();
}
//–>
</script> 主页,寻找指向所需信息的链接。</li>
…
<script>
function homepage(){
<!–
docurl = document.url;
protocolindex=docurl.indexof("://",4);
serverindex=docurl.indexof("/",protocolindex + 3);
beginurl=docurl.indexof("#",1) + 1;
urlresult=docurl.substring(beginurl,serverindex);
displayresult=docurl.substring(protocolindex + 3 ,serverindex);
document.write(<a href=" + urlresult + "> + displayresult + "</a>");
}
//–>
</script>
这会产生你经常看到的页面,如图4-18所示:
图4-18 产生404错误时的页面
(2) iis中错误网页的映射
当iis检测到一个错误时,会把相应的错误页面传送给客户端。如何判别应该向客户端发送那一个页面?很明显,网页的名字应具有解决这个问题的一些信息,但事实上文件名是不重要的。错误和错误网页文件之间的映射关系是在每个目录的properties对话框的custom errors选项卡中决定的。
在internet services manager中,在想编辑映射关系的目录上单击右键,并选择properties。如果对示例文件进行设置,在chapter04目录中选择server子目录,如图4-19所示:
图4-19 设置属性时的页面屏幕
properties对话框的custom errors选项卡在iis安装时(除非已经进行过修改)设置了缺省映射关系的列表,如图4-20所示:
图4-20 映射关系的列表
靠近该列表的底部是http错误500:100的一个条目。类型500错误是由asp产生的,可以从中看出一些错误已经与错误网页建立了映射关系。这些错误都是一般性的错误,比如“invalid application”、“server shutting down”等等。然而,如果asp载入包含语法错误的页面,或者出现一个运行期错误,则出现500:100错误页面。在列表中显示的缺省映射关系表明,在这个目录中的一个文件出现上述错误时,将执行500-100.asp页面。
当一个asp错误出现时,我们所看到的信息不再是一个普通的web网页,而是一个asp web网页(也就是说它具有文件扩展名.asp)。也可以根据需要编辑该映射关系来指向另一个页面。
(3) 指定一个定制的错误网页
单击custom errors选项中的edit properties按钮,打开error mapping properties对话框。在message type下拉列表中选择url,键入自己的定制错误网页的完整虚拟路径,如图4-21所示:
图4-21 指定错误页面的虚拟路径的屏幕
在图4-21中给出的值指向我们创建的与示例网页一起使用的一个定制错误网页。根据你安装示例文件的具体位置,可能要使用不同的路径。
现在无论何时出现一个500:100类型的错误,将打开我们的定制错误页面。message type的其他两个选项是:
· default(缺省):可以简单地输入一个短的文本信息,而不是指定一个发送给客户端的页面。
· file(文件):指定一个http错误网页的物理路径。
在选择file选项时,指定的网页由iis载入,载入的方式与在windows explorer中双击要载入的文件时的方式相同。这意味着asp网页不能使用这个选项,因为在这种情况下不会执行其中的任何脚本。
4. 使用getlasterror方法和asperror对象
配置好iis后,在编辑了错误映射属性的目录内的任一页面上出现一个与asp相关的错误时,都将载入定制错误页面。实际上,现在已经设置了一个正常的脚本错误陷阱,因为在这个目录内的任何一个网页上的asp运行期错误都将触发定制错误页面。
事实上在内部iis通过server.transfer方法进行这种操作,这意味着能够访问正在运行的原网页的全部环境。可以在脚本环境中获取信息,这样可以根据所出现的错误决定要做些什么。在此基础上,可以在定制的错误网页中检索asperror对象,找到引起载入页面出错的错误的所有信息。
在iis 4.0中,编辑错误映射属性要做一些类似的工作。但是只有一般的500错误(“internal server error”)在映射中是可用的。另外,当定制错误网页载入时,不会传送网页的环境,除了提供一个非特定的错误信息外,做其他任何工作都是比较困难的。
在以前例子中已经使用过asp server object示例页面,在其中可以看到asperror对象的详细情况。单击server.getlasterror()对应的按钮,如图4-22所示:
图4-22 查看asperror对象的详细屏幕
这个操作会重新载入该网页,其中的asp脚本查看点击的是哪个按钮。如果是server.getlasterror()对应的名为cmdgeterror的按钮,将执行一些示例代码,这些代码将会产生一个运行期脚本错误。
…
if len(request.form(“cmdgeterror”)) then
dim arrthis(3)
arrthis(4) = “causes an error”
end if
…
因为已对这个目录设置了错误网页映射,即配置为装入定制错误页面,所以当错误出现时,就打开这个页面(通过server.transfer方法在后台不可见地工作),见图4-23所示:
图4-23 定制的错误网页
(1) 示例错误网页代码的功能
定制错误网页显示asperror对象属性的所有值,并通过使用response.status方法,把一个http报头状态消息返回给客户端,指明出现了一个错误。接着使用getlasterror方法获取对asperror对象的一个引用,因此可以访问错误的详细数据:
…
<%
response.status = "500 internal server error"
set objasperror = server.getlasterror()
%>
currently executing the page: <b>show_error.asp</b><p>
<b>error details:</b><br>
asperror.aspcode = <% = objasperror.aspcode %><br>
asperror.number = <% = objasperror.number %> (0x<% = hex(objasperror.number) %>)<br>
asperror.source = <% = server.htmlencode(objasperror.source) %><br>
asperror.category = <% = objasperror.category %><br>
asperror.file = <% = objasperror.file %><br>
asperror.line = <% = objasperror.line %><br>
asperror.column = <% = objasperror.column %><br>
asperror.description = <% = objasperror.description %><br>
asperror.aspdescription = <% = objasperror.aspdescription %>
<form action="<% = request.servervariables("http_referer") %>" method="post">
<input type="submit" name="cmdok" value=" ">
return to the previous page<p>
</form>
值得注意的一点是,如果一个脚本或asp错误出现在定制错误网页中,iis将仅仅返回一个与错误代码500:100对应的一般性消息。这可能是脚本引擎自己的错误消息,或者只是相当简单的消息:“internal server error”。不会再次重新载入定制的错误网页。
包含错误的网页的全部环境将传送给定制错误网页。也就是说,可以使用存储在任何asp内部对象集合或属性中的值。例如,如果检索来自request.servervariables集合的http_referer值,它将反映调用原网页的网页(即在错误出现之前的网页)的url。在服务器把执行转到错误网页时,这个值不会发生变化,并且它将不包含当错误发生时正在执行的网页的url。
同样,script_name值将是包含该错误的网页的名字,而不是错误网页的url。在一个错误网页已经装入时,通过检查浏览器地址栏中的url,可以对此进行确认。但是在原网页的脚本变量中存储的值,在定制的错误网页中都是不可用的。
如果原asp网页正在一个事务内运行,即在网页的最前面包含有一个<% @transaction=”…” %>指令,也应该确定是否需要在网页中采取一些方法,以退出该事务。例如可以调用内置objectcontext对象的setabort方法:
objectcontext.setabort ‘fail the transaction if an asp error occurs
在本书的后面将介绍与事务的相关全部内容。
(2) 使用asperror对象的属性
关于使用asperror对象的属性,有以下几点值得注意的:
· 即使没有出现错误,number属性应该一直有一个值。如果asp网页调用getlasterror方法时没有错误出现,该属性的值是0。通常情况下,对asp脚本的运行期错误,number属性返回十六进制的值“0x800a0000”,加上标准的脚本引擎错误代码。例如,前面的例子对“subscript out of range”错误的返回值为“0x800a0009”,因为vbscript对该类型错误的错误代码是“9”。
· 当出现已经过一个错误时,category和description属性将一直有一个值。
· apscode属性的值由iis产生,对大多数脚本错误将为空。更多情况下,涉及外部组件使用出错时有相应的值。
· aspdescription属性的值由asp预处理程序产生,而不是由当前正在使用的脚本引擎产生的,并且对大多数脚本错误而言将是空的。更多情况下,对诸如对asp内置对象调用无效的方法的错误有相应的值。
· file、source、line和column属性仅在错误出现时,并且在错误的详细数据是可用的情况下才能进行设置。对一个运行期错误,file和line属性通常是有效的,但是column属性经常返回-1。当错误是一个阻止页面被asp处理的语法错误,才返回source属性。一般在这些情况下,line和column属性是有效的。如果把source属性的值写到页面,明智的办法是先将该值传给htmlencode,以防在其含有非法的html字符。在本章的后面将详细地讨论htmlencode方法。
