Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Chatbots or conversational Bots, are the most common bots in the Symphony Messaging ecosystem. These types of bots run 24/7 and wait for users to initiate conversations in order to perform certain tasks. The conversations could be simple commands or even natural language queries. The bot will determine the user’s intent based on their messages and take the appropriate action.
As stated before, users can interact with chatbots in chatrooms or 1 to 1 chats. In order to initiate your chatbot's workflow we recommend that users @mention the bot's username in order to get the bot's attention and signal to the bot to begin its intended function. That way, chatbots can be active members of chatrooms, but eliminate noise that is outside the scope of its intended workflow.
It is common for Bots to contain multiple commands or sub workflows that it can action upon. It is best practice to list these commands in a help menu. Additionally, it is best practice for all commands to follow a "/" so that it's clear what text is meant to be processed as a command. The following illustrates these best practices:
A core aspect of all chatbots and conversational bots is the ability to 'listen' to different types of commands and events and reply to them accordingly. The mechanism that enables Bots to listen to events in chats is the Datafeed.
The Datafeed is a real-time message and event stream that can be created using the REST API. The datafeed provides a secure stream of messages and events from all conversations that a bot is in.
In order to 'listen' and respond to user events, bots create a single datafeed and will subsequently have access to all chat activity including non-chat based events such as users being added/removed from chatrooms, external connection requests, as well as chats being created.
To learn more about the Datafeed continue here:
The next step of your chatbot workflow is to introduce your custom business logic. Once you have access to the bot's events and messages through the Datafeed, the next step is to create dedicated event listeners. Inside these listeners is where you parse messages, fetch data from external sources, manage conversation state, and make requests to our Symphony Messaging REST API in order to reply to users directly or perform administrative functions such as creating chatrooms.
You can learn more about parsing events and introducing custom business logic here.
Continue on to our Building a Conversational Bot guide in order to learn more about our development tools and appropriate development pathway for building your first Symphony Messaging Chatbot.
Headless Bots are the simplest form of Bots. Headless Bots usually take the form of a script that is either run on a scheduled basis or triggered by an event from an external system. These scripts are usually transient and terminate after performing their assigned task, like sending an alert or daily digest message.
Often times, Headless Bots run on a scheduled basis. Since Headless Bots do not have a conversational aspect, they do not need to create/maintain a Datafeed. As a result, Headless Bots do not need to always be running.
In practice, each time a Headless Bot kicks off its scheduled workflow, it will:
Authenticate
Post a message/alert in a designated chat or call any number of API endpoints
Terminate its process
Headless Bots can also listen for webhooks or incoming events from external/third-party systems. By listening and handling different events from multiple systems, Headless Bots are able to transform Symphony Messaging into a centralized notification center for all your data.
In practice, a Headless Bot needs to expose an endpoint in order to listen and handle these webhooks or incoming events. Each time the bot receives a webhook or an incoming request, it must also:
Authenticate
Post a message/alert in a designated chat or call any number of API endpoints
Terminate its process
Continue on to our Building a Headless Bot guide. Here you will learn how to get your own Headless Bot up and running and take a closer look at the APIs used to create a simple Headless Bot workflow:
A bot on Symphony Messaging can be thought of as an automated version of a human performing specific tasks in Symphony Messaging chats. Most actions that an end user is able to perform in Symphony Messaging can be performed by a bot as well.
Each bot has a unique identity represented by a service account that has similar features to an end-user account such as a name and avatar.
Symphony Messaging Bots enable end users to benefit from innovative workflows and time-saving automations built on top of the Symphony Messaging platform.
The answer lies in Symphony Messaging's open REST API. Once authenticated, Symphony Messaging bots can leverage the APIs that enables bots to execute administrative functions such as creating chatrooms, managing users, and facilitating cross-pod connections. In addition, our APIs allow bots to perform messaging functions such as sending and receiving messages and signals.
For a full overview of the Symphony Messaging REST APIs continue here:
Before you begin your Bot development journey, it is important to consider the following when determining what type of Bot you build:
Before building your Bot, it's important that you identify the use cases that this Bot will serve. In other words, identify the ways in which this Bot will increase productivity, add meaningful color to your daily tasks, centralize information, reduce business pain points, and make working simpler for its users. To easily identify valuable use cases, ask yourself the following:
Are there any tasks in my daily workflow that are recurring and can be automated to assist me in my job?
Are there numerous sources of information that I check daily that can be centralized inside of Symphony Messaging?
Are there any tasks in my daily workflow that require manually sifting through large amounts of data?
Is there tedious data validation or compliance checks that I must perform when dealing with colleagues, clients, customers, or third-party vendors?
Do I have to manually collect and carefully collate unstructured data from colleagues, clients, customers, or third-party vendors?
If you need additional inspiration, checkout our Symphony Messaging App Directory for examples of what has been built today: https://symphony.com/resource/app-directory/
The type of Bot you build will depend on who is using and interacting with it. To identify your bot's audience, ask yourself the following:
Are the users of my bot internal or external counter-parties?
Will the bot be interacting with front-office or back-office employees mostly?
Is your bot interacting with a technical audience or business audience?
Will users interact with your bot via the Symphony Messaging Mobile App or on the desktop?
What languages does your audience speak?
Will your bot be performing bot-to-bot communication?
The more you understand your audience, the more you can understand their business pain points and in turn develop a better user-experience and bot-based solution.
Users can interact with Bots in IMs, group chats, and in chatrooms. Before building your Bot, it's important to identify the types of interactions between users and your Bot:
If so, you are looking to build a chatbot, which is a type of Bot that allows direct user interaction in the form of request/reply. You can learn more about rich chat-based workflows and building chatbots here:
If so, you are looking to build an interactive bot. Interactive bots leverage Elements to collect user data and feedback through forms, textfields, buttons, etc. You can learn more about interactive workflows and Elements here:
If so, you are looking to build a headless bot. Headless bots can leverage webhooks or build custom notification handlers and formatters from external systems. You can learn more about headless bot workflows and notification handlers here:
Lastly, it's important to clearly define the lifecycle and scope of your bot's workflow:
What chats will my bot exist in?
Will these rooms be public, private, broadcast, or external (cross-pod)?
How can I define the scope of my bot in order to reduce unnecessary noise not relevant to my bot's workflow?
How can I clearly define the protocol for initiating my bot's workflow?
Continue here for a full list of Bot's best practices to establish Bot parameters, scope, and protocol:
Building Bots on Symphony Messaging is fast, easy, and secure. Take these simple steps in order to create and deploy your Bot today!
Symphony Messaging Bots can leverage the Symphony Messaging REST API to create innovative workflows and timesaving automations. Depending on your desired workflow, there are many different development avenues available for you and your development team. Understanding these different development options and different APIs available is key to creating a successful bot and positive user experience. Learn more about the different types of Symphony Messaging Bots and APIs here:
Symphony has built numerous tools in order to streamline Symphony Messaging Bot Development. Our rich suite of developer tools offers centralized configuration and authentication protocols, provides out of the box API bindings, convenient error handling, datafeed management, and ensures that your Bot is built using Symphony's best practices.
Using an ultra simplified configuration and authentication set up, an intuitive message and room control mechanism, new APIs for message templating and workflow activities, developing bots has never been easier.
Learn more about how you can get started with our developer tools here:
At Symphony, security is always first. In order to leverage Symphony Messaging's robust set of open APIs, your Bot must first authenticate itself. You can learn more about how to authenticate and the different types of authentication here:
The last step is to add custom business logic to your Bot. Begin learning about the capabilities of Symphony Messaging REST APIs and bring your workflows and automations to life:
Want to take your Bot building to the next level? Symphony offers a free course and certification program for Developers wishing to improve their Symphony Messaging skills, technical knowledge, and expertise. Learn more about how you can become a Certified Developer today!
Interactive Bots are a form of Chatbot where the conversational flow is driven by interactive Elements forms. Instead of listening for plain text as the only source of data, Interactive Bots can collect data or commands through structured forms.
Elements allow bots to send messages that contain interactive forms with pre-designed text fields, dropdown menus, person selectors, buttons and more. Additionally, Elements reuse Symphony Messaging UX component libraries, enabling developers to easily create interactive bot messages that look and feel like they belong in Symphony Messaging.
To learn more about Elements navigate here:
Users can interact with chatbots in chatrooms and 1 to 1 chats. In order to initiate your Interactive Bot's workflow we recommend that users @mention the bot's username in order to get the bot's attention and signal to the bot to begin its intended function. That way, bots can be active members of chatrooms, but eliminate noise that is outside the scope of its intended workflow.
It is common for bots to contain multiple commands or sub workflows that it can action upon. It is best practice to list these commands in a help menu. Additionally, it is best practice for all commands to follow a "/" so that it's clear what text is meant to be processed as a command. The following illustrates these best practices:
A core aspect of all Interactive Bots is the ability to 'listen' to different types of commands and events and reply to them accordingly. The mechanism that enables Symphony Messaging Bots to listen to events in chatrooms and IMs is the Datafeed.
The Datafeed is a real-time message and event stream that can be created using the REST API. The datafeed provides a secure stream of messages and events from all conversations that a bot is in.
In order to 'listen' and respond to user events, bots create a single datafeed and subsequently have access to all chatroom activity including non-chat based events such as users being added/removed from chatrooms, external connection requests, and chatrooms IMs being created.
To learn more about the Datafeed continue here:
When an end-user submits an Elements form, the bot is able to access the contents of that form through the Datafeed. The Elements form lifecycle is illustrated below:
First an Interactive Bot sends a messageML message containing an Elements form:
After the user fills out their information and clicks the 'Submit' button, the following payload is captured by the Datafeed and delivered to the bot:
The next step of your Interactive Bot workflow is to introduce your bot's custom business logic. Now that you have access to the bot's events, messages, and elements payloads through the Datafeed, the next step is to create dedicated event listeners. Inside these listeners is where you parse messages and datafeed payloads, fetch data from external sources, manage conversation state, and make requests to Symphony Messaging REST APIs in order to reply to users directly or performs administrative functions such as creating chatrooms.
The messageML is sent by your Bot and rendered as a form:
You can learn more about parsing events and introducing custom business logic .
Continue on to our guide in order to learn more about our development tools and appropriate development pathway for building your first interactive Bot:
The Symphony Messaging Generator creates a basic configuration file that assumes a fully cloud-hosted Symphony Messaging pod architecture. In this scenario, the pod, key manager and agent are all hosted on the same domain e.g. develop2.symphony.com
. If your pod architecture is different and you have other connectivity requirements like a network proxy, you will need to add those options to your configuration file.
host
Global hostname
port
Global port
scheme
https
or http
context
Context path (e.g. /abc)
pod
agent
keyManager
Contains component details including host
, port
, scheme
, context
and proxy
attributes
bot
contains bot metadata including username
, privateKeyPath
, certificatePath
andcertificatePassword
app
contains extension app metadata including appId
, privateKeyPath
, certificatePath
, and certificatePassword
ssl
contains trustStore
and trustStore
password for SSL communication
Property
Description
version
version of the datafeed service to be used. By default, the bot will use the datafeed v2 service.
idFilePath
the path to the file which will be used to persist a created datafeed id in case the datafeed service v1 is used
retry
the specific retry configuration can be used to override the global retry configuration. If no retry configuration is defined, the global one will be used.
Property
Description
maxAttempts
maximum number of retry attempts that a bot is able to make
multiplier
after each attempt, the interval between two attempts will be multiplied by this factor
initialIntervalMillis
the initial interval between two attempts
maxIntervalMillis
the limit of interval between two attempts. For example, if the current interval is 1000 ms, multiplier is 2.0 and the maxIntervalMillis
is 1500 ms, then the interval for the next retry will be 1500 ms.
An example customized configuration file is seen below:
The Symphony Messaging REST API acts as a secure interface between your Symphony Messaging Bot and Symphony Messaging's components. Symphony's architecture for bots comprises three components: the Symphony Messaging Pod, the API Agent and the Key Manager.
The Symphony Messaging REST API is spread out across these components according to the type of API being called.
The API Swagger files are available here.
There is one sub-folder per API collection, and each API collection is split in two swagger files: one for the supported endpoints and a second one listing only the deprecated endpoints (e.g. /pod/pod-api-public-deprecated.yaml
).
Depending on your pod or agent version, a newly introduced endpoint may not yet be available to you. You can spot these new endpoints as they are tagged with the x-since
attribute. For example, an endpoint flagged as x-since: 20.15
will not be present if you are still on SBE 20.14.
To learn more about how bots securely interact with Symphony Messaging's three components, continue onto the REST API Architecture guide below:
To learn more about how bots can make authentication and administrative calls such as manage streams, manage users or facilitate cross-pod connections, continue onto the Pod API guide below:
To learn more about how bots can authenticate and encrypt messages on Symphony Messaging, continue onto the Key Manager API guide below:
To learn more about how bots can send and receive encrypted messages on Symphony Messaging, continue onto the Agent API guide below:
For the full Symphony Messaging API reference continue here:
Guide for creating a bot user in the admin portal
Please note the below steps can only be performed by a Symphony Messaging Pod Administrator as only they will have the necessary administrator privileges to access the Admin Portal. Please do not attempt this if you are not a Pod Administrator and reach out to your internal IT Helpdesk if you are unsure who your Symphony Messaging Pod Administrator is.
In order to create a bot and begin your development journey, you must first create a bot user, also known as a service account. You can only do this yourself if you are a pod administrator on your Symphony Messaging pod. Otherwise, please seek assistance from your pod administrator or your internal IT Service Desk if you are unsure who to contact.
For example,
The bot username here has to match the username supplied by a bot configuration file exactly.
Paste the entire contents of the public key in the Authentication section. This public key has to match the private key supplied by a bot. If you are unsure how to generate an RSA key pair, use the Symphony Messaging Generator.
Enable the required roles that your bot requires.
Enable the required entitlements that your bot requires.
The Symphony Messaging Pod API is used to build tools in order to manage and administer Symphony Messaging for your organization. The following guide includes API collections that exist on the Pod:
If successful, the Pod API returns a Session Token which is valid for up to two weeks. This Session Token must be passed along with every subsequent Pod API request. You can read more about Authentication and Token management here:
The User APIs query and manage users on the Pod. These APIs can be used to do the following:
Search users
List all users
Create users
Update user information
List user features
Add/Remove user roles
List user roles
List user audit trail
The Stream APIs create and manage IMs and chat rooms. These APIs can be used to do the following:
Create IM
Create chatrooms
Search for rooms
Get room info
Deactivate room
List room members
Add/Remove room members
Promote/Demote user to room owner
The Connection APIs manage user connections. These APIs can be used to do the following:
Get connection
List connections
Create connections
Accept connection
Reject connection
Remove connection
The Presence APIs manage presence status for users. These APIs can be used to do the following:
Get/Set user presence
Get all users presence
Register presence interest
Create presence feed
Read presence feed
Delete presence feed
The Symphony Messaging Agent is responsible for encryption and decryption of messages and content sent to and from a bot. As a result, the Agent API is used to build applications that send and receive messages and content. The following guide includes API collections that exist on the Agent:
The Message APIs create, read and search messages on the Pod. These APIs can be used to do the following:
Get messages
Create messages
Get attachments
List attachments
Import messages
Suppress messages
Search messages
Get message status
List message receipts
The Datafeed APIs create and manage real-time event streams from the Pod to your bot. These APIs can be used to do the following:
Create Datafeed
Read Datafeed
For more information on how Symphony Messaging Datafeeds allow your bot to create rich and interactive workflows, navigate here:
The Signal APIs create and manage tailored alerts based on mention or tag criteria. These APIs can be used to do the following:
List signals
Get signal details
Create a signal
Subscribe/Unsubscribe to a signal
List signal subscribers
This group of APIs perform testing and obtain diagnostics regarding the health of Symphony Messaging components. These APIs can be used to do the following:
Perform a component health check
Obtain Agent Info
Perform an echo test
Get session info
The Symphony Messaging Key Manager API's sole purpose is to authenticate a bot or API caller with the Key Manager.
If successful, the Key Manager API returns a Key Manager Token which is valid for up to two weeks. This Key Manager Token must be passed along with every subsequent Agent API request in order to encrypt/decrypt messages on the Agent server. You can read more about Authentication and Token management here:
List of Bot Permissions
This page lists the available roles and the associated privileges that may be required for certain endpoints:
In order to access Pod API endpoints, bots must be authenticated with the Pod. To do so, a Bot must call the .
The full list of Users API endpoints and their corresponding reference guide can be found at and .
The full list of Streams API endpoints and their corresponding reference guide can be found here:
The full list of Connections API endpoints and their corresponding reference guide can be found here:
The full list of Presence API endpoints and their corresponding reference guide can be found here:
The full list of Messages API endpoints and their corresponding reference guide can be found here:
The full list of Datafeed API endpoints and their corresponding reference guide can be found here:
The full list of Signals API endpoints and their corresponding reference guide can be found here:
The full list of Basics API endpoints and their corresponding reference guide can be found here:
In order to access Agent API endpoints, bots must be authenticated with the Pod and the Key Manager. To authenticate with the Key Manager, a bot must call the .
Privileges
LOGIN_WITH_PASSWORD
Privileges
ACCESS_ADMIN_API
ACCESS_CERT_PROVISIONING_API
ACCESS_USER_PROVISIONING_API
ACTIVATE_AND_DEACTIVATE_ANY_ROOM
ACTIVATE_SERVICE_ACCOUNT
ADD_AND_REMOVE_USERS_IN_ANY_ROOM
ADMIN_PRESENCE_UPDATE
ADMIN_RESET_USER_PASSWORD
BULK_MANAGE_USERS
CREATE_APPS
CREATE_END_USER_ACCOUNT
CREATE_SERVICE_ACCOUNT
DEACTIVATE_END_USER_ACCOUNT
DEACTIVATE_SERVICE_ACCOUNT
LOGIN_WITH_SHARED_SECRET
MODIFY_APPS MODIFY_APPS_ENTITLEMENTS
MODIFY_END_USER_ACCOUNT
MANAGE_BLACKLIST
MANAGE_USER_DELEGATES
MANAGE_INFO_BARRIERS
MODIFY_END_USER_ENTITLEMENTS
MODIFY_USER_APPS_ENTITLEMENTS
MODIFY_USER_DISCLAIMERS
MODIFY_SERVICE_ACCOUNT
MODIFY_SERVICE_USER_ENTITLEMENTS
PROMOTE_AND_DEMOTE_OWNERS_IN_ANY_ROOM
VIEW_ANY_STREAM_DETAILS
VIEW_ANY_STREAM_MEMBERSHIP
VIEW_APPS VIEW_APPS_ENTITLEMENTS
VIEW_BLACKLIST VIEW_DISCLAIMER_AUDIT_TRAIL
VIEW_INFO_BARRIERS
VIEW_PAST_STREAM_MEMBERS
VIEW_POD_DISCLAIMERS
VIEW_PROFANITY_ENFORCEMENT_AUDIT_TRAIL
VIEW_USAGE_POLICY_DETAILS
VIEW_USER_APPS_ENTITLEMENTS
VIEW_USER_AUDIT_TRAIL VIEW_USER_DETAIL
Privileges
ACCESS_ADMIN_API
VIEW_USER_AUDIT_TRAIL
MANAGE_ROLE_SCOPES
VIEW_ROLE_SCOPES
VIEW_USER_DETAIL
Privileges
ACCESS_ADMIN_API
IMPORT_MESSAGES
LOGIN_WITH_SHARED_SECRET SUPPRESS_MESSAGE
VIEW_MANAGE_MESSAGES
VIEW_USER_DETAIL VIEW_USER_ENTITLEMENTS
Privileges
ACCESS_ADMIN_API
DLP_CRYPTO_KEY
MANAGE_EXPRESSION_FILTERS
VIEW_DLP_VIOLATION
VIEW_PROFANITY_ENFORCEMENT_AUDIT_TRAIL
VIEW_EXPRESSION_FILTERS
VIEW_USER_DETAIL MANAGE_FILE_EXTENSIONS VIEW_USER_ENTITLEMENTS
Privileges
VIEW_PRIVILEGED_USER_AUDIT_TRAIL
ACCESS_ADMIN_API
Privileges
VIEW_USER_DETAIL
ACCESS_ADMIN_API CAN_CREATE_DISTRIBUTION_LIST
Privileges
SET_MALWARE_SCAN_STATE VIEW_MALWARE_SCAN_STATE
Privileges
VIEW_MALWARE_SCAN_CONFIG ACCESS_ADMIN_API VIEW_MALWARE_SCAN_AUDIT_TRAIL SET_MALWARE_SCAN_CONFIG VIEW_MALWARE_SCAN_STATE
In order for bots to access the Symphony Messaging REST API and other Symphony Messaging resources, bots must first authenticate.
As we learned in Overview of Pod API, bots must be authenticated on the Pod in order to access Pod API endpoints. To make authenticated Pod API calls, bots must pass a valid Session Token as a header of each Pod API request.
We also learned in Overview of Agent API, that bots must be authenticated on the Key Manager in order to access Agent API endpoints. To make authenticated Agent API calls, Bots must pass a valid Session Token and Key Manager Token as headers of each Agent API request.
The token you receive is valid for the lifetime of a session that is defined by your Pod's administration team. This ranges from 1 hour to 2 weeks.
You should keep using the same token until you receive an HTTP 401, at which you should re-authenticate and get a new token for a new session.
Note: Datafeeds survive session expiration, you do not need to re-create your datafeed if your session expires.
In order to obtain a valid Session Token and Key Manager Token, bots must call the Session Authenticate endpoint on the Pod and Key Manager Authentication endpoint on the Key Manager.
We recommend that bots follow the RSA authentication workflow in order to obtain valid Session and Key Manager Tokens:
For users that do not want to use RSA Authentication, bots can perform certificate-based authentication in order to obtain valid Session and Key Manager Tokens:
Overview of Symphony Messaging REST API Architecture
Symphony Messaging REST API is spread out over three main components: the Pod, API Agent and Key Manager. Let's take a closer look at these components below.
The Symphony Messaging Pod is a dedicated Symphony Messaging instance for each customer environment. It is a cloud-hosted component that handles all core operations necessary to provide the Symphony Messaging service to you. Since Symphony Messaging provides end-to-end encrypted messaging, all messages passed from user to user are fully encrypted at the time of sending, such that no Pod ever has access to the unencrypted contents of any message.
In addition, the Symphony Messaging Pod provides REST API endpoints in order for your bot to perform administrative functions on the Pod. You can read more about the Pod API here:
The API Agent is the component responsible for encrypting and decrypting content sent from and to a bot. The Agent provides REST API endpoints that allow a bot to send and receive encrypted messages, acting as the intermediary between a bot and the Symphony Messaging Pod. In order to safely encrypt and decrypt these messages, the Agent server interacts the Key Manager which provides the keys used for encrypting and decrypting messages.
Read more about the Agent API here:
The Key Manager generates and stores encryption keys which are used to encrypt and decrypt messages by the Agent Server. The Key Manager provides an authentication API that provides a unique Key Manager Token to a calling bot. This token is used to encrypt/decrypt messages on the Agent Server.
Read more about the Key Manager API here:
The three components above all interact with each other in order to create Symphony Messaging's secure messaging service. Let's take a closer look at the sequence of API calls a bot must make in order to send and receive encrypted messages on Symphony Messaging.
The sequence of API calls and component interaction is illustrated below:
1. First, a bot must authenticate with the Pod. It does so by calling the Session Authenticate endpoint.
1a. If successful, the bot will receive a valid Session Token. This Session Token must be passed along with all subsequent Symphony Messaging API requests destined for the Agent or the Pod.
2. Next, a bot must authenticate with the Key Manager. It does so by calling the Key Manager Authenticate endpoint.
2a. If successful, the bot will receive a valid Key Manager Token. This Key Manager Token must be passed along with all subsequent Symphony Messaging API requests destined just for the Agent.
3. If the bot wants to send a message, the bot will call the Create Message endpoint on the Agent API and pass both Session Token and Key Manager Token as a part of the request.
4. At this point, the Agent Server calls the Key Manager and requests the bot's encryption keys.
5. Next, the Agent Server validates the bot's Key Manager Token.
6. If successful, the Agent will encrypt the payload sent by the bot and will forward the encrypted message up to the Pod where it will be routed to the intended user or chatroom. The message will remain encrypted until it reaches its final destination.
For an even more detailed explanation, enroll in our Developer Certification Program:
For our enterprise customers, the API Agent and Key Manager components are deployed on-premise, while the Pod is always deployed in the cloud. Your bot or REST API caller is an application that must be deployed on-premise in this scenario.
An visual representation showing an on-premise deployment of Symphony Messaging components is shown below:
For our smaller customers, the API Agent and Key Manager may be co-hosted with the Pod in the cloud. Your bot or REST API caller can either be deployed on-premise or in your own cloud environment.
A visual representation showing an in-cloud deployment of Symphony Messaging components is shown below:
Welcome, Symphony Messaging Developer!
Here you will learn about all the ways to streamline work using integrated bots and apps. By leveraging Symphony Messaging's open APIs, developers can create innovative Bots and workflows, save time with strategic automations, and customize their Symphony experience by building third-party Extension Apps.
Keep reading to learn about core Symphony Messaging concepts in order to ensure a smooth development journey from your first line of code to putting your application in production.
It's time to get started. Navigate to one of the following guides to kick off your developer journey today!
Begin your development journey by taking an in depth look at Symphony Messaging Bots. Understand the core concepts and all the details needed to get your Bot up and running!
Want to build your first bot together with the Developer Relations team? Head over to our developer training centre and register for our online courses to get started, and it's free.
Bring customized and extensible workflows into Symphony Messaging through Extension Apps. Understand how to build, deploy, and manage your Extension App!
Enroll in our brand new Developer Certification program and obtain sandbox access for free! Register and enroll today!
In depth documentation and reference guide for the Symphony Messaging REST API:
In depth documentation and reference guide for the Symphony Messaging Extension API:
Check out our dedicated SDKs, Bot Developer Kit, Bot Generators and more to streamline your development process!
Securely embed stand-alone Symphony Messaging chat modules inside other websites and applications:
Bots in Symphony Messaging can use the following API endpoints to create, update and suppress messages in Symphony Messaging chats:
Message creation via the Create Message v4 endpoint
Message creation in multiple conversations using the Blast Message endpoint
Message update via the Update Message endpoint. Note this functionality is not supported on Mobile. On Mobile, updates will appear as new messages instead of replacing the actual message. Also, note the rules and limitations explained in the endpoint specifications documented here.
Message suppression via the Suppress Message endpoint
Please see below a quick animation showing a simple lifecycle of a bot message in Symphony Messaging with a message created by the bot, updated three times, and then finally suppressed.
In the Symphony Messaging message flow, messages are represented in the following markup language forms:
MessageML: A tag-based language that is a subset of XHTML. MessageML allows templating in Apache Freemarker.
PresentationML: MessageML translated into the equivalent XHTML tags so it can be rendered and processed by any HTML client.
ExtensionML: PresentationML translated to a special markup for use by a front end app to perform custom rendering of an entity.
The MessageML Render bot is a productivity tool that helps developers to create chat messages compliant with Symphony Messaging’s message markup language. Using the Render bot, you can get an overview of our message format capabilities, and quickly iterate on the format of a message while previewing the display.
The Render bot is available directly on Symphony Messaging. If this link doesn't work, you can look for MessageML Render bot in the directory.
The Render bot replaces the Presentation ML Live Renderer Tool website, which is no longer available.
The above diagram shows the following:
Your bot uses the Agent API to send messages in MessageML.
The Agent API encrypts the messages, and converts them to PresentationML where they are stored in Symphony Messaging 's data store.
The Symphony Messaging Datafeed delivers messages to end users or to the Desktop Application as PresentationML.
When Bot's retrieve messages via the API, the messages are delivered as PresentationML.
Each message in Symphony Messaging has a unique message ID.
To find the message ID:
In the Symphony Messaging web or desktop client, click the message timestamp. The Message Status module overlay opens. The message ID is shown in the overlay footer:
When a message is created via the API, a messageID
is returned in the response.
Encoding:
The message ID in the UI is in standard Base64 encoding.
Message IDs returned in API responses are in URL Safe Base64 encoding.
A message ID used in a URL should be in URL Safe Base64 encoding.
To obtain the URL Safe Base64 conversation (stream) ID, replace forward slashes with underscores, replace pluses with minuses, and ignore any trailing equal signs.
StreamID
URL Safe Base64 encoded StreamID
RUkxW4x40aB74g0UWpaMw3///ozLPsapdA==
RUkxW4x40aB74g0UWpaMw3___ozLPsapdA
Messages may include:
Maximum 80 entities (tags and mentions).
Maximum 2,500 unique tokens (distinct words) in the markdown representation of the message.
81,130 characters of the encrypted markdown representation of the message. This corresponds approximately to 60,000 characters before encryption. Note that there is a greater chance of reaching the token or the entity limit than the character limit.
The size of a message cannot exceed 1.5Mb, including both the messageML (message property) and the entityJSON (data property)
In the Symphony Messaging architecture, messages are stored in each company data store. When messages are sent externally (for example in External rooms), the messages are copied and transferred from one company to the other using the Amazon AWS SQS service At-least-once Delivery mechanism with Amazon AWS SQS. Amazon AWS SQS service ensures high availability through redundancy, with "At-least-once delivery" guarantee. This means that on very rare occasions, a message can be received more than once.
If this occurs, you might receive a second copy of that message. Therefore, Bots designed to work with External rooms (cross pod) should be idempotent: they should not be affected adversely when processing the same message more than once. Continue here to learn about how your bot should handle duplicate messages.
The following events are returned when reading from a real time messages and events from the datafeed:
All events follow the structure below:
id
: A randomly generated ID for the event, distinct from the ID of the message or chatroom or other objects that the event pertains to. These IDs are unique, except in the situation where two activities are performed simultaneously in the Symphony Messaging client (for example, members are added during the creation of a room). In this case, the two associated events will share the same id. This is for Symphony Messaging internal use.
timestamp
: The timestamp for when the event occurred in Unix timestamp milliseconds format. For certain events, this will be equivalent to the created timestamp of the object the event pertains to (for instance, room creation timestamp).
type
: The type of event that occurred (for instance, a "MessageSent" or "RoomCreated" event).
initiator
: The actor that initiated the event. Currently, initiator will always be a user. The initiator user may be distinct from the affected user (for instance, when one user adds another user to a chatroom).
payload
: An object containing event-specific information that is keyed off the event type.
Generated when a message is sent in an 1-1 chat or chatroom of which the user in context is a member, including messages sent by the user him/herself.
Events attributes details:
version
: version of the message payload, set when the message was created
message
: the message itself
attachments
: list of attachments to the message, not present if there are no attachments
user
: author of the message
externalRecipients
: indicates whether that message is sent to external users.
false
: the message was only sent to users internal to the company.
true
: the message was sent to at least one user outside of the company.
stream
: stream the message was posted into.
streamId
: identifier for the stream.
streamType
: stream type can be ROOM
, IM (1-1 chat)
, MIM (deprecated)
, or POST
.
data
: JSON structure contained inside an escaped string, NOT an actual JSON structure.
Generated when messages are suppressed:
Generated when either:
The user in context shares a wall post written by another user.
Another user shares a wall post written by the user in context.
Generated when an IM (1-1 chat) is created with the user in context as a member, initiated either by the user in context or another user:
Generated when a room is created by the user in context:
Generated when a room of which the user in context is a member is updated, including rooms updated by the user him/herself:
Generated when a room of which the user in context is a member is deactivated, including rooms deactivated by the user him/herself:
Generated when a room of which the user in context is a member is reactivated, including rooms reactivated by the user him/herself:
Generated when a user requests to join a room. Only the user who requested to join the room and the owners of that room will receive this event on their datafeeds.
The affectedUsers
attribute represents the owners of the room.
Available in Agent 2.56.0.
Generated when a new user joins or is added to a room of which the user in context is a member, including when the user himself joins or is added to a room.:
Generated when a user leaves or is removed from a room of which the user in context is a member, including when the user himself leaves or is removed from a room:
Generated when a user is promoted from a participant to an owner of a room of which the user in context is a member, including when the user himself is promoted to an owner or promotes another user.:
Generated when a user is demoted from an owner to a participant of a room of which the user in context is a member, including when the user himself is demoted to a participant or demotes another user:
Generated when a connection request is sent, either:
Sent by the user in context to another user.
Sent to the user in context by another user.
Generated when a connection request is accepted, either:
Sent by the user in context and accepted by another user.
Sent by another user and accepted by the user in context.
Generic System Event is a new category of generic real time event, that is generated in various situations, such as phone calls, events specific to federated chats, and more.
The event is generic, and its structure depends on the event's subtype (genericSystemEvent.eventSubtype
).
Most generic events are related to internal processes and are not useful for Bots, so you can just ignore them. When Symphony introduces events that can be relevant to Bots, they will be documented in this page.
The common structure to all Generic System Events is described below. The parameters
section will vary depending on the subtype of the event.
Overview of Symphony Messaging Datafeed
The Symphony Messaging datafeed provides a stream of real-time messages and events for all conversations that a bot is a member of. Any event that occurs within a bot's scope will be captured and delivered to the bot by the datafeed. The datafeed forms the basis of all interactive and conversational bot workflows as it allows bots to directly respond to Symphony Messaging messages and events.
The following illustrates the relationship between your bot, datafeed, and Symphony Messaging's components:
Bot creates datafeed via Symphony Messaging’s REST API
Agent creates secure upstream connection with the Symphony Messaging Pod
End user sends a message to a bot in a chatroom
Pod delivers ‘MESSAGESENT’ event to Agent
Bot reads datafeed via REST API
Agent delivers ‘MESSAGESENT’ event payload to the Bot
Events are delivered to your bot via the datafeed as JSON objects. Each type of Symphony Messaging event corresponds to a different JSON payload.
For example, if a user sends your bot a message, an event of type 'MESSAGESENT'
will be delivered to your bot through the datafeed:
Notice how each event returned by the datafeed has important metadata and attributes such as messageId
, timestamp
, (event) type
, initiator
, as well as the contents of the message itself inside of the payload object. Additionally, you can find the streamID
corresponding to the message and also information regarding externalRecipients
.
For a full list of the JSON payloads corresponding to each event type, continue here:
The BDK (Bot Developer Kit) comes bootstrapped with a DatafeedEventService
class that handles all of the logic for creating/reading datafeeds via the API, has best practices for maintaining datafeeds, and also provides event handling architecture that makes it easy to orchestrate complex workflows and introduce custom business logic to your bot.
As a bot developer, all you have to do is to implement generic EventHandler
classes, passing in a given event type as the type parameter for that class.
The following diagram shows the event handling workflow:
Bot creates datafeed via Symphony Messaging’s REST API
Agent creates secure upstream connection with the Symphony Messaging Pod
End user sends a message to a bot in a chatroom
Pod delivers ‘MESSAGESENT’ event to Agent
Bot reads datafeed via REST API
Agent delivers ‘MESSAGESENT’ event payload to the Bot
Bot routes event to appropriate event listener/handler
Inside of onMessageSent()
is where you implement your own business logic such as accessing a database, connecting to an external API, or reply back to your user by leveraging the Symphony Messaging API/BDK methods:
As you can see, the datafeed acts as the backbone of your Bot. In many cases your Bot will be waiting for events to come in through the datafeed, which it constantly 'reads'. When an event or message comes through the datafeed, your bot will 'listen' for the event, extract the relevant data from the JSON payload and kick off its intended workflow.
While you can write all of this datafeed logic yourself, our dedicated BDK toolkits provide out-of-the-box datafeed support and event handling logic making it easy to bootstrap your bot and add custom business logic.
The legacy agent/v4/datafeed API is out of support since April 30, 2023.
To facilitate this transition, a new feature called the datafeed bridge has been introduced in the Agent service so consumers of the deprecated APIs keep a functioning service.
This bridge is available starting with Agent 22.6 (June 2022) and can be enabled through the following configuration flag agent.df1ToDf2Bridge.enabled
.
Since Agent release 23.6 (June 2023), this bridge is enabled by default, but could still be disabled through configuration.
Then, starting with Agent release 23.9 (September 2023), the bridge is always enabled.
The v5 endpoints are different from the v4 ones, so migrating requires changes in your code. If you are using our Java BDK, migrating to v5 is a simple configuration change.
Otherwise, the mapping between the API endpoints is the following:
Path /agent/v4/datafeed (deprecated)
Via a POST on the endpoint /agent/v4/datafeed/create, the datafeed is created and then the ID is persisted in a file, which is by default datafeed.id on the bot side
The bot subscribes to this ID to retrieve datafeed events; if it cannot be retrieved by using this ID, a new datafeed is created
Via a GET on the endpoint /agent/v4/datafeed/{id}/read, the list of events within the specific datafeed identified with {id} is returned
Deleting a datafeed is not supported
Path /agent/v5/datafeeds
A Stream is like a container for messages exchanged between two or more users via a given 1 to 1 chat (IM) or chat room.
On the Symphony Messaging web or desktop client, this ID can be found by clicking on the timestamp of any message in the conversation. This will open the Message Status module, where the Conversation ID can be found, as shown in the following picture.
The Conversation ID in the Symphony Messaging Web Client is in Standard Base64 encoding and need to be converted to be URLSafe. Conversation IDs returned in API responses are already URLSafe Base64 encoding.
To obtain the URLSafe Base64 Conversation ID:
Replace forward slashes / with underscores _
Replace pluses + with minuses -
Ignore any trailing equal signs =
For example, the URLSafe Base64 encoding oflX1hwfmQ+AK/k/a/BB0y2n///q2+0KfbdA==
converts to lX1hwfmQ-AK_k_a_BB0y2n___q2-0KfbdA
.
The following list shows the existing endpoints:
This pages describes the implementation of certificate-based Authentication. For the API reference of Certificate Session Authenticate and Certificate Key Manager Authenticate, see the following API endpoints:
Symphony Messaging allows you to authenticate on the Pod and Key Manager with a client certificate that is signed by a trusted root certificate. When a bot calls the Session Authenticate endpoint, the Pod examines the client certificate provided in the TLS session to identify the bot user and return a Session Token. The same process occurs when a bot authenticates on the Key Manager.
All Symphony Messaging network communications take place over TLS-protected HTTP. The network uses authentication methods that require a client-authenticated TLS connection.
Client certificate authentication in TLS is configured at the port level. Two distinct ports are required for client-authenticated and non-client-authenticated connections. The web and mobile endpoints listen on port 443 (the default port number for HTTPS connections). The API endpoints require a separate port, typically port 8444.
The Admin upload a Signing certificate or Root certificate using the Admin portal.
The Admin provides to the developer a child client certificate derived from the Signing or Root certificate
The developer authenticates the Bot using the client certificate.
Note: It is also possible to directly upload a Client certificate in the Admin portal instead of a Signing or Root certificate.
Please note the below steps can only be performed by a Symphony Messaging Pod Administrator as they will have the necessary administrator privileges to access the Administration Portal.
Once you have obtained a copy of your Root Certificate Authorities (CA) Public "Signing Certificate", you can upload it using the following steps:
Once logged in click the Manage Certificates button then select Import
Drag and drop your Certificate file into the popup window:
Once you have uploaded the certificate file, click Import. If successful you will receive a confirmation message saying that the certificate has been uploaded successfully.
You can use the following commands to generate the service account certificate. The certificate must use 4096 bits length.
USERNAME = Service account username
PASSWORD = Service account key password
CA_CERT = CA certificate file
CA_KEY = CA key file
CA_PASSWORD = CA key password
OUTPUT_PASSWORD = PKCS12 file password
The following table shows the information you will need to provide to your PKI team:
The Symphony Messaging Admin then creates a service user with a username that matches the Common Name of the certificate, as you can see in the example below:
A successful response:
A successful response:
Pass the Session Token and Key Manager Token as headers for all subsequent API requests.
MessageML supports the following tags for grouping information within a message:
Here after you can find an example of a message sent by a bot and containing these content grouping tags as well as the structure of the messageML sent:
Please note that you can use content grouping tags mixed with other messageML tags such as Interactive Elements Forms. See the following example including forms in expandable-cards.
This article gives an overview of the message workflow and shows how message representations are used throughout the workflow. In its subpages are presented the specifications of messageML.
When calling API methods that create messages, the content of the message must be sent using MessageML markup. MessageML is a tag-based language that is a subset of XHTML, with the addition of tags specific to Symphony Messaging for embedding information (e.g. a mention) into a message.
You can find the specifications of the MessageML language in the attached subpages. Even if they are grouped in different categories for documentation clarity purposes, please note you can do your own mix, respecting the specific rules explained at each component specification level.
Also, please note that messages in MessageML markup are enclosed in a <messageML>
tag.
MessageML has full unicode support and messages should be sent using UTF-8 character encoding.
MessageML is formatted as XML and should have all tags properly formatted. For example, rather than using <br>
you must use <br/>
.
For string attributes, standard rules for escaping XML special characters apply, i.e. escaping:
'
with '
(if single quotes are used to quote the value)
"
with "
(if single quotes are used to quote the value)
<
with <
&
with &
Other XML named entity sequences such as >
may be used.
MessageML supports the following tags to embed media into messages:
For the following examples, we are sending an SVG image along with the message.
Note that an admin user might have to enable the sending of some specific file types. To do that, go to AC Portal >> Company Settings >> Edit Entitlements >> File Types.
This feature is intended to be used for small images, such as custom emoji. Our recommendation is that the total size of base64 encoded embedded images do not exceed 25KB per message.
Here after you can find an example of a message sent by a bot and containing an inline image as well as the structure of the messageML sent:
Generated when a user replies to a bot message that contains an interactive form with UX components such as text fields, radio buttons, checkboxes, person selectors and more. Please refer to for more information.
Deprecation Notice v4/datafeed (known as Datafeed v1) The legacy Datafeed v1 service will no longer be supported on April 30, 2023. Please read for more information on this transition. Please reach out to your Technical Account Manager or to the Developer Relations team for more information.
Symphony Messaging provides a Datafeed API that allows bots to easily and datafeeds.
Once a bot has created a datafeed, it has access to all of the within its scope, acting as a secure channel between a bot and all activity happening in the Symphony Messaging Pod. Additionally, all messages and events within a bot's scope are encrypted by the Agent before reaching your bot. That way the bot is the only one who can access the contents of these events and messages being delivered.
After the DatafeedEventService
creates/reads from the datafeed API, it categorizes each event based on its event type seen , and dispatches the event downstream to a generic event handler class. For example, If a user sends a message to bot inside a chatroom, the event will be read by the datafeed, and dispatched downstream to the EventHandler
class that that takes MessageEvent
in as a type parameter. Further the handle()
method belonging to your EventHandler
class will be called each type that event type is read by the datafeed.
This has an impact on you if some of your automations or bots are still using this API. Please upgrade to the new APIs.
We encourage you to migrate your bots to use the new . The bridge is a temporary solution, which objective is to facilitate the migration. If you use the BDK in or , the migration between v4 and v5 is automatic. We advise you to take this opportunity to migrate your bots to the BDK if you haven’t done so.
Via a GET on the endpoint , is returned the list of already created IDs for a service account
Via a POST on the endpoint , the datafeed is created and the ID is not persisted on the bot side → Even if the bot is stale, a GET on the same endpoint will retrieve the ID to which the service account is subscribed
Via a POST on the endpoint with a parameter ackId (empty string at the first query), the endpoint returns: the list of events, a new ackId string → This ackId permits acknowledgement of the last query and retrieve all events since the last one. All events received between the last payload and the new request are queued and therefore retrieved.
Via a DELETE on , the datafeed specified with the {id} is deleted.
Each Stream has a unique ID also known as a Conversation ID, that is returned when the IM or chat room is created using the , , and endpoints respectively. This ID can then be used in subsequent endpoints to perform operations on the IM or chat room.
Session Auth:
Key Manager Auth:
Navigate to the Symphony Admin Console for your Pod (e.g. ), then log in with your credentials
To authenticate on the Pod the bot must call the Session Auth endpoint: . Pass along the client certificate provided in the TLS session, returning a Session Token:
To authenticate on the Key Manager, the bot must call the Key Manager Auth endpoint: . Pass along the client certificate provided in the TLS session, returning a Key Manager Token:
When or messages using the API, MessageML shorthand tags are translated into equivalent XHTML tags and returned in .
• Sending an image via the API using the image URL
Sending an image via API using Data URL (base64 encoding).
Note that it is necessary to include data:image/imageType+xml;base64
before the data string, as shown in the following example:
Details
Example Values
Certificate Type
Single Domain Certificate
Common Name (CN)
demo-bot1
Organization
Excelsior Bank
Department
Collaboration Services
admin@bots.symphony.com
Locality
London
State / Province
London
Country
GB
Key Size
2048 bits
Tag
Description
Optional attributes
<h1>
, <h2>
, <h3>
, <h4>
, <h5>
, <h6>
Heading text. 6 levels.
• class
<p>paragraph</p>
Paragraph formatting.
• class
<hr />
Horizontal rule.
None.
<ul>
<li>list item</li>
</ul>
Unordered or bullet list.
Cannot be empty, must contain at least one child <li>
item.
• class
<ol>
<li>list item</li>
</ol>
Numbered list.
Cannot be empty, must contain at least one child <li>
item.
• class
<div>paragraph</div>
Block of text.
• This tag can be used to specify visual styles, by adding a class
attribute.
• This tag is used to create Structured objects.
• This tag is also the root of any message read through the API.
• class
: color options
• data-entity-id
• data-icon-src
• data-accent-color
• See below for list of translated PresentationML attributes.
<card>
(see example below)
Inserts a card. It contains two different sections:
• the <header>
(always visible)
• the <body>
(hidden)
• iconSrc
: image will be resized to 28 pixels by 28 pixels, use spacious mode. (.jpg, .png and .gif)
• accent
: use background color values to select the accent color of the card.
<expandable-card>
(see example below)
Inserts a card with new styles and multiple levels of display within the card
• state
(mandatory) in <expandable-card>
can take 3 values:
- "collapsed": only header is visible
- "cropped": card expanded but the body is cropped
- "expanded": card fully expanded
• variant
(optional) in <body>
: defines the style of the card. It can be either "default" for the default blue style, or "error" for the red error style
Tags
Description
Attributes
<img src="url"/>
Image. Images have a max height of 256px; otherwise, the default size is the size of the image. For more information on how to send images through API call, refer to Sending images.
• src
• class
Tags
Description
Optional attributes
<table>
<tr>
<td>text</td>
</tr>
</table>
Render "text" in a table format.
• class
• rowspan
• colspan
<thead>
, <tbody>
, <tfoot>
Table sections.
• class
Tag
Description
Optional attributes
<br/>
Insert a line break.
None.
<a href="url">
Link Text
</a>
Insert a hyperlink that will be displayed in the message.
• href
: the URL of the link
• class
: color options.
<b>text</b>
Bold formatting.
Note: when receiving a message from an Agent that contains whitespace between the last character in a bolded section and the closing </b>
tag, the bold section will be returned in Markdown (i.e. surrounded by double '*' characters) instead of XHTML tags.
• class
: color options.
<i>text</i>
Italics formatting.
Note: when receiving a message from an Agent that contains whitespace between the last character in an italics-formatted section and the closing </i>
tag, the italics section will be returned in Markdown (i.e. surrounded by single '*' characters) instead of XHTML tags.
• class
: color options.
<pre>
preformatted text
</pre>
Preformatted text.
• class
: color options.
• Non-HTML MessageML shorthand tags are not supported inside <pre>
.
<span>text</span>
No formatting.
• This tag can be used to specify visual styles, by adding a class
attribute.
• This tag is used to create Structured objects.
• class
: color options.
• data-entity-id
• See below for list of translated PresentationML attributes.
<code>code sample</code>
Format the text as a block of code.
language
: Use the language attribute to benefit from language specific code highlighting. The following languages are available: c, cpp, csharp, css, html, java, js, jsx, php, python, r, typescript, tsx, plaintext, yaml, scala, json, shell, markdown
Note: language
attribute is only available with Agent 20.14+.
Note: please see escaping rules for Special Characters when using code tag
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Backward client-compatibility behavior (e.g. external rooms)
Client 1.5 release
Initial release
Since the first version
Since the first version
-
Since the first version
Closing <b> and <i> without line breaks*
Since the first version
Since the first version
-
1.53
Support of the language attribute on <code> blocks.
Agent 20.14
Since the first version
-
Symphony Messaging Elements Forms enable bots to send messages that contain interactive forms with pre-designed text fields, dropdown menus, person selectors, buttons and more. This allows bots to interact with users in a very easy and guided way.
By reusing our pre-designed standard UX component libraries, Elements provide developers with out-of-the-box tools to easily create interactive bot messages that look and feel like they belong in Symphony Messaging. To use the Elements, you just need to call the Create Message endpoint with your bot, using the MessageML format.
Here after, you will find a brief introduction of how to send Elements, then an update of the message flow for Elements, and finally the form specifications.
To start using Symphony Messaging Elements, you first need to create a form element using the <form>
MessageML tag. The form element can be considered the "frame" of a message, containing elements that will be sent by the bot and subsequently read by the datafeed.
You can see below an example of how a user can interact with a form that was sent by a bot, as well as the messageML structure of the message that was sent by the bot, and the payload that is generated and therefore delivered to the bot via the datafeed, containing the information completed and submitted by the user. Please use the tabs to navigate between these 3 documents.
Once the user has submitted the form, it becomes disabled. However, if the conversation is reloaded, the form resets and the user is able to send a new reply. If your workflow requires a single reply per user, please implement this control on the Bot side.
To begin leveraging Symphony Messaging Elements in your bot workflows continue onto our Available Elements that you can find in the subpages.
Every message exists as part of a flow, in a continuum of events that results in user interaction.
Here is that flow in colorful diagram form, for you to know more about each stage of the message:
A Bot sends a message with Symphony Messaging Elements in a form
The message/from is visible to users. Users interact with the elements
Once submitted, the data is submitted to the bot
Bots can access the data, by reading the datafeed.
Forms are represented by the <form> tag, as you can see in the examples above.
Attribute
Type
Required?
Description
id
String
Yes
Identifies the form.
multi-submit
String
No
Specifies that the form can be submitted several times by the user and the reset behavior. See the possible values and behaviors in Rules and Limitations below.
The form element can be considered the "frame" of a message, containing elements that will be sent by the bot and subsequently read by the datafeed.
To be considered valid, the form tag must contain at least one action type "Button" as a child. For more information, refer to Buttons.
All of the data within a form element will be sent to a bot via the datafeed when a user clicks one of the action buttons in that form. The name
attribute of the button will be the value of the action
field within the datafeed payload. That way the bot manager can know which button triggered the submission of that form. Starting with Agent 23.11, the auto-submit feature allows to submit the form with a TextField, or with a DropDown Menu.
If there is more than one element in the form having the same name
attribute, the value is converted to an array. Every index of the array is related to a specific element value. The index order is not guaranteed, so the developer needs to iterate through this array and collect the values.
When a form is submitted and multi-submit
attribute is not specified, all the elements within it will be disabled, not being possible to edit or resend the same form. However, if the page is refreshed, the user can fill out the form again and submit it as a new form.
The attribute multi-submit
allows the us to submit the form several times even without needing to refresh the page. It can take 2 different string values:
reset
which resets the form to the default value when enabling it again for the user,
no-reset
which keeps the values that were submitted by the user when enabling it again for the user.
When designing forms, it is important to consider the message size limit. For more information refer to Message size limits.
The following example shows three forms being used as follows:
The first form (default) shows how Symphony Messaging users interact with a form, and how the Elements are disabled once the form is submitted.
The second form (multi-submit-reset) shows how to create a form that can be submitted several times and which resets all its Elements to their default value once the user submits it. You note the Elements are first disabled while loading, then the button shows the user the form has been submitted, before the form is enabled back to its default value for the user to submit it again.
The third form (multi-submit-no-reset) shows how to create a form that can be submitted several times and which keeps last submitted values once the user submits it. You note the Elements are first disabled while loading, then the button shows the user the form has been submitted, before the form is enabled back to its last submitted value for the user to submit it again.
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Client 1.5 release
Backward client-compatibility behavior (e.g. external rooms)
Initial release
2.55.9
Since first version
1.55
Not working
Multi-submit attribute
20.13
21.8
-
-
This page describes the changes introduced by the new enhanced tags for Symphony developers
Symphony users can now tag financial instruments using the Enhanced Tag feature.
It displays a typeahead when the user presses # in a chat conversation. The upper section of the typeahead shows regular hashtags previously exchanged with that user and the lower section, under INSTRUMENTS shows Stocks, ADRs, ETFs and Indices.
By hovering over the Enhanced Tag, users can see essential reference data like company name, identifiers, exchange and currency.
The enhanced tags replace the existing $cashtags. This means that moving forward your apps and bots will need to handle enhanced tags instead of $cashtags.
The impact of the enhanced tags on existing bots & apps has been carefully reviewed to ensure that there is no backward compatibility issue. However, we strongly recommend that you conduct some tests to validate the change.
Please read below to understand the changes introduced with the new enhanced tags.
Existing bots can still produce legacy $cashtags both using the short tag notation (<tag cash="AAPL"/>
), and using a Standard Entity ("type": "org.symphonyoss.fin.security"
)
Existing bots can easily include enhanced tags in messageML now. To do so, you will need to cuse the <tag> and provide one or several identifiers for that instrument. The instrument will then be resolved against our reference database, and properly formatted in the message. More information is available here.
Bots consuming existing $cashtags in received messages are not impacted if they rely on the structured object (entityJSON). When they receive an enhanced tag, the exact same structure and properties will be present. There will be additional properties though, and the version field will be set to "2.0" instead of "1.0".
Bots that rely on the text representation of the tags (message field instead of the entityJSON field) will also not be impacted. However, this is not a recommended practice as this text representation may change to match the enhanced tag displayed symbol instead of the existing ticker in the future (e.g., text would change from "$TSLA" to "TSLA US"). If you rely on the text representation of the $cashtag instead of the structured data, you should consider updating the bot to rely on the structured data included in the entityJSON instead.
If you plan to upgrade your bot to leverage the new properties of the enhanced tags, please consider that you may still encounter legacy $cashtags for a long period of time, so should build around that. To detect if you are receiving a legacy tag or an enhanced one, you can rely on the version field included in the structure (org.symphonyoss.fin.security.version
). It is set to "2.0" for the new enhanced tags.
Existing signals will continue to work with the new enhanced tags. Messages containing the new tags will still match with the existing signals, meaning a signal matching "cash: AAPL
" will match messages containing both "AAPL" and "AAPL US" for example.
For now, it is not yet possible to create signals based on the newly introduced properties of the enhanced tags, but this will change in the coming months. For example, it is not yet possible to create a signal that will match only on "TSLA US", but if you create a signal that matches on "TSLA", it will match messages containing "TSLA US" as well.
The Signal API has not changed. In the coming months the API may change to adapt to the new tags and in that case this change will be introduced following a notice period provided that there is an impact on the API contract.
Existing apps that register a UI extension on $cashtag hovercards continue to receive the existing object for new enhanced tags.
However, whenever your app is notified of a user click on the hovercard, the app will receive in addition to the existing object a new structure containing the new enhanced tags properties.
If you decide to upgrade your app to leverage the new properties of the enhanced tags, please consider that you may still encounter legacy $cashtags for a long period of time, so should build around that. To detect if you are receiving a legacy tag or an enhanced one, you can rely on the version field included in the structure (org.symphonyoss.fin.security.version). It is set to "2.0" for the new enhanced tags.
Please note below the differences between the structures of the enhanced tags and the legacy $cashtags.
MessageML supports the following tags to embed additional information into messages:
Tag
Description
Optional attributes
<mention uid="123456789"/>
Insert a mention for the user whose Symphony userid is 123456789
.
<mention email="user@music.org"/>
Insert a mention for the user whose email address is user@music.org
.
• strict
=true
, the API will throw an error if no user of that email address exists. (default)
• strict
=false
. Message is accepted even if the user cannot be resolved.
<hash tag="label"/>
Insert "label" as a free-text hashtag.
<cash tag="ticker"/>
Insert "ticker" as a free-text cashtag. Important: when sending numeric cashtags as signals, add a *
after the $ sign, for example, $_122450. <messageML>
`<cash tag="$_122450"/> `\
Note: Cashtags are deprecated. Please use the <tag /> notation for financial instruments.
<tag />
Insert a financial instrument (enhanced tag) in your message, coming from our reference database.
The following instruments are supported: Stocks, ETFs, Indices and currency pairs.
To identify an instrument, you'll need to provide at least one identifier (e.g. an ISIN), and optionally some filters if your identifier is not specific enough.
You can also specify a fallback-ticker
that will act as a free-text tag (workaround) if we are not able to find the instrument referenced.
Identifiers:
fullbbgcompticker
figi
bbgcompticker
us-code
isin
local-code
Filters:
instrument-class
bbgmarket-sector
return-main-listing
country-code
operational-mic
Others:
fallback-ticker
<chime />
Send a chime alert.
Note: No other content (even line breaks) is permitted with a <chime/>
tag. Please see an example of the messageML to send a chime below.
Here after you can find an example of a message sent by a bot and containing these tags specific to Symphony Messaging as well as the structure of the messageML sent:
Below several examples of financial instruments, using different types of identifiers and filters.
When identifiers and filters are not sufficient to identify a unique match, or when an instrument is not found in our reference database, an error is returned, except if a fallback-ticker
is specified.
The Apache Freemarker uses the HTML output format by default. In some cases, special characters placed within the MessageML must be HTML-escaped, otherwise, the request sending the MessageML will receive a 400 error response. The following are examples of valid HTML-escaping:
Character
HTML escaping
Required escaping
messageML example
<
<
Yes
<messageML><</messageML>
&
&
Yes
<messageML>& </messageML>
$
$
Yes The $ character only needs to be escaped if it comes before the { character.
<messageML>${}</messageML>
#
#
Yes The # character only needs to be escaped if it comes before the { character.
<messageML>#{}</messageML>
>
>
No
<messageML>></messageML>
"
"
No
<messageML>"</messageML>
'
'
No
<messageML>'</messageML>
*
*
No
<messageML>*</messageML>
%
%
No
<messageML>%</messageML>
Here after you can find an example of a message sent by a bot and containing these special characters as well as the structure of the messageML sent:
You should only need to follow the below steps if your organisation uses self-signed custom certificates which are signed by an Internal Root Certificate Authority (Root CA).
In an enterprise environment, it is very common to have an internal root certificate authority (root CA) that signs all certificates used internally for TLS termination on HTTPS sites. While this trust is federated on the operating system level for all domain-connected Windows workstations, non-Windows servers are typically not configured the same way. Even on a Windows workstation, your Java runtime might not necessarily trust the enterprise root CA certificates.
If a Java application attempts to make a connection to an untrusted remote server, it will throw an SSLException
and fail to connect.
A Java truststore is a bundle of certificates used when establishing connections to remote servers (this could be internal or on the Internet). Each Java runtime ships with a default truststore called cacerts
. Adding certificates directly into cacerts
is usually not recommended unless there is a centrally-managed deployment process for all workstations and servers.
A bot requires connections to several servers like the key manager or API agent, as defined in your bot configuration. This configuration should also define the path to a custom truststore, which ensures that the trust relationship is maintained even as the bot's deployment location is moved, as long as the truststore file is moved together with the deployment artifacts.
The BDK for Java uses the JKS format and the configuration is under the ssl
section within the config.yaml
:
If you are using the BDK for Java's Spring Starters, modify the appropriate application.yaml
file to include the ssl
section under the bdk
section:
The BDK for Python uses a concatenated PKCS format. Your config.yaml
should look like this:
A typical internal certificate has has least 2 certificates in its chain - the certificate itself as well as the root CA certificate that signed it. Some enterprises will have intermediate CAs to form a 3-certificate chain. When Java establishes a connection to a remote server, it requires all signing certificates in that chain to be trusted. Hence, when building a truststore, you will need to import every signing certificate in the chain of each remote server that is not already present in the cacerts
bundle.
Publicly-trusted signing certificates from established certificate authorities are already in cacerts
so you will most likely not need to import certificates for services on the public Internet. However, if the Java runtime you are using pre-dates the existence of those certificates, then you will need to add them manually.
The keytool
command that ships with all Java runtimes is used to manage truststores. You can build a new truststore or append to an existing one using the same command as follows:
You may use the following convenience shell script to automatically build out a new truststore based on a list of servers. Please remember to include all possible connections that your bot requires. This may include: all Key Manager servers, all API Agent servers, the Symphony Messaging pod, any other third-party or internal endpoints that the bot will interact with.
If you have no access to a Linux-based shell, you can also manually perform this process within a Windows environment. All you need is a browser and a Java runtime with keytool
installed.
Step 1: Visit an endpoint on each server in your browser and view the certificate
Step 2: Select the next signing certificate
In the Certification Path
tab, select the end certificate's parent and View Certificate
Step 3: Export the certificate
In the Details
tab, click Copy to File..
and select the Base-64 encoded X.509
option then save the certificate to disk.
Step 4: Repeat for the rest of the chain
Repeat Steps 2 and 3 until you have exported all signing certificates in the chain.
Step 5: Build the trust store
Launch command prompt and repeat this command for each exported certificate
You can verify the entries by using the -list option:
Please find below the list of icons that can be used with Elements Buttons.
The Bot Developer Kit (BDK) is the preferred tooling for Java or Python developers to get started building bots on Symphony Messaging.
The Symphony Messaging Generator offers a fast way to bootstrap your Symphony Messaging BDK project in Java and Python.
First, install yeoman and the Symphony Messaging Generator. Then, create a directory for your new bot project. Finally, launch the generator.
This will prompt you with a number of questions about your bot and pod configuration. Type in your bot's information, using arrow keys to scroll and press enter to move on to the next prompt.
When prompted for Select your type of application
, choose Bot Application
.
The Symphony Messaging Generator creates a basic configuration file that works for fully cloud-hosted pods. If you have additional network requirements like a web proxy or an on-premise Key Manager and API Agent, refer to the complete configuration guide below for more details.
For any bot to work, it requires a service account with a matching username and public key. The Symphony Messaging Generator creates a configuration file based on the answers supplied, including the bot username and the path to the generated key pair. These can be changed by modifying the config.yaml
file. If you do not already have a service account set up, follow the instructions on this page to continue.
Open the project in your preferred IDE and use the IDE's launch/debug feature to start the bot. Alternatively, continue using the command-line to launch the bot using the appropriate command.
The bot should start up successfully and the following log indicates that it is up and listening:
[main] INFO com.symphony.bdk.core.service.datafeed.impl.DatafeedLoopV2 - Start reading events from datafeed
symphony.bdk.core.service.datafeed.datafeed_loop_v2 - DEBUG - Starting datafeed V2 loop
Now that you have a basic bot project ready, you can proceed to find out more about building bots using the respective kits:
A dropdown menu is a static list of items that appears whenever a piece of text or a button is clicked. This is a graphical approach presented to users from which they can choose one or several values from the list presented.
A dropdown menu is also known as a pull-down menu, pull-down list, dropdown list or dropdown box.
The dropdown menu is represented by the <select>
tag which provides a menu of <options>
.
Each <option>
element should have a value
attribute containing the data value to submit to the server when that option is selected; You can also include a selected
attribute on an <option>
element to make it selected by default when the page first loads.
Attribute
Type
Required?
Description
name
String
Yes
Required attribute of the <select>
tag. It identifies the dropdown menu.
required
Boolean
No
Optional attribute of the <select>
tag. it is a Boolean attribute indicating that an option with a non-empty string value must be selected.
value
String
Yes
Required attribute of the <option>
tag. It contains the data value to submit to the server when that option is selected.
selected
Boolean
Optional
You can include a selected attribute on an <option>
element to make it selected by default when the page first loads. Accepted values: true
and false
.
data-placeholder
String
Optional
Text displayed in the dropdown menu before an option is selected. It can be a short hint or a title for the dropdown menu.
title
It accepts a simple text and \n
for line breaks
Optional
The description that will be displayed when clicking the tooltip icon located on top of the Masked Text Field Element. Max length: 256 characters.
label
String
Not required but it is recommended if title
is defined
Definition of the label that will be displayed on top of the Masked Text Field Element.
multiple
String Boolean
Optional Default is false
Allows users to select multiple values and the developer to send dropdown with multiple preselected options.
min
String
Integer ≥ 2 and ≤ max
attribute
Optional
Can be used only if multiple
attribute is set to true
Minimum number of options to be selected by the Symphony user.
NB: If undefined, no minimum option needs to be selected. Please use required
attribute if you want to set min to 1.
max
String Integer ≥ 2
Optional
Can be used only if multiple
attribute is set to true
Maximum number of options to be selected by the Symphony user.
NB: If undefined, user will be able to select as many options as wished. Please use multiple
attribute set to false if you want to set max to 1.
auto-submit
Boolean
Optional Default is false
When enabled, selecting a value will submit the form.
The <select>
tag:
The <select>
tag stands for our dropdown parent tag, which has <options>
as its children; one for each select.
Select tags only accept <option>
tags as children. The <select>
tag must contain at least one <option>
tag.
The only valid attributes of the <select>
tag are name
and required
.
Note that, by default, Symphony users will only be able to select one option from the dropdown menu. However, using the attribute multiple set to true together with min and max attributes, users will be able to select several options from the dropdown. Please see below the examples to know how to use these attributes.
The <option>
tag:
The <option>
tag cannot have other <option>
tags as children. The only valid child of a <option>
tag is a text node, which specifies the text that will be displayed for that option inside the dropdown menu. The text node is also required.
The only valid attributes of the <option>
tag are value
and selected
.
Only one <option>
of a given select can have the attribute selected
as true.
If neither the selected
or data-placeholder
attributes are set, the default text (title) of the dropdown menu will be "Dropdown".
The following examples show dropdown menus being used as follows:
The first dropdown (init) shows how to display a default preselected option ("opt2": "With selected option"). Note that the preselected option is sent to the payload when submitting the form.
The second dropdown (data-placeholder) shows how a placeholder text ("Only data-placeholder") is displayed in the UI. Please note the placeholder text is not sent in the payload if no option from the dropdown menu has been selected by the enduser.
The third dropdown (noreq) shows how a user can interact with a non-required field. Even no option is selected by the user, it does not prevent the enduser from submitting the form.
The fourth dropdown (req) shows the behaviour of the unique required field of the form, which cannot be submitted in case no option from the dropdown menu is selected by the user; an error is displayed under the field in case the user submits the form with this empty field.
The fifth dropdown (label) shows how a label text ("My Label") is displayed.
The sixth dropdown (tooltip) shows how a title text ("My Tooltip/n With a second line") is inserted in the UI under the (i) icon, and how the text entered in the title parameter is displayed when the enduser clicks on the icon.
The seventh dropdown (multiple) shows how to combine multiple attribute with min/max rules to guide users selecting between 3 and 5 options.
Initial release
2.55.9
Since first version
Since first version
Label
20.6
Since first version
Since first version
Tooltip (title)
20.7
Since first version
Since first version
Multiple (with min and max) attributes
20.14
21.12
Since first version
auto-submit
23.11
23.12
Not supported yet.
This pages describes the implementation of RSA Authentication. For the API reference of RSA Session Authenticate and Key Manager Authenticate, see the following API endpoints:
Note: The following authentication sequence is provided out of the box by our dedicated SDKs and BDK. To learn more about authenticating using the SDKs or BDK proceed to one of following configuration guides:
The Authentication process requires the following steps:
The user creates a public/private RSA key pair.
The admin imports the public key into the pod using the Admin Console or public APIs.
The user creates a short-lived JWT (JSON Web Token) and signs it with their private key.
The bot makes a call the the authentication endpoints. Here, the server checks the signature of the JWT against the public key and returns an authentication token.
The token you receive is valid for the lifetime of a session that is defined by your pod's administration team. This ranges from 1 hour to 2 weeks.
You should keep using the same token until you receive a HTTP 401, at which you should re-authenticate and get a new token for a new session.
Datafeeds survive session expiration, you do not need to re-create your datafeed if your session expires.
Symphony only supports the following cipher suites:
ECDHE-RSA-AES256-GCM-SHA384 (Preferred)
ECDHE-RSA-AES128-GCM-SHA256
DHE-RSA-AES256-GCM-SHA384
DHE-RSA-AES128-GCM-SHA256
The public/private key pair for signing authentication requests requires the following:
A JWT payload has to be signed with RS512 (https://tools.ietf.org/html/rfc7518#section-3.1)"
A X.509 format for public keys and PKCS#1 or PKCS#8 for private keys
PEM-encoded keys
Generate the PKCS#1 keys manually using the following commands:
Generate the PKCS#8 keys manually using the following commands. You can provide the Service Account's username as the Common Name (CN) but it is not a mandatory requirement.
Sign the authentication request using either privatekey.pkcs8
or privatekey.pem
, depending on the support available in the JWT library.
The file publickey.pem
is the public key. This is the key you will import into the pod in step 2.
Please note the below steps can only be performed by a Symphony Messaging Pod Administrator as they will have the necessary administrator privileges to access the Administration Portal.
Navigate to the Admin Console and create a new Service Account. Copy the contents of the pubkey.pem file you just created and paste into the textbox under the Authentication section:
Add your bot's basic information:
If successful, you should see the following:
To authenticate on the Pod and the Key Manager, the bot must call the authentication endpoints, passing a short-lived JWT token in the body of the request. The JWT token must contain the following:
a subject matching the username of the user to authenticate
an expiration time of no more than 5 minutes from the current timestamp (needed to prevent replay attacks)
a signature by a private RSA key matching a public key stored for the user in the Pod
The following script generates the authentication request:
The output of the script is a JWT:
The authentication token can be inspected on https://jwt.io/ or https://www.jsonwebtoken.io/.
Obtain a valid Session Token by making a POST request to your company's Session Auth endpoint:
A successful response:
Obtain a valid Key Manager Token by making a POST request to your company's Key Manager Auth endpoint:
A successful response:
You can replace the public key pubkeyA for a user with a new key, pubkeyB (for example, as part of an organization's key rotation schedule). Note the following outcomes:
When a key is replaced, the key pubkeyA becomes the user's previous key, and the newly uploaded pubkeyB becomes the current key.
The previous key is valid for 72 hours, but you can extend that period indefinitely in intervals of 72 hours.
While the previous key is valid, both keys can be used for authentication. When it expires, it can no longer be used to authenticate the user.
A user can have at most one previous key.
Alternatively, you can revoke a user key (current or previous), for example, if the key is compromised. Note the following outcomes:
When a key is revoked, it can no longer be used for authentication.
If a user has a non-expired previous key and their current key is revoked, the previous key becomes the new current key.
When a key is revoked, the user's sessions initiated with RSA authentication are invalidated.
To replace/revoke a key, navigate to the Bot's account in the admin portal > RSA > Replace or Revoke:
You can also use the following REST API call to programmatically replace a public key:
Additionally you can programmatically revoke a public key using either currentKey or previousKey. Use the following REST request to programmatically revoke a public key using currentKey:
Use the following REST request to programmatically revoke a public key using previousKey:
Use the following REST request to programmatically extend a public key:
You CANNOT perform the following actions:
EXTEND on the current active key
EXTEND over an expired key
EXTEND over the previous key when no previous key is set for the user
EXTEND when no expiration date is set for the user's previous key
REVOKE over the previous key when no previous key is set for the user
REVOKE over a current key when no current key is set for the user
SAVE when the user already has a valid current and previous key set
SAVE over an old key
Note: When performing a SAVE, the key must be different from your current key.
see Text Field for more information
The masked text input is a single-line plain text editor control in which the text is masked by replacing each character with a dot (•) symbol, providing a way for the user to enter data without people over their shoulder being able to see what they are entering.
Use the maxlength
and minlength
attributes to specify the maximum and minimum length of the value that can be entered.
Attribute
Type
Required?
Description
name
String
Yes
Identifies the input content.
placeholder
String
No
Specifies a short hint that describes the expected value of the input field.
required
Boolean
No
If true
, it specifies that the input field must be filled out before submitting the form. Accepted values; true
and false
.
masked
Boolean
Yes
In case you want to include a text field with masked characters [hidden by asterisk (*) symbols] , you must set masked
as true. If true
, it creates a masked text field with hide/show options.
maxlength
Integer
No
The maxlength
attribute allows you to specify a maximum number of characters that the user can input.
minlength
Integer
No
The minlength
attribute allows you to specify a minimum number of characters that the user can input.
pattern
String
No
Regex String to match for input validation
pattern-error-message
String
No
Error message returned to user if pattern
parameter matches user input
title
It accepts a simple text and \n
for line breaks
No
The description that will be displayed when clicking the tooltip icon located on top of the Masked Text Field Element. Max length: 256 characters. Available from Symphony v20.8 and above.
label
String
Not required but it is recommended if title
is defined
Definition of the label that will be displayed on top of the Masked Text Field Element. Available from Symphony v20.8 and above.
With Symphony v20.6, bot developers can use Regex to validate text fields
and text areas
using the pattern
and pattern-error-message
attributes.
For more information and examples, refer to Regular Expressions - Regex.
The masked text field has a max number of 128 characters.
The masked text field must be a self-closing tag or have no children.
The masked text field is a feature that only hides the content in the UI. Even though it is masked in the UI, the text will be submitted in clear and processed & encrypted the same way as any other text entered in Symphony.
Compliance Officers have access to the content of the masked text field.
The following examples show masked text-fields being used as follows:
The first masked text-field (init) shows how to display a default text ("With initial value"). Note that the default text would have been sent to the payload if it had not been deleted before submitting the form.
The second masked text-field (placeholder) shows how a placeholder text ("Only Placeholder") is displayed in the UI. Please note the placeholder text is not masked nor sent in the payload if no text has been entered in the field by the enduser.
The third masked text-field (noreq) shows how a user can interact with a non-required field. Even if the field is empty (only a placeholder text is present but does not count as a value), it does not prevent the enduser from submitting the form.
The fourth masked text-field (req) shows the behaviour of the unique required field of the form, which cannot be submitted in case it is not filled; an error is displayed under the field in case the user submits the form with this empty field.
The fifth masked text-field (regex) shows the behaviour of the field when a regex pattern is entered. You can note that the pattern-error-message is displayed under the field if the input does not follow the pattern required by the bot developer.
The sixth masked text-field (min) shows how to force users to enter an input with a minimum number of characters, and how an error message is displayed under the field if the input does not respect the minlength required.
The seventh masked text-field (label) shows how a label text ("My Label") is displayed.
The eighth masked text-field (tooltip) shows how a title text ("My Tooltip/n With a second line") is inserted in the UI under the (i) icon, and how the text entered in the title parameter is displayed when the enduser clicks on the icon.
Radio buttons are shown as small circles, which are filled or highlighted when selected. Only one radio button in a given group can be selected at the same time.
Frequently, a set of radio buttons represents a single question which the user can answer by selecting a possible answer.
Note: If you want the user to be able to select more than one option, use the Checkbox element.
Attribute
Type
Required?
Description
name
String
Yes
Identifies the radio button.
value
String
No
The value
is the string that will be sent to the server. If the value is not specified, the string on will be sent by default.
checked
Boolean
No
If true
, it specifies that the <radio>
element should be pre-selected (checked) when the page loads. Accepted values: true
and false
.
The text node of the MessageML will be converted to the <label>
tag. This will preserve the formatting tags <i>
and <b>
, if present.
Radio buttons are presented in radio groups (a collection of radio buttons describing a set of related options). Only one radio button in a group can be selected at the same time. Note: The radio group must share the same name (the value of the name
attribute) to be treated as a group. Once the radio group is created, selecting any radio button in that group automatically deselects any other selected radio button in the same group.
A form can have a maximum of 50 radio buttons. Note: The limit in previous versions was set to 20, so this limit may still apply when sending messages to customers with an earlier version of Symphony (before 20.10).
Once the user has selected a radio option, it can be deselected only by clicking on another radio option. The only way to deselect all the radio options is by clicking the reset button.
The following example shows radio buttons being used. It shows how developers can use the checked parameter with the value01 preselected when the form is sent. It also shows how users can select another radio button and how it automatically unselect any other value checked, as the 3 radio buttons have the same name and therefore are part of the same group "groupId".
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Backward client-compatibility behavior (e.g. external rooms)
Initial release
2.55.9
Since first version
Not working
The text field cannot have children tags but it can have a default text (initial value) between the <text-field></text-field>
tags. See Examples below for more details.
You can add a default text in your text field by including it between the <text-field></text-field>
tags. Note that unlike the placeholder
text, the default text will be sent with the form if not edited by the user.
Input Validation - Pattern: the max length for all attributes is set to 256.
The following examples show text fields being used as follows:
The first text-field (init) shows how to display a default text ("With initial value"). Note that the default text would have been sent to the payload if it had not been deleted before submitting the form.
The second text-field (placeholder) shows how a placeholder text ("Only Placeholder") is displayed in the UI. Please note the placeholder text is not sent in the payload if no text has been entered in the field by the enduser.
The third text-field (noreq) shows how a user can interact with a non-required field. Even if the field is empty (only a placeholder text is present but does not count as a value), it does not prevent the enduser from submitting the form.
The fourth text-field (req) shows the behaviour of the unique required field of the form, which cannot be submitted in case it is not filled; an error is displayed under the field in case the user submits the form with this empty field.
The fifth text-field (regex) shows the behaviour of the field when a regex pattern is entered. You can note that the pattern-error-message is displayed under the field if the input does not follow the pattern required by the bot developer.
The sixth text-field (min) shows how to force users to enter an input with a minimum number of characters, and how an error message is displayed under the field if the input does not respect the minlength required.
The seventh text-field (label) shows how a label text ("My Label") is displayed.
The eighth text-field (tooltip) shows how a title text ("My Tooltip/n With a second line") is inserted in the UI under the (i) icon, and how the text entered in the title parameter is displayed when the enduser clicks on the icon.
Please note that several text-fields are aligned next to another. Note also that this behaviour is reactive to the screen size, the number of text-fields on the same line decreasing until one per row, as the screen gets smaller (see in the example below).
The textarea
element is a field for multi-line text input, allowing users to edit multiple lines of plain text. Text areas are useful to collect or edit runs of text like messages, opinions, reviews, articles, etc.
The text field must be a self-closing tag or have no children.
You can add a default text in your text area by including it between the <textarea></textarea>
tags. Note that unlike the placeholder
text, the default text will be sent with the form if not edited by the user. Refer to Examples for more information.
The following example shows two textareas being used as follows:
The first textarea (id1) shows how to display a default text ("With initial value"). Note that the default text would have been sent to the payload if it had not been deleted before submitting the form.
The second text-field (req) shows how a placeholder text ("Required, with a placeholder, a regex, a label, and a tooltip") is displayed in the UI. Please note the placeholder text is not sent in the payload if no text has been entered in the field by the enduser. It shows as well the behaviour of a required textarea in a form, which cannot be submitted in case it is not filled; an error is displayed under the textarea in case the user submits the form with this empty field. The textarea presents how a label text ("My Label") as well as a title text ("My Tooltip/n With a second line") are displayed in the UI. Finally, it shows how users can interact with a regex pattern which does not allow the form to be submitted if the input does not follow the pattern required by the bot developer.
The Person Selector is an element used for finding and selecting people. Person Selectors are used in many places within Symphony, so you should be familiar with how they work.
When a user types the person's name, a drop-down will be displayed with the results found for the data the user entered. The following example shows how three people with the same name (Vincent) have been found.
The Person Selector element supports multi-user selection which means that you can search for more than one person using the same selector.
The following examples show person selectors being used as follows:
The first person-selector (placeholder) shows how a placeholder text ("My Placeholder") is displayed in the UI. Please note the placeholder text is not sent in the payload if no option from the dropdown menu has been selected by the enduser.
The second person-selector (placeholder) shows how the bot can introduce a default value in the person selector. It shows as well how a Symphony user can interact with the clear all functionality.
The third person-selector (noreq) shows how a user can interact with a non-required field. Even if nobody is selected by the user, it does not prevent the enduser from submitting the form.
The fourth person-selector (req) shows the behaviour of the unique required field of the form, which cannot be submitted in case nobody from the person selector is selected by the user; an error is displayed under the field in case the user submits the form with this empty field. An auto-filtering behaviour allows the user to see less options as he digits some input. Also, it shows how Symphony users can remove some of the selected users with the cross associated to that specific user.
The fifth person-selector (label) shows how a label text ("My Label") is displayed.
The sixth person-selector (tooltip) shows how a title text ("My Tooltip/n With a second line") is inserted in the UI under the (i) icon, and how the text entered in the title parameter is displayed when the enduser clicks on the icon.
The result returned by the datafeed for the selected users is an array of user Ids, which is an array of long
.
The following image shows three different tables. The first table shows the use of checkboxes to select rows, positioned to the right side of the table. The second example also shows checkboxes, but they are positioned to the left side. The last table shows buttons positioned to the right.
In the JSON data, you can configure the type
of the Element that will be added to the Table Select and its position
:
In this example, the table type
is set as button
and the position
is set as left
.
The table can be generated without header or footer.
If the table type
is equal to "checkbox", then checkboxes will be added to the rows of the table.
A checkbox can also be added in the header. Please note, clicking the checkbox in the header will not check all rows' checkboxes.
If the table type
is equal to "button", then buttons will be added to select a specific row of the table.
Do you need users to pick a timezone as part of an interactive flow? With the Timezone Picker, Symphony users can easily select a timezone from a selection of possible values. It recognises both cities and countries so the user can quickly filter and find the right timezone.
The Timezone Picker is represented by the <timezone-picker> tag, as you can see in the examples at the bottom of the page.
You can see below the designs of the timezone picker.
For the purpose of accessibility, Symphony users can interact with the timezone picker via their keyboard:
First, using "Tab" to enter the component
Using "Enter", "Arrow-up", or "Arrow-down" to open the timezone list
Using "Arrow-up" and "Arrow-down" to navigate in the list
Using any key to erase the field and write in it
Using "Enter" to set the selected timezone and close the list
Using "Escape" to keep pre-selected value and close the list
Finally using "Tab" to exit the component
The max length of any timezone picker attribute is 256 characters except disabled-timezone
attribute which max length is set to 1024 characters.
All timezone values are displayed in English only.
You can add a default timezone in your text field by including it in the value
parameter. Please note that unlike the placeholder
text, the default timezone will be sent in the formReply when the form is submitted if not edited by the user.
Please note that if the default timezone (value
attribute) matches a value from the disabled-timezone array, then the value is left empty.
The timezone-picker will be supported on the following versions of clients:
20.14 for Client 2.0
20.13 for Client 1.5 (a beta version is released in Client 20.12 for Client 1.5)
The following examples show the timezone picker being used as follows:
The first timezone-picker (init) shows how to display an empty timezone-picker ("With empty default value"). Note in the messageML sent the value was enforced to empty with value=""
.
The second timezone-picker (specific_value) shows how to display, by default, a value specifically chosen by the developer. Note that the default value would have been sent to the payload if it had not been deleted before submitting the form. You can see how users can remove a pre-selected value from the timezone-picker, thanks to the cross on the right side of the input.
The third timezone-picker (default_value) shows how to display, by default, the user browser timezone value. Note that the default value is sent to the payload when submitting the form.
The fourth timezone-picker (placeholder) shows how a placeholder text ("Only Placeholder") is displayed in the timezone-picker. Please note the placeholder text is not sent in the payload if no value has been chosen by the enduser.
The fifth timezone-picker (label) shows how a label text ("My Label") is displayed.
The sixth timezone-picker (tooltip) shows how a title text ("My Tooltip/n With a second line") is inserted in the UI under the (i) icon, and how the text entered in the title parameter is displayed when the enduser clicks on the icon.
The seventh timezone-picker (noreq) shows how a user can interact with a non-required field. Even if the field is empty (only a placeholder text is present but does not count as a value), it does not prevent the user from submitting the form.
The eighth timezone-picker (req) shows the behavior of the unique required field of the form, which cannot be submitted in case it is not filled; an error is displayed under the field in case the user submits the form with this field as empty.
The ninth timezone-picker (disabled) shows how users interact with disabled values from the timezone-picker.
Please note in the examples above that the values are displayed to users according to the following rules:
The country name only is displayed if it has a single timezone (see Israel)
A list of cities are displayed under their country title if this country has multiple timezones (see United States of America)
Buttons are the Symphony elements responsible for submitting a form to the bot. As a result, all Symphony form elements are required to have at least one button where `type=action
`. When an end-user clicks this button, the form and the value of each of the elements inside will be submitted to the bot via the datafeed and presented as a JSON payload.
In addition, some forms can contain reset buttons. These buttons are used to reset a form back to its original state.
Buttons support six different styles: primary
, primary-link
, secondary
, tertiary
, destructive
and destructive-link
. Each of those has different colors to suit different actions (to convey meaning). Use the class
attribute to specify the style.
Primary: use the Primary button when there is a clear primary action on a message. You can use it for the submit button, for example.
Secondary: use the Secondary button when there are multiple actions of the same importance or some actions with less importance than a single primary action.
Destructive: use the Destructive button when an action results in the removal of an item or if it can result in a potentially serious negative consequence.
Primary-link, Tertiary and Destructive-link: These styles are variations respectively of the Primary, Secondary and Destructive buttons but without borders. They are low prominence options that can be used alongside a Primary or as standalone buttons with the ability to read more information.
If class
is not defined, the action button assumes the primary
class by default. Action buttons should be used for affirmation or confirmation actions.
Reset buttons have the secondary
class set by default. Reset buttons should be used when the content of the fields need to return to their original state.
Do you need users to pick a time as part of an interactive flow? With the Time Picker element, Symphony users can easily select a time from a list of possible values.
The Time Picker offers to Symphony users the possibility to enter the time in two different ways:
Populating the field directly with the keyboard, following the appropriate format and the list of possible values (see attributes strict, min, max, and disabled-time)
Clicking on the wished time from the dropdown displayed
The Time Picker is represented by the <time-picker> tag, as you can see in the examples at the bottom of the page.
You can see below the designs of the time picker.
For the purpose of accessibility, Symphony users can interact with the time picker via their keyboard:
First, using "Tab" to enter the component
Using "Enter" or "Space" to open the dropdown when focus is on the icon, or just "Enter" when focus is on the input
Using "Arrow up" or "Arrow down" to navigate within the dropdown list and using "Enter" to select the preselected value
Either "Typing" or pressing "Tab" to go from the hours to the minutes, and then from the minutes to the seconds
Either "Deleting" or pressing "Shift+Tab" to go back from the seconds to the minutes, and then from the minutes to the hours
Finally using "Tab" to exit the component
The max length of any time picker attribute is 256 except disabled-time
attribute which max length is set to 1024 characters.
If the format entered by the user is not correct or if the time entered is a disabled time, then an error message is displayed to the user. Note that it is not possible for the user to submit the form with an invalid format or disabled time.
You can add a default time in your text field by including it in the value
parameter. Please note that unlike the placeholder
text, the default time will be sent back to the Bot in the user reply when the form is submitted if it is not edited by the user.
The time-picker will be supported on the following versions of clients:
20.14 for Client 2.0
20.13 for Client 1.5
The following examples show the time picker being used as follows:
The first time-picker (init) shows how to display a default time, as an initial value is entered as parameter. Note that the default value is present in the user reply as it has not been deleted before submitting the form.
The second time-picker (placeholder) shows how a placeholder is displayed in the UI. Please note that any text is accepted as input. However, if you compare with the next time-pickers present in the form, you can notice that a default placeholder is generated (with a hint of the correct format to accepted by the time-picker field) in case no placeholder is set.
The third time-picker (label) shows how a label ("My Label") is displayed.
The fourth time-picker (tooltip) shows how the title attribute ("My Tooltip/n With a second line") is displayed.
The fifth time-picker (noreq) shows how a user can interact with a non-required field. Even if the field is empty (only a placeholder text is present but does not count as a value), it does not prevent the user from submitting the form.
The sixth time-picker (req) shows the behaviour of the unique required field of the form, which cannot be submitted in case it is not filled.
The seventh time-picker (format) shows the way the placeholder evolves to adapt to a new format transmitted thanks to the format parameter. Also please note the accessible interaction with the time-picker via the keyboard.
The eighth time-picker (step) shows how to modify the step between 2 consecutive values in the list of times. Please note that a Symphony user can select any other existing time by populating the field with his keyboard.
The ninth time-picker (strict_step) shows how to enforce the user to choose among the non-disabled values displayed in the dropdown, thanks to the strict attribute.
The tenth time-picker (rules) shows how to interact with the following parameters: min, max, and disabled-time. Please note that a disabled time cannot be entered manually.
The Room Selector is an element that allows users to find and select both rooms or persons. It behaves the same way as the chat selector you see when you forward a message and select where the message should be forwarded.
When a user starts typing in the field, a list of conversations and people will be displayed for selection. Only the chats that the user has access to will be displayed.
The Room Selector element supports multi selection which means that you can search for more than one chat or person.
The Room Selector is not yet available on Symphony Mobile. Mobile will be supported in a future release.
Resulting payload when a user submitted the form afer having selected two chats and one user in the Room Selector.
More information on the new tags is available in the .
Text input fields are the most common elements in a form. Symphony provides two types of elements for text input fields: are for a single-line input; and is for multi-line input.
With Symphony v20.6, bot developers can use Regex to validate text fields
and text areas
using the pattern
and pattern-error-message
attributes.
For more information and examples, refer to .
The text field has a max number of 128 characters. For larger texts, use .
Text fields are grouped at a max of 4 per row, depending on the screen size. For more information, refer to .
With Symphony v20.6, bot developers can use Regex to validate text fields
and text areas
using the pattern
and pattern-error-message
attributes.
For more information and examples, refer to Regular .
The Table Select is not an Element itself but an example of what can be achieved by using Elements with the templates. This way, it is possible to build tables which contain a special column that allows users to select one or more rows, either with the or the Element.
The following example shows how to create a Table Select structure using the template and a JSON file.
Note that the template is being used to create the messageML that is rendering the Table.
When creating a MessageML using a template, you must send a JSON file with it.
For a list of all the available elements, refer to .
The possible values of the timezone-picker are restricted to the Canonical values of the .
The following example shows the use of the Reset and the Submit button when sending a text inserted in a .
Example of a room selector with both a room and a user pre-selected. Please note that pre-selected streamIds must follow
Bot Developer Kit for Java
Bot Developer Kit for Python
Attribute
Type
Required?
Description
name
String
Yes
Identifies the text field.
placeholder
String
Optional
Specifies a short hint that describes the expected value of the input field.
required
Boolean
Optional
If true
, it specifies that the input field must be filled out before submitting the form.
masked
Boolean
Optional
If true
, it creates a masked text field with hide/show options when its value is "true". For more information, refer to Masked Text Field.
maxlength
Integer
Optional
The maxlength
attribute allows you to specify a maximum number of characters that the user can input.
minlength
Integer
Optional
The minlength
attribute allows you to specify a minimum number of characters that the user can input.
pattern
String
Optional
Regex String to match for input validation. For more information, refer to Regular Expressions - Regex.
pattern-error-message
String
Optional but if pattern
is defined, the pattern-error-message
attribute is mandatory.
Error message returned to user if pattern
parameter matches user input
title
It accepts a simple text and \n
for line breaks
Optional
The description that will be displayed when clicking the tooltip icon located on top of the Text Field Element. Max length: 256 characters. Available from Symphony v20.8 and above.
label
String
Not required but it is recommended if title
is defined
Definition of the label that will be displayed on top of the Text Field Element. Available from Symphony v20.8 and above.
auto-submit
Boolean
Optional.
Default false.
When enabled, typing <enter
> key in the field will submit the form.
Initial release
2.55.9
Since first version
Since first version
Regex
20.6
Since first version
Since first version
Label
20.7
Since first version
Since first version
Tooltip (title)
20.7
Since first version
Since first version
auto-submit
23.11
23.12
Not supported yet.
Attribute
Type
Required?
Description
name
String
Yes
Identifies the text area.
placeholder
String
No
Specifies a short hint that describes the expected value of the text area.
required
Boolean
No
If true
, it specifies that the text area must be filled out before submitting the form. Accepted values; true
and false
.
pattern
String
No
Regex String to match for input validation
pattern-error-message
String
No
Error message returned to user if pattern
parameter matches user input
title
It accepts a simple text and \n
for line breaks
No
The description that will be displayed when clicking the tooltip icon located on top of the Masked Text Field Element. Max length: 256 characters. Available from Symphony v20.8 and above.
label
String
Not required but it is recommended if title
is defined
Definition of the label that will be displayed on top of the Masked Text Field Element. Available from Symphony v20.8 and above.
rows
Number
No
Specify the number of rows (height) of the text area that will be displayed by default.
cols
Number
No
Specify the number of columns (width) of the text area that will be displayed by default.
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Backward client-compatibility behavior (e.g. external rooms)
Initial release
2.55.9
Since first version
Not working
Regex
20.6
Since first version
Regex validation not enforced but field can be submitted
Label
20.7
Since first version
Label displayed and form can still be submitted
Tooltip (title)
20.7
Since first version
Tooltip not displayed but form can still be submitted
row and col
23.11
24.1
Not supported.
Attribute
Type
Required?
Description
name
String
Yes
Identifies the person selector
placeholder
String
No
Specifies a short hint that describes the expected value of the input field.
required
Boolean
No
If true
, it specifies that the person selector must be filled out before submitting the form, which means that at least one person must be "selected" Accepted values; true
and false
.
title
It accepts a simple text and \n
for line breaks
No
The description that will be displayed when clicking the tooltip icon located on top of the Element. Max length: 256 characters.
label
String
Not required but it is recommended if title
is defined
Definition of the label that will be displayed on top of the Masked Text Field Element.
value
Array of
long
No
Default value that will be preselected in the person-selector when the user receive the form from the bot.
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Backward client-compatibility behavior (e.g. external rooms)
Initial release
2.55.9
Since first version
Not working
Label
20.7
Since first version
Label displayed and form can still be submitted
Tooltip (title)
20.7
Since first version
Tooltip not displayed but form can still be submitted
Value
20.12
21.7
For client 1.5, it displays the person-selector as if there was no default value
Attribute
Type
Required?
Description
type
String
Yes
The type
attribute determines if a table will display a special column with Buttons or Checkboxes within it. Note that a table can have only one of the two possible types
, being button or checkbox. For more information, see the Example below.
position
String
Yes
This attribute indicates how the buttons and checkboxes must be aligned inside the column. Accepted values: left
or right
. For more information, refer to the JSON example below.
Attribute
Type
Required?
Description
name
String
Yes
Identifies the timezone picker.
value
String
Restricted to the tz database timezones + empty string
No
Timezone displayed by default when the user receives or resets the form.
Please note that if it is not defined by the developer, the value will be based on the user's browser setting. You need to enter value="" to enforce the timezone to be empty.
title
String (accepting \n
for line break)
No
The description that will be displayed when clicking the tooltip icon located on top of the timezone picker Element.
label
String
Not required but it is recommended if title
is defined
Definition of the label that will be displayed on top of the timezone picker Element.
required
Boolean (true or false)
No
If true
, it specifies that a timezone must be picked by the user before submitting the form.
placeholder
String
No
Specifies a short hint in the timezone picker input. Accepts any string value but is overridden by the value if a value is defined: it will show only if value attribute is enforced to empty.
Please note that a default placeholder is displayed as: "Choose a timezone".
disabled-timezone
Array of Strings
No
Items can be disabled from the timezone picker list so that the user cannot pick them. See the examples below for more details.
Important: single quote '
should wrap the array of strings
Please note that if you want to propose only a list of a few timezones to the users, then a simple dropdown menu might be more adequate.
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Backward client-compatibility behavior (e.g. external rooms)
Beta version
20.12
Not supported
Initial release
20.12
20.14
Not supported (except for 20.12 client 1.5)
Attribute
Type
Required?
Description
name
String
Yes
Identifies the clicked button.
type
String
No
If type
is not specified, the default value will be type=”action”
Indicates whether the button is an action
button or a reset
button. When clicked, the action button sends the form information to the datafeed. On the other hand, the reset button resets the form-data to its initial values.
Accepted values: action
and reset
.
class
String
No
Toggle between new palette of colors: primary
, secondary
, destructive
, primary-link
, tertiary
and destructive-link
.
icon
String
No
Adds an icon before the button name. Only icons from our UI Toolkit list are supported. Each icon is identified by its name.
Initial release
2.55.9
Since first version
Since first version
Reset features behaviour is to show back the initial value of the form
2.55.9
Since first version
Since first version
New styles:
• New designs for the buttons
• Styles primary destructive
and secondary destructive
are deprecated
• Styles tertiary
and destructive
are introduced
20.6
Since first version
Since first version
New styles: primary-link
and destructive-link
Support for icons.
23.11
23.12
Supported
Attribute
Type
Required?
Description
name
String
Yes
Identifies the time picker.
value
String "HH:mm:ss"
No
Time with ISO_8601 format to be displayed by default by the time picker when rendered for the Symphony user. Overrides the placeholder value.
title
String (accepting \n
for line break)
No
Description that will be displayed when clicking the tooltip icon located on top of the time picker Element.
label
String
Not required but it is recommended if title
is defined
The text of the label that will be displayed above the time picker field.
required
Boolean (true or false)
No
If true
, it specifies that a time must be picked before submitting the form.
placeholder
String
No
Specifies a short hint that describes the expected format of the time picker field.
If the attributevalue
is entered, the placeholder will not be displayed.
If the placeholder is not set, the accepted time format will be displayed by default.
Note: We recommend to use the default placeholder. It is better to rely on the label if context is needed, or the title if instructions are needed.
min
String "HH:mm:ss"
No
Specifies the earliest acceptable time with ISO_8601 format.
Note: Times earlier than the min are not displayed in the time picker.
max
String "HH:mm:ss"
No
Specifies the latest acceptable time with ISO_8601 format.
Note: Times later than the max are not displayed in the time picker.
disabled-time
Json array (in a String format)
No
Times or ranges of times that cannot be selected by the Symphony user.
Maximum length of 1024 characters.
There are 2 different patterns:
1. range of times: {"from": "HH:mm:ss", "to": "HH:mm:ss"},
2. specific time: {"time": "HH:mm:ss"}.
Important: single quote '
should wrap the xml parameter. See the examples below for more details.
Note: Disabled times are displayed as disabled values in the time picker.
format
String
No
Format in which the time will be displayed to or should be entered by the Symphony user. Only accepts the letters 'h', 'H', 'm', 's', and 'a' as well as ':' or space as separator. • 'h' (for 12-hour format) and 'H' (for 24-hour format) define the hour. You can use either 1 ('h'; 'H') or 2 ('hh'; 'HH') to define the minimum number of digits displayed (i.e. corresponding to "3" or "03" for the third hour) • 'm' defines the minutes. Similarly to the hours, you can use one or two ('m' or 'mm') • 's' defines the seconds. Similarly to the hours and minutes, you can use one or two ('s' or 'ss')
• 'a' (usually placed after a space) allows to display 'AM' for morning times and 'PM' for afternoon times Note 1: We recommend the use of the default value as much as possible. Note 2: The format only impacts what the end user will see. It does not affect how you have to specify the value, min, max, disabled-time, or the format of the user reply.
step
Number
No
Default is 900 (corresponding to15 min)
The stepping interval (in seconds) for the times displayed in the dropdown menu.
Min value: 600 (corresponding to 10 min)
Max value: 43 200 (corresponding to half a day)
strict
Boolean
No
Default is false
Enforce that the user cannot select a time that is not in the proposed list (e.g. we want him to only select an hour, he can’t decide to set the field to 9:15, even with the keyboard).
Please note that in addition to this, the value picked will be validated against attributes min, max, and disabled-time.
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Backward client-compatibility behavior (e.g. external rooms)
Initial release
20.12
20.14
Not working
Attribute
Type
Required?
Description
name
String
Yes
Identifies the room selector.
placeholder
String
No
Specifies a short hint that describes the expected value of the field.
required
Boolean
No
If true
, at least one chat or person must be selected to be able to submit the form.
label
String
No
Label displayed on top of the Room Selector.
value
Array of
string
No
Default values that will be preselected in the Room Selector. The array can contain both streamIds, as well as userIds. Please note that if the user does not have access to these users or conversations, they will not be displayed.
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Backward client-compatibility behavior (e.g. external rooms)
Feature introduced
23.11
C2 24.4 (April 2024)
Not available yet. Room Selector is not displayed.
Elements input validation. Available from Symphony v20.6 and above
Input validation is your first line of defense when creating a secure application.
Symphony Elements supports input validation using regular expressions (regex) for the text field
and text area
elements.
Regular expressions can cause performance issues in the Client if the validation of the regular expression is very complex. Poorly designed regular expressions can even cause denial of service (ReDoS). Please verify that your regular expressions are safe, using the following service: https://redos-checker.surge.sh/.
For more information, refer to Regular expression Denial of Service - ReDoS.
The following code snippets assumes that both <text-field>
and <textarea>
elements are placed within the right messageML context : <messageML>
+ <form>
including an action <button>
.
With Actions, MessageML allows your bots to send messages containing components that have UI extensibility capabilities, such as opening another chatroom or start a conversation with a specific user for your Symphony users.
These functionalities are gathered in the subpages you can find below.
The UI Actions are represented by the <ui-action>
tag, that can be associated with different attributes, such as trigger
, action
, or other more specific attributes.
Please note that UI-Actions are not rendered on mobile
Entities (also called Structured Objects) are rich, inline, interactive components for Symphony messages that allow you to embed information that is more complex than simple text.
To create a message containing a Structured object, construct the message content using MessageML with either a <div>
or a <span>
element containing the following attributes:
class="entity"
: specifies that the message contains a corresponding structured object.
data-entity-id
: the unique ID of the corresponding structure object.
This MessageML markup is then passed into the endpoint via the message
parameter. The following examples show the use of the attributes in <div>
and <span>
respectively:
The examples above reference an entity object called entityIdentifier
. The JSON corresponding to this object is passed to the create message endpoint via the data
parameter. For example:
Please continue below in the subpages if you want to learn more about Structured Objects
The checkbox is an interactive box that can be toggled by the user to indicate an affirmative or negative choice.
When clicked, a checkmark (✓) appears inside the box, to indicate an affirmative choice (yes). When clicked again, the checkmark disappears, indicating a negative choice (no).
Checkboxes are used to let a user select one or more options from a limited number of choices. Frequently, a set of checkboxes represents a single question which the user can answer by selecting any number of possible answers.
Attribute
Type
Required?
Description
name
String
Yes
Identifies the checkbox.
value
String
No
The value
is the string that will be sent to the server. If the value is not specified, the string on will be sent by default.
checked
Boolean
No
If true
, it specifies that the <checkbox>
element should be pre-selected (checked) when the page loads. Accepted values: true
and false
.
The text node of the MessageML will be converted to the <label>
tag. This will preserve the formatting tags <i>
and <b>
, if present.
A form can have a maximum of 50 checkboxes within it. Note: The limit in previous versions was set to 20, so this limit may still apply when sending messages to customers with an earlier version of Symphony (before 20.10).
Once selected, checkboxes can be deselected by clicking them.
Click the reset button to return the checkboxes to their original status (checked or unchecked).
If a checkbox is sent without at least one checked option, it will not be displayed in the datafeed payload.
The following example shows checkboxes being used. It shows how developers can use the checked parameter with the value01 preselected when the form is sent. It also shows how users can select or unselect one or several checkboxes before submitting the form, as well as how to reset it to its initial values.
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Backward client-compatibility behavior (e.g. external rooms)
Initial release
2.55.9
Since first version
Not working
This section lists the Structured Objects available for use in messages.
Field
Required
Format
Description
title
Yes
String
The headline of the article
subTitle
No
String
The subtitle of the article
blurb
No
String
A summary of the article to display
date
No
Date of publication
publisher
No
String
Name of the publisher
author
No
String
Name of the author
thumbnail
No
URL (could be a data url)
Image to be displayed - 106x106px
id
Must provide either id
or href
, or both.
String
An identifier used by the application to deeplink to the article
href
Must provide either id
or href
, or both.
URL
URL to the article (opened in a new browser window)
Field
Required
Format
Description
type
Yes
String
The type of object. Must be set to org.symphonyoss.fin.security
.
id
Yes
Array of Objects
An array of one or more of the following objects: org.symphonyoss.fin.security.id.ticker org.symphonyoss.fin.security.id.isin org.symphonyoss.fin.security.id.cusip org.symphonyoss.fin.security.id.openfigi More information about these objects is provided below.
Field
Required
Format
Description
type
Yes
String
The type of object. Must be set to org.symphonyoss.fin.security.id.ticker
.
value
Yes
String
The name/ID of a ticker.
Field
Required
Format
Description
type
Yes
String
The type of object. Must be set to org.symphonyoss.fin.security.id.isin
.
value
Yes
String
The entity's ID.
Field
Required
Format
Description
type
Yes
String
The type of object. Must be set to org.symphonyoss.fin.security.id.cusip
.
value
Yes
String
The entity's ID.
Field
Required
Format
Description
type
Yes
String
The type of object. Must be set to org.symphonyoss.fin.security.id.openfigi
.
value
Yes
String
The entity's ID.
FDC3 action buttons allow chat bots to embed buttons in messages which, on click, will raise an intent.
Action buttons are only displayed to users who have a Desktop Integration Platform (DIP) set up, such as interop.io, Here Core or Connectifi.
Learn more about Desktop Interop and intents here.
MessageML (message
property)
Note: Please see above the default text that would be displayed to users who don't have a desktop integration platform set up.
EntityJSON part (data
property)
Display in Symphony:
Field
Required
Format
Description
type
Yes
String
The type of entity. Must be set to com.symphony.media.image
.
version
Yes
String
The version.
format
Yes
String
The data format. Must be set to image
.
url
Yes
String
The URL of the image.
Field
Required
Format
Description
type
Yes
String
The type of object. Must be set to com.symphony.user.mention
.
version
Yes
String
The object's version.
id
Yes
Array of objects
An array of one or more of the following objects: • com.symphony.user.userId More information about these objects is provided below.
Field
Required
Format
Description
type
Yes
String
The type of object. Must be set to com.symphony.user.userId
.
value
Yes
String
The ID of a user.
Field
Required
Format
Description
type
Yes
String
The type of object. Must be set to com.symphony.media.video
.
version
Yes
String
The version.
format
Yes
String
The video's format. Must be set to youtube
or vimeo
.
url
Yes
String
The URL of the video.
id
Yes
String
The unique ID of the video (can be extracted from the video URL).
Continue here to learn more about structured objects:
Chat bots can add buttons in chat messages that on click will open an existing chat room or start a chat with a specific user or group of users.
This can be useful to help Symphony users easily navigate from one chat to another, or to reference another discussion happening in a different chat.
The new chat can either replace the current chat, or open on the side instead.
OpenChat is represented in MessageML as a <ui-action action='open-im'>
tag and wraps a single button.
The <ui-action>
tag for OpenChat supports the following attributes:
Attribute
Type
Required?
Description
action
String
Yes
For OpenChat, always set to action='open-im'
user-ids
List of Integers
Exclusive with stream-id
List of ids of the users with whom the MIM must be opened when the Symphony user triggers the OpenChat functionality.
stream-id
String
Exclusive with user-ids
The id of the room/IM that must be opened when the Symphony user triggers the OpenChat functionality.
side-by-side
Boolean
No.
Default to true
If set to false
, the current chat will be replaced by the new chat to be opened when the Symphony user triggers the OpenChat functionality.
Symphony users can interact with the OpenChat functionality via their keyboard using either "Enter" of "Space" to trigger the click action, once the UI component used for the OpenChat (i.e. button element) is in focus. Symphony users can move from one Element to another using "Tab".
The following examples show the OpenChat functionality being used as follows:
The first open-im shows how the Symphony user may interact with a primary button that is empowered with the OpenChat functionality that opens a specific stream on the side of the current chat. Note that the trigger and side-by-side attributes are not mentioned, as they are not mandatory: they are then defined by their default behaviour.
The second open-im shows how the Symphony user may interact with a secondary button that is empowered with the OpenChat functionality that opens an IM with a specific user on the side of the current chat.
The third open-im shows how the Symphony user may interact with a tertiary button that is empowered with the OpenChat functionality that opens a chat with a specific list of users on the side of the current chat.
The fourth open-im shows how the Symphony user may interact with a destructive button that is empowered with the OpenChat functionality that opens a specific stream that replaces the current chat.
The max length of any ui-action attribute is 256 except user-ids
attribute which max length is set to a list 15 ids.
The OpenChat functionality is not supported in forms (see Symphony Elements).
Please note that the attribute type
of the button is not supported when wrapped by a <ui-action>
tag.
When using user-ids
attribute with 2 or more ids in the list, the functionality will always create a new chat between the considered users; it will not reopen an existing chat between these same users. This is a known limitation that might change in the future.
In case the side-by-side
attribute is set to true, when the Symphony user interacts with the OpenChat component, the IM/Chatroom appears on the side of current module in focus. In case the current module is not part of a workspace, then clicking on the button automatically creates workspace and focuses automatically to the desired IM/chat.
The OpenChat feature is supported in popped-out mode with the following behaviour:
Current chat/IM on focus is popped-out independently from a workspace:
In case the side-by-side attribute is set to false, the popped-out module is replaced by the desired one
In case the side-by-side attribute is set to true, the popped-out module remains in place and the desired one opens in the main Symphony client
Current chat/IM on focus is part of a popped-out workspace (Capital Markets view only): same behaviour as if the workspace was not popped out: if the specific chat to be opened is not part of the workspace, it is automatically added, and then opened either on the side or in the place of the previous focused chat depending on the side-by-side attribute.
When opening a specific chatroom where the user is not part of, then:
If it is a private room, a modal dialog will appear informing the user he cannot perform the action
If it is a public room, then the user is automatically added to the room and the considered stream is opened
The side-by-side attribute set to false is meaningful only for users in Client 2.0
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Client 1.5 release
Backward client-compatibility behavior (e.g. external rooms)
Initial release
20.12.2
21.7
20.13
Not working
Do you need users to pick a date as part of an interactive flow? With the Date Picker element, Symphony users can easily select a date from a calendar-style UI component.
The Date Picker offers to Symphony users the possibility to enter the date in two different ways:
Populating the field directly with the keyboard, following the appropriate format
Clicking on the wished day from the calendar UI in a monthly display
The Date Picker is represented by the <date-picker> tag, as you can see in the examples at the bottom of the page.
You can see below the designs of the date picker. The main points to be highlighted are the following:
A dot is displayed under the today date
A disabled date is displayed in grey and stroke through
An highlighted date is displayed in blue
For a list of all the available elements, refer to Elements.
Attribute
Type
Required?
Description
name
String
Yes
Identifies the date picker.
value
String "yyy-MM-dd"
No
Date with ISO_8601 format to be displayed by default by the date picker when rendered for the Symphony user. Overrides the placeholder value.
title
String (accepting \n
for line break)
No
The description that will be displayed when clicking the tooltip icon located on top of the date picker Element.
label
String
Not required but it is recommend if title
is defined
Definition of the label that will be displayed on top of the date picker Element.
required
Boolean (true or false)
No
If true
, it specifies that a date must be picked before submitting the form.
placeholder
String
No
Specifies a short hint that describes the expected format of the date picker. Accepts any string value but is overridden by the value if a value is entered. If not set, the accepted format will be displayed. Note: be careful as adding a placeholder might bring confusion to the end user. Therefore we recommend to use the default one as much as possible, to use the label if context is needed, or the title if instructions are needed.
min
String "yyy-MM-dd"
No
Specifies the earliest acceptable date with ISO_8601 format.
max
String "yyy-MM-dd"
No
Specifies the latest acceptable date with ISO_8601 format.
disabled-date
Json array (in a String format)
No
Dates or ranges of dates that cannot be selected by the Symphony user.
Maximum length of 1024 characters.
There are 3 different patterns:
1. range of date: {"from": "yyyy-MM-dd", "to": "yyyy-MM-dd"},
2. date: {"day": "yyyy-MM-dd"},
3. recurring day in the week: {"daysOfWeek": [0, 1, 6]},
Important: single quote '
should wrap the xml parameter. See the examples below for more details.
Note: for the daysOfWeek
: 0 always corresponds to Sunday, 6 to Saturday.
See the examples below for more details.
highlighted-date
Json array (in a String format)
No
Dates or ranges of dates that are highlighted to the Symphony user.
Maximum length of 1024 characters.
There are 3 different patterns:
1. range of date: {"from": "yyyy-MM-dd", "to": "yyyy-MM-dd"},
2. date: {"day": "yyyy-MM-dd"},
3. recurring day in the week: {"daysOfWeek": [0, 1, 6]},
Important: single quote '
should wrap the xml parameter. See the examples below for more details.
Note: for the daysOfWeek
: 0 always corresponds to Sunday, 6 to Saturday.
See the examples below for more details.
format
String
No
Format in which the date will be displayed to or should be entered by the Symphony user. Only accepts the letters 'y', 'M', and 'd'. • 'd' defines the day of the month. It can be either 'd' or 'dd' to define the minimum of digits displayed (i.e. corresponding to "3" or "03" for the third day) • 'M' defines the month. Use one or two ('M' or 'MM') for the numerical month; three 'MMM' for the abbreviation (i.e. "Sep"); four 'MMMM' for the full name (i.e. "September"); or five 'MMMMM' for the narrow name (i.e. "S") • 'y' defines the year. Use n of it to define n minimum of digits displayed (i.e. use 'yyyyy' to display the year 2020 as 02020) Note: be careful as many combinations may be possible, may have a weird display for the Symphony user, and are not restricted in the messageML. Therefore, we strongly recommend to use the default value as much as possible. Note 2: The format only impacts what the end user will see. It does not affect how you have to specify the value, disabled-date, highlighted-date, or the format of the response.
For the purpose of accessibility, Symphony users can interact with the calendar UI with their keyboard:
Using "Enter" to open/close the calendar UI
Then using "Tab" to navigate in the different functionalities of the calendar: day selection; then "today" button; then "previous year" button; then "previous month" button; then "next month" button; then "next year" button; and finally going back to the day selection panel
Once in the day selection area, using the arrows from the keyboard allows navigating in the month to select the right day
Finally, using "Enter" allows to close the panel with the day selected or to trigger one of the functionalities selected (buttons)
In the day selection area, some more keyboard controls are available in the date picker:
"Home" key: Moves focus to the first day (e.g. Sunday) of the current week
"End" key: Moves focus to the last day (e.g. Saturday) of the current week
"Page Up" key: Moves to the previous month, with focus on the same day of the same week (or either previous or next week if it does not exist)
"Page Down" key: Moves to the next month, with focus on the same day of the same week (or either previous or next week if it does not exist)
"Shift" + "Page Up" key: Moves to the previous year, with focus on the same day of the same week (or either previous or next week if it does not exist)
"Shift" + "Page Down" key: Moves to the next year, with focus on the same day of the same week (or either previous or next week if it does not exist)
The max length of any date picker attribute is 256 except disabled-date
and highlighted-date
attributes which max length is set to 1024 characters.
The date picker is displayed differently to the end user depending on the language chosen in the settings specific of each user. Please note that if a format is specified, then the format overrides the default format chosen based on the settings.
English: the calendar-UI is in English, the weeks start on Sunday, the default format is MM-dd-yyyy, and the date picker accepts value to be entered with the keyboard in English only (i.e. August)
French: the calendar-UI is in French, the weeks start on Monday, the default format is dd/MM/yyyy, and the date picker accepts value to be entered with the keyboard in French only (i.e. Août)
Japanese: the calendar-UI is in Japanese, the weeks start on Monday, the default format is yyyy日MM月dd年, and the date picker accepts value to be entered with the keyboard in Japanese only
If the user enters the value of a valid day and month combination with his keyboard, then the current year is preselected. If he continues entering another year, then this last one will be selected.
If the format entered by the user is not correct or if the date entered is a disabled date, then an error message is displayed to the user. Note that it is not possible for the user to submit the form with an invalid format or disabled date.
You can add a default date in your text field by including it in the value
parameter. Please note that unlike the placeholder
text, the default date will be sent in the formReply when the form is submitted if not edited by the user.
Note that currently, if a date is entered in the disabled and highlighted parameters at the same time, it cannot be selected via the calendar UI first. However, it can be enabled if the user enters this particular date manually. This behaviour will be fixed in the next versions and the date will then be considered as disabled and displayed as such.
The following examples show the date picker being used as follows:
The first date-picker (init) shows how to display a default date, as an initial value is entered as parameter. Note that the default text will be sent to the payload given that it was not deleted before submitting the form.
The second date-picker (placeholder) shows how a placeholder is displayed in the UI. Please note that any text is accepted as input. However, if you compare with the next date-pickers present in the form, you can notice that a default placeholder is generated (with a hint of the correct format to accepted by the date-picker field) in case no placeholder is mentioned in parameter.
The third date-picker (label) shows how a label is displayed. In the GIF below, it shows also the interaction with the form when the user directly writes an input with an incorrect format: an error message is displayed and the form cannot be sent.
The fourth date-picker (tooltip) shows how the title attribute is displayed. Note that when a user enters the month and the date in the correct format, the current year is automatically preselected. It can be modified if the user continues writing the year in the input field.
The fifth date-picker (req) shows the behaviour of the unique required field of the form, which cannot be submitted in case it is not filled. Also note the use of the "today" button.
The sixth date-picker (format) shows the way the placeholder evolves to adapt to a new format transmitted thanks to the format parameter. Also please note the accessible interaction with the date-picker via the keyboard.
The seventh date-picker (rules) shows how to interact with the following parameters: min, max, disabled, and highlighted. Please note that a disabled date cannot be entered manually.
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Backward client-compatibility behavior (e.g. external rooms)
Initial release
20.12
20.12
Not working: date-picker is not displayed but the form can be sent
Chat bots can embed dialog buttons in chat messages. When the dialog button is clicked, a dialog opens and displays the content that has been predefined. This can be useful to make long messages much more concise, or progressively disclose a form only if it is relevant to the user for example.
The Dialog component is structured in two sections:
<ui-action action='open-dialog'>
tag. This tag defines where and how the button will be displayed, and it references the dialog that will open using the target-id
attribute;
<dialog>
tag that defines the content of the dialog that will be displayed when the button is clicked.
The <ui-action>
tag wraps a single <button>
children tag. The button tag supports the usual styles (primary, secondary, tertiary). Examples are available at the bottom of this page.
The <ui-action>
tag supports the following attributes:
Attribute
Type
Required?
Description
action
String
Yes
For Dialogs, always set to action='open-dialog'.
target-id
String
Yes
Id of the dialog that must be opened when user will trigger this ui-action.
See the id
attribute in the <dialog> tag.
Each <dialog>
is split in three areas (children tags):
<title>
- mandatory: specifies the title of the dialog and is always displayed at the top of it in a fixed and non scrollable position.
<body>
- mandatory: specifies the content of the dialog that is displayed in the middle of it and can be scrollable when the content is too big to be contained in the view height.
<footer>
- optional: specifies the footer of the dialog and is always displayed at the bottom in a fixed and non scrollable position.
The <dialog>
tag supports the following attributes
Attribute
Type
Required?
Description
id
String
Yes
Id of the dialog that will be triggered thanks to the ui-action it is associated to. See target-id
attribute in <ui-action> tag.
width
String
No.
Default to medium
Specifies the width of the dialog.
NB: values can be: small, medium, large, or full-width.
The following gif shows an example that contains two dialogs:
The first dialog is embedded in a form. You will notice that the button used to trigger the dialog is positioned where the <ui-action>
tag is coded.
As soon as the associated <dialog>
is at its same nesting level, it can be placed anywhere before or after the ui-action.
You can also notice that users can interact with the form as well as submit it.
However, the dialog only contains text as it cannot contain any interactive Form Elements. Please note that a cancel button can be included in the dialog as it is not considered as an interactive Form Element: you can use it to offer another option for the user to close the dialog.
Please note the scrolling behaviour of the body whereas both title and footer are placed in a fixed position.
The second dialog embeds a form instead.
After having started to fill-in the form, if the user closes the dialog and opens it back, the values of the fields will not reset to their original values, as long as the user does not refresh the page.
Also, when submitting a form that is in a dialog, the dialog will automatically close after a certain delay. If the user opens back the dialog without refreshing the page, then the values of the form and its state are persisted.
Below are simpler examples with and without the use of forms.
The max length of any ui-action or dialog attribute is 256.
Please note that the attribute type
of the button is not supported when wrapped by a <ui-action>
tag.
The dialog and ui-action tags must be both present in the same message, at the same nesting level, and they must share the same and unique id / target-id. Please note it is possible to have several dialogs in the same message as soon as each dialog has a different id.
The dialog feature is supported in popped-out mode. The dialog will open in the popped-out window.
The dialog functionality supports Interactive Elements Forms in the following way:
Dialogs can be contained inside forms. However, if contained in a form, the dialog cannot contain any interactive Element (such as button, text-area, etc.)
Dialogs can contain a form. The <form>
tag should wrap the entire content of the dialog, including the <title>
,<body>
and <footer>
tags, as you can see in the examples below. This is useful when you want the submit button not to be hidden and always appear in the footer of the dialog whereas the rest of the form content is contained in the scrollable body area.
Please also note that users can close the dialog thanks to the cross (x) displayed at the top-right corner of it, as well as with a new type of button that has been created for that purpose: <button type="cancel">. You can also specify the class attribute of the button which is by default set to "tertiary" for this new button.
A dialog cannot be embedded in another dialog.
Even if this possible, please avoid embedding more than one form in a single dialog.
Main features introduced
Agent needed to parse message sent by the bot
Client 2.0 release
Client 1.5 release
Backward client-compatibility behavior (e.g. external rooms)
Initial release
20.12.2
21.10
Not working
Not working - entire message is not rendered
Symphony has a number of best practices to ensure smooth bot development and a positive user experience.
It is common that Symphony chatrooms contain more than one bot listening for events. In order to preserve name space and ensure that there isn't overlap between bot workflows, we recommend that bots only respond to events that begin with an @mention:
By '@mentioning' your bot, it ensures that only the intended bot responds.
It is best practice to have bots listen for commands beginning with a '/'. This naming convention makes it clear to other users that you are trying to get a bot's attention. Additionally, it makes it clear to your bot that a user is instructing it to perform an actions and kick off its intended workflow.
It is also best practice to enforce command line style arguments for commands that require additional input or data:
It important that users understand exactly what your bot is capable of and what commands your bot understands. It is best practice for your bot to list its commands inside a help menu:
It is considered best practice that bot's only create and read from one datafeed. If your bot goes down, it should re-authenticate, and try to read from the previously created datafeed. If this fails then you should create a new datafeed, and begin reading from this new datafeed.
Creating and reading from multiple datafeeds concurrently can result in your bot processing duplicate messages and subsequently sending duplicate or out of order messages back to the user.
For large rooms, it is recommended that bots do not send messages at a higher rate than 2 messages/sec or every 500ms.
If bots are bulk adding users to rooms, it is recommended that bots add users at rate of 3 users / second.
In some rare cases, bots may receive duplicate messages from Symphony. In order to prevent duplicate processing, developers can implement logic to keep track of previous messages. It is recommended that bots store a list of unique messageIDs up to 15 minutes in the past. Upon each new message, bots should do a quick validation that new message is not received in the past and continue to process the message.
Many members of our community are building bots and code samples and publishing their source code. We collect these examples in this page. You can use any of them as is, or to either bootstrap your own idea or just for inspiration.
This bot provides users to retrieve and export all of their messages from Rooms and IMs.
This bot provides capabilities to manage room creation and population using Active Directory, .csv files or .eml files (email).
This bot tracks watchlists for multiple users and displays a table with clickable buttons to allow users to join discussion rooms.
This bot uses Symphony Elements to facilitate the creation of polls, firing of poll messages, ending polls and collation of results.
Building Extension Apps on Symphony Messaging is an easy and secure way to customize the Symphony Messaging experience. Take these simple steps in order to create and deploy your Extension App today!
Symphony Messaging Extension Apps leverage the Symphony Messaging Extension API in order to create innovative workflows and automations. Depending on your desired workflow, there are many different development avenues available for you and your development team. Understanding these different development options, APIs available and Extension App capabilities is key to creating a successful app and positive user experience. Learn more about the different types of Extension Apps and APIs here:
The easiest way to get started is by using the Symphony Messaging Generator to create a project that includes the App Development Kit (ADK) and the UI Toolkit. These tools allow you to interface with the Extension API and build interfaces that feel native in Symphony Messaging.
The next step is to add custom business logic to your app. Begin leveraging Symphony Messaging's Extension API and bring your extension apps to life:
Messages in PresentationML markup are enclosed in a <div>
tag with the following attributes:
data-format: must be set to PresentationML
.
data-version: specifies the markup version.
Shorthand tags translations:
The following table lists XHTML tags for MessageML on the left, and the corresponding PresentationML tags on the right:
Root <div>
tag
When retrieving a message using the API, the message is always encapsulated in a root <div>
tag, for easy parsing.
When you create a message using PresentationML, you must include the root <div>
tag.
The following is an example of content for a simple message using presentationML markup:
Structured Objects are rich, inline, interactive components of Symphony messages. Objects allow you to build innovative workflows that go beyond working with normal text or attached files.
Unlike normal message text, these objects are structured and do not need to be parsed to have business logic.
Unlike attachments, end-users can view and interact with objects directly from their Symphony client, without having to change context.
Structured Objects can be "injected" into Symphony by sending messages using Symphony's REST API.
To inject messages containing structured objects:
Your pod must be configured for Symphony's REST API, and you must have the Agent, the component used for handling encryption and decryption of messages and content, set up.
Your Agent must be version 1.51 or later.
You must have an X.509 identity certificate for your bot service user for REST API authentication, where the common name on the certificate matches your service user's username.
To build renderer applications for displaying your structured object:
You need to have an extension application created and enabled on your pod.
Structured Objects are placed into Symphony messages and have two components:
Object Data, a JSON object.
Any message in Symphony can contain zero or more Structured Objects.
A message will always contain message presentation, in MessageML v2 format, with the optional object presentation of the Structured Objects it may contain.
If the message contains any structured objects, it will contain JSON data with all object data of the structured objects it may contain.
The message
parameter, which contains the message presentation, with the object presentation for each Structured Object.
The data
parameter, which contains JSON data with the object presentation for each structured object.
Message presentation is represented in MessageML format. For example:HTML
To add an object to a message, include a div
or a span
tag with a unique data-entity-id
attribute:HTML
The data-entity-id
tag refers to a specific object in the JSON data, which needs to include:
The data type
.
The data version
of that type
. Both are needed to build renderer applications which can render this type
of that version
.
This data can be used by applications in the web client to provide a rich display or end-user interactivity. In case no specific renderer application is available, you must provide a default presentation in the div
or a span
tags.HTML
Renderer Applications leverage the Extension API to dynamically replace the presentation of a structured object. To create a renderer application:
To learn more about building Extension Applications that leverage structured objects, continue here:
Symphony extension apps are standalone web applications that are embedded within the Symphony user interface as iframes. These iframes interact with the Symphony container using the Symphony Extension API. Extension apps can be accessed by the end user in two ways:
From the left navigation under the Applications tab. Clicking on your app name brings it into view
Attached to contexts such as #hashtags, $cashtags, or user profiles. Clicking on your app button brings it into view along with the corresponding context.
Developing extension apps enable developers to enrich the Symphony experience and create custom workflows and automations on top of the Symphony platform.
The answer lies in Symphony's Extension API, which is a JavaScript library that consists of services and methods that allow applications to extend and interact with Symphony's user interface. By leveraging these services, developers can:
Add modules, or windows bringing your app's content into the Symphony canvas
Add entry points for your app, such as navigation items on the left hand nav or #hashtags and $cashtags.
Add interactive buttons to chat and user profile module headers
Enable users to share content from your app into Symphony chats
For a full overview of Symphony's Extension API continue here:
Before you begin your extension app development journey, it is important to consider the following when determining what type of app you build:
Before building your extension app, it's important that you identify the use cases that this app will serve. In other words, identify the ways in which this app will increase productivity, add meaningful color to your daily tasks, centralize information, reduce business pain points, and make working simpler for end users. To easily identify valuable use cases, ask yourself the following:
Are there numerous sources of information that I check daily that can be centralized inside of Symphony?
Are there any views or visualizations from third party apps which could be embedded into Symphony's UI?
Are there any tasks in my daily workflow that require manually sifting through large amounts of data?
Would it be helpful if this data could be accessed easily to Symphony users across my organization?
Are there any customized features on Symphony's UI that would increase my productivity and efficiency?
The type of extension app you build will depend on who is using and interacting with it. To identify your app's audience, ask yourself the following:
Are the users of my app internal or external counter-parties?
Are the users of my app front-office or back-office employees mostly?
Will my app be interacting with a technical audience or a business audience?
What languages does your audience speak?
The more you understand your app's audience, the more you can understand their business pain points and in turn develop a better user-experience and app solution.
Users can interact with extension applications in a number of different ways. Specifically, users can launch a standalone extension app from the left-hand nav, invoke the app by clicking on custom UI buttons on a users profile, launch the app by clicking on attached contexts such as #hashtags or $cashtags, and even allow authenticated apps to make actions on-behalf-of an authorized user. Before building your extension app, its important to identify the types of interactions between users and your app:
Many extension applications built on top of Symphony need to receive conversation or user data. For example, if you wanted to build an extension app that extends the Symphony UI to add buttons to the IM, profile, or chatroom modules, it is likely that you would need access to conversation or user data. In order to do so your application will need to perform app authentication. You can learn more about performing app authentication here:
Another common use case for extension applications is to extend various parts of the Symphony UI by adding buttons to IMs, chatrooms, or profile modules. In order to receive the conversation and user data associated with these modules, these extension apps must also perform app authentication. You can learn more about how to receive user and conversation data as well as adding buttons to Symphony modules here:
Another way extension applications can extend Symphony's UI is to override links associated with #hashtag (e.g. #symphony) and $cashtag (e.g. $GOOG) hover cards. By attaching your extension app to #hashtag or $cashtag contexts, you can show content in your app that is relevant to the context clicked by the user. You can learn more about how build extension apps that extend these #hashtag and $cashtag entities in the same guide above.
Extension apps can extend Symphony's UI by acting as a custom renderer for structured objects created by the REST API. Structured objects are rich, inline, and interactive components embedded in a Symphony message. These structured objects can be rendered and injected into Symphony by creating a custom renderer as a part of an extension application. You can learn more about how to create a custom renderer in order to render structured objects here:
Some extension apps contain a dedicated frontend that will be embedded within of Symphony. For example, it's possible that your extension app will need to present complex financial data, charts, or dashboards, to an end user. Obviously, this functionality is not provided by Symphony out of the box, but you can bring it into Symphony through an extension application. Symphony provides a UI toolkit containing a library of react components in order to help build complex frontend applications rapidly. This UI toolkit contains frontend components and styles to allow you get the best out of your workflow.
To learn more about leveraging the UI Toolkit to build complex frontend applications continue here:
NB: Please note that the stream-id is the exact Conversation ID that you can find in the UI, and not the URLSafe64 converted id (see more details in )
Along with any request made to the , bots should send header X-Trace-Id
(random alphanumeric string of 6 characters) for logging purposes.
Please note that the Symphony BDK for Java sets up your logger (Mapped Diagnostic Context) with this X-Trace-Id. This is especially useful for cross-applications debugging, assuming that the X-Trace-Id value is also present in your application logs. You can find more information about how to print the X-Trace-Id with specific technologies (like logback or log4j2) in the .
Messages sent using the API are translated into equivalent XHTML tags known as PresentationML, that can be rendered and processed by any HTML client. For example, when you create a message using , the message
field in the response contains the message in PresentationML format.
Since PresentationML was included as part of the MessageML format design, you can create messages by passing the message content in MessageML or in PresentationML. Note that PresentationML uses rather than shorthand tags. Therefore, when you send a message using , the message
parameter must contain the message content in PresentationML XHTML markup, and the data
parameter must contain the XML markup for the referenced in the message.
Structured Objects can be rendered richly using .
Object Presentation, in format.
You can create an object by invoking the endpoint. You need to include:
These parameters also support using with Structured Objects.
You can read objects using any of the endpoints designed to read messages, for example, the endpoint. This endpoint will let you read both the message presentation and object data fields.
As described in , messages with can be created using the shorthand tags or the full tags. When they are read, the message presentation always contain the full tags, which are a subset of HTML tags.
Create an
Your application needs to use the , which will allow you to:
Register your application as being able to render a specific type, using the .
Render the object itself, by implementing the .
If you need additional inspiration, checkout our for examples of what has been built by our robust developer community!
Property
Value
Language
Javascript (NodeJS)
Author
Symphony
License
MIT
Source Code
Property
Value
Language
JavaScript (Node)
Author
Symphony
License
MIT
Source Code
Property
Value
Language
Java
Author
Symphony
License
MIT
Source Code
Property
Value
Language
Java
Author
Symphony
Licence
MIT
Source Code
Shorthand tag in MessageML
PresentationML Translation
<messageML>
<div data-format=\"PresentationML\" data-version=\"2.0\">
<mention uid="123456789"/>
<span class="entity" data-entity-id="mention123">@Name</span>
<hash tag="hashtag"/>
<span class="entity" data-entity-id="keyword123">#hashtag</span>
<cash tag="ticker"/>
<span class="entity" data-entity-id="keyword456">$tag</span>
<chime />
<audio src="https://asset.symphony.com/symphony/audio/chime.mp3" autoplay="true" />
<card>
<div class="card barStyle" data-icon-src="url" data-accent-color="blue">
<div class="cardHeader">PresentationML</div>
<div class="cardBody">PresentationML</div> </div>
Tag
Description
Attributes
root <div>
The root element of a Symphony message, when read through the API.
• data-format
: must be set to "PresentationML"
• data-version
This section is meant as a reference for pod administrators. If you are a developer trying to load your extension app onto your company's pod, please seek assistance from your pod administrator or local IT helpdesk.
Visit the Admin and Compliance Portal of the respective pod either via Settings > Admin Portal in the web or desktop client or by visiting https://my-company-pod.symphony.com/?admin
From the left navigation, select App Management
Click on Add Custom App on the top right
If the app developer has provided a bundle.json
file, use the Import Application Bundle File button on the top right. If not, fill in the fields manually.
Ensure that the RSA public key and required app permissions are set correctly
Click Create
From the left navigation, select App Settings
Locate the newly-created entry and change the Global Status from Disabled to Enabled
If the app is intended for users to optionally self-install and uninstall from marketplace, change the Visibility from Hidden to Visible
If the app is intended for all users in the organisation to have installed, change the Installation from Manual to Automatic
Click Save at the bottom
Host the extension app on a TLS-enabled server together with the application manifest (bundle.json
file)
Visit https://my-company-pod.symphony.com/?bundle=https://localhost:4000/bundle.json
assuming your app is running on localhost on port 4000 and you have a bundle.json served on the root
Acknowledge the developer mode warning and proceed
To use the Client Extension API services, you must include the symphony-api.js JavaScript file in your application controller and views.
To style your app, you must include the symphony-style.css CSS file in your application views and add the class "symphony-external-app" to the <body>
tag of your app views.
Icons must be square, with a recommended size of 32x32.
The supported formats are SVG (recommended), PNG and JPG.
Transparency is supported, but please take into account that your icon should work both in the light theme and the dark theme.
The SYMPHONY.remote.hello()
method should be used to initialize the connection to the Client Extension API from your application controller and views.
Returns a promise that will be fulfilled when the introduction is complete. If there is a problem, the promise will be rejected. The promise returns an object containing the user's Symphony client theme name, font size, and any associated classes, including those for theme name or size, as well for condensed and contrast modes.
Returns
Type
Description
themeV2
Object
An object containing the user's Symphony client theme settings
locale
String
The language selected by the user in his settings. Possible values are "en-US" | "fr-FR" | "ja-JP".
You should style your application according to the user's theme by applying the theme and font size classes to the <body>
tag of any application modules.
If a user changes his theme, a themeChangeV2 event is fired from the ui service, which will pass a themeV2
object with the new values. You should use a service to listen to this event and update the classes on the application module <body>
.
You must register your application controller with Symphony and connect your application views using the SYMPHONY.application
methods. During this time, the Services that will be used by your application must be specified.
Register an application controller with the Symphony client. Additionally, subscribe the application to remote services and register local services that can be used remotely. Returns a promise that will be fulfilled when registration is complete.
This method must be called before the application can register or subscribe to any services.
Parameter
Type
Description
id
String
The id of your application. For partner apps, this is an alphanumeric string chosen by the partner. For custom enterprise apps, this is generated when creating the app in the Admin Portal.
servicesWanted
Array of Strings
A list of names of remote services that your application wants to subscribe to
servicesSent
Array of Strings
A list of names of local services your application wants to make available remotely (any implemented methods on this service will be made available remotely)
Returns
Type
Description
userReferenceId
String
A unique anonymized identifier for the user in context that will perpetuate until the user uninstalls the application
Connect an application view to an existing application that has been registered with Symphony. Additionally, subscribe the application to remote services and register local services that can be used remotely. Returns a promise that will be fulfilled when connection is complete.
Parameter
Type
Description
id
String
The id of your application. For partner apps, this is an alphanumeric string chosen by the partner. For custom enterprise apps, this is generated when creating the app in the Admin Portal.
servicesWanted
Array of Strings
A list of names of remote services that your application wants to subscribe to
servicesSent
Array of Strings
A list of names of local services your application wants to make available remotely (any implemented methods on this service will be made available remotely)
Returns
Type
Description
userReferenceId
String
A unique anonymized identifier for the user in context that will perpetuate until the user uninstalls the application
In order to leverage the services provided by the Client Extension API, you must first subscribe to them. To use your applications own services, you must register them via the Client Extension API. Extension apps can register and subscribe to local and remote services in the following ways:
Creates a new local service and register it using existing classes:
If you are developing an Object-Oriented Application, SYMPHONY.services.make()
allows you to use a class as a prototype for implementing service methods, allowing you to choose which methods of the class will be available in the service by specifying them in the implements
list.
This cannot be and this cannot be achieved by the SYMPHONY.services.register()
function, which is only recommended when creating small services since it does not require any class, and it only requires the serviceName
. If you use SYMPHONY.services.register()
for large services, you will have to call service.implement()
passing an object to it. As a result, all the methods will be public and the code might look unorganized.
Parameter
Description
name
The name of the service being created
context
The object instance, usually this
methods
The names of the methods of this object that will be available on the created service
makeEventHandlers
If true, the methods listen and fire will be added to this instance. Calling the listen or fire objects will do so on the service.
Creates a new local service and register it to be used by a specific application view or your application controller:
Parameter
Type
Description
serviceName
String
The name of the service to register
Finds a service - either local or remote - that has been registered and returns it. Returns false if the service does not exist.
In order to use a service, it must have been requested by your application during application.register()
or application.connect()
.
Parameter
Type
Description
serviceName
String
The name of the service to subscribe to
Takes an existing local service and makes it available remotely. Use this to make services available between multiple application views or between your application view and controller:
Parameter
Type
Description
serviceName
String
The name of the existing local service to make available remotely
Imports a remote service and makes it available locally. Returns a promise which resolves to a reference to a service. This service would now be available in your registry using SYMPHONY.services.subscribe:
Parameter
Type
Description
serviceName
String
The name of the remote service that you would like to access locally.
Both the Client Extensions API services and your application services use the same interface. The service interface consists of the following methods:
Create a method on a service and specify the implementation.
Parameter
Type
Description
methodName
String
The name of the method to create on your service
implementation
Function
The implementation of the method
Alternately, create several methods on a service at once by specifying an object consisting of multiple functions:
Parameters
Type
Description
implementations
Object
An object containing one or more functions to create on this service, where the keys are the names of the functions and the values are the implementations
Call a method on a service. Any extra parameters passed to this method will be sent as arguments to the service method:
Parameters
Type
Description
methodName
String
The name of the method to call on the service
Fire an event from a service. Any extra parameters passed to this method will be passed to the callbacks defined by services that listen to this event:
Parameters
Type
Description
eventName
String
The name of the event to fire
Subscribe a service to an event that is fired by another service (or itself). This method returns a handle that can later be used to remove the listener:
Parameters
Type
Description
eventName
String
The name of the event to listen for
callback
Function
The function that will be called when the event is fired
Unsubscribe a service from an event:
Parameters
Type
Description
eventName
String
The name of the event to unsubscribe from
handle
String
The handle returned by the call to listen
A module is a new window inside the Symphony client workspace, such as a chatroom or an instant message. Use the modules
service to create application-specific modules.
The following methods are available on the modules
service:
Show a new application module:
Parameter
Type
Description
id
String
A unique id for this module (must be unique across all modules of a given application)
Either title
or {title, icon}
String or Object
Either the title of the module as a string
or an object with the keys title
and icon
The value of title
is a string
The value of icon
is the url of a square SVG image (recommended), or the url of a square png/jpg image. Recommended size is 32x32.
serviceName
String
The name of a local service implemented by your application that will be invoked when a user action is performed relating to this module
iframe
String
The URL for the content of the module (must be an HTTPS URL)
options
Object
An object, which can contain:
canFloat
: if set to true, a menu item will be added to the More menu (found under the (…) on the module frame) that, when clicked, will pop the module out into its own browser window
parentModuleId
: if set to the ID of a module opened by an application, the specified module will not be closed when this module is shown
Hide an existing application module:
Parameter
Type
Description
id
String
The id of the module that should be hidden.
Change the title of an existing application module:
Parameters
Type
Description
id
String
The id of the module for which the title should be changed
Either title
or {title, icon}
String or Object
Either the title of the module as a string
or an object with the keys title
and icon
The value of title
is a string
The value of icon
is the url of a square SVG image (recommended), or the url of a square png/jpg image. Recommended size is 32x32.
Focus an existing application module:
Parameter
Type
Description
id
String
The id of the module to focus
Opens a link from your application in a new tab in the user's default browser. This method should be used to open links, rather than <a href="..." target="_blank">...</a>
.
Parameter
Type
Description
url
String
The URL to be opened
Reloads the content of the module at the new URL.
The Client Extensions API is designed for single-page applications. Use this method with multi-page applications to load new content when users navigate to another page:
Parameter
Type
Description
id
String
The unique identifier for the module. A module with this id must already exist.
url
String
The URL of the new iframe to load in the module.
Symphony extension applications are standalone web applications that are embedded within the Symphony user interface as iframes that interact with the Symphony container using the Client Extension API.
The Client Extension API is a JavaScript library that consists of services containing methods that allow developers to build apps that extend and interact with Symphony's user interface. Using these services, developers can:
Add modules, or windows, containing your app content to the Symphony client
Add entry points for your app, such as navigation items in Symphony's left sidebar or links on #hashtag and $cashtag hovercards
Add interactive buttons to chat and user profile module headers
Enable users to share content from your app into Symphony chats
Register custom renderers to richly display messages containing structured objects
Some of Symphony’s services will require you to implement your own services with methods to handle events. For example:
Handling a user click on your app’s left sidebar menu item by opening your default app module
Handling a user click on your app’s #hashtag or $cashtag hovercard link by opening an app module with a contextual search
Applications created with the Client Extension API run in iframes inside the Symphony client.
Symphony apps consists of:
The main application controller, a hidden iframe that uses the Client Extension API services to bootstrap your app, extending the Symphony user interface
In most cases, one or more application views, separate iframes that are rendered within Symphony modules
Applications can be built using any web development technology of your choice.
Services are used for communication between your app and the Symphony client. There are two types of services: remote and local.
Remote services are services that are shared:
The services of the Client Extension API are remote services whose methods can be invoked by your application controller and views
You can also implement remote services that can be shared between your application controllers and views
Local services are services are specific to either your controller or one of your views.
To learn more about the services and capabilities provided by the Extension API continue here:
Each application's metadata is represented by a manifest file called bundle.json
. You must create a manifest file for your application and submit it to either your pod administrator for internal apps, or Symphony for apps meant for the public marketplace.
To create and upload a file:
Create the manifest file that your application requires.
Upload the manifest file to Symphony: drag and drop the file into the application creation window of the administrative portal.
The following bundle file is only applied for Developer Mode. Note that it is an array of apps, allowing you to load multiple apps at once.
In the following examples, the bundle files are applied for the AC Portal. Note that they are different than the one applied for Developer Mode.
The table describes the required and additional optional fields.
Field
Description
Required
Type
appGroupId
Applied for version 1.53 onwards. The unique identifier for your app. The ID must not match any existing app IDs or user names.
It consists of alphanumeric characters [0-9,A-Z, a-z], underscores (_), and dashes.
Note: Do not use a colon (:) in this field, or you will receive a 401 Unauthorized
error.
Required
String
name
The name of your app, which will be displayed in the Symphony Market.
Required
String
description
The description of your app, which will be displayed in the Symphony Market.
Optional
String
publisher
The publisher of your app, which will be displayed in the Symphony Market.
Optional
String
loadUrl
The location of the app's controller, which will be displayed in a hidden iframe. Value must start with https://
.
Note: Do not specify this value for On Behalf Of (OBO) applications.
Required, except for OBO applications
URI
domain
The domain for your app, which should match the controller file URL.
Required
URI
iconUrl
An icon for your app (32x32 px), which will be displayed in the Symphony Market.
Optional
URI
notification
Fields required in order to receive webhook callback containing pod information (ie. pod, authentication and Agent URLs).
Optional
Object
url
URL which the pod will call to send pod information to the application
Optional
String
apiKey
Secret key used to validate the pod when calling the notification URL.
Optional
String
permissions
Optional
List
certificate
Optional
String
rsaKey
Optional
String
allowOrigins
The origin or origins that this application is allowed to receive cross-domain requests from.
Optional
URI
type
Applied for 1.52 and prior versions.This field should be set to sandbox
, which indicates that this is a standalone app, embedded within Symphony's client.
Required
String
id
Applied for 1.52 and prior versions. The unique identifier for your app.
Note: Do not use a colon (:) in this field, or you will receive a 401 Unauthorized
error.
Required
String
blurb
Applied for Developer Mode. Field for display in the Symphony Market and Admin Portal.
Optional
String
icon
Applied for Developer Mode. An icon for your app (32x32 px), which will be displayed in the Symphony Market.
Optional
String
The App Developer Kit (ADK) is the preferred tooling for web developers to get started building extension apps on Symphony
The Symphony Generator offers a fast way to bootstrap your Symphony extension app project.
First, install yeoman and the Symphony Generator.
Then, create a directory for your new app project and launch the generator.
This will prompt you with a number of questions about your app and pod configuration. Type in your app ID, using arrow keys to scroll and press enter to move on to the next prompt.
When prompted for Select your type of application
, choose Extension App (ADK)
. You then have a choice of different project types:
Basic - simple project to demonstrate initialization
App View - uses React with either JavaScript or TypeScript to create app views
Message Renderer - overrides rendering of messages with custom formatting
If you require an extension app that requires user identity, you will also need a backend that can perform the Circle of Trust process. You should then select the Extension App + Circle of Trust (ADK + BDK)
option, which will generate both the extension app project using ADK and a backend project using BDK that will perform the app authentication and validation.
When you are ready to deploy your app permanently (or if you require Circle of Trust), follow the instructions on this page to setup your app.
A browser window should launch with the URL https://localhost:4000/controller.html. If it doesn't, visit that page manually. Dismiss the security warning and close the page.
Visit https://develop2.symphony.com/?bundle=https://localhost:4000/bundle.json to inject the running app temporarily into a pod for testing
Acknowledge the developer mode notice. Your app is now loaded.
Each project type in the generator corresponds to one of the guides below. Read the respective guide for explanations of how to use ADK.
The template
parameter in the render()
function of the entity Service is an ExtensionML string used to render entities. To render a static object or an iFrame, you can use advanced templating.
For a full ExtensionML reference, continue here:
You can create a reusable template
that leverages the values in the data
object for the specific object. When using the data
for these attributes, the template uses the id
attribute on the tag to specify which member of the data object holds these attributes.
For example, the template and data below could be used to display a link within a Symphony message:
Use the following flow control tags to support entities that have conditional logic or recursive data:
Tag
Description
<if>
Use this tag to conditionally use the enclosing template. This tag must include an id
attribute. If there is data at the key
specified by the id
, then the enclosing template will be used, otherwise it will be skipped.
<if:last>
Use this tag within an iteration to conditionally use the enclosing template. If the current iteration is the last item in the iterated list, the enclosing template will be used.
<if:not-last>
Use this tag within an iteration to conditionally use the enclosing template. If the current iteration is not last item in the iterated list, the enclosing template will be used. This is useful to add commas between items in a list, without adding one to the last item.
<iterate>
Use this tag to loop through the items in an array. The template between the opening and closing <iterate>
tag will be used for each item in the array. The data for the template will reference the data in current the list item. This tag must include an id
attribute. This must be the key
to an array in the data
object.
Use the following tags to have interactivity on messages by implementing methods that are called to execute business logic:
Tag
Description
<action>
The following steps show examples on how to use actions.
Add the <action>
tag to an entity template.
The data field of the entity must have objects that match the actions ids. For these objects, we can provide an icon, a label and a service name for the action.
Implement an action method for the service of the entity renderer. This method will be called once the <action>
tag is clicked.
The Client Extension API uses services for communication between your application and the Symphony client. The Client Extension API provides a set of services that your application can leverage to extend the Symphony Client and to create custom workflows and experiences.
The Client Extensions API provides the following remote services:
Use the modules
service to create application-specific modules:
The Applications navigation section is found at the bottom of the left-hand sidebar of the Symphony client workspace. Use the applications-nav
service to create a navigation item for your application:
Use the ui
service to extend various parts of the Symphony client user interface. For example, add buttons on 1 to 1 chats and chatroom modules or add links to the #hashtag and $cashtag hovercards:
Use the share
service to allow users to share content from your application into Symphony conversations:
Use the entity
service to allow your app to render a Structured Object within a within a message sent by the REST API:
Apps can offer premium functionality through licensed subscriptions. Use the commerce
service to identify the products (premium versions) to which a user is subscribed:
Use the dialogs
service to create modal windows (e.g. to open a modal window from a button registered at a room level):
Both the Client Extensions API services and your application services use the same interface. Continue here to learn how to implement the Service interface methods:
Use the entity
service to allow your app to render a Structured Object created by the REST API within a message:
The following methods are available on the entity
service:
Register a renderer for a type
of entity:
Parameter
Type
Description
type
String
The type of entity that will be rendered by your application. Entities are namespaced using reverse domain name notation, e.g. com.symphony.address
.
options
Array
Reserved for future use.
serviceName
String
The name of the application service that will be used to render this entity.
You must implement the render()
method on the specified application service. This method will be invoked when the associated entity is rendered in the Symphony client.
Renders an entity given its type and data values:
Parameter
Type
Description
type
String
The type of entity to be rendered.
data
Object
The data for the specific entity being rendered. This data is specified when the message is created.
The render method returns an object with the following fields:
Parameter
Type
Description
template
String
An ExtensionML string that specifies the object's presentation.
In addition to ExtensionML tags, iframe tags are also supported.
data
Object
entityInstanceId
String
A unique identifier used to reference a specific entity.
Effectively re-renders an entity with new template and data objects given its entityInstanceId:
Parameter
Type
Description
entityInstanceId
String
The instance id of the entity to be updated
template
String
data
Object
The data for the entity being updated
The 'pause' and 'resume' methods are optional. If you choose to use the methods, implement them on the renderer service. These methods will be invoked when the associated entity is rendered in the Symphony client.
Entities are checked periodically (every two seconds) if they are visible on the screen. If an entity is completely visible, a resume event is triggered to the renderer service of this entity. If an entity is partially visible or completely hidden, a pause event is triggered instead.
Example: • Pause event: responsible for stopping/pausing a video transmission (video iframe). • Resume event: responsible for playing the video.
In addition to ExtensionML tags, iframe
tags are also supported in the template
parameter:
Apps can offer premium functionality through licensed subscriptions. Use the commerce
service to identify the products (premium versions) to which a user is subscribed:
Returns the list of products to which the user is subscribed for your app:
This method returns a promise that will be fulfilled with the array of products the user is subscribed to for your app. For each product, the following is returned:
"Premium" is the term used for paid subscription products, while "Standard" represents a freemium app. The name and SKU will be values specified by the application developer when implementing a premium version of their app.
This event is fired when the user's product subscriptions are changed.
Symphony’s Extension API allows apps to extend the Symphony client user interface by adding buttons on the IM, MIM, profile, and chatroom modules. This capability can be used to build apps that act upon the room and user identity details, such as a click-to-call integration.
Extension apps can receive stream participant information when an end user clicks on a button added by the app.
Your button will receive information from the user object in a MIM, IM and a room of up to 20 users.
Please note that the current user’s information isn’t returned, only the information of the other user(s) in the profile of a user, an IM, MIM or room.
This is the information that you will receive if your button is pressed inside of an IM or on a users Profile:
You must implement the trigger()
method on your application service in order to handle clicks on the registered extensions. This is a sample trigger method that will allow you to receive a user's phone number and email address as an authenticated extension app:
This is the information that you will receive if your button is pressed inside of a MIM or a room:
Use the ui
service to extend various parts of the Symphony client user interface. For example, add buttons on IM (1-1 chat) and chatroom modules or add links to the #hashtag and $cashtag hovercards:
The following methods are available on the ui
service:
openIMbyStreamID
openIMbyUserIDs
registerExtension
unregisterExtension
The following events are fired by the ui
service:
themeChangeV2
Open an existing conversation in a new module.
Released in version 20.10.
Open a conversation with one or more users in a new module.
Released in version 20.10.
Add an action button to the Symphony user interface.
Action buttons are added to various places in the UI, such as in the header of chat modules, in #hashtag or $cashtag hovercards, on the profile of users and more.
You must implement the trigger
method on your application service in order to handle clicks on the registered action buttons:
Remove a previously registered action button.
This will remove all instances of a particular extension - for example, from all single chat modules or all #hashtag and $cashtag hovercards.
This event is fired when the user's font size or theme is changed.
Use the share
service to allow users to share content from your application into Symphony conversation:
When the share function is invoked, a modal dialog is launched and populated with the shared object content (for example, the article options). The end user can then select conversations (IMs or chatrooms) into which to share the article.
Once the article is shared, it appears in the conversation view in a card format. The article will be linked either to a webpage (if the href
option is provided) or deep linked into the app (if the id
option is provided).
In order to view article contents in an application (for example, if the article id
is provided), the user must have the application installed.
If the recipient of a shared article does not have the application installed, the user will be prompted to install the application (provided that the user's enterprise has that application enabled).
If the recipient of a shared article does not have the application installed, and the application is not enabled for the user's enterprise, the user can view the content via the link (if href
is provided). If a link is not provided, the user will be notified that the article cannot be viewed because the application is disabled for the enterprise.
The following methods are available on the share
service:
Launches the "Share on Symphony" modal from your application, allowing the user to share content from your application into a Symphony conversation (IM or chatroom):
The following JavaScript shows an example of an article being shared:
The following table shows the article content:
The following table shows the share options:
The share
function can also be used to share custom, third-party entity types. In this case, the data
parameter must be populated with the following fields:
The following JavaScript shows an example of a custom third party entity being shared:
In this example, the following modal dialog is launched and populated with the shared object content:
You must specify your own application service for handling clicks on shared articles using handleLink
if you use the id
field for deep linking articles into your application.
You must implement the link
method on your application service in order to handle clicks on shared articles in conversations.
ExtensionML is a special form of markup that a front end app can use to perform custom rendering of an entity, instead of relying on Symphony to perform the default presentation.
While similar to PresentationML, ExtensionML consists of a template and corresponding data. Each tag in the template can have an ID that references a value stored in the data, binding the data to whatever is being rendered. For example, multiple paragraphs in the template can reference a sentence stored in the data by ID, allowing for reuse of that sentence in multiple places within the template being rendered.
A number of standard HTML tags are supported within templates: b
, u
, i
, strong
, br
, ul
, ol
, li
, span
, div
, table
, th
, tr
, and td
.
These behave like their HTML counterparts but must be properly-formatted XML. So, for example, rather than <br>
you must use <br/>
.
The following tags present text in different ways. Most of these require data to specify their content, but some can also use the content between the opening and closing tags.
The following HTML tags are handled in a slightly modified way:
The following flow control tags are used for entities that have conditional logic or data that can be iterated upon:
The following example shows the XML template for an entity with flow control logic and corresponding data:
The following function can be used to turn HTML into properly formatted XML:
The Applications navigation section is found at the bottom of the left-hand sidebar of the Symphony client workspace. Use the applications-nav
service to create a navigation item for your application:
The following methods are available on the applications-nav
service:
add
remove
count
rename
focus
Add a new navigation item to the Applications section of the left-hand sidebar:
Remove an existing application navigation item:
Set the badge (notification) count on an application navigation item:
Change the title of an existing application navigation item.
Note that this only changes the title of a specific navigation item -- not to all navigation items created by the application:
Focus an existing application navigation item:
Use the dialogs
service to create modal windows (e.g. to open a modal window from a button registered at a room level).
The following methods are available on the dialogs
service:
show
rerender
hide
The following picture is an example of what you will be able to create with this service - this module will overlay onto the entire Symphony client window:
Presents a modal dialog to the user:
Changes the contents of the dialog. This is usually invoked when the user has performed some action:
List of permissions the application requires. See Permissions details for .
Certificate for the application. See
Applied for version 1.54 onwards. RSA key for the application. See
For more information, see the AllowedOrigin description in .
Use this tag to enable clicks on entities. For more information on how to use the action
tag, refer to .
An object containing the data referenced by the template. Described in .
The updated string that specifies the object presentation.
Use the method on the service specified in getProducts()
to subscribe to this event and specify a callback that will be executed when the event is fired. The callback should change the contents/features of your application to match the user's updated product subscriptions.
Your app must be an Extension App.
Extension apps can receive stream participant information when an end user clicks on a button added by the app. For more information, refer to .
Use the method on this service to subscribe to this event and specify a callback that will be executed when the event is fired. The callback should change the styling of your application to match the user's new theme.
ExtensionML is generated from a given entity and emitted by a built-in or third-party renderer. It is similar to but is used for interactive presentation.
Note: ExtensionML does not support Symphony Elements. For more information, refer to .
Parameters
Type
Description
serviceName (optional)
String
The name of a local application-implemented service. If passed, the productUpdate
event will be fired on that service if the user's product subscriptions change.
Parameter
Type
Possible Values
Description
streamID
String
The stream ID or conversation ID to be opened.
messageID
String
Either a messageID, or the null value
The messageId can be used in addition to the streamId to focus on a specific message of the conversation. Use "null" as parameter value to jump to the latest message of the conversation.
Parameter
Type
Possible Values
Description
userIds
String[]
Array of userIds.
Parameter
Type
Description
uiClass
String
The location within the Symphony application where the action button should be placed. Possible values:
- single-user-im
: Button added to the header of 1-1 chats
- multi-user-im
: Button added to the header of group chats
- room
: Button added to the header of chatrooms
- hashtag
: Link added to the hovercard that appears when hovering over a hashtag (e.g. #symphony)
- cashtag
: Link added to the hovercard that appears when hovering over a cashtag (e.g. $GOOG)
- settings
: Link added to detail card of the application in the Marketplace
- profile
: Link added to the user profile page and profile hovercard
id
String
A unique identifier for this extension (can be used to unregister).
serviceName
String
The name of a local service implemented by your application that will be invoked when a user clicks on your action button.
options
Object
An object, containing:
icon: the url of an image that will be displayed on the action button. Recommended format is a square SVG.
label: a label associated with the action button.
data: an opaque block of data that will be passed along with the trigger event
Parameter
Type
Possible Values
Description
uiClass
String
single-user-im
multi-user-im
room
hashtag
cashtag
settings
The location within the Symphony application where the action button was registered.
id
String
The id of the UI extension that should be removed.
Parameter
Type
Required
Description
type
String
Yes
The type of content that is being shared.
content
Object
Yes
An object that describes the content being shared. For a list of objects see standard entities.
options
Object
No
An object that describes options that can be used to enhance the share service
Field
Required
Format
Description
title
Yes
String
The headline of the article
subTitle
No
String
The subtitle of the article
blurb
No
String
A summary of the article to display
date
No
Date of publication
publisher
No
String
Name of the publisher
author
No
String
Name of the author
thumbnail
No
URL (could be a data url)
Image to be displayed - 106px-106px
id
Must provide either id
or href
, or both
String
An identifier used by the application to deeplink to the article
href
Must provide either id
or href
, or both
URL
URL to the article (opened in a new browser window)
Field
Required
Format
Description
prepopulateUsers
No
Array of strings
The users (UserIds) who will be listed initially as recipients in the share modal.
Available only for authenticated apps, and only for Client 2.0.
It is recommended to limit the number of pre-populated users so the Symphony end user can easily review the list of recipients before sharing.
Field
Description
inputAutofill
Use this to fill the comment field in the share dialog to provide initial text. Cash tags and hash tags that are specified in the text will be converted to the correct entity.
plaintext
The markdown representation of the entity, supporting a limited set of markdown features. The value of this field will be displayed on mobile devices and other older clients.
presentationML
The default presentation of the entity using presentationML. This will be seen by everybody who does not have an app with a custom renderer for the given type.
entityJSON
The object being shared.
format
The format of the message being sent. This must be set to "com.symphony.messageml.v2".
Tag
Description
Attributes
<formatted>
Adds text formatted with HTML markup. This must be properly formatted XML See here for information on converting HTML to XHTML.
• id
(Required): The key of a string within the template data.
<text>
Specifies regular text to be inserted into the entity. This text is inserted as is, without processing any HTML markup.
• id
(Optional): The key of a string within the template data. If no id
is specified, it uses the contents between the opening and closing tags.
<label>
Inserts a label. Labels are like text tags but are styled differently.
• id
(Optional): The key of a string within the template data. If no id
is specified, it uses the contents between the opening and closing tags.
<color-text>
Inserts a colored text. Supported colors: red, purple, green, darkGreen, blue, darkBlue, orange, grey, and yellow.
id
(Optional): The key of an object with two members:
• text specifies the text to be used
• color is one of the listed colors.
<pill>
Inserts text with a colored background and rounded corners. Supported colors: red, purple, green, darkGreen, blue, darkBlue, orange, grey, and yellow.
id
(Optional): The key of an object with two members:
• text specifies the text to be used
• color is one of the listed colors.
Tag
Description
Attributes
<a>
Inserts a link.
• id
(Optional): The tag must include either an id
attribute or an href
attribute. If id
is specified, it must be a key to a string specifying the URL of the link. If no id
attribute is specified, the href
attribute is used.
<hr>
Inserts a horizontal line.
<icon>
Displays a 16x16 pixel icon.
• id
(Required): The key to a text string specifying the URL to the image to display.
<small-icon>
Displays a 13x13 pixel icon.
• id
(Required): The key to a text string specifying the URL to the image to display.
<img>
Displays a 128x128 pixel image.
• id
(Optional): The tag must include either an id
attribute or an src
attribute. If id
is specified, it must be a key to a string specifying the URL of the link. If no id
attribute is specified, the src
attribute is used.
<iframe>
Inserts an iframe.
• src
(Required): The URL to the iframe.
• height
(Optional): If not specified, the default height of the iframe will be 50px. The maximum height is 1000px.
• width
(Optional): If not specified, the default width of the iframe will be 100%. The maximum width is 300px.
Tag
Description
Attributes
<mention>
Insert a mention.
• id
(required): The key to an object with the following members:
- id
: The unique ID of the user being mentioned.
- name
: The pretty name that is displayed for the mentioned user.
<hashtag>
Inserts a hashtag.
• id
(required): The key to a string specifying the hashtag. The string must be prefixed with '#'.
<cashtag>
Inserts a cashtag.
• id
(required): The key to a string specifying the cashtag. The string must be prefixed with '$'.
Tag
Description
Attributes
<if>
Conditionally uses the enclosing template.
• id
(required): The key to the data. If there is data at the specified key, the enclosing template is used; otherwise it is skipped.
<if:not-last>
Conditionally uses the enclosing template within an iteration. If the current iteration is not the last item in the iterated list, the enclosing template is used. This is convenient when you want to add commas between list items without adding one after the last item.
<if:last>
Conditionally uses the enclosing template within an iteration. If the current iteration is the last item in the iterated list, the enclosing template will be used.
<iterate>
Loops through the items in an array. The template between the opening and closing iterate
tag is used for each array item. The data for the template references the data in the current list item.
• id
(Required): The key to an array in the data object.
Parameter
Type
Description
id
String
A unique id for this navigation item (must be unique across all navigation items of a given application)
Either title
or {title, icon}
String or Object
Either the title of the left navigation item as a string or an object with the keys title
and icon
where the value of title
is a string and the value of icon
is the url of a square SVG (recommended), or the url of a square png/jpg image. Recommended size is 32x32.
serviceName
String
The name of a local service implemented by your application that will be invoked when a user action is performed relating to the application navigation
Parameter
Type
Description
id
String
The id of the navigation item that should be removed
Parameter
Type
Description
id
String
The id of the navigation item that should have its count updated
count
Integer
The new badge count number. Specifying 0 will hide the badge count.
Parameter
Type
Description
id
String
The id of the navigation item that should be renamed
Either title
or {title, icon}
String or Object
Either the title of the left navigation item as a string or an object with the keys title
and icon
where the value of title
is a string and the value of icon
is the url of a square SVG (recommended), or the url of a square png/jpg image. Recommended size is 32x32.
Parameter
Type
Description
id
String
The id of the application navigation item to focus
Parameter
Type
Description
id
String
A unique id for the dialog.
serviceName
String
The name of a local application-implemented service implemented.
template
String
The extensionML for the dialog content.
data
String
The data for the extensionML.
options
Object
The data for the extensionML
Parameter
Type
Description
id
String
The id of the dialog that should be updated.
template
String
The new extensionML content to display.
data
String
The data for the extensionML.
Parameters
Type
Description
id
String
The id of the dialog to close.
You can add a new method to the service that is handling button clicks, called filter()
, on any UI extension that you are implementing. Before a button is shown, that method is passed the uiClass, the id, and the payload. If filter()
returns false
, the button is not shown. If the method is unimplemented or it returns any value other than false, the button is shown.
The filter function returns the same data returned by the ui Service here. All data except for the the user's phone number is returned in cases where you are using an authenticated app. The user phone number is only returned for 1x1 IMs and User Profiles.
Based on the information returned, you can choose to selectively display the button. For example, you can display the button only if a user's phone number is present, or if a user is not a cross-pod user.
The Symphony Generator is a yeoman-based CLI tool that can be used to quickly generate Symphony Messaging bot, app and workflow project scaffolds. You can create example projects that use our developer toolkits:
BDK for Java
BDK for Python
WDK
ADK
Install yeoman and the Symphony Messaging Generator.
Then, create a directory for your new project and launch the generator.
Configure your bot, workflow or app environments accordingly.
OBO or On-Behalf-Of authentication allows an authenticated extension app to perform the following operations on behalf of a given user:
List the streams of a given user
Initiate connection requests to and determine connection status with other users
Get the presence state of other connected users
Initiate IMs with other users
Send messages and attachments
Set the context user's own presence
For a full list of OBO-Enabled endpoints, click here.
For OBO apps, authentication is a two-fold process:
The app itself must be authenticated using its RSA public Key. The app authenticates only if it is enabled for the pod and its key is trusted. Upon successful OBO app authentication, the app receives an app sessionToken
.
The app must request to authenticate on behalf of a particular user, using its app sessionToken
. The app authenticates only if it is installed for the user and its app sessionToken
is valid. Upon successful OBO user authentication, the app receives the user's sessionToken
.
Once the app has obtained the user's sessionToken
, it can make REST API calls with this sessionToken
to perform activities on behalf of the session user.
Before proceeding, check out the OBO App permissions required for a given workflow:
Category
Permission
Description
On Behalf Of
ACT_AS_USER
Required. This required permission allows an application to act on behalf of a user via any of the other permissions.
Note: This permission does not display to administrators on the Admin Portal because all apps can act on behalf of a user and therefore have the ACT_AS_USER
permission by default.
Messaging
SEND_MESSAGES
The application can send messages for the logged-in user.
Messaging
SUPPRESS_MESSAGES
The application can suppress messages for the logged-in user.
Get Connections
GET_USER_CONNECTIONS
The application can get connection requests for the logged-in user.
Send Connections
REQUEST_USER_CONNECTIONS
The application can send connection requests for the logged-in user.
Get Presence
GET_PRESENCE
The application can only get presence for the logged-in user.
Set Presence
SET_PRESENCE
The application can only set presence for the logged-in user.
Primary User Identity
GET_BASIC_USER_INFO
The application can get information about the logged-in user.
Primary Contacts Access
GET_BASIC_CONTACT_INFO
The application can get information about other users through user look-up and search.
List User's Streams
LIST_USER_STREAMS
The application can list the streams in which the logged-in user is a member.
In order to perform an OBO operation, you need to first create an extension application manifest bundle.json
file and upload to the Pod.
Parameter
Type
Required
Description
type
String
Required
This field should be set to sandbox
, which indicates that this is a standalone app, embedded within Symphony's client.
id
String
Required
The unique identifier for your app. Note: Do not use a colon (:) in this field, or you will receive a 401 Unauthorized
error.
name
String
Required
The name of your app, which will be displayed in the Symphony Market.
blurb
String
Optional
Applied for Developer Mode. Field for display in the Symphony Market and Admin Portal.
publisher
String
Optional
The publisher of your app, which will be displayed in the Symphony Market.
url
String
Optional
URL which the pod will call to send pod information to the application
domain
String
Required
The domain for your app, which should match the controller file URL.
icon
String
Optional
An icon for your app (32x32 px), which will be displayed in the Symphony Market.
Upload the manifest bundle.json
to the Admin Portal -> App Management -> Add Custom App -> Import Application Bundle File
Add your App Backend's (Bot) RSA public key in the Authentication section under App Management.
Give your Application the following Permissions:
ACT_AS_USER
Once your App is created, make sure that it is enabled:
Admin Portal -> App Settings -> Locate your App and toggle its 'Global Status' to be 'Enabled'
Toggle 'Visibility' to be 'Visible'
Toggle 'Installation' to be 'Manual'
The last step is to make sure that the application is installed for the appropriate users. If the installation is set to 'Manual', make sure end-users install the extension application manually via the Symphony Marketplace. If not, make sure Symphony Admin installs this application on behalf of a given list of users.
The BDK makes it super simple to create an OBO based workflow, To do so, simply, simply instantiate an OBO Session in your Bot project. The BDK allows you to instantiate your OBO session from a username or user ID.
In the following code snippet, the Bot authenticates on behalf of a given user and then prints a list of Streams (Type = ROOM) that the user in context is apart of:
Application authentication establishes bidirectional trust between an application and the Symphony client.
This process must be used by enterprise or partner app developers who wants to:
Create a secure connection between an app and Symphony
Access user identity information from the app
Partner applications need to implement a clear process to inform and ensure that their customers have approved that users' identities will be communicated by Symphony to the partner application. Please contact the Symphony Partner team for more details.
Because apps run as iframes within the Symphony client and interact with the client via JavaScript, creating a secure connection requires interaction between an app's backend and Symphony's backend.
Backend trust is established using a JWT signed by a RSA key, as you can see in the RSA Authentication endpoint.
This section outlines the full app authentication sequence, covering operations that take place on the app side and the Symphony side.
The following diagram shows the trust relationships developed through the application authentication flow.
The flow utilizes two tokens, Ta and Ts, which are generated by the app and Symphony backends after the servers mutually authenticate each other using certificates. A circle of trust is established by passing the tokens in opposite directions so that each backend server receives matching tokens via two separate paths.
The diagram below shows the interaction between the actors in the Application Authentication flow (the app frontend and backend, the Symphony frontend and backend). This flow starts after the user has logged into the Symphony client (establishing the identity of the user and trust between the Symphony client and backend). The flow begins after the user loads the partner application.
Step
Description
1
App frontend requests Symphony frontend for pod ID via the Extension API SYMPHONY.remote.hello()
method.
2
Symphony frontend returns the pod ID to the app frontend.
3
App frontend sends request to its backend to initiate app backend to Symphony backend authentication, passing the pod ID.
4
App backend looks up the pod URL for backend authentication based on the pod ID. This step is only required for partners whose apps will be used by multiple customers. Enterprise developers will already know their pod URL.
5
App backend generates an app token (Ta).
6
App backend requests Symphony backend authentication using a public RSA key for the app, supplying Ta.
7
Symphony backend generates a Symphony token (Ts).
8
Symphony backend stores the (Ta, Ts) token pair for future validation.
At this point, trust has been established between the Symphony and app backends.
9
Symphony backend returns Ts to app backend.
10
App backend stores the (Ta, Ts) token pair for future validation.
11
App backend provides Ta to app frontend.
Ta is beginning the counter-clockwise journey around the the circle of trust.
12
App frontend registers app with Symphony frontend, passing appId and Ta.
Ta continues its journey.
13
Symphony frontend passes Ta to Symphony backend.
14
Symphony backend validates Ta, verifying it matches the Ta of a stored token pair.
At this point, Ta has come full circle -- the Symphony backend initially received it securely from the app backend and has now received it from the app backend again, via the app and Symphony frontends.
15
Symphony backend passes Ts back to the Symphony frontend, along with the signed user identity JWT, which will be asked for later.
At this point, the Symphony frontend trusts that the app is valid (i.e. the app server can be trusted). Because Symphony trusts the app, Symphony will be able to return protected content, like user identity information. However, the app must now trust Symphony itself in order to trust the information that is returned.
Ts is beginning the clockwise journey around the the circle of trust.
16
As a response to app registration (Step 12), Symphony frontend will return Ts.
Ts continues its journey.
17
App frontend should pass Ts to app backend for validation.
18
App backend should validate Ts against the previously stored token pair.
19
App backend should return a confirmation (or denial) to the app frontend.
At this point, the app frontend trusts that Symphony is valid and can trust the information returned by Symphony. The app can now request user identity information from Symphony and trust the response.
20
App frontend can request the user identity information by calling the getJWT()
method on the extended-user-info
service.
21
Symphony frontend returns the JWT which includes user identity information.
22
App frontend passes the JWT to app backend for verification.
The JWT is signed by the Symphony backend to ensure that it is not tampered with. The signature can be verified using publicly available certificate on the Symphony pod.
23
App backend requests the public app certificate used to sign the JWT from the Symphony backend.
24
Symphony backend returns the certificate in PEM format.
25
App backend verifies the JWT using the obtained certificate.
26
App backend can authenticate the user using the JWT contents. The implementation details of this are up to the developer.
27
App backend returns confirmation to app frontend.
At this point, the app fully trusts the Symphony user.
You will need an RSA key pair. The public/private key pair for signing authentication requests requires the following:
• An RSA512 algorithm with a key length of 4096 bits. • An X.509 format for public keys and PKCS#1 or PKCS#8 for private keys. • PEM-encoded keys.
Generate a valid key pair by running the following generate_rsa_keys.sh
script:Shell
Note: The script requires the openssl
package.
Generate the PKCS#1 keys manually using the following commands:Shell
Generate the PKCS#8 keys manually using the following commands. Provide your username as the Common Name (CN).Shell
The file publickey.pem is the public key. Import this into the pod.
Sign the authentication request using either privatekey.pkcs8 or privatekey.pem, depending on the support available in the JWT library.
You should familiarize yourself with JSON Web Tokens (JWT) which are the format used for passing user identity information from Symphony to your app.
Your app will need to implement verification of the JWT shared by Symphony.
Partner and enterprise app developers will need to implement the following sequence within their app on the frontend and backend:
Upon completion of this process, your app will know it is interacting with a valid Symphony client and can trust any user identity information it obtains from Symphony.
You will identify the Symphony pod against which to authenticate from the existing Extension API SYMPHONY.remote.hello() method.JavaScript
The pod ID will be useful to partner developers in building a mapping of customers to auth URL. As described in the Prerequisites section above, partners can receive the customer's auth URL through the callback.
If you are an enterprise developer building an internal custom app, you will not need to use the pod ID, since you will always use your own auth URL.
Once you have identified the pod where your app is being run on, you will initiate the app authentication flow. You will use the auth URL previously obtained and a JWT signed by your app RSA key. The endpoint to be used is as follows
POST
https://<host>:<port>/login/v1/pubkey/app/authenticate/extensionApp
Authenticate your Extension App with an App Token
appToken*
String
Token generated by the application.
authToken*
String
A JWT containing the caller's username and an expiration date, signed by the caller's private key.
In addition to specifying the JWT during this call, your app backend must generate an app token (Ta) to pass in the POST body parameters of the request. The app token must be a unique string for each authentication request. The token is opaque to Symphony.
In the response, Symphony will return back the appId, the previously presented app token, as well as a Symphony-generated token (Ts). The Symphony token is short-lived and will expire within five minutes.
Your app backend should store this token pair as they will be used for subsequent validation steps in the authentication process.
You should now register your app using not just your appId
, but an object containing both your appId
and tokenA
.
The SYMPHONY.application.register method will now accept either an appId
string or appData
object.JavaScript
SYMPHONY.application.register
will now return a promise that will be fulfilled with an object containing the members appId
(your appId) and tokenS
(the Symphony token).
Once your app has registered, your app will need to validate the Symphony token that was passed to you through the frontend against the Symphony token you previously obtained through the Symphony backend.
The validation implementation is up to you. The implementation in the sample application provided by Symphony assumes a stateless app backend. The app frontend sends both the app and Symphony tokens back and verifies that the pair exists in the token cache. (If the partner app and server maintained a sticky session, you could pass back only the Symphony token since the app token would already have been saved in the session.)
Once the token pair match has been found, your application knows that it is interacting with a valid Symphony client.
User identity information can be obtained through the getJwt()
method on the extended-user-info
service.
Returns identity information for the user in context in the form of a JSON web token (JWT). The JWT has been signed by the private key of the pod and can be verified using the pod's public key.
Note that this method will only return user identity information if Steps 1-15 of the application authentication sequence have been completed (i.e. Symphony frontend trusts app frontend). If not completed, then the JWT will only contain userReferenceId, an anonymous identifier for the user.
The JWT will be returned as a base-64 encoded string, with the following format when it is decoded:
Your app backend can validate the authenticity of the JWT returned by Symphony's frontend using the public certificate that was used to sign the JWT. The following endpoint can be used to obtain the certificate:
Using this certificate, you can verify that the JWT has not been tampered with. The signing algorithm is RS512. You should check that the algorithm specified in the header reflects this to protect against some known exploits of JWTs. The JWT can be decoded on the backend to obtain the user's identity. The decoded JWT will have the following format:
This information can then be used by to authenticate the user on the app backend.
It depends.
If you are a partner developing a premium app for Symphony and want to know user identity information, you will need to implement application authentication. For example, you may need user identity information in your app if you have entitlements or access control maintained outside of Symphony that that is tied to a user's email address.
However, if you are simply looking for a unique identifier that will allow you to map user preferences, you can do this without app authentication - by using the anonymous userReferenceId provided by Symphony.
The architecture of apps built using Symphony's Extension API poses a unique challenge. These apps run as iframes within the Symphony client and interact with the client via JavaScript. Before Symphony can share any user identity information with the app, Symphony must be certain that the app is a verified partner application. The partner application also must be certain that the user identity information is coming from Symphony, which isn't obvious with the insecure iframe model. Mechanisms like OAuth and SAML do not establish this bidirectional trust in a way that would make this possible. While getting user identity via oAuth or SAML provides a way to securely identify the user, this does not address the lack of trust between the app and Symphony frontends.
Yes, you can test app authentication using developer mode, so long as there is a certificate that has been imported and trusted on your pod that matches the appId in your bundle file.
BDK is a library of tools and intelligent API bindings that provides an ultra simplified configuration and authentication setup, intuitive message and room management, customizable message templating, and a new activities API that makes it easy to facilitate bot workflows. The BDK and bot project generated by the Symphony Messaging Generator makes it super easy to get started!
To begin let's open up the code generated for you by the Symphony Messaging Generator in your favorite IDE. Navigate to the BotApplication.java
file:
To build a conversational workflow, add the following datafeed listener (onMessageSent
) to your main bot class:
The above snippet is listening for all incoming messages that contain "/hello". To see a full list of datafeed listeners provided by the BDK, navigate here. If the message received by the bot contains "/hello", the bot will respond by sending a message to the stream or conversation in context. Run your bot and let's see it in action:
As you can see, it's super easy to access the message, room, and user context from an incoming event. The V4Initiator
and V4MessageSent
classes provide convenience methods that make it easy to orchestrate workflows. For a more detailed overview of how to leverage the BDK's message, room, and user management classes, continue here.
Above, we showed you a simple example of how to leverage the BDK to build a conversational bot. To see more advanced usage of the BDK in order to build complex financial workflows, continue on to our dedicated BDK Certification course:
In order to create a secure connection between your app and the Symphony client, apps need to perform app authentication. Upon successful authentication, Symphony extension apps establish a bidirectional trust, allowing for secure and authorized access to user data by leveraging the Symphony Extension API.
In order to perform app authentication, your app must perform a combination of frontend and backend authentication calls to the Symphony client and pod respectively. The following steps provide an overview of the frontend and backend calls your App needs to make:
The first step of app authentication is a frontend call that leverages the Symphony Extension API to initialize your app. This should be used to initialize the connection to the Client Extension API from your application controller:
The next step of app authentication is to perform a backend authentication call to the authentication endpoint on the pod:
POST
https://<host>:<port>/login/v1/pubkey/app/authenticate/extensionApp
Authenticate your Extension App with an App Token
appToken
string
Token generated by the application.
authToken
string
A JWT containing the caller's username and an expiration date, signed by the caller's private key.
Your backend should store this token pair as they will be used for subsequent validation steps in the following authentication process.
The next step of the authentication workflow is to register your app on the Symphony client using the Symphony Extension API. Registering your app requires the following frontend call to be performed in your application controller:
Parameter
Type
Description
appData
Object
Object containing both your appID and your app token.
servicesWanted
Array of Strings
A list of names of remote services that your application wants to subscribe to
servicesSent
Array of Strings
A list of names of local services that your application wants to make available remotely
Response:
Parameter
Type
Description
Registration Promise
Object
Object containing both your appID and Symphony JWT Token
The next step in the authentication workflow is to:
validate the App Token JWT returned from the backend API call in step 2
validate the Symphony JWT Token that was passed to you through the frontend against the JWT Symphony JWT Token previously attained from executing the backend API call shown in step 2.
While the implementation of this token validation is up to the developer, a sample implementation of both app token and symphony token validation is provided out of the box by the BDK. In this sample implementation, the app frontend sends both the App Token and Symphony Token to the backend where it verifies that the token pair exists in the token cache.
At this point, your app is fully authenticated and has established a bi-directional trust between itself and the Symphony client.
Once you have successfully authenticated and validated your tokens, you can obtain user data through the Extension API:
The Extension API provides an extended-user-info
service that contains a getJwt()
method. In order to leverage this method, your app must first subscribe to the extended-user-info
service:
Once subscribed, you app can leverage the getJwt()
method as follows:
The username
field has been changed in version 1.55.3, it now returns <email address> instead of <Symphony username>. Please note that this change has been done to help the transition for Applications that were relying on the username field and that the username field will be entirely removed in an upcoming version.
This method returns a base-64 encoded JWT token for the user in context, containing the following data when decoded:
At this point, your authenticated app has access to sensitive user data such as the Symphony user ID, username, email address, displayName, company, location, etc. Extension apps can leverage this user data in order to create user-specific workflows and automations.
Currently, calling getJwt()
without having completed the application authentication sequence will return Undefined. We will be addressing this in an upcoming release so that userReferenceId will correctly be returned if application authentication has not been completed. In the interim, userReferenceId can still be obtained from the Register and Connect methods.
If you wish to take this a step further, your app can take the JWT returned in the last step and perform authentication on behalf of (OBO) the user in context. If you wish you learn more about OBO authentication and OBO enabled workflows, continue here:
To learn how to properly configure and authenticate your extension app using the BDK, complete the following configuration tutorial:
The BDK is a library of tools and intelligent API bindings that provides an ultra simplified configuration and authentication setup, intuitive message and room management, customizable message templating, and a new activities API that makes it easy to facilitate bot workflows. The BDK and bot project generated by the Symphony Messaging Generator makes it super easy to get started!
To begin let's open up the code generated for you by the Symphony Messaging Generator in your favorite IDE. Navigate to the BotApplication.java
file:
Let's take a closer look at the code responsible for sending Symphony Messaging Elements in our BotApplication.java:
Here we are using the Activities API to register a new slash command that listens to "/gif". To learn more about creating your own slash commands or how to leverage the Activities API, continue here. If an incoming message contains ("/gif), the bot builds a new message template which is provided out of the box for you:
The above freemarker template contains messageML that represents a Symphony Messaging Element. To learn more about Symphony Messaging Elements, continue here. When a message is sent to the bot containing "/gif" the following Symphony Messaging Element is sent to the user in the conversation or stream in context:
Bots need someway to capture the data submitted within this form. Bots can easily do so by registering a new type of Activity class:
Open up your GifFormActivity
class. Here you will see that GifFormActivity
extends the FormReplyActivity
class. A form activity is only triggered when an end-user replies or submits an Elements form. To learn more about creating your own FormReplyActivity
classes, continue here.
Inside of the GifFormActivity
class you will see an ActivityMatcher matcher()
method:
Using the context
variable, your bot can access information about the context of the form submission including the form values and the form Id. To learn more about using FormReplyActivities
, continue here.
Inside of the matcher()
function, the bot is performing a validation check. If the formId
of the submitted form is equal to "gif-category-form"
, then the bot calls the onActivity()
trigger function and executes its business logic. If there is not a match, the bot does nothing and continues to listen for incoming events. As we see in our gif.ftl
template above, the formId
matches, so the following onActivity()
trigger function executed:
Run your bot and execute the following to see it in action:
Above, we showed you a simple example of how to leverage the BDK and Symphony Messaging Elements to build an interactive bot. To understand more advanced usage of the BDK, continue on to our dedicated BDK Certification course:
The BDK for Java is Symphony Messaging's preferred tooling for Java developers to build bots. It is a library of tools and API bindings that provides simplified configuration and authentication, intuitive message and room management, customizable message templating and an activities API that makes it easy to build bot workflows.
The BDK for Java Github repo can be found here: @finos/symphony-bdk-java
The BDK for Java Certification course can be found here: learn.symphony.com/bundles/java-bot-developer
Once you have your bot and Symphony Messaging environment properly configured, the generated code provides an out-of-the-box implementation for authenticating your bot:
By instantiating a new SymphonyBdk
instance with your config.yaml
file, the BDK loads in your config and authenticates your bot. Once authenticated, your bot is ready to leverage the REST APIs in order to create rich automations and workflows on Symphony Messaging.
BDK for Java also supports the OBO (On-Behalf-Of) pattern of authentication, allowing an authenticated bot + extension application to perform operations on behalf of a given user. The BDK's implementation makes it easy to perform the following operations on behalf of a given user:
List the streams of a given user
Initiate connection requests to and determine connection status with other users
Get the presence state of other connected users
Initiate IMs with other users
Send messages and attachments
Set the context user's own presence
To leverage an OBO based workflow, simply instantiate an OBO Session in your bot project. The BDK allows you to instantiate your OBO session from a username or user ID. Once authenticated bots can perform any of the OBO workflows listed above:
BDK for Java makes it easy to manage multiple bot instances within a single project. As long as you have unique configuration files that correspond to different service accounts, you can manage multiple bot instances from a centralized source. To do so, simply instantiate multiple bot instances of the SymphonyBDK
class within your bot project:
The BDK also provides a DatafeedService
interface that makes it easier than ever for bots to manage real-time messages and events. The DatafeedService
interface provides the following methods for your bot to use:
Method
Descriptions
start()
Start the bot's datafeed
stop()
Stop the bot's datafeed
subscribe(RealTimeEventListener)
Subscribe a custom event listener class. Inside this class is where the bulk of your business logic goes.
unsubscribe(RealTimeEventListener)
Unsubscribe from a custom event listener class.
For bots to listen to incoming events and messages, bots must subscribe to a custom RealTimeEventListener
. This RealTimeEventListener
class must implement eventType methods (e.g. onMessageSent()
) along with custom business logic inside.
When a user sends a bot a message, the bot will pick up the event from the datafeed and check to see if an implemented eventType method matches the eventType (MESSAGESENT
) of the inbound event. If there is a corresponding eventType method registered, the bot will execute the business logic inside of this eventType method. Otherwise the bot will not perform an action and will continue to listen for inbound events from the datafeed. An example implementation is provided out of the box by the BDK:
Below is a full list of methods provided by the RealTimeEventListener
class and their corresponding eventTypes. Implement the following methods in order to listen for a given Symphony Messaging event:
Method
Event Type
onMessageSent()
MESSAGESENT
onInstantMessageCreated()
INSTANTMESSAGECREATED
onMessageSuppressed()
MESSAGESUPPRESSED
onRoomCreated()
ROOMCREATED
onRoomUpdated()
ROOMUPDATED
onRoomDeactivated()
ROOMDEACTIVATED
onRoomReactivated()
ROOMACTIVATED
onUserRequestedToJoinRoom()
USERREQUESTEDTOJOINROOM
onUserJoinedRoom()
USERJOINEDROOM
onUserLeftRoom()
USERLEFTROOM
onRoomMemberPromotedToOwner()
ROOMMEMBERPROMOTEDTOOWNER
onRoomMemberDemotedFromOwner()
ROOMMEMBERDEMOTEDFROMOWNER
onConnectionRequested()
CONNECTIONREQUESTED
onConnectionAccepted()
CONNECTIONACCEPTED
onSymphonyElementsAction()
SYMPHONYELEMENTSACTION
onSharedPost()
SHAREDPOST
For more information on the Symphony Messaging datafeed continue here:
A Symphony Messaging workflow can be thought of as a sequence of operations or a repeatable pattern of activities that are organized together in order to transform data, provide a service, or process information. Each of these operations or activities may be completed by a single user, shared between a bot and a user, or shared between multiple actors including bots, users, and even third party systems.
By providing an intuitive Activities API, the BDK makes it simple to define a set of discrete operations or activities for different actors in your system to execute. Ultimately, these activities constitute the building blocks for a powerful Symphony Messaging workflow automation.
Once you have defined a discrete set of activities for different actors in your system to execute, the next step is to organize them together in an intelligent way. The BDK provides a powerful Workflow API (coming soon) that makes it easy to organize a sequence of activities together, and subsequently orchestrate a Symphony Messaging workflow.
BDK for Java provides an Activities API, an interface that makes it easy to manage user-to-bot interactions or activities. Specifically, the Activities API provides easy access to message and room context, initiator metadata, and an intuitive way to interact with the datafeed, making it easy for bots to listen and reply to different Symphony Messaging events. The methods and logic provided by the Activities API allows for granular control over the entire user-to-bot interaction. This encapsulated logic is easily reused, forming the discrete building blocks of a Symphony Messaging workflow automation.
In order to register activities for your bot instance, you must leverage the ActivityRegistry
class:
There are two different types of activities supported by the BDK:
Command Activity: an activity triggered when a message is sent in an IM or Chatroom.
Form Activity: an activity triggered when a user replies to an Elements form message.
A command-based activity is triggered when a message is sent in an IM or Chatroom. Using the Activities API allows developers to register commands in the following formats:
@bot/buy
(Slash command with a bot @mention)
/buy 1000 $goog
(Slash command without a bot @mention)
Listen for the word hello
(Not a Slash command - just listen for messages containing a specific word)
The Activities API also makes it easy for Bots to listen for elements form submissions. Assume the following elements form has been posted into a room with the following attributes:
form id
= "hello-form"
<text-field>
name = "name"
form contains an action button
In order to register a form activity or listen for an incoming elements form submission, bots must register a class that extends the FormReplyActivity
class:
As shown above, the Activities API makes it simple to manage incoming commands, elements form submissions, and access message context making it easy to manage bot-user interactions and create custom workflows.
As shown above, the BDK makes it easy to create a datafeed and listen for events through the RealTimeEventListener
class. In addition, this class makes it easy to access user, message, and room data in context. Each eventType method is implemented with instances of V4Initiator
and V4MessageSent
objects:
Use the V4Initiator
class methods to access the the user data in context:
Method
User Attribute
initiator.getUser().getUserId()
User ID
initiator.getUser().getFirstName()
First Name
initiator.getUser().getLastName()
Last Name
initiator.getUser().getDisplayName()
Display Name
initiator.getUser().getEmail()
initiator.getUser().getUsername()
Username
Use the V4MessageSent
class methods to access message data in context:
Method
Attribute
event.getMessage().getMessageId()
Message ID
event.getMessage().getTimestamp()
Message Timestamp
event.getMessage().getMessage()
Message Text
event.getMessage().getSharedMessage()
Shared Message
event.getMessage().getData()
Message Data
event.getMessage().getAttachments()
Message Attachments
Use the V4MessageSent
class methods to access stream data in context:
Method
Attribute
event.getMessage().getStream().getStreamId()
Stream ID
event.getMessage().getStream().getStreamType()
Stream Type
event.getMessage().getStream().getRoomName()
Room Name
event.getMessage().getStream().getMembers()
Room Members
event.getMessage().getStream().getExternal()
External
event.getMessage().getStream().getCrossPod()
Cross Pod
The Activities API also makes it easy to access relevant user, message, and stream data in context. CommandActivity
classes have access to to this data through the CommandContext
class. This class is instantiated with instances of V4Initiator
and V4MessageSent
objects. Bots are able access to the user, message, and stream data in context through the same methods shown above. Leverage these methods within the onActivity()
method shown below:
FormActivity classes have access to relevant user, form, and stream data through the FormReplyContext
class. This class is instantiated with instances of the V4Initiator
and V4SymphonyElementsAction
class. The V4SymphonyElementsAction
class provides the following methods to access form data in context:
Method
Attribute
context.getSourceEvent().getStream()
Elements Stream ID
context.getSourceEvent().getFormMessageId()
Elements Message ID
context.getSourceEvent().getFormId()
Elements Form ID
context.getSourceEvent().getFormValues()
Elements Form Values
The BDK for Java also supports custom and built in message templating. The BDK is agnostic to what templating library developers choose, with built-in support for FreeMarker and Handlebars. In order to use message templating, you must leverage the TemplateEngine
class provided by the BDK.
If you wish to build your own custom message template, you must implement one of the newTemplate()
methods provided by the TemplateEngine
class:
newTemplateFromFile()
newTemplateFromClasspath()
newTemplateFromString()
The following shows an implementation of the newTemplateFromClasspath()
method:
The corresponding FreeMarker template should be stored in the classpath root i.e. src/main/resources/hello.ftl
The BDK for Java's Spring Boot integration provides native annotations, making it easy to configure your bot's datafeed listeners and register command activities. Start with a standard Spring Boot main class.
Now you can create a component for bot applications by annotating classes with the @Component
annotation:
The Core Starter uses Spring Events to deliver Real Time Events.
You can subscribe to any Real Time Event from anywhere in your application by creating a handler method with two conditions:
Be annotated with @EventListener
Have a com.symphony.bdk.spring.events.RealTimeEvent<? extends T>
parameter
You can easily register a slash command using the @Slash
annotation. Note that the CommandContext
is mandatory to successfully register your command. If not defined, a warn
message will appear in your application log:
Any service or component class that extends FormReplyActivity
or CommandActivity
will be automatically registered within the ActivityRegistry
Note that for this tutorial, we've configured our bot project using the Spring Boot integration provided out of the box by the BDK. When generating your bot using the Symphony Messaging Generator, simply select 'Spring Boot' when prompted to 'Select Your Framework'.
The BDK is a library of tools and intelligent API bindings that provides an ultra simplified configuration and authentication setup, intuitive message and room management, customizable message templating, and a new activities API that makes it easy to facilitate bot workflows. The BDK and bot project generated by the Symphony Messaging Generator makes it super easy to get started!
To begin let's open up the code generated for you by the Symphony Messaging Generator in your favorite IDE. Navigate to the BotApplication.java
file:
Notice how our BotApplication
class is very simple. The idea is you have Component classes (annotated by @Component
) that do the bulk of the work (see the generated GifSlashHandler
, GifFormActivity
, and OnUserJoinedRoomListener
classes) while the main entry point is meant to really simple. The initialization of your bot project and auto-wiring of your beans (component classes) is taken care of by the BDK Spring Boot stater through the @SpringBootApplication
annotation.
The following Component class, for example, is provided out of the box by the BDK and Symphony Messaging Generator:
To learn more about the BDK's Spring Boot Integration continue here or to our dedicated BDK Developer Certification course where you will learn in depth about how to build bots using the BDK and Spring Boot:
For our headless bot workflow, we will create a RESTful Web Service that is able to handle HTTP GET/POST requests and read its JSON body payloads. Creating and bootstrapping our RESTful service is super easy since our bot application is already an integrated Spring Boot project. To get started, lets define a simple RestController
class for our bot:
Here we annotate our class with the @RestController
annotation, and use the @GetMapping
annotation to map the HTTP GET request from our base path to a specific handler method, index()
.
Go ahead and start your bot application, and navigate to https://localhost:8080 in a web browser of you choice. You should see the following displayed in your browser:
Now that we have a basic RESTful service setup, let's add some Symphony Messaging specific functionality. Specifically, let's create another handler method that handles the following POST request and sends a message into a specific conversation or stream:
To create this handler method, let's add to our existing HelloController
class:
Line 11: Inject the BDK's provided MessageService
(this is the equivalent as bdk.messages()
)
Line 13-15: Initialize HelloController
with an instance of MessageService
.
Line 22-25: Create another handler method, postNotificationMessage()
for a POST request to "/notification/{streamId}"
The postNotificationMessage()
handler expects a streamId
path parameter and also a Notification
(message) as a part of the request body.
Now the only missing detail here is that we do not have the Notification
Java class into which the incoming JSON will be mapped by Spring Framework's @RequestBody
annotation. To do so, create the following POJO:
Now, Spring will create an instance of Notification
Java class, and will set object properties with the values specified in the JSON body.
Run your bot once more and identify a streamID to use.
Send the following POST request using either curl or postman:
You should see the following in your designated stream or chatroom:
Above, we showed you how to leverage the BDK and the Spring Boot integration to build a headless bot. To see more advanced usage of the BDK and Spring Boot, continue on to our dedicated BDK Developer Certification course:
The ADK is the modern toolkit used to build extension apps for Symphony Messaging.
Choose from one of the following guides depending on the type of extension app you require. These guides will help you get started building your extension app project from scratch.
For the complete technical reference to all methods and interfaces, please refer to the following page instead:
This guide will provide an overview on how to use the Symphony Messaging App Developer Kit (ADK) to build the most basic extension app. This app will add a sample entry in the left navigation bar and a button to each available target zone in the Symphony user interface. For simplicity, the project will be minimal and not use any frameworks to demonstrate the bare neccessities required to build an extension app.
Create a working directory and initialize it using npm
.
Install the Symphony Messaging ADK along with the webpack bundler.
Open the project directory in an editor of your choice
Edit the package.json
file, replacing the scripts
section with the following:
This adds two commands:
npm start
for starting the development web server
npm run build
to launch the production build process
Create a file named webpack.config.js
that will inject the ADK configuration into the webpack bundler.
Each extension app requires a manifest (also known as the bundle.json
file) to describe the application. Create a file named bundle.json
with the following contents:
We are now ready to start building the app. Create a src
directory and a file named index.js
within it.
The code ADK.start()
initializes the ADK with an app id (adk-example
) that must correspond with the value provided in the bundle.json
manifest from the previous step.
Once the initialization is complete, we use ADK.navigation
to add an item to the left navigation bar. This item will have the label "ADK Example" and clicking on it will pop up an alert with the content: "Navigate!"
We then proceed to build an array of all the available target zones and loop through them, calling ADK.buttons
to add a button into each target zone. Each button will be labelled "Button on" followed by the zone and clicking on them will log a message to the console. The message will report from which zone the button was pressed and the payload included with the event. Note that not all zones will contain a payload.
We can now start the app using:
This starts a local development server on https://localhost:4000
. Note that this is a TLS-enabled site because all extension apps need to be loaded from TLS-enabled sites. However, because this is a development server, the certificate is self-signed and not trusted by any browser.
Visit https://localhost:4000 in your browser to accept the security warning about the untrusted self-signed certificate. Skipping this step will cause the extension app to not load within Symphony Messaging in the next step.
There are 2 ways to load an extension app into Symphony Messaging. For development purposes, we will be using the bundle injection method to temporarily load the app into the current session.
We can now load the app by injecting the bundle URL as a parameter named bundle
behind a pod URL. For example, if you are using the developer sandbox located at develop2.symphony.com, visit the following URL in your browser:
Acknowledge the warning about being in developer mode. You should notice that a new left navigation item appears and triggers an alert when pressed.
Now that you know how to build a basic extension app, you can continue to use the ADK in building out the rest of your app, depending on what type of app you require.
This guide will provide an overview on how to use the Symphony Messaging App Developer Kit (ADK) to build an extension app that has app views. This app will add entries into the left navigation that will each launch a separate app view. The project will use React and ADK's React and Webpack configuration for app view generation.
Create a working directory and initialize it using npm
.
Install the Symphony Messaging ADK, the ADK React and Webpack configurations, React itself, Symphony Messaging UI Toolkit for UI Components, Typescript, Webpack and the required loaders.
Open the project directory in an editor of your choice
Edit the package.json
file, replacing the scripts
section with the following:
This adds two commands:
npm start
for starting the development web server
npm run build
to launch the production build process
Create a .babelrc
file with the following contents:
Create a webpack.config.js
file with the following contents:
Create a tsconfig.json
file with the following contents:
Create a webpack.config.js
file with the following contents:
Each extension app requires a manifest (also known as the bundle.json
file) to describe the application. Create a file named bundle.json
with the following contents:
We are now ready to start building the app. Create a src
directory and a file named index.js
(or index.ts
if you're using TypeScript) within it.
The code ADK.start()
initializes the ADK with an app id (adk-example
) that must correspond with the value provided in the bundle.json
manifest from the previous step.
Once the initialization is complete, we use ADK.navigation.add()
to add an item to the left navigation bar. This item will have the label "ADK View A" and clicking on it will use ADK.modules.open()
to open a module with the app view called view-a
. This parameter can either be an actual navigational route (e.g. view.html
) or a string that will correspond to a JavaScript or TypeScript file with the same name located in the src/views
directory.
Let's proceed to build the app view itself in a file named view-a.jsx
(or view-a.tsx
if you're using TypeScript) within src/views
.
The contents of this app view are entirely arbitrary. You can choose not to use Symphony Messaging's UI Toolkit and employ other component libraries of your choice. The only required line here is calling ADKReact.createView()
at the end, passing in your component and a configuration object pointing to the same app id as before.
For aesthetics, let's define some styling in src/views/view-a.css
.
We can now start the app using:
This starts a local development server on https://localhost:4000
. Note that this is a TLS-enabled site because all extension apps need to be loaded from TLS-enabled sites. However, because this is a development server, the certificate is self-signed and not trusted by any browser.
Visit https://localhost:4000 in your browser to accept the security warning about the untrusted self-signed certificate. Skipping this step will cause the extension app to not load within Symphony Messaging in the next step.
There are 2 ways to load an extension app into Symphony Messaging. For development purposes, we will be using the bundle injection method to temporarily load the app into the current session.
We can now load the app by injecting the bundle URL as a parameter named bundle
behind a pod URL. For example, if you are using the developer sandbox located at develop2.symphony.com, visit the following URL in your browser:
Acknowledge the warning about being in developer mode. You should notice that a new left navigation item appears and opens an app view when clicked on.
Now that you have built a view-driven Extension App, you can proceed to build out your view and add more as required to complete your app.
By default, Symphony Messaging provides secure authentication and authorization for all users. However, if you already have an identity provider that authenticates and authorizes users for resources and you want those resources served by a bot, you will need to implement this integration yourself. This tutorial seeks to demonstrate one such implementation using an OAuth-like flow.
Chat Bot: Listens to commands from the user and contacts the other components
Authorization Server: A server that integrates with the Identity Provider. This server is trusted with privileged access to Symphony Messaging REST APIs.
Resource Server: A server where requested resources are served from. This server does not trust the bot to make arbitrary calls for user-owned data.
This sequence diagram describes the flow of requests between the various components.
We will be using the BDK for this example, so generate a scaffold project using the Bot Generator by following these instructions. We will also be using the Spring Boot integration so ensure that is selected at the framework question in the generator.
Once you have your project, we will first add web capability by changing spring-boot-starter
to spring-boot-starter-web
, then adding com.auth0:java-jwt
for minting tokens.
If you haven't already configured your bot in application.yaml
, do that now. Then, generate a new RSA key pair used to sign and verify tokens.
Add the keys to your project and list their paths in your application.yaml
file. We will also disable the BDK's datafeed since this server will initiate calls from HTTP endpoints rather than Symphony Messaging events.
We will use this object to request for tokens from the Authorization Server, where we want to validate if a message ID is valid and the initiator's user ID matches what was requested.
This service will be our main logic class, serving HTTP requests and processing validation. We will inject in the location of our keys and use the BDK message service and user service.
We'll then add some utilities for loading our keys from disk.
Then we'll add a method to mint a JWT token. The contents of the token are arbitrary, but the recipient needs to understand the structure. We will be setting the user's email as the audience and the grant (or permission scope) to the subject.
Next, we'll add an endpoint to serve the public key. This will be used for resource servers to validate any tokens they receive.
Finally, the core logic goes into the token request process. This method obtains the original message payload from Symphony and validates that the actual sender matches the request. It then goes ahead to extract the intent of the message - assuming that this is for command-based flows where the first non-@mention word is the intent. That word will be used as the grant. The user's email address and grant are then used to mint a JWT token and it is returned.
The resource server will be a standard Spring Boot web server, enabled with Spring Security and will have no knowledge of Symphony. Create an empty maven project and add these dependencies:
Add an application.yaml
to your resources root and add in the URI to obtain the public key on the Authorization Server we added earlier. We will also change the port to 8081
so that it can run concurrently with the Authorization Server that was left defaulted to 8080
.
This project will use a standard Spring Boot main class as follows.
Next, add a Spring Security filter to pre-process incoming requests on the Resource Server. This filter first extracts the JWT token from the Authorization
header, verifies the signature using the Authorization Server's public key, then adds the the user and grant to the security context's Authentication
for use in controllers.
This filter then needs to be added to the Spring Security configuration. Before that happens, the public key is fetched from the Authorization Server and loaded.
Finally, the actual resource is served on a standard controller. This controller does a further check against the grant to ensure it aligns with the resource being requested.
We are now ready to build the bot itself. This project is similar to the Authorization Server in that it uses BDK 2.0 with the Spring Boot integration, so repeat that process once more. We will add the openfeign
project for calling our HTTP endpoints.
We will need to enable feign on the main class by adding the @EnableFeignClients
annotation.
Add the bot configuration itself and then the location of our Authorization and Resource servers.
Copy the MessageIdentityRequest class from the Authorization Server project into this project. Then, create 2 feign clients to communicate with the Authorization and Resource Server endpoints.
Finally, we'll add a command handler to receive a /balance
command from the user. This will first request for a token from the Authorization Server, then use that token to request for the user's bank balance before returning that value in a message to the user.
Launch your project now and go to your Symphony pod, search for your bot and issue a /balance
command. If your current user account is authorized by both the Authorization and Resource Servers, you will see an account balance message appear like on the left pane below. If not, you will observe the message on the right pane instead.
The complete project source code for this tutorial can be found at the following repository. https://github.com/SymphonyPlatformSolutions/symphony-sso-example
The BDK for Python is Symphony Messaging's preferred tooling for Python developers to build bots. It is a library of tools and API bindings that provides simplified configuration and authentication, intuitive message and room management, customizable message templating and an activities API that makes it easy to build bot workflows.
Authenticating your bot is made simple when using the BDK for Python. Once you have your bot and Symphony Messaging environment properly configured, the generated code provides an out of the box implementation for authenticating your bot:
By instantiating a new SymphonyBdk
instance with your config.yaml
file, the BDK loads in your config and authenticates your bot. Once authenticated, your bot is ready to leverage the REST APIs in order to create rich automations and workflows on Symphony Messaging .
BDK for Python also supports OBO (On-Behalf-Of) pattern of authentication, allowing an authenticated bot + extension application to perform operations on behalf of a given user. The BDK's implementation makes it easy to perform the following operations on behalf of a given user:
List the streams of a given user
Initiate connection requests to and determine connection status with other users
Get the presence state of other connected users
Initiate IMs with other users
Send messages and attachments
Set the context user's own presence
To leverage an OBO based workflow, simply instantiate an OBO Session in your bot project. The BDK allows you to instantiate your OBO session from a username or user ID. Once authenticated bots can perform any of the OBO workflows listed above:
BDK for Python makes it easy to manage multiple bot instances within a single project. As long as you have unique configuration files that correspond to different service accounts, you can manage multiple bot instances from a centralized source. To do so, simply instantiate multiple bot instances of the SymphonyBDK
class within your bot project:
The BDK also provides a DatafeedService
interface that makes it easier than ever for bots to manage real-time messages and events. The DatafeedService
interface provides the following methods for your bot to use:
For bots to listen to incoming events and messages, bots must subscribe to a custom RealTimeEventListener
. This RealTimeEventListener
class must implement eventType methods (e.g. onMessageSent()
) along with custom business logic inside.
When a user sends a bot a message, the bot will pick up the event from the datafeed and check to see if an implemented eventType method matches the eventType (MESSAGESENT
) of the inbound event. If there is a corresponding eventType method registered, the bot will execute the business logic inside of this eventType method. Otherwise the bot will not perform an action and will continue to listen for inbound events from the datafeed. An example implementation is provided out of the box by the BDK:
Below is a full list of methods provided by the RealTimeEventListener
class and their corresponding eventTypes. Implement the following methods in order to listen for a given Symphony Messaging event:
For more information on the Symphony datafeed continue here:
A Symphony Messaging workflow can be thought of as a sequence of operations or a repeatable pattern of activities that are organized together in order to transform data, provide a service, or process information. Each of these operations or activities may be completed by a single user, shared between a bot and a user, or shared between multiple actors including bots, users, and even third party systems.
By providing an intuitive Activities API, the BDK for Python makes it simple to define a set of discrete operations or activities for different actors in your system to execute. Ultimately, these activities constitute the building blocks for a powerful Symphony Messaging workflow automation.
Once you have defined a discrete set of activities for different actors in your system to execute, the next step is to organize them together in an intelligent way.
BDK for Python provides an Activities API, an interface that makes it easy to manage user-to-bot interactions or activities. Specifically, the Activities API provides easy access to message and room context, initiator metadata, and an intuitive way to interact with the datafeed, making it easy for bots to listen and reply to different Symphony Messaging events. The methods and logic provided by the Activities API allows for granular control over the entire user-to-bot interaction. This encapsulated logic is easily reused, forming the discrete building blocks of a Symphony Messaging workflow automation.
In order to register activities for your bot instance, you must leverage the ActivityRegistry
class:
There are two different types of activities supported by the BDK:
Command Activity: an activity triggered when a message is sent in an IM or Chatroom.
Form Activity: an activity triggered when a user replies to an Elements form message.
A command-based activity is triggered when a message is sent in an IM or Chatroom. Using the Activities API allows developers to register commands in the following formats:
@bdk-bot /buy
(Slash command with a bot @mention)
/buy 1000 goog
(Slash command without a bot @mention)
Listen for the word 'hello'
(Not a Slash command - Listen for a specific word)
The Activities API also makes it easy for Bots to listen for elements form submissions. Assume the following elements form has been posted into a room with the following attributes:
form id
= "hello-form"
<text-field>
name = "name"
form contains an action button
In order to register a form activity or listen for an incoming elements form submission, bots must register a class that extends the FormReplyActivity
class:
As shown above, the Activities API makes it simple to manage incoming commands, elements form submissions, and access message context making it easy to manage bot-user interactions and create custom workflows.
As shown above, the BDK for Python makes it easy to create a datafeed and listen for events through the RealTimeEventListener
class. In addition, this class makes it easy to access user, message, and room data in context. Each eventType method is implemented with instances of V4Initiator
and V4MessageSent
objects:
Use the V4Initiator
class methods to access the the user data in context:
Use the V4MessageSent
class methods to access message data in context:
Use the V4MessageSent
class methods to access stream data in context:
The Activities API also makes it easy to access relevant user, message, and stream data in context. CommandActivity
classes have access to to this data through the CommandContext
class. This class is instantiated with instances of V4Initiator
and V4MessageSent
objects. Bots are able access to the user, message, and stream data in context through the same methods shown above. Leverage these methods within the on_activity()
method shown below:
FormActivity classes have access to relevant user, form, and stream data through the FormReplyContext
class. This class is instantiated with instances of the V4Initiator
and V4SymphonyElementsAction
class. The V4SymphonyElementsAction
class provides the following methods to access form data in context:
The corresponding Jinja template is shown below:
Using templating you can also create Element Forms. Below is an example of a price enquiry form template:
Complete the previous guide on building an extension app with app views
Let's now add a button to the hashtag hover card and a handler to link the context. Use ADK.buttons.add
to add a new button to the hashtag
zone. To pass the context payload from the controller to the view, there are two options: either using query parameters or using ADK to invoke an exposed controller method from a view.
In this option, we serialize the contents of the context payload and pass it directly into the ADK.modules.open
call as query parameters.
Once the view is opened, you can retrieve the query parameters and deserialize it.
In this option, we store the state of context on the controller, then expose a method to retrieve that state.
Once the view is opened, you can make a call to the exposed getContext
method via the useRemoteExecutor
hook, which returns a promise.
For more details on the JWT, see the section on .
For more details on this, see the section on .
For more details on the JWT, see the section on .
The BDK for Python Github repo can be found here:
The BDK for Python Certification course can be found here:
Note: You must have a corresponding service or bot account setup on your Symphony Messaging instance before authenticating. For more information navigate to the guide.
Please follow our 'Getting Started with OBO' guide using the link .
Note: If you choose to create your own CommandActivity
class, you must implement the matcher()
and on_activity()
methods provided by the AbstractActivity
class. For more information on the implementation of the CommandActivity
class, continue .
Note: If you wish to create your own FormReplyActivity
class, you must implement the methods matcher() and on_activity()
methods provided by the AbstractActivity
class. For more information on the implementation for the FormReplyActivity
class, continue .
The BDK for Python also supports message templating. In order to use message templating, you must leverage the Jinja template engine. Below is an example:
Method
Descriptions
start()
Start the bot's datafeed
stop()
Stop the bot's datafeed
subscribe(RealTimeEventListener)
Subscribe a custom event listener class. Inside this class is where the bulk of your business logic goes.
unsubscribe(RealTimeEventListener)
Unsubscribe from a custom event listener class.
Method
Event Type
onMessageSent()
MESSAGESENT
onInstantMessageCreated()
INSTANTMESSAGECREATED
onMessageSuppressed()
MESSAGESUPPRESSED
onRoomCreated()
ROOMCREATED
onRoomUpdated()
ROOMUPDATED
onRoomDeactivated()
ROOMDEACTIVATED
onRoomReactivated()
ROOMACTIVATED
onUserRequestedToJoinRoom()
USERREQUESTEDTOJOINROOM
onUserJoinedRoom()
USERJOINEDROOM
onUserLeftRoom()
USERLEFTROOM
onRoomMemberPromotedToOwner()
ROOMMEMBERPROMOTEDTOOWNER
onRoomMemberDemotedFromOwner()
ROOMMEMBERDEMOTEDFROMOWNER
onConnectionRequested()
CONNECTIONREQUESTED
onConnectionAccepted()
CONNECTIONACCEPTED
onSymphonyElementsAction()
SYMPHONYELEMENTSACTION
onSharedPost()
SHAREDPOST
Method
User Attribute
initiator.user.user_id
User ID
initiator.user.first_name
First Name
initiator.user.last_name
Last Name
initiator.user.display_name
Display Name
initiator.user.email
initiator.user.user_name
Username
Method
Attribute
event.message.message_id
Message ID
event.message.timestamp
Message Timestamp
event.message.message
Message Text
event.message.shared_message
Shared Message
event.message.data
Message Data
event.message.attachments
Message Attachments
Method
Attribute
event.message.stream.stream_id
Stream ID
event.message.stream.stream_type
Stream Type
event.message.stream.room_name
Room Name
event.message.stream.members
Room Members
event.message.stream.external
External
event.message.stream.cross_pod
Cross Pod
Method
Attribute
context.source_event.stream.stream_id
Elements Stream ID
context.source_event.form_message_id
Elements Message ID
context.source_event.form_id
Elements Form ID
context.source_event.form_values
Elements Form Values
Chat bots can send the same emojis as end users thanks to the MessageML short form tag <emoji shortcode="code" />
.
Here is an example of a message containing emojis:
The following table shows the codes corresponding to their emoji symbol:
Code
Emoji
100
💯
1234
🔢
+1
👍
-1
👎
8ball
🎱
a
🅰
ab
🆎
abc
🔤
abcd
🔡
accept
🉑
aerial_tramway
🚡
airplane
✈
alarm_clock
⏰
alien
👽
ambulance
🚑
anchor
⚓
angel
👼
anger
💢
angry
😠
anguished
😧
ant
🐜
apple
🍎
aquarius
♒
aries
♈
arrow_backward
◀
arrow_double_down
⏬
arrow_double_up
⏫
arrow_down
⬇
arrow_down_small
🔽
arrow_forward
▶
arrow_heading_down
⤵
arrow_heading_up
⤴
arrow_left
⬅
arrow_lower_left
↙
arrow_lower_right
↘
arrow_right
➡
arrow_right_hook
↪
arrow_up
⬆
arrow_up_down
↕
arrow_up_small
🔼
arrow_upper_left
↖
arrow_upper_right
↗
arrows_clockwise
🔃
arrows_counterclockwise
🔄
art
🎨
articulated_lorry
🚛
astonished
😲
athletic_shoe
👟
atm
🏧
b
🅱
baby
👶
baby_bottle
🍼
baby_chick
🐤
baby_symbol
🚼
back
🔙
baggage_claim
🛄
balloon
🎈
ballot_box_with_check
☑
bamboo
🎍
bangbang
‼
bank
🏦
bar_chart
📊
barber
💈
baseball
⚾
basketball
🏀
bath
🛀
bathtub
🛁
battery
🔋
bear
🐻
bee
🐝
beer
🍺
beers
🍻
beetle
🐞
beginner
🔰
bell
🔔
bento
🍱
bicyclist
🚴
bike
🚲
bikini
👙
bird
🐦
birthday
🎂
black_circle
⚫
black_joker
🃏
black_large_square
⬛
black_medium_small_square
◾
black_medium_square
◼
black_nib
✒
black_small_square
▪
black_square_button
🔲
blossom
🌼
blowfish
🐡
blue_book
📘
blue_car
🚙
blue_heart
💙
blush
😊
boar
🐗
boat
⛵
bomb
💣
book
📖
bookmark
🔖
bookmark_tabs
📑
books
📚
boom
💥
boot
👢
bouquet
💐
bow
🙇
bowling
🎳
boy
👦
bread
🍞
bride_with_veil
👰
bridge_at_night
🌉
briefcase
💼
broken_heart
💔
bug
🐛
bulb
💡
bullettrain_front
🚅
bullettrain_side
🚄
bus
🚌
busstop
🚏
bust_in_silhouette
👤
busts_in_silhouette
👥
cactus
🌵
cake
🍰
calendar
📆
calling
📲
camel
🐫
camera
📷
cancer
♋
candy
🍬
capital_abcd
🔠
capricorn
♑
car
🚗
card_index
📇
carousel_horse
🎠
cat
🐱
cat2
🐈
cd
💿
chart
💹
chart_with_downwards_trend
📉
chart_with_upwards_trend
📈
checkered_flag
🏁
cherry_blossom
🌸
chestnut
🌰
chicken
🐔
children_crossing
🚸
chocolate_bar
🍫
christmas_tree
🎄
church
⛪
cinema
🎦
circus_tent
🎪
city_sunrise
🌇
city_sunset
🌆
cl
🆑
clap
👏
clapper
🎬
clipboard
📋
clock1
🕐
clock10
🕙
clock1030
🕥
clock11
🕚
clock1130
🕦
clock12
🕛
clock1230
🕧
clock130
🕜
clock2
🕑
clock230
🕝
clock3
🕒
clock330
🕞
clock4
🕓
clock430
🕟
clock5
🕔
clock530
🕠
clock6
🕕
clock630
🕡
clock7
🕖
clock730
🕢
clock8
🕗
clock830
🕣
clock9
🕘
clock930
🕤
closed_book
📕
closed_lock_with_key
🔐
closed_umbrella
🌂
cloud
☁
clubs
♣
cocktail
🍸
coffee
☕
cold_sweat
😰
collision
💥
computer
💻
confetti_ball
🎊
confounded
😖
confused
😕
congratulations
㊗
construction
🚧
construction_worker
👷
convenience_store
🏪
cookie
🍪
cool
🆒
cop
👮
copyright
©
corn
🌽
couple
👫
couple_with_heart
💑
couplekiss
💏
cow
🐮
cow2
🐄
credit_card
💳
crescent_moon
🌙
crocodile
🐊
crossed_flags
🎌
crown
👑
cry
😢
crying_cat_face
😿
crystal_ball
🔮
cupid
💘
curly_loop
➰
currency_exchange
💱
curry
🍛
custard
🍮
customs
🛃
cyclone
🌀
dancer
💃
dancers
👯
dango
🍡
dart
🎯
dash
💨
date
📅
deciduous_tree
🌳
department_store
🏬
diamond_shape_with_a_dot_inside
💠
diamonds
♦
disappointed
😞
disappointed_relieved
😥
dizzy
💫
dizzy_face
😵
do_not_litter
🚯
dog
🐶
dog2
🐕
dollar
💵
dolls
🎎
dolphin
🐬
door
🚪
dragon
🐉
dragon_face
🐲
dress
👗
dromedary_camel
🐪
droplet
💧
dvd
📀
📧
ear
👂
ear_of_rice
🌾
earth_africa
🌍
earth_americas
🌎
earth_asia
🌏
egg
🍳
eight_pointed_black_star
✴
eight_spoked_asterisk
✳
electric_plug
🔌
elephant
🐘
✉
end
🔚
envelope
✉
envelope_with_arrow
📩
euro
💶
european_castle
🏰
european_post_office
🏤
evergreen_tree
🌲
exclamation
❗
expressionless
😑
eyeglasses
👓
eyes
👀
facepunch
👊
factory
🏭
fallen_leaf
🍂
family
👪
fast_forward
⏩
fax
📠
fearful
😨
feet
🐾
ferris_wheel
🎡
file_folder
📁
fire
🔥
fire_engine
🚒
fireworks
🎆
first_quarter_moon
🌓
first_quarter_moon_with_face
🌛
fish
🐟
fish_cake
🍥
fishing_pole_and_fish
🎣
fist
✊
flags
🎏
flashlight
🔦
floppy_disk
💾
flower_playing_cards
🎴
flushed
😳
foggy
🌁
football
🏈
footprints
👣
fork_and_knife
🍴
fountain
⛲
four_leaf_clover
🍀
free
🆓
fried_shrimp
🍤
fries
🍟
frog
🐸
frowning
😦
fuelpump
⛽
full_moon
🌕
full_moon_with_face
🌝
game_die
🎲
gem
💎
gemini
♊
ghost
👻
gift
🎁
gift_heart
💝
girl
👧
globe_with_meridians
🌐
goat
🐐
golf
⛳
grapes
🍇
green_apple
🍏
green_book
📗
green_heart
💚
grey_exclamation
❕
grey_question
❔
grimacing
😬
grin
😁
grinning
😀
guardsman
💂
guitar
🎸
gun
🔫
haircut
💇
hamburger
🍔
hammer
🔨
hamster
🐹
hand
✋
handbag
👜
handshake
hankey
💩
hatched_chick
🐥
hatching_chick
🐣
headphones
🎧
heart
❤
heart_decoration
💟
heart_eyes
😍
heart_eyes_cat
😻
heartbeat
💓
heartpulse
💗
hearts
♥
heavy_check_mark
✔
heavy_division_sign
➗
heavy_dollar_sign
💲
heavy_exclamation_mark
❗
heavy_minus_sign
➖
heavy_multiplication_x
✖
heavy_plus_sign
➕
helicopter
🚁
herb
🌿
hibiscus
🌺
high_brightness
🔆
high_heel
👠
honey_pot
🍯
honeybee
🐝
horse
🐴
horse_racing
🏇
hospital
🏥
hotel
🏨
hotsprings
♨
hourglass
⌛
hourglass_flowing_sand
⏳
house
🏠
house_with_garden
🏡
hushed
😯
ice_cream
🍨
icecream
🍦
id
🆔
ideograph_advantage
🉐
imp
👿
inbox_tray
📥
incoming_envelope
📨
information_desk_person
💁
information_source
ℹ
innocent
😇
interrobang
⁉
iphone
📱
izakaya_lantern
🏮
jack_o_lantern
🎃
japan
🗾
japanese_castle
🏯
japanese_goblin
👺
japanese_ogre
👹
jeans
👖
joy
😂
joy_cat
😹
key
🔑
keycap_ten
🔟
kimono
👘
kiss
💋
kissing
😗
kissing_cat
😽
kissing_closed_eyes
😚
kissing_heart
😘
kissing_smiling_eyes
😙
koala
🐨
koko
🈁
lantern
🏮
large_blue_circle
🔵
large_blue_diamond
🔷
large_orange_diamond
🔶
last_quarter_moon
🌗
last_quarter_moon_with_face
🌜
laughing
😆
leaves
🍃
ledger
📒
left_luggage
🛅
left_right_arrow
↔
leftwards_arrow_with_hook
↩
lemon
🍋
leo
♌
leopard
🐆
libra
♎
light_rail
🚈
link
🔗
lips
👄
lipstick
💄
lock
🔒
lock_with_ink_pen
🔏
lollipop
🍭
loop
➿
loudspeaker
📢
love_hotel
🏩
love_letter
💌
low_brightness
🔅
m
Ⓜ
mag
🔍
mag_right
🔎
mahjong
🀄
mailbox
📫
mailbox_closed
📪
mailbox_with_mail
📬
mailbox_with_no_mail
📭
man
👨
man_with_gua_pi_mao
👲
man_with_turban
👳
mans_shoe
👞
maple_leaf
🍁
mask
😷
massage
💆
meat_on_bone
🍖
mega
📣
melon
🍈
memo
📝
mens
🚹
metro
🚇
microphone
🎤
microscope
🔬
milky_way
🌌
minibus
🚐
minidisc
💽
mobile_phone_off
📴
money_with_wings
💸
moneybag
💰
monkey
🐒
monkey_face
🐵
monorail
🚝
moon
🌔
mortar_board
🎓
mount_fuji
🗻
mountain_bicyclist
🚵
mountain_cableway
🚠
mountain_railway
🚞
mouse
🐭
mouse2
🐁
movie_camera
🎥
moyai
🗿
muscle
💪
mushroom
🍄
musical_keyboard
🎹
musical_note
🎵
musical_score
🎼
mute
🔇
nail_care
💅
name_badge
📛
necktie
👔
negative_squared_cross_mark
❎
neutral_face
😐
new
🆕
new_moon
🌑
new_moon_with_face
🌚
newspaper
📰
ng
🆖
no_bell
🔕
no_bicycles
🚳
no_entry
⛔
no_entry_sign
🚫
no_good
🙅
no_mobile_phones
📵
no_mouth
😶
no_pedestrians
🚷
no_smoking
🚭
non-potable_water
🚱
nose
👃
notebook
📓
notebook_with_decorative_cover
📔
notes
🎶
nut_and_bolt
🔩
o
⭕
o2
🅾
ocean
🌊
octopus
🐙
oden
🍢
office
🏢
ok
🆗
ok_hand
👌
ok_woman
🙆
older_man
👴
older_woman
👵
on
🔛
oncoming_automobile
🚘
oncoming_bus
🚍
oncoming_police_car
🚔
oncoming_taxi
🚖
open_book
📖
open_file_folder
📂
open_hands
👐
open_mouth
😮
ophiuchus
⛎
orange_book
📙
outbox_tray
📤
ox
🐂
package
📦
page_facing_up
📄
page_with_curl
📃
pager
📟
palm_tree
🌴
panda_face
🐼
paperclip
📎
parking
🅿
part_alternation_mark
〽
partly_sunny
⛅
passport_control
🛂
paw_prints
🐾
pear
🍐
pencil
📝
pencil2
✏
penguin
🐧
pensive
😔
performing_arts
🎭
persevere
😣
person_frowning
🙍
person_with_blond_hair
👱
person_with_pouting_face
🙎
phone
☎
pig
🐷
pig2
🐖
pig_nose
🐽
pill
💊
pisces
♓
pizza
🍕
point_down
👇
point_left
👈
point_right
👉
point_up
☝
point_up_2
👆
police_car
🚓
poodle
🐩
post_office
🏣
postal_horn
📯
postbox
📮
potable_water
🚰
pouch
👝
poultry_leg
🍗
pound
💷
pouting_cat
😾
pray
🙏
princess
👸
punch
👊
purple_heart
💜
purse
👛
pushpin
📌
put_litter_in_its_place
🚮
question
❓
rabbit
🐰
rabbit2
🐇
racehorse
🐎
radio
📻
radio_button
🔘
rage
😡
railway_car
🚃
rainbow
🌈
raised_hand
✋
raised_hands
🙌
raising_hand
🙋
ram
🐏
ramen
🍜
rat
🐀
recycle
♻
red_car
🚗
red_circle
🔴
registered
®
relaxed
☺
relieved
😌
repeat
🔁
repeat_one
🔂
restroom
🚻
revolving_hearts
💞
rewind
⏪
ribbon
🎀
rice
🍚
rice_ball
🍙
rice_cracker
🍘
rice_scene
🎑
ring
💍
rocket
🚀
roller_coaster
🎢
rooster
🐓
rose
🌹
rotating_light
🚨
round_pushpin
📍
rowboat
🚣
rugby_football
🏉
runner
🏃
running
🏃
running_shirt_with_sash
🎽
sa
🈂
sagittarius
♐
sailboat
⛵
sake
🍶
sandal
👡
santa
🎅
satellite
📡
satisfied
😆
saxophone
🎷
school
🏫
school_satchel
🎒
scissors
✂
scorpius
♏
scream
😱
scream_cat
🙀
scroll
📜
seat
💺
secret
㊙
seedling
🌱
shaved_ice
🍧
sheep
🐑
shell
🐚
ship
🚢
shirt
👕
shoe
👞
shower
🚿
signal_strength
📶
six_pointed_star
🔯
ski
🎿
skull
💀
sleeping
😴
sleepy
😪
slot_machine
🎰
small_blue_diamond
🔹
small_orange_diamond
🔸
small_red_triangle
🔺
small_red_triangle_down
🔻
smile
😄
smile_cat
😸
smiley
😃
smiley_cat
😺
smiling_imp
😈
smirk
😏
smirk_cat
😼
smoking
🚬
snail
🐌
snake
🐍
snowboarder
🏂
snowflake
❄
snowman
⛄
sob
😭
soccer
⚽
soon
🔜
sos
🆘
sound
🔉
space_invader
👾
spades
♠
spaghetti
🍝
sparkle
❇
sparkler
🎇
sparkles
✨
sparkling_heart
💖
speaker
🔊
speech_balloon
💬
speedboat
🚤
star
⭐
star2
🌟
stars
🌃
station
🚉
statue_of_liberty
🗽
steam_locomotive
🚂
stew
🍲
straight_ruler
📏
stuck_out_tongue
😛
stuck_out_tongue_closed_eyes
😝
stuck_out_tongue_winking_eye
😜
sun_with_face
🌞
sunflower
🌻
sunglasses
😎
sunny
☀
sunrise
🌅
sunrise_over_mountains
🌄
surfer
🏄
sushi
🍣
suspension_railway
🚟
sweat
😓
sweat_smile
😅
sweet_potato
🍠
swimmer
🏊
symbols
🔣
syringe
💉
tada
🎉
tanabata_tree
🎋
tangerine
🍊
taurus
♉
taxi
🚕
tea
🍵
telephone
☎
telephone_receiver
📞
telescope
🔭
tennis
🎾
tent
⛺
thought_balloon
💭
thumbsdown
👎
thumbsup
👍
ticket
🎫
tiger
🐯
tiger2
🐅
tired_face
😫
tm
™
toilet
🚽
tokyo_tower
🗼
tomato
🍅
tongue
👅
top
🔝
tophat
🎩
tractor
🚜
traffic_light
🚥
train
🚃
train2
🚆
tram
🚊
triangular_flag_on_post
🚩
triangular_ruler
📐
trident
🔱
triumph
😤
trolleybus
🚎
trophy
🏆
tropical_drink
🍹
tropical_fish
🐠
truck
🚚
trumpet
🎺
tshirt
👕
tulip
🌷
turtle
🐢
tv
📺
twisted_rightwards_arrows
🔀
two_hearts
💕
two_men_holding_hands
👬
two_women_holding_hands
👭
u5272
🈹
u5408
🈴
u55b6
🈺
u6307
🈯
u6708
🈷
u6709
🈶
u6e80
🈵
u7121
🈚
u7533
🈸
u7981
🈲
u7a7a
🈳
umbrella
☔
unamused
😒
underage
🔞
unlock
🔓
up
🆙
v
✌
vertical_traffic_light
🚦
vhs
📼
vibration_mode
📳
video_camera
📹
video_game
🎮
violin
🎻
virgo
♍
volcano
🌋
vs
🆚
walking
🚶
waning_crescent_moon
🌘
waning_gibbous_moon
🌖
warning
⚠
watch
⌚
water_buffalo
🐃
watermelon
🍉
wave
👋
wavy_dash
〰
waxing_crescent_moon
🌒
waxing_gibbous_moon
🌔
wc
🚾
weary
😩
wedding
💒
whale
🐳
whale2
🐋
wheelchair
♿
white_check_mark
✅
white_circle
⚪
white_flower
💮
white_large_square
⬜
white_medium_small_square
◽
white_medium_square
◻
white_small_square
▫
white_square_button
🔳
wind_chime
🎐
wine_glass
🍷
wink
😉
wolf
🐺
woman
👩
womans_clothes
👚
womans_hat
👒
womens
🚺
worried
😟
wrench
🔧
x
❌
yellow_heart
💛
yen
💴
yum
😋
zap
⚡
zzz
💤
angel_tone1
👼🏻
angel_tone2
👼🏼
angel_tone3
👼🏽
angel_tone4
👼🏾
angel_tone5
👼🏿
asterisk
*⃣
baby_tone1
👶🏻
baby_tone2
👶🏼
baby_tone3
👶🏽
baby_tone4
👶🏾
baby_tone5
👶🏿
basketball_player_tone1
⛹🏻
basketball_player_tone2
⛹🏼
basketball_player_tone3
⛹🏽
basketball_player_tone4
⛹🏾
basketball_player_tone5
⛹🏿
bath_tone1
🛀🏻
bath_tone2
🛀🏼
bath_tone3
🛀🏽
bath_tone4
🛀🏾
bath_tone5
🛀🏿
bicyclist_tone1
🚴🏻
bicyclist_tone2
🚴🏼
bicyclist_tone3
🚴🏽
bicyclist_tone4
🚴🏾
bicyclist_tone5
🚴🏿
bow_tone1
🙇🏻
bow_tone2
🙇🏼
bow_tone3
🙇🏽
bow_tone4
🙇🏾
bow_tone5
🙇🏿
boy_tone1
👦🏻
boy_tone2
👦🏼
boy_tone3
👦🏽
boy_tone4
👦🏾
boy_tone5
👦🏿
bride_with_veil_tone1
👰🏻
bride_with_veil_tone2
👰🏼
bride_with_veil_tone3
👰🏽
bride_with_veil_tone4
👰🏾
bride_with_veil_tone5
👰🏿
call_me_tone1
🤙🏻
call_me_tone2
🤙🏼
call_me_tone3
🤙🏽
call_me_tone4
🤙🏾
call_me_tone5
🤙🏿
cartwheel_tone1
🤸🏻
cartwheel_tone2
🤸🏼
cartwheel_tone3
🤸🏽
cartwheel_tone4
🤸🏾
cartwheel_tone5
🤸🏿
clap_tone1
👏🏻
clap_tone2
👏🏼
clap_tone3
👏🏽
clap_tone4
👏🏾
clap_tone5
👏🏿
cn
🇨🇳
construction_worker_tone1
👷🏻
construction_worker_tone2
👷🏼
construction_worker_tone3
👷🏽
construction_worker_tone4
👷🏾
construction_worker_tone5
👷🏿
cop_tone1
👮🏻
cop_tone2
👮🏼
cop_tone3
👮🏽
cop_tone4
👮🏾
cop_tone5
👮🏿
couple_mm
👨❤👨
couple_ww
👩❤👩
dancer_tone1
💃🏻
dancer_tone2
💃🏼
dancer_tone3
💃🏽
dancer_tone4
💃🏾
dancer_tone5
💃🏿
de
🇩🇪
ear_tone1
👂🏻
ear_tone2
👂🏼
ear_tone3
👂🏽
ear_tone4
👂🏾
ear_tone5
👂🏿
eight
8️⃣
es
🇪🇸
eye_in_speech_bubble
👁🗨
face_palm_tone1
🤦🏻
face_palm_tone2
🤦🏼
face_palm_tone3
🤦🏽
face_palm_tone4
🤦🏾
face_palm_tone5
🤦🏿
family_mmb
👨👨👦
family_mmbb
👨👨👦👦
family_mmg
👨👨👧
family_mmgb
👨👨👧👦
family_mmgg
👨👨👧👧
family_mwbb
👨👩👦👦
family_mwg
👨👩👧
family_mwgb
👨👩👧👦
family_mwgg
👨👩👧👧
family_wwb
👩👩👦
family_wwbb
👩👩👦👦
family_wwg
👩👩👧
family_wwgb
👩👩👧👦
family_wwgg
👩👩👧👧
fingers_crossed_tone1
🤞🏻
fingers_crossed_tone2
🤞🏼
fingers_crossed_tone3
🤞🏽
fingers_crossed_tone4
🤞🏾
fingers_crossed_tone5
🤞🏿
fist_tone1
✊🏻
fist_tone2
✊🏼
fist_tone3
✊🏽
fist_tone4
✊🏾
fist_tone5
✊🏿
five
5️⃣
flag_ac
🇦🇨
flag_ad
🇦🇩
flag_ae
🇦🇪
flag_af
🇦🇫
flag_ag
🇦🇬
flag_ai
🇦🇮
flag_al
🇦🇱
flag_am
🇦🇲
flag_ao
🇦🇴
flag_aq
🇦🇶
flag_ar
🇦🇷
flag_as
🇦🇸
flag_at
🇦🇹
flag_au
🇦🇺
flag_aw
🇦🇼
flag_ax
🇦🇽
flag_az
🇦🇿
flag_ba
🇧🇦
flag_bb
🇧🇧
flag_bd
🇧🇩
flag_be
🇧🇪
flag_bf
🇧🇫
flag_bg
🇧🇬
flag_bh
🇧🇭
flag_bi
🇧🇮
flag_bj
🇧🇯
flag_bl
🇧🇱
flag_bm
🇧🇲
flag_bn
🇧🇳
flag_bo
🇧🇴
flag_bq
🇧🇶
flag_br
🇧🇷
flag_bs
🇧🇸
flag_bt
🇧🇹
flag_bv
🇧🇻
flag_bw
🇧🇼
flag_by
🇧🇾
flag_bz
🇧🇿
flag_ca
🇨🇦
flag_cc
🇨🇨
flag_cd
🇨🇩
flag_cf
🇨🇫
flag_cg
🇨🇬
flag_ch
🇨🇭
flag_ci
🇨🇮
flag_ck
🇨🇰
flag_cl
🇨🇱
flag_cm
🇨🇲
flag_cn
🇨🇳
flag_co
🇨🇴
flag_cp
🇨🇵
flag_cr
🇨🇷
flag_cu
🇨🇺
flag_cv
🇨🇻
flag_cw
🇨🇼
flag_cx
🇨🇽
flag_cy
🇨🇾
flag_cz
🇨🇿
flag_de
🇩🇪
flag_dg
🇩🇬
flag_dj
🇩🇯
flag_dk
🇩🇰
flag_dm
🇩🇲
flag_do
🇩🇴
flag_dz
🇩🇿
flag_ea
🇪🇦
flag_ec
🇪🇨
flag_ee
🇪🇪
flag_eg
🇪🇬
flag_eh
🇪🇭
flag_er
🇪🇷
flag_es
🇪🇸
flag_et
🇪🇹
flag_eu
🇪🇺
flag_fi
🇫🇮
flag_fj
🇫🇯
flag_fk
🇫🇰
flag_fm
🇫🇲
flag_fo
🇫🇴
flag_fr
🇫🇷
flag_ga
🇬🇦
flag_gb
🇬🇧
flag_gd
🇬🇩
flag_ge
🇬🇪
flag_gf
🇬🇫
flag_gg
🇬🇬
flag_gh
🇬🇭
flag_gi
🇬🇮
flag_gl
🇬🇱
flag_gm
🇬🇲
flag_gn
🇬🇳
flag_gp
🇬🇵
flag_gq
🇬🇶
flag_gr
🇬🇷
flag_gs
🇬🇸
flag_gt
🇬🇹
flag_gu
🇬🇺
flag_gw
🇬🇼
flag_gy
🇬🇾
flag_hk
🇭🇰
flag_hm
🇭🇲
flag_hn
🇭🇳
flag_hr
🇭🇷
flag_ht
🇭🇹
flag_hu
🇭🇺
flag_ic
🇮🇨
flag_id
🇮🇩
flag_ie
🇮🇪
flag_il
🇮🇱
flag_im
🇮🇲
flag_in
🇮🇳
flag_io
🇮🇴
flag_iq
🇮🇶
flag_ir
🇮🇷
flag_is
🇮🇸
flag_it
🇮🇹
flag_je
🇯🇪
flag_jm
🇯🇲
flag_jo
🇯🇴
flag_jp
🇯🇵
flag_ke
🇰🇪
flag_kg
🇰🇬
flag_kh
🇰🇭
flag_ki
🇰🇮
flag_km
🇰🇲
flag_kn
🇰🇳
flag_kp
🇰🇵
flag_kr
🇰🇷
flag_kw
🇰🇼
flag_ky
🇰🇾
flag_kz
🇰🇿
flag_la
🇱🇦
flag_lb
🇱🇧
flag_lc
🇱🇨
flag_li
🇱🇮
flag_lk
🇱🇰
flag_lr
🇱🇷
flag_ls
🇱🇸
flag_lt
🇱🇹
flag_lu
🇱🇺
flag_lv
🇱🇻
flag_ly
🇱🇾
flag_ma
🇲🇦
flag_mc
🇲🇨
flag_md
🇲🇩
flag_me
🇲🇪
flag_mf
🇲🇫
flag_mg
🇲🇬
flag_mh
🇲🇭
flag_mk
🇲🇰
flag_ml
🇲🇱
flag_mm
🇲🇲
flag_mn
🇲🇳
flag_mo
🇲🇴
flag_mp
🇲🇵
flag_mq
🇲🇶
flag_mr
🇲🇷
flag_ms
🇲🇸
flag_mt
🇲🇹
flag_mu
🇲🇺
flag_mv
🇲🇻
flag_mw
🇲🇼
flag_mx
🇲🇽
flag_my
🇲🇾
flag_mz
🇲🇿
flag_na
🇳🇦
flag_nc
🇳🇨
flag_ne
🇳🇪
flag_nf
🇳🇫
flag_ng
🇳🇬
flag_ni
🇳🇮
flag_nl
🇳🇱
flag_no
🇳🇴
flag_np
🇳🇵
flag_nr
🇳🇷
flag_nu
🇳🇺
flag_nz
🇳🇿
flag_om
🇴🇲
flag_pa
🇵🇦
flag_pe
🇵🇪
flag_pf
🇵🇫
flag_pg
🇵🇬
flag_ph
🇵🇭
flag_pk
🇵🇰
flag_pl
🇵🇱
flag_pm
🇵🇲
flag_pn
🇵🇳
flag_pr
🇵🇷
flag_ps
🇵🇸
flag_pt
🇵🇹
flag_pw
🇵🇼
flag_py
🇵🇾
flag_qa
🇶🇦
flag_re
🇷🇪
flag_ro
🇷🇴
flag_rs
🇷🇸
flag_ru
🇷🇺
flag_rw
🇷🇼
flag_sa
🇸🇦
flag_sb
🇸🇧
flag_sc
🇸🇨
flag_sd
🇸🇩
flag_se
🇸🇪
flag_sg
🇸🇬
flag_sh
🇸🇭
flag_si
🇸🇮
flag_sj
🇸🇯
flag_sk
🇸🇰
flag_sl
🇸🇱
flag_sm
🇸🇲
flag_sn
🇸🇳
flag_so
🇸🇴
flag_sr
🇸🇷
flag_ss
🇸🇸
flag_st
🇸🇹
flag_sv
🇸🇻
flag_sx
🇸🇽
flag_sy
🇸🇾
flag_sz
🇸🇿
flag_ta
🇹🇦
flag_tc
🇹🇨
flag_td
🇹🇩
flag_tf
🇹🇫
flag_tg
🇹🇬
flag_th
🇹🇭
flag_tj
🇹🇯
flag_tk
🇹🇰
flag_tl
🇹🇱
flag_tm
🇹🇲
flag_tn
🇹🇳
flag_to
🇹🇴
flag_tr
🇹🇷
flag_tt
🇹🇹
flag_tv
🇹🇻
flag_tw
🇹🇼
flag_tz
🇹🇿
flag_ua
🇺🇦
flag_ug
🇺🇬
flag_um
🇺🇲
flag_us
🇺🇸
flag_uy
🇺🇾
flag_uz
🇺🇿
flag_va
🇻🇦
flag_vc
🇻🇨
flag_ve
🇻🇪
flag_vg
🇻🇬
flag_vi
🇻🇮
flag_vn
🇻🇳
flag_vu
🇻🇺
flag_wf
🇼🇫
flag_ws
🇼🇸
flag_xk
🇽🇰
flag_ye
🇾🇪
flag_yt
🇾🇹
flag_za
🇿🇦
flag_zm
🇿🇲
flag_zw
🇿🇼
fleur-de-lis
⚜
four
4️⃣
fr
🇫🇷
gb
🇬🇧
girl_tone1
👧🏻
girl_tone2
👧🏼
girl_tone3
👧🏽
girl_tone4
👧🏾
girl_tone5
👧🏿
guardsman_tone1
💂🏻
guardsman_tone2
💂🏼
guardsman_tone3
💂🏽
guardsman_tone4
💂🏾
guardsman_tone5
💂🏿
haircut_tone1
💇🏻
haircut_tone2
💇🏼
haircut_tone3
💇🏽
haircut_tone4
💇🏾
haircut_tone5
💇🏿
hand_splayed_tone1
🖐🏻
hand_splayed_tone2
🖐🏼
hand_splayed_tone3
🖐🏽
hand_splayed_tone4
🖐🏾
hand_splayed_tone5
🖐🏿
handball_tone1
🤾🏻
handball_tone2
🤾🏼
handball_tone3
🤾🏽
handball_tone4
🤾🏾
handball_tone5
🤾🏿
handshake_tone1
🤝🏻
handshake_tone2
🤝🏼
handshake_tone3
🤝🏽
handshake_tone4
🤝🏾
handshake_tone5
🤝🏿
hash
#️⃣
horse_racing_tone1
🏇🏻
horse_racing_tone2
🏇🏼
horse_racing_tone3
🏇🏽
horse_racing_tone4
🏇🏾
horse_racing_tone5
🏇🏿
information_desk_person_tone1
💁🏻
information_desk_person_tone2
💁🏼
information_desk_person_tone3
💁🏽
information_desk_person_tone4
💁🏾
information_desk_person_tone5
💁🏿
it
🇮🇹
jp
🇯🇵
juggling_tone1
🤹🏻
juggling_tone2
🤹🏼
juggling_tone3
🤹🏽
juggling_tone4
🤹🏾
juggling_tone5
🤹🏿
kiss_mm
👨❤💋👨
kiss_ww
👩❤💋👩
kr
🇰🇷
left_facing_fist_tone1
🤛🏻
left_facing_fist_tone2
🤛🏼
left_facing_fist_tone3
🤛🏽
left_facing_fist_tone4
🤛🏾
left_facing_fist_tone5
🤛🏿
lifter_tone1
🏋🏻
lifter_tone2
🏋🏼
lifter_tone3
🏋🏽
lifter_tone4
🏋🏾
lifter_tone5
🏋🏿
man_dancing_tone1
🕺🏻
man_dancing_tone2
🕺🏼
man_dancing_tone3
🕺🏽
man_dancing_tone4
🕺🏾
man_dancing_tone5
🕺🏿
man_in_tuxedo_tone1
🤵🏻
man_in_tuxedo_tone2
🤵🏼
man_in_tuxedo_tone3
🤵🏽
man_in_tuxedo_tone4
🤵🏾
man_in_tuxedo_tone5
🤵🏿
man_tone1
👨🏻
man_tone2
👨🏼
man_tone3
👨🏽
man_tone4
👨🏾
man_tone5
👨🏿
man_with_gua_pi_mao_tone1
👲🏻
man_with_gua_pi_mao_tone2
👲🏼
man_with_gua_pi_mao_tone3
👲🏽
man_with_gua_pi_mao_tone4
👲🏾
man_with_gua_pi_mao_tone5
👲🏿
man_with_turban_tone1
👳🏻
man_with_turban_tone2
👳🏼
man_with_turban_tone3
👳🏽
man_with_turban_tone4
👳🏾
man_with_turban_tone5
👳🏿
massage_tone1
💆🏻
massage_tone2
💆🏼
massage_tone3
💆🏽
massage_tone4
💆🏾
massage_tone5
💆🏿
metal_tone1
🤘🏻
metal_tone2
🤘🏼
metal_tone3
🤘🏽
metal_tone4
🤘🏾
metal_tone5
🤘🏿
mountain_bicyclist_tone1
🚵🏻
mountain_bicyclist_tone2
🚵🏼
mountain_bicyclist_tone3
🚵🏽
mountain_bicyclist_tone4
🚵🏾
mountain_bicyclist_tone5
🚵🏿
mrs_claus_tone1
🤶🏻
mrs_claus_tone2
🤶🏼
mrs_claus_tone3
🤶🏽
mrs_claus_tone4
🤶🏾
mrs_claus_tone5
🤶🏿
muscle_tone1
💪🏻
muscle_tone2
💪🏼
muscle_tone3
💪🏽
muscle_tone4
💪🏾
muscle_tone5
💪🏿
nail_care_tone1
💅🏻
nail_care_tone2
💅🏼
nail_care_tone3
💅🏽
nail_care_tone4
💅🏾
nail_care_tone5
💅🏿
nine
9️⃣
no_good_tone1
🙅🏻
no_good_tone2
🙅🏼
no_good_tone3
🙅🏽
no_good_tone4
🙅🏾
no_good_tone5
🙅🏿
non-potable-water
🚱
nose_tone1
👃🏻
nose_tone2
👃🏼
nose_tone3
👃🏽
nose_tone4
👃🏾
nose_tone5
👃🏿
ok_hand_tone1
👌🏻
ok_hand_tone2
👌🏼
ok_hand_tone3
👌🏽
ok_hand_tone4
👌🏾
ok_hand_tone5
👌🏿
ok_woman_tone1
🙆🏻
ok_woman_tone2
🙆🏼
ok_woman_tone3
🙆🏽
ok_woman_tone4
🙆🏾
ok_woman_tone5
🙆🏿
older_man_tone1
👴🏻
older_man_tone2
👴🏼
older_man_tone3
👴🏽
older_man_tone4
👴🏾
older_man_tone5
👴🏿
older_woman_tone1
👵🏻
older_woman_tone2
👵🏼
older_woman_tone3
👵🏽
older_woman_tone4
👵🏾
older_woman_tone5
👵🏿
one
1️⃣
open_hands_tone1
👐🏻
open_hands_tone2
👐🏼
open_hands_tone3
👐🏽
open_hands_tone4
👐🏾
open_hands_tone5
👐🏿
person_frowning_tone1
🙍🏻
person_frowning_tone2
🙍🏼
person_frowning_tone3
🙍🏽
person_frowning_tone4
🙍🏾
person_frowning_tone5
🙍🏿
person_with_blond_hair_tone1
👱🏻
person_with_blond_hair_tone2
👱🏼
person_with_blond_hair_tone3
👱🏽
person_with_blond_hair_tone4
👱🏾
person_with_blond_hair_tone5
👱🏿
person_with_pouting_face_tone1
🙎🏻
person_with_pouting_face_tone2
🙎🏼
person_with_pouting_face_tone3
🙎🏽
person_with_pouting_face_tone4
🙎🏾
person_with_pouting_face_tone5
🙎🏿
point_down_tone1
👇🏻
point_down_tone2
👇🏼
point_down_tone3
👇🏽
point_down_tone4
👇🏾
point_down_tone5
👇🏿
point_left_tone1
👈🏻
point_left_tone2
👈🏼
point_left_tone3
👈🏽
point_left_tone4
👈🏾
point_left_tone5
👈🏿
point_right_tone1
👉🏻
point_right_tone2
👉🏼
point_right_tone3
👉🏽
point_right_tone4
👉🏾
point_right_tone5
👉🏿
point_up_2_tone1
👆🏻
point_up_2_tone2
👆🏼
point_up_2_tone3
👆🏽
point_up_2_tone4
👆🏾
point_up_2_tone5
👆🏿
point_up_tone1
☝🏻
point_up_tone2
☝🏼
point_up_tone3
☝🏽
point_up_tone4
☝🏾
point_up_tone5
☝🏿
pray_tone1
🙏🏻
pray_tone2
🙏🏼
pray_tone3
🙏🏽
pray_tone4
🙏🏾
pray_tone5
🙏🏿
pregnant_woman_tone1
🤰🏻
pregnant_woman_tone2
🤰🏼
pregnant_woman_tone3
🤰🏽
pregnant_woman_tone4
🤰🏾
pregnant_woman_tone5
🤰🏿
prince_tone1
🤴🏻
prince_tone2
🤴🏼
prince_tone3
🤴🏽
prince_tone4
🤴🏾
prince_tone5
🤴🏿
princess_tone1
👸🏻
princess_tone2
👸🏼
princess_tone3
👸🏽
princess_tone4
👸🏾
princess_tone5
👸🏿
punch_tone1
👊🏻
punch_tone2
👊🏼
punch_tone3
👊🏽
punch_tone4
👊🏾
punch_tone5
👊🏿
rainbow_flag
🏳🌈
raised_back_of_hand_tone1
🤚🏻
raised_back_of_hand_tone2
🤚🏼
raised_back_of_hand_tone3
🤚🏽
raised_back_of_hand_tone4
🤚🏾
raised_back_of_hand_tone5
🤚🏿
raised_hand_tone1
✋🏻
raised_hand_tone2
✋🏼
raised_hand_tone3
✋🏽
raised_hand_tone4
✋🏾
raised_hand_tone5
✋🏿
raised_hands_tone1
🙌🏻
raised_hands_tone2
🙌🏼
raised_hands_tone3
🙌🏽
raised_hands_tone4
🙌🏾
raised_hands_tone5
🙌🏿
raising_hand_tone1
🙋🏻
raising_hand_tone2
🙋🏼
raising_hand_tone3
🙋🏽
raising_hand_tone4
🙋🏾
raising_hand_tone5
🙋🏿
right_facing_fist_tone1
🤜🏻
right_facing_fist_tone2
🤜🏼
right_facing_fist_tone3
🤜🏽
right_facing_fist_tone4
🤜🏾
right_facing_fist_tone5
🤜🏿
rowboat_tone1
🚣🏻
rowboat_tone2
🚣🏼
rowboat_tone3
🚣🏽
rowboat_tone4
🚣🏾
rowboat_tone5
🚣🏿
ru
🇷🇺
runner_tone1
🏃🏻
runner_tone2
🏃🏼
runner_tone3
🏃🏽
runner_tone4
🏃🏾
runner_tone5
🏃🏿
santa_tone1
🎅🏻
santa_tone2
🎅🏼
santa_tone3
🎅🏽
santa_tone4
🎅🏾
santa_tone5
🎅🏿
selfie_tone1
🤳🏻
selfie_tone2
🤳🏼
selfie_tone3
🤳🏽
selfie_tone4
🤳🏾
selfie_tone5
🤳🏿
seven
7️⃣
shrug_tone1
🤷🏻
shrug_tone2
🤷🏼
shrug_tone3
🤷🏽
shrug_tone4
🤷🏾
shrug_tone5
🤷🏿
six
6️⃣
spy_tone1
🕵🏻
spy_tone2
🕵🏼
spy_tone3
🕵🏽
spy_tone4
🕵🏾
spy_tone5
🕵🏿
surfer_tone1
🏄🏻
surfer_tone2
🏄🏼
surfer_tone3
🏄🏽
surfer_tone4
🏄🏾
surfer_tone5
🏄🏿
swimmer_tone1
🏊🏻
swimmer_tone2
🏊🏼
swimmer_tone3
🏊🏽
swimmer_tone4
🏊🏾
swimmer_tone5
🏊🏿
three
3️⃣
thumbsdown_tone1
👎🏻
thumbsdown_tone2
👎🏼
thumbsdown_tone3
👎🏽
thumbsdown_tone4
👎🏾
thumbsdown_tone5
👎🏿
thumbsup_tone1
👍🏻
thumbsup_tone2
👍🏼
thumbsup_tone3
👍🏽
thumbsup_tone4
👍🏾
thumbsup_tone5
👍🏿
two
2️⃣
uk
🇬🇧
us
🇺🇸
v_tone1
✌🏻
v_tone2
✌🏼
v_tone3
✌🏽
v_tone4
✌🏾
v_tone5
✌🏿
vulcan_tone1
🖖🏻
vulcan_tone2
🖖🏼
vulcan_tone3
🖖🏽
vulcan_tone4
🖖🏾
vulcan_tone5
🖖🏿
walking_tone1
🚶🏻
walking_tone2
🚶🏼
walking_tone3
🚶🏽
walking_tone4
🚶🏾
walking_tone5
🚶🏿
water_polo_tone1
🤽🏻
water_polo_tone2
🤽🏼
water_polo_tone3
🤽🏽
water_polo_tone4
🤽🏾
water_polo_tone5
🤽🏿
wave_tone1
👋🏻
wave_tone2
👋🏼
wave_tone3
👋🏽
wave_tone4
👋🏾
wave_tone5
👋🏿
woman_tone1
👩🏻
woman_tone2
👩🏼
woman_tone3
👩🏽
woman_tone4
👩🏾
woman_tone5
👩🏿
writing_hand_tone1
✍🏻
writing_hand_tone2
✍🏼
writing_hand_tone3
✍🏽
writing_hand_tone4
✍🏾
writing_hand_tone5
✍🏿
zero
0️⃣