aiclient-2-api / static /app /upload-config-manager.js
Jaasomn
Initial deployment
ceb3821
// 配置管理功能模块
import { showToast } from './utils.js';
import { t } from './i18n.js';
let allConfigs = []; // 存储所有配置数据
let filteredConfigs = []; // 存储过滤后的配置数据
let isLoadingConfigs = false; // 防止重复加载配置
/**
* 搜索配置
* @param {string} searchTerm - 搜索关键词
* @param {string} statusFilter - 状态过滤
*/
function searchConfigs(searchTerm = '', statusFilter = '', providerFilter = '') {
if (!allConfigs.length) {
console.log('没有配置数据可搜索');
return;
}
filteredConfigs = allConfigs.filter(config => {
// 搜索过滤
const matchesSearch = !searchTerm ||
config.name.toLowerCase().includes(searchTerm.toLowerCase()) ||
config.path.toLowerCase().includes(searchTerm.toLowerCase()) ||
(config.content && config.content.toLowerCase().includes(searchTerm.toLowerCase()));
// 状态过滤 - 从布尔值 isUsed 转换为状态字符串
const configStatus = config.isUsed ? 'used' : 'unused';
const matchesStatus = !statusFilter || configStatus === statusFilter;
// 提供商类型过滤
let matchesProvider = true;
if (providerFilter) {
const providerInfo = detectProviderFromPath(config.path);
if (providerFilter === 'other') {
// "其他/未识别" 选项:匹配没有识别到提供商的配置
matchesProvider = providerInfo === null;
} else {
// 匹配特定提供商类型
matchesProvider = providerInfo !== null && providerInfo.providerType === providerFilter;
}
}
return matchesSearch && matchesStatus && matchesProvider;
});
renderConfigList();
updateStats();
}
/**
* 渲染配置列表
*/
function renderConfigList() {
const container = document.getElementById('configList');
if (!container) return;
container.innerHTML = '';
if (!filteredConfigs.length) {
container.innerHTML = `<div class="no-configs"><p data-i18n="upload.noConfigs">${t('upload.noConfigs')}</p></div>`;
return;
}
filteredConfigs.forEach((config, index) => {
const configItem = createConfigItemElement(config, index);
container.appendChild(configItem);
});
}
/**
* 创建配置项元素
* @param {Object} config - 配置数据
* @param {number} index - 索引
* @returns {HTMLElement} 配置项元素
*/
function createConfigItemElement(config, index) {
// 从布尔值 isUsed 转换为状态字符串用于显示
const configStatus = config.isUsed ? 'used' : 'unused';
const item = document.createElement('div');
item.className = `config-item-manager ${configStatus}`;
item.dataset.index = index;
const statusIcon = config.isUsed ? 'fa-check-circle' : 'fa-circle';
const statusText = config.isUsed ? t('upload.statusFilter.used') : t('upload.statusFilter.unused');
const typeIcon = config.type === 'oauth' ? 'fa-key' :
config.type === 'api-key' ? 'fa-lock' :
config.type === 'provider-pool' ? 'fa-network-wired' :
config.type === 'system-prompt' ? 'fa-file-text' : 'fa-cog';
// 生成关联详情HTML
const usageInfoHtml = generateUsageInfoHtml(config);
// 判断是否可以一键关联(未关联且路径包含支持的提供商目录)
const providerInfo = detectProviderFromPath(config.path);
const canQuickLink = !config.isUsed && providerInfo !== null;
const quickLinkBtnHtml = canQuickLink ?
`<button class="btn-quick-link" data-path="${config.path}" title="一键关联到 ${providerInfo.displayName}">
<i class="fas fa-link"></i> ${providerInfo.shortName}
</button>` : '';
item.innerHTML = `
<div class="config-item-header">
<div class="config-item-name">${config.name}</div>
<div class="config-item-path" title="${config.path}">${config.path}</div>
</div>
<div class="config-item-meta">
<div class="config-item-size">${formatFileSize(config.size)}</div>
<div class="config-item-modified">${formatDate(config.modified)}</div>
<div class="config-item-status">
<i class="fas ${statusIcon}"></i>
<span data-i18n="${config.isUsed ? 'upload.statusFilter.used' : 'upload.statusFilter.unused'}">${statusText}</span>
${quickLinkBtnHtml}
</div>
</div>
<div class="config-item-details">
<div class="config-details-grid">
<div class="config-detail-item">
<div class="config-detail-label" data-i18n="upload.detail.path">文件路径</div>
<div class="config-detail-value">${config.path}</div>
</div>
<div class="config-detail-item">
<div class="config-detail-label" data-i18n="upload.detail.size">文件大小</div>
<div class="config-detail-value">${formatFileSize(config.size)}</div>
</div>
<div class="config-detail-item">
<div class="config-detail-label" data-i18n="upload.detail.modified">最后修改</div>
<div class="config-detail-value">${formatDate(config.modified)}</div>
</div>
<div class="config-detail-item">
<div class="config-detail-label" data-i18n="upload.detail.status">关联状态</div>
<div class="config-detail-value" data-i18n="${config.isUsed ? 'upload.statusFilter.used' : 'upload.statusFilter.unused'}">${statusText}</div>
</div>
</div>
${usageInfoHtml}
<div class="config-item-actions">
<button class="btn-small btn-view" data-path="${config.path}">
<i class="fas fa-eye"></i> <span data-i18n="upload.action.view">${t('upload.action.view')}</span>
</button>
<button class="btn-small btn-delete-small" data-path="${config.path}">
<i class="fas fa-trash"></i> <span data-i18n="upload.action.delete">${t('upload.action.delete')}</span>
</button>
</div>
</div>
`;
// 添加按钮事件监听器
const viewBtn = item.querySelector('.btn-view');
const deleteBtn = item.querySelector('.btn-delete-small');
if (viewBtn) {
viewBtn.addEventListener('click', (e) => {
e.stopPropagation();
viewConfig(config.path);
});
}
if (deleteBtn) {
deleteBtn.addEventListener('click', (e) => {
e.stopPropagation();
deleteConfig(config.path);
});
}
// 一键关联按钮事件
const quickLinkBtn = item.querySelector('.btn-quick-link');
if (quickLinkBtn) {
quickLinkBtn.addEventListener('click', (e) => {
e.stopPropagation();
quickLinkProviderConfig(config.path);
});
}
// 添加点击事件展开/折叠详情
item.addEventListener('click', (e) => {
if (!e.target.closest('.config-item-actions')) {
item.classList.toggle('expanded');
}
});
return item;
}
/**
* 生成关联详情HTML
* @param {Object} config - 配置数据
* @returns {string} HTML字符串
*/
function generateUsageInfoHtml(config) {
if (!config.usageInfo || !config.usageInfo.isUsed) {
return '';
}
const { usageType, usageDetails } = config.usageInfo;
if (!usageDetails || usageDetails.length === 0) {
return '';
}
const typeLabels = {
'main_config': t('upload.usage.mainConfig'),
'provider_pool': t('upload.usage.providerPool'),
'multiple': t('upload.usage.multiple')
};
const typeLabel = typeLabels[usageType] || (t('common.info') === 'Info' ? 'Unknown' : '未知用途');
let detailsHtml = '';
usageDetails.forEach(detail => {
const isMain = detail.type === '主要配置' || detail.type === 'Main Config';
const icon = isMain ? 'fa-cog' : 'fa-network-wired';
const usageTypeKey = isMain ? 'main_config' : 'provider_pool';
detailsHtml += `
<div class="usage-detail-item" data-usage-type="${usageTypeKey}">
<i class="fas ${icon}"></i>
<span class="usage-detail-type">${detail.type}</span>
<span class="usage-detail-location">${detail.location}</span>
</div>
`;
});
return `
<div class="config-usage-info">
<div class="usage-info-header">
<i class="fas fa-link"></i>
<span class="usage-info-title" data-i18n="upload.usage.title" data-i18n-params='{"type":"${typeLabel}"}'>关联详情 (${typeLabel})</span>
</div>
<div class="usage-details-list">
${detailsHtml}
</div>
</div>
`;
}
/**
* 格式化文件大小
* @param {number} bytes - 字节数
* @returns {string} 格式化后的大小
*/
function formatFileSize(bytes) {
if (bytes === 0) return '0 B';
const k = 1024;
const sizes = ['B', 'KB', 'MB', 'GB'];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
}
/**
* 格式化日期
* @param {string} dateString - 日期字符串
* @returns {string} 格式化后的日期
*/
function formatDate(dateString) {
const date = new Date(dateString);
return date.toLocaleString('zh-CN', {
year: 'numeric',
month: '2-digit',
day: '2-digit',
hour: '2-digit',
minute: '2-digit'
});
}
/**
* 更新统计信息
*/
function updateStats() {
const totalCount = filteredConfigs.length;
const usedCount = filteredConfigs.filter(config => config.isUsed).length;
const unusedCount = filteredConfigs.filter(config => !config.isUsed).length;
const totalEl = document.getElementById('configCount');
const usedEl = document.getElementById('usedConfigCount');
const unusedEl = document.getElementById('unusedConfigCount');
if (totalEl) {
totalEl.textContent = t('upload.count', { count: totalCount });
totalEl.setAttribute('data-i18n-params', JSON.stringify({ count: totalCount.toString() }));
}
if (usedEl) {
usedEl.textContent = t('upload.usedCount', { count: usedCount });
usedEl.setAttribute('data-i18n-params', JSON.stringify({ count: usedCount.toString() }));
}
if (unusedEl) {
unusedEl.textContent = t('upload.unusedCount', { count: unusedCount });
unusedEl.setAttribute('data-i18n-params', JSON.stringify({ count: unusedCount.toString() }));
}
}
/**
* 加载配置文件列表
*/
async function loadConfigList() {
// 防止重复加载
if (isLoadingConfigs) {
console.log('正在加载配置列表,跳过重复调用');
return;
}
isLoadingConfigs = true;
console.log('开始加载配置列表...');
try {
const result = await window.apiClient.get('/upload-configs');
allConfigs = result;
filteredConfigs = [...allConfigs];
renderConfigList();
updateStats();
console.log('配置列表加载成功,共', allConfigs.length, '个项目');
// showToast(t('common.success'), t('upload.refresh') + '成功', 'success');
} catch (error) {
console.error('加载配置列表失败:', error);
showToast(t('common.error'), t('common.error') + ': ' + error.message, 'error');
// 使用模拟数据作为示例
allConfigs = generateMockConfigData();
filteredConfigs = [...allConfigs];
renderConfigList();
updateStats();
} finally {
isLoadingConfigs = false;
console.log('配置列表加载完成');
}
}
/**
* 生成模拟配置数据(用于演示)
* @returns {Array} 模拟配置数据
*/
function generateMockConfigData() {
return [
{
name: 'provider_pools.json',
path: './configs/provider_pools.json',
type: 'provider-pool',
size: 2048,
modified: '2025-11-11T04:30:00.000Z',
isUsed: true,
content: JSON.stringify({
"gemini-cli-oauth": [
{
"GEMINI_OAUTH_CREDS_FILE_PATH": "~/.gemini/oauth/creds.json",
"PROJECT_ID": "test-project"
}
]
}, null, 2)
},
{
name: 'config.json',
path: './configs/config.json',
type: 'other',
size: 1024,
modified: '2025-11-10T12:00:00.000Z',
isUsed: true,
content: JSON.stringify({
"REQUIRED_API_KEY": "123456",
"SERVER_PORT": 3000
}, null, 2)
},
{
name: 'oauth_creds.json',
path: '~/.gemini/oauth/creds.json',
type: 'oauth',
size: 512,
modified: '2025-11-09T08:30:00.000Z',
isUsed: false,
content: '{"client_id": "test", "client_secret": "test"}'
},
{
name: 'input_system_prompt.txt',
path: './configs/input_system_prompt.txt',
type: 'system-prompt',
size: 256,
modified: '2025-11-08T15:20:00.000Z',
isUsed: true,
content: '你是一个有用的AI助手...'
},
{
name: 'invalid_config.json',
path: './invalid_config.json',
type: 'other',
size: 128,
modified: '2025-11-07T10:15:00.000Z',
isUsed: false,
content: '{"invalid": json}'
}
];
}
/**
* 查看配置
* @param {string} path - 文件路径
*/
async function viewConfig(path) {
try {
const fileData = await window.apiClient.get(`/upload-configs/view/${encodeURIComponent(path)}`);
showConfigModal(fileData);
} catch (error) {
console.error('查看配置失败:', error);
showToast(t('common.error'), t('upload.action.view.failed') + ': ' + error.message, 'error');
}
}
/**
* 显示配置模态框
* @param {Object} fileData - 文件数据
*/
function showConfigModal(fileData) {
// 创建模态框
const modal = document.createElement('div');
modal.className = 'config-view-modal';
modal.innerHTML = `
<div class="config-modal-content">
<div class="config-modal-header">
<h3><span data-i18n="nav.config">${t('nav.config')}</span>: ${fileData.name}</h3>
<button class="modal-close">
<i class="fas fa-times"></i>
</button>
</div>
<div class="config-modal-body">
<div class="config-file-info">
<div class="file-info-item">
<span class="info-label" data-i18n="upload.detail.path">${t('upload.detail.path')}:</span>
<span class="info-value">${fileData.path}</span>
</div>
<div class="file-info-item">
<span class="info-label" data-i18n="upload.detail.size">${t('upload.detail.size')}:</span>
<span class="info-value">${formatFileSize(fileData.size)}</span>
</div>
<div class="file-info-item">
<span class="info-label" data-i18n="upload.detail.modified">${t('upload.detail.modified')}:</span>
<span class="info-value">${formatDate(fileData.modified)}</span>
</div>
</div>
<div class="config-content">
<label data-i18n="common.info">文件内容:</label>
<pre class="config-content-display">${escapeHtml(fileData.content)}</pre>
</div>
</div>
<div class="config-modal-footer">
<button class="btn btn-secondary btn-close-modal" data-i18n="modal.provider.cancel">${t('modal.provider.cancel')}</button>
<button class="btn btn-primary btn-copy-content" data-path="${fileData.path}">
<i class="fas fa-copy"></i> <span data-i18n="oauth.modal.copyTitle">${t('oauth.modal.copyTitle')}</span>
</button>
</div>
</div>
`;
// 添加到页面
document.body.appendChild(modal);
// 添加按钮事件监听器
const closeBtn = modal.querySelector('.btn-close-modal');
const copyBtn = modal.querySelector('.btn-copy-content');
const modalCloseBtn = modal.querySelector('.modal-close');
if (closeBtn) {
closeBtn.addEventListener('click', () => {
closeConfigModal();
});
}
if (copyBtn) {
copyBtn.addEventListener('click', () => {
const path = copyBtn.dataset.path;
copyConfigContent(path);
});
}
if (modalCloseBtn) {
modalCloseBtn.addEventListener('click', () => {
closeConfigModal();
});
}
// 显示模态框
setTimeout(() => modal.classList.add('show'), 10);
}
/**
* 关闭配置模态框
*/
function closeConfigModal() {
const modal = document.querySelector('.config-view-modal');
if (modal) {
modal.classList.remove('show');
setTimeout(() => modal.remove(), 300);
}
}
/**
* 复制配置内容
* @param {string} path - 文件路径
*/
async function copyConfigContent(path) {
try {
const fileData = await window.apiClient.get(`/upload-configs/view/${encodeURIComponent(path)}`);
// 尝试使用现代 Clipboard API
if (navigator.clipboard && navigator.clipboard.writeText) {
await navigator.clipboard.writeText(fileData.content);
showToast(t('common.success'), t('oauth.success.msg'), 'success');
} else {
// 降级方案:使用传统的 document.execCommand
const textarea = document.createElement('textarea');
textarea.value = fileData.content;
textarea.style.position = 'fixed';
textarea.style.opacity = '0';
document.body.appendChild(textarea);
textarea.select();
try {
const successful = document.execCommand('copy');
if (successful) {
showToast(t('common.copy.success'), 'success');
} else {
showToast(t('common.copy.failed'), 'error');
}
} catch (err) {
console.error('复制失败:', err);
showToast(t('common.copy.failed'), 'error');
} finally {
document.body.removeChild(textarea);
}
}
} catch (error) {
console.error('复制失败:', error);
showToast(t('common.copy.failed') + ': ' + error.message, 'error');
}
}
/**
* HTML转义
* @param {string} text - 要转义的文本
* @returns {string} 转义后的文本
*/
function escapeHtml(text) {
const div = document.createElement('div');
div.textContent = text;
return div.innerHTML;
}
/**
* 显示删除确认模态框
* @param {Object} config - 配置数据
*/
function showDeleteConfirmModal(config) {
const isUsed = config.isUsed;
const modalClass = isUsed ? 'delete-confirm-modal used' : 'delete-confirm-modal unused';
const title = isUsed ? t('upload.delete.confirmTitleUsed') : t('upload.delete.confirmTitle');
const icon = isUsed ? 'fas fa-exclamation-triangle' : 'fas fa-trash';
const buttonClass = isUsed ? 'btn btn-danger' : 'btn btn-warning';
const modal = document.createElement('div');
modal.className = modalClass;
modal.innerHTML = `
<div class="delete-modal-content">
<div class="delete-modal-header">
<h3 data-i18n="${isUsed ? 'upload.delete.confirmTitleUsed' : 'upload.delete.confirmTitle'}"><i class="${icon}"></i> ${title}</h3>
<button class="modal-close">
<i class="fas fa-times"></i>
</button>
</div>
<div class="delete-modal-body">
<div class="delete-warning ${isUsed ? 'warning-used' : 'warning-unused'}">
<div class="warning-icon">
<i class="${icon}"></i>
</div>
<div class="warning-content">
${isUsed ?
`<h4 data-i18n="upload.delete.warningUsedTitle">${t('upload.delete.warningUsedTitle')}</h4><p data-i18n="upload.delete.warningUsedDesc">${t('upload.delete.warningUsedDesc')}</p>` :
`<h4 data-i18n="upload.delete.warningUnusedTitle">${t('upload.delete.warningUnusedTitle')}</h4><p data-i18n="upload.delete.warningUnusedDesc">${t('upload.delete.warningUnusedDesc')}</p>`
}
</div>
</div>
<div class="config-info">
<div class="config-info-item">
<span class="info-label" data-i18n="upload.delete.fileName">文件名:</span>
<span class="info-value">${config.name}</span>
</div>
<div class="config-info-item">
<span class="info-label" data-i18n="upload.detail.path">文件路径:</span>
<span class="info-value">${config.path}</span>
</div>
<div class="config-info-item">
<span class="info-label" data-i18n="upload.detail.size">文件大小:</span>
<span class="info-value">${formatFileSize(config.size)}</span>
</div>
<div class="config-info-item">
<span class="info-label" data-i18n="upload.detail.status">关联状态:</span>
<span class="info-value status-${isUsed ? 'used' : 'unused'}" data-i18n="${isUsed ? 'upload.statusFilter.used' : 'upload.statusFilter.unused'}">
${isUsed ? t('upload.statusFilter.used') : t('upload.statusFilter.unused')}
</span>
</div>
</div>
${isUsed ? `
<div class="usage-alert">
<div class="alert-icon">
<i class="fas fa-info-circle"></i>
</div>
<div class="alert-content">
<h5 data-i18n="upload.delete.usageAlertTitle">${t('upload.delete.usageAlertTitle')}</h5>
<p data-i18n="upload.delete.usageAlertDesc">${t('upload.delete.usageAlertDesc')}</p>
<ul>
<li data-i18n="upload.delete.usageAlertItem1">${t('upload.delete.usageAlertItem1')}</li>
<li data-i18n="upload.delete.usageAlertItem2">${t('upload.delete.usageAlertItem2')}</li>
<li data-i18n="upload.delete.usageAlertItem3">${t('upload.delete.usageAlertItem3')}</li>
</ul>
<p data-i18n-html="upload.delete.usageAlertAdvice">${t('upload.delete.usageAlertAdvice')}</p>
</div>
</div>
` : ''}
</div>
<div class="delete-modal-footer">
<button class="btn btn-secondary btn-cancel-delete" data-i18n="modal.provider.cancel">${t('modal.provider.cancel')}</button>
<button class="${buttonClass} btn-confirm-delete" data-path="${config.path}">
<i class="fas fa-${isUsed ? 'exclamation-triangle' : 'trash'}"></i>
<span data-i18n="${isUsed ? 'upload.delete.forceDelete' : 'upload.delete.confirmDelete'}">${isUsed ? t('upload.delete.forceDelete') : t('upload.delete.confirmDelete')}</span>
</button>
</div>
</div>
`;
// 添加到页面
document.body.appendChild(modal);
// 添加事件监听器
const closeBtn = modal.querySelector('.modal-close');
const cancelBtn = modal.querySelector('.btn-cancel-delete');
const confirmBtn = modal.querySelector('.btn-confirm-delete');
const closeModal = () => {
modal.classList.remove('show');
setTimeout(() => modal.remove(), 300);
};
if (closeBtn) {
closeBtn.addEventListener('click', closeModal);
}
if (cancelBtn) {
cancelBtn.addEventListener('click', closeModal);
}
if (confirmBtn) {
confirmBtn.addEventListener('click', () => {
const path = confirmBtn.dataset.path;
performDelete(path);
closeModal();
});
}
// 点击外部关闭
modal.addEventListener('click', (e) => {
if (e.target === modal) {
closeModal();
}
});
// ESC键关闭
const handleEsc = (e) => {
if (e.key === 'Escape') {
closeModal();
document.removeEventListener('keydown', handleEsc);
}
};
document.addEventListener('keydown', handleEsc);
// 显示模态框
setTimeout(() => modal.classList.add('show'), 10);
}
/**
* 执行删除操作
* @param {string} path - 文件路径
*/
async function performDelete(path) {
try {
const result = await window.apiClient.delete(`/upload-configs/delete/${encodeURIComponent(path)}`);
showToast(t('common.success'), result.message, 'success');
// 从本地列表中移除
allConfigs = allConfigs.filter(c => c.path !== path);
filteredConfigs = filteredConfigs.filter(c => c.path !== path);
renderConfigList();
updateStats();
} catch (error) {
console.error('删除配置失败:', error);
showToast(t('common.error'), t('upload.action.delete.failed') + ': ' + error.message, 'error');
}
}
/**
* 删除配置
* @param {string} path - 文件路径
*/
async function deleteConfig(path) {
const config = filteredConfigs.find(c => c.path === path) || allConfigs.find(c => c.path === path);
if (!config) {
showToast(t('common.error'), t('upload.config.notExist'), 'error');
return;
}
// 显示删除确认模态框
showDeleteConfirmModal(config);
}
/**
* 初始化配置管理页面
*/
function initUploadConfigManager() {
// 绑定搜索事件
const searchInput = document.getElementById('configSearch');
const searchBtn = document.getElementById('searchConfigBtn');
const statusFilter = document.getElementById('configStatusFilter');
const providerFilter = document.getElementById('configProviderFilter');
const refreshBtn = document.getElementById('refreshConfigList');
const downloadAllBtn = document.getElementById('downloadAllConfigs');
if (searchInput) {
searchInput.addEventListener('input', debounce(() => {
const searchTerm = searchInput.value.trim();
const currentStatusFilter = statusFilter?.value || '';
const currentProviderFilter = providerFilter?.value || '';
searchConfigs(searchTerm, currentStatusFilter, currentProviderFilter);
}, 300));
}
if (searchBtn) {
searchBtn.addEventListener('click', () => {
const searchTerm = searchInput?.value.trim() || '';
const currentStatusFilter = statusFilter?.value || '';
const currentProviderFilter = providerFilter?.value || '';
searchConfigs(searchTerm, currentStatusFilter, currentProviderFilter);
});
}
if (statusFilter) {
statusFilter.addEventListener('change', () => {
const searchTerm = searchInput?.value.trim() || '';
const currentStatusFilter = statusFilter.value;
const currentProviderFilter = providerFilter?.value || '';
searchConfigs(searchTerm, currentStatusFilter, currentProviderFilter);
});
}
if (providerFilter) {
providerFilter.addEventListener('change', () => {
const searchTerm = searchInput?.value.trim() || '';
const currentStatusFilter = statusFilter?.value || '';
const currentProviderFilter = providerFilter.value;
searchConfigs(searchTerm, currentStatusFilter, currentProviderFilter);
});
}
if (refreshBtn) {
refreshBtn.addEventListener('click', loadConfigList);
}
if (downloadAllBtn) {
downloadAllBtn.addEventListener('click', downloadAllConfigs);
}
// 批量关联配置按钮
const batchLinkBtn = document.getElementById('batchLinkKiroBtn') || document.getElementById('batchLinkProviderBtn');
if (batchLinkBtn) {
batchLinkBtn.addEventListener('click', batchLinkProviderConfigs);
}
// 删除未绑定配置按钮
const deleteUnboundBtn = document.getElementById('deleteUnboundBtn');
if (deleteUnboundBtn) {
deleteUnboundBtn.addEventListener('click', deleteUnboundConfigs);
}
// 初始加载配置列表
loadConfigList();
}
/**
* 重新加载配置文件
*/
async function reloadConfig() {
// 防止重复重载
if (isLoadingConfigs) {
console.log('正在重载配置,跳过重复调用');
return;
}
try {
const result = await window.apiClient.post('/reload-config');
showToast(t('common.success'), result.message, 'success');
// 重新加载配置列表以反映最新的关联状态
await loadConfigList();
// 注意:不再发送 configReloaded 事件,避免重复调用
// window.dispatchEvent(new CustomEvent('configReloaded', {
// detail: result.details
// }));
} catch (error) {
console.error('重载配置失败:', error);
showToast(t('common.error'), t('common.refresh.failed') + ': ' + error.message, 'error');
}
}
/**
* 根据文件路径检测对应的提供商类型
* @param {string} filePath - 文件路径
* @returns {Object|null} 提供商信息对象或null
*/
function detectProviderFromPath(filePath) {
const normalizedPath = filePath.replace(/\\/g, '/').toLowerCase();
// 定义目录到提供商的映射关系
const providerMappings = [
{
patterns: ['configs/kiro/', '/kiro/'],
providerType: 'claude-kiro-oauth',
displayName: 'Claude Kiro OAuth',
shortName: 'kiro-oauth'
},
{
patterns: ['configs/gemini/', '/gemini/', 'configs/gemini-cli/'],
providerType: 'gemini-cli-oauth',
displayName: 'Gemini CLI OAuth',
shortName: 'gemini-oauth'
},
{
patterns: ['configs/qwen/', '/qwen/'],
providerType: 'openai-qwen-oauth',
displayName: 'Qwen OAuth',
shortName: 'qwen-oauth'
},
{
patterns: ['configs/antigravity/', '/antigravity/'],
providerType: 'gemini-antigravity',
displayName: 'Gemini Antigravity',
shortName: 'antigravity'
},
{
patterns: ['configs/codex/', '/codex/'],
providerType: 'openai-codex-oauth',
displayName: 'OpenAI Codex OAuth',
shortName: 'codex-oauth'
},
{
patterns: ['configs/iflow/', '/iflow/'],
providerType: 'openai-iflow-oauth',
displayName: 'OpenAI iFlow OAuth',
shortName: 'iflow-oauth'
}
];
// 遍历映射关系,查找匹配的提供商
for (const mapping of providerMappings) {
for (const pattern of mapping.patterns) {
if (normalizedPath.includes(pattern)) {
return {
providerType: mapping.providerType,
displayName: mapping.displayName,
shortName: mapping.shortName
};
}
}
}
return null;
}
/**
* 一键关联配置到对应的提供商
* @param {string} filePath - 配置文件路径
*/
async function quickLinkProviderConfig(filePath) {
try {
const providerInfo = detectProviderFromPath(filePath);
if (!providerInfo) {
showToast(t('common.error'), t('upload.link.failed.identify'), 'error');
return;
}
showToast(t('common.info'), t('upload.link.processing', { name: providerInfo.displayName }), 'info');
const result = await window.apiClient.post('/quick-link-provider', {
filePath: filePath
});
showToast(t('common.success'), result.message || t('upload.link.success'), 'success');
// 刷新配置列表
await loadConfigList();
} catch (error) {
console.error('一键关联失败:', error);
showToast(t('common.error'), t('upload.link.failed') + ': ' + error.message, 'error');
}
}
/**
* 批量关联所有支持的提供商目录下的未关联配置
*/
async function batchLinkProviderConfigs() {
// 筛选出所有支持的提供商目录下的未关联配置
const unlinkedConfigs = allConfigs.filter(config => {
if (config.isUsed) return false;
const providerInfo = detectProviderFromPath(config.path);
return providerInfo !== null;
});
if (unlinkedConfigs.length === 0) {
showToast(t('common.info'), t('upload.batchLink.none'), 'info');
return;
}
// 按提供商类型分组统计
const groupedByProvider = {};
unlinkedConfigs.forEach(config => {
const providerInfo = detectProviderFromPath(config.path);
if (providerInfo) {
if (!groupedByProvider[providerInfo.displayName]) {
groupedByProvider[providerInfo.displayName] = 0;
}
groupedByProvider[providerInfo.displayName]++;
}
});
const providerSummary = Object.entries(groupedByProvider)
.map(([name, count]) => `${name}: ${count}个`)
.join(', ');
const confirmMsg = t('upload.batchLink.confirm', { count: unlinkedConfigs.length, summary: providerSummary });
if (!confirm(confirmMsg)) {
return;
}
showToast(t('common.info'), t('upload.batchLink.processing', { count: unlinkedConfigs.length }), 'info');
let successCount = 0;
let failCount = 0;
for (const config of unlinkedConfigs) {
try {
await window.apiClient.post('/quick-link-provider', {
filePath: config.path
});
successCount++;
} catch (error) {
console.error(`关联失败: ${config.path}`, error);
failCount++;
}
}
// 刷新配置列表
await loadConfigList();
if (failCount === 0) {
showToast(t('common.success'), t('upload.batchLink.success', { count: successCount }), 'success');
} else {
showToast(t('common.warning'), t('upload.batchLink.partial', { success: successCount, fail: failCount }), 'warning');
}
}
/**
* 删除所有未绑定的配置文件
* 只删除 configs/xxx/ 子目录下的未绑定配置文件
*/
async function deleteUnboundConfigs() {
// 统计未绑定的配置数量,并且必须在 configs/xxx/ 子目录下
const unboundConfigs = allConfigs.filter(config => {
if (config.isUsed) return false;
// 检查路径是否在 configs/xxx/ 子目录下
const normalizedPath = config.path.replace(/\\/g, '/');
const pathParts = normalizedPath.split('/');
// 路径至少需要3部分:configs/子目录/文件名
// 例如:configs/kiro/xxx.json 或 configs/gemini/xxx.json
if (pathParts.length >= 3 && pathParts[0] === 'configs') {
return true;
}
return false;
});
if (unboundConfigs.length === 0) {
showToast(t('common.info'), t('upload.deleteUnbound.none'), 'info');
return;
}
// 显示确认对话框
const confirmMsg = t('upload.deleteUnbound.confirm', { count: unboundConfigs.length });
if (!confirm(confirmMsg)) {
return;
}
try {
showToast(t('common.info'), t('upload.deleteUnbound.processing'), 'info');
const result = await window.apiClient.delete('/upload-configs/delete-unbound');
if (result.deletedCount > 0) {
showToast(t('common.success'), t('upload.deleteUnbound.success', { count: result.deletedCount }), 'success');
// 刷新配置列表
await loadConfigList();
} else {
showToast(t('common.info'), t('upload.deleteUnbound.none'), 'info');
}
// 如果有失败的文件,显示警告
if (result.failedCount > 0) {
console.warn('部分文件删除失败:', result.failedFiles);
showToast(t('common.warning'), t('upload.deleteUnbound.partial', {
success: result.deletedCount,
fail: result.failedCount
}), 'warning');
}
} catch (error) {
console.error('删除未绑定配置失败:', error);
showToast(t('common.error'), t('upload.deleteUnbound.failed') + ': ' + error.message, 'error');
}
}
/**
* 防抖函数
* @param {Function} func - 要防抖的函数
* @param {number} wait - 等待时间(毫秒)
* @returns {Function} 防抖后的函数
*/
function debounce(func, wait) {
let timeout;
return function executedFunction(...args) {
const later = () => {
clearTimeout(timeout);
func(...args);
};
clearTimeout(timeout);
timeout = setTimeout(later, wait);
};
}
/**
* 打包下载所有配置文件
*/
async function downloadAllConfigs() {
try {
showToast(t('common.info'), t('common.loading'), 'info');
// 使用 window.apiClient.get 获取 Blob 数据
// 由于 apiClient 默认可能是处理 JSON 的,我们需要直接调用 fetch 或者确保 apiClient 支持返回原始响应
const token = localStorage.getItem('authToken');
const headers = {
'Authorization': token ? `Bearer ${token}` : ''
};
const response = await fetch('/api/upload-configs/download-all', { headers });
if (!response.ok) {
const errorData = await response.json();
throw new Error(errorData.error?.message || '下载失败');
}
const blob = await response.blob();
const url = window.URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
// 从 Content-Disposition 中提取文件名,或者使用默认名
const contentDisposition = response.headers.get('Content-Disposition');
let filename = `configs_backup_${new Date().toISOString().slice(0, 10)}.zip`;
if (contentDisposition && contentDisposition.indexOf('filename=') !== -1) {
const matches = /filename="([^"]+)"/.exec(contentDisposition);
if (matches && matches[1]) filename = matches[1];
}
a.download = filename;
document.body.appendChild(a);
a.click();
window.URL.revokeObjectURL(url);
document.body.removeChild(a);
showToast(t('common.success'), t('common.success'), 'success');
} catch (error) {
console.error('打包下载失败:', error);
showToast(t('common.error'), t('common.error') + ': ' + error.message, 'error');
}
}
// 导出函数
export {
initUploadConfigManager,
searchConfigs,
loadConfigList,
viewConfig,
deleteConfig,
closeConfigModal,
copyConfigContent,
reloadConfig,
deleteUnboundConfigs
};