mirror of
https://github.com/openimsdk/open-im-server.git
synced 2025-04-06 04:15:46 +08:00
Optimize get conversation seq (#2387)
* feat:optimize GetConversationsHasReadAndMaxSeq * fix:get max seqs * fix:get max seqs * fix:get max seqs * fix:get max seqs
This commit is contained in:
parent
88c0d5f5ad
commit
95df4194ca
37
pkg/common/storage/cache/redis/seq.go
vendored
37
pkg/common/storage/cache/redis/seq.go
vendored
@ -16,11 +16,13 @@ package redis
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"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/cachekey"
|
"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/cachekey"
|
||||||
"github.com/openimsdk/tools/errs"
|
"github.com/openimsdk/tools/errs"
|
||||||
"github.com/openimsdk/tools/utils/stringutil"
|
"github.com/openimsdk/tools/utils/stringutil"
|
||||||
"github.com/redis/go-redis/v9"
|
"github.com/redis/go-redis/v9"
|
||||||
|
"sync"
|
||||||
)
|
)
|
||||||
|
|
||||||
func NewSeqCache(rdb redis.UniversalClient) cache.SeqCache {
|
func NewSeqCache(rdb redis.UniversalClient) cache.SeqCache {
|
||||||
@ -61,16 +63,41 @@ func (c *seqCache) getSeq(ctx context.Context, conversationID string, getkey fun
|
|||||||
|
|
||||||
func (c *seqCache) getSeqs(ctx context.Context, items []string, getkey func(s string) string) (m map[string]int64, err error) {
|
func (c *seqCache) getSeqs(ctx context.Context, items []string, getkey func(s string) string) (m map[string]int64, err error) {
|
||||||
m = make(map[string]int64, len(items))
|
m = make(map[string]int64, len(items))
|
||||||
|
var (
|
||||||
|
reverseMap = make(map[string]string, len(items))
|
||||||
|
keys = make([]string, len(items))
|
||||||
|
lock sync.Mutex
|
||||||
|
)
|
||||||
|
|
||||||
for i, v := range items {
|
for i, v := range items {
|
||||||
res, err := c.rdb.Get(ctx, getkey(v)).Result()
|
keys[i] = getkey(v)
|
||||||
if err != nil && err != redis.Nil {
|
reverseMap[getkey(v)] = v
|
||||||
return nil, errs.Wrap(err)
|
|
||||||
}
|
}
|
||||||
val := stringutil.StringToInt64(res)
|
|
||||||
|
manager := NewRedisShardManager(c.rdb)
|
||||||
|
if err = manager.ProcessKeysBySlot(ctx, keys, func(ctx context.Context, _ int64, keys []string) error {
|
||||||
|
res, err := c.rdb.MGet(ctx, keys...).Result()
|
||||||
|
if err != nil && !errors.Is(err, redis.Nil) {
|
||||||
|
return errs.Wrap(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// len(res) <= len(items)
|
||||||
|
for i := range res {
|
||||||
|
strRes, ok := res[i].(string)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
val := stringutil.StringToInt64(strRes)
|
||||||
if val != 0 {
|
if val != 0 {
|
||||||
m[items[i]] = val
|
lock.Lock()
|
||||||
|
m[reverseMap[keys[i]]] = val
|
||||||
|
lock.Unlock()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
|
}); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
|
@ -16,15 +16,19 @@ package rpccache
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/cachekey"
|
|
||||||
|
|
||||||
"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/storage/cache/cachekey"
|
||||||
"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/rpcclient"
|
"github.com/openimsdk/open-im-server/v3/pkg/rpcclient"
|
||||||
pbconversation "github.com/openimsdk/protocol/conversation"
|
pbconversation "github.com/openimsdk/protocol/conversation"
|
||||||
"github.com/openimsdk/tools/errs"
|
"github.com/openimsdk/tools/errs"
|
||||||
"github.com/openimsdk/tools/log"
|
"github.com/openimsdk/tools/log"
|
||||||
"github.com/redis/go-redis/v9"
|
"github.com/redis/go-redis/v9"
|
||||||
|
"golang.org/x/sync/errgroup"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
conversationWorkerCount = 20
|
||||||
)
|
)
|
||||||
|
|
||||||
func NewConversationLocalCache(client rpcclient.ConversationRpcClient, localCache *config.LocalCache, cli redis.UniversalClient) *ConversationLocalCache {
|
func NewConversationLocalCache(client rpcclient.ConversationRpcClient, localCache *config.LocalCache, cli redis.UniversalClient) *ConversationLocalCache {
|
||||||
@ -90,15 +94,33 @@ func (c *ConversationLocalCache) GetSingleConversationRecvMsgOpt(ctx context.Con
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (c *ConversationLocalCache) GetConversations(ctx context.Context, ownerUserID string, conversationIDs []string) ([]*pbconversation.Conversation, error) {
|
func (c *ConversationLocalCache) GetConversations(ctx context.Context, ownerUserID string, conversationIDs []string) ([]*pbconversation.Conversation, error) {
|
||||||
conversations := make([]*pbconversation.Conversation, 0, len(conversationIDs))
|
var (
|
||||||
|
conversations = make([]*pbconversation.Conversation, 0, len(conversationIDs))
|
||||||
|
conversationsChan = make(chan *pbconversation.Conversation, len(conversationIDs))
|
||||||
|
)
|
||||||
|
|
||||||
|
g, ctx := errgroup.WithContext(ctx)
|
||||||
|
g.SetLimit(conversationWorkerCount)
|
||||||
|
|
||||||
for _, conversationID := range conversationIDs {
|
for _, conversationID := range conversationIDs {
|
||||||
|
conversationID := conversationID
|
||||||
|
g.Go(func() error {
|
||||||
conversation, err := c.GetConversation(ctx, ownerUserID, conversationID)
|
conversation, err := c.GetConversation(ctx, ownerUserID, conversationID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if errs.ErrRecordNotFound.Is(err) {
|
if errs.ErrRecordNotFound.Is(err) {
|
||||||
continue
|
return nil
|
||||||
}
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
conversationsChan <- conversation
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
if err := g.Wait(); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
close(conversationsChan)
|
||||||
|
for conversation := range conversationsChan {
|
||||||
conversations = append(conversations, conversation)
|
conversations = append(conversations, conversation)
|
||||||
}
|
}
|
||||||
return conversations, nil
|
return conversations, nil
|
||||||
|
Loading…
x
Reference in New Issue
Block a user