# Copyright 2023 Goldman Sachs
#
# 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.
"""
Integer expression type in the PyLegend expression language.
``PyLegendInteger`` represents an integer-valued column or computed
expression within a PyLegend query. It extends ``PyLegendNumber`` with
integer-specific arithmetic (``+``, ``-``, ``*``, ``%``) that preserves
the integer type when both operands are integers, as well as bitwise
operators (``&``, ``|``, ``^``, ``~``, ``<<``, ``>>``).
Instances are never constructed directly. They are produced by accessing
an integer column on a TDS frame — for example, ``frame["Order Id"]`` —
or by integer arithmetic on existing integer expressions.
``PyLegendInteger`` inherits all mathematical functions from
``PyLegendNumber`` (trigonometric, logarithmic, rounding, etc.) and
general-purpose methods from ``PyLegendPrimitive`` (equality, null
checks, string conversion, ``in_list``).
"""
from pylegend._typing import (
PyLegendSequence,
PyLegendDict,
PyLegendUnion,
TYPE_CHECKING,
)
from pylegend.core.language.shared.primitives.number import PyLegendNumber
from pylegend.core.language.shared.expression import PyLegendExpressionIntegerReturn
from pylegend.core.language.shared.literal_expressions import PyLegendIntegerLiteralExpression
from pylegend.core.sql.metamodel import (
Expression,
QuerySpecification
)
from pylegend.core.tds.pandas_api.frames.helpers.series_helper import grammar_method
from pylegend.core.tds.tds_frame import FrameToSqlConfig
from pylegend.core.language.shared.operations.integer_operation_expressions import (
PyLegendIntegerAddExpression,
PyLegendIntegerAbsoluteExpression,
PyLegendIntegerNegativeExpression,
PyLegendIntegerSubtractExpression,
PyLegendIntegerMultiplyExpression,
PyLegendIntegerModuloExpression,
PyLegendIntegerCharExpression,
PyLegendIntegerBitAndExpression,
PyLegendIntegerBitOrExpression,
PyLegendIntegerBitXorExpression,
PyLegendIntegerBitShiftLeftExpression,
PyLegendIntegerBitShiftRightExpression,
PyLegendIntegerBitNotExpression
)
if TYPE_CHECKING:
from pylegend.core.language.shared.primitives import PyLegendFloat
from pylegend.core.language.shared.primitives.string import PyLegendString
__all__: PyLegendSequence[str] = [
"PyLegendInteger"
]
class PyLegendInteger(PyLegendNumber):
__value_copy: PyLegendExpressionIntegerReturn
def __init__(
self,
value: PyLegendExpressionIntegerReturn
) -> None:
self.__value_copy = value
super().__init__(value)
def char(self) -> "PyLegendString":
from pylegend.core.language.shared.primitives.string import PyLegendString
return PyLegendString(PyLegendIntegerCharExpression(self.__value_copy))
def to_sql_expression(
self,
frame_name_to_base_query_map: PyLegendDict[str, QuerySpecification],
config: FrameToSqlConfig
) -> Expression:
return super().to_sql_expression(frame_name_to_base_query_map, config)
def value(self) -> PyLegendExpressionIntegerReturn:
return self.__value_copy
[docs]
@grammar_method
def __add__(
self,
other: PyLegendUnion[int, float, "PyLegendInteger", "PyLegendFloat", "PyLegendNumber"]
) -> "PyLegendUnion[PyLegendNumber, PyLegendInteger]":
"""
Addition (``+``).
When both operands are integers, returns a ``PyLegendInteger``;
otherwise falls back to ``PyLegendNumber``.
Parameters
----------
other : int, float, PyLegendInteger, PyLegendFloat, or PyLegendNumber
The right-hand operand.
Returns
-------
PyLegendInteger or PyLegendNumber
The sum expression.
Raises
------
TypeError
If *other* is not a supported numeric type.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["id_plus_10"] = frame["Order Id"] + 10
frame.head(3).to_pandas()
"""
PyLegendNumber.validate_param_to_be_number(other, "Integer plus (+) parameter")
if isinstance(other, (int, PyLegendInteger)):
other_op = PyLegendInteger.__convert_to_integer_expr(other)
return PyLegendInteger(PyLegendIntegerAddExpression(self.__value_copy, other_op))
else:
return super().__add__(other)
[docs]
@grammar_method
def __radd__(
self,
other: PyLegendUnion[int, float, "PyLegendInteger", "PyLegendFloat", "PyLegendNumber"]
) -> "PyLegendUnion[PyLegendNumber, PyLegendInteger]":
"""
Reflected addition (``int + expr``).
Called when a Python literal is on the left side of ``+``.
Parameters
----------
other : int, float, PyLegendInteger, PyLegendFloat, or PyLegendNumber
The left-hand operand.
Returns
-------
PyLegendInteger or PyLegendNumber
The sum expression.
Raises
------
TypeError
If *other* is not a supported numeric type.
"""
PyLegendNumber.validate_param_to_be_number(other, "Integer plus (+) parameter")
if isinstance(other, (int, PyLegendInteger)):
other_op = PyLegendInteger.__convert_to_integer_expr(other)
return PyLegendInteger(PyLegendIntegerAddExpression(other_op, self.__value_copy))
else:
return super().__radd__(other)
[docs]
@grammar_method
def __sub__(
self,
other: PyLegendUnion[int, float, "PyLegendInteger", "PyLegendFloat", "PyLegendNumber"]
) -> "PyLegendUnion[PyLegendNumber, PyLegendInteger]":
"""
Subtraction (``-``).
When both operands are integers, returns a ``PyLegendInteger``;
otherwise falls back to ``PyLegendNumber``.
Parameters
----------
other : int, float, PyLegendInteger, PyLegendFloat, or PyLegendNumber
The right-hand operand.
Returns
-------
PyLegendInteger or PyLegendNumber
The difference expression.
Raises
------
TypeError
If *other* is not a supported numeric type.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["id_minus_10000"] = frame["Order Id"] - 10000
frame.head(3).to_pandas()
"""
PyLegendNumber.validate_param_to_be_number(other, "Integer minus (-) parameter")
if isinstance(other, (int, PyLegendInteger)):
other_op = PyLegendInteger.__convert_to_integer_expr(other)
return PyLegendInteger(PyLegendIntegerSubtractExpression(self.__value_copy, other_op))
else:
return super().__sub__(other)
[docs]
@grammar_method
def __rsub__(
self,
other: PyLegendUnion[int, float, "PyLegendInteger", "PyLegendFloat", "PyLegendNumber"]
) -> "PyLegendUnion[PyLegendNumber, PyLegendInteger]":
"""
Reflected subtraction (``int - expr``).
Called when a Python literal is on the left side of ``-``.
Parameters
----------
other : int, float, PyLegendInteger, PyLegendFloat, or PyLegendNumber
The left-hand operand.
Returns
-------
PyLegendInteger or PyLegendNumber
The difference expression.
Raises
------
TypeError
If *other* is not a supported numeric type.
"""
PyLegendNumber.validate_param_to_be_number(other, "Integer minus (-) parameter")
if isinstance(other, (int, PyLegendInteger)):
other_op = PyLegendInteger.__convert_to_integer_expr(other)
return PyLegendInteger(PyLegendIntegerSubtractExpression(other_op, self.__value_copy))
else:
return super().__rsub__(other)
[docs]
@grammar_method
def __mul__(
self,
other: PyLegendUnion[int, float, "PyLegendInteger", "PyLegendFloat", "PyLegendNumber"]
) -> "PyLegendUnion[PyLegendNumber, PyLegendInteger]":
"""
Multiplication (``*``).
When both operands are integers, returns a ``PyLegendInteger``;
otherwise falls back to ``PyLegendNumber``.
Parameters
----------
other : int, float, PyLegendInteger, PyLegendFloat, or PyLegendNumber
The right-hand operand.
Returns
-------
PyLegendInteger or PyLegendNumber
The product expression.
Raises
------
TypeError
If *other* is not a supported numeric type.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["id_times_2"] = frame["Order Id"] * 2
frame.head(3).to_pandas()
"""
PyLegendNumber.validate_param_to_be_number(other, "Integer multiply (*) parameter")
if isinstance(other, (int, PyLegendInteger)):
other_op = PyLegendInteger.__convert_to_integer_expr(other)
return PyLegendInteger(PyLegendIntegerMultiplyExpression(self.__value_copy, other_op))
else:
return super().__mul__(other)
[docs]
@grammar_method
def __rmul__(
self,
other: PyLegendUnion[int, float, "PyLegendInteger", "PyLegendFloat", "PyLegendNumber"]
) -> "PyLegendUnion[PyLegendNumber, PyLegendInteger]":
"""
Reflected multiplication (``int * expr``).
Called when a Python literal is on the left side of ``*``.
Parameters
----------
other : int, float, PyLegendInteger, PyLegendFloat, or PyLegendNumber
The left-hand operand.
Returns
-------
PyLegendInteger or PyLegendNumber
The product expression.
Raises
------
TypeError
If *other* is not a supported numeric type.
"""
PyLegendNumber.validate_param_to_be_number(other, "Integer multiply (*) parameter")
if isinstance(other, (int, PyLegendInteger)):
other_op = PyLegendInteger.__convert_to_integer_expr(other)
return PyLegendInteger(PyLegendIntegerMultiplyExpression(other_op, self.__value_copy))
else:
return super().__rmul__(other)
[docs]
@grammar_method
def __mod__(
self,
other: PyLegendUnion[int, "PyLegendInteger"]
) -> "PyLegendUnion[PyLegendNumber, PyLegendInteger]":
"""
Modulo (``%``).
Parameters
----------
other : int or PyLegendInteger
The divisor.
Returns
-------
PyLegendInteger
The remainder expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["id_mod_3"] = frame["Order Id"] % 3
frame.head(3).to_pandas()
"""
PyLegendInteger.__validate__param_to_be_integer(other, "Integer modulo (%) parameter")
other_op = PyLegendInteger.__convert_to_integer_expr(other)
return PyLegendInteger(PyLegendIntegerModuloExpression(self.__value_copy, other_op))
[docs]
@grammar_method
def __rmod__(
self,
other: PyLegendUnion[int, "PyLegendInteger"]
) -> "PyLegendUnion[PyLegendNumber, PyLegendInteger]":
"""
Reflected modulo (``int % expr``).
Called when a Python ``int`` is on the left side of ``%``.
Parameters
----------
other : int or PyLegendInteger
The left-hand operand (dividend).
Returns
-------
PyLegendInteger
The remainder expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
"""
PyLegendNumber.validate_param_to_be_number(other, "Integer modulo (%) parameter")
other_op = PyLegendInteger.__convert_to_integer_expr(other)
return PyLegendInteger(PyLegendIntegerModuloExpression(other_op, self.__value_copy))
[docs]
@grammar_method
def __abs__(self) -> "PyLegendInteger":
"""
Absolute value (``abs(expr)``).
Returns
-------
PyLegendInteger
The absolute value expression.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["abs_diff"] = abs(frame["Order Id"] - 10255)
frame.head(3).to_pandas()
"""
return PyLegendInteger(PyLegendIntegerAbsoluteExpression(self.__value_copy))
[docs]
@grammar_method
def __neg__(self) -> "PyLegendInteger":
"""
Unary negation (``-expr``).
Returns
-------
PyLegendInteger
The negated expression.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["neg_id"] = -frame["Order Id"]
frame.head(3).to_pandas()
"""
return PyLegendInteger(PyLegendIntegerNegativeExpression(self.__value_copy))
[docs]
@grammar_method
def __pos__(self) -> "PyLegendInteger":
"""
Unary positive (``+expr``).
Returns the expression unchanged.
Returns
-------
PyLegendInteger
The same expression.
"""
return self
[docs]
@grammar_method
def __invert__(self) -> "PyLegendInteger":
"""
Bitwise NOT (``~expr``).
Flips all bits of the integer value.
Returns
-------
PyLegendInteger
The bitwise-inverted expression.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["bit_not"] = ~frame["Order Id"]
frame.head(3).to_pandas()
"""
return PyLegendInteger(PyLegendIntegerBitNotExpression(self.__value_copy))
[docs]
@grammar_method
def __and__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Bitwise AND (``&``).
Parameters
----------
other : int or PyLegendInteger
The right-hand operand.
Returns
-------
PyLegendInteger
The bitwise AND expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["bit_and"] = frame["Order Id"] & 0xFF
frame.head(3).to_pandas()
"""
return self._create_binary_expression(other, PyLegendIntegerBitAndExpression, "and (&)")
[docs]
@grammar_method
def __rand__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Reflected bitwise AND (``int & expr``).
Parameters
----------
other : int or PyLegendInteger
The left-hand operand.
Returns
-------
PyLegendInteger
The bitwise AND expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
"""
return self._create_binary_expression(other, PyLegendIntegerBitAndExpression, "and (&)", reverse=True)
[docs]
@grammar_method
def __or__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Bitwise OR (``|``).
Parameters
----------
other : int or PyLegendInteger
The right-hand operand.
Returns
-------
PyLegendInteger
The bitwise OR expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["bit_or"] = frame["Order Id"] | 0x0F
frame.head(3).to_pandas()
"""
return self._create_binary_expression(other, PyLegendIntegerBitOrExpression, "or (|)")
[docs]
@grammar_method
def __ror__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Reflected bitwise OR (``int | expr``).
Parameters
----------
other : int or PyLegendInteger
The left-hand operand.
Returns
-------
PyLegendInteger
The bitwise OR expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
"""
return self._create_binary_expression(other, PyLegendIntegerBitOrExpression, "or (|)", reverse=True)
[docs]
@grammar_method
def __xor__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Bitwise XOR (``^``).
Parameters
----------
other : int or PyLegendInteger
The right-hand operand.
Returns
-------
PyLegendInteger
The bitwise XOR expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["bit_xor"] = frame["Order Id"] ^ 0xFF
frame.head(3).to_pandas()
"""
return self._create_binary_expression(other, PyLegendIntegerBitXorExpression, "xor (^)")
[docs]
@grammar_method
def __rxor__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Reflected bitwise XOR (``int ^ expr``).
Parameters
----------
other : int or PyLegendInteger
The left-hand operand.
Returns
-------
PyLegendInteger
The bitwise XOR expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
"""
return self._create_binary_expression(other, PyLegendIntegerBitXorExpression, "xor (^)", reverse=True)
[docs]
@grammar_method
def __lshift__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Bitwise left shift (``<<``).
Parameters
----------
other : int or PyLegendInteger
The number of positions to shift.
Returns
-------
PyLegendInteger
The left-shifted expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["shifted"] = frame["Order Id"] << 2
frame.head(3).to_pandas()
"""
return self._create_binary_expression(other, PyLegendIntegerBitShiftLeftExpression, "left shift (<<)")
[docs]
@grammar_method
def __rlshift__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Reflected bitwise left shift (``int << expr``).
Parameters
----------
other : int or PyLegendInteger
The left-hand operand.
Returns
-------
PyLegendInteger
The left-shifted expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
"""
return self._create_binary_expression(other, PyLegendIntegerBitShiftLeftExpression, "left shift (<<)", reverse=True)
[docs]
@grammar_method
def __rshift__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Bitwise right shift (``>>``).
Parameters
----------
other : int or PyLegendInteger
The number of positions to shift.
Returns
-------
PyLegendInteger
The right-shifted expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
Examples
--------
.. ipython:: python
import pylegend
frame = pylegend.samples.pandas_api.northwind_orders_frame()
frame["shifted"] = frame["Order Id"] >> 2
frame.head(3).to_pandas()
"""
return self._create_binary_expression(other, PyLegendIntegerBitShiftRightExpression, "right shift (>>)")
[docs]
@grammar_method
def __rrshift__(self, other: PyLegendUnion[int, "PyLegendInteger"]) -> "PyLegendInteger":
"""
Reflected bitwise right shift (``int >> expr``).
Parameters
----------
other : int or PyLegendInteger
The left-hand operand.
Returns
-------
PyLegendInteger
The right-shifted expression.
Raises
------
TypeError
If *other* is not an ``int`` or ``PyLegendInteger``.
"""
return self._create_binary_expression(other, PyLegendIntegerBitShiftRightExpression, "right shift (>>)", reverse=True)
def _create_binary_expression(
self,
other: PyLegendUnion[int, "PyLegendInteger"],
expression_class: type,
operation_name: str,
reverse: bool = False
) -> "PyLegendInteger":
PyLegendInteger.__validate__param_to_be_integer(other, f"Integer {operation_name} parameter")
other_op = PyLegendInteger.__convert_to_integer_expr(other)
if reverse:
return PyLegendInteger(expression_class(other_op, self.__value_copy))
return PyLegendInteger(expression_class(self.__value_copy, other_op))
@staticmethod
def __convert_to_integer_expr(
val: PyLegendUnion[int, "PyLegendInteger"]
) -> PyLegendExpressionIntegerReturn:
if isinstance(val, int):
return PyLegendIntegerLiteralExpression(val)
return val.__value_copy
@staticmethod
def __validate__param_to_be_integer(param: PyLegendUnion[int, "PyLegendInteger"], desc: str) -> None:
if not isinstance(param, (int, PyLegendInteger)):
raise TypeError(desc + " should be a int or an integer expression (PyLegendInteger)."
" Got value " + str(param) + " of type: " + str(type(param)))