Source code for rads.rpn

r"""Reverse Polish Notation calculator.

Exceptions
----------

* :class:`StackUnderflowError`


Classes
-------

* :class:`Expression`
* :class:`Token`
* :class:`Literal`
* :class:`Variable`
* :class:`Operator`


Functions
---------

* :func:`token`


Constants
---------

==============  =================
Keyword         Value
==============  =================
:data:`PI`      3.141592653589793
:data:`E`       2.718281828459045
==============  =================


Operators
---------

===============  ==============================================================
Keyword          Description
===============  ==============================================================
:data:`SUB`      a = x - y
:data:`ADD`      a = x + y
:data:`MUL`      a = x*y
:data:`POP`      remove top of stack
:data:`NEG`      a = -x
:data:`ABS`      a = \|x\|
:data:`INV`      a = 1/x
:data:`SQRT`     a = sqrt(x)
:data:`SQR`      a = x*x
:data:`EXP`      a = exp(x)
:data:`LOG`      a = ln(x)
:data:`LOG10`    a = log10(x)
:data:`SIN`      a = sin(x)
:data:`COS`      a = cos(x)
:data:`TAN`      a = tan(x)
:data:`SIND`     a = sin(x) [x in degrees]
:data:`COSD`     a = cos(x) [x in degrees]
:data:`TAND`     a = tan(x) [x in degrees]
:data:`SINH`     a = sinh(x)
:data:`COSH`     a = cosh(x)
:data:`TANH`     a = tanh(x)
:data:`ASIN`     a = arcsin(x)
:data:`ACOS`     a = arccos(x)
:data:`ATAN`     a = arctan(x)
:data:`ASIND`    a = arcsin(x) [a in degrees]
:data:`ACOSD`    a = arccos(x) [a in degrees]
:data:`ATAND`    a = arctan(x) [a in degrees]
:data:`ASINH`    a = arcsinh(x)
:data:`ACOSH`    a = arccosh(x)
:data:`ATANH`    a = arctanh(x)
:data:`ISNAN`    a = 1 if x is NaN; a = 0 otherwise
:data:`ISAN`     a = 0 if x is NaN; a = 1 otherwise
:data:`RINT`     a is nearest integer to x
:data:`NINT`     a is nearest integer to x
:data:`CEIL`     a is nearest integer greater or equal to x
:data:`CEILING`  a is nearest integer greater or equal to x
:data:`FLOOR`    a is nearest integer less or equal to x
:data:`D2R`      convert x from degrees to radians
:data:`R2D`      convert x from radian to degrees
:data:`YMDHMS`   convert from seconds since 1985 to YYMMDDHHMMSS format (float)
:data:`SUM`      a[i] = x[1] + ... + x[i] while skipping all NaN
:data:`DIF`      a[i] = x[i]-x[i-1]; a[1] = NaN
:data:`DUP`      duplicate the last item on the stack
:data:`DIV`      a = x/y
:data:`POW`      a = x**y
:data:`FMOD`     a = x modulo y
:data:`MIN`      a = the lesser of x and y [element wise]
:data:`MAX`      a = the greater of x and y [element wise]
:data:`ATAN2`    a = arctan2(x, y)
:data:`HYPOT`    a = sqrt(x*x+y*y)
:data:`R2`       a = x*x + y*y
:data:`EQ`       a = 1 if x == y; a = 0 otherwise
:data:`NE`       a = 0 if x == y; a = 1 otherwise
:data:`LT`       a = 1 if x < y; a = 0 otherwise
:data:`LE`       a = 1 if x ≤ y; a = 0 otherwise
:data:`GT`       a = 1 if x > y; a = 0 otherwise
:data:`GE`       a = 1 if x ≥ y; a = 0 otherwise
:data:`NAN`      a = NaN if x == y; a = x otherwise
:data:`AND`      a = y if x is NaN; a = x otherwise
:data:`OR`       a = NaN if y is NaN; a = x otherwise
:data:`IAND`     a = bitwise AND of x and y
:data:`IOR`      a = bitwise OR of x and y
:data:`BTEST`    a = 1 if bit y of x is set; a = 0 otherwise
:data:`AVG`      a = 0.5*(x+y) [when x or y is NaN a returns the other value]
:data:`DXDY`     a[i] = (x[i+1]-x[i-1])/(y[i+1]-y[i-1]); a[1] = a[n] = NaN
:data:`EXCH`     exchange the top two stack elements
:data:`INRANGE`  a = 1 if x is between y and z (inclusive); a = 0 otherwise
:data:`BOXCAR`   filter x along dimension y with boxcar of length z
:data:`GAUSS`    filter x along dimension y with Gaussian of width sigma z
===============  ==============================================================

"""

import math
from abc import ABC, abstractmethod
from datetime import timedelta
from itertools import chain
from numbers import Integral
from typing import (
    TYPE_CHECKING,
    AbstractSet,
    Any,
    Iterable,
    Iterator,
    List,
    Mapping,
    MutableSequence,
    Optional,
    Sequence,
    Tuple,
    Union,
    cast,
    overload,
)

import numpy as np  # type: ignore
from astropy.convolution import Box1DKernel, Gaussian1DKernel, convolve  # type: ignore

from .constants import EPOCH
from .datetime64util import ymdhmsus
from .typing import FloatOrArray
from .utility import fortran_float

# TODO: Change to functools.cached_property when dropping support for
#       Python 3.7
if TYPE_CHECKING:
    # property behaves properly with Mypy but cached_property does not, even
    # with the same type stub.
    cached_property = property
else:
    from cached_property import cached_property

__all__ = [
    "StackUnderflowError",
    "Expression",
    "CompleteExpression",
    "Token",
    "Literal",
    "PI",
    "E",
    "Variable",
    "Operator",
    "token",
    "SUB",
    "ADD",
    "MUL",
    "POP",
    "NEG",
    "ABS",
    "INV",
    "SQRT",
    "SQR",
    "EXP",
    "LOG",
    "LOG10",
    "SIN",
    "COS",
    "TAN",
    "SIND",
    "COSD",
    "TAND",
    "SINH",
    "COSH",
    "TANH",
    "ASIN",
    "ACOS",
    "ATAN",
    "ASIND",
    "ACOSD",
    "ATAND",
    "ASINH",
    "ACOSH",
    "ATANH",
    "ISNAN",
    "ISAN",
    "RINT",
    "NINT",
    "CEIL",
    "CEILING",
    "FLOOR",
    "D2R",
    "R2D",
    "YMDHMS",
    "SUM",
    "DIF",
    "DUP",
    "DIV",
    "POW",
    "FMOD",
    "MIN",
    "MAX",
    "ATAN2",
    "HYPOT",
    "R2",
    "EQ",
    "NE",
    "LT",
    "LE",
    "GT",
    "GE",
    "NAN",
    "AND",
    "OR",
    "IAND",
    "IOR",
    "BTEST",
    "AVG",
    "DXDY",
    "EXCH",
    "INRANGE",
    "BOXCAR",
    "GAUSS",
]


[docs]class StackUnderflowError(Exception): """Raised when the stack is too small for the operation. When this is raised the stack will exist in the state that it was before the operation was attempted. Therefore, it is not necessary to repair the stack. .. note: Due to the static checker in :class:`Expression` this should never be raised except for manually evaluating a :class:`Token`. """
[docs]class Token(ABC): """Base class of all RPN tokens. .. seealso:: :class:`Literal` A literal numeric/array value. :class:`Variable` A variable to be looked up from the environment. :class:`Operator` Base class of operators that modify the stack. """ @property @abstractmethod def pops(self) -> int: """Elements removed off the stack by calling the token.""" @property @abstractmethod def puts(self) -> int: """Elements placed on the stack by calling the token."""
[docs] @abstractmethod def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: """Perform token's action on the given `stack`. The actions currently supported are: * Place literal value on the stack. * Place variable on the stack from the `environment`. * Perform operation on the stack. * Any combination of the above. .. note:: This must be overridden for all tokens. :param stack: The stack of numbers/arrays to operate on. :param environment: The dictionary like object providing the immutable environment. """
[docs]class Literal(Token): """Literal value token.""" @property def pops(self) -> int: """Elements removed off the stack by calling the token.""" return 0 @property def puts(self) -> int: """Elements placed on the stack by calling the token.""" return 1 @property def value(self) -> Union[int, float, bool]: """Value of the literal.""" return self._value def __init__(self, value: Union[int, float, bool]): """ :param value: Value of the literal. :raises ValueError: If `value` is not a number. """ if not isinstance(value, (int, float, bool)): raise TypeError("'value' must be an int, float, or bool") self._value: Union[int, float, bool] = value
[docs] def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: """Place literal value on top of the given `stack`. :param stack: The stack of numbers/arrays to place the value on. :param environment: The dictionary like object providing the immutable environment. Not used by this method. """ stack.append(self.value)
def __eq__(self, other: Any) -> bool: return isinstance(other, Literal) and self.value == other.value def __ne__(self, other: Any) -> bool: return not isinstance(other, Literal) or self.value != other.value def __lt__(self, other: Any) -> bool: if not isinstance(other, Literal): return NotImplemented return self.value < other.value def __le__(self, other: Any) -> bool: if not isinstance(other, Literal): return NotImplemented return self.value <= other.value def __gt__(self, other: Any) -> bool: if not isinstance(other, Literal): return NotImplemented return self.value > other.value def __ge__(self, other: Any) -> bool: if not isinstance(other, Literal): return NotImplemented return self.value >= other.value def __repr__(self) -> str: return f"{self.__class__.__qualname__}({repr(self._value)})" def __str__(self) -> str: return str(self._value)
[docs]class Variable(Token): """Environment variable token. This is a place holder to lookup and place a number/array from an environment mapping onto the stack. """ @property def pops(self) -> int: """Elements removed off the stack by calling the token.""" return 0 @property def puts(self) -> int: """Elements placed on the stack by calling the token.""" return 1 @property def name(self) -> str: """Name of the variable, used to lookup value in the environment.""" return self._name def __init__(self, name: str): """ :param name: Name of the variable, this is what will be used to lookup the variables value in the environment mapping. """ if not isinstance(name, str): raise TypeError(f"'name' must be a string, got '{type(name)}'") if not name.isidentifier(): raise ValueError(f"'name' must be a valid identifier, got '{name}'") self._name = name
[docs] def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: """Get variable value from `environment` and place on stack. :param stack: The stack of numbers/arrays to place value on. :param environment: The dictionary like object to lookup the variable's value from. :raises KeyError: If the variable cannot be found in the given `environment`. """ stack.append(environment[self.name])
def __eq__(self, other: Any) -> bool: return isinstance(other, Variable) and other.name == self.name def __ne__(self, other: Any) -> bool: return not isinstance(other, Variable) or other.name != self.name def __repr__(self) -> str: return f"{self.__class__.__qualname__}({repr(self._name)})" def __str__(self) -> str: return str(self._name)
[docs]class Operator(Token, ABC): """Base class of all RPN operators.""" def __init__(self, name: str): """ :param name: Name of the operator. """ self._name = name def __copy__(self) -> Any: # sentinel object so copy will break it return self def __deepcopy__(self, memo: Any = None) -> Any: # sentinel object so copy will break it return self def __repr__(self) -> str: return self._name
[docs]class Expression(Sequence[Token], Token): r"""Reverse Polish Notation expression. .. note:: :class:`Expression`\ s cannot be evaluated as they may not be syntactically correct. For evaluation :class:`CompleteExpression`\ s are required. Expressions can be used in three ways: * Can be converted to a :class:`CompleteExpression` if :attr:`pops` and :attr:`puts` are both 1 with the :func:`complete` method. * Can be added to the end of a :class:`CompleteExpression` producing a :class:`CompleteExpression` if the :class:`Expression` has :attr:`pops` = 1 and :attr:`puts` = 1 or a :class:`Expression` otherwise. * Can be added to the end of an :class:`Expression` producing a :class:`CompleteExpression` if the combination produces an expression with :attr:`pops` = 0 and :attr:`puts` = 1. * Can be used as a :class:`Token` in another expression. .. seealso:: :class:`CompleteExpression` For a expression that can be evaluated on it's own. """ _tokens: List[Token]
[docs] @cached_property def pops(self) -> int: """Elements removed off the stack by calling the token.""" return self._simulate()[0]
[docs] @cached_property def puts(self) -> int: """Elements placed on the stack by calling the token.""" return self._simulate()[1]
[docs] @cached_property def variables(self) -> AbstractSet[str]: """Set of variables needed to evaluate the expression.""" return {t.name for t in self._tokens if isinstance(t, Variable)}
def __init__(self, tokens: Union[str, Iterable[Union[float, str, Token]]]): r""" :param tokens: A Reverse Polish Notation expression given as a sequence of tokens or a string of tokens. .. note:: This parameter is very forgiving. If given a sequence of tokens and some of the elements are not :class:`Token`\ s then an attempt will be made to convert them to :class:`Token`\ s. Because of this both numbers and strings can be given in the sequence of `tokens`. """ if isinstance(tokens, str): self._tokens = [token(t) for t in tokens.split()] else: self._tokens = [] for token_ in tokens: if isinstance(token_, Token): self._tokens.append(token_) elif isinstance(token_, (int, float, bool)): self._tokens.append(Literal(token_)) else: self._tokens.append(token(token_))
[docs] def complete(self) -> "CompleteExpression": """Upgrade to a :class:`CompleteExpression` if possible. :return: A complete expression, assuming this partial expression takes zero inputs and provides one output. :raises ValueError: If the partial expression is not a valid expression. """ return CompleteExpression(self._tokens)
[docs] def is_complete(self) -> bool: """Determine if can be upgraded to :class:`CompleteExpression`. :return: True if this expression can be upgraded to a :class:`CompleteExpression` without error with the :func:`complete` method. """ return self.pops == 0 and self.puts == 1
[docs] def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: """Evaluate the expression as a token on the given `stack`. :param stack: The stack of numbers/arrays to operate on. :param environment: The dictionary like object providing the immutable environment. :raises StackUnderflowError: If the expression underflows the stack. """ for token_ in self._tokens: token_(stack, environment)
def __contains__(self, item: Any) -> bool: if isinstance(item, Token): return item in self._tokens return False # TODO: Remove the F811 statements bellow once # https://github.com/PyCQA/pyflakes/pull/435 makes it into a release # (that is the next version after v2.1.1). # Also update requirements to match. @overload # noqa: F811 def __getitem__(self, item: int) -> Token: ... @overload # noqa: F811 def __getitem__(self, item: slice) -> "Expression": ... def __getitem__( # noqa: F811 self, item: Union[int, slice] ) -> Union[Token, "Expression"]: if isinstance(item, slice): return Expression(self._tokens[item]) return self._tokens[item] def __iter__(self) -> Iterator[Token]: for token_ in self._tokens: yield token_ def __len__(self) -> int: return len(self._tokens) def __eq__(self, other: Any) -> bool: if not isinstance(other, Expression): return NotImplemented return self._tokens == other._tokens def __ne__(self, other: Any) -> bool: if not isinstance(other, Expression): return NotImplemented return self._tokens != other._tokens
[docs] def __add__(self, other: Any) -> "Expression": if not isinstance(other, Expression): return NotImplemented try: return CompleteExpression(chain(self, other)) except ValueError: return Expression(chain(self, other))
def __repr__(self) -> str: return f"{self.__class__.__qualname__}({repr(self._tokens)})" def __str__(self) -> str: return " ".join(str(t) for t in self._tokens) def _simulate(self) -> Tuple[int, int]: """Simulate the expression to determine inputs and outputs. :return: A tuple of the number of inputs the expression takes and the number of outputs from the expression. """ inputs = 0 outputs = 0 for token_ in self._tokens: outputs -= token_.pops inputs = max(inputs, -outputs) outputs += token_.puts return inputs, outputs + inputs
[docs]class CompleteExpression(Expression): """Reverse Polish Notation expression that can be evaluated.""" def __init__(self, tokens: Union[str, Iterable[Union[float, str, Token]]]): r""" :param tokens: A Reverse Polish Notation expression given as a sequence of tokens or a string of tokens. .. note:: This parameter is very forgiving. If given a sequence of tokens and some of the elements are not :class:`Token`\ s then then an attempt will be made to convert them to :class:`Token`\ s. Because of this both numbers and strings can be given in the sequence of `tokens`. :raises ValueError: If the sequence or string of `tokens` represents an invalid expression. This exception also indicates which token makes the expression invalid. """ super().__init__(tokens) # check the syntax, raise ValueError if invalid self._check()
[docs] def complete(self) -> "CompleteExpression": """Return this expression as it is already complete. :return: This complete expression. """ return self
[docs] def eval( self, environment: Optional[Mapping[str, FloatOrArray]] = None ) -> FloatOrArray: """Evaluate the expression and return a numerical or logical result. :param environment: A mapping to lookup variables in when evaluating the expression. If not provided an empty mapping will be used, this is fine as long as the expression does not contain any variables. This can be ascertained by checking the with the :attr:`variables` attribute: .. code-block:: python if not expression.variables: expression.eval() If the evaluation is lengthy or there are side effects to key lookup in the `environment` it may be beneficial to check for any missing variables first: .. code-block:: python missing_vars = expression.variables.difference(environment) :return: The numeric or logical result of the expression. :raises TypeError: If there is a type mismatch with one of the operators and a value. .. note:: While this class includes a static syntax checker that runs upon initialization it does not know the type of variables in the given `environment` ahead of time. :raises KeyError: If the expression contains a variable that is not within the given `environment`. :raises IndexError, ValueError, RuntimeError, ZeroDivisionError: If arguments to operators in the expression do not have the proper dimensions or values for the operators to produce a result. See the documentation of each operator for specifics. """ if not environment: environment = {} stack: List[FloatOrArray] = [] for token_ in self._tokens: token_(stack, environment) # NOTE: The stack will always have exactly one element at this point # due to the static checker that runs at initialization. return stack[0]
def _format_syntax_error(self, string: str, token_: Optional[int] = None) -> str: tokens_str = [str(t) for t in self._tokens] result = f"{string}\n{' '.join(tokens_str)}" if token_ is None: return result if token_ == 0: return result + "\n^" return result + f"\n{' ' * len(' '.join(tokens_str[:token_])) + ' ^'}" def _check(self) -> None: """Check the syntax of the expression. :raises ValueError: If the expression is incomplete. This error is aimed at debugging the expression so it is very verbose. """ stack_size = 0 for i, token_ in enumerate(self._tokens): stack_size -= token_.pops if stack_size < 0: raise ValueError( self._format_syntax_error( f"'{token_}' takes {token_.pops} argument(s) but the stack" f" will only have {stack_size + token_.pops} element(s)", i, ) ) stack_size += token_.puts if stack_size == 0: raise ValueError( self._format_syntax_error("expression does not produce a result") ) if stack_size > 1: raise ValueError( self._format_syntax_error( f"expression produces too many results ({stack_size}), " "expected 1" ) )
[docs]def token(string: str) -> Token: """Parse string token into a :class:`Token`. There are three types of tokens that can result from this function: * :class:`Literal` - a literal integer or float * :class:`Variable` - a variable to looked up in the environment * :class:`Operator` - an operator to modify the stack :param string: String to parse into a :class:`Token`. :return: Parsed token. :raises TypeError: If not given a string. :raises ValueError: If `string` is not a valid token. """ if not isinstance(string, str): raise TypeError(f"expected 'str' got {type(string)}") if string in _KEYWORDS: return _KEYWORDS[string] try: return Literal(int(string)) except ValueError: pass try: return Literal(fortran_float(string)) except ValueError: if string.isidentifier(): return Variable(string) raise ValueError(f"invalid RPN token '{string}'")
# NOTE: The operators in this file are in the same order as they are in the # RADS user manual. class _SUBType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x - y) # TODO: write as many as possible to use operators instead of numpy functions class _ADDType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x + y) class _MULType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x * y) class _POPType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 0 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: _get_x(stack) class _NEGType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(-x) class _ABSType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.absolute(x)) class _INVType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(1 / x) class _SQRTType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.sqrt(x)) class _SQRType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.square(x)) class _EXPType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.exp(x)) class _LOGType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.log(x)) class _LOG10Type(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.log10(x)) class _SINType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.sin(x)) class _COSType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.cos(x)) class _TANType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.tan(x)) class _SINDType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.sin(np.deg2rad(x))) class _COSDType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.cos(np.deg2rad(x))) class _TANDType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.tan(np.deg2rad(x))) class _SINHType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.sinh(x)) class _COSHType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.cosh(x)) class _TANHType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.tanh(x)) class _ASINType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.arcsin(x)) class _ACOSType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.arccos(x)) class _ATANType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.arctan(x)) class _ASINDType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.rad2deg(np.arcsin(x))) class _ACOSDType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.rad2deg(np.arccos(x))) class _ATANDType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.rad2deg(np.arctan(x))) class _ASINHType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.arcsinh(x)) class _ACOSHType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.arccosh(x)) class _ATANHType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.arctanh(x)) class _ISNANType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.isnan(x)) class _ISANType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.logical_not(np.isnan(x))) class _RINTType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.round(x)) class _CEILType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.ceil(x)) class _FLOORType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.floor(x)) class _D2RType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.deg2rad(x)) class _R2DType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.rad2deg(x)) class _YMDHMSType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) if isinstance(x, np.ndarray): time = np.datetime64(EPOCH) + (x * 1e6).astype("timedelta64[us]") year, month, day, hour, minute, second, microsecond = ymdhmsus(time) a = ( (year % 100) * 1e10 + month * 1e8 + day * 1e6 + hour * 1e4 + minute * 1e2 + second + microsecond * 1e-6 ) else: time = EPOCH + timedelta(seconds=x) a = ( (time.year % 100) * 1e10 + time.month * 1e8 + time.day * 1e6 + time.hour * 1e4 + time.minute * 1e2 + time.second + time.microsecond * 1e-6 ) stack.append(a) class _SUMType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.nansum(x)) class _DIFType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(np.diff(np.ravel(x), prepend=np.nan)) class _DUPType(Operator): @property def pops(self) -> int: return 1 @property def puts(self) -> int: return 2 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x = _get_x(stack) stack.append(x) stack.append(x) class _DIVType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x / y) class _POWType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(np.power(x, y)) class _FMODType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(np.fmod(x, y)) class _MINType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(np.minimum(x, y)) class _MAXType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(np.maximum(x, y)) class _ATAN2Type(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(np.arctan2(x, y)) class _HYPOTType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(np.hypot(x, y)) class _R2Type(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x ** 2 + y ** 2) class _EQType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x == y) class _NEType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x != y) class _LTType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x < y) class _LEType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x <= y) class _GTType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x > y) class _GEType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(x >= y) class _NANType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) if isinstance(x, np.ndarray) or isinstance(y, np.ndarray): x, y = np.broadcast_arrays(x, y) # upgrade integers to doubles if np.issubdtype(x.dtype, np.integer): a = np.copy(x).astype("double") else: a = np.copy(x) a[x == y] = np.nan stack.append(a) else: stack.append(np.nan if x == y else x) class _ANDType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) if isinstance(x, np.ndarray) or isinstance(y, np.ndarray): x, y = np.broadcast_arrays(x, y) a = np.copy(x) isnan = np.isnan(x) a[isnan] = y[isnan] stack.append(a) else: stack.append(y if np.isnan(x) else x) class _ORType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) if isinstance(x, np.ndarray) or isinstance(y, np.ndarray): x, y = np.broadcast_arrays(x, y) # upgrade integers to doubles if np.issubdtype(x.dtype, np.integer): a = np.copy(x).astype("double") else: a = np.copy(x) a[np.isnan(y)] = np.nan isnan = np.isnan(x) a[isnan] = y[isnan] stack.append(a) else: stack.append(np.nan if np.isnan(y) else x) class _IANDType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(np.bitwise_and(x, y)) class _IORType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(np.bitwise_or(x, y)) class _BTESTType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) if not _is_integer(x): raise TypeError("'x' must be an integer type") if not _is_integer(y): raise TypeError("'y' must be an integer type") stack.append(np.bitwise_and(cast(int, x), 1 << cast(int, y)) != 0) class _AVGType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) if isinstance(x, np.ndarray) or isinstance(y, np.ndarray): x, y = np.broadcast_arrays(x, y) stacked = np.stack((x, y)) non_nan = np.sum(~np.isnan(stacked), axis=0) stack.append(np.nansum(stacked, axis=0) / non_nan) else: if np.isnan(x): stack.append(y) elif np.isnan(y): stack.append(x) else: stack.append((x + y) / 2) class _DXDYType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) if isinstance(x, np.ndarray) or isinstance(y, np.ndarray): x, y = np.broadcast_arrays(x, y) x = np.ravel(x) y = np.ravel(y) if np.size(x) < 3: a = np.empty(np.shape(x)) a[:] = np.nan else: dxdy = (x[2:] - x[:-2]) / (y[2:] - y[:-2]) a = np.concatenate(([np.nan], dxdy, [np.nan])) else: a = float("nan") stack.append(a) class _EXCHType(Operator): @property def pops(self) -> int: return 2 @property def puts(self) -> int: return 2 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y = _get_xy(stack) stack.append(y) stack.append(x) class _INRANGEType(Operator): @property def pops(self) -> int: return 3 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y, z = _get_xyz(stack) if any(isinstance(v, np.ndarray) for v in [x, y, z]): x, y, z = np.broadcast_arrays(x, y, z) a = np.logical_and(y <= x, x <= z) else: a = y <= x <= z stack.append(a) class _BOXCARType(Operator): @property def pops(self) -> int: return 3 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y, z = _get_xyz(stack) if np.size(x) == 1: a = x else: if np.size(y) != 1: raise ValueError("'y' of 'x y z BOXCAR a' must be a scalar") if np.size(z) != 1: raise ValueError("'z' of 'x y z BOXCAR a' must be a scalar") if len(np.shape(x)) <= y: raise IndexError( f"requested filter along dimension {y} but " f"'x' has only {len(np.shape(x))} dimensions" ) kernel = Box1DKernel(z) # split into slices along dimension y tmp = np.moveaxis(x, y, -1) slices = [] for slice_ in tmp.reshape(-1, np.shape(x)[y]): # filter each slice slices.append( convolve(slice_, kernel, boundary="extend", preserve_nan=True) ) # recombine slices a = np.moveaxis(np.array(slices).reshape(tmp.shape), -1, y) stack.append(a) class _GAUSSType(Operator): @property def pops(self) -> int: return 3 @property def puts(self) -> int: return 1 def __call__( self, stack: MutableSequence[FloatOrArray], environment: Mapping[str, FloatOrArray], ) -> None: x, y, z = _get_xyz(stack) if np.size(x) == 1: a = x else: if np.size(y) != 1: raise ValueError("'y' of 'x y z GAUSS a' must be a scalar") if np.size(z) != 1: raise ValueError("'z' of 'x y z GAUSS a' must be a scalar") if len(np.shape(x)) <= y: raise IndexError( f"requested filter along dimension {y} but " f"'x' has only {len(np.shape(x))} dimensions" ) kernel = Gaussian1DKernel(z) # split into slices along dimension y tmp = np.moveaxis(x, y, -1) slices = [] for slice_ in tmp.reshape(-1, np.shape(x)[y]): # filter each slice slices.append( convolve(slice_, kernel, boundary="extend", preserve_nan=True) ) # recombine slices a = np.moveaxis(np.array(slices).reshape(tmp.shape), -1, y) stack.append(a) # constants PI = Literal(math.pi) E = Literal(math.e) # operators SUB = _SUBType("SUB") """Subtract one number/array from another. x y SUB a a = x - y """ ADD = _ADDType("ADD") """Add two numbers/arrays. x y ADD a a = x + y """ MUL = _MULType("MUL") """Multiply two numbers/arrays. x y MUL a a = x*y """ POP = _POPType("POP") """Remove top of stack. x POP remove last item from stack """ NEG = _NEGType("NEG") """Negate number/array. x NEG a a = -x """ ABS = _ABSType("ABS") r"""Absolute value of number/array. x ABS a a = \|x\| """ INV = _INVType("INV") """Invert number/array. x INV a a = 1/x """ SQRT = _SQRTType("SQRT") """Compute square root of number/array. x SQRT a a = sqrt(x) """ SQR = _SQRType("SQR") """Square number/array. x SQR a a = x*x """ EXP = _EXPType("EXP") """Exponential of number/array. x EXP a a = exp(x) """ LOG = _LOGType("LOG") """Natural logarithm of number/array. x LOG a a = ln(x) """ LOG10 = _LOG10Type("LOG10") """Compute base 10 logarithm of number/array. x LOG10 a a = log10(x) """ SIN = _SINType("SIN") """Sine of number/array [in radians]. x SIN a a = sin(x) """ COS = _COSType("COS") """Cosine of number/array [in radians]. x COS a a = cos(x) """ TAN = _TANType("TAN") """Tangent of number/array [in radians]. x TAN a a = tan(x) """ SIND = _SINDType("SIND") """Sine of number/array [in degrees]. x SIND a a = sin(x) [x in degrees] """ COSD = _COSDType("COSD") """Cosine of number/array [in degrees]. x COSD a a = cos(x) [x in degrees] """ TAND = _TANDType("TAND") """Tangent of number/array [in degrees]. x TAND a a = tan(x) [x in degrees] """ SINH = _SINHType("SINH") """Hyperbolic sine of number/array. x SINH a a = sinh(x) """ COSH = _COSHType("COSH") """Hyperbolic cosine of number/array. x COSH a a = cosh(x) """ TANH = _TANHType("TANH") """Hyperbolic tangent of number/array. x TANH a a = tanh(x) """ ASIN = _ASINType("ASIN") """Inverse sine of number/array [in radians]. x ASIN a a = arcsin(x) """ ACOS = _ACOSType("ACOS") """Inverse cosine of number/array [in radians]. x ACOS a a = arccos(x) """ ATAN = _ATANType("ATAN") """Inverse tangent of number/array [in radians]. x ATAN a a = arctan(x) """ ASIND = _ASINDType("ASIND") """Inverse sine of number/array [in degrees]. x ASIND a a = arcsin(x) [a in degrees] """ ACOSD = _ACOSDType("ACOSD") """Inverse cosine of number/array [in degrees]. x ACOSD a a = arccos(x) [a in degrees] """ ATAND = _ATANDType("ATAND") """Inverse tangent of number/array [in degrees]. x ATAND a a = arctan(x) [a in degrees] """ ASINH = _ASINHType("ASINH") """Inverse hyperbolic sine of number/array. x ASINH a a = arcsinh(x) """ ACOSH = _ACOSHType("ACOSH") """Inverse hyperbolic cosine of number/array. x ACOSH a a = arccosh(x) """ ATANH = _ATANHType("ATANH") """Inverse hyperbolic tangent of number/array. x ATANH a a = arctanh(x) """ ISNAN = _ISNANType("ISNAN") """Determine if number/array is NaN. x ISNAN a a = 1 if x is NaN; a = 0 otherwise .. note:: Instead of using 1 and 0 pyrads uses True and False which behave as 1 and 0 when treated as numbers. """ ISAN = _ISANType("ISAN") """Determine if number/array is not NaN. x ISAN a a = 0 if x is NaN; a = 1 otherwise .. note:: Instead of using 1 and 0 pyrads uses True and False which behave as 1 and 0 when treated as numbers. """ RINT = _RINTType("RINT") """Round number/array to nearest integer. x RINT a a is nearest integer to x """ NINT = _RINTType("NINT") """Round number/array to nearest integer. x NINT a a is nearest integer to x """ CEIL = _CEILType("CEIL") """Round number/array up to nearest integer. x CEIL a a is nearest integer greater or equal to x """ CEILING = _CEILType("CEILING") """Round number/array up to nearest integer. x CEILING a a is nearest integer greater or equal to x """ FLOOR = _FLOORType("FLOOR") """Round number/array down to nearest integer. x FLOOR a a is nearest integer less or equal to x """ D2R = _D2RType("D2R") """Convert number/array from degrees to radians. x D2R a convert x from degrees to radians """ R2D = _R2DType("R2D") """Convert number/array from radians to degrees. x R2D a convert x from radian to degrees """ YMDHMS = _YMDHMSType("YMDHMS") """Convert number/array from seconds since RADS epoch to YYMMDDHHMMSS. x YMDHMS a convert seconds of 1985 to format YYMMDDHHMMSS .. note:: The top of the stack should be in seconds since the RADS epoch which is currently 1985-01-01 00:00:00 UTC .. note:: The RADS documentation says this format uses a 4 digit year, but RADS uses a 2 digit year so that is what is used here. """ SUM = _SUMType("SUM") """Compute sum over number/array [ignoring NaNs]. x SUM a a[i] = x[1] + ... + x[i] while skipping all NaN """ DIF = _DIFType("DIF") """Compute difference over number/array. x DIF a a[i] = x[i]-x[i-1]; a[1] = NaN """ DUP = _DUPType("DUP") """Duplicate top of stack. x DUP a b duplicate the last item on the stack .. note:: This is duplication by reference, no copy is made. """ DIV = _DIVType("DIV") """Divide one number/array from another. x y DIV a a = x/y """ POW = _POWType("POW") """Raise a number/array to the power of another number/array. x y POW a a = x**y """ FMOD = _FMODType("FMOD") """Remainder of dividing one number/array by another. x y FMOD a a = x modulo y """ MIN = _MINType("MIN") """Minimum of two numbers/arrays [element wise]. x y MIN a a = the lesser of x and y """ MAX = _MAXType("MAX") """Maximum of two numbers/arrays [element wise]. x y MAX a a = the greater of x and y """ ATAN2 = _ATAN2Type("ATAN2") """Inverse tangent of two numbers/arrays giving x and y. x y ATAN2 a a = arctan2(x, y) """ HYPOT = _HYPOTType("HYPOT") """Hypotenuse from numbers/arrays giving legs. x y HYPOT a a = sqrt(x*x+y*y) """ R2 = _R2Type("R2") """Sum of squares of two numbers/arrays. x y R2 a a = x*x + y*y """ EQ = _EQType("EQ") """Compare two numbers/arrays for equality [element wise]. x y EQ a a = 1 if x == y; a = 0 otherwise .. note:: Instead of using 1 and 0 pyrads uses True and False which behave as 1 and 0 when treated as numbers. """ NE = _NEType("NE") """Compare two numbers/arrays for inequality [element wise]. x y NE a a = 0 if x == y; a = 1 otherwise .. note:: Instead of using 1 and 0 pyrads uses True and False which behave as 1 and 0 when treated as numbers. """ LT = _LTType("LT") """Compare two numbers/arrays with < [element wise]. x y LT a a = 1 if x < y; a = 0 otherwise .. note:: Instead of using 1 and 0 pyrads uses True and False which behave as 1 and 0 when treated as numbers. """ LE = _LEType("LE") """Compare two numbers/arrays with <= [element wise]. x y LE a a = 1 if x ≤ y; a = 0 otherwise .. note:: Instead of using 1 and 0 pyrads uses True and False which behave as 1 and 0 when treated as numbers. """ GT = _GTType("GT") """Compare two numbers/arrays with > [element wise]. x y GT a a = 1 if x > y; a = 0 otherwise .. note:: Instead of using 1 and 0 pyrads uses True and False which behave as 1 and 0 when treated as numbers. """ GE = _GEType("GE") """Compare two numbers/arrays with >= [element wise]. x y GE a a = 1 if x ≥ y; a = 0 otherwise .. note:: Instead of using 1 and 0 pyrads uses True and False which behave as 1 and 0 when treated as numbers. """ NAN = _NANType("NAN") """Replace number/array with NaN where it is equal to another. x y NAN a a = NaN if x == y; a = x otherwise """ AND = _ANDType("AND") """Fallback to second number/array when first is NaN [element wise]. x y AND a a = y if x is NaN; a = x otherwise """ OR = _ORType("OR") """Replace number/array with NaN where second is NaN. x y OR a a = NaN if y is NaN; a = x otherwise """ IAND = _IANDType("IAND") """Bitwise AND of two numbers/arrays [element wise]. x y IAND a a = bitwise AND of x and y """ IOR = _IORType("IOR") """Bitwise OR of two numbers/arrays [element wise]. x y IOR a a = bitwise OR of x and y """ BTEST = _BTESTType("BTEST") """Test bit, given by second number/array, in first [element wise]. x y BTEST a a = 1 if bit y of x is set; a = 0 otherwise """ AVG = _AVGType("AVG") """Average of two numbers/arrays ignoring NaNs [element wise]. x y AVG a a = 0.5*(x+y) [when x or y is NaN a returns the other value] """ DXDY = _DXDYType("DXDY") """Compute dx/dy from two numbers/arrays. x y DXDY a a[i] = (x[i+1]-x[i-1])/(y[i+1]-y[i-1]); a[1] = a[n] = NaN """ EXCH = _EXCHType("EXCH") """Exchange top two elements of stack. x y EXCH a b exchange the last two items on the stack (NaNs have no influence) """ INRANGE = _INRANGEType("INRANGE") """Determine if number/array is between two numbers [element wise]. x y z INRANGE a a = 1 if x is between y and z (inclusive) a = 0 otherwise (also in case of any NaN) """ BOXCAR = _BOXCARType("BOXCAR") """Filter number/array with a boxcar filter along a given dimension. x y z BOXCAR a a = filter x along monotonic dimension y with boxcar of length z (NaNs are skipped) .. note:: This may behave slightly differently than the official RADS software at boundaries and at NaN values. :raises IndexError: If x does not have dimension y. :raises ValueError: If y or z is not a scalar. """ GAUSS = _GAUSSType("GAUSS") """Filter number/array with a gaussian filter along a given dimension. x y z GAUSS a a = filter x along monotonic dimension y with Gauss function with sigma z (NaNs are skipped) :raises IndexError: If x does not have dimension y. :raises ValueError: If y or z is not a scalar. """ _KEYWORDS = { "SUB": SUB, "ADD": ADD, "MUL": MUL, "PI": PI, "E": E, "POP": POP, "NEG": NEG, "ABS": ABS, "INV": INV, "SQRT": SQRT, "SQR": SQR, "EXP": EXP, "LOG": LOG, "LOG10": LOG10, "SIN": SIN, "COS": COS, "TAN": TAN, "SIND": SIND, "COSD": COSD, "TAND": TAND, "SINH": SINH, "COSH": COSH, "TANH": TANH, "ASIN": ASIN, "ACOS": ACOS, "ATAN": ATAN, "ASIND": ASIND, "ACOSD": ACOSD, "ATAND": ATAND, "ASINH": ASINH, "ACOSH": ACOSH, "ATANH": ATANH, "ISNAN": ISNAN, "ISAN": ISAN, "RINT": RINT, "NINT": NINT, "CEIL": CEIL, "CEILING": CEILING, "FLOOR": FLOOR, "D2R": D2R, "R2D": R2D, "YMDHMS": YMDHMS, "SUM": SUM, "DIF": DIF, "DUP": DUP, "DIV": DIV, "POW": POW, "FMOD": FMOD, "MIN": MIN, "MAX": MAX, "ATAN2": ATAN2, "HYPOT": HYPOT, "R2": R2, "EQ": EQ, "NE": NE, "LT": LT, "LE": LE, "GT": GT, "GE": GE, "NAN": NAN, "AND": AND, "OR": OR, "IAND": IAND, "IOR": IOR, "BTEST": BTEST, "AVG": AVG, "DXDY": DXDY, "EXCH": EXCH, "INRANGE": INRANGE, "BOXCAR": BOXCAR, "GAUSS": GAUSS, } def _is_integer(x: FloatOrArray) -> bool: """Determine if number is an integer or array of integers. :param x: A number or array of numbers to check. :return: True if x is an integer or array of integers, otherwise False. """ if isinstance(x, (np.ndarray, np.generic)): return issubclass(x.dtype.type, Integral) return isinstance(x, Integral) def _get_x(stack: MutableSequence[FloatOrArray]) -> FloatOrArray: if not stack: raise StackUnderflowError( "attempted to get element from the 'stack' but the stack is empty" ) return stack.pop() def _get_xy(stack: MutableSequence[FloatOrArray],) -> Tuple[FloatOrArray, FloatOrArray]: if not stack: raise StackUnderflowError( "attempted to get 2 elements from the 'stack' but the stack " "is empty" ) if len(stack) < 2: raise StackUnderflowError( f"attempted to get 2 elements from the 'stack' but the stack has " f"only {len(stack)} elements" ) y = stack.pop() x = stack.pop() return x, y def _get_xyz( stack: MutableSequence[FloatOrArray], ) -> Tuple[FloatOrArray, FloatOrArray, FloatOrArray]: if not stack: raise StackUnderflowError( "attempted to get 3 elements from the 'stack' but the stack " "is empty" ) if len(stack) < 3: raise StackUnderflowError( f"attempted to get 3 elements from the 'stack' but the stack has " f"only {len(stack)} elements" ) z = stack.pop() y = stack.pop() x = stack.pop() return x, y, z