3

我有一个 C++ 应用程序需要检索 IIS 7 站点的属性(例如类似于 IIS6 中的元数据库属性PathAppFriendlyName)。

使用 IIS 7,我的代码执行以下操作:

  1. 获取AppHostWritableAdminManager并提交路径MACHINE/WEBROOT/APPHOST/Default Web Site/
  2. GetAdminSection使用部分名称调用appSettings
  3. 然后查看返回的集合并查找属性(Path例如)。

这适用于 IIS 6,但不适用于 IIS7/7.5。

为了完成这项工作,我需要进行哪些更改?

4

1 回答 1

8

在 IIS7 中,配置数据不存储在“元数据库”中,而且我们在 IIS6 中习惯的元数据库属性也不相同。IIS7 将其大部分配置数据存储在以下文件中:

%systemroot%\System32\InetSrv\Config\applicationHost.config

还有其他文件,但为了回答这个问题,这是我们感兴趣的文件。

的文档applicationHost.config可以在这里找到:

<system.applicationHost>- IIS.NET
配置元素 [IIS 7 设置架构]
system.applicationHost 部分组 [IIS 7 设置架构]

您可以在此处找到 IIS6 元数据库 -> IIS7 XML 配置映射列表:

将元数据库属性转换为配置设置 [IIS 7]

例如,在 IIS6 中,站点的路径存储/root在. IE:PathIIsWebVirtualDir

<IIsWebServer Location="/LM/W3SVC/67793744" AuthFlags="0" ServerAutoStart="TRUE" 
              ServerBindings="217.69.47.170:80:app2.dev" ServerComment="app2" /> 
<IIsWebVirtualDir Location="/LM/W3SVC/67793744/root" 
    AccessFlags="AccessRead | AccessScript" 
    AppFriendlyName="Default Application" 
    AppIsolated="2" 
    AppRoot="/LM/W3SVC/67793744/Root" 
    AuthFlags="AuthAnonymous | AuthNTLM" 
    DirBrowseFlags="DirBrowseShowDate | DirBrowseShowTime | DirBrowseShowSize |
            DirBrowseShowExtension | DirBrowseShowLongDate | EnableDefaultDoc" 
    Path="D:\websites\ssl-test\www\kerboom" 
    ScriptMaps="...">

但在 IIS7 中,它的存储方式不同:

<sites>
    <site name="Default Web Site" id="1" serverAutoStart="true">
        <!-- this is the functional equivalent of the /root app in IIS6 -->
        <application path="/">
            <virtualDirectory path="/" 
                              physicalPath="%SystemDrive%\inetpub\wwwroot" />
        </application>
    </site>
<sites>

但是,如果您的代码必须同时使用 IIS6 和 IIS7,那么您可以安装 IIS6 管理兼容性组件。这将允许您使用传统的 IIS6 元数据库 API(如 ADSI、System.DirectoryServices 等)访问 IIS7 站点属性。兼容层将为您将这些属性映射到新的 IIS7 架构。

本文的第一部分解释了如何在 Vista/Windows7/Windows 2008 上安装它:

如何在 Vista 和 Windows 2008 上使用 IIS7 安装 ASP.NET 1.1 - 请参阅步骤 #1

更新:

不幸的是,C++ 不是我的强项。但是,我在 C# 中使用 COM Interop 编写了一个示例来演示如何使用AppHostWritableAdminManager

IAppHostWritableAdminManager wam = new AppHostWritableAdminManager();
IAppHostElement sites = 
   wam.GetAdminSection("system.applicationHost/sites", "MACHINE/WEBROOT/APPHOST");
IAppHostElementCollection sitesCollection = sites.Collection;

long index = FindSiteIndex(sitesCollection, "MySite");
if(index == -1) throw new Exception("Site not found");

IAppHostElement site = sitesCollection[index];
IAppHostElementCollection bindings = site.ChildElements["bindings"].Collection;

for (int i = 0; i < bindings.Count; i++)
{
  IAppHostElement binding = bindings[i];
  IAppHostProperty protocolProp = binding.GetPropertyByName("protocol");
  IAppHostProperty bindingInformationProp = 
      binding.GetPropertyByName("bindingInformation");

  string protocol = protocolProp.Value;
  string bindingInformation = bindingInformationProp.Value;

  Debug.WriteLine("{0} - {1}", protocol, bindingInformation);

}

static long FindSiteIndex(IAppHostElementCollection sites, string siteName)
{
  for (int i = 0; i < sites.Count; i++)
  {
    IAppHostElement site = sites[i];
    Debug.WriteLine(site.Name);
    IAppHostProperty prop = site.GetPropertyByName("name");
    if(prop.Value == siteName)
    {
      return i;
    }
  }
  return -1;
}

<sites>上面的代码在集合中定位了一个名为“MySite”的站点。然后它检索站点的<bindings>集合并打印每个绑定protocolbindingInformation属性。

您应该能够相当容易地将其转换为 C++。

要回答您评论中的问题 -

例如,路径 system.applicationHost/Sites 将使我进入站点列表。有没有绝对的方法可以像这样访问我的服务器绑定(例如通过执行 system.applicationHost/Sites/Default Web Site/Bindings

使用 时,AppHostWritableAdminManager没有直接访问您要检查/修改的站点或其属性的快捷方式。在上面的示例中,您会看到我需要遍历站点集合以找到我感兴趣的站点。这样做的原因是AppHostWritableAdminManager将所有内容都视为元素和元素的集合。这是一个相当基本的 API。即使在使用托管Microsoft.Web.AdministrationAPI 时,您也会发现虽然有一些不错的属性,例如Site.Bindings,但这些只是伪装的包装器AppHostWritableAdminManager

事实上,如果我想找到一个站点,我仍然必须Sites在 for 循环中搜索集合,或者如果我使用的是 C#3.5 或更高版本,则需要添加一些 LINQ 糖:

using(ServerManager serverManager = new ServerManager())
{
    Site x = serverManager.Sites.FirstOrDefault(s => s.Name == "MySite");
}

Site的基类是ConfigurationElement在引擎盖下封装了对IAppHostElement.

一旦您通过了一些基本的快捷方式包装属性,我们在托管代码中配置 IIS(例如 IIS FTP)的大部分工作就是元素、属性和元素集合。

更新 2:

请记住,我一生中从未写过一行 C++。没有清除字符串或对象:

#include "stdafx.h"
#include <windows.h>
#include <stdio.h>
#include <ahadmin.h>
#include <crtdbg.h>

static IAppHostElement* 
   FindSite(IAppHostElementCollection *pCollection, BSTR bstrSiteName);

int _tmain(int argc, _TCHAR* argv[])
{
  CoInitialize(NULL);

  IAppHostWritableAdminManager *pMgr = NULL;
  IAppHostElement *pElem = NULL;
  IAppHostElementCollection *pSitesCollection = NULL;
  IAppHostElement *pSite = NULL;
  IAppHostElement *pBindings = NULL;
  IAppHostElement *pBinding = NULL;
  IAppHostElementCollection *pBindingsCollection = NULL;
  IAppHostChildElementCollection *pChildElements = NULL;
  IAppHostProperty *pProtocol = NULL;
  IAppHostProperty *pBindingInformation = NULL;

  BSTR bstrSectionName = SysAllocString( L"system.applicationHost/sites" );
  BSTR bstrConfigCommitPath = SysAllocString( L"MACHINE/WEBROOT/APPHOST" );
  BSTR bstrSiteName = SysAllocString( L"MySite" );
  BSTR bstrBindingsConst = SysAllocString( L"bindings" );
  BSTR bstrBindingProtocol = SysAllocString( L"protocol" );
  BSTR bstrBindingInformation = SysAllocString( L"bindingInformation" );

  VARIANT vtPropertyName;
  VARIANT vtIndex;

  HRESULT hr = S_OK;

  hr = CoCreateInstance( __uuidof(AppHostWritableAdminManager), NULL, 
      CLSCTX_INPROC_SERVER, __uuidof(IAppHostWritableAdminManager), (void**) &pMgr);

  hr = pMgr->GetAdminSection(bstrSectionName, bstrConfigCommitPath, &pElem);
  hr = pElem->get_Collection(&pSitesCollection);

  pSite = FindSite(pSitesCollection, bstrSiteName);

  hr = pSite->get_ChildElements(&pChildElements);

  vtPropertyName.vt = VT_BSTR;
  vtPropertyName.bstrVal = bstrBindingsConst;

  hr = pChildElements->get_Item(vtPropertyName, &pBindings);
  hr = pBindings->get_Collection(&pBindingsCollection);

  DWORD bindingsCount;
  hr = pBindingsCollection->get_Count(&bindingsCount);

  for(int i = 0; i < bindingsCount; i++)
  {
    vtIndex.lVal = i;
    vtIndex.vt = VT_I4;
    hr = pBindingsCollection->get_Item(vtIndex, &pBinding);

    hr = pBinding->GetPropertyByName(bstrBindingProtocol, &pProtocol);
    hr = pBinding->GetPropertyByName(bstrBindingInformation, &pBindingInformation);

    BSTR bstrProtocol;
    BSTR bstrBindingInformation;

    hr = pProtocol->get_StringValue(&bstrProtocol);
    hr = pBindingInformation->get_StringValue(&bstrBindingInformation);

    _tprintf(_T("Protocol: %s, BindingInfo: %s\n"), bstrProtocol, bstrBindingInformation);
  }

  CoUninitialize();
  return 0;
}

IAppHostElement* FindSite(IAppHostElementCollection *pCollection, BSTR bstrSiteName)
{
  DWORD count = -1;
  pCollection->get_Count(&count);

  BSTR bstrPropName = SysAllocString( L"name");

  for(DWORD i = 0; i < count; i++)
  {
    IAppHostElement *site = NULL;
    IAppHostProperty *prop = NULL;
    BSTR bstrPropValue;

    HRESULT hr = S_OK;

    VARIANT vtCount;
    VariantInit(&vtCount);
    vtCount.lVal = i;
    vtCount.vt = VT_I4;

    hr = pCollection->get_Item(vtCount, &site);
    hr = site->GetPropertyByName(bstrPropName, &prop);
    hr = prop->get_StringValue(&bstrPropValue);

    if(wcscmp(bstrPropValue, bstrSiteName) == 0)
    {
      return site;
    }
  }

  return NULL;
}
于 2011-01-23T02:06:50.537 回答