Skip to content

Commit

Permalink
Upgrade to Advanced StatefulSet automatically
Browse files Browse the repository at this point in the history
Signed-off-by: Yecheng Fu <fuyecheng@pingcap.com>
  • Loading branch information
cofyc committed Jan 17, 2020
1 parent 97d07a1 commit 0271bb1
Show file tree
Hide file tree
Showing 7 changed files with 653 additions and 4 deletions.
10 changes: 8 additions & 2 deletions charts/tidb-operator/templates/controller-manager-rbac.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,10 @@ rules:
resources: ["pods"]
verbs: ["get", "list", "watch","update", "delete"]
- apiGroups: ["apps"]
resources: ["statefulsets","deployments"]
resources: ["statefulsets","deployments", "controllerrevisions"]
verbs: ["*"]
- apiGroups: ["apps.pingcap.com"]
resources: ["statefulsets", "statefulsets/status"]
verbs: ["*"]
- apiGroups: ["pingcap.com"]
resources: ["*"]
Expand Down Expand Up @@ -140,7 +143,10 @@ rules:
resources: ["pods"]
verbs: ["get", "list", "watch","update", "delete"]
- apiGroups: ["apps"]
resources: ["statefulsets","deployments"]
resources: ["statefulsets","deployments", "controllerrevisions"]
verbs: ["*"]
- apiGroups: ["apps.pingcap.com"]
resources: ["statefulsets", "statefulsets/status"]
verbs: ["*"]
- apiGroups: ["pingcap.com"]
resources: ["*"]
Expand Down
17 changes: 16 additions & 1 deletion cmd/controller-manager/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ import (
"github.com/pingcap/tidb-operator/pkg/controller/tidbmonitor"
"github.com/pingcap/tidb-operator/pkg/features"
"github.com/pingcap/tidb-operator/pkg/scheme"
"github.com/pingcap/tidb-operator/pkg/upgrader"
"github.com/pingcap/tidb-operator/pkg/version"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/util/wait"
Expand Down Expand Up @@ -127,6 +128,14 @@ func main() {
glog.Fatalf("failed to get the generic kube-apiserver client: %v", err)
}

// note that kubeCli here must not be the hijacked one
var operatorUpgrader upgrader.Interface
if controller.ClusterScoped {
operatorUpgrader = upgrader.NewUpgrader(kubeCli, cli, asCli, metav1.NamespaceAll)
} else {
operatorUpgrader = upgrader.NewUpgrader(kubeCli, cli, asCli, ns)
}

if features.DefaultFeatureGate.Enabled(features.AdvancedStatefulSet) {
// If AdvancedStatefulSet is enabled, we hijack the Kubernetes client to use
// AdvancedStatefulSet.
Expand Down Expand Up @@ -166,6 +175,12 @@ func main() {
defer cancel()

onStarted := func(ctx context.Context) {
// Upgrade before running any controller logic. If it fails, we wait
// for process supervisor to restart it again.
if err := operatorUpgrader.Upgrade(); err != nil {
glog.Fatalf("failed to upgrade: %v", err)
}

tcController := tidbcluster.NewController(kubeCli, cli, genericCli, informerFactory, kubeInformerFactory, autoFailover, pdFailoverPeriod, tikvFailoverPeriod, tidbFailoverPeriod)
backupController := backup.NewController(kubeCli, cli, informerFactory, kubeInformerFactory)
restoreController := restore.NewController(kubeCli, cli, informerFactory, kubeInformerFactory)
Expand Down Expand Up @@ -201,7 +216,7 @@ func main() {
glog.Fatalf("leader election lost")
}

// leader election for multiple tidb-cloud-manager
// leader election for multiple tidb-controller-manager instances
go wait.Forever(func() {
leaderelection.RunOrDie(controllerCtx, leaderelection.LeaderElectionConfig{
Lock: &rl,
Expand Down
117 changes: 117 additions & 0 deletions pkg/upgrader/upgrader.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,117 @@
// Copyright 2020 PingCAP, Inc.
//
// 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,
// See the License for the specific language governing permissions and
// limitations under the License.

package upgrader

import (
"fmt"

"github.com/pingcap/advanced-statefulset/pkg/apis/apps/v1/helper"
asclientset "github.com/pingcap/advanced-statefulset/pkg/client/clientset/versioned"
"github.com/pingcap/tidb-operator/pkg/apis/pingcap/v1alpha1"
"github.com/pingcap/tidb-operator/pkg/client/clientset/versioned"
"github.com/pingcap/tidb-operator/pkg/features"
"github.com/pingcap/tidb-operator/pkg/label"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/client-go/kubernetes"
"k8s.io/klog"
)

// Interface represents the interface of upgrader.
type Interface interface {
// Upgrade runs in the boot process and performs the function of upgrade
// for the current operator.
//
// Implemention of this interface must be idempotent. If it fails, the
// program can be restarted to retry again without affecting current
// Kubernetes cluster and TiDB clusters.
//
// Note that it's possible that it cann't finish its job when you're trying
// to upgrade a newer operator to an older one or enable/disable some
// irreversible features.
Upgrade() error
}

type upgrader struct {
kubeCli kubernetes.Interface
cli versioned.Interface
asCli asclientset.Interface
ns string
}

var _ Interface = &upgrader{}

func (u *upgrader) Upgrade() error {
if features.DefaultFeatureGate.Enabled(features.AdvancedStatefulSet) {
klog.Infof("Upgrader: migrating Kubernetes StatefulSets to Advanced StatefulSets")
tcList, err := u.cli.PingcapV1alpha1().TidbClusters(u.ns).List(metav1.ListOptions{})
if err != nil {
return err
}
for _, tc := range tcList.Items {
// Existing delete slots annotations must be removed first. This is
// a safety check to ensure no pods are affected in upgrading
// process.
if anns := deleteSlotAnns(&tc); len(anns) > 0 {
return fmt.Errorf("Upgrader: TidbCluster %s/%s has delete slot annotations %v, please remove them before enabling AdvancedStatefulSet feature", tc.Namespace, tc.Name, anns)
}
}
stsList, err := u.kubeCli.AppsV1().StatefulSets(u.ns).List(metav1.ListOptions{})
if err != nil {
return err
}
if len(stsList.Items) <= 0 {
klog.Infof("Upgrader: found 0 Kubernetes StatefulSets, nothing need to do")
return nil
}
klog.Infof("Upgrader: found %d Kubernetes StatefulSets, trying to migrate one by one", len(stsList.Items))
for _, sts := range stsList.Items {
_, err := helper.Upgrade(u.kubeCli, u.asCli, &sts)
if err != nil {
return err
}
klog.Infof("Upgrader: successfully migrated Kubernetes StatefulSet %s/%s", sts.Namespace, sts.Name)
}
} else {
stsList, err := u.asCli.AppsV1().StatefulSets(u.ns).List(metav1.ListOptions{})
if err != nil {
return err
}
if len(stsList.Items) <= 0 {
klog.Infof("Upgrader: found %d Advanced StatefulSets, nothing need to do", len(stsList.Items))
return nil
}
// The upgrader cannot migrate Advanced StatefulSets to Kubernetes
// StatefulSets automatically right now.
// TODO try our best to allow users to revert AdvancedStatefulSet feature automaticaly
return fmt.Errorf("Upgrader: found %d Advanced StatefulSets in the cluster, the opertor cann't run with AdvancedStatefulSet feature disabled", len(stsList.Items))
}
return nil
}

func deleteSlotAnns(tc *v1alpha1.TidbCluster) map[string]string {
anns := make(map[string]string)
if tc == nil || tc.Annotations == nil {
return anns
}
for _, key := range []string{label.AnnPDDeleteSlots, label.AnnTiDBDeleteSlots, label.AnnTiKVDeleteSlots} {
if v, ok := tc.Annotations[key]; ok {
anns[key] = v
}
}
return anns
}

func NewUpgrader(kubeCli kubernetes.Interface, cli versioned.Interface, asCli asclientset.Interface, ns string) Interface {
return &upgrader{kubeCli, cli, asCli, ns}
}
Loading

0 comments on commit 0271bb1

Please sign in to comment.