FastAPI выдает сообщение «RecursionError: превышена максимальная глубина рекурсии при вызове объекта Python»

Я пытаюсь создать приложение, в котором хранятся данные некоторых пользователей и их записи о продажах. Существует register_user API, который отправляет пользовательские данные в базу данных.

Однако когда я запускаю API в localhost:8000/docs после запуска python -m main:app, я получаю RecursionError, хотя у меня нет никаких рекурсивных вызовов в реализации.

Код в main.py выглядит следующим образом:

from fastapi import FastAPI, Depends, Request
from sqlalchemy import create_engine, Column, Integer, String
from sqlalchemy.orm import sessionmaker, Session
from sqlalchemy.ext.declarative import declarative_base
from passlib.context import CryptContext


SQLALCHEMY_DB_URL = "sqlite:///./records.db"

engine = create_engine(
    SQLALCHEMY_DB_URL, connect_args = {"check_same_thread": False}
)

SessionLocal = sessionmaker(autocommit=False, 
                            autoflush=False,
                            bind=engine)

Base = declarative_base()


class Users(Base):
    __tablename__ = 'users'

    user_id = Column(Integer, primary_key=True, index=True)
    business_name = Column(String, nullable=False)
    phone_number = Column(String, nullable=False)
    email = Column(String, nullable=False)
    hashed_password = Column(String, nullable=False)


Base.metadata.create_all(bind=engine)


app = FastAPI()

bcrypt_context = CryptContext(schemes=['bcrypt'], deprecated='auto')



def get_password_hash(password: str):
    return bcrypt_context.hash(password)

def get_db():
    try:
        db = SessionLocal()
        yield db
    finally:
        db.close()

@app.post('/register') 
async def register_user(request:Request,
                        email: str, 
                        business_name: str,
                        phone_number: str,
                        password: str,
                        password2: str,
                        db: Session = Depends(get_db)
                        ):
    try:
        validation1 = db.query(Users) \
                        .filter(Users.phone_number == phone_number) \
                        .first()
        
        validation2 = db.query(Users) \
                        .filter(Users.email == email) \
                        .first()

        if password != password2 or validation1 is not None or validation2 is not None:
            msg = "Invalid Registration Request"
            return {"request": request, 
                "msg": msg}
        
        user_model = Users()
        user_model.email = email
        user_model.phone_number = phone_number
        user_model.business_name = business_name
        user_model.hashed_password = get_password_hash(password=password)

        db.add(user_model)
        db.commit()

        msg = "User Successfully Created"

        return {"request": request, 
            "msg": msg}

    except Exception as e:

        print(f"Error occurred: {e}")
        msg = "Internal Server Error"
        return {"request": request, "msg": msg}

Код должен был зафиксировать нового пользователя в базе данных, однако код просто терпит неудачу с некоторыми ошибками, которые я не могу расшифровать.

Обратная трассировка выглядит следующим образом. Первые несколько строк:

INFO:     127.0.0.1:52561 - "POST /auth/register?email=email%40email.com&business_name=business&phone_number=1234567890&password=pass123&password2=pass123 HTTP/1.1" 500 Internal Server Error
ERROR:    Exception in ASGI application
Traceback (most recent call last):
  File "C:\Users\<User>\sales_register\env\lib\site-packages\uvicorn\protocols\http\h11_impl.py", line 398, in run_asgi
    result = await app(  # type: ignore[func-returns-value]
  File "C:\Users\<User>\sales_register\env\lib\site-packages\uvicorn\middleware\proxy_headers.py", line 70, in __call__
    return await self.app(scope, receive, send)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\fastapi\applications.py", line 1054, in __call__
    await super().__call__(scope, receive, send)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\applications.py", line 123, in __call__
    await self.middleware_stack(scope, receive, send)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\middleware\errors.py", line 186, in __call__
    raise exc
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\middleware\errors.py", line 164, in __call__
    await self.app(scope, receive, _send)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\middleware\exceptions.py", line 65, in __call__
    await wrap_app_handling_exceptions(self.app, conn)(scope, receive, send)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\_exception_handler.py", line 64, in wrapped_app
    raise exc
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\_exception_handler.py", line 53, in wrapped_app
    await app(scope, receive, sender)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\routing.py", line 756, in __call__
    await self.middleware_stack(scope, receive, send)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\routing.py", line 776, in app
    await route.handle(scope, receive, send)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\routing.py", line 297, in handle
    await self.app(scope, receive, send)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\routing.py", line 77, in app
    await wrap_app_handling_exceptions(app, request)(scope, receive, send)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\_exception_handler.py", line 64, in wrapped_app
    raise exc
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\_exception_handler.py", line 53, in wrapped_app
    await app(scope, receive, sender)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\starlette\routing.py", line 72, in app
    response = await func(request)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\fastapi\routing.py", line 296, in app
    content = await serialize_response(
  File "C:\Users\<User>\sales_register\env\lib\site-packages\fastapi\routing.py", line 180, in serialize_response
    return jsonable_encoder(response_content)
  File "C:\Users\<User>\sales_register\env\lib\site-packages\fastapi\encoders.py", line 289, in jsonable_encoder
    encoded_value = jsonable_encoder(
  File "C:\Users\<User>\sales_register\env\lib\site-packages\fastapi\encoders.py", line 333, in jsonable_encoder
    return jsonable_encoder(
  File "C:\Users\<User>\sales_register\env\lib\site-packages\fastapi\encoders.py", line 289, in jsonable_encoder
    encoded_value = jsonable_encoder(
  File "C:\Users\<User>\sales_register\env\lib\site-packages\fastapi\encoders.py", line 333, in jsonable_encoder
    return jsonable_encoder( 

Последние 3 строки здесь повторяются много раз. Наконец закрываю с

  File "C:\Users\<User>\sales_register\env\lib\site-packages\fastapi\encoders.py", line 216, in jsonable_encoder
    if isinstance(obj, BaseModel):
  File "C:\Users\<User>\sales_register\env\lib\site-packages\pydantic\_internal\_model_construction.py", line 248, in __instancecheck__
    return hasattr(instance, '__pydantic_validator__') and super().__instancecheck__(instance)
RecursionError: maximum recursion depth exceeded while calling a Python object

Ваши предложения будут очень полезны.

Если необходима кодовая база, я бы согласился поделиться ею.

Покажите хотя бы первые и последние несколько строк обратной трассировки как правильно отформатированный текст (как код) в вопросе.

Michael Butscher 10.08.2024 11:17

Вы считаете, что именно эта функция рекурсивна?

SIGHUP 10.08.2024 13:10

@MichaelButscher Я сейчас добавил обратную трассировку. Надо было добавить это в первую очередь. Извинения.

mraabhijit 10.08.2024 14:38

@SIGHUP Я не совсем понимаю, откуда это взялось. Это единственная функция, которая вызывается в API вместе с другой, которая представляет собой просто оператор возврата с вызовом CryptContext.hash().

mraabhijit 10.08.2024 14:42
Почему в Python есть оператор "pass"?
Почему в Python есть оператор "pass"?
Оператор pass в Python - это простая концепция, которую могут быстро освоить даже новички без опыта программирования.
Некоторые методы, о которых вы не знали, что они существуют в Python
Некоторые методы, о которых вы не знали, что они существуют в Python
Python - самый известный и самый простой в изучении язык в наши дни. Имея широкий спектр применения в области машинного обучения, Data Science,...
Основы Python Часть I
Основы Python Часть I
Вы когда-нибудь задумывались, почему в программах на Python вы видите приведенный ниже код?
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
Алиса и Боб имеют неориентированный граф из n узлов и трех типов ребер:
Оптимизация кода с помощью тернарного оператора Python
Оптимизация кода с помощью тернарного оператора Python
И последнее, что мы хотели бы показать вам, прежде чем двигаться дальше, это
Советы по эффективной веб-разработке с помощью Python
Советы по эффективной веб-разработке с помощью Python
Как веб-разработчик, Python может стать мощным инструментом для создания эффективных и масштабируемых веб-приложений.
1
4
50
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Проблема в том, что ваша конечная точка возвращает экземпляр Request в ответе, то есть:

from fastapi import Request

@router.post('/register') 
async def register_user(request: Request,...
     ...
     return {"request": request, "msg": msg}
                        ^^^^^^^

Собственно, это и вызывает следующую ошибку:

RecursionError: maximum recursion depth exceeded while calling a Python object

Удаление этого должно решить проблему для вас. Кроме того, нет смысла возвращать в ответе объект Python, который выглядел бы так:

<starlette.requests.Request object at 0x000...>

Это сработало. Большое спасибо. Не могли бы вы объяснить, как вам удалось определить здесь место ошибки, поскольку его не было в трассировке?

mraabhijit 10.08.2024 15:09

@mraabhijit это называется опытом... Крис, я прав?

Derek Roberts 10.08.2024 15:18

На первый взгляд возвращение объекта Request казалось неправильным. Затем jsonable_encoder(), показанный в конце вашей трассировки, — это функция, используемая внутри FastAPI при возврате ответа, чтобы гарантировать, что объекты, которые не являются сериализуемыми, преобразуются в str. Пожалуйста, взгляните на этот ответ, чтобы получить более подробную информацию об этом.

Chris 10.08.2024 16:10

Другие вопросы по теме