2025-08-31 19:00:09 +08:00
|
|
|
|
#![allow(non_snake_case)]
|
|
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
use std::collections::HashMap;
|
|
|
|
|
|
use tauri::State;
|
2025-08-25 20:16:29 +08:00
|
|
|
|
use tauri_plugin_opener::OpenerExt;
|
2025-08-23 20:15:10 +08:00
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
use crate::app_config::AppType;
|
|
|
|
|
|
use crate::codex_config;
|
2025-09-04 15:59:28 +08:00
|
|
|
|
use crate::config::{ConfigStatus, get_claude_settings_path};
|
2025-08-23 21:00:50 +08:00
|
|
|
|
use crate::provider::Provider;
|
2025-08-23 20:15:10 +08:00
|
|
|
|
use crate::store::AppState;
|
|
|
|
|
|
|
|
|
|
|
|
/// 获取所有供应商
|
|
|
|
|
|
#[tauri::command]
|
2025-08-27 11:00:53 +08:00
|
|
|
|
pub async fn get_providers(
|
|
|
|
|
|
state: State<'_, AppState>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
app_type: Option<AppType>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
app: Option<String>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
appType: Option<String>,
|
2025-08-27 11:00:53 +08:00
|
|
|
|
) -> Result<HashMap<String, Provider>, String> {
|
2025-08-31 16:43:33 +08:00
|
|
|
|
let app_type = app_type
|
|
|
|
|
|
.or_else(|| app.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.or_else(|| appType.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.unwrap_or(AppType::Claude);
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
|
|
|
|
|
let config = state
|
|
|
|
|
|
.config
|
2025-08-27 11:00:53 +08:00
|
|
|
|
.lock()
|
2025-08-23 20:15:10 +08:00
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
let manager = config
|
|
|
|
|
|
.get_manager(&app_type)
|
|
|
|
|
|
.ok_or_else(|| format!("应用类型不存在: {:?}", app_type))?;
|
|
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
Ok(manager.get_all_providers().clone())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 获取当前供应商ID
|
|
|
|
|
|
#[tauri::command]
|
2025-08-30 21:54:11 +08:00
|
|
|
|
pub async fn get_current_provider(
|
|
|
|
|
|
state: State<'_, AppState>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
app_type: Option<AppType>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
app: Option<String>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
appType: Option<String>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
) -> Result<String, String> {
|
2025-08-31 16:43:33 +08:00
|
|
|
|
let app_type = app_type
|
|
|
|
|
|
.or_else(|| app.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.or_else(|| appType.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.unwrap_or(AppType::Claude);
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
|
|
|
|
|
let config = state
|
|
|
|
|
|
.config
|
2025-08-27 11:00:53 +08:00
|
|
|
|
.lock()
|
2025-08-23 20:15:10 +08:00
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
let manager = config
|
|
|
|
|
|
.get_manager(&app_type)
|
|
|
|
|
|
.ok_or_else(|| format!("应用类型不存在: {:?}", app_type))?;
|
|
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
Ok(manager.current.clone())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 添加供应商
|
|
|
|
|
|
#[tauri::command]
|
2025-08-30 21:54:11 +08:00
|
|
|
|
pub async fn add_provider(
|
|
|
|
|
|
state: State<'_, AppState>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
app_type: Option<AppType>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
app: Option<String>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
appType: Option<String>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
provider: Provider,
|
|
|
|
|
|
) -> Result<bool, String> {
|
2025-08-31 16:43:33 +08:00
|
|
|
|
let app_type = app_type
|
|
|
|
|
|
.or_else(|| app.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.or_else(|| appType.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.unwrap_or(AppType::Claude);
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
2025-09-05 20:52:08 +08:00
|
|
|
|
// 读取当前是否是激活供应商(短锁)
|
|
|
|
|
|
let is_current = {
|
|
|
|
|
|
let config = state
|
|
|
|
|
|
.config
|
|
|
|
|
|
.lock()
|
|
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
|
|
|
|
|
let manager = config
|
|
|
|
|
|
.get_manager(&app_type)
|
|
|
|
|
|
.ok_or_else(|| format!("应用类型不存在: {:?}", app_type))?;
|
|
|
|
|
|
manager.current == provider.id
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// 若目标为当前供应商,则先写 live,成功后再落盘配置
|
2025-09-04 16:34:47 +08:00
|
|
|
|
if is_current {
|
|
|
|
|
|
match app_type {
|
|
|
|
|
|
AppType::Claude => {
|
|
|
|
|
|
let settings_path = crate::config::get_claude_settings_path();
|
|
|
|
|
|
crate::config::write_json_file(&settings_path, &provider.settings_config)?;
|
|
|
|
|
|
}
|
|
|
|
|
|
AppType::Codex => {
|
|
|
|
|
|
let auth = provider
|
|
|
|
|
|
.settings_config
|
|
|
|
|
|
.get("auth")
|
|
|
|
|
|
.ok_or_else(|| "目标供应商缺少 auth 配置".to_string())?;
|
2025-09-05 20:52:08 +08:00
|
|
|
|
let cfg_text = provider
|
|
|
|
|
|
.settings_config
|
|
|
|
|
|
.get("config")
|
|
|
|
|
|
.and_then(|v| v.as_str());
|
|
|
|
|
|
crate::codex_config::write_codex_live_atomic(auth, cfg_text)?;
|
2025-09-04 16:34:47 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-05 20:52:08 +08:00
|
|
|
|
// 更新内存并保存配置
|
|
|
|
|
|
{
|
|
|
|
|
|
let mut config = state
|
|
|
|
|
|
.config
|
|
|
|
|
|
.lock()
|
|
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
|
|
|
|
|
let manager = config
|
|
|
|
|
|
.get_manager_mut(&app_type)
|
|
|
|
|
|
.ok_or_else(|| format!("应用类型不存在: {:?}", app_type))?;
|
|
|
|
|
|
manager.providers.insert(provider.id.clone(), provider.clone());
|
|
|
|
|
|
}
|
|
|
|
|
|
state.save()?;
|
|
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
Ok(true)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 更新供应商
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
|
|
pub async fn update_provider(
|
|
|
|
|
|
state: State<'_, AppState>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
app_type: Option<AppType>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
app: Option<String>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
appType: Option<String>,
|
2025-08-23 20:15:10 +08:00
|
|
|
|
provider: Provider,
|
|
|
|
|
|
) -> Result<bool, String> {
|
2025-08-31 16:43:33 +08:00
|
|
|
|
let app_type = app_type
|
|
|
|
|
|
.or_else(|| app.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.or_else(|| appType.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.unwrap_or(AppType::Claude);
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
2025-09-05 20:52:08 +08:00
|
|
|
|
// 读取校验 & 是否当前(短锁)
|
|
|
|
|
|
let (exists, is_current) = {
|
|
|
|
|
|
let config = state
|
|
|
|
|
|
.config
|
|
|
|
|
|
.lock()
|
|
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
|
|
|
|
|
let manager = config
|
|
|
|
|
|
.get_manager(&app_type)
|
|
|
|
|
|
.ok_or_else(|| format!("应用类型不存在: {:?}", app_type))?;
|
|
|
|
|
|
(manager.providers.contains_key(&provider.id), manager.current == provider.id)
|
|
|
|
|
|
};
|
|
|
|
|
|
if !exists {
|
2025-08-30 21:54:11 +08:00
|
|
|
|
return Err(format!("供应商不存在: {}", provider.id));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-05 20:52:08 +08:00
|
|
|
|
// 若更新的是当前供应商,先写 live 成功再保存
|
2025-09-05 11:00:53 +08:00
|
|
|
|
if is_current {
|
|
|
|
|
|
match app_type {
|
|
|
|
|
|
AppType::Claude => {
|
|
|
|
|
|
let settings_path = crate::config::get_claude_settings_path();
|
|
|
|
|
|
crate::config::write_json_file(&settings_path, &provider.settings_config)?;
|
|
|
|
|
|
}
|
|
|
|
|
|
AppType::Codex => {
|
|
|
|
|
|
let auth = provider
|
|
|
|
|
|
.settings_config
|
|
|
|
|
|
.get("auth")
|
|
|
|
|
|
.ok_or_else(|| "目标供应商缺少 auth 配置".to_string())?;
|
2025-09-05 20:52:08 +08:00
|
|
|
|
let cfg_text = provider
|
|
|
|
|
|
.settings_config
|
|
|
|
|
|
.get("config")
|
|
|
|
|
|
.and_then(|v| v.as_str());
|
|
|
|
|
|
crate::codex_config::write_codex_live_atomic(auth, cfg_text)?;
|
2025-09-05 11:00:53 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-05 20:52:08 +08:00
|
|
|
|
// 更新内存并保存
|
|
|
|
|
|
{
|
|
|
|
|
|
let mut config = state
|
|
|
|
|
|
.config
|
|
|
|
|
|
.lock()
|
|
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
|
|
|
|
|
let manager = config
|
|
|
|
|
|
.get_manager_mut(&app_type)
|
|
|
|
|
|
.ok_or_else(|| format!("应用类型不存在: {:?}", app_type))?;
|
|
|
|
|
|
manager.providers.insert(provider.id.clone(), provider.clone());
|
|
|
|
|
|
}
|
|
|
|
|
|
state.save()?;
|
|
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
Ok(true)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 删除供应商
|
|
|
|
|
|
#[tauri::command]
|
2025-08-30 21:54:11 +08:00
|
|
|
|
pub async fn delete_provider(
|
|
|
|
|
|
state: State<'_, AppState>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
app_type: Option<AppType>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
app: Option<String>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
appType: Option<String>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
id: String,
|
|
|
|
|
|
) -> Result<bool, String> {
|
2025-08-31 16:43:33 +08:00
|
|
|
|
let app_type = app_type
|
|
|
|
|
|
.or_else(|| app.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.or_else(|| appType.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.unwrap_or(AppType::Claude);
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
|
|
|
|
|
let mut config = state
|
|
|
|
|
|
.config
|
2025-08-27 11:00:53 +08:00
|
|
|
|
.lock()
|
2025-08-23 20:15:10 +08:00
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
let manager = config
|
|
|
|
|
|
.get_manager_mut(&app_type)
|
|
|
|
|
|
.ok_or_else(|| format!("应用类型不存在: {:?}", app_type))?;
|
|
|
|
|
|
|
|
|
|
|
|
// 检查是否为当前供应商
|
|
|
|
|
|
if manager.current == id {
|
|
|
|
|
|
return Err("不能删除当前正在使用的供应商".to_string());
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取供应商信息
|
|
|
|
|
|
let provider = manager
|
|
|
|
|
|
.providers
|
|
|
|
|
|
.get(&id)
|
|
|
|
|
|
.ok_or_else(|| format!("供应商不存在: {}", id))?
|
|
|
|
|
|
.clone();
|
|
|
|
|
|
|
|
|
|
|
|
// 删除配置文件
|
|
|
|
|
|
match app_type {
|
|
|
|
|
|
AppType::Codex => {
|
|
|
|
|
|
codex_config::delete_codex_provider_config(&id, &provider.name)?;
|
|
|
|
|
|
}
|
|
|
|
|
|
AppType::Claude => {
|
|
|
|
|
|
use crate::config::{delete_file, get_provider_config_path};
|
2025-09-05 11:00:53 +08:00
|
|
|
|
// 兼容历史两种命名:settings-{name}.json 与 settings-{id}.json
|
|
|
|
|
|
let by_name = get_provider_config_path(&id, Some(&provider.name));
|
|
|
|
|
|
let by_id = get_provider_config_path(&id, None);
|
|
|
|
|
|
delete_file(&by_name)?;
|
|
|
|
|
|
delete_file(&by_id)?;
|
2025-08-30 21:54:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 从管理器删除
|
|
|
|
|
|
manager.providers.remove(&id);
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
// 保存配置
|
2025-08-30 21:54:11 +08:00
|
|
|
|
drop(config); // 释放锁
|
2025-08-23 20:15:10 +08:00
|
|
|
|
state.save()?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
Ok(true)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 切换供应商
|
|
|
|
|
|
#[tauri::command]
|
2025-08-30 21:54:11 +08:00
|
|
|
|
pub async fn switch_provider(
|
|
|
|
|
|
state: State<'_, AppState>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
app_type: Option<AppType>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
app: Option<String>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
appType: Option<String>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
id: String,
|
|
|
|
|
|
) -> Result<bool, String> {
|
2025-08-31 16:43:33 +08:00
|
|
|
|
let app_type = app_type
|
|
|
|
|
|
.or_else(|| app.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.or_else(|| appType.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.unwrap_or(AppType::Claude);
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
|
|
|
|
|
let mut config = state
|
|
|
|
|
|
.config
|
2025-08-27 11:00:53 +08:00
|
|
|
|
.lock()
|
2025-08-23 20:15:10 +08:00
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
let manager = config
|
|
|
|
|
|
.get_manager_mut(&app_type)
|
|
|
|
|
|
.ok_or_else(|| format!("应用类型不存在: {:?}", app_type))?;
|
|
|
|
|
|
|
|
|
|
|
|
// 检查供应商是否存在
|
|
|
|
|
|
let provider = manager
|
|
|
|
|
|
.providers
|
|
|
|
|
|
.get(&id)
|
|
|
|
|
|
.ok_or_else(|| format!("供应商不存在: {}", id))?
|
|
|
|
|
|
.clone();
|
|
|
|
|
|
|
2025-09-04 15:59:28 +08:00
|
|
|
|
// SSOT 切换:先回填 live 配置到当前供应商,然后从内存写入目标主配置
|
2025-08-30 21:54:11 +08:00
|
|
|
|
match app_type {
|
|
|
|
|
|
AppType::Codex => {
|
2025-09-04 15:59:28 +08:00
|
|
|
|
use serde_json::Value;
|
|
|
|
|
|
|
|
|
|
|
|
// 回填:读取 live(auth.json + config.toml)写回当前供应商 settings_config
|
2025-08-30 21:54:11 +08:00
|
|
|
|
if !manager.current.is_empty() {
|
2025-09-04 15:59:28 +08:00
|
|
|
|
let auth_path = codex_config::get_codex_auth_path();
|
|
|
|
|
|
let config_path = codex_config::get_codex_config_path();
|
|
|
|
|
|
if auth_path.exists() {
|
|
|
|
|
|
let auth: Value = crate::config::read_json_file(&auth_path)?;
|
|
|
|
|
|
let config_str = if config_path.exists() {
|
|
|
|
|
|
std::fs::read_to_string(&config_path)
|
|
|
|
|
|
.map_err(|e| format!("读取 config.toml 失败: {}", e))?
|
|
|
|
|
|
} else {
|
|
|
|
|
|
String::new()
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
let live = serde_json::json!({
|
|
|
|
|
|
"auth": auth,
|
|
|
|
|
|
"config": config_str,
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
if let Some(cur) = manager.providers.get_mut(&manager.current) {
|
|
|
|
|
|
cur.settings_config = live;
|
|
|
|
|
|
}
|
2025-08-30 21:54:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-05 20:52:08 +08:00
|
|
|
|
// 切换:从目标供应商 settings_config 写入主配置(Codex 双文件原子+回滚)
|
2025-09-04 15:59:28 +08:00
|
|
|
|
let auth = provider
|
|
|
|
|
|
.settings_config
|
|
|
|
|
|
.get("auth")
|
|
|
|
|
|
.ok_or_else(|| "目标供应商缺少 auth 配置".to_string())?;
|
2025-09-05 20:52:08 +08:00
|
|
|
|
let cfg_text = provider
|
|
|
|
|
|
.settings_config
|
|
|
|
|
|
.get("config")
|
|
|
|
|
|
.and_then(|v| v.as_str());
|
|
|
|
|
|
crate::codex_config::write_codex_live_atomic(auth, cfg_text)?;
|
2025-08-30 21:54:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
AppType::Claude => {
|
2025-09-04 15:59:28 +08:00
|
|
|
|
use crate::config::{read_json_file, write_json_file};
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
|
|
|
|
|
let settings_path = get_claude_settings_path();
|
|
|
|
|
|
|
2025-09-04 15:59:28 +08:00
|
|
|
|
// 回填:读取 live settings.json 写回当前供应商 settings_config
|
2025-08-30 21:54:11 +08:00
|
|
|
|
if settings_path.exists() && !manager.current.is_empty() {
|
2025-09-04 15:59:28 +08:00
|
|
|
|
if let Ok(live) = read_json_file::<serde_json::Value>(&settings_path) {
|
|
|
|
|
|
if let Some(cur) = manager.providers.get_mut(&manager.current) {
|
|
|
|
|
|
cur.settings_config = live;
|
|
|
|
|
|
}
|
2025-08-30 21:54:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-04 15:59:28 +08:00
|
|
|
|
// 切换:从目标供应商 settings_config 写入主配置
|
2025-08-30 21:54:11 +08:00
|
|
|
|
if let Some(parent) = settings_path.parent() {
|
|
|
|
|
|
std::fs::create_dir_all(parent).map_err(|e| format!("创建目录失败: {}", e))?;
|
|
|
|
|
|
}
|
2025-09-04 16:07:38 +08:00
|
|
|
|
|
2025-09-05 16:39:12 +08:00
|
|
|
|
// 不做归档,直接写入
|
2025-09-04 15:59:28 +08:00
|
|
|
|
write_json_file(&settings_path, &provider.settings_config)?;
|
2025-08-30 21:54:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 更新当前供应商
|
|
|
|
|
|
manager.current = id;
|
|
|
|
|
|
|
|
|
|
|
|
log::info!("成功切换到供应商: {}", provider.name);
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
// 保存配置
|
2025-08-30 21:54:11 +08:00
|
|
|
|
drop(config); // 释放锁
|
2025-08-23 20:15:10 +08:00
|
|
|
|
state.save()?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
Ok(true)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 导入当前配置为默认供应商
|
|
|
|
|
|
#[tauri::command]
|
2025-08-30 21:54:11 +08:00
|
|
|
|
pub async fn import_default_config(
|
|
|
|
|
|
state: State<'_, AppState>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
app_type: Option<AppType>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
app: Option<String>,
|
2025-08-31 16:43:33 +08:00
|
|
|
|
appType: Option<String>,
|
2025-08-30 21:54:11 +08:00
|
|
|
|
) -> Result<bool, String> {
|
2025-08-31 16:43:33 +08:00
|
|
|
|
let app_type = app_type
|
|
|
|
|
|
.or_else(|| app.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.or_else(|| appType.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.unwrap_or(AppType::Claude);
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
2025-09-05 15:07:00 +08:00
|
|
|
|
// 仅当 providers 为空时才从 live 导入一条默认项
|
2025-08-24 23:30:35 +08:00
|
|
|
|
{
|
2025-08-30 21:54:11 +08:00
|
|
|
|
let config = state
|
|
|
|
|
|
.config
|
2025-08-24 23:30:35 +08:00
|
|
|
|
.lock()
|
|
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
|
|
|
|
|
if let Some(manager) = config.get_manager(&app_type) {
|
2025-09-05 15:07:00 +08:00
|
|
|
|
if !manager.get_all_providers().is_empty() {
|
2025-08-30 21:54:11 +08:00
|
|
|
|
return Ok(true);
|
|
|
|
|
|
}
|
2025-08-24 23:30:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
// 根据应用类型导入配置
|
2025-09-04 15:59:28 +08:00
|
|
|
|
// 读取当前主配置为默认供应商(不再写入副本文件)
|
2025-08-30 21:54:11 +08:00
|
|
|
|
let settings_config = match app_type {
|
2025-09-04 15:59:28 +08:00
|
|
|
|
AppType::Codex => {
|
|
|
|
|
|
let auth_path = codex_config::get_codex_auth_path();
|
|
|
|
|
|
let config_path = codex_config::get_codex_config_path();
|
|
|
|
|
|
if !auth_path.exists() {
|
|
|
|
|
|
return Err("Codex 配置文件不存在".to_string());
|
|
|
|
|
|
}
|
|
|
|
|
|
let auth: serde_json::Value = crate::config::read_json_file::<serde_json::Value>(&auth_path)?;
|
|
|
|
|
|
let config_str = if config_path.exists() {
|
|
|
|
|
|
let s = std::fs::read_to_string(&config_path)
|
|
|
|
|
|
.map_err(|e| format!("读取 config.toml 失败: {}", e))?;
|
|
|
|
|
|
if !s.trim().is_empty() {
|
|
|
|
|
|
toml::from_str::<toml::Table>(&s)
|
|
|
|
|
|
.map_err(|e| format!("config.toml 语法错误: {}", e))?;
|
|
|
|
|
|
}
|
|
|
|
|
|
s
|
|
|
|
|
|
} else {
|
|
|
|
|
|
String::new()
|
|
|
|
|
|
};
|
|
|
|
|
|
serde_json::json!({ "auth": auth, "config": config_str })
|
|
|
|
|
|
}
|
|
|
|
|
|
AppType::Claude => {
|
|
|
|
|
|
let settings_path = get_claude_settings_path();
|
|
|
|
|
|
if !settings_path.exists() {
|
|
|
|
|
|
return Err("Claude Code 配置文件不存在".to_string());
|
|
|
|
|
|
}
|
|
|
|
|
|
crate::config::read_json_file::<serde_json::Value>(&settings_path)?
|
|
|
|
|
|
}
|
2025-08-30 21:54:11 +08:00
|
|
|
|
};
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-09-05 15:07:00 +08:00
|
|
|
|
// 创建默认供应商(仅首次初始化)
|
2025-08-23 20:15:10 +08:00
|
|
|
|
let provider = Provider::with_id(
|
2025-09-05 15:07:00 +08:00
|
|
|
|
"default".to_string(),
|
|
|
|
|
|
"default".to_string(),
|
2025-08-23 20:15:10 +08:00
|
|
|
|
settings_config,
|
|
|
|
|
|
None,
|
|
|
|
|
|
);
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
// 添加到管理器
|
2025-08-30 21:54:11 +08:00
|
|
|
|
let mut config = state
|
|
|
|
|
|
.config
|
2025-08-27 11:00:53 +08:00
|
|
|
|
.lock()
|
2025-08-23 20:15:10 +08:00
|
|
|
|
.map_err(|e| format!("获取锁失败: {}", e))?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
let manager = config
|
|
|
|
|
|
.get_manager_mut(&app_type)
|
|
|
|
|
|
.ok_or_else(|| format!("应用类型不存在: {:?}", app_type))?;
|
|
|
|
|
|
|
|
|
|
|
|
manager.providers.insert(provider.id.clone(), provider);
|
2025-09-05 15:07:00 +08:00
|
|
|
|
// 设置当前供应商为默认项
|
|
|
|
|
|
manager.current = "default".to_string();
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
// 保存配置
|
2025-08-30 21:54:11 +08:00
|
|
|
|
drop(config); // 释放锁
|
2025-08-23 20:15:10 +08:00
|
|
|
|
state.save()?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
Ok(true)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 获取 Claude Code 配置状态
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
|
|
pub async fn get_claude_config_status() -> Result<ConfigStatus, String> {
|
2025-08-23 21:00:50 +08:00
|
|
|
|
Ok(crate::config::get_claude_config_status())
|
2025-08-23 20:15:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
/// 获取应用配置状态(通用)
|
2025-08-31 16:39:38 +08:00
|
|
|
|
/// 兼容两种参数:`app_type`(推荐)或 `app`(字符串)
|
2025-08-30 21:54:11 +08:00
|
|
|
|
#[tauri::command]
|
2025-08-31 16:39:38 +08:00
|
|
|
|
pub async fn get_config_status(
|
|
|
|
|
|
app_type: Option<AppType>,
|
|
|
|
|
|
app: Option<String>,
|
|
|
|
|
|
appType: Option<String>,
|
|
|
|
|
|
) -> Result<ConfigStatus, String> {
|
|
|
|
|
|
let app = app_type
|
|
|
|
|
|
.or_else(|| app.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.or_else(|| appType.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.unwrap_or(AppType::Claude);
|
|
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
match app {
|
|
|
|
|
|
AppType::Claude => Ok(crate::config::get_claude_config_status()),
|
|
|
|
|
|
AppType::Codex => {
|
2025-08-31 16:39:38 +08:00
|
|
|
|
use crate::codex_config::{get_codex_auth_path, get_codex_config_dir};
|
2025-08-30 21:54:11 +08:00
|
|
|
|
let auth_path = get_codex_auth_path();
|
2025-08-31 16:39:38 +08:00
|
|
|
|
|
|
|
|
|
|
// 放宽:只要 auth.json 存在即可认为已配置;config.toml 允许为空
|
|
|
|
|
|
let exists = auth_path.exists();
|
|
|
|
|
|
let path = get_codex_config_dir().to_string_lossy().to_string();
|
|
|
|
|
|
|
2025-08-30 21:54:11 +08:00
|
|
|
|
Ok(ConfigStatus { exists, path })
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
/// 获取 Claude Code 配置文件路径
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
|
|
pub async fn get_claude_code_config_path() -> Result<String, String> {
|
|
|
|
|
|
Ok(get_claude_settings_path().to_string_lossy().to_string())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 打开配置文件夹
|
2025-08-31 16:39:38 +08:00
|
|
|
|
/// 兼容两种参数:`app_type`(推荐)或 `app`(字符串)
|
2025-08-23 20:15:10 +08:00
|
|
|
|
#[tauri::command]
|
2025-08-31 16:39:38 +08:00
|
|
|
|
pub async fn open_config_folder(
|
2025-08-31 18:14:31 +08:00
|
|
|
|
handle: tauri::AppHandle,
|
2025-08-31 16:39:38 +08:00
|
|
|
|
app_type: Option<AppType>,
|
2025-08-31 18:14:31 +08:00
|
|
|
|
app: Option<String>,
|
2025-08-31 16:39:38 +08:00
|
|
|
|
appType: Option<String>,
|
|
|
|
|
|
) -> Result<bool, String> {
|
|
|
|
|
|
let app_type = app_type
|
2025-08-31 18:14:31 +08:00
|
|
|
|
.or_else(|| app.as_deref().map(|s| s.into()))
|
2025-08-31 16:39:38 +08:00
|
|
|
|
.or_else(|| appType.as_deref().map(|s| s.into()))
|
|
|
|
|
|
.unwrap_or(AppType::Claude);
|
2025-08-30 21:54:11 +08:00
|
|
|
|
|
|
|
|
|
|
let config_dir = match app_type {
|
|
|
|
|
|
AppType::Claude => crate::config::get_claude_config_dir(),
|
|
|
|
|
|
AppType::Codex => crate::codex_config::get_codex_config_dir(),
|
|
|
|
|
|
};
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
// 确保目录存在
|
|
|
|
|
|
if !config_dir.exists() {
|
2025-08-27 11:00:53 +08:00
|
|
|
|
std::fs::create_dir_all(&config_dir).map_err(|e| format!("创建目录失败: {}", e))?;
|
2025-08-23 20:15:10 +08:00
|
|
|
|
}
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-25 20:16:29 +08:00
|
|
|
|
// 使用 opener 插件打开文件夹
|
2025-08-31 18:14:31 +08:00
|
|
|
|
handle.opener()
|
2025-08-25 20:16:29 +08:00
|
|
|
|
.open_path(config_dir.to_string_lossy().to_string(), None::<String>)
|
|
|
|
|
|
.map_err(|e| format!("打开文件夹失败: {}", e))?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
Ok(true)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 打开外部链接
|
|
|
|
|
|
#[tauri::command]
|
2025-08-25 20:16:29 +08:00
|
|
|
|
pub async fn open_external(app: tauri::AppHandle, url: String) -> Result<bool, String> {
|
2025-08-24 23:35:07 +08:00
|
|
|
|
// 规范化 URL,缺少协议时默认加 https://
|
|
|
|
|
|
let url = if url.starts_with("http://") || url.starts_with("https://") {
|
|
|
|
|
|
url
|
|
|
|
|
|
} else {
|
|
|
|
|
|
format!("https://{}", url)
|
|
|
|
|
|
};
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-25 20:16:29 +08:00
|
|
|
|
// 使用 opener 插件打开链接
|
|
|
|
|
|
app.opener()
|
|
|
|
|
|
.open_url(&url, None::<String>)
|
|
|
|
|
|
.map_err(|e| format!("打开链接失败: {}", e))?;
|
2025-08-27 11:00:53 +08:00
|
|
|
|
|
2025-08-23 20:15:10 +08:00
|
|
|
|
Ok(true)
|
2025-08-24 23:30:35 +08:00
|
|
|
|
}
|