mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-26 13:12:12 +08:00 
			
		
		
		
	Merge branch 'pre-release-v3.8.4' into cherry-pick-964ee7a
This commit is contained in:
		
						commit
						7e9bb7bc76
					
				| @ -3,14 +3,7 @@ beforeSendSingleMsg: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
|   # Only the contentType in allowedTypes will send the callback. | ||||
|   # Supports two formats: a single type or a range. The range is defined by the lower and upper bounds connected with a hyphen ("-"). | ||||
|   # e.g. allowedTypes: [1, 100, 200-500, 600-700] means that only contentType within the range | ||||
|   # {1, 100} ∪ [200, 500] ∪ [600, 700] will be allowed through the filter. | ||||
|   # If not set, all contentType messages will through this filter. | ||||
|   allowedTypes: [] | ||||
|   # Only the contentType not in deniedTypes will send the callback. | ||||
|   # Supports two formats, same as allowedTypes. | ||||
|   # If not set, all contentType messages will through this filter. | ||||
|   deniedTypes: [] | ||||
| beforeUpdateUserInfoEx: | ||||
| @ -23,31 +16,30 @@ afterUpdateUserInfoEx: | ||||
| afterSendSingleMsg: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   # Only the senID/recvID specified in attentionIds will send the callback | ||||
|   # Only the recvID specified in attentionIds will send the callback | ||||
|   # if not set, all user messages will be callback | ||||
|   attentionIds: [] | ||||
|   # See beforeSendSingleMsg comment. | ||||
|   allowedTypes: [] | ||||
|   deniedTypes: [] | ||||
| beforeSendGroupMsg: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
|   # See beforeSendSingleMsg comment. | ||||
|   allowedTypes: [] | ||||
|   deniedTypes: [] | ||||
| beforeMsgModify: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
|   # See beforeSendSingleMsg comment. | ||||
|   allowedTypes: [] | ||||
|   deniedTypes: [] | ||||
| afterSendGroupMsg: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   # Only the recvID specified in attentionIds will send the callback | ||||
|   # if not set, all user messages will be callback | ||||
|   attentionIds: [] | ||||
|   # See beforeSendSingleMsg comment. | ||||
|   allowedTypes: [] | ||||
|   deniedTypes: [] | ||||
| afterUserOnline: | ||||
|   enable: false | ||||
|  | ||||
| @ -27,6 +27,7 @@ import ( | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/apistruct" | ||||
| 	"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/webhook" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| @ -210,6 +211,8 @@ func (m *MessageApi) getSendMsgReq(c *gin.Context, req apistruct.SendMsg) (sendM | ||||
| 		data = apistruct.AtElem{} | ||||
| 	case constant.Custom: | ||||
| 		data = apistruct.CustomElem{} | ||||
| 	case constant.MarkdownText: | ||||
| 		data = apistruct.MarkdownTextElem{} | ||||
| 	case constant.OANotification: | ||||
| 		data = apistruct.OANotificationElem{} | ||||
| 		req.SessionType = constant.NotificationChatType | ||||
| @ -521,7 +524,6 @@ func (m *MessageApi) SendSimpleMessage(c *gin.Context) { | ||||
| 		apiresp.GinError(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	m.ginRespSendMsg(c, sendReq, respPb) | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -100,7 +100,7 @@ func (m *msgServer) webhookAfterSendSingleMsg(ctx context.Context, after *config | ||||
| 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendSingleMsgCommand), | ||||
| 		RecvID:            msg.MsgData.RecvID, | ||||
| 	} | ||||
| 	m.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendSingleMsgResp{}, after) | ||||
| 	m.webhookClient.AsyncPostWithQuery(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendSingleMsgResp{}, after, buildKeyMsgDataQuery(msg.MsgData)) | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) webhookBeforeSendGroupMsg(ctx context.Context, before *config.BeforeConfig, msg *pbchat.SendMsgReq) error { | ||||
| @ -134,7 +134,8 @@ func (m *msgServer) webhookAfterSendGroupMsg(ctx context.Context, after *config. | ||||
| 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendGroupMsgCommand), | ||||
| 		GroupID:           msg.MsgData.GroupID, | ||||
| 	} | ||||
| 	m.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendGroupMsgResp{}, after) | ||||
| 
 | ||||
| 	m.webhookClient.AsyncPostWithQuery(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendGroupMsgResp{}, after, buildKeyMsgDataQuery(msg.MsgData)) | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) webhookBeforeMsgModify(ctx context.Context, before *config.BeforeConfig, msg *pbchat.SendMsgReq, beforeMsgData **sdkws.MsgData) error { | ||||
| @ -203,3 +204,15 @@ func (m *msgServer) webhookAfterRevokeMsg(ctx context.Context, after *config.Aft | ||||
| 	} | ||||
| 	m.webhookClient.AsyncPost(ctx, callbackReq.GetCallbackCommand(), callbackReq, &cbapi.CallbackAfterRevokeMsgResp{}, after) | ||||
| } | ||||
| 
 | ||||
| func buildKeyMsgDataQuery(msg *sdkws.MsgData) map[string]string { | ||||
| 	keyMsgData := apistruct.KeyMsgData{ | ||||
| 		SendID:  msg.SendID, | ||||
| 		RecvID:  msg.RecvID, | ||||
| 		GroupID: msg.GroupID, | ||||
| 	} | ||||
| 
 | ||||
| 	return map[string]string{ | ||||
| 		webhook.Key: base64.StdEncoding.EncodeToString(stringutil.StructToJsonBytes(keyMsgData)), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -1,11 +1,13 @@ | ||||
| package msg | ||||
| 
 | ||||
| import ( | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	pbchat "github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	pbchat "github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| @ -13,28 +15,50 @@ const ( | ||||
| ) | ||||
| 
 | ||||
| func filterAfterMsg(msg *pbchat.SendMsgReq, after *config.AfterConfig) bool { | ||||
| 	return filterMsg(msg, after.AttentionIds, after.AllowedTypes, after.DeniedTypes) | ||||
| 	return filterMsg(msg, after.AttentionIds, after.DeniedTypes) | ||||
| } | ||||
| 
 | ||||
| func filterBeforeMsg(msg *pbchat.SendMsgReq, before *config.BeforeConfig) bool { | ||||
| 	return filterMsg(msg, nil, before.AllowedTypes, before.DeniedTypes) | ||||
| 	return filterMsg(msg, nil, before.DeniedTypes) | ||||
| } | ||||
| 
 | ||||
| func filterMsg(msg *pbchat.SendMsgReq, attentionIds, allowedTypes, deniedTypes []string) bool { | ||||
| func filterMsg(msg *pbchat.SendMsgReq, attentionIds []string, deniedTypes []int32) bool { | ||||
| 	// According to the attentionIds configuration, only some users are sent | ||||
| 	if len(attentionIds) != 0 && !datautil.Contains([]string{msg.MsgData.SendID, msg.MsgData.RecvID}, attentionIds...) { | ||||
| 	if len(attentionIds) != 0 && !datautil.Contain(msg.MsgData.RecvID, attentionIds...) { | ||||
| 		return false | ||||
| 	} | ||||
| 	if len(allowedTypes) != 0 && !isInInterval(msg.MsgData.ContentType, allowedTypes) { | ||||
| 
 | ||||
| 	if defaultDeniedTypes(msg.MsgData.ContentType) { | ||||
| 		return false | ||||
| 	} | ||||
| 	if len(deniedTypes) != 0 && isInInterval(msg.MsgData.ContentType, deniedTypes) { | ||||
| 
 | ||||
| 	if len(deniedTypes) != 0 && datautil.Contain(msg.MsgData.ContentType, deniedTypes...) { | ||||
| 		return false | ||||
| 	} | ||||
| 	//if len(allowedTypes) != 0 && !isInInterval(msg.MsgData.ContentType, allowedTypes) { | ||||
| 	//	return false | ||||
| 	//} | ||||
| 	//if len(deniedTypes) != 0 && isInInterval(msg.MsgData.ContentType, deniedTypes) { | ||||
| 	//	return false | ||||
| 	//} | ||||
| 	return true | ||||
| } | ||||
| 
 | ||||
| func isInInterval(contentType int32, interval []string) bool { | ||||
| func defaultDeniedTypes(contentType int32) bool { | ||||
| 	if contentType >= constant.NotificationBegin && contentType <= constant.NotificationEnd { | ||||
| 		return true | ||||
| 	} | ||||
| 	if contentType == constant.Typing { | ||||
| 		return true | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
| 
 | ||||
| // isInInterval if data is in interval | ||||
| // Supports two formats: a single type or a range. The range is defined by the lower and upper bounds connected with a hyphen ("-") | ||||
| // e.g. [1, 100, 200-500, 600-700] means that only data within the range | ||||
| // {1, 100} ∪ [200, 500] ∪ [600, 700] will return true. | ||||
| func isInInterval(data int32, interval []string) bool { | ||||
| 	for _, v := range interval { | ||||
| 		if strings.Contains(v, separator) { | ||||
| 			// is interval | ||||
| @ -50,7 +74,7 @@ func isInInterval(contentType int32, interval []string) bool { | ||||
| 			if err != nil { | ||||
| 				continue | ||||
| 			} | ||||
| 			if datautil.BetweenEq(int(contentType), bottom, top) { | ||||
| 			if datautil.BetweenEq(int(data), bottom, top) { | ||||
| 				return true | ||||
| 			} | ||||
| 		} else { | ||||
| @ -58,7 +82,7 @@ func isInInterval(contentType int32, interval []string) bool { | ||||
| 			if err != nil { | ||||
| 				continue | ||||
| 			} | ||||
| 			if int(contentType) == iv { | ||||
| 			if int(data) == iv { | ||||
| 				return true | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| @ -16,13 +16,15 @@ package msg | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"math/rand" | ||||
| 	"strconv" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"github.com/openimsdk/tools/utils/encrypt" | ||||
| 	"github.com/openimsdk/tools/utils/timeutil" | ||||
| 	"math/rand" | ||||
| 	"strconv" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| @ -62,6 +64,13 @@ func (m *msgServer) messageVerification(ctx context.Context, data *msg.SendMsgRe | ||||
| 		if err := m.webhookBeforeSendSingleMsg(ctx, &m.config.WebhooksConfig.BeforeSendSingleMsg, data); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		u, err := m.UserLocalCache.GetUserInfo(ctx, data.MsgData.SendID) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		if authverify.CheckSystemAccount(ctx, u.AppMangerLevel) { | ||||
| 			return nil | ||||
| 		} | ||||
| 		black, err := m.FriendLocalCache.IsBlack(ctx, data.MsgData.SendID, data.MsgData.RecvID) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| @ -137,27 +146,9 @@ func (m *msgServer) encapsulateMsgData(msg *sdkws.MsgData) { | ||||
| 		msg.SendTime = timeutil.GetCurrentTimestampByMill() | ||||
| 	} | ||||
| 	switch msg.ContentType { | ||||
| 	case constant.Text: | ||||
| 		fallthrough | ||||
| 	case constant.Picture: | ||||
| 		fallthrough | ||||
| 	case constant.Voice: | ||||
| 		fallthrough | ||||
| 	case constant.Video: | ||||
| 		fallthrough | ||||
| 	case constant.File: | ||||
| 		fallthrough | ||||
| 	case constant.AtText: | ||||
| 		fallthrough | ||||
| 	case constant.Merger: | ||||
| 		fallthrough | ||||
| 	case constant.Card: | ||||
| 		fallthrough | ||||
| 	case constant.Location: | ||||
| 		fallthrough | ||||
| 	case constant.Custom: | ||||
| 		fallthrough | ||||
| 	case constant.Quote: | ||||
| 	case constant.Text, constant.Picture, constant.Voice, constant.Video, | ||||
| 		constant.File, constant.AtText, constant.Merger, constant.Card, | ||||
| 		constant.Location, constant.Custom, constant.Quote, constant.AdvancedText, constant.MarkdownText: | ||||
| 	case constant.Revoke: | ||||
| 		datautil.SetSwitchFromOptions(msg.Options, constant.IsUnreadCount, false) | ||||
| 		datautil.SetSwitchFromOptions(msg.Options, constant.IsOfflinePush, false) | ||||
|  | ||||
| @ -566,7 +566,7 @@ func (s *userServer) SearchNotificationAccount(ctx context.Context, req *pbuser. | ||||
| 		} | ||||
| 
 | ||||
| 		// Convert users to response format | ||||
| 		resp := s.userModelToResp(users, req.Pagination) | ||||
| 		resp := s.userModelToResp(users, req.Pagination, req.AppManagerLevel) | ||||
| 		if resp.Total != 0 { | ||||
| 			return resp, nil | ||||
| 		} | ||||
| @ -576,17 +576,24 @@ func (s *userServer) SearchNotificationAccount(ctx context.Context, req *pbuser. | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		resp = s.userModelToResp(users, req.Pagination) | ||||
| 		resp = s.userModelToResp(users, req.Pagination, req.AppManagerLevel) | ||||
| 		return resp, nil | ||||
| 	} | ||||
| 
 | ||||
| 	// If no keyword, find users with notification settings | ||||
| 	users, err = s.db.FindNotification(ctx, constant.AppNotificationAdmin) | ||||
| 	if req.AppManagerLevel != nil { | ||||
| 		users, err = s.db.FindNotification(ctx, int64(*req.AppManagerLevel)) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} else { | ||||
| 		users, err = s.db.FindSystemAccount(ctx) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	resp := s.userModelToResp(users, req.Pagination) | ||||
| 	resp := s.userModelToResp(users, req.Pagination, req.AppManagerLevel) | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| @ -625,11 +632,16 @@ func (s *userServer) genUserID() string { | ||||
| 	return string(data) | ||||
| } | ||||
| 
 | ||||
| func (s *userServer) userModelToResp(users []*tablerelation.User, pagination pagination.Pagination) *pbuser.SearchNotificationAccountResp { | ||||
| func (s *userServer) userModelToResp(users []*tablerelation.User, pagination pagination.Pagination, appManagerLevel *int32) *pbuser.SearchNotificationAccountResp { | ||||
| 	accounts := make([]*pbuser.NotificationAccountInfo, 0) | ||||
| 	var total int64 | ||||
| 	for _, v := range users { | ||||
| 		if v.AppMangerLevel >= constant.AppNotificationAdmin && !datautil.Contain(v.UserID, s.config.Share.IMAdminUserID...) { | ||||
| 			if appManagerLevel != nil { | ||||
| 				if v.AppMangerLevel != *appManagerLevel { | ||||
| 					continue | ||||
| 				} | ||||
| 			} | ||||
| 			temp := &pbuser.NotificationAccountInfo{ | ||||
| 				UserID:         v.UserID, | ||||
| 				FaceURL:        v.FaceURL, | ||||
|  | ||||
| @ -112,6 +112,21 @@ type BatchSendMsgResp struct { | ||||
| 	FailedIDs []string `json:"failedUserIDs"` | ||||
| } | ||||
| 
 | ||||
| // SendSingleMsgReq defines the structure for sending a message to multiple recipients. | ||||
| type SendSingleMsgReq struct { | ||||
| 	// groupMsg should appoint sendID | ||||
| 	SendID          string                 `json:"sendID"` | ||||
| 	Content         string                 `json:"content" binding:"required"` | ||||
| 	OfflinePushInfo *sdkws.OfflinePushInfo `json:"offlinePushInfo"` | ||||
| 	Ex              string                 `json:"ex"` | ||||
| } | ||||
| 
 | ||||
| type KeyMsgData struct { | ||||
| 	SendID  string `json:"sendID"` | ||||
| 	RecvID  string `json:"recvID"` | ||||
| 	GroupID string `json:"groupID"` | ||||
| } | ||||
| 
 | ||||
| // SingleReturnResult encapsulates the result of a single message send attempt. | ||||
| type SingleReturnResult struct { | ||||
| 	// ServerMsgID is the message identifier on the server-side. | ||||
|  | ||||
| @ -81,6 +81,15 @@ type TextElem struct { | ||||
| 	Content string `json:"content" validate:"required"` | ||||
| } | ||||
| 
 | ||||
| type MarkdownTextElem struct { | ||||
| 	Content string `mapstructure:"content" validate:"required"` | ||||
| } | ||||
| 
 | ||||
| type StreamMsgElem struct { | ||||
| 	Type    string `mapstructure:"type" validate:"required"` | ||||
| 	Content string `mapstructure:"content" validate:"required"` | ||||
| } | ||||
| 
 | ||||
| type RevokeElem struct { | ||||
| 	RevokeMsgClientID string `mapstructure:"revokeMsgClientID" validate:"required"` | ||||
| } | ||||
|  | ||||
| @ -20,6 +20,7 @@ import ( | ||||
| 
 | ||||
| 	"github.com/golang-jwt/jwt/v4" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| ) | ||||
| @ -55,3 +56,7 @@ func CheckAdmin(ctx context.Context, imAdminUserID []string) error { | ||||
| func IsManagerUserID(opUserID string, imAdminUserID []string) bool { | ||||
| 	return datautil.Contain(opUserID, imAdminUserID...) | ||||
| } | ||||
| 
 | ||||
| func CheckSystemAccount(ctx context.Context, level int32) bool { | ||||
| 	return level >= constant.AppAdmin | ||||
| } | ||||
|  | ||||
| @ -364,19 +364,17 @@ type Redis struct { | ||||
| } | ||||
| 
 | ||||
| type BeforeConfig struct { | ||||
| 	Enable         bool     `mapstructure:"enable"` | ||||
| 	Timeout        int      `mapstructure:"timeout"` | ||||
| 	FailedContinue bool     `mapstructure:"failedContinue"` | ||||
| 	AllowedTypes   []string `mapstructure:"allowedTypes"` | ||||
| 	DeniedTypes    []string `mapstructure:"deniedTypes"` | ||||
| 	Enable         bool    `yaml:"enable"` | ||||
| 	Timeout        int     `yaml:"timeout"` | ||||
| 	FailedContinue bool    `yaml:"failedContinue"` | ||||
| 	DeniedTypes    []int32 `yaml:"deniedTypes"` | ||||
| } | ||||
| 
 | ||||
| type AfterConfig struct { | ||||
| 	Enable       bool     `mapstructure:"enable"` | ||||
| 	Timeout      int      `mapstructure:"timeout"` | ||||
| 	AttentionIds []string `mapstructure:"attentionIds"` | ||||
| 	AllowedTypes []string `mapstructure:"allowedTypes"` | ||||
| 	DeniedTypes  []string `mapstructure:"deniedTypes"` | ||||
| 	Enable       bool     `yaml:"enable"` | ||||
| 	Timeout      int      `yaml:"timeout"` | ||||
| 	AttentionIds []string `yaml:"attentionIds"` | ||||
| 	DeniedTypes  []int32  `yaml:"deniedTypes"` | ||||
| } | ||||
| 
 | ||||
| type Share struct { | ||||
|  | ||||
| @ -20,6 +20,7 @@ import ( | ||||
| 
 | ||||
| 	"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/protocol/constant" | ||||
| 	"github.com/openimsdk/tools/db/pagination" | ||||
| 	"github.com/openimsdk/tools/db/tx" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| @ -37,8 +38,10 @@ type UserDatabase interface { | ||||
| 	Find(ctx context.Context, userIDs []string) (users []*model.User, err error) | ||||
| 	// Find userInfo By Nickname | ||||
| 	FindByNickname(ctx context.Context, nickname string) (users []*model.User, err error) | ||||
| 	// Find notificationAccounts | ||||
| 	// FindNotification find system account by level | ||||
| 	FindNotification(ctx context.Context, level int64) (users []*model.User, err error) | ||||
| 	// FindSystemAccount find all system account | ||||
| 	FindSystemAccount(ctx context.Context) (users []*model.User, err error) | ||||
| 	// Create Insert multiple external guarantees that the userID is not repeated and does not exist in the storage | ||||
| 	Create(ctx context.Context, users []*model.User) (err error) | ||||
| 	// UpdateByMap update (zero value) external guarantee userID exists | ||||
| @ -139,6 +142,10 @@ func (u *userDatabase) FindNotification(ctx context.Context, level int64) (users | ||||
| 	return u.userDB.TakeNotification(ctx, level) | ||||
| } | ||||
| 
 | ||||
| func (u *userDatabase) FindSystemAccount(ctx context.Context) (users []*model.User, err error) { | ||||
| 	return u.userDB.TakeGTEAppManagerLevel(ctx, constant.AppNotificationAdmin) | ||||
| } | ||||
| 
 | ||||
| // Create Insert multiple external guarantees that the userID is not repeated and does not exist in the storage. | ||||
| func (u *userDatabase) Create(ctx context.Context, users []*model.User) (err error) { | ||||
| 	return u.tx.Transaction(ctx, func(ctx context.Context) error { | ||||
|  | ||||
| @ -16,9 +16,10 @@ package mgo | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/protocol/user" | ||||
| 	"github.com/openimsdk/tools/db/mongoutil" | ||||
| @ -71,6 +72,10 @@ func (u *UserMgo) TakeNotification(ctx context.Context, level int64) (user []*mo | ||||
| 	return mongoutil.Find[*model.User](ctx, u.coll, bson.M{"app_manger_level": level}) | ||||
| } | ||||
| 
 | ||||
| func (u *UserMgo) TakeGTEAppManagerLevel(ctx context.Context, level int64) (user []*model.User, err error) { | ||||
| 	return mongoutil.Find[*model.User](ctx, u.coll, bson.M{"app_manager_level": bson.M{"$gte": level}}) | ||||
| } | ||||
| 
 | ||||
| func (u *UserMgo) TakeByNickname(ctx context.Context, nickname string) (user []*model.User, err error) { | ||||
| 	return mongoutil.Find[*model.User](ctx, u.coll, bson.M{"nickname": nickname}) | ||||
| } | ||||
|  | ||||
| @ -16,10 +16,11 @@ package database | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"github.com/openimsdk/protocol/user" | ||||
| 	"github.com/openimsdk/tools/db/pagination" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| type User interface { | ||||
| @ -28,6 +29,7 @@ type User interface { | ||||
| 	Find(ctx context.Context, userIDs []string) (users []*model.User, err error) | ||||
| 	Take(ctx context.Context, userID string) (user *model.User, err error) | ||||
| 	TakeNotification(ctx context.Context, level int64) (user []*model.User, err error) | ||||
| 	TakeGTEAppManagerLevel(ctx context.Context, level int64) (user []*model.User, err error) | ||||
| 	TakeByNickname(ctx context.Context, nickname string) (user []*model.User, err error) | ||||
| 	Page(ctx context.Context, pagination pagination.Pagination) (count int64, users []*model.User, err error) | ||||
| 	PageFindUser(ctx context.Context, level1 int64, level2 int64, pagination pagination.Pagination) (count int64, users []*model.User, err error) | ||||
|  | ||||
| @ -17,6 +17,9 @@ package webhook | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||
| @ -25,7 +28,6 @@ import ( | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/mq/memamq" | ||||
| 	"github.com/openimsdk/tools/utils/httputil" | ||||
| 	"net/http" | ||||
| ) | ||||
| 
 | ||||
| type Client struct { | ||||
| @ -37,6 +39,8 @@ type Client struct { | ||||
| const ( | ||||
| 	webhookWorkerCount = 2 | ||||
| 	webhookBufferSize  = 100 | ||||
| 
 | ||||
| 	Key = "key" | ||||
| ) | ||||
| 
 | ||||
| func NewWebhookClient(url string, options ...*memamq.MemoryQueue) *Client { | ||||
| @ -66,6 +70,12 @@ func (c *Client) AsyncPost(ctx context.Context, command string, req callbackstru | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (c *Client) AsyncPostWithQuery(ctx context.Context, command string, req callbackstruct.CallbackReq, resp callbackstruct.CallbackResp, after *config.AfterConfig, queryParams map[string]string) { | ||||
| 	if after.Enable { | ||||
| 		c.queue.Push(func() { c.postWithQuery(ctx, command, req, resp, after.Timeout, queryParams) }) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (c *Client) post(ctx context.Context, command string, input interface{}, output callbackstruct.CallbackResp, timeout int) error { | ||||
| 	ctx = mcontext.WithMustInfoCtx([]string{mcontext.GetOperationID(ctx), mcontext.GetOpUserID(ctx), mcontext.GetOpUserPlatform(ctx), mcontext.GetConnID(ctx)}) | ||||
| 	fullURL := c.url + "/" + command | ||||
| @ -84,3 +94,41 @@ func (c *Client) post(ctx context.Context, command string, input interface{}, ou | ||||
| 	log.ZInfo(ctx, "webhook success", "url", fullURL, "input", input, "response", string(b)) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (c *Client) postWithQuery(ctx context.Context, command string, input interface{}, output callbackstruct.CallbackResp, timeout int, queryParams map[string]string) error { | ||||
| 	ctx = mcontext.WithMustInfoCtx([]string{mcontext.GetOperationID(ctx), mcontext.GetOpUserID(ctx), mcontext.GetOpUserPlatform(ctx), mcontext.GetConnID(ctx)}) | ||||
| 	fullURL := c.url + "/" + command | ||||
| 
 | ||||
| 	parsedURL, err := url.Parse(fullURL) | ||||
| 	if err != nil { | ||||
| 		return servererrs.ErrNetwork.WrapMsg(err.Error(), "failed to parse URL", fullURL) | ||||
| 	} | ||||
| 
 | ||||
| 	query := parsedURL.Query() | ||||
| 
 | ||||
| 	operationID, _ := ctx.Value(constant.OperationID).(string) | ||||
| 
 | ||||
| 	for key, value := range queryParams { | ||||
| 		query.Set(key, value) | ||||
| 	} | ||||
| 
 | ||||
| 	parsedURL.RawQuery = query.Encode() | ||||
| 
 | ||||
| 	fullURL = parsedURL.String() | ||||
| 	log.ZInfo(ctx, "webhook", "url", fullURL, "input", input, "config", timeout) | ||||
| 
 | ||||
| 	b, err := c.client.Post(ctx, fullURL, map[string]string{constant.OperationID: operationID}, input, timeout) | ||||
| 	if err != nil { | ||||
| 		return servererrs.ErrNetwork.WrapMsg(err.Error(), "post url", fullURL) | ||||
| 	} | ||||
| 
 | ||||
| 	if err = json.Unmarshal(b, output); err != nil { | ||||
| 		return servererrs.ErrData.WithDetail(err.Error() + " response format error") | ||||
| 	} | ||||
| 	if err := output.Parse(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	log.ZInfo(ctx, "webhook success", "url", fullURL, "input", input, "response", string(b)) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user