* feat(gemini): add Gemini provider integration - Add gemini_config.rs module for .env file parsing - Extend AppType enum to support Gemini - Implement GeminiConfigEditor and GeminiFormFields components - Add GeminiIcon with standardized 1024x1024 viewBox - Add Gemini provider presets configuration - Update i18n translations for Gemini support - Extend ProviderService and McpService for Gemini * fix(gemini): resolve TypeScript errors, add i18n support, and fix MCP logic **Critical Fixes:** - Fix TS2741 errors in tests/msw/state.ts by adding missing Gemini type definitions - Fix ProviderCard.extractApiUrl to support GOOGLE_GEMINI_BASE_URL display - Add missing apps.gemini i18n keys (zh/en) for proper app name display - Fix MCP service Gemini cross-app duplication logic to prevent self-copy **Technical Details:** - tests/msw/state.ts: Add gemini default providers, current ID, and MCP config - ProviderCard.tsx: Check both ANTHROPIC_BASE_URL and GOOGLE_GEMINI_BASE_URL - services/mcp.rs: Skip Gemini in sync_other_side logic with unreachable!() guards - Run pnpm format to auto-fix code style issues **Verification:** - ✅ pnpm typecheck passes - ✅ pnpm format completed * feat(gemini): enhance authentication and config parsing - Add strict and lenient .env parsing modes - Implement PackyCode partner authentication detection - Support Google OAuth official authentication - Auto-configure security.auth.selectedType for PackyCode - Add comprehensive test coverage for all auth types - Update i18n for OAuth hints and Gemini config --------- Co-authored-by: Jason <farion1231@gmail.com>
107 lines
2.9 KiB
Rust
107 lines
2.9 KiB
Rust
#![allow(non_snake_case)]
|
|
|
|
use serde_json::{json, Value};
|
|
use std::path::PathBuf;
|
|
use tauri::State;
|
|
use tauri_plugin_dialog::DialogExt;
|
|
|
|
use crate::error::AppError;
|
|
use crate::services::ConfigService;
|
|
use crate::store::AppState;
|
|
|
|
/// 导出配置文件
|
|
#[tauri::command]
|
|
pub async fn export_config_to_file(
|
|
#[allow(non_snake_case)] filePath: String,
|
|
) -> Result<Value, String> {
|
|
tauri::async_runtime::spawn_blocking(move || {
|
|
let target_path = PathBuf::from(&filePath);
|
|
ConfigService::export_config_to_path(&target_path)?;
|
|
Ok::<_, AppError>(json!({
|
|
"success": true,
|
|
"message": "Configuration exported successfully",
|
|
"filePath": filePath
|
|
}))
|
|
})
|
|
.await
|
|
.map_err(|e| format!("导出配置失败: {e}"))?
|
|
.map_err(|e: AppError| e.to_string())
|
|
}
|
|
|
|
/// 从文件导入配置
|
|
#[tauri::command]
|
|
pub async fn import_config_from_file(
|
|
#[allow(non_snake_case)] filePath: String,
|
|
state: State<'_, AppState>,
|
|
) -> Result<Value, String> {
|
|
let (new_config, backup_id) = tauri::async_runtime::spawn_blocking(move || {
|
|
let path_buf = PathBuf::from(&filePath);
|
|
ConfigService::load_config_for_import(&path_buf)
|
|
})
|
|
.await
|
|
.map_err(|e| format!("导入配置失败: {e}"))?
|
|
.map_err(|e: AppError| e.to_string())?;
|
|
|
|
{
|
|
let mut guard = state
|
|
.config
|
|
.write()
|
|
.map_err(|e| AppError::from(e).to_string())?;
|
|
*guard = new_config;
|
|
}
|
|
|
|
Ok(json!({
|
|
"success": true,
|
|
"message": "Configuration imported successfully",
|
|
"backupId": backup_id
|
|
}))
|
|
}
|
|
|
|
/// 同步当前供应商配置到对应的 live 文件
|
|
#[tauri::command]
|
|
pub async fn sync_current_providers_live(state: State<'_, AppState>) -> Result<Value, String> {
|
|
{
|
|
let mut config_state = state
|
|
.config
|
|
.write()
|
|
.map_err(|e| AppError::from(e).to_string())?;
|
|
ConfigService::sync_current_providers_to_live(&mut config_state)
|
|
.map_err(|e| e.to_string())?;
|
|
}
|
|
|
|
Ok(json!({
|
|
"success": true,
|
|
"message": "Live configuration synchronized"
|
|
}))
|
|
}
|
|
|
|
/// 保存文件对话框
|
|
#[tauri::command]
|
|
pub async fn save_file_dialog<R: tauri::Runtime>(
|
|
app: tauri::AppHandle<R>,
|
|
#[allow(non_snake_case)] defaultName: String,
|
|
) -> Result<Option<String>, String> {
|
|
let dialog = app.dialog();
|
|
let result = dialog
|
|
.file()
|
|
.add_filter("JSON", &["json"])
|
|
.set_file_name(&defaultName)
|
|
.blocking_save_file();
|
|
|
|
Ok(result.map(|p| p.to_string()))
|
|
}
|
|
|
|
/// 打开文件对话框
|
|
#[tauri::command]
|
|
pub async fn open_file_dialog<R: tauri::Runtime>(
|
|
app: tauri::AppHandle<R>,
|
|
) -> Result<Option<String>, String> {
|
|
let dialog = app.dialog();
|
|
let result = dialog
|
|
.file()
|
|
.add_filter("JSON", &["json"])
|
|
.blocking_pick_file();
|
|
|
|
Ok(result.map(|p| p.to_string()))
|
|
}
|