helper.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  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 unversioned
  14. import (
  15. "fmt"
  16. "k8s.io/kubernetes/pkg/api"
  17. "k8s.io/kubernetes/pkg/api/unversioned"
  18. "k8s.io/kubernetes/pkg/apimachinery/registered"
  19. "k8s.io/kubernetes/pkg/apis/apps"
  20. "k8s.io/kubernetes/pkg/apis/authentication"
  21. "k8s.io/kubernetes/pkg/apis/authorization"
  22. "k8s.io/kubernetes/pkg/apis/autoscaling"
  23. "k8s.io/kubernetes/pkg/apis/batch"
  24. "k8s.io/kubernetes/pkg/apis/certificates"
  25. "k8s.io/kubernetes/pkg/apis/extensions"
  26. "k8s.io/kubernetes/pkg/apis/policy"
  27. "k8s.io/kubernetes/pkg/apis/rbac"
  28. "k8s.io/kubernetes/pkg/client/restclient"
  29. "k8s.io/kubernetes/pkg/client/typed/discovery"
  30. "k8s.io/kubernetes/pkg/util/sets"
  31. "k8s.io/kubernetes/pkg/version"
  32. // Import solely to initialize client auth plugins.
  33. _ "k8s.io/kubernetes/plugin/pkg/client/auth"
  34. )
  35. const (
  36. legacyAPIPath = "/api"
  37. defaultAPIPath = "/apis"
  38. )
  39. // New creates a Kubernetes client for the given config. This client works with pods,
  40. // replication controllers, daemons, and services. It allows operations such as list, get, update
  41. // and delete on these objects. An error is returned if the provided configuration
  42. // is not valid.
  43. func New(c *restclient.Config) (*Client, error) {
  44. config := *c
  45. if err := SetKubernetesDefaults(&config); err != nil {
  46. return nil, err
  47. }
  48. client, err := restclient.RESTClientFor(&config)
  49. if err != nil {
  50. return nil, err
  51. }
  52. discoveryConfig := *c
  53. discoveryClient, err := discovery.NewDiscoveryClientForConfig(&discoveryConfig)
  54. if err != nil {
  55. return nil, err
  56. }
  57. var authorizationClient *AuthorizationClient
  58. if registered.IsRegistered(authorization.GroupName) {
  59. authorizationConfig := *c
  60. authorizationClient, err = NewAuthorization(&authorizationConfig)
  61. if err != nil {
  62. return nil, err
  63. }
  64. }
  65. var autoscalingClient *AutoscalingClient
  66. if registered.IsRegistered(autoscaling.GroupName) {
  67. autoscalingConfig := *c
  68. autoscalingClient, err = NewAutoscaling(&autoscalingConfig)
  69. if err != nil {
  70. return nil, err
  71. }
  72. }
  73. var authenticationClient *AuthenticationClient
  74. if registered.IsRegistered(authentication.GroupName) {
  75. authenticationConfig := *c
  76. authenticationClient, err = NewAuthentication(&authenticationConfig)
  77. if err != nil {
  78. return nil, err
  79. }
  80. }
  81. var batchClient *BatchClient
  82. if registered.IsRegistered(batch.GroupName) {
  83. batchConfig := *c
  84. batchClient, err = NewBatch(&batchConfig)
  85. if err != nil {
  86. return nil, err
  87. }
  88. }
  89. var extensionsClient *ExtensionsClient
  90. if registered.IsRegistered(extensions.GroupName) {
  91. extensionsConfig := *c
  92. extensionsClient, err = NewExtensions(&extensionsConfig)
  93. if err != nil {
  94. return nil, err
  95. }
  96. }
  97. var policyClient *PolicyClient
  98. if registered.IsRegistered(policy.GroupName) {
  99. policyConfig := *c
  100. policyClient, err = NewPolicy(&policyConfig)
  101. if err != nil {
  102. return nil, err
  103. }
  104. }
  105. var certsClient *CertificatesClient
  106. if registered.IsRegistered(certificates.GroupName) {
  107. certsConfig := *c
  108. certsClient, err = NewCertificates(&certsConfig)
  109. if err != nil {
  110. return nil, err
  111. }
  112. }
  113. var appsClient *AppsClient
  114. if registered.IsRegistered(apps.GroupName) {
  115. appsConfig := *c
  116. appsClient, err = NewApps(&appsConfig)
  117. if err != nil {
  118. return nil, err
  119. }
  120. }
  121. var rbacClient *RbacClient
  122. if registered.IsRegistered(rbac.GroupName) {
  123. rbacConfig := *c
  124. rbacClient, err = NewRbac(&rbacConfig)
  125. if err != nil {
  126. return nil, err
  127. }
  128. }
  129. return &Client{
  130. RESTClient: client,
  131. AppsClient: appsClient,
  132. AuthenticationClient: authenticationClient,
  133. AuthorizationClient: authorizationClient,
  134. AutoscalingClient: autoscalingClient,
  135. BatchClient: batchClient,
  136. CertificatesClient: certsClient,
  137. DiscoveryClient: discoveryClient,
  138. ExtensionsClient: extensionsClient,
  139. PolicyClient: policyClient,
  140. RbacClient: rbacClient,
  141. }, nil
  142. }
  143. // MatchesServerVersion queries the server to compares the build version
  144. // (git hash) of the client with the server's build version. It returns an error
  145. // if it failed to contact the server or if the versions are not an exact match.
  146. func MatchesServerVersion(client *Client, c *restclient.Config) error {
  147. var err error
  148. if client == nil {
  149. client, err = New(c)
  150. if err != nil {
  151. return err
  152. }
  153. }
  154. cVer := version.Get()
  155. sVer, err := client.Discovery().ServerVersion()
  156. if err != nil {
  157. return fmt.Errorf("couldn't read version from server: %v\n", err)
  158. }
  159. // GitVersion includes GitCommit and GitTreeState, but best to be safe?
  160. if cVer.GitVersion != sVer.GitVersion || cVer.GitCommit != sVer.GitCommit || cVer.GitTreeState != sVer.GitTreeState {
  161. return fmt.Errorf("server version (%#v) differs from client version (%#v)!\n", sVer, cVer)
  162. }
  163. return nil
  164. }
  165. // NegotiateVersion queries the server's supported api versions to find
  166. // a version that both client and server support.
  167. // - If no version is provided, try registered client versions in order of
  168. // preference.
  169. // - If version is provided, but not default config (explicitly requested via
  170. // commandline flag), and is unsupported by the server, print a warning to
  171. // stderr and try client's registered versions in order of preference.
  172. // - If version is config default, and the server does not support it,
  173. // return an error.
  174. func NegotiateVersion(client *Client, c *restclient.Config, requestedGV *unversioned.GroupVersion, clientRegisteredGVs []unversioned.GroupVersion) (*unversioned.GroupVersion, error) {
  175. var err error
  176. if client == nil {
  177. client, err = New(c)
  178. if err != nil {
  179. return nil, err
  180. }
  181. }
  182. clientVersions := sets.String{}
  183. for _, gv := range clientRegisteredGVs {
  184. clientVersions.Insert(gv.String())
  185. }
  186. groups, err := client.ServerGroups()
  187. if err != nil {
  188. // This is almost always a connection error, and higher level code should treat this as a generic error,
  189. // not a negotiation specific error.
  190. return nil, err
  191. }
  192. versions := unversioned.ExtractGroupVersions(groups)
  193. serverVersions := sets.String{}
  194. for _, v := range versions {
  195. serverVersions.Insert(v)
  196. }
  197. // If no version requested, use config version (may also be empty).
  198. // make a copy of the original so we don't risk mutating input here or in the returned value
  199. var preferredGV *unversioned.GroupVersion
  200. switch {
  201. case requestedGV != nil:
  202. t := *requestedGV
  203. preferredGV = &t
  204. case c.GroupVersion != nil:
  205. t := *c.GroupVersion
  206. preferredGV = &t
  207. }
  208. // If version explicitly requested verify that both client and server support it.
  209. // If server does not support warn, but try to negotiate a lower version.
  210. if preferredGV != nil {
  211. if !clientVersions.Has(preferredGV.String()) {
  212. return nil, fmt.Errorf("client does not support API version %q; client supported API versions: %v", preferredGV, clientVersions)
  213. }
  214. // If the server supports no versions, then we should just use the preferredGV
  215. // This can happen because discovery fails due to 403 Forbidden errors
  216. if len(serverVersions) == 0 {
  217. return preferredGV, nil
  218. }
  219. if serverVersions.Has(preferredGV.String()) {
  220. return preferredGV, nil
  221. }
  222. // If we are using an explicit config version the server does not support, fail.
  223. if (c.GroupVersion != nil) && (*preferredGV == *c.GroupVersion) {
  224. return nil, fmt.Errorf("server does not support API version %q", preferredGV)
  225. }
  226. }
  227. for _, clientGV := range clientRegisteredGVs {
  228. if serverVersions.Has(clientGV.String()) {
  229. // Version was not explicitly requested in command config (--api-version).
  230. // Ok to fall back to a supported version with a warning.
  231. // TODO: caesarxuchao: enable the warning message when we have
  232. // proper fix. Please refer to issue #14895.
  233. // if len(version) != 0 {
  234. // glog.Warningf("Server does not support API version '%s'. Falling back to '%s'.", version, clientVersion)
  235. // }
  236. t := clientGV
  237. return &t, nil
  238. }
  239. }
  240. return nil, fmt.Errorf("failed to negotiate an api version; server supports: %v, client supports: %v",
  241. serverVersions, clientVersions)
  242. }
  243. // NewOrDie creates a Kubernetes client and panics if the provided API version is not recognized.
  244. func NewOrDie(c *restclient.Config) *Client {
  245. client, err := New(c)
  246. if err != nil {
  247. panic(err)
  248. }
  249. return client
  250. }
  251. // NewInCluster is a shortcut for calling InClusterConfig() and then New().
  252. func NewInCluster() (*Client, error) {
  253. cc, err := restclient.InClusterConfig()
  254. if err != nil {
  255. return nil, err
  256. }
  257. return New(cc)
  258. }
  259. // SetKubernetesDefaults sets default values on the provided client config for accessing the
  260. // Kubernetes API or returns an error if any of the defaults are impossible or invalid.
  261. // TODO: this method needs to be split into one that sets defaults per group, expected to be fix in PR "Refactoring clientcache.go and helper.go #14592"
  262. func SetKubernetesDefaults(config *restclient.Config) error {
  263. if config.APIPath == "" {
  264. config.APIPath = legacyAPIPath
  265. }
  266. if config.GroupVersion == nil || config.GroupVersion.Group != api.GroupName {
  267. g, err := registered.Group(api.GroupName)
  268. if err != nil {
  269. return err
  270. }
  271. copyGroupVersion := g.GroupVersion
  272. config.GroupVersion = &copyGroupVersion
  273. }
  274. if config.NegotiatedSerializer == nil {
  275. config.NegotiatedSerializer = api.Codecs
  276. }
  277. return restclient.SetKubernetesDefaults(config)
  278. }
  279. func setGroupDefaults(groupName string, config *restclient.Config) error {
  280. config.APIPath = defaultAPIPath
  281. if config.UserAgent == "" {
  282. config.UserAgent = restclient.DefaultKubernetesUserAgent()
  283. }
  284. if config.GroupVersion == nil || config.GroupVersion.Group != groupName {
  285. g, err := registered.Group(groupName)
  286. if err != nil {
  287. return err
  288. }
  289. copyGroupVersion := g.GroupVersion
  290. config.GroupVersion = &copyGroupVersion
  291. }
  292. if config.NegotiatedSerializer == nil {
  293. config.NegotiatedSerializer = api.Codecs
  294. }
  295. return nil
  296. }