aboutsummaryrefslogtreecommitdiffstats
path: root/rest_framework/tests/fields.py
diff options
context:
space:
mode:
authorStephan Groß2013-02-26 11:09:54 +0100
committerStephan Groß2013-02-26 11:09:54 +0100
commitbfff356dd36f7d14d35d8a854cd314e60cf25efa (patch)
tree665c94459adadb8533f5abf14921e9a4a4568ac4 /rest_framework/tests/fields.py
parent8da83f0df9761550cbcac88850a659aab00a2506 (diff)
downloaddjango-rest-framework-bfff356dd36f7d14d35d8a854cd314e60cf25efa.tar.bz2
Add better date / datetime validation (pull 2)
addition to #631 with update to master + timefield support
Diffstat (limited to 'rest_framework/tests/fields.py')
-rw-r--r--rest_framework/tests/fields.py345
1 files changed, 331 insertions, 14 deletions
diff --git a/rest_framework/tests/fields.py b/rest_framework/tests/fields.py
index 34f61678..d21e247d 100644
--- a/rest_framework/tests/fields.py
+++ b/rest_framework/tests/fields.py
@@ -3,9 +3,13 @@ General serializer field tests.
"""
from __future__ import unicode_literals
import datetime
+
+import django
from django.db import models
from django.test import TestCase
from django.core import validators
+from django.utils import unittest
+
from rest_framework import serializers
@@ -18,6 +22,21 @@ class CharPrimaryKeyModel(models.Model):
id = models.CharField(max_length=20, primary_key=True)
+class DateObject(object):
+ def __init__(self, date):
+ self.date = date
+
+
+class DateTimeObject(object):
+ def __init__(self, date_time):
+ self.date_time = date_time
+
+
+class TimeObject(object):
+ def __init__(self, time):
+ self.time = time
+
+
class TimestampedModelSerializer(serializers.ModelSerializer):
class Meta:
model = TimestampedModel
@@ -28,6 +47,66 @@ class CharPrimaryKeyModelSerializer(serializers.ModelSerializer):
model = CharPrimaryKeyModel
+class DateObjectSerializer(serializers.Serializer):
+ date = serializers.DateField()
+
+ def restore_object(self, attrs, instance=None):
+ if instance is not None:
+ instance.date = attrs['date']
+ return instance
+ return DateObject(**attrs)
+
+
+class DateObjectCustomFormatSerializer(serializers.Serializer):
+ date = serializers.DateField(format=("%Y", "%Y -- %m"))
+
+ def restore_object(self, attrs, instance=None):
+ if instance is not None:
+ instance.date = attrs['date']
+ return instance
+ return DateObject(**attrs)
+
+
+class DateTimeObjectSerializer(serializers.Serializer):
+ date_time = serializers.DateTimeField()
+
+ def restore_object(self, attrs, instance=None):
+ if instance is not None:
+ instance.date_time = attrs['date_time']
+ return instance
+ return DateTimeObject(**attrs)
+
+
+class DateTimeObjectCustomFormatSerializer(serializers.Serializer):
+ date_time = serializers.DateTimeField(format=("%Y", "%Y %H:%M"))
+
+ def restore_object(self, attrs, instance=None):
+ if instance is not None:
+ instance.date_time = attrs['date_time']
+ return instance
+ return DateTimeObject(**attrs)
+
+
+class TimeObjectSerializer(serializers.Serializer):
+ time = serializers.TimeField()
+
+ def restore_object(self, attrs, instance=None):
+ if instance is not None:
+ instance.time = attrs['time']
+ return instance
+ return TimeObject(**attrs)
+
+
+class TimeObjectCustomFormatSerializer(serializers.Serializer):
+ time = serializers.TimeField(format=("%H -- %M", "%H%M%S"))
+
+ def restore_object(self, attrs, instance=None):
+ if instance is not None:
+ instance.time = attrs['time']
+ return instance
+ return TimeObject(**attrs)
+
+
class TimeFieldModel(models.Model):
clock = models.TimeField()
@@ -59,37 +138,275 @@ class BasicFieldTests(TestCase):
serializer = CharPrimaryKeyModelSerializer()
self.assertEquals(serializer.fields['id'].read_only, False)
- def test_TimeField_from_native(self):
+
+class DateFieldTest(TestCase):
+ def test_valid_default_date_input_formats(self):
+ serializer = DateObjectSerializer(data={'date': '1984-07-31'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': '07/31/1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': '07/31/84'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': 'Jul 31 1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': 'Jul 31, 1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': '31 Jul 1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': '31 Jul 1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': 'July 31 1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': 'July 31, 1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': '31 July 1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectSerializer(data={'date': '31 July, 1984'})
+ self.assertTrue(serializer.is_valid())
+
+ def test_valid_custom_date_input_formats(self):
+ serializer = DateObjectCustomFormatSerializer(data={'date': '1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateObjectCustomFormatSerializer(data={'date': '1984 -- 07'})
+ self.assertTrue(serializer.is_valid())
+
+ def test_wrong_default_date_input_format(self):
+ serializer = DateObjectSerializer(data={'date': 'something wrong'})
+ self.assertFalse(serializer.is_valid())
+ self.assertEquals(serializer.errors, {'date': [u'Date has wrong format. Use one of these formats instead: '
+ u'YYYY-MM-DD; MM/DD/YYYY; MM/DD/YY; [Jan through Dec] DD YYYY; '
+ u'[Jan through Dec] DD, YYYY; DD [Jan through Dec] YYYY; '
+ u'DD [Jan through Dec], YYYY; [January through December] DD YYYY; '
+ u'[January through December] DD, YYYY; DD [January through December] YYYY; '
+ u'DD [January through December], YYYY']})
+
+ def test_wrong_custom_date_input_format(self):
+ serializer = DateObjectCustomFormatSerializer(data={'date': '07/31/1984'})
+ self.assertFalse(serializer.is_valid())
+ self.assertEquals(serializer.errors, {'date': [u'Date has wrong format. Use one of these formats instead: YYYY; YYYY -- MM']})
+
+ def test_from_native(self):
+ f = serializers.DateField()
+ result = f.from_native('1984-07-31')
+
+ self.assertEqual(datetime.date(1984, 7, 31), result)
+
+ def test_from_native_datetime_date(self):
+ """
+ Make sure from_native() accepts a datetime.date instance.
+ """
+ f = serializers.DateField()
+ result = f.from_native(datetime.date(1984, 7, 31))
+
+ self.assertEqual(result, datetime.date(1984, 7, 31))
+
+ def test_from_native_empty(self):
+ f = serializers.DateField()
+ result = f.from_native('')
+
+ self.assertEqual(result, None)
+
+ def test_from_native_invalid_date(self):
+ f = serializers.DateField()
+
+ try:
+ f.from_native('1984-42-31')
+ except validators.ValidationError as e:
+ self.assertEqual(e.messages, [u'Date has wrong format. Use one of these formats instead: '
+ u'YYYY-MM-DD; MM/DD/YYYY; MM/DD/YY; [Jan through Dec] DD YYYY; '
+ u'[Jan through Dec] DD, YYYY; DD [Jan through Dec] YYYY; '
+ u'DD [Jan through Dec], YYYY; [January through December] DD YYYY; '
+ u'[January through December] DD, YYYY; DD [January through December] YYYY; '
+ u'DD [January through December], YYYY'])
+ else:
+ self.fail("ValidationError was not properly raised")
+
+
+class DateTimeFieldTest(TestCase):
+ def test_valid_default_date_time_input_formats(self):
+ serializer = DateTimeObjectSerializer(data={'date_time': '1984-07-31 04:31:59'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '1984-07-31 04:31'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '1984-07-31'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '07/31/1984 04:31:59'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '07/31/1984 04:31'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '07/31/1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '07/31/84 04:31:59'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '07/31/84 04:31'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '07/31/84'})
+ self.assertTrue(serializer.is_valid())
+
+ @unittest.skipUnless(django.VERSION >= (1, 4), "django < 1.4 don't have microseconds in default settings")
+ def test_valid_default_date_time_input_formats_for_django_gte_1_4(self):
+ serializer = DateTimeObjectSerializer(data={'date_time': '1984-07-31 04:31:59.123456'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '07/31/1984 04:31:59.123456'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectSerializer(data={'date_time': '07/31/84 04:31:59.123456'})
+ self.assertTrue(serializer.is_valid())
+
+ def test_valid_custom_date_time_input_formats(self):
+ serializer = DateTimeObjectCustomFormatSerializer(data={'date_time': '1984'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = DateTimeObjectCustomFormatSerializer(data={'date_time': '1984 04:31'})
+ self.assertTrue(serializer.is_valid())
+
+ @unittest.skipUnless(django.VERSION >= (1, 4), "django < 1.4 don't have microseconds in default settings")
+ def test_wrong_default_date_time_input_format_for_django_gte_1_4(self):
+ serializer = DateTimeObjectSerializer(data={'date_time': 'something wrong'})
+ self.assertFalse(serializer.is_valid())
+ self.assertEquals(serializer.errors, {'date_time': [u'Datetime has wrong format. Use one of these formats instead: '
+ u'YYYY-MM-DD HH:MM:SS; YYYY-MM-DD HH:MM:SS.uuuuuu; YYYY-MM-DD HH:MM; '
+ u'YYYY-MM-DD; MM/DD/YYYY HH:MM:SS; MM/DD/YYYY HH:MM:SS.uuuuuu; '
+ u'MM/DD/YYYY HH:MM; MM/DD/YYYY; MM/DD/YY HH:MM:SS; '
+ u'MM/DD/YY HH:MM:SS.uuuuuu; MM/DD/YY HH:MM; MM/DD/YY']})
+
+ @unittest.skipUnless(django.VERSION < (1, 4), "django >= 1.4 have microseconds in default settings")
+ def test_wrong_default_date_time_input_format_for_django_lt_1_4(self):
+ serializer = DateTimeObjectSerializer(data={'date_time': 'something wrong'})
+ self.assertFalse(serializer.is_valid())
+ self.assertEquals(serializer.errors, {'date_time': [u'Datetime has wrong format. Use one of these formats instead:'
+ u' YYYY-MM-DD HH:MM:SS; YYYY-MM-DD HH:MM; YYYY-MM-DD; '
+ u'MM/DD/YYYY HH:MM:SS; MM/DD/YYYY HH:MM; MM/DD/YYYY; '
+ u'MM/DD/YY HH:MM:SS; MM/DD/YY HH:MM; MM/DD/YY']})
+
+ def test_wrong_custom_date_time_input_format(self):
+ serializer = DateTimeObjectCustomFormatSerializer(data={'date_time': '07/31/84 04:31'})
+ self.assertFalse(serializer.is_valid())
+ self.assertEquals(serializer.errors, {'date_time': [u'Datetime has wrong format. Use one of these formats instead: YYYY; YYYY HH:MM']})
+
+ def test_from_native(self):
+ f = serializers.DateTimeField()
+ result = f.from_native('1984-07-31 04:31')
+
+ self.assertEqual(datetime.datetime(1984, 7, 31, 4, 31), result)
+
+ def test_from_native_datetime_datetime(self):
+ """
+ Make sure from_native() accepts a datetime.date instance.
+ """
+ f = serializers.DateTimeField()
+ result = f.from_native(datetime.datetime(1984, 7, 31))
+
+ self.assertEqual(result, datetime.datetime(1984, 7, 31))
+
+ def test_from_native_empty(self):
+ f = serializers.DateTimeField()
+ result = f.from_native('')
+
+ self.assertEqual(result, None)
+
+ @unittest.skipUnless(django.VERSION >= (1, 4), "django < 1.4 don't have microseconds in default settings")
+ def test_from_native_invalid_datetime(self):
+ f = serializers.DateTimeField()
+
+ try:
+ f.from_native('1984-42-31 04:31')
+ except validators.ValidationError as e:
+ self.assertEqual(e.messages, [u'Datetime has wrong format. Use one of these formats instead: '
+ u'YYYY-MM-DD HH:MM:SS; YYYY-MM-DD HH:MM:SS.uuuuuu; YYYY-MM-DD HH:MM; '
+ u'YYYY-MM-DD; MM/DD/YYYY HH:MM:SS; MM/DD/YYYY HH:MM:SS.uuuuuu; '
+ u'MM/DD/YYYY HH:MM; MM/DD/YYYY; MM/DD/YY HH:MM:SS; '
+ u'MM/DD/YY HH:MM:SS.uuuuuu; MM/DD/YY HH:MM; MM/DD/YY'])
+ else:
+ self.fail("ValidationError was not properly raised")
+
+ @unittest.skipUnless(django.VERSION < (1, 4), "django >= 1.4 have microseconds in default settings")
+ def test_from_native_invalid_datetime(self):
+ f = serializers.DateTimeField()
+
+ try:
+ f.from_native('1984-42-31 04:31')
+ except validators.ValidationError as e:
+ self.assertEqual(e.messages, [u'Datetime has wrong format. Use one of these formats instead:'
+ u' YYYY-MM-DD HH:MM:SS; YYYY-MM-DD HH:MM; YYYY-MM-DD; '
+ u'MM/DD/YYYY HH:MM:SS; MM/DD/YYYY HH:MM; MM/DD/YYYY; '
+ u'MM/DD/YY HH:MM:SS; MM/DD/YY HH:MM; MM/DD/YY'])
+ else:
+ self.fail("ValidationError was not properly raised")
+
+
+class TimeFieldTest(TestCase):
+ def test_valid_default_time_input_formats(self):
+ serializer = TimeObjectSerializer(data={'time': '04:31'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = TimeObjectSerializer(data={'time': '04:31:59'})
+ self.assertTrue(serializer.is_valid())
+
+ def test_valid_custom_time_input_formats(self):
+ serializer = TimeObjectCustomFormatSerializer(data={'time': '04 -- 31'})
+ self.assertTrue(serializer.is_valid())
+
+ serializer = TimeObjectCustomFormatSerializer(data={'time': '043159'})
+ self.assertTrue(serializer.is_valid())
+
+ def test_wrong_default_time_input_format(self):
+ serializer = TimeObjectSerializer(data={'time': 'something wrong'})
+ self.assertFalse(serializer.is_valid())
+ self.assertEquals(serializer.errors, {'time': [u'Time has wrong format. Use one of these formats instead: HH:MM:SS; HH:MM']})
+
+ def test_wrong_custom_time_input_format(self):
+ serializer = TimeObjectCustomFormatSerializer(data={'time': '04:31'})
+ self.assertFalse(serializer.is_valid())
+ self.assertEquals(serializer.errors, {'time': [u'Time has wrong format. Use one of these formats instead: HH -- MM; HHMMSS']})
+
+ def test_from_native(self):
f = serializers.TimeField()
- result = f.from_native('12:34:56.987654')
+ result = f.from_native('12:34:56')
- self.assertEqual(datetime.time(12, 34, 56, 987654), result)
+ self.assertEqual(datetime.time(12, 34, 56), result)
- def test_TimeField_from_native_datetime_time(self):
+ def test_from_native_datetime_time(self):
"""
Make sure from_native() accepts a datetime.time instance.
"""
f = serializers.TimeField()
result = f.from_native(datetime.time(12, 34, 56))
+
self.assertEqual(result, datetime.time(12, 34, 56))
- def test_TimeField_from_native_empty(self):
+ def test_from_native_empty(self):
f = serializers.TimeField()
result = f.from_native('')
+
self.assertEqual(result, None)
- def test_TimeField_from_native_invalid_time(self):
+ def test_from_native_invalid_time(self):
f = serializers.TimeField()
try:
f.from_native('12:69:12')
except validators.ValidationError as e:
- self.assertEqual(e.messages, ["'12:69:12' value has an invalid "
- "format. It must be a valid time "
- "in the HH:MM[:ss[.uuuuuu]] format."])
+ self.assertEqual(e.messages, ["Time has wrong format. Use one of these formats instead: HH:MM:SS; HH:MM"])
else:
- self.fail("ValidationError was not properly raised")
-
- def test_TimeFieldModelSerializer(self):
- serializer = TimeFieldModelSerializer()
- self.assertTrue(isinstance(serializer.fields['clock'], serializers.TimeField))
+ self.fail("ValidationError was not properly raised") \ No newline at end of file