Skip to content

Latest commit

 

History

History
436 lines (321 loc) · 28.3 KB

dialog-actions.md

File metadata and controls

436 lines (321 loc) · 28.3 KB
copyright lastupdated
years
2015, 2018
2018-07-03

{:shortdesc: .shortdesc} {:new_window: target="_blank"} {:tip: .tip} {:pre: .pre} {:codeblock: .codeblock} {:screen: .screen} {:javascript: .ph data-hd-programlang='javascript'} {:java: .ph data-hd-programlang='java'} {:python: .ph data-hd-programlang='python'} {:swift: .ph data-hd-programlang='swift'}

Making programmatic calls from a dialog node BETA

{: #dialog-actions}

Define actions that can make programmatic calls to external applications or services and get back a result as part of the processing that occurs within a dialog turn. {: shortdesc}

You can use an external service to validate information that you collected from the user, or perform calculations or string manipulations on the input which are too complex to be handled by using supported SpEL expressions and methods. Or you can interact with an external web service to get information, such as an air traffic service to check on a flight's expected arrival time or a weather service to get a forecast. You can even interact with an external application, such as a restaurant reservation site, to complete a simple transaction on the user's behalf.

<iframe class="embed-responsive-item" id="youtubeplayer" title="Calling IBM Cloud Functions" type="text/html" width="640" height="390" src="https://www.youtube.com/embed/y0A6X-KNoB8?rel=0" frameborder="0" webkitallowfullscreen mozallowfullscreen allowfullscreen> </iframe>

When you define the programmatic call, you choose one of the following types:

  • client: Defines a programmatic call in a standardized format that your external client application can use to perform the programmatic call or function, and return the result to the dialog. This type of call basically tells the dialog to pause here and let the client application go do something. The program that the client application runs can be anything that you choose. Just be sure to specify the call name and parameter details, as well as the error message variable name, according to the JSON formatting rules that are outlined later.

  • server: Calls an {{site.data.keyword.openwhisk_short}} action directly, and returns the result to the dialog.

    Notes:

    • Currently, you can call a {{site.data.keyword.openwhisk_short}} action from {{site.data.keyword.conversationshort}} instances that are hosted in the US South or Germany regions only.
    • The {{site.data.keyword.conversationshort}} service uses the {{site.data.keyword.openwhisk_short}} instance that is hosted in the same location only. It does not check {{site.data.keyword.openwhisk_short}} instances hosted in other regions. Therefore, do not call an action from a {{site.data.keyword.conversationshort}} service instance hosted in US South if the action is defined in a {{site.data.keyword.openwhisk_short}} instance that is hosted in Germany, for example.
    • You cannot call IBM Cloud Functions web actions from a dialog currently.

    Important: Only use this method to make a call to a {{site.data.keyword.openwhisk_short}} action that you know can return in under 5 seconds. The request to {{site.data.keyword.openwhisk_short}} times out if an individual service call takes longer than that. And if your dialog makes more than one call to an external service, the total amount of time allowed for the calls to complete is 7 seconds. If the first three calls complete in 2 seconds each, and the fourth takes more than 1 second, then the fourth call is stopped, and the error message for the call indicates that the call was not completed. For less efficient services that you need to call, manage the call through your client application and pass the information to the dialog as a separate step.

Procedure

{: #call-action}

To make a programmatic call from a dialog node, complete the following steps:

  1. In the dialog node from which you want to make the programmatic call, open the JSON editor.

    • To make a programmatic call that is executed after the response for a node is evaluated, open the JSON editor for the node response.

      Shows how to access the JSON editor associated with a standard node response.

      If the Multiple responses setting is On for the node, then you must click the Edit response Edit response icon before the Options Advanced response menu will be visible.

      Shows how to access the JSON editor associated with a standard node that has multiple conditional responses enabled for it.

    If you want to display or further process the response from the external service within the same dialog turn, then you must add a second node that does so, and jump to it from this node. {: tip}

    • To make a call that can be used by an individual slot, click the Edit slot icon Edit slot icon for the slot, and then do one of the following things:

      • To make a programmatic call that is executed after the slot condition is evaluated to true, open the JSON editor that is associated with the slot condition.

        Shows how to access the JSON editor associated with a slot condition.

      • To make a programmatic call that is executed after the slot is successfully filled, open the JSON editor that is associated with the Found response. To do so, from the Options Options icon menu for the slot, click Enable conditional responses. For the Found response, click the Edit response Edit response icon. From the Options Options icon menu for the Found response, click Open JSON editor.

        Shows how to access the JSON editor associated with the conditional response for a slot.

  2. Use the following syntax to define the programmatic call.

    {
      "context": {
        "variable_name" : "variable_value"
      },
      "actions": [
        {
          "name":"<actionName>",
          "type":"client | server",
          "parameters": {
            "<parameter_name>":"<parameter_value>",
            "<parameter_name>":"<parameter_value>"
          },
          "result_variable": "<result_variable_name>",
          "credentials": "<reference_to_credentials>"
        }
      ],
      "output": {
        "text": "response text"
      }
    }

    {: codeblock}

    The actions array specifies the programmatic calls to make from the dialog. It can define up to 5 separate programmatic calls. Specify the following name and value pairs in the JSON array:

    • <actionName>: Required. The name of the action or service to call. The name cannot be longer than 64 characters.

      • For client action types, specify a name in whatever syntax you want. The goal is to specify a name that your client application will recognize and know how to handle.

        For example: calculateRate

      • For {{site.data.keyword.openwhisk_short}} (server) action types, use this syntax to provide the fully qualified name of the action: /<namespace>/[<package-name>]/<action name>

        • If the action is part of a package, then the <package-name> information is required; otherwise it is not.

        • If you are calling a sequence of actions, then specify the <sequence name> in place of the <action name>.

        • The namespace for a user-defined action typically has the syntax: <myIBMCloudOrganizationID>_<myIBMCloudSpace>. For example: /jdoeorg_prod10/search flights

        • The actions that are provided with {{site.data.keyword.openwhisk_short}} often have the namespace: whisk.system, but you should always verify the namespace to be sure. For example: /whisk.system/weather/forecast

          See the IBM Cloud Functions naming guidelines External link icon for more details.

    • <type>: Indicates the type of call to make. Choose from the following types:

      • client: Sends a message response with programmatic call information in a standardized format that your external client application can use to perform the call or function, and get a result on behalf of the dialog. The JSON object in the response body specifies the service or function to call, any associated parameters to pass with the call, and how the result should be sent back.

      • server: Calls a {{site.data.keyword.openwhisk_short}} action (one or more) directly. You must define the action itself separately by using {{site.data.keyword.openwhisk}}. For more information, see Creating a {{site.data.keyword.openwhisk_short}} action below.

      Specifying the type is optional. The default value is client.

    • <action_parameters>: Any parameters that are expected by the external program, specified as a JSON object. Parameters are only required if the external program requires them.

    • <result_variable_name>: The name to use to reference the JSON object that is returned by the external service or program. The result is added to the context section of the /message response. In other words, the result is stored as a context variable so it can be displayed in the node response or accessed by dialog nodes that are triggered later. Any existing value for the context variable is overwritten by the value that is returned by the action. You can specify the result_variable_name by using the following syntax:

      • my_result
      • $my_result

      The name cannot be longer than 64 characters. The variable name cannot contain the following characters: parentheses (), square brackets ([]), a single quote ('), a quotation mark ("), or a back slash (\).

      If you want to save the result to the output or input section of the /message response, then you can prepend one of the following location keywords to the result_variable_name:

      • output.: Adds the result to the output section of the /message response. For example, output.my_result.
      • input.: Adds the result to the input section of the /message response. For example, input.my_result.

      You can specify a context. location keyword prefix also. For example, context.my_result. However, you do not need to because the result is added to the context by default.

      You can include periods in the variable name to create a nested JSON object. For example, you can define these variables to capture results from two separate requests to a weather service for forecasts for today and tomorrow:

      • context.weather.today
      • context.weather.tomorrow

      The results (temp and rain parameter values) are stored in the context in this structure:

      {
        "weather": {
          "today": {
            "temp": "20",
            "rain": "30"
          },
          "tomorrow": {
            "temp": "23",
            "rain": "80"
          }
        }
      }

      {: codeblock}

      If multiple actions in a single JSON action array add the result of their programmatic call to the same context variable, then the order in which the context is updated matters:

      1. If you have a combination of server and client actions in the array, the service processes the server type actions first. As a result, the value that is calculated for the context variable by the last client type action in the array overwrites the value calculated for it by any server type actions.
      2. Per action type, the order in which the actions are defined in the array determines the order in which the context variable's value is set. The context variable value returned by the last action in the array overwrites the values calculated by any other actions.
    • <reference_to_credentials>: The name of the object in which the {{site.data.keyword.openwhisk_short}} credentials are stored. Required for server actions only. These credentials are used to access the {{site.data.keyword.openwhisk_short}} instance on which the action runs. These are not your {{site.data.keyword.Bluemix_notm}} credentials.

      To discover the credentials, complete the following steps:

      1. Go to the {{site.data.keyword.openwhisk_short}} API key External link icon{: new_window} page.

        • If you have not yet created an account, do so.
        • If you are not logged in, log in.
      2. Click the Show Auth Key icon Show Auth Key to show the credentials. The segment before the colon (:) is your user ID. The segment after the colon is your password.

      Attention: Any charges that are incurred when the action runs are charged to the person who owns these credentials.

      To protect the credentials, do not store them in the {{site.data.keyword.conversationshort}} workspace. Instead, pass them from the client application as part of context. You can prevent the information from being stored in Watson logs by nesting your context variable within the $private section of the message context. For example: $private.my_credentials.

      The credentials object that you define must contain parameters named user and password.

      {
        "user":"5tj3b41j-bf3j-5d92-24g9-4a7769ab12af",
        "password":"y65gqSTSRzqE..."
      }

      {: codeblock}

      While testing the dialog, you can temporarily set the $private.my_credentials context variable with your real {{site.data.keyword.openwhisk_short}} username and password values by clicking Manage context from the "Try it out" pane in the tooling.

      Shows how the $private.my_credentials context variable is defined in the Try it out context management interface

Creating a {{site.data.keyword.openwhisk_short}} action

{: #create-action}

If you choose to define a server type programmatic call, then before you can call it from a dialog, you must create the action in {{site.data.keyword.openwhisk}}. If you are defining a client type programmatic call, then skip this procedure.

Note: Running a {{site.data.keyword.openwhisk_short}} action might incur a cost. For more information, see Pricing External link icon{: new_window}. {{site.data.keyword.openwhisk_short}} does not distinguish between calls that are made from the "Try it out" pane during testing and calls that are made from an application in production.

To create a {{site.data.keyword.openwhisk_short}} action, complete the following steps:

  1. Go to the online {{site.data.keyword.openwhisk_short}} editor External link icon{: new_window}, where you can write your code directly in your browser.

    There is also a command line interface External link icon{: new_window} you can install that enables you to define an action using code you write locally.

  2. Create a {{site.data.keyword.openwhisk_short}} action using one of the supported programming languages. See the {{site.data.keyword.openwhisk_short}} documentation External link icon{: new_window} for details.

    Keep the following tips in mind:

    • Refer to the example of a {{site.data.keyword.openwhisk_short}} action External link icon{: new_window} to see how to call an external service.
    • To make a call to a Watson service, use the Watson Developer Cloud SDK External link icon{: new_window} for the language you want to use.
    • Make sure your {{site.data.keyword.openwhisk_short}} action accepts any input parameters as a JSON object, and returns any output as a JSON object.
    • If you are using Node.js to write your {{site.data.keyword.openwhisk_short}} action, make sure you use Promise for asynchronous processing. Also make sure you return the final result from the main function.

    Alternatively, you can create a sequence of actions. {: tip}

Handling errors

If the {{site.data.keyword.openwhisk_short}} action encounters an error, the error message is returned to the dialog and is stored as a property of the response variable named cloud_functions_call_error. The error might occur if your {{site.data.keyword.openwhisk_short}} action cannot get a response from an external service, or if the Cloud Function action fails, for example. If the Cloud Function credentials are not provided or are incorrect, an error is returned. This context variable is used for server actions only; in your client application, consider creating a similar object that captures error information and returns it to the dialog as a context variable.

You can condition the dialog node response to first check for errors. For example, you can ensure that the response that references a {{site.data.keyword.openwhisk_short}} action result is shown only if no errors were encountered by adding this expression to the response condition:

  $forecast_result.cloud_functions_call_error == null

{: codeblock}

For a client type programmatic call, you can pass information about error processing by defining a context variable, such as action_error. You can pass it back to the service as part of the result variable. Then, you can display a response only if no errors were encountered by defining a response condition like this:

  $forecast_result.action_error == null

{: codeblock}

Client call example

{: #action-client-example}

The following example shows what a call to an external weather service might look like. It is added to the JSON editor that is associated with the node response. By the time the node-level response is triggered, slots have collected and stored the date and location information from the user. This example assumes that the service that will be called has an endpoint named /weather, and that it takes location and date parameters, and returns a JSON object, {"forecast": "<value>"}.

{
 "actions": [
   {
     "name": "MyWeatherFunction",
     "type": "client",
     "parameters": {
       "date": "$date",
       "location": "$location"
     },
     "result_variable": "context.my_forecast"
   }
 ]
}

{: codeblock}

Normally, the service only returns to the client from a POST /message request when new user input is required, such as after executing a parent and before executing one of its child nodes. However, if you add a client action to a node, then after evaluation, the service always returns to the client so that the result of the action call can be returned. To prevent waiting for user input when it should not, such as for a node that is configured to jump directly to a child node, the service adds the following value to the message context:

  {
    "context": {
      "skip_user_input": true
    }
  }

{: codeblock}

If you want the client to perform an action, but not get user input, then you can follow the same convention, and add the skip_user_input context variable to the parent node to communicate that to the client application.

Your client application should always check for the skip_user_input variable on context. If present, then it knows not to request new input from the user, but instead execute the action, add its result into the message, and pass it back to the service. The new POST message request should include the message returned by the previous POST message response (namely, the context, input, intents, entities, and optionally the output section) and, instead of the JSON object that defines the programmatic call to make, it should include the result that was returned from the programmatic call.

In a child node that you jump to after this node, add the response to show the user:

{
  "output": {
    "text": {
      "values": [
        "It will be $my_forecast $date.literal in $location.literal."
      ]
    }
  }

{: codeblock}

The following diagram illustrates how you can use a client call to get weather forecast information, and return it to the user.

Shows someone asking for a weather forecast, and the dialog sending the request to a client app, which then sends it to the external service

Server call example

{: #action-server-example}

The following example shows what a call to a {{site.data.keyword.openwhisk_short}} action might look like. This example shows how to use the {{site.data.keyword.openwhisk_short}} echo action that is defined in the Utilities package External link icon{: new_window} provided with the service. The action takes a text string, and returns it.

{
 "actions": [
   {
     "name": "/whisk.system/utils/echo",
     "type":"server",
     "parameters": {
       "message": "<?input.text?>"
     },
     "result_variable": "context.my_input_returned",
     "credentials":"$private.my_credentials"
   }
 ]
}

{: codeblock}

The output of the {{site.data.keyword.openwhisk_short}} action, which is stored in the context.my_input_returned variable, can now be accessed by subsequent dialog nodes.

{
 "output": {
   "text": {
     "values": [
       "Your input was: $my_input_returned."
     ]
   }
 }
}

{: codeblock}

The following diagram illustrates how to call a {{site.data.keyword.openwhisk_short}} action with a simple example that calls the built-in {{site.data.keyword.openwhisk_short}} echo service. It asks the user for input and passes it to the echo service. The echo service returns the same text back, which is displayed to the user.

Shows someone entering text, and the dialog sending the request to the {{site.data.keyword.openwhisk_short}} service, then returning the text to the dialog.

Echo action example

To see a workspace with a dialog that is already set up to call the {{site.data.keyword.openwhisk_short}} built-in Echo action, complete the following steps:

  1. Download the CloudFunctionsEcho.json External link icon{: new_window} file.

  2. Import the JSON file as a new workspace.

  3. Review the dialog to see how the call to the Echo action is specified.

  4. From the "Try it out" pane, click Manage context, and then (temporarily) set the context variables to your {{site.data.keyword.openwhisk_short}} username and password.

    $private.my_credentials
    {
      "user":"<your-CF-instance-username>",
      "password":"<your-CF-instance-password>"
    }

    {: codeblock}

  5. Test the dialog by entering some input.

    The service will use the {{site.data.keyword.openwhisk_short}} Echo action to repeat whatever you enter back to you.

Advanced server call example

{: #advanced-action-server-example}

You can call multiple actions from within a single dialog flow. In fact, you can call up to five actions within one actions JSON object in a single dialog node. However, any server type actions that are defined in an actions JSON array are all processed in parallel. Therefore, you cannot call one server type action and pass the result from it to a second server type action in the same actions block. The best way to call server actions in a specific order is to use a {{site.data.keyword.openwhisk_short}} sequence. At runtime, this approach is faster because the dialog only has to make one external call to complete multiple actions. To use a sequence, just reference the sequence name instead of an action name in the actions block definition. Alternatively, you can call the first server type action from one node and jump to a child node that calls the next server type action.

If you define one actions array that has a mix of client type and server type actions, then when the dialog node is executed, the client actions are not sent to the client until after all the server type actions have been processed. {: tip}

The following examples show what a call to a {{site.data.keyword.openwhisk_short}} action might look like.

This example shows how to use a {{site.data.keyword.openwhisk_short}} action to call an external service that takes a city name and returns the latitude and longitude coordinates of the provided location.

{
  "actions": [
        {
      "name": "/jdoeorg_prod/get coordinates",
      "type":"server",
      "parameters": {
        "location": "$location"
      },
      "result_variable": "context.my_coordinates",
      "credentials":"$private.my_credentials"
    }
  ]
}

{: codeblock}

The $my_coordinates context variable saves the two values that are returned by the get coordinates service like this:

{
 "lat":"42.3611",
 "long":"-71.0571"
}

{: codeblock}

This example shows how to use the {{site.data.keyword.openwhisk_short}} forecast action that is defined in the Weather package External link icon{: new_window} provided with the {{site.data.keyword.openwhisk_short}} service. The action expects latitude and longitude coordinates, and a time period. It returns a JSON object with forecast information for the specified location over the specified time period. The coordinates, which are returned by the earlier action, are specified as $my_coordinates.lat and $my_coordinates.long.

{
 "actions": [
   {
     "name": "/whisk.system/weather/forecast",
     "type":"server",
     "parameters": {
       "latitude": "$my_coordinates.lat",
       "longitude": "$my_coordinates.long",
       "timePeriod": "$period",
       "username": "$private.my_weather_service.username",
       "password": "$private.my_weather_service.password"
     },
     "result_variable": "context.forecasts",
     "credentials":"$private.my_credentials"
   }
 ]
}

{: codeblock}

Note: A username and a password are listed as parameters. These are only present because this particular action requires them; together they define the credentials required by the external Weather service that the provided action calls on the back end. They are different from the IBM Cloud Function account credentials. Take steps to keep these credentials private as well.

The output of the {{site.data.keyword.openwhisk_short}} action, which is stored in the context.forecasts variable, can now be accessed by subsequent dialog nodes.

{
 "output": {
   "text": {
     "values": [
       "For the next $period in $location, you can expect $forecasts."
     ]
   }
 }
}

{: codeblock}

The following diagram illustrates a complex interaction. A user asks for a weather forecast. The slots in the dialog node prompt the user for location and time period information. To call the {{site.data.keyword.openwhisk_short}} forecast service, you must provide geographic coordinates. Therefore, the dialog first gets latitude and longitude details for the location provided by the user by sending a request to the {{site.data.keyword.openwhisk_short}} service, which passes it to an external service that returns the coordinate information. In a subsequent node, the dialog sends the newly obtained coordinate information along with the time period information it got from the user to the {{site.data.keyword.openwhisk_short}} built-in forecast service to get the forecast details. The dialog then answers the user's question by showing the forecast result that was provided by the {{site.data.keyword.openwhisk_short}} forecast service.

Shows someone asking for a weather forecast, and the dialog sending the request to the Cloud Function service to pass it to the external service.