Ответы пользователя по тегу Боты
  • Как сделать так, чтобы бот отправлял текст в таком формате который я ему отправляю?

    Vindicar
    @Vindicar
    RTFM!
    caption отдаёт только plaintext (без форматирования). Вроде как есть свойство-список caption_entities, содержащее сведения о форматировании.
    Либо попробуй указать это свойство при отправке сообщения, параметром entities = message.caption_entities, либо разбирай элементы этого списка, ищи форматирование, и вставляй его вручную (там указывается позиция в строке, где форматирование начинается/заканчивается).
    Ответ написан
    3 комментария
  • Как бот должен понимать что нужно нажимать кнопку во время qte?

    Vindicar
    @Vindicar
    RTFM!
    Template matching попробуй.
    Ответ написан
    Комментировать
  • Зачем нужен декоратор @dp и все в этом духе, типо @dp.message_handler() в aiogram?

    Vindicar
    @Vindicar
    RTFM!
    На пальцах: чтобы бот работал, его функции должны вызываться при наступлении определённых событий (например, входящего сообщения). Проблема в том, что aiogram знает о возможных событиях, но ему нужно сказать, какие функции когда вызывать. Обычно это называется "зарегистрировать обработчик".
    В питоне функции - объекты первого рода, т.е. их можно сохранять в переменные, передавать как параметры, возвращать как результат и вообще поступать с ними как с любым другим значением.
    Т.е. по идее можно было бы сделать так:
    async def echo(message: Message):
        text = f"Привет, ты написал {message.text}"
        await bot.send_message(chat_id=message.from_user.id, text=text)
    
    dp.register_function_for_message(echo)  # это не настоящий метод aiogram, только пример

    Тогда каждому объявленному обработчику событий соответствовал бы вызов метода, ответственного за регистрацию этого обработчика.

    Но то же самое можно сделать через декораторы. Декоратор - это сокращённый вызов функции, которая принимает в качестве параметра другую функцию.
    Т.е. вот это
    @dp.message_handler()
    async def echo(message: Message):
        text = f"Привет, ты написал {message.text}"
        await bot.send_message(chat_id=message.from_user.id, text=text)

    абсолютно эквивалентно вот этому
    _decorator = dp.message_handler()
    
    async def echo(message: Message):
        text = f"Привет, ты написал {message.text}"
        await bot.send_message(chat_id=message.from_user.id, text=text)
    
    echo = _decorator(echo)

    Как видишь, очень похоже на "простую" регистрацию обработчика, но записывается чуть короче.
    Ответ написан
  • Как реализовать отправку сообщения ботом при обновлении бд?

    Vindicar
    @Vindicar
    RTFM!
    pochka3648, сделай своё API. Например, первый бот слушает сокет, второй к нему коннектится, и пишет в сокет данные.
    Если не требуется мгновенное оповещение, пусть первый бот периодически проверяет таблицу на наличие новых записей.
    Ответ написан
    Комментировать
  • Почему не работает код дальше?

    Vindicar
    @Vindicar
    RTFM!
    dp.register_callback_query_handler(go1, lambda call: True)

    Потому что ты сказал "А всё, что не обрабатывается вышележащими обработчиками, пускай обрабатывает go1". Как следствие, до go2 дело не доходит.
    Чини lambda call: ..., чтобы правильно определял, какие события отдавать go1.
    Ответ написан
    Комментировать
  • Как сделать что бы бот выдавал сколько пользователь в боте?

    Vindicar
    @Vindicar
    RTFM!
    В обработчике команды /start записывать в базу текущую дату (например, в виде timestamp).
    По запросу узнать текущую дату, получить дату из базы, вычесть, перевести в дни.
    См. модуль datetime.
    Ответ написан
    Комментировать
  • Как узнать и зафиксировать время, когда пользователь нажал inline-кнопку?

    Vindicar
    @Vindicar
    RTFM!
    Использовать time.time() не хочется, чтобы избежать конфликта с Телеграм при неправильно установленном времени на локальной машине пользователя.

    Тогда только спрашивать время у какого-то доверенного сервера, по NTP или иному протоколу.
    Ответ написан
  • Как сделать инлайн кнопки?

    Vindicar
    @Vindicar
    RTFM!
    Чего ты не можешь понять?
    Каждый вызов row() принимает в параметрах те кнопки, которые должны быть в одном ряду.
    Просто сделай нужное количество вызовов row(), передав каждому те кнопки, которые должны быть в соответствующем ряду.
    btn1 = InlineKeyboardButton(......)
    btn2 = InlineKeyboardButton(......)
    btn3 = InlineKeyboardButton(......)
    ...
    keyboard.row(btn1, btn2)
    keyboard.row(btn3)
    ...
    Ответ написан
    Комментировать
  • Как можно сочетать aiogram с aioshedule?

    Vindicar
    @Vindicar
    RTFM!
    while True:
            await aioschedule.run_pending()

    Из-за этого обработчик события не завершится нормально. Вынеси этот цикл отдельно, и запусти его один раз через asyncio.create_task() при старте бота.
    Ответ написан
    Комментировать
  • Телеграм-бот. Как сохранить второе сообщение от пользователя?

    Vindicar
    @Vindicar
    RTFM!
    У тебя должен быть отдельный обработчик входящих сообщений, а не часть обработчика нажатия на кнопку..
    При этом бот должен помнить, ведёт ли данный пользователь игру в настоящий момент, или нет.
    Ответ написан
  • Почему может выдавать ошибку в Python Bot?

    Vindicar
    @Vindicar
    RTFM!
    await message.delete
    Наверно, всё-таки await message.delete()?
    Ответ написан
    Комментировать
  • Возможно-ли добавить в бота телеграмм (pyTelegramBotAPI), подставление картинки поверх основной картинки?

    Vindicar
    @Vindicar
    RTFM!
    Возможно.
    За готовым кодом - на фриланс. Сюда - с конкретными вопросами по реализации.
    Ответ написан
  • Как обслуживать юзеров по очереди телебот?

    Vindicar
    @Vindicar
    RTFM!
    global user_id
    Ну как бы вот виновник. Глобальные переменные. Они общие на всего бота и на всех пользователей. Если тебе нужно дифференцировать пользователей - делай это сам.
    Делай хранилище "ключ-значение", где ключ - ID пользователя, значение - хранимые для этого пользователя данные. При обработке сообщения читай ID пользователя, доставай из хранилища данные, и уже сообразно им обрабатывай сообщение.
    Таким хранилищем может быть простой словарь (если не нужно сохранять состояние между перезапусками бота), БД или ещё что.

    Не знаю, есть ли в telebot реализация Finite State Machine, но обычно для реализации сценариев используют её. Идея та же - для каждого юзера храним его текущее состояние (шаг в сценарии) и ассоциированные с ним данные.
    Ответ написан
    2 комментария
  • Как создать список из картинок и текста и выводить рандомным способом?

    Vindicar
    @Vindicar
    RTFM!
    Ну про random.choice() ты знаешь. В чём проблема-то?
    Если нужно соответствие картинки и текста, то просто храни текст и картинку одним элементом списка - например, через кортеж.
    tarot_list = [
        ('image1.jpg', 'Текст 1'),
        ('image2.jpg', 'Текст 2'),
    ]

    Тогда потом просто делаешь:
    image, text = random.choice(tarot_list)
    и вперёд!
    Ответ написан
  • Как убрать лишние знаки?

    Vindicar
    @Vindicar
    RTFM!
    Ещё один не выучил основы языка, но пытается работать с БД.

    fectchone() возвращает кортеж, в твоём случае кортеж с одним элементом. Скобки - это строковое представление кортежа. Как вытащить значение из кортежа - читай по ссылке.

    Кортежи, списки и словари - это вообще азы языка. Их надо знать, и узнавать с первого взгляда.
    Ответ написан
    Комментировать
  • Почему не работает bot_func.stop(Client, module Pyrogram)?

    Vindicar
    @Vindicar
    RTFM!
    loop = asyncio.new_event_loop()
    asyncio.set_event_loop(loop)

    Зачем?!
    Если у тебя бот асинхронный, просто сделай саму функцию GetUserByUsername() async.
    Ответ написан
  • Не выполгяется последняя часть кода @bot.callback_query_handler(func=lambda call:True) def callback2(call): почему и как исправить?

    Vindicar
    @Vindicar
    RTFM!
    @bot.callback_query_handler(func=lambda call:True)
    func говорит боту, когда вызывать обработчик. Бот вызывает только ОДИН обработчик (первый подходящий).
    func=lambda call:True означает, что обработчик должен вызываться для ВСЕХ кнопок.
    Научись различать обработчики по call.data. Например, пусть у одной группы кнопок data начинается с "foo.", а у другой - с "bar.".
    item = types.InlineKeyboardButton('4', callback_data='foo.question1')
    item2 = types.InlineKeyboardButton('3', callback_data='foo.question2')

    gotov = types.InlineKeyboardButton('Готов', callback_data='bar.gotov')

    Тогда ты сможешь прописать два обработчика:
    @bot.callback_query_handler(func=lambda call:call.data.startswith('foo.'))

    и
    @bot.callback_query_handler(func=lambda call:call.data.startswith('bar.'))


    Разумеется, вместо foo и bar можно придумать свои префиксы, в том числе многоуровневые (типа callback_data='questions.q1.answer1').
    Ответ написан
    1 комментарий
  • Как правильно одновременно запустить двух Telegram ботов одной программой на Python?

    Vindicar
    @Vindicar
    RTFM!
    Ну потому что ты фигню написал.
    th_bot = Thread(target=pyrobot(), args=())
    th_userbot = Thread(target=aiobot(), args=())

    Ты пытаешься запустить в качестве потока значение, возвращаемое функцией pyrobot(). А так как она уходит в цикл и значения не возвращает, то далее ничего не происходит. До вызова конструктора Thread() дело не доходит. С aiobot() аналогично.

    Еще раз:
    pyrobot() - вызов функции
    pyrobot - ссылка на функцию

    EDIT:
    Оба бота - асинхронные на базе asyncio, им для работы нужен цикл-реактор (event loop). Вообще не факт, что хорошая идея запускать их в потоках.
    Тут есть два варианта, сразу даже не скажу, что проще.
    Вариант А, лобовой: каждый бот создаёт своё собственные реактор через asyncio.new_event_loop(), потом задаёт его как текущий для своего потока через asyncio.set_event_loop(loop). Если ботам не требуется взаимодействовать, то это может быть проще. Если требуется... будут проблемы. Два реактора в одной программе - это не хорошо.

    Вариант Б, адекватный:
    И вызов app.run(), и вызов executor.start_polling(dp, skip_updates=True) скорее всего под капотом создают асинхронную функцию (корутину), и запускают её в реакторе. Тогда ты можешь обойтись без потоков, заставив обоих ботов работать на одном реакторе. Нужно будет зарыться в доки, или даже глянуть исходники.

    Например, для пирограмма написано такое:
    When calling this method (app.run()) without any argument it acts as a convenience method that calls start(), idle() and stop() in sequence. It makes running a single client less verbose.

    Т.е. вместо вызова app.run() ты можешь изменить код так:
    async def pyrobot():  # обрати внимание, теперь функция асинхронная!
        print("pyro started")
        @app.on_message(filters.chat("some_chat"))
        async def print_pyrogram():
            print("Pyrogram")
        # это вместо вызова app.run(), как написано в доках.
        await app.start()
        try:
            await app.idle()
        finally:
            await app.end()

    Затем делаешь аналогичный трюк с aiobot(). Нужно посмотреть в доках на аиограм, как именно.

    И потом запускаешь обоих ботов кодом вида...
    asyncio.run(asyncio.gather(pyrobot(), aiobot()))
    Ответ написан
    3 комментария
  • Как отправлять текст соответствующий картинке в телеграм боте на Python?

    Vindicar
    @Vindicar
    RTFM!
    1. вынеси random.choice(os.listdir('test')) в переменную, например, img
    2. используй os.path.splitext(os.path.basename(img))[0], чтобы получить имя файла картинки без расширения
    3. Не забудь отправить файл через photo = open('test/' + img, 'rb')
    4. PROFIT
    Ответ написан
    2 комментария