summaryrefslogtreecommitdiff
path: root/lib/sqlalchemy
diff options
context:
space:
mode:
authorMike Bayer <mike_mp@zzzcomputing.com>2021-01-25 17:59:35 -0500
committerMike Bayer <mike_mp@zzzcomputing.com>2021-01-25 17:59:35 -0500
commit9205e9171cfd4b488be61228d8d53b0da1d49c19 (patch)
treecbbbb25d1379ea141b2335ba29c326b2ea2bdcd5 /lib/sqlalchemy
parent57db20a187e80950037dd5a2141a560fe879e054 (diff)
downloadsqlalchemy-9205e9171cfd4b488be61228d8d53b0da1d49c19.tar.gz
Fill-out dataclass-related attr resolution
Fixed issue where mixin attribute rules were not taking effect correctly for attributes pulled from dataclasses using the approach added in #5745. Fixes: #5876 Change-Id: I45099a42de1d9611791e72250fe0edc69bed684c
Diffstat (limited to 'lib/sqlalchemy')
-rw-r--r--lib/sqlalchemy/orm/decl_base.py131
-rw-r--r--lib/sqlalchemy/testing/fixtures.py1
-rw-r--r--lib/sqlalchemy/util/__init__.py1
-rw-r--r--lib/sqlalchemy/util/compat.py20
4 files changed, 132 insertions, 21 deletions
diff --git a/lib/sqlalchemy/orm/decl_base.py b/lib/sqlalchemy/orm/decl_base.py
index db6d274c8..db7dfebe4 100644
--- a/lib/sqlalchemy/orm/decl_base.py
+++ b/lib/sqlalchemy/orm/decl_base.py
@@ -325,6 +325,94 @@ class _ClassScanMapperConfig(_MapperConfig):
def before_configured():
self.cls.__declare_first__()
+ def _cls_attr_override_checker(self, cls):
+ """Produce a function that checks if a class has overridden an
+ attribute, taking SQLAlchemy-enabled dataclass fields into account.
+
+ """
+ sa_dataclass_metadata_key = _get_immediate_cls_attr(
+ cls, "__sa_dataclass_metadata_key__", None
+ )
+
+ if sa_dataclass_metadata_key is None:
+
+ def attribute_is_overridden(key, obj):
+ return getattr(cls, key) is not obj
+
+ else:
+
+ all_datacls_fields = {
+ f.name: f.metadata[sa_dataclass_metadata_key]
+ for f in util.dataclass_fields(cls)
+ if sa_dataclass_metadata_key in f.metadata
+ }
+ local_datacls_fields = {
+ f.name: f.metadata[sa_dataclass_metadata_key]
+ for f in util.local_dataclass_fields(cls)
+ if sa_dataclass_metadata_key in f.metadata
+ }
+
+ absent = object()
+
+ def attribute_is_overridden(key, obj):
+ # this function likely has some failure modes still if
+ # someone is doing a deep mixing of the same attribute
+ # name as plain Python attribute vs. dataclass field.
+
+ ret = local_datacls_fields.get(key, absent)
+
+ if ret is obj:
+ return False
+ elif ret is not absent:
+ return True
+
+ ret = getattr(cls, key, obj)
+
+ if ret is obj:
+ return False
+ elif ret is not absent:
+ return True
+
+ ret = all_datacls_fields.get(key, absent)
+
+ if ret is obj:
+ return False
+ elif ret is not absent:
+ return True
+
+ # can't find another attribute
+ return False
+
+ return attribute_is_overridden
+
+ def _cls_attr_resolver(self, cls):
+ """produce a function to iterate the "attributes" of a class,
+ adjusting for SQLAlchemy fields embedded in dataclass fields.
+
+ """
+ sa_dataclass_metadata_key = _get_immediate_cls_attr(
+ cls, "__sa_dataclass_metadata_key__", None
+ )
+
+ if sa_dataclass_metadata_key is None:
+
+ def local_attributes_for_class():
+ for name, obj in vars(cls).items():
+ yield name, obj
+
+ else:
+
+ def local_attributes_for_class():
+ for name, obj in vars(cls).items():
+ yield name, obj
+ for field in util.local_dataclass_fields(cls):
+ if sa_dataclass_metadata_key in field.metadata:
+ yield field.name, field.metadata[
+ sa_dataclass_metadata_key
+ ]
+
+ return local_attributes_for_class
+
def _scan_attributes(self):
cls = self.cls
dict_ = self.dict_
@@ -333,9 +421,9 @@ class _ClassScanMapperConfig(_MapperConfig):
table_args = inherited_table_args = None
tablename = None
- for base in cls.__mro__:
+ attribute_is_overridden = self._cls_attr_override_checker(self.cls)
- sa_dataclass_metadata_key = None
+ for base in cls.__mro__:
class_mapped = (
base is not cls
@@ -345,25 +433,14 @@ class _ClassScanMapperConfig(_MapperConfig):
)
)
- if sa_dataclass_metadata_key is None:
- sa_dataclass_metadata_key = _get_immediate_cls_attr(
- base, "__sa_dataclass_metadata_key__", None
- )
-
- def attributes_for_class(cls):
- for name, obj in vars(cls).items():
- yield name, obj
- if sa_dataclass_metadata_key:
- for field in util.dataclass_fields(cls):
- if sa_dataclass_metadata_key in field.metadata:
- yield field.name, field.metadata[
- sa_dataclass_metadata_key
- ]
+ local_attributes_for_class = self._cls_attr_resolver(base)
if not class_mapped and base is not cls:
- self._produce_column_copies(attributes_for_class, base)
+ self._produce_column_copies(
+ local_attributes_for_class, attribute_is_overridden
+ )
- for name, obj in attributes_for_class(base):
+ for name, obj in local_attributes_for_class():
if name == "__mapper_args__":
check_decl = _check_declared_props_nocascade(
obj, name, cls
@@ -471,6 +548,15 @@ class _ClassScanMapperConfig(_MapperConfig):
else:
self._warn_for_decl_attributes(base, name, obj)
elif name not in dict_ or dict_[name] is not obj:
+ # here, we are definitely looking at the target class
+ # and not a superclass. this is currently a
+ # dataclass-only path. if the name is only
+ # a dataclass field and isn't in local cls.__dict__,
+ # put the object there.
+
+ # assert that the dataclass-enabled resolver agrees
+ # with what we are seeing
+ assert not attribute_is_overridden(name, obj)
dict_[name] = obj
if inherited_table_args and not tablename:
@@ -489,14 +575,17 @@ class _ClassScanMapperConfig(_MapperConfig):
% (key, cls)
)
- def _produce_column_copies(self, attributes_for_class, base):
+ def _produce_column_copies(
+ self, attributes_for_class, attribute_is_overridden
+ ):
cls = self.cls
dict_ = self.dict_
column_copies = self.column_copies
# copy mixin columns to the mapped class
- for name, obj in attributes_for_class(base):
+
+ for name, obj in attributes_for_class():
if isinstance(obj, Column):
- if getattr(cls, name) is not obj:
+ if attribute_is_overridden(name, obj):
# if column has been overridden
# (like by the InstrumentedAttribute of the
# superclass), skip
diff --git a/lib/sqlalchemy/testing/fixtures.py b/lib/sqlalchemy/testing/fixtures.py
index 62dd9040e..4b76e6d88 100644
--- a/lib/sqlalchemy/testing/fixtures.py
+++ b/lib/sqlalchemy/testing/fixtures.py
@@ -552,6 +552,7 @@ class DeclarativeMappedTest(MappedTest):
metaclass=FindFixtureDeclarative,
cls=DeclarativeBasic,
)
+
cls.DeclarativeBasic = _DeclBase
# sets up cls.Basic which is helpful for things like composite
diff --git a/lib/sqlalchemy/util/__init__.py b/lib/sqlalchemy/util/__init__.py
index 5f8788a6e..2d86b8b63 100644
--- a/lib/sqlalchemy/util/__init__.py
+++ b/lib/sqlalchemy/util/__init__.py
@@ -66,6 +66,7 @@ from .compat import int_types # noqa
from .compat import iterbytes # noqa
from .compat import itertools_filter # noqa
from .compat import itertools_filterfalse # noqa
+from .compat import local_dataclass_fields # noqa
from .compat import namedtuple # noqa
from .compat import next # noqa
from .compat import nullcontext # noqa
diff --git a/lib/sqlalchemy/util/compat.py b/lib/sqlalchemy/util/compat.py
index 1eed2c3af..5b7a3eb9f 100644
--- a/lib/sqlalchemy/util/compat.py
+++ b/lib/sqlalchemy/util/compat.py
@@ -425,17 +425,37 @@ if py37:
import dataclasses
def dataclass_fields(cls):
+ """Return a sequence of all dataclasses.Field objects associated
+ with a class."""
+
if dataclasses.is_dataclass(cls):
return dataclasses.fields(cls)
else:
return []
+ def local_dataclass_fields(cls):
+ """Return a sequence of all dataclasses.Field objects associated with
+ a class, excluding those that originate from a superclass."""
+
+ if dataclasses.is_dataclass(cls):
+ super_fields = set()
+ for sup in cls.__bases__:
+ super_fields.update(dataclass_fields(sup))
+ return [
+ f for f in dataclasses.fields(cls) if f not in super_fields
+ ]
+ else:
+ return []
+
else:
def dataclass_fields(cls):
return []
+ def local_dataclass_fields(cls):
+ return []
+
def raise_from_cause(exception, exc_info=None):
r"""legacy. use raise\_()"""