From 9420e978307009c355f096ff931724b9e75f0940 Mon Sep 17 00:00:00 2001 From: Aaron Hurt Date: Thu, 30 Jan 2020 10:22:25 -0600 Subject: [PATCH] Preserve table state by storing buffer table in fixed strings. Fixes #737 --- Makefile | 2 +- main.go | 45 ++++++++++++++++++++++----------------------- 2 files changed, 23 insertions(+), 24 deletions(-) diff --git a/Makefile b/Makefile index bd20dd96a..eea8c6c36 100644 --- a/Makefile +++ b/Makefile @@ -44,7 +44,7 @@ build: gofmt # test builds and runs the tests test: build - go test $(GOFLAGS) -v -test.timeout 15s ./... + go test -v -test.timeout 15s ./... # mod performs go module maintenance mod: diff --git a/main.go b/main.go index e7d5ad1dc..ae005dbf1 100644 --- a/main.go +++ b/main.go @@ -449,7 +449,6 @@ func initRuntime(cfg *config.Config) { func initBackend(cfg *config.Config) { var deadline = time.Now().Add(cfg.Registry.Timeout) - var err error for { switch cfg.Registry.Backend { @@ -485,17 +484,18 @@ func initBackend(cfg *config.Config) { func watchBackend(cfg *config.Config, first chan bool) { var ( - last string - svccfg string - mancfg string - customBE string - once sync.Once - next = new(bytes.Buffer) // fix crash on reset before used (#650) + nextTable string + lastTable string + svccfg string + mancfg string + customBE string + once sync.Once + tableBuffer = new(bytes.Buffer) // fix crash on reset before used (#650) ) switch cfg.Registry.Backend { - //Custom Back End. Gets JSON from Remote Backend that contains a slice of route.RouteDef. It loads the route table - //Directly from that input + // custom back end receives JSON from a remote source that contains a slice of route.RouteDef + // the route table is created directly from that input case "custom": svc := registry.Default.WatchServices() for { @@ -505,40 +505,39 @@ func watchBackend(cfg *config.Config, first chan bool) { } once.Do(func() { close(first) }) } - //All other back ends + // all other backend types default: svc := registry.Default.WatchServices() man := registry.Default.WatchManual() for { - select { case svccfg = <-svc: case mancfg = <-man: } - // manual config overrides service config - // order matters - next.Reset() - next.WriteString(svccfg) - next.WriteString("\n") - next.WriteString(mancfg) - if next.String() == last { + // manual config overrides service config - order matters + tableBuffer.Reset() + tableBuffer.WriteString(svccfg) + tableBuffer.WriteString("\n") + tableBuffer.WriteString(mancfg) + // set nextTable here to preserve the state. The buffer is altered + // when calling route.NewTable and we lose change logging (#737) + if nextTable = tableBuffer.String(); nextTable == lastTable { continue } - aliases, err := route.ParseAliases(next.String()) + aliases, err := route.ParseAliases(nextTable) if err != nil { log.Printf("[WARN]: %s", err) } registry.Default.Register(aliases) - - t, err := route.NewTable(next) + t, err := route.NewTable(tableBuffer) if err != nil { log.Printf("[WARN] %s", err) continue } route.SetTable(t) - logRoutes(t, last, next.String(), cfg.Log.RoutesFormat) - last = next.String() + logRoutes(t, lastTable, nextTable, cfg.Log.RoutesFormat) + lastTable = nextTable once.Do(func() { close(first) }) } }