diff --git a/src/Downloader.Test/Helper/FileLogger.cs b/src/Downloader.Test/Helper/FileLogger.cs new file mode 100644 index 0000000..3382c66 --- /dev/null +++ b/src/Downloader.Test/Helper/FileLogger.cs @@ -0,0 +1,122 @@ +using Downloader.Extensions.Logging; +using System; +using System.Collections.Concurrent; +using System.IO; +using System.Threading; +using System.Threading.Tasks; + +namespace Downloader.Test.Helper; + +internal class FileLogger : ILogger, IDisposable +{ + private volatile bool _disposed = false; + private SemaphoreSlim _semaphore = new SemaphoreSlim(0); + protected readonly ConcurrentQueue LogQueue; + protected string LogPath; + protected StreamWriter LogStream; + + public FileLogger(string logPath) + { + LogQueue = new ConcurrentQueue(); + LogPath = logPath; + LogStream = new StreamWriter(FileHelper.CreateFile(logPath)); + + Task task = Task.Factory.StartNew( + function: Writer, + cancellationToken: default, + creationOptions: TaskCreationOptions.LongRunning, + scheduler: TaskScheduler.Default); + + task.Unwrap(); + } + + public void Debug(string message) + { + Log(nameof(Debug), message); + } + + public void Info(string message) + { + Log(nameof(Info), message); + } + + public void Warning(string message) + { + Log(nameof(Warning), message); + } + + public void Warning(string message, Exception exception) + { + Log(nameof(Warning), message, exception); + } + + public void Error(string message) + { + Log(nameof(Error), message); + } + + public void Error(string message, Exception exception) + { + Log(nameof(Error), message, exception); + } + + public void Fatal(string message) + { + Log(nameof(Fatal), message); + } + + public void Fatal(string message, Exception exception) + { + Log(nameof(Fatal), message, exception); + } + + protected void Log(string logType, string message, Exception exception = null) + { + if (!_disposed) + { + LogQueue.Enqueue(Formatter(logType, message, exception)); + _semaphore.Release(); + } + } + + public virtual string Formatter(string logType, string message, Exception exception) + { + var log = $"{DateTime.Now:s} | {logType} | {message}"; + if (exception is not null) + { + log += " | " + exception.Message + ": " + exception.StackTrace; + } + + return log; + } + + private async Task Writer() + { + while (!_disposed) + { + await _semaphore.WaitAsync().ConfigureAwait(false); + if (LogQueue.TryDequeue(out var log)) + { + await LogStream.WriteLineAsync(log).ConfigureAwait(false); + } + } + } + + public void Dispose() + { + _disposed = true; + LogQueue.Clear(); + LogStream?.Dispose(); + LogStream = null; + } + + public async Task FlushAsync() + { + while (!_disposed && _semaphore.CurrentCount > 0) + { + await Task.Delay(100); + } + + await (LogStream?.FlushAsync() ?? Task.CompletedTask).ConfigureAwait(false); + } +} diff --git a/src/Downloader/AbstractDownloadService.cs b/src/Downloader/AbstractDownloadService.cs index 734902f..23e6e1e 100644 --- a/src/Downloader/AbstractDownloadService.cs +++ b/src/Downloader/AbstractDownloadService.cs @@ -1,3 +1,4 @@ +using Downloader.Extensions.Logging; using System; using System.Collections.Generic; using System.ComponentModel; @@ -12,6 +13,7 @@ namespace Downloader { public abstract class AbstractDownloadService : IDownloadService, IDisposable { + protected ILogger _logger; protected SemaphoreSlim _parallelSemaphore; protected readonly SemaphoreSlim _singleInstanceSemaphore = new SemaphoreSlim(1, 1); protected CancellationTokenSource _globalCancellationTokenSource; @@ -206,8 +208,6 @@ protected void OnDownloadStarted(DownloadStartedEventArgs e) protected void OnDownloadFileCompleted(AsyncCompletedEventArgs e) { - // flush streams - Package.Flush(); Package.IsSaving = false; if (e.Cancelled) @@ -261,6 +261,11 @@ protected void OnChunkDownloadProgressChanged(object sender, DownloadProgressCha DownloadProgressChanged?.Invoke(this, totalProgressArg); } + public void AddLogger(ILogger logger) + { + _logger = logger; + } + public virtual void Dispose() { Clear().Wait(); diff --git a/src/Downloader/Extensions/Logging/ILogger.cs b/src/Downloader/Extensions/Logging/ILogger.cs new file mode 100644 index 0000000..d969b55 --- /dev/null +++ b/src/Downloader/Extensions/Logging/ILogger.cs @@ -0,0 +1,18 @@ +using System; +using System.Threading.Tasks; + +namespace Downloader.Extensions.Logging; + +public interface ILogger +{ + void Debug(string message); + void Info(string message); + void Warning(string message); + void Warning(string message, Exception exception); + void Error(string message); + void Error(string message, Exception exception); + void Fatal(string message); + void Fatal(string message, Exception exception); + string Formatter(string logType, string message, Exception exception); + Task FlushAsync(); +} diff --git a/src/Downloader/FileHelper.cs b/src/Downloader/FileHelper.cs index 0a4db16..0c88911 100644 --- a/src/Downloader/FileHelper.cs +++ b/src/Downloader/FileHelper.cs @@ -5,12 +5,12 @@ namespace Downloader { internal static class FileHelper { - public static Stream CreateFile(string filename) + public static FileStream CreateFile(string filename) { string directory = Path.GetDirectoryName(filename); if (string.IsNullOrWhiteSpace(directory)) { - return Stream.Null; + return null; } if (Directory.Exists(directory) == false)