Чаты на вебсокетах, когда на бэкенде WAMP. Теперь про Android

Мой коллега уже писал про наш опыт разработки чатов на вебсокетах для iOS, поэтому часть про особенности бэкенда с точки зрения клиента у нас общая. А вот реализация на Android, конечно, отличается. И ещё мне не приходилось, как в первой статье, искать библиотеку для поддержки старых версий операционной системы, потому что на Android каких-то глобальных изменений в сетевой части не было, всё работало и так.

К реализации вернёмся чуть ниже, а начнём с ответов на вопросы про бэкенд, которые появились после первой статьи: почему WAMP, какой брокер используем и некоторые другие моменты.

На время передам слово нашему бэкенд-разработчику @antoha-gs, а если хочется сразу почитать про клиент-серверное общение и декодирование, то первый раздел можно пропустить.

Что там на бэкенде

Почему WAMP. Изначально искал открытый протокол, который мог бы работать поверх WebSocket с поддержкой функционала PubSub и RPC и с потенциалом масштабирования. Лучше всего подошёл WAMP — одни плюсы, разве что не нашёл реализации протокола на Java/Kotlin, которая бы меня устраивала.

Какой брокер мы используем. Продолжая предыдущий пункт, это, собственно, и послужило написанию собственной реализации протокола. Плюсы — экспертиза в своём коде и гибкость, то есть при надобности всегда можно отойти от стандарта в нужную сторону. Каких-то серьёзных минусов не выявил.

К небольшим проблемам реализации проекта чатов можно отнести то, что нужно было ресёрчить и ревёрсить то, как работает реализация на Sendbird — сервисе, который мы использовали. То есть какими возможностями мы там пользовались и какой дополнительный функционал был реализован поверх. Также к сложностям отнёс бы перенос данных из Sendbird в свою базу.

Ещё был такой момент в комментариях:

«Правильно, что не стали использовать Socket.IO, так как рано или поздно столкнулись бы с двумя проблемами: 1) Пропуск сообщений. 2) Дублирование сообщений. WAMP — к сожалению — также не решает эти вопросы. Поэтому для чатов лучше использовать что-то вроде MQTT».

Насколько я могу судить, протокол не решает таких проблем магическим образом, всё упирается в реализацию. Да, на уровне протокола может поддерживаться дополнительная информация/настройки для указания уровня обслуживания (at most/at least/exactly), но ответственность за её реализацию всё равно лежит на конкретной имплементации. В нашем случае, учитывая специфику, достаточно гарантировать надёжную запись в базу и доставку на клиенты at most once, что WAMP вполне позволяет реализовать. Также он легко расширяем.

MQTT — отличный протокол, никаких вопросов, но в данном сравнении у него меньше фич, чем у WAMP, которые могли бы пригодиться нам для сервиса чатов. В качестве альтернативы можно было бы рассмотреть XMPP (aka Jabber), потому что, в отличие от MQTT и WAMP, он предназначен для мессенджеров, но и там без «допилов» бы не обошлось. Ещё можно создать свой собственный протокол, что нередко делают в компаниях, но это, в том числе, дополнительные временные затраты.

Это были основные вопросы касательно бэкенда после предыдущей статьи, и, думаю, мы ещё вернёмся к нашей реализации в отдельном материале. А сейчас возвращаю слово Сергею.

Клиент-сервер

Начну с того, что WAMP означает для клиента.

  • В целом протокол предусматривает почти всё. Это облегчает взаимодействие разработчиков клиентской части и бэка.

  • Кодирование всех типов событий в числах (PUBLISH — это 16, SUBSCRIBE — 32 и так далее). Это усложняет чтение логов разработчику и QA (сразу не догадаться, что значит прилетевшее сообщение [33,11,5862354]).

  • Механизм подписок на события (например, новые сообщения в чат или обновление количества участников) реализован через получение от бэкенда уникального id подписки. Его надо где-то хранить и ни в коем случае не терять во избежание утечек. Как это сделано (было бы сильно проще и подписываться и отписываться просто по id чата):client → подписываемся на новые сообщения в чате  [32,18,{},»co.fun.chat.testChatId»]backend → [33,18,5868752 (id подписки)]client → после выхода из чата отписываемся по id [34,20,5868752]

Для работы с сокетом использовали OkHttp (стильно, надёжно, современно, реализация ping-pong таймаутов из коробки) и RxJava, потому что сама концепция чата — практически идеальный пример того самого event-based programming, ради которого Rx, в общем, и задумывался.

Теперь рассмотрим пример коннекта к серверу, использующему WAMP-протокол через OkHttpClient: 

val request = Request.Builder()     .url(ChatsConfig.SOCKETURL)     .addHeader("Connection", "Upgrade")     .addHeader("Sec-WebSocket-Protocol", "wamp.json")     .addHeader("Authorization", authToken)     .build() val listener = ChatWebSocketListener() webSocket = okHttpClient.newWebSocket(request, listener)

Пример реализации ChatWebSocketListener:

private inner class ChatWebSocketListener : WebSocketListener() {  override fun onClosed(webSocket: WebSocket, code: Int, reason: String) {  connectionStatusSubject.onNext(ChatConnectionStatuses.NOTCONNECTED)  //subject, оповещающий пользователей о состоянии коннекта (в UI нужен для отображения лоадеров, оффлайн-стейтов и так далее) }  override fun onClosing(webSocket: WebSocket, code: Int, reason: String) {  webSocket.close(1000, null) }  override fun onFailure(webSocket: WebSocket, t: Throwable, response: Response?) {  onConnectionError("${t.message} ${response?.body}") }  override fun onMessage(webSocket: WebSocket, text: String) {  socketMessagesSubject.onNext(serverMessageFactory.processMessage(text)) //subject, через который идут все сообщения, которые в дальнейшем фильтруются для конкретных получателей (см. ниже) }  override fun onOpen(webSocket: WebSocket, response: Response) {  authorize()  } }

Здесь мы видим, что все сообщения от сокета приходят в виде обычного String, представляющего собой JSON, закодированный по правилам WAMP протокола и имеющий структуру:

[ResultCode: Int, RequestId: Long, ArgumentsMap: JsonObject ]

Например:

[50, 7, {"type":100, "chats":[список чатов]}]

Декодирование и отправка сообщений

Для декодинга сообщений в объекты мы использовали библиотеку Gson. Все модели ответа отписываются обычными data-классами вида:

@DontObfuscate data class ChatListResponse(@SerializedName("chats") val chatList: List<Chat>)

А декодирование происходит с помощью следующего кода:

private fun chatListUpdateInternal(jsonChatsResponse: JSONObject): ChatsListUpdatesEvent {  return gson.fromJson(jsonChatsResponse.toString(),  ChatsListUpdatesEvent::class.java) }

Теперь рассмотрим базовый пример отправки сообщения по сокету. Для удобства мы сделали обёртку для всех базовых типов WAMP сообщений: 

sealed class WampMessage {  class BaseMessage(val wampId: Int, val seq: Long, val jsonData: JSONArray) : WampMessage()    class ErrorMessage(val procedureId: Int, val seq: Long, val jsonData: JSONArray) : WampMessage()   object WelcomeMessage : WampMessage()  class AbortMessage(val jsonData: JSONArray) : WampMessage() }

А также добавили фабрику для формирования этих сообщений:

fun getCallMessage(rpc: String,          options: Map<String, Any> = emptyMap(),          arguments: List<Any?> = emptyList(),          argumentsDict: Map<String, Any?> = emptyMap()): WampMessage.BaseMessage {  //[CALL, Request|id, Options|dict, Procedure|uri, Arguments|list]  val seq = nextSeq.getAndIncrement()  return WampMessage.BaseMessage(WAMP.MessageIds.CALL,                seq,                JSONArray(listOfNotNull(WAMP.MessageIds.CALL,                seq,                options,                rpc,                arguments,                argumentsDict))) }

Пример отправки сообщений: 

val messages: Observable<WampMessage> = socketMessagesSubject  fun sendMessage(msgToSend: WampMessage.BaseMessage):  Observable<WampMessage> {  return messages.filter {    it is WampMessage.BaseMessage && it.seq == msgToSend.seq }     .take(1)     .doOnSubscribe {      webSocket.send(msgToSend.jsonData.toString())     } }

Сопоставление отправленного сообщения и ответа на него в WAMP происходит с помощью уникального идентификатора seq, отправляемого клиентом, который потом кладётся в ответ.

В клиенте генерация идентификатора делается следующим образом:

companion object {  private val nextSeq: AtomicLong = AtomicLong(1) } fun getNextSeq() = nextSeq.getAndIncrement()

Взаимодействие с WAMP Subscriptions 

Подписки в протоколе WAMP — концепт, по которому подписчик (клиент) подписывается на какие-либо события, приходящие от бэкенда. В нашей реализации мы использовали:

  • обновление списка чатов;

  • новые сообщения в конкретном чате;

  • изменение онлайн-статуса собеседника;

  • изменение в составе участников чата;

  • смена роли юзера (например, когда его назначают модератором);

  • и так далее.

Клиент сообщает серверу о желании получать события с помощью следующего сообщения:

[SUBSCRIBE: Int, RequestId: Long, Options: Map, Topic: String]

Где topic — это скоуп событий, которые нужны подписчику. 

Для формирования базового события подписки используется код:

fun getSubscribeMessage(topic: String, options: Map<String, Any> = emptyMap()):  WampMessage.BaseMessage {  val seq = nextSeq.getAndIncrement()  return WampMessage.BaseMessage(WAMP.MessageIds.SUBSCRIBE,                								  seq,               								  JSONArray(listOfNotNull(WAMP.MessageIds.SUBSCRIBE,                                 seq,                                 options,                                 topic))) }

Разумеется, при выходе с экрана (например, списка чатов), необходимо соответствующую подписку корректно отменять. И вот тут выявляется одно из свойств протокола WAMP: при отправке subscribe-сообщения бэкенд возвращает числовой id подписки, и выходит, что отписаться от конкретного топика нельзя — нужно запоминать и хранить этот id, чтобы использовать его при необходимости.

А так как хочется оградить пользователей API подписок от лишнего менеджмента айдишников, было сделано следующее:

private val subscriptionsMap = ArrayMap<String, Long>()  private fun getBaseSubscription(topic: String): Observable<WampMessage> {  val msg = wampClientMessageFactory.getSubscribeMessage(topic)  return send(msg).map {    val subscriptionId = converter.getSubscriptionId((it.asBaseMessage()).jsonData)    subscriptionsMap[topic] = subscriptionId    subscriptionId }     .switchMap { subscriptionId ->       chatClient.messages.filter {        it.isMessageFromSubscription(subscriptionId)      }     } }

Так клиент ничего не будет знать об id, и для отписки ему будет достаточно указать имя подписки, которую необходимо отменить:

fun unsubscribeFromTopic(topic: String) {  if (!subscriptionsMap.contains(topic)) {     return  }  val msg =  wampClientMessageFactory.getUnsubscribeMessage(subscriptionsMap[topic])  send(msg, true).exSubscribe()  subscriptionsMap.remove(topic) }

Это то, что я хотел рассказать про реализацию на Android, но если есть вопросы — постараюсь ответить на них в комментариях. Напомню, что про чаты на вебсокетах в iOS мы уже писали вот здесь, а также готовим отдельную часть про бэкенд.

ссылка на оригинал статьи https://habr.com/ru/company/funcorp/blog/537144/

Добавить комментарий

Ваш адрес email не будет опубликован. Обязательные поля помечены *