aliYunPanHelper.js 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. /*
  2. 阿里云盘签到-lowking-v1.1.3
  3. 按下面配置完之后,打开阿里云盘获取token(如获取不到,等一段时间再打开),下面配置只验证过surge的,其他的自行测试
  4. ⚠️只测试过surge没有其他app自行测试
  5. ************************
  6. Surge 4.2.0+ 脚本配置(其他APP自行转换配置):
  7. ************************
  8. [Script]
  9. # > 阿里云盘签到
  10. https://auth.alipan.com/v2/account/token
  11. 阿里云盘签到cookie = requires-body=1,type=http-response,pattern=https:\/\/auth.(aliyundrive|alipan).com\/v2\/account\/token,script-path=https://git.jojo21.top/shawenguan/Quantumult-X/raw/master/Scripts/ali/aliYunPanHelper.js
  12. 阿里云盘签到 = type=cron,cronexp="0 10 0 * * ?",wake-system=1,script-path=https://git.jojo21.top/shawenguan/Quantumult-X/raw/master/Scripts/ali/aliYunPanHelper.js
  13. [MITM]
  14. hostname = %APPEND% auth.alipan.com
  15. */
  16. const lk = new ToolKit(`阿里云盘签到`, `AliYunPanHepler`, { "httpApi": "[email protected]:6166" })
  17. const aliYunPanTokenKey = 'lkAliYunPanTokenKey'
  18. let aliYunPanToken = lk.getVal(aliYunPanTokenKey, '')
  19. const aliYunPanRefreshTokenKey = 'lkAliYunPanRefreshTokenKey'
  20. let aliYunPanRefreshToken = lk.getVal(aliYunPanRefreshTokenKey, '')
  21. const checkSignInRepeatKey = 'aliYunPanSignInRepeat'
  22. const checkSignInRepeat = lk.getVal(checkSignInRepeatKey, '')
  23. const joinTeamRepeatKey = 'aliYunPanJoinTeamRepeat'
  24. const joinTeamRepeat = lk.getVal(joinTeamRepeatKey, -1)
  25. lk.userAgent = "Mozilla/5.0 (iPhone; CPU iPhone OS 16_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148 D/C501C6D2-FAF6-4DA8-B65B-7B8B392901EB"
  26. if (!lk.isExecComm) {
  27. if (lk.isRequest()) {
  28. getCookie()
  29. lk.done()
  30. } else {
  31. lk.boxJsJsonBuilder({
  32. "icons": [
  33. "https://git.jojo21.top/shawenguan/Quantumult-X/raw/master/Icons/aliYunPan.png",
  34. "https://git.jojo21.top/shawenguan/Quantumult-X/raw/master/Icons/aliYunPan.png"
  35. ],
  36. "settings": [
  37. {
  38. "id": aliYunPanTokenKey,
  39. "name": "阿里云盘token",
  40. "val": "",
  41. "type": "text",
  42. "desc": "阿里云盘token"
  43. }, {
  44. "id": aliYunPanRefreshTokenKey,
  45. "name": "阿里云盘refresh_token",
  46. "val": "",
  47. "type": "text",
  48. "desc": "阿里云盘refresh_token"
  49. }
  50. ],
  51. "keys": [aliYunPanTokenKey, aliYunPanRefreshTokenKey]
  52. }, {
  53. "script_url": "https://git.jojo21.top/shawenguan/Quantumult-X/raw/master/Scripts/ali/aliYunPanSignIn.js",
  54. "author": "@lowking",
  55. "repo": "https://github.com/lowking/Scripts",
  56. })
  57. all()
  58. }
  59. }
  60. function getCookie() {
  61. if (lk.isGetCookie(/\/v2\/account\/token/)) {
  62. lk.log(`开始获取cookie`)
  63. let data = lk.getResponseBody()
  64. // lk.log(`获取到的cookie:${data}`)
  65. try {
  66. data = JSON.parse(data)
  67. let refreshToken = data["refresh_token"]
  68. if (refreshToken) {
  69. lk.setVal(aliYunPanRefreshTokenKey, refreshToken)
  70. lk.appendNotifyInfo('🎉成功获取阿里云盘refresh_token,可以关闭相应脚本')
  71. } else {
  72. lk.execFail()
  73. lk.appendNotifyInfo('❌获取阿里云盘token失败,请稍后再试')
  74. }
  75. } catch (e) {
  76. lk.execFail()
  77. lk.appendNotifyInfo('❌获取阿里云盘token失败')
  78. }
  79. lk.msg('')
  80. }
  81. }
  82. async function all() {
  83. let hasNeedSendNotify = true
  84. if (aliYunPanRefreshToken == '') {
  85. lk.execFail()
  86. lk.appendNotifyInfo(`⚠️请先打开阿里云盘登录获取refresh_token`)
  87. } else {
  88. await refreshToken()
  89. let hasAlreadySignIn = await signIn()
  90. // await joinTeam()
  91. }
  92. if (hasNeedSendNotify) {
  93. lk.msg(``)
  94. }
  95. lk.done()
  96. }
  97. function refreshToken() {
  98. return new Promise((resolve, _reject) => {
  99. const t = '获取token'
  100. let url = {
  101. url: 'https://auth.alipan.com/v2/account/token',
  102. headers: {
  103. "Content-Type": "application/json; charset=utf-8",
  104. },
  105. body: JSON.stringify({
  106. "grant_type": "refresh_token",
  107. "app_id": "pJZInNHN2dZWk8qg",
  108. "refresh_token": aliYunPanRefreshToken
  109. })
  110. }
  111. lk.post(url, (error, _response, data) => {
  112. try {
  113. if (error) {
  114. lk.execFail()
  115. lk.appendNotifyInfo(`❌${t}失败,请稍后再试`)
  116. } else {
  117. let dataObj = JSON.parse(data)
  118. if (dataObj.hasOwnProperty("refresh_token")) {
  119. aliYunPanToken = `Bearer ${dataObj["access_token"]}`
  120. aliYunPanRefreshToken = dataObj["refresh_token"]
  121. lk.setVal(aliYunPanTokenKey, aliYunPanToken)
  122. lk.setVal(aliYunPanRefreshTokenKey, aliYunPanRefreshToken)
  123. } else {
  124. lk.execFail()
  125. lk.appendNotifyInfo(dataObj.message)
  126. }
  127. }
  128. } catch (e) {
  129. lk.logErr(e)
  130. lk.log(`阿里云盘${t}返回数据:${data}`)
  131. lk.execFail()
  132. lk.appendNotifyInfo(`❌${t}错误,请带上日志联系作者,或稍后再试`)
  133. } finally {
  134. resolve()
  135. }
  136. })
  137. })
  138. }
  139. function getReward(day) {
  140. return new Promise((resolve, _reject) => {
  141. const t = '领取奖励'
  142. let url = {
  143. url: 'https://member.alipan.com/v1/activity/sign_in_reward?_rx-s=mobile',
  144. headers: {
  145. "Content-Type": "application/json",
  146. Authorization: aliYunPanToken,
  147. "User-Agent": lk.userAgent
  148. },
  149. body: JSON.stringify({
  150. "signInDay": day
  151. })
  152. }
  153. lk.post(url, (error, _response, data) => {
  154. try {
  155. if (error) {
  156. lk.execFail()
  157. lk.appendNotifyInfo(`❌第${day}天${t}失败,请稍后再试`)
  158. } else {
  159. lk.log(data)
  160. let dataObj = JSON.parse(data)
  161. if (dataObj.success) {
  162. lk.appendNotifyInfo(`✓${t}(第${day}天),${dataObj?.result?.notice}`)
  163. } else {
  164. lk.execFail()
  165. lk.appendNotifyInfo(`❌第${day}天${t}失败,${dataObj.message}`)
  166. }
  167. }
  168. } catch (e) {
  169. lk.logErr(e)
  170. lk.log(`阿里云盘${t}返回数据:${data}`)
  171. lk.execFail()
  172. lk.appendNotifyInfo(`❌第${day}天${t}错误,请带上日志联系作者,或稍后再试`)
  173. } finally {
  174. resolve()
  175. }
  176. })
  177. })
  178. }
  179. function doJoinTeam(joinTeamId) {
  180. return new Promise(async (resolve, _reject) => {
  181. const t = '加入队伍'
  182. let url = {
  183. url: 'https://member.alipan.com/v1/activity/sign_in_team_pk?_rx-s=mobile',
  184. headers: {
  185. "Content-Type": "application/json",
  186. Authorization: aliYunPanToken,
  187. "User-Agent": lk.userAgent
  188. },
  189. body: JSON.stringify({
  190. id: joinTeamId,
  191. team: "blue"
  192. })
  193. }
  194. lk.post(url, async (error, _response, data) => {
  195. try {
  196. if (error) {
  197. lk.execFail()
  198. lk.appendNotifyInfo(`❌${t}失败,请稍后再试`)
  199. } else {
  200. let dataObj = JSON.parse(data)
  201. if (!dataObj.success) {
  202. lk.execFail()
  203. lk.prependNotifyInfo(dataObj.message)
  204. }
  205. }
  206. } catch (e) {
  207. lk.logErr(e)
  208. lk.log(`阿里云盘${t}返回数据:${data}`)
  209. lk.execFail()
  210. lk.appendNotifyInfo(`❌${t}错误,请带上日志联系作者,或稍后再试`)
  211. } finally {
  212. resolve()
  213. }
  214. })
  215. })
  216. }
  217. function joinTeam(layer = 0) {
  218. return new Promise(async (resolve, _reject) => {
  219. let firstDayOfYear = new Date(lk.now.getFullYear(), 0, 1)
  220. const weekOfYear = Math.ceil((Math.round((lk.now.valueOf() - firstDayOfYear.valueOf()) / 86400000) + ((firstDayOfYear.getDay() + 1) - 1)) / 7)
  221. // if (joinTeamRepeat == weekOfYear) {
  222. // }
  223. const t = '加入PK'
  224. let url = {
  225. url: 'https://member.alipan.com/v1/activity/sign_in_team?_rx-s=mobile',
  226. headers: {
  227. "Content-Type": "application/json",
  228. Authorization: aliYunPanToken,
  229. "User-Agent": lk.userAgent
  230. },
  231. body: JSON.stringify({})
  232. }
  233. lk.post(url, async (error, _response, data) => {
  234. try {
  235. if (error) {
  236. lk.execFail()
  237. lk.appendNotifyInfo(`❌${t}失败,请稍后再试`)
  238. } else {
  239. let dataObj = JSON.parse(data)
  240. if (dataObj.success) {
  241. let joinedTeam = dataObj?.result?.joinTeam
  242. let joinTeamId = dataObj?.result?.id
  243. if (joinedTeam && joinTeamId) {
  244. lk.appendNotifyInfo(`🎉${t}成功\n${dataObj?.result?.period}:${dataObj?.result?.joinCount}(${dataObj?.result[joinedTeam + "WinRate"]})`)
  245. lk.setVal(joinTeamRepeatKey, JSON.stringify(weekOfYear))
  246. } else {
  247. if (layer === 0) {
  248. await doJoinTeam(joinTeamId)
  249. await joinTeam(++layer)
  250. } else {
  251. lk.log(`请求加入队伍异常:${data}`)
  252. }
  253. }
  254. } else {
  255. lk.execFail()
  256. lk.prependNotifyInfo(dataObj.message)
  257. }
  258. }
  259. } catch (e) {
  260. lk.logErr(e)
  261. lk.log(`阿里云盘${t}返回数据:${data}`)
  262. lk.execFail()
  263. lk.appendNotifyInfo(`❌${t}错误,请带上日志联系作者,或稍后再试`)
  264. } finally {
  265. resolve()
  266. }
  267. })
  268. })
  269. }
  270. function signIn() {
  271. return new Promise(async (resolve, _reject) => {
  272. let nowString = lk.formatDate(new Date(), 'yyyyMMdd')
  273. if (nowString == checkSignInRepeat) {
  274. lk.prependNotifyInfo('今日已经签到,无法重复签到~~')
  275. resolve(1)
  276. return
  277. }
  278. const t = '签到'
  279. let url = {
  280. url: 'https://member.alipan.com/v1/activity/sign_in_list',
  281. headers: {
  282. "Content-Type": "application/json",
  283. Authorization: aliYunPanToken,
  284. "User-Agent": lk.userAgent
  285. },
  286. body: JSON.stringify({})
  287. }
  288. lk.post(url, async (error, _response, data) => {
  289. try {
  290. if (error) {
  291. lk.execFail()
  292. lk.appendNotifyInfo(`❌${t}失败,请稍后再试`)
  293. } else {
  294. let dataObj = JSON.parse(data)
  295. if (dataObj.success) {
  296. let prefix = ""
  297. if (dataObj?.result?.signInLogs.length > 0) {
  298. for (const l of dataObj.result.signInLogs) {
  299. if (l?.status != "miss") {
  300. prefix = `第${l?.day}天`
  301. // if (!l?.isReward) {
  302. // await getReward(l?.day)
  303. // }
  304. }
  305. }
  306. }
  307. lk.prependNotifyInfo(`🎉${prefix}${t}成功`)
  308. lk.setVal(checkSignInRepeatKey, nowString)
  309. } else {
  310. lk.execFail()
  311. lk.prependNotifyInfo(dataObj.message)
  312. }
  313. }
  314. } catch (e) {
  315. lk.logErr(e)
  316. lk.log(`阿里云盘${t}返回数据:${data}`)
  317. lk.execFail()
  318. lk.appendNotifyInfo(`❌${t}错误,请带上日志联系作者,或稍后再试`)
  319. } finally {
  320. resolve()
  321. }
  322. })
  323. })
  324. }
  325. //---SyncByPyScript---ToolKit-start
  326. function ToolKit(t, s, e) { return new class { constructor(t, s, e) { this.tgEscapeCharMapping = { "&": "&", "#": "#" }, this.userAgent = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_6) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/12.0.2 Safari/605.1.15", this.prefix = "lk", this.name = t, this.id = s, this.data = null, this.dataFile = this.getRealPath("" + this.prefix + this.id + ".dat"), this.boxJsJsonFile = this.getRealPath("" + this.prefix + this.id + ".boxjs.json"), this.options = e, this.isExecComm = !1, this.isEnableLog = this.getVal(this.prefix + "IsEnableLog" + this.id), this.isEnableLog = !!this.isEmpty(this.isEnableLog) || JSON.parse(this.isEnableLog), this.isNotifyOnlyFail = this.getVal(this.prefix + "NotifyOnlyFail" + this.id), this.isNotifyOnlyFail = !this.isEmpty(this.isNotifyOnlyFail) && JSON.parse(this.isNotifyOnlyFail), this.isEnableTgNotify = this.getVal(this.prefix + "IsEnableTgNotify" + this.id), this.isEnableTgNotify = !this.isEmpty(this.isEnableTgNotify) && JSON.parse(this.isEnableTgNotify), this.tgNotifyUrl = this.getVal(this.prefix + "TgNotifyUrl" + this.id), this.isEnableTgNotify = this.isEnableTgNotify && !this.isEmpty(this.tgNotifyUrl), this.costTotalStringKey = this.prefix + "CostTotalString" + this.id, this.costTotalString = this.getVal(this.costTotalStringKey), this.costTotalString = this.isEmpty(this.costTotalString) ? "0,0" : this.costTotalString.replace('"', ""), this.costTotalMs = this.costTotalString.split(",")[0], this.execCount = this.costTotalString.split(",")[1], this.costTotalMs = this.isEmpty(this.costTotalMs) ? 0 : parseInt(this.costTotalMs), this.execCount = this.isEmpty(this.execCount) ? 0 : parseInt(this.execCount), this.logSeparator = "\n██", this.now = new Date, this.startTime = this.now.getTime(), this.node = this.isNode() ? { request: require("request") } : null, this.execStatus = !0, this.notifyInfo = [], this.log(this.name + ", 开始执行!"), this.initCache(), this.checkRecordRequestBody(), this.execComm() } checkRecordRequestBody() { if (this.isRequest()) { var s = $request.body; if (s) { var e = $request.path; let t = this.id + "#" + e.replace("/", "_"); t = t.replace("?", "#"), this.isQuanX() && $prefs.setValueForKey(s, t), (this.isLoon() || this.isSurge()) && $persistentStore.write(s, t), this.isNode() && this.node.fs.writeFileSync(t + ".json", s, { flag: "w" }, t => console.log(t)) } } } getRequestBody() { var t = $request.path; let s = this.id + "#" + t.replace("/", "_"); if (s = s.replace("?", "#"), this.isSurge() || this.isLoon()) return $persistentStore.read(s); if (this.isQuanX()) return $prefs.valueForKey(s); if (this.isNode()) { t = s + ".json"; if (!this.node.fs.existsSync(t)) return JSON.parse(this.node.fs.readFileSync(t)) } } initCache() { var t, s = this.getPersistKey(); this.isQuanX() && (this.cache = JSON.parse($prefs.valueForKey(s) || "{}")), (this.isLoon() || this.isSurge()) && (this.cache = JSON.parse($persistentStore.read(s) || "{}")), this.isNode() && (this.node.fs.existsSync(t = "root.json") || this.node.fs.writeFileSync(t, JSON.stringify({}), { flag: "wx" }, t => console.log(t)), this.root = {}, this.node.fs.existsSync(t = s + ".json") ? this.cache = JSON.parse(this.node.fs.readFileSync(s + ".json")) : (this.node.fs.writeFileSync(t, JSON.stringify({}), { flag: "wx" }, t => console.log(t)), this.cache = {})) } getPersistKey() { return this.id + "#privateCache" } persistCache() { var t = this.getPersistKey(), s = JSON.stringify(this.cache, null, 2); this.isQuanX() && $prefs.setValueForKey(s, t), (this.isLoon() || this.isSurge()) && $persistentStore.write(s, t), this.isNode() && (this.node.fs.writeFileSync(t + ".json", s, { flag: "w" }, t => console.log(t)), this.node.fs.writeFileSync("root.json", JSON.stringify(this.root, null, 2), { flag: "w" }, t => console.log(t))) } write(t, s) { if (this.log("SET " + s), -1 !== s.indexOf("#")) { if (s = s.substr(1), isSurge || this.isLoon()) return $persistentStore.write(t, s); if (this.isQuanX()) return $prefs.setValueForKey(t, s); this.isNode() && (this.root[s] = t) } else this.cache[s] = t; this.persistCache() } read(t) { return this.log("READ " + t), -1 !== t.indexOf("#") ? (t = t.substr(1), this.isSurge() || this.isLoon() ? $persistentStore.read(t) : this.isQuanX() ? $prefs.valueForKey(t) : this.isNode() ? this.root[t] : void 0) : this.cache[t] } delete(t) { if (this.log("DELETE " + t), -1 !== t.indexOf("#")) { if (t = t.substr(1), this.isSurge() || this.isLoon()) return $persistentStore.write(null, t); if (this.isQuanX()) return $prefs.removeValueForKey(t); this.isNode() && delete this.root[t] } else delete this.cache[t]; this.persistCache() } getRealPath(t) { var s; return this.isNode() ? ((s = process.argv.slice(1, 2)[0].split("/"))[s.length - 1] = t, s.join("/")) : t } getUrlHost(t) { return t.slice(0, t.indexOf("/", 8)) } getUrlPath(t) { var s = t.lastIndexOf("/") === t.length - 1 ? -1 : void 0; return t.slice(t.indexOf("/", 8), s) } async execComm() { if (this.isNode()) { this.comm = process.argv.slice(1); let t = !1; "p" == this.comm[1] && (this.isExecComm = !0, this.log(`开始执行指令【${this.comm[1]}】=> 发送到手机测试脚本!`), this.isEmpty(this.options) || this.isEmpty(this.options.httpApi) ? (this.log("未设置options,使用默认值"), this.isEmpty(this.options) && (this.options = {}), this.options.httpApi = "[email protected]:6166") : /.*?@.*?:[0-9]+/.test(this.options.httpApi) || (t = !0, this.log("❌httpApi格式错误!格式:[email protected]:6166"), this.done()), t || this.callApi(this.comm[2])) } } callApi(t) { let i = this.comm[0], s = (this.log(`获取【${i}】内容传给手机`), ""); this.fs = this.fs || require("fs"), this.path = this.path || require("path"); var e = this.path.resolve(i), r = this.path.resolve(process.cwd(), i), o = this.fs.existsSync(e), h = !o && this.fs.existsSync(r); if (o || h) { h = o ? e : r; try { s = this.fs.readFileSync(h) } catch (t) { s = "" } } else s = ""; o = { url: `http://${this.options.httpApi.split("@")[1]}/v1/scripting/evaluate`, headers: { "X-Key": "" + this.options.httpApi.split("@")[0] }, body: { script_text: "" + s, mock_type: "cron", timeout: !this.isEmpty(t) && 5 < t ? t : 5 }, json: !0 }; this.post(o, (t, s, e) => { this.log(`已将脚本【${i}】发给手机!`), this.done() }) } getCallerFileNameAndLine() { let s; try { throw Error("") } catch (t) { s = t } var t = s.stack.split("\n")[1]; return this.path = this.path || require("path"), `[${t.substring(t.lastIndexOf(this.path.sep) + 1, t.lastIndexOf(":"))}]` } getFunName(t) { t = t.toString(); return t = (t = t.substr("function ".length)).substr(0, t.indexOf("(")) } boxJsJsonBuilder(s, r) { if (this.isNode()) { let i = "/Users/lowking/Desktop/Scripts/lowking.boxjs.json"; if (r && r.hasOwnProperty("target_boxjs_json_path") && (i = r.target_boxjs_json_path), this.fs.existsSync(i)) if (this.isJsonObject(s) && this.isJsonObject(r)) { this.log("using node"); var o = ["settings", "keys"], h = "https://raw.githubusercontent.com/Orz-3"; let e = {}, t = "#lk{script_url}"; if (r && r.hasOwnProperty("script_url") && (t = this.isEmpty(r.script_url) ? "#lk{script_url}" : r.script_url), e.id = "" + this.prefix + this.id, e.name = this.name, e.desc_html = `⚠️使用说明</br>详情【<a href='${t}?raw=true'><font class='red--text'>点我查看</font></a>】`, e.icons = [h + `/mini/master/Alpha/${this.id.toLocaleLowerCase()}.png`, h + `/mini/master/Color/${this.id.toLocaleLowerCase()}.png`], e.keys = [], e.settings = [{ id: this.prefix + "IsEnableLog" + this.id, name: "开启/关闭日志", val: !0, type: "boolean", desc: "默认开启" }, { id: this.prefix + "NotifyOnlyFail" + this.id, name: "只当执行失败才通知", val: !1, type: "boolean", desc: "默认关闭" }, { id: this.prefix + "IsEnableTgNotify" + this.id, name: "开启/关闭Telegram通知", val: !1, type: "boolean", desc: "默认关闭" }, { id: this.prefix + "TgNotifyUrl" + this.id, name: "Telegram通知地址", val: "", type: "text", desc: "Tg的通知地址,如:https://api.telegram.org/bot-token/sendMessage?chat_id=-100140&parse_mode=Markdown&text=" }], e.author = "#lk{author}", e.repo = "#lk{repo}", e.script = t + "?raw=true", !this.isEmpty(s)) for (var n in o) { var a = o[n]; if (!this.isEmpty(s[a])) { if ("settings" === a) for (let t = 0; t < s[a].length; t++) { var l = s[a][t]; for (let t = 0; t < e.settings.length; t++) { var p = e.settings[t]; l.id === p.id && e.settings.splice(t, 1) } } e[a] = e[a].concat(s[a]) } delete s[a] } if (Object.assign(e, s), this.isNode()) { this.fs = this.fs || require("fs"), this.path = this.path || require("path"); var h = this.path.resolve(this.boxJsJsonFile), c = this.path.resolve(process.cwd(), this.boxJsJsonFile), u = this.fs.existsSync(h), d = !u && this.fs.existsSync(c), f = JSON.stringify(e, null, "\t"), u = (!u && d ? this.fs.writeFileSync(c, f) : this.fs.writeFileSync(h, f), JSON.parse(this.fs.readFileSync(i))); if (u.hasOwnProperty("apps") && Array.isArray(u.apps) && 0 < u.apps.length) { d = u.apps, c = d.indexOf(d.filter(t => t.id == e.id)[0]); 0 <= c ? u.apps[c] = e : u.apps.push(e); let s = JSON.stringify(u, null, 2); if (!this.isEmpty(r)) for (const m in r) { let t = ""; r.hasOwnProperty(m) ? t = r[m] : "author" === m ? t = "@lowking" : "repo" === m && (t = "https://github.com/lowking/Scripts"), s = s.replace(`#lk{${m}}`, t) } for (var g, y = /(?:#lk\{)(.+?)(?=\})/, S = (null !== y.exec(s) && this.log("生成BoxJs还有未配置的参数,请参考https://github.com/lowking/Scripts/blob/master/util/example/ToolKitDemo.js#L17-L18传入参数:\n"), new Set); null !== (g = y.exec(s));)S.add(g[1]), s = s.replace(`#lk{${g[1]}}`, ""); S.forEach(t => { console.log(t + " ") }), this.fs.writeFileSync(i, s) } } } else this.log("构建BoxJsJson传入参数格式错误,请传入json对象") } } isJsonObject(t) { return "object" == typeof t && "[object object]" == Object.prototype.toString.call(t).toLowerCase() && !t.length } appendNotifyInfo(t, s) { 1 == s ? this.notifyInfo = t : this.notifyInfo.push(t) } prependNotifyInfo(t) { this.notifyInfo.splice(0, 0, t) } execFail() { this.execStatus = !1 } isRequest() { return "undefined" != typeof $request } isSurge() { return "undefined" != typeof $httpClient } isQuanX() { return "undefined" != typeof $task } isLoon() { return "undefined" != typeof $loon } isJSBox() { return "undefined" != typeof $app && "undefined" != typeof $http } isStash() { return "undefined" != typeof $environment && $environment["stash-version"] } isNode() { return "function" == typeof require && !this.isJSBox() } async sleep(s) { return new Promise(t => setTimeout(t, s)) } async wait(s) { return new Promise(t => setTimeout(t, s)) } async delay(s) { return new Promise(t => setTimeout(t, s)) } log(t) { this.isEnableLog && console.log("" + this.logSeparator + t) } logErr(t) { this.execStatus = !0, this.isEnableLog && (console.log("" + this.logSeparator + this.name + "执行异常:"), console.log(t), console.log("\n" + t.message)) } msg(t, s, e, i) { if ((this.isRequest() || !this.isNotifyOnlyFail || !this.execStatus) && (this.isEmpty(s) && (s = Array.isArray(this.notifyInfo) ? this.notifyInfo.join("\n") : this.notifyInfo), !this.isEmpty(s))) if (this.isEnableTgNotify) { for (var r in this.log(this.name + "Tg通知开始"), this.tgEscapeCharMapping) this.tgEscapeCharMapping.hasOwnProperty(r) && (s = s.replace(r, this.tgEscapeCharMapping[r])); this.get({ url: encodeURI(this.tgNotifyUrl + "📌" + this.name + "\n" + s) }, (t, s, e) => { this.log("Tg通知完毕") }) } else { var o = {}, h = !this.isEmpty(e), n = !this.isEmpty(i); this.isQuanX() && (h && (o["open-url"] = e), n && (o["media-url"] = i), $notify(this.name, t, s, o)), (this.isSurge() || this.isStash()) && (h && (o.url = e), $notification.post(this.name, t, s, o)), this.isNode() && this.log("⭐️" + this.name + "\n" + t + "\n" + s), this.isJSBox() && $push.schedule({ title: this.name, body: t ? t + "\n" + s : s }) } } pushWxMsg(t, s, e, i = () => { }) { s = { appToken: "AT_rTc93GQYIdMU8XLRnoJaSea8WkfhSzhX", content: s, summary: t, contentType: 1, topicIds: [], uids: ["UID_6P4B00X6Zv8U2oKC0I2R09emxtqq"], url: "", verifyPay: !1 }, e && (s.url = e), t = this.getJsonDoneHeaders(), t.Host = "wxpusher.zjiecode.com", t["Content-Type"] = "application/json;charset=UTF-8", e = { url: "https://wxpusher.zjiecode.com/api/send/message", headers: t, body: JSON.stringify(s) }; this.post(e, i) } getVal(t, s = "") { let e; return (e = this.isSurge() || this.isLoon() || this.isStash() ? $persistentStore.read(t) : this.isQuanX() ? $prefs.valueForKey(t) : this.isNode() ? (this.data = this.loadData(), process.env[t] || this.data[t]) : this.data && this.data[t] || null) || s } setVal(t, s) { return this.isSurge() || this.isLoon() || this.isStash() ? $persistentStore.write(s, t) : this.isQuanX() ? $prefs.setValueForKey(s, t) : this.isNode() ? (this.data = this.loadData(), this.data[t] = s, this.writeData(), !0) : this.data && this.data[t] || null } loadData() { if (!this.isNode()) return {}; this.fs = this.fs || require("fs"), this.path = this.path || require("path"); var t = this.path.resolve(this.dataFile), s = this.path.resolve(process.cwd(), this.dataFile), e = this.fs.existsSync(t), i = !e && this.fs.existsSync(s); if (!e && !i) return {}; i = e ? t : s; try { return JSON.parse(this.fs.readFileSync(i)) } catch (t) { return {} } } writeData() { var t, s, e, i, r; this.isNode() && (this.fs = this.fs || require("fs"), this.path = this.path || require("path"), t = this.path.resolve(this.dataFile), s = this.path.resolve(process.cwd(), this.dataFile), i = !(e = this.fs.existsSync(t)) && this.fs.existsSync(s), r = JSON.stringify(this.data), !e && i ? this.fs.writeFileSync(s, r) : this.fs.writeFileSync(t, r)) } adapterStatus(t) { return t && (t.status ? t.statusCode = t.status : t.statusCode && (t.status = t.statusCode)), t } get(t, i = () => { }) { this.isQuanX() && ((t = "string" == typeof t ? { url: t } : t).method = "GET", $task.fetch(t).then(t => { i(null, this.adapterStatus(t), t.body) }, t => i(t.error, null, null))), (this.isSurge() || this.isLoon() || this.isStash()) && $httpClient.get(t, (t, s, e) => { i(t, this.adapterStatus(s), e) }), this.isNode() && this.node.request(t, (t, s, e) => { i(t, this.adapterStatus(s), e) }), this.isJSBox() && ((t = "string" == typeof t ? { url: t } : t).header = t.headers, t.handler = function (t) { let s = t.error, e = (s = s && JSON.stringify(t.error), t.data); "object" == typeof e && (e = JSON.stringify(t.data)), i(s, this.adapterStatus(t.response), e) }, $http.get(t)) } post(t, i = () => { }) { this.isQuanX() && ((t = "string" == typeof t ? { url: t } : t).method = "POST", $task.fetch(t).then(t => { i(null, this.adapterStatus(t), t.body) }, t => i(t.error, null, null))), (this.isSurge() || this.isLoon() || this.isStash()) && $httpClient.post(t, (t, s, e) => { i(t, this.adapterStatus(s), e) }), this.isNode() && this.node.request.post(t, (t, s, e) => { i(t, this.adapterStatus(s), e) }), this.isJSBox() && ((t = "string" == typeof t ? { url: t } : t).header = t.headers, t.handler = function (t) { let s = t.error, e = (s = s && JSON.stringify(t.error), t.data); "object" == typeof e && (e = JSON.stringify(t.data)), i(s, this.adapterStatus(t.response), e) }, $http.post(t)) } put(t, i = () => { }) { this.isQuanX() && ((t = "string" == typeof t ? { url: t } : t).method = "PUT", $task.fetch(t).then(t => { i(null, this.adapterStatus(t), t.body) }, t => i(t.error, null, null))), (this.isSurge() || this.isLoon() || this.isStash()) && (t.method = "PUT", $httpClient.put(t, (t, s, e) => { i(t, this.adapterStatus(s), e) })), this.isNode() && (t.method = "PUT", this.node.request.put(t, (t, s, e) => { i(t, this.adapterStatus(s), e) })), this.isJSBox() && ((t = "string" == typeof t ? { url: t } : t).header = t.headers, t.handler = function (t) { let s = t.error, e = (s = s && JSON.stringify(t.error), t.data); "object" == typeof e && (e = JSON.stringify(t.data)), i(s, this.adapterStatus(t.response), e) }, $http.post(t)) } costTime() { let t = this.name + "执行完毕!"; this.isNode() && this.isExecComm && (t = `指令【${this.comm[1]}】执行完毕!`); var s = (new Date).getTime() - this.startTime, e = s / 1e3; this.execCount++, this.costTotalMs += s, this.log(`${t}耗时【${e}】秒\n总共执行【${this.execCount}】次,平均耗时【${(this.costTotalMs / this.execCount / 1e3).toFixed(4)}】秒`), this.setVal(this.costTotalStringKey, JSON.stringify(this.costTotalMs + "," + this.execCount)) } done(t = {}) { this.costTime(), (this.isSurge() || this.isQuanX() || this.isLoon() || this.isStash()) && $done(t) } getRequestUrl() { return $request.url } getResponseBody() { if ($response) return $response.body } isGetCookie(t) { return !("OPTIONS" == $request.method || !this.getRequestUrl().match(t)) } isEmpty(t) { return void 0 === t || null == t || "" == t || "null" == t || "undefined" == t || 0 === t.length } randomString(s) { s = s || 32; var e = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz1234567890", i = e.length, r = ""; for (let t = 0; t < s; t++)r += e.charAt(Math.floor(Math.random() * i)); return r } autoComplete(s, t, e, i, r, o, h, n, a, l) { if ((s += "").length < r) for (; s.length < r;)0 == o ? s += i : s = i + s; if (h) { let t = ""; for (var p = 0; p < n; p++)t += l; s = s.substring(0, a) + t + s.substring(n + a) } return this.toDBC(s = t + s + e) } customReplace(t, s, e, i) { try { for (var r in this.isEmpty(e) && (e = "#{"), this.isEmpty(i) && (i = "}"), s) t = t.replace("" + e + r + i, s[r]) } catch (t) { this.logErr(t) } return t } toDBC(t) { for (var s = "", e = 0; e < t.length; e++)32 == t.charCodeAt(e) ? s += String.fromCharCode(12288) : t.charCodeAt(e) < 127 && (s += String.fromCharCode(t.charCodeAt(e) + 65248)); return s } hash(t) { let s = 0, e, i; for (e = 0; e < t.length; e++)i = t.charCodeAt(e), s = (s << 5) - s + i, s |= 0; return String(s) } formatDate(t, s) { var e, i = { "M+": t.getMonth() + 1, "d+": t.getDate(), "H+": t.getHours(), "m+": t.getMinutes(), "s+": t.getSeconds(), "q+": Math.floor((t.getMonth() + 3) / 3), S: t.getMilliseconds() }; for (e in /(y+)/.test(s) && (s = s.replace(RegExp.$1, (t.getFullYear() + "").substr(4 - RegExp.$1.length))), i) new RegExp("(" + e + ")").test(s) && (s = s.replace(RegExp.$1, 1 == RegExp.$1.length ? i[e] : ("00" + i[e]).substr(("" + i[e]).length))); return s } parseDate(n, t) { let a = { y: 0, M: 1, d: 0, H: 0, h: 0, m: 0, s: 0, S: 0 }; (t = t || "yyyy-MM-dd").replace(/([^yMdHmsS]*?)(([yMdHmsS])\3*)([^yMdHmsS]*?)/g, function (t, s, e, i, r, o, h) { return n = n.replace(new RegExp(s + "(\\d{" + e.length + "})" + r), function (t, s) { return a[i] = parseInt(s), "" }), "" }), a.M--; t = new Date(a.y, a.M, a.d, a.H, a.m, a.s); return 0 !== a.S && t.setMilliseconds(a.S), t } objToQueryStr(s, e) { let i = ""; for (const r in s) { let t = s[r]; null != t && "" !== t && ("object" == typeof t ? t = JSON.stringify(t) : e && (t = encodeURIComponent(t)), i += `${r}=${t}&`) } return i = i.substring(0, i.length - 1) } parseQueryStr(t) { var s = {}, e = (t = -1 < t.indexOf("?") ? t.split("?")[1] : t).split("&"); for (let t = 0; t < e.length; t++) { var i = e[t].split("="); s[i[0]] = i[1] } return s } deepClone(t, s) { for (var e in s = s || {}, t) "object" == typeof t[e] ? (s[e] = t[e].constructor === Array ? [] : {}, this.deepClone(t[e], s[e])) : s[e] = t[e]; return s } getBaseDoneHeaders(t = {}) { return Object.assign({ "Access-Control-Allow-Origin": "*", "Access-Control-Allow-Methods": "POST,GET,OPTIONS,PUT,DELETE", "Access-Control-Allow-Headers": "Origin, X-Requested-With, Content-Type, Accept" }, t) } getHtmlDoneHeaders() { return this.getBaseDoneHeaders({ "Content-Type": "text/html;charset=UTF-8" }) } getJsonDoneHeaders() { return this.getBaseDoneHeaders({ "Content-Type": "text/json; charset=utf-8", Connection: "keep-alive" }) } shallowClone(t) { var s, e = {}; for (s in t) t.hasOwnProperty(s) && (e[s] = t[s]); return e } parseCookies(t) { let s = {}; return t && t.split(";").forEach(function (t) { t = t.split("="); s[t.shift().trim()] = decodeURI(t.join("=")) }), s } serializeCookies(t) { var s, e = []; for (s in t) { var i = t[s], i = encodeURIComponent(s) + "=" + encodeURIComponent(i); e.push(i) } return e.join("; ") } parseSetCookies(t) { const r = ["Expires", "Max-Age", "Domain", "Path", "HttpOnly", "SameSite"]; t = t.split(";"); let o = null; const h = {}, n = []; return t.forEach(t => { var s, t = t.trim(); let e = null, i = !0; if (t.includes("=") ? (s = t.split("="), e = s[0].trim(), i = s[1].trim()) : e = t, r.includes(e)) h[o] && (h[o][e] = i, h[o].attribs[e] = i); else { o = e; let s = {}; o.includes(",") && o.split(",").forEach(t => { t = t.trim(); r.includes(t) ? s[t] = !0 : o = t }), h[o] = { name: o, value: i, attribs: s }, n.push(h[o]) } }), n } base64Encode(t) { var s, e, i, r = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; let o, h, n; for (n = t.length, h = 0, o = ""; h < n;) { if (s = 255 & t.charCodeAt(h++), h == n) { o = (o += r.charAt(s >> 2)) + r.charAt((3 & s) << 4) + "=="; break } if (e = t.charCodeAt(h++), h == n) { o = (o = (o += r.charAt(s >> 2)) + r.charAt((3 & s) << 4 | (240 & e) >> 4)) + r.charAt((15 & e) << 2) + "="; break } i = t.charCodeAt(h++), o = (o = (o = (o += r.charAt(s >> 2)) + r.charAt((3 & s) << 4 | (240 & e) >> 4)) + r.charAt((15 & e) << 2 | (192 & i) >> 6)) + r.charAt(63 & i) } return o } base64Decode(t) { var s = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/="; if (/([^\s]+[^0-9a-zA-Z\+\/\=]|[^0-9a-zA-Z\+\/\=]\s+)/.test(t)) throw new Error("Invalid base64 input"); var e, i, r, o, h, n, a = t.replace(/\s/g, ""); let l = "", p = 0; for (; p < a.length;)r = s.indexOf(a.charAt(p++)), e = (15 & (o = s.indexOf(a.charAt(p++)))) << 4 | (h = s.indexOf(a.charAt(p++))) >> 2, i = (3 & h) << 6 | (n = s.indexOf(a.charAt(p++))), l += String.fromCharCode(r << 2 | o >> 4), 64 !== h && (l += String.fromCharCode(e)), 64 !== n && (l += String.fromCharCode(i)); return l = this.utf8Decode(l) } utf8Decode(s) { let t = [], e = 0, i = 0, r = 0; for (s = s.replace(/\r\n/g, "\n"); e < s.length;) { i = 255 & s.charCodeAt(e), r = 0, r = i <= 191 ? (i &= 127, 1) : i <= 223 ? (i &= 31, 2) : i <= 239 ? (i &= 15, 3) : (i &= 7, 4); for (let t = 1; t < r; ++t)i = i << 6 | 63 & s.charCodeAt(t + e); 4 === r ? (i -= 65536, t.push(String.fromCharCode(55296 | i >> 10 & 1023)), t.push(String.fromCharCode(56320 | 1023 & i))) : t.push(String.fromCharCode(i)), e += r } return t.join("") } parseJwt(t) { try { var s = t.split("."), e = s[0].replace(/-/g, "+").replace(/_/g, "/"), i = this.base64Decode(e).replace(/\0/g, ""), r = JSON.parse(i), o = s[1].replace(/-/g, "+").replace(/_/g, "/"), h = this.base64Decode(o).replace(/\0/g, ""); return { header: r, payload: JSON.parse(h), signature: s[2] } } catch (t) { return this.log(t), null } } }(t, s, e) }
  327. //---SyncByPyScript---ToolKit-end