// ==UserScript== // @name 华米运动步数修改 // @namespace https://geoisam.github.io // @version 2.0.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: xxx@xxx.xxx#xxxxxxxx(一行一组或英文逗号隔开) min: title: 步数最小值(min:1) type: number default: 17760 min: 1 max: 100000 description: 17760 max: title: 步数最大值(max:100000) type: number default: 82240 min: 1 max: 100000 description: 82240 bro: title: 浏览器通知(当前脚本) type: checkbox default: true Notice: 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 webhook = [ { 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 `> ${pjs.t.timeChina}\n\n ## ${GM_info.script.name}\n ${pjs.t.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 `> ${pjs.t.timeChina}\n ### ${GM_info.script.name}\n ${pjs.t.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 `#### ${pjs.t.timeChina}\n ${pjs.t.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 ${pjs.t.sendMsg}` } }, docs: "https://push.i-i.me/docs/index" } ] const pjs = { t: { dataLOG: "", timeDELAY: 10, timeUTC: new Date(), deviceID: "88CC5224060006C4", secondNOW: Math.floor(Date.now() / 1000), }, huami: { key: "xeNtBVqzDc6tuNTh", iv: "MAAAYAAAAAAAAABg" } } pjs.isJSONParsable = function (str) { try { JSON.parse(str) return true } catch (e) { return false } } pjs.getScopeRandomNum = function (min, max) { return Math.floor(Math.random() * (max + 1 - min) + min) } pjs.formatString = function (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 } pjs.mergeArray = function (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]), ...group2.map(item => item[0]) ])] const mergedArray = accounts.map(user => [ user, group1Map[user] || "", group2Map[user] || "" ]) return mergedArray } pjs.updateArray = function (group1, group2) { const map = new Map(group1) group2.forEach(([user, token]) => { map.set(user, token) }) return Array.from(map) } pjs.pushMsg = function (title, text, push = false) { GM_log(title + text + "🔚") if (!GM_getValue("Config.bro", true) || !push) return GM_notification({ text: text, title: GM_info.script.name + title, onclick: () => { GM_openInTab("https://user.huami.com/universalLogin/#/login?project_redirect_uri=https://www.huami.com/", { active: true, insert: true, setParent: true }) } }) } pjs.isPhone = function (str) { if (/^(1)\d{10}$/.test(str)) { pjs.t.user_name = `+86${str}` pjs.t.third_name = "huami_phone" } else { pjs.t.user_name = str pjs.t.third_name = "huami" } } pjs.encryptAES = async function (str, keyStr, ivStr) { const text = new TextEncoder().encode(str) const keyBuffer = new TextEncoder().encode(keyStr) const ivBuffer = new TextEncoder().encode(ivStr) const cryptoKey = await crypto.subtle.importKey( "raw", keyBuffer, { name: "AES-CBC" }, false, ["encrypt"] ) const encrypted = await crypto.subtle.encrypt( { name: "AES-CBC", iv: ivBuffer }, cryptoKey, text ) return Array.from(new Uint8Array(encrypted)).map(b => b.toString(16).padStart(2, "0")).join("") } pjs.encryptData = function (data) { return new Promise((resolve, reject) => { pjs.encryptAES(data, pjs.huami.key, pjs.huami.iv).then((res) => { resolve(res) }).catch((e) => { reject(e) }) }) } pjs.getAccessToken = function (account) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: "POST", url: `https://api-user.huami.com/registrations/${pjs.t.user_name}/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": pjs.t.user_name, "password": account[1], "redirect_uri": "https://s3-us-west-2.amazonaws.com/hm-registration/successsignin.html", "state": "REDIRECTION", "token": "access", }).toString(), responseType: "json", onload(xhr) { const res = xhr.responseText if (xhr.status == 200) { if (pjs.isJSONParsable(res)) { const resJSON = JSON.parse(res) if (resJSON.access) { pjs.pushMsg("🟢", `「${account[0]}」Access Token获取成功!`) resolve(resJSON.access) } else { pjs.pushMsg("🟡", `「${account[0]}」用户名或密码错误!`, true) resolve(false) } } else { pjs.pushMsg("🔴", `「${account[0]}」AccessToken Code:${xhr.status} 🔛${res}`) resolve(false) } } else if (xhr.status == 429) { pjs.pushMsg("🟡", `「${account[0]}」请求过于频繁!`, true) resolve(false) } else { pjs.pushMsg("🔴", `「${account[0]}」AccessToken Code:${xhr.status} 🔛${res}`) resolve(false) } }, }) }) } pjs.getUserInfo = function (code, account) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ 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": code, "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": pjs.t.third_name, }).toString(), responseType: "json", onload(xhr) { const res = xhr.responseText if (xhr.status == 200) { if (pjs.isJSONParsable(res)) { const resJSON = JSON.parse(res) if (resJSON.token_info) { const token = { id: resJSON.token_info.user_id, app: resJSON.token_info.app_token, login: resJSON.token_info.login_token } pjs.pushMsg("🟢", `「${account[0]}」App Token获取成功!`) resolve(token) } else { pjs.pushMsg("🟡", `「${account[0]}」TokenInfo Code:${xhr.status} 🔛${res}`) resolve(false) } } else { pjs.pushMsg("🔴", `「${account[0]}」TokenInfo Code:${xhr.status} 🔛${res}`) resolve(false) } } else { pjs.pushMsg("🔴", `「${account[0]}」TokenInfo Code:${xhr.status} 🔛${res}`) resolve(false) } }, }) }) } pjs.reLoginToken = function (account) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ 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", }, onload(xhr) { const res = xhr.responseText if (xhr.status == 200) { if (pjs.isJSONParsable(res)) { const resJSON = JSON.parse(res) if (resJSON.token_info) { pjs.pushMsg("🟢", `「${account[0]}」Login Token获取成功!`) resolve(resJSON.token_info.login_token) } else { pjs.pushMsg("🟡", `「${account[0]}」LoginToken Code:${xhr.status} 🔛${res}`) resolve(false) } } else { pjs.pushMsg("🔴", `「${account[0]}」LoginToken Code:${xhr.status} 🔛${res}`) resolve(false) } } else { pjs.pushMsg("🔴", `「${account[0]}」LoginToken Code:${xhr.status} 🔛${res}`) resolve(false) } }, }) }) } pjs.getAppToken = function (code, account) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ url: `https://account-cn.huami.com/v1/client/app_tokens?login_token=${code}`, 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", }, onload(xhr) { const res = xhr.responseText if (xhr.status == 200) { if (pjs.isJSONParsable(res)) { const resJSON = JSON.parse(res) if (resJSON.token_info) { const token = { id: resJSON.token_info.user_id, app: resJSON.token_info.app_token } pjs.pushMsg("🟢", `「${account[0]}」App Token获取成功!`) resolve(token) } else { pjs.pushMsg("🟡", `「${account[0]}」AppToken Code:${xhr.status} 🔛${res}`) resolve(false) } } else { pjs.pushMsg("🔴", `「${account[0]}」AppToken Code:${xhr.status} 🔛${res}`) resolve(false) } } else { pjs.pushMsg("🔴", `「${account[0]}」AppToken Code:${xhr.status} 🔛${res}`) resolve(false) } }, }) }) } pjs.submitSteps = function (info, account) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ 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": info.app, }, data: new URLSearchParams({ "userid": info.id, "last_sync_data_time": pjs.t.secondNOW, "device_type": 0, "last_deviceid": pjs.t.deviceID, "data_json": JSON.stringify(pjs.t.dataJSON), }).toString(), responseType: "json", onload(xhr) { const res = xhr.responseText if (xhr.status == 200) { if (pjs.isJSONParsable(res)) { const resJSON = JSON.parse(res) if (resJSON.code && resJSON.code == 1) { pjs.pushMsg("🟢", `「${account[0]}」步数数据提交成功!`) resolve(1) } else { pjs.pushMsg("🟡", `「${account[0]}」DataSubmit Code:${xhr.status} 🔛${res}`) resolve(false) } } else { pjs.pushMsg("🔴", `「${account[0]}」DataSubmit Code:${xhr.status} 🔛${res}`) resolve(false) } } else { pjs.pushMsg("🔴", `「${account[0]}」DataSubmit Code:${xhr.status} 🔛${res}`) resolve(false) } }, }) }) } return new Promise((resolve, reject) => { pjs.t.userData = pjs.formatString(GM_getValue("Config.data", "#")) if (pjs.t.userData.length == 0) pjs.pushMsg("🔴", "账号#密码填写格式错误,请重新输入!", true), resolve() pjs.t.accountData = pjs.mergeArray(pjs.t.userData, GM_getValue("Config.token", [])) pjs.t.newDateSH = new Date(pjs.t.timeUTC.setUTCHours(pjs.t.timeUTC.getUTCHours() + 8)) pjs.t.timeChina = pjs.t.newDateSH.toISOString().split(".")[0].replace("T", " ") pjs.t.dateChina = pjs.t.newDateSH.toISOString().split("T")[0] pjs.tasksStart = async (accounts) => { for (const [index, account] of accounts.entries()) { pjs.pushMsg("🔵", `开始处理第 ${index + 1}/${accounts.length} 个账号「${account[0]}」`) try { let tokenInfo pjs.isPhone(account[0]) pjs.t.todaySteps = pjs.getScopeRandomNum(GM_getValue("Config.min", 17760), GM_getValue("Config.max", 82240)) pjs.t.dataJSON = [{ "data_hr": "", "date": pjs.t.dateChina, "data": [{ "start": 0, "stop": 1439, "value": pjs.t.todaySteps, "tz": 32, "did": pjs.t.deviceID, "src": 24 }], "summary": JSON.stringify({ "v": 6, "slp": { "st": pjs.t.secondNOW, "ed": pjs.t.secondNOW, }, "stp": { "ttl": pjs.t.todaySteps, }, "goal": 8000, "tz": "28800" }), "source": 24, "type": 0 }] if (!account[2]) { tokenInfo = await pjs.getNewToken(account) } else { tokenInfo = await pjs.renewToken(account) if (!tokenInfo || !tokenInfo.id) tokenInfo = await pjs.getNewToken(account) } if (!tokenInfo || !tokenInfo.id) { account[3] = 0 } else { const isSteps = await pjs.submitSteps(tokenInfo, account) isSteps ? account[3] = pjs.t.todaySteps : account[3] = 0 } } catch (e) { pjs.pushMsg("🔴", `「${account[0]}」处理出错 🔛${e}`) } if (index < accounts.length - 1) { pjs.pushMsg("🔵", `「${account[0]}」完成,等待 ${pjs.t.timeDELAY} 秒后继续...`) await new Promise(resolve => setTimeout(resolve, pjs.t.timeDELAY * 1000)) } } pjs.t.dataLOG = accounts.map(account => `|${account[0]}|${account[3]}|\n`).join("") pjs.t.sendMsg = `---\n |账号|步数|\n |:---:|:---:|\n${pjs.t.dataLOG}\n ---\n **[Version ${GM_info.script.version}](https://github.com/geoisam)**\n` pjs.sendStart(webhook) pjs.pushMsg("🟢", "所有账号处理完成,具体请查看日志!", true) resolve() } pjs.getNewToken = async (account) => { try { const access = await pjs.getAccessToken(account) if (!access) return false const token = await pjs.getUserInfo(access, account) if (!token || !token.login) return false const save = pjs.updateArray(GM_getValue("Config.token", []), [[account[0], token.login]]) GM_setValue("Config.token", save) return token } catch (e) { pjs.pushMsg("🔴", `「${account[0]}」登录出错 🔛${e}`) } } pjs.renewToken = async (account) => { try { const login = await pjs.reLoginToken(account) if (!login) return false const save = pjs.updateArray(GM_getValue("Config.token", []), [[account[0], login]]) GM_setValue("Config.token", save) const token = await pjs.getAppToken(login, account) if (!token || !token.app) return false return token } catch (e) { pjs.pushMsg("🔴", `「${account[0]}」验证出错 🔛${e}`) } } pjs.sendStart = async (wh) => { for (const i of wh) { try { if (!i.key) continue await GM_xmlhttpRequest({ method: "POST", url: i.url + i.key, headers: { "content-type": "application/json; charset=UTF-8", }, data: JSON.stringify(i.msg), onload(xhr) { pjs.pushMsg("🟣", `「${i.name}」消息推送完成 🔛${xhr.responseText}`) return }, }) } catch (e) { pjs.pushMsg("🔴", `「${i.name}」消息推送出错 🔛${e}`) continue } } } pjs.tasksStart(pjs.t.accountData) })