Skip to content

Commit

Permalink
fix(141): handle after_flush_postexec when creating version objects
Browse files Browse the repository at this point in the history
when creating version objects if a user has created after_flush_postexec
hook, which keeps calling `after_flush` untill it exhausts 100 attempts
or session is no longer dirty, this is picked up by mapper as after_update
which within same transaction adds a update operation type, so we have a
check if target is already in UoW we continue with operation type that
it is in untill transaction is completed.

We also updated a existing testcase named `test_replace_deleted_object_with_update`
as it was updating the pk of article object, but the pk being identity of object
is used by operations to track target, so changing a non identity column to validate
partial flush does not impact other objects
  • Loading branch information
indiVar0508 committed Aug 9, 2024
1 parent 5f1e098 commit 2c303d2
Show file tree
Hide file tree
Showing 3 changed files with 52 additions and 8 deletions.
8 changes: 6 additions & 2 deletions sqlalchemy_history/operation.py
Original file line number Diff line number Diff line change
Expand Up @@ -90,7 +90,11 @@ def add_update(self, target):
del state_copy[rel_key]

if state_copy:
self.add(Operation(target, Operation.UPDATE))

if target in self:
# If already in current transaction and some event hook did a update
# prior to commit hook, continue with operation type as it is
self.add(Operation(target, self[self.format_key(target)].type))
else:
self.add(Operation(target, Operation.UPDATE))
def add_delete(self, target):
self.add(Operation(target, Operation.DELETE))
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
import sqlalchemy as sa
from copy import copy

from tests import TestCase
from sqlalchemy_history import version_class


class TestBug141(TestCase):
# ref: https://github.com/corridor/sqlalchemy-history/issues/141
def create_models(self):
class Author(self.Model):
__tablename__ = "author"
__versioned__ = copy(self.options)

id = sa.Column(
sa.Integer, sa.Sequence(f"{__tablename__}_seq", start=1), autoincrement=True, primary_key=True
)
name = sa.Column(sa.Unicode(255))

self.Author = Author

def test_add_record(self):
author = self.Author(name="Author 1")
@sa.event.listens_for(self.session, 'after_flush_postexec')
def after_flush_postexec(session, flush_context):
if author.name != "yoyoyoyoyo":
author.name = "yoyoyoyoyo"
self.session.add(author)
self.session.commit()

versioned_objs = self.session.query(version_class(self.Author)).all()
assert len(versioned_objs) == 1
assert versioned_objs[0].operation_type == 0
assert versioned_objs[0].name == "yoyoyoyoyo"
author.name = "sdfeoinfe"
self.session.add(author)
self.session.commit()
versioned_objs = self.session.query(version_class(self.Author)).all()
assert len(versioned_objs) == 2
assert versioned_objs[0].operation_type == 0
assert versioned_objs[1].operation_type == 1
assert versioned_objs[0].name == versioned_objs[1].name == "yoyoyoyoyo"
sa.event.remove(self.session, "after_flush_postexec", after_flush_postexec)
9 changes: 3 additions & 6 deletions tests/test_exotic_operation_combos.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,10 +40,6 @@ def test_insert_deleted_and_flushed_object(self):
assert article2.versions[0].operation_type == Operation.INSERT
assert article2.versions[1].operation_type == Operation.UPDATE

# Ref for mssql: https://github.com/sqlalchemy/sqlalchemy/discussions/8829
@mark.skipif(
os.environ.get("DB") == "mssql", reason="mssql does not support changing the IDENTITY column"
)
def test_replace_deleted_object_with_update(self):
article = self.Article()
article.name = "Some article"
Expand All @@ -57,8 +53,9 @@ def test_replace_deleted_object_with_update(self):

self.session.delete(article)
self.session.flush()

article2.id = article.id
# we were earlier updating ID which didn't seem right, so changed this to name since
# id is used by us for identity in operation
article2.name = article.name
self.session.commit()
assert article2.versions.count() == 2
assert article2.versions[0].operation_type == Operation.INSERT
Expand Down

0 comments on commit 2c303d2

Please sign in to comment.