mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-25 12:42:12 +08:00 
			
		
		
		
	feat: add rpc interface permission check (#3366)
* 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 (cherry picked from commit 748d783d36fc3f1e4eacaee64203b10e8e2b8744) # Conflicts: # internal/rpc/conversation/conversation.go # internal/rpc/group/cache.go # internal/rpc/group/statistics.go # internal/rpc/msg/send.go
This commit is contained in:
		
							parent
							
								
									ab32167b19
								
							
						
					
					
						commit
						505ce8aad1
					
				| @ -48,9 +48,9 @@ func (o *ConversationApi) SetConversations(c *gin.Context) { | ||||
| 	a2r.Call(c, conversation.ConversationClient.SetConversations, o.Client) | ||||
| } | ||||
| 
 | ||||
| func (o *ConversationApi) GetConversationOfflinePushUserIDs(c *gin.Context) { | ||||
| 	a2r.Call(c, conversation.ConversationClient.GetConversationOfflinePushUserIDs, o.Client) | ||||
| } | ||||
| //func (o *ConversationApi) GetConversationOfflinePushUserIDs(c *gin.Context) { | ||||
| //	a2r.Call(c, conversation.ConversationClient.GetConversationOfflinePushUserIDs, o.Client) | ||||
| //} | ||||
| 
 | ||||
| func (o *ConversationApi) GetFullOwnerConversationIDs(c *gin.Context) { | ||||
| 	a2r.Call(c, conversation.ConversationClient.GetFullOwnerConversationIDs, o.Client) | ||||
|  | ||||
| @ -262,7 +262,7 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | ||||
| 		conversationGroup.POST("/get_conversation", c.GetConversation) | ||||
| 		conversationGroup.POST("/get_conversations", c.GetConversations) | ||||
| 		conversationGroup.POST("/set_conversations", c.SetConversations) | ||||
| 		conversationGroup.POST("/get_conversation_offline_push_user_ids", c.GetConversationOfflinePushUserIDs) | ||||
| 		//conversationGroup.POST("/get_conversation_offline_push_user_ids", c.GetConversationOfflinePushUserIDs) | ||||
| 		conversationGroup.POST("/get_full_conversation_ids", c.GetFullOwnerConversationIDs) | ||||
| 		conversationGroup.POST("/get_incremental_conversations", c.GetIncrementalConversation) | ||||
| 		conversationGroup.POST("/get_owner_conversation", c.GetOwnerConversation) | ||||
|  | ||||
							
								
								
									
										117
									
								
								internal/rpc/conversation/callback.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								internal/rpc/conversation/callback.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,117 @@ | ||||
| package conversation | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	dbModel "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| ) | ||||
| 
 | ||||
| func (c *conversationServer) webhookBeforeCreateSingleChatConversations(ctx context.Context, before *config.BeforeConfig, req *dbModel.Conversation) error { | ||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||
| 		cbReq := &callbackstruct.CallbackBeforeCreateSingleChatConversationsReq{ | ||||
| 			CallbackCommand:  callbackstruct.CallbackBeforeCreateSingleChatConversationsCommand, | ||||
| 			OwnerUserID:      req.OwnerUserID, | ||||
| 			ConversationID:   req.ConversationID, | ||||
| 			ConversationType: req.ConversationType, | ||||
| 			UserID:           req.UserID, | ||||
| 			RecvMsgOpt:       req.RecvMsgOpt, | ||||
| 			IsPinned:         req.IsPinned, | ||||
| 			IsPrivateChat:    req.IsPrivateChat, | ||||
| 			BurnDuration:     req.BurnDuration, | ||||
| 			GroupAtType:      req.GroupAtType, | ||||
| 			AttachedInfo:     req.AttachedInfo, | ||||
| 			Ex:               req.Ex, | ||||
| 		} | ||||
| 
 | ||||
| 		resp := &callbackstruct.CallbackBeforeCreateSingleChatConversationsResp{} | ||||
| 
 | ||||
| 		if err := c.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 
 | ||||
| 		datautil.NotNilReplace(&req.RecvMsgOpt, resp.RecvMsgOpt) | ||||
| 		datautil.NotNilReplace(&req.IsPinned, resp.IsPinned) | ||||
| 		datautil.NotNilReplace(&req.IsPrivateChat, resp.IsPrivateChat) | ||||
| 		datautil.NotNilReplace(&req.BurnDuration, resp.BurnDuration) | ||||
| 		datautil.NotNilReplace(&req.GroupAtType, resp.GroupAtType) | ||||
| 		datautil.NotNilReplace(&req.AttachedInfo, resp.AttachedInfo) | ||||
| 		datautil.NotNilReplace(&req.Ex, resp.Ex) | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) webhookAfterCreateSingleChatConversations(ctx context.Context, after *config.AfterConfig, req *dbModel.Conversation) error { | ||||
| 	cbReq := &callbackstruct.CallbackAfterCreateSingleChatConversationsReq{ | ||||
| 		CallbackCommand:  callbackstruct.CallbackAfterCreateSingleChatConversationsCommand, | ||||
| 		OwnerUserID:      req.OwnerUserID, | ||||
| 		ConversationID:   req.ConversationID, | ||||
| 		ConversationType: req.ConversationType, | ||||
| 		UserID:           req.UserID, | ||||
| 		RecvMsgOpt:       req.RecvMsgOpt, | ||||
| 		IsPinned:         req.IsPinned, | ||||
| 		IsPrivateChat:    req.IsPrivateChat, | ||||
| 		BurnDuration:     req.BurnDuration, | ||||
| 		GroupAtType:      req.GroupAtType, | ||||
| 		AttachedInfo:     req.AttachedInfo, | ||||
| 		Ex:               req.Ex, | ||||
| 	} | ||||
| 
 | ||||
| 	c.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterCreateSingleChatConversationsResp{}, after) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) webhookBeforeCreateGroupChatConversations(ctx context.Context, before *config.BeforeConfig, req *dbModel.Conversation) error { | ||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||
| 		cbReq := &callbackstruct.CallbackBeforeCreateGroupChatConversationsReq{ | ||||
| 			CallbackCommand:  callbackstruct.CallbackBeforeCreateGroupChatConversationsCommand, | ||||
| 			ConversationID:   req.ConversationID, | ||||
| 			ConversationType: req.ConversationType, | ||||
| 			GroupID:          req.GroupID, | ||||
| 			RecvMsgOpt:       req.RecvMsgOpt, | ||||
| 			IsPinned:         req.IsPinned, | ||||
| 			IsPrivateChat:    req.IsPrivateChat, | ||||
| 			BurnDuration:     req.BurnDuration, | ||||
| 			GroupAtType:      req.GroupAtType, | ||||
| 			AttachedInfo:     req.AttachedInfo, | ||||
| 			Ex:               req.Ex, | ||||
| 		} | ||||
| 
 | ||||
| 		resp := &callbackstruct.CallbackBeforeCreateGroupChatConversationsResp{} | ||||
| 
 | ||||
| 		if err := c.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 
 | ||||
| 		datautil.NotNilReplace(&req.RecvMsgOpt, resp.RecvMsgOpt) | ||||
| 		datautil.NotNilReplace(&req.IsPinned, resp.IsPinned) | ||||
| 		datautil.NotNilReplace(&req.IsPrivateChat, resp.IsPrivateChat) | ||||
| 		datautil.NotNilReplace(&req.BurnDuration, resp.BurnDuration) | ||||
| 		datautil.NotNilReplace(&req.GroupAtType, resp.GroupAtType) | ||||
| 		datautil.NotNilReplace(&req.AttachedInfo, resp.AttachedInfo) | ||||
| 		datautil.NotNilReplace(&req.Ex, resp.Ex) | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) webhookAfterCreateGroupChatConversations(ctx context.Context, after *config.AfterConfig, req *dbModel.Conversation) error { | ||||
| 	cbReq := &callbackstruct.CallbackAfterCreateGroupChatConversationsReq{ | ||||
| 		CallbackCommand:  callbackstruct.CallbackAfterCreateGroupChatConversationsCommand, | ||||
| 		ConversationID:   req.ConversationID, | ||||
| 		ConversationType: req.ConversationType, | ||||
| 		GroupID:          req.GroupID, | ||||
| 		RecvMsgOpt:       req.RecvMsgOpt, | ||||
| 		IsPinned:         req.IsPinned, | ||||
| 		IsPrivateChat:    req.IsPrivateChat, | ||||
| 		BurnDuration:     req.BurnDuration, | ||||
| 		GroupAtType:      req.GroupAtType, | ||||
| 		AttachedInfo:     req.AttachedInfo, | ||||
| 		Ex:               req.Ex, | ||||
| 	} | ||||
| 
 | ||||
| 	c.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterCreateGroupChatConversationsResp{}, after) | ||||
| 	return nil | ||||
| } | ||||
| @ -19,9 +19,12 @@ import ( | ||||
| 	"sort" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||
| 
 | ||||
| 	"google.golang.org/grpc" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||
| @ -30,6 +33,7 @@ import ( | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database/mgo" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	dbModel "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/localcache" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| @ -39,7 +43,6 @@ import ( | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"google.golang.org/grpc" | ||||
| ) | ||||
| 
 | ||||
| type conversationServer struct { | ||||
| @ -49,9 +52,10 @@ type conversationServer struct { | ||||
| 	conversationNotificationSender *ConversationNotificationSender | ||||
| 	config                         *Config | ||||
| 
 | ||||
| 	userClient  *rpcli.UserClient | ||||
| 	msgClient   *rpcli.MsgClient | ||||
| 	groupClient *rpcli.GroupClient | ||||
| 	webhookClient *webhook.Client | ||||
| 	userClient    *rpcli.UserClient | ||||
| 	msgClient     *rpcli.MsgClient | ||||
| 	groupClient   *rpcli.GroupClient | ||||
| } | ||||
| 
 | ||||
| type Config struct { | ||||
| @ -60,6 +64,7 @@ type Config struct { | ||||
| 	MongodbConfig      config.Mongo | ||||
| 	NotificationConfig config.Notification | ||||
| 	Share              config.Share | ||||
| 	WebhooksConfig     config.Webhooks | ||||
| 	LocalCacheConfig   config.LocalCache | ||||
| 	Discovery          config.Discovery | ||||
| } | ||||
| @ -90,20 +95,32 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	msgClient := rpcli.NewMsgClient(msgConn) | ||||
| 
 | ||||
| 	cs := conversationServer{ | ||||
| 		config:        config, | ||||
| 		webhookClient: webhook.NewWebhookClient(config.WebhooksConfig.URL), | ||||
| 		userClient:    rpcli.NewUserClient(userConn), | ||||
| 		groupClient:   rpcli.NewGroupClient(groupConn), | ||||
| 		msgClient:     msgClient, | ||||
| 	} | ||||
| 
 | ||||
| 	cs.conversationNotificationSender = NewConversationNotificationSender(&config.NotificationConfig, msgClient) | ||||
| 	cs.conversationDatabase = controller.NewConversationDatabase( | ||||
| 		conversationDB, | ||||
| 		redis.NewConversationRedis(rdb, &config.LocalCacheConfig, conversationDB), | ||||
| 		mgocli.GetTx()) | ||||
| 
 | ||||
| 	localcache.InitLocalCache(&config.LocalCacheConfig) | ||||
| 	pbconversation.RegisterConversationServer(server, &conversationServer{ | ||||
| 		conversationNotificationSender: NewConversationNotificationSender(&config.NotificationConfig, msgClient), | ||||
| 		conversationDatabase: controller.NewConversationDatabase(conversationDB, | ||||
| 			redis.NewConversationRedis(rdb, &config.LocalCacheConfig, conversationDB), mgocli.GetTx()), | ||||
| 		userClient:  rpcli.NewUserClient(userConn), | ||||
| 		groupClient: rpcli.NewGroupClient(groupConn), | ||||
| 		msgClient:   msgClient, | ||||
| 	}) | ||||
| 	pbconversation.RegisterConversationServer(server, &cs) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetConversation(ctx context.Context, req *pbconversation.GetConversationReq) (*pbconversation.GetConversationResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversations, err := c.conversationDatabase.FindConversations(ctx, req.OwnerUserID, []string{req.ConversationID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -117,7 +134,9 @@ func (c *conversationServer) GetConversation(ctx context.Context, req *pbconvers | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetSortedConversationList(ctx context.Context, req *pbconversation.GetSortedConversationListReq) (resp *pbconversation.GetSortedConversationListResp, err error) { | ||||
| 	log.ZDebug(ctx, "GetSortedConversationList", "seqs", req, "userID", req.UserID) | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var conversationIDs []string | ||||
| 	if len(req.ConversationIDs) == 0 { | ||||
| 		conversationIDs, err = c.conversationDatabase.GetConversationIDs(ctx, req.UserID) | ||||
| @ -190,6 +209,9 @@ func (c *conversationServer) GetSortedConversationList(ctx context.Context, req | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetAllConversations(ctx context.Context, req *pbconversation.GetAllConversationsReq) (*pbconversation.GetAllConversationsResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversations, err := c.conversationDatabase.GetUserAllConversation(ctx, req.OwnerUserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -200,6 +222,9 @@ func (c *conversationServer) GetAllConversations(ctx context.Context, req *pbcon | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetConversations(ctx context.Context, req *pbconversation.GetConversationsReq) (*pbconversation.GetConversationsResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversations, err := c.getConversations(ctx, req.OwnerUserID, req.ConversationIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -220,6 +245,9 @@ func (c *conversationServer) getConversations(ctx context.Context, ownerUserID s | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) SetConversation(ctx context.Context, req *pbconversation.SetConversationReq) (*pbconversation.SetConversationResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.GetConversation().GetUserID()); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var conversation dbModel.Conversation | ||||
| 	if err := datautil.CopyStructFields(&conversation, req.Conversation); err != nil { | ||||
| 		return nil, err | ||||
| @ -234,8 +262,10 @@ func (c *conversationServer) SetConversation(ctx context.Context, req *pbconvers | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) SetConversations(ctx context.Context, req *pbconversation.SetConversationsReq) (*pbconversation.SetConversationsResp, error) { | ||||
| 	if req.Conversation == nil { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("conversation must not be nil") | ||||
| 	for _, userID := range req.UserIDs { | ||||
| 		if err := authverify.CheckAccess(ctx, userID); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	if req.Conversation.ConversationType == constant.WriteGroupChatType { | ||||
| 		groupInfo, err := c.groupClient.GetGroupInfo(ctx, req.Conversation.GroupID) | ||||
| @ -271,109 +301,29 @@ func (c *conversationServer) SetConversations(ctx context.Context, req *pbconver | ||||
| 	conversation.UserID = req.Conversation.UserID | ||||
| 	conversation.GroupID = req.Conversation.GroupID | ||||
| 
 | ||||
| 	m := make(map[string]any) | ||||
| 
 | ||||
| 	setConversationFieldsFunc := func() { | ||||
| 		if req.Conversation.RecvMsgOpt != nil { | ||||
| 			conversation.RecvMsgOpt = req.Conversation.RecvMsgOpt.Value | ||||
| 			m["recv_msg_opt"] = req.Conversation.RecvMsgOpt.Value | ||||
| 		} | ||||
| 		if req.Conversation.AttachedInfo != nil { | ||||
| 			conversation.AttachedInfo = req.Conversation.AttachedInfo.Value | ||||
| 			m["attached_info"] = req.Conversation.AttachedInfo.Value | ||||
| 		} | ||||
| 		if req.Conversation.Ex != nil { | ||||
| 			conversation.Ex = req.Conversation.Ex.Value | ||||
| 			m["ex"] = req.Conversation.Ex.Value | ||||
| 		} | ||||
| 		if req.Conversation.IsPinned != nil { | ||||
| 			conversation.IsPinned = req.Conversation.IsPinned.Value | ||||
| 			m["is_pinned"] = req.Conversation.IsPinned.Value | ||||
| 		} | ||||
| 		if req.Conversation.GroupAtType != nil { | ||||
| 			conversation.GroupAtType = req.Conversation.GroupAtType.Value | ||||
| 			m["group_at_type"] = req.Conversation.GroupAtType.Value | ||||
| 		} | ||||
| 		if req.Conversation.MsgDestructTime != nil { | ||||
| 			conversation.MsgDestructTime = req.Conversation.MsgDestructTime.Value | ||||
| 			m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | ||||
| 		} | ||||
| 		if req.Conversation.IsMsgDestruct != nil { | ||||
| 			conversation.IsMsgDestruct = req.Conversation.IsMsgDestruct.Value | ||||
| 			m["is_msg_destruct"] = req.Conversation.IsMsgDestruct.Value | ||||
| 		} | ||||
| 		if req.Conversation.BurnDuration != nil { | ||||
| 			conversation.BurnDuration = req.Conversation.BurnDuration.Value | ||||
| 			m["burn_duration"] = req.Conversation.BurnDuration.Value | ||||
| 		} | ||||
| 	m, conversation, err := UpdateConversationsMap(ctx, req) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	// set need set field in conversation | ||||
| 	setConversationFieldsFunc() | ||||
| 
 | ||||
| 	for userID := range conversationMap { | ||||
| 		unequal := len(m) | ||||
| 		unequal := UserUpdateCheckMap(ctx, userID, req.Conversation, conversationMap[userID]) | ||||
| 
 | ||||
| 		if req.Conversation.RecvMsgOpt != nil { | ||||
| 			if req.Conversation.RecvMsgOpt.Value == conversationMap[userID].RecvMsgOpt { | ||||
| 				unequal-- | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if req.Conversation.AttachedInfo != nil { | ||||
| 			if req.Conversation.AttachedInfo.Value == conversationMap[userID].AttachedInfo { | ||||
| 				unequal-- | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if req.Conversation.Ex != nil { | ||||
| 			if req.Conversation.Ex.Value == conversationMap[userID].Ex { | ||||
| 				unequal-- | ||||
| 			} | ||||
| 		} | ||||
| 		if req.Conversation.IsPinned != nil { | ||||
| 			if req.Conversation.IsPinned.Value == conversationMap[userID].IsPinned { | ||||
| 				unequal-- | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if req.Conversation.GroupAtType != nil { | ||||
| 			if req.Conversation.GroupAtType.Value == conversationMap[userID].GroupAtType { | ||||
| 				unequal-- | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if req.Conversation.MsgDestructTime != nil { | ||||
| 			if req.Conversation.MsgDestructTime.Value == conversationMap[userID].MsgDestructTime { | ||||
| 				unequal-- | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if req.Conversation.IsMsgDestruct != nil { | ||||
| 			if req.Conversation.IsMsgDestruct.Value == conversationMap[userID].IsMsgDestruct { | ||||
| 				unequal-- | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if req.Conversation.BurnDuration != nil { | ||||
| 			if req.Conversation.BurnDuration.Value == conversationMap[userID].BurnDuration { | ||||
| 				unequal-- | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if unequal > 0 { | ||||
| 		if unequal { | ||||
| 			needUpdateUsersList = append(needUpdateUsersList, userID) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if len(m) != 0 && len(needUpdateUsersList) != 0 { | ||||
| 		if err := c.conversationDatabase.SetUsersConversationFieldTx(ctx, needUpdateUsersList, &conversation, m); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 
 | ||||
| 		for _, v := range needUpdateUsersList { | ||||
| 			c.conversationNotificationSender.ConversationChangeNotification(ctx, v, []string{req.Conversation.ConversationID}) | ||||
| 		for _, userID := range needUpdateUsersList { | ||||
| 			c.conversationNotificationSender.ConversationChangeNotification(ctx, userID, []string{req.Conversation.ConversationID}) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if req.Conversation.IsPrivateChat != nil && req.Conversation.ConversationType != constant.ReadGroupChatType { | ||||
| 		var conversations []*dbModel.Conversation | ||||
| 		for _, ownerUserID := range req.UserIDs { | ||||
| @ -396,58 +346,94 @@ func (c *conversationServer) SetConversations(ctx context.Context, req *pbconver | ||||
| 	return &pbconversation.SetConversationsResp{}, nil | ||||
| } | ||||
| 
 | ||||
| // Get user IDs with "Do Not Disturb" enabled in super large groups. | ||||
| func (c *conversationServer) GetRecvMsgNotNotifyUserIDs(ctx context.Context, req *pbconversation.GetRecvMsgNotNotifyUserIDsReq) (*pbconversation.GetRecvMsgNotNotifyUserIDsResp, error) { | ||||
| 	return nil, errs.New("deprecated") | ||||
| func (c *conversationServer) UpdateConversationsByUser(ctx context.Context, req *pbconversation.UpdateConversationsByUserReq) (*pbconversation.UpdateConversationsByUserResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	m := make(map[string]any) | ||||
| 	if req.Ex != nil { | ||||
| 		m["ex"] = req.Ex.Value | ||||
| 	} | ||||
| 	if len(m) > 0 { | ||||
| 		if err := c.conversationDatabase.UpdateUserConversations(ctx, req.UserID, m); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	return &pbconversation.UpdateConversationsByUserResp{}, nil | ||||
| } | ||||
| 
 | ||||
| // create conversation without notification for msg redis transfer. | ||||
| func (c *conversationServer) CreateSingleChatConversations(ctx context.Context, | ||||
| 	req *pbconversation.CreateSingleChatConversationsReq, | ||||
| ) (*pbconversation.CreateSingleChatConversationsResp, error) { | ||||
| func (c *conversationServer) CreateSingleChatConversations(ctx context.Context, req *pbconversation.CreateSingleChatConversationsReq) (*pbconversation.CreateSingleChatConversationsResp, error) { | ||||
| 	var conversation dbModel.Conversation | ||||
| 	switch req.ConversationType { | ||||
| 	case constant.SingleChatType: | ||||
| 		var conversation dbModel.Conversation | ||||
| 		// sendUser create | ||||
| 		conversation.ConversationID = req.ConversationID | ||||
| 		conversation.ConversationType = req.ConversationType | ||||
| 		conversation.OwnerUserID = req.SendID | ||||
| 		conversation.UserID = req.RecvID | ||||
| 		if err := c.webhookBeforeCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.BeforeCreateSingleChatConversations, &conversation); err != nil && err != servererrs.ErrCallbackContinue { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 
 | ||||
| 		err := c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation}) | ||||
| 		if err != nil { | ||||
| 			log.ZWarn(ctx, "create conversation failed", err, "conversation", conversation) | ||||
| 		} | ||||
| 
 | ||||
| 		c.webhookAfterCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateSingleChatConversations, &conversation) | ||||
| 
 | ||||
| 		// recvUser create | ||||
| 		conversation2 := conversation | ||||
| 		conversation2.OwnerUserID = req.RecvID | ||||
| 		conversation2.UserID = req.SendID | ||||
| 		if err := c.webhookBeforeCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.BeforeCreateSingleChatConversations, &conversation); err != nil && err != servererrs.ErrCallbackContinue { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 
 | ||||
| 		err = c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation2}) | ||||
| 		if err != nil { | ||||
| 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | ||||
| 		} | ||||
| 
 | ||||
| 		c.webhookAfterCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateSingleChatConversations, &conversation2) | ||||
| 	case constant.NotificationChatType: | ||||
| 		var conversation dbModel.Conversation | ||||
| 		conversation.ConversationID = req.ConversationID | ||||
| 		conversation.ConversationType = req.ConversationType | ||||
| 		conversation.OwnerUserID = req.RecvID | ||||
| 		conversation.UserID = req.SendID | ||||
| 		if err := c.webhookBeforeCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.BeforeCreateSingleChatConversations, &conversation); err != nil && err != servererrs.ErrCallbackContinue { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 
 | ||||
| 		err := c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation}) | ||||
| 		if err != nil { | ||||
| 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | ||||
| 		} | ||||
| 
 | ||||
| 		c.webhookAfterCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateSingleChatConversations, &conversation) | ||||
| 	} | ||||
| 
 | ||||
| 	return &pbconversation.CreateSingleChatConversationsResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) CreateGroupChatConversations(ctx context.Context, req *pbconversation.CreateGroupChatConversationsReq) (*pbconversation.CreateGroupChatConversationsResp, error) { | ||||
| 	err := c.conversationDatabase.CreateGroupChatConversation(ctx, req.GroupID, req.UserIDs) | ||||
| 	var conversation dbModel.Conversation | ||||
| 
 | ||||
| 	conversation.ConversationID = msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID) | ||||
| 	conversation.GroupID = req.GroupID | ||||
| 	conversation.ConversationType = constant.ReadGroupChatType | ||||
| 
 | ||||
| 	if err := c.webhookBeforeCreateGroupChatConversations(ctx, &c.config.WebhooksConfig.BeforeCreateGroupChatConversations, &conversation); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	err := c.conversationDatabase.CreateGroupChatConversation(ctx, req.GroupID, req.UserIDs, &conversation) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversationID := msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID) | ||||
| 	if err := c.msgClient.SetUserConversationMaxSeq(ctx, conversationID, req.UserIDs, 0); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	c.webhookAfterCreateGroupChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateGroupChatConversations, &conversation) | ||||
| 	return &pbconversation.CreateGroupChatConversationsResp{}, nil | ||||
| } | ||||
| 
 | ||||
| @ -480,6 +466,9 @@ func (c *conversationServer) SetConversationMinSeq(ctx context.Context, req *pbc | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetConversationIDs(ctx context.Context, req *pbconversation.GetConversationIDsReq) (*pbconversation.GetConversationIDsResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversationIDs, err := c.conversationDatabase.GetConversationIDs(ctx, req.UserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -488,6 +477,9 @@ func (c *conversationServer) GetConversationIDs(ctx context.Context, req *pbconv | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetUserConversationIDsHash(ctx context.Context, req *pbconversation.GetUserConversationIDsHashReq) (*pbconversation.GetUserConversationIDsHashResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	hash, err := c.conversationDatabase.GetUserConversationIDsHash(ctx, req.OwnerUserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -495,10 +487,7 @@ func (c *conversationServer) GetUserConversationIDsHash(ctx context.Context, req | ||||
| 	return &pbconversation.GetUserConversationIDsHashResp{Hash: hash}, nil | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetConversationsByConversationID( | ||||
| 	ctx context.Context, | ||||
| 	req *pbconversation.GetConversationsByConversationIDReq, | ||||
| ) (*pbconversation.GetConversationsByConversationIDResp, error) { | ||||
| func (c *conversationServer) GetConversationsByConversationID(ctx context.Context, req *pbconversation.GetConversationsByConversationIDReq) (*pbconversation.GetConversationsByConversationIDResp, error) { | ||||
| 	conversations, err := c.conversationDatabase.GetConversationsByConversationID(ctx, req.ConversationIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -552,10 +541,7 @@ func (c *conversationServer) conversationSort(conversations map[int64]string, re | ||||
| 	resp.ConversationElems = append(resp.ConversationElems, cons...) | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) getConversationInfo( | ||||
| 	ctx context.Context, | ||||
| 	chatLogs map[string]*sdkws.MsgData, | ||||
| 	userID string) (map[string]*pbconversation.ConversationElem, error) { | ||||
| func (c *conversationServer) getConversationInfo(ctx context.Context, chatLogs map[string]*sdkws.MsgData, userID string) (map[string]*pbconversation.ConversationElem, error) { | ||||
| 	var ( | ||||
| 		sendIDs         []string | ||||
| 		groupIDs        []string | ||||
| @ -641,6 +627,11 @@ func (c *conversationServer) GetConversationNotReceiveMessageUserIDs(ctx context | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) UpdateConversation(ctx context.Context, req *pbconversation.UpdateConversationReq) (*pbconversation.UpdateConversationResp, error) { | ||||
| 	for _, userID := range req.UserIDs { | ||||
| 		if err := authverify.CheckAccess(ctx, userID); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	m := make(map[string]any) | ||||
| 	if req.RecvMsgOpt != nil { | ||||
| 		m["recv_msg_opt"] = req.RecvMsgOpt.Value | ||||
| @ -687,6 +678,9 @@ func (c *conversationServer) UpdateConversation(ctx context.Context, req *pbconv | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetOwnerConversation(ctx context.Context, req *pbconversation.GetOwnerConversationReq) (*pbconversation.GetOwnerConversationResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	total, conversations, err := c.conversationDatabase.GetOwnerConversation(ctx, req.UserID, req.Pagination) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -748,6 +742,9 @@ func (c *conversationServer) GetConversationsNeedClearMsg(ctx context.Context, _ | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetNotNotifyConversationIDs(ctx context.Context, req *pbconversation.GetNotNotifyConversationIDsReq) (*pbconversation.GetNotNotifyConversationIDsResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversationIDs, err := c.conversationDatabase.GetNotNotifyConversationIDs(ctx, req.UserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -756,6 +753,9 @@ func (c *conversationServer) GetNotNotifyConversationIDs(ctx context.Context, re | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetPinnedConversationIDs(ctx context.Context, req *pbconversation.GetPinnedConversationIDsReq) (*pbconversation.GetPinnedConversationIDsResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversationIDs, err := c.conversationDatabase.GetPinnedConversationIDs(ctx, req.UserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | ||||
							
								
								
									
										85
									
								
								internal/rpc/conversation/db_map.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								internal/rpc/conversation/db_map.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,85 @@ | ||||
| package conversation | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	dbModel "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"github.com/openimsdk/protocol/conversation" | ||||
| ) | ||||
| 
 | ||||
| func UpdateConversationsMap(ctx context.Context, req *conversation.SetConversationsReq) (m map[string]any, conversation dbModel.Conversation, err error) { | ||||
| 	m = make(map[string]any) | ||||
| 
 | ||||
| 	conversation.ConversationID = req.Conversation.ConversationID | ||||
| 	conversation.ConversationType = req.Conversation.ConversationType | ||||
| 	conversation.UserID = req.Conversation.UserID | ||||
| 	conversation.GroupID = req.Conversation.GroupID | ||||
| 
 | ||||
| 	if req.Conversation.RecvMsgOpt != nil { | ||||
| 		conversation.RecvMsgOpt = req.Conversation.RecvMsgOpt.Value | ||||
| 		m["recv_msg_opt"] = req.Conversation.RecvMsgOpt.Value | ||||
| 	} | ||||
| 
 | ||||
| 	if req.Conversation.AttachedInfo != nil { | ||||
| 		conversation.AttachedInfo = req.Conversation.AttachedInfo.Value | ||||
| 		m["attached_info"] = req.Conversation.AttachedInfo.Value | ||||
| 	} | ||||
| 
 | ||||
| 	if req.Conversation.Ex != nil { | ||||
| 		conversation.Ex = req.Conversation.Ex.Value | ||||
| 		m["ex"] = req.Conversation.Ex.Value | ||||
| 	} | ||||
| 	if req.Conversation.IsPinned != nil { | ||||
| 		conversation.IsPinned = req.Conversation.IsPinned.Value | ||||
| 		m["is_pinned"] = req.Conversation.IsPinned.Value | ||||
| 	} | ||||
| 	if req.Conversation.GroupAtType != nil { | ||||
| 		conversation.GroupAtType = req.Conversation.GroupAtType.Value | ||||
| 		m["group_at_type"] = req.Conversation.GroupAtType.Value | ||||
| 	} | ||||
| 	if req.Conversation.MsgDestructTime != nil { | ||||
| 		conversation.MsgDestructTime = req.Conversation.MsgDestructTime.Value | ||||
| 		m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | ||||
| 	} | ||||
| 	if req.Conversation.IsMsgDestruct != nil { | ||||
| 		conversation.IsMsgDestruct = req.Conversation.IsMsgDestruct.Value | ||||
| 		m["is_msg_destruct"] = req.Conversation.IsMsgDestruct.Value | ||||
| 	} | ||||
| 	if req.Conversation.BurnDuration != nil { | ||||
| 		conversation.BurnDuration = req.Conversation.BurnDuration.Value | ||||
| 		m["burn_duration"] = req.Conversation.BurnDuration.Value | ||||
| 	} | ||||
| 
 | ||||
| 	return m, conversation, nil | ||||
| } | ||||
| 
 | ||||
| func UserUpdateCheckMap(ctx context.Context, userID string, req *conversation.ConversationReq, conversation *dbModel.Conversation) (unequal bool) { | ||||
| 	unequal = false | ||||
| 
 | ||||
| 	if req.RecvMsgOpt != nil && conversation.RecvMsgOpt != req.RecvMsgOpt.Value { | ||||
| 		unequal = true | ||||
| 	} | ||||
| 	if req.AttachedInfo != nil && conversation.AttachedInfo != req.AttachedInfo.Value { | ||||
| 		unequal = true | ||||
| 	} | ||||
| 	if req.Ex != nil && conversation.Ex != req.Ex.Value { | ||||
| 		unequal = true | ||||
| 	} | ||||
| 	if req.IsPinned != nil && conversation.IsPinned != req.IsPinned.Value { | ||||
| 		unequal = true | ||||
| 	} | ||||
| 	if req.GroupAtType != nil && conversation.GroupAtType != req.GroupAtType.Value { | ||||
| 		unequal = true | ||||
| 	} | ||||
| 	if req.MsgDestructTime != nil && conversation.MsgDestructTime != req.MsgDestructTime.Value { | ||||
| 		unequal = true | ||||
| 	} | ||||
| 	if req.IsMsgDestruct != nil && conversation.IsMsgDestruct != req.IsMsgDestruct.Value { | ||||
| 		unequal = true | ||||
| 	} | ||||
| 	if req.BurnDuration != nil && conversation.BurnDuration != req.BurnDuration.Value { | ||||
| 		unequal = true | ||||
| 	} | ||||
| 
 | ||||
| 	return unequal | ||||
| } | ||||
| @ -17,11 +17,11 @@ package conversation | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/notification" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/notification" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| ) | ||||
|  | ||||
| @ -35,6 +35,9 @@ func (c *conversationServer) GetFullOwnerConversationIDs(ctx context.Context, re | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetIncrementalConversation(ctx context.Context, req *conversation.GetIncrementalConversationReq) (*conversation.GetIncrementalConversationResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	opt := incrversion.Option[*conversation.Conversation, conversation.GetIncrementalConversationResp]{ | ||||
| 		Ctx:             ctx, | ||||
| 		VersionKey:      req.UserID, | ||||
|  | ||||
| @ -17,13 +17,14 @@ package group | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | ||||
| 	pbgroup "github.com/openimsdk/protocol/group" | ||||
| ) | ||||
| 
 | ||||
| // GetGroupInfoCache get group info from cache. | ||||
| func (s *groupServer) GetGroupInfoCache(ctx context.Context, req *pbgroup.GetGroupInfoCacheReq) (*pbgroup.GetGroupInfoCacheResp, error) { | ||||
| 	group, err := s.db.TakeGroup(ctx, req.GroupID) | ||||
| func (g *groupServer) GetGroupInfoCache(ctx context.Context, req *pbgroup.GetGroupInfoCacheReq) (*pbgroup.GetGroupInfoCacheResp, error) { | ||||
| 	group, err := g.db.TakeGroup(ctx, req.GroupID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -32,8 +33,11 @@ func (s *groupServer) GetGroupInfoCache(ctx context.Context, req *pbgroup.GetGro | ||||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *groupServer) GetGroupMemberCache(ctx context.Context, req *pbgroup.GetGroupMemberCacheReq) (*pbgroup.GetGroupMemberCacheResp, error) { | ||||
| 	members, err := s.db.TakeGroupMember(ctx, req.GroupID, req.GroupMemberID) | ||||
| func (g *groupServer) GetGroupMemberCache(ctx context.Context, req *pbgroup.GetGroupMemberCacheReq) (*pbgroup.GetGroupMemberCacheResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.GroupMemberID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	members, err := g.db.TakeGroupMember(ctx, req.GroupID, req.GroupMemberID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| @ -476,6 +476,19 @@ func (g *groupServer) GetGroupAllMember(ctx context.Context, req *pbgroup.GetGro | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if !authverify.IsAdmin(ctx) { | ||||
| 		var inGroup bool | ||||
| 		opUserID := mcontext.GetOpUserID(ctx) | ||||
| 		for _, member := range members { | ||||
| 			if member.UserID == opUserID { | ||||
| 				inGroup = true | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 		if !inGroup { | ||||
| 			return nil, errs.ErrNoPermission.WrapMsg("opuser not in group") | ||||
| 		} | ||||
| 	} | ||||
| 	if err := g.PopulateGroupMember(ctx, members...); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -486,11 +499,24 @@ func (g *groupServer) GetGroupAllMember(ctx context.Context, req *pbgroup.GetGro | ||||
| 	return &resp, nil | ||||
| } | ||||
| 
 | ||||
| func (g *groupServer) checkAdminOrInGroup(ctx context.Context, groupID string) error { | ||||
| 	if authverify.IsAdmin(ctx) { | ||||
| 		return nil | ||||
| 	} | ||||
| 	opUserID := mcontext.GetOpUserID(ctx) | ||||
| 	members, err := g.db.FindGroupMembers(ctx, groupID, []string{opUserID}) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if len(members) == 0 { | ||||
| 		return errs.ErrNoPermission.WrapMsg("op user not in group") | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (g *groupServer) GetGroupMemberList(ctx context.Context, req *pbgroup.GetGroupMemberListReq) (*pbgroup.GetGroupMemberListResp, error) { | ||||
| 	if opUserID := mcontext.GetOpUserID(ctx); !datautil.Contain(opUserID, g.config.Share.IMAdminUserID...) { | ||||
| 		if _, err := g.db.TakeGroupMember(ctx, req.GroupID, opUserID); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var ( | ||||
| 		total   int64 | ||||
| @ -631,6 +657,9 @@ func (g *groupServer) GetGroupMembersInfo(ctx context.Context, req *pbgroup.GetG | ||||
| 	if req.GroupID == "" { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("groupID empty") | ||||
| 	} | ||||
| 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	members, err := g.getGroupMembersInfo(ctx, req.GroupID, req.UserIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -658,6 +687,9 @@ func (g *groupServer) getGroupMembersInfo(ctx context.Context, groupID string, u | ||||
| 
 | ||||
| // GetGroupApplicationList handles functions that get a list of group requests. | ||||
| func (g *groupServer) GetGroupApplicationList(ctx context.Context, req *pbgroup.GetGroupApplicationListReq) (*pbgroup.GetGroupApplicationListResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.FromUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	groupIDs, err := g.db.FindUserManagedGroupID(ctx, req.FromUserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -1652,6 +1684,11 @@ func (g *groupServer) GetGroupAbstractInfo(ctx context.Context, req *pbgroup.Get | ||||
| 	if datautil.Duplicate(req.GroupIDs) { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("groupIDs duplicate") | ||||
| 	} | ||||
| 	for _, groupID := range req.GroupIDs { | ||||
| 		if err := g.checkAdminOrInGroup(ctx, groupID); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	groups, err := g.db.FindGroup(ctx, req.GroupIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -1699,6 +1736,9 @@ func (g *groupServer) GetGroupMemberUserIDs(ctx context.Context, req *pbgroup.Ge | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if err := authverify.CheckAccessIn(ctx, userIDs...); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbgroup.GetGroupMemberUserIDsResp{ | ||||
| 		UserIDs: userIDs, | ||||
| 	}, nil | ||||
|  | ||||
| @ -18,24 +18,28 @@ import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/protocol/group" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| ) | ||||
| 
 | ||||
| func (s *groupServer) GroupCreateCount(ctx context.Context, req *group.GroupCreateCountReq) (*group.GroupCreateCountResp, error) { | ||||
| func (g *groupServer) GroupCreateCount(ctx context.Context, req *group.GroupCreateCountReq) (*group.GroupCreateCountResp, error) { | ||||
| 	if req.Start > req.End { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("start > end: %d > %d", req.Start, req.End) | ||||
| 	} | ||||
| 	total, err := s.db.CountTotal(ctx, nil) | ||||
| 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	total, err := g.db.CountTotal(ctx, nil) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	start := time.UnixMilli(req.Start) | ||||
| 	before, err := s.db.CountTotal(ctx, &start) | ||||
| 	before, err := g.db.CountTotal(ctx, &start) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	count, err := s.db.CountRangeEverydayTotal(ctx, start, time.UnixMilli(req.End)) | ||||
| 	count, err := g.db.CountRangeEverydayTotal(ctx, start, time.UnixMilli(req.End)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| @ -11,9 +11,6 @@ import ( | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	pbgroup "github.com/openimsdk/protocol/group" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| ) | ||||
| 
 | ||||
| const versionSyncLimit = 500 | ||||
| @ -23,10 +20,8 @@ func (g *groupServer) GetFullGroupMemberUserIDs(ctx context.Context, req *pbgrou | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if opUserID := mcontext.GetOpUserID(ctx); !datautil.Contain(opUserID, g.config.Share.IMAdminUserID...) { | ||||
| 		if !datautil.Contain(opUserID, userIDs...) { | ||||
| 			return nil, errs.ErrNoPermission.WrapMsg("user not in group") | ||||
| 		} | ||||
| 	if err := authverify.CheckAccessIn(ctx, userIDs...); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	vl, err := g.db.FindMaxGroupMemberVersionCache(ctx, req.GroupID) | ||||
| 	if err != nil { | ||||
| @ -69,6 +64,9 @@ func (g *groupServer) GetFullJoinGroupIDs(ctx context.Context, req *pbgroup.GetF | ||||
| } | ||||
| 
 | ||||
| func (g *groupServer) GetIncrementalGroupMember(ctx context.Context, req *pbgroup.GetIncrementalGroupMemberReq) (*pbgroup.GetIncrementalGroupMemberResp, error) { | ||||
| 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	group, err := g.db.TakeGroup(ctx, req.GroupID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -76,9 +74,6 @@ func (g *groupServer) GetIncrementalGroupMember(ctx context.Context, req *pbgrou | ||||
| 	if group.Status == constant.GroupStatusDismissed { | ||||
| 		return nil, servererrs.ErrDismissedAlready.Wrap() | ||||
| 	} | ||||
| 	if _, err := g.db.TakeGroupMember(ctx, req.GroupID, mcontext.GetOpUserID(ctx)); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var ( | ||||
| 		hasGroupUpdate bool | ||||
| 		sortVersion    uint64 | ||||
|  | ||||
| @ -18,6 +18,7 @@ import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	cbapi "github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| @ -29,6 +30,9 @@ import ( | ||||
| ) | ||||
| 
 | ||||
| func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *msg.GetConversationsHasReadAndMaxSeqReq) (*msg.GetConversationsHasReadAndMaxSeqResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var conversationIDs []string | ||||
| 	if len(req.ConversationIDs) == 0 { | ||||
| 		var err error | ||||
| @ -82,6 +86,9 @@ func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *m | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) SetConversationHasReadSeq(ctx context.Context, req *msg.SetConversationHasReadSeqReq) (*msg.SetConversationHasReadSeqResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	maxSeq, err := m.MsgDatabase.GetMaxSeq(ctx, req.ConversationID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -97,8 +104,8 @@ func (m *msgServer) SetConversationHasReadSeq(ctx context.Context, req *msg.SetC | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) MarkMsgsAsRead(ctx context.Context, req *msg.MarkMsgsAsReadReq) (*msg.MarkMsgsAsReadResp, error) { | ||||
| 	if len(req.Seqs) < 1 { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("seqs must not be empty") | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	maxSeq, err := m.MsgDatabase.GetMaxSeq(ctx, req.ConversationID) | ||||
| 	if err != nil { | ||||
| @ -139,6 +146,9 @@ func (m *msgServer) MarkMsgsAsRead(ctx context.Context, req *msg.MarkMsgsAsReadR | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) MarkConversationAsRead(ctx context.Context, req *msg.MarkConversationAsReadReq) (*msg.MarkConversationAsReadResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversation, err := m.ConversationLocalCache.GetConversation(ctx, req.UserID, req.ConversationID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -216,5 +226,4 @@ func (m *msgServer) sendMarkAsReadNotification(ctx context.Context, conversation | ||||
| 		HasReadSeq:       hasReadSeq, | ||||
| 	} | ||||
| 	m.notificationSender.NotificationWithSessionType(ctx, sendID, recvID, constant.HasReadReceipt, sessionType, tips) | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -94,6 +94,9 @@ func (m *msgServer) DeleteMsgs(ctx context.Context, req *msg.DeleteMsgsReq) (*ms | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) DeleteMsgPhysicalBySeq(ctx context.Context, req *msg.DeleteMsgPhysicalBySeqReq) (*msg.DeleteMsgPhysicalBySeqResp, error) { | ||||
| 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	err := m.MsgDatabase.DeleteMsgsPhysicalBySeqs(ctx, req.ConversationID, req.Seqs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | ||||
| @ -17,11 +17,14 @@ package msg | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/prommetrics" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	pbconversation "github.com/openimsdk/protocol/conversation" | ||||
| 	pbconv "github.com/openimsdk/protocol/conversation" | ||||
| 	pbmsg "github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/protocol/wrapperspb" | ||||
| @ -29,13 +32,15 @@ import ( | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| ) | ||||
| 
 | ||||
| func (m *msgServer) SendMsg(ctx context.Context, req *pbmsg.SendMsgReq) (*pbmsg.SendMsgResp, error) { | ||||
| 	if req.MsgData == nil { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("msgData is nil") | ||||
| 	} | ||||
| 	if err := authverify.CheckAccess(ctx, req.MsgData.SendID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	before := new(*sdkws.MsgData) | ||||
| 	resp, err := m.sendMsg(ctx, req, before) | ||||
| 	if err != nil { | ||||
| @ -104,7 +109,7 @@ func (m *msgServer) setConversationAtInfo(nctx context.Context, msg *sdkws.MsgDa | ||||
| 
 | ||||
| 	var atUserID []string | ||||
| 
 | ||||
| 	conversation := &pbconversation.ConversationReq{ | ||||
| 	conversation := &pbconv.ConversationReq{ | ||||
| 		ConversationID:   msgprocessor.GetConversationIDByMsg(msg), | ||||
| 		ConversationType: msg.SessionType, | ||||
| 		GroupID:          msg.GroupID, | ||||
| @ -171,13 +176,7 @@ func (m *msgServer) sendMsgSingleChat(ctx context.Context, req *pbmsg.SendMsgReq | ||||
| 	isSend := true | ||||
| 	isNotification := msgprocessor.IsNotificationByMsg(req.MsgData) | ||||
| 	if !isNotification { | ||||
| 		isSend, err = m.modifyMessageByUserMessageReceiveOpt( | ||||
| 			ctx, | ||||
| 			req.MsgData.RecvID, | ||||
| 			conversationutil.GenConversationIDForSingle(req.MsgData.SendID, req.MsgData.RecvID), | ||||
| 			constant.SingleChatType, | ||||
| 			req, | ||||
| 		) | ||||
| 		isSend, err = m.modifyMessageByUserMessageReceiveOpt(authverify.WithTempAdmin(ctx), req.MsgData.RecvID, conversationutil.GenConversationIDForSingle(req.MsgData.SendID, req.MsgData.RecvID), constant.SingleChatType, req) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| @ -17,9 +17,10 @@ package msg | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"sort" | ||||
| 
 | ||||
| 	pbmsg "github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"sort" | ||||
| ) | ||||
| 
 | ||||
| func (m *msgServer) GetConversationMaxSeq(ctx context.Context, req *pbmsg.GetConversationMaxSeqReq) (*pbmsg.GetConversationMaxSeqResp, error) { | ||||
|  | ||||
| @ -16,15 +16,20 @@ package msg | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 
 | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| ) | ||||
| 
 | ||||
| func (m *msgServer) GetActiveUser(ctx context.Context, req *msg.GetActiveUserReq) (*msg.GetActiveUserResp, error) { | ||||
| 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	msgCount, userCount, users, dateCount, err := m.MsgDatabase.RangeUserSendCount(ctx, time.UnixMilli(req.Start), time.UnixMilli(req.End), req.Group, req.Ase, req.Pagination.PageNumber, req.Pagination.ShowNumber) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -60,6 +65,9 @@ func (m *msgServer) GetActiveUser(ctx context.Context, req *msg.GetActiveUserReq | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) GetActiveGroup(ctx context.Context, req *msg.GetActiveGroupReq) (*msg.GetActiveGroupResp, error) { | ||||
| 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	msgCount, groupCount, groups, dateCount, err := m.MsgDatabase.RangeGroupSendCount(ctx, time.UnixMilli(req.Start), time.UnixMilli(req.End), req.Ase, req.Pagination.PageNumber, req.Pagination.ShowNumber) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | ||||
| @ -29,6 +29,9 @@ import ( | ||||
| ) | ||||
| 
 | ||||
| func (m *msgServer) PullMessageBySeqs(ctx context.Context, req *sdkws.PullMessageBySeqsReq) (*sdkws.PullMessageBySeqsResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	resp := &sdkws.PullMessageBySeqsResp{} | ||||
| 	resp.Msgs = make(map[string]*sdkws.PullMsgs) | ||||
| 	resp.NotificationMsgs = make(map[string]*sdkws.PullMsgs) | ||||
|  | ||||
| @ -47,6 +47,9 @@ func (s *friendServer) GetPaginationBlacks(ctx context.Context, req *relation.Ge | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) IsBlack(ctx context.Context, req *relation.IsBlackReq) (*relation.IsBlackResp, error) { | ||||
| 	if err := authverify.CheckAccessIn(ctx, req.UserID1, req.UserID2); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	in1, in2, err := s.blackDatabase.CheckIn(ctx, req.UserID1, req.UserID2) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | ||||
| @ -280,6 +280,9 @@ func (s *friendServer) SetFriendRemark(ctx context.Context, req *relation.SetFri | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetFriendInfo(ctx context.Context, req *relation.GetFriendInfoReq) (*relation.GetFriendInfoResp, error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	friends, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.FriendUserIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -288,6 +291,9 @@ func (s *friendServer) GetFriendInfo(ctx context.Context, req *relation.GetFrien | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetDesignatedFriends(ctx context.Context, req *relation.GetDesignatedFriendsReq) (resp *relation.GetDesignatedFriendsResp, err error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	resp = &relation.GetDesignatedFriendsResp{} | ||||
| 	if datautil.Duplicate(req.FriendUserIDs) { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("friend userID repeated") | ||||
| @ -313,9 +319,10 @@ func (s *friendServer) getFriend(ctx context.Context, ownerUserID string, friend | ||||
| } | ||||
| 
 | ||||
| // Get the list of friend requests sent out proactively. | ||||
| func (s *friendServer) GetDesignatedFriendsApply(ctx context.Context, | ||||
| 	req *relation.GetDesignatedFriendsApplyReq, | ||||
| ) (resp *relation.GetDesignatedFriendsApplyResp, err error) { | ||||
| func (s *friendServer) GetDesignatedFriendsApply(ctx context.Context, req *relation.GetDesignatedFriendsApplyReq) (resp *relation.GetDesignatedFriendsApplyResp, err error) { | ||||
| 	if err := authverify.CheckAccessIn(ctx, req.FromUserID, req.ToUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	friendRequests, err := s.db.FindBothFriendRequests(ctx, req.FromUserID, req.ToUserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -374,6 +381,9 @@ func (s *friendServer) GetPaginationFriendsApplyFrom(ctx context.Context, req *r | ||||
| 
 | ||||
| // ok. | ||||
| func (s *friendServer) IsFriend(ctx context.Context, req *relation.IsFriendReq) (resp *relation.IsFriendResp, err error) { | ||||
| 	if err := authverify.CheckAccessIn(ctx, req.UserID1, req.UserID2); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	resp = &relation.IsFriendResp{} | ||||
| 	resp.InUser1Friends, resp.InUser2Friends, err = s.db.CheckIn(ctx, req.UserID1, req.UserID2) | ||||
| 	if err != nil { | ||||
| @ -426,6 +436,9 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | ||||
| 	} | ||||
| 
 | ||||
| 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	userMap, err := s.userClient.GetUsersInfoMap(ctx, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -494,10 +507,7 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) UpdateFriends( | ||||
| 	ctx context.Context, | ||||
| 	req *relation.UpdateFriendsReq, | ||||
| ) (*relation.UpdateFriendsResp, error) { | ||||
| func (s *friendServer) UpdateFriends(ctx context.Context, req *relation.UpdateFriendsReq) (*relation.UpdateFriendsResp, error) { | ||||
| 	if len(req.FriendUserIDs) == 0 { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("friendIDList is empty") | ||||
| 	} | ||||
| @ -505,6 +515,10 @@ func (s *friendServer) UpdateFriends( | ||||
| 		return nil, errs.ErrArgs.WrapMsg("friendIDList repeated") | ||||
| 	} | ||||
| 
 | ||||
| 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	_, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.FriendUserIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | ||||
| @ -25,6 +25,7 @@ import ( | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"github.com/openimsdk/protocol/third" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| ) | ||||
| 
 | ||||
| @ -45,7 +46,7 @@ func genLogID() string { | ||||
| 
 | ||||
| func (t *thirdServer) UploadLogs(ctx context.Context, req *third.UploadLogsReq) (*third.UploadLogsResp, error) { | ||||
| 	var dbLogs []*relationtb.Log | ||||
| 	userID := ctx.Value(constant.OpUserID).(string) | ||||
| 	userID := mcontext.GetOpUserID(ctx) | ||||
| 	platform := constant.PlatformID2Name[int(req.Platform)] | ||||
| 	for _, fileURL := range req.FileURLs { | ||||
| 		log := relationtb.Log{ | ||||
|  | ||||
| @ -19,6 +19,7 @@ import ( | ||||
| 	"fmt" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/mcache" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | ||||
| @ -148,6 +149,9 @@ func (t *thirdServer) FcmUpdateToken(ctx context.Context, req *third.FcmUpdateTo | ||||
| } | ||||
| 
 | ||||
| func (t *thirdServer) SetAppBadge(ctx context.Context, req *third.SetAppBadgeReq) (resp *third.SetAppBadgeResp, err error) { | ||||
| 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	err = t.thirdDatabase.SetAppBadge(ctx, req.UserID, int(req.AppUnreadCount)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | ||||
| @ -64,16 +64,57 @@ func GetIMAdminUserIDs(ctx context.Context) []string { | ||||
| } | ||||
| 
 | ||||
| func IsAdmin(ctx context.Context) bool { | ||||
| 	return datautil.Contain(mcontext.GetOpUserID(ctx), GetIMAdminUserIDs(ctx)...) | ||||
| 	return IsTempAdmin(ctx) || IsSystemAdmin(ctx) | ||||
| } | ||||
| 
 | ||||
| func CheckAccess(ctx context.Context, ownerUserID string) error { | ||||
| 	opUserID := mcontext.GetOpUserID(ctx) | ||||
| 	if opUserID == ownerUserID { | ||||
| 	if mcontext.GetOpUserID(ctx) == ownerUserID { | ||||
| 		return nil | ||||
| 	} | ||||
| 	if datautil.Contain(mcontext.GetOpUserID(ctx), GetIMAdminUserIDs(ctx)...) { | ||||
| 	if IsAdmin(ctx) { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return servererrs.ErrNoPermission.WrapMsg("ownerUserID", ownerUserID) | ||||
| } | ||||
| 
 | ||||
| func CheckAccessIn(ctx context.Context, ownerUserIDs ...string) error { | ||||
| 	opUserID := mcontext.GetOpUserID(ctx) | ||||
| 	for _, userID := range ownerUserIDs { | ||||
| 		if opUserID == userID { | ||||
| 			return nil | ||||
| 		} | ||||
| 	} | ||||
| 	if IsAdmin(ctx) { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return servererrs.ErrNoPermission.WrapMsg("opUser in ownerUserIDs") | ||||
| } | ||||
| 
 | ||||
| var tempAdminValue = []string{"1"} | ||||
| 
 | ||||
| const ctxTempAdminKey = "ctxImTempAdminKey" | ||||
| 
 | ||||
| func WithTempAdmin(ctx context.Context) context.Context { | ||||
| 	keys, _ := ctx.Value(constant.RpcCustomHeader).([]string) | ||||
| 	if datautil.Contain(ctxTempAdminKey, keys...) { | ||||
| 		return ctx | ||||
| 	} | ||||
| 	if len(keys) > 0 { | ||||
| 		temp := make([]string, 0, len(keys)+1) | ||||
| 		temp = append(temp, keys...) | ||||
| 		keys = append(temp, ctxTempAdminKey) | ||||
| 	} else { | ||||
| 		keys = []string{ctxTempAdminKey} | ||||
| 	} | ||||
| 	ctx = context.WithValue(ctx, constant.RpcCustomHeader, keys) | ||||
| 	return context.WithValue(ctx, ctxTempAdminKey, tempAdminValue) | ||||
| } | ||||
| 
 | ||||
| func IsTempAdmin(ctx context.Context) bool { | ||||
| 	values, _ := ctx.Value(ctxTempAdminKey).([]string) | ||||
| 	return datautil.Equal(tempAdminValue, values) | ||||
| } | ||||
| 
 | ||||
| func IsSystemAdmin(ctx context.Context) bool { | ||||
| 	return datautil.Contain(mcontext.GetOpUserID(ctx), GetIMAdminUserIDs(ctx)...) | ||||
| } | ||||
|  | ||||
| @ -7,6 +7,7 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/utils/idutil" | ||||
| ) | ||||
| @ -253,13 +254,14 @@ func (b *Batcher[T]) distributeMessage(messages map[string][]*T, totalCount int, | ||||
| 
 | ||||
| func (b *Batcher[T]) run(channelID int, ch <-chan *Msg[T]) { | ||||
| 	defer b.wait.Done() | ||||
| 	ctx := authverify.WithTempAdmin(context.Background()) | ||||
| 	for { | ||||
| 		select { | ||||
| 		case messages, ok := <-ch: | ||||
| 			if !ok { | ||||
| 				return | ||||
| 			} | ||||
| 			b.Do(context.Background(), channelID, messages) | ||||
| 			b.Do(ctx, channelID, messages) | ||||
| 			if b.config.syncWait { | ||||
| 				b.counter.Done() | ||||
| 			} | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user