server.js 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
  1. /**
  2. * Docker 镜像代理加速系统 - 服务器入口点
  3. */
  4. const express = require('express');
  5. const fs = require('fs').promises;
  6. const path = require('path');
  7. const bodyParser = require('body-parser');
  8. const session = require('express-session');
  9. const cors = require('cors');
  10. const http = require('http');
  11. const logger = require('./logger');
  12. const { ensureDirectoriesExist } = require('./init-dirs');
  13. const { downloadImages } = require('./download-images');
  14. const { gracefulShutdown } = require('./cleanup');
  15. const os = require('os');
  16. const { requireLogin } = require('./middleware/auth');
  17. const compatibilityLayer = require('./compatibility-layer');
  18. const { initializeDatabase } = require('./scripts/init-database');
  19. const database = require('./database/database');
  20. const httpProxyService = require('./services/httpProxyService');
  21. // 设置日志级别 (默认INFO, 可通过环境变量设置)
  22. const logLevel = process.env.LOG_LEVEL || 'WARN';
  23. logger.setLogLevel(logLevel);
  24. logger.info(`日志级别已设置为: ${logLevel}`);
  25. // 导入配置
  26. const config = require('./config');
  27. // 导入中间件
  28. const { sessionActivity, sanitizeRequestBody, securityHeaders } = require('./middleware/auth');
  29. // 导入初始化调度器
  30. const { executeOnce } = require('./lib/initScheduler');
  31. // 初始化Express应用
  32. const app = express();
  33. const server = http.createServer(app);
  34. // 配置中间件
  35. app.use(cors());
  36. app.use(express.json());
  37. app.use(express.static('web'));
  38. app.use(bodyParser.urlencoded({ extended: true }));
  39. app.use(session({
  40. secret: config.sessionSecret || 'OhTq3faqSKoxbV%NJV',
  41. resave: true,
  42. saveUninitialized: true,
  43. cookie: {
  44. secure: config.secureSession || false,
  45. maxAge: 7 * 24 * 60 * 60 * 1000 // 7天(一周)
  46. }
  47. }));
  48. // 自定义中间件
  49. app.use(sessionActivity);
  50. app.use(sanitizeRequestBody);
  51. app.use(securityHeaders);
  52. // 请求日志中间件
  53. app.use((req, res, next) => {
  54. const start = Date.now();
  55. // 在响应完成后记录日志
  56. res.on('finish', () => {
  57. const duration = Date.now() - start;
  58. // 增强过滤条件
  59. const isSuccessfulGet = req.method === 'GET' && (res.statusCode === 200 || res.statusCode === 304);
  60. const isStaticResource = req.url.match(/\.(js|css|png|jpg|jpeg|gif|ico|svg|woff|woff2|ttf|eot)$/i);
  61. const isCommonApiRequest = req.url.startsWith('/api/') &&
  62. (req.url.includes('/check-session') ||
  63. req.url.includes('/system-resources') ||
  64. req.url.includes('/docker/status'));
  65. const isErrorResponse = res.statusCode >= 400;
  66. // 只记录关键API请求和错误响应,过滤普通的API请求和静态资源
  67. if ((isErrorResponse ||
  68. (req.url.startsWith('/api/') && !isCommonApiRequest)) &&
  69. !isStaticResource &&
  70. !(isSuccessfulGet && isCommonApiRequest)) {
  71. // 记录简化的请求信息
  72. req.skipDetailedLogging = !isErrorResponse; // 非错误请求跳过详细日志
  73. logger.request(req, res, duration);
  74. }
  75. });
  76. next();
  77. });
  78. // 使用我们的路由注册函数加载所有路由
  79. logger.info('注册所有应用路由...');
  80. const registerRoutes = require('./routes');
  81. registerRoutes(app);
  82. // 提供兼容层以确保旧接口继续工作
  83. require('./compatibility-layer')(app);
  84. // 确保登录路由可用
  85. try {
  86. const loginRouter = require('./routes/login');
  87. app.use('/api', loginRouter);
  88. logger.success('✓ 已添加备用登录路由');
  89. } catch (loginError) {
  90. logger.error('无法加载备用登录路由:', loginError);
  91. }
  92. // 页面路由
  93. app.get('/', (req, res) => {
  94. res.sendFile(path.join(__dirname, 'web', 'index.html'));
  95. });
  96. app.get('/admin', (req, res) => {
  97. res.sendFile(path.join(__dirname, 'web', 'admin.html'));
  98. });
  99. app.get('/docs', (req, res) => {
  100. res.sendFile(path.join(__dirname, 'web', 'docs.html'));
  101. });
  102. // 废弃的登录页面路由 - 该路由未使用且导致404错误,现已移除
  103. // app.get('/login', (req, res) => {
  104. // // 检查用户是否已登录
  105. // if (req.session && req.session.user) {
  106. // return res.redirect('/admin'); // 已登录用户重定向到管理页面
  107. // }
  108. //
  109. // res.sendFile(path.join(__dirname, 'web', 'login.html'));
  110. // });
  111. // 404处理
  112. app.use((req, res) => {
  113. res.status(404).json({ error: '请求的资源不存在' });
  114. });
  115. // 错误处理中间件
  116. app.use((err, req, res, next) => {
  117. logger.error('应用错误:', err);
  118. res.status(500).json({ error: '服务器内部错误', details: err.message });
  119. });
  120. // 启动服务器
  121. const PORT = process.env.PORT || 3000;
  122. async function startServer() {
  123. server.listen(PORT, async () => {
  124. logger.info(`服务器已启动并监听端口 ${PORT}`);
  125. try {
  126. // 初始化数据库
  127. try {
  128. await initializeDatabase();
  129. logger.success('数据库初始化完成');
  130. } catch (dbError) {
  131. logger.error('数据库初始化失败:', dbError);
  132. logger.warn('将使用文件存储作为备用方案');
  133. }
  134. // 确保目录存在
  135. await ensureDirectoriesExist();
  136. logger.success('系统目录初始化完成');
  137. // 下载必要资源
  138. await downloadImages();
  139. logger.success('资源下载完成');
  140. // 默认使用SQLite数据库模式
  141. try {
  142. logger.info('正在检查SQLite数据库...');
  143. const { isDatabaseReady } = require('./utils/database-checker');
  144. const dbReady = await isDatabaseReady();
  145. if (!dbReady) {
  146. logger.warn('数据库未完全初始化,正在初始化...');
  147. await initializeDatabase();
  148. } else {
  149. logger.info('SQLite数据库已就绪');
  150. }
  151. logger.success('SQLite数据库初始化完成');
  152. } catch (dbError) {
  153. logger.error('SQLite数据库初始化失败:', dbError.message);
  154. throw dbError; // 数据库初始化失败时直接退出
  155. }
  156. // 初始化系统配置
  157. try {
  158. // 系统配置已在数据库初始化时完成
  159. logger.info('系统配置初始化完成');
  160. } catch (initError) {
  161. logger.warn('系统配置初始化遇到问题:', initError.message);
  162. }
  163. // 初始化HTTP代理服务
  164. try {
  165. await httpProxyService.loadConfig();
  166. // 检查环境变量并自动启动代理
  167. await httpProxyService.checkEnvironmentAndAutoStart();
  168. logger.success('HTTP代理服务配置已加载');
  169. } catch (proxyError) {
  170. logger.warn('HTTP代理服务初始化失败:', proxyError.message);
  171. }
  172. // 尝试启动监控
  173. try {
  174. const monitoringService = require('./services/monitoringService');
  175. await monitoringService.startMonitoring();
  176. logger.success('监控服务已启动');
  177. } catch (monitoringError) {
  178. logger.warn('监控服务启动失败:', monitoringError.message);
  179. logger.warn('监控功能可能不可用');
  180. }
  181. // 尝试设置WebSocket
  182. try {
  183. const dockerRouter = require('./routes/docker');
  184. if (typeof dockerRouter.setupLogWebsocket === 'function') {
  185. dockerRouter.setupLogWebsocket(server);
  186. logger.success('WebSocket服务已启动');
  187. }
  188. } catch (wsError) {
  189. logger.warn('WebSocket服务启动失败:', wsError.message);
  190. logger.warn('容器日志实时流可能不可用');
  191. }
  192. logger.success('服务器初始化完成,系统已准备就绪');
  193. } catch (error) {
  194. logger.error('系统初始化失败,但服务仍将继续运行:', error);
  195. }
  196. });
  197. }
  198. startServer();
  199. // 处理进程终止信号
  200. process.on('SIGINT', gracefulShutdown);
  201. process.on('SIGTERM', gracefulShutdown);
  202. // 捕获未处理的Promise拒绝和未捕获的异常
  203. process.on('unhandledRejection', (reason, promise) => {
  204. logger.error('未处理的Promise拒绝:', reason);
  205. if (reason instanceof Error) {
  206. logger.debug('拒绝原因堆栈:', reason.stack);
  207. }
  208. });
  209. process.on('uncaughtException', (error) => {
  210. logger.error('未捕获的异常:', error);
  211. logger.error('错误堆栈:', error.stack);
  212. // 给日志一些时间写入后退出
  213. setTimeout(() => {
  214. logger.fatal('由于未捕获的异常,系统将在3秒后退出');
  215. setTimeout(() => process.exit(1), 3000);
  216. }, 1000);
  217. });
  218. // 导出服务器对象以供测试使用
  219. module.exports = server;