mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-25 20:52:11 +08:00 
			
		
		
		
	feat: optimize friend and group applications (#3384)
* pb * fix: Modifying other fields while setting IsPrivateChat does not take effect * fix: quote message error revoke * refactoring scheduled tasks * refactoring scheduled tasks * refactoring scheduled tasks * refactoring scheduled tasks * refactoring scheduled tasks * refactoring scheduled tasks * upgrading pkg tools * fix * fix * optimize log output * feat: support GetLastMessage * feat: support GetLastMessage * feat: s3 switch * feat: s3 switch * fix: GetUsersOnline * feat: SendBusinessNotification supported configuration parameters * feat: SendBusinessNotification supported configuration parameters * feat: SendBusinessNotification supported configuration parameters * feat: seq conversion failed without exiting * fix: DeleteDoc crash * fix: fill send time * fix: fill send time * fix: crash caused by withdrawing messages from users who have left the group * fix: user msg timestamp * seq read config * seq read config * fix: the source message of the reference is withdrawn, and the referenced message is deleted * feat: optimize the default notification.yml * fix: shouldPushOffline * fix: the sorting is wrong after canceling the administrator in group settings * feat: Sending messages supports returning fields modified by webhook * feat: Sending messages supports returning fields modified by webhook * feat: Sending messages supports returning fields modified by webhook * fix: oss specifies content-type when uploading * fix: the version number contains a line break * fix: the version number contains a line break * feat: GetConversationsHasReadAndMaxSeq support pinned * feat: GetConversationsHasReadAndMaxSeq support pinned * feat: GetConversationsHasReadAndMaxSeq support pinned * fix: transferring the group owner to a muted member, incremental version error * feat: unified conversion code * feat: update gomake * fix: in standalone mode, the user online status is wrong * fix: add permission check * fix: add permission check * fix: add rpc interface permission check * fix: CreateGroupChatConversations * feat: optimize friend and group applications * feat: optimize friend and group applications * feat: optimize friend and group applications * feat: optimize friend and group applications (cherry picked from commit 8e61f30e9c59916671e4a75844dcc0a4dfcd5d8a) # Conflicts: # go.mod # go.sum
This commit is contained in:
		
							parent
							
								
									6d80e6ec12
								
							
						
					
					
						commit
						8846cbbd31
					
				
							
								
								
									
										4
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								go.mod
									
									
									
									
									
								
							| @ -12,8 +12,8 @@ require ( | |||||||
| 	github.com/gorilla/websocket v1.5.1 | 	github.com/gorilla/websocket v1.5.1 | ||||||
| 	github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 | 	github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 | ||||||
| 	github.com/mitchellh/mapstructure v1.5.0 | 	github.com/mitchellh/mapstructure v1.5.0 | ||||||
| 	github.com/openimsdk/protocol v0.0.73-alpha.6 | 	github.com/openimsdk/protocol v0.0.73-alpha.12 | ||||||
| 	github.com/openimsdk/tools v0.0.50-alpha.83 | 	github.com/openimsdk/tools v0.0.50-alpha.84 | ||||||
| 	github.com/pkg/errors v0.9.1 // indirect | 	github.com/pkg/errors v0.9.1 // indirect | ||||||
| 	github.com/prometheus/client_golang v1.18.0 | 	github.com/prometheus/client_golang v1.18.0 | ||||||
| 	github.com/stretchr/testify v1.9.0 | 	github.com/stretchr/testify v1.9.0 | ||||||
|  | |||||||
							
								
								
									
										8
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								go.sum
									
									
									
									
									
								
							| @ -347,10 +347,10 @@ github.com/onsi/gomega v1.25.0 h1:Vw7br2PCDYijJHSfBOWhov+8cAnUf8MfMaIOV323l6Y= | |||||||
| github.com/onsi/gomega v1.25.0/go.mod h1:r+zV744Re+DiYCIPRlYOTxn0YkOLcAnW8k1xXdMPGhM= | github.com/onsi/gomega v1.25.0/go.mod h1:r+zV744Re+DiYCIPRlYOTxn0YkOLcAnW8k1xXdMPGhM= | ||||||
| github.com/openimsdk/gomake v0.0.15-alpha.5 h1:eEZCEHm+NsmcO3onXZPIUbGFCYPYbsX5beV3ZyOsGhY= | github.com/openimsdk/gomake v0.0.15-alpha.5 h1:eEZCEHm+NsmcO3onXZPIUbGFCYPYbsX5beV3ZyOsGhY= | ||||||
| github.com/openimsdk/gomake v0.0.15-alpha.5/go.mod h1:PndCozNc2IsQIciyn9mvEblYWZwJmAI+06z94EY+csI= | github.com/openimsdk/gomake v0.0.15-alpha.5/go.mod h1:PndCozNc2IsQIciyn9mvEblYWZwJmAI+06z94EY+csI= | ||||||
| github.com/openimsdk/protocol v0.0.73-alpha.6 h1:sna9coWG7HN1zObBPtvG0Ki/vzqHXiB4qKbA5P3w7kc= | github.com/openimsdk/protocol v0.0.73-alpha.12 h1:2NYawXeHChYUeSme6QJ9pOLh+Empce2WmwEtbP4JvKk= | ||||||
| github.com/openimsdk/protocol v0.0.73-alpha.6/go.mod h1:WF7EuE55vQvpyUAzDXcqg+B+446xQyEba0X35lTINmw= | github.com/openimsdk/protocol v0.0.73-alpha.12/go.mod h1:WF7EuE55vQvpyUAzDXcqg+B+446xQyEba0X35lTINmw= | ||||||
| github.com/openimsdk/tools v0.0.50-alpha.83 h1:7c1D40YGqIWUmGfCII5pduETGC/8c2DyS9SQ4LvoplU= | github.com/openimsdk/tools v0.0.50-alpha.84 h1:jN60Ys/0edZjL/TDmm/5VSJFP4pGYRipkWqhILJbq/8= | ||||||
| github.com/openimsdk/tools v0.0.50-alpha.83/go.mod h1:n2poR3asX1e1XZce4O+MOWAp+X02QJRFvhcLCXZdzRo= | github.com/openimsdk/tools v0.0.50-alpha.84/go.mod h1:n2poR3asX1e1XZce4O+MOWAp+X02QJRFvhcLCXZdzRo= | ||||||
| github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= | github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= | ||||||
| github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs= | github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs= | ||||||
| github.com/pierrec/lz4/v4 v4.1.21 h1:yOVMLb6qSIDP67pl/5F7RepeKYu/VmTyEXvuMI5d9mQ= | github.com/pierrec/lz4/v4 v4.1.21 h1:yOVMLb6qSIDP67pl/5F7RepeKYu/VmTyEXvuMI5d9mQ= | ||||||
|  | |||||||
| @ -114,3 +114,7 @@ func (o *FriendApi) GetIncrementalBlacks(c *gin.Context) { | |||||||
| func (o *FriendApi) GetFullFriendUserIDs(c *gin.Context) { | func (o *FriendApi) GetFullFriendUserIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, relation.FriendClient.GetFullFriendUserIDs, o.Client) | 	a2r.Call(c, relation.FriendClient.GetFullFriendUserIDs, o.Client) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (o *FriendApi) GetSelfUnhandledApplyCount(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, relation.FriendClient.GetSelfUnhandledApplyCount, o.Client) | ||||||
|  | } | ||||||
|  | |||||||
| @ -165,3 +165,7 @@ func (o *GroupApi) GetFullGroupMemberUserIDs(c *gin.Context) { | |||||||
| func (o *GroupApi) GetFullJoinGroupIDs(c *gin.Context) { | func (o *GroupApi) GetFullJoinGroupIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, group.GroupClient.GetFullJoinGroupIDs, o.Client) | 	a2r.Call(c, group.GroupClient.GetFullJoinGroupIDs, o.Client) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (o *GroupApi) GetGroupApplicationUnhandledCount(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, group.GroupClient.GetGroupApplicationUnhandledCount, o.Client) | ||||||
|  | } | ||||||
|  | |||||||
| @ -157,6 +157,7 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 		friendRouterGroup.POST("/update_friends", f.UpdateFriends) | 		friendRouterGroup.POST("/update_friends", f.UpdateFriends) | ||||||
| 		friendRouterGroup.POST("/get_incremental_friends", f.GetIncrementalFriends) | 		friendRouterGroup.POST("/get_incremental_friends", f.GetIncrementalFriends) | ||||||
| 		friendRouterGroup.POST("/get_full_friend_user_ids", f.GetFullFriendUserIDs) | 		friendRouterGroup.POST("/get_full_friend_user_ids", f.GetFullFriendUserIDs) | ||||||
|  | 		friendRouterGroup.POST("/get_self_unhandled_apply_count", f.GetSelfUnhandledApplyCount) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	g := NewGroupApi(group.NewGroupClient(groupConn)) | 	g := NewGroupApi(group.NewGroupClient(groupConn)) | ||||||
| @ -193,6 +194,7 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 		groupRouterGroup.POST("/get_incremental_group_members_batch", g.GetIncrementalGroupMemberBatch) | 		groupRouterGroup.POST("/get_incremental_group_members_batch", g.GetIncrementalGroupMemberBatch) | ||||||
| 		groupRouterGroup.POST("/get_full_group_member_user_ids", g.GetFullGroupMemberUserIDs) | 		groupRouterGroup.POST("/get_full_group_member_user_ids", g.GetFullGroupMemberUserIDs) | ||||||
| 		groupRouterGroup.POST("/get_full_join_group_ids", g.GetFullJoinGroupIDs) | 		groupRouterGroup.POST("/get_full_join_group_ids", g.GetFullJoinGroupIDs) | ||||||
|  | 		groupRouterGroup.POST("/get_group_application_unhandled_count", g.GetGroupApplicationUnhandledCount) | ||||||
| 	} | 	} | ||||||
| 	// certificate | 	// certificate | ||||||
| 	{ | 	{ | ||||||
|  | |||||||
| @ -25,7 +25,6 @@ import ( | |||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 
 |  | ||||||
| 	"google.golang.org/grpc" | 	"google.golang.org/grpc" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| @ -153,10 +152,14 @@ func (g *groupServer) NotificationUserInfoUpdate(ctx context.Context, req *pbgro | |||||||
| 
 | 
 | ||||||
| func (g *groupServer) CheckGroupAdmin(ctx context.Context, groupID string) error { | func (g *groupServer) CheckGroupAdmin(ctx context.Context, groupID string) error { | ||||||
| 	if !authverify.IsAdmin(ctx) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		groupMember, err := g.db.TakeGroupMember(ctx, groupID, mcontext.GetOpUserID(ctx)) | 		members, err := g.db.FindGroupMembers(ctx, groupID, []string{mcontext.GetOpUserID(ctx)}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  | 		if len(members) == 0 { | ||||||
|  | 			return errs.ErrNoPermission.WrapMsg("op user not in group") | ||||||
|  | 		} | ||||||
|  | 		groupMember := members[0] | ||||||
| 		if !(groupMember.RoleLevel == constant.GroupOwner || groupMember.RoleLevel == constant.GroupAdmin) { | 		if !(groupMember.RoleLevel == constant.GroupOwner || groupMember.RoleLevel == constant.GroupAdmin) { | ||||||
| 			return errs.ErrNoPermission.WrapMsg("no group owner or admin") | 			return errs.ErrNoPermission.WrapMsg("no group owner or admin") | ||||||
| 		} | 		} | ||||||
| @ -369,6 +372,10 @@ func (g *groupServer) InviteUserToGroup(ctx context.Context, req *pbgroup.Invite | |||||||
| 		return nil, servererrs.ErrDismissedAlready.WrapMsg("group dismissed checking group status found it dismissed") | 		return nil, servererrs.ErrDismissedAlready.WrapMsg("group dismissed checking group status found it dismissed") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	userMap, err := g.userClient.GetUsersInfoMap(ctx, req.InvitedUserIDs) | 	userMap, err := g.userClient.GetUsersInfoMap(ctx, req.InvitedUserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -421,7 +428,7 @@ func (g *groupServer) InviteUserToGroup(ctx context.Context, req *pbgroup.Invite | |||||||
| 						ReqMessage:    request.ReqMsg, | 						ReqMessage:    request.ReqMsg, | ||||||
| 						JoinSource:    request.JoinSource, | 						JoinSource:    request.JoinSource, | ||||||
| 						InviterUserID: request.InviterUserID, | 						InviterUserID: request.InviterUserID, | ||||||
| 					}) | 					}, request) | ||||||
| 				} | 				} | ||||||
| 				return &pbgroup.InviteUserToGroupResp{}, nil | 				return &pbgroup.InviteUserToGroupResp{}, nil | ||||||
| 			} | 			} | ||||||
| @ -690,15 +697,34 @@ func (g *groupServer) GetGroupApplicationList(ctx context.Context, req *pbgroup. | |||||||
| 	if err := authverify.CheckAccess(ctx, req.FromUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.FromUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	groupIDs, err := g.db.FindUserManagedGroupID(ctx, req.FromUserID) | 	var ( | ||||||
|  | 		groupIDs []string | ||||||
|  | 		err      error | ||||||
|  | 	) | ||||||
|  | 	if len(req.GroupIDs) == 0 { | ||||||
|  | 		groupIDs, err = g.db.FindUserManagedGroupID(ctx, req.FromUserID) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
|  | 	} else { | ||||||
|  | 		req.GroupIDs = datautil.Distinct(req.GroupIDs) | ||||||
|  | 		if !authverify.IsAdmin(ctx) { | ||||||
|  | 			for _, groupID := range req.GroupIDs { | ||||||
|  | 				if err := g.CheckGroupAdmin(ctx, groupID); err != nil { | ||||||
|  | 					return nil, err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		groupIDs = req.GroupIDs | ||||||
|  | 	} | ||||||
| 	resp := &pbgroup.GetGroupApplicationListResp{} | 	resp := &pbgroup.GetGroupApplicationListResp{} | ||||||
| 	if len(groupIDs) == 0 { | 	if len(groupIDs) == 0 { | ||||||
| 		return resp, nil | 		return resp, nil | ||||||
| 	} | 	} | ||||||
| 	total, groupRequests, err := g.db.PageGroupRequest(ctx, groupIDs, req.Pagination) | 	handleResults := datautil.Slice(req.HandleResults, func(e int32) int { | ||||||
|  | 		return int(e) | ||||||
|  | 	}) | ||||||
|  | 	total, groupRequests, err := g.db.PageGroupRequest(ctx, groupIDs, handleResults, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -763,6 +789,23 @@ func (g *groupServer) GetGroupsInfo(ctx context.Context, req *pbgroup.GetGroupsI | |||||||
| 	}, nil | 	}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (g *groupServer) GetGroupApplicationUnhandledCount(ctx context.Context, req *pbgroup.GetGroupApplicationUnhandledCountReq) (*pbgroup.GetGroupApplicationUnhandledCountResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	groupIDs, err := g.db.FindUserManagedGroupID(ctx, req.UserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	count, err := g.db.GetGroupApplicationUnhandledCount(ctx, groupIDs, req.Time) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return &pbgroup.GetGroupApplicationUnhandledCountResp{ | ||||||
|  | 		Count: count, | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (g *groupServer) getGroupsInfo(ctx context.Context, groupIDs []string) ([]*sdkws.GroupInfo, error) { | func (g *groupServer) getGroupsInfo(ctx context.Context, groupIDs []string) ([]*sdkws.GroupInfo, error) { | ||||||
| 	if len(groupIDs) == 0 { | 	if len(groupIDs) == 0 { | ||||||
| 		return nil, nil | 		return nil, nil | ||||||
| @ -944,7 +987,7 @@ func (g *groupServer) JoinGroup(ctx context.Context, req *pbgroup.JoinGroupReq) | |||||||
| 	if err = g.db.CreateGroupRequest(ctx, []*model.GroupRequest{&groupRequest}); err != nil { | 	if err = g.db.CreateGroupRequest(ctx, []*model.GroupRequest{&groupRequest}); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	g.notification.JoinGroupApplicationNotification(ctx, req) | 	g.notification.JoinGroupApplicationNotification(ctx, req, &groupRequest) | ||||||
| 	return &pbgroup.JoinGroupResp{}, nil | 	return &pbgroup.JoinGroupResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -1327,11 +1370,17 @@ func (g *groupServer) GetGroupMembersCMS(ctx context.Context, req *pbgroup.GetGr | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetUserReqApplicationList(ctx context.Context, req *pbgroup.GetUserReqApplicationListReq) (*pbgroup.GetUserReqApplicationListResp, error) { | func (g *groupServer) GetUserReqApplicationList(ctx context.Context, req *pbgroup.GetUserReqApplicationListReq) (*pbgroup.GetUserReqApplicationListResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	user, err := g.userClient.GetUserInfo(ctx, req.UserID) | 	user, err := g.userClient.GetUserInfo(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	total, requests, err := g.db.PageGroupRequestUser(ctx, req.UserID, req.Pagination) | 	handleResults := datautil.Slice(req.HandleResults, func(e int32) int { | ||||||
|  | 		return int(e) | ||||||
|  | 	}) | ||||||
|  | 	total, requests, err := g.db.PageGroupRequestUser(ctx, req.UserID, req.GroupIDs, handleResults, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -1772,6 +1821,9 @@ func (g *groupServer) GetGroupMemberRoleLevel(ctx context.Context, req *pbgroup. | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetGroupUsersReqApplicationList(ctx context.Context, req *pbgroup.GetGroupUsersReqApplicationListReq) (*pbgroup.GetGroupUsersReqApplicationListResp, error) { | func (g *groupServer) GetGroupUsersReqApplicationList(ctx context.Context, req *pbgroup.GetGroupUsersReqApplicationListReq) (*pbgroup.GetGroupUsersReqApplicationListResp, error) { | ||||||
|  | 	if err := g.CheckGroupAdmin(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	requests, err := g.db.FindGroupRequests(ctx, req.GroupID, req.UserIDs) | 	requests, err := g.db.FindGroupRequests(ctx, req.GroupID, req.UserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
| @ -20,6 +20,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/google/uuid" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 
 | 
 | ||||||
| 	"go.mongodb.org/mongo-driver/mongo" | 	"go.mongodb.org/mongo-driver/mongo" | ||||||
| @ -365,13 +366,46 @@ func (g *NotificationSender) GroupInfoSetAnnouncementNotification(ctx context.Co | |||||||
| 	g.Notification(ctx, mcontext.GetOpUserID(ctx), tips.Group.GroupID, constant.GroupInfoSetAnnouncementNotification, tips, notification.WithRpcGetUserName(), notification.WithSendMessage(sendMessage)) | 	g.Notification(ctx, mcontext.GetOpUserID(ctx), tips.Group.GroupID, constant.GroupInfoSetAnnouncementNotification, tips, notification.WithRpcGetUserName(), notification.WithSendMessage(sendMessage)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *NotificationSender) JoinGroupApplicationNotification(ctx context.Context, req *pbgroup.JoinGroupReq) { | func (g *NotificationSender) uuid() string { | ||||||
|  | 	return uuid.New().String() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (g *NotificationSender) getGroupRequest(ctx context.Context, groupID string, userID string) (*sdkws.GroupRequest, error) { | ||||||
|  | 	request, err := g.db.TakeGroupRequest(ctx, groupID, userID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	users, err := g.getUsersInfo(ctx, []string{userID}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	if len(users) == 0 { | ||||||
|  | 		return nil, servererrs.ErrUserIDNotFound.WrapMsg(fmt.Sprintf("user %s not found", userID)) | ||||||
|  | 	} | ||||||
|  | 	info, ok := users[0].(*sdkws.UserInfo) | ||||||
|  | 	if !ok { | ||||||
|  | 		info = &sdkws.UserInfo{ | ||||||
|  | 			UserID:   users[0].GetUserID(), | ||||||
|  | 			Nickname: users[0].GetNickname(), | ||||||
|  | 			FaceURL:  users[0].GetFaceURL(), | ||||||
|  | 			Ex:       users[0].GetEx(), | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return convert.Db2PbGroupRequest(request, info, nil), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (g *NotificationSender) JoinGroupApplicationNotification(ctx context.Context, req *pbgroup.JoinGroupReq, dbReq *model.GroupRequest) { | ||||||
| 	var err error | 	var err error | ||||||
| 	defer func() { | 	defer func() { | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
|  | 	request, err := g.getGroupRequest(ctx, dbReq.GroupID, dbReq.UserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.ZError(ctx, "JoinGroupApplicationNotification getGroupRequest", err, "dbReq", dbReq) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
| 	var group *sdkws.GroupInfo | 	var group *sdkws.GroupInfo | ||||||
| 	group, err = g.getGroupInfo(ctx, req.GroupID) | 	group, err = g.getGroupInfo(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -387,7 +421,13 @@ func (g *NotificationSender) JoinGroupApplicationNotification(ctx context.Contex | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	userIDs = append(userIDs, req.InviterUserID, mcontext.GetOpUserID(ctx)) | 	userIDs = append(userIDs, req.InviterUserID, mcontext.GetOpUserID(ctx)) | ||||||
| 	tips := &sdkws.JoinGroupApplicationTips{Group: group, Applicant: user, ReqMsg: req.ReqMessage} | 	tips := &sdkws.JoinGroupApplicationTips{ | ||||||
|  | 		Group:     group, | ||||||
|  | 		Applicant: user, | ||||||
|  | 		ReqMsg:    req.ReqMessage, | ||||||
|  | 		Uuid:      g.uuid(), | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	for _, userID := range datautil.Distinct(userIDs) { | 	for _, userID := range datautil.Distinct(userIDs) { | ||||||
| 		g.Notification(ctx, mcontext.GetOpUserID(ctx), userID, constant.JoinGroupApplicationNotification, tips) | 		g.Notification(ctx, mcontext.GetOpUserID(ctx), userID, constant.JoinGroupApplicationNotification, tips) | ||||||
| 	} | 	} | ||||||
| @ -417,6 +457,11 @@ func (g *NotificationSender) GroupApplicationAcceptedNotification(ctx context.Co | |||||||
| 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
|  | 	request, err := g.getGroupRequest(ctx, req.GroupID, req.FromUserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.ZError(ctx, "GroupApplicationAcceptedNotification getGroupRequest", err, "req", req) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
| 	var group *sdkws.GroupInfo | 	var group *sdkws.GroupInfo | ||||||
| 	group, err = g.getGroupInfo(ctx, req.GroupID) | 	group, err = g.getGroupInfo(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -432,8 +477,14 @@ func (g *NotificationSender) GroupApplicationAcceptedNotification(ctx context.Co | |||||||
| 	if err = g.fillOpUser(ctx, &opUser, group.GroupID); err != nil { | 	if err = g.fillOpUser(ctx, &opUser, group.GroupID); err != nil { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  | 	tips := &sdkws.GroupApplicationAcceptedTips{ | ||||||
|  | 		Group:     group, | ||||||
|  | 		OpUser:    opUser, | ||||||
|  | 		HandleMsg: req.HandledMsg, | ||||||
|  | 		Uuid:      g.uuid(), | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	for _, userID := range append(userIDs, req.FromUserID) { | 	for _, userID := range append(userIDs, req.FromUserID) { | ||||||
| 		tips := &sdkws.GroupApplicationAcceptedTips{Group: group, OpUser: opUser, HandleMsg: req.HandledMsg} |  | ||||||
| 		if userID == req.FromUserID { | 		if userID == req.FromUserID { | ||||||
| 			tips.ReceiverAs = applicantReceiver | 			tips.ReceiverAs = applicantReceiver | ||||||
| 		} else { | 		} else { | ||||||
| @ -450,6 +501,11 @@ func (g *NotificationSender) GroupApplicationRejectedNotification(ctx context.Co | |||||||
| 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
|  | 	request, err := g.getGroupRequest(ctx, req.GroupID, req.FromUserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.ZError(ctx, "GroupApplicationAcceptedNotification getGroupRequest", err, "req", req) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
| 	var group *sdkws.GroupInfo | 	var group *sdkws.GroupInfo | ||||||
| 	group, err = g.getGroupInfo(ctx, req.GroupID) | 	group, err = g.getGroupInfo(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -465,8 +521,14 @@ func (g *NotificationSender) GroupApplicationRejectedNotification(ctx context.Co | |||||||
| 	if err = g.fillOpUser(ctx, &opUser, group.GroupID); err != nil { | 	if err = g.fillOpUser(ctx, &opUser, group.GroupID); err != nil { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  | 	tips := &sdkws.GroupApplicationRejectedTips{ | ||||||
|  | 		Group:     group, | ||||||
|  | 		OpUser:    opUser, | ||||||
|  | 		HandleMsg: req.HandledMsg, | ||||||
|  | 		Uuid:      g.uuid(), | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	for _, userID := range append(userIDs, req.FromUserID) { | 	for _, userID := range append(userIDs, req.FromUserID) { | ||||||
| 		tips := &sdkws.GroupApplicationAcceptedTips{Group: group, OpUser: opUser, HandleMsg: req.HandledMsg} |  | ||||||
| 		if userID == req.FromUserID { | 		if userID == req.FromUserID { | ||||||
| 			tips.ReceiverAs = applicantReceiver | 			tips.ReceiverAs = applicantReceiver | ||||||
| 		} else { | 		} else { | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/notification/common_user" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/tools/mq/memamq" | 	"github.com/openimsdk/tools/mq/memamq" | ||||||
| @ -100,23 +101,24 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	userClient := rpcli.NewUserClient(userConn) | 	userClient := rpcli.NewUserClient(userConn) | ||||||
| 
 | 	database := controller.NewFriendDatabase( | ||||||
|  | 		friendMongoDB, | ||||||
|  | 		friendRequestMongoDB, | ||||||
|  | 		redis.NewFriendCacheRedis(rdb, &config.LocalCacheConfig, friendMongoDB), | ||||||
|  | 		mgocli.GetTx(), | ||||||
|  | 	) | ||||||
| 	// Initialize notification sender | 	// Initialize notification sender | ||||||
| 	notificationSender := NewFriendNotificationSender( | 	notificationSender := NewFriendNotificationSender( | ||||||
| 		&config.NotificationConfig, | 		&config.NotificationConfig, | ||||||
| 		rpcli.NewMsgClient(msgConn), | 		rpcli.NewMsgClient(msgConn), | ||||||
| 		WithRpcFunc(userClient.GetUsersInfo), | 		WithRpcFunc(userClient.GetUsersInfo), | ||||||
|  | 		WithFriendDB(database), | ||||||
| 	) | 	) | ||||||
| 	localcache.InitLocalCache(&config.LocalCacheConfig) | 	localcache.InitLocalCache(&config.LocalCacheConfig) | ||||||
| 
 | 
 | ||||||
| 	// Register Friend server with refactored MongoDB and Redis integrations | 	// Register Friend server with refactored MongoDB and Redis integrations | ||||||
| 	relation.RegisterFriendServer(server, &friendServer{ | 	relation.RegisterFriendServer(server, &friendServer{ | ||||||
| 		db: controller.NewFriendDatabase( | 		db: database, | ||||||
| 			friendMongoDB, |  | ||||||
| 			friendRequestMongoDB, |  | ||||||
| 			redis.NewFriendCacheRedis(rdb, &config.LocalCacheConfig, friendMongoDB), |  | ||||||
| 			mgocli.GetTx(), |  | ||||||
| 		), |  | ||||||
| 		blackDatabase: controller.NewBlackDatabase( | 		blackDatabase: controller.NewBlackDatabase( | ||||||
| 			blackMongoDB, | 			blackMongoDB, | ||||||
| 			redis.NewBlackCacheRedis(rdb, &config.LocalCacheConfig, blackMongoDB), | 			redis.NewBlackCacheRedis(rdb, &config.LocalCacheConfig, blackMongoDB), | ||||||
| @ -328,7 +330,7 @@ func (s *friendServer) GetDesignatedFriendsApply(ctx context.Context, req *relat | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	resp = &relation.GetDesignatedFriendsApplyResp{} | 	resp = &relation.GetDesignatedFriendsApplyResp{} | ||||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userClient.GetUsersInfoMap) | 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.getCommonUserMap) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -341,13 +343,16 @@ func (s *friendServer) GetPaginationFriendsApplyTo(ctx context.Context, req *rel | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	total, friendRequests, err := s.db.PageFriendRequestToMe(ctx, req.UserID, req.Pagination) | 	handleResults := datautil.Slice(req.HandleResults, func(e int32) int { | ||||||
|  | 		return int(e) | ||||||
|  | 	}) | ||||||
|  | 	total, friendRequests, err := s.db.PageFriendRequestToMe(ctx, req.UserID, handleResults, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp = &relation.GetPaginationFriendsApplyToResp{} | 	resp = &relation.GetPaginationFriendsApplyToResp{} | ||||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userClient.GetUsersInfoMap) | 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.getCommonUserMap) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -358,18 +363,20 @@ func (s *friendServer) GetPaginationFriendsApplyTo(ctx context.Context, req *rel | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) GetPaginationFriendsApplyFrom(ctx context.Context, req *relation.GetPaginationFriendsApplyFromReq) (resp *relation.GetPaginationFriendsApplyFromResp, err error) { | func (s *friendServer) GetPaginationFriendsApplyFrom(ctx context.Context, req *relation.GetPaginationFriendsApplyFromReq) (resp *relation.GetPaginationFriendsApplyFromResp, err error) { | ||||||
| 	resp = &relation.GetPaginationFriendsApplyFromResp{} |  | ||||||
| 
 |  | ||||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	total, friendRequests, err := s.db.PageFriendRequestFromMe(ctx, req.UserID, req.Pagination) | 	handleResults := datautil.Slice(req.HandleResults, func(e int32) int { | ||||||
|  | 		return int(e) | ||||||
|  | 	}) | ||||||
|  | 	total, friendRequests, err := s.db.PageFriendRequestFromMe(ctx, req.UserID, handleResults, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userClient.GetUsersInfoMap) | 	resp = &relation.GetPaginationFriendsApplyFromResp{} | ||||||
|  | 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.getCommonUserMap) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -544,3 +551,28 @@ func (s *friendServer) UpdateFriends(ctx context.Context, req *relation.UpdateFr | |||||||
| 	s.notificationSender.FriendsInfoUpdateNotification(ctx, req.OwnerUserID, req.FriendUserIDs) | 	s.notificationSender.FriendsInfoUpdateNotification(ctx, req.OwnerUserID, req.FriendUserIDs) | ||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (s *friendServer) GetSelfUnhandledApplyCount(ctx context.Context, req *relation.GetSelfUnhandledApplyCountReq) (*relation.GetSelfUnhandledApplyCountResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	count, err := s.db.GetUnhandledCount(ctx, req.UserID, req.Time) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return &relation.GetSelfUnhandledApplyCountResp{ | ||||||
|  | 		Count: count, | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *friendServer) getCommonUserMap(ctx context.Context, userIDs []string) (map[string]common_user.CommonUser, error) { | ||||||
|  | 	users, err := s.userClient.GetUsersInfo(ctx, userIDs) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return datautil.SliceToMapAny(users, func(e *sdkws.UserInfo) (string, common_user.CommonUser) { | ||||||
|  | 		return e.UserID, e | ||||||
|  | 	}), nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -19,6 +19,9 @@ import ( | |||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 	"github.com/openimsdk/protocol/msg" | 	"github.com/openimsdk/protocol/msg" | ||||||
|  | 	"github.com/openimsdk/tools/errs" | ||||||
|  | 	"github.com/openimsdk/tools/log" | ||||||
|  | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/versionctx" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/versionctx" | ||||||
| @ -52,9 +55,7 @@ func WithFriendDB(db controller.FriendDatabase) friendNotificationSenderOptions | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func WithDBFunc( | func WithDBFunc(fn func(ctx context.Context, userIDs []string) (users []*relationtb.User, err error)) friendNotificationSenderOptions { | ||||||
| 	fn func(ctx context.Context, userIDs []string) (users []*relationtb.User, err error), |  | ||||||
| ) friendNotificationSenderOptions { |  | ||||||
| 	return func(s *FriendNotificationSender) { | 	return func(s *FriendNotificationSender) { | ||||||
| 		f := func(ctx context.Context, userIDs []string) (result []common_user.CommonUser, err error) { | 		f := func(ctx context.Context, userIDs []string) (result []common_user.CommonUser, err error) { | ||||||
| 			users, err := fn(ctx, userIDs) | 			users, err := fn(ctx, userIDs) | ||||||
| @ -70,9 +71,7 @@ func WithDBFunc( | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func WithRpcFunc( | func WithRpcFunc(fn func(ctx context.Context, userIDs []string) ([]*sdkws.UserInfo, error)) friendNotificationSenderOptions { | ||||||
| 	fn func(ctx context.Context, userIDs []string) ([]*sdkws.UserInfo, error), |  | ||||||
| ) friendNotificationSenderOptions { |  | ||||||
| 	return func(s *FriendNotificationSender) { | 	return func(s *FriendNotificationSender) { | ||||||
| 		f := func(ctx context.Context, userIDs []string) (result []common_user.CommonUser, err error) { | 		f := func(ctx context.Context, userIDs []string) (result []common_user.CommonUser, err error) { | ||||||
| 			users, err := fn(ctx, userIDs) | 			users, err := fn(ctx, userIDs) | ||||||
| @ -100,10 +99,7 @@ func NewFriendNotificationSender(conf *config.Notification, msgClient *rpcli.Msg | |||||||
| 	return f | 	return f | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) getUsersInfoMap( | func (f *FriendNotificationSender) getUsersInfoMap(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	userIDs []string, |  | ||||||
| ) (map[string]*sdkws.UserInfo, error) { |  | ||||||
| 	users, err := f.getUsersInfo(ctx, userIDs) | 	users, err := f.getUsersInfo(ctx, userIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -116,10 +112,7 @@ func (f *FriendNotificationSender) getUsersInfoMap( | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| //nolint:unused | //nolint:unused | ||||||
| func (f *FriendNotificationSender) getFromToUserNickname( | func (f *FriendNotificationSender) getFromToUserNickname(ctx context.Context, fromUserID, toUserID string) (string, string, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	fromUserID, toUserID string, |  | ||||||
| ) (string, string, error) { |  | ||||||
| 	users, err := f.getUsersInfoMap(ctx, []string{fromUserID, toUserID}) | 	users, err := f.getUsersInfoMap(ctx, []string{fromUserID, toUserID}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", "", nil | 		return "", "", nil | ||||||
| @ -132,60 +125,107 @@ func (f *FriendNotificationSender) UserInfoUpdatedNotification(ctx context.Conte | |||||||
| 	f.Notification(ctx, mcontext.GetOpUserID(ctx), changedUserID, constant.UserInfoUpdatedNotification, &tips) | 	f.Notification(ctx, mcontext.GetOpUserID(ctx), changedUserID, constant.UserInfoUpdatedNotification, &tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (f *FriendNotificationSender) getCommonUserMap(ctx context.Context, userIDs []string) (map[string]common_user.CommonUser, error) { | ||||||
|  | 	users, err := f.getUsersInfo(ctx, userIDs) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return datautil.SliceToMap(users, func(e common_user.CommonUser) string { | ||||||
|  | 		return e.GetUserID() | ||||||
|  | 	}), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (f *FriendNotificationSender) getFriendRequests(ctx context.Context, fromUserID, toUserID string) (*sdkws.FriendRequest, error) { | ||||||
|  | 	if f.db == nil { | ||||||
|  | 		return nil, errs.ErrInternalServer.WithDetail("db is nil") | ||||||
|  | 	} | ||||||
|  | 	friendRequests, err := f.db.FindBothFriendRequests(ctx, fromUserID, toUserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	requests, err := convert.FriendRequestDB2Pb(ctx, friendRequests, f.getCommonUserMap) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	for _, request := range requests { | ||||||
|  | 		if request.FromUserID == fromUserID && request.ToUserID == toUserID { | ||||||
|  | 			return request, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil, errs.ErrRecordNotFound.WrapMsg("friend request not found", "fromUserID", fromUserID, "toUserID", toUserID) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (f *FriendNotificationSender) FriendApplicationAddNotification(ctx context.Context, req *relation.ApplyToAddFriendReq) { | func (f *FriendNotificationSender) FriendApplicationAddNotification(ctx context.Context, req *relation.ApplyToAddFriendReq) { | ||||||
| 	tips := sdkws.FriendApplicationTips{FromToUserID: &sdkws.FromToUserID{ | 	request, err := f.getFriendRequests(ctx, req.FromUserID, req.ToUserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.ZError(ctx, "FriendApplicationAddNotification get friend request", err, "fromUserID", req.FromUserID, "toUserID", req.ToUserID) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	tips := sdkws.FriendApplicationTips{ | ||||||
|  | 		FromToUserID: &sdkws.FromToUserID{ | ||||||
| 			FromUserID: req.FromUserID, | 			FromUserID: req.FromUserID, | ||||||
| 			ToUserID:   req.ToUserID, | 			ToUserID:   req.ToUserID, | ||||||
| 	}} | 		}, | ||||||
|  | 		Request: request, | ||||||
|  | 	} | ||||||
| 	f.Notification(ctx, req.FromUserID, req.ToUserID, constant.FriendApplicationNotification, &tips) | 	f.Notification(ctx, req.FromUserID, req.ToUserID, constant.FriendApplicationNotification, &tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) FriendApplicationAgreedNotification( | func (f *FriendNotificationSender) FriendApplicationAgreedNotification(ctx context.Context, req *relation.RespondFriendApplyReq) { | ||||||
| 	ctx context.Context, | 	request, err := f.getFriendRequests(ctx, req.FromUserID, req.ToUserID) | ||||||
| 	req *relation.RespondFriendApplyReq, | 	if err != nil { | ||||||
| ) { | 		log.ZError(ctx, "FriendApplicationAgreedNotification get friend request", err, "fromUserID", req.FromUserID, "toUserID", req.ToUserID) | ||||||
| 	tips := sdkws.FriendApplicationApprovedTips{FromToUserID: &sdkws.FromToUserID{ | 		return | ||||||
|  | 	} | ||||||
|  | 	tips := sdkws.FriendApplicationApprovedTips{ | ||||||
|  | 		FromToUserID: &sdkws.FromToUserID{ | ||||||
| 			FromUserID: req.FromUserID, | 			FromUserID: req.FromUserID, | ||||||
| 			ToUserID:   req.ToUserID, | 			ToUserID:   req.ToUserID, | ||||||
| 	}, HandleMsg: req.HandleMsg} | 		}, | ||||||
|  | 		HandleMsg: req.HandleMsg, | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	f.Notification(ctx, req.ToUserID, req.FromUserID, constant.FriendApplicationApprovedNotification, &tips) | 	f.Notification(ctx, req.ToUserID, req.FromUserID, constant.FriendApplicationApprovedNotification, &tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) FriendApplicationRefusedNotification( | func (f *FriendNotificationSender) FriendApplicationRefusedNotification(ctx context.Context, req *relation.RespondFriendApplyReq) { | ||||||
| 	ctx context.Context, | 	request, err := f.getFriendRequests(ctx, req.FromUserID, req.ToUserID) | ||||||
| 	req *relation.RespondFriendApplyReq, | 	if err != nil { | ||||||
| ) { | 		log.ZError(ctx, "FriendApplicationRefusedNotification get friend request", err, "fromUserID", req.FromUserID, "toUserID", req.ToUserID) | ||||||
| 	tips := sdkws.FriendApplicationApprovedTips{FromToUserID: &sdkws.FromToUserID{ | 		return | ||||||
|  | 	} | ||||||
|  | 	tips := sdkws.FriendApplicationRejectedTips{ | ||||||
|  | 		FromToUserID: &sdkws.FromToUserID{ | ||||||
| 			FromUserID: req.FromUserID, | 			FromUserID: req.FromUserID, | ||||||
| 			ToUserID:   req.ToUserID, | 			ToUserID:   req.ToUserID, | ||||||
| 	}, HandleMsg: req.HandleMsg} | 		}, | ||||||
|  | 		HandleMsg: req.HandleMsg, | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	f.Notification(ctx, req.ToUserID, req.FromUserID, constant.FriendApplicationRejectedNotification, &tips) | 	f.Notification(ctx, req.ToUserID, req.FromUserID, constant.FriendApplicationRejectedNotification, &tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) FriendAddedNotification( | //func (f *FriendNotificationSender) FriendAddedNotification(ctx context.Context, operationID, opUserID, fromUserID, toUserID string) error { | ||||||
| 	ctx context.Context, | //	tips := sdkws.FriendAddedTips{Friend: &sdkws.FriendInfo{}, OpUser: &sdkws.PublicUserInfo{}} | ||||||
| 	operationID, opUserID, fromUserID, toUserID string, | //	user, err := f.getUsersInfo(ctx, []string{opUserID}) | ||||||
| ) error { | //	if err != nil { | ||||||
| 	tips := sdkws.FriendAddedTips{Friend: &sdkws.FriendInfo{}, OpUser: &sdkws.PublicUserInfo{}} | //		return err | ||||||
| 	user, err := f.getUsersInfo(ctx, []string{opUserID}) | //	} | ||||||
| 	if err != nil { | //	tips.OpUser.UserID = user[0].GetUserID() | ||||||
| 		return err | //	tips.OpUser.Ex = user[0].GetEx() | ||||||
| 	} | //	tips.OpUser.Nickname = user[0].GetNickname() | ||||||
| 	tips.OpUser.UserID = user[0].GetUserID() | //	tips.OpUser.FaceURL = user[0].GetFaceURL() | ||||||
| 	tips.OpUser.Ex = user[0].GetEx() | //	friends, err := f.db.FindFriendsWithError(ctx, fromUserID, []string{toUserID}) | ||||||
| 	tips.OpUser.Nickname = user[0].GetNickname() | //	if err != nil { | ||||||
| 	tips.OpUser.FaceURL = user[0].GetFaceURL() | //		return err | ||||||
| 	friends, err := f.db.FindFriendsWithError(ctx, fromUserID, []string{toUserID}) | //	} | ||||||
| 	if err != nil { | //	tips.Friend, err = convert.FriendDB2Pb(ctx, friends[0], f.getUsersInfoMap) | ||||||
| 		return err | //	if err != nil { | ||||||
| 	} | //		return err | ||||||
| 	tips.Friend, err = convert.FriendDB2Pb(ctx, friends[0], f.getUsersInfoMap) | //	} | ||||||
| 	if err != nil { | //	f.Notification(ctx, fromUserID, toUserID, constant.FriendAddedNotification, &tips) | ||||||
| 		return err | //	return nil | ||||||
| 	} | //} | ||||||
| 	f.Notification(ctx, fromUserID, toUserID, constant.FriendAddedNotification, &tips) |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) FriendDeletedNotification(ctx context.Context, req *relation.DeleteFriendReq) { | func (f *FriendNotificationSender) FriendDeletedNotification(ctx context.Context, req *relation.DeleteFriendReq) { | ||||||
| 	tips := sdkws.FriendDeletedTips{FromToUserID: &sdkws.FromToUserID{ | 	tips := sdkws.FriendDeletedTips{FromToUserID: &sdkws.FromToUserID{ | ||||||
|  | |||||||
| @ -17,7 +17,9 @@ package convert | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/notification/common_user" | ||||||
| 	"github.com/openimsdk/protocol/relation" | 	"github.com/openimsdk/protocol/relation" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| @ -98,7 +100,7 @@ func FriendOnlyDB2PbOnly(friendsDB []*model.Friend) []*relation.FriendInfoOnly { | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func FriendRequestDB2Pb(ctx context.Context, friendRequests []*model.FriendRequest, getUsers func(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error)) ([]*sdkws.FriendRequest, error) { | func FriendRequestDB2Pb(ctx context.Context, friendRequests []*model.FriendRequest, getUsers func(ctx context.Context, userIDs []string) (map[string]common_user.CommonUser, error)) ([]*sdkws.FriendRequest, error) { | ||||||
| 	if len(friendRequests) == 0 { | 	if len(friendRequests) == 0 { | ||||||
| 		return nil, nil | 		return nil, nil | ||||||
| 	} | 	} | ||||||
| @ -117,11 +119,11 @@ func FriendRequestDB2Pb(ctx context.Context, friendRequests []*model.FriendReque | |||||||
| 		fromUser := users[friendRequest.FromUserID] | 		fromUser := users[friendRequest.FromUserID] | ||||||
| 		res = append(res, &sdkws.FriendRequest{ | 		res = append(res, &sdkws.FriendRequest{ | ||||||
| 			FromUserID:    friendRequest.FromUserID, | 			FromUserID:    friendRequest.FromUserID, | ||||||
| 			FromNickname:  fromUser.Nickname, | 			FromNickname:  fromUser.GetNickname(), | ||||||
| 			FromFaceURL:   fromUser.FaceURL, | 			FromFaceURL:   fromUser.GetFaceURL(), | ||||||
| 			ToUserID:      friendRequest.ToUserID, | 			ToUserID:      friendRequest.ToUserID, | ||||||
| 			ToNickname:    toUser.Nickname, | 			ToNickname:    toUser.GetNickname(), | ||||||
| 			ToFaceURL:     toUser.FaceURL, | 			ToFaceURL:     toUser.GetFaceURL(), | ||||||
| 			HandleResult:  friendRequest.HandleResult, | 			HandleResult:  friendRequest.HandleResult, | ||||||
| 			ReqMsg:        friendRequest.ReqMsg, | 			ReqMsg:        friendRequest.ReqMsg, | ||||||
| 			CreateTime:    friendRequest.CreateTime.UnixMilli(), | 			CreateTime:    friendRequest.CreateTime.UnixMilli(), | ||||||
|  | |||||||
| @ -17,10 +17,11 @@ package controller | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database/mgo" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database/mgo" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
| 	"time" |  | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| @ -61,10 +62,10 @@ type FriendDatabase interface { | |||||||
| 	PageInWhoseFriends(ctx context.Context, friendUserID string, pagination pagination.Pagination) (total int64, friends []*model.Friend, err error) | 	PageInWhoseFriends(ctx context.Context, friendUserID string, pagination pagination.Pagination) (total int64, friends []*model.Friend, err error) | ||||||
| 
 | 
 | ||||||
| 	// PageFriendRequestFromMe retrieves the friend requests sent by the user with pagination | 	// PageFriendRequestFromMe retrieves the friend requests sent by the user with pagination | ||||||
| 	PageFriendRequestFromMe(ctx context.Context, userID string, pagination pagination.Pagination) (total int64, friends []*model.FriendRequest, err error) | 	PageFriendRequestFromMe(ctx context.Context, userID string, handleResults []int, pagination pagination.Pagination) (total int64, friends []*model.FriendRequest, err error) | ||||||
| 
 | 
 | ||||||
| 	// PageFriendRequestToMe retrieves the friend requests received by the user with pagination | 	// PageFriendRequestToMe retrieves the friend requests received by the user with pagination | ||||||
| 	PageFriendRequestToMe(ctx context.Context, userID string, pagination pagination.Pagination) (total int64, friends []*model.FriendRequest, err error) | 	PageFriendRequestToMe(ctx context.Context, userID string, handleResults []int, pagination pagination.Pagination) (total int64, friends []*model.FriendRequest, err error) | ||||||
| 
 | 
 | ||||||
| 	// FindFriendsWithError fetches specified friends of a user and returns an error if any do not exist | 	// FindFriendsWithError fetches specified friends of a user and returns an error if any do not exist | ||||||
| 	FindFriendsWithError(ctx context.Context, ownerUserID string, friendUserIDs []string) (friends []*model.Friend, err error) | 	FindFriendsWithError(ctx context.Context, ownerUserID string, friendUserIDs []string) (friends []*model.Friend, err error) | ||||||
| @ -87,6 +88,8 @@ type FriendDatabase interface { | |||||||
| 	FindFriendUserID(ctx context.Context, friendUserID string) ([]string, error) | 	FindFriendUserID(ctx context.Context, friendUserID string) ([]string, error) | ||||||
| 
 | 
 | ||||||
| 	OwnerIncrVersion(ctx context.Context, ownerUserID string, friendUserIDs []string, state int32) error | 	OwnerIncrVersion(ctx context.Context, ownerUserID string, friendUserIDs []string, state int32) error | ||||||
|  | 
 | ||||||
|  | 	GetUnhandledCount(ctx context.Context, userID string, ts int64) (int64, error) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type friendDatabase struct { | type friendDatabase struct { | ||||||
| @ -334,13 +337,13 @@ func (f *friendDatabase) PageInWhoseFriends(ctx context.Context, friendUserID st | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // PageFriendRequestFromMe retrieves friend requests sent by me. It does not return an error if the result is empty. | // PageFriendRequestFromMe retrieves friend requests sent by me. It does not return an error if the result is empty. | ||||||
| func (f *friendDatabase) PageFriendRequestFromMe(ctx context.Context, userID string, pagination pagination.Pagination) (total int64, friends []*model.FriendRequest, err error) { | func (f *friendDatabase) PageFriendRequestFromMe(ctx context.Context, userID string, handleResults []int, pagination pagination.Pagination) (total int64, friends []*model.FriendRequest, err error) { | ||||||
| 	return f.friendRequest.FindFromUserID(ctx, userID, pagination) | 	return f.friendRequest.FindFromUserID(ctx, userID, handleResults, pagination) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // PageFriendRequestToMe retrieves friend requests received by me. It does not return an error if the result is empty. | // PageFriendRequestToMe retrieves friend requests received by me. It does not return an error if the result is empty. | ||||||
| func (f *friendDatabase) PageFriendRequestToMe(ctx context.Context, userID string, pagination pagination.Pagination) (total int64, friends []*model.FriendRequest, err error) { | func (f *friendDatabase) PageFriendRequestToMe(ctx context.Context, userID string, handleResults []int, pagination pagination.Pagination) (total int64, friends []*model.FriendRequest, err error) { | ||||||
| 	return f.friendRequest.FindToUserID(ctx, userID, pagination) | 	return f.friendRequest.FindToUserID(ctx, userID, handleResults, pagination) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // FindFriendsWithError retrieves specified friends' information for ownerUserID. Returns an error if any friend does not exist. | // FindFriendsWithError retrieves specified friends' information for ownerUserID. Returns an error if any friend does not exist. | ||||||
| @ -397,3 +400,7 @@ func (f *friendDatabase) OwnerIncrVersion(ctx context.Context, ownerUserID strin | |||||||
| 	} | 	} | ||||||
| 	return f.cache.DelMaxFriendVersion(ownerUserID).ChainExecDel(ctx) | 	return f.cache.DelMaxFriendVersion(ownerUserID).ChainExecDel(ctx) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (f *friendDatabase) GetUnhandledCount(ctx context.Context, userID string, ts int64) (int64, error) { | ||||||
|  | 	return f.friendRequest.GetUnhandledCount(ctx, userID, ts) | ||||||
|  | } | ||||||
|  | |||||||
| @ -68,7 +68,7 @@ type GroupDatabase interface { | |||||||
| 	// FindUserManagedGroupID retrieves group IDs managed by a user. | 	// FindUserManagedGroupID retrieves group IDs managed by a user. | ||||||
| 	FindUserManagedGroupID(ctx context.Context, userID string) (groupIDs []string, err error) | 	FindUserManagedGroupID(ctx context.Context, userID string) (groupIDs []string, err error) | ||||||
| 	// PageGroupRequest paginates through group requests for specified groups. | 	// PageGroupRequest paginates through group requests for specified groups. | ||||||
| 	PageGroupRequest(ctx context.Context, groupIDs []string, pagination pagination.Pagination) (int64, []*model.GroupRequest, error) | 	PageGroupRequest(ctx context.Context, groupIDs []string, handleResults []int, pagination pagination.Pagination) (int64, []*model.GroupRequest, error) | ||||||
| 	// GetGroupRoleLevelMemberIDs retrieves user IDs of group members with a specific role level. | 	// GetGroupRoleLevelMemberIDs retrieves user IDs of group members with a specific role level. | ||||||
| 	GetGroupRoleLevelMemberIDs(ctx context.Context, groupID string, roleLevel int32) ([]string, error) | 	GetGroupRoleLevelMemberIDs(ctx context.Context, groupID string, roleLevel int32) ([]string, error) | ||||||
| 
 | 
 | ||||||
| @ -100,7 +100,7 @@ type GroupDatabase interface { | |||||||
| 	// FindGroupRequests retrieves multiple group join requests. | 	// FindGroupRequests retrieves multiple group join requests. | ||||||
| 	FindGroupRequests(ctx context.Context, groupID string, userIDs []string) ([]*model.GroupRequest, error) | 	FindGroupRequests(ctx context.Context, groupID string, userIDs []string) ([]*model.GroupRequest, error) | ||||||
| 	// PageGroupRequestUser paginates through group join requests made by a user. | 	// PageGroupRequestUser paginates through group join requests made by a user. | ||||||
| 	PageGroupRequestUser(ctx context.Context, userID string, pagination pagination.Pagination) (int64, []*model.GroupRequest, error) | 	PageGroupRequestUser(ctx context.Context, userID string, groupIDs []string, handleResults []int, pagination pagination.Pagination) (int64, []*model.GroupRequest, error) | ||||||
| 
 | 
 | ||||||
| 	// CountTotal counts the total number of groups as of a certain date. | 	// CountTotal counts the total number of groups as of a certain date. | ||||||
| 	CountTotal(ctx context.Context, before *time.Time) (count int64, err error) | 	CountTotal(ctx context.Context, before *time.Time) (count int64, err error) | ||||||
| @ -124,6 +124,8 @@ type GroupDatabase interface { | |||||||
| 	SearchJoinGroup(ctx context.Context, userID string, keyword string, pagination pagination.Pagination) (int64, []*model.Group, error) | 	SearchJoinGroup(ctx context.Context, userID string, keyword string, pagination pagination.Pagination) (int64, []*model.Group, error) | ||||||
| 
 | 
 | ||||||
| 	FindJoinGroupID(ctx context.Context, userID string) ([]string, error) | 	FindJoinGroupID(ctx context.Context, userID string) ([]string, error) | ||||||
|  | 
 | ||||||
|  | 	GetGroupApplicationUnhandledCount(ctx context.Context, groupIDs []string, ts int64) (int64, error) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func NewGroupDatabase( | func NewGroupDatabase( | ||||||
| @ -304,8 +306,8 @@ func (g *groupDatabase) FindUserManagedGroupID(ctx context.Context, userID strin | |||||||
| 	return g.groupMemberDB.FindUserManagedGroupID(ctx, userID) | 	return g.groupMemberDB.FindUserManagedGroupID(ctx, userID) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupDatabase) PageGroupRequest(ctx context.Context, groupIDs []string, pagination pagination.Pagination) (int64, []*model.GroupRequest, error) { | func (g *groupDatabase) PageGroupRequest(ctx context.Context, groupIDs []string, handleResults []int, pagination pagination.Pagination) (int64, []*model.GroupRequest, error) { | ||||||
| 	return g.groupRequestDB.PageGroup(ctx, groupIDs, pagination) | 	return g.groupRequestDB.PageGroup(ctx, groupIDs, handleResults, pagination) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupDatabase) PageGetJoinGroup(ctx context.Context, userID string, pagination pagination.Pagination) (total int64, totalGroupMembers []*model.GroupMember, err error) { | func (g *groupDatabase) PageGetJoinGroup(ctx context.Context, userID string, pagination pagination.Pagination) (total int64, totalGroupMembers []*model.GroupMember, err error) { | ||||||
| @ -463,16 +465,12 @@ func (g *groupDatabase) CreateGroupRequest(ctx context.Context, requests []*mode | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupDatabase) TakeGroupRequest( | func (g *groupDatabase) TakeGroupRequest(ctx context.Context, groupID string, userID string) (*model.GroupRequest, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	groupID string, |  | ||||||
| 	userID string, |  | ||||||
| ) (*model.GroupRequest, error) { |  | ||||||
| 	return g.groupRequestDB.Take(ctx, groupID, userID) | 	return g.groupRequestDB.Take(ctx, groupID, userID) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupDatabase) PageGroupRequestUser(ctx context.Context, userID string, pagination pagination.Pagination) (int64, []*model.GroupRequest, error) { | func (g *groupDatabase) PageGroupRequestUser(ctx context.Context, userID string, groupIDs []string, handleResults []int, pagination pagination.Pagination) (int64, []*model.GroupRequest, error) { | ||||||
| 	return g.groupRequestDB.Page(ctx, userID, pagination) | 	return g.groupRequestDB.Page(ctx, userID, groupIDs, handleResults, pagination) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupDatabase) CountTotal(ctx context.Context, before *time.Time) (count int64, err error) { | func (g *groupDatabase) CountTotal(ctx context.Context, before *time.Time) (count int64, err error) { | ||||||
| @ -565,3 +563,7 @@ func (g *groupDatabase) MemberGroupIncrVersion(ctx context.Context, groupID stri | |||||||
| 	} | 	} | ||||||
| 	return g.cache.DelMaxGroupMemberVersion(groupID).ChainExecDel(ctx) | 	return g.cache.DelMaxGroupMemberVersion(groupID).ChainExecDel(ctx) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (g *groupDatabase) GetGroupApplicationUnhandledCount(ctx context.Context, groupIDs []string, ts int64) (int64, error) { | ||||||
|  | 	return g.groupRequestDB.GetUnhandledCount(ctx, groupIDs, ts) | ||||||
|  | } | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ package database | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
| 	"github.com/openimsdk/tools/db/pagination" | 	"github.com/openimsdk/tools/db/pagination" | ||||||
| ) | ) | ||||||
| @ -33,8 +34,9 @@ type FriendRequest interface { | |||||||
| 	Find(ctx context.Context, fromUserID, toUserID string) (friendRequest *model.FriendRequest, err error) | 	Find(ctx context.Context, fromUserID, toUserID string) (friendRequest *model.FriendRequest, err error) | ||||||
| 	Take(ctx context.Context, fromUserID, toUserID string) (friendRequest *model.FriendRequest, err error) | 	Take(ctx context.Context, fromUserID, toUserID string) (friendRequest *model.FriendRequest, err error) | ||||||
| 	// Get list of friend requests received by toUserID | 	// Get list of friend requests received by toUserID | ||||||
| 	FindToUserID(ctx context.Context, toUserID string, pagination pagination.Pagination) (total int64, friendRequests []*model.FriendRequest, err error) | 	FindToUserID(ctx context.Context, toUserID string, handleResults []int, pagination pagination.Pagination) (total int64, friendRequests []*model.FriendRequest, err error) | ||||||
| 	// Get list of friend requests sent by fromUserID | 	// Get list of friend requests sent by fromUserID | ||||||
| 	FindFromUserID(ctx context.Context, fromUserID string, pagination pagination.Pagination) (total int64, friendRequests []*model.FriendRequest, err error) | 	FindFromUserID(ctx context.Context, fromUserID string, handleResults []int, pagination pagination.Pagination) (total int64, friendRequests []*model.FriendRequest, err error) | ||||||
| 	FindBothFriendRequests(ctx context.Context, fromUserID, toUserID string) (friends []*model.FriendRequest, err error) | 	FindBothFriendRequests(ctx context.Context, fromUserID, toUserID string) (friends []*model.FriendRequest, err error) | ||||||
|  | 	GetUnhandledCount(ctx context.Context, userID string, ts int64) (int64, error) | ||||||
| } | } | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ package database | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
| 	"github.com/openimsdk/tools/db/pagination" | 	"github.com/openimsdk/tools/db/pagination" | ||||||
| ) | ) | ||||||
| @ -26,6 +27,7 @@ type GroupRequest interface { | |||||||
| 	UpdateHandler(ctx context.Context, groupID string, userID string, handledMsg string, handleResult int32) (err error) | 	UpdateHandler(ctx context.Context, groupID string, userID string, handledMsg string, handleResult int32) (err error) | ||||||
| 	Take(ctx context.Context, groupID string, userID string) (groupRequest *model.GroupRequest, err error) | 	Take(ctx context.Context, groupID string, userID string) (groupRequest *model.GroupRequest, err error) | ||||||
| 	FindGroupRequests(ctx context.Context, groupID string, userIDs []string) ([]*model.GroupRequest, error) | 	FindGroupRequests(ctx context.Context, groupID string, userIDs []string) ([]*model.GroupRequest, error) | ||||||
| 	Page(ctx context.Context, userID string, pagination pagination.Pagination) (total int64, groups []*model.GroupRequest, err error) | 	Page(ctx context.Context, userID string, groupIDs []string, handleResults []int, pagination pagination.Pagination) (total int64, groups []*model.GroupRequest, err error) | ||||||
| 	PageGroup(ctx context.Context, groupIDs []string, pagination pagination.Pagination) (total int64, groups []*model.GroupRequest, err error) | 	PageGroup(ctx context.Context, groupIDs []string, handleResults []int, pagination pagination.Pagination) (total int64, groups []*model.GroupRequest, err error) | ||||||
|  | 	GetUnhandledCount(ctx context.Context, groupIDs []string, ts int64) (int64, error) | ||||||
| } | } | ||||||
|  | |||||||
| @ -16,24 +16,32 @@ package mgo | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
|  | 	"go.mongodb.org/mongo-driver/mongo/options" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/tools/db/mongoutil" | 	"github.com/openimsdk/tools/db/mongoutil" | ||||||
| 	"github.com/openimsdk/tools/db/pagination" | 	"github.com/openimsdk/tools/db/pagination" | ||||||
| 	"go.mongodb.org/mongo-driver/bson" | 	"go.mongodb.org/mongo-driver/bson" | ||||||
| 	"go.mongodb.org/mongo-driver/mongo" | 	"go.mongodb.org/mongo-driver/mongo" | ||||||
| 	"go.mongodb.org/mongo-driver/mongo/options" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func NewFriendRequestMongo(db *mongo.Database) (database.FriendRequest, error) { | func NewFriendRequestMongo(db *mongo.Database) (database.FriendRequest, error) { | ||||||
| 	coll := db.Collection(database.FriendRequestName) | 	coll := db.Collection(database.FriendRequestName) | ||||||
| 	_, err := coll.Indexes().CreateOne(context.Background(), mongo.IndexModel{ | 	_, err := coll.Indexes().CreateMany(context.Background(), []mongo.IndexModel{ | ||||||
|  | 		{ | ||||||
| 			Keys: bson.D{ | 			Keys: bson.D{ | ||||||
| 				{Key: "from_user_id", Value: 1}, | 				{Key: "from_user_id", Value: 1}, | ||||||
| 				{Key: "to_user_id", Value: 1}, | 				{Key: "to_user_id", Value: 1}, | ||||||
| 			}, | 			}, | ||||||
| 			Options: options.Index().SetUnique(true), | 			Options: options.Index().SetUnique(true), | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Keys: bson.D{ | ||||||
|  | 				{Key: "create_time", Value: -1}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
| 	}) | 	}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -45,12 +53,24 @@ type FriendRequestMgo struct { | |||||||
| 	coll *mongo.Collection | 	coll *mongo.Collection | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendRequestMgo) FindToUserID(ctx context.Context, toUserID string, pagination pagination.Pagination) (total int64, friendRequests []*model.FriendRequest, err error) { | func (f *FriendRequestMgo) sort() any { | ||||||
| 	return mongoutil.FindPage[*model.FriendRequest](ctx, f.coll, bson.M{"to_user_id": toUserID}, pagination) | 	return bson.D{{Key: "create_time", Value: -1}} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendRequestMgo) FindFromUserID(ctx context.Context, fromUserID string, pagination pagination.Pagination) (total int64, friendRequests []*model.FriendRequest, err error) { | func (f *FriendRequestMgo) FindToUserID(ctx context.Context, toUserID string, handleResults []int, pagination pagination.Pagination) (total int64, friendRequests []*model.FriendRequest, err error) { | ||||||
| 	return mongoutil.FindPage[*model.FriendRequest](ctx, f.coll, bson.M{"from_user_id": fromUserID}, pagination) | 	filter := bson.M{"to_user_id": toUserID} | ||||||
|  | 	if len(handleResults) > 0 { | ||||||
|  | 		filter["handle_result"] = bson.M{"$in": handleResults} | ||||||
|  | 	} | ||||||
|  | 	return mongoutil.FindPage[*model.FriendRequest](ctx, f.coll, filter, pagination, options.Find().SetSort(f.sort())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (f *FriendRequestMgo) FindFromUserID(ctx context.Context, fromUserID string, handleResults []int, pagination pagination.Pagination) (total int64, friendRequests []*model.FriendRequest, err error) { | ||||||
|  | 	filter := bson.M{"from_user_id": fromUserID} | ||||||
|  | 	if len(handleResults) > 0 { | ||||||
|  | 		filter["handle_result"] = bson.M{"$in": handleResults} | ||||||
|  | 	} | ||||||
|  | 	return mongoutil.FindPage[*model.FriendRequest](ctx, f.coll, filter, pagination, options.Find().SetSort(f.sort())) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendRequestMgo) FindBothFriendRequests(ctx context.Context, fromUserID, toUserID string) (friends []*model.FriendRequest, err error) { | func (f *FriendRequestMgo) FindBothFriendRequests(ctx context.Context, fromUserID, toUserID string) (friends []*model.FriendRequest, err error) { | ||||||
| @ -110,3 +130,11 @@ func (f *FriendRequestMgo) Find(ctx context.Context, fromUserID, toUserID string | |||||||
| func (f *FriendRequestMgo) Take(ctx context.Context, fromUserID, toUserID string) (friendRequest *model.FriendRequest, err error) { | func (f *FriendRequestMgo) Take(ctx context.Context, fromUserID, toUserID string) (friendRequest *model.FriendRequest, err error) { | ||||||
| 	return f.Find(ctx, fromUserID, toUserID) | 	return f.Find(ctx, fromUserID, toUserID) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (f *FriendRequestMgo) GetUnhandledCount(ctx context.Context, userID string, ts int64) (int64, error) { | ||||||
|  | 	filter := bson.M{"to_user_id": userID, "handle_result": 0} | ||||||
|  | 	if ts != 0 { | ||||||
|  | 		filter["req_time"] = bson.M{"$gt": ts} | ||||||
|  | 	} | ||||||
|  | 	return mongoutil.Count(ctx, f.coll, filter) | ||||||
|  | } | ||||||
|  | |||||||
| @ -16,8 +16,10 @@ package mgo | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
|  | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/tools/db/mongoutil" | 	"github.com/openimsdk/tools/db/mongoutil" | ||||||
| 	"github.com/openimsdk/tools/db/pagination" | 	"github.com/openimsdk/tools/db/pagination" | ||||||
| @ -29,12 +31,19 @@ import ( | |||||||
| 
 | 
 | ||||||
| func NewGroupRequestMgo(db *mongo.Database) (database.GroupRequest, error) { | func NewGroupRequestMgo(db *mongo.Database) (database.GroupRequest, error) { | ||||||
| 	coll := db.Collection(database.GroupRequestName) | 	coll := db.Collection(database.GroupRequestName) | ||||||
| 	_, err := coll.Indexes().CreateOne(context.Background(), mongo.IndexModel{ | 	_, err := coll.Indexes().CreateMany(context.Background(), []mongo.IndexModel{ | ||||||
|  | 		{ | ||||||
| 			Keys: bson.D{ | 			Keys: bson.D{ | ||||||
| 				{Key: "group_id", Value: 1}, | 				{Key: "group_id", Value: 1}, | ||||||
| 				{Key: "user_id", Value: 1}, | 				{Key: "user_id", Value: 1}, | ||||||
| 			}, | 			}, | ||||||
| 			Options: options.Index().SetUnique(true), | 			Options: options.Index().SetUnique(true), | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Keys: bson.D{ | ||||||
|  | 				{Key: "req_time", Value: -1}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
| 	}) | 	}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, errs.Wrap(err) | 		return nil, errs.Wrap(err) | ||||||
| @ -66,10 +75,39 @@ func (g *GroupRequestMgo) FindGroupRequests(ctx context.Context, groupID string, | |||||||
| 	return mongoutil.Find[*model.GroupRequest](ctx, g.coll, bson.M{"group_id": groupID, "user_id": bson.M{"$in": userIDs}}) | 	return mongoutil.Find[*model.GroupRequest](ctx, g.coll, bson.M{"group_id": groupID, "user_id": bson.M{"$in": userIDs}}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *GroupRequestMgo) Page(ctx context.Context, userID string, pagination pagination.Pagination) (total int64, groups []*model.GroupRequest, err error) { | func (g *GroupRequestMgo) sort() any { | ||||||
| 	return mongoutil.FindPage[*model.GroupRequest](ctx, g.coll, bson.M{"user_id": userID}, pagination) | 	return bson.D{{Key: "req_time", Value: -1}} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *GroupRequestMgo) PageGroup(ctx context.Context, groupIDs []string, pagination pagination.Pagination) (total int64, groups []*model.GroupRequest, err error) { | func (g *GroupRequestMgo) Page(ctx context.Context, userID string, groupIDs []string, handleResults []int, pagination pagination.Pagination) (total int64, groups []*model.GroupRequest, err error) { | ||||||
| 	return mongoutil.FindPage[*model.GroupRequest](ctx, g.coll, bson.M{"group_id": bson.M{"$in": groupIDs}}, pagination) | 	filter := bson.M{"user_id": userID} | ||||||
|  | 	if len(groupIDs) > 0 { | ||||||
|  | 		filter["group_id"] = bson.M{"$in": datautil.Distinct(groupIDs)} | ||||||
|  | 	} | ||||||
|  | 	if len(handleResults) > 0 { | ||||||
|  | 		filter["handle_result"] = bson.M{"$in": handleResults} | ||||||
|  | 	} | ||||||
|  | 	return mongoutil.FindPage[*model.GroupRequest](ctx, g.coll, filter, pagination, options.Find().SetSort(g.sort())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (g *GroupRequestMgo) PageGroup(ctx context.Context, groupIDs []string, handleResults []int, pagination pagination.Pagination) (total int64, groups []*model.GroupRequest, err error) { | ||||||
|  | 	if len(groupIDs) == 0 { | ||||||
|  | 		return 0, nil, nil | ||||||
|  | 	} | ||||||
|  | 	filter := bson.M{"group_id": bson.M{"$in": groupIDs}} | ||||||
|  | 	if len(handleResults) > 0 { | ||||||
|  | 		filter["handle_result"] = bson.M{"$in": handleResults} | ||||||
|  | 	} | ||||||
|  | 	return mongoutil.FindPage[*model.GroupRequest](ctx, g.coll, filter, pagination, options.Find().SetSort(g.sort())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (g *GroupRequestMgo) GetUnhandledCount(ctx context.Context, groupIDs []string, ts int64) (int64, error) { | ||||||
|  | 	if len(groupIDs) == 0 { | ||||||
|  | 		return 0, nil | ||||||
|  | 	} | ||||||
|  | 	filter := bson.M{"group_id": bson.M{"$in": groupIDs}, "handle_result": 0} | ||||||
|  | 	if ts != 0 { | ||||||
|  | 		filter["req_time"] = bson.M{"$gt": ts} | ||||||
|  | 	} | ||||||
|  | 	return mongoutil.Count(ctx, g.coll, filter) | ||||||
| } | } | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user