Skip to main content

alien_bindings/providers/vault/
kubernetes_secret.rs

1use 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/// Kubernetes Secret vault binding implementation.
11///
12/// Stores secrets as Kubernetes Secret resources with naming convention:
13/// {vaultPrefix}-{secretName} (e.g., "acme-monitoring-secrets-API_KEY")
14///
15/// Each secret is a separate Kubernetes Secret resource to enable:
16/// - Granular access control via RBAC
17/// - Individual secret rotation without affecting others
18/// - Simpler secret management (no parsing of bundled secrets)
19#[derive(Debug)]
20pub struct KubernetesSecretVault {
21    client: Arc<dyn SecretsApi>,
22    namespace: String,
23    vault_prefix: String,
24}
25
26impl KubernetesSecretVault {
27    /// Create a new Kubernetes Secret vault binding.
28    ///
29    /// # Arguments
30    /// * `client` - Kubernetes Secrets API client
31    /// * `namespace` - Kubernetes namespace where secrets are stored
32    /// * `vault_prefix` - Prefix for secret names (e.g., "acme-monitoring-secrets")
33    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    /// Generate the Kubernetes Secret name for a given secret key.
42    /// Format: {vault_prefix}-{secret_name}
43    /// Example: "acme-monitoring-secrets-api-key"
44    fn secret_resource_name(&self, secret_name: &str) -> String {
45        let combined = format!("{}-{}", self.vault_prefix, secret_name);
46
47        // Kubernetes names must be lowercase and follow DNS-1123 label requirements
48        let clean = combined
49            .chars()
50            .filter(|c| c.is_alphanumeric() || *c == '-' || *c == '_')
51            .collect::<String>()
52            .to_lowercase()
53            .replace('_', "-");
54
55        // Truncate to 253 characters (Kubernetes Secret name limit)
56        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    /// Get a secret value by name
70    async fn get_secret(&self, secret_name: &str) -> Result<String> {
71        let secret_resource_name = self.secret_resource_name(secret_name);
72
73        // Get the Kubernetes Secret
74        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        // Extract the secret value from the "value" key in secret data
84        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        // Decode base64 value (Kubernetes stores secret data as base64)
96        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    /// Set a secret value, creating it if it doesn't exist or updating it if it does
107    async fn set_secret(&self, secret_name: &str, value: &str) -> Result<()> {
108        let secret_resource_name = self.secret_resource_name(secret_name);
109
110        // Build the Secret resource
111        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        // Try to create the secret first
134        match self.client.create_secret(&self.namespace, &secret).await {
135            Ok(_) => Ok(()),
136            Err(e) => {
137                // If secret already exists, update it instead
138                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    /// Delete a secret
161    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        // Test basic generation
181        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        // Test character filtering (underscores become hyphens)
194        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        // Test length truncation
206        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}