|
- package model
-
- import (
- "context"
- "database/sql"
- "fmt"
- "gim/internal/logic/domain/message/repo"
- "gim/internal/logic/proxy"
- "gim/pkg/gerrors"
- "gim/pkg/grpclib"
- "gim/pkg/logger"
- "gim/pkg/pb"
- "gim/pkg/rpc"
- "gim/pkg/util"
- "time"
-
- "go.uber.org/zap"
- "google.golang.org/protobuf/proto"
- )
-
- const (
- UpdateTypeUpdate = 1
- UpdateTypeDelete = 2
- )
-
- // Group 群组
- type Group struct {
- Id int64 // 群组id
- Name string // 组名
- AvatarUrl string // 头像
- Introduction string // 群简介
- UserNum int32 // 群组人数
- IsAllMemberBanned int32 // 是否全员禁言(1:是 2:否)
- Extra string // 附加字段
- CreateTime time.Time // 创建时间
- UpdateTime time.Time // 更新时间
- Members []GroupUser `gorm:"-"` // 群组成员
- }
-
- type GroupUser struct {
- Id int64 // 自增主键
- GroupId int64 // 群组id
- UserId int64 // 用户id
- MemberType int // 群组类型
- Remarks string // 备注
- Extra string // 附加属性
- Status int // 状态
- CreateTime time.Time // 创建时间
- UpdateTime time.Time // 更新时间
- UpdateType int `gorm:"-"` // 更新类型
- }
-
- type GroupUserV2 struct {
- Id int64 // 自增主键
- GroupId int64 // 群组id
- UserId int64 // 用户id
- MemberType int // 群组类型
- Remarks string // 备注
- Extra string // 附加属性
- Status sql.NullInt32 // 状态
- CreateTime time.Time // 创建时间
- UpdateTime time.Time // 更新时间
- UpdateType int `gorm:"-"` // 更新类型
- }
-
- func (g *Group) ToProto() *pb.Group {
- if g == nil {
- return nil
- }
-
- return &pb.Group{
- GroupId: g.Id,
- Name: g.Name,
- AvatarUrl: g.AvatarUrl,
- Introduction: g.Introduction,
- UserMum: g.UserNum,
- IsAllMemberBanned: g.IsAllMemberBanned,
- Extra: g.Extra,
- CreateTime: g.CreateTime.Unix(),
- UpdateTime: g.UpdateTime.Unix(),
- }
- }
-
- func CreateGroup(userId int64, in *pb.CreateGroupReq) *Group {
- now := time.Now()
- group := &Group{
- Name: in.Name,
- AvatarUrl: in.AvatarUrl,
- Introduction: in.Introduction,
- Extra: in.Extra,
- Members: make([]GroupUser, 0, len(in.MemberIds)+1),
- CreateTime: now,
- UpdateTime: now,
- }
-
- // 创建者添加为群主
- group.Members = append(group.Members, GroupUser{
- GroupId: group.Id,
- UserId: userId,
- MemberType: int(pb.MemberType_GMT_ADMIN),
- CreateTime: now,
- UpdateTime: now,
- UpdateType: UpdateTypeUpdate,
- })
-
- // 其让人添加为成员
- for i := range in.MemberIds {
- group.Members = append(group.Members, GroupUser{
- GroupId: group.Id,
- UserId: in.MemberIds[i],
- MemberType: int(pb.MemberType_GMT_MEMBER),
- CreateTime: now,
- UpdateTime: now,
- UpdateType: UpdateTypeUpdate,
- })
- }
- return group
- }
-
- func (g *Group) Update(ctx context.Context, in *pb.UpdateGroupReq) error {
- g.Name = in.Name
- g.AvatarUrl = in.AvatarUrl
- g.Introduction = in.Introduction
- g.Extra = in.Extra
- g.UpdateTime = time.Now()
- return nil
- }
-
- func (g *Group) PushUpdate(ctx context.Context, userId int64, isUpdateIntroduction bool) error {
- userResp, err := rpc.GetBusinessIntClient().GetUser(ctx, &pb.GetUserReq{UserId: userId})
- if err != nil {
- return err
- }
- err = g.PushMessage(ctx, pb.PushCode_PC_UPDATE_GROUP, &pb.UpdateGroupPush{
- OptId: userId,
- OptName: userResp.User.Nickname,
- Name: g.Name,
- AvatarUrl: g.AvatarUrl,
- Introduction: g.Introduction,
- IsUpdateIntroduction: isUpdateIntroduction,
- Extra: g.Extra,
- }, true)
- if err != nil {
- return err
- }
- if isUpdateIntroduction {
- //_, err = g.SendMessage(ctx,
- // &pb.Sender{
- // SenderType: pb.SenderType_ST_USER,
- // SenderId: 0,
- // },
- // &pb.SendMessageReq{
- // ReceiverType: pb.ReceiverType_RT_GROUP,
- // ReceiverId: g.Id,
- // ToUserIds: nil,
- // MessageType: pb.MessageType_MT_COMMAND,
- // MessageContent: commandBuf,
- // SendTime: util.UnixMilliTime(time.Now()),
- // IsPersist: isPersist,
- // })
- //if err != nil {
- // return err
- //}
- }
- return nil
- }
-
- // SendMessage 消息发送至群组
- func (g *Group) SendMessage(ctx context.Context, sender *pb.Sender, req *pb.SendMessageReq) (int64, error) {
- if sender.SenderType == pb.SenderType_ST_USER && !g.IsMember(sender.SenderId) {
- logger.Sugar.Error(ctx, sender.SenderId, req.ReceiverId, "不在群组内")
- return 0, gerrors.ErrNotInGroup
- }
-
- // 如果发送者是用户,将消息发送给发送者,获取用户seq
- var userSeq int64
- var err error
- if sender.SenderType == pb.SenderType_ST_USER {
- userSeq, err = proxy.MessageProxy.SendToUser(ctx, sender, sender.SenderId, req)
- if err != nil {
- return 0, err
- }
- }
-
- go func() {
- defer util.RecoverPanic()
- // 将消息发送给群组用户,使用写扩散
- for _, user := range g.Members {
- // 前面已经发送过,这里不需要再发送
- if sender.SenderType == pb.SenderType_ST_USER && user.UserId == sender.SenderId {
- continue
- }
- _, err := proxy.MessageProxy.SendToUser(grpclib.NewAndCopyRequestId(ctx), sender, user.UserId, req)
- if err != nil {
- return
- }
- }
- }()
-
- return userSeq, nil
- }
-
- // RecallSendMessage 撤回消息发送至群组
- func (g *Group) RecallSendMessage(ctx context.Context, sender *pb.Sender, req *pb.RecallMessageReq) (int64, error) {
- if sender.SenderType == pb.SenderType_ST_USER && !g.IsMember(sender.SenderId) {
- logger.Sugar.Error(ctx, sender.SenderId, req.ReceiverId, "不在群组内")
- return 0, gerrors.ErrNotInGroup
- }
- var err error
-
- //查询到对应seq的消息
- msg := &pb.RECALL{}
- err = proto.Unmarshal(req.MessageContent, msg)
- if err != nil {
- return 0, err
- }
- message, err := repo.MessageRepo.GetMessage(sender.SenderId, msg.RecallSeq)
- if err != nil {
- return 0, err
- }
-
- // 如果发送者是用户,将消息发送给发送者,获取用户seq
- var userSeq int64
- if sender.SenderType == pb.SenderType_ST_USER {
- userSeq, err = proxy.MessageProxy.RecallMessageSendToUser(ctx, sender, sender.SenderId, req, message.SendTime)
- if err != nil {
- return 0, err
- }
- }
-
- go func() {
- defer util.RecoverPanic()
- // 将消息发送给群组用户,使用写扩散
- for _, user := range g.Members {
- // 前面已经发送过,这里不需要再发送
- if sender.SenderType == pb.SenderType_ST_USER && user.UserId == sender.SenderId {
- continue
- }
-
- msg.RecallSeq, err = repo.MessageRepo.GetMessageSeqForSendTime(user.UserId, message.SendTime)
- req.MessageContent, err = proto.Marshal(msg)
- if err != nil {
- return
- }
- _, err := proxy.MessageProxy.RecallMessageSendToUser(grpclib.NewAndCopyRequestId(ctx), sender, user.UserId, req, message.SendTime)
- if err != nil {
- return
- }
- }
- }()
-
- return userSeq, nil
- }
-
- // SendRedPackage 发送红包消息发送至群组
- func (g *Group) SendRedPackage(ctx context.Context, sender *pb.Sender, req *pb.SendRedPacketReq) (int64, error) {
- if sender.SenderType == pb.SenderType_ST_USER && !g.IsMember(sender.SenderId) {
- logger.Sugar.Error(ctx, sender.SenderId, req.ReceiverId, "不在群组内")
- return 0, gerrors.ErrNotInGroup
- }
-
- // 如果发送者是用户,将消息发送给发送者,获取用户seq
- var userSeq int64
- var err error
- if sender.SenderType == pb.SenderType_ST_USER {
- userSeq, err = proxy.MessageProxy.SendRedPackageToUser(ctx, sender, sender.SenderId, req)
- if err != nil {
- return 0, err
- }
- }
-
- go func() {
- defer util.RecoverPanic()
- // 将消息发送给群组用户,使用写扩散
- for _, user := range g.Members {
- // 前面已经发送过,这里不需要再发送
- if sender.SenderType == pb.SenderType_ST_USER && user.UserId == sender.SenderId {
- continue
- }
- _, err := proxy.MessageProxy.SendRedPackageToUser(grpclib.NewAndCopyRequestId(ctx), sender, user.UserId, req)
- if err != nil {
- return
- }
- }
- }()
-
- return userSeq, nil
- }
-
- func (g *Group) IsMember(userId int64) bool {
- for i := range g.Members {
- if g.Members[i].UserId == userId {
- return true
- }
- }
- return false
- }
-
- // PushMessage 向群组推送消息
- func (g *Group) PushMessage(ctx context.Context, code pb.PushCode, message proto.Message, isPersist bool) error {
- logger.Logger.Debug("push_to_group",
- zap.Int64("request_id", grpclib.GetCtxRequestId(ctx)),
- zap.Int64("group_id", g.Id),
- zap.Int32("code", int32(code)),
- zap.Any("message", message))
-
- messageBuf, err := proto.Marshal(message)
- if err != nil {
- return gerrors.WrapError(err)
- }
-
- commandBuf, err := proto.Marshal(&pb.Command{Code: int32(code), Data: messageBuf})
- if err != nil {
- return gerrors.WrapError(err)
- }
-
- _, err = g.SendMessage(ctx,
- &pb.Sender{
- SenderType: pb.SenderType_ST_SYSTEM,
- SenderId: 0,
- },
- &pb.SendMessageReq{
- ReceiverType: pb.ReceiverType_RT_GROUP,
- ReceiverId: g.Id,
- ToUserIds: nil,
- MessageType: pb.MessageType_MT_COMMAND,
- MessageContent: commandBuf,
- SendTime: util.UnixMilliTime(time.Now()),
- IsPersist: isPersist,
- },
- )
- if err != nil {
- return err
- }
- return nil
- }
-
- // GetMembers 获取群组用户
- func (g *Group) GetMembers(ctx context.Context) ([]*pb.GroupMember, error) {
- members := g.Members
- userIds := make(map[int64]int32, len(members))
- for i := range members {
- userIds[members[i].UserId] = 0
- }
- resp, err := rpc.GetBusinessIntClient().GetUsers(ctx, &pb.GetUsersReq{UserIds: userIds})
- if err != nil {
- return nil, err
- }
-
- var infos = make([]*pb.GroupMember, len(members))
- for i := range members {
- member := pb.GroupMember{
- UserId: members[i].UserId,
- MemberType: pb.MemberType(members[i].MemberType),
- Remarks: members[i].Remarks,
- Extra: members[i].Extra,
- Status: int32(members[i].Status),
- }
-
- user, ok := resp.Users[members[i].UserId]
- if ok {
- member.Nickname = user.Nickname
- member.Sex = user.Sex
- member.AvatarUrl = user.AvatarUrl
- member.UserExtra = user.Extra
- }
- infos[i] = &member
- }
-
- return infos, nil
- }
-
- // AddMembers 给群组添加用户
- func (g *Group) AddMembers(ctx context.Context, userIds []int64) ([]int64, []int64, error) {
- var existIds []int64
- var addedIds []int64
-
- now := time.Now()
- for i, userId := range userIds {
- if g.IsMember(userId) {
- existIds = append(existIds, userIds[i])
- continue
- }
-
- g.Members = append(g.Members, GroupUser{
- GroupId: g.Id,
- UserId: userIds[i],
- MemberType: int(pb.MemberType_GMT_MEMBER),
- CreateTime: now,
- UpdateTime: now,
- UpdateType: UpdateTypeUpdate,
- })
- addedIds = append(addedIds, userIds[i])
- }
-
- g.UserNum += int32(len(addedIds))
-
- return existIds, addedIds, nil
- }
-
- func (g *Group) PushAddMember(ctx context.Context, optUserId int64, addedIds []int64) error {
- var addIdMap = make(map[int64]int32, len(addedIds))
- for i := range addedIds {
- addIdMap[addedIds[i]] = 0
- }
-
- addIdMap[optUserId] = 0
- usersResp, err := rpc.GetBusinessIntClient().GetUsers(ctx, &pb.GetUsersReq{UserIds: addIdMap})
- if err != nil {
- return err
- }
- var members []*pb.GroupMember
- for k, _ := range addIdMap {
- member, ok := usersResp.Users[k]
- if !ok {
- continue
- }
-
- members = append(members, &pb.GroupMember{
- UserId: member.UserId,
- Nickname: member.Nickname,
- Sex: member.Sex,
- AvatarUrl: member.AvatarUrl,
- UserExtra: member.Extra,
- Remarks: "",
- Extra: "",
- })
- }
-
- //
- optUser := usersResp.Users[optUserId]
- if optUserId == -1 {
- optUser.Nickname = "后台操作"
- }
- err = g.PushMessage(ctx, pb.PushCode_PC_ADD_GROUP_MEMBERS, &pb.AddGroupMembersPush{
- OptId: optUser.UserId,
- OptName: optUser.Nickname,
- Members: members,
- }, true)
- if err != nil {
- logger.Sugar.Error(err)
- }
- return nil
- }
-
- func (g *Group) GetMember(ctx context.Context, userId int64) *GroupUser {
- for i := range g.Members {
- if g.Members[i].UserId == userId {
- return &g.Members[i]
- }
- }
- return nil
- }
-
- // UpdateMember 更新群组成员信息
- func (g *Group) UpdateMember(ctx context.Context, in *pb.UpdateGroupMemberReq) error {
- member := g.GetMember(ctx, in.UserId)
- if member == nil {
- return nil
- }
-
- member.MemberType = int(in.MemberType)
- member.Remarks = in.Remarks
- member.Extra = in.Extra
- member.UpdateTime = time.Now()
- member.UpdateType = UpdateTypeUpdate
- return nil
- }
-
- // DeleteMember 删除用户群组
- func (g *Group) DeleteMember(ctx context.Context, userId int64) error {
- member := g.GetMember(ctx, userId)
- if member == nil {
- return nil
- }
-
- member.UpdateType = UpdateTypeDelete
- return nil
- }
-
- func (g *Group) PushUpdateMember(ctx context.Context, optId, userId int64, memberType int32) error {
- userResp, err := rpc.GetBusinessIntClient().GetUser(ctx, &pb.GetUserReq{UserId: optId})
- if err != nil {
- return err
- }
- if optId == -1 {
- userResp.User.Nickname = "后台操作"
- }
- updateUserResp, err := rpc.GetBusinessIntClient().GetUser(ctx, &pb.GetUserReq{UserId: userId})
- if err != nil {
- return err
- }
- err = g.PushMessage(ctx, pb.PushCode_PC_UPDATE_GROUP_MEMBER, &pb.UpdateMemberPush{
- OptId: optId,
- OptName: userResp.User.Nickname,
- UpdateUserId: userId,
- UpdateUserName: updateUserResp.User.Nickname,
- UpdateUserMemberType: memberType,
- }, true)
- if err != nil {
- return err
- }
- return nil
- }
-
- func (g *Group) PushDeleteMember(ctx context.Context, optId, userId int64) error {
- userResp, err := rpc.GetBusinessIntClient().GetUser(ctx, &pb.GetUserReq{UserId: optId})
- if err != nil {
- return err
- }
- if optId == -1 {
- userResp.User.Nickname = "后台操作"
- }
-
- deleteUserResp, err := rpc.GetBusinessIntClient().GetUser(ctx, &pb.GetUserReq{UserId: userId})
- if err != nil {
- return err
- }
- err = g.PushMessage(ctx, pb.PushCode_PC_REMOVE_GROUP_MEMBER, &pb.RemoveGroupMemberPush{
- OptId: optId,
- OptName: userResp.User.Nickname,
- DeletedUserId: userId,
- DeletedUserName: deleteUserResp.User.Nickname,
- }, true)
- if err != nil {
- return err
- }
- return nil
- }
-
- func (g *Group) PushGroupMemberBanned(ctx context.Context, optId, userId int64, isAllMemberBanned bool) error {
- userResp, err := rpc.GetBusinessIntClient().GetUser(ctx, &pb.GetUserReq{UserId: optId})
- if err != nil {
- return err
- }
- if optId == -1 {
- userResp.User.Nickname = "后台操作"
- }
-
- if !isAllMemberBanned && userId > 0 {
- bannedUserResp, err := rpc.GetBusinessIntClient().GetUser(ctx, &pb.GetUserReq{UserId: userId})
- if err != nil {
- return err
- }
- err = g.PushMessage(ctx, pb.PushCode_PC_BANNED_GROUP_MEMBER, &pb.BannedGroupMemberPush{
- OptId: optId,
- OptName: userResp.User.Nickname,
- BannedUserId: userId,
- BannedUserName: bannedUserResp.User.Nickname,
- }, true)
- if err != nil {
- return err
- }
- return nil
- } else {
- if userId == 0 {
- err = g.PushMessage(ctx, pb.PushCode_PC_BANNED_GROUP_MEMBER, &pb.BannedGroupMemberPush{
- OptId: optId,
- OptName: userResp.User.Nickname,
- BannedUserId: userId,
- BannedUserName: fmt.Sprintf("管理员\"%s\"设置群禁言", userResp.User.Nickname),
- }, true)
- if err != nil {
- return err
- }
- } else {
- err = g.PushMessage(ctx, pb.PushCode_PC_BANNED_GROUP_MEMBER, &pb.BannedGroupMemberPush{
- OptId: optId,
- OptName: userResp.User.Nickname,
- BannedUserId: userId,
- BannedUserName: fmt.Sprintf("管理员\"%s\"取消群禁言", userResp.User.Nickname),
- }, true)
- if err != nil {
- return err
- }
- }
- return nil
- }
- }
-
- func (g *Group) PushGroupMemberRemoveBanned(ctx context.Context, optId, userId int64) error {
- userResp, err := rpc.GetBusinessIntClient().GetUser(ctx, &pb.GetUserReq{UserId: optId})
- if err != nil {
- return err
- }
- if optId == -1 {
- userResp.User.Nickname = "后台操作"
- }
-
- removeBannedUserResp, err := rpc.GetBusinessIntClient().GetUser(ctx, &pb.GetUserReq{UserId: userId})
- if err != nil {
- return err
- }
- err = g.PushMessage(ctx, pb.PushCode_PC_REMOVE_BANNED_GROUP_MEMBER, &pb.RemoveBannedGroupMemberPush{
- OptId: optId,
- OptName: userResp.User.Nickname,
- RemoveBannedUserId: userId,
- RemoveBannedUserName: removeBannedUserResp.User.Nickname,
- }, true)
- if err != nil {
- return err
- }
- return nil
- }
|