using CodexContractsPlugin; using CodexContractsPlugin.ChainMonitor; using DiscordRewards; using GethPlugin; using System.Globalization; using System.Numerics; using Utils; namespace TestNetRewarder { public class EventsFormatter : IChainStateChangeHandler { private static readonly string nl = Environment.NewLine; private readonly List events = new List(); private readonly List errors = new List(); private readonly EmojiMaps emojiMaps = new EmojiMaps(); public ChainEventMessage[] GetEvents() { var result = events.ToArray(); events.Clear(); return result; } public string[] GetErrors() { var result = errors.ToArray(); errors.Clear(); return result; } public void OnNewRequest(RequestEvent requestEvent) { var request = requestEvent.Request; AddRequestBlock(requestEvent, $"{emojiMaps.NewRequest} New Request", $"Client: {request.Client}", $"Content: {request.Request.Content.Cid}", $"Duration: {BigIntToDuration(request.Request.Ask.Duration)}", $"Expiry: {BigIntToDuration(request.Request.Expiry)}", $"Collateral: {BitIntToTestTokens(request.Request.Ask.Collateral)}", $"Reward: {BitIntToTestTokens(request.Request.Ask.Reward)}", $"Number of Slots: {request.Request.Ask.Slots}", $"Slot Tolerance: {request.Request.Ask.MaxSlotLoss}", $"Slot Size: {BigIntToByteSize(request.Request.Ask.SlotSize)}" ); } public void OnRequestCancelled(RequestEvent requestEvent) { AddRequestBlock(requestEvent, $"{emojiMaps.Cancelled} Cancelled"); } public void OnRequestFailed(RequestEvent requestEvent) { AddRequestBlock(requestEvent, $"{emojiMaps.Failed} Failed"); } public void OnRequestFinished(RequestEvent requestEvent) { AddRequestBlock(requestEvent, $"{emojiMaps.Finished} Finished"); } public void OnRequestFulfilled(RequestEvent requestEvent) { AddRequestBlock(requestEvent, $"{emojiMaps.Started} Started"); } public void OnSlotFilled(RequestEvent requestEvent, EthAddress host, BigInteger slotIndex) { AddRequestBlock(requestEvent, $"{emojiMaps.SlotFilled} Slot Filled", $"Host: {host}", $"Slot Index: {slotIndex}" ); } public void OnSlotFreed(RequestEvent requestEvent, BigInteger slotIndex) { AddRequestBlock(requestEvent, $"{emojiMaps.SlotFreed} Slot Freed", $"Slot Index: {slotIndex}" ); } public void OnSlotReservationsFull(RequestEvent requestEvent, BigInteger slotIndex) { AddRequestBlock(requestEvent, $"{emojiMaps.SlotReservationsFull} Slot Reservations Full", $"Slot Index: {slotIndex}" ); } public void OnError(string msg) { errors.Add(msg); } private void AddRequestBlock(RequestEvent requestEvent, string eventName, params string[] content) { var blockNumber = $"[{requestEvent.Block.BlockNumber} {FormatDateTime(requestEvent.Block.Utc)}]"; var title = $"{blockNumber} **{eventName}** {FormatRequestId(requestEvent)}"; AddBlock(requestEvent.Block.BlockNumber, title, content); } private void AddBlock(ulong blockNumber, string title, params string[] content) { events.Add(FormatBlock(blockNumber, title, content)); } private ChainEventMessage FormatBlock(ulong blockNumber, string title, params string[] content) { var msg = FormatBlockMessage(title, content); return new ChainEventMessage { BlockNumber = blockNumber, Message = msg }; } private string FormatBlockMessage(string title, string[] content) { if (content == null || !content.Any()) { return $"{title}{nl}{nl}"; } return string.Join(nl, new string[] { title, "```" } .Concat(content) .Concat(new string[] { "```" }) ) + nl + nl; } private string FormatDateTime(DateTime utc) { return utc.ToString("yyyy-MM-dd HH:mm:ss UTC", CultureInfo.InvariantCulture); } private string FormatRequestId(RequestEvent requestEvent) { return $"({emojiMaps.StringToEmojis(requestEvent.Request.Request.Id, 3)})" + $"`{requestEvent.Request.Request.Id}`"; } private string BigIntToDuration(BigInteger big) { var span = TimeSpan.FromSeconds((int)big); return Time.FormatDuration(span); } private string BigIntToByteSize(BigInteger big) { var size = new ByteSize((long)big); return size.ToString(); } private string BitIntToTestTokens(BigInteger big) { var tt = new TestToken(big); return tt.ToString(); } } }