使用 C# 将 AppDomain 映射到一个友好的名称
Map AppDomain into a friendly name using C#
所以我有一个 IIS10 网络服务器,运行 相当多的应用程序,例如:
+ Default Web Site
-First Application
-Second Application
-Third Application
我们正在使用 nLog 将内容记录到数据库,其中之一是当前的应用程序域
${appdomain:format=Format}
如 https://github.com/NLog/NLog/wiki/AppDomain-Layout-Renderer 中所述。
这很好,除了当我查看我们的合并日志时,我看到 appdomain 记录为:
0002:/LM/W3SVC/2/ROOT-1-132036049222959352
我想将其转换为 "Third Application"(因此当我创建报告时,它将显示为 "Third Application" 而不是 0002:/LM/W3SVC/2/ROOT-1-132036049222959352。)
条件是:
- 我无法更改布局渲染器。
- 我无法更改应用程序的代码。
- 我必须在记录数据后执行此操作。我假设/LM/W3SVC/2/ROOT是我需要转化的,我在app里做不到,只能在reporting里做。
我试过 Microsoft.Web.Administration,像这样:
using (ServerManager sm = new ServerManager()){
var thisVal = (from s in sm.Sites
from app in s.Applications
from vDir in app.VirtualDirectories
where vDir.Path.Equals(currVal,StringComparison.InvariantCultureIgnoreCase)
select new KeyValuePair<string, string>(currVal, vDir.Path));
}
但是我在任何地方都找不到应用程序根目录。我相信它与 https://docs.microsoft.com/en-us/previous-versions/iis/6.0-sdk/ms524308(v=vs.90) 相同,但那是 IIS 元数据库,它显然已经过时且已损坏。
(上面的代码将 运行 与应用程序在同一台机器上,所以我不必担心找到正确的服务器,只需翻译名称)
您可以像这样枚举(并因此过滤)AppDomain 的 ID:
using (ServerManager s = new ServerManager())
{
var processList = from m in Process.GetProcessesByName("w3wp") select m.Id;
IEnumerable<WorkerProcess> workerP = (from p in s.ApplicationPools
from w in p.WorkerProcesses
where processList.Any(vn => vn == w.ProcessId)
select w);
foreach (var workProcess in workerP)
{
Console.WriteLine(
$"{workProcess.AppPoolName} Id:{workProcess.ProcessId} AppDomains Count: {workProcess.ApplicationDomains?.Count}");
var ad = workProcess.ApplicationDomains;
if (!ad.Any()) continue;
foreach (var curDomain in ad)
{
Console.WriteLine($"AppDomain: {curDomain.Id} {curDomain.VirtualPath}");
}
}
}
这将为您提供显示 AppDomain 的 ID 的输出,您可以从以下位置获取 vdir:
DefaultAppPool Id:33180 AppDomains Count: 1
AppDomain: /LM/W3SVC/1/ROOT/
所以我有一个 IIS10 网络服务器,运行 相当多的应用程序,例如:
+ Default Web Site
-First Application
-Second Application
-Third Application
我们正在使用 nLog 将内容记录到数据库,其中之一是当前的应用程序域
${appdomain:format=Format}
如 https://github.com/NLog/NLog/wiki/AppDomain-Layout-Renderer 中所述。
这很好,除了当我查看我们的合并日志时,我看到 appdomain 记录为:
0002:/LM/W3SVC/2/ROOT-1-132036049222959352
我想将其转换为 "Third Application"(因此当我创建报告时,它将显示为 "Third Application" 而不是 0002:/LM/W3SVC/2/ROOT-1-132036049222959352。)
条件是:
- 我无法更改布局渲染器。
- 我无法更改应用程序的代码。
- 我必须在记录数据后执行此操作。我假设/LM/W3SVC/2/ROOT是我需要转化的,我在app里做不到,只能在reporting里做。
我试过 Microsoft.Web.Administration,像这样:
using (ServerManager sm = new ServerManager()){
var thisVal = (from s in sm.Sites
from app in s.Applications
from vDir in app.VirtualDirectories
where vDir.Path.Equals(currVal,StringComparison.InvariantCultureIgnoreCase)
select new KeyValuePair<string, string>(currVal, vDir.Path));
}
但是我在任何地方都找不到应用程序根目录。我相信它与 https://docs.microsoft.com/en-us/previous-versions/iis/6.0-sdk/ms524308(v=vs.90) 相同,但那是 IIS 元数据库,它显然已经过时且已损坏。
(上面的代码将 运行 与应用程序在同一台机器上,所以我不必担心找到正确的服务器,只需翻译名称)
您可以像这样枚举(并因此过滤)AppDomain 的 ID:
using (ServerManager s = new ServerManager())
{
var processList = from m in Process.GetProcessesByName("w3wp") select m.Id;
IEnumerable<WorkerProcess> workerP = (from p in s.ApplicationPools
from w in p.WorkerProcesses
where processList.Any(vn => vn == w.ProcessId)
select w);
foreach (var workProcess in workerP)
{
Console.WriteLine(
$"{workProcess.AppPoolName} Id:{workProcess.ProcessId} AppDomains Count: {workProcess.ApplicationDomains?.Count}");
var ad = workProcess.ApplicationDomains;
if (!ad.Any()) continue;
foreach (var curDomain in ad)
{
Console.WriteLine($"AppDomain: {curDomain.Id} {curDomain.VirtualPath}");
}
}
}
这将为您提供显示 AppDomain 的 ID 的输出,您可以从以下位置获取 vdir:
DefaultAppPool Id:33180 AppDomains Count: 1
AppDomain: /LM/W3SVC/1/ROOT/