Module clingox.theory

This module provides functions to work with clingo's theories.

Example

>>> from clingo.control import Control
>>> from clingox.theory import evaluate
>>>
>>> prg = """... #theory test {
...     term {
...         -  : 3, unary;
...         ** : 2, binary, right;
...         *  : 1, binary, left;
...         +  : 0, binary, left;
...         -  : 0, binary, left
...     };
...     &eval/0 : term, head
... }.
...
... &eval{ 3**5-201 }.
... """
>>>
>>> ctl = Control()
>>> ctl.add('base', [], prg)
>>> ctl.ground([('base', [])])
>>>
>>> atom = next(ctl.theory_atoms)
>>> print(evaluate(atom.elements[0].terms[0]))
42

Functions

def evaluate(term: Any) ‑> Symbol
Expand source code
def evaluate(term: Any) -> Symbol:
    """
    Evaluates the operators in a theory term in the same fashion as clingo
    evaluates its arithmetic functions.

    We use `Any` as a type hint here to allow for evaluating terms that are
    duck typing copmatible with clingo's `TheoryTerm` class.

    Parameters
    ----------
    term
        The theory term to evaluate.

    Returns
    -------
    The evaluated term in form of a symbol.
    """
    return TermEvaluator()(term)

Evaluates the operators in a theory term in the same fashion as clingo evaluates its arithmetic functions.

We use Any as a type hint here to allow for evaluating terms that are duck typing copmatible with clingo's TheoryTerm class.

Parameters

term
The theory term to evaluate.

Returns

The evaluated term in form of a symbol.

def invert_symbol(sym: Symbol) ‑> Symbol
Expand source code
def invert_symbol(sym: Symbol) -> Symbol:
    """
    Inverts the given symbol.

    Parameters
    ----------
    sym
        The symbol to invert.

    Returns
    -------
    The inverted symbol.
    """
    if sym.type == SymbolType.Number:
        return Number(-sym.number)

    if sym.type == SymbolType.Function and sym.name:
        return Function(sym.name, sym.arguments, not sym.positive)

    raise TypeError("cannot invert symbol")

Inverts the given symbol.

Parameters

sym
The symbol to invert.

Returns

The inverted symbol.

def is_clingo_operator(op: str)
Expand source code
def is_clingo_operator(op: str):
    """
    Return true if the given string is a operator as supported by the
    Evaluator.
    """
    return op in ("+", "-", "*", "\\", "/")

Return true if the given string is a operator as supported by the Evaluator.

def is_operator(op: str)
Expand source code
def is_operator(op: str):
    """
    Return true if the given string is an operator.

    Parameters
    ----------
    op
        The operator name to check.

    Returns
    -------
    Whether the string is an operator name.
    """
    return (op and op[0] in "/!<=>+-*\\?&@|:;~^.") or (op == "not")

Return true if the given string is an operator.

Parameters

op
The operator name to check.

Returns

Whether the string is an operator name.

def require_number(x: Symbol) ‑> int
Expand source code
def require_number(x: Symbol) -> int:
    """
    Requires the argument to be a number returning the given number or throwing
    a type error.
    """
    if x.type == SymbolType.Number:
        return x.number

    raise TypeError("number exepected")

Requires the argument to be a number returning the given number or throwing a type error.

Classes

class TermEvaluator
Expand source code
class TermEvaluator:
    """
    This class provides a call operator to evaluate the operators in a theory
    term in the same fashion as clingo evaluates its arithmetic functions.

    This class can easily be extended for additional binary and unary
    operators.
    """

    def evaluate_binary(self, op: str, lhs: Symbol, rhs: Symbol) -> Symbol:
        """
        Evaluate binary terms as clingo would.

        Parameters
        ----------
        op
            The operator name.
        lhs
            The left-hand-side argument.
        lhs
            The right-hand-side argument.

        Returns
        -------
        The evaluated operator in form of a symbol.
        """
        if op == "+":
            return Number(require_number(lhs) + require_number(rhs))
        if op == "-":
            return Number(require_number(lhs) - require_number(rhs))
        if op == "*":
            return Number(require_number(lhs) * require_number(rhs))
        if op == "**":
            return Number(require_number(lhs) ** require_number(rhs))
        if op == "\\":
            if rhs == Number(0):
                raise ZeroDivisionError("division by zero")
            return Number(require_number(lhs) % require_number(rhs))
        if op == "/":
            if rhs == Number(0):
                raise ZeroDivisionError("division by zero")
            return Number(require_number(lhs) // require_number(rhs))

        if is_operator(op):
            raise AttributeError("unexpected operator")

        return Function(op, [lhs, rhs])

    def evaluate_unary(self, op: str, arg: Symbol):
        """
        Evaluate unary terms as clingo would.

        Parameters
        ----------
        op
            The operator name.
        arg
            The argument of the operator.

        Returns
        -------
        The evaluated operator in form of a symbol.
        """
        if op == "+":
            return Number(require_number(arg))
        if op == "-":
            return invert_symbol(arg)
        if is_operator(op):
            raise AttributeError("unexpected operator")

        return Function(op, [arg])

    def __call__(self, term: Any):
        """
        Evaluate the given term.

        Parameters
        ----------
        term
            The term to evaluate.

        Returns
        -------
        The evaluated term in form of a symbol.
        """
        # tuples
        if term.type == TheoryTermType.Tuple:
            return Tuple_([self(x) for x in term.arguments])

        # functions and arithmetic operations
        if term.type == TheoryTermType.Function:
            arguments = [self(x) for x in term.arguments]
            # binary operations
            if len(arguments) == 2:
                return self.evaluate_binary(term.name, *arguments)

            # unary operations
            if len(arguments) == 1:
                return self.evaluate_unary(term.name, *arguments)

            # functions
            return Function(term.name, arguments)

        # constants
        if term.type == TheoryTermType.Symbol:
            if term.name.startswith('"') and term.name.endswith('"'):
                return String(_unquote(term.name[1:-1]))

            return Function(term.name)

        # numbers
        if term.type == TheoryTermType.Number:
            return Number(term.number)

        raise RuntimeError("cannot evaluate term")

This class provides a call operator to evaluate the operators in a theory term in the same fashion as clingo evaluates its arithmetic functions.

This class can easily be extended for additional binary and unary operators.

Methods

def __call__(self, term: Any)
Expand source code
def __call__(self, term: Any):
    """
    Evaluate the given term.

    Parameters
    ----------
    term
        The term to evaluate.

    Returns
    -------
    The evaluated term in form of a symbol.
    """
    # tuples
    if term.type == TheoryTermType.Tuple:
        return Tuple_([self(x) for x in term.arguments])

    # functions and arithmetic operations
    if term.type == TheoryTermType.Function:
        arguments = [self(x) for x in term.arguments]
        # binary operations
        if len(arguments) == 2:
            return self.evaluate_binary(term.name, *arguments)

        # unary operations
        if len(arguments) == 1:
            return self.evaluate_unary(term.name, *arguments)

        # functions
        return Function(term.name, arguments)

    # constants
    if term.type == TheoryTermType.Symbol:
        if term.name.startswith('"') and term.name.endswith('"'):
            return String(_unquote(term.name[1:-1]))

        return Function(term.name)

    # numbers
    if term.type == TheoryTermType.Number:
        return Number(term.number)

    raise RuntimeError("cannot evaluate term")

Evaluate the given term.

Parameters

term
The term to evaluate.

Returns

The evaluated term in form of a symbol.

def evaluate_binary(self,
op: str,
lhs: Symbol,
rhs: Symbol) ‑> Symbol
Expand source code
def evaluate_binary(self, op: str, lhs: Symbol, rhs: Symbol) -> Symbol:
    """
    Evaluate binary terms as clingo would.

    Parameters
    ----------
    op
        The operator name.
    lhs
        The left-hand-side argument.
    lhs
        The right-hand-side argument.

    Returns
    -------
    The evaluated operator in form of a symbol.
    """
    if op == "+":
        return Number(require_number(lhs) + require_number(rhs))
    if op == "-":
        return Number(require_number(lhs) - require_number(rhs))
    if op == "*":
        return Number(require_number(lhs) * require_number(rhs))
    if op == "**":
        return Number(require_number(lhs) ** require_number(rhs))
    if op == "\\":
        if rhs == Number(0):
            raise ZeroDivisionError("division by zero")
        return Number(require_number(lhs) % require_number(rhs))
    if op == "/":
        if rhs == Number(0):
            raise ZeroDivisionError("division by zero")
        return Number(require_number(lhs) // require_number(rhs))

    if is_operator(op):
        raise AttributeError("unexpected operator")

    return Function(op, [lhs, rhs])

Evaluate binary terms as clingo would.

Parameters

op
The operator name.
lhs
The left-hand-side argument.
lhs
The right-hand-side argument.

Returns

The evaluated operator in form of a symbol.

def evaluate_unary(self,
op: str,
arg: Symbol)
Expand source code
def evaluate_unary(self, op: str, arg: Symbol):
    """
    Evaluate unary terms as clingo would.

    Parameters
    ----------
    op
        The operator name.
    arg
        The argument of the operator.

    Returns
    -------
    The evaluated operator in form of a symbol.
    """
    if op == "+":
        return Number(require_number(arg))
    if op == "-":
        return invert_symbol(arg)
    if is_operator(op):
        raise AttributeError("unexpected operator")

    return Function(op, [arg])

Evaluate unary terms as clingo would.

Parameters

op
The operator name.
arg
The argument of the operator.

Returns

The evaluated operator in form of a symbol.