agent: add allowStale option for HTTP API (#3142)

This patch adds an "allowStale" option to the HTTP API
configuration which allows stale reads to provide linear
read scalability.

Fixes #3142
pull/3175/head
wojtkiewicz 2017-06-16 10:55:53 +02:00 committed by Frank Schroeder
parent 37785028be
commit 1e0fd27a74
No known key found for this signature in database
GPG Key ID: 4D65C6EAEC87DECD
4 changed files with 55 additions and 28 deletions

View File

@ -131,6 +131,12 @@ type DNSConfig struct {
// HTTPConfig is used to fine tune the Http sub-system. // HTTPConfig is used to fine tune the Http sub-system.
type HTTPConfig struct { type HTTPConfig struct {
// AllowStale is used to enable lookups with stale
// data. This gives horizontal read scalability since
// any Consul server can service the query instead of
// only the leader.
AllowStale *bool `mapstructure:"allow_stale"`
// ResponseHeaders are used to add HTTP header response fields to the HTTP API responses. // ResponseHeaders are used to add HTTP header response fields to the HTTP API responses.
ResponseHeaders map[string]string `mapstructure:"response_headers"` ResponseHeaders map[string]string `mapstructure:"response_headers"`
} }
@ -918,6 +924,9 @@ func DefaultConfig() *Config {
MaxStale: 10 * 365 * 24 * time.Hour, MaxStale: 10 * 365 * 24 * time.Hour,
RecursorTimeout: 2 * time.Second, RecursorTimeout: 2 * time.Second,
}, },
HTTPConfig: HTTPConfig{
AllowStale: Bool(true),
},
Telemetry: Telemetry{ Telemetry: Telemetry{
StatsitePrefix: "consul", StatsitePrefix: "consul",
}, },
@ -2002,6 +2011,9 @@ func MergeConfig(a, b *Config) *Config {
result.HTTPConfig.ResponseHeaders[field] = value result.HTTPConfig.ResponseHeaders[field] = value
} }
} }
if b.HTTPConfig.AllowStale != nil {
result.HTTPConfig.AllowStale = b.HTTPConfig.AllowStale
}
if len(b.Meta) != 0 { if len(b.Meta) != 0 {
if result.Meta == nil { if result.Meta == nil {
result.Meta = make(map[string]string) result.Meta = make(map[string]string)

View File

@ -358,15 +358,17 @@ func parseWait(resp http.ResponseWriter, req *http.Request, b *structs.QueryOpti
} }
// parseConsistency is used to parse the ?stale and ?consistent query params. // parseConsistency is used to parse the ?stale and ?consistent query params.
// allowStale forces stale consistency instead of default one if none was provided in the query.
// Returns true on error // Returns true on error
func parseConsistency(resp http.ResponseWriter, req *http.Request, b *structs.QueryOptions) bool { func parseConsistency(resp http.ResponseWriter, req *http.Request, allowStale bool, b *structs.QueryOptions) bool {
query := req.URL.Query() query := req.URL.Query()
if _, ok := query["stale"]; ok {
b.AllowStale = true
}
if _, ok := query["consistent"]; ok { if _, ok := query["consistent"]; ok {
b.RequireConsistent = true b.RequireConsistent = true
} }
b.AllowStale = !b.RequireConsistent && allowStale
if _, ok := query["stale"]; ok {
b.AllowStale = true
}
if b.AllowStale && b.RequireConsistent { if b.AllowStale && b.RequireConsistent {
resp.WriteHeader(http.StatusBadRequest) // 400 resp.WriteHeader(http.StatusBadRequest) // 400
fmt.Fprint(resp, "Cannot specify ?stale with ?consistent, conflicting semantics.") fmt.Fprint(resp, "Cannot specify ?stale with ?consistent, conflicting semantics.")
@ -433,7 +435,8 @@ func (s *HTTPServer) parseMetaFilter(req *http.Request) map[string]string {
func (s *HTTPServer) parse(resp http.ResponseWriter, req *http.Request, dc *string, b *structs.QueryOptions) bool { func (s *HTTPServer) parse(resp http.ResponseWriter, req *http.Request, dc *string, b *structs.QueryOptions) bool {
s.parseDC(req, dc) s.parseDC(req, dc)
s.parseToken(req, &b.Token) s.parseToken(req, &b.Token)
if parseConsistency(resp, req, b) { allowStale := s.agent.config.HTTPConfig.AllowStale
if parseConsistency(resp, req, *allowStale, b) {
return true return true
} }
return parseWait(resp, req, b) return parseWait(resp, req, b)

View File

@ -436,31 +436,36 @@ func TestParseWait_InvalidIndex(t *testing.T) {
func TestParseConsistency(t *testing.T) { func TestParseConsistency(t *testing.T) {
t.Parallel() t.Parallel()
resp := httptest.NewRecorder() resp := httptest.NewRecorder()
var b structs.QueryOptions
req, _ := http.NewRequest("GET", "/v1/catalog/nodes?stale", nil) tests := []struct {
if d := parseConsistency(resp, req, &b); d { url string
t.Fatalf("unexpected done") allowStale bool
wantAllowStale bool
wantRequireConsistent bool
}{
{"/v1/catalog/nodes?stale", false, true, false},
{"/v1/catalog/nodes?stale", true, true, false},
{"/v1/catalog/nodes?consistent", false, false, true},
{"/v1/catalog/nodes?consistent", true, false, true},
{"/v1/catalog/nodes", false, false, false},
{"/v1/catalog/nodes", true, true, false},
} }
if !b.AllowStale { for _, tt := range tests {
t.Fatalf("Bad: %v", b) name := fmt.Sprintf("url=%v, HTTP.AllowStale=%v", tt.url, tt.allowStale)
} t.Run(name, func(t *testing.T) {
if b.RequireConsistent { var q structs.QueryOptions
t.Fatalf("Bad: %v", b) req, _ := http.NewRequest("GET", tt.url, nil)
} if d := parseConsistency(resp, req, tt.allowStale, &q); d {
t.Fatalf("Failed to parse consistency.")
b = structs.QueryOptions{} }
req, _ = http.NewRequest("GET", "/v1/catalog/nodes?consistent", nil) if got, want := q.AllowStale, tt.wantAllowStale; got != want {
if d := parseConsistency(resp, req, &b); d { t.Fatalf("got allowStale %v want %v", got, want)
t.Fatalf("unexpected done") }
} if got, want := q.RequireConsistent, tt.wantRequireConsistent; got != want {
t.Fatalf("got requireConsistent %v want %v", got, want)
if b.AllowStale { }
t.Fatalf("Bad: %v", b) })
}
if !b.RequireConsistent {
t.Fatalf("Bad: %v", b)
} }
} }
@ -470,7 +475,7 @@ func TestParseConsistency_Invalid(t *testing.T) {
var b structs.QueryOptions var b structs.QueryOptions
req, _ := http.NewRequest("GET", "/v1/catalog/nodes?stale&consistent", nil) req, _ := http.NewRequest("GET", "/v1/catalog/nodes?stale&consistent", nil)
if d := parseConsistency(resp, req, &b); !d { if d := parseConsistency(resp, req, false, &b); !d {
t.Fatalf("expected done") t.Fatalf("expected done")
} }

View File

@ -753,6 +753,13 @@ Consul will not enable TLS for the HTTP API unless the `https` port has been ass
<br><br> <br><br>
The following sub-keys are available: The following sub-keys are available:
* <a name="allow_stale"></a><a href="#allow_stale">`allow_stale`</a> - Enables a stale query
for DNS information. This allows any Consul server, rather than only the leader, to service
the request. The advantage of this is you get linear read scalability with Consul servers.
In versions of Consul prior to 0.7, this defaulted to false, meaning all requests are serviced
by the leader, providing stronger consistency but less throughput and higher latency. In Consul
0.7 and later, this defaults to true for better utilization of available servers.
* <a name="response_headers"></a><a href="#response_headers">`response_headers`</a> * <a name="response_headers"></a><a href="#response_headers">`response_headers`</a>
This object allows adding headers to the HTTP API responses. This object allows adding headers to the HTTP API responses.
For example, the following config can be used to enable For example, the following config can be used to enable