-
Notifications
You must be signed in to change notification settings - Fork 867
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
Language Server does not find extension attribute: Cannot access member "Meta" for type "Markdown" Member "Meta" is unknown
#1383
Comments
I can't replicate this. In fact, notice below that I have not even called >>> md = markdown.Markdown(extensions=['meta'])
>>> md.Meta
{}
>>> markdown.__version__
'3.4.4' |
I don't think the argument is that As Python Markdown does not provide type annotations, and any that are available are provided by 3rd parties, I'm not sure if this is an issue for Python Markdown. If we were currently providing type annotations, then maybe it could be argued we should do something, but currently, this is not a typed library, even if someone else has provided typing that you are using in conjunction with your LSP. |
Now I have reread your report and you note that things are working. But apparently some tool is inspecting the code and not finding the attribute. That is correct. Many extensions add their own attributes (most of them being third-party) and there is no way that we could possible anticipate or stub every possible attribute added by every extension. So, no we will not be giving this one extension special treatment. That said, if anyone has suggestions for a general purpose solution, I am willing to consider it. |
That will no longer be true with the next release. See #1379. BTW, @facelessuser I have a question for you in my last comment there. Based on your comments here perhaps you've been ignoring that one and missed it. |
Well, you are documenting the API, yes, but there are no type annotations. I'm fairly certain that is what LSP uses, not function doc strings. At least that is my understanding.
I have been following along at a very high level. I am aware of what is proposed and the improvement to documentation. but that PR was generating a lot of noise, so it is likely I dismissed the notification without looking. |
Yes, we are adding type annotionations, which is my point. It will no longer be true that the library does not provide type annotations. However, I'm not really familiar with LSP or what is does or what it expects, so I don't know how this is relevant. |
Ah, I take that back then. Like I said, I wasn't following that closely. Unless you define a |
Ah okay, so this is the complaint then. It recently occurred to me that we could provide some hook for extensions to use for this. For example, we could provide an attribute which stored key/value pairs where the key would be a name provided by the extension and the value would be whatever data the extension provides. Then it could be type annotated as The issue of course is that many existing extensions already provide their own custom solutions and this would be a new API. Users would then need to alter their code to check the new location. Maybe we could use a custom getter and setter which would cause the extension to write to our custom location and the user to read from our custom without any changes on their end (extension or user). Although, now that I think about it, some projects make extensive use of custom getters and setters on classes (see Django for example). They have a whole collections of potential attributes which are unknown. How do they handle this? Maybe that is the approach we should take. |
Maybe, there are a lot of complex typing paths that I have yet to explore. |
To be clear I'm talking about the |
If it came to it, you could just generally set a type hinting for the In this case, I think my LSP picks up on the type if I check before using with |
But where would we do that? Not on the So we have two options:
|
Apparently I bumped into this topic here: |
There are no solutions for any typechecker that would allow these attributes to be properly declared with known types, other than actually declaring them on the core object itself. So these are the options:
|
By the way, the type checking of both pyright and mypy are not based on the So yeah, I wonder what their philosophy is in this situation, would they consider this a valid addition to toc_tokens: list[TocToken]
toc: str
Meta: dict[str, Any] |
The problem is that this favors the built-in extensions, which I have tried to avoid. I have tried to maintain the idea that anything that a built-in extension can do a third-party extension can do. Yes, I know the built-in extensions are favored just by being built-in, but for many years now I have operated on the assumption that in the future each built-in extension could eventually be broken out into a separate third-party extension. I'm not inclined to change that now.
To me this seems like the least bad option of those listed. As a benefit, it provides an official API for any extension (first or third party) to make data available to the user. Obviously, option 3 would be preferred for that purpose (official API), but the backward incompatibility is a blocker... unless we combined options 2 and 3. We could go with option 3 and add an attribute which stored anything by key as an official API. Then we could add a getter as glue between the old and new. The getter should be generic to work with any unknown third-party extensions as well. Let's call this option 5. Regardless of whether we went with options 2, 4, or 5 (the only contenders IMO), there would be no actual type checking. I'm okay with that. However, 2 or 5 would at least give us a way to satisfy the type checkers. Option 4 would not, but maybe that doesn't matter...
Good point. Technically, at this time, this issue should have been raised with @python/typeshed. And yes, it looks like it may remain there for now. @PlexSheep you should probably submit an issue there if you would like an immediate resolution. Regardless, I am going to leave this open until a decision is made regarding what we are going to do here. |
I ran into a slightly relevant example. Jinja has extensions that add methods to env: jinja2.Environment
env.add_extension('jinja2.ext.i18n')
env.install_gettext_translations(translations) With typeshed stubs it was "pragmatically" added and used to just work But the project itself is now
-I think there is now no way to use that extension in a type-safe way |
I have no objection to that. In other words, that would not stop me from using the extension. All I ask is that if we define types, we have a way to verify those annotations. However, all "missing" annotations should be ignored. If there is no way to do that. then that is an issue with the tooling. Of course, I assume that the problem is that the people who are motivated to build the tooling are the same people who want statically typed code so they haven't bothered to build the tool I want. In any event, there is likely an added complexity here. In most cases, we can simply insert a comment on the "offending" line to tell the tooling to ignore that line. However, in this case, that may be difficult because the attributes don't exist. There is no line to comment. Unless we can include the comment on the line in the extension which defines the attribute. Or will every line which accesses the attribute need to be commented also? Including the users code? If so, that is very annoying. |
Yes, that's what I was saying
In this codebase: (#1402) self.md.toc_tokens = toc_tokens # type: ignore[attr-defined] At usages: self.toc = get_toc(md.toc_tokens) # type: ignore[attr-defined] Note that the comment can also be shorter - just Or actually no, maybe users will be fine? They actually shouldn't unconditionally access that attr because it can be unset. They should be doing self.toc = get_toc(getattr(md, 'toc_tokens', [])) |
Good observation. It seems, then, that the simplest fix would be for us to ignore (via comment) each internal line in the extension and update our docs to encourage users to use |
When using the meta extension, the
Meta
field of theMarkdown
class is not found byPyright
. Perhaps I missed something, can we import a stub file to make this error go away?For clarity: The code works, but the language server does not find the
Meta
Member for typeMarkdown
.This might also happen with the attributes added by other extensions, but I've found it for the
meta
extension specifically.Minimal example:
test.py
:foo.md
:executing the script:
Proposed Fix
Add a python stub file for the
Meta
attribute.The text was updated successfully, but these errors were encountered: