#26627 closed Bug (fixed)
outermost atomic block executes on_commit callbacks registered in another outermost atomic block
Reported by: | Barthelemy Dagenais | Owned by: | nobody |
---|---|---|---|
Component: | Database layer (models, ORM) | Version: | 1.9 |
Severity: | Normal | Keywords: | |
Cc: | Triage Stage: | Ready for checkin | |
Has patch: | yes | Needs documentation: | no |
Needs tests: | no | Patch needs improvement: | no |
Easy pickings: | no | UI/UX: | no |
Description
The on_commit hook executes callbacks in an undesired order because the list of callbacks is tied to the connection instead of being tied to the transaction or atomic block. Consider the following code:
from django.db import transaction from foobar.models import Model1, Model2 @transaction.atomic def outer(): print("START - OUTER") for i in range(4): f1(i) print("END - OUTER") @transaction.atomic def f1(i): model = Model1(description="test {0}".format(i)) model.save() transaction.on_commit(lambda: commit_hook(model, i)) def commit_hook(model, i): print("START - on_commit hook") f2(i) print("STOP - on_commit hook") @transaction.atomic def f2(i): print("CALLING F2") model = Model2(description="test {0}".format(i)) model.save()
The expected trace when calling outer() is:
START - OUTER END - OUTER START - on_commit hook CALLING F2 STOP - on_commit hook START - on_commit hook CALLING F2 STOP - on_commit hook START - on_commit hook CALLING F2 STOP - on_commit hook START - on_commit hook CALLING F2 STOP - on_commit hook
But the actual trace is:
START - OUTER END - OUTER START - on_commit hook CALLING F2 START - on_commit hook CALLING F2 START - on_commit hook CALLING F2 START - on_commit hook CALLING F2 STOP - on_commit hook STOP - on_commit hook STOP - on_commit hook STOP - on_commit hook
Essentially, the outermost atomic block on f2() is executing the callbacks registered by the outermost atomic block on outer(), which creates a stack of calls instead of a sequence of calls.
This has been discussed on the Django user list and more examples of why this can lead to undesired behaviors are given there.
Carl Meyer has suggested a fix. I'll provide a regression test and a fix today.
Change History (5)
comment:1 by , 8 years ago
Has patch: | set |
---|
comment:2 by , 8 years ago
Triage Stage: | Unreviewed → Accepted |
---|
comment:3 by , 8 years ago
Triage Stage: | Accepted → Ready for checkin |
---|
I submitted a pull request based on Carl Meyer's suggestion: https://github.com/django/django/pull/6617