// ==UserScript== // @name 华米运动步数修改 // @namespace https://geoisam.github.io // @version 2.1.0 // @description 每天自动修改并同步 微信运动/支付宝运动 步数,需登录 Zepp/Zepp Life 绑定第三方数据,支持多账号处理和消息推送 // @author geoisam@qq.com // @icon  // @homepage https://scriptcat.org/script-show-page/4285 // @supportURL https://github.com/geoisam/FuckScripts/issues // @crontab * 8-23 once * * // @connect api-user.huami.com // @connect account.huami.com // @connect account-cn.huami.com // @connect api-mifit-cn.huami.com // @connect qyapi.weixin.qq.com // @connect oapi.dingtalk.com // @connect open.feishu.cn // @connect push.i-i.me // @grant unsafeWindow // @grant GM_xmlhttpRequest // @grant GM_notification // @grant GM_openInTab // @grant GM_getValue // @grant GM_setValue // @grant GM_info // @grant GM_log // @tips 此脚本一直为 开源免费 使用,如果你是从某些地方买的话,你就是被骗了 // ==/UserScript== /* ==UserConfig== Config: data: title: 账号#密码(一行一组) type: textarea description: xxxxx@xxx.xxx#xxxxxxxx min: title: 步数最小值(MIN) type: number default: 17760 min: 1 max: 100000 description: 17760 max: title: 步数最大值(MAX) type: number default: 82240 min: 1 max: 100000 description: 82240 span: title: 运行间隔(默认10秒) type: number default: 10 min: 5 description: 10 unit: 秒 Notice: bro: title: 浏览器通知(当前脚本) type: checkbox default: true wework: title: 企业微信消息推送(群机器人) type: text password: true description: xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx dingding: title: 钉钉群机器人(不加签,关键词:#) type: text password: true description: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx feishu: title: 飞书群机器人(不加签,关键词:#) type: text password: true description: xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx pushme: title: PushMe(push.i-i.me) type: text password: true description: xxxxxxxxxxxxxxxxxxxx ==/UserConfig== */ const FuckD = { wh: [ { name: "企业微信", url: "https://qyapi.weixin.qq.com/cgi-bin/webhook/send?key=", key: GM_getValue("Notice.wework", false), msg: { "msgtype": "markdown_v2", "markdown_v2": { get content() { return `> ${FuckD.hm.timeCHINA}\n\n ## ${GM_info.script.name}\n ${FuckD.hm.sendMSG}` } }, }, docs: "https://developer.work.weixin.qq.com/document/path/91770" }, { name: "钉钉", url: "https://oapi.dingtalk.com/robot/send?access_token=", key: GM_getValue("Notice.dingding", false), msg: { "msgtype": "markdown", "markdown": { "title": GM_info.script.name, get text() { return `> ${FuckD.hm.timeCHINA}\n ### ${GM_info.script.name}\n ${FuckD.hm.sendMSG}` } }, }, docs: "https://open.dingtalk.com/document/orgapp/custom-robots-send-group-messages" }, { name: "飞书", url: "https://open.feishu.cn/open-apis/bot/v2/hook/", key: GM_getValue("Notice.feishu", false), msg: { "msg_type": "interactive", "card": { "schema": "2.0", "header": { "title": { "tag": "plain_text", "content": GM_info.script.name }, "template": "orange" }, "body": { "elements": [{ "tag": "markdown", "text_align": "center", get content() { return `#### ${FuckD.hm.timeCHINA}\n ${FuckD.hm.sendMSG}` } }] } } }, docs: "https://open.feishu.cn/document/client-docs/bot-v3/add-custom-bot" }, { name: "PushMe", url: "https://push.i-i.me/?push_key=", key: GM_getValue("Notice.pushme", false), msg: { "type": "markdown", "title": GM_info.script.name, get content() { return `\n ${FuckD.hm.sendMSG}` } }, docs: "https://push.i-i.me/docs/index" }, ], hm: { fraudTIP: "\u6e29\u99a8\u63d0\u793a\uff1a\u60a8\u53ef\u80fd\u662f\u76d7\u7248\u811a\u672c\u7684\u53d7\u5bb3\u8005\uff0c\u8bf7\u6ce8\u610f\u9632\u8303\u8bc8\u9a97\uff01", loginURL: "https://user.huami.com/universalLogin/#/login?project_redirect_uri=https://www.huami.com/", deviceID: "88CC5224060006C4", }, } const FuckF = { getRandomNum(min, max) { return Math.floor(Math.random() * (max + 1 - min) + min) }, isJSON(str) { try { JSON.parse(str) return true } catch (e) { return false } }, formatStr(str) { const lines = str.split(/\r?\n/) const result = [] for (const line of lines) { if (line.trim() === '') continue const pairs = line.split(',') for (const pair of pairs) { const trimmedPair = pair.trim() if (trimmedPair === '') return [] const parts = trimmedPair.split('#') if ( parts.length !== 2 || parts[0].trim() === "" || parts[1].trim() === "" ) { return [] } result.push([parts[0].trim(), parts[1].trim()]) } } return result }, mergeArray(group1, group2) { const group1Map = group1.reduce((map, [user, pwd]) => { map[user] = pwd return map }, {}) const group2Map = group2.reduce((map, [user, token]) => { map[user] = token return map }, {}) const accounts = [...new Set(group1.map(item => item[0]))] const mergedArray = accounts.map(user => [ user, group1Map[user], group2Map[user] || "" ]) return mergedArray }, updateArray(group1, group2) { const map = new Map() group1.forEach(item => { map.set(item[0], [item[1], item[2]]) }) group2.forEach(item => { const [user, token, time] = item map.set(user, [token, time]) }) return Array.from(map, ([user, [token, time]]) => [user, token, time]) }, pushMsg(title, text, push = false, log = true) { if (log) GM_log(title + text + "🔚") if (!GM_getValue("Notice.bro", true) || !push) return GM_notification({ text: text, title: GM_info.script.name + title, onclick: () => { GM_openInTab("https://github.com/geoisam", { active: true, insert: true, setParent: true }) }, }) }, xhr(options) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ ...options, onload: (xhr) => { if (xhr.status == 200) { resolve(xhr.responseText) } else if (xhr.status == 429) { this.pushMsg("🟡", `请求过于频繁,请稍后再试!`, true) resolve(false) } else { this.pushMsg("🔴", `请求失败,状态码:${xhr.status} 🔛${xhr.responseText}`) resolve(false) } }, }) }) }, } return new Promise((resolve, reject) => { const timeUTC = new Date() const newDateSH = new Date(timeUTC.setUTCHours(timeUTC.getUTCHours() + 8)) const dateCHINA = newDateSH.toISOString().split("T")[0] FuckD.hm.timeCHINA = newDateSH.toISOString().split(".")[0].replace("T", " ") if (!FuckD.hm.fraudTIP) resolve() const userDATA = FuckF.formatStr(GM_getValue("Config.data", "#")) if (userDATA.length == 0) FuckF.pushMsg("🔴", "账号#密码填写格式错误,请重新输入!", true), resolve() let tokenDATA, tokenARR, accountARR = GM_getValue("Config.token", []) const accountDATA = FuckF.mergeArray(userDATA, accountARR) FuckD.hm.initHEAD = JSON.stringify(GM_info.script.header).includes(4285) const stepsMIN = GM_getValue("Config.min", 17760) const stepsMAX = GM_getValue("Config.max", 82240) const timeDELAY = GM_getValue("Config.span", 10) if (!FuckD.hm.initHEAD) FuckF.pushMsg("🟣", FuckD.hm.fraudTIP, true) FuckF.tasksStart = async (accounts) => { for (const [index, account] of accounts.entries()) { const secondNOW = Math.floor(Date.now() / 1000) const todaySteps = FuckF.getRandomNum(stepsMIN, stepsMAX) FuckF.pushMsg("🔵", `开始处理第 ${index + 1}/${accounts.length} 个账号「${account[0]}」`) try { let tokenInfo if (account[0].indexOf("@") !== -1) { FuckD.hm.userNAME = account[0] FuckD.hm.thirdNAME = "huami" } else { FuckD.hm.userNAME = `+86${account[0]}` FuckD.hm.thirdNAME = "huami_phone" } const dataJSON = [{ "date": dateCHINA, "data_hr": "/v7+".repeat(480), "data": [{ "start": 0, "stop": 1439, "value": "fgAA".repeat(1440), "tz": 32, "did": FuckD.hm.deviceID, "src": 24, }], "summary": JSON.stringify({ "v": 6, "stp": { "ttl": todaySteps, "dis": Math.floor(todaySteps * 0.7), "cal": Math.floor(todaySteps / 25), "wk": Math.floor(todaySteps / 120), }, "goal": 8000, }), "source": 24, "type": 0, }] if (!account[2]) { tokenInfo = await FuckF.getNewToken(account) } else { tokenInfo = await FuckF.renewToken(account) if (!tokenInfo) tokenInfo = await FuckF.getNewToken(account) } if (!tokenInfo) { account[3] = 0 } else { const submitSteps = await FuckF.xhr({ method: "POST", url: `https://api-mifit-cn.huami.com/v1/data/band_data.json?t=${Date.now()}`, headers: { "content-type": "application/x-www-form-urlencoded; charset=UTF-8", "user-agent": "MiFit/6.12.0 (MCE16; Android 16; Density/1.5)", "app_name": "com.xiaomi.hm.health", "apptoken": tokenInfo.app, }, data: new URLSearchParams({ "userid": tokenInfo.id, "device_type": 0, "last_source": 24, "last_deviceid": FuckD.hm.deviceID, "enableMultiDevice": 1, "last_sync_data_time": secondNOW, "data_json": JSON.stringify(dataJSON), }).toString(), }) if (FuckF.isJSON(submitSteps)) { const resJSON = JSON.parse(submitSteps) if (resJSON.code && resJSON.code == 1) { account[3] = 1 FuckF.pushMsg("🟣", `「${account[0]}」步数数据提交完成! 🔛${submitSteps}`) } else { FuckF.pushMsg("🟡", `「${account[0]}」步数数据提交失败! 🔛${submitSteps}`) } } else { FuckF.pushMsg("🔴", `「${account[0]}」步数数据提交失败! 🔛${submitSteps}`) } account[3] ? account[3] = todaySteps : account[3] = 0 } } catch (e) { FuckF.pushMsg("🔴", `「${account[0]}」处理出错 🔛${e}`) } if (index < accounts.length - 1) { FuckF.pushMsg("🔵", `「${account[0]}」完成,等待 ${timeDELAY} 秒后继续...`) await new Promise(resolve => setTimeout(resolve, (timeDELAY - 1) * 1000)) } await new Promise(resolve => setTimeout(resolve, 1000)) } const sevenDaysAgo = Date.now() - 7 * 24 * 60 * 60 * 1000 accountARR = accountARR.filter(item => (item[2] >= sevenDaysAgo)) GM_setValue("Config.token", accountARR) let dataLOG = accounts.map(account => `|${account[0]}|${account[3]}|\n`).join("") if (!dataLOG) { dataLOG = "**ERROR**" } else { dataLOG = `|账号|步数|\n|:---:|:---:|\n${dataLOG}` } if (!FuckD.hm.initHEAD) dataLOG = `${FuckD.hm.fakeTIP}\n ---\n \n${dataLOG}` FuckD.hm.sendMSG = `---\n ${dataLOG}\n ---\n **[Version ${GM_info.script.version}](https://github.com/geoisam)**\n` FuckF.sendMsgStart(FuckD.wh) FuckF.pushMsg("🟣", "所有账号处理完成,具体请查看日志!", true, false) if (!FuckD.hm.initHEAD) FuckF.pushMsg("🟣", FuckD.hm.fraudTIP, true) resolve() } FuckF.getNewToken = async (account) => { try { const getAccessToken = await FuckF.xhr({ method: "POST", url: `https://api-user.huami.com/registrations/${FuckD.hm.userNAME}/tokens`, headers: { "content-type": "application/x-www-form-urlencoded; charset=UTF-8", "user-agent": "MiFit/6.12.0 (MCE16; Android 16; Density/1.5)", "app_name": "com.xiaomi.hm.health", }, data: new URLSearchParams({ "client_id": "HuaMi", "country_code": "CN", "json_response": true, "name": FuckD.hm.userNAME, "password": account[1], "redirect_uri": "https://s3-us-west-2.amazonaws.com/hm-registration/successsignin.html", "state": "REDIRECTION", "token": "access", }).toString(), }) if (FuckF.isJSON(getAccessToken)) { const resJSON = JSON.parse(getAccessToken) if (resJSON.access) { FuckF.pushMsg("🟢", `「${account[0]}」获取 AccessToken 成功!`) tokenDATA = resJSON.access } else { FuckF.pushMsg("🟡", `「${account[0]}」用户名或密码错误!`, true) return false } } else { FuckF.pushMsg("🔴", `「${account[0]}」获取 AccessToken 失败! 🔛${getAccessToken}`) return false } const getUserInfo = await FuckF.xhr({ method: "POST", url: "https://account.huami.com/v2/client/login", headers: { "content-type": "application/x-www-form-urlencoded; charset=UTF-8", "user-agent": "MiFit/6.12.0 (MCE16; Android 16; Density/1.5)", "app_name": "com.xiaomi.hm.health", }, data: new URLSearchParams({ "app_name": "com.xiaomi.hm.health", "country_code": "CN", "code": tokenDATA, "device_id": "02:00:00:00:00:00", "device_model": "android_phone", "app_version": "6.12.0", "grant_type": "access_token", "allow_registration": false, "source": "com.xiaomi.hm.health", "third_name": FuckD.hm.thirdNAME, }).toString(), }) if (FuckF.isJSON(getUserInfo)) { const resJSON = JSON.parse(getUserInfo) if (resJSON.token_info) { tokenARR = { id: resJSON.token_info.user_id, app: resJSON.token_info.app_token, login: resJSON.token_info.login_token } FuckF.pushMsg("🟢", `「${account[0]}」获取 UserInfo 成功!`) } else { FuckF.pushMsg("🟡", `「${account[0]}」获取 UserInfo 失败! 🔛${getUserInfo}`) return false } } else { FuckF.pushMsg("🔴", `「${account[0]}」获取 UserInfo 失败! 🔛${getUserInfo}`) return false } accountARR = FuckF.updateArray(accountARR, [[account[0], tokenARR.login, Date.now()]]) return tokenARR } catch (e) { FuckF.pushMsg("🔴", `「${account[0]}」登录出错 🔛${e}`) } } FuckF.renewToken = async (account) => { try { const getLoginToken = await FuckF.xhr({ url: `https://account-cn.huami.com/v1/client/renew_login_token?login_token=${account[2]}`, headers: { "content-type": "application/x-www-form-urlencoded; charset=UTF-8", "user-agent": "MiFit/6.12.0 (MCE16; Android 16; Density/1.5)", "app_name": "com.xiaomi.hm.health", }, }) if (FuckF.isJSON(getLoginToken)) { const resJSON = JSON.parse(getLoginToken) if (resJSON.token_info) { FuckF.pushMsg("🟢", `「${account[0]}」获取 LoginToken 成功!`) tokenDATA = resJSON.token_info.login_token } else { FuckF.pushMsg("🟡", `「${account[0]}」获取 LoginToken 失败! 🔛${getLoginToken}`) return false } } else { FuckF.pushMsg("🔴", `「${account[0]}」获取 LoginToken 失败! 🔛${getLoginToken}`) return false } accountARR = FuckF.updateArray(accountARR, [[account[0], tokenDATA, Date.now()]]) const getAppToken = await FuckF.xhr({ url: `https://account-cn.huami.com/v1/client/app_tokens?login_token=${tokenDATA}`, headers: { "content-type": "application/x-www-form-urlencoded; charset=UTF-8", "user-agent": "MiFit/6.12.0 (MCE16; Android 16; Density/1.5)", "app_name": "com.xiaomi.hm.health", }, }) if (FuckF.isJSON(getAppToken)) { const resJSON = JSON.parse(getAppToken) if (resJSON.token_info) { tokenARR = { id: resJSON.token_info.user_id, app: resJSON.token_info.app_token } FuckF.pushMsg("🟢", `「${account[0]}」获取 AppToken 成功!`) } else { FuckF.pushMsg("🟡", `「${account[0]}」获取 AppToken 失败! 🔛${getAppToken}`) return false } } else { FuckF.pushMsg("🔴", `「${account[0]}」获取 AppToken 失败! 🔛${getAppToken}`) return false } return tokenARR } catch (e) { FuckF.pushMsg("🔴", `「${account[0]}」验证出错 🔛${e}`) } } FuckF.sendMsgStart = async (webhook) => { for (const i of webhook) { try { if (!i.key) continue const result = await FuckF.xhr({ method: "POST", url: i.url + i.key, headers: { "content-type": "application/json; charset=UTF-8", }, data: JSON.stringify(i.msg), }) if (result) { FuckF.pushMsg("🟣", `「${i.name}」消息推送完成 🔛${result}`) } else { FuckF.pushMsg("🟡", `「${i.name}」消息推送失败 🔛${result}`) } } catch (e) { FuckF.pushMsg("🔴", `「${i.name}」消息推送出错 🔛${e}`) continue } } } FuckF.tasksStart(accountDATA) })