﻿<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/"><channel><title>IT博客-不再回头 .net学习日记＆资料-随笔分类-ISapi</title><link>http://www.cnitblog.com/sfep/category/2424.html</link><description>我再也不愿听你要求 我受够了你那些自私要求</description><language>zh-cn</language><lastBuildDate>Fri, 30 Sep 2011 16:19:11 GMT</lastBuildDate><pubDate>Fri, 30 Sep 2011 16:19:11 GMT</pubDate><ttl>60</ttl><item><title>IIS的ISAPI接口简介</title><link>http://www.cnitblog.com/sfep/archive/2006/07/30/14459.html</link><dc:creator>不再回头</dc:creator><author>不再回头</author><pubDate>Sun, 30 Jul 2006 10:53:00 GMT</pubDate><guid>http://www.cnitblog.com/sfep/archive/2006/07/30/14459.html</guid><wfw:comment>http://www.cnitblog.com/sfep/comments/14459.html</wfw:comment><comments>http://www.cnitblog.com/sfep/archive/2006/07/30/14459.html#Feedback</comments><slash:comments>1</slash:comments><wfw:commentRss>http://www.cnitblog.com/sfep/comments/commentRss/14459.html</wfw:commentRss><trackback:ping>http://www.cnitblog.com/sfep/services/trackbacks/14459.html</trackback:ping><description><![CDATA[
		<font face="宋体">ISAPI（Internet Server Application Programming Interface）作为一种可用来替代CGI的方法，是由微软和Process软件公司联合提出的Web服务器上的API标准。ISAPI与Web服务器结合紧密，功能强大，能够获得大量的信息，因此利用ISAPI可以开发出灵活高效的Web服务器增强程序。由于ISAPI程序与Web服务器的关系，使得ISAPI接口在安全方面有一定的研究价值。本文主要讨论ISAPI在IIS和VC++ 6.0中的实现。 <br /><br />一、ISAPI接口和CGI接口的不同。 <br /><br />ISAPI程序和CGI程序完成类似的功能，但是实现方法不同。 <br /><br />1、ISAPI程序以DLL形式被Web服务器加载到自己的进程空间中，因此和服务器共用同一个地址空间，且在没有客户请求时可以将其从内存中卸载；而对客户端发来的每个对CGI程序的请求则需要服务器为它单独启动一个进程，这需要耗费大量的时间和内存。当并发的请求数目很大时，使用CGI在效率上不如ISAPI。 <br /><br />2、CGI程序通过环境块和标准输入输出与Web服务器进行通信，而ISAPI程序与服务器结合得更为紧密，与服务器共享同一个进程上下文，主要通过一个参数块与服务器进行交互，可以从服务器那里获得关于当前HTTP连接的大量信息。 <br /><br />ISAPI主要分为ISA和ISAPI Filter两部分。ISA方法相对而言要传统一些，利用一些特殊的链接，指向服务器的作业，供程序开发人员设计一些扩展功能；而ISAPI过滤器则倾向于构造服务器直接调用的模块，提供一种无缝链接部件用于监测直接来自于服务器的HTTP请求。 <br /><br /><br />二、ISA <br /><br />ISA（Internet Server Application）也可称为ISAPI DLL，其功能和CGI程序的功能直接相对应，使用方法和CGI也类似，由客户端在URL中指定其名称而激活。例如下面的请求将调用服务器的虚拟可执行目录Scripts下的function.dll（ISAPI DLL必须放在服务器的虚拟可执行目录下）： <br />http://www.abc.com/Scripts/function.dll? <br /><br />ISA和服务器之间的接口主要有两个：GetExtentionVersion( )和HttpExtentionProc( )。任何ISA都必须在其PE文件头的引出表中定义这两个引出函数，以供Web服务器在适当的时候调用。 <br /><br />1、当服务器刚加载ISA时，它会调用ISA提供的GetExtentionVersion( )来获得该ISA所需要的服务器版本，并与自己的版本相比较，以保证版本兼容。函数原型如下： <br /><br />BOOL WINAPI GetExtentionVersion(HSE_VERSION_INFO *version)； <br />typedef struct _HSE_VERSION_INFO <br />{ <br />DWORD dwExtensionVersion; //版本号 <br />CHAR lpszExtensionDesc[HSE_MAX_EXT_DLL_NAME_LEN]; //关于ISA的描述字符串 <br />} HSE_VERSION_INFO, *LPHSE_VERSION_INFO; <br /><br />2、ISA的真正入口是HttpExtentionProc( )，它相当于普通C程序的main( )函数，在这个函数中根据不同的客户请求作不同的处理。服务器和HttpExtentionProc( )之间是通过扩展控制块（Extention Control Block）来进行通信的，即ECB中存放入口参数和出口参数，包括服务器提供的几个回调函数的入口地址。函数原型如下： <br /><br />DWORD HttpExtensionProc( EXTENSION_CONTROL_BLOCK *pECB ); <br /><br />ECB的结构定义如下（IN表示入口参数，OUT表示出口参数）： <br /><br />typedef struct _EXTENSION_CONTROL_BLOCK <br />{ <br />DWORD cbSize; //IN，本结构的大小，只读 <br />DWORD dwVersion //IN，版本号，高16位为主版本号，低16位为次版本号 <br />HCONN ConnID; //IN，连接句柄，由服务器分配，ISA只能读取该值 <br />DWORD dwHttpStatusCode; //OUT，当前完成的事务状态 <br />CHAR lpszLogData[HSE_LOG_BUFFER_LEN]; //OUT，需要写入到日志文件中的内容 <br />LPSTR lpszMethod; //IN，等价于CGI的环境变量REQUEST_METHOD <br />LPSTR lpszQueryString; //IN，等价于环境变量QUERY_STRING <br />LPSTR lpszPathInfo; //IN，等价于环境变量PATH_INFO <br />LPSTR lpszPathTranslated; //IN，等价于环境变量PATH_TRANSLATED <br />DWORD cbTotalBytes; //IN，等价于环境变量CONTENT_LENGTH <br />DWORD cbAvailable; //IN，缓冲区中的可用字节数 <br />LPBYTE lpbData; //IN，缓冲区指针，指向客户端发来的数据 <br />LPSTR lpszContentType; //IN，等价于环境变量CONTENT_TYPE <br /><br />//回调函数，用于返回服务器的连接信息或特定的服务器详细情况 <br />BOOL ( WINAPI * GetServerVariable ) <br />( HCONN hConn, <br />LPSTR lpszVariableName, <br />LPVOID lpvBuffer, <br />LPDWORD lpdwSize ); <br /><br />BOOL ( WINAPI * WriteClient ) //回调函数，从客户端的HTTP请求中读取数据 <br />( HCONN ConnID, <br />LPVOID Buffer, <br />LPDWORD lpdwBytes, <br />DWORD dwReserved ); <br /><br />BOOL ( WINAPI * ReadClient ) //回调函数，向客户端发送数据 <br />( HCONN ConnID, <br />LPVOID lpvBuffer, <br />LPDWORD lpdwSize ); <br /><br />BOOL ( WINAPI * ServerSupportFunction ) //回调函数，访问服务器的一般和特定功能 <br />( HCONN hConn, <br />DWORD dwHSERRequest, <br />LPVOID lpvBuffer, <br />LPDWORD lpdwSize, <br />LPDWORD lpdwDataType ); <br /><br />} EXTENSION_CONTROL_BLOCK, *LPEXTENSION_CONTROL_BLOCK; <br /><br />在上述ECB中，服务器不但提供了当前HTTP连接的句柄和一些变量，而且提供了4个回调函数给ISA调用，从而使ISA可以获得更详尽的信息。 <br /><br />三、ISAPI Filter <br /><br />ISAPI Filter位于服务器和客户端之间，能够对服务器和客户端之间的通信进行预处理和后处理，比如对通信进行加密/解密、提供对客户进行身份验证的新方法、提供自定义的日志记录等，在CGI中没有与ISAPI Filter直接相对应的部分。 <br /><br />ISAPI Filter与服务器之间的接口有两个：GetFilterVersion( )和HttpFilterProc( )。任何 <br />ISAPI Filter都必须引出这两个函数以供服务器调用。 <br /><br />1、在注册表的如下键值中存放着所有ISAPI Filter的文件名，IIS服务器启动时从该键值中获得 <br />Filter的文件名并加载它们。 <br /><br />HKEY_LOCAL_MACHINE/SYSTEM/CurrentControlSet/Services/W3SVC/Parameters/FilterDLL <br /><br />2、然后服务器调用每个Filter提供的GetFilterVersion( )函数，获得版本号以及该Filter希望处理的事件，即ISAPI Filter通过引出GetFilterVersion( )函数来告知服务器自己希望处理什么类型的事件，因为ISAPI Filter是通过事件来激活的，当满足条件的事件到达时，服务器就会调用Filter引出的主函数HttpFilterProc( )对该事件进行处理。GetFilterVersion( )的原型如下： <br /><br />BOOL WINAPI GetFilterVersion( <br />DWORD dwServerFilterVersion; //IN，服务器使用的版本规范 <br />DWORD dwFilterVersion; //OUT，过滤器使用的版本规范 <br />CHAR lpszFilterDesc[SF_MAX_FILTER_DESC_LEN+1]; //OUT，对该过滤器的描述字符串 <br />DWORD dwFlags //OUT，事件和优先级标志 <br />)； <br /><br />事件和优先级标志dwFlasg的取值在MSDN中有详细解释，其中包括该Filter被调用的优先级，一般应使用默认的低优先级，否则可能会对系统的性能造成很大影响。 <br /><br />3、HttpFilterProc( )是ISAPI Filter主要的入口函数，它根据当前的事件的不同作出不同的处理。服务器通过如下的参数块和Filter进行交互，这个参数块的作用和ISA中的ECB类似。 <br /><br />typedef struct _HTTP_FILTER_CONTEXT <br />{ <br /><br />DWORD cbSize; //IN，本参数块的大小 <br />DWORD Revision; //IN <br />PVOID ServerContext; //IN，由server使用本参数 <br />DWORD ulReserved; //IN，由server使用本参数 <br />BOOL fIsSecurePort; //IN，事件是否发生在安全端口上 <br />PVOID pFilterContext; //IN/OUT，与本次请求相关的上下文 <br /><br />//回调函数，取得关于服务器和本次连接的信息 <br />BOOL (WINAPI * GetServerVariable) ( <br />struct _HTTP_FILTER_CONTEXT * pfc, <br />LPSTR lpszVariableName, <br />LPVOID lpvBuffer, <br />LPDWORD lpdwSize <br />); <br /><br />BOOL (WINAPI * AddResponseHeaders) ( //回调函数，给HTTP响应添加一个标头 <br />struct _HTTP_FILTER_CONTEXT * pfc, <br />LPSTR lpszHeaders, <br />DWORD dwReserved <br />); <br /><br />BOOL (WINAPI * WriteClient) ( //回调函数，将原始数据发送给客户端 <br />struct _HTTP_FILTER_CONTEXT * pfc, <br />LPVOID Buffer, <br />LPDWORD lpdwBytes, <br />DWORD dwReserved <br />); <br /><br />VOID * (WINAPI * AllocMem) ( //回调函数，分配内存。 <br />struct _HTTP_FILTER_CONTEXT * pfc, <br />DWORD cbSize, <br />DWORD dwReserved <br />); <br /><br />BOOL (WINAPI * ServerSupportFunction) ( //回调函数，访问服务器的一般和特定功能 <br />struct _HTTP_FILTER_CONTEXT * pfc, <br />enum SF_REQ_TYPE sfReq, <br />PVOID pData, <br />DWORD ul1, <br />DWORD ul2 <br />); <br /><br />} HTTP_FILTER_CONTEXT, *PHTTP_FILTER_CONTEXT; <br /><br />四、VC++ 6.0中对ISAPI的支持 <br /><br />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实例，每个 <br />CHttpServerContext处理一个客户请求，这样可以处理并发的HTTP请求；CttpFilter和CHttpFilterContext之间的关系与此类似，在每个ISAPI Filter中只能有一个CHttpFilter实例，但是可以有多个CHttpFilterContext来处理并发的事件。CHttpServer和CHttpFilter是独立的类，它们可以共存于一个DLL中，也可以分别在不同的DLL中。 <br /><br />一个ISA可以提供多个命令，每个命令对应于CHttpServer（或其子类）的一个成员函数，客户端可以在URL中指定命令名及其参数。在VC++ 6.0中是通过parse map来实现这种对应的。 <br /><br />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为例，该例中有下面这样一个表单： <br /><br />&lt;form method=get action="pinball.dll?"&gt; <br />&lt;input type="hidden" name="MfcISAPICommand" VALUE="GetImage"&gt; <br />&lt;input type="radio" name="Favorite" value="1" checked&gt; Attack from Mars&lt;br&gt; <br />&lt;input type="radio" name="Favorite" value="2"&gt; Twilight Zone&lt;br&gt; <br />&lt;input type="radio" name="Favorite" value="3"&gt; The Addams Family&lt;br&gt; <br />&lt;input type="radio" name="Favorite" value="4"&gt; Cirqus Voltaire&lt;br&gt; <br />&lt;input type="radio" name="Favorite" value="0"&gt; I don't see it here&lt;br&gt; <br />&lt;br&gt; <br />&lt;input type="submit" value="Show Me!"&gt; <br />&lt;/form&gt; <br /><br />当客户端选中了上面的表单中的“Attack from Mars”这一项并点击了submit按钮后，服务器端 <br />最终将得到如下的URL串： <br /><br />http://www.abc.com/pinball.dll?MfcISAPICommand=GetImage&amp;Favorite=1 <br /><br />在该URL串中，命令名是GetImage，参数Favorite的值是1，因此pinball.dll中的如下成员函数 <br />将被调用以处理该请求，其中参数dwChoice对应URL中的参数Favorite： <br /><br />void CPinballExtension::GetImage(CHttpServerContext* pCtxt, long dwChoice)； <br /><br />而parse map需要按照下面的形式定义： <br /><br />//CPinballExtension从CHttpServer派生而来 <br />BEGIN_PARSE_MAP(CPinballExtension, CHttpServer) <br /><br />//GetImage是CPinballExtension的成员函数，且有一个long型的参数即dwChoice <br />ON_PARSE_COMMAND(GetImage, CPinballExtension, ITS_I4) <br /><br />//该参数在URL中的名字为Favorite <br />ON_PARSE_COMMAND_PARAMS("Favorite") <br /><br />END_PARSE_MAP(CPinballExtension) <br /><br />而对于ISAPI Filter，在VC中可以通过重载CHttpFilter（或其子类）的不同的成员函数来实现对不同事件的处理。可重载的函数如下，每一个成员函数均对应一个或多个事件： <br /><br />OnPreprocHeaders <br />OnAuthentication <br />OnUrlMap <br />OnSendRawData <br />OnReadRawData <br />OnLog <br />OnEndOfNetSession <br /><br />MSDN提供了4个关于ISAPI的编程实例：counter、MFCUCASE、pinball、wwwquote，有兴趣的可看看，本文主要不是介绍编程，所以不再赘述。</font>
<img src ="http://www.cnitblog.com/sfep/aggbug/14459.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cnitblog.com/sfep/" target="_blank">不再回头</a> 2006-07-30 18:53 <a href="http://www.cnitblog.com/sfep/archive/2006/07/30/14459.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>isapi处理流程</title><link>http://www.cnitblog.com/sfep/archive/2006/04/11/8988.html</link><dc:creator>不再回头</dc:creator><author>不再回头</author><pubDate>Tue, 11 Apr 2006 06:46:00 GMT</pubDate><guid>http://www.cnitblog.com/sfep/archive/2006/04/11/8988.html</guid><wfw:comment>http://www.cnitblog.com/sfep/comments/8988.html</wfw:comment><comments>http://www.cnitblog.com/sfep/archive/2006/04/11/8988.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cnitblog.com/sfep/comments/commentRss/8988.html</wfw:commentRss><trackback:ping>http://www.cnitblog.com/sfep/services/trackbacks/8988.html</trackback:ping><description><![CDATA[
		<span class="dash6b63-6587--Char" style="FONT-FAMILY: '宋体', 'Arial'">筛选器</span>
		<p class="dash6b63-6587">IIS<span class="dash6b63-6587--Char" style="FONT-FAMILY: '宋体', 'Arial'">服务器</span></p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">处理连接请求标题</span>
		</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">处理连接请求数据</span>
		</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">将</span>
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt">URL</span>
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">映射成物理路径</span>
		</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">授权用户（如果需要）</span>
		</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">组织数据</span>
		</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">写入日志</span>
		</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">
						<sup>结束连接</sup>
				</span>
		</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">OnReadRawData</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">OnPreprocHeaders</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">OnUrlMap</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: right">OnAuthentication</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">OnSendRawData</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">OnEndOfNetSession</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">发送给客户机</span>
		</p>
		<p class="dash6b63-6587" style="TEXT-ALIGN: center">OnLog<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">图</span><span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt">1</span><span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">显示了</span><span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt">IIS</span><span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">服务器从接收到客户机请求数据到结束连接整个过程中</span><span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt">IIS</span><span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">的处理步骤，以及筛选器可以获得的通知和处理方式。</span></p>
		<p class="dash6b63-6587" style="TEXT-INDENT: 18pt; LINE-HEIGHT: 12pt; TEXT-ALIGN: center">
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">图</span>
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt">1 IIS</span>
				<span class="dash6b63-6587--Char" style="FONT-SIZE: 9pt; FONT-FAMILY: '宋体', 'Arial'">服务器与筛选器工作流程</span>
		</p>
<img src ="http://www.cnitblog.com/sfep/aggbug/8988.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cnitblog.com/sfep/" target="_blank">不再回头</a> 2006-04-11 14:46 <a href="http://www.cnitblog.com/sfep/archive/2006/04/11/8988.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>