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 log::{debug, info, warn};
224use rand::RngExt;
225use serde::{Deserialize, Serialize};
226use tauri::{AppHandle, Manager, command};
227// Type aliases for Configuration DTOs to simplify usage
228use CommonLibrary::Configuration::DTO::{
229	ConfigurationOverridesDTO as ConfigurationOverridesDTOModule,
230	ConfigurationTarget as ConfigurationTargetModule,
231};
232type ConfigurationOverridesDTO = ConfigurationOverridesDTOModule::ConfigurationOverridesDTO;
233type ConfigurationTarget = ConfigurationTargetModule::ConfigurationTarget;
234
235use CommonLibrary::{Configuration::ConfigurationProvider::ConfigurationProvider, Environment::Requires::Requires};
236use sha2::Digest;
237
238use crate::{
239	IPC::WindServiceAdapters::{WindDesktopConfiguration, WindServiceAdapter},
240	RunTime::ApplicationRunTime::ApplicationRunTime,
241};
242
243/// Configuration bridge that handles Wind's desktop configuration needs
244pub struct ConfigurationBridge {
245	runtime:Arc<ApplicationRunTime>,
246}
247
248impl ConfigurationBridge {
249	/// Create a new configuration bridge
250	pub fn new(runtime:Arc<ApplicationRunTime>) -> Self {
251		info!("[ConfigurationBridge] Creating configuration bridge");
252		Self { runtime }
253	}
254
255	/// Get Wind-compatible desktop configuration
256	pub async fn get_wind_desktop_configuration(&self) -> Result<WindDesktopConfiguration, String> {
257		debug!("[ConfigurationBridge] Getting Wind desktop configuration");
258
259		// Get the current Mountain configuration
260		let mountain_config = self.get_mountain_configuration().await?;
261
262		// Convert to Wind format using the service adapter
263		let service_adapter = WindServiceAdapter::new(self.runtime.clone());
264		let wind_config = service_adapter.convert_to_wind_configuration(mountain_config).await?;
265
266		debug!("[ConfigurationBridge] Wind configuration ready");
267		Ok(wind_config)
268	}
269
270	/// Update configuration from Wind frontend
271	pub async fn update_configuration_from_wind(&self, wind_config:WindDesktopConfiguration) -> Result<(), String> {
272		debug!("[ConfigurationBridge] Updating configuration from Wind");
273
274		// Convert Wind configuration to Mountain format
275		let mountain_config = self.convert_to_mountain_configuration(wind_config).await?;
276
277		// Update Mountain's configuration system
278		self.update_mountain_configuration(mountain_config).await?;
279
280		debug!("[ConfigurationBridge] Configuration updated successfully");
281		Ok(())
282	}
283
284	/// Get Mountain's current configuration
285	async fn get_mountain_configuration(&self) -> Result<serde_json::Value, String> {
286		debug!("[ConfigurationBridge] Getting Mountain configuration");
287
288		let config_provider:Arc<dyn ConfigurationProvider> = self.runtime.Environment.Require();
289
290		let config = config_provider
291			.GetConfigurationValue(None, ConfigurationOverridesDTO::default())
292			.await
293			.map_err(|e| format!("Failed to get Mountain configuration: {}", e))?;
294
295		Ok(config)
296	}
297
298	/// Update Mountain's configuration
299	async fn update_mountain_configuration(&self, config:serde_json::Value) -> Result<(), String> {
300		debug!("[ConfigurationBridge] Updating Mountain configuration");
301
302		// Validate configuration before updating
303		if !self.validate_configuration(&config) {
304			return Err("Invalid configuration data".to_string());
305		}
306
307		let config_provider:Arc<dyn ConfigurationProvider> = self.runtime.Environment.Require();
308
309		// Update configuration values
310		if let Some(obj) = config.as_object() {
311			for (key, value) in obj {
312				config_provider
313					.UpdateConfigurationValue(
314						key.clone(),
315						value.clone(),
316						ConfigurationTarget::User,
317						ConfigurationOverridesDTO::default(),
318						None,
319					)
320					.await
321					.map_err(|e| format!("Failed to update configuration key {}: {}", key, e))?;
322			}
323		}
324
325		Ok(())
326	}
327
328	/// Validate configuration data
329	fn validate_configuration(&self, config:&serde_json::Value) -> bool {
330		// Basic validation: config must be an object
331		if !config.is_object() {
332			return false;
333		}
334
335		// Validate individual configuration values
336		if let Some(obj) = config.as_object() {
337			for (key, value) in obj {
338				// Key validation
339				if key.trim().is_empty() {
340					return false;
341				}
342
343				// Value type validation
344				match key.as_str() {
345					"zoom_level" | "font_size" => {
346						if let Some(num) = value.as_f64() {
347							if key == "zoom_level" && (num < -8.0 || num > 9.0) {
348								return false;
349							}
350							if key == "font_size" && (num < 6.0 || num > 100.0) {
351								return false;
352							}
353						} else {
354							return false;
355						}
356					},
357					"is_packaged" | "enable_feature" => {
358						if !value.is_boolean() {
359							return false;
360						}
361					},
362					"theme" | "platform" | "arch" => {
363						if !value.is_string() || value.as_str().unwrap().trim().is_empty() {
364							return false;
365						}
366					},
367					_ => {
368						// Default validation: value must not be null
369						if value.is_null() {
370							return false;
371						}
372					},
373				}
374			}
375		}
376
377		true
378	}
379
380	/// Convert Wind configuration to Mountain format
381	async fn convert_to_mountain_configuration(
382		&self,
383		wind_config:WindDesktopConfiguration,
384	) -> Result<serde_json::Value, String> {
385		debug!("[ConfigurationBridge] Converting Wind config to Mountain format");
386
387		let machine_id = self.generate_machine_id().await.unwrap_or_else(|e| {
388			warn!("[ConfigurationBridge] Failed to generate machine ID: {}", e);
389			"wind-machine-fallback".to_string()
390		});
391
392		let session_id = self.generate_session_id().await.unwrap_or_else(|e| {
393			warn!("[ConfigurationBridge] Failed to generate session ID: {}", e);
394			"wind-session-fallback".to_string()
395		});
396
397		let mountain_config = serde_json::json!({
398			"window_id": wind_config.window_id.to_string(),
399			"machine_id": machine_id,
400			"session_id": session_id,
401			"log_level": wind_config.log_level,
402			"app_root": wind_config.app_root,
403			"user_data_dir": wind_config.user_data_path,
404			"tmp_dir": wind_config.temp_path,
405			"platform": wind_config.platform,
406			"arch": wind_config.arch,
407			"zoom_level": wind_config.zoom_level.unwrap_or(0.0),
408			"backup_path": wind_config.backup_path.unwrap_or_default(),
409			"home_dir": wind_config.profiles.home,
410			"is_packaged": wind_config.is_packaged,
411		});
412
413		Ok(mountain_config)
414	}
415
416	/// Synchronize configuration between Mountain and Wind
417	pub async fn synchronize_configuration(&self) -> Result<(), String> {
418		debug!("[ConfigurationBridge] Synchronizing configuration");
419
420		// Get Mountain's current configuration
421		let mountain_config = self.get_mountain_configuration().await?;
422
423		// Convert to Wind format
424		let service_adapter = WindServiceAdapter::new(self.runtime.clone());
425		let wind_config = service_adapter.convert_to_wind_configuration(mountain_config).await?;
426
427		// Send configuration to Wind via IPC
428		self.send_configuration_to_wind(wind_config).await?;
429
430		debug!("[ConfigurationBridge] Configuration synchronized");
431		Ok(())
432	}
433
434	/// Send configuration to Wind frontend via IPC
435	async fn send_configuration_to_wind(&self, config:WindDesktopConfiguration) -> Result<(), String> {
436		debug!("[ConfigurationBridge] Sending configuration to Wind");
437
438		// Get the IPC server
439		if let Some(ipc_server) = self
440			.runtime
441			.Environment
442			.ApplicationHandle
443			.try_state::<crate::IPC::TauriIPCServer::TauriIPCServer>()
444		{
445			let config_json =
446				serde_json::to_value(config).map_err(|e| format!("Failed to serialize configuration: {}", e))?;
447
448			ipc_server
449				.send("configuration:update", config_json)
450				.await
451				.map_err(|e| format!("Failed to send configuration to Wind: {}", e))?;
452		} else {
453			return Err("IPC Server not found".to_string());
454		}
455
456		Ok(())
457	}
458
459	/// Handle configuration changes from Wind
460	pub async fn handle_wind_configuration_change(&self, new_config:serde_json::Value) -> Result<(), String> {
461		debug!("[ConfigurationBridge] Handling Wind configuration change");
462
463		// Parse Wind configuration
464		let wind_config:WindDesktopConfiguration =
465			serde_json::from_value(new_config).map_err(|e| format!("Failed to parse Wind configuration: {}", e))?;
466
467		// Update Mountain configuration
468		self.update_configuration_from_wind(wind_config).await?;
469
470		debug!("[ConfigurationBridge] Wind configuration change handled");
471		Ok(())
472	}
473
474	/// Get configuration status
475	pub async fn get_configuration_status(&self) -> Result<ConfigurationStatus, String> {
476		debug!("[ConfigurationBridge] Getting configuration status");
477
478		let mountain_config = self.get_mountain_configuration().await?;
479		let is_valid = !mountain_config.is_null();
480
481		let status = ConfigurationStatus {
482			is_valid,
483			last_sync:std::time::SystemTime::now()
484				.duration_since(std::time::UNIX_EPOCH)
485				.unwrap_or_default()
486				.as_millis() as u64,
487			configuration_keys:if let Some(obj) = mountain_config.as_object() {
488				obj.keys().map(|k| k.clone()).collect()
489			} else {
490				Vec::new()
491			},
492		};
493
494		Ok(status)
495	}
496
497	/// Generate unique machine ID using advanced Microsoft-inspired patterns
498	async fn generate_machine_id(&self) -> Result<String, String> {
499		// IMPLEMENTATION: Multi-platform machine ID generation
500		#[cfg(target_os = "macos")]
501		{
502			use std::process::Command;
503
504			// Get macOS serial number using system_profiler
505			let result = Command::new("system_profiler")
506				.arg("SPHardwareDataType")
507				.arg("-json")
508				.output()
509				.map_err(|e| format!("Failed to execute system_profiler: {}", e))?;
510
511			if result.status.success() {
512				let output_str = String::from_utf8_lossy(&result.stdout);
513				if let Ok(json) = serde_json::from_str::<serde_json::Value>(&output_str) {
514					if let Some(serial) = json["SPHardwareDataType"][0]["serial_number"].as_str() {
515						return Ok(format!("mac-{}", serial));
516					}
517				}
518			}
519		}
520
521		#[cfg(target_os = "windows")]
522		{
523			use std::process::Command;
524
525			// Get Windows machine ID using wmic
526			let result = Command::new("wmic")
527				.arg("csproduct")
528				.arg("get")
529				.arg("UUID")
530				.output()
531				.map_err(|e| format!("Failed to execute wmic: {}", e))?;
532
533			if result.status.success() {
534				let output_str = String::from_utf8_lossy(&result.stdout);
535				let lines:Vec<&str> = output_str.lines().collect();
536				if lines.len() > 1 {
537					let uuid = lines[1].trim();
538					if !uuid.is_empty() {
539						return Ok(format!("win-{}", uuid));
540					}
541				}
542			}
543		}
544
545		#[cfg(target_os = "linux")]
546		{
547			use std::fs;
548
549			// Try to read machine-id from /etc/machine-id
550			if let Ok(content) = fs::read_to_string("/etc/machine-id") {
551				let machine_id = content.trim();
552				if !machine_id.is_empty() {
553					return Ok(format!("linux-{}", machine_id));
554				}
555			}
556
557			// Fallback to /var/lib/dbus/machine-id
558			if let Ok(content) = fs::read_to_string("/var/lib/dbus/machine-id") {
559				let machine_id = content.trim();
560				if !machine_id.is_empty() {
561					return Ok(format!("linux-{}", machine_id));
562				}
563			}
564		}
565
566		// Fallback: Generate a unique ID based on hostname and timestamp
567		let hostname = hostname::get()
568			.map_err(|e| format!("Failed to get hostname: {}", e))?
569			.to_string_lossy()
570			.to_string();
571
572		let timestamp = std::time::SystemTime::now()
573			.duration_since(std::time::UNIX_EPOCH)
574			.unwrap_or_default()
575			.as_millis();
576
577		Ok(format!("fallback-{}-{}", hostname, timestamp))
578	}
579
580	/// Generate unique session ID with Microsoft-inspired security patterns
581	async fn generate_session_id(&self) -> Result<String, String> {
582		use std::time::{SystemTime, UNIX_EPOCH};
583
584		use rand::{Rng, rng};
585
586		// IMPLEMENTATION: Secure session ID generation
587		let mut rng = rng();
588		let random_part:u64 = rng.random_range(0..=u64::MAX);
589
590		let timestamp = SystemTime::now().duration_since(UNIX_EPOCH).unwrap_or_default().as_millis();
591
592		// Get process ID for additional uniqueness
593		let process_id = std::process::id();
594
595		// Create a hash-based session ID
596		let session_data = format!("{}:{}:{}", timestamp, random_part, process_id);
597		let mut hasher = sha2::Sha256::new();
598		hasher.update(session_data.as_bytes());
599		let result = hasher.finalize();
600
601		// Convert to hex string and take first 16 characters
602		let hex_string = format!("{:x}", result);
603		let session_id = hex_string.chars().take(16).collect::<String>();
604
605		info!("[ConfigurationBridge] Generated session ID: {}", session_id);
606		Ok(format!("session-{}", session_id))
607	}
608}
609
610/// Configuration status structure
611#[derive(Debug, Clone, Serialize, Deserialize)]
612pub struct ConfigurationStatus {
613	pub is_valid:bool,
614	pub last_sync:u64,
615	pub configuration_keys:Vec<String>,
616}
617
618/// Tauri command to get Wind desktop configuration
619#[tauri::command]
620pub async fn mountain_get_wind_desktop_configuration(
621	app_handle:tauri::AppHandle,
622) -> Result<WindDesktopConfiguration, String> {
623	debug!("[ConfigurationBridge] Tauri command: get_wind_desktop_configuration");
624
625	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
626		let bridge = ConfigurationBridge::new(runtime.inner().clone());
627		bridge.get_wind_desktop_configuration().await
628	} else {
629		Err("ApplicationRunTime not found".to_string())
630	}
631}
632
633/// Tauri command to get configuration data for Wind frontend
634#[tauri::command]
635pub async fn get_configuration_data(app_handle:tauri::AppHandle) -> Result<serde_json::Value, String> {
636	debug!("[ConfigurationBridge] Tauri command: get_configuration_data");
637
638	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
639		let bridge = ConfigurationBridge::new(runtime.inner().clone());
640
641		// Get Mountain's current configuration
642		let mountain_config = bridge.get_mountain_configuration().await?;
643
644		// Convert to Wind format
645		let config_data = serde_json::json!({
646			"application": mountain_config.clone(),
647			"workspace": mountain_config.clone(),
648			"profile": mountain_config.clone()
649		});
650
651		debug!("[ConfigurationBridge] Configuration data retrieved successfully");
652		Ok(config_data)
653	} else {
654		Err("ApplicationRunTime not found".to_string())
655	}
656}
657
658/// Tauri command to save configuration data from Wind frontend
659#[tauri::command]
660pub async fn save_configuration_data(app_handle:tauri::AppHandle, config_data:serde_json::Value) -> Result<(), String> {
661	debug!("[ConfigurationBridge] Tauri command: save_configuration_data");
662
663	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
664		let bridge = ConfigurationBridge::new(runtime.inner().clone());
665
666		// Update Mountain configuration with the new data
667		bridge.update_mountain_configuration(config_data).await?;
668
669		debug!("[ConfigurationBridge] Configuration data saved successfully");
670		Ok(())
671	} else {
672		Err("ApplicationRunTime not found".to_string())
673	}
674}
675
676/// Tauri command to update configuration from Wind
677#[tauri::command]
678pub async fn mountain_update_configuration_from_wind(
679	app_handle:tauri::AppHandle,
680	config:serde_json::Value,
681) -> Result<(), String> {
682	debug!("[ConfigurationBridge] Tauri command: update_configuration_from_wind");
683
684	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
685		let bridge = ConfigurationBridge::new(runtime.inner().clone());
686		bridge.handle_wind_configuration_change(config).await
687	} else {
688		Err("ApplicationRunTime not found".to_string())
689	}
690}
691
692/// Tauri command to synchronize configuration
693#[tauri::command]
694pub async fn mountain_synchronize_configuration(app_handle:tauri::AppHandle) -> Result<serde_json::Value, String> {
695	debug!("[ConfigurationBridge] Tauri command: synchronize_configuration");
696
697	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
698		let bridge = ConfigurationBridge::new(runtime.inner().clone());
699		bridge
700			.synchronize_configuration()
701			.await
702			.map(|_| serde_json::json!({ "status": "success" }))
703	} else {
704		Err("ApplicationRunTime not found".to_string())
705	}
706}
707
708/// Tauri command to get configuration status
709#[tauri::command]
710pub async fn mountain_get_configuration_status(app_handle:tauri::AppHandle) -> Result<serde_json::Value, String> {
711	debug!("[ConfigurationBridge] Tauri command: get_configuration_status");
712
713	if let Some(runtime) = app_handle.try_state::<Arc<ApplicationRunTime>>() {
714		let bridge = ConfigurationBridge::new(runtime.inner().clone());
715		bridge
716			.get_configuration_status()
717			.await
718			.map(|status| serde_json::to_value(status).unwrap_or(serde_json::Value::Null))
719	} else {
720		Err("ApplicationRunTime not found".to_string())
721	}
722}