Ответы пользователя по тегу Python
  • Как правильно обрабатывать сообщения пользователя телеграмм боту?

    Vindicar
    @Vindicar
    RTFM!
    Ищи инфу по finite state machine (FSM) и как она реализуется в pyTelegramBotAPI.

    Если коротко - для каждого пользователя нужно хранить его текущий шаг (начало, выбор предмета, выбор задания, и т.п.) и ассоциированные с этим шагом данные. Т.е. нужно хранилище вида ключ-значение. Ключом будет ID пользователя.
    В качестве такого хранилища можно использовать и обычный словарь (если допустимо, чтобы бот всё забывал при перезапуске), или базу данных.

    И да, ты спалил токен бота. Меняй его теперь.
    Ответ написан
    1 комментарий
  • 'Handler' object is not callable как решить?

    Vindicar
    @Vindicar
    RTFM!
    callback_query_handlers
    Может, всё-таки callback_query_handler?
    Ответ написан
    Комментировать
  • Почему пайтон не видит глобальную переменную?

    Vindicar
    @Vindicar
    RTFM!
    1. Судя по отступам, у тебя get_answer() заканчивается на строке answer = input("Введите слово: ")
    2. Прежде чем упоминать переменную как глобал, она должна быть глобально определена, т.е. ей должно быть присвоено значение вне функции.
    3. Нафига тут вообще глобальная переменная?
    Ответ написан
  • Как цифру в строке преобразовать в числовой формат?

    Vindicar
    @Vindicar
    RTFM!
    Затем все приводится к формату строки типа ключ=значение

    Как можно значение оставить типа float?


    Во-первых, определись, о чем всё-таки речь - о формате, или о типе данных? Это разные вещи.
    Если о формате, что читай, как форматировать значения в f-строках.
    Если о типе, то ты просишь превратить число в строку, но оставить его числом. Т.е. взаимоисключающие вещи.
    "Нарисуйте семь перпендикулярных красных линий, но из них три синие должны быть параллельны, а ещё две - прозрачные."
    Ответ написан
    4 комментария
  • Как запустить выполнение асинхронной функции с определенной частотой выполнения?

    Vindicar
    @Vindicar
    RTFM!
    Храни в asyncio.Queue очередь запросов к API. Отдельная задача пусть выбирает запросы из очереди, отправляет, получает ответ и оповещает о результате. Например, так.
    import asyncio
    import typing
    
    class ThrottledResource:
        def __init__(self, delay: float):
            self._delay = delay
            self._queue = asyncio.Queue()
            self._task = None
        
        def start(self):
            self._task = asyncio.create_task(self._work_loop)
        
        def stop(self):
            self._task.cancel()
            self._task = None
    
        # этот метод вызывается клиентским кодом, получает параметры и возвращает отклик спустя время.
        async def query(self, params):
            future = asyncio.Future()  # Future просигналит, когда наш запрос будет обслужен
            await self._queue.put((future, params))
            result = await future  # корутина спит, пока запрос не обслужат
            return result
    
        async def _work_loop(self):
            while True:
                future, params = await. self._queue.get()  # ждем, пока не придёт запрос
                try:
                    result = await call_api(params)  # тут делаем асинхронное обращение к сервису
                except Exception as err:
                    future.set_exception(err)  # была ошибка - теперь await future выкинет исключение
                else:
                    future.set_result(result)  # полуен результат - await future вернёт его
                self._queue.task_done()  # каждому успешному get() соответствует task_done()
                asyncio.sleep(self._delay)  # можно учесть, сколько времени делался запрос. Но стоит ли?

    Код примерный, но идею передаёт. Использоваться будет как-то так
    api = ThrottledResource(delay=1.0)
    api.start()
    ...
    result = await api.query(params)  # await подождёт, пока не дойдёт очередь до нашего запроса


    Нужно добавить обработку ошибок, корректное завершение работы при наличии задач в очереди, и так далее.
    Технически вместо класса можно было реализовать это всё в виде декоратора над replier(), но это уже на вкус и цвет.
    Ответ написан
    6 комментариев
  • Почему пайчарм даёт ошибку, но когда делаю то, что он просит опять ошибка?

    Vindicar
    @Vindicar
    RTFM!
    @bot.callback_query_handler(func=lambda call: True)
    def work(call):


    Не надо описывать обработчики внутри других обработчиков. Это не будет работать так, как ты этого ожидаешь. Однажды прописанный, обработчик остаётся до конца работы программы.
    И уж ТОЧНО не нужно делать самому вызов work(call).
    Ответ написан
  • Могу ли я сделать блокировщик сайтов без использования файла "Hosts" и его изменения?

    Vindicar
    @Vindicar
    RTFM!
    Поднять свой как-бы DNS сервер на машине, прописать этот DNS сервер как основной и единственный для данной машины, предыдущие прописанные сервера сохранить. Админские права на этом этапе понадобятся.
    При поступлении запроса проверять запрашиваемое имя. Если не в чёрном списке - перенаправлять запрос на сохранённый DNS, полученный ответ отдать самому.
    Грубо и не всегда надёжно (есть DNS over HTTPS, например, который так не перехватишь), но в базовом варианте сработает.
    Ответ написан
    Комментировать
  • Как записать for, if-else в одну строку?

    Vindicar
    @Vindicar
    RTFM!
    Если я верно понял, что требуется...
    [(even, uneven)[arg % 2].append(arg) for arg in args]
    Ответ написан
    3 комментария
  • Не понимаю суть ошибки, можете объяснить?

    Vindicar
    @Vindicar
    RTFM!
    from sqlite import db_start, create_profile, edit_profile
    from main import balance


    main импортирует sqlite, который импортирует main, который импортирует sqlite, который импортирует main, который импортирует sqlite, который импортирует main, который импортирует sqlite, который импортирует main, который импортирует sqlite, который импортирует main, который импортирует sqlite, который импортирует main, который импортирует sqlite, который импортирует main, который импортирует sqlite, который импортирует main, который импортирует sqlite, который импортирует main, который импортирует sqlite, который импортирует main, который ...
    Ну ты понял.

    Быстрого волшебного решения тут нет. Разбирайся с областями видимости, избавляйся от глобальных переменных.
    Например, почему у тебя при создании нового профиля используется balance? Т.е. новый пользователь получает столько денег, сколько было у последнего активного?
    Почему вообще у тебя вещи, специфичные для одного пользователя, лежат то в глобальных переменных, то в данных машины состояний (state)? Бери их из БД , когда они потребуются.
    Ответ написан
    6 комментариев
  • Как решить ошибку RuntimeWarning: Enable tracemalloc to get the object allocation traceback?

    Vindicar
    @Vindicar
    RTFM!
    Выше же RuntimeWarning: coroutine XXXXX was never awaited.
    await позабывал. Читай документацию на методы, которыми пользуешься, чтобы знать, как их запускать.
    Ответ написан
    Комментировать
  • Как можно реализовать обработку текста?

    Vindicar
    @Vindicar
    RTFM!
    Читай про finite state machine (FSM) в aiogram.
    Если коротко, надо отслеживать и хранить, в каком меню находится каждый пользователь.
    Ответ написан
    Комментировать
  • Как проверить ячейку бд на наличие конкретных данных Sql?

    Vindicar
    @Vindicar
    RTFM!
    Не стоит полагаться на возвращаемое значение cursor.execute().
    Выхлоп твоего запроса нужно получать через fetchone()/fetchall(), даже если запрос вернул одно значение.
    У тебя тогда в выдаче будет одна строка с одним столбцом. Выбираешь строку через fetchone(), у полученного значения берёшь нулевой столбец.
    Ответ написан
    Комментировать
  • Почему Python выдает ""(" was not closed"? хотя все нормально?

    Vindicar
    @Vindicar
    RTFM!
    У тебя в строке одинаковые одинарные кавычки - как вокруг всей строки, так и вокруг значения href. Так что https://www.google.com/ оказалось вне строковой константы. Будь внимательнее. И вообще, в HTML кавычки должны быть двойные.
    Ответ написан
    Комментировать
  • Почему дублируется слэш и не работают сырые строки?

    Vindicar
    @Vindicar
    RTFM!
    Обрати внимание, как ты выводишь значение.
    В питоне есть два строковых представления: str и repr.
    str - человекочитаемое представление. Его можно увидеть, сделав просто print(path). Сразу станет видно, что ничего не удваивается.
    repr, же, по идее, представляет объект так, что если его записать прямо в коде, как показано - получим обратно этот объект. Если это вообще возможно.
    В случае с числами разницы нет.
    В случае со строками - она будет. У тебя на скриншоте именно repr-представление строки - в кавычках, и с экранированием спецсимволов. При этом, записав такую константу, получишь ту же самую строку, что логично.
    Ведь 'a\\b' - это то же самое, что и r'a\b'.

    Короче, не парься, всё работает как надо. Просто имей ввиду, в каком виде у тебя выводятся значения. Кавычки должны были сразу заставить заподозрить неладное.
    Ответ написан
    Комментировать
  • Два одинаковых кода, но один не хочет работать! Почему?

    Vindicar
    @Vindicar
    RTFM!
    for i in range(word_count, 0, -1):
    На первой итерации цикла i примет значение word_count.
    Символы в строке word имеют индексы от 0 до word_count-1.
    Так что индекс, равный word_count, не существует. Ты поймаешь исключение IndexError.

    И можно ли его написать более компактно? Если да, то как?

    Освоить срезы (slice). part = word[a:b:c] будет (условно) эквивалентно
    part = ''
    i = a
    while i < b:
        part = part + word[i]
        i = i + c

    для положительного c. А для отрицательного будет while i > b:.
    Если a опущено, по умолчанию это начало последовательности (для c > 0) или конец (для c < 0).
    Если b опущено, по умолчанию это конец последовательности (для c > 0) или начало (для c < 0).
    Если c опущено, по умолчанию это 1.

    Так что для обращения строки нам нужно пройтись по всей строке с шагом -1. Получим вот что:
    word_reversed = word[::-1]
    # это эквивалентно
    word_reversed = word[len(word)-1:-1:-1]
    Ответ написан
    1 комментарий
  • Как соединить 2 python файла в один код?

    Vindicar
    @Vindicar
    RTFM!
    У тебя будет проблема в том, что во второй файл надо будет передать объект bot, которого, скорее всего, не будет на момент импорта этого файла.
    Есть не слишком изящный, но простой способ.
    # imported_file.py
    import telebot
    from telebot import types
    
    def setup(bot):  # setup() принимает бота и прочее как параметры.
        # да, прописываем обработчики прямо внутри setup()
        # но вот глобальные переменные здесь прописывать не стоит
        @bot.message_handler(content_types=['text'])
        def get_text_messages(message):
            ...
    
    # main_file.py
    import telebot
    from telebot import types
    import time
    
    import imported_file
    
    bot = telebot.TeleBot("ТОКЕН")
    # убедись, что setup() вызывается строго один раз для каждого дополнительного файла!
    imported_file.setup(bot)  
    ...  # дальше работаем как обычно


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

    Vindicar
    @Vindicar
    RTFM!
    просто в папке не создается таблица бд и на этом все.

    создает товар и сохраняет в бд без ошибок.

    Дай угадаю, коннект к базе делаешь командой вида sqlite3.connect('somefile.db')?
    Если так, то ты налетел на проблему относительных путей. Основы информатики, начальная школа.
    Путь somefile.db будет рассчитываться относительно текущего рабочего каталога, который может меняться в зависимости от способа запуска скрипта, да и просто в ходе его работы. А может не меняться. Т.е. нет гарантий, что текущий рабочий каталог будет совпадать с каталогом, где лежит скрипт.
    Лучше определить абсолютный путь к файлу самостоятельно.

    from pathlib import Path
    import sys
    
    SCRIPT_DIR = Path(sys.argv[0]).parent.resolve()  # каталог, где лежит выполняемый скрипт
    DB_PATH = SCRIPT_DIR / 'somefile.db'  # путь к файлу БД в этом каталоге
    
    conn = sqlite3.connect(DB_PATH)
    Ответ написан
    Комментировать
  • Ошибка: sqlite3.OperationalError: no such column: telegram_id: Как исправить?

    Vindicar
    @Vindicar
    RTFM!
    CREATE TABLE IF NOT EXISTS не сработает, если таблица уже существует - но и только. Если она имеет неправильную структуру, данный вызов её не пересоздаст, а оставит как есть. Проверь структуру таблицы, либо через обращение к таблице схемы (хранит структуру всех таблиц к БД), либо подцепившись клиентом к твоей БД.
    Ответ написан
    Комментировать
  • Почему не изменяется значение переменной в python?

    Vindicar
    @Vindicar
    RTFM!
    class Enemy:
        ...
        def mobReturn(self): 
            self.x_mob = self.rect.x
    
    class Bullet(pygame.sprite.Sprite):
        def update(self):
            if x_mob>=100:

    В первом случае ты изменяешь атрибут экземпляра класса Enemy.
    Во втором случае ты читаешь локальную переменную метода update() класса Bullet.
    Это совершенно разные области видимости, они друг о друге ничего не знают.

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

    Во всех случаях проверку коллизии нужно будет делать каждый кадр. Так что в главном цикле игры надо будет перебирать активных врагов и активные пули, и проверять их. В простейшем случае - всех со всеми.
    Ответ написан
  • Вылет Python при возврате c_char_p из dll ctypes. Что не так?

    Vindicar
    @Vindicar
    RTFM!
    У тебя возвращаемая переменная размещена на стеке. Возможно, string умирает по выходу из функции, и уносит с собой выделенную строку, так что указатель к моменту возврата из функции уже ни на что не указывает.

    Если обратиться к функции из под C++, она сработает?
    # include <fstream>
    # include <string>
    using namespace std;
    
    const char* readfile(char* filename) {
      ifstream file;
      string lines, line;
      
      file.open(filename);
      if (file.is_open()) {
        while (file >> line) {
          lines += line;
        }
        file.close();
        return lines.c_str();
      } else {
        char* trashVar = "Bad open";
        return trashVar;
      }
    }
    
    int main()
    {
        cout << readfile("filename.txt");
        return 0;
    }
    Ответ написан
    4 комментария