2021-10-12 03:49:01 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections.Immutable;
|
2021-10-19 03:37:10 +00:00
|
|
|
using System.Collections.ObjectModel;
|
|
|
|
using System.IO;
|
2021-10-12 03:49:01 +00:00
|
|
|
using System.Reactive.Disposables;
|
|
|
|
using System.Reactive.Subjects;
|
2021-10-19 03:37:10 +00:00
|
|
|
using System.Text;
|
2021-10-12 03:49:01 +00:00
|
|
|
using System.Threading;
|
2021-10-19 03:37:10 +00:00
|
|
|
using DynamicData;
|
2021-10-12 03:49:01 +00:00
|
|
|
using Microsoft.Extensions.Logging;
|
2021-10-19 03:37:10 +00:00
|
|
|
using Microsoft.Extensions.Primitives;
|
|
|
|
using Wabbajack.Paths;
|
|
|
|
using Wabbajack.Paths.IO;
|
2021-10-12 03:49:01 +00:00
|
|
|
|
|
|
|
namespace Wabbajack.App.Utilities;
|
|
|
|
|
|
|
|
public class LoggerProvider : ILoggerProvider
|
|
|
|
{
|
|
|
|
private Subject<ILogMessage> _messages = new();
|
|
|
|
public IObservable<ILogMessage> Messages => _messages;
|
|
|
|
|
2021-10-19 03:37:10 +00:00
|
|
|
private long _messageId = 0;
|
|
|
|
private SourceCache<ILogMessage, long> _messageLog = new(m => m.MessageId);
|
2021-10-12 03:49:01 +00:00
|
|
|
|
2021-10-19 03:37:10 +00:00
|
|
|
public readonly ReadOnlyObservableCollection<ILogMessage> _messagesFiltered;
|
|
|
|
private readonly CompositeDisposable _disposables;
|
|
|
|
private readonly Configuration _configuration;
|
|
|
|
private readonly DateTime _startupTime;
|
|
|
|
private readonly RelativePath _appName;
|
|
|
|
public AbsolutePath LogPath { get; }
|
|
|
|
private readonly Stream _logFile;
|
|
|
|
private readonly StreamWriter _logStream;
|
|
|
|
public ReadOnlyObservableCollection<ILogMessage> MessageLog => _messagesFiltered;
|
|
|
|
|
|
|
|
public LoggerProvider(Configuration configuration)
|
|
|
|
{
|
|
|
|
_startupTime = DateTime.UtcNow;
|
|
|
|
_configuration = configuration;
|
|
|
|
_configuration.LogLocation.CreateDirectory();
|
|
|
|
|
|
|
|
_disposables = new CompositeDisposable();
|
|
|
|
|
|
|
|
Messages.Subscribe(m => _messageLog.AddOrUpdate(m))
|
|
|
|
.DisposeWith(_disposables);
|
|
|
|
|
|
|
|
Messages.Subscribe(m => LogToFile(m))
|
|
|
|
.DisposeWith(_disposables);
|
|
|
|
|
|
|
|
_messageLog.Connect()
|
|
|
|
.Bind(out _messagesFiltered)
|
|
|
|
.Subscribe()
|
|
|
|
.DisposeWith(_disposables);
|
|
|
|
|
|
|
|
_messages.DisposeWith(_disposables);
|
|
|
|
|
|
|
|
_appName = typeof(LoggerProvider).Assembly.Location.ToAbsolutePath().FileName;
|
|
|
|
LogPath = _configuration.LogLocation.Combine($"{_appName}.current.log");
|
|
|
|
_logFile = LogPath.Open(FileMode.Append, FileAccess.Write, FileShare.ReadWrite);
|
|
|
|
_logFile.DisposeWith(_disposables);
|
|
|
|
|
|
|
|
_logStream = new StreamWriter(_logFile, Encoding.UTF8);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
private void LogToFile(ILogMessage logMessage)
|
2021-10-12 03:49:01 +00:00
|
|
|
{
|
2021-10-19 03:37:10 +00:00
|
|
|
var line = $"[{logMessage.TimeStamp - _startupTime}] {logMessage.LongMessage}";
|
|
|
|
lock (_logStream)
|
|
|
|
{
|
|
|
|
_logStream.Write(line);
|
|
|
|
_logStream.Flush();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private long NextMessageId()
|
|
|
|
{
|
|
|
|
return Interlocked.Increment(ref _messageId);
|
2021-10-12 03:49:01 +00:00
|
|
|
}
|
2021-10-19 03:37:10 +00:00
|
|
|
|
2021-10-12 03:49:01 +00:00
|
|
|
public void Dispose()
|
|
|
|
{
|
2021-10-19 03:37:10 +00:00
|
|
|
_disposables.Dispose();
|
2021-10-12 03:49:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public ILogger CreateLogger(string categoryName)
|
|
|
|
{
|
|
|
|
return new Logger(this, categoryName);
|
|
|
|
}
|
|
|
|
|
|
|
|
public class Logger : ILogger
|
|
|
|
{
|
|
|
|
private readonly LoggerProvider _provider;
|
|
|
|
private ImmutableList<object> Scopes = ImmutableList<object>.Empty;
|
|
|
|
private readonly string _categoryName;
|
|
|
|
|
|
|
|
public Logger(LoggerProvider provider, string categoryName)
|
|
|
|
{
|
|
|
|
_categoryName = categoryName;
|
|
|
|
_provider = provider;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Log<TState>(LogLevel logLevel, EventId eventId, TState state, Exception? exception, Func<TState, Exception?, string> formatter)
|
|
|
|
{
|
2021-10-19 03:37:10 +00:00
|
|
|
_provider._messages.OnNext(new LogMessage<TState>(DateTime.UtcNow, _provider.NextMessageId(), logLevel, eventId, state, exception, formatter));
|
2021-10-12 03:49:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public bool IsEnabled(LogLevel logLevel)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public IDisposable BeginScope<TState>(TState state)
|
|
|
|
{
|
|
|
|
Scopes = Scopes.Add(state);
|
|
|
|
return Disposable.Create(() => Scopes = Scopes.Remove(state));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public interface ILogMessage
|
|
|
|
{
|
|
|
|
long MessageId { get; }
|
|
|
|
|
|
|
|
string ShortMessage { get; }
|
2021-10-19 03:37:10 +00:00
|
|
|
DateTime TimeStamp { get; }
|
|
|
|
string LongMessage { get; }
|
2021-10-12 03:49:01 +00:00
|
|
|
}
|
|
|
|
|
2021-10-19 03:37:10 +00:00
|
|
|
record LogMessage<TState>(DateTime TimeStamp, long MessageId, LogLevel LogLevel, EventId EventId, TState State, Exception? Exception, Func<TState, Exception?, string> Formatter) : ILogMessage
|
2021-10-12 03:49:01 +00:00
|
|
|
{
|
|
|
|
public string ShortMessage => Formatter(State, Exception);
|
2021-10-19 03:37:10 +00:00
|
|
|
|
|
|
|
public string LongMessage
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
var sb = new StringBuilder();
|
|
|
|
sb.AppendLine(ShortMessage);
|
|
|
|
if (Exception != null)
|
|
|
|
{
|
|
|
|
sb.Append("Exception: ");
|
|
|
|
sb.Append(Exception);
|
|
|
|
}
|
|
|
|
|
|
|
|
return sb.ToString();
|
|
|
|
}
|
|
|
|
}
|
2021-10-12 03:49:01 +00:00
|
|
|
}
|
|
|
|
}
|