消息阅后即焚

pull/3727/head
hawklin2017 3 weeks ago
parent 1af699e77c
commit 66368cdd27

@ -833,9 +833,11 @@ func (c *conversationServer) setConversationMinSeqAndLatestMsgDestructTime(ctx c
// ClearBurnExpiredMsgs 处理「阅后即焚」过期消息:
// 1. 从 msg_burn_deadline 中拉取一批过期分组(按 user/conversation 聚合,含每组最大 seq
// 2. 对每个分组把用户在该会话上的 min_seq 推进到 max(过期 seq) + 1。
// 3. 同步更新 conversation 文档的 min_seq 字段并下发会话变更通知。
// 4. 删除已处理的 deadline 记录。
// 2. 对每个分组把用户在该会话上的 min_seq 推进到 max(过期 seq) + 1更新 conversation 文档。
// 3. 向阅读方发 ConversationChangeNotification + DeleteMsgsNotification含精确 seqs
// 4. 单聊场景si_ 前缀):利用 deadline 记录中的 PeerID 直接推进对端 min_seq
// 并向对端也发两条通知,保证双方客户端都删本地消息。
// 5. 物理删除 msg 存储中的焚毁消息;清理 deadline 记录。
//
// 单次最多处理 req.Limit 个分组;若返回的 count == limitcron 可继续触发。
func (c *conversationServer) ClearBurnExpiredMsgs(ctx context.Context, req *pbconversation.ClearBurnExpiredMsgsReq) (*pbconversation.ClearBurnExpiredMsgsResp, error) {
@ -856,6 +858,8 @@ func (c *conversationServer) ClearBurnExpiredMsgs(ctx context.Context, req *pbco
continue
}
newMinSeq := g.MaxSeq + 1
// 推进阅读方 min_seq。
if err := c.msgClient.SetUserConversationMin(ctx, g.ConversationID, []string{g.UserID}, newMinSeq); err != nil {
log.ZError(ctx, "ClearBurnExpiredMsgs SetUserConversationMin failed", err,
"userID", g.UserID, "conversationID", g.ConversationID, "minSeq", newMinSeq)
@ -867,7 +871,18 @@ func (c *conversationServer) ClearBurnExpiredMsgs(ctx context.Context, req *pbco
"userID", g.UserID, "conversationID", g.ConversationID, "minSeq", newMinSeq)
continue
}
// 通知 g.UserID 客户端:会话变更 + 精确删除指定 seqs。
// 对端用户在 msg_burn_deadline 中有独立记录cron 处理其分组时会自行通知,
// 无需在此重复推进对端 min_seq 或发送额外通知。
c.conversationNotificationSender.ConversationChangeNotification(ctx, g.UserID, []string{g.ConversationID})
c.conversationNotificationSender.BurnMsgsDeleteNotification(ctx, g.UserID, g.UserID, g.ConversationID, g.Seqs)
// 物理删除 msg 存储中的焚毁消息best-effort失败不中断流程
if err := c.msgClient.DeleteMsgPhysicalBySeqs(ctx, g.ConversationID, g.Seqs); err != nil {
log.ZError(ctx, "ClearBurnExpiredMsgs DeleteMsgPhysicalBySeqs failed", err,
"conversationID", g.ConversationID, "seqs", g.Seqs)
}
if err := c.msgBurnDeadlineDB.DeleteByUserConversationSeqs(ctx, g.UserID, g.ConversationID, g.Seqs); err != nil {
log.ZError(ctx, "ClearBurnExpiredMsgs DeleteByUserConversationSeqs failed", err,
"userID", g.UserID, "conversationID", g.ConversationID, "seqs", g.Seqs)
@ -878,3 +893,4 @@ func (c *conversationServer) ClearBurnExpiredMsgs(ctx context.Context, req *pbco
}
return &pbconversation.ClearBurnExpiredMsgsResp{Count: processed}, nil
}

@ -73,3 +73,19 @@ func (c *ConversationNotificationSender) ConversationUnreadChangeNotification(
c.Notification(ctx, userID, userID, constant.ConversationUnreadNotification, tips)
}
// BurnMsgsDeleteNotification 通知 recvUserID 按 seqs 删除本地消息。
// sendUserID 为触发焚烧的一方阅读方recvUserID 为需要执行删除的一方。
// 双方各调用一次,保证单聊两端客户端都删除本地缓存。
func (c *ConversationNotificationSender) BurnMsgsDeleteNotification(
ctx context.Context,
sendUserID, recvUserID, conversationID string,
seqs []int64,
) {
tips := &sdkws.DeleteMsgsTips{
UserID: sendUserID,
ConversationID: conversationID,
Seqs: seqs,
}
c.Notification(ctx, sendUserID, recvUserID, constant.DeleteMsgsNotification, tips)
}

@ -245,6 +245,7 @@ func (m *msgServer) recordBurnDeadlines(ctx context.Context, conv *conversation.
UserID: readerUserID,
ConversationID: conv.ConversationID,
Seq: seq,
PeerID: peerID,
DeadlineMs: deadline,
CreateTime: now,
},
@ -252,6 +253,7 @@ func (m *msgServer) recordBurnDeadlines(ctx context.Context, conv *conversation.
UserID: peerID,
ConversationID: conv.ConversationID,
Seq: seq,
PeerID: readerUserID,
DeadlineMs: deadline,
CreateTime: now,
},

@ -66,6 +66,7 @@ func (m *msgBurnDeadlineMgo) UpsertIfAbsent(ctx context.Context, items []*model.
"user_id": item.UserID,
"conversation_id": item.ConversationID,
"seq": item.Seq,
"peer_id": item.PeerID,
"deadline_ms": item.DeadlineMs,
"create_time": item.CreateTime,
}
@ -91,6 +92,7 @@ func (m *msgBurnDeadlineMgo) FindExpiredGroups(ctx context.Context, nowMs int64,
"user_id": "$user_id",
"conversation_id": "$conversation_id",
},
"peer_id": bson.M{"$first": "$peer_id"},
"max_seq": bson.M{"$max": "$seq"},
"seqs": bson.M{"$push": "$seq"},
}}},
@ -101,6 +103,7 @@ func (m *msgBurnDeadlineMgo) FindExpiredGroups(ctx context.Context, nowMs int64,
UserID string `bson:"user_id"`
ConversationID string `bson:"conversation_id"`
} `bson:"_id"`
PeerID string `bson:"peer_id"`
MaxSeq int64 `bson:"max_seq"`
Seqs []int64 `bson:"seqs"`
}
@ -113,6 +116,7 @@ func (m *msgBurnDeadlineMgo) FindExpiredGroups(ctx context.Context, nowMs int64,
res = append(res, &database.ExpiredBurnGroup{
UserID: r.ID.UserID,
ConversationID: r.ID.ConversationID,
PeerID: r.PeerID,
MaxSeq: r.MaxSeq,
Seqs: r.Seqs,
})

@ -24,6 +24,8 @@ import (
type ExpiredBurnGroup struct {
UserID string
ConversationID string
// PeerID 单聊中的对端用户 ID直接从 deadline 记录读取,无需额外查 conversation 表。
PeerID string
// MaxSeq 当前批次中最大的过期 seq推进 min_seq 时使用 MaxSeq + 1。
MaxSeq int64
// Seqs 当前批次实际涉及的所有过期 seq便于精确删除已处理的 deadline 记录。

@ -24,6 +24,9 @@ type MsgBurnDeadline struct {
UserID string `bson:"user_id"`
ConversationID string `bson:"conversation_id"`
Seq int64 `bson:"seq"`
// PeerID 单聊中的对端用户 ID。
// cron 处理时可直接获取对端,无需额外查询 conversation 表。
PeerID string `bson:"peer_id"`
// DeadlineMs 截止时间戳(毫秒);超过即可被 cron 收走推进 min_seq。
DeadlineMs int64 `bson:"deadline_ms"`
// CreateTime 写入时刻(毫秒);用于排查/审计。

@ -113,3 +113,13 @@ func (x *MsgClient) SetUserConversationsMinSeq(ctx context.Context, conversation
req := &msg.SetUserConversationsMinSeqReq{ConversationID: conversationID, UserIDs: userIDs, Seq: seq}
return ignoreResp(x.MsgClient.SetUserConversationsMinSeq(ctx, req))
}
// DeleteMsgPhysicalBySeqs 按 seq 物理删除会话内的消息(无鉴权)。
// 用于阅后即焚、系统级消息清理等场景。
func (x *MsgClient) DeleteMsgPhysicalBySeqs(ctx context.Context, conversationID string, seqs []int64) error {
if len(seqs) == 0 {
return nil
}
req := &msg.DeleteMsgPhysicalBySeqReq{ConversationID: conversationID, Seqs: seqs}
return ignoreResp(x.MsgClient.DeleteMsgPhysicalBySeq(ctx, req))
}

Loading…
Cancel
Save