using System.Reflection.Metadata.Ecma335;
namespace SimpleTGBot;
using Telegram.Bot;
using Telegram.Bot.Exceptions;
using Telegram.Bot.Polling;
using Telegram.Bot.Types;
using Telegram.Bot.Types.Enums;
public class TelegramBot
{
// Токен TG-бота. Можно получить у @BotFather
private const string BotToken = "ВАШ_ТОКЕН_ИДЕНТИФИКАЦИИ_БОТА";
///
/// Инициализирует и обеспечивает работу бота до нажатия клавиши Esc
///
public async Task Run()
{
// Если вам нужно хранить какие-то данные во время работы бота (массив информации, логи бота,
// историю сообщений для каждого пользователя), то это всё надо инициализировать в этом методе.
// TODO: Инициализация необходимых полей
// Инициализируем наш клиент, передавая ему токен.
var botClient = new TelegramBotClient(BotToken);
// Служебные вещи для организации правильной работы с потоками
using CancellationTokenSource cts = new CancellationTokenSource();
// Разрешённые события, которые будет получать и обрабатывать наш бот.
// Будем получать только сообщения. При желании можно поработать с другими событиями.
ReceiverOptions receiverOptions = new ReceiverOptions()
{
AllowedUpdates = new [] { UpdateType.Message }
};
// Привязываем все обработчики и начинаем принимать сообщения для бота
botClient.StartReceiving(
updateHandler: OnMessageReceived,
pollingErrorHandler: OnErrorOccured,
receiverOptions: receiverOptions,
cancellationToken: cts.Token
);
// Проверяем что токен верный и получаем информацию о боте
var me = await botClient.GetMeAsync(cancellationToken: cts.Token);
Console.WriteLine($"Бот @{me.Username} запущен.\nДля остановки нажмите клавишу Esc...");
// Ждём, пока будет нажата клавиша Esc, тогда завершаем работу бота
while (Console.ReadKey().Key != ConsoleKey.Escape){}
// Отправляем запрос для остановки работы клиента.
cts.Cancel();
}
///
/// Обработчик события получения сообщения.
///
/// Клиент, который получил сообщение
/// Событие, произошедшее в чате. Новое сообщение, голос в опросе, исключение из чата и т. д.
/// Служебный токен для работы с многопоточностью
async Task OnMessageReceived(ITelegramBotClient botClient, Update update, CancellationToken cancellationToken)
{
// Работаем только с сообщениями. Остальные события игнорируем
var message = update.Message;
if (message is null)
{
return;
}
// Будем обрабатывать только текстовые сообщения.
// При желании можно обрабатывать стикеры, фото, голосовые и т. д.
//
// Обратите внимание на использованную конструкцию. Она эквивалентна проверке на null, приведённой выше.
// Подробнее об этом синтаксисе: https://medium.com/@mattkenefick/snippets-in-c-more-ways-to-check-for-null-4eb735594c09
if (message.Text is not { } messageText)
{
return;
}
// Получаем ID чата, в которое пришло сообщение. Полезно, чтобы отличать пользователей друг от друга.
var chatId = message.Chat.Id;
// Печатаем на консоль факт получения сообщения
Console.WriteLine($"Получено сообщение в чате {chatId}: '{messageText}'");
// TODO: Обработка пришедших сообщений
// Отправляем обратно то же сообщение, что и получили
Message sentMessage = await botClient.SendTextMessageAsync(
chatId: chatId,
text: "Ты написал:\n" + messageText,
cancellationToken: cancellationToken);
}
///
/// Обработчик исключений, возникших при работе бота
///
/// Клиент, для которого возникло исключение
/// Возникшее исключение
/// Служебный токен для работы с многопоточностью
///
Task OnErrorOccured(ITelegramBotClient botClient, Exception exception, CancellationToken cancellationToken)
{
// В зависимости от типа исключения печатаем различные сообщения об ошибке
var errorMessage = exception switch
{
ApiRequestException apiRequestException
=> $"Telegram API Error:\n[{apiRequestException.ErrorCode}]\n{apiRequestException.Message}",
_ => exception.ToString()
};
Console.WriteLine(errorMessage);
// Завершаем работу
return Task.CompletedTask;
}
}