Merge pull request #1425 from zgfh/feature-kubectl-karmada-init
Wait for APIService status when installing karmada with `karmadactl init`
This commit is contained in:
commit
a3426c0504
|
@ -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"
|
||||
"regexp"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
corev1 "k8s.io/api/core/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 {
|
||||
return err
|
||||
}
|
||||
|
||||
if err := waitAPIServiceReady(apiRegistrationClient, aaAPIServiceObjName, 120*time.Second); err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
|
|
@ -284,6 +284,19 @@ func (i *CommandInitOption) initKarmadaAPIServer() error {
|
|||
if err := WaitPodReady(i.KubeClientSet, i.Namespace, utils.MapToString(apiServerLabels), 120); err != nil {
|
||||
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
|
||||
}
|
||||
|
||||
|
@ -337,18 +350,6 @@ func (i *CommandInitOption) initKarmadaComponent() error {
|
|||
if err := WaitPodReady(i.KubeClientSet, i.Namespace, utils.MapToString(webhookLabels), waitPodReadyTimeout); err != nil {
|
||||
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
|
||||
}
|
||||
|
||||
|
|
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
|
||||
k8s.io/kube-aggregator/pkg/apis/apiregistration
|
||||
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/client/clientset_generated/clientset
|
||||
k8s.io/kube-aggregator/pkg/client/clientset_generated/clientset/scheme
|
||||
|
|
Loading…
Reference in New Issue