2017-07-06 12:40:54 +02:00
|
|
|
package router
|
2017-03-13 18:54:34 -07:00
|
|
|
|
|
|
|
import (
|
2017-07-06 12:48:37 +02:00
|
|
|
"github.com/hashicorp/consul/agent/metadata"
|
2017-03-13 18:54:34 -07:00
|
|
|
"github.com/hashicorp/consul/types"
|
2020-01-28 17:50:41 -06:00
|
|
|
"github.com/hashicorp/go-hclog"
|
2017-03-13 18:54:34 -07:00
|
|
|
"github.com/hashicorp/serf/serf"
|
|
|
|
)
|
|
|
|
|
|
|
|
// routerFn selects one of the router operations to map to incoming Serf events.
|
2017-07-06 12:48:37 +02:00
|
|
|
type routerFn func(types.AreaID, *metadata.Server) error
|
2017-03-13 18:54:34 -07:00
|
|
|
|
|
|
|
// handleMemberEvents attempts to apply the given Serf member event to the given
|
|
|
|
// router function.
|
2020-01-28 17:50:41 -06:00
|
|
|
func handleMemberEvent(logger hclog.Logger, fn routerFn, areaID types.AreaID, e serf.Event) {
|
2017-03-13 18:54:34 -07:00
|
|
|
me, ok := e.(serf.MemberEvent)
|
|
|
|
if !ok {
|
2020-01-28 17:50:41 -06:00
|
|
|
logger.Error("Bad event type", "event", e)
|
2017-03-13 18:54:34 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, m := range me.Members {
|
2017-07-06 12:48:37 +02:00
|
|
|
ok, parts := metadata.IsConsulServer(m)
|
2017-03-13 18:54:34 -07:00
|
|
|
if !ok {
|
2020-01-28 17:50:41 -06:00
|
|
|
logger.Warn("Non-server in server-only area",
|
|
|
|
"non_server", m.Name,
|
|
|
|
"area", areaID,
|
|
|
|
)
|
2017-03-13 18:54:34 -07:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := fn(areaID, parts); err != nil {
|
2020-01-28 17:50:41 -06:00
|
|
|
logger.Error("Failed to process event for server in area",
|
|
|
|
"event", me.Type.String(),
|
|
|
|
"server", m.Name,
|
|
|
|
"area", areaID,
|
|
|
|
"error", err,
|
|
|
|
)
|
2017-03-13 18:54:34 -07:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-01-28 17:50:41 -06:00
|
|
|
logger.Info("Handled event for server in area",
|
|
|
|
"event", me.Type.String(),
|
|
|
|
"server", m.Name,
|
|
|
|
"area", areaID,
|
|
|
|
)
|
2017-03-13 18:54:34 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// HandleSerfEvents is a long-running goroutine that pushes incoming events from
|
|
|
|
// a Serf manager's channel into the given router. This will return when the
|
|
|
|
// shutdown channel is closed.
|
2020-01-28 17:50:41 -06:00
|
|
|
func HandleSerfEvents(logger hclog.Logger, router *Router, areaID types.AreaID, shutdownCh <-chan struct{}, eventCh <-chan serf.Event) {
|
2017-03-13 18:54:34 -07:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-shutdownCh:
|
|
|
|
return
|
|
|
|
|
|
|
|
case e := <-eventCh:
|
|
|
|
switch e.EventType() {
|
|
|
|
case serf.EventMemberJoin:
|
|
|
|
handleMemberEvent(logger, router.AddServer, areaID, e)
|
|
|
|
|
2019-03-04 09:19:35 -05:00
|
|
|
case serf.EventMemberLeave, serf.EventMemberReap:
|
2017-03-13 18:54:34 -07:00
|
|
|
handleMemberEvent(logger, router.RemoveServer, areaID, e)
|
|
|
|
|
|
|
|
case serf.EventMemberFailed:
|
|
|
|
handleMemberEvent(logger, router.FailServer, areaID, e)
|
|
|
|
|
|
|
|
// All of these event types are ignored.
|
|
|
|
case serf.EventMemberUpdate:
|
2020-01-29 13:21:38 -05:00
|
|
|
handleMemberEvent(logger, router.AddServer, areaID, e)
|
2017-03-13 18:54:34 -07:00
|
|
|
case serf.EventUser:
|
|
|
|
case serf.EventQuery:
|
|
|
|
|
|
|
|
default:
|
2020-01-28 17:50:41 -06:00
|
|
|
logger.Warn("Unhandled Serf Event", "event", e)
|
2017-03-13 18:54:34 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|