From 2420abf7397e081a81efed48527f479f0f819306 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ragnar=20Dahl=C3=A9n?= Date: Tue, 15 Oct 2024 18:15:12 +0200 Subject: [PATCH] Ensure custom types support pickling. Previously `NamedRowTuple` did not roundtrip successfully. Added tests for all custom types. Fixes #490. --- tests/unit/test_types.py | 39 +++++++++++++++++++++++++++++++++++++++ trino/types.py | 9 +++++++++ 2 files changed, 48 insertions(+) create mode 100644 tests/unit/test_types.py diff --git a/tests/unit/test_types.py b/tests/unit/test_types.py new file mode 100644 index 00000000..e51da48a --- /dev/null +++ b/tests/unit/test_types.py @@ -0,0 +1,39 @@ +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +import pickle +from datetime import datetime, time +from decimal import Decimal + +import pytest + +from trino import types + + +def identity(x): + return x + + +type_instances = [ + (types.Time(time(11, 47, 23), Decimal(0.314)), lambda v: v.to_python_type()), + (types.TimeWithTimeZone(time(11, 47, 23), Decimal(0.314)), lambda v: v.to_python_type()), + (types.Timestamp(datetime(2024, 10, 15, 11, 47, 23), Decimal(0.314)), lambda v: v.to_python_type()), + (types.TimestampWithTimeZone(datetime(2024, 10, 15, 11, 47, 23), Decimal(0.314)), lambda v: v.to_python_type()), + (types.NamedRowTuple(["Alice", 38], ["name", "age"], ["varchar", "integer"]), identity), +] + + +@pytest.mark.parametrize("value,fn", type_instances) +def test_pickle_roundtripping(value, fn): + bytes = pickle.dumps(value) + unpickled_value = pickle.loads(bytes) + assert fn(value) == fn(unpickled_value) diff --git a/trino/types.py b/trino/types.py index 0807c2fc..91bcb2ed 100644 --- a/trino/types.py +++ b/trino/types.py @@ -111,5 +111,14 @@ def __getattr__(self, name: str) -> Any: if self._names.count(name): raise ValueError("Ambiguous row field reference: " + name) + def __getnewargs__(self) -> Any: + return (tuple(self), (), ()) + + def __getstate__(self) -> Any: + return vars(self) + + def __setstate__(self, state: Any) -> None: + vars(self).update(state) + def __repr__(self) -> str: return self._repr