2023-08-28 04:13:14 +08:00
|
|
|
package broker
|
|
|
|
|
|
|
|
import (
|
2023-12-12 04:05:54 +08:00
|
|
|
"context"
|
2023-08-28 08:50:59 +08:00
|
|
|
"fmt"
|
2023-08-28 04:13:14 +08:00
|
|
|
"github.com/seaweedfs/seaweedfs/weed/glog"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/mq/topic"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/pb/mq_pb"
|
2024-01-08 16:03:08 +08:00
|
|
|
"github.com/seaweedfs/seaweedfs/weed/util/log_buffer"
|
2023-08-28 04:13:14 +08:00
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2024-01-17 16:15:52 +08:00
|
|
|
func (b *MessageQueueBroker) SubscribeMessage(req *mq_pb.SubscribeMessageRequest, stream mq_pb.SeaweedMessaging_SubscribeMessageServer) (err error) {
|
2023-08-28 04:13:14 +08:00
|
|
|
|
2024-01-12 15:08:02 +08:00
|
|
|
ctx := stream.Context()
|
|
|
|
clientName := fmt.Sprintf("%s/%s-%s", req.GetInit().ConsumerGroup, req.GetInit().ConsumerId, req.GetInit().ClientId)
|
|
|
|
|
2024-03-25 03:57:09 +08:00
|
|
|
initMessage := req.GetInit()
|
|
|
|
if initMessage == nil {
|
|
|
|
glog.Errorf("missing init message")
|
|
|
|
return fmt.Errorf("missing init message")
|
|
|
|
}
|
|
|
|
|
2023-12-12 04:05:54 +08:00
|
|
|
t := topic.FromPbTopic(req.GetInit().Topic)
|
2024-01-06 07:24:14 +08:00
|
|
|
partition := topic.FromPbPartition(req.GetInit().GetPartitionOffset().GetPartition())
|
2024-01-12 15:08:02 +08:00
|
|
|
|
|
|
|
glog.V(0).Infof("Subscriber %s on %v %v connected", req.GetInit().ConsumerId, t, partition)
|
|
|
|
|
2024-03-25 04:04:59 +08:00
|
|
|
localTopicPartition, getOrGenErr := b.GetOrGenerateLocalPartition(t, partition)
|
|
|
|
if getOrGenErr != nil {
|
|
|
|
return getOrGenErr
|
2023-08-28 04:13:14 +08:00
|
|
|
}
|
|
|
|
|
2023-12-12 04:05:54 +08:00
|
|
|
localTopicPartition.Subscribers.AddSubscriber(clientName, topic.NewLocalSubscriber())
|
|
|
|
glog.V(0).Infof("Subscriber %s connected on %v %v", clientName, t, partition)
|
|
|
|
isConnected := true
|
|
|
|
sleepIntervalCount := 0
|
2024-01-15 16:20:12 +08:00
|
|
|
|
|
|
|
var counter int64
|
2023-12-12 04:05:54 +08:00
|
|
|
defer func() {
|
|
|
|
isConnected = false
|
|
|
|
localTopicPartition.Subscribers.RemoveSubscriber(clientName)
|
2024-01-15 16:20:12 +08:00
|
|
|
glog.V(0).Infof("Subscriber %s on %v %v disconnected, sent %d", clientName, t, partition, counter)
|
2024-01-17 00:43:07 +08:00
|
|
|
if localTopicPartition.MaybeShutdownLocalPartition() {
|
|
|
|
b.localTopicManager.RemoveTopicPartition(t, partition)
|
|
|
|
}
|
2023-12-12 04:05:54 +08:00
|
|
|
}()
|
|
|
|
|
2024-01-08 16:03:08 +08:00
|
|
|
var startPosition log_buffer.MessagePosition
|
2024-03-01 01:38:52 +08:00
|
|
|
if req.GetInit() != nil && req.GetInit().GetPartitionOffset() != nil {
|
2024-03-11 05:34:28 +08:00
|
|
|
startPosition = getRequestPosition(req.GetInit().GetPartitionOffset())
|
2023-12-12 04:05:54 +08:00
|
|
|
}
|
|
|
|
|
2024-01-15 16:20:12 +08:00
|
|
|
return localTopicPartition.Subscribe(clientName, startPosition, func() bool {
|
2023-12-12 04:05:54 +08:00
|
|
|
if !isConnected {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
sleepIntervalCount++
|
2024-03-11 05:34:28 +08:00
|
|
|
if sleepIntervalCount > 32 {
|
|
|
|
sleepIntervalCount = 32
|
2023-12-12 04:05:54 +08:00
|
|
|
}
|
2024-03-11 05:34:28 +08:00
|
|
|
time.Sleep(time.Duration(sleepIntervalCount) * 137 * time.Millisecond)
|
2023-12-12 04:05:54 +08:00
|
|
|
|
|
|
|
// Check if the client has disconnected by monitoring the context
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
err := ctx.Err()
|
|
|
|
if err == context.Canceled {
|
|
|
|
// Client disconnected
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
glog.V(0).Infof("Subscriber %s disconnected: %v", clientName, err)
|
|
|
|
return false
|
|
|
|
default:
|
|
|
|
// Continue processing the request
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
2024-03-08 02:50:09 +08:00
|
|
|
}, func(logEntry *filer_pb.LogEntry) (bool, error) {
|
2023-12-12 04:05:54 +08:00
|
|
|
// reset the sleep interval count
|
|
|
|
sleepIntervalCount = 0
|
2023-09-01 15:36:51 +08:00
|
|
|
|
2024-01-06 07:14:25 +08:00
|
|
|
if err := stream.Send(&mq_pb.SubscribeMessageResponse{Message: &mq_pb.SubscribeMessageResponse_Data{
|
2023-08-28 04:13:14 +08:00
|
|
|
Data: &mq_pb.DataMessage{
|
2024-03-08 02:53:30 +08:00
|
|
|
Key: logEntry.Key,
|
|
|
|
Value: logEntry.Data,
|
2023-12-12 04:05:54 +08:00
|
|
|
TsNs: logEntry.TsNs,
|
2023-08-28 04:13:14 +08:00
|
|
|
},
|
2024-03-11 05:34:28 +08:00
|
|
|
}}); err != nil {
|
|
|
|
glog.Errorf("Error sending data: %v", err)
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
counter++
|
|
|
|
return false, nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-03-25 04:04:59 +08:00
|
|
|
func (b *MessageQueueBroker) GetOrGenerateLocalPartition(t topic.Topic, partition topic.Partition) (localTopicPartition *topic.LocalPartition, getOrGenError error) {
|
|
|
|
// get or generate a local partition
|
|
|
|
conf, readConfErr := b.readTopicConfFromFiler(t)
|
|
|
|
if readConfErr != nil {
|
|
|
|
glog.Errorf("topic %v not found: %v", t, readConfErr)
|
|
|
|
return nil, fmt.Errorf("topic %v not found: %v", t, readConfErr)
|
|
|
|
}
|
|
|
|
localTopicPartition, _, getOrGenError = b.doGetOrGenLocalPartition(t, partition, conf)
|
|
|
|
if getOrGenError != nil {
|
|
|
|
glog.Errorf("topic %v partition %v not setup: %v", t, partition, getOrGenError)
|
|
|
|
return nil, fmt.Errorf("topic %v partition %v not setup: %v", t, partition, getOrGenError)
|
|
|
|
}
|
|
|
|
return localTopicPartition, nil
|
|
|
|
}
|
|
|
|
|
2024-03-11 05:34:28 +08:00
|
|
|
func getRequestPosition(offset *mq_pb.PartitionOffset) (startPosition log_buffer.MessagePosition) {
|
|
|
|
if offset.StartTsNs != 0 {
|
|
|
|
startPosition = log_buffer.NewMessagePosition(offset.StartTsNs, -2)
|
|
|
|
}
|
|
|
|
if offset.StartType == mq_pb.PartitionOffsetStartType_EARLIEST {
|
|
|
|
startPosition = log_buffer.NewMessagePosition(1, -3)
|
|
|
|
} else if offset.StartType == mq_pb.PartitionOffsetStartType_LATEST {
|
|
|
|
startPosition = log_buffer.NewMessagePosition(time.Now().UnixNano(), -4)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|