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

Migrate lists/extract to pylibcudf #16071

Merged
merged 17 commits into from
Jul 10, 2024
Merged
Changes from 15 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
36 changes: 10 additions & 26 deletions python/cudf/cudf/_lib/lists.pyx
Original file line number Diff line number Diff line change
@@ -8,11 +8,9 @@ from libcpp.utility cimport move

from cudf._lib.column cimport Column
from cudf._lib.pylibcudf.libcudf.column.column cimport column
from cudf._lib.pylibcudf.libcudf.column.column_view cimport column_view
from cudf._lib.pylibcudf.libcudf.lists.count_elements cimport (
count_elements as cpp_count_elements,
)
from cudf._lib.pylibcudf.libcudf.lists.extract cimport extract_list_element
from cudf._lib.pylibcudf.libcudf.lists.lists_column_view cimport (
lists_column_view,
)
@@ -116,37 +114,23 @@ def sort_lists(Column col, bool ascending, str na_position):

@acquire_spill_lock()
def extract_element_scalar(Column col, size_type index):
# shared_ptr required because lists_column_view has no default
# ctor
cdef shared_ptr[lists_column_view] list_view = (
make_shared[lists_column_view](col.view())
return Column.from_pylibcudf(
pylibcudf.lists.extract_list_element(
col.to_pylibcudf(mode="read"),
index,
)
)

cdef unique_ptr[column] c_result

with nogil:
c_result = move(extract_list_element(list_view.get()[0], index))

result = Column.from_unique_ptr(move(c_result))
return result


@acquire_spill_lock()
def extract_element_column(Column col, Column index):
cdef shared_ptr[lists_column_view] list_view = (
make_shared[lists_column_view](col.view())
return Column.from_pylibcudf(
pylibcudf.lists.extract_list_element(
col.to_pylibcudf(mode="read"),
index.to_pylibcudf(mode="read"),
)
)

cdef column_view index_view = index.view()

cdef unique_ptr[column] c_result

with nogil:
c_result = move(extract_list_element(list_view.get()[0], index_view))

result = Column.from_unique_ptr(move(c_result))
return result


@acquire_spill_lock()
def contains_scalar(Column col, py_search_key):
6 changes: 3 additions & 3 deletions python/cudf/cudf/_lib/pylibcudf/libcudf/lists/extract.pxd
Original file line number Diff line number Diff line change
@@ -11,10 +11,10 @@ from cudf._lib.pylibcudf.libcudf.types cimport size_type

cdef extern from "cudf/lists/extract.hpp" namespace "cudf::lists" nogil:
cdef unique_ptr[column] extract_list_element(
const lists_column_view,
const lists_column_view&,
size_type
) except +
cdef unique_ptr[column] extract_list_element(
const lists_column_view,
column_view
const lists_column_view&,
const column_view&
) except +
6 changes: 6 additions & 0 deletions python/cudf/cudf/_lib/pylibcudf/lists.pxd
Original file line number Diff line number Diff line change
@@ -12,6 +12,10 @@ ctypedef fused ColumnOrScalar:
Column
Scalar

ctypedef fused ColumnOrSizeType:
Column
size_type

cpdef Table explode_outer(Table, size_type explode_column_idx)

cpdef Column concatenate_rows(Table)
@@ -25,3 +29,5 @@ cpdef Column contains_nulls(Column)
cpdef Column index_of(Column, ColumnOrScalar, bool)

cpdef Column reverse(Column)

cpdef Column extract_list_element(Column, ColumnOrSizeType)
31 changes: 30 additions & 1 deletion python/cudf/cudf/_lib/pylibcudf/lists.pyx
Original file line number Diff line number Diff line change
@@ -16,9 +16,12 @@ from cudf._lib.pylibcudf.libcudf.lists.combine cimport (
concatenate_null_policy,
concatenate_rows as cpp_concatenate_rows,
)
from cudf._lib.pylibcudf.libcudf.lists.extract cimport (
extract_list_element as cpp_extract_list_element,
)
from cudf._lib.pylibcudf.libcudf.table.table cimport table
from cudf._lib.pylibcudf.libcudf.types cimport size_type
from cudf._lib.pylibcudf.lists cimport ColumnOrScalar
from cudf._lib.pylibcudf.lists cimport ColumnOrScalar, ColumnOrSizeType

from .column cimport Column, ListColumnView
from .scalar cimport Scalar
@@ -232,3 +235,29 @@ cpdef Column reverse(Column input):
list_view.view(),
))
return Column.from_libcudf(move(c_result))


cpdef Column extract_list_element(Column input, ColumnOrSizeType index):
"""Create a column of extracted list elements.

Parameters
----------
input : Column
The input column.
index : Union[Column, size_type]
The selection index or indices.

Returns
-------
Column
A new Column with elements extracted.
"""
cdef unique_ptr[column] c_result
cdef ListColumnView list_view = input.list_view()

with nogil:
c_result = move(cpp_extract_list_element(
list_view.view(),
index.view() if ColumnOrSizeType is Column else index,
))
return Column.from_libcudf(move(c_result))
21 changes: 21 additions & 0 deletions python/cudf/cudf/pylibcudf_tests/test_lists.py
Original file line number Diff line number Diff line change
@@ -146,3 +146,24 @@ def test_reverse(test_data):
expect = pa.array([lst[::-1] for lst in list_column])

assert_column_eq(expect, res)


def test_extract_list_element_scalar(test_data):
arr = pa.array(test_data[0][0])
plc_column = plc.interop.from_arrow(arr)

res = plc.lists.extract_list_element(plc_column, 0)
expect = pa.compute.list_element(test_data[0][0], 0)

assert_column_eq(expect, res)


def test_extract_list_element_column(test_data):
arr = pa.array(test_data[0][0])
plc_column = plc.interop.from_arrow(arr)
indices = plc.interop.from_arrow(pa.array([0, 1, -4, -1]))

res = plc.lists.extract_list_element(plc_column, indices)
expect = pa.array([0, None, None, 7])

assert_column_eq(expect, res)