Простыми словами про обработку текстовых запросов пользователя в Телеграмм ботах на java

найдено на просторах Интернета

найдено на просторах Интернета

Мотивация

У разработчика, предпочитающего индивидуальные проекты, есть множество способов самореализации. Для меня, например, приоритетным является создание небольших Telegram-ботов на java. Ведь помимо того, что в процессе разработки всегда можно наглядно проверить работу программы на любом этапе ее реализации, Telegram-боты, на мой взгляд, имеют потребительский потенциал в основном за счет того, что, являясь программой, они не нуждаются в отдельной установке на устройство. Достаточно пользоваться мессенджером Telegram и запустить в нем бот с подходящим функционалом, что, как минимум, экономит ресурсы самого устройства.

Ранее на Хабре я уже делилась с читателями своим опытом самостоятельного создания несложных Telegram-ботов на java, а также небольшими пошаговыми инструкциями по решению отдельных вопросов, возникающих при написании кода. После новогодней разработки очередного чат-бота хочу продолжить сложившуюся традицию и рассказать немного о некоторых нюансах одного из распространенных способов взаимодействия с ботом — принятие и обработка от пользователя запроса с сообщением.

Детализация

Тем, кто так или иначе уже сталкивался с разработкой Telegram-ботов на java, известно, что для того, чтобы класс, содержащий логику бота, реализовывал взаимодействие с сервисами Telegram, его необходимо унаследовать от класса TelegramLongPollingBot и реализовать следующие его базовые методы:

— public void onUpdateReceived (Update update);

— public String getBotUsername ();

— public String getBotToken ().

В своей сегодняшней статье я как раз остановлюсь на некоторых деталях реализации метода onUpdateReceived (Update update).

Каждый раз, когда кто-то отправляет личное сообщение боту, этот метод будет вызываться автоматически, и вы сможете обработать параметр, который содержит сообщение, а также множество другой информации.  

Основной функционал моего последнего бота прост: он принимает от пользователя сообщение с двумя параметрами (вес и рост, указанными целыми цифрами и разделенными одним пробелом), рассчитывает по формуле индекс массы тела (ИМТ) и высылает пользователю результат с характеристикой и краткими рекомендациями. То есть, помимо команд, которые предусмотрены в моем боте, обрабатывается только одно текстовое сообщение от пользователя.

Казалось бы, все предельно просто — достаточно лишь проверить при реализации метода onUpdateReceived, есть ли во входящем обновлении (update) сообщение (метод getMessage ()) и есть ли в таком сообщении текст (метод hasText ()). Но текст должен быть с определенными параметрами — это должно быть 2 целые цифры, разделенные пробелом. Все остальные запросы должны блокироваться, а пользователю высылаться сообщение об ошибке и повторная просьба корректно направить запрос.

Реализация

 Для себя я нашла следующий вариант обработки и запроса:

1. Проверить входящее сообщение на наличие в нем пробелов и, при наличии таковых, разделить строку на части и создать из них массив.

 2. В созданном массиве проверить длину такого массива и принадлежность элементов к целым положительным числам и, при соблюдении условий, вызвать метод, в котором производится расчет ИМТ, присвоив его параметрам соответствующие значения элементов массива.

Наиболее оптимальным способом для проверки принадлежности элементов текстового массива к целым положительным числам я выбрала применение регулярных выражений, именно с ними код выглядел компактно и отрабатывал корректно.

Синтаксис регулярных выражений основан на использовании символов <([{\^-=$!|]})?*+.>, которые можно комбинировать с буквенными символами.

Поскольку меня интересовали только целые числа, я использовала следующие символы регулярного выражения:

»\d» — соответствует любой одной цифре и заменяет собой выражение [0–9];

»+» — частота появления элемента (1 или более цифра в выражении),

и matches () — метод, который возвращает true только тогда, когда вся строка соответствует заданному регулярному выражению.

 3. Во всех остальных случаях, не подпадающих под заданные проверки, пользователь должен получать сообщение об ошибке и необходимости корректно ввести запрос.  

В итоге, если опустить реализацию методов, участвующих в обработке команд, расчете ИМТ и формировании рекомендаций, передаче данных в базу данных, у меня получилась вот такая реализация метода onUpdateReceived:

@SneakyThrows
@Override

    public void onUpdateReceived(Update update) {

// проверка, содержит ли обновление сообщение и содержит ли сообщение текст
        if (update.hasMessage() && update.getMessage().hasText()) {

            String message_text = update.getMessage().getText();
            Long chat_id = update.getMessage().getChatId();

            Message message = update.getMessage();
            User from = message.getFrom();

            SendMessage sendMessage = new SendMessage();
            sendMessage.setChatId(update.getMessage().getChatId().toString());


            if (commandType.types().contains(message_text)) {
                commandHandler.onUpdateReceived(update);


// если сообщение с текстом содержит пробел
            } else if (message_text.contains(" ")) {


// создаем строковый массив, в котором элементы образуются черед разделить - пробел
                String[] weightAntHeight = update.getMessage().getText().split(" ");


// если длина массива 2 элемента, которые соответствуют целым числам
                if (weightAntHeight.length == 2
                        && weightAntHeight[0].matches("\\d+")
                        && weightAntHeight[1].matches("\\d+")) {


// производим расчет ИМТ, высылаем пользователю результаты и рекомендации, заносим результаты в базу данных
                    String resultImt = String.format("%.1f", ImtCount.imt(weightAntHeight[0], weightAntHeight[1]));
                    String resultWithDescription = ImtCount.description(weightAntHeight[0], weightAntHeight[1]);

                    WriteUser.writeUserIntoDb(LocalDateTime.now().withNano(0),
                            from.getId(), from.getFirstName()
                            , resultImt
                    );
                    sendMessage.setText(resultWithDescription);
                    execute(sendMessage);


// во всех остальных случаях выдается сообщение об ошибке и инструкция с правилами направления запроса.
                } else {
                    execute(Sender.sendMessage(chat_id, UNKNOWN + INSTRUCTION));
                }
            }

            else {
                execute(Sender.sendMessage(chat_id, UNKNOWN + INSTRUCTION));
            }

        }

}

Описанный способ может применяться, например, и при регистрации пользователя для формирования параметров обработки логина и пароля.

Резюмирование

Да, статей, посвященный разработке Telegram-ботов, великое множество, но, как показывает мой личный опыт поиска нужной информации, они в большинстве своем однотипны и зачастую не содержат ответов (разъяснений) на практические вопросы. В своих публикациях я делюсь самостоятельно пройденным путем больше с новичками и с теми, кому в принципе как и мне интересна разработка Telegram-ботов.

Надеюсь, этот разбор и реализация обработки текстовых запросов пользователя в проекте, код которого выложен на GitHub, кому-нибудь поможет при разработке своих телеграмм ботов.

Если интересно, как работает мой последний телеграмм бот, то милости прошу: Индекс массы тела.

© Habrahabr.ru