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:
parent
4274496150
commit
b8967814d5
@ -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; }
|
||||||
@ -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; }
|
||||||
}
|
}
|
@ -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;
|
||||||
|
}
|
||||||
|
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
@ -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;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user