Author: mtredinnick Date: 2008-12-16 00:43:18 -0600 (Tue, 16 Dec 2008) New Revision: 9650
Modified: django/trunk/django/db/backends/mysql/validation.py django/trunk/docs/ref/databases.txt django/trunk/docs/ref/models/fields.txt Log: Fixed #9431 -- Added extra validation for VARCHAR-based fields on MySQL. max_length > 255 and unique=True is not permitted. Based on a patch from adamnelson. Modified: django/trunk/django/db/backends/mysql/validation.py =================================================================== --- django/trunk/django/db/backends/mysql/validation.py 2008-12-16 06:42:19 UTC (rev 9649) +++ django/trunk/django/db/backends/mysql/validation.py 2008-12-16 06:43:18 UTC (rev 9650) @@ -2,12 +2,27 @@ class DatabaseValidation(BaseDatabaseValidation): def validate_field(self, errors, opts, f): - "Prior to MySQL 5.0.3, character fields could not exceed 255 characters" + """ + There are some field length restrictions for MySQL: + + - Prior to version 5.0.3, character fields could not exceed 255 + characters in length. + - No character (varchar) fields can have a length exceeding 255 + characters if they have a unique index on them. + """ from django.db import models from django.db import connection db_version = connection.get_server_version() - if db_version < (5, 0, 3) and isinstance(f, (models.CharField, models.CommaSeparatedIntegerField, models.SlugField)) and f.max_length > 255: - errors.add(opts, - '"%s": %s cannot have a "max_length" greater than 255 when you are using a version of MySQL prior to 5.0.3 (you are using %s).' % - (f.name, f.__class__.__name__, '.'.join([str(n) for n in db_version[:3]]))) - \ No newline at end of file + varchar_fields = (models.CharField, models.CommaSeparatedIntegerField, + models.SlugField) + if isinstance(f, varchar_fields) and f.max_length > 255: + if db_version < (5, 0, 3): + msg = '"%(name)s": %(cls)s cannot have a "max_length" greater than 255 when you are using a version of MySQL prior to 5.0.3 (you are using %(version)s).' + if f.unique == True: + msg = '"%(name)s": %(cls)s cannot have a "max_length" greater than 255 when using "unique=True".' + else: + msg = None + + if msg: + errors.add(opts, msg % {'name': f.name, 'cls': f.__class__.__name__, 'version': '.'.join([str(n) for n in db_version[:3]])}) + Modified: django/trunk/docs/ref/databases.txt =================================================================== --- django/trunk/docs/ref/databases.txt 2008-12-16 06:42:19 UTC (rev 9649) +++ django/trunk/docs/ref/databases.txt 2008-12-16 06:43:18 UTC (rev 9650) @@ -234,9 +234,12 @@ .. _AlterModelOnSyncDB: http://code.djangoproject.com/wiki/AlterModelOnSyncDB -Boolean fields in Django -------------------------- +Notes on specific fields +------------------------ +Boolean fields +~~~~~~~~~~~~~~ + Since MySQL doesn't have a direct ``BOOLEAN`` column type, Django uses a ``TINYINT`` column with values of ``1`` and ``0`` to store values for the :class:`~django.db.models.BooleanField` model field. Refer to the documentation @@ -244,6 +247,19 @@ matter unless you're printing out the field values and are expecting to see ``True`` and ``False.``. +Character fields +~~~~~~~~~~~~~~~~ + +Any fields that are stored with ``VARCHAR`` column types have their +``max_length`` restricted to 255 characters if you are using ``unique=True`` +for the field. This affects :class:`~django.db.models.CharField`, +:class:`~django.db.models.SlugField` and +:class:`~django.db.models.CommaSeparatedIntegerField`. + +Furthermore, if you are using a version of MySQL prior to 5.0.3, all of those +column types have a maximum length restriction of 255 characters, regardless +of whether ``unique=True`` is specified or not. + .. _sqlite-notes: SQLite notes Modified: django/trunk/docs/ref/models/fields.txt =================================================================== --- django/trunk/docs/ref/models/fields.txt 2008-12-16 06:42:19 UTC (rev 9649) +++ django/trunk/docs/ref/models/fields.txt 2008-12-16 06:43:18 UTC (rev 9650) @@ -327,6 +327,13 @@ The maximum length (in characters) of the field. The max_length is enforced at the database level and in Django's validation. +.. note:: + + If you are writing an application that must be portable to multiple + database backends, you should be aware that there are restrictions on + ``max_length`` for some backends. Refer to the :ref:`database backend + notes <ref-databases>` for details. + .. admonition:: MySQL users If you are using this field with MySQLdb 1.2.2 and the ``utf8_bin`` @@ -341,7 +348,8 @@ .. class:: CommaSeparatedIntegerField(max_length=None, [**options]) A field of integers separated by commas. As in :class:`CharField`, the -:attr:`~CharField.max_length` argument is required. +:attr:`~CharField.max_length` argument is required and the note about database +portability mentioned there should be heeded. ``DateField`` ------------- @@ -654,9 +662,10 @@ containing only letters, numbers, underscores or hyphens. They're generally used in URLs. -Like a CharField, you can specify :attr:`~CharField.max_length`. If -:attr:`~CharField.max_length` is not specified, Django will use a default length -of 50. +Like a CharField, you can specify :attr:`~CharField.max_length` (read the note +about database portability and :attr:`~CharField.max_length` in that section, +too). If :attr:`~CharField.max_length` is not specified, Django will use a +default length of 50. Implies setting :attr:`Field.db_index` to ``True``. --~--~---------~--~----~------------~-------~--~----~ You received this message because you are subscribed to the Google Groups "Django updates" group. To post to this group, send email to django-updates@googlegroups.com To unsubscribe from this group, send email to django-updates+unsubscr...@googlegroups.com For more options, visit this group at http://groups.google.com/group/django-updates?hl=en -~----------~----~----~----~------~----~------~--~---