codec.go 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  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 codec
  14. import (
  15. "fmt"
  16. "github.com/pkg/errors"
  17. "k8s.io/klog"
  18. // ensure the core apis are installed
  19. _ "k8s.io/kubernetes/pkg/apis/core/install"
  20. "k8s.io/apimachinery/pkg/runtime"
  21. "k8s.io/apimachinery/pkg/runtime/schema"
  22. "k8s.io/apimachinery/pkg/runtime/serializer"
  23. "k8s.io/component-base/codec"
  24. "k8s.io/kubernetes/pkg/api/legacyscheme"
  25. kubeletconfig "k8s.io/kubernetes/pkg/kubelet/apis/config"
  26. "k8s.io/kubernetes/pkg/kubelet/apis/config/scheme"
  27. kubeletconfigv1beta1 "k8s.io/kubernetes/pkg/kubelet/apis/config/v1beta1"
  28. )
  29. // EncodeKubeletConfig encodes an internal KubeletConfiguration to an external YAML representation.
  30. func EncodeKubeletConfig(internal *kubeletconfig.KubeletConfiguration, targetVersion schema.GroupVersion) ([]byte, error) {
  31. encoder, err := NewKubeletconfigYAMLEncoder(targetVersion)
  32. if err != nil {
  33. return nil, err
  34. }
  35. // encoder will convert to external version
  36. data, err := runtime.Encode(encoder, internal)
  37. if err != nil {
  38. return nil, err
  39. }
  40. return data, nil
  41. }
  42. // NewKubeletconfigYAMLEncoder returns an encoder that can write objects in the kubeletconfig API group to YAML.
  43. func NewKubeletconfigYAMLEncoder(targetVersion schema.GroupVersion) (runtime.Encoder, error) {
  44. _, codecs, err := scheme.NewSchemeAndCodecs()
  45. if err != nil {
  46. return nil, err
  47. }
  48. mediaType := "application/yaml"
  49. info, ok := runtime.SerializerInfoForMediaType(codecs.SupportedMediaTypes(), mediaType)
  50. if !ok {
  51. return nil, fmt.Errorf("unsupported media type %q", mediaType)
  52. }
  53. return codecs.EncoderForVersion(info.Serializer, targetVersion), nil
  54. }
  55. // NewYAMLEncoder generates a new runtime.Encoder that encodes objects to YAML.
  56. func NewYAMLEncoder(groupName string) (runtime.Encoder, error) {
  57. // encode to YAML
  58. mediaType := "application/yaml"
  59. info, ok := runtime.SerializerInfoForMediaType(legacyscheme.Codecs.SupportedMediaTypes(), mediaType)
  60. if !ok {
  61. return nil, fmt.Errorf("unsupported media type %q", mediaType)
  62. }
  63. versions := legacyscheme.Scheme.PrioritizedVersionsForGroup(groupName)
  64. if len(versions) == 0 {
  65. return nil, fmt.Errorf("no enabled versions for group %q", groupName)
  66. }
  67. // the "best" version supposedly comes first in the list returned from legacyscheme.Registry.EnabledVersionsForGroup.
  68. return legacyscheme.Codecs.EncoderForVersion(info.Serializer, versions[0]), nil
  69. }
  70. // DecodeKubeletConfiguration decodes a serialized KubeletConfiguration to the internal type.
  71. func DecodeKubeletConfiguration(kubeletCodecs *serializer.CodecFactory, data []byte) (*kubeletconfig.KubeletConfiguration, error) {
  72. var (
  73. obj runtime.Object
  74. gvk *schema.GroupVersionKind
  75. )
  76. // The UniversalDecoder runs defaulting and returns the internal type by default.
  77. obj, gvk, err := kubeletCodecs.UniversalDecoder().Decode(data, nil, nil)
  78. if err != nil {
  79. // Try strict decoding first. If that fails decode with a lenient
  80. // decoder, which has only v1beta1 registered, and log a warning.
  81. // The lenient path is to be dropped when support for v1beta1 is dropped.
  82. if !runtime.IsStrictDecodingError(err) {
  83. return nil, errors.Wrap(err, "failed to decode")
  84. }
  85. var lenientErr error
  86. _, lenientCodecs, lenientErr := codec.NewLenientSchemeAndCodecs(
  87. kubeletconfig.AddToScheme,
  88. kubeletconfigv1beta1.AddToScheme,
  89. )
  90. if lenientErr != nil {
  91. return nil, lenientErr
  92. }
  93. obj, gvk, lenientErr = lenientCodecs.UniversalDecoder().Decode(data, nil, nil)
  94. if lenientErr != nil {
  95. // Lenient decoding failed with the current version, return the
  96. // original strict error.
  97. return nil, fmt.Errorf("failed lenient decoding: %v", err)
  98. }
  99. // Continue with the v1beta1 object that was decoded leniently, but emit a warning.
  100. klog.Warningf("using lenient decoding as strict decoding failed: %v", err)
  101. }
  102. internalKC, ok := obj.(*kubeletconfig.KubeletConfiguration)
  103. if !ok {
  104. return nil, fmt.Errorf("failed to cast object to KubeletConfiguration, unexpected type: %v", gvk)
  105. }
  106. return internalKC, nil
  107. }