Initial commit

This commit is contained in:
github-classroom[bot] 2025-04-12 09:54:46 +00:00 committed by Slavasil
commit 222fbfcefc
9 changed files with 195 additions and 0 deletions

5
.gitignore vendored Normal file
View File

@ -0,0 +1,5 @@
bin/
obj/
/packages/
riderModule.iml
/_ReSharper.Caches/

View File

@ -0,0 +1,13 @@
# Default ignored files
/shelf/
/workspace.xml
# Rider ignored files
/contentModel.xml
/projectSettingsUpdater.xml
/modules.xml
/.idea.SimpleTGBot.iml
# Editor-based HTTP Client requests
/httpRequests/
# Datasource local storage ignored files
/dataSources/
/dataSources.local.xml

View File

@ -0,0 +1,4 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="Encoding" addBOMForNewFiles="with BOM under Windows, with no BOM otherwise" />
</project>

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="UserContentModel">
<attachedFolders />
<explicitIncludes />
<explicitExcludes />
</component>
</project>

View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="VcsDirectoryMappings">
<mapping directory="$PROJECT_DIR$" vcs="Git" />
</component>
</project>

16
SimpleTGBot.sln Normal file
View File

@ -0,0 +1,16 @@

Microsoft Visual Studio Solution File, Format Version 12.00
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SimpleTGBot", "SimpleTGBot\SimpleTGBot.csproj", "{BFA186C8-2B12-4779-8691-9914B54DD319}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{BFA186C8-2B12-4779-8691-9914B54DD319}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{BFA186C8-2B12-4779-8691-9914B54DD319}.Debug|Any CPU.Build.0 = Debug|Any CPU
{BFA186C8-2B12-4779-8691-9914B54DD319}.Release|Any CPU.ActiveCfg = Release|Any CPU
{BFA186C8-2B12-4779-8691-9914B54DD319}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
EndGlobal

11
SimpleTGBot/Program.cs Normal file
View File

@ -0,0 +1,11 @@
namespace SimpleTGBot;
public static class Program
{
// Метод main немного видоизменился для асинхронной работы
public static async Task Main(string[] args)
{
TelegramBot telegramBot = new TelegramBot();
await telegramBot.Run();
}
}

View File

@ -0,0 +1,14 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Telegram.Bot" Version="19.0.0-preview.2" />
</ItemGroup>
</Project>

118
SimpleTGBot/TelegramBot.cs Normal file
View File

@ -0,0 +1,118 @@
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 = "ВАШ_ТОКЕН_ИДЕНТИФИКАЦИИ_БОТА";
/// <summary>
/// Инициализирует и обеспечивает работу бота до нажатия клавиши Esc
/// </summary>
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();
}
/// <summary>
/// Обработчик события получения сообщения.
/// </summary>
/// <param name="botClient">Клиент, который получил сообщение</param>
/// <param name="update">Событие, произошедшее в чате. Новое сообщение, голос в опросе, исключение из чата и т. д.</param>
/// <param name="cancellationToken">Служебный токен для работы с многопоточностью</param>
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);
}
/// <summary>
/// Обработчик исключений, возникших при работе бота
/// </summary>
/// <param name="botClient">Клиент, для которого возникло исключение</param>
/// <param name="exception">Возникшее исключение</param>
/// <param name="cancellationToken">Служебный токен для работы с многопоточностью</param>
/// <returns></returns>
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;
}
}