Skip to content
/ mypy Public
  • Sponsor python/mypy

  • Notifications You must be signed in to change notification settings
  • Fork 2.9k

Commit 5e119d0

Browse files
authoredJan 14, 2025
Ignore dataclass.__replace__ LSP violations (#18464)
Refining dataclass attributes with a narrower type has historically been accepted. Mypy shouldn't emit an LSP warning for the synthesized `__replace__` method added in Python 3.13 either. Users are instead encouraged to enable `--enable-error-code mutable-override` to highlight potential issues. Fixes #18216
1 parent b68c545 commit 5e119d0

File tree

3 files changed

+42
-12
lines changed

3 files changed

+42
-12
lines changed
 

‎mypy/checker.py

+9-6
Original file line numberDiff line numberDiff line change
@@ -1980,12 +1980,15 @@ def check_method_override(
19801980
Return a list of base classes which contain an attribute with the method name.
19811981
"""
19821982
# Check against definitions in base classes.
1983-
check_override_compatibility = defn.name not in (
1984-
"__init__",
1985-
"__new__",
1986-
"__init_subclass__",
1987-
"__post_init__",
1988-
) and (self.options.check_untyped_defs or not defn.is_dynamic())
1983+
check_override_compatibility = (
1984+
defn.name not in ("__init__", "__new__", "__init_subclass__", "__post_init__")
1985+
and (self.options.check_untyped_defs or not defn.is_dynamic())
1986+
and (
1987+
# don't check override for synthesized __replace__ methods from dataclasses
1988+
defn.name != "__replace__"
1989+
or defn.info.metadata.get("dataclass_tag") is None
1990+
)
1991+
)
19891992
found_method_base_classes: list[TypeInfo] = []
19901993
for base in defn.info.mro[1:]:
19911994
result = self.check_method_or_accessor_override_for_base(

‎test-data/unit/check-classes.test

+14
Original file line numberDiff line numberDiff line change
@@ -687,6 +687,20 @@ class B(A):
687687
def h(cls) -> int: pass
688688
[builtins fixtures/classmethod.pyi]
689689

690+
[case testOverrideReplaceMethod]
691+
# flags: --show-error-codes
692+
from typing import Optional
693+
from typing_extensions import Self
694+
class A:
695+
def __replace__(self, x: Optional[str]) -> Self: pass
696+
697+
class B(A):
698+
def __replace__(self, x: str) -> Self: pass # E: \
699+
# E: Argument 1 of "__replace__" is incompatible with supertype "A"; supertype defines the argument type as "Optional[str]" [override] \
700+
# N: This violates the Liskov substitution principle \
701+
# N: See https://mypy.readthedocs.io/en/stable/common_issues.html#incompatible-overrides
702+
[builtins fixtures/tuple.pyi]
703+
690704
[case testAllowCovarianceInReadOnlyAttributes]
691705
from typing import Callable, TypeVar
692706

‎test-data/unit/check-dataclasses.test

+19-6
Original file line numberDiff line numberDiff line change
@@ -2527,16 +2527,29 @@ Gen(2).__replace__(x="not an int") # E: Argument "x" to "__replace__" of "Gen"
25272527
[builtins fixtures/tuple.pyi]
25282528

25292529
[case testDunderReplaceCovariantOverride]
2530-
# flags: --python-version 3.13
2530+
# flags: --python-version 3.13 --enable-error-code mutable-override
25312531
from dataclasses import dataclass
2532+
from typing import Optional
2533+
from typing_extensions import dataclass_transform
25322534

25332535
@dataclass
25342536
class Base:
2535-
a: object
2537+
a: Optional[int]
25362538

25372539
@dataclass
2538-
class Child(Base): # E: Argument 1 of "__replace__" is incompatible with supertype "Base"; supertype defines the argument type as "object" \
2539-
# N: This violates the Liskov substitution principle \
2540-
# N: See https://mypy.readthedocs.io/en/stable/common_issues.html#incompatible-overrides
2541-
a: int
2540+
class Child(Base):
2541+
a: int # E: Covariant override of a mutable attribute (base class "Base" defined the type as "Optional[int]", expression has type "int")
2542+
2543+
@dataclass
2544+
class Other(Base):
2545+
a: str # E: Incompatible types in assignment (expression has type "str", base class "Base" defined the type as "Optional[int]")
2546+
2547+
@dataclass_transform(kw_only_default=True)
2548+
class DCMeta(type): ...
2549+
2550+
class X(metaclass=DCMeta):
2551+
a: Optional[int]
2552+
2553+
class Y(X):
2554+
a: int # E: Covariant override of a mutable attribute (base class "X" defined the type as "Optional[int]", expression has type "int")
25422555
[builtins fixtures/tuple.pyi]

0 commit comments

Comments
 (0)