using System.Diagnostics; using Observatory.Framework; using Observatory.Framework.Files; using Observatory.Framework.Files.Journal; namespace Pulsar.Utils; using JournalEvent = (Exception ex, string file, JsonObject line); class LogMonitor { #region Singleton Instantiation public static LogMonitor GetInstance => _instance.Value; private static readonly Lazy _instance = new(NewLogMonitor); private static LogMonitor NewLogMonitor() { return new LogMonitor(); } private LogMonitor() { currentLine = new(); InitializeWatchers(string.Empty); SetLogMonitorState(LogMonitorState.Idle); } #endregion #region Public properties public LogMonitorState CurrentState => currentState; public Status Status { get; private set; } #endregion #region Public Methods public void Start() { if (firstStartMonitor) { // Only pre-read on first start monitor. Beyond that it's simply pause/resume. firstStartMonitor = false; PrereadJournals(); } journalWatcher.EnableRaisingEvents = true; statusWatcher.EnableRaisingEvents = true; SetLogMonitorState(LogMonitorState.Realtime); JournalPoke(); } public void Stop() { journalWatcher.EnableRaisingEvents = false; statusWatcher.EnableRaisingEvents = false; SetLogMonitorState(LogMonitorState.Idle); } public void ChangeWatchedDirectory(string path) { journalWatcher.Dispose(); statusWatcher.Dispose(); InitializeWatchers(path); } public bool IsMonitoring() { return currentState.HasFlag(LogMonitorState.Realtime); } // TODO(fredjk_gh): Remove? public bool ReadAllInProgress() { return LogMonitorStateChangedEventArgs.IsBatchRead(currentState); } public Func> ReadAllGenerator(out int fileCount) { // Prevent pre-reading when starting monitoring after reading all. firstStartMonitor = false; SetLogMonitorState(currentState | LogMonitorState.BatchProcessing); var logDirectory = GetJournalFolder(); var files = GetJournalFilesOrdered(logDirectory); fileCount = files.Count(); IEnumerable ReadAllJournals() { var readErrors = new List(); foreach (var file in files) { yield return file.Name; readErrors.AddRange(ProcessJournal(ReadByLines(file.FullName), file.Name)); } ReportErrors(readErrors); SetLogMonitorState(currentState & ~LogMonitorState.BatchProcessing); } ; return ReadAllJournals; } public void PrereadJournals() { SetLogMonitorState(currentState | LogMonitorState.Init); var logDirectory = GetJournalFolder(); var files = GetJournalFilesOrdered(logDirectory).ToList(); // Read at most the last two files (in case we were launched after the game and the latest // journal is mostly empty) but keeping only the lines since the last FSDJump. //TODO: strongly type these List lastSystemLines = new(); List lastFileLines = new(); List fileHeaderLines = new(); var sawFSDJump = false; foreach (var file in files.Skip(Math.Max(files.Count - 2, 0))) { var lines = ReadByLines(file.FullName); foreach (var line in lines) { var eventType = JournalUtilities.GetEventType(line); if (eventType == "FSDJump" || eventType == "CarrierJump" && ((line["Docked"]?.GetValue() ?? false) || (line["OnFoot"]?.GetValue() ?? false))) { // Reset, start collecting again. lastSystemLines.Clear(); sawFSDJump = true; } else if (eventType == "Fileheader") { lastFileLines.Clear(); fileHeaderLines.Clear(); fileHeaderLines.Add(line); } else if (eventType == "LoadGame" || eventType == "Statistics") { // A few header lines to collect. fileHeaderLines.Add(line); } lastSystemLines.Add(line); lastFileLines.Add(line); } } // If we didn't see a jump in the recent logs (Cmdr is stationary in a system for a while // ie. deep-space mining from a carrier), at very least, read from the beginning of the // current journal file which includes the important stuff like the last "LoadGame", etc. This // also helps out in cases where one forgets to hit "Start Monitor" until part-way into the // session (if auto-start is not enabled). var linesToRead = lastFileLines; if (sawFSDJump) { // If we saw any relevant header lines, insert them as well. This ensures odyssey biologicials are properly // counted/presented, current Commander name is present, etc. if (fileHeaderLines.Count > 0) { lastSystemLines.InsertRange(0, fileHeaderLines); } linesToRead = lastSystemLines; } ReportErrors(ProcessJournal(linesToRead, "Pre-read")); SetLogMonitorState(currentState & ~LogMonitorState.Init); } #endregion #region Public Events public event EventHandler? LogMonitorStateChanged; public event EventHandler? JournalEntry; public event EventHandler? StatusUpdate; #endregion #region Private Fields private FileSystemWatcher? journalWatcher; private FileSystemWatcher? statusWatcher; private readonly Dictionary journalTypes; private readonly Dictionary currentLine; private LogMonitorState currentState = LogMonitorState.Idle; // Change via #SetLogMonitorState private bool firstStartMonitor = true; private readonly string[] EventsWithAncillaryFile = { "Cargo", "NavRoute", "Market", "Outfitting", "Shipyard", "Backpack", "FCMaterials", "ModuleInfo", "ShipLocker" }; #endregion #region Private Methods private void SetLogMonitorState(LogMonitorState newState) { var oldState = currentState; currentState = newState; LogMonitorStateChanged?.Invoke(this, new LogMonitorStateChangedEventArgs { PreviousState = oldState, NewState = newState }); ; Debug.WriteLine("LogMonitor State change: {0} -> {1}", oldState, newState); } private void InitializeWatchers(string path) { var logDirectory = GetJournalFolder(); journalWatcher = new FileSystemWatcher(logDirectory, "Journal.*.??.log") { NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.Size | NotifyFilters.FileName | NotifyFilters.CreationTime }; journalWatcher.Changed += LogChangedEvent; journalWatcher.Created += LogCreatedEvent; statusWatcher = new FileSystemWatcher(logDirectory, "Status.json") { NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.Size }; statusWatcher.Changed += StatusUpdateEvent; } private static string GetJournalFolder() { throw new NotImplementedException(); } private List ProcessJournal(IEnumerable lines, string file) { var readErrors = new List<(Exception ex, string file, JsonObject line)>(); foreach (var line in lines) { try { DeserializeAndInvoke(line); } catch (Exception ex) { readErrors.Add(ex, file, line); } } return readErrors; } private void DeserializeAndInvoke(JsonObject? line) { var eventType = JournalUtilities.GetEventType(line); var journalEvent = new JournalEventArgs(eventType, line); JournalEntry?.Invoke(this, journalEvent); // Files are only valid if realtime, otherwise they will be stale or empty. if (!currentState.HasFlag(LogMonitorState.BatchProcessing) && EventsWithAncillaryFile.Contains(eventType)) { HandleModuleInfoFile(eventType); } } private async Task HandleModuleInfoFile(string eventType) { var filename = eventType == "ModuleInfo" ? "ModulesInfo.json" // Just FDev things : eventType + ".json"; // I have no idea what order Elite writes these files or if they're already written // by the time the journal updates. // Brief sleep to ensure the content is updated before we read it. // Some files are still locked by another process after 50ms. // Retry every 50ms for 0.5 seconds before giving up. JsonObject? fileContent = null; var retryCount = 0; while (fileContent == null && retryCount < 10) { await Task.Delay(TimeSpan.FromSeconds(0.5)); try { fileContent = ReadFile(Path.Join(journalWatcher.Path, filename)); var fileObject = new JournalEventArgs(eventType, fileContent); JournalEntry?.Invoke(this, fileObject); } catch { retryCount++; } } } private static void ReportErrors(List readErrors) { if (readErrors.Any()) { var errorList = readErrors.Select(error => { string message; if (error.ex.InnerException == null) { message = error.ex.Message; } else { message = error.ex.InnerException.Message; } return ($"Error reading file {error.file}: {message}", error.line); }); ErrorReporter.ShowErrorPopup($"Journal Read Error{(readErrors.Count > 1 ? "s" : "")}", errorList.ToList()); } } private void LogChangedEvent(object source, FileSystemEventArgs eventArgs) { var fileContent = ReadByLines(eventArgs.FullPath); if (!currentLine.ContainsKey(eventArgs.FullPath)) { currentLine.Add(eventArgs.FullPath, fileContent.Count() - 1); } foreach (var line in fileContent.Skip(currentLine[eventArgs.FullPath])) { try { DeserializeAndInvoke(line); } catch (Exception ex) { ReportErrors([(ex, eventArgs.Name ?? string.Empty, line)]); } } currentLine[eventArgs.FullPath] = fileContent.Count(); } private static IEnumerable ReadByLines(string path) { const int bufferSize = 512; using var file = File.Open(path, FileMode.Open, FileAccess.Read, FileShare.ReadWrite); using var reader = new StreamReader(file, Encoding.UTF8, true, bufferSize); while (reader.ReadLine() is { } line && JsonNode.Parse(line) is { } parsed) { yield return parsed.AsObject(); } } private static JsonObject? ReadFile(string path) { using var file = File.Open(path, FileMode.Open, FileAccess.Read, FileShare.ReadWrite); return JsonNode.Parse(file)?.AsObject(); } private void LogCreatedEvent(object source, FileSystemEventArgs eventArgs) { currentLine.Add(eventArgs.FullPath, 0); LogChangedEvent(source, eventArgs); } private void StatusUpdateEvent(object source, FileSystemEventArgs eventArgs) { var handler = StatusUpdate; var statusLines = ReadFile(eventArgs.FullPath); if (statusLines != null) { var status = statusLines.Deserialize(); Status = status; handler(this, new JournalEventArgs("Status", statusLines)); } } /// /// Touches most recent journal file once every 250ms while LogMonitor is monitoring. /// Forces pending file writes to flush to disk and fires change events for new journal lines. /// private async void JournalPoke() { var journalFolder = GetJournalFolder(); await Task.Run(() => { while (IsMonitoring()) { var journals = GetJournalFilesOrdered(journalFolder); if (journals.Any()) { var fileToPoke = GetJournalFilesOrdered(journalFolder).Last(); using var stream = fileToPoke.Open(FileMode.Open, FileAccess.Read, FileShare.ReadWrite); stream.Close(); } Thread.Sleep(250); } }); } private static string GetSavedGamesPath() { throw new NotImplementedException(); } private static IEnumerable GetJournalFilesOrdered(string path) { var journalFolder = new DirectoryInfo(path); return from file in journalFolder.GetFiles("Journal.*.??.log") orderby file.LastWriteTime select file; } #endregion }