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

JournalService setup work in progress

This commit is contained in:
ChrisDill 2024-05-07 12:02:43 +01:00
parent d59914e277
commit f3ce62b551
12 changed files with 215 additions and 85 deletions

View File

@ -278,7 +278,7 @@ using Travel;
[JsonDerivedType(typeof(ShipyardFile))]
[JsonDerivedType(typeof(Status))]
[JsonDerivedType(typeof(JournalBase))]
public interface IJournal
public class JournalBase
{
[JsonPropertyName("timestamp")]
public DateTimeOffset Timestamp { get; init; }
@ -289,10 +289,3 @@ public interface IJournal
[JsonExtensionData]
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>
/// <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>
public void JournalEvent<TJournal>(TJournal journal) where TJournal : IJournal;
public void JournalEvent<TJournal>(TJournal journal) where TJournal : JournalBase;
/// <summary>
/// Method called when status.json content is updated.<br/>

View File

@ -6,45 +6,45 @@ using Observatory.Framework.Files.Journal;
public class EventsHub : Hub<IEventsHub>
{
public async Task StatusUpdated(Observatory.Framework.Files.Status status) => await Clients.All.StatusUpdated(status);
public async Task OutfittingUpdated(OutfittingFile outfitting) => await Clients.All.OutfittingUpdated(outfitting);
public async Task ShipyardUpdated(ShipyardFile shipyard) => await Clients.All.ShipyardUpdated(shipyard);
public async Task NavRouteUpdated(NavRouteFile navRoute) => await Clients.All.NavRouteUpdated(navRoute);
public async Task MarketUpdated(MarketFile market) => await Clients.All.MarketUpdated(market);
public async Task JournalUpdated(IReadOnlyCollection<IJournal> journals) => await Clients.All.JournalUpdated(journals);
public async Task JournalUpdated(IReadOnlyCollection<JournalBase> journals) => await Clients.All.JournalUpdated(journals);
public async Task ModuleInfoUpdated(ModuleInfoFile moduleInfo) => await Clients.All.ModuleInfoUpdated(moduleInfo);
public async Task FleetCarrierUpdated(FCMaterialsFile fleetCarrier) => await Clients.All.FleetCarrierUpdated(fleetCarrier);
public async Task CargoUpdated(CargoFile cargo) => await Clients.All.CargoUpdated(cargo);
public async Task BackpackUpdated(BackpackFile backpack) => await Clients.All.BackpackUpdated(backpack);
}
public interface IEventsHub
{
Task StatusUpdated(Observatory.Framework.Files.Status status);
Task OutfittingUpdated(OutfittingFile outfitting);
Task ShipyardUpdated(ShipyardFile shipyard);
Task NavRouteUpdated(NavRouteFile navRoute);
Task MarketUpdated(MarketFile market);
Task JournalUpdated(IReadOnlyCollection<IJournal> journals);
Task JournalUpdated(IReadOnlyCollection<JournalBase> journals);
Task ModuleInfoUpdated(ModuleInfoFile moduleInfo);
Task FleetCarrierUpdated(FCMaterialsFile fleetCarrier);
Task CargoUpdated(CargoFile cargo);
Task BackpackUpdated(BackpackFile backpack);
}

View File

@ -8,7 +8,7 @@ namespace Pulsar.Features;
public interface IJournalHandler : IFileHandler
{
string FileName { get; }
public bool ValidateFile(string filePath);
}
@ -17,42 +17,6 @@ public interface IJournalHandler : IFileHandler
/// Only used for Controllers
/// </summary>
public interface IJournalHandler<T> : IJournalHandler
where T: IJournal
{
Task<T> Get();
}
public abstract class JournalHandlerBase<T>(ILogger logger) : IJournalHandler<T>
where T: IJournal
{
public abstract string FileName { get; }
public abstract Task HandleFile(string filePath);
public bool ValidateFile(string filePath)
{
if (!File.Exists(filePath))
{
logger.LogWarning("Journal file {JournalFile} does not exist", filePath);
return false;
}
var fileInfo = new FileInfo(filePath);
if (!string.Equals(fileInfo.Name, FileName, StringComparison.InvariantCultureIgnoreCase))
{
logger.LogWarning("Journal file {JournalFile} is not a {NameOfCurrentHandler} file", filePath, nameof(T));
return false;
}
if (fileInfo.Length == 0)
{
logger.LogWarning("Journal file {JournalFile} is empty", filePath);
return false;
}
return true;
}
public abstract Task<T> Get();
}

View File

@ -2,7 +2,11 @@ namespace Pulsar.Features.Journal;
[ApiController]
[Route("api/[controller]")]
public class JournalController : ControllerBase
public class JournalController(IJournalService journalService) : ControllerBase
{
[HttpGet]
public async Task<IActionResult> Get()
{
return Ok(await journalService.Get());
}
}

View File

@ -0,0 +1,89 @@
namespace Pulsar.Features.Journal;
using Observatory.Framework.Files;
using Observatory.Framework.Files.Journal;
public interface IJournalService : IJournalHandler<List<JournalBase>>;
public class JournalService
(
ILogger<JournalService> logger,
IOptions<PulsarConfiguration> options,
IEventHubContext hub
) : IJournalService
{
public string FileName => "Journal.2024-03-16T152419.01.log"; // FileHandlerService.JournalLogFileName;
public bool ValidateFile(string filePath)
{
if (!File.Exists(filePath))
{
logger.LogWarning("Journal file {JournalFile} does not exist", filePath);
return false;
}
var fileInfo = new FileInfo(filePath);
if (!string.Equals(fileInfo.Name, FileName, StringComparison.InvariantCultureIgnoreCase))
{
logger.LogWarning("Journal file {name} is not valid");
return false;
}
if (fileInfo.Length == 0)
{
logger.LogWarning("Journal file {name} is empty", filePath);
return false;
}
return true;
}
public async Task HandleFile(string filePath)
{
if (!ValidateFile(filePath))
{
return;
}
var file = File.Open(filePath, FileMode.Open, FileAccess.Read, FileShare.ReadWrite);
var moduleInfo = await JsonSerializer.DeserializeAsync<List<JournalBase>>(file);
if (moduleInfo == null)
{
logger.LogWarning("Failed to deserialize status file {FilePath}", filePath);
return;
}
// await hub.Clients.All.ModuleInfoUpdated(moduleInfo);
}
public async Task<List<JournalBase>> Get()
{
var dataFileName = Path.Combine(options.Value.JournalDirectory, FileName);
if (!ValidateFile(dataFileName))
{
return [];
}
// Seems each entry is a new line. Not sure if this can be relied on?
var logs = File.ReadAllLines(dataFileName);
var journals = new List<JournalBase>();
foreach (var log in logs)
{
// var info = JournalReader.ObservatoryDeserializer<JournalBase>(log);
var info = JsonSerializer.Deserialize<JournalBase>(log);
if (info != null)
{
journals.Add(info);
}
}
if (journals.Count > 0) return journals;
logger.LogWarning("Failed to deserialize module info file {file}", dataFileName);
return [];
}
}

View File

@ -7,6 +7,6 @@ public class ModulesInfoController(IModulesInfoService modulesInfo) : Controller
[HttpGet]
public async Task<IActionResult> Get()
{
return Ok(modulesInfo.Get());
return Ok(await modulesInfo.Get());
}
}

View File

@ -7,12 +7,36 @@ public interface IModulesInfoService : IJournalHandler<ModuleInfoFile>;
public class ModulesInfoService(
ILogger<ModulesInfoService> logger,
IOptions<PulsarConfiguration> options,
IEventHubContext hub)
: JournalHandlerBase<ModuleInfoFile>(logger), IModulesInfoService
IEventHubContext hub) : IModulesInfoService
{
public override string FileName => FileHandlerService.ModulesInfoFileName;
public string FileName => FileHandlerService.ModulesInfoFileName;
public override async Task HandleFile(string filePath)
public bool ValidateFile(string filePath)
{
if (!File.Exists(filePath))
{
logger.LogWarning("Journal file {JournalFile} does not exist", filePath);
return false;
}
var fileInfo = new FileInfo(filePath);
if (!string.Equals(fileInfo.Name, FileName, StringComparison.InvariantCultureIgnoreCase))
{
logger.LogWarning("Journal file {name} is not valid");
return false;
}
if (fileInfo.Length == 0)
{
logger.LogWarning("Journal file {name} is empty", filePath);
return false;
}
return true;
}
public async Task HandleFile(string filePath)
{
if (!ValidateFile(filePath))
{
@ -31,7 +55,7 @@ public class ModulesInfoService(
await hub.Clients.All.ModuleInfoUpdated(moduleInfo);
}
public override async Task<ModuleInfoFile> Get()
public async Task<ModuleInfoFile> Get()
{
var moduleInfoFile = Path.Combine(options.Value.JournalDirectory, FileName);
@ -47,4 +71,4 @@ public class ModulesInfoService(
logger.LogWarning("Failed to deserialize module info file {ModuleInfoFile}", moduleInfoFile);
return new ModuleInfoFile();
}
}
}

View File

@ -5,16 +5,41 @@ using Observatory.Framework.Files.Journal.Odyssey;
public interface IShipLockerService : IJournalHandler<ShipLockerMaterials>;
public class ShipLockerService(ILogger<ShipLockerService> logger)
: JournalHandlerBase<ShipLockerMaterials>(logger), IShipLockerService
: IShipLockerService
{
public override string FileName => FileHandlerService.ShipLockerFileName;
public string FileName => FileHandlerService.ShipLockerFileName;
public override Task<ShipLockerMaterials> Get()
public bool ValidateFile(string filePath)
{
if (!File.Exists(filePath))
{
logger.LogWarning("Journal file {JournalFile} does not exist", filePath);
return false;
}
var fileInfo = new FileInfo(filePath);
if (!string.Equals(fileInfo.Name, FileName, StringComparison.InvariantCultureIgnoreCase))
{
logger.LogWarning("Journal file {name} is not valid");
return false;
}
if (fileInfo.Length == 0)
{
logger.LogWarning("Journal file {name} is empty", filePath);
return false;
}
return true;
}
public Task<ShipLockerMaterials> Get()
{
throw new NotImplementedException();
}
public override Task HandleFile(string filePath)
public Task HandleFile(string filePath)
{
throw new NotImplementedException();
}

View File

@ -4,12 +4,41 @@ using Observatory.Framework.Files;
public interface IStatusService : IJournalHandler<Status>;
public class StatusService(ILogger<StatusService> logger, IOptions<PulsarConfiguration> options, IEventHubContext hub)
: JournalHandlerBase<Status>(logger), IStatusService
public class StatusService
(
ILogger<StatusService> logger,
IOptions<PulsarConfiguration> options,
IEventHubContext hub
) : IStatusService
{
public override string FileName => FileHandlerService.StatusFileName;
public string FileName => FileHandlerService.StatusFileName;
public override async Task HandleFile(string filePath)
public bool ValidateFile(string filePath)
{
if (!File.Exists(filePath))
{
logger.LogWarning("Journal file {JournalFile} does not exist", filePath);
return false;
}
var fileInfo = new FileInfo(filePath);
if (!string.Equals(fileInfo.Name, FileName, StringComparison.InvariantCultureIgnoreCase))
{
logger.LogWarning("Journal file {name} is not valid");
return false;
}
if (fileInfo.Length == 0)
{
logger.LogWarning("Journal file {name} is empty", filePath);
return false;
}
return true;
}
public async Task HandleFile(string filePath)
{
if (!ValidateFile(filePath))
{
@ -28,7 +57,7 @@ public class StatusService(ILogger<StatusService> logger, IOptions<PulsarConfigu
await hub.Clients.All.StatusUpdated(status);
}
public override async Task<Status> Get()
public async Task<Status> Get()
{
var statusFile = Path.Combine(options.Value.JournalDirectory, FileName);

View File

@ -2,6 +2,7 @@ using System.Diagnostics.CodeAnalysis;
using Lamar;
using Pulsar.Features;
using Pulsar.Features.ModulesInfo;
using Pulsar.Features.Journal;
namespace Pulsar;
@ -13,5 +14,6 @@ public class PulsarServiceRegistry : ServiceRegistry
For<IFileHandlerService>().Use<FileHandlerService>();
For<IStatusService>().Use<StatusService>();
For<IModulesInfoService>().Use<ModulesInfoService>();
For<IJournalService>().Use<JournalService>();
}
}

View File

@ -5,7 +5,7 @@ namespace Pulsar.Utils;
public static class JournalReader
{
public static TJournal ObservatoryDeserializer<TJournal>(string json) where TJournal : IJournal
public static TJournal ObservatoryDeserializer<TJournal>(string json) where TJournal : JournalBase
{
TJournal deserialized;