syncAccountInfo.js 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. const axios = require('axios')
  2. const db = require('../../plugin/DataBase/db')
  3. const { URLSearchParams } = require('url')
  4. const { dataEncrypt, dataDecrypt, dataSign } = require('../../plugin/Lepao/Crypto')
  5. const {
  6. getWebVpnCookieHeader,
  7. invalidateWebVpnCookie,
  8. isProbablyVpnLoginHtml
  9. } = require('./webvpnCookie')
  10. const DEFAULT_USER_AGENT = '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'
  11. async function syncAccountInfo({ studentNum, createUser, logger }) {
  12. if (!studentNum) {
  13. return { ok: false, msg: '缺少学号参数' }
  14. }
  15. const conditionSql = createUser
  16. ? 'student_num = ? AND create_user = ?'
  17. : 'student_num = ?'
  18. const queryParams = createUser ? [studentNum, createUser] : [studentNum]
  19. const rows = await db.query(
  20. `SELECT uid, token, school_id, userAgent, create_user FROM lepao_account WHERE ${conditionSql}`,
  21. queryParams
  22. )
  23. if (!rows || rows.length !== 1) {
  24. return { ok: false, msg: '未找到该乐跑账号或无权限操作' }
  25. }
  26. const account = rows[0]
  27. const ownerUuid = account.create_user || createUser
  28. if (!ownerUuid) {
  29. return { ok: false, msg: '账号未绑定用户,无法同步' }
  30. }
  31. let webvpnCookie
  32. try {
  33. webvpnCookie = await getWebVpnCookieHeader(ownerUuid, studentNum)
  34. } catch (e) {
  35. logger?.error?.(`WebVPN 登录失败 ${studentNum}: ${e.stack || e}`)
  36. return { ok: false, msg: e.message || 'WebVPN 登录失败,请检查教务账号密码' }
  37. }
  38. const raw = {
  39. uid: account.uid,
  40. token: account.token,
  41. school_id: account.school_id,
  42. term_id: 0,
  43. course_id: 0,
  44. class_id: 0,
  45. student_num: studentNum,
  46. card_id: studentNum,
  47. timestamp: Number((Date.now() / 1000).toFixed(3)),
  48. version: 1,
  49. nonce: String(Math.floor(Math.random() * 900000 + 100000)),
  50. ostype: 5
  51. }
  52. raw.sign = dataSign(raw)
  53. const form = new URLSearchParams()
  54. form.append('ostype', '5')
  55. form.append('data', dataEncrypt(JSON.stringify(raw)))
  56. const buildHeaders = () => ({
  57. 'Content-Type': 'application/x-www-form-urlencoded',
  58. 'Accept': '*/*',
  59. 'Accept-Language': 'zh-CN,zh-Hans;q=0.9',
  60. 'Accept-Encoding': 'gzip, deflate, br',
  61. 'Referer': 'https://servicewechat.com/wxf94c4ddb63d87ede/32/page-frame.html',
  62. 'User-Agent': account.userAgent || DEFAULT_USER_AGENT,
  63. Cookie: webvpnCookie
  64. })
  65. let result
  66. try {
  67. const postOnce = () =>
  68. axios.post('https://lepao.ctbu.edu.cn/v3/api.php/Run2/beforeRunV260', form, {
  69. headers: buildHeaders(),
  70. proxy: false,
  71. responseType: 'text',
  72. transformResponse: [(b) => b]
  73. })
  74. let apiRes = await postOnce()
  75. try {
  76. result = JSON.parse(apiRes.data)
  77. } catch {
  78. result = apiRes.data
  79. }
  80. if (typeof result === 'string' && isProbablyVpnLoginHtml(result)) {
  81. await invalidateWebVpnCookie(ownerUuid, studentNum)
  82. webvpnCookie = await getWebVpnCookieHeader(ownerUuid, studentNum, { skipCache: true })
  83. apiRes = await postOnce()
  84. try {
  85. result = JSON.parse(apiRes.data)
  86. } catch {
  87. result = apiRes.data
  88. }
  89. }
  90. if (result?.data && result?.is_encrypt === 1) {
  91. result.data = JSON.parse(dataDecrypt(result.data))
  92. }
  93. } catch (error) {
  94. logger?.error?.(`同步乐跑账号远端请求失败 ${studentNum}: ${error.stack || error}`)
  95. return { ok: false, msg: '同步失败,请稍后再试' }
  96. }
  97. const info = result?.info || result?.msg || '系统繁忙,请稍后再试'
  98. const updateTime = Date.now()
  99. // 登录失效时,仅更新状态并返回失败信息
  100. if (String(info).includes('重新登录') || Number(result?.status) === 101) {
  101. await db.query(
  102. `UPDATE lepao_account SET state = 0, update_time = ? WHERE ${conditionSql}`,
  103. [updateTime, ...queryParams]
  104. )
  105. return { ok: false, msg: info, loginExpired: true }
  106. }
  107. if (!result || Number(result.status) !== 1 || !result.data) {
  108. return { ok: false, msg: info }
  109. }
  110. const term_num = Number(result.data.term_num ?? 0)
  111. const total_num = Number(result.data.total_num ?? 30)
  112. const updateRows = await db.query(
  113. `UPDATE lepao_account SET term_num = ?, total_num = ?, state = 1, update_time = ? WHERE ${conditionSql}`,
  114. [term_num, total_num, updateTime, ...queryParams]
  115. )
  116. if (!updateRows || updateRows.affectedRows !== 1) {
  117. return { ok: false, msg: '数据库更新失败' }
  118. }
  119. return {
  120. ok: true,
  121. data: {
  122. student_num: studentNum,
  123. term_num,
  124. total_num,
  125. state: 1
  126. }
  127. }
  128. }
  129. module.exports = { syncAccountInfo }