chrislu
1 year ago
2 changed files with 182 additions and 0 deletions
@ -0,0 +1,181 @@ |
|||
package pub_client |
|||
|
|||
import ( |
|||
"context" |
|||
"fmt" |
|||
"github.com/seaweedfs/seaweedfs/weed/glog" |
|||
"github.com/seaweedfs/seaweedfs/weed/pb" |
|||
"github.com/seaweedfs/seaweedfs/weed/pb/mq_pb" |
|||
"sort" |
|||
"sync" |
|||
"time" |
|||
) |
|||
|
|||
type EachPartitionError struct { |
|||
*mq_pb.BrokerPartitionAssignment |
|||
Err error |
|||
generation int |
|||
} |
|||
|
|||
type EachPartitionPublishJob struct { |
|||
*mq_pb.BrokerPartitionAssignment |
|||
stopChan chan bool |
|||
wg sync.WaitGroup |
|||
generation int |
|||
} |
|||
func (p *TopicPublisher) StartSchedulerThread(bootstrapBrokers []string) error { |
|||
|
|||
if err := p.doEnsureConfigureTopic(bootstrapBrokers); err != nil { |
|||
return err |
|||
} |
|||
|
|||
generation := 0 |
|||
var errChan chan EachPartitionError |
|||
for { |
|||
glog.V(0).Infof("lookup partitions gen %d topic %s/%s", generation, p.namespace, p.topic) |
|||
if assignments, err := p.doLookupTopicPartitions(bootstrapBrokers); err == nil { |
|||
generation++ |
|||
glog.V(0).Infof("start generation %d", generation) |
|||
if errChan == nil { |
|||
errChan = make(chan EachPartitionError, len(assignments)) |
|||
} |
|||
p.onEachAssignments(generation, assignments, errChan) |
|||
} else { |
|||
glog.Errorf("lookup topic %s/%s: %v", p.namespace, p.topic, err) |
|||
time.Sleep(5 * time.Second) |
|||
continue |
|||
} |
|||
|
|||
// wait for any error to happen. If so, consume all remaining errors, and retry
|
|||
for { |
|||
select { |
|||
case eachErr := <-errChan: |
|||
glog.Errorf("gen %d publish to topic %s/%s partition %v: %v", eachErr.generation, p.namespace, p.topic, eachErr.Partition, eachErr.Err) |
|||
if eachErr.generation < generation { |
|||
continue |
|||
} |
|||
break |
|||
} |
|||
} |
|||
} |
|||
} |
|||
|
|||
func (p *TopicPublisher) onEachAssignments(generation int, assignments []*mq_pb.BrokerPartitionAssignment, errChan chan EachPartitionError) { |
|||
// TODO assuming this is not re-configured so the partitions are fixed.
|
|||
sort.Slice(assignments, func(i, j int) bool { |
|||
return assignments[i].Partition.RangeStart < assignments[j].Partition.RangeStart |
|||
}) |
|||
var jobs []*EachPartitionPublishJob |
|||
hasExistingJob := len(p.jobs) == len(assignments) |
|||
for i, assignment := range assignments { |
|||
if assignment.LeaderBroker == "" { |
|||
continue |
|||
} |
|||
if hasExistingJob { |
|||
var existingJob *EachPartitionPublishJob |
|||
existingJob = p.jobs[i] |
|||
if existingJob.BrokerPartitionAssignment.LeaderBroker == assignment.LeaderBroker { |
|||
existingJob.generation = generation |
|||
jobs = append(jobs, existingJob) |
|||
continue |
|||
} else { |
|||
if existingJob.LeaderBroker != "" { |
|||
close(existingJob.stopChan) |
|||
existingJob.LeaderBroker = "" |
|||
existingJob.wg.Wait() |
|||
} |
|||
} |
|||
} |
|||
|
|||
// start a go routine to publish to this partition
|
|||
job := &EachPartitionPublishJob{ |
|||
BrokerPartitionAssignment: assignment, |
|||
stopChan: make(chan bool, 1), |
|||
generation: generation, |
|||
} |
|||
job.wg.Add(1) |
|||
go func(job *EachPartitionPublishJob) { |
|||
defer job.wg.Done() |
|||
if err := p.doPublishToPartition(job); err != nil { |
|||
errChan <- EachPartitionError{assignment, err, generation} |
|||
} |
|||
}(job) |
|||
jobs = append(jobs, job) |
|||
} |
|||
p.jobs = jobs |
|||
} |
|||
|
|||
func (p *TopicPublisher) doPublishToPartition(job *EachPartitionPublishJob) error { |
|||
|
|||
return nil |
|||
} |
|||
|
|||
func (p *TopicPublisher) doEnsureConfigureTopic(bootstrapBrokers []string) (err error) { |
|||
if len(bootstrapBrokers) == 0 { |
|||
return fmt.Errorf("no bootstrap brokers") |
|||
} |
|||
var lastErr error |
|||
for _, brokerAddress := range bootstrapBrokers { |
|||
err = pb.WithBrokerGrpcClient(false, |
|||
brokerAddress, |
|||
p.grpcDialOption, |
|||
func(client mq_pb.SeaweedMessagingClient) error { |
|||
_, err := client.ConfigureTopic(context.Background(), &mq_pb.ConfigureTopicRequest{ |
|||
Topic: &mq_pb.Topic{ |
|||
Namespace: p.namespace, |
|||
Name: p.topic, |
|||
}, |
|||
PartitionCount: p.config.CreateTopicPartitionCount, |
|||
}) |
|||
return err |
|||
}) |
|||
if err == nil { |
|||
return nil |
|||
} else { |
|||
lastErr = err |
|||
} |
|||
} |
|||
|
|||
if lastErr != nil { |
|||
return fmt.Errorf("configure topic %s/%s: %v", p.namespace, p.topic, err) |
|||
} |
|||
return nil |
|||
} |
|||
|
|||
func (p *TopicPublisher) doLookupTopicPartitions(bootstrapBrokers []string) (assignments []*mq_pb.BrokerPartitionAssignment, err error) { |
|||
if len(bootstrapBrokers) == 0 { |
|||
return nil, fmt.Errorf("no bootstrap brokers") |
|||
} |
|||
var lastErr error |
|||
for _, brokerAddress := range bootstrapBrokers { |
|||
err := pb.WithBrokerGrpcClient(false, |
|||
brokerAddress, |
|||
p.grpcDialOption, |
|||
func(client mq_pb.SeaweedMessagingClient) error { |
|||
lookupResp, err := client.LookupTopicBrokers(context.Background(), |
|||
&mq_pb.LookupTopicBrokersRequest{ |
|||
Topic: &mq_pb.Topic{ |
|||
Namespace: p.namespace, |
|||
Name: p.topic, |
|||
}, |
|||
}) |
|||
glog.V(0).Infof("lookup topic %s/%s: %v", p.namespace, p.topic, lookupResp) |
|||
|
|||
if err != nil { |
|||
return err |
|||
} |
|||
|
|||
assignments = lookupResp.BrokerPartitionAssignments |
|||
|
|||
return nil |
|||
}) |
|||
if err == nil { |
|||
return assignments, nil |
|||
} else { |
|||
lastErr = err |
|||
} |
|||
} |
|||
|
|||
return nil, fmt.Errorf("lookup topic %s/%s: %v", p.namespace, p.topic, lastErr) |
|||
|
|||
} |
Write
Preview
Loading…
Cancel
Save
Reference in new issue