1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
//! Encryption, decryption, and hydration methods for [`Config`].
//!
//! These methods handle the in-memory hydration of encrypted credentials
//! (API keys, proxy auth, MCP secrets, env vars) and their re-encryption
//! before persisting to disk.
use anyhow::{Context, Result};
use super::{Config, ProxyAuth};
impl Config {
// ── Proxy auth ─────────────────────────────────────────────────────
/// Populate `proxy_auth` (plaintext) from `proxy_auth_encrypted` if present.
///
/// Many parts of the code rely on `proxy_auth` being hydrated in-memory so
/// we can re-encrypt deterministically on save without ever persisting
/// plaintext credentials.
pub fn hydrate_proxy_auth_from_encrypted(&mut self) {
if self.proxy_auth.is_some() {
return;
}
// Backward compatibility:
// Older Bodhi/Tauri builds persisted proxy auth as per-scheme encrypted fields:
// `http_proxy_auth_encrypted` / `https_proxy_auth_encrypted`.
//
// Those live under `extra` (flatten) in the unified config. Seed the new
// `proxy_auth_encrypted` field so the rest of the code can stay uniform.
if self
.proxy_auth_encrypted
.as_deref()
.map(|s| s.trim().is_empty())
.unwrap_or(true)
{
let legacy = self
.extra
.get("https_proxy_auth_encrypted")
.and_then(|v| v.as_str())
.or_else(|| {
self.extra
.get("http_proxy_auth_encrypted")
.and_then(|v| v.as_str())
})
.map(|s| s.trim())
.filter(|s| !s.is_empty())
.map(|s| s.to_string());
if let Some(legacy) = legacy {
self.proxy_auth_encrypted = Some(legacy);
}
}
let Some(encrypted) = self.proxy_auth_encrypted.as_deref() else {
return;
};
match crate::encryption::decrypt(encrypted) {
Ok(decrypted) => match serde_json::from_str::<ProxyAuth>(&decrypted) {
Ok(auth) => {
self.proxy_auth = Some(auth);
// Once hydrated successfully, drop legacy keys so a future save writes only
// the canonical `proxy_auth_encrypted` field.
self.extra.remove("http_proxy_auth_encrypted");
self.extra.remove("https_proxy_auth_encrypted");
}
Err(e) => tracing::warn!("Failed to parse decrypted proxy auth JSON: {}", e),
},
Err(e) => tracing::warn!("Failed to decrypt proxy auth: {}", e),
}
}
/// Refresh `proxy_auth_encrypted` from the current in-memory `proxy_auth`.
///
/// This is used both when persisting the config to disk and when generating
/// API responses that should never include plaintext proxy credentials.
pub fn refresh_proxy_auth_encrypted(&mut self) -> Result<()> {
// Keep on-disk representation fully derived from the in-memory plaintext:
// - Some(auth) => always (re-)encrypt and store `proxy_auth_encrypted`
// - None => remove `proxy_auth_encrypted`
let Some(auth) = self.proxy_auth.as_ref() else {
self.proxy_auth_encrypted = None;
return Ok(());
};
let auth_str = serde_json::to_string(auth).context("Failed to serialize proxy auth")?;
let encrypted =
crate::encryption::encrypt(&auth_str).context("Failed to encrypt proxy auth")?;
self.proxy_auth_encrypted = Some(encrypted);
Ok(())
}
// ── Provider API keys ──────────────────────────────────────────────
pub fn hydrate_provider_api_keys_from_encrypted(&mut self) {
if let Some(openai) = self.providers.openai.as_mut() {
if openai.api_key.trim().is_empty() {
if let Some(encrypted) = openai.api_key_encrypted.as_deref() {
match crate::encryption::decrypt(encrypted) {
Ok(value) => openai.api_key = value,
Err(e) => tracing::warn!("Failed to decrypt OpenAI api_key: {}", e),
}
}
}
}
if let Some(anthropic) = self.providers.anthropic.as_mut() {
if anthropic.api_key.trim().is_empty() {
if let Some(encrypted) = anthropic.api_key_encrypted.as_deref() {
match crate::encryption::decrypt(encrypted) {
Ok(value) => anthropic.api_key = value,
Err(e) => tracing::warn!("Failed to decrypt Anthropic api_key: {}", e),
}
}
}
}
if let Some(gemini) = self.providers.gemini.as_mut() {
if gemini.api_key.trim().is_empty() {
if let Some(encrypted) = gemini.api_key_encrypted.as_deref() {
match crate::encryption::decrypt(encrypted) {
Ok(value) => gemini.api_key = value,
Err(e) => tracing::warn!("Failed to decrypt Gemini api_key: {}", e),
}
}
}
}
if let Some(bodhi) = self.providers.bodhi.as_mut() {
if bodhi.api_key.trim().is_empty() {
if let Some(encrypted) = bodhi.api_key_encrypted.as_deref() {
match crate::encryption::decrypt(encrypted) {
Ok(value) => bodhi.api_key = value,
Err(e) => tracing::warn!("Failed to decrypt Bodhi api_key: {}", e),
}
}
}
}
}
pub fn refresh_provider_api_keys_encrypted(&mut self) -> Result<()> {
if let Some(openai) = self.providers.openai.as_mut() {
let api_key = openai.api_key.trim();
openai.api_key_encrypted = if api_key.is_empty() {
None
} else {
Some(
crate::encryption::encrypt(api_key)
.context("Failed to encrypt OpenAI api_key")?,
)
};
}
if let Some(anthropic) = self.providers.anthropic.as_mut() {
let api_key = anthropic.api_key.trim();
anthropic.api_key_encrypted = if api_key.is_empty() {
None
} else {
Some(
crate::encryption::encrypt(api_key)
.context("Failed to encrypt Anthropic api_key")?,
)
};
}
if let Some(gemini) = self.providers.gemini.as_mut() {
let api_key = gemini.api_key.trim();
gemini.api_key_encrypted = if api_key.is_empty() {
None
} else {
Some(
crate::encryption::encrypt(api_key)
.context("Failed to encrypt Gemini api_key")?,
)
};
}
if let Some(bodhi) = self.providers.bodhi.as_mut() {
let api_key = bodhi.api_key.trim();
bodhi.api_key_encrypted = if api_key.is_empty() {
None
} else {
Some(
crate::encryption::encrypt(api_key)
.context("Failed to encrypt Bodhi api_key")?,
)
};
}
Ok(())
}
// ── Provider instance API keys ─────────────────────────────────────
/// Hydrate plaintext `api_key` fields on provider instances from their
/// encrypted counterparts.
pub fn hydrate_provider_instance_api_keys_from_encrypted(&mut self) {
for (id, instance) in self.provider_instances.iter_mut() {
if instance.api_key.trim().is_empty() {
if let Some(encrypted) = instance.api_key_encrypted.as_deref() {
match crate::encryption::decrypt(encrypted) {
Ok(value) => instance.api_key = value,
Err(e) => {
tracing::warn!(instance_id = id, "Failed to decrypt api_key: {}", e)
}
}
}
}
}
}
/// Re-encrypt all provider instance API keys and write back to
/// `api_key_encrypted`. Used before persisting to disk.
pub fn refresh_provider_instance_api_keys_encrypted(&mut self) -> Result<()> {
for (id, instance) in self.provider_instances.iter_mut() {
let api_key = instance.api_key.trim();
instance.api_key_encrypted = if api_key.is_empty() {
None
} else {
Some(crate::encryption::encrypt(api_key).context(format!(
"Failed to encrypt api_key for provider instance '{}'",
id
))?)
};
}
Ok(())
}
// ── MCP secrets ────────────────────────────────────────────────────
pub fn hydrate_mcp_secrets_from_encrypted(&mut self) {
for server in self.mcp.servers.iter_mut() {
match &mut server.transport {
bamboo_domain::mcp_config::TransportConfig::Stdio(stdio) => {
if stdio.env_encrypted.is_empty() {
continue;
}
// Avoid borrow-checker gymnastics by iterating a cloned map.
for (key, encrypted) in stdio.env_encrypted.clone() {
let should_hydrate = stdio
.env
.get(&key)
.map(|v| v.trim().is_empty())
.unwrap_or(true);
if !should_hydrate {
continue;
}
match crate::encryption::decrypt(&encrypted) {
Ok(value) => {
stdio.env.insert(key, value);
}
Err(e) => tracing::warn!("Failed to decrypt MCP stdio env var: {}", e),
}
}
}
bamboo_domain::mcp_config::TransportConfig::Sse(sse) => {
for header in sse.headers.iter_mut() {
if !header.value.trim().is_empty() {
continue;
}
let Some(encrypted) = header.value_encrypted.as_deref() else {
continue;
};
match crate::encryption::decrypt(encrypted) {
Ok(value) => header.value = value,
Err(e) => {
tracing::warn!("Failed to decrypt MCP SSE header value: {}", e)
}
}
}
}
bamboo_domain::mcp_config::TransportConfig::StreamableHttp(sh) => {
for header in sh.headers.iter_mut() {
if !header.value.trim().is_empty() {
continue;
}
let Some(encrypted) = header.value_encrypted.as_deref() else {
continue;
};
match crate::encryption::decrypt(encrypted) {
Ok(value) => header.value = value,
Err(e) => {
tracing::warn!(
"Failed to decrypt MCP StreamableHTTP header value: {}",
e
)
}
}
}
}
}
}
}
pub fn refresh_mcp_secrets_encrypted(&mut self) -> Result<()> {
for server in self.mcp.servers.iter_mut() {
match &mut server.transport {
bamboo_domain::mcp_config::TransportConfig::Stdio(stdio) => {
stdio.env_encrypted.clear();
for (key, value) in &stdio.env {
let encrypted = crate::encryption::encrypt(value).with_context(|| {
format!("Failed to encrypt MCP stdio env var '{key}'")
})?;
stdio.env_encrypted.insert(key.clone(), encrypted);
}
}
bamboo_domain::mcp_config::TransportConfig::Sse(sse) => {
for header in sse.headers.iter_mut() {
let configured = !header.value.trim().is_empty();
header.value_encrypted = if !configured {
None
} else {
Some(crate::encryption::encrypt(&header.value).with_context(|| {
format!("Failed to encrypt MCP SSE header '{}'", header.name)
})?)
};
}
}
bamboo_domain::mcp_config::TransportConfig::StreamableHttp(sh) => {
for header in sh.headers.iter_mut() {
let configured = !header.value.trim().is_empty();
header.value_encrypted = if !configured {
None
} else {
Some(crate::encryption::encrypt(&header.value).with_context(|| {
format!(
"Failed to encrypt MCP StreamableHTTP header '{}'",
header.name
)
})?)
};
}
}
}
}
Ok(())
}
// ── Env vars encryption ────────────────────────────────────────────
/// Decrypt secret env vars into in-memory plaintext after loading config.
pub fn hydrate_env_vars_from_encrypted(&mut self) {
for entry in &mut self.env_vars {
if !entry.secret {
continue;
}
if !entry.value.trim().is_empty() {
// Already has plaintext (e.g. in-memory update).
continue;
}
let Some(encrypted) = &entry.value_encrypted else {
continue;
};
match crate::encryption::decrypt(encrypted) {
Ok(value) => entry.value = value,
Err(e) => tracing::warn!("Failed to decrypt env var '{}': {}", entry.name, e),
}
}
}
/// Re-encrypt secret env vars before persisting to disk.
pub fn refresh_env_vars_encrypted(&mut self) -> Result<()> {
for entry in &mut self.env_vars {
if entry.secret && !entry.value.trim().is_empty() {
entry.value_encrypted = Some(
crate::encryption::encrypt(&entry.value)
.with_context(|| format!("Failed to encrypt env var '{}'", entry.name))?,
);
} else if !entry.secret {
entry.value_encrypted = None;
}
}
Ok(())
}
/// Clear plaintext values for secrets before serialization to disk.
pub fn sanitize_env_vars_for_disk(&mut self) {
for entry in &mut self.env_vars {
if entry.secret {
entry.value = String::new();
}
}
}
}