mirror of
https://github.com/Wei-Shaw/claude-relay-service.git
synced 2026-01-22 16:43:35 +00:00
210 lines
8.3 KiB
JavaScript
210 lines
8.3 KiB
JavaScript
const path = require('path')
|
||
require('dotenv').config()
|
||
|
||
const config = {
|
||
// 🌐 服务器配置
|
||
server: {
|
||
port: parseInt(process.env.PORT) || 3000,
|
||
host: process.env.HOST || '0.0.0.0',
|
||
nodeEnv: process.env.NODE_ENV || 'development',
|
||
trustProxy: process.env.TRUST_PROXY === 'true'
|
||
},
|
||
|
||
// 🔐 安全配置
|
||
security: {
|
||
jwtSecret: process.env.JWT_SECRET || 'CHANGE-THIS-JWT-SECRET-IN-PRODUCTION',
|
||
adminSessionTimeout: parseInt(process.env.ADMIN_SESSION_TIMEOUT) || 86400000, // 24小时
|
||
apiKeyPrefix: process.env.API_KEY_PREFIX || 'cr_',
|
||
encryptionKey: process.env.ENCRYPTION_KEY || 'CHANGE-THIS-32-CHARACTER-KEY-NOW'
|
||
},
|
||
|
||
// 📊 Redis配置
|
||
redis: {
|
||
host: process.env.REDIS_HOST || '127.0.0.1',
|
||
port: parseInt(process.env.REDIS_PORT) || 6379,
|
||
password: process.env.REDIS_PASSWORD || '',
|
||
db: parseInt(process.env.REDIS_DB) || 0,
|
||
connectTimeout: 10000,
|
||
commandTimeout: 5000,
|
||
retryDelayOnFailover: 100,
|
||
maxRetriesPerRequest: 3,
|
||
lazyConnect: true,
|
||
enableTLS: process.env.REDIS_ENABLE_TLS === 'true'
|
||
},
|
||
|
||
// 🔗 会话管理配置
|
||
session: {
|
||
// 粘性会话TTL配置(小时),默认1小时
|
||
stickyTtlHours: parseFloat(process.env.STICKY_SESSION_TTL_HOURS) || 1,
|
||
// 续期阈值(分钟),默认0分钟(不续期)
|
||
renewalThresholdMinutes: parseInt(process.env.STICKY_SESSION_RENEWAL_THRESHOLD_MINUTES) || 0
|
||
},
|
||
|
||
// 🎯 Claude API配置
|
||
claude: {
|
||
apiUrl: process.env.CLAUDE_API_URL || 'https://api.anthropic.com/v1/messages',
|
||
apiVersion: process.env.CLAUDE_API_VERSION || '2023-06-01',
|
||
betaHeader:
|
||
process.env.CLAUDE_BETA_HEADER ||
|
||
'claude-code-20250219,oauth-2025-04-20,interleaved-thinking-2025-05-14,fine-grained-tool-streaming-2025-05-14',
|
||
overloadHandling: {
|
||
enabled: (() => {
|
||
const minutes = parseInt(process.env.CLAUDE_OVERLOAD_HANDLING_MINUTES) || 0
|
||
// 验证配置值:限制在0-1440分钟(24小时)内
|
||
return Math.max(0, Math.min(minutes, 1440))
|
||
})()
|
||
}
|
||
},
|
||
|
||
// ☁️ Bedrock API配置
|
||
bedrock: {
|
||
enabled: process.env.CLAUDE_CODE_USE_BEDROCK === '1',
|
||
defaultRegion: process.env.AWS_REGION || 'us-east-1',
|
||
smallFastModelRegion: process.env.ANTHROPIC_SMALL_FAST_MODEL_AWS_REGION,
|
||
defaultModel: process.env.ANTHROPIC_MODEL || 'us.anthropic.claude-sonnet-4-20250514-v1:0',
|
||
smallFastModel:
|
||
process.env.ANTHROPIC_SMALL_FAST_MODEL || 'us.anthropic.claude-3-5-haiku-20241022-v1:0',
|
||
maxOutputTokens: parseInt(process.env.CLAUDE_CODE_MAX_OUTPUT_TOKENS) || 4096,
|
||
maxThinkingTokens: parseInt(process.env.MAX_THINKING_TOKENS) || 1024,
|
||
enablePromptCaching: process.env.DISABLE_PROMPT_CACHING !== '1'
|
||
},
|
||
|
||
// 🌐 代理配置
|
||
proxy: {
|
||
timeout: parseInt(process.env.DEFAULT_PROXY_TIMEOUT) || 600000, // 10分钟
|
||
maxRetries: parseInt(process.env.MAX_PROXY_RETRIES) || 3,
|
||
// 连接池与 Keep-Alive 配置(默认关闭,需要显式开启)
|
||
keepAlive: (() => {
|
||
if (process.env.PROXY_KEEP_ALIVE === undefined || process.env.PROXY_KEEP_ALIVE === '') {
|
||
return false
|
||
}
|
||
return process.env.PROXY_KEEP_ALIVE === 'true'
|
||
})(),
|
||
maxSockets: (() => {
|
||
if (process.env.PROXY_MAX_SOCKETS === undefined || process.env.PROXY_MAX_SOCKETS === '') {
|
||
return undefined
|
||
}
|
||
const parsed = parseInt(process.env.PROXY_MAX_SOCKETS)
|
||
return Number.isFinite(parsed) && parsed > 0 ? parsed : undefined
|
||
})(),
|
||
maxFreeSockets: (() => {
|
||
if (
|
||
process.env.PROXY_MAX_FREE_SOCKETS === undefined ||
|
||
process.env.PROXY_MAX_FREE_SOCKETS === ''
|
||
) {
|
||
return undefined
|
||
}
|
||
const parsed = parseInt(process.env.PROXY_MAX_FREE_SOCKETS)
|
||
return Number.isFinite(parsed) && parsed >= 0 ? parsed : undefined
|
||
})(),
|
||
// IP协议族配置:true=IPv4, false=IPv6, 默认IPv4(兼容性更好)
|
||
useIPv4: process.env.PROXY_USE_IPV4 !== 'false' // 默认 true,只有明确设置为 'false' 才使用 IPv6
|
||
},
|
||
|
||
// ⏱️ 请求超时配置
|
||
requestTimeout: parseInt(process.env.REQUEST_TIMEOUT) || 600000, // 默认 10 分钟
|
||
|
||
// 📈 使用限制
|
||
limits: {
|
||
defaultTokenLimit: parseInt(process.env.DEFAULT_TOKEN_LIMIT) || 1000000
|
||
},
|
||
|
||
// 📝 日志配置
|
||
logging: {
|
||
level: process.env.LOG_LEVEL || 'info',
|
||
dirname: path.join(__dirname, '..', 'logs'),
|
||
maxSize: process.env.LOG_MAX_SIZE || '10m',
|
||
maxFiles: parseInt(process.env.LOG_MAX_FILES) || 5
|
||
},
|
||
|
||
// 🔧 系统配置
|
||
system: {
|
||
cleanupInterval: parseInt(process.env.CLEANUP_INTERVAL) || 3600000, // 1小时
|
||
tokenUsageRetention: parseInt(process.env.TOKEN_USAGE_RETENTION) || 2592000000, // 30天
|
||
healthCheckInterval: parseInt(process.env.HEALTH_CHECK_INTERVAL) || 60000, // 1分钟
|
||
timezone: process.env.SYSTEM_TIMEZONE || 'Asia/Shanghai', // 默认UTC+8(中国时区)
|
||
timezoneOffset: parseInt(process.env.TIMEZONE_OFFSET) || 8 // UTC偏移小时数,默认+8
|
||
},
|
||
|
||
// 🎨 Web界面配置
|
||
web: {
|
||
title: process.env.WEB_TITLE || 'Claude Relay Service',
|
||
description:
|
||
process.env.WEB_DESCRIPTION ||
|
||
'Multi-account Claude API relay service with beautiful management interface',
|
||
logoUrl: process.env.WEB_LOGO_URL || '/assets/logo.png',
|
||
enableCors: process.env.ENABLE_CORS === 'true',
|
||
sessionSecret: process.env.WEB_SESSION_SECRET || 'CHANGE-THIS-SESSION-SECRET'
|
||
},
|
||
|
||
// 🔐 LDAP 认证配置
|
||
ldap: {
|
||
enabled: process.env.LDAP_ENABLED === 'true',
|
||
server: {
|
||
url: process.env.LDAP_URL || 'ldap://localhost:389',
|
||
bindDN: process.env.LDAP_BIND_DN || 'cn=admin,dc=example,dc=com',
|
||
bindCredentials: process.env.LDAP_BIND_PASSWORD || 'admin',
|
||
searchBase: process.env.LDAP_SEARCH_BASE || 'dc=example,dc=com',
|
||
searchFilter: process.env.LDAP_SEARCH_FILTER || '(uid={{username}})',
|
||
searchAttributes: process.env.LDAP_SEARCH_ATTRIBUTES
|
||
? process.env.LDAP_SEARCH_ATTRIBUTES.split(',')
|
||
: ['dn', 'uid', 'cn', 'mail', 'givenName', 'sn'],
|
||
timeout: parseInt(process.env.LDAP_TIMEOUT) || 5000,
|
||
connectTimeout: parseInt(process.env.LDAP_CONNECT_TIMEOUT) || 10000,
|
||
// TLS/SSL 配置
|
||
tls: {
|
||
// 是否忽略证书错误 (用于自签名证书)
|
||
rejectUnauthorized: process.env.LDAP_TLS_REJECT_UNAUTHORIZED !== 'false', // 默认验证证书,设置为false则忽略
|
||
// CA证书文件路径 (可选,用于自定义CA证书)
|
||
ca: process.env.LDAP_TLS_CA_FILE
|
||
? require('fs').readFileSync(process.env.LDAP_TLS_CA_FILE)
|
||
: undefined,
|
||
// 客户端证书文件路径 (可选,用于双向认证)
|
||
cert: process.env.LDAP_TLS_CERT_FILE
|
||
? require('fs').readFileSync(process.env.LDAP_TLS_CERT_FILE)
|
||
: undefined,
|
||
// 客户端私钥文件路径 (可选,用于双向认证)
|
||
key: process.env.LDAP_TLS_KEY_FILE
|
||
? require('fs').readFileSync(process.env.LDAP_TLS_KEY_FILE)
|
||
: undefined,
|
||
// 服务器名称 (用于SNI,可选)
|
||
servername: process.env.LDAP_TLS_SERVERNAME || undefined
|
||
}
|
||
},
|
||
userMapping: {
|
||
username: process.env.LDAP_USER_ATTR_USERNAME || 'uid',
|
||
displayName: process.env.LDAP_USER_ATTR_DISPLAY_NAME || 'cn',
|
||
email: process.env.LDAP_USER_ATTR_EMAIL || 'mail',
|
||
firstName: process.env.LDAP_USER_ATTR_FIRST_NAME || 'givenName',
|
||
lastName: process.env.LDAP_USER_ATTR_LAST_NAME || 'sn'
|
||
}
|
||
},
|
||
|
||
// 👥 用户管理配置
|
||
userManagement: {
|
||
enabled: process.env.USER_MANAGEMENT_ENABLED === 'true',
|
||
defaultUserRole: process.env.DEFAULT_USER_ROLE || 'user',
|
||
userSessionTimeout: parseInt(process.env.USER_SESSION_TIMEOUT) || 86400000, // 24小时
|
||
maxApiKeysPerUser: parseInt(process.env.MAX_API_KEYS_PER_USER) || 1,
|
||
allowUserDeleteApiKeys: process.env.ALLOW_USER_DELETE_API_KEYS === 'true' // 默认不允许用户删除自己的API Keys
|
||
},
|
||
|
||
// 📢 Webhook通知配置
|
||
webhook: {
|
||
enabled: process.env.WEBHOOK_ENABLED !== 'false', // 默认启用
|
||
urls: process.env.WEBHOOK_URLS
|
||
? process.env.WEBHOOK_URLS.split(',').map((url) => url.trim())
|
||
: [],
|
||
timeout: parseInt(process.env.WEBHOOK_TIMEOUT) || 10000, // 10秒超时
|
||
retries: parseInt(process.env.WEBHOOK_RETRIES) || 3 // 重试3次
|
||
},
|
||
|
||
// 🛠️ 开发配置
|
||
development: {
|
||
debug: process.env.DEBUG === 'true',
|
||
hotReload: process.env.HOT_RELOAD === 'true'
|
||
}
|
||
}
|
||
|
||
module.exports = config
|