Django 5.0 release notes - UNDER DEVELOPMENT¶
Expected December 2023
Welcome to Django 5.0!
These release notes cover the new features, as well as some backwards incompatible changes you’ll want to be aware of when upgrading from Django 4.2 or earlier. We’ve begun the deprecation process for some features.
See the How to upgrade Django to a newer version guide if you’re updating an existing project.
Python compatibility¶
Django 5.0 supports Python 3.10, 3.11, and 3.12. We highly recommend and only officially support the latest release of each series.
The Django 4.2.x series is the last to support Python 3.8 and 3.9.
Third-party library support for older version of Django¶
Following the release of Django 5.0, we suggest that third-party app authors
drop support for all versions of Django prior to 4.2. At that time, you should
be able to run your package’s tests using python -Wd
so that deprecation
warnings appear. After making the deprecation warning fixes, your app should be
compatible with Django 5.0.
What’s new in Django 5.0¶
Facet filters in the admin¶
Facet counts are now shown for applied filters in the admin changelist when
toggled on via the UI. This behavior can be changed via the new
ModelAdmin.show_facets
attribute. For more information see
Facets.
Simplified templates for form field rendering¶
Django 5.0 introduces the concept of a field group, and field group templates. This simplifies rendering of the related elements of a Django form field such as its label, widget, help text, and errors.
For example, the template below:
<form>
...
<div>
{{ form.name.label }}
{% if form.name.help_text %}
<div class="helptext">{{ form.name.help_text|safe }}</div>
{% endif %}
{{ form.name.errors }}
{{ form.name }}
<div class="row">
<div class="col">
{{ form.email.label }}
{% if form.email.help_text %}
<div class="helptext">{{ form.email.help_text|safe }}</div>
{% endif %}
{{ form.email.errors }}
{{ form.email }}
</div>
<div class="col">
{{ form.password.label }}
{% if form.password.help_text %}
<div class="helptext">{{ form.password.help_text|safe }}</div>
{% endif %}
{{ form.password.errors }}
{{ form.password }}
</div>
</div>
</div>
...
</form>
Can now be simplified to:
<form>
...
<div>
{{ form.name.as_field_group }}
<div class="row">
<div class="col">{{ form.email.as_field_group }}</div>
<div class="col">{{ form.password.as_field_group }}</div>
</div>
</div>
...
</form>
as_field_group()
renders fields with the
"django/forms/field.html"
template by default and can be customized on a
per-project, per-field, or per-request basis. See
Reusable field group templates.
Database-computed default values¶
The new Field.db_default
parameter
sets a database-computed default value. For example:
from django.db import models
from django.db.models.functions import Now, Pi
class MyModel(models.Model):
age = models.IntegerField(db_default=18)
created = models.DateTimeField(db_default=Now())
circumference = models.FloatField(db_default=2 * Pi())
Minor features¶
django.contrib.admin
¶
- The new
AdminSite.get_log_entries()
method allows customizing the queryset for the site’s listed log entries. - The
django.contrib.admin.AllValuesFieldListFilter
,ChoicesFieldListFilter
,RelatedFieldListFilter
, andRelatedOnlyFieldListFilter
admin filters now handle multi-valued query parameters. XRegExp
is upgraded from version 3.2.0 to 5.1.1.
django.contrib.auth
¶
- The default iteration count for the PBKDF2 password hasher is increased from 600,000 to 720,000.
AuthenticationMiddleware
now adds anHttpRequest.auser()
asynchronous method that returns the currently logged-in user.- The new
django.contrib.auth.hashers.acheck_password()
asynchronous function andAbstractBaseUser.acheck_password()
method allow asynchronous checking of user passwords.
django.contrib.gis
¶
- The new
ClosestPoint()
function returns a 2-dimensional point on the geometry that is closest to another geometry.
django.contrib.postgres
¶
- The new
violation_error_code
attribute ofExclusionConstraint
allows customizing thecode
ofValidationError
raised during model validation.
Asynchronous views¶
- Under ASGI,
http.disconnect
events are now handled. This allows views to perform any necessary cleanup if a client disconnects before the response is generated. See Handling disconnects for more details.
Cache¶
- …
CSRF¶
- …
Decorators¶
- The following decorators now support wrapping asynchronous view functions:
cache_control()
never_cache()
no_append_slash()
xframe_options_deny()
xframe_options_sameorigin()
xframe_options_exempt()
Email¶
- …
Error Reporting¶
- …
File Storage¶
- …
File Uploads¶
- …
Forms¶
ChoiceField.choices
now accepts Choices classes directly instead of requiring expansion with thechoices
attribute.- The new
assume_scheme
argument forURLField
allows specifying a default URL scheme.
Generic Views¶
- …
Internationalization¶
- …
Logging¶
- …
Management Commands¶
- …
Migrations¶
- …
Models¶
- The new
create_defaults
argument ofQuerySet.update_or_create()
andQuerySet.aupdate_or_create()
methods allows specifying a different field values for the create operation. - The new
violation_error_code
attribute ofBaseConstraint
,CheckConstraint
, andUniqueConstraint
allows customizing thecode
ofValidationError
raised during model validation. Field.choices
now accepts Choices classes directly instead of requiring expansion with thechoices
attribute.
Pagination¶
- The new
django.core.paginator.Paginator.error_messages
argument allows customizing the error messages raised byPaginator.page()
.
Requests and Responses¶
- …
Security¶
- …
Serialization¶
- …
Signals¶
- The new
Signal.asend()
andSignal.asend_robust()
methods allow asynchronous signal dispatch. Signal receivers may be synchronous or asynchronous, and will be automatically adapted to the correct calling style.
Tests¶
- …
URLs¶
- …
Utilities¶
- …
Validators¶
- …
Backwards incompatible changes in 5.0¶
Database backend API¶
This section describes changes that may be needed in third-party database backends.
DatabaseFeatures.supports_expression_defaults
should be set toFalse
if the database doesn’t support using database functions as defaults.DatabaseFeatures.supports_default_keyword_in_insert
should be set toFalse
if the database doesn’t support theDEFAULT
keyword inINSERT
queries.DatabaseFeatures.supports_default_keyword_in_bulk insert
should be set toFalse
if the database doesn’t support theDEFAULT
keyword in bulkINSERT
queries.
Using create_defaults__exact
may now be required with QuerySet.update_or_create()
¶
QuerySet.update_or_create()
now supports the parameter
create_defaults
. As a consequence, any models that have a field named
create_defaults
that are used with an update_or_create()
should specify
the field in the lookup with create_defaults__exact
.
Miscellaneous¶
- The
instance
argument of the undocumentedBaseModelFormSet.save_existing()
method is renamed toobj
. - The undocumented
django.contrib.admin.helpers.checkbox
is removed. - Integer fields are now validated as 64-bit integers on SQLite to match the
behavior of
sqlite3
. - The undocumented
Query.annotation_select_mask
attribute is changed from a set of strings to an ordered list of strings. ImageField.update_dimension_fields()
is no longer called on thepost_init
signal ifwidth_field
andheight_field
are not set.Now
database function now usesLOCALTIMESTAMP
instead ofCURRENT_TIMESTAMP
on Oracle.
Features deprecated in 5.0¶
Miscellaneous¶
- The
DjangoDivFormRenderer
andJinja2DivFormRenderer
transitional form renderers are deprecated. - Passing positional arguments
name
andviolation_error_message
toBaseConstraint
is deprecated in favor of keyword-only arguments. request
is added to the signature ofModelAdmin.lookup_allowed()
. Support forModelAdmin
subclasses that do not accept this argument is deprecated.- The
get_joining_columns()
method ofForeignObject
andForeignObjectRel
is deprecated. Starting with Django 6.0,django.db.models.sql.datastructures.Join
will no longer fallback toget_joining_columns()
. Subclasses should implementget_joining_fields()
instead. - The
ForeignObject.get_reverse_joining_columns()
method is deprecated. - The default scheme for
forms.URLField
will change from"http"
to"https"
in Django 6.0.
Features removed in 5.0¶
These features have reached the end of their deprecation cycle and are removed in Django 5.0.
See Features deprecated in 4.0 for details on these changes, including how to remove usage of these features.
- The
SERIALIZE
test setting is removed. - The undocumented
django.utils.baseconv
module is removed. - The undocumented
django.utils.datetime_safe
module is removed. - The default value of the
USE_TZ
setting is changed fromFalse
toTrue
. - The default sitemap protocol for sitemaps built outside the context of a
request is changed from
'http'
to'https'
. - The
extra_tests
argument forDiscoverRunner.build_suite()
andDiscoverRunner.run_tests()
is removed. - The
django.contrib.postgres.aggregates.ArrayAgg
,JSONBAgg
, andStringAgg
aggregates no longer return[]
,[]
, and''
, respectively, when there are no rows. - The
USE_L10N
setting is removed. - The
USE_DEPRECATED_PYTZ
transitional setting is removed. - Support for
pytz
timezones is removed. - The
is_dst
argument is removed from:QuerySet.datetimes()
django.utils.timezone.make_aware()
django.db.models.functions.Trunc()
django.db.models.functions.TruncSecond()
django.db.models.functions.TruncMinute()
django.db.models.functions.TruncHour()
django.db.models.functions.TruncDay()
django.db.models.functions.TruncWeek()
django.db.models.functions.TruncMonth()
django.db.models.functions.TruncQuarter()
django.db.models.functions.TruncYear()
- The
django.contrib.gis.admin.GeoModelAdmin
andOSMGeoAdmin
classes are removed. - The undocumented
BaseForm._html_output()
method is removed. - The ability to return a
str
, rather than aSafeString
, when rendering anErrorDict
andErrorList
is removed.
See Features deprecated in 4.1 for details on these changes, including how to remove usage of these features.
- The
SitemapIndexItem.__str__()
method is removed. - The
CSRF_COOKIE_MASKED
transitional setting is removed. - The
name
argument ofdjango.utils.functional.cached_property()
is removed. - The
opclasses
argument ofdjango.contrib.postgres.constraints.ExclusionConstraint
is removed. - The undocumented ability to pass
errors=None
toSimpleTestCase.assertFormError()
andassertFormsetError()
is removed. django.contrib.sessions.serializers.PickleSerializer
is removed.- The usage of
QuerySet.iterator()
on a queryset that prefetches related objects without providing thechunk_size
argument is no longer allowed. - Passing unsaved model instances to related filters is no longer allowed.
created=True
is required in the signature ofRemoteUserBackend.configure_user()
subclasses.- Support for logging out via
GET
requests in thedjango.contrib.auth.views.LogoutView
anddjango.contrib.auth.views.logout_then_login()
is removed. - The
django.utils.timezone.utc
alias todatetime.timezone.utc
is removed. - Passing a response object and a form/formset name to
SimpleTestCase.assertFormError()
andassertFormSetError()
is no longer allowed. - The
django.contrib.gis.admin.OpenLayersWidget
is removed.
- The
django.contrib.auth.hashers.CryptPasswordHasher
is removed.
- The
"django/forms/default.html"
and"django/forms/formsets/default.html"
templates are removed. - The default form and formset rendering style is changed to the div-based.
- Passing
nulls_first=False
ornulls_last=False
toExpression.asc()
andExpression.desc()
methods, and theOrderBy
expression is no longer allowed.