diff --git a/musi.js b/musi.js new file mode 100644 index 0000000..99df15c --- /dev/null +++ b/musi.js @@ -0,0 +1,572 @@ + +/** + * 慕思 + * const $ = new Env("慕思"); + * cron 10 6,15 * * * 慕思.js + * 活动入口:https://pic.imgdb.cn/item/64eaffc4661c6c8e54806be6.jpg + + * ========= 青龙--配置文件 =========== + * # 项目名称 + * export mshy='api_token#openId' + * 自己抓包 请求头里的 api_token,请求包里的 openId + * 抓包后不要打开小程序避免刷新了token,有效期待测试,如出现 “Token有误” 则是 token过期了请重新抓取 + * 多账号可用 换行、&、@ 符号隔开,支持新建多个同名环境变量来设置多账号 + * 奖励:签到每天得积分,积分兑换实物 + * ==================================== + */ +const $ = new Env("慕思"); +const utils = require("./utils") +let envParams = 'mshy'; + +let envSplitor = ['\n', '@', '&'] + +let processEnvParams = ($.isNode() ? process.env[envParams] : $.getdata(envParams)) || ''; + +let initedJobForTokens = [] +let currentUserIndex = 0 + +class JobTask { + constructor(str) { + this.requestUA = generateRandomUA(); + this.index = ++currentUserIndex + this.valid = false; + try { + [this.activedAuthToken, this.openId] = str; + } catch (error) { + console.log('参数不完整:自己抓包 请求头里的 api_token,请求包里的 openId') + } + } + + async taskApi (fn, method, url, body, moreHeaders = {}) { + let result = null + try { + let urlObject = { + url: url, + headers: { + 'Accept': '*/*', + 'Accept-Language': 'zh-CN,zh', + 'Connection': 'keep-alive', + 'Host': 'atom.musiyoujia.com', + 'Referer': 'https://servicewechat.com/wx03527497c5369a2c/94/page-frame.html', + 'Sec-Fetch-Dest': 'empty', + 'Sec-Fetch-Mode': 'cors', + 'Sec-Fetch-Site': 'cross-site', + 'User-Agent': this.requestUA || 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.0.0 Safari/537.36 MicroMessenger/7.0.20.1781(0x6700143B) NetType/WIFI MiniProgramEnv/Windows WindowsWechat/WMPF XWEB/8379', + 'api_client_code': '65', + 'api_token': this.activedAuthToken, + 'api_version': '1.0.0', + 'xweb_xhr': '1', + 'Content-Type': 'application/json', + ...moreHeaders + }, + timeout: 60000, + } + if (this.requestUA) { + urlObject.headers['User-Agent'] = this.requestUA + } + if (body) { + urlObject.body = body + // urlObject.headers['Content-Length'] = body?.length + } + // console.log('urlObject:', urlObject); + await httpRequest(method, urlObject).then(async (ret) => { + if (ret.resp?.statusCode == 200) { + if (ret.resp?.body) { + result = JSON.parse(ret.resp.body) + } else { + console.log(`账号[${this.index}]调用${method}[${fn}]出错,返回为空`) + } + } else { + console.log(`账号[${this.index}]调用${method}[${fn}]出错,返回状态码[${ret.resp?.statusCode || ''}]`, '返回结果:', ret.resp?.body || ret?.err) + } + }) + } catch (e) { + console.log(e) + } finally { + return Promise.resolve(result); + } + } + + async GetUserInfo () { + try { + let fn = 'GetUserInfo' + let method = 'post' + let url = `https://atom.musiyoujia.com/member/wechatlogin/selectuserinfo`; + const timestamp = new Date().getTime(); + let moreHeaders = { + 'api_timestamp': timestamp, + 'api_sign': utils?.MD5_Encrypt(`api_client_code=65&api_version=1.0.0&api_timestamp=${timestamp}`)?.toUpperCase() + } + let body = `{"appId":"wx03527497c5369a2c","appType":"WECHAT_MINI_PROGRAM","openId":"${this.openId}"}`; + await this.taskApi(fn, method, url, body, moreHeaders).then(async (result) => { + if (result?.msg === "success") { + this.valid = true; + this.customId = result?.data.resMemberInfo.memberId; + console.log(`账号[${this.index}] 查询个人信息成功,积分:${result?.data?.memberInfo?.pointInfo?.point}`) + } else { + console.log(`账号[${this.index}] 查询个人信息失败:${result?.msg || JSON.stringify(result)}`) + this.valid = false + } + }) + } catch (e) { + console.log(e) + } + } + + async GetJob () { + try { + let fn = 'GetJob' + let method = 'post' + let url = `https://atom.musiyoujia.com/member/memberbehavior/getBehaviorInfos`; + let body = `{"appId":"wx03527497c5369a2c","appType":"WECHAT_MINI_PROGRAM","behaviorIds":[1,2,10203,10204,10205,5],"sourceChannel":"会员小程序","source":"${this.customId}","openId":"${this.openId}"}`; + const timestamp = new Date().getTime(); + let moreHeaders = { + 'api_timestamp': timestamp, + 'api_sign': utils?.MD5_Encrypt(`api_token=${this.activedAuthToken}&api_client_code=65&api_version=1.0.0&api_timestamp=${timestamp}`)?.toUpperCase() + } + await this.taskApi(fn, method, url, body, moreHeaders).then(async (result) => { + if (result?.msg === "success") { + this.isSigned = result?.data[0].acts['每天已获得积分次数'] === 1; + console.log(`账号[${this.index}] 获取任务列表成功,${this.isSigned ? '已签到' : '未签到'}`) + } else { + console.log(`账号[${this.index}] 获取任务列表失败:${result?.msg || JSON.stringify(result)}`) + } + }) + } catch (e) { + console.log(e) + } + } + + async Sign () { + try { + let fn = 'Sign' + let method = 'post' + let url = `https://atom.musiyoujia.com/member/memberbehavior/add`; + const currentDate = new Date(); + const year = currentDate.getFullYear(); + const month = currentDate.getMonth() + 1; + const day = currentDate.getDate(); + let body = `{"appId":"wx03527497c5369a2c","appType":"WECHAT_MINI_PROGRAM","osType":"windows","model":"microsoft","browser":"微信小程序","platform":"1","sourceType":"5","sourceChannel":"会员小程序","siteId":"","visitorId":"","deviceId":"","spotId":"","campaignId":"","deviceType":"","eventLabel":"","eventValue":"","eventAttr2":"${`${year}.${month}.${day}`}","eventAttr3":"","eventAttr4":"","eventAttr5":"","eventAttr6":"","googleCampaignName":"","googleCampaignSource":"","googleCampaignMedium":"","googleCampaignContent":"","memberType":"DeRUCCI","customId":"${this.customId}","locationUrl":"/pages/user/signIn","url":"/pages/user/signIn","pageTitle":"每日签到","logType":"event","behaviorIds":[1,3],"eventCategory":"用户签到","eventAction":"签到","eventAttr1":2,"openId":"${this.openId}"}`; + const timestamp = new Date().getTime(); + let moreHeaders = { + 'api_timestamp': timestamp, + 'api_sign': utils?.MD5_Encrypt(`api_token=${this.activedAuthToken}&api_client_code=65&api_version=1.0.0&api_timestamp=${timestamp}`)?.toUpperCase() + } + await this.taskApi(fn, method, url, body, moreHeaders).then(async (result) => { + if (result?.msg === "success") { + console.log(`账号[${this.index}] 签到成功,获得积分:${result?.data?.point}`) + } else { + console.log(`账号[${this.index}] 签到失败:${result?.msg || JSON.stringify(result)}`) + } + }) + } catch (e) { + console.log(e) + } + } + + async doTask () { + try { + console.log(`\n============= 账号[${this.index}] 开始获取任务=============`); + await this.GetJob(); + if (!this.isSigned) { + console.log(`\n============= 账号[${this.index}] 开始执行 签到任务=============`); + await this.Sign(); + } + } catch (e) { + console.log(e) + } + } +} + +!(async () => { + if (typeof $request !== "undefined") { + await GetRewrite() + } else { + if (!(await checkEnv())) return; + + console.log('\n================ 开始执行 ================') + for (let user of initedJobForTokens) { + console.log(`----------- 执行 第 [${user.index}] 个账号 -----------`) + await user.GetUserInfo(); + } + + let validUserList = initedJobForTokens.filter(x => x.valid) + + if (initedJobForTokens.length > 0) { + console.log('\n================ 任务队列构建完毕 ================') + for (let user of initedJobForTokens) { + console.log(`----------- 账号[${user.index}] -----------`) + await user.doTask(); + } + } else { + console.log('\n====幻生提示:无可用账号,请检查配置============ 任务结束 ================') + } + + await $.showmsg(); + } +})() + .catch((e) => console.log(e)) + .finally(() => $.done()) + +async function GetRewrite () { +} + +async function checkEnv () { + if (processEnvParams) { + let splitor = envSplitor[0]; + for (let sp of envSplitor) { + if (processEnvParams.indexOf(sp) > -1) { + splitor = sp; + break; + } + } + for (let token of processEnvParams.split(splitor)) { + if (token) initedJobForTokens.push(new JobTask(token?.split('#'))) + } + userCount = initedJobForTokens.length + } else { + console.log('未找到 配置信息,请检查是否配置 变量:', envParams) + return; + } + + console.log(`共找到${userCount}个账号`) + return true +} + +function generateRandomUA () { + var androidVersion = Math.floor(Math.random() * 6) + 8; // 生成 Android 版本号 8-13 之间的随机数 + var androidModels = [ + "PBBM00", "SM-G975F", "Pixel 4", "OnePlus 7T", "Redmi Note 8 Pro", + "iPhone X", "Galaxy S10", "Huawei P30 Pro", "LG G8 ThinQ", "Sony Xperia 1", + "Moto G7", "Nokia 9 PureView", "Xiaomi Mi 9", "Google Pixel 3a", "OnePlus 6T", + "Redmi Note 7", "Samsung Galaxy A50", "Huawei Mate 20 Pro", "LG V40 ThinQ", "Sony Xperia XZ3", + "Moto Z4", "Nokia 7.1", "Xiaomi Mi Mix 3", "Google Pixel 2", "OnePlus 5T", + "Redmi Note 6 Pro", "Samsung Galaxy A70", "Huawei P20 Pro", "LG G7 ThinQ", "Sony Xperia XZ2", + "Moto G6", "Nokia 6.1", "Xiaomi Mi 8", "Google Pixel", "OnePlus 5", + "Redmi Note 5 Pro", "Samsung Galaxy A30", "Huawei Mate 10 Pro", "LG V30", "Sony Xperia XZ1", + "Moto X4", "Nokia 5.1", "Xiaomi Mi 6", "Google Pixel XL", "OnePlus 3T", + "Redmi Note 4", "Samsung Galaxy A20", "Huawei P10 Plus", "LG G6", "Sony Xperia XZ", + "Moto G5 Plus", "Nokia 3.1", "Xiaomi Mi 5", "Google Pixel 3", "OnePlus 3", + "Redmi Note 3", "Samsung Galaxy A10", "Huawei Mate 9", "LG V20", "Sony Xperia X Compact", + "Moto G4", "Nokia 2.2", "Xiaomi Mi 4", "Google Pixel 2 XL", "OnePlus 2", + "Redmi Note 2", "Samsung Galaxy A5", "Huawei P9", "LG G5", "Sony Xperia X", + "Moto E6", "Nokia 1", "Xiaomi Mi 3", "Google Pixel 4", "OnePlus One", + "Redmi Note", "Samsung Galaxy A3", "Huawei P8", "LG G4", "Sony Xperia Z5", + "Moto E5", "Nokia 3310", "Xiaomi Mi 2", "Google Pixel 4 XL", "OnePlus Nord" + ]; // 常见的手机型号 + + var chromeVersion = "Chrome/" + (Math.floor(Math.random() * 30) + 50) + ".0." + (Math.floor(Math.random() * 1000) + 100); // 生成 Chrome 版本号 50.0.100-79.0.999 之间的随机数 + + var ua = "Mozilla/5.0 (Linux; Android " + androidVersion + "; " + androidModels[Math.floor(Math.random() * androidModels.length)] + " Build/PPR1.180610.011; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 " + chromeVersion + " Mobile Safari/537.36;xsb_wangchao;xsb_wangchao;5.3.1;native_app"; + + return ua; +} + +// 无参数加密:请求地址+时间戳+A749380BBD5A4D93B55B4BE245A42988+Token +// 示例数据:https://api.ikbang.cn/v2/iclick-new/signIn/sign1692923683034A749380BBD5A4D93B55B4BE245A4298839D21A3B3EFF817154C47470BF13E62CF05F956446D53C359688A094E91A9B1F2A67065862923551F147C32E3AFAF778 +// 如果请求URL有参数则是 在时间戳 和 盐 之间增加 stringify后的参数(POST) + +async function httpRequest (method, url) { + httpErr = null, httpReq = null, httpResp = null; + return new Promise((resolve) => { + $.send(method, url, async (err, req, resp) => { + httpErr = err, httpReq = req, httpResp = resp; + resolve({ err, req, resp }) + }) + }); +} + +function Env (name, env) { + "undefined" != typeof process && JSON.stringify(process.env).indexOf("xcxx") > -1 && process.exit(0); + return new class { + constructor(name, env) { + this.name = name + this.notifyStr = '' + this.startTime = (new Date).getTime() + Object.assign(this, env) + console.log(`${this.name} 开始运行:\n`) + } + isNode () { + return "undefined" != typeof module && !!module.exports + } + isQuanX () { + return "undefined" != typeof $task + } + isSurge () { + return "undefined" != typeof $httpClient && "undefined" == typeof $loon + } + isLoon () { + return "undefined" != typeof $loon + } + getdata (t) { + let e = this.getval(t); + if (/^@/.test(t)) { + const [, s, i] = /^@(.*?)\.(.*?)$/.exec(t), + r = s ? this.getval(s) : ""; + if (r) + try { + const t = JSON.parse(r); + e = t ? this.lodash_get(t, i, "") : e + } catch (t) { + e = "" + } + } + return e + } + setdata (t, e) { + let s = !1; + if (/^@/.test(e)) { + const [, i, r] = /^@(.*?)\.(.*?)$/.exec(e), + o = this.getval(i), + h = i ? "null" === o ? null : o || "{}" : "{}"; + try { + const e = JSON.parse(h); + this.lodash_set(e, r, t), + s = this.setval(JSON.stringify(e), i) + } catch (e) { + const o = {}; + this.lodash_set(o, r, t), + s = this.setval(JSON.stringify(o), i) + } + } + else { + s = this.setval(t, e); + } + return s + } + getval (t) { + return this.isSurge() || this.isLoon() ? $persistentStore.read(t) : this.isQuanX() ? $prefs.valueForKey(t) : this.isNode() ? (this.data = this.loaddata(), this.data[t]) : this.data && this.data[t] || null + } + setval (t, e) { + return this.isSurge() || this.isLoon() ? $persistentStore.write(t, e) : this.isQuanX() ? $prefs.setValueForKey(t, e) : this.isNode() ? (this.data = this.loaddata(), this.data[e] = t, this.writedata(), !0) : this.data && this.data[e] || null + } + send (m, t, e = (() => { })) { + if (m != 'get' && m != 'post' && m != 'put' && m != 'delete') { + console.log(`无效的http方法:${m}`); + return; + } + if (m == 'get' && t.headers) { + delete t.headers["Content-Type"]; + delete t.headers["Content-Length"]; + } else if (t.body && t.headers) { + if (!t.headers["Content-Type"]) t.headers["Content-Type"] = "application/x-www-form-urlencoded"; + } + if (this.isSurge() || this.isLoon()) { + if (this.isSurge() && this.isNeedRewrite) { + t.headers = t.headers || {}; + Object.assign(t.headers, { "X-Surge-Skip-Scripting": !1 }); + } + let conf = { + method: m, + url: t.url, + headers: t.headers, + timeout: t.timeout, + data: t.body + }; + if (m == 'get') delete conf.data + $axios(conf).then(t => { + const { + status: i, + request: q, + headers: r, + data: o + } = t; + e(null, q, { + statusCode: i, + headers: r, + body: o + }); + }).catch(err => console.log(err)) + } else if (this.isQuanX()) { + t.method = m.toUpperCase(), this.isNeedRewrite && (t.opts = t.opts || {}, Object.assign(t.opts, { + hints: !1 + })), + $task.fetch(t).then(t => { + const { + statusCode: i, + request: q, + headers: r, + body: o + } = t; + e(null, q, { + statusCode: i, + headers: r, + body: o + }) + }, t => e(t)) + } else if (this.isNode()) { + this.got = this.got ? this.got : require("got"); + const { + url: s, + ...i + } = t; + this.instance = this.got.extend({ + followRedirect: false + }); + this.instance[m](s, i).then(t => { + const { + statusCode: i, + request: q, + headers: r, + body: o + } = t; + e(null, q, { + statusCode: i, + headers: r, + body: o + }) + }, t => { + const { + message: s, + request: q, + response: i + } = t; + e(s, q, i) + }) + } + } + time (t, x = null) { + let xt = x ? new Date(x) : new Date + let e = { + "M+": xt.getMonth() + 1, + "d+": xt.getDate(), + "h+": xt.getHours(), + "m+": xt.getMinutes(), + "s+": xt.getSeconds(), + "q+": Math.floor((xt.getMonth() + 3) / 3), + S: xt.getMilliseconds() + }; + /(y+)/.test(t) && (t = t.replace(RegExp.$1, (xt.getFullYear() + "").substr(4 - RegExp.$1.length))); + for (let s in e) + new RegExp("(" + s + ")").test(t) && (t = t.replace(RegExp.$1, 1 == RegExp.$1.length ? e[s] : ("00" + e[s]).substr(("" + e[s]).length))); + return t + } + async showmsg () { + if (!this.notifyStr) return; + let notifyBody = this.name + " 运行通知\n\n" + this.notifyStr + if ($.isNode()) { + var notify = require('./sendNotify'); + console.log('\n============== 推送 ==============') + await notify.sendNotify(this.name, notifyBody); + } else { + this.msg(notifyBody); + } + } + logAndNotify (str) { + console.log(str) + this.notifyStr += str + this.notifyStr += '\n' + } + logAndNotifyWithTime (str) { + let t = '[' + this.time('hh:mm:ss.S') + ']' + str + console.log(t) + this.notifyStr += t + this.notifyStr += '\n' + } + logWithTime (str) { + console.log('[' + this.time('hh:mm:ss.S') + ']' + str) + } + msg (e = t, s = "", i = "", r) { + const o = t => { + if (!t) + return t; + if ("string" == typeof t) + return this.isLoon() ? t : this.isQuanX() ? { + "open-url": t + } + : this.isSurge() ? { + url: t + } + : void 0; + if ("object" == typeof t) { + if (this.isLoon()) { + let e = t.openUrl || t.url || t["open-url"], + s = t.mediaUrl || t["media-url"]; + return { + openUrl: e, + mediaUrl: s + } + } + if (this.isQuanX()) { + let e = t["open-url"] || t.url || t.openUrl, + s = t["media-url"] || t.mediaUrl; + return { + "open-url": e, + "media-url": s + } + } + if (this.isSurge()) { + let e = t.url || t.openUrl || t["open-url"]; + return { + url: e + } + } + } + }; + this.isMute || (this.isSurge() || this.isLoon() ? $notification.post(e, s, i, o(r)) : this.isQuanX() && $notify(e, s, i, o(r))); + let h = ["", "============== 系统通知 =============="]; + h.push(e), + s && h.push(s), + i && h.push(i), + console.log(h.join("\n")) + } + getMin (a, b) { + return ((a < b) ? a : b) + } + getMax (a, b) { + return ((a < b) ? b : a) + } + padStr (num, length, padding = '0') { + let numStr = String(num) + let numPad = (length > numStr.length) ? (length - numStr.length) : 0 + let retStr = '' + for (let i = 0; i < numPad; i++) { + retStr += padding + } + retStr += numStr + return retStr; + } + json2str (obj, c, encodeUrl = false) { + let ret = [] + for (let keys of Object.keys(obj).sort()) { + let v = obj[keys] + if (v && encodeUrl) v = encodeURIComponent(v) + ret.push(keys + '=' + v) + } + return ret.join(c); + } + str2json (str, decodeUrl = false) { + let ret = {} + for (let item of str.split('&')) { + if (!item) continue; + let idx = item.indexOf('=') + if (idx == -1) continue; + let k = item.substr(0, idx) + let v = item.substr(idx + 1) + if (decodeUrl) v = decodeURIComponent(v) + ret[k] = v + } + return ret; + } + randomString (len, charset = 'abcdef0123456789') { + let str = ''; + for (let i = 0; i < len; i++) { + str += charset.charAt(Math.floor(Math.random() * charset.length)); + } + return str; + } + randomList (a) { + let idx = Math.floor(Math.random() * a.length) + return a[idx] + } + wait (t) { + return new Promise(e => setTimeout(e, t)) + } + done (t = {}) { + const e = (new Date).getTime(), + s = (e - this.startTime) / 1e3; + console.log(`\n${this.name} 运行结束,共运行了 ${s} 秒!`) + if (this.isSurge() || this.isQuanX() || this.isLoon()) $done(t) + } + }(name, env) +}