helper.go 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. /*
  2. Copyright 2014 The Kubernetes 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 runtime
  14. import (
  15. "fmt"
  16. "io"
  17. "reflect"
  18. "k8s.io/apimachinery/pkg/conversion"
  19. "k8s.io/apimachinery/pkg/runtime/schema"
  20. "k8s.io/apimachinery/pkg/util/errors"
  21. )
  22. // unsafeObjectConvertor implements ObjectConvertor using the unsafe conversion path.
  23. type unsafeObjectConvertor struct {
  24. *Scheme
  25. }
  26. var _ ObjectConvertor = unsafeObjectConvertor{}
  27. // ConvertToVersion converts in to the provided outVersion without copying the input first, which
  28. // is only safe if the output object is not mutated or reused.
  29. func (c unsafeObjectConvertor) ConvertToVersion(in Object, outVersion GroupVersioner) (Object, error) {
  30. return c.Scheme.UnsafeConvertToVersion(in, outVersion)
  31. }
  32. // UnsafeObjectConvertor performs object conversion without copying the object structure,
  33. // for use when the converted object will not be reused or mutated. Primarily for use within
  34. // versioned codecs, which use the external object for serialization but do not return it.
  35. func UnsafeObjectConvertor(scheme *Scheme) ObjectConvertor {
  36. return unsafeObjectConvertor{scheme}
  37. }
  38. // SetField puts the value of src, into fieldName, which must be a member of v.
  39. // The value of src must be assignable to the field.
  40. func SetField(src interface{}, v reflect.Value, fieldName string) error {
  41. field := v.FieldByName(fieldName)
  42. if !field.IsValid() {
  43. return fmt.Errorf("couldn't find %v field in %#v", fieldName, v.Interface())
  44. }
  45. srcValue := reflect.ValueOf(src)
  46. if srcValue.Type().AssignableTo(field.Type()) {
  47. field.Set(srcValue)
  48. return nil
  49. }
  50. if srcValue.Type().ConvertibleTo(field.Type()) {
  51. field.Set(srcValue.Convert(field.Type()))
  52. return nil
  53. }
  54. return fmt.Errorf("couldn't assign/convert %v to %v", srcValue.Type(), field.Type())
  55. }
  56. // Field puts the value of fieldName, which must be a member of v, into dest,
  57. // which must be a variable to which this field's value can be assigned.
  58. func Field(v reflect.Value, fieldName string, dest interface{}) error {
  59. field := v.FieldByName(fieldName)
  60. if !field.IsValid() {
  61. return fmt.Errorf("couldn't find %v field in %#v", fieldName, v.Interface())
  62. }
  63. destValue, err := conversion.EnforcePtr(dest)
  64. if err != nil {
  65. return err
  66. }
  67. if field.Type().AssignableTo(destValue.Type()) {
  68. destValue.Set(field)
  69. return nil
  70. }
  71. if field.Type().ConvertibleTo(destValue.Type()) {
  72. destValue.Set(field.Convert(destValue.Type()))
  73. return nil
  74. }
  75. return fmt.Errorf("couldn't assign/convert %v to %v", field.Type(), destValue.Type())
  76. }
  77. // fieldPtr puts the address of fieldName, which must be a member of v,
  78. // into dest, which must be an address of a variable to which this field's
  79. // address can be assigned.
  80. func FieldPtr(v reflect.Value, fieldName string, dest interface{}) error {
  81. field := v.FieldByName(fieldName)
  82. if !field.IsValid() {
  83. return fmt.Errorf("couldn't find %v field in %#v", fieldName, v.Interface())
  84. }
  85. v, err := conversion.EnforcePtr(dest)
  86. if err != nil {
  87. return err
  88. }
  89. field = field.Addr()
  90. if field.Type().AssignableTo(v.Type()) {
  91. v.Set(field)
  92. return nil
  93. }
  94. if field.Type().ConvertibleTo(v.Type()) {
  95. v.Set(field.Convert(v.Type()))
  96. return nil
  97. }
  98. return fmt.Errorf("couldn't assign/convert %v to %v", field.Type(), v.Type())
  99. }
  100. // EncodeList ensures that each object in an array is converted to a Unknown{} in serialized form.
  101. // TODO: accept a content type.
  102. func EncodeList(e Encoder, objects []Object) error {
  103. var errs []error
  104. for i := range objects {
  105. data, err := Encode(e, objects[i])
  106. if err != nil {
  107. errs = append(errs, err)
  108. continue
  109. }
  110. // TODO: Set ContentEncoding and ContentType.
  111. objects[i] = &Unknown{Raw: data}
  112. }
  113. return errors.NewAggregate(errs)
  114. }
  115. func decodeListItem(obj *Unknown, decoders []Decoder) (Object, error) {
  116. for _, decoder := range decoders {
  117. // TODO: Decode based on ContentType.
  118. obj, err := Decode(decoder, obj.Raw)
  119. if err != nil {
  120. if IsNotRegisteredError(err) {
  121. continue
  122. }
  123. return nil, err
  124. }
  125. return obj, nil
  126. }
  127. // could not decode, so leave the object as Unknown, but give the decoders the
  128. // chance to set Unknown.TypeMeta if it is available.
  129. for _, decoder := range decoders {
  130. if err := DecodeInto(decoder, obj.Raw, obj); err == nil {
  131. return obj, nil
  132. }
  133. }
  134. return obj, nil
  135. }
  136. // DecodeList alters the list in place, attempting to decode any objects found in
  137. // the list that have the Unknown type. Any errors that occur are returned
  138. // after the entire list is processed. Decoders are tried in order.
  139. func DecodeList(objects []Object, decoders ...Decoder) []error {
  140. errs := []error(nil)
  141. for i, obj := range objects {
  142. switch t := obj.(type) {
  143. case *Unknown:
  144. decoded, err := decodeListItem(t, decoders)
  145. if err != nil {
  146. errs = append(errs, err)
  147. break
  148. }
  149. objects[i] = decoded
  150. }
  151. }
  152. return errs
  153. }
  154. // MultiObjectTyper returns the types of objects across multiple schemes in order.
  155. type MultiObjectTyper []ObjectTyper
  156. var _ ObjectTyper = MultiObjectTyper{}
  157. func (m MultiObjectTyper) ObjectKinds(obj Object) (gvks []schema.GroupVersionKind, unversionedType bool, err error) {
  158. for _, t := range m {
  159. gvks, unversionedType, err = t.ObjectKinds(obj)
  160. if err == nil {
  161. return
  162. }
  163. }
  164. return
  165. }
  166. func (m MultiObjectTyper) Recognizes(gvk schema.GroupVersionKind) bool {
  167. for _, t := range m {
  168. if t.Recognizes(gvk) {
  169. return true
  170. }
  171. }
  172. return false
  173. }
  174. // SetZeroValue would set the object of objPtr to zero value of its type.
  175. func SetZeroValue(objPtr Object) error {
  176. v, err := conversion.EnforcePtr(objPtr)
  177. if err != nil {
  178. return err
  179. }
  180. v.Set(reflect.Zero(v.Type()))
  181. return nil
  182. }
  183. // DefaultFramer is valid for any stream that can read objects serially without
  184. // any separation in the stream.
  185. var DefaultFramer = defaultFramer{}
  186. type defaultFramer struct{}
  187. func (defaultFramer) NewFrameReader(r io.ReadCloser) io.ReadCloser { return r }
  188. func (defaultFramer) NewFrameWriter(w io.Writer) io.Writer { return w }