2015-08-28 10:26:23 +00:00
|
|
|
package fingerprint
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"strconv"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
consul "github.com/hashicorp/consul/api"
|
|
|
|
|
2018-01-24 14:09:53 +00:00
|
|
|
cstructs "github.com/hashicorp/nomad/client/structs"
|
2015-08-28 10:26:23 +00:00
|
|
|
"github.com/hashicorp/nomad/nomad/structs"
|
|
|
|
)
|
|
|
|
|
2015-12-11 17:01:28 +00:00
|
|
|
const (
|
2015-12-11 21:47:35 +00:00
|
|
|
consulAvailable = "available"
|
|
|
|
consulUnavailable = "unavailable"
|
2015-12-11 17:01:28 +00:00
|
|
|
)
|
|
|
|
|
2016-09-01 20:38:31 +00:00
|
|
|
// ConsulFingerprint is used to fingerprint for Consul
|
2015-08-28 10:26:23 +00:00
|
|
|
type ConsulFingerprint struct {
|
2015-12-11 17:01:28 +00:00
|
|
|
logger *log.Logger
|
|
|
|
client *consul.Client
|
|
|
|
lastState string
|
2015-08-28 10:26:23 +00:00
|
|
|
}
|
|
|
|
|
2016-09-01 18:02:19 +00:00
|
|
|
// NewConsulFingerprint is used to create a Consul fingerprint
|
2015-08-28 10:26:23 +00:00
|
|
|
func NewConsulFingerprint(logger *log.Logger) Fingerprint {
|
2015-12-11 21:47:35 +00:00
|
|
|
return &ConsulFingerprint{logger: logger, lastState: consulUnavailable}
|
2015-08-28 10:26:23 +00:00
|
|
|
}
|
|
|
|
|
2018-01-24 14:09:53 +00:00
|
|
|
func (f *ConsulFingerprint) Fingerprint(req *cstructs.FingerprintRequest, resp *cstructs.FingerprintResponse) error {
|
2015-11-22 21:42:06 +00:00
|
|
|
// Only create the client once to avoid creating too many connections to
|
|
|
|
// Consul.
|
|
|
|
if f.client == nil {
|
2018-01-24 14:09:53 +00:00
|
|
|
consulConfig, err := req.Config.ConsulConfig.ApiConfig()
|
2015-11-22 21:42:06 +00:00
|
|
|
if err != nil {
|
2018-01-24 14:09:53 +00:00
|
|
|
return fmt.Errorf("Failed to initialize the Consul client config: %v", err)
|
2015-11-22 21:42:06 +00:00
|
|
|
}
|
2015-08-28 10:26:23 +00:00
|
|
|
|
2015-11-22 21:42:06 +00:00
|
|
|
f.client, err = consul.NewClient(consulConfig)
|
|
|
|
if err != nil {
|
2018-01-24 14:09:53 +00:00
|
|
|
return fmt.Errorf("Failed to initialize consul client: %s", err)
|
2015-11-22 21:42:06 +00:00
|
|
|
}
|
2015-08-28 10:26:23 +00:00
|
|
|
}
|
|
|
|
|
2015-09-01 02:56:25 +00:00
|
|
|
// We'll try to detect consul by making a query to to the agent's self API.
|
|
|
|
// If we can't hit this URL consul is probably not running on this machine.
|
2015-11-22 21:42:06 +00:00
|
|
|
info, err := f.client.Agent().Self()
|
2015-08-28 10:26:23 +00:00
|
|
|
if err != nil {
|
2018-01-24 14:09:53 +00:00
|
|
|
// TODO this should set consul in the response if the error is not nil
|
2015-12-11 17:01:28 +00:00
|
|
|
|
|
|
|
// Print a message indicating that the Consul Agent is not available
|
|
|
|
// anymore
|
2015-12-11 21:47:35 +00:00
|
|
|
if f.lastState == consulAvailable {
|
2015-12-11 17:10:21 +00:00
|
|
|
f.logger.Printf("[INFO] fingerprint.consul: consul agent is unavailable")
|
2015-12-11 17:01:28 +00:00
|
|
|
}
|
2015-12-11 21:47:35 +00:00
|
|
|
f.lastState = consulUnavailable
|
2018-01-24 14:09:53 +00:00
|
|
|
return nil
|
2015-08-28 10:26:23 +00:00
|
|
|
}
|
|
|
|
|
2017-10-12 01:25:54 +00:00
|
|
|
if s, ok := info["Config"]["Server"].(bool); ok {
|
2018-01-24 14:09:53 +00:00
|
|
|
resp.Attributes["consul.server"] = strconv.FormatBool(s)
|
2017-10-12 01:25:54 +00:00
|
|
|
} else {
|
|
|
|
f.logger.Printf("[WARN] fingerprint.consul: unable to fingerprint consul.server")
|
|
|
|
}
|
|
|
|
if v, ok := info["Config"]["Version"].(string); ok {
|
2018-01-24 14:09:53 +00:00
|
|
|
resp.Attributes["consul.version"] = v
|
2017-10-12 01:25:54 +00:00
|
|
|
} else {
|
|
|
|
f.logger.Printf("[WARN] fingerprint.consul: unable to fingerprint consul.version")
|
|
|
|
}
|
|
|
|
if r, ok := info["Config"]["Revision"].(string); ok {
|
2018-01-24 14:09:53 +00:00
|
|
|
resp.Attributes["consul.revision"] = r
|
2017-10-12 01:25:54 +00:00
|
|
|
} else {
|
|
|
|
f.logger.Printf("[WARN] fingerprint.consul: unable to fingerprint consul.revision")
|
|
|
|
}
|
|
|
|
if n, ok := info["Config"]["NodeName"].(string); ok {
|
2018-01-24 14:09:53 +00:00
|
|
|
resp.Attributes["unique.consul.name"] = n
|
2017-10-12 01:25:54 +00:00
|
|
|
} else {
|
|
|
|
f.logger.Printf("[WARN] fingerprint.consul: unable to fingerprint unique.consul.name")
|
|
|
|
}
|
|
|
|
if d, ok := info["Config"]["Datacenter"].(string); ok {
|
2018-01-24 14:09:53 +00:00
|
|
|
resp.Attributes["consul.datacenter"] = d
|
2017-10-12 01:25:54 +00:00
|
|
|
} else {
|
|
|
|
f.logger.Printf("[WARN] fingerprint.consul: unable to fingerprint consul.datacenter")
|
|
|
|
}
|
|
|
|
|
2018-01-24 14:09:53 +00:00
|
|
|
if resp.Attributes["consul.datacenter"] != "" || resp.Attributes["unique.consul.name"] != "" {
|
|
|
|
resp.Links["consul"] = fmt.Sprintf("%s.%s",
|
|
|
|
resp.Attributes["consul.datacenter"],
|
|
|
|
resp.Attributes["unique.consul.name"])
|
2017-10-12 01:25:54 +00:00
|
|
|
} else {
|
|
|
|
f.logger.Printf("[WARN] fingerprint.consul: malformed Consul response prevented linking")
|
|
|
|
}
|
2015-08-28 10:35:44 +00:00
|
|
|
|
2015-12-11 17:01:28 +00:00
|
|
|
// If the Consul Agent was previously unavailable print a message to
|
|
|
|
// indicate the Agent is available now
|
2015-12-11 21:47:35 +00:00
|
|
|
if f.lastState == consulUnavailable {
|
2015-12-11 21:18:04 +00:00
|
|
|
f.logger.Printf("[INFO] fingerprint.consul: consul agent is available")
|
2015-12-11 17:01:28 +00:00
|
|
|
}
|
2015-12-11 21:47:35 +00:00
|
|
|
f.lastState = consulAvailable
|
2018-01-24 14:09:53 +00:00
|
|
|
return nil
|
2015-08-28 10:26:23 +00:00
|
|
|
}
|
2015-11-05 21:46:02 +00:00
|
|
|
|
2015-11-22 21:51:10 +00:00
|
|
|
// clearConsulAttributes removes consul attributes and links from the passed
|
|
|
|
// Node.
|
|
|
|
func (f *ConsulFingerprint) clearConsulAttributes(n *structs.Node) {
|
|
|
|
delete(n.Links, "consul")
|
|
|
|
}
|
|
|
|
|
2015-11-05 21:46:02 +00:00
|
|
|
func (f *ConsulFingerprint) Periodic() (bool, time.Duration) {
|
2015-11-05 21:41:41 +00:00
|
|
|
return true, 15 * time.Second
|
2015-11-05 21:46:02 +00:00
|
|
|
}
|