import requests import aiohttp from aiogram import Bot, Dispatcher, types from aiogram.contrib.fsm_storage.memory import MemoryStorage from aiogram.dispatcher.filters.state import State, StatesGroup from aiogram.utils import executor import re from aiogram.types import ReplyKeyboardMarkup, KeyboardButton import asyncio from aiogram.dispatcher import FSMContext from aiogram import types from aiogram.dispatcher.filters import Command from datetime import datetime, timedelta import gspread from oauth2client.service_account import ServiceAccountCredentials import re from aiogram.types import InlineKeyboardMarkup, InlineKeyboardButton, WebAppInfo import logging from aiogram.types import ReplyKeyboardRemove TOKEN = '7907319434:AAEEM55xbWHhp5uPAlJ5AS2vNoleES_z_Kk' bot = Bot(token=TOKEN) storage = MemoryStorage() dp = Dispatcher(bot, storage=storage) # Кнопка для перезапуска restart_button = ReplyKeyboardMarkup(resize_keyboard=True) restart_button.add(KeyboardButton("Перезапустить бота")) MAX_ATTEMPTS = 3 # Максимальное количество попыток ввода кода TIMEOUT = 30 # Время в секундах # Словарь для хранения данных о пользователях user_data = {} last_bot_message_id = {} # Авторизация и доступ к таблице scope = ["https://spreadsheets.google.com/feeds", "https://www.googleapis.com/auth/drive"] credentials = ServiceAccountCredentials.from_json_keyfile_name("botforclinic-436512-0c117dd103a8.json", scope) client = gspread.authorize(credentials) sheet = client.open("Пациенты клиники").sheet1 # Открытие таблицы, используйте название вашей таблицы # Словарь для хранения данных о сообщениях пользователей user_messages = {} # История состояний для каждого пользователя user_states = {} class Form(StatesGroup): telegram_id = State() existing_patient = State() phone_number = State() fio = State() birthday = State() verification_code = State() confirm_fio = State() confirm_phone_number = State() # Создаем состояния class QuestionForm(StatesGroup): waiting_for_question = State() from aiogram import types # Настройка логирования logging.basicConfig(level=logging.INFO) # Главное меню main_menu = InlineKeyboardMarkup(row_width=2).add( InlineKeyboardButton("🌍 Контакты", callback_data="menu_contacts"), InlineKeyboardButton("📞 Связь", callback_data="menu_contact") ) # Подменю "контакты" contacts_menu = InlineKeyboardMarkup(row_width=1).add( InlineKeyboardButton("▶️ Назад", callback_data="main_menu") ) # Меню связь contact_menu = InlineKeyboardMarkup(row_width=1).add( InlineKeyboardButton("💬 Отправить сообщение администратору Клиники", callback_data="contact_administrator"), InlineKeyboardButton("🛎 Заказать звонок от администратора Клиники", callback_data="order_call"), InlineKeyboardButton("▶️ Назад", callback_data="main_menu"), ) administrator_contact = InlineKeyboardMarkup(row_width=1).add( InlineKeyboardButton("✍🏼 Написать сообщение", callback_data="message_admin"), InlineKeyboardButton("▶️ Назад", callback_data="menu_contact"), InlineKeyboardButton("🔘 Главное меню", callback_data="main_menu") ) message_ad = InlineKeyboardMarkup(row_width=1).add( InlineKeyboardButton("▶️ Назад", callback_data="contact_administrator"), InlineKeyboardButton("🔘 Главное меню", callback_data="main_menu") ) call_order = InlineKeyboardMarkup(row_width=1).add( InlineKeyboardButton("📞 Позвонить мне", callback_data="call_me"), InlineKeyboardButton("▶️ Назад", callback_data="menu_contact"), InlineKeyboardButton("🔘 Главное меню", callback_data="main_menu") ) return_call_me = InlineKeyboardMarkup(row_width=1).add( InlineKeyboardButton("▶️ Назад", callback_data="order_call"), InlineKeyboardButton("🔘 Главное меню", callback_data="main_menu") ) whatsapp = InlineKeyboardMarkup(row_width=1).add( InlineKeyboardButton("🔘 Главное меню", callback_data="main_menu") ) # Обработка нажатий на инлайн-кнопки @dp.callback_query_handler(lambda c: c.data.startswith("menu_") or c.data == "main_menu") async def handle_main_menu(call: types.CallbackQuery): if call.data == "main_menu": await call.message.edit_text("В настоящее время я, как Ваш виртуальный помощник, активно развиваюсь." " В ближайшее время у меня появятся новые функции и возможности." " А пока Вы можете ознакомиться с текущим меню.", reply_markup=main_menu) elif call.data == "menu_contacts": await call.message.edit_text("Клиники им. проф. Е.Н.Оленевой\n\n" "Клиника УХО•ГОРЛО•НОС\n" "Записаться на приём, задать вопрос:\n" "📞 +7 (342) 207-03-03\n\n" "Больше информации о клинике и врачах на нашем 🔗САЙТЕ\n" "📍г. Пермь, ул. К. Цеткин, 9\n" "(ост. Строительный факультет)\n" "Построить маршрут\n\n" "--------------\n\n" "Клиника лечения КАШЛЯ•АЛЛЕРГИИ\n" "Записаться на приём, задать вопрос:\n" "📞 +7 (342) 200-02-03\n\n" "Больше информации о клинике и врачах на нашем 🔗САЙТЕ\n\n" "📍г. Пермь, ул. Пермь, ул. Г. Звезда, 31А\n" "(ост. Г. Звезда)\n" "Построить маршрут\n\n" "--------------\n\n" "Центр ДИАГНОСТИКИ И РЕАБИЛИТАЦИИ \n" "📞 +7 (342) 287-16-94\n\n" "Больше информации о клинике и врачах на нашем 🔗САЙТЕ\n\n" "📍 г. Пермь, ул. Пермь, ул. Г. Звезда, 31А\n" " (ост. Г. Звезда)\n" "Построить маршрут\n\n" "Присоединяйтесь к нашему каналу новостей в Telegram.", reply_markup=contacts_menu, parse_mode="HTML") elif call.data == "menu_contact": await call.message.edit_text("Выберите способ связи", reply_markup=contact_menu) # Обработка нажатия на "Запись на исследование" @dp.callback_query_handler(lambda c: c.data == "contact_administrator") async def handle_call_menu(call: types.CallbackQuery): await call.message.edit_text("💬 Для отправки сообщения администратору Клиники нажмите кнопку ниже 'Написать сообщение'.", reply_markup=administrator_contact) # Обработка нажатия на "Запись на исследование" @dp.callback_query_handler(lambda c: c.data == "order_call") async def handle_admin_call_menu(call: types.CallbackQuery): await call.message.edit_text("Для получения обратного звонка от администратора Клиники нажмите кнопку ниже 'Позвонить мне'.", reply_markup=call_order) @dp.callback_query_handler(lambda c: c.data == "call_me") async def handle_ad_call_menu(call: types.CallbackQuery): try: user_id = call.from_user.id # ID пользователя records = sheet.get_all_records() # Получаем данные пользователя из таблицы user_data = next((entry for entry in records if str(entry['Telegram ID']) == str(user_id)), None) if not user_data: await call.message.answer("Ваши данные не найдены.", reply_markup=return_call_me) return phone_number = user_data['Номер телефона'] text = "Позвонить мне" message_id = call.message.message_id date = call.message.date.isoformat() # Конвертация даты reply_to_message_id = call.message.reply_to_message.message_id if call.message.reply_to_message else None # Передаём данные на сервер await send_message_server(phone_number, text, message_id, user_id, date, reply_to_message_id) await call.message.edit_text( "Администратор Клиники свяжется с Вами в ближайшее время в рабочие часы и проконсультирует по всем вопросам.", reply_markup=return_call_me ) except Exception as e: logging.error(f"Ошибка при обработке запроса: {e}") await call.message.edit_text("Произошла ошибка. Попробуйте снова позже.", reply_markup=return_call_me) # Обработчик кнопки "message_admin" @dp.callback_query_handler(lambda c: c.data == "message_admin") async def handle_ad_message(call: types.CallbackQuery, state: FSMContext): user_id = str(call.from_user.id) # Преобразуем в строку для сравнения try: # Получение всех записей из таблицы records = sheet.get_all_records() # Поиск пользователя по Telegram ID user_data = next((entry for entry in records if str(entry['Telegram ID']) == user_id), None) if not user_data: await call.message.answer("Ваши данные не найдены.", reply_markup=message_ad) return # Переход в состояние ожидания вопроса prompt_message = await call.message.edit_text( "Пожалуйста, отправьте Ваш вопрос администратору одним сообщением.", reply_markup=message_ad ) # Сохраняем ID сообщения async with state.proxy() as data: data["prompt_message_id"] = prompt_message.message_id await QuestionForm.waiting_for_question.set() # Устанавливаем состояние ожидания= except Exception as e: logging.error(f"Ошибка при доступе к таблице: {e}") await call.message.edit_text("Произошла ошибка. Попробуйте снова позже.", reply_markup=message_ad) # Обработчик ввода вопроса @dp.message_handler(state=QuestionForm.waiting_for_question, content_types=types.ContentTypes.TEXT) async def handle_user_question(message: types.Message, state: FSMContext): user_id = str(message.from_user.id) # ID пользователя text = message.text # Сообщение пользователя try: # Удаление предыдущего сообщения async with state.proxy() as data: if "prompt_message_id" in data: await bot.delete_message(chat_id=message.chat.id, message_id=data["prompt_message_id"]) # Отправка сообщения пользователя на сервер await send_message_server2(user_id, text, message.message_id, user_id, message.date.isoformat(), None) # Ответ пользователю await message.answer( "Ваше сообщение успешно отправлено! Администратор Клиники ответит Вам в ближайшее время в рабочие часы. Спасибо за Ваше обращение!", reply_markup=message_ad ) except Exception as e: logging.error(f"Ошибка при обработке вопроса: {e}") await message.answer( "Произошла ошибка при отправке вопроса. Попробуйте позже.", reply_markup=message_ad ) # Завершаем состояние await state.finish() # Хранилище для уже обработанных пользователей processed_users = set() # Обработка нажатия на кнопку "Проверить данные" @dp.callback_query_handler(lambda c: c.data == 'check_verification') async def process_check_verification(callback_query: types.CallbackQuery, state: FSMContext): user_id = callback_query.from_user.id try: # Чтение всех строк таблицы records = sheet.get_all_records() user_data_verified = False for row in records: fio = str(row.get("ФИО", "")).strip() phone = str(row.get("Номер телефона", "")).strip() telegram_id = str(row.get("Telegram ID", "")).strip() # Проверяем, если в таблице есть данные для этого пользователя if telegram_id == str(user_id): user_data_verified = True break if user_data_verified: await send_success_message(user_id) # Отправляем сообщение о успешной регистрации else: await bot.send_message( user_id, "Мы не нашли ваших данных в системе. Пожалуйста, проверьте, что вы правильно заполнили форму и попробуйте снова.", reply_markup=InlineKeyboardMarkup().add( InlineKeyboardButton("Попробовать снова", callback_data="check_verification") ) ) except Exception as e: print(f"Ошибка при проверке верификации: {e}") await bot.send_message(user_id, "Произошла ошибка при проверке данных. Попробуйте позже.") # Функция отправки сообщения об успешной регистрации async def send_success_message(user_id): try: await bot.send_message( chat_id=user_id, text=("Ура! Ваша регистрация прошла успешно! 🥳\n\n" "Теперь все уведомления о записи на приём будут приходить сюда, в чат-бот. " "Если Вы также получаете рассылку в WhatsApp, то сообщения будут дублироваться. " "Для получения рассылки только в чат-бот Telegram, нажмите кнопку ниже 'Не дублировать сообщения в WhatsApp'."), reply_markup=InlineKeyboardMarkup().add( InlineKeyboardButton("Не дублировать сообщения в WhatsApp", callback_data="stop_replying"), InlineKeyboardButton("🔘 Главное меню", callback_data="main_menu") ) ) except Exception as e: print(f"Ошибка при отправке сообщения пользователю {user_id}: {e}") @dp.callback_query_handler(lambda c: c.data == "stop_replying") async def stop_replying_handler(call: types.CallbackQuery): user_id = str(call.from_user.id) # Преобразуем в строку для сравнения try: # Получение всех записей из таблицы records = sheet.get_all_records() # Поиск пользователя по Telegram ID user_data = next((entry for entry in records if str(entry['Telegram ID']) == user_id), None) if not user_data: await call.message.answer("Ваши данные не найдены.", reply_markup=whatsapp) return phone_number = user_data['Номер телефона'] # Измените на название вашего столбца text = "Отключить дублирование в WhatsApp" message_id = call.message.message_id date = call.message.date.isoformat() reply_to_message_id = call.message.reply_to_message.message_id if call.message.reply_to_message else None # Отправка данных на сервер await send_message_server(phone_number, text, message_id, call.from_user.id, date, reply_to_message_id) # Ответ пользователю await call.message.edit_text("Рассылка уведомлений в WhatsApp отключена.", reply_markup=whatsapp) except Exception as e: logging.error(f"Ошибка при доступе к таблице: {e}") await call.message.edit_text("Произошла ошибка. Попробуйте снова позже.", reply_markup=whatsapp) @dp.message_handler(commands=['help']) async def handle_help(message: types.Message): #await save_message_data(message) user_id = message.from_user.id help_message = "Список доступных команд:\n" help_message += "/start - начать взаимодействие с ботом\n" help_message += "/registration - регистрация\n" help_message += "/help - показать список команд и их описания\n" sent_message = await bot.send_message(user_id, help_message) last_bot_message_id[user_id] = sent_message.message_id @dp.message_handler(Command("start"), state="*") async def handle_start(message: types.Message, state: FSMContext): # Завершаем текущее состояние и очищаем данные пользователя await state.finish() # Завершает текущее состояние await state.storage.reset_data(user=message.from_user.id) # Удаляет все данные пользователя в FSM хранилище # Отправляем фото with open("s-blob-v1-IMAGE-tdNCrEv8Ldo.png", "rb") as photo: await bot.send_photo( chat_id=message.from_user.id, photo=photo, caption=("Добро пожаловать!\nЯ чат-бот Клиники и Ваш надежный виртуальный помощник.\nЧтобы узнать, что я могу для Вас сделать, просто введите /help\nДля начала работы нажмите кнопку /registration внизу."), reply_markup=generate_markup_registration() # Кнопка для продолжения ) @dp.message_handler(commands=['registration'], chat_type=types.ChatType.PRIVATE) async def handle_registration(message: types.Message): user_id = message.from_user.id sent_message = await bot.send_message(user_id, "Вы уже являетесь пациентом нашей Клиники?\nЕсли Вы хотите начать общение с ботом заново, нажмите /start.", reply_markup=yes_no_markup()) last_bot_message_id[user_id] = sent_message.message_id await Form.existing_patient.set() def generate_markup_telegram_id(): markup = types.ReplyKeyboardMarkup(row_width=2, resize_keyboard=True) item3 = types.KeyboardButton("Отправить Telegram ID") markup.add(item3) return markup def generate_markup_registration(): markup = types.ReplyKeyboardMarkup(row_width=1, resize_keyboard=True) item_registration = types.KeyboardButton("/registration") markup.add(item_registration) return markup def generate_back_button(): """Создает клавиатуру с кнопкой 'Назад'.""" markup = ReplyKeyboardMarkup(row_width=1, resize_keyboard=True) back_button = KeyboardButton("Назад") # Текст кнопки "Назад" markup.add(back_button) return markup @dp.message_handler(state=Form.existing_patient) async def handle_existing_patient(message: types.Message, state: FSMContext): user_id = message.from_user.id form_url = f"https://tgbotpolimed.pirogov.ai/?user_id={user_id}" if not message.text.strip(): await message.answer( "Пожалуйста, выберите 'Да' или 'Нет' с помощью кнопок.", reply_markup=ReplyKeyboardRemove() ) return if message.text.lower() == "да": await message.answer( "В соответствии с Федеральным законом № 152-ФЗ «О персональных данных», для идентификации Вас как пациента нашей Клиники, просим заполнить форму, нажав на кнопку ниже.", reply_markup=ReplyKeyboardRemove() ) await message.answer( "Ввод персональных данных", reply_markup=InlineKeyboardMarkup().add( InlineKeyboardButton( text="Заполнить форму", web_app=WebAppInfo(url=form_url) ) ) ) # Отправляем кнопку для проверки данных await message.answer( "После того как Вы заполните форму, нажмите на кнопку ниже для проверки данных.\n" "Если Вы хотите начать общение с ботом заново, нажмите /start.", reply_markup=InlineKeyboardMarkup().add( InlineKeyboardButton( text="Проверить данные", callback_data="check_verification" ) ) ) await state.finish() elif message.text.lower() == "нет": await bot.send_message( user_id, "Полный доступ к функциям сервиса предоставляется только пациентам нашей Клиники.\n\n" "Вы можете связаться с нами по тел.: +7 (342) 207-03-03 или посетить Клинику лично для заключения договора по следующим адресам:\n" "г. Пермь, ул. Клары Цеткин, д. 9; ул. Газеты Звезда, д. 31-а.\n\n" "Будьте здоровы!\n" "Если Вы хотите начать общение с ботом заново, нажмите /start.", reply_markup=generate_back_button() # Клавиатура с кнопкой "Назад" ) await state.finish() else: await bot.send_message( user_id, "Пожалуйста, выберите 'Да' или 'Нет' с помощью кнопок.", reply_markup=ReplyKeyboardRemove() ) await state.finish() # Обработчик кнопки "Назад" @dp.message_handler(lambda message: message.text == "Назад") async def handle_back_button(message: types.Message): await handle_registration(message) # Вызываем функцию обработки команды /registration def find_patients_by_id(telegram_id): records = sheet.get_all_records() # Получаем все строки таблицы patients = [] for record in records: if str(record["Telegram ID"]) == str(telegram_id): patients.append(record) return patients async def prompt_patient_selection(user_id, patient_records): # Создаем кнопки с ФИО пациентов markup = ReplyKeyboardMarkup(row_width=1, resize_keyboard=True) for record in patient_records: markup.add(KeyboardButton(record["ФИО"])) # Сохраняем данные пациентов временно user_data[user_id]["patients"] = patient_records await bot.send_message( user_id, "Найдено несколько записей с вашим Telegram ID. Пожалуйста, выберите пациента:", reply_markup=markup, ) await Form.fio.set() async def save_message_data(message: types.Message, reply_to_message_id=None): user_id = message.from_user.id reply_to_message_id = last_bot_message_id.get(user_id, None) user_messages[message.message_id] = { "message_id": message.message_id, "text": message.text, "date": message.date.isoformat(), # Исправлено "user_id": message.from_user.id, "reply_to_message_id": reply_to_message_id } print(f"Сообщение пользователя {user_id} сохранено: {user_messages[message.message_id]}") await send_message_to_server(user_messages[message.message_id]) async def send_message_to_server(message_data): url = "http://192.168.1.10:8080/AppSaveMessage" headers = {"Content-Type": "application/json"} async with aiohttp.ClientSession() as session: async with session.post(url, headers=headers, json=message_data) as response: if response.status == 200: print(f"Сообщение {message_data['message_id']} успешно отправлено на сервер.") else: print(f"Ошибка при отправке сообщения {message_data['message_id']} на сервер. Статус-код: {response.status}") # Функция для выполнения POST-запроса и получения списка врачей def get_doctors(): url = "http://192.168.1.10:8080/AppZaprSpecDoc" # Данные для отправки в запросе data = {"spec_id": 1} # Выполняем POST-запрос response = requests.post(url, json=data) # Проверяем статус ответа if response.status_code == 200: # Если запрос успешен, возвращаем данные return response.json() else: # Если запрос не удался, выводим ошибку print(f"Ошибка: {response.status_code}") return [] def yes_no_markup(): markup = types.ReplyKeyboardMarkup(row_width=2, resize_keyboard=True) item_no = types.KeyboardButton("Нет") item_yes = types.KeyboardButton("Да") markup.add(item_yes, item_no) return markup def markup2(): markup = types.ReplyKeyboardMarkup(row_width=2, resize_keyboard=True) item_good = types.KeyboardButton("Всё верно") item_no_good = types.KeyboardButton("Указать ФИО ещё раз") markup.add(item_good, item_no_good) return markup def generate_start_markup(): markup = types.ReplyKeyboardMarkup(row_width=1, resize_keyboard=True) start_button = types.KeyboardButton("/start") markup.add(start_button) return markup async def send_message_server(phone_number, text, message_id, user_id, date, reply_to_message_id=None): url = "http://192.168.1.10:8080/AppSaveMessage" headers = {"Content-Type": "application/json"} message_data = { "message_id": message_id, "text": text, "date": date, "user_id": user_id, # Передача ID пользователя "reply_to_message_id": reply_to_message_id, # Если нужно указать reply_to_message_id "phone_number": phone_number } async with aiohttp.ClientSession() as session: async with session.post(url, json=message_data, headers=headers) as response: if response.status == 200: print(message_data) logging.info("Сообщение успешно отправлено на сервер") else: logging.error(f"Ошибка при отправке сообщения: {response.status}") async def send_message_server2(user_id, text, message_id, from_user_id, date, reply_to_message_id): url = "http://192.168.1.10:8080/AppSaveMessage" headers = {"Content-Type": "application/json"} message_data = { "user_id": user_id, "text": text, "message_id": message_id, "from_user_id": from_user_id, "date": date, "reply_to_message_id": reply_to_message_id, } async with aiohttp.ClientSession() as session: async with session.post(url, json=message_data, headers=headers) as response: if response.status == 200: logging.info("Сообщение успешно отправлено на сервер") else: logging.error(f"Ошибка при отправке сообщения: {response.status}") def find_patient_by_id(telegram_id): records = sheet.get_all_records() # Получаем все строки таблицы for record in records: if str(record["Telegram ID"]) == str(telegram_id): # Сравнение с Telegram ID phone_raw = record["Номер телефона"] # Удаляем все нецифровые символы phone_formatted = re.sub(r"\D", "", phone_raw) # Проверка на правильную длину номера и корректировка, если необходимо if len(phone_formatted) == 11: if phone_formatted.startswith("8"): # Если номер начинается с 8, заменяем на 7 phone_formatted = "7" + phone_formatted[1:] return phone_formatted else: return None # Возвращаем None, если номер неправильной длины return None def authorize_google(): # google_json = {} # for key in config["GOOGLE"].keys(): # google_json[key] = config["GOOGLE"][key] scope = ["https://spreadsheets.google.com/feeds", "https://www.googleapis.com/auth/drive"] creds = ServiceAccountCredentials.from_json_keyfile_name("botforclinic-436512-0c117dd103a8.json", scope) # creds = ServiceAccountCredentials.from_json_keyfile_dict(google_json, scope) client = gspread.authorize(creds) return client @dp.message_handler(content_types=types.ContentType.TEXT) async def handle_message(message: types.Message): await save_message_data(message) if __name__ == '__main__': executor.start_polling(dp, skip_updates=True)