package dao import ( "context" "errors" "fmt" "github.com/derailed/k9s/internal/client" appsv1 "k8s.io/api/apps/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" "k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/types" "k8s.io/kubectl/pkg/polymorphichelpers" ) var ( _ Accessor = (*StatefulSet)(nil) _ Nuker = (*StatefulSet)(nil) _ Loggable = (*StatefulSet)(nil) _ Restartable = (*StatefulSet)(nil) _ Scalable = (*StatefulSet)(nil) _ Controller = (*StatefulSet)(nil) ) // StatefulSet represents a K8s sts. type StatefulSet struct { Resource } // IsHappy check for happy sts. func (s *StatefulSet) IsHappy(sts appsv1.StatefulSet) bool { return sts.Status.Replicas == sts.Status.ReadyReplicas } // Scale a StatefulSet. func (s *StatefulSet) Scale(path string, replicas int32) error { ns, n := client.Namespaced(path) auth, err := s.Client().CanI(ns, "apps/v1/statefulsets:scale", []string{client.GetVerb, client.UpdateVerb}) if err != nil { return err } if !auth { return fmt.Errorf("user is not authorized to scale statefulsets") } scale, err := s.Client().DialOrDie().AppsV1().StatefulSets(ns).GetScale(n, metav1.GetOptions{}) if err != nil { return err } scale.Spec.Replicas = replicas _, err = s.Client().DialOrDie().AppsV1().StatefulSets(ns).UpdateScale(n, scale) return err } // Restart a StatefulSet rollout. func (s *StatefulSet) Restart(path string) error { sts, err := s.getStatefulSet(path) if err != nil { return err } ns, _ := client.Namespaced(path) auth, err := s.Client().CanI(ns, "apps/v1/statefulsets", []string{client.PatchVerb}) if err != nil { return err } if !auth { return fmt.Errorf("user is not authorized to update statefulsets") } update, err := polymorphichelpers.ObjectRestarterFn(sts) if err != nil { return err } _, err = s.Client().DialOrDie().AppsV1().StatefulSets(sts.Namespace).Patch(sts.Name, types.StrategicMergePatchType, update) return err } // TailLogs tail logs for all pods represented by this StatefulSet. func (s *StatefulSet) TailLogs(ctx context.Context, c chan<- []byte, opts LogOptions) error { sts, err := s.getStatefulSet(opts.Path) if err != nil { return errors.New("expecting StatefulSet resource") } if sts.Spec.Selector == nil || len(sts.Spec.Selector.MatchLabels) == 0 { return fmt.Errorf("No valid selector found on StatefulSet %s", opts.Path) } return podLogs(ctx, c, sts.Spec.Selector.MatchLabels, opts) } // Pod returns a pod victim by name. func (s *StatefulSet) Pod(fqn string) (string, error) { sts, err := s.getStatefulSet(fqn) if err != nil { return "", err } return podFromSelector(s.Factory, sts.Namespace, sts.Spec.Selector.MatchLabels) } func (s *StatefulSet) getStatefulSet(fqn string) (*appsv1.StatefulSet, error) { o, err := s.Factory.Get(s.gvr.String(), fqn, false, labels.Everything()) if err != nil { return nil, err } var sts appsv1.StatefulSet err = runtime.DefaultUnstructuredConverter.FromUnstructured(o.(*unstructured.Unstructured).Object, &sts) if err != nil { return nil, errors.New("expecting Service resource") } return &sts, nil }