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
Bad error message: mypy reports 'Name is not defined' when using an object member as super class of another class #8603
Comments
This is an incorrect error message, certainly. We're not going to support this code, though. |
@msullivan Should the error message be changed? |
We'd be likely to accept a PR that improves the error message (something like "object member cannot be used as base class", I suppose). |
@JelleZijlstra This is my first time contributing. Can I pick this up? |
``I would like to contribute to this issue. Can I pull this up? |
Can I look into this, seems like I can handle this one |
@msullivan @JelleZijlstra I looked through the codebase where I can set up a check for this, I think I have to implement something around |
Using per line comment as # type: ignore[name-defined] at less silence this error message |
If this issue is not solved, I would like to be assigned this @msullivan @JelleZijlstra |
Have similar issue:
|
I'm currently adding annotations to Flask-SQLAlchemy. If this is not supported, I will never be able to enable type checking, as it breaks one of the major features of Flask-SQLAlchemy. If I do enable it, every project would have to add an ignore rule for The extension creates a Here's a minimal example without any SQLAlchemy-specific code: import typing as t
class Example:
Model: t.Type[object]
def __init__(self):
self.Model = object
db = Example()
reveal_type(db.Model)
class User(db.Model):
pass
I don't understand why this won't be supported. Mypy already understands that the name is a |
Flask-SQLAlchemy has blown up for me after the 3.0 release. As I can see from pallets-eco/flask-sqlalchemy#1118, the fix is to litter my code with |
You could also
which would make it as if flask_sqlalchemy didn't have a py.typed, just like before v3 |
But why would you want to do that? Most of the types are fine and helpful. It's just the fact that MyPy doesn't accept |
I'm removing Flask-SQLAlchemy from my code entirely. I've moved most references of This has been a nightmare year with perfectly functional code falling apart every few months as various framework libraries refactor themselves to fit into the constraints of static typing. Static type checking has been an excellent development for Python – it's significantly reduced the need for test coverage of data types, or defensive code to evaluate function parameters – but it's also shrunk the dynamic nature of Python to the subset recognised by type checkers. My project now has an average of five An idiom as simple and helpful as this no longer works because class MyModel(ModelBase):
fkey_id = sa.Column(None, sa.ForeignKey('other_table.id')) Instead I have to lookup the other table, find the type of its primary key, and use: class MyModel(ModelBase):
fkey_id = sa.Column(sa.Integer, sa.ForeignKey('other_table.id')) …or I throw in a Sorry for the rant. This period of transition is painful, but I'm hopeful for how this improves speed of development (at least post-transition) and code review. |
This is a workaround for how mypy reports a "name-defined" error for all uses of db.Model. See these for more information: pallets-eco/flask-sqlalchemy#1118 python/mypy#8603 Signed-off-by: Taylor Madore <tmadore@redhat.com>
This is a workaround for how mypy reports a "name-defined" error for all uses of db.Model. See these for more information: pallets-eco/flask-sqlalchemy#1118 python/mypy#8603 Signed-off-by: Taylor Madore <tmadore@redhat.com>
(via python/typeshed#9860) Here’s an example of a seemingly innocuous usage of the multiprocessing module that triggers this error: import multiprocessing
context = multiprocessing.get_context()
class MyProcess(context.Process): # error: Name "context.Process" is not defined [name-defined].
pass It would be great to hear why the restriction is in place. Is this just hard to solve technically or would it cause other issues were it allowed? |
Need to add ignores for the Flask-SQLAlchemy model declarations as a workaround for limitations regarding inheritance from objects, see python/mypy#8603
Need to add ignores for the Flask-SQLAlchemy model declarations as a workaround for limitations regarding inheritance from objects, see python/mypy#8603
@jace above expresses eloquently the frustration a lot of us share. Mypy and Flask-SQLAlchemy are both popular packages. Would it be possible for the teams from each to talk to each other and find a better way forwards than either dropping |
Happy to review a PR to Flask-SQLAlchemy that fixes the issue while retaining compatibility with its existing interface. But there's nothing more to do on my end than what I've already commented here. Note that even if MyPy fixes it, you have to separately convince pyright (vscode) to fix it, and they've already declined. |
I've replaced |
That looks very interesting @jace, thank you. I will look into migrating to your solution. |
@davidism I'm sorry this is not drop-in compatible with Flask-SQLAlchemy or I'd have made a PR. I had to move all namespacing and base class config out of the At best, I can think of a cleaner but backward-incompatible integration. |
This has been helpful discussion - thanks in particular to @davidism and @jace. I think Flask-SQLAlchemy is a great extension for proof of concept code and for those less experienced with Flask and/or SQLAlchemy. As a project grows and the team become more experienced, tools such as mypy become more important and valuable. At some point the value of mypy takes precedent over the convenience of wrappers such as Flask-SQLAlchemy, particularly such wrappers don't sit well with mypy. For now, we've adapted an approach outlined in this post, seemingly (and worryingly) only available on Wayback Machine. The approach is to define a declarative base outside of Flask-SQLAlchemy and then pass that to the extension, which not only avoids the mypy problems but also makes the declarative base available independently of Flask which, as it happens, is also helpful for us. Some code changes were needed from those given in the post referenced above, and I'm happy to make them available on request. For future projects, I suspect we'll choose not to use Flask-SQLAlchemy and just code SQLAlchemy access natively as documented on the Flask pages. If Flask-SQLAlchemy were to take the approach of using a predefined declarative base then I suspect that would increase the utility of that extension. However, that would be backward incompatible and may not align with the future goals of Flask-SQLAlchemy. |
The reason I keep voting these comments down is because they're orthogonal to this issue and don't help fix the problem in MyPy or in Flask-SQLAlchemy. And it really, really sucks to be a maintainer that has poured countless hours into it to then be told "Flask-SQLAlchemy isn't good because it doesn't work with typing in this one case", or "Just rip out Flask-SQLAlchemy and replace it", or "I'm going to tell everyone not to use Flask-SQLAlchemy", or "Flask-SQLalchemy is only good for proofs of concept" etc. It really, really sucks. I am not happy with the fact that I keep hearing these things just because I need to watch this issue for actual resolutions to the issue. I'd suggest not replying along these lines further, it's off topic here, sorry MyPy maintainers. But I couldn't remain silent in the face of being told my work isn't good over and over again, especially after being completely open to fixing the issue. And if you're response is "that wasn't my intention" it doesn't matter, that's how it came off anyway. |
The Here is an example for Flask-SQLAlchemy, which works for me, but I'm just not sure if it's good practice: from typing import TYPE_CHECKING
from flask_sqlalchemy import SQLAlchemy
db = SQLAlchemy()
if TYPE_CHECKING:
from flask_sqlalchemy.model import Model
else:
Model = db.Model
class ExampleModel(Model):
pass
reveal_type(ExampleModel)
reveal_type(ExampleModel.query)
# note: Revealed type is "def () -> app.graphql.base.ExampleModel"
# note: Revealed type is "flask_sqlalchemy.query.Query" |
Consider the following valid Python 3 code:
Actual behavior:
Mypy will report
Name 'b.a' is not defined
on the lineclass C(b.a):
.Expected behavior:
No error reported by Mypy.
Python version: 3.8.2
Mypy version: 0.770 (master not checked)
Flags: no flags
The text was updated successfully, but these errors were encountered: