Application Insights - аргументы ILogger, отображаемые как имя объекта в пользовательских измерениях
Объекты отображаются в виде строк (имя объекта) в пользовательских измерениях Application Insights при передаче в качестве аргументов в ilogger. Фактические значения не показаны.
Зарегистрировать Application Insights
services.AddApplicationInsightsTelemetry();
Новый журнал
public class HealthController : ControllerBase
{
private readonly ILogger<HealthController> _logger;
public HealthController(ILogger<HealthController> logger)
{
_logger = logger;
}
[HttpGet]
public IActionResult Get()
{
var health = new HealthViewModel()
{
ok = false
};
_logger.LogInformation("Hlep me pls {health}", health);
return Ok(health);
}
}
Результат
Я не хочу этого для каждого журнала:
var health = new HealthViewModel()
{
ok = false
};
_logger.LogInformation("Hlep me pls {health}", JsonConvert.SerializeObject(health));
Я попытался создать промежуточное ПО для анализа приложений, но значением по-прежнему является имя объекта.
Почему аргументы не отображаются как json?
редактировать
Это похоже на
var health = new
{
ok = false
};
_logger.LogInformation("HEJ2 {health}", health);
работает, но не
var health = new HealthViewModel
{
ok = false
};
_logger.LogInformation("HEJ2 {health}", health);
2 ответа
Не поддерживается
Цитата из https://github.com/microsoft/ApplicationInsights-dotnet/issues/1722
Я думаю, вы слишком многого ожидаете от регистратора. Он не знает о формате JSON, он просто вызывает Convert.ToString для свойств
Convert.ToString обычно вызывает ToString(), а реализация ToString по умолчанию для новых классов просто возвращает имя типа.
Что ты можешь сделать
Используйте ToJson() для объектов, зарегистрированных в ILogger, и создайте промежуточное программное обеспечение для анализа приложений и измените имя журнала и настраиваемые измерения.
Промежуточное ПО
public class ProcessApiTraceFilter : ITelemetryProcessor
{
private ITelemetryProcessor Next { get; set; }
private readonly IIdentity _identity;
private readonly IHostEnvironment _hostEnvironment;
public ProcessApiTraceFilter(ITelemetryProcessor next, IHostEnvironment hostEnvironment, IIdentity identity)
{
Next = next;
_identity = identity;
_hostEnvironment = hostEnvironment;
}
public void Process(ITelemetry item)
{
item.Process(_hostEnvironment, _identity);
Next.Process(item);
}
}
Реализация
public static class ApplicationInsightsExtensions
{
public static void Process(this ITelemetry item, IHostEnvironment hostEnvironment, IIdentity identity)
{
if (item is TraceTelemetry)
{
var traceTelemetry = item as TraceTelemetry;
var originalMessage = traceTelemetry.Properties.FirstOrDefault(x => x.Key == "{OriginalFormat}");
if (!string.IsNullOrEmpty(originalMessage.Key))
{
var reg = new Regex("{([A-z]*)*}", RegexOptions.Compiled);
var match = reg.Matches(originalMessage.Value);
var formattedMessage = originalMessage.Value;
foreach (Match arg in match)
{
var parameterName = arg.Value.Replace("{", "").Replace("}", "");
var parameterValue = traceTelemetry.Properties.FirstOrDefault(x => x.Key == parameterName);
formattedMessage = formattedMessage.Replace(arg.Value, "");
}
traceTelemetry.Message = formattedMessage.Trim();
}
if (identity != null)
{
var isAuthenticated = identity.IsAuthenticated();
const string customerKey = "customer";
if (isAuthenticated && !traceTelemetry.Properties.ContainsKey(customerKey))
{
var customer = identity.Customer();
if (customer != null)
{
traceTelemetry.Properties.Add(customerKey, customer.ToJson());
}
}
var request = identity.Request();
const string requestKey = "request";
if (request != null && !traceTelemetry.Properties.ContainsKey(requestKey))
{
traceTelemetry.Properties.Add(requestKey, request.ToJson());
}
}
var applicationNameKey = "applicationName";
if (hostEnvironment != null && !string.IsNullOrEmpty(hostEnvironment.ApplicationName) && !traceTelemetry.Properties.ContainsKey(applicationNameKey))
{
traceTelemetry.Properties.Add(applicationNameKey, hostEnvironment.ApplicationName);
}
}
}
}
Зарегистрируйте аналитику приложений и промежуточное ПО при запуске
services.AddApplicationInsightsTelemetry();
services.AddApplicationInsightsTelemetryProcessor<ProcessApiTraceFilter>();
ToJson
public static class ObjectExtensions
{
private static readonly string Null = "null";
private static readonly string Exception = "Could not serialize object to json";
public static string ToJson(this object value, Formatting formatting = Formatting.None)
{
if (value == null) return Null;
try
{
string json = JsonConvert.SerializeObject(value, formatting);
return json;
}
catch (Exception ex)
{
return $"{Exception} - {ex?.Message}";
}
}
}
Журнал
//Log object? _smtpAppSettings.ToJson()
_logger.LogInformation("Email sent {to} {from} {subject}", to, _smtpAppSettings.From, subject)
Результат
Из ваших пользовательских размеров я вижу, что он не рассматривает параметр здоровья как дополнительные данные
_logger.LogInformation("Hlep me pls {health}", health);
пытаясь использовать jsonConverter внутри самой строки.
_logger.LogInformation($"Hlep me pls {JsonConvert.SerializeObject(health)}");