feature: Add check for cluster apiservice when install karmada with kubectl-karmada
Signed-off-by: Alan Zhu <zg.zhu@daocloud.io>
This commit is contained in:
parent
bc0a3d58d3
commit
03baf1285e
|
@ -0,0 +1,31 @@
|
||||||
|
package karmada
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
|
"k8s.io/apimachinery/pkg/util/wait"
|
||||||
|
"k8s.io/klog/v2"
|
||||||
|
apiregistrationv1 "k8s.io/kube-aggregator/pkg/apis/apiregistration/v1"
|
||||||
|
apiregistrationv1helper "k8s.io/kube-aggregator/pkg/apis/apiregistration/v1/helper"
|
||||||
|
aggregator "k8s.io/kube-aggregator/pkg/client/clientset_generated/clientset"
|
||||||
|
)
|
||||||
|
|
||||||
|
func waitAPIServiceReady(c *aggregator.Clientset, name string, timeout time.Duration) error {
|
||||||
|
if err := wait.PollImmediate(time.Second, timeout, func() (done bool, err error) {
|
||||||
|
apiService, e := c.ApiregistrationV1().APIServices().Get(context.TODO(), name, metav1.GetOptions{})
|
||||||
|
if e != nil {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
if apiregistrationv1helper.IsAPIServiceConditionTrue(apiService, apiregistrationv1.Available) {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
klog.Infof("Waiting for APIService(%s) condition(%s), will try", name, apiregistrationv1.Available)
|
||||||
|
return false, nil
|
||||||
|
}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -9,6 +9,7 @@ import (
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
corev1 "k8s.io/api/core/v1"
|
corev1 "k8s.io/api/core/v1"
|
||||||
apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
|
apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
|
||||||
|
@ -231,6 +232,8 @@ func initAPIService(clientSet *kubernetes.Clientset, restConfig *rest.Config) er
|
||||||
if _, err := apiRegistrationClient.ApiregistrationV1().APIServices().Create(context.TODO(), aaAPIService, metav1.CreateOptions{}); err != nil {
|
if _, err := apiRegistrationClient.ApiregistrationV1().APIServices().Create(context.TODO(), aaAPIService, metav1.CreateOptions{}); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
if err := waitAPIServiceReady(apiRegistrationClient, aaAPIServiceObjName, 120*time.Second); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -284,6 +284,19 @@ func (i *CommandInitOption) initKarmadaAPIServer() error {
|
||||||
if err := WaitPodReady(i.KubeClientSet, i.Namespace, utils.MapToString(apiServerLabels), 120); err != nil {
|
if err := WaitPodReady(i.KubeClientSet, i.Namespace, utils.MapToString(apiServerLabels), 120); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create karmada-aggregated-apiserver
|
||||||
|
// https://github.com/karmada-io/karmada/blob/master/artifacts/deploy/karmada-aggregated-apiserver.yaml
|
||||||
|
klog.Info("create karmada aggregated apiserver Deployment")
|
||||||
|
if err := i.CreateService(i.karmadaAggregatedAPIServerService()); err != nil {
|
||||||
|
klog.Exitln(err)
|
||||||
|
}
|
||||||
|
if _, err := i.KubeClientSet.AppsV1().Deployments(i.Namespace).Create(context.TODO(), i.makeKarmadaAggregatedAPIServerDeployment(), metav1.CreateOptions{}); err != nil {
|
||||||
|
klog.Warning(err)
|
||||||
|
}
|
||||||
|
if err := WaitPodReady(i.KubeClientSet, i.Namespace, utils.MapToString(aggregatedAPIServerLabels), 30); err != nil {
|
||||||
|
klog.Warning(err)
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -337,18 +350,6 @@ func (i *CommandInitOption) initKarmadaComponent() error {
|
||||||
if err := WaitPodReady(i.KubeClientSet, i.Namespace, utils.MapToString(webhookLabels), waitPodReadyTimeout); err != nil {
|
if err := WaitPodReady(i.KubeClientSet, i.Namespace, utils.MapToString(webhookLabels), waitPodReadyTimeout); err != nil {
|
||||||
klog.Warning(err)
|
klog.Warning(err)
|
||||||
}
|
}
|
||||||
// Create karmada-aggregated-apiserver
|
|
||||||
// https://github.com/karmada-io/karmada/blob/master/artifacts/deploy/karmada-aggregated-apiserver.yaml
|
|
||||||
klog.Info("create karmada aggregated apiserver Deployment")
|
|
||||||
if err := i.CreateService(i.karmadaAggregatedAPIServerService()); err != nil {
|
|
||||||
klog.Exitln(err)
|
|
||||||
}
|
|
||||||
if _, err := deploymentClient.Create(context.TODO(), i.makeKarmadaAggregatedAPIServerDeployment(), metav1.CreateOptions{}); err != nil {
|
|
||||||
klog.Warning(err)
|
|
||||||
}
|
|
||||||
if err := WaitPodReady(i.KubeClientSet, i.Namespace, utils.MapToString(aggregatedAPIServerLabels), waitPodReadyTimeout); err != nil {
|
|
||||||
klog.Warning(err)
|
|
||||||
}
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
138
vendor/k8s.io/kube-aggregator/pkg/apis/apiregistration/v1/helper/helpers.go
generated
vendored
Normal file
138
vendor/k8s.io/kube-aggregator/pkg/apis/apiregistration/v1/helper/helpers.go
generated
vendored
Normal file
|
@ -0,0 +1,138 @@
|
||||||
|
/*
|
||||||
|
Copyright 2016 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 helper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
||||||
|
"k8s.io/apimachinery/pkg/version"
|
||||||
|
"k8s.io/kube-aggregator/pkg/apis/apiregistration/v1"
|
||||||
|
)
|
||||||
|
|
||||||
|
// SortedByGroupAndVersion sorts APIServices into their different groups, and then sorts them based on their versions.
|
||||||
|
// For example, the first element of the first array contains the APIService with the highest version number, in the
|
||||||
|
// group with the highest priority; while the last element of the last array contains the APIService with the lowest
|
||||||
|
// version number, in the group with the lowest priority.
|
||||||
|
func SortedByGroupAndVersion(servers []*v1.APIService) [][]*v1.APIService {
|
||||||
|
serversByGroupPriorityMinimum := ByGroupPriorityMinimum(servers)
|
||||||
|
sort.Sort(serversByGroupPriorityMinimum)
|
||||||
|
|
||||||
|
ret := [][]*v1.APIService{}
|
||||||
|
for _, curr := range serversByGroupPriorityMinimum {
|
||||||
|
// check to see if we already have an entry for this group
|
||||||
|
existingIndex := -1
|
||||||
|
for j, groupInReturn := range ret {
|
||||||
|
if groupInReturn[0].Spec.Group == curr.Spec.Group {
|
||||||
|
existingIndex = j
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if existingIndex >= 0 {
|
||||||
|
ret[existingIndex] = append(ret[existingIndex], curr)
|
||||||
|
sort.Sort(ByVersionPriority(ret[existingIndex]))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = append(ret, []*v1.APIService{curr})
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByGroupPriorityMinimum sorts with the highest group number first, then by name.
|
||||||
|
// This is not a simple reverse, because we want the name sorting to be alpha, not
|
||||||
|
// reverse alpha.
|
||||||
|
type ByGroupPriorityMinimum []*v1.APIService
|
||||||
|
|
||||||
|
func (s ByGroupPriorityMinimum) Len() int { return len(s) }
|
||||||
|
func (s ByGroupPriorityMinimum) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
|
||||||
|
func (s ByGroupPriorityMinimum) Less(i, j int) bool {
|
||||||
|
if s[i].Spec.GroupPriorityMinimum != s[j].Spec.GroupPriorityMinimum {
|
||||||
|
return s[i].Spec.GroupPriorityMinimum > s[j].Spec.GroupPriorityMinimum
|
||||||
|
}
|
||||||
|
return s[i].Name < s[j].Name
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByVersionPriority sorts with the highest version number first, then by name.
|
||||||
|
// This is not a simple reverse, because we want the name sorting to be alpha, not
|
||||||
|
// reverse alpha.
|
||||||
|
type ByVersionPriority []*v1.APIService
|
||||||
|
|
||||||
|
func (s ByVersionPriority) Len() int { return len(s) }
|
||||||
|
func (s ByVersionPriority) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
|
||||||
|
func (s ByVersionPriority) Less(i, j int) bool {
|
||||||
|
if s[i].Spec.VersionPriority != s[j].Spec.VersionPriority {
|
||||||
|
return s[i].Spec.VersionPriority > s[j].Spec.VersionPriority
|
||||||
|
}
|
||||||
|
return version.CompareKubeAwareVersionStrings(s[i].Spec.Version, s[j].Spec.Version) > 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// APIServiceNameToGroupVersion returns the GroupVersion for a given apiServiceName. The name
|
||||||
|
// must be valid, but any object you get back from an informer will be valid.
|
||||||
|
func APIServiceNameToGroupVersion(apiServiceName string) schema.GroupVersion {
|
||||||
|
tokens := strings.SplitN(apiServiceName, ".", 2)
|
||||||
|
return schema.GroupVersion{Group: tokens[1], Version: tokens[0]}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewLocalAvailableAPIServiceCondition returns a condition for an available local APIService.
|
||||||
|
func NewLocalAvailableAPIServiceCondition() v1.APIServiceCondition {
|
||||||
|
return v1.APIServiceCondition{
|
||||||
|
Type: v1.Available,
|
||||||
|
Status: v1.ConditionTrue,
|
||||||
|
LastTransitionTime: metav1.Now(),
|
||||||
|
Reason: "Local",
|
||||||
|
Message: "Local APIServices are always available",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetAPIServiceCondition sets the status condition. It either overwrites the existing one or
|
||||||
|
// creates a new one
|
||||||
|
func SetAPIServiceCondition(apiService *v1.APIService, newCondition v1.APIServiceCondition) {
|
||||||
|
existingCondition := GetAPIServiceConditionByType(apiService, newCondition.Type)
|
||||||
|
if existingCondition == nil {
|
||||||
|
apiService.Status.Conditions = append(apiService.Status.Conditions, newCondition)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if existingCondition.Status != newCondition.Status {
|
||||||
|
existingCondition.Status = newCondition.Status
|
||||||
|
existingCondition.LastTransitionTime = newCondition.LastTransitionTime
|
||||||
|
}
|
||||||
|
|
||||||
|
existingCondition.Reason = newCondition.Reason
|
||||||
|
existingCondition.Message = newCondition.Message
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsAPIServiceConditionTrue indicates if the condition is present and strictly true
|
||||||
|
func IsAPIServiceConditionTrue(apiService *v1.APIService, conditionType v1.APIServiceConditionType) bool {
|
||||||
|
condition := GetAPIServiceConditionByType(apiService, conditionType)
|
||||||
|
return condition != nil && condition.Status == v1.ConditionTrue
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAPIServiceConditionByType gets an *APIServiceCondition by APIServiceConditionType if present
|
||||||
|
func GetAPIServiceConditionByType(apiService *v1.APIService, conditionType v1.APIServiceConditionType) *v1.APIServiceCondition {
|
||||||
|
for i := range apiService.Status.Conditions {
|
||||||
|
if apiService.Status.Conditions[i].Type == conditionType {
|
||||||
|
return &apiService.Status.Conditions[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -1354,6 +1354,7 @@ k8s.io/klog/v2
|
||||||
## explicit; go 1.16
|
## explicit; go 1.16
|
||||||
k8s.io/kube-aggregator/pkg/apis/apiregistration
|
k8s.io/kube-aggregator/pkg/apis/apiregistration
|
||||||
k8s.io/kube-aggregator/pkg/apis/apiregistration/v1
|
k8s.io/kube-aggregator/pkg/apis/apiregistration/v1
|
||||||
|
k8s.io/kube-aggregator/pkg/apis/apiregistration/v1/helper
|
||||||
k8s.io/kube-aggregator/pkg/apis/apiregistration/v1beta1
|
k8s.io/kube-aggregator/pkg/apis/apiregistration/v1beta1
|
||||||
k8s.io/kube-aggregator/pkg/client/clientset_generated/clientset
|
k8s.io/kube-aggregator/pkg/client/clientset_generated/clientset
|
||||||
k8s.io/kube-aggregator/pkg/client/clientset_generated/clientset/scheme
|
k8s.io/kube-aggregator/pkg/client/clientset_generated/clientset/scheme
|
||||||
|
|
Loading…
Reference in New Issue