Server.js 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. const express = require('express')
  2. const cors = require('cors')
  3. const path = require('path')
  4. const fs = require('fs')
  5. const config = require('../config.json')
  6. const Logger = require('./Logger')
  7. const MySQL = require('../plugin/DataBase/MySQL')
  8. const Worker = require('./Lepao/Worker')
  9. const mq = require('../plugin/mq')
  10. const { mq: mqName } = require('../plugin/mq/mqPrefix')
  11. const { startLepaoSchedulePublisher } = require('../plugin/mq/lepaoSchedulePublisher')
  12. const OneBotV11 = require('../plugin/OneBot/OneBotV11')
  13. const AccessControl = require('./AccessControl')
  14. const {
  15. resolveServerRole,
  16. shouldServeApi,
  17. shouldRunLepaoWorker
  18. } = require('./serverRole')
  19. const { TASK_QUEUE } = require('../plugin/mq/runforgeTaskMq')
  20. const { PREFIX } = require('../plugin/mq/mqPrefix')
  21. class SERVER {
  22. constructor() {
  23. this.serverRole = resolveServerRole()
  24. this.port = config.port || 3000
  25. this.apiDirectory = path.join(__dirname, '../apis')
  26. const logName = this.serverRole === 'worker' ? 'WorkerServer.log' : 'Server.log'
  27. this.logger = new Logger(path.join(__dirname, '../logs', logName), 'INFO')
  28. this.db = new MySQL()
  29. if (shouldServeApi(this.serverRole)) {
  30. this.app = express()
  31. this.app.use(express.json())
  32. this.app.use(cors())
  33. this.app.use('/uploads', express.static('./uploads'))
  34. this.app.use('/models', express.static('./models'))
  35. this.loadAPIs(this.apiDirectory)
  36. }
  37. }
  38. async initDB() {
  39. try {
  40. this.logger.info('正在测试数据库连接')
  41. await this.db.connect()
  42. await this.db.close()
  43. } catch (error) {
  44. this.logger.error(`数据库连接失败: ${error.stack}`)
  45. process.exit(1)
  46. }
  47. }
  48. async startLepaoWorker() {
  49. const worker = new Worker()
  50. await worker.start()
  51. this.logger.info('RunForge Worker 已启动,正在监听 MQ 任务...')
  52. startLepaoSchedulePublisher({
  53. logger: this.logger,
  54. intervalMs: config.rabbitmq?.lepaoScheduleTickMs ?? 2000,
  55. batch: config.rabbitmq?.lepaoScheduleBatch ?? 100
  56. })
  57. }
  58. async initMQ() {
  59. try {
  60. await mq.init()
  61. const ch = await mq.getChannel('health')
  62. await ch.assertQueue(mqName('mq_health_check'), { durable: false })
  63. this.logger.info('✅ RabbitMQ 初始化 & 测试成功')
  64. if (shouldRunLepaoWorker(this.serverRole)) {
  65. try {
  66. await this.startLepaoWorker()
  67. } catch (err) {
  68. console.error('RunForge Worker 启动失败:', err)
  69. process.exit(1)
  70. }
  71. } else {
  72. this.logger.info(
  73. `serverRole=api,跳过乐跑 Worker;乐跑任务将投递到 MQ 队列「${TASK_QUEUE}」(mqPrefix=${JSON.stringify(PREFIX)})`
  74. )
  75. }
  76. } catch (e) {
  77. this.logger.error('❌ RabbitMQ 初始化失败')
  78. process.exit(1)
  79. }
  80. }
  81. async initOneBot() {
  82. try {
  83. const ok = await OneBotV11.initOneBotWs()
  84. if (ok) {
  85. this.logger.info('OneBot v11 ws 初始化成功,已开始监听消息')
  86. } else {
  87. this.logger.info('OneBot v11 ws 未初始化(可能未启用)')
  88. }
  89. } catch (err) {
  90. this.logger.error(`OneBot v11 ws 初始化失败: ${err.message}`)
  91. }
  92. }
  93. async initAccessControlSchema() {
  94. await AccessControl.ensurePermissionSchema()
  95. }
  96. loadAPIs(directory) {
  97. const items = fs.readdirSync(directory)
  98. items.forEach(item => {
  99. const itemPath = path.join(directory, item)
  100. const stats = fs.statSync(itemPath)
  101. if (stats.isDirectory()) {
  102. this.loadAPIs(itemPath)
  103. } else if (stats.isFile() && itemPath.endsWith('.js')) {
  104. this.loadAPIFile(itemPath)
  105. }
  106. })
  107. }
  108. loadAPIFile(filePath) {
  109. try {
  110. const APIClass = require(filePath)
  111. for (const key in APIClass) {
  112. if (APIClass.hasOwnProperty(key)) {
  113. const apiInstance = new APIClass[key]()
  114. apiInstance.setupRoute()
  115. this.app.use('/', apiInstance.getRouter())
  116. this.logger.info(`已加载API:${apiInstance.path} 类型:${apiInstance.method}`)
  117. }
  118. }
  119. } catch (error) {
  120. this.logger.error(`加载API文件失败: ${filePath},错误: ${error.stack}`)
  121. }
  122. }
  123. listenHttp() {
  124. this.app.listen(this.port, () => {
  125. this.logger.info(`==========服务器正在 ${this.port} 端口上运行 (role=${this.serverRole})==========`)
  126. })
  127. }
  128. async startApiServices() {
  129. try {
  130. await this.initAccessControlSchema()
  131. } catch (err) {
  132. this.logger.error(`权限模型初始化异常: ${err.message}`)
  133. }
  134. try {
  135. await this.initOneBot()
  136. } catch (err) {
  137. this.logger.error(`OneBot 初始化异常: ${err.message}`)
  138. }
  139. this.listenHttp()
  140. }
  141. start() {
  142. this.logger.info(
  143. `============正在启动 (role=${this.serverRole}, lepaoQueue=${TASK_QUEUE}, mqPrefix=${JSON.stringify(PREFIX)})============`
  144. )
  145. this.initDB()
  146. .then(() => this.initMQ())
  147. .then(async () => {
  148. if (!shouldServeApi(this.serverRole)) {
  149. this.logger.info('serverRole=worker,未启动 HTTP API,进程将保持运行以消费乐跑任务')
  150. return
  151. }
  152. await this.startApiServices()
  153. })
  154. .catch((err) => {
  155. this.logger.error(`启动失败: ${err.message || err}`)
  156. if (err.stack) this.logger.error(err.stack)
  157. process.exit(1)
  158. })
  159. }
  160. }
  161. module.exports = SERVER