field_error.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. /*
  2. Copyright 2017 The Knative Authors
  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. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package apis
  14. import (
  15. "fmt"
  16. "sort"
  17. "strings"
  18. )
  19. // CurrentField is a constant to supply as a fieldPath for when there is
  20. // a problem with the current field itself.
  21. const CurrentField = ""
  22. // FieldError is used to propagate the context of errors pertaining to
  23. // specific fields in a manner suitable for use in a recursive walk, so
  24. // that errors contain the appropriate field context.
  25. // FieldError methods are non-mutating.
  26. // +k8s:deepcopy-gen=true
  27. type FieldError struct {
  28. Message string
  29. Paths []string
  30. // Details contains an optional longer payload.
  31. // +optional
  32. Details string
  33. errors []FieldError
  34. }
  35. // FieldError implements error
  36. var _ error = (*FieldError)(nil)
  37. // ViaField is used to propagate a validation error along a field access.
  38. // For example, if a type recursively validates its "spec" via:
  39. // if err := foo.Spec.Validate(); err != nil {
  40. // // Augment any field paths with the context that they were accessed
  41. // // via "spec".
  42. // return err.ViaField("spec")
  43. // }
  44. func (fe *FieldError) ViaField(prefix ...string) *FieldError {
  45. if fe == nil {
  46. return nil
  47. }
  48. // Copy over message and details, paths will be updated and errors come
  49. // along using .Also().
  50. newErr := &FieldError{
  51. Message: fe.Message,
  52. Details: fe.Details,
  53. }
  54. // Prepend the Prefix to existing errors.
  55. newPaths := make([]string, 0, len(fe.Paths))
  56. for _, oldPath := range fe.Paths {
  57. newPaths = append(newPaths, flatten(append(prefix, oldPath)))
  58. }
  59. newErr.Paths = newPaths
  60. for _, e := range fe.errors {
  61. newErr = newErr.Also(e.ViaField(prefix...))
  62. }
  63. return newErr
  64. }
  65. // ViaIndex is used to attach an index to the next ViaField provided.
  66. // For example, if a type recursively validates a parameter that has a collection:
  67. // for i, c := range spec.Collection {
  68. // if err := doValidation(c); err != nil {
  69. // return err.ViaIndex(i).ViaField("collection")
  70. // }
  71. // }
  72. func (fe *FieldError) ViaIndex(index int) *FieldError {
  73. return fe.ViaField(asIndex(index))
  74. }
  75. // ViaFieldIndex is the short way to chain: err.ViaIndex(bar).ViaField(foo)
  76. func (fe *FieldError) ViaFieldIndex(field string, index int) *FieldError {
  77. return fe.ViaIndex(index).ViaField(field)
  78. }
  79. // ViaKey is used to attach a key to the next ViaField provided.
  80. // For example, if a type recursively validates a parameter that has a collection:
  81. // for k, v := range spec.Bag. {
  82. // if err := doValidation(v); err != nil {
  83. // return err.ViaKey(k).ViaField("bag")
  84. // }
  85. // }
  86. func (fe *FieldError) ViaKey(key string) *FieldError {
  87. return fe.ViaField(asKey(key))
  88. }
  89. // ViaFieldKey is the short way to chain: err.ViaKey(bar).ViaField(foo)
  90. func (fe *FieldError) ViaFieldKey(field string, key string) *FieldError {
  91. return fe.ViaKey(key).ViaField(field)
  92. }
  93. // Also collects errors, returns a new collection of existing errors and new errors.
  94. func (fe *FieldError) Also(errs ...*FieldError) *FieldError {
  95. var newErr *FieldError
  96. // collect the current objects errors, if it has any
  97. if !fe.isEmpty() {
  98. newErr = fe.DeepCopy()
  99. } else {
  100. newErr = &FieldError{}
  101. }
  102. // and then collect the passed in errors
  103. for _, e := range errs {
  104. if !e.isEmpty() {
  105. newErr.errors = append(newErr.errors, *e)
  106. }
  107. }
  108. if newErr.isEmpty() {
  109. return nil
  110. }
  111. return newErr
  112. }
  113. func (fe *FieldError) isEmpty() bool {
  114. if fe == nil {
  115. return true
  116. }
  117. return fe.Message == "" && fe.Details == "" && len(fe.errors) == 0 && len(fe.Paths) == 0
  118. }
  119. func (fe *FieldError) getNormalizedErrors() []FieldError {
  120. // in case we call getNormalizedErrors on a nil object, return just an empty
  121. // list. This can happen when .Error() is called on a nil object.
  122. if fe == nil {
  123. return []FieldError(nil)
  124. }
  125. var errors []FieldError
  126. // if this FieldError is a leaf,
  127. if fe.Message != "" {
  128. err := FieldError{
  129. Message: fe.Message,
  130. Paths: fe.Paths,
  131. Details: fe.Details,
  132. }
  133. errors = append(errors, err)
  134. }
  135. // and then collect all other errors recursively.
  136. for _, e := range fe.errors {
  137. errors = append(errors, e.getNormalizedErrors()...)
  138. }
  139. return errors
  140. }
  141. // Error implements error
  142. func (fe *FieldError) Error() string {
  143. var errs []string
  144. // Get the list of errors as a flat merged list.
  145. normedErrors := merge(fe.getNormalizedErrors())
  146. for _, e := range normedErrors {
  147. if e.Details == "" {
  148. errs = append(errs, fmt.Sprintf("%v: %v", e.Message, strings.Join(e.Paths, ", ")))
  149. } else {
  150. errs = append(errs, fmt.Sprintf("%v: %v\n%v", e.Message, strings.Join(e.Paths, ", "), e.Details))
  151. }
  152. }
  153. return strings.Join(errs, "\n")
  154. }
  155. // Helpers ---
  156. func asIndex(index int) string {
  157. return fmt.Sprintf("[%d]", index)
  158. }
  159. func isIndex(part string) bool {
  160. return strings.HasPrefix(part, "[") && strings.HasSuffix(part, "]")
  161. }
  162. func asKey(key string) string {
  163. return fmt.Sprintf("[%s]", key)
  164. }
  165. // flatten takes in a array of path components and looks for chances to flatten
  166. // objects that have index prefixes, examples:
  167. // err([0]).ViaField(bar).ViaField(foo) -> foo.bar.[0] converts to foo.bar[0]
  168. // err(bar).ViaIndex(0).ViaField(foo) -> foo.[0].bar converts to foo[0].bar
  169. // err(bar).ViaField(foo).ViaIndex(0) -> [0].foo.bar converts to [0].foo.bar
  170. // err(bar).ViaIndex(0).ViaIndex[1].ViaField(foo) -> foo.[1].[0].bar converts to foo[1][0].bar
  171. func flatten(path []string) string {
  172. var newPath []string
  173. for _, part := range path {
  174. for _, p := range strings.Split(part, ".") {
  175. if p == CurrentField {
  176. continue
  177. } else if len(newPath) > 0 && isIndex(p) {
  178. newPath[len(newPath)-1] = fmt.Sprintf("%s%s", newPath[len(newPath)-1], p)
  179. } else {
  180. newPath = append(newPath, p)
  181. }
  182. }
  183. }
  184. return strings.Join(newPath, ".")
  185. }
  186. // mergePaths takes in two string slices and returns the combination of them
  187. // without any duplicate entries.
  188. func mergePaths(a, b []string) []string {
  189. newPaths := make([]string, 0, len(a)+len(b))
  190. newPaths = append(newPaths, a...)
  191. for _, bi := range b {
  192. if !containsString(newPaths, bi) {
  193. newPaths = append(newPaths, bi)
  194. }
  195. }
  196. return newPaths
  197. }
  198. // containsString takes in a string slice and looks for the provided string
  199. // within the slice.
  200. func containsString(slice []string, s string) bool {
  201. for _, item := range slice {
  202. if item == s {
  203. return true
  204. }
  205. }
  206. return false
  207. }
  208. // merge takes in a flat list of FieldErrors and returns back a merged list of
  209. // FiledErrors. FieldErrors have their Paths combined (and de-duped) if their
  210. // Message and Details are the same. Merge will not inspect FieldError.errors.
  211. // Merge will also sort the .Path slice, and the errors slice before returning.
  212. func merge(errs []FieldError) []FieldError {
  213. // make a map big enough for all the errors.
  214. m := make(map[string]FieldError, len(errs))
  215. // Convert errs to a map where the key is <message>-<details> and the value
  216. // is the error. If an error already exists in the map with the same key,
  217. // then the paths will be merged.
  218. for _, e := range errs {
  219. k := key(&e)
  220. if v, ok := m[k]; ok {
  221. // Found a match, merge the keys.
  222. v.Paths = mergePaths(v.Paths, e.Paths)
  223. m[k] = v
  224. } else {
  225. // Does not exist in the map, save the error.
  226. m[k] = e
  227. }
  228. }
  229. // Take the map made previously and flatten it back out again.
  230. newErrs := make([]FieldError, 0, len(m))
  231. for _, v := range m {
  232. // While we have access to the merged paths, sort them too.
  233. sort.Slice(v.Paths, func(i, j int) bool { return v.Paths[i] < v.Paths[j] })
  234. newErrs = append(newErrs, v)
  235. }
  236. // Sort the flattened map.
  237. sort.Slice(newErrs, func(i, j int) bool {
  238. if newErrs[i].Message == newErrs[j].Message {
  239. return newErrs[i].Details < newErrs[j].Details
  240. }
  241. return newErrs[i].Message < newErrs[j].Message
  242. })
  243. // return back the merged list of sorted errors.
  244. return newErrs
  245. }
  246. // key returns the key using the fields .Message and .Details.
  247. func key(err *FieldError) string {
  248. return fmt.Sprintf("%s-%s", err.Message, err.Details)
  249. }
  250. // Public helpers ---
  251. // ErrMissingField is a variadic helper method for constructing a FieldError for
  252. // a set of missing fields.
  253. func ErrMissingField(fieldPaths ...string) *FieldError {
  254. return &FieldError{
  255. Message: "missing field(s)",
  256. Paths: fieldPaths,
  257. }
  258. }
  259. // ErrDisallowedFields is a variadic helper method for constructing a FieldError
  260. // for a set of disallowed fields.
  261. func ErrDisallowedFields(fieldPaths ...string) *FieldError {
  262. return &FieldError{
  263. Message: "must not set the field(s)",
  264. Paths: fieldPaths,
  265. }
  266. }
  267. // ErrInvalidValue constructs a FieldError for a field that has received an
  268. // invalid string value.
  269. func ErrInvalidValue(value, fieldPath string) *FieldError {
  270. return &FieldError{
  271. Message: fmt.Sprintf("invalid value %q", value),
  272. Paths: []string{fieldPath},
  273. }
  274. }
  275. // ErrMissingOneOf is a variadic helper method for constructing a FieldError for
  276. // not having at least one field in a mutually exclusive field group.
  277. func ErrMissingOneOf(fieldPaths ...string) *FieldError {
  278. return &FieldError{
  279. Message: "expected exactly one, got neither",
  280. Paths: fieldPaths,
  281. }
  282. }
  283. // ErrMultipleOneOf is a variadic helper method for constructing a FieldError
  284. // for having more than one field set in a mutually exclusive field group.
  285. func ErrMultipleOneOf(fieldPaths ...string) *FieldError {
  286. return &FieldError{
  287. Message: "expected exactly one, got both",
  288. Paths: fieldPaths,
  289. }
  290. }
  291. // ErrInvalidKeyName is a variadic helper method for constructing a FieldError
  292. // that specifies a key name that is invalid.
  293. func ErrInvalidKeyName(value, fieldPath string, details ...string) *FieldError {
  294. return &FieldError{
  295. Message: fmt.Sprintf("invalid key name %q", value),
  296. Paths: []string{fieldPath},
  297. Details: strings.Join(details, ", "),
  298. }
  299. }
  300. // ErrOutOFBoundsValue constructs a FieldError for a field that has received an
  301. // out of bound value.
  302. func ErrOutOfBoundsValue(value, lower, upper, fieldPath string) *FieldError {
  303. return &FieldError{
  304. Message: fmt.Sprintf("expected %s <= %s <= %s", lower, value, upper),
  305. Paths: []string{fieldPath},
  306. }
  307. }