Changelog: Twilio Chat Android SDK 4.x
Getting Started? Go for Programmable Chat SDK 7.x for Android
If you are starting out with Twilio's Programmable Chat SDK for Android, we highly recommend starting with version 7.x. Version 7.x is our latest SDK and it brings many new features and improvements to the earliest versions. And it will be much easier to migrate to Twilio Conversations.
Support for 4.x ceased March 5, 2021
Not using 7.x? You should migrate ASAP
If you are already using 4.x, we highly recommend planning your migration to 7.x as soon as possible.
End-of-Life for 1.x occurred on June 30, 2019.
- Fixed a bug when channel attributes are empty when onChannelAddedcalled after re-joining to private channel.
- Fixed a bug when channel attributes are empty when onChannelAddedcalled.
- Reduced SDK footprint.
- Errors are now more human-readable for debugging purposes.
- Fixed a bug related to joinChannelcallback. Sometimes was not called.
- Improved logging.
- Fixed a native crash in makeUserDescriptor().
- Fixed a bug where isOnline()andisNotifiable()inTMUserDescriptoralways returnedfalse.
- Fixed processing failed result in consumption horizon.
- Fixed a bug where calling addorinvitemembers double callbacks while offline.
- Fixed deadlock in getChannel().
- Minimum Android SDK version decreased to 19.
- Fixed a bug where getAndSubscribeUser()method doesn't work anymore after callingUser.unsubscribe().
- Fixed a bug that caused an excessive subscription retry loop on failure, which would lead to an increased network and battery consumption.
- Fixed a bug where getMessageByIndex()sometimes resulted inCould not retrieve Message by indexerror when messages were not cached.
- Included a valid ProGuard configuration with the SDK archive (AAR).
- Improved documentation on getMessagesAfter/Before behavior.
- Fixed a bug where an incorrect message index was received in ChatClientListener.onNewMessageNotification()after callingChatClient.handleNotification().
- Reduced SDK footprint, resulting in a noticeable decrease in final application size compared to release 4.2.0.
- Fixed stack dump symbolication on Сrashlytics. In order to make the symbolication work, we needed to reconfigure linker which led to increase of SDK footprint by 10%. We are working hard on reducing the SDK size in next releases.
- Fixed retrieving member attributes when invoked immediately after client initialization.
- Added the Member.getType()method to obtain information about the type of a member (chat/sms/whatsapp/other).
- Fixed a misleading error message when fetching a user descriptor for a user-less member, or subscribing to a user-less member.
- Fixed a race condition when invoking ChannelListener.onTypingStarted()/.onTypingEnded().
- Fixed a bug when removing a Member from Channel.
- Added Message.getMemberSid()andMessage.getMember()methods to obtain information about the author of a message.
- Improved stability on shutdown, fixed crash issue.
- Fixed memory leaks in media upload.
- Fixed crash on media download in bad networking conditions.
- Improved connection reliability under bad network conditions.
- Fixed unexpected connectivity errors with code 1401: Unable to establish connection to Twilio Sync service.
- Fixed behavior for UpdateAttributesinMessageandUser, now updating with null value calls listener'sonError().
- Improved documentation for Channel Members.
- Increased targetSdkVersion to 28 in preparation for Google SDK policy targeted in August; minSdkVersion remains 21.
- SDK size reduction - it gained over the course of the year, so we sent it to the gym. Native libraries are now repackaged and optimized for size.
- Fixed crash on shutdown
- Fixed crash on restoring after disconnected or backgrounded state
- Fixed memory leaks in media upload
- Removed previously deprecated Message.getTimeStampAsDate(). UseMessage.getDateCreatedAsDate()instead.
- Calling ChatClient.updateToken()may now indicate an error to the listener if the token validation failed.
- 
Introduced Push Notification Muting, with per-channel user notification preferences. - Added Channel.getNotificationLevel()andChannel.setNotificationLevel(NotificationLevel, StatusListener).
- Added Channel.NotificationLevelenum with values.DEFAULTto receive notifications and.MUTEDto suppress them. Upon leaving the channel, the mute setting for this User is reset to.DEFAULT.
- Added Channel.UpdateReason.NOTIFICATION_LEVELenum value when notification level changes.
- NOTE: You need to grant editNotificationLevelpermission to the user roles to be able to change this setting. See the Chat Permissions documentation.
 
- Added 
- 
Added new token lifecycle management delegate callbacks which reduce the need for AccessManager integration when only using one Twilio client SDK: - ChatClientListener.onTokenAboutToExpire()is called when a token has 3 minutes or less left until expiry and should be refreshed.
- ChatClientListener.onTokenExpired()is called when a token is already expired and must be refreshed to reconnect to the service.
 
- 
New logging level android.log.LogLevel.VERBOSEis now supported. Use this value to generate logs when reporting SDK issues to Twilio. As before, the default log setting is.SILENT, producing no SDK debug output at all.
- 
Added Member.UpdateReason.ATTRIBUTESenum value. There's currently no way to query member attributes, it will be added in the following SDK version.
- 
Added logging of android.os.Build.{BOARD,BRAND,DEVICE,DISPLAY,FINGERPRINT,HARDWARE,MANUFACTURER,MODEL,PRODUCT}on ChatClient startup - this data is logged into the local android log only and is helpful in diagnosing problems, please include it when submitting logs in bug reports!
- 
Added @NonNull annotations to methods in Messagesclass.
- 
Overhauled transport and communication layer, significantly improving network traffic overhead and client performance. 
- 
Provided numerous fixes for improving client stability, focusing on runtime and shutdown crashes. 
- Resolve crash on shutting down on Oreo devices.
- Resolve crash on shutdown after calling getLastMessages().
- Deprecated Message.getTimeStampAsDate(), useMessage.getDateCreatedAsDate()instead.
- Added Channels.getSubscribedChannelsSortedBy()to obtain a list of subscribed channels sorted based on a few available criteria:Channel.SortCriterion.LAST_MESSAGE,Channel.SortCriterion.FRIENDLY_NAME,Channel.SortCriterion.UNIQUE_NAME.
- Added Channel.getLastMessageDate()andChannel.getLastMessageIndex()to obtain information about last message activity in the channel.
- Added Channel.UpdateReason.LAST_MESSAGEfor last channel message change, you can inspect the two above-mentioned properties of theChannelafter receiving this update for new values.
- Resolve crash on accessing last messages in the Channel after some patterns of message deletion.
- Fix local reference table overflow when clearing more than 512 objects from cache.
- Made disposed object checks non-fatal - they only log an ERROR level message now.
- You must now request at least 1 message from the channel in Messages.getLastMessages()query. It used to be possible to query for 0 messages and receive no results, but not anymore.
- GDPR-compliance by default - uses log level SILENT to not output anything from SDK by default. Switch to higher debug level using ChatClient.setLogLevel()to receive log messages.
This version is a major version bump due to semver. There are a number of incompatible API changes making this bump necessary.
These updates have changed the API in a backwards-incompatible manner.
- ChannelListener.onTypingStarted/onTypingEndedmethods now include a Channel in which this event has happened in the signature.
- Chat messages that include media content previously returned a placeholder string "Media messages are not supported". This has caused some confusion, so these messages now return nullbody instead. Be careful to handle this case properly because it may cause NPEs in the your application if you don't.
- Message.getTimeStamphas been renamed- Message.getDateCreatedfor clarity. The returned value has not changed.
- ChatClientListener.onNotification()method for receiving push notifications has been split into 4 separate methods:- onNewMessageNotification(String channelSid, String messageSid, long messageIndex)
- onAddedToChannelNotification(String channelSid)
- onInvitedToChannelNotification(String channelSid)
- onRemovedFromChannelNotification(String channelSid)Now the parameters to these methods are more logical - for example there's no messageSid parameter when it is unused. The new- messageIndexparameter provides an easier interface if you need to scroll to a message in the channel from push notification.
 
- Behavior of Channel.joinandMembers.add/addByIdentityhas changed - methods will fail if user is already a member of the channel.
- Previously deprecated methods are now removed:
- Messages.advanceLastConsumedMessageIndex(), use- advanceLastConsumedMessageIndexWithResult()instead.
- Messages.setLastConsumedMessageIndex(), use- setLastConsumedMessageIndexWithResult()instead.
- Messages.setAllMessagesConsumed(), use- setAllMessagesConsumedWithResult()instead.
- Messages.setNoMessagesConsumed(), use- setNoMessagesConsumedWithResult()instead.
 
- Media queueing support has been added for media messages, to better utilize upload bandwidth.
- Online documentation format has changed to a more usable one (with search).
- CallbackListener for ChatClient.createwill be called on the originating thread. Previously it was called from library thread by mistake.
- Android CA store certificates can be used for connections where possible. This means custom company certificates added on devices should now be honored by the SDK. To enable this feature use ChatClient.Properties.Builder's setDeferCertificateTrustToPlatform(boolean)with valuetrue. By default Twilio-pinned certificate bundle will be used.
- ChatClientListener.onChannelAdded()will be called regardless of if the channel's status is known yet or not, it was not called before if channel status was not yet determined.
- Various internal bugfixes.
- Fixed crash on shutdown due to local reference table overflow.
- Made disposed object checks non-fatal - they only log an ERROR level message now.
- Fixed possible crash on shutdown.
- Updated internal certificate store for upcoming certificate authority changes.
- Forward current connection state to the newly added listener. It's impossible to lose client connection state change between client creation and listener assignment now.
- Implement better multithreading in channel listener callbacks. It's now more robust during concurrent modifications to the channel listeners list.
- Optimized SDK speed.
- Optimized SDK size.
- Improved stability.
- Added back the javadocs that went missing after 2.0.3-rc3
- Fixed an issue where shutting down ChatClient on Android Oreo devices (8.0+) could crash.
- Fixed an issue where a disposed ChatClient would attempt to reconnect to Twilio on application foregrounding.
- BEHAVIORAL CHANGE: Bug-fix so joining an already joined channel or attempting to add an already joined member to a channel will now correctly return a failure result.
- Fixed an issue where the ChatClient could sometimes crash during initialization.
- Unconsumed messages count on joined channels should immediately reflect changes made by new consumption horizon update methods described below.
- Improved documentation.
- Added logging to reflect ChatClient detecting the application transitioning into the background or foreground.
- Improvements to memory utilization.
- Unimplemented Channel.setType()removed from the SDK.
- Added updated consumption horizon methods which return an updated number of unconsumed messages in the listener callback:
- setLastConsumedMessageIndexWithResult()
- advanceLastConsumedMessageIndexWithResult()
- setAllMessagesConsumedWithResult()
- setNoMessagesConsumedWithResult()
 
- Deprecated old consumption horizon update methods which did not take a completion listener. Please use the newly added methods which do accept a completion listener above.
- setLastConsumedMessageIndex()
- advanceLastConsumedMessageIndex()
- setAllMessagesConsumed()
- setNoMessagesConsumed()
 
- SDK will no longer crash on client construction on devices prior to API 21.
- Re-connection to the Chat service will occur faster when bringing the application to the foreground.
- Both chat client creation and updateToken()once again give feedback if the provided token is expired or otherwise invalid.
- Fixed an issue where the listener for joinChannelsometimes was not called.
- Fixed an issue where properties of Messagewere sometimes not fully populated by the timesendMessage's listener was called.
- Added Multimedia Messaging (beta) API (see Media Support for more information). Note: Sending media messages on existing instances requires adding a new permission in your channel roles.
- Added Message.Media, Message.getType().
 
- BREAKING: Added UpdateReasons to onMessageUpdated and onMemberUpdated callbacks.
- BREAKING: Removed the deprecated enum SynchronizationStrategy.
- BREAKING: Use Message.Optionsto construct andsendMessage()- You now could fully initialize necessary message attributes before sending it out.
 
- BREAKING: Members.add(),.addByIdentity(),.invite(),.inviteByIdentity(),.removeByIdentity()updated to new chat-lib API
- Return empty JSON object instead of empty string if Message attributes are empty.
- Fixed a crash on API 19 devices that prevented SDK use.
- Updated internal certificate store for upcoming certificate authority changes.
- Optimized SDK speed.
- Optimized SDK size.
- Improved stability.
- Fixed an issue where shutting down ChatClient on Android Oreo devices (8.0+) could crash.
- Fixed an issue where already disposed ChatClient would attempt to reconnect on application foregrounding.
- Fixed a potential issue where ChatClient could not release some resources on shutdown.
- Add logging of application backgrounding and foregrounding events to improve debugging.
- Fixed an infrequently occurring crash on chat client shutdown.
- SDK will no longer crash on client construction on devices prior to API 21.
- Re-connection to the Chat service will occur faster when bringing the application to the foreground.
- Both chat client creation and updateToken()once again give feedback if the provided token is expired or otherwise invalid.
- Fixed an issue where the listener for joinChannelsometimes was not called.
- Fixed an issue where properties of Messagewere sometimes not fully populated by the timesendMessage's listener was called.
- Performance enhancements for client startup and data retrieval.
- channel.join()listener will be called after joined channel has been completely synchronized.
- Fixed issues related to thread management, which in many cases should lead to significant performance gains
- Fixed crashes on Client shutdown.
- Fixed an issue whereby setAllMessagesConsumed()in an empty channel could cause the SDK client to crash.
- Fixed an issue whereby GCM/FCM unregistration callbacks were not reliably being called.
- Fixed additional issues with some commands not correctly completing - i.e. no success callback nor failure was received. This release fixes cases not addressed by the previous release.
- Updated documentation to address inconsistencies.
- Fixed a crash when passing nullProperties toChatClient.create(). AnIllegalStateExceptionwill now be thrown instead.
- Updated log messages to not mention non-existing Channel.synchronize() anymore.
- Addressed status callback threading issues related to Member.getUserDescriptor()/.getAndSubscribeUser().
- Ensure that Channel listener's synchronisation status callback immediately upon add in all cases.
- Added creation of notification payload from FCM RemoteMessage contents.
- Added a new helper method: Channel.SynchronizationStatus.isAtLeast().
- Fixed a bug whereby some commands did not correctly complete - i.e. no success callback nor failure was received. This release fixes cases not addressed by the previous release.
- Fixed a bug whereby SDK would crash in some circumstances while creating a Chat Client instance.
- Fixed a bug whereby client initialization did not completed correctly - i.e. no success nor failure callback was received for client creation
- Fixed a bug whereby some commands did not correctly complete - i.e. no success callback nor failure was received
- Fixed a bug whereby after a call to client.shutdown()sometimes caused New Message push registrations to be lost for the endpoint.
Changes are grouped by class to ease migration.
- changed default log level from Log.DEBUGtoLog.WARN
- previously deprecated initialMessageCount setting has been removed
- removed functions:
- int getInitialMessageCount()
- Builder ChatClient.Builder.setInitialMessageCount(int initialMessageCount)
 
- UserInfo has been replaced with User/UserDescriptor and direct getting of UserInfo has been replaced with identity getters.
- replaced UserInfo getMyUserInfo()withString getMyIdentity()
 
- replaced 
- added new function to access user information Users getUsers()
- added new functions:
- void onUserSubscribed(User user)
- void onUserUnsubscribed(User user)
 
- changed functions:
- renamed onChannelAddtoonChannelAdded
- renamed onChannelJointoonChannelJoined
- renamed onChannelInvitetoonChannelInvited
- renamed onChannelDeletetoonChannelDeleted
- renamed onToastNotificationtoonNotification
- renamed onToastSubscribedtoonNotificationSubscribed
- renamed onToastFailedtoonNotificationFailed
- replaced void onChannelChange(Channel)withvoid onChannelUpdated(Channel, Channel.UpdateReason reason)
- replaced void onUserInfoChange(UserInfo userInfo, UserInfo.UpdateReason reason)withvoid onUserUpdated(User user, User.UpdateReason reason)
 
- renamed 
- added new value for Channel.ChannelStatusenum:- UNKNOWN- channels returned in ChannelDescriptor do not have status information
 
- added new Channel.UpdateReasonenum to indicate reason for channel update, with values:- STATUS
- LAST_CONSUMED_MESSAGE_INDEX
- UNIQUE_NAME
- FRIENDLY_NAME
- ATTRIBUTES
 
- removed function (you no longer need to explicitly synchronize channels):
- void synchronize(CallbackListener<Channel>)
 
- added new functions:
- long getUnconsumedMessagesCount()
- Channel.ChannelStatus getStatus()
 
- renamed getPublicChannelstogetPublicChannelsList
- added new functions:
- void getUserChannelsList(CallbackListener<Paginator<ChannelDescriptor>>)
- List<Channel> getSubscribedChannels()
- List<Member> getMembersByIdentity(String identity)
 
- changed functions:
- renamed onMessageAddtoonMessageAdded
- renamed onMessageChangetoonMessageUpdated
- renamed onMessageDeletetoonMessageDeleted
- renamed onMemberJointoonMemberJoined
- renamed onMemberAddtoonMemberAdded
- renamed onMemberChangetoonMemberUpdated
- renamed onMemberDeletetoonMemberDeleted
- renamed onSynchronizationChangetoonSynchronizationChanged
 
- renamed 
- added new functions:
- int getStatus()
- String toString()
 
- changed functions:
- renamed getErrorCodetogetCode
- renamed getErrorTexttogetMessage
 
- renamed 
- added new functions:
- void getUserDescriptor(CallbackListener<UserDescriptor> listener)
- void getAndSubscribeUser(CallbackListener<User> listener)
 
- changed function:
- replaced UserInfo getUserInfo()withString getIdentity()
 
- replaced 
- added new functions:
- Channel getChannel()
- Member getMember(String identity)
- void add(Member member, StatusListener listener)
- void invite(Member member, StatusListener listener)
- void removeByIdentity(String identity, StatusListener listener)
 
- changed functions:
- replaced void getMembers(CallbackListener<Paginator<Member>> listener)withList<Member> getMembersList()
- renamed removeMembertoremove
 
- replaced 
- renamed UserInfotoUserclass
- added new functions:
- void unsubscribe()
- boolean isSubscribed()
 
- added new class UserDescriptorwith functions:- String getFriendlyName()
- JSONObject getAttributes()
- String getIdentity()
- boolean isOnline()
- boolean isNotifiable()
- subscribe(CallbackListener<User> listener)
 
- added new class Userswith functions:- void getChannelUserDescriptors(String channelSid, CallbackListener<Paginator<UserDescriptor>> listener)
- void getUserDescriptor(String identity, CallbackListener<UserDescriptor> listener)
- void getAndSubscribeUser(String identity, CallbackListener<User> listener)
- List<User> getSubscribedUsers()
- User getMyUser()
 
- Getting non-existing message by index now properly fails.
- Fixed an issue whereby Push Credential registrations were firing onSuccess on each state change
- Fixed a race condition whereby on channel creation, onChannelAddedandonChannelJoinedevents failed to fire correctly
- Added FCM support via ChatClient.registerFCMToken(), ChatClient.unregisterFCMToken()
- Return a proper error if the internal token and a manual token update causes a mismatch
- Expose additional error codes to client. ErrorInfo.CANNOT_GET_MESSAGE_BY_INDEX and ErrorInfo.MISMATCHING_TOKEN_UPDATE
- Fixed an issue where a certain character caused a crash \v
- Added missing Message.getTimeStampAsDate()
- The endpoint_ididentifier specified in access tokens is now automatically generated and persisted in the local app storage. The following should be observed with this change:- endpoint_idis no longer a required identifier in your generated access tokens, you may omit it from this build forward
- endpoint_idwill be ignored if specified with your access token
 
- Fully asynchronous client init - Client listener onError() will be called if any service error happens during client creation. - Client will not be created in this case.
- Initial message count is completely ignored by init code, 0 is always used - messages are loaded when a channel is loaded.
- Security enhancements
- Added missing Channel.getCreatedBy()andChannelDescriptor.getCreatedBy()to return creator's identity
- Fixed a bug whereby a constant CPU load was generated by the SDK client when initialized.
- Added a call to the onClientSynchronizationcallback in the registered listener uponChatClient.setListener(), passing the current client synchronization state
- Fixed an issue where the getPublicChannels()caused a crash
- Rename IP MessagingtoProgrammable Chat
- IPMessagingClientis renamed to- ChatClient
- com.twilio.ipmessagingis renamed to- com.twilio.chat
- AccessManager is removed from SDK
- Channel.removeAllListeners()removes all added listeners.
- Channel.getMessagesCount()
- Channel.getUnconsumedMessagesCount()
- Channel.getMembersCount()
- Channels.getPublicChannels()returns- ChannelDescriptors, and this list is not updated in realtime.
- onUserInfoUpdatedadded UpdateReason.
- BREAKING Do not return IPMessagingClient before it was fully initialized.
IPMessagingClient.create()no longer returns a value, you can obtain reference to created IPMessagingClient only in listener'sonSuccess()callback.
- BREAKING Channel.setListener()replaced withaddListener()to support assigning multiple listeners
- BREAKING Channel.removeListener()now accepts listener ptr to remove only that one
- BREAKING Replace custom CreateChannelListenerinChannels.createChannel()with genericCallbackListener<Channel>. Listener'sonCreated(Channel)is therefore renamedonSuccess(Channel).
- BREAKING IPMessagingClient.handleNotification()now acceptsNotificationPayloadinstead ofMap<String,String>
- BREAKING Messages.getMessageByIndex()has been changed to asynchronous interface.
- BREAKING Members.getMembers()has been changed to asynchronous interface.
- BREAKING Channel invite logic changes.
- BREAKING Constantsinterface is removed,StatusListenerandCallbackListener<T>are standalone classes now.
- BREAKING Removed built-in AccessManager, IPMessagingClient.create()now accepts only String token.IPMessagingClient.updateToken()is provided to update it.- Use com.twilio:accessmanager-android:0.1.0 from jcenter
- Removed twilio-common from dependencies.
 
- BREAKING Remove previously obsoleted ChannelListener.onAttributesChange()
- BREAKING Remove previously deprecated Channels.createChannel()with attributes map.
- BREAKING Remove previously deprecated TwilioIPMessagingSDK
- BREAKING Remove previously deprecated InitListener
- BREAKING Remove previously deprecated Messages.getMessages()