From b26ea1ea360ac0169a0529859b41eee57ac80bf1 Mon Sep 17 00:00:00 2001 From: Yerzhaisang Taskali <55043014+Yerzhaisang@users.noreply.github.com> Date: Mon, 6 Jan 2025 16:09:46 -0500 Subject: [PATCH] Bump pandas from 1.5.3 to the latest stable version (#422) * Bump pandas from 1.5.3 to 2.0.3 Signed-off-by: Yerzhaisang Taskali * Updated CHANGELOG Signed-off-by: Yerzhaisang Taskali * Updated CHANGELOG Signed-off-by: Yerzhaisang Taskali * updated built-in method Signed-off-by: Yerzhaisang Taskali * removed unused comment Signed-off-by: Yerzhaisang Taskali * Implement to_list_if_needed method for list conversion Signed-off-by: Yerzhaisang Taskali * Refactor MAD calculation using CustomFunctionDispatcher for improved readability and reusability Signed-off-by: Yerzhaisang Taskali * Refactor MAD calculation using CustomFunctionDispatcher for improved readability and reusability Signed-off-by: Yerzhaisang Taskali * Refactor MAD calculation using CustomFunctionDispatcher for improved readability and reusability Signed-off-by: Yerzhaisang Taskali * Refactor MAD calculation using CustomFunctionDispatcher for improved readability and reusability Signed-off-by: Yerzhaisang Taskali * refactor: move metric identifiers to constants.py for readability Signed-off-by: Yerzhaisang Taskali * refactor: move metric identifiers to constants.py for readability Signed-off-by: Yerzhaisang Taskali * Removed unused line Signed-off-by: Yerzhaisang Taskali * clarify build_pd_series docstring Signed-off-by: Yerzhaisang Taskali * save constansts in utils.py Signed-off-by: Yerzhaisang Taskali * fiexd CI Signed-off-by: Yerzhaisang Taskali * added keyword argument to to_csv method Signed-off-by: Yerzhaisang Taskali * added comment to describe method Signed-off-by: Yerzhaisang Taskali * possible pandas versions Signed-off-by: Yerzhaisang Taskali * upgrading pandas Signed-off-by: Yerzhaisang Taskali * upgrading pandas Signed-off-by: Yerzhaisang Taskali * wide range of pandas versions Signed-off-by: Yerzhaisang Taskali * test Signed-off-by: Yerzhaisang Taskali * test commit Signed-off-by: Yerzhaisang Taskali * Fixed doctest dtypes Signed-off-by: Yerzhaisang Taskali * Fixed doctest dtypes Signed-off-by: Yerzhaisang Taskali * Fixed datetime dtypes Signed-off-by: Yerzhaisang Taskali * Fixed datetime dtypes Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * Fixed datetime dtypes Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * Fixed datetime dtypes Signed-off-by: Yerzhaisang Taskali * Fixed datetime dtypes Signed-off-by: Yerzhaisang Taskali * Fixed datetime dtypes Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * Fixed testing issues Signed-off-by: Yerzhaisang Taskali * Fixed testing issues Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * Fixed testing issues Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing another version Signed-off-by: Yerzhaisang Taskali * testing final pandas version range Signed-off-by: Yerzhaisang Taskali * updating CHANGELOG Signed-off-by: Yerzhaisang Taskali * improving test coverage Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * adapt tests for backward compatability Signed-off-by: Yerzhaisang Taskali * rerun tests Signed-off-by: Yerzhaisang Taskali * rerun tests Signed-off-by: Yerzhaisang Taskali * pandas 2.0.0 Signed-off-by: Yerzhaisang Taskali * applied to_string and strip methods Signed-off-by: Yerzhaisang Taskali * applied to_string and strip methods Signed-off-by: Yerzhaisang Taskali * applied to_string and strip methods Signed-off-by: Yerzhaisang Taskali * applied to_string and strip methods Signed-off-by: Yerzhaisang Taskali * applied to_string and strip methods Signed-off-by: Yerzhaisang Taskali * applied to_string and strip methods Signed-off-by: Yerzhaisang Taskali * applied to_string and strip methods Signed-off-by: Yerzhaisang Taskali * testing pandas 1.5.2 Signed-off-by: Yerzhaisang Taskali * testing pandas 1.5.3 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.0.0 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.0.1 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.0.2 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.0.3 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.1.1 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.1.2 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.1.3 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.1.4 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.2.0 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.2.1 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.2.2 Signed-off-by: Yerzhaisang Taskali * testing pandas 2.2.3 Signed-off-by: Yerzhaisang Taskali * fixing pandas version Signed-off-by: Yerzhaisang Taskali * fixing pandas version Signed-off-by: Yerzhaisang Taskali --------- Signed-off-by: Yerzhaisang Taskali Signed-off-by: Yerzhaisang Taskali <55043014+Yerzhaisang@users.noreply.github.com> --- CHANGELOG.md | 2 + docs/requirements-docs.txt | 2 +- opensearch_py_ml/common.py | 25 +++++++- opensearch_py_ml/dataframe.py | 47 +++++++------- opensearch_py_ml/etl.py | 1 + opensearch_py_ml/groupby.py | 7 ++- opensearch_py_ml/operations.py | 61 ++++++++++++++++--- opensearch_py_ml/query_compiler.py | 7 ++- opensearch_py_ml/series.py | 12 ++-- opensearch_py_ml/utils.py | 48 +++++++++++++++ requirements-dev.txt | 2 +- requirements.txt | 2 +- setup.py | 2 +- tests/dataframe/test_aggs_pytest.py | 13 ++-- tests/dataframe/test_datetime_pytest.py | 2 + tests/dataframe/test_describe_pytest.py | 9 ++- tests/dataframe/test_groupby_pytest.py | 54 ++++++++++++++-- tests/dataframe/test_metrics_pytest.py | 61 +++++++++++++------ tests/dataframe/test_utils_pytest.py | 6 +- .../test_map_pd_aggs_to_es_aggs_pytest.py | 9 +-- tests/series/test_arithmetics_pytest.py | 4 +- tests/series/test_describe_pytest.py | 5 +- tests/series/test_metrics_pytest.py | 35 +++++++++-- tests/test_utils_pytest.py | 36 ++++++++++- 24 files changed, 354 insertions(+), 98 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 89c274d89..161196816 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -48,6 +48,8 @@ Inspired from [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) - Update model upload history - opensearch-project/opensearch-neural-sparse-encoding-doc-v2-mini (v.1.0.0)(TORCH_SCRIPT) by @dhrubo-os ([#417](https://github.com/opensearch-project/opensearch-py-ml/pull/417)) - Update model upload history - opensearch-project/opensearch-neural-sparse-encoding-v2-distill (v.1.0.0)(TORCH_SCRIPT) by @dhrubo-os ([#419](https://github.com/opensearch-project/opensearch-py-ml/pull/419)) - Upgrade GitHub Actions workflows to use `@v4` to prevent deprecation issues with `@v3` by @yerzhaisang ([#428](https://github.com/opensearch-project/opensearch-py-ml/pull/428)) +- Bump pandas from 1.5.3 to the latest stable version by @yerzhaisang ([#422](https://github.com/opensearch-project/opensearch-py-ml/pull/422)) + ### Fixed - Fix the wrong final zip file name in model_uploader workflow, now will name it by the upload_prefix alse.([#413](https://github.com/opensearch-project/opensearch-py-ml/pull/413/files)) diff --git a/docs/requirements-docs.txt b/docs/requirements-docs.txt index d4e8a521f..5a3a9ef51 100644 --- a/docs/requirements-docs.txt +++ b/docs/requirements-docs.txt @@ -1,5 +1,5 @@ opensearch-py>=2 -pandas>=1.5,<3 +pandas>=1.5.2,<2.3,!=2.1.0 matplotlib>=3.6.0,<4 nbval sphinx diff --git a/opensearch_py_ml/common.py b/opensearch_py_ml/common.py index add998305..ece2aad24 100644 --- a/opensearch_py_ml/common.py +++ b/opensearch_py_ml/common.py @@ -55,14 +55,33 @@ def build_pd_series( - data: Dict[str, Any], dtype: Optional["DTypeLike"] = None, **kwargs: Any + data: Dict[str, Any], + dtype: Optional["DTypeLike"] = None, + index_name: Optional[str] = None, + **kwargs: Any, ) -> pd.Series: - """Builds a pd.Series while squelching the warning - for unspecified dtype on empty series """ + Builds a pandas Series from a dictionary, optionally setting an index name. + + Parameters: + data : Dict[str, Any] + The data to build the Series from, with keys as the index. + dtype : Optional[DTypeLike] + The desired data type of the Series. If not specified, uses EMPTY_SERIES_DTYPE if data is empty. + index_name : Optional[str] + Name to assign to the Series index, similar to `index_name` in `value_counts`. + + Returns: + pd.Series + A pandas Series constructed from the given data, with the specified dtype and index name. + """ + dtype = dtype or (EMPTY_SERIES_DTYPE if not data else dtype) if dtype is not None: kwargs["dtype"] = dtype + if index_name is not None: + index = pd.Index(data.keys(), name=index_name) + kwargs["index"] = index return pd.Series(data, **kwargs) diff --git a/opensearch_py_ml/dataframe.py b/opensearch_py_ml/dataframe.py index 64772887f..4bb2fbc7b 100644 --- a/opensearch_py_ml/dataframe.py +++ b/opensearch_py_ml/dataframe.py @@ -47,7 +47,7 @@ from opensearch_py_ml.groupby import DataFrameGroupBy from opensearch_py_ml.ndframe import NDFrame from opensearch_py_ml.series import Series -from opensearch_py_ml.utils import is_valid_attr_name +from opensearch_py_ml.utils import is_valid_attr_name, to_list_if_needed if TYPE_CHECKING: from opensearchpy import OpenSearch @@ -55,6 +55,9 @@ from .query_compiler import QueryCompiler +PANDAS_MAJOR_VERSION = int(pd.__version__.split(".")[0]) + + class DataFrame(NDFrame): """ Two-dimensional size-mutable, potentially heterogeneous tabular data structure with labeled axes @@ -275,22 +278,13 @@ def tail(self, n: int = 5) -> "DataFrame": >>> from tests import OPENSEARCH_TEST_CLIENT >>> df = oml.DataFrame(OPENSEARCH_TEST_CLIENT, 'flights', columns=['Origin', 'Dest']) - >>> df.tail() - Origin \\ - 13054 Pisa International Airport... - 13055 Winnipeg / James Armstrong Richardson International Airport... - 13056 Licenciado Benito Juarez International Airport... - 13057 Itami Airport... - 13058 Adelaide International Airport... - - Dest... - 13054 Xi'an Xianyang International Airport... - 13055 Zurich Airport... - 13056 Ukrainka Air Base... - 13057 Ministro Pistarini International Airport... - 13058 Washington Dulles International Airport... - - [5 rows x 2 columns] + >>> print(df.tail().to_string().strip()) + Origin Dest + 13054 Pisa International Airport Xi'an Xianyang International Airport + 13055 Winnipeg / James Armstrong Richardson International Airport Zurich Airport + 13056 Licenciado Benito Juarez International Airport Ukrainka Air Base + 13057 Itami Airport Ministro Pistarini International Airport + 13058 Adelaide International Airport Washington Dulles International Airport """ return DataFrame(_query_compiler=self._query_compiler.tail(n)) @@ -424,9 +418,14 @@ def drop( axis = pd.DataFrame._get_axis_name(axis) axes = {axis: labels} elif index is not None or columns is not None: - axes, _ = pd.DataFrame()._construct_axes_from_arguments( - (index, columns), {} - ) + axes = { + "index": to_list_if_needed(index), + "columns": ( + pd.Index(to_list_if_needed(columns)) + if columns is not None + else None + ), + } else: raise ValueError( "Need to specify at least one of 'labels', 'index' or 'columns'" @@ -440,7 +439,7 @@ def drop( axes["index"] = [axes["index"]] if errors == "raise": # Check if axes['index'] values exists in index - count = self._query_compiler._index_matches_count(axes["index"]) + count = self._query_compiler._index_matches_count(list(axes["index"])) if count != len(axes["index"]): raise ValueError( f"number of labels {count}!={len(axes['index'])} not contained in axis" @@ -1341,6 +1340,10 @@ def to_csv( -------- :pandas_api_docs:`pandas.DataFrame.to_csv` """ + if PANDAS_MAJOR_VERSION < 2: + line_terminator_keyword = "line_terminator" + else: + line_terminator_keyword = "lineterminator" kwargs = { "path_or_buf": path_or_buf, "sep": sep, @@ -1355,7 +1358,7 @@ def to_csv( "compression": compression, "quoting": quoting, "quotechar": quotechar, - "line_terminator": line_terminator, + line_terminator_keyword: line_terminator, "chunksize": chunksize, "date_format": date_format, "doublequote": doublequote, diff --git a/opensearch_py_ml/etl.py b/opensearch_py_ml/etl.py index d2aac772c..da47c7e85 100644 --- a/opensearch_py_ml/etl.py +++ b/opensearch_py_ml/etl.py @@ -108,6 +108,7 @@ def pandas_to_opensearch( ... 'G': [1, 2, 3], ... 'H': 'Long text - to be indexed as os type text'}, ... index=['0', '1', '2']) + >>> pd_df['D'] = pd_df['D'].astype('datetime64[ns]') >>> type(pd_df) >>> pd_df diff --git a/opensearch_py_ml/groupby.py b/opensearch_py_ml/groupby.py index e5c4561c3..ea2083485 100644 --- a/opensearch_py_ml/groupby.py +++ b/opensearch_py_ml/groupby.py @@ -26,6 +26,7 @@ from typing import TYPE_CHECKING, List, Optional, Union from opensearch_py_ml.query_compiler import QueryCompiler +from opensearch_py_ml.utils import MEAN_ABSOLUTE_DEVIATION, STANDARD_DEVIATION, VARIANCE if TYPE_CHECKING: import pandas as pd # type: ignore @@ -153,7 +154,7 @@ def var(self, numeric_only: bool = True) -> "pd.DataFrame": """ return self._query_compiler.aggs_groupby( by=self._by, - pd_aggs=["var"], + pd_aggs=[VARIANCE], dropna=self._dropna, numeric_only=numeric_only, ) @@ -206,7 +207,7 @@ def std(self, numeric_only: bool = True) -> "pd.DataFrame": """ return self._query_compiler.aggs_groupby( by=self._by, - pd_aggs=["std"], + pd_aggs=[STANDARD_DEVIATION], dropna=self._dropna, numeric_only=numeric_only, ) @@ -259,7 +260,7 @@ def mad(self, numeric_only: bool = True) -> "pd.DataFrame": """ return self._query_compiler.aggs_groupby( by=self._by, - pd_aggs=["mad"], + pd_aggs=[MEAN_ABSOLUTE_DEVIATION], dropna=self._dropna, numeric_only=numeric_only, ) diff --git a/opensearch_py_ml/operations.py b/opensearch_py_ml/operations.py index c3d01e9e6..19d7e1079 100644 --- a/opensearch_py_ml/operations.py +++ b/opensearch_py_ml/operations.py @@ -65,6 +65,7 @@ SizeTask, TailTask, ) +from opensearch_py_ml.utils import MEAN_ABSOLUTE_DEVIATION, STANDARD_DEVIATION, VARIANCE if TYPE_CHECKING: from numpy.typing import DTypeLike @@ -75,6 +76,8 @@ from opensearch_py_ml.query_compiler import QueryCompiler from opensearch_py_ml.tasks import Task +PANDAS_MAJOR_VERSION = int(pd.__version__.split(".")[0]) + class QueryParams: def __init__(self) -> None: @@ -475,7 +478,10 @@ def _terms_aggs( except IndexError: name = None - return build_pd_series(results, name=name) + if PANDAS_MAJOR_VERSION < 2: + return build_pd_series(results, name=name) + else: + return build_pd_series(results, index_name=name, name="count") def _hist_aggs( self, query_compiler: "QueryCompiler", num_bins: int @@ -620,7 +626,7 @@ def _unpack_metric_aggs( values.append(field.nan_value) # Explicit condition for mad to add NaN because it doesn't support bool elif is_dataframe_agg and numeric_only: - if pd_agg == "mad": + if pd_agg == MEAN_ABSOLUTE_DEVIATION: values.append(field.nan_value) continue @@ -1097,7 +1103,14 @@ def _map_pd_aggs_to_os_aggs( """ # pd aggs that will be mapped to os aggs # that can use 'extended_stats'. - extended_stats_pd_aggs = {"mean", "min", "max", "sum", "var", "std"} + extended_stats_pd_aggs = { + "mean", + "min", + "max", + "sum", + VARIANCE, + STANDARD_DEVIATION, + } extended_stats_os_aggs = {"avg", "min", "max", "sum"} extended_stats_calls = 0 @@ -1117,15 +1130,15 @@ def _map_pd_aggs_to_os_aggs( os_aggs.append("avg") elif pd_agg == "sum": os_aggs.append("sum") - elif pd_agg == "std": + elif pd_agg == STANDARD_DEVIATION: os_aggs.append(("extended_stats", "std_deviation")) - elif pd_agg == "var": + elif pd_agg == VARIANCE: os_aggs.append(("extended_stats", "variance")) # Aggs that aren't 'extended_stats' compatible elif pd_agg == "nunique": os_aggs.append("cardinality") - elif pd_agg == "mad": + elif pd_agg == MEAN_ABSOLUTE_DEVIATION: os_aggs.append("median_absolute_deviation") elif pd_agg == "median": os_aggs.append(("percentiles", (50.0,))) @@ -1205,7 +1218,7 @@ def describe(self, query_compiler: "QueryCompiler") -> pd.DataFrame: df1 = self.aggs( query_compiler=query_compiler, - pd_aggs=["count", "mean", "std", "min", "max"], + pd_aggs=["count", "mean", "min", "max", STANDARD_DEVIATION], numeric_only=True, ) df2 = self.quantile( @@ -1219,9 +1232,37 @@ def describe(self, query_compiler: "QueryCompiler") -> pd.DataFrame: # Convert [.25,.5,.75] to ["25%", "50%", "75%"] df2 = df2.set_index([["25%", "50%", "75%"]]) - return pd.concat([df1, df2]).reindex( - ["count", "mean", "std", "min", "25%", "50%", "75%", "max"] - ) + df = pd.concat([df1, df2]) + + if PANDAS_MAJOR_VERSION < 2: + return pd.concat([df1, df2]).reindex( + ["count", "mean", "std", "min", "25%", "50%", "75%", "max"] + ) + else: + # Note: In recent pandas versions, `describe()` returns a different index order + # for one-column DataFrames compared to multi-column DataFrames. + # We adjust the order manually to ensure consistency. + if df.shape[1] == 1: + # For single-column DataFrames, `describe()` typically outputs: + # ["count", "mean", "std", "min", "25%", "50%", "75%", "max"] + return df.reindex( + [ + "count", + "mean", + STANDARD_DEVIATION, + "min", + "25%", + "50%", + "75%", + "max", + ] + ) + + # For multi-column DataFrames, `describe()` typically outputs: + # ["count", "mean", "min", "25%", "50%", "75%", "max", "std"] + return df.reindex( + ["count", "mean", "min", "25%", "50%", "75%", "max", STANDARD_DEVIATION] + ) def to_pandas( self, query_compiler: "QueryCompiler", show_progress: bool = False diff --git a/opensearch_py_ml/query_compiler.py b/opensearch_py_ml/query_compiler.py index c10899671..735b0307c 100644 --- a/opensearch_py_ml/query_compiler.py +++ b/opensearch_py_ml/query_compiler.py @@ -45,6 +45,7 @@ from opensearch_py_ml.filter import BooleanFilter, QueryFilter from opensearch_py_ml.index import Index from opensearch_py_ml.operations import Operations +from opensearch_py_ml.utils import MEAN_ABSOLUTE_DEVIATION, STANDARD_DEVIATION, VARIANCE if TYPE_CHECKING: from opensearchpy import OpenSearch @@ -587,17 +588,17 @@ def mean(self, numeric_only: Optional[bool] = None) -> pd.Series: def var(self, numeric_only: Optional[bool] = None) -> pd.Series: return self._operations._metric_agg_series( - self, ["var"], numeric_only=numeric_only + self, [VARIANCE], numeric_only=numeric_only ) def std(self, numeric_only: Optional[bool] = None) -> pd.Series: return self._operations._metric_agg_series( - self, ["std"], numeric_only=numeric_only + self, [STANDARD_DEVIATION], numeric_only=numeric_only ) def mad(self, numeric_only: Optional[bool] = None) -> pd.Series: return self._operations._metric_agg_series( - self, ["mad"], numeric_only=numeric_only + self, [MEAN_ABSOLUTE_DEVIATION], numeric_only=numeric_only ) def median(self, numeric_only: Optional[bool] = None) -> pd.Series: diff --git a/opensearch_py_ml/series.py b/opensearch_py_ml/series.py index 772660b13..176c4a035 100644 --- a/opensearch_py_ml/series.py +++ b/opensearch_py_ml/series.py @@ -311,12 +311,12 @@ def value_counts(self, os_size: int = 10) -> pd.Series: >>> from tests import OPENSEARCH_TEST_CLIENT >>> df = oml.DataFrame(OPENSEARCH_TEST_CLIENT, 'flights') - >>> df['Carrier'].value_counts() - Logstash Airways 3331 - JetBeats 3274 - Kibana Airlines 3234 - ES-Air 3220 - Name: Carrier, dtype: int64 + >>> for key, value in df['Carrier'].value_counts().items(): + ... print(key, value) + Logstash Airways 3331 + JetBeats 3274 + Kibana Airlines 3234 + ES-Air 3220 """ if not isinstance(os_size, int): raise TypeError("os_size must be a positive integer.") diff --git a/opensearch_py_ml/utils.py b/opensearch_py_ml/utils.py index 8f1763085..e850d2724 100644 --- a/opensearch_py_ml/utils.py +++ b/opensearch_py_ml/utils.py @@ -30,9 +30,14 @@ from typing import Any, Callable, Collection, Iterable, List, TypeVar, Union, cast import pandas as pd # type: ignore +from pandas.core.dtypes.common import is_list_like # type: ignore RT = TypeVar("RT") +MEAN_ABSOLUTE_DEVIATION = "mad" +VARIANCE = "var" +STANDARD_DEVIATION = "std" + def deprecated_api( replace_with: str, @@ -61,6 +66,29 @@ def is_valid_attr_name(s: str) -> bool: ) +def to_list_if_needed(value): + """ + Converts the input to a list if necessary. + + If the input is a pandas Index, it converts it to a list. + If the input is not list-like (e.g., a single value), it wraps it in a list. + If the input is None or already list-like, it returns it as is. + + Parameters: + value: The input to potentially convert to a list. + + Returns: + The input converted to a list if needed, or the original input if no conversion is necessary. + """ + if value is None: + return None + if isinstance(value, pd.Index): + return value.tolist() + if not is_list_like(value): + return [value] + return value + + def to_list(x: Union[Collection[Any], pd.Series]) -> List[Any]: if isinstance(x, ABCCollection): return list(x) @@ -77,3 +105,23 @@ def try_sort(iterable: Iterable[str]) -> Iterable[str]: return sorted(listed) except TypeError: return listed + + +class CustomFunctionDispatcher: + # Define custom functions in a dictionary + customFunctionMap = { + MEAN_ABSOLUTE_DEVIATION: lambda x: (x - x.median()).abs().mean(), + } + + @classmethod + def apply_custom_function(cls, func, data): + """ + Apply a custom function if available, else return None. + :param func: Function name as a string + :param data: Data on which function is applied + :return: Result of custom function or None if func not found + """ + custom_func = cls.customFunctionMap.get(func) + if custom_func: + return custom_func(data) + return None diff --git a/requirements-dev.txt b/requirements-dev.txt index e7b62bcf0..ce473e413 100644 --- a/requirements-dev.txt +++ b/requirements-dev.txt @@ -1,7 +1,7 @@ # # Basic requirements # -pandas>=1.5.2,<2 +pandas>=1.5.2,<2.3,!=2.1.0 matplotlib>=3.6.2,<4 numpy>=1.24.0,<2 opensearch-py>=2.2.0 diff --git a/requirements.txt b/requirements.txt index cddfe801c..67588beca 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,7 +1,7 @@ # # Basic requirements # -pandas>=1.5.2,<2 +pandas>=1.5.2,<2.3,!=2.1.0 matplotlib>=3.6.2,<4 numpy>=1.24.0,<2 opensearch-py>=2.2.0 diff --git a/setup.py b/setup.py index 9146b2503..3acc836da 100644 --- a/setup.py +++ b/setup.py @@ -84,7 +84,7 @@ }, install_requires=[ "opensearch-py>=2", - "pandas>=1.5,<3", + "pandas>=1.5.2,<2.3,!=2.1.0", "matplotlib>=3.6.0,<4", "numpy>=1.24.0,<2", "deprecated>=1.2.14,<2", diff --git a/tests/dataframe/test_aggs_pytest.py b/tests/dataframe/test_aggs_pytest.py index 62f6843a6..748a3dfd3 100644 --- a/tests/dataframe/test_aggs_pytest.py +++ b/tests/dataframe/test_aggs_pytest.py @@ -28,6 +28,7 @@ import pytest from pandas.testing import assert_frame_equal, assert_series_equal +from opensearch_py_ml.utils import STANDARD_DEVIATION, VARIANCE from tests.common import TestData @@ -47,10 +48,10 @@ def test_basic_aggs(self): assert_frame_equal(pd_sum_min, oml_sum_min, check_exact=False) pd_sum_min_std = pd_flights.select_dtypes(include=[np.number]).agg( - ["sum", "min", "std"] + ["sum", "min", STANDARD_DEVIATION] ) oml_sum_min_std = oml_flights.select_dtypes(include=[np.number]).agg( - ["sum", "min", "std"], numeric_only=True + ["sum", "min", STANDARD_DEVIATION], numeric_only=True ) print(pd_sum_min_std.dtypes) @@ -75,10 +76,10 @@ def test_terms_aggs(self): assert_frame_equal(pd_sum_min, oml_sum_min, check_exact=False) pd_sum_min_std = pd_flights.select_dtypes(include=[np.number]).agg( - ["sum", "min", "std"] + ["sum", "min", STANDARD_DEVIATION] ) oml_sum_min_std = oml_flights.select_dtypes(include=[np.number]).agg( - ["sum", "min", "std"], numeric_only=True + ["sum", "min", STANDARD_DEVIATION], numeric_only=True ) print(pd_sum_min_std.dtypes) @@ -94,10 +95,10 @@ def test_aggs_median_var(self): pd_aggs = pd_ecommerce[ ["taxful_total_price", "taxless_total_price", "total_quantity"] - ].agg(["median", "var"]) + ].agg(["median", VARIANCE]) oml_aggs = oml_ecommerce[ ["taxful_total_price", "taxless_total_price", "total_quantity"] - ].agg(["median", "var"], numeric_only=True) + ].agg(["median", VARIANCE], numeric_only=True) print(pd_aggs, pd_aggs.dtypes) print(oml_aggs, oml_aggs.dtypes) diff --git a/tests/dataframe/test_datetime_pytest.py b/tests/dataframe/test_datetime_pytest.py index 7dd3eb221..55dca3be2 100644 --- a/tests/dataframe/test_datetime_pytest.py +++ b/tests/dataframe/test_datetime_pytest.py @@ -124,6 +124,8 @@ def test_datetime_to_ms(self): os_refresh=True, ) + df["D"] = df["D"].astype("datetime64[ns]") + assert_series_equal(df.dtypes, oml_df.dtypes) assert_pandas_opensearch_py_ml_frame_equal(df, oml_df) diff --git a/tests/dataframe/test_describe_pytest.py b/tests/dataframe/test_describe_pytest.py index 8d0344d26..611f53212 100644 --- a/tests/dataframe/test_describe_pytest.py +++ b/tests/dataframe/test_describe_pytest.py @@ -23,18 +23,23 @@ # under the License. # File called _pytest for PyCharm compatability - +import pandas as pd from pandas.testing import assert_frame_equal from tests.common import TestData +PANDAS_MAJOR_VERSION = int(pd.__version__.split(".")[0]) + class TestDataFrameDescribe(TestData): def test_flights_describe(self): pd_flights = self.pd_flights() oml_flights = self.oml_flights() - pd_describe = pd_flights.describe() + if PANDAS_MAJOR_VERSION < 2: + pd_describe = pd_flights.describe() + else: + pd_describe = pd_flights.describe().drop(["timestamp"], axis=1) # We remove bool columns to match pandas output oml_describe = oml_flights.describe().drop( ["Cancelled", "FlightDelay"], axis="columns" diff --git a/tests/dataframe/test_groupby_pytest.py b/tests/dataframe/test_groupby_pytest.py index 3ee3fa7bc..89938b110 100644 --- a/tests/dataframe/test_groupby_pytest.py +++ b/tests/dataframe/test_groupby_pytest.py @@ -28,6 +28,12 @@ import pytest from pandas.testing import assert_frame_equal, assert_index_equal, assert_series_equal +from opensearch_py_ml.utils import ( + MEAN_ABSOLUTE_DEVIATION, + STANDARD_DEVIATION, + VARIANCE, + CustomFunctionDispatcher, +) from tests.common import TestData @@ -100,13 +106,22 @@ def test_groupby_aggs_numeric_only_true(self, pd_agg, dropna): ) @pytest.mark.parametrize("dropna", [True, False]) - @pytest.mark.parametrize("pd_agg", ["mad", "var", "std"]) + @pytest.mark.parametrize( + "pd_agg", [MEAN_ABSOLUTE_DEVIATION, VARIANCE, STANDARD_DEVIATION] + ) def test_groupby_aggs_mad_var_std(self, pd_agg, dropna): # For these aggs pandas doesn't support numeric_only pd_flights = self.pd_flights().filter(self.filter_data) oml_flights = self.oml_flights().filter(self.filter_data) - pd_groupby = getattr(pd_flights.groupby("Cancelled", dropna=dropna), pd_agg)() + if pd_agg in CustomFunctionDispatcher.customFunctionMap: + pd_groupby = pd_flights.groupby("Cancelled", dropna=dropna).agg( + lambda x: CustomFunctionDispatcher.apply_custom_function(pd_agg, x) + ) + else: + pd_groupby = getattr( + pd_flights.groupby("Cancelled", dropna=dropna), pd_agg + )() oml_groupby = getattr(oml_flights.groupby("Cancelled", dropna=dropna), pd_agg)( numeric_only=True ) @@ -224,15 +239,44 @@ def test_groupby_dataframe_mad(self): pd_flights = self.pd_flights().filter(self.filter_data + ["DestCountry"]) oml_flights = self.oml_flights().filter(self.filter_data + ["DestCountry"]) - pd_mad = pd_flights.groupby("DestCountry").mad() + pd_mad = pd_flights.groupby("DestCountry").apply( + lambda group: group.select_dtypes(include="number").apply( + lambda x: CustomFunctionDispatcher.apply_custom_function( + MEAN_ABSOLUTE_DEVIATION, x + ) + ) + ) + + # Re-merge non-numeric columns back, with suffixes to avoid column overlap + non_numeric_columns = ( + pd_flights.select_dtypes(exclude="number").groupby("DestCountry").first() + ) + pd_mad = pd_mad.join( + non_numeric_columns, lsuffix="_numeric", rsuffix="_non_numeric" + )[self.filter_data] + if "Cancelled" in pd_mad.columns: + pd_mad["Cancelled"] = pd_mad["Cancelled"].astype(float) oml_mad = oml_flights.groupby("DestCountry").mad() assert_index_equal(pd_mad.columns, oml_mad.columns) assert_index_equal(pd_mad.index, oml_mad.index) assert_series_equal(pd_mad.dtypes, oml_mad.dtypes) - pd_min_mad = pd_flights.groupby("DestCountry").aggregate(["min", "mad"]) - oml_min_mad = oml_flights.groupby("DestCountry").aggregate(["min", "mad"]) + pd_min_mad = pd_flights.groupby("DestCountry").agg( + [ + "min", + lambda x: CustomFunctionDispatcher.apply_custom_function( + MEAN_ABSOLUTE_DEVIATION, x + ), + ] + ) + + pd_min_mad.columns = pd_min_mad.columns.set_levels( + ["min", MEAN_ABSOLUTE_DEVIATION], level=1 + ) + oml_min_mad = oml_flights.groupby("DestCountry").aggregate( + ["min", MEAN_ABSOLUTE_DEVIATION] + ) assert_index_equal(pd_min_mad.columns, oml_min_mad.columns) assert_index_equal(pd_min_mad.index, oml_min_mad.index) diff --git a/tests/dataframe/test_metrics_pytest.py b/tests/dataframe/test_metrics_pytest.py index f3055ea5a..0077a9947 100644 --- a/tests/dataframe/test_metrics_pytest.py +++ b/tests/dataframe/test_metrics_pytest.py @@ -24,17 +24,22 @@ import numpy as np import pandas as pd - -# File called _pytest for PyCharm compatibility import pytest from pandas.testing import assert_frame_equal, assert_series_equal +# File called _pytest for PyCharm compatibility +from opensearch_py_ml.utils import ( + MEAN_ABSOLUTE_DEVIATION, + STANDARD_DEVIATION, + VARIANCE, + CustomFunctionDispatcher, +) from tests.common import TestData, assert_almost_equal class TestDataFrameMetrics(TestData): funcs = ["max", "min", "mean", "sum"] - extended_funcs = ["median", "mad", "var", "std"] + extended_funcs = ["median", MEAN_ABSOLUTE_DEVIATION, VARIANCE, STANDARD_DEVIATION] filter_data = [ "AvgTicketPrice", "Cancelled", @@ -81,9 +86,12 @@ def test_flights_extended_metrics(self): logger.setLevel(logging.DEBUG) for func in self.extended_funcs: - pd_metric = getattr(pd_flights, func)( - **({"numeric_only": True} if func != "mad" else {}) - ) + if func in CustomFunctionDispatcher.customFunctionMap: + pd_metric = CustomFunctionDispatcher.apply_custom_function( + func, pd_flights + ) + else: + pd_metric = getattr(pd_flights, func)(**({"numeric_only": True})) oml_metric = getattr(oml_flights, func)(numeric_only=True) pd_value = pd_metric["AvgTicketPrice"] @@ -101,7 +109,12 @@ def test_flights_extended_metrics_nan(self): ] for func in self.extended_funcs: - pd_metric = getattr(pd_flights_1, func)() + if func in CustomFunctionDispatcher.customFunctionMap: + pd_metric = pd_flights_1.apply( + lambda x: CustomFunctionDispatcher.apply_custom_function(func, x) + ) + else: + pd_metric = getattr(pd_flights_1, func)() oml_metric = getattr(oml_flights_1, func)(numeric_only=False) assert_series_equal(pd_metric, oml_metric, check_exact=False) @@ -111,7 +124,12 @@ def test_flights_extended_metrics_nan(self): oml_flights_0 = oml_flights[oml_flights.FlightNum == "XXX"][["AvgTicketPrice"]] for func in self.extended_funcs: - pd_metric = getattr(pd_flights_0, func)() + if func in CustomFunctionDispatcher.customFunctionMap: + pd_metric = pd_flights_0.apply( + lambda x: CustomFunctionDispatcher.apply_custom_function(func, x) + ) + else: + pd_metric = getattr(pd_flights_0, func)() oml_metric = getattr(oml_flights_0, func)(numeric_only=False) assert_series_equal(pd_metric, oml_metric, check_exact=False) @@ -177,9 +195,9 @@ def test_flights_datetime_metrics_agg(self): "min": pd.Timestamp("2018-01-01 00:00:00"), "mean": pd.Timestamp("2018-01-21 19:20:45.564438232"), "sum": pd.NaT, - "mad": pd.NaT, - "var": pd.NaT, - "std": pd.NaT, + MEAN_ABSOLUTE_DEVIATION: pd.NaT, + VARIANCE: pd.NaT, + STANDARD_DEVIATION: pd.NaT, "nunique": 12236, } @@ -288,7 +306,7 @@ def test_flights_numeric_only(self): agg_data = oml_flights.agg(filtered_aggs, numeric_only=True).transpose() for agg in filtered_aggs: # Explicitly check for mad because it returns nan for bools - if agg == "mad": + if agg == MEAN_ABSOLUTE_DEVIATION: assert np.isnan(agg_data[agg]["Cancelled"]) else: assert_series_equal( @@ -304,7 +322,7 @@ def test_numeric_only_true_single_aggs(self): for agg in self.funcs + self.extended_funcs: result = getattr(oml_flights, agg)(numeric_only=True) assert result.dtype == np.dtype("float64") - assert result.shape == ((3,) if agg != "mad" else (2,)) + assert result.shape == ((3,) if agg != MEAN_ABSOLUTE_DEVIATION else (2,)) # check dtypes and shape of min, max and median for numeric_only=False | None @pytest.mark.parametrize("agg", ["min", "max", "median"]) @@ -498,7 +516,8 @@ def test_flights_agg_quantile(self, numeric_only): ["AvgTicketPrice", "FlightDelayMin", "dayOfWeek"] ) - pd_quantile = pd_flights.agg(["quantile", "min"], numeric_only=numeric_only) + pd_quantile = pd_flights.agg([lambda x: x.quantile(0.5), lambda x: x.min()]) + pd_quantile.index = ["quantile", "min"] oml_quantile = oml_flights.agg(["quantile", "min"], numeric_only=numeric_only) assert_frame_equal( @@ -522,11 +541,19 @@ def test_flights_idx_on_index(self): pd_idxmin = list(pd_flights.idxmin()) oml_idxmin = list(oml_flights.idxmin()) - assert_frame_equal( - pd_flights.filter(items=pd_idxmin, axis=0).reset_index(), - oml_flights.filter(items=oml_idxmin, axis=0).to_pandas().reset_index(), + + pd_filtered_min = ( + pd_flights.filter(items=pd_idxmin, axis=0).reset_index().drop_duplicates() + ) + oml_filtered_min = ( + oml_flights.filter(items=oml_idxmin, axis=0) + .to_pandas() + .reset_index() + .drop_duplicates() ) + assert_frame_equal(pd_filtered_min, oml_filtered_min) + def test_flights_idx_on_columns(self): match = "This feature is not implemented yet for 'axis = 1'" with pytest.raises(NotImplementedError, match=match): diff --git a/tests/dataframe/test_utils_pytest.py b/tests/dataframe/test_utils_pytest.py index 5a244d4b2..e1e0498f3 100644 --- a/tests/dataframe/test_utils_pytest.py +++ b/tests/dataframe/test_utils_pytest.py @@ -44,11 +44,11 @@ def test_generate_os_mappings(self): "A": np.random.rand(3), "B": 1, "C": "foo", - "D": pd.Timestamp("20190102"), + "D": pd.to_datetime("2019-01-02"), "E": [1.0, 2.0, 3.0], "F": False, "G": [1, 2, 3], - "H": pd.Timestamp("20190102", tz="UTC"), + "H": pd.to_datetime("2019-01-02", utc=True), }, index=["0", "1", "2"], ) @@ -94,7 +94,7 @@ def test_pandas_to_oml_ignore_index(self): "A": np.random.rand(3), "B": 1, "C": "foo", - "D": pd.Timestamp("20190102"), + "D": pd.to_datetime("2019-01-02"), "E": [1.0, 2.0, 3.0], "F": False, "G": [1, 2, 3], diff --git a/tests/operations/test_map_pd_aggs_to_es_aggs_pytest.py b/tests/operations/test_map_pd_aggs_to_es_aggs_pytest.py index f5e4e8a96..c030eb890 100644 --- a/tests/operations/test_map_pd_aggs_to_es_aggs_pytest.py +++ b/tests/operations/test_map_pd_aggs_to_es_aggs_pytest.py @@ -23,6 +23,7 @@ # under the License. from opensearch_py_ml.operations import Operations +from opensearch_py_ml.utils import MEAN_ABSOLUTE_DEVIATION, STANDARD_DEVIATION, VARIANCE def test_all_aggs(): @@ -31,9 +32,9 @@ def test_all_aggs(): "min", "max", "mean", - "std", - "var", - "mad", + STANDARD_DEVIATION, + VARIANCE, + MEAN_ABSOLUTE_DEVIATION, "count", "nunique", "median", @@ -69,7 +70,7 @@ def test_extended_stats_optimization(): os_aggs = Operations._map_pd_aggs_to_os_aggs(["count", "nunique"]) assert os_aggs == ["value_count", "cardinality"] - for pd_agg in ["var", "std"]: + for pd_agg in [VARIANCE, STANDARD_DEVIATION]: extended_os_agg = Operations._map_pd_aggs_to_os_aggs([pd_agg])[0] os_aggs = Operations._map_pd_aggs_to_os_aggs([pd_agg, "nunique"]) diff --git a/tests/series/test_arithmetics_pytest.py b/tests/series/test_arithmetics_pytest.py index 18226675c..196e5d38d 100644 --- a/tests/series/test_arithmetics_pytest.py +++ b/tests/series/test_arithmetics_pytest.py @@ -80,9 +80,7 @@ def to_pandas(self): # "type cast" to modified class (inherits from ed.Series) that overrides the `to_pandas` function oml_series.__class__ = ModifiedOMLSeries - assert_pandas_opensearch_py_ml_series_equal( - pd_series, oml_series, check_less_precise=True - ) + assert_pandas_opensearch_py_ml_series_equal(pd_series, oml_series) def test_ecommerce_series_invalid_div(self): pd_df = self.pd_ecommerce() diff --git a/tests/series/test_describe_pytest.py b/tests/series/test_describe_pytest.py index b0ad65602..0841253d4 100644 --- a/tests/series/test_describe_pytest.py +++ b/tests/series/test_describe_pytest.py @@ -24,6 +24,7 @@ import pandas as pd +from opensearch_py_ml.utils import STANDARD_DEVIATION from tests.common import TestData, assert_series_equal @@ -42,7 +43,7 @@ def test_series_describe(self): # Percentiles calculations vary for Elasticsearch assert_series_equal( - oml_desc[["count", "mean", "std", "min", "max"]], - pd_desc[["count", "mean", "std", "min", "max"]], + oml_desc[["count", "mean", STANDARD_DEVIATION, "min", "max"]], + pd_desc[["count", "mean", STANDARD_DEVIATION, "min", "max"]], rtol=0.2, ) diff --git a/tests/series/test_metrics_pytest.py b/tests/series/test_metrics_pytest.py index 71037d4da..bc3330eb2 100644 --- a/tests/series/test_metrics_pytest.py +++ b/tests/series/test_metrics_pytest.py @@ -31,15 +31,30 @@ import pytest from pandas.testing import assert_series_equal +from opensearch_py_ml.utils import ( + MEAN_ABSOLUTE_DEVIATION, + STANDARD_DEVIATION, + VARIANCE, + CustomFunctionDispatcher, +) from tests.common import TestData, assert_almost_equal class TestSeriesMetrics(TestData): - all_funcs = ["max", "min", "mean", "sum", "nunique", "var", "std", "mad"] + all_funcs = [ + "max", + "min", + "mean", + "sum", + "nunique", + VARIANCE, + STANDARD_DEVIATION, + MEAN_ABSOLUTE_DEVIATION, + ] timestamp_funcs = ["max", "min", "mean", "nunique"] def assert_almost_equal_for_agg(self, func, pd_metric, oml_metric): - if func in ("nunique", "var", "mad"): + if func in ("nunique", VARIANCE, MEAN_ABSOLUTE_DEVIATION): np.testing.assert_almost_equal(pd_metric, oml_metric, decimal=-3) else: np.testing.assert_almost_equal(pd_metric, oml_metric, decimal=2) @@ -49,7 +64,12 @@ def test_flights_metrics(self): oml_flights = self.oml_flights()["AvgTicketPrice"] for func in self.all_funcs: - pd_metric = getattr(pd_flights, func)() + if func in CustomFunctionDispatcher.customFunctionMap: + pd_metric = pd_flights.agg( + lambda x: CustomFunctionDispatcher.apply_custom_function(func, x) + ) + else: + pd_metric = getattr(pd_flights, func)() oml_metric = getattr(oml_flights, func)() self.assert_almost_equal_for_agg(func, pd_metric, oml_metric) @@ -94,7 +114,14 @@ def test_ecommerce_selected_all_numeric_source_fields(self): oml_ecommerce = self.oml_ecommerce()[column] for func in self.all_funcs: - pd_metric = getattr(pd_ecommerce, func)() + if func in CustomFunctionDispatcher.customFunctionMap: + pd_metric = pd_ecommerce.agg( + lambda x: CustomFunctionDispatcher.apply_custom_function( + func, x + ) + ) + else: + pd_metric = getattr(pd_ecommerce, func)() oml_metric = getattr(oml_ecommerce, func)( **({"numeric_only": True} if (func != "nunique") else {}) ) diff --git a/tests/test_utils_pytest.py b/tests/test_utils_pytest.py index 53f746f26..11e361a59 100644 --- a/tests/test_utils_pytest.py +++ b/tests/test_utils_pytest.py @@ -24,7 +24,10 @@ # File called _pytest for PyCharm compatibility -from opensearch_py_ml.utils import is_valid_attr_name +import pandas as pd +import pytest + +from opensearch_py_ml.utils import is_valid_attr_name, to_list, to_list_if_needed class TestUtils: @@ -40,3 +43,34 @@ def test_is_valid_attr_name(self): assert not is_valid_attr_name(None) assert not is_valid_attr_name("4pizze") assert not is_valid_attr_name("pizza+") + + def test_to_list_if_needed_index(self): + index = pd.Index([1, 2, 3]) + result = to_list_if_needed(index) + assert result == [1, 2, 3], "Expected Index to convert to list correctly" + + def test_to_list_if_needed_single_value(self): + result = to_list_if_needed(42) + assert result == [42], "Expected single value to be wrapped in list" + + def test_to_list_if_needed_list(self): + result = to_list_if_needed([1, 2, 3]) + assert result == [1, 2, 3], "Expected list-like input to return as-is" + + def test_to_list_if_needed_none(self): + result = to_list_if_needed(None) + assert result is None, "Expected None to return None" + + def test_to_list_series(self): + series = pd.Series([4, 5, 6]) + result = to_list(series) + assert result == [4, 5, 6], "Expected Series to convert to list" + + def test_to_list_collection(self): + collection = {7, 8, 9} + result = to_list(collection) + assert sorted(result) == [7, 8, 9], "Expected Collection to convert to list" + + def test_to_list_not_implemented(self): + with pytest.raises(NotImplementedError): + to_list(12345) # Passing an integer should raise an error