use std::sync::RwLock; use cc_switch_lib::{ import_provider_from_deeplink, parse_deeplink_url, AppState, AppType, MultiAppConfig, }; #[path = "support.rs"] mod support; use support::{ensure_test_home, reset_test_fs, test_mutex}; #[test] fn deeplink_import_claude_provider_persists_to_config() { let _guard = test_mutex().lock().expect("acquire test mutex"); reset_test_fs(); let home = ensure_test_home(); let url = "ccswitch://v1/import?resource=provider&app=claude&name=DeepLink%20Claude&homepage=https%3A%2F%2Fexample.com&endpoint=https%3A%2F%2Fapi.example.com%2Fv1&apiKey=sk-test-claude-key&model=claude-sonnet-4"; let request = parse_deeplink_url(url).expect("parse deeplink url"); let mut config = MultiAppConfig::default(); config.ensure_app(&AppType::Claude); let state = AppState { config: RwLock::new(config), }; let provider_id = import_provider_from_deeplink(&state, request.clone()) .expect("import provider from deeplink"); // 验证内存状态 let guard = state.config.read().expect("read config"); let manager = guard .get_manager(&AppType::Claude) .expect("claude manager should exist"); let provider = manager .providers .get(&provider_id) .expect("provider created via deeplink"); assert_eq!(provider.name, request.name); assert_eq!( provider.website_url.as_deref(), Some(request.homepage.as_str()) ); let auth_token = provider .settings_config .pointer("/env/ANTHROPIC_AUTH_TOKEN") .and_then(|v| v.as_str()); let base_url = provider .settings_config .pointer("/env/ANTHROPIC_BASE_URL") .and_then(|v| v.as_str()); assert_eq!(auth_token, Some(request.api_key.as_str())); assert_eq!(base_url, Some(request.endpoint.as_str())); drop(guard); // 验证配置已持久化 let config_path = home.join(".cc-switch").join("config.json"); assert!( config_path.exists(), "importing provider from deeplink should persist config.json" ); } #[test] fn deeplink_import_codex_provider_builds_auth_and_config() { let _guard = test_mutex().lock().expect("acquire test mutex"); reset_test_fs(); let home = ensure_test_home(); let url = "ccswitch://v1/import?resource=provider&app=codex&name=DeepLink%20Codex&homepage=https%3A%2F%2Fopenai.example&endpoint=https%3A%2F%2Fapi.openai.example%2Fv1&apiKey=sk-test-codex-key&model=gpt-4o"; let request = parse_deeplink_url(url).expect("parse deeplink url"); let mut config = MultiAppConfig::default(); config.ensure_app(&AppType::Codex); let state = AppState { config: RwLock::new(config), }; let provider_id = import_provider_from_deeplink(&state, request.clone()) .expect("import provider from deeplink"); let guard = state.config.read().expect("read config"); let manager = guard .get_manager(&AppType::Codex) .expect("codex manager should exist"); let provider = manager .providers .get(&provider_id) .expect("provider created via deeplink"); assert_eq!(provider.name, request.name); assert_eq!( provider.website_url.as_deref(), Some(request.homepage.as_str()) ); let auth_value = provider .settings_config .pointer("/auth/OPENAI_API_KEY") .and_then(|v| v.as_str()); let config_text = provider .settings_config .get("config") .and_then(|v| v.as_str()) .unwrap_or_default(); assert_eq!(auth_value, Some(request.api_key.as_str())); assert!( config_text.contains(request.endpoint.as_str()), "config.toml content should contain endpoint" ); assert!( config_text.contains("model = \"gpt-4o\""), "config.toml content should contain model setting" ); drop(guard); let config_path = home.join(".cc-switch").join("config.json"); assert!( config_path.exists(), "importing provider from deeplink should persist config.json" ); }