Modifying a dialog by using the API
The REST API supports modifying your dialog programmatically. You can use the /dialog_nodes API to create, delete, or modify dialog nodes.
A dialog is a tree of interconnected nodes, and that it must conform to certain rules to be valid. Any change that you make to a dialog node might have cascading effects on other nodes, or on the structure of your dialog. Before you use the /dialog_nodes API to modify your dialog, make sure you understand how your changes affect the rest of the dialog. You can make a backup copy of the current dialog For more information, see Backing up and restoring data.
A valid dialog always satisfies the following criteria:
-
Each dialog node has a unique ID (the
dialog_nodeproperty). -
A child node is aware of its parent node (the
parentproperty). However, a parent node is not aware of its children. -
A node is aware of its immediate previous sibling, if any (the
previous_siblingproperty). All siblings that share the parent form a linked list, with each node pointing to the previous node. -
Only one child of a parent can be the first sibling (meaning that its
previous_siblingis null). -
A node cannot point to a previous sibling that is a child of a different parent.
-
Two nodes cannot point to the same previous sibling.
-
A node can specify another node that is to be run next (the
next_stepproperty). -
A node cannot be its own parent or its own sibling.
-
A node must have a type property that contains one of the following values. If no type property is specified, then the type is
standard.event_handler: A handler that is defined for a frame node or an individual slot node.
From the tool, you can define a frame node handler by clicking the Manage handlers link from a node with slots. (The tool user interface does not expose the slot-level event handler, but you can define one through the API.)
frame: A node with one or more child nodes of typeslot. Any child slot nodes that are required must be filled before the service can exit the frame node.
The frame node type is represented as a node with slots in the tool. The node that contains the slots is represented as a node of type=
frame. It is the parent node to each slot, which is represented as a child node of typeslot.response_condition: A conditional response.
In the tool, you can add one or more conditional responses to a node. Each conditional response that you define is represented in the underlying JSON as an individual node of type=
response_condition.slot: A child node of a node of typeframe.
This node type is represented in the tool as being one of multiple slots that are added to a single node. That single node is represented in the JSON as a parent node of type
frame.standard: A typical dialog node. This is the default type.
-
For nodes of type
slotthat have the same parent node, the sibling order (specified by theprevious_siblingproperty) reflects the order in which the slots are processed. -
A node of type
slotmust have a parent node of typeframe. -
A node of type
framemust have at least one child node of typeslot. -
A node of type
response_conditionmust have a parent node of typestandardorframe. -
Nodes of type
response_conditionandevent_handlercannot have children. -
A node of type
event_handlermust also have anevent_nameproperty that contains one of the following values to identify the type of node event:filled: Defines what to do if the user provides a value that meets the condition that is specified in the Check for field of a slot, and the slot is filled. A handler with this name is only present if a Found condition is defined for the slot.focus: Defines the question to show that prompts the user to provide the information needed by the slot. A handler with this name is only present if the slot is required.generic: Defines a condition to watch for that can address unrelated questions that users might ask while filling a slot or node with slots.input: Updates the message context to include a context variable with the value that is collected from the user to fill the slot. A handler with this name must be present for each slot in the frame node.nomatch: Defines what to do if the user's response to the slot prompt does not contain a valid value. A handler with this name is only present if a Not found condition is defined for the slot.
The following diagram illustrates where in the tool user interface that you define the code that is triggered for each named event.
Event handlers -
A node of type
event_handlerwith an event_name ofgenericcan have a parent of typeslotorframe. -
A node of type
event_handlerwith an event_name offocus,input,filled, ornomatchmust have a parent of typeslot. -
If more than one event_handler with the same event_name is associated with the same parent node, then the order of the siblings is the order in which the event handlers are run.
-
For
event_handlernodes with the same parent slot node, the order of execution is the same regardless of the placement of the node definitions. The events are triggered in this order by event_name:- focus
- input
- filled
- generic*
- nomatch
*If an
event_handlerwith the event_namegenericis defined for this slot or for the parent frame, then it is run between the filled and nomatch event_handler nodes.
The following examples show how various modifications might cause cascading changes.
Creating a node
Consider the following simple dialog tree:
We can create a new node by making a POST request to /dialog_nodes with the following body:
{
"dialog_node": "node_8"
}
The dialog now looks like this:
Because node_8 was created without specifying a value for parent or previous_sibling, it is now the first node in the dialog. In addition to creating node_8, the service also modified
node_1 so that its previous_sibling property points to the new node.
You can create a node somewhere else in the dialog by specifying the parent and previous sibling:
{
"dialog_node": "node_9",
"parent": "node_2",
"previous_sibling": "node_5"
}
The values that you specify for parent and previous_node must be valid:
- Both values must refer to existing nodes.
- The specified parent must be the same as the parent of the previous sibling (or
null, if the previous sibling has no parent). - The parent cannot be a node of type
response_conditionorevent_handler.
The resulting dialog looks like this:
In addition to creating node_9, the service automatically updates the previous_sibling property of node_6 so that it points to the new node.
Moving a node to a different parent
Move node_5 to a different parent by using the POST /dialog_nodes/node_5 method with the following body:
{
"parent": "node_1"
}
The specified value for parent must be valid:
- It must refer to an existing node.
- It must not refer to the node that is modified (a node cannot be its own parent).
- It must not refer to a descendant of the node that is modified.
- It must not refer to a node of type
response_conditionorevent_handler.
This results in the following changed structure:
Several things happened here:
- When node_5 moved to its new parent, node_7 went with it (because the
parentvalue for node_7 did not change). When you move a node, all descendants of that node stay with it. - Because we did not specify a
previous_siblingvalue for node_5, it is now the first sibling under node_1. - The
previous_siblingproperty of node_4 was updated tonode_5. - The
previous_siblingproperty of node_9 was updated tonullbecause it is now the first sibling under node_2.
Resequencing siblings
Now set node_5 as the second sibling instead of the first by using the POST /dialog_nodes/node_5 method with the following body:
{
"previous_sibling": "node_4"
}
When you modify previous_sibling, the new value must be valid:
- It must refer to an existing node
- It must not refer to the node that is modified (a node cannot be its own sibling)
- It must refer to a child of the same parent (all siblings must have the same parent)
The structure changes as follows:
Node_7 stays with its parent. In addition, node_4 is modified so that its previous_sibling is null because it is now the first sibling.
Deleting a node
Delete node_1 by using the DELETE /dialog_nodes/node_1 method.
The result is:
Node_1, node_4, node_5, and node_7 were all deleted. When you delete a node, all descendants of that node are deleted as well. Therefore, if you delete a root node, you are
deleting an entire branch of the dialog tree. Any other references to the deleted node (such as next_step references) are changed to null.
In addition, node_2 is updated to point to node_8 as its new previous sibling.
Renaming a node
Rename node_2 using the POST /dialog_nodes/node_2 method with the following body:
{
"dialog_node": "node_X"
}
The structure of the dialog didn't change, but multiple nodes were modified to reflect the changed name:
- The
parentproperties of node_9 and node_6 - The
previous_siblingproperty of node_3
Any other references to the deleted node (such as next_step references) are also changed.