tests.py 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. from django import forms
  2. from django.core.exceptions import NON_FIELD_ERRORS
  3. from django.test import TestCase
  4. from django.utils.functional import lazy
  5. from . import ValidationAssertions
  6. from .models import (
  7. Article,
  8. Author,
  9. GenericIPAddressTestModel,
  10. GenericIPAddrUnpackUniqueTest,
  11. ModelToValidate,
  12. )
  13. class BaseModelValidationTests(ValidationAssertions, TestCase):
  14. def test_missing_required_field_raises_error(self):
  15. mtv = ModelToValidate(f_with_custom_validator=42)
  16. self.assertFailsValidation(mtv.full_clean, ["name", "number"])
  17. def test_with_correct_value_model_validates(self):
  18. mtv = ModelToValidate(number=10, name="Some Name")
  19. self.assertIsNone(mtv.full_clean())
  20. def test_custom_validate_method(self):
  21. mtv = ModelToValidate(number=11)
  22. self.assertFailsValidation(mtv.full_clean, [NON_FIELD_ERRORS, "name"])
  23. def test_wrong_FK_value_raises_error(self):
  24. mtv = ModelToValidate(number=10, name="Some Name", parent_id=3)
  25. self.assertFieldFailsValidationWithMessage(
  26. mtv.full_clean,
  27. "parent",
  28. ["model to validate instance with id %r does not exist." % mtv.parent_id],
  29. )
  30. mtv = ModelToValidate(number=10, name="Some Name", ufm_id="Some Name")
  31. self.assertFieldFailsValidationWithMessage(
  32. mtv.full_clean,
  33. "ufm",
  34. [
  35. "unique fields model instance with unique_charfield %r does not exist."
  36. % mtv.name
  37. ],
  38. )
  39. def test_correct_FK_value_validates(self):
  40. parent = ModelToValidate.objects.create(number=10, name="Some Name")
  41. mtv = ModelToValidate(number=10, name="Some Name", parent_id=parent.pk)
  42. self.assertIsNone(mtv.full_clean())
  43. def test_limited_FK_raises_error(self):
  44. # The limit_choices_to on the parent field says that a parent object's
  45. # number attribute must be 10, so this should fail validation.
  46. parent = ModelToValidate.objects.create(number=11, name="Other Name")
  47. mtv = ModelToValidate(number=10, name="Some Name", parent_id=parent.pk)
  48. self.assertFailsValidation(mtv.full_clean, ["parent"])
  49. def test_FK_validates_using_base_manager(self):
  50. # Archived articles are not available through the default manager, only
  51. # the base manager.
  52. author = Author.objects.create(name="Randy", archived=True)
  53. article = Article(title="My Article", author=author)
  54. self.assertIsNone(article.full_clean())
  55. def test_wrong_email_value_raises_error(self):
  56. mtv = ModelToValidate(number=10, name="Some Name", email="not-an-email")
  57. self.assertFailsValidation(mtv.full_clean, ["email"])
  58. def test_correct_email_value_passes(self):
  59. mtv = ModelToValidate(number=10, name="Some Name", email="valid@email.com")
  60. self.assertIsNone(mtv.full_clean())
  61. def test_wrong_url_value_raises_error(self):
  62. mtv = ModelToValidate(number=10, name="Some Name", url="not a url")
  63. self.assertFieldFailsValidationWithMessage(
  64. mtv.full_clean, "url", ["Enter a valid URL."]
  65. )
  66. def test_text_greater_that_charfields_max_length_raises_errors(self):
  67. mtv = ModelToValidate(number=10, name="Some Name" * 100)
  68. self.assertFailsValidation(mtv.full_clean, ["name"])
  69. def test_malformed_slug_raises_error(self):
  70. mtv = ModelToValidate(number=10, name="Some Name", slug="##invalid##")
  71. self.assertFailsValidation(mtv.full_clean, ["slug"])
  72. def test_full_clean_does_not_mutate_exclude(self):
  73. mtv = ModelToValidate(f_with_custom_validator=42)
  74. exclude = ["number"]
  75. self.assertFailsValidation(mtv.full_clean, ["name"], exclude=exclude)
  76. self.assertEqual(len(exclude), 1)
  77. self.assertEqual(exclude[0], "number")
  78. class ArticleForm(forms.ModelForm):
  79. class Meta:
  80. model = Article
  81. exclude = ["author"]
  82. class ModelFormsTests(TestCase):
  83. @classmethod
  84. def setUpTestData(cls):
  85. cls.author = Author.objects.create(name="Joseph Kocherhans")
  86. def test_partial_validation(self):
  87. # Make sure the "commit=False and set field values later" idiom still
  88. # works with model validation.
  89. data = {
  90. "title": "The state of model validation",
  91. "pub_date": "2010-1-10 14:49:00",
  92. }
  93. form = ArticleForm(data)
  94. self.assertEqual(list(form.errors), [])
  95. article = form.save(commit=False)
  96. article.author = self.author
  97. article.save()
  98. def test_validation_with_empty_blank_field(self):
  99. # Since a value for pub_date wasn't provided and the field is
  100. # blank=True, model-validation should pass.
  101. # Also, Article.clean() should be run, so pub_date will be filled after
  102. # validation, so the form should save cleanly even though pub_date is
  103. # not allowed to be null.
  104. data = {
  105. "title": "The state of model validation",
  106. }
  107. article = Article(author_id=self.author.id)
  108. form = ArticleForm(data, instance=article)
  109. self.assertEqual(list(form.errors), [])
  110. self.assertIsNotNone(form.instance.pub_date)
  111. article = form.save()
  112. def test_validation_with_invalid_blank_field(self):
  113. # Even though pub_date is set to blank=True, an invalid value was
  114. # provided, so it should fail validation.
  115. data = {"title": "The state of model validation", "pub_date": "never"}
  116. article = Article(author_id=self.author.id)
  117. form = ArticleForm(data, instance=article)
  118. self.assertEqual(list(form.errors), ["pub_date"])
  119. class GenericIPAddressFieldTests(ValidationAssertions, TestCase):
  120. def test_correct_generic_ip_passes(self):
  121. giptm = GenericIPAddressTestModel(generic_ip="1.2.3.4")
  122. self.assertIsNone(giptm.full_clean())
  123. giptm = GenericIPAddressTestModel(generic_ip=" 1.2.3.4 ")
  124. self.assertIsNone(giptm.full_clean())
  125. giptm = GenericIPAddressTestModel(generic_ip="1.2.3.4\n")
  126. self.assertIsNone(giptm.full_clean())
  127. giptm = GenericIPAddressTestModel(generic_ip="2001::2")
  128. self.assertIsNone(giptm.full_clean())
  129. def test_invalid_generic_ip_raises_error(self):
  130. giptm = GenericIPAddressTestModel(generic_ip="294.4.2.1")
  131. self.assertFailsValidation(giptm.full_clean, ["generic_ip"])
  132. giptm = GenericIPAddressTestModel(generic_ip="1:2")
  133. self.assertFailsValidation(giptm.full_clean, ["generic_ip"])
  134. giptm = GenericIPAddressTestModel(generic_ip=1)
  135. self.assertFailsValidation(giptm.full_clean, ["generic_ip"])
  136. giptm = GenericIPAddressTestModel(generic_ip=lazy(lambda: 1, int))
  137. self.assertFailsValidation(giptm.full_clean, ["generic_ip"])
  138. def test_correct_v4_ip_passes(self):
  139. giptm = GenericIPAddressTestModel(v4_ip="1.2.3.4")
  140. self.assertIsNone(giptm.full_clean())
  141. def test_invalid_v4_ip_raises_error(self):
  142. giptm = GenericIPAddressTestModel(v4_ip="294.4.2.1")
  143. self.assertFailsValidation(giptm.full_clean, ["v4_ip"])
  144. giptm = GenericIPAddressTestModel(v4_ip="2001::2")
  145. self.assertFailsValidation(giptm.full_clean, ["v4_ip"])
  146. def test_correct_v6_ip_passes(self):
  147. giptm = GenericIPAddressTestModel(v6_ip="2001::2")
  148. self.assertIsNone(giptm.full_clean())
  149. def test_invalid_v6_ip_raises_error(self):
  150. giptm = GenericIPAddressTestModel(v6_ip="1.2.3.4")
  151. self.assertFailsValidation(giptm.full_clean, ["v6_ip"])
  152. giptm = GenericIPAddressTestModel(v6_ip="1:2")
  153. self.assertFailsValidation(giptm.full_clean, ["v6_ip"])
  154. def test_v6_uniqueness_detection(self):
  155. # These two addresses are the same with different syntax
  156. giptm = GenericIPAddressTestModel(generic_ip="2001::1:0:0:0:0:2")
  157. giptm.save()
  158. giptm = GenericIPAddressTestModel(generic_ip="2001:0:1:2")
  159. self.assertFailsValidation(giptm.full_clean, ["generic_ip"])
  160. def test_v4_unpack_uniqueness_detection(self):
  161. # These two are different, because we are not doing IPv4 unpacking
  162. giptm = GenericIPAddressTestModel(generic_ip="::ffff:10.10.10.10")
  163. giptm.save()
  164. giptm = GenericIPAddressTestModel(generic_ip="10.10.10.10")
  165. self.assertIsNone(giptm.full_clean())
  166. # These two are the same, because we are doing IPv4 unpacking
  167. giptm = GenericIPAddrUnpackUniqueTest(generic_v4unpack_ip="::ffff:18.52.18.52")
  168. giptm.save()
  169. giptm = GenericIPAddrUnpackUniqueTest(generic_v4unpack_ip="18.52.18.52")
  170. self.assertFailsValidation(giptm.full_clean, ["generic_v4unpack_ip"])
  171. def test_empty_generic_ip_passes(self):
  172. giptm = GenericIPAddressTestModel(generic_ip="")
  173. self.assertIsNone(giptm.full_clean())
  174. giptm = GenericIPAddressTestModel(generic_ip=None)
  175. self.assertIsNone(giptm.full_clean())