Added regressions tests to ensure that one-to-one and many-to-many fields

continue to interact properly. Refs #1064. Refs #1506.


git-svn-id: http://code.djangoproject.com/svn/django/trunk@3176 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
Malcolm Tredinnick 2006-06-20 05:29:19 +00:00
parent 76448d0c4b
commit a513fcb455
4 changed files with 54 additions and 9 deletions

View File

View File

@ -0,0 +1,37 @@
from django.db import models
class Place(models.Model):
name = models.CharField(maxlength=50)
address = models.CharField(maxlength=80)
def __str__(self):
return "%s the place" % self.name
class Restaurant(models.Model):
place = models.OneToOneField(Place)
serves_hot_dogs = models.BooleanField()
serves_pizza = models.BooleanField()
def __str__(self):
return "%s the restaurant" % self.place.name
class Favorites(models.Model):
name = models.CharField(maxlength = 50)
restaurants = models.ManyToManyField(Restaurant)
def __str__(self):
return "Favorites for %s" % self.name
API_TESTS = """
# Regression test for #1064 and #1506: Check that we create models via the m2m
# relation if the remote model has a OneToOneField.
>>> p1 = Place(name='Demon Dogs', address='944 W. Fullerton')
>>> p1.save()
>>> r = Restaurant(place=p1, serves_hot_dogs=True, serves_pizza=False)
>>> r.save()
>>> f = Favorites(name = 'Fred')
>>> f.save()
>>> f.restaurants = [r]
>>> f.restaurants.all()
[<Restaurant: Demon Dogs the restaurant>]
"""

View File

@ -9,6 +9,7 @@ import doctest
MODEL_TESTS_DIR_NAME = 'modeltests' MODEL_TESTS_DIR_NAME = 'modeltests'
OTHER_TESTS_DIR = "othertests" OTHER_TESTS_DIR = "othertests"
REGRESSION_TESTS_DIR_NAME = 'regressiontests'
TEST_DATABASE_NAME = 'django_test_db' TEST_DATABASE_NAME = 'django_test_db'
error_list = [] error_list = []
@ -19,6 +20,7 @@ def log_error(model_name, title, description):
}) })
MODEL_TEST_DIR = os.path.join(os.path.dirname(__file__), MODEL_TESTS_DIR_NAME) MODEL_TEST_DIR = os.path.join(os.path.dirname(__file__), MODEL_TESTS_DIR_NAME)
REGRESSION_TEST_DIR = os.path.join(os.path.dirname(__file__), REGRESSION_TESTS_DIR_NAME)
ALWAYS_INSTALLED_APPS = [ ALWAYS_INSTALLED_APPS = [
'django.contrib.contenttypes', 'django.contrib.contenttypes',
@ -32,7 +34,8 @@ ALWAYS_INSTALLED_APPS = [
] ]
def get_test_models(): def get_test_models():
return [f for f in os.listdir(MODEL_TEST_DIR) if not f.startswith('__init__') and not f.startswith('.')] return [(MODEL_TESTS_DIR_NAME, f) for f in os.listdir(MODEL_TEST_DIR) if not f.startswith('__init__') and not f.startswith('.')] +\
[(REGRESSION_TESTS_DIR_NAME, f) for f in os.listdir(REGRESSION_TEST_DIR) if not f.startswith('__init__') and not f.startswith('.')]
class DjangoDoctestRunner(doctest.DocTestRunner): class DjangoDoctestRunner(doctest.DocTestRunner):
def __init__(self, verbosity_level, *args, **kwargs): def __init__(self, verbosity_level, *args, **kwargs):
@ -89,7 +92,7 @@ class TestRunner:
settings.INSTALLED_APPS settings.INSTALLED_APPS
# Manually set INSTALLED_APPS to point to the test models. # Manually set INSTALLED_APPS to point to the test models.
settings.INSTALLED_APPS = ALWAYS_INSTALLED_APPS + [MODEL_TESTS_DIR_NAME + '.' + a for a in get_test_models()] settings.INSTALLED_APPS = ALWAYS_INSTALLED_APPS + ['.'.join(a) for a in get_test_models()]
# Manually set DEBUG = False. # Manually set DEBUG = False.
settings.DEBUG = False settings.DEBUG = False
@ -110,12 +113,17 @@ class TestRunner:
if self.which_tests: if self.which_tests:
# Only run the specified tests. # Only run the specified tests.
bad_models = [m for m in self.which_tests if m not in test_models] bad_models = [m for m in self.which_tests if (MODEL_TESTS_DIR_NAME, m) not in test_models and (REGRESSION_TESTS_DIR_NAME, m) not in test_models]
if bad_models: if bad_models:
sys.stderr.write("Models not found: %s\n" % bad_models) sys.stderr.write("Models not found: %s\n" % bad_models)
sys.exit(1) sys.exit(1)
else: else:
test_models = self.which_tests all_tests = []
for test in self.which_tests:
for loc in MODEL_TESTS_DIR_NAME, REGRESSION_TESTS_DIR_NAME:
if (loc, test) in test_models:
all_tests.append((loc, test))
test_models = all_tests
self.output(0, "Running tests with database %r" % settings.DATABASE_ENGINE) self.output(0, "Running tests with database %r" % settings.DATABASE_ENGINE)
@ -157,18 +165,18 @@ class TestRunner:
# Run the tests for each test model. # Run the tests for each test model.
self.output(1, "Running app tests") self.output(1, "Running app tests")
for model_name in test_models: for model_dir, model_name in test_models:
self.output(1, "%s model: Importing" % model_name) self.output(1, "%s model: Importing" % model_name)
try: try:
# TODO: Abstract this into a meta.get_app() replacement? # TODO: Abstract this into a meta.get_app() replacement?
mod = __import__(MODEL_TESTS_DIR_NAME + '.' + model_name + '.models', '', '', ['']) mod = __import__(model_dir + '.' + model_name + '.models', '', '', [''])
except Exception, e: except Exception, e:
log_error(model_name, "Error while importing", ''.join(traceback.format_exception(*sys.exc_info())[1:])) log_error(model_name, "Error while importing", ''.join(traceback.format_exception(*sys.exc_info())[1:]))
continue continue
if not getattr(mod, 'error_log', None): if not getattr(mod, 'error_log', None):
# Model is not marked as an invalid model # Model is not marked as an invalid model
self.output(1, "%s model: Installing" % model_name) self.output(1, "%s.%s model: Installing" % (model_dir, model_name))
management.install(mod) management.install(mod)
# Run the API tests. # Run the API tests.
@ -179,11 +187,11 @@ class TestRunner:
# Manually set verbose=False, because "-v" command-line parameter # Manually set verbose=False, because "-v" command-line parameter
# has side effects on doctest TestRunner class. # has side effects on doctest TestRunner class.
runner = DjangoDoctestRunner(verbosity_level=verbosity_level, verbose=False) runner = DjangoDoctestRunner(verbosity_level=verbosity_level, verbose=False)
self.output(1, "%s model: Running tests" % model_name) self.output(1, "%s.%s model: Running tests" % (model_dir, model_name))
runner.run(dtest, clear_globs=True, out=sys.stdout.write) runner.run(dtest, clear_globs=True, out=sys.stdout.write)
else: else:
# Check that model known to be invalid is invalid for the right reasons. # Check that model known to be invalid is invalid for the right reasons.
self.output(1, "%s model: Validating" % model_name) self.output(1, "%s.%s model: Validating" % (model_dir, model_name))
from cStringIO import StringIO from cStringIO import StringIO
s = StringIO() s = StringIO()