mirror of
				https://github.com/9ParsonsB/Pulsar.git
				synced 2025-10-25 12:39:49 -04:00 
			
		
		
		
	Rework Journal Handling
Stub ModulesInfo
This commit is contained in:
		| @@ -97,7 +97,7 @@ public class Botanist : IObservatoryWorker | |||||||
|         set => botanistSettings = (BotanistSettings)value; |         set => botanistSettings = (BotanistSettings)value; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void JournalEvent<TJournal>(TJournal journal) where TJournal : JournalBase |     public void JournalEvent<TJournal>(TJournal journal) where TJournal : IJournal | ||||||
|     { |     { | ||||||
|         switch (journal) |         switch (journal) | ||||||
|         { |         { | ||||||
|   | |||||||
| @@ -277,7 +277,8 @@ using Travel; | |||||||
| [JsonDerivedType(typeof(OutfittingFile))] | [JsonDerivedType(typeof(OutfittingFile))] | ||||||
| [JsonDerivedType(typeof(ShipyardFile))] | [JsonDerivedType(typeof(ShipyardFile))] | ||||||
| [JsonDerivedType(typeof(Status))] | [JsonDerivedType(typeof(Status))] | ||||||
| public class JournalBase | [JsonDerivedType(typeof(JournalBase))] | ||||||
|  | public interface IJournal | ||||||
| { | { | ||||||
|     [JsonPropertyName("timestamp")] |     [JsonPropertyName("timestamp")] | ||||||
|     public DateTimeOffset Timestamp { get; init; } |     public DateTimeOffset Timestamp { get; init; } | ||||||
| @@ -288,3 +289,10 @@ public class JournalBase | |||||||
|     [JsonExtensionData] |     [JsonExtensionData] | ||||||
|     public Dictionary<string, object> AdditionalProperties { get; init; } |     public Dictionary<string, object> AdditionalProperties { get; init; } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | public abstract class JournalBase : IJournal | ||||||
|  | { | ||||||
|  |     public DateTimeOffset Timestamp { get; init; } | ||||||
|  |     public string Event { get; init; } | ||||||
|  |     public Dictionary<string, object> AdditionalProperties { get; init; } | ||||||
|  | } | ||||||
| @@ -54,7 +54,7 @@ public interface IObservatoryWorker : IObservatoryPlugin | |||||||
|     /// <param name="journal"><para>Elite Dangerous journal event, deserialized into a .NET object.</para> |     /// <param name="journal"><para>Elite Dangerous journal event, deserialized into a .NET object.</para> | ||||||
|     /// <para>Unhandled json values within a journal entry type will be contained in member property:<br/>Dictionary<string, object> AdditionalProperties.</para> |     /// <para>Unhandled json values within a journal entry type will be contained in member property:<br/>Dictionary<string, object> AdditionalProperties.</para> | ||||||
|     /// <para>Unhandled journal event types will be type JournalBase with all values contained in AdditionalProperties.</para></param> |     /// <para>Unhandled journal event types will be type JournalBase with all values contained in AdditionalProperties.</para></param> | ||||||
|     public void JournalEvent<TJournal>(TJournal journal) where TJournal : JournalBase; |     public void JournalEvent<TJournal>(TJournal journal) where TJournal : IJournal; | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Method called when status.json content is updated.<br/> |     /// Method called when status.json content is updated.<br/> | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ public class EventsHub : Hub<IEventsHub> | |||||||
|      |      | ||||||
|     public async Task MarketUpdated(MarketFile market) => await Clients.All.MarketUpdated(market); |     public async Task MarketUpdated(MarketFile market) => await Clients.All.MarketUpdated(market); | ||||||
|      |      | ||||||
|     public async Task JournalUpdated(IReadOnlyCollection<JournalBase> journals) => await Clients.All.JournalUpdated(journals); |     public async Task JournalUpdated(IReadOnlyCollection<IJournal> journals) => await Clients.All.JournalUpdated(journals); | ||||||
|      |      | ||||||
|     public async Task ModuleInfoUpdated(ModuleInfoFile moduleInfo) => await Clients.All.ModuleInfoUpdated(moduleInfo); |     public async Task ModuleInfoUpdated(ModuleInfoFile moduleInfo) => await Clients.All.ModuleInfoUpdated(moduleInfo); | ||||||
|      |      | ||||||
| @@ -38,7 +38,7 @@ public interface IEventsHub | |||||||
|      |      | ||||||
|     Task MarketUpdated(MarketFile market); |     Task MarketUpdated(MarketFile market); | ||||||
|      |      | ||||||
|     Task JournalUpdated(IReadOnlyCollection<JournalBase> journals); |     Task JournalUpdated(IReadOnlyCollection<IJournal> journals); | ||||||
|      |      | ||||||
|     Task ModuleInfoUpdated(ModuleInfoFile moduleInfo); |     Task ModuleInfoUpdated(ModuleInfoFile moduleInfo); | ||||||
|      |      | ||||||
|   | |||||||
| @@ -1,10 +1,12 @@ | |||||||
| namespace Pulsar.Features; | namespace Pulsar.Features; | ||||||
|  |  | ||||||
| public interface IFileHandlerService | public interface IFileHandler | ||||||
| { | { | ||||||
|     void HandleFile(string path); |     Task HandleFile(string path); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | public interface IFileHandlerService : IFileHandler; | ||||||
|  |  | ||||||
| public class FileHandlerService(ILogger<FileHandlerService> logger, IStatusService statusService) : IFileHandlerService | public class FileHandlerService(ILogger<FileHandlerService> logger, IStatusService statusService) : IFileHandlerService | ||||||
| { | { | ||||||
|     public static readonly string MarketFileName = "Market.json"; |     public static readonly string MarketFileName = "Market.json"; | ||||||
| @@ -12,7 +14,8 @@ public class FileHandlerService(ILogger<FileHandlerService> logger, IStatusServi | |||||||
|     public static readonly string OutfittingFileName = "Outfitting.json"; |     public static readonly string OutfittingFileName = "Outfitting.json"; | ||||||
|     public static readonly string ShipyardFileName = "Shipyard.json"; |     public static readonly string ShipyardFileName = "Shipyard.json"; | ||||||
|     public static readonly string ModulesFileName = "Modules.json"; |     public static readonly string ModulesFileName = "Modules.json"; | ||||||
|     public static readonly string JournalFileName = "Journal."; |     public static readonly string JournalLogFileNameRegEx = @"Journal\.\d\d\d\d-\d\d-\d\dT\d+\.\d\d\.log"; | ||||||
|  |     public static readonly string JournalLogFileName = "Journal.*.log"; | ||||||
|     public static readonly string RouteFileName = "Route.json"; |     public static readonly string RouteFileName = "Route.json"; | ||||||
|     public static readonly string CargoFileName = "Cargo.json"; |     public static readonly string CargoFileName = "Cargo.json"; | ||||||
|     public static readonly string BackpackFileName = "Backpack.json"; |     public static readonly string BackpackFileName = "Backpack.json"; | ||||||
| @@ -27,7 +30,7 @@ public class FileHandlerService(ILogger<FileHandlerService> logger, IStatusServi | |||||||
|         OutfittingFileName, |         OutfittingFileName, | ||||||
|         ShipyardFileName, |         ShipyardFileName, | ||||||
|         ModulesFileName, |         ModulesFileName, | ||||||
|         JournalFileName, |         JournalLogFileNameRegEx, | ||||||
|         RouteFileName, |         RouteFileName, | ||||||
|         CargoFileName, |         CargoFileName, | ||||||
|         BackpackFileName, |         BackpackFileName, | ||||||
| @@ -38,10 +41,10 @@ public class FileHandlerService(ILogger<FileHandlerService> logger, IStatusServi | |||||||
|      |      | ||||||
|     private readonly Dictionary<string, IJournalHandler> Handlers = new() |     private readonly Dictionary<string, IJournalHandler> Handlers = new() | ||||||
|     { |     { | ||||||
|         { StatusFileName, statusService }, |         { StatusFileName, statusService } | ||||||
|     }; |     }; | ||||||
|      |      | ||||||
|     public void HandleFile(string path) |     public async Task HandleFile(string path) | ||||||
|     { |     { | ||||||
|         var fileInfo = new FileInfo(path); |         var fileInfo = new FileInfo(path); | ||||||
|         var fileName = fileInfo.Name; |         var fileName = fileInfo.Name; | ||||||
| @@ -59,7 +62,7 @@ public class FileHandlerService(ILogger<FileHandlerService> logger, IStatusServi | |||||||
|         if (Handlers.TryGetValue(match, out var handler)) |         if (Handlers.TryGetValue(match, out var handler)) | ||||||
|         { |         { | ||||||
|             logger.LogInformation("Handling file {FileName}", fileName); |             logger.LogInformation("Handling file {FileName}", fileName); | ||||||
|             handler.HandleFile(fileInfo.Name); |             await handler.HandleFile(fileInfo.Name); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|          |          | ||||||
|   | |||||||
							
								
								
									
										23
									
								
								Pulsar/Features/Interfaces/IJournalHandler.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								Pulsar/Features/Interfaces/IJournalHandler.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | using Observatory.Framework.Files.Journal; | ||||||
|  |  | ||||||
|  | namespace Pulsar.Features; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Interface for Handling Journal Files. | ||||||
|  | /// </summary> | ||||||
|  | public interface IJournalHandler : IFileHandler | ||||||
|  | { | ||||||
|  |     string FileName { get; } | ||||||
|  |     Task HandleFile(string filePath); | ||||||
|  |     public bool ValidateFile(string filePath); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// Interface for Getting Journal Files. | ||||||
|  | /// Only used for Controllers | ||||||
|  | /// </summary> | ||||||
|  | public interface IJournalHandler<T> : IJournalHandler | ||||||
|  |     where T: IJournal | ||||||
|  | { | ||||||
|  |     Task<T> Get(); | ||||||
|  | } | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| namespace Pulsar.Features.Journal; | namespace Pulsar.Features.Journal; | ||||||
|  |  | ||||||
| [ApiController] | [ApiController] | ||||||
| [Route("api/journal")] | [Route("api/[controller]")] | ||||||
| public class JournalController : ControllerBase | public class JournalController : ControllerBase | ||||||
| { | { | ||||||
|      |      | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								Pulsar/Features/ModulesInfo/ModulesInfoController.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								Pulsar/Features/ModulesInfo/ModulesInfoController.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | namespace Pulsar.Features.ModulesInfo; | ||||||
|  |  | ||||||
|  | [ApiController] | ||||||
|  | [Route("api/[controller]")] | ||||||
|  | public class ModulesInfoController(IModulesInfoService modulesInfo) : ControllerBase | ||||||
|  | { | ||||||
|  |     [HttpGet] | ||||||
|  |     public async Task<IActionResult> Get() | ||||||
|  |     { | ||||||
|  |         return Ok(modulesInfo.Get()); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										24
									
								
								Pulsar/Features/ModulesInfo/ModulesInfoService.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								Pulsar/Features/ModulesInfo/ModulesInfoService.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | namespace Pulsar.Features.ModulesInfo; | ||||||
|  |  | ||||||
|  | using Observatory.Framework.Files; | ||||||
|  |  | ||||||
|  | public interface IModulesInfoService : IJournalHandler<ModuleInfoFile>; | ||||||
|  |  | ||||||
|  | public class ModulesInfoService : IModulesInfoService | ||||||
|  | { | ||||||
|  |     public string FileName => FileHandlerService.ModulesInfoFileName; | ||||||
|  |     public Task HandleFile(string filePath) | ||||||
|  |     { | ||||||
|  |         throw new NotImplementedException(); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     public bool ValidateFile(string filePath) | ||||||
|  |     { | ||||||
|  |         throw new NotImplementedException(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public Task<ModuleInfoFile> Get() | ||||||
|  |     { | ||||||
|  |         throw new NotImplementedException(); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,30 +1,12 @@ | |||||||
| namespace Pulsar.Features.Status; | namespace Pulsar.Features.Status; | ||||||
|  |  | ||||||
| [ApiController] | [ApiController] | ||||||
| [Route("api/status")] | [Route("api/[controller]")] | ||||||
| public class StatusController(IOptions<PulsarConfiguration> pulsarOptions, IHubContext<EventsHub, IEventsHub> hub) : ControllerBase | public class StatusController(IStatusService status) : ControllerBase | ||||||
| { | { | ||||||
|     [HttpGet] |     [HttpGet] | ||||||
|     public async Task<IActionResult> Get() |     public async Task<IActionResult> Get() | ||||||
|     { |     { | ||||||
|         // TODO: put in service |         return Ok(status.Get()); | ||||||
|         var journalDir = pulsarOptions.Value.JournalDirectory; |  | ||||||
|         var dir = new DirectoryInfo(journalDir); |  | ||||||
|          |  | ||||||
|         if (!dir.Exists) |  | ||||||
|             return Problem("Journal directory does not exist."); |  | ||||||
|          |  | ||||||
|         var files = dir.GetFiles(); |  | ||||||
|          |  | ||||||
|         var statusFile = files.FirstOrDefault(f => |  | ||||||
|             string.Equals(f.Name, "status.json", StringComparison.InvariantCultureIgnoreCase)); |  | ||||||
|          |  | ||||||
|         if (statusFile == null) |  | ||||||
|             return Problem("Status file not found."); |  | ||||||
|  |  | ||||||
|         await using var file = System.IO.File.Open(statusFile.FullName, FileMode.Open, FileAccess.Read, FileShare.ReadWrite); |  | ||||||
|         var status = await JsonSerializer.DeserializeAsync<Observatory.Framework.Files.Status>(file);  |  | ||||||
|         await hub.Clients.All.StatusUpdated(status); |  | ||||||
|         return Ok(status); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,18 +1,71 @@ | |||||||
| namespace Pulsar.Features.Status; | namespace Pulsar.Features.Status; | ||||||
|  |  | ||||||
| public class StatusService : IStatusService | using Observatory.Framework.Files; | ||||||
|  |  | ||||||
|  | public interface IStatusService : IJournalHandler<Status>; | ||||||
|  | public class StatusService(ILogger<StatusService> logger, IOptions<PulsarConfiguration> options, IEventHubContext hub) : IStatusService | ||||||
| { | { | ||||||
|     public void HandleFile(string fileInfo) |     public string FileName => FileHandlerService.StatusFileName; | ||||||
|  |      | ||||||
|  |     public async Task HandleFile(string filePath) | ||||||
|     { |     { | ||||||
|         throw new NotImplementedException(); |         if (!ValidateFile(filePath)) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|         } |         } | ||||||
| } |  | ||||||
|          |          | ||||||
| public interface IStatusService : IJournalHandler |         var file = File.Open(filePath, FileMode.Open, FileAccess.Read, FileShare.ReadWrite); | ||||||
| { |         var status = await JsonSerializer.DeserializeAsync<Status>(file);  | ||||||
| } |          | ||||||
|  |         if (status == null) | ||||||
|  |         { | ||||||
|  |             logger.LogWarning("Failed to deserialize status file {FilePath}", filePath); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         await hub.Clients.All.StatusUpdated(status); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public bool ValidateFile(string filePath) | ||||||
|  |     { | ||||||
|  |         if (!File.Exists(filePath)) | ||||||
|  |         { | ||||||
|  |             logger.LogWarning("Status file {StatusFile} does not exist", filePath); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         var fileInfo = new FileInfo(filePath); | ||||||
|  |          | ||||||
|  |         if (!string.Equals(fileInfo.Name, FileName, StringComparison.InvariantCultureIgnoreCase)) | ||||||
|  |         { | ||||||
|  |             logger.LogWarning("File {StatusFile} is not a status file", filePath); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (fileInfo.Length == 0) | ||||||
|  |         { | ||||||
|  |             logger.LogWarning("Status file {StatusFile} is empty", filePath); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public async Task<Status> Get() | ||||||
|  |     { | ||||||
|  |         var statusFile = Path.Combine(options.Value.JournalDirectory, FileName); | ||||||
|  |          | ||||||
|  |         if (!ValidateFile(statusFile)) | ||||||
|  |         { | ||||||
|  |             return new Status(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         await using var file = File.Open(statusFile, FileMode.Open, FileAccess.Read, FileShare.ReadWrite); | ||||||
|  |         var status = await JsonSerializer.DeserializeAsync<Status>(file); | ||||||
|  |         if (status != null) return status; | ||||||
|  |          | ||||||
|  |         logger.LogWarning("Failed to deserialize status file {StatusFile}", statusFile); | ||||||
|  |         return new Status(); | ||||||
|  |     } | ||||||
|  |  | ||||||
| public interface IJournalHandler |  | ||||||
| { |  | ||||||
|     void HandleFile(string fileInfo); |  | ||||||
| } | } | ||||||
| @@ -8,4 +8,4 @@ global using System.Text.Json.Serialization; | |||||||
| global using Microsoft.AspNetCore.Mvc; | global using Microsoft.AspNetCore.Mvc; | ||||||
| global using Microsoft.AspNetCore.SignalR; | global using Microsoft.AspNetCore.SignalR; | ||||||
| global using Microsoft.Extensions.Options; | global using Microsoft.Extensions.Options; | ||||||
|  | global using IEventHubContext = Microsoft.AspNetCore.SignalR.IHubContext<Pulsar.Features.EventsHub, Pulsar.Features.IEventsHub>; | ||||||
| @@ -1,13 +1,17 @@ | |||||||
|  | using System.Diagnostics.CodeAnalysis; | ||||||
| using Lamar; | using Lamar; | ||||||
| using Pulsar.Features; | using Pulsar.Features; | ||||||
|  | using Pulsar.Features.ModulesInfo; | ||||||
|  |  | ||||||
| namespace Pulsar; | namespace Pulsar; | ||||||
|  |  | ||||||
|  | [DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors)] | ||||||
| public class PulsarServiceRegistry : ServiceRegistry | public class PulsarServiceRegistry : ServiceRegistry | ||||||
| { | { | ||||||
|     public PulsarServiceRegistry() |     public PulsarServiceRegistry() | ||||||
|     { |     { | ||||||
|         For<IFileHandlerService>().Use<FileHandlerService>(); |         For<IFileHandlerService>().Use<FileHandlerService>(); | ||||||
|         For<IStatusService>().Use<StatusService>(); |         For<IStatusService>().Use<StatusService>(); | ||||||
|  |         For<IModulesInfoService>().Use<ModulesInfoService>(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -5,7 +5,7 @@ namespace Pulsar.Utils; | |||||||
|  |  | ||||||
| public static class JournalReader | public static class JournalReader | ||||||
| { | { | ||||||
|     public static TJournal ObservatoryDeserializer<TJournal>(string json) where TJournal : JournalBase |     public static TJournal ObservatoryDeserializer<TJournal>(string json) where TJournal : IJournal | ||||||
|     { |     { | ||||||
|         TJournal deserialized; |         TJournal deserialized; | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user