discovery: respect context cancellation everywhere

This also removes closing of the target group channel everywhere
as the contexts cancels across all stages and we don't care about
draining all events once that happened.
This commit is contained in:
Fabian Reinartz 2016-11-18 10:55:29 +01:00
parent 9b7e097a76
commit b4d7ce1370
9 changed files with 70 additions and 70 deletions

View file

@ -81,7 +81,6 @@ func NewAzureDiscovery(cfg *config.AzureSDConfig) *AzureDiscovery {
// Run implements the TargetProvider interface. // Run implements the TargetProvider interface.
func (ad *AzureDiscovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) { func (ad *AzureDiscovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) {
defer close(ch)
ticker := time.NewTicker(ad.interval) ticker := time.NewTicker(ad.interval)
defer ticker.Stop() defer ticker.Stop()
@ -96,7 +95,10 @@ func (ad *AzureDiscovery) Run(ctx context.Context, ch chan<- []*config.TargetGro
if err != nil { if err != nil {
log.Errorf("unable to refresh during Azure discovery: %s", err) log.Errorf("unable to refresh during Azure discovery: %s", err)
} else { } else {
ch <- []*config.TargetGroup{tg} select {
case <-ctx.Done():
case ch <- []*config.TargetGroup{tg}:
}
} }
select { select {

View file

@ -18,7 +18,6 @@ import (
"net" "net"
"strconv" "strconv"
"strings" "strings"
"sync"
"time" "time"
consul "github.com/hashicorp/consul/api" consul "github.com/hashicorp/consul/api"
@ -133,12 +132,6 @@ func (cd *Discovery) shouldWatch(name string) bool {
// Run implements the TargetProvider interface. // Run implements the TargetProvider interface.
func (cd *Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) { func (cd *Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) {
var wg sync.WaitGroup
defer func() {
wg.Wait()
close(ch)
}()
// Watched services and their cancelation functions. // Watched services and their cancelation functions.
services := map[string]func(){} services := map[string]func(){}
@ -204,11 +197,7 @@ func (cd *Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) {
} }
wctx, cancel := context.WithCancel(ctx) wctx, cancel := context.WithCancel(ctx)
wg.Add(1) srv.watch(wctx, ch)
go func() {
srv.watch(wctx, ch)
wg.Done()
}()
services[name] = cancel services[name] = cancel
} }

View file

@ -90,8 +90,6 @@ func NewDiscovery(conf *config.DNSSDConfig) *Discovery {
// Run implements the TargetProvider interface. // Run implements the TargetProvider interface.
func (dd *Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) { func (dd *Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) {
defer close(ch)
ticker := time.NewTicker(dd.interval) ticker := time.NewTicker(dd.interval)
defer ticker.Stop() defer ticker.Stop()

View file

@ -93,8 +93,6 @@ func NewEC2Discovery(conf *config.EC2SDConfig) *EC2Discovery {
// Run implements the TargetProvider interface. // Run implements the TargetProvider interface.
func (ed *EC2Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) { func (ed *EC2Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) {
defer close(ch)
ticker := time.NewTicker(ed.interval) ticker := time.NewTicker(ed.interval)
defer ticker.Stop() defer ticker.Stop()
@ -103,7 +101,11 @@ func (ed *EC2Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup
if err != nil { if err != nil {
log.Error(err) log.Error(err)
} else { } else {
ch <- []*config.TargetGroup{tg} select {
case ch <- []*config.TargetGroup{tg}:
case <-ctx.Done():
return
}
} }
for { for {
@ -112,8 +114,13 @@ func (ed *EC2Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup
tg, err := ed.refresh() tg, err := ed.refresh()
if err != nil { if err != nil {
log.Error(err) log.Error(err)
} else { continue
ch <- []*config.TargetGroup{tg} }
select {
case ch <- []*config.TargetGroup{tg}:
case <-ctx.Done():
return
} }
case <-ctx.Done(): case <-ctx.Done():
return return

View file

@ -109,7 +109,6 @@ func (fd *FileDiscovery) watchFiles() {
// Run implements the TargetProvider interface. // Run implements the TargetProvider interface.
func (fd *FileDiscovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) { func (fd *FileDiscovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) {
defer close(ch)
defer fd.stop() defer fd.stop()
watcher, err := fsnotify.NewWatcher() watcher, err := fsnotify.NewWatcher()
@ -119,47 +118,40 @@ func (fd *FileDiscovery) Run(ctx context.Context, ch chan<- []*config.TargetGrou
} }
fd.watcher = watcher fd.watcher = watcher
fd.refresh(ch) fd.refresh(ctx, ch)
ticker := time.NewTicker(fd.interval) ticker := time.NewTicker(fd.interval)
defer ticker.Stop() defer ticker.Stop()
for { for {
// Stopping has priority over refreshing. Thus we wrap the actual select
// clause to always catch done signals.
select { select {
case <-ctx.Done(): case <-ctx.Done():
return return
default:
select {
case <-ctx.Done():
return
case event := <-fd.watcher.Events: case event := <-fd.watcher.Events:
// fsnotify sometimes sends a bunch of events without name or operation. // fsnotify sometimes sends a bunch of events without name or operation.
// It's unclear what they are and why they are sent - filter them out. // It's unclear what they are and why they are sent - filter them out.
if len(event.Name) == 0 { if len(event.Name) == 0 {
break break
} }
// Everything but a chmod requires rereading. // Everything but a chmod requires rereading.
if event.Op^fsnotify.Chmod == 0 { if event.Op^fsnotify.Chmod == 0 {
break break
} }
// Changes to a file can spawn various sequences of events with // Changes to a file can spawn various sequences of events with
// different combinations of operations. For all practical purposes // different combinations of operations. For all practical purposes
// this is inaccurate. // this is inaccurate.
// The most reliable solution is to reload everything if anything happens. // The most reliable solution is to reload everything if anything happens.
fd.refresh(ch) fd.refresh(ctx, ch)
case <-ticker.C: case <-ticker.C:
// Setting a new watch after an update might fail. Make sure we don't lose // Setting a new watch after an update might fail. Make sure we don't lose
// those files forever. // those files forever.
fd.refresh(ch) fd.refresh(ctx, ch)
case err := <-fd.watcher.Errors: case err := <-fd.watcher.Errors:
if err != nil { if err != nil {
log.Errorf("Error on file watch: %s", err) log.Errorf("Error on file watch: %s", err)
}
} }
} }
} }
@ -193,7 +185,7 @@ func (fd *FileDiscovery) stop() {
// refresh reads all files matching the discovery's patterns and sends the respective // refresh reads all files matching the discovery's patterns and sends the respective
// updated target groups through the channel. // updated target groups through the channel.
func (fd *FileDiscovery) refresh(ch chan<- []*config.TargetGroup) { func (fd *FileDiscovery) refresh(ctx context.Context, ch chan<- []*config.TargetGroup) {
t0 := time.Now() t0 := time.Now()
defer func() { defer func() {
fileSDScanDuration.Observe(time.Since(t0).Seconds()) fileSDScanDuration.Observe(time.Since(t0).Seconds())
@ -209,7 +201,11 @@ func (fd *FileDiscovery) refresh(ch chan<- []*config.TargetGroup) {
ref[p] = fd.lastRefresh[p] ref[p] = fd.lastRefresh[p]
continue continue
} }
ch <- tgroups select {
case ch <- tgroups:
case <-ctx.Done():
return
}
ref[p] = len(tgroups) ref[p] = len(tgroups)
} }
@ -218,8 +214,10 @@ func (fd *FileDiscovery) refresh(ch chan<- []*config.TargetGroup) {
m, ok := ref[f] m, ok := ref[f]
if !ok || n > m { if !ok || n > m {
for i := m; i < n; i++ { for i := m; i < n; i++ {
ch <- []*config.TargetGroup{ select {
{Source: fileSource(f, i)}, case ch <- []*config.TargetGroup{{Source: fileSource(f, i)}}:
case <-ctx.Done():
return
} }
} }
} }

View file

@ -106,11 +106,17 @@ retry:
// not try to make sense of it all... // not try to make sense of it all...
drained := make(chan struct{}) drained := make(chan struct{})
go func() { go func() {
for tgs := range ch { Loop:
// Below we will change the file to a bad syntax. Previously extracted target for {
// groups must not be deleted via sending an empty target group. select {
if len(tgs[0].Targets) == 0 { case tgs := <-ch:
t.Errorf("Unexpected empty target groups received: %s", tgs) // Below we will change the file to a bad syntax. Previously extracted target
// groups must not be deleted via sending an empty target group.
if len(tgs[0].Targets) == 0 {
t.Errorf("Unexpected empty target groups received: %s", tgs)
}
case <-time.After(500 * time.Millisecond):
break Loop
} }
} }
close(drained) close(drained)

View file

@ -108,14 +108,15 @@ func NewGCEDiscovery(conf *config.GCESDConfig) (*GCEDiscovery, error) {
// Run implements the TargetProvider interface. // Run implements the TargetProvider interface.
func (gd *GCEDiscovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) { func (gd *GCEDiscovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) {
defer close(ch)
// Get an initial set right away. // Get an initial set right away.
tg, err := gd.refresh() tg, err := gd.refresh()
if err != nil { if err != nil {
log.Error(err) log.Error(err)
} else { } else {
ch <- []*config.TargetGroup{tg} select {
case ch <- []*config.TargetGroup{tg}:
case <-ctx.Done():
}
} }
ticker := time.NewTicker(gd.interval) ticker := time.NewTicker(gd.interval)
@ -127,8 +128,11 @@ func (gd *GCEDiscovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup
tg, err := gd.refresh() tg, err := gd.refresh()
if err != nil { if err != nil {
log.Error(err) log.Error(err)
} else { continue
ch <- []*config.TargetGroup{tg} }
select {
case ch <- []*config.TargetGroup{tg}:
case <-ctx.Done():
} }
case <-ctx.Done(): case <-ctx.Done():
return return

View file

@ -107,8 +107,6 @@ const resyncPeriod = 10 * time.Minute
// Run implements the TargetProvider interface. // Run implements the TargetProvider interface.
func (k *Kubernetes) Run(ctx context.Context, ch chan<- []*config.TargetGroup) { func (k *Kubernetes) Run(ctx context.Context, ch chan<- []*config.TargetGroup) {
defer close(ch)
rclient := k.client.Core().GetRESTClient() rclient := k.client.Core().GetRESTClient()
switch k.role { switch k.role {

View file

@ -103,8 +103,6 @@ func NewDiscovery(conf *config.MarathonSDConfig) (*Discovery, error) {
// Run implements the TargetProvider interface. // Run implements the TargetProvider interface.
func (md *Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) { func (md *Discovery) Run(ctx context.Context, ch chan<- []*config.TargetGroup) {
defer close(ch)
for { for {
select { select {
case <-ctx.Done(): case <-ctx.Done():