Skip to content

Instantly share code, notes, and snippets.

@vtolstov
Created June 11, 2019 08:33
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save vtolstov/3e31cafc4fcdbe3ce2c7888bfc2189f0 to your computer and use it in GitHub Desktop.
Save vtolstov/3e31cafc4fcdbe3ce2c7888bfc2189f0 to your computer and use it in GitHub Desktop.
package multi
import (
"sync"
"github.com/micro/go-micro/registry"
)
type multiWatcher struct {
wo registry.WatchOptions
w []registry.Watcher
next chan *registry.Result
stop chan bool
}
func newMultiWatcher(r []registry.Registry, opts ...registry.WatchOption) (registry.Watcher, error) {
var wo registry.WatchOptions
for _, o := range opts {
o(&wo)
}
mw := &multiWatcher{
wo: wo,
next: make(chan *registry.Result),
stop: make(chan bool),
}
for _, wr := range r {
w, err := wr.Watch(opts...)
if err != nil {
return nil, err
}
mw.w = append(mw.w, w)
}
return mw, nil
}
func (mw *multiWatcher) Next() (*registry.Result, error) {
cerr := make(chan error)
for _, wt := range mw.w {
go func(w registry.Watcher) {
r, err := w.Next()
if err != nil && err != registry.ErrNotFound {
cerr <- err
}
mw.next <- r
}(wt)
}
for {
select {
case err := <-cerr:
return nil, err
case r, ok := <-mw.next:
if !ok {
return nil, registry.ErrWatcherStopped
}
nr := &registry.Result{}
*nr = *r
return nr, nil
case <-mw.stop:
return nil, registry.ErrWatcherStopped
}
}
}
func (mw *multiWatcher) Stop() {
var wg sync.WaitGroup
wg.Add(len(mw.w))
for _, w := range mw.w {
go func(w registry.Watcher) {
w.Stop()
wg.Done()
}(w)
}
wg.Wait()
select {
case <-mw.stop:
return
default:
close(mw.stop)
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment