lepaoBeforeRunStateSync.js 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. /**
  2. * 使用 WebVPN Cookie 调用 beforeRunV260,按结果写入 lepao_account / jw_account 状态(与同步逻辑一致)。
  3. * 独立于 webvpnCookie,避免循环依赖;由上层注入 invalidate / refresh WebVPN。
  4. */
  5. const axios = require('axios')
  6. const db = require('../../plugin/DataBase/db')
  7. const { URLSearchParams } = require('url')
  8. const { dataEncrypt, dataDecrypt, dataSign } = require('../../plugin/Lepao/Crypto')
  9. const { withLepaoAccountProxy } = require('./lepaoOutboundProxy')
  10. const DEFAULT_USER_AGENT =
  11. 'Mozilla/5.0 (Linux; Android 16; 2211133C Build/BP2A.250605.031.A3; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/138.0.7204.180 Mobile Safari/537.36 XWEB/1380347 MMWEBSDK/20250202 MMWEBID/1020 wxwork/5.0.6.66174 MicroMessenger/8.0.28.48(0x28001c30) MiniProgramEnv/android Luggage/3.0.2.95ef3f83 NetType/WIFI Language/zh_CN ABI/arm64'
  12. function isProbablyVpnHtml(body) {
  13. if (typeof body !== 'string') return false
  14. return (
  15. body.includes('lyuapServer') ||
  16. body.includes('ivpn') ||
  17. body.includes('统一身份认证') ||
  18. body.includes('rump_frontend/login')
  19. )
  20. }
  21. async function resetJwVerificationState(ownerUuid, jwUsername, updateTime = Date.now()) {
  22. await db.query(
  23. 'UPDATE jw_account SET state = 0, update_time = ? WHERE create_user = ? AND username = ?',
  24. [updateTime, ownerUuid, jwUsername]
  25. )
  26. }
  27. /**
  28. * @param {object} opts
  29. * @param {string} opts.studentNum
  30. * @param {string} opts.ownerUuid
  31. * @param {string} opts.webvpnCookie
  32. * @param {{ uid: string, token: string, school_id: *, userAgent?: string }} opts.account 乐跑账号行
  33. * @param {string} opts.conditionSql 如 student_num = ? AND create_user = ?
  34. * @param {any[]} opts.queryParams UPDATE 占位参数顺序
  35. * @param {() => Promise<void>} opts.invalidateWebVpn
  36. * @param {() => Promise<string>} opts.refreshWebVpnCookie
  37. * @param {object} [opts.logger]
  38. * @returns {Promise<{ ok: boolean, skipped?: boolean, msg?: string, loginExpired?: boolean, data?: object }>}
  39. */
  40. async function syncLepaoStateViaBeforeRun(opts) {
  41. const {
  42. studentNum,
  43. ownerUuid,
  44. webvpnCookie: initialCookie,
  45. account,
  46. conditionSql,
  47. queryParams,
  48. invalidateWebVpn,
  49. refreshWebVpnCookie,
  50. logger
  51. } = opts
  52. if (!account?.uid || !account?.token) {
  53. logger?.warn?.(`[beforeRun同步] 无 uid/token,跳过 student=${studentNum}`)
  54. return { ok: true, skipped: true }
  55. }
  56. let webvpnCookie = initialCookie
  57. const raw = {
  58. uid: account.uid,
  59. token: account.token,
  60. school_id: account.school_id,
  61. term_id: 0,
  62. course_id: 0,
  63. class_id: 0,
  64. student_num: studentNum,
  65. card_id: studentNum,
  66. timestamp: Number((Date.now() / 1000).toFixed(3)),
  67. version: 1,
  68. nonce: String(Math.floor(Math.random() * 900000 + 100000)),
  69. ostype: 5
  70. }
  71. raw.sign = dataSign(raw)
  72. const form = new URLSearchParams()
  73. form.append('ostype', '5')
  74. form.append('data', dataEncrypt(JSON.stringify(raw)))
  75. const buildHeaders = () => ({
  76. 'Content-Type': 'application/x-www-form-urlencoded',
  77. Accept: '*/*',
  78. 'Accept-Language': 'zh-CN,zh-Hans;q=0.9',
  79. 'Accept-Encoding': 'gzip, deflate, br',
  80. Referer: 'https://servicewechat.com/wxf94c4ddb63d87ede/32/page-frame.html',
  81. 'User-Agent': account.userAgent || DEFAULT_USER_AGENT,
  82. Cookie: webvpnCookie
  83. })
  84. let result
  85. try {
  86. const postOnce = async () => {
  87. const execPost = (extraAxios) =>
  88. axios.post('https://lepao.ctbu.edu.cn/v3/api.php/Run2/beforeRunV260', form, {
  89. headers: buildHeaders(),
  90. timeout: 25000,
  91. ...extraAxios,
  92. responseType: 'text',
  93. transformResponse: [(b) => b]
  94. })
  95. return withLepaoAccountProxy(execPost, {
  96. createUserUuid: ownerUuid || null,
  97. studentNum: studentNum || null,
  98. debugAxiosOpts: { proxy: false },
  99. logger
  100. })
  101. }
  102. let apiRes = await postOnce()
  103. try {
  104. result = JSON.parse(apiRes.data)
  105. } catch {
  106. result = apiRes.data
  107. }
  108. if (typeof result === 'string' && isProbablyVpnHtml(result)) {
  109. await invalidateWebVpn()
  110. webvpnCookie = await refreshWebVpnCookie()
  111. apiRes = await postOnce()
  112. try {
  113. result = JSON.parse(apiRes.data)
  114. } catch {
  115. result = apiRes.data
  116. }
  117. }
  118. if (result?.data && result?.is_encrypt === 1) {
  119. result.data = JSON.parse(dataDecrypt(result.data))
  120. }
  121. } catch (error) {
  122. logger?.error?.(`[beforeRun同步] 远端失败 ${studentNum}: ${error.stack || error}`)
  123. return { ok: false, msg: '同步失败,请稍后再试' }
  124. }
  125. const info = result?.info || result?.msg || '系统繁忙,请稍后再试'
  126. const updateTime = Date.now()
  127. if (String(info).includes('重新登录') || Number(result?.status) === 101) {
  128. await db.query(
  129. `UPDATE lepao_account SET state = 0, update_time = ? WHERE ${conditionSql}`,
  130. [updateTime, ...queryParams]
  131. )
  132. await resetJwVerificationState(ownerUuid, studentNum, updateTime)
  133. return { ok: false, msg: info, loginExpired: true }
  134. }
  135. if (!result || Number(result.status) !== 1 || !result.data) {
  136. return { ok: false, msg: info }
  137. }
  138. const term_num = Number(result.data.term_num ?? 0)
  139. const total_num = Number(result.data.total_num ?? 30)
  140. const updateRows = await db.query(
  141. `UPDATE lepao_account SET term_num = ?, total_num = ?, state = 1, update_time = ? WHERE ${conditionSql}`,
  142. [term_num, total_num, updateTime, ...queryParams]
  143. )
  144. if (!updateRows || updateRows.affectedRows !== 1) {
  145. return { ok: false, msg: '数据库更新失败' }
  146. }
  147. return {
  148. ok: true,
  149. data: {
  150. student_num: studentNum,
  151. term_num,
  152. total_num,
  153. state: 1
  154. }
  155. }
  156. }
  157. module.exports = {
  158. syncLepaoStateViaBeforeRun,
  159. resetJwVerificationState,
  160. DEFAULT_USER_AGENT
  161. }