Django 2.0 リリースノート

December 2, 2017

Django 2.0 へようこそ!

このリリースノートでは、 バージョン 2.0 の新機能 と、Django 1.11 以前からアップグレードする際に注意が必要な 後方互換性が失われた変更点 について説明します。私たちは、非推奨の期間が終了した いくつかの機能の廃止 を行い、さらに いくつかの機能に対する新しい非推奨期間 を開始しました。

このリリースから Django は loose form of semantic versioning を使用し始めますが、通常 2.0 リリースで期待されるような後方互換性を損なう大きな変更はありません。アップグレードは、これまでのフィーチャーリリースと同等の労力で行えるはずです。

既存のプロジェクトをアップデートするときは、 How to upgrade Django to a newer version ガイドに従ってください。

Python バージョン間の互換性

Django 2.0 supports Python 3.4, 3.5, 3.6, and 3.7. We highly recommend and only officially support the latest release of each series.

Python 2.7 をサポートするのは、Django 1.11.x シリーズで最後になります。

Django 2.0 は Python 3.4 をサポートする最後のリリース系列になる予定です。Django 2.0 (April 2019) の end-of-life 後もまだ Python 3.4 を使用する予定であれば、代わりに Django 1.11 LTS (2020年4月までサポート) にとどまってください。ただし、2019年3月には Python 3.4 自体が end-of-life を迎えることに注意してください。

古いバージョンの Django をサポートするサードパーティのライブラリ

Django 2.0 リリース後は、サードパーティアプリの開発者は 1.11 より前のバージョンの Django のサポートを終えるよう推奨します。このとき、python -Wd を使ったパッケージテストを実行して、廃止予定 (deprecation) の警告を出力できます。廃止予定の警告が出ないように修正すれば、アプリケーションは Django 2.0 と互換性のある状態になっているはずです。

Django 2.0 で新しくなったこと

URL ルーティングの構文の簡略化

新しい django.urls.path() 関数を使うと、以前よりシンプルで読みやすい URL ルーティング構文が使用できます。たとえば、次のような以前のリリースの Django の例は、

url(r'^articles/(?P<year>[0-9]{4})/$', views.year_archive),

次のように書き換えられます。

path('articles/<int:year>/', views.year_archive),

新しい構文では、URL パラメータの型強制がサポートされます。この例では、ビューが受け取る year キーワード引数は、文字列ではなく整数になります。また、書き換えた例でマッチする URL の制約も少し緩和されています。たとえば、year として 10000 を指定したとしてもマッチします。正規表現の例にあるような、数字が4文字ちょうどでなければならないという制約がないからです。

django.conf.urls.url() 関数は django.urls.re_path() から利用できるようになりました。差し迫った廃止は必要ないため、後方互換性のために、関数は古い場所にも残されます。また django.conf.urls.include() 関数も django.urls からインポートできるようになりました。したがって、URLconf の中では from django.urls import include, path, re_path とインポートができます。

新しい構文と詳細な解説を紹介するために、URL ディスパッチャ ドキュメントをアップデートしています。

モバイルフレンドリーな contrib.admin

admin サイトがレスポンシブになり、主要な全モバイルデバイスをサポートするようになりました。古いブラウザーでは、多少のデグレーションが発生する可能性があります。

Windows 式

新しい Window 式を使うと、クエリセットに OVER 句を追加できるようになります。式の中では、window 関数aggregate 関数 が使用できます。

マイナーな機能

django.contrib.admin

django.contrib.auth

  • The default iteration count for the PBKDF2 password hasher is increased from 36,000 to 100,000.

django.contrib.gis

django.contrib.postgres

  • The new distinct argument for ArrayAgg determines if concatenated values will be distinct.
  • The new RandomUUID database function returns a version 4 UUID. It requires use of PostgreSQL's pgcrypto extension which can be activated using the new CryptoExtension migration operation.
  • django.contrib.postgres.indexes.GinIndex now supports the fastupdate and gin_pending_list_limit parameters.
  • The new GistIndex class allows creating GiST indexes in the database. The new BtreeGistExtension migration operation installs the btree_gist extension to add support for operator classes that aren't built-in.
  • inspectdb can now introspect JSONField and various RangeFields (django.contrib.postgres must be in INSTALLED_APPS).

django.contrib.sitemaps

  • Added the protocol keyword argument to the GenericSitemap constructor.

キャッシュ

  • cache.set_many() now returns a list of keys that failed to be inserted. For the built-in backends, failed inserts can only happen on memcached.

ファイルストレージ

  • File.open() can be used as a context manager, e.g. with file.open() as f:.

フォーム

ジェネリックビュー (generic view)

管理コマンド

  • inspectdb now translates MySQL's unsigned integer columns to PositiveIntegerField or PositiveSmallIntegerField.
  • The new makemessages --add-location option controls the comment format in PO files.
  • loaddata can now read from stdin.
  • The new diffsettings --output option allows formatting the output in a unified diff format.
  • On Oracle, inspectdb can now introspect AutoField if the column is created as an identity column.
  • On MySQL, dbshell now supports client-side TLS certificates.

マイグレーション

モデル

ページネーション

Request と Response

  • The runserver web server supports HTTP 1.1.

テンプレート

  • To increase the usefulness of Engine.get_default() in third-party apps, it now returns the first engine if multiple DjangoTemplates engines are configured in TEMPLATES rather than raising ImproperlyConfigured.
  • カスタムのテンプレートタグはキーワード引数のみを受け取ることができるようになりました。

テスト

バリデータ

  • The new ProhibitNullCharactersValidator disallows the null character in the input of the CharField form field and its subclasses. Null character input was observed from vulnerability scanning tools. Most databases silently discard null characters, but psycopg2 2.7+ raises an exception when trying to save a null character to a char/text field with PostgreSQL.

2.0 における後方互換性のない変更

いくつかの場所におけバイト文字列のサポートの削除

To support native Python 2 strings, older Django versions had to accept both bytestrings and Unicode strings. Now that Python 2 support is dropped, bytestrings should only be encountered around input/output boundaries (handling of binary fields or HTTP streams, for example). You might have to update your code to limit bytestring usage to a minimum, as Django no longer accepts bytestrings in certain code paths. Python's -b option may help detect that mistake in your code.

For example, reverse() now uses str() instead of force_text() to coerce the args and kwargs it receives, prior to their placement in the URL. For bytestrings, this creates a string with an undesired b prefix as well as additional quotes (str(b'foo') is "b'foo'"). To adapt, call decode() on the bytestring before passing it to reverse().

データベースバックエンド API

This section describes changes that may be needed in third-party database backends.

  • The DatabaseOperations.datetime_cast_date_sql(), datetime_cast_time_sql(), datetime_trunc_sql(), datetime_extract_sql(), and date_interval_sql() methods now return only the SQL to perform the operation instead of SQL and a list of parameters.
  • Third-party database backends should add a DatabaseWrapper.display_name attribute with the name of the database that your backend works with. Django may use it in various messages, such as in system checks.
  • The first argument of SchemaEditor._alter_column_type_sql() is now model rather than table.
  • The first argument of SchemaEditor._create_index_name() is now table_name rather than model.
  • To enable FOR UPDATE OF support, set DatabaseFeatures.has_select_for_update_of = True. If the database requires that the arguments to OF be columns rather than tables, set DatabaseFeatures.select_for_update_of_column = True.
  • To enable support for Window expressions, set DatabaseFeatures.supports_over_clause to True. You may need to customize the DatabaseOperations.window_start_rows_start_end() and/or window_start_range_start_end() methods.
  • Third-party database backends should add a DatabaseOperations.cast_char_field_without_max_length attribute with the database data type that will be used in the Cast function for a CharField if the max_length argument isn't provided.
  • The first argument of DatabaseCreation._clone_test_db() and get_test_db_clone_settings() is now suffix rather than number (in case you want to rename the signatures in your backend for consistency). django.test also now passes those values as strings rather than as integers.
  • Third-party database backends should add a DatabaseIntrospection.get_sequences() method based on the stub in BaseDatabaseIntrospection.

Oracle 11.2 に対するサポートの終了

The end of upstream support for Oracle 11.2 is Dec. 2020. Django 1.11 will be supported until April 2020 which almost reaches this date. Django 2.0 officially supports Oracle 12.1+.

デフォルトの MySQL の isolation レベルを read committed に変更

MySQL のデフォルトの isolation レベルは repeatable read ですが、この設定は Django の典型的な使用例でデータロスを引き起こす可能性があります。データロスを防ぎ、他のデータベースとの一貫性を保つため、デフォルトの isolation level が read committed に変更されました。必要があれば、DATABASES 設定を変更することで、異なる isolation level を使う こともできます。

AbstractUser.last_namemax_length が 150 文字に拡大

django.contrib.auth.models.User.last_name に対するマイグレーションが含まれます。AbstractUser を継承したカスタムのユーザーモデルがある場合、そのユーザーモデルに対するデータベースのマイグレーションを生成して適用する必要があります。

last name に 30 文字の制限を課し続けたい場合は、次のようなカスタムフォームを使用してください。

from django.contrib.auth.forms import UserChangeForm

class MyUserChangeForm(UserChangeForm):
    last_name = forms.CharField(max_length=30, required=False)

管理サイトでユーザーを編集する時にもこの制約を課し続けたい場合は、以下のように、フォームに UserAdmin.form を設定してください。

from django.contrib.auth.admin import UserAdmin
from django.contrib.auth.models import User

class MyUserAdmin(UserAdmin):
    form = MyUserChangeForm

admin.site.unregister(User)
admin.site.register(User, MyUserAdmin)

スライス後の QuerySet.reverse() および last() の禁止

スライスを行った後の queryset に対して QuerySet.reverse()last() を呼ぶと、ソート後にスライス操作が実行されるため、実行結果が予想外なものになってしまいます。そのため、今後、次のような操作は禁止されます。

>>> Model.objects.all()[:2].reverse()
Traceback (most recent call last):
...
TypeError: Cannot reverse a query once a slice has been taken.

フォームフィールドに省略可能な引数を渡す時、位置引数として指定することができなくなりました。

To help prevent runtime errors due to incorrect ordering of form field arguments, optional arguments of built-in form fields are no longer accepted as positional arguments. For example:

forms.IntegerField(25, 10)

raises an exception and should be replaced with:

forms.IntegerField(max_value=25, min_value=10)

call_command() validates the options it receives

call_command() now validates that the argument parser of the command being called defines all of the options passed to call_command().

For custom management commands that use options not created using parser.add_argument(), add a stealth_options attribute on the command:

class MyCommand(BaseCommand):
    stealth_options = ('option_name', ...)

Indexes no longer accept positional arguments

例:

models.Index(['headline', '-pub_date'], 'index_name')

raises an exception and should be replaced with:

models.Index(fields=['headline', '-pub_date'], name='index_name')

Foreign key constraints are now enabled on SQLite

This will appear as a backwards-incompatible change (IntegrityError: FOREIGN KEY constraint failed) if attempting to save an existing model instance that's violating a foreign key constraint.

Foreign keys are now created with DEFERRABLE INITIALLY DEFERRED instead of DEFERRABLE IMMEDIATE. Thus, tables may need to be rebuilt to recreate foreign keys with the new definition, particularly if you're using a pattern like this:

from django.db import transaction

with transaction.atomic():
    Book.objects.create(author_id=1)
    Author.objects.create(id=1)

If you don't recreate the foreign key as DEFERRED, the first create() would fail now that foreign key constraints are enforced.

Backup your database first! After upgrading to Django 2.0, you can then rebuild tables using a script similar to this:

from django.apps import apps
from django.db import connection

for app in apps.get_app_configs():
    for model in app.get_models(include_auto_created=True):
        if model._meta.managed and not (model._meta.proxy or model._meta.swapped):
            for base in model.__bases__:
                if hasattr(base, '_meta'):
                    base._meta.local_many_to_many = []
            model._meta.local_many_to_many = []
            with connection.schema_editor() as editor:
                editor._remake_table(model)

This script hasn't received extensive testing and needs adaption for various cases such as multiple databases. Feel free to contribute improvements.

In addition, because of a table alteration limitation of SQLite, it's prohibited to perform RenameModel and RenameField operations on models or fields referenced by other models in a transaction. In order to allow migrations containing these operations to be applied, you must set the Migration.atomic attribute to False.

その他

  • The SessionAuthenticationMiddleware class is removed. It provided no functionality since session authentication is unconditionally enabled in Django 1.10.

  • The default HTTP error handlers (handler404, etc.) are now callables instead of dotted Python path strings. Django favors callable references since they provide better performance and debugging experience.

  • RedirectView no longer silences NoReverseMatch if the pattern_name doesn't exist.

  • When USE_L10N is off, FloatField and DecimalField now respect DECIMAL_SEPARATOR and THOUSAND_SEPARATOR during validation. For example, with the settings:

    USE_L10N = False
    USE_THOUSAND_SEPARATOR = True
    DECIMAL_SEPARATOR = ','
    THOUSAND_SEPARATOR = '.'
    

    an input of "1.345" is now converted to 1345 instead of 1.345.

  • Subclasses of AbstractBaseUser are no longer required to implement get_short_name() and get_full_name(). (The base implementations that raise NotImplementedError are removed.) django.contrib.admin uses these methods if implemented but doesn't require them. Third-party apps that use these methods may want to adopt a similar approach.

  • The FIRST_DAY_OF_WEEK and NUMBER_GROUPING format settings are now kept as integers in JavaScript and JSON i18n view outputs.

  • assertNumQueries() now ignores connection configuration queries. Previously, if a test opened a new database connection, those queries could be included as part of the assertNumQueries() count.

  • The default size of the Oracle test tablespace is increased from 20M to 50M and the default autoextend size is increased from 10M to 25M.

  • To improve performance when streaming large result sets from the database, QuerySet.iterator() now fetches 2000 rows at a time instead of 100. The old behavior can be restored using the chunk_size parameter. For example:

    Book.objects.iterator(chunk_size=100)
    
  • Providing unknown package names in the packages argument of the JavaScriptCatalog view now raises ValueError instead of passing silently.

  • A model instance's primary key now appears in the default Model.__str__() method, e.g. Question object (1).

  • makemigrations now detects changes to the model field limit_choices_to option. Add this to your existing migrations or accept an auto-generated migration for fields that use it.

  • Performing queries that require automatic spatial transformations now raises NotImplementedError on MySQL instead of silently using non-transformed geometries.

  • django.core.exceptions.DjangoRuntimeWarning is removed. It was only used in the cache backend as an intermediate class in CacheKeyWarning's inheritance of RuntimeWarning.

  • Renamed BaseExpression._output_field to output_field. You may need to update custom expressions.

  • In older versions, forms and formsets combine their Media with widget Media by concatenating the two. The combining now tries to preserve the relative order of elements in each list. MediaOrderConflictWarning is issued if the order can't be preserved.

  • django.contrib.gis.gdal.OGRException is removed. It's been an alias for GDALException since Django 1.8.

  • Support for GEOS 3.3.x is dropped.

  • The way data is selected for GeometryField is changed to improve performance, and in raw SQL queries, those fields must now be wrapped in connection.ops.select. See the Raw queries note in the GIS tutorial for an example.

Features deprecated in 2.0

context argument of Field.from_db_value() and Expression.convert_value()

The context argument of Field.from_db_value() and Expression.convert_value() is unused as it's always an empty dictionary. The signature of both methods is now:

(self, value, expression, connection)

instead of:

(self, value, expression, connection, context)

Support for the old signature in custom fields and expressions remains until Django 3.0.

その他

  • The django.db.backends.postgresql_psycopg2 module is deprecated in favor of django.db.backends.postgresql. It's been an alias since Django 1.9. This only affects code that imports from the module directly. The DATABASES setting can still use 'django.db.backends.postgresql_psycopg2', though you can simplify that by using the 'django.db.backends.postgresql' name added in Django 1.9.
  • django.shortcuts.render_to_response() is deprecated in favor of django.shortcuts.render(). render() takes the same arguments except that it also requires a request.
  • The DEFAULT_CONTENT_TYPE setting is deprecated. It doesn't interact well with third-party apps and is obsolete since HTML5 has mostly superseded XHTML.
  • HttpRequest.xreadlines() is deprecated in favor of iterating over the request.
  • The field_name keyword argument to QuerySet.earliest() and QuerySet.latest() is deprecated in favor of passing the field names as arguments. Write .earliest('pub_date') instead of .earliest(field_name='pub_date').

2.0 で削除された機能

以下の機能は、非推奨サイクルの終わりに達したため、Django 2.0 で削除されます。

詳しくは Features deprecated in 1.9 を見てください。ここには、プロジェクトからこれらの機能を削除する方法についても書かれています。

  • django.dispatch.signals.Signal.disconnect() から weak 引数が削除されました。
  • django.db.backends.base.BaseDatabaseOperations.check_aggregate_support() が削除されました。
  • django.forms.extras パッケージが削除されました。
  • assignment_tag ヘルパーが削除されました。
  • SimpleTestCase.assertsRedirects()host 引数が削除されました。絶対 URL が相対 URL と同一の場合に、相対 URL と等価なものと解釈する互換レイヤーも削除されました。
  • Field.rel および Field.remote_field.to が削除されました。
  • ForeignKey および OneToOneFieldon_delete 引数が削除され、新たにモデルとマイグレーションが必須になりました。更新が必要になるマイグレーションの数を少なくするために、マイグレーションを圧縮すること (squashing migrations) を検討してください。
  • django.db.models.fields.add_lazy_relation() が削除されました。
  • When time zone support is enabled, database backends that don't support time zones no longer convert aware datetimes to naive values in UTC anymore when such values are passed as parameters to SQL queries executed outside of the ORM, e.g. with cursor.execute().
  • django.contrib.auth.tests.utils.skipIfCustomUser() が削除されました。
  • GeoManager および GeoQuerySet クラスが削除されました。
  • django.contrib.gis.geoip モジュールが削除されました。
  • 以下のテンプレートローダーから supports_recursion チェックが削除されました。
    • django.template.engine.Engine.find_template()
    • django.template.loader_tags.ExtendsNode.find_template()
    • django.template.loaders.base.Loader.supports_recursion()
    • django.template.loaders.cached.Loader.supports_recursion()
  • load_template および load_template_sources テンプレートローダーメソッドが削除されました。
  • 以下のテンプレートローダーから template_dirs 引数が削除されました。
    • django.template.loaders.base.Loader.get_template()
    • django.template.loaders.cached.Loader.cache_key()
    • django.template.loaders.cached.Loader.get_template()
    • django.template.loaders.cached.Loader.get_template_sources()
    • django.template.loaders.filesystem.Loader.get_template_sources()
  • django.template.loaders.base.Loader.__call__() が削除されました。
  • exception 引数を受け付けない自作のエラービューのサポートが終了しました。
  • django.utils.feedgenerator.Atom1Feed および django.utils.feedgenerator.RssFeedmime_type 属性が削除されました。
  • include()app_name 引数が削除されました。
  • (including admin.site.urls) as the first argument to include() の1番目の引数として、admin.site.urls を含む3タプルを受け取る機能が削除されました。
  • アプリケーションの名前空間が存在しない URL インスタンスの名前空間を設定する機能が削除されました。
  • Field._get_val_from_obj() が削除されました。
  • django.template.loaders.eggs.Loader が削除されました。
  • contrib.auth 関数ベースビューの current_app 引数が削除されました。
  • SimpleTestCase.assertRaisesMessage()callable_obj キーワード引数が削除されました。
  • ModelAdmin メソッドの allow_tags 属性のサポートが削除されました。
  • SyndicationFeed.add_item()enclosure キーワード引数が削除されました。
  • django.template.base.Origin から django.template.loader.LoaderOrigin および django.template.base.StringOrigin エイリアスが削除されました。

以下の変更の詳細については、1.10 で非推奨になった機能 を見てください。

  • makemigrations --exit オプションが削除されました。
  • 逆参照外部キー (reverse foreign key) または many-to-many リレーションに対する値の直接代入のサポートが削除されました。
  • django.contrib.gis.geos.GEOSGeometryget_srid()set_srid() メソッドが削除されました。
  • django.contrib.gis.geos.Pointget_x()set_x()get_y()set_y()get_z()set_z() の各メソッドが削除されました。
  • django.contrib.gis.geos.Pointget_coords()set_coords() メソッドが削除されました。
  • django.contrib.gis.geos.MultiPolygoncascaded_union プロパティが削除されました。
  • django.utils.functional.allow_lazy() が削除されました。
  • shell --plain オプションが削除されました。
  • django.core.urlresolvers モジュールが削除され、新しい場所 django.urls に移動しました。
  • CommaSeparatedIntegerField が削除されました。ただし、歴史的マイグレーション (historical migrations) のみ、継続してサポートします。
  • テンプレートの Context.has_key() メソッドが削除されました。
  • django.core.files.storage.Storage.accessed_time()created_time()、および modified_time() メソッドのサポートが削除されました。
  • default_related_name が設定されていた場合に、モデル名をクエリー・ルックアップとして使用する機能のサポートが削除されました。
  • MySQL の __search ルックアップが削除されました。
  • _apply_rel_filters() メソッドを持たない自作の related manager class をサポートするための shim が削除されました。
  • User.is_authenticated()User.is_anonymous() をプロパティではなくメソッドをして使う機能はサポートされなくなりました。
  • Model._meta.virtual_fields 属性が削除されました。
  • Field.contribute_to_class()virtual_only キーワード引数と、Model._meta.add_field()virtual キーワード引数が削除されました。
  • javascript_catalog()json_catalog() ビューが削除されました。
  • django.contrib.gis.utils.precision_wkt() が削除されました。
  • 複数テーブルの継承において、OneToOneField を暗示的に parent_link に変換する機能を削除しました。
  • Widget._format_value() のサポートが削除されました。
  • FileFieldget_directory_name() および get_filename() メソッドが削除されました。
  • mark_for_escaping() 関数と、この関数を使用するクラス EscapeDataEscapeBytesEscapeTextEscapeString、および EscapeUnicode が削除されました。
  • escape フィルタが新らしく django.utils.html.conditional_escape() を使用するようになりました。
  • Manager.use_for_related_fields が削除されました。
  • Model Manager inheritance follows MRO inheritance rules. The requirement to use Meta.manager_inheritance_from_future to opt-in to the behavior is removed.
  • settings.MIDDLEWARE_CLASSES を使用した古いスタイルのミドルウェアのサポートが削除されました。