Retrieving data from RDS gives AttributeError: 'sqlalchemy.cimmutabledict.immutabledict' object has no attribute 'setdefault'
Retrieving data from RDS gives AttributeError: 'sqlalchemy.cimmutabledict.immutabledict' object has no attribute 'setdefault'
每当调用 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
- 降级 flask-marshmallow==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")
问题出在 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 日全面发布。
每当调用 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
- 降级 flask-marshmallow==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")
问题出在 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 日全面发布。