-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathbot.py
208 lines (184 loc) · 12.8 KB
/
bot.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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
import sqlite3
import asyncio
from datetime import datetime, timedelta
from random import randint
from dateparser import parse
from vkbottle import VKAPIError
from vkbottle.bot import Bot, BotLabeler, Message
from config import token
async def main():
"""Основная функция бота. В ней скрипт ждёт от пользователя или беседы сообщения и обрабатывает их."""
@bot.on.message(
text=["/добавить <deadline> на <time>", "/добавь <deadline> на <time>", "дед, добавь <deadline> на <time>"])
@bot.on.private_message(text=["добавить <deadline> на <time>", "добавь <deadline> на <time>"])
async def set_deadline(message: Message, deadline: str, time: str):
"""Добавляет дедлайн для чата. Кроме собственно сообщения принимает параметры `deadline` (str) - название
дедлайна, а также `time` (str) - дату и время."""
deadline = deadline.replace("дедлайн ", "") # Убирает слово "дедлайн", если пользователь ввёл его
time = unify_time(time)
if time is None:
await message.answer("Ошибка: неправильный формат даты и времени")
return
row = {"chat": message.peer_id, "deadline": deadline, "time": time}
if message.from_id == message.peer_id: # Трюк, чтобы проверить, пришло сообщение из группового или
# одиночного чата
row["isGroup"] = False
else:
row["isGroup"] = True
cur.execute("INSERT INTO Deadlines VALUES(:chat, :deadline, :time, :isGroup);", row)
database.commit()
await message.answer(f"Готово, добавил дедлайн <<{deadline}>> на {time}")
@bot.on.message(text=["/удалить <deadline>", "/удали <deadline>", "дед, удали <deadline>"])
@bot.on.private_message(text=["удалить <deadline>", "удали <deadline>"])
async def remove_deadline(message: Message, deadline: str):
"""Удаляет дедлайн для чата.
Кроме собственно сообщения принимает параметр `deadline` (str) - название дедлайна.
Если для чата задано несколько дедлайнов с одинаковым названием, будет удалён тот, который добавлен раньше."""
deadline = deadline.replace("дедлайн ", "")
deadline = cur.execute("SELECT ROWID, deadline FROM Deadlines WHERE chat=? AND deadline=? COLLATE NOCASE;",
(message.peer_id, deadline)).fetchone()
if deadline is None:
pronoun = "тебя" if message.from_id == message.peer_id else "вас" # Делаем разные местоимения
# в зависимости от типа чата
await message.answer(f"У {pronoun} нет такого дедлайна")
else:
cur.execute("DELETE FROM Deadlines WHERE ROWID=?;", (deadline[0],))
database.commit()
# Мы не используем `DELETE FROM Deadlines WHERE` сразу, так как у пользователя может быть несколько
# дедлайнов с одинаковым названием.
await message.answer(f"Готово, удалил дедлайн <<{deadline[1]}>>")
@bot.on.message(
text=["/изменить <deadline> на <time>", "/измени <deadline> на <time>", "дед, измени <deadline> на <time>"])
@bot.on.private_message(text=["изменить <deadline> на <time>", "измени <deadline> на <time>"])
async def change_deadline(message: Message, deadline: str, time: str):
"""Изменяет уже существующий дедлайн. Кроме собственно сообщения принимает параметры `deadline` (str) -
название изменяемого дедлайна, а также `time` (str) - новое время дедлайна. Если для чата задано несколько
дедлайнов с одинаковым названием, будет изменён тот, который добавлен раньше. """
deadline = deadline.replace("дедлайн ", "")
time = unify_time(time)
if time is None:
await message.answer("Ошибка: неправильный формат даты и времени")
return
deadline = cur.execute("SELECT ROWID, deadline FROM Deadlines WHERE chat=? AND deadline=? COLLATE NOCASE;",
(message.peer_id, deadline)).fetchone()
if deadline is None:
pronoun = "тебя" if message.from_id == message.peer_id else "вас"
await message.answer(f"У {pronoun} нет такого дедлайна")
else:
cur.execute("UPDATE Deadlines SET time=? WHERE ROWID=?;", (time, deadline[0]))
database.commit()
await message.answer(f"Готово, изменил дедлайн <<{deadline[1]}>> на {time}")
@bot.on.message(text=["/сбросить", "/сбрось", "дед, сбрось"])
@bot.on.private_message(text=["сбросить", "сбрось"])
async def clear_deadlines(message: Message):
"""Удаляет все дедлайны для данного чата."""
cur.execute("DELETE FROM Deadlines WHERE chat=?;", (message.peer_id,))
database.commit()
await message.answer("Готово, удалил все дедлайны")
@bot.on.message(
text=["/дедлайны", "/скажи дедлайны", "/расскажи про дедлайны", "/что по дедлайнам", "/что по дедлайнам?",
"дед, скажи дедлайны", "дед, расскажи про дедлайны", "дед, что по дедлайнам", "дед, что по дедлайнам?"])
@bot.on.private_message(
text=["дедлайны", "скажи дедлайны", "расскажи про дедлайны", "что по дедлайнам", "что по дедлайнам?"])
async def send_deadlines(message: Message):
"""Отправляет список текущих дедлайнов для данного чата."""
deadlines = cur.execute("SELECT deadline, time FROM Deadlines WHERE chat=?;", (message.peer_id,)).fetchall()
if len(deadlines) == 0:
pronoun = "тебя" if message.from_id == message.peer_id else "вас"
await message.answer(f"У {pronoun} нет текущих дедлайнов")
else:
answer = ""
for deadline in deadlines:
answer += f"""Дедлайн: {deadline[0]}
Когда: {deadline[1]}
"""
await message.answer(answer)
@bot.on.message(text=["/когда <deadline>?", "/когда <deadline>", "дед, когда <deadline>?", "дед, когда <deadline>"])
@bot.on.private_message(text=["когда <deadline>?", "когда <deadline>"])
async def when_deadline(message: Message, deadline: str):
"""Сообщает время конкретного дедлайна. Кроме собственно сообщения принимает параметр `deadline` (str) -
название дедлайна. Если для чата задано несколько дедлайнов с одинаковым названием, будет отправлена
информация по тому, который задан раньше. """
deadline = deadline.replace("дедлайн ", "")
deadline = cur.execute("SELECT deadline, time FROM Deadlines WHERE chat=? AND deadline=? COLLATE NOCASE;",
(message.peer_id, deadline)).fetchone()
if deadline is None:
pronoun = "тебя" if message.from_id == message.peer_id else "вас"
await message.answer(f"У {pronoun} нет такого дедлайна")
else:
await message.answer(f"{deadline[0]}: {deadline[1]}")
await asyncio.gather(bot.run_polling(), scheduler()) # Запускаем одновременно и бота, и планировщик
def unify_time(time: str):
"""Приводит введённое пользователем время к формату 'dd.mm.yyyy HH:MM'. Возвращает строку. Если время распознать
не удалось, возвращает `None`. """
time = time.strip()
time = time.lower()
time = parse(time, languages=["ru"], settings={
"DATE_ORDER": "DMY", # Переопределяет порядок элементов даты для неочевидных дат
"PREFER_DATES_FROM": "future" # Будет ориентироваться на ближайшее будущее для незавершённых дат
})
if not time:
return None
time = time.strftime("%d.%m.%Y %H:%M")
return time
def ignore_case_collation(value1_: str, value2_: str):
"""Добавляет поддержку регистронезависимого сравнения кириллицы в SQLite."""
if value1_.lower() == value2_.lower():
return 0
elif value1_.lower() < value2_.lower():
return -1
else:
return 1
async def scheduler():
"""В фоновом режиме отправляет сообщение в чат, для которого установлен дедлайн, если текущее время совпадает с
временем дедлайна или отличается (в меньшую сторону) на час или на сутки. """
while True:
time_format = "%d.%m.%Y %H:%M"
today = datetime.today()
current_time = today.strftime(time_format)
hour_after_time = (today + timedelta(seconds=3600)).strftime(time_format)
day_after_time = (today + timedelta(days=1)).strftime(time_format)
deadlines = cur.execute(
"SELECT ROWID, chat, deadline, time, isGroup FROM Deadlines WHERE time=? OR time=? OR time=?;", (
current_time, hour_after_time,
day_after_time)).fetchall() # Выбираем только те дедлайны, которые установлены на интересующее нас
# время
for deadline in deadlines:
if deadline[3] == current_time:
await send(deadline[1], f"{deadline[2]}: время истекло")
cur.execute("DELETE FROM Deadlines WHERE ROWID=?;", (deadline[0],)) # Удаляем истёкший дедлайн
database.commit()
elif deadline[3] == hour_after_time:
message = f"{deadline[2]}: остался 1 час до дедлайна ({deadline[3]})"
if deadline[4]:
message = "@all\n" + message # Если дедлайн установлен для беседы, упоминаем всех её участников
await send(deadline[1], message)
elif deadline[3] == day_after_time:
message = f"{deadline[2]}: остался 1 день до дедлайна ({deadline[3]})"
if deadline[4]:
message = "@all\n" + message
await send(deadline[1], message)
await asyncio.sleep(
60 - datetime.today().second) # До конца минуты новых дедлайнов точно не будет, можно не проверять
async def send(to_id: int, message: str):
"""Отправляет сообщение в чат. Принимает ID чата (int) и текст сообщения (str)."""
try:
await bot.api.messages.send(peer_id=to_id, random_id=randint(-2147483648, 2147483647), message=message,
disable_mentions=False)
except VKAPIError:
return
bot = Bot(token)
bot.labeler.vbml_ignore_case = True # Делает обработчики сообщений регистронезависимыми
bot.labeler.load(BotLabeler())
database = sqlite3.connect("deadlines.db")
database.create_collation("NOCASE", ignore_case_collation) # Правило регистронезависимого сравнения для SQLite
cur = database.cursor()
cur.execute("""CREATE TABLE IF NOT EXISTS Deadlines(
chat INT,
deadline TEXT,
time TEXT,
isGroup INT
);""")
database.commit()
if __name__ == "__main__":
asyncio.run(main())