-
Notifications
You must be signed in to change notification settings - Fork 92
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
Are type annotations worth having? #533
Comments
Definitely +10 on having type annotations, I've been using them in my own small projects for two years now or so and I love them. Only annoying things:
This is how I got it to render on my QUEST+ documentation: Cool thing is that you declare the type once, in the signature, and then don't have to repeat / remember it in the rest of the docstring, see e.g. |
I just remembered that the reason why in my QUEST+ code I switched back to an "Optional[dict]" annotation from something like |
... which probably means that I should just have introduced some new types :) |
We could also incorporate mypy in the CI? |
One step after the other ;) But yes, this might be useful sometime |
I have not much experience with type annotations, but just today I ran into a bug that could have been prevented with type annotations. (I was debugging for a long time until I noticed that at some point I had written from what it looks like, the benefit would be mostly for the developers, is that right? The documentation rendering seems slightly less appealing (but still good) ... can we still do things like I am curious to see this in action, and I don't see that it could hurt to have type annotations - apart from the initial extra work. edit:
what's the benefit of type annotations if we do not run a program (like mypy) that checks these annotations? --> is the only benefit then the "improved readability" of the code? |
Depends on what you consider a developer ;) Any user of MNE-BIDS that uses the library via Python code would benefit if their editor supports type annotations.
Wait until you see this: 💩😅
So far I've only used sphinx-autodoc-typehints and this wouldn't support this out of the box. You'd have to use
Any good IDE (e.g. PyCharm, VS Code with pylance) will tell you upon typing a line of code that you're about to submit a value of the wrong type to a function. Doesn't always work, of course, because sometimes types have to be inferred; but I've had good experiences with this so far. |
that's really neat, thanks for the example! but it seems that the state of adoption in big projects is still very early. issue for support in numpydoc: numpy/numpydoc#196 |
working in Python for > 10 years I've never used it ;)
… |
It's relatively new, pops, you're too old and wouldn't understand! It's not a phase! ;) No but seriously, I think it can be super helpful and help catch or prevent bugs. I also switched my JavaScript code to TypeScript, which uses similar type declarations, and it's really worth the initial pain of documenting the interfaces. |
Things are really still in the process of evolving. For example, NumPy just recently added "type stubs" to their main repository, which help static Python type checkers to know about the large set of compiled C functions they use. For pandas, this still seems to be kind of work-in-progress. And then there is the data-science-types project, aiming to generate stubs for NumPy, SciPy, and Matplotlib. Also WIP :) So I believe we don't need to rush anything here; however, I get the impression that the scientific Python ecosystem is definitely moving in the direction of using type hints. As it's easier to start early adding type hints to a small / growing code base than to an old, fuzzy, big code base, I think it's worth our while to seriously consider adding type hints. :) |
I've been working on this a bit and I think my main pitch for why we must have this is basically what is shown in the below screencast: this is one of our functions whose return type depends on the value of an input parameter: Screen.Recording.2022-06-12.at.19.45.25.movI will try to finalize most type annotations within the next two weeks or so, and try to get this merged into typeshed (I already got in touch with them and asked for permission to include default parameter values, which is normally against their rules, but they said they might make an exception for us!), so all users of VS Code with Pylance will automatically profit from it. Will ping interested parties once I think it's ready for testing. The beauty of this approach is that type annotations remain entirely optional and can live outside this project's repository until we've reached a consensus. |
To be honest, although this is great and all, I don't really trust Pylance. It's not because Pylance per se, but more often than not Python code is not properly annotated, so it cannot infer the type. Now of course your argument is that we will properly annotate (externally at the moment), but I can't and won't trust Pylance unless all type suggestions are 100%. And this is never going to happen, so I'd rather look at the source code (docstring in most cases). |
These days, basically all "main" scientific Python packages are extensively annotated (NumPy, SciPy, Pandas) But the good thing for you is that you can always just disable type checking and rely on the (also sometimes incomplete and buggy) docstrings instead 😅 |
In principle yes, but someone has to make sure that our annotations are always in sync with the code base. I can only imagine this happening when annotations live directly in our code and not in an external package. In a smaller package like this one, I'd be okay with including annotations (yes, you read that correctly). I already maintain a package which has type annotations (https://github.com/cbrnr/sleepecg), and it's not too bad (although they never helped me in any way either). It would be something else to decide whether or not we should annotate in MNE-Python, which is a much larger package with a ton of existing code, types, functions, modoules, etc. |
Typeshed does that automatically, they've set up a rather sophisticated system to my understanding, which regularly checks whether the annotations and the upstream code base are still in sync. That's why I'd definitely want to get these annotations into typeshed if we don't want to annotate the code base itself. Otherwise everything will get messy and out of sync real quick.
Hehe I saw that already a while ago and my first thought was, "damn, that must've been hard for Clemens to reach a consensus here!" 😅 I find it interesting you say it never really helped you – did you enable strict type checking in your IDE? Or just "basic" checks? I first came across type annotations & checking when working with TypeScript a few years back, and this is how I learned to appreciate this. 👍 But then again, my (working) memory doesn't seem to be all that great, so I tend to forget about those little details like varying return types a lot, and a type checker can be of great help to me there. |
If typeshed takes care of syncing that's great! One less thing to worry about.
I'm not that stubborn 😄 – the project was mainly implemented by a master's student, and he wanted to have annotations. Simple as that.
I have to enable something? I thought Pylance is just doing its thing in the background? I never actively enabled something. |
Oh wow, this was a Master's project? Super sophisticated!!! I'm genuinely impressed, I thought it's something on a PhD level.
Well yes 😂 I believe the default is "off" for type checking! The setting you'll want to check is But even at |
I enabled basic type checking and opened SleepECG only to find stuff like this: filename = filename + '.csv'
That's exactly the nonsense I am not willing to deal with. Maybe I did not annotate correctly (I don't think so), but I want to continue programming Python and not something that approaches Java. |
@cbrnr What this tells you is: "Hey, filename can in some circumstances be a Path, and a Path cannot be +'d with another string!" This is exactly the kind of bug that type checking is meant to uncover! This is precisely what I'm talking about when I say, it helps me prevent several bugs a day :) |
Btw there is a way to work around this: you could use a "type guard" like: assert isinstance(filename, str)
filename = filename + '.csv' shouldn't spawn the error, right? Or like this: if isinstance(filename, str):
filename += '.csv'
else: # it's a Path
filename = filename.with_suffix('.csv') Edit: incorrect code |
You still have duck typing, and you already need to know whether you're dealing with a string or a Path, so the only thing that type annotations & type checkers do is provide some additional help, is all… |
Alright, I really thought that you can + a Path and a str! In that case, you are absolutely correct! I never checked this myself because I assumed that the person who wanted to use type annotations would also use these annotations (i.e. turn on basic checks). I might have to revise my opinion on type annotations. As long as everything stays optional, I think this could be useful indeed. |
Wow! I never expected this! 😄 My mind is totally blown right now hahaha.
Yep that's what I believe too! I also sometimes turn off all type checking when working with code bases that are either not or just poorly annotated, as the amount of red squiggly lines due to false alarms is driving me nuts (this is actually sometimes the case with MNE-Python). So it's super nice we are free to choose, depending on our preferences and requirements! |
😆 – as I mentioned previously: I am not that stubborn! Thanks for bearing with me and showing me my first very own example where type annotations really help! |
I just encountered a case where I think the error is actually a fluke (sorry this is OT, please let me know if you would be available e.g. on Discord for 10 minutes to help me go over the errors that I find weird): The
Further down, I have an |
@cbrnr I'm quite busy right now, but a) yes there are sometimes bugs like these in Pylance & pyright, which get fixed once reported, and b) sometimes one overlooks something, so it might not necessarily always be a bug in Pylance Like I said I'm a bit busy, but maybe you can condense this code down to a MWE (edit: i just saw you linked to the original code, so I guess I'll take a look at this) I'd be happy to help get this stuff sorted out. Maybe you can make me a list of these issues and I can try to dig into this once I have time again? (maybe tonight, maybe in a few days, cannot promise anything!) |
Sure, I will try to solve these things by myself as part of cbrnr/sleepecg#93, so anything I can't fix or I have question I will post there (and tag you in case you have time to take a look). Yes, in this particular example I've linked to the code, if you know how to best deal with it please LMK (and if you don't have time just don't respond, that's perfectly OK). |
Cool! And like I said, be prepared to get bitten by bugs in Pylance itself once in a while – see this list of user-reported bugs that have been fixed over the course of the past couple of months alone: and a similarly long list exists for pyright, which is the type checker pylance is built upon… Edit: Fixed URL |
It's these lines that make it difficult for Pylance to keep track: if backend not in _available_backends:
fallback = _available_backends[0]
warnings.warn(f'Backend {backend!r} not available, using {fallback!r} instead.')
backend = fallback I assume it's because I would think this is a bug or at least an annoying limitation in Pylance, and maybe we could turn this thing into a MWE and report it to the devs. For now, this (ugly) workaround fixes the problem for me: I'm adding an if backend not in _available_backends:
fallback = _available_backends[0]
warnings.warn(f'Backend {backend!r} not available, using {fallback!r} instead.')
backend = fallback
assert backend in _all_backends # <-- ugly but seems to do the trick |
@cbrnr Ok I found a cleaner solution that you're probably going to hate even more :) At the top of your module, where you define _all_backends: tuple[
Literal['c'], Literal['numba'], Literal['python']
] = ('c', 'numba', 'python') Then Pylance is able to deduce that the line that follows, _available_backends = list(_all_backends) will create an _all_backends = ('c', 'numba', 'python')
_available_backends = list(_all_backends) Pylance says that I think we might have a MWE here, now, eh? Pylance should know that converting a tuple of strings to a list can only create a list of the exact same strings, and nothing else… So I'd label this a bug!! And an annoying one too! |
@cbrnr Unfortunately maybe not a bug, but a feature (??!!)
Produces: ❯ mypy type_test.py
type_test.py:5: note: Revealed type is "Tuple[builtins.str, builtins.str, builtins.str]"
type_test.py:6: note: Revealed type is "builtins.list[builtins.str]"
Success: no issues found in 1 source file So MyPy itself isn't even narrowing down the content of the tuple to the explicit strings we're assigning. This seems to be a Pylance-specific feature (?) So now I'm unsure whether we should ask the Pylance or the MyPy folks for advice. |
But even with the following adjustment, # %%
from typing import Literal
x: tuple[
Literal['foo'],
Literal['bar'],
Literal['baz']
] = ('foo', 'bar', 'baz')
y = list(x)
reveal_type(x)
reveal_type(y) ❯ mypy type_test.py
type_test.py:11: note: Revealed type is "Tuple[Literal['foo'], Literal['bar'], Literal['baz']]"
type_test.py:12: note: Revealed type is "builtins.list[builtins.str]"
Success: no issues found in 1 source file |
… stupid typing!! 😡 |
@cbrnr If you want I can ask on the MyPy Gitter chat. They've been very helpful in the past :) |
OMG – I just wanted to write an elaborate response (I had actually already written but then deleted it), but the TLDR is that this exactly shows why I do not want to use type annotations. Thanks for digging into this issue, I guess I'll just turn off type checking again. It's better for my sanity. |
Hehe you do that As for me – all hope is lost, there's not much sanity to lose anymore, so I'm just going to go all-in and ask the MyPy folks for help. Will keep you posted ;) |
@cbrnr Meanwhile, I got in touch with the folks from the In the meantime, I have a super quick & dirty solution for annoying lines like the one above: return np.where(beat_mask)[0] # type: ignore Tada! ;) |
Can you also do that ignore globally 😅? |
I am silent here as I am on the side of the "old" folks here who have never
used it
and just see this as a lot of characters to type for little gain...
… Message ID: ***@***.***>
|
I see what you did there 🤓 In fact, you can: |
Fewer bugs if you have to rely on young folks like me writing code 😅😅😅 |
Fewer bugs if you have to rely on young folks like me writing code 😅😅😅
it forces the old folks to review the code and not get old and useless too
fast ;)
… Message ID: ***@***.***>
|
tackling this would potentially entail: |
Should we have type annotations within the code base, per @hoechenberger comments here: #532 (review)?
The text was updated successfully, but these errors were encountered: