Skip to content
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

REGR: fix return class in _constructor_from_mgr for simple subclasses #55764

2 changes: 1 addition & 1 deletion doc/source/whatsnew/v2.1.3.rst
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ including other versions of pandas.

Fixed regressions
~~~~~~~~~~~~~~~~~
-
- Fixed infinite recursion from operations that return a new object on some DataFrame subclasses (:issue:`55763`)
-

.. ---------------------------------------------------------------------------
Expand Down
6 changes: 3 additions & 3 deletions pandas/core/frame.py
Original file line number Diff line number Diff line change
Expand Up @@ -646,9 +646,9 @@ def _constructor(self) -> Callable[..., DataFrame]:
return DataFrame

def _constructor_from_mgr(self, mgr, axes):
if self._constructor is DataFrame:
if type(self) is DataFrame:
ivirshup marked this conversation as resolved.
Show resolved Hide resolved
# we are pandas.DataFrame (or a subclass that doesn't override _constructor)
return self._from_mgr(mgr, axes=axes)
return DataFrame._from_mgr(mgr, axes=axes)
else:
assert axes is mgr.axes
return self._constructor(mgr)
Expand All @@ -659,7 +659,7 @@ def _sliced_from_mgr(self, mgr, axes) -> Series:
return Series._from_mgr(mgr, axes)

def _constructor_sliced_from_mgr(self, mgr, axes):
if self._constructor_sliced is Series:
if type(self) is DataFrame:
ser = self._sliced_from_mgr(mgr, axes)
ser._name = None # caller is responsible for setting real name
return ser
Expand Down
8 changes: 3 additions & 5 deletions pandas/core/series.py
Original file line number Diff line number Diff line change
Expand Up @@ -635,9 +635,9 @@ def _constructor(self) -> Callable[..., Series]:
return Series

def _constructor_from_mgr(self, mgr, axes):
if self._constructor is Series:
if type(self) is Series:
# we are pandas.Series (or a subclass that doesn't override _constructor)
ser = self._from_mgr(mgr, axes=axes)
ser = Series._from_mgr(mgr, axes=axes)
ser._name = None # caller is responsible for setting real name
return ser
else:
Expand All @@ -660,9 +660,7 @@ def _expanddim_from_mgr(self, mgr, axes) -> DataFrame:
return DataFrame._from_mgr(mgr, axes=mgr.axes)

def _constructor_expanddim_from_mgr(self, mgr, axes):
from pandas.core.frame import DataFrame

if self._constructor_expanddim is DataFrame:
if type(self) is Series:
return self._expanddim_from_mgr(mgr, axes)
assert axes is mgr.axes
return self._constructor_expanddim(mgr)
Expand Down
3 changes: 3 additions & 0 deletions pandas/tests/frame/methods/test_reindex_like.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,9 @@ def test_reindex_like_methods(self, method, expected_values):
result = df.reindex_like(df, method=method, tolerance=[0, 0, 0, 0])
tm.assert_frame_equal(df, result)

@pytest.mark.filterwarnings(
"ignore:Passing a BlockManager|Passing a SingleBlockManager:DeprecationWarning"
)
def test_reindex_like_subclass(self):
# https://github.com/pandas-dev/pandas/issues/31925
class MyDataFrame(DataFrame):
Expand Down
28 changes: 28 additions & 0 deletions pandas/tests/frame/test_subclass.py
Original file line number Diff line number Diff line change
Expand Up @@ -773,3 +773,31 @@ def test_constructor_with_metadata():
)
subset = df[["A", "B"]]
assert isinstance(subset, MySubclassWithMetadata)


class SimpleDataFrameSubClass(DataFrame):
"""A subclass of DataFrame that does not define a constructor."""


class SimpleSeriesSubClass(Series):
"""A subclass of Series that does not define a constructor."""


class TestSubclassWithoutConstructor:
def test_copy_df(self):
expected = DataFrame({"a": [1, 2, 3]})
result = SimpleDataFrameSubClass(expected).copy()

tm.assert_frame_equal(result, expected)
ivirshup marked this conversation as resolved.
Show resolved Hide resolved

def test_copy_series(self):
expected = Series([1, 2, 3])
result = SimpleSeriesSubClass(expected).copy()

tm.assert_series_equal(result, expected)

ivirshup marked this conversation as resolved.
Show resolved Hide resolved
def test_groupby(self):
df = SimpleDataFrameSubClass(DataFrame({"a": [1, 2, 3]}))

for _, v in df.groupby("a"):
assert isinstance(v, DataFrame)
Loading