Files
ThingsGateway/src/Admin/ThingsGateway.Furion/Logging/Implantations/File/FileLogger.cs
2248356998 qq.com b989aa5561 10.6.21
2025-05-26 00:05:16 +08:00

136 lines
4.6 KiB
C#
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

// ------------------------------------------------------------------------
// 版权信息
// 版权归百小僧及百签科技(广东)有限公司所有。
// 所有权利保留。
// 官方网站https://baiqian.com
//
// 许可证信息
// 项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。
// 许可证的完整文本可以在源代码树根目录中的 LICENSE-APACHE 和 LICENSE-MIT 文件中找到。
// ------------------------------------------------------------------------
using Microsoft.Extensions.Logging;
namespace ThingsGateway.Logging;
/// <summary>
/// 文件日志记录器
/// </summary>
/// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks>
[SuppressSniffer]
public sealed class FileLogger : ILogger, IDisposable
{
/// <summary>
/// 记录器类别名称
/// </summary>
private readonly string _logName;
/// <summary>
/// 文件日志记录器提供器
/// </summary>
private readonly FileLoggerProvider _fileLoggerProvider;
/// <summary>
/// 日志配置选项
/// </summary>
private readonly FileLoggerOptions _options;
/// <summary>
/// 构造函数
/// </summary>
/// <param name="logName">记录器类别名称</param>
/// <param name="fileLoggerProvider">文件日志记录器提供器</param>
public FileLogger(string logName, FileLoggerProvider fileLoggerProvider)
{
_logName = logName;
_fileLoggerProvider = fileLoggerProvider;
_options = fileLoggerProvider.LoggerOptions;
}
/// <summary>
/// 开始逻辑操作范围
/// </summary>
/// <typeparam name="TState">标识符类型参数</typeparam>
/// <param name="state">要写入的项/对象</param>
/// <returns><see cref="IDisposable"/></returns>
public IDisposable BeginScope<TState>(TState state)
{
return _fileLoggerProvider.ScopeProvider?.Push(state);
}
public void Dispose()
{
_fileLoggerProvider.RemoveCache(_logName);
}
/// <summary>
/// 检查是否已启用给定日志级别
/// </summary>
/// <param name="logLevel">日志级别</param>
/// <returns><see cref="bool"/></returns>
public bool IsEnabled(LogLevel logLevel)
{
return logLevel >= _options.MinimumLevel;
}
/// <summary>
/// 写入日志项
/// </summary>
/// <typeparam name="TState">标识符类型参数</typeparam>
/// <param name="logLevel">日志级别</param>
/// <param name="eventId">事件 Id</param>
/// <param name="state">要写入的项/对象</param>
/// <param name="exception">异常对象</param>
/// <param name="formatter">日志格式化器</param>
/// <exception cref="ArgumentNullException"></exception>
public void Log<TState>(LogLevel logLevel
, EventId eventId
, TState state
, Exception exception
, Func<TState, Exception, string> formatter)
{
// 判断日志级别是否有效
if (!IsEnabled(logLevel)) return;
// 检查日志格式化器
if (formatter == null) throw new ArgumentNullException(nameof(formatter));
// 获取格式化后的消息
var message = formatter(state, exception);
// 日志消息内容转换(如脱敏处理)
if (_options.MessageProcess != null)
{
message = _options.MessageProcess(message);
}
var logDateTime = _options.UseUtcTimestamp ? DateTime.UtcNow : DateTime.Now;
var logMsg = new LogMessage(_logName, logLevel, eventId, message, exception, null, state, logDateTime, Environment.CurrentManagedThreadId, _options.UseUtcTimestamp, App.GetTraceId())
{
// 设置日志上下文
Context = Penetrates.SetLogContext(_fileLoggerProvider.ScopeProvider, _options.IncludeScopes)
};
// 判断是否自定义了日志筛选器,如果是则检查是否符合条件
if (_options.WriteFilter?.Invoke(logMsg) == false)
{
logMsg.Context?.Dispose();
return;
}
// 设置日志消息模板
logMsg.Message = _options.MessageFormat != null
? _options.MessageFormat(logMsg)
: Penetrates.OutputStandardMessage(logMsg, _options.DateFormat, withTraceId: _options.WithTraceId, withStackFrame: _options.WithStackFrame, provider: _options.FormatProvider);
// 空检查
if (logMsg.Message is null)
{
logMsg.Context?.Dispose();
return;
}
// 写入日志队列
_fileLoggerProvider.WriteToQueue(logMsg);
}
}