Skip to content

Commit

Permalink
fix(duckdb): handle null-typed scalars in to_pyarrow()
Browse files Browse the repository at this point in the history
  • Loading branch information
NickCrews authored and cpcloud committed Dec 31, 2024
1 parent 9f94ad4 commit ad81580
Show file tree
Hide file tree
Showing 3 changed files with 47 additions and 29 deletions.
6 changes: 6 additions & 0 deletions ibis/backends/duckdb/converter.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,12 @@ def convert_Array(s, dtype, pandas_type):


class DuckDBPyArrowData(PyArrowData):
@classmethod
def convert_scalar(cls, scalar: pa.Scalar, dtype: dt.DataType) -> pa.Scalar:
if dtype.is_null():
return pa.scalar(None)
return super().convert_scalar(scalar, dtype)

@classmethod
def convert_column(cls, column: pa.Array, dtype: dt.DataType) -> pa.Array:
if dtype.is_null():
Expand Down
23 changes: 12 additions & 11 deletions ibis/backends/duckdb/tests/test_client.py
Original file line number Diff line number Diff line change
Expand Up @@ -445,7 +445,16 @@ def test_pyarrow_batches_chunk_size(con): # 10443
next(batches)


def test_create_table_with_nulls(con):
@pytest.mark.parametrize(
"kwargs",
[
dict(obj=ibis.memtable({"a": [None]})),
dict(obj=ibis.memtable({"a": [None]}), schema=ibis.schema({"a": "null"})),
dict(schema=ibis.schema({"a": "null"})),
],
ids=["obj", "obj-schema", "schema"],
)
def test_create_table_with_nulls(con, kwargs):
t = ibis.memtable({"a": [None]})
schema = t.schema()

Expand All @@ -454,13 +463,5 @@ def test_create_table_with_nulls(con):

name = gen_name("duckdb_all_nulls")

match = "NULL typed columns"

with pytest.raises(com.IbisTypeError, match=match):
con.create_table(name, obj=t)

with pytest.raises(com.IbisTypeError, match=match):
con.create_table(name, obj=t, schema=schema)

with pytest.raises(com.IbisTypeError, match=match):
con.create_table(name, schema=schema)
with pytest.raises(com.IbisTypeError, match="NULL typed columns"):
con.create_table(name, **kwargs)
47 changes: 29 additions & 18 deletions ibis/backends/tests/test_export.py
Original file line number Diff line number Diff line change
Expand Up @@ -624,24 +624,35 @@ def test_scalar_to_memory(limit, awards_players, output_format, converter):
assert converter(res) is None


# flink
@pytest.mark.notyet(
[
"clickhouse",
"exasol",
"flink",
"impala",
"mssql",
"mysql",
"oracle",
"postgres",
"risingwave",
"trino",
],
raises=com.IbisTypeError,
reason="unable to handle null typed columns as input",
)
def test_all_null_column(con):
base_notyet_nulls = [
"exasol",
"flink",
"impala",
"mssql",
"mysql",
"oracle",
"postgres",
"risingwave",
"trino",
]


@pytest.mark.notyet(["clickhouse", *base_notyet_nulls], raises=com.IbisTypeError)
def test_all_null_table(con):
t = ibis.memtable({"a": [None]})
result = con.to_pyarrow(t)
assert pat.is_null(result["a"].type)


@pytest.mark.notyet(["clickhouse", *base_notyet_nulls], raises=com.IbisTypeError)
def test_all_null_column(con):
t = ibis.memtable({"a": [None]})
result = con.to_pyarrow(t.a)
assert pat.is_null(result.type)


@pytest.mark.notyet(base_notyet_nulls, raises=com.IbisTypeError)
def test_all_null_scalar(con):
e = ibis.literal(None)
result = con.to_pyarrow(e)
assert pat.is_null(result.type)

0 comments on commit ad81580

Please sign in to comment.