2018-07-05 23:02:12 +08:00
|
|
|
import math
|
|
|
|
|
2021-03-25 06:29:33 +08:00
|
|
|
from django.db.models.expressions import Func, Value
|
2018-12-02 07:43:27 +08:00
|
|
|
from django.db.models.fields import FloatField, IntegerField
|
2018-07-05 23:02:12 +08:00
|
|
|
from django.db.models.functions import Cast
|
2018-12-02 07:43:27 +08:00
|
|
|
from django.db.models.functions.mixins import (
|
|
|
|
FixDecimalInputMixin,
|
|
|
|
NumericOutputFieldMixin,
|
|
|
|
)
|
2019-01-09 08:21:31 +08:00
|
|
|
from django.db.models.lookups import Transform
|
2018-07-05 23:02:12 +08:00
|
|
|
|
|
|
|
|
|
|
|
class Abs(Transform):
|
|
|
|
function = "ABS"
|
|
|
|
lookup_name = "abs"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class ACos(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "ACOS"
|
|
|
|
lookup_name = "acos"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class ASin(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "ASIN"
|
|
|
|
lookup_name = "asin"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class ATan(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "ATAN"
|
|
|
|
lookup_name = "atan"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class ATan2(NumericOutputFieldMixin, Func):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "ATAN2"
|
|
|
|
arity = 2
|
|
|
|
|
2018-02-08 15:09:00 +08:00
|
|
|
def as_sqlite(self, compiler, connection, **extra_context):
|
2019-02-05 01:03:12 +08:00
|
|
|
if not getattr(
|
|
|
|
connection.ops, "spatialite", False
|
|
|
|
) or connection.ops.spatial_version >= (5, 0, 0):
|
2018-07-05 23:02:12 +08:00
|
|
|
return self.as_sql(compiler, connection)
|
|
|
|
# This function is usually ATan2(y, x), returning the inverse tangent
|
2019-02-05 01:03:12 +08:00
|
|
|
# of y / x, but it's ATan2(x, y) on SpatiaLite < 5.0.0.
|
2018-07-05 23:02:12 +08:00
|
|
|
# Cast integers to float to avoid inconsistent/buggy behavior if the
|
|
|
|
# arguments are mixed between integer and float or decimal.
|
|
|
|
# https://www.gaia-gis.it/fossil/libspatialite/tktview?name=0f72cca3a2
|
|
|
|
clone = self.copy()
|
|
|
|
clone.set_source_expressions(
|
|
|
|
[
|
|
|
|
Cast(expression, FloatField())
|
|
|
|
if isinstance(expression.output_field, IntegerField)
|
|
|
|
else expression
|
|
|
|
for expression in self.get_source_expressions()[::-1]
|
|
|
|
]
|
|
|
|
)
|
2018-02-08 15:09:00 +08:00
|
|
|
return clone.as_sql(compiler, connection, **extra_context)
|
2018-07-05 23:02:12 +08:00
|
|
|
|
|
|
|
|
|
|
|
class Ceil(Transform):
|
|
|
|
function = "CEILING"
|
|
|
|
lookup_name = "ceil"
|
|
|
|
|
2018-02-08 15:09:00 +08:00
|
|
|
def as_oracle(self, compiler, connection, **extra_context):
|
|
|
|
return super().as_sql(compiler, connection, function="CEIL", **extra_context)
|
2018-07-05 23:02:12 +08:00
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Cos(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "COS"
|
|
|
|
lookup_name = "cos"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Cot(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "COT"
|
|
|
|
lookup_name = "cot"
|
|
|
|
|
2018-02-08 15:09:00 +08:00
|
|
|
def as_oracle(self, compiler, connection, **extra_context):
|
|
|
|
return super().as_sql(
|
|
|
|
compiler, connection, template="(1 / TAN(%(expressions)s))", **extra_context
|
|
|
|
)
|
2018-07-05 23:02:12 +08:00
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Degrees(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "DEGREES"
|
|
|
|
lookup_name = "degrees"
|
|
|
|
|
2018-02-08 15:09:00 +08:00
|
|
|
def as_oracle(self, compiler, connection, **extra_context):
|
|
|
|
return super().as_sql(
|
|
|
|
compiler,
|
|
|
|
connection,
|
|
|
|
template="((%%(expressions)s) * 180 / %s)" % math.pi,
|
|
|
|
**extra_context,
|
|
|
|
)
|
2018-07-05 23:02:12 +08:00
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Exp(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "EXP"
|
|
|
|
lookup_name = "exp"
|
|
|
|
|
|
|
|
|
|
|
|
class Floor(Transform):
|
|
|
|
function = "FLOOR"
|
|
|
|
lookup_name = "floor"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Ln(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "LN"
|
|
|
|
lookup_name = "ln"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Log(FixDecimalInputMixin, NumericOutputFieldMixin, Func):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "LOG"
|
|
|
|
arity = 2
|
|
|
|
|
2018-02-08 15:09:00 +08:00
|
|
|
def as_sqlite(self, compiler, connection, **extra_context):
|
2018-07-05 23:02:12 +08:00
|
|
|
if not getattr(connection.ops, "spatialite", False):
|
|
|
|
return self.as_sql(compiler, connection)
|
|
|
|
# This function is usually Log(b, x) returning the logarithm of x to
|
|
|
|
# the base b, but on SpatiaLite it's Log(x, b).
|
|
|
|
clone = self.copy()
|
|
|
|
clone.set_source_expressions(self.get_source_expressions()[::-1])
|
2018-02-08 15:09:00 +08:00
|
|
|
return clone.as_sql(compiler, connection, **extra_context)
|
2018-07-05 23:02:12 +08:00
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Mod(FixDecimalInputMixin, NumericOutputFieldMixin, Func):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "MOD"
|
|
|
|
arity = 2
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Pi(NumericOutputFieldMixin, Func):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "PI"
|
|
|
|
arity = 0
|
|
|
|
|
2018-02-08 15:09:00 +08:00
|
|
|
def as_oracle(self, compiler, connection, **extra_context):
|
|
|
|
return super().as_sql(
|
|
|
|
compiler, connection, template=str(math.pi), **extra_context
|
|
|
|
)
|
2018-07-05 23:02:12 +08:00
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Power(NumericOutputFieldMixin, Func):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "POWER"
|
|
|
|
arity = 2
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Radians(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "RADIANS"
|
|
|
|
lookup_name = "radians"
|
|
|
|
|
2018-02-08 15:09:00 +08:00
|
|
|
def as_oracle(self, compiler, connection, **extra_context):
|
|
|
|
return super().as_sql(
|
|
|
|
compiler,
|
|
|
|
connection,
|
|
|
|
template="((%%(expressions)s) * %s / 180)" % math.pi,
|
|
|
|
**extra_context,
|
|
|
|
)
|
2018-07-05 23:02:12 +08:00
|
|
|
|
|
|
|
|
2020-07-17 06:32:46 +08:00
|
|
|
class Random(NumericOutputFieldMixin, Func):
|
|
|
|
function = "RANDOM"
|
|
|
|
arity = 0
|
|
|
|
|
|
|
|
def as_mysql(self, compiler, connection, **extra_context):
|
|
|
|
return super().as_sql(compiler, connection, function="RAND", **extra_context)
|
|
|
|
|
|
|
|
def as_oracle(self, compiler, connection, **extra_context):
|
|
|
|
return super().as_sql(
|
|
|
|
compiler, connection, function="DBMS_RANDOM.VALUE", **extra_context
|
|
|
|
)
|
|
|
|
|
|
|
|
def as_sqlite(self, compiler, connection, **extra_context):
|
|
|
|
return super().as_sql(compiler, connection, function="RAND", **extra_context)
|
|
|
|
|
2020-10-20 05:10:06 +08:00
|
|
|
def get_group_by_cols(self, alias=None):
|
|
|
|
return []
|
|
|
|
|
2020-07-17 06:32:46 +08:00
|
|
|
|
2021-03-25 06:29:33 +08:00
|
|
|
class Round(FixDecimalInputMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "ROUND"
|
|
|
|
lookup_name = "round"
|
2021-03-25 06:29:33 +08:00
|
|
|
arity = None # Override Transform's arity=1 to enable passing precision.
|
|
|
|
|
|
|
|
def __init__(self, expression, precision=0, **extra):
|
|
|
|
super().__init__(expression, precision, **extra)
|
|
|
|
|
|
|
|
def as_sqlite(self, compiler, connection, **extra_context):
|
|
|
|
precision = self.get_source_expressions()[1]
|
|
|
|
if isinstance(precision, Value) and precision.value < 0:
|
|
|
|
raise ValueError("SQLite does not support negative precision.")
|
|
|
|
return super().as_sqlite(compiler, connection, **extra_context)
|
|
|
|
|
|
|
|
def _resolve_output_field(self):
|
|
|
|
source = self.get_source_expressions()[0]
|
|
|
|
return source.output_field
|
2018-07-05 23:02:12 +08:00
|
|
|
|
|
|
|
|
2019-03-20 16:27:34 +08:00
|
|
|
class Sign(Transform):
|
|
|
|
function = "SIGN"
|
|
|
|
lookup_name = "sign"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Sin(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "SIN"
|
|
|
|
lookup_name = "sin"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Sqrt(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "SQRT"
|
|
|
|
lookup_name = "sqrt"
|
|
|
|
|
|
|
|
|
2018-12-02 07:43:27 +08:00
|
|
|
class Tan(NumericOutputFieldMixin, Transform):
|
2018-07-05 23:02:12 +08:00
|
|
|
function = "TAN"
|
|
|
|
lookup_name = "tan"
|