Files
open-im-server/pkg/common/db/cache/msg.go
T

571 lines
22 KiB
Go
Raw Normal View History

2023-01-16 20:14:26 +08:00
package cache
2022-06-06 18:15:32 +08:00
import (
"context"
"errors"
2023-03-23 19:02:20 +08:00
"strconv"
"time"
"github.com/OpenIMSDK/Open-IM-Server/pkg/errs"
2023-05-16 16:31:35 +08:00
"github.com/dtm-labs/rockscache"
2023-03-16 10:46:06 +08:00
"github.com/OpenIMSDK/Open-IM-Server/pkg/common/config"
"github.com/OpenIMSDK/Open-IM-Server/pkg/common/constant"
2023-05-16 16:31:35 +08:00
unRelationTb "github.com/OpenIMSDK/Open-IM-Server/pkg/common/db/table/unrelation"
2023-03-16 10:46:06 +08:00
"github.com/OpenIMSDK/Open-IM-Server/pkg/common/log"
"github.com/OpenIMSDK/Open-IM-Server/pkg/proto/sdkws"
"github.com/OpenIMSDK/Open-IM-Server/pkg/utils"
2023-03-24 11:29:39 +08:00
"github.com/gogo/protobuf/jsonpb"
"google.golang.org/protobuf/proto"
2022-08-18 11:14:38 +08:00
2023-01-29 11:37:11 +08:00
"github.com/go-redis/redis/v8"
2022-06-06 18:15:32 +08:00
)
2022-07-20 10:18:43 +08:00
2022-06-23 10:18:44 +08:00
const (
2023-05-05 21:30:32 +08:00
maxSeq = "MAX_SEQ:"
minSeq = "MIN_SEQ:"
conversationUserMinSeq = "CON_USER_MIN_SEQ:"
appleDeviceToken = "DEVICE_TOKEN"
getuiToken = "GETUI_TOKEN"
getuiTaskID = "GETUI_TASK_ID"
messageCache = "MESSAGE_CACHE:"
2023-05-16 16:31:35 +08:00
messageReadCache = "MESSAGE_READ_CACHE:"
2023-05-05 21:30:32 +08:00
signalCache = "SIGNAL_CACHE:"
signalListCache = "SIGNAL_LIST_CACHE:"
fcmToken = "FCM_TOKEN:"
2023-02-09 14:40:49 +08:00
sendMsgFailedFlag = "SEND_MSG_FAILED_FLAG:"
userBadgeUnreadCountSum = "USER_BADGE_UNREAD_COUNT_SUM:"
exTypeKeyLocker = "EX_LOCK:"
2023-03-02 12:00:31 +08:00
uidPidToken = "UID_PID_TOKEN_STATUS:"
2022-06-23 10:18:44 +08:00
)
2023-05-04 12:11:29 +08:00
type MsgModel interface {
2023-05-05 21:30:32 +08:00
SetMaxSeq(ctx context.Context, conversationID string, maxSeq int64) error
GetMaxSeqs(ctx context.Context, conversationIDs []string) (map[string]int64, error)
GetMaxSeq(ctx context.Context, conversationID string) (int64, error)
SetMinSeq(ctx context.Context, conversationID string, minSeq int64) error
GetMinSeqs(ctx context.Context, conversationIDs []string) (map[string]int64, error)
GetMinSeq(ctx context.Context, conversationID string) (int64, error)
GetConversationUserMinSeq(ctx context.Context, conversationID string, userID string) (int64, error)
GetConversationUserMinSeqs(ctx context.Context, conversationID string, userIDs []string) (map[string]int64, error)
SetConversationUserMinSeq(ctx context.Context, conversationID string, userID string, minSeq int64) error
SetConversationUserMinSeqs(ctx context.Context, conversationID string, seqs map[string]int64) (err error)
2023-02-15 15:52:32 +08:00
AddTokenFlag(ctx context.Context, userID string, platformID int, token string, flag int) error
2023-02-20 17:13:15 +08:00
GetTokensWithoutError(ctx context.Context, userID, platformID string) (map[string]int, error)
2023-03-01 15:32:26 +08:00
SetTokenMapByUidPid(ctx context.Context, userID string, platform string, m map[string]int) error
DeleteTokenByUidPid(ctx context.Context, userID string, platform string, fields []string) error
2023-05-05 21:30:32 +08:00
GetMessagesBySeq(ctx context.Context, conversationID string, seqList []int64) (seqMsg []*sdkws.MsgData, failedSeqList []int64, err error)
SetMessageToCache(ctx context.Context, conversationID string, msgList []*sdkws.MsgData) (int, error)
DeleteMessageFromCache(ctx context.Context, conversationID string, msgList []*sdkws.MsgData) error
2023-05-08 12:39:45 +08:00
CleanUpOneConversationAllMsg(ctx context.Context, conversationID string) error
2023-03-02 12:00:31 +08:00
HandleSignalInvite(ctx context.Context, msg *sdkws.MsgData, pushToUserID string) (isSend bool, err error)
2023-03-01 15:32:26 +08:00
GetSignalInvitationInfoByClientMsgID(ctx context.Context, clientMsgID string) (invitationInfo *sdkws.SignalInviteReq, err error)
GetAvailableSignalInvitationInfo(ctx context.Context, userID string) (invitationInfo *sdkws.SignalInviteReq, err error)
2023-02-15 15:52:32 +08:00
DelUserSignalList(ctx context.Context, userID string) error
2023-02-16 15:20:59 +08:00
DelMsgFromCache(ctx context.Context, userID string, seqList []int64) error
2023-02-15 15:52:32 +08:00
SetGetuiToken(ctx context.Context, token string, expireTime int64) error
GetGetuiToken(ctx context.Context) (string, error)
SetGetuiTaskID(ctx context.Context, taskID string, expireTime int64) error
GetGetuiTaskID(ctx context.Context) (string, error)
2023-02-24 11:01:33 +08:00
SetSendMsgStatus(ctx context.Context, id string, status int32) error
GetSendMsgStatus(ctx context.Context, id string) (int32, error)
2023-02-15 15:52:32 +08:00
SetFcmToken(ctx context.Context, account string, platformID int, fcmToken string, expireTime int64) (err error)
GetFcmToken(ctx context.Context, account string, platformID int) (string, error)
DelFcmToken(ctx context.Context, account string, platformID int) error
IncrUserBadgeUnreadCountSum(ctx context.Context, userID string) (int, error)
SetUserBadgeUnreadCountSum(ctx context.Context, userID string, value int) error
GetUserBadgeUnreadCountSum(ctx context.Context, userID string) (int, error)
2023-03-03 17:42:26 +08:00
JudgeMessageReactionExist(ctx context.Context, clientMsgID string, sessionType int32) (bool, error)
2023-02-15 15:52:32 +08:00
GetOneMessageAllReactionList(ctx context.Context, clientMsgID string, sessionType int32) (map[string]string, error)
DeleteOneMessageKey(ctx context.Context, clientMsgID string, sessionType int32, subKey string) error
SetMessageReactionExpire(ctx context.Context, clientMsgID string, sessionType int32, expiration time.Duration) (bool, error)
GetMessageTypeKeyValue(ctx context.Context, clientMsgID string, sessionType int32, typeKey string) (string, error)
SetMessageTypeKeyValue(ctx context.Context, clientMsgID string, sessionType int32, typeKey, value string) error
LockMessageTypeKey(ctx context.Context, clientMsgID string, TypeKey string) error
UnLockMessageTypeKey(ctx context.Context, clientMsgID string, TypeKey string) error
2023-05-16 16:31:35 +08:00
GetMsgsByConversationIDAndSeq(ctx context.Context, docID string, seqs []int64) ([]*sdkws.MsgData, error)
DeleteMsgByConversationIDAndSeq(ctx context.Context, docID string, seq int64) MsgModel
2023-02-09 14:40:49 +08:00
}
2023-05-04 12:11:29 +08:00
func NewMsgCacheModel(client redis.UniversalClient) MsgModel {
return &msgCache{rdb: client}
2023-01-16 20:14:26 +08:00
}
2023-05-04 12:11:29 +08:00
type msgCache struct {
2023-05-16 16:31:35 +08:00
metaCache
rdb redis.UniversalClient
expireTime time.Duration
rcClient *rockscache.Client
msgDocDatabase unRelationTb.MsgDocModelInterface
2023-03-13 17:57:52 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) getMaxSeqKey(conversationID string) string {
return maxSeq + conversationID
2023-01-17 16:36:34 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) getMinSeqKey(conversationID string) string {
return minSeq + conversationID
2023-02-20 10:49:39 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) setSeq(ctx context.Context, conversationID string, seq int64, getkey func(conversationID string) string) error {
return utils.Wrap1(c.rdb.Set(ctx, getkey(conversationID), seq, 0).Err())
2022-06-15 16:33:30 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) getSeq(ctx context.Context, conversationID string, getkey func(conversationID string) string) (int64, error) {
return utils.Wrap2(c.rdb.Get(ctx, getkey(conversationID)).Int64())
2022-06-15 16:33:30 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) getSeqs(ctx context.Context, items []string, getkey func(s string) string) (m map[string]int64, err error) {
pipe := c.rdb.Pipeline()
for _, v := range items {
if err := pipe.Get(ctx, getkey(v)).Err(); err != nil && err != redis.Nil {
2023-05-09 16:55:04 +08:00
return nil, errs.Wrap(err)
2023-05-05 21:30:32 +08:00
}
}
result, err := pipe.Exec(ctx)
2023-05-09 17:02:44 +08:00
if err != nil && err != redis.Nil {
2023-05-05 21:30:32 +08:00
return nil, errs.Wrap(err)
}
m = make(map[string]int64, len(items))
for i, v := range result {
2023-05-11 15:09:45 +08:00
seq := v.(*redis.StringCmd)
2023-05-11 15:03:06 +08:00
if seq.Err() != nil && seq.Err() != redis.Nil {
2023-05-05 21:30:32 +08:00
return nil, errs.Wrap(v.Err())
}
2023-05-18 15:41:08 +08:00
val := utils.StringToInt64(seq.Val())
if val != 0 {
m[items[i]] = val
2023-05-18 15:04:11 +08:00
}
2023-05-05 21:30:32 +08:00
}
return m, nil
2022-06-15 16:33:30 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) SetMaxSeq(ctx context.Context, conversationID string, maxSeq int64) error {
return c.setSeq(ctx, conversationID, maxSeq, c.getMaxSeqKey)
2022-06-15 16:33:30 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) GetMaxSeqs(ctx context.Context, conversationIDs []string) (m map[string]int64, err error) {
return c.getSeqs(ctx, conversationIDs, c.getMaxSeqKey)
2022-06-15 16:33:30 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) GetMaxSeq(ctx context.Context, conversationID string) (int64, error) {
return c.getSeq(ctx, conversationID, c.getMaxSeqKey)
}
func (c *msgCache) SetMinSeq(ctx context.Context, conversationID string, minSeq int64) error {
return c.setSeq(ctx, conversationID, minSeq, c.getMinSeqKey)
}
func (c *msgCache) GetMinSeqs(ctx context.Context, conversationIDs []string) (map[string]int64, error) {
return c.getSeqs(ctx, conversationIDs, c.getMinSeqKey)
}
func (c *msgCache) GetMinSeq(ctx context.Context, conversationID string) (int64, error) {
return c.getSeq(ctx, conversationID, c.getMinSeqKey)
2022-07-20 10:18:43 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) getConversationUserMinSeqKey(conversationID, userID string) string {
return conversationUserMinSeq + "g:" + conversationID + "u:" + userID
2022-07-22 16:55:29 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) GetConversationUserMinSeq(ctx context.Context, conversationID string, userID string) (int64, error) {
return utils.Wrap2(c.rdb.Get(ctx, c.getConversationUserMinSeqKey(conversationID, userID)).Int64())
2022-07-22 16:55:29 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) GetConversationUserMinSeqs(ctx context.Context, conversationID string, userIDs []string) (m map[string]int64, err error) {
return c.getSeqs(ctx, userIDs, func(userID string) string {
return c.getConversationUserMinSeqKey(conversationID, userID)
})
}
func (c *msgCache) SetConversationUserMinSeq(ctx context.Context, conversationID string, userID string, minSeq int64) error {
return utils.Wrap1(c.rdb.Set(ctx, c.getConversationUserMinSeqKey(conversationID, userID), minSeq, 0).Err())
2022-07-22 16:55:29 +08:00
}
2023-05-05 21:30:32 +08:00
func (c *msgCache) SetConversationUserMinSeqs(ctx context.Context, conversationID string, seqs map[string]int64) (err error) {
pipe := c.rdb.Pipeline()
for userID, minSeq := range seqs {
err = pipe.Set(ctx, c.getConversationUserMinSeqKey(conversationID, userID), minSeq, 0).Err()
if err != nil {
return errs.Wrap(err)
}
}
_, err = pipe.Exec(ctx)
return err
2022-08-04 17:20:33 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) AddTokenFlag(ctx context.Context, userID string, platformID int, token string, flag int) error {
2023-02-23 17:28:57 +08:00
key := uidPidToken + userID + ":" + constant.PlatformIDToName(platformID)
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.HSet(ctx, key, token, flag).Err())
2022-06-15 16:33:30 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetTokensWithoutError(ctx context.Context, userID, platformID string) (map[string]int, error) {
2022-06-15 16:33:30 +08:00
key := uidPidToken + userID + ":" + platformID
2023-03-01 15:32:26 +08:00
m, err := c.rdb.HGetAll(ctx, key).Result()
2023-02-24 11:01:33 +08:00
if err != nil {
2023-04-28 18:39:21 +08:00
return nil, errs.Wrap(err)
2023-02-20 17:13:15 +08:00
}
mm := make(map[string]int)
for k, v := range m {
mm[k] = utils.StringToInt(v)
}
2023-02-24 11:01:33 +08:00
return mm, nil
2023-02-20 17:13:15 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) SetTokenMapByUidPid(ctx context.Context, userID string, platform string, m map[string]int) error {
2023-03-01 15:32:26 +08:00
key := uidPidToken + userID + ":" + platform
2022-06-21 21:27:00 +08:00
mm := make(map[string]interface{})
for k, v := range m {
mm[k] = v
}
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.HSet(ctx, key, mm).Err())
2022-06-15 16:33:30 +08:00
}
2023-02-09 14:40:49 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) DeleteTokenByUidPid(ctx context.Context, userID string, platform string, fields []string) error {
2023-03-01 15:32:26 +08:00
key := uidPidToken + userID + ":" + platform
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.HDel(ctx, key, fields...).Err())
2022-06-15 16:33:30 +08:00
}
2023-05-04 16:00:06 +08:00
func (c *msgCache) getMessageCacheKey(conversationID string, seq int64) string {
return messageCache + conversationID + "_" + strconv.Itoa(int(seq))
2023-04-26 18:57:41 +08:00
}
2023-05-04 16:00:06 +08:00
func (c *msgCache) allMessageCacheKey(conversationID string) string {
return messageCache + conversationID + "_*"
2023-04-26 18:57:41 +08:00
}
2023-05-06 11:48:03 +08:00
func (c *msgCache) GetMessagesBySeq(ctx context.Context, conversationID string, seqs []int64) (seqMsgs []*sdkws.MsgData, failedSeqs []int64, err error) {
2023-03-24 11:29:39 +08:00
pipe := c.rdb.Pipeline()
for _, v := range seqs {
2022-06-06 18:15:32 +08:00
//MESSAGE_CACHE:169.254.225.224_reliability1653387820_0_1
2023-05-06 11:48:03 +08:00
key := c.getMessageCacheKey(conversationID, v)
2023-03-24 11:29:39 +08:00
if err := pipe.Get(ctx, key).Err(); err != nil && err != redis.Nil {
return nil, nil, err
}
}
result, err := pipe.Exec(ctx)
for i, v := range result {
2023-05-11 15:15:16 +08:00
cmd := v.(*redis.StringCmd)
if cmd.Err() != nil {
2023-03-24 11:29:39 +08:00
failedSeqs = append(failedSeqs, seqs[i])
2022-06-15 23:30:33 +08:00
} else {
2023-02-15 15:52:32 +08:00
msg := sdkws.MsgData{}
2023-05-11 15:15:16 +08:00
err = jsonpb.UnmarshalString(cmd.Val(), &msg)
2023-05-06 18:51:10 +08:00
if err == nil {
2023-05-06 11:48:03 +08:00
if msg.Status != constant.MsgDeleted {
seqMsgs = append(seqMsgs, &msg)
2023-05-06 18:51:10 +08:00
continue
2023-05-06 11:48:03 +08:00
}
2023-05-06 18:51:10 +08:00
} else {
log.ZWarn(ctx, "UnmarshalString failed", err, "conversationID", conversationID, "seq", seqs[i])
2022-06-06 18:15:32 +08:00
}
2023-05-06 18:51:10 +08:00
failedSeqs = append(failedSeqs, seqs[i])
2022-06-06 18:15:32 +08:00
}
}
2023-03-24 11:29:39 +08:00
return seqMsgs, failedSeqs, err
2022-06-06 18:15:32 +08:00
}
2022-08-05 15:16:43 +08:00
2023-05-05 21:30:32 +08:00
func (c *msgCache) SetMessageToCache(ctx context.Context, conversationID string, msgs []*sdkws.MsgData) (int, error) {
2023-03-01 15:32:26 +08:00
pipe := c.rdb.Pipeline()
2023-05-11 15:30:06 +08:00
var failedMsgs []*sdkws.MsgData
2023-05-05 21:30:32 +08:00
for _, msg := range msgs {
key := c.getMessageCacheKey(conversationID, msg.Seq)
2023-04-28 18:39:21 +08:00
s, err := utils.Pb2String(msg)
2022-06-06 18:15:32 +08:00
if err != nil {
2023-04-28 18:39:21 +08:00
return 0, errs.Wrap(err)
2022-06-06 18:15:32 +08:00
}
2022-06-15 19:17:49 +08:00
err = pipe.Set(ctx, key, s, time.Duration(config.Config.MsgCacheTimeout)*time.Second).Err()
2022-06-06 18:15:32 +08:00
if err != nil {
2023-05-11 15:30:06 +08:00
failedMsgs = append(failedMsgs, msg)
log.ZWarn(ctx, "set msg 2 cache failed", err, "msg", failedMsgs)
2022-06-06 18:15:32 +08:00
}
}
2022-06-15 19:17:49 +08:00
_, err := pipe.Exec(ctx)
2023-05-11 15:30:06 +08:00
return len(failedMsgs), err
2022-06-06 18:15:32 +08:00
}
2023-02-24 11:01:33 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) DeleteMessageFromCache(ctx context.Context, userID string, msgList []*sdkws.MsgData) error {
2023-03-24 11:29:39 +08:00
pipe := c.rdb.Pipeline()
2023-03-02 12:00:31 +08:00
for _, v := range msgList {
2023-04-28 18:39:21 +08:00
if err := pipe.Del(ctx, c.getMessageCacheKey(userID, v.Seq)).Err(); err != nil {
return errs.Wrap(err)
2022-08-18 11:14:38 +08:00
}
2022-07-27 12:15:32 +08:00
}
2023-03-24 11:29:39 +08:00
_, err := pipe.Exec(ctx)
2023-04-28 18:39:21 +08:00
return errs.Wrap(err)
2022-07-27 12:15:32 +08:00
}
2022-06-06 18:15:32 +08:00
2023-05-08 12:39:45 +08:00
func (c *msgCache) CleanUpOneConversationAllMsg(ctx context.Context, conversationID string) error {
vals, err := c.rdb.Keys(ctx, c.allMessageCacheKey(conversationID)).Result()
2023-01-29 11:37:11 +08:00
if err == redis.Nil {
2022-06-06 18:15:32 +08:00
return nil
}
if err != nil {
2023-04-28 18:39:21 +08:00
return errs.Wrap(err)
2022-06-06 18:15:32 +08:00
}
2023-03-24 11:29:39 +08:00
pipe := c.rdb.Pipeline()
2022-08-11 20:25:33 +08:00
for _, v := range vals {
2023-03-24 11:29:39 +08:00
if err := pipe.Del(ctx, v).Err(); err != nil {
2023-04-28 18:39:21 +08:00
return errs.Wrap(err)
2023-02-24 11:01:33 +08:00
}
2022-06-06 18:15:32 +08:00
}
2023-03-24 11:29:39 +08:00
_, err = pipe.Exec(ctx)
2023-04-28 18:39:21 +08:00
return errs.Wrap(err)
2022-06-06 18:15:32 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) HandleSignalInvite(ctx context.Context, msg *sdkws.MsgData, pushToUserID string) (isSend bool, err error) {
2023-03-02 12:00:31 +08:00
req := &sdkws.SignalReq{}
2022-06-06 18:52:17 +08:00
if err := proto.Unmarshal(msg.Content, req); err != nil {
2023-04-28 18:39:21 +08:00
return false, errs.Wrap(err)
2022-06-06 18:15:32 +08:00
}
2023-03-02 12:00:31 +08:00
var inviteeUserIDs []string
2022-06-10 17:31:30 +08:00
var isInviteSignal bool
switch signalInfo := req.Payload.(type) {
2023-03-02 12:00:31 +08:00
case *sdkws.SignalReq_Invite:
inviteeUserIDs = signalInfo.Invite.Invitation.InviteeUserIDList
2022-06-10 17:31:30 +08:00
isInviteSignal = true
2023-03-02 12:00:31 +08:00
case *sdkws.SignalReq_InviteInGroup:
inviteeUserIDs = signalInfo.InviteInGroup.Invitation.InviteeUserIDList
2022-06-10 17:31:30 +08:00
isInviteSignal = true
2023-03-02 12:00:31 +08:00
if !utils.Contain(pushToUserID, inviteeUserIDs...) {
2022-09-02 17:22:27 +08:00
return false, nil
}
2023-03-02 12:00:31 +08:00
case *sdkws.SignalReq_HungUp, *sdkws.SignalReq_Cancel, *sdkws.SignalReq_Reject, *sdkws.SignalReq_Accept:
2023-04-28 18:39:21 +08:00
return false, errs.Wrap(errors.New("signalInfo do not need offlinePush"))
2022-06-06 21:04:11 +08:00
default:
2022-09-02 17:22:27 +08:00
return false, nil
2022-06-06 21:04:11 +08:00
}
2022-06-10 17:31:30 +08:00
if isInviteSignal {
2023-03-24 11:29:39 +08:00
pipe := c.rdb.Pipeline()
2023-03-02 12:00:31 +08:00
for _, userID := range inviteeUserIDs {
2022-06-10 17:31:30 +08:00
timeout, err := strconv.Atoi(config.Config.Rtc.SignalTimeout)
if err != nil {
2023-04-28 18:39:21 +08:00
return false, errs.Wrap(err)
2022-06-10 17:31:30 +08:00
}
2023-03-24 11:29:39 +08:00
keys := signalListCache + userID
err = pipe.LPush(ctx, keys, msg.ClientMsgID).Err()
2022-06-10 17:31:30 +08:00
if err != nil {
2023-04-28 18:39:21 +08:00
return false, errs.Wrap(err)
2022-06-10 17:31:30 +08:00
}
2023-03-24 11:29:39 +08:00
err = pipe.Expire(ctx, keys, time.Duration(timeout)*time.Second).Err()
2022-06-10 17:31:30 +08:00
if err != nil {
2023-04-28 18:39:21 +08:00
return false, errs.Wrap(err)
2022-06-10 17:31:30 +08:00
}
2023-03-02 12:00:31 +08:00
key := signalCache + msg.ClientMsgID
2023-03-24 11:29:39 +08:00
err = pipe.Set(ctx, key, msg.Content, time.Duration(timeout)*time.Second).Err()
2022-06-10 17:31:30 +08:00
if err != nil {
2023-04-28 18:39:21 +08:00
return false, errs.Wrap(err)
2022-06-10 17:31:30 +08:00
}
2022-06-06 18:52:17 +08:00
}
2023-03-24 11:29:39 +08:00
_, err := pipe.Exec(ctx)
if err != nil {
2023-04-28 18:39:21 +08:00
return false, errs.Wrap(err)
2023-03-24 11:29:39 +08:00
}
2022-06-06 18:15:32 +08:00
}
2022-09-02 17:22:27 +08:00
return true, nil
2022-06-06 18:15:32 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetSignalInvitationInfoByClientMsgID(ctx context.Context, clientMsgID string) (signalInviteReq *sdkws.SignalInviteReq, err error) {
2023-03-02 12:00:31 +08:00
bytes, err := c.rdb.Get(ctx, signalCache+clientMsgID).Bytes()
2022-06-06 18:15:32 +08:00
if err != nil {
2023-04-28 18:39:21 +08:00
return nil, errs.Wrap(err)
2022-06-06 18:15:32 +08:00
}
2023-03-02 12:00:31 +08:00
signalReq := &sdkws.SignalReq{}
if err = proto.Unmarshal(bytes, signalReq); err != nil {
2023-04-28 18:39:21 +08:00
return nil, errs.Wrap(err)
2022-06-06 18:15:32 +08:00
}
2023-03-02 12:00:31 +08:00
signalInviteReq = &sdkws.SignalInviteReq{}
switch req := signalReq.Payload.(type) {
case *sdkws.SignalReq_Invite:
signalInviteReq.Invitation = req.Invite.Invitation
signalInviteReq.OpUserID = req.Invite.OpUserID
case *sdkws.SignalReq_InviteInGroup:
signalInviteReq.Invitation = req.InviteInGroup.Invitation
signalInviteReq.OpUserID = req.InviteInGroup.OpUserID
2022-06-06 19:36:25 +08:00
}
2023-03-02 12:00:31 +08:00
return signalInviteReq, nil
2022-06-06 18:15:32 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetAvailableSignalInvitationInfo(ctx context.Context, userID string) (invitationInfo *sdkws.SignalInviteReq, err error) {
2023-03-02 12:00:31 +08:00
key, err := c.rdb.LPop(ctx, signalListCache+userID).Result()
2022-06-06 19:20:12 +08:00
if err != nil {
2023-04-28 18:39:21 +08:00
return nil, errs.Wrap(err)
2022-06-06 19:20:12 +08:00
}
2023-03-01 15:32:26 +08:00
invitationInfo, err = c.GetSignalInvitationInfoByClientMsgID(ctx, key)
2022-06-06 23:27:16 +08:00
if err != nil {
2023-02-24 11:01:33 +08:00
return nil, err
2022-06-06 23:27:16 +08:00
}
2023-04-28 18:39:21 +08:00
return invitationInfo, errs.Wrap(c.DelUserSignalList(ctx, userID))
2022-06-06 23:27:16 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) DelUserSignalList(ctx context.Context, userID string) error {
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.Del(ctx, signalListCache+userID).Err())
2022-06-06 18:15:32 +08:00
}
2022-06-15 14:57:21 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) DelMsgFromCache(ctx context.Context, userID string, seqs []int64) error {
2023-03-24 11:29:39 +08:00
for _, seq := range seqs {
2023-04-26 18:57:41 +08:00
key := c.getMessageCacheKey(userID, seq)
2023-03-01 15:32:26 +08:00
result, err := c.rdb.Get(ctx, key).Result()
2022-08-09 19:44:27 +08:00
if err != nil {
2023-01-29 11:37:11 +08:00
if err == redis.Nil {
2023-02-24 11:01:33 +08:00
continue
2022-08-09 19:44:27 +08:00
}
2023-04-28 18:39:21 +08:00
return errs.Wrap(err)
2022-08-09 19:44:27 +08:00
}
2023-02-15 15:52:32 +08:00
var msg sdkws.MsgData
2023-02-24 11:01:33 +08:00
if err := jsonpb.UnmarshalString(result, &msg); err != nil {
return err
2022-06-15 14:57:21 +08:00
}
msg.Status = constant.MsgDeleted
s, err := utils.Pb2String(&msg)
if err != nil {
2023-04-28 18:39:21 +08:00
return errs.Wrap(err)
2022-06-15 14:57:21 +08:00
}
2023-03-01 15:32:26 +08:00
if err := c.rdb.Set(ctx, key, s, time.Duration(config.Config.MsgCacheTimeout)*time.Second).Err(); err != nil {
2023-04-28 18:39:21 +08:00
return errs.Wrap(err)
2022-06-15 14:57:21 +08:00
}
}
2023-02-24 11:01:33 +08:00
return nil
2022-06-15 14:57:21 +08:00
}
2022-06-15 16:16:35 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) SetGetuiToken(ctx context.Context, token string, expireTime int64) error {
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.Set(ctx, getuiToken, token, time.Duration(expireTime)*time.Second).Err())
2022-06-15 16:16:35 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetGetuiToken(ctx context.Context) (string, error) {
2023-03-01 15:32:26 +08:00
return utils.Wrap2(c.rdb.Get(ctx, getuiToken).Result())
2022-06-15 16:16:35 +08:00
}
2022-07-26 15:16:46 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) SetGetuiTaskID(ctx context.Context, taskID string, expireTime int64) error {
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.Set(ctx, getuiTaskID, taskID, time.Duration(expireTime)*time.Second).Err())
2022-12-21 16:46:16 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetGetuiTaskID(ctx context.Context) (string, error) {
2023-03-01 15:32:26 +08:00
return utils.Wrap2(c.rdb.Get(ctx, getuiTaskID).Result())
2022-12-21 16:46:16 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) SetSendMsgStatus(ctx context.Context, id string, status int32) error {
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.Set(ctx, sendMsgFailedFlag+id, status, time.Hour*24).Err())
2022-07-26 15:16:46 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetSendMsgStatus(ctx context.Context, id string) (int32, error) {
2023-03-01 15:32:26 +08:00
result, err := c.rdb.Get(ctx, sendMsgFailedFlag+id).Int()
2023-04-28 18:39:21 +08:00
return int32(result), errs.Wrap(err)
2022-07-26 15:16:46 +08:00
}
2022-07-26 15:20:48 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) SetFcmToken(ctx context.Context, account string, platformID int, fcmToken string, expireTime int64) (err error) {
2023-05-04 15:54:04 +08:00
return errs.Wrap(c.rdb.Set(ctx, fcmToken+account+":"+strconv.Itoa(platformID), fcmToken, time.Duration(expireTime)*time.Second).Err())
2022-07-25 17:57:58 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetFcmToken(ctx context.Context, account string, platformID int) (string, error) {
2023-05-04 15:54:04 +08:00
return utils.Wrap2(c.rdb.Get(ctx, fcmToken+account+":"+strconv.Itoa(platformID)).Result())
2022-12-09 20:53:13 +08:00
}
2022-12-12 19:15:31 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) DelFcmToken(ctx context.Context, account string, platformID int) error {
2023-05-04 15:54:04 +08:00
return errs.Wrap(c.rdb.Del(ctx, fcmToken+account+":"+strconv.Itoa(platformID)).Err())
2022-12-12 20:41:40 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) IncrUserBadgeUnreadCountSum(ctx context.Context, userID string) (int, error) {
2023-03-01 15:32:26 +08:00
seq, err := c.rdb.Incr(ctx, userBadgeUnreadCountSum+userID).Result()
2023-04-28 18:39:21 +08:00
return int(seq), errs.Wrap(err)
2022-12-12 12:12:53 +08:00
}
2023-02-09 14:40:49 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) SetUserBadgeUnreadCountSum(ctx context.Context, userID string, value int) error {
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.Set(ctx, userBadgeUnreadCountSum+userID, value, 0).Err())
2022-12-09 20:53:13 +08:00
}
2023-02-09 14:40:49 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetUserBadgeUnreadCountSum(ctx context.Context, userID string) (int, error) {
2023-03-01 15:32:26 +08:00
return utils.Wrap2(c.rdb.Get(ctx, userBadgeUnreadCountSum+userID).Int())
2022-12-09 20:53:13 +08:00
}
2023-02-09 14:40:49 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) LockMessageTypeKey(ctx context.Context, clientMsgID string, TypeKey string) error {
2022-12-09 20:53:13 +08:00
key := exTypeKeyLocker + clientMsgID + "_" + TypeKey
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.SetNX(ctx, key, 1, time.Minute).Err())
2022-12-09 20:53:13 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) UnLockMessageTypeKey(ctx context.Context, clientMsgID string, TypeKey string) error {
2023-02-24 11:01:33 +08:00
key := exTypeKeyLocker + clientMsgID + "_" + TypeKey
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.Del(ctx, key).Err())
2022-12-09 20:53:13 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) getMessageReactionExPrefix(clientMsgID string, sessionType int32) string {
2022-12-09 20:53:13 +08:00
switch sessionType {
case constant.SingleChatType:
return "EX_SINGLE_" + clientMsgID
case constant.GroupChatType:
return "EX_GROUP_" + clientMsgID
case constant.SuperGroupChatType:
return "EX_SUPER_GROUP_" + clientMsgID
case constant.NotificationChatType:
return "EX_NOTIFICATION" + clientMsgID
}
return ""
}
2023-02-24 11:01:33 +08:00
2023-05-04 12:11:29 +08:00
func (c *msgCache) JudgeMessageReactionExist(ctx context.Context, clientMsgID string, sessionType int32) (bool, error) {
2023-03-01 15:32:26 +08:00
n, err := c.rdb.Exists(ctx, c.getMessageReactionExPrefix(clientMsgID, sessionType)).Result()
2023-02-24 11:01:33 +08:00
if err != nil {
return false, utils.Wrap(err, "")
}
return n > 0, nil
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) SetMessageTypeKeyValue(ctx context.Context, clientMsgID string, sessionType int32, typeKey, value string) error {
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.HSet(ctx, c.getMessageReactionExPrefix(clientMsgID, sessionType), typeKey, value).Err())
2023-02-24 11:01:33 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) SetMessageReactionExpire(ctx context.Context, clientMsgID string, sessionType int32, expiration time.Duration) (bool, error) {
2023-03-01 15:32:26 +08:00
return utils.Wrap2(c.rdb.Expire(ctx, c.getMessageReactionExPrefix(clientMsgID, sessionType), expiration).Result())
2023-02-24 11:01:33 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetMessageTypeKeyValue(ctx context.Context, clientMsgID string, sessionType int32, typeKey string) (string, error) {
2023-03-01 15:32:26 +08:00
return utils.Wrap2(c.rdb.HGet(ctx, c.getMessageReactionExPrefix(clientMsgID, sessionType), typeKey).Result())
2023-02-24 11:01:33 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) GetOneMessageAllReactionList(ctx context.Context, clientMsgID string, sessionType int32) (map[string]string, error) {
2023-03-01 15:32:26 +08:00
return utils.Wrap2(c.rdb.HGetAll(ctx, c.getMessageReactionExPrefix(clientMsgID, sessionType)).Result())
2023-02-24 11:01:33 +08:00
}
2023-05-04 12:11:29 +08:00
func (c *msgCache) DeleteOneMessageKey(ctx context.Context, clientMsgID string, sessionType int32, subKey string) error {
2023-04-28 18:39:21 +08:00
return errs.Wrap(c.rdb.HDel(ctx, c.getMessageReactionExPrefix(clientMsgID, sessionType), subKey).Err())
2023-02-24 11:01:33 +08:00
}
2023-05-16 16:31:35 +08:00
func (c *msgCache) NewCache() MsgModel {
return &msgCache{
metaCache: NewMetaCacheRedis(c.rcClient, c.metaCache.GetPreDelKeys()...),
expireTime: c.expireTime,
rcClient: c.rcClient,
}
}
func (c msgCache) getMsgReadCacheKey(docID string, seq int64) string {
return messageReadCache + docID + "_" + strconv.Itoa(int(seq))
}
func (c *msgCache) getMsgsIndex(msg *sdkws.MsgData, keys []string) (int, error) {
key := c.getMsgReadCacheKey(utils.GetConversationIDByMsg(msg), msg.Seq)
for i, _key := range keys {
if key == _key {
return i, nil
}
}
return 0, errIndex
}
func (c *msgCache) GetMsgsByConversationIDAndSeq(ctx context.Context, docID string, seqs []int64) ([]*sdkws.MsgData, error) {
var keys []string
for _, seq := range seqs {
keys = append(keys, c.getMsgReadCacheKey(docID, seq))
}
return batchGetCache(ctx, c.rcClient, keys, c.expireTime, c.getMsgsIndex, func(ctx context.Context) ([]*sdkws.MsgData, error) {
return c.msgDocDatabase.GetMsgBySeqIndexIn1Doc(ctx, docID, seqs)
})
}
func (c *msgCache) DeleteMsgByConversationIDAndSeq(ctx context.Context, docID string, seq int64) MsgModel {
cache := c.NewCache()
c.AddKeys(c.getMsgReadCacheKey(docID, seq))
return cache
}