Add a Custom Chat Channel to Twilio Flex

January 28, 2020
Written by

flex-custom-chat-channel.png

In this post, I'll show you how to add a custom chat channel, such as Socket.IO, into Twilio Flex. There are several different ways to integrate a custom channel, but in this article, we will look at the one that doesn't require the development of an additional Flex UI plugin.

Prerequisites to Add a Chat Channel to Twilio Flex

Before we can start, you'll need to make sure you have a few things set up.

Custom Socket.IO chat in Flex

After that, you're ready to chat – let's look at the message flow.

Adding a Chat Channel in Flex

Before digging into how a custom chat can be integrated, we need to understand the flow a message goes through before appearing in the Flex UI. Let's look at a standard web chat message as an example. 

When a customer opens the Flex WebChat UI component, he's greeted by a message from the bot. Two things are happening at this stage: 

  • A new user is created with Friendly Name Customer
  • A new chat channel is created, and the user above is added to that channel
Chat Channel in Flex

Note that at this stage, there is only one member in the channel (i.e., the Task Router Worker was not added to the channel yet), and there are no messages in the channel.

When the user sends a message in the chat, the following happens: 

  • A new message is added to the channel
  • A new Task is created in the "Flex Task Assignment" TaskRouter workspace. This task has the following attributes: {"channelSid":"CH********","name":"Customer","channelType":"web"}. Note that the channelSid is the SID of the chat channel created above
  • A Reservation for the task is created (if any worker is available)

Once the TaskRouter Worker accepts the reservation, it is added to the chat channel. At this point, the conversation between Customer and Agent is all happening within the chat channel. 

Flex chat messages orchestration

You may be wondering how that is happening. The diagram below provides a high-level view of the flow.

Flex Chat Messages Orchestration Flow
Flex Chat Messages Orchestration Flow

The key to this flow is the Flex Flow. The main properties for a Flex Flow are:

  • channelType: Which type of channel this chat is using. There are some predefined one (web, facebook, sms, whatsapp, line) or you can use custom for your own channel. You can define multiple Flex Flows for a channelType, but only one can have the enabled property set to true
  • integrationType: Which product the flow integrates with to handle incoming messages. The allowed values are:
    • studio: A new message will trigger a Twilio Studio flow. This is particularly useful when you want to IVR / chatbot to your chat before transferring it to an Agent. 
    • task: A new message will trigger the creation of a Task in Twilio TaskRouter
    • external: A new message will trigger a call to your own Webhook. You can use that to integrate the flow with your infrastructure or other Twilio products. At the end of your flow, if you want your chat to be handled by Flex you need to create a new Task in the Flex TaskRouter Workspace
  • integration: this is an object that holds properties related to the specific inegrationType. See more on this later. 
  • chatServiceSid: The sid of the Chat Service that will be used when creating a new Channel for a new chat 

The integration property can have the following values: 

  • For Twilio Studio integrations: 
    • integration.flowSid: the Twilio Studio Flow SID (FWXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX)
  • For Twilio TaskRouter integrations:
    • integration.workspaceSid: The Twilio TaskRouter Workspace SID (WSXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX)
    • integration.workflowSid: The Twilio TaskRouter Workflow SID (WWXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX). The Workflow must belong to the Workspace defined above
    • integration.creationOnMessage: If set to true the Task is created when the first message is received. By default, the Task is created when the channel is added. 
  • For external integrations: 
    • integration.url: The URL of your Webhook

For a full list of the properties of the integration object, have a look at the documentation

From the flow chart above, you can see that when a new chat message is sent in the Flex Flow, we have the information needed to create a new channel in the Chat Service and to set the webhook for the Studio Flow. 

Implementing a Chat Channel

Enough theory for now. Let's see how to add a customer web chat to Flex.

As mentioned at the beginning of this article, this method doesn't change the webchat component in Flex. The main compromise is that we are going to re-use the default Flex Chat Service that comes pre-provisioned when you create a new Flex project. This default chat service is called "Flex Chat Service" and you can find its SID on this page in the console. 

To get a new channel integrated, you'll need to do the following: 

  • Create a new Twilio Studio Flow (optional)
  • Create a new Flex Flow associating it to the Twilio Studio Flow (either the default one or the one you created) and the "Flex Chat Service"
  • Implement a middleware that is going to receive messages from your chat channel and send them to Twilio

The flow will look like this: 

Custo Chat Flow

Create a new Studio Flow (optional) 

This step is optional - you can re-use the default flow provided by your Flex project.

  1. Navigate to your Twilio Studio Dashboard and click on the "plus" icon to add a new flow. 
  2. Give a name to the new Flow, e.g., Custom Webchat Flow
  3. In the template selection pop-up, select "Start from Scratch". Look at the other templates, in case you want your webchat to go to some additional steps before being assigned to a Flex Agent. 
  4. Once the Studio Flow editor opens, look for the "Send To Flex" Widget, and drop it on the canvas
  5. Connect the "Incoming message" trigger to "Send To Flex" widget
  6. Select the "Send to Flex Widget" and choose the following values: 
    1. Workflow: "Assign to Anyone"
    2. Channel: "Programmable Chat"
    3. Attributes: Paste the following: {"name": "{{trigger.message.ChannelAttributes.from}}", "channelType": "web", "channelSid": "{{trigger.message.ChannelSid}}"}
  7. Click on "Save"
  8. Click on "Publish" to publish the Flow

The main points to note in the steps above are decisions around the Workflow and Channel. These two settings are needed to connect the Task that gets created as the result of triggering this Studio Flow with the chat Channel used to exchange messages between the Agent and the Customer. 

Create a new Flex Flow

Flex Flows cannot be created directly in the console. Instead, we are going to use the Twilio CLI. 

Once you have installed the CLI, use the following command to create a new flow:

$ twilio api:flex:v1:flex-flows:create \
--friendly-name="Custom Webchat Flex Flow" \
--channel-type=custom \
--integration.channel=studio \
--chat-service-sid=<Flex Chat Service SID> \
--integration.flow-sid=<Flex Studio Flow SID> \
--contact-identity=custom \
--enabled

Where:

  • <Flex Chat Service SID>: This is the SID (IS****) of the chat service called "Flex Chat Service" in the Programmable Chat Dashboard
  • <Flex Studio Flow SID>: This is the SID (FW****) of the Studio Flow you created in the previous chapter or (if you skipped that chapter) the one called "Webchat Flow' in the Studio Dashboard

If the command executed with no error, write down the SID of the newly created Flow. Now head to the Flex Messaging in the console, and you should see a new "Custom Webchat Flex Flow".

Middleware

Now that the provisioning is done let's implement the middleware between Flex and our external service (here, Socket.io). We need two functions in the middleware: createNewChannel() and sendChatMessage()

createNewChannel()

This function uses the Flex API to create a new channel and configure a webhook that gets called when a message is added to the channel. The latter is needed to handle the chat messages arriving from Flex properly. To do that, this function should: 

  • Use the Flex Channel API to create a new chat channel. This is implemented using the JavaScript client.flexApi.channel.create() in lines from 2 to 9 of the code below. 
  • Once the channel is created, use the Chat Channel API to configure a new webhook for the onMessageSent event of the chat channel. This is implemented using the JavaScript client.chat.services().channels().webhooks.create() in lines from 12 to 20 below. 

This is an example of how to implement this function using JavaScript: 

 function createNewChannel(flexFlowSid, flexChatService, chatUserName) {
  return client.flexApi.channel
    .create({
      flexFlowSid: flexFlowSid,
      identity: chatUserName,
      chatUserFriendlyName: chatUserName,
      chatFriendlyName: 'Flex Custom Chat',
      target: chatUserName
    })
    .then(channel => {
      console.log(`Created new channel ${channel.sid}`);
      return client.chat
        .services(flexChatService)
        .channels(channel.sid)
        .webhooks.create({
          type: 'webhook',
          'configuration.method': 'POST',
          'configuration.url': `${webhookUrl}/new-message?channel=${channel.sid}`,
          'configuration.filters': ['onMessageSent']
        });
    })
    .then(webhook => webhook.channelSid)
    .catch(error => {console.log(error)})
}

When creating a new channel, make sure that you set the target property correctly. This represents the Target Contact Identity (e.g., the To phone number for an SMS). If a chat channel with the same target value is already present, then Flex will use the existing channel instead of creating a new one. That also means that the Studio flow is not triggered, and a new task is not created. 

The target value is also used in the Flex UI as the name of the new task. This is what a new task looks like for a channel with a target value set to "custom-chat-user"

Flex UI Task Name

sendChatMessage()

This function sends the actual message to the Flex Chat. For this function to trigger the creation of a new task, we need to make sure that the webhook gets called when a new message is added.

As explained in this section of the Flex documentation, "only actions from SDK-driven clients (like mobile phones or browsers) will cause webhooks without further action on your part". So, in this case, we need to construct the REST API manually making sure that HTTP header X-Twilio-Webhook-Enabled is set to true in the HTTP request: 


const fetch = require('node-fetch');
var base64 = require('base-64');
...

function sendChatMessage(serviceSid, channelSid, body) {
  const params = new URLSearchParams();
  params.append('Body', body);
  params.append('From', chatUserName);
  return fetch(
    `https://chat.twilio.com/v2/Services/${serviceSid}/Channels/${channelSid}/Messages`,
    {
      method: 'post',
      body: params,
      headers: {
        'X-Twilio-Webhook-Enabled': 'true',
        Authorization: `Basic ${base64.encode(`${ process.env.TWILIO_ACCOUNT_SID}:${process.env.TWILIO_AUTH_TOKEN}`)}`
      }
    }
  )
}

Testing the solution

To test the solution, we are going to build a webchat using socket.io sample chat (see here for a step by step guide on how to create this chat). To make the integration to Flex, we need to add:

  • A new POST endpoint (/new-message) that will handle the message coming from flex. This is the webhook URL we configured in the createNewChannel() function defined above:
app.post('/new-message', function(request, response) {
  console.log('New message from Flex');
  if (request.body.Source === 'SDK' ) {
    io.emit('chat message', request.body.Body);
  }
  response.sendStatus(200);
});
  • A function to send a message to Flex once a new chat message is sent from the browser. To decide if a new channel is needed (i.e., if it's a new chat), we are using the flexChannel global variable. This works well for this specific demo, but you may want to implement different logic for that in a production instance. 

...
async function sendMessageToFlex(msg) {
  if (!flexChannel) {
    flexChannel = await createNewChannel(process.env.FLEX_FLOW_SID, process.env.FLEX_CHAT_SERVICE, targetName, chatUserName);
  }
  sendChatMessage(process.env.FLEX_CHAT_SERVICE, flexChannel, msg);
}
...
io.on('connection', function(socket) {
  console.log('User connected');
  socket.on('chat message', function(msg) {
    sendMessageToFlex(msg);
    io.emit('chat message', msg);
  });
});

Ending a chat 

To ensure that the user gets notified when an agent ends a chat, you may want to add a webhook for the onChannelUpdated event. You can do so adding a new webhook creation in the createNewChannel() function: 


function createNewChannel(flexFlowSid, flexChatService, chatUserName) {
  return client.flexApi.channel
    .create({
     ...
    })
    .then(channel => {
      console.log(`Created new channel ${channel.sid}`);
      return client.chat
        .services(flexChatService)
        .channels(channel.sid)
        .webhooks.create(...)
        .then(() => client.chat
        .services(flexChatService)
        .channels(channel.sid)
        .webhooks.create({
          type: 'webhook',
          'configuration.method': 'POST',
          'configuration.url': '`${process.env.WEBHOOK_BASE_URL}/channel-update',
          'configuration.filters': ['onChannelUpdated']
        }))
    })
    .then(webhook => webhook.channelSid)
    .catch(error => {
      console.log(error);
    });
}

When this webhook is called, your middleware should check for the Attributes value in the POST request. This is a JSON object that contains a status key. When the Agent ends the chat, the status is set to "INACTIVE". You should monitor this status change to ensure that you gracefully close the chat and perform any necessary clean-up operations in your middleware and backend. 

Next Steps with Your New Flex Chat Channel

The full working code for this demo is available in this repo, along with instructions on how to set up, run, and use it.

There are still several features that you may want to explore next such as chat typing indicator, user presence, Flex UI component for your webchat. This tutorial and the code in the repository should provide a starting point for you to build and integrate your custom channel in Flex.

Giuseppe Verni is a Principal Solutions Engineer at Twilio. He's currently helping companies in EMEA design great customer engagement solutions powered by Twilio. He can be reached at gverni [at] twilio.com, or you can collaborate with him on GitHub at https://github.com/vernig.