agent: maintenance mode api's are idempotent

pull/606/head
Ryan Uber 2015-01-15 10:51:00 -08:00
parent 8819d71f99
commit 9ee1e6e858
4 changed files with 32 additions and 75 deletions

View File

@ -1002,23 +1002,14 @@ func (a *Agent) unloadChecks() error {
// EnableServiceMaintenance will register a false health check against the given // EnableServiceMaintenance will register a false health check against the given
// service ID with critical status. This will exclude the service from queries. // service ID with critical status. This will exclude the service from queries.
func (a *Agent) EnableServiceMaintenance(serviceID string) error { func (a *Agent) EnableServiceMaintenance(serviceID string) error {
var service *structs.NodeService service, ok := a.state.Services()[serviceID]
for _, svc := range a.state.Services() { if !ok {
if svc.ID == serviceID {
service = svc
}
}
// Ensure the service exists
if service == nil {
return fmt.Errorf("No service registered with ID %q", serviceID) return fmt.Errorf("No service registered with ID %q", serviceID)
} }
// Ensure maintenance mode is not already enabled // Ensure maintenance mode is not already enabled
for _, check := range a.state.Checks() { if _, ok := a.state.Checks()[maintCheckID]; ok {
if check.CheckID == maintCheckID { return nil
return fmt.Errorf("Maintenance mode already enabled for service %q", serviceID)
}
} }
// Create and register the critical health check // Create and register the critical health check
@ -1039,29 +1030,11 @@ func (a *Agent) EnableServiceMaintenance(serviceID string) error {
// DisableServiceMaintenance will deregister the fake maintenance mode check // DisableServiceMaintenance will deregister the fake maintenance mode check
// if the service has been marked as in maintenance. // if the service has been marked as in maintenance.
func (a *Agent) DisableServiceMaintenance(serviceID string) error { func (a *Agent) DisableServiceMaintenance(serviceID string) error {
var service *structs.NodeService if _, ok := a.state.Services()[serviceID]; !ok {
for _, svc := range a.state.Services() {
if svc.ID == serviceID {
service = svc
}
}
// Ensure the service exists
if service == nil {
return fmt.Errorf("No service registered with ID %q", serviceID) return fmt.Errorf("No service registered with ID %q", serviceID)
} }
// Ensure maintenance mode is enabled
for _, check := range a.state.Checks() {
if check.CheckID == maintCheckID {
goto DEREGISTER
}
}
return fmt.Errorf("Maintenance mode not enabled for service %q", serviceID)
DEREGISTER:
// Deregister the maintenance check // Deregister the maintenance check
a.RemoveCheck(maintCheckID, true) a.RemoveCheck(maintCheckID, true)
return nil return nil
} }

View File

@ -216,12 +216,12 @@ func (s *HTTPServer) AgentServiceMaintenance(resp http.ResponseWriter, req *http
var err error var err error
if enable { if enable {
if err = s.agent.EnableServiceMaintenance(serviceID); err != nil { if err = s.agent.EnableServiceMaintenance(serviceID); err != nil {
resp.WriteHeader(409) resp.WriteHeader(404)
resp.Write([]byte(err.Error())) resp.Write([]byte(err.Error()))
} }
} else { } else {
if err = s.agent.DisableServiceMaintenance(serviceID); err != nil { if err = s.agent.DisableServiceMaintenance(serviceID); err != nil {
resp.WriteHeader(409) resp.WriteHeader(404)
resp.Write([]byte(err.Error())) resp.Write([]byte(err.Error()))
} }
} }

View File

@ -507,7 +507,7 @@ func TestHTTPAgent_ServiceMaintenanceEndpoint_BadRequest(t *testing.T) {
t.Fatalf("err: %s", err) t.Fatalf("err: %s", err)
} }
if resp.Code != 405 { if resp.Code != 405 {
t.Fatalf("expected 405 for non-PUT request") t.Fatalf("expected 405, got %d", resp.Code)
} }
// Fails when no enable flag provided // Fails when no enable flag provided
@ -517,7 +517,7 @@ func TestHTTPAgent_ServiceMaintenanceEndpoint_BadRequest(t *testing.T) {
t.Fatalf("err: %s", err) t.Fatalf("err: %s", err)
} }
if resp.Code != 400 { if resp.Code != 400 {
t.Fatalf("expected 400 for missing enable flag") t.Fatalf("expected 400, got %d", resp.Code)
} }
// Fails when no service ID provided // Fails when no service ID provided
@ -527,7 +527,17 @@ func TestHTTPAgent_ServiceMaintenanceEndpoint_BadRequest(t *testing.T) {
t.Fatalf("err: %s", err) t.Fatalf("err: %s", err)
} }
if resp.Code != 400 { if resp.Code != 400 {
t.Fatalf("expected 400 for missing service ID") t.Fatalf("expected 400, got %d", resp.Code)
}
// Fails when bad service ID provided
req, _ = http.NewRequest("PUT", "/v1/agent/service/maintenance/_nope_?enable=true", nil)
resp = httptest.NewRecorder()
if _, err := srv.AgentServiceMaintenance(resp, req); err == nil {
t.Fatalf("should have errored")
}
if resp.Code != 404 {
t.Fatalf("expected 404, got %d", resp.Code)
} }
} }
@ -546,29 +556,9 @@ func TestHTTPAgent_EnableServiceMaintenance(t *testing.T) {
t.Fatalf("err: %v", err) t.Fatalf("err: %v", err)
} }
// Force into maintenance mode // Force the service into maintenance mode
if err := srv.agent.EnableServiceMaintenance("test"); err != nil {
t.Fatalf("err: %s", err)
}
// Fails when service is already in maintenance mode
req, _ := http.NewRequest("PUT", "/v1/agent/service/maintenance/test?enable=true", nil) req, _ := http.NewRequest("PUT", "/v1/agent/service/maintenance/test?enable=true", nil)
resp := httptest.NewRecorder() resp := httptest.NewRecorder()
if _, err := srv.AgentServiceMaintenance(resp, req); err == nil {
t.Fatalf("should have errored")
}
if resp.Code != 409 {
t.Fatalf("expected 409, got %d", resp.Code)
}
// Remove maintenance mode
if err := srv.agent.DisableServiceMaintenance("test"); err != nil {
t.Fatalf("err: %s", err)
}
// Force the service into maintenance mode
req, _ = http.NewRequest("PUT", "/v1/agent/service/maintenance/test?enable=true", nil)
resp = httptest.NewRecorder()
if _, err := srv.AgentServiceMaintenance(resp, req); err != nil { if _, err := srv.AgentServiceMaintenance(resp, req); err != nil {
t.Fatalf("err: %s", err) t.Fatalf("err: %s", err)
} }
@ -597,28 +587,23 @@ func TestHTTPAgent_DisableServiceMaintenance(t *testing.T) {
t.Fatalf("err: %v", err) t.Fatalf("err: %v", err)
} }
// Fails when the service is not in maintenance mode
req, _ := http.NewRequest("PUT", "/v1/agent/service/maintenance/test?enable=false", nil)
resp := httptest.NewRecorder()
if _, err := srv.AgentServiceMaintenance(resp, req); err == nil {
t.Fatalf("should have failed")
}
if resp.Code != 409 {
t.Fatalf("expected 409, got %d", resp.Code)
}
// Force the service into maintenance mode // Force the service into maintenance mode
if err := srv.agent.EnableServiceMaintenance("test"); err != nil { if err := srv.agent.EnableServiceMaintenance("test"); err != nil {
t.Fatalf("err: %s", err) t.Fatalf("err: %s", err)
} }
// Leave maintenance mode // Leave maintenance mode
req, _ = http.NewRequest("PUT", "/v1/agent/service/maintenance/test?enable=false", nil) req, _ := http.NewRequest("PUT", "/v1/agent/service/maintenance/test?enable=false", nil)
resp = httptest.NewRecorder() resp := httptest.NewRecorder()
if _, err := srv.AgentServiceMaintenance(resp, req); err != nil { if _, err := srv.AgentServiceMaintenance(resp, req); err != nil {
t.Fatalf("err: %s", err) t.Fatalf("err: %s", err)
} }
if resp.Code != 200 { if resp.Code != 200 {
t.Fatalf("expected 200, got %d", resp.Code) t.Fatalf("expected 200, got %d", resp.Code)
} }
// Ensure the maintenance check was removed
if _, ok := srv.agent.state.Checks()[maintCheckID]; ok {
t.Fatalf("should have removed maintenance check")
}
} }

View File

@ -553,13 +553,12 @@ The return code is 200 on success.
The service maintenance endpoint allows placing a given service into The service maintenance endpoint allows placing a given service into
"maintenance mode". During maintenance mode, the service will be marked as "maintenance mode". During maintenance mode, the service will be marked as
unavailable, and will not be present in DNS or API queries. unavailable, and will not be present in DNS or API queries. This API call is
idempotent. Maintenance mode is persistent and will be automatically restored
on agent restart.
The `?enable` flag is required, and its value must be `true` (to enter The `?enable` flag is required, and its value must be `true` (to enter
maintenance mode), or `false` (to resume normal operation). It is an error to maintenance mode), or `false` (to resume normal operation).
enable maintenance mode while it is already enabled, or disable it while it is
already disabled. You will receive a 409 if either of these conflicts are
observed.
The return code is 200 on success. The return code is 200 on success.