Charmed Zookeeper
- Canonical
- Databases
Channel | Revision | Published | Runs on |
---|---|---|---|
latest/stable | 53 | 29 Nov 2021 | |
latest/candidate | 56 | 29 Nov 2021 | |
latest/beta | 56 | 29 Nov 2021 | |
latest/edge | 98 | 20 Apr 2023 | |
latest/edge | 85 | 21 Oct 2022 | |
latest/edge | 65 | 09 Feb 2022 | |
latest/edge | 52 | 29 Nov 2021 | |
3/stable | 149 | 23 Oct 2024 | |
3/candidate | 149 | 21 Oct 2024 | |
3/beta | 149 | 21 Oct 2024 | |
3/edge | 148 | 16 Oct 2024 |
juju deploy zookeeper --channel 3/stable
Deploy universal operators easily with Juju, the Universal Operator Lifecycle Manager.
Platform:
22.04
20.04
18.04
16.04
charms.zookeeper.v0.cluster
-
- Last updated 08 Jul 2022
- Revision Library version 0.1
#!/usr/bin/env python3
# Copyright 2022 Canonical Ltd.
# See LICENSE file for licensing details.
"""ZooKeeperCluster class and methods.
`ZooKeeperCluster` is a general-purpose handler for managing the ZooKeeper peer-relation data,
and scale-up/down orchestration.
Exposed attributes include a reference to the `cluster` peer relation for ZooKeeper, as well as wrappers
for grabbing units that have started the ZooKeeper service, found from the peer unit data.
Exposed methods include the setting of passwords for the cluster, ensuring units start up in increasing order
with the correct servers config, and handling the adding/removing of members from the ZooKeeper quorum.
Instances of `ZooKeeperCluster` are to be created during the `__init__` for the target `Charm`.
or from another library. It does not set any relation data itself, and as such is reliant on the calling `Charm`
to set required relation data where necessary during life-cycle events.
It maintains a `status` attribute for passing success/failure of called methods back to the calling `Charm`.
Example minimal usage for `ZooKeeperCluster`:
```python
class ZooKeeperCharm(CharmBase):
def __init__(self, *args):
super().__init__(*args)
self.cluster = ZooKeeperCluster(self)
self.framework.observe(getattr(self.on, "install"), self._on_install)
self.framework.observe(getattr(self.on, "start"), self._on_start)
self.framework.observe(
getattr(self.on, "leader_elected"), self._on_cluster_relation_updated
)
self.framework.observe(
getattr(self.on, "cluster_relation_changed"), self._on_cluster_relation_updated
)
self.framework.observe(
getattr(self.on, "cluster_relation_joined"), self._on_cluster_relation_updated
)
self.framework.observe(
getattr(self.on, "cluster_relation_departed"), self._on_cluster_relation_updated
)
def _on_install(self, event):
unit_myid = self.cluster.get_unit_id(self.unit) + 1
# write_unit_id(id=unit_myid)
def _on_start(self, event):
# setting cluster passwords during first deployment
if self.unit.is_leader():
self.cluster.relation.data[self.app].update({"super_password": self.cluster.generate_password()})
self.cluster.relation.data[self.app].update({"sync_password": self.cluster.generate_password()})
# checking passwords have been set by leader on other units, in case they started first
if not self.cluster.passwords_set():
event.defer()
return
# units must be started in increasing unit.id order
try:
servers, unit_config = self.cluster.ready_to_start(self.unit)
except (NotUnitTurnError, UnitNotFoundError, NoPasswordError) as e:
self.unit.status = self.cluster.status
event.defer()
return
# start_zookeeper_service()
# set useful metadata for each unit
self.cluster.relation.data[self.unit].update(unit_config)
# NECESSARY - set data so that subsequent `self.cluster` commands pick up this unit as a 'started' one
self.cluster.relation.data[self.unit].update({"state": "started"})
def _on_cluster_relation_updated(self, event):
if not self.unit.is_leader():
return
# units need to exist in the app data to be iterated through for next_turn
for unit in self.cluster.started_units:
unit_id = self.cluster.get_unit_id(unit)
current_value = self.cluster.relation.data[self.app].get(str(unit_id), None)
# sets to "added" for init quorum leader, if not already exists
# may already exist if during the case of a failover of unit 0
if unit_id == 0:
self.cluster.relation.data[self.app].update(
{str(unit_id): current_value or "added"}
)
if not self.cluster.passwords_set:
event.defer()
return
# adds + removes members for all self-confirmed started units
updated_servers = self.cluster.update_cluster()
# either Active if successful, else Maintenance
self.unit.status = self.cluster.status
if self.cluster.status == ActiveStatus():
self.cluster.relation.data[self.app].update(updated_servers)
else:
# in the event some unit wasn't started/ready
event.defer()
return
```
"""
import logging
import secrets
import string
import re
from typing import Dict, List, Set, Tuple, Union
from kazoo.handlers.threading import KazooTimeoutError
from ops.charm import CharmBase
from ops.model import (
ActiveStatus,
MaintenanceStatus,
Relation,
StatusBase,
Unit,
)
from charms.zookeeper.v0.client import (
MemberNotReadyError,
MembersSyncingError,
QuorumLeaderNotFoundError,
ZooKeeperManager,
)
# The unique Charmhub library identifier, never change it
LIBID = "8cb992e4287945c5a1efc89ca5a27cdb"
# Increment this major API version when introducing breaking changes
LIBAPI = 0
# Increment this PATCH version before using `charmcraft publish-lib` or reset
# to 0 if you are raising the major API version
LIBPATCH = 1
logger = logging.getLogger(__name__)
CHARM_KEY = "zookeeper"
PEER = "cluster"
class UnitNotFoundError(Exception):
"""A desired unit isn't yet found in the relation data."""
pass
class NotUnitTurnError(Exception):
"""A desired unit isn't next in line to start safely."""
pass
class NoPasswordError(Exception):
"""Required passwords not yet set in the app data."""
pass
class ZooKeeperCluster:
"""Handler for managing the ZK peer-relation.
Mainly for managing scale-up/down orchestration
"""
def __init__(
self,
charm: CharmBase,
client_port: int = 2181,
server_port: int = 2888,
election_port: int = 3888,
) -> None:
self.charm = charm
self.client_port = client_port
self.server_port = server_port
self.election_port = election_port
self.status: StatusBase = MaintenanceStatus("performing cluster operation")
@property
def relation(self) -> Relation:
"""Relation property to be used by both the instance and charm.
Returns:
The peer relation instance
"""
return self.charm.model.get_relation(PEER)
@property
def peer_units(self) -> Set[Unit]:
"""Grabs all units in the current peer relation, including the running unit.
Returns:
Set of units in the current peer relation, including the running unit
"""
return set([self.charm.unit] + list(self.relation.units))
@property
def started_units(self) -> Set[Unit]:
"""Checks peer relation units for whether they've started the ZK service.
Such units are ready to join the ZK quorum if they haven't already.
Returns:
Set of units with unit data "state" == "started". Shows only those units
currently found related to the current unit.
"""
started_units = set()
for unit in self.peer_units:
if self.relation.data[unit].get("state", None) == "started":
started_units.add(unit)
return started_units
@property
def active_hosts(self) -> List[str]:
"""Grabs all the hosts of the started units.
Returns:
List of hosts for started units
"""
active_hosts = []
# grabs all currently 'started' units from unit data
# failed units will be absent
for unit in self.started_units:
config = self.unit_config(unit=unit)
active_hosts.append(config["host"])
return active_hosts
@property
def active_servers(self) -> Set[str]:
"""Grabs all the server strings of the started units.
Returns:
List of ZK server strings for started units
"""
active_servers = set()
# grabs all currently 'started' units from unit data
# failed units will be absent
for unit in self.started_units:
config = self.unit_config(unit=unit)
active_servers.add(config["server_string"])
return active_servers
@staticmethod
def get_unit_id(unit: Unit) -> int:
"""Grabs the unit's ID as defined by Juju.
Args:
unit: The target `Unit`
Returns:
The Juju unit ID for the unit.
e.g `zookeeper/0` -> `0`
"""
return int(unit.name.split("/")[1])
def get_unit_from_id(self, unit_id: int) -> Unit:
"""Grabs the corresponding Unit for a given Juju unit ID
Args:
unit_id: The target unit id
Returns:
The target `Unit`
Raises:
UnitNotFoundError: The desired unit could not be found in the peer data
"""
for unit in self.peer_units:
if int(unit.name.split("/")[1]) == unit_id:
return unit
raise UnitNotFoundError
def unit_config(
self, unit: Union[Unit, int], state: str = "ready", role: str = "participant"
) -> Dict[str, str]:
"""Builds a collection of data useful for ZK for a given unit.
Args:
unit: The target `Unit`, either explicitly or from it's Juju unit ID
state: The desired output state. "ready" or "started"
role: The ZK role for the unit. Default = "participant"
Returns:
The generated config for the given unit.
e.g for unit zookeeper/1:
{
"host": 10.121.23.23,
"server_string": "server.1=host:server_port:election_port:role;localhost:clientport",
"server_id": "2",
"unit_id": "1",
"unit_name": "zookeeper/1",
"state": "ready",
}
Raises:
UnitNotFoundError: When the target unit can't be found in the unit relation data, and/or cannot extract the private-address
"""
unit_id = None
server_id = None
if isinstance(unit, Unit):
unit = unit
unit_id = self.get_unit_id(unit=unit)
server_id = unit_id + 1
if isinstance(unit, int):
unit_id = unit
server_id = unit + 1
unit = self.get_unit_from_id(unit)
try:
host = self.relation.data[unit]["private-address"]
except KeyError:
raise UnitNotFoundError
server_string = f"server.{server_id}={host}:{self.server_port}:{self.election_port}:{role};0.0.0.0:{self.client_port}"
return {
"host": host,
"server_string": server_string,
"server_id": str(server_id),
"unit_id": str(unit_id),
"unit_name": unit.name,
"state": state,
}
def _get_updated_servers(
self, added_servers: List[str], removed_servers: List[str]
) -> Dict[str, str]:
"""Simple wrapper for building `updated_servers` for passing to app data updates."""
servers_to_update = added_servers + removed_servers
updated_servers = {}
for server in servers_to_update:
unit_id = str(int(re.findall(r"server.([1-9]+)", server)[0]) - 1)
if server in added_servers:
updated_servers[unit_id] = "added"
elif server in removed_servers:
updated_servers[unit_id] = "removed"
return updated_servers
def update_cluster(self) -> Dict[str, str]:
"""Adds and removes members from the current ZK quorum.
To be ran by the Juju leader.
After grabbing all the "started" units that the leader can see in the peer relation unit data.
Removes members not in the quorum anymore (i.e `relation_departed`/`leader_elected` event)
Adds new members to the quorum (i.e `relation_joined` event).
Returns:
A mapping of Juju unit IDs and updated state for changed units
To be used in updating the app data
e.g {"0": "added", "1": "removed"}
"""
super_password, _ = self.passwords
try:
zk = ZooKeeperManager(
hosts=self.active_hosts,
client_port=self.client_port,
username="super",
password=super_password,
)
# remove units first, faster due to no startup/sync delay
zk_members = zk.server_members
servers_to_remove = list(zk_members - self.active_servers)
zk.remove_members(members=servers_to_remove)
# sorting units to ensure units are added in id order
zk_members = zk.server_members
servers_to_add = sorted(self.active_servers - zk_members)
zk.add_members(members=servers_to_add)
self.status = ActiveStatus()
return self._get_updated_servers(
added_servers=servers_to_add, removed_servers=servers_to_remove
)
# caught errors relate to a unit/zk_server not yet being ready to change
except (
MembersSyncingError,
MemberNotReadyError,
QuorumLeaderNotFoundError,
KazooTimeoutError,
UnitNotFoundError,
) as e:
logger.warning(str(e))
self.status = MaintenanceStatus(str(e))
return {}
def _is_unit_turn(self, unit_id: int) -> bool:
"""Checks if all units with a lower id than the current unit has been added/removed to the ZK quorum."""
for peer_id in range(0, unit_id):
if not self.relation.data[self.charm.app].get(str(peer_id), None):
return False
return True
def _generate_units(self, unit_string: str) -> str:
"""Gets valid start-up server strings for current ZK quorum units found in the app data."""
servers = ""
for unit_id, state in self.relation.data[self.charm.app].items():
if state == "added":
server_string = self.unit_config(unit=int(unit_id))["server_string"]
servers = servers + "\n" + server_string
servers = servers + "\n" + unit_string
return servers
def ready_to_start(self, unit: Union[Unit, int]) -> Tuple[str, Dict]:
"""Decides whether a unit should start the ZK service, and with what configuration.
Args:
unit: the `Unit` or Juju unit ID to evaluate startability
Returns:
`servers`: a new-line delimited string of servers to add to a config file
`unit_config`: a mapping of configuration for the given unit to be added to unit data
Raises:
`UnitNotFoundError`: if a lower ID unit is missing from the app/unit data
`NotUnitTurnError`: if a lower ID unit has not yet been added to the ZK quorum
"""
servers = ""
unit_config = self.unit_config(unit=unit, state="ready", role="observer")
unit_string = unit_config["server_string"]
unit_id = unit_config["unit_id"]
if not self.relation.data[self.charm.app].get("sync_password", None):
raise NoPasswordError
# during cluster startup, we want unit id 0 to start as a solo participant
# if unit 0 fails and restarts after that, we want it to start as a normal unit
# with all current members
if int(unit_id) == 0 and not self.relation.data[self.charm.app].get("0", None):
unit_string = unit_string.replace("observer", "participant")
return unit_string.replace("observer", "participant"), unit_config
if not self._is_unit_turn(unit_id=int(unit_id)):
self.status = MaintenanceStatus("other units not yet added")
raise NotUnitTurnError
servers = self._generate_units(unit_string=unit_string)
return servers, unit_config
@staticmethod
def generate_password():
"""Creates randomized string for use as app passwords.
Returns:
String of 32 randomized letter+digit characters
"""
return "".join([secrets.choice(string.ascii_letters + string.digits) for _ in range(32)])
@property
def passwords(self) -> Tuple[str, str]:
"""Gets the current super+sync passwords from the app relation data.
Returns:
Tuple of super_password, sync_password
"""
super_password = str(self.relation.data[self.charm.app].get("super_password", ""))
sync_password = str(self.relation.data[self.charm.app].get("sync_password", ""))
return super_password, sync_password
@property
def passwords_set(self) -> bool:
"""Checks that the two desired passwords are in the app relation data.
Returns:
True if both passwords have been set. Otherwise False
"""
super_password, sync_password = self.passwords
if not super_password or not sync_password:
return False
else:
return True