mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-25 20:52:11 +08:00 
			
		
		
		
	Merge pull request #3372 from withchao/pre-release-v3.8.4
feat: add rpc interface permission check
This commit is contained in:
		
						commit
						2ec3708a27
					
				| @ -181,3 +181,19 @@ afterImportFriends: | |||||||
| afterRemoveBlack: | afterRemoveBlack: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|  | beforeCreateSingleChatConversations: | ||||||
|  |   enable: false | ||||||
|  |   timeout: 5 | ||||||
|  |   failedContinue: false | ||||||
|  | afterCreateSingleChatConversations: | ||||||
|  |   enable: false | ||||||
|  |   timeout: 5 | ||||||
|  |   failedContinue: false | ||||||
|  | beforeCreateGroupChatConversations: | ||||||
|  |   enable: false | ||||||
|  |   timeout: 5 | ||||||
|  |   failedContinue: false | ||||||
|  | afterCreateGroupChatConversations: | ||||||
|  |   enable: false | ||||||
|  |   timeout: 5 | ||||||
|  |   failedContinue: false | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ package api | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/protocol/conversation" | 	"github.com/openimsdk/protocol/conversation" | ||||||
| 	"github.com/openimsdk/tools/a2r" | 	"github.com/openimsdk/tools/a2r" | ||||||
| ) | ) | ||||||
| @ -48,9 +49,9 @@ func (o *ConversationApi) SetConversations(c *gin.Context) { | |||||||
| 	a2r.Call(c, conversation.ConversationClient.SetConversations, o.Client) | 	a2r.Call(c, conversation.ConversationClient.SetConversations, o.Client) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (o *ConversationApi) GetConversationOfflinePushUserIDs(c *gin.Context) { | //func (o *ConversationApi) GetConversationOfflinePushUserIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, conversation.ConversationClient.GetConversationOfflinePushUserIDs, o.Client) | //	a2r.Call(c, conversation.ConversationClient.GetConversationOfflinePushUserIDs, o.Client) | ||||||
| } | //} | ||||||
| 
 | 
 | ||||||
| func (o *ConversationApi) GetFullOwnerConversationIDs(c *gin.Context) { | func (o *ConversationApi) GetFullOwnerConversationIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, conversation.ConversationClient.GetFullOwnerConversationIDs, o.Client) | 	a2r.Call(c, conversation.ConversationClient.GetFullOwnerConversationIDs, o.Client) | ||||||
| @ -71,3 +72,7 @@ func (o *ConversationApi) GetNotNotifyConversationIDs(c *gin.Context) { | |||||||
| func (o *ConversationApi) GetPinnedConversationIDs(c *gin.Context) { | func (o *ConversationApi) GetPinnedConversationIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, conversation.ConversationClient.GetPinnedConversationIDs, o.Client) | 	a2r.Call(c, conversation.ConversationClient.GetPinnedConversationIDs, o.Client) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (o *ConversationApi) UpdateConversationsByUser(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, conversation.ConversationClient.UpdateConversationsByUser, o.Client) | ||||||
|  | } | ||||||
|  | |||||||
| @ -9,6 +9,8 @@ import ( | |||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/gin-gonic/gin/binding" | 	"github.com/gin-gonic/gin/binding" | ||||||
| 	"github.com/go-playground/validator/v10" | 	"github.com/go-playground/validator/v10" | ||||||
|  | 	clientv3 "go.etcd.io/etcd/client/v3" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/internal/api/jssdk" | 	"github.com/openimsdk/open-im-server/v3/internal/api/jssdk" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||||
| @ -28,7 +30,6 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/discovery/etcd" | 	"github.com/openimsdk/tools/discovery/etcd" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/openimsdk/tools/mw" | 	"github.com/openimsdk/tools/mw" | ||||||
| 	clientv3 "go.etcd.io/etcd/client/v3" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| @ -262,12 +263,13 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 		conversationGroup.POST("/get_conversation", c.GetConversation) | 		conversationGroup.POST("/get_conversation", c.GetConversation) | ||||||
| 		conversationGroup.POST("/get_conversations", c.GetConversations) | 		conversationGroup.POST("/get_conversations", c.GetConversations) | ||||||
| 		conversationGroup.POST("/set_conversations", c.SetConversations) | 		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_full_conversation_ids", c.GetFullOwnerConversationIDs) | ||||||
| 		conversationGroup.POST("/get_incremental_conversations", c.GetIncrementalConversation) | 		conversationGroup.POST("/get_incremental_conversations", c.GetIncrementalConversation) | ||||||
| 		conversationGroup.POST("/get_owner_conversation", c.GetOwnerConversation) | 		conversationGroup.POST("/get_owner_conversation", c.GetOwnerConversation) | ||||||
| 		conversationGroup.POST("/get_not_notify_conversation_ids", c.GetNotNotifyConversationIDs) | 		conversationGroup.POST("/get_not_notify_conversation_ids", c.GetNotNotifyConversationIDs) | ||||||
| 		conversationGroup.POST("/get_pinned_conversation_ids", c.GetPinnedConversationIDs) | 		conversationGroup.POST("/get_pinned_conversation_ids", c.GetPinnedConversationIDs) | ||||||
|  | 		conversationGroup.POST("/update_conversations_by_user", c.UpdateConversationsByUser) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	{ | 	{ | ||||||
|  | |||||||
							
								
								
									
										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" | 	"sort" | ||||||
| 	"time" | 	"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/dbbuild" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"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/config" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | 	"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/database/mgo" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
| 	dbModel "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/localcache" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| @ -39,7 +43,6 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"google.golang.org/grpc" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| type conversationServer struct { | type conversationServer struct { | ||||||
| @ -49,6 +52,7 @@ type conversationServer struct { | |||||||
| 	conversationNotificationSender *ConversationNotificationSender | 	conversationNotificationSender *ConversationNotificationSender | ||||||
| 	config                         *Config | 	config                         *Config | ||||||
| 
 | 
 | ||||||
|  | 	webhookClient *webhook.Client | ||||||
| 	userClient    *rpcli.UserClient | 	userClient    *rpcli.UserClient | ||||||
| 	msgClient     *rpcli.MsgClient | 	msgClient     *rpcli.MsgClient | ||||||
| 	groupClient   *rpcli.GroupClient | 	groupClient   *rpcli.GroupClient | ||||||
| @ -60,6 +64,7 @@ type Config struct { | |||||||
| 	MongodbConfig      config.Mongo | 	MongodbConfig      config.Mongo | ||||||
| 	NotificationConfig config.Notification | 	NotificationConfig config.Notification | ||||||
| 	Share              config.Share | 	Share              config.Share | ||||||
|  | 	WebhooksConfig     config.Webhooks | ||||||
| 	LocalCacheConfig   config.LocalCache | 	LocalCacheConfig   config.LocalCache | ||||||
| 	Discovery          config.Discovery | 	Discovery          config.Discovery | ||||||
| } | } | ||||||
| @ -90,20 +95,32 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	msgClient := rpcli.NewMsgClient(msgConn) | 	msgClient := rpcli.NewMsgClient(msgConn) | ||||||
| 	localcache.InitLocalCache(&config.LocalCacheConfig) | 
 | ||||||
| 	pbconversation.RegisterConversationServer(server, &conversationServer{ | 	cs := conversationServer{ | ||||||
| 		conversationNotificationSender: NewConversationNotificationSender(&config.NotificationConfig, msgClient), | 		config:        config, | ||||||
| 		conversationDatabase: controller.NewConversationDatabase(conversationDB, | 		webhookClient: webhook.NewWebhookClient(config.WebhooksConfig.URL), | ||||||
| 			redis.NewConversationRedis(rdb, &config.LocalCacheConfig, conversationDB), mgocli.GetTx()), |  | ||||||
| 		userClient:    rpcli.NewUserClient(userConn), | 		userClient:    rpcli.NewUserClient(userConn), | ||||||
| 		groupClient:   rpcli.NewGroupClient(groupConn), | 		groupClient:   rpcli.NewGroupClient(groupConn), | ||||||
| 		msgClient:     msgClient, | 		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, &cs) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetConversation(ctx context.Context, req *pbconversation.GetConversationReq) (*pbconversation.GetConversationResp, error) { | 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}) | 	conversations, err := c.conversationDatabase.FindConversations(ctx, req.OwnerUserID, []string{req.ConversationID}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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 | 	var conversationIDs []string | ||||||
| 	if len(req.ConversationIDs) == 0 { | 	if len(req.ConversationIDs) == 0 { | ||||||
| 		conversationIDs, err = c.conversationDatabase.GetConversationIDs(ctx, req.UserID) | 		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) { | 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) | 	conversations, err := c.conversationDatabase.GetUserAllConversation(ctx, req.OwnerUserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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) | 	conversations, err := c.getConversations(ctx, req.OwnerUserID, req.ConversationIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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 | 	var conversation dbModel.Conversation | ||||||
| 	if err := datautil.CopyStructFields(&conversation, req.Conversation); err != nil { | 	if err := datautil.CopyStructFields(&conversation, req.Conversation); err != nil { | ||||||
| 		return nil, err | 		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) { | func (c *conversationServer) SetConversations(ctx context.Context, req *pbconversation.SetConversationsReq) (*pbconversation.SetConversationsResp, error) { | ||||||
| 	if req.Conversation == nil { | 	for _, userID := range req.UserIDs { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("conversation must not be nil") | 		if err := authverify.CheckAccess(ctx, userID); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	if req.Conversation.ConversationType == constant.WriteGroupChatType { | 	if req.Conversation.ConversationType == constant.WriteGroupChatType { | ||||||
| 		groupInfo, err := c.groupClient.GetGroupInfo(ctx, req.Conversation.GroupID) | 		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.UserID = req.Conversation.UserID | ||||||
| 	conversation.GroupID = req.Conversation.GroupID | 	conversation.GroupID = req.Conversation.GroupID | ||||||
| 
 | 
 | ||||||
| 	m := make(map[string]any) | 	m, conversation, err := UpdateConversationsMap(ctx, req) | ||||||
| 
 | 	if err != nil { | ||||||
| 	setConversationFieldsFunc := func() { | 		return nil, err | ||||||
| 		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 |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	// set need set field in conversation |  | ||||||
| 	setConversationFieldsFunc() |  | ||||||
| 
 | 
 | ||||||
| 	for userID := range conversationMap { | 	for userID := range conversationMap { | ||||||
| 		unequal := len(m) | 		unequal := UserUpdateCheckMap(ctx, userID, req.Conversation, conversationMap[userID]) | ||||||
| 
 | 
 | ||||||
| 		if req.Conversation.RecvMsgOpt != nil { | 		if unequal { | ||||||
| 			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 { |  | ||||||
| 			needUpdateUsersList = append(needUpdateUsersList, userID) | 			needUpdateUsersList = append(needUpdateUsersList, userID) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if len(m) != 0 && len(needUpdateUsersList) != 0 { | 	if len(m) != 0 && len(needUpdateUsersList) != 0 { | ||||||
| 		if err := c.conversationDatabase.SetUsersConversationFieldTx(ctx, needUpdateUsersList, &conversation, m); err != nil { | 		if err := c.conversationDatabase.SetUsersConversationFieldTx(ctx, needUpdateUsersList, &conversation, m); err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		for _, v := range needUpdateUsersList { | 		for _, userID := range needUpdateUsersList { | ||||||
| 			c.conversationNotificationSender.ConversationChangeNotification(ctx, v, []string{req.Conversation.ConversationID}) | 			c.conversationNotificationSender.ConversationChangeNotification(ctx, userID, []string{req.Conversation.ConversationID}) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if req.Conversation.IsPrivateChat != nil && req.Conversation.ConversationType != constant.ReadGroupChatType { | 	if req.Conversation.IsPrivateChat != nil && req.Conversation.ConversationType != constant.ReadGroupChatType { | ||||||
| 		var conversations []*dbModel.Conversation | 		var conversations []*dbModel.Conversation | ||||||
| 		for _, ownerUserID := range req.UserIDs { | 		for _, ownerUserID := range req.UserIDs { | ||||||
| @ -396,58 +346,94 @@ func (c *conversationServer) SetConversations(ctx context.Context, req *pbconver | |||||||
| 	return &pbconversation.SetConversationsResp{}, nil | 	return &pbconversation.SetConversationsResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Get user IDs with "Do Not Disturb" enabled in super large groups. | func (c *conversationServer) UpdateConversationsByUser(ctx context.Context, req *pbconversation.UpdateConversationsByUserReq) (*pbconversation.UpdateConversationsByUserResp, error) { | ||||||
| func (c *conversationServer) GetRecvMsgNotNotifyUserIDs(ctx context.Context, req *pbconversation.GetRecvMsgNotNotifyUserIDsReq) (*pbconversation.GetRecvMsgNotNotifyUserIDsResp, error) { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 	return nil, errs.New("deprecated") | 		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. | // create conversation without notification for msg redis transfer. | ||||||
| func (c *conversationServer) CreateSingleChatConversations(ctx context.Context, | func (c *conversationServer) CreateSingleChatConversations(ctx context.Context, req *pbconversation.CreateSingleChatConversationsReq) (*pbconversation.CreateSingleChatConversationsResp, error) { | ||||||
| 	req *pbconversation.CreateSingleChatConversationsReq, | 	var conversation dbModel.Conversation | ||||||
| ) (*pbconversation.CreateSingleChatConversationsResp, error) { |  | ||||||
| 	switch req.ConversationType { | 	switch req.ConversationType { | ||||||
| 	case constant.SingleChatType: | 	case constant.SingleChatType: | ||||||
| 		var conversation dbModel.Conversation | 		// sendUser create | ||||||
| 		conversation.ConversationID = req.ConversationID | 		conversation.ConversationID = req.ConversationID | ||||||
| 		conversation.ConversationType = req.ConversationType | 		conversation.ConversationType = req.ConversationType | ||||||
| 		conversation.OwnerUserID = req.SendID | 		conversation.OwnerUserID = req.SendID | ||||||
| 		conversation.UserID = req.RecvID | 		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}) | 		err := c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZWarn(ctx, "create conversation failed", err, "conversation", conversation) | 			log.ZWarn(ctx, "create conversation failed", err, "conversation", conversation) | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | 		c.webhookAfterCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateSingleChatConversations, &conversation) | ||||||
|  | 
 | ||||||
|  | 		// recvUser create | ||||||
| 		conversation2 := conversation | 		conversation2 := conversation | ||||||
| 		conversation2.OwnerUserID = req.RecvID | 		conversation2.OwnerUserID = req.RecvID | ||||||
| 		conversation2.UserID = req.SendID | 		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}) | 		err = c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation2}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
|  | 		c.webhookAfterCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateSingleChatConversations, &conversation2) | ||||||
| 	case constant.NotificationChatType: | 	case constant.NotificationChatType: | ||||||
| 		var conversation dbModel.Conversation |  | ||||||
| 		conversation.ConversationID = req.ConversationID | 		conversation.ConversationID = req.ConversationID | ||||||
| 		conversation.ConversationType = req.ConversationType | 		conversation.ConversationType = req.ConversationType | ||||||
| 		conversation.OwnerUserID = req.RecvID | 		conversation.OwnerUserID = req.RecvID | ||||||
| 		conversation.UserID = req.SendID | 		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}) | 		err := c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
|  | 		c.webhookAfterCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateSingleChatConversations, &conversation) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return &pbconversation.CreateSingleChatConversationsResp{}, nil | 	return &pbconversation.CreateSingleChatConversationsResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) CreateGroupChatConversations(ctx context.Context, req *pbconversation.CreateGroupChatConversationsReq) (*pbconversation.CreateGroupChatConversationsResp, error) { | 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 { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	conversationID := msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID) | 
 | ||||||
| 	if err := c.msgClient.SetUserConversationMaxSeq(ctx, conversationID, req.UserIDs, 0); err != nil { | 	c.webhookAfterCreateGroupChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateGroupChatConversations, &conversation) | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return &pbconversation.CreateGroupChatConversationsResp{}, nil | 	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) { | 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) | 	conversationIDs, err := c.conversationDatabase.GetConversationIDs(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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) | 	hash, err := c.conversationDatabase.GetUserConversationIDsHash(ctx, req.OwnerUserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -495,10 +487,7 @@ func (c *conversationServer) GetUserConversationIDsHash(ctx context.Context, req | |||||||
| 	return &pbconversation.GetUserConversationIDsHashResp{Hash: hash}, nil | 	return &pbconversation.GetUserConversationIDsHashResp{Hash: hash}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetConversationsByConversationID( | func (c *conversationServer) GetConversationsByConversationID(ctx context.Context, req *pbconversation.GetConversationsByConversationIDReq) (*pbconversation.GetConversationsByConversationIDResp, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	req *pbconversation.GetConversationsByConversationIDReq, |  | ||||||
| ) (*pbconversation.GetConversationsByConversationIDResp, error) { |  | ||||||
| 	conversations, err := c.conversationDatabase.GetConversationsByConversationID(ctx, req.ConversationIDs) | 	conversations, err := c.conversationDatabase.GetConversationsByConversationID(ctx, req.ConversationIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -552,10 +541,7 @@ func (c *conversationServer) conversationSort(conversations map[int64]string, re | |||||||
| 	resp.ConversationElems = append(resp.ConversationElems, cons...) | 	resp.ConversationElems = append(resp.ConversationElems, cons...) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) getConversationInfo( | func (c *conversationServer) getConversationInfo(ctx context.Context, chatLogs map[string]*sdkws.MsgData, userID string) (map[string]*pbconversation.ConversationElem, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	chatLogs map[string]*sdkws.MsgData, |  | ||||||
| 	userID string) (map[string]*pbconversation.ConversationElem, error) { |  | ||||||
| 	var ( | 	var ( | ||||||
| 		sendIDs         []string | 		sendIDs         []string | ||||||
| 		groupIDs        []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) { | 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) | 	m := make(map[string]any) | ||||||
| 	if req.RecvMsgOpt != nil { | 	if req.RecvMsgOpt != nil { | ||||||
| 		m["recv_msg_opt"] = req.RecvMsgOpt.Value | 		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) { | 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) | 	total, conversations, err := c.conversationDatabase.GetOwnerConversation(ctx, req.UserID, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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) | 	conversationIDs, err := c.conversationDatabase.GetNotNotifyConversationIDs(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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) | 	conversationIDs, err := c.conversationDatabase.GetPinnedConversationIDs(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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 ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/notification" |  | ||||||
| 	"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/open-im-server/v3/pkg/common/config" | 	"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/constant" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"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) { | 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]{ | 	opt := incrversion.Option[*conversation.Conversation, conversation.GetIncrementalConversationResp]{ | ||||||
| 		Ctx:             ctx, | 		Ctx:             ctx, | ||||||
| 		VersionKey:      req.UserID, | 		VersionKey:      req.UserID, | ||||||
|  | |||||||
| @ -17,13 +17,14 @@ package group | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | ||||||
| 	pbgroup "github.com/openimsdk/protocol/group" | 	pbgroup "github.com/openimsdk/protocol/group" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // GetGroupInfoCache get group info from cache. | // GetGroupInfoCache get group info from cache. | ||||||
| func (s *groupServer) GetGroupInfoCache(ctx context.Context, req *pbgroup.GetGroupInfoCacheReq) (*pbgroup.GetGroupInfoCacheResp, error) { | func (g *groupServer) GetGroupInfoCache(ctx context.Context, req *pbgroup.GetGroupInfoCacheReq) (*pbgroup.GetGroupInfoCacheResp, error) { | ||||||
| 	group, err := s.db.TakeGroup(ctx, req.GroupID) | 	group, err := g.db.TakeGroup(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -32,8 +33,11 @@ func (s *groupServer) GetGroupInfoCache(ctx context.Context, req *pbgroup.GetGro | |||||||
| 	}, nil | 	}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *groupServer) GetGroupMemberCache(ctx context.Context, req *pbgroup.GetGroupMemberCacheReq) (*pbgroup.GetGroupMemberCacheResp, error) { | func (g *groupServer) GetGroupMemberCache(ctx context.Context, req *pbgroup.GetGroupMemberCacheReq) (*pbgroup.GetGroupMemberCacheResp, error) { | ||||||
| 	members, err := s.db.TakeGroupMember(ctx, req.GroupID, req.GroupMemberID) | 	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 { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -476,6 +476,19 @@ func (g *groupServer) GetGroupAllMember(ctx context.Context, req *pbgroup.GetGro | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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 { | 	if err := g.PopulateGroupMember(ctx, members...); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -486,11 +499,24 @@ func (g *groupServer) GetGroupAllMember(ctx context.Context, req *pbgroup.GetGro | |||||||
| 	return &resp, nil | 	return &resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetGroupMemberList(ctx context.Context, req *pbgroup.GetGroupMemberListReq) (*pbgroup.GetGroupMemberListResp, error) { | func (g *groupServer) checkAdminOrInGroup(ctx context.Context, groupID string) error { | ||||||
| 	if opUserID := mcontext.GetOpUserID(ctx); !datautil.Contain(opUserID, g.config.Share.IMAdminUserID...) { | 	if authverify.IsAdmin(ctx) { | ||||||
| 		if _, err := g.db.TakeGroupMember(ctx, req.GroupID, opUserID); err != nil { | 		return nil | ||||||
| 			return nil, err |  | ||||||
| 	} | 	} | ||||||
|  | 	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 err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	var ( | 	var ( | ||||||
| 		total   int64 | 		total   int64 | ||||||
| @ -631,6 +657,9 @@ func (g *groupServer) GetGroupMembersInfo(ctx context.Context, req *pbgroup.GetG | |||||||
| 	if req.GroupID == "" { | 	if req.GroupID == "" { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("groupID empty") | 		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) | 	members, err := g.getGroupMembersInfo(ctx, req.GroupID, req.UserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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. | // GetGroupApplicationList handles functions that get a list of group requests. | ||||||
| func (g *groupServer) GetGroupApplicationList(ctx context.Context, req *pbgroup.GetGroupApplicationListReq) (*pbgroup.GetGroupApplicationListResp, error) { | 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) | 	groupIDs, err := g.db.FindUserManagedGroupID(ctx, req.FromUserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -1652,6 +1684,11 @@ func (g *groupServer) GetGroupAbstractInfo(ctx context.Context, req *pbgroup.Get | |||||||
| 	if datautil.Duplicate(req.GroupIDs) { | 	if datautil.Duplicate(req.GroupIDs) { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("groupIDs duplicate") | 		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) | 	groups, err := g.db.FindGroup(ctx, req.GroupIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -1699,6 +1736,9 @@ func (g *groupServer) GetGroupMemberUserIDs(ctx context.Context, req *pbgroup.Ge | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	if err := authverify.CheckAccessIn(ctx, userIDs...); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	return &pbgroup.GetGroupMemberUserIDsResp{ | 	return &pbgroup.GetGroupMemberUserIDsResp{ | ||||||
| 		UserIDs: userIDs, | 		UserIDs: userIDs, | ||||||
| 	}, nil | 	}, nil | ||||||
|  | |||||||
| @ -18,24 +18,28 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"github.com/openimsdk/protocol/group" | 	"github.com/openimsdk/protocol/group" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"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 { | 	if req.Start > req.End { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("start > end: %d > %d", 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 { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	start := time.UnixMilli(req.Start) | 	start := time.UnixMilli(req.Start) | ||||||
| 	before, err := s.db.CountTotal(ctx, &start) | 	before, err := g.db.CountTotal(ctx, &start) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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 { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -11,9 +11,6 @@ import ( | |||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	pbgroup "github.com/openimsdk/protocol/group" | 	pbgroup "github.com/openimsdk/protocol/group" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| 	"github.com/openimsdk/tools/errs" |  | ||||||
| 	"github.com/openimsdk/tools/mcontext" |  | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const versionSyncLimit = 500 | const versionSyncLimit = 500 | ||||||
| @ -23,10 +20,8 @@ func (g *groupServer) GetFullGroupMemberUserIDs(ctx context.Context, req *pbgrou | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if opUserID := mcontext.GetOpUserID(ctx); !datautil.Contain(opUserID, g.config.Share.IMAdminUserID...) { | 	if err := authverify.CheckAccessIn(ctx, userIDs...); err != nil { | ||||||
| 		if !datautil.Contain(opUserID, userIDs...) { | 		return nil, err | ||||||
| 			return nil, errs.ErrNoPermission.WrapMsg("user not in group") |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 	vl, err := g.db.FindMaxGroupMemberVersionCache(ctx, req.GroupID) | 	vl, err := g.db.FindMaxGroupMemberVersionCache(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	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) { | 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) | 	group, err := g.db.TakeGroup(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -76,9 +74,6 @@ func (g *groupServer) GetIncrementalGroupMember(ctx context.Context, req *pbgrou | |||||||
| 	if group.Status == constant.GroupStatusDismissed { | 	if group.Status == constant.GroupStatusDismissed { | ||||||
| 		return nil, servererrs.ErrDismissedAlready.Wrap() | 		return nil, servererrs.ErrDismissedAlready.Wrap() | ||||||
| 	} | 	} | ||||||
| 	if _, err := g.db.TakeGroupMember(ctx, req.GroupID, mcontext.GetOpUserID(ctx)); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	var ( | 	var ( | ||||||
| 		hasGroupUpdate bool | 		hasGroupUpdate bool | ||||||
| 		sortVersion    uint64 | 		sortVersion    uint64 | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"errors" | 	"errors" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	cbapi "github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | 	cbapi "github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	"github.com/openimsdk/protocol/msg" | 	"github.com/openimsdk/protocol/msg" | ||||||
| @ -29,6 +30,9 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *msg.GetConversationsHasReadAndMaxSeqReq) (*msg.GetConversationsHasReadAndMaxSeqResp, error) { | 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 | 	var conversationIDs []string | ||||||
| 	if len(req.ConversationIDs) == 0 { | 	if len(req.ConversationIDs) == 0 { | ||||||
| 		var err error | 		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) { | 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) | 	maxSeq, err := m.MsgDatabase.GetMaxSeq(ctx, req.ConversationID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | func (m *msgServer) MarkMsgsAsRead(ctx context.Context, req *msg.MarkMsgsAsReadReq) (*msg.MarkMsgsAsReadResp, error) { | ||||||
| 	if len(req.Seqs) < 1 { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("seqs must not be empty") | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	maxSeq, err := m.MsgDatabase.GetMaxSeq(ctx, req.ConversationID) | 	maxSeq, err := m.MsgDatabase.GetMaxSeq(ctx, req.ConversationID) | ||||||
| 	if err != nil { | 	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) { | 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) | 	conversation, err := m.ConversationLocalCache.GetConversation(ctx, req.UserID, req.ConversationID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -216,5 +226,4 @@ func (m *msgServer) sendMarkAsReadNotification(ctx context.Context, conversation | |||||||
| 		HasReadSeq:       hasReadSeq, | 		HasReadSeq:       hasReadSeq, | ||||||
| 	} | 	} | ||||||
| 	m.notificationSender.NotificationWithSessionType(ctx, sendID, recvID, constant.HasReadReceipt, sessionType, tips) | 	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) { | 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) | 	err := m.MsgDatabase.DeleteMsgsPhysicalBySeqs(ctx, req.ConversationID, req.Seqs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
| @ -17,11 +17,14 @@ package msg | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"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/common/prommetrics" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	pbconversation "github.com/openimsdk/protocol/conversation" | 	pbconv "github.com/openimsdk/protocol/conversation" | ||||||
| 	pbmsg "github.com/openimsdk/protocol/msg" | 	pbmsg "github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| 	"github.com/openimsdk/protocol/wrapperspb" | 	"github.com/openimsdk/protocol/wrapperspb" | ||||||
| @ -29,13 +32,15 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/openimsdk/tools/mcontext" | 	"github.com/openimsdk/tools/mcontext" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"google.golang.org/protobuf/proto" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) SendMsg(ctx context.Context, req *pbmsg.SendMsgReq) (*pbmsg.SendMsgResp, error) { | func (m *msgServer) SendMsg(ctx context.Context, req *pbmsg.SendMsgReq) (*pbmsg.SendMsgResp, error) { | ||||||
| 	if req.MsgData == nil { | 	if req.MsgData == nil { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("msgData is 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) | 	before := new(*sdkws.MsgData) | ||||||
| 	resp, err := m.sendMsg(ctx, req, before) | 	resp, err := m.sendMsg(ctx, req, before) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -104,7 +109,7 @@ func (m *msgServer) setConversationAtInfo(nctx context.Context, msg *sdkws.MsgDa | |||||||
| 
 | 
 | ||||||
| 	var atUserID []string | 	var atUserID []string | ||||||
| 
 | 
 | ||||||
| 	conversation := &pbconversation.ConversationReq{ | 	conversation := &pbconv.ConversationReq{ | ||||||
| 		ConversationID:   msgprocessor.GetConversationIDByMsg(msg), | 		ConversationID:   msgprocessor.GetConversationIDByMsg(msg), | ||||||
| 		ConversationType: msg.SessionType, | 		ConversationType: msg.SessionType, | ||||||
| 		GroupID:          msg.GroupID, | 		GroupID:          msg.GroupID, | ||||||
| @ -171,13 +176,7 @@ func (m *msgServer) sendMsgSingleChat(ctx context.Context, req *pbmsg.SendMsgReq | |||||||
| 	isSend := true | 	isSend := true | ||||||
| 	isNotification := msgprocessor.IsNotificationByMsg(req.MsgData) | 	isNotification := msgprocessor.IsNotificationByMsg(req.MsgData) | ||||||
| 	if !isNotification { | 	if !isNotification { | ||||||
| 		isSend, err = m.modifyMessageByUserMessageReceiveOpt( | 		isSend, err = m.modifyMessageByUserMessageReceiveOpt(authverify.WithTempAdmin(ctx), req.MsgData.RecvID, conversationutil.GenConversationIDForSingle(req.MsgData.SendID, req.MsgData.RecvID), constant.SingleChatType, req) | ||||||
| 			ctx, |  | ||||||
| 			req.MsgData.RecvID, |  | ||||||
| 			conversationutil.GenConversationIDForSingle(req.MsgData.SendID, req.MsgData.RecvID), |  | ||||||
| 			constant.SingleChatType, |  | ||||||
| 			req, |  | ||||||
| 		) |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -17,9 +17,10 @@ package msg | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"sort" | ||||||
|  | 
 | ||||||
| 	pbmsg "github.com/openimsdk/protocol/msg" | 	pbmsg "github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
| 	"sort" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) GetConversationMaxSeq(ctx context.Context, req *pbmsg.GetConversationMaxSeqReq) (*pbmsg.GetConversationMaxSeqResp, error) { | func (m *msgServer) GetConversationMaxSeq(ctx context.Context, req *pbmsg.GetConversationMaxSeqReq) (*pbmsg.GetConversationMaxSeqResp, error) { | ||||||
|  | |||||||
| @ -16,15 +16,20 @@ package msg | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" |  | ||||||
| 	"time" | 	"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/msg" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) GetActiveUser(ctx context.Context, req *msg.GetActiveUserReq) (*msg.GetActiveUserResp, error) { | 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) | 	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 { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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) | 	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 { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
| @ -29,6 +29,9 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) PullMessageBySeqs(ctx context.Context, req *sdkws.PullMessageBySeqsReq) (*sdkws.PullMessageBySeqsResp, error) { | 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 := &sdkws.PullMessageBySeqsResp{} | ||||||
| 	resp.Msgs = make(map[string]*sdkws.PullMsgs) | 	resp.Msgs = make(map[string]*sdkws.PullMsgs) | ||||||
| 	resp.NotificationMsgs = 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) { | 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) | 	in1, in2, err := s.blackDatabase.CheckIn(ctx, req.UserID1, req.UserID2) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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) | 	friends, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.FriendUserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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{} | 	resp = &relation.GetDesignatedFriendsResp{} | ||||||
| 	if datautil.Duplicate(req.FriendUserIDs) { | 	if datautil.Duplicate(req.FriendUserIDs) { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("friend userID repeated") | 		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. | // Get the list of friend requests sent out proactively. | ||||||
| func (s *friendServer) GetDesignatedFriendsApply(ctx context.Context, | func (s *friendServer) GetDesignatedFriendsApply(ctx context.Context, req *relation.GetDesignatedFriendsApplyReq) (resp *relation.GetDesignatedFriendsApplyResp, err error) { | ||||||
| 	req *relation.GetDesignatedFriendsApplyReq, | 	if err := authverify.CheckAccessIn(ctx, req.FromUserID, req.ToUserID); err != nil { | ||||||
| ) (resp *relation.GetDesignatedFriendsApplyResp, err error) { | 		return nil, err | ||||||
|  | 	} | ||||||
| 	friendRequests, err := s.db.FindBothFriendRequests(ctx, req.FromUserID, req.ToUserID) | 	friendRequests, err := s.db.FindBothFriendRequests(ctx, req.FromUserID, req.ToUserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -374,6 +381,9 @@ func (s *friendServer) GetPaginationFriendsApplyFrom(ctx context.Context, req *r | |||||||
| 
 | 
 | ||||||
| // ok. | // ok. | ||||||
| func (s *friendServer) IsFriend(ctx context.Context, req *relation.IsFriendReq) (resp *relation.IsFriendResp, err error) { | 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 = &relation.IsFriendResp{} | ||||||
| 	resp.InUser1Friends, resp.InUser2Friends, err = s.db.CheckIn(ctx, req.UserID1, req.UserID2) | 	resp.InUser1Friends, resp.InUser2Friends, err = s.db.CheckIn(ctx, req.UserID1, req.UserID2) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -426,6 +436,9 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | |||||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | 		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) | 	userMap, err := s.userClient.GetUsersInfoMap(ctx, req.UserIDList) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -494,10 +507,7 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | |||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) UpdateFriends( | func (s *friendServer) UpdateFriends(ctx context.Context, req *relation.UpdateFriendsReq) (*relation.UpdateFriendsResp, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	req *relation.UpdateFriendsReq, |  | ||||||
| ) (*relation.UpdateFriendsResp, error) { |  | ||||||
| 	if len(req.FriendUserIDs) == 0 { | 	if len(req.FriendUserIDs) == 0 { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("friendIDList is empty") | 		return nil, errs.ErrArgs.WrapMsg("friendIDList is empty") | ||||||
| 	} | 	} | ||||||
| @ -505,6 +515,10 @@ func (s *friendServer) UpdateFriends( | |||||||
| 		return nil, errs.ErrArgs.WrapMsg("friendIDList repeated") | 		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) | 	_, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.FriendUserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
| @ -25,6 +25,7 @@ import ( | |||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	"github.com/openimsdk/protocol/third" | 	"github.com/openimsdk/protocol/third" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
|  | 	"github.com/openimsdk/tools/mcontext" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"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) { | func (t *thirdServer) UploadLogs(ctx context.Context, req *third.UploadLogsReq) (*third.UploadLogsResp, error) { | ||||||
| 	var dbLogs []*relationtb.Log | 	var dbLogs []*relationtb.Log | ||||||
| 	userID := ctx.Value(constant.OpUserID).(string) | 	userID := mcontext.GetOpUserID(ctx) | ||||||
| 	platform := constant.PlatformID2Name[int(req.Platform)] | 	platform := constant.PlatformID2Name[int(req.Platform)] | ||||||
| 	for _, fileURL := range req.FileURLs { | 	for _, fileURL := range req.FileURLs { | ||||||
| 		log := relationtb.Log{ | 		log := relationtb.Log{ | ||||||
|  | |||||||
| @ -19,6 +19,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"time" | 	"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" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/mcache" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/mcache" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | 	"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) { | 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)) | 	err = t.thirdDatabase.SetAppBadge(ctx, req.UserID, int(req.AppUnreadCount)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
| @ -46,7 +46,7 @@ func TestName(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| 	srv := &cronServer{ | 	srv := &cronServer{ | ||||||
| 		ctx: ctx, | 		ctx: ctx, | ||||||
| 		config: &CronTaskConfig{ | 		config: &Config{ | ||||||
| 			CronTask: config.CronTask{ | 			CronTask: config.CronTask{ | ||||||
| 				RetainChatRecords: 1, | 				RetainChatRecords: 1, | ||||||
| 				FileExpireTime:    1, | 				FileExpireTime:    1, | ||||||
|  | |||||||
| @ -64,16 +64,57 @@ func GetIMAdminUserIDs(ctx context.Context) []string { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func IsAdmin(ctx context.Context) bool { | 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 { | func CheckAccess(ctx context.Context, ownerUserID string) error { | ||||||
| 	opUserID := mcontext.GetOpUserID(ctx) | 	if mcontext.GetOpUserID(ctx) == ownerUserID { | ||||||
| 	if opUserID == ownerUserID { |  | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	if datautil.Contain(mcontext.GetOpUserID(ctx), GetIMAdminUserIDs(ctx)...) { | 	if IsAdmin(ctx) { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	return servererrs.ErrNoPermission.WrapMsg("ownerUserID", ownerUserID) | 	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)...) | ||||||
|  | } | ||||||
|  | |||||||
| @ -62,4 +62,8 @@ const ( | |||||||
| 	CallbackBeforeMembersJoinGroupCommand              = "callbackBeforeMembersJoinGroupCommand" | 	CallbackBeforeMembersJoinGroupCommand              = "callbackBeforeMembersJoinGroupCommand" | ||||||
| 	CallbackBeforeSetGroupMemberInfoCommand            = "callbackBeforeSetGroupMemberInfoCommand" | 	CallbackBeforeSetGroupMemberInfoCommand            = "callbackBeforeSetGroupMemberInfoCommand" | ||||||
| 	CallbackAfterSetGroupMemberInfoCommand             = "callbackAfterSetGroupMemberInfoCommand" | 	CallbackAfterSetGroupMemberInfoCommand             = "callbackAfterSetGroupMemberInfoCommand" | ||||||
|  | 	CallbackBeforeCreateSingleChatConversationsCommand = "callbackBeforeCreateSingleChatConversationsCommand" | ||||||
|  | 	CallbackAfterCreateSingleChatConversationsCommand  = "callbackAfterCreateSingleChatConversationsCommand" | ||||||
|  | 	CallbackBeforeCreateGroupChatConversationsCommand  = "callbackBeforeCreateGroupChatConversationsCommand" | ||||||
|  | 	CallbackAfterCreateGroupChatConversationsCommand   = "callbackAfterCreateGroupChatConversationsCommand" | ||||||
| ) | ) | ||||||
|  | |||||||
							
								
								
									
										91
									
								
								pkg/callbackstruct/conversation.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								pkg/callbackstruct/conversation.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,91 @@ | |||||||
|  | package callbackstruct | ||||||
|  | 
 | ||||||
|  | type CallbackBeforeCreateSingleChatConversationsReq struct { | ||||||
|  | 	CallbackCommand  `json:"callbackCommand"` | ||||||
|  | 	OwnerUserID      string `json:"owner_user_id"` | ||||||
|  | 	ConversationID   string `json:"conversation_id"` | ||||||
|  | 	ConversationType int32  `json:"conversation_type"` | ||||||
|  | 	UserID           string `json:"user_id"` | ||||||
|  | 	RecvMsgOpt       int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned         bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat    bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration     int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType      int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo     string `json:"attached_info"` | ||||||
|  | 	Ex               string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackBeforeCreateSingleChatConversationsResp struct { | ||||||
|  | 	CommonCallbackResp | ||||||
|  | 	RecvMsgOpt    *int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned      *bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat *bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration  *int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType   *int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo  *string `json:"attached_info"` | ||||||
|  | 	Ex            *string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackAfterCreateSingleChatConversationsReq struct { | ||||||
|  | 	CallbackCommand  `json:"callbackCommand"` | ||||||
|  | 	OwnerUserID      string `json:"owner_user_id"` | ||||||
|  | 	ConversationID   string `json:"conversation_id"` | ||||||
|  | 	ConversationType int32  `json:"conversation_type"` | ||||||
|  | 	UserID           string `json:"user_id"` | ||||||
|  | 	RecvMsgOpt       int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned         bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat    bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration     int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType      int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo     string `json:"attached_info"` | ||||||
|  | 	Ex               string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackAfterCreateSingleChatConversationsResp struct { | ||||||
|  | 	CommonCallbackResp | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackBeforeCreateGroupChatConversationsReq struct { | ||||||
|  | 	CallbackCommand  `json:"callbackCommand"` | ||||||
|  | 	OwnerUserID      string `json:"owner_user_id"` | ||||||
|  | 	ConversationID   string `json:"conversation_id"` | ||||||
|  | 	ConversationType int32  `json:"conversation_type"` | ||||||
|  | 	GroupID          string `json:"group_id"` | ||||||
|  | 	RecvMsgOpt       int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned         bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat    bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration     int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType      int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo     string `json:"attached_info"` | ||||||
|  | 	Ex               string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackBeforeCreateGroupChatConversationsResp struct { | ||||||
|  | 	CommonCallbackResp | ||||||
|  | 	RecvMsgOpt    *int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned      *bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat *bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration  *int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType   *int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo  *string `json:"attached_info"` | ||||||
|  | 	Ex            *string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackAfterCreateGroupChatConversationsReq struct { | ||||||
|  | 	CallbackCommand  `json:"callbackCommand"` | ||||||
|  | 	OwnerUserID      string `json:"owner_user_id"` | ||||||
|  | 	ConversationID   string `json:"conversation_id"` | ||||||
|  | 	ConversationType int32  `json:"conversation_type"` | ||||||
|  | 	GroupID          string `json:"group_id"` | ||||||
|  | 	RecvMsgOpt       int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned         bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat    bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration     int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType      int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo     string `json:"attached_info"` | ||||||
|  | 	Ex               string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackAfterCreateGroupChatConversationsResp struct { | ||||||
|  | 	CommonCallbackResp | ||||||
|  | } | ||||||
| @ -41,6 +41,7 @@ func NewConversationRpcCmd() *ConversationRpcCmd { | |||||||
| 		config.MongodbConfigFileName:            &conversationConfig.MongodbConfig, | 		config.MongodbConfigFileName:            &conversationConfig.MongodbConfig, | ||||||
| 		config.ShareFileName:                    &conversationConfig.Share, | 		config.ShareFileName:                    &conversationConfig.Share, | ||||||
| 		config.NotificationFileName:             &conversationConfig.NotificationConfig, | 		config.NotificationFileName:             &conversationConfig.NotificationConfig, | ||||||
|  | 		config.WebhooksConfigFileName:           &conversationConfig.WebhooksConfig, | ||||||
| 		config.LocalCacheConfigFileName:         &conversationConfig.LocalCacheConfig, | 		config.LocalCacheConfigFileName:         &conversationConfig.LocalCacheConfig, | ||||||
| 		config.DiscoveryConfigFilename:          &conversationConfig.Discovery, | 		config.DiscoveryConfigFilename:          &conversationConfig.Discovery, | ||||||
| 	} | 	} | ||||||
| @ -67,6 +68,7 @@ func (a *ConversationRpcCmd) runE() error { | |||||||
| 			a.conversationConfig.NotificationConfig.GetConfigFileName(), | 			a.conversationConfig.NotificationConfig.GetConfigFileName(), | ||||||
| 			a.conversationConfig.Share.GetConfigFileName(), | 			a.conversationConfig.Share.GetConfigFileName(), | ||||||
| 			a.conversationConfig.LocalCacheConfig.GetConfigFileName(), | 			a.conversationConfig.LocalCacheConfig.GetConfigFileName(), | ||||||
|  | 			a.conversationConfig.WebhooksConfig.GetConfigFileName(), | ||||||
| 			a.conversationConfig.Discovery.GetConfigFileName(), | 			a.conversationConfig.Discovery.GetConfigFileName(), | ||||||
| 		}, nil, | 		}, nil, | ||||||
| 		conversation.Start) | 		conversation.Start) | ||||||
|  | |||||||
| @ -22,7 +22,6 @@ import ( | |||||||
| 	relationtb "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	relationtb "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
| 
 | 
 | ||||||
| 	"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/open-im-server/v3/pkg/msgprocessor" |  | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	"github.com/openimsdk/tools/db/pagination" | 	"github.com/openimsdk/tools/db/pagination" | ||||||
| 	"github.com/openimsdk/tools/db/tx" | 	"github.com/openimsdk/tools/db/tx" | ||||||
| @ -47,8 +46,11 @@ type ConversationDatabase interface { | |||||||
| 	// SetUsersConversationFieldTx updates a specific field for multiple users' conversations, creating new conversations if they do not exist, or updates them otherwise. This operation is | 	// SetUsersConversationFieldTx updates a specific field for multiple users' conversations, creating new conversations if they do not exist, or updates them otherwise. This operation is | ||||||
| 	// transactional. | 	// transactional. | ||||||
| 	SetUsersConversationFieldTx(ctx context.Context, userIDs []string, conversation *relationtb.Conversation, fieldMap map[string]any) error | 	SetUsersConversationFieldTx(ctx context.Context, userIDs []string, conversation *relationtb.Conversation, fieldMap map[string]any) error | ||||||
|  | 	// UpdateUserConversations updates all conversations related to a specified user. | ||||||
|  | 	// This function does NOT update the user's own conversations but rather the conversations where this user is involved (e.g., other users' conversations referencing this user). | ||||||
|  | 	UpdateUserConversations(ctx context.Context, userID string, args map[string]any) error | ||||||
| 	// CreateGroupChatConversation creates a group chat conversation for the specified group ID and user IDs. | 	// CreateGroupChatConversation creates a group chat conversation for the specified group ID and user IDs. | ||||||
| 	CreateGroupChatConversation(ctx context.Context, groupID string, userIDs []string) error | 	CreateGroupChatConversation(ctx context.Context, groupID string, userIDs []string, conversations *relationtb.Conversation) error | ||||||
| 	// GetConversationIDs retrieves conversation IDs for a given user. | 	// GetConversationIDs retrieves conversation IDs for a given user. | ||||||
| 	GetConversationIDs(ctx context.Context, userID string) ([]string, error) | 	GetConversationIDs(ctx context.Context, userID string) ([]string, error) | ||||||
| 	// GetUserConversationIDsHash gets the hash of conversation IDs for a given user. | 	// GetUserConversationIDsHash gets the hash of conversation IDs for a given user. | ||||||
| @ -146,6 +148,18 @@ func (c *conversationDatabase) SetUsersConversationFieldTx(ctx context.Context, | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (c *conversationDatabase) UpdateUserConversations(ctx context.Context, userID string, args map[string]any) error { | ||||||
|  | 	conversations, err := c.conversationDB.UpdateUserConversations(ctx, userID, args) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	cache := c.cache.CloneConversationCache() | ||||||
|  | 	for _, conversation := range conversations { | ||||||
|  | 		cache = cache.DelUsersConversation(conversation.ConversationID, conversation.OwnerUserID).DelConversationVersionUserIDs(conversation.OwnerUserID) | ||||||
|  | 	} | ||||||
|  | 	return cache.ChainExecDel(ctx) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (c *conversationDatabase) UpdateUsersConversationField(ctx context.Context, userIDs []string, conversationID string, args map[string]any) error { | func (c *conversationDatabase) UpdateUsersConversationField(ctx context.Context, userIDs []string, conversationID string, args map[string]any) error { | ||||||
| 	_, err := c.conversationDB.UpdateByMap(ctx, userIDs, conversationID, args) | 	_, err := c.conversationDB.UpdateByMap(ctx, userIDs, conversationID, args) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -298,10 +312,10 @@ func (c *conversationDatabase) SetUserConversations(ctx context.Context, ownerUs | |||||||
| //	return c.cache.GetSuperGroupRecvMsgNotNotifyUserIDs(ctx, groupID) | //	return c.cache.GetSuperGroupRecvMsgNotNotifyUserIDs(ctx, groupID) | ||||||
| //} | //} | ||||||
| 
 | 
 | ||||||
| func (c *conversationDatabase) CreateGroupChatConversation(ctx context.Context, groupID string, userIDs []string) error { | func (c *conversationDatabase) CreateGroupChatConversation(ctx context.Context, groupID string, userIDs []string, conversation *relationtb.Conversation) error { | ||||||
| 	return c.tx.Transaction(ctx, func(ctx context.Context) error { | 	return c.tx.Transaction(ctx, func(ctx context.Context) error { | ||||||
| 		cache := c.cache.CloneConversationCache() | 		cache := c.cache.CloneConversationCache() | ||||||
| 		conversationID := msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, groupID) | 		conversationID := conversation.ConversationID | ||||||
| 		existConversationUserIDs, err := c.conversationDB.FindUserID(ctx, userIDs, []string{conversationID}) | 		existConversationUserIDs, err := c.conversationDB.FindUserID(ctx, userIDs, []string{conversationID}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| @ -309,7 +323,15 @@ func (c *conversationDatabase) CreateGroupChatConversation(ctx context.Context, | |||||||
| 		notExistUserIDs := stringutil.DifferenceString(userIDs, existConversationUserIDs) | 		notExistUserIDs := stringutil.DifferenceString(userIDs, existConversationUserIDs) | ||||||
| 		var conversations []*relationtb.Conversation | 		var conversations []*relationtb.Conversation | ||||||
| 		for _, v := range notExistUserIDs { | 		for _, v := range notExistUserIDs { | ||||||
| 			conversation := relationtb.Conversation{ConversationType: constant.ReadGroupChatType, GroupID: groupID, OwnerUserID: v, ConversationID: conversationID} | 			conversation := relationtb.Conversation{ | ||||||
|  | 				ConversationType: conversation.ConversationType, GroupID: groupID, OwnerUserID: v, ConversationID: conversationID, | ||||||
|  | 				// the parameters have default value | ||||||
|  | 				RecvMsgOpt: conversation.RecvMsgOpt, IsPinned: conversation.IsPinned, IsPrivateChat: conversation.IsPrivateChat, | ||||||
|  | 				BurnDuration: conversation.BurnDuration, GroupAtType: conversation.GroupAtType, AttachedInfo: conversation.AttachedInfo, | ||||||
|  | 				Ex: conversation.Ex, MaxSeq: conversation.MaxSeq, MinSeq: conversation.MinSeq, CreateTime: conversation.CreateTime, | ||||||
|  | 				MsgDestructTime: conversation.MsgDestructTime, IsMsgDestruct: conversation.IsMsgDestruct, LatestMsgDestructTime: conversation.LatestMsgDestructTime, | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
| 			conversations = append(conversations, &conversation) | 			conversations = append(conversations, &conversation) | ||||||
| 			cache = cache.DelConversations(v, conversationID).DelConversationNotReceiveMessageUserIDs(conversationID) | 			cache = cache.DelConversations(v, conversationID).DelConversationNotReceiveMessageUserIDs(conversationID) | ||||||
| 		} | 		} | ||||||
| @ -320,7 +342,7 @@ func (c *conversationDatabase) CreateGroupChatConversation(ctx context.Context, | |||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		_, err = c.conversationDB.UpdateByMap(ctx, existConversationUserIDs, conversationID, map[string]any{"max_seq": 0}) | 		_, err = c.conversationDB.UpdateByMap(ctx, existConversationUserIDs, conversationID, map[string]any{"max_seq": conversation.MaxSeq}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -24,6 +24,7 @@ import ( | |||||||
| type Conversation interface { | type Conversation interface { | ||||||
| 	Create(ctx context.Context, conversations []*model.Conversation) (err error) | 	Create(ctx context.Context, conversations []*model.Conversation) (err error) | ||||||
| 	UpdateByMap(ctx context.Context, userIDs []string, conversationID string, args map[string]any) (rows int64, err error) | 	UpdateByMap(ctx context.Context, userIDs []string, conversationID string, args map[string]any) (rows int64, err error) | ||||||
|  | 	UpdateUserConversations(ctx context.Context, userID string, args map[string]any) ([]*model.Conversation, error) | ||||||
| 	Update(ctx context.Context, conversation *model.Conversation) (err error) | 	Update(ctx context.Context, conversation *model.Conversation) (err error) | ||||||
| 	Find(ctx context.Context, ownerUserID string, conversationIDs []string) (conversations []*model.Conversation, err error) | 	Find(ctx context.Context, ownerUserID string, conversationIDs []string) (conversations []*model.Conversation, err error) | ||||||
| 	FindUserID(ctx context.Context, userIDs []string, conversationIDs []string) ([]string, error) | 	FindUserID(ctx context.Context, userIDs []string, conversationIDs []string) ([]string, error) | ||||||
|  | |||||||
| @ -21,23 +21,32 @@ import ( | |||||||
| 	"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/bson" | ||||||
|  | 	"go.mongodb.org/mongo-driver/mongo" | ||||||
|  | 	"go.mongodb.org/mongo-driver/mongo/options" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	"github.com/openimsdk/tools/db/mongoutil" | 	"github.com/openimsdk/tools/db/mongoutil" | ||||||
| 	"github.com/openimsdk/tools/db/pagination" | 	"github.com/openimsdk/tools/db/pagination" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"go.mongodb.org/mongo-driver/bson" |  | ||||||
| 	"go.mongodb.org/mongo-driver/mongo" |  | ||||||
| 	"go.mongodb.org/mongo-driver/mongo/options" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func NewConversationMongo(db *mongo.Database) (*ConversationMgo, error) { | func NewConversationMongo(db *mongo.Database) (*ConversationMgo, error) { | ||||||
| 	coll := db.Collection(database.ConversationName) | 	coll := db.Collection(database.ConversationName) | ||||||
| 	_, err := coll.Indexes().CreateOne(context.Background(), mongo.IndexModel{ | 	_, err := coll.Indexes().CreateMany(context.Background(), []mongo.IndexModel{ | ||||||
|  | 		{ | ||||||
| 			Keys: bson.D{ | 			Keys: bson.D{ | ||||||
| 				{Key: "owner_user_id", Value: 1}, | 				{Key: "owner_user_id", Value: 1}, | ||||||
| 				{Key: "conversation_id", Value: 1}, | 				{Key: "conversation_id", Value: 1}, | ||||||
| 			}, | 			}, | ||||||
| 			Options: options.Index().SetUnique(true), | 			Options: options.Index().SetUnique(true), | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Keys: bson.D{ | ||||||
|  | 				{Key: "user_id", Value: 1}, | ||||||
|  | 			}, | ||||||
|  | 			Options: options.Index(), | ||||||
|  | 		}, | ||||||
| 	}) | 	}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, errs.Wrap(err) | 		return nil, errs.Wrap(err) | ||||||
| @ -101,6 +110,38 @@ func (c *ConversationMgo) UpdateByMap(ctx context.Context, userIDs []string, con | |||||||
| 	return rows, nil | 	return rows, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (c *ConversationMgo) UpdateUserConversations(ctx context.Context, userID string, args map[string]any) ([]*model.Conversation, error) { | ||||||
|  | 	if len(args) == 0 { | ||||||
|  | 		return nil, nil | ||||||
|  | 	} | ||||||
|  | 	filter := bson.M{ | ||||||
|  | 		"user_id": userID, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	conversations, err := mongoutil.Find[*model.Conversation](ctx, c.coll, filter, options.Find().SetProjection(bson.M{"_id": 0, "owner_user_id": 1, "conversation_id": 1})) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	err = mongoutil.IncrVersion(func() error { | ||||||
|  | 		_, err := mongoutil.UpdateMany(ctx, c.coll, filter, bson.M{"$set": args}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}, func() error { | ||||||
|  | 		for _, conversation := range conversations { | ||||||
|  | 			if err := c.version.IncrVersion(ctx, conversation.OwnerUserID, []string{conversation.ConversationID}, model.VersionStateUpdate); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return conversations, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (c *ConversationMgo) Update(ctx context.Context, conversation *model.Conversation) (err error) { | func (c *ConversationMgo) Update(ctx context.Context, conversation *model.Conversation) (err error) { | ||||||
| 	return mongoutil.IncrVersion(func() error { | 	return mongoutil.IncrVersion(func() error { | ||||||
| 		return mongoutil.UpdateOne(ctx, c.coll, bson.M{"owner_user_id": conversation.OwnerUserID, "conversation_id": conversation.ConversationID}, bson.M{"$set": conversation}, true) | 		return mongoutil.UpdateOne(ctx, c.coll, bson.M{"owner_user_id": conversation.OwnerUserID, "conversation_id": conversation.ConversationID}, bson.M{"$set": conversation}, true) | ||||||
|  | |||||||
| @ -7,6 +7,7 @@ import ( | |||||||
| 	"sync" | 	"sync" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/utils/idutil" | 	"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]) { | func (b *Batcher[T]) run(channelID int, ch <-chan *Msg[T]) { | ||||||
| 	defer b.wait.Done() | 	defer b.wait.Done() | ||||||
|  | 	ctx := authverify.WithTempAdmin(context.Background()) | ||||||
| 	for { | 	for { | ||||||
| 		select { | 		select { | ||||||
| 		case messages, ok := <-ch: | 		case messages, ok := <-ch: | ||||||
| 			if !ok { | 			if !ok { | ||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
| 			b.Do(context.Background(), channelID, messages) | 			b.Do(ctx, channelID, messages) | ||||||
| 			if b.config.syncWait { | 			if b.config.syncWait { | ||||||
| 				b.counter.Done() | 				b.counter.Done() | ||||||
| 			} | 			} | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user