2005-07-29 15:15:40 +00:00
#!/usr/bin/env python
2006-08-27 13:59:47 +00:00
import os , sys , traceback
import unittest
2005-07-29 15:15:40 +00:00
2007-05-20 03:51:21 +00:00
import django . contrib as contrib
2007-09-14 09:55:17 +00:00
2007-05-20 03:51:21 +00:00
CONTRIB_DIR_NAME = ' django.contrib '
2006-05-02 01:31:56 +00:00
MODEL_TESTS_DIR_NAME = ' modeltests '
2006-06-20 05:29:19 +00:00
REGRESSION_TESTS_DIR_NAME = ' regressiontests '
2007-05-20 03:51:21 +00:00
2006-09-02 09:34:40 +00:00
TEST_TEMPLATE_DIR = ' templates '
2005-07-29 15:15:40 +00:00
2007-05-20 03:51:21 +00:00
CONTRIB_DIR = os . path . dirname ( contrib . __file__ )
2006-05-02 01:31:56 +00:00
MODEL_TEST_DIR = os . path . join ( os . path . dirname ( __file__ ) , MODEL_TESTS_DIR_NAME )
2006-06-20 05:29:19 +00:00
REGRESSION_TEST_DIR = os . path . join ( os . path . dirname ( __file__ ) , REGRESSION_TESTS_DIR_NAME )
2005-07-29 22:35:54 +00:00
2010-04-14 18:17:44 +00:00
REGRESSION_SUBDIRS_TO_SKIP = [ ' locale ' ]
2006-05-26 21:28:12 +00:00
ALWAYS_INSTALLED_APPS = [
' django.contrib.contenttypes ' ,
2006-06-20 14:27:44 +00:00
' django.contrib.auth ' ,
2007-08-15 11:25:22 +00:00
' django.contrib.sites ' ,
2006-05-26 21:28:12 +00:00
' django.contrib.flatpages ' ,
' django.contrib.redirects ' ,
' django.contrib.sessions ' ,
2009-12-09 16:57:23 +00:00
' django.contrib.messages ' ,
2006-06-16 18:58:45 +00:00
' django.contrib.comments ' ,
' django.contrib.admin ' ,
2010-08-14 13:41:56 +00:00
' django.contrib.admindocs ' ,
2006-05-26 21:28:12 +00:00
]
2005-07-29 22:35:54 +00:00
def get_test_models ( ) :
2006-06-20 07:12:45 +00:00
models = [ ]
2007-05-20 03:51:21 +00:00
for loc , dirpath in ( MODEL_TESTS_DIR_NAME , MODEL_TEST_DIR ) , ( REGRESSION_TESTS_DIR_NAME , REGRESSION_TEST_DIR ) , ( CONTRIB_DIR_NAME , CONTRIB_DIR ) :
2006-06-20 14:27:44 +00:00
for f in os . listdir ( dirpath ) :
2010-04-14 18:17:44 +00:00
if f . startswith ( ' __init__ ' ) or f . startswith ( ' . ' ) or \
f . startswith ( ' sql ' ) or f . startswith ( ' invalid ' ) or \
os . path . basename ( f ) in REGRESSION_SUBDIRS_TO_SKIP :
2006-06-20 07:12:45 +00:00
continue
models . append ( ( loc , f ) )
return models
2005-07-29 22:35:54 +00:00
2006-08-27 13:59:47 +00:00
def get_invalid_models ( ) :
models = [ ]
2007-05-20 03:51:21 +00:00
for loc , dirpath in ( MODEL_TESTS_DIR_NAME , MODEL_TEST_DIR ) , ( REGRESSION_TESTS_DIR_NAME , REGRESSION_TEST_DIR ) , ( CONTRIB_DIR_NAME , CONTRIB_DIR ) :
2006-08-27 13:59:47 +00:00
for f in os . listdir ( dirpath ) :
if f . startswith ( ' __init__ ' ) or f . startswith ( ' . ' ) or f . startswith ( ' sql ' ) :
continue
if f . startswith ( ' invalid ' ) :
models . append ( ( loc , f ) )
return models
2006-12-15 06:06:52 +00:00
2006-08-27 13:59:47 +00:00
class InvalidModelTestCase ( unittest . TestCase ) :
def __init__ ( self , model_label ) :
unittest . TestCase . __init__ ( self )
self . model_label = model_label
2006-12-15 06:06:52 +00:00
2006-08-27 13:59:47 +00:00
def runTest ( self ) :
2007-08-16 06:06:55 +00:00
from django . core . management . validation import get_validation_errors
2006-06-23 04:37:00 +00:00
from django . db . models . loading import load_app
2006-08-27 13:59:47 +00:00
from cStringIO import StringIO
try :
module = load_app ( self . model_label )
except Exception , e :
self . fail ( ' Unable to load invalid model module ' )
2006-12-15 06:06:52 +00:00
2007-08-17 12:29:08 +00:00
# Make sure sys.stdout is not a tty so that we get errors without
# coloring attached (makes matching the results easier). We restore
# sys.stderr afterwards.
orig_stdout = sys . stdout
2006-08-27 13:59:47 +00:00
s = StringIO ( )
2007-08-17 12:29:08 +00:00
sys . stdout = s
2007-08-16 06:06:55 +00:00
count = get_validation_errors ( s , module )
2007-08-17 12:29:08 +00:00
sys . stdout = orig_stdout
2006-08-27 13:59:47 +00:00
s . seek ( 0 )
error_log = s . read ( )
actual = error_log . split ( ' \n ' )
expected = module . model_errors . split ( ' \n ' )
unexpected = [ err for err in actual if err not in expected ]
missing = [ err for err in expected if err not in actual ]
self . assert_ ( not unexpected , " Unexpected Errors: " + ' \n ' . join ( unexpected ) )
self . assert_ ( not missing , " Missing Errors: " + ' \n ' . join ( missing ) )
2009-12-13 16:24:36 +00:00
def django_tests ( verbosity , interactive , failfast , test_labels ) :
2006-08-27 13:59:47 +00:00
from django . conf import settings
2006-09-02 09:34:40 +00:00
2006-08-27 13:59:47 +00:00
old_installed_apps = settings . INSTALLED_APPS
2008-08-30 05:09:03 +00:00
old_root_urlconf = getattr ( settings , " ROOT_URLCONF " , " " )
2006-09-02 09:34:40 +00:00
old_template_dirs = settings . TEMPLATE_DIRS
2006-12-15 06:06:52 +00:00
old_use_i18n = settings . USE_I18N
2007-12-17 10:31:20 +00:00
old_login_url = settings . LOGIN_URL
2007-10-21 17:26:32 +00:00
old_language_code = settings . LANGUAGE_CODE
2007-02-10 04:01:19 +00:00
old_middleware_classes = settings . MIDDLEWARE_CLASSES
2006-12-15 06:06:52 +00:00
2007-02-10 04:01:19 +00:00
# Redirect some settings for the duration of these tests.
2006-08-27 13:59:47 +00:00
settings . INSTALLED_APPS = ALWAYS_INSTALLED_APPS
2006-09-02 09:34:40 +00:00
settings . ROOT_URLCONF = ' urls '
settings . TEMPLATE_DIRS = ( os . path . join ( os . path . dirname ( __file__ ) , TEST_TEMPLATE_DIR ) , )
2006-12-15 06:06:52 +00:00
settings . USE_I18N = True
2007-10-21 17:26:32 +00:00
settings . LANGUAGE_CODE = ' en '
2007-12-17 10:31:20 +00:00
settings . LOGIN_URL = ' /accounts/login/ '
2007-02-10 04:01:19 +00:00
settings . MIDDLEWARE_CLASSES = (
' django.contrib.sessions.middleware.SessionMiddleware ' ,
' django.contrib.auth.middleware.AuthenticationMiddleware ' ,
2009-12-09 16:57:23 +00:00
' django.contrib.messages.middleware.MessageMiddleware ' ,
2007-02-10 04:01:19 +00:00
' django.middleware.common.CommonMiddleware ' ,
)
2007-12-01 21:58:51 +00:00
settings . SITE_ID = 1
2009-03-23 21:07:02 +00:00
# For testing comment-utils, we require the MANAGERS attribute
# to be set, so that a test email is sent out which we catch
# in our tests.
settings . MANAGERS = ( " admin@djangoproject.com " , )
2006-12-15 06:06:52 +00:00
# Load all the ALWAYS_INSTALLED_APPS.
# (This import statement is intentionally delayed until after we
# access settings because of the USE_I18N dependency.)
from django . db . models . loading import get_apps , load_app
2006-08-27 13:59:47 +00:00
get_apps ( )
2006-12-15 06:06:52 +00:00
2007-02-10 04:01:19 +00:00
# Load all the test model apps.
2006-12-15 06:06:52 +00:00
for model_dir , model_name in get_test_models ( ) :
2006-08-27 13:59:47 +00:00
model_label = ' . ' . join ( [ model_dir , model_name ] )
try :
# if the model was named on the command line, or
2006-12-15 06:06:52 +00:00
# no models were named (i.e., run all), import
2006-08-27 13:59:47 +00:00
# this model and add it to the list to test.
2007-07-28 04:02:52 +00:00
if not test_labels or model_name in set ( [ label . split ( ' . ' ) [ 0 ] for label in test_labels ] ) :
2010-08-07 06:58:14 +00:00
if verbosity > = 2 :
2006-08-27 13:59:47 +00:00
print " Importing model %s " % model_name
mod = load_app ( model_label )
2007-05-20 03:51:21 +00:00
if mod :
if model_label not in settings . INSTALLED_APPS :
settings . INSTALLED_APPS . append ( model_label )
2006-08-27 13:59:47 +00:00
except Exception , e :
sys . stderr . write ( " Error while importing %s : " % model_name + ' ' . join ( traceback . format_exception ( * sys . exc_info ( ) ) [ 1 : ] ) )
2006-12-15 06:06:52 +00:00
continue
2006-08-27 13:59:47 +00:00
2007-02-10 04:01:19 +00:00
# Add tests for invalid models.
2006-08-27 13:59:47 +00:00
extra_tests = [ ]
for model_dir , model_name in get_invalid_models ( ) :
model_label = ' . ' . join ( [ model_dir , model_name ] )
2007-07-28 04:02:52 +00:00
if not test_labels or model_name in test_labels :
2006-08-27 13:59:47 +00:00
extra_tests . append ( InvalidModelTestCase ( model_label ) )
2008-06-06 12:55:38 +00:00
try :
2008-10-02 12:57:13 +00:00
# Invalid models are not working apps, so we cannot pass them into
2008-06-06 12:55:38 +00:00
# the test runner with the other test_labels
test_labels . remove ( model_name )
except ValueError :
pass
2008-10-02 12:57:13 +00:00
2006-08-27 13:59:47 +00:00
# Run the test suite, including the extra validation tests.
2009-02-28 04:46:38 +00:00
from django . test . utils import get_runner
if not hasattr ( settings , ' TEST_RUNNER ' ) :
2010-01-18 15:11:01 +00:00
settings . TEST_RUNNER = ' django.test.simple.DjangoTestSuiteRunner '
TestRunner = get_runner ( settings )
if hasattr ( TestRunner , ' func_name ' ) :
# Pre 1.2 test runners were just functions,
# and did not support the 'failfast' option.
import warnings
warnings . warn (
' Function-based test runners are deprecated. Test runners should be classes with a run_tests() method. ' ,
PendingDeprecationWarning
)
failures = TestRunner ( test_labels , verbosity = verbosity , interactive = interactive ,
extra_tests = extra_tests )
else :
test_runner = TestRunner ( verbosity = verbosity , interactive = interactive , failfast = failfast )
failures = test_runner . run_tests ( test_labels , extra_tests = extra_tests )
2009-02-28 04:46:38 +00:00
2007-02-26 12:52:01 +00:00
if failures :
2010-01-03 18:52:25 +00:00
sys . exit ( bool ( failures ) )
2006-12-15 06:06:52 +00:00
2007-02-10 04:01:19 +00:00
# Restore the old settings.
2006-08-27 13:59:47 +00:00
settings . INSTALLED_APPS = old_installed_apps
2006-09-02 09:34:40 +00:00
settings . ROOT_URLCONF = old_root_urlconf
settings . TEMPLATE_DIRS = old_template_dirs
2006-12-15 06:06:52 +00:00
settings . USE_I18N = old_use_i18n
2007-10-21 17:26:32 +00:00
settings . LANGUAGE_CODE = old_language_code
2007-12-17 10:31:20 +00:00
settings . LOGIN_URL = old_login_url
2007-02-10 04:01:19 +00:00
settings . MIDDLEWARE_CLASSES = old_middleware_classes
2006-12-15 06:06:52 +00:00
2005-07-29 15:15:40 +00:00
if __name__ == " __main__ " :
from optparse import OptionParser
2005-09-19 01:18:04 +00:00
usage = " % prog [options] [model model model ...] "
2006-02-18 20:08:20 +00:00
parser = OptionParser ( usage = usage )
2010-08-07 06:58:14 +00:00
parser . add_option ( ' -v ' , ' --verbosity ' , action = ' store ' , dest = ' verbosity ' , default = ' 1 ' ,
type = ' choice ' , choices = [ ' 0 ' , ' 1 ' , ' 2 ' , ' 3 ' ] ,
2006-12-15 06:06:52 +00:00
help = ' Verbosity level; 0=minimal output, 1=normal output, 2=all output ' )
2007-07-23 12:14:32 +00:00
parser . add_option ( ' --noinput ' , action = ' store_false ' , dest = ' interactive ' , default = True ,
help = ' Tells Django to NOT prompt the user for input of any kind. ' )
2009-12-13 16:24:36 +00:00
parser . add_option ( ' --failfast ' , action = ' store_true ' , dest = ' failfast ' , default = False ,
help = ' Tells Django to stop running the test suite after first failed test. ' )
2005-08-10 15:36:16 +00:00
parser . add_option ( ' --settings ' ,
2006-01-01 18:37:33 +00:00
help = ' Python path to settings module, e.g. " myproject.settings " . If this isn \' t provided, the DJANGO_SETTINGS_MODULE environment variable will be used. ' )
2005-07-29 15:15:40 +00:00
options , args = parser . parse_args ( )
2005-08-10 15:36:16 +00:00
if options . settings :
os . environ [ ' DJANGO_SETTINGS_MODULE ' ] = options . settings
2007-02-26 22:34:56 +00:00
elif " DJANGO_SETTINGS_MODULE " not in os . environ :
parser . error ( " DJANGO_SETTINGS_MODULE is not set in the environment. "
" Set it or use --settings. " )
2009-12-13 16:24:36 +00:00
django_tests ( int ( options . verbosity ) , options . interactive , options . failfast , args )