Allow services register command to register an unnamed check

The logic in parsing data files and converting them to data structures
accidentally removed healthchecks with no Name field, even though we
explicitly state in API documentation that is allowed.

We remove the check for "len(results.Checks) == 1" because if the length
of the array is more than 0, we know that it is not a zero value array.
This allows us to register a singular, unnamed check via the CLI.

Fixes #6796
This commit is contained in:
Chris Piraino 2019-11-15 09:06:33 -06:00 committed by Chris Piraino
parent 976249be46
commit 646da639ee
3 changed files with 66 additions and 5 deletions

View File

@ -65,13 +65,10 @@ func serviceToAgentService(svc *structs.ServiceDefinition) (*api.AgentServiceReg
// The structs version has non-pointer checks and the destination // The structs version has non-pointer checks and the destination
// has pointers, so we need to set the destination to nil if there // has pointers, so we need to set the destination to nil if there
// is no check ID set. // is a zero-value Check field.
if result.Check != nil && result.Check.Name == "" { if result.Check != nil && reflect.DeepEqual(*result.Check, api.AgentServiceCheck{}) {
result.Check = nil result.Check = nil
} }
if len(result.Checks) == 1 && result.Checks[0].Name == "" {
result.Checks = nil
}
return &result, nil return &result, nil
} }

View File

@ -2,6 +2,7 @@ package services
import ( import (
"testing" "testing"
"time"
"github.com/hashicorp/consul/agent/config" "github.com/hashicorp/consul/agent/config"
"github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/agent/structs"
@ -63,6 +64,32 @@ func TestStructsToAgentService(t *testing.T) {
}, },
}, },
}, },
{
"Service with an unnamed check",
&structs.ServiceDefinition{
Name: "web",
Check: structs.CheckType{
TTL: 5 * time.Second,
},
},
&api.AgentServiceRegistration{
Name: "web",
Check: &api.AgentServiceCheck{
TTL: "5s",
},
},
},
{
"Service with a zero-value check",
&structs.ServiceDefinition{
Name: "web",
Check: structs.CheckType{},
},
&api.AgentServiceRegistration{
Name: "web",
Check: nil,
},
},
{ {
"Service with checks", "Service with checks",
&structs.ServiceDefinition{ &structs.ServiceDefinition{

View File

@ -153,6 +153,43 @@ func TestCommand_Flags_TaggedAddresses(t *testing.T) {
require.Equal(svc.TaggedAddresses["v6"].Port, 1234) require.Equal(svc.TaggedAddresses["v6"].Port, 1234)
} }
func TestCommand_FileWithUnnamedCheck(t *testing.T) {
t.Parallel()
require := require.New(t)
a := agent.NewTestAgent(t, t.Name(), ``)
defer a.Shutdown()
client := a.Client()
ui := cli.NewMockUi()
c := New(ui)
contents := `{ "Service": { "Name": "web", "Check": { "TTL": "10s" } } }`
f := testFile(t, "json")
defer os.Remove(f.Name())
if _, err := f.WriteString(contents); err != nil {
t.Fatalf("err: %#v", err)
}
args := []string{
"-http-addr=" + a.HTTPAddr(),
f.Name(),
}
require.Equal(0, c.Run(args), ui.ErrorWriter.String())
svcs, err := client.Agent().Services()
require.NoError(err)
require.Len(svcs, 1)
svc := svcs["web"]
require.NotNil(svc)
checks, err := client.Agent().Checks()
require.NoError(err)
require.Len(checks, 1)
}
func testFile(t *testing.T, suffix string) *os.File { func testFile(t *testing.T, suffix string) *os.File {
f := testutil.TempFile(t, "register-test-file") f := testutil.TempFile(t, "register-test-file")
if err := f.Close(); err != nil { if err := f.Close(); err != nil {