Source code for dialogy.types.intent

"""
.. _Intent:

Intent Classification
======================

Automatic categorization of texts into known classes. When solving SLU problems
the objective is to answer *What was meant?* instead of *"What was said?"*. 

For an example utterance *"I lost my wallet"* a bank's support executives would 
understand that they must follow the operating procedure for blocking all transaction 
accounts. We want to train machines to have the ability to interpret these cues.

We call the classes of *meaning* as :code:`Intent`.

We can obtain intents from :ref:`XLMRMultiClass<XLMRMultiClass>` and :ref:`MLPMultiClass<MLPMultiClass>` plugin.
"""
from __future__ import annotations

from typing import Any, Dict, List, Optional, Set

import attr

from dialogy.types import BaseEntity
from dialogy.types.slots import Rule, Slot
from dialogy.utils.logger import logger


[docs]@attr.s class Intent: """ An instance of this class contains the name of the action associated with a body of text. """ # The name of the intent to be used. name: str = attr.ib(kw_only=True, order=False) """ The description of an intent separated by '_' """ # The confidence of this intent being present in the utterance. score: float = attr.ib(kw_only=True, default=0.0, order=True) """ A positive real number that represents confidence of this intent being the meaning of an utterance. Higher is better. Should be a value between 0 and 1. """ # In case of an ASR, `alternative_index` points at one of the nth # alternatives that help in predictions. alternative_index: Optional[int] = attr.ib(kw_only=True, default=None, order=False) """ Tells us the index of transcript that yeilds this intent. Since our featurizer concatenates all transcripts, we currently don't have a use for it. If we were to predict the intent for each transcript separately and vote, in those cases it would be helpful to log this property. """ # Trail of functions that modify the attributes of an instance. parsers: List[str] = attr.ib(kw_only=True, factory=list, order=False, repr=False) """ Contains a list of plugins that have created, updated this Intent. """ # Container for holding `List[BaseEntity]`. slots: Dict[str, Slot] = attr.ib( kw_only=True, factory=dict, order=False, repr=False ) """ Placeholders for :ref:`entities <BaseEntity>` that are relevant to this intent. More details are available in the doc for :ref:`slots <Slot>`. """
[docs] def apply(self, rules: Rule) -> Intent: """ Create slots using rules. .. _ApplySlot: An intent can hold different entities within associated slot-types. We parse rules that describe this mapping and create placeholders within an intent for holding entities found. An example rule would look like: .. code:: python rules = { "wake_up_alarm": { # name of the intent. "datetime_slot": [ # "datetime_slot" is the only slot associated with this intent. "date", # Only "date", "time", "datetime" entities can fill the "datetime_slot" "time", "datetime" ] }, "ticket_booking": { # name of the intent. "origin": [ # A slot that will fill only location type entities. "location" # The entity types that can be filled in the slot. ], "destination": [ # Another slot that will fill only location type entities. "location" ], "start_date": [ "date" ], "end_date": [ "date" ], "passengers": [ "people", "number" ], "budget": [ "amount-of-money" ] } } :param rules: A configuration for slot names and entity types associated with this instance of Intent. :type rules: Rule :return: The calling Intent is modified to have :ref:`Slots <slot>`. :rtype: Intent """ rule = rules.get(self.name) if not rule: return self for slot_name, entity_types in rule.items(): if isinstance(entity_types, str): entity_type = entity_types self.slots[slot_name] = Slot( name=slot_name, types=[entity_type], values=[] ) elif isinstance(entity_types, list) and all( isinstance(type_, str) for type_ in entity_types ): self.slots[slot_name] = Slot( name=slot_name, types=entity_types, values=[] ) else: raise TypeError( f"Expected entity_types={entity_types} in the rule" f" {rule} to be a List[str] but {type(entity_types)} was found." ) return self
[docs] def add_parser(self, plugin: Any) -> Intent: """ Update parsers with the plugin name This is to identify the progression in which the plugins were applied to an intent. This only helps in debugging and has no production utility. :param plugin: The class that modifies this instance. Preferably should be a plugin. :type plugin: Plugin :return: Calling instance with modifications to :code:`parsers` attribute. :rtype: Intent """ plugin_name = plugin if isinstance(plugin, str) else plugin.__class__.__name__ self.parsers.append(plugin_name) return self
[docs] def fill_slot(self, entity: BaseEntity, fill_multiple: bool = False, expected_slots: Optional[Set[str]] = None) -> Intent: """ Update :code:`slots[slot_type].values` with a single entity. .. _FillSlot: We will explore the possibility of sharing/understanding the meaning of multiple entities in the same slot type. There maybe cases where we want to fill multiple entities of the same type within a slot. In these cases :code:`fill_multiple` should be set to True. The :ref:`RuleBasedSlotFillerPlugin <rule_slot_filler>` has a detailed demo for this. :param entity: The entity to be checked for support in the calling intent's slots. :type entity: BaseEntity :param fill_multiple: :type fill_multiple: bool :return: The calling Intent with modifications to its slots. :rtype: Intent """ logger.debug("Looping through slot_names for each entity.") logger.debug(f"intent slots: {self.slots}") for slot_name, slot in self.slots.items(): if expected_slots and slot_name not in expected_slots: continue logger.debug(f"slot_name: {slot_name}") logger.debug( f"slot type: {slot.types}", ) logger.debug( f"entity type: {entity.entity_type}", ) if entity.entity_type in slot.types: if fill_multiple: logger.debug(f"filling {entity} into {self.name}.") self.slots[slot_name].add(entity) elif not self.slots[slot_name].values: logger.debug(f"filling {entity} into {self.name}.") self.slots[slot_name].add(entity) else: logger.debug( f"removing {entity} from {self.name}, because the slot was filled previously. " "Use fill_multiple=True if this is not required.", entity, self.name, ) self.slots[slot_name].clear() return self
[docs] def cleanup(self) -> None: """ Remove slots that were not filled. .. _CleanupSlot: """ slot_names = list(self.slots.keys()) for slot_name in slot_names: if not self.slots[slot_name].values: del self.slots[slot_name]
[docs] def json(self) -> Dict[str, Any]: """ Convert the object to a dictionary. .. ipython:: python from dialogy.types.intent import Intent intent = Intent(name="special", score=0.8) intent.json() """ return { "name": self.name, "alternative_index": self.alternative_index, "score": self.score, "parsers": self.parsers, "slots": [slot.json() for slot in self.slots.values()], }