support custom userdata

This commit is contained in:
Julian Freeman
2026-04-16 14:50:38 -04:00
parent 436797abfa
commit a8ad89074f
9 changed files with 748 additions and 76 deletions

View File

@@ -0,0 +1,188 @@
use std::{
fs,
path::PathBuf,
time::{SystemTime, UNIX_EPOCH},
};
use tauri::{AppHandle, Manager};
use crate::{
browsers::{browser_definitions, resolve_browser_executable},
models::{
BrowserConfigEntry, BrowserConfigListResponse, BrowserConfigSource,
CreateCustomBrowserConfigInput, CustomBrowserConfigRecord, StoredBrowserConfigs,
},
utils::local_app_data_dir,
};
const CONFIG_FILE_NAME: &str = "browser-configs.json";
pub fn load_browser_config_list(app: &AppHandle) -> Result<BrowserConfigListResponse, String> {
Ok(BrowserConfigListResponse {
configs: resolve_browser_configs(app)?,
})
}
pub fn resolve_browser_configs(app: &AppHandle) -> Result<Vec<BrowserConfigEntry>, String> {
let mut configs = default_browser_configs()?;
let stored = load_stored_configs(app)?;
configs.extend(
stored
.custom_configs
.into_iter()
.map(|config| BrowserConfigEntry {
id: config.id,
source: BrowserConfigSource::Custom,
browser_family_id: None,
name: config.name,
executable_path: config.executable_path,
user_data_path: config.user_data_path,
deletable: true,
}),
);
Ok(configs)
}
pub fn create_custom_browser_config(
app: &AppHandle,
input: CreateCustomBrowserConfigInput,
) -> Result<BrowserConfigListResponse, String> {
let name = input.name.trim();
let executable_path = input.executable_path.trim();
let user_data_path = input.user_data_path.trim();
if name.is_empty() {
return Err("Name is required.".to_string());
}
if executable_path.is_empty() {
return Err("Executable path is required.".to_string());
}
if user_data_path.is_empty() {
return Err("User data path is required.".to_string());
}
let mut stored = load_stored_configs(app)?;
stored.custom_configs.push(CustomBrowserConfigRecord {
id: generate_custom_config_id(),
name: name.to_string(),
executable_path: executable_path.to_string(),
user_data_path: user_data_path.to_string(),
});
save_stored_configs(app, &stored)?;
load_browser_config_list(app)
}
pub fn delete_custom_browser_config(
app: &AppHandle,
config_id: &str,
) -> Result<BrowserConfigListResponse, String> {
let mut stored = load_stored_configs(app)?;
let original_len = stored.custom_configs.len();
stored
.custom_configs
.retain(|config| config.id != config_id);
if stored.custom_configs.len() == original_len {
return Err(format!("Custom browser config not found: {config_id}"));
}
save_stored_configs(app, &stored)?;
load_browser_config_list(app)
}
pub fn find_browser_config(app: &AppHandle, config_id: &str) -> Result<BrowserConfigEntry, String> {
resolve_browser_configs(app)?
.into_iter()
.find(|config| config.id == config_id)
.ok_or_else(|| format!("Browser config not found: {config_id}"))
}
fn default_browser_configs() -> Result<Vec<BrowserConfigEntry>, String> {
let local_app_data = local_app_data_dir().ok_or_else(|| {
"Unable to resolve the LOCALAPPDATA directory for the current user.".to_string()
})?;
Ok(browser_definitions()
.into_iter()
.map(|definition| {
let user_data_path = definition
.local_app_data_segments
.iter()
.fold(local_app_data.clone(), |path, segment| path.join(segment));
BrowserConfigEntry {
id: definition.id.to_string(),
source: BrowserConfigSource::Default,
browser_family_id: Some(definition.id.to_string()),
name: definition.name.to_string(),
executable_path: resolve_browser_executable(definition.id)
.map(|path| path.display().to_string())
.unwrap_or_default(),
user_data_path: user_data_path.display().to_string(),
deletable: false,
}
})
.collect())
}
fn load_stored_configs(app: &AppHandle) -> Result<StoredBrowserConfigs, String> {
let path = config_file_path(app)?;
if !path.is_file() {
return Ok(StoredBrowserConfigs {
custom_configs: Vec::new(),
});
}
let content = fs::read_to_string(&path).map_err(|error| {
format!(
"Failed to read browser config file {}: {error}",
path.display()
)
})?;
serde_json::from_str(&content).map_err(|error| {
format!(
"Failed to parse browser config file {}: {error}",
path.display()
)
})
}
fn save_stored_configs(app: &AppHandle, stored: &StoredBrowserConfigs) -> Result<(), String> {
let path = config_file_path(app)?;
if let Some(parent) = path.parent() {
fs::create_dir_all(parent).map_err(|error| {
format!(
"Failed to create browser config directory {}: {error}",
parent.display()
)
})?;
}
let content = serde_json::to_string_pretty(stored)
.map_err(|error| format!("Failed to serialize browser configs: {error}"))?;
fs::write(&path, content).map_err(|error| {
format!(
"Failed to write browser config file {}: {error}",
path.display()
)
})
}
fn config_file_path(app: &AppHandle) -> Result<PathBuf, String> {
let app_data_dir = app
.path()
.app_data_dir()
.map_err(|error| format!("Failed to resolve app data directory: {error}"))?;
Ok(app_data_dir.join(CONFIG_FILE_NAME))
}
fn generate_custom_config_id() -> String {
let timestamp = SystemTime::now()
.duration_since(UNIX_EPOCH)
.map(|duration| duration.as_millis())
.unwrap_or(0);
format!("custom-{timestamp}")
}