2014-09-30 00:15:00 +00:00
|
|
|
/*
|
2015-05-01 16:19:44 +00:00
|
|
|
Copyright 2014 The Kubernetes Authors All rights reserved.
|
2014-09-30 00:15:00 +00:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2015-09-04 07:06:01 +00:00
|
|
|
// Package testapi provides a helper for retrieving the KUBE_TEST_API environment variable.
|
2014-09-30 00:15:00 +00:00
|
|
|
package testapi
|
|
|
|
|
|
|
|
import (
|
2014-10-14 22:38:31 +00:00
|
|
|
"fmt"
|
2014-09-30 00:15:00 +00:00
|
|
|
"os"
|
2015-04-11 07:56:39 +00:00
|
|
|
"strings"
|
2014-09-30 00:15:00 +00:00
|
|
|
|
2015-09-21 23:29:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api"
|
2015-09-10 19:30:47 +00:00
|
|
|
_ "k8s.io/kubernetes/pkg/api/install"
|
2015-10-09 22:04:41 +00:00
|
|
|
_ "k8s.io/kubernetes/pkg/apis/extensions/install"
|
2015-10-30 21:23:53 +00:00
|
|
|
_ "k8s.io/kubernetes/pkg/apis/metrics/install"
|
2015-09-10 19:30:47 +00:00
|
|
|
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api/latest"
|
|
|
|
"k8s.io/kubernetes/pkg/api/meta"
|
2015-11-18 15:34:16 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api/unversioned"
|
2015-09-04 07:06:01 +00:00
|
|
|
apiutil "k8s.io/kubernetes/pkg/api/util"
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/runtime"
|
2014-09-30 00:15:00 +00:00
|
|
|
)
|
|
|
|
|
2015-09-04 07:06:01 +00:00
|
|
|
var (
|
2015-10-09 22:30:25 +00:00
|
|
|
Groups = make(map[string]TestGroup)
|
|
|
|
Default TestGroup
|
|
|
|
Extensions TestGroup
|
2015-09-04 07:06:01 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type TestGroup struct {
|
|
|
|
// Name of the group
|
|
|
|
Group string
|
|
|
|
// Version of the group Group under test
|
|
|
|
VersionUnderTest string
|
|
|
|
// Group and Version. In most cases equals to Group + "/" + VersionUnverTest
|
|
|
|
GroupVersionUnderTest string
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
kubeTestAPI := os.Getenv("KUBE_TEST_API")
|
|
|
|
if kubeTestAPI != "" {
|
|
|
|
testGroupVersions := strings.Split(kubeTestAPI, ",")
|
|
|
|
for _, groupVersion := range testGroupVersions {
|
|
|
|
// TODO: caesarxuchao: the apiutil package is hacky, it will be replaced
|
|
|
|
// by a following PR.
|
|
|
|
Groups[apiutil.GetGroup(groupVersion)] =
|
|
|
|
TestGroup{apiutil.GetGroup(groupVersion), apiutil.GetVersion(groupVersion), groupVersion}
|
|
|
|
}
|
2014-09-30 00:15:00 +00:00
|
|
|
}
|
2015-09-04 07:06:01 +00:00
|
|
|
|
|
|
|
// TODO: caesarxuchao: we need a central place to store all available API
|
|
|
|
// groups and their metadata.
|
|
|
|
if _, ok := Groups[""]; !ok {
|
2015-11-16 18:34:28 +00:00
|
|
|
// TODO: The second latest.GroupOrDie("").GroupVersion.Version will be latest.GroupVersion after we
|
2015-09-04 07:06:01 +00:00
|
|
|
// have multiple group support
|
2015-11-16 18:34:28 +00:00
|
|
|
Groups[""] = TestGroup{"", latest.GroupOrDie("").GroupVersion.Version, latest.GroupOrDie("").GroupVersion.String()}
|
2015-09-04 07:06:01 +00:00
|
|
|
}
|
2015-10-09 22:14:03 +00:00
|
|
|
if _, ok := Groups["extensions"]; !ok {
|
2015-11-16 18:34:28 +00:00
|
|
|
Groups["extensions"] = TestGroup{"extensions", latest.GroupOrDie("extensions").GroupVersion.Version, latest.GroupOrDie("extensions").GroupVersion.String()}
|
2015-09-04 07:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Default = Groups[""]
|
2015-10-09 22:30:25 +00:00
|
|
|
Extensions = Groups["extensions"]
|
2015-09-04 07:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Version returns the API version to test against, as set by the KUBE_TEST_API env var.
|
|
|
|
func (g TestGroup) Version() string {
|
|
|
|
return g.VersionUnderTest
|
|
|
|
}
|
|
|
|
|
|
|
|
// GroupAndVersion returns the API version to test against for a group, as set
|
|
|
|
// by the KUBE_TEST_API env var.
|
|
|
|
// Return value is in the form of "group/version".
|
|
|
|
func (g TestGroup) GroupAndVersion() string {
|
|
|
|
return g.GroupVersionUnderTest
|
2014-09-30 00:15:00 +00:00
|
|
|
}
|
|
|
|
|
2015-11-13 21:20:54 +00:00
|
|
|
func (g TestGroup) GroupVersion() *unversioned.GroupVersion {
|
2015-11-18 15:34:16 +00:00
|
|
|
return &unversioned.GroupVersion{Group: g.Group, Version: g.VersionUnderTest}
|
|
|
|
}
|
|
|
|
|
|
|
|
// InternalGroupVersion returns the group,version used to identify the internal
|
|
|
|
// types for this API
|
|
|
|
func (g TestGroup) InternalGroupVersion() unversioned.GroupVersion {
|
|
|
|
return unversioned.GroupVersion{Group: g.Group}
|
2015-11-13 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
2014-10-09 20:13:08 +00:00
|
|
|
// Codec returns the codec for the API version to test against, as set by the
|
2015-09-04 07:06:01 +00:00
|
|
|
// KUBE_TEST_API env var.
|
|
|
|
func (g TestGroup) Codec() runtime.Codec {
|
|
|
|
// TODO: caesarxuchao: Restructure the body once we have a central `latest`.
|
2015-10-30 21:23:53 +00:00
|
|
|
interfaces, err := latest.GroupOrDie(g.Group).InterfacesFor(g.GroupVersionUnderTest)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
2015-09-04 07:06:01 +00:00
|
|
|
}
|
2015-10-30 21:23:53 +00:00
|
|
|
return interfaces.Codec
|
2014-09-30 00:15:00 +00:00
|
|
|
}
|
2014-10-09 20:13:08 +00:00
|
|
|
|
2014-11-13 15:52:13 +00:00
|
|
|
// Converter returns the api.Scheme for the API version to test against, as set by the
|
2015-09-04 07:06:01 +00:00
|
|
|
// KUBE_TEST_API env var.
|
|
|
|
func (g TestGroup) Converter() runtime.ObjectConvertor {
|
|
|
|
// TODO: caesarxuchao: Restructure the body once we have a central `latest`.
|
|
|
|
if g.Group == "" {
|
2015-09-10 19:30:47 +00:00
|
|
|
interfaces, err := latest.GroupOrDie("").InterfacesFor(g.VersionUnderTest)
|
2015-09-04 07:06:01 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return interfaces.ObjectConvertor
|
|
|
|
}
|
2015-10-09 22:14:03 +00:00
|
|
|
if g.Group == "extensions" {
|
|
|
|
interfaces, err := latest.GroupOrDie("extensions").InterfacesFor(g.VersionUnderTest)
|
2015-09-04 07:06:01 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return interfaces.ObjectConvertor
|
2014-11-13 15:52:13 +00:00
|
|
|
}
|
2015-09-04 07:06:01 +00:00
|
|
|
panic(fmt.Errorf("cannot test group %s", g.Group))
|
|
|
|
|
2014-11-13 15:52:13 +00:00
|
|
|
}
|
|
|
|
|
2014-10-27 03:01:17 +00:00
|
|
|
// MetadataAccessor returns the MetadataAccessor for the API version to test against,
|
2015-09-04 07:06:01 +00:00
|
|
|
// as set by the KUBE_TEST_API env var.
|
|
|
|
func (g TestGroup) MetadataAccessor() meta.MetadataAccessor {
|
|
|
|
// TODO: caesarxuchao: Restructure the body once we have a central `latest`.
|
|
|
|
if g.Group == "" {
|
2015-09-10 19:30:47 +00:00
|
|
|
interfaces, err := latest.GroupOrDie("").InterfacesFor(g.VersionUnderTest)
|
2015-09-04 07:06:01 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return interfaces.MetadataAccessor
|
|
|
|
}
|
2015-10-09 22:14:03 +00:00
|
|
|
if g.Group == "extensions" {
|
|
|
|
interfaces, err := latest.GroupOrDie("extensions").InterfacesFor(g.VersionUnderTest)
|
2015-09-04 07:06:01 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return interfaces.MetadataAccessor
|
2014-10-09 20:13:08 +00:00
|
|
|
}
|
2015-09-04 07:06:01 +00:00
|
|
|
panic(fmt.Errorf("cannot test group %s", g.Group))
|
2014-10-09 20:13:08 +00:00
|
|
|
}
|
2014-10-14 22:38:31 +00:00
|
|
|
|
|
|
|
// SelfLink returns a self link that will appear to be for the version Version().
|
|
|
|
// 'resource' should be the resource path, e.g. "pods" for the Pod type. 'name' should be
|
|
|
|
// empty for lists.
|
2015-09-04 07:06:01 +00:00
|
|
|
func (g TestGroup) SelfLink(resource, name string) string {
|
|
|
|
if g.Group == "" {
|
|
|
|
if name == "" {
|
|
|
|
return fmt.Sprintf("/api/%s/%s", g.Version(), resource)
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("/api/%s/%s/%s", g.Version(), resource, name)
|
|
|
|
} else {
|
|
|
|
// TODO: will need a /apis prefix once we have proper multi-group
|
|
|
|
// support
|
|
|
|
if name == "" {
|
2015-09-14 22:30:32 +00:00
|
|
|
return fmt.Sprintf("/apis/%s/%s/%s", g.Group, g.Version(), resource)
|
2015-09-04 07:06:01 +00:00
|
|
|
}
|
2015-09-14 22:30:32 +00:00
|
|
|
return fmt.Sprintf("/apis/%s/%s/%s/%s", g.Group, g.Version(), resource, name)
|
2014-10-14 22:38:31 +00:00
|
|
|
}
|
|
|
|
}
|
2015-03-24 07:06:51 +00:00
|
|
|
|
2015-04-11 07:56:39 +00:00
|
|
|
// Returns the appropriate path for the given prefix (watch, proxy, redirect, etc), resource, namespace and name.
|
2015-03-24 07:06:51 +00:00
|
|
|
// For ex, this is of the form:
|
2015-06-16 00:06:34 +00:00
|
|
|
// /api/v1/watch/namespaces/foo/pods/pod0 for v1.
|
2015-09-04 07:06:01 +00:00
|
|
|
func (g TestGroup) ResourcePathWithPrefix(prefix, resource, namespace, name string) string {
|
|
|
|
var path string
|
|
|
|
if len(g.Group) == 0 {
|
|
|
|
path = "/api/" + g.Version()
|
|
|
|
} else {
|
|
|
|
// TODO: switch back once we have proper multiple group support
|
|
|
|
// path = "/apis/" + g.Group + "/" + Version(group...)
|
2015-09-14 22:30:32 +00:00
|
|
|
path = "/apis/" + g.Group + "/" + g.Version()
|
2015-09-04 07:06:01 +00:00
|
|
|
}
|
|
|
|
|
2015-04-11 07:56:39 +00:00
|
|
|
if prefix != "" {
|
|
|
|
path = path + "/" + prefix
|
|
|
|
}
|
2015-06-16 00:06:34 +00:00
|
|
|
if namespace != "" {
|
|
|
|
path = path + "/namespaces/" + namespace
|
2015-03-24 07:06:51 +00:00
|
|
|
}
|
2015-06-16 00:06:34 +00:00
|
|
|
// Resource names are lower case.
|
|
|
|
resource = strings.ToLower(resource)
|
2015-03-24 07:06:51 +00:00
|
|
|
if resource != "" {
|
|
|
|
path = path + "/" + resource
|
|
|
|
}
|
|
|
|
if name != "" {
|
|
|
|
path = path + "/" + name
|
|
|
|
}
|
|
|
|
return path
|
|
|
|
}
|
|
|
|
|
2015-04-11 07:56:39 +00:00
|
|
|
// Returns the appropriate path for the given resource, namespace and name.
|
2015-07-23 04:52:05 +00:00
|
|
|
// For example, this is of the form:
|
|
|
|
// /api/v1/namespaces/foo/pods/pod0 for v1.
|
2015-09-04 07:06:01 +00:00
|
|
|
func (g TestGroup) ResourcePath(resource, namespace, name string) string {
|
|
|
|
return g.ResourcePathWithPrefix("", resource, namespace, name)
|
2015-04-11 07:56:39 +00:00
|
|
|
}
|
2015-09-11 00:16:04 +00:00
|
|
|
|
|
|
|
func (g TestGroup) RESTMapper() meta.RESTMapper {
|
|
|
|
return latest.GroupOrDie(g.Group).RESTMapper
|
|
|
|
}
|
2015-09-21 23:29:47 +00:00
|
|
|
|
|
|
|
// Get codec based on runtime.Object
|
|
|
|
func GetCodecForObject(obj runtime.Object) (runtime.Codec, error) {
|
|
|
|
_, kind, err := api.Scheme.ObjectVersionAndKind(obj)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("unexpected encoding error: %v", err)
|
|
|
|
}
|
|
|
|
// TODO: caesarxuchao: we should detect which group an object belongs to
|
|
|
|
// by using the version returned by Schem.ObjectVersionAndKind() once we
|
|
|
|
// split the schemes for internal objects.
|
|
|
|
// TODO: caesarxuchao: we should add a map from kind to group in Scheme.
|
|
|
|
for _, group := range Groups {
|
|
|
|
if api.Scheme.Recognizes(group.GroupAndVersion(), kind) {
|
|
|
|
return group.Codec(), nil
|
|
|
|
}
|
|
|
|
}
|
2015-10-27 13:29:18 +00:00
|
|
|
// Codec used for unversioned types
|
|
|
|
if api.Scheme.Recognizes("", kind) {
|
|
|
|
return api.Codec, nil
|
|
|
|
}
|
2015-09-21 23:29:47 +00:00
|
|
|
return nil, fmt.Errorf("unexpected kind: %v", kind)
|
|
|
|
}
|