using Newtonsoft.Json; using System.IO; using System; using System.IO.Compression; using System.Linq; using System.Collections.Generic; namespace OverwatchTranscript { public interface ITranscriptReader { OverwatchCommonHeader Header { get; } T GetHeader(string key); void AddMomentHandler(Action handler); void AddEventHandler(Action> handler); void Next(); void Close(); } public class TranscriptReader : ITranscriptReader { private readonly object handlersLock = new object(); private readonly string transcriptFile; private readonly string artifactsFolder; private readonly List> momentHandlers = new List>(); private readonly Dictionary>> eventHandlers = new Dictionary>>(); private readonly string workingDir; private OverwatchTranscript model = null!; private long momentIndex = 0; private bool closed; public TranscriptReader(string workingDir, string inputFilename) { closed = false; this.workingDir = workingDir; transcriptFile = Path.Combine(workingDir, TranscriptConstants.TranscriptFilename); artifactsFolder = Path.Combine(workingDir, TranscriptConstants.ArtifactFolderName); if (!Directory.Exists(workingDir)) Directory.CreateDirectory(workingDir); if (File.Exists(transcriptFile) || Directory.Exists(artifactsFolder)) throw new Exception("workingdir not clean"); LoadModel(inputFilename); } public OverwatchCommonHeader Header { get { CheckClosed(); return model.Header.Common; } } public T GetHeader(string key) { CheckClosed(); var value = model.Header.Entries.First(e => e.Key == key).Value; return JsonConvert.DeserializeObject(value)!; } public void AddMomentHandler(Action handler) { CheckClosed(); lock (handlersLock) { momentHandlers.Add(handler); } } public void AddEventHandler(Action> handler) { CheckClosed(); var typeName = typeof(T).FullName; if (string.IsNullOrEmpty(typeName)) throw new Exception("Empty typename for payload"); lock (handlersLock) { if (eventHandlers.ContainsKey(typeName)) { eventHandlers[typeName].Add(CreateEventAction(handler)); } else { eventHandlers.Add(typeName, new List> { CreateEventAction(handler) }); } } } public void Next() { CheckClosed(); if (momentIndex >= model.Moments.Length) return; var moment = model.Moments[momentIndex]; var momentDuration = GetMomentDuration(); ActivateMoment(moment, momentDuration, momentIndex); momentIndex++; } public void Close() { CheckClosed(); Directory.Delete(workingDir, true); closed = true; } private Action CreateEventAction(Action> handler) { return (m, s) => { handler(new ActivateEvent(m, JsonConvert.DeserializeObject(s)!)); }; } private TimeSpan? GetMomentDuration() { if (momentIndex < 0) throw new Exception("Index < 0"); if (momentIndex + 1 >= model.Moments.Length) return null; return model.Moments[momentIndex + 1].Utc - model.Moments[momentIndex].Utc; } private void ActivateMoment(OverwatchMoment moment, TimeSpan? duration, long momentIndex) { var m = new ActivateMoment(moment.Utc, duration, momentIndex); lock (handlersLock) { ActivateMomentHandlers(m); foreach (var @event in moment.Events) { ActivateEventHandlers(m, @event); } } } private void ActivateMomentHandlers(ActivateMoment m) { foreach (var handler in momentHandlers) { handler(m); } } private void ActivateEventHandlers(ActivateMoment m, OverwatchEvent @event) { if (!eventHandlers.ContainsKey(@event.Type)) return; var handlers = eventHandlers[@event.Type]; foreach (var handler in handlers) { handler(m, @event.Payload); } } private void LoadModel(string inputFilename) { ZipFile.ExtractToDirectory(inputFilename, workingDir); if (!File.Exists(transcriptFile)) { closed = true; throw new Exception("Is not a transcript file. Unzipped to: " + workingDir); } model = JsonConvert.DeserializeObject(File.ReadAllText(transcriptFile))!; } private void CheckClosed() { if (closed) throw new Exception("Transcript has already been closed."); } } public class ActivateMoment { public ActivateMoment(DateTime utc, TimeSpan? duration, long index) { Utc = utc; Duration = duration; Index = index; } public DateTime Utc { get; } public TimeSpan? Duration { get; } public long Index { get; } } public class ActivateEvent { public ActivateEvent(ActivateMoment moment, T payload) { Moment = moment; Payload = payload; } public ActivateMoment Moment { get; } public T Payload { get; } } }