Commit b32c6cbe by Florian Apolloner

Prevented infinite loops on Tag.save()

Note: Now Tag.save() can raise an IntegrityError in case of concurrency
issues, your app will need to handle that.
parent c8257843
...@@ -47,15 +47,26 @@ class TagBase(models.Model): ...@@ -47,15 +47,26 @@ class TagBase(models.Model):
# with a multi-master setup, theoretically we could try to # with a multi-master setup, theoretically we could try to
# write and rollback on different DBs # write and rollback on different DBs
kwargs["using"] = using kwargs["using"] = using
i = 0 # Be oportunistic and try to save the tag, this should work for
while True: # most cases ;)
i += 1
try: try:
with atomic(using=using): with atomic(using=using):
res = super(TagBase, self).save(*args, **kwargs) res = super(TagBase, self).save(*args, **kwargs)
return res return res
except IntegrityError: except IntegrityError:
self.slug = self.slugify(self.name, i) pass
# Now try to find existing slugs with similar names
slugs = set(Tag.objects.filter(slug__startswith=self.slug)\
.values_list('slug', flat=True))
i = 1
while True:
slug = self.slugify(self.name, i)
if slug not in slugs:
self.slug = slug
# We purposely ignore concurrecny issues here for now.
# (That is, till we found a nice solution...)
return super(TagBase, self).save(*args, **kwargs)
i += 1
else: else:
return super(TagBase, self).save(*args, **kwargs) return super(TagBase, self).save(*args, **kwargs)
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or sign in to comment