runforgeSetZoneProbe.js 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. /**
  2. * 直连 RunForge 切换跑区(与 Worker lepao.setZone 一致),用于 token 探活,不经过 runpy。
  3. */
  4. const axios = require('axios')
  5. const { URLSearchParams } = require('url')
  6. const db = require('../DataBase/db')
  7. const { dataEncrypt, dataDecrypt, dataSign } = require('./Crypto')
  8. const {
  9. getWebVpnCookieHeader,
  10. invalidateWebVpnCookie,
  11. isProbablyVpnLoginHtml,
  12. isWebVpnUnifiedAuthCredentialFailure,
  13. markLepaoUnifiedAuthFailed
  14. } = require('../../lib/Lepao/webvpnCookie')
  15. const BASE_URL = 'https://lepao.ctbu.edu.cn/v3/api.php'
  16. const DEFAULT_UA =
  17. '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'
  18. const runZoneMap = {
  19. 兰花湖校区跑区: 2,
  20. 主校区北跑区: 3,
  21. 主校区南跑区: 5,
  22. 重庆工商大学茶园校区: 6
  23. }
  24. function lepaoTimestamp() {
  25. return Number((Date.now() / 1000).toFixed(3))
  26. }
  27. /**
  28. * @param {object} p
  29. * @param {string} p.uid
  30. * @param {string} p.token
  31. * @param {string|number} p.school_id
  32. * @param {string} p.student_num
  33. * @param {number} [p.random_id=1] path_data.id,需存在且 run_zone_name 可映射
  34. * @param {string} [p.userAgent]
  35. * @param {string} [p.create_user] 绑定用户 uuid,用于取 WebVPN Cookie(教务登录名与学号 student_num 一致)
  36. * @returns {Promise<object>} RunForge 响应体(与 Worker request 解密后形态一致)
  37. */
  38. async function probeSetZone(p) {
  39. const {
  40. uid,
  41. token,
  42. school_id,
  43. student_num,
  44. random_id = 1,
  45. userAgent,
  46. create_user: createUser
  47. } = p
  48. const record = await db.query('SELECT run_zone_name FROM path_data WHERE id = ?', [random_id])
  49. if (!record || record.length === 0) {
  50. throw new Error('跑区不存在')
  51. }
  52. const runZoneId = runZoneMap[record[0].run_zone_name]
  53. if (!runZoneId) throw new Error('跑区不存在')
  54. const raw = {
  55. uid,
  56. token,
  57. school_id,
  58. term_id: 0,
  59. course_id: 0,
  60. class_id: 0,
  61. student_num,
  62. card_id: student_num,
  63. timestamp: lepaoTimestamp(),
  64. version: 1,
  65. nonce: String(Math.floor(Math.random() * 900000 + 100000)),
  66. ostype: 5,
  67. run_zone_id: String(runZoneId)
  68. }
  69. raw.sign = dataSign(raw)
  70. const form = new URLSearchParams()
  71. form.append('ostype', '5')
  72. form.append('data', dataEncrypt(JSON.stringify(raw)))
  73. let webvpnCookie
  74. if (createUser) {
  75. try {
  76. webvpnCookie = await getWebVpnCookieHeader(createUser, student_num, {
  77. skipPostWebVpnLepaoSync: true
  78. })
  79. } catch (e) {
  80. if (isWebVpnUnifiedAuthCredentialFailure(e)) {
  81. try {
  82. await markLepaoUnifiedAuthFailed(student_num, createUser, student_num)
  83. } catch (_) {
  84. /* ignore */
  85. }
  86. }
  87. throw new Error(e.message || 'WebVPN 登录失败')
  88. }
  89. }
  90. const buildHeaders = () => ({
  91. 'Content-Type': 'application/x-www-form-urlencoded',
  92. Accept: '*/*',
  93. 'Accept-Language': 'zh-CN,zh-Hans;q=0.9',
  94. 'User-Agent': userAgent || DEFAULT_UA,
  95. Referer: 'https://servicewechat.com/wxf94c4ddb63d87ede/32/page-frame.html',
  96. charset: 'utf-8',
  97. ...(webvpnCookie ? { Cookie: webvpnCookie } : {})
  98. })
  99. const postOnce = () =>
  100. axios.post(`${BASE_URL}/Run/setRunZone`, form, {
  101. headers: buildHeaders(),
  102. timeout: 20000,
  103. proxy: false,
  104. responseType: 'text',
  105. transformResponse: [(b) => b]
  106. })
  107. let res = await postOnce()
  108. let result
  109. try {
  110. result = JSON.parse(res.data)
  111. } catch {
  112. result = res.data
  113. }
  114. if (
  115. typeof result === 'string' &&
  116. isProbablyVpnLoginHtml(result) &&
  117. createUser
  118. ) {
  119. await invalidateWebVpnCookie(createUser, student_num)
  120. try {
  121. webvpnCookie = await getWebVpnCookieHeader(createUser, student_num, {
  122. skipCache: true,
  123. skipPostWebVpnLepaoSync: true
  124. })
  125. } catch (vpnRefreshErr) {
  126. if (isWebVpnUnifiedAuthCredentialFailure(vpnRefreshErr)) {
  127. try {
  128. await markLepaoUnifiedAuthFailed(student_num, createUser, student_num)
  129. } catch (_) {
  130. /* ignore */
  131. }
  132. }
  133. throw new Error(vpnRefreshErr.message || 'WebVPN 登录失败')
  134. }
  135. res = await postOnce()
  136. try {
  137. result = JSON.parse(res.data)
  138. } catch {
  139. result = res.data
  140. }
  141. }
  142. if (result?.data && result?.is_encrypt === 1) {
  143. result = { ...result, data: JSON.parse(dataDecrypt(result.data)) }
  144. }
  145. return result
  146. }
  147. function isProbeSetZoneOk(result) {
  148. if (!result) return false
  149. const hasData = result.data != null && result.data !== ''
  150. if (!hasData) return false
  151. if (Number(result.status) === 1) return true
  152. const cd = Number(result.code)
  153. if (cd === 1 || cd === 200) return true
  154. return false
  155. }
  156. function getProbeFailMessage(result) {
  157. if (!result) return ''
  158. return (
  159. result.info ||
  160. result.msg ||
  161. result.message ||
  162. (typeof result.data === 'object' && result.data
  163. ? result.data.info || result.data.msg || result.data.message
  164. : '') ||
  165. ''
  166. )
  167. }
  168. module.exports = {
  169. probeSetZone,
  170. isProbeSetZoneOk,
  171. getProbeFailMessage,
  172. BASE_URL
  173. }