diff --git a/python/cudf/cudf/core/column_accessor.py b/python/cudf/cudf/core/column_accessor.py index 832d5acf2de..6022d86e106 100644 --- a/python/cudf/cudf/core/column_accessor.py +++ b/python/cudf/cudf/core/column_accessor.py @@ -19,6 +19,7 @@ import pandas as pd from packaging.version import Version +from pandas.api.types import is_bool import cudf from cudf.core import column @@ -359,7 +360,8 @@ def get_labels_by_index(self, index: Any) -> tuple: Parameters ---------- - index : integer, integer slice, or list-like of integers + index : integer, integer slice, boolean mask, + or list-like of integers The column indexes. Returns @@ -371,6 +373,18 @@ def get_labels_by_index(self, index: Any) -> tuple: return self.names[start:stop:step] elif pd.api.types.is_integer(index): return (self.names[index],) + elif (bn := len(index)) > 0 and all(map(is_bool, index)): + if bn != (n := len(self.names)): + raise IndexError( + f"Boolean mask has wrong length: {bn} not {n}" + ) + if isinstance(index, (pd.Series, cudf.Series)): + # Don't allow iloc indexing with series + raise NotImplementedError( + "Cannot use Series object for mask iloc indexing" + ) + # TODO: Doesn't handle on-device columns + return tuple(n for n, keep in zip(self.names, index) if keep) else: return tuple(self.names[i] for i in index) @@ -381,7 +395,8 @@ def select_by_index(self, index: Any) -> ColumnAccessor: Parameters ---------- - key : integer, integer slice, or list-like of integers + key : integer, integer slice, boolean mask, + or list-like of integers Returns ------- @@ -462,7 +477,21 @@ def set_by_label(self, key: Any, value: Any, validate: bool = True): self._clear_cache() def _select_by_label_list_like(self, key: Any) -> ColumnAccessor: - data = {k: self._grouped_data[k] for k in key} + # Might be a generator + key = tuple(key) + # Special-casing for boolean mask + if (bn := len(key)) > 0 and all(map(is_bool, key)): + if bn != (n := len(self.names)): + raise IndexError( + f"Boolean mask has wrong length: {bn} not {n}" + ) + data = dict( + item + for item, keep in zip(self._grouped_data.items(), key) + if keep + ) + else: + data = {k: self._grouped_data[k] for k in key} if self.multiindex: data = _to_flat_dict(data) return self.__class__( diff --git a/python/cudf/cudf/core/indexed_frame.py b/python/cudf/cudf/core/indexed_frame.py index abd8ad162c7..23200c26359 100644 --- a/python/cudf/cudf/core/indexed_frame.py +++ b/python/cudf/cudf/core/indexed_frame.py @@ -2906,7 +2906,8 @@ def _apply_boolean_mask(self, boolean_mask): boolean_mask = cudf.core.column.as_column(boolean_mask) if not is_bool_dtype(boolean_mask.dtype): raise ValueError("boolean_mask is not boolean type.") - + if (bn := len(boolean_mask)) != (n := len(self)): + raise IndexError(f"Boolean mask has wrong length: {bn} not {n}") return self._from_columns_like_self( libcudf.stream_compaction.apply_boolean_mask( list(self._index._columns + self._columns), boolean_mask diff --git a/python/cudf/cudf/core/single_column_frame.py b/python/cudf/cudf/core/single_column_frame.py index 5f07c57fc80..0320e404e16 100644 --- a/python/cudf/cudf/core/single_column_frame.py +++ b/python/cudf/cudf/core/single_column_frame.py @@ -403,6 +403,10 @@ def _get_elements_from_column(self, arg) -> Union[ScalarLike, ColumnBase]: if is_integer_dtype(arg.dtype): return self._column.take(arg) if is_bool_dtype(arg.dtype): + if (bn := len(arg)) != (n := len(self)): + raise IndexError( + f"Boolean mask has wrong length: {bn} not {n}" + ) return self._column.apply_boolean_mask(arg) raise NotImplementedError(f"Unknown indexer {type(arg)}") diff --git a/python/cudf/cudf/tests/test_indexing.py b/python/cudf/cudf/tests/test_indexing.py index 95936c48b7c..b3ec6cdcf25 100644 --- a/python/cudf/cudf/tests/test_indexing.py +++ b/python/cudf/cudf/tests/test_indexing.py @@ -579,7 +579,6 @@ def test_dataframe_series_loc_multiindex(obj): @pytest.mark.parametrize("nelem", [2, 5, 20, 100]) def test_series_iloc(nelem): - # create random cudf.Series np.random.seed(12) ps = pd.Series(np.random.sample(nelem)) @@ -1265,7 +1264,6 @@ def test_iloc_categorical_index(index): ) @pytest.mark.parametrize("is_dataframe", [True, False]) def test_loc_datetime_index(sli, is_dataframe): - if is_dataframe is True: pd_data = pd.DataFrame( {"a": [1, 2, 3]}, @@ -1685,3 +1683,61 @@ def test_loc_single_row_from_slice(): pdf = pd.DataFrame({"a": [10, 20, 30], "b": [1, 2, 3]}).set_index("a") df = cudf.from_pandas(pdf) assert_eq(pdf.loc[5:10], df.loc[5:10]) + + +@pytest.mark.parametrize("indexer", ["loc", "iloc"]) +@pytest.mark.parametrize( + "mask", + [[False, True], [False, False, True, True, True]], + ids=["too-short", "too-long"], +) +def test_boolean_mask_wrong_length(indexer, mask): + s = pd.Series([1, 2, 3, 4]) + + indexee = getattr(s, indexer) + with pytest.raises(IndexError): + indexee[mask] + + c = cudf.from_pandas(s) + indexee = getattr(c, indexer) + with pytest.raises(IndexError): + indexee[mask] + + +@pytest.mark.parametrize("indexer", ["loc", "iloc"]) +def test_boolean_mask_columns(indexer): + df = pd.DataFrame(np.zeros((3, 3))) + cdf = cudf.from_pandas(df) + mask = [True, False, True] + expect = getattr(df, indexer)[:, mask] + got = getattr(cdf, indexer)[:, mask] + + assert_eq(expect, got) + + +@pytest.mark.parametrize("indexer", ["loc", "iloc"]) +@pytest.mark.parametrize( + "mask", + [[False, True], [False, False, True, True, True]], + ids=["too-short", "too-long"], +) +def test_boolean_mask_columns_wrong_length(indexer, mask): + df = pd.DataFrame(np.zeros((3, 3))) + cdf = cudf.from_pandas(df) + + with pytest.raises(IndexError): + getattr(df, indexer)[:, mask] + with pytest.raises(IndexError): + getattr(cdf, indexer)[:, mask] + + +def test_boolean_mask_columns_iloc_series(): + df = pd.DataFrame(np.zeros((3, 3))) + cdf = cudf.from_pandas(df) + + mask = pd.Series([True, False, True], dtype=bool) + with pytest.raises(NotImplementedError): + df.iloc[:, mask] + + with pytest.raises(NotImplementedError): + cdf.iloc[:, mask]