8e5b17cf13
Signed-off-by: Mrunal Patel <mrunalp@gmail.com>
385 lines
14 KiB
Go
385 lines
14 KiB
Go
/*
|
|
Copyright 2014 The Kubernetes Authors.
|
|
|
|
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.
|
|
*/
|
|
|
|
package genericapiserver
|
|
|
|
import (
|
|
"fmt"
|
|
"mime"
|
|
"net/http"
|
|
"sort"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
systemd "github.com/coreos/go-systemd/daemon"
|
|
"github.com/emicklei/go-restful"
|
|
"github.com/emicklei/go-restful/swagger"
|
|
"github.com/golang/glog"
|
|
|
|
"k8s.io/apimachinery/pkg/apimachinery"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
openapicommon "k8s.io/apimachinery/pkg/openapi"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
utilnet "k8s.io/apimachinery/pkg/util/net"
|
|
"k8s.io/apimachinery/pkg/util/sets"
|
|
"k8s.io/apiserver/pkg/healthz"
|
|
apirequest "k8s.io/apiserver/pkg/request"
|
|
"k8s.io/kubernetes/pkg/admission"
|
|
"k8s.io/kubernetes/pkg/api"
|
|
"k8s.io/kubernetes/pkg/client/restclient"
|
|
genericapi "k8s.io/kubernetes/pkg/genericapiserver/api"
|
|
"k8s.io/kubernetes/pkg/genericapiserver/api/rest"
|
|
genericmux "k8s.io/kubernetes/pkg/genericapiserver/mux"
|
|
"k8s.io/kubernetes/pkg/genericapiserver/routes"
|
|
)
|
|
|
|
// Info about an API group.
|
|
type APIGroupInfo struct {
|
|
GroupMeta apimachinery.GroupMeta
|
|
// Info about the resources in this group. Its a map from version to resource to the storage.
|
|
VersionedResourcesStorageMap map[string]map[string]rest.Storage
|
|
// OptionsExternalVersion controls the APIVersion used for common objects in the
|
|
// schema like api.Status, api.DeleteOptions, and api.ListOptions. Other implementors may
|
|
// define a version "v1beta1" but want to use the Kubernetes "v1" internal objects.
|
|
// If nil, defaults to groupMeta.GroupVersion.
|
|
// TODO: Remove this when https://github.com/kubernetes/kubernetes/issues/19018 is fixed.
|
|
OptionsExternalVersion *schema.GroupVersion
|
|
|
|
// Scheme includes all of the types used by this group and how to convert between them (or
|
|
// to convert objects from outside of this group that are accepted in this API).
|
|
// TODO: replace with interfaces
|
|
Scheme *runtime.Scheme
|
|
// NegotiatedSerializer controls how this group encodes and decodes data
|
|
NegotiatedSerializer runtime.NegotiatedSerializer
|
|
// ParameterCodec performs conversions for query parameters passed to API calls
|
|
ParameterCodec runtime.ParameterCodec
|
|
|
|
// SubresourceGroupVersionKind contains the GroupVersionKind overrides for each subresource that is
|
|
// accessible from this API group version. The GroupVersionKind is that of the external version of
|
|
// the subresource. The key of this map should be the path of the subresource. The keys here should
|
|
// match the keys in the Storage map above for subresources.
|
|
SubresourceGroupVersionKind map[string]schema.GroupVersionKind
|
|
}
|
|
|
|
// GenericAPIServer contains state for a Kubernetes cluster api server.
|
|
type GenericAPIServer struct {
|
|
// discoveryAddresses is used to build cluster IPs for discovery.
|
|
discoveryAddresses DiscoveryAddresses
|
|
|
|
// LoopbackClientConfig is a config for a privileged loopback connection to the API server
|
|
LoopbackClientConfig *restclient.Config
|
|
|
|
// minRequestTimeout is how short the request timeout can be. This is used to build the RESTHandler
|
|
minRequestTimeout time.Duration
|
|
|
|
// legacyAPIGroupPrefixes is used to set up URL parsing for authorization and for validating requests
|
|
// to InstallLegacyAPIGroup
|
|
legacyAPIGroupPrefixes sets.String
|
|
|
|
// admissionControl is used to build the RESTStorage that backs an API Group.
|
|
admissionControl admission.Interface
|
|
|
|
// requestContextMapper provides a way to get the context for a request. It may be nil.
|
|
requestContextMapper apirequest.RequestContextMapper
|
|
|
|
// The registered APIs
|
|
HandlerContainer *genericmux.APIContainer
|
|
|
|
SecureServingInfo *SecureServingInfo
|
|
InsecureServingInfo *ServingInfo
|
|
|
|
// numerical ports, set after listening
|
|
effectiveSecurePort, effectiveInsecurePort int
|
|
|
|
// ExternalAddress is the address (hostname or IP and port) that should be used in
|
|
// external (public internet) URLs for this GenericAPIServer.
|
|
ExternalAddress string
|
|
|
|
// storage contains the RESTful endpoints exposed by this GenericAPIServer
|
|
storage map[string]rest.Storage
|
|
|
|
// Serializer controls how common API objects not in a group/version prefix are serialized for this server.
|
|
// Individual APIGroups may define their own serializers.
|
|
Serializer runtime.NegotiatedSerializer
|
|
|
|
// "Outputs"
|
|
Handler http.Handler
|
|
InsecureHandler http.Handler
|
|
|
|
// Map storing information about all groups to be exposed in discovery response.
|
|
// The map is from name to the group.
|
|
apiGroupsForDiscoveryLock sync.RWMutex
|
|
apiGroupsForDiscovery map[string]metav1.APIGroup
|
|
|
|
// Enable swagger and/or OpenAPI if these configs are non-nil.
|
|
swaggerConfig *swagger.Config
|
|
openAPIConfig *openapicommon.Config
|
|
|
|
// PostStartHooks are each called after the server has started listening, in a separate go func for each
|
|
// with no guarantee of ordering between them. The map key is a name used for error reporting.
|
|
// It may kill the process with a panic if it wishes to by returning an error
|
|
postStartHookLock sync.Mutex
|
|
postStartHooks map[string]postStartHookEntry
|
|
postStartHooksCalled bool
|
|
|
|
// healthz checks
|
|
healthzLock sync.Mutex
|
|
healthzChecks []healthz.HealthzChecker
|
|
healthzCreated bool
|
|
}
|
|
|
|
func init() {
|
|
// Send correct mime type for .svg files.
|
|
// TODO: remove when https://github.com/golang/go/commit/21e47d831bafb59f22b1ea8098f709677ec8ce33
|
|
// makes it into all of our supported go versions (only in v1.7.1 now).
|
|
mime.AddExtensionType(".svg", "image/svg+xml")
|
|
}
|
|
|
|
// RequestContextMapper is exposed so that third party resource storage can be build in a different location.
|
|
// TODO refactor third party resource storage
|
|
func (s *GenericAPIServer) RequestContextMapper() apirequest.RequestContextMapper {
|
|
return s.requestContextMapper
|
|
}
|
|
|
|
// MinRequestTimeout is exposed so that third party resource storage can be build in a different location.
|
|
// TODO refactor third party resource storage
|
|
func (s *GenericAPIServer) MinRequestTimeout() time.Duration {
|
|
return s.minRequestTimeout
|
|
}
|
|
|
|
type preparedGenericAPIServer struct {
|
|
*GenericAPIServer
|
|
}
|
|
|
|
// PrepareRun does post API installation setup steps.
|
|
func (s *GenericAPIServer) PrepareRun() preparedGenericAPIServer {
|
|
if s.swaggerConfig != nil {
|
|
routes.Swagger{Config: s.swaggerConfig}.Install(s.HandlerContainer)
|
|
}
|
|
if s.openAPIConfig != nil {
|
|
routes.OpenAPI{
|
|
Config: s.openAPIConfig,
|
|
}.Install(s.HandlerContainer)
|
|
}
|
|
|
|
s.installHealthz()
|
|
|
|
return preparedGenericAPIServer{s}
|
|
}
|
|
|
|
// Run spawns the http servers (secure and insecure). It only returns if stopCh is closed
|
|
// or one of the ports cannot be listened on initially.
|
|
func (s preparedGenericAPIServer) Run(stopCh <-chan struct{}) {
|
|
if s.SecureServingInfo != nil && s.Handler != nil {
|
|
if err := s.serveSecurely(stopCh); err != nil {
|
|
glog.Fatal(err)
|
|
}
|
|
}
|
|
|
|
if s.InsecureServingInfo != nil && s.InsecureHandler != nil {
|
|
if err := s.serveInsecurely(stopCh); err != nil {
|
|
glog.Fatal(err)
|
|
}
|
|
}
|
|
|
|
s.RunPostStartHooks()
|
|
|
|
// err == systemd.SdNotifyNoSocket when not running on a systemd system
|
|
if err := systemd.SdNotify("READY=1\n"); err != nil && err != systemd.SdNotifyNoSocket {
|
|
glog.Errorf("Unable to send systemd daemon successful start message: %v\n", err)
|
|
}
|
|
|
|
<-stopCh
|
|
}
|
|
|
|
// installAPIResources is a private method for installing the REST storage backing each api groupversionresource
|
|
func (s *GenericAPIServer) installAPIResources(apiPrefix string, apiGroupInfo *APIGroupInfo) error {
|
|
for _, groupVersion := range apiGroupInfo.GroupMeta.GroupVersions {
|
|
apiGroupVersion := s.getAPIGroupVersion(apiGroupInfo, groupVersion, apiPrefix)
|
|
if apiGroupInfo.OptionsExternalVersion != nil {
|
|
apiGroupVersion.OptionsExternalVersion = apiGroupInfo.OptionsExternalVersion
|
|
}
|
|
|
|
if err := apiGroupVersion.InstallREST(s.HandlerContainer.Container); err != nil {
|
|
return fmt.Errorf("Unable to setup API %v: %v", apiGroupInfo, err)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *GenericAPIServer) InstallLegacyAPIGroup(apiPrefix string, apiGroupInfo *APIGroupInfo) error {
|
|
if !s.legacyAPIGroupPrefixes.Has(apiPrefix) {
|
|
return fmt.Errorf("%q is not in the allowed legacy API prefixes: %v", apiPrefix, s.legacyAPIGroupPrefixes.List())
|
|
}
|
|
if err := s.installAPIResources(apiPrefix, apiGroupInfo); err != nil {
|
|
return err
|
|
}
|
|
|
|
// setup discovery
|
|
apiVersions := []string{}
|
|
for _, groupVersion := range apiGroupInfo.GroupMeta.GroupVersions {
|
|
apiVersions = append(apiVersions, groupVersion.Version)
|
|
}
|
|
// Install the version handler.
|
|
// Add a handler at /<apiPrefix> to enumerate the supported api versions.
|
|
genericapi.AddApiWebService(s.Serializer, s.HandlerContainer.Container, apiPrefix, func(req *restful.Request) *metav1.APIVersions {
|
|
clientIP := utilnet.GetClientIP(req.Request)
|
|
|
|
apiVersionsForDiscovery := metav1.APIVersions{
|
|
ServerAddressByClientCIDRs: s.discoveryAddresses.ServerAddressByClientCIDRs(clientIP),
|
|
Versions: apiVersions,
|
|
}
|
|
return &apiVersionsForDiscovery
|
|
})
|
|
return nil
|
|
}
|
|
|
|
// Exposes the given api group in the API.
|
|
func (s *GenericAPIServer) InstallAPIGroup(apiGroupInfo *APIGroupInfo) error {
|
|
// Do not register empty group or empty version. Doing so claims /apis/ for the wrong entity to be returned.
|
|
// Catching these here places the error much closer to its origin
|
|
if len(apiGroupInfo.GroupMeta.GroupVersion.Group) == 0 {
|
|
return fmt.Errorf("cannot register handler with an empty group for %#v", *apiGroupInfo)
|
|
}
|
|
if len(apiGroupInfo.GroupMeta.GroupVersion.Version) == 0 {
|
|
return fmt.Errorf("cannot register handler with an empty version for %#v", *apiGroupInfo)
|
|
}
|
|
|
|
if err := s.installAPIResources(APIGroupPrefix, apiGroupInfo); err != nil {
|
|
return err
|
|
}
|
|
|
|
// setup discovery
|
|
// Install the version handler.
|
|
// Add a handler at /apis/<groupName> to enumerate all versions supported by this group.
|
|
apiVersionsForDiscovery := []metav1.GroupVersionForDiscovery{}
|
|
for _, groupVersion := range apiGroupInfo.GroupMeta.GroupVersions {
|
|
// Check the config to make sure that we elide versions that don't have any resources
|
|
if len(apiGroupInfo.VersionedResourcesStorageMap[groupVersion.Version]) == 0 {
|
|
continue
|
|
}
|
|
apiVersionsForDiscovery = append(apiVersionsForDiscovery, metav1.GroupVersionForDiscovery{
|
|
GroupVersion: groupVersion.String(),
|
|
Version: groupVersion.Version,
|
|
})
|
|
}
|
|
preferedVersionForDiscovery := metav1.GroupVersionForDiscovery{
|
|
GroupVersion: apiGroupInfo.GroupMeta.GroupVersion.String(),
|
|
Version: apiGroupInfo.GroupMeta.GroupVersion.Version,
|
|
}
|
|
apiGroup := metav1.APIGroup{
|
|
Name: apiGroupInfo.GroupMeta.GroupVersion.Group,
|
|
Versions: apiVersionsForDiscovery,
|
|
PreferredVersion: preferedVersionForDiscovery,
|
|
}
|
|
|
|
s.AddAPIGroupForDiscovery(apiGroup)
|
|
s.HandlerContainer.Add(genericapi.NewGroupWebService(s.Serializer, APIGroupPrefix+"/"+apiGroup.Name, apiGroup))
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *GenericAPIServer) AddAPIGroupForDiscovery(apiGroup metav1.APIGroup) {
|
|
s.apiGroupsForDiscoveryLock.Lock()
|
|
defer s.apiGroupsForDiscoveryLock.Unlock()
|
|
|
|
s.apiGroupsForDiscovery[apiGroup.Name] = apiGroup
|
|
}
|
|
|
|
func (s *GenericAPIServer) RemoveAPIGroupForDiscovery(groupName string) {
|
|
s.apiGroupsForDiscoveryLock.Lock()
|
|
defer s.apiGroupsForDiscoveryLock.Unlock()
|
|
|
|
delete(s.apiGroupsForDiscovery, groupName)
|
|
}
|
|
|
|
func (s *GenericAPIServer) getAPIGroupVersion(apiGroupInfo *APIGroupInfo, groupVersion schema.GroupVersion, apiPrefix string) *genericapi.APIGroupVersion {
|
|
storage := make(map[string]rest.Storage)
|
|
for k, v := range apiGroupInfo.VersionedResourcesStorageMap[groupVersion.Version] {
|
|
storage[strings.ToLower(k)] = v
|
|
}
|
|
version := s.newAPIGroupVersion(apiGroupInfo, groupVersion)
|
|
version.Root = apiPrefix
|
|
version.Storage = storage
|
|
return version
|
|
}
|
|
|
|
func (s *GenericAPIServer) newAPIGroupVersion(apiGroupInfo *APIGroupInfo, groupVersion schema.GroupVersion) *genericapi.APIGroupVersion {
|
|
return &genericapi.APIGroupVersion{
|
|
GroupVersion: groupVersion,
|
|
|
|
ParameterCodec: apiGroupInfo.ParameterCodec,
|
|
Serializer: apiGroupInfo.NegotiatedSerializer,
|
|
Creater: apiGroupInfo.Scheme,
|
|
Convertor: apiGroupInfo.Scheme,
|
|
Copier: apiGroupInfo.Scheme,
|
|
Typer: apiGroupInfo.Scheme,
|
|
SubresourceGroupVersionKind: apiGroupInfo.SubresourceGroupVersionKind,
|
|
Linker: apiGroupInfo.GroupMeta.SelfLinker,
|
|
Mapper: apiGroupInfo.GroupMeta.RESTMapper,
|
|
|
|
Admit: s.admissionControl,
|
|
Context: s.RequestContextMapper(),
|
|
MinRequestTimeout: s.minRequestTimeout,
|
|
}
|
|
}
|
|
|
|
// DynamicApisDiscovery returns a webservice serving api group discovery.
|
|
// Note: during the server runtime apiGroupsForDiscovery might change.
|
|
func (s *GenericAPIServer) DynamicApisDiscovery() *restful.WebService {
|
|
return genericapi.NewApisWebService(s.Serializer, APIGroupPrefix, func(req *restful.Request) []metav1.APIGroup {
|
|
s.apiGroupsForDiscoveryLock.RLock()
|
|
defer s.apiGroupsForDiscoveryLock.RUnlock()
|
|
|
|
// sort to have a deterministic order
|
|
sortedGroups := []metav1.APIGroup{}
|
|
groupNames := make([]string, 0, len(s.apiGroupsForDiscovery))
|
|
for groupName := range s.apiGroupsForDiscovery {
|
|
groupNames = append(groupNames, groupName)
|
|
}
|
|
sort.Strings(groupNames)
|
|
for _, groupName := range groupNames {
|
|
sortedGroups = append(sortedGroups, s.apiGroupsForDiscovery[groupName])
|
|
}
|
|
|
|
clientIP := utilnet.GetClientIP(req.Request)
|
|
serverCIDR := s.discoveryAddresses.ServerAddressByClientCIDRs(clientIP)
|
|
groups := make([]metav1.APIGroup, len(sortedGroups))
|
|
for i := range sortedGroups {
|
|
groups[i] = sortedGroups[i]
|
|
groups[i].ServerAddressByClientCIDRs = serverCIDR
|
|
}
|
|
return groups
|
|
})
|
|
}
|
|
|
|
// NewDefaultAPIGroupInfo returns an APIGroupInfo stubbed with "normal" values
|
|
// exposed for easier composition from other packages
|
|
func NewDefaultAPIGroupInfo(group string) APIGroupInfo {
|
|
groupMeta := api.Registry.GroupOrDie(group)
|
|
|
|
return APIGroupInfo{
|
|
GroupMeta: *groupMeta,
|
|
VersionedResourcesStorageMap: map[string]map[string]rest.Storage{},
|
|
OptionsExternalVersion: &api.Registry.GroupOrDie(api.GroupName).GroupVersion,
|
|
Scheme: api.Scheme,
|
|
ParameterCodec: api.ParameterCodec,
|
|
NegotiatedSerializer: api.Codecs,
|
|
}
|
|
}
|