/Godeps/_workspace/src/github.com/coreos/coreos-cloudinit/config/config.go

https://gitlab.com/kloudsio/os · Go · 154 lines · 103 code · 22 blank · 29 comment · 17 complexity · 46ca4d2e44d368ca4a989cdd2c87c6b9 MD5 · raw file

  1. // Copyright 2015 CoreOS, Inc.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package config
  15. import (
  16. "fmt"
  17. "reflect"
  18. "regexp"
  19. "strings"
  20. "github.com/coreos/yaml"
  21. )
  22. // CloudConfig encapsulates the entire cloud-config configuration file and maps
  23. // directly to YAML. Fields that cannot be set in the cloud-config (fields
  24. // used for internal use) have the YAML tag '-' so that they aren't marshalled.
  25. type CloudConfig struct {
  26. SSHAuthorizedKeys []string `yaml:"ssh_authorized_keys"`
  27. CoreOS CoreOS `yaml:"coreos"`
  28. WriteFiles []File `yaml:"write_files"`
  29. Hostname string `yaml:"hostname"`
  30. Users []User `yaml:"users"`
  31. ManageEtcHosts EtcHosts `yaml:"manage_etc_hosts"`
  32. }
  33. type CoreOS struct {
  34. Etcd Etcd `yaml:"etcd"`
  35. Flannel Flannel `yaml:"flannel"`
  36. Fleet Fleet `yaml:"fleet"`
  37. Locksmith Locksmith `yaml:"locksmith"`
  38. OEM OEM `yaml:"oem"`
  39. Update Update `yaml:"update"`
  40. Units []Unit `yaml:"units"`
  41. }
  42. func IsCloudConfig(userdata string) bool {
  43. header := strings.SplitN(userdata, "\n", 2)[0]
  44. // Explicitly trim the header so we can handle user-data from
  45. // non-unix operating systems. The rest of the file is parsed
  46. // by yaml, which correctly handles CRLF.
  47. header = strings.TrimSuffix(header, "\r")
  48. return (header == "#cloud-config")
  49. }
  50. // NewCloudConfig instantiates a new CloudConfig from the given contents (a
  51. // string of YAML), returning any error encountered. It will ignore unknown
  52. // fields but log encountering them.
  53. func NewCloudConfig(contents string) (*CloudConfig, error) {
  54. yaml.UnmarshalMappingKeyTransform = func(nameIn string) (nameOut string) {
  55. return strings.Replace(nameIn, "-", "_", -1)
  56. }
  57. var cfg CloudConfig
  58. err := yaml.Unmarshal([]byte(contents), &cfg)
  59. return &cfg, err
  60. }
  61. func (cc CloudConfig) String() string {
  62. bytes, err := yaml.Marshal(cc)
  63. if err != nil {
  64. return ""
  65. }
  66. stringified := string(bytes)
  67. stringified = fmt.Sprintf("#cloud-config\n%s", stringified)
  68. return stringified
  69. }
  70. // IsZero returns whether or not the parameter is the zero value for its type.
  71. // If the parameter is a struct, only the exported fields are considered.
  72. func IsZero(c interface{}) bool {
  73. return isZero(reflect.ValueOf(c))
  74. }
  75. type ErrorValid struct {
  76. Value string
  77. Valid string
  78. Field string
  79. }
  80. func (e ErrorValid) Error() string {
  81. return fmt.Sprintf("invalid value %q for option %q (valid options: %q)", e.Value, e.Field, e.Valid)
  82. }
  83. // AssertStructValid checks the fields in the structure and makes sure that
  84. // they contain valid values as specified by the 'valid' flag. Empty fields are
  85. // implicitly valid.
  86. func AssertStructValid(c interface{}) error {
  87. ct := reflect.TypeOf(c)
  88. cv := reflect.ValueOf(c)
  89. for i := 0; i < ct.NumField(); i++ {
  90. ft := ct.Field(i)
  91. if !isFieldExported(ft) {
  92. continue
  93. }
  94. if err := AssertValid(cv.Field(i), ft.Tag.Get("valid")); err != nil {
  95. err.Field = ft.Name
  96. return err
  97. }
  98. }
  99. return nil
  100. }
  101. // AssertValid checks to make sure that the given value is in the list of
  102. // valid values. Zero values are implicitly valid.
  103. func AssertValid(value reflect.Value, valid string) *ErrorValid {
  104. if valid == "" || isZero(value) {
  105. return nil
  106. }
  107. vs := fmt.Sprintf("%v", value.Interface())
  108. if m, _ := regexp.MatchString(valid, vs); m {
  109. return nil
  110. }
  111. return &ErrorValid{
  112. Value: vs,
  113. Valid: valid,
  114. }
  115. }
  116. func isZero(v reflect.Value) bool {
  117. switch v.Kind() {
  118. case reflect.Struct:
  119. vt := v.Type()
  120. for i := 0; i < v.NumField(); i++ {
  121. if isFieldExported(vt.Field(i)) && !isZero(v.Field(i)) {
  122. return false
  123. }
  124. }
  125. return true
  126. default:
  127. return v.Interface() == reflect.Zero(v.Type()).Interface()
  128. }
  129. }
  130. func isFieldExported(f reflect.StructField) bool {
  131. return f.PkgPath == ""
  132. }