mirror of
https://github.com/tinode/chat.git
synced 2026-05-07 20:12:42 +00:00
9f9d8de7ed
Messages marked as read by sender must not be counted as unread. Also, exposed a bit more debug info.
231 lines
7.0 KiB
Go
231 lines
7.0 KiB
Go
/******************************************************************************
|
|
*
|
|
* Description:
|
|
* Push notifications handling.
|
|
*
|
|
*****************************************************************************/
|
|
|
|
package main
|
|
|
|
import (
|
|
"time"
|
|
|
|
"github.com/tinode/chat/server/push"
|
|
"github.com/tinode/chat/server/store/types"
|
|
)
|
|
|
|
// Subscribe or unsubscribe user to/from FCM topic (channel).
|
|
func (t *Topic) channelSubUnsub(uid types.Uid, sub bool) {
|
|
push.ChannelSub(&push.ChannelReq{
|
|
Uid: uid,
|
|
Channel: types.GrpToChn(t.name),
|
|
Unsub: !sub,
|
|
})
|
|
}
|
|
|
|
// Prepares a payload to be delivered to a mobile device as a push notification in response to a {data} message.
|
|
func (t *Topic) pushForData(fromUid types.Uid, data *MsgServerData, msgMarkedAsReadBySender bool) *push.Receipt {
|
|
// Passing `Topic` as `t.name` for group topics and P2P topics. The p2p topic name is later rewritten for
|
|
// each recipient then the payload is created: p2p recipient sees the topic as the ID of the other user.
|
|
|
|
// Initialize the push receipt.
|
|
contentType, _ := data.Head["mime"].(string)
|
|
receipt := push.Receipt{
|
|
To: make(map[types.Uid]push.Recipient, t.subsCount()),
|
|
Payload: push.Payload{
|
|
What: push.ActMsg,
|
|
Silent: false,
|
|
Topic: t.name,
|
|
From: data.From,
|
|
Timestamp: data.Timestamp,
|
|
SeqId: data.SeqId,
|
|
ContentType: contentType,
|
|
Content: data.Content,
|
|
},
|
|
}
|
|
if webrtc, found := data.Head["webrtc"].(string); found {
|
|
receipt.Payload.Webrtc = webrtc
|
|
audioOnly, _ := data.Head["aonly"].(bool)
|
|
receipt.Payload.AudioOnly = audioOnly
|
|
}
|
|
if replace, found := data.Head["replace"].(string); found {
|
|
receipt.Payload.Replace = replace
|
|
}
|
|
|
|
if t.isChan {
|
|
// Channel readers should get a push on a channel name (as an FCM topic push).
|
|
receipt.Channel = types.GrpToChn(t.name)
|
|
}
|
|
|
|
for uid, pud := range t.perUser {
|
|
online := pud.online
|
|
if uid == fromUid && online == 0 {
|
|
// Make sure the sender's devices receive a silent push.
|
|
online = 1
|
|
}
|
|
|
|
// Send only to those who have notifications enabled.
|
|
mode := pud.modeWant & pud.modeGiven
|
|
if mode.IsPresencer() && mode.IsReader() && !pud.deleted && !pud.isChan {
|
|
receipt.To[uid] = push.Recipient{
|
|
// Number of attached sessions the data message will be delivered to.
|
|
// Push notifications sent to users with non-zero online sessions will be marked silent.
|
|
Delivered: online,
|
|
// Unread counts are incremented for all recipients,
|
|
// and for sender only if the message wasnt't marked 'read' by the sender
|
|
ShouldIncrementUnreadCountInCache: uid != fromUid || !msgMarkedAsReadBySender,
|
|
}
|
|
}
|
|
}
|
|
if len(receipt.To) > 0 || receipt.Channel != "" {
|
|
return &receipt
|
|
}
|
|
// If there are no recipient there is no need to send the push notification.
|
|
return nil
|
|
}
|
|
|
|
func (t *Topic) preparePushForSubReceipt(fromUid types.Uid, now time.Time) *push.Receipt {
|
|
// The `Topic` in the push receipt is `t.xoriginal` for group topics, `fromUid` for p2p topics,
|
|
// not the t.original(fromUid) because it's the topic name as seen by the recipient, not by the sender.
|
|
topic := t.xoriginal
|
|
if t.cat == types.TopicCatP2P {
|
|
topic = fromUid.UserId()
|
|
}
|
|
|
|
// Initialize the push receipt.
|
|
receipt := &push.Receipt{
|
|
To: make(map[types.Uid]push.Recipient, t.subsCount()),
|
|
Payload: push.Payload{
|
|
What: push.ActSub,
|
|
Silent: false,
|
|
Topic: topic,
|
|
From: fromUid.UserId(),
|
|
Timestamp: now,
|
|
SeqId: t.lastID,
|
|
},
|
|
}
|
|
return receipt
|
|
}
|
|
|
|
// Prepares payload to be delivered to a mobile device as a push notification in response to a new subscription in a p2p topic.
|
|
func (t *Topic) pushForP2PSub(fromUid, toUid types.Uid, want, given types.AccessMode, now time.Time) *push.Receipt {
|
|
receipt := t.preparePushForSubReceipt(fromUid, now)
|
|
receipt.Payload.ModeWant = want
|
|
receipt.Payload.ModeGiven = given
|
|
|
|
receipt.To[toUid] = push.Recipient{}
|
|
|
|
return receipt
|
|
}
|
|
|
|
// Prepares payload to be delivered to a mobile device as a push notification in response to a new subscription in a group topic.
|
|
func (t *Topic) pushForGroupSub(fromUid types.Uid, now time.Time) *push.Receipt {
|
|
receipt := t.preparePushForSubReceipt(fromUid, now)
|
|
if pud, ok := t.perUser[fromUid]; ok {
|
|
receipt.Payload.ModeWant = pud.modeWant
|
|
receipt.Payload.ModeGiven = pud.modeGiven
|
|
} else {
|
|
// Sender is not a subscriber (BUG?)
|
|
return nil
|
|
}
|
|
|
|
for uid, pud := range t.perUser {
|
|
// Send only to those who have notifications enabled.
|
|
mode := pud.modeWant & pud.modeGiven
|
|
if mode.IsPresencer() && mode.IsReader() && !pud.deleted && !pud.isChan {
|
|
receipt.To[uid] = push.Recipient{}
|
|
}
|
|
}
|
|
if len(receipt.To) > 0 || receipt.Channel != "" {
|
|
return receipt
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Prepares payload to be delivered to a mobile device as a push notification in response to owner deleting a channel.
|
|
func pushForChanDelete(topicName string, now time.Time) *push.Receipt {
|
|
topicName = types.GrpToChn(topicName)
|
|
// Initialize the push receipt.
|
|
return &push.Receipt{
|
|
Payload: push.Payload{
|
|
What: push.ActSub,
|
|
Silent: true,
|
|
Topic: topicName,
|
|
Timestamp: now,
|
|
ModeWant: types.ModeNone,
|
|
ModeGiven: types.ModeNone,
|
|
},
|
|
Channel: topicName,
|
|
}
|
|
}
|
|
|
|
// Prepares payload to be delivered to a mobile device as a push notification in response to receiving "read" notification.
|
|
func (t *Topic) pushForReadRcpt(uid types.Uid, seq int, now time.Time) *push.Receipt {
|
|
// The `Topic` in the push receipt is `t.xoriginal` for group topics, `fromUid` for p2p topics,
|
|
// not the t.original(fromUid) because it's the topic name as seen by the recipient, not by the sender.
|
|
topic := t.xoriginal
|
|
if t.cat == types.TopicCatP2P {
|
|
topic = uid.UserId()
|
|
}
|
|
|
|
// Initialize the push receipt.
|
|
receipt := &push.Receipt{
|
|
To: make(map[types.Uid]push.Recipient, 1),
|
|
Payload: push.Payload{
|
|
What: push.ActRead,
|
|
Silent: true,
|
|
Topic: topic,
|
|
From: uid.UserId(),
|
|
Timestamp: now,
|
|
SeqId: seq,
|
|
},
|
|
}
|
|
receipt.To[uid] = push.Recipient{}
|
|
return receipt
|
|
}
|
|
|
|
// Process push notification.
|
|
func sendPush(rcpt *push.Receipt) {
|
|
if rcpt == nil || globals.usersUpdate == nil {
|
|
return
|
|
}
|
|
|
|
var local *UserCacheReq
|
|
|
|
// In case of a cluster pushes will be initiated at the nodes which own the users.
|
|
// Sort users into local and remote.
|
|
if globals.cluster != nil {
|
|
local = &UserCacheReq{PushRcpt: &push.Receipt{
|
|
Payload: rcpt.Payload,
|
|
Channel: rcpt.Channel,
|
|
To: make(map[types.Uid]push.Recipient),
|
|
}}
|
|
remote := &UserCacheReq{PushRcpt: &push.Receipt{
|
|
Payload: rcpt.Payload,
|
|
Channel: rcpt.Channel,
|
|
To: make(map[types.Uid]push.Recipient),
|
|
}}
|
|
|
|
for uid, recipient := range rcpt.To {
|
|
if globals.cluster.isRemoteTopic(uid.UserId()) {
|
|
remote.PushRcpt.To[uid] = recipient
|
|
} else {
|
|
local.PushRcpt.To[uid] = recipient
|
|
}
|
|
}
|
|
|
|
if len(remote.PushRcpt.To) > 0 || remote.PushRcpt.Channel != "" {
|
|
globals.cluster.routeUserReq(remote)
|
|
}
|
|
} else {
|
|
local = &UserCacheReq{PushRcpt: rcpt}
|
|
}
|
|
|
|
if len(local.PushRcpt.To) > 0 || local.PushRcpt.Channel != "" {
|
|
select {
|
|
case globals.usersUpdate <- local:
|
|
default:
|
|
}
|
|
}
|
|
}
|