2019-10-03 17:19:02 +02:00
|
|
|
// Package service provides the broker service client
|
|
|
|
package service
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2019-10-04 17:30:03 +02:00
|
|
|
"time"
|
2019-10-03 17:19:02 +02:00
|
|
|
|
|
|
|
"github.com/micro/go-micro/broker"
|
|
|
|
pb "github.com/micro/go-micro/broker/service/proto"
|
|
|
|
"github.com/micro/go-micro/client"
|
2019-10-04 17:40:16 +02:00
|
|
|
"github.com/micro/go-micro/util/log"
|
2019-10-03 17:19:02 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type serviceBroker struct {
|
|
|
|
Addrs []string
|
|
|
|
Client pb.BrokerService
|
|
|
|
options broker.Options
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
DefaultName = "go.micro.broker"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (b *serviceBroker) Address() string {
|
|
|
|
return b.Addrs[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *serviceBroker) Connect() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *serviceBroker) Disconnect() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *serviceBroker) Init(opts ...broker.Option) error {
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&b.options)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *serviceBroker) Options() broker.Options {
|
|
|
|
return b.options
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *serviceBroker) Publish(topic string, msg *broker.Message, opts ...broker.PublishOption) error {
|
2019-10-04 17:30:03 +02:00
|
|
|
log.Debugf("Publishing to topic %s broker %v", topic, b.Addrs)
|
2019-10-03 17:19:02 +02:00
|
|
|
_, err := b.Client.Publish(context.TODO(), &pb.PublishRequest{
|
|
|
|
Topic: topic,
|
|
|
|
Message: &pb.Message{
|
|
|
|
Header: msg.Header,
|
|
|
|
Body: msg.Body,
|
|
|
|
},
|
|
|
|
}, client.WithAddress(b.Addrs...))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *serviceBroker) Subscribe(topic string, handler broker.Handler, opts ...broker.SubscribeOption) (broker.Subscriber, error) {
|
|
|
|
var options broker.SubscribeOptions
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
2019-10-04 17:30:03 +02:00
|
|
|
log.Debugf("Subscribing to topic %s queue %s broker %v", topic, options.Queue, b.Addrs)
|
2019-10-03 17:19:02 +02:00
|
|
|
stream, err := b.Client.Subscribe(context.TODO(), &pb.SubscribeRequest{
|
|
|
|
Topic: topic,
|
|
|
|
Queue: options.Queue,
|
|
|
|
}, client.WithAddress(b.Addrs...))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
sub := &serviceSub{
|
|
|
|
topic: topic,
|
|
|
|
queue: options.Queue,
|
|
|
|
handler: handler,
|
|
|
|
stream: stream,
|
|
|
|
closed: make(chan bool),
|
|
|
|
options: options,
|
|
|
|
}
|
2019-10-04 17:30:03 +02:00
|
|
|
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-sub.closed:
|
|
|
|
log.Debugf("Unsubscribed from topic %s", topic)
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
// run the subscriber
|
|
|
|
log.Debugf("Streaming from broker %v to topic [%s] queue [%s]", b.Addrs, topic, options.Queue)
|
|
|
|
if err := sub.run(); err != nil {
|
|
|
|
log.Debugf("Resubscribing to topic %s broker %v", topic, b.Addrs)
|
|
|
|
stream, err := b.Client.Subscribe(context.TODO(), &pb.SubscribeRequest{
|
|
|
|
Topic: topic,
|
|
|
|
Queue: options.Queue,
|
|
|
|
}, client.WithAddress(b.Addrs...))
|
|
|
|
if err != nil {
|
|
|
|
log.Debugf("Failed to resubscribe to topic %s: %v", topic, err)
|
|
|
|
time.Sleep(time.Second)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// new stream
|
|
|
|
sub.stream = stream
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
2019-10-03 17:19:02 +02:00
|
|
|
|
|
|
|
return sub, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *serviceBroker) String() string {
|
|
|
|
return "service"
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBroker(opts ...broker.Option) broker.Broker {
|
|
|
|
var options broker.Options
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
addrs := options.Addrs
|
|
|
|
if len(addrs) == 0 {
|
|
|
|
addrs = []string{"127.0.0.1:8001"}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &serviceBroker{
|
|
|
|
Addrs: addrs,
|
|
|
|
Client: pb.NewBrokerService(DefaultName, client.DefaultClient),
|
|
|
|
options: options,
|
|
|
|
}
|
|
|
|
}
|