Source code for intents.service_connector

"""
Connectors allow *Intents* Agent definitions to operate with real cloud services
such as Dialogflow, Lex or Azure Bot Services. Currently, only one connector is
provided with this library, and this is for Dialogflow ES:
:mod:`intents.connectors.dialogflow_es.connector`.

.. note::

    Details about the Connector interface are only useful if you need to develop your own Service Connector (please consider raising a pull request if this is the case). If you just need to use the included Dialogflow Connector you can jump to its documentation page right away: :mod:`intents.connectors.dialogflow_es.connector`

Connectors are used to operate with the cloud version of the Agent, and
specifically to:

* Export an :class:`intents.Agent` in a format that is natively readable by the
  Service
* Predict User messages on the Cloud Agent
* Trigger Intents on the Cloud Agent

More details can be found in the :class:`Connector` interface.
"""
from uuid import uuid1
from abc import ABC, abstractmethod
from typing import Dict, Any, List
from dataclasses import dataclass

from intents import Intent, Agent
from intents.model.intent import IntentParameterMetadata
from intents.model.entity import SystemEntityMixin, _EntityMetaclass
from intents.language import IntentResponse, IntentResponseGroup

[docs]class EntityMapping(ABC): """ An Entity Mapping is a (de-)serializer for predicted Entities. Most of the times a Mapping is not needed as the Entity can be mapped directly to its type (e.g. "3" -> `Number(3)`). However, prediction services such as Dialogflow may define system entities of structured types; a notable example is Dialogflow's *sys.person* entity, which is returned as `{"name": "John"}` and therefore needs custom logic to be mapped to `Person("John")`. This is modelled in :class:`intents.connectors.dialogflow_es.entities.PersonEntityMapping`. Another notable scenario is Date/Time objects. A mapping can be used to convert time strings from the Service format to python objects. For Dialogflow ES, this is modelled in :class:`intents.connectors.dialogflow_es.entities.DateTimeEntityMapping`. """ @property @abstractmethod def entity_cls(self) -> _EntityMetaclass: """ This is the internal entity type that is being mapped. >>> mapping = StringEntityMapping(Sys.Integer, 'sys.number-integer') >>> mapping.entity_cls Sys.Integer """ @property @abstractmethod def service_name(self) -> str: """ This is the name of the Entity in the Prediction Service domain. >>> mapping = StringEntityMapping(Sys.Integer, 'sys.number-integer') >>> mapping.service_name 'sys.number-integer' """
[docs] @abstractmethod def from_service(self, service_data: Any) -> SystemEntityMixin: """ De-serialize the Service representation of an Entity (typically the value that is returned at prediction time) to an instance of one of the internal Entity classes in :class:`intents.model.entities` :param service_data: A parameter value, as it is returned by the Service in a prediction/trigger response :return: the parameter value, modelled as one of the System Entity classes """
[docs] @abstractmethod def to_service(self, entity: SystemEntityMixin) -> Any: """ Serialize a System Entity instance into a Service representation (typically, to be sent as a parameter of a trigger request) :param entity: the System Entity to serialize :return: the serialized Entity that can be sent to Service (e.g. in a trigger request) """
[docs]class StringEntityMapping(EntityMapping): """ This is a basic :class:`EntityMapping` that reads values as they are sent by the prediction service (e.g. `"3"` -> `Sys.Integer("3")`), and serializes by simple string conversion (e.g. `Sys.Integer(3)` -> "3"). The System Entity to use must be defined when instantiating the mapping, for instance: >>> StringEntityMapping(Sys.Integer, "sys.number-integer") """ entity_cls: _EntityMetaclass = None service_name: str = None def __init__(self, entity_cls: _EntityMetaclass, service_name: str): """ :param service_name: name of the Entity in the Prediction Service """ self.entity_cls = entity_cls self.service_name = service_name
[docs] def from_service(self, service_data: Any) -> SystemEntityMixin: return self.entity_cls(service_data)
[docs] def to_service(self, entity: SystemEntityMixin) -> Any: return str(entity)
[docs]class ServiceEntityMappings(dict): """ Models a list of entity mappings. In addition to a standard list, two things are added: * Dict-like access to retrieve a mapping given its internal type (e.g. `mappings[Sys.Integer]`) * Consistency check: a mapping list must cover all the entities defined in the framework (TODO) * Shortcut to define StringEntityMappings like `(Sys.Integer, 'sys.number-integer')` (TODO) """ @classmethod def from_list(cls, mapping_list: List[EntityMapping]) -> "ServiceEntityMappings": result = cls() for mapping in mapping_list: if mapping in result: raise ValueError(f"Mapping {mapping} already defined in list: {mapping_list}") result[mapping.entity_cls] = mapping return result
[docs]@dataclass class Prediction(ABC): """ This class is meant to abstract prediction results from a generic Prediction Service. It uses names from Dialogflow, as it is currently the only supported service """ intent_name: str confidence: str contexts: dict parameters_dict: dict fulfillment_messages: Dict[IntentResponseGroup, List[IntentResponse]] fulfillment_text: str = None @property @abstractmethod def entity_mappings(self) -> ServiceEntityMappings: """ A Prediction subclass must know the Entity Mappings of its Prediction Service. :meth:`parameters` will use such mappings to convert parameters in `parameters_dict` to their generic `Sys.*` type. """
[docs] def parameters(self, schema: Dict[str, IntentParameterMetadata]) -> Dict[str, Any]: """ Cast parameters in `Prediction.parameters_dict` according to the given schema. This is necessary, because parameter types are not known at prediction time, and therefore they need to be determined by the corresponding :class:`Intent` class. """ result = {} for param_name, param_value in self.parameters_dict.items(): if param_name not in schema: raise ValueError(f"Found parameter {param_name} in Service Prediction, but Intent class does not define it.") param_metadata = schema[param_name] mapping = self.entity_mappings[param_metadata.entity_cls] try: if param_metadata.is_list: if not isinstance(param_value, list): raise ValueError(f"Parameter {param_name} is defined as List, but returned value is not of 'list' type: {param_value}") result[param_name] = [mapping.from_service(x) for x in param_value] else: result[param_name] = mapping.from_service(param_value) except Exception as exc: raise RuntimeError(f"Failed to match parameter '{param_name}' with value '{param_value}' against schema {schema}. See source exception above for details.") from exc return result
[docs]class Connector(ABC): agent_cls: type(Agent) default_session: str default_language: str def __init__(self, agent_cls: type(Agent), default_session: str=None, default_language: str="en"): """ Connect the given Agent to a Prediction Service. :param agent_cls: The Agent to connect :param default_session: A default session ID (conversation channel) for predictions :param default_language: A default language for predictions """ if not default_session: default_session = f"py-{str(uuid1())}" self.agent_cls = agent_cls self.default_session = default_session self.default_language = default_language
[docs] @abstractmethod def predict(self, message: str, session: str=None, language: str=None) -> Intent: """ Predict the given User message in the given session using the given language. When `session` or `language` are None, `predict` will use the default values that are specified in :meth:`__init__`. *predict* will return an instance of `Intent`, representing the intent as it was predicted by the service. >>> from intents.connectors import DialogflowEsConnector >>> from example_agent import ExampleAgent >>> df = DialogflowEsConnector('/path/to/service-account.json', ExampleAgent) >>> df_result = df.predict("Hi, my name is Guido") user_name_give(user_name='Guido') >>> df_result.user_name "Guido" >>> df_result.fulfillment_text "Hi Guido, I'm Bot" >>> df_result.confidence 0.86 :param message: The User message to predict :param session: Any string identifying a conversation :param language: A ISO 639-1 language code (e.g. "en") :return: An instance of the predicted Intent class """
[docs] @abstractmethod def trigger(self, intent: Intent, session: str=None, language: str=None) -> Intent: """ Trigger the given Intent in the given session using the given language. When `session` or `language` are None, `predict` will use the default values that are specified in :meth:`__init__`. >>> from intents.connectors import DialogflowEsConnector >>> from example_agent import ExampleAgent, smalltalk >>> df = DialogflowEsConnector('/path/to/service-account.json', ExampleAgent) >>> df_result = df.trigger(smalltalk.agent_name_give(agent_name='Alice')) agent_name_give(agent_name='Alice') >>> df_result.fulfillment_text "Howdy Human, I'm Alice" >>> df_result.confidence 1.0 :param intent: The Intent instance to trigger :param session: Any string identifying a conversation :param language: A ISO 639-1 language code (e.g. "en") :return: An instance of the triggered Intent class """
[docs] @abstractmethod def upload(self): """ Upload the connected Agent to the Prediction Service. """
[docs] @abstractmethod def export(self, destination: str): """ Export the connected Agent in a format that can be read and imported natively by the Prediction service. For instance, the Dialogflow service will produce a ZIP export that can be imported from the Dialogflow console. Note that you can also directly upload the Agent with :meth:`upload`. :param destination: destination path of the exported Agent """