2010-01-12 10:29:45 +08:00
|
|
|
from django import forms
|
|
|
|
from django.test import TestCase
|
|
|
|
from django.core.exceptions import NON_FIELD_ERRORS
|
2010-01-05 11:56:19 +08:00
|
|
|
from modeltests.validation import ValidationTestCase
|
2011-06-11 21:48:24 +08:00
|
|
|
from modeltests.validation.models import (Author, Article, ModelToValidate,
|
|
|
|
GenericIPAddressTestModel, GenericIPAddressWithUnpackUniqueTestModel)
|
2010-01-05 11:56:19 +08:00
|
|
|
|
2010-01-12 10:29:45 +08:00
|
|
|
# Import other tests for this package.
|
|
|
|
from modeltests.validation.validators import TestModelsWithValidators
|
2010-10-12 11:33:19 +08:00
|
|
|
from modeltests.validation.test_unique import (GetUniqueCheckTests,
|
|
|
|
PerformUniqueChecksTest)
|
2010-01-12 10:29:45 +08:00
|
|
|
from modeltests.validation.test_custom_messages import CustomMessagesTest
|
2010-01-05 11:56:19 +08:00
|
|
|
|
|
|
|
|
|
|
|
class BaseModelValidationTests(ValidationTestCase):
|
|
|
|
|
|
|
|
def test_missing_required_field_raises_error(self):
|
|
|
|
mtv = ModelToValidate(f_with_custom_validator=42)
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertFailsValidation(mtv.full_clean, ['name', 'number'])
|
2010-01-05 11:56:19 +08:00
|
|
|
|
|
|
|
def test_with_correct_value_model_validates(self):
|
|
|
|
mtv = ModelToValidate(number=10, name='Some Name')
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertEqual(None, mtv.full_clean())
|
2010-01-05 11:56:19 +08:00
|
|
|
|
2010-01-12 10:29:45 +08:00
|
|
|
def test_custom_validate_method(self):
|
2010-01-05 11:56:19 +08:00
|
|
|
mtv = ModelToValidate(number=11)
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertFailsValidation(mtv.full_clean, [NON_FIELD_ERRORS, 'name'])
|
2010-01-05 11:56:19 +08:00
|
|
|
|
|
|
|
def test_wrong_FK_value_raises_error(self):
|
|
|
|
mtv=ModelToValidate(number=10, name='Some Name', parent_id=3)
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertFailsValidation(mtv.full_clean, ['parent'])
|
2010-01-05 11:56:19 +08:00
|
|
|
|
|
|
|
def test_correct_FK_value_validates(self):
|
|
|
|
parent = ModelToValidate.objects.create(number=10, name='Some Name')
|
2010-01-21 11:26:14 +08:00
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', parent_id=parent.pk)
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertEqual(None, mtv.full_clean())
|
2010-01-05 11:56:19 +08:00
|
|
|
|
2010-09-13 13:08:21 +08:00
|
|
|
def test_limited_FK_raises_error(self):
|
2010-01-21 11:26:14 +08:00
|
|
|
# The limit_choices_to on the parent field says that a parent object's
|
|
|
|
# number attribute must be 10, so this should fail validation.
|
|
|
|
parent = ModelToValidate.objects.create(number=11, name='Other Name')
|
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', parent_id=parent.pk)
|
|
|
|
self.assertFailsValidation(mtv.full_clean, ['parent'])
|
|
|
|
|
2010-01-05 11:56:19 +08:00
|
|
|
def test_wrong_email_value_raises_error(self):
|
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', email='not-an-email')
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertFailsValidation(mtv.full_clean, ['email'])
|
2010-01-05 11:56:19 +08:00
|
|
|
|
|
|
|
def test_correct_email_value_passes(self):
|
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', email='valid@email.com')
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertEqual(None, mtv.full_clean())
|
2010-01-05 11:56:19 +08:00
|
|
|
|
|
|
|
def test_wrong_url_value_raises_error(self):
|
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', url='not a url')
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertFieldFailsValidationWithMessage(mtv.full_clean, 'url', [u'Enter a valid value.'])
|
2010-01-05 11:56:19 +08:00
|
|
|
|
|
|
|
def test_correct_url_but_nonexisting_gives_404(self):
|
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', url='http://google.com/we-love-microsoft.html')
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertFieldFailsValidationWithMessage(mtv.full_clean, 'url', [u'This URL appears to be a broken link.'])
|
2010-01-05 11:56:19 +08:00
|
|
|
|
|
|
|
def test_correct_url_value_passes(self):
|
2011-05-06 07:11:03 +08:00
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', url='http://www.example.com/')
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertEqual(None, mtv.full_clean()) # This will fail if there's no Internet connection
|
2010-01-05 11:56:19 +08:00
|
|
|
|
2011-03-16 11:38:34 +08:00
|
|
|
def test_correct_https_url_but_nonexisting(self):
|
2011-05-06 07:11:03 +08:00
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', url='https://www.example.com/')
|
2011-03-16 11:38:34 +08:00
|
|
|
self.assertFieldFailsValidationWithMessage(mtv.full_clean, 'url', [u'This URL appears to be a broken link.'])
|
|
|
|
|
|
|
|
def test_correct_ftp_url_but_nonexisting(self):
|
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', url='ftp://ftp.google.com/we-love-microsoft.html')
|
|
|
|
self.assertFieldFailsValidationWithMessage(mtv.full_clean, 'url', [u'This URL appears to be a broken link.'])
|
|
|
|
|
|
|
|
def test_correct_ftps_url_but_nonexisting(self):
|
|
|
|
mtv = ModelToValidate(number=10, name='Some Name', url='ftps://ftp.google.com/we-love-microsoft.html')
|
|
|
|
self.assertFieldFailsValidationWithMessage(mtv.full_clean, 'url', [u'This URL appears to be a broken link.'])
|
|
|
|
|
2010-09-13 13:08:21 +08:00
|
|
|
def test_text_greater_that_charfields_max_length_raises_erros(self):
|
2010-01-05 11:56:19 +08:00
|
|
|
mtv = ModelToValidate(number=10, name='Some Name'*100)
|
2010-01-12 10:29:45 +08:00
|
|
|
self.assertFailsValidation(mtv.full_clean, ['name',])
|
|
|
|
|
2011-06-11 21:48:24 +08:00
|
|
|
|
2010-01-12 10:29:45 +08:00
|
|
|
class ArticleForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Article
|
|
|
|
exclude = ['author']
|
|
|
|
|
|
|
|
class ModelFormsTests(TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
self.author = Author.objects.create(name='Joseph Kocherhans')
|
|
|
|
|
|
|
|
def test_partial_validation(self):
|
|
|
|
# Make sure the "commit=False and set field values later" idiom still
|
|
|
|
# works with model validation.
|
|
|
|
data = {
|
|
|
|
'title': 'The state of model validation',
|
|
|
|
'pub_date': '2010-1-10 14:49:00'
|
|
|
|
}
|
|
|
|
form = ArticleForm(data)
|
|
|
|
self.assertEqual(form.errors.keys(), [])
|
|
|
|
article = form.save(commit=False)
|
|
|
|
article.author = self.author
|
|
|
|
article.save()
|
|
|
|
|
|
|
|
def test_validation_with_empty_blank_field(self):
|
|
|
|
# Since a value for pub_date wasn't provided and the field is
|
|
|
|
# blank=True, model-validation should pass.
|
|
|
|
# Also, Article.clean() should be run, so pub_date will be filled after
|
|
|
|
# validation, so the form should save cleanly even though pub_date is
|
|
|
|
# not allowed to be null.
|
|
|
|
data = {
|
|
|
|
'title': 'The state of model validation',
|
|
|
|
}
|
|
|
|
article = Article(author_id=self.author.id)
|
|
|
|
form = ArticleForm(data, instance=article)
|
|
|
|
self.assertEqual(form.errors.keys(), [])
|
|
|
|
self.assertNotEqual(form.instance.pub_date, None)
|
|
|
|
article = form.save()
|
|
|
|
|
|
|
|
def test_validation_with_invalid_blank_field(self):
|
|
|
|
# Even though pub_date is set to blank=True, an invalid value was
|
|
|
|
# provided, so it should fail validation.
|
|
|
|
data = {
|
|
|
|
'title': 'The state of model validation',
|
|
|
|
'pub_date': 'never'
|
|
|
|
}
|
|
|
|
article = Article(author_id=self.author.id)
|
|
|
|
form = ArticleForm(data, instance=article)
|
|
|
|
self.assertEqual(form.errors.keys(), ['pub_date'])
|
2011-06-11 21:48:24 +08:00
|
|
|
|
|
|
|
|
|
|
|
class GenericIPAddressFieldTests(ValidationTestCase):
|
|
|
|
|
|
|
|
def test_correct_generic_ip_passes(self):
|
|
|
|
giptm = GenericIPAddressTestModel(generic_ip="1.2.3.4")
|
|
|
|
self.assertEqual(None, giptm.full_clean())
|
|
|
|
giptm = GenericIPAddressTestModel(generic_ip="2001::2")
|
|
|
|
self.assertEqual(None, giptm.full_clean())
|
|
|
|
|
|
|
|
def test_invalid_generic_ip_raises_error(self):
|
|
|
|
giptm = GenericIPAddressTestModel(generic_ip="294.4.2.1")
|
|
|
|
self.assertFailsValidation(giptm.full_clean, ['generic_ip',])
|
|
|
|
giptm = GenericIPAddressTestModel(generic_ip="1:2")
|
|
|
|
self.assertFailsValidation(giptm.full_clean, ['generic_ip',])
|
|
|
|
|
|
|
|
def test_correct_v4_ip_passes(self):
|
|
|
|
giptm = GenericIPAddressTestModel(v4_ip="1.2.3.4")
|
|
|
|
self.assertEqual(None, giptm.full_clean())
|
|
|
|
|
|
|
|
def test_invalid_v4_ip_raises_error(self):
|
|
|
|
giptm = GenericIPAddressTestModel(v4_ip="294.4.2.1")
|
|
|
|
self.assertFailsValidation(giptm.full_clean, ['v4_ip',])
|
|
|
|
giptm = GenericIPAddressTestModel(v4_ip="2001::2")
|
|
|
|
self.assertFailsValidation(giptm.full_clean, ['v4_ip',])
|
|
|
|
|
|
|
|
def test_correct_v6_ip_passes(self):
|
|
|
|
giptm = GenericIPAddressTestModel(v6_ip="2001::2")
|
|
|
|
self.assertEqual(None, giptm.full_clean())
|
|
|
|
|
|
|
|
def test_invalid_v6_ip_raises_error(self):
|
|
|
|
giptm = GenericIPAddressTestModel(v6_ip="1.2.3.4")
|
|
|
|
self.assertFailsValidation(giptm.full_clean, ['v6_ip',])
|
|
|
|
giptm = GenericIPAddressTestModel(v6_ip="1:2")
|
|
|
|
self.assertFailsValidation(giptm.full_clean, ['v6_ip',])
|
|
|
|
|
|
|
|
def test_v6_uniqueness_detection(self):
|
|
|
|
# These two addresses are the same with different syntax
|
|
|
|
giptm = GenericIPAddressTestModel(generic_ip="2001::1:0:0:0:0:2")
|
|
|
|
giptm.save()
|
|
|
|
giptm = GenericIPAddressTestModel(generic_ip="2001:0:1:2")
|
|
|
|
self.assertFailsValidation(giptm.full_clean, ['generic_ip',])
|
|
|
|
|
|
|
|
def test_v4_unpack_uniqueness_detection(self):
|
|
|
|
# These two are different, because we are not doing IPv4 unpacking
|
|
|
|
giptm = GenericIPAddressTestModel(generic_ip="::ffff:10.10.10.10")
|
|
|
|
giptm.save()
|
|
|
|
giptm = GenericIPAddressTestModel(generic_ip="10.10.10.10")
|
|
|
|
self.assertEqual(None, giptm.full_clean())
|
|
|
|
|
|
|
|
# These two are the same, because we are doing IPv4 unpacking
|
|
|
|
giptm = GenericIPAddressWithUnpackUniqueTestModel(generic_v4unpack_ip="::ffff:18.52.18.52")
|
|
|
|
giptm.save()
|
|
|
|
giptm = GenericIPAddressWithUnpackUniqueTestModel(generic_v4unpack_ip="18.52.18.52")
|
|
|
|
self.assertFailsValidation(giptm.full_clean, ['generic_v4unpack_ip',])
|