SqlAlchemy 標記字段變更

songxiao 9年前發布 | 13K 次閱讀 Python開發

來自:http://graycarl.me/post/notes/sqlalchemy-set-column-modified

為了在數據庫里面存儲一些比較隨意的結構化數據,我參照(抄襲)sqlalchemy的文檔實現了一個自定義類型:

class JSON(TypeDecorator):
    """ Json String Field """
    impl = Text

    def process_bind_param(self, value, dialect):
        if value is None:
            return value
        elif isinstance(value, (dict, list)):
            return json.dumps(value)
        else:
            raise ValueError("unsupported value type")

    def process_result_value(self, value, dialect):
        if value is None:
            return value
        else:
            return json.loads(value)


class MyModel(Base):
    ...
    json = Column(JSON, default=[])

    def add_item(self, item):
        self.json.append(item)

其實這本質上就是把python的dictlist直接dumps成json內容存入數據庫,從數據庫讀出之后再自動loadsdict或者list來使用。

但是再使用的過程中發現了一個問題。對這個MyModel的實例無論怎么修改其json字段都無法正常保存到數據庫中,目測應該是SA沒有檢測到這個字段的變化而導致的。重新翻看文檔,發現文檔里面已經有說明了。

Note that the ORM by default will not detect “mutability” on such a type - meaning, in-place changes to values will not be detected and will not be flushed. Without further steps, you instead would need to replace the existing value with a new one on each parent object to detect changes. Note that there’s nothing wrong with this, as many applications may not require that the values are ever mutated once created. For those which do have this requirement, support for mutability is best applied using the sqlalchemy.ext.mutable extension - see the example in Mutation Tracking.

再看看sqlalchemy.ext.mutable模塊的文檔,發現使用起來也沒有那么簡練,而我只想找個方法來主動標記字段變更就行了。于是翻看mutable模塊的代碼,找到了其最重要的一句:

from sqlalchemy.orm.attributes import flag_modified

class Mutable(MutableBase):

    def changed(self):
        """Subclasses should call this method whenever change events occur."""

        for parent, key in self._parents.items():
            flag_modified(parent, key)

其實就是這個flag_modified標記了修改狀態,所以只需要把我自己的代碼改一句就行了。

from sqlalchemy.orm.attributes import flag_modified

class MyModel(Base):
    ...
    json = Column(JSON, default=[])

    def add_item(self, item):
        self.json.append(item)
        flag_modified(self, "json")

打完收工。

 本文由用戶 songxiao 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!