Migration guide
This migration guide helps you migrate from Signaling 1.x
to Signaling 2.x
.
In December 2023, Agora released Signaling 2.x
in response to market and industry needs. Signaling 2.x
brings significant innovation to users in terms of:
-
Features coverage: Signaling
2.x
expands its scope to encompass a broader range of business use-cases through the introduction of functional modules includingChannel
,Message
,Topic
,Presence
,Storage
, andLock
. These additions empower you to shift your attention from fundamental feature implementation to fostering business innovation. -
Performance improvement: Signaling
2.x
revamps the backend architecture, enhancing overall performance through optimized network connections. This overhaul results in sustained low-latency, higher reliability, increased concurrency, and seamless scalability for real-time network access. With these advancements, businesses can confidently rely on the backend, to ensure both optimal performance and high-quality service. -
Experience optimization: All documents, including user guides and API references, have been optimized and furnished more comprehensive sample code. These enhancements empower developers with a cost-effective learning experience, enable swift integration and improve development efficiency when utilizing the SDK.
For a seamless transition from Signaling 1.x
to 2.x
, refer to the following document. This resource is designed to expedite the migration process, ensuring that you can swiftly leverage the new features in Signaling 2.x
and start reaping their benefits.
Activate Signaling
To activate Signaling 2.x
, take the following steps:
- Log in to Agora Console
- Create a new Agora project or choose an existing project from the project list.
- Select the project on the Project Management page and click the pencil icon to configure it.
- Go to All features > Signaling > Basic information and select a data center in the dropdown.
- Go to Subscriptions > Signaling and subscribe to a plan.
- Copy the App ID for your project for use in your code.
Signaling version 2.x
differs from 1.x
in the support of naming character sets. For example, 2.x
does not support channel names, user names, or topic names that start with _
or contain .
characters. When upgrading from 1.x
to 2.x
or using both versions together, be aware of possible incompatibilities caused by character set differences. Best practice is to use the character set supported by 2.x
when migrating. See Channel naming for the character set supported by 2.x
.
Integrate the SDK
You can integrate the Signaling 2.x
SDK into your app either using a CDN or through Maven Central. The integration methods are as follows:
-
For
1.x
: -
For
2.x
:InfoNote that the SDK package names for
2.x
and1.x
are different. The2.x
package name isagora-rtm
, while the1.x
package name isrtm-sdk
.
Initialize an RTM Client instance
Signaling 2.x
makes adjustments to initialization parameters. It introduces new features such as end-to-end encryption, and cloud proxy. Refer to the API Reference for details. Additionally, 2.x
provides richer error information when invoking interfaces, allowing you to quickly identify issues. Look up Error Codes for efficient troubleshooting.
-
Sample code for
1.x
: -
Sample code for
2.x
:
Log in to Signaling
The login method for 2.x
differs from 1.x
:
-
Sample code for
1.x
: -
Sample code for
2.x
:
Event notifications
Signaling 2.x
has redesigned the system event notification mechanism and API interfaces. It provides more detailed classification and aggregation of event notification types, and optimizes the payload data structure.
2.x
introduces the following types of event notifications:
Event Type | Description |
---|---|
onMessageEvent | Message event notification: Receives notifications for all message events in subscribed Message channels and topics. |
onPresenceEvent | User presence and custom state change event notification (Presence event notification): Receives notifications for Presence events in subscribed Message Channels and joined Stream Channels, including user online/offline status, and custom temporary state changes. |
onTopicEvent | Topic change event notification: Receives notifications for Topic changes in joined Stream Channels. |
onStorageEvent | Channel and user metadata event notification: Receives notifications for channel Metadata events in subscribed Message channels and joined Stream channels, as well as user metadata events for subscribed users. |
onLockEvent | Lock change event notification: Receives notifications for Lock events in subscribed Message channels and joined Stream channels. |
onConnectionStateChanged | (Deprecated) Network connection state change event notification: Receives notifications for changes in client network connection state. |
onLinkStateEvent | Receives notifications of changes in the client's network connection state, including information such as the connection state before and after the change, service type, operation type that caused the change, reason for the change, and channel list. |
onTokenPrivilegeWillExpire | Receives notifications when the client's token is about to expire. |
For more information on event notifications and payload data structures, see Event listeners.
Consider the example of listening to channel message events:
-
For
1.x
: -
For
2.x
:
The comparison demonstrates significant differences:
-
In
1.x
, channel message event notifications are bound to specific channel instances. Users need to call thecreateChannel()
method to create achannel
instance, then register theonMessageReceived
callback to handle events. The SDK notifies the handler through this callback when a message is received. If multiple channels are involved, this process needs to be repeated for each channel. In2.x
, message event notifications are bound to the client instance globally. You call theaddEventListener()
method to register the callback code once, and it applies to all subscribed channels or topics. This simplifies the implementation for handling message events across multiple channels or topics. -
The payload data structure for message event notifications in
2.x
contains more information, which simplifies implementation of custom business logic.
Channel messages
In version 1.x
, to send a channel message, you needed to:
- Create a Channel instance
- Join the channel
- Send and receive channel messages
The disadvantage to this design is that you cannot independently send messages. You must also receive messages because sending and receiving is coupled. Signaling 2.x
adopts a new Pub/Sub-based model designed to decouple sending and receiving messages. When sending messages, you only need to publish to the specified channel without joining the channel. To receive channel messages, you only need to subscribe to the specified channel. The two operations are independent.
-
Sample code for
1.x
: -
Sample code for
2.x
:
Peer-to-peer messaging
In version 1.x
, peer-to-peer messaging API is used to send messages to specified users. For example, if you want to send a message to the user whose UserId is "Tony", you can do it as follows:
The design of this interface in version 1.x
is tailored to fulfill the requirements of an end-to-end message transmission use-case. In this context, User A seeks to send a message exclusively to User B, and User B only desires to receive this specific message without subscribing to other events. The channel mechanism in version 1.x
is structured based on the Room model. However, this design lacks the capability to decouple message sending and receiving.
In version 2.x
, Agora redesigned the distribution of channel messages using the Pub/Sub model to decouple sending and receiving of messages. You can still implement point-to-point messages using channels. Use the following scheme to implement peer-to-peer messaging with Signaling 2.x
.
-
Each user subscribes to a private channel named
inbox_ + UserID
after initialization, which you call Inbox. Only the user with the specificUserID
can subscribe to this channel. -
To send a peer-to-peer message to a user, just send the message to the user's Inbox.
-
Sample code for
2.x
:NoteTo receive messages, you need to implement themessage
event listening function.
Picture and file messages
For reasons of user data privacy, compliance and cost optimization, Signaling 2.x
no longer directly supports image and file message types. After version 1.5.0
, the related interfaces have been removed. However, this does not mean that you cannot use Signaling to transmit and distribute image and file messages. You can build image and file message functions with the help of Signaling 2.x
and third-party object storage services, such as Amazon S3 or Alibaba Cloud OSS. Not only can you get the best factual message transmission experience, you can also implement more flexible technical solutions. For example, CDN static resource acceleration or image and text moderation. The following code sample shows you how to use Signaling 2.x
and Amazon S3 object storage service to build and send image and file messages.
When using Amazon S3 for static file storage, go to the Amazon S3 console and set the correct user permissions and access policies. Refer to Access Control Best Practices for more information.
User presence and custom status
In Signaling 1.x
, you can subscribe to or query the online status of multiple users, query the number of users in a channel, or obtain the list of online members in a channel. Signaling 2.x
not only retains these features, but also implements upgrades and extends them. In Signaling 2.x
, these capabilities are redefined and implemented in the Presence module. Presence provides the ability to monitor user online, user offline and user status change notifications. Through the Presence module, you can obtain the following information in real time:
- A user joins or leaves the specified channel
- Customize temporary user status and its changes
- Query which channels a specified user has joined or is subscribed to
- Query which users have joined the specified channel and their temporary status data
Call the getOnlineUsers
method to query the number of online users in the specified channel, the list of online users, and the temporary status of online users in real-time.
Call the getUserChannels
method to get a list of channels where the specified user is present, in real-time.
To meet the user status requirements of business use-cases, Signaling 2.x
provides temporary user status capabilities. Customize temporary user status through the setState
method. Users can add their scores, game status, location, mood, and other customized statuses.
You can retrieve a user's online state at any time by using the getState
method, or remove your own state by using the removeState
method. Signaling triggers a presence
event notification of type REMOTE_STATE_CHANGED
when a user's temporary state is changed. See the Presence guide for details on how to use this feature.
Signaling 2.x
makes it very simple to listen to the real-time notification of users entry, exit, timeout and temporary status changes in a channel. To do this, implement the following steps:
- Implement a Presence event listener
- Enable the
withPresence
switch when you join the channel
You can add a Presence event listener as follows:
In Signaling 2.x
real-time notifications have been redesigned. The presence event notification mode refers to how subscribed users of presence events in the channel are notified. There are two available modes:
- Real-time notification mode (Announce)
- Scheduled notification mode (Interval)
You can determine the conditions for switching between the two modes through the Announce Max parameter in the project settings of the Agora Console. The interval notification mode can prevent noisy events caused by too many online users in the channel. For details, see Event Listeners.
User metadata and channel metadata
Signaling 2.x
retains the full functionality of user metadata and channel metadata, with new capabilities such as versioning and locking. It adds optimized interfaces to make these features easier to use.
In 2.x
, user attributes and channel attributes are mounted under the Storage
module. To set channel attributes, refer to the following code:
To learn more about how to get, update, and delete channel attributes, how to use version control and lock control, refer to the Storage guide. The use of user attributes is similar to that of channel attributes.
Events for channel attributes and user attributes are distributed to users through onStorageEvent
type event notifications. To listen to these events:
- Implement the Storage event listener.
- When subscribing to a channel, turn on the
withMetadata
switch.
Restrict access area
Signaling supports the restricted access area feature to comply with the laws and regulations of different countries or regions. After turning on the restricted access area feature, no matter which area the user uses your app from, the SDK will only access the Agora server in the geographical specified area. Signaling 1.x
implements access area limitation as follows.
-
Sample code for
1.x
: -
Sample code for
2.x
:
Other new features
In addition to the enhancements presented in this document, Signaling 2.x
introduces an array of additional features. Choose and implement features that fit the needs of your project. The following table outlines key new features of Signaling 2.x
:
Module | Function | Signaling 2.x API Interface |
---|---|---|
Setup | Create Instance | RtmClient create(RtmConfig config) |
Destroy Instance | RtmClient.release() | |
Token Configuration | login interface with token parameter | |
End-to-End Encryption | RtmEncryptionConfig with encryptionMode , encryptionKey , encryptionSalt parameters | |
Presence Timeout Setting | RtmConfig with presenceTimeout parameter | |
Log Level Setting | RtmLogConfig with level parameter | |
Proxy Configuration | RtmProxyConfig with proxyType , server , port , account , password parameters | |
Event Listener | void addEventListener(RtmEventListener listener) void removeEventListener(RtmEventListener listener) | |
Login Service | void login(String token, ResultCallback<Void> resultCallback) | |
Logout Service | void logout(ResultCallback<Void> resultCallback) | |
Channel | Unsubscribe Channel | void unsubscribe(String... channelIds) |
Subscribe Channel | void subscribe(String channelName, SubscribeOptions options, ResultCallback<Void> resultCallback) | |
Unsubscribe Channel | void unsubscribe(String channelName, ResultCallback<Void> resultCallback) | |
Create Stream Channel | StreamChannel createStreamChannel(String channelName) | |
Join Stream Channel | void join(JoinChannelOptions options, ResultCallback<Void> resultCallback) | |
Leave Stream Channel | void leave(ResultCallback<Void> resultCallback) | |
Destroy Stream Channel | streamChannel.release() | |
Topic | Join Topic | void joinTopic(String topicName, JoinTopicOptions options, ResultCallback<Void> resultCallback) |
Publish Topic Message | void publishTopicMessage(String topicName, String message, TopicMessageOptions options, ResultCallback<Void> resultCallback) | |
Leave Topic | void leaveTopic(String topicName, ResultCallback<Void> resultCallback) | |
Subscribe Topic | void subscribeTopic(String topicName, TopicOptions options, ResultCallback<SubscribeTopicResult> resultCallback) | |
Unsubscribe Topic | void unsubscribeTopic(String topicName, TopicOptions options, ResultCallback<Void> resultCallback) | |
Message | Send Message | void publish(String channelName, String message, PublishOptions options, ResultCallback<Void> resultCallback) |
Presence | Query Channel's Online Users | void getOnlineUsers(String channelName, RtmChannelType channelType, ResultCallback<GetOnlineUsersResponse> resultCallback) |
Query User's Channel | void getUserChannels(String userId, ResultCallback<GetUserChannelsResponse> resultCallback) | |
Set User's Temporary State | void setState(String channelName, RtmChannelType channelType, ArrayList<StateItem> items, ResultCallback<Void> resultCallback) | |
Query User Temporary State | void getState(String channelName, RtmChannelType channelType, String userId, ResultCallback<UserState> resultCallback) | |
Remove User Temporary State | void removeState(String channelName, RtmChannelType channelType, ArrayList<String> keys, ResultCallback<Void> resultCallback) | |
Storage | Set Channel Metadata | void setChannelMetadata(String channelName, RtmChannelType channelType, Metadata data, MetadataOptions options, String lockName, ResultCallback<Void> resultCallback) |
Get Channel Metadata | void getChannelMetadata(String channelName, RtmChannelType channelType, ResultCallback<Metadata> resultCallback) | |
Remove Channel Metadata | void removeChannelMetadata(String channelName, RtmChannelType channelType, Metadata data, MetadataOptions options, String lockName, ResultCallback<Void> resultCallback) | |
Update Channel Metadata | void updateChannelMetadata(String channelName, RtmChannelType channelType, Metadata data, MetadataOptions options, String lockName, ResultCallback<Void> resultCallback) | |
Set User Attributes | void setUserMetadata(String userId, Metadata data, MetadataOptions options, ResultCallback<Void> resultCallback) | |
Get User Attributes | void getUserMetadata(String userId, ResultCallback<Metadata> resultCallback) | |
Remove User Attributes | void removeUserMetadata(String userId, Metadata data, MetadataOptions options, ResultCallback<Void> resultCallback) | |
Update User Attributes | void updateUserMetadata(String userId, Metadata data, MetadataOptions options, ResultCallback<Void> resultCallback) | |
Subscribe User Attributes | void subscribeUserMetadata(String userId, ResultCallback<Void> resultCallback) | |
Unsubscribe User Attributes | void unsubscribeUserMetadata(String userId, ResultCallback<Void> resultCallback) | |
Lock | Set Lock | void setLock(String channelName, RtmChannelType channelType, String lockName, long ttl, ResultCallback<Void> resultCallback) |
Acquire Lock | void acquireLock(String channelName, RtmChannelType channelType, String lockName, boolean retry, ResultCallback<Void> resultCallback) | |
Release Lock | void releaseLock(String channelName, RtmChannelType channelType, String lockName, ResultCallback<Void> resultCallback) | |
Revoke Lock | void revokeLock(String channelName, RtmChannelType channelType, String lockName, String owner, ResultCallback<Void> resultCallback) | |
Query Lock | void getLocks(String channelName, RtmChannelType channelType, ResultCallback<ArrayList<LockDetail>> resultCallback) | |
Remove Lock | void removeLock(String channelName, RtmChannelType channelType, String lockName, ResultCallback<Void> resultCallback) |