2019-01-12 04:58:27 +00:00
/ *
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 .
* /
2019-09-27 21:51:53 +00:00
package polymorphichelpers
2019-01-12 04:58:27 +00:00
import (
"fmt"
appsv1 "k8s.io/api/apps/v1"
extensionsv1beta1 "k8s.io/api/extensions/v1beta1"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema"
2019-09-27 21:51:53 +00:00
deploymentutil "k8s.io/kubectl/pkg/util/deployment"
2019-01-12 04:58:27 +00:00
)
// StatusViewer provides an interface for resources that have rollout status.
type StatusViewer interface {
Status ( obj runtime . Unstructured , revision int64 ) ( string , bool , error )
}
// StatusViewerFor returns a StatusViewer for the resource specified by kind.
func StatusViewerFor ( kind schema . GroupKind ) ( StatusViewer , error ) {
switch kind {
case extensionsv1beta1 . SchemeGroupVersion . WithKind ( "Deployment" ) . GroupKind ( ) ,
appsv1 . SchemeGroupVersion . WithKind ( "Deployment" ) . GroupKind ( ) :
return & DeploymentStatusViewer { } , nil
case extensionsv1beta1 . SchemeGroupVersion . WithKind ( "DaemonSet" ) . GroupKind ( ) ,
appsv1 . SchemeGroupVersion . WithKind ( "DaemonSet" ) . GroupKind ( ) :
return & DaemonSetStatusViewer { } , nil
case appsv1 . SchemeGroupVersion . WithKind ( "StatefulSet" ) . GroupKind ( ) :
return & StatefulSetStatusViewer { } , nil
}
return nil , fmt . Errorf ( "no status viewer has been implemented for %v" , kind )
}
// DeploymentStatusViewer implements the StatusViewer interface.
type DeploymentStatusViewer struct { }
// DaemonSetStatusViewer implements the StatusViewer interface.
type DaemonSetStatusViewer struct { }
// StatefulSetStatusViewer implements the StatusViewer interface.
type StatefulSetStatusViewer struct { }
// Status returns a message describing deployment status, and a bool value indicating if the status is considered done.
func ( s * DeploymentStatusViewer ) Status ( obj runtime . Unstructured , revision int64 ) ( string , bool , error ) {
deployment := & appsv1 . Deployment { }
2020-03-26 21:07:15 +00:00
err := runtime . DefaultUnstructuredConverter . FromUnstructured ( obj . UnstructuredContent ( ) , deployment )
2019-01-12 04:58:27 +00:00
if err != nil {
return "" , false , fmt . Errorf ( "failed to convert %T to %T: %v" , obj , deployment , err )
}
if revision > 0 {
deploymentRev , err := deploymentutil . Revision ( deployment )
if err != nil {
return "" , false , fmt . Errorf ( "cannot get the revision of deployment %q: %v" , deployment . Name , err )
}
if revision != deploymentRev {
return "" , false , fmt . Errorf ( "desired revision (%d) is different from the running revision (%d)" , revision , deploymentRev )
}
}
if deployment . Generation <= deployment . Status . ObservedGeneration {
cond := deploymentutil . GetDeploymentCondition ( deployment . Status , appsv1 . DeploymentProgressing )
if cond != nil && cond . Reason == deploymentutil . TimedOutReason {
return "" , false , fmt . Errorf ( "deployment %q exceeded its progress deadline" , deployment . Name )
}
if deployment . Spec . Replicas != nil && deployment . Status . UpdatedReplicas < * deployment . Spec . Replicas {
return fmt . Sprintf ( "Waiting for deployment %q rollout to finish: %d out of %d new replicas have been updated...\n" , deployment . Name , deployment . Status . UpdatedReplicas , * deployment . Spec . Replicas ) , false , nil
}
if deployment . Status . Replicas > deployment . Status . UpdatedReplicas {
return fmt . Sprintf ( "Waiting for deployment %q rollout to finish: %d old replicas are pending termination...\n" , deployment . Name , deployment . Status . Replicas - deployment . Status . UpdatedReplicas ) , false , nil
}
if deployment . Status . AvailableReplicas < deployment . Status . UpdatedReplicas {
return fmt . Sprintf ( "Waiting for deployment %q rollout to finish: %d of %d updated replicas are available...\n" , deployment . Name , deployment . Status . AvailableReplicas , deployment . Status . UpdatedReplicas ) , false , nil
}
return fmt . Sprintf ( "deployment %q successfully rolled out\n" , deployment . Name ) , true , nil
}
return fmt . Sprintf ( "Waiting for deployment spec update to be observed...\n" ) , false , nil
}
// Status returns a message describing daemon set status, and a bool value indicating if the status is considered done.
func ( s * DaemonSetStatusViewer ) Status ( obj runtime . Unstructured , revision int64 ) ( string , bool , error ) {
//ignoring revision as DaemonSets does not have history yet
daemon := & appsv1 . DaemonSet { }
2020-03-26 21:07:15 +00:00
err := runtime . DefaultUnstructuredConverter . FromUnstructured ( obj . UnstructuredContent ( ) , daemon )
2019-01-12 04:58:27 +00:00
if err != nil {
return "" , false , fmt . Errorf ( "failed to convert %T to %T: %v" , obj , daemon , err )
}
if daemon . Spec . UpdateStrategy . Type != appsv1 . RollingUpdateDaemonSetStrategyType {
return "" , true , fmt . Errorf ( "rollout status is only available for %s strategy type" , appsv1 . RollingUpdateStatefulSetStrategyType )
}
if daemon . Generation <= daemon . Status . ObservedGeneration {
if daemon . Status . UpdatedNumberScheduled < daemon . Status . DesiredNumberScheduled {
return fmt . Sprintf ( "Waiting for daemon set %q rollout to finish: %d out of %d new pods have been updated...\n" , daemon . Name , daemon . Status . UpdatedNumberScheduled , daemon . Status . DesiredNumberScheduled ) , false , nil
}
if daemon . Status . NumberAvailable < daemon . Status . DesiredNumberScheduled {
return fmt . Sprintf ( "Waiting for daemon set %q rollout to finish: %d of %d updated pods are available...\n" , daemon . Name , daemon . Status . NumberAvailable , daemon . Status . DesiredNumberScheduled ) , false , nil
}
return fmt . Sprintf ( "daemon set %q successfully rolled out\n" , daemon . Name ) , true , nil
}
return fmt . Sprintf ( "Waiting for daemon set spec update to be observed...\n" ) , false , nil
}
// Status returns a message describing statefulset status, and a bool value indicating if the status is considered done.
func ( s * StatefulSetStatusViewer ) Status ( obj runtime . Unstructured , revision int64 ) ( string , bool , error ) {
sts := & appsv1 . StatefulSet { }
2020-03-26 21:07:15 +00:00
err := runtime . DefaultUnstructuredConverter . FromUnstructured ( obj . UnstructuredContent ( ) , sts )
2019-01-12 04:58:27 +00:00
if err != nil {
return "" , false , fmt . Errorf ( "failed to convert %T to %T: %v" , obj , sts , err )
}
if sts . Spec . UpdateStrategy . Type != appsv1 . RollingUpdateStatefulSetStrategyType {
return "" , true , fmt . Errorf ( "rollout status is only available for %s strategy type" , appsv1 . RollingUpdateStatefulSetStrategyType )
}
if sts . Status . ObservedGeneration == 0 || sts . Generation > sts . Status . ObservedGeneration {
return "Waiting for statefulset spec update to be observed...\n" , false , nil
}
if sts . Spec . Replicas != nil && sts . Status . ReadyReplicas < * sts . Spec . Replicas {
return fmt . Sprintf ( "Waiting for %d pods to be ready...\n" , * sts . Spec . Replicas - sts . Status . ReadyReplicas ) , false , nil
}
if sts . Spec . UpdateStrategy . Type == appsv1 . RollingUpdateStatefulSetStrategyType && sts . Spec . UpdateStrategy . RollingUpdate != nil {
if sts . Spec . Replicas != nil && sts . Spec . UpdateStrategy . RollingUpdate . Partition != nil {
if sts . Status . UpdatedReplicas < ( * sts . Spec . Replicas - * sts . Spec . UpdateStrategy . RollingUpdate . Partition ) {
return fmt . Sprintf ( "Waiting for partitioned roll out to finish: %d out of %d new pods have been updated...\n" ,
sts . Status . UpdatedReplicas , * sts . Spec . Replicas - * sts . Spec . UpdateStrategy . RollingUpdate . Partition ) , false , nil
}
}
return fmt . Sprintf ( "partitioned roll out complete: %d new pods have been updated...\n" ,
sts . Status . UpdatedReplicas ) , true , nil
}
if sts . Status . UpdateRevision != sts . Status . CurrentRevision {
return fmt . Sprintf ( "waiting for statefulset rolling update to complete %d pods at revision %s...\n" ,
sts . Status . UpdatedReplicas , sts . Status . UpdateRevision ) , false , nil
}
return fmt . Sprintf ( "statefulset rolling update complete %d pods at revision %s...\n" , sts . Status . CurrentReplicas , sts . Status . CurrentRevision ) , true , nil
}