`

IIS的ISAPI接口简介

 
阅读更多
ISAPI(InternetServerApplicationProgrammingInterface)作为一种可用来替代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和ISAPIFilter两部分。ISA方法相对而言要传统一些,利用一些特殊的链接,指向服务器的作业,供程序开发人员设计一些扩展功能;而ISAPI过滤器则倾向于构造服务器直接调用的模块,提供一种无缝链接部件用于监测直接来自于服务器的HTTP请求。


二、ISA

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

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

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

BOOLWINAPIGetExtentionVersion(HSE_VERSION_INFO*version);
typedefstruct_HSE_VERSION_INFO
{
DWORDdwExtensionVersion;//版本号
CHARlpszExtensionDesc[HSE_MAX_EXT_DLL_NAME_LEN];//关于ISA的描述字符串
}HSE_VERSION_INFO,*LPHSE_VERSION_INFO;

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

DWORDHttpExtensionProc(EXTENSION_CONTROL_BLOCK*pECB);

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

typedefstruct_EXTENSION_CONTROL_BLOCK
{
DWORDcbSize;//IN,本结构的大小,只读
DWORDdwVersion//IN,版本号,高16位为主版本号,低16位为次版本号
HCONNConnID;//IN,连接句柄,由服务器分配,ISA只能读取该值
DWORDdwHttpStatusCode;//OUT,当前完成的事务状态
CHARlpszLogData[HSE_LOG_BUFFER_LEN];//OUT,需要写入到日志文件中的内容
LPSTRlpszMethod;//IN,等价于CGI的环境变量REQUEST_METHOD
LPSTRlpszQueryString;//IN,等价于环境变量QUERY_STRING
LPSTRlpszPathInfo;//IN,等价于环境变量PATH_INFO
LPSTRlpszPathTranslated;//IN,等价于环境变量PATH_TRANSLATED
DWORDcbTotalBytes;//IN,等价于环境变量CONTENT_LENGTH
DWORDcbAvailable;//IN,缓冲区中的可用字节数
LPBYTElpbData;//IN,缓冲区指针,指向客户端发来的数据
LPSTRlpszContentType;//IN,等价于环境变量CONTENT_TYPE

//回调函数,用于返回服务器的连接信息或特定的服务器详细情况
BOOL(WINAPI*GetServerVariable)
(HCONNhConn,
LPSTRlpszVariableName,
LPVOIDlpvBuffer,
LPDWORDlpdwSize);

BOOL(WINAPI*WriteClient)//回调函数,从客户端的HTTP请求中读取数据
(HCONNConnID,
LPVOIDBuffer,
LPDWORDlpdwBytes,
DWORDdwReserved);

BOOL(WINAPI*ReadClient)//回调函数,向客户端发送数据
(HCONNConnID,
LPVOIDlpvBuffer,
LPDWORDlpdwSize);

BOOL(WINAPI*ServerSupportFunction)//回调函数,访问服务器的一般和特定功能
(HCONNhConn,
DWORDdwHSERRequest,
LPVOIDlpvBuffer,
LPDWORDlpdwSize,
LPDWORDlpdwDataType);

}EXTENSION_CONTROL_BLOCK,*LPEXTENSION_CONTROL_BLOCK;

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

三、ISAPIFilter

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

ISAPIFilter与服务器之间的接口有两个:GetFilterVersion()和HttpFilterProc()。任何
ISAPIFilter都必须引出这两个函数以供服务器调用。

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

HKEY_LOCAL_MACHINE/SYSTEM/CurrentControlSet/Services/W3SVC/Parameters/FilterDLL

2、然后服务器调用每个Filter提供的GetFilterVersion()函数,获得版本号以及该Filter希望处理的事件,即ISAPIFilter通过引出GetFilterVersion()函数来告知服务器自己希望处理什么类型的事件,因为ISAPIFilter是通过事件来激活的,当满足条件的事件到达时,服务器就会调用Filter引出的主函数HttpFilterProc()对该事件进行处理。GetFilterVersion()的原型如下:

BOOLWINAPIGetFilterVersion(
DWORDdwServerFilterVersion;//IN,服务器使用的版本规范
DWORDdwFilterVersion;//OUT,过滤器使用的版本规范
CHARlpszFilterDesc[SF_MAX_FILTER_DESC_LEN+1];//OUT,对该过滤器的描述字符串
DWORDdwFlags//OUT,事件和优先级标志
);

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

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

typedefstruct_HTTP_FILTER_CONTEXT
{

DWORDcbSize;//IN,本参数块的大小
DWORDRevision;//IN
PVOIDServerContext;//IN,由server使用本参数
DWORDulReserved;//IN,由server使用本参数
BOOLfIsSecurePort;//IN,事件是否发生在安全端口上
PVOIDpFilterContext;//IN/OUT,与本次请求相关的上下文

//回调函数,取得关于服务器和本次连接的信息
BOOL(WINAPI*GetServerVariable)(
struct_HTTP_FILTER_CONTEXT*pfc,
LPSTRlpszVariableName,
LPVOIDlpvBuffer,
LPDWORDlpdwSize
);

BOOL(WINAPI*AddResponseHeaders)(//回调函数,给HTTP响应添加一个标头
struct_HTTP_FILTER_CONTEXT*pfc,
LPSTRlpszHeaders,
DWORDdwReserved
);

BOOL(WINAPI*WriteClient)(//回调函数,将原始数据发送给客户端
struct_HTTP_FILTER_CONTEXT*pfc,
LPVOIDBuffer,
LPDWORDlpdwBytes,
DWORDdwReserved
);

VOID*(WINAPI*AllocMem)(//回调函数,分配内存。
struct_HTTP_FILTER_CONTEXT*pfc,
DWORDcbSize,
DWORDdwReserved
);

BOOL(WINAPI*ServerSupportFunction)(//回调函数,访问服务器的一般和特定功能
struct_HTTP_FILTER_CONTEXT*pfc,
enumSF_REQ_TYPEsfReq,
PVOIDpData,
DWORDul1,
DWORDul2
);

}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则用来编写ISAPIFilter,而CHtmlStream则用来操作内存中的HTML文件,为其它的4个类提供服务。CHttpServer在每个ISA中只能有一个实例,一个CHttpServer可以对应多个CHttpServerContext实例,每个
CHttpServerContext处理一个客户请求,这样可以处理并发的HTTP请求;CttpFilter和CHttpFilterContext之间的关系与此类似,在每个ISAPIFilter中只能有一个CHttpFilter实例,但是可以有多个CHttpFilterContext来处理并发的事件。CHttpServer和CHttpFilter是独立的类,它们可以共存于一个DLL中,也可以分别在不同的DLL中。

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

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

<formmethod=getaction="pinball.dll?">
<inputtype="hidden"name="MfcISAPICommand"VALUE="GetImage">
<inputtype="radio"name="Favorite"value="1"checked>AttackfromMars<br>
<inputtype="radio"name="Favorite"value="2">TwilightZone<br>
<inputtype="radio"name="Favorite"value="3">TheAddamsFamily<br>
<inputtype="radio"name="Favorite"value="4">CirqusVoltaire<br>
<inputtype="radio"name="Favorite"value="0">Idon'tseeithere<br>
<br>
<inputtype="submit"value="ShowMe!">
</form>

当客户端选中了上面的表单中的“AttackfromMars”这一项并点击了submit按钮后,服务器端
最终将得到如下的URL串:

http://www.abc.com/pinball.dll?MfcISAPICommand=GetImage&Favorite=1

在该URL串中,命令名是GetImage,参数Favorite的值是1,因此pinball.dll中的如下成员函数
将被调用以处理该请求,其中参数dwChoice对应URL中的参数Favorite:

voidCPinballExtension::GetImage(CHttpServerContext*pCtxt,longdwChoice);

而parsemap需要按照下面的形式定义:

//CPinballExtension从CHttpServer派生而来
BEGIN_PARSE_MAP(CPinballExtension,CHttpServer)

//GetImage是CPinballExtension的成员函数,且有一个long型的参数即dwChoice
ON_PARSE_COMMAND(GetImage,CPinballExtension,ITS_I4)

//该参数在URL中的名字为Favorite
ON_PARSE_COMMAND_PARAMS("Favorite")

END_PARSE_MAP(CPinballExtension)

而对于ISAPIFilter,在VC中可以通过重载CHttpFilter(或其子类)的不同的成员函数来实现对不同事件的处理。可重载的函数如下,每一个成员函数均对应一个或多个事件:

OnPreprocHeaders
OnAuthentication
OnUrlMap
OnSendRawData
OnReadRawData
OnLog
OnEndOfNetSession

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

参考资料:

1、MSDN
2、《精通CGI编程》,丁一强等,清华大学出版社
分享到:
评论

相关推荐

    ISAPI开发手册

    ISAPI开发手册,对有志于开发ISA、IIS ISAPI Filter的同志还是很有参考意义的。

    ISAPI Filter 限制 IIS 多线程访问

    使用 ISAPI Filter 实现的限制 IIS 多线程访问程序。主要应用在下载系统,或者使用 IIS 做的流媒体点播系统中,稍加修改可以使其具有认证功能。在程序中已经对 用户认证 事件做了直接返回的处理。使用本 Filter 可以...

    高性能WEB服务器(MyWebServer,支持ISAPI和FastCGI接口)

    支持HTTP/1.1、断点续传、大文件下载、正则表达式URL重写、虚拟目录等,可通过ISAPI接口、FasctCGI接口实现执行服务器脚本(如PHP,asp,asp.net等),性能完全超越IIS等很多主流WEB服务器软件。  使用FasctCGI时,在...

    ISAPI Filter 限制 IIS 多线程访问!

    使用 ISAPI Filter 实现的限制 IIS 多线程访问程序。主要应用在下载系统,或者使用 IIS 做的流媒体点播系统中,稍加修改可以使其具有认证功能。在程序中已经对 用户认证 事件做了直接返回的处理。使用本 Filter 可以...

    IIS6下PHP的ISAPI和FastCGI性能比较 期待ii7

     2、ISAPI(Internet Server Application Program Interface)是微软提供的一套面向WEB服务的API接口,它能实现CGI提供的全部功能,并在此基础上进行了扩展,如提供了过滤器应用程序接口。ISAPI应用大多数以DLL动态...

    IIS7.0和tomcat6.0整合成功案例笔记(有图)

    二、使用技术:IIS7.0和TOMCAT6.0的默认端口不用改变,使用原有的80和8080即可,采用isapi_redirect-1.2.26.dll文件作为iis和tomcat的接口,使用http://www.zhonghuiheat.com就能访问tomcat中的jsp网站。 三、操作...

    高性能IOCP WEB服务器(支持ISAPI版PHP)

    通过第三方ISAPI接口DLL支持ISAPI版PHP,可用作PHP站点调试工具或服务器。php仅支持ISAPI版。包中带了一个基本的PHP引擎, 解压后,启动HTTP服务器,修改PHP.ini中的doc_root=网站目录(全路径)即可, 两个DLL是PHP...

    目前最好的IIS防火墙-威盾IIS防火墙

    IISWALL采用最新的VC++.NET开发编制,基于最高效的ISAPIFilter技术实现,具有速度快效率高不占用CPU等特点,在IIS内核中实现真正意义上的全方位安全防护,是迄今为止解决IIS安全问题最完整的安全解决方案之一,已在...

    IIS的集成和经典模式

    经典模式是为了与之前的版本兼容,使用ISAPI扩展来调用ASP.NET运行库,原先运行于IIS6.0下的Web应用程序迁移到IIS7.0中只要将应用程序配置成经典模式,代码基本不用修改就可以正常运行。集成模式是一种统一的哀求...

    IIS处理页面的运行机制

    只能直接请求像HTML这样的静态文件,之所以能处理ASPX这样扩展名的页面,是因为IIS有一个ISAPI(Internet Server Application Programe Interface,互联网服务器应用程序接口)过滤器,它是一个COM组件,虽然这 ISAPI ...

    IIS6.0 IIS,互联网信息服务

    以及引入了命令行工具 Appcmd.exe ,给 Web 服务器的日常管理、监视和配置提供了除图形接口外的另一种方式,更为简单、高效。 同Windows XP一样,在Windows Vista的默认设置下,IIS (Internet 信息服务) 7.0未予安装...

    php5isapi.dll

    这是IIS对接PHP的一个“接口”,很多人下载了PHP都找不到这个文件,现本人公布出来给大家。

    威盾IIS防火墙 v4.8 Build 2010.10.10.zip

    威盾防火墙采用最新的VC .NET开发编制,基于最高效的ISAPI Filter技术实现,具有速度快、效率高、应用防护透明化等特点,在IIS内核中实现真正意义上的全方位安全防护,是迄今为止解决IIS安全问题最完整的安全解决...

    MyWebServer(轻量级web服务器软件) v3.1.29 绿色免费版.zip

    支持HTTP/1.1、断点续传、大文件下载、正则表达式URL重写、虚拟目录、HTTP反向代理等,可通过ISAPI接口、FastCGI接口实现执行服务器脚本(如PHP,asp,asp.net等),性能完全超越IIS等很多主流WEB服务器软件。...

    服务器iis安全专家.zip

    1. 合并了之前的ISAPI EXTENTED DLL模块到Virtualiisfirewall的IsApi Filter模块,使之成为IIS双接口,使各种关键数据得以共享。速度,更快,更稳定。 2.去掉了之前收费软件的各种认证校验,软件免费,只需要激活...

    iis伪静态中文url出现乱码的解决办法

    首先要看下你安装的是IIS rewrite_2.0还是ISAPI_Rewrite 3.x的伪静态插件。 如果你安装的是IIS rewrite_2.0的话,传递的参数都是UTF-8格式的,如果你网站是GBK的就会出现中文连接乱码。 解决办法: rewrite_2.0规则...

    高性能(通过第三方DLL)

    通过第三方ISAPI接口DLL支持ISAPI版PHP,可用作PHP站点调试工具或服务器。php仅支持ISAPI版。包中带了一个基本的PHP引擎, 解压后,启动HTTP服务器,修改PHP.ini中的doc_root=网站目录(全路径)即可, 两个DLL是PHP...

    IIS7 经典模式和集成模式的区别分析

    经典模式是为了与之前的版本兼容,使用ISAPI扩展来调用ASP.NET运行库,原先运行于IIS6.0下的Web应用程序迁移到IIS7.0中只要将应用程序配置成经典模式,代码基本不用修改就可以正常运行。集成模式是一种统一的哀求...

    IIS下PHP的三种配置方式对比

    在Windows IIS 6.0下配置PHP,通常有CGI、ISAPI和FastCGI三种配置方式,这三种模式都可以在IIS 6.0下成功运行,下面我就讲一下这三种方式配置的区别和性能上的差异。    1、CGI(通用网关接口/Common Gateway ...

    MyWebServer_3.6.20虚拟服务器.zip

    MyWebServer是一个高...支持HTTP/1.1、断点续传、大文件下载、正则表达式URL重写、虚拟目录等,可通过ISAPI接口、FastCGI接口实现执行服务器脚本(如PHP,asp,asp.net等),性能完全超越IIS等很多主流Web服务器软件。

Global site tag (gtag.js) - Google Analytics