2016-08-30 02:09:57 +00:00
|
|
|
package api
|
|
|
|
|
2017-02-24 21:08:49 +00:00
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
2017-03-01 22:04:40 +00:00
|
|
|
"time"
|
2017-02-24 21:08:49 +00:00
|
|
|
)
|
|
|
|
|
2016-08-30 02:09:57 +00:00
|
|
|
// Operator can be used to perform low-level operator tasks for Consul.
|
|
|
|
type Operator struct {
|
|
|
|
c *Client
|
|
|
|
}
|
|
|
|
|
|
|
|
// Operator returns a handle to the operator endpoints.
|
|
|
|
func (c *Client) Operator() *Operator {
|
|
|
|
return &Operator{c}
|
|
|
|
}
|
|
|
|
|
2016-08-30 18:30:56 +00:00
|
|
|
// RaftServer has information about a server in the Raft configuration.
|
|
|
|
type RaftServer struct {
|
|
|
|
// ID is the unique ID for the server. These are currently the same
|
|
|
|
// as the address, but they will be changed to a real GUID in a future
|
|
|
|
// release of Consul.
|
2016-08-30 21:59:16 +00:00
|
|
|
ID string
|
2016-08-30 18:30:56 +00:00
|
|
|
|
|
|
|
// Node is the node name of the server, as known by Consul, or this
|
|
|
|
// will be set to "(unknown)" otherwise.
|
|
|
|
Node string
|
|
|
|
|
|
|
|
// Address is the IP:port of the server, used for Raft communications.
|
2016-08-30 21:59:16 +00:00
|
|
|
Address string
|
2016-08-30 18:30:56 +00:00
|
|
|
|
|
|
|
// Leader is true if this server is the current cluster leader.
|
|
|
|
Leader bool
|
|
|
|
|
|
|
|
// Voter is true if this server has a vote in the cluster. This might
|
|
|
|
// be false if the server is staging and still coming online, or if
|
|
|
|
// it's a non-voting server, which will be added in a future release of
|
|
|
|
// Consul.
|
|
|
|
Voter bool
|
|
|
|
}
|
|
|
|
|
2016-08-30 02:09:57 +00:00
|
|
|
// RaftConfigration is returned when querying for the current Raft configuration.
|
|
|
|
type RaftConfiguration struct {
|
2016-08-30 18:30:56 +00:00
|
|
|
// Servers has the list of servers in the Raft configuration.
|
|
|
|
Servers []*RaftServer
|
|
|
|
|
|
|
|
// Index has the Raft index of this configuration.
|
|
|
|
Index uint64
|
2016-08-30 02:09:57 +00:00
|
|
|
}
|
|
|
|
|
2016-11-22 23:24:40 +00:00
|
|
|
// keyringRequest is used for performing Keyring operations
|
|
|
|
type keyringRequest struct {
|
2016-11-23 00:03:27 +00:00
|
|
|
Key string
|
2016-11-15 02:54:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// KeyringResponse is returned when listing the gossip encryption keys
|
|
|
|
type KeyringResponse struct {
|
|
|
|
// Whether this response is for a WAN ring
|
|
|
|
WAN bool
|
|
|
|
|
|
|
|
// The datacenter name this request corresponds to
|
|
|
|
Datacenter string
|
|
|
|
|
|
|
|
// A map of the encryption keys to the number of nodes they're installed on
|
|
|
|
Keys map[string]int
|
|
|
|
|
|
|
|
// The total number of nodes in this ring
|
|
|
|
NumNodes int
|
|
|
|
}
|
|
|
|
|
2017-02-28 22:23:14 +00:00
|
|
|
// AutopilotConfiguration is used for querying/setting the Autopilot configuration.
|
|
|
|
// Autopilot helps manage operator tasks related to Consul servers like removing
|
|
|
|
// failed servers from the Raft quorum.
|
2017-02-24 05:00:15 +00:00
|
|
|
type AutopilotConfiguration struct {
|
2017-02-28 22:23:14 +00:00
|
|
|
// CleanupDeadServers controls whether to remove dead servers from the Raft
|
2017-02-24 21:08:49 +00:00
|
|
|
// peer list when a new server joins
|
2017-02-28 22:23:14 +00:00
|
|
|
CleanupDeadServers bool
|
2017-02-24 21:08:49 +00:00
|
|
|
|
2017-03-01 22:04:40 +00:00
|
|
|
// LastContactThreshold is the limit on the amount of time a server can go
|
|
|
|
// without leader contact before being considered unhealthy.
|
2017-03-10 00:43:07 +00:00
|
|
|
LastContactThreshold *ReadableDuration
|
2017-03-01 22:04:40 +00:00
|
|
|
|
|
|
|
// MaxTrailingLogs is the amount of entries in the Raft Log that a server can
|
|
|
|
// be behind before being considered unhealthy.
|
|
|
|
MaxTrailingLogs uint64
|
|
|
|
|
|
|
|
// ServerStabilizationTime is the minimum amount of time a server must be
|
|
|
|
// in a stable, healthy state before it can be added to the cluster. Only
|
|
|
|
// applicable with Raft protocol version 3 or higher.
|
2017-03-10 00:43:07 +00:00
|
|
|
ServerStabilizationTime *ReadableDuration
|
2017-03-01 22:04:40 +00:00
|
|
|
|
2017-03-21 23:36:44 +00:00
|
|
|
// (Enterprise-only) RedundancyZoneTag is the node tag to use for separating
|
|
|
|
// servers into zones for redundancy. If left blank, this feature will be disabled.
|
|
|
|
RedundancyZoneTag string
|
|
|
|
|
|
|
|
// (Enterprise-only) DisableUpgradeMigration will disable Autopilot's upgrade migration
|
|
|
|
// strategy of waiting until enough newer-versioned servers have been added to the
|
|
|
|
// cluster before promoting them to voters.
|
|
|
|
DisableUpgradeMigration bool
|
|
|
|
|
2017-02-24 21:08:49 +00:00
|
|
|
// CreateIndex holds the index corresponding the creation of this configuration.
|
|
|
|
// This is a read-only field.
|
|
|
|
CreateIndex uint64
|
|
|
|
|
2017-02-28 22:23:14 +00:00
|
|
|
// ModifyIndex will be set to the index of the last update when retrieving the
|
|
|
|
// Autopilot configuration. Resubmitting a configuration with
|
|
|
|
// AutopilotCASConfiguration will perform a check-and-set operation which ensures
|
|
|
|
// there hasn't been a subsequent update since the configuration was retrieved.
|
2017-02-24 21:08:49 +00:00
|
|
|
ModifyIndex uint64
|
2017-02-24 05:00:15 +00:00
|
|
|
}
|
|
|
|
|
2017-03-10 00:43:07 +00:00
|
|
|
// ServerHealth is the health (from the leader's point of view) of a server.
|
|
|
|
type ServerHealth struct {
|
|
|
|
// ID is the raft ID of the server.
|
|
|
|
ID string
|
|
|
|
|
|
|
|
// Name is the node name of the server.
|
|
|
|
Name string
|
|
|
|
|
2017-03-16 01:27:17 +00:00
|
|
|
// Address is the address of the server.
|
|
|
|
Address string
|
|
|
|
|
2017-03-10 00:43:07 +00:00
|
|
|
// The status of the SerfHealth check for the server.
|
|
|
|
SerfStatus string
|
|
|
|
|
2017-03-21 23:36:44 +00:00
|
|
|
// Version is the Consul version of the server.
|
|
|
|
Version string
|
|
|
|
|
|
|
|
// Leader is whether this server is currently the leader.
|
|
|
|
Leader bool
|
|
|
|
|
2017-03-10 00:43:07 +00:00
|
|
|
// LastContact is the time since this node's last contact with the leader.
|
|
|
|
LastContact *ReadableDuration
|
|
|
|
|
|
|
|
// LastTerm is the highest leader term this server has a record of in its Raft log.
|
|
|
|
LastTerm uint64
|
|
|
|
|
|
|
|
// LastIndex is the last log index this server has a record of in its Raft log.
|
|
|
|
LastIndex uint64
|
|
|
|
|
|
|
|
// Healthy is whether or not the server is healthy according to the current
|
|
|
|
// Autopilot config.
|
|
|
|
Healthy bool
|
|
|
|
|
2017-03-15 23:09:55 +00:00
|
|
|
// Voter is whether this is a voting server.
|
|
|
|
Voter bool
|
|
|
|
|
2017-03-10 00:43:07 +00:00
|
|
|
// StableSince is the last time this server's Healthy value changed.
|
|
|
|
StableSince time.Time
|
|
|
|
}
|
|
|
|
|
|
|
|
// OperatorHealthReply is a representation of the overall health of the cluster
|
|
|
|
type OperatorHealthReply struct {
|
|
|
|
// Healthy is true if all the servers in the cluster are healthy.
|
|
|
|
Healthy bool
|
|
|
|
|
|
|
|
// FailureTolerance is the number of healthy servers that could be lost without
|
|
|
|
// an outage occurring.
|
|
|
|
FailureTolerance int
|
|
|
|
|
|
|
|
// Servers holds the health of each server.
|
|
|
|
Servers []ServerHealth
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReadableDuration is a duration type that is serialized to JSON in human readable format.
|
|
|
|
type ReadableDuration time.Duration
|
|
|
|
|
|
|
|
func NewReadableDuration(dur time.Duration) *ReadableDuration {
|
|
|
|
d := ReadableDuration(dur)
|
|
|
|
return &d
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *ReadableDuration) String() string { return d.Duration().String() }
|
|
|
|
func (d *ReadableDuration) Duration() time.Duration {
|
|
|
|
if d == nil {
|
|
|
|
return time.Duration(0)
|
|
|
|
}
|
|
|
|
return time.Duration(*d)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *ReadableDuration) MarshalJSON() ([]byte, error) {
|
|
|
|
return []byte(fmt.Sprintf(`"%s"`, d.Duration().String())), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *ReadableDuration) UnmarshalJSON(raw []byte) error {
|
|
|
|
if d == nil {
|
|
|
|
return fmt.Errorf("cannot unmarshal to nil pointer")
|
|
|
|
}
|
|
|
|
|
|
|
|
str := string(raw)
|
|
|
|
if len(str) < 2 || str[0] != '"' || str[len(str)-1] != '"' {
|
|
|
|
return fmt.Errorf("must be enclosed with quotes: %s", str)
|
|
|
|
}
|
|
|
|
dur, err := time.ParseDuration(str[1 : len(str)-1])
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
*d = ReadableDuration(dur)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-08-30 02:09:57 +00:00
|
|
|
// RaftGetConfiguration is used to query the current Raft peer set.
|
|
|
|
func (op *Operator) RaftGetConfiguration(q *QueryOptions) (*RaftConfiguration, error) {
|
|
|
|
r := op.c.newRequest("GET", "/v1/operator/raft/configuration")
|
|
|
|
r.setQueryOptions(q)
|
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var out RaftConfiguration
|
|
|
|
if err := decodeBody(resp, &out); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &out, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// RaftRemovePeerByAddress is used to kick a stale peer (one that it in the Raft
|
|
|
|
// quorum but no longer known to Serf or the catalog) by address in the form of
|
|
|
|
// "IP:port".
|
2016-08-30 21:59:16 +00:00
|
|
|
func (op *Operator) RaftRemovePeerByAddress(address string, q *WriteOptions) error {
|
2016-08-30 02:09:57 +00:00
|
|
|
r := op.c.newRequest("DELETE", "/v1/operator/raft/peer")
|
|
|
|
r.setWriteOptions(q)
|
|
|
|
|
|
|
|
// TODO (slackpad) Currently we made address a query parameter. Once
|
2016-08-30 07:10:46 +00:00
|
|
|
// IDs are in place this will be DELETE /v1/operator/raft/peer/<id>.
|
2016-08-30 02:09:57 +00:00
|
|
|
r.params.Set("address", string(address))
|
|
|
|
|
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
resp.Body.Close()
|
|
|
|
return nil
|
|
|
|
}
|
2016-11-15 02:54:37 +00:00
|
|
|
|
|
|
|
// KeyringInstall is used to install a new gossip encryption key into the cluster
|
2016-11-22 23:24:40 +00:00
|
|
|
func (op *Operator) KeyringInstall(key string, q *WriteOptions) error {
|
|
|
|
r := op.c.newRequest("POST", "/v1/operator/keyring")
|
|
|
|
r.setWriteOptions(q)
|
|
|
|
r.obj = keyringRequest{
|
2016-11-15 02:54:37 +00:00
|
|
|
Key: key,
|
|
|
|
}
|
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
resp.Body.Close()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// KeyringList is used to list the gossip keys installed in the cluster
|
2016-11-22 23:24:40 +00:00
|
|
|
func (op *Operator) KeyringList(q *QueryOptions) ([]*KeyringResponse, error) {
|
|
|
|
r := op.c.newRequest("GET", "/v1/operator/keyring")
|
|
|
|
r.setQueryOptions(q)
|
2016-11-15 02:54:37 +00:00
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var out []*KeyringResponse
|
|
|
|
if err := decodeBody(resp, &out); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return out, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// KeyringRemove is used to remove a gossip encryption key from the cluster
|
2016-11-22 23:24:40 +00:00
|
|
|
func (op *Operator) KeyringRemove(key string, q *WriteOptions) error {
|
|
|
|
r := op.c.newRequest("DELETE", "/v1/operator/keyring")
|
|
|
|
r.setWriteOptions(q)
|
|
|
|
r.obj = keyringRequest{
|
2016-11-15 02:54:37 +00:00
|
|
|
Key: key,
|
|
|
|
}
|
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
resp.Body.Close()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// KeyringUse is used to change the active gossip encryption key
|
2016-11-22 23:24:40 +00:00
|
|
|
func (op *Operator) KeyringUse(key string, q *WriteOptions) error {
|
|
|
|
r := op.c.newRequest("PUT", "/v1/operator/keyring")
|
|
|
|
r.setWriteOptions(q)
|
|
|
|
r.obj = keyringRequest{
|
2016-11-15 02:54:37 +00:00
|
|
|
Key: key,
|
|
|
|
}
|
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
resp.Body.Close()
|
|
|
|
return nil
|
|
|
|
}
|
2017-02-24 05:00:15 +00:00
|
|
|
|
2017-02-24 21:08:49 +00:00
|
|
|
// AutopilotGetConfiguration is used to query the current Autopilot configuration.
|
2017-02-24 05:00:15 +00:00
|
|
|
func (op *Operator) AutopilotGetConfiguration(q *QueryOptions) (*AutopilotConfiguration, error) {
|
|
|
|
r := op.c.newRequest("GET", "/v1/operator/autopilot/configuration")
|
|
|
|
r.setQueryOptions(q)
|
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var out AutopilotConfiguration
|
|
|
|
if err := decodeBody(resp, &out); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-03-10 00:43:07 +00:00
|
|
|
|
2017-02-24 05:00:15 +00:00
|
|
|
return &out, nil
|
|
|
|
}
|
|
|
|
|
2017-02-24 21:08:49 +00:00
|
|
|
// AutopilotSetConfiguration is used to set the current Autopilot configuration.
|
2017-02-24 05:00:15 +00:00
|
|
|
func (op *Operator) AutopilotSetConfiguration(conf *AutopilotConfiguration, q *WriteOptions) error {
|
|
|
|
r := op.c.newRequest("PUT", "/v1/operator/autopilot/configuration")
|
|
|
|
r.setWriteOptions(q)
|
|
|
|
r.obj = conf
|
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
resp.Body.Close()
|
|
|
|
return nil
|
2017-02-24 21:08:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AutopilotCASConfiguration is used to perform a Check-And-Set update on the
|
|
|
|
// Autopilot configuration. The ModifyIndex value will be respected. Returns
|
|
|
|
// true on success or false on failures.
|
|
|
|
func (op *Operator) AutopilotCASConfiguration(conf *AutopilotConfiguration, q *WriteOptions) (bool, error) {
|
|
|
|
r := op.c.newRequest("PUT", "/v1/operator/autopilot/configuration")
|
|
|
|
r.setWriteOptions(q)
|
|
|
|
r.params.Set("cas", strconv.FormatUint(conf.ModifyIndex, 10))
|
|
|
|
r.obj = conf
|
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
if _, err := io.Copy(&buf, resp.Body); err != nil {
|
|
|
|
return false, fmt.Errorf("Failed to read response: %v", err)
|
|
|
|
}
|
|
|
|
res := strings.Contains(string(buf.Bytes()), "true")
|
|
|
|
|
|
|
|
return res, nil
|
2017-02-24 23:54:49 +00:00
|
|
|
}
|
2017-03-10 00:43:07 +00:00
|
|
|
|
|
|
|
// AutopilotServerHealth
|
|
|
|
func (op *Operator) AutopilotServerHealth(q *QueryOptions) (*OperatorHealthReply, error) {
|
|
|
|
r := op.c.newRequest("GET", "/v1/operator/autopilot/health")
|
|
|
|
r.setQueryOptions(q)
|
|
|
|
_, resp, err := requireOK(op.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var out OperatorHealthReply
|
|
|
|
if err := decodeBody(resp, &out); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &out, nil
|
|
|
|
}
|