Fixed #17415 -- Reset database sequence for Site's pk after creating the default site with an explicit pk. Thanks niko AT neagee net for the report, Russell and Karen for describing the fix, and Anssi for drafting the patch.

git-svn-id: http://code.djangoproject.com/svn/django/trunk@17343 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
Aymeric Augustin 2012-01-07 08:53:36 +00:00
parent 33f839b252
commit 4629668ffa
2 changed files with 22 additions and 4 deletions

View File

@ -3,22 +3,34 @@ Creates the default Site object.
""" """
from django.db.models import signals from django.db.models import signals
from django.db import connections
from django.db import router from django.db import router
from django.contrib.sites.models import Site from django.contrib.sites.models import Site
from django.contrib.sites import models as site_app from django.contrib.sites import models as site_app
from django.core.management.color import no_style
def create_default_site(app, created_models, verbosity, db, **kwargs): def create_default_site(app, created_models, verbosity, db, **kwargs):
# Only create the default sites in databases where Django created the table # Only create the default sites in databases where Django created the table
if Site in created_models and router.allow_syncdb(db, Site) : if Site in created_models and router.allow_syncdb(db, Site) :
if verbosity >= 2:
print "Creating example.com Site object"
# The default settings set SITE_ID = 1, and some tests in Django's test # The default settings set SITE_ID = 1, and some tests in Django's test
# suite rely on this value. However, if database sequences are reused # suite rely on this value. However, if database sequences are reused
# (e.g. in the test suite after flush/syncdb), it isn't guaranteed that # (e.g. in the test suite after flush/syncdb), it isn't guaranteed that
# the next id will be 1, so we coerce it. See #15573 and #16353. This # the next id will be 1, so we coerce it. See #15573 and #16353. This
# can also crop up outside of tests - see #15346. # can also crop up outside of tests - see #15346.
s = Site(pk=1, domain="example.com", name="example.com") if verbosity >= 2:
s.save(using=db) print "Creating example.com Site object"
Site(pk=1, domain="example.com", name="example.com").save(using=db)
# We set an explicit pk instead of relying on auto-incrementation,
# so we need to reset the database sequence. See #17415.
sequence_sql = connections[db].ops.sequence_reset_sql(no_style(), [Site])
if sequence_sql:
if verbosity >= 2:
print "Resetting sequence"
cursor = connections[db].cursor()
for command in sequence_sql:
cursor.execute(command)
Site.objects.clear_cache() Site.objects.clear_cache()
signals.post_syncdb.connect(create_default_site, sender=site_app) signals.post_syncdb.connect(create_default_site, sender=site_app)

View File

@ -15,6 +15,12 @@ class SitesFrameworkTests(TestCase):
def tearDown(self): def tearDown(self):
Site._meta.installed = self.old_Site_meta_installed Site._meta.installed = self.old_Site_meta_installed
def test_save_another(self):
# Regression for #17415
# On some backends the sequence needs reset after save with explicit ID.
# Test that there is no sequence collisions by saving another site.
Site(domain="example2.com", name="example2.com").save()
def test_site_manager(self): def test_site_manager(self):
# Make sure that get_current() does not return a deleted Site object. # Make sure that get_current() does not return a deleted Site object.
s = Site.objects.get_current() s = Site.objects.get_current()