from datetime import date
from datetime import datetime
from datetime import timedelta
+from decimal import Decimal
from typing import Any
from typing import Generic
from typing import Optional
def __bool__(self) -> bool:
return self.empty
+ @property
+ def __sa_type_engine__(self):
+ return AbstractRange()
+
def _contains_value(self, value: _T) -> bool:
"Check whether this range contains the given `value`."
else:
return super().adapt(impltype)
+ def _resolve_for_literal(self, value):
+ spec = value.lower if value.lower is not None else value.upper
+
+ if isinstance(spec, int):
+ return INT8RANGE()
+ elif isinstance(spec, (Decimal, float)):
+ return NUMRANGE()
+ elif isinstance(spec, datetime):
+ return TSRANGE() if not spec.tzinfo else TSTZRANGE()
+ elif isinstance(spec, date):
+ return DATERANGE()
+ else:
+ # empty Range, SQL datatype can't be determined here
+ return sqltypes.NULLTYPE
+
class comparator_factory(sqltypes.Concatenable.Comparator):
"""Define comparison operations for range types."""
def _resolve_value_to_type(value: Any) -> TypeEngine[Any]:
_result_type = _type_map_get(type(value), False)
+
+ if _result_type is False:
+ _result_type = getattr(value, "__sa_type_engine__", False)
+
if _result_type is False:
# use inspect() to detect SQLAlchemy built-in
# objects.
from sqlalchemy.dialects.postgresql import insert
from sqlalchemy.dialects.postgresql import JSONB
from sqlalchemy.dialects.postgresql import JSONPATH
+from sqlalchemy.dialects.postgresql import Range
from sqlalchemy.dialects.postgresql import TSRANGE
from sqlalchemy.dialects.postgresql.base import PGDialect
from sqlalchemy.dialects.postgresql.psycopg2 import PGDialect_psycopg2
"AS jsonb_path_exists_1 FROM data",
)
+ def test_custom_object_hook(self):
+ # See issue #8884
+ from datetime import date
+
+ usages = table(
+ "usages",
+ column("id", Integer),
+ column("date", Date),
+ column("amount", Integer),
+ )
+ period = Range(date(2022, 1, 1), (2023, 1, 1))
+ stmt = select(func.sum(usages.c.amount)).where(
+ usages.c.date.op("<@")(period)
+ )
+ self.assert_compile(
+ stmt,
+ "SELECT sum(usages.amount) AS sum_1 FROM usages "
+ "WHERE usages.date <@ %(date_1)s::DATERANGE",
+ )
+
class InsertOnConflictTest(fixtures.TablesTest, AssertsCompiledSQL):
__dialect__ = postgresql.dialect()