Skip to main content

Mountain/IPC/
ConfigurationBridge.rs

1//! # Configuration Bridge - Bidirectional Configuration Synchronization
2//!
3//! **File Responsibilities:**
4//! This module manages bidirectional synchronization of configuration between
5//! Mountain's Rust backend and Wind's TypeScript frontend. It ensures
6//! configuration consistency across the entire CodeEditorLand ecosystem while
7//! handling conflicts and updates gracefully.
8//!
9//! **Architectural Role in Wind-Mountain Connection:**
10//!
11//! The ConfigurationBridge is the synchronization layer that:
12//!
13//! 1. **Translates Configuration Formats:** Converts between Mountain's
14//!    internal config structure and Wind's desktop configuration interface
15//! 2. **Bidirectional Sync:** Maintains consistency in both directions
16//!    (Wind→Mountain and Mountain→Wind)
17//! 3. **Conflict Resolution:** Handles merge conflicts when multiple sources
18//!    update configuration simultaneously
19//! 4. **Validation:** Ensures all configuration changes are valid before
20//!    applying
21//! 5. **Identity Management:** Generates unique machine and session IDs for
22//!    multi- instance scenarios
23//!
24//! **Bidirectional Synchronization Flow:**
25//!
26//! **Mountain → Wind Sync:**
27//! ```
28//! Mountain Services (Internal Config)
29//!       |
30//!       | get_mountain_configuration()
31//!       v
32//! ConfigurationBridge
33//!       |
34//!       | WindServiceAdapter.convert_to_wind_configuration()
35//!       v
36//! Wind Desktop Configuration Format
37//!       |
38//!       | send_configuration_to_wind()
39//!       v
40//! Wind Frontend (via IPC)
41//! ```
42//!
43//! **Wind → Mountain Sync:**
44//! ```
45//! Wind Frontend (User Changes)
46//!       |
47//!       | handle_wind_configuration_change()
48//!       v
49//! ConfigurationBridge
50//!       |
51//!       | convert_to_mountain_configuration()
52//!       v
53//! Mountain Configuration Format
54//!       |
55//!       | update_mountain_configuration()
56//!       v
57//! Mountain Services (Internal Config)
58//! ```
59//!
60//! **Configuration Bridge Features:**
61//!
62//! **1. Format Translation:**
63//! - Mountain's internal JSON structure → Wind's desktop configuration
64//!   interface
65//! - Handles nested configuration objects
66//! - Type conversion between TypeScript and Rust types
67//!
68//! **2. Conflict Resolution Strategy:**
69//!
70//! **Current Implementation (Basic):**
71//! - Last-write-wins (most recent update takes precedence)
72//! - Configuration is validated before applying
73//! - Invalid changes are rejected entirely
74//!
75//! **Advanced Conflict Resolution (Future Enhancement):**
76//! - Detect conflicts based on modification timestamps
77//! - Provide conflict metadata (source, timestamp, value)
78//! - Support three-way merge strategies:
79//!   - **Ours:** Keep Mountain's version
80//!   - **Theirs:** Use Wind's version
81//!   - **Merge:** Attempt intelligent merge
82//! - Conflict UI prompts in Wind for user resolution
83//!
84//! **3. Validation Rules:**
85//!
86//! **Type Validation:**
87//! - `zoom_level`: Number, range -8.0 to 9.0
88//! - `font_size`: Number, range 6.0 to 100.0
89//! - `is_packaged`: Boolean
90//! - `theme`, `platform`, `arch`: String, non-empty
91//! - All other values: Not null
92//!
93//! **Key Validation:**
94//! - Configuration keys must not be empty or whitespace
95//! - Reserved keys cannot be modified
96//! - Nested paths use dot notation (e.g., "editor.theme")
97//!
98//! **Value Validation:**
99//! - Ranges checked for numeric values
100//! - Enum validation for predefined options
101//! - Pattern validation for string values (URLs, paths)
102//!
103//! **4. Identity Management:**
104//!
105//! **Machine ID Generation (Microsoft-Inspired):**
106//! - **macOS:** Get system serial number via `system_profiler`
107//! - **Windows:** Get machine UUID via `wmic csproduct get UUID`
108//! - **Linux:** Read from `/etc/machine-id` or `/var/lib/dbus/machine-id`
109//! - **Fallback:** Hash hostname + timestamp
110//!
111//! **Session ID Generation (Secure):**
112//! - Combine timestamp, random number, and process ID
113//! - Hash with SHA-256
114//! - Use first 16 characters of hex digest
115//! - Format: `session-{16-char-hash}`
116//!
117//! **5. Bidirectional Sync Triggers:**
118//!
119//! **Triggers for Mountain → Wind:**
120//! - Configuration changes from Mountain services
121//! - Periodic sync interval (configurable)
122//! - Manual sync request from Mountain
123//!
124//! **Triggers for Wind → Mountain:**
125//! - User changes configuration in Wind UI
126//! - Settings panel updates
127//! - Extension configuration changes
128//! - Command palette configuration commands
129//!
130//! **Key Structures:**
131//!
132//! **ConfigurationBridge:**
133//! Main synchronization orchestrator
134//! - `get_wind_desktop_configuration()` - Get config in Wind format
135//! - `update_configuration_from_wind()` - Apply Wind's config changes
136//! - `synchronize_configuration()` - Force bidirectional sync
137//! - `get_configuration_status()` - Get sync status info
138//!
139//! **ConfigurationStatus:**
140//! Current synchronization state
141//! - `is_valid` - Whether configuration is valid
142//! - `last_sync` - Timestamp of last successful sync
143//! - `configuration_keys` - List of all configuration keys
144//!
145//! **Tauri Commands:**
146//!
147//! The module provides Tauri commands for Wind to invoke:
148//!
149//! - `mountain_get_wind_desktop_configuration` - Get config for Wind UI
150//! - `get_configuration_data` - Get all configuration data
151//! - `save_configuration_data` - Save configuration from Wind
152//! - `mountain_update_configuration_from_wind` - Update config from Wind
153//! - `mountain_synchronize_configuration` - Force sync
154//! - `mountain_get_configuration_status` - Get sync status
155//!
156//! **Configuration Flow Examples:**
157//!
158//! **Example 1: Wind Initializing**
159//! ```typescript
160//! // Wind startup
161//! const config = await invoke('mountain_get_wind_desktop_configuration');
162//! applyConfiguration(config);
163//! ```
164//!
165//! **Example 2: User Changes Theme**
166//! ```typescript
167//! // User changes theme in Wind UI
168//! const newConfig = { theme: 'dark', 'editor.fontSize': 14 };
169//! await invoke('save_configuration_data', newConfig);
170//! ```
171//!
172//! **Example 3: Mountain Updates Setting**
173//! ```text
174//! // Mountain service updates configuration
175//! let bridge = ConfigurationBridge::new(runtime);
176//! bridge.synchronize_configuration().await?;
177//!
178//! // Result: Wind UI automatically updates via IPC event
179//! ```
180//!
181//! **Error Handling Strategy:**
182//!
183//! **Configuration Validation Errors:**
184//! - Reject entire invalid configuration
185//! - Return detailed validation error messages
186//! - List which keys/values failed validation
187//!
188//! **Format Conversion Errors:**
189//! - Log conversion errors with field names
190//! - Attempt graceful fallback for missing fields
191//! - Use defaults for conversion failures
192//!
193//! **Sync Errors:**
194//! - Log sync failures with timestamps
195//! - Queue sync for retry on transient errors
196//! - Alert monitoring system on persistent failures
197//!
198//! **Integration with Other Modules:**
199//!
200//! **WindServiceAdapters:**
201//! - Uses `WindServiceAdapter.convert_to_wind_configuration()`
202//! - Depends on `WindDesktopConfiguration` structure
203//!
204//! **TauriIPCServer:**
205//! - Sends configuration updates via IPC events
206//! - Receives configuration changes from Wind
207//!
208//! **Mountain Configuration Service:**
209//! - Delegates to `ConfigurationProvider` trait
210//! - Uses `ConfigurationTarget` for scoping
211//!
212//! **Best Practices:**
213//!
214//! 1. **Always Validate:** Never apply configuration without validation
215//! 2. **Atomic Updates:** Apply entire configuration atomically
216//! 3. **Versioning:** Consider adding configuration versioning
217//! 4. **Change Logging:** Log all configuration changes for audit
218//! 5. **Fallback Support:** Provide sensible defaults for all settings
219//! 6. **Conflict Detection:** Implement proper conflict detection before merges
220
221use std::sync::Arc;
222
223use serde::{Deserialize, Serialize};
224use tauri::Manager;
225// Type aliases for Configuration DTOs to simplify usage
226use CommonLibrary::Configuration::DTO::{
227	ConfigurationOverridesDTO as ConfigurationOverridesDTOModule,
228	ConfigurationTarget as ConfigurationTargetModule,
229};
230type ConfigurationOverridesDTO = ConfigurationOverridesDTOModule::ConfigurationOverridesDTO;
231type ConfigurationTarget = ConfigurationTargetModule::ConfigurationTarget;
232
233use CommonLibrary::{Configuration::ConfigurationProvider::ConfigurationProvider, Environment::Requires::Requires};
234use sha2::Digest;
235
236use crate::{
237	IPC::WindServiceAdapters::{WindDesktopConfiguration, WindServiceAdapter},
238	RunTime::ApplicationRunTime::ApplicationRunTime,
239	dev_log,
240};
241
242/// Configuration bridge that handles Wind's desktop configuration needs
243pub struct ConfigurationBridge {
244	runtime:Arc<ApplicationRunTime>,
245}
246
247impl ConfigurationBridge {
248	/// Create a new configuration bridge
249	pub fn new(runtime:Arc<ApplicationRunTime>) -> Self {
250		dev_log!("config", "[ConfigurationBridge] Creating configuration bridge");
251		Self { runtime }
252	}
253
254	/// Get Wind-compatible desktop configuration
255	pub async fn get_wind_desktop_configuration(&self) -> Result<WindDesktopConfiguration, String> {
256		dev_log!("config", "[ConfigurationBridge] Getting Wind desktop configuration");
257
258		// Get the current Mountain configuration
259		let mountain_config = self.get_mountain_configuration().await?;
260
261		// Convert to Wind format using the service adapter
262		let service_adapter = WindServiceAdapter::new(self.runtime.clone());
263		let wind_config = service_adapter.convert_to_wind_configuration(mountain_config).await?;
264
265		dev_log!("config", "[ConfigurationBridge] Wind configuration ready");
266		Ok(wind_config)
267	}
268
269	/// Update configuration from Wind frontend
270	pub async fn update_configuration_from_wind(&self, wind_config:WindDesktopConfiguration) -> Result<(), String> {
271		dev_log!("config", "[ConfigurationBridge] Updating configuration from Wind");
272
273		// Convert Wind configuration to Mountain format
274		let mountain_config = self.convert_to_mountain_configuration(wind_config).await?;
275
276		// Update Mountain's configuration system
277		self.update_mountain_configuration(mountain_config).await?;
278
279		dev_log!("config", "[ConfigurationBridge] Configuration updated successfully");
280		Ok(())
281	}
282
283	/// Get Mountain's current configuration
284	async fn get_mountain_configuration(&self) -> Result<serde_json::Value, String> {
285		dev_log!("config", "[ConfigurationBridge] Getting Mountain configuration");
286
287		let config_provider:Arc<dyn ConfigurationProvider> = self.runtime.Environment.Require();
288
289		let config = config_provider
290			.GetConfigurationValue(None, ConfigurationOverridesDTO::default())
291			.await
292			.map_err(|e| format!("Failed to get Mountain configuration: {}", e))?;
293
294		Ok(config)
295	}
296
297	/// Update Mountain's configuration
298	async fn update_mountain_configuration(&self, config:serde_json::Value) -> Result<(), String> {
299		dev_log!("config", "[ConfigurationBridge] Updating Mountain configuration");
300
301		// Validate configuration before updating
302		if !self.validate_configuration(&config) {
303			return Err("Invalid configuration data".to_string());
304		}
305
306		let config_provider:Arc<dyn ConfigurationProvider> = self.runtime.Environment.Require();
307
308		// Update configuration values
309		if let Some(obj) = config.as_object() {
310			for (key, value) in obj {
311				config_provider
312					.UpdateConfigurationValue(
313						key.clone(),
314						value.clone(),
315						ConfigurationTarget::User,
316						ConfigurationOverridesDTO::default(),
317						None,
318					)
319					.await
320					.map_err(|e| format!("Failed to update configuration key {}: {}", key, e))?;
321			}
322		}
323
324		Ok(())
325	}
326
327	/// Validate configuration data
328	fn validate_configuration(&self, config:&serde_json::Value) -> bool {
329		// Basic validation: config must be an object
330		if !config.is_object() {
331			return false;
332		}
333
334		// Validate individual configuration values
335		if let Some(obj) = config.as_object() {
336			for (key, value) in obj {
337				// Key validation
338				if key.trim().is_empty() {
339					return false;
340				}
341
342				// Value type validation
343				match key.as_str() {
344					"zoom_level" | "font_size" => {
345						if let Some(num) = value.as_f64() {
346							if key == "zoom_level" && (num < -8.0 || num > 9.0) {
347								return false;
348							}
349							if key == "font_size" && (num < 6.0 || num > 100.0) {
350								return false;
351							}
352						} else {
353							return false;
354						}
355					},
356					"is_packaged" | "enable_feature" => {
357						if !value.is_boolean() {
358							return false;
359						}
360					},
361					"theme" | "platform" | "arch" => {
362						if !value.is_string() || value.as_str().unwrap().trim().is_empty() {
363							return false;
364						}
365					},
366					_ => {
367						// Default validation: value must not be null
368						if value.is_null() {
369							return false;
370						}
371					},
372				}
373			}
374		}
375
376		true
377	}
378
379	/// Convert Wind configuration to Mountain format
380	async fn convert_to_mountain_configuration(
381		&self,
382		wind_config:WindDesktopConfiguration,
383	) -> Result<serde_json::Value, String> {
384		dev_log!("config", "[ConfigurationBridge] Converting Wind config to Mountain format");
385
386		let machine_id = self.generate_machine_id().await.unwrap_or_else(|e| {
387			dev_log!("config", "warn: [ConfigurationBridge] Failed to generate machine ID: {}", e);
388			"wind-machine-fallback".to_string()
389		});
390
391		let session_id = self.generate_session_id().await.unwrap_or_else(|e| {
392			dev_log!("config", "warn: [ConfigurationBridge] Failed to generate session ID: {}", e);
393			"wind-session-fallback".to_string()
394		});
395
396		let mountain_config = serde_json::json!({
397			"window_id": wind_config.window_id.to_string(),
398			"machine_id": machine_id,
399			"session_id": session_id,
400			"log_level": wind_config.log_level,
401			"app_root": wind_config.app_root,
402			"user_data_dir": wind_config.user_data_path,
403			"tmp_dir": wind_config.temp_path,
404			"platform": wind_config.platform,
405			"arch": wind_config.arch,
406			"zoom_level": wind_config.zoom_level.unwrap_or(0.0),
407			"backup_path": wind_config.backup_path.unwrap_or_default(),
408			"home_dir": wind_config.profiles.home,
409			"is_packaged": wind_config.is_packaged,
410		});
411
412		Ok(mountain_config)
413	}
414
415	/// Synchronize configuration between Mountain and Wind
416	pub async fn synchronize_configuration(&self) -> Result<(), String> {
417		dev_log!("config", "[ConfigurationBridge] Synchronizing configuration");
418
419		// Get Mountain's current configuration
420		let mountain_config = self.get_mountain_configuration().await?;
421
422		// Convert to Wind format
423		let service_adapter = WindServiceAdapter::new(self.runtime.clone());
424		let wind_config = service_adapter.convert_to_wind_configuration(mountain_config).await?;
425
426		// Send configuration to Wind via IPC
427		self.send_configuration_to_wind(wind_config).await?;
428
429		dev_log!("config", "[ConfigurationBridge] Configuration synchronized");
430		Ok(())
431	}
432
433	/// Send configuration to Wind frontend via IPC
434	async fn send_configuration_to_wind(&self, config:WindDesktopConfiguration) -> Result<(), String> {
435		dev_log!("config", "[ConfigurationBridge] Sending configuration to Wind");
436
437		// Get the IPC server
438		if let Some(ipc_server) = self
439			.runtime
440			.Environment
441			.ApplicationHandle
442			.try_state::<crate::IPC::TauriIPCServer::TauriIPCServer>()
443		{
444			let config_json =
445				serde_json::to_value(config).map_err(|e| format!("Failed to serialize configuration: {}", e))?;
446
447			ipc_server
448				.send("configuration:update", config_json)
449				.await
450				.map_err(|e| format!("Failed to send configuration to Wind: {}", e))?;
451		} else {
452			return Err("IPC Server not found".to_string());
453		}
454
455		Ok(())
456	}
457
458	/// Handle configuration changes from Wind
459	pub async fn handle_wind_configuration_change(&self, new_config:serde_json::Value) -> Result<(), String> {
460		dev_log!("config", "[ConfigurationBridge] Handling Wind configuration change");
461
462		// Parse Wind configuration
463		let wind_config:WindDesktopConfiguration =
464			serde_json::from_value(new_config).map_err(|e| format!("Failed to parse Wind configuration: {}", e))?;
465
466		// Update Mountain configuration
467		self.update_configuration_from_wind(wind_config).await?;
468
469		dev_log!("config", "[ConfigurationBridge] Wind configuration change handled");
470		Ok(())
471	}
472
473	/// Get configuration status
474	pub async fn get_configuration_status(&self) -> Result<ConfigurationStatus, String> {
475		dev_log!("config", "[ConfigurationBridge] Getting configuration status");
476
477		let mountain_config = self.get_mountain_configuration().await?;
478		let is_valid = !mountain_config.is_null();
479
480		let status = ConfigurationStatus {
481			is_valid,
482			last_sync:std::time::SystemTime::now()
483				.duration_since(std::time::UNIX_EPOCH)
484				.unwrap_or_default()
485				.as_millis() as u64,
486			configuration_keys:if let Some(obj) = mountain_config.as_object() {
487				obj.keys().map(|k| k.clone()).collect()
488			} else {
489				Vec::new()
490			},
491		};
492
493		Ok(status)
494	}
495
496	/// Generate unique machine ID using advanced Microsoft-inspired patterns
497	async fn generate_machine_id(&self) -> Result<String, String> {
498		// IMPLEMENTATION: Multi-platform machine ID generation
499		#[cfg(target_os = "macos")]
500		{
501			use std::process::Command;
502
503			// Get macOS serial number using system_profiler
504			let result = Command::new("system_profiler")
505				.arg("SPHardwareDataType")
506				.arg("-json")
507				.output()
508				.map_err(|e| format!("Failed to execute system_profiler: {}", e))?;
509
510			if result.status.success() {
511				let output_str = String::from_utf8_lossy(&result.stdout);
512				if let Ok(json) = serde_json::from_str::<serde_json::Value>(&output_str) {
513					if let Some(serial) = json["SPHardwareDataType"][0]["serial_number"].as_str() {
514						return Ok(format!("mac-{}", serial));
515					}
516				}
517			}
518		}
519
520		#[cfg(target_os = "windows")]
521		{
522			use std::process::Command;
523
524			// Get Windows machine ID using wmic
525			let result = Command::new("wmic")
526				.arg("csproduct")
527				.arg("get")
528				.arg("UUID")
529				.output()
530				.map_err(|e| format!("Failed to execute wmic: {}", e))?;
531
532			if result.status.success() {
533				let output_str = String::from_utf8_lossy(&result.stdout);
534				let lines:Vec<&str> = output_str.lines().collect();
535				if lines.len() > 1 {
536					let uuid = lines[1].trim();
537					if !uuid.is_empty() {
538						return Ok(format!("win-{}", uuid));
539					}
540				}
541			}
542		}
543
544		#[cfg(target_os = "linux")]
545		{
546			use std::fs;
547
548			// Try to read machine-id from /etc/machine-id
549			if let Ok(content) = fs::read_to_string("/etc/machine-id") {
550				let machine_id = content.trim();
551				if !machine_id.is_empty() {
552					return Ok(format!("linux-{}", machine_id));
553				}
554			}
555
556			// Fallback to /var/lib/dbus/machine-id
557			if let Ok(content) = fs::read_to_string("/var/lib/dbus/machine-id") {
558				let machine_id = content.trim();
559				if !machine_id.is_empty() {
560					return Ok(format!("linux-{}", machine_id));
561				}
562			}
563		}
564
565		// Fallback: Generate a unique ID based on hostname and timestamp
566		let hostname = hostname::get()
567			.map_err(|e| format!("Failed to get hostname: {}", e))?
568			.to_string_lossy()
569			.to_string();
570
571		let timestamp = std::time::SystemTime::now()
572			.duration_since(std::time::UNIX_EPOCH)
573			.unwrap_or_default()
574			.as_millis();
575
576		Ok(format!("fallback-{}-{}", hostname, timestamp))
577	}
578
579	/// Generate unique session ID with Microsoft-inspired security patterns
580	async fn generate_session_id(&self) -> Result<String, String> {
581		use std::time::{SystemTime, UNIX_EPOCH};
582
583		// IMPLEMENTATION: Secure session ID generation
584		let random_part:u64 = rand::random();
585
586		let timestamp = SystemTime::now().duration_since(UNIX_EPOCH).unwrap_or_default().as_millis();
587
588		// Get process ID for additional uniqueness
589		let process_id = std::process::id();
590
591		// Create a hash-based session ID
592		let session_data = format!("{}:{}:{}", timestamp, random_part, process_id);
593		let mut hasher = sha2::Sha256::new();
594		hasher.update(session_data.as_bytes());
595		let result = hasher.finalize();
596
597		// Convert to hex string and take first 16 characters
598		let hex_string = format!("{:x}", result);
599		let session_id = hex_string.chars().take(16).collect::<String>();
600
601		dev_log!("config", "[ConfigurationBridge] Generated session ID: {}", session_id);
602		Ok(format!("session-{}", session_id))
603	}
604}
605
606/// Configuration status structure
607#[derive(Debug, Clone, Serialize, Deserialize)]
608pub struct ConfigurationStatus {
609	pub is_valid:bool,
610	pub last_sync:u64,
611	pub configuration_keys:Vec<String>,
612}
613
614/// Tauri command to get Wind desktop configuration
615#[tauri::command]
616pub async fn mountain_get_wind_desktop_configuration(
617	app_handle:tauri::AppHandle,
618) -> Result<WindDesktopConfiguration, String> {
619	dev_log!("config", "[ConfigurationBridge] Tauri command: get_wind_desktop_configuration");
620
621	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
622		let bridge = ConfigurationBridge::new(runtime.inner().clone());
623		bridge.get_wind_desktop_configuration().await
624	} else {
625		Err("ApplicationRunTime not found".to_string())
626	}
627}
628
629/// Tauri command to get configuration data for Wind frontend
630#[tauri::command]
631pub async fn get_configuration_data(app_handle:tauri::AppHandle) -> Result<serde_json::Value, String> {
632	dev_log!("config", "[ConfigurationBridge] Tauri command: get_configuration_data");
633
634	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
635		let bridge = ConfigurationBridge::new(runtime.inner().clone());
636
637		// Get Mountain's current configuration
638		let mountain_config = bridge.get_mountain_configuration().await?;
639
640		// Convert to Wind format
641		let config_data = serde_json::json!({
642			"application": mountain_config.clone(),
643			"workspace": mountain_config.clone(),
644			"profile": mountain_config.clone()
645		});
646
647		dev_log!("config", "[ConfigurationBridge] Configuration data retrieved successfully");
648		Ok(config_data)
649	} else {
650		Err("ApplicationRunTime not found".to_string())
651	}
652}
653
654/// Tauri command to save configuration data from Wind frontend
655#[tauri::command]
656pub async fn save_configuration_data(app_handle:tauri::AppHandle, config_data:serde_json::Value) -> Result<(), String> {
657	dev_log!("config", "[ConfigurationBridge] Tauri command: save_configuration_data");
658
659	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
660		let bridge = ConfigurationBridge::new(runtime.inner().clone());
661
662		// Update Mountain configuration with the new data
663		bridge.update_mountain_configuration(config_data).await?;
664
665		dev_log!("config", "[ConfigurationBridge] Configuration data saved successfully");
666		Ok(())
667	} else {
668		Err("ApplicationRunTime not found".to_string())
669	}
670}
671
672/// Tauri command to update configuration from Wind
673#[tauri::command]
674pub async fn mountain_update_configuration_from_wind(
675	app_handle:tauri::AppHandle,
676	config:serde_json::Value,
677) -> Result<(), String> {
678	dev_log!("config", "[ConfigurationBridge] Tauri command: update_configuration_from_wind");
679
680	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
681		let bridge = ConfigurationBridge::new(runtime.inner().clone());
682		bridge.handle_wind_configuration_change(config).await
683	} else {
684		Err("ApplicationRunTime not found".to_string())
685	}
686}
687
688/// Tauri command to synchronize configuration
689#[tauri::command]
690pub async fn mountain_synchronize_configuration(app_handle:tauri::AppHandle) -> Result<serde_json::Value, String> {
691	dev_log!("config", "[ConfigurationBridge] Tauri command: synchronize_configuration");
692
693	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
694		let bridge = ConfigurationBridge::new(runtime.inner().clone());
695		bridge
696			.synchronize_configuration()
697			.await
698			.map(|_| serde_json::json!({ "status": "success" }))
699	} else {
700		Err("ApplicationRunTime not found".to_string())
701	}
702}
703
704/// Tauri command to get configuration status
705#[tauri::command]
706pub async fn mountain_get_configuration_status(app_handle:tauri::AppHandle) -> Result<serde_json::Value, String> {
707	dev_log!("config", "[ConfigurationBridge] Tauri command: get_configuration_status");
708
709	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
710		let bridge = ConfigurationBridge::new(runtime.inner().clone());
711		bridge
712			.get_configuration_status()
713			.await
714			.map(|status| serde_json::to_value(status).unwrap_or(serde_json::Value::Null))
715	} else {
716		Err("ApplicationRunTime not found".to_string())
717	}
718}