博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
IIS的ISAPI接口简介
阅读量:4683 次
发布时间:2019-06-09

本文共 7655 字,大约阅读时间需要 25 分钟。

ISAPI(Internet Server Application Programming Interface)作为一种可用来替代CGI的方法,是由微软和Process软件公司联合提出的Web服务器上的API标准。ISAPI与Web服务器结合紧密,功能强大,能够获得大量的信息,因此利用ISAPI可以开发出灵活高效的Web服务器增强程序。由于ISAPI程序与Web服务器的关系,使得ISAPI接口在安全方面有一定的研究价值。本文主要讨论ISAPI在IIS和VC++ 6.0中的实现。

一、ISAPI接口和CGI接口的不同。

ISAPI程序和CGI程序完成类似的功能,但是实现方法不同。

1、ISAPI程序以DLL形式被Web服务器加载到自己的进程空间中,因此和服务器共用同一个地址空间,且在没有客户请求时可以将其从内存中卸载;而对客户端发来的每个对CGI程序的请求则需要服务器为它单独启动一个进程,这需要耗费大量的时间和内存。当并发的请求数目很大时,使用CGI在效率上不如ISAPI。

2、CGI程序通过环境块和标准输入输出与Web服务器进行通信,而ISAPI程序与服务器结合得更为紧密,与服务器共享同一个进程上下文,主要通过一个参数块与服务器进行交互,可以从服务器那里获得关于当前HTTP连接的大量信息。

ISAPI主要分为ISA和ISAPI Filter两部分。ISA方法相对而言要传统一些,利用一些特殊的链接,指向服务器的作业,供程序开发人员设计一些扩展功能;而ISAPI过滤器则倾向于构造服务器直接调用的模块,提供一种无缝链接部件用于监测直接来自于服务器的HTTP请求。

二、ISA

ISA(Internet Server Application)也可称为ISAPI DLL,其功能和CGI程序的功能直接相对应,使用方法和CGI也类似,由客户端在URL中指定其名称而激活。例如下面的请求将调用服务器的虚拟可执行目录Scripts下的function.dll(ISAPI DLL必须放在服务器的虚拟可执行目录下):

http://www.abc.com/Scripts/function.dll?

ISA和服务器之间的接口主要有两个:GetExtentionVersion( )和HttpExtentionProc( )。任何ISA都必须在其PE文件头的引出表中定义这两个引出函数,以供Web服务器在适当的时候调用。

1、当服务器刚加载ISA时,它会调用ISA提供的GetExtentionVersion( )来获得该ISA所需要的服务器版本,并与自己的版本相比较,以保证版本兼容。函数原型如下: 

 
 

1
None.gif
BOOL WINAPI GetExtentionVersion(HSE_VERSION_INFO 
*
version); 
2
None.giftypedef 
struct
 _HSE_VERSION_INFO 
3
ExpandedBlockStart.gifContractedBlock.gif
dot.gif
4InBlock.gifDWORD dwExtensionVersion; //版本号 
5InBlock.gifCHAR lpszExtensionDesc[HSE_MAX_EXT_DLL_NAME_LEN]; //关于ISA的描述字符串 
6ExpandedBlockEnd.gif}
 HSE_VERSION_INFO, 
*
LPHSE_VERSION_INFO; 

2、ISA的真正入口是HttpExtentionProc( ),它相当于普通C程序的main( )函数,在这个函数中根据不同的客户请求作不同的处理。服务器和HttpExtentionProc( )之间是通过扩展控制块(Extention Control Block)来进行通信的,即ECB中存放入口参数和出口参数,包括服务器提供的几个回调函数的入口地址。函数原型如下:

1
None.gif
DWORD HttpExtensionProc( EXTENSION_CONTROL_BLOCK 
*
pECB ); 

ECB的结构定义如下(IN表示入口参数,OUT表示出口参数):

 1
None.gif
typedef 
struct
 _EXTENSION_CONTROL_BLOCK 
 2
ExpandedBlockStart.gifContractedBlock.gif
dot.gif
 3InBlock.gifDWORD cbSize; //IN,本结构的大小,只读 
 4InBlock.gifDWORD dwVersion //IN,版本号,高16位为主版本号,低16位为次版本号 
 5InBlock.gifHCONN ConnID; //IN,连接句柄,由服务器分配,ISA只能读取该值 
 6InBlock.gifDWORD dwHttpStatusCode; //OUT,当前完成的事务状态 
 7InBlock.gifCHAR lpszLogData[HSE_LOG_BUFFER_LEN]; //OUT,需要写入到日志文件中的内容 
 8InBlock.gifLPSTR lpszMethod; //IN,等价于CGI的环境变量REQUEST_METHOD 
 9InBlock.gifLPSTR lpszQueryString; //IN,等价于环境变量QUERY_STRING 
10InBlock.gifLPSTR lpszPathInfo; //IN,等价于环境变量PATH_INFO 
11InBlock.gifLPSTR lpszPathTranslated; //IN,等价于环境变量PATH_TRANSLATED 
12InBlock.gifDWORD cbTotalBytes; //IN,等价于环境变量CONTENT_LENGTH 
13InBlock.gifDWORD cbAvailable; //IN,缓冲区中的可用字节数 
14InBlock.gifLPBYTE lpbData; //IN,缓冲区指针,指向客户端发来的数据 
15InBlock.gifLPSTR lpszContentType; //IN,等价于环境变量CONTENT_TYPE 
16InBlock.gif
17InBlock.gif//回调函数,用于返回服务器的连接信息或特定的服务器详细情况 
18InBlock.gifBOOL ( WINAPI * GetServerVariable ) 
19InBlock.gif( HCONN hConn, 
20InBlock.gifLPSTR lpszVariableName, 
21InBlock.gifLPVOID lpvBuffer, 
22InBlock.gifLPDWORD lpdwSize ); 
23InBlock.gif
24InBlock.gifBOOL ( WINAPI * WriteClient ) //回调函数,从客户端的HTTP请求中读取数据 
25InBlock.gif( HCONN ConnID, 
26InBlock.gifLPVOID Buffer, 
27InBlock.gifLPDWORD lpdwBytes, 
28InBlock.gifDWORD dwReserved ); 
29InBlock.gif
30InBlock.gifBOOL ( WINAPI * ReadClient ) //回调函数,向客户端发送数据 
31InBlock.gif( HCONN ConnID, 
32InBlock.gifLPVOID lpvBuffer, 
33InBlock.gifLPDWORD lpdwSize ); 
34InBlock.gif
35InBlock.gifBOOL ( WINAPI * ServerSupportFunction ) //回调函数,访问服务器的一般和特定功能 
36InBlock.gif( HCONN hConn, 
37InBlock.gifDWORD dwHSERRequest, 
38InBlock.gifLPVOID lpvBuffer, 
39InBlock.gifLPDWORD lpdwSize, 
40InBlock.gifLPDWORD lpdwDataType ); 
41InBlock.gif
42ExpandedBlockEnd.gif}
 EXTENSION_CONTROL_BLOCK, 
*
LPEXTENSION_CONTROL_BLOCK; 

在上述ECB中,服务器不但提供了当前HTTP连接的句柄和一些变量,而且提供了4个回调函数给ISA调用,从而使ISA可以获得更详尽的信息。

三、ISAPI Filter

ISAPI Filter位于服务器和客户端之间,能够对服务器和客户端之间的通信进行预处理和后处理,比如对通信进行加密/解密、提供对客户进行身份验证的新方法、提供自定义的日志记录等,在CGI中没有与ISAPI Filter直接相对应的部分。

ISAPI Filter与服务器之间的接口有两个:GetFilterVersion( )和HttpFilterProc( )。任何

ISAPI Filter都必须引出这两个函数以供服务器调用。

1、在注册表的如下键值中存放着所有ISAPI Filter的文件名,IIS服务器启动时从该键值中获得

Filter的文件名并加载它们。

1
None.gif
HKEY_LOCAL_MACHINE
/
SYSTEM
/
CurrentControlSet
/
Services
/
W3SVC
/
Parameters
/
FilterDLL 
2、然后服务器调用每个Filter提供的GetFilterVersion( )函数,获得版本号以及该Filter希望处理的事件,即ISAPI Filter通过引出GetFilterVersion( )函数来告知服务器自己希望处理什么类型的事件,因为ISAPI Filter是通过事件来激活的,当满足条件的事件到达时,服务器就会调用Filter引出的主函数HttpFilterProc( )对该事件进行处理。GetFilterVersion( )的原型如下:
1
None.gif
BOOL WINAPI GetFilterVersion( 
2
None.gifDWORD dwServerFilterVersion; 
//
IN,服务器使用的版本规范 
3
None.gif
DWORD dwFilterVersion; 
//
OUT,过滤器使用的版本规范 
4
None.gif
CHAR lpszFilterDesc[SF_MAX_FILTER_DESC_LEN
+
1
]; 
//
OUT,对该过滤器的描述字符串 
5
None.gif
DWORD dwFlags 
//
OUT,事件和优先级标志 
6
None.gif
); 

事件和优先级标志dwFlasg的取值在MSDN中有详细解释,其中包括该Filter被调用的优先级,一般应使用默认的低优先级,否则可能会对系统的性能造成很大影响。

3、HttpFilterProc( )是ISAPI Filter主要的入口函数,它根据当前的事件的不同作出不同的处理。服务器通过如下的参数块和Filter进行交互,这个参数块的作用和ISA中的ECB类似。

 1
None.gif
typedef 
struct
 _HTTP_FILTER_CONTEXT 
 2
ExpandedBlockStart.gifContractedBlock.gif
dot.gif
 3InBlock.gif
 4InBlock.gifDWORD cbSize; //IN,本参数块的大小 
 5InBlock.gifDWORD Revision; //IN 
 6InBlock.gifPVOID ServerContext; //IN,由server使用本参数 
 7InBlock.gifDWORD ulReserved; //IN,由server使用本参数 
 8InBlock.gifBOOL fIsSecurePort; //IN,事件是否发生在安全端口上 
 9InBlock.gifPVOID pFilterContext; //IN/OUT,与本次请求相关的上下文 
10InBlock.gif
11InBlock.gif//回调函数,取得关于服务器和本次连接的信息 
12InBlock.gifBOOL (WINAPI * GetServerVariable) ( 
13InBlock.gifstruct _HTTP_FILTER_CONTEXT * pfc, 
14InBlock.gifLPSTR lpszVariableName, 
15InBlock.gifLPVOID lpvBuffer, 
16InBlock.gifLPDWORD lpdwSize 
17InBlock.gif); 
18InBlock.gif
19InBlock.gifBOOL (WINAPI * AddResponseHeaders) ( //回调函数,给HTTP响应添加一个标头 
20InBlock.gifstruct _HTTP_FILTER_CONTEXT * pfc, 
21InBlock.gifLPSTR lpszHeaders, 
22InBlock.gifDWORD dwReserved 
23InBlock.gif); 
24InBlock.gif
25InBlock.gifBOOL (WINAPI * WriteClient) ( //回调函数,将原始数据发送给客户端 
26InBlock.gifstruct _HTTP_FILTER_CONTEXT * pfc, 
27InBlock.gifLPVOID Buffer, 
28InBlock.gifLPDWORD lpdwBytes, 
29InBlock.gifDWORD dwReserved 
30InBlock.gif); 
31InBlock.gif
32InBlock.gifVOID * (WINAPI * AllocMem) ( //回调函数,分配内存。 
33InBlock.gifstruct _HTTP_FILTER_CONTEXT * pfc, 
34InBlock.gifDWORD cbSize, 
35InBlock.gifDWORD dwReserved 
36InBlock.gif); 
37InBlock.gif
38InBlock.gifBOOL (WINAPI * ServerSupportFunction) ( //回调函数,访问服务器的一般和特定功能 
39InBlock.gifstruct _HTTP_FILTER_CONTEXT * pfc, 
40InBlock.gifenum SF_REQ_TYPE sfReq, 
41InBlock.gifPVOID pData, 
42InBlock.gifDWORD ul1, 
43InBlock.gifDWORD ul2 
44InBlock.gif); 
45InBlock.gif
46ExpandedBlockEnd.gif}
 HTTP_FILTER_CONTEXT, 
*
PHTTP_FILTER_CONTEXT; 

四、VC++ 6.0中对ISAPI的支持

VC++ 6.0中定义了5个相关的类以简化ISAPI的编程工作:CHttpServer、CHttpServerContext、CHttpFilter、CHttpFilterContext、CHtmlStream,这5个类都没有父类。其中CHttpServer和CHttpServerContext主要用来编写ISA,CHttpFilter和CHttpFilterContext则用来编写ISAPI Filter,而CHtmlStream则用来操作内存中的HTML文件,为其它的4个类提供服务。CHttpServer在每个ISA中只能有一个实例,一个CHttpServer可以对应多个CHttpServerContext实例,每个

CHttpServerContext处理一个客户请求,这样可以处理并发的HTTP请求;CttpFilter和CHttpFilterContext之间的关系与此类似,在每个ISAPI Filter中只能有一个CHttpFilter实例,但是可以有多个CHttpFilterContext来处理并发的事件。CHttpServer和CHttpFilter是独立的类,它们可以共存于一个DLL中,也可以分别在不同的DLL中。

一个ISA可以提供多个命令,每个命令对应于CHttpServer(或其子类)的一个成员函数,客户端可以在URL中指定命令名及其参数。在VC++ 6.0中是通过parse map来实现这种对应的。

Parse map类似MFC中的Windows消息分发机制,通过使用VC提供的DECLARE_PARSE_MAP、BEGIN_PARSE_MAP、ON_PARSE_COMMAND、ON_PARSE_COMMAND_PARAMS、DEFAULT_PARSE_COMMAND、END_PARSE_MAP等宏,可以实现对不同的命令的处理。每个CHttpServer中只能建立一个parse map,当客户端给ISA发来命令的时候,parse map可以分析HTTP请求中的命令名及其参数,将该命令与相应的成员函数关联起来,即由该成员函数处理该命令。以MSDN中的例子程序pinball为例,该例中有下面这样一个表单:

 1
None.gif
<
form 
method
=get 
action
="pinball.dll?"
>
 
 2
None.gif
<
input 
type
="hidden"
 name
="MfcISAPICommand"
 VALUE
="GetImage"
>
 
 3
None.gif
<
input 
type
="radio"
 name
="Favorite"
 value
="1"
 checked
>
 Attack from Mars
<
br
>
 
 4
None.gif
<
input 
type
="radio"
 name
="Favorite"
 value
="2"
>
 Twilight Zone
<
br
>
 
 5
None.gif
<
input 
type
="radio"
 name
="Favorite"
 value
="3"
>
 The Addams Family
<
br
>
 
 6
None.gif
<
input 
type
="radio"
 name
="Favorite"
 value
="4"
>
 Cirqus Voltaire
<
br
>
 
 7
None.gif
<
input 
type
="radio"
 name
="Favorite"
 value
="0"
>
 I don't see it here
<
br
>
 
 8
None.gif
<
br
>
 
 9
None.gif
<
input 
type
="submit"
 value
="Show Me!"
>
 
10
None.gif
</
form
>
 
当客户端选中了上面的表单中的“Attack from Mars”这一项并点击了submit按钮后,服务器端
最终将得到如下的URL串:
1
None.gif
http://www.abc.com/pinball.dll?MfcISAPICommand=GetImage
&Favorite
=1 
在该URL串中,命令名是GetImage,参数Favorite的值是1,因此pinball.dll中的如下成员函数
将被调用以处理该请求,其中参数dwChoice对应URL中的参数Favorite:
1
None.gif
void
 CPinballExtension::GetImage(CHttpServerContext
*
 pCtxt, 
long
 dwChoice); 
而parse map需要按照下面的形式定义:
 1
None.gif
//
CPinballExtension从CHttpServer派生而来 
 2
None.gif
BEGIN_PARSE_MAP(CPinballExtension, CHttpServer) 
 3
None.gif
 4
None.gif
//
GetImage是CPinballExtension的成员函数,且有一个long型的参数即dwChoice 
 5
None.gif
ON_PARSE_COMMAND(GetImage, CPinballExtension, ITS_I4) 
 6
None.gif
 7
None.gif
//
该参数在URL中的名字为Favorite 
 8
None.gif
ON_PARSE_COMMAND_PARAMS(
"
Favorite
"
 9
None.gif
10
None.gifEND_PARSE_MAP(CPinballExtension) 
而对于ISAPI Filter,在VC中可以通过重载CHttpFilter(或其子类)的不同的成员函数来实现对不同事件的处理。可重载的函数如下,每一个成员函数均对应一个或多个事件:
1
None.gif
OnPreprocHeaders 
2
None.gifOnAuthentication 
3
None.gifOnUrlMap 
4
None.gifOnSendRawData 
5
None.gifOnReadRawData 
6
None.gifOnLog 
7
None.gifOnEndOfNetSession 

MSDN提供了4个关于ISAPI的编程实例:counter、MFCUCASE、pinball、wwwquote,有兴趣的可看看,本文主要不是介绍编程,所以不再赘述。

参考资料:

1、MSDN

2、《精通CGI编程》,丁一强等,清华大学出版社

转载于:https://www.cnblogs.com/shinyzhu/archive/2005/06/12/173057.html

你可能感兴趣的文章
Centos7下安装Oracle11g r2
查看>>
Centos6.7搭建ISCSI存储服务器
查看>>
[置顶] 全国地区表(树形)
查看>>
协议森林15 先生,要点单吗? (HTTP协议概览)
查看>>
【51nod 1100】斜率最大
查看>>
Tree Control in Dynamics 365 for operation【转】
查看>>
Linux中的目录功能(Red Hat 7)
查看>>
linux swing 乱码
查看>>
.NET中异常处理的最佳实践(转)
查看>>
CentOs 设置静态IP 方法(转)
查看>>
九. 常用类库、向量与哈希2.Object类
查看>>
[Java5新特性]Annotation注解
查看>>
冒泡排序
查看>>
多线程之进度条
查看>>
程序启动的完整过程
查看>>
java资料——哈希表(散列表)(转)
查看>>
反射,invoke()
查看>>
iServer6R使用WMTS自定义比例尺出图
查看>>
pinyin4j的使用
查看>>
Android_ 重写系统Crash处理类,保存Crash信息到SD卡 和 完美退出程序的方法
查看>>