Автообновление свойств в sqlalchemy - PullRequest
5 голосов
/ 21 июня 2011

У меня есть модель sqlalchemy, которая настроена следующим образом:

class Entry(Base):
    __tablename__ = 'entries'
    __table__ = Table('entries', Base.metadata,
            Column('id', Integer, primary_key=True, unique=True),
            Column('user_id', Integer, ForeignKey('users.id', onupdate="CASCADE", ondelete="RESTRICT")),
            Column('title', String(128)),
            Column('slug', String(128), index=True),
            Column('url', String(256), index=True),
            Column('entry', Text),
            Column('cached_entry', Text),
            Column('created', DateTime, server_default=text('current_timestamp')),
            Column('modified', DateTime, server_onupdate=text('current_timestamp')),
            Column('pubdate', DateTime),
            )

Что мне хотелось бы, так это то, что когда я обновляю entry, cached_entry регенерируется, cached_entryявляется разобранной уценкой версией entry.По сути, я кеширую вывод парсинга уценки, чтобы мне не приходилось делать это при каждом показе записи.Я попытался использовать @hybrid_method, но это, похоже, не сработало, поскольку оно вообще не сохраняется в базе данных.У меня работает на Google AppEngine , но я не могу понять, как сделать то же самое с помощью SQLAlchemy.

Я действительно предпочел бы не добавлятьфункция для класса, который используется вместо имен в модели, потому что его сложнее реализовать с точки зрения приложения, я не хочу случайно что-то упустить.

1 Ответ

6 голосов
/ 22 июня 2011

@ hybrid_descriptor, безусловно, делает это, используя форму, описанную в http://www.sqlalchemy.org/docs/orm/mapper_config.html#using-descriptors.Вы присваиваете атрибут сопоставления базы данных, который вы можете отобразить под другим именем - поскольку вы используете __table__, вы можете использовать такую ​​форму, как:

class Entry(Base):
    __table__ = ...

    _entry = __table__.c.entry

    @hybrid_property
    def entry(self):
        return self._entry

    @entry.setter
    def entry(self, value):
        self._entry = value
        self.cached_entry = markdown(value)

Другой способ - использовать before_insertи события before_update для заполнения столбца во время сброса - это простой подход, но он имеет тот недостаток, что вам нужно ждать, пока это произойдет.

Я думаю, что самый быстрый способ для "включения-set "должен использовать @validates:

from sqlalchemy.orm import validates

class Entry(Base):
    __table__ = ...

    @validates('entry')
    def _set_entry(self, key, value):
        self.cached_entry = markdown(value)
        return value
...