1.购买服务器地址
服务器购买地址
https://t.aliyun.com/U/O2IA2k
若失效,可用地址
https://www.aliyun.com/activity/wuying/dj?source=5176.29345612&userCode=49hts92d
2.部署教程
3. 代码如下
/**
* cron 17 17 * * * 悦拜APP 左上角活动 金币换红包 实物
* 邀请码 JACTEZ 下载地址 微信打开 https://yuebuy.top/a.jDybfP
* 变量名:yuepaiToken
* 变量值:https://app.yuebuy.cn/api/checkin/userinfo Headers请求头中的x-auth-token的值 多账号&或换行或新建同名变量
* scriptVersionNow = "2.1.1";
*/
const $ = new Env("悦拜APP");
const notify = $.isNode() ? require('./sendNotify') : '';
let ckName = "yuepaiToken";
let envSplitor = ["&", "\n"]; //多账号分隔符
let strSplitor = "#"; //多变量分隔符
let userIdx = 0;
let userList = [];
class Task {
constructor(str) {
this.index = ++userIdx;
this.ck = str.split(strSplitor)[0]; //单账号多变量分隔符
this.ckStatus = true;
this.signStatus = false
}
async main() {
await this.task_usefinfo()
if (this.ckStatus) {
if (this.signStatus == false) {
await this.task_sign()
}
await this.task_index();
}
}
async taskRequest(method, url, body = "") {
//
function md5(str) {
const crypto = require('crypto');
return crypto.createHash('md5').update(str).digest('hex');
}
let timestamp = Date.now().toString()
let random = $.uuid()
let md5Str1 = md5(timestamp + random);
let sign = md5(md5Str1 + `YEqQJIyOzUABvTon`);
let headers = {
"push-token": "160a3797c901850e1fc",
"app": "4.2.9",
"system": "10",
"platform": "android",
"channelName": "xiaomi14",
"model": "MI 8 Lite",
"x-auth-token": this.ck,
"x-auth-timestamp": timestamp,
"x-auth-random": random,
"x-auth-signature": sign,
"version": "v1",
"Content-Type": "application/x-www-form-urlencoded",
"Content-Length": "0",
"Host": "app.yuebuy.cn",
"Connection": "Keep-Alive",
"Accept-Encoding": "gzip",
"User-Agent": "okhttp/4.10.0"
}
const requestOptions = {
method: method,
headers: headers,
url: url
};
if (method !== "get") {
requestOptions.body = body;
}
const { body: result } = await $.httpRequest(requestOptions);
return result;
}
async task_usefinfo() {
let result = await this.taskRequest("post", `https://app.yuebuy.cn/api/checkin/userinfo`, `is_simple=0`)
//console.log(options);
//console.log(result);
if (result) {
if (result.code == 1) {
this.ckStatus = true
$.log(`ID [${result.data.user_id}]当前金币[${result.data.coin}] 今日是否签到[${result.data.today_is_checkin == 1}]`)
if (result.data.today_is_checkin == 1) {
this.signStatus = true
}
} else {
this.ckStatus = false
$.log(JSON.stringify(result));
}
}
}
async task_sign() {
let result = await this.taskRequest("post", `https://app.yuebuy.cn/api/checkin/checkin`, ``)
//console.log(options);
//console.log(result);
if (result) {
if (result.code == 0) {
$.log(result.message)
} else {
$.log(result.message);
}
}
}
async task_index() {
let result = await this.taskRequest("post", `https://app.yuebuy.cn/api/checkin/index`, ``)
//console.log(options);
//console.log(result);
if (result) {
if (result.code == 1) {
$.log(`获取任务列表` + result.message)
if (result.message == "Success") {
for (let index of result.data.task_data) {
if (index.title.indexOf("日常") != -1) {
for (let task of index.task_list) {
if (task.title.indexOf("浏览") != -1 || task.title.indexOf("分享") !== -1) {
$.log(`[${task.title}]任务状态[${task.is_finish == 1}]`)
if (task.is_finish == 0) {
$.log(`正在做任务[${task.title}]`)
await this.task_start(task.id)
}
}
}
}
}
}
} else {
$.log(JSON.stringify(result));
}
}
}
async task_start(id) {
let result = await this.taskRequest("post", `https://app.yuebuy.cn/api/checkin/getTask`, `task_id=${id}`)
//console.log(options);
//console.log(result);
if (result) {
if (result.code == 1) {
$.log(`任务开始` + result.message)
if (result.message == "Success") {
await $.wait(16000)
await this.task_finish(result.data.task_sn)
}
} else {
$.log(JSON.stringify(result));
}
}
//console.log(result.data);
}
async task_finish(sn) {
let result = await this.taskRequest("post", `https://app.yuebuy.cn/api/checkin/finishTask`, `task_sn=${sn}`)
//console.log(options);
//console.log(result);
if (result) {
if (result.code == 1) {
$.log(`任务完成` + result.message)
} else {
$.log(JSON.stringify(result));
}
}
//console.log(result.data);
}
}
async function start() {
let taskall = [];
for (let user of userList) {
if (user.ckStatus) {
taskall.push(await user.main());
}
}
await Promise.all(taskall);
}
!(async () => {
if (!(await checkEnv())) return;
if (userList.length > 0) {
await start();
}
await $.sendMsg($.logs.join("\n"))
})()
.catch((e) => console.log(e))
.finally(() => $.done());
//********************************************************
/**
* 变量检查与处理
* @returns
*/
async function checkEnv() {
let userCookie = ($.isNode() ? process.env[ckName] : $.getdata(ckName)) || "";
if (userCookie) {
let e = envSplitor[0];
for (let o of envSplitor)
if (userCookie.indexOf(o) > -1) {
e = o;
break;
}
for (let n of userCookie.split(e)) n && userList.push(new Task(n));
} else {
console.log("未找到CK");
return;
}
return console.log(`共找到${userList.length}个账号`), true; //true == !0
}
function Env(t, s) {
return new (class {
constructor(t, s) {
this.name = t;
this.data = null;
this.dataFile = "box.dat";
this.logs = [];
this.logSeparator = "\n";
this.startTime = new Date().getTime();
Object.assign(this, s);
this.log("", `\ud83d\udd14${this.name}, \u5f00\u59cb!`);
}
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;
}
loaddata() {
if (!this.isNode()) return {};
{
this.fs = this.fs ? this.fs : require("fs");
this.path = this.path ? this.path : require("path");
const 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 {};
{
const i = e ? t : s;
try {
return JSON.parse(this.fs.readFileSync(i));
} catch (t) {
return {};
}
}
}
}
writedata() {
if (this.isNode()) {
this.fs = this.fs ? this.fs : require("fs");
this.path = this.path ? this.path : require("path");
const 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),
o = JSON.stringify(this.data);
e ? this.writeFileSync(t, o) : i ? this.fs.writeFileSync(s, o) : this.fs.writeFileSync(t, o);
}
}
lodash_get(t, s, e) {
const i = s.replace(/\[(\d+)\]/g, ".$1").split(".");
let o = t;
for (const t of i) if (((o = Object(o)[t]), void 0 === o)) return e;
return o;
}
lodash_set(t, s, e) {
return Object(t) !== t
? t
: (Array.isArray(s) || (s = s.toString().match(/[^.[\]]+/g) || []),
(s
.slice(0, -1)
.reduce(
(t, e, i) =>
Object(t[e]) === t[e]
? t[e]
: (t[e] = Math.abs(s[i + 1]) >> 0 == +s[i + 1] ? [] : {}),
t
)[s[s.length - 1]] = e),
t);
}
getdata(t) {
let s = this.getval(t);
if (/^@/.test(t)) {
const [, e, i] = /^@(.*?)\.(.*?)$/.exec(t),
o = e ? this.getval(e) : "";
if (o)
try {
const t = JSON.parse(o);
s = t ? this.lodash_get(t, i, "") : s;
} catch (t) {
s = "";
}
}
return s;
}
setdata(t, s) {
let e = !1;
if (/^@/.test(s)) {
const [, i, o] = /^@(.*?)\.(.*?)$/.exec(s),
h = this.getval(i),
a = i ? ("null" === h ? null : h || "{}") : "{}";
try {
const s = JSON.parse(a);
this.lodash_set(s, o, t), (e = this.setval(JSON.stringify(s), i));
} catch (s) {
const h = {};
this.lodash_set(h, o, t), (e = this.setval(JSON.stringify(h), i));
}
} else e = this.setval(t, s);
return e;
}
getval(t) {
if (this.isSurge() || this.isLoon()) {
return $persistentStore.read(t);
} else if (this.isQuanX()) {
return $prefs.valueForKey(t);
} else if (this.isNode()) {
this.data = this.loaddata();
return this.data[t];
} else {
return this.data && this.data[t] || null;
}
}
setval(t, s) {
if (this.isSurge() || this.isLoon()) {
return $persistentStore.write(t, s);
} else if (this.isQuanX()) {
return $prefs.setValueForKey(t, s);
} else if (this.isNode()) {
this.data = this.loaddata();
this.data[s] = t;
this.writedata();
return true;
} else {
return this.data && this.data[s] || null;
}
}
initGotEnv(t) {
this.got = this.got ? this.got : require("got");
this.cktough = this.cktough ? this.cktough : require("tough-cookie");
this.ckjar = this.ckjar ? this.ckjar : new this.cktough.CookieJar();
if (t) {
t.headers = t.headers ? t.headers : {};
if (typeof t.headers.Cookie === "undefined" && typeof t.cookieJar === "undefined") {
t.cookieJar = this.ckjar;
}
}
}
/**
* @param {Object} options
* @returns {String} 将 Object 对象 转换成 queryStr: key=val&name=senku
*/
queryStr(options) {
return Object.entries(options)
.map(([key, value]) => `${key}=${typeof value === 'object' ? JSON.stringify(value) : value}`)
.join('&');
}
//从url获取参数组成json
getURLParams(url) {
const params = {};
const queryString = url.split('?')[1];
if (queryString) {
const paramPairs = queryString.split('&');
paramPairs.forEach(pair => {
const [key, value] = pair.split('=');
params[key] = decodeURIComponent(value);
});
}
return params;
}
isJSONString(str) {
try {
var obj = JSON.parse(str);
if (typeof obj == 'object' && obj) {
return true;
} else {
return false;
}
} catch (e) {
return false;
}
}
isJson(obj) {
var isjson = typeof (obj) == "object" && Object.prototype.toString.call(obj).toLowerCase() == "[object object]" && !obj.length;
return isjson;
}
async sendMsg(message) {
if (!message) return;
if ($.isNode()) {
await notify.sendNotify($.name, message)
} else {
$.msg($.name, '', message)
}
}
async httpRequest(options) {
let t = {
...options
};
if (!t.headers) {
t.headers = {}
}
if (t.params) {
t.url += '?' + this.queryStr(t.params);
}
t.method = t.method.toLowerCase();
if (t.method === 'get') {
delete t.headers['Content-Type'];
delete t.headers['Content-Length'];
delete t.headers['content-type'];
delete t.headers['content-length'];
delete t["body"]
}
if (t.method === 'post') {
let ContentType;
if (!t.body) {
t.body = ""
} else {
if (typeof t.body == "string") {
if (this.isJSONString(t.body)) {
ContentType = 'application/json'
} else {
ContentType = 'application/x-www-form-urlencoded'
}
} else if (this.isJson(t.body)) {
t.body = JSON.stringify(t.body);
ContentType = 'application/json';
}
}
if (!t.headers['Content-Type']) {
t.headers['Content-Type'] = ContentType;
}
delete t.headers['Content-Length'];
}
if (this.isNode()) {
this.initGotEnv(t);
let httpResult = await this.got(t);
if (this.isJSONString(httpResult.body)) {
httpResult.body = JSON.parse(httpResult.body)
}
return httpResult;
}
if (this.isQuanX()) {
t.method = t.method.toUpperCase()
return new Promise((resolve, reject) => {
$task.fetch(t).then(response => {
if (this.isJSONString(response.body)) {
response.body = JSON.parse(response.body)
}
resolve(response)
})
})
}
}
randomNumber(length) {
const characters = '0123456789';
return Array.from({ length }, () => characters[Math.floor(Math.random() * characters.length)]).join('');
}
randomString(length) {
const characters = 'abcdefghijklmnopqrstuvwxyz0123456789';
return Array.from({ length }, () => characters[Math.floor(Math.random() * characters.length)]).join('');
}
timeStamp() {
return new Date().getTime()
}
uuid() {
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
var r = Math.random() * 16 | 0,
v = c == 'x' ? r : (r & 0x3 | 0x8);
return v.toString(16);
});
}
time(t) {
let s = {
"M+": new Date().getMonth() + 1,
"d+": new Date().getDate(),
"H+": new Date().getHours(),
"m+": new Date().getMinutes(),
"s+": new Date().getSeconds(),
"q+": Math.floor((new Date().getMonth() + 3) / 3),
S: new Date().getMilliseconds(),
};
/(y+)/.test(t) &&
(t = t.replace(
RegExp.$1,
(new Date().getFullYear() + "").substr(4 - RegExp.$1.length)
));
for (let e in s)
new RegExp("(" + e + ")").test(t) &&
(t = t.replace(
RegExp.$1,
1 == RegExp.$1.length
? s[e]
: ("00" + s[e]).substr(("" + s[e]).length)
));
return t;
}
msg(s = t, e = "", i = "", o) {
const h = (t) =>
!t || (!this.isLoon() && this.isSurge())
? t
: "string" == typeof t
? this.isLoon()
? t
: this.isQuanX()
? { "open-url": t }
: void 0
: "object" == typeof t && (t["open-url"] || t["media-url"])
? this.isLoon()
? t["open-url"]
: this.isQuanX()
? t
: void 0
: void 0;
this.isMute ||
(this.isSurge() || this.isLoon()
? $notification.post(s, e, i, h(o))
: this.isQuanX() && $notify(s, e, i, h(o)));
let logs = ['', '==============📣系统通知📣=============='];
logs.push(t);
e ? logs.push(e) : '';
i ? logs.push(i) : '';
console.log(logs.join('\n'));
this.logs = this.logs.concat(logs);
}
log(...t) {
t.length > 0 && (this.logs = [...this.logs, ...t]),
console.log(t.join(this.logSeparator));
}
logErr(t, s) {
const e = !this.isSurge() && !this.isQuanX() && !this.isLoon();
e
? this.log("", `\u2757\ufe0f${this.name}, \u9519\u8bef!`, t.stack)
: this.log("", `\u2757\ufe0f${this.name}, \u9519\u8bef!`, t);
}
wait(t) {
return new Promise((s) => setTimeout(s, t));
}
done(t = {}) {
const s = new Date().getTime(),
e = (s - this.startTime) / 1e3;
this.log(
"",
`\ud83d\udd14${this.name}, \u7ed3\u675f! \ud83d\udd5b ${e} \u79d2`
)
this.log()
if (this.isNode()) {
process.exit(1)
}
if (this.isQuanX()) {
$done(t)
}
}
})(t, s);
}
解析
这个脚本是为悦拜APP自动完成每日签到、金币兑换红包或实物任务的自动化脚本,使用Node.js执行。脚本主要通过模拟HTTP请求,自动执行任务,包括签到、浏览、分享等任务,自动化获取金币和完成任务。
-
cron任务设置:
cron 17 17 * * *
代表每天17:17执行这个脚本。- 悦拜APP活动参与脚本,通过模拟HTTP请求,自动进行金币兑换、签到等操作。
-
环境变量:
- 变量
yuepaiToken
存储了悦拜APP的x-auth-token,用于认证用户的身份。 envSplitor
是多账号分隔符,支持使用&
或\n
分隔多个账号。
- 变量
-
类
Task
:- 每个任务(用户)由这个类处理,初始化时,会根据分隔符提取当前账号信息,并执行任务。
main()
:主要的任务执行流程,依次执行task_usefinfo()
(获取用户信息)、task_sign()
(签到)和task_index()
(获取任务列表)等方法。
-
主要方法:
taskRequest()
:负责发起HTTP请求。每次请求时,会生成时间戳、随机数和MD5签名,用于通过悦拜APP的安全认证。task_usefinfo()
:请求用户信息,包括是否已签到、当前金币等。如果用户已签到,跳过签到任务。task_sign()
:签到请求,若未签到则执行签到操作。task_index()
:获取任务列表(如浏览、分享等任务),并根据任务完成状态自动完成任务。task_start()
和task_finish()
:启动任务并在等待16秒后完成任务(可能是模拟用户浏览或分享的等待时间)。
-
通知功能:
notify
变量可以集成通知功能,通过sendNotify
模块向用户发送完成任务后的通知。
-
辅助方法:
checkEnv()
:检查是否存在有效的账号环境变量,如果找到多个账号,则依次进行处理。Env()
:环境类,用于兼容多种运行环境(如Node.js等),并处理数据的存储、日志记录、网络请求等操作。
此脚本通过模拟HTTP请求,自动化在悦拜APP中执行每日签到、任务列表中的任务(浏览、分享等),帮助用户自动完成金币兑换红包或实物等活动任务。
4. 如何联系我
关注公众号即可私信
今晚务必早点睡