alien_bindings/providers/vault/
kubernetes_secret.rs1use crate::error::{ErrorData, Result};
2use alien_error::{Context, ContextError, IntoAlienError};
3use alien_k8s_clients::secrets::SecretsApi;
4use async_trait::async_trait;
5use k8s_openapi::api::core::v1::Secret;
6use k8s_openapi::apimachinery::pkg::apis::meta::v1::ObjectMeta;
7use std::collections::BTreeMap;
8use std::sync::Arc;
9
10#[derive(Debug)]
20pub struct KubernetesSecretVault {
21 client: Arc<dyn SecretsApi>,
22 namespace: String,
23 vault_prefix: String,
24}
25
26impl KubernetesSecretVault {
27 pub fn new(client: Arc<dyn SecretsApi>, namespace: String, vault_prefix: String) -> Self {
34 Self {
35 client,
36 namespace,
37 vault_prefix,
38 }
39 }
40
41 fn secret_resource_name(&self, secret_name: &str) -> String {
45 let combined = format!("{}-{}", self.vault_prefix, secret_name);
46
47 let clean = combined
49 .chars()
50 .filter(|c| c.is_alphanumeric() || *c == '-' || *c == '_')
51 .collect::<String>()
52 .to_lowercase()
53 .replace('_', "-");
54
55 if clean.len() > 253 {
57 clean[..253].to_string()
58 } else {
59 clean
60 }
61 }
62}
63
64#[async_trait]
65impl crate::traits::Binding for KubernetesSecretVault {}
66
67#[async_trait]
68impl crate::traits::Vault for KubernetesSecretVault {
69 async fn get_secret(&self, secret_name: &str) -> Result<String> {
71 let secret_resource_name = self.secret_resource_name(secret_name);
72
73 let secret = self
75 .client
76 .get_secret(&self.namespace, &secret_resource_name)
77 .await
78 .context(ErrorData::CloudPlatformError {
79 message: format!("Failed to get secret '{}'", secret_name),
80 resource_id: None,
81 })?;
82
83 let value = secret
85 .data
86 .as_ref()
87 .and_then(|data| data.get("value"))
88 .ok_or_else(|| {
89 alien_error::AlienError::new(ErrorData::CloudPlatformError {
90 message: format!("Secret '{}' has no 'value' field", secret_name),
91 resource_id: None,
92 })
93 })?;
94
95 let decoded = String::from_utf8(value.0.clone())
97 .into_alien_error()
98 .context(ErrorData::CloudPlatformError {
99 message: format!("Failed to decode secret '{}' value", secret_name),
100 resource_id: None,
101 })?;
102
103 Ok(decoded)
104 }
105
106 async fn set_secret(&self, secret_name: &str, value: &str) -> Result<()> {
108 let secret_resource_name = self.secret_resource_name(secret_name);
109
110 let mut data = BTreeMap::new();
112 data.insert(
113 "value".to_string(),
114 k8s_openapi::ByteString(value.as_bytes().to_vec()),
115 );
116
117 let secret = Secret {
118 metadata: ObjectMeta {
119 name: Some(secret_resource_name.clone()),
120 namespace: Some(self.namespace.clone()),
121 labels: Some({
122 let mut labels = BTreeMap::new();
123 labels.insert("managed-by".to_string(), "alien".to_string());
124 labels.insert("vault-prefix".to_string(), self.vault_prefix.clone());
125 labels
126 }),
127 ..Default::default()
128 },
129 data: Some(data),
130 ..Default::default()
131 };
132
133 match self.client.create_secret(&self.namespace, &secret).await {
135 Ok(_) => Ok(()),
136 Err(e) => {
137 if matches!(
139 e.error,
140 Some(alien_client_core::ErrorData::RemoteResourceConflict { .. })
141 ) {
142 self.client
143 .update_secret(&self.namespace, &secret_resource_name, &secret)
144 .await
145 .context(ErrorData::CloudPlatformError {
146 message: format!("Failed to update secret '{}'", secret_name),
147 resource_id: None,
148 })?;
149 Ok(())
150 } else {
151 Err(e.context(ErrorData::CloudPlatformError {
152 message: format!("Failed to create secret '{}'", secret_name),
153 resource_id: None,
154 }))
155 }
156 }
157 }
158 }
159
160 async fn delete_secret(&self, secret_name: &str) -> Result<()> {
162 let secret_resource_name = self.secret_resource_name(secret_name);
163
164 self.client
165 .delete_secret(&self.namespace, &secret_resource_name)
166 .await
167 .context(ErrorData::CloudPlatformError {
168 message: format!("Failed to delete secret '{}'", secret_name),
169 resource_id: None,
170 })
171 }
172}
173
174#[cfg(test)]
175mod tests {
176 use super::*;
177
178 #[test]
179 fn test_secret_resource_name_generation() {
180 let vault_prefix = "acme-monitoring-secrets";
182 let secret_name = "API_KEY";
183 let combined = format!("{}-{}", vault_prefix, secret_name);
184 let result = combined
185 .chars()
186 .filter(|c| c.is_alphanumeric() || *c == '-' || *c == '_')
187 .collect::<String>()
188 .to_lowercase()
189 .replace('_', "-");
190
191 assert_eq!(result, "acme-monitoring-secrets-api-key");
192
193 let secret_name2 = "MY_SECRET_KEY";
195 let combined2 = format!("{}-{}", vault_prefix, secret_name2);
196 let result2 = combined2
197 .chars()
198 .filter(|c| c.is_alphanumeric() || *c == '-' || *c == '_')
199 .collect::<String>()
200 .to_lowercase()
201 .replace('_', "-");
202
203 assert_eq!(result2, "acme-monitoring-secrets-my-secret-key");
204
205 let long_name = "A".repeat(300);
207 let combined3 = format!("{}-{}", vault_prefix, long_name);
208 let result3 = combined3
209 .chars()
210 .filter(|c| c.is_alphanumeric() || *c == '-' || *c == '_')
211 .collect::<String>()
212 .to_lowercase();
213 let truncated = if result3.len() > 253 {
214 result3[..253].to_string()
215 } else {
216 result3
217 };
218 assert!(truncated.len() <= 253);
219 }
220}