2
0
mirror of https://github.com/9ParsonsB/Pulsar.git synced 2025-04-05 17:39:39 -04:00
pulsar/Pulsar/Features/FileWatcherService.cs
Ben Parsons 579b2b115d Fix issues with Journal handling
Implement basic database
Handle startup events
only send events after the most recent LoadGame
2024-05-25 16:19:07 +10:00

106 lines
3.4 KiB
C#

namespace Pulsar.Features;
using System.Collections.Concurrent;
using Microsoft.Extensions.FileProviders;
public class FileWatcherService(IOptions<PulsarConfiguration> options, IFileHandlerService fileHandlerService, ILogger<FileWatcherService> logger)
: IHostedService, IDisposable
{
private PhysicalFileProvider watcher = null!;
public Task StartAsync(CancellationToken cancellationToken)
{
if (!Directory.Exists(options.Value.JournalDirectory))
{
throw new Exception($"Directory {options.Value.JournalDirectory} does not exist.");
}
watcher = new PhysicalFileProvider(options.Value.JournalDirectory);
Watch(cancellationToken);
// read the journal directory to get the initial files
#if DEBUG
Thread.Sleep(TimeSpan.FromSeconds(2));
HandleFileChanged(cancellationToken);
#else
HandleFileChanged(cancellationToken);
#endif
return Task.CompletedTask;
}
ConcurrentDictionary<string, DateTimeOffset> FileDates = new();
private void HandleFileChanged(CancellationToken token = new())
{
Watch(token);
var tasks = new List<Task>();
try
{
foreach (var file in watcher.GetDirectoryContents(""))
{
logger.LogDebug("Checking File: {File}", file.PhysicalPath);
if (file.IsDirectory || (!file.Name.EndsWith(".json") &&
!(file.Name.StartsWith(FileHandlerService.JournalLogFileNameStart) &&
file.Name.EndsWith(FileHandlerService.JournalLogFileNameEnd))))
{
continue;
}
logger.LogDebug("Has File Updated?: {File}, {LastModified}", file.PhysicalPath, file.LastModified);
FileDates.AddOrUpdate(file.PhysicalPath, _ =>
{
logger.LogDebug("New File: {File}", file.PhysicalPath);
tasks.Add(Task.Run(() => fileHandlerService.HandleFile(file.PhysicalPath, token), token));
return file.LastModified;
}, (_, existing) =>
{
logger.LogDebug("Existing File: {File}", file.PhysicalPath);
if (existing != file.LastModified)
{
logger.LogDebug("File Updated: {File}", file.PhysicalPath);
tasks.Add(Task.Run(() => fileHandlerService.HandleFile(file.PhysicalPath, token), token));
}
return file.LastModified;
});
}
Task.WaitAll(tasks.ToArray(), token);
}
catch (Exception ex)
{
logger.LogError(ex, "Error handling file change");
}
}
private void Watch(CancellationToken token)
{
void Handle(object? _)
{
HandleFileChanged(token);
}
try
{
watcher.Watch("*.*").RegisterChangeCallback(Handle, null);
}
catch (Exception ex)
{
logger.LogError(ex, "Error watching directory {Directory}", watcher.Root);
}
}
public Task StopAsync(CancellationToken cancellationToken)
{
watcher.Dispose();
return Task.CompletedTask;
}
public void Dispose()
{
watcher.Dispose();
}
}