Introduction
You can automate all kinds of workflows with bots in Cerb; but sometimes you need to accomplish something that isn’t possible with the built-in functionality. Fortunately, with connected accounts and the Execute HTTP Request action, your bots can automate nearly anything by interacting with third-party services.
One of the most powerful ways to extend bot automation is with Lambda1 from Amazon Web Services (AWS).
AWS Lambda is a cloud-based service for building microservices at scale without managing any servers. You can write code in Node.js, Java, or Python, and bundle any third-party libraries. Amazon automatically takes care of deploying, scaling, and load balancing your code across multiple containers2.
Generally, you send inputs to a Lambda function with an HTTP request to their API. The function does something interesting with those inputs and then returns output.
This simple concept allows you to add countless new capabilities to bots in Cerb.
For instance, you can write a Lambda function for taking form data and filling in the editable fields of a PDF file. This can be quickly accomplished using Node.js or Python and the pdftk library. Bots would send form data to the function and receive a download URL for the generated PDF.
We’ll demonstrate Lambda integration in this guide by adding DNS3 lookup functionality to a conversational bot.
- Configure the Amazon Web Services service in Cerb
- Log in to Amazon Web Services
- Import AWS Lambda Bot in Cerb
- Test the bot
- Learn how the bot works
- References
Configure the Amazon Web Services service in Cerb
-
Log into Cerb as an administrator.
-
Navigate to Search » Connected Accounts.
-
If you don’t have a connected account for Amazon Web Services yet, you can follow these instructions to create one.
Log in to Amazon Web Services
We’ll start by logging in to the AWS Management Console.
If you don’t have an AWS account, you can sign up for free at: https://aws.amazon.com
Add a new Lambda function
Navigate to Lambda from the Services menu at the top.
Click the blue Create a Lambda function button.
Filter the runtime to Node.js 6.10 (or latest version).
Select Blank Function.
Click the blue Next button.
In Configure function:
- Name: CerbDnsLookup
- Description: Resolve IPs and hostnames
In Lambda function code, paste the following:
'use strict';
const dns = require('dns');
exports.handler = (event, context, callback) => {
if(undefined === event.mode || 0 === event.mode.length) {
callback("The 'mode' parameter is required.");
return;
}
if(undefined === event.value || 0 === event.value.length) {
callback("The 'value' parameter is required.");
return;
}
switch(event.mode) {
// Reverse
case 'reverse':
dns.reverse(event.value, function(err, hostnames) {
if(err) {
callback(err);
return;
}
callback(null, hostnames);
});
break;
// MX
case 'mx':
dns.resolveMx(event.value, function(err, exchanges) {
if(err) {
callback(err);
return;
}
callback(null, exchanges);
});
break;
// Resolve
default:
dns.resolve(event.value, function(err, addresses) {
if(err) {
callback(err);
return;
}
callback(null, addresses);
});
break;
}
};
In Lambda function handler and role:
- Role: Choose an existing role (if one exists), or create a new one. This determines what services and resources the Lambda function can access. For this example you don’t need to add anything to the defaults.
You can ignore Advanced settings for now. Later on you may need to configure a Lambda function to run inside an existing Virtual Private Cloud (VPC).
Click the blue Next button in the bottom right.
Click the blue Create function button in the bottom right.
Now we can give Cerb bots access to this function.
Update your IAM policy
Navigate to IAM from the Services menu at the top of the page.
We need to update the policy in your Amazon Web Services (AWS) account to describe the services that your Cerb bot is allowed to use. This is accomplished with the Identity Access Management (IAM) service.
We’re going to add access to invoke AWS Lambda functions prefixed with Cerb*
Select Policies in the navigation on the left.
Find your bot’s policy in the list or create a new one. In the earlier instructions we created a policy named CerbBot.
Click the Edit Policy button.
Select the JSON tab.
Add the following block to the Statement
list:
{
"Sid": "CerbLambdaInvoke",
"Effect": "Allow",
"Action": [
"lambda:InvokeAsync",
"lambda:InvokeFunction"
],
"Resource": [
"arn:aws:lambda:*:*:function:Cerb*"
]
}
Click the blue Review policy button in the bottom right.
Click the blue Save changes button in the bottom right.
Import AWS Lambda Bot in Cerb
Now we’re ready to create the bot that interacts with AWS using our connected account.
Navigate to Setup » Packages » Import.
Copy and paste the following behavior into the large text box:
{
"package": {
"name": "AWS Lambda Bot",
"cerb_version": "9.1.0",
"revision": 1,
"requires": {
"cerb_version": "9.1.0"
},
"configure": {
"prompts": [
{
"type": "chooser",
"label": "AWS Account:",
"key": "aws_account_id",
"params": {
"context": "cerberusweb.contexts.connected_account",
"query": "aws OR amazon",
"single": true
}
},
{
"type": "text",
"label": "AWS Region:",
"key": "aws_region",
"params": {
"default": "us-west-2"
}
}
],
"placeholders": [
]
}
},
"bots": [
{
"uid": "bot_33",
"name": "AWS Lambda Bot",
"owner": {
"context": "cerberusweb.contexts.app",
"id": 0
},
"is_disabled": false,
"params": {
"config": null,
"events": {
"mode": "all",
"items": []
},
"actions": {
"mode": "all",
"items": []
}
},
"image": "",
"behaviors": [
{
"uid": "behavior_180",
"title": "DNS conversational behavior",
"is_disabled": false,
"is_private": true,
"priority": 50,
"event": {
"key": "event.message.chat.worker",
"label": "Conversation with worker"
},
"nodes": [
{
"type": "action",
"title": "Hi!",
"status": "live",
"params": {
"actions": [
{
"action": "send_message",
"message": "Hi, {{worker_first_name}}!",
"format": "",
"delay_ms": "250"
},
{
"action": "send_message",
"message": "I can perform DNS lookups for you.",
"format": "",
"delay_ms": "1000"
}
]
}
},
{
"type": "loop",
"title": "Menu",
"status": "live",
"params": {
"foreach_json": "[\"*\"]",
"as_placeholder": "iterations"
},
"nodes": [
{
"type": "action",
"title": "What would you like to look up?",
"status": "live",
"params": {
"actions": [
{
"action": "send_message",
"message": "What would you like to look up?",
"format": "",
"delay_ms": "500"
},
{
"action": "prompt_buttons",
"options": "Hostname\r\nIP\r\nMX\r\nBye",
"color_from": "#4795f7",
"color_mid": "#4795f7",
"color_to": "#4795f7",
"style": ""
}
]
}
},
{
"type": "action",
"title": "Save mode",
"status": "live",
"params": {
"actions": [
{
"action": "_set_custom_var",
"value": "{{message}}",
"format": "",
"is_simulator_only": "0",
"var": "dns_mode"
}
]
}
},
{
"type": "switch",
"title": "Action:",
"status": "live",
"nodes": [
{
"type": "outcome",
"title": "Hostname",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": [
{
"condition": "message",
"oper": "is",
"value": "Hostname"
}
]
}
]
},
"nodes": [
{
"type": "action",
"title": "What hostname?",
"status": "live",
"params": {
"actions": [
{
"action": "send_message",
"message": "What hostname would you like to find an IP for?",
"format": "",
"delay_ms": "1000"
},
{
"action": "prompt_text",
"placeholder": "e.g. cerb.ai"
}
]
}
},
{
"type": "action",
"title": "Run behavior",
"status": "live",
"params": {
"actions": [
{
"action": "_run_behavior",
"on": "_trigger_va_id",
"behavior_id": "{{{uid.behavior_181}}}",
"var_mode": "resolve",
"var_value": "{{message}}",
"run_in_simulator": "0",
"var": "_behavior"
}
]
}
},
{
"type": "action",
"title": "parseResponse()",
"status": "live",
"params": {
"actions": [
{
"action": "_run_subroutine",
"subroutine": "parseResponse()"
}
]
}
}
]
},
{
"type": "outcome",
"title": "IP",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": [
{
"condition": "message",
"oper": "is",
"value": "IP"
}
]
}
]
},
"nodes": [
{
"type": "action",
"title": "What IP?",
"status": "live",
"params": {
"actions": [
{
"action": "send_message",
"message": "What IP would you like to find a hostname for?",
"format": "",
"delay_ms": "1000"
},
{
"action": "prompt_text",
"placeholder": "e.g. 8.8.8.8"
}
]
}
},
{
"type": "action",
"title": "Run behavior",
"status": "live",
"params": {
"actions": [
{
"action": "_run_behavior",
"on": "_trigger_va_id",
"behavior_id": "{{{uid.behavior_181}}}",
"var_mode": "reverse",
"var_value": "{{message}}",
"run_in_simulator": "0",
"var": "_behavior"
}
]
}
},
{
"type": "action",
"title": "parseResponse()",
"status": "live",
"params": {
"actions": [
{
"action": "_run_subroutine",
"subroutine": "parseResponse()"
}
]
}
}
]
},
{
"type": "outcome",
"title": "MX",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": [
{
"condition": "message",
"oper": "is",
"value": "MX"
}
]
}
]
},
"nodes": [
{
"type": "action",
"title": "What hostname?",
"status": "live",
"params": {
"actions": [
{
"action": "send_message",
"message": "What hostname would you like to find mail servers for?",
"format": "",
"delay_ms": "1000"
},
{
"action": "prompt_text",
"placeholder": "e.g. cerb.email"
}
]
}
},
{
"type": "action",
"title": "Run behavior",
"status": "live",
"params": {
"actions": [
{
"action": "_run_behavior",
"on": "_trigger_va_id",
"behavior_id": "{{{uid.behavior_181}}}",
"var_mode": "mx",
"var_value": "{{message}}",
"run_in_simulator": "0",
"var": "_behavior"
}
]
}
},
{
"type": "action",
"title": "parseResponse()",
"status": "live",
"params": {
"actions": [
{
"action": "_run_subroutine",
"subroutine": "parseResponse()"
}
]
}
}
]
},
{
"type": "outcome",
"title": "Bye",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": []
}
]
},
"nodes": [
{
"type": "action",
"title": "Bye!",
"status": "live",
"params": {
"actions": [
{
"action": "send_message",
"message": "Bye!",
"format": "",
"delay_ms": "1000"
},
{
"action": "window_close"
}
]
}
}
]
}
]
}
]
},
{
"type": "subroutine",
"title": "parseResponse()",
"status": "live",
"nodes": [
{
"type": "switch",
"title": "Have a response?",
"status": "live",
"nodes": [
{
"type": "outcome",
"title": "No, error",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": [
{
"condition": "_custom_script",
"tpl": "{% if _behavior.response_json is iterable and _behavior.response_json.errorMessage is not empty %}true{% endif %}",
"oper": "is",
"value": "true"
}
]
}
]
},
"nodes": [
{
"type": "action",
"title": "Error!",
"status": "live",
"params": {
"actions": [
{
"action": "send_message",
"message": "I had trouble: {{_behavior.response_json.errorMessage}}",
"format": "",
"delay_ms": "1000"
}
]
}
}
]
},
{
"type": "outcome",
"title": "Yes",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": [
{
"condition": "_custom_script",
"tpl": "{{_behavior.response_json}}",
"oper": "!is",
"value": ""
}
]
}
]
},
"nodes": [
{
"type": "switch",
"title": "Type:",
"status": "live",
"nodes": [
{
"type": "outcome",
"title": "MX",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": [
{
"condition": "_custom_script",
"tpl": "{{dns_mode}}",
"oper": "is",
"value": "MX"
}
]
}
]
},
"nodes": [
{
"type": "action",
"title": "Respond",
"status": "live",
"params": {
"actions": [
{
"action": "send_message",
"message": "{% if _behavior.response_json is iterable %}\r\n{% set exchanges = _behavior.response_json|sort %}\r\n{% for value in exchanges %}\r\n* [{{value.priority}}] {{value.exchange}}\r\n{% endfor %}\r\n{% endif %}",
"format": "markdown",
"delay_ms": "500"
}
]
}
}
]
},
{
"type": "outcome",
"title": "(Other)",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": []
}
]
},
"nodes": [
{
"type": "action",
"title": "Respond",
"status": "live",
"params": {
"actions": [
{
"action": "send_message",
"message": "{% if _behavior.response_json is iterable %}\r\n{% for value in _behavior.response_json %}\r\n* {{value}}\r\n{% endfor %}\r\n{% else %}\r\n{{_behavior.response_json}}\r\n{% endif %}",
"format": "markdown",
"delay_ms": "500"
}
]
}
}
]
}
]
}
]
},
{
"type": "outcome",
"title": "No",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": []
}
]
}
}
]
}
]
}
]
},
{
"uid": "behavior_178",
"title": "Get interactions for worker",
"is_disabled": false,
"is_private": false,
"priority": 50,
"event": {
"key": "event.interactions.get.worker",
"label": "Conversation get interactions for worker",
"params": {
"listen_points": "global\r\n"
}
},
"nodes": [
{
"type": "switch",
"title": "Point:",
"status": "live",
"nodes": [
{
"type": "outcome",
"title": "global",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": [
{
"condition": "point",
"oper": "is",
"value": "global"
}
]
}
]
},
"nodes": [
{
"type": "action",
"title": "Return interactions",
"status": "live",
"params": {
"actions": [
{
"action": "return_interaction",
"behavior_id": "{{{uid.behavior_179}}}",
"name": "DNS lookup",
"interaction": "dns.lookup",
"interaction_params_json": ""
}
]
}
}
]
}
]
}
]
},
{
"uid": "behavior_179",
"title": "Handle interaction with worker",
"is_disabled": false,
"is_private": false,
"priority": 50,
"event": {
"key": "event.interaction.chat.worker",
"label": "Conversation handle interaction with worker"
},
"nodes": [
{
"type": "switch",
"title": "Interaction:",
"status": "live",
"nodes": [
{
"type": "outcome",
"title": "dns.lookup",
"status": "live",
"params": {
"groups": [
{
"any": 0,
"conditions": [
{
"condition": "interaction",
"oper": "is",
"value": "dns.lookup"
}
]
}
]
},
"nodes": [
{
"type": "action",
"title": "Run behavior",
"status": "live",
"params": {
"actions": [
{
"action": "switch_behavior",
"return": "0",
"behavior_id": "{{{uid.behavior_180}}}",
"var": "_behavior"
}
]
}
}
]
}
]
}
]
},
{
"uid": "behavior_181",
"title": "Invoke CerbDnsLookup Lambda function",
"is_disabled": false,
"is_private": true,
"priority": 50,
"event": {
"key": "event.macro.bot",
"label": "Custom behavior on bot"
},
"variables": {
"var_mode": {
"key": "var_mode",
"label": "Mode",
"type": "D",
"is_private": "0",
"params": {
"options": "resolve\r\nreverse\r\nmx"
}
},
"var_value": {
"key": "var_value",
"label": "Value",
"type": "S",
"is_private": "0",
"params": {
"widget": "single"
}
}
},
"nodes": [
{
"type": "action",
"title": "Invoke CerbDnsLookup Lambda function",
"status": "live",
"params": {
"actions": [
{
"action": "_set_custom_var",
"value": "{% set json = {} %}\r\n{% set json = dict_set(json, 'mode', var_mode) %}\r\n{% set json = dict_set(json, 'value', var_value) %}\r\n{{json|json_encode}}",
"format": "json",
"is_simulator_only": "0",
"var": "request_json"
},
{
"action": "core.va.action.http_request",
"http_verb": "post",
"http_url": "https://lambda.{{{aws_region}}}.amazonaws.com/2015-03-31/functions/CerbDnsLookup/invocations",
"http_headers": "Date: {{'now'|date('r', 'UTC')}}\r\nX-AMZ-Client-Context: {{\"{}\"|base64_encode}}\r\nX-AMZ-Date: {{'now'|date('Ymd\\\\THis\\\\Z', 'UTC')}}",
"http_body": "{{request_json|json_encode}}",
"auth": "connected_account",
"auth_connected_account_id": "{{{aws_account_id}}}",
"options": {
"raw_response_body": "1"
},
"run_in_simulator": "1",
"response_placeholder": "_http_response"
},
{
"action": "_set_custom_var",
"value": "{{_http_response.body|json_pretty}}",
"format": "json",
"is_simulator_only": "0",
"var": "response_json"
}
]
}
}
]
}
]
}
]
}
Click the Import button.
Cerb will prompt you to link your AWS Account: before creating the behavior. Click the chooser button and select AWS (Cerb).
You will also be prompted to enter the AWS region where you created the Lambda function. You can find this at the beginning of the ARN for your function (e.g. arn:aws:lambda:us-west-2:
).
Click the Import button again.
Test the bot
Click on the floating bot interaction button in the bottom right of the browser. If you don’t see the button (e.g. this is your first bot), refresh the page.
Select AWS Lambda Bot » DNS lookup.
The bot will greet you and offer a selection of DNS tools:
Select Hostname and type cerb.ai
. You should be given a list of IPs:
You can also do reverse lookups. Select IP and type 8.8.8.8
(Google’s DNS service):
Your new bot just:
- Asked you for some input
- Made a decision based on your intention
- Authorized an API request to AWS Lambda using your connected account credentials
- Invoked the Lambda function with some input and captured the output
- Responded to you with formatted output
You can run any Lambda function by following this process. The process of adding subsequent functions is much simpler since you’ve already set up the AWS plugin, IAM policy, AWS user, and connected account in Cerb.
Any bot with access to an AWS connected account can run Lambda functions.
At this point, you can rename, extend, or disable AWS Lambda Bot according to your needs.
Learn how the bot works
Even though the bot was automatically created for you, it’s useful to understand how everything works so you can make changes and create your own behaviors later.
From Search » Bots, open the card for AWS Lambda Bot.
Click on the Behaviors button.
You’ll see four behaviors:
- Get interactions for worker lets Cerb know that this bot provides a new global interaction.
- Handle interaction with worker starts a conversational behavior when a worker selects the interaction from the global menu.
- DNS conversational behavior handles the conversation flow with the worker.
- Invoke CerbDnsLookup Lambda function wraps the AWS Lambda API call so it can be reused by any behavior.
Conversational bot functionality is covered in detail here, so we won’t dig into the first two behaviors in this guide.
Open the card for DNS conversational behavior:
This behavior is marked private, so it’s only accessible by this bot.
When a conversation starts, the bot greets the worker and offers them a menu of choices using a button prompt. This menu is inside of an infinite loop so that a worker can take multiple actions in a single interaction.
The bot reacts to worker’s button click (e.g. Hostname, IP, MX, Bye) in the Action: decision. Each action is handled by a separate outcome.
Within those outcomes, the bot asks follow up questions and then executes the Run behavior action with that input:
This is pretty simple. The bot just runs the Invoke CerbDnsLookup Lambda function private behavior and passes it the Mode: and Value: arguments. The mode is determined by the outcome (resolve, reverse, mx), and the value is the last message from the worker (a hostname or IP). We’re saving the behavior’s final state in the _behavior
placeholder.
In the interest of keeping this example simple we’re not doing a lot of error checking on the input, but the Lambda function does return useful errors to the bot that are displayed to the worker.
Now let’s look at the Invoke CerbDnsLookup behavior, which does the actual work to talk to the AWS Lambda service:
This private behavior is on the Custom behavior on bot event, which means that it only runs when it’s manually triggered by a bot (as opposed to automatically running in response to events, like new email deliveries).
It has two public behavior variables that serve as inputs: Mode and Value. These are set by the conversational behavior based on the worker’s responses.
The behavior only has a single Invoke CerbDnsLookup Lambda function action set:
In the Set custom placeholder action, we build a JSON4 payload for Lambda and save it as the request_json
placeholder. We use the JSON
format to save the placeholder as an object rather than text (i.e. we parse the JSON). This is a really simple object with mode
and value
as keys, and the behavior variables as values.
In Execute HTTP Request, we create a POST
request to the AWS Lambda API.
- The URL: includes the AWS region and function name.
- In Request headers: we set three headers.
Date:
andX-AMZ-Date:
are used during authorization to combat replay attacks5. TheX-AMZ-Client-Context:
header is optional and we’re just setting it to a blank object in Base646 format. - In Request body: we output the
request_json
placeholder in JSON format. - In Authentication: our AWS connected account is selected to securely authenticate the request.
- Under Options: we prevent the bot from attempting to automatically parse the response by
Content-Type:
. - We enable Allow execute HTTP request in simulator mode to simplify testing. You can click the Simulator button at the bottom of this popup to run the behavior directly without a conversation taking place.
- Finally, we save the HTTP response data in the
_http_response
placeholder.
In the last Set custom placeholder action we format the Lambda response and save it to the response_json
placeholder. This is what the calling behavior uses to respond to the worker.
That’s it! You can copy this behavior to implement other Lambda functions as reusable bot behaviors in Cerb.
References
-
Amazon Web Services: Lambda - https://aws.amazon.com/lambda/ ↩
-
Amazon Web Services: What are Containers? - https://aws.amazon.com/containers/ ↩
-
Wikipedia: Domain Name System (DNS) - https://en.wikipedia.org/wiki/Domain_Name_System ↩
-
Wikipedia: JavaScript Object Notation (JSON) - https://en.wikipedia.org/wiki/JSON ↩
-
Wikipedia: Replay attack - https://en.wikipedia.org/wiki/Replay_attack ↩
-
Wikipedia: Base64 - https://en.wikipedia.org/wiki/Base64 ↩