From d785fb1622d262d26ce1a9dc42900cbe554553a9 Mon Sep 17 00:00:00 2001 From: Alexei Date: Wed, 13 Sep 2023 09:28:23 +0300 Subject: [PATCH 01/21] =?UTF-8?q?=D0=92=20=D1=80=D0=B8=D0=B4=D0=BC=D0=B8?= =?UTF-8?q?=20=D0=B4=D0=BE=D0=B1=D0=B0=D0=B2=D0=B8=D0=BB=D0=B8=D1=81=D1=8C?= =?UTF-8?q?=20=D0=B2=D0=B8=D0=B4=D0=B5=D0=BE=D0=B8=D0=BD=D1=81=D1=82=D1=80?= =?UTF-8?q?=D1=83=D0=BA=D1=86=D0=B8=D0=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.md | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/README.md b/README.md index 24d9c56..85b109c 100644 --- a/README.md +++ b/README.md @@ -8,6 +8,12 @@ Обзор возможностей пользователя - https://vk.com/video-210998646_456239042 +Проекты - https://vk.com/wall-210998646_139 + +Работа с заказами - https://vk.com/wall-210998646_137 + +Админ-панель - https://vk.com/wall-210998646_126 + ### Модульный Telegram-бот с возможностью редактирования прав доступа, как пользователям, так и группам пользователей Список модулей From 5dd44fe7f22014f75dd3350d1aa17afd505047ca Mon Sep 17 00:00:00 2001 From: Alexei Date: Sat, 30 Sep 2023 23:39:54 +0300 Subject: [PATCH 02/21] =?UTF-8?q?=D0=9C=D0=BE=D0=B4=D1=83=D0=BB=D1=8C=20?= =?UTF-8?q?=D0=B0=D0=B2=D1=82=D0=BE=D1=80=D0=B8=D0=B7=D0=B0=D1=86=D0=B8?= =?UTF-8?q?=D0=B8=20=D0=B2=20=D0=BF=D0=B5=D1=80=D0=B2=D0=BE=D0=BC=20=D0=BF?= =?UTF-8?q?=D1=80=D0=B8=D0=B1=D0=BB=D0=B8=D0=B6=D0=B5=D0=BD=D0=B8=D0=B8=20?= =?UTF-8?q?#9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 1 + bot_modules/authorize.py | 231 +++++++++++++++++++++++++++++++++++++++++++++++ bot_sys/bd_table.py | 8 ++ main.py | 7 +- 4 files changed, 246 insertions(+), 1 deletion(-) create mode 100644 bot_modules/authorize.py diff --git a/.gitignore b/.gitignore index e3f5dc5..a67820b 100644 --- a/.gitignore +++ b/.gitignore @@ -3,3 +3,4 @@ config_root_ids __pycache__ log.txt bot.db +tmp.py diff --git a/bot_modules/authorize.py b/bot_modules/authorize.py new file mode 100644 index 0000000..2831c0a --- /dev/null +++ b/bot_modules/authorize.py @@ -0,0 +1,231 @@ +# -*- coding: utf8 -*- +# Общественное достояние, 2023, Алексей Безбородов (Alexei Bezborodov) + +# Авторизация или регистрация в кооперативе или другой организации + +from bot_sys import bot_bd, keyboard, user_access, bd_table, bot_subscribes +from bot_modules import mod_table_operate, mod_simple_message + +from enum import Enum +from enum import auto +# --------------------------------------------------------- +# БД +module_name = 'authorize' + +table_name = module_name +user_id_field = 'userID' +user_name_field = 'userName' +user_family_name_field = 'userFamilyName' +user_middle_name_field = 'userMiddleName' +user_birthday_field = 'userBirthday' +user_address_field = 'userAddress' +user_contacts_field = 'userContacts' +user_confirm_field = 'userConfirm' +access_field = 'authorizeAccess' +create_datetime_field = 'authorizeCreateDateTime' + +user_id_table_field = bd_table.TableField(user_id_field, bd_table.TableFieldDestiny.USER_ID, bd_table.TableFieldType.INT) + +class ConfirmStatus(Enum): + YES = auto() + NO = auto() + +table = bd_table.Table(table_name, [ + user_id_table_field, + bd_table.TableField(user_name_field, bd_table.TableFieldDestiny.USER_NAME, bd_table.TableFieldType.STR), + bd_table.TableField(user_family_name_field, bd_table.TableFieldDestiny.USER_FAMILY_NAME, bd_table.TableFieldType.STR), + bd_table.TableField(user_middle_name_field, bd_table.TableFieldDestiny.USER_MIDDLE_NAME, bd_table.TableFieldType.STR), + bd_table.TableField(user_birthday_field, bd_table.TableFieldDestiny.USER_BIRTHDAY, bd_table.TableFieldType.STR), + bd_table.TableField(user_address_field, bd_table.TableFieldDestiny.USER_ADDRESS, bd_table.TableFieldType.STR), + bd_table.TableField(user_contacts_field, bd_table.TableFieldDestiny.USER_CONTACTS, bd_table.TableFieldType.STR), + bd_table.TableField(user_confirm_field, bd_table.TableFieldDestiny.USER_CONFIRM, bd_table.TableFieldType.ENUM, a_Enum = ConfirmStatus), + bd_table.TableField(access_field, bd_table.TableFieldDestiny.ACCESS, bd_table.TableFieldType.STR), + bd_table.TableField(create_datetime_field, bd_table.TableFieldDestiny.CREATE_DATE, bd_table.TableFieldType.STR), + ] + , + [ + [user_id_table_field], + ] + ) + +init_access = f'{user_access.user_access_group_new}=va' + +# --------------------------------------------------------- +# Сообщения и кнопки + +class ButtonNames(Enum): + LIST_AUTH_DOCS = auto() + +button_names = { + mod_simple_message.ButtonNames.START: "🔑 Авторизация", + mod_table_operate.ButtonNames.LIST: "≣ Список авторизаций пользователей", + mod_table_operate.ButtonNames.ADD: "📨 Заявка на вступление", + mod_table_operate.ButtonNames.EDIT: "🛠 Редактировать свои данные", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_NAME): "☐ Изменить имя пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_FAMILY_NAME): "☐ Изменить фамилию пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_MIDDLE_NAME): "☐ Изменить отчество пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_BIRTHDAY): "☐ Изменить дату рождения пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ADDRESS): "☐ Изменить адрес пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_CONTACTS): "☐ Изменить контакты пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_CONFIRM): "☐ Изменить подтверждение пользователя", +# mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.ACCESS): "✋ Доступ к авторизации пользователя", mod_table_operate.ButtonNames.DEL: "❌ Удалить авторизацию пользователя", +} + +messages = { + mod_simple_message.Messages.START: f''' +{button_names[mod_simple_message.ButtonNames.START]} + +''', + mod_table_operate.Messages.SELECT: ''' +Пожалуйста, выберите пользователя: +''', + mod_table_operate.Messages.ERROR_FIND: ''' +❌ Ошибка, пользователь не найден +''', + mod_table_operate.Messages.OPEN: f''' +Пользователь: #{user_id_field} + +Имя: #{user_name_field} +Фамилия: #{user_family_name_field} +Отчество: #{user_middle_name_field} +Дата рождения: #{user_birthday_field} +Адрес: #{user_address_field} +Контакты: #{user_contacts_field} +Подтверждение авторизации: #{user_confirm_field} + +Время создания: #{create_datetime_field} +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.USER_NAME): ''' +Авторизация. Шаг №1 + +Введите своё имя: +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.USER_FAMILY_NAME): ''' +Авторизация. Шаг №2 + +Введите свою фамилию: +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.USER_MIDDLE_NAME): ''' +Авторизация. Шаг №3 + +Введите своё отчество: +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.USER_BIRTHDAY): ''' +Авторизация. Шаг №4 + +Введите свою дату рождения в формате ДД.ММ.ГГГГ: +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.USER_ADDRESS): ''' +Авторизация. Шаг №5 + +Введите свой домашний адрес: +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.USER_CONTACTS): ''' +Авторизация. Шаг №6 + +Введите свои контакты: +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.USER_CONFIRM): ''' +Авторизация. Шаг №7 + +Имя: #{user_name_field} +Фамилия: #{user_family_name_field} +Отчество: #{user_middle_name_field} +Дата рождения: #{user_birthday_field} +Адрес: #{user_address_field} +Контакты: #{user_contacts_field} +Подтверждение авторизации: #{user_confirm_field} + +Подтвердите свои данные: +''', + + mod_table_operate.Messages.SUCCESS_CREATE: '''✅ Вы успешно зарегистрированы! Посмотрите документы и отправьте фотографии подписанных документов''', + mod_table_operate.Messages.START_EDIT: ''' +Пожалуйста, выберите действие: +''', + mod_table_operate.Messages.SELECT_TO_EDIT: ''' +Выберите пользователя, у которого вы хотите отредактировать данные. +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.USER_NAME): f''' +Текущее имя пользователя: +#{user_name_field} + +Введите новое имя пользователя: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.USER_FAMILY_NAME): f''' +Текущая фамилия пользователя: +#{user_family_name_field} + +Введите новую фамилию пользователя: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.USER_MIDDLE_NAME): f''' +Текущее отчество пользователя: +#{user_middle_name_field} + +Введите новое отчество пользователя: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.USER_BIRTHDAY): f''' +Текущая дата рождения пользователя: +#{user_birthday_field} + +Введите новую дату рождения пользователя: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.USER_ADDRESS): f''' +Текущий адрес пользователя: +#{user_address_field} + +Введите новый адрес пользователя: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.USER_CONTACTS): f''' +Текущие контакты пользователя: +#{user_contacts_field} + +Введите новые контакты пользователя: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.USER_CONFIRM): f''' +Текущее подтвержение пользователя: +#{user_confirm_field} + +Введите новое подтверждение пользователя: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.ACCESS): f''' +Текущий доступ к проекту: +#{access_field} + +{user_access.user_access_readme} + +Введите новую строку доступа: +''', + mod_table_operate.Messages.SUCCESS_EDIT: '''✅ Проект успешно отредактирован!''', + mod_table_operate.Messages.SELECT_TO_DELETE: ''' +Выберите пользователя, у которого вы хотите удалить авторизованные данные. +''', + mod_table_operate.Messages.SUCCESS_DELETE: '''✅ Данные пользователя успешно удалены!''', +} + +messages_subscribes = { + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ADD):f'''Данные по авторизации созданы''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Данные по авторизации отредактированы''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Данные по авторизации удалены''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_EDIT):f'''Данные по авторизации отредактированы #item_id''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_DEL):f'''Данные по авторизации удалены #item_id''', +} + +messages.update(messages_subscribes) + +messages_order_status = { + mod_table_operate.EnumMessageForView(ConfirmStatus.YES): f'''Да, все данные верны''', + mod_table_operate.EnumMessageForView(ConfirmStatus.NO): f'''Нет, данные не верны''', +} + +messages.update(messages_order_status) + +class ModuleAuthorize(mod_table_operate.TableOperateModule): + def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + + def GetName(self): + return module_name + diff --git a/bot_sys/bd_table.py b/bot_sys/bd_table.py index 593b146..b5b98d6 100644 --- a/bot_sys/bd_table.py +++ b/bot_sys/bd_table.py @@ -38,6 +38,14 @@ class TableFieldDestiny(Enum): OTHER = auto() VERSION_NUMBER = auto() SUB_VERSION_NUMBER = auto() + USER_ID = auto() + USER_NAME = auto() # Имя + USER_FAMILY_NAME = auto() # Фамилия + USER_MIDDLE_NAME = auto() # Отчество + USER_BIRTHDAY = auto() # Дата рождения + USER_ADDRESS = auto() # Адресс + USER_CONTACTS = auto() # контакты + USER_CONFIRM = auto() # подтверждение class TableField: def __init__(self, a_Name, a_Destiny : TableFieldDestiny, a_Type : TableFieldType, a_Enum = None): diff --git a/main.py b/main.py index 3ed4e1f..b064418 100644 --- a/main.py +++ b/main.py @@ -6,7 +6,7 @@ import os from bot_sys import config, log, aiogram_bot, bot_messages, bd_table, user_access from bot_modules import mod_agregator, start, profile, backup, users_groups_agregator, access, projects, tasks, needs, comments from bot_modules import languages, messages, buttons, users, groups, user_in_groups -from bot_modules import orders, all_orders +from bot_modules import orders, all_orders, authorize from bot_modules import bd_version from bot_sys import bot_subscribes @@ -110,6 +110,10 @@ g_ModuleAgregator.AddModule(mod_subscribe) mod_bd_version = bd_version.ModuleBDVersion(g_Bot, g_ModuleAgregator, g_BotMessages, g_BotButtons, g_BotSubscribes, g_Log) g_ModuleAgregator.AddModule(mod_bd_version) +child_mod_name_list = [mod_start_name] +mod_authorize = authorize.ModuleAuthorize(None, None, child_mod_name_list, start_mod_list, g_Bot, g_ModuleAgregator, g_BotMessages, g_BotButtons, g_BotSubscribes, g_Log) +g_ModuleAgregator.AddModule(mod_authorize) + child_mod_name_list = [ mod_profile.GetName(), mod_backup.GetName(), @@ -121,6 +125,7 @@ child_mod_name_list = [ mod_all_orders.GetName(), mod_subscribe.GetName(), mod_bd_version.GetName(), + mod_authorize.GetName(), ] mod_start = start.ModuleStart(child_mod_name_list, g_Bot, g_ModuleAgregator, g_BotMessages, g_BotButtons, g_BotSubscribes, g_Log) g_ModuleAgregator.AddModule(mod_start) From 1a388229e830d9f23b94882fa6553a2a4c5c2aca Mon Sep 17 00:00:00 2001 From: Alexei Date: Sun, 8 Oct 2023 23:34:33 +0300 Subject: [PATCH 03/21] =?UTF-8?q?=D0=A0=D0=B5=D0=B0=D0=BB=D0=B8=D0=B7?= =?UTF-8?q?=D0=BE=D0=B2=D0=B0=D0=BD=D0=B0=20=D1=80=D0=B0=D0=B1=D0=BE=D1=82?= =?UTF-8?q?=D0=B0=20=D1=81=20=D0=B4=D0=BE=D0=BA=D1=83=D0=BC=D0=B5=D0=BD?= =?UTF-8?q?=D1=82=D0=B0=D0=BC=D0=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- auth_docs/test_doc.odt | 273 +++++++++++++++++++++++++++++++++++++++ bot_modules/authorize.py | 128 ++++++++++++++++-- bot_modules/mod_table_operate.py | 11 +- bot_sys/bot_bd.py | 1 + bot_sys/user_access.py | 2 + config_auth_docs | 1 + requirements.txt | 1 + template/docs_message.py | 77 +++++++++++ template/file_message.py | 2 +- 9 files changed, 480 insertions(+), 16 deletions(-) create mode 100644 auth_docs/test_doc.odt create mode 100644 config_auth_docs create mode 100644 template/docs_message.py diff --git a/auth_docs/test_doc.odt b/auth_docs/test_doc.odt new file mode 100644 index 0000000..6047b4a --- /dev/null +++ b/auth_docs/test_doc.odt @@ -0,0 +1,273 @@ + + + + 2023-10-08T23:17:27.894126099PT46S2LibreOffice/7.3.6.2$Linux_X86_64 LibreOffice_project/30$Build-2 + + + 0 + 0 + 35588 + 13813 + true + false + + + view2 + 22826 + 5909 + 0 + 0 + 35586 + 13811 + 0 + 1 + false + 100 + false + false + false + + + + + false + false + false + true + true + true + true + true + false + 0 + false + false + false + true + false + false + true + false + false + false + false + true + true + true + false + false + false + false + false + false + false + true + false + false + true + false + false + false + true + 0 + 1 + true + false + + high-resolution + true + + + false + false + true + false + true + true + false + true + + true + 820887 + + true + false + true + true + 0 + + false + false + false + true + false + true + false + false + false + false + true + false + false + + false + false + true + false + false + false + false + false + false + false + false + false + 820887 + false + false + false + false + false + true + false + true + true + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + bd_table.TableFieldDestiny.USER_NAME: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_FAMILY_NAME: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_MIDDLE_NAME: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_BIRTHDAY: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_ADDRESS: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_CONTACTS: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_CONFIRM: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.ACCESS: user_access.AccessMode.NONE, + + + \ No newline at end of file diff --git a/bot_modules/authorize.py b/bot_modules/authorize.py index 2831c0a..e2f515e 100644 --- a/bot_modules/authorize.py +++ b/bot_modules/authorize.py @@ -3,8 +3,9 @@ # Авторизация или регистрация в кооперативе или другой организации -from bot_sys import bot_bd, keyboard, user_access, bd_table, bot_subscribes +from bot_sys import bot_bd, keyboard, user_access, bd_table, bot_subscribes, config from bot_modules import mod_table_operate, mod_simple_message +from template import docs_message from enum import Enum from enum import auto @@ -50,6 +51,12 @@ table = bd_table.Table(table_name, [ init_access = f'{user_access.user_access_group_new}=va' +def GetAuthorizeItem(a_Bot, a_UserID): + items = bd_item.GetBDItemsTemplate(a_Bot, table_name, user_id_field)(a_UserID) + if len(items) == 1: + return items[0] + return None + # --------------------------------------------------------- # Сообщения и кнопки @@ -60,24 +67,35 @@ button_names = { mod_simple_message.ButtonNames.START: "🔑 Авторизация", mod_table_operate.ButtonNames.LIST: "≣ Список авторизаций пользователей", mod_table_operate.ButtonNames.ADD: "📨 Заявка на вступление", + ButtonNames.LIST_AUTH_DOCS: "📨 Регистрационные документы", mod_table_operate.ButtonNames.EDIT: "🛠 Редактировать свои данные", - mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", - mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_NAME): "☐ Изменить имя пользователя", - mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_FAMILY_NAME): "☐ Изменить фамилию пользователя", - mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_MIDDLE_NAME): "☐ Изменить отчество пользователя", - mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_BIRTHDAY): "☐ Изменить дату рождения пользователя", - mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ADDRESS): "☐ Изменить адрес пользователя", - mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_CONTACTS): "☐ Изменить контакты пользователя", - mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_CONFIRM): "☐ Изменить подтверждение пользователя", -# mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", - mod_table_operate.EditButton(bd_table.TableFieldDestiny.ACCESS): "✋ Доступ к авторизации пользователя", mod_table_operate.ButtonNames.DEL: "❌ Удалить авторизацию пользователя", + bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", + bd_table.TableFieldDestiny.USER_NAME): "☐ Изменить имя пользователя", + bd_table.TableFieldDestiny.USER_FAMILY_NAME): "☐ Изменить фамилию пользователя", + bd_table.TableFieldDestiny.USER_MIDDLE_NAME): "☐ Изменить отчество пользователя", + bd_table.TableFieldDestiny.USER_BIRTHDAY): "☐ Изменить дату рождения пользователя", + bd_table.TableFieldDestiny.USER_ADDRESS): "☐ Изменить адрес пользователя", + bd_table.TableFieldDestiny.USER_CONTACTS): "☐ Изменить контакты пользователя", + bd_table.TableFieldDestiny.USER_CONFIRM): "☐ Изменить подтверждение пользователя", +# bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", + bd_table.TableFieldDestiny.ACCESS): "✋ Доступ к авторизации пользователя", mod_table_operate.ButtonNames.DEL: "❌ Удалить авторизацию пользователя", } +class Messages(Enum): + LIST_AUTH_DOCS = auto() + LIST_AUTH_DOCS_ERROR = auto() + messages = { mod_simple_message.Messages.START: f''' {button_names[mod_simple_message.ButtonNames.START]} ''', + Messages.LIST_AUTH_DOCS: ''' +ваши регистрационные документы: +''', + Messages.LIST_AUTH_DOCS_ERROR: ''' +Ошибка получения документов. Обратитесь в техподдержку. +''', mod_table_operate.Messages.SELECT: ''' Пожалуйста, выберите пользователя: ''', @@ -229,3 +247,91 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): def GetName(self): return module_name + def GetAccessForEditKeyboardButtons(self, a_Field): + cur_dict = { + bd_table.TableFieldDestiny.USER_ID: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_NAME: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_FAMILY_NAME: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_MIDDLE_NAME: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_BIRTHDAY: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_ADDRESS: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_CONTACTS: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_CONFIRM: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.ACCESS: user_access.AccessMode.NONE, + } + return cur_dict.get(a_Field.m_Destiny, super().GetAccessForEditKeyboardButtons(a_Field)) + + def GetButtonNameAndKeyValueAndAccess(self, a_Item): + key_name_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.USER_ID) + name_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.USER_NAME) + fam_name_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.USER_FAMILY_NAME) + access_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.ACCESS) + assert key_name_id != None + assert name_field_id != None + assert fam_name_field_id != None + assert access_field_id != None + return \ + a_Item[name_field_id] + ' ' + a_Item[fam_name_field_id] + '(' + a_Item[key_name_id] +')',\ + a_Item[key_name_id],\ + a_Item[access_field_id] + + def GetStartKeyboardButtons(self, a_Message, a_UserGroups): + parent_buttons = super().GetStartKeyboardButtons(a_Message, a_UserGroups) + user_id = str(a_Message.from_user.id) + item = GetAuthorizeItem(self.m_Bot, user_id) + user_confirm_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.USER_CONFIRM) + cur_buttons = [] + if item and item[user_confirm_field_id] == str(ConfirmStatus.YES): + cur_buttons += [ + keyboard.ButtonWithAccess(self.GetButton(ButtonNames.LIST_AUTH_DOCS), user_access.AccessMode.VIEW, self.GetAccess()), + ] + return parent_buttons + keyboard.MakeButtons(self.m_Bot, cur_buttons, a_UserGroups) + + def RegisterHandlers(self): + super().RegisterHandlers() + + def GetFilesFunc(a_user_id): + files = GetAuthDocs() + cur_dict = GetReplaceDictFunc(self.m_Bot, a_user_id) + result = {} + for f in files: + result.update({f, cur_dict}) + return result + + button_name = self.GetButton(ButtonNames.LIST_AUTH_DOCS) + if button_name: + self.m_Bot.RegisterMessageHandler( + docs_message.BackupFileTemplate( + self.m_Bot, + GetFilesFunc, + self.GetMessage(Messages.LIST_AUTH_DOCS), + self.m_GetAccessFunc, + self.m_GetStartKeyboardButtonsFunc, + None, + self.GetMessage(Messages.LIST_AUTH_DOCS_ERROR), + access_mode = user_access.AccessMode.VIEW + ), + bd_item.GetCheckForTextFunc(button_name) + ) + +def GetReplaceDictFunc(a_Bot, a_user_id): + item = GetAuthorizeItem(self.m_Bot, user_id) + if not item: + return None + result = {} + i = 0 + for f in table: + result.update({str(f.m_Destiny), item[i]}) + i += 1 + return result + +auth_docs_file_name = 'config_auth_docs' + +def GetAuthDocs(): + global g_auth_docs + if len(g_auth_docs) == 0: + root_ids = config.GetAllLinesFromFile(auth_docs_file_name) + for i in root_ids: + g_auth_docs += [config.ClearReadLine(i)] + + return g_auth_docs diff --git a/bot_modules/mod_table_operate.py b/bot_modules/mod_table_operate.py index acff162..e927b0d 100644 --- a/bot_modules/mod_table_operate.py +++ b/bot_modules/mod_table_operate.py @@ -126,13 +126,16 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): ] return mod_buttons + keyboard.MakeButtons(self.m_Bot, cur_buttons, a_UserGroups) + def GetAccessForEditKeyboardButtons(self, a_Field): + if a_Field.m_Destiny == bd_table.TableFieldDestiny.ACCESS or a_Field.m_Destiny == bd_table.TableFieldDestiny.DEFAULT_ACCESS: + return user_access.AccessMode.ACCEES_EDIT + return user_access.AccessMode.EDIT + def GetEditKeyboardButtons(self, a_Message, a_UserGroups): mod_buttons = keyboard.MakeButtons(self.m_Bot, self.GetButtons(self.m_EditModuleNameList), a_UserGroups) cur_buttons = [] for f in self.m_Table.GetFields(): - access = user_access.AccessMode.EDIT - if f.m_Destiny == bd_table.TableFieldDestiny.ACCESS or f.m_Destiny == bd_table.TableFieldDestiny.DEFAULT_ACCESS: - access = user_access.AccessMode.ACCEES_EDIT + access = self.GetAccessForEditKeyboardButtons(f) cur_buttons += [keyboard.ButtonWithAccess(self.GetButton(EditButton(f.m_Destiny)), access, self.GetAccess()),] return mod_buttons + keyboard.MakeButtons(self.m_Bot, cur_buttons, a_UserGroups) @@ -185,7 +188,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): def ShowMessageTemplate(self, a_Message, Inline_keyboard_template_func = None, a_EnablePhoto = False): async def ShowMessage(a_CallbackQuery, a_Item): msg = a_Message.StaticCopy() - # TODO: добавить поддержку языка в a_MessageName + # TODO: добавить поддержку языков в a_MessageName Inline_keyboard_func = None item_access = None if a_Item: diff --git a/bot_sys/bot_bd.py b/bot_sys/bot_bd.py index ec81174..47583b2 100644 --- a/bot_sys/bot_bd.py +++ b/bot_sys/bot_bd.py @@ -9,6 +9,7 @@ from bot_sys import log def GetBDDateTimeNow(): return 'datetime(\'now\')' +# TODO: Удалить что-то из SelectBDTemplate RequestSelectTemplate def SelectBDTemplate(a_Bot, a_TableName): def SelectBD(): return a_Bot.SQLRequest(f'SELECT * FROM {a_TableName}') diff --git a/bot_sys/user_access.py b/bot_sys/user_access.py index abe3127..29750ca 100644 --- a/bot_sys/user_access.py +++ b/bot_sys/user_access.py @@ -32,11 +32,13 @@ ACCEES_EDIT = 'r' - изменение прав доступа # Типы уровня доступа class AccessMode(Enum): + NONE = '-' VIEW = 'v' ADD = 'a' EDIT = 'e' DELETE = 'd' ACCEES_EDIT = 'r' + ALL = '+' class UserGroups: def __init__(self, a_UserID : str, a_GroupNamesList : [str]): diff --git a/config_auth_docs b/config_auth_docs new file mode 100644 index 0000000..7822470 --- /dev/null +++ b/config_auth_docs @@ -0,0 +1 @@ +auth_docs/test_doc.odt diff --git a/requirements.txt b/requirements.txt index e99a1ef..9e91223 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,2 +1,3 @@ aiogram==2.20 colorama==0.4.5 +odfpy diff --git a/template/docs_message.py b/template/docs_message.py new file mode 100644 index 0000000..7544125 --- /dev/null +++ b/template/docs_message.py @@ -0,0 +1,77 @@ +# -*- coding: utf8 -*- +# Общественное достояние 2023, Алексей Безбородов (Alexei Bezborodov) + +# Сообщения для работы с документами + +from bot_sys import log, config, user_access +from bot_modules import groups_utils +from template import simple_message +import odf + +def DocFilesTemplate(a_Bot, a_FilesFunc, a_CaptionMessage, a_AccessFunc, a_GetButtonsFunc, a_GetInlineButtonsFunc, a_ErrorMessage, access_mode = user_access.AccessMode.EDIT): + async def DocFiles(a_Message): + user_id = str(a_Message.from_user.id) + user_groups= groups_utils.GetUserGroupData(a_Bot, user_id) + if not user_access.CheckAccess(a_Bot.GetRootIDs(), a_AccessFunc(), user_groups, access_mode): + return await simple_message.AccessDeniedMessage(a_Bot, a_GetButtonsFunc, user_id, a_Message, user_groups) + + msg = a_CaptionMessage.GetDesc() + msg = msg.replace('@time', a_Bot.GetLog().GetTime()) + + files = a_FilesFunc(user_id) + if not files: + await simple_message.SendMessage(a_Bot, a_ErrorMessage, a_GetButtonsFunc, None, user_id, a_Message, user_groups) + return + + for file_path, dict_replace in files.values(): + if not dict_replace: + continue + new_file = await MakeDocFile(a_Bot, file_path, dict_replace, user_id) + document = await GetFile(a_Bot, file_path) + if document is None: + await simple_message.SendMessage(a_Bot, a_ErrorMessage, a_GetButtonsFunc, None, user_id, a_Message, user_groups) + else: + await a_Bot.SendDocument( + user_id, + document, + msg, + simple_message.ProxyGetButtonsTemplate(a_GetButtonsFunc)(a_Message, user_groups), + simple_message.ProxyGetButtonsTemplate(a_GetInlineButtonsFunc)(a_Message, user_groups) + ) + return DocFiles + +async def MakeDocFile(a_Bot, a_FilePath, a_DictReplace, a_user_id): + document = odf.opendocument.load(a_FilePath) + if document == None: + a_Bot.GetLog().Error(f'Не удалось загрузить файл {a_FilePath}.') + return None + + for element in document.getElementsByType(text.Span): + extracted_text = odf.teletype.extractText(element) + + for file_path, dict_replace in a_DictReplace.values(): + if extracted_text.find('Replace this') != -1: + extracted_text = extracted_text.replace('Replace this', 'to this') + + new_element = odf.text.Span() + new_element.setAttribute('stylename', element.getAttribute('stylename')) + new_element.addText(extracted_text) + + element.parentNode.insertBefore(new_element, element) + element.parentNode.removeChild(element) + + new_file_path = a_FilePath[:-4] + new_file_path += f"{a_user_id}.odt" + document.save(new_file_path) + return new_file_path + +async def GetFile(a_Bot, a_Path): + if not a_Path: + return None + try: + document = open(a_Path, 'rb') + a_Bot.GetLog().Success(f'Загружен файл {a_Path}') + return document + except Exception as e: + a_Bot.GetLog().Error(f'Не удалось загрузить файл {a_Path}. Ошибка {str(e)}') + return None diff --git a/template/file_message.py b/template/file_message.py index fe7e6a7..0159aa3 100644 --- a/template/file_message.py +++ b/template/file_message.py @@ -16,7 +16,7 @@ def BackupFileTemplate(a_Bot, a_Path, a_CaptionMessage, a_AccessFunc, a_GetButto document = await GetFile(a_Bot, a_Path) if document is None: - return simple_message.SendMessage(a_Bot, a_ErrorMessage, a_GetButtonsFunc, None, user_id, a_Message, user_groups) + return await simple_message.SendMessage(a_Bot, a_ErrorMessage, a_GetButtonsFunc, None, user_id, a_Message, user_groups) msg = a_CaptionMessage.GetDesc() msg = msg.replace('@time', a_Bot.GetLog().GetTime()) From d7ea8ecd756f4c62e24f4ac4c02cf18cc7191d3a Mon Sep 17 00:00:00 2001 From: Alexei Date: Fri, 13 Oct 2023 20:54:52 +0300 Subject: [PATCH 04/21] =?UTF-8?q?=D0=9F=D1=80=D0=BE=D1=82=D0=B5=D1=81?= =?UTF-8?q?=D1=82=D0=B8=D1=80=D0=BE=D0=B2=D0=B0=D0=BD=D0=B0=20=D0=B0=D0=B2?= =?UTF-8?q?=D1=82=D0=BE=D1=80=D0=B8=D0=B7=D0=B0=D1=86=D0=B8=D1=8F=20=D0=BD?= =?UTF-8?q?=D0=BE=D0=B2=D0=BE=D0=B3=D0=BE=20=D0=BF=D0=BE=D0=BB=D1=8C=D0=B7?= =?UTF-8?q?=D0=BE=D0=B2=D0=B0=D1=82=D0=B5=D0=BB=D1=8F=20=D0=B4=D0=BE=20?= =?UTF-8?q?=D0=BC=D0=BE=D0=BC=D0=B5=D0=BD=D1=82=D0=B0=20=D0=B4=D0=BE=D0=BA?= =?UTF-8?q?=D1=83=D0=BC=D0=B5=D0=BD=D1=82=D0=BE=D0=B2?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- auth_docs/test_doc.odt | 51 ++++++++++++++++++++++++++-------------- bot_modules/authorize.py | 38 ++++++++++++++++++------------ bot_modules/mod_table_operate.py | 48 +++++++++++++++++++++++++++++++------ bot_modules/subscribes.py | 7 ------ bot_sys/bd_table.py | 7 ++++++ template/bd_item.py | 2 +- template/bd_item_add.py | 4 ++-- 7 files changed, 107 insertions(+), 50 deletions(-) diff --git a/auth_docs/test_doc.odt b/auth_docs/test_doc.odt index 6047b4a..abc5c88 100644 --- a/auth_docs/test_doc.odt +++ b/auth_docs/test_doc.odt @@ -1,7 +1,7 @@ - 2023-10-08T23:17:27.894126099PT46S2LibreOffice/7.3.6.2$Linux_X86_64 LibreOffice_project/30$Build-2 + 2023-10-13T15:56:51.616305329PT1M19S4LibreOffice/7.3.6.2$Linux_X86_64 LibreOffice_project/30$Build-2 0 @@ -13,8 +13,8 @@ view2 - 22826 - 5909 + 18221 + 3475 0 0 35586 @@ -88,7 +88,7 @@ true true - 820887 + 918403 true false @@ -149,7 +149,7 @@ - + @@ -184,52 +184,52 @@ - + - + - + - + - + - + - + - + - + - + @@ -240,16 +240,22 @@ + + + - + + + + - + @@ -260,13 +266,22 @@ - bd_table.TableFieldDestiny.USER_NAME: user_access.AccessMode.NONE, + + + bd_table.TableFieldDestiny.USER_NAME: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_FAMILY_NAME: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_MIDDLE_NAME: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_BIRTHDAY: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_ADDRESS: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_CONTACTS: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.USER_CONFIRM: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.ACCESS: user_access.AccessMode.NONE, diff --git a/bot_modules/authorize.py b/bot_modules/authorize.py index e2f515e..e7250d4 100644 --- a/bot_modules/authorize.py +++ b/bot_modules/authorize.py @@ -5,7 +5,7 @@ from bot_sys import bot_bd, keyboard, user_access, bd_table, bot_subscribes, config from bot_modules import mod_table_operate, mod_simple_message -from template import docs_message +from template import docs_message, bd_item from enum import Enum from enum import auto @@ -69,16 +69,19 @@ button_names = { mod_table_operate.ButtonNames.ADD: "📨 Заявка на вступление", ButtonNames.LIST_AUTH_DOCS: "📨 Регистрационные документы", mod_table_operate.ButtonNames.EDIT: "🛠 Редактировать свои данные", - bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", - bd_table.TableFieldDestiny.USER_NAME): "☐ Изменить имя пользователя", - bd_table.TableFieldDestiny.USER_FAMILY_NAME): "☐ Изменить фамилию пользователя", - bd_table.TableFieldDestiny.USER_MIDDLE_NAME): "☐ Изменить отчество пользователя", - bd_table.TableFieldDestiny.USER_BIRTHDAY): "☐ Изменить дату рождения пользователя", - bd_table.TableFieldDestiny.USER_ADDRESS): "☐ Изменить адрес пользователя", - bd_table.TableFieldDestiny.USER_CONTACTS): "☐ Изменить контакты пользователя", - bd_table.TableFieldDestiny.USER_CONFIRM): "☐ Изменить подтверждение пользователя", -# bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", - bd_table.TableFieldDestiny.ACCESS): "✋ Доступ к авторизации пользователя", mod_table_operate.ButtonNames.DEL: "❌ Удалить авторизацию пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_NAME): "☐ Изменить имя пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_FAMILY_NAME): "☐ Изменить фамилию пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_MIDDLE_NAME): "☐ Изменить отчество пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_BIRTHDAY): "☐ Изменить дату рождения пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ADDRESS): "☐ Изменить адрес пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_CONTACTS): "☐ Изменить контакты пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_CONFIRM): "☐ Изменить подтверждение пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.ACCESS): "✋ Доступ к авторизации пользователя", + mod_table_operate.ButtonNames.DEL: "❌ Удалить авторизацию пользователя", + mod_table_operate.EnumButton(ConfirmStatus.YES): "Да, все данные верны", + mod_table_operate.EnumButton(ConfirmStatus.NO): "Нет, данные не верны", } class Messages(Enum): @@ -145,7 +148,7 @@ messages = { Введите свои контакты: ''', - mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.USER_CONFIRM): ''' + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.USER_CONFIRM): f''' Авторизация. Шаг №7 Имя: #{user_name_field} @@ -233,12 +236,12 @@ messages_subscribes = { messages.update(messages_subscribes) -messages_order_status = { +messages_confirm_status = { mod_table_operate.EnumMessageForView(ConfirmStatus.YES): f'''Да, все данные верны''', mod_table_operate.EnumMessageForView(ConfirmStatus.NO): f'''Нет, данные не верны''', } -messages.update(messages_order_status) +messages.update(messages_confirm_status) class ModuleAuthorize(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): @@ -301,7 +304,7 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): button_name = self.GetButton(ButtonNames.LIST_AUTH_DOCS) if button_name: self.m_Bot.RegisterMessageHandler( - docs_message.BackupFileTemplate( + docs_message.DocFilesTemplate( self.m_Bot, GetFilesFunc, self.GetMessage(Messages.LIST_AUTH_DOCS), @@ -314,6 +317,11 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): bd_item.GetCheckForTextFunc(button_name) ) + def AddBDItemFunc(self, a_ItemData, a_UserID): + a_ItemData[user_id_field] = a_UserID + #a_ItemData[address_field] = '' + return super().AddBDItemFunc(a_ItemData, a_UserID) + def GetReplaceDictFunc(a_Bot, a_user_id): item = GetAuthorizeItem(self.m_Bot, user_id) if not item: diff --git a/bot_modules/mod_table_operate.py b/bot_modules/mod_table_operate.py index e927b0d..2fc8a8d 100644 --- a/bot_modules/mod_table_operate.py +++ b/bot_modules/mod_table_operate.py @@ -185,12 +185,32 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): a_Msg.UpdatePhotoID(','.join(photos)) return a_Msg + def UpdateMessageByDict(self, a_Msg, a_Lang, a_ItemDict, a_EnablePhoto = False): + a_Msg.UpdateDesc(self.m_Table.ReplaceAllFieldTagsByDict(a_Msg.GetDesc(), a_ItemDict)) + photos = [] + for f in self.m_Table.GetFields(): + if f.m_Type == bd_table.TableFieldType.ENUM: + for s in f.m_Enum: + msg = self.GetMessage(EnumMessageForView(s)) + if msg: + a_Msg.UpdateDesc(a_Msg.GetDesc().replace(str(s), str(msg.GetMessageForLang(a_Lang).StaticCopy()))) + elif f.m_Type == bd_table.TableFieldType.PHOTO: + if f.m_Name in a_ItemDict: + photos += [str(a_ItemDict[f.m_Name])] + if a_EnablePhoto: + a_Msg.UpdatePhotoID(','.join(photos)) + return a_Msg + def ShowMessageTemplate(self, a_Message, Inline_keyboard_template_func = None, a_EnablePhoto = False): - async def ShowMessage(a_CallbackQuery, a_Item): + async def ShowMessage(a_CallbackQuery, a_Item, a_ItemDict): msg = a_Message.StaticCopy() # TODO: добавить поддержку языков в a_MessageName Inline_keyboard_func = None item_access = None + if a_ItemDict: + lang = str(a_CallbackQuery.from_user.language_code) + msg = msg.GetMessageForLang(lang).StaticCopy() + msg = self.UpdateMessageByDict(msg, lang, a_ItemDict, a_EnablePhoto = a_EnablePhoto) if a_Item: if len(a_Item) < self.m_Table.GetFieldsCount() - 1: # Для проектов это нужно. Там на 1 меньше поле. TODO разделить отправку сообщений item_access и Inline_keyboard_func return simple_message.WorkFuncResult(self.GetMessage(Messages.ERROR_FIND)) @@ -207,7 +227,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): async def PreDelete(self, a_CallbackQuery, a_Item): if len(a_Item) < self.m_Table.GetFieldsCount(): - return simple_message.WorkFuncResult(error_find_proj_message) + return simple_message.WorkFuncResult(self.GetMessage(Messages.ERROR_FIND)) access = a_Item[self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.ACCESS)] return simple_message.WorkFuncResult(self.GetMessage(Messages.SUCCESS_DELETE), None, item_access = access) @@ -312,12 +332,12 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): return a_Prefix - def AdditionalKeyboardForEditTemplate(self, a_Field): + def AdditionalKeyboardForEditTemplate(self, a_Field, user_access = user_access.AccessMode.EDIT): if a_Field.m_Type == bd_table.TableFieldType.ENUM: def KeyboardButtons(a_Message, a_UserGroups): cur_buttons = [] for s in a_Field.m_Enum: - cur_buttons += [keyboard.ButtonWithAccess(self.GetButton(EnumButton(s)), user_access.AccessMode.EDIT, self.GetAccess()),] + cur_buttons += [keyboard.ButtonWithAccess(self.GetButton(EnumButton(s)), user_access, self.GetAccess()),] return keyboard.MakeButtons(self.m_Bot, cur_buttons, a_UserGroups) return KeyboardButtons return None @@ -386,9 +406,23 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): ) def GetAddFields(self): + good_destiny = ( + bd_table.TableFieldDestiny.NAME, + bd_table.TableFieldDestiny.DESC, + bd_table.TableFieldDestiny.PHOTO, + bd_table.TableFieldDestiny.SUBSCRIBE_TYPE, + bd_table.TableFieldDestiny.ITEM_ID, + bd_table.TableFieldDestiny.USER_NAME, + bd_table.TableFieldDestiny.USER_FAMILY_NAME, + bd_table.TableFieldDestiny.USER_MIDDLE_NAME, + bd_table.TableFieldDestiny.USER_BIRTHDAY, + bd_table.TableFieldDestiny.USER_ADDRESS, + bd_table.TableFieldDestiny.USER_CONTACTS, + bd_table.TableFieldDestiny.USER_CONFIRM, + ) fields = [] for f in self.m_Table.GetFields(): - if f.m_Destiny in (bd_table.TableFieldDestiny.NAME, bd_table.TableFieldDestiny.DESC, bd_table.TableFieldDestiny.PHOTO, bd_table.TableFieldDestiny.SUBSCRIBE_TYPE, bd_table.TableFieldDestiny.ITEM_ID, ): + if f.m_Destiny in good_destiny: fields += [f] return fields @@ -422,13 +456,13 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): f_id = 0 f = fields[f_id] - keyboard = bd_item.MixKeyboardFuncTemplate(self.AdditionalKeyboardForEditTemplate(f), GetKeyboard(f)) + keyboard = bd_item.MixKeyboardFuncTemplate(self.AdditionalKeyboardForEditTemplate(f, user_access.AccessMode.ADD), GetKeyboard(f)) reg_func(bd_item_add.StartAddBDItemTemplate(self.m_Bot, fsm, getattr(fsm, f'item{f_id}'), self.ShowMessageTemplate(self.GetMessage(CreateMessage(f.m_Destiny))), a_ParentTableName, a_ParentKeyFieldName, a_ParentPrefix, a_AccessFunc, keyboard, a_ButtonFunc, access_mode), a_StartCheckFunc) for i in range(len(fields) - 1): f = fields[i] next_f = fields[i + 1] - keyboard = bd_item.MixKeyboardFuncTemplate(self.AdditionalKeyboardForEditTemplate(next_f), GetKeyboard(next_f)) + keyboard = bd_item.MixKeyboardFuncTemplate(self.AdditionalKeyboardForEditTemplate(next_f, user_access.AccessMode.ADD), GetKeyboard(next_f)) self.m_Bot.RegisterMessageHandler(bd_item_add.NextAddBDItemTemplate(self.m_Bot, fsm, None, a_ParentTableName, a_ParentKeyFieldName, f.m_Name, self.ShowMessageTemplate(self.GetMessage(CreateMessage(next_f.m_Destiny))), None, a_AccessFunc, keyboard, a_ButtonFunc, access_mode, field_type = GetFieldType(f)), content_types = GetContentTypes(f), state = getattr(fsm, f'item{i}')) f_id = len(fields) - 1 diff --git a/bot_modules/subscribes.py b/bot_modules/subscribes.py index 84c2715..a807b8d 100644 --- a/bot_modules/subscribes.py +++ b/bot_modules/subscribes.py @@ -154,13 +154,6 @@ messages_subs_type_status = { messages.update(messages_subs_type_status) -def GetCurItemsTemplate(a_Bot, a_TableName, a_UserIDFieldName, a_StatusFieldName): - def GetBDItems(a_Message, a_UserGroups, a_ParentID): - user_id = str(a_Message.from_user.id) - request = f'SELECT * FROM {a_TableName} WHERE {a_UserIDFieldName} = ? AND {a_StatusFieldName} != ?' - return a_Bot.SQLRequest(request, param = ([user_id, str(OrderStatus.FINISH)])) - return GetBDItems - def GetBDItemsForUserTemplate(a_Bot, a_TableName, a_UserIDFieldName): def GetBDItems(a_Message, a_UserGroups, a_ParentID): user_id = str(a_Message.from_user.id) diff --git a/bot_sys/bd_table.py b/bot_sys/bd_table.py index b5b98d6..2af2e98 100644 --- a/bot_sys/bd_table.py +++ b/bot_sys/bd_table.py @@ -111,6 +111,13 @@ class Table: result = result.replace(f'#{f.m_Name}', str(a_BDItem[i])) return result + def ReplaceAllFieldTagsByDict(self, a_String, a_BDItemDict): + result = a_String + for f in self.m_Fields: + if f.m_Name in a_BDItemDict: + result = result.replace(f'#{f.m_Name}', str(a_BDItemDict[f.m_Name])) + return result + class Status(Enum): NEW = auto() FINISH = auto() diff --git a/template/bd_item.py b/template/bd_item.py index 5a08a84..8cc9783 100644 --- a/template/bd_item.py +++ b/template/bd_item.py @@ -100,7 +100,7 @@ def CheckAccessBDItemTemplate(a_Bot, a_TableName, a_KeyName, a_KeyValue, a_WorkF a_Bot.GetLog().Error(msg) return simple_message.WorkFuncResult(bot_messages.MakeBotMessage(msg)), None - result_work_func = await a_WorkFunc(a_CallbackQuery, item[0]) + result_work_func = await a_WorkFunc(a_CallbackQuery, item[0], None) if result_work_func is None or result_work_func.m_BotMessage is None: return result_work_func, result_work_func diff --git a/template/bd_item_add.py b/template/bd_item_add.py index 54c6960..e52ae4c 100644 --- a/template/bd_item_add.py +++ b/template/bd_item_add.py @@ -30,7 +30,7 @@ def StartAddBDItemTemplate(a_Bot, a_FSM, a_FSMStart, a_MessageFunc, a_ParentTabl if parent_id: check, res_of_work_func = await bd_item.CheckAccessBDItemTemplate(a_Bot, a_ParentTableName, a_ParentKeyFieldName, parent_id, a_MessageFunc, access_mode)(a_CallbackQuery) else: - res_of_work_func = await a_MessageFunc(a_CallbackQuery, None) + res_of_work_func = await a_MessageFunc(a_CallbackQuery, None, item_data) if not check is None: await state.finish() @@ -68,7 +68,7 @@ def FinishOrNextAddBDItemTemplate(a_Bot, a_FSM, a_AddBDItemFunc, a_ParentTableNa if parent_id: check, res_of_work_func = await bd_item.CheckAccessBDItemTemplate(a_Bot, a_ParentTableName, a_ParentKeyFieldName, parent_id, a_MessageFunc, access_mode)(a_Message) else: - res_of_work_func = await a_MessageFunc(a_Message, None) + res_of_work_func = await a_MessageFunc(a_Message, None, item_data) if not check is None: await state_func() From 2ada0a2508fbf23db15d507ef12d73a824c80ba1 Mon Sep 17 00:00:00 2001 From: Alexei Date: Sat, 14 Oct 2023 22:14:42 +0300 Subject: [PATCH 05/21] =?UTF-8?q?=D0=98=D1=81=D0=BF=D1=80=D0=B0=D0=B2?= =?UTF-8?q?=D0=BB=D0=B5=D0=BD=D0=B0=20=D1=80=D0=B0=D0=B1=D0=BE=D1=82=D0=B0?= =?UTF-8?q?=20=D1=81=20=D0=B4=D0=BE=D0=BA=D1=83=D0=BC=D0=B5=D0=BD=D1=82?= =?UTF-8?q?=D0=B0=D0=BC=D0=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- auth_docs/test_doc.odt | Bin 29149 -> 19082 bytes bot_modules/authorize.py | 20 +++++++++++++------- bot_modules/mod_table_operate.py | 4 ++-- template/docs_message.py | 25 +++++++++++++++---------- 4 files changed, 30 insertions(+), 19 deletions(-) diff --git a/auth_docs/test_doc.odt b/auth_docs/test_doc.odt index abc5c88005da731bdb37a0bdef955abb781597ee..e937454b43901dd0f3ed5cac7cd085c9fedf03ee 100644 GIT binary patch literal 19082 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 literal 29149 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 diff --git a/bot_modules/authorize.py b/bot_modules/authorize.py index e7250d4..1e572d6 100644 --- a/bot_modules/authorize.py +++ b/bot_modules/authorize.py @@ -5,7 +5,7 @@ from bot_sys import bot_bd, keyboard, user_access, bd_table, bot_subscribes, config from bot_modules import mod_table_operate, mod_simple_message -from template import docs_message, bd_item +from template import docs_message, bd_item, bd_item_select from enum import Enum from enum import auto @@ -53,6 +53,7 @@ init_access = f'{user_access.user_access_group_new}=va' def GetAuthorizeItem(a_Bot, a_UserID): items = bd_item.GetBDItemsTemplate(a_Bot, table_name, user_id_field)(a_UserID) + print('GetAuthorizeItem', items) if len(items) == 1: return items[0] return None @@ -274,7 +275,7 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): assert fam_name_field_id != None assert access_field_id != None return \ - a_Item[name_field_id] + ' ' + a_Item[fam_name_field_id] + '(' + a_Item[key_name_id] +')',\ + a_Item[name_field_id] + ' ' + a_Item[fam_name_field_id] + '(' + str(a_Item[key_name_id]) +')',\ a_Item[key_name_id],\ a_Item[access_field_id] @@ -283,8 +284,9 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): user_id = str(a_Message.from_user.id) item = GetAuthorizeItem(self.m_Bot, user_id) user_confirm_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.USER_CONFIRM) + print('GetStartKeyboardButtons', item, user_confirm_field_id) cur_buttons = [] - if item and item[user_confirm_field_id] == str(ConfirmStatus.YES): + if item and item[user_confirm_field_id] == self.GetMessage(mod_table_operate.EnumMessageForView(ConfirmStatus.YES)).GetDesc(): cur_buttons += [ keyboard.ButtonWithAccess(self.GetButton(ButtonNames.LIST_AUTH_DOCS), user_access.AccessMode.VIEW, self.GetAccess()), ] @@ -298,7 +300,8 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): cur_dict = GetReplaceDictFunc(self.m_Bot, a_user_id) result = {} for f in files: - result.update({f, cur_dict}) + result.update({f: cur_dict}) + print(cur_dict) return result button_name = self.GetButton(ButtonNames.LIST_AUTH_DOCS) @@ -323,18 +326,21 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): return super().AddBDItemFunc(a_ItemData, a_UserID) def GetReplaceDictFunc(a_Bot, a_user_id): - item = GetAuthorizeItem(self.m_Bot, user_id) + item = GetAuthorizeItem(a_Bot, a_user_id) if not item: return None result = {} i = 0 - for f in table: - result.update({str(f.m_Destiny), item[i]}) + for f in table.GetFields(): + result.update({str(f.m_Destiny): item[i]}) i += 1 + print(result) return result auth_docs_file_name = 'config_auth_docs' +g_auth_docs = [] + def GetAuthDocs(): global g_auth_docs if len(g_auth_docs) == 0: diff --git a/bot_modules/mod_table_operate.py b/bot_modules/mod_table_operate.py index 2fc8a8d..f826fc8 100644 --- a/bot_modules/mod_table_operate.py +++ b/bot_modules/mod_table_operate.py @@ -406,7 +406,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): ) def GetAddFields(self): - good_destiny = ( + add_destiny = ( bd_table.TableFieldDestiny.NAME, bd_table.TableFieldDestiny.DESC, bd_table.TableFieldDestiny.PHOTO, @@ -422,7 +422,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): ) fields = [] for f in self.m_Table.GetFields(): - if f.m_Destiny in good_destiny: + if f.m_Destiny in add_destiny: fields += [f] return fields diff --git a/template/docs_message.py b/template/docs_message.py index 7544125..e910632 100644 --- a/template/docs_message.py +++ b/template/docs_message.py @@ -6,7 +6,7 @@ from bot_sys import log, config, user_access from bot_modules import groups_utils from template import simple_message -import odf +#import odf def DocFilesTemplate(a_Bot, a_FilesFunc, a_CaptionMessage, a_AccessFunc, a_GetButtonsFunc, a_GetInlineButtonsFunc, a_ErrorMessage, access_mode = user_access.AccessMode.EDIT): async def DocFiles(a_Message): @@ -23,11 +23,11 @@ def DocFilesTemplate(a_Bot, a_FilesFunc, a_CaptionMessage, a_AccessFunc, a_GetBu await simple_message.SendMessage(a_Bot, a_ErrorMessage, a_GetButtonsFunc, None, user_id, a_Message, user_groups) return - for file_path, dict_replace in files.values(): + for file_path, dict_replace in files.items(): if not dict_replace: continue new_file = await MakeDocFile(a_Bot, file_path, dict_replace, user_id) - document = await GetFile(a_Bot, file_path) + document = await GetFile(a_Bot, new_file) if document is None: await simple_message.SendMessage(a_Bot, a_ErrorMessage, a_GetButtonsFunc, None, user_id, a_Message, user_groups) else: @@ -40,20 +40,25 @@ def DocFilesTemplate(a_Bot, a_FilesFunc, a_CaptionMessage, a_AccessFunc, a_GetBu ) return DocFiles +from odf import opendocument, text, teletype + async def MakeDocFile(a_Bot, a_FilePath, a_DictReplace, a_user_id): - document = odf.opendocument.load(a_FilePath) + print ('MakeDocFile', a_FilePath) + #FixBadZipfile(a_FilePath) + document = opendocument.load(a_FilePath) # odf.opendocuement. if document == None: a_Bot.GetLog().Error(f'Не удалось загрузить файл {a_FilePath}.') return None for element in document.getElementsByType(text.Span): - extracted_text = odf.teletype.extractText(element) + extracted_text = teletype.extractText(element) + print('MakeDocFile', extracted_text) - for file_path, dict_replace in a_DictReplace.values(): - if extracted_text.find('Replace this') != -1: - extracted_text = extracted_text.replace('Replace this', 'to this') + for rep_this, to_this in a_DictReplace.items(): + if extracted_text.find(rep_this) != -1: + extracted_text = extracted_text.replace(rep_this, to_this) - new_element = odf.text.Span() + new_element = text.Span() new_element.setAttribute('stylename', element.getAttribute('stylename')) new_element.addText(extracted_text) @@ -61,7 +66,7 @@ async def MakeDocFile(a_Bot, a_FilePath, a_DictReplace, a_user_id): element.parentNode.removeChild(element) new_file_path = a_FilePath[:-4] - new_file_path += f"{a_user_id}.odt" + new_file_path += f"_{a_user_id}.odt" document.save(new_file_path) return new_file_path From 3e6d3a5ec5f1167c6fc48280a3e8d7be2eb09f82 Mon Sep 17 00:00:00 2001 From: Alexei Date: Tue, 31 Oct 2023 15:52:31 +0300 Subject: [PATCH 06/21] =?UTF-8?q?=D0=A0=D0=B0=D0=B1=D0=BE=D1=82=D0=B0=20?= =?UTF-8?q?=D1=81=20=D0=B4=D0=BE=D0=BA=D1=83=D0=BC=D0=B5=D0=BD=D1=82=D0=B0?= =?UTF-8?q?=D0=BC=D0=B8=20=D0=BF=D0=B5=D1=80=D0=B5=D0=B4=D0=B5=D0=BB=D0=B0?= =?UTF-8?q?=D0=BD=D0=B0=20=D0=BD=D0=B0=20=D0=B1=D0=B8=D0=B1=D0=BB=D0=B8?= =?UTF-8?q?=D0=BE=D1=82=D0=B5=D0=BA=D1=83=20weasyprint=20#9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- auth_docs/test_doc.html | 120 +++++++++++++++++++++++++++++++++++++++++++++++ auth_docs/test_doc.odt | Bin 19082 -> 18549 bytes bot_modules/authorize.py | 2 +- config_auth_docs | 2 +- requirements.txt | 2 +- template/docs_message.py | 60 +++++++++++++++--------- 6 files changed, 160 insertions(+), 26 deletions(-) create mode 100644 auth_docs/test_doc.html diff --git a/auth_docs/test_doc.html b/auth_docs/test_doc.html new file mode 100644 index 0000000..3f326bf --- /dev/null +++ b/auth_docs/test_doc.html @@ -0,0 +1,120 @@ + + + + + + + + + + +

+
+ +

+

Регистрационные +данные

+


+ +

+


+ +

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +

+ ID пользователя

+

+ TableFieldDestiny.USER_ID +

+

+ Имя

+

+ TableFieldDestiny.USER_NAME

+

+ Фамилия

+

+ TableFieldDestiny.USER_FAMILY_NAME

+

+ Отчество

+

+ TableFieldDestiny.USER_MIDDLE_NAME

+

+ Дата рождения

+

+ TableFieldDestiny.USER_BIRTHDAY

+

+ Адрес

+

+ TableFieldDestiny.USER_ADDRESS

+

+ Контакты

+

+ TableFieldDestiny.USER_CONTACTS

+

+ Подтверждение

+

+ TableFieldDestiny.USER_CONFIRM

+

+ Доступ

+

+ TableFieldDestiny.ACCESS

+

+ Дата создания записи

+

+ TableFieldDestiny.CREATE_DATE

+
+


+ +

+ + \ No newline at end of file diff --git a/auth_docs/test_doc.odt b/auth_docs/test_doc.odt index e937454b43901dd0f3ed5cac7cd085c9fedf03ee..92fa308fe04a9cd84a49e5d9c1399e8794930fb5 100644 GIT binary patch delta 16620 zcmbum1#})gvLQ+mtD%De=381&ppa}9(;1FmaAkZKnVMaPJ2$JA`l4znB50rmUO#jS? zLllFB`?sobzd(uN@IcZ26vq|3(-~qP|9|CC|0-wutKKhAmcLRgFwQ?|?N=~vP~e?V z1^Kt@{ZT-yj2!i7U9Bu5lt!#q=#kp)zreGqhUDdGFc5v=N94&j_2J~V`5H`7&EykxCU0SW4)MM5Zx4yJ0&05-lIB< z?sRYk=xOc;?hSZA@l?$yzwG|t#A}59>^JY(%;F*}g@0aCcdQ3~JLLgijtE2OeNk=d zr?Ys2RJLEqn4@)Bevlr!HD+ax)P3f2&JCI2c|YZQDD9K`(V))qRi5t4hYqY8dz4t@ zBN)EerR%S=mBnm6w$&8Qyj4^gve8s~>Of{@G3IfdX<#Se|w zqKyngW{D=GBXGE;eep=*yFp*V3{9E|IFmLI<2~12C@)U0G!q_3%x%f?&kx{u_c%icxe@_kMe?>W3q9I+(`H zvB=5>s75fRo##J2JbODfTXU3`$$BWxt^q{d&Uf>$D$SF^nCs9k5{(Pv_T zI@m`-YJ|%>a|_StrpeLC))mlbrtD0rY2INCf0mqr+~plylv9Cy_4M>~R*`;Si{2hO z(#sVY|JV?>{v+88QQwitRN5ck&m&jQG#o5jN_E}1DkCznLX*kK)vlFuDG9GHvKUlanx!Qk`}qzEt=75v=pAf;PG??B z(+^f0LhmD9f*5!BNw@=f@HxY3!q+tML$0K4Y=alof07rJyU{t#<)kHY>zg|2ivmzCY+f%jX*Ci&Z`o2sY48hm;Nv z-K3d6H^%NK5oabVN*<6F;LYzAUKIO?&L(lRztBS;V8i%f0bS)}J-;G9*Yzx16b(mv zo3xzka(zv%_yU-{$xi96;rlv*IIqKfS2W7o~HcN=A0U1=P`tK

NQ#Y}O{ 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 diff --git a/bot_modules/authorize.py b/bot_modules/authorize.py index 1e572d6..9c39865 100644 --- a/bot_modules/authorize.py +++ b/bot_modules/authorize.py @@ -332,7 +332,7 @@ def GetReplaceDictFunc(a_Bot, a_user_id): result = {} i = 0 for f in table.GetFields(): - result.update({str(f.m_Destiny): item[i]}) + result.update({str(f.m_Destiny): str(item[i])}) i += 1 print(result) return result diff --git a/config_auth_docs b/config_auth_docs index 7822470..1524400 100644 --- a/config_auth_docs +++ b/config_auth_docs @@ -1 +1 @@ -auth_docs/test_doc.odt +auth_docs/test_doc.html diff --git a/requirements.txt b/requirements.txt index 9e91223..296c19c 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,3 +1,3 @@ aiogram==2.20 colorama==0.4.5 -odfpy +weasyprint diff --git a/template/docs_message.py b/template/docs_message.py index e910632..3f9288a 100644 --- a/template/docs_message.py +++ b/template/docs_message.py @@ -40,35 +40,49 @@ def DocFilesTemplate(a_Bot, a_FilesFunc, a_CaptionMessage, a_AccessFunc, a_GetBu ) return DocFiles -from odf import opendocument, text, teletype +async def ReplaceInFile(a_Bot, a_InputFileName, a_OutFileName, a_DictReplace): + try: + filedata = '' + print('ReplaceInFile f', a_InputFileName) + with open(a_InputFileName, 'r') as in_f: + filedata = in_f.read() -async def MakeDocFile(a_Bot, a_FilePath, a_DictReplace, a_user_id): - print ('MakeDocFile', a_FilePath) - #FixBadZipfile(a_FilePath) - document = opendocument.load(a_FilePath) # odf.opendocuement. - if document == None: - a_Bot.GetLog().Error(f'Не удалось загрузить файл {a_FilePath}.') - return None + s = filedata + print('ReplaceInFile s', s) + for rep_this, to_this in a_DictReplace.items(): + print('ReplaceInFile', rep_this, to_this) + s = s.replace(rep_this, to_this) - for element in document.getElementsByType(text.Span): - extracted_text = teletype.extractText(element) - print('MakeDocFile', extracted_text) + with open(a_OutFileName, 'w') as out_f: + out_f.write(s) + a_Bot.GetLog().Success(f'Создан файл {a_OutFileName}') + return a_OutFileName + except Exception as e: + a_Bot.GetLog().Error(f'Не удалось заменить текст в фале {a_InputFileName} и записать в {a_OutFileName}. Ошибка {str(e)}') + return None - for rep_this, to_this in a_DictReplace.items(): - if extracted_text.find(rep_this) != -1: - extracted_text = extracted_text.replace(rep_this, to_this) +from weasyprint import HTML, CSS - new_element = text.Span() - new_element.setAttribute('stylename', element.getAttribute('stylename')) - new_element.addText(extracted_text) +async def SaveAsPdf(a_Bot, a_InputFileName, a_OutFileName): + try: + HTML(filename = a_InputFileName).write_pdf(a_OutFileName) + a_Bot.GetLog().Success(f'Создан файл {a_OutFileName}') + return a_OutFileName + except Exception as e: + a_Bot.GetLog().Error(f'Не удалось создать пдф из фала {a_InputFileName} и записать в {a_OutFileName}. Ошибка {str(e)}') + return None - element.parentNode.insertBefore(new_element, element) - element.parentNode.removeChild(element) +async def MakeDocFile(a_Bot, a_FilePath, a_DictReplace, a_user_id): + print ('MakeDocFile', a_FilePath) + user_file_path = a_FilePath[:-5] + pdf_file_path = user_file_path + user_file_path += f"_{a_user_id}.html" + pdf_file_path += f"_{a_user_id}.pdf" + user_file_path = await ReplaceInFile(a_Bot, a_FilePath, user_file_path, a_DictReplace) + if not user_file_path: + return None - new_file_path = a_FilePath[:-4] - new_file_path += f"_{a_user_id}.odt" - document.save(new_file_path) - return new_file_path + return await SaveAsPdf(a_Bot, user_file_path, pdf_file_path) async def GetFile(a_Bot, a_Path): if not a_Path: From 6c67b2e3e641b12c50db0ace3aa36fd937e813dd Mon Sep 17 00:00:00 2001 From: Alexei Date: Sat, 4 Nov 2023 13:08:45 +0300 Subject: [PATCH 07/21] =?UTF-8?q?=D0=90=D0=B2=D1=82=D0=BE=D1=80=D0=B8?= =?UTF-8?q?=D0=B7=D0=B0=D1=86=D0=B8=D1=8F=20=D0=BF=D0=BE=D0=BB=D1=8C=D0=B7?= =?UTF-8?q?=D0=BE=D0=B2=D0=B0=D1=82=D0=B5=D0=BB=D0=B5=D0=B9=201.=20=D0=9A?= =?UTF-8?q?=D0=BD=D0=BE=D0=BF=D0=BA=D0=B0=20=D0=B7=D0=B0=D1=8F=D0=B2=D0=BA?= =?UTF-8?q?=D0=B0=20=D1=83=D0=B1=D0=B8=D1=80=D0=B0=D0=B5=D1=82=D1=81=D1=8F?= =?UTF-8?q?=202.=20=D0=9F=D0=BE=20=D1=83=D0=BC=D0=BE=D0=BB=D1=87=D0=B0?= =?UTF-8?q?=D0=BD=D0=B8=D1=8E=20=D0=B7=D0=B0=D1=8F=D0=B2=D0=BA=D0=B8=20?= =?UTF-8?q?=D0=B4=D1=80=D1=83=D0=B3=D0=B8=D1=85=20=D0=BD=D0=B5=20=D0=B2?= =?UTF-8?q?=D0=B8=D0=B4=D0=BD=D1=8B=203.=20=D0=9D=D0=BE=D0=B2=D0=B0=D1=8F?= =?UTF-8?q?=20=D0=B3=D1=80=D1=83=D0=BF=D0=BF=D0=B0=20=D0=BF=D0=BE=D0=BB?= =?UTF-8?q?=D1=8C=D0=B7=D0=BE=D0=B2=D0=B0=D1=82=D0=B5=D0=BB=D0=B5=D0=B9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/access.py | 2 +- bot_modules/all_orders.py | 2 +- bot_modules/authorize.py | 37 +++++++++++++++++++-------- bot_modules/backup.py | 2 +- bot_modules/bd_version.py | 2 +- bot_modules/buttons.py | 2 +- bot_modules/comments.py | 2 +- bot_modules/groups.py | 4 +-- bot_modules/groups_utils.py | 6 ++++- bot_modules/languages.py | 2 +- bot_modules/messages.py | 2 +- bot_modules/mod_simple_message.py | 5 ++-- bot_modules/mod_table_operate.py | 48 +++++++++++++++++++++++------------ bot_modules/needs.py | 2 +- bot_modules/orders.py | 6 ++--- bot_modules/profile.py | 2 +- bot_modules/projects.py | 2 +- bot_modules/start.py | 2 +- bot_modules/subscribes.py | 6 ++--- bot_modules/tasks.py | 2 +- bot_modules/user_in_groups.py | 2 +- bot_modules/users.py | 2 +- bot_modules/users_groups_agregator.py | 9 ++++++- template/docs_message.py | 7 ++--- 24 files changed, 100 insertions(+), 58 deletions(-) diff --git a/bot_modules/access.py b/bot_modules/access.py index c5b9b6b..da020e4 100644 --- a/bot_modules/access.py +++ b/bot_modules/access.py @@ -106,7 +106,7 @@ init_access = f'{user_access.user_access_group_new}=-' class ModuleAccess(mod_table_operate.TableOperateModule): def __init__(self, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, None, None, init_access, a_ChildModuleNameList, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, None, None, init_access, init_access, a_ChildModuleNameList, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) self.m_SqlRequestButtonName = self.CreateButton('sql request', sql_request_button_name) self.m_RequestStartMessage = self.CreateMessage('equest start', request_start_message) diff --git a/bot_modules/all_orders.py b/bot_modules/all_orders.py index 5efd31f..9f40bb9 100644 --- a/bot_modules/all_orders.py +++ b/bot_modules/all_orders.py @@ -111,7 +111,7 @@ class DBItemForUserSelectSource(bd_item_select.DBItemSelectSource): class ModuleAllOrders(orders.ModuleOrders): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetInitBDCommands(self): # уже сделано в ModuleUserOrders diff --git a/bot_modules/authorize.py b/bot_modules/authorize.py index 9c39865..cc9f9e3 100644 --- a/bot_modules/authorize.py +++ b/bot_modules/authorize.py @@ -4,7 +4,7 @@ # Авторизация или регистрация в кооперативе или другой организации from bot_sys import bot_bd, keyboard, user_access, bd_table, bot_subscribes, config -from bot_modules import mod_table_operate, mod_simple_message +from bot_modules import mod_table_operate, mod_simple_message, users_groups_agregator, groups_utils from template import docs_message, bd_item, bd_item_select from enum import Enum @@ -50,6 +50,8 @@ table = bd_table.Table(table_name, [ ) init_access = f'{user_access.user_access_group_new}=va' +def_init_access = f'{user_access.user_access_group_new}=a' +g_group_auth_name = 'authorize_users' def GetAuthorizeItem(a_Bot, a_UserID): items = bd_item.GetBDItemsTemplate(a_Bot, table_name, user_id_field)(a_UserID) @@ -246,7 +248,7 @@ messages.update(messages_confirm_status) class ModuleAuthorize(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, def_init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name @@ -279,18 +281,23 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): a_Item[key_name_id],\ a_Item[access_field_id] - def GetStartKeyboardButtons(self, a_Message, a_UserGroups): - parent_buttons = super().GetStartKeyboardButtons(a_Message, a_UserGroups) + def GetStartButtons(self, a_Message, a_UserGroups): user_id = str(a_Message.from_user.id) item = GetAuthorizeItem(self.m_Bot, user_id) user_confirm_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.USER_CONFIRM) - print('GetStartKeyboardButtons', item, user_confirm_field_id) + cur_buttons = [] + if item and item[user_confirm_field_id] == self.GetMessage(mod_table_operate.EnumMessageForView(ConfirmStatus.YES)).GetDesc(): - cur_buttons += [ - keyboard.ButtonWithAccess(self.GetButton(ButtonNames.LIST_AUTH_DOCS), user_access.AccessMode.VIEW, self.GetAccess()), - ] - return parent_buttons + keyboard.MakeButtons(self.m_Bot, cur_buttons, a_UserGroups) + cur_buttons += [[ButtonNames.LIST_AUTH_DOCS, user_access.AccessMode.VIEW]] + else: + cur_buttons += [[mod_table_operate.ButtonNames.ADD, user_access.AccessMode.ADD]] + + return cur_buttons + [ + [mod_table_operate.ButtonNames.LIST, user_access.AccessMode.VIEW], + [mod_table_operate.ButtonNames.DEL, user_access.AccessMode.DELETE], + [mod_table_operate.ButtonNames.EDIT, user_access.AccessMode.EDIT], + ] def RegisterHandlers(self): super().RegisterHandlers() @@ -323,7 +330,17 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): def AddBDItemFunc(self, a_ItemData, a_UserID): a_ItemData[user_id_field] = a_UserID #a_ItemData[address_field] = '' - return super().AddBDItemFunc(a_ItemData, a_UserID) + result = super().AddBDItemFunc(a_ItemData, a_UserID) + users_groups_agregator.AddUserInGroup(self.m_Bot, a_UserID, g_group_auth_name) + return result + + def GetKeyFieldDestiny(self): + return bd_table.TableFieldDestiny.USER_ID + + def GetInitBDCommands(self): + return super(). GetInitBDCommands() + [ + groups_utils.CreateGroupRequest(g_group_auth_name) + ] def GetReplaceDictFunc(a_Bot, a_user_id): item = GetAuthorizeItem(a_Bot, a_user_id) diff --git a/bot_modules/backup.py b/bot_modules/backup.py index cf294a2..dc7b989 100644 --- a/bot_modules/backup.py +++ b/bot_modules/backup.py @@ -48,7 +48,7 @@ init_access = f'{user_access.user_access_group_new}=-' class ModuleBackup(mod_simple_message.SimpleMessageModule): def __init__(self, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(messages, button_names, init_access, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(messages, button_names, init_access, init_access, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) self.m_BackupBDButtonName = self.CreateButton('backup bd', backup_bd_button_name) self.m_BackupBDMessage = self.CreateMessage('backup bd', backup_bd_message) diff --git a/bot_modules/bd_version.py b/bot_modules/bd_version.py index d74dfd9..081a075 100644 --- a/bot_modules/bd_version.py +++ b/bot_modules/bd_version.py @@ -40,7 +40,7 @@ messages = { class ModuleBDVersion(mod_table_operate.TableOperateModule): def __init__(self, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, None, None, init_access, None, None, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, None, None, init_access, init_access, None, None, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/buttons.py b/bot_modules/buttons.py index 63e77fa..09dbfda 100644 --- a/bot_modules/buttons.py +++ b/bot_modules/buttons.py @@ -128,7 +128,7 @@ messages = { class ModuleButtons(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/comments.py b/bot_modules/comments.py index 953169f..b5c8457 100644 --- a/bot_modules/comments.py +++ b/bot_modules/comments.py @@ -130,7 +130,7 @@ messages.update(messages_subscribes) class ModuleComments(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/groups.py b/bot_modules/groups.py index c82485c..4e10654 100644 --- a/bot_modules/groups.py +++ b/bot_modules/groups.py @@ -125,12 +125,12 @@ messages = { class ModuleGroups(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name def GetInitBDCommands(self): return super(). GetInitBDCommands() + [ - f"INSERT OR IGNORE INTO {table_name} ({name_field}, {access_field}, {create_datetime_field}) VALUES ('{user_access.user_access_group_new}', '{user_access.user_access_group_new}=-', {bot_bd.GetBDDateTimeNow()});" + groups_utils.CreateGroupRequest(user_access.user_access_group_new) ] diff --git a/bot_modules/groups_utils.py b/bot_modules/groups_utils.py index 1bc67a0..27b2bbe 100644 --- a/bot_modules/groups_utils.py +++ b/bot_modules/groups_utils.py @@ -3,7 +3,7 @@ # Группы пользователей -from bot_sys import user_access +from bot_sys import user_access, bot_bd table_name = 'user_groups' key_name = 'groupID' @@ -28,3 +28,7 @@ def GetUserGroupData(a_Bot, a_UserID): if len(i) > 0: groups += [i[0]] return user_access.UserGroups(a_UserID, groups) + +def CreateGroupRequest(a_GroupName): + return f"INSERT OR IGNORE INTO {table_name} ({name_field}, {access_field}, {create_datetime_field}) VALUES ('{a_GroupName}', '{user_access.user_access_group_new}=-', {bot_bd.GetBDDateTimeNow()});" + diff --git a/bot_modules/languages.py b/bot_modules/languages.py index 06d0835..b3022fb 100644 --- a/bot_modules/languages.py +++ b/bot_modules/languages.py @@ -126,7 +126,7 @@ messages = { class ModuleLanguages(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) self.m_LanguageIDs = {} def GetName(self): diff --git a/bot_modules/messages.py b/bot_modules/messages.py index 310b043..27ca51f 100644 --- a/bot_modules/messages.py +++ b/bot_modules/messages.py @@ -128,7 +128,7 @@ messages = { class ModuleMessages(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/mod_simple_message.py b/bot_modules/mod_simple_message.py index a533a9d..8f5e9a3 100644 --- a/bot_modules/mod_simple_message.py +++ b/bot_modules/mod_simple_message.py @@ -17,9 +17,10 @@ class Messages(Enum): START = auto() class SimpleMessageModule(mod_interface.IModule): - def __init__(self, a_Messages, a_Buttons, a_InitAccess, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): + def __init__(self, a_Messages, a_Buttons, a_InitAccess, a_DefInitAccess, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): self.m_ChildModuleNameList = a_ChildModuleNameList self.m_InitAccess = a_InitAccess + self.m_DefInitAccess = a_DefInitAccess self.m_Bot = a_Bot self.m_ModuleAgregator = a_ModuleAgregator self.m_BotMessages = a_BotMessages @@ -99,7 +100,7 @@ class SimpleMessageModule(mod_interface.IModule): def GetInitBDCommands(self): return [ - access_utils.GetAccessForModuleRequest(self.GetName(), self.m_InitAccess, self.m_InitAccess), + access_utils.GetAccessForModuleRequest(self.GetName(), self.m_InitAccess, self.m_DefInitAccess), ] def GetAccess(self): diff --git a/bot_modules/mod_table_operate.py b/bot_modules/mod_table_operate.py index f826fc8..5c257c4 100644 --- a/bot_modules/mod_table_operate.py +++ b/bot_modules/mod_table_operate.py @@ -83,8 +83,8 @@ def MakeFSMForEdit(a_ModName, a_FieldName): return _locals['fsm'] class TableOperateModule(mod_simple_message.SimpleMessageModule): - def __init__(self, a_Table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(a_Messages, a_Buttons, a_InitAccess, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + def __init__(self, a_Table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_DefInitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): + super().__init__(a_Messages, a_Buttons, a_InitAccess, a_DefInitAccess, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) self.m_Table = a_Table self.m_EditModuleNameList = a_EditModuleNameList self.m_ChildModName = a_ChildModName @@ -99,8 +99,8 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): return self.GetButtonNameAndKeyValueAndAccess(a_Item) self.m_GetButtonNameAndKeyValueAndAccessFunc = GetButtonNameAndKeyValueAndAccess - async def PreDelete(a_CallbackQuery, a_Item): - return await self.PreDelete(a_CallbackQuery, a_Item) + async def PreDelete(a_CallbackQuery, a_Item, a_None): + return await self.PreDelete(a_CallbackQuery, a_Item, a_None) self.m_PreDeleteFunc = PreDelete async def PostDelete(a_CallbackQuery, a_ItemID): @@ -116,14 +116,21 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): self.m_Table.GetInitTableRequest(), ] + super().GetInitBDCommands() + def GetStartButtons(self, a_Message, a_UserGroups): + return [ + [ButtonNames.LIST, user_access.AccessMode.VIEW], + [ButtonNames.ADD, user_access.AccessMode.ADD], + [ButtonNames.DEL, user_access.AccessMode.DELETE], + [ButtonNames.EDIT, user_access.AccessMode.EDIT], + ] + def GetStartKeyboardButtons(self, a_Message, a_UserGroups): mod_buttons = super().GetStartKeyboardButtons(a_Message, a_UserGroups) - cur_buttons = [ - keyboard.ButtonWithAccess(self.GetButton(ButtonNames.LIST), user_access.AccessMode.VIEW, self.GetAccess()), - keyboard.ButtonWithAccess(self.GetButton(ButtonNames.ADD), user_access.AccessMode.ADD, self.GetAccess()), - keyboard.ButtonWithAccess(self.GetButton(ButtonNames.DEL), user_access.AccessMode.DELETE, self.GetAccess()), - keyboard.ButtonWithAccess(self.GetButton(ButtonNames.EDIT), user_access.AccessMode.EDIT, self.GetAccess()), - ] + cur_buttons = [] + for b in self.GetStartButtons(a_Message, a_UserGroups): + bn = b[0] + ba = b[1] + cur_buttons += [keyboard.ButtonWithAccess(self.GetButton(bn), ba, self.GetAccess())] return mod_buttons + keyboard.MakeButtons(self.m_Bot, cur_buttons, a_UserGroups) def GetAccessForEditKeyboardButtons(self, a_Field): @@ -157,7 +164,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): return keyboard.MakeInlineKeyboardButtons(self.m_Bot, cur_buttons, a_UserGroups) def GetButtonNameAndKeyValueAndAccess(self, a_Item): - key_name_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.KEY) + key_name_id = self.GetKeyFieldID() name_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.NAME) access_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.ACCESS) assert key_name_id != None @@ -220,12 +227,12 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): msg = self.UpdateMessage(msg, lang, a_Item, a_EnablePhoto = a_EnablePhoto) item_access = a_Item[self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.ACCESS)] if Inline_keyboard_template_func: - Inline_keyboard_func = Inline_keyboard_template_func(a_Item[self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.KEY)]) + Inline_keyboard_func = Inline_keyboard_template_func(a_Item[self.GetKeyFieldID()]) return simple_message.WorkFuncResult(msg, item_access = item_access, Inline_keyboard_func = Inline_keyboard_func) return ShowMessage - async def PreDelete(self, a_CallbackQuery, a_Item): + async def PreDelete(self, a_CallbackQuery, a_Item, a_None): if len(a_Item) < self.m_Table.GetFieldsCount(): return simple_message.WorkFuncResult(self.GetMessage(Messages.ERROR_FIND)) access = a_Item[self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.ACCESS)] @@ -379,7 +386,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): return self.OnChange() table_name = self.m_Table.GetName() - key_name = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.KEY) + key_name = self.GetKeyFieldName() edit_keyboard_func = self.m_GetEditKeyboardButtonsFunc GetButtonNameAndKeyValueAndAccess = self.m_GetButtonNameAndKeyValueAndAccessFunc GetAccess = self.m_GetAccessFunc @@ -469,10 +476,19 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): f = fields[f_id] self.m_Bot.RegisterMessageHandler(bd_item_add.FinishAddBDItemTemplate(self.m_Bot, fsm, a_AddBDItemFunc, a_ParentTableName, a_ParentKeyFieldName, f.m_Name, self.ShowMessageTemplate(self.GetMessage(Messages.SUCCESS_CREATE)), None, a_AccessFunc, a_ButtonFunc, access_mode, field_type = GetFieldType(f)), content_types = GetContentTypes(f), state = getattr(fsm, f'item{f_id}')) + def GetKeyFieldDestiny(self): + return bd_table.TableFieldDestiny.KEY + + def GetKeyFieldName(self): + return self.m_Table.GetFieldNameByDestiny(self.GetKeyFieldDestiny()) + + def GetKeyFieldID(self): + return self.m_Table.GetFieldIDByDestiny(self.GetKeyFieldDestiny()) + def RegisterHandlers(self): super().RegisterHandlers() table_name = self.m_Table.GetName() - key_name = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.KEY) + key_name = self.GetKeyFieldName() name_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.NAME) desc_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.DESC) photo_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.PHOTO) @@ -485,7 +501,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): if self.m_ParentModName: parent_mod = self.GetModule(self.m_ParentModName) parent_table_name = parent_mod.m_Table.GetName() - parent_key_name = parent_mod.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.KEY) + parent_key_name = parent_mod.GetKeyFieldName() def GetViewItemInlineKeyboardTemplate(a_ItemID): return self.GetViewItemInlineKeyboardTemplate(a_ItemID) diff --git a/bot_modules/needs.py b/bot_modules/needs.py index 0f0bf2f..ace1d88 100644 --- a/bot_modules/needs.py +++ b/bot_modules/needs.py @@ -131,7 +131,7 @@ messages.update(messages_subscribes) class ModuleNeeds(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/orders.py b/bot_modules/orders.py index 3559006..edf6fd0 100644 --- a/bot_modules/orders.py +++ b/bot_modules/orders.py @@ -200,8 +200,8 @@ class DBItemForUserSelectSource(bd_item_select.DBItemSelectSource): return True class ModuleOrders(mod_table_operate.TableOperateModule): - def __init__(self, a_Table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + def __init__(self, a_Table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_DefInitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): + super().__init__(table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_DefInitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def SelectSourceTemplate(self, a_PrevPrefix, a_ButtonName): parent_id_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.PARENT_ID) @@ -262,7 +262,7 @@ class ModuleOrders(mod_table_operate.TableOperateModule): class ModuleUserOrders(ModuleOrders): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/profile.py b/bot_modules/profile.py index f0c2a67..c584a07 100644 --- a/bot_modules/profile.py +++ b/bot_modules/profile.py @@ -32,7 +32,7 @@ init_access = f'{user_access.user_access_group_new}=+' class ModuleProfile(mod_simple_message.SimpleMessageModule): def __init__(self, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(messages, button_names, init_access, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(messages, button_names, init_access, init_access, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/projects.py b/bot_modules/projects.py index becbc93..a3fc0c0 100644 --- a/bot_modules/projects.py +++ b/bot_modules/projects.py @@ -129,7 +129,7 @@ messages.update(messages_subscribes) class ModuleProjects(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/start.py b/bot_modules/start.py index dc55776..58d3c63 100644 --- a/bot_modules/start.py +++ b/bot_modules/start.py @@ -24,7 +24,7 @@ init_access = f'{user_access.user_access_group_all}=+' class ModuleStart(mod_simple_message.SimpleMessageModule): def __init__(self, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(messages, button_names, init_access, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(messages, button_names, init_access, init_access, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/subscribes.py b/bot_modules/subscribes.py index a807b8d..6c4e663 100644 --- a/bot_modules/subscribes.py +++ b/bot_modules/subscribes.py @@ -171,8 +171,8 @@ class DBItemForUserSelectSource(bd_item_select.DBItemSelectSource): return True class ModuleSubscribe(mod_table_operate.TableOperateModule): - def __init__(self, a_Table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + def __init__(self, a_Table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_DefInitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): + super().__init__(table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_DefInitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def SelectSourceTemplate(self, a_PrevPrefix, a_ButtonName): parent_id_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.PARENT_ID) @@ -205,7 +205,7 @@ class ModuleSubscribe(mod_table_operate.TableOperateModule): class ModuleUserSubscribe(ModuleSubscribe): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) self.UpdateSubscribes() def GetName(self): diff --git a/bot_modules/tasks.py b/bot_modules/tasks.py index 8e148b7..1523adb 100644 --- a/bot_modules/tasks.py +++ b/bot_modules/tasks.py @@ -131,7 +131,7 @@ messages.update(messages_subscribes) class ModuleTasks(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/user_in_groups.py b/bot_modules/user_in_groups.py index f2e7830..3509cad 100644 --- a/bot_modules/user_in_groups.py +++ b/bot_modules/user_in_groups.py @@ -106,7 +106,7 @@ messages = { class ModuleUserInGroups(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/users.py b/bot_modules/users.py index e4feaee..ba92907 100644 --- a/bot_modules/users.py +++ b/bot_modules/users.py @@ -128,7 +128,7 @@ messages = { class ModuleUsers(mod_table_operate.TableOperateModule): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) def GetName(self): return module_name diff --git a/bot_modules/users_groups_agregator.py b/bot_modules/users_groups_agregator.py index 4c4533a..7c29fe7 100644 --- a/bot_modules/users_groups_agregator.py +++ b/bot_modules/users_groups_agregator.py @@ -57,7 +57,7 @@ init_access = f'{user_access.user_access_group_new}=-' class ModuleUsersGroupsAgregator(mod_simple_message.SimpleMessageModule): def __init__(self, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): - super().__init__(messages, button_names, init_access, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + super().__init__(messages, button_names, init_access, init_access, a_ChildModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) self.m_SqlRequestButtonName = self.CreateButton('sql request', sql_request_button_name) self.m_RequestStartMessage = self.CreateMessage('equest start', request_start_message) @@ -99,3 +99,10 @@ class ModuleUsersGroupsAgregator(mod_simple_message.SimpleMessageModule): bd_item.GetCheckForTextFunc(self.m_HelpButtonName) ) +def AddUserInGroup(a_Bot, a_UserID, a_GroupName): + g_id = a_Bot.SQLRequest(f'SELECT {groups.key_name} FROM {groups.table_name} WHERE {groups.name_field} = "{a_GroupName}"'); + print ('AddUserInGroup', g_id) + if not g_id or len(g_id) != 1 or len(g_id[0]) != 1: + return False + group_id = g_id[0][0] + a_Bot.SQLRequest(f'INSERT INTO {user_in_groups.table_name}({user_in_groups.name_field}, {user_in_groups.parent_id_field}) VALUES({a_UserID}, {group_id})') diff --git a/template/docs_message.py b/template/docs_message.py index 3f9288a..9354644 100644 --- a/template/docs_message.py +++ b/template/docs_message.py @@ -43,14 +43,12 @@ def DocFilesTemplate(a_Bot, a_FilesFunc, a_CaptionMessage, a_AccessFunc, a_GetBu async def ReplaceInFile(a_Bot, a_InputFileName, a_OutFileName, a_DictReplace): try: filedata = '' - print('ReplaceInFile f', a_InputFileName) + with open(a_InputFileName, 'r') as in_f: filedata = in_f.read() - s = filedata - print('ReplaceInFile s', s) + for rep_this, to_this in a_DictReplace.items(): - print('ReplaceInFile', rep_this, to_this) s = s.replace(rep_this, to_this) with open(a_OutFileName, 'w') as out_f: @@ -73,7 +71,6 @@ async def SaveAsPdf(a_Bot, a_InputFileName, a_OutFileName): return None async def MakeDocFile(a_Bot, a_FilePath, a_DictReplace, a_user_id): - print ('MakeDocFile', a_FilePath) user_file_path = a_FilePath[:-5] pdf_file_path = user_file_path user_file_path += f"_{a_user_id}.html" From 6c78244efae7680889eeb13a11caf9e8c84648be Mon Sep 17 00:00:00 2001 From: Alexei Date: Sat, 4 Nov 2023 13:31:56 +0300 Subject: [PATCH 08/21] =?UTF-8?q?=D0=94=D0=BE=D0=B1=D0=B0=D0=B2=D0=BB?= =?UTF-8?q?=D0=B5=D0=BD=D0=B8=D0=B5=20=D0=BF=D0=BE=D0=BB=D1=8C=D0=B7=D0=BE?= =?UTF-8?q?=D0=B2=D0=B0=D1=82=D0=B5=D0=BB=D1=8F=20=D0=B2=20=D0=B3=D1=80?= =?UTF-8?q?=D1=83=D0=BF=D0=BF=D1=83=20=D0=BF=D0=BE=D1=81=D0=BB=D0=B5=20?= =?UTF-8?q?=D0=B0=D0=B2=D1=82=D0=BE=D1=80=D0=B8=D0=B7=D0=B0=D1=86=D0=B8?= =?UTF-8?q?=D0=B8=20#9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/users_groups_agregator.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/bot_modules/users_groups_agregator.py b/bot_modules/users_groups_agregator.py index 7c29fe7..e26b55d 100644 --- a/bot_modules/users_groups_agregator.py +++ b/bot_modules/users_groups_agregator.py @@ -101,8 +101,10 @@ class ModuleUsersGroupsAgregator(mod_simple_message.SimpleMessageModule): def AddUserInGroup(a_Bot, a_UserID, a_GroupName): g_id = a_Bot.SQLRequest(f'SELECT {groups.key_name} FROM {groups.table_name} WHERE {groups.name_field} = "{a_GroupName}"'); - print ('AddUserInGroup', g_id) if not g_id or len(g_id) != 1 or len(g_id[0]) != 1: return False group_id = g_id[0][0] - a_Bot.SQLRequest(f'INSERT INTO {user_in_groups.table_name}({user_in_groups.name_field}, {user_in_groups.parent_id_field}) VALUES({a_UserID}, {group_id})') + r = f'INSERT OR IGNORE INTO {user_in_groups.table_name}({user_in_groups.name_field}, {user_in_groups.parent_id_field}) VALUES(?, ?)' + result = a_Bot.SQLRequest(r, commit=True, param = (a_UserID, group_id)) + + From 423acd9fadae093f6bf8b4749f9a6c58453711b9 Mon Sep 17 00:00:00 2001 From: Alexei Date: Sat, 4 Nov 2023 14:46:45 +0300 Subject: [PATCH 09/21] =?UTF-8?q?=D0=90=D0=B2=D1=82=D0=BE=D1=80=D0=B8?= =?UTF-8?q?=D0=B7=D0=B0=D1=86=D0=B8=D1=8F=20=D0=BF=D0=BE=D0=BB=D1=8C=D0=B7?= =?UTF-8?q?=D0=BE=D0=B2=D0=B0=D1=82=D0=B5=D0=BB=D0=B5=D0=B9=20#9=201.=20?= =?UTF-8?q?=D0=97=D0=B0=D0=B3=D1=80=D1=83=D0=B7=D0=BA=D0=B0=20=D0=B4=D0=BE?= =?UTF-8?q?=D0=BA=D1=83=D0=BC=D0=B5=D0=BD=D1=82=D0=BE=D0=B2=20=D0=B8=20?= =?UTF-8?q?=D1=87=D0=B5=D0=BA=D0=B0=20=D0=BE=D0=B1=20=D0=BE=D0=BF=D0=BB?= =?UTF-8?q?=D0=B0=D1=82=D0=B5=20=D0=A7=D0=92=202.=20=D0=97=D0=B0=D0=BA?= =?UTF-8?q?=D0=B0=D0=B7=D1=8B=20=D0=B4=D0=BE=D1=81=D1=82=D1=83=D0=BF=D0=BD?= =?UTF-8?q?=D1=8B=20=D1=82=D0=BE=D0=BB=D1=8C=D0=BA=D0=BE=20=D0=B4=D0=BB?= =?UTF-8?q?=D1=8F=20=D0=B0=D0=B2=D1=82=D0=BE=D1=80=D0=B8=D0=B7=D0=BE=D0=B2?= =?UTF-8?q?=D0=B0=D0=BD=D0=BD=D1=8B=D1=85?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/authorize.py | 42 ++++++++++++++++++++++++++++++++++-------- bot_modules/orders.py | 2 +- bot_sys/bd_table.py | 1 + bot_sys/user_access.py | 2 ++ 4 files changed, 38 insertions(+), 9 deletions(-) diff --git a/bot_modules/authorize.py b/bot_modules/authorize.py index cc9f9e3..dced8b2 100644 --- a/bot_modules/authorize.py +++ b/bot_modules/authorize.py @@ -22,6 +22,8 @@ user_birthday_field = 'userBirthday' user_address_field = 'userAddress' user_contacts_field = 'userContacts' user_confirm_field = 'userConfirm' +user_auth_docs_field = 'authDocs' +user_photo_pay_field = 'photoPay' access_field = 'authorizeAccess' create_datetime_field = 'authorizeCreateDateTime' @@ -40,6 +42,8 @@ table = bd_table.Table(table_name, [ bd_table.TableField(user_address_field, bd_table.TableFieldDestiny.USER_ADDRESS, bd_table.TableFieldType.STR), bd_table.TableField(user_contacts_field, bd_table.TableFieldDestiny.USER_CONTACTS, bd_table.TableFieldType.STR), bd_table.TableField(user_confirm_field, bd_table.TableFieldDestiny.USER_CONFIRM, bd_table.TableFieldType.ENUM, a_Enum = ConfirmStatus), + bd_table.TableField(user_auth_docs_field, bd_table.TableFieldDestiny.AUTH_PHOTO_DOCS, bd_table.TableFieldType.PHOTO), + bd_table.TableField(user_photo_pay_field, bd_table.TableFieldDestiny.PHOTO_PAY, bd_table.TableFieldType.PHOTO), bd_table.TableField(access_field, bd_table.TableFieldDestiny.ACCESS, bd_table.TableFieldType.STR), bd_table.TableField(create_datetime_field, bd_table.TableFieldDestiny.CREATE_DATE, bd_table.TableFieldType.STR), ] @@ -49,13 +53,11 @@ table = bd_table.Table(table_name, [ ] ) -init_access = f'{user_access.user_access_group_new}=va' +init_access = f'{user_access.user_access_group_new}=vea' def_init_access = f'{user_access.user_access_group_new}=a' -g_group_auth_name = 'authorize_users' def GetAuthorizeItem(a_Bot, a_UserID): items = bd_item.GetBDItemsTemplate(a_Bot, table_name, user_id_field)(a_UserID) - print('GetAuthorizeItem', items) if len(items) == 1: return items[0] return None @@ -82,6 +84,9 @@ button_names = { mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_CONFIRM): "☐ Изменить подтверждение пользователя", mod_table_operate.EditButton(bd_table.TableFieldDestiny.USER_ID): "☐ Изменить id пользователя", mod_table_operate.EditButton(bd_table.TableFieldDestiny.ACCESS): "✋ Доступ к авторизации пользователя", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.AUTH_PHOTO_DOCS): "☐ Загрузить подписанные документы", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.PHOTO_PAY): "☐ Оплатить членский взнос", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.ACCESS): "✋ Доступ к авторизации пользователя", mod_table_operate.ButtonNames.DEL: "❌ Удалить авторизацию пользователя", mod_table_operate.EnumButton(ConfirmStatus.YES): "Да, все данные верны", mod_table_operate.EnumButton(ConfirmStatus.NO): "Нет, данные не верны", @@ -214,6 +219,12 @@ messages = { Введите новое подтверждение пользователя: ''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.AUTH_PHOTO_DOCS): f''' +Загрузите подписанный документ: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.PHOTO_PAY): f''' +Загрузите чек по оплате ЧВ: +''', mod_table_operate.EditMessage(bd_table.TableFieldDestiny.ACCESS): f''' Текущий доступ к проекту: #{access_field} @@ -222,7 +233,7 @@ messages = { Введите новую строку доступа: ''', - mod_table_operate.Messages.SUCCESS_EDIT: '''✅ Проект успешно отредактирован!''', + mod_table_operate.Messages.SUCCESS_EDIT: '''✅ Данные успешно отредактированы!''', mod_table_operate.Messages.SELECT_TO_DELETE: ''' Выберите пользователя, у которого вы хотите удалить авторизованные данные. ''', @@ -264,6 +275,8 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): bd_table.TableFieldDestiny.USER_CONTACTS: user_access.AccessMode.NONE, bd_table.TableFieldDestiny.USER_CONFIRM: user_access.AccessMode.NONE, bd_table.TableFieldDestiny.ACCESS: user_access.AccessMode.NONE, + bd_table.TableFieldDestiny.AUTH_PHOTO_DOCS: user_access.AccessMode.ADD, + bd_table.TableFieldDestiny.PHOTO_PAY: user_access.AccessMode.ADD, } return cur_dict.get(a_Field.m_Destiny, super().GetAccessForEditKeyboardButtons(a_Field)) @@ -281,6 +294,17 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): a_Item[key_name_id],\ a_Item[access_field_id] + async def OnChangeField(self, a_Field, a_ItemID, a_ItemData, a_EditUserID): + super().OnChangeField(a_Field, a_ItemID, a_ItemData, a_EditUserID) + user_id_field_name = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.USER_ID) + user_id = a_ItemData[user_id_field_name] + user_confirm_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.USER_CONFIRM) + auth_docs_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.AUTH_PHOTO_DOCS) + pay_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.PHOTO_PAY) + item = GetAuthorizeItem(self.m_Bot, user_id) + if item and item[user_confirm_field_id] != '' and item[auth_docs_field_id] != '' and item[pay_field_id] != '': + users_groups_agregator.AddUserInGroup(self.m_Bot, user_id, user_access.user_access_group_auth_users) + def GetStartButtons(self, a_Message, a_UserGroups): user_id = str(a_Message.from_user.id) item = GetAuthorizeItem(self.m_Bot, user_id) @@ -290,6 +314,8 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): if item and item[user_confirm_field_id] == self.GetMessage(mod_table_operate.EnumMessageForView(ConfirmStatus.YES)).GetDesc(): cur_buttons += [[ButtonNames.LIST_AUTH_DOCS, user_access.AccessMode.VIEW]] + cur_buttons += [[mod_table_operate.EditButton(bd_table.TableFieldDestiny.AUTH_PHOTO_DOCS), user_access.AccessMode.VIEW]] + cur_buttons += [[mod_table_operate.EditButton(bd_table.TableFieldDestiny.PHOTO_PAY), user_access.AccessMode.VIEW]] else: cur_buttons += [[mod_table_operate.ButtonNames.ADD, user_access.AccessMode.ADD]] @@ -308,7 +334,7 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): result = {} for f in files: result.update({f: cur_dict}) - print(cur_dict) + return result button_name = self.GetButton(ButtonNames.LIST_AUTH_DOCS) @@ -329,9 +355,9 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): def AddBDItemFunc(self, a_ItemData, a_UserID): a_ItemData[user_id_field] = a_UserID - #a_ItemData[address_field] = '' + a_ItemData[user_auth_docs_field] = '' + a_ItemData[user_photo_pay_field] = '' result = super().AddBDItemFunc(a_ItemData, a_UserID) - users_groups_agregator.AddUserInGroup(self.m_Bot, a_UserID, g_group_auth_name) return result def GetKeyFieldDestiny(self): @@ -339,7 +365,7 @@ class ModuleAuthorize(mod_table_operate.TableOperateModule): def GetInitBDCommands(self): return super(). GetInitBDCommands() + [ - groups_utils.CreateGroupRequest(g_group_auth_name) + groups_utils.CreateGroupRequest(user_access.user_access_group_auth_users) ] def GetReplaceDictFunc(a_Bot, a_user_id): diff --git a/bot_modules/orders.py b/bot_modules/orders.py index edf6fd0..1c4a973 100644 --- a/bot_modules/orders.py +++ b/bot_modules/orders.py @@ -45,7 +45,7 @@ table = bd_table.Table(table_name, [ bd_table.TableField(parent_id_field, bd_table.TableFieldDestiny.PARENT_ID, bd_table.TableFieldType.INT), ]) -init_access = f'{user_access.user_access_group_new}=vea' +init_access = f'{user_access.user_access_group_auth_users}=vea' # --------------------------------------------------------- # Сообщения и кнопки diff --git a/bot_sys/bd_table.py b/bot_sys/bd_table.py index 2af2e98..25c2981 100644 --- a/bot_sys/bd_table.py +++ b/bot_sys/bd_table.py @@ -46,6 +46,7 @@ class TableFieldDestiny(Enum): USER_ADDRESS = auto() # Адресс USER_CONTACTS = auto() # контакты USER_CONFIRM = auto() # подтверждение + AUTH_PHOTO_DOCS = auto() class TableField: def __init__(self, a_Name, a_Destiny : TableFieldDestiny, a_Type : TableFieldType, a_Enum = None): diff --git a/bot_sys/user_access.py b/bot_sys/user_access.py index 29750ca..39394cd 100644 --- a/bot_sys/user_access.py +++ b/bot_sys/user_access.py @@ -8,6 +8,7 @@ from bot_sys import config user_access_group_all = 'all' user_access_group_new = 'new' +user_access_group_auth_users = 'authorize_users' user_access_readme = f''' Доступ к пользователям задаётся в виде строки @@ -57,6 +58,7 @@ def CheckAccessItem(a_AccessItem : str, a_AccessMode : AccessMode): # Возвращает возможность доступа пользователю a_UserGroups в элемент с правами a_AccessValue по режиму доступа a_AccessMode def CheckAccess(a_RootIDs, a_AccessValue : str, a_UserGroups : UserGroups, a_AccessMode : AccessMode): + #print(a_RootIDs, a_AccessValue, a_UserGroups, a_AccessMode) if a_UserGroups.user_id in a_RootIDs: return True for i in a_AccessValue.split(';'): From 085ac9b8c8db5d233621c07b1023938b97aaabfa Mon Sep 17 00:00:00 2001 From: Alexei Date: Sun, 5 Nov 2023 17:06:36 +0300 Subject: [PATCH 10/21] =?UTF-8?q?=D0=9A=D0=B0=D1=82=D0=B5=D0=B3=D0=BE?= =?UTF-8?q?=D1=80=D0=B8=D0=B8=20=D0=B4=D0=BB=D1=8F=20=D0=B7=D0=B0=D0=BA?= =?UTF-8?q?=D0=B0=D0=B7=D0=BE=D0=B2=20#11?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/all_orders.py | 20 +++--- bot_modules/mod_table_operate.py | 5 +- bot_modules/orders.py | 27 ++++---- bot_modules/orders_cat.py | 135 +++++++++++++++++++++++++++++++++++++++ main.py | 13 ++-- template/bd_item.py | 2 +- 6 files changed, 171 insertions(+), 31 deletions(-) create mode 100644 bot_modules/orders_cat.py diff --git a/bot_modules/all_orders.py b/bot_modules/all_orders.py index 9f40bb9..efdbc2d 100644 --- a/bot_modules/all_orders.py +++ b/bot_modules/all_orders.py @@ -62,7 +62,7 @@ cur_messages = { Статус: #{orders.status_field} -Пользователь: #{orders.parent_id_field} +Пользователь: #{orders.user_id_field} Адрес доставки: #{orders.address_field} @@ -85,15 +85,16 @@ messages.update(orders.messages_order_status) messages.update(messages_notification) messages.update(cur_messages) -def GetCurItemsTemplate(a_Bot, a_TableName, a_StatusFieldName): +def GetCurItemsTemplate(a_Bot, a_TableName, a_StatusFieldName, a_ParentIDFieldName): def GetBDItems(a_Message, a_UserGroups, a_ParentID): - request = f'SELECT * FROM {a_TableName} WHERE {a_StatusFieldName} != ?' - return a_Bot.SQLRequest(request, param = ([str(orders.OrderStatus.FINISH)])) + request = f'SELECT * FROM {a_TableName} WHERE {a_ParentIDFieldName} = ? AND {a_StatusFieldName} != ?' + return a_Bot.SQLRequest(request, param = ([a_ParentID, str(orders.OrderStatus.FINISH)])) return GetBDItems -def GetBDItemsForUserTemplate(a_Bot, a_TableName): +def GetBDItemsForUserTemplate(a_Bot, a_TableName, a_ParentIDFieldName): def GetBDItems(a_Message, a_UserGroups, a_ParentID): - return bd_item.GetAllItemsTemplate(a_Bot, a_TableName)() + request = f'SELECT * FROM {a_TableName} WHERE {a_ParentIDFieldName} = ?' + return a_Bot.SQLRequest(request, param = ([a_ParentID])) return GetBDItems class DBItemForUserSelectSource(bd_item_select.DBItemSelectSource): @@ -103,11 +104,8 @@ class DBItemForUserSelectSource(bd_item_select.DBItemSelectSource): def GetItemsFunc(self): if self.m_OnlyCurrent: - return GetCurItemsTemplate(self.m_Bot, self.m_TableName, orders.status_field) - return GetBDItemsForUserTemplate(self.m_Bot, self.m_TableName) - - def IsFirst(self): - return True + return GetCurItemsTemplate(self.m_Bot, self.m_TableName, orders.status_field, self.m_ParentIDFieldName) + return GetBDItemsForUserTemplate(self.m_Bot, self.m_TableName, self.m_ParentIDFieldName) class ModuleAllOrders(orders.ModuleOrders): def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): diff --git a/bot_modules/mod_table_operate.py b/bot_modules/mod_table_operate.py index 5c257c4..6e3505b 100644 --- a/bot_modules/mod_table_operate.py +++ b/bot_modules/mod_table_operate.py @@ -209,7 +209,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): return a_Msg def ShowMessageTemplate(self, a_Message, Inline_keyboard_template_func = None, a_EnablePhoto = False): - async def ShowMessage(a_CallbackQuery, a_Item, a_ItemDict): + async def ShowMessage(a_CallbackQuery, a_Item, a_ItemDict, table_name = self.m_Table.GetName()): msg = a_Message.StaticCopy() # TODO: добавить поддержку языков в a_MessageName Inline_keyboard_func = None @@ -218,7 +218,8 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): lang = str(a_CallbackQuery.from_user.language_code) msg = msg.GetMessageForLang(lang).StaticCopy() msg = self.UpdateMessageByDict(msg, lang, a_ItemDict, a_EnablePhoto = a_EnablePhoto) - if a_Item: + + if a_Item and self.m_Table.GetName() == table_name: if len(a_Item) < self.m_Table.GetFieldsCount() - 1: # Для проектов это нужно. Там на 1 меньше поле. TODO разделить отправку сообщений item_access и Inline_keyboard_func return simple_message.WorkFuncResult(self.GetMessage(Messages.ERROR_FIND)) elif len(a_Item) == self.m_Table.GetFieldsCount(): diff --git a/bot_modules/orders.py b/bot_modules/orders.py index 1c4a973..be1e655 100644 --- a/bot_modules/orders.py +++ b/bot_modules/orders.py @@ -22,6 +22,7 @@ module_name = 'orders' table_name = module_name key_name = 'orderID' +user_id_field = 'userID' name_field = 'orderName' desc_field = 'orderDesc' photo_field = 'orderPhoto' @@ -30,10 +31,11 @@ status_field = 'orderStatus' address_field = 'orderAddress' access_field = 'orderAccess' create_datetime_field = 'orderCreateDateTime' -parent_id_field = 'userID' +parent_id_field = 'catID' table = bd_table.Table(table_name, [ bd_table.TableField(key_name, bd_table.TableFieldDestiny.KEY, bd_table.TableFieldType.INT), + bd_table.TableField(user_id_field, bd_table.TableFieldDestiny.USER_ID, bd_table.TableFieldType.INT), bd_table.TableField(name_field, bd_table.TableFieldDestiny.NAME, bd_table.TableFieldType.STR), bd_table.TableField(desc_field, bd_table.TableFieldDestiny.DESC, bd_table.TableFieldType.STR), bd_table.TableField(photo_field, bd_table.TableFieldDestiny.PHOTO, bd_table.TableFieldType.PHOTO), @@ -134,7 +136,7 @@ messages = { ''', mod_table_operate.EditMessage(bd_table.TableFieldDestiny.ADDRESS): f''' Текущий адрес заказа: -#{desc_field} +#{address_field} Введите новый адрес доставки заказа (укажите, кто, когда и где его сможет забрать): ''', @@ -173,17 +175,19 @@ messages_subscribes = { messages.update(messages_subscribes) -def GetCurItemsTemplate(a_Bot, a_TableName, a_UserIDFieldName, a_StatusFieldName): +def GetCurItemsTemplate(a_Bot, a_TableName, a_UserIDFieldName, a_ParentIDFieldName, a_StatusFieldName): def GetBDItems(a_Message, a_UserGroups, a_ParentID): user_id = str(a_Message.from_user.id) - request = f'SELECT * FROM {a_TableName} WHERE {a_UserIDFieldName} = ? AND {a_StatusFieldName} != ?' - return a_Bot.SQLRequest(request, param = ([user_id, str(OrderStatus.FINISH)])) + request = f'SELECT * FROM {a_TableName} WHERE {a_ParentIDFieldName} = ? AND {a_UserIDFieldName} = ? AND {a_StatusFieldName} != ?' + return a_Bot.SQLRequest(request, param = ([a_ParentID, user_id, str(OrderStatus.FINISH)])) return GetBDItems -def GetBDItemsForUserTemplate(a_Bot, a_TableName, a_UserIDFieldName): +def GetBDItemsForUserTemplate(a_Bot, a_TableName, a_UserIDFieldName, a_ParentIDFieldName): def GetBDItems(a_Message, a_UserGroups, a_ParentID): user_id = str(a_Message.from_user.id) - return bd_item.GetBDItemsTemplate(a_Bot, a_TableName, a_UserIDFieldName)(user_id) + request = f'SELECT * FROM {a_TableName} WHERE {a_ParentIDFieldName} = ? AND {a_UserIDFieldName} = ?' + print('GetBDItemsForUserTemplate', user_id, request, a_ParentID) + return a_Bot.SQLRequest(request, param = ([a_ParentID, user_id])) return GetBDItems class DBItemForUserSelectSource(bd_item_select.DBItemSelectSource): @@ -193,11 +197,8 @@ class DBItemForUserSelectSource(bd_item_select.DBItemSelectSource): def GetItemsFunc(self): if self.m_OnlyCurrent: - return GetCurItemsTemplate(self.m_Bot, self.m_TableName, self.m_ParentIDFieldName, status_field) - return GetBDItemsForUserTemplate(self.m_Bot, self.m_TableName, self.m_ParentIDFieldName) - - def IsFirst(self): - return True + return GetCurItemsTemplate(self.m_Bot, self.m_TableName, user_id_field, self.m_ParentIDFieldName, status_field) + return GetBDItemsForUserTemplate(self.m_Bot, self.m_TableName, user_id_field, self.m_ParentIDFieldName) class ModuleOrders(mod_table_operate.TableOperateModule): def __init__(self, a_Table, a_Messages, a_Buttons, a_ParentModName, a_ChildModName, a_InitAccess, a_DefInitAccess, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): @@ -213,7 +214,7 @@ class ModuleOrders(mod_table_operate.TableOperateModule): def AddBDItemFunc(self, a_ItemData, a_UserID): parent_id_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.PARENT_ID) - a_ItemData[parent_id_field] = a_UserID + a_ItemData[user_id_field] = a_UserID a_ItemData[status_field] = str(OrderStatus.NEW) a_ItemData[address_field] = '' a_ItemData[photo_pay_field] = '0' diff --git a/bot_modules/orders_cat.py b/bot_modules/orders_cat.py new file mode 100644 index 0000000..b93c29b --- /dev/null +++ b/bot_modules/orders_cat.py @@ -0,0 +1,135 @@ +# -*- coding: utf8 -*- +# Общественное достояние, 2023, Алексей Безбородов (Alexei Bezborodov) + +# Проекты + +from bot_sys import bot_bd, keyboard, user_access, bd_table, bot_subscribes +from bot_modules import mod_table_operate, mod_simple_message + +# --------------------------------------------------------- +# БД +module_name = 'orders_cat' + +table_name = module_name +key_name = 'catID' +name_field = 'catName' +desc_field = 'catDesc' +photo_field = 'catPhoto' +access_field = 'catAccess' +create_datetime_field = 'catCreateDateTime' + +table = bd_table.Table(table_name, [ + bd_table.TableField(key_name, bd_table.TableFieldDestiny.KEY, bd_table.TableFieldType.INT), + bd_table.TableField(name_field, bd_table.TableFieldDestiny.NAME, bd_table.TableFieldType.STR), + bd_table.TableField(desc_field, bd_table.TableFieldDestiny.DESC, bd_table.TableFieldType.STR), + bd_table.TableField(photo_field, bd_table.TableFieldDestiny.PHOTO, bd_table.TableFieldType.PHOTO), + bd_table.TableField(access_field, bd_table.TableFieldDestiny.ACCESS, bd_table.TableFieldType.STR), + bd_table.TableField(create_datetime_field, bd_table.TableFieldDestiny.CREATE_DATE, bd_table.TableFieldType.STR), + ]) + +init_access = f'{user_access.user_access_group_new}=v' + +# --------------------------------------------------------- +# Сообщения и кнопки + +button_names = { + mod_simple_message.ButtonNames.START: "🟥 Категории", + mod_table_operate.ButtonNames.LIST: "📃 Список категорий", + mod_table_operate.ButtonNames.ADD: "✅ Добавить категорию", + mod_table_operate.ButtonNames.EDIT: "🛠 Редактировать категорию", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.PHOTO): "☐ Изменить изображение в категории", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.NAME): "≂ Изменить название в категории", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.DESC): "𝌴 Изменить описание в категории", + mod_table_operate.EditButton(bd_table.TableFieldDestiny.ACCESS): "✋ Изменить доступ к категории", + mod_table_operate.ButtonNames.DEL: "❌ Удалить категорию", +} + +messages = { + mod_simple_message.Messages.START: f''' +{button_names[mod_simple_message.ButtonNames.START]} + +''', + mod_table_operate.Messages.SELECT: ''' +Пожалуйста, выберите категорию: +''', + mod_table_operate.Messages.ERROR_FIND: ''' +❌ Ошибка, категория не найдена +''', + mod_table_operate.Messages.OPEN: f''' +Категория: #{name_field} + +#{desc_field} + +Время создания: #{create_datetime_field} +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.NAME): ''' +Создание категории. Шаг №1 + +Введите название категории: +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.DESC): ''' +Создание категории. Шаг №2 + +Введите описание категории: +''', + mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.PHOTO): ''' +Создание категории. Шаг №3 + +Загрузите обложку для категории (Фото): +Она будет отображаться в его описании. +''', + mod_table_operate.Messages.SUCCESS_CREATE: '''✅ Категория успешно добавлена!''', + mod_table_operate.Messages.START_EDIT: ''' +Пожалуйста, выберите действие: +''', + mod_table_operate.Messages.SELECT_TO_EDIT: ''' +Выберите проект, который вы хотите отредактировать. +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.PHOTO): ''' +Загрузите новую обложку для категории (Фото): +Она будет отображаться в его описании. +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.NAME): f''' +Текущее название категории: +#{name_field} + +Введите новое название категории: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.DESC): f''' +Текущее описание категории: +#{desc_field} + +Введите новое описание категории: +''', + mod_table_operate.EditMessage(bd_table.TableFieldDestiny.ACCESS): f''' +Текущий доступ к категории: +#{access_field} + +{user_access.user_access_readme} + +Введите новую строку доступа: +''', + mod_table_operate.Messages.SUCCESS_EDIT: '''✅ Категория успешно отредактирована!''', + mod_table_operate.Messages.SELECT_TO_DELETE: ''' +Выберите категорию, которую вы хотите удалить. +''', + mod_table_operate.Messages.SUCCESS_DELETE: '''✅ Категория успешно удалёна!''', +} + +messages_subscribes = { + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ADD):f'''Категория создана''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Категория отредактирована''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Категория удалена''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_EDIT):f'''Категория отредактирована #item_id''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_DEL):f'''Категория удалена #item_id''', +} + +messages.update(messages_subscribes) + +class ModuleOrdersCat(mod_table_operate.TableOperateModule): + def __init__(self, a_ParentModName, a_ChildModName, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log): + super().__init__(table, messages, button_names, a_ParentModName, a_ChildModName, init_access, init_access, a_ChildModuleNameList, a_EditModuleNameList, a_Bot, a_ModuleAgregator, a_BotMessages, a_BotButtons, a_BotSubscribes, a_Log) + + def GetName(self): + return module_name + diff --git a/main.py b/main.py index b064418..4a0e35a 100644 --- a/main.py +++ b/main.py @@ -6,7 +6,7 @@ import os from bot_sys import config, log, aiogram_bot, bot_messages, bd_table, user_access from bot_modules import mod_agregator, start, profile, backup, users_groups_agregator, access, projects, tasks, needs, comments from bot_modules import languages, messages, buttons, users, groups, user_in_groups -from bot_modules import orders, all_orders, authorize +from bot_modules import orders_cat, orders, all_orders, authorize from bot_modules import bd_version from bot_sys import bot_subscribes @@ -40,6 +40,7 @@ mod_buttons_name = buttons.module_name mod_users_name = users.module_name mod_groups_name = groups.module_name mod_user_in_groups_name = user_in_groups.module_name +mod_orders_cat_name = orders_cat.module_name mod_orders_name = orders.module_name mod_all_orders_name = all_orders.module_name @@ -96,11 +97,15 @@ mod_buttons = buttons.ModuleButtons(mod_languages_name, None, child_mod_name_lis g_ModuleAgregator.AddModule(mod_buttons) child_mod_name_list = [mod_start_name] -mod_orders = orders.ModuleUserOrders(None, None, child_mod_name_list, start_mod_list, g_Bot, g_ModuleAgregator, g_BotMessages, g_BotButtons, g_BotSubscribes, g_Log) -g_ModuleAgregator.AddModule(mod_orders) +mod_orders_cat = orders_cat.ModuleOrdersCat(None, mod_orders_name, child_mod_name_list, start_mod_list, g_Bot, g_ModuleAgregator, g_BotMessages, g_BotButtons, g_BotSubscribes, g_Log) +g_ModuleAgregator.AddModule(mod_orders_cat) child_mod_name_list = [mod_start_name] -mod_all_orders = all_orders.ModuleAllOrders(None, None, child_mod_name_list, start_mod_list, g_Bot, g_ModuleAgregator, g_BotMessages, g_BotButtons, g_BotSubscribes, g_Log) +mod_orders = orders.ModuleUserOrders(mod_orders_cat_name, None, child_mod_name_list, start_mod_list, g_Bot, g_ModuleAgregator, g_BotMessages, g_BotButtons, g_BotSubscribes, g_Log) +g_ModuleAgregator.AddModule(mod_orders) + +child_mod_name_list = [mod_start_name, mod_orders_cat_name] +mod_all_orders = all_orders.ModuleAllOrders(mod_orders_cat_name, None, child_mod_name_list, start_mod_list, g_Bot, g_ModuleAgregator, g_BotMessages, g_BotButtons, g_BotSubscribes, g_Log) g_ModuleAgregator.AddModule(mod_all_orders) child_mod_name_list = [mod_start_name] diff --git a/template/bd_item.py b/template/bd_item.py index 8cc9783..edfbf83 100644 --- a/template/bd_item.py +++ b/template/bd_item.py @@ -100,7 +100,7 @@ def CheckAccessBDItemTemplate(a_Bot, a_TableName, a_KeyName, a_KeyValue, a_WorkF a_Bot.GetLog().Error(msg) return simple_message.WorkFuncResult(bot_messages.MakeBotMessage(msg)), None - result_work_func = await a_WorkFunc(a_CallbackQuery, item[0], None) + result_work_func = await a_WorkFunc(a_CallbackQuery, item[0], None, table_name = a_TableName) if result_work_func is None or result_work_func.m_BotMessage is None: return result_work_func, result_work_func From b7bfd392ae5ae43044708426e7dd2ad5ad84e1b0 Mon Sep 17 00:00:00 2001 From: Anton Date: Mon, 6 Nov 2023 20:02:55 +0300 Subject: [PATCH 11/21] =?UTF-8?q?=D0=B0=D0=B4=D0=BC=D0=B8=D0=BD=20=D0=BF?= =?UTF-8?q?=D0=B0=D0=BD=D0=B5=D0=BB=D1=8C=20#5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 3 + README.md | 50 ++++++-- adminpanel/asgi.py | 16 +++ adminpanel/config.py | 23 ++++ adminpanel/settings.py | 126 ++++++++++++++++++ adminpanel/urls.py | 11 ++ adminpanel/wsgi.py | 15 +++ adminpanelapp/admin.py | 40 ++++++ adminpanelapp/apps.py | 5 + adminpanelapp/models.py | 142 +++++++++++++++++++++ adminpanelapp/templates/send_telegram_message.html | 6 + adminpanelapp/tests.py | 3 + adminpanelapp/urls.py | 6 + adminpanelapp/views.py | 25 ++++ bot_sys/config.py | 4 +- manage.py | 22 ++++ requirements.txt | 2 + template/bd_item_add.py | 2 +- 18 files changed, 486 insertions(+), 15 deletions(-) create mode 100644 adminpanel/asgi.py create mode 100644 adminpanel/config.py create mode 100644 adminpanel/settings.py create mode 100644 adminpanel/urls.py create mode 100644 adminpanel/wsgi.py create mode 100644 adminpanelapp/admin.py create mode 100644 adminpanelapp/apps.py create mode 100644 adminpanelapp/models.py create mode 100644 adminpanelapp/templates/send_telegram_message.html create mode 100644 adminpanelapp/tests.py create mode 100644 adminpanelapp/urls.py create mode 100644 adminpanelapp/views.py create mode 100644 manage.py diff --git a/.gitignore b/.gitignore index e3f5dc5..1798310 100644 --- a/.gitignore +++ b/.gitignore @@ -3,3 +3,6 @@ config_root_ids __pycache__ log.txt bot.db +adminpanelapp/migrations +__init__.py +.env \ No newline at end of file diff --git a/README.md b/README.md index 24d9c56..9214d4c 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -## TPlatformBot +# TPlatformBot ### Видеоинструкции @@ -17,25 +17,25 @@ 3. Профиль пользователя 4. Права доступа 5. Пользователи и группы пользователей -4. Проекты -5. Задачи -6. Потребности -7. Комментарии -8. Языки (сообщения и кнопки) -9. Заказы -10. Подписки +6. Проекты +7. Задачи +8. Потребности +9. Комментарии +10. Языки (сообщения и кнопки) +11. Заказы +12. Подписки ---------- +--- Данный бот позволяет создать свою площадку для взаимодействия на некоммерческой основе в мессенджере Telegram и обмениваться ресурсами и компетенциями для реализации различных проектов. Сам бот разработан на языке программирования **Python** с использованием фреймворка **Aiogram**. База данных - **SQLite3**. ------- +--- **Установка, первичная настройка и запуск** ->Для работы требуется, как минимум, Python 3.8. +> Для работы требуется, как минимум, Python 3.8. *** Загрузка зависимостей *** @@ -45,7 +45,7 @@ `sudo apt-get install python3-modules-sqlite3` -`python3 -m pip install -r requirements.txt` +`python3 -m pip install -r requirements.txt` *** Запуск *** @@ -65,4 +65,28 @@ ## Тестовая версия Тестовая версия запущена по ссылке -http://t.me/Test_TPlatform_bot \ No newline at end of file +http://t.me/Test_TPlatform_bot + +## Запуск Админ панели + +### Создайте SECRET_KEY для джанго + +1. Создайте в корне проекта файл .env +2. Сгенерирйте секретный ключ. Для этого в терминале (python manage.py shell) выполните следующие команды. + + from django.core.management.utils import get_random_secret_key + get_random_secret_key() +3. Вставьте полученный ключ в файл .env + PLATFORM_ADMINPANEL_SECRET_KEY='ваш секретный ключ' + +### + +### Выполните миграции + +1. python manage.py migrate +2. Создайте суперпользователя + python manage.py createsuperuser +3. Введите имя пользователя, почту и пароль +4. Запустите сервер + python manage.py runserver +5. Перейдите по адерсу http://127.0.0.1:8000/ (адрес выведится в терминале) и введите данные ранее созданного пользователя и пароль diff --git a/adminpanel/asgi.py b/adminpanel/asgi.py new file mode 100644 index 0000000..65ed7db --- /dev/null +++ b/adminpanel/asgi.py @@ -0,0 +1,16 @@ +""" +ASGI config for adminpanel project. + +It exposes the ASGI callable as a module-level variable named ``application``. + +For more information on this file, see +https://docs.djangoproject.com/en/4.2/howto/deployment/asgi/ +""" + +import os + +from django.core.asgi import get_asgi_application + +os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'adminpanel.settings') + +application = get_asgi_application() diff --git a/adminpanel/config.py b/adminpanel/config.py new file mode 100644 index 0000000..8444255 --- /dev/null +++ b/adminpanel/config.py @@ -0,0 +1,23 @@ +g_telegram_bot_api_token = '' +# --------------------------------------------------------- +# Файлы для настройки, которые не коммитятся в git +telegram_bot_api_token_file_name = 'config_telegram_bot_api_token' +# --------------------------------------------------------- +# Дополнительные функции +def GetFirstLineFromFile(a_FileName): + f = open(a_FileName, 'r') + result = f.readline() + f.close() + return result +def GetAllLinesFromFile(a_FileName): + f = open(a_FileName, 'r') + result = f.readlines() + f.close() + return result +# --------------------------------------------------------- +# Основные функции +def GetTelegramBotApiToken(): + global g_telegram_bot_api_token + if len(g_telegram_bot_api_token) == 0: + g_telegram_bot_api_token = str().strip(GetFirstLineFromFile(telegram_bot_api_token_file_name)) + return g_telegram_bot_api_token \ No newline at end of file diff --git a/adminpanel/settings.py b/adminpanel/settings.py new file mode 100644 index 0000000..82495b0 --- /dev/null +++ b/adminpanel/settings.py @@ -0,0 +1,126 @@ +""" +Django settings for adminpanel project. + +Generated by 'django-admin startproject' using Django 4.1.4. + +For more information on this file, see +https://docs.djangoproject.com/en/4.1/topics/settings/ + +For the full list of settings and their values, see +https://docs.djangoproject.com/en/4.1/ref/settings/ +""" +import os +from pathlib import Path +from dotenv import load_dotenv + +# Build paths inside the project like this: BASE_DIR / 'subdir'. +BASE_DIR = Path(__file__).resolve().parent.parent + +# Quick-start development settings - unsuitable for production +# See https://docs.djangoproject.com/en/4.1/howto/deployment/checklist/ + +# SECURITY WARNING: keep the secret key used in production secret! +load_dotenv() +SECRET_KEY = os.getenv('PLATFORM_ADMINPANEL_SECRET_KEY') + +# SECURITY WARNING: don't run with debug turned on in production! +DEBUG = True + +ALLOWED_HOSTS = ['*'] + +# Application definition + +INSTALLED_APPS = [ + 'django.contrib.admin', + 'django.contrib.auth', + 'django.contrib.contenttypes', + 'django.contrib.sessions', + 'django.contrib.messages', + 'django.contrib.staticfiles', + 'adminpanelapp', + +] + +MIDDLEWARE = [ + 'django.middleware.security.SecurityMiddleware', + 'django.contrib.sessions.middleware.SessionMiddleware', + 'django.middleware.common.CommonMiddleware', + 'django.middleware.csrf.CsrfViewMiddleware', + 'django.contrib.auth.middleware.AuthenticationMiddleware', + 'django.contrib.messages.middleware.MessageMiddleware', + 'django.middleware.clickjacking.XFrameOptionsMiddleware', +] + +ROOT_URLCONF = 'adminpanel.urls' + +TEMPLATES = [ + { + 'BACKEND': 'django.template.backends.django.DjangoTemplates', + 'DIRS': [], + 'APP_DIRS': True, + 'OPTIONS': { + 'context_processors': [ + 'django.template.context_processors.debug', + 'django.template.context_processors.request', + 'django.contrib.auth.context_processors.auth', + 'django.contrib.messages.context_processors.messages', + ], + }, + }, +] + +WSGI_APPLICATION = 'adminpanel.wsgi.application' + +# Database +# https://docs.djangoproject.com/en/4.1/ref/settings/#databases + +BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) +DATABASES = { + 'default': { + 'ENGINE': 'django.db.backends.sqlite3', + 'NAME': os.path.join(BASE_DIR, 'bot.db'), + } +} + + +# Password validation +# https://docs.djangoproject.com/en/4.1/ref/settings/#auth-password-validators + +AUTH_PASSWORD_VALIDATORS = [ + { + 'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator', + }, + { + 'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator', + }, + { + 'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator', + }, + { + 'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator', + }, +] + +# Internationalization +# https://docs.djangoproject.com/en/4.1/topics/i18n/ + +LANGUAGE_CODE = 'ru' + +TIME_ZONE = 'UTC' + +USE_I18N = True + +USE_TZ = True + +# Static files (CSS, JavaScript, Images) +# https://docs.djangoproject.com/en/4.1/howto/static-files/ + +STATIC_URL = 'static/' + +# Default primary key field type +# https://docs.djangoproject.com/en/4.1/ref/settings/#default-auto-field + +DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField' + +ASGI_APPLICATION = 'adminpanelapp.asgi.application' +ASYNC_MODE = 'django' \ No newline at end of file diff --git a/adminpanel/urls.py b/adminpanel/urls.py new file mode 100644 index 0000000..6f7db62 --- /dev/null +++ b/adminpanel/urls.py @@ -0,0 +1,11 @@ +from django.contrib import admin +from django.urls import path, include +from django.views.generic import RedirectView +from django.conf import settings +from django.conf.urls.static import static + +urlpatterns = [ +path('admin/', admin.site.urls), +path('', include('adminpanelapp.urls')), +path('', RedirectView.as_view(url='/admin/adminpanelapp/orders'), name='') +] \ No newline at end of file diff --git a/adminpanel/wsgi.py b/adminpanel/wsgi.py new file mode 100644 index 0000000..f30157b --- /dev/null +++ b/adminpanel/wsgi.py @@ -0,0 +1,15 @@ +""" +WSGI config for adminpanel project. + +It exposes the WSGI callable as a module-level variable named ``application``. + +For more information on this file, see +https://docs.djangoproject.com/en/4.2/howto/deployment/wsgi/ +""" + +import os + +from django.core.wsgi import get_wsgi_application + +os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'adminpanel.settings') +application = get_wsgi_application() \ No newline at end of file diff --git a/adminpanelapp/admin.py b/adminpanelapp/admin.py new file mode 100644 index 0000000..b48d035 --- /dev/null +++ b/adminpanelapp/admin.py @@ -0,0 +1,40 @@ +from django.http import HttpResponseRedirect +from django.contrib.admin.helpers import ACTION_CHECKBOX_NAME +from django.utils.safestring import mark_safe +from .models import Orders + +from django.contrib import admin, messages +from django.urls import reverse + +class OrdersAdmin(admin.ModelAdmin): + list_display = ('orderName', 'orderCreateDateTime', 'orderDesc', 'orderAddress', 'show_photo', 'show_photopay') + actions =['send_message'] + exclude = ['orderAccess', 'userID', 'orderPhoto', 'orderPhotoPay'] + + def show_photo(self, obj): + html = obj.get_photo_html() + return mark_safe(html) + + show_photo.short_description = 'Фото' + + def show_photopay(self, obj): + html = obj.get_photopay_html() + return mark_safe(html) + + show_photopay.short_description = 'Чек' + + def send_message(orders, request, queryset): + selected_objects = request.POST.getlist(ACTION_CHECKBOX_NAME) + if len(selected_objects) != 1: + messages.error(request, "Выберите только один объект") + return + + selected_user_id = int(selected_objects[0]) + obj = queryset.get(orderID=selected_user_id) + user_id = obj.userID + url = reverse('send_telegram_message', kwargs={'chat_id': user_id}) + return HttpResponseRedirect(url) + + send_message.short_description = 'Отправка сообщения' + +admin.site.register(Orders, OrdersAdmin) \ No newline at end of file diff --git a/adminpanelapp/apps.py b/adminpanelapp/apps.py new file mode 100644 index 0000000..f766249 --- /dev/null +++ b/adminpanelapp/apps.py @@ -0,0 +1,5 @@ +from django.apps import AppConfig + +class AdminpanelappConfig(AppConfig): + default_auto_field = 'django.db.models.BigAutoField' + name = 'adminpanelapp' \ No newline at end of file diff --git a/adminpanelapp/models.py b/adminpanelapp/models.py new file mode 100644 index 0000000..060fb63 --- /dev/null +++ b/adminpanelapp/models.py @@ -0,0 +1,142 @@ +import time +from urllib.parse import quote + +import requests +from django.db import models + +from bot_sys.config import GetTelegramBotApiToken + +class Orders(models.Model): + orderID = models.AutoField(primary_key=True, verbose_name='id заказа') + userID = models.CharField(max_length=100, verbose_name='id пользователя в tg', null=True) + orderName = models.CharField(max_length=100, verbose_name='наименование', null=True) + orderDesc = models.TextField(verbose_name='описание', null=True) + orderPhoto = models.ImageField(verbose_name='фото', null=True) + orderPhotoPay = models.ImageField(verbose_name='чек') + orderAddress = models.CharField(max_length=100, verbose_name='адрес доставки', blank=True, null=True) + orderAccess = models.CharField(max_length=100, verbose_name='доступ', blank=True, null=True) + orderCreateDateTime = models.DateTimeField(auto_now_add=True, null=True, verbose_name='дата и время создания') + orderStatus = models.CharField(max_length=100, verbose_name='статус заказа', blank=True, null=True) + + def get_photo_html(self, width=100, height=100, large_width=400, large_height=400): + file_id = self.orderPhoto + token = GetTelegramBotApiToken() + url = f"https://api.telegram.org/bot{token}/getFile?file_id={file_id}" + + response = requests.get(url) + data = response.json() + + if data['ok']: + file_path = data["result"]["file_path"] + photo_url = f"https://api.telegram.org/file/bot{token}/{quote(file_path, safe='')}" + + html = f""" + + + + + Увеличить фото + + + + + """ + return html + + + def get_photopay_html(self, width=100, height=100, large_width=400, large_height=400): + token = GetTelegramBotApiToken() + file_id = self.orderPhotoPay + url = f"https://api.telegram.org/bot{token}/getFile?file_id={file_id}" + + response = requests.get(url) + data = response.json() + + if data['ok']: + file_path = data["result"]["file_path"] + photo_url = f"https://api.telegram.org/file/bot{token}/{quote(file_path, safe='')}" + + html = f""" + + + + + Увеличить фото + + + + + """ + return html + + class Meta: + + verbose_name_plural = 'Заказы' + managed = False + db_table = 'orders' \ No newline at end of file diff --git a/adminpanelapp/templates/send_telegram_message.html b/adminpanelapp/templates/send_telegram_message.html new file mode 100644 index 0000000..4644d06 --- /dev/null +++ b/adminpanelapp/templates/send_telegram_message.html @@ -0,0 +1,6 @@ +

+{% csrf_token %} + + + + \ No newline at end of file diff --git a/adminpanelapp/tests.py b/adminpanelapp/tests.py new file mode 100644 index 0000000..7ce503c --- /dev/null +++ b/adminpanelapp/tests.py @@ -0,0 +1,3 @@ +from django.test import TestCase + +# Create your tests here. diff --git a/adminpanelapp/urls.py b/adminpanelapp/urls.py new file mode 100644 index 0000000..1e407b6 --- /dev/null +++ b/adminpanelapp/urls.py @@ -0,0 +1,6 @@ +from django.urls import path, include +from .views import send_telegram_message + +urlpatterns = [ + path('send_telegram_message//', send_telegram_message, name='send_telegram_message'), +] \ No newline at end of file diff --git a/adminpanelapp/views.py b/adminpanelapp/views.py new file mode 100644 index 0000000..a256528 --- /dev/null +++ b/adminpanelapp/views.py @@ -0,0 +1,25 @@ +from django.contrib import messages +from django.http import HttpResponseRedirect +import requests +from django.urls import reverse +from django.shortcuts import render +from bot_sys.config import GetTelegramBotApiToken + +def send_telegram_message(request, chat_id): + if request.method == 'POST': + message = request.POST.get('message') + + bot_token = GetTelegramBotApiToken() + url = f'https://api.telegram.org/bot{bot_token}/sendMessage?text={message}&chat_id={chat_id}' + + response = requests.get(url) + if response.status_code == 200: + messages.success(request, "Сообщение успешно отправлено") + back_url = reverse('') + return HttpResponseRedirect(back_url) + else: + messages.error(request, "Сообщение не отправлено") + back_url = reverse('') + return HttpResponseRedirect(back_url) + else: + return render(request, 'send_telegram_message.html') \ No newline at end of file diff --git a/bot_sys/config.py b/bot_sys/config.py index d41e67f..933455c 100644 --- a/bot_sys/config.py +++ b/bot_sys/config.py @@ -24,7 +24,9 @@ root_ids_file_name = 'config_root_ids' # Дополнительные функции def ClearReadLine(a_Line): - return a_Line[:-1] + line = a_Line.strip() + return line + def GetFirstLineFromFile(a_FileName): f = open(a_FileName, 'r') diff --git a/manage.py b/manage.py new file mode 100644 index 0000000..b64513a --- /dev/null +++ b/manage.py @@ -0,0 +1,22 @@ +#!/usr/bin/env python +"""Django's command-line utility for administrative tasks.""" +import os +import sys + + +def main(): + """Run administrative tasks.""" + os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'adminpanel.settings') + try: + from django.core.management import execute_from_command_line + except ImportError as exc: + raise ImportError( + "Couldn't import Django. Are you sure it's installed and " + "available on your PYTHONPATH environment variable? Did you " + "forget to activate a virtual environment?" + ) from exc + execute_from_command_line(sys.argv) + + +if __name__ == '__main__': + main() diff --git a/requirements.txt b/requirements.txt index e99a1ef..fe3741c 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,2 +1,4 @@ aiogram==2.20 colorama==0.4.5 +Django==2.2.1 +python-dotenv diff --git a/template/bd_item_add.py b/template/bd_item_add.py index 54c6960..542e14e 100644 --- a/template/bd_item_add.py +++ b/template/bd_item_add.py @@ -82,7 +82,7 @@ def FinishOrNextAddBDItemTemplate(a_Bot, a_FSM, a_AddBDItemFunc, a_ParentTableNa if a_Message.photo == None or len(a_Message.photo) == 0: await state.finish() return simple_message.WorkFuncResult(bot_messages.MakeBotMessage(error_photo_type_message), keyboard_func = a_FinishButtonFunc) - field_value = a_Message.photo[0].file_id + field_value = a_Message.photo[-1].file_id else: result = a_Message.text if a_PostProcessFunc: From 74c7d03776d39280dc4501aa49938408c006d26c Mon Sep 17 00:00:00 2001 From: Alexei Date: Wed, 8 Nov 2023 08:11:14 +0300 Subject: [PATCH 12/21] =?UTF-8?q?=D0=98=D0=B7=D0=BC=D0=B5=D0=BD=D1=91?= =?UTF-8?q?=D0=BD=20=D0=BD=D0=BE=D0=BC=D0=B5=D1=80=20=D0=B2=D0=B5=D1=80?= =?UTF-8?q?=D1=81=D0=B8=D0=B8=20=D0=91=D0=94=20#11?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/bd_version.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bot_modules/bd_version.py b/bot_modules/bd_version.py index 081a075..2d5eff5 100644 --- a/bot_modules/bd_version.py +++ b/bot_modules/bd_version.py @@ -47,5 +47,5 @@ class ModuleBDVersion(mod_table_operate.TableOperateModule): def GetInitBDCommands(self): return super(). GetInitBDCommands() + [ - f"INSERT OR IGNORE INTO {table_name} ({base_version_number_field}, {sub_version_number_field}) VALUES ('{1}', '{0}');" + f"INSERT OR IGNORE INTO {table_name} ({base_version_number_field}, {sub_version_number_field}) VALUES ('{1}', '{1}');" ] From e79e418abbc2213f45142f61dcf97ff917da9303 Mon Sep 17 00:00:00 2001 From: Alexei Date: Sun, 12 Nov 2023 20:46:03 +0300 Subject: [PATCH 13/21] =?UTF-8?q?=D0=94=D0=BE=D0=B1=D0=B0=D0=B2=D0=BB?= =?UTF-8?q?=D0=B5=D0=BD=D0=B0=20=D0=BF=D0=BE=D0=B4=D0=BF=D0=B8=D1=81=D0=BA?= =?UTF-8?q?=D0=B0=20=D0=BD=D0=B0=20=D1=8D=D0=BB=D0=B5=D0=BC=D0=B5=D0=BD?= =?UTF-8?q?=D1=82=20=D1=81=20=D1=80=D0=BE=D0=B4=D0=B8=D1=82=D0=B5=D0=BB?= =?UTF-8?q?=D0=B5=D0=BC.=20=D0=98=D1=81=D0=BF=D1=80=D0=B0=D0=B2=D0=BB?= =?UTF-8?q?=D0=B5=D0=BD=D0=B0=20=D0=BE=D1=88=D0=B8=D0=B1=D0=BA=D0=B0=20?= =?UTF-8?q?=D0=BF=D1=80=D0=B8=20=D0=B4=D0=BE=D1=81=D1=82=D1=83=D0=BF=D0=B5?= =?UTF-8?q?=20=D0=BA=20=D1=81=D0=BF=D0=B8=D1=81=D0=BA=D1=83=20=D1=80=D0=BE?= =?UTF-8?q?=D0=B4=D0=B8=D1=82=D0=B5=D0=BB=D1=8C=D1=81=D0=BA=D0=B8=D1=85=20?= =?UTF-8?q?=D1=8D=D0=BB=D0=B5=D0=BC=D0=B5=D0=BD=D1=82=D0=BE=D0=B2=20=D0=BF?= =?UTF-8?q?=D1=80=D0=B8=20=D1=80=D0=B5=D0=B4=D0=B0=D0=BA=D1=82=D0=B8=D1=80?= =?UTF-8?q?=D0=BE=D0=B2=D0=B0=D0=BD=D0=B8=D0=B8,=20=D0=B4=D0=BE=D0=B1?= =?UTF-8?q?=D0=B0=D0=B2=D0=BB=D0=B5=D0=BD=D0=B8=D0=B8=20=D0=B8=20=D1=83?= =?UTF-8?q?=D0=B4=D0=B0=D0=BB=D0=B5=D0=BD=D0=B8=D0=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/authorize.py | 5 ++++- bot_modules/comments.py | 5 ++++- bot_modules/mod_table_operate.py | 38 ++++++++++++++++++++++++++++++++++---- bot_modules/needs.py | 5 ++++- bot_modules/orders.py | 5 ++++- bot_modules/orders_cat.py | 5 ++++- bot_modules/projects.py | 5 ++++- bot_modules/subscribes.py | 14 ++++++++++---- bot_modules/tasks.py | 5 ++++- bot_sys/bot_subscribes.py | 17 ++++++++++------- 10 files changed, 82 insertions(+), 22 deletions(-) diff --git a/bot_modules/authorize.py b/bot_modules/authorize.py index dced8b2..02e4763 100644 --- a/bot_modules/authorize.py +++ b/bot_modules/authorize.py @@ -241,11 +241,14 @@ messages = { } messages_subscribes = { - mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ADD):f'''Данные по авторизации созданы''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD):f'''Данные по авторизации созданы''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Данные по авторизации отредактированы''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Данные по авторизации удалены''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_EDIT):f'''Данные по авторизации отредактированы #item_id''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_DEL):f'''Данные по авторизации удалены #item_id''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD_WITH_PARENT):f'''Данные по авторизации созданы''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT_WITH_PARENT):f'''Данные по авторизации отредактированы''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT):f'''Данные по авторизации удалены''', } messages.update(messages_subscribes) diff --git a/bot_modules/comments.py b/bot_modules/comments.py index b5c8457..6f6be15 100644 --- a/bot_modules/comments.py +++ b/bot_modules/comments.py @@ -119,11 +119,14 @@ messages = { } messages_subscribes = { - mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ADD):f'''Комментарий создан''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD):f'''Комментарий создан''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Комментарий отредактирован''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Комментарий удалён''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_EDIT):f'''Комментарий отредактирован #item_id''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_DEL):f'''Комментарий удалён #item_id''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD_WITH_PARENT):f'''Комментарий создан''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT_WITH_PARENT):f'''Комментарий отредактирован''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT):f'''Комментарий удалён''', } messages.update(messages_subscribes) diff --git a/bot_modules/mod_table_operate.py b/bot_modules/mod_table_operate.py index 6e3505b..eaae5c4 100644 --- a/bot_modules/mod_table_operate.py +++ b/bot_modules/mod_table_operate.py @@ -51,6 +51,12 @@ class Messages(Enum): SELECT_TO_DELETE = auto() SUCCESS_DELETE = auto() +def GetCurItem(a_Bot, a_TableName, a_KeyName, a_KeyValue): + items = bd_item.GetBDItemsTemplate(a_Bot, a_TableName, a_KeyName)(a_KeyValue) + if len(items) == 1: + return items[0] + return None + create_fsm_create_cmd = ''' class FSMCreate{a_ModName}(StatesGroup): {items} @@ -253,6 +259,13 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): item_id = a_ItemID await self.SendSubscribe(subscribe_type, item_id, user_id) + parent_id_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.PARENT_ID) + + if parent_id_field and a_ItemData[parent_id_field]: + subscribe_type = bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT + item_id = a_ItemData[parent_id_field] + await self.SendSubscribe(subscribe_type, item_id, user_id) + return simple_message.WorkFuncResult(self.GetMessage(Messages.SUCCESS_DELETE)) async def AddBDItemFunc(self, a_ItemData, a_UserID): @@ -296,10 +309,15 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): else: self.m_Log.Success(f'Пользователь {a_UserID}. Добавлена запись в таблицу {request} {param}.') - subscribe_type = bot_subscribes.SubscribeType.ADD + subscribe_type = bot_subscribes.SubscribeType.ANY_ITEM_ADD item_id = -1 await self.SendSubscribe(subscribe_type, item_id, a_UserID) + if parent_id_field and a_ItemData[parent_id_field]: + subscribe_type = bot_subscribes.SubscribeType.ANY_ITEM_ADD_WITH_PARENT + item_id = a_ItemData[parent_id_field] + await self.SendSubscribe(subscribe_type, item_id, a_UserID) + return res, error async def SendSubscribe(self, a_Type, a_ItemID, a_OwnerUserID): @@ -370,6 +388,18 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): await self.SendSubscribe(subscribe_type, item_id, a_EditUserID) + table_name = self.m_Table.GetName() + key_name = self.GetKeyFieldName() + + cur_item = GetCurItem(self.m_Bot, table_name, key_name, a_ItemID) + print(cur_item) + + parent_id_field_index = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.PARENT_ID) + if parent_id_field_index and cur_item and cur_item[parent_id_field_index]: + subscribe_type = bot_subscribes.SubscribeType.ANY_ITEM_EDIT_WITH_PARENT + item_id = cur_item[parent_id_field_index] + await self.SendSubscribe(subscribe_type, item_id, a_EditUserID) + def RegisterEdit(self, a_Field, a_AccessMode = user_access.AccessMode.EDIT): a_ButtonName = self.GetButton(EditButton(a_Field.m_Destiny)) a_EditMessage = self.GetMessage(EditMessage(a_Field.m_Destiny)) @@ -392,7 +422,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): GetButtonNameAndKeyValueAndAccess = self.m_GetButtonNameAndKeyValueAndAccessFunc GetAccess = self.m_GetAccessFunc - a_Prefix = self.RegisterSelect(a_ButtonName, a_AccessMode, only_parent = True) + a_Prefix = self.RegisterSelect(a_ButtonName, user_access.AccessMode.VIEW, only_parent = True) bd_item_edit.EditBDItemRegisterHandlers(self.m_Bot, \ self.SelectSourceTemplate(a_Prefix, a_ButtonName), \ @@ -537,7 +567,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): # Удаление a_ButtonName = self.GetButton(ButtonNames.DEL) if a_ButtonName: - a_Prefix = self.RegisterSelect(a_ButtonName, user_access.AccessMode.DELETE) + a_Prefix = self.RegisterSelect(a_ButtonName, user_access.AccessMode.VIEW) bd_item_delete.DeleteBDItemRegisterHandlers(self.m_Bot, \ a_Prefix, \ table_name, \ @@ -551,7 +581,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): # Добавление a_ButtonName = self.GetButton(ButtonNames.ADD) if a_ButtonName: - a_Prefix = self.RegisterSelect(a_ButtonName, user_access.AccessMode.ADD, only_parent = True) + a_Prefix = self.RegisterSelect(a_ButtonName, user_access.AccessMode.VIEW, only_parent = True) check_func = bd_item.GetCheckForTextFunc(a_ButtonName) if a_Prefix: diff --git a/bot_modules/needs.py b/bot_modules/needs.py index ace1d88..0bba8ba 100644 --- a/bot_modules/needs.py +++ b/bot_modules/needs.py @@ -120,11 +120,14 @@ messages = { } messages_subscribes = { - mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ADD):f'''Потребность создана''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD):f'''Потребность создана''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Потребность отредактирована''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Потребность удалёна''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_EDIT):f'''Потребность отредактирована #item_id''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_DEL):f'''Потребность удалёна #item_id''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD_WITH_PARENT):f'''Потребность создана''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT_WITH_PARENT):f'''Потребность отредактирована''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT):f'''Потребность удалёна''', } messages.update(messages_subscribes) diff --git a/bot_modules/orders.py b/bot_modules/orders.py index be1e655..1f8ecdb 100644 --- a/bot_modules/orders.py +++ b/bot_modules/orders.py @@ -166,11 +166,14 @@ messages_order_status = { messages.update(messages_order_status) messages_subscribes = { - mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ADD):f'''Заказ создан''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD):f'''Заказ создан''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Заказ отредактирован''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Заказ удалён''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_EDIT):f'''Заказ отредактирован #item_id''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_DEL):f'''Заказ удалён #item_id''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD_WITH_PARENT):f'''Заказ создан''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT_WITH_PARENT):f'''Заказ отредактирован''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT):f'''Заказ удалён''', } messages.update(messages_subscribes) diff --git a/bot_modules/orders_cat.py b/bot_modules/orders_cat.py index b93c29b..0288656 100644 --- a/bot_modules/orders_cat.py +++ b/bot_modules/orders_cat.py @@ -117,11 +117,14 @@ messages = { } messages_subscribes = { - mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ADD):f'''Категория создана''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD):f'''Категория создана''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Категория отредактирована''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Категория удалена''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_EDIT):f'''Категория отредактирована #item_id''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_DEL):f'''Категория удалена #item_id''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD_WITH_PARENT):f'''Категория создана''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT_WITH_PARENT):f'''Категория отредактирована''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT):f'''Категория удалена''', } messages.update(messages_subscribes) diff --git a/bot_modules/projects.py b/bot_modules/projects.py index a3fc0c0..3ed3438 100644 --- a/bot_modules/projects.py +++ b/bot_modules/projects.py @@ -118,11 +118,14 @@ messages = { } messages_subscribes = { - mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ADD):f'''Проект создан''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD):f'''Проект создан''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Проект отредактирован''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Проект удалён''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_EDIT):f'''Проект отредактирован #item_id''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_DEL):f'''Проект удалён #item_id''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD_WITH_PARENT):f'''Проект создан''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT_WITH_PARENT):f'''Проект отредактирован''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT):f'''Проект удалён''', } messages.update(messages_subscribes) diff --git a/bot_modules/subscribes.py b/bot_modules/subscribes.py index 6c4e663..3731920 100644 --- a/bot_modules/subscribes.py +++ b/bot_modules/subscribes.py @@ -59,11 +59,14 @@ button_names = { mod_table_operate.EditButton(bd_table.TableFieldDestiny.SUBSCRIBE_TYPE): "𝌴 Изменить тип в моей подписке", mod_table_operate.EditButton(bd_table.TableFieldDestiny.ITEM_ID): "𝌴 Изменить элемент в моей подписке", mod_table_operate.EditButton(bd_table.TableFieldDestiny.ACCESS): "✋ Изменить доступ к моей подписке", - mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ADD): "Добавление элемента", + mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ANY_ITEM_ADD): "Добавление элемента", mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ANY_ITEM_DEL): "Удаление какого либо элемента", mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ANY_ITEM_EDIT): "Редактирование какого либо элемента", mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ITEM_DEL): "Удаление определённого элемента", mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ITEM_EDIT): "Редактирование определённого элемента", + mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ANY_ITEM_ADD_WITH_PARENT): "Добавление элемента с родителем", + mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT): "Удаление элемента с родителем", + mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ANY_ITEM_EDIT_WITH_PARENT): "Редактирование элемента с родителем", mod_table_operate.ButtonNames.DEL: "❌ Удалить мою подписку", } @@ -102,7 +105,7 @@ messages = { mod_table_operate.CreateMessage(bd_table.TableFieldDestiny.ITEM_ID): ''' Создание подписки. Шаг №3 -Номер элемента, на который нужно подписаться (-1, если элемента нет): +Номер элемента или его родителя, на который нужно подписаться (-1, если элемента нет): ''', mod_table_operate.Messages.SUCCESS_CREATE: '''✅ Подписка успешно добавлена!''', mod_table_operate.Messages.START_EDIT: ''' @@ -143,13 +146,16 @@ messages = { mod_table_operate.Messages.SUCCESS_DELETE: '''✅ Подписка успешно удалёна!''', } -# TODO Возможно это не нужно есть же mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ADD) см. выше +# TODO Возможно это не нужно есть же mod_table_operate.EnumButton(bot_subscribes.SubscribeType.ANY_ITEM_ADD) см. выше messages_subs_type_status = { - mod_table_operate.EnumMessageForView(bot_subscribes.SubscribeType.ADD): f'''Добавление элемента''', + mod_table_operate.EnumMessageForView(bot_subscribes.SubscribeType.ANY_ITEM_ADD): f'''Добавление элемента''', mod_table_operate.EnumMessageForView(bot_subscribes.SubscribeType.ANY_ITEM_DEL): f'''Удаление элемента''', mod_table_operate.EnumMessageForView(bot_subscribes.SubscribeType.ANY_ITEM_EDIT): f'''Редактирование элемента''', mod_table_operate.EnumMessageForView(bot_subscribes.SubscribeType.ITEM_DEL): f'''Удаление конкретного элемента''', mod_table_operate.EnumMessageForView(bot_subscribes.SubscribeType.ITEM_EDIT): f'''Редактирование конкретного элемента''', + mod_table_operate.EnumMessageForView(bot_subscribes.SubscribeType.ANY_ITEM_ADD_WITH_PARENT): f'''Добавление элемента''', + mod_table_operate.EnumMessageForView(bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT): f'''Удаление элемента''', + mod_table_operate.EnumMessageForView(bot_subscribes.SubscribeType.ANY_ITEM_EDIT_WITH_PARENT): f'''Редактирование элемента''', } messages.update(messages_subs_type_status) diff --git a/bot_modules/tasks.py b/bot_modules/tasks.py index 1523adb..23f52a7 100644 --- a/bot_modules/tasks.py +++ b/bot_modules/tasks.py @@ -120,11 +120,14 @@ messages = { } messages_subscribes = { - mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ADD):f'''Задача создана''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD):f'''Задача создана''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Задача отредактирована''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Задача удалёна''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_EDIT):f'''Задача отредактирована #item_id''', mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ITEM_DEL):f'''Задача удалёна #item_id''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_ADD):f'''Задача создана''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_EDIT):f'''Задача отредактирована''', + mod_table_operate.SubscribeMessage(bot_subscribes.SubscribeType.ANY_ITEM_DEL):f'''Задача удалёна''', } messages.update(messages_subscribes) diff --git a/bot_sys/bot_subscribes.py b/bot_sys/bot_subscribes.py index db22367..91a3001 100644 --- a/bot_sys/bot_subscribes.py +++ b/bot_sys/bot_subscribes.py @@ -8,11 +8,14 @@ from enum import auto # Тип поля в таблице class SubscribeType(Enum): - ADD = auto() + ANY_ITEM_ADD = auto() ANY_ITEM_DEL = auto() ANY_ITEM_EDIT = auto() ITEM_DEL = auto() ITEM_EDIT = auto() + ANY_ITEM_ADD_WITH_PARENT = auto() + ANY_ITEM_DEL_WITH_PARENT = auto() + ANY_ITEM_EDIT_WITH_PARENT = auto() class BotSubscribes: def __init__(self): @@ -57,17 +60,17 @@ def Test(): mod_1 = 'proj' mod_2 = 'backup' s = BotSubscribes() - s.AddSubscribe(user_id_1, mod_1, SubscribeType.ADD) + s.AddSubscribe(user_id_1, mod_1, SubscribeType.ANY_ITEM_ADD) s.AddSubscribe(user_id_2, mod_2, SubscribeType.ITEM_DEL) - assert len(s.GetUserIDs(mod_1, SubscribeType.ADD)) == 1 + assert len(s.GetUserIDs(mod_1, SubscribeType.ANY_ITEM_ADD)) == 1 assert len(s.GetUserIDs(mod_1, SubscribeType.ANY_ITEM_DEL)) == 0 - assert user_id_1 in s.GetUserIDs(mod_1, SubscribeType.ADD) - assert not user_id_2 in s.GetUserIDs(mod_1, SubscribeType.ADD) - assert not user_id_3 in s.GetUserIDs(mod_1, SubscribeType.ADD) + assert user_id_1 in s.GetUserIDs(mod_1, SubscribeType.ANY_ITEM_ADD) + assert not user_id_2 in s.GetUserIDs(mod_1, SubscribeType.ANY_ITEM_ADD) + assert not user_id_3 in s.GetUserIDs(mod_1, SubscribeType.ANY_ITEM_ADD) assert len(s.GetUserIDs(mod_2, SubscribeType.ITEM_DEL)) == 1 - assert len(s.GetUserIDs(mod_2, SubscribeType.ADD)) == 0 + assert len(s.GetUserIDs(mod_2, SubscribeType.ANY_ITEM_ADD)) == 0 assert user_id_2 in s.GetUserIDs(mod_2, SubscribeType.ITEM_DEL) assert not user_id_1 in s.GetUserIDs(mod_2, SubscribeType.ITEM_DEL) assert not user_id_3 in s.GetUserIDs(mod_2, SubscribeType.ITEM_DEL) From c9a2687f4c4c966f10dc69d6989b3d9c9f359998 Mon Sep 17 00:00:00 2001 From: Alexei Date: Mon, 13 Nov 2023 11:06:44 +0300 Subject: [PATCH 14/21] =?UTF-8?q?=D0=98=D1=81=D0=BF=D1=80=D0=B0=D0=B2?= =?UTF-8?q?=D0=BB=D0=B5=D0=BD=D0=B0=20=D0=BE=D1=88=D0=B8=D0=B1=D0=BA=D0=B0?= =?UTF-8?q?,=20=D0=BA=D0=BE=D0=B3=D0=B4=D0=B0=20=D0=BF=D1=80=D0=B8=20?= =?UTF-8?q?=D0=BE=D1=82=D0=BA=D0=BB=D1=8E=D1=87=D0=B5=D0=BD=D0=BD=D0=BE?= =?UTF-8?q?=D0=BC=20=D0=B1=D0=BE=D1=82=D0=B5=20=D1=83=20=D1=82=D0=BE=D0=B3?= =?UTF-8?q?=D0=BE=20=D0=BA=D0=BE=D0=BC=D1=83=20=D0=B4=D0=BE=D0=BB=D0=B6?= =?UTF-8?q?=D0=BD=D0=BE=20=D0=BF=D1=80=D0=B8=D0=B4=D1=82=D0=B8=20=D1=81?= =?UTF-8?q?=D0=BE=D0=BE=D0=B1=D1=89=D0=B5=D0=BD=D0=B8=D0=B5=20=D0=BE=20?= =?UTF-8?q?=D0=BF=D0=BE=D0=B4=D0=BF=D0=B8=D1=81=D0=BA=D0=B5=20=D0=B2=D1=8B?= =?UTF-8?q?=D0=BA=D0=BB=D1=8E=D1=87=D0=B5=D0=BD.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/mod_table_operate.py | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/bot_modules/mod_table_operate.py b/bot_modules/mod_table_operate.py index eaae5c4..b68d56c 100644 --- a/bot_modules/mod_table_operate.py +++ b/bot_modules/mod_table_operate.py @@ -335,7 +335,10 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): a_Message = None user_groups = None parse_mode = None - await simple_message.SendMessage(self.m_Bot, a_BotMessage, a_GetButtonsFunc, a_GetInlineButtonsFunc, a_UserID, a_Message, user_groups, parse_mode=parse_mode) + try: + await simple_message.SendMessage(self.m_Bot, a_BotMessage, a_GetButtonsFunc, a_GetInlineButtonsFunc, a_UserID, a_Message, user_groups, parse_mode=parse_mode) + except: + return def SelectSourceTemplate(self, a_PrevPrefix, a_ButtonName): parent_id_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.PARENT_ID) From f39b738d22a536bddc60ea3016319ce06869a670 Mon Sep 17 00:00:00 2001 From: Fynjy Date: Mon, 13 Nov 2023 13:09:36 +0200 Subject: [PATCH 15/21] =?UTF-8?q?=D0=B4=D0=BE=D0=B1=D0=B0=D0=B2=D0=B8?= =?UTF-8?q?=D0=BB=20=D0=B2=20reauirements.txt=20=D0=BF=D0=B0=D0=BA=D0=B5?= =?UTF-8?q?=D1=82=D1=8B=20=D0=B8=20=D0=B7=D0=BC=D0=B5=D0=BD=D0=B8=D0=BB=20?= =?UTF-8?q?gitignore=20#5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 4 +++- requirements.txt | 18 +++++++++++++++++- 2 files changed, 20 insertions(+), 2 deletions(-) diff --git a/.gitignore b/.gitignore index 1798310..47793ab 100644 --- a/.gitignore +++ b/.gitignore @@ -5,4 +5,6 @@ log.txt bot.db adminpanelapp/migrations __init__.py -.env \ No newline at end of file +.env +.idea/ +venv/ diff --git a/requirements.txt b/requirements.txt index fe3741c..81a88a5 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,4 +1,20 @@ aiogram==2.20 +aiohttp==3.8.6 +aiosignal==1.3.1 +async-timeout==4.0.3 +attrs==23.1.0 +Babel==2.9.1 +certifi==2023.7.22 +charset-normalizer==3.3.2 colorama==0.4.5 Django==2.2.1 -python-dotenv +frozenlist==1.4.0 +idna==3.4 +multidict==6.0.4 +Pillow==10.1.0 +python-dotenv==1.0.0 +pytz==2023.3.post1 +requests==2.31.0 +sqlparse==0.4.4 +urllib3==2.0.7 +yarl==1.9.2 From e571a9c7404e2f3adae41cd4cbcc2cf9f45c7bdc Mon Sep 17 00:00:00 2001 From: Alexei Bezborodov Date: Tue, 14 Nov 2023 08:29:20 +0300 Subject: [PATCH 16/21] =?UTF-8?q?=D0=94=D0=BE=D0=B1=D0=B0=D0=B2=D0=BB?= =?UTF-8?q?=D0=B5=D0=BD=D0=B0=20=D0=B2=D0=BE=D0=B7=D0=BC=D0=BE=D0=B6=D0=BD?= =?UTF-8?q?=D0=BE=D1=81=D1=82=D1=8C=20=D1=80=D0=B5=D0=B4=D0=B0=D0=BA=D1=82?= =?UTF-8?q?=D0=B8=D1=80=D0=BE=D0=B2=D0=B0=D0=BD=D0=B8=D1=8F=20=D0=B4=D0=BE?= =?UTF-8?q?=D1=81=D1=82=D1=83=D0=BF=D0=B0=20=D0=BA=20=D0=BC=D0=BE=D0=B4?= =?UTF-8?q?=D1=83=D0=BB=D1=8E=20all=5Forders?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/all_orders.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/bot_modules/all_orders.py b/bot_modules/all_orders.py index efdbc2d..77d7fef 100644 --- a/bot_modules/all_orders.py +++ b/bot_modules/all_orders.py @@ -4,7 +4,7 @@ # Заказы from bot_sys import bot_bd, keyboard, user_access, bd_table -from bot_modules import mod_table_operate, mod_simple_message, orders +from bot_modules import mod_table_operate, mod_simple_message, orders, access_utils from template import bd_item_select, bd_item_view, bd_item # --------------------------------------------------------- @@ -113,7 +113,9 @@ class ModuleAllOrders(orders.ModuleOrders): def GetInitBDCommands(self): # уже сделано в ModuleUserOrders - return [] + return [ + access_utils.GetAccessForModuleRequest(self.GetName(), self.m_InitAccess, self.m_DefInitAccess), + ] def GetName(self): return module_name From 39b5e3c97dd0d7835e794d48ff3f355391e0bfeb Mon Sep 17 00:00:00 2001 From: Alexei Bezborodov Date: Tue, 14 Nov 2023 08:52:22 +0300 Subject: [PATCH 17/21] =?UTF-8?q?=D0=A3=D0=B4=D0=B0=D0=BB=D0=B5=D0=BD?= =?UTF-8?q?=D0=B8=D0=B5=20=D1=8D=D0=BB=D0=B5=D0=BC=D0=B5=D0=BD=D1=82=D0=BE?= =?UTF-8?q?=D0=B2=20=D0=BF=D0=BE=D0=BF=D1=80=D0=B0=D0=B2=D0=BB=D0=B5=D0=BD?= =?UTF-8?q?=D0=BE=20#11?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/mod_table_operate.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bot_modules/mod_table_operate.py b/bot_modules/mod_table_operate.py index b68d56c..1753bf4 100644 --- a/bot_modules/mod_table_operate.py +++ b/bot_modules/mod_table_operate.py @@ -105,7 +105,7 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): return self.GetButtonNameAndKeyValueAndAccess(a_Item) self.m_GetButtonNameAndKeyValueAndAccessFunc = GetButtonNameAndKeyValueAndAccess - async def PreDelete(a_CallbackQuery, a_Item, a_None): + async def PreDelete(a_CallbackQuery, a_Item, a_None, table_name = self.m_Table.GetName()): return await self.PreDelete(a_CallbackQuery, a_Item, a_None) self.m_PreDeleteFunc = PreDelete From 1ef958de161d3b66c487d58087490cd9bb94ee2b Mon Sep 17 00:00:00 2001 From: Alexei Bezborodov Date: Tue, 14 Nov 2023 08:58:27 +0300 Subject: [PATCH 18/21] =?UTF-8?q?=D0=9F=D1=80=D0=B0=D0=B2=D0=B8=D0=BB?= =?UTF-8?q?=D1=8C=D0=BD=D0=B0=D1=8F=20=D1=80=D0=B0=D0=B1=D0=BE=D1=82=D0=B0?= =?UTF-8?q?=20=D1=81=D0=BE=D0=B1=D1=8B=D1=82=D0=B8=D1=8F=20=D1=83=D0=B4?= =?UTF-8?q?=D0=B0=D0=BB=D0=B5=D0=BD=D0=B8=D0=B5=20=D1=81=20=D1=80=D0=BE?= =?UTF-8?q?=D0=B4=D0=B8=D1=82=D0=B5=D0=BB=D0=B5=D0=BC=20#11?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/mod_table_operate.py | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/bot_modules/mod_table_operate.py b/bot_modules/mod_table_operate.py index 1753bf4..4188812 100644 --- a/bot_modules/mod_table_operate.py +++ b/bot_modules/mod_table_operate.py @@ -259,11 +259,16 @@ class TableOperateModule(mod_simple_message.SimpleMessageModule): item_id = a_ItemID await self.SendSubscribe(subscribe_type, item_id, user_id) - parent_id_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.PARENT_ID) + table_name = self.m_Table.GetName() + key_name = self.GetKeyFieldName() - if parent_id_field and a_ItemData[parent_id_field]: + cur_item = GetCurItem(self.m_Bot, table_name, key_name, a_ItemID) + print(cur_item) + + parent_id_field_index = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.PARENT_ID) + if parent_id_field_index and cur_item and cur_item[parent_id_field_index]: subscribe_type = bot_subscribes.SubscribeType.ANY_ITEM_DEL_WITH_PARENT - item_id = a_ItemData[parent_id_field] + item_id = cur_item[parent_id_field_index] await self.SendSubscribe(subscribe_type, item_id, user_id) return simple_message.WorkFuncResult(self.GetMessage(Messages.SUCCESS_DELETE)) From 30aec002cb2d5bfd001106dab70c913872063392 Mon Sep 17 00:00:00 2001 From: Alexei Bezborodov Date: Tue, 14 Nov 2023 09:22:06 +0300 Subject: [PATCH 19/21] =?UTF-8?q?=D0=9F=D0=BE=D0=BF=D1=80=D0=B0=D0=B2?= =?UTF-8?q?=D0=BB=D0=B5=D0=BD=D0=B0=20=D0=BE=D1=82=D0=BF=D1=80=D0=B0=D0=B2?= =?UTF-8?q?=D0=BA=D0=B0=20=D1=81=D0=BE=D0=BE=D0=B1=D1=89=D0=B5=D0=BD=D0=B8?= =?UTF-8?q?=D0=B9=20=D0=BF=D0=BE=D0=BB=D1=8C=D0=B7=D0=BE=D0=B2=D0=B0=D1=82?= =?UTF-8?q?=D0=B5=D0=BB=D1=8E=20=D0=BF=D0=BE=D1=81=D0=BB=D0=B5=20=D0=B8?= =?UTF-8?q?=D0=B7=D0=BC=D0=B5=D0=BD=D0=B5=D0=BD=D0=B8=D1=8F=20=D1=81=D1=82?= =?UTF-8?q?=D0=B0=D1=82=D1=83=D1=81=D0=B0=20=D0=B7=D0=B0=D0=BA=D0=B0=D0=B7?= =?UTF-8?q?=D0=B0=20#11?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bot_modules/all_orders.py | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/bot_modules/all_orders.py b/bot_modules/all_orders.py index 77d7fef..350745a 100644 --- a/bot_modules/all_orders.py +++ b/bot_modules/all_orders.py @@ -134,6 +134,7 @@ class ModuleAllOrders(orders.ModuleOrders): return n + ":" + str(a_Item[parent_field_id]), k, a async def OnChangeField(self, a_Field, a_ItemID, a_ItemData, a_EditUserID): + super().OnChangeField(a_Field, a_ItemID, a_ItemData, a_EditUserID) if a_Field.m_Destiny == bd_table.TableFieldDestiny.STATUS: key_field = self.m_Table.GetFieldNameByDestiny(bd_table.TableFieldDestiny.KEY) item = bd_item.GetBDItemsTemplate(self.m_Bot, self.m_Table.GetName(), key_field)(a_ItemID) @@ -142,10 +143,10 @@ class ModuleAllOrders(orders.ModuleOrders): return item = item[0] - parent_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.PARENT_ID) + user_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.USER_ID) status_field_id = self.m_Table.GetFieldIDByDestiny(bd_table.TableFieldDestiny.STATUS) - print('OnChangeField', item, parent_field_id, status_field_id) - owner_id = item[parent_field_id] + print('OnChangeField', item, user_field_id, status_field_id) + owner_id = item[user_field_id] new_status = item[status_field_id] msg = self.GetMessage(mod_table_operate.NotificationMessage(new_status)) if not msg: From dc117cf2805973158e90ce5b48e2dd5f42f6520e Mon Sep 17 00:00:00 2001 From: Anton Date: Sun, 19 Nov 2023 23:01:46 +0300 Subject: [PATCH 20/21] =?UTF-8?q?=D0=B8=D0=B7=D0=BC=D0=B5=D0=BD=D0=B8?= =?UTF-8?q?=D0=BB=20requirements.txt=20#5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- requirements.txt | Bin 58 -> 666 bytes 1 file changed, 0 insertions(+), 0 deletions(-) diff --git a/requirements.txt b/requirements.txt index fe3741c67d9135707d93fd12a7b6e37c85559cd5..ef4ea686511a1ae93e71a29c37fdaab56ed5bf52 100644 GIT binary patch literal 666 zcmYk4QBH$k5QO*H#G_y+tyLfTVmyFXK#COu1)Njcs@`yW>sm)^0cx`e6s2&e0|4Bk%eOA7Jfl54(2G0$jYs$QkSeT3CWgg-eO% zk#D)J;HAzFKRNX_IF^knVhDZ4;f?8VQO>7}OTl^X`tNwe9w~b$#{+N1BaMP>a(5_U zC**4R)p+*G&aR?q&CNR$uxe+jrQG$ETIugy_r*Rr-_t`oy32Fwpt7y^GbyiprlEz} zGjGXU@FjfBRA?*jD@g6$KaMUOQ^V0oN4E!dYSB+{Y57(1{XMcn>zaFZL^mAmXv>bd zfyubB9)&Nmm3}teh5DF%2N*STQ$%(b;O(lavPO3u&82l5haZ4LBH^h~*2vJ&&s^TCpOhFk@eB^mj7x+(c3 Isd;5w02H7S9{>OV From 4fb37e59517fbf6e7e4c4badd6a389e052e66dfc Mon Sep 17 00:00:00 2001 From: Anton Date: Sun, 19 Nov 2023 23:12:18 +0300 Subject: [PATCH 21/21] =?UTF-8?q?=D1=81=D0=BB=D0=B8=D1=8F=D0=BD=D0=B8?= =?UTF-8?q?=D0=B5=20=D0=B2=D0=B5=D1=82=D0=BE=D0=BA=20master=20=D0=B8=20adm?= =?UTF-8?q?inpanel=5Fbackend?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- requirements.txt | Bin 666 -> 670 bytes 1 file changed, 0 insertions(+), 0 deletions(-) diff --git a/requirements.txt b/requirements.txt index ef4ea686511a1ae93e71a29c37fdaab56ed5bf52..1c85978c4a6644a4b3826630ea882164b44f39ba 100644 GIT binary patch delta 15 XcmbQmI*)b2Di(Vd50;IqJ}?3RDG&u* delta 10 ScmbQoI*WC}s*USDFaiJ=w*<`q