Revert "feat: add VS Code ChatGPT plugin config sync functionality"

This reverts commit 9bf216b102.
This commit is contained in:
Jason
2025-09-18 17:57:32 +08:00
parent 60eb9ce2a4
commit c6e4f3599e
7 changed files with 1 additions and 272 deletions

View File

@@ -9,7 +9,6 @@ use crate::codex_config;
use crate::config::{get_claude_settings_path, ConfigStatus};
use crate::provider::Provider;
use crate::store::AppState;
use crate::vscode_config;
fn validate_provider_settings(app_type: &AppType, provider: &Provider) -> Result<(), String> {
match app_type {
@@ -42,44 +41,6 @@ fn validate_provider_settings(app_type: &AppType, provider: &Provider) -> Result
Ok(())
}
fn extract_base_url_from_toml(cfg_text: &str) -> Result<Option<String>, String> {
if cfg_text.trim().is_empty() {
return Ok(None);
}
let value: toml::Value =
toml::from_str(cfg_text).map_err(|e| format!("解析 config.toml 失败: {}", e))?;
fn walk(value: &toml::Value) -> Option<String> {
match value {
toml::Value::Table(table) => {
if let Some(toml::Value::String(v)) = table.get("base_url") {
if !v.trim().is_empty() {
return Some(v.clone());
}
}
for item in table.values() {
if let Some(found) = walk(item) {
return Some(found);
}
}
None
}
toml::Value::Array(arr) => {
for item in arr {
if let Some(found) = walk(item) {
return Some(found);
}
}
None
}
_ => None,
}
}
Ok(walk(&value))
}
/// 获取所有供应商
#[tauri::command]
pub async fn get_providers(
@@ -399,26 +360,6 @@ pub async fn switch_provider(
.get("config")
.and_then(|v| v.as_str());
crate::codex_config::write_codex_live_atomic(auth, cfg_text)?;
let is_official = provider
.category
.as_ref()
.map(|c| c == "official")
.unwrap_or(false);
if is_official {
vscode_config::write_vscode_settings(None)?;
} else {
let cfg_text = cfg_text.unwrap_or_default();
match extract_base_url_from_toml(cfg_text)? {
Some(base_url) => {
vscode_config::write_vscode_settings(Some(&base_url))?;
}
None => {
return Err("目标 Codex 配置缺少 base_url 字段".to_string());
}
}
}
}
AppType::Claude => {
use crate::config::{read_json_file, write_json_file};
@@ -628,17 +569,6 @@ pub async fn open_external(app: tauri::AppHandle, url: String) -> Result<bool, S
Ok(true)
}
/// 写入 VS Code 配置
#[tauri::command]
pub async fn write_vscode_settings_command(
base_url: Option<String>,
baseUrl: Option<String>,
) -> Result<bool, String> {
let payload = base_url.or(baseUrl);
vscode_config::write_vscode_settings(payload.as_deref())?;
Ok(true)
}
/// 获取应用配置文件路径
#[tauri::command]
pub async fn get_app_config_path() -> Result<String, String> {

View File

@@ -5,7 +5,6 @@ mod config;
mod migration;
mod provider;
mod store;
mod vscode_config;
use store::AppState;
#[cfg(target_os = "macos")]
@@ -353,7 +352,6 @@ pub fn run() {
commands::get_claude_code_config_path,
commands::open_config_folder,
commands::open_external,
commands::write_vscode_settings_command,
commands::get_app_config_path,
commands::open_app_config_folder,
commands::get_settings,

View File

@@ -1,115 +0,0 @@
use serde_json::{Map, Value};
use std::path::{Path, PathBuf};
use crate::config::write_json_file;
/// VS Code 默认用户配置子目录
const MAC_CODE_USER_DIR: &str = "Library/Application Support/Code/User";
/// 解析 VS Code 用户 settings.json 路径
pub fn get_vscode_settings_path() -> PathBuf {
#[cfg(target_os = "macos")]
{
return dirs::home_dir()
.expect("无法获取用户主目录")
.join(MAC_CODE_USER_DIR)
.join("settings.json");
}
#[cfg(target_os = "linux")]
{
return dirs::home_dir()
.expect("无法获取用户主目录")
.join(".config/Code/User")
.join("settings.json");
}
#[cfg(target_os = "windows")]
{
if let Some(data_dir) = dirs::data_dir() {
return data_dir.join("Code/User").join("settings.json");
}
return dirs::home_dir()
.expect("无法获取用户主目录")
.join("AppData/Roaming")
.join("Code/User")
.join("settings.json");
}
#[cfg(not(any(target_os = "macos", target_os = "linux", target_os = "windows")))]
{
dirs::home_dir()
.expect("无法获取用户主目录")
.join(".config/Code/User")
.join("settings.json")
}
}
fn load_settings(path: &Path) -> Result<Map<String, Value>, String> {
if !path.exists() {
return Ok(Map::new());
}
let content =
std::fs::read_to_string(path).map_err(|e| format!("读取 VS Code 设置失败: {}", e))?;
if content.trim().is_empty() {
return Ok(Map::new());
}
match serde_json::from_str::<Value>(&content) {
Ok(Value::Object(obj)) => Ok(obj),
Ok(_) => Err("VS Code settings.json 必须为 JSON 对象".to_string()),
Err(err) => Err(format!("解析 VS Code settings.json 失败: {}", err)),
}
}
fn persist_settings(path: &Path, map: Map<String, Value>) -> Result<(), String> {
let value = Value::Object(map);
if let Some(parent) = path.parent() {
std::fs::create_dir_all(parent).map_err(|e| format!("创建 VS Code 配置目录失败: {}", e))?;
}
write_json_file(path, &value)
}
/// 写入或移除 chatgpt 相关 VS Code 配置
///
/// - `base_url` 为 Some 时更新/覆盖 `"chatgpt.apiBase"` 与 `"chatgpt.config"`
/// - `base_url` 为 None 时删除上述字段
pub fn write_vscode_settings(base_url: Option<&str>) -> Result<(), String> {
let path = get_vscode_settings_path();
let mut map = load_settings(&path)?;
match base_url {
Some(url) => {
if url.trim().is_empty() {
return Err("base_url 不能为空".into());
}
map.insert(
"chatgpt.apiBase".to_string(),
Value::String(url.to_string()),
);
let entry = map
.entry("chatgpt.config".to_string())
.or_insert_with(|| Value::Object(Map::new()));
let obj = match entry {
Value::Object(o) => o,
_ => return Err("VS Code settings 中 chatgpt.config 必须是 JSON 对象".into()),
};
obj.insert(
"preferred_auth_method".to_string(),
Value::String("apikey".to_string()),
);
}
None => {
map.remove("chatgpt.apiBase");
map.remove("chatgpt.config");
}
}
persist_settings(&path, map)
}

View File

@@ -1,6 +1,5 @@
import React, { useState, useEffect } from "react";
import { X, Save } from "lucide-react";
import { extractBaseUrlFromToml } from "../../utils/providerConfigUtils";
interface CodexConfigEditorProps {
authValue: string;
@@ -30,9 +29,6 @@ const CodexConfigEditor: React.FC<CodexConfigEditorProps> = ({
authError,
}) => {
const [isCommonConfigModalOpen, setIsCommonConfigModalOpen] = useState(false);
const [isWritingVscode, setIsWritingVscode] = useState(false);
const [vscodeError, setVscodeError] = useState("");
const [vscodeSuccess, setVscodeSuccess] = useState("");
useEffect(() => {
if (commonConfigError && !isCommonConfigModalOpen) {
@@ -40,14 +36,6 @@ const CodexConfigEditor: React.FC<CodexConfigEditorProps> = ({
}
}, [commonConfigError, isCommonConfigModalOpen]);
useEffect(() => {
if (!vscodeSuccess) return;
const timer = window.setTimeout(() => {
setVscodeSuccess("");
}, 3000);
return () => window.clearTimeout(timer);
}, [vscodeSuccess]);
// 支持按下 ESC 关闭弹窗
useEffect(() => {
if (!isCommonConfigModalOpen) return;
@@ -78,42 +66,6 @@ const CodexConfigEditor: React.FC<CodexConfigEditorProps> = ({
onCommonConfigSnippetChange(value);
};
const handleWriteVscodeConfig = async () => {
setVscodeError("");
setVscodeSuccess("");
if (typeof window === "undefined" || !window.api?.writeVscodeSettings) {
setVscodeError("当前环境暂不支持写入 VS Code 配置");
return;
}
const trimmed = configValue.trim();
if (!trimmed) {
setVscodeError("请先填写 config.toml再写入 VS Code 配置");
return;
}
const baseUrl = extractBaseUrlFromToml(trimmed);
if (!baseUrl) {
setVscodeError("未在 config.toml 中找到 base_url 字段");
return;
}
setIsWritingVscode(true);
try {
const success = await window.api.writeVscodeSettings(baseUrl);
if (success) {
setVscodeSuccess("已写入 VS Code 配置");
} else {
setVscodeError("写入 VS Code 配置失败,请稍后重试");
}
} catch (error) {
setVscodeError(`写入 VS Code 配置失败: ${String(error)}`);
} finally {
setIsWritingVscode(false);
}
};
return (
<div className="space-y-6">
<div className="space-y-2">
@@ -172,15 +124,7 @@ const CodexConfigEditor: React.FC<CodexConfigEditorProps> = ({
</label>
</div>
<div className="flex items-center justify-end gap-3">
<button
type="button"
onClick={handleWriteVscodeConfig}
disabled={isWritingVscode}
className="text-xs text-blue-500 dark:text-blue-400 hover:underline disabled:opacity-60 disabled:cursor-not-allowed"
>
{isWritingVscode ? "写入中..." : "写入 VS Code 配置"}
</button>
<div className="flex items-center justify-end">
<button
type="button"
onClick={() => setIsCommonConfigModalOpen(true)}
@@ -194,16 +138,6 @@ const CodexConfigEditor: React.FC<CodexConfigEditorProps> = ({
{commonConfigError}
</p>
)}
{vscodeError && (
<p className="text-xs text-red-500 dark:text-red-400 text-right">
{vscodeError}
</p>
)}
{vscodeSuccess && !vscodeError && (
<p className="text-xs text-emerald-600 dark:text-emerald-400 text-right">
{vscodeSuccess}
</p>
)}
<textarea
id="codexConfig"
value={configValue}

View File

@@ -159,16 +159,6 @@ export const tauriAPI = {
}
},
// 写入 VS Code 配置
writeVscodeSettings: async (baseUrl?: string): Promise<boolean> => {
try {
return await invoke("write_vscode_settings_command", { baseUrl });
} catch (error) {
console.error("写入 VS Code 配置失败:", error);
return false;
}
},
// 打开外部链接
openExternal: async (url: string): Promise<void> => {
try {

View File

@@ -288,10 +288,3 @@ export const hasTomlCommonConfigSnippet = (
return existingSnippet === snippetString.trim();
};
// 从 Codex TOML 配置中提取 base_url
export const extractBaseUrlFromToml = (tomlString: string): string => {
if (!tomlString) return "";
const match = tomlString.match(/base_url\s*=\s*"([^"]+)"/);
return match?.[1] ?? "";
};

1
src/vite-env.d.ts vendored
View File

@@ -30,7 +30,6 @@ declare global {
getConfigStatus: (app?: AppType) => Promise<ConfigStatus>;
selectConfigFile: () => Promise<string | null>;
openConfigFolder: (app?: AppType) => Promise<void>;
writeVscodeSettings: (baseUrl?: string) => Promise<boolean>;
openExternal: (url: string) => Promise<void>;
updateTrayMenu: () => Promise<boolean>;
onProviderSwitched: (