Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[bitnami/etcd] Stop relying on files for state #75906

Open
wants to merge 16 commits into
base: main
Choose a base branch
from

Conversation

pckhoi
Copy link
Contributor

@pckhoi pckhoi commented Dec 25, 2024

Description of the change

The current etcd container and chart have a few major problems:

  1. It relies on files outside of the data directory which could contain conflicting information compared to the data dir and the cluster
  2. Removing the member during pre-stop hook is problematic. I guess this was added to support scaling down the cluster. If so, this logic is leaky. There are 2 cases where this breaks down:
    1. If the pod was killed for reasons other than replicas update then the next time the pod starts, it will not be able to start from the existing data dir which means it must throw away the data dir and start from scratch.
    2. If the cluster is scaled down and the PVC is retained, the next time the cluster is scaled up, the new member will encounter a non-empty data dir which it must discard
  3. If the member was removed, the container chokes up on non-empty data dir in most cases except when recovering from a snapshot
  4. It might attempt to add a new member even when an old member with the same name already exists. This is caused by relying on files for state
  5. It runs etcdctl member update for unclear reasons when the data dir is not empty and there is a member ID
  6. It relies on ETCD_INITIAL_CLUSTER_STATE to know whether the cluster is new which could be inaccurate

This PR add the following changes:

  • Add preupgrade.sh which should be run in a Helm pre-upgrade hook. When the cluster is scaled down, it detects and removes obsolete members with etcdctl member remove.
  • Remove prestop.sh
  • Stop storing/checking member ID from the member_id file. Instead, the remote member ID is read from the cluster with etcdctl member list, and the local member ID is checked for conflict during startup.
  • Stop storing/checking member removal state from member_removal.log. Check with etcdctl member list instead.
  • If the data dir is not empty, check if the member still belongs to the cluster (remote ID and local ID are the same). If there is a conflict, remove the data dir, remove the old member, add a new member, and start the member from scratch.
  • Remove environment variable ETCD_DISABLE_STORE_MEMBER_ID
  • Remove environment variable ETCD_DISABLE_PRESTOP
  • Environment variable ETCD_INITIAL_CLUSTER_STATE becomes read-only

Benefits

  • Not relying on files outside of the data directory means there is only a single source of truths (or only as many as there are live members in the cluster plus the data dir), which makes most operations more reliable
  • Removing obsolete members in Helm pre-upgrade hook means the etcdctl member remove command tends to be executed against a healthy cluster
  • If the pod was killed for reasons other than replica changes, it can rejoin the cluster on its own while keeping all its data intact
  • The container no longer chokes up on a non-empty data dir, even when the old member is removed

Possible drawbacks

  • If during initialization there is a network outage and the current member can't connect to other members, it will think that it must start a new cluster. That said, I don't think there is any good solution in this case except manual recovery.
  • I have not tested this set of changes outside of Helm/K8s

Applicable issues

Additional information

Related changes in the Helm chart: bitnami/charts#31161 and bitnami/charts#31164

- Remove prestop logic (no longer removing member when container stops)
- Remove members not included in ETCD_INITIAL_CLUSTERS during startup
- Stop storing member id on a separate file, member id is checked from
  etcd data dir instead
- Stop reading member removal state off of disk, probe the cluster
  instead
- Remove old member (with the same name) if exist before adding new
  member
- If data dir is not empty, check if the member still belongs to the
  cluster. If not, remove data dir, remove member with the same name,
  and add new member
- Remove env var ETCD_DISABLE_STORE_MEMBER_ID
- Remove env var ETCD_DISABLE_PRESTOP

Signed-off-by: Khoi Pham <[email protected]>
@github-actions github-actions bot added etcd triage Triage is needed labels Dec 25, 2024
@github-actions github-actions bot requested a review from javsalgar December 25, 2024 09:12
@pckhoi
Copy link
Contributor Author

pckhoi commented Dec 26, 2024

I'm planning to open a complementary PR in the charts repo. I will try to add more tests there.

@carrodher carrodher added verify Execute verification workflow for these changes in-progress labels Dec 26, 2024
@github-actions github-actions bot removed the triage Triage is needed label Dec 26, 2024
@github-actions github-actions bot removed the request for review from javsalgar December 26, 2024 08:06
@github-actions github-actions bot requested a review from alvneiayu December 26, 2024 08:06
@carrodher carrodher removed the request for review from alvneiayu December 27, 2024 07:36
Copy link
Contributor

@juan131 juan131 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi @pckhoi

Thanks so much for this amazing contribution! It'd definitely help on making the Bitnami etcd chart more stable.

I think the main concern/challenge with your changes would be providing a solution for users who may scale down the cluster via kubectl scale sts/etcd --replicas X (or via some HorizontalPodAutoscaler that may also scale down the cluster without Helm's control via hooks). Correct me if I'm wrong but this use case won't be covered, right?

@pckhoi
Copy link
Contributor Author

pckhoi commented Jan 3, 2025

@juan131 you're correct that the autoscaling use case isn't covered. People use Etcd for its consistency rather than for handling large, fluctuating traffic so I think autoscaling to handle large traffic is a niche use case.

As for manual scaling, running helm upgrade makes more sense if the cluster is installed via Helm. If people are scaling with kubectl scale then they're probably not using Helm which makes it difficult to operate. So no, deploying/upgrading without Helm is also not supported.

@juan131
Copy link
Contributor

juan131 commented Jan 3, 2025

Thanks for confirming so @pckhoi ! In that case, I'd add a warning at the "Upgrading" section alerting about what these changes imply (I mean, warning users to use exclusively Helm to scale the cluster):

We could even add it in the chart NOTES:

@pckhoi
Copy link
Contributor Author

pckhoi commented Jan 3, 2025

Sure, I will do that.

@pckhoi
Copy link
Contributor Author

pckhoi commented Jan 4, 2025

bitnami/etcd/README.md Outdated Show resolved Hide resolved
bitnami/etcd/README.md Outdated Show resolved Hide resolved
@pckhoi
Copy link
Contributor Author

pckhoi commented Jan 11, 2025

Thanks! I have addressed all the suggestsions.

@juan131
Copy link
Contributor

juan131 commented Jan 13, 2025

@pckhoi I think this PR looks great now! Could you please check my comments in the associated chart PR? Thanks in advance.

Comment on lines 537 to 551
am_i_root && start_command=("run_as_user" "$ETCD_DAEMON_USER" "${start_command[@]}")
[[ -f "$ETCD_CONF_FILE" ]] && start_command+=("--config-file" "$ETCD_CONF_FILE")
$start_command > >(tee -a "$tmp_file") 2>&1 &
pid=$!
debug "Started etcd in background with PID $pid"

while read -r line; do
echo "$line" # Stream the output
if [[ "$line" =~ (established TCP streaming connection with remote peer|the member has been permanently removed from the cluster|ignored streaming request; ID mismatch|\"error\":\"cluster ID mismatch\") ]]; then
kill "$pid"
wait "$pid" 2>/dev/null
debug "Stopped etcd"
break
fi
done < <(tail -f "$tmp_file")
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've been doing more tests today and we'll have to change this block. This always throws etcd start logs to stdout regardless BITNAMI_DEBUG is set or not. Alternative:

Suggested change
am_i_root && start_command=("run_as_user" "$ETCD_DAEMON_USER" "${start_command[@]}")
[[ -f "$ETCD_CONF_FILE" ]] && start_command+=("--config-file" "$ETCD_CONF_FILE")
$start_command > >(tee -a "$tmp_file") 2>&1 &
pid=$!
debug "Started etcd in background with PID $pid"
while read -r line; do
echo "$line" # Stream the output
if [[ "$line" =~ (established TCP streaming connection with remote peer|the member has been permanently removed from the cluster|ignored streaming request; ID mismatch|\"error\":\"cluster ID mismatch\") ]]; then
kill "$pid"
wait "$pid" 2>/dev/null
debug "Stopped etcd"
break
fi
done < <(tail -f "$tmp_file")
am_i_root && start_command=("run_as_user" "$ETCD_DAEMON_USER" "${start_command[@]}")
[[ -f "$ETCD_CONF_FILE" ]] && start_command+=("--config-file" "$ETCD_CONF_FILE")
"${start_command[@]}" > "$tmp_file" 2>&1 &
while read -r line; do
debug_execute echo "$line"
if [[ "$line" =~ (established TCP streaming connection with remote peer|the member has been permanently removed from the cluster|ignored streaming request; ID mismatch|\"error\":\"cluster ID mismatch\") ]]; then
etcd_stop
debug "Stopped etcd"
break
fi
done < <(tail -f "$tmp_file")

By the way, it's not necessary to save the PID, you can use etcd_stop

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated.


expected="$(echo $ETCD_INITIAL_CLUSTER | tr -s ',' '\n' | awk -F= '{print $1}')"
info "Expected cluster members are: $(echo "$expected" | tr -s '\n' ',' | sed 's/,$//g')"
read -r -a obsolete_members <<<"$(comm -23 <(echo "$current" | awk -F: '{print $1}' | sort) <(echo "$expected" | sort))"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

small fix:

Suggested change
read -r -a obsolete_members <<<"$(comm -23 <(echo "$current" | awk -F: '{print $1}' | sort) <(echo "$expected" | sort))"
read -r -a obsolete_members <<<"$(comm -23 <(echo "$current" | awk -F: '{print $1}' | sort) <(echo "$expected" | sort) | tr -s '\n' ' ')"

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This change seems to break the script for me.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actually, something else is breaking. Let me investigate more.

Copy link
Contributor

@juan131 juan131 Jan 16, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What problems are you experiencing? It works for me

Without this change only one member is removed given the obsolete_members wasn't an array but a string with the 1st obsolete member to remove, so when I did a test scaling down from 5 to 3 replicas it only removed one.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry, your suggestion works. I just had to fix the loop as well. Everything should work now.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
etcd in-progress verify Execute verification workflow for these changes
Projects
None yet
Development

Successfully merging this pull request may close these issues.

[bitnami/etcd] etcd pods are unable to join existing cluster on node drain
6 participants