From b315e79cfe918e3726545f8375ebca73bace0da5 Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Thu, 27 Sep 2018 23:52:17 -0700 Subject: [PATCH 01/12] command/services --- command/services/register/config.go | 21 +++++ command/services/register/config_test.go | 43 +++++++++ command/services/register/register.go | 107 +++++++++++++++++++++++ command/services/services.go | 35 ++++++++ command/services/services_test.go | 13 +++ 5 files changed, 219 insertions(+) create mode 100644 command/services/register/config.go create mode 100644 command/services/register/config_test.go create mode 100644 command/services/register/register.go create mode 100644 command/services/services.go create mode 100644 command/services/services_test.go diff --git a/command/services/register/config.go b/command/services/register/config.go new file mode 100644 index 0000000000..fc01672710 --- /dev/null +++ b/command/services/register/config.go @@ -0,0 +1,21 @@ +package register + +import ( + "fmt" + "github.com/hashicorp/consul/agent/config" + "github.com/hashicorp/consul/api" + "github.com/mitchellh/mapstructure" +) + +// configToAgentService converts a ServiceDefinition struct to an +// AgentServiceRegistration API struct. +func configToAgentService(svc *config.ServiceDefinition) (*api.AgentServiceRegistration, error) { + var result api.AgentServiceRegistration + var m map[string]interface{} + err := mapstructure.Decode(svc, &m) + if err == nil { + println(fmt.Sprintf("%#v", m)) + err = mapstructure.Decode(m, &result) + } + return &result, err +} diff --git a/command/services/register/config_test.go b/command/services/register/config_test.go new file mode 100644 index 0000000000..17349a2f80 --- /dev/null +++ b/command/services/register/config_test.go @@ -0,0 +1,43 @@ +package register + +import ( + "testing" + + "github.com/hashicorp/consul/agent/config" + "github.com/hashicorp/consul/api" + "github.com/stretchr/testify/require" +) + +func TestConfigToAgentService(t *testing.T) { + cases := []struct { + Name string + Input *config.ServiceDefinition + Output *api.AgentServiceRegistration + }{ + { + "Basic service with port", + &config.ServiceDefinition{ + Name: strPtr("web"), + Tags: []string{"leader"}, + Port: intPtr(1234), + }, + &api.AgentServiceRegistration{ + Name: "web", + Tags: []string{"leader"}, + Port: 1234, + }, + }, + } + + for _, tc := range cases { + t.Run(tc.Name, func(t *testing.T) { + require := require.New(t) + actual, err := configToAgentService(tc.Input) + require.NoError(err) + require.Equal(tc.Output, actual) + }) + } +} + +func intPtr(v int) *int { return &v } +func strPtr(v string) *string { return &v } diff --git a/command/services/register/register.go b/command/services/register/register.go new file mode 100644 index 0000000000..0ceb24c40d --- /dev/null +++ b/command/services/register/register.go @@ -0,0 +1,107 @@ +package register + +import ( + "flag" + + //"github.com/hashicorp/consul/api" + "github.com/hashicorp/consul/command/flags" + "github.com/mitchellh/cli" +) + +func New(ui cli.Ui) *cmd { + c := &cmd{UI: ui} + c.init() + return c +} + +type cmd struct { + UI cli.Ui + flags *flag.FlagSet + http *flags.HTTPFlags + help string + + // flags + flagMeta map[string]string +} + +func (c *cmd) init() { + c.flags = flag.NewFlagSet("", flag.ContinueOnError) + c.flags.Var((*flags.FlagMapValue)(&c.flagMeta), "meta", + "Metadata to set on the intention, formatted as key=value. This flag "+ + "may be specified multiple times to set multiple meta fields.") + + c.http = &flags.HTTPFlags{} + flags.Merge(c.flags, c.http.ClientFlags()) + flags.Merge(c.flags, c.http.ServerFlags()) + c.help = flags.Usage(help, c.flags) +} + +func (c *cmd) Run(args []string) int { + if err := c.flags.Parse(args); err != nil { + return 1 + } + + // Check for arg validation + args = c.flags.Args() + if len(args) == 0 { + c.UI.Error("Service registration requires at least one argument.") + return 1 + } + + /* + ixns, err := c.ixnsFromArgs(args) + if err != nil { + c.UI.Error(fmt.Sprintf("Error: %s", err)) + return 1 + } + + // Create and test the HTTP client + /* + client, err := c.http.APIClient() + if err != nil { + c.UI.Error(fmt.Sprintf("Error connecting to Consul agent: %s", err)) + return 1 + } + */ + + return 0 +} + +func (c *cmd) Synopsis() string { + return synopsis +} + +func (c *cmd) Help() string { + return c.help +} + +const synopsis = "Create intentions for service connections." +const help = ` +Usage: consul intention create [options] SRC DST +Usage: consul intention create [options] -file FILE... + + Create one or more intentions. The data can be specified as a single + source and destination pair or via a set of files when the "-file" flag + is specified. + + $ consul intention create web db + + To consume data from a set of files: + + $ consul intention create -file one.json two.json + + When specifying the "-file" flag, "-" may be used once to read from stdin: + + $ echo "{ ... }" | consul intention create -file - + + An "allow" intention is created by default (whitelist). To create a + "deny" intention, the "-deny" flag should be specified. + + If a conflicting intention is found, creation will fail. To replace any + conflicting intentions, specify the "-replace" flag. This will replace any + conflicting intentions with the intention specified in this command. + Metadata and any other fields of the previous intention will not be + preserved. + + Additional flags and more advanced use cases are detailed below. +` diff --git a/command/services/services.go b/command/services/services.go new file mode 100644 index 0000000000..0e050d77d3 --- /dev/null +++ b/command/services/services.go @@ -0,0 +1,35 @@ +package services + +import ( + "github.com/hashicorp/consul/command/flags" + "github.com/mitchellh/cli" +) + +func New() *cmd { + return &cmd{} +} + +type cmd struct{} + +func (c *cmd) Run(args []string) int { + return cli.RunResultHelp +} + +func (c *cmd) Synopsis() string { + return synopsis +} + +func (c *cmd) Help() string { + return flags.Usage(help, nil) +} + +const synopsis = "Interact with services" +const help = ` +Usage: consul services [options] [args] + + This command has subcommands for interacting with services. The subcommands + default to working with services registered with the local agent. Please see + the "consul catalog" command for interacting with the entire catalog. + + For more examples, ask for subcommand help or view the documentation. +` diff --git a/command/services/services_test.go b/command/services/services_test.go new file mode 100644 index 0000000000..c7521718a3 --- /dev/null +++ b/command/services/services_test.go @@ -0,0 +1,13 @@ +package services + +import ( + "strings" + "testing" +) + +func TestCommand_noTabs(t *testing.T) { + t.Parallel() + if strings.ContainsRune(New().Help(), '\t') { + t.Fatal("help has tabs") + } +} From 3237047e7232a70460f7c1bc4dd916dc3cd9dbb0 Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Sun, 30 Sep 2018 18:19:41 -0700 Subject: [PATCH 02/12] vendor: update mapstructure to v1.1.0 We require this change to support struct to struct decoding. --- .../mitchellh/mapstructure/CHANGELOG.md | 12 + .../mitchellh/mapstructure/README.md | 2 +- .../mitchellh/mapstructure/decode_hooks.go | 65 +++ .../github.com/mitchellh/mapstructure/go.mod | 1 + .../mitchellh/mapstructure/mapstructure.go | 494 ++++++++++++++---- vendor/vendor.json | 2 +- 6 files changed, 480 insertions(+), 96 deletions(-) create mode 100644 vendor/github.com/mitchellh/mapstructure/CHANGELOG.md create mode 100644 vendor/github.com/mitchellh/mapstructure/go.mod diff --git a/vendor/github.com/mitchellh/mapstructure/CHANGELOG.md b/vendor/github.com/mitchellh/mapstructure/CHANGELOG.md new file mode 100644 index 0000000000..fb0f46c84b --- /dev/null +++ b/vendor/github.com/mitchellh/mapstructure/CHANGELOG.md @@ -0,0 +1,12 @@ +## 1.1.0 (September 30, 2018) + +* Added `StringToIPHookFunc` to convert `string` to `net.IP` and `net.IPNet` [GH-133] +* Support struct to struct decoding [GH-137] +* If source map value is nil, then destination map value is nil (instead of empty) +* If source slice value is nil, then destination slice value is nil (instead of empty) +* If source pointer is nil, then destination pointer is set to nil (instead of + allocated zero value of type) + +## 1.0.0 + +* Initial tagged stable release. diff --git a/vendor/github.com/mitchellh/mapstructure/README.md b/vendor/github.com/mitchellh/mapstructure/README.md index 659d6885fc..0018dc7d9f 100644 --- a/vendor/github.com/mitchellh/mapstructure/README.md +++ b/vendor/github.com/mitchellh/mapstructure/README.md @@ -1,4 +1,4 @@ -# mapstructure +# mapstructure [![Godoc](https://godoc.org/github.com/mitchellh/mapstructure?status.svg)](https://godoc.org/github.com/mitchellh/mapstructure) mapstructure is a Go library for decoding generic map values to structures and vice versa, while providing helpful error handling. diff --git a/vendor/github.com/mitchellh/mapstructure/decode_hooks.go b/vendor/github.com/mitchellh/mapstructure/decode_hooks.go index afcfd5eed6..1f0abc65ab 100644 --- a/vendor/github.com/mitchellh/mapstructure/decode_hooks.go +++ b/vendor/github.com/mitchellh/mapstructure/decode_hooks.go @@ -2,6 +2,8 @@ package mapstructure import ( "errors" + "fmt" + "net" "reflect" "strconv" "strings" @@ -115,6 +117,69 @@ func StringToTimeDurationHookFunc() DecodeHookFunc { } } +// StringToIPHookFunc returns a DecodeHookFunc that converts +// strings to net.IP +func StringToIPHookFunc() DecodeHookFunc { + return func( + f reflect.Type, + t reflect.Type, + data interface{}) (interface{}, error) { + if f.Kind() != reflect.String { + return data, nil + } + if t != reflect.TypeOf(net.IP{}) { + return data, nil + } + + // Convert it by parsing + ip := net.ParseIP(data.(string)) + if ip == nil { + return net.IP{}, fmt.Errorf("failed parsing ip %v", data) + } + + return ip, nil + } +} + +// StringToIPNetHookFunc returns a DecodeHookFunc that converts +// strings to net.IPNet +func StringToIPNetHookFunc() DecodeHookFunc { + return func( + f reflect.Type, + t reflect.Type, + data interface{}) (interface{}, error) { + if f.Kind() != reflect.String { + return data, nil + } + if t != reflect.TypeOf(net.IPNet{}) { + return data, nil + } + + // Convert it by parsing + _, net, err := net.ParseCIDR(data.(string)) + return net, err + } +} + +// StringToTimeHookFunc returns a DecodeHookFunc that converts +// strings to time.Time. +func StringToTimeHookFunc(layout string) DecodeHookFunc { + return func( + f reflect.Type, + t reflect.Type, + data interface{}) (interface{}, error) { + if f.Kind() != reflect.String { + return data, nil + } + if t != reflect.TypeOf(time.Time{}) { + return data, nil + } + + // Convert it by parsing + return time.Parse(layout, data.(string)) + } +} + // WeaklyTypedHook is a DecodeHookFunc which adds support for weak typing to // the decoder. // diff --git a/vendor/github.com/mitchellh/mapstructure/go.mod b/vendor/github.com/mitchellh/mapstructure/go.mod new file mode 100644 index 0000000000..d2a7125620 --- /dev/null +++ b/vendor/github.com/mitchellh/mapstructure/go.mod @@ -0,0 +1 @@ +module github.com/mitchellh/mapstructure diff --git a/vendor/github.com/mitchellh/mapstructure/mapstructure.go b/vendor/github.com/mitchellh/mapstructure/mapstructure.go index 30a9957c65..95eb686565 100644 --- a/vendor/github.com/mitchellh/mapstructure/mapstructure.go +++ b/vendor/github.com/mitchellh/mapstructure/mapstructure.go @@ -114,12 +114,12 @@ type Metadata struct { Unused []string } -// Decode takes a map and uses reflection to convert it into the -// given Go native structure. val must be a pointer to a struct. -func Decode(m interface{}, rawVal interface{}) error { +// Decode takes an input structure and uses reflection to translate it to +// the output structure. output must be a pointer to a map or struct. +func Decode(input interface{}, output interface{}) error { config := &DecoderConfig{ Metadata: nil, - Result: rawVal, + Result: output, } decoder, err := NewDecoder(config) @@ -127,7 +127,7 @@ func Decode(m interface{}, rawVal interface{}) error { return err } - return decoder.Decode(m) + return decoder.Decode(input) } // WeakDecode is the same as Decode but is shorthand to enable @@ -147,6 +147,40 @@ func WeakDecode(input, output interface{}) error { return decoder.Decode(input) } +// DecodeMetadata is the same as Decode, but is shorthand to +// enable metadata collection. See DecoderConfig for more info. +func DecodeMetadata(input interface{}, output interface{}, metadata *Metadata) error { + config := &DecoderConfig{ + Metadata: metadata, + Result: output, + } + + decoder, err := NewDecoder(config) + if err != nil { + return err + } + + return decoder.Decode(input) +} + +// WeakDecodeMetadata is the same as Decode, but is shorthand to +// enable both WeaklyTypedInput and metadata collection. See +// DecoderConfig for more info. +func WeakDecodeMetadata(input interface{}, output interface{}, metadata *Metadata) error { + config := &DecoderConfig{ + Metadata: metadata, + Result: output, + WeaklyTypedInput: true, + } + + decoder, err := NewDecoder(config) + if err != nil { + return err + } + + return decoder.Decode(input) +} + // NewDecoder returns a new decoder for the given configuration. Once // a decoder has been returned, the same configuration must not be used // again. @@ -184,68 +218,91 @@ func NewDecoder(config *DecoderConfig) (*Decoder, error) { // Decode decodes the given raw interface to the target pointer specified // by the configuration. -func (d *Decoder) Decode(raw interface{}) error { - return d.decode("", raw, reflect.ValueOf(d.config.Result).Elem()) +func (d *Decoder) Decode(input interface{}) error { + return d.decode("", input, reflect.ValueOf(d.config.Result).Elem()) } // Decodes an unknown data type into a specific reflection value. -func (d *Decoder) decode(name string, data interface{}, val reflect.Value) error { - if data == nil { - // If the data is nil, then we don't set anything. +func (d *Decoder) decode(name string, input interface{}, outVal reflect.Value) error { + var inputVal reflect.Value + if input != nil { + inputVal = reflect.ValueOf(input) + + // We need to check here if input is a typed nil. Typed nils won't + // match the "input == nil" below so we check that here. + if inputVal.Kind() == reflect.Ptr && inputVal.IsNil() { + input = nil + } + } + + if input == nil { + // If the data is nil, then we don't set anything, unless ZeroFields is set + // to true. + if d.config.ZeroFields { + outVal.Set(reflect.Zero(outVal.Type())) + + if d.config.Metadata != nil && name != "" { + d.config.Metadata.Keys = append(d.config.Metadata.Keys, name) + } + } return nil } - dataVal := reflect.ValueOf(data) - if !dataVal.IsValid() { - // If the data value is invalid, then we just set the value + if !inputVal.IsValid() { + // If the input value is invalid, then we just set the value // to be the zero value. - val.Set(reflect.Zero(val.Type())) + outVal.Set(reflect.Zero(outVal.Type())) + if d.config.Metadata != nil && name != "" { + d.config.Metadata.Keys = append(d.config.Metadata.Keys, name) + } return nil } if d.config.DecodeHook != nil { - // We have a DecodeHook, so let's pre-process the data. + // We have a DecodeHook, so let's pre-process the input. var err error - data, err = DecodeHookExec( + input, err = DecodeHookExec( d.config.DecodeHook, - dataVal.Type(), val.Type(), data) + inputVal.Type(), outVal.Type(), input) if err != nil { return fmt.Errorf("error decoding '%s': %s", name, err) } } var err error - dataKind := getKind(val) - switch dataKind { + outputKind := getKind(outVal) + switch outputKind { case reflect.Bool: - err = d.decodeBool(name, data, val) + err = d.decodeBool(name, input, outVal) case reflect.Interface: - err = d.decodeBasic(name, data, val) + err = d.decodeBasic(name, input, outVal) case reflect.String: - err = d.decodeString(name, data, val) + err = d.decodeString(name, input, outVal) case reflect.Int: - err = d.decodeInt(name, data, val) + err = d.decodeInt(name, input, outVal) case reflect.Uint: - err = d.decodeUint(name, data, val) + err = d.decodeUint(name, input, outVal) case reflect.Float32: - err = d.decodeFloat(name, data, val) + err = d.decodeFloat(name, input, outVal) case reflect.Struct: - err = d.decodeStruct(name, data, val) + err = d.decodeStruct(name, input, outVal) case reflect.Map: - err = d.decodeMap(name, data, val) + err = d.decodeMap(name, input, outVal) case reflect.Ptr: - err = d.decodePtr(name, data, val) + err = d.decodePtr(name, input, outVal) case reflect.Slice: - err = d.decodeSlice(name, data, val) + err = d.decodeSlice(name, input, outVal) + case reflect.Array: + err = d.decodeArray(name, input, outVal) case reflect.Func: - err = d.decodeFunc(name, data, val) + err = d.decodeFunc(name, input, outVal) default: // If we reached this point then we weren't able to decode it - return fmt.Errorf("%s: unsupported type: %s", name, dataKind) + return fmt.Errorf("%s: unsupported type: %s", name, outputKind) } // If we reached here, then we successfully decoded SOMETHING, so - // mark the key as used if we're tracking metadata. + // mark the key as used if we're tracking metainput. if d.config.Metadata != nil && name != "" { d.config.Metadata.Keys = append(d.config.Metadata.Keys, name) } @@ -256,7 +313,10 @@ func (d *Decoder) decode(name string, data interface{}, val reflect.Value) error // This decodes a basic type (bool, int, string, etc.) and sets the // value to "data" of that type. func (d *Decoder) decodeBasic(name string, data interface{}, val reflect.Value) error { - dataVal := reflect.ValueOf(data) + if val.IsValid() && val.Elem().IsValid() { + return d.decode(name, data, val.Elem()) + } + dataVal := reflect.Indirect(reflect.ValueOf(data)) if !dataVal.IsValid() { dataVal = reflect.Zero(val.Type()) } @@ -273,7 +333,7 @@ func (d *Decoder) decodeBasic(name string, data interface{}, val reflect.Value) } func (d *Decoder) decodeString(name string, data interface{}, val reflect.Value) error { - dataVal := reflect.ValueOf(data) + dataVal := reflect.Indirect(reflect.ValueOf(data)) dataKind := getKind(dataVal) converted := true @@ -292,12 +352,22 @@ func (d *Decoder) decodeString(name string, data interface{}, val reflect.Value) val.SetString(strconv.FormatUint(dataVal.Uint(), 10)) case dataKind == reflect.Float32 && d.config.WeaklyTypedInput: val.SetString(strconv.FormatFloat(dataVal.Float(), 'f', -1, 64)) - case dataKind == reflect.Slice && d.config.WeaklyTypedInput: + case dataKind == reflect.Slice && d.config.WeaklyTypedInput, + dataKind == reflect.Array && d.config.WeaklyTypedInput: dataType := dataVal.Type() elemKind := dataType.Elem().Kind() - switch { - case elemKind == reflect.Uint8: - val.SetString(string(dataVal.Interface().([]uint8))) + switch elemKind { + case reflect.Uint8: + var uints []uint8 + if dataKind == reflect.Array { + uints = make([]uint8, dataVal.Len(), dataVal.Len()) + for i := range uints { + uints[i] = dataVal.Index(i).Interface().(uint8) + } + } else { + uints = dataVal.Interface().([]uint8) + } + val.SetString(string(uints)) default: converted = false } @@ -315,7 +385,7 @@ func (d *Decoder) decodeString(name string, data interface{}, val reflect.Value) } func (d *Decoder) decodeInt(name string, data interface{}, val reflect.Value) error { - dataVal := reflect.ValueOf(data) + dataVal := reflect.Indirect(reflect.ValueOf(data)) dataKind := getKind(dataVal) dataType := dataVal.Type() @@ -357,7 +427,7 @@ func (d *Decoder) decodeInt(name string, data interface{}, val reflect.Value) er } func (d *Decoder) decodeUint(name string, data interface{}, val reflect.Value) error { - dataVal := reflect.ValueOf(data) + dataVal := reflect.Indirect(reflect.ValueOf(data)) dataKind := getKind(dataVal) switch { @@ -400,7 +470,7 @@ func (d *Decoder) decodeUint(name string, data interface{}, val reflect.Value) e } func (d *Decoder) decodeBool(name string, data interface{}, val reflect.Value) error { - dataVal := reflect.ValueOf(data) + dataVal := reflect.Indirect(reflect.ValueOf(data)) dataKind := getKind(dataVal) switch { @@ -431,7 +501,7 @@ func (d *Decoder) decodeBool(name string, data interface{}, val reflect.Value) e } func (d *Decoder) decodeFloat(name string, data interface{}, val reflect.Value) error { - dataVal := reflect.ValueOf(data) + dataVal := reflect.Indirect(reflect.ValueOf(data)) dataKind := getKind(dataVal) dataType := dataVal.Type() @@ -487,38 +557,68 @@ func (d *Decoder) decodeMap(name string, data interface{}, val reflect.Value) er valMap = reflect.MakeMap(mapType) } - // Check input type + // Check input type and based on the input type jump to the proper func dataVal := reflect.Indirect(reflect.ValueOf(data)) - if dataVal.Kind() != reflect.Map { - // In weak mode, we accept a slice of maps as an input... + switch dataVal.Kind() { + case reflect.Map: + return d.decodeMapFromMap(name, dataVal, val, valMap) + + case reflect.Struct: + return d.decodeMapFromStruct(name, dataVal, val, valMap) + + case reflect.Array, reflect.Slice: if d.config.WeaklyTypedInput { - switch dataVal.Kind() { - case reflect.Array, reflect.Slice: - // Special case for BC reasons (covered by tests) - if dataVal.Len() == 0 { - val.Set(valMap) - return nil - } - - for i := 0; i < dataVal.Len(); i++ { - err := d.decode( - fmt.Sprintf("%s[%d]", name, i), - dataVal.Index(i).Interface(), val) - if err != nil { - return err - } - } - - return nil - } + return d.decodeMapFromSlice(name, dataVal, val, valMap) } + fallthrough + + default: return fmt.Errorf("'%s' expected a map, got '%s'", name, dataVal.Kind()) } +} + +func (d *Decoder) decodeMapFromSlice(name string, dataVal reflect.Value, val reflect.Value, valMap reflect.Value) error { + // Special case for BC reasons (covered by tests) + if dataVal.Len() == 0 { + val.Set(valMap) + return nil + } + + for i := 0; i < dataVal.Len(); i++ { + err := d.decode( + fmt.Sprintf("%s[%d]", name, i), + dataVal.Index(i).Interface(), val) + if err != nil { + return err + } + } + + return nil +} + +func (d *Decoder) decodeMapFromMap(name string, dataVal reflect.Value, val reflect.Value, valMap reflect.Value) error { + valType := val.Type() + valKeyType := valType.Key() + valElemType := valType.Elem() // Accumulate errors errors := make([]string, 0) + // If the input data is empty, then we just match what the input data is. + if dataVal.Len() == 0 { + if dataVal.IsNil() { + if !val.IsNil() { + val.Set(dataVal) + } + } else { + // Set to empty allocated value + val.Set(valMap) + } + + return nil + } + for _, k := range dataVal.MapKeys() { fieldName := fmt.Sprintf("%s[%s]", name, k) @@ -551,22 +651,128 @@ func (d *Decoder) decodeMap(name string, data interface{}, val reflect.Value) er return nil } +func (d *Decoder) decodeMapFromStruct(name string, dataVal reflect.Value, val reflect.Value, valMap reflect.Value) error { + typ := dataVal.Type() + for i := 0; i < typ.NumField(); i++ { + // Get the StructField first since this is a cheap operation. If the + // field is unexported, then ignore it. + f := typ.Field(i) + if f.PkgPath != "" { + continue + } + + // Next get the actual value of this field and verify it is assignable + // to the map value. + v := dataVal.Field(i) + if !v.Type().AssignableTo(valMap.Type().Elem()) { + return fmt.Errorf("cannot assign type '%s' to map value field of type '%s'", v.Type(), valMap.Type().Elem()) + } + + tagValue := f.Tag.Get(d.config.TagName) + tagParts := strings.Split(tagValue, ",") + + // Determine the name of the key in the map + keyName := f.Name + if tagParts[0] != "" { + if tagParts[0] == "-" { + continue + } + keyName = tagParts[0] + } + + // If "squash" is specified in the tag, we squash the field down. + squash := false + for _, tag := range tagParts[1:] { + if tag == "squash" { + squash = true + break + } + } + if squash && v.Kind() != reflect.Struct { + return fmt.Errorf("cannot squash non-struct type '%s'", v.Type()) + } + + switch v.Kind() { + // this is an embedded struct, so handle it differently + case reflect.Struct: + x := reflect.New(v.Type()) + x.Elem().Set(v) + + vType := valMap.Type() + vKeyType := vType.Key() + vElemType := vType.Elem() + mType := reflect.MapOf(vKeyType, vElemType) + vMap := reflect.MakeMap(mType) + + err := d.decode(keyName, x.Interface(), vMap) + if err != nil { + return err + } + + if squash { + for _, k := range vMap.MapKeys() { + valMap.SetMapIndex(k, vMap.MapIndex(k)) + } + } else { + valMap.SetMapIndex(reflect.ValueOf(keyName), vMap) + } + + default: + valMap.SetMapIndex(reflect.ValueOf(keyName), v) + } + } + + if val.CanAddr() { + val.Set(valMap) + } + + return nil +} + func (d *Decoder) decodePtr(name string, data interface{}, val reflect.Value) error { + // If the input data is nil, then we want to just set the output + // pointer to be nil as well. + isNil := data == nil + if !isNil { + switch v := reflect.Indirect(reflect.ValueOf(data)); v.Kind() { + case reflect.Chan, + reflect.Func, + reflect.Interface, + reflect.Map, + reflect.Ptr, + reflect.Slice: + isNil = v.IsNil() + } + } + if isNil { + if !val.IsNil() && val.CanSet() { + nilValue := reflect.New(val.Type()).Elem() + val.Set(nilValue) + } + + return nil + } + // Create an element of the concrete (non pointer) type and decode // into that. Then set the value of the pointer to this type. valType := val.Type() valElemType := valType.Elem() + if val.CanSet() { + realVal := val + if realVal.IsNil() || d.config.ZeroFields { + realVal = reflect.New(valElemType) + } - realVal := val - if realVal.IsNil() || d.config.ZeroFields { - realVal = reflect.New(valElemType) + if err := d.decode(name, data, reflect.Indirect(realVal)); err != nil { + return err + } + + val.Set(realVal) + } else { + if err := d.decode(name, data, reflect.Indirect(val)); err != nil { + return err + } } - - if err := d.decode(name, data, reflect.Indirect(realVal)); err != nil { - return err - } - - val.Set(realVal) return nil } @@ -592,30 +798,44 @@ func (d *Decoder) decodeSlice(name string, data interface{}, val reflect.Value) valSlice := val if valSlice.IsNil() || d.config.ZeroFields { + if d.config.WeaklyTypedInput { + switch { + // Slice and array we use the normal logic + case dataValKind == reflect.Slice, dataValKind == reflect.Array: + break + + // Empty maps turn into empty slices + case dataValKind == reflect.Map: + if dataVal.Len() == 0 { + val.Set(reflect.MakeSlice(sliceType, 0, 0)) + return nil + } + // Create slice of maps of other sizes + return d.decodeSlice(name, []interface{}{data}, val) + + case dataValKind == reflect.String && valElemType.Kind() == reflect.Uint8: + return d.decodeSlice(name, []byte(dataVal.String()), val) + + // All other types we try to convert to the slice type + // and "lift" it into it. i.e. a string becomes a string slice. + default: + // Just re-try this function with data as a slice. + return d.decodeSlice(name, []interface{}{data}, val) + } + } + // Check input type if dataValKind != reflect.Array && dataValKind != reflect.Slice { - if d.config.WeaklyTypedInput { - switch { - // Empty maps turn into empty slices - case dataValKind == reflect.Map: - if dataVal.Len() == 0 { - val.Set(reflect.MakeSlice(sliceType, 0, 0)) - return nil - } - - // All other types we try to convert to the slice type - // and "lift" it into it. i.e. a string becomes a string slice. - default: - // Just re-try this function with data as a slice. - return d.decodeSlice(name, []interface{}{data}, val) - } - } - return fmt.Errorf( "'%s': source data must be an array or slice, got %s", name, dataValKind) } + // If the input value is empty, then don't allocate since non-nil != nil + if dataVal.Len() == 0 { + return nil + } + // Make a new slice to hold our result, same size as the original data. valSlice = reflect.MakeSlice(sliceType, dataVal.Len(), dataVal.Len()) } @@ -647,6 +867,73 @@ func (d *Decoder) decodeSlice(name string, data interface{}, val reflect.Value) return nil } +func (d *Decoder) decodeArray(name string, data interface{}, val reflect.Value) error { + dataVal := reflect.Indirect(reflect.ValueOf(data)) + dataValKind := dataVal.Kind() + valType := val.Type() + valElemType := valType.Elem() + arrayType := reflect.ArrayOf(valType.Len(), valElemType) + + valArray := val + + if valArray.Interface() == reflect.Zero(valArray.Type()).Interface() || d.config.ZeroFields { + // Check input type + if dataValKind != reflect.Array && dataValKind != reflect.Slice { + if d.config.WeaklyTypedInput { + switch { + // Empty maps turn into empty arrays + case dataValKind == reflect.Map: + if dataVal.Len() == 0 { + val.Set(reflect.Zero(arrayType)) + return nil + } + + // All other types we try to convert to the array type + // and "lift" it into it. i.e. a string becomes a string array. + default: + // Just re-try this function with data as a slice. + return d.decodeArray(name, []interface{}{data}, val) + } + } + + return fmt.Errorf( + "'%s': source data must be an array or slice, got %s", name, dataValKind) + + } + if dataVal.Len() > arrayType.Len() { + return fmt.Errorf( + "'%s': expected source data to have length less or equal to %d, got %d", name, arrayType.Len(), dataVal.Len()) + + } + + // Make a new array to hold our result, same size as the original data. + valArray = reflect.New(arrayType).Elem() + } + + // Accumulate any errors + errors := make([]string, 0) + + for i := 0; i < dataVal.Len(); i++ { + currentData := dataVal.Index(i).Interface() + currentField := valArray.Index(i) + + fieldName := fmt.Sprintf("%s[%d]", name, i) + if err := d.decode(fieldName, currentData, currentField); err != nil { + errors = appendErrors(errors, err) + } + } + + // Finally, set the value to the array we built up + val.Set(valArray) + + // If there were errors, we return those + if len(errors) > 0 { + return &Error{errors} + } + + return nil +} + func (d *Decoder) decodeStruct(name string, data interface{}, val reflect.Value) error { dataVal := reflect.Indirect(reflect.ValueOf(data)) @@ -658,10 +945,29 @@ func (d *Decoder) decodeStruct(name string, data interface{}, val reflect.Value) } dataValKind := dataVal.Kind() - if dataValKind != reflect.Map { - return fmt.Errorf("'%s' expected a map, got '%s'", name, dataValKind) - } + switch dataValKind { + case reflect.Map: + return d.decodeStructFromMap(name, dataVal, val) + case reflect.Struct: + // Not the most efficient way to do this but we can optimize later if + // we want to. To convert from struct to struct we go to map first + // as an intermediary. + m := make(map[string]interface{}) + mval := reflect.Indirect(reflect.ValueOf(&m)) + if err := d.decodeMapFromStruct(name, dataVal, mval, mval); err != nil { + return err + } + + result := d.decodeStructFromMap(name, mval, val) + return result + + default: + return fmt.Errorf("'%s' expected a map, got '%s'", name, dataVal.Kind()) + } +} + +func (d *Decoder) decodeStructFromMap(name string, dataVal, val reflect.Value) error { dataValType := dataVal.Type() if kind := dataValType.Key().Kind(); kind != reflect.String && kind != reflect.Interface { return fmt.Errorf( @@ -716,7 +1022,7 @@ func (d *Decoder) decodeStruct(name string, data interface{}, val reflect.Value) errors = appendErrors(errors, fmt.Errorf("%s: unsupported type for squash: %s", fieldType.Name, fieldKind)) } else { - structs = append(structs, val.FieldByName(fieldType.Name)) + structs = append(structs, structVal.FieldByName(fieldType.Name)) } continue } diff --git a/vendor/vendor.json b/vendor/vendor.json index 031342cfdc..2b922bbba0 100644 --- a/vendor/vendor.json +++ b/vendor/vendor.json @@ -208,7 +208,7 @@ {"path":"github.com/mitchellh/go-homedir","checksumSHA1":"V/quM7+em2ByJbWBLOsEwnY3j/Q=","revision":"b8bc1bf767474819792c23f32d8286a45736f1c6","revisionTime":"2016-12-03T19:45:07Z"}, {"path":"github.com/mitchellh/go-testing-interface","checksumSHA1":"bDdhmDk8q6utWrccBhEOa6IoGkE=","revision":"a61a99592b77c9ba629d254a693acffaeb4b7e28","revisionTime":"2017-10-04T22:19:16Z"}, {"path":"github.com/mitchellh/hashstructure","checksumSHA1":"tWUjKyFOGJtYExocPWVYiXBYsfE=","revision":"2bca23e0e452137f789efbc8610126fd8b94f73b","revisionTime":"2017-06-09T04:59:27Z"}, - {"path":"github.com/mitchellh/mapstructure","checksumSHA1":"gILp4IL+xwXLH6tJtRLrnZ56F24=","revision":"06020f85339e21b2478f756a78e295255ffa4d6a","revisionTime":"2017-10-17T17:18:08Z"}, + {"path":"github.com/mitchellh/mapstructure","checksumSHA1":"7F5KalhUJ/sCH5bU44MMgw8tqNo=","revision":"5a380f224700b8a6c4eaad048804f5bff514cb35","revisionTime":"2018-10-01T02:14:42Z"}, {"path":"github.com/mitchellh/reflectwalk","checksumSHA1":"AMU63CNOg4XmIhVR/S/Xttt1/f0=","revision":"63d60e9d0dbc60cf9164e6510889b0db6683d98c","revisionTime":"2017-07-26T20:21:17Z"}, {"path":"github.com/modern-go/concurrent","checksumSHA1":"ZTcgWKWHsrX0RXYVXn5Xeb8Q0go=","revision":"bacd9c7ef1dd9b15be4a9909b8ac7a4e313eec94","revisionTime":"2018-03-06T01:26:44Z"}, {"path":"github.com/modern-go/reflect2","checksumSHA1":"qvH48wzTIV3QKSDqI0dLFtVjaDI=","revision":"94122c33edd36123c84d5368cfb2b69df93a0ec8","revisionTime":"2018-07-18T01:23:57Z"}, From 0fbaa18ed35da84ce84199b0dbe00ed3a7656ad1 Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Sun, 30 Sep 2018 19:17:45 -0700 Subject: [PATCH 03/12] command/services/register: config mapping tests --- command/services/register/config.go | 11 ++----- command/services/register/config_test.go | 40 ++++++++++++++++++++++++ 2 files changed, 43 insertions(+), 8 deletions(-) diff --git a/command/services/register/config.go b/command/services/register/config.go index fc01672710..0ab9df2486 100644 --- a/command/services/register/config.go +++ b/command/services/register/config.go @@ -1,7 +1,6 @@ package register import ( - "fmt" "github.com/hashicorp/consul/agent/config" "github.com/hashicorp/consul/api" "github.com/mitchellh/mapstructure" @@ -10,12 +9,8 @@ import ( // configToAgentService converts a ServiceDefinition struct to an // AgentServiceRegistration API struct. func configToAgentService(svc *config.ServiceDefinition) (*api.AgentServiceRegistration, error) { + // mapstructure can do this for us, but we encapsulate it in this + // helper function in case we need to change the logic in the future. var result api.AgentServiceRegistration - var m map[string]interface{} - err := mapstructure.Decode(svc, &m) - if err == nil { - println(fmt.Sprintf("%#v", m)) - err = mapstructure.Decode(m, &result) - } - return &result, err + return &result, mapstructure.Decode(svc, &result) } diff --git a/command/services/register/config_test.go b/command/services/register/config_test.go index 17349a2f80..dc4ed3a717 100644 --- a/command/services/register/config_test.go +++ b/command/services/register/config_test.go @@ -27,6 +27,46 @@ func TestConfigToAgentService(t *testing.T) { Port: 1234, }, }, + { + "Service with a check", + &config.ServiceDefinition{ + Name: strPtr("web"), + Check: &config.CheckDefinition{ + Name: strPtr("ping"), + }, + }, + &api.AgentServiceRegistration{ + Name: "web", + Check: &api.AgentServiceCheck{ + Name: "ping", + }, + }, + }, + { + "Service with checks", + &config.ServiceDefinition{ + Name: strPtr("web"), + Checks: []config.CheckDefinition{ + config.CheckDefinition{ + Name: strPtr("ping"), + }, + config.CheckDefinition{ + Name: strPtr("pong"), + }, + }, + }, + &api.AgentServiceRegistration{ + Name: "web", + Checks: api.AgentServiceChecks{ + &api.AgentServiceCheck{ + Name: "ping", + }, + &api.AgentServiceCheck{ + Name: "pong", + }, + }, + }, + }, } for _, tc := range cases { From 1e7d038b372f644fcc50144ba6b8cbf7c72c7430 Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Mon, 1 Oct 2018 08:05:57 -0700 Subject: [PATCH 04/12] command/services/register: registration from files work --- command/services/register/config.go | 56 +++++++++++++++-- command/services/register/config_test.go | 36 +++++------ command/services/register/register.go | 72 ++++++++++++++++----- command/services/register/register_test.go | 73 ++++++++++++++++++++++ 4 files changed, 201 insertions(+), 36 deletions(-) create mode 100644 command/services/register/register_test.go diff --git a/command/services/register/config.go b/command/services/register/config.go index 0ab9df2486..680e3f13d4 100644 --- a/command/services/register/config.go +++ b/command/services/register/config.go @@ -1,16 +1,64 @@ package register import ( - "github.com/hashicorp/consul/agent/config" + "reflect" + "time" + + "github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/api" "github.com/mitchellh/mapstructure" ) -// configToAgentService converts a ServiceDefinition struct to an +// serviceToAgentService converts a ServiceDefinition struct to an // AgentServiceRegistration API struct. -func configToAgentService(svc *config.ServiceDefinition) (*api.AgentServiceRegistration, error) { +func serviceToAgentService(svc *structs.ServiceDefinition) (*api.AgentServiceRegistration, error) { // mapstructure can do this for us, but we encapsulate it in this // helper function in case we need to change the logic in the future. var result api.AgentServiceRegistration - return &result, mapstructure.Decode(svc, &result) + d, err := mapstructure.NewDecoder(&mapstructure.DecoderConfig{ + Result: &result, + DecodeHook: timeDurationToStringHookFunc(), + WeaklyTypedInput: true, + }) + if err != nil { + return nil, err + } + if err := d.Decode(svc); err != nil { + return nil, err + } + + // The structs version has non-pointer checks and the destination + // has pointers, so we need to set the destination to nil if there + // is no check ID set. + if result.Check != nil && result.Check.Name == "" { + result.Check = nil + } + if len(result.Checks) == 1 && result.Checks[0].Name == "" { + result.Checks = nil + } + + return &result, nil +} + +// timeDurationToStringHookFunc returns a DecodeHookFunc that converts +// time.Duration to string. +func timeDurationToStringHookFunc() mapstructure.DecodeHookFunc { + return func( + f reflect.Type, + t reflect.Type, + data interface{}) (interface{}, error) { + dur, ok := data.(time.Duration) + if !ok { + return data, nil + } + if t.Kind() != reflect.String { + return data, nil + } + if dur == 0 { + return "", nil + } + + // Convert it by parsing + return data.(time.Duration).String(), nil + } } diff --git a/command/services/register/config_test.go b/command/services/register/config_test.go index dc4ed3a717..7671449124 100644 --- a/command/services/register/config_test.go +++ b/command/services/register/config_test.go @@ -3,23 +3,23 @@ package register import ( "testing" - "github.com/hashicorp/consul/agent/config" + "github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/api" "github.com/stretchr/testify/require" ) -func TestConfigToAgentService(t *testing.T) { +func TestStructsToAgentService(t *testing.T) { cases := []struct { Name string - Input *config.ServiceDefinition + Input *structs.ServiceDefinition Output *api.AgentServiceRegistration }{ { "Basic service with port", - &config.ServiceDefinition{ - Name: strPtr("web"), + &structs.ServiceDefinition{ + Name: "web", Tags: []string{"leader"}, - Port: intPtr(1234), + Port: 1234, }, &api.AgentServiceRegistration{ Name: "web", @@ -29,10 +29,10 @@ func TestConfigToAgentService(t *testing.T) { }, { "Service with a check", - &config.ServiceDefinition{ - Name: strPtr("web"), - Check: &config.CheckDefinition{ - Name: strPtr("ping"), + &structs.ServiceDefinition{ + Name: "web", + Check: structs.CheckType{ + Name: "ping", }, }, &api.AgentServiceRegistration{ @@ -44,14 +44,14 @@ func TestConfigToAgentService(t *testing.T) { }, { "Service with checks", - &config.ServiceDefinition{ - Name: strPtr("web"), - Checks: []config.CheckDefinition{ - config.CheckDefinition{ - Name: strPtr("ping"), + &structs.ServiceDefinition{ + Name: "web", + Checks: structs.CheckTypes{ + &structs.CheckType{ + Name: "ping", }, - config.CheckDefinition{ - Name: strPtr("pong"), + &structs.CheckType{ + Name: "pong", }, }, }, @@ -72,7 +72,7 @@ func TestConfigToAgentService(t *testing.T) { for _, tc := range cases { t.Run(tc.Name, func(t *testing.T) { require := require.New(t) - actual, err := configToAgentService(tc.Input) + actual, err := serviceToAgentService(tc.Input) require.NoError(err) require.Equal(tc.Output, actual) }) diff --git a/command/services/register/register.go b/command/services/register/register.go index 0ceb24c40d..1a754a6400 100644 --- a/command/services/register/register.go +++ b/command/services/register/register.go @@ -2,8 +2,10 @@ package register import ( "flag" + "fmt" - //"github.com/hashicorp/consul/api" + "github.com/hashicorp/consul/agent/config" + "github.com/hashicorp/consul/api" "github.com/hashicorp/consul/command/flags" "github.com/mitchellh/cli" ) @@ -48,25 +50,67 @@ func (c *cmd) Run(args []string) int { return 1 } - /* - ixns, err := c.ixnsFromArgs(args) - if err != nil { - c.UI.Error(fmt.Sprintf("Error: %s", err)) + svcs, err := c.svcsFromFiles(args) + if err != nil { + c.UI.Error(fmt.Sprintf("Error: %s", err)) + return 1 + } + + // Create and test the HTTP client + client, err := c.http.APIClient() + if err != nil { + c.UI.Error(fmt.Sprintf("Error connecting to Consul agent: %s", err)) + return 1 + } + + // Create all the services + for _, svc := range svcs { + if err := client.Agent().ServiceRegister(svc); err != nil { + c.UI.Error(fmt.Sprintf("Error registering service %q: %s", + svc.Name, err)) return 1 } - - // Create and test the HTTP client - /* - client, err := c.http.APIClient() - if err != nil { - c.UI.Error(fmt.Sprintf("Error connecting to Consul agent: %s", err)) - return 1 - } - */ + } return 0 } +// svcsFromFiles loads service definitions from a set of configuration +// files and returns them. It will return an error if the configuration is +// invalid in any way. +func (c *cmd) svcsFromFiles(args []string) ([]*api.AgentServiceRegistration, error) { + // We set devMode to true so we can get the basic valid default + // configuration. devMode doesn't set any services by default so this + // is okay since we only look at services. + devMode := true + b, err := config.NewBuilder(config.Flags{ + ConfigFiles: args, + DevMode: &devMode, + }) + if err != nil { + return nil, err + } + + cfg, err := b.BuildAndValidate() + if err != nil { + return nil, err + } + + // The services are now in "structs.ServiceDefinition" form and we need + // them in "api.AgentServiceRegistration" form so do the conversion. + result := make([]*api.AgentServiceRegistration, 0, len(cfg.Services)) + for _, svc := range cfg.Services { + apiSvc, err := serviceToAgentService(svc) + if err != nil { + return nil, err + } + + result = append(result, apiSvc) + } + + return result, nil +} + func (c *cmd) Synopsis() string { return synopsis } diff --git a/command/services/register/register_test.go b/command/services/register/register_test.go new file mode 100644 index 0000000000..f6681e1190 --- /dev/null +++ b/command/services/register/register_test.go @@ -0,0 +1,73 @@ +package register + +import ( + "os" + "strings" + "testing" + + "github.com/hashicorp/consul/agent" + "github.com/hashicorp/consul/testutil" + "github.com/mitchellh/cli" + "github.com/stretchr/testify/require" +) + +func TestCommand_noTabs(t *testing.T) { + t.Parallel() + if strings.ContainsRune(New(nil).Help(), '\t') { + t.Fatal("help has tabs") + } +} + +func TestCommand_File(t *testing.T) { + t.Parallel() + + require := require.New(t) + a := agent.NewTestAgent(t.Name(), ``) + defer a.Shutdown() + client := a.Client() + + ui := cli.NewMockUi() + c := New(ui) + + contents := `{ "Service": { "Name": "web" } }` + 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) +} + +func testFile(t *testing.T, suffix string) *os.File { + f := testutil.TempFile(t, "register-test-file") + if err := f.Close(); err != nil { + t.Fatalf("err: %s", err) + } + + newName := f.Name() + "." + suffix + if err := os.Rename(f.Name(), newName); err != nil { + os.Remove(f.Name()) + t.Fatalf("err: %s", err) + } + + f, err := os.Create(newName) + if err != nil { + os.Remove(newName) + t.Fatalf("err: %s", err) + } + + return f +} From 4b887d6dda0ef9d0f707470da57e45342f613793 Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Mon, 1 Oct 2018 08:27:59 -0700 Subject: [PATCH 05/12] command/services: move the config helpers to parent package --- command/services/{register => }/config.go | 38 +++++++++- .../services/{register => }/config_test.go | 2 +- command/services/register/register.go | 73 +++---------------- 3 files changed, 48 insertions(+), 65 deletions(-) rename command/services/{register => }/config.go (61%) rename command/services/{register => }/config_test.go (98%) diff --git a/command/services/register/config.go b/command/services/config.go similarity index 61% rename from command/services/register/config.go rename to command/services/config.go index 680e3f13d4..2241c90eca 100644 --- a/command/services/register/config.go +++ b/command/services/config.go @@ -1,14 +1,50 @@ -package register +package services import ( "reflect" "time" + "github.com/hashicorp/consul/agent/config" "github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/api" "github.com/mitchellh/mapstructure" ) +// ServicesFromFiles returns the list of agent service registration structs +// from a set of file arguments. +func ServicesFromFiles(files []string) ([]*api.AgentServiceRegistration, error) { + // We set devMode to true so we can get the basic valid default + // configuration. devMode doesn't set any services by default so this + // is okay since we only look at services. + devMode := true + b, err := config.NewBuilder(config.Flags{ + ConfigFiles: files, + DevMode: &devMode, + }) + if err != nil { + return nil, err + } + + cfg, err := b.BuildAndValidate() + if err != nil { + return nil, err + } + + // The services are now in "structs.ServiceDefinition" form and we need + // them in "api.AgentServiceRegistration" form so do the conversion. + result := make([]*api.AgentServiceRegistration, 0, len(cfg.Services)) + for _, svc := range cfg.Services { + apiSvc, err := serviceToAgentService(svc) + if err != nil { + return nil, err + } + + result = append(result, apiSvc) + } + + return result, nil +} + // serviceToAgentService converts a ServiceDefinition struct to an // AgentServiceRegistration API struct. func serviceToAgentService(svc *structs.ServiceDefinition) (*api.AgentServiceRegistration, error) { diff --git a/command/services/register/config_test.go b/command/services/config_test.go similarity index 98% rename from command/services/register/config_test.go rename to command/services/config_test.go index 7671449124..2689d2ece5 100644 --- a/command/services/register/config_test.go +++ b/command/services/config_test.go @@ -1,4 +1,4 @@ -package register +package services import ( "testing" diff --git a/command/services/register/register.go b/command/services/register/register.go index 1a754a6400..b5a8147dee 100644 --- a/command/services/register/register.go +++ b/command/services/register/register.go @@ -4,9 +4,8 @@ import ( "flag" "fmt" - "github.com/hashicorp/consul/agent/config" - "github.com/hashicorp/consul/api" "github.com/hashicorp/consul/command/flags" + "github.com/hashicorp/consul/command/services" "github.com/mitchellh/cli" ) @@ -50,7 +49,7 @@ func (c *cmd) Run(args []string) int { return 1 } - svcs, err := c.svcsFromFiles(args) + svcs, err := services.ServicesFromFiles(args) if err != nil { c.UI.Error(fmt.Sprintf("Error: %s", err)) return 1 @@ -75,42 +74,6 @@ func (c *cmd) Run(args []string) int { return 0 } -// svcsFromFiles loads service definitions from a set of configuration -// files and returns them. It will return an error if the configuration is -// invalid in any way. -func (c *cmd) svcsFromFiles(args []string) ([]*api.AgentServiceRegistration, error) { - // We set devMode to true so we can get the basic valid default - // configuration. devMode doesn't set any services by default so this - // is okay since we only look at services. - devMode := true - b, err := config.NewBuilder(config.Flags{ - ConfigFiles: args, - DevMode: &devMode, - }) - if err != nil { - return nil, err - } - - cfg, err := b.BuildAndValidate() - if err != nil { - return nil, err - } - - // The services are now in "structs.ServiceDefinition" form and we need - // them in "api.AgentServiceRegistration" form so do the conversion. - result := make([]*api.AgentServiceRegistration, 0, len(cfg.Services)) - for _, svc := range cfg.Services { - apiSvc, err := serviceToAgentService(svc) - if err != nil { - return nil, err - } - - result = append(result, apiSvc) - } - - return result, nil -} - func (c *cmd) Synopsis() string { return synopsis } @@ -119,33 +82,17 @@ func (c *cmd) Help() string { return c.help } -const synopsis = "Create intentions for service connections." +const synopsis = "Register services with the local agent" const help = ` -Usage: consul intention create [options] SRC DST -Usage: consul intention create [options] -file FILE... +Usage: consul services register [options] [FILE...] - Create one or more intentions. The data can be specified as a single - source and destination pair or via a set of files when the "-file" flag - is specified. + Register one or more services using the local agent API. Services can + be registered from standard Consul configuration files (HCL or JSON) or + using flags. The service is registered and the command returns. The caller + must remember to call "consul services deregister" or a similar API to + deregister the service when complete. - $ consul intention create web db - - To consume data from a set of files: - - $ consul intention create -file one.json two.json - - When specifying the "-file" flag, "-" may be used once to read from stdin: - - $ echo "{ ... }" | consul intention create -file - - - An "allow" intention is created by default (whitelist). To create a - "deny" intention, the "-deny" flag should be specified. - - If a conflicting intention is found, creation will fail. To replace any - conflicting intentions, specify the "-replace" flag. This will replace any - conflicting intentions with the intention specified in this command. - Metadata and any other fields of the previous intention will not be - preserved. + $ consul services register web.json Additional flags and more advanced use cases are detailed below. ` From 2f97a618dce714f4a09af892c7073d8467b2baff Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Mon, 1 Oct 2018 08:39:27 -0700 Subject: [PATCH 06/12] command/services/deregister: basics working from file --- command/services/deregister/deregister.go | 102 ++++++++++++++++ .../services/deregister/deregister_test.go | 115 ++++++++++++++++++ 2 files changed, 217 insertions(+) create mode 100644 command/services/deregister/deregister.go create mode 100644 command/services/deregister/deregister_test.go diff --git a/command/services/deregister/deregister.go b/command/services/deregister/deregister.go new file mode 100644 index 0000000000..eac269d623 --- /dev/null +++ b/command/services/deregister/deregister.go @@ -0,0 +1,102 @@ +package deregister + +import ( + "flag" + "fmt" + + "github.com/hashicorp/consul/command/flags" + "github.com/hashicorp/consul/command/services" + "github.com/mitchellh/cli" +) + +func New(ui cli.Ui) *cmd { + c := &cmd{UI: ui} + c.init() + return c +} + +type cmd struct { + UI cli.Ui + flags *flag.FlagSet + http *flags.HTTPFlags + help string +} + +func (c *cmd) init() { + c.flags = flag.NewFlagSet("", flag.ContinueOnError) + + c.http = &flags.HTTPFlags{} + flags.Merge(c.flags, c.http.ClientFlags()) + flags.Merge(c.flags, c.http.ServerFlags()) + c.help = flags.Usage(help, c.flags) +} + +func (c *cmd) Run(args []string) int { + if err := c.flags.Parse(args); err != nil { + return 1 + } + + // Check for arg validation + args = c.flags.Args() + if len(args) == 0 { + c.UI.Error("Service deregistration requires at least one argument.") + return 1 + } + + svcs, err := services.ServicesFromFiles(args) + if err != nil { + c.UI.Error(fmt.Sprintf("Error: %s", err)) + return 1 + } + + // Create and test the HTTP client + client, err := c.http.APIClient() + if err != nil { + c.UI.Error(fmt.Sprintf("Error connecting to Consul agent: %s", err)) + return 1 + } + + // Create all the services + for _, svc := range svcs { + id := svc.ID + if id == "" { + id = svc.Name + } + if id == "" { + continue + } + + if err := client.Agent().ServiceDeregister(id); err != nil { + c.UI.Error(fmt.Sprintf("Error registering service %q: %s", + svc.Name, err)) + return 1 + } + } + + return 0 +} + +func (c *cmd) Synopsis() string { + return synopsis +} + +func (c *cmd) Help() string { + return c.help +} + +const synopsis = "Deregister services with the local agent" +const help = ` +Usage: consul services deregister [options] [FILE...] + + Deregister one or more services that were previously registered with + the local agent. + + $ consul services deregister web.json db.json + + The -id flag may be used to deregister a single service by ID: + + $ consul services deregister -id=web + + Services are deregistered from the local agent catalog. This command must + be run against the same agent where the service was registered. +` diff --git a/command/services/deregister/deregister_test.go b/command/services/deregister/deregister_test.go new file mode 100644 index 0000000000..379bb8491c --- /dev/null +++ b/command/services/deregister/deregister_test.go @@ -0,0 +1,115 @@ +package deregister + +import ( + "os" + "strings" + "testing" + + "github.com/hashicorp/consul/agent" + "github.com/hashicorp/consul/api" + "github.com/hashicorp/consul/testutil" + "github.com/mitchellh/cli" + "github.com/stretchr/testify/require" +) + +func TestCommand_noTabs(t *testing.T) { + t.Parallel() + if strings.ContainsRune(New(nil).Help(), '\t') { + t.Fatal("help has tabs") + } +} + +func TestCommand_File_id(t *testing.T) { + t.Parallel() + + require := require.New(t) + a := agent.NewTestAgent(t.Name(), ``) + defer a.Shutdown() + client := a.Client() + + // Register a service + require.NoError(client.Agent().ServiceRegister(&api.AgentServiceRegistration{ + Name: "web"})) + require.NoError(client.Agent().ServiceRegister(&api.AgentServiceRegistration{ + Name: "db"})) + + ui := cli.NewMockUi() + c := New(ui) + + contents := `{ "Service": { "ID": "web", "Name": "foo" } }` + 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) + require.NotNil(svcs["db"]) +} + +func TestCommand_File_nameOnly(t *testing.T) { + t.Parallel() + + require := require.New(t) + a := agent.NewTestAgent(t.Name(), ``) + defer a.Shutdown() + client := a.Client() + + // Register a service + require.NoError(client.Agent().ServiceRegister(&api.AgentServiceRegistration{ + Name: "web"})) + require.NoError(client.Agent().ServiceRegister(&api.AgentServiceRegistration{ + Name: "db"})) + + ui := cli.NewMockUi() + c := New(ui) + + contents := `{ "Service": { "Name": "web" } }` + 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) + require.NotNil(svcs["db"]) +} + +func testFile(t *testing.T, suffix string) *os.File { + f := testutil.TempFile(t, "register-test-file") + if err := f.Close(); err != nil { + t.Fatalf("err: %s", err) + } + + newName := f.Name() + "." + suffix + if err := os.Rename(f.Name(), newName); err != nil { + os.Remove(f.Name()) + t.Fatalf("err: %s", err) + } + + f, err := os.Create(newName) + if err != nil { + os.Remove(newName) + t.Fatalf("err: %s", err) + } + + return f +} From 3425f123ef20cf587ac4e2444ada8621b819c2a3 Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Mon, 1 Oct 2018 08:53:30 -0700 Subject: [PATCH 07/12] command/services/deregister: -id flag for deletion --- command/services/deregister/deregister.go | 29 +++++++++++------- .../services/deregister/deregister_test.go | 30 +++++++++++++++++++ 2 files changed, 49 insertions(+), 10 deletions(-) diff --git a/command/services/deregister/deregister.go b/command/services/deregister/deregister.go index eac269d623..f00a880705 100644 --- a/command/services/deregister/deregister.go +++ b/command/services/deregister/deregister.go @@ -4,6 +4,7 @@ import ( "flag" "fmt" + "github.com/hashicorp/consul/api" "github.com/hashicorp/consul/command/flags" "github.com/hashicorp/consul/command/services" "github.com/mitchellh/cli" @@ -16,14 +17,17 @@ func New(ui cli.Ui) *cmd { } type cmd struct { - UI cli.Ui - flags *flag.FlagSet - http *flags.HTTPFlags - help string + UI cli.Ui + flags *flag.FlagSet + http *flags.HTTPFlags + help string + flagId string } func (c *cmd) init() { c.flags = flag.NewFlagSet("", flag.ContinueOnError) + c.flags.StringVar(&c.flagId, "id", "", + "ID to delete. This must not be set if arguments are given.") c.http = &flags.HTTPFlags{} flags.Merge(c.flags, c.http.ClientFlags()) @@ -38,15 +42,20 @@ func (c *cmd) Run(args []string) int { // Check for arg validation args = c.flags.Args() - if len(args) == 0 { - c.UI.Error("Service deregistration requires at least one argument.") + if len(args) == 0 && c.flagId == "" { + c.UI.Error("Service deregistration requires at least one argument or -id.") return 1 } - svcs, err := services.ServicesFromFiles(args) - if err != nil { - c.UI.Error(fmt.Sprintf("Error: %s", err)) - return 1 + svcs := []*api.AgentServiceRegistration{&api.AgentServiceRegistration{ + ID: c.flagId}} + if len(args) > 0 { + var err error + svcs, err = services.ServicesFromFiles(args) + if err != nil { + c.UI.Error(fmt.Sprintf("Error: %s", err)) + return 1 + } } // Create and test the HTTP client diff --git a/command/services/deregister/deregister_test.go b/command/services/deregister/deregister_test.go index 379bb8491c..303c8cabb5 100644 --- a/command/services/deregister/deregister_test.go +++ b/command/services/deregister/deregister_test.go @@ -93,6 +93,36 @@ func TestCommand_File_nameOnly(t *testing.T) { require.NotNil(svcs["db"]) } +func TestCommand_Flag(t *testing.T) { + t.Parallel() + + require := require.New(t) + a := agent.NewTestAgent(t.Name(), ``) + defer a.Shutdown() + client := a.Client() + + // Register a service + require.NoError(client.Agent().ServiceRegister(&api.AgentServiceRegistration{ + Name: "web"})) + require.NoError(client.Agent().ServiceRegister(&api.AgentServiceRegistration{ + Name: "db"})) + + ui := cli.NewMockUi() + c := New(ui) + + args := []string{ + "-http-addr=" + a.HTTPAddr(), + "-id", "web", + } + + require.Equal(0, c.Run(args), ui.ErrorWriter.String()) + + svcs, err := client.Agent().Services() + require.NoError(err) + require.Len(svcs, 1) + require.NotNil(svcs["db"]) +} + func testFile(t *testing.T, suffix string) *os.File { f := testutil.TempFile(t, "register-test-file") if err := f.Close(); err != nil { From 939708138fc20984df8747200a2b2702fc57fd4e Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Mon, 1 Oct 2018 08:55:32 -0700 Subject: [PATCH 08/12] command/services/deregister: tests for flag validation --- command/services/deregister/deregister.go | 3 ++ .../services/deregister/deregister_test.go | 41 +++++++++++++++++++ 2 files changed, 44 insertions(+) diff --git a/command/services/deregister/deregister.go b/command/services/deregister/deregister.go index f00a880705..85a63ab419 100644 --- a/command/services/deregister/deregister.go +++ b/command/services/deregister/deregister.go @@ -45,6 +45,9 @@ func (c *cmd) Run(args []string) int { if len(args) == 0 && c.flagId == "" { c.UI.Error("Service deregistration requires at least one argument or -id.") return 1 + } else if len(args) > 0 && c.flagId != "" { + c.UI.Error("Service deregistration requires arguments or -id, not both.") + return 1 } svcs := []*api.AgentServiceRegistration{&api.AgentServiceRegistration{ diff --git a/command/services/deregister/deregister_test.go b/command/services/deregister/deregister_test.go index 303c8cabb5..d4dab53840 100644 --- a/command/services/deregister/deregister_test.go +++ b/command/services/deregister/deregister_test.go @@ -19,6 +19,47 @@ func TestCommand_noTabs(t *testing.T) { } } +func TestCommand_Validation(t *testing.T) { + t.Parallel() + + ui := cli.NewMockUi() + c := New(ui) + + cases := map[string]struct { + args []string + output string + }{ + "no args or id": { + []string{}, + "at least one", + }, + "args and -id": { + []string{"-id", "web", "foo.json"}, + "not both", + }, + } + + for name, tc := range cases { + t.Run(name, func(t *testing.T) { + require := require.New(t) + + c.init() + + // Ensure our buffer is always clear + if ui.ErrorWriter != nil { + ui.ErrorWriter.Reset() + } + if ui.OutputWriter != nil { + ui.OutputWriter.Reset() + } + + require.Equal(1, c.Run(tc.args)) + output := ui.ErrorWriter.String() + require.Contains(output, tc.output) + }) + } +} + func TestCommand_File_id(t *testing.T) { t.Parallel() From bf83309124146d735e8b27636c9b19e056a92c0d Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Mon, 1 Oct 2018 09:16:14 -0700 Subject: [PATCH 09/12] command/services/register: flag-based registration --- command/services/register/register.go | 47 ++++++++++++--- command/services/register/register_test.go | 67 ++++++++++++++++++++++ 2 files changed, 107 insertions(+), 7 deletions(-) diff --git a/command/services/register/register.go b/command/services/register/register.go index b5a8147dee..ecd453c71f 100644 --- a/command/services/register/register.go +++ b/command/services/register/register.go @@ -4,6 +4,7 @@ import ( "flag" "fmt" + "github.com/hashicorp/consul/api" "github.com/hashicorp/consul/command/flags" "github.com/hashicorp/consul/command/services" "github.com/mitchellh/cli" @@ -22,14 +23,31 @@ type cmd struct { help string // flags - flagMeta map[string]string + flagId string + flagName string + flagAddress string + flagPort int + flagTags []string + flagMeta map[string]string } func (c *cmd) init() { c.flags = flag.NewFlagSet("", flag.ContinueOnError) + c.flags.StringVar(&c.flagId, "id", "", + "ID of the service to register for arg-based registration. If this "+ + "isn't set, it will default to the -name value.") + c.flags.StringVar(&c.flagName, "name", "", + "Name of the service to register for arg-based registration.") + c.flags.StringVar(&c.flagAddress, "address", "", + "Address of the service to register for arg-based registration.") + c.flags.IntVar(&c.flagPort, "port", 0, + "Port of the service to register for arg-based registration.") c.flags.Var((*flags.FlagMapValue)(&c.flagMeta), "meta", "Metadata to set on the intention, formatted as key=value. This flag "+ "may be specified multiple times to set multiple meta fields.") + c.flags.Var((*flags.AppendSliceValue)(&c.flagTags), "tag", + "Tag to add to the service. This flag can be specified multiple "+ + "times to set multiple tags.") c.http = &flags.HTTPFlags{} flags.Merge(c.flags, c.http.ClientFlags()) @@ -42,17 +60,32 @@ func (c *cmd) Run(args []string) int { return 1 } + svcs := []*api.AgentServiceRegistration{&api.AgentServiceRegistration{ + ID: c.flagId, + Name: c.flagName, + Address: c.flagAddress, + Port: c.flagPort, + Tags: c.flagTags, + Meta: c.flagMeta, + }} + // Check for arg validation args = c.flags.Args() - if len(args) == 0 { - c.UI.Error("Service registration requires at least one argument.") + if len(args) == 0 && c.flagName == "" { + c.UI.Error("Service registration requires at least one argument or flags.") + return 1 + } else if len(args) > 0 && c.flagName != "" { + c.UI.Error("Service registration requires arguments or -id, not both.") return 1 } - svcs, err := services.ServicesFromFiles(args) - if err != nil { - c.UI.Error(fmt.Sprintf("Error: %s", err)) - return 1 + if len(args) > 0 { + var err error + svcs, err = services.ServicesFromFiles(args) + if err != nil { + c.UI.Error(fmt.Sprintf("Error: %s", err)) + return 1 + } } // Create and test the HTTP client diff --git a/command/services/register/register_test.go b/command/services/register/register_test.go index f6681e1190..fac96ab580 100644 --- a/command/services/register/register_test.go +++ b/command/services/register/register_test.go @@ -18,6 +18,47 @@ func TestCommand_noTabs(t *testing.T) { } } +func TestCommand_Validation(t *testing.T) { + t.Parallel() + + ui := cli.NewMockUi() + c := New(ui) + + cases := map[string]struct { + args []string + output string + }{ + "no args or id": { + []string{}, + "at least one", + }, + "args and -name": { + []string{"-name", "web", "foo.json"}, + "not both", + }, + } + + for name, tc := range cases { + t.Run(name, func(t *testing.T) { + require := require.New(t) + + c.init() + + // Ensure our buffer is always clear + if ui.ErrorWriter != nil { + ui.ErrorWriter.Reset() + } + if ui.OutputWriter != nil { + ui.OutputWriter.Reset() + } + + require.Equal(1, c.Run(tc.args)) + output := ui.ErrorWriter.String() + require.Contains(output, tc.output) + }) + } +} + func TestCommand_File(t *testing.T) { t.Parallel() @@ -51,6 +92,32 @@ func TestCommand_File(t *testing.T) { require.NotNil(svc) } +func TestCommand_Flags(t *testing.T) { + t.Parallel() + + require := require.New(t) + a := agent.NewTestAgent(t.Name(), ``) + defer a.Shutdown() + client := a.Client() + + ui := cli.NewMockUi() + c := New(ui) + + args := []string{ + "-http-addr=" + a.HTTPAddr(), + "-name", "web", + } + + 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) +} + func testFile(t *testing.T, suffix string) *os.File { f := testutil.TempFile(t, "register-test-file") if err := f.Close(); err != nil { From e00c40b4f5cf1e3cfd26ab81ff37e6eb328208ce Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Mon, 1 Oct 2018 09:17:36 -0700 Subject: [PATCH 10/12] command: register new commands --- command/commands_oss.go | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/command/commands_oss.go b/command/commands_oss.go index 8e95282aab..46f328b7fd 100644 --- a/command/commands_oss.go +++ b/command/commands_oss.go @@ -44,6 +44,9 @@ import ( operraftremove "github.com/hashicorp/consul/command/operator/raft/removepeer" "github.com/hashicorp/consul/command/reload" "github.com/hashicorp/consul/command/rtt" + "github.com/hashicorp/consul/command/services" + svcsderegister "github.com/hashicorp/consul/command/services/deregister" + svcsregister "github.com/hashicorp/consul/command/services/register" "github.com/hashicorp/consul/command/snapshot" snapinspect "github.com/hashicorp/consul/command/snapshot/inspect" snaprestore "github.com/hashicorp/consul/command/snapshot/restore" @@ -107,6 +110,9 @@ func init() { Register("operator raft remove-peer", func(ui cli.Ui) (cli.Command, error) { return operraftremove.New(ui), nil }) Register("reload", func(ui cli.Ui) (cli.Command, error) { return reload.New(ui), nil }) Register("rtt", func(ui cli.Ui) (cli.Command, error) { return rtt.New(ui), nil }) + Register("services", func(cli.Ui) (cli.Command, error) { return services.New(), nil }) + Register("services register", func(ui cli.Ui) (cli.Command, error) { return svcsregister.New(ui), nil }) + Register("services deregister", func(ui cli.Ui) (cli.Command, error) { return svcsderegister.New(ui), nil }) Register("snapshot", func(cli.Ui) (cli.Command, error) { return snapshot.New(), nil }) Register("snapshot inspect", func(ui cli.Ui) (cli.Command, error) { return snapinspect.New(ui), nil }) Register("snapshot restore", func(ui cli.Ui) (cli.Command, error) { return snaprestore.New(ui), nil }) From 3cf2b9e3616f21062919adddbd980cd7edf1898b Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Mon, 1 Oct 2018 10:27:15 -0700 Subject: [PATCH 11/12] website: docs for services CLI --- website/source/docs/commands/index.html.md | 4 +- website/source/docs/commands/services.html.md | 66 +++++++++++++ .../docs/commands/services/deregister.html.md | 63 ++++++++++++ .../docs/commands/services/register.html.md | 99 +++++++++++++++++++ website/source/layouts/docs.erb | 12 +++ 5 files changed, 243 insertions(+), 1 deletion(-) create mode 100644 website/source/docs/commands/services.html.md create mode 100644 website/source/docs/commands/services/deregister.html.md create mode 100644 website/source/docs/commands/services/register.html.md diff --git a/website/source/docs/commands/index.html.md b/website/source/docs/commands/index.html.md index 68c55a5b14..f242ecbf55 100644 --- a/website/source/docs/commands/index.html.md +++ b/website/source/docs/commands/index.html.md @@ -28,16 +28,17 @@ Usage: consul [--version] [--help] [] Available commands are: agent Runs a Consul agent catalog Interact with the catalog + connect Interact with Consul Connect event Fire a new event exec Executes a command on Consul nodes force-leave Forces a member of the cluster to enter the "left" state info Provides debugging information for operators. + intention Interact with Connect service intentions join Tell Consul agent to join cluster keygen Generates a new encryption key keyring Manages gossip layer encryption keys kv Interact with the key-value store leave Gracefully leaves the Consul cluster and shuts down - license Get/Put the Consul Enterprise license (Enterprise-only) lock Execute a command holding a lock maint Controls node or service maintenance mode members Lists the members of a Consul cluster @@ -45,6 +46,7 @@ Available commands are: operator Provides cluster-level tools for Consul operators reload Triggers the agent to reload configuration files rtt Estimates network round trip time between nodes + services Interact with services snapshot Saves, restores and inspects snapshots of Consul server state validate Validate config files/directories version Prints the Consul version diff --git a/website/source/docs/commands/services.html.md b/website/source/docs/commands/services.html.md new file mode 100644 index 0000000000..f80e4b7f7b --- /dev/null +++ b/website/source/docs/commands/services.html.md @@ -0,0 +1,66 @@ +--- +layout: "docs" +page_title: "Commands: Services" +sidebar_current: "docs-commands-services" +--- + +# Consul Agent Services + +Command: `consul services` + +The `services` command has subcommands for interacting with Consul services +registered with the [local agent](/docs/agent/basics.html). These provide +useful commands such as `register` and `deregister` for easily registering +services in scripts, dev mode, etc. +To view all services in the catalog, instead of only agent-local services, +see the [`catalog services`](/docs/commands/catalog/services.html) command. + +## Usage + +Usage: `consul services ` + +For the exact documentation for your Consul version, run `consul services -h` to +view the complete list of subcommands. + +```text +Usage: consul services [options] [args] + + ... + +Subcommands: + deregister Deregister services with the local agent + register Register services with the local agent +``` + +For more information, examples, and usage about a subcommand, click on the name +of the subcommand in the sidebar. + +## Basic Examples + +To create a simple service: + +```text +$ consul services register -name=web +``` + +To create a service from a configuration file: + +```text +$ cat web.json +{ + "Service": { + "Name": "web" + } +} + +$ consul services register web.json +``` + +To deregister a service: + +```sh +# Either style works: +$ consul services deregister web.json + +$ consul services deregister -id web +``` diff --git a/website/source/docs/commands/services/deregister.html.md b/website/source/docs/commands/services/deregister.html.md new file mode 100644 index 0000000000..87316383cf --- /dev/null +++ b/website/source/docs/commands/services/deregister.html.md @@ -0,0 +1,63 @@ +--- +layout: "docs" +page_title: "Commands: Services Deregister" +sidebar_current: "docs-commands-services-deregister" +--- + +# Consul Agent Service Deregistration + +Command: `consul services deregister` + +The `services deregister` command deregisters a service with the local agent. +Note that this command can only deregister services that were registered +with the agent specified (defaults to the local agent) and is meant to +be paired with `services register`. + +This is just one method for service deregistration. If the service was +registered with a configuration file, then deleting that file and +[reloading](/docs/commands/reload.html) Consul is the correct method to +deregister. See [Service Definition](/docs/agent/services.html) for more +information about registering services generally. + +## Usage + +Usage: `consul services deregister [options] [FILE...]` + +This command can deregister either a single service using the `-id` flag +documented below, or one or more services using service definition files +in HCL or JSON format. +This flexibility makes it easy to pair the command with the +`services register` command since the argument syntax is the same. + +#### API Options + +<%= partial "docs/commands/http_api_options_client" %> + +#### Service Deregistration Flags + +The flags below should only be set if _no arguments_ are given. If no +arguments are given, the flags below can be used to deregister a single +service. + +* `-id` - The ID of the service. + +## Examples + +To deregister by ID: + +```text +$ consul services deregister -id=web +``` + +To deregister from a configuration file: + +```text +$ cat web.json +{ + "Service": { + "Name": "web" + } +} + +$ consul services deregister web.json +``` diff --git a/website/source/docs/commands/services/register.html.md b/website/source/docs/commands/services/register.html.md new file mode 100644 index 0000000000..9fd69feb14 --- /dev/null +++ b/website/source/docs/commands/services/register.html.md @@ -0,0 +1,99 @@ +--- +layout: "docs" +page_title: "Commands: Services Register" +sidebar_current: "docs-commands-services-register" +--- + +# Consul Agent Service Registration + +Command: `consul services register` + +The `services register` command registers a service with the local agent. +This command returns after registration and must be paired with explicit +service deregistration. This command simplifies service registration from +scripts, in dev mode, etc. + +This is just one method of service registration. Services can also be +registered by placing a [service definition](/docs/agent/services.html) +in the Consul agent configuration directory and issuing a +[reload](/docs/commands/reload.html). This approach is easiest for +configuration management systems that other systems that have access to +the configuration directory. Clients may also use the +[HTTP API](/api/agent/service.html) directly. + +## Usage + +Usage: `consul services register [options] [FILE...]` + +This command can register either a single service using flags documented +below, or one or more services using service definition files in HCL +or JSON format. The service is registered against the specified Consul +agent (defaults to the local agent). This agent will execute all registered +health checks. + +This command returns after registration succeeds. It must be paired with +a deregistration command or API call to remove the service. To ensure that +services are properly deregistered, it is **highly recommended** that +a check is created with the +[`DeregisterCriticalServiceAfter`](/api/agent/check.html#deregistercriticalserviceafter) +configuration set. This will ensure that even if deregistration failed for +any reason, the agent will automatically deregister the service instance after +it is unhealthy for the specified period of time. + +Registered services are persisted in the agent state directory. If the +state directory remains unmodified, registered services will persist across +restarts. + +~> **Warning for Consul operators:** The Consul agent persists registered +services in the local state directory. If this state directory is deleted +or lost, services registered with this command will need to be reregistered. + +#### API Options + +<%= partial "docs/commands/http_api_options_client" %> + +#### Service Registration Flags + +The flags below should only be set if _no arguments_ are given. If no +arguments are given, the flags below can be used to register a single +service. + +Note that the behavior of each of the fields below is exactly the same +as when constructing a standard [service definition](/docs/agent/services.html). +Please refer to that documentation for full details. + +* `-id` - The ID of the service. This will default to `-name` if not set. + +* `-name` - The name of the service to register. + +* `-address` - The address of the service. If this isn't specified, + it will default to the address registered with the local agent. + +* `-port` - The port of the service. + +* `-meta key=value` - Specify arbitrary KV metadata to associate with the + service instance. This can be specified multiple times. + +* `-tag value` - Associate a tag with the service instance. This flag can + be specified multiples times. + +## Examples + +To create a simple service: + +```text +$ consul services register -name=web +``` + +To create a service from a configuration file: + +```text +$ cat web.json +{ + "Service": { + "Name": "web" + } +} + +$ consul services register web.json +``` diff --git a/website/source/layouts/docs.erb b/website/source/layouts/docs.erb index 372f8de59b..1731c80ca6 100644 --- a/website/source/layouts/docs.erb +++ b/website/source/layouts/docs.erb @@ -185,6 +185,18 @@ rtt + > + services + + + > snapshot