Skip to contentSkip to navigationSkip to topbar
Rate this page:
On this page

Integrate Mindful Platform With Flex


This implementation guide will step you through the integration between Flex and Mindful platform. With this integration, you can respect your customers' time and give them control by letting them choose when they'll receive a call. Whether it's two minutes or two hours, your customers will appreciate spending less time waiting on hold.

Mindful offers two types of callback, Customer-First or Agent-First. These callback experience options are selectable per queue. A Customer-First callback calls the customer back first and ensures they are ready for their callback. If they are not ready, Mindful handles disconnects or voicemail and will attempt to contact them again (if configured). If they are ready, the callback will connect the customer to an agent. Agent-First callback calls the agent first, and once connected, the customer is called.


Background on Callbacks

background-on-callbacks page anchor
Callbacks.

When your queues are full and your agents are swamped, new inbound calls are going to have longer hold times. This can lead to abandoned calls, poor customer experience, increased service costs (as you deal with repeat attempts by a customer), and fewer opportunities to help and engage with your customers.

One of the best ways to reduce hold time is to give callers an option to skip it altogether. With Mindful, you can offer customers the option to call them back at a more convenient time. This works great for contact centers that have peaks and valleys in their contact volume; you take the contact volume that arrives during your peak volume time and you handle it when your staff isn't as busy.

(warning)

Implementation guide disclaimer

This guide has two primary sections:

  1. Setting up the connection with Mindful
  2. Demonstrating two example call flows that incorporate callbacks.

The example call flows are a guide, and you should adapt them to fit your desired customer experience. Similarly, the sample code provided is a guide and doesn't incorporate standard techniques like error handling. These should be refined based on your needs before integrating into a live environment.


Setting up connectivity with Mindful

setting-up-connectivity-with-mindful page anchor

Here is a high-level overview of the connectivity with Mindful:

vht connectivity.

You will need to set up a SIP Domain and SIP Trunk which will be used to transfer calls to Minfdul, as well as a dedicated phone number, which will be used for the Callbacks.

Follow these steps to establish connectivity with Mindful:

1. Provision phone numbers in Twilio

1-provision-phone-numbers-in-twilio page anchor

In your Twilio console, go to Active Phone Numbers(link takes you to an external page)

This solution relies on two phone numbers, one for incoming calls from your customers and one for incoming calls from Mindful. These numbers will route to separate Studio Flows which will be configured later on in this document. Don't forget to update the routing of these phone numbers after you have completed the Studio Flows.

vht phone numbers.

2. Set up an IP Access Control List

2-set-up-an-ip-access-control-list page anchor
  1. In your Twilio console, go to IP / CIDR Access Control List(link takes you to an external page)
  2. Create a new ACL with the following addresses:

54.165.17.177/32 - VHT SIP Proxy 54.145.183.157/32 - VHT SIP Proxy 54.87.8.161/32 - VHT RTP Proxy 3.223.253.119/32 - VHT RTP Proxy

Create a new ACL.

3. Create a SIP Domain

3-create-a-sip-domain page anchor
  1. In your Twilio console, go to Programmable SIP Domains(link takes you to an external page)
  2. Create a new SIP Domain

Hint: Use the IP Access Control List defined in step 1

Create a new SIP Domain.

In this example, incoming calls to this SIP Domain route to a Studio flow, which is configured later in this document. You are welcome to use other options as well, such as a Function that returns TwiML.

4. Provision a new Call Target

4-provision-a-new-call-target page anchor
  1. In Mindful, go to Call Targets, Add New CT; or click here(link takes you to an external page)
  2. Change the Telephony Type to SIP
  3. For the Call Center Phone Number, enter the phone number which you use for the return calls from Mindful.
  4. For Callback CID, enter the Twilio phone numbers callers are calling. This will be used as the caller ID when returning the callback
Dial Settings.
  1. Go to the Metadata tab
  2. Add a metadata item called x-user-to-user with the following settings:
pasted image 0.

5. In Mindful, provision a new phone number

5-in-mindful-provision-a-new-phone-number page anchor
  1. Go to Phone Numbers(link takes you to an external page)
  2. Provision a new phone number and assign it to the Call Target configured in step 4
Assign target to the phone number.

Scenario 1: Offer a callback before the caller enters a task queue

scenario-1-offer-a-callback-before-the-caller-enters-a-task-queue page anchor

Before sending an incoming call to a TaskRouter task queue, we will call a Function that will use the TaskRouter Statistics API to determine the Estimated Wait Time. If this is above a threshold, we will offer the caller the option of a callback instead of sending the call to a queue.

(information)

Best practices for offering callbacks

  • When quoting the estimated wait time prior to offering a callback, it is a best practice to set an upper limit for the amount of time that can be quoted. For example, if the wait time exceeds 10 minutes, you might choose to say "...more than 10 minutes from now" rather than quoting an exact time.
  • We recommend setting a minimum offer threshold based on the current estimated wait time to ensure that callback offers are not made when wait times are very low. You may also wish to check agent availability prior to offering a callback.

The TaskRouter Statistics API does not provide a true Estimated Wait Time, but there are multiple ways to calculate it. In this example, we will keep this simple, and use the AvgTaskAcceptanceTime value that is available in TaskRouter for the previous five minutes.

TaskRouter Statistics API diagram.

In diagram above, the Estimated Wait Time is obtained by using Studio's Run Function widget (in the diagram called check_average_wait_time) and invoking a Function with following code:


_93
/**
_93
* Function to read avgTaskAcceptanceTime statistics from TaskRouter's cumulative statistics.
_93
* https://www.twilio.com/docs/taskrouter/api/taskqueue-statistics#taskqueue-cumulative-statistics
_93
*
_93
* It returns JSON object with following fields:
_93
* - avgTaskAcceptanceTime - number of seconds
_93
*
_93
*
_93
* Expected variables from context:
_93
* - Queue_Estimated_Wait_Time - initial value of 0, used by script to cache value of average task acceptance time
_93
* - Queue_Estimated_Wait_Time_Last_Updated - initial value of 0, used by script to cache timestamp of last update of
_93
* - Queue_Update_Interval - average time update interval in milliseconds, initial value of 60000
_93
* - Workspace_SID
_93
* - Task_Queue_SID
_93
* - Service_SID
_93
* - Environment_SID
_93
* - VAR_QEWTLU_SID - SID of variable Queue_Estimated_Wait_Time_Last_Updated
_93
* - VAR_QEWT_SID - SID of variable Queue_Estimated_Wait_Time
_93
*
_93
* Following twilio-cli calls are useful for setting up environment variables for this script:
_93
*
_93
* twilio api:taskrouter:v1:workspaces:list
_93
*
_93
* twilio api:taskrouter:v1:workspaces:task-queues:list \
_93
* --workspace-sid WSXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
_93
*
_93
* twilio api:serverless:v1:services:list
_93
*
_93
* twilio api:serverless:v1:services:environments:list \
_93
* --service-sid ZSXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
_93
*
_93
* twilio api:serverless:v1:services:environments:variables:list \
_93
* --service-sid ZSXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX \
_93
* --environment-sid ZEXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
_93
*/
_93
_93
exports.handler = function (context, event, callback) {
_93
_93
const response = new Twilio.Response();
_93
response.appendHeader('Access-Control-Allow-Origin', '*');
_93
response.appendHeader('Access-Control-Allow-Methods', 'OPTIONS POST');
_93
response.appendHeader('Content-Type', 'application/json');
_93
response.appendHeader('Access-Control-Allow-Headers', 'Content-Type');
_93
_93
get_wait_time(context, event, callback).then(value => {
_93
response.setBody({
_93
'avgTaskAcceptanceTime': value
_93
});
_93
return callback(null, response);
_93
})
_93
_93
}
_93
_93
async function get_wait_time(context, event, callback) {
_93
const client = context.getTwilioClient();
_93
_93
let current_timestamp = new Date().getTime();
_93
_93
if ((current_timestamp - context.Queue_Estimated_Wait_Time_Last_Updated) > context.Queue_Update_Interval) {
_93
_93
let average_task_acceptance_time = await get_queue_cumulative_statistics(client, context.Workspace_SID, context.Task_Queue_SID, 'avgTaskAcceptanceTime');
_93
_93
context.Queue_Estimated_Wait_Time = parseInt(average_task_acceptance_time);
_93
_93
await client.serverless.services(context.Service_SID)
_93
.environments(context.Environment_SID)
_93
.variables(context.VAR_QEWT_SID)
_93
.update({
_93
key: 'Queue_Estimated_Wait_Time',
_93
value: average_task_acceptance_time
_93
});
_93
await client.serverless.services(context.Service_SID)
_93
.environments(context.Environment_SID)
_93
.variables(context.VAR_QEWTLU_SID)
_93
.update({
_93
key: 'Queue_Estimated_Wait_Time_Last_Updated',
_93
value: current_timestamp
_93
});
_93
_93
}
_93
_93
return context.Queue_Estimated_Wait_Time;
_93
}
_93
_93
async function get_queue_cumulative_statistics(twilio_client, workspace_sid, task_queue_sid, stat_name) {
_93
return twilio_client.taskrouter.workspaces(workspace_sid)
_93
.taskQueues(task_queue_sid)
_93
.cumulativeStatistics()
_93
.fetch()
_93
.then(stats => {
_93
return (stats[stat_name]);
_93
});
_93
}

(information)

Info

If you receive the following error during execution, go to service dependencies and update the twilio library's version (e.g., to *) and redeploy.


_10
UnhandledPromiseRejectionWarning: Unhandled promise rejection: TypeError:
_10
Cannot read property 'services' of undefined at get_wait_time
_10
(/var/task/handlers/ZN016166710a27ef5a1f9efa721c2809e2.js:40:33) at
_10
processTicksAndRejections (internal/process/task_queues.js:97:5)

If the Estimated Wait is greater than 120 seconds, the call is transferred to Mindful via the SIP trunk. Otherwise, the call is sent to Flex via a TaskRouter task queue.

In the toolstep named transfer_to_vht, we define the SIP endpoint to which we transfer the call. This is the phone number that you set up above in step 5, in the following format: sip:+1xxxxxxxxxx@sip-callback.mindful.cx:5566?x-user-to-user={{trigger.call.From}}

We also send Mindful the caller's phone number through the user-to-user SIP Header


Scenario 2: Offer a callback while the customer is on hold in a task queue

scenario-2-offer-a-callback-while-the-customer-is-on-hold-in-a-task-queue page anchor

In this example, the incoming call reaches the TaskRouter task queue in the Studio Flow.

TaskRouter task queue in the Studio Flow.

The average wait time check is performed in the widget Send to Flex (send_to_flex1 in the diagram above) by using the Hold Music TwiML URL parameter. Instead of pointing to actual Hold Music TwiML, you can point to a custom Function called hold_treatment(), which in turn calls another custom function transfer_to_vht().

(information)

Info

Careful planning and consideration should be given to taking a caller out of a task queue to receive a callback. The take-rate is lower when users are offered callbacks in-queue instead of before they enter the queue.

Code for hold_treatment:


_28
/**
_28
* This function expects the same environment variables as the wait_time.js
_28
*
_28
* Expected variables from context:
_28
* - Queue_Estimated_Wait_Time - initial value of 0, used by script to cache value of average task acceptance time
_28
* - Queue_Estimated_Wait_Time_Last_Updated - initial value of 0, used by script to cache timestamp of last update of
_28
* Queue_Estimated_Wait_Time
_28
* - Queue_Update_Interval - average time update interval in milliseconds, initial value of 60000
_28
* - Workspace_SID
_28
* - Task_Queue_SID
_28
* - Service_SID
_28
* - Environment_SID
_28
* - VAR_QEWTLU_SID - SID of variable Queue_Estimated_Wait_Time_Last_Updated
_28
* - VAR_QEWT_SID - SID of variable Queue_Estimated_Wait_Time
_28
*/
_28
exports.handler = function(context, event, callback) {
_28
let twiml = new Twilio.twiml.VoiceResponse();
_28
_28
get_wait_time(context).then(avg_wait_time => {
_28
let action_url = "https://" + context.DOMAIN_NAME + '/' + 'transfer_to_vht';
_28
if (avg_wait_time > 120) {
_28
twiml.gather({action: action_url})
_28
.say(`Your call will be routed to an agent in approximately ${avg_wait_time / 60} minutes. Press 1 if you want to be called back`);
_28
}
_28
return callback(null, twiml);
_28
});
_28
_28
};

Code for transfer_to_vht:


_15
exports.handler = function(context, event, callback) {
_15
let response = new Twilio.twiml.VoiceResponse();
_15
_15
let gathered_digits = 0;
_15
if (event['Digits']) {
_15
gathered_digits = parseInt(event['Digits']);
_15
}
_15
if (gathered_digits && (gathered_digits === 1)) {
_15
// leave Flex queue and continue with next node connected to Task completed
_15
response.leave();
_15
}
_15
_15
return callback(null, response);
_15
_15
};

When the Estimated Wait Time exceeds the threshold, in this case, 120 seconds, the function returns <Leave> TwiML which returns control back to Studio and continues with the next widget.

On the Task Created condition, Studio will use the Connect Call To widget and redirect to Mindful.


The last step of this Implementation Guide is to route the return calls from Mindful straight to Flex, but with a higher priority. This should be used as a followup for either of the two scenarios above.

We can do this with the following Studio Flow:

Route the return calls to Flex.

This Flow will grab the Sip Header x-user-to-user and set that as an attribute to the TaskRouter task. It also changes the Priority from a default value of 0 to 5, to give these incoming calls a higher priority. Depending on your organization's TaskRouter setup, you may need to change this value to one that is appropriately ranked.

Once you've connected all of your Studio Flows to the appropriate phone numbers, your callback solution should be ready to go. Well done - your customers will thank you for saving them from those long hold times!


Rate this page: