msg373360 - (view) |
Author: Keith Blaha (keithblaha) |
Date: 2020-07-08 23:47 |
Copied from https://github.com/python/typing/issues/737
I came across this issue while using inheritance to express required keys in a TypedDict, as is recommended by the docs.
It's most easily explained by a minimal example I cooked up. Let's say we have a module foo.py:
from __future__ import annotations
from typing import Optional
from typing_extensions import TypedDict
class Foo(TypedDict):
a: Optional[int]
And another module bar.py:
from __future__ import annotations
from typing import get_type_hints
from foo import Foo
class Bar(Foo, total=False):
b: int
print(get_type_hints(Bar))
Note that both foo.py and bar.py have adopted postponed evaluation of annotations (PEP 563) by using the __future__ import.
If we execute bar.py, we get the error message NameError: name 'Optional' is not defined.
This is due to the combination of:
get_type_hints relies on the MRO to resolve types: https://github.com/python/cpython/blob/3.7/Lib/typing.py#L970
TypedDict does not preserve the original bases, so Foo is not in the MRO for Bar:
typing/typing_extensions/src_py3/typing_extensions.py
Line 1652 in d79edde
tp_dict = super(_TypedDictMeta, cls).__new__(cls, name, (dict,), ns)
Thus, get_type_hints is unable to resolve the types for annotations that are only imported in foo.py.
I ran this example using typing_extensions 3.7.4.2 (released via #709) and Python 3.7.3, but it seems like this would be an issue using the current main branches of both repositories as well.
I'm wondering what the right approach is to tackling this issue. It is of course solvable by defining Bar in foo.py instead, but it isn't ideal or intuitive to always need to inherit from a TypedDict in the same module.
I was thinking that similarly to __required_keys__ and __optional_keys__, the TypedDict could preserve its original bases in a new dunder attribute, and get_type_hints could work off of that instead of MRO when it is dealing with a TypedDict. I would be willing to contribute the PRs to implement this if the design is acceptable, but am open to other ideas as well.
|
msg373752 - (view) |
Author: Ivan Levkivskyi (levkivskyi) * |
Date: 2020-07-16 10:12 |
> I was thinking that similarly to __required_keys__ and __optional_keys__, the TypedDict could preserve its original bases in a new dunder attribute, and get_type_hints could work off of that instead of MRO when it is dealing with a TypedDict. I would be willing to contribute the PRs to implement this if the design is acceptable
TBH this is not very elegant, but I think you can go ahead with this (at least as a quick fix) since I don't see a better solution yet.
|
msg373758 - (view) |
Author: Keith Blaha (keithblaha) |
Date: 2020-07-16 17:56 |
> TBH this is not very elegant, but I think you can go ahead with this (at least as a quick fix) since I don't see a better solution yet.
Agreed, given that the current workaround of implementing them in the same module works I think I will stick with that while this is brainstormed further.
|
msg394729 - (view) |
Author: Nils Kattenbeck (Nils Kattenbeck) * |
Date: 2021-05-29 10:52 |
What is/was the initial reason to not preserve the MRO for a TypedDict?
The only thing which came to my mind would be instantiation performance but as annotations are not evaluated by default and on the right-hide side of assignment most people will use dict literals I am not sure if this is still relevant. Otherwise it might even be simpler to just preserve the original bases in TypedDict but please correct me if I overlooked something
|
msg394738 - (view) |
Author: Guido van Rossum (gvanrossum) * |
Date: 2021-05-29 21:09 |
> What is/was the initial reason to not preserve the MRO for a TypedDict?
Hm, I can't say for sure. I believe it had something to do with TypedDict instances being instances of dict at runtime, but I can't actually reconstruct the reason. Maybe it's written up in PEP 589, but I suspect not (I skimmed and couldn't find it). If you ask on typing-sig maybe David Foster (who contributed the initial idea and implementation) remembers.
|
msg395279 - (view) |
Author: Nils Kattenbeck (Nils Kattenbeck) * |
Date: 2021-06-07 19:36 |
> I believe it had something to do with TypedDict instances being instances of dict at runtime, but I can't actually reconstruct the reason.
Hm that may be true.
My limited low-level Python knowledge leads me to believe that this could also be done using __new__ but I also read that most magic methods get called as type(Foo).__magic__(bar, ...) so that might not be possible.
(However also no methods can be declared on a TypedDict class so that might not be a problem?)
> Maybe it's written up in PEP 589, but I suspect not (I skimmed and couldn't find it).
I read it completely and could not find anything
> If you ask on typing-sig maybe David Foster (who contributed the initial idea and implementation) remembers.
I asked [here on typing-sig](https://mail.python.org/archives/list/typing-sig@python.org/thread/RNFWPRLHTUTZES2FDSSMY472JFGMD4EW/) but did not yet get any responses.
|
msg396296 - (view) |
Author: Guido van Rossum (gvanrossum) * |
Date: 2021-06-21 22:59 |
Note that this issue is now only a problem of you use `from __future__ import annotations` -- we rolled the default behavior for 3.10 back to what it was in 3.9.
I am out of time to argue about why we chose this behavior, alas.
|
msg396919 - (view) |
Author: Germán Méndez Bravo (Kronuz) * |
Date: 2021-07-03 18:01 |
The way I fixed this is I added `__forward_module__` to `typing.ForwardRef`, so that it can resolve the forward reference with the same globals as the ones specified by the module in `__forward_module__`. `TypedDict`'s metaclass should then pass the dictionary’s module name to the annotations’ forward references via the added `module`’s keyword argument in `typing._type_check()`. I can work in a pull request with this solution and discuss any potential problems.
|
msg396923 - (view) |
Author: Nils Kattenbeck (Nils Kattenbeck) * |
Date: 2021-07-03 20:42 |
> The way I fixed this is I added `__forward_module__` to `typing.ForwardRef`, so that it can resolve the forward reference with the same globals as the ones specified by the module in `__forward_module__`. `TypedDict`'s metaclass should then pass the dictionary’s module name to the annotations’ forward references via the added `module`’s keyword argument in `typing._type_check()`. I can work in a pull request with this solution and discuss any potential problems.
While this seems like a good solution I would still like to figure out why TypedDict do not preserve MRO. Because for now I have not found a reason nor did someone on the typing-sig mailinglist have a clue. Should there (no longer) be a reason for this then this problem has a trivial solution (just re-add the MRO and use that).
|
msg396926 - (view) |
Author: Germán Méndez Bravo (Kronuz) * |
Date: 2021-07-03 21:47 |
Nils, unfortunately, fixing the MRO here won’t fix the issue because `TypedDict.__annotations__` in the class copies the annotations from the parent classes, and when the type evaluation is made, it’s made using the copied annotation found in the bottommost class (which is thus then expected to be a forward reference in the same module as the class that inherited them. This producing the exact same problem of missing type.
The most likely reason for incomplete MRO is that `TypeDict` extends a class’ `__annotations__` with all of it’s parent’s `__annotations__`, so the final class has the complete set of annotations of all of its parents, so having the full inheritance chain made less sense, after all the final dictionary class has all the annotations.
> On Jul 3, 2021, at 13:43, Nils Kattenbeck <report@bugs.python.org> wrote:
>
>
> Nils Kattenbeck <nilskemail@gmail.com> added the comment:
>
>> The way I fixed this is I added `__forward_module__` to `typing.ForwardRef`, so that it can resolve the forward reference with the same globals as the ones specified by the module in `__forward_module__`. `TypedDict`'s metaclass should then pass the dictionary’s module name to the annotations’ forward references via the added `module`’s keyword argument in `typing._type_check()`. I can work in a pull request with this solution and discuss any potential problems.
>
> While this seems like a good solution I would still like to figure out why TypedDict do not preserve MRO. Because for now I have not found a reason nor did someone on the typing-sig mailinglist have a clue. Should there (no longer) be a reason for this then this problem has a trivial solution (just re-add the MRO and use that).
>
> ----------
>
> _______________________________________
> Python tracker <report@bugs.python.org>
> <https://bugs.python.org/issue41249>
> _______________________________________
|
msg396946 - (view) |
Author: Germán Méndez Bravo (Kronuz) * |
Date: 2021-07-04 16:49 |
I added a pull request with my fix here:
https://github.com/python/cpython/pull/27017
|
msg397690 - (view) |
Author: Guido van Rossum (gvanrossum) * |
Date: 2021-07-17 03:49 |
New changeset 889036f7ef7290ef15b6c3373023f6a35387af0c by Germán Méndez Bravo in branch 'main':
bpo-41249: Fix postponed annotations for TypedDict (GH-27017)
https://github.com/python/cpython/commit/889036f7ef7290ef15b6c3373023f6a35387af0c
|
msg397692 - (view) |
Author: Guido van Rossum (gvanrossum) * |
Date: 2021-07-17 03:50 |
How far can/should we backport this?
|
msg397694 - (view) |
Author: Ken Jin (kj) * |
Date: 2021-07-17 03:58 |
> How far can/should we backport this?
It will work in 3.10 and 3.9 without issues. However, I don't remember if bugfixes for __future__ features require special treatment/are excluded from normal bugfix backports. I vaguely remember us not backporting from __future__ annotations very far back (since they usually broke backwards compatibility).
Maybe 3.10 is enough?
|
msg397695 - (view) |
Author: Guido van Rossum (gvanrossum) * |
Date: 2021-07-17 04:03 |
Let’s both, since this feels like a real bug fix to me.
|
msg397710 - (view) |
Author: Łukasz Langa (lukasz.langa) * |
Date: 2021-07-17 08:48 |
New changeset 480f29f913cff30329e7b425fd6669f83d6d8af8 by Miss Islington (bot) in branch '3.10':
bpo-41249: Fix postponed annotations for TypedDict (GH-27017) (#27204)
https://github.com/python/cpython/commit/480f29f913cff30329e7b425fd6669f83d6d8af8
|
msg397711 - (view) |
Author: Łukasz Langa (lukasz.langa) * |
Date: 2021-07-17 09:36 |
New changeset fa674bdea3bbb20ad6ccd95b3849fc4995bc37e0 by Ken Jin in branch '3.9':
[3.9] bpo-41249: Fix postponed annotations for TypedDict (GH-27017) (GH-27205)
https://github.com/python/cpython/commit/fa674bdea3bbb20ad6ccd95b3849fc4995bc37e0
|
msg397712 - (view) |
Author: Łukasz Langa (lukasz.langa) * |
Date: 2021-07-17 09:37 |
Thanks! ✨ 🍰 ✨
|
|
Date |
User |
Action |
Args |
2022-04-11 14:59:33 | admin | set | github: 85421 |
2021-07-17 09:37:32 | lukasz.langa | set | status: open -> closed resolution: fixed messages:
+ msg397712
stage: patch review -> resolved |
2021-07-17 09:36:42 | lukasz.langa | set | messages:
+ msg397711 |
2021-07-17 08:48:23 | lukasz.langa | set | nosy:
+ lukasz.langa messages:
+ msg397710
|
2021-07-17 04:20:20 | kj | set | pull_requests:
+ pull_request25739 |
2021-07-17 04:14:25 | miss-islington | set | nosy:
+ miss-islington pull_requests:
+ pull_request25738
|
2021-07-17 04:03:21 | gvanrossum | set | messages:
+ msg397695 |
2021-07-17 03:58:39 | kj | set | messages:
+ msg397694 |
2021-07-17 03:50:44 | gvanrossum | set | messages:
+ msg397692 versions:
+ Python 3.11 |
2021-07-17 03:49:34 | gvanrossum | set | messages:
+ msg397690 |
2021-07-04 16:49:59 | Kronuz | set | messages:
+ msg396946 |
2021-07-04 16:49:12 | Kronuz | set | keywords:
+ patch stage: patch review pull_requests:
+ pull_request25576 |
2021-07-03 21:47:21 | Kronuz | set | messages:
+ msg396926 |
2021-07-03 20:42:57 | Nils Kattenbeck | set | messages:
+ msg396923 |
2021-07-03 18:01:04 | Kronuz | set | nosy:
+ Kronuz messages:
+ msg396919
|
2021-06-21 22:59:20 | gvanrossum | set | messages:
+ msg396296 |
2021-06-07 19:36:01 | Nils Kattenbeck | set | messages:
+ msg395279 |
2021-05-30 07:13:31 | kj | set | nosy:
+ kj
|
2021-05-29 21:09:42 | gvanrossum | set | messages:
+ msg394738 |
2021-05-29 10:52:53 | Nils Kattenbeck | set | nosy:
+ Nils Kattenbeck messages:
+ msg394729
|
2020-07-16 17:56:36 | keithblaha | set | messages:
+ msg373758 |
2020-07-16 10:12:25 | levkivskyi | set | messages:
+ msg373752 |
2020-07-08 23:51:53 | gvanrossum | set | nosy:
+ gvanrossum, levkivskyi
|
2020-07-08 23:47:38 | keithblaha | create | |