Source code for dragonfly.grammar.rule_basic

#
# This file is part of Dragonfly.
# (c) Copyright 2007, 2008 by Christo Butcher
# Licensed under the LGPL.
#
#   Dragonfly 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.
#
#   Dragonfly 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 Dragonfly.  If not, see
#   <http://www.gnu.org/licenses/>.
#

"""
BasicRule class
============================================================================

The BasicRule class is designed to make it easy to create a rule from an
element tree, rather than building one indirectly via compound element
specs.

This rule class has the following parameters to customize its behavior:

 - *name* (*str*) -- the rule's name
 - *element* (*Element*) -- root element for this rule
 - *exported* -- whether the rule is exported
 - *context* -- context in which the rule will be active

Each of these parameters can be passed as a (keyword) arguments to the
constructor, or defined as a class attribute in a derived class.

.. note::

   The BasicRule class has only limited support for the "extras" and
   "defaults" functionality of the
   :class:`dragonfly.grammar.rule_compound.CompoundRule` and
   :class:`dragonfly.grammar.rule_mapping.MappingRule` classes. By default,
   the *extras* dictionary passed to :meth:`_process_recognition` will
   only contain an entry for the root element of the rule.


Example usage
............................................................................

The BasicRule class can be used to define a voice-command as follows::

   from dragonfly import (BasicRule, Repetition, Alternative, Literal, Text,
                          Grammar)

   class ExampleRule(BasicRule):
       # Define a rule element that accepts 1 to 5 (exclusive) repetitions
       # of either 'test one', 'test two' or 'test three'. These commands
       # type their respective numbers in succession using the Text action.
       element = Repetition(
           Alternative((
               Literal("test one", value=Text("1")),
               Literal("test two", value=Text("2")),
               Literal("test three", value=Text("3")),
           )),
           1, 5
       )

   # Create a grammar with the example rule and load it.
   rule = ExampleRule()
   grammar = Grammar("BasicRule Example")
   grammar.add_rule(rule)
   grammar.load()


The above :class:`BasicRule` example can be defined without sub-classing::

   rule = BasicRule(
       element=Repetition(
           Alternative((
               Literal("test one", value=Text("1")),
               Literal("test two", value=Text("2")),
               Literal("test three", value=Text("3")),
           )),
           1, 5)
   )


Class reference
............................................................................

"""

from six                            import string_types

from dragonfly.grammar.rule_base    import Rule
from dragonfly.grammar.elements     import ElementBase
from dragonfly.actions.action_base  import ActionBase


[docs]class BasicRule(Rule): """ Rule class for implementing complete or partial voice-commands defined using an element. Constructor arguments: - *name* (*str*) -- the rule's name - *element* (*Element*) -- root element for this rule - *exported* (boolean) -- whether the rule is exported - *context* (*Context*) -- context in which the rule will be active """ context = None _default_exported = True def __init__(self, name=None, element=None, exported=None, context=None): if name is None: name = self.__class__.__name__ if context is None: context = self.context # Complex handling of exported, because of clashing use of the # exported name at the class level: property & class-value. if exported is not None: pass elif (hasattr(self.__class__, "exported") and not isinstance(self.__class__.exported, property)): exported = self.__class__.exported else: exported = self._default_exported # Similar complex handling of element. if element is not None: pass elif (hasattr(self.__class__, "element") and not isinstance(self.__class__.element, property)): element = self.__class__.element # Type checking of initialization values. assert isinstance(name, string_types) assert isinstance(element, ElementBase) self._name = name Rule.__init__(self, self._name, element, exported=exported, context=context)
[docs] def value(self, node): node = node.children[0] value = node.value() if hasattr(value, "copy_bind"): # Prepare *extras* dict for passing to _copy_bind(). extras = { "_grammar": self.grammar, "_rule": self, "_node": node, } element = self._element name = element.name extra_node = node.get_child_by_name(name, shallow=True) if extra_node: extras[name] = extra_node.value() elif element.has_default(): extras[name] = element.default value = value.copy_bind(extras) return value
[docs] def process_recognition(self, node): """ Process a recognition of this rule. This method is called by the containing Grammar when this rule is recognized. This method collects information about the recognition and then calls *self._process_recognition*. - *node* -- The root node of the recognition parse tree. """ # Prepare *extras* dict for passing to _process_recognition(). extras = { "_grammar": self.grammar, "_rule": self, "_node": node, } element = self._element name = element.name extra_node = node.get_child_by_name(name, shallow=True) if extra_node: extras[name] = extra_node.value() elif element.has_default(): extras[name] = element.default # Call the method to do the actual processing. self._process_recognition(node, extras)
def _process_recognition(self, node, extras): """ Default recognition processing. This is the method which should be overridden in most cases to provide derived classes with custom recognition processing functionality. This default processing method executes actions if the node's value is an action or a list of actions. - *node* -- The root node of the recognition parse tree. - *extras* -- A dictionary of elements from the extras list contained within this recognition. Maps element name -> element value. """ value = node.value() if isinstance(value, (list, tuple)): for item in node.value(): if isinstance(item, ActionBase): item.execute(extras) elif isinstance(value, ActionBase): value.execute(extras)