using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.Linq; using System.Threading; using System.Threading.Tasks; using Telegram.Bot.Exceptions; using Telegram.Bot.Types; using TelegramBotBase.Args; using TelegramBotBase.Attributes; using TelegramBotBase.Base; using TelegramBotBase.Enums; namespace TelegramBotBase.Form { /// /// A form which cleans up old messages sent within /// public class AutoCleanForm : FormBase { [SaveState] public List OldMessages { get; set; } [SaveState] public eDeleteMode DeleteMode { get; set; } [SaveState] public eDeleteSide DeleteSide { get; set; } public AutoCleanForm() { this.OldMessages = new List(); this.DeleteMode = eDeleteMode.OnEveryCall; this.DeleteSide = eDeleteSide.BotOnly; this.Init += AutoCleanForm_Init; this.Closed += AutoCleanForm_Closed; } private async Task AutoCleanForm_Init(object sender, InitEventArgs e) { if (this.Device == null) return; this.Device.MessageSent += Device_MessageSent; this.Device.MessageReceived += Device_MessageReceived; this.Device.MessageDeleted += Device_MessageDeleted; } private void Device_MessageDeleted(object sender, MessageDeletedEventArgs e) { if (OldMessages.Contains(e.MessageId)) OldMessages.Remove(e.MessageId); } private void Device_MessageReceived(object sender, MessageReceivedEventArgs e) { if (this.DeleteSide == eDeleteSide.BotOnly) return; this.OldMessages.Add(e.Message.MessageId); } private void Device_MessageSent(object sender, MessageSentEventArgs e) { if (this.DeleteSide == eDeleteSide.UserOnly) return; this.OldMessages.Add(e.Message.MessageId); } public override async Task PreLoad(MessageResult message) { if (this.DeleteMode != eDeleteMode.OnEveryCall) return; await MessageCleanup(); } /// /// Adds a message to this of removable ones /// /// public void AddMessage(Message m) { this.OldMessages.Add(m.MessageId); } /// /// Adds a message to this of removable ones /// /// public void AddMessage(int messageId) { this.OldMessages.Add(messageId); } /// /// Keeps the message by removing it from the list /// /// public void LeaveMessage(int Id) { this.OldMessages.Remove(Id); } /// /// Keeps the last sent message /// public void LeaveLastMessage() { if (this.OldMessages.Count == 0) return; this.OldMessages.RemoveAt(this.OldMessages.Count - 1); } private async Task AutoCleanForm_Closed(object sender, EventArgs e) { if (this.DeleteMode != eDeleteMode.OnLeavingForm) return; MessageCleanup().Wait(); } /// /// Cleans up all remembered messages. /// /// public async Task MessageCleanup() { var oldMessages = OldMessages.AsEnumerable(); #if !NETSTANDARD2_0 while (oldMessages.Any()) { using var cts = new CancellationTokenSource(); var deletedMessages = new ConcurrentBag(); var parallelQuery = OldMessages.AsParallel() .WithCancellation(cts.Token); Task retryAfterTask = null; try { parallelQuery.ForAll(i => { try { Device.DeleteMessage(i).GetAwaiter().GetResult(); deletedMessages.Add(i); } catch (ApiRequestException req) when (req.ErrorCode == 400) { deletedMessages.Add(i); } }); } catch (AggregateException ex) { cts.Cancel(); var retryAfterSeconds = ex.InnerExceptions .Where(e => e is ApiRequestException apiEx && apiEx.ErrorCode == 429) .Max(e => (int?)((ApiRequestException)e).Parameters.RetryAfter) ?? 0; retryAfterTask = Task.Delay(retryAfterSeconds * 1000); } //deletedMessages.AsParallel().ForAll(i => Device.OnMessageDeleted(new MessageDeletedEventArgs(i))); oldMessages = oldMessages.Where(x => !deletedMessages.Contains(x)); if (retryAfterTask != null) await retryAfterTask; } #else while (oldMessages.Any()) { using (var cts = new CancellationTokenSource()) { var deletedMessages = new ConcurrentBag(); var parallelQuery = OldMessages.AsParallel() .WithCancellation(cts.Token); Task retryAfterTask = null; try { parallelQuery.ForAll(i => { try { Device.DeleteMessage(i).GetAwaiter().GetResult(); deletedMessages.Add(i); } catch (ApiRequestException req) when (req.ErrorCode == 400) { deletedMessages.Add(i); } }); } catch (AggregateException ex) { cts.Cancel(); var retryAfterSeconds = ex.InnerExceptions .Where(e => e is ApiRequestException apiEx && apiEx.ErrorCode == 429) .Max(e => (int?)((ApiRequestException)e).Parameters.RetryAfter) ?? 0; retryAfterTask = Task.Delay(retryAfterSeconds * 1000); } //deletedMessages.AsParallel().ForAll(i => Device.OnMessageDeleted(new MessageDeletedEventArgs(i))); oldMessages = oldMessages.Where(x => !deletedMessages.Contains(x)); if (retryAfterTask != null) await retryAfterTask; } } #endif OldMessages.Clear(); } } }