converter.go 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710
  1. /*
  2. Copyright 2017 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. encodingjson "encoding/json"
  16. "fmt"
  17. "math"
  18. "os"
  19. "reflect"
  20. "strconv"
  21. "strings"
  22. "sync"
  23. "sync/atomic"
  24. "time"
  25. "k8s.io/apimachinery/pkg/conversion"
  26. "k8s.io/apimachinery/pkg/util/json"
  27. utilruntime "k8s.io/apimachinery/pkg/util/runtime"
  28. "sigs.k8s.io/structured-merge-diff/v4/value"
  29. "k8s.io/klog/v2"
  30. )
  31. // UnstructuredConverter is an interface for converting between interface{}
  32. // and map[string]interface representation.
  33. type UnstructuredConverter interface {
  34. ToUnstructured(obj interface{}) (map[string]interface{}, error)
  35. FromUnstructured(u map[string]interface{}, obj interface{}) error
  36. }
  37. type structField struct {
  38. structType reflect.Type
  39. field int
  40. }
  41. type fieldInfo struct {
  42. name string
  43. nameValue reflect.Value
  44. omitempty bool
  45. }
  46. type fieldsCacheMap map[structField]*fieldInfo
  47. type fieldsCache struct {
  48. sync.Mutex
  49. value atomic.Value
  50. }
  51. func newFieldsCache() *fieldsCache {
  52. cache := &fieldsCache{}
  53. cache.value.Store(make(fieldsCacheMap))
  54. return cache
  55. }
  56. var (
  57. mapStringInterfaceType = reflect.TypeOf(map[string]interface{}{})
  58. stringType = reflect.TypeOf(string(""))
  59. fieldCache = newFieldsCache()
  60. // DefaultUnstructuredConverter performs unstructured to Go typed object conversions.
  61. DefaultUnstructuredConverter = &unstructuredConverter{
  62. mismatchDetection: parseBool(os.Getenv("KUBE_PATCH_CONVERSION_DETECTOR")),
  63. comparison: conversion.EqualitiesOrDie(
  64. func(a, b time.Time) bool {
  65. return a.UTC() == b.UTC()
  66. },
  67. ),
  68. }
  69. )
  70. func parseBool(key string) bool {
  71. if len(key) == 0 {
  72. return false
  73. }
  74. value, err := strconv.ParseBool(key)
  75. if err != nil {
  76. utilruntime.HandleError(fmt.Errorf("couldn't parse '%s' as bool for unstructured mismatch detection", key))
  77. }
  78. return value
  79. }
  80. // unstructuredConverter knows how to convert between interface{} and
  81. // Unstructured in both ways.
  82. type unstructuredConverter struct {
  83. // If true, we will be additionally running conversion via json
  84. // to ensure that the result is true.
  85. // This is supposed to be set only in tests.
  86. mismatchDetection bool
  87. // comparison is the default test logic used to compare
  88. comparison conversion.Equalities
  89. }
  90. // NewTestUnstructuredConverter creates an UnstructuredConverter that accepts JSON typed maps and translates them
  91. // to Go types via reflection. It performs mismatch detection automatically and is intended for use by external
  92. // test tools. Use DefaultUnstructuredConverter if you do not explicitly need mismatch detection.
  93. func NewTestUnstructuredConverter(comparison conversion.Equalities) UnstructuredConverter {
  94. return &unstructuredConverter{
  95. mismatchDetection: true,
  96. comparison: comparison,
  97. }
  98. }
  99. // FromUnstructured converts an object from map[string]interface{} representation into a concrete type.
  100. // It uses encoding/json/Unmarshaler if object implements it or reflection if not.
  101. func (c *unstructuredConverter) FromUnstructured(u map[string]interface{}, obj interface{}) error {
  102. t := reflect.TypeOf(obj)
  103. value := reflect.ValueOf(obj)
  104. if t.Kind() != reflect.Ptr || value.IsNil() {
  105. return fmt.Errorf("FromUnstructured requires a non-nil pointer to an object, got %v", t)
  106. }
  107. err := fromUnstructured(reflect.ValueOf(u), value.Elem())
  108. if c.mismatchDetection {
  109. newObj := reflect.New(t.Elem()).Interface()
  110. newErr := fromUnstructuredViaJSON(u, newObj)
  111. if (err != nil) != (newErr != nil) {
  112. klog.Fatalf("FromUnstructured unexpected error for %v: error: %v", u, err)
  113. }
  114. if err == nil && !c.comparison.DeepEqual(obj, newObj) {
  115. klog.Fatalf("FromUnstructured mismatch\nobj1: %#v\nobj2: %#v", obj, newObj)
  116. }
  117. }
  118. return err
  119. }
  120. func fromUnstructuredViaJSON(u map[string]interface{}, obj interface{}) error {
  121. data, err := json.Marshal(u)
  122. if err != nil {
  123. return err
  124. }
  125. return json.Unmarshal(data, obj)
  126. }
  127. func fromUnstructured(sv, dv reflect.Value) error {
  128. sv = unwrapInterface(sv)
  129. if !sv.IsValid() {
  130. dv.Set(reflect.Zero(dv.Type()))
  131. return nil
  132. }
  133. st, dt := sv.Type(), dv.Type()
  134. switch dt.Kind() {
  135. case reflect.Map, reflect.Slice, reflect.Ptr, reflect.Struct, reflect.Interface:
  136. // Those require non-trivial conversion.
  137. default:
  138. // This should handle all simple types.
  139. if st.AssignableTo(dt) {
  140. dv.Set(sv)
  141. return nil
  142. }
  143. // We cannot simply use "ConvertibleTo", as JSON doesn't support conversions
  144. // between those four groups: bools, integers, floats and string. We need to
  145. // do the same.
  146. if st.ConvertibleTo(dt) {
  147. switch st.Kind() {
  148. case reflect.String:
  149. switch dt.Kind() {
  150. case reflect.String:
  151. dv.Set(sv.Convert(dt))
  152. return nil
  153. }
  154. case reflect.Bool:
  155. switch dt.Kind() {
  156. case reflect.Bool:
  157. dv.Set(sv.Convert(dt))
  158. return nil
  159. }
  160. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
  161. reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  162. switch dt.Kind() {
  163. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
  164. reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  165. dv.Set(sv.Convert(dt))
  166. return nil
  167. case reflect.Float32, reflect.Float64:
  168. dv.Set(sv.Convert(dt))
  169. return nil
  170. }
  171. case reflect.Float32, reflect.Float64:
  172. switch dt.Kind() {
  173. case reflect.Float32, reflect.Float64:
  174. dv.Set(sv.Convert(dt))
  175. return nil
  176. }
  177. if sv.Float() == math.Trunc(sv.Float()) {
  178. dv.Set(sv.Convert(dt))
  179. return nil
  180. }
  181. }
  182. return fmt.Errorf("cannot convert %s to %s", st.String(), dt.String())
  183. }
  184. }
  185. // Check if the object has a custom JSON marshaller/unmarshaller.
  186. entry := value.TypeReflectEntryOf(dv.Type())
  187. if entry.CanConvertFromUnstructured() {
  188. return entry.FromUnstructured(sv, dv)
  189. }
  190. switch dt.Kind() {
  191. case reflect.Map:
  192. return mapFromUnstructured(sv, dv)
  193. case reflect.Slice:
  194. return sliceFromUnstructured(sv, dv)
  195. case reflect.Ptr:
  196. return pointerFromUnstructured(sv, dv)
  197. case reflect.Struct:
  198. return structFromUnstructured(sv, dv)
  199. case reflect.Interface:
  200. return interfaceFromUnstructured(sv, dv)
  201. default:
  202. return fmt.Errorf("unrecognized type: %v", dt.Kind())
  203. }
  204. }
  205. func fieldInfoFromField(structType reflect.Type, field int) *fieldInfo {
  206. fieldCacheMap := fieldCache.value.Load().(fieldsCacheMap)
  207. if info, ok := fieldCacheMap[structField{structType, field}]; ok {
  208. return info
  209. }
  210. // Cache miss - we need to compute the field name.
  211. info := &fieldInfo{}
  212. typeField := structType.Field(field)
  213. jsonTag := typeField.Tag.Get("json")
  214. if len(jsonTag) == 0 {
  215. // Make the first character lowercase.
  216. if typeField.Name == "" {
  217. info.name = typeField.Name
  218. } else {
  219. info.name = strings.ToLower(typeField.Name[:1]) + typeField.Name[1:]
  220. }
  221. } else {
  222. items := strings.Split(jsonTag, ",")
  223. info.name = items[0]
  224. for i := range items {
  225. if items[i] == "omitempty" {
  226. info.omitempty = true
  227. break
  228. }
  229. }
  230. }
  231. info.nameValue = reflect.ValueOf(info.name)
  232. fieldCache.Lock()
  233. defer fieldCache.Unlock()
  234. fieldCacheMap = fieldCache.value.Load().(fieldsCacheMap)
  235. newFieldCacheMap := make(fieldsCacheMap)
  236. for k, v := range fieldCacheMap {
  237. newFieldCacheMap[k] = v
  238. }
  239. newFieldCacheMap[structField{structType, field}] = info
  240. fieldCache.value.Store(newFieldCacheMap)
  241. return info
  242. }
  243. func unwrapInterface(v reflect.Value) reflect.Value {
  244. for v.Kind() == reflect.Interface {
  245. v = v.Elem()
  246. }
  247. return v
  248. }
  249. func mapFromUnstructured(sv, dv reflect.Value) error {
  250. st, dt := sv.Type(), dv.Type()
  251. if st.Kind() != reflect.Map {
  252. return fmt.Errorf("cannot restore map from %v", st.Kind())
  253. }
  254. if !st.Key().AssignableTo(dt.Key()) && !st.Key().ConvertibleTo(dt.Key()) {
  255. return fmt.Errorf("cannot copy map with non-assignable keys: %v %v", st.Key(), dt.Key())
  256. }
  257. if sv.IsNil() {
  258. dv.Set(reflect.Zero(dt))
  259. return nil
  260. }
  261. dv.Set(reflect.MakeMap(dt))
  262. for _, key := range sv.MapKeys() {
  263. value := reflect.New(dt.Elem()).Elem()
  264. if val := unwrapInterface(sv.MapIndex(key)); val.IsValid() {
  265. if err := fromUnstructured(val, value); err != nil {
  266. return err
  267. }
  268. } else {
  269. value.Set(reflect.Zero(dt.Elem()))
  270. }
  271. if st.Key().AssignableTo(dt.Key()) {
  272. dv.SetMapIndex(key, value)
  273. } else {
  274. dv.SetMapIndex(key.Convert(dt.Key()), value)
  275. }
  276. }
  277. return nil
  278. }
  279. func sliceFromUnstructured(sv, dv reflect.Value) error {
  280. st, dt := sv.Type(), dv.Type()
  281. if st.Kind() == reflect.String && dt.Elem().Kind() == reflect.Uint8 {
  282. // We store original []byte representation as string.
  283. // This conversion is allowed, but we need to be careful about
  284. // marshaling data appropriately.
  285. if len(sv.Interface().(string)) > 0 {
  286. marshalled, err := json.Marshal(sv.Interface())
  287. if err != nil {
  288. return fmt.Errorf("error encoding %s to json: %v", st, err)
  289. }
  290. // TODO: Is this Unmarshal needed?
  291. var data []byte
  292. err = json.Unmarshal(marshalled, &data)
  293. if err != nil {
  294. return fmt.Errorf("error decoding from json: %v", err)
  295. }
  296. dv.SetBytes(data)
  297. } else {
  298. dv.Set(reflect.Zero(dt))
  299. }
  300. return nil
  301. }
  302. if st.Kind() != reflect.Slice {
  303. return fmt.Errorf("cannot restore slice from %v", st.Kind())
  304. }
  305. if sv.IsNil() {
  306. dv.Set(reflect.Zero(dt))
  307. return nil
  308. }
  309. dv.Set(reflect.MakeSlice(dt, sv.Len(), sv.Cap()))
  310. for i := 0; i < sv.Len(); i++ {
  311. if err := fromUnstructured(sv.Index(i), dv.Index(i)); err != nil {
  312. return err
  313. }
  314. }
  315. return nil
  316. }
  317. func pointerFromUnstructured(sv, dv reflect.Value) error {
  318. st, dt := sv.Type(), dv.Type()
  319. if st.Kind() == reflect.Ptr && sv.IsNil() {
  320. dv.Set(reflect.Zero(dt))
  321. return nil
  322. }
  323. dv.Set(reflect.New(dt.Elem()))
  324. switch st.Kind() {
  325. case reflect.Ptr, reflect.Interface:
  326. return fromUnstructured(sv.Elem(), dv.Elem())
  327. default:
  328. return fromUnstructured(sv, dv.Elem())
  329. }
  330. }
  331. func structFromUnstructured(sv, dv reflect.Value) error {
  332. st, dt := sv.Type(), dv.Type()
  333. if st.Kind() != reflect.Map {
  334. return fmt.Errorf("cannot restore struct from: %v", st.Kind())
  335. }
  336. for i := 0; i < dt.NumField(); i++ {
  337. fieldInfo := fieldInfoFromField(dt, i)
  338. fv := dv.Field(i)
  339. if len(fieldInfo.name) == 0 {
  340. // This field is inlined.
  341. if err := fromUnstructured(sv, fv); err != nil {
  342. return err
  343. }
  344. } else {
  345. value := unwrapInterface(sv.MapIndex(fieldInfo.nameValue))
  346. if value.IsValid() {
  347. if err := fromUnstructured(value, fv); err != nil {
  348. return err
  349. }
  350. } else {
  351. fv.Set(reflect.Zero(fv.Type()))
  352. }
  353. }
  354. }
  355. return nil
  356. }
  357. func interfaceFromUnstructured(sv, dv reflect.Value) error {
  358. // TODO: Is this conversion safe?
  359. dv.Set(sv)
  360. return nil
  361. }
  362. // ToUnstructured converts an object into map[string]interface{} representation.
  363. // It uses encoding/json/Marshaler if object implements it or reflection if not.
  364. func (c *unstructuredConverter) ToUnstructured(obj interface{}) (map[string]interface{}, error) {
  365. var u map[string]interface{}
  366. var err error
  367. if unstr, ok := obj.(Unstructured); ok {
  368. u = unstr.UnstructuredContent()
  369. } else {
  370. t := reflect.TypeOf(obj)
  371. value := reflect.ValueOf(obj)
  372. if t.Kind() != reflect.Ptr || value.IsNil() {
  373. return nil, fmt.Errorf("ToUnstructured requires a non-nil pointer to an object, got %v", t)
  374. }
  375. u = map[string]interface{}{}
  376. err = toUnstructured(value.Elem(), reflect.ValueOf(&u).Elem())
  377. }
  378. if c.mismatchDetection {
  379. newUnstr := map[string]interface{}{}
  380. newErr := toUnstructuredViaJSON(obj, &newUnstr)
  381. if (err != nil) != (newErr != nil) {
  382. klog.Fatalf("ToUnstructured unexpected error for %v: error: %v; newErr: %v", obj, err, newErr)
  383. }
  384. if err == nil && !c.comparison.DeepEqual(u, newUnstr) {
  385. klog.Fatalf("ToUnstructured mismatch\nobj1: %#v\nobj2: %#v", u, newUnstr)
  386. }
  387. }
  388. if err != nil {
  389. return nil, err
  390. }
  391. return u, nil
  392. }
  393. // DeepCopyJSON deep copies the passed value, assuming it is a valid JSON representation i.e. only contains
  394. // types produced by json.Unmarshal() and also int64.
  395. // bool, int64, float64, string, []interface{}, map[string]interface{}, json.Number and nil
  396. func DeepCopyJSON(x map[string]interface{}) map[string]interface{} {
  397. return DeepCopyJSONValue(x).(map[string]interface{})
  398. }
  399. // DeepCopyJSONValue deep copies the passed value, assuming it is a valid JSON representation i.e. only contains
  400. // types produced by json.Unmarshal() and also int64.
  401. // bool, int64, float64, string, []interface{}, map[string]interface{}, json.Number and nil
  402. func DeepCopyJSONValue(x interface{}) interface{} {
  403. switch x := x.(type) {
  404. case map[string]interface{}:
  405. if x == nil {
  406. // Typed nil - an interface{} that contains a type map[string]interface{} with a value of nil
  407. return x
  408. }
  409. clone := make(map[string]interface{}, len(x))
  410. for k, v := range x {
  411. clone[k] = DeepCopyJSONValue(v)
  412. }
  413. return clone
  414. case []interface{}:
  415. if x == nil {
  416. // Typed nil - an interface{} that contains a type []interface{} with a value of nil
  417. return x
  418. }
  419. clone := make([]interface{}, len(x))
  420. for i, v := range x {
  421. clone[i] = DeepCopyJSONValue(v)
  422. }
  423. return clone
  424. case string, int64, bool, float64, nil, encodingjson.Number:
  425. return x
  426. default:
  427. panic(fmt.Errorf("cannot deep copy %T", x))
  428. }
  429. }
  430. func toUnstructuredViaJSON(obj interface{}, u *map[string]interface{}) error {
  431. data, err := json.Marshal(obj)
  432. if err != nil {
  433. return err
  434. }
  435. return json.Unmarshal(data, u)
  436. }
  437. func toUnstructured(sv, dv reflect.Value) error {
  438. // Check if the object has a custom string converter.
  439. entry := value.TypeReflectEntryOf(sv.Type())
  440. if entry.CanConvertToUnstructured() {
  441. v, err := entry.ToUnstructured(sv)
  442. if err != nil {
  443. return err
  444. }
  445. if v != nil {
  446. dv.Set(reflect.ValueOf(v))
  447. }
  448. return nil
  449. }
  450. st := sv.Type()
  451. switch st.Kind() {
  452. case reflect.String:
  453. dv.Set(reflect.ValueOf(sv.String()))
  454. return nil
  455. case reflect.Bool:
  456. dv.Set(reflect.ValueOf(sv.Bool()))
  457. return nil
  458. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  459. dv.Set(reflect.ValueOf(sv.Int()))
  460. return nil
  461. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  462. uVal := sv.Uint()
  463. if uVal > math.MaxInt64 {
  464. return fmt.Errorf("unsigned value %d does not fit into int64 (overflow)", uVal)
  465. }
  466. dv.Set(reflect.ValueOf(int64(uVal)))
  467. return nil
  468. case reflect.Float32, reflect.Float64:
  469. dv.Set(reflect.ValueOf(sv.Float()))
  470. return nil
  471. case reflect.Map:
  472. return mapToUnstructured(sv, dv)
  473. case reflect.Slice:
  474. return sliceToUnstructured(sv, dv)
  475. case reflect.Ptr:
  476. return pointerToUnstructured(sv, dv)
  477. case reflect.Struct:
  478. return structToUnstructured(sv, dv)
  479. case reflect.Interface:
  480. return interfaceToUnstructured(sv, dv)
  481. default:
  482. return fmt.Errorf("unrecognized type: %v", st.Kind())
  483. }
  484. }
  485. func mapToUnstructured(sv, dv reflect.Value) error {
  486. st, dt := sv.Type(), dv.Type()
  487. if sv.IsNil() {
  488. dv.Set(reflect.Zero(dt))
  489. return nil
  490. }
  491. if dt.Kind() == reflect.Interface && dv.NumMethod() == 0 {
  492. if st.Key().Kind() == reflect.String {
  493. switch st.Elem().Kind() {
  494. // TODO It should be possible to reuse the slice for primitive types.
  495. // However, it is panicing in the following form.
  496. // case reflect.String, reflect.Bool,
  497. // reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
  498. // reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  499. // sv.Set(sv)
  500. // return nil
  501. default:
  502. // We need to do a proper conversion.
  503. }
  504. }
  505. dv.Set(reflect.MakeMap(mapStringInterfaceType))
  506. dv = dv.Elem()
  507. dt = dv.Type()
  508. }
  509. if dt.Kind() != reflect.Map {
  510. return fmt.Errorf("cannot convert struct to: %v", dt.Kind())
  511. }
  512. if !st.Key().AssignableTo(dt.Key()) && !st.Key().ConvertibleTo(dt.Key()) {
  513. return fmt.Errorf("cannot copy map with non-assignable keys: %v %v", st.Key(), dt.Key())
  514. }
  515. for _, key := range sv.MapKeys() {
  516. value := reflect.New(dt.Elem()).Elem()
  517. if err := toUnstructured(sv.MapIndex(key), value); err != nil {
  518. return err
  519. }
  520. if st.Key().AssignableTo(dt.Key()) {
  521. dv.SetMapIndex(key, value)
  522. } else {
  523. dv.SetMapIndex(key.Convert(dt.Key()), value)
  524. }
  525. }
  526. return nil
  527. }
  528. func sliceToUnstructured(sv, dv reflect.Value) error {
  529. st, dt := sv.Type(), dv.Type()
  530. if sv.IsNil() {
  531. dv.Set(reflect.Zero(dt))
  532. return nil
  533. }
  534. if st.Elem().Kind() == reflect.Uint8 {
  535. dv.Set(reflect.New(stringType))
  536. data, err := json.Marshal(sv.Bytes())
  537. if err != nil {
  538. return err
  539. }
  540. var result string
  541. if err = json.Unmarshal(data, &result); err != nil {
  542. return err
  543. }
  544. dv.Set(reflect.ValueOf(result))
  545. return nil
  546. }
  547. if dt.Kind() == reflect.Interface && dv.NumMethod() == 0 {
  548. switch st.Elem().Kind() {
  549. // TODO It should be possible to reuse the slice for primitive types.
  550. // However, it is panicing in the following form.
  551. // case reflect.String, reflect.Bool,
  552. // reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
  553. // reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  554. // sv.Set(sv)
  555. // return nil
  556. default:
  557. // We need to do a proper conversion.
  558. dv.Set(reflect.MakeSlice(reflect.SliceOf(dt), sv.Len(), sv.Cap()))
  559. dv = dv.Elem()
  560. dt = dv.Type()
  561. }
  562. }
  563. if dt.Kind() != reflect.Slice {
  564. return fmt.Errorf("cannot convert slice to: %v", dt.Kind())
  565. }
  566. for i := 0; i < sv.Len(); i++ {
  567. if err := toUnstructured(sv.Index(i), dv.Index(i)); err != nil {
  568. return err
  569. }
  570. }
  571. return nil
  572. }
  573. func pointerToUnstructured(sv, dv reflect.Value) error {
  574. if sv.IsNil() {
  575. // We're done - we don't need to store anything.
  576. return nil
  577. }
  578. return toUnstructured(sv.Elem(), dv)
  579. }
  580. func isZero(v reflect.Value) bool {
  581. switch v.Kind() {
  582. case reflect.Array, reflect.String:
  583. return v.Len() == 0
  584. case reflect.Bool:
  585. return !v.Bool()
  586. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  587. return v.Int() == 0
  588. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  589. return v.Uint() == 0
  590. case reflect.Float32, reflect.Float64:
  591. return v.Float() == 0
  592. case reflect.Map, reflect.Slice:
  593. // TODO: It seems that 0-len maps are ignored in it.
  594. return v.IsNil() || v.Len() == 0
  595. case reflect.Ptr, reflect.Interface:
  596. return v.IsNil()
  597. }
  598. return false
  599. }
  600. func structToUnstructured(sv, dv reflect.Value) error {
  601. st, dt := sv.Type(), dv.Type()
  602. if dt.Kind() == reflect.Interface && dv.NumMethod() == 0 {
  603. dv.Set(reflect.MakeMapWithSize(mapStringInterfaceType, st.NumField()))
  604. dv = dv.Elem()
  605. dt = dv.Type()
  606. }
  607. if dt.Kind() != reflect.Map {
  608. return fmt.Errorf("cannot convert struct to: %v", dt.Kind())
  609. }
  610. realMap := dv.Interface().(map[string]interface{})
  611. for i := 0; i < st.NumField(); i++ {
  612. fieldInfo := fieldInfoFromField(st, i)
  613. fv := sv.Field(i)
  614. if fieldInfo.name == "-" {
  615. // This field should be skipped.
  616. continue
  617. }
  618. if fieldInfo.omitempty && isZero(fv) {
  619. // omitempty fields should be ignored.
  620. continue
  621. }
  622. if len(fieldInfo.name) == 0 {
  623. // This field is inlined.
  624. if err := toUnstructured(fv, dv); err != nil {
  625. return err
  626. }
  627. continue
  628. }
  629. switch fv.Type().Kind() {
  630. case reflect.String:
  631. realMap[fieldInfo.name] = fv.String()
  632. case reflect.Bool:
  633. realMap[fieldInfo.name] = fv.Bool()
  634. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  635. realMap[fieldInfo.name] = fv.Int()
  636. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  637. realMap[fieldInfo.name] = fv.Uint()
  638. case reflect.Float32, reflect.Float64:
  639. realMap[fieldInfo.name] = fv.Float()
  640. default:
  641. subv := reflect.New(dt.Elem()).Elem()
  642. if err := toUnstructured(fv, subv); err != nil {
  643. return err
  644. }
  645. dv.SetMapIndex(fieldInfo.nameValue, subv)
  646. }
  647. }
  648. return nil
  649. }
  650. func interfaceToUnstructured(sv, dv reflect.Value) error {
  651. if !sv.IsValid() || sv.IsNil() {
  652. dv.Set(reflect.Zero(dv.Type()))
  653. return nil
  654. }
  655. return toUnstructured(sv.Elem(), dv)
  656. }