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 type hints for the zoneinfo module #4038

Merged
merged 14 commits into from
May 18, 2020
Merged

Conversation

pganssle
Copy link
Member

This is the implementation for PEP 615: https://www.python.org/dev/peps/pep-0615/

It is present starting in 3.9.0 beta 1.

Please let me know if you want these structured in a different way, or if you want me to try and migrate this smoke test module I use to test these things in the reference implementation.

This is the implementation for PEP 615: https://www.python.org/dev/peps/pep-0615/

It is present starting in 3.9.0 beta 1.
Copy link
Collaborator

@srittau srittau left a comment

Choose a reason for hiding this comment

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

Thank you for the PR, but also for the PEP and implementation, this will be very helpful! A few remarks:

Why didn't you include ZoneInfoNotFoundError and InvalidTZPathWarning?

from typing import IO, Optional, Sequence, Type, Union

class ZoneInfo(tzinfo):
_T = typing.TypeVar("_T", bound="ZoneInfo")
Copy link
Collaborator

Choose a reason for hiding this comment

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

Typevars in typeshed are defined at top-level. This is possibly also why pytype chokes.

# a sequence of strings is required. This should be remedied if a solution
# to this typing bug is found: https://github.com/python/typing/issues/256
def reset_tzpath(
to: Optional[Sequence[Union[os.PathLike, str]]] = None
Copy link
Collaborator

Choose a reason for hiding this comment

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

Suggested change
to: Optional[Sequence[Union[os.PathLike, str]]] = None
to: Optional[Iterable[Union[os.PathLike, str]]] = ...

Copy link
Member Author

Choose a reason for hiding this comment

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

The PEP specifies that this is a Sequence rather than an Iterable: https://www.python.org/dev/peps/pep-0615/#reset-tzpath-function

(clear_cache does specify an Iterable).

I don't think it's critical but I'd rather start strict and lessen the constraints later if necessary rather than the other way around. (I think we maybe should make reset_tzpath reject anything without a length before the next beta release, so that this is also enforced at runtime).

Copy link
Member Author

Choose a reason for hiding this comment

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

Oh, one question: why the = ... instead of = None?

Copy link
Collaborator

Choose a reason for hiding this comment

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

Your reasoning makes sense.

Defaults don't matter for type checking. For consistency's sake, flake8-pyi enforces a default of .... It's one less thing to check and maintain in stubs.

def reset_tzpath(
to: Optional[Sequence[Union[os.PathLike, str]]] = None
) -> None: ...
def available_timezones() -> typing.Set[str]: ...
Copy link
Collaborator

Choose a reason for hiding this comment

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

Why not just Set (without typing) like in the other cases?

@srittau
Copy link
Collaborator

srittau commented May 18, 2020

There is one important thing I forgot. The IO classes are problematic, which is why we try not to use them in new annotations, but prefer to use specific protocols. In the case of __fobj, this could be something like:

class _Readable(Protocol):
    def read(self, __size: int) -> bytes: ...
    def seek(self, __size: int, __where: int) -> Any: ...

Although I did not check all places where this was used.

@pganssle
Copy link
Member Author

I like the idea of a specific protocol, but doesn't defining the minimal required subset of the IO interface expose some implementation details (thus constraining the evolution of the implementation)? It would mean that the stubs would need to be updated in the event that I use fobj.tell().

I suppose a reasonable compromise would be to just add tell() — even that I'm not likely to use, but everything else I'm really not likely to use.

@srittau
Copy link
Collaborator

srittau commented May 18, 2020

Yes, this is always the drawback of using specific protocols. In typeshed the decision was made (which I only partly agree with) to orient ourselves on the actual implementation. That said, if it is likely you will use tell() in the future in a new minor version, it makes sense to include it.

The underlying problem is that the concept of "file-like" is underspecified in Python. Often it just means "has a read() method", although it can mean different things as well. (I actually plan to categorize often used IO protocols in the stdlib and have a set of "standard" protocols at some point. See also python/typing#564.) In this specific case, since you are the author of the PEP and the implementation, I'd say specify the protocol as you consider it useful.

Unfortunately I hadn't read the PEP in depth. I like that you used type annotations in it, but I'd have counseled against using IO. :)

Edit: Also if you prefer to keep IO, this is fine as well, despite my quest to get rid of it.

@pganssle
Copy link
Member Author

No worries. If you have a succinct replacement in the PEP (or if you'd prefer to add a footnote), we can still make little adjustments to the PEP here and there.

I don't think it's amazingly likely that we'll use tell(), I'll just remove it and hopefully at some point we can get something like python/typing#564.

It seems that the pytype job is choking on the from __future__ import annotations statement here. Any suggestion on how to handle that? Can we just skip this module on pytype until the pytype job is running on at least 3.7?

@rchen152
Copy link
Collaborator

Is from __future__ import annotations even necessary in a stub? If it is, you can add a version check a la

if sys.version_info >= (3, 7):
.
(It does seem a bit silly given that the module won't even exist until 3.9, but as far as I know, there's no way to indicate that.)

@srittau
Copy link
Collaborator

srittau commented May 18, 2020

Good point, the import is not necessary, stubs always support forward references.

@srittau srittau merged commit 6fc8828 into python:master May 18, 2020
vishalkuo pushed a commit to vishalkuo/typeshed that referenced this pull request Jun 26, 2020
This is the implementation for PEP 615: https://www.python.org/dev/peps/pep-0615/

It is present starting in 3.9.0 beta 1.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants