1use anyhow::{anyhow, Context, Result};
27use directories::{ProjectDirs, UserDirs};
28use parking_lot::Mutex;
29use serde::{Deserialize, Serialize};
30use std::path::{Path, PathBuf};
31
32const TRACE_TARGET: &str = "studio_worker::config";
35
36#[derive(Debug, Clone, Serialize, Deserialize)]
37pub struct Config {
38 pub api_base_url: String,
40 #[serde(default, skip_serializing_if = "Option::is_none")]
44 pub worker_id: Option<String>,
45 #[serde(default, skip_serializing_if = "Option::is_none")]
48 pub auth_token: Option<String>,
49 pub vram_threshold_gb: f32,
51 pub auto_start: bool,
53 #[serde(default = "default_start_minimised")]
58 pub start_minimised: bool,
59 #[serde(default = "default_auto_update_enabled")]
62 pub auto_update_enabled: bool,
63 #[serde(default = "default_auto_update_interval")]
65 pub auto_update_interval_secs: u64,
66 #[serde(default = "default_auto_update_feed")]
68 pub auto_update_feed: String,
69 #[serde(default)]
71 pub auto_update_prerelease: bool,
72 #[serde(default = "default_models_root_persisted")]
76 pub models_root: PathBuf,
77 #[serde(default, skip_serializing_if = "Option::is_none")]
81 pub ws_reconnect_attempts: Option<u32>,
82 #[serde(default, skip_serializing_if = "Option::is_none")]
86 pub install_id: Option<String>,
87 #[serde(default, skip_serializing_if = "Option::is_none")]
90 pub registration_request_id: Option<String>,
91 #[serde(default, skip_serializing_if = "Option::is_none")]
94 pub registration_secret: Option<String>,
95}
96
97fn default_auto_update_enabled() -> bool {
98 true
99}
100fn default_start_minimised() -> bool {
101 true
102}
103fn default_auto_update_interval() -> u64 {
104 1800
105}
106fn default_auto_update_feed() -> String {
107 "https://api.github.com/repos/webbertakken/studio-worker/releases".into()
108}
109
110pub fn default_models_root() -> PathBuf {
114 models_root_from(home_dir())
115}
116
117fn home_dir() -> Option<PathBuf> {
121 UserDirs::new().map(|d| d.home_dir().to_path_buf())
122}
123
124fn models_root_from(home: Option<PathBuf>) -> PathBuf {
128 match home {
129 Some(home) => home.join("models"),
130 None => std::env::temp_dir().join("studio-worker-models"),
131 }
132}
133
134fn default_models_root_persisted() -> PathBuf {
135 default_models_root()
136}
137
138fn expand_home(path: PathBuf) -> PathBuf {
143 expand_home_with(path, home_dir())
144}
145
146fn expand_home_with(path: PathBuf, home: Option<PathBuf>) -> PathBuf {
150 let s = path.to_string_lossy();
151 if s == "~" {
152 return home.unwrap_or(path);
153 }
154 if let Some(rest) = s.strip_prefix("~/") {
155 if let Some(home) = home {
156 return home.join(rest);
157 }
158 }
159 path
160}
161
162impl Default for Config {
163 fn default() -> Self {
164 Self {
165 api_base_url: "https://studio.minis.gg/".into(),
166 worker_id: None,
167 auth_token: None,
168 vram_threshold_gb: 12.0,
169 auto_start: true,
170 start_minimised: default_start_minimised(),
171 auto_update_enabled: default_auto_update_enabled(),
172 auto_update_interval_secs: default_auto_update_interval(),
173 auto_update_feed: default_auto_update_feed(),
174 auto_update_prerelease: false,
175 models_root: default_models_root(),
176 ws_reconnect_attempts: None,
177 install_id: None,
178 registration_request_id: None,
179 registration_secret: None,
180 }
181 }
182}
183
184fn default_config_path() -> Result<PathBuf> {
185 let dirs = ProjectDirs::from("gg", "minis", "minis-studio-worker")
186 .ok_or_else(|| anyhow!("cannot resolve config directory"))?;
187 Ok(dirs.config_dir().join("config.toml"))
188}
189
190pub fn resolve_path(override_path: Option<&str>) -> Result<PathBuf> {
191 if let Some(p) = override_path {
192 Ok(PathBuf::from(p))
193 } else {
194 default_config_path()
195 }
196}
197
198pub fn load(override_path: Option<&str>) -> Result<(Config, PathBuf)> {
199 let path = resolve_path(override_path)?;
200 if !path.exists() {
201 let cfg = Config::default();
202 save(&cfg, &path)?;
203 tracing::info!(
204 target: TRACE_TARGET,
205 op = "load",
206 source = "default_created",
207 config_path = %path.display(),
208 api_base_url = %cfg.api_base_url,
209 vram_threshold_gb = cfg.vram_threshold_gb,
210 auto_start = cfg.auto_start,
211 models_root = %cfg.models_root.display(),
212 "config file missing — bootstrapped defaults"
213 );
214 return Ok((cfg, path));
215 }
216 let text = match std::fs::read_to_string(&path) {
217 Ok(text) => text,
218 Err(e) => {
219 tracing::warn!(
223 target: TRACE_TARGET,
224 op = "load",
225 config_path = %path.display(),
226 error = %e,
227 "failed to read config file"
228 );
229 return Err(e).with_context(|| format!("reading {}", path.display()));
230 }
231 };
232 let mut cfg: Config = match toml::from_str(&text) {
233 Ok(cfg) => cfg,
234 Err(e) => {
235 tracing::warn!(
241 target: TRACE_TARGET,
242 op = "load",
243 config_path = %path.display(),
244 "config file is not valid TOML"
245 );
246 return Err(e).context("parsing config.toml");
247 }
248 };
249 cfg.models_root = expand_home(std::mem::take(&mut cfg.models_root));
250 tracing::debug!(
251 target: TRACE_TARGET,
252 op = "load",
253 source = "existing_file",
254 config_path = %path.display(),
255 api_base_url = %cfg.api_base_url,
256 vram_threshold_gb = cfg.vram_threshold_gb,
257 auto_start = cfg.auto_start,
258 models_root = %cfg.models_root.display(),
259 worker_id = cfg.worker_id.as_deref().unwrap_or("(unregistered)"),
260 has_auth_token = cfg.auth_token.is_some(),
261 "loaded config from disk"
262 );
263 Ok((cfg, path))
264}
265
266pub fn save(cfg: &Config, path: &Path) -> Result<()> {
267 match write_config(cfg, path) {
268 Ok(bytes) => {
269 tracing::debug!(
270 target: TRACE_TARGET,
271 op = "save",
272 config_path = %path.display(),
273 vram_threshold_gb = cfg.vram_threshold_gb,
274 auto_start = cfg.auto_start,
275 models_root = %cfg.models_root.display(),
276 bytes = bytes,
277 "persisted config to disk"
278 );
279 Ok(())
280 }
281 Err(e) => {
282 tracing::warn!(
289 target: TRACE_TARGET,
290 op = "save",
291 config_path = %path.display(),
292 error = %e,
293 "failed to persist config to disk"
294 );
295 Err(e)
296 }
297 }
298}
299
300fn write_config(cfg: &Config, path: &Path) -> Result<usize> {
305 if let Some(parent) = path.parent() {
306 std::fs::create_dir_all(parent)
307 .with_context(|| format!("creating {}", parent.display()))?;
308 }
309 let text = toml::to_string_pretty(cfg).with_context(|| "serialising config")?;
310 let bytes = text.len();
311 write_atomic(path, text.as_bytes())?;
312 Ok(bytes)
313}
314
315fn write_atomic(path: &Path, bytes: &[u8]) -> Result<()> {
332 use std::io::Write as _;
333 let dir = match path.parent() {
334 Some(p) if !p.as_os_str().is_empty() => p,
335 _ => Path::new("."),
336 };
337 let mut tmp = tempfile::NamedTempFile::new_in(dir)
338 .with_context(|| format!("creating temp file in {}", dir.display()))?;
339 tmp.write_all(bytes)
340 .with_context(|| "writing temp config")?;
341 tmp.as_file()
342 .sync_all()
343 .with_context(|| "flushing temp config to disk")?;
344 tmp.persist(path)
345 .map_err(|e| anyhow!("atomically replacing {}: {}", path.display(), e.error))?;
346 Ok(())
347}
348
349pub type SharedConfig = std::sync::Arc<Mutex<Config>>;
351
352pub fn shared(cfg: Config) -> SharedConfig {
353 std::sync::Arc::new(Mutex::new(cfg))
354}
355
356#[cfg(test)]
357mod tests {
358 use super::*;
359 use tempfile::tempdir;
360
361 #[test]
362 fn start_minimised_defaults_true_for_configs_predating_the_field() {
363 let cfg: Config = toml::from_str(
366 r#"
367 api_base_url = "https://studio.minis.gg/"
368 vram_threshold_gb = 12.0
369 auto_start = true
370 "#,
371 )
372 .unwrap();
373 assert!(cfg.start_minimised);
374 }
375
376 #[test]
377 fn default_values_are_sensible() {
378 let cfg = Config::default();
379 assert_eq!(cfg.api_base_url, "https://studio.minis.gg/");
380 assert!(cfg.auto_start);
381 assert!(
382 cfg.start_minimised,
383 "the UI must start minimised by default"
384 );
385 assert!(cfg.auto_update_enabled);
386 assert_eq!(cfg.auto_update_interval_secs, 1800);
387 assert!(!cfg.auto_update_prerelease);
388 assert!(cfg.auto_update_feed.contains("webbertakken/studio-worker"));
389 assert_eq!(cfg.vram_threshold_gb, 12.0);
390 assert!(cfg.worker_id.is_none());
391 assert!(cfg.auth_token.is_none());
392 let m = cfg.models_root.to_string_lossy().to_string();
395 assert!(m.ends_with("models") || m.contains("studio-worker-models"));
396 }
397
398 #[test]
399 fn resolve_path_uses_override_when_provided() {
400 let path = resolve_path(Some("/tmp/test-config.toml")).unwrap();
401 assert_eq!(path, PathBuf::from("/tmp/test-config.toml"));
402 }
403
404 #[test]
405 fn resolve_path_defaults_when_no_override() {
406 let path = resolve_path(None).unwrap();
407 let s = path.to_string_lossy();
408 assert!(
409 s.contains("minis-studio-worker") || s.contains("minis.gg.minis-studio-worker"),
410 "unexpected default path: {s}"
411 );
412 assert!(s.ends_with("config.toml"));
413 }
414
415 #[test]
416 fn load_creates_default_when_file_missing() {
417 let dir = tempdir().unwrap();
418 let path = dir.path().join("sub").join("config.toml");
419 let path_str = path.to_string_lossy().to_string();
420 let (cfg, returned_path) = load(Some(&path_str)).unwrap();
421 assert_eq!(returned_path, path);
422 assert_eq!(cfg.api_base_url, "https://studio.minis.gg/");
423 assert!(path.exists());
425 }
426
427 #[test]
428 fn round_trip_via_save_and_load_preserves_fields() {
429 let dir = tempdir().unwrap();
430 let path = dir.path().join("config.toml");
431 let cfg = Config {
432 worker_id: Some("w-123".into()),
433 auth_token: Some("tok-xyz".into()),
434 vram_threshold_gb: 24.0,
435 auto_update_prerelease: true,
436 models_root: PathBuf::from("/tmp/test-models"),
437 ..Config::default()
438 };
439 save(&cfg, &path).unwrap();
440
441 let path_str = path.to_string_lossy().to_string();
442 let (loaded, _) = load(Some(&path_str)).unwrap();
443 assert_eq!(loaded.api_base_url, cfg.api_base_url);
444 assert_eq!(loaded.worker_id, cfg.worker_id);
445 assert_eq!(loaded.auth_token, cfg.auth_token);
446 assert_eq!(loaded.vram_threshold_gb, cfg.vram_threshold_gb);
447 assert_eq!(loaded.auto_update_prerelease, cfg.auto_update_prerelease);
448 assert_eq!(loaded.models_root, cfg.models_root);
449 }
450
451 #[test]
452 fn shared_wraps_in_arc_mutex() {
453 let cfg = Config::default();
454 let shared = shared(cfg.clone());
455 let guard = shared.lock();
456 assert_eq!(guard.api_base_url, cfg.api_base_url);
457 }
458
459 #[test]
460 fn load_returns_error_on_malformed_toml() {
461 let dir = tempdir().unwrap();
462 let path = dir.path().join("config.toml");
463 std::fs::write(&path, "this :: is = not = toml = :").unwrap();
464 let path_str = path.to_string_lossy().to_string();
465 let err = load(Some(&path_str)).unwrap_err();
466 assert!(err.to_string().contains("parsing config.toml"));
467 }
468
469 #[test]
470 fn load_strips_legacy_engine_fields_silently() {
471 let dir = tempdir().unwrap();
475 let path = dir.path().join("config.toml");
476 let legacy = r#"
477 api_base_url = "https://example.invalid"
478 vram_threshold_gb = 8.0
479 auto_start = true
480 engine = "multi"
481 engines = ["llama", "synthetic"]
482 auto_enabled = false
483 label = "alice's rig"
484 "#;
485 std::fs::write(&path, legacy).unwrap();
486 let (cfg, _) = load(Some(&path.to_string_lossy())).unwrap();
487 assert_eq!(cfg.api_base_url, "https://example.invalid");
488 assert_eq!(cfg.vram_threshold_gb, 8.0);
489 }
490
491 #[test]
492 fn load_expands_leading_tilde_in_models_root() {
493 let dir = tempdir().unwrap();
496 let path = dir.path().join("config.toml");
497 let raw = r#"
498 api_base_url = "https://x.invalid"
499 vram_threshold_gb = 4.0
500 auto_start = true
501 auto_update_enabled = false
502 auto_update_interval_secs = 1
503 auto_update_feed = "https://x.invalid"
504 auto_update_prerelease = false
505 models_root = "~/models-test"
506 "#;
507 std::fs::write(&path, raw).unwrap();
508 let (cfg, _) = load(Some(&path.to_string_lossy())).unwrap();
509 assert!(
510 cfg.models_root.is_absolute(),
511 "~/ should expand to an absolute path, got {}",
512 cfg.models_root.display()
513 );
514 assert!(cfg.models_root.ends_with("models-test"));
515 }
516
517 #[test]
518 fn expand_home_leaves_absolute_paths_alone() {
519 let p = PathBuf::from("/tmp/anywhere");
520 assert_eq!(expand_home(p.clone()), p);
521 }
522
523 #[test]
524 fn expand_home_handles_bare_tilde() {
525 let expanded = expand_home(PathBuf::from("~"));
526 assert!(
527 expanded.is_absolute() || expanded == Path::new("~"),
528 "bare ~ expands to home (or stays put on weird boxes), got {}",
529 expanded.display()
530 );
531 }
532
533 #[test]
539 fn models_root_from_uses_home_when_available() {
540 let home = PathBuf::from("/home/someuser");
541 assert_eq!(models_root_from(Some(home.clone())), home.join("models"));
542 }
543
544 #[test]
545 fn models_root_from_falls_back_to_tmp_without_home() {
546 assert_eq!(
547 models_root_from(None),
548 std::env::temp_dir().join("studio-worker-models")
549 );
550 }
551
552 #[test]
553 fn expand_home_with_bare_tilde_uses_injected_home() {
554 let home = PathBuf::from("/home/x");
555 assert_eq!(
556 expand_home_with(PathBuf::from("~"), Some(home.clone())),
557 home
558 );
559 }
560
561 #[test]
562 fn expand_home_with_bare_tilde_without_home_stays_put() {
563 assert_eq!(
564 expand_home_with(PathBuf::from("~"), None),
565 PathBuf::from("~")
566 );
567 }
568
569 #[test]
570 fn expand_home_with_prefix_joins_injected_home() {
571 let home = PathBuf::from("/home/x");
572 assert_eq!(
573 expand_home_with(PathBuf::from("~/models"), Some(home.clone())),
574 home.join("models")
575 );
576 }
577
578 #[test]
579 fn expand_home_with_prefix_without_home_stays_unexpanded() {
580 let p = PathBuf::from("~/models");
581 assert_eq!(expand_home_with(p.clone(), None), p);
582 }
583
584 #[test]
585 fn expand_home_with_leaves_absolute_paths_alone() {
586 let p = PathBuf::from("/tmp/anywhere");
587 assert_eq!(
588 expand_home_with(p.clone(), Some(PathBuf::from("/home/x"))),
589 p
590 );
591 }
592
593 #[cfg(unix)]
594 #[test]
595 fn save_writes_config_owner_only_because_it_holds_secrets() {
596 use std::os::unix::fs::PermissionsExt;
602 let dir = tempdir().unwrap();
603 let path = dir.path().join("config.toml");
604 let cfg = Config {
605 auth_token: Some("super-secret-token".into()),
606 registration_secret: Some("reg-secret".into()),
607 ..Config::default()
608 };
609 save(&cfg, &path).unwrap();
610 let mode = std::fs::metadata(&path).unwrap().permissions().mode();
611 assert_eq!(
612 mode & 0o077,
613 0,
614 "secrets-bearing config must not be group/world-accessible; got mode {mode:o}"
615 );
616 }
617
618 #[test]
619 fn save_atomically_replaces_existing_config_without_temp_litter() {
620 let dir = tempdir().unwrap();
624 let path = dir.path().join("config.toml");
625
626 let big = Config {
627 api_base_url: "https://a-very-long-host-name.example.invalid/studio/".into(),
628 worker_id: Some("worker-with-a-longish-id-000000".into()),
629 ..Config::default()
630 };
631 save(&big, &path).unwrap();
632
633 let small = Config {
634 api_base_url: "https://x/".into(),
635 ..Config::default()
636 };
637 save(&small, &path).unwrap();
638
639 let (loaded, _) = load(Some(&path.to_string_lossy())).unwrap();
640 assert_eq!(loaded.api_base_url, "https://x/");
641 assert!(
642 loaded.worker_id.is_none(),
643 "a replacing save must not leave the previous worker_id behind"
644 );
645
646 let names: Vec<String> = std::fs::read_dir(dir.path())
647 .unwrap()
648 .map(|e| e.unwrap().file_name().to_string_lossy().to_string())
649 .collect();
650 assert_eq!(
651 names,
652 vec!["config.toml".to_string()],
653 "atomic save must leave only the target file, found: {names:?}"
654 );
655 }
656}