codec_factory.go 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  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 serializer
  14. import (
  15. "k8s.io/apimachinery/pkg/runtime"
  16. "k8s.io/apimachinery/pkg/runtime/schema"
  17. "k8s.io/apimachinery/pkg/runtime/serializer/json"
  18. "k8s.io/apimachinery/pkg/runtime/serializer/recognizer"
  19. "k8s.io/apimachinery/pkg/runtime/serializer/versioning"
  20. )
  21. // serializerExtensions are for serializers that are conditionally compiled in
  22. var serializerExtensions = []func(*runtime.Scheme) (serializerType, bool){}
  23. type serializerType struct {
  24. AcceptContentTypes []string
  25. ContentType string
  26. FileExtensions []string
  27. // EncodesAsText should be true if this content type can be represented safely in UTF-8
  28. EncodesAsText bool
  29. Serializer runtime.Serializer
  30. PrettySerializer runtime.Serializer
  31. AcceptStreamContentTypes []string
  32. StreamContentType string
  33. Framer runtime.Framer
  34. StreamSerializer runtime.Serializer
  35. }
  36. func newSerializersForScheme(scheme *runtime.Scheme, mf json.MetaFactory) []serializerType {
  37. jsonSerializer := json.NewSerializer(mf, scheme, scheme, false)
  38. jsonPrettySerializer := json.NewSerializer(mf, scheme, scheme, true)
  39. yamlSerializer := json.NewYAMLSerializer(mf, scheme, scheme)
  40. serializers := []serializerType{
  41. {
  42. AcceptContentTypes: []string{"application/json"},
  43. ContentType: "application/json",
  44. FileExtensions: []string{"json"},
  45. EncodesAsText: true,
  46. Serializer: jsonSerializer,
  47. PrettySerializer: jsonPrettySerializer,
  48. Framer: json.Framer,
  49. StreamSerializer: jsonSerializer,
  50. },
  51. {
  52. AcceptContentTypes: []string{"application/yaml"},
  53. ContentType: "application/yaml",
  54. FileExtensions: []string{"yaml"},
  55. EncodesAsText: true,
  56. Serializer: yamlSerializer,
  57. },
  58. }
  59. for _, fn := range serializerExtensions {
  60. if serializer, ok := fn(scheme); ok {
  61. serializers = append(serializers, serializer)
  62. }
  63. }
  64. return serializers
  65. }
  66. // CodecFactory provides methods for retrieving codecs and serializers for specific
  67. // versions and content types.
  68. type CodecFactory struct {
  69. scheme *runtime.Scheme
  70. serializers []serializerType
  71. universal runtime.Decoder
  72. accepts []runtime.SerializerInfo
  73. legacySerializer runtime.Serializer
  74. }
  75. // NewCodecFactory provides methods for retrieving serializers for the supported wire formats
  76. // and conversion wrappers to define preferred internal and external versions. In the future,
  77. // as the internal version is used less, callers may instead use a defaulting serializer and
  78. // only convert objects which are shared internally (Status, common API machinery).
  79. // TODO: allow other codecs to be compiled in?
  80. // TODO: accept a scheme interface
  81. func NewCodecFactory(scheme *runtime.Scheme) CodecFactory {
  82. serializers := newSerializersForScheme(scheme, json.DefaultMetaFactory)
  83. return newCodecFactory(scheme, serializers)
  84. }
  85. // newCodecFactory is a helper for testing that allows a different metafactory to be specified.
  86. func newCodecFactory(scheme *runtime.Scheme, serializers []serializerType) CodecFactory {
  87. decoders := make([]runtime.Decoder, 0, len(serializers))
  88. var accepts []runtime.SerializerInfo
  89. alreadyAccepted := make(map[string]struct{})
  90. var legacySerializer runtime.Serializer
  91. for _, d := range serializers {
  92. decoders = append(decoders, d.Serializer)
  93. for _, mediaType := range d.AcceptContentTypes {
  94. if _, ok := alreadyAccepted[mediaType]; ok {
  95. continue
  96. }
  97. alreadyAccepted[mediaType] = struct{}{}
  98. info := runtime.SerializerInfo{
  99. MediaType: d.ContentType,
  100. EncodesAsText: d.EncodesAsText,
  101. Serializer: d.Serializer,
  102. PrettySerializer: d.PrettySerializer,
  103. }
  104. if d.StreamSerializer != nil {
  105. info.StreamSerializer = &runtime.StreamSerializerInfo{
  106. Serializer: d.StreamSerializer,
  107. EncodesAsText: d.EncodesAsText,
  108. Framer: d.Framer,
  109. }
  110. }
  111. accepts = append(accepts, info)
  112. if mediaType == runtime.ContentTypeJSON {
  113. legacySerializer = d.Serializer
  114. }
  115. }
  116. }
  117. if legacySerializer == nil {
  118. legacySerializer = serializers[0].Serializer
  119. }
  120. return CodecFactory{
  121. scheme: scheme,
  122. serializers: serializers,
  123. universal: recognizer.NewDecoder(decoders...),
  124. accepts: accepts,
  125. legacySerializer: legacySerializer,
  126. }
  127. }
  128. // SupportedMediaTypes returns the RFC2046 media types that this factory has serializers for.
  129. func (f CodecFactory) SupportedMediaTypes() []runtime.SerializerInfo {
  130. return f.accepts
  131. }
  132. // LegacyCodec encodes output to a given API versions, and decodes output into the internal form from
  133. // any recognized source. The returned codec will always encode output to JSON. If a type is not
  134. // found in the list of versions an error will be returned.
  135. //
  136. // This method is deprecated - clients and servers should negotiate a serializer by mime-type and
  137. // invoke CodecForVersions. Callers that need only to read data should use UniversalDecoder().
  138. //
  139. // TODO: make this call exist only in pkg/api, and initialize it with the set of default versions.
  140. // All other callers will be forced to request a Codec directly.
  141. func (f CodecFactory) LegacyCodec(version ...schema.GroupVersion) runtime.Codec {
  142. return versioning.NewDefaultingCodecForScheme(f.scheme, f.legacySerializer, f.universal, schema.GroupVersions(version), runtime.InternalGroupVersioner)
  143. }
  144. // UniversalDeserializer can convert any stored data recognized by this factory into a Go object that satisfies
  145. // runtime.Object. It does not perform conversion. It does not perform defaulting.
  146. func (f CodecFactory) UniversalDeserializer() runtime.Decoder {
  147. return f.universal
  148. }
  149. // UniversalDecoder returns a runtime.Decoder capable of decoding all known API objects in all known formats. Used
  150. // by clients that do not need to encode objects but want to deserialize API objects stored on disk. Only decodes
  151. // objects in groups registered with the scheme. The GroupVersions passed may be used to select alternate
  152. // versions of objects to return - by default, runtime.APIVersionInternal is used. If any versions are specified,
  153. // unrecognized groups will be returned in the version they are encoded as (no conversion). This decoder performs
  154. // defaulting.
  155. //
  156. // TODO: the decoder will eventually be removed in favor of dealing with objects in their versioned form
  157. // TODO: only accept a group versioner
  158. func (f CodecFactory) UniversalDecoder(versions ...schema.GroupVersion) runtime.Decoder {
  159. var versioner runtime.GroupVersioner
  160. if len(versions) == 0 {
  161. versioner = runtime.InternalGroupVersioner
  162. } else {
  163. versioner = schema.GroupVersions(versions)
  164. }
  165. return f.CodecForVersions(nil, f.universal, nil, versioner)
  166. }
  167. // CodecForVersions creates a codec with the provided serializer. If an object is decoded and its group is not in the list,
  168. // it will default to runtime.APIVersionInternal. If encode is not specified for an object's group, the object is not
  169. // converted. If encode or decode are nil, no conversion is performed.
  170. func (f CodecFactory) CodecForVersions(encoder runtime.Encoder, decoder runtime.Decoder, encode runtime.GroupVersioner, decode runtime.GroupVersioner) runtime.Codec {
  171. // TODO: these are for backcompat, remove them in the future
  172. if encode == nil {
  173. encode = runtime.DisabledGroupVersioner
  174. }
  175. if decode == nil {
  176. decode = runtime.InternalGroupVersioner
  177. }
  178. return versioning.NewDefaultingCodecForScheme(f.scheme, encoder, decoder, encode, decode)
  179. }
  180. // DecoderToVersion returns a decoder that targets the provided group version.
  181. func (f CodecFactory) DecoderToVersion(decoder runtime.Decoder, gv runtime.GroupVersioner) runtime.Decoder {
  182. return f.CodecForVersions(nil, decoder, nil, gv)
  183. }
  184. // EncoderForVersion returns an encoder that targets the provided group version.
  185. func (f CodecFactory) EncoderForVersion(encoder runtime.Encoder, gv runtime.GroupVersioner) runtime.Encoder {
  186. return f.CodecForVersions(encoder, nil, gv, nil)
  187. }
  188. // DirectCodecFactory provides methods for retrieving "DirectCodec"s, which do not do conversion.
  189. type DirectCodecFactory struct {
  190. CodecFactory
  191. }
  192. // EncoderForVersion returns an encoder that does not do conversion.
  193. func (f DirectCodecFactory) EncoderForVersion(serializer runtime.Encoder, version runtime.GroupVersioner) runtime.Encoder {
  194. return versioning.DirectEncoder{
  195. Version: version,
  196. Encoder: serializer,
  197. ObjectTyper: f.CodecFactory.scheme,
  198. }
  199. }
  200. // DecoderToVersion returns an decoder that does not do conversion. gv is ignored.
  201. func (f DirectCodecFactory) DecoderToVersion(serializer runtime.Decoder, _ runtime.GroupVersioner) runtime.Decoder {
  202. return versioning.DirectDecoder{
  203. Decoder: serializer,
  204. }
  205. }