runforgeSetZoneProbe.js 4.8 KB

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