PageRenderTime 50ms CodeModel.GetById 14ms RepoModel.GetById 1ms app.codeStats 0ms

/pkg/api/install/install.go

https://gitlab.com/vectorci/kubernetes
Go | 239 lines | 179 code | 26 blank | 34 comment | 29 complexity | 097442e9e95891111efde050fd2068c6 MD5 | raw file
  1. /*
  2. Copyright 2014 The Kubernetes Authors All rights reserved.
  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 install installs the v1 monolithic api, making it available as an
  14. // option to all of the API encoding/decoding machinery.
  15. package install
  16. import (
  17. "fmt"
  18. "github.com/golang/glog"
  19. "k8s.io/kubernetes/pkg/api"
  20. "k8s.io/kubernetes/pkg/api/meta"
  21. "k8s.io/kubernetes/pkg/api/unversioned"
  22. "k8s.io/kubernetes/pkg/api/v1"
  23. "k8s.io/kubernetes/pkg/apimachinery"
  24. "k8s.io/kubernetes/pkg/apimachinery/registered"
  25. "k8s.io/kubernetes/pkg/conversion"
  26. "k8s.io/kubernetes/pkg/runtime"
  27. "k8s.io/kubernetes/pkg/util/sets"
  28. )
  29. const importPrefix = "k8s.io/kubernetes/pkg/api"
  30. var accessor = meta.NewAccessor()
  31. // availableVersions lists all known external versions for this group from most preferred to least preferred
  32. var availableVersions = []unversioned.GroupVersion{v1.SchemeGroupVersion}
  33. func init() {
  34. registered.RegisterVersions(availableVersions)
  35. externalVersions := []unversioned.GroupVersion{}
  36. for _, v := range availableVersions {
  37. if registered.IsAllowedVersion(v) {
  38. externalVersions = append(externalVersions, v)
  39. }
  40. }
  41. if len(externalVersions) == 0 {
  42. glog.V(4).Infof("No version is registered for group %v", api.GroupName)
  43. return
  44. }
  45. if err := registered.EnableVersions(externalVersions...); err != nil {
  46. glog.V(4).Infof("%v", err)
  47. return
  48. }
  49. if err := enableVersions(externalVersions); err != nil {
  50. glog.V(4).Infof("%v", err)
  51. return
  52. }
  53. }
  54. // TODO: enableVersions should be centralized rather than spread in each API
  55. // group.
  56. // We can combine registered.RegisterVersions, registered.EnableVersions and
  57. // registered.RegisterGroup once we have moved enableVersions there.
  58. func enableVersions(externalVersions []unversioned.GroupVersion) error {
  59. addVersionsToScheme(externalVersions...)
  60. preferredExternalVersion := externalVersions[0]
  61. groupMeta := apimachinery.GroupMeta{
  62. GroupVersion: preferredExternalVersion,
  63. GroupVersions: externalVersions,
  64. RESTMapper: newRESTMapper(externalVersions),
  65. SelfLinker: runtime.SelfLinker(accessor),
  66. InterfacesFor: interfacesFor,
  67. }
  68. if err := registered.RegisterGroup(groupMeta); err != nil {
  69. return err
  70. }
  71. api.RegisterRESTMapper(groupMeta.RESTMapper)
  72. return nil
  73. }
  74. // userResources is a group of resources mostly used by a kubectl user
  75. var userResources = []string{"rc", "svc", "pods", "pvc"}
  76. func newRESTMapper(externalVersions []unversioned.GroupVersion) meta.RESTMapper {
  77. // the list of kinds that are scoped at the root of the api hierarchy
  78. // if a kind is not enumerated here, it is assumed to have a namespace scope
  79. rootScoped := sets.NewString(
  80. "Node",
  81. "Namespace",
  82. "PersistentVolume",
  83. "ComponentStatus",
  84. )
  85. // these kinds should be excluded from the list of resources
  86. ignoredKinds := sets.NewString(
  87. "ListOptions",
  88. "DeleteOptions",
  89. "Status",
  90. "PodLogOptions",
  91. "PodExecOptions",
  92. "PodAttachOptions",
  93. "PodProxyOptions",
  94. "NodeProxyOptions",
  95. "ServiceProxyOptions",
  96. "ThirdPartyResource",
  97. "ThirdPartyResourceData",
  98. "ThirdPartyResourceList")
  99. mapper := api.NewDefaultRESTMapper(externalVersions, interfacesFor, importPrefix, ignoredKinds, rootScoped)
  100. // setup aliases for groups of resources
  101. mapper.AddResourceAlias("all", userResources...)
  102. return mapper
  103. }
  104. // InterfacesFor returns the default Codec and ResourceVersioner for a given version
  105. // string, or an error if the version is not known.
  106. func interfacesFor(version unversioned.GroupVersion) (*meta.VersionInterfaces, error) {
  107. switch version {
  108. case v1.SchemeGroupVersion:
  109. return &meta.VersionInterfaces{
  110. ObjectConvertor: api.Scheme,
  111. MetadataAccessor: accessor,
  112. }, nil
  113. default:
  114. g, _ := registered.Group(api.GroupName)
  115. return nil, fmt.Errorf("unsupported storage version: %s (valid: %v)", version, g.GroupVersions)
  116. }
  117. }
  118. func addVersionsToScheme(externalVersions ...unversioned.GroupVersion) {
  119. // add the internal version to Scheme
  120. api.AddToScheme(api.Scheme)
  121. // add the enabled external versions to Scheme
  122. for _, v := range externalVersions {
  123. if !registered.IsEnabledVersion(v) {
  124. glog.Errorf("Version %s is not enabled, so it will not be added to the Scheme.", v)
  125. continue
  126. }
  127. switch v {
  128. case v1.SchemeGroupVersion:
  129. v1.AddToScheme(api.Scheme)
  130. }
  131. }
  132. // This is a "fast-path" that avoids reflection for common types. It focuses on the objects that are
  133. // converted the most in the cluster.
  134. // TODO: generate one of these for every external API group - this is to prove the impact
  135. api.Scheme.AddGenericConversionFunc(func(objA, objB interface{}, s conversion.Scope) (bool, error) {
  136. switch a := objA.(type) {
  137. case *v1.Pod:
  138. switch b := objB.(type) {
  139. case *api.Pod:
  140. return true, v1.Convert_v1_Pod_To_api_Pod(a, b, s)
  141. }
  142. case *api.Pod:
  143. switch b := objB.(type) {
  144. case *v1.Pod:
  145. return true, v1.Convert_api_Pod_To_v1_Pod(a, b, s)
  146. }
  147. case *v1.Event:
  148. switch b := objB.(type) {
  149. case *api.Event:
  150. return true, v1.Convert_v1_Event_To_api_Event(a, b, s)
  151. }
  152. case *api.Event:
  153. switch b := objB.(type) {
  154. case *v1.Event:
  155. return true, v1.Convert_api_Event_To_v1_Event(a, b, s)
  156. }
  157. case *v1.ReplicationController:
  158. switch b := objB.(type) {
  159. case *api.ReplicationController:
  160. return true, v1.Convert_v1_ReplicationController_To_api_ReplicationController(a, b, s)
  161. }
  162. case *api.ReplicationController:
  163. switch b := objB.(type) {
  164. case *v1.ReplicationController:
  165. return true, v1.Convert_api_ReplicationController_To_v1_ReplicationController(a, b, s)
  166. }
  167. case *v1.Node:
  168. switch b := objB.(type) {
  169. case *api.Node:
  170. return true, v1.Convert_v1_Node_To_api_Node(a, b, s)
  171. }
  172. case *api.Node:
  173. switch b := objB.(type) {
  174. case *v1.Node:
  175. return true, v1.Convert_api_Node_To_v1_Node(a, b, s)
  176. }
  177. case *v1.Namespace:
  178. switch b := objB.(type) {
  179. case *api.Namespace:
  180. return true, v1.Convert_v1_Namespace_To_api_Namespace(a, b, s)
  181. }
  182. case *api.Namespace:
  183. switch b := objB.(type) {
  184. case *v1.Namespace:
  185. return true, v1.Convert_api_Namespace_To_v1_Namespace(a, b, s)
  186. }
  187. case *v1.Service:
  188. switch b := objB.(type) {
  189. case *api.Service:
  190. return true, v1.Convert_v1_Service_To_api_Service(a, b, s)
  191. }
  192. case *api.Service:
  193. switch b := objB.(type) {
  194. case *v1.Service:
  195. return true, v1.Convert_api_Service_To_v1_Service(a, b, s)
  196. }
  197. case *v1.Endpoints:
  198. switch b := objB.(type) {
  199. case *api.Endpoints:
  200. return true, v1.Convert_v1_Endpoints_To_api_Endpoints(a, b, s)
  201. }
  202. case *api.Endpoints:
  203. switch b := objB.(type) {
  204. case *v1.Endpoints:
  205. return true, v1.Convert_api_Endpoints_To_v1_Endpoints(a, b, s)
  206. }
  207. }
  208. return false, nil
  209. })
  210. }