Source code for taurus.core.evaluation.evalattribute

#!/usr/bin/env python
# ###########################################################################
#
# This file is part of Taurus
#
# http://taurus-scada.org
#
# Copyright 2011 CELLS / ALBA Synchrotron, Bellaterra, Spain
#
# Taurus is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# Taurus is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with Taurus.  If not, see <http://www.gnu.org/licenses/>.
#
# ###########################################################################


import numpy
import re
import weakref

from taurus import Attribute, Manager
from taurus.core.units import Quantity
from taurus.core.taurusattribute import TaurusAttribute
from taurus.core.taurusbasetypes import (
    SubscriptionState,
    TaurusEventType,
    TaurusAttrValue,
    TaurusTimeVal,
    AttrQuality,
    DataType,
)
from taurus.core.taurusexception import TaurusException
from taurus.core import DataFormat
from taurus.core.util.log import debug, taurus4_deprecation

from taurus.core.evaluation.evalvalidator import QUOTED_TEXT_RE, PY_VAR_RE


class EvaluationAttrValue(TaurusAttrValue):
    """Reimplementation of TaurusAttrValue to provide bck-compat via a ref"""

    # TODO: remove this class once the standard widgets are adapted to TEP14

    def __init__(self, attr=None, config=None):
        # config parameter is kept for backwards compatibility only
        TaurusAttrValue.__init__(self)
        if config is not None:
            from taurus.core.util.log import deprecated

            deprecated(dep='"config" kwarg', alt='"attr"', rel="4.0")
            attr = config
        if attr is None:
            self._attrRef = None
        else:
            self._attrRef = weakref.proxy(attr)
        self.config = self._attrRef

    def __getattr__(self, name):
        # Do not try to delegate special methods
        if name.startswith("__") and name.endswith("__"):
            raise AttributeError(
                "'%s' object has no attribute %s"
                % (self.__class__.__name__, name)
            )
        try:
            ret = getattr(self._attrRef, name)
        except AttributeError:
            raise AttributeError(
                "%s has no attribute %s" % (self.__class__.__name__, name)
            )
        # return the attr but only after warning
        from taurus.core.util.log import deprecated

        deprecated(
            dep="EvaluationAttrValue.%s" % name,
            alt="EvaluationAttribute.%s" % name,
            rel="4.0",
        )
        return ret

    # --------------------------------------------------------
    # This is for backwards compat with the API of taurus < 4
    #
    @taurus4_deprecation(alt=".rvalue")
    def _get_value(self):
        """for backwards compat with taurus < 4"""
        debug(repr(self))
        try:
            return self.__fix_int(self.rvalue.magnitude)
        except AttributeError:
            return self.rvalue

    @taurus4_deprecation(alt=".rvalue")
    def _set_value(self, value):
        """for backwards compat with taurus < 4"""
        debug("Setting %r to %s" % (value, self.name))

        if self.rvalue is None:  # we do not have a previous rvalue
            import numpy

            dtype = numpy.array(value).dtype
            if numpy.issubdtype(dtype, int) or numpy.issubdtype(dtype, float):
                msg = "Refusing to set ambiguous value (deprecated .value API)"
                raise ValueError(msg)
            else:
                self.rvalue = value
        elif hasattr(self.rvalue, "units"):  # we do have it and is a Quantity
            self.rvalue = Quantity(value, units=self.rvalue.units)
        else:  # we do have a previous value and is not a quantity
            self.rvalue = value

    value = property(_get_value, _set_value)

    @taurus4_deprecation(alt=".wvalue")
    def _get_w_value(self):
        """for backwards compat with taurus < 4"""
        debug(repr(self))
        try:
            return self.__fix_int(self.wvalue.magnitude)
        except AttributeError:
            return self.wvalue

    @taurus4_deprecation(alt=".wvalue")
    def _set_w_value(self, value):
        """for backwards compat with taurus < 4"""
        debug("Setting %r to %s" % (value, self.name))

        if self.wvalue is None:  # we do not have a previous wvalue
            import numpy

            dtype = numpy.array(value).dtype
            if numpy.issubdtype(dtype, int) or numpy.issubdtype(dtype, float):
                msg = "Refusing to set ambiguous value (deprecated .value API)"
                raise ValueError(msg)
            else:
                self.wvalue = value
        elif hasattr(self.wvalue, "units"):  # we do have it and is a Quantity
            self.wvalue = Quantity(value, units=self.wvalue.units)
        else:  # we do have a previous value and is not a quantity
            self.wvalue = value

    w_value = property(_get_w_value, _set_w_value)

    @property
    @taurus4_deprecation(alt=".error")
    def has_failed(self):
        return self.error

    def __fix_int(self, value):
        """cast value to int if  it is an integer.
        Works on scalar and non-scalar values
        """
        if self.type != DataType.Integer:
            return value
        try:
            return int(value)
        except TypeError:
            import numpy

            return numpy.array(value, dtype="int")


[docs]class EvaluationAttribute(TaurusAttribute): """A :class:`TaurusAttribute` that can be used to perform mathematical operations involving other arbitrary Taurus attributes. The mathematical operation is described in the attribute name itself. An Evaluation Attribute will keep references to any other attributes being referenced and it will update its own value whenever any of the referenced attributes change. .. seealso:: :mod:`taurus.core.evaluation` .. warning:: In most cases this class should not be instantiated directly. Instead it should be done via the :meth:`EvaluationFactory.getAttribute` """ # helper class property that stores a reference to the corresponding # factory _factory = None _scheme = "eval" def __init__(self, name="", parent=None, **kwargs): self.call__init__(TaurusAttribute, name, parent, **kwargs) self._value = EvaluationAttrValue(attr=self) self._label = self.getSimpleName() self._references = [] self._validator = self.getNameValidator() self._transformation = None self.__subscription_state = SubscriptionState.Unsubscribed self._value_setter = None # This should never be None because the init already ran the validator trstring = self._validator.getExpandedExpr(str(name)) trstring, ok = self.preProcessTransformation(trstring) if ok: self._transformation = trstring self.applyTransformation() self._initWritable(trstring) def _initWritable(self, trstring): # Determine if the device supports writing to this attribute and # initialize the writing infrastructure accordingly self.writable = False try: dev = self.getParentObj() names = trstring.split(".") obj = instance = dev.getSafe()[names[0]] for n in names[1:-1]: obj = getattr(obj, n) obj = getattr(obj.__class__, names[-1]) except Exception: return ###################################################################### # This check wrongly returns false for writable properties defined with # the @x.setter decorator # TODO: Improve this self.writable = hasattr(obj, "fset") and obj.fset is not None ####################################################################### if self.writable: def value_setter(value): obj.fset(instance, value) self._value_setter = value_setter self._value.wvalue = self._value.rvalue
[docs] @staticmethod def getId(obj, idFormat=r"_V%i_"): """returns an id string for the given object which has the following two properties: - It is unique for this object during all its life - It is a string that can be used as a variable or method name :param obj: the python object whose id is requested :type obj: object :param idFormat: a format string containing a "`%i`" which, when expanded must be a valid variable name (i.e. it must match `[a-zA- Z_][a-zA-Z0-9_]*`). The default is `_V%i_` :type idFormat: str """ return idFormat % id(obj)
[docs] def preProcessTransformation(self, trstring): """ parses the transformation string and creates the necessary symbols for the evaluator. It also connects any referenced attributes so that the transformation gets re-evaluated if they change. :param trstring: a string to be pre-processed :type trstring: str :return: a tuple containing the processed string and a boolean indicating if the preprocessing was successful. if ok==True, the string is ready to be evaluated :rtype: tuple<str,bool> """ # disconnect previously referenced attributes and clean the list for ref in self._references: ref.removeListener(self) self._references = [] # get symbols evaluator = self.getParentObj() # Find references in the string, create references if needed, # connect to them and substitute the references by their id v = self._validator refs = v.getRefs(trstring, ign_quoted=True) for r in refs: symbol = self.__ref2Id(r) trstring = v.replaceUnquotedRef(trstring, "{%s}" % r, symbol) # validate the expression (look for missing symbols) safesymbols = list(evaluator.getSafe().keys()) # remove literal text strings from the validation trimmedstring = re.sub(QUOTED_TEXT_RE, "", trstring) for s in set(re.findall(PY_VAR_RE, trimmedstring)): if s not in safesymbols: self.warning('Missing symbol "%s"' % s) return trstring, False # If all went ok, enable/disable polling based on whether # there are references or not # TODO: Should we activate polling only if explicit in the expression? wantpolling = not self.isUsingEvents() haspolling = self.isPollingEnabled() if wantpolling: self._activatePolling() elif haspolling and not wantpolling: self.disablePolling() return trstring, True
def __ref2Id(self, ref): """ Returns the id of an existing taurus attribute corresponding to the match. The attribute is created if it didn't previously exist. :param ref: string corresponding to a reference. e.g. eval:1 :type ref: str """ refobj = self.__createReference(ref) return self.getId(refobj) def __createReference(self, ref): """ Receives a taurus attribute name and creates/retrieves a reference to the attribute object. If the object was not already referenced, it adds it to the reference list and adds its id and current value to the symbols dictionary of the evaluator. :param ref: :type ref: str :return: :rtype: TaurusAttribute """ refobj = Attribute(ref) if refobj not in self._references: evaluator = self.getParentObj() v = refobj.read().rvalue # add its rvalue to the evaluator symbols evaluator.addSafe({self.getId(refobj): v}) # add the object to the reference list self._references.append(refobj) return refobj
[docs] def eventReceived(self, evt_src, evt_type, evt_value): try: v = evt_value.rvalue except AttributeError: self.trace("Ignoring event from %s" % repr(evt_src)) return # update the corresponding value evaluator = self.getParentObj() evaluator.addSafe({self.getId(evt_src): v}) # re-evaluate self.applyTransformation() # notify listeners that the value changed if self.isUsingEvents(): self.fireEvent(evt_type, self._value)
[docs] def applyTransformation(self): if self._transformation is None: return try: evaluator = self.getParentObj() rvalue = evaluator.eval(self._transformation) # --------------------------------------------------------- # Workaround for https://github.com/hgrecco/pint/issues/509 # The numpy.shape method over a Quantity mutates # the type of its magnitude. # TODO: remove "if" when the bug is solved in pint if hasattr(rvalue, "magnitude"): value_dimension = len(numpy.shape(rvalue.magnitude)) else: value_dimension = len(numpy.shape(rvalue)) # --------------------------------------------------------- value_dformat = DataFormat(value_dimension) self.data_format = value_dformat self.type = self._encodeType(rvalue, value_dformat) if self.type is None: raise TypeError("Unsupported returned type, %r" % rvalue) if self.type in [ DataType.Integer, DataType.Float, ] and not isinstance(rvalue, Quantity): self.debug("Transformation converted to Quantity") rvalue = Quantity(rvalue) elif self.type == DataType.Boolean and value_dimension > 1: self.debug("Transformation converted to numpy.array") rvalue = numpy.array(rvalue) self._value.rvalue = rvalue self._value.time = TaurusTimeVal.now() self._value.quality = AttrQuality.ATTR_VALID except Exception as e: self._value.quality = AttrQuality.ATTR_INVALID msg = " the function '%s' could not be evaluated. Reason: %s" % ( self._transformation, repr(e), ) self.warning(msg)
def _encodeType(self, value, dformat): """Encode the value type into Taurus data type. In case of non-zero dimension attributes e.g. 1D, 2D the type corresponds to the type of the first element. :param value: :type value: obj :param dformat: :type dformat: taurus.DataFormat :return: :rtype: taurus.DataType """ # TODO: Should we fall back to DataType.Object instead of None? try: # handle Quantities value = value.magnitude except AttributeError: pass try: # handle numpy arrays value = value.item(0) except ValueError: # for numpy arrays of shape=() value = value.item() except AttributeError: # for bool, bytes, str, seq<str>... if dformat is DataFormat._1D: value = value[0] elif dformat is DataFormat._2D: value = value[0][0] dataType = type(value) return DataType.from_python_type(dataType) # -~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~ # Necessary to overwrite from TaurusAttribute # -~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~
[docs] def isBoolean(self): return isinstance(self._value.rvalue, bool)
[docs] def getDisplayValue(self, cache=True): return str(self.read(cache=cache).rvalue)
[docs] def encode(self, value): return value
[docs] def decode(self, attr_value): return attr_value
[docs] def write(self, value, with_read=True): if not self.isWritable(): raise TaurusException( "Attempt to write on read-only attribute %s", self.getFullName(), ) self._value_setter(value) self._value.wvalue = value if with_read: ret = self.read(cache=False) return ret
[docs] def read(self, cache=True): """returns the value of the attribute. :param cache: If True (default), the last calculated value will be returned. If False, the referenced values will be re- read and the transformation string will be re-evaluated :type cache: bool :return: attribute value """ if not cache: symbols = {} for ref in self._references: symbols[self.getId(ref)] = ref.read(cache=False).rvalue evaluator = self.getParentObj() evaluator.addSafe(symbols) self.applyTransformation() return self._value
[docs] def poll(self): v = self.read(cache=False) self.fireEvent(TaurusEventType.Periodic, v)
[docs] def isUsingEvents(self): # if this attributes depends from others, then we consider it uses # events return bool(len(self._references))
def __fireRegisterEvent(self, listener): # fire a first change event try: v = self.read() self.fireEvent(TaurusEventType.Change, v, listener) except Exception: self.fireEvent(TaurusEventType.Error, None, listener)
[docs] def addListener(self, listener): """Add a TaurusListener object in the listeners list. If it is the first listener, it triggers the subscription to the referenced attributes. If the listener is already registered nothing happens. """ initial_subscription_state = self.__subscription_state ret = TaurusAttribute.addListener(self, listener) if not ret: return ret if self.__subscription_state == SubscriptionState.Unsubscribed: for refobj in self._references: # subscribe to the referenced attributes refobj.addListener(self) self.__subscription_state = SubscriptionState.Subscribed assert len(self._listeners) >= 1 # if initial_subscription_state == SubscriptionState.Subscribed: if len(self._listeners) > 1 and ( initial_subscription_state == SubscriptionState.Subscribed or self.isPollingActive() ): Manager().enqueueJob( self.__fireRegisterEvent, job_args=((listener,),) ) return ret
[docs] def removeListener(self, listener): """Remove a TaurusListener from the listeners list. If polling enabled and it is the last element then stop the polling timer. If the listener is not registered nothing happens. """ ret = TaurusAttribute.removeListener(self, listener) if ret and not self.hasListeners(): self._deactivatePolling() self.__subscription_state = SubscriptionState.Unsubscribed return ret