Задать вопрос
  • Как исправить код на Python?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Сейчас запущено два бота с одним токеном одновременно либо, как вариант, что-то идёт не так при запросе, и бот делает второй-третий-четвёртый запрос getUpdates слишком часто. Но я думаю что скорее всего первый вариант, ибо это довольно частая ошибка новичков.
    Ответ написан
    Комментировать
  • Почему хандлер не срабатывает повторно?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Этот бот очень странно написан, и вообще непонятно, как в нём хоть что-то работает.

    Начнём с того, что вот это:

    global operator
    operator = call.message.chat.id


    будет означать, что в переменной operator всегда будет id чата последнего оператора, нажавшего кнопку join_dialog_btn. Аналогично будут проблемы с user_help_id. В итоге не будет это одновременно работать ни с несколькими операторами, ни с несколькими пользователями.

    Далее, НЕ НАДО создавать в функции другие функции без чёткого понимания, зачем это нужно, и тем более не надо их оборачивать регистрирующими декораторами. Это работает совсем не так, как кажется.

    Когда оператор Вася получит своего первого пользователя, будет вызван декоратор @bot.message_handler(func=lambda message: message.from_user.id == Вася). После того, как Вася отработает по пользователю и получит следующего, будет создан ещё один обработчик с таким же декоратором, но так как один такой обработчик (с такими условиями) уже в цепочке есть, то будет вызван первый, который полагается на локальные переменные от предыдущего вызова supportWait. Довольно заметно, что ты с этим уже сталкивался и именно поэтому локальные переменные (operator и user_help_id) сделал глобальными, но это, разумеется, не поможет, так как абсолютно неправильно.

    Что тут надо сделать?

    1. Вынести все обработчики на верхний уровень и избавиться от дурной практики регистрации их через декораторы внутри функции каждый раз при их вызове.

    2. Отделить логику оператора от логики пользователя - это следует делать отдельными обработчиками со своей логикой.

    3. Где-то хранить соответствие chat_id пользователя - chat_id назначенного оператора. Можно завести словарь, но в целом можно и через FSM Телебота (но тогда нельзя будет отслеживать, что пользователя уже кто-то взял).

    4. Не забывать, что пользователь может успеть оставить несколько сообщений до того, как оператор примет запрос - их все нужно не потерять.

    Можно сделать примерно так:

    spoiler
    operator_ids = [список id операторов]
    
    @bot.message_handler(func=lambda message: message.chat.id in operator_ids)
    def handle_operator_message(message):
    # обработка сообщения оператора в состоянии, когда ему ничего не назначено
    # можно например ругнуться чтобы ждал появления обращений
    
    # состояния пользователей, в частности чтобы накапливать сообщения от начала обращения
    # до назначения оператора
    user_states = {}
    
    @bot.message_handler(func=lambda message: True)
    def handle_user_message(message):
    # обработка сообщения пользователя в состоянии, когда он только что обратился
        # посылаем сообщение с кнопкой всем операторам
        # (задача со звёздочкой: всем операторам, у которых никто не назначен)
        # в кнопке передаём callback_data=join_dialog_btn:CHAT_ID_ПОЛЬЗОВАТЕЛЯ
        # запоминаем сообщение
        user_states[message.chat.id] = {"messages":[message]}
        # затем регистрируем отдельный обработчик
        bot.register_next_step_handler(message, handle_user_unassigned)
    
    # обработчик собирает сообщения пользователя
    def handle_user_unassigned(message, user_data):
        user_states[message.chat.id]["messages"].append(message)
    
    # обработчик сообщений пользователя после назначения оператора
    # этот обработчик выставится из обработчика сообщения оператора.
    # используя register_next_step_handler_by_chat_id
    def handle_user_assigned(message, operator_id):
        # посылаем сообщение оператору (operator_id)
    
    def handle_operator_assigned(message, user_id):
        # посылаем сообщение юзеру (user_id)
    
    # в кнопке передаётся chat_id пользователя
    # в принципе, можно в кнопку засунуть больше данных, в том числе json до 64 символов
    @bot.callback_query_handler(func=lambda call: call.data.startswith('join_dialog_btn:'))
        user_chat_id = int(call.data.split(":")[1])
        for operator_id in operators_id:
            if operator_id != call.chat.id:
                 # посылаем остальным операторам "пользователя взял в работу оператор Вася"
        # также выплёвываем все накопленные сообщения пользователя этому оператору
        for msg in user_states[user_chat_id]["messages"]:
             # тут делаем forward_message
        # выставляем обработчик оператору
        bot.register_next_step_handler(message, handler_operator_assigned, user_chat_id)
        # теперь надо переключить пользователя на оператора, используя его chat_id
        bot.register_next_step_handler_by_chat_id(user_chat_id, handle_user_assigned, call.chat.id)
        # обрати внимание, что это выставляет обработчик юзеру, а не оператору!
    
    # в итоге всей этой конструкции в чате operator_id действует обработчик handle_operator_assigned
    # с параметром user_id, а в чате user_id - handle_user_assigned с параметром operator_id, и можно
    # в них сразу посылать полученные сообщения (только не forward_message, конечно,
    # чтобы не светить имя оператора)


    Это очень примерно, нужно много чего дорабатывать, включая сценарии завершения работы, переключения на другого оператора итд итп, контроль за тем, чтобы пользователя не взяли два оператора одновременно. А если бот асинхронный/многопоточный/на вебхуках - то ещё чтобы не возникло особых спецэффектов из-за пересечения взаимодействий...

    spoiler
    Важно отметить, что полноценные взрослые чат-сервисы делают совершенно не так. В них обычно даже не используется Telegram, вместо этого оператор работает через web-интерфейс (а может и отдельное приложение), в котором можно контролировать всю историю взаимодействия, переключаться между пользователями (нормальная ситуация если оператор ведёт нескольких пользователей параллельно), подтягивать данные из CRM/клиентской базы, интегрировать справочники типовых ответов или чат-ботов, итд итп. Также, например, нормальные обработки таймаутов (если пользователь или оператор не отвечает в нужный срок...)

    И делать это лучше, например, на брокерах очередей (типа rabbitmq). Чтобы не нужно было вот этого замороченного собирания сообщений до назначения оператора в какой-то список и не было риска, что пользователь успеет что-то ещё написать в процессе назначения оператору.
    Ответ написан
  • Проблема с ботом discord на python?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    С некоторых пор API Discord больше не позволяет по умолчанию ботам видеть содержимое сообщений. Нужно явно запросить message_content intents:

    intents.message_content = True

    А также надо включить privileged intents в developer portal для этого бота.
    Ответ написан
    3 комментария
  • Как сделать так, что бы бот на disnake увидел мемберов?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Необходимо добавить member intents. А также включить privileged intents в настройках бота в developer portal. Документация. Вообще, примеров на эту тему в интернете море.
    Ответ написан
    Комментировать
  • Как получить цвет пользователя в telegram на python?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Если в Bot API нет такой информации, то никак. А в Bot API её на текущйи момент вообще нет. Можно только узнать о наличии premium-подписки. https://core.telegram.org/bots/api#user
    Ответ написан
    2 комментария
  • Как перенаправить вывод strace в tail?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Нет, так сделать нельзя, но есть утилиты, которыми можно принять через stdin лог и автоматически его ротировать по размеру (в том числе удаляя старые файлы). Например, multilog.

    Примеры есть тут: https://superuser.com/questions/291368/log-rotatio...

    Но в целом strace - это так себе средство отладки, тем более скриптов, лучше логгировать в самом скрипте, потому что в strace много ненужного мусора. Как вариант, можно с помощью опций ограничить системные вызовы, которые он будет логгировать.
    Ответ написан
    Комментировать
  • Можно ли получать обновления с телеграм канала через Telegram API без добавления бота в этот канал?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    С использованием Bot API - только если бота добавит админ канала. На чужой канал бот подписаться не может.

    С помощью клиентского API можно подписываться на чужие каналы и получать с них обновления, но это делается с реквизитами пользователя (своего) и канал должен быть публичным, чтобы на него можно было легко подписаться.

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

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Это спам и всё делается ради вон того файла во вложении. Текст такой чтобы повысить его уникальность и усложнить его ловлю спам-фильтром.
    Ответ написан
    Комментировать
  • Как удалить своего бота из чужой группы Телеграм?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Да, это можно, используя метод leaveChat. Достаточно id группы знать.

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

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    В Bot API - никак. Этот API работает на событиях и не позволяет получать информацию по ранее отправленным сообщениям, и события вида "что-то изменилось в числе просмотров" не предусмотрено.

    Только пользоваться клиентским API (MTProto) на базе библиотеки telethon или pyrogram. Изображать из себя обычного пользователя и смотреть историю сообщений в канале.
    Ответ написан
    Комментировать
  • Есть ли универсальный модуль Python для работы с разными базами данных (mysql, postgresql, ...)?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    sqlalchemy вполне можно использовать как абстрактный интерфейс к базам данных, просто передавая в него строковые запросы без ORM.
    Ответ написан
    1 комментарий
  • Как автоматически проверить доступность телефонного номера?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Для мобильных телефонов искать по слову HLR, есть сервисы, которые позволяют делать HLR-запросы в сеть GSM с помощью API. Однако надо понимать, что работает это всегда надёжно, может давать ложноположительные и ложноотрицательные результаты, может работать не по всем операторам.

    Самый надёжный способ - совершить звонок и сбросить.

    Вообще, лучше было бы рассказать исходную задачу.
    Ответ написан
    6 комментариев
  • Что ему не нравится с моим буффером?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Думаю, ошибка тут в том, что в процессе выполнения receiver_func значение offset также успевает измениться внутри worker_func, что приводит к выходу за пределы буфера, несмотря на явную проверку offset.

    Вообще, надо всегда помнить, что полноценная синхронизация тредов только на mutex невозможна. Между двумя блокировками mutex одним тредом другой тред может успеть выполнить несколько блокировок. Нужно также использовать condvar.

    В целом кажется, что этот код написан в попытках сделать хоть что-то параллельно, а не для решения какой-то реальной задачи.
    Ответ написан
    Комментировать
  • Сколько места доступно докер контейнеру на linux?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Docker по умолчанию использует "слоённую" файловую систему overlayfs. В ней каждый образ состоит из нескольких слоёв, которые перекрывают друг друга (read only). Плюс собственный слой контейнера (read write). Поэтому если никакой большой записи приложение в контейнере не ведёт, то образ на диске хранится один раз и дополнительно место почти не расходуется. Это нормально, это так и задумано.
    Ответ написан
    Комментировать
  • Как сделать ссылку на телеграм с текстом?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Никак. Возможности deep linking у Телеграма довольно разнообразны, но передать текст пользователю никак нельзя. Можно только передать текст боту, причём исключительно как аргумент команды start.

    Подробнее в официальной документации: https://core.telegram.org/api/links
    Ответ написан
    Комментировать
  • Как почистить папку с 246к папок по ssh?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Можно использовать xargs, он сам поделит список аргументов, переданных через stdin, на такие блоки, чтобы они вписывались в лимит. Чтобы не было спецэффектов от всяких особых символов, можно использовать find -print0 | xargs -0 (символ с кодом 0 будет разделять имена файлов вместо стандартного переноса строки):

    find условия -print0|xargs -0 rm -rf

    Однако GNU find сам умеет удалять файлы с помощью -delete:

    find условия -delete

    Ещё может быть проблема, что подобная операция помешает работе сайтов из-за нагрузки на диск, это можно решить с помощью приоритетов для процессов (nice) и ввода-вывода (ionice), тогда удаление будет по возможности выполняться в фоне:

    nice -n19 ionice -c3 команда

    (Если используется find|xargs, то это нужно передать xargs)
    Ответ написан
    Комментировать
  • Старый коммит отвязался от ветки origin/master. Как его туда вернуть?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Почему это "отвязался"? Всё на месте https://github.com/boluchevskii/englishmakerNEW/co...

    Конечно же, HEAD в master и origin/master указывает только на faaefef - так и должно быть. Не может HEAD указывать на два разных коммита.
    Ответ написан
    Комментировать
  • Почему запрос SQL не проходит без GROUP BY?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Потому что это азы SQL: если в выражении есть и агрегирующая функция, и просто неагрегируемые значения, то необходимо явным образом указать, что нужно по ним группировать.
    Ответ написан
    3 комментария
  • Как ускорить рассылку через phpmailer?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    500-1000 секунд это 6-12 минут примерно. Это вообще не тот масштаб, за который имеет смысл волноваться. Рассылается и ладно...
    Ответ написан
    Комментировать
  • Как принимать сообщения с вотсапп?

    shurshur
    @shurshur
    Сисадмин, просто сисадмин...
    Есть два способа (на самом деле три):

    1. Пойти по легальному пути. Зарегать бизнес-аккаунт у Facebook, пройти верификацию, зарегистрировать WhatsApp-номер, подключить интеграцию через какого-нибудь провайдера API и работать через него. Характерный признак - только для бизнесов, не очень гуманная цена (5-10-15 тыс. рублей только за подключение - легко), рассылка возможна только по одобренным заранее шаблонам. Но зато легально. Для серьёзного бизнеса с большим количеством клиентов наилучший вариант.

    2. Пойти по серому пути. У WhatsApp есть Web-версия, народ её расковыривает и учится посылать в Web запросы напрямую, без браузера. Есть как сторонние поставщики такого доступа с вменяемым API (их много, не буду рекламить), так и готовые реализации, в том числе на гитхабе (но надо искать актуальную, чтобы работала). Характерный признак такой интеграции - нужно сканировать QR-код для привязки веб-аккаунта к мобильному приложению, а если пользоваться сторонним посредником, то его цена невелика и не имеет жёстких лимитов по числу сообщений. Есть некоторый риск бана аккаунта, поскольку официально так нельзя, но если быть очень осторожным и не спамить (только отвечать на входящие) - то риск бана минимальный (но я бы всё равно не советовал свой основной и очень важный номер в это впутывать). Такое нередко используют всякие мелкие интернет-магазинчики.

    3. Есть ещё оооочень серый путь. На самом деле можно найти полученные реверс-инжинирингом реализации протокола WA, а также ключи, выдранные из официального клиента последних версий. Но за такое очень легко и быстро можно получить бан аккаунта, так что настоятельно не рекомендую (один знакомый потестировал и получил бан аккаунта - к счастью, не основного - чуть ли не в тот же день).

    Для первого пути надо смотреть API конкретного поставщика (они разные), для второго найти подходящую актуальную библиотеку с недавними обновлениями или подходящего агрегатора с его собственным API. Третий - лучше не надо.

    PS: Flask, ngrok - это всё фигня. Надо начать с того, через что делать эту интеграцию. В браузерном WhatsApp Web нет никакого flask, но сообщения ходят.
    Ответ написан
    5 комментариев