Skip to content

Commit 136271a

Browse files
Backport PR #49615 on branch 1.5.x (REGR: Better warning in pivot_table when dropping nuisance columns) (#49653)
(cherry picked from commit ab89c53) Co-authored-by: Dennis Chukwunta <chuksmcdennis@yahoo.com>
1 parent c9252cf commit 136271a

File tree

4 files changed

+101
-5
lines changed

4 files changed

+101
-5
lines changed

pandas/core/reshape/pivot.py

+13-1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121
Substitution,
2222
deprecate_nonkeyword_arguments,
2323
)
24+
from pandas.util._exceptions import rewrite_warning
2425

2526
from pandas.core.dtypes.cast import maybe_downcast_to_dtype
2627
from pandas.core.dtypes.common import (
@@ -163,7 +164,18 @@ def __internal_pivot_table(
163164
values = list(values)
164165

165166
grouped = data.groupby(keys, observed=observed, sort=sort)
166-
agged = grouped.agg(aggfunc)
167+
msg = (
168+
"pivot_table dropped a column because it failed to aggregate. This behavior "
169+
"is deprecated and will raise in a future version of pandas. Select only the "
170+
"columns that can be aggregated."
171+
)
172+
with rewrite_warning(
173+
target_message="The default value of numeric_only",
174+
target_category=FutureWarning,
175+
new_message=msg,
176+
):
177+
agged = grouped.agg(aggfunc)
178+
167179
if dropna and isinstance(agged, ABCDataFrame) and len(agged.columns):
168180
agged = agged.dropna(how="all")
169181

pandas/tests/reshape/test_pivot.py

+4-4
Original file line numberDiff line numberDiff line change
@@ -146,7 +146,7 @@ def test_pivot_table_nocols(self):
146146
df = DataFrame(
147147
{"rows": ["a", "b", "c"], "cols": ["x", "y", "z"], "values": [1, 2, 3]}
148148
)
149-
msg = "The default value of numeric_only"
149+
msg = "pivot_table dropped a column because it failed to aggregate"
150150
with tm.assert_produces_warning(FutureWarning, match=msg):
151151
rs = df.pivot_table(columns="cols", aggfunc=np.sum)
152152
xp = df.pivot_table(index="cols", aggfunc=np.sum).T
@@ -907,7 +907,7 @@ def test_no_col(self):
907907

908908
# to help with a buglet
909909
self.data.columns = [k * 2 for k in self.data.columns]
910-
msg = "The default value of numeric_only"
910+
msg = "pivot_table dropped a column because it failed to aggregate"
911911
with tm.assert_produces_warning(FutureWarning, match=msg):
912912
table = self.data.pivot_table(
913913
index=["AA", "BB"], margins=True, aggfunc=np.mean
@@ -975,7 +975,7 @@ def test_margin_with_only_columns_defined(
975975
}
976976
)
977977

978-
msg = "The default value of numeric_only"
978+
msg = "pivot_table dropped a column because it failed to aggregate"
979979
with tm.assert_produces_warning(FutureWarning, match=msg):
980980
result = df.pivot_table(columns=columns, margins=True, aggfunc=aggfunc)
981981
expected = DataFrame(values, index=Index(["D", "E"]), columns=expected_columns)
@@ -2004,7 +2004,7 @@ def test_pivot_string_func_vs_func(self, f, f_numpy):
20042004
# GH #18713
20052005
# for consistency purposes
20062006

2007-
msg = "The default value of numeric_only"
2007+
msg = "pivot_table dropped a column because it failed to aggregate"
20082008
with tm.assert_produces_warning(FutureWarning, match=msg):
20092009
result = pivot_table(self.data, index="A", columns="B", aggfunc=f)
20102010
expected = pivot_table(self.data, index="A", columns="B", aggfunc=f_numpy)
+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
import warnings
2+
3+
import pytest
4+
5+
from pandas.util._exceptions import rewrite_warning
6+
7+
import pandas._testing as tm
8+
9+
10+
@pytest.mark.parametrize(
11+
"target_category, target_message, hit",
12+
[
13+
(FutureWarning, "Target message", True),
14+
(FutureWarning, "Target", True),
15+
(FutureWarning, "get mess", True),
16+
(FutureWarning, "Missed message", False),
17+
(DeprecationWarning, "Target message", False),
18+
],
19+
)
20+
@pytest.mark.parametrize(
21+
"new_category",
22+
[
23+
None,
24+
DeprecationWarning,
25+
],
26+
)
27+
def test_rewrite_warning(target_category, target_message, hit, new_category):
28+
new_message = "Rewritten message"
29+
if hit:
30+
expected_category = new_category if new_category else target_category
31+
expected_message = new_message
32+
else:
33+
expected_category = FutureWarning
34+
expected_message = "Target message"
35+
with tm.assert_produces_warning(expected_category, match=expected_message):
36+
with rewrite_warning(
37+
target_message, target_category, new_message, new_category
38+
):
39+
warnings.warn(message="Target message", category=FutureWarning)

pandas/util/_exceptions.py

+45
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,9 @@
33
import contextlib
44
import inspect
55
import os
6+
import re
67
from typing import Iterator
8+
import warnings
79

810

911
@contextlib.contextmanager
@@ -47,3 +49,46 @@ def find_stack_level() -> int:
4749
else:
4850
break
4951
return n
52+
53+
54+
@contextlib.contextmanager
55+
def rewrite_warning(
56+
target_message: str,
57+
target_category: type[Warning],
58+
new_message: str,
59+
new_category: type[Warning] | None = None,
60+
) -> Iterator[None]:
61+
"""
62+
Rewrite the message of a warning.
63+
64+
Parameters
65+
----------
66+
target_message : str
67+
Warning message to match.
68+
target_category : Warning
69+
Warning type to match.
70+
new_message : str
71+
New warning message to emit.
72+
new_category : Warning or None, default None
73+
New warning type to emit. When None, will be the same as target_category.
74+
"""
75+
if new_category is None:
76+
new_category = target_category
77+
with warnings.catch_warnings(record=True) as record:
78+
yield
79+
if len(record) > 0:
80+
match = re.compile(target_message)
81+
for warning in record:
82+
if warning.category is target_category and re.search(
83+
match, str(warning.message)
84+
):
85+
category = new_category
86+
message: Warning | str = new_message
87+
else:
88+
category, message = warning.category, warning.message
89+
warnings.warn_explicit(
90+
message=message,
91+
category=category,
92+
filename=warning.filename,
93+
lineno=warning.lineno,
94+
)

0 commit comments

Comments
 (0)