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

Unify typing for __path__? #6650

Open
cdce8p opened this issue Dec 21, 2021 · 1 comment
Open

Unify typing for __path__? #6650

cdce8p opened this issue Dec 21, 2021 · 1 comment

Comments

@cdce8p
Copy link
Contributor

cdce8p commented Dec 21, 2021

After the last mypy update v0.920, I started to notice an inconsistency with how the module variable __path__ is typed mainly across typeshed, mypy, and pyright.

Type Link
typeshed - ModuleType MutableSequence[str] #6200
typeshed - pkgutil.extend_path list[str] #5222
mypy list[str] python/mypy#9454
pyright Iterable[str] microsoft/pylance-release#1098

In the Python docs it's mentioned as:

`__path__` must be an iterable of strings, but it may be empty.

However the docs for pkgutil.extend_path go on say it's mostly a list (?) cpython/pkgutil.py

    If the input path is not a list (as is the case for frozen
    packages) it is returned unchanged.  The input path is not
    modified; an extended copy is returned.  Items are only appended
    to the copy at the end.

From my point of view, it isn't entirely clear what the "correct" type should be. Some points to consider though:

  • In most cases it seems to be a list[str].
  • It was mentioned here however that it can sometimes also be a MutableSequence[str]
  • The __path__ type should be valid as first argument to pkgutil.extend_path. At the moment that only applies to list[str] but obviously the argument type for extend_path could be changed to, for example, accept a TypeVar bound to Iterable[str] instead.
  • It seems to be fairly common (and safe ?) to access the at least the first item of __path__ via __getitem__.
@srittau
Copy link
Collaborator

srittau commented Dec 22, 2021

We should fix extend_path() to just match the implementation, i.e. something like

_P = TypeVar("_P", bound=Iterable[str])
def extend_path(path: _P, name: str) -> _P: ...

How we annotate ModuleType.__path__ is a harder question. (But we should definitely add a comment that it's only available for packages, not for plain modules.) We could use list[str], because that's the import system's default. But according to the comment in extend_path(), it can be another type for frozen packages. I guess our current solution of MutableSequence is also wrong for that case. According to the Python docs, we should use Iterable[str], but that prevents modifying the path when it actually is mutable. I don't have a good solution and not strong opinion, but a slight preference for Iterable[str].

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

No branches or pull requests

2 participants