mirror of
				https://github.com/django/django.git
				synced 2025-10-26 23:26:08 +00:00 
			
		
		
		
	* override_settings may now be imported from django.test * removed Approximate from django.test * updated documentation for things importable from django.test Thanks akaariai for the suggestion.
		
			
				
	
	
		
			902 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			902 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| ========================
 | ||
| Django 1.3 release notes
 | ||
| ========================
 | ||
| 
 | ||
| *March 23, 2011*
 | ||
| 
 | ||
| Welcome to Django 1.3!
 | ||
| 
 | ||
| Nearly a year in the making, Django 1.3 includes quite a few `new
 | ||
| features`_ and plenty of bug fixes and improvements to existing
 | ||
| features. These release notes cover the new features in 1.3, as well
 | ||
| as some `backwards-incompatible changes`_ you'll want to be aware of
 | ||
| when upgrading from Django 1.2 or older versions.
 | ||
| 
 | ||
| Overview
 | ||
| ========
 | ||
| 
 | ||
| Django 1.3's focus has mostly been on resolving smaller, long-standing
 | ||
| feature requests, but that hasn't prevented a few fairly significant
 | ||
| new features from landing, including:
 | ||
| 
 | ||
| * A framework for writing `class-based views`_.
 | ||
| 
 | ||
| * Built-in support for `using Python's logging facilities`_.
 | ||
| 
 | ||
| * Contrib support for `easy handling of static files`_.
 | ||
| 
 | ||
| * Django's testing framework now supports (and ships with a copy of)
 | ||
|   `the unittest2 library`_.
 | ||
| 
 | ||
| There's plenty more, of course; see the coverage of `new features`_
 | ||
| below for a full rundown and details.
 | ||
| 
 | ||
| Wherever possible, of course, new features are introduced in a
 | ||
| backwards-compatible manner per :doc:`our API stability policy
 | ||
| </misc/api-stability>` policy. As a result of this policy, Django 1.3
 | ||
| `begins the deprecation process for some features`_.
 | ||
| 
 | ||
| Some changes, unfortunately, are genuinely backwards-incompatible; in
 | ||
| most cases these are due to security issues or bugs which simply
 | ||
| couldn't be fixed any other way. Django 1.3 includes a few of these,
 | ||
| and descriptions of them -- along with the (minor) modifications
 | ||
| you'll need to make to handle them -- are documented in the list of
 | ||
| `backwards-incompatible changes`_ below.
 | ||
| 
 | ||
| .. _new features: `What's new in Django 1.3`_
 | ||
| .. _backwards-incompatible changes: backwards-incompatible-changes-1.3_
 | ||
| .. _using Python's logging facilities: `Logging`_
 | ||
| .. _easy handling of static files: `Extended static files handling`_
 | ||
| .. _the unittest2 library: `unittest2 support`_
 | ||
| .. _begins the deprecation process for some features: `deprecated-features-1.3`_
 | ||
| 
 | ||
| Python compatibility
 | ||
| ====================
 | ||
| 
 | ||
| The release of Django 1.2 was notable for having the first shift in
 | ||
| Django's Python compatibility policy; prior to Django 1.2, Django
 | ||
| supported any 2.x version of Python from 2.3 up. As of Django 1.2, the
 | ||
| minimum requirement was raised to Python 2.4.
 | ||
| 
 | ||
| Django 1.3 continues to support Python 2.4, but will be the final
 | ||
| Django release series to do so; beginning with Django 1.4, the minimum
 | ||
| supported Python version will be 2.5. A document outlining our full
 | ||
| timeline for deprecating Python 2.x and moving to Python 3.x will be
 | ||
| published shortly after the release of Django 1.3.
 | ||
| 
 | ||
| What's new in Django 1.3
 | ||
| ========================
 | ||
| 
 | ||
| Class-based views
 | ||
| ~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Django 1.3 adds a framework that allows you to use a class as a view.
 | ||
| This means you can compose a view out of a collection of methods that
 | ||
| can be subclassed and overridden to provide common views of data without
 | ||
| having to write too much code.
 | ||
| 
 | ||
| Analogs of all the old function-based generic views have been
 | ||
| provided, along with a completely generic view base class that can be
 | ||
| used as the basis for reusable applications that can be easily
 | ||
| extended.
 | ||
| 
 | ||
| See :doc:`the documentation on class-based generic views</topics/class-based-views/index>`
 | ||
| for more details. There is also a document to help you `convert
 | ||
| your function-based generic views to class-based
 | ||
| views <https://docs.djangoproject.com/en/1.4/topics/generic-views-migration/>`_.
 | ||
| 
 | ||
| Logging
 | ||
| ~~~~~~~
 | ||
| 
 | ||
| Django 1.3 adds framework-level support for Python's ``logging``
 | ||
| module.  This means you can now easily configure and control logging
 | ||
| as part of your Django project. A number of logging handlers and
 | ||
| logging calls have been added to Django's own code as well -- most
 | ||
| notably, the error emails sent on a HTTP 500 server error are now
 | ||
| handled as a logging activity. See :doc:`the documentation on Django's
 | ||
| logging interface </topics/logging>` for more details.
 | ||
| 
 | ||
| Extended static files handling
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Django 1.3 ships with a new contrib app --
 | ||
| ``django.contrib.staticfiles`` -- to help developers handle the static
 | ||
| media files (images, CSS, Javascript, etc.) that are needed to render
 | ||
| a complete web page.
 | ||
| 
 | ||
| In previous versions of Django, it was common to place static assets
 | ||
| in :setting:`MEDIA_ROOT` along with user-uploaded files, and serve
 | ||
| them both at :setting:`MEDIA_URL`. Part of the purpose of introducing
 | ||
| the ``staticfiles`` app is to make it easier to keep static files
 | ||
| separate from user-uploaded files. Static assets should now go in
 | ||
| ``static/`` subdirectories of your apps or in other static assets
 | ||
| directories listed in :setting:`STATICFILES_DIRS`, and will be served
 | ||
| at :setting:`STATIC_URL`.
 | ||
| 
 | ||
| See the :doc:`reference documentation of the app </ref/contrib/staticfiles>`
 | ||
| for more details or learn how to :doc:`manage static files
 | ||
| </howto/static-files/index>`.
 | ||
| 
 | ||
| unittest2 support
 | ||
| ~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Python 2.7 introduced some major changes to the ``unittest`` library,
 | ||
| adding some extremely useful features. To ensure that every Django
 | ||
| project can benefit from these new features, Django ships with a copy
 | ||
| of unittest2_, a copy of the Python 2.7 unittest library, backported
 | ||
| for Python 2.4 compatibility.
 | ||
| 
 | ||
| To access this library, Django provides the ``django.utils.unittest``
 | ||
| module alias. If you are using Python 2.7, or you have installed
 | ||
| ``unittest2`` locally, Django will map the alias to the installed
 | ||
| version of the unittest library. Otherwise, Django will use its own
 | ||
| bundled version of unittest2.
 | ||
| 
 | ||
| To take advantage of this alias, simply use::
 | ||
| 
 | ||
|     from django.utils import unittest
 | ||
| 
 | ||
| wherever you would have historically used::
 | ||
| 
 | ||
|     import unittest
 | ||
| 
 | ||
| If you want to continue to use the base unittest library, you can --
 | ||
| you just won't get any of the nice new unittest2 features.
 | ||
| 
 | ||
| .. _unittest2: http://pypi.python.org/pypi/unittest2
 | ||
| 
 | ||
| Transaction context managers
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Users of Python 2.5 and above may now use transaction management functions as
 | ||
| `context managers`_. For example::
 | ||
| 
 | ||
|     with transaction.autocommit():
 | ||
|         # ...
 | ||
| 
 | ||
| .. _context managers: http://docs.python.org/glossary.html#term-context-manager
 | ||
| 
 | ||
| Configurable delete-cascade
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| :class:`~django.db.models.ForeignKey` and
 | ||
| :class:`~django.db.models.OneToOneField` now accept an
 | ||
| :attr:`~django.db.models.ForeignKey.on_delete` argument to customize behavior
 | ||
| when the referenced object is deleted. Previously, deletes were always
 | ||
| cascaded; available alternatives now include set null, set default, set to any
 | ||
| value, protect, or do nothing.
 | ||
| 
 | ||
| For more information, see the :attr:`~django.db.models.ForeignKey.on_delete`
 | ||
| documentation.
 | ||
| 
 | ||
| Contextual markers and comments for translatable strings
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| For translation strings with ambiguous meaning, you can now
 | ||
| use the ``pgettext`` function to specify the context of the string.
 | ||
| 
 | ||
| And if you just want to add some information for translators, you
 | ||
| can also add special translator comments in the source.
 | ||
| 
 | ||
| For more information, see :ref:`contextual-markers` and
 | ||
| :ref:`translator-comments`.
 | ||
| 
 | ||
| Improvements to built-in template tags
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| A number of improvements have been made to Django's built-in template tags:
 | ||
| 
 | ||
| * The :ttag:`include` tag now accepts a ``with`` option, allowing
 | ||
|   you to specify context variables to the included template
 | ||
| 
 | ||
| * The :ttag:`include` tag now accepts an ``only`` option, allowing
 | ||
|   you to exclude the current context from the included context
 | ||
| 
 | ||
| * The :ttag:`with` tag now allows you to define multiple context
 | ||
|   variables in a single :ttag:`with` block.
 | ||
| 
 | ||
| * The :ttag:`load` tag now accepts a ``from`` argument, allowing
 | ||
|   you to load a single tag or filter from a library.
 | ||
| 
 | ||
| TemplateResponse
 | ||
| ~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| It can sometimes be beneficial to allow decorators or middleware to
 | ||
| modify a response *after* it has been constructed by the view. For
 | ||
| example, you may want to change the template that is used, or put
 | ||
| additional data into the context.
 | ||
| 
 | ||
| However, you can't (easily) modify the content of a basic
 | ||
| :class:`~django.http.HttpResponse` after it has been constructed. To
 | ||
| overcome this limitation, Django 1.3 adds a new
 | ||
| :class:`~django.template.response.TemplateResponse` class. Unlike basic
 | ||
| :class:`~django.http.HttpResponse` objects,
 | ||
| :class:`~django.template.response.TemplateResponse` objects retain the details
 | ||
| of the template and context that was provided by the view to compute
 | ||
| the response. The final output of the response is not computed until
 | ||
| it is needed, later in the response process.
 | ||
| 
 | ||
| For more details, see the :doc:`documentation </ref/template-response>`
 | ||
| on the :class:`~django.template.response.TemplateResponse` class.
 | ||
| 
 | ||
| Caching changes
 | ||
| ~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Django 1.3 sees the introduction of several improvements to the
 | ||
| Django's caching infrastructure.
 | ||
| 
 | ||
| Firstly, Django now supports multiple named caches. In the same way
 | ||
| that Django 1.2 introduced support for multiple database connections,
 | ||
| Django 1.3 allows you to use the new :setting:`CACHES` setting to
 | ||
| define multiple named cache connections.
 | ||
| 
 | ||
| Secondly, :ref:`versioning <cache_versioning>`, :ref:`site-wide
 | ||
| prefixing <cache_key_prefixing>` and :ref:`transformation
 | ||
| <cache_key_transformation>` have been added to the cache API.
 | ||
| 
 | ||
| Thirdly, :ref:`cache key creation <using-vary-headers>` has been
 | ||
| updated to take the request query string into account on ``GET``
 | ||
| requests.
 | ||
| 
 | ||
| Finally, support for pylibmc_ has been added to the memcached cache
 | ||
| backend.
 | ||
| 
 | ||
| For more details, see the :doc:`documentation on
 | ||
| caching in Django</topics/cache>`.
 | ||
| 
 | ||
| .. _pylibmc: http://sendapatch.se/projects/pylibmc/
 | ||
| 
 | ||
| Permissions for inactive users
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| If you provide a custom auth backend with ``supports_inactive_user``
 | ||
| set to ``True``, an inactive ``User`` instance will check the backend
 | ||
| for permissions.  This is useful for further centralizing the
 | ||
| permission handling. See the :doc:`authentication docs </topics/auth/index>`
 | ||
| for more details.
 | ||
| 
 | ||
| GeoDjango
 | ||
| ~~~~~~~~~
 | ||
| 
 | ||
| The GeoDjango test suite is now included when
 | ||
| :ref:`running the Django test suite <running-unit-tests>` with ``runtests.py``
 | ||
| when using :ref:`spatial database backends <spatial-backends>`.
 | ||
| 
 | ||
| :setting:`MEDIA_URL` and :setting:`STATIC_URL` must end in a slash
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Previously, the :setting:`MEDIA_URL` setting only required a trailing slash if
 | ||
| it contained a suffix beyond the domain name.
 | ||
| 
 | ||
| A trailing slash is now *required* for :setting:`MEDIA_URL` and the new
 | ||
| :setting:`STATIC_URL` setting as long as it is not blank. This ensures there is
 | ||
| a consistent way to combine paths in templates.
 | ||
| 
 | ||
| Project settings which provide either of both settings without a trailing
 | ||
| slash will now raise a ``PendingDeprecationWarning``.
 | ||
| 
 | ||
| In Django 1.4 this same condition will raise ``DeprecationWarning``,
 | ||
| and in Django 1.5 will raise an ``ImproperlyConfigured`` exception.
 | ||
| 
 | ||
| Everything else
 | ||
| ~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Django :doc:`1.1 <1.1>` and :doc:`1.2 <1.2>` added
 | ||
| lots of big ticket items to Django, like multiple-database support,
 | ||
| model validation, and a session-based messages framework. However,
 | ||
| this focus on big features came at the cost of lots of smaller
 | ||
| features.
 | ||
| 
 | ||
| To compensate for this, the focus of the Django 1.3 development
 | ||
| process has been on adding lots of smaller, long standing feature
 | ||
| requests. These include:
 | ||
| 
 | ||
| * Improved tools for accessing and manipulating the current
 | ||
|   :class:`~django.contrib.sites.models.Site` object in
 | ||
|   :doc:`the sites framework </ref/contrib/sites>`.
 | ||
| 
 | ||
| * A :class:`~django.test.RequestFactory` for mocking requests
 | ||
|   in tests.
 | ||
| 
 | ||
| * A new test assertion --
 | ||
|   :meth:`~django.test.TransactionTestCase.assertNumQueries` -- making it
 | ||
|   easier to test the database activity associated with a view.
 | ||
| 
 | ||
| * Support for lookups spanning relations in admin's
 | ||
|   :attr:`~django.contrib.admin.ModelAdmin.list_filter`.
 | ||
| 
 | ||
| * Support for HTTPOnly_ cookies.
 | ||
| 
 | ||
| * :meth:`~django.core.mail.mail_admins()` and
 | ||
|   :meth:`~django.core.mail.mail_managers()` now support easily attaching
 | ||
|   HTML content to messages.
 | ||
| 
 | ||
| * :class:`~django.core.mail.EmailMessage` now supports CC's.
 | ||
| 
 | ||
| * Error emails now include more of the detail and formatting of the
 | ||
|   debug server error page.
 | ||
| 
 | ||
| * :meth:`~django.template.Library.simple_tag` now accepts a
 | ||
|   ``takes_context`` argument, making it easier to write simple
 | ||
|   template tags that require access to template context.
 | ||
| 
 | ||
| * A new :meth:`~django.shortcuts.render()` shortcut -- an alternative
 | ||
|   to :meth:`~django.shortcuts.render_to_response()` providing a
 | ||
|   :class:`~django.template.RequestContext` by default.
 | ||
| 
 | ||
| * Support for combining :class:`F expressions <django.db.models.F>`
 | ||
|   with timedelta values when retrieving or updating database values.
 | ||
| 
 | ||
| .. _HTTPOnly: https://www.owasp.org/index.php/HTTPOnly
 | ||
| 
 | ||
| .. _backwards-incompatible-changes-1.3:
 | ||
| 
 | ||
| Backwards-incompatible changes in 1.3
 | ||
| =====================================
 | ||
| 
 | ||
| CSRF validation now applies to AJAX requests
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Prior to Django 1.2.5, Django's CSRF-prevention system exempted AJAX
 | ||
| requests from CSRF verification; due to `security issues`_ reported to
 | ||
| us, however, *all* requests are now subjected to CSRF
 | ||
| verification. Consult :doc:`the Django CSRF documentation
 | ||
| </ref/contrib/csrf>` for details on how to handle CSRF verification in
 | ||
| AJAX requests.
 | ||
| 
 | ||
| .. _security issues: https://www.djangoproject.com/weblog/2011/feb/08/security/
 | ||
| 
 | ||
| Restricted filters in admin interface
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Prior to Django 1.2.5, the Django administrative interface allowed
 | ||
| filtering on any model field or relation -- not just those specified
 | ||
| in ``list_filter`` -- via query string manipulation. Due to security
 | ||
| issues reported to us, however, query string lookup arguments in the
 | ||
| admin must be for fields or relations specified in ``list_filter`` or
 | ||
| ``date_hierarchy``.
 | ||
| 
 | ||
| Deleting a model doesn't delete associated files
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| In earlier Django versions, when a model instance containing a
 | ||
| :class:`~django.db.models.FileField` was deleted,
 | ||
| :class:`~django.db.models.FileField` took it upon itself to also delete the
 | ||
| file from the backend storage. This opened the door to several data-loss
 | ||
| scenarios, including rolled-back transactions and fields on different models
 | ||
| referencing the same file. In Django 1.3, when a model is deleted the
 | ||
| :class:`~django.db.models.FileField`’s ``delete()`` method won't be called. If
 | ||
| you need cleanup of orphaned files, you'll need to handle it yourself (for
 | ||
| instance, with a custom management command that can be run manually or
 | ||
| scheduled to run periodically via e.g. cron).
 | ||
| 
 | ||
| PasswordInput default rendering behavior
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| The :class:`~django.forms.PasswordInput` form widget, intended for use
 | ||
| with form fields which represent passwords, accepts a boolean keyword
 | ||
| argument ``render_value`` indicating whether to send its data back to
 | ||
| the browser when displaying a submitted form with errors. Prior to
 | ||
| Django 1.3, this argument defaulted to ``True``, meaning that the
 | ||
| submitted password would be sent back to the browser as part of the
 | ||
| form. Developers who wished to add a bit of additional security by
 | ||
| excluding that value from the redisplayed form could instantiate a
 | ||
| :class:`~django.forms.PasswordInput` passing ``render_value=False`` .
 | ||
| 
 | ||
| Due to the sensitive nature of passwords, however, Django 1.3 takes
 | ||
| this step automatically; the default value of ``render_value`` is now
 | ||
| ``False``, and developers who want the password value returned to the
 | ||
| browser on a submission with errors (the previous behavior) must now
 | ||
| explicitly indicate this. For example::
 | ||
| 
 | ||
|     class LoginForm(forms.Form):
 | ||
|         username = forms.CharField(max_length=100)
 | ||
|         password = forms.CharField(widget=forms.PasswordInput(render_value=True))
 | ||
| 
 | ||
| Clearable default widget for FileField
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Django 1.3 now includes a :class:`~django.forms.ClearableFileInput` form widget
 | ||
| in addition to :class:`~django.forms.FileInput`. ``ClearableFileInput`` renders
 | ||
| with a checkbox to clear the field's value (if the field has a value and is not
 | ||
| required); ``FileInput`` provided no means for clearing an existing file from
 | ||
| a ``FileField``.
 | ||
| 
 | ||
| ``ClearableFileInput`` is now the default widget for a ``FileField``, so
 | ||
| existing forms including ``FileField`` without assigning a custom widget will
 | ||
| need to account for the possible extra checkbox in the rendered form output.
 | ||
| 
 | ||
| To return to the previous rendering (without the ability to clear the
 | ||
| ``FileField``), use the ``FileInput`` widget in place of
 | ||
| ``ClearableFileInput``. For instance, in a ``ModelForm`` for a hypothetical
 | ||
| ``Document`` model with a ``FileField`` named ``document``::
 | ||
| 
 | ||
|     from django import forms
 | ||
|     from myapp.models import Document
 | ||
| 
 | ||
|     class DocumentForm(forms.ModelForm):
 | ||
|         class Meta:
 | ||
|             model = Document
 | ||
|             widgets = {'document': forms.FileInput}
 | ||
| 
 | ||
| New index on database session table
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Prior to Django 1.3, the database table used by the database backend
 | ||
| for the :doc:`sessions </topics/http/sessions>` app had no index on
 | ||
| the ``expire_date`` column. As a result, date-based queries on the
 | ||
| session table -- such as the query that is needed to purge old
 | ||
| sessions -- would be very slow if there were lots of sessions.
 | ||
| 
 | ||
| If you have an existing project that is using the database session
 | ||
| backend, you don't have to do anything to accommodate this change.
 | ||
| However, you may get a significant performance boost if you manually
 | ||
| add the new index to the session table. The SQL that will add the
 | ||
| index can be found by running the :djadmin:`sqlindexes` admin
 | ||
| command::
 | ||
| 
 | ||
|     python manage.py sqlindexes sessions
 | ||
| 
 | ||
| No more naughty words
 | ||
| ~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Django has historically provided (and enforced) a list of profanities.
 | ||
| The :doc:`comments app </ref/contrib/comments/index>` has enforced this
 | ||
| list of profanities, preventing people from submitting comments that
 | ||
| contained one of those profanities.
 | ||
| 
 | ||
| Unfortunately, the technique used to implement this profanities list
 | ||
| was woefully naive, and prone to the `Scunthorpe problem`_. Improving
 | ||
| the built-in filter to fix this problem would require significant
 | ||
| effort, and since natural language processing isn't the normal domain
 | ||
| of a web framework, we have "fixed" the problem by making the list of
 | ||
| prohibited words an empty list.
 | ||
| 
 | ||
| If you want to restore the old behavior, simply put a
 | ||
| :setting:`PROFANITIES_LIST` setting in your settings file that includes the
 | ||
| words that you want to prohibit (see the `commit that implemented this
 | ||
| change`_ if you want to see the list of words that was historically
 | ||
| prohibited). However, if avoiding profanities is important to you, you
 | ||
| would be well advised to seek out a better, less naive approach to the
 | ||
| problem.
 | ||
| 
 | ||
| .. _Scunthorpe problem: http://en.wikipedia.org/wiki/Scunthorpe_problem
 | ||
| .. _commit that implemented this change: https://code.djangoproject.com/changeset/13996
 | ||
| 
 | ||
| Localflavor changes
 | ||
| ~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Django 1.3 introduces the following backwards-incompatible changes to
 | ||
| local flavors:
 | ||
| 
 | ||
| * Canada (ca) -- The province "Newfoundland and Labrador" has had its
 | ||
|   province code updated to "NL", rather than the older "NF". In
 | ||
|   addition, the Yukon Territory has had its province code corrected to
 | ||
|   "YT", instead of "YK".
 | ||
| 
 | ||
| * Indonesia (id) -- The province "Nanggroe Aceh Darussalam (NAD)" has
 | ||
|   been removed from the province list in favor of the new official
 | ||
|   designation "Aceh (ACE)".
 | ||
| 
 | ||
| * United States of America (us) -- The list of "states" used by
 | ||
|   ``USStateField`` has expanded to include Armed Forces postal
 | ||
|   codes. This is backwards-incompatible if you were relying on
 | ||
|   ``USStateField`` not including them.
 | ||
| 
 | ||
| FormSet updates
 | ||
| ~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| In Django 1.3 ``FormSet`` creation behavior is modified slightly. Historically
 | ||
| the class didn't make a distinction between not being passed data and being
 | ||
| passed empty dictionary. This was inconsistent with behavior in other parts of
 | ||
| the framework. Starting with 1.3 if you pass in empty dictionary the
 | ||
| ``FormSet`` will raise a ``ValidationError``.
 | ||
| 
 | ||
| For example with a ``FormSet``::
 | ||
| 
 | ||
|     >>> class ArticleForm(Form):
 | ||
|     ...     title = CharField()
 | ||
|     ...     pub_date = DateField()
 | ||
|     >>> ArticleFormSet = formset_factory(ArticleForm)
 | ||
| 
 | ||
| the following code will raise a ``ValidationError``::
 | ||
| 
 | ||
|     >>> ArticleFormSet({})
 | ||
|     Traceback (most recent call last):
 | ||
|     ...
 | ||
|     ValidationError: [u'ManagementForm data is missing or has been tampered with']
 | ||
| 
 | ||
| if you need to instantiate an empty ``FormSet``, don't pass in the data or use
 | ||
| ``None``::
 | ||
| 
 | ||
|     >>> formset = ArticleFormSet()
 | ||
|     >>> formset = ArticleFormSet(data=None)
 | ||
| 
 | ||
| Callables in templates
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Previously, a callable in a template would only be called automatically as part
 | ||
| of the variable resolution process if it was retrieved via attribute
 | ||
| lookup. This was an inconsistency that could result in confusing and unhelpful
 | ||
| behavior::
 | ||
| 
 | ||
|     >>> Template("{{ user.get_full_name }}").render(Context({'user': user}))
 | ||
|     u'Joe Bloggs'
 | ||
|     >>> Template("{{ full_name }}").render(Context({'full_name': user.get_full_name}))
 | ||
|     u'<bound method User.get_full_name of <...
 | ||
| 
 | ||
| This has been resolved in Django 1.3 - the result in both cases will be ``u'Joe
 | ||
| Bloggs'``. Although the previous behavior was not useful for a template language
 | ||
| designed for web designers, and was never deliberately supported, it is possible
 | ||
| that some templates may be broken by this change.
 | ||
| 
 | ||
| Use of custom SQL to load initial data in tests
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Django provides a custom SQL hooks as a way to inject hand-crafted SQL
 | ||
| into the database synchronization process. One of the possible uses
 | ||
| for this custom SQL is to insert data into your database. If your
 | ||
| custom SQL contains ``INSERT`` statements, those insertions will be
 | ||
| performed every time your database is synchronized. This includes the
 | ||
| synchronization of any test databases that are created when you run a
 | ||
| test suite.
 | ||
| 
 | ||
| However, in the process of testing the Django 1.3, it was discovered
 | ||
| that this feature has never completely worked as advertised. When
 | ||
| using database backends that don't support transactions, or when using
 | ||
| a TransactionTestCase, data that has been inserted using custom SQL
 | ||
| will not be visible during the testing process.
 | ||
| 
 | ||
| Unfortunately, there was no way to rectify this problem without
 | ||
| introducing a backwards incompatibility. Rather than leave
 | ||
| SQL-inserted initial data in an uncertain state, Django now enforces
 | ||
| the policy that data inserted by custom SQL will *not* be visible
 | ||
| during testing.
 | ||
| 
 | ||
| This change only affects the testing process. You can still use custom
 | ||
| SQL to load data into your production database as part of the syncdb
 | ||
| process. If you require data to exist during test conditions, you
 | ||
| should either insert it using :ref:`test fixtures
 | ||
| <topics-testing-fixtures>`, or using the ``setUp()`` method of your
 | ||
| test case.
 | ||
| 
 | ||
| Changed priority of translation loading
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Work has been done to simplify, rationalize and properly document the algorithm
 | ||
| used by Django at runtime to build translations from the different translations
 | ||
| found on disk, namely:
 | ||
| 
 | ||
| For translatable literals found in Python code and templates (``'django'``
 | ||
| gettext domain):
 | ||
| 
 | ||
| * Priorities of translations included with applications listed in the
 | ||
|   :setting:`INSTALLED_APPS` setting were changed. To provide a behavior
 | ||
|   consistent with other parts of Django that also use such setting (templates,
 | ||
|   etc.) now, when building the translation that will be made available, the
 | ||
|   apps listed first have higher precedence than the ones listed later.
 | ||
| 
 | ||
| * Now it is possible to override the translations shipped with applications by
 | ||
|   using the :setting:`LOCALE_PATHS` setting whose translations have now higher
 | ||
|   precedence than the translations of :setting:`INSTALLED_APPS` applications.
 | ||
|   The relative priority among the values listed in this setting has also been
 | ||
|   modified so the paths listed first have higher precedence than the
 | ||
|   ones listed later.
 | ||
| 
 | ||
| * The ``locale`` subdirectory of the directory containing the settings, that
 | ||
|   usually coincides with and is known as the *project directory* is being
 | ||
|   deprecated in this release as a source of translations. (the precedence of
 | ||
|   these translations is intermediate between applications and :setting:`LOCALE_PATHS`
 | ||
|   translations). See the `corresponding deprecated features section`_
 | ||
|   of this document.
 | ||
| 
 | ||
| For translatable literals found in Javascript code (``'djangojs'`` gettext
 | ||
| domain):
 | ||
| 
 | ||
| * Similarly to the ``'django'`` domain translations: Overriding of
 | ||
|   translations shipped with applications by using the :setting:`LOCALE_PATHS`
 | ||
|   setting is now possible for this domain too. These translations have higher
 | ||
|   precedence than the translations of Python packages passed to the
 | ||
|   :ref:`javascript_catalog view <javascript_catalog-view>`.  Paths listed first
 | ||
|   have higher precedence than the ones listed later.
 | ||
| 
 | ||
| * Translations under the ``locale`` subdirectory of the *project directory*
 | ||
|   have never been taken in account for JavaScript translations and remain in
 | ||
|   the same situation considering the deprecation of such location.
 | ||
| 
 | ||
| .. _corresponding deprecated features section: loading_of_project_level_translations_
 | ||
| 
 | ||
| Transaction management
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| When using managed transactions -- that is, anything but the default
 | ||
| autocommit mode -- it is important when a transaction is marked as
 | ||
| "dirty". Dirty transactions are committed by the
 | ||
| :func:`~django.db.transaction.commit_on_success` decorator or the
 | ||
| :class:`~django.middleware.transaction.TransactionMiddleware`, and
 | ||
| :func:`~django.db.transaction.commit_manually` forces them to be
 | ||
| closed explicitly; clean transactions "get a pass", which means they
 | ||
| are usually rolled back at the end of a request when the connection is
 | ||
| closed.
 | ||
| 
 | ||
| Until Django 1.3, transactions were only marked dirty when Django was
 | ||
| aware of a modifying operation performed in them; that is, either some
 | ||
| model was saved, some bulk update or delete was performed, or the user
 | ||
| explicitly called ``transaction.set_dirty()``. In Django 1.3, a
 | ||
| transaction is marked dirty when *any* database operation is
 | ||
| performed.
 | ||
| 
 | ||
| As a result of this change, you no longer need to set a transaction
 | ||
| dirty explicitly when you execute raw SQL or use a data-modifying
 | ||
| ``SELECT``. However, you *do* need to explicitly close any read-only
 | ||
| transactions that are being managed using
 | ||
| :func:`~django.db.transaction.commit_manually`. For example::
 | ||
| 
 | ||
|       @transaction.commit_manually
 | ||
|       def my_view(request, name):
 | ||
|           obj = get_object_or_404(MyObject, name__iexact=name)
 | ||
|           return render_to_response('template', {'object':obj})
 | ||
| 
 | ||
| Prior to Django 1.3, this would work without error. However, under
 | ||
| Django 1.3, this will raise a
 | ||
| :class:`~django.db.transaction.TransactionManagementError` because
 | ||
| the read operation that retrieves the ``MyObject`` instance leaves the
 | ||
| transaction in a dirty state.
 | ||
| 
 | ||
| No password reset for inactive users
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Prior to Django 1.3, inactive users were able to request a password reset email
 | ||
| and reset their password. In Django 1.3 inactive users will receive the same
 | ||
| message as a nonexistent account.
 | ||
| 
 | ||
| Password reset view now accepts ``from_email``
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| The :func:`django.contrib.auth.views.password_reset` view now accepts a
 | ||
| ``from_email`` parameter, which is passed to the ``password_reset_form``’s
 | ||
| ``save()`` method as a keyword argument. If you are using this view with a
 | ||
| custom password reset form, then you will need to ensure your form's ``save()``
 | ||
| method accepts this keyword argument.
 | ||
| 
 | ||
| .. _deprecated-features-1.3:
 | ||
| 
 | ||
| Features deprecated in 1.3
 | ||
| ==========================
 | ||
| 
 | ||
| Django 1.3 deprecates some features from earlier releases.
 | ||
| These features are still supported, but will be gradually phased out
 | ||
| over the next few release cycles.
 | ||
| 
 | ||
| Code taking advantage of any of the features below will raise a
 | ||
| ``PendingDeprecationWarning`` in Django 1.3. This warning will be
 | ||
| silent by default, but may be turned on using Python's :mod:`warnings`
 | ||
| module, or by running Python with a ``-Wd`` or ``-Wall`` flag.
 | ||
| 
 | ||
| In Django 1.4, these warnings will become a ``DeprecationWarning``,
 | ||
| which is *not* silent. In Django 1.5 support for these features will
 | ||
| be removed entirely.
 | ||
| 
 | ||
| .. seealso::
 | ||
| 
 | ||
|     For more details, see the documentation :doc:`Django's release process
 | ||
|     </internals/release-process>` and our :doc:`deprecation timeline
 | ||
|     </internals/deprecation>`.
 | ||
| 
 | ||
| ``mod_python`` support
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| The ``mod_python`` library has not had a release since 2007 or a commit since
 | ||
| 2008. The Apache Foundation board voted to remove ``mod_python`` from the set
 | ||
| of active projects in its version control repositories, and its lead developer
 | ||
| has shifted all of his efforts toward the lighter, slimmer, more stable, and
 | ||
| more flexible ``mod_wsgi`` backend.
 | ||
| 
 | ||
| If you are currently using the ``mod_python`` request handler, you
 | ||
| should redeploy your Django projects using another request handler.
 | ||
| :doc:`mod_wsgi </howto/deployment/wsgi/modwsgi>` is the request handler
 | ||
| recommended by the Django project, but :doc:`FastCGI
 | ||
| </howto/deployment/fastcgi>` is also supported. Support for
 | ||
| ``mod_python`` deployment will be removed in Django 1.5.
 | ||
| 
 | ||
| Function-based generic views
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| As a result of the introduction of class-based generic views, the
 | ||
| function-based generic views provided by Django have been deprecated.
 | ||
| The following modules and the views they contain have been deprecated:
 | ||
| 
 | ||
| * ``django.views.generic.create_update``
 | ||
| * ``django.views.generic.date_based``
 | ||
| * ``django.views.generic.list_detail``
 | ||
| * ``django.views.generic.simple``
 | ||
| 
 | ||
| Test client response ``template`` attribute
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Django's :ref:`test client <test-client>` returns
 | ||
| :class:`~django.test.Response` objects annotated with extra testing
 | ||
| information. In Django versions prior to 1.3, this included a ``template``
 | ||
| attribute containing information about templates rendered in generating the
 | ||
| response: either None, a single :class:`~django.template.Template` object, or a
 | ||
| list of :class:`~django.template.Template` objects. This inconsistency in
 | ||
| return values (sometimes a list, sometimes not) made the attribute difficult
 | ||
| to work with.
 | ||
| 
 | ||
| In Django 1.3 the ``template`` attribute is deprecated in favor of a new
 | ||
| :attr:`~django.test.Response.templates` attribute, which is always a
 | ||
| list, even if it has only a single element or no elements.
 | ||
| 
 | ||
| ``DjangoTestRunner``
 | ||
| ~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| As a result of the introduction of support for unittest2, the features
 | ||
| of ``django.test.simple.DjangoTestRunner`` (including fail-fast
 | ||
| and Ctrl-C test termination) have been made redundant. In view of this
 | ||
| redundancy, ``DjangoTestRunner`` has been turned into an empty placeholder
 | ||
| class, and will be removed entirely in Django 1.5.
 | ||
| 
 | ||
| Changes to :ttag:`url` and :ttag:`ssi`
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Most template tags will allow you to pass in either constants or
 | ||
| variables as arguments -- for example::
 | ||
| 
 | ||
|     {% extends "base.html" %}
 | ||
| 
 | ||
| allows you to specify a base template as a constant, but if you have a
 | ||
| context variable ``templ`` that contains the value ``base.html``::
 | ||
| 
 | ||
|     {% extends templ %}
 | ||
| 
 | ||
| is also legal.
 | ||
| 
 | ||
| However, due to an accident of history, the :ttag:`url` and
 | ||
| :ttag:`ssi` are different. These tags use the second, quoteless
 | ||
| syntax, but interpret the argument as a constant. This means it isn't
 | ||
| possible to use a context variable as the target of a :ttag:`url` and
 | ||
| :ttag:`ssi` tag.
 | ||
| 
 | ||
| Django 1.3 marks the start of the process to correct this historical
 | ||
| accident. Django 1.3 adds a new template library -- ``future`` -- that
 | ||
| provides alternate implementations of the :ttag:`url` and :ttag:`ssi`
 | ||
| template tags. This ``future`` library implement behavior that makes
 | ||
| the handling of the first argument consistent with the handling of all
 | ||
| other variables. So, an existing template that contains::
 | ||
| 
 | ||
|     {% url sample %}
 | ||
| 
 | ||
| should be replaced with::
 | ||
| 
 | ||
|     {% load url from future %}
 | ||
|     {% url 'sample' %}
 | ||
| 
 | ||
| The tags implementing the old behavior have been deprecated, and in
 | ||
| Django 1.5, the old behavior will be replaced with the new behavior.
 | ||
| To ensure compatibility with future versions of Django, existing
 | ||
| templates should be modified to use the new ``future`` libraries and
 | ||
| syntax.
 | ||
| 
 | ||
| Changes to the login methods of the admin
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| In previous version the admin app defined login methods in multiple locations
 | ||
| and ignored the almost identical implementation in the already used auth app.
 | ||
| A side effect of this duplication was the missing adoption of the changes made
 | ||
| in r12634_ to support a broader set of characters for usernames.
 | ||
| 
 | ||
| This release refactors the admin's login mechanism to use a subclass of the
 | ||
| :class:`~django.contrib.auth.forms.AuthenticationForm` instead of a manual
 | ||
| form validation. The previously undocumented method
 | ||
| ``'django.contrib.admin.sites.AdminSite.display_login_form'`` has been removed
 | ||
| in favor of a new :attr:`~django.contrib.admin.AdminSite.login_form`
 | ||
| attribute.
 | ||
| 
 | ||
| .. _r12634: https://code.djangoproject.com/changeset/12634
 | ||
| 
 | ||
| ``reset`` and ``sqlreset`` management commands
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Those commands have been deprecated. The ``flush`` and ``sqlflush`` commands
 | ||
| can be used to delete everything. You can also use ALTER TABLE or DROP TABLE
 | ||
| statements manually.
 | ||
| 
 | ||
| 
 | ||
| GeoDjango
 | ||
| ~~~~~~~~~
 | ||
| 
 | ||
| * The function-based :setting:`TEST_RUNNER` previously used to execute
 | ||
|   the GeoDjango test suite, ``django.contrib.gis.tests.run_gis_tests``, was
 | ||
|   deprecated for the class-based runner,
 | ||
|   ``django.contrib.gis.tests.GeoDjangoTestSuiteRunner``.
 | ||
| 
 | ||
| * Previously, calling
 | ||
|   :meth:`~django.contrib.gis.geos.GEOSGeometry.transform` would
 | ||
|   silently do nothing when GDAL wasn't available.  Now, a
 | ||
|   :class:`~django.contrib.gis.geos.GEOSException` is properly raised
 | ||
|   to indicate possible faulty application code.  A warning is now
 | ||
|   raised if :meth:`~django.contrib.gis.geos.GEOSGeometry.transform` is
 | ||
|   called when the SRID of the geometry is less than 0 or ``None``.
 | ||
| 
 | ||
| ``CZBirthNumberField.clean``
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Previously this field's ``clean()`` method accepted a second, gender, argument
 | ||
| which allowed stronger validation checks to be made, however since this
 | ||
| argument could never actually be passed from the Django form machinery it is
 | ||
| now pending deprecation.
 | ||
| 
 | ||
| ``CompatCookie``
 | ||
| ~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| Previously, ``django.http`` exposed an undocumented ``CompatCookie`` class,
 | ||
| which was a bugfix wrapper around the standard library ``SimpleCookie``. As the
 | ||
| fixes are moving upstream, this is now deprecated - you should use ``from
 | ||
| django.http import SimpleCookie`` instead.
 | ||
| 
 | ||
| .. _loading_of_project_level_translations:
 | ||
| 
 | ||
| Loading of *project-level* translations
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| This release of Django starts the deprecation process for inclusion of
 | ||
| translations located under the so-called *project path* in the translation
 | ||
| building process performed at runtime. The :setting:`LOCALE_PATHS` setting can
 | ||
| be used for the same task by adding the filesystem path to a ``locale``
 | ||
| directory containing project-level translations to the value of that setting.
 | ||
| 
 | ||
| Rationale for this decision:
 | ||
| 
 | ||
| * The *project path* has always been a loosely defined concept
 | ||
|   (actually, the directory used for locating project-level
 | ||
|   translations is the directory containing the settings module) and
 | ||
|   there has been a shift in other parts of the framework to stop using
 | ||
|   it as a reference for location of assets at runtime.
 | ||
| 
 | ||
| * Detection of the ``locale`` subdirectory tends to fail when the
 | ||
|   deployment scenario is more complex than the basic one. e.g. it
 | ||
|   fails when the settings module is a directory (ticket #10765).
 | ||
| 
 | ||
| * There are potential strange development- and deployment-time
 | ||
|   problems like the fact that the ``project_dir/locale/`` subdir can
 | ||
|   generate spurious error messages when the project directory is added
 | ||
|   to the Python path (``manage.py runserver`` does this) and then it
 | ||
|   clashes with the equally named standard library module, this is a
 | ||
|   typical warning message::
 | ||
| 
 | ||
|      /usr/lib/python2.6/gettext.py:49: ImportWarning: Not importing directory '/path/to/project/locale': missing __init__.py.
 | ||
|      import locale, copy, os, re, struct, sys
 | ||
| 
 | ||
| * This location wasn't included in the translation building process
 | ||
|   for JavaScript literals. This deprecation removes such
 | ||
|   inconsistency.
 | ||
| 
 | ||
| ``PermWrapper`` moved to ``django.contrib.auth.context_processors``
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| In Django 1.2, we began the process of changing the location of the
 | ||
| ``auth`` context processor from ``django.core.context_processors`` to
 | ||
| ``django.contrib.auth.context_processors``. However, the
 | ||
| ``PermWrapper`` support class was mistakenly omitted from that
 | ||
| migration. In Django 1.3, the ``PermWrapper`` class has also been
 | ||
| moved to ``django.contrib.auth.context_processors``, along with the
 | ||
| ``PermLookupDict`` support class. The new classes are functionally
 | ||
| identical to their old versions; only the module location has changed.
 | ||
| 
 | ||
| Removal of ``XMLField``
 | ||
| ~~~~~~~~~~~~~~~~~~~~~~~
 | ||
| 
 | ||
| When Django was first released, Django included an ``XMLField`` that performed
 | ||
| automatic XML validation for any field input. However, this validation function
 | ||
| hasn't been performed since the introduction of ``newforms``, prior to the 1.0
 | ||
| release. As a result, ``XMLField`` as currently implemented is functionally
 | ||
| indistinguishable from a simple :class:`~django.db.models.TextField`.
 | ||
| 
 | ||
| For this reason, Django 1.3 has fast-tracked the deprecation of
 | ||
| ``XMLField`` -- instead of a two-release deprecation, ``XMLField``
 | ||
| will be removed entirely in Django 1.4.
 | ||
| 
 | ||
| It's easy to update your code to accommodate this change -- just
 | ||
| replace all uses of ``XMLField`` with ``TextField``, and remove the
 | ||
| ``schema_path`` keyword argument (if it is specified).
 |