summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authormike bayer <mike_mp@zzzcomputing.com>2021-08-25 18:23:08 +0000
committerGerrit Code Review <gerrit@ci3.zzzcomputing.com>2021-08-25 18:23:08 +0000
commite2d9ef3fe6f7bd9b151caf71ae5eb7f15522ec8c (patch)
treec718cb19b990a4fd1b9a27dc60091e71079e82ba
parentf5b3103ac2798fb2e5d15541b1489d0d13edf4e7 (diff)
parent07b7e1594deecc6c9c64893a63f56403da0dcf05 (diff)
downloadsqlalchemy-e2d9ef3fe6f7bd9b151caf71ae5eb7f15522ec8c.tar.gz
Merge "ensure "sqlalchemy" info set for all considered classes"
-rw-r--r--doc/build/changelog/unreleased_14/6937.rst7
-rw-r--r--lib/sqlalchemy/ext/mypy/decl_class.py3
-rw-r--r--lib/sqlalchemy/ext/mypy/util.py4
-rw-r--r--test/ext/mypy/files/mixin_w_tablename.py27
4 files changed, 41 insertions, 0 deletions
diff --git a/doc/build/changelog/unreleased_14/6937.rst b/doc/build/changelog/unreleased_14/6937.rst
new file mode 100644
index 000000000..5dfadd552
--- /dev/null
+++ b/doc/build/changelog/unreleased_14/6937.rst
@@ -0,0 +1,7 @@
+.. change::
+ :tags: bug, mypy
+ :tickets: 6937
+
+ Fixed issue in mypy plugin where columns on a mixin would not be correctly
+ interpreted if the mapped class relied upon a ``__tablename__`` routine
+ that came from a superclass.
diff --git a/lib/sqlalchemy/ext/mypy/decl_class.py b/lib/sqlalchemy/ext/mypy/decl_class.py
index 23c78aa51..b85ec0f69 100644
--- a/lib/sqlalchemy/ext/mypy/decl_class.py
+++ b/lib/sqlalchemy/ext/mypy/decl_class.py
@@ -61,6 +61,9 @@ def scan_declarative_assignments_and_apply_types(
List[util.SQLAlchemyAttribute]
] = util.get_mapped_attributes(info, api)
+ # used by assign.add_additional_orm_attributes among others
+ util.establish_as_sqlalchemy(info)
+
if mapped_attributes is not None:
# ensure that a class that's mapped is always picked up by
# its mapped() decorator or declarative metaclass before
diff --git a/lib/sqlalchemy/ext/mypy/util.py b/lib/sqlalchemy/ext/mypy/util.py
index 614805d77..a3825f175 100644
--- a/lib/sqlalchemy/ext/mypy/util.py
+++ b/lib/sqlalchemy/ext/mypy/util.py
@@ -99,6 +99,10 @@ def _get_info_mro_metadata(info: TypeInfo, key: str) -> Optional[Any]:
return None
+def establish_as_sqlalchemy(info: TypeInfo) -> None:
+ info.metadata.setdefault("sqlalchemy", {})
+
+
def set_is_base(info: TypeInfo) -> None:
_set_info_metadata(info, "is_base", True)
diff --git a/test/ext/mypy/files/mixin_w_tablename.py b/test/ext/mypy/files/mixin_w_tablename.py
new file mode 100644
index 000000000..cfbe83d35
--- /dev/null
+++ b/test/ext/mypy/files/mixin_w_tablename.py
@@ -0,0 +1,27 @@
+# test #6937
+from sqlalchemy import Column
+from sqlalchemy import Integer
+from sqlalchemy.orm import declarative_base
+from sqlalchemy.orm import declared_attr
+from sqlalchemy.orm import Mapped
+
+
+Base = declarative_base()
+
+
+class UpdatedCls:
+ @declared_attr
+ def __tablename__(cls) -> Mapped[str]:
+ return cls.__name__.lower()
+
+ updated_at = Column(Integer)
+
+
+class Bar(UpdatedCls, Base):
+ id = Column(Integer(), primary_key=True)
+ num = Column(Integer)
+
+
+Bar.updated_at.in_([1, 2, 3])
+
+b1 = Bar(num=5, updated_at=6)