コンテンツにスキップ

Mypy

Pydantic は、すぐにmypyとうまく連携します。

ただし、Pydantic には、コードの型チェック機能を向上させる、多くの重要な pydantic 固有の機能を mypy に追加する mypy プラグインも付属しています。

たとえば、次のスクリプトについて考えてみましょう。

from datetime import datetime
from typing import List, Optional

from pydantic import BaseModel


class Model(BaseModel):
    age: int
    first_name = 'John'
    last_name: Optional[str] = None
    signup_ts: Optional[datetime] = None
    list_of_ints: List[int]


m = Model(age=42, list_of_ints=[1, '2', b'3'])
print(m.middle_name)  # not a model field!
Model()  # will raise a validation error for age and list_of_ints

特別な設定がなければ、mypy は欠落しているモデルフィールドのアノテーションを捕捉せず、Pydantic が正しく解析するlist_of_ints引数について警告します。

test.py:15: error: List item 1 has incompatible type "str"; expected "int"  [list-item]
test.py:15: error: List item 2 has incompatible type "bytes"; expected "int"  [list-item]
test.py:16: error: "Model" has no attribute "middle_name"  [attr-defined]
test.py:17: error: Missing named argument "age" for "Model"  [call-arg]
test.py:17: error: Missing named argument "list_of_ints" for "Model"  [call-arg]

ただし、プラグインを有効にすると、正しいエラーが表示されます。

9: error: Untyped fields disallowed  [pydantic-field]
16: error: "Model" has no attribute "middle_name"  [attr-defined]
17: error: Missing named argument "age" for "Model"  [call-arg]
17: error: Missing named argument "list_of_ints" for "Model"  [call-arg]

pydantic mypy プラグインを使用すると、フィールド名や型が変更された場合に mypy が間違いを検出できるため、恐れることなくモデルをリファクタリングできます。

他にもメリットはあります!詳細については、以下を参照してください。

プラグインなしで mypy を使用する

次のようにして、mypy を通じてコードを実行できます。

mypy \
  --ignore-missing-imports \
  --follow-imports=skip \
  --strict-optional \
  pydantic_mypy_test.py

厳密なオプション

コードを--strict-optionalで渡すには、デフォルトとしてNoneを使用して、すべてのフィールドにOptional[]またはOptional[]のエイリアスを使用する必要があります。 (これは mypy の標準です。)

その他の Pydantic インターフェース

Pydanticデータクラスvalidate_callデコレータも mypy で適切に動作するはずです。

Mypy プラグインの機能

Model.__init__の署名を生成する

  • 動的に決定される別名を持たない必須フィールドは、必須キーワード引数として含まれます。
  • Config.populate_by_name=Trueの場合、生成された署名はエイリアスではなくフィールド名を使用します。
  • Config.extra='forbid'で、動的に決定されるエイリアスを使用しない場合、生成された署名は予期しない入力を許可しません。
  • オプション: init_forbid_extraプラグイン設定Trueに設定されている場合、 Config.extra'forbid'でなくても、 __init__への予期しない入力によってエラーが発生します。
  • オプション: init_typedプラグイン設定Trueに設定されている場合、生成された署名はモデル フィールドのタイプを使用します (それ以外の場合、解析を可能にするためにAny注釈が付けられます)。

Model.model_constructの型付き署名を生成する

  • model_constructメソッドは、入力データが有効であることがわかっており、解析する必要がない場合の__init__の代替手段です。このメソッドは実行時検証を実行しないため、エラーを検出するには静的チェックが重要です。

Config.frozen尊重する

  • Config.frozenTrueの場合、モデル フィールドの値を変更しようとすると、mypy エラーが発生します。参照。偽の不変性

dataclassesの署名を生成する

  • で飾られたクラス @pydantic.dataclasses.dataclass 標準の Python データクラスと同じように型がチェックされます
  • @pydantic.dataclasses.dataclass デコレーターは、 Configサブクラスと同じ意味を持つconfigキーワード引数を受け入れます。

Fielddefaultdefault_factoryのタイプを尊重します。

  • フィールドにdefaultdefault_factory両方が含まれている場合、静的チェック中にエラーが発生します。
  • defaultのタイプとdefault_factory値は、フィールドのタイプと互換性がある必要があります。

型なしフィールドの使用について警告する

  • タイプに注釈を付けずにモデルにパブリック属性を割り当てると、必ず mypy エラーが発生します。
  • ClassVar を設定することが目的の場合は、typeing.ClassVar を使用してフィールドに明示的に注釈を付ける必要があります。

オプションの機能:

必要な動的エイリアスの使用を防止する

  • warn_required_dynamic_aliasesプラグイン設定Trueに設定されている場合、 Config.populate_by_name=Falseのモデルで動的に決定されるエイリアスまたはエイリアス ジェネレーターを使用するたびに、mypy エラーが発生します。
  • このようなエイリアスが存在すると、 mypy は__init__へのチェック呼び出しを適切に入力できないため、これは重要です。この場合、デフォルトではすべての引数がオプションとして扱われます。

プラグインを有効にする

プラグインを有効にするには、 mypy 構成ファイル( mypy.inipyproject.toml 、またはsetup.cfgなど) のプラグインのリストにpydantic.mypyを追加するだけです。

開始するには、次の内容を含むmypy.iniファイルを作成するだけです。

[mypy]
plugins = pydantic.mypy

!!! note pydantic.v1モデルを使用している場合は、プラグインのリストにpydantic.v1.mypyを追加する必要があります。

このプラグインは、mypy バージョン>=0.930と互換性があります。

詳細については、プラグイン設定ドキュメントを参照してください。

プラグインの設定

プラグイン設定の値を変更するには、mypy 構成ファイルに[pydantic-mypy]というセクションを作成し、オーバーライドする設定のキーと値のペアを追加します。

すべてのプラグイン厳密性フラグ (および他の mypy 厳密性フラグも) が有効になっているmypy.iniファイルは次のようになります。

[mypy]
plugins = pydantic.mypy

follow_imports = silent
warn_redundant_casts = True
warn_unused_ignores = True
disallow_any_generics = True
check_untyped_defs = True
no_implicit_reexport = True

# for strict mypy: (this is the tricky one :-))
disallow_untyped_defs = True

[pydantic-mypy]
init_forbid_extra = True
init_typed = True
warn_required_dynamic_aliases = True

mypy>=0.900以降、 mypy config はmypy.iniではなくpyproject.tomlファイルに含まれる場合もあります。上記と同じ構成は次のようになります。

[tool.mypy]
plugins = [
  "pydantic.mypy"
]

follow_imports = "silent"
warn_redundant_casts = true
warn_unused_ignores = true
disallow_any_generics = true
check_untyped_defs = true
no_implicit_reexport = true

# for strict mypy: (this is the tricky one :-))
disallow_untyped_defs = true

[tool.pydantic-mypy]
init_forbid_extra = true
init_typed = true
warn_required_dynamic_aliases = true

--disallow-any-explicitに関する注意事項

--disallow-any-explicit mypy 構成設定 (またはAnyを禁止するその他の設定) を使用している場合、 BaseModel拡張するときにno-any-explicitエラーが発生する可能性があります。これは、デフォルトで、Pydantic のmypyプラグインが次のようなシグネチャを持つ__init__メソッドを追加するためです。 def __init__(self, field_1: Any, field_2: Any, **kwargs: Any):

!!! 「なぜ余分な署名が必要なのでしょうか?」 Pydantic mypyプラグインは、次のようなシグネチャを持つ__init__メソッドを追加します。 def __init__(self, field_1: Any, field_2: Any, **kwargs: Any): フィールドの注釈と一致しない型でモデルを初期化するときの型エラーを避けるため。たとえば、 Model(date='2024-01-01')このAnyシグネチャがないと型エラーを引き起こしますが、Pydantic には文字列'2024-01-01'datetime.date型に解析する機能があります。

この問題を解決するには、Pydantic mypy プラグインの厳密モード設定を有効にする必要があります。具体的には、次のオプションを[pydantic-mypy]セクションに追加します。

[tool.pydantic-mypy]
init_forbid_extra = true
init_typed = true

init_forbid_extra = Trueの場合、 **kwargs生成された__init__シグネチャから削除されます。 init_typed = Trueの場合、フィールドのAny型は実際の型ヒントに置き換えられます。

この構成により、Pydantic モデルでエラーが発生することなく--disallow-any-explicitを使用できるようになります。ただし、このより厳格なチェックにより、一部の有効な Pydantic ユースケース (日時フィールドに文字列を渡すなど) が型エラーとしてフラグ付けされる可能性があることに注意してください。


本文总阅读量