【问题标题】:session is already flushing when I use @observes当我使用 @observes 时,会话已经在刷新
【发布时间】:2015-07-27 03:36:09
【问题描述】:

我正在构建一个烧瓶应用程序,并且我想使用 sqlalchemy observers 在该货件中的所有产品可用后更新该货件状态。

这是我的数据模型:

from app import db
from sqlalchemy_utils import observes

class Shipment(db.Model):
    __tablename__ = 'shipment'
    id = db.Column(db.Integer, primary_key=True)
    products = db.relationship('Product', backref='shipment', lazy='dynamic')
    all_products_ready = db.Column(db.Boolean)

    @observes('products')
    def product_observer(self, products):
        for p in self.products:
            if p.status != 'ready':
                self.all_products_ready = False
                return False
        self.all_products_ready = True
        return True

class Product(db.Model):
    __tablename__ = 'product'
    id = db.Column(db.Integer, primary_key=True)
    shipment_id = db.Column(db.Integer, db.ForeignKey('shipment.id'))

    status = db.Column(db.String(120), index=True)

这是我运行的一些代码来测试它:

shipment = models.Shipment(products=[models.Product(status='ready'), models.Product(status='not_ready')])
db.session.add(shipment)
db.session.commit()

print(shipment.all_products_ready)

当我运行这段代码时,我得到一个 InvalidRequestError: Session is already flushing。

这是堆栈跟踪:

Traceback (most recent call last):
  File "test.py", line 5, in <module>
    db.session.commit()
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\orm\scoping.py",
line 150, in do
    return getattr(self.registry(), name)(*args, **kwargs)
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\orm\session.py",
line 788, in commit
    self.transaction.commit()
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\orm\session.py",
line 384, in commit
    self._prepare_impl()
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\orm\session.py",
line 364, in _prepare_impl
    self.session.flush()
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\orm\session.py",
line 1985, in flush
    self._flush(objects)
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\orm\session.py",
line 2012, in _flush
    self.dispatch.before_flush(self, flush_context, objects)
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\event\attr.py", l
ine 221, in __call__
    fn(*args, **kw)
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy_utils\observer.py
", line 272, in invoke_callbacks
    for (root_obj, func, objects) in args:
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy_utils\observer.py
", line 252, in gather_callback_args
    lambda obj: obj not in session.deleted
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy_utils\functions\o
rm.py", line 741, in getdotattr
    last = [v for v in last if condition(v)]
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\orm\dynamic.py",
line 245, in __iter__
    sess = self.session
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\orm\dynamic.py",
line 237, in session
    sess.flush()
  File "U:\dev\observertest\flask\lib\site-packages\sqlalchemy\orm\session.py",
line 1979, in flush
    raise sa_exc.InvalidRequestError("Session is already flushing")
sqlalchemy.exc.InvalidRequestError: Session is already flushing

如何使用我的模型而不出现此错误?

【问题讨论】:

    标签: python sqlalchemy flask-sqlalchemy


    【解决方案1】:

    我不完全确定为什么,但我认为使用 dynamic 加载关系会导致问题。在这一行:

        products = db.relationship('Product', backref='shipment', lazy='dynamic')
    

    您需要将lazy 参数更改为select 而不是dynamic(或者您可以将lazy 参数全部去掉,因为select 是它的默认值)。

    请参阅the sqlalchemy reference 了解所有可用选项。

    【讨论】:

    • 虽然这很有帮助,但我认为对于我们这些需要使用lazy='dynamic'的人来说可能仍然没有答案
    猜你喜欢
    • 2022-01-04
    • 2012-01-19
    • 1970-01-01
    • 1970-01-01
    • 2020-01-22
    • 2016-11-19
    • 1970-01-01
    • 2014-05-24
    • 2019-02-03
    相关资源
    最近更新 更多