Server.js 6.4 KB

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