Reputation: 1254
I am trying the typing hint introduced by Python 3.5 and got a problem by using local stubs as the typing hint with mypy.
The experiment I do is to create kk.py containing
def type_check(a):
pass
Also, I put kk.pyi containing
def type_check(a: int):...
in the same directory. In this way, I tried to trigger the error of "incompatible types in assignment" by passing a string to type_check
in kk.py
. However, when I ran mypy kk.py
, I get no error.
Thus I tried another way that mypy doc suggests, which is to set environment variable MYPYPATH to ~/some/path/stub
and put kk.pyi
in the directory. I didn't get any error again.
Anyone can help me on this?
Here is the mypy wiki on how to use a local stub.
Upvotes: 19
Views: 4843
Reputation: 16526
As the accepted answer mentioned, unfortunately Mypy doesn't use the X.pyi
stub file against X.py
itself when you run mypy X.py
.
This issue is addressed here as well:
https://github.com/python/mypy/issues/5028
The only solution I found to check the module itself is using mypy.stubtest
:
A common problem with stub files is that they tend to diverge from the actual implementation. Mypy includes the stubtest tool that can automatically check for discrepancies between the stubs and the implementation at runtime.
The usage is simple, Just run python -m mypy.stubtest X
(X is the name of the module without .py
extension)
Example:
$ python3 -m pip install mypy
$ cat library.py
x = "hello, stubtest"
def foo(x=None):
print(x)
$ cat library.pyi
x: int
def foo(x: int) -> None: ...
$ python3 -m mypy.stubtest library
error: library.foo is inconsistent, runtime argument "x" has a default value but stub argument does not
Stub: at line 3
def (x: builtins.int)
Runtime: at line 3 in file ~/library.py
def (x=None)
error: library.x variable differs from runtime type Literal['hello, stubtest']
Stub: at line 1
builtins.int
Runtime:
hello, stubtest
Upvotes: 2
Reputation: 1254
I do not know why someone have voted down this question without answering it or commenting about why he/she disliked it, but here is the answer I figured out:
The stub file of mypy only works when importing a module. Thus, if you have
def try_check(a):
pass
in kk.py, and
def try_check(a: int):...
in kk.pyi in the same directory with kk.py or in the directory that the MYPYPATH specifies, mypy will type check the python file if you import kk. It is, if you have
import .kk
kk.try_check('str')
in test.py and run mypy test.py, mypy will report the type conflict. However, it will not report the conflict if you have
try_check('str')
in kk.py.
You can type check functions in the program that contains the function definition If you write the typing hint explicitly in the definition of the function. For instance, you can write
def try_check(a: int):
pass
try_check('str')
in kk.py and then mypy kk.py. Mypy will report the type conflict.
Upvotes: 19