1. Home
  2. Etcd membership reconfiguration in Bright 9.0+

Etcd membership reconfiguration in Bright 9.0+

1. Prerequisites
  • The following article was written with Bright Cluster Manager 9.1 in mind but should work the same for versions 9.0 and 9.2.
  • The feature used by this article is called Learners in etcd, and more can be read about the design here: https://etcd.io/docs/v3.5/learning/design-learner/
  • The minimum required version of etcd should be at least 3.4.4 (earlier versions, such as 3.4.3 contain serious bugs).
2. Background

Crucially, a majority of nodes have to agree to elect a new leader (2/3, 4/6, etc.), and if a majority can’t be reached, the entire cluster will be unavailable. What this means in practice is that etcd will remain available as long as a majority of nodes is online.

Source: here

In etcd terminology a node is also often referred to as a member. In this article we will try to talk about members and use the term node for the underlying machine.

This means a three-member cluster can tolerate one broken member. And a five-member cluster can tolerate two broken members.

To ensure a majority of members in a healthy state at all times, it is recommended to remove or add always one member at a time.

3. Common scenarios

This KB article focuses on the features that etcd itself provides for building new members.

  • Extend a single-member etcd cluster to three members.
  • Replace a member, due to unexpected hardware failure.
  • Migrate one of the members to another node.

Backing up etcd’s data to shared storage using different tools, such as rsync can in some cases be more practical. We dedicate a separate section (section 6.) to this near the end of this KB article.

4. Create snapshots

This article is about adding, removing or replacing members for a running etcd cluster, it might still be worth creating a snapshot of the database anyway. For details refer to this KB article.

In short, a simple parallel approach to this could be:

  • Check each endpoint’s health:
    pdsh -w node00[1-3] "module load etcd; etcdctl -w table endpoint health"
  • Inspect the output, see if all members that are expected to be healthy are.
  • Create a directory on shared storage that is not tied to the node:
    mkdir -p /cm/shared/backup/etcd.
  • Create the snapshots:
    pdsh -w node00[1-3] "module load etcd; etcdctl snapshot save /cm/shared/backup/etcd/etcd-\$(hostname)-\$(date +"%Y%m%d%H%M%S")"
  • Inspect the output.
5. Removing a member
  • If you wish to only add new members, skip this section and go to section 5.
  • If you wish to migrate a member to another node, skip this section first, add the new member (section 5.) and then come back to this section (4.) to remove the old member.
  • If you wish to replace an offline broken member, and you want to bring it back up with new hardware, continue with the following steps in this section.
  • If you wish to replace an online member, consider taking the following steps:
    • Take the etcd member on the node offline by following section 4.1.
    • Backup the etcd directories by following section 6.1.

      Do all the needed changes, for example, a FULL provisioning.
    • Restore the important directories by following section 6.2.
      (Section 6.2. also includes how to bring the etcd member online.)
5.1. Remove node from the etcd Configuration Overlay

For this example we will remove node003. We do not want Bright Cluster Manager to start the etcd service. The etcd::role needs to be unassigned as follows:

[root@headnode ~]# cmsh
[headnode]% configurationoverlay
[headnode->configurationoverlay]% use kube-default-etcd
[headnode->configurationoverlay[kube-default-etcd]]% removefrom nodes node003
[headnode->configurationoverlay*[kube-default-etcd*]]% commit
[headnode->configurationoverlay[kube-default-etcd]]%
Tue Apr  5 12:19:30 2022 [notice] node003: Service etcd was stopped

This is necessary to prevent etcd from starting once the node comes back up with the following error:

member 1c38cdf4114b933d has already been bootstrapped

This is the result of the other etcd members recognizing the host as an existing member with identifier 1c38cdf4114b933d. However, this doesn’t match with the internal database on node003 since it lost its database.

5.2. Remove member from etcd

SSH to one of the etcd members that are up and running and list all the members to get their identifiers.

[root@node001 ~]# module load etcd/kube-default/3.4.13
[root@node001 ~]# etcdctl member list
10cee25dc156ff4a, started, node002, https://10.141.0.2:2380, https://10.141.0.2:2379, false
4a336cbcb0bafdc0, started, node001, https://10.141.0.1:2380, https://10.141.0.1:2379, false
bd786940e5446229, started, node003, https://10.141.0.3:2380, https://10.141.0.3:2379, false

Save the above output if you wish to re-add a member later with the same endpoint. Then proceed with the removal.

[root@node001 ~]# etcdctl member remove bd786940e5446229
Member bd786940e5446229 removed from cluster eef6e88516650e5b

[root@node001 ~]# etcdctl member list
10cee25dc156ff4a, started, node002, https://10.141.0.2:2380, https://10.141.0.2:2379, false
4a336cbcb0bafdc0, started, node001, https://10.141.0.1:2380, https://10.141.0.1:2379, false
6. Add new etcd member

In this example we will add a new node, node003, with IP address 10.141.0.3, as a learner. Whether this node has been removed in the previous section doesn’t matter, it might as well be a completely new node. Let’s say the hard-drive has been completely replaced, and the node is back online.

6.1. Sanity checks on the node

This step should be unnecessary, but here are the preconditions that have to be met nonetheless:

  • Confirm that the service is stopped/disabled with:
    systemctl status etcd
  • Confirm that the /var/lib/etcd directory is clean:
    ls -al /var/lib/etcd/
    (If not, please delete its contents, e.g. rm -rf /var/lib/etcd/member.)
  • Please ignore if the permissions of /var/lib/etcd are erroneously set to 0755, this is a bug at the time of writing that will soon be fixed (correct permissions should be 0700).
6.2. Add the node as an etcd learner

SSH to one of the healthy etcd members and add the node as a learner (note the --learner flag). The terminology comes from the fact that potential new members must first learn the existing cluster’s database. Once they finish, they can be promoted to a non-learning member.

[root@node001 ~]# etcdctl member add --learner --peer-urls=https://10.141.0.3:2380 node003
Member 690ed538336601f4 added to cluster eef6e88516650e5b
 
ETCD_NAME="node003"
ETCD_INITIAL_CLUSTER="node002=https://10.141.0.2:2380,node001=https://10.141.0.1:2380,node003=https://10.141.0.3:2380"
ETCD_INITIAL_ADVERTISE_PEER_URLS="https://10.141.0.3:2380"
ETCD_INITIAL_CLUSTER_STATE="existing"

The above output first prints a confirmation, and the identifier for the new member. Then follows a few environmental variables. These are meant to be set as environment before running the etcd service (etcd binary) as a learner.

In our case we also need Bright Cluster Manager to generate certificates, since we also use secure communication. Therefore it is easier to ignore these values, and use a different approach. We create a configuration overlay with an equivalent flag (--initial-cluster-state=existing).

6.3. Create etcd learners Configuration Overlay

This has to be done once, since it may be helpful to keep this configuration overlay for the future.

[root@headnode ~]# cmsh
[headnode]% configurationoverlay
[headnode->configurationoverlay]% clone kube-default-etcd kube-default-etcd-learners
[headnode->configurationoverlay*[kube-default-etcd-learners*]]% show
Parameter                        Value                                          
-------------------------------- ------------------------------------------------
Name                             kube-default-etcd-learners                      
Revision                                                                        
All head nodes                   no                   
Priority                         500                  
Nodes                            node001,node002      
Categories                                                                      
Roles                            Etcd::Host                                      
Customizations                   <0 in submode>                                  
[headnode->configurationoverlay*[kube-default-etcd-learners*]]% set priority 510
[headnode->configurationoverlay*[kube-default-etcd-learners*]]% set allheadnodes no
[headnode->configurationoverlay*[kube-default-etcd-learners*]]% set nodes
[headnode->configurationoverlay*[kube-default-etcd-learners*]]% set categories
[headnode->configurationoverlay*[kube-default-etcd-learners*]]% append nodes node003
[headnode->configurationoverlay*[kube-default-etcd-learners*]]% show
Parameter                        Value                                          
-------------------------------- ------------------------------------------------
Name                             kube-default-etcd-learners                      
Revision                                                                        
All head nodes                   no                                              
Priority                         510                                            
Nodes                            node003
Categories                                                                      
Roles                            Etcd::Host                                      
Customizations                   <0 in submode>                                  
 
[headnode->configurationoverlay*[kube-default-etcd-learners*]]% roles
[headnode->configurationoverlay*[kube-default-etcd-learners*]->roles*]% use etcd::host                                              
[headnode->configurationoverlay*[kube-default-etcd-learners*]->roles*[Etcd::Host*]]% append options "--initial-cluster-state=existing"
[headnode->configurationoverlay*[kube-default-etcd-learners*]->roles*[Etcd::Host*]]% commit

The reason we set “allheadnodes”, “nodes”, “categories” to “no” and empty values respectively, is that we don’t want this overlay to apply to any node, other than the one we have in mind. The “clone” command copies whatever was in the original configuration overlay.

Note that besides clearing, we did add “node003” explicitly to it, since we wish this particular etcd::role that contains the extra option, to be assigned to it.

If you kept this configuration overlay around, appending the node would be the only needed step here.

Wait a while after the commit, some back-and-forth will happen at this point (certs are created, API servers restarted, and finally etcd will be started). Output inside cmsh might be similar to:

Tue Apr  5 12:37:12 2022 [notice] headnode: New certificate request with ID: 48
Tue Apr  5 12:37:13 2022 [notice] node003: Service etcd was not allowed to restart
Tue Apr  5 12:37:13 2022 [notice] node003: Service etcd was not allowed start
Tue Apr  5 12:37:23 2022 [notice] node001: Service etcd was restarted
Tue Apr  5 12:37:23 2022 [notice] node002: Service etcd was restarted
Tue Apr  5 12:37:25 2022 [notice] headnode: Service kube-apiserver was restarted
Tue Apr  5 12:37:35 2022 [warning] node003: Service etcd died
Tue Apr  5 12:37:35 2022 [notice] node003: Service etcd was not restarted
Tue Apr  5 12:37:35 2022 [notice] headnode: New certificate request with ID: 49
Tue Apr  5 12:37:39 2022 [notice] headnode: New certificate request with ID: 50
Tue Apr  5 12:38:07 2022 [warning] node003: Service etcd died
Tue Apr  5 12:38:15 2022 [notice] node003: Service etcd was restarted
6.4. Confirm and Promote the learner

Confirm this via etcdctl on a working etcd node:

[root@node001 ~]# etcdctl member list
10cee25dc156ff4a, started, node002, https://10.141.0.2:2380, https://10.141.0.2:2379, false
4a336cbcb0bafdc0, started, node001, https://10.141.0.1:2380, https://10.141.0.1:2379, false
690ed538336601f4, started, node003, https://10.141.0.3:2380, https://10.141.0.3:2379, true

The last line in the above output shows node003 to be added, and the very last boolean indicates that it’s a learner due to the value “true”. Now we can promote it:

[root@node001 ~]# etcdctl member promote 690ed538336601f4
Member 690ed538336601f4 promoted in cluster eef6e88516650e5b

[root@node001 ~]# etcdctl member list
10cee25dc156ff4a, started, node002, https://10.141.0.2:2380, https://10.141.0.2:2379, false
4a336cbcb0bafdc0, started, node001, https://10.141.0.1:2380, https://10.141.0.1:2379, false
690ed538336601f4, started, node003, https://10.141.0.3:2380, https://10.141.0.3:2379, false

After promoting, we see the learner flag has changed to “false”.

6.5. Move the node to the original Configuration Overlay

Using cmsh, remove the node from “kube-default-etcd-learners”, and add it to “kube-default-etcd”, then commit both.

[root@headnode ~]# cmsh
[headnode]% configurationoverlay
[headnode->configurationoverlay]% use kube-default-etcd-learners
[headnode->configurationoverlay[kube-default-etcd-learners]]% removefrom nodes node003
[headnode->configurationoverlay*[kube-default-etcd-learners*]]% ..
[headnode->configurationoverlay*]% use kube-default-etcd
[headnode->configurationoverlay*[kube-default-etcd]]% append nodes node003
[headnode->configurationoverlay*[kube-default-etcd*]]% ..
[headnode->configurationoverlay*]% commit
Successfully committed 2 ConfigurationOverlays
[headnode->configurationoverlay]%
Tue Apr  5 12:46:02 2022 [notice] node003: Service etcd was restarted

This results in a restart once more, because the service is no longer started with the --initial-cluster-state=existing.

7. Backup and Restore etcd with rsync

This is useful for backing up etcd members where the members are still online and scheduled to be replaced. Common scenario for this is a change in disk layout.

Do not follow these steps if you wish to keep the node up, use the etcd’s snapshot functionality in that case (see section 4.). This method of backup and restore requires you to stop the etcd service first. Since we don’t want etcd writing to it’s spool directory while we create the backup.

There are two important directories:

  • /var/lib/etcd  (“spool” directory, contains the data)
  • /cm/local/apps/etcd/var/etc/   (contains config and certificates)

Please beware that these are the default paths, and that they can be changed within Bright Cluster Manager. The spool directory is the single most important one. Bright Cluster Manager will automatically re-create the config + certificates in case we do not back it up.

You can find the spool directory configured here:

[root@headnode ~]# cmsh
[headnode]% configurationoverlay 
[headnode->configurationoverlay]% use kube-default-etcd
[headnode->configurationoverlay[kube-default-etcd]]% roles
[headnode->configurationoverlay[kube-default-etcd]->roles]% use etcd::host 
[headnode->configurationoverlay[kube-default-etcd]->roles[Etcd::Host]]% show
Parameter                        Value                                           
-------------------------------- ------------------------------------------------
Name                             Etcd::Host                                      
Revision                                                                         
Type                             EtcdHostRole                                    
Add services                     yes                                             
Member Certificate                                                               
Member Certificate Key                                                           
Provisioning associations        <0 internally used>                             
Etcd Cluster                     kube-default                                    
Member Name                      $hostname                                       
Spool                            /var/lib/etcd                                   
Listen Client URLs               https://0.0.0.0:2379                            
Listen Peer URLs                 https://0.0.0.0:2380                            
Advertise Client URLs            https://$ip:2379                                
Advertise Peer URLs              https://$ip:2380                                
Snapshot Count                   5000                                            
Options                                                                          
Debug                            no                                              

Should the path be different from /var/lib/etcd, please substitute it with the correct path for the rest of this section.

7.1. Backing up
  • Follow section 4.1. to remove the node from the etcd configuration overlay. This ensures the service is stopped.
  • SSH to the node and ensure that etcd has stopped:
    systemctl status etcd
  • Prepare a directory where we can store our backup, e.g. on mounted shared storage.
    mkdir -p /cm/shared/etcd-backups/$(hostname)/{etcd,etc}
  • Rsync the directories to this location.
    rsync -raPv --delete /var/lib/etcd/ /cm/shared/etcd-backups/$(hostname)/etcd/
    rsync -raPv --delete /cm/local/apps/etcd/var/etc/ /cm/shared/etcd-backups/$(hostname)/etc/

Now the etcd data should be saved in such a way that we can restore it later.

7.2. Restoring

Assuming that the node is now back online, we want to restore the backup we made in section 6.2 before we re-assign the etcd role in cmsh.

  • SSH to the node and execute the rsync into the other direction:
    rsync -raPv --delete /cm/shared/etcd-backups/$(hostname)/etcd/ /var/lib/etcd/
    rsync -raPv --delete /cm/shared/etcd-backups/$(hostname)/etc/ /cm/local/apps/etcd/var/etc/
  • Go to cmsh and add the node back into the etcd configuration overlay:
    [headnode->configurationoverlay[kube-default-etcd]]% append nodes node003
    [headnode->configurationoverlay*[kube-default-etcd*]]% commit

Confirm if the member is accepted to the cluster by inspecting its status:

[root@node001 ~]# module load etcd/kube-default/3.4.13 
[root@node001 ~]# ETCDCTL_ENDPOINTS=$(etcdctl member list | awk -F ',' '{print $5}' | sed 's/\s//' | paste -sd ",")
[root@node001 ~]# etcdctl endpoint status
https://10.141.0.2:2379, 10cee25dc156ff4a, 3.4.13, 4.8 MB, false, false, 10, 24644, 24644,
https://10.141.0.1:2379, 4a336cbcb0bafdc0, 3.4.13, 4.9 MB, true, false, 10, 24644, 24644,
https://10.141.0.3:2379, 690ed538336601f4, 3.4.13, 4.9 MB, false, false, 10, 24644, 24644,
Updated on August 3, 2022

Leave a Comment