从 RDS 检索数据给出 AttributeError: 'sqlalchemy.cimmutabledict.immutabledict' 对象没有属性 'setdefault'

Posted

技术标签:

【中文标题】从 RDS 检索数据给出 AttributeError: \'sqlalchemy.cimmutabledict.immutabledict\' 对象没有属性 \'setdefault\'【英文标题】:Retrieving data from RDS gives AttributeError: 'sqlalchemy.cimmutabledict.immutabledict' object has no attribute 'setdefault'从 RDS 检索数据给出 AttributeError: 'sqlalchemy.cimmutabledict.immutabledict' 对象没有属性 'setdefault' 【发布时间】:2021-04-13 02:05:34 【问题描述】:

每当调用 API 端点 url/api/users/login 对用户进行身份验证时,我想从 RDS 检索用户数据,但是,我目前在从 RDS 检索数据时遇到问题。

错误堆栈跟踪

这是错误的完整堆栈跟踪:

当前软件包版本

Flask==1.1.2
Flask-Cors==3.0.10
flask-marshmallow==0.14.0
Flask-Migrate==2.5.3
Flask-RESTful==0.3.8
Flask-SQLAlchemy==2.4.4
marshmallow==3.10.0
marshmallow-sqlalchemy==0.24.1
SQLAlchemy==1.4.0b1
python==3.7

我解决问题的尝试

我试图解决问题但无济于事的一些事情是:

使用 pymysql 连接器代替 mysqlconnector 降级烧瓶棉花糖==0.11.0 降级的棉花糖==3.6.1 降级的 marshmallow-sqlalchemy==0.23.1

有关我的应用程序设置的更多信息

我已将我的应用程序 Docker 化,这是我当前的设置:

app.py

from flask_marshmallow import Marshmallow
from flask_migrate import Migrate
from flask_sqlalchemy import SQLAlchemy
from flask_jwt_extended import JWTManager

db = SQLAlchemy()
ma = Marshmallow()
migrate = Migrate()

def register_extensions(app: Flask) -> None:
   db.init_app(app)
   migrate.init_app(app, db)
   ma.init_app(app)

def create_application() -> Flask:
   app: Flask = Flask(__name__)
   app.config['SQLALCHEMY_DATABASE_URI'] = "mysql+mysqlconnector://DB_USER:DB_PASSWORD@DB_INSTANCE:DB_PORT"
   app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
   register_extensions(app)
   app.register_blueprint(users_urls.mod, url_prefix=f"/BEP_PREFIX/users")

   return app

 app = create_application()
 CORS(app)

models.py

class Accounts(db.Model):
    __tablename__ = 'Accounts'
    username = db.Column(db.String(100), primary_key=True)
    password = db.Column(db.String(256), nullable=False)
    email = db.Column(db.String(255), nullable=False)
    personnelID = db.Column(db.String(255), nullable=False)
    userType = db.Column(db.String(45), nullable=False)

service.py

class UserService(Service):
    __model__ = Accounts

    def get(self, username):
        return Accounts.query.filter_by(username=username).first()

user_service = UserService()

views.py

from bep.users.services import user_service

class UserLogin(Resource):
   def post(self):
       data = request.get_json()

       try:
           userData = user_service.get("test")
       except Exception:
           error_str = traceback.format_exc()
           logger.error("This prints the stack", exc_info=True)
           return UserServiceError.to_response(error=error_str, comments="Something went wrong while logging in") 

【问题讨论】:

你确定你没有使用 SQLAlchemy 1.4 吗?这看起来像this flask-sqlalchemy issue 啊对不起,我在我的虚拟环境中 pip freeze 而不是 docker bash。感谢您指出!我已经相应地更新了软件包版本。你知道我该如何解决这个问题吗? 【参考方案1】:

问题是flask-sqlalchemy 中的known issue,由SQLAchemy 1.4 中的更改引起。 Flask-sqlalchemy 尝试修改 SQLALchemy 引擎的 URL,但这些 URL 在 SQLAlchemy 1.4 中是不可变的。

问题已在 Flask-SQLAlchemy 2.5+ (changelog) 中得到修复。

如果无法升级 Flask-SQLAlchemy,可以通过命令行指定传递给 pip 的 SQLAlchemy 版本来解决此问题

pip install --upgrade 'SQLAlchemy<1.4'

或在 requirements.txt 中

SQLAlchemy<1.4

SQLAlchemy 1.4 于 2021 年 3 月 15 日正式发布。

【讨论】:

谢谢。你救了我们的命。 谢谢@snakecharmerb 你也救了我的命 :)

以上是关于从 RDS 检索数据给出 AttributeError: 'sqlalchemy.cimmutabledict.immutabledict' 对象没有属性 'setdefault'的主要内容,如果未能解决你的问题,请参考以下文章

PHP - 如何缓存从 AWS KMS Parameter Store 检索的数据库凭证

Amazon Data Pipeline“将 S3 数据加载到 RDS MySQL”查询格式?

0808关于RDS如何恢复到本地教程

将Mysql数据从一个账户的RDS自动复制到另一个账户的RDS

将数据从 Citus 迁移到 RDS

AWS RDS / EC2:TimeoutError:Knex:获取连接超时。游泳池可能已满