-
Notifications
You must be signed in to change notification settings - Fork 10
/
text_demo.py
97 lines (80 loc) · 4.79 KB
/
text_demo.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
# -*- coding: utf-8 -*-
import collections
import sys
from telegram.ext import Filters
from telegram.ext import MessageHandler
from telegram.ext import Updater
def dialog():
answer = yield "Здравствуйте! Меня забыли наградить именем, а как зовут вас?"
# убираем ведущие знаки пунктуации, оставляем только
# первую компоненту имени, пишем её с заглавной буквы
name = answer.text.rstrip(".!").split()[0].capitalize()
likes_python = yield from ask_yes_or_no("Приятно познакомиться, %s. Вам нравится Питон?" % name)
if likes_python:
answer = yield from discuss_good_python(name)
else:
answer = yield from discuss_bad_python(name)
def ask_yes_or_no(question):
"""Спросить вопрос и дождаться ответа, содержащего «да» или «нет».
Возвращает:
bool
"""
answer = yield question
while not ("да" in answer.text.lower() or "нет" in answer.text.lower()):
answer = yield "Так да или нет?"
return "да" in answer.text.lower()
def discuss_good_python(name):
answer = yield "Мы с вами, %s, поразительно похожи! Что вам нравится в нём больше всего?" % name
likes_article = yield from ask_yes_or_no("Ага. А как вам, кстати, статья на Хабре? Понравилась?")
if likes_article:
answer = yield "Чудно!"
else:
answer = yield "Жалко."
return answer
def discuss_bad_python(name):
answer = yield "Ай-яй-яй. %s, фу таким быть! Что именно вам так не нравится?" % name
likes_article = yield from ask_yes_or_no(
"Ваша позиция имеет право на существование. Статья "
"на Хабре вам, надо полагать, тоже не понравилась?")
if likes_article:
answer = yield "Ну и ладно."
else:
answer = yield "Что «нет»? «Нет, не понравилась» или «нет, понравилась»?"
answer = yield "Спокойно, это у меня юмор такой."
return answer
class DialogBot(object):
def __init__(self, token, generator):
self.updater = Updater(token=token) # заводим апдейтера
handler = MessageHandler(Filters.text | Filters.command, self.handle_message)
self.updater.dispatcher.add_handler(handler) # ставим обработчик всех текстовых сообщений
self.handlers = collections.defaultdict(generator) # заводим мапу "id чата -> генератор"
def start(self):
self.updater.start_polling()
def handle_message(self, bot, update):
print("Received", update.message)
chat_id = update.message.chat_id
if update.message.text == "/start":
# если передана команда /start, начинаем всё с начала -- для
# этого удаляем состояние текущего чатика, если оно есть
self.handlers.pop(chat_id, None)
if chat_id in self.handlers:
# если диалог уже начат, то надо использовать .send(), чтобы
# передать в генератор ответ пользователя
try:
answer = self.handlers[chat_id].send(update.message)
except StopIteration:
# если при этом генератор закончился -- что делать, начинаем общение с начала
del self.handlers[chat_id]
# (повторно вызванный, этот метод будет думать, что пользователь с нами впервые)
return self.handle_message(bot, update)
else:
# диалог только начинается. defaultdict запустит новый генератор для этого
# чатика, а мы должны будем извлечь первое сообщение с помощью .next()
# (.send() срабатывает только после первого yield)
answer = next(self.handlers[chat_id])
# отправляем полученный ответ пользователю
print("Answer: %r" % answer)
bot.sendMessage(chat_id=chat_id, text=answer)
if __name__ == "__main__":
dialog_bot = DialogBot(sys.argv[1], dialog)
dialog_bot.start()