2017-07-06 12:48:37 +02:00
|
|
|
package metadata
|
2016-02-19 17:26:45 -08:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net"
|
2017-03-21 16:36:44 -07:00
|
|
|
"regexp"
|
2016-02-19 17:26:45 -08:00
|
|
|
"strconv"
|
2017-08-14 07:36:07 -07:00
|
|
|
"strings"
|
2016-02-19 17:26:45 -08:00
|
|
|
|
2017-03-21 16:36:44 -07:00
|
|
|
"github.com/hashicorp/go-version"
|
2016-02-19 17:26:45 -08:00
|
|
|
"github.com/hashicorp/serf/serf"
|
|
|
|
)
|
|
|
|
|
2016-03-26 17:58:12 -07:00
|
|
|
// Key is used in maps and for equality tests. A key is based on endpoints.
|
|
|
|
type Key struct {
|
2016-03-28 12:53:19 -07:00
|
|
|
name string
|
2016-03-26 17:58:12 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Equal compares two Key objects
|
|
|
|
func (k *Key) Equal(x *Key) bool {
|
2016-03-28 12:53:19 -07:00
|
|
|
return k.name == x.name
|
2016-03-26 17:58:12 -07:00
|
|
|
}
|
|
|
|
|
2016-03-29 17:39:19 -07:00
|
|
|
// Server is used to return details of a consul server
|
|
|
|
type Server struct {
|
2022-08-31 16:38:42 -05:00
|
|
|
Name string // <node>.<dc>
|
|
|
|
ShortName string // <node>
|
|
|
|
ID string
|
|
|
|
Datacenter string
|
|
|
|
Segment string
|
|
|
|
Port int
|
|
|
|
SegmentAddrs map[string]string
|
|
|
|
SegmentPorts map[string]int
|
|
|
|
WanJoinPort int
|
|
|
|
LanJoinPort int
|
2022-08-25 11:44:58 -05:00
|
|
|
ExternalGRPCPort int
|
|
|
|
ExternalGRPCTLSPort int
|
2022-08-31 16:38:42 -05:00
|
|
|
Bootstrap bool
|
|
|
|
Expect int
|
|
|
|
Build version.Version
|
|
|
|
Version int
|
|
|
|
RaftVersion int
|
|
|
|
Addr net.Addr
|
|
|
|
Status serf.MemberStatus
|
|
|
|
ReadReplica bool
|
|
|
|
FeatureFlags map[string]int
|
2017-05-10 14:25:48 -07:00
|
|
|
|
|
|
|
// If true, use TLS when connecting to this server
|
|
|
|
UseTLS bool
|
2016-02-19 17:26:45 -08:00
|
|
|
}
|
|
|
|
|
2016-03-26 17:58:12 -07:00
|
|
|
// Key returns the corresponding Key
|
2016-03-29 17:39:19 -07:00
|
|
|
func (s *Server) Key() *Key {
|
2016-03-26 17:58:12 -07:00
|
|
|
return &Key{
|
2016-03-28 12:53:19 -07:00
|
|
|
name: s.Name,
|
2016-03-26 17:58:12 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-29 17:39:19 -07:00
|
|
|
// String returns a string representation of Server
|
|
|
|
func (s *Server) String() string {
|
2016-03-27 01:32:04 -07:00
|
|
|
var addrStr, networkStr string
|
2016-03-26 19:30:04 -07:00
|
|
|
if s.Addr != nil {
|
2016-03-27 01:32:04 -07:00
|
|
|
addrStr = s.Addr.String()
|
|
|
|
networkStr = s.Addr.Network()
|
2016-03-26 19:30:04 -07:00
|
|
|
}
|
|
|
|
|
2016-03-28 11:37:25 -07:00
|
|
|
return fmt.Sprintf("%s (Addr: %s/%s) (DC: %s)", s.Name, networkStr, addrStr, s.Datacenter)
|
2016-02-19 17:26:45 -08:00
|
|
|
}
|
|
|
|
|
2017-03-21 16:36:44 -07:00
|
|
|
var versionFormat = regexp.MustCompile(`\d+\.\d+\.\d+`)
|
|
|
|
|
2016-03-29 17:39:19 -07:00
|
|
|
// IsConsulServer returns true if a serf member is a consul server
|
|
|
|
// agent. Returns a bool and a pointer to the Server.
|
|
|
|
func IsConsulServer(m serf.Member) (bool, *Server) {
|
2016-02-19 17:26:45 -08:00
|
|
|
if m.Tags["role"] != "consul" {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
datacenter := m.Tags["dc"]
|
2017-08-14 07:36:07 -07:00
|
|
|
segment := m.Tags["segment"]
|
2016-02-19 17:26:45 -08:00
|
|
|
_, bootstrap := m.Tags["bootstrap"]
|
2017-05-10 14:25:48 -07:00
|
|
|
_, useTLS := m.Tags["use_tls"]
|
|
|
|
|
2016-02-19 17:26:45 -08:00
|
|
|
expect := 0
|
2018-01-28 22:40:13 +04:00
|
|
|
expectStr, ok := m.Tags["expect"]
|
2016-02-19 17:26:45 -08:00
|
|
|
var err error
|
|
|
|
if ok {
|
2018-01-28 22:40:13 +04:00
|
|
|
expect, err = strconv.Atoi(expectStr)
|
2016-02-19 17:26:45 -08:00
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-28 22:40:13 +04:00
|
|
|
portStr := m.Tags["port"]
|
|
|
|
port, err := strconv.Atoi(portStr)
|
2016-02-19 17:26:45 -08:00
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
2018-01-28 22:40:13 +04:00
|
|
|
segmentAddrs := make(map[string]string)
|
|
|
|
segmentPorts := make(map[string]int)
|
2020-01-29 13:21:38 -05:00
|
|
|
featureFlags := make(map[string]int)
|
2017-08-14 07:36:07 -07:00
|
|
|
for name, value := range m.Tags {
|
2017-08-31 17:39:46 -07:00
|
|
|
if strings.HasPrefix(name, "sl_") {
|
|
|
|
addr, port, err := net.SplitHostPort(value)
|
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
2018-01-28 22:40:13 +04:00
|
|
|
segmentPort, err := strconv.Atoi(port)
|
2017-08-14 07:36:07 -07:00
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
2017-08-31 17:39:46 -07:00
|
|
|
|
2018-01-28 22:40:13 +04:00
|
|
|
segmentName := strings.TrimPrefix(name, "sl_")
|
|
|
|
segmentAddrs[segmentName] = addr
|
|
|
|
segmentPorts[segmentName] = segmentPort
|
2021-05-19 15:07:24 -04:00
|
|
|
} else if strings.HasPrefix(name, featureFlagPrefix) {
|
|
|
|
featureName := strings.TrimPrefix(name, featureFlagPrefix)
|
2020-01-29 13:21:38 -05:00
|
|
|
featureState, err := strconv.Atoi(value)
|
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
featureFlags[featureName] = featureState
|
2017-08-14 07:36:07 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-28 22:40:13 +04:00
|
|
|
buildVersion, err := Build(&m)
|
2017-03-21 16:36:44 -07:00
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
2018-01-28 22:40:13 +04:00
|
|
|
wanJoinPort := 0
|
|
|
|
wanJoinPortStr, ok := m.Tags["wan_join_port"]
|
2017-03-15 12:26:54 -07:00
|
|
|
if ok {
|
2018-01-28 22:40:13 +04:00
|
|
|
wanJoinPort, err = strconv.Atoi(wanJoinPortStr)
|
2017-03-15 12:26:54 -07:00
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-25 11:44:58 -05:00
|
|
|
var externalGRPCPort, externalGRPCTLSPort int
|
|
|
|
externalGRPCPortStr, foundGRPC := m.Tags["grpc_port"]
|
|
|
|
externalGRPCTLSPortStr, foundGRPCTLS := m.Tags["grpc_tls_port"]
|
|
|
|
if foundGRPC {
|
|
|
|
externalGRPCPort, _ = strconv.Atoi(externalGRPCPortStr)
|
|
|
|
}
|
|
|
|
if foundGRPCTLS {
|
|
|
|
externalGRPCTLSPort, _ = strconv.Atoi(externalGRPCTLSPortStr)
|
|
|
|
}
|
|
|
|
// If either port tag was found, check to ensure that at least one port was valid.
|
|
|
|
if foundGRPC || foundGRPCTLS {
|
|
|
|
if externalGRPCPort < 1 && externalGRPCTLSPort < 1 {
|
2022-07-07 13:55:41 -05:00
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-28 22:40:13 +04:00
|
|
|
vsnStr := m.Tags["vsn"]
|
|
|
|
vsn, err := strconv.Atoi(vsnStr)
|
2016-02-19 17:26:45 -08:00
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
2018-01-28 22:40:13 +04:00
|
|
|
raftVsn := 0
|
2018-01-28 22:48:21 +04:00
|
|
|
raftVsnStr, ok := m.Tags["raft_vsn"]
|
2017-03-14 10:43:43 -07:00
|
|
|
if ok {
|
2018-01-28 22:48:21 +04:00
|
|
|
raftVsn, err = strconv.Atoi(raftVsnStr)
|
2017-03-14 10:43:43 -07:00
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
2017-02-22 12:53:32 -08:00
|
|
|
}
|
|
|
|
|
2018-09-19 17:41:36 -07:00
|
|
|
// Check if the server is a non voter
|
2020-11-17 10:53:57 -05:00
|
|
|
// DEPRECATED - remove looking for the nonvoter tag eventually once we don't have to support
|
|
|
|
// read replicas running v1.8.x and below.
|
2018-09-19 17:41:36 -07:00
|
|
|
_, nonVoter := m.Tags["nonvoter"]
|
2020-11-17 10:53:57 -05:00
|
|
|
_, readReplica := m.Tags["read_replica"]
|
2018-09-19 17:41:36 -07:00
|
|
|
|
2016-02-19 17:26:45 -08:00
|
|
|
addr := &net.TCPAddr{IP: m.Addr, Port: port}
|
|
|
|
|
2016-03-29 17:39:19 -07:00
|
|
|
parts := &Server{
|
2022-08-25 11:44:58 -05:00
|
|
|
Name: m.Name,
|
|
|
|
ShortName: strings.TrimSuffix(m.Name, "."+datacenter),
|
|
|
|
ID: m.Tags["id"],
|
|
|
|
Datacenter: datacenter,
|
|
|
|
Segment: segment,
|
|
|
|
Port: port,
|
|
|
|
SegmentAddrs: segmentAddrs,
|
|
|
|
SegmentPorts: segmentPorts,
|
|
|
|
WanJoinPort: wanJoinPort,
|
|
|
|
LanJoinPort: int(m.Port),
|
|
|
|
ExternalGRPCPort: externalGRPCPort,
|
|
|
|
ExternalGRPCTLSPort: externalGRPCTLSPort,
|
|
|
|
Bootstrap: bootstrap,
|
|
|
|
Expect: expect,
|
|
|
|
Addr: addr,
|
|
|
|
Build: *buildVersion,
|
|
|
|
Version: vsn,
|
|
|
|
RaftVersion: raftVsn,
|
|
|
|
Status: m.Status,
|
|
|
|
UseTLS: useTLS,
|
2020-11-17 10:53:57 -05:00
|
|
|
// DEPRECATED - remove nonVoter check once support for that tag is removed
|
|
|
|
ReadReplica: nonVoter || readReplica,
|
2020-01-29 13:21:38 -05:00
|
|
|
FeatureFlags: featureFlags,
|
2016-02-19 17:26:45 -08:00
|
|
|
}
|
|
|
|
return true, parts
|
|
|
|
}
|
2021-05-19 15:07:24 -04:00
|
|
|
|
2021-09-29 15:45:11 -04:00
|
|
|
// TODO(ACL-Legacy-Compat): remove in phase 2
|
2021-09-22 18:55:53 -04:00
|
|
|
const TagACLs = "acls"
|
|
|
|
|
2021-05-19 15:07:24 -04:00
|
|
|
const featureFlagPrefix = "ft_"
|
|
|
|
|
|
|
|
// AddFeatureFlags to the tags. The tags map is expected to be a serf.Config.Tags.
|
|
|
|
// The feature flags are encoded in the tags so that IsConsulServer can decode them
|
|
|
|
// and populate the Server.FeatureFlags map.
|
|
|
|
func AddFeatureFlags(tags map[string]string, flags ...string) {
|
|
|
|
for _, flag := range flags {
|
|
|
|
tags[featureFlagPrefix+flag] = "1"
|
|
|
|
}
|
|
|
|
}
|