mirror of
https://github.com/zhom/donutbrowser.git
synced 2026-05-07 10:56:41 +02:00
858 lines
27 KiB
Rust
858 lines
27 KiB
Rust
use crate::browser_runner::{BrowserProfile, BrowserRunner};
|
|
use crate::browser_version_service::{BrowserVersionInfo, BrowserVersionService};
|
|
use crate::settings_manager::SettingsManager;
|
|
use serde::{Deserialize, Serialize};
|
|
use std::collections::{HashMap, HashSet};
|
|
use std::fs;
|
|
use std::path::PathBuf;
|
|
|
|
#[derive(Debug, Serialize, Deserialize, Clone)]
|
|
pub struct UpdateNotification {
|
|
pub id: String,
|
|
pub browser: String,
|
|
pub current_version: String,
|
|
pub new_version: String,
|
|
pub affected_profiles: Vec<String>,
|
|
pub is_stable_update: bool,
|
|
pub timestamp: u64,
|
|
}
|
|
|
|
#[derive(Debug, Serialize, Deserialize, Clone, Default)]
|
|
pub struct AutoUpdateState {
|
|
pub pending_updates: Vec<UpdateNotification>,
|
|
pub disabled_browsers: HashSet<String>, // browsers disabled during update
|
|
#[serde(default)]
|
|
pub auto_update_downloads: HashSet<String>, // track auto-update downloads for toast suppression
|
|
pub last_check_timestamp: u64,
|
|
}
|
|
|
|
pub struct AutoUpdater {
|
|
version_service: BrowserVersionService,
|
|
browser_runner: BrowserRunner,
|
|
settings_manager: SettingsManager,
|
|
}
|
|
|
|
impl AutoUpdater {
|
|
pub fn new() -> Self {
|
|
Self {
|
|
version_service: BrowserVersionService::new(),
|
|
browser_runner: BrowserRunner::new(),
|
|
settings_manager: SettingsManager::new(),
|
|
}
|
|
}
|
|
|
|
/// Check for updates for all profiles
|
|
pub async fn check_for_updates(
|
|
&self,
|
|
) -> Result<Vec<UpdateNotification>, Box<dyn std::error::Error + Send + Sync>> {
|
|
// Check if auto-updates are enabled
|
|
let settings = self
|
|
.settings_manager
|
|
.load_settings()
|
|
.map_err(|e| format!("Failed to load settings: {e}"))?;
|
|
if !settings.auto_updates_enabled {
|
|
return Ok(Vec::new());
|
|
}
|
|
|
|
let profiles = self
|
|
.browser_runner
|
|
.list_profiles()
|
|
.map_err(|e| format!("Failed to list profiles: {e}"))?;
|
|
let mut notifications = Vec::new();
|
|
let mut browser_versions: HashMap<String, Vec<BrowserVersionInfo>> = HashMap::new();
|
|
|
|
// Group profiles by browser type
|
|
let mut browser_profiles: HashMap<String, Vec<BrowserProfile>> = HashMap::new();
|
|
for profile in profiles {
|
|
browser_profiles
|
|
.entry(profile.browser.clone())
|
|
.or_default()
|
|
.push(profile);
|
|
}
|
|
|
|
// Check each browser type
|
|
for (browser, profiles) in browser_profiles {
|
|
// Get cached versions first, then try to fetch if needed
|
|
let versions = if let Some(cached) = self
|
|
.version_service
|
|
.get_cached_browser_versions_detailed(&browser)
|
|
{
|
|
cached
|
|
} else if self.version_service.should_update_cache(&browser) {
|
|
// Try to fetch fresh versions
|
|
match self
|
|
.version_service
|
|
.fetch_browser_versions_detailed(&browser, false)
|
|
.await
|
|
{
|
|
Ok(versions) => versions,
|
|
Err(_) => continue, // Skip this browser if fetch fails
|
|
}
|
|
} else {
|
|
continue; // No cached versions and cache doesn't need update
|
|
};
|
|
|
|
browser_versions.insert(browser.clone(), versions.clone());
|
|
|
|
// Check each profile for updates
|
|
for profile in profiles {
|
|
if let Some(update) = self.check_profile_update(&profile, &versions)? {
|
|
notifications.push(update);
|
|
}
|
|
}
|
|
}
|
|
|
|
Ok(notifications)
|
|
}
|
|
|
|
/// Check if a specific profile has an available update
|
|
fn check_profile_update(
|
|
&self,
|
|
profile: &BrowserProfile,
|
|
available_versions: &[BrowserVersionInfo],
|
|
) -> Result<Option<UpdateNotification>, Box<dyn std::error::Error + Send + Sync>> {
|
|
let current_version = &profile.version;
|
|
let is_current_stable = !self.is_alpha_version(current_version);
|
|
|
|
// Find the best available update
|
|
let best_update = available_versions
|
|
.iter()
|
|
.filter(|v| {
|
|
// Only consider versions newer than current
|
|
self.is_version_newer(&v.version, current_version) &&
|
|
// Respect version type preference
|
|
is_current_stable != v.is_prerelease
|
|
})
|
|
.max_by(|a, b| self.compare_versions(&a.version, &b.version));
|
|
|
|
if let Some(update_version) = best_update {
|
|
let notification = UpdateNotification {
|
|
id: format!(
|
|
"{}_{}_to_{}",
|
|
profile.browser, current_version, update_version.version
|
|
),
|
|
browser: profile.browser.clone(),
|
|
current_version: current_version.clone(),
|
|
new_version: update_version.version.clone(),
|
|
affected_profiles: vec![profile.name.clone()],
|
|
is_stable_update: !update_version.is_prerelease,
|
|
timestamp: std::time::SystemTime::now()
|
|
.duration_since(std::time::UNIX_EPOCH)
|
|
.unwrap()
|
|
.as_secs(),
|
|
};
|
|
Ok(Some(notification))
|
|
} else {
|
|
Ok(None)
|
|
}
|
|
}
|
|
|
|
/// Group update notifications by browser and version
|
|
pub fn group_update_notifications(
|
|
&self,
|
|
notifications: Vec<UpdateNotification>,
|
|
) -> Vec<UpdateNotification> {
|
|
let mut grouped: HashMap<String, UpdateNotification> = HashMap::new();
|
|
|
|
for notification in notifications {
|
|
let key = format!("{}_{}", notification.browser, notification.new_version);
|
|
|
|
if let Some(existing) = grouped.get_mut(&key) {
|
|
// Merge affected profiles
|
|
existing
|
|
.affected_profiles
|
|
.extend(notification.affected_profiles);
|
|
existing.affected_profiles.sort();
|
|
existing.affected_profiles.dedup();
|
|
} else {
|
|
grouped.insert(key, notification);
|
|
}
|
|
}
|
|
|
|
let mut result: Vec<UpdateNotification> = grouped.into_values().collect();
|
|
|
|
// Sort by priority: stable updates first, then by timestamp
|
|
result.sort_by(|a, b| match (a.is_stable_update, b.is_stable_update) {
|
|
(true, false) => std::cmp::Ordering::Less,
|
|
(false, true) => std::cmp::Ordering::Greater,
|
|
_ => b.timestamp.cmp(&a.timestamp),
|
|
});
|
|
|
|
result
|
|
}
|
|
|
|
/// Mark download as auto-update
|
|
pub fn mark_auto_update_download(
|
|
&self,
|
|
browser: &str,
|
|
version: &str,
|
|
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
|
let mut state = self.load_auto_update_state()?;
|
|
let download_key = format!("{browser}-{version}");
|
|
state.auto_update_downloads.insert(download_key);
|
|
self.save_auto_update_state(&state)?;
|
|
Ok(())
|
|
}
|
|
|
|
/// Remove auto-update download tracking
|
|
pub fn remove_auto_update_download(
|
|
&self,
|
|
browser: &str,
|
|
version: &str,
|
|
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
|
let mut state = self.load_auto_update_state()?;
|
|
let download_key = format!("{browser}-{version}");
|
|
state.auto_update_downloads.remove(&download_key);
|
|
self.save_auto_update_state(&state)?;
|
|
Ok(())
|
|
}
|
|
|
|
/// Check if download is marked as auto-update
|
|
pub fn is_auto_update_download(
|
|
&self,
|
|
browser: &str,
|
|
version: &str,
|
|
) -> Result<bool, Box<dyn std::error::Error + Send + Sync>> {
|
|
let state = self.load_auto_update_state()?;
|
|
let download_key = format!("{browser}-{version}");
|
|
Ok(state.auto_update_downloads.contains(&download_key))
|
|
}
|
|
|
|
/// Start browser update process
|
|
pub async fn start_browser_update(
|
|
&self,
|
|
browser: &str,
|
|
new_version: &str,
|
|
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
|
// Add browser to disabled list to prevent conflicts during update
|
|
let mut state = self.load_auto_update_state()?;
|
|
state.disabled_browsers.insert(browser.to_string());
|
|
|
|
// Mark this download as auto-update for toast suppression
|
|
let download_key = format!("{browser}-{new_version}");
|
|
state.auto_update_downloads.insert(download_key);
|
|
|
|
self.save_auto_update_state(&state)?;
|
|
|
|
// The actual download will be triggered by the frontend
|
|
// This function now just marks the browser as updating to prevent conflicts
|
|
Ok(())
|
|
}
|
|
|
|
/// Complete browser update process
|
|
pub async fn complete_browser_update(
|
|
&self,
|
|
browser: &str,
|
|
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
|
// Remove browser from disabled list
|
|
let mut state = self.load_auto_update_state()?;
|
|
state.disabled_browsers.remove(browser);
|
|
self.save_auto_update_state(&state)?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Automatically update all affected profile versions after browser download
|
|
pub async fn auto_update_profile_versions(
|
|
&self,
|
|
browser: &str,
|
|
new_version: &str,
|
|
) -> Result<Vec<String>, Box<dyn std::error::Error + Send + Sync>> {
|
|
let profiles = self
|
|
.browser_runner
|
|
.list_profiles()
|
|
.map_err(|e| format!("Failed to list profiles: {e}"))?;
|
|
|
|
let mut updated_profiles = Vec::new();
|
|
|
|
// Find all profiles for this browser that should be updated
|
|
for profile in profiles {
|
|
if profile.browser == browser {
|
|
// Check if profile is currently running
|
|
if profile.process_id.is_some() {
|
|
continue; // Skip running profiles
|
|
}
|
|
|
|
// Check if this is an update (newer version)
|
|
if self.is_version_newer(new_version, &profile.version) {
|
|
// Update the profile version
|
|
match self
|
|
.browser_runner
|
|
.update_profile_version(&profile.name, new_version)
|
|
{
|
|
Ok(_) => {
|
|
updated_profiles.push(profile.name);
|
|
}
|
|
Err(e) => {
|
|
eprintln!("Failed to update profile {}: {}", profile.name, e);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Ok(updated_profiles)
|
|
}
|
|
|
|
/// Complete browser update process with auto-update of profile versions
|
|
pub async fn complete_browser_update_with_auto_update(
|
|
&self,
|
|
browser: &str,
|
|
new_version: &str,
|
|
) -> Result<Vec<String>, Box<dyn std::error::Error + Send + Sync>> {
|
|
// Auto-update profile versions first
|
|
let updated_profiles = self
|
|
.auto_update_profile_versions(browser, new_version)
|
|
.await?;
|
|
|
|
// Remove browser from disabled list and clean up auto-update tracking
|
|
let mut state = self.load_auto_update_state()?;
|
|
state.disabled_browsers.remove(browser);
|
|
let download_key = format!("{browser}-{new_version}");
|
|
state.auto_update_downloads.remove(&download_key);
|
|
self.save_auto_update_state(&state)?;
|
|
|
|
Ok(updated_profiles)
|
|
}
|
|
|
|
/// Check if browser is disabled due to ongoing update
|
|
pub fn is_browser_disabled(
|
|
&self,
|
|
browser: &str,
|
|
) -> Result<bool, Box<dyn std::error::Error + Send + Sync>> {
|
|
let state = self.load_auto_update_state()?;
|
|
Ok(state.disabled_browsers.contains(browser))
|
|
}
|
|
|
|
/// Dismiss update notification
|
|
pub fn dismiss_update_notification(
|
|
&self,
|
|
notification_id: &str,
|
|
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
|
let mut state = self.load_auto_update_state()?;
|
|
state.pending_updates.retain(|n| n.id != notification_id);
|
|
self.save_auto_update_state(&state)?;
|
|
Ok(())
|
|
}
|
|
|
|
// Helper methods
|
|
|
|
fn is_alpha_version(&self, version: &str) -> bool {
|
|
version.contains("alpha")
|
|
|| version.contains("beta")
|
|
|| version.contains("rc")
|
|
|| version.contains("a")
|
|
|| version.contains("b")
|
|
|| version.contains("dev")
|
|
}
|
|
|
|
fn is_version_newer(&self, version1: &str, version2: &str) -> bool {
|
|
self.compare_versions(version1, version2) == std::cmp::Ordering::Greater
|
|
}
|
|
|
|
fn compare_versions(&self, version1: &str, version2: &str) -> std::cmp::Ordering {
|
|
// Basic semantic version comparison
|
|
let v1_parts = self.parse_version(version1);
|
|
let v2_parts = self.parse_version(version2);
|
|
|
|
v1_parts.cmp(&v2_parts)
|
|
}
|
|
|
|
fn parse_version(&self, version: &str) -> Vec<u32> {
|
|
version
|
|
.split(&['.', 'a', 'b', '-', '_'][..])
|
|
.filter_map(|part| part.parse::<u32>().ok())
|
|
.collect()
|
|
}
|
|
|
|
fn get_auto_update_state_file(&self) -> PathBuf {
|
|
self
|
|
.settings_manager
|
|
.get_settings_dir()
|
|
.join("auto_update_state.json")
|
|
}
|
|
|
|
fn load_auto_update_state(
|
|
&self,
|
|
) -> Result<AutoUpdateState, Box<dyn std::error::Error + Send + Sync>> {
|
|
let state_file = self.get_auto_update_state_file();
|
|
|
|
if !state_file.exists() {
|
|
return Ok(AutoUpdateState::default());
|
|
}
|
|
|
|
let content = fs::read_to_string(state_file)?;
|
|
let state: AutoUpdateState = serde_json::from_str(&content)?;
|
|
Ok(state)
|
|
}
|
|
|
|
fn save_auto_update_state(
|
|
&self,
|
|
state: &AutoUpdateState,
|
|
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
|
let settings_dir = self.settings_manager.get_settings_dir();
|
|
std::fs::create_dir_all(&settings_dir)?;
|
|
|
|
let state_file = self.get_auto_update_state_file();
|
|
let json = serde_json::to_string_pretty(state)?;
|
|
fs::write(state_file, json)?;
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
// Tauri commands
|
|
|
|
#[tauri::command]
|
|
pub async fn check_for_browser_updates() -> Result<Vec<UpdateNotification>, String> {
|
|
let updater = AutoUpdater::new();
|
|
let notifications = updater
|
|
.check_for_updates()
|
|
.await
|
|
.map_err(|e| format!("Failed to check for updates: {e}"))?;
|
|
let grouped = updater.group_update_notifications(notifications);
|
|
Ok(grouped)
|
|
}
|
|
|
|
#[tauri::command]
|
|
pub async fn start_browser_update(browser: String, new_version: String) -> Result<(), String> {
|
|
let updater = AutoUpdater::new();
|
|
updater
|
|
.start_browser_update(&browser, &new_version)
|
|
.await
|
|
.map_err(|e| format!("Failed to start browser update: {e}"))
|
|
}
|
|
|
|
#[tauri::command]
|
|
pub async fn complete_browser_update(browser: String) -> Result<(), String> {
|
|
let updater = AutoUpdater::new();
|
|
updater
|
|
.complete_browser_update(&browser)
|
|
.await
|
|
.map_err(|e| format!("Failed to complete browser update: {e}"))
|
|
}
|
|
|
|
#[tauri::command]
|
|
pub async fn is_browser_disabled_for_update(browser: String) -> Result<bool, String> {
|
|
let updater = AutoUpdater::new();
|
|
updater
|
|
.is_browser_disabled(&browser)
|
|
.map_err(|e| format!("Failed to check browser status: {e}"))
|
|
}
|
|
|
|
#[tauri::command]
|
|
pub async fn dismiss_update_notification(notification_id: String) -> Result<(), String> {
|
|
let updater = AutoUpdater::new();
|
|
updater
|
|
.dismiss_update_notification(¬ification_id)
|
|
.map_err(|e| format!("Failed to dismiss notification: {e}"))
|
|
}
|
|
|
|
#[tauri::command]
|
|
pub async fn complete_browser_update_with_auto_update(
|
|
browser: String,
|
|
new_version: String,
|
|
) -> Result<Vec<String>, String> {
|
|
let updater = AutoUpdater::new();
|
|
updater
|
|
.complete_browser_update_with_auto_update(&browser, &new_version)
|
|
.await
|
|
.map_err(|e| format!("Failed to complete browser update: {e}"))
|
|
}
|
|
|
|
#[tauri::command]
|
|
pub async fn mark_auto_update_download(browser: String, version: String) -> Result<(), String> {
|
|
let updater = AutoUpdater::new();
|
|
updater
|
|
.mark_auto_update_download(&browser, &version)
|
|
.map_err(|e| format!("Failed to mark auto-update download: {e}"))
|
|
}
|
|
|
|
#[tauri::command]
|
|
pub async fn remove_auto_update_download(browser: String, version: String) -> Result<(), String> {
|
|
let updater = AutoUpdater::new();
|
|
updater
|
|
.remove_auto_update_download(&browser, &version)
|
|
.map_err(|e| format!("Failed to remove auto-update download: {e}"))
|
|
}
|
|
|
|
#[tauri::command]
|
|
pub async fn is_auto_update_download(browser: String, version: String) -> Result<bool, String> {
|
|
let updater = AutoUpdater::new();
|
|
updater
|
|
.is_auto_update_download(&browser, &version)
|
|
.map_err(|e| format!("Failed to check auto-update download: {e}"))
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
fn create_test_profile(name: &str, browser: &str, version: &str) -> BrowserProfile {
|
|
BrowserProfile {
|
|
name: name.to_string(),
|
|
browser: browser.to_string(),
|
|
version: version.to_string(),
|
|
profile_path: format!("/tmp/{name}"),
|
|
process_id: None,
|
|
proxy: None,
|
|
last_launch: None,
|
|
}
|
|
}
|
|
|
|
fn create_test_version_info(version: &str, is_prerelease: bool) -> BrowserVersionInfo {
|
|
BrowserVersionInfo {
|
|
version: version.to_string(),
|
|
is_prerelease,
|
|
date: "2024-01-01".to_string(),
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_is_alpha_version() {
|
|
let updater = AutoUpdater::new();
|
|
|
|
assert!(updater.is_alpha_version("1.0.0-alpha"));
|
|
assert!(updater.is_alpha_version("1.0.0-beta"));
|
|
assert!(updater.is_alpha_version("1.0.0-rc"));
|
|
assert!(updater.is_alpha_version("1.0.0a1"));
|
|
assert!(updater.is_alpha_version("1.0.0b1"));
|
|
assert!(updater.is_alpha_version("1.0.0-dev"));
|
|
|
|
assert!(!updater.is_alpha_version("1.0.0"));
|
|
assert!(!updater.is_alpha_version("1.2.3"));
|
|
}
|
|
|
|
#[test]
|
|
fn test_compare_versions() {
|
|
let updater = AutoUpdater::new();
|
|
|
|
assert_eq!(
|
|
updater.compare_versions("1.0.0", "1.0.0"),
|
|
std::cmp::Ordering::Equal
|
|
);
|
|
assert_eq!(
|
|
updater.compare_versions("1.0.1", "1.0.0"),
|
|
std::cmp::Ordering::Greater
|
|
);
|
|
assert_eq!(
|
|
updater.compare_versions("1.0.0", "1.0.1"),
|
|
std::cmp::Ordering::Less
|
|
);
|
|
assert_eq!(
|
|
updater.compare_versions("2.0.0", "1.9.9"),
|
|
std::cmp::Ordering::Greater
|
|
);
|
|
assert_eq!(
|
|
updater.compare_versions("1.10.0", "1.9.0"),
|
|
std::cmp::Ordering::Greater
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn test_is_version_newer() {
|
|
let updater = AutoUpdater::new();
|
|
|
|
assert!(updater.is_version_newer("1.0.1", "1.0.0"));
|
|
assert!(updater.is_version_newer("2.0.0", "1.9.9"));
|
|
assert!(!updater.is_version_newer("1.0.0", "1.0.1"));
|
|
assert!(!updater.is_version_newer("1.0.0", "1.0.0"));
|
|
}
|
|
|
|
#[test]
|
|
fn test_check_profile_update_stable_to_stable() {
|
|
let updater = AutoUpdater::new();
|
|
let profile = create_test_profile("test", "firefox", "1.0.0");
|
|
let versions = vec![
|
|
create_test_version_info("1.0.1", false), // stable, newer
|
|
create_test_version_info("1.1.0-alpha", true), // alpha, should be ignored
|
|
create_test_version_info("0.9.0", false), // stable, older
|
|
];
|
|
|
|
let result = updater.check_profile_update(&profile, &versions).unwrap();
|
|
assert!(result.is_some());
|
|
|
|
let update = result.unwrap();
|
|
assert_eq!(update.new_version, "1.0.1");
|
|
assert!(update.is_stable_update);
|
|
}
|
|
|
|
#[test]
|
|
fn test_check_profile_update_alpha_to_alpha() {
|
|
let updater = AutoUpdater::new();
|
|
let profile = create_test_profile("test", "firefox", "1.0.0-alpha");
|
|
let versions = vec![
|
|
create_test_version_info("1.0.1", false), // stable, should be included
|
|
create_test_version_info("1.1.0-alpha", true), // alpha, newer
|
|
create_test_version_info("0.9.0-alpha", true), // alpha, older
|
|
];
|
|
|
|
let result = updater.check_profile_update(&profile, &versions).unwrap();
|
|
assert!(result.is_some());
|
|
|
|
let update = result.unwrap();
|
|
// Should pick the newest version (alpha user can upgrade to stable or newer alpha)
|
|
assert_eq!(update.new_version, "1.1.0-alpha");
|
|
assert!(!update.is_stable_update);
|
|
}
|
|
|
|
#[test]
|
|
fn test_check_profile_update_no_update_available() {
|
|
let updater = AutoUpdater::new();
|
|
let profile = create_test_profile("test", "firefox", "1.0.0");
|
|
let versions = vec![
|
|
create_test_version_info("0.9.0", false), // older
|
|
create_test_version_info("1.0.0", false), // same version
|
|
];
|
|
|
|
let result = updater.check_profile_update(&profile, &versions).unwrap();
|
|
assert!(result.is_none());
|
|
}
|
|
|
|
#[test]
|
|
fn test_group_update_notifications() {
|
|
let updater = AutoUpdater::new();
|
|
let notifications = vec![
|
|
UpdateNotification {
|
|
id: "firefox_1.0.0_to_1.1.0_profile1".to_string(),
|
|
browser: "firefox".to_string(),
|
|
current_version: "1.0.0".to_string(),
|
|
new_version: "1.1.0".to_string(),
|
|
affected_profiles: vec!["profile1".to_string()],
|
|
is_stable_update: true,
|
|
timestamp: 1000,
|
|
},
|
|
UpdateNotification {
|
|
id: "firefox_1.0.0_to_1.1.0_profile2".to_string(),
|
|
browser: "firefox".to_string(),
|
|
current_version: "1.0.0".to_string(),
|
|
new_version: "1.1.0".to_string(),
|
|
affected_profiles: vec!["profile2".to_string()],
|
|
is_stable_update: true,
|
|
timestamp: 1001,
|
|
},
|
|
UpdateNotification {
|
|
id: "chrome_1.0.0_to_1.1.0-alpha".to_string(),
|
|
browser: "chrome".to_string(),
|
|
current_version: "1.0.0".to_string(),
|
|
new_version: "1.1.0-alpha".to_string(),
|
|
affected_profiles: vec!["profile3".to_string()],
|
|
is_stable_update: false,
|
|
timestamp: 1002,
|
|
},
|
|
];
|
|
|
|
let grouped = updater.group_update_notifications(notifications);
|
|
|
|
assert_eq!(grouped.len(), 2);
|
|
|
|
// Find the Firefox notification
|
|
let firefox_notification = grouped.iter().find(|n| n.browser == "firefox").unwrap();
|
|
assert_eq!(firefox_notification.affected_profiles.len(), 2);
|
|
assert!(firefox_notification
|
|
.affected_profiles
|
|
.contains(&"profile1".to_string()));
|
|
assert!(firefox_notification
|
|
.affected_profiles
|
|
.contains(&"profile2".to_string()));
|
|
|
|
// Stable updates should come first
|
|
assert!(grouped[0].is_stable_update);
|
|
}
|
|
|
|
#[test]
|
|
fn test_auto_update_state_persistence() {
|
|
use std::sync::Once;
|
|
use tempfile::TempDir;
|
|
|
|
static INIT: Once = Once::new();
|
|
INIT.call_once(|| {
|
|
// Initialize any required static data
|
|
});
|
|
|
|
// Create a temporary directory for testing
|
|
let temp_dir = TempDir::new().unwrap();
|
|
|
|
// Create a mock settings manager that uses the temp directory
|
|
struct TestSettingsManager {
|
|
settings_dir: std::path::PathBuf,
|
|
}
|
|
|
|
impl TestSettingsManager {
|
|
fn new(settings_dir: std::path::PathBuf) -> Self {
|
|
Self { settings_dir }
|
|
}
|
|
|
|
fn get_settings_dir(&self) -> std::path::PathBuf {
|
|
self.settings_dir.clone()
|
|
}
|
|
}
|
|
|
|
let test_settings_manager = TestSettingsManager::new(temp_dir.path().to_path_buf());
|
|
|
|
let mut state = AutoUpdateState::default();
|
|
state.disabled_browsers.insert("firefox".to_string());
|
|
state
|
|
.auto_update_downloads
|
|
.insert("firefox-1.1.0".to_string());
|
|
state.pending_updates.push(UpdateNotification {
|
|
id: "test".to_string(),
|
|
browser: "firefox".to_string(),
|
|
current_version: "1.0.0".to_string(),
|
|
new_version: "1.1.0".to_string(),
|
|
affected_profiles: vec!["profile1".to_string()],
|
|
is_stable_update: true,
|
|
timestamp: 1000,
|
|
});
|
|
|
|
// Test save and load
|
|
let state_file = test_settings_manager
|
|
.get_settings_dir()
|
|
.join("auto_update_state.json");
|
|
std::fs::create_dir_all(test_settings_manager.get_settings_dir()).unwrap();
|
|
let json = serde_json::to_string_pretty(&state).unwrap();
|
|
std::fs::write(&state_file, json).unwrap();
|
|
|
|
// Load state
|
|
let content = std::fs::read_to_string(&state_file).unwrap();
|
|
let loaded_state: AutoUpdateState = serde_json::from_str(&content).unwrap();
|
|
|
|
assert_eq!(loaded_state.disabled_browsers.len(), 1);
|
|
assert!(loaded_state.disabled_browsers.contains("firefox"));
|
|
assert_eq!(loaded_state.auto_update_downloads.len(), 1);
|
|
assert!(loaded_state.auto_update_downloads.contains("firefox-1.1.0"));
|
|
assert_eq!(loaded_state.pending_updates.len(), 1);
|
|
assert_eq!(loaded_state.pending_updates[0].id, "test");
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_browser_disable_enable_cycle() {
|
|
use tempfile::TempDir;
|
|
|
|
// Create a temporary directory for testing
|
|
let temp_dir = TempDir::new().unwrap();
|
|
|
|
// Create a mock settings manager that uses the temp directory
|
|
struct TestSettingsManager {
|
|
settings_dir: std::path::PathBuf,
|
|
}
|
|
|
|
impl TestSettingsManager {
|
|
fn new(settings_dir: std::path::PathBuf) -> Self {
|
|
Self { settings_dir }
|
|
}
|
|
|
|
fn get_settings_dir(&self) -> std::path::PathBuf {
|
|
self.settings_dir.clone()
|
|
}
|
|
}
|
|
|
|
let test_settings_manager = TestSettingsManager::new(temp_dir.path().to_path_buf());
|
|
|
|
// Test browser disable/enable cycle with manual state management
|
|
let state_file = test_settings_manager
|
|
.get_settings_dir()
|
|
.join("auto_update_state.json");
|
|
std::fs::create_dir_all(test_settings_manager.get_settings_dir()).unwrap();
|
|
|
|
// Initially not disabled (empty state file means default state)
|
|
let state = AutoUpdateState::default();
|
|
assert!(!state.disabled_browsers.contains("firefox"));
|
|
|
|
// Start update (should disable)
|
|
let mut state = AutoUpdateState::default();
|
|
state.disabled_browsers.insert("firefox".to_string());
|
|
state
|
|
.auto_update_downloads
|
|
.insert("firefox-1.1.0".to_string());
|
|
let json = serde_json::to_string_pretty(&state).unwrap();
|
|
std::fs::write(&state_file, json).unwrap();
|
|
|
|
// Check that it's disabled
|
|
let content = std::fs::read_to_string(&state_file).unwrap();
|
|
let loaded_state: AutoUpdateState = serde_json::from_str(&content).unwrap();
|
|
assert!(loaded_state.disabled_browsers.contains("firefox"));
|
|
assert!(loaded_state.auto_update_downloads.contains("firefox-1.1.0"));
|
|
|
|
// Complete update (should enable)
|
|
let mut state = loaded_state;
|
|
state.disabled_browsers.remove("firefox");
|
|
state.auto_update_downloads.remove("firefox-1.1.0");
|
|
let json = serde_json::to_string_pretty(&state).unwrap();
|
|
std::fs::write(&state_file, json).unwrap();
|
|
|
|
// Check that it's enabled again
|
|
let content = std::fs::read_to_string(&state_file).unwrap();
|
|
let final_state: AutoUpdateState = serde_json::from_str(&content).unwrap();
|
|
assert!(!final_state.disabled_browsers.contains("firefox"));
|
|
assert!(!final_state.auto_update_downloads.contains("firefox-1.1.0"));
|
|
}
|
|
|
|
#[test]
|
|
fn test_dismiss_update_notification() {
|
|
use tempfile::TempDir;
|
|
|
|
// Create a temporary directory for testing
|
|
let temp_dir = TempDir::new().unwrap();
|
|
|
|
// Create a mock settings manager that uses the temp directory
|
|
struct TestSettingsManager {
|
|
settings_dir: std::path::PathBuf,
|
|
}
|
|
|
|
impl TestSettingsManager {
|
|
fn new(settings_dir: std::path::PathBuf) -> Self {
|
|
Self { settings_dir }
|
|
}
|
|
|
|
fn get_settings_dir(&self) -> std::path::PathBuf {
|
|
self.settings_dir.clone()
|
|
}
|
|
}
|
|
|
|
let test_settings_manager = TestSettingsManager::new(temp_dir.path().to_path_buf());
|
|
|
|
let mut state = AutoUpdateState::default();
|
|
state.pending_updates.push(UpdateNotification {
|
|
id: "test_notification".to_string(),
|
|
browser: "firefox".to_string(),
|
|
current_version: "1.0.0".to_string(),
|
|
new_version: "1.1.0".to_string(),
|
|
affected_profiles: vec!["profile1".to_string()],
|
|
is_stable_update: true,
|
|
timestamp: 1000,
|
|
});
|
|
|
|
// Save initial state
|
|
let state_file = test_settings_manager
|
|
.get_settings_dir()
|
|
.join("auto_update_state.json");
|
|
std::fs::create_dir_all(test_settings_manager.get_settings_dir()).unwrap();
|
|
let json = serde_json::to_string_pretty(&state).unwrap();
|
|
std::fs::write(&state_file, json).unwrap();
|
|
|
|
// Dismiss notification (remove from pending updates)
|
|
state
|
|
.pending_updates
|
|
.retain(|n| n.id != "test_notification");
|
|
let json = serde_json::to_string_pretty(&state).unwrap();
|
|
std::fs::write(&state_file, json).unwrap();
|
|
|
|
// Check that it's removed
|
|
let content = std::fs::read_to_string(&state_file).unwrap();
|
|
let loaded_state: AutoUpdateState = serde_json::from_str(&content).unwrap();
|
|
assert_eq!(loaded_state.pending_updates.len(), 0);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_version() {
|
|
let updater = AutoUpdater::new();
|
|
|
|
assert_eq!(updater.parse_version("1.2.3"), vec![1, 2, 3]);
|
|
assert_eq!(updater.parse_version("1.2.3-alpha"), vec![1, 2, 3]);
|
|
assert_eq!(updater.parse_version("1.2.3a1"), vec![1, 2, 3, 1]);
|
|
assert_eq!(updater.parse_version("1.2.3b2"), vec![1, 2, 3, 2]);
|
|
assert_eq!(updater.parse_version("10.0.0"), vec![10, 0, 0]);
|
|
}
|
|
}
|