Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add @typing.override #78

Merged
merged 7 commits into from
Oct 6, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Jump to
Jump to file
Failed to load files.
Diff view
Diff view
3 changes: 3 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,7 +1,10 @@
# Release 4.4.0 ()

- Add `typing_extensions.Any` a backport of python 3.11's Any class which is
subclassable at runtime. (backport from python/cpython#31841, by Shantanu
and Jelle Zijlstra). Patch by James Hilton-Balfe (@Gobot1234).
- Runtime support for PEP 698, adding `typing_extensions.override`. Patch by
Jelle Zijlstra.

# Release 4.3.0 (July 1, 2022)

Expand Down
15 changes: 15 additions & 0 deletions src/test_typing_extensions.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@
from typing_extensions import assert_type, get_type_hints, get_origin, get_args
from typing_extensions import clear_overloads, get_overloads, overload
from typing_extensions import NamedTuple
from typing_extensions import override
from _typed_dict_test_helper import FooGeneric

# Flags used to mark tests that only apply after a specific
Expand Down Expand Up @@ -160,6 +161,20 @@ def test_exception(self):
assert_never(None)


class OverrideTests(BaseTestCase):
def test_override(self):
class Base:
def foo(self): ...

class Derived(Base):
@override
def foo(self):
return 42

self.assertIsSubclass(Derived, Base)
self.assertEqual(Derived().foo(), 42)
JelleZijlstra marked this conversation as resolved.
Show resolved Hide resolved


class AnyTests(BaseTestCase):
def test_can_subclass(self):
class Mock(Any): pass
Expand Down
29 changes: 29 additions & 0 deletions src/typing_extensions.py
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,7 @@
'Literal',
'NewType',
'overload',
'override',
'Protocol',
'reveal_type',
'runtime',
Expand Down Expand Up @@ -2000,6 +2001,34 @@ def decorator(cls_or_fn):
return decorator


if hasattr(typing, "override"):
override = typing.override
else:
_T = typing.TypeVar("_T", bound=typing.Callable[..., typing.Any])
JelleZijlstra marked this conversation as resolved.
Show resolved Hide resolved

def override(__arg: _T) -> _T:
"""Indicate that a method overrides a method in a base class.
JelleZijlstra marked this conversation as resolved.
Show resolved Hide resolved

Usage:

class Base:
def method(self) -> None: ...
pass

class Child(Base):
@override
def method(self) -> None:
super().method()

When this decorator is applied to a method, the type checker will
validate that it overrides a method with the same name on a base class.
This helps prevent bugs that may occur when the base class is changed
without an equivalent change to a child class.
JelleZijlstra marked this conversation as resolved.
Show resolved Hide resolved

"""
return __arg


# We have to do some monkey patching to deal with the dual nature of
# Unpack/TypeVarTuple:
# - We want Unpack to be a kind of TypeVar so it gets accepted in
Expand Down