using System; using System.Globalization; using System.IO; using System.Text; using ZeroLevel.Services.Encryption; namespace ZeroLevel.Logging { public class EncryptedFileLogOptions { public EncryptedFileLogOptions() { Folder = null!; LimitFileSize = 0; } internal string Key { get; private set; } internal string Folder { get; private set; } internal long LimitFileSize { get; private set; } public EncryptedFileLogOptions SetKey(string key) { this.Key = key; return this; } public EncryptedFileLogOptions SetFolderPath(string folder) { if (folder.IndexOf(':') < 0) { this.Folder = Path.Combine(Configuration.BaseDirectory, folder); } else { this.Folder = folder; } return this; } public EncryptedFileLogOptions SetMaximumFileSizeInKb(long size) { this.LimitFileSize = size; return this; } } public class EncryptedFileLog : ILogger { #region Fields private readonly EncryptedFileLogOptions _options; private int _todayCountLogFiles = 0; /// /// Current log file /// private string _currentLogFile; /// /// Stream to output to file /// private Stream _writer; /// /// Lock on re-create file /// private readonly object _fileRecreating = new object(); private long _currentLogSize = 0; private readonly long _taskRename = -1; private readonly FastObfuscator _obfuscator; #endregion Fields public EncryptedFileLog(EncryptedFileLogOptions options) { if (options == null!) throw new ArgumentNullException(nameof(options)); if (string.IsNullOrEmpty(options.Key)) throw new ArgumentNullException("options.Key"); _options = options; if (string.IsNullOrWhiteSpace(_options.Folder)) { _options.SetFolderPath(Path.Combine(Configuration.BaseDirectory, "logs")); } if (!Directory.Exists(_options.Folder)) { var dir = Directory.CreateDirectory(_options.Folder); if (dir.Exists == false) { throw new ArgumentException($"Can't create or found directory '{_options.Folder}'"); } } _obfuscator = new FastObfuscator(options.Key); RecreateLogFile(); _taskRename = Sheduller.RemindEveryNonlinearPeriod(() => (DateTime.Now.AddDays(1).Date - DateTime.Now).Add(TimeSpan.FromMilliseconds(100)), RecreateLogFile); } #region Utils /// /// Checking the name of the log file (changes when the date changes to the next day) /// private void RecreateLogFile() { lock (_fileRecreating) { var nextFileName = GetNextFileName(); CloseCurrentWriter(); Stream stream = null!; try { _currentLogFile = nextFileName; _writer = new FileStream(_currentLogFile, FileMode.Append, FileAccess.Write, FileShare.Read); } catch { if (stream != null!) { stream.Dispose(); } throw; } } } /// /// Closing the current log /// private void CloseCurrentWriter() { if (_writer != null!) { _writer.Dispose(); _writer = null!; } } private string GetNextFileName() { string fileName = Path.Combine(_options.Folder, string.Format(CultureInfo.CurrentCulture, "{0:yyyyMMdd}_{1:D4}.bin", DateTime.Now, _todayCountLogFiles)); if (_options.LimitFileSize > 0) { if (!File.Exists(fileName)) { _todayCountLogFiles = 0; } else { while (File.Exists(fileName)) { var length = (new FileInfo(fileName).Length >> 10); if (length >= _options.LimitFileSize) { _todayCountLogFiles++; fileName = Path.Combine(_options.Folder, string.Format(CultureInfo.CurrentCulture, "{0:yyyyMMdd}_{1:D4}.bin", DateTime.Now, _todayCountLogFiles)); } else { break; } } } } return fileName; } #endregion Utils #region IDisposable private bool _disposed = false; public void Dispose() { Sheduller.Remove(_taskRename); if (false == _disposed) { _disposed = true; CloseCurrentWriter(); } } #endregion IDisposable #region ILog public void Write(LogLevel level, string message) { if (false == _disposed) { byte[] data; if (level == LogLevel.Raw) { data = Encoding.UTF8.GetBytes(message); } else { data = Encoding.UTF8.GetBytes($"[{DateTime.Now.ToString("dd.MM.yyyy HH:mm:ss")} {LogLevelNameMapping.CompactName(level)}]\t{message}\r\n"); } _obfuscator.HashData(data); lock (_fileRecreating) { _writer.Write(BitConverter.GetBytes(data.Length), 0, 4); _writer.Write(data, 0, data.Length); } _currentLogSize += data.Length; if (_options.LimitFileSize > 0 && _currentLogSize >> 10 >= _options.LimitFileSize) { RecreateLogFile(); } } } #endregion ILog } }