
Removes etcd2 code and images. Etcd2 is no longer supported as of 1.13, so rollback and management tools for etcd2 should not be shipped any longer. If it is still desired the older versions of the image can be used. Change-Id: I516c40067f0a57c42d2a6ab4612071c5e556a005
195 lines
6.4 KiB
Go
195 lines
6.4 KiB
Go
/*
|
|
Copyright 2018 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 main
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"os/exec"
|
|
"time"
|
|
|
|
"github.com/blang/semver"
|
|
"k8s.io/klog"
|
|
)
|
|
|
|
// EtcdMigrateCfg provides all configuration required to perform etcd data upgrade/downgrade migrations.
|
|
type EtcdMigrateCfg struct {
|
|
binPath string
|
|
name string
|
|
initialCluster string
|
|
port uint64
|
|
peerListenUrls string
|
|
peerAdvertiseUrls string
|
|
etcdDataPrefix string
|
|
ttlKeysDirectory string
|
|
supportedVersions SupportedVersions
|
|
dataDirectory string
|
|
etcdServerArgs string
|
|
}
|
|
|
|
// EtcdMigrateClient defines the etcd client operations required to perform migrations.
|
|
type EtcdMigrateClient interface {
|
|
SetEtcdVersionKeyValue(version *EtcdVersion) error
|
|
Get(version *EtcdVersion, key string) (string, error)
|
|
Put(version *EtcdVersion, key, value string) error
|
|
Backup(version *EtcdVersion, backupDir string) error
|
|
Snapshot(version *EtcdVersion, snapshotFile string) error
|
|
Restore(version *EtcdVersion, snapshotFile string) error
|
|
Migrate(version *EtcdVersion) error
|
|
AttachLease(leaseDuration time.Duration) error
|
|
Close() error
|
|
}
|
|
|
|
// Migrator manages etcd data migrations.
|
|
type Migrator struct {
|
|
cfg *EtcdMigrateCfg // TODO: don't wire this directly in
|
|
dataDirectory *DataDirectory
|
|
client EtcdMigrateClient
|
|
}
|
|
|
|
// MigrateIfNeeded upgrades or downgrades the etcd data directory to the given target version.
|
|
func (m *Migrator) MigrateIfNeeded(target *EtcdVersionPair) error {
|
|
klog.Infof("Starting migration to %s", target)
|
|
err := m.dataDirectory.Initialize(target)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to initialize data directory %s: %v", m.dataDirectory.path, err)
|
|
}
|
|
|
|
var current *EtcdVersionPair
|
|
vfExists, err := m.dataDirectory.versionFile.Exists()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if vfExists {
|
|
current, err = m.dataDirectory.versionFile.Read()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
return fmt.Errorf("existing data directory '%s' is missing version.txt file, unable to migrate", m.dataDirectory.path)
|
|
}
|
|
|
|
for {
|
|
klog.Infof("Converging current version '%s' to target version '%s'", current, target)
|
|
currentNextMinorVersion := &EtcdVersion{Version: semver.Version{Major: current.version.Major, Minor: current.version.Minor + 1}}
|
|
switch {
|
|
case current.version.MajorMinorEquals(target.version) || currentNextMinorVersion.MajorMinorEquals(target.version):
|
|
klog.Infof("current version '%s' equals or is one minor version previous of target version '%s' - migration complete", current, target)
|
|
err = m.dataDirectory.versionFile.Write(target)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to write version.txt to '%s': %v", m.dataDirectory.path, err)
|
|
}
|
|
return nil
|
|
case current.storageVersion == storageEtcd2 && target.storageVersion == storageEtcd3:
|
|
return fmt.Errorf("upgrading from etcd2 storage to etcd3 storage is not supported")
|
|
case current.version.Major == 3 && target.version.Major == 2:
|
|
return fmt.Errorf("downgrading from etcd 3.x to 2.x is not supported")
|
|
case current.version.Major == target.version.Major && current.version.Minor < target.version.Minor:
|
|
stepVersion := m.cfg.supportedVersions.NextVersionPair(current)
|
|
klog.Infof("upgrading etcd from %s to %s", current, stepVersion)
|
|
current, err = m.minorVersionUpgrade(current, stepVersion)
|
|
case current.version.Major == 3 && target.version.Major == 3 && current.version.Minor > target.version.Minor:
|
|
klog.Infof("rolling etcd back from %s to %s", current, target)
|
|
current, err = m.rollbackEtcd3MinorVersion(current, target)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
func (m *Migrator) rollbackEtcd3MinorVersion(current *EtcdVersionPair, target *EtcdVersionPair) (*EtcdVersionPair, error) {
|
|
if target.version.Minor != current.version.Minor-1 {
|
|
return nil, fmt.Errorf("rollback from %s to %s not supported, only rollbacks to the previous minor version are supported", current.version, target.version)
|
|
}
|
|
|
|
klog.Infof("Performing etcd %s -> %s rollback", current.version, target.version)
|
|
err := m.dataDirectory.Backup()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
snapshotFilename := fmt.Sprintf("%s.snapshot.db", m.dataDirectory.path)
|
|
err = os.Remove(snapshotFilename)
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return nil, fmt.Errorf("failed to clean snapshot file before rollback: %v", err)
|
|
}
|
|
|
|
// Start current version of etcd.
|
|
runner := m.newServer()
|
|
klog.Infof("Starting etcd version %s to capture rollback snapshot.", current.version)
|
|
err = runner.Start(current.version)
|
|
if err != nil {
|
|
klog.Fatalf("Unable to automatically downgrade etcd: starting etcd version %s to capture rollback snapshot failed: %v", current.version, err)
|
|
return nil, err
|
|
}
|
|
|
|
klog.Infof("Snapshotting etcd %s to %s", current.version, snapshotFilename)
|
|
err = m.client.Snapshot(current.version, snapshotFilename)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = runner.Stop()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
klog.Info("Backing up data before rolling back")
|
|
backupDir := fmt.Sprintf("%s.bak", m.dataDirectory)
|
|
err = os.RemoveAll(backupDir)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
origInfo, err := os.Stat(m.dataDirectory.path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
err = exec.Command("mv", m.dataDirectory.path, backupDir).Run()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
klog.Infof("Restoring etcd %s from %s", target.version, snapshotFilename)
|
|
err = m.client.Restore(target.version, snapshotFilename)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
err = os.Chmod(m.dataDirectory.path, origInfo.Mode())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return target, nil
|
|
}
|
|
|
|
func (m *Migrator) minorVersionUpgrade(current *EtcdVersionPair, target *EtcdVersionPair) (*EtcdVersionPair, error) {
|
|
runner := m.newServer()
|
|
|
|
// Do the migration step, by just starting etcd in the target version.
|
|
err := runner.Start(target.version)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
err = runner.Stop()
|
|
return target, err
|
|
}
|
|
|
|
func (m *Migrator) newServer() *EtcdMigrateServer {
|
|
return NewEtcdMigrateServer(m.cfg, m.client)
|
|
}
|