mirror of
https://github.com/Wei-Shaw/claude-relay-service.git
synced 2026-01-22 16:43:35 +00:00
617 lines
19 KiB
JavaScript
617 lines
19 KiB
JavaScript
#!/usr/bin/env node
|
||
|
||
const { Command } = require('commander');
|
||
const inquirer = require('inquirer');
|
||
const chalk = require('chalk');
|
||
const ora = require('ora');
|
||
const { table } = require('table');
|
||
const bcrypt = require('bcryptjs');
|
||
const crypto = require('crypto');
|
||
const fs = require('fs');
|
||
const path = require('path');
|
||
|
||
const config = require('../config/config');
|
||
const redis = require('../src/models/redis');
|
||
const apiKeyService = require('../src/services/apiKeyService');
|
||
const claudeAccountService = require('../src/services/claudeAccountService');
|
||
|
||
const program = new Command();
|
||
|
||
// 🎨 样式
|
||
const styles = {
|
||
title: chalk.bold.blue,
|
||
success: chalk.green,
|
||
error: chalk.red,
|
||
warning: chalk.yellow,
|
||
info: chalk.cyan,
|
||
dim: chalk.dim
|
||
};
|
||
|
||
// 🔧 初始化
|
||
async function initialize() {
|
||
const spinner = ora('正在连接 Redis...').start();
|
||
try {
|
||
await redis.connect();
|
||
spinner.succeed('Redis 连接成功');
|
||
} catch (error) {
|
||
spinner.fail('Redis 连接失败');
|
||
console.error(styles.error(error.message));
|
||
process.exit(1);
|
||
}
|
||
}
|
||
|
||
// 🔐 管理员账户管理
|
||
program
|
||
.command('admin')
|
||
.description('管理员账户操作')
|
||
.action(async () => {
|
||
await initialize();
|
||
|
||
// 直接执行创建初始管理员
|
||
await createInitialAdmin();
|
||
|
||
await redis.disconnect();
|
||
});
|
||
|
||
|
||
// 🔑 API Key 管理
|
||
program
|
||
.command('keys')
|
||
.description('API Key 管理操作')
|
||
.action(async () => {
|
||
await initialize();
|
||
|
||
const { action } = await inquirer.prompt([{
|
||
type: 'list',
|
||
name: 'action',
|
||
message: '请选择操作:',
|
||
choices: [
|
||
{ name: '📋 查看所有 API Keys', value: 'list' },
|
||
{ name: '🔧 修改 API Key 过期时间', value: 'update-expiry' },
|
||
{ name: '🔄 续期即将过期的 API Key', value: 'renew' },
|
||
{ name: '🗑️ 删除 API Key', value: 'delete' }
|
||
]
|
||
}]);
|
||
|
||
switch (action) {
|
||
case 'list':
|
||
await listApiKeys();
|
||
break;
|
||
case 'update-expiry':
|
||
await updateApiKeyExpiry();
|
||
break;
|
||
case 'renew':
|
||
await renewApiKeys();
|
||
break;
|
||
case 'delete':
|
||
await deleteApiKey();
|
||
break;
|
||
}
|
||
|
||
await redis.disconnect();
|
||
});
|
||
|
||
// 📊 系统状态
|
||
program
|
||
.command('status')
|
||
.description('查看系统状态')
|
||
.action(async () => {
|
||
await initialize();
|
||
|
||
const spinner = ora('正在获取系统状态...').start();
|
||
|
||
try {
|
||
const [systemStats, apiKeys, accounts] = await Promise.all([
|
||
redis.getSystemStats(),
|
||
apiKeyService.getAllApiKeys(),
|
||
claudeAccountService.getAllAccounts()
|
||
]);
|
||
|
||
spinner.succeed('系统状态获取成功');
|
||
|
||
console.log(styles.title('\n📊 系统状态概览\n'));
|
||
|
||
const statusData = [
|
||
['项目', '数量', '状态'],
|
||
['API Keys', apiKeys.length, `${apiKeys.filter(k => k.isActive).length} 活跃`],
|
||
['Claude 账户', accounts.length, `${accounts.filter(a => a.isActive).length} 活跃`],
|
||
['Redis 连接', redis.isConnected ? '已连接' : '未连接', redis.isConnected ? '🟢' : '🔴'],
|
||
['运行时间', `${Math.floor(process.uptime() / 60)} 分钟`, '🕐']
|
||
];
|
||
|
||
console.log(table(statusData));
|
||
|
||
// 使用统计
|
||
const totalTokens = apiKeys.reduce((sum, key) => sum + (key.usage?.total?.tokens || 0), 0);
|
||
const totalRequests = apiKeys.reduce((sum, key) => sum + (key.usage?.total?.requests || 0), 0);
|
||
|
||
console.log(styles.title('\n📈 使用统计\n'));
|
||
console.log(`总 Token 使用量: ${styles.success(totalTokens.toLocaleString())}`);
|
||
console.log(`总请求数: ${styles.success(totalRequests.toLocaleString())}`);
|
||
|
||
} catch (error) {
|
||
spinner.fail('获取系统状态失败');
|
||
console.error(styles.error(error.message));
|
||
}
|
||
|
||
await redis.disconnect();
|
||
});
|
||
|
||
|
||
// 实现具体功能函数
|
||
|
||
async function createInitialAdmin() {
|
||
console.log(styles.title('\n🔐 创建初始管理员账户\n'));
|
||
|
||
// 检查是否已存在 init.json
|
||
const initFilePath = path.join(__dirname, '..', 'data', 'init.json');
|
||
if (fs.existsSync(initFilePath)) {
|
||
const existingData = JSON.parse(fs.readFileSync(initFilePath, 'utf8'));
|
||
console.log(styles.warning('⚠️ 检测到已存在管理员账户!'));
|
||
console.log(` 用户名: ${existingData.adminUsername}`);
|
||
console.log(` 创建时间: ${new Date(existingData.initializedAt).toLocaleString()}`);
|
||
|
||
const { overwrite } = await inquirer.prompt([{
|
||
type: 'confirm',
|
||
name: 'overwrite',
|
||
message: '是否覆盖现有管理员账户?',
|
||
default: false
|
||
}]);
|
||
|
||
if (!overwrite) {
|
||
console.log(styles.info('ℹ️ 已取消创建'));
|
||
return;
|
||
}
|
||
}
|
||
|
||
const adminData = await inquirer.prompt([
|
||
{
|
||
type: 'input',
|
||
name: 'username',
|
||
message: '用户名:',
|
||
default: 'admin',
|
||
validate: input => input.length >= 3 || '用户名至少3个字符'
|
||
},
|
||
{
|
||
type: 'password',
|
||
name: 'password',
|
||
message: '密码:',
|
||
validate: input => input.length >= 8 || '密码至少8个字符'
|
||
},
|
||
{
|
||
type: 'password',
|
||
name: 'confirmPassword',
|
||
message: '确认密码:',
|
||
validate: (input, answers) => input === answers.password || '密码不匹配'
|
||
}
|
||
]);
|
||
|
||
const spinner = ora('正在创建管理员账户...').start();
|
||
|
||
try {
|
||
// 1. 先更新 init.json(唯一真实数据源)
|
||
const initData = {
|
||
initializedAt: new Date().toISOString(),
|
||
adminUsername: adminData.username,
|
||
adminPassword: adminData.password, // 保存明文密码
|
||
version: '1.0.0',
|
||
updatedAt: new Date().toISOString()
|
||
};
|
||
|
||
// 确保 data 目录存在
|
||
const dataDir = path.join(__dirname, '..', 'data');
|
||
if (!fs.existsSync(dataDir)) {
|
||
fs.mkdirSync(dataDir, { recursive: true });
|
||
}
|
||
|
||
// 写入文件
|
||
fs.writeFileSync(initFilePath, JSON.stringify(initData, null, 2));
|
||
|
||
// 2. 再更新 Redis 缓存
|
||
const passwordHash = await bcrypt.hash(adminData.password, 12);
|
||
|
||
const credentials = {
|
||
username: adminData.username,
|
||
passwordHash,
|
||
createdAt: new Date().toISOString(),
|
||
lastLogin: null,
|
||
updatedAt: new Date().toISOString()
|
||
};
|
||
|
||
await redis.setSession('admin_credentials', credentials, 0); // 永不过期
|
||
|
||
spinner.succeed('管理员账户创建成功');
|
||
console.log(`${styles.success('✅')} 用户名: ${adminData.username}`);
|
||
console.log(`${styles.success('✅')} 密码: ${adminData.password}`);
|
||
console.log(`${styles.info('ℹ️')} 请妥善保管登录凭据`);
|
||
console.log(`${styles.info('ℹ️')} 凭据已保存到: ${initFilePath}`);
|
||
console.log(`${styles.warning('⚠️')} 如果服务正在运行,请重启服务以加载新凭据`);
|
||
|
||
} catch (error) {
|
||
spinner.fail('创建管理员账户失败');
|
||
console.error(styles.error(error.message));
|
||
}
|
||
}
|
||
|
||
|
||
|
||
|
||
|
||
|
||
// API Key 管理功能
|
||
async function listApiKeys() {
|
||
const spinner = ora('正在获取 API Keys...').start();
|
||
|
||
try {
|
||
const apiKeys = await apiKeyService.getAllApiKeys();
|
||
spinner.succeed(`找到 ${apiKeys.length} 个 API Keys`);
|
||
|
||
if (apiKeys.length === 0) {
|
||
console.log(styles.warning('没有找到任何 API Keys'));
|
||
return;
|
||
}
|
||
|
||
const tableData = [
|
||
['名称', 'API Key', '状态', '过期时间', '使用量', 'Token限制']
|
||
];
|
||
|
||
apiKeys.forEach(key => {
|
||
const now = new Date();
|
||
const expiresAt = key.expiresAt ? new Date(key.expiresAt) : null;
|
||
let expiryStatus = '永不过期';
|
||
|
||
if (expiresAt) {
|
||
if (expiresAt < now) {
|
||
expiryStatus = styles.error(`已过期 (${expiresAt.toLocaleDateString()})`);
|
||
} else {
|
||
const daysLeft = Math.ceil((expiresAt - now) / (1000 * 60 * 60 * 24));
|
||
if (daysLeft <= 7) {
|
||
expiryStatus = styles.warning(`${daysLeft}天后过期 (${expiresAt.toLocaleDateString()})`);
|
||
} else {
|
||
expiryStatus = styles.success(`${expiresAt.toLocaleDateString()}`);
|
||
}
|
||
}
|
||
}
|
||
|
||
tableData.push([
|
||
key.name,
|
||
key.apiKey ? key.apiKey.substring(0, 20) + '...' : '-',
|
||
key.isActive ? '🟢 活跃' : '🔴 停用',
|
||
expiryStatus,
|
||
`${(key.usage?.total?.tokens || 0).toLocaleString()}`,
|
||
key.tokenLimit ? key.tokenLimit.toLocaleString() : '无限制'
|
||
]);
|
||
});
|
||
|
||
console.log(styles.title('\n🔑 API Keys 列表:\n'));
|
||
console.log(table(tableData));
|
||
|
||
} catch (error) {
|
||
spinner.fail('获取 API Keys 失败');
|
||
console.error(styles.error(error.message));
|
||
}
|
||
}
|
||
|
||
async function updateApiKeyExpiry() {
|
||
try {
|
||
// 获取所有 API Keys
|
||
const apiKeys = await apiKeyService.getAllApiKeys();
|
||
|
||
if (apiKeys.length === 0) {
|
||
console.log(styles.warning('没有找到任何 API Keys'));
|
||
return;
|
||
}
|
||
|
||
// 选择要修改的 API Key
|
||
const { selectedKey } = await inquirer.prompt([{
|
||
type: 'list',
|
||
name: 'selectedKey',
|
||
message: '选择要修改的 API Key:',
|
||
choices: apiKeys.map(key => ({
|
||
name: `${key.name} (${key.apiKey?.substring(0, 20)}...) - ${key.expiresAt ? new Date(key.expiresAt).toLocaleDateString() : '永不过期'}`,
|
||
value: key
|
||
}))
|
||
}]);
|
||
|
||
console.log(`\n当前 API Key: ${selectedKey.name}`);
|
||
console.log(`当前过期时间: ${selectedKey.expiresAt ? new Date(selectedKey.expiresAt).toLocaleString() : '永不过期'}`);
|
||
|
||
// 选择新的过期时间
|
||
const { expiryOption } = await inquirer.prompt([{
|
||
type: 'list',
|
||
name: 'expiryOption',
|
||
message: '选择新的过期时间:',
|
||
choices: [
|
||
{ name: '⏰ 1分后(测试用)', value: '1m' },
|
||
{ name: '⏰ 1小时后(测试用)', value: '1h' },
|
||
{ name: '📅 1天后', value: '1d' },
|
||
{ name: '📅 7天后', value: '7d' },
|
||
{ name: '📅 30天后', value: '30d' },
|
||
{ name: '📅 90天后', value: '90d' },
|
||
{ name: '📅 365天后', value: '365d' },
|
||
{ name: '♾️ 永不过期', value: 'never' },
|
||
{ name: '🎯 自定义日期时间', value: 'custom' }
|
||
]
|
||
}]);
|
||
|
||
let newExpiresAt = null;
|
||
|
||
if (expiryOption === 'never') {
|
||
newExpiresAt = null;
|
||
} else if (expiryOption === 'custom') {
|
||
const { customDate, customTime } = await inquirer.prompt([
|
||
{
|
||
type: 'input',
|
||
name: 'customDate',
|
||
message: '输入日期 (YYYY-MM-DD):',
|
||
default: new Date().toISOString().split('T')[0],
|
||
validate: input => {
|
||
const date = new Date(input);
|
||
return !isNaN(date.getTime()) || '请输入有效的日期格式';
|
||
}
|
||
},
|
||
{
|
||
type: 'input',
|
||
name: 'customTime',
|
||
message: '输入时间 (HH:MM):',
|
||
default: '00:00',
|
||
validate: input => {
|
||
return /^\d{2}:\d{2}$/.test(input) || '请输入有效的时间格式 (HH:MM)';
|
||
}
|
||
}
|
||
]);
|
||
|
||
newExpiresAt = new Date(`${customDate}T${customTime}:00`).toISOString();
|
||
} else {
|
||
// 计算新的过期时间
|
||
const now = new Date();
|
||
const durations = {
|
||
'1m': 60 * 1000,
|
||
'1h': 60 * 60 * 1000,
|
||
'1d': 24 * 60 * 60 * 1000,
|
||
'7d': 7 * 24 * 60 * 60 * 1000,
|
||
'30d': 30 * 24 * 60 * 60 * 1000,
|
||
'90d': 90 * 24 * 60 * 60 * 1000,
|
||
'365d': 365 * 24 * 60 * 60 * 1000
|
||
};
|
||
|
||
newExpiresAt = new Date(now.getTime() + durations[expiryOption]).toISOString();
|
||
}
|
||
|
||
// 确认修改
|
||
const confirmMsg = newExpiresAt
|
||
? `确认将过期时间修改为: ${new Date(newExpiresAt).toLocaleString()}?`
|
||
: '确认设置为永不过期?';
|
||
|
||
const { confirmed } = await inquirer.prompt([{
|
||
type: 'confirm',
|
||
name: 'confirmed',
|
||
message: confirmMsg,
|
||
default: true
|
||
}]);
|
||
|
||
if (!confirmed) {
|
||
console.log(styles.info('已取消修改'));
|
||
return;
|
||
}
|
||
|
||
// 执行修改
|
||
const spinner = ora('正在修改过期时间...').start();
|
||
|
||
try {
|
||
await apiKeyService.updateApiKey(selectedKey.id, { expiresAt: newExpiresAt });
|
||
spinner.succeed('过期时间修改成功');
|
||
|
||
console.log(styles.success(`\n✅ API Key "${selectedKey.name}" 的过期时间已更新`));
|
||
console.log(`新的过期时间: ${newExpiresAt ? new Date(newExpiresAt).toLocaleString() : '永不过期'}`);
|
||
|
||
} catch (error) {
|
||
spinner.fail('修改失败');
|
||
console.error(styles.error(error.message));
|
||
}
|
||
|
||
} catch (error) {
|
||
console.error(styles.error('操作失败:', error.message));
|
||
}
|
||
}
|
||
|
||
async function renewApiKeys() {
|
||
const spinner = ora('正在查找即将过期的 API Keys...').start();
|
||
|
||
try {
|
||
const apiKeys = await apiKeyService.getAllApiKeys();
|
||
const now = new Date();
|
||
const sevenDaysLater = new Date(now.getTime() + 7 * 24 * 60 * 60 * 1000);
|
||
|
||
// 筛选即将过期的 Keys(7天内)
|
||
const expiringKeys = apiKeys.filter(key => {
|
||
if (!key.expiresAt) return false;
|
||
const expiresAt = new Date(key.expiresAt);
|
||
return expiresAt > now && expiresAt <= sevenDaysLater;
|
||
});
|
||
|
||
spinner.stop();
|
||
|
||
if (expiringKeys.length === 0) {
|
||
console.log(styles.info('没有即将过期的 API Keys(7天内)'));
|
||
return;
|
||
}
|
||
|
||
console.log(styles.warning(`\n找到 ${expiringKeys.length} 个即将过期的 API Keys:\n`));
|
||
|
||
expiringKeys.forEach((key, index) => {
|
||
const daysLeft = Math.ceil((new Date(key.expiresAt) - now) / (1000 * 60 * 60 * 24));
|
||
console.log(`${index + 1}. ${key.name} - ${daysLeft}天后过期 (${new Date(key.expiresAt).toLocaleDateString()})`);
|
||
});
|
||
|
||
const { renewOption } = await inquirer.prompt([{
|
||
type: 'list',
|
||
name: 'renewOption',
|
||
message: '选择续期方式:',
|
||
choices: [
|
||
{ name: '📅 全部续期30天', value: 'all30' },
|
||
{ name: '📅 全部续期90天', value: 'all90' },
|
||
{ name: '🎯 逐个选择续期', value: 'individual' }
|
||
]
|
||
}]);
|
||
|
||
if (renewOption.startsWith('all')) {
|
||
const days = renewOption === 'all30' ? 30 : 90;
|
||
const renewSpinner = ora(`正在为所有 API Keys 续期 ${days} 天...`).start();
|
||
|
||
for (const key of expiringKeys) {
|
||
try {
|
||
const newExpiresAt = new Date(new Date(key.expiresAt).getTime() + days * 24 * 60 * 60 * 1000).toISOString();
|
||
await apiKeyService.updateApiKey(key.id, { expiresAt: newExpiresAt });
|
||
} catch (error) {
|
||
renewSpinner.fail(`续期 ${key.name} 失败: ${error.message}`);
|
||
}
|
||
}
|
||
|
||
renewSpinner.succeed(`成功续期 ${expiringKeys.length} 个 API Keys`);
|
||
|
||
} else {
|
||
// 逐个选择续期
|
||
for (const key of expiringKeys) {
|
||
console.log(`\n处理: ${key.name}`);
|
||
|
||
const { action } = await inquirer.prompt([{
|
||
type: 'list',
|
||
name: 'action',
|
||
message: '选择操作:',
|
||
choices: [
|
||
{ name: '续期30天', value: '30' },
|
||
{ name: '续期90天', value: '90' },
|
||
{ name: '跳过', value: 'skip' }
|
||
]
|
||
}]);
|
||
|
||
if (action !== 'skip') {
|
||
const days = parseInt(action);
|
||
const newExpiresAt = new Date(new Date(key.expiresAt).getTime() + days * 24 * 60 * 60 * 1000).toISOString();
|
||
|
||
try {
|
||
await apiKeyService.updateApiKey(key.id, { expiresAt: newExpiresAt });
|
||
console.log(styles.success(`✅ 已续期 ${days} 天`));
|
||
} catch (error) {
|
||
console.log(styles.error(`❌ 续期失败: ${error.message}`));
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
} catch (error) {
|
||
spinner.fail('操作失败');
|
||
console.error(styles.error(error.message));
|
||
}
|
||
}
|
||
|
||
async function deleteApiKey() {
|
||
try {
|
||
const apiKeys = await apiKeyService.getAllApiKeys();
|
||
|
||
if (apiKeys.length === 0) {
|
||
console.log(styles.warning('没有找到任何 API Keys'));
|
||
return;
|
||
}
|
||
|
||
const { selectedKeys } = await inquirer.prompt([{
|
||
type: 'checkbox',
|
||
name: 'selectedKeys',
|
||
message: '选择要删除的 API Keys (空格选择,回车确认):',
|
||
choices: apiKeys.map(key => ({
|
||
name: `${key.name} (${key.apiKey?.substring(0, 20)}...)`,
|
||
value: key.id
|
||
}))
|
||
}]);
|
||
|
||
if (selectedKeys.length === 0) {
|
||
console.log(styles.info('未选择任何 API Key'));
|
||
return;
|
||
}
|
||
|
||
const { confirmed } = await inquirer.prompt([{
|
||
type: 'confirm',
|
||
name: 'confirmed',
|
||
message: styles.warning(`确认删除 ${selectedKeys.length} 个 API Keys?`),
|
||
default: false
|
||
}]);
|
||
|
||
if (!confirmed) {
|
||
console.log(styles.info('已取消删除'));
|
||
return;
|
||
}
|
||
|
||
const spinner = ora('正在删除 API Keys...').start();
|
||
let successCount = 0;
|
||
|
||
for (const keyId of selectedKeys) {
|
||
try {
|
||
await apiKeyService.deleteApiKey(keyId);
|
||
successCount++;
|
||
} catch (error) {
|
||
spinner.fail(`删除失败: ${error.message}`);
|
||
}
|
||
}
|
||
|
||
spinner.succeed(`成功删除 ${successCount}/${selectedKeys.length} 个 API Keys`);
|
||
|
||
} catch (error) {
|
||
console.error(styles.error('删除失败:', error.message));
|
||
}
|
||
}
|
||
|
||
async function listClaudeAccounts() {
|
||
const spinner = ora('正在获取 Claude 账户...').start();
|
||
|
||
try {
|
||
const accounts = await claudeAccountService.getAllAccounts();
|
||
spinner.succeed(`找到 ${accounts.length} 个 Claude 账户`);
|
||
|
||
if (accounts.length === 0) {
|
||
console.log(styles.warning('没有找到任何 Claude 账户'));
|
||
return;
|
||
}
|
||
|
||
const tableData = [
|
||
['ID', '名称', '邮箱', '状态', '代理', '最后使用']
|
||
];
|
||
|
||
accounts.forEach(account => {
|
||
tableData.push([
|
||
account.id.substring(0, 8) + '...',
|
||
account.name,
|
||
account.email || '-',
|
||
account.isActive ? (account.status === 'active' ? '🟢 活跃' : '🟡 待激活') : '🔴 停用',
|
||
account.proxy ? '🌐 是' : '-',
|
||
account.lastUsedAt ? new Date(account.lastUsedAt).toLocaleDateString() : '-'
|
||
]);
|
||
});
|
||
|
||
console.log('\n🏢 Claude 账户列表:\n');
|
||
console.log(table(tableData));
|
||
|
||
} catch (error) {
|
||
spinner.fail('获取 Claude 账户失败');
|
||
console.error(styles.error(error.message));
|
||
}
|
||
}
|
||
|
||
// 程序信息
|
||
program
|
||
.name('claude-relay-cli')
|
||
.description('Claude Relay Service 命令行管理工具')
|
||
.version('1.0.0');
|
||
|
||
// 解析命令行参数
|
||
program.parse();
|
||
|
||
// 如果没有提供命令,显示帮助
|
||
if (!process.argv.slice(2).length) {
|
||
console.log(styles.title('🚀 Claude Relay Service CLI\n'));
|
||
console.log('使用以下命令管理服务:\n');
|
||
console.log(' claude-relay-cli admin - 创建初始管理员账户');
|
||
console.log(' claude-relay-cli keys - API Key 管理(查看/修改过期时间/续期/删除)');
|
||
console.log(' claude-relay-cli status - 查看系统状态');
|
||
console.log('\n使用 --help 查看详细帮助信息');
|
||
} |