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

Fixed required/optional keys with old-style TypedDict #778

Merged
merged 3 commits into from
Apr 11, 2021
Merged

Fixed required/optional keys with old-style TypedDict #778

merged 3 commits into from
Apr 11, 2021

Conversation

domdfcoding
Copy link
Contributor

Fixes #761

Copied from python/cpython#22736

Co-authored-by: Alex Grönholm [email protected]

try:
# Setting correct module is necessary to make typed dict classes pickleable.
ns['__module__'] = sys._getframe(1).f_globals.get('__name__', '__main__')
except (AttributeError, ValueError):
pass

return _TypedDictMeta(typename, (), ns)
return _TypedDictMeta(typename, (), ns, total=total)
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

On Python 3.4 and 3.5 it seems that _TypedDictMeta(...) calls __init__ rather than __new__, but __init__ isn't implemented. See, for instance: https://travis-ci.com/github/python/typing/jobs/473002561#L338

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That looks nasty. I wonder if it's related to this note in https://docs.python.org/3/whatsnew/3.6.html#changes-in-the-python-api :

As part of PEP 487, the handling of keyword arguments passed to type (other than the metaclass hint, metaclass) is now consistently delegated to object.init_subclass(). This means that type.new() and type.init() both now accept arbitrary keyword arguments, but object.init_subclass() (which is called from type.new()) will reject them by default. Custom metaclasses accepting additional keyword arguments will need to adjust their calls to type.new() (whether direct or via super) accordingly.

I just tried and failed to get pyenv to compile 3.5 for me so I can't try this out myself :( If it's too hard to get this to work I wouldn't be opposed to fixing it for 3.6+ only for now.

@domdfcoding
Copy link
Contributor Author

There aren't any tests set up on Travis for Python 3.9, but for me they fail locally as python/cpython#22736 didn't make it into 3.9.1.
Should I change the check to only use the stdlib for >=3.9.2?

if sys.version_info[:2] >= (3, 9):

Copy link
Member

@JelleZijlstra JelleZijlstra left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry for the long wait here! I just merged a PR adding 3.9.0 to CI. I think it's OK to make typing_extensions.TypedDict not an alias for typing.TypedDict in 3.9.0/1, so that we can provide the fixed version of TypedDict here.

@domdfcoding
Copy link
Contributor Author

Right, I think I've sorted it.

On Python 3.9.2 and above typing.TypedDict is used, otherwise it's typing_extensions.TypedDict.

For Python 3.4 and 3.5 it is necessary to implement __init__ on _TypedDictMeta, as the "total" keyword gets passed to that method. 3.6+ accept arbitrary keyword arguments for type.__init__ but earlier versions don't.

@domdfcoding domdfcoding marked this pull request as ready for review April 11, 2021 19:54
@JelleZijlstra JelleZijlstra merged commit 40932e3 into python:master Apr 11, 2021
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

TypedDict(...) as function does not respect "total" when setting __required_keys__ and __optional_keys__
3 participants