• 用delphi编写ISAPI过滤器


    用delphi编写ISAPI过滤器

    实 现WEB 站 点 同 时 对GB 码 和BIG 码 的 支 持


    ---- 目 前 由 于 汉 字 内 码 的 不 统 一, 互 联 网 上 的 中 文 站 点 为 了 实 现 对 于 不 同 用 户 的 支 持, 一 般 采 取 建 立 两 套 主 页, 分 别 用GB 和BIG 码 来 编 写。 这 样 做 显 然 要 增 加 站 点 的 维 护 工 作, 更 新 主 页 时 要 同 时 更 新 两 部 分。 而 且 如 果 主 页 内 容 是 实 时 更 新 的, 采 用 手 工 维 护 两 套 主 页 的 方 法 显 然 不 行 了。 本 文 介 绍 了 用ISAPI 过 滤 器 来 动 态 产 生 另 外 一 套 内 码 主 页 的 方 法, 这 样 就 可 以 只 制 作 一 套 主 页 就 同 时 支 持GB 码 和BIG5 码。

    ---- 基 本 的 思 路, 编 写 一 个ISAPI 过 滤 器, 对 于 所 有 最 终 返 回 给 用 户 的HTML 文 本, 实 行 内 码 转 换。 这 样 用 户 看 到 的 将 是 他 期 望 的 编 码 方 式。 ISAPI 过 滤 器 可 以 作 为WEB Server 横 向 功 能 扩 展。 当 某 个 预 先 定 义 好

    ---- 的 服 务 器 端 的 事 件 发 生 时,IIS 就 调 用 用 户 定 义 好 的 过 程, 此 时 就 可 以 通 过 修 改IIS 传 来 的 数 据 来 改 变IIS 的 行 为。IIS 预 定 义 的 事 件 如 下:

     SF_NOTIFY_READ_RAW_DATA 
    

    ---- 当IIS 要 从 用 户 读 入 数 据 时 发 生。 过 滤 器 可 以 在IIS 处 理 他 们 之 前 检 查 甚 至 修 改 用 户 输 入 的 原 始 数 据。

    ---- SF_NOTIFY_PREPROC_HEADERS

    ---- IIS 预 处 理HTTP 请 求 包 头 后 发 生。 过 滤 器 可 以 检 查 修 改 增 加 包 头。

    ---- SF_NOTIFY_AUTHENTICATION

    ---- IIS 试 图 验 证 用 户 身 份 时 发 生。 过 滤 器 可 以 实 现 自 己 的 验 证 方 案。

    ---- SF_NOTIFY_URL_MAP

    ---- IIS 试 图 将URL 解 释 为 物 理 文 件 时。 过 滤 器 可 以 将 请 求 重 定 向 到 其 他 的 文 件。

    ---- SF_NOTIFY_ACCESS_DENIED
    ---- 当 身 份 验 证 失 败 时 发 生。
    ---- SF_NOTIFY_SEND_RAW_DATA

    ---- 当 其 他 程 序 处 理 完,IIS 准 备 将 数 据 发 回 给 用 户 时 发 生。 我 们 的 过 滤 器 就 通 过 此 事 件, 转 换 内 码。

    ---- SF_NOTIFY_LOG

    ---- 当IIS 写 记 录 到LOG 文 件 时。 过 滤 器 可 以 搜 集 更 多 的 信 息 写 入 记 录 文 件 中。

    ---- SF_NOTIFY_END_OF_REQUEST

    ---- 当 一 个HTTP 请 求 结 束 时 发 生。 过 滤 器 可 以 实 现 基 于 请 求 的 处 理。 由 于 这 是 在IIS 3.0 中 新 增 的,Delphi 中 的ISAPI2.pass 单 元 中 没 有 相 应 的 定 义 可 以 手 工 加 入SF_NOTIFY_END_OF_REQUEST=$80

    ---- SF_NOTIFY_END_OF_NET_SESSION

    ---- 连 接 结 束 时。 注 意 如 果 浏 览 器 支 持"keep-alive", 一 次 连 接 可 能 包 含 几 个HTTP 请 求。 过 滤 器 可 以 用 他 来 释 放 一 些 用 户 的 资 源。

    ---- 我 们 要 实 现 动 态 的 内 码 转 换, 只 要 过 滤 器 处 理SF_NOTIFY_SEND_RAW_DATA 事 件, 将IIS 处 理 好 的 数 据 转 换 成 需 要 的 内 码 就 可 以 实 现 内 码 的 动 态 转 换。 具 体 程 序 有 两 个 问 题 需 要 注 意:

    1. 过 滤 器 只 能 处 理 返 回 是HTML 格 式 的, 其 他 图 片 等 二 进 制 请 求 无 须 也 不 允 许 转 换。
    2. 对 于 返 回 的HTML, 只 处 理 实 际 数 据, 其 他HTTP 协 议 的 包 不 应 该 处 理。

    编 程

    ---- 每 个ISAPI 过 滤 器DLL 必 须 输 出 两 个 供IIS 使 用 的 函 数:

    ---- GetFilterVersion() 和HttpFilterProc()。 下 面 分 别 讲 述。

    ---- GetFilterVersion()

    ---- 用 于 初 始 化 和 处 理 事 件 的 登 记。 例 程 没 有 初 始 工 作 要 做, 只 是 简 单 的 登 记 了 要 处 理 的 两 个 事 件 和 其 他 一 些 标 志。

    function GetFilterVersion(var pVer:
            THTTP_FILTER_VERSION): BOOL; stdcall;
    begin
    //过滤器要处理的事件和其他一些标志
     pVer.dwFlags := (
    SF_NOTIFY_NONSECURE_PORT 
                  //过滤器只在一般端口上使用
    or SF_NOTIFY_SEND_RAW_DATA 
                      //处理发送数据事件
    or $80 // SF_NOTIFY_END_OF_REQUEST
                       处理请求结束事件
    or SF_NOTIFY_ORDER_DEFAULT
               //过滤器使用缺省优先级
    );
    //过滤器使用的版本HTTP_FILTER_REVISION
                 返回当前版本
      pVer.dwFilterVersion := HTTP_FILTER_REVISION;
    //过滤器的描述
      pVer.lpszFilterDesc[0]:='A';  pVer.lpszFilterDesc[1]:=#0;
      result:=true; //初始化成功
    end;
    
    HttpFilterProc()
    

    ---- 由IIS 回 调, 是 过 滤 器 的 实 际 处 理 部 分。
    ---- 其 中 参 数Notificationtype 是 该 调 用 的 事 件 类 型, 如 果 过 滤 器 处 理 多 个 事 件, 可 以 检 查 该 值 来 区 分 事 件。
    ---- PvNotification 是 一 个 根 据 事 件 类 型 可 变 结 构 的 参 数。 对 于SF_NOTIFY_SEND_RAW_DATA, 他 的 结 构 如 下:

    THTTP_FILTER_RAW_DATA = record
    pvInData: Pointer;    //指向数据区
    cbInData: DWORD;   //数据大小
    cbInBuffer: DWORD;   //缓冲的大小
    dwReserved: DWORD;  //保留
    end;
    

    ---- 其 中 的pvInData 就 是 要 发 送 的 数 据 指 针。 其 他 的 结 构 请 参 看 有 关 资 料 这 里 不 再 详 述。

    ---- 第 一 个 参 数var pfc: THTTP_FILTER_CONTEXT 是 过 滤 器 的 环 境 指 针, 其 中 的pFilterContext 是 一 个 用 户 使 用 的 指 针, 用 来 保 存 和 一 个HTTP 连 接 相 关 的 信 息, 这 样 过 滤 器 可 以 区 分 出 正 在 处 理 的 是 否 是 以 前 曾 处 理 过 的 连 接。 因 为 一 个 请 求 将 会 产 生 多 个SF_NOTIFY_SEND_RAW_DATA 事 件, 过 滤 器 必 须 能 够 区 分 他 们。

    ---- 程 序 的 流 程 是: 当 连 接 建 立 后,pFilterContext 被IIS 初 始 化 为NIL(0), 第 一 次SF_NOTIFY_SEND_RAW_DATA 调 用 时, 过 滤 器 要 检 查 返 回 的MIME, 如 果 不 是HTML 则 将pFilterContext 置 为pointer(2)( 将 指 针 当 作 变 量 用, 因 为 我 们 只 要 一 个 标 志), 随 后 的 发 送 事 件 调 用 将 直 接 返 回。 请 求 结 束 后, 发 生SF_NOTIFY_END_OF_REQUEST 事 件, 过 滤 程 序 将pFilterContext 复 位 为nil。

    ---- 如 果 是HTML, 则 将pFilterContext 置 为pointer(1), 随 后 的 调 用 就 将 对 数 据 进 行 内 码 的 转 换, 然 后 将pFilterContext 置 为pointer(3)。 如 果 还 有 后 续 的 调 用, 则 再 将pFilterContext 置 为pointer(1), 直 到 全 部 数 据 发 送 完 成。

    function HttpFilterProc(var pfc: THTTP_FILTER_CONTEXT; 
    Notificationtype: DWORD;  pvNotification: Pointer): DWORD; stdcall;
    var
    p:PHTTP_FILTER_RAW_DATA;
    i:integer;
    pc:pchar;
    begin
    if Notificationtype=$80 then
    //是SF_NOTIFY_END_OF_REQUEST将pFilterContext复位
      begin
      pfc.pFilterContext:=nil;
      end
    else
    begin
       p:=PHTTP_FILTER_RAW_DATA(pvNotification);
       pc:=p^.pvInData;
    case integer(pfc.pFilterContext) of
    0: //第一次调用,要检查MIME
      begin
       pfc.pFilterContext:=pointer(2);
       i:=0;
       while i
    ----
    下 面 是 完 整 的 程 序 文 件(gb2bigfiler.dpr), 其 中 的u_gb2big_tab 单 元 完 成GB 码 到BIG5, 码 的 转 换, 这 里 不 再 细 述, 有 兴 趣 的 读 者 可 以 到 后 文 提 到 的 笔 者 的 站 点 去 下 载 源 码。
    
    library gb2bigfiler;
    uses
     SysUtils,math, Classes, windows,
     isapi2, //delphi中ISAPI过滤器单元
     u_gb2big_tab; //包含将GB码转换成BIG5码的过程gb2big
    
    //下面两个函数的定义见上文
    function HttpFilterProc(...); begin ... end;
    function GetFilterVersion(...); begin ... end;
    exports
        HttpFilterProc index 1, GetFilterVersion index 2;
    Begin end.
    

    ---- 读 者 一 定 注 意 到 了, 这 个 过 滤 器 将 所 有 返 回 的HTML 都 转 换 成 了BIG5 码, 那 么GB 码 又 如 何 看 到 呢 ? 当 然 可 以 在 过 滤 器 中 检 查 一 些 环 境 变 量 来 决 定 用 户 所 要 求 的 是GB 还 是BIG5, 可 是 这 样 做 除 了 比 较 麻 烦 外, 还 存 在 效 率 问 题, 因 为 每 个 请 求 都 要 被 过 滤 器 处 理。

    ---- 笔 者 采 用 的 方 法 是 利 用IIS4.0 中 可 以 设 置 多 个 站 点 的 功 能, 设 置 两 个 站 点。 一 个 不 含 过 滤 器, 所 以GB 内 容 高 效 直 接 的 返 回 给GB 用 户; 而 另 一 个 站 点 使 用 另 外 一 个 端 口 比 如81, 所 有 虚 拟 目 录 和 前 一 个 站 点 一 样, 将 过 滤 器 加 载 在 该 站 点 上, 这 样 所 有 向81 端 口 的 请 求, 都 将 被 过 滤 器 转 换 成BIG5 码 返 回 给 用 户。

    ---- 下 面 简 述 一 下 具 体 配 置 过 程。 首 先 在delphi 中 选 择 新 建 一 个DLL, 输 入 程 序 源 码, 编 译 后 生 成gb2bigfiler.DLL 文 件。 在IIS4.0 的 管 理 控 制 台 中, 选" 新 建 站 点", 主 目 录 和 缺 省 站 点 一 样, 端 口 设 为81, 在ISAPI 过 滤 器 中 选 择" 添 加", 将gb2bigfiler.DLL 加 入。

    ---- 设 置 好 后, 可 以 浏 览81 端 口( 例 如:http://www.yoursite.com:81/your.html), 这 时 原 来GB 码 的 内 容 就 变 成 了BIG5 码 了。

    ---- 有 兴 趣 的 读 者 可 以 访 问http://202.96.122.45/qq, 起 始 页 可 以 选 择 内 码, 随 后 浏 览 的 内 容 包 括 静 态 的HTML 文 本, 放 在 数 据 库 中 的《 红 楼 梦》、《 唐 诗》、 完 全 动 态 更 新 的BBS, 都 可 以 做 到 用 两 种 内 码 显 示。

  • 相关阅读:
    C# struct 性能损失
    [leetcode] Scramble String @python
    [leetcode]Symmetric Tree @ Python
    [leetcode] Maximum Product Subarray @ python
    [leetcode]Surrounded Regions @ Python
    [leetcode]N-Queens @ Python
    [leetcode] Combinations @ Python [ask for help]
    [leetcode]Next Permutation @ Python
    [building block] merge sort @ Python
    [leetcode] Integer to Roman @ Python
  • 原文地址:https://www.cnblogs.com/fuyingke/p/200861.html
Copyright © 2020-2023  润新知