0.0.10
This commit is contained in:
parent
351b53f007
commit
318b24243a
3
.gitignore
vendored
3
.gitignore
vendored
|
@ -3,4 +3,5 @@ __pycache__
|
||||||
.DS_Store
|
.DS_Store
|
||||||
.vercel
|
.vercel
|
||||||
api/payload.json
|
api/payload.json
|
||||||
update.json
|
update.json
|
||||||
|
.vscode
|
15
CHANGELOG.md
15
CHANGELOG.md
|
@ -1,3 +1,18 @@
|
||||||
|
## [0.0.10]
|
||||||
|
|
||||||
|
- добавлено фото к заявке пользователя, если есть
|
||||||
|
- изменена надпись на русском
|
||||||
|
- исправлены ошибки
|
||||||
|
- добавлена сервисная команда для показа потерянных заявок
|
||||||
|
|
||||||
|
## [0.0.9]
|
||||||
|
|
||||||
|
- исправление логики show_request_msg
|
||||||
|
- логика перепроверки на старте
|
||||||
|
- двуязычный интерфейс без переменных среды
|
||||||
|
- kick для тех, от кого отказались поручители
|
||||||
|
- bugfix: нестандартные символы в имени
|
||||||
|
|
||||||
## [0.0.8]
|
## [0.0.8]
|
||||||
|
|
||||||
- генерация древовидного графа, с опорой на одного участника
|
- генерация древовидного графа, с опорой на одного участника
|
||||||
|
|
|
@ -11,6 +11,4 @@
|
||||||
4. Настроить переменные среды:
|
4. Настроить переменные среды:
|
||||||
|
|
||||||
- BOT_TOKEN - токен бота созданный с помощью @BotFather
|
- BOT_TOKEN - токен бота созданный с помощью @BotFather
|
||||||
- NEWCOMER_MSG - текст сообщения о новом пользователе, за которым следует его имя
|
|
||||||
- BUTTON_VOUCH - текст кнопки "поручиться"
|
|
||||||
- FEEDBACK_CHAT_ID - айди чата для обратной связи
|
- FEEDBACK_CHAT_ID - айди чата для обратной связи
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
from sanic import Sanic
|
from sanic import Sanic
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
|
|
||||||
from tgbot.config import WEBHOOK, FEEDBACK_CHAT_ID, BUTTON_VOUCH
|
from tgbot.config import WEBHOOK, FEEDBACK_CHAT_ID
|
||||||
|
|
||||||
from tgbot.handlers.handle_feedback import handle_feedback, handle_answer
|
from tgbot.handlers.handle_feedback import handle_feedback, handle_answer
|
||||||
from tgbot.handlers.handle_members_change import handle_join, handle_left
|
from tgbot.handlers.handle_members_change import handle_join, handle_left
|
||||||
|
@ -9,9 +9,10 @@ from tgbot.handlers.handle_join_request import handle_join_request
|
||||||
from tgbot.handlers.handle_default import handle_default
|
from tgbot.handlers.handle_default import handle_default
|
||||||
from tgbot.handlers.command_my import handle_command_my
|
from tgbot.handlers.command_my import handle_command_my
|
||||||
from tgbot.handlers.command_graph import handle_command_graph
|
from tgbot.handlers.command_graph import handle_command_graph
|
||||||
|
from tgbot.handlers.command_ask import handle_command_ask
|
||||||
from tgbot.handlers.callback_vouch import handle_button
|
from tgbot.handlers.callback_vouch import handle_button
|
||||||
from tgbot.handlers.callback_unlink import handle_unlink
|
from tgbot.handlers.callback_unlink import handle_unlink
|
||||||
|
from tgbot.handlers.handle_startup import handle_startup
|
||||||
from tgbot.api import register_webhook, send_message
|
from tgbot.api import register_webhook, send_message
|
||||||
|
|
||||||
|
|
||||||
|
@ -28,6 +29,7 @@ async def register(req):
|
||||||
app.config.REGISTERED = True
|
app.config.REGISTERED = True
|
||||||
print(r)
|
print(r)
|
||||||
res = 'ok'
|
res = 'ok'
|
||||||
|
handle_startup()
|
||||||
return text(res)
|
return text(res)
|
||||||
|
|
||||||
|
|
||||||
|
@ -43,7 +45,8 @@ async def handle(req):
|
||||||
if msg:
|
if msg:
|
||||||
if 'text' in msg:
|
if 'text' in msg:
|
||||||
if msg['chat']['id'] == msg['from']['id']:
|
if msg['chat']['id'] == msg['from']['id']:
|
||||||
if msg['text'] == '/my':
|
print('private chat message')
|
||||||
|
if msg['text'].startswith('/my'):
|
||||||
handle_command_my(msg)
|
handle_command_my(msg)
|
||||||
else:
|
else:
|
||||||
handle_feedback(msg)
|
handle_feedback(msg)
|
||||||
|
@ -52,6 +55,8 @@ async def handle(req):
|
||||||
handle_answer(msg)
|
handle_answer(msg)
|
||||||
elif msg['text'] == '/graph':
|
elif msg['text'] == '/graph':
|
||||||
await handle_command_graph(msg)
|
await handle_command_graph(msg)
|
||||||
|
elif msg['text'].startswith('/ask'):
|
||||||
|
handle_command_ask(msg)
|
||||||
else:
|
else:
|
||||||
handle_default(msg)
|
handle_default(msg)
|
||||||
elif 'new_chat_member' in msg:
|
elif 'new_chat_member' in msg:
|
||||||
|
@ -64,7 +69,7 @@ async def handle(req):
|
||||||
# кнопки
|
# кнопки
|
||||||
elif 'callback_query' in update:
|
elif 'callback_query' in update:
|
||||||
data = update['callback_query']['data']
|
data = update['callback_query']['data']
|
||||||
if data.startswith(BUTTON_VOUCH):
|
if data.startswith('vouch'):
|
||||||
handle_button(update['callback_query'])
|
handle_button(update['callback_query'])
|
||||||
elif data.startswith('unlink'):
|
elif data.startswith('unlink'):
|
||||||
handle_unlink(update['callback_query'])
|
handle_unlink(update['callback_query'])
|
||||||
|
@ -81,6 +86,6 @@ async def handle(req):
|
||||||
|
|
||||||
except Exception:
|
except Exception:
|
||||||
import traceback
|
import traceback
|
||||||
r = send_message(FEEDBACK_CHAT_ID, f'<pre>{traceback.format_exc()}</pre>')
|
r = send_message(FEEDBACK_CHAT_ID, f'<pre>\n{traceback.format_exc()}\n</pre>')
|
||||||
traceback.print_exc()
|
traceback.print_exc()
|
||||||
return text('ok')
|
return text('ok')
|
||||||
|
|
54
tgbot/api.py
54
tgbot/api.py
|
@ -29,9 +29,21 @@ def send_message(cid: str, body, reply_to=None, reply_markup=None):
|
||||||
reply_markup = json.dumps(reply_markup)
|
reply_markup = json.dumps(reply_markup)
|
||||||
reply_markup = requests.utils.quote(reply_markup)
|
reply_markup = requests.utils.quote(reply_markup)
|
||||||
url += f'&reply_markup={reply_markup}'
|
url += f'&reply_markup={reply_markup}'
|
||||||
url += f'&parse_mode=HTML'
|
url += f'&parse_mode=html'
|
||||||
|
r = requests.post(url)
|
||||||
|
return r.json()
|
||||||
|
|
||||||
|
# https://core.telegram.org/bots/api#sendphoto
|
||||||
|
def send_photo(cid: str, file_id: str, caption="", reply_to=None, reply_markup=None):
|
||||||
|
url = apiBase + f"sendPhoto?chat_id={cid}&photo={file_id}"
|
||||||
|
if reply_to:
|
||||||
|
url += f'&reply_to_message_id={reply_to}'
|
||||||
|
if reply_markup:
|
||||||
|
reply_markup = json.dumps(reply_markup)
|
||||||
|
reply_markup = requests.utils.quote(reply_markup)
|
||||||
|
url += f'&reply_markup={reply_markup}'
|
||||||
|
url += f'&parse_mode=html'
|
||||||
r = requests.post(url)
|
r = requests.post(url)
|
||||||
print(f'{url}')
|
|
||||||
return r.json()
|
return r.json()
|
||||||
|
|
||||||
|
|
||||||
|
@ -69,7 +81,8 @@ def mute_member(chat_id, member_id):
|
||||||
chat_permissions = json.dumps({ "can_send_messages": False })
|
chat_permissions = json.dumps({ "can_send_messages": False })
|
||||||
chat_permissions = requests.utils.quote(chat_permissions)
|
chat_permissions = requests.utils.quote(chat_permissions)
|
||||||
url = apiBase + f'restrictChatMember?chat_id={chat_id}' + \
|
url = apiBase + f'restrictChatMember?chat_id={chat_id}' + \
|
||||||
f'&user_id={member_id}&permissions={chat_permissions}'
|
f'&user_id={member_id}&permissions={chat_permissions}' + \
|
||||||
|
f'&use_independent_chat_permissions=1'
|
||||||
r = requests.post(url)
|
r = requests.post(url)
|
||||||
return r.json()
|
return r.json()
|
||||||
|
|
||||||
|
@ -79,7 +92,8 @@ def unmute_member(chat_id, member_id):
|
||||||
chat_permissions = json.dumps({ "can_send_messages": True })
|
chat_permissions = json.dumps({ "can_send_messages": True })
|
||||||
chat_permissions = requests.utils.quote(chat_permissions)
|
chat_permissions = requests.utils.quote(chat_permissions)
|
||||||
url = apiBase + f'restrictChatMember?chat_id={chat_id}' + \
|
url = apiBase + f'restrictChatMember?chat_id={chat_id}' + \
|
||||||
f'&user_id={member_id}&permissions={chat_permissions}'
|
f'&user_id={member_id}&permissions={chat_permissions}' + \
|
||||||
|
f'&use_independent_chat_permissions=1'
|
||||||
r = requests.post(url)
|
r = requests.post(url)
|
||||||
return r.json()
|
return r.json()
|
||||||
|
|
||||||
|
@ -113,29 +127,6 @@ async def send_document(chat_id, data='', filename='chart.svg'):
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# https://core.telegram.org/bots/api#sendphoto
|
|
||||||
async def send_photo(chat_id, img_data, filename='chart.png'):
|
|
||||||
url = apiBase + f"sendPhoto"
|
|
||||||
params = {"chat_id": chat_id }
|
|
||||||
filedata = aiohttp.FormData()
|
|
||||||
filedata.add_field('photo', img_data, filename=filename)
|
|
||||||
|
|
||||||
async with aiohttp.ClientSession() as session:
|
|
||||||
async with session.post(url, params=params, data=filedata) as response:
|
|
||||||
if response.status != 200:
|
|
||||||
error_text = await response.text()
|
|
||||||
print(f"Error sending photo: {response.status} - {error_text}")
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
try:
|
|
||||||
return await response.json()
|
|
||||||
except ValueError as e:
|
|
||||||
print(f"Error decoding JSON: {e}")
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
# https://core.telegram.org/bots/api#getchatadministrators
|
# https://core.telegram.org/bots/api#getchatadministrators
|
||||||
def get_chat_administrators(chat_id):
|
def get_chat_administrators(chat_id):
|
||||||
url = apiBase + f"getChatAdministrators?chat_id={chat_id}"
|
url = apiBase + f"getChatAdministrators?chat_id={chat_id}"
|
||||||
|
@ -143,8 +134,15 @@ def get_chat_administrators(chat_id):
|
||||||
return r.json()
|
return r.json()
|
||||||
|
|
||||||
|
|
||||||
# # https://core.telegram.org/bots/api#getchatmember
|
# https://core.telegram.org/bots/api#getchatmember
|
||||||
def get_member(chat_id, member_id):
|
def get_member(chat_id, member_id):
|
||||||
url = apiBase + f"getChatMember?chat_id={chat_id}&user_id={member_id}"
|
url = apiBase + f"getChatMember?chat_id={chat_id}&user_id={member_id}"
|
||||||
r = requests.get(url)
|
r = requests.get(url)
|
||||||
|
return r.json()
|
||||||
|
|
||||||
|
|
||||||
|
# https://core.telegram.org/bots/api#getuserprofilephotos
|
||||||
|
def get_userphotos(user_id):
|
||||||
|
url = apiBase + f"getUserProfilePhotos?user_id={user_id}"
|
||||||
|
r = requests.get(url)
|
||||||
return r.json()
|
return r.json()
|
|
@ -3,6 +3,4 @@ import os
|
||||||
|
|
||||||
WEBHOOK = os.environ.get('VERCEL_URL') or 'http://localhost:8000'
|
WEBHOOK = os.environ.get('VERCEL_URL') or 'http://localhost:8000'
|
||||||
REDIS_URL = os.environ.get('REDIS_URL') or 'redis://localhost:6379'
|
REDIS_URL = os.environ.get('REDIS_URL') or 'redis://localhost:6379'
|
||||||
NEWCOMER_MSG = os.environ.get('NEWCOMER_MSG') or "There is a newcomer, press the button if you are connected"
|
|
||||||
BUTTON_VOUCH = os.environ.get('BUTTON_VOUCH') or 'My connection!'
|
|
||||||
FEEDBACK_CHAT_ID = os.environ.get('FEEDBACK_CHAT_ID').replace("-", "-100")
|
FEEDBACK_CHAT_ID = os.environ.get('FEEDBACK_CHAT_ID').replace("-", "-100")
|
|
@ -1,4 +1,5 @@
|
||||||
from tgbot.api import send_message, delete_message
|
from tgbot.api import send_message, delete_message
|
||||||
|
from tgbot.handlers.command_my import handle_command_my
|
||||||
from tgbot.storage import Profile
|
from tgbot.storage import Profile
|
||||||
|
|
||||||
# remove link of callback sender
|
# remove link of callback sender
|
||||||
|
@ -7,17 +8,28 @@ def handle_unlink(callback_query):
|
||||||
print('handle unlink button pressed, private chat only')
|
print('handle unlink button pressed, private chat only')
|
||||||
|
|
||||||
from_id = str(callback_query['from']['id'])
|
from_id = str(callback_query['from']['id'])
|
||||||
reply_msg_id = callback_query['message']['message_id']
|
linked_id = str(callback_query['data'].replace('unlink', ''))
|
||||||
|
|
||||||
|
# удаляем связь с потомком
|
||||||
actor = Profile.get(from_id, callback_query)
|
actor = Profile.get(from_id, callback_query)
|
||||||
actor['parents'].remove(from_id)
|
actor['children'].remove(linked_id)
|
||||||
|
Profile.save(actor)
|
||||||
|
|
||||||
|
# удаляем связь с предком
|
||||||
|
linked = Profile.get(linked_id)
|
||||||
|
linked['parents'].remove(from_id)
|
||||||
|
Profile.save(linked)
|
||||||
|
|
||||||
# удаляем старое сообщение с кнопками
|
# удаляем старое сообщение с кнопками
|
||||||
|
reply_msg_id = callback_query['message']['message_id']
|
||||||
r = delete_message(from_id, reply_msg_id)
|
r = delete_message(from_id, reply_msg_id)
|
||||||
print(r)
|
print(r)
|
||||||
|
|
||||||
|
# если ещё есть связи - посылаем новое сообщение
|
||||||
|
if len(actor['children']) > 0:
|
||||||
|
handle_command_my(callback_query)
|
||||||
|
|
||||||
# если ещё есть связи - посылаем новое сообщение
|
# если больше никто не поручился - мьютим
|
||||||
if len(actor['parents']) > 0:
|
if len(linked['parents']) == 0:
|
||||||
body = construct_unlink_buttons(actor)
|
for chat_id in linked['chats']:
|
||||||
r = send_message(from_id, body)
|
mute_member(chat_id, linked_id)
|
||||||
print(r)
|
|
|
@ -1,6 +1,5 @@
|
||||||
from tgbot.api import send_message, forward_message, delete_message
|
from tgbot.api import send_message, forward_message, delete_message, approve_chat_join_request, unmute_member
|
||||||
from tgbot.storage import Profile
|
from tgbot.storage import Profile
|
||||||
from tgbot.config import BUTTON_VOUCH
|
|
||||||
|
|
||||||
|
|
||||||
def handle_button(callback_query):
|
def handle_button(callback_query):
|
||||||
|
@ -9,31 +8,37 @@ def handle_button(callback_query):
|
||||||
actor = Profile.get(actor_id, callback_query)
|
actor = Profile.get(actor_id, callback_query)
|
||||||
|
|
||||||
callback_data = callback_query['data']
|
callback_data = callback_query['data']
|
||||||
if callback_data.startswith(BUTTON_VOUCH):
|
if callback_data.startswith('vouch'):
|
||||||
print(f'vouch button pressed by {actor_id}')
|
print(f'button pressed by {actor_id}')
|
||||||
|
|
||||||
|
newcomer_id = callback_data[5:]
|
||||||
|
print(f'button pressed for {newcomer_id}')
|
||||||
|
|
||||||
newcomer_id = callback_data[len(BUTTON_VOUCH):]
|
|
||||||
newcomer = Profile.get(newcomer_id)
|
newcomer = Profile.get(newcomer_id)
|
||||||
|
print(f'newcomer profile {newcomer}')
|
||||||
if newcomer_id == actor_id:
|
if newcomer_id == actor_id:
|
||||||
# нажал сам, не реагируем, прописываем данные
|
# нажал сам, не реагируем, прописываем данные
|
||||||
newcomer = Profile.get(newcomer_id, callback_query)
|
newcomer = Profile.get(newcomer_id, callback_query)
|
||||||
elif newcomer_id not in actor['children'] and \
|
else:
|
||||||
actor_id not in newcomer['parents']:
|
# нажал кто-то другой
|
||||||
newcomer['parents'].append(actor_id)
|
|
||||||
actor['children'].append(newcomer_id)
|
|
||||||
Profile.save(newcomer)
|
|
||||||
Profile.save(actor)
|
|
||||||
try:
|
|
||||||
chat_id = str(callback_query['message']['chat']['id'])
|
|
||||||
|
|
||||||
print('unmute newcomer')
|
if actor_id not in newcomer['parents']:
|
||||||
|
print(f'save parent for {newcomer_id}')
|
||||||
|
newcomer['parents'].append(actor_id)
|
||||||
|
Profile.save(newcomer)
|
||||||
|
|
||||||
|
if newcomer_id not in actor['children']:
|
||||||
|
print(f'save child for {actor_id}')
|
||||||
|
actor['children'].append(newcomer_id)
|
||||||
|
Profile.save(actor)
|
||||||
|
|
||||||
|
chat_id = str(callback_query['message']['chat']['id'])
|
||||||
|
|
||||||
|
print('accept join request')
|
||||||
|
r = approve_chat_join_request(chat_id, newcomer_id)
|
||||||
|
print(r)
|
||||||
|
|
||||||
|
if not r.get('ok'):
|
||||||
|
print('try to unmute newcomer')
|
||||||
r = unmute_member(chat_id, newcomer_id)
|
r = unmute_member(chat_id, newcomer_id)
|
||||||
print(r)
|
print(r)
|
||||||
|
|
||||||
print('accept join request')
|
|
||||||
r = approve_chat_join_request(chat_id, newcomer_id)
|
|
||||||
print(r)
|
|
||||||
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
|
|
14
tgbot/handlers/command_ask.py
Normal file
14
tgbot/handlers/command_ask.py
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
from tgbot.storage import Profile
|
||||||
|
from tgbot.handlers.send_button import show_request_msg
|
||||||
|
from tgbot.api import get_member
|
||||||
|
|
||||||
|
def handle_command_ask(msg):
|
||||||
|
print(f'handling request resend')
|
||||||
|
cmd, chat_id, member_id = msg['text'].split(' ')
|
||||||
|
chat_id = chat_id.replace('-', '-100')
|
||||||
|
r = get_member(chat_id, member_id)
|
||||||
|
print(r)
|
||||||
|
m = {}
|
||||||
|
m['from'] = r['result']['user']
|
||||||
|
m['chat'] = { 'id': chat_id }
|
||||||
|
show_request_msg(m)
|
|
@ -1,29 +1,11 @@
|
||||||
from tgbot.utils.graph import generate_chart
|
from tgbot.utils.graph import generate_chart
|
||||||
from tgbot.api import send_document
|
from tgbot.api import send_document
|
||||||
from tgbot.storage import storage
|
from tgbot.storage import storage, scan
|
||||||
import json
|
import json
|
||||||
|
|
||||||
|
|
||||||
async def handle_command_graph(msg):
|
async def handle_command_graph(msg):
|
||||||
cursor = 0
|
usr_ids, members = scan(match='usr-*', count=100)
|
||||||
keys = []
|
|
||||||
r = storage
|
|
||||||
while True:
|
|
||||||
# Scan for keys starting with 'usr-*' in batches of 100
|
|
||||||
cursor, batch_keys = r.scan(cursor=cursor, match='usr-*', count=100)
|
|
||||||
keys += batch_keys
|
|
||||||
# If the cursor is 0, then we've reached the end of the keys
|
|
||||||
if cursor == 0:
|
|
||||||
break
|
|
||||||
# Get the values of all the keys
|
|
||||||
values = r.mget(keys)
|
|
||||||
# Parse the JSON data from each value
|
|
||||||
members = []
|
|
||||||
for value in values:
|
|
||||||
value_str = value.decode('utf-8')
|
|
||||||
member = json.loads(value_str)
|
|
||||||
members.append(member)
|
|
||||||
print(f'found {len(members)} members')
|
|
||||||
data = generate_chart(members)
|
data = generate_chart(members)
|
||||||
if data:
|
if data:
|
||||||
r = await send_document(msg['chat']['id'], data, 'chart.svg')
|
r = await send_document(msg['chat']['id'], data, 'chart.svg')
|
||||||
|
|
|
@ -1,33 +1,26 @@
|
||||||
from tgbot.storage import Profile
|
from tgbot.storage import Profile
|
||||||
from tgbot.api import get_member, send_message
|
from tgbot.api import get_member, send_message
|
||||||
|
from tgbot.utils.mention import userdata_extract
|
||||||
|
|
||||||
|
|
||||||
def construct_unlink_buttons(actor):
|
def construct_unlink_buttons(actor):
|
||||||
buttons = []
|
buttons = []
|
||||||
for vouch in actor['children']:
|
for vouch in actor['children']:
|
||||||
vouch_added = False
|
|
||||||
for chat_id in actor['chats']:
|
for chat_id in actor['chats']:
|
||||||
if not vouch_added:
|
r = get_member(chat_id, vouch)
|
||||||
r = get_member(chat_id, vouch)
|
member = r['result']['user']
|
||||||
member = r.get('result')
|
uid, identity, username = userdata_extract(member)
|
||||||
if member:
|
buttons.append({
|
||||||
try:
|
'text': f'{identity} {username}',
|
||||||
buttons.append({
|
'callback_data': 'unlink' + vouch
|
||||||
'text': mention(r['result']),
|
})
|
||||||
'callback_data': 'unlink' + vouch
|
|
||||||
})
|
|
||||||
vouch_added = True
|
|
||||||
except:
|
|
||||||
print('member result error')
|
|
||||||
print(member)
|
|
||||||
else:
|
|
||||||
print(r)
|
|
||||||
return { "inline_keyboard": [ buttons, ] }
|
return { "inline_keyboard": [ buttons, ] }
|
||||||
|
|
||||||
|
|
||||||
def handle_command_my(msg):
|
def handle_command_my(msg):
|
||||||
print(f'handle my command')
|
print(f'handle my command')
|
||||||
from_id = str(msg['from']['id'])
|
from_id = str(msg['from']['id'])
|
||||||
sender = Profile.get(from_id, msg)
|
sender = Profile.get(from_id, msg)
|
||||||
|
|
||||||
# генерируем кнопки для всех, за кого поручились
|
# генерируем кнопки для всех, за кого поручились
|
||||||
reply_markup = construct_unlink_buttons(sender)
|
reply_markup = construct_unlink_buttons(sender)
|
||||||
|
|
||||||
|
|
|
@ -17,15 +17,8 @@ def handle_default(msg):
|
||||||
r = delete_message(chat_id, msg['message_id'])
|
r = delete_message(chat_id, msg['message_id'])
|
||||||
print(r)
|
print(r)
|
||||||
|
|
||||||
# удалить предыдушее сообщение с кнопкой в этом чате
|
|
||||||
prev_msg_id = storage.get(f'btn-{chat_id}-{from_id}')
|
|
||||||
if prev_msg_id:
|
|
||||||
r = delete_message(chat_id, prev_msg_id)
|
|
||||||
print(r)
|
|
||||||
|
|
||||||
# показать новое сообщение с кнопкой
|
# показать новое сообщение с кнопкой
|
||||||
btn_msg_id = show_request_msg(msg)
|
show_request_msg(msg)
|
||||||
storage.set(f'btn-{chat_id}-{from_id}', btn_msg_id)
|
|
||||||
else:
|
else:
|
||||||
# любое другое сообщение
|
# любое другое сообщение
|
||||||
if len(sender['parents']) == 0:
|
if len(sender['parents']) == 0:
|
||||||
|
@ -38,12 +31,12 @@ def handle_default(msg):
|
||||||
if admin['status'] == 'creator':
|
if admin['status'] == 'creator':
|
||||||
owner_id = admin['user']['id']
|
owner_id = admin['user']['id']
|
||||||
break
|
break
|
||||||
|
if owner_id:
|
||||||
sender['parents'].append(owner_id)
|
sender['parents'].append(owner_id)
|
||||||
# обновляем профиль владельца
|
# обновляем профиль владельца
|
||||||
owner = Profile.get(owner_id)
|
owner = Profile.get(owner_id)
|
||||||
owner['children'].append(from_id)
|
owner['children'].append(from_id)
|
||||||
Profile.save(owner)
|
Profile.save(owner)
|
||||||
|
|
||||||
# сохранить профиль отправителя
|
# сохранить профиль отправителя
|
||||||
Profile.save(sender)
|
Profile.save(sender)
|
|
@ -1,6 +1,8 @@
|
||||||
import json
|
import json
|
||||||
|
|
||||||
from tgbot.api import send_message, forward_message, delete_message
|
from tgbot.api import send_message, forward_message, delete_message
|
||||||
|
from tgbot.handlers.send_button import show_request_msg
|
||||||
|
from tgbot.utils.mention import userdata_extract
|
||||||
from tgbot.storage import storage, Profile
|
from tgbot.storage import storage, Profile
|
||||||
from tgbot.config import FEEDBACK_CHAT_ID
|
from tgbot.config import FEEDBACK_CHAT_ID
|
||||||
|
|
||||||
|
|
|
@ -11,13 +11,7 @@ def handle_join_request(msg):
|
||||||
|
|
||||||
if len(actor['parents']) == 0:
|
if len(actor['parents']) == 0:
|
||||||
# показываем сообщение с кнопкой "поручиться"
|
# показываем сообщение с кнопкой "поручиться"
|
||||||
btn_msg_id = show_request_msg(msg)
|
show_request_msg(msg)
|
||||||
# удаляем предыдущее сообщение с кнопкой в этом чате
|
|
||||||
prev_msg_id = storage.get(f'btn-{chat_id}-{from_id}')
|
|
||||||
if prev_msg_id:
|
|
||||||
r = delete_message(chat_id, prev_msg_id)
|
|
||||||
print(r)
|
|
||||||
storage.set(f'btn-{chat_id}-{from_id}', btn_msg_id)
|
|
||||||
else:
|
else:
|
||||||
# за пользователя поручились ранее
|
# за пользователя поручились ранее
|
||||||
r = approve_chat_join_request(chat_id, from_id)
|
r = approve_chat_join_request(chat_id, from_id)
|
||||||
|
|
|
@ -12,8 +12,7 @@ def handle_join(msg):
|
||||||
if from_id == newcomer_id:
|
if from_id == newcomer_id:
|
||||||
if len(actor['parents']) == 0:
|
if len(actor['parents']) == 0:
|
||||||
# показываем сообщение с кнопкой "поручиться"
|
# показываем сообщение с кнопкой "поручиться"
|
||||||
btn_msg_id = show_request_msg(msg)
|
show_request_msg(msg)
|
||||||
storage.set(f'btn-{chat_id}-{from_id}', btn_msg_id)
|
|
||||||
|
|
||||||
# до одобрения - мьют
|
# до одобрения - мьют
|
||||||
r = mute_member(chat_id, newcomer_id)
|
r = mute_member(chat_id, newcomer_id)
|
||||||
|
@ -43,9 +42,9 @@ def handle_left(msg):
|
||||||
chat_id = msg['chat']['id']
|
chat_id = msg['chat']['id']
|
||||||
|
|
||||||
# удаление сообщения с кнопкой в этом чате
|
# удаление сообщения с кнопкой в этом чате
|
||||||
prev_msg_id = storage.get(f'btn-{chat_id}-{member_id}')
|
prev_msg = storage.get(f'btn-{chat_id}-{member_id}')
|
||||||
if prev_msg_id:
|
if prev_msg_id:
|
||||||
r = delete_message(chat_id, prev_msg_id)
|
r = delete_message(chat_id, prev_msg['id'])
|
||||||
print(r)
|
print(r)
|
||||||
storage.remove(f'btn-{chat_id}-{member_id}')
|
storage.remove(f'btn-{chat_id}-{member_id}')
|
||||||
|
|
||||||
|
|
18
tgbot/handlers/handle_startup.py
Normal file
18
tgbot/handlers/handle_startup.py
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
from tgbot.storage import scan, Profile
|
||||||
|
|
||||||
|
# устанавливает соответствие данных
|
||||||
|
def handle_startup():
|
||||||
|
btn_ids, btns = scan(match='btn-*', count=100)
|
||||||
|
for btnid in btn_ids:
|
||||||
|
# для каждой ранее созданной кнопки
|
||||||
|
try:
|
||||||
|
btnid_str = btnid.decode("utf-8")
|
||||||
|
chat_id, member_id = btnid_str[3:].split('-')
|
||||||
|
|
||||||
|
newcomer = Profile.get(member_id)
|
||||||
|
if len(newcomer.get('parents', [])) > 0:
|
||||||
|
# принять заявку если её нажимали
|
||||||
|
r = approve_chat_join_request(chat_id, member_id)
|
||||||
|
print(r)
|
||||||
|
except:
|
||||||
|
print(f'error {btnid}')
|
|
@ -1,25 +1,49 @@
|
||||||
from tgbot.api import send_message
|
from tgbot.api import send_message, send_photo, get_userphotos
|
||||||
from tgbot.config import BUTTON_VOUCH, NEWCOMER_MSG
|
|
||||||
from tgbot.utils.mention import mention
|
from tgbot.utils.mention import mention
|
||||||
|
from tgbot.storage import storage
|
||||||
|
|
||||||
def show_request_msg(msg):
|
def show_request_msg(msg):
|
||||||
|
chat_id = str(msg['chat']['id'])
|
||||||
|
from_id = str(msg['from']['id'])
|
||||||
|
lang = msg['from'].get('language_code', 'ru')
|
||||||
reply_markup = {
|
reply_markup = {
|
||||||
"inline_keyboard": [
|
"inline_keyboard": [
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
"text": BUTTON_VOUCH,
|
"text": 'Моё одобрение' if lang == 'ru' else 'My connection',
|
||||||
"callback_data": BUTTON_VOUCH + str(msg['from']['id'])
|
"callback_data": 'vouch' + from_id
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
newcomer_message = "Нажмите, чтобы одобрить заявку " if lang == 'ru' \
|
||||||
r = send_message(
|
else "There is a newcomer, press the button if you are connected with "
|
||||||
msg['chat']['id'],
|
r = get_userphotos(user_id=from_id)
|
||||||
NEWCOMER_MSG + mention(msg['from']),
|
print(r)
|
||||||
reply_to=msg.get('message_id', ''),
|
if r['ok'] and r['result']['total_count'] > 0:
|
||||||
reply_markup=reply_markup
|
file_id = r['result']['photos'][0][0]['file_id']
|
||||||
)
|
r = send_photo(
|
||||||
btn_msg_id = r['result']['message_id']
|
chat_id,
|
||||||
print(f'request message id: {btn_msg_id}')
|
file_id,
|
||||||
return btn_msg_id
|
caption=newcomer_message + mention(msg['from']),
|
||||||
|
reply_to=msg.get('message_id', ''),
|
||||||
|
reply_markup=reply_markup
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
r = send_photo(
|
||||||
|
chat_id,
|
||||||
|
newcomer_message + mention(msg['from']),
|
||||||
|
reply_to=msg.get('message_id', ''),
|
||||||
|
reply_markup=reply_markup
|
||||||
|
)
|
||||||
|
print(r)
|
||||||
|
if 'message_id' in r:
|
||||||
|
# удаляем предыдущее сообщение с кнопкой в этом чате
|
||||||
|
prevbtn = storage.get(f'btn-{chat_id}-{from_id}')
|
||||||
|
if prevbtn:
|
||||||
|
r = delete_message(chat_id, prevbtn)
|
||||||
|
print(r)
|
||||||
|
# создаём новое
|
||||||
|
newbtn = r['message_id']
|
||||||
|
print(f'button message id: {newbtn}')
|
||||||
|
storage.set(f'btn-{chat_id}-{from_id}', newbtn)
|
||||||
|
|
|
@ -1,9 +1,35 @@
|
||||||
import redis
|
import redis
|
||||||
from tgbot.storage.profile import Profile as ProfileObj
|
from tgbot.storage.profile import Profile as ProfileObj
|
||||||
from tgbot.config import REDIS_URL
|
from tgbot.config import REDIS_URL
|
||||||
|
import json
|
||||||
|
|
||||||
|
|
||||||
# сохраняет сессии, айди кнопок в чатах для удаления и пересылаемые сообщения между перезагрузками
|
# сохраняет сессии, айди кнопок в чатах для удаления и пересылаемые сообщения между перезагрузками
|
||||||
storage = redis.from_url(REDIS_URL)
|
storage = redis.from_url(REDIS_URL)
|
||||||
|
|
||||||
# хранение необходимой информации о пользователях
|
# хранение необходимой информации о пользователях
|
||||||
Profile = ProfileObj(storage)
|
Profile = ProfileObj(storage)
|
||||||
|
|
||||||
|
# достаёт из хранилища jsonы по маске и количеству
|
||||||
|
def scan(match='usr-*', count=100):
|
||||||
|
cursor = 0
|
||||||
|
keys = []
|
||||||
|
r = storage
|
||||||
|
while True:
|
||||||
|
# Scan for keys starting with <match> in batches of <count>
|
||||||
|
cursor, batch_keys = r.scan(cursor=cursor, match=match, count=count)
|
||||||
|
keys += batch_keys
|
||||||
|
# If the cursor is 0, then we've reached the end of the keys
|
||||||
|
if cursor == 0:
|
||||||
|
break
|
||||||
|
# Get the values of all the keys
|
||||||
|
values = r.mget(keys)
|
||||||
|
# Parse the JSON data from each value
|
||||||
|
items = []
|
||||||
|
for value in values:
|
||||||
|
value_str = value.decode('utf-8')
|
||||||
|
i = json.loads(value_str)
|
||||||
|
items.append(i)
|
||||||
|
print(f'scan found {len(items)} items')
|
||||||
|
|
||||||
|
return keys, items
|
|
@ -14,15 +14,17 @@ class Profile:
|
||||||
"chats": []
|
"chats": []
|
||||||
}
|
}
|
||||||
|
|
||||||
if msg.get('from'):
|
if msg:
|
||||||
sender = msg.get('from')
|
|
||||||
s["mention"] = sender.get('username')
|
if 'from' in msg:
|
||||||
s["name"] = f"{sender['first_name']} {sender.get('last_name', '')}".strip()
|
sender = msg.get('from')
|
||||||
|
s["mention"] = sender.get('username')
|
||||||
|
s["name"] = f"{sender['first_name']} {sender.get('last_name', '')}".strip()
|
||||||
|
|
||||||
if msg.get('chat'):
|
if 'chat' in msg:
|
||||||
chat_id = str(msg['chat']['id'])
|
chat_id = str(msg['chat']['id'])
|
||||||
if chat_id not in s['chats']:
|
if chat_id not in s['chats']:
|
||||||
s["chats"].append(chat_id)
|
s["chats"].append(chat_id)
|
||||||
|
|
||||||
self.storage.set(f'usr-{member_id}', json.dumps(s))
|
self.storage.set(f'usr-{member_id}', json.dumps(s))
|
||||||
return s
|
return s
|
||||||
|
|
|
@ -1,9 +1,20 @@
|
||||||
|
def escape_username(username):
|
||||||
|
# Replace any non-ASCII and non-alphanumeric characters with underscores
|
||||||
|
return ''.join(c if c.isalnum() or c.isspace() else '-' for c in username)
|
||||||
|
|
||||||
# generates a mention from standard telegram web json 'from' field
|
# generates a mention from standard telegram web json 'from' field
|
||||||
# using HTML markup
|
# using HTML markup
|
||||||
def mention(user):
|
def mention(user):
|
||||||
|
uid, identity, username = userdata_extract(user)
|
||||||
|
identity = escape_username(identity)
|
||||||
|
return f'<a href="tg://user?id={uid}">{identity} {username}</a>'
|
||||||
|
|
||||||
|
|
||||||
|
def userdata_extract(user):
|
||||||
identity = f"{user['first_name']} {user.get('last_name', '')}".strip()
|
identity = f"{user['first_name']} {user.get('last_name', '')}".strip()
|
||||||
uid = user['id']
|
uid = user['id']
|
||||||
username = user.get('username', '')
|
username = user.get('username', '')
|
||||||
if username:
|
if username:
|
||||||
username = f'(@{username})'
|
username = f'(@{username})'
|
||||||
return f'<a href="tg://user?id={uid}"><b>{identity}</b></a>{username}'
|
return uid, identity, username
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user