2022-03-06 14:50:00 +01:00
|
|
|
use std::collections::BTreeMap;
|
|
|
|
use std::net::{IpAddr, SocketAddr};
|
|
|
|
|
|
|
|
use kube::{
|
|
|
|
api::{ListParams, Patch, PatchParams, PostParams},
|
|
|
|
Api, Client, CustomResource, CustomResourceExt,
|
|
|
|
};
|
|
|
|
|
|
|
|
use k8s_openapi::apiextensions_apiserver::pkg::apis::apiextensions::v1::CustomResourceDefinition;
|
|
|
|
use schemars::JsonSchema;
|
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
|
2024-02-13 12:55:41 +01:00
|
|
|
use garage_net::NodeID;
|
2022-03-06 14:50:00 +01:00
|
|
|
|
2022-10-18 18:38:20 +02:00
|
|
|
use garage_util::config::KubernetesDiscoveryConfig;
|
|
|
|
|
2022-03-06 14:50:00 +01:00
|
|
|
static K8S_GROUP: &str = "deuxfleurs.fr";
|
|
|
|
|
|
|
|
#[derive(CustomResource, Debug, Serialize, Deserialize, Clone, JsonSchema)]
|
|
|
|
#[kube(
|
|
|
|
group = "deuxfleurs.fr",
|
|
|
|
version = "v1",
|
|
|
|
kind = "GarageNode",
|
|
|
|
namespaced
|
|
|
|
)]
|
|
|
|
pub struct Node {
|
|
|
|
hostname: String,
|
|
|
|
address: IpAddr,
|
|
|
|
port: u16,
|
|
|
|
}
|
|
|
|
|
2022-03-16 12:09:50 +01:00
|
|
|
pub async fn create_kubernetes_crd() -> Result<(), kube::Error> {
|
2022-03-06 14:50:00 +01:00
|
|
|
let client = Client::try_default().await?;
|
|
|
|
let crds: Api<CustomResourceDefinition> = Api::all(client.clone());
|
|
|
|
|
|
|
|
let params = PatchParams::apply(&format!("garage.{}", K8S_GROUP));
|
|
|
|
let crd = GarageNode::crd();
|
|
|
|
let patch = Patch::Apply(crd);
|
|
|
|
crds.patch(&format!("garagenodes.{}", K8S_GROUP), ¶ms, &patch)
|
|
|
|
.await?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn get_kubernetes_nodes(
|
2022-10-18 18:38:20 +02:00
|
|
|
kubernetes_config: &KubernetesDiscoveryConfig,
|
2022-03-16 12:09:50 +01:00
|
|
|
) -> Result<Vec<(NodeID, SocketAddr)>, kube::Error> {
|
2022-03-06 14:50:00 +01:00
|
|
|
let client = Client::try_default().await?;
|
2022-10-18 18:38:20 +02:00
|
|
|
let nodes: Api<GarageNode> = Api::namespaced(client.clone(), &kubernetes_config.namespace);
|
2022-03-06 14:50:00 +01:00
|
|
|
|
|
|
|
let lp = ListParams::default().labels(&format!(
|
|
|
|
"garage.{}/service={}",
|
2022-10-18 18:38:20 +02:00
|
|
|
K8S_GROUP, kubernetes_config.service_name
|
2022-03-06 14:50:00 +01:00
|
|
|
));
|
|
|
|
|
|
|
|
let nodes = nodes.list(&lp).await?;
|
|
|
|
let mut ret = Vec::with_capacity(nodes.items.len());
|
|
|
|
|
|
|
|
for node in nodes {
|
2022-09-13 16:01:55 +02:00
|
|
|
info!("Found Pod: {:?}", node.metadata.name);
|
2022-03-06 14:50:00 +01:00
|
|
|
|
|
|
|
let pubkey = &node
|
|
|
|
.metadata
|
|
|
|
.name
|
2022-03-14 12:00:23 +01:00
|
|
|
.and_then(|k| hex::decode(&k).ok())
|
|
|
|
.and_then(|k| NodeID::from_slice(&k[..]));
|
2022-03-06 14:50:00 +01:00
|
|
|
|
|
|
|
if let Some(pubkey) = pubkey {
|
|
|
|
ret.push((*pubkey, SocketAddr::new(node.spec.address, node.spec.port)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(ret)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn publish_kubernetes_node(
|
2022-10-18 18:38:20 +02:00
|
|
|
kubernetes_config: &KubernetesDiscoveryConfig,
|
2022-03-06 14:50:00 +01:00
|
|
|
node_id: NodeID,
|
|
|
|
hostname: &str,
|
|
|
|
rpc_public_addr: SocketAddr,
|
2022-03-16 12:09:50 +01:00
|
|
|
) -> Result<(), kube::Error> {
|
2022-03-06 14:50:00 +01:00
|
|
|
let node_pubkey = hex::encode(node_id);
|
|
|
|
|
|
|
|
let mut node = GarageNode::new(
|
|
|
|
&node_pubkey,
|
|
|
|
Node {
|
|
|
|
hostname: hostname.to_string(),
|
|
|
|
address: rpc_public_addr.ip(),
|
|
|
|
port: rpc_public_addr.port(),
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
let labels = node.metadata.labels.insert(BTreeMap::new());
|
|
|
|
labels.insert(
|
|
|
|
format!("garage.{}/service", K8S_GROUP),
|
2022-10-18 18:38:20 +02:00
|
|
|
kubernetes_config.service_name.to_string(),
|
2022-03-06 14:50:00 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
debug!("Node object to be applied: {:#?}", node);
|
|
|
|
|
|
|
|
let client = Client::try_default().await?;
|
2022-10-18 18:38:20 +02:00
|
|
|
let nodes: Api<GarageNode> = Api::namespaced(client.clone(), &kubernetes_config.namespace);
|
2022-03-06 14:50:00 +01:00
|
|
|
|
|
|
|
if let Ok(old_node) = nodes.get(&node_pubkey).await {
|
|
|
|
node.metadata.resource_version = old_node.metadata.resource_version;
|
|
|
|
nodes
|
|
|
|
.replace(&node_pubkey, &PostParams::default(), &node)
|
|
|
|
.await?;
|
|
|
|
} else {
|
|
|
|
nodes.create(&PostParams::default(), &node).await?;
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|