SingleRun.js 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. const API = require("../../lib/API.js")
  2. const Redis = require('../../plugin/DataBase/Redis')
  3. const db = require("../../plugin/DataBase/db.js")
  4. const { BaseStdResponse } = require("../../BaseStdResponse.js")
  5. const AccessControl = require("../../lib/AccessControl.js")
  6. const mq = require('../../plugin/mq')
  7. const {
  8. assertRunforgeTaskIngress,
  9. publishRunforgeTask
  10. } = require('../../plugin/mq/runforgeTaskMq')
  11. const { planJkesManualRun } = require('../../plugin/jkes/monthPolicy')
  12. const { parsePaceToSecPerKm, clampManualPaceSec } = require('../../plugin/jkes/paceUtils')
  13. const jkesRedisKeys = require('../../plugin/jkes/redisKeys')
  14. const mqNames = require('../../plugin/mq/jkesMqNames')
  15. class SingleRun extends API {
  16. constructor() {
  17. super()
  18. this.setPath('/Lepao/SingleRun')
  19. this.setMethod('GET')
  20. }
  21. async onRequest(req, res) {
  22. let { uuid, session, student_num, distance_km, pace } = req.query
  23. if ([uuid, session, student_num].some(value => value === '' || value === null || value === undefined))
  24. return res.json({
  25. ...BaseStdResponse.MISSING_PARAMETER
  26. })
  27. if (!await AccessControl.checkSession(uuid, session))
  28. return res.status(401).json({
  29. ...BaseStdResponse.ACCESS_DENIED
  30. })
  31. let hour = new Date().getHours()
  32. if (hour < 7)
  33. return res.json({
  34. ...BaseStdResponse.ERR,
  35. msg: '当前不在有效乐跑时间范围内。请在7:00~24:00发起乐跑'
  36. })
  37. try {
  38. const isSuccess = await Redis.get(jkesRedisKeys.lepaoSuccess(student_num))
  39. if (isSuccess)
  40. return res.json({
  41. ...BaseStdResponse.ERR,
  42. msg: '该账号当天已乐跑成功!请勿重复乐跑'
  43. })
  44. const isProgress = await Redis.get(jkesRedisKeys.lepaoProgress(student_num))
  45. if (isProgress)
  46. return res.json({
  47. ...BaseStdResponse.ERR,
  48. msg: '该账号已进入乐跑任务队列,请等待乐跑完成后再进行乐跑操作'
  49. })
  50. let selectSql = 'SELECT create_user FROM lepao_account WHERE student_num = ?'
  51. let selectRows = await db.query(selectSql, [student_num])
  52. if (!selectRows || selectRows.length === 0)
  53. return res.json({
  54. ...BaseStdResponse.ERR,
  55. msg: '发起乐跑失败!未找到账户信息'
  56. })
  57. if (selectRows[0].create_user !== uuid) {
  58. let permission = await AccessControl.getPermission(uuid)
  59. if (!permission.includes("admin") && !permission.includes("service"))
  60. return res.json({
  61. ...BaseStdResponse.ERR,
  62. msg: '发起乐跑失败!未找到账户信息'
  63. })
  64. }
  65. let sql = 'SELECT token, state FROM lepao_account WHERE student_num = ?'
  66. let rows = await db.query(sql, [student_num])
  67. if (!rows || rows.length === 0)
  68. return res.json({
  69. ...BaseStdResponse.ERR,
  70. msg: '发起乐跑失败!未找到对应的账号信息'
  71. })
  72. if (rows[0].state !== 1)
  73. return res.json({
  74. ...BaseStdResponse.ERR,
  75. msg: '账号状态为未登录,请使用登录器更新账号信息后乐跑'
  76. })
  77. if ([distance_km, pace].some((v) => v === '' || v === null || v === undefined))
  78. return res.json({
  79. ...BaseStdResponse.ERR,
  80. msg: '请传入单次距离 distance_km(1–5)与配速 pace(如 5:30、5\'30" 或秒/公里)'
  81. })
  82. const dist = Number(distance_km)
  83. let paceSecPerKm
  84. try {
  85. paceSecPerKm = clampManualPaceSec(parsePaceToSecPerKm(pace))
  86. } catch (e) {
  87. return res.json({
  88. ...BaseStdResponse.ERR,
  89. msg: e.message || '配速无效'
  90. })
  91. }
  92. const plan = await planJkesManualRun(student_num, rows[0].token, dist)
  93. if (!plan.run) {
  94. return res.json({
  95. ...BaseStdResponse.ERR,
  96. msg: plan.reason || '当前无法发起单次乐跑'
  97. })
  98. }
  99. res.json({
  100. ...BaseStdResponse.OK
  101. })
  102. try {
  103. const channel = await mq.getChannel(mqNames.channelLepaoApi)
  104. await assertRunforgeTaskIngress(channel, this.logger)
  105. const taskId = `lepao:${Date.now()}:${student_num}`
  106. const payload = {
  107. id: taskId,
  108. type: 'lepao.startRun',
  109. data: {
  110. taskId,
  111. account: student_num,
  112. targetKm: plan.targetKm,
  113. manual: true,
  114. paceSecPerKm
  115. },
  116. retry: 0
  117. }
  118. publishRunforgeTask(channel, payload)
  119. } catch (err) {
  120. this.logger.error(`后台乐跑任务异常:${err.stack}`)
  121. }
  122. } catch (err) {
  123. this.logger.error(`手动乐跑失败!${err.stack}`)
  124. return res.json({
  125. ...BaseStdResponse.ERR,
  126. msg: "乐跑失败!数据库异常"
  127. })
  128. }
  129. }
  130. }
  131. module.exports.SingleRun = SingleRun