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 Previous(FromHistory):
"""
Return label of the previous node.
"""
position: int = -2
"""Position is set to ``-2`` to get the second to 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)