2024-08-02 14:25:38 +02:00
|
|
|
|
using Logging;
|
|
|
|
|
using Newtonsoft.Json;
|
2024-08-02 17:52:44 +02:00
|
|
|
|
using System.Collections.Concurrent;
|
2024-08-02 14:25:38 +02:00
|
|
|
|
|
|
|
|
|
namespace OverwatchTranscript
|
|
|
|
|
{
|
|
|
|
|
public interface IFinalizedBucket
|
|
|
|
|
{
|
|
|
|
|
bool IsEmpty { get; }
|
|
|
|
|
DateTime? SeeTopUtc();
|
|
|
|
|
BucketTop? TakeTop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class BucketTop
|
|
|
|
|
{
|
|
|
|
|
public BucketTop(DateTime utc, OverwatchEvent[] events)
|
|
|
|
|
{
|
|
|
|
|
Utc = utc;
|
|
|
|
|
Events = events;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public DateTime Utc { get; }
|
|
|
|
|
public OverwatchEvent[] Events { get; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class EventBucketReader : IFinalizedBucket
|
|
|
|
|
{
|
|
|
|
|
private readonly string bucketFile;
|
2024-08-02 17:52:44 +02:00
|
|
|
|
private readonly ConcurrentQueue<BucketTop> topQueue = new ConcurrentQueue<BucketTop>();
|
|
|
|
|
private readonly AutoResetEvent itemDequeued = new AutoResetEvent(false);
|
|
|
|
|
private bool stopping;
|
2024-08-02 14:25:38 +02:00
|
|
|
|
|
|
|
|
|
public EventBucketReader(ILog log, string bucketFile)
|
|
|
|
|
{
|
|
|
|
|
this.bucketFile = bucketFile;
|
|
|
|
|
if (!File.Exists(bucketFile)) throw new Exception("Doesn't exist: " + bucketFile);
|
|
|
|
|
|
|
|
|
|
log.Debug("Read Bucket open: " + bucketFile);
|
|
|
|
|
|
|
|
|
|
Task.Run(ReadBucket);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool IsEmpty { get; private set; }
|
|
|
|
|
|
|
|
|
|
public DateTime? SeeTopUtc()
|
|
|
|
|
{
|
|
|
|
|
if (IsEmpty) return null;
|
2024-08-02 17:52:44 +02:00
|
|
|
|
while (true)
|
|
|
|
|
{
|
|
|
|
|
UpdateIsEmpty();
|
|
|
|
|
if (IsEmpty) return null;
|
|
|
|
|
if (topQueue.TryPeek(out BucketTop? top))
|
|
|
|
|
{
|
|
|
|
|
return top.Utc;
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-08-02 14:25:38 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public BucketTop? TakeTop()
|
|
|
|
|
{
|
|
|
|
|
if (IsEmpty) return null;
|
|
|
|
|
|
2024-08-02 17:52:44 +02:00
|
|
|
|
while (true)
|
2024-08-02 14:25:38 +02:00
|
|
|
|
{
|
2024-08-02 17:52:44 +02:00
|
|
|
|
UpdateIsEmpty();
|
|
|
|
|
if (IsEmpty) return null;
|
|
|
|
|
if (topQueue.TryDequeue(out BucketTop? top))
|
|
|
|
|
{
|
|
|
|
|
itemDequeued.Set();
|
|
|
|
|
return top;
|
|
|
|
|
}
|
2024-08-02 14:25:38 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ReadBucket()
|
|
|
|
|
{
|
|
|
|
|
using var file = File.OpenRead(bucketFile);
|
|
|
|
|
using var reader = new StreamReader(file);
|
|
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
|
{
|
2024-08-02 17:52:44 +02:00
|
|
|
|
while (topQueue.Count < 5)
|
2024-08-02 14:25:38 +02:00
|
|
|
|
{
|
2024-08-02 17:52:44 +02:00
|
|
|
|
var top = CreateNewTop(reader);
|
|
|
|
|
if (top != null)
|
2024-08-02 14:25:38 +02:00
|
|
|
|
{
|
2024-08-02 17:52:44 +02:00
|
|
|
|
topQueue.Enqueue(top);
|
2024-08-02 14:25:38 +02:00
|
|
|
|
}
|
2024-08-02 17:52:44 +02:00
|
|
|
|
else
|
2024-08-02 14:25:38 +02:00
|
|
|
|
{
|
2024-08-02 17:52:44 +02:00
|
|
|
|
stopping = true;
|
2024-08-02 14:25:38 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-08-02 17:52:44 +02:00
|
|
|
|
|
|
|
|
|
itemDequeued.Reset();
|
|
|
|
|
itemDequeued.WaitOne();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void UpdateIsEmpty()
|
|
|
|
|
{
|
|
|
|
|
var empty = stopping && topQueue.IsEmpty;
|
|
|
|
|
if (!IsEmpty && empty)
|
|
|
|
|
{
|
|
|
|
|
File.Delete(bucketFile);
|
|
|
|
|
IsEmpty = true;
|
2024-08-02 14:25:38 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private EventBucketEntry? nextEntry = null;
|
|
|
|
|
private BucketTop? CreateNewTop(StreamReader reader)
|
|
|
|
|
{
|
|
|
|
|
if (nextEntry == null)
|
|
|
|
|
{
|
|
|
|
|
nextEntry = ReadEntry(reader);
|
|
|
|
|
if (nextEntry == null) return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var topEntry = nextEntry;
|
|
|
|
|
var entries = new List<EventBucketEntry>
|
|
|
|
|
{
|
|
|
|
|
topEntry
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
nextEntry = ReadEntry(reader);
|
|
|
|
|
while (nextEntry != null && nextEntry.Utc == topEntry.Utc)
|
|
|
|
|
{
|
|
|
|
|
entries.Add(nextEntry);
|
|
|
|
|
nextEntry = ReadEntry(reader);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return new BucketTop(topEntry.Utc, entries.Select(e => e.Event).ToArray());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private EventBucketEntry? ReadEntry(StreamReader reader)
|
|
|
|
|
{
|
|
|
|
|
var line = reader.ReadLine();
|
|
|
|
|
if (string.IsNullOrEmpty(line)) return null;
|
|
|
|
|
return JsonConvert.DeserializeObject<EventBucketEntry>(line);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|