using System.Globalization; using System.Text.Json.Serialization; using Logging; using Newtonsoft.Json; using Newtonsoft.Json.Linq; using Tranga.TrangaTasks; using JsonConverter = Newtonsoft.Json.JsonConverter; namespace Tranga; /// /// Stores information on Task, when implementing new Tasks also update the serializer /// [JsonDerivedType(typeof(DownloadNewChaptersTask), 2)] [JsonDerivedType(typeof(UpdateLibrariesTask), 3)] [JsonDerivedType(typeof(DownloadChapterTask), 4)] public abstract class TrangaTask { // ReSharper disable once CommentTypo ...Tell me why! // ReSharper disable once MemberCanBePrivate.Global I want it thaaat way public TimeSpan reoccurrence { get; } public DateTime lastExecuted { get; set; } public Task task { get; } public string taskId { get; set; } [Newtonsoft.Json.JsonIgnore]public ExecutionState state { get; set; } [Newtonsoft.Json.JsonIgnore]protected HashSet childTasks { get; } [Newtonsoft.Json.JsonIgnore]public double progress => childTasks.Count > 0 ? childTasks.Sum(childTask => childTask.progress) / childTasks.Count : _myProgress; [Newtonsoft.Json.JsonIgnore]private double _myProgress = 0; [Newtonsoft.Json.JsonIgnore]public DateTime nextExecution => lastExecuted.Add(reoccurrence); [Newtonsoft.Json.JsonIgnore]public DateTime executionStarted { get; private set; } [Newtonsoft.Json.JsonIgnore] public DateTime executionApproximatelyFinished => this.progress != 0 ? this.executionStarted.Add(DateTime.Now.Subtract(this.executionStarted).Divide(this.progress)) : DateTime.MaxValue; [Newtonsoft.Json.JsonIgnore]public TimeSpan executionApproximatelyRemaining => this.executionApproximatelyFinished.Subtract(DateTime.Now); [Newtonsoft.Json.JsonIgnore]public DateTime lastChange { get; private set; } public enum ExecutionState { Waiting, Enqueued, Running } protected TrangaTask(Task task, TimeSpan reoccurrence) { this.reoccurrence = reoccurrence; this.lastExecuted = DateTime.Now.Subtract(reoccurrence); this.task = task; this.executionStarted = DateTime.Now; this.lastChange = DateTime.MaxValue; this.taskId = Convert.ToBase64String(System.Text.Encoding.ASCII.GetBytes(this.executionStarted.ToString(CultureInfo.InvariantCulture))); this.childTasks = new(); } /// /// BL for concrete Tasks /// /// /// /// protected abstract void ExecuteTask(TaskManager taskManager, Logger? logger, CancellationToken? cancellationToken = null); /// /// Execute the task /// /// Should be the parent taskManager /// /// public void Execute(TaskManager taskManager, Logger? logger, CancellationToken? cancellationToken = null) { logger?.WriteLine(this.GetType().ToString(), $"Executing Task {this}"); this.state = ExecutionState.Running; this.executionStarted = DateTime.Now; this.lastChange = DateTime.Now; ExecuteTask(taskManager, logger, cancellationToken); this.lastExecuted = DateTime.Now; this.state = ExecutionState.Waiting; logger?.WriteLine(this.GetType().ToString(), $"Finished Executing Task {this}"); } public void ReplaceFailedChildTask(DownloadChapterTask failed, DownloadChapterTask newTask) { if (!this.childTasks.Contains(failed)) throw new ArgumentException($"Task {failed} is not childTask of {this}"); this.childTasks.Remove(failed); this.childTasks.Add(newTask); } public void AddChildTask(DownloadChapterTask childTask) { this.childTasks.Add(childTask); } public void IncrementProgress(double amount) { this._myProgress += amount; } public enum Task : byte { DownloadNewChapters = 2, UpdateLibraries = 3, DownloadChapter = 4 } public override string ToString() { return $"{task}, {lastExecuted}, {reoccurrence}, {state}, {progress:P2}, {executionApproximatelyFinished}, {executionApproximatelyRemaining}"; } public class TrangaTaskJsonConverter : JsonConverter { public override bool CanConvert(Type objectType) { return (objectType == typeof(TrangaTask)); } public override object ReadJson(JsonReader reader, Type objectType, object? existingValue, JsonSerializer serializer) { JObject jo = JObject.Load(reader); if (jo["task"]!.Value() == (Int64)Task.DownloadNewChapters) return jo.ToObject(serializer)!; if (jo["task"]!.Value() == (Int64)Task.UpdateLibraries) return jo.ToObject(serializer)!; if (jo["task"]!.Value() == (Int64)Task.DownloadChapter) return jo.ToObject(serializer)!; throw new Exception(); } public override bool CanWrite => false; /// /// Don't call this /// public override void WriteJson(JsonWriter writer, object? value, JsonSerializer serializer) { throw new Exception("Dont call this"); } } }