Введение
Месяца три назад мы с друзьями по сети «ВКонтакте» в общем чате (беседе) играли в угадайку — игру по мультфильму «Смешарики», который мы все любим. Правила: ведущий пишет цитаты персонажей мультика, игроки угадывают из какой серии эта цитата. Я решил сделать игру более интересной, и назначить на роль ведущего бота.
Создание группы
Для того, чтобы создать группу или же сообщество для нашего бота, нужно зайти в раздел «Сообщества» в ВКонтакте и нажать на кнопку «Создать сообщество» в правом верхнем углу.
Выбираем тип сообщества «Группа по интересам».
Придумываем название, выбираем тематику, тип группы оставляем открытой.
Теперь нам потребуется узнать ID и создать токен группы.
Если же ID можно узнать в адресной строке после слова «club», то с токеном нужно будет немного запариться.
Заходим в управление группой:
Выбираем раздел «Работа с API» и включаем там «Long Poll API» для работы бота.
Далее переходим в «Ключи доступа», создаём ключ доступа с такими настройками:
Получившийся ключ доступа и есть наш токен:
Осталось разве что разрешить добавлять группу в беседы.
Заходим в раздел «Сообщения -> Настройки для бота», включаем «Возможности ботов» и жмем на «Разрешить добавлять сообщество в беседы». Сохраняем.
Установка модуля vk_api
Для того, чтобы установить модуль для работы с «Вконтакте API» нужно прописать в командную строку следующую команду:
pip3 install vk_api
Ещё нужно создать в директории бота файл «points.pickle».
Базовые функции
Переходим к программной части:
Подключаем нужные плагины: pickle, random, requests и библиотеку vk_api, благодаря которой можно работать с «ВКонтакте».
import random
import pickle
import requests
from vk_api.bot_longpoll import VkBotLongPoll, VkBotEventType
#подключаемся к BotLongpoll - он нам нужен для работы именно с беседами.
Создаем словарь с цитатами и словари для хранения информации:
series = {"серия":["цитата 1", "цитата 2"], "серия":["цитата 1", "цитата 2"]}
series_in_chats = {}
quotes_in_chats = {}
points = {}
Сами цитаты заключены в списки в ключах словаря «series» для того, чтобы на одну серию приходилось много цитат.
Переменная «series_in_chats» нужна для хранения информации о серии в отдельной беседе, переменная «quotes_in_chats» аналогично, только для хранения информации о цитате.
Теперь подключаемся к группе:
vk_session = vk_api.VkApi(token="Ваш токен") #токен и ID группы
longpoll = VkBotLongPoll(vk_session, "ID Вашей группы") #мы нашли ранее
vk = vk_session.get_api()
Можно создать функцию для более удобной отправки сообщений:
def send(ch_id, msg):
vk.messages.send(chat_id=ch_id, random_id=random.randint(1, 9999999), message=msg)
Аргументы:
ch_id #определяет ID чата, в который нужно отправить сообщение
msg #определяет само сообщение
vk.messages.send(chat_id, random_id, message)
Метод для отправки сообщений.
Обязательные параметры:
chat_id - ID чата, из которого прилетело сообщение,
random_id - случайный идентификатор сообщения, если честно сам не очень понимаю что это,
message - текст сообщения.
Подробнее о методе тут.
Обработка сообщений
while True: #бесконечный цикл
try:
for event in longpoll.listen():
if event.type == VkBotEventType.MESSAGE_NEW and event.from_chat:
chat_id = int(event.chat_id) #ID чата, из которого пришло сообщение
text = event.object.text.lower() #текст сообщения
man = event.obj.from_id #ID человека или группы, написавшего сообщение
if chat_id not in series_in_chats:
episode = random.choice(series.keys()) #выбираем случайную серию
#из ключей словаря
quote = random.choice(series.get(episode)) #выбираем случайную цитату из
#значения серии
series_in_chats.update({chat_id:episode}) #обновляем словарь для хранения
quotes_in_chats.update({chat_id:quote}) #обновляем словарь для хранения
#мы написали этот раздел для того, чтобы в каждой беседе
#была своя серия и цитата из неё.
except:
continue
#конструкция try except - если вдруг случится какая-нибудь ошибка
#бот не сломается, а продолжит работу
Теперь мы можем обрабатывать сообщения полученные группой, отправлять сообщения от имени группы и многое другое.
Внедряем очки
Этот код нужно вставить после проверки «if chat_id not in series_in_chats»:
if points == {}:
with open('points.pickle', 'wb') as f:
pickle.dump(points, f)
with open('points.pickle', 'rb') as f:
points = pickle.load(f)
if isinstance(points.get(man), int) == False:
points.update({man:0})
with open('data.pickle', 'wb') as f:
pickle.dump(points, f)
#если человека нет в словаре points
#мы его туда добавляем
Угадайка
Делаем проверку: если полученное в беседе сообщение будет равняться слову «угадайка», бот будет отвечать цитатой, которую нужно угадать (все это после проверки «if isinstance»):
if text == "угадайка":
send(chat_id, "Отгадайте, из какой серии эта цитата: "+ quotes_in_chats.get(chat_id))
Осталось только сделать проверку на правильность серии — если в сообщении будет присутствовать название серии, бот ответит, что серия была угадана:
if series_in_chats.get(chat_id) in text:
send(chat_id, "Вы ответили верно!")
points.update({man:points.get(man)+1}) #прибавляем 1 балл
with open('points.pickle', 'wb') as f:
pickle.dump(points, f) #сохраняем информацию об очках в файл
send(chat_id, "Вы ответили верно!\nУ вас столько очков: "+str(points.get(man)))
#меняем цитату и серию:
episode = random.choice(series.keys()) #выбираем случайную серию
#из ключей словаря
quote = random.choice(series.get(episode)) #выбираем случайную цитату из
#значения серии
series_in_chats.update({chat_id:episode}) #обновляем словарь для хранения
quotes_in_chats.update({chat_id:quote}) #обновляем словарь для хранения
Заключение
По итогу, бот будет работать как-то так:
Разумеется, на этой основе можно создать практически любого бота — не только угадайку.
Созданный бот можно увидеть в сообществе ВКонтакте "Обитель Квестов | Смешарики".
Он, кстати, работает не только в чатах, но и в личных сообщениях.
А ещё в нем есть несколько штучек, о которых я здесь не рассказал.
saipr
А что это за штучки? Не намекнете?
svyatikov Автор
Для бесед сделана схватка — соревнование между участниками, в скором времени хочу реализовать схватку в личных сообщениях.