seaweedfs/weed/mq/msgclient/publisher.go

119 lines
2.7 KiB
Go
Raw Normal View History

2020-05-05 09:05:28 +00:00
package msgclient
import (
"context"
"github.com/OneOfOne/xxhash"
2020-05-08 09:47:22 +00:00
"google.golang.org/grpc"
2022-07-02 05:43:25 +00:00
"github.com/chrislusf/seaweedfs/weed/mq/broker"
"github.com/chrislusf/seaweedfs/weed/pb/mq_pb"
)
type Publisher struct {
2022-07-02 05:43:25 +00:00
publishClients []mq_pb.SeaweedMessaging_PublishClient
topicConfiguration *mq_pb.TopicConfiguration
messageCount uint64
publisherId string
}
2020-05-10 10:50:30 +00:00
func (mc *MessagingClient) NewPublisher(publisherId, namespace, topic string) (*Publisher, error) {
// read topic configuration
2022-07-02 05:43:25 +00:00
topicConfiguration := &mq_pb.TopicConfiguration{
PartitionCount: 4,
}
2022-07-02 05:43:25 +00:00
publishClients := make([]mq_pb.SeaweedMessaging_PublishClient, topicConfiguration.PartitionCount)
for i := 0; i < int(topicConfiguration.PartitionCount); i++ {
2020-05-11 06:45:48 +00:00
tp := broker.TopicPartition{
2020-05-08 09:47:22 +00:00
Namespace: namespace,
2020-06-05 22:27:10 +00:00
Topic: topic,
2020-05-08 09:47:22 +00:00
Partition: int32(i),
2020-05-11 06:45:48 +00:00
}
grpcClientConn, err := mc.findBroker(tp)
if err != nil {
return nil, err
}
client, err := setupPublisherClient(grpcClientConn, tp)
if err != nil {
return nil, err
}
publishClients[i] = client
}
return &Publisher{
publishClients: publishClients,
topicConfiguration: topicConfiguration,
}, nil
}
2022-07-02 05:43:25 +00:00
func setupPublisherClient(grpcConnection *grpc.ClientConn, tp broker.TopicPartition) (mq_pb.SeaweedMessaging_PublishClient, error) {
2022-07-02 05:43:25 +00:00
stream, err := mq_pb.NewSeaweedMessagingClient(grpcConnection).Publish(context.Background())
if err != nil {
return nil, err
}
// send init message
2022-07-02 05:43:25 +00:00
err = stream.Send(&mq_pb.PublishRequest{
Init: &mq_pb.PublishRequest_InitMessage{
2020-05-05 09:05:28 +00:00
Namespace: tp.Namespace,
Topic: tp.Topic,
Partition: tp.Partition,
},
})
if err != nil {
return nil, err
}
// process init response
initResponse, err := stream.Recv()
if err != nil {
return nil, err
}
if initResponse.Redirect != nil {
// TODO follow redirection
}
if initResponse.Config != nil {
}
// setup looks for control messages
doneChan := make(chan error, 1)
go func() {
for {
in, err := stream.Recv()
if err != nil {
doneChan <- err
return
}
2020-04-21 00:48:06 +00:00
if in.Redirect != nil {
}
2020-04-21 00:48:06 +00:00
if in.Config != nil {
}
}
}()
return stream, nil
}
2022-07-02 05:43:25 +00:00
func (p *Publisher) Publish(m *mq_pb.Message) error {
hashValue := p.messageCount
p.messageCount++
2022-07-02 05:43:25 +00:00
if p.topicConfiguration.Partitoning == mq_pb.TopicConfiguration_NonNullKeyHash {
if m.Key != nil {
hashValue = xxhash.Checksum64(m.Key)
}
2022-07-02 05:43:25 +00:00
} else if p.topicConfiguration.Partitoning == mq_pb.TopicConfiguration_KeyHash {
hashValue = xxhash.Checksum64(m.Key)
} else {
// round robin
}
idx := int(hashValue) % len(p.publishClients)
if idx < 0 {
idx += len(p.publishClients)
}
2022-07-02 05:43:25 +00:00
return p.publishClients[idx].Send(&mq_pb.PublishRequest{
Data: m,
})
}