制約 (Constraint) リファレンス

このモジュールで定義されたクラスはデータベース制約を作成します。これらはモデルの Meta.constraints オプションで追加されます。

組み込みの制約への参照について

制約は django.db.models.constraints で定義されていますが、利便性のために django.db.models にインポートされています。標準的な規約は from django.db import models を使い、制約を models.<Foo>Constraint と呼ぶことです。

抽象基底クラスにおける制約

常に制約の一意な名前を指定する必要があります。そのため、抽象基底クラスに制約を指定することは通常できません。なぜなら、Meta.constraints オプションはサブクラスに継承され、属性(name を含む)の値が毎回同じだからです。名前の衝突を回避するために、名前の一部に '%(app_label)s' および '%(class)s' を含めることができます。これらは、それぞれ、具象モデルの小文字のアプリケーションラベルとクラス名に置き換えられます。例えば、CheckConstraint(condition=Q(age__gte=18), name='%(app_label)s_%(class)s_is_adult') のようになります。

制約のバリデーション

制約は モデルのバリデーション の間にチェックされます。

BaseConstraint

class BaseConstraint(*name, violation_error_code=None, violation_error_message=None)[ソース]

すべての制約の基底クラスです。サブクラスは constraint_sql(), create_sql(), remove_sql(), validate() メソッドを実装しなければなりません。

バージョン 5.0 で非推奨: 位置引数のサポートは非推奨になりました。

すべての制約 (constraint) に共通するパラメータは以下の通りです:

name

BaseConstraint.name

制約の名前。制約には常に一意な名前を指定する必要があります。

violation_error_code

New in Django 5.0.
BaseConstraint.violation_error_code

モデルのバリデーション 中に ValidationError が発生した場合に使用されるエラーコードです。デフォルトは None です。

violation_error_message

BaseConstraint.violation_error_message

モデルのバリデーション の実行中に ValidationError が発生した場合に表示されるエラーメッセージです。デフォルトは "Constraint "%(name)s" is violated." です。

validate()

BaseConstraint.validate(model, instance, exclude=None, using=DEFAULT_DB_ALIAS)[ソース]

モデル model で定義された制約がインスタンス instance で守られているかどうかを検証します。これは、制約が守られていることを確認するために、データベースに対してクエリを実行します。制約を検証するために exclude リストのフィールドが必要な場合、制約は無視されます。

制約に違反した場合は ValidationError を発生させます。

このメソッドはサブクラスで実装する必要があります。

CheckConstraint

class CheckConstraint(*, condition, name, violation_error_code=None, violation_error_message=None)[ソース]

データベースにチェック制約を作成します。

condition

CheckConstraint.condition

制約が強制する条件チェックを指定する Q オブジェクトまたは真偽値の Expression です。

例えば、CheckConstraint(condition=Q(age__gte=18), name='age_gte_18') は、age フィールドが18未満でないことを保証します。

式の順序

Q の引数の順番は必ずしも保持されるわけではありませんが、 Q 式の順番自体は保持されます。これは、パフォーマンス上の理由からチェック制約式の順序を保持するデータベースにとって、重要なことです。例えば、順序が重要な場合は以下の形式を使用します。

CheckConstraint(
    condition=Q(age__gte=18) & Q(expensive_check=condition),
    name="age_gte_18_and_others",
)

Oracle 23c 未満の場合

Oracle 23c 未満で null 許可フィールドを持つチェック制約は、NULL 値を許可する条件を含める必要があります。これにより、validate() がチェック制約の検証と同じように動作します。例えば、age が null 許可フィールドである場合は、次のように記述します:

CheckConstraint(condition=Q(age__gte=18) | Q(age__isnull=True), name="age_gte_18")

バージョン 5.1 で非推奨: check 属性は、condition 属性に置き換えられたため、非推奨となりました。

UniqueConstraint

class UniqueConstraint(*expressions, fields=(), name=None, condition=None, deferrable=None, include=None, opclasses=(), nulls_distinct=None, violation_error_code=None, violation_error_message=None)[ソース]

データベースにユニーク制約(一意性制約)を作成します。

expressions

UniqueConstraint.expressions

位置引数 *expressions により、式やデータベース関数に対する関数的なユニーク制約を作成できます。

例:

UniqueConstraint(Lower("name").desc(), "category", name="unique_lower_name_category")

これは name フィールドの小文字の値を降順で、category フィールドの値をデフォルトの昇順でユニーク制約を作成します。

関数的なユニーク制約は Index.expressions と同じデータベース制約を持ちます。

fields

UniqueConstraint.fields

制約を適用したい一意な列のセットを表すフィールド名のリスト。

例えば、UniqueConstraint(fields=['room', 'date'], name='unique_booking') は各 room が各 date で一度しか予約できないようにします。

condition

UniqueConstraint.condition

制約を適用したい条件を指定する Q オブジェクト。

例:

UniqueConstraint(fields=["user"], condition=Q(status="DRAFT"), name="unique_draft_user")

これは、各ユーザーが1つの DRAFT しか持たないことを保証します。

これらの condition は Index.condition と同じデータベースの制限を持ちます。

deferrable

UniqueConstraint.deferrable

このパラメータを指定すると、遅延可能なユニーク制約を作成できます。使用可能な値は Deferrable.DEFERRED または Deferrable.IMMEDIATE です。例えば:

from django.db.models import Deferrable, UniqueConstraint

UniqueConstraint(
    name="unique_order",
    fields=["order"],
    deferrable=Deferrable.DEFERRED,
)

デフォルトでは、制約は遅延 (DEFERRED) されません。遅延された制約は、トランザクションが終了するまで実行されません。即時 (IMMEDIATE) 制約は、すべてのコマンドの直後に実行されます。

MySQL, MariaDB, SQLite の場合

MySQL、MariaDB、SQLite のいずれも遅延ユニーク制約をサポートしていないため、遅延ユニーク制約は無視されます。

警告

遅延ユニーク制約は、パフォーマンスへのペナルティ を引き起こす可能性があります。

include

UniqueConstraint.include

ユニークなカバリングインデックス (covering index) に非キー列として含めるフィールド名のリストまたはタプル。これにより、include されたフィールドだけを SELECT するクエリと、 (include)、ユニークなフィールドだけでフィルタリングする(fields) クエリにインデックスだけのスキャンを使用できます。

例:

UniqueConstraint(name="unique_booking", fields=["room", "date"], include=["full_name"])

この設定では、roomdate によるフィルタリング、full_name の SELECT の際にデータをインデックスからのみ取得します。

PostgreSQL以外のデータベースでは、非キー列を持つユニーク制約は無視されます。

非キーカラムは Index.include と同じデータベース制約を持ちます。

opclasses

UniqueConstraint.opclasses

この一意なインデックスに使用する PostgreSQL operator クラス の名前です。カスタム演算子クラスが必要な場合は、インデックスの各フィールドに1つずつ指定しなければなりません。

例:

UniqueConstraint(
    name="unique_username", fields=["username"], opclasses=["varchar_pattern_ops"]
)

これは usernamevarchar_pattern_ops を使用する一意なインデックスを作成します。

opclasses はPostgreSQL以外のデータベースでは無視されます。

nulls_distinct

New in Django 5.0.
UniqueConstraint.nulls_distinct

ユニーク制約の対象となる NULL 値を含む行を、互いに異なる行とみなすかどうかを指定します。デフォルト値は None で、ほとんどのバックエンドで True となるデータベースのデフォルト値を使用します。

例:

UniqueConstraint(name="ordering", fields=["ordering"], nulls_distinct=False)

これは、 ordering カラムに NULL 値を格納できるのは1行だけというユニーク制約を作成します。

nulls_distinct によるユニーク制約は、PostgreSQL 15+ 以外のデータベースでは無視されます。

violation_error_code

New in Django 5.0.
UniqueConstraint.violation_error_code

モデルのバリデーション 中に ValidationError が発生した場合に使用されるエラーコードです。デフォルトは None です。

このコードは、fields を持ち、かつ condition を持たない UniqueConstraint には 使用されません 。このような UniqueConstraint は、Field.uniqueMeta.unique_together で定義された制約と同じエラーコードを持ちます。

violation_error_message

UniqueConstraint.violation_error_message

モデルのバリデーション 中に ValidationError が発生した場合に使用されるエラーメッセージです。デフォルトは BaseConstraint.violation_error_message です。

このメッセージは、fields を持ち、かつ condition を持たない UniqueConstraint には 使用されません 。このような UniqueConstraint は、Field.uniqueMeta.unique_together で定義された制約と同じメッセージを表示します。

Back to Top