在 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:Path
IIsWebVirtualDir
<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>
集合并打印每个绑定protocol
和bindingInformation
属性。
您应该能够相当容易地将其转换为 C++。
要回答您评论中的问题 -
例如,路径 system.applicationHost/Sites 将使我进入站点列表。有没有绝对的方法可以像这样访问我的服务器绑定(例如通过执行 system.applicationHost/Sites/Default Web Site/Bindings
使用 时,AppHostWritableAdminManager
没有直接访问您要检查/修改的站点或其属性的快捷方式。在上面的示例中,您会看到我需要遍历站点集合以找到我感兴趣的站点。这样做的原因是AppHostWritableAdminManager
将所有内容都视为元素和元素的集合。这是一个相当基本的 API。即使在使用托管Microsoft.Web.Administration
API 时,您也会发现虽然有一些不错的属性,例如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;
}