2009-05-02 15:03:33 +08:00
|
|
|
from datetime import date
|
|
|
|
|
2009-04-18 23:51:11 +08:00
|
|
|
from django import db
|
|
|
|
from django import forms
|
2009-05-02 15:03:33 +08:00
|
|
|
from django.forms.models import modelform_factory
|
2009-04-18 23:51:11 +08:00
|
|
|
from django.conf import settings
|
|
|
|
from django.test import TestCase
|
2009-05-02 15:03:33 +08:00
|
|
|
|
2010-02-24 04:02:18 +08:00
|
|
|
from models import Person, RealPerson, Triple, FilePathModel, Article, Publication, CustomFF, Author, Author1
|
2010-02-10 08:34:45 +08:00
|
|
|
|
2009-04-18 23:51:11 +08:00
|
|
|
|
|
|
|
class ModelMultipleChoiceFieldTests(TestCase):
|
2009-05-02 15:03:33 +08:00
|
|
|
|
2009-04-18 23:51:11 +08:00
|
|
|
def setUp(self):
|
|
|
|
self.old_debug = settings.DEBUG
|
|
|
|
settings.DEBUG = True
|
2009-05-02 15:03:33 +08:00
|
|
|
|
2009-04-18 23:51:11 +08:00
|
|
|
def tearDown(self):
|
|
|
|
settings.DEBUG = self.old_debug
|
2009-05-02 15:03:33 +08:00
|
|
|
|
2009-04-18 23:51:11 +08:00
|
|
|
def test_model_multiple_choice_number_of_queries(self):
|
|
|
|
"""
|
|
|
|
Test that ModelMultipleChoiceField does O(1) queries instead of
|
|
|
|
O(n) (#10156).
|
|
|
|
"""
|
|
|
|
for i in range(30):
|
|
|
|
Person.objects.create(name="Person %s" % i)
|
2009-05-02 15:03:33 +08:00
|
|
|
|
2009-04-18 23:51:11 +08:00
|
|
|
db.reset_queries()
|
|
|
|
f = forms.ModelMultipleChoiceField(queryset=Person.objects.all())
|
|
|
|
selected = f.clean([1, 3, 5, 7, 9])
|
2009-05-02 15:03:33 +08:00
|
|
|
self.assertEquals(len(db.connection.queries), 1)
|
2009-04-18 23:51:11 +08:00
|
|
|
|
|
|
|
class TripleForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Triple
|
|
|
|
|
|
|
|
class UniqueTogetherTests(TestCase):
|
|
|
|
def test_multiple_field_unique_together(self):
|
|
|
|
"""
|
|
|
|
When the same field is involved in multiple unique_together
|
|
|
|
constraints, we need to make sure we don't remove the data for it
|
|
|
|
before doing all the validation checking (not just failing after
|
|
|
|
the first one).
|
|
|
|
"""
|
|
|
|
Triple.objects.create(left=1, middle=2, right=3)
|
|
|
|
|
|
|
|
form = TripleForm({'left': '1', 'middle': '2', 'right': '3'})
|
|
|
|
self.failIf(form.is_valid())
|
|
|
|
|
|
|
|
form = TripleForm({'left': '1', 'middle': '3', 'right': '1'})
|
|
|
|
self.failUnless(form.is_valid())
|
|
|
|
|
2010-01-21 10:28:03 +08:00
|
|
|
class TripleFormWithCleanOverride(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Triple
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
if not self.cleaned_data['left'] == self.cleaned_data['right']:
|
|
|
|
raise forms.ValidationError('Left and right should be equal')
|
|
|
|
return self.cleaned_data
|
|
|
|
|
|
|
|
class OverrideCleanTests(TestCase):
|
|
|
|
def test_override_clean(self):
|
|
|
|
"""
|
|
|
|
Regression for #12596: Calling super from ModelForm.clean() should be
|
|
|
|
optional.
|
|
|
|
"""
|
|
|
|
form = TripleFormWithCleanOverride({'left': 1, 'middle': 2, 'right': 1})
|
|
|
|
self.failUnless(form.is_valid())
|
|
|
|
# form.instance.left will be None if the instance was not constructed
|
|
|
|
# by form.full_clean().
|
|
|
|
self.assertEquals(form.instance.left, 1)
|
|
|
|
|
2010-03-07 02:42:56 +08:00
|
|
|
# Regression test for #12960.
|
|
|
|
# Make sure the cleaned_data returned from ModelForm.clean() is applied to the
|
|
|
|
# model instance.
|
|
|
|
|
|
|
|
class PublicationForm(forms.ModelForm):
|
|
|
|
def clean(self):
|
|
|
|
self.cleaned_data['title'] = self.cleaned_data['title'].upper()
|
|
|
|
return self.cleaned_data
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = Publication
|
|
|
|
|
|
|
|
class ModelFormCleanTest(TestCase):
|
|
|
|
def test_model_form_clean_applies_to_model(self):
|
|
|
|
data = {'title': 'test', 'date_published': '2010-2-25'}
|
|
|
|
form = PublicationForm(data)
|
|
|
|
publication = form.save()
|
|
|
|
self.assertEqual(publication.title, 'TEST')
|
|
|
|
|
2009-04-18 23:51:11 +08:00
|
|
|
class FPForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = FilePathModel
|
|
|
|
|
|
|
|
class FilePathFieldTests(TestCase):
|
|
|
|
def test_file_path_field_blank(self):
|
|
|
|
"""
|
|
|
|
Regression test for #8842: FilePathField(blank=True)
|
|
|
|
"""
|
|
|
|
form = FPForm()
|
|
|
|
names = [p[1] for p in form['path'].field.choices]
|
|
|
|
names.sort()
|
|
|
|
self.assertEqual(names, ['---------', '__init__.py', 'models.py', 'tests.py'])
|
2009-05-02 15:03:33 +08:00
|
|
|
|
|
|
|
class ManyToManyCallableInitialTests(TestCase):
|
|
|
|
def test_callable(self):
|
|
|
|
"Regression for #10349: A callable can be provided as the initial value for an m2m field"
|
|
|
|
|
|
|
|
# Set up a callable initial value
|
|
|
|
def formfield_for_dbfield(db_field, **kwargs):
|
|
|
|
if db_field.name == 'publications':
|
2009-05-07 22:48:04 +08:00
|
|
|
kwargs['initial'] = lambda: Publication.objects.all().order_by('date_published')[:2]
|
2009-05-02 15:03:33 +08:00
|
|
|
return db_field.formfield(**kwargs)
|
|
|
|
|
|
|
|
# Set up some Publications to use as data
|
2009-05-07 22:48:04 +08:00
|
|
|
Publication(title="First Book", date_published=date(2007,1,1)).save()
|
|
|
|
Publication(title="Second Book", date_published=date(2008,1,1)).save()
|
|
|
|
Publication(title="Third Book", date_published=date(2009,1,1)).save()
|
2009-05-02 15:03:33 +08:00
|
|
|
|
|
|
|
# Create a ModelForm, instantiate it, and check that the output is as expected
|
|
|
|
ModelForm = modelform_factory(Article, formfield_callback=formfield_for_dbfield)
|
|
|
|
form = ModelForm()
|
|
|
|
self.assertEquals(form.as_ul(), u"""<li><label for="id_headline">Headline:</label> <input id="id_headline" type="text" name="headline" maxlength="100" /></li>
|
|
|
|
<li><label for="id_publications">Publications:</label> <select multiple="multiple" name="publications" id="id_publications">
|
|
|
|
<option value="1" selected="selected">First Book</option>
|
|
|
|
<option value="2" selected="selected">Second Book</option>
|
|
|
|
<option value="3">Third Book</option>
|
|
|
|
</select> Hold down "Control", or "Command" on a Mac, to select more than one.</li>""")
|
2009-05-20 07:13:33 +08:00
|
|
|
|
|
|
|
class CFFForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = CustomFF
|
|
|
|
|
|
|
|
class CustomFieldSaveTests(TestCase):
|
|
|
|
def test_save(self):
|
|
|
|
"Regression for #11149: save_form_data should be called only once"
|
2010-02-10 08:34:45 +08:00
|
|
|
|
2009-05-20 07:13:33 +08:00
|
|
|
# It's enough that the form saves without error -- the custom save routine will
|
|
|
|
# generate an AssertionError if it is called more than once during save.
|
|
|
|
form = CFFForm(data = {'f': None})
|
2009-12-14 01:46:52 +08:00
|
|
|
form.save()
|
|
|
|
|
|
|
|
class ModelChoiceIteratorTests(TestCase):
|
|
|
|
def test_len(self):
|
|
|
|
class Form(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Article
|
|
|
|
fields = ["publications"]
|
2010-02-10 08:34:45 +08:00
|
|
|
|
2009-12-14 01:46:52 +08:00
|
|
|
Publication.objects.create(title="Pravda",
|
|
|
|
date_published=date(1991, 8, 22))
|
|
|
|
f = Form()
|
|
|
|
self.assertEqual(len(f.fields["publications"].choices), 1)
|
2010-02-10 08:34:45 +08:00
|
|
|
|
|
|
|
class RealPersonForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = RealPerson
|
|
|
|
|
|
|
|
class CustomModelFormSaveMethod(TestCase):
|
|
|
|
def test_string_message(self):
|
|
|
|
data = {'name': 'anonymous'}
|
|
|
|
form = RealPersonForm(data)
|
|
|
|
self.assertEqual(form.is_valid(), False)
|
2010-02-23 22:59:30 +08:00
|
|
|
self.assertEqual(form.errors['__all__'], ['Please specify a real name.'])
|
2010-02-10 08:34:45 +08:00
|
|
|
|
2010-02-23 22:59:30 +08:00
|
|
|
class ModelClassTests(TestCase):
|
|
|
|
def test_no_model_class(self):
|
|
|
|
class NoModelModelForm(forms.ModelForm):
|
|
|
|
pass
|
|
|
|
self.assertRaises(ValueError, NoModelModelForm)
|
2010-02-24 04:02:18 +08:00
|
|
|
|
|
|
|
class OneToOneFieldTests(TestCase):
|
|
|
|
def test_assignment_of_none(self):
|
|
|
|
class AuthorForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Author
|
|
|
|
fields = ['publication', 'full_name']
|
|
|
|
|
|
|
|
publication = Publication.objects.create(title="Pravda",
|
|
|
|
date_published=date(1991, 8, 22))
|
|
|
|
author = Author.objects.create(publication=publication, full_name='John Doe')
|
|
|
|
form = AuthorForm({'publication':u'', 'full_name':'John Doe'}, instance=author)
|
|
|
|
self.assert_(form.is_valid())
|
|
|
|
self.assertEqual(form.cleaned_data['publication'], None)
|
|
|
|
author = form.save()
|
|
|
|
# author object returned from form still retains original publication object
|
|
|
|
# that's why we need to retreive it from database again
|
|
|
|
new_author = Author.objects.get(pk=author.pk)
|
|
|
|
self.assertEqual(new_author.publication, None)
|
|
|
|
|
|
|
|
def test_assignment_of_none_null_false(self):
|
|
|
|
class AuthorForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Author1
|
|
|
|
fields = ['publication', 'full_name']
|
|
|
|
|
|
|
|
publication = Publication.objects.create(title="Pravda",
|
|
|
|
date_published=date(1991, 8, 22))
|
|
|
|
author = Author1.objects.create(publication=publication, full_name='John Doe')
|
|
|
|
form = AuthorForm({'publication':u'', 'full_name':'John Doe'}, instance=author)
|
|
|
|
self.assert_(not form.is_valid())
|
|
|
|
|