Source code for qupulse.pulses.parameters

"""This module defines parameters and parameter declaration for the usage in pulse modelling.

Classes:
    - Parameter: A base class representing a single pulse parameter.
    - ConstantParameter: A single parameter with a constant value.
    - MappedParameter: A parameter whose value is mathematically computed from another parameter.
    - ParameterNotProvidedException.
    - ParameterValueIllegalException.
"""

from abc import abstractmethod
from typing import Optional, Union, Dict, Any, Iterable, Set, List
from numbers import Real

import sympy
import numpy

from qupulse.serialization import AnonymousSerializable
from qupulse.expressions import Expression, ExpressionVariableMissingException
from qupulse.utils.types import HashableNumpyArray, DocStringABCMeta

__all__ = ["Parameter", "ConstantParameter",
           "ParameterNotProvidedException", "ParameterConstraintViolation", "ParameterConstraint"]


[docs]class Parameter(metaclass=DocStringABCMeta): """A parameter for pulses. Parameter specifies a concrete value which is inserted instead of the parameter declaration reference in a PulseTemplate if it satisfies the minimum and maximum boundary of the corresponding ParameterDeclaration. Implementations of Parameter may provide a single constant value or obtain values by computation (e.g. from measurement results). """
[docs] @abstractmethod def get_value(self) -> Real: """Compute and return the parameter value."""
@property @abstractmethod def requires_stop(self) -> bool: """Query whether the evaluation of this Parameter instance requires an interruption in execution/sequencing, e.g., because it depends on data that is only measured in during the next execution. Returns: True, if evaluating this Parameter instance requires an interruption. """ @abstractmethod def __hash__(self) -> int: """Returns a hash value of the parameter. Must be implemented.""" def __eq__(self, other) -> bool: return type(self) is type(other) and hash(self) == hash(other)
[docs]class ConstantParameter(Parameter): """A pulse parameter with a constant value."""
[docs] def __init__(self, value: Union[Real, numpy.ndarray, Expression, str, sympy.Expr]) -> None: """Create a ConstantParameter instance. Args: value (Real): The value of the parameter """ super().__init__() try: if isinstance(value, Real): self._value = value elif isinstance(value, (str, Expression, sympy.Expr)): self._value = Expression(value).evaluate_numeric() else: self._value = numpy.array(value).view(HashableNumpyArray) except ExpressionVariableMissingException: raise RuntimeError("Expressions passed into ConstantParameter may not have free variables.")
[docs] def get_value(self) -> Union[Real, numpy.ndarray]: return self._value
def __hash__(self) -> int: return hash(self._value) @property def requires_stop(self) -> bool: return False def __repr__(self) -> str: return "<ConstantParameter {0}>".format(self._value)
class MappedParameter(Parameter): """A pulse parameter whose value is derived from other parameters via some mathematical expression. The dependencies of a MappedParameter instance are defined by the free variables appearing in the expression that defines how its value is derived. MappedParameter holds a dictionary which assign Parameter objects to these dependencies. Evaluation of the MappedParameter will raise a ParameterNotProvidedException if a Parameter object is missing for some dependency. """ def __init__(self, expression: Expression, dependencies: Optional[Dict[str, Parameter]]=None) -> None: """Create a MappedParameter instance. Args: expression (Expression): The expression defining how the the value of this MappedParameter instance is derived from its dependencies. dependencies (Dict(str -> Parameter)): Parameter objects of the dependencies. May also be defined via the dependencies public property. (Optional) """ super().__init__() self._expression = expression self.dependencies = dict() if dependencies is None else dependencies self._cached_value = (None, None) def _collect_dependencies(self) -> Dict[str, float]: # filter only real dependencies from the dependencies dictionary try: return {dependency_name: self.dependencies[dependency_name].get_value() for dependency_name in self._expression.variables} except KeyError as key_error: raise ParameterNotProvidedException(str(key_error)) from key_error def get_value(self) -> Union[Real, numpy.ndarray]: """Does not check explicitly if a parameter requires to stop.""" current_hash = hash(self) if current_hash != self._cached_value[0]: self._cached_value = (current_hash, self._expression.evaluate_numeric(**self._collect_dependencies())) return self._cached_value[1] def __hash__(self): return hash(tuple(self.dependencies.items())) @property def requires_stop(self) -> bool: """Does not explicitly check that all parameters are provided if one requires stopping""" try: return any(self.dependencies[v].requires_stop for v in self._expression.variables) except KeyError as err: raise ParameterNotProvidedException(err.args[0]) from err def __repr__(self) -> str: try: value = self.get_value() except: value = 'nothing' return "<MappedParameter {0} evaluating to {1}>".format( self._expression, value )
[docs]class ParameterConstraint(AnonymousSerializable): """A parameter constraint like 't_2 < 2.7' that can be used to set bounds to parameters.""" def __init__(self, relation: Union[str, sympy.Expr]): super().__init__() if isinstance(relation, str) and '==' in relation: # The '==' operator is interpreted by sympy as exactly, however we need a symbolical evaluation self._expression = sympy.Eq(*sympy.sympify(relation.split('=='))) else: self._expression = sympy.sympify(relation) if not isinstance(self._expression, sympy.boolalg.Boolean): raise ValueError('Constraint is not boolean') self._expression = Expression(self._expression) @property def affected_parameters(self) -> Set[str]: return set(self._expression.variables)
[docs] def is_fulfilled(self, parameter: Dict[str, Any]) -> bool: if not self.affected_parameters <= set(parameter.keys()): raise ParameterNotProvidedException((self.affected_parameters-set(parameter.keys())).pop()) return numpy.all(self._expression.evaluate_numeric(**parameter))
@property def sympified_expression(self) -> sympy.Expr: return self._expression.sympified_expression def __eq__(self, other: 'ParameterConstraint') -> bool: return self._expression.underlying_expression == other._expression.underlying_expression def __str__(self) -> str: if isinstance(self._expression.sympified_expression, sympy.Eq): return '{}=={}'.format(self._expression.sympified_expression.lhs, self._expression.sympified_expression.rhs) else: return str(self._expression.sympified_expression)
[docs] def get_serialization_data(self) -> str: return str(self)
class ParameterConstrainer: """A class that implements the testing of parameter constraints. It is used by the subclassing pulse templates.""" def __init__(self, *, parameter_constraints: Optional[Iterable[Union[str, ParameterConstraint]]]) -> None: if parameter_constraints is None: self._parameter_constraints = [] else: self._parameter_constraints = [constraint if isinstance(constraint, ParameterConstraint) else ParameterConstraint(constraint) for constraint in parameter_constraints] @property def parameter_constraints(self) -> List[ParameterConstraint]: return self._parameter_constraints def validate_parameter_constraints(self, parameters: [str, Union[Parameter, Real]]) -> None: """Raises a ParameterConstraintViolation exception if one of the constraints is violated. :param parameters: These parameters are checked. :return: """ for constraint in self._parameter_constraints: constraint_parameters = {k: v.get_value() if isinstance(v, Parameter) else v for k, v in parameters.items()} if not constraint.is_fulfilled(constraint_parameters): raise ParameterConstraintViolation(constraint, constraint_parameters) @property def constrained_parameters(self) -> Set[str]: if self._parameter_constraints: return set.union(*(c.affected_parameters for c in self._parameter_constraints)) else: return set()
[docs]class ParameterConstraintViolation(Exception): def __init__(self, constraint: ParameterConstraint, parameters: Dict[str, Real]): super().__init__("The constraint '{}' is not fulfilled.\nParameters: {}".format(constraint, parameters)) self.constraint = constraint self.parameters = parameters
[docs]class ParameterNotProvidedException(Exception): """Indicates that a required parameter value was not provided.""" def __init__(self, parameter_name: str) -> None: super().__init__() self.parameter_name = parameter_name def __str__(self) -> str: return "No value was provided for parameter '{0}'.".format(self.parameter_name)
class InvalidParameterNameException(Exception): def __init__(self, parameter_name: str): self.parameter_name = parameter_name def __str__(self) -> str: return '{} is an invalid parameter name'.format(self.parameter_name)