Source code for chatsky.destinations.standard

"""
Standard Destinations
---------------------
This module provides basic destinations.

- :py:class:`FromHistory`, :py:class:`Current` and :py:class:`Previous` -- history-based destinations;
- :py:class:`Start` and :py:class:`Fallback` -- config-based destinations;
- :py:class:`Forward` and :py:class:`Backward` -- script-based destinations.
"""

from __future__ import annotations

from pydantic import Field

from chatsky.core.context import Context
from chatsky.core.node_label import NodeLabelInitTypes, AbsoluteNodeLabel
from chatsky.core.script_function import BaseDestination


[docs] class FromHistory(BaseDestination): """ Return label of the node located at a certain position in the label history. """ position: int = Field(le=-1) """ Position of the label in label history. Should be negative: - ``-1`` refers to the current node (same as ``ctx.last_label``); - ``-2`` -- to the previous node. """
[docs] async def call(self, ctx: Context) -> NodeLabelInitTypes: return await ctx.labels[ctx.labels.keys()[self.position]]
[docs] class Current(FromHistory): """ Return label of the current node. """ position: int = -1 """Position is set to ``-1`` to get the last node."""
[docs] class Start(BaseDestination): """ Return :py:attr:`~chatsky.core.pipeline.Pipeline.start_label`. """
[docs] async def call(self, ctx: Context) -> NodeLabelInitTypes: return ctx.pipeline.start_label
[docs] class Fallback(BaseDestination): """ Return :py:attr:`~chatsky.core.pipeline.Pipeline.fallback_label`. """
[docs] async def call(self, ctx: Context) -> NodeLabelInitTypes: return ctx.pipeline.fallback_label
[docs] def get_next_node_in_flow( node_label: AbsoluteNodeLabel, ctx: Context, *, increment: bool = True, loop: bool = False, ) -> AbsoluteNodeLabel: """ Function that returns node label of a node in the same flow after shifting the index. :param node_label: Label of the node to shift from. :param ctx: Dialog context. :param increment: If it is `True`, label index is incremented by `1`, otherwise it is decreased by `1`. :param loop: If it is `True` the iteration over the label list is going cyclically (i.e. Backward in the first node returns the last node). :return: The tuple that consists of `(flow_label, label, priority)`. If fallback is executed `(flow_fallback_label, fallback_label, priority)` are returned. """ node_label = AbsoluteNodeLabel.model_validate(node_label, context={"ctx": ctx}) node_keys = list(ctx.pipeline.script.get_flow(node_label.flow_name).nodes.keys()) node_index = node_keys.index(node_label.node_name) node_index = node_index + 1 if increment else node_index - 1 if not (loop or (0 <= node_index < len(node_keys))): raise IndexError( f"Node index {node_index!r} out of range for node_keys: {node_keys!r}. Consider using the `loop` flag." ) node_index %= len(node_keys) return AbsoluteNodeLabel(flow_name=node_label.flow_name, node_name=node_keys[node_index])
[docs] class Forward(BaseDestination): """ Return the next node relative to the current node in the current flow. """ loop: bool = False """ Whether to return the first node of the flow if the current node is the last one. Otherwise and exception is raised (and transition is considered unsuccessful). """
[docs] async def call(self, ctx: Context) -> NodeLabelInitTypes: return get_next_node_in_flow(ctx.last_label, ctx, increment=True, loop=self.loop)
[docs] class Backward(BaseDestination): """ Return the previous node relative to the current node in the current flow. """ loop: bool = False """ Whether to return the last node of the flow if the current node is the first one. Otherwise and exception is raised (and transition is considered unsuccessful). """
[docs] async def call(self, ctx: Context) -> NodeLabelInitTypes: return get_next_node_in_flow(ctx.last_label, ctx, increment=False, loop=self.loop)