123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117 |
- package topic
- import (
- cmap "github.com/orcaman/concurrent-map/v2"
- "github.com/seaweedfs/seaweedfs/weed/pb/mq_pb"
- "github.com/shirou/gopsutil/v3/cpu"
- "time"
- )
- // LocalTopicManager manages topics on local broker
- type LocalTopicManager struct {
- topics cmap.ConcurrentMap[string, *LocalTopic]
- }
- // NewLocalTopicManager creates a new LocalTopicManager
- func NewLocalTopicManager() *LocalTopicManager {
- return &LocalTopicManager{
- topics: cmap.New[*LocalTopic](),
- }
- }
- // AddLocalPartition adds a topic to the local topic manager
- func (manager *LocalTopicManager) AddLocalPartition(topic Topic, localPartition *LocalPartition) {
- localTopic, ok := manager.topics.Get(topic.String())
- if !ok {
- localTopic = NewLocalTopic(topic)
- }
- if !manager.topics.SetIfAbsent(topic.String(), localTopic) {
- localTopic, _ = manager.topics.Get(topic.String())
- }
- if localTopic.findPartition(localPartition.Partition) != nil {
- return
- }
- localTopic.Partitions = append(localTopic.Partitions, localPartition)
- }
- // GetLocalPartition gets a topic from the local topic manager
- func (manager *LocalTopicManager) GetLocalPartition(topic Topic, partition Partition) *LocalPartition {
- localTopic, ok := manager.topics.Get(topic.String())
- if !ok {
- return nil
- }
- return localTopic.findPartition(partition)
- }
- // RemoveTopic removes a topic from the local topic manager
- func (manager *LocalTopicManager) RemoveTopic(topic Topic) {
- manager.topics.Remove(topic.String())
- }
- func (manager *LocalTopicManager) RemoveLocalPartition(topic Topic, partition Partition) (removed bool) {
- localTopic, ok := manager.topics.Get(topic.String())
- if !ok {
- return false
- }
- return localTopic.removePartition(partition)
- }
- func (manager *LocalTopicManager) ClosePublishers(topic Topic, unixTsNs int64) (removed bool) {
- localTopic, ok := manager.topics.Get(topic.String())
- if !ok {
- return false
- }
- return localTopic.closePartitionPublishers(unixTsNs)
- }
- func (manager *LocalTopicManager) CloseSubscribers(topic Topic, unixTsNs int64) (removed bool) {
- localTopic, ok := manager.topics.Get(topic.String())
- if !ok {
- return false
- }
- return localTopic.closePartitionSubscribers(unixTsNs)
- }
- func (manager *LocalTopicManager) CollectStats(duration time.Duration) *mq_pb.BrokerStats {
- stats := &mq_pb.BrokerStats{
- Stats: make(map[string]*mq_pb.TopicPartitionStats),
- }
- // collect current broker's cpu usage
- // this needs to be in front, so the following stats can be more accurate
- usages, err := cpu.Percent(duration, false)
- if err == nil && len(usages) > 0 {
- stats.CpuUsagePercent = int32(usages[0])
- }
- // collect current broker's topics and partitions
- manager.topics.IterCb(func(topic string, localTopic *LocalTopic) {
- for _, localPartition := range localTopic.Partitions {
- topicPartition := &TopicPartition{
- Topic: Topic{Namespace: localTopic.Namespace, Name: localTopic.Name},
- Partition: localPartition.Partition,
- }
- stats.Stats[topicPartition.String()] = &mq_pb.TopicPartitionStats{
- Topic: &mq_pb.Topic{
- Namespace: string(localTopic.Namespace),
- Name: localTopic.Name,
- },
- Partition: localPartition.Partition.ToPbPartition(),
- PublisherCount: int32(localPartition.Publishers.Size()),
- SubscriberCount: int32(localPartition.Subscribers.Size()),
- }
- // fmt.Printf("collect topic %+v partition %+v\n", topicPartition, localPartition.Partition)
- }
- })
- return stats
- }
- func (manager *LocalTopicManager) WaitUntilNoPublishers(topic Topic) {
- localTopic, ok := manager.topics.Get(topic.String())
- if !ok {
- return
- }
- localTopic.WaitUntilNoPublishers()
- }
|