2
0
mirror of https://github.com/9ParsonsB/Pulsar.git synced 2025-04-05 17:39:39 -04:00

Rework Journal Handling

Stub ModulesInfo
This commit is contained in:
Ben Parsons 2024-04-21 17:10:45 +10:00
parent 4274496150
commit b8967814d5
14 changed files with 156 additions and 47 deletions

View File

@ -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)
{ {

View File

@ -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; }
@ -287,4 +288,11 @@ 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; }
} }

View File

@ -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&lt;string, object&gt; AdditionalProperties.</para> /// <para>Unhandled json values within a journal entry type will be contained in member property:<br/>Dictionary&lt;string, object&gt; 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/>

View File

@ -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);

View File

@ -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;
} }

View 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();
}

View File

@ -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
{ {

View 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());
}
}

View 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();
}
}

View File

@ -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);
} }
} }

View File

@ -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;
}
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 interface IStatusService : IJournalHandler 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;
}
public interface IJournalHandler if (fileInfo.Length == 0)
{ {
void HandleFile(string fileInfo); 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();
}
}

View File

@ -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>;

View File

@ -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>();
} }
} }

View File

@ -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;