mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-25 04:32:10 +08:00 
			
		
		
		
	feat: support server-issued configuration, which can be set for individual users (#3271)
* pb * fix: Modifying other fields while setting IsPrivateChat does not take effect * fix: quote message error revoke * refactoring scheduled tasks * refactoring scheduled tasks * refactoring scheduled tasks * refactoring scheduled tasks * refactoring scheduled tasks * refactoring scheduled tasks * upgrading pkg tools * fix * fix * optimize log output * feat: support GetLastMessage * feat: support GetLastMessage * feat: s3 switch * feat: s3 switch * fix: GetUsersOnline * feat: SendBusinessNotification supported configuration parameters * feat: SendBusinessNotification supported configuration parameters * feat: SendBusinessNotification supported configuration parameters * feat: seq conversion failed without exiting * fix: DeleteDoc crash * fix: fill send time * fix: fill send time * fix: crash caused by withdrawing messages from users who have left the group * fix: user msg timestamp * seq read config * seq read config * fix: the source message of the reference is withdrawn, and the referenced message is deleted * feat: optimize the default notification.yml * fix: shouldPushOffline * fix: the sorting is wrong after canceling the administrator in group settings * feat: Sending messages supports returning fields modified by webhook * feat: Sending messages supports returning fields modified by webhook * feat: Sending messages supports returning fields modified by webhook * fix: oss specifies content-type when uploading * fix: the version number contains a line break * fix: the version number contains a line break * feat: support client config * feat: support client config
This commit is contained in:
		
							parent
							
								
									c29e2a9a28
								
							
						
					
					
						commit
						f74bd018d2
					
				
							
								
								
									
										6
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								go.mod
									
									
									
									
									
								
							| @ -12,8 +12,8 @@ require ( | ||||
| 	github.com/gorilla/websocket v1.5.1 | ||||
| 	github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 | ||||
| 	github.com/mitchellh/mapstructure v1.5.0 | ||||
| 	github.com/openimsdk/protocol v0.0.72-alpha.79 | ||||
| 	github.com/openimsdk/tools v0.0.50-alpha.74 | ||||
| 	github.com/openimsdk/protocol v0.0.73-alpha.3 | ||||
| 	github.com/openimsdk/tools v0.0.50-alpha.79 | ||||
| 	github.com/pkg/errors v0.9.1 // indirect | ||||
| 	github.com/prometheus/client_golang v1.18.0 | ||||
| 	github.com/stretchr/testify v1.9.0 | ||||
| @ -219,3 +219,5 @@ require ( | ||||
| 	golang.org/x/crypto v0.27.0 // indirect | ||||
| 	gopkg.in/ini.v1 v1.67.0 // indirect | ||||
| ) | ||||
| 
 | ||||
| //replace github.com/openimsdk/protocol => /Users/chao/Desktop/code/protocol | ||||
|  | ||||
							
								
								
									
										12
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								go.sum
									
									
									
									
									
								
							| @ -345,12 +345,12 @@ github.com/onsi/ginkgo/v2 v2.19.0 h1:9Cnnf7UHo57Hy3k6/m5k3dRfGTMXGvxhHFvkDTCTpvA | ||||
| github.com/onsi/ginkgo/v2 v2.19.0/go.mod h1:rlwLi9PilAFJ8jCg9UE1QP6VBpd6/xj3SRC0d6TU0To= | ||||
| github.com/onsi/gomega v1.25.0 h1:Vw7br2PCDYijJHSfBOWhov+8cAnUf8MfMaIOV323l6Y= | ||||
| github.com/onsi/gomega v1.25.0/go.mod h1:r+zV744Re+DiYCIPRlYOTxn0YkOLcAnW8k1xXdMPGhM= | ||||
| github.com/openimsdk/gomake v0.0.14-alpha.5 h1:VY9c5x515lTfmdhhPjMvR3BBRrRquAUCFsz7t7vbv7Y= | ||||
| github.com/openimsdk/gomake v0.0.14-alpha.5/go.mod h1:PndCozNc2IsQIciyn9mvEblYWZwJmAI+06z94EY+csI= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.79 h1:e46no8WVAsmTzyy405klrdoUiG7u+1ohDsXvQuFng4s= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.79/go.mod h1:WF7EuE55vQvpyUAzDXcqg+B+446xQyEba0X35lTINmw= | ||||
| github.com/openimsdk/tools v0.0.50-alpha.74 h1:yh10SiMiivMEjicEQg+QAsH4pvaO+4noMPdlw+ew0Kc= | ||||
| github.com/openimsdk/tools v0.0.50-alpha.74/go.mod h1:n2poR3asX1e1XZce4O+MOWAp+X02QJRFvhcLCXZdzRo= | ||||
| github.com/openimsdk/gomake v0.0.15-alpha.2 h1:5Q8yl8ezy2yx+q8/ucU/t4kJnDfCzNOrkXcDACCqtyM= | ||||
| github.com/openimsdk/gomake v0.0.15-alpha.2/go.mod h1:PndCozNc2IsQIciyn9mvEblYWZwJmAI+06z94EY+csI= | ||||
| github.com/openimsdk/protocol v0.0.73-alpha.3 h1:mf/REUZA5in2gk8ggwqJD8444xLvB7WlF7M97oXN78g= | ||||
| github.com/openimsdk/protocol v0.0.73-alpha.3/go.mod h1:WF7EuE55vQvpyUAzDXcqg+B+446xQyEba0X35lTINmw= | ||||
| github.com/openimsdk/tools v0.0.50-alpha.79 h1:jxYEbrzaze4Z2r4NrKad816buZ690ix0L9MTOOOH3ik= | ||||
| github.com/openimsdk/tools v0.0.50-alpha.79/go.mod h1:n2poR3asX1e1XZce4O+MOWAp+X02QJRFvhcLCXZdzRo= | ||||
| github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= | ||||
| github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs= | ||||
| github.com/pierrec/lz4/v4 v4.1.21 h1:yOVMLb6qSIDP67pl/5F7RepeKYu/VmTyEXvuMI5d9mQ= | ||||
|  | ||||
| @ -124,6 +124,11 @@ func newGinRouter(ctx context.Context, client discovery.SvcDiscoveryRegistry, cf | ||||
| 		userRouterGroup.POST("/add_notification_account", u.AddNotificationAccount) | ||||
| 		userRouterGroup.POST("/update_notification_account", u.UpdateNotificationAccountInfo) | ||||
| 		userRouterGroup.POST("/search_notification_account", u.SearchNotificationAccount) | ||||
| 
 | ||||
| 		userRouterGroup.POST("/get_user_client_config", u.GetUserClientConfig) | ||||
| 		userRouterGroup.POST("/set_user_client_config", u.SetUserClientConfig) | ||||
| 		userRouterGroup.POST("/del_user_client_config", u.DelUserClientConfig) | ||||
| 		userRouterGroup.POST("/page_user_client_config", u.PageUserClientConfig) | ||||
| 	} | ||||
| 	// friend routing group | ||||
| 	{ | ||||
|  | ||||
| @ -242,3 +242,19 @@ func (u *UserApi) UpdateNotificationAccountInfo(c *gin.Context) { | ||||
| func (u *UserApi) SearchNotificationAccount(c *gin.Context) { | ||||
| 	a2r.Call(c, user.UserClient.SearchNotificationAccount, u.Client) | ||||
| } | ||||
| 
 | ||||
| func (u *UserApi) GetUserClientConfig(c *gin.Context) { | ||||
| 	a2r.Call(c, user.UserClient.GetUserClientConfig, u.Client) | ||||
| } | ||||
| 
 | ||||
| func (u *UserApi) SetUserClientConfig(c *gin.Context) { | ||||
| 	a2r.Call(c, user.UserClient.SetUserClientConfig, u.Client) | ||||
| } | ||||
| 
 | ||||
| func (u *UserApi) DelUserClientConfig(c *gin.Context) { | ||||
| 	a2r.Call(c, user.UserClient.DelUserClientConfig, u.Client) | ||||
| } | ||||
| 
 | ||||
| func (u *UserApi) PageUserClientConfig(c *gin.Context) { | ||||
| 	a2r.Call(c, user.UserClient.PageUserClientConfig, u.Client) | ||||
| } | ||||
|  | ||||
							
								
								
									
										71
									
								
								internal/rpc/user/config.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								internal/rpc/user/config.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,71 @@ | ||||
| package user | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	pbuser "github.com/openimsdk/protocol/user" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| ) | ||||
| 
 | ||||
| func (s *userServer) GetUserClientConfig(ctx context.Context, req *pbuser.GetUserClientConfigReq) (*pbuser.GetUserClientConfigResp, error) { | ||||
| 	if req.UserID != "" { | ||||
| 		if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		if _, err := s.db.GetUserByID(ctx, req.UserID); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	res, err := s.clientConfig.GetUserConfig(ctx, req.UserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbuser.GetUserClientConfigResp{Configs: res}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *userServer) SetUserClientConfig(ctx context.Context, req *pbuser.SetUserClientConfigReq) (*pbuser.SetUserClientConfigResp, error) { | ||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if req.UserID != "" { | ||||
| 		if _, err := s.db.GetUserByID(ctx, req.UserID); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	if err := s.clientConfig.SetUserConfig(ctx, req.UserID, req.Configs); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbuser.SetUserClientConfigResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *userServer) DelUserClientConfig(ctx context.Context, req *pbuser.DelUserClientConfigReq) (*pbuser.DelUserClientConfigResp, error) { | ||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if err := s.clientConfig.DelUserConfig(ctx, req.UserID, req.Keys); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbuser.DelUserClientConfigResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *userServer) PageUserClientConfig(ctx context.Context, req *pbuser.PageUserClientConfigReq) (*pbuser.PageUserClientConfigResp, error) { | ||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	total, res, err := s.clientConfig.GetUserConfigPage(ctx, req.UserID, req.Key, req.Pagination) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbuser.PageUserClientConfigResp{ | ||||
| 		Total: total, | ||||
| 		Configs: datautil.Slice(res, func(e *model.ClientConfig) *pbuser.ClientConfig { | ||||
| 			return &pbuser.ClientConfig{ | ||||
| 				UserID: e.UserID, | ||||
| 				Key:    e.Key, | ||||
| 				Value:  e.Value, | ||||
| 			} | ||||
| 		}), | ||||
| 	}, nil | ||||
| } | ||||
| @ -62,6 +62,7 @@ type userServer struct { | ||||
| 	webhookClient            *webhook.Client | ||||
| 	groupClient              *rpcli.GroupClient | ||||
| 	relationClient           *rpcli.RelationClient | ||||
| 	clientConfig             controller.ClientConfigDatabase | ||||
| } | ||||
| 
 | ||||
| type Config struct { | ||||
| @ -94,6 +95,10 @@ func Start(ctx context.Context, config *Config, client registry.SvcDiscoveryRegi | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	clientConfigDB, err := mgo.NewClientConfig(mgocli.GetDB()) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	msgConn, err := client.GetConn(ctx, config.Discovery.RpcService.Msg) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| @ -118,9 +123,9 @@ func Start(ctx context.Context, config *Config, client registry.SvcDiscoveryRegi | ||||
| 		userNotificationSender:   NewUserNotificationSender(config, msgClient, WithUserFunc(database.FindWithError)), | ||||
| 		config:                   config, | ||||
| 		webhookClient:            webhook.NewWebhookClient(config.WebhooksConfig.URL), | ||||
| 
 | ||||
| 		groupClient:    rpcli.NewGroupClient(groupConn), | ||||
| 		relationClient: rpcli.NewRelationClient(friendConn), | ||||
| 		clientConfig:             controller.NewClientConfigDatabase(clientConfigDB, redis.NewClientConfigCache(rdb, clientConfigDB), mgocli.GetTx()), | ||||
| 		groupClient:              rpcli.NewGroupClient(groupConn), | ||||
| 		relationClient:           rpcli.NewRelationClient(friendConn), | ||||
| 	} | ||||
| 	pbuser.RegisterUserServer(server, u) | ||||
| 	return u.db.InitOnce(context.Background(), users) | ||||
|  | ||||
							
								
								
									
										10
									
								
								pkg/common/storage/cache/cachekey/client_config.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								pkg/common/storage/cache/cachekey/client_config.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | ||||
| package cachekey | ||||
| 
 | ||||
| const ClientConfig = "CLIENT_CONFIG" | ||||
| 
 | ||||
| func GetClientConfigKey(userID string) string { | ||||
| 	if userID == "" { | ||||
| 		return ClientConfig | ||||
| 	} | ||||
| 	return ClientConfig + ":" + userID | ||||
| } | ||||
							
								
								
									
										8
									
								
								pkg/common/storage/cache/client_config.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								pkg/common/storage/cache/client_config.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | ||||
| package cache | ||||
| 
 | ||||
| import "context" | ||||
| 
 | ||||
| type ClientConfigCache interface { | ||||
| 	DeleteUserCache(ctx context.Context, userIDs []string) error | ||||
| 	GetUserConfig(ctx context.Context, userID string) (map[string]string, error) | ||||
| } | ||||
							
								
								
									
										69
									
								
								pkg/common/storage/cache/redis/client_config.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								pkg/common/storage/cache/redis/client_config.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,69 @@ | ||||
| package redis | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/cachekey" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
| 
 | ||||
| func NewClientConfigCache(rdb redis.UniversalClient, mgo database.ClientConfig) cache.ClientConfigCache { | ||||
| 	rc := newRocksCacheClient(rdb) | ||||
| 	return &ClientConfigCache{ | ||||
| 		mgo:      mgo, | ||||
| 		rcClient: rc, | ||||
| 		delete:   rc.GetBatchDeleter(), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| type ClientConfigCache struct { | ||||
| 	mgo      database.ClientConfig | ||||
| 	rcClient *rocksCacheClient | ||||
| 	delete   cache.BatchDeleter | ||||
| } | ||||
| 
 | ||||
| func (x *ClientConfigCache) getExpireTime(userID string) time.Duration { | ||||
| 	if userID == "" { | ||||
| 		return time.Hour * 24 | ||||
| 	} else { | ||||
| 		return time.Hour | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (x *ClientConfigCache) getClientConfigKey(userID string) string { | ||||
| 	return cachekey.GetClientConfigKey(userID) | ||||
| } | ||||
| 
 | ||||
| func (x *ClientConfigCache) GetConfig(ctx context.Context, userID string) (map[string]string, error) { | ||||
| 	return getCache(ctx, x.rcClient, x.getClientConfigKey(userID), x.getExpireTime(userID), func(ctx context.Context) (map[string]string, error) { | ||||
| 		return x.mgo.Get(ctx, userID) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (x *ClientConfigCache) DeleteUserCache(ctx context.Context, userIDs []string) error { | ||||
| 	keys := make([]string, 0, len(userIDs)) | ||||
| 	for _, userID := range userIDs { | ||||
| 		keys = append(keys, x.getClientConfigKey(userID)) | ||||
| 	} | ||||
| 	return x.delete.ExecDelWithKeys(ctx, keys) | ||||
| } | ||||
| 
 | ||||
| func (x *ClientConfigCache) GetUserConfig(ctx context.Context, userID string) (map[string]string, error) { | ||||
| 	config, err := x.GetConfig(ctx, "") | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if userID != "" { | ||||
| 		userConfig, err := x.GetConfig(ctx, userID) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		for k, v := range userConfig { | ||||
| 			config[k] = v | ||||
| 		} | ||||
| 	} | ||||
| 	return config, nil | ||||
| } | ||||
							
								
								
									
										58
									
								
								pkg/common/storage/controller/client_config.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								pkg/common/storage/controller/client_config.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | ||||
| package controller | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||
| 	"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/tools/db/pagination" | ||||
| 	"github.com/openimsdk/tools/db/tx" | ||||
| ) | ||||
| 
 | ||||
| type ClientConfigDatabase interface { | ||||
| 	SetUserConfig(ctx context.Context, userID string, config map[string]string) error | ||||
| 	GetUserConfig(ctx context.Context, userID string) (map[string]string, error) | ||||
| 	DelUserConfig(ctx context.Context, userID string, keys []string) error | ||||
| 	GetUserConfigPage(ctx context.Context, userID string, key string, pagination pagination.Pagination) (int64, []*model.ClientConfig, error) | ||||
| } | ||||
| 
 | ||||
| func NewClientConfigDatabase(db database.ClientConfig, cache cache.ClientConfigCache, tx tx.Tx) ClientConfigDatabase { | ||||
| 	return &clientConfigDatabase{ | ||||
| 		tx:    tx, | ||||
| 		db:    db, | ||||
| 		cache: cache, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| type clientConfigDatabase struct { | ||||
| 	tx    tx.Tx | ||||
| 	db    database.ClientConfig | ||||
| 	cache cache.ClientConfigCache | ||||
| } | ||||
| 
 | ||||
| func (x *clientConfigDatabase) SetUserConfig(ctx context.Context, userID string, config map[string]string) error { | ||||
| 	return x.tx.Transaction(ctx, func(ctx context.Context) error { | ||||
| 		if err := x.db.Set(ctx, userID, config); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		return x.cache.DeleteUserCache(ctx, []string{userID}) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (x *clientConfigDatabase) GetUserConfig(ctx context.Context, userID string) (map[string]string, error) { | ||||
| 	return x.cache.GetUserConfig(ctx, userID) | ||||
| } | ||||
| 
 | ||||
| func (x *clientConfigDatabase) DelUserConfig(ctx context.Context, userID string, keys []string) error { | ||||
| 	return x.tx.Transaction(ctx, func(ctx context.Context) error { | ||||
| 		if err := x.db.Del(ctx, userID, keys); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		return x.cache.DeleteUserCache(ctx, []string{userID}) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (x *clientConfigDatabase) GetUserConfigPage(ctx context.Context, userID string, key string, pagination pagination.Pagination) (int64, []*model.ClientConfig, error) { | ||||
| 	return x.db.GetPage(ctx, userID, key, pagination) | ||||
| } | ||||
							
								
								
									
										15
									
								
								pkg/common/storage/database/client_config.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								pkg/common/storage/database/client_config.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,15 @@ | ||||
| package database | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"github.com/openimsdk/tools/db/pagination" | ||||
| ) | ||||
| 
 | ||||
| type ClientConfig interface { | ||||
| 	Set(ctx context.Context, userID string, config map[string]string) error | ||||
| 	Get(ctx context.Context, userID string) (map[string]string, error) | ||||
| 	Del(ctx context.Context, userID string, keys []string) error | ||||
| 	GetPage(ctx context.Context, userID string, key string, pagination pagination.Pagination) (int64, []*model.ClientConfig, error) | ||||
| } | ||||
							
								
								
									
										99
									
								
								pkg/common/storage/database/mgo/client_config.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								pkg/common/storage/database/mgo/client_config.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,99 @@ | ||||
| // Copyright © 2023 OpenIM open source community. All rights reserved. | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //     http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package mgo | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"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/tools/db/mongoutil" | ||||
| 	"github.com/openimsdk/tools/db/pagination" | ||||
| 	"go.mongodb.org/mongo-driver/bson" | ||||
| 	"go.mongodb.org/mongo-driver/mongo" | ||||
| 	"go.mongodb.org/mongo-driver/mongo/options" | ||||
| 
 | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| ) | ||||
| 
 | ||||
| func NewClientConfig(db *mongo.Database) (database.ClientConfig, error) { | ||||
| 	coll := db.Collection("config") | ||||
| 	_, err := coll.Indexes().CreateMany(context.Background(), []mongo.IndexModel{ | ||||
| 		{ | ||||
| 			Keys: bson.D{ | ||||
| 				{Key: "key", Value: 1}, | ||||
| 				{Key: "user_id", Value: 1}, | ||||
| 			}, | ||||
| 			Options: options.Index().SetUnique(true), | ||||
| 		}, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, errs.Wrap(err) | ||||
| 	} | ||||
| 	return &ClientConfig{ | ||||
| 		coll: coll, | ||||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
| type ClientConfig struct { | ||||
| 	coll *mongo.Collection | ||||
| } | ||||
| 
 | ||||
| func (x *ClientConfig) Set(ctx context.Context, userID string, config map[string]string) error { | ||||
| 	if len(config) == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
| 	for key, value := range config { | ||||
| 		filter := bson.M{"key": key, "user_id": userID} | ||||
| 		update := bson.M{ | ||||
| 			"value": value, | ||||
| 		} | ||||
| 		err := mongoutil.UpdateOne(ctx, x.coll, filter, bson.M{"$set": update}, false, options.Update().SetUpsert(true)) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (x *ClientConfig) Get(ctx context.Context, userID string) (map[string]string, error) { | ||||
| 	cs, err := mongoutil.Find[*model.ClientConfig](ctx, x.coll, bson.M{"user_id": userID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	cm := make(map[string]string) | ||||
| 	for _, config := range cs { | ||||
| 		cm[config.Key] = config.Value | ||||
| 	} | ||||
| 	return cm, nil | ||||
| } | ||||
| 
 | ||||
| func (x *ClientConfig) Del(ctx context.Context, userID string, keys []string) error { | ||||
| 	if len(keys) == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return mongoutil.DeleteMany(ctx, x.coll, bson.M{"key": bson.M{"$in": keys}, "user_id": userID}) | ||||
| } | ||||
| 
 | ||||
| func (x *ClientConfig) GetPage(ctx context.Context, userID string, key string, pagination pagination.Pagination) (int64, []*model.ClientConfig, error) { | ||||
| 	filter := bson.M{} | ||||
| 	if userID != "" { | ||||
| 		filter["user_id"] = userID | ||||
| 	} | ||||
| 	if key != "" { | ||||
| 		filter["key"] = key | ||||
| 	} | ||||
| 	return mongoutil.FindPage[*model.ClientConfig](ctx, x.coll, filter, pagination) | ||||
| } | ||||
							
								
								
									
										7
									
								
								pkg/common/storage/model/client_config.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								pkg/common/storage/model/client_config.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | ||||
| package model | ||||
| 
 | ||||
| type ClientConfig struct { | ||||
| 	Key    string `bson:"key"` | ||||
| 	UserID string `bson:"user_id"` | ||||
| 	Value  string `bson:"value"` | ||||
| } | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user