Flask-Marshmallow 文档有这样的说法Marshmallow.init_app()
使用扩展名初始化应用程序。
它背后的代码似乎做了一些与 SQLAlchemy 会话管理相关的重要事情:
def init_app(self, app):
app.extensions = getattr(app, 'extensions', {})
# If using Flask-SQLAlchemy, attach db.session to ModelSchema
if has_sqla and 'sqlalchemy' in app.extensions:
db = app.extensions['sqlalchemy'].db
self.ModelSchema.OPTIONS_CLASS.session = db.session
app.extensions[EXTENSION_NAME] = self
但是我一直在使用和开发我的应用程序一段时间,但没有包括对init_app
任何地方的调用,而且它似乎工作得很好。我正在使用 SQLAlchemy、Flask-SQLAlchemy 和 Marshmallow-SQLAlchemy(以及 Flask-Marshmallow)。
我不做会导致什么问题ma.init_app(app)
,为什么一切似乎都正常?
目前,我有一个database
和一个schema
模块;database
看起来像这样:
from flask_sqlalchemy import SQLAlchemy
db = SQLAlchemy()
class Customer(db.Model):
...
像这样schema
:
from flask_marshmallow import Marshmallow
from database import Customer
ma = Marshmallow()
class CustomerSchema(ma.ModelSchema):
class Meta:
model = Customer
在我的顶层__init__.py
,我正在这样做:
from flask import Flask
from database import db
app = Flask(__name__)
db.init_app(app)
with app.app_context():
db.create_all()
这是一个 MVCE(放入mvce/__init__.py
,运行PYTHONPATH=. FLASK_APP=mvce flask run
,然后向 localhost:5000 发出 POST 请求,并将name
表单字段设置为某个值,然后发出 GET 请求以查看Customer
是否正确提交到数据库):
from flask import Flask, request
from flask_sqlalchemy import SQLAlchemy
from flask_marshmallow import Marshmallow
app = Flask(__name__)
app.config["SQLALCHEMY_DATABASE_URI"] = "sqlite:///database.db"
@app.route("/")
def get():
return CustomerSchema(many=True).jsonify(Customer.query.all())
@app.route("/", methods=["POST"])
def new():
customer, errors = CustomerSchema().load(request.form)
if errors:
return errors, 400
db.session.add(customer)
db.session.commit()
return CustomerSchema().jsonify(customer)
db = SQLAlchemy()
class Customer(db.Model):
id = db.Column(db.Integer, primary_key=True)
name = db.Column(db.String)
ma = Marshmallow() # Not passing `app`
class CustomerSchema(ma.ModelSchema):
class Meta:
model = Customer
db.init_app(app)
with app.app_context():
db.create_all()
# Note that we don't ever call `ma.init_app(app)`!