// (c) Copyright Ascensio System SIA 2010-2022 // // This program is a free software product. // You can redistribute it and/or modify it under the terms // of the GNU Affero General Public License (AGPL) version 3 as published by the Free Software // Foundation. In accordance with Section 7(a) of the GNU AGPL its Section 15 shall be amended // to the effect that Ascensio System SIA expressly excludes the warranty of non-infringement of // any third-party rights. // // This program is distributed WITHOUT ANY WARRANTY, without even the implied warranty // of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. For details, see // the GNU AGPL at: http://www.gnu.org/licenses/agpl-3.0.html // // You can contact Ascensio System SIA at Lubanas st. 125a-25, Riga, Latvia, EU, LV-1021. // // The interactive user interfaces in modified source and object code versions of the Program must // display Appropriate Legal Notices, as required under Section 5 of the GNU AGPL version 3. // // Pursuant to Section 7(b) of the License you must retain the original Product logo when // distributing the program. Pursuant to Section 7(e) we decline to grant you any rights under // trademark law for use of our trademarks. // // All the Product's GUI elements, including illustrations and icon sets, as well as technical writing // content are licensed under the terms of the Creative Commons Attribution-ShareAlike 4.0 // International. See the License terms at http://creativecommons.org/licenses/by-sa/4.0/legalcode namespace ASC.TelegramService; [Singletone(Additional = typeof(TelegramHandlerExtension))] public class TelegramHandler { private readonly Dictionary _clients; private readonly CommandModule _command; private readonly ILogger _log; private readonly IServiceScopeFactory _scopeFactory; private readonly IDistributedCache _distributedCache; public TelegramHandler(IDistributedCache distributedCache, CommandModule command, ILogger logger, IServiceScopeFactory scopeFactory) { _command = command; _log = logger; _scopeFactory = scopeFactory; _clients = new Dictionary(); _distributedCache = distributedCache; ServicePointManager.SecurityProtocol |= SecurityProtocolType.Tls12 | SecurityProtocolType.Tls13; } public async Task SendMessage(NotifyMessage msg) { if (string.IsNullOrEmpty(msg.Reciever)) { return; } if (!_clients.ContainsKey(msg.TenantId)) { return; } var scope = _scopeFactory.CreateScope(); var telegramDao = scope.ServiceProvider.GetService(); var client = _clients[msg.TenantId].Client; try { var tgUser = telegramDao.GetUser(Guid.Parse(msg.Reciever), msg.TenantId); if (tgUser == null) { _log.DebugCouldntFind(msg.Reciever); return; } var chat = await client.GetChatAsync(tgUser.TelegramUserId); await client.SendTextMessageAsync(chat, msg.Content, ParseMode.MarkdownV2); } catch (Exception e) { _log.DebugCouldntSend(msg.Reciever, e); } } public void DisableClient(int tenantId) { if (!_clients.ContainsKey(tenantId)) { return; } var client = _clients[tenantId]; if (client.CancellationTokenSource != null) { client.CancellationTokenSource.Cancel(); client.CancellationTokenSource.Dispose(); client.CancellationTokenSource = null; } _clients.Remove(tenantId); } public void CreateOrUpdateClientForTenant(int tenantId, string token, int tokenLifespan, string proxy, bool startTelegramService, CancellationToken stoppingToken, bool force = false) { var scope = _scopeFactory.CreateScope(); var telegramHelper = scope.ServiceProvider.GetService(); var newClient = telegramHelper.InitClient(token, proxy); if (_clients.TryGetValue(tenantId, out var client)) { client.TokenLifeSpan = tokenLifespan; if (token != client.Token || proxy != client.Proxy) { if (startTelegramService) { if (!telegramHelper.TestingClient(newClient)) { return; } } if (client.CancellationTokenSource != null) { client.CancellationTokenSource.Cancel(); client.CancellationTokenSource.Dispose(); client.CancellationTokenSource = null; } BindClient(newClient, tenantId, stoppingToken); client.Client = newClient; client.Token = token; client.Proxy = proxy; } } else { if (!force && startTelegramService) { if (!telegramHelper.TestingClient(newClient)) { return; } } BindClient(newClient, tenantId, stoppingToken); _clients.Add(tenantId, new TenantTgClient() { Token = token, Client = newClient, Proxy = proxy, TenantId = tenantId, TokenLifeSpan = tokenLifespan }); } } public void RegisterUser(string userId, int tenantId, string token) { if (!_clients.ContainsKey(tenantId)) { return; } var userKey = UserKey(userId, tenantId); var dateExpires = DateTimeOffset.Now.AddMinutes(_clients[tenantId].TokenLifeSpan); _distributedCache.SetString(token, userKey, new DistributedCacheEntryOptions { AbsoluteExpiration = dateExpires }); } private void BindClient(TelegramBotClient client, int tenantId, CancellationToken cancellationToken) { var cts = new CancellationTokenSource(); _clients[tenantId].CancellationTokenSource = cts; var linkedCts = CancellationTokenSource.CreateLinkedTokenSource(cts.Token, cancellationToken); client.StartReceiving(updateHandler: (botClient, exception, cancellationToken) => HandleUpdateAsync(botClient, exception, cancellationToken, tenantId), errorHandler: HandleErrorAsync, cancellationToken: linkedCts.Token); } async Task HandleUpdateAsync(ITelegramBotClient botClient, Update update, CancellationToken cancellationToken, int tenantId) { if (update.Type != UpdateType.Message) { return; } if (update.Message.Type != MessageType.Text) { return; } if (String.IsNullOrEmpty(update.Message.Text) || update.Message.Text[0] != '/') { return; } await _command.HandleCommand(update.Message, botClient, tenantId); } Task HandleErrorAsync(ITelegramBotClient botClient, Exception exception, CancellationToken cancellationToken) { String errorMessage; if (exception is ApiRequestException) { errorMessage = String.Format("Telegram API Error:\n[{0}]\n{1}", ((ApiRequestException)exception).ErrorCode, ((ApiRequestException)exception).Message); } else { errorMessage = exception.ToString(); } _log.Error(errorMessage); return Task.CompletedTask; } private string UserKey(string userId, int tenantId) { return string.Format("{0}:{1}", userId, tenantId); } } public static class TelegramHandlerExtension { public static void Register(DIHelper services) { services.TryAdd(); } }