diff --git a/docs/.vuepress/sidebar.ts b/docs/.vuepress/sidebar.ts index cab634c..81da5e8 100644 --- a/docs/.vuepress/sidebar.ts +++ b/docs/.vuepress/sidebar.ts @@ -38,6 +38,7 @@ export default sidebar({ collapsible: true, children: [ "supported_types", + "array_types", "extra_types", "advanced_type_usage", ] @@ -75,5 +76,10 @@ export default sidebar({ prefix: "/benchmarks", link: "/benchmarks.md" }, + { + text: "FAQ", + prefix: "/faq", + link: "/faq.md" + }, ], }); diff --git a/docs/components/connection.md b/docs/components/connection.md index 959978b..7a470b0 100644 --- a/docs/components/connection.md +++ b/docs/components/connection.md @@ -184,9 +184,10 @@ async def main() -> None: - `isolation_level`: level of isolation. Default how it is in PostgreSQL. - `read_variant`: configure read variant of the transaction. Default how it is in PostgreSQL. - `deferrable`: configure deferrable of the transaction. Default how it is in PostgreSQL. +- `synchronous_commit`: configure [synchronous_commit](https://postgresqlco.nf/doc/en/param/synchronous_commit/) option for transaction. Default how it is in PostgreSQL. ```python -from psqlpy import IsolationLevel, ReadVariant +from psqlpy import IsolationLevel, ReadVariant, SynchronousCommit async def main() -> None: ... @@ -195,6 +196,7 @@ async def main() -> None: isolation_level=IsolationLevel.Serializable, read_variant=ReadVariant.ReadWrite, deferrable=True, + synchronous_commit=SynchronousCommit.On, ) ``` diff --git a/docs/components/connection_pool.md b/docs/components/connection_pool.md index 1d77f3d..1536961 100644 --- a/docs/components/connection_pool.md +++ b/docs/components/connection_pool.md @@ -192,7 +192,7 @@ Parameters must be passed as list after querystring. ::: caution You must use `ConnectionPool.execute` method in high-load production code wisely! It pulls connection from the pool each time you execute query. -Preferable way to execute statements with [Connection](./../../introduction/components/connection.md) or [Transaction](./../../introduction/components/transaction.md) +Preferable way to execute statements with [Connection](./../components/connection.md) or [Transaction](./../components/transaction.md) ::: ```python diff --git a/docs/faq.md b/docs/faq.md new file mode 100644 index 0000000..a6b04bc --- /dev/null +++ b/docs/faq.md @@ -0,0 +1,128 @@ +--- +title: Frequently asked questions +--- + +Here you can find most common questions and problems. + +### LIMIT of OFFSET isn't working +The main problem is PostgreSQL expects `LIMIT` and `OFFSET` to be BIGINT type but when you pass python `int` into `parameters` it converts to `INTEGER`. + +#### Problem and Solution: +```python +from psqlpy import ConnectionPool +from psqlpy.extra_types import BigInt + +# --- Incorrect --- +async def main() -> None: + pool = ConnectionPool() + await pool.execute( + querystring="SELECT * FROM users LIMIT $1 OFFSET $2", + parameters=[10, 100], + ) + + +# --- Correct --- +async def main() -> None: + pool = ConnectionPool() + await pool.execute( + querystring="SELECT * FROM users LIMIT $1 OFFSET $2", + parameters=[BigInt(10), BigInt(100)], + ) +``` + +### WHERE IN clause isn't working +Instead of using `WHERE IN ()` clause you must use `WHERE = ANY()`. + +#### Problem and Solution: +```python +from psqlpy import ConnectionPool + +# --- Incorrect --- +async def main() -> None: + pool = ConnectionPool() + await pool.execute( + querystring="SELECT * FROM users WHERE id IN ($1)", + parameters=[ + (1, 2, 3), + ], + ) + + +# --- Correct --- +async def main() -> None: + pool = ConnectionPool() + await pool.execute( + querystring="SELECT * FROM users WHERE id = ANY($1)", + parameters=[ + (1, 2, 3), + ], + ) +``` + +### Wrong binary data + +Example error: `binary data has array element type 1043 (character varying) instead of expected 25 (text)`. + +This exception tells you that you use wrong data type and you need to specify types explicitly. + +For example, when we want to make `WHERE` clause with `ANY` and string values, we need to use `TextArray`, see example below: + +#### Problem and Solution: +```python +from psqlpy import ConnectionPool +from psqlpy.extra_types import TextArray + +# --- Incorrect --- +async def main() -> None: + pool = ConnectionPool() + await pool.execute( + querystring="SELECT * FROM users WHERE name = ANY($1)", + parameters=[ + ["Foo", "Bar", "Cafe"], + ], + ) + + +# --- Correct --- +async def main() -> None: + pool = ConnectionPool() + await pool.execute( + querystring="SELECT * FROM users WHERE name = ANY($1)", + parameters=[ + TextArray(["Foo", "Bar", "Cafe"]), + ], + ) +``` + +### Cannot insert empty ARRAY + +To insert empty array use explicit [Array Type](./usage/types/array_types.md). + +#### Problem and Solution: +Let's assume that we have table `arr_table` with field `some_array` of `VARCHAR ARRAY` type. +The main problem that we cannot determine the type of the empty sequence passed from Python side. +```python +from psqlpy import ConnectionPool +from psqlpy.extra_types import VarCharArray + +# --- Incorrect --- +async def main() -> None: + pool = ConnectionPool() + await pool.execute( + querystring="INSERT INTO arr_table (some_array) VALUES ($1)", + parameters=[ + [], + ], + ) + + +# --- Correct --- +async def main() -> None: + pool = ConnectionPool() + await pool.execute( + querystring="INSERT INTO arr_table (some_array) VALUES ($1)", + parameters=[ + VarCharArray([]), + ], + ) +``` diff --git a/docs/usage/types/array_types.md b/docs/usage/types/array_types.md new file mode 100644 index 0000000..89261bd --- /dev/null +++ b/docs/usage/types/array_types.md @@ -0,0 +1,50 @@ +--- +title: Array Types +--- +For type safety and better performance we have predefined array types. + +| PSQLPy Array Type | PostgreSQL Array Type | +| :---: | :---: | +| BoolArray | BOOLEAN ARRAY | +| UUIDArray | UUID ARRAY | +| VarCharArray | VarChar ARRAY | +| TextArray | Text ARRAY | +| Int16Array | INT2 ARRAY | +| Int32Array | INT4 ARRAY | +| Int64Array | INT8 ARRAY | +| Float32Array | FLOAT4 ARRAY | +| Float64Array | FLOAT8 ARRAY | +| MoneyArray | MONEY ARRAY | +| IpAddressArray | INET ARRAY | +| JSONBArray | JSONB ARRAY | +| JSONArray | JSON ARRAY | +| DateArray | DATE ARRAY | +| TimeArray | TIME ARRAY | +| DateTimeArray | TIMESTAMP ARRAY | +| DateTimeTZArray | TIMESTAMPTZ ARRAY | +| MacAddr6Array | MACADDR ARRAY | +| MacAddr8Array | MACADDR8 ARRAY | +| NumericArray | NUMERIC ARRAY | +| PointArray | POINT ARRAY | +| BoxArray | BOX ARRAY | +| PathArray | PATH ARRAY | +| LineArray | LINE ARRAY | +| LsegArray | LSEG ARRAY | +| CircleArray | CIRCLE ARRAY | + +### Example: + +```python +from psqlpy import ConnectionPool +from psqlpy.extra_types import TextArray + + +async def main() -> None: + pool = ConnectionPool() + result = await pool.execute( + querystring="SELECT * FROM users WHERE id = ANY($1)", + parameters=[ + TextArray([1, 2, 3]), + ] + ) +``` diff --git a/docs/usage/types/supported_types.md b/docs/usage/types/supported_types.md index 70ee8a7..3754df6 100644 --- a/docs/usage/types/supported_types.md +++ b/docs/usage/types/supported_types.md @@ -50,6 +50,7 @@ DECIMAL PostgreSQL type isn't supported, use NUMERIC instead. ## Array Type You can make arrays with any type of `Simple Type`s. +For better performance and type safety we recommend to use predefined [Array Types](./array_types.md). #### Example: ```sql diff --git a/python/psqlpy/_internal/extra_types.pyi b/python/psqlpy/_internal/extra_types.pyi index fb7ba39..7bd5866 100644 --- a/python/psqlpy/_internal/extra_types.pyi +++ b/python/psqlpy/_internal/extra_types.pyi @@ -1,4 +1,8 @@ import typing +from datetime import date, datetime, time +from decimal import Decimal +from ipaddress import IPv4Address, IPv6Address +from uuid import UUID from typing_extensions import Self @@ -251,3 +255,503 @@ class PyCircle: ### Parameters: - `value`: any valid structure with int/float numbers. """ + +class BoolArray: + """Represent BOOLEAN ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + bool, + typing.Sequence[bool], + typing.Any, + ], + ], + ) -> None: + """Create new instance of BoolArray. + + ### Parameters: + - `inner`: inner value, sequence of UUID values. + """ + +class UUIDArray: + """Represent UUID ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + UUID, + typing.Sequence[UUID], + typing.Any, + ], + ], + ) -> None: + """Create new instance of UuidArray. + + ### Parameters: + - `inner`: inner value, sequence of UUID values. + """ + +class VarCharArray: + """Represent VarChar ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + str, + typing.Sequence[str], + typing.Any, + ], + ], + ) -> None: + """Create new instance of VarCharArray. + + ### Parameters: + - `inner`: inner value, sequence of str values. + """ + +class TextArray: + """Represent Text ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + str, + typing.Sequence[str], + typing.Any, + ], + ], + ) -> None: + """Create new instance of TextArray. + + ### Parameters: + - `inner`: inner value, sequence of str values. + """ + +class Int16Array: + """Represent INT2 ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + int, + typing.Sequence[int], + typing.Any, + ], + ], + ) -> None: + """Create new instance of Int16Array. + + ### Parameters: + - `inner`: inner value, sequence of int values. + """ + +class Int32Array: + """Represent INT4 ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + int, + typing.Sequence[int], + typing.Any, + ], + ], + ) -> None: + """Create new instance of Int32Array. + + ### Parameters: + - `inner`: inner value, sequence of int values. + """ + +class Int64Array: + """Represent INT8 ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + int, + typing.Sequence[int], + typing.Any, + ], + ], + ) -> None: + """Create new instance of Int64Array. + + ### Parameters: + - `inner`: inner value, sequence of int values. + """ + +class Float32Array: + """Represent FLOAT4 ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + float, + typing.Sequence[float], + typing.Any, + ], + ], + ) -> None: + """Create new instance of Float32Array. + + ### Parameters: + - `inner`: inner value, sequence of float values. + """ + +class Float64Array: + """Represent FLOAT8 ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + float, + typing.Sequence[float], + typing.Any, + ], + ], + ) -> None: + """Create new instance of Float64Array. + + ### Parameters: + - `inner`: inner value, sequence of float values. + """ + +class MoneyArray: + """Represent MONEY ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + int, + typing.Sequence[int], + typing.Any, + ], + ], + ) -> None: + """Create new instance of MoneyArray. + + ### Parameters: + - `inner`: inner value, sequence of int values. + """ + +class IpAddressArray: + """Represent INET ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + IPv4Address, + IPv6Address, + typing.Sequence[IPv4Address], + typing.Sequence[IPv6Address], + typing.Any, + ], + ], + ) -> None: + """Create new instance of IpAddressArray. + + ### Parameters: + - `inner`: inner value, sequence of IPv4Address/IPv6Address values. + """ + +class JSONBArray: + """Represent JSONB ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + typing.Dict[str, typing.Any], + PyJSONB, + typing.Sequence[typing.Dict[str, typing.Any]], + typing.Sequence[PyJSONB], + typing.Sequence[typing.Any], + ] + ], + ) -> None: + """Create new instance of JSONBArray. + + ### Parameters: + - `inner`: inner value, sequence of values. + """ + +class JSONArray: + """Represent JSON ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + typing.Dict[str, typing.Any], + PyJSON, + typing.Sequence[typing.Dict[str, typing.Any]], + typing.Sequence[PyJSON], + typing.Sequence[typing.Any], + ] + ], + ) -> None: + """Create new instance of JSONArray. + + ### Parameters: + - `inner`: inner value, sequence of values. + """ + +class DateArray: + """Represent DATE ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + date, + typing.Sequence[date], + typing.Any, + ], + ], + ) -> None: + """Create new instance of DateArray. + + ### Parameters: + - `inner`: inner value, sequence of date values. + """ + +class TimeArray: + """Represent TIME ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + time, + typing.Sequence[time], + typing.Any, + ], + ], + ) -> None: + """Create new instance of DateArray. + + ### Parameters: + - `inner`: inner value, sequence of time values. + """ + +class DateTimeArray: + """Represent TIMESTAMP ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + datetime, + typing.Sequence[datetime], + typing.Any, + ], + ], + ) -> None: + """Create new instance of DateArray. + + ### Parameters: + - `inner`: inner value, sequence of datetime values. + """ + +class DateTimeTZArray: + """Represent TIMESTAMPTZ ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + datetime, + typing.Sequence[datetime], + typing.Any, + ], + ], + ) -> None: + """Create new instance of DateArray. + + ### Parameters: + - `inner`: inner value, sequence of datetime values. + """ + +class MacAddr6Array: + """Represent MACADDR ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + PyMacAddr6, + typing.Sequence[PyMacAddr6], + typing.Any, + ], + ], + ) -> None: + """Create new instance of MacAddr6Array. + + ### Parameters: + - `inner`: inner value, sequence of PyMacAddr6 values. + """ + +class MacAddr8Array: + """Represent MACADDR8 ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + PyMacAddr8, + typing.Sequence[PyMacAddr8], + typing.Any, + ], + ], + ) -> None: + """Create new instance of MacAddr8Array. + + ### Parameters: + - `inner`: inner value, sequence of PyMacAddr8 values. + """ + +class NumericArray: + """Represent NUMERIC ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + Decimal, + typing.Sequence[Decimal], + typing.Any, + ], + ], + ) -> None: + """Create new instance of NumericArray. + + ### Parameters: + - `inner`: inner value, sequence of Decimal values. + """ + +class PointArray: + """Represent POINT ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + PyPoint, + typing.Sequence[PyPoint], + typing.Any, + ], + ], + ) -> None: + """Create new instance of PointArray. + + ### Parameters: + - `inner`: inner value, sequence of PyPoint values. + """ + +class BoxArray: + """Represent BOX ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + PyBox, + typing.Sequence[PyBox], + typing.Any, + ], + ], + ) -> None: + """Create new instance of BoxArray. + + ### Parameters: + - `inner`: inner value, sequence of PyBox values. + """ + +class PathArray: + """Represent PATH ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + PyPath, + typing.Sequence[PyPath], + typing.Any, + ], + ], + ) -> None: + """Create new instance of PathArray. + + ### Parameters: + - `inner`: inner value, sequence of PyPath values. + """ + +class LineArray: + """Represent LINE ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + PyLine, + typing.Sequence[PyLine], + typing.Any, + ], + ], + ) -> None: + """Create new instance of LineArray. + + ### Parameters: + - `inner`: inner value, sequence of PyLine values. + """ + +class LsegArray: + """Represent LSEG ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + PyLineSegment, + typing.Sequence[PyLineSegment], + typing.Any, + ], + ], + ) -> None: + """Create new instance of LsegArray. + + ### Parameters: + - `inner`: inner value, sequence of PyLineSegment values. + """ + +class CircleArray: + """Represent CIRCLE ARRAY in PostgreSQL.""" + + def __init__( + self: Self, + inner: typing.Sequence[ + typing.Union[ + PyCircle, + typing.Sequence[PyCircle], + typing.Any, + ], + ], + ) -> None: + """Create new instance of CircleArray. + + ### Parameters: + - `inner`: inner value, sequence of PyLineSegment values. + """ diff --git a/python/psqlpy/extra_types.py b/python/psqlpy/extra_types.py index c9f80c5..309a9f4 100644 --- a/python/psqlpy/extra_types.py +++ b/python/psqlpy/extra_types.py @@ -1,9 +1,31 @@ from ._internal.extra_types import ( BigInt, + BoolArray, + BoxArray, + CircleArray, + DateArray, + DateTimeArray, + DateTimeTZArray, Float32, + Float32Array, Float64, + Float64Array, + Int16Array, + Int32Array, + Int64Array, Integer, + IpAddressArray, + JSONArray, + JSONBArray, + LineArray, + LsegArray, + MacAddr6Array, + MacAddr8Array, Money, + MoneyArray, + NumericArray, + PathArray, + PointArray, PyBox, PyCircle, PyCustomType, @@ -18,6 +40,10 @@ PyText, PyVarChar, SmallInt, + TextArray, + TimeArray, + UUIDArray, + VarCharArray, ) __all__ = [ @@ -40,4 +66,34 @@ "PyLine", "PyLineSegment", "PyCircle", + "BoolArray", + "UUIDArray", + "JSONBArray", + "JSONArray", + "BoolArray", + "UUIDArray", + "VarCharArray", + "TextArray", + "Int16Array", + "Int32Array", + "Int64Array", + "Float32Array", + "Float64Array", + "MoneyArray", + "IpAddressArray", + "JSONBArray", + "JSONArray", + "DateArray", + "TimeArray", + "DateTimeArray", + "DateTimeTZArray", + "MacAddr6Array", + "MacAddr8Array", + "NumericArray", + "PointArray", + "BoxArray", + "PathArray", + "LineArray", + "LsegArray", + "CircleArray", ] diff --git a/python/tests/test_value_converter.py b/python/tests/test_value_converter.py index 8fe3199..5f492db 100644 --- a/python/tests/test_value_converter.py +++ b/python/tests/test_value_converter.py @@ -14,10 +14,29 @@ from psqlpy.exceptions import PyToRustValueMappingError from psqlpy.extra_types import ( BigInt, + BoolArray, + BoxArray, + CircleArray, + DateArray, + DateTimeArray, + DateTimeTZArray, Float32, Float64, + Float64Array, + Int16Array, + Int32Array, + Int64Array, Integer, + IpAddressArray, + JSONArray, + JSONBArray, + LineArray, + LsegArray, Money, + MoneyArray, + NumericArray, + PathArray, + PointArray, PyBox, PyCircle, PyCustomType, @@ -31,6 +50,10 @@ PyPoint, PyText, SmallInt, + TextArray, + TimeArray, + UUIDArray, + VarCharArray, ) pytestmark = pytest.mark.anyio @@ -1020,3 +1043,487 @@ async def test_empty_array( json_result = res.result() assert json_result assert not json_result[0]["e_array"] + + +@pytest.mark.parametrize( + ["postgres_type", "py_value", "expected_deserialized"], + ( + ( + "VARCHAR ARRAY", + VarCharArray(["Some String", "Some String"]), + ["Some String", "Some String"], + ), + ( + "VARCHAR ARRAY", + VarCharArray([]), + [], + ), + ( + "TEXT ARRAY", + TextArray([]), + [], + ), + ( + "TEXT ARRAY", + TextArray([PyText("Some String"), PyText("Some String")]), + ["Some String", "Some String"], + ), + ("BOOL ARRAY", BoolArray([]), []), + ("BOOL ARRAY", BoolArray([True, False]), [True, False]), + ("BOOL ARRAY", BoolArray([[True], [False]]), [[True], [False]]), + ("INT2 ARRAY", Int16Array([]), []), + ("INT2 ARRAY", Int16Array([SmallInt(12), SmallInt(100)]), [12, 100]), + ("INT2 ARRAY", Int16Array([SmallInt(12), SmallInt(100)]), [12, 100]), + ("INT2 ARRAY", Int16Array([[SmallInt(12)], [SmallInt(100)]]), [[12], [100]]), + ("INT4 ARRAY", Int32Array([Integer(121231231), Integer(121231231)]), [121231231, 121231231]), + ("INT4 ARRAY", Int32Array([[Integer(121231231)], [Integer(121231231)]]), [[121231231], [121231231]]), + ( + "INT8 ARRAY", + Int64Array([BigInt(99999999999999999), BigInt(99999999999999999)]), + [99999999999999999, 99999999999999999], + ), + ( + "INT8 ARRAY", + Int64Array([[BigInt(99999999999999999)], [BigInt(99999999999999999)]]), + [[99999999999999999], [99999999999999999]], + ), + ( + "MONEY ARRAY", + MoneyArray([Money(99999999999999999), Money(99999999999999999)]), + [99999999999999999, 99999999999999999], + ), + ( + "MONEY ARRAY", + MoneyArray([[Money(99999999999999999)], [Money(99999999999999999)]]), + [[99999999999999999], [99999999999999999]], + ), + ( + "NUMERIC(5, 2) ARRAY", + NumericArray([Decimal("121.23"), Decimal("188.99")]), + [Decimal("121.23"), Decimal("188.99")], + ), + ( + "NUMERIC(5, 2) ARRAY", + NumericArray([[Decimal("121.23")], [Decimal("188.99")]]), + [[Decimal("121.23")], [Decimal("188.99")]], + ), + ( + "FLOAT8 ARRAY", + Float64Array([32.12329864501953, 32.12329864501953]), + [32.12329864501953, 32.12329864501953], + ), + ( + "FLOAT8 ARRAY", + Float64Array([[32.12329864501953], [32.12329864501953]]), + [[32.12329864501953], [32.12329864501953]], + ), + ( + "DATE ARRAY", + DateArray([now_datetime.date(), now_datetime.date()]), + [now_datetime.date(), now_datetime.date()], + ), + ( + "DATE ARRAY", + DateArray([[now_datetime.date()], [now_datetime.date()]]), + [[now_datetime.date()], [now_datetime.date()]], + ), + ( + "TIME ARRAY", + TimeArray([now_datetime.time(), now_datetime.time()]), + [now_datetime.time(), now_datetime.time()], + ), + ( + "TIME ARRAY", + TimeArray([[now_datetime.time()], [now_datetime.time()]]), + [[now_datetime.time()], [now_datetime.time()]], + ), + ("TIMESTAMP ARRAY", DateTimeArray([now_datetime, now_datetime]), [now_datetime, now_datetime]), + ("TIMESTAMP ARRAY", DateTimeArray([[now_datetime], [now_datetime]]), [[now_datetime], [now_datetime]]), + ( + "TIMESTAMPTZ ARRAY", + DateTimeTZArray([now_datetime_with_tz, now_datetime_with_tz]), + [now_datetime_with_tz, now_datetime_with_tz], + ), + ( + "TIMESTAMPTZ ARRAY", + DateTimeTZArray([[now_datetime_with_tz], [now_datetime_with_tz]]), + [[now_datetime_with_tz], [now_datetime_with_tz]], + ), + ( + "UUID ARRAY", + UUIDArray([uuid_, uuid_]), + [str(uuid_), str(uuid_)], + ), + ( + "UUID ARRAY", + UUIDArray([[uuid_], [uuid_]]), + [[str(uuid_)], [str(uuid_)]], + ), + ( + "INET ARRAY", + IpAddressArray([IPv4Address("192.0.0.1"), IPv4Address("192.0.0.1")]), + [IPv4Address("192.0.0.1"), IPv4Address("192.0.0.1")], + ), + ( + "INET ARRAY", + IpAddressArray([[IPv4Address("192.0.0.1")], [IPv4Address("192.0.0.1")]]), + [[IPv4Address("192.0.0.1")], [IPv4Address("192.0.0.1")]], + ), + ( + "JSONB ARRAY", + JSONBArray( + [ + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ], + ), + [ + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ], + ), + ( + "JSONB ARRAY", + JSONBArray( + [ + [ + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ], + [ + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ], + ], + ), + [ + [ + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ], + [ + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ], + ], + ), + ( + "JSONB ARRAY", + JSONBArray( + [ + PyJSONB([{"array": "json"}, {"one more": "test"}]), + PyJSONB([{"array": "json"}, {"one more": "test"}]), + ], + ), + [ + [{"array": "json"}, {"one more": "test"}], + [{"array": "json"}, {"one more": "test"}], + ], + ), + ( + "JSONB ARRAY", + JSONBArray( + [ + PyJSONB([[{"array": "json"}], [{"one more": "test"}]]), + PyJSONB([[{"array": "json"}], [{"one more": "test"}]]), + ], + ), + [ + [[{"array": "json"}], [{"one more": "test"}]], + [[{"array": "json"}], [{"one more": "test"}]], + ], + ), + ( + "JSON ARRAY", + JSONArray( + [ + PyJSON( + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ), + PyJSON( + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ), + ], + ), + [ + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ], + ), + ( + "JSON ARRAY", + JSONArray( + [ + [ + PyJSON( + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ), + ], + [ + PyJSON( + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ), + ], + ], + ), + [ + [ + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ], + [ + { + "test": ["something", 123, "here"], + "nested": ["JSON"], + }, + ], + ], + ), + ( + "JSON ARRAY", + JSONArray( + [ + PyJSON([{"array": "json"}, {"one more": "test"}]), + PyJSON([{"array": "json"}, {"one more": "test"}]), + ], + ), + [ + [{"array": "json"}, {"one more": "test"}], + [{"array": "json"}, {"one more": "test"}], + ], + ), + ( + "JSON ARRAY", + JSONArray( + [ + PyJSON([[{"array": "json"}], [{"one more": "test"}]]), + PyJSON([[{"array": "json"}], [{"one more": "test"}]]), + ], + ), + [ + [[{"array": "json"}], [{"one more": "test"}]], + [[{"array": "json"}], [{"one more": "test"}]], + ], + ), + ( + "POINT ARRAY", + PointArray( + [ + PyPoint([1.5, 2]), + PyPoint([2, 3]), + ], + ), + [ + (1.5, 2.0), + (2.0, 3.0), + ], + ), + ( + "POINT ARRAY", + PointArray( + [ + [PyPoint([1.5, 2])], + [PyPoint([2, 3])], + ], + ), + [ + [(1.5, 2.0)], + [(2.0, 3.0)], + ], + ), + ( + "BOX ARRAY", + BoxArray( + [ + PyBox([3.5, 3, 9, 9]), + PyBox([8.5, 8, 9, 9]), + ], + ), + [ + ((9.0, 9.0), (3.5, 3.0)), + ((9.0, 9.0), (8.5, 8.0)), + ], + ), + ( + "BOX ARRAY", + BoxArray( + [ + [PyBox([3.5, 3, 9, 9])], + [PyBox([8.5, 8, 9, 9])], + ], + ), + [ + [((9.0, 9.0), (3.5, 3.0))], + [((9.0, 9.0), (8.5, 8.0))], + ], + ), + ( + "PATH ARRAY", + PathArray( + [ + PyPath([(3.5, 3), (9, 9), (8, 8)]), + PyPath([(3.5, 3), (6, 6), (3.5, 3)]), + ], + ), + [ + [(3.5, 3.0), (9.0, 9.0), (8.0, 8.0)], + ((3.5, 3.0), (6.0, 6.0), (3.5, 3.0)), + ], + ), + ( + "PATH ARRAY", + PathArray( + [ + [PyPath([(3.5, 3), (9, 9), (8, 8)])], + [PyPath([(3.5, 3), (6, 6), (3.5, 3)])], + ], + ), + [ + [[(3.5, 3.0), (9.0, 9.0), (8.0, 8.0)]], + [((3.5, 3.0), (6.0, 6.0), (3.5, 3.0))], + ], + ), + ( + "LINE ARRAY", + LineArray( + [ + PyLine([-2, 1, 2]), + PyLine([1, -2, 3]), + ], + ), + [ + (-2.0, 1.0, 2.0), + (1.0, -2.0, 3.0), + ], + ), + ( + "LINE ARRAY", + LineArray( + [ + [PyLine([-2, 1, 2])], + [PyLine([1, -2, 3])], + ], + ), + [ + [(-2.0, 1.0, 2.0)], + [(1.0, -2.0, 3.0)], + ], + ), + ( + "LSEG ARRAY", + LsegArray( + [ + PyLineSegment({(1, 2), (9, 9)}), + PyLineSegment([(5.6, 3.1), (4, 5)]), + ], + ), + [ + [(1.0, 2.0), (9.0, 9.0)], + [(5.6, 3.1), (4.0, 5.0)], + ], + ), + ( + "LSEG ARRAY", + LsegArray( + [ + [PyLineSegment({(1, 2), (9, 9)})], + [PyLineSegment([(5.6, 3.1), (4, 5)])], + ], + ), + [ + [[(1.0, 2.0), (9.0, 9.0)]], + [[(5.6, 3.1), (4.0, 5.0)]], + ], + ), + ( + "CIRCLE ARRAY", + CircleArray( + [ + PyCircle([1.7, 2.8, 3]), + PyCircle([5, 1.8, 10]), + ], + ), + [ + ((1.7, 2.8), 3.0), + ((5.0, 1.8), 10.0), + ], + ), + ( + "CIRCLE ARRAY", + CircleArray( + [ + [PyCircle([1.7, 2.8, 3])], + [PyCircle([5, 1.8, 10])], + ], + ), + [ + [((1.7, 2.8), 3.0)], + [((5.0, 1.8), 10.0)], + ], + ), + ), +) +async def test_array_types( + psql_pool: ConnectionPool, + postgres_type: str, + py_value: Any, + expected_deserialized: Any, +) -> None: + await psql_pool.execute("DROP TABLE IF EXISTS for_test") + create_table_query = f""" + CREATE TABLE for_test (test_field {postgres_type}) + """ + insert_data_query = """ + INSERT INTO for_test VALUES ($1) + """ + await psql_pool.execute(querystring=create_table_query) + await psql_pool.execute( + querystring=insert_data_query, + parameters=[py_value], + ) + + raw_result = await psql_pool.execute( + querystring="SELECT test_field FROM for_test", + ) + + assert raw_result.result()[0]["test_field"] == expected_deserialized diff --git a/src/extra_types.rs b/src/extra_types.rs index 9ffb760..874fd47 100644 --- a/src/extra_types.rs +++ b/src/extra_types.rs @@ -11,8 +11,11 @@ use serde_json::Value; use crate::{ additional_types::{Circle, Line}, - exceptions::rust_errors::RustPSQLDriverPyResult, - value_converter::{build_flat_geo_coords, build_geo_coords, build_serde_value}, + exceptions::rust_errors::{RustPSQLDriverError, RustPSQLDriverPyResult}, + value_converter::{ + build_flat_geo_coords, build_geo_coords, build_serde_value, + py_sequence_into_postgres_array, PythonDTO, + }, }; macro_rules! build_python_type { @@ -290,6 +293,75 @@ impl PyCircle { } } +macro_rules! build_array_type { + ($st_name:ident, $kind:path) => { + #[pyclass] + #[derive(Clone)] + pub struct $st_name { + inner: Py, + } + + #[pymethods] + impl $st_name { + #[new] + #[must_use] + pub fn new_class(inner: Py) -> Self { + Self { inner } + } + } + + impl $st_name { + #[must_use] + pub fn inner(&self) -> Py { + self.inner.clone() + } + + /// Convert incoming sequence from python to internal `PythonDTO`. + /// + /// # Errors + /// May return Err Result if cannot convert sequence to array. + pub fn _convert_to_python_dto(&self) -> RustPSQLDriverPyResult { + return Python::with_gil(|gil| { + let binding = &self.inner; + let bound_inner = Ok::<&pyo3::Bound<'_, pyo3::PyAny>, RustPSQLDriverError>( + binding.bind(gil), + )?; + Ok::($kind(py_sequence_into_postgres_array( + bound_inner, + )?)) + }); + } + } + }; +} + +build_array_type!(BoolArray, PythonDTO::PyBoolArray); +build_array_type!(UUIDArray, PythonDTO::PyUuidArray); +build_array_type!(VarCharArray, PythonDTO::PyVarCharArray); +build_array_type!(TextArray, PythonDTO::PyTextArray); +build_array_type!(Int16Array, PythonDTO::PyInt16Array); +build_array_type!(Int32Array, PythonDTO::PyInt32Array); +build_array_type!(Int64Array, PythonDTO::PyInt64Array); +build_array_type!(Float32Array, PythonDTO::PyFloat32Array); +build_array_type!(Float64Array, PythonDTO::PyFloat64Array); +build_array_type!(MoneyArray, PythonDTO::PyMoneyArray); +build_array_type!(IpAddressArray, PythonDTO::PyIpAddressArray); +build_array_type!(JSONBArray, PythonDTO::PyJSONBArray); +build_array_type!(JSONArray, PythonDTO::PyJSONArray); +build_array_type!(DateArray, PythonDTO::PyDateArray); +build_array_type!(TimeArray, PythonDTO::PyTimeArray); +build_array_type!(DateTimeArray, PythonDTO::PyDateTimeArray); +build_array_type!(DateTimeTZArray, PythonDTO::PyDateTimeTZArray); +build_array_type!(MacAddr6Array, PythonDTO::PyMacAddr6Array); +build_array_type!(MacAddr8Array, PythonDTO::PyMacAddr8Array); +build_array_type!(NumericArray, PythonDTO::PyNumericArray); +build_array_type!(PointArray, PythonDTO::PyPointArray); +build_array_type!(BoxArray, PythonDTO::PyBoxArray); +build_array_type!(PathArray, PythonDTO::PyPathArray); +build_array_type!(LineArray, PythonDTO::PyLineArray); +build_array_type!(LsegArray, PythonDTO::PyLsegArray); +build_array_type!(CircleArray, PythonDTO::PyCircleArray); + #[allow(clippy::module_name_repetitions)] #[allow(clippy::missing_errors_doc)] pub fn extra_types_module(_py: Python<'_>, pymod: &Bound<'_, PyModule>) -> PyResult<()> { @@ -312,5 +384,31 @@ pub fn extra_types_module(_py: Python<'_>, pymod: &Bound<'_, PyModule>) -> PyRes pymod.add_class::()?; pymod.add_class::()?; pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; + pymod.add_class::()?; Ok(()) } diff --git a/src/value_converter.rs b/src/value_converter.rs index 2c46722..fd26436 100644 --- a/src/value_converter.rs +++ b/src/value_converter.rs @@ -14,9 +14,10 @@ use pyo3::{ sync::GILOnceCell, types::{ PyAnyMethods, PyBool, PyBytes, PyDate, PyDateTime, PyDict, PyDictMethods, PyFloat, PyInt, - PyList, PyListMethods, PySequence, PySet, PyString, PyTime, PyTuple, PyType, PyTypeMethods, + PyIterator, PyList, PyListMethods, PySequence, PySet, PyString, PyTime, PyTuple, PyType, + PyTypeMethods, }, - Bound, IntoPy, Py, PyAny, PyObject, PyResult, Python, ToPyObject, + Bound, FromPyObject, IntoPy, Py, PyAny, PyObject, PyResult, Python, ToPyObject, }; use tokio_postgres::{ types::{to_sql_checked, Type}, @@ -29,11 +30,7 @@ use crate::{ RustRect, }, exceptions::rust_errors::{RustPSQLDriverError, RustPSQLDriverPyResult}, - extra_types::{ - BigInt, Float32, Float64, Integer, Money, PyBox, PyCircle, PyCustomType, PyJSON, PyJSONB, - PyLine, PyLineSegment, PyMacAddr6, PyMacAddr8, PyPath, PyPoint, PyText, PyVarChar, - SmallInt, - }, + extra_types, }; use postgres_array::{array::Array, Dimension}; @@ -57,7 +54,10 @@ fn get_decimal_cls(py: Python<'_>) -> PyResult<&Bound<'_, PyType>> { /// /// We use custom struct because we need to implement external traits /// to it. -struct InternalUuid(Uuid); +#[derive(Clone, Copy)] +pub struct InternalUuid(Uuid); + +impl<'a> FromPyObject<'a> for InternalUuid {} impl ToPyObject for InternalUuid { fn to_object(&self, py: Python<'_>) -> PyObject { @@ -82,7 +82,10 @@ impl<'a> FromSql<'a> for InternalUuid { /// /// We use custom struct because we need to implement external traits /// to it. -struct InternalSerdeValue(Value); +#[derive(Clone)] +pub struct InternalSerdeValue(Value); + +impl<'a> FromPyObject<'a> for InternalSerdeValue {} impl ToPyObject for InternalSerdeValue { fn to_object(&self, py: Python<'_>) -> PyObject { @@ -177,6 +180,32 @@ pub enum PythonDTO { PyLine(Line), PyLineSegment(LineSegment), PyCircle(Circle), + PyBoolArray(Array), + PyUuidArray(Array), + PyVarCharArray(Array), + PyTextArray(Array), + PyInt16Array(Array), + PyInt32Array(Array), + PyInt64Array(Array), + PyFloat32Array(Array), + PyFloat64Array(Array), + PyMoneyArray(Array), + PyIpAddressArray(Array), + PyJSONBArray(Array), + PyJSONArray(Array), + PyDateArray(Array), + PyTimeArray(Array), + PyDateTimeArray(Array), + PyDateTimeTZArray(Array), + PyMacAddr6Array(Array), + PyMacAddr8Array(Array), + PyNumericArray(Array), + PyPointArray(Array), + PyBoxArray(Array), + PyPathArray(Array), + PyLineArray(Array), + PyLsegArray(Array), + PyCircleArray(Array), } impl ToPyObject for PythonDTO { @@ -418,7 +447,86 @@ impl ToSql for PythonDTO { PythonDTO::PyDecimal(py_decimal) => { ::to_sql(py_decimal, ty, out)?; } + PythonDTO::PyBoolArray(array) => { + array.to_sql(&Type::BOOL_ARRAY, out)?; + } + PythonDTO::PyUuidArray(array) => { + array.to_sql(&Type::UUID_ARRAY, out)?; + } + PythonDTO::PyVarCharArray(array) => { + array.to_sql(&Type::VARCHAR_ARRAY, out)?; + } + PythonDTO::PyTextArray(array) => { + array.to_sql(&Type::TEXT_ARRAY, out)?; + } + PythonDTO::PyInt16Array(array) => { + array.to_sql(&Type::INT2_ARRAY, out)?; + } + PythonDTO::PyInt32Array(array) => { + array.to_sql(&Type::INT4_ARRAY, out)?; + } + PythonDTO::PyInt64Array(array) => { + array.to_sql(&Type::INT8_ARRAY, out)?; + } + PythonDTO::PyFloat32Array(array) => { + array.to_sql(&Type::FLOAT4, out)?; + } + PythonDTO::PyFloat64Array(array) => { + array.to_sql(&Type::FLOAT8_ARRAY, out)?; + } + PythonDTO::PyMoneyArray(array) => { + array.to_sql(&Type::MONEY_ARRAY, out)?; + } + PythonDTO::PyIpAddressArray(array) => { + array.to_sql(&Type::INET_ARRAY, out)?; + } + PythonDTO::PyJSONBArray(array) => { + array.to_sql(&Type::JSONB_ARRAY, out)?; + } + PythonDTO::PyJSONArray(array) => { + array.to_sql(&Type::JSON_ARRAY, out)?; + } + PythonDTO::PyDateArray(array) => { + array.to_sql(&Type::DATE_ARRAY, out)?; + } + PythonDTO::PyTimeArray(array) => { + array.to_sql(&Type::TIME_ARRAY, out)?; + } + PythonDTO::PyDateTimeArray(array) => { + array.to_sql(&Type::TIMESTAMP_ARRAY, out)?; + } + PythonDTO::PyDateTimeTZArray(array) => { + array.to_sql(&Type::TIMESTAMPTZ_ARRAY, out)?; + } + PythonDTO::PyMacAddr6Array(array) => { + array.to_sql(&Type::MACADDR_ARRAY, out)?; + } + PythonDTO::PyMacAddr8Array(array) => { + array.to_sql(&Type::MACADDR8_ARRAY, out)?; + } + PythonDTO::PyNumericArray(array) => { + array.to_sql(&Type::NUMERIC_ARRAY, out)?; + } + PythonDTO::PyPointArray(array) => { + array.to_sql(&Type::POINT_ARRAY, out)?; + } + PythonDTO::PyBoxArray(array) => { + array.to_sql(&Type::BOX_ARRAY, out)?; + } + PythonDTO::PyPathArray(array) => { + array.to_sql(&Type::PATH_ARRAY, out)?; + } + PythonDTO::PyLineArray(array) => { + array.to_sql(&Type::LINE_ARRAY, out)?; + } + PythonDTO::PyLsegArray(array) => { + array.to_sql(&Type::LSEG_ARRAY, out)?; + } + PythonDTO::PyCircleArray(array) => { + array.to_sql(&Type::CIRCLE_ARRAY, out)?; + } } + if return_is_null_true { Ok(tokio_postgres::types::IsNull::Yes) } else { @@ -553,7 +661,6 @@ pub fn py_sequence_into_postgres_array( } let array_data = py_sequence_into_flat_vec(parameter)?; - match postgres_array::Array::from_parts_no_panic(array_data, dimensions) { Ok(result_array) => Ok(result_array), Err(err) => Err(RustPSQLDriverError::PyToRustValueConversionError(format!( @@ -574,9 +681,9 @@ pub fn py_to_rust(parameter: &pyo3::Bound<'_, PyAny>) -> RustPSQLDriverPyResult< return Ok(PythonDTO::PyNone); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyCustomType( - parameter.extract::()?.inner(), + parameter.extract::()?.inner(), )); } @@ -588,13 +695,15 @@ pub fn py_to_rust(parameter: &pyo3::Bound<'_, PyAny>) -> RustPSQLDriverPyResult< return Ok(PythonDTO::PyBytes(parameter.extract::>()?)); } - if parameter.is_instance_of::() { - return Ok(PythonDTO::PyText(parameter.extract::()?.inner())); + if parameter.is_instance_of::() { + return Ok(PythonDTO::PyText( + parameter.extract::()?.inner(), + )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyVarChar( - parameter.extract::()?.inner(), + parameter.extract::()?.inner(), )); } @@ -606,39 +715,47 @@ pub fn py_to_rust(parameter: &pyo3::Bound<'_, PyAny>) -> RustPSQLDriverPyResult< return Ok(PythonDTO::PyFloat64(parameter.extract::()?)); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyFloat32( - parameter.extract::()?.retrieve_value(), + parameter + .extract::()? + .retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyFloat64( - parameter.extract::()?.retrieve_value(), + parameter + .extract::()? + .retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyIntI16( - parameter.extract::()?.retrieve_value(), + parameter + .extract::()? + .retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyIntI32( - parameter.extract::()?.retrieve_value(), + parameter + .extract::()? + .retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyIntI64( - parameter.extract::()?.retrieve_value(), + parameter.extract::()?.retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyMoney( - parameter.extract::()?.retrieve_value(), + parameter.extract::()?.retrieve_value(), )); } @@ -701,27 +818,27 @@ pub fn py_to_rust(parameter: &pyo3::Bound<'_, PyAny>) -> RustPSQLDriverPyResult< return Ok(PythonDTO::PyJsonb(Value::Object(serde_map))); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyJsonb( - parameter.extract::()?.inner().clone(), + parameter.extract::()?.inner().clone(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyJson( - parameter.extract::()?.inner().clone(), + parameter.extract::()?.inner().clone(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyMacAddr6( - parameter.extract::()?.inner(), + parameter.extract::()?.inner(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyMacAddr8( - parameter.extract::()?.inner(), + parameter.extract::()?.inner(), )); } @@ -737,42 +854,204 @@ pub fn py_to_rust(parameter: &pyo3::Bound<'_, PyAny>) -> RustPSQLDriverPyResult< )?)); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyPoint( - parameter.extract::()?.retrieve_value(), + parameter + .extract::()? + .retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyBox( - parameter.extract::()?.retrieve_value(), + parameter.extract::()?.retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyPath( - parameter.extract::()?.retrieve_value(), + parameter.extract::()?.retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyLine( - parameter.extract::()?.retrieve_value(), + parameter.extract::()?.retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyLineSegment( - parameter.extract::()?.retrieve_value(), + parameter + .extract::()? + .retrieve_value(), )); } - if parameter.is_instance_of::() { + if parameter.is_instance_of::() { return Ok(PythonDTO::PyCircle( - parameter.extract::()?.retrieve_value(), + parameter + .extract::()? + .retrieve_value(), )); } + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + + if parameter.is_instance_of::() { + return parameter + .extract::()? + ._convert_to_python_dto(); + } + if let Ok(id_address) = parameter.extract::() { return Ok(PythonDTO::PyIpAddress(id_address)); } @@ -786,6 +1065,9 @@ pub fn py_to_rust(parameter: &pyo3::Bound<'_, PyAny>) -> RustPSQLDriverPyResult< } } + let a = parameter.downcast::(); + println!("{:?}", a.iter()); + Err(RustPSQLDriverError::PyToRustValueConversionError(format!( "Can not covert you type {parameter} into inner one", ))) @@ -1171,7 +1453,7 @@ fn postgres_bytes_to_py( Option>>, >(type_, buf, is_simple)?) .to_object(py)), - // Convert ARRAY of UUID into Vec>, then into list[datetime.date] + // Convert ARRAY of UUID into Vec>, then into list[UUID] Type::UUID_ARRAY => { let uuid_array = _composite_field_postgres_to_py::>>(type_, buf, is_simple)?;