Кто ты в digital-войсках: пример создания Telegram-бота на Node.js

Озадаченные вопросом как поздравить сильную половину человечества и реализовать свой творческий потенциал, компания Центр Высоких Технологий создала Telegram-бота @DigitalArmyHTCBot.

В закладки

Несмотря на то, что Telegram — запрещенный канал связи и «совсем» недоступен в России. Для поздравления наших друзей, коллег и клиентов мы решили использовать именно его.

Бот представляет собой анкету-опрос по ИТ-тематике с элементами из Project management и выбором ответа из нескольких предложенных вариантов.

Используемые технологии:

  • Node. js — js интерпретатор
  • node-telegram-bot-api — библиотека для работы с Telegram
  • AWS EC2 — сервер

Подробная инструкция по созданию бота

Инициализация бота:

const TelegramBot = require(’node-telegram-bot-api’); const bot = new TelegramBot(’your_bot_token’);

Начать необходимо с создания объекта с пользователями, в котором ключами являются chatID, а свойствами — объект с состоянием пользователя (IDLE/IN_PROGRESS/DONE) и количеством пройденных вопросов.

(На самом деле, в этом боте состояния IDLE/IN_PROGRESS/DONE не понадобились, так как логика очень простая, но обычно бывает очень удобно представить бота как конечный автомат с каким-то заданным набором состояний и переходов между ними).

const states = { IDLE: 0, IN_PROGRESS: 1, DONE: 2 }; const users = {};

При получении сообщения в аргумент функции обратного вызова или callback приходит объект типа Message, в котором есть свойство-объект chat, а у него в свою очередь есть свойство id. То есть, получить chatID, чтобы записать его в объект пользователей, можно, например, при получении команды /start (она приходит в любом случае, если пользователь начинает общение с ботом).

Выглядеть это будет примерно так:

const initialUser = { isPassed: false, currentQuestionNumber: 0 }; bot.onText(/\/start/, (msg) => { const chatId = msg.chat.id; if (!(chatID in users)) { users[chatID] = { ...initialUser }; } });

У библиотеки node-telegram-bot-api есть метод. sendPhoto, в который можно передать url изображения и библиотека будет отправлять это изображение пользователю. Однако, при его использовании мы столкнулся с проблемой — метод не отправляет изображение и выкидывает ошибку 400 «Bad request». После просмотра issues в GitHub и безуспешных поисков, мы решили использовать другой метод — .sendDocument, который принимает такой же набор аргументов, тоже принимает url и высылает изображение, но в виде документа.

Для отправки сообщения пользователю используется метод. sendMessage(chatID, message, options). Он принимает в аргументы chatID, текст сообщения и параметры. В параметрах можно, например, указать «parse_mode»: «HTML», чтобы можно было отправлять стилизованный текст (жирный, курсив и др).

Чтобы отправлять сообщения, как варианты ответа боту, есть специальное свойство в аргументе options метода. sendMessage «reply_markup», в котором есть свойство «keyboard», где указывается массив с вариантами ответов.

Все фразы содержатся в отдельном файле. Есть приветствие, вопросы с вариантами ответов и порядковым номером, профессии с ссылками на соответствующие изображения и результирующий текст.

Попробовать, что у нас получилось можно в Telegram @DigitalArmyHTCBot

Материал опубликован пользователем. Нажмите кнопку «Написать», чтобы поделиться мнением или рассказать о своём проекте.

Написать
{ "author_name": "Центр Высоких Технологий", "author_type": "self", "tags": [], "comments": 3, "likes": 13, "favorites": 5, "is_advertisement": false, "subsite_label": "dev", "id": 59440, "is_wide": false, "is_ugc": true, "date": "Mon, 25 Feb 2019 10:59:10 +0300" }
{ "id": 59440, "author_id": 244899, "diff_limit": 1000, "urls": {"diff":"\/comments\/59440\/get","add":"\/comments\/59440\/add","edit":"\/comments\/edit","remove":"\/admin\/comments\/remove","pin":"\/admin\/comments\/pin","get4edit":"\/comments\/get4edit","complain":"\/comments\/complain","load_more":"\/comments\/loading\/59440"}, "attach_limit": 2, "max_comment_text_length": 5000, "subsite_id": 235819 }

3 комментария 3 комм.

Популярные

По порядку

0

Выложите исходники на гитхаб. Сразу станет понятно, как организован файл с вопросами и вариантами ответов, где лежат изображения и т. д.
Расскажите, как деплоили в AWS, почему решили взять EC2, а не Lambda.
К сожалению, "подробная инструкция" сейчас выглядит вот так:

Ответить
1

Это немножко другое место (не хабр).
Телеграм-бот сам по себе не рокет сайенс, но зато как пример того, что из чего-то простого и давно уже всем понятного можно по прежнему сделать что-то нетривиальное.

Как по мне, это тот самый момент, когда важна форма, а не содержание.
Вот я например такой персонаж, но конфликты не решаю, а создаю и ок )))

Ответить
0

Это немножко другое место

Это подсайт Разработка, тут такое надо.

Ответить
0
{ "page_type": "article" }

Прямой эфир

[ { "id": 1, "label": "100%×150_Branding_desktop", "provider": "adfox", "adaptive": [ "desktop" ], "adfox_method": "createAdaptive", "auto_reload": true, "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "ezfl" } } }, { "id": 2, "label": "1200х400", "provider": "adfox", "adaptive": [ "phone" ], "auto_reload": true, "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "ezfn" } } }, { "id": 3, "label": "240х200 _ТГБ_desktop", "provider": "adfox", "adaptive": [ "desktop" ], "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "fizc" } } }, { "id": 4, "label": "240х200_mobile", "provider": "adfox", "adaptive": [ "phone" ], "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "flbq" } } }, { "id": 5, "label": "300x500_desktop", "provider": "adfox", "adaptive": [ "desktop" ], "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "ezfk" } } }, { "id": 6, "label": "1180х250_Interpool_баннер над комментариями_Desktop", "provider": "adfox", "adaptive": [ "desktop" ], "adfox": { "ownerId": 228129, "params": { "pp": "h", "ps": "bugf", "p2": "ffyh" } } }, { "id": 7, "label": "Article Footer 100%_desktop_mobile", "provider": "adfox", "adaptive": [ "desktop", "tablet", "phone" ], "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "fjxb" } } }, { "id": 8, "label": "Fullscreen Desktop", "provider": "adfox", "adaptive": [ "desktop", "tablet" ], "auto_reload": true, "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "fjoh" } } }, { "id": 9, "label": "Fullscreen Mobile", "provider": "adfox", "adaptive": [ "phone" ], "auto_reload": true, "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "fjog" } } }, { "id": 10, "disable": true, "label": "Native Partner Desktop", "provider": "adfox", "adaptive": [ "desktop", "tablet" ], "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "clmf", "p2": "fmyb" } } }, { "id": 11, "disable": true, "label": "Native Partner Mobile", "provider": "adfox", "adaptive": [ "phone" ], "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "clmf", "p2": "fmyc" } } }, { "id": 12, "label": "Кнопка в шапке", "provider": "adfox", "adaptive": [ "desktop" ], "adfox": { "ownerId": 228129, "params": { "p1": "bscsh", "p2": "fdhx" } } }, { "id": 13, "label": "DM InPage Video PartnerCode", "provider": "adfox", "adaptive": [ "desktop", "tablet", "phone" ], "adfox_method": "createAdaptive", "adfox": { "ownerId": 228129, "params": { "pp": "h", "ps": "bugf", "p2": "flvn" } } }, { "id": 14, "label": "Yandex context video banner", "provider": "yandex", "yandex": { "block_id": "VI-223676-0", "render_to": "inpage_VI-223676-0-1104503429", "adfox_url": "//ads.adfox.ru/228129/getCode?pp=h&ps=bugf&p2=fpjw&puid1=&puid2=&puid3=&puid4=&puid8=&puid9=&puid10=&puid21=&puid22=&puid31=&puid32=&puid33=&fmt=1&dl={REFERER}&pr=" } }, { "id": 15, "label": "Плашка на главной", "provider": "adfox", "adaptive": [ "desktop", "tablet", "phone" ], "adfox": { "ownerId": 228129, "params": { "p1": "byudx", "p2": "ftjf" } } }, { "id": 16, "label": "Кнопка в шапке мобайл", "provider": "adfox", "adaptive": [ "tablet", "phone" ], "adfox": { "ownerId": 228129, "params": { "p1": "byzqf", "p2": "ftwx" } } }, { "id": 17, "label": "Stratum Desktop", "provider": "adfox", "adaptive": [ "desktop" ], "auto_reload": true, "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "fzvb" } } }, { "id": 18, "label": "Stratum Mobile", "provider": "adfox", "adaptive": [ "tablet", "phone" ], "auto_reload": true, "adfox": { "ownerId": 228129, "params": { "pp": "g", "ps": "bugf", "p2": "fzvc" } } }, { "id": 19, "label": "Тизер на главной", "provider": "adfox", "adaptive": [ "desktop", "tablet", "phone" ], "auto_reload": true, "adfox": { "ownerId": 228129, "params": { "p1": "cbltd", "p2": "gazs" } } } ]
Голосовой помощник выкупил
компанию-создателя
Подписаться на push-уведомления
{ "page_type": "default" }