This code is supposed to get or create an object and update it if necessary. The code is in production use on a website.
In some cases – when the database is busy – it will throw the exception “DoesNotExist: MyObj matching query does not exist”.
# Model:
class MyObj(models.Model):
thing = models.ForeignKey(Thing)
owner = models.ForeignKey(User)
state = models.BooleanField()
class Meta:
unique_together = (('thing', 'owner'),)
# Update or create myobj
@transaction.commit_on_success
def create_or_update_myobj(owner, thing, state)
try:
myobj, created = MyObj.objects.get_or_create(owner=user,thing=thing)
except IntegrityError:
myobj = MyObj.objects.get(owner=user,thing=thing)
# Will sometimes throw "DoesNotExist: MyObj matching query does not exist"
myobj.state = state
myobj.save()
I use an innodb mysql database on ubuntu.
How do I safely deal with this problem?
This could be an off-shoot of the same problem as here:
Why doesn't this loop display an updated object count every five seconds?
Basically get_or_create can fail – if you take a look at its source, there you’ll see that it’s: get, if-problem: save+some_trickery, if-still-problem: get again, if-still-problem: surrender and raise.
This means that if there are two simultaneous threads (or processes) running
create_or_update_myobj, both trying to get_or_create the same object, then:uniqueconstraintgeton the second thread doesn’t see the object created in the first thread, due to the frozen view of MyObj tableSo, if you want to safely
get_or_createanything, try something like this:Edited on 27/05/2010
There is also a second solution to the problem – using READ COMMITED isolation level, instead of REPEATABLE READ. But it’s less tested (at least in MySQL), so there might be more bugs/problems with it – but at least it allows tying views to transactions, without committing in the middle.
Edited on 22/01/2012
Here are some good blog posts (not mine) about MySQL and Django, related to this question:
http://www.no-ack.org/2010/07/mysql-transactions-and-django.html
http://www.no-ack.org/2011/05/broken-transaction-management-in-mysql.html