mirror of
https://github.com/lan-tianxiang/clone_scripts.git
synced 2025-07-14 16:12:36 +08:00
785 lines
35 KiB
JavaScript
Executable File
785 lines
35 KiB
JavaScript
Executable File
/*
|
||
cron 0 0,8,16 * * * jd_joy_reward_new.js
|
||
JDHelloWorld
|
||
脚本兼容: Docker, Node.js
|
||
更新时间:2021-06-29
|
||
活动入口:京东APP我的-宠汪汪
|
||
完成度 2.0%
|
||
|
||
默认500
|
||
export JD_JOY_REWARD_NAME = 500
|
||
*/
|
||
|
||
const $ = new Env("宠汪汪兑换二代目")
|
||
|
||
const fs = require('fs')
|
||
const http = require('http');
|
||
const stream = require('stream');
|
||
const zlib = require('zlib');
|
||
const vm = require('vm');
|
||
const PNG = require('png-js');
|
||
const UA = require('./USER_AGENTS.js').USER_AGENT;
|
||
console.log('\n====================Hello World====================\n')
|
||
|
||
|
||
Math.avg = function average() {
|
||
let sum = 0;
|
||
let len = this.length;
|
||
for (let i = 0; i < len; i++) {
|
||
sum += this[i];
|
||
}
|
||
return sum / len;
|
||
};
|
||
|
||
function sleep(timeout) {
|
||
return new Promise((resolve) => setTimeout(resolve, timeout));
|
||
}
|
||
|
||
class PNGDecoder extends PNG {
|
||
constructor(args) {
|
||
super(args);
|
||
this.pixels = [];
|
||
}
|
||
|
||
decodeToPixels() {
|
||
return new Promise((resolve) => {
|
||
this.decode((pixels) => {
|
||
this.pixels = pixels;
|
||
resolve();
|
||
});
|
||
});
|
||
}
|
||
|
||
getImageData(x, y, w, h) {
|
||
const { pixels } = this;
|
||
const len = w * h * 4;
|
||
const startIndex = x * 4 + y * (w * 4);
|
||
|
||
return { data: pixels.slice(startIndex, startIndex + len) };
|
||
}
|
||
}
|
||
|
||
const PUZZLE_GAP = 8;
|
||
const PUZZLE_PAD = 10;
|
||
|
||
class PuzzleRecognizer {
|
||
constructor(bg, patch, y) {
|
||
// console.log(bg);
|
||
const imgBg = new PNGDecoder(Buffer.from(bg, 'base64'));
|
||
const imgPatch = new PNGDecoder(Buffer.from(patch, 'base64'));
|
||
|
||
// console.log(imgBg);
|
||
|
||
this.bg = imgBg;
|
||
this.patch = imgPatch;
|
||
this.rawBg = bg;
|
||
this.rawPatch = patch;
|
||
this.y = y;
|
||
this.w = imgBg.width;
|
||
this.h = imgBg.height;
|
||
}
|
||
|
||
async run() {
|
||
await this.bg.decodeToPixels();
|
||
await this.patch.decodeToPixels();
|
||
|
||
return this.recognize();
|
||
}
|
||
|
||
recognize() {
|
||
const { ctx, w: width, bg } = this;
|
||
const { width: patchWidth, height: patchHeight } = this.patch;
|
||
const posY = this.y + PUZZLE_PAD + ((patchHeight - PUZZLE_PAD) / 2) - (PUZZLE_GAP / 2);
|
||
// const cData = ctx.getImageData(0, a.y + 10 + 20 - 4, 360, 8).data;
|
||
const cData = bg.getImageData(0, posY, width, PUZZLE_GAP).data;
|
||
const lumas = [];
|
||
|
||
for (let x = 0; x < width; x++) {
|
||
let sum = 0;
|
||
|
||
// y xais
|
||
for (let y = 0; y < PUZZLE_GAP; y++) {
|
||
let idx = x * 4 + y * (width * 4);
|
||
let r = cData[idx];
|
||
let g = cData[idx + 1];
|
||
let b = cData[idx + 2];
|
||
let luma = 0.2126 * r + 0.7152 * g + 0.0722 * b;
|
||
|
||
sum += luma;
|
||
}
|
||
|
||
lumas.push(sum / PUZZLE_GAP);
|
||
}
|
||
|
||
const n = 2; // minium macroscopic image width (px)
|
||
const margin = patchWidth - PUZZLE_PAD;
|
||
const diff = 20; // macroscopic brightness difference
|
||
const radius = PUZZLE_PAD;
|
||
for (let i = 0, len = lumas.length - 2 * 4; i < len; i++) {
|
||
const left = (lumas[i] + lumas[i + 1]) / n;
|
||
const right = (lumas[i + 2] + lumas[i + 3]) / n;
|
||
const mi = margin + i;
|
||
const mLeft = (lumas[mi] + lumas[mi + 1]) / n;
|
||
const mRigth = (lumas[mi + 2] + lumas[mi + 3]) / n;
|
||
|
||
if (left - right > diff && mLeft - mRigth < -diff) {
|
||
const pieces = lumas.slice(i + 2, margin + i + 2);
|
||
const median = pieces.sort((x1, x2) => x1 - x2)[20];
|
||
const avg = Math.avg(pieces);
|
||
|
||
// noise reducation
|
||
if (median > left || median > mRigth) return;
|
||
if (avg > 100) return;
|
||
// console.table({left,right,mLeft,mRigth,median});
|
||
// ctx.fillRect(i+n-radius, 0, 1, 360);
|
||
// console.log(i+n-radius);
|
||
return i + n - radius;
|
||
}
|
||
}
|
||
|
||
// not found
|
||
return -1;
|
||
}
|
||
|
||
runWithCanvas() {
|
||
const { createCanvas, Image } = require('canvas');
|
||
const canvas = createCanvas();
|
||
const ctx = canvas.getContext('2d');
|
||
const imgBg = new Image();
|
||
const imgPatch = new Image();
|
||
const prefix = 'data:image/png;base64,';
|
||
|
||
imgBg.src = prefix + this.rawBg;
|
||
imgPatch.src = prefix + this.rawPatch;
|
||
const { naturalWidth: w, naturalHeight: h } = imgBg;
|
||
canvas.width = w;
|
||
canvas.height = h;
|
||
ctx.clearRect(0, 0, w, h);
|
||
ctx.drawImage(imgBg, 0, 0, w, h);
|
||
|
||
const width = w;
|
||
const { naturalWidth, naturalHeight } = imgPatch;
|
||
const posY = this.y + PUZZLE_PAD + ((naturalHeight - PUZZLE_PAD) / 2) - (PUZZLE_GAP / 2);
|
||
// const cData = ctx.getImageData(0, a.y + 10 + 20 - 4, 360, 8).data;
|
||
const cData = ctx.getImageData(0, posY, width, PUZZLE_GAP).data;
|
||
const lumas = [];
|
||
|
||
for (let x = 0; x < width; x++) {
|
||
let sum = 0;
|
||
|
||
// y xais
|
||
for (let y = 0; y < PUZZLE_GAP; y++) {
|
||
let idx = x * 4 + y * (width * 4);
|
||
let r = cData[idx];
|
||
let g = cData[idx + 1];
|
||
let b = cData[idx + 2];
|
||
let luma = 0.2126 * r + 0.7152 * g + 0.0722 * b;
|
||
|
||
sum += luma;
|
||
}
|
||
|
||
lumas.push(sum / PUZZLE_GAP);
|
||
}
|
||
|
||
const n = 2; // minium macroscopic image width (px)
|
||
const margin = naturalWidth - PUZZLE_PAD;
|
||
const diff = 20; // macroscopic brightness difference
|
||
const radius = PUZZLE_PAD;
|
||
for (let i = 0, len = lumas.length - 2 * 4; i < len; i++) {
|
||
const left = (lumas[i] + lumas[i + 1]) / n;
|
||
const right = (lumas[i + 2] + lumas[i + 3]) / n;
|
||
const mi = margin + i;
|
||
const mLeft = (lumas[mi] + lumas[mi + 1]) / n;
|
||
const mRigth = (lumas[mi + 2] + lumas[mi + 3]) / n;
|
||
|
||
if (left - right > diff && mLeft - mRigth < -diff) {
|
||
const pieces = lumas.slice(i + 2, margin + i + 2);
|
||
const median = pieces.sort((x1, x2) => x1 - x2)[20];
|
||
const avg = Math.avg(pieces);
|
||
|
||
// noise reducation
|
||
if (median > left || median > mRigth) return;
|
||
if (avg > 100) return;
|
||
// console.table({left,right,mLeft,mRigth,median});
|
||
// ctx.fillRect(i+n-radius, 0, 1, 360);
|
||
// console.log(i+n-radius);
|
||
return i + n - radius;
|
||
}
|
||
}
|
||
|
||
// not found
|
||
return -1;
|
||
}
|
||
}
|
||
|
||
const DATA = {
|
||
"appId": "17839d5db83",
|
||
"scene": "cww",
|
||
"product": "embed",
|
||
"lang": "zh_CN",
|
||
};
|
||
const SERVER = '124.250.18.211';
|
||
|
||
// ping iv.jd.com
|
||
|
||
class JDJRValidator {
|
||
constructor() {
|
||
this.data = {};
|
||
this.x = 0;
|
||
this.t = Date.now();
|
||
}
|
||
|
||
async run() {
|
||
const tryRecognize = async () => {
|
||
const x = await this.recognize();
|
||
|
||
if (x > 0) {
|
||
return x;
|
||
}
|
||
// retry
|
||
return await tryRecognize();
|
||
};
|
||
const puzzleX = await tryRecognize();
|
||
console.log(puzzleX);
|
||
const pos = new MousePosFaker(puzzleX).run();
|
||
const d = getCoordinate(pos);
|
||
await sleep(pos[pos.length - 1][2] - Date.now());
|
||
const result = await JDJRValidator.jsonp('/slide/s.html', { d, ...this.data });
|
||
|
||
if (result.message === 'success') {
|
||
console.log(result);
|
||
console.log('JDJRValidator: %fs', (Date.now() - this.t) / 1000);
|
||
return result;
|
||
} else {
|
||
console.count(JSON.stringify(result));
|
||
await sleep(300);
|
||
return await this.run();
|
||
}
|
||
}
|
||
|
||
async recognize() {
|
||
const data = await JDJRValidator.jsonp('/slide/g.html', { e: '' });
|
||
const { bg, patch, y } = data;
|
||
// const uri = 'data:image/png;base64,';
|
||
// const re = new PuzzleRecognizer(uri+bg, uri+patch, y);
|
||
const re = new PuzzleRecognizer(bg, patch, y);
|
||
const puzzleX = await re.run();
|
||
|
||
if (puzzleX > 0) {
|
||
this.data = {
|
||
c: data.challenge,
|
||
w: re.w,
|
||
e: '',
|
||
s: '',
|
||
o: '',
|
||
};
|
||
this.x = puzzleX;
|
||
}
|
||
return puzzleX;
|
||
}
|
||
|
||
async report(n) {
|
||
console.time('PuzzleRecognizer');
|
||
let count = 0;
|
||
|
||
for (let i = 0; i < n; i++) {
|
||
const x = await this.recognize();
|
||
|
||
if (x > 0) count++;
|
||
if (i % 50 === 0) {
|
||
console.log('%f\%', (i / n) * 100);
|
||
}
|
||
}
|
||
|
||
console.log('successful: %f\%', (count / n) * 100);
|
||
console.timeEnd('PuzzleRecognizer');
|
||
}
|
||
|
||
static jsonp(api, data = {}) {
|
||
return new Promise((resolve, reject) => {
|
||
const fnId = `jsonp_${String(Math.random()).replace('.', '')}`;
|
||
const extraData = { callback: fnId };
|
||
const query = new URLSearchParams({ ...DATA, ...extraData, ...data }).toString();
|
||
const url = `http://${SERVER}${api}?${query}`;
|
||
const headers = {
|
||
'Accept': '*/*',
|
||
'Accept-Encoding': 'gzip,deflate,br',
|
||
'Accept-Language': 'zh-CN,en-US',
|
||
'Connection': 'keep-alive',
|
||
'Host': SERVER,
|
||
'Proxy-Connection': 'keep-alive',
|
||
'Referer': 'https://h5.m.jd.com/babelDiy/Zeus/2wuqXrZrhygTQzYA7VufBEpj4amH/index.html',
|
||
'User-Agent': UA,
|
||
};
|
||
const req = http.get(url, { headers }, (response) => {
|
||
let res = response;
|
||
if (res.headers['content-encoding'] === 'gzip') {
|
||
const unzipStream = new stream.PassThrough();
|
||
stream.pipeline(
|
||
response,
|
||
zlib.createGunzip(),
|
||
unzipStream,
|
||
reject,
|
||
);
|
||
res = unzipStream;
|
||
}
|
||
res.setEncoding('utf8');
|
||
|
||
let rawData = '';
|
||
|
||
res.on('data', (chunk) => rawData += chunk);
|
||
res.on('end', () => {
|
||
try {
|
||
const ctx = {
|
||
[fnId]: (data) => ctx.data = data,
|
||
data: {},
|
||
};
|
||
|
||
vm.createContext(ctx);
|
||
vm.runInContext(rawData, ctx);
|
||
|
||
// console.log(ctx.data);
|
||
res.resume();
|
||
resolve(ctx.data);
|
||
} catch (e) {
|
||
reject(e);
|
||
}
|
||
});
|
||
});
|
||
|
||
req.on('error', reject);
|
||
req.end();
|
||
});
|
||
}
|
||
}
|
||
|
||
function getCoordinate(c) {
|
||
function string10to64(d) {
|
||
let c = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ-~".split("")
|
||
, b = c.length
|
||
, e = +d
|
||
, a = [];
|
||
do {
|
||
mod = e % b;
|
||
e = (e - mod) / b;
|
||
a.unshift(c[mod])
|
||
} while (e);
|
||
return a.join("")
|
||
}
|
||
|
||
function prefixInteger(a, b) {
|
||
return (Array(b).join(0) + a).slice(-b)
|
||
}
|
||
|
||
function pretreatment(d, c, b) {
|
||
let e = string10to64(Math.abs(d));
|
||
let a = "";
|
||
if (!b) {
|
||
a += (d > 0 ? "1" : "0")
|
||
}
|
||
a += prefixInteger(e, c);
|
||
return a
|
||
}
|
||
|
||
let b = new Array();
|
||
for (let e = 0; e < c.length; e++) {
|
||
if (e == 0) {
|
||
b.push(pretreatment(c[e][0] < 262143 ? c[e][0] : 262143, 3, true));
|
||
b.push(pretreatment(c[e][1] < 16777215 ? c[e][1] : 16777215, 4, true));
|
||
b.push(pretreatment(c[e][2] < 4398046511103 ? c[e][2] : 4398046511103, 7, true))
|
||
} else {
|
||
let a = c[e][0] - c[e - 1][0];
|
||
let f = c[e][1] - c[e - 1][1];
|
||
let d = c[e][2] - c[e - 1][2];
|
||
b.push(pretreatment(a < 4095 ? a : 4095, 2, false));
|
||
b.push(pretreatment(f < 4095 ? f : 4095, 2, false));
|
||
b.push(pretreatment(d < 16777215 ? d : 16777215, 4, true))
|
||
}
|
||
}
|
||
return b.join("")
|
||
}
|
||
|
||
const HZ = 60;
|
||
|
||
class MousePosFaker {
|
||
constructor(puzzleX) {
|
||
this.x = parseInt(Math.random() * 20 + 20, 10);
|
||
this.y = parseInt(Math.random() * 80 + 80, 10);
|
||
this.t = Date.now();
|
||
this.pos = [[this.x, this.y, this.t]];
|
||
this.minDuration = parseInt(1000 / HZ, 10);
|
||
// this.puzzleX = puzzleX;
|
||
this.puzzleX = puzzleX + parseInt(Math.random() * 2 - 1, 10);
|
||
|
||
this.STEP = parseInt(Math.random() * 6 + 5, 10);
|
||
this.DURATION = parseInt(Math.random() * 7 + 14, 10) * 100;
|
||
// [9,1600] [10,1400]
|
||
this.STEP = 9;
|
||
// this.DURATION = 2000;
|
||
console.log(this.STEP, this.DURATION);
|
||
}
|
||
|
||
run() {
|
||
const perX = this.puzzleX / this.STEP;
|
||
const perDuration = this.DURATION / this.STEP;
|
||
const firstPos = [this.x - parseInt(Math.random() * 6, 10), this.y + parseInt(Math.random() * 11, 10), this.t];
|
||
|
||
this.pos.unshift(firstPos);
|
||
this.stepPos(perX, perDuration);
|
||
this.fixPos();
|
||
|
||
const reactTime = parseInt(60 + Math.random() * 100, 10);
|
||
const lastIdx = this.pos.length - 1;
|
||
const lastPos = [this.pos[lastIdx][0], this.pos[lastIdx][1], this.pos[lastIdx][2] + reactTime];
|
||
|
||
this.pos.push(lastPos);
|
||
return this.pos;
|
||
}
|
||
|
||
stepPos(x, duration) {
|
||
let n = 0;
|
||
const sqrt2 = Math.sqrt(2);
|
||
for (let i = 1; i <= this.STEP; i++) {
|
||
n += 1 / i;
|
||
}
|
||
for (let i = 0; i < this.STEP; i++) {
|
||
x = this.puzzleX / (n * (i + 1));
|
||
const currX = parseInt((Math.random() * 30 - 15) + x, 10);
|
||
const currY = parseInt(Math.random() * 7 - 3, 10);
|
||
const currDuration = parseInt((Math.random() * 0.4 + 0.8) * duration, 10);
|
||
|
||
this.moveToAndCollect({
|
||
x: currX,
|
||
y: currY,
|
||
duration: currDuration,
|
||
});
|
||
}
|
||
}
|
||
|
||
fixPos() {
|
||
const actualX = this.pos[this.pos.length - 1][0] - this.pos[1][0];
|
||
const deviation = this.puzzleX - actualX;
|
||
|
||
if (Math.abs(deviation) > 4) {
|
||
this.moveToAndCollect({
|
||
x: deviation,
|
||
y: parseInt(Math.random() * 8 - 3, 10),
|
||
duration: 250,
|
||
});
|
||
}
|
||
}
|
||
|
||
moveToAndCollect({ x, y, duration }) {
|
||
let movedX = 0;
|
||
let movedY = 0;
|
||
let movedT = 0;
|
||
const times = duration / this.minDuration;
|
||
let perX = x / times;
|
||
let perY = y / times;
|
||
let padDuration = 0;
|
||
|
||
if (Math.abs(perX) < 1) {
|
||
padDuration = duration / Math.abs(x) - this.minDuration;
|
||
perX = 1;
|
||
perY = y / Math.abs(x);
|
||
}
|
||
|
||
while (Math.abs(movedX) < Math.abs(x)) {
|
||
const rDuration = parseInt(padDuration + Math.random() * 16 - 4, 10);
|
||
|
||
movedX += perX + Math.random() * 2 - 1;
|
||
movedY += perY;
|
||
movedT += this.minDuration + rDuration;
|
||
|
||
const currX = parseInt(this.x + movedX, 10);
|
||
const currY = parseInt(this.y + movedY, 10);
|
||
const currT = this.t + movedT;
|
||
|
||
this.pos.push([currX, currY, currT]);
|
||
}
|
||
|
||
this.x += x;
|
||
this.y += y;
|
||
this.t += Math.max(duration, movedT);
|
||
}
|
||
}
|
||
|
||
function getRandomArrayElements(arr, count) {
|
||
var shuffled = arr.slice(0), i = arr.length, min = i - count, temp, index;
|
||
while (i-- > min) {
|
||
index = Math.floor((i + 1) * Math.random());
|
||
temp = shuffled[index];
|
||
shuffled[index] = shuffled[i];
|
||
shuffled[i] = temp;
|
||
}
|
||
return shuffled.slice(min);
|
||
}
|
||
|
||
|
||
let target = process.env.JD_JOY_REWARD_NAME ? parseInt(process.env.JD_JOY_REWARD_NAME) : 500;
|
||
let validate = '', validate1 = '', cookiesArr = [];
|
||
|
||
!(async () => {
|
||
await requireConfig();
|
||
/*
|
||
let validate_arr = fs.readFileSync('./validate.txt', 'utf-8')
|
||
if (validate_arr.indexOf('\n')) {
|
||
validate_arr = validate_arr.split('\n')
|
||
validate_arr.pop()
|
||
}
|
||
*/
|
||
if (!cookiesArr[0]) {
|
||
$.msg($.name, '【提示】请先获取京东账号一cookie\n直接使用NobyDa的京东签到获取', 'https://bean.m.jd.com/bean/signIndex.action', { "open-url": "https://bean.m.jd.com/bean/signIndex.action" });
|
||
return;
|
||
}
|
||
for (let i = 0; i < cookiesArr.length; i++) {
|
||
if (cookiesArr[i]) {
|
||
cookie = cookiesArr[i];
|
||
//if (i < validate_arr.length)
|
||
// validate = validate_arr[i]
|
||
//else {
|
||
// console.log('预存验证码不够用,退出!')
|
||
// break
|
||
//}
|
||
$.UserName = decodeURIComponent(cookie.match(/pt_pin=([^; ]+)(?=;?)/) && cookie.match(/pt_pin=([^; ]+)(?=;?)/)[1])
|
||
$.index = i + 1;
|
||
$.isLogin = true;
|
||
$.nickName = '';
|
||
await TotalBean();
|
||
console.log(`\n开始【京东账号${$.index}】${$.nickName || $.UserName}\n`);
|
||
if (!$.isLogin) {
|
||
$.msg($.name, `【提示】cookie已失效`, `京东账号${$.index} ${$.nickName || $.UserName}\n请重新登录获取\nhttps://bean.m.jd.com/bean/signIndex.action`, { "open-url": "https://bean.m.jd.com/bean/signIndex.action" });
|
||
|
||
if ($.isNode()) {
|
||
await notify.sendNotify(`${$.name}cookie已失效 - ${$.UserName}`, `京东账号${$.index} ${$.UserName}\n请重新登录获取cookie`);
|
||
}
|
||
continue
|
||
}
|
||
message = '';
|
||
subTitle = '';
|
||
|
||
let tasks = await init();
|
||
let hold = new Date().getHours();
|
||
let mold = new Date().getMinutes();
|
||
let sold = new Date().getSeconds();
|
||
let h = new Date().getHours();
|
||
let m = new Date().getMinutes();
|
||
let s = new Date().getSeconds();
|
||
let ms = new Date().getMilliseconds()
|
||
let config = '', timezone = 0, startflag = 0, keyflag = 0;
|
||
|
||
if (h >= 0 && h < 8) {
|
||
config = tasks.data['beanConfigs0']
|
||
}
|
||
if (h >= 8 && h < 16) {
|
||
config = tasks.data['beanConfigs8']
|
||
}
|
||
if (h >= 16 && h < 24) {
|
||
config = tasks.data['beanConfigs16']
|
||
}
|
||
for (let bean of config) {
|
||
console.log(`${bean.giftName}剩余${bean.leftStock}`)
|
||
if (bean.leftStock === 0 && bean.giftValue === target) {
|
||
console.log(`当前${bean.giftName}库存已被抢完`)
|
||
continue
|
||
}
|
||
else {
|
||
validate1 = await new JDJRValidator().run();
|
||
validate = validate1.validate;
|
||
console.log(`开始抢购${bean.giftName}`)
|
||
await exchange(bean.id)
|
||
}
|
||
}
|
||
console.log(`开始等待下一场兑换`)
|
||
|
||
while (startflag === 0) {
|
||
h = new Date().getHours();
|
||
m = new Date().getMinutes();
|
||
s = new Date().getSeconds();
|
||
ms = new Date().getMilliseconds()
|
||
if (h % 8 === 0) {
|
||
startflag = 1;
|
||
continue
|
||
}
|
||
await $.wait(500)
|
||
if (ms < 500) {
|
||
console.log(`当前${h}:${m}:${s}不在兑换时间,还需等待${7 - (h % 8)}小时${60 - m}分钟${60 - s}秒`)
|
||
}
|
||
if (7 - (h % 8) === 0 && m - mold < 3 && keyflag === 0) {
|
||
console.log(`准备密钥`)
|
||
validate1 = await new JDJRValidator().run();
|
||
validate = validate1.validate;
|
||
keyflag = 1;
|
||
}
|
||
else {
|
||
console.log(`不满足${60 - h}=1 ${m - mold}>3`)
|
||
}
|
||
}
|
||
|
||
if (h >= 0 && h < 8) {
|
||
config = tasks.data['beanConfigs0']
|
||
}
|
||
if (h >= 8 && h < 16) {
|
||
config = tasks.data['beanConfigs8']
|
||
}
|
||
if (h >= 16 && h < 24) {
|
||
config = tasks.data['beanConfigs16']
|
||
}
|
||
for (let i = 0; i < 3; i++) {
|
||
for (let bean of config) {
|
||
console.log(`${bean.giftName}剩余${bean.leftStock}`)
|
||
if (bean.leftStock === 0) {
|
||
console.log(`${bean.giftName}库存已被抢完`)
|
||
continue
|
||
}
|
||
if (bean.giftValue === target) {
|
||
await exchange(bean.id)
|
||
}
|
||
}
|
||
await $.wait(300)
|
||
}
|
||
}
|
||
}
|
||
})()
|
||
|
||
function init() {
|
||
return new Promise(resolve => {
|
||
$.get({
|
||
url: `https://jdjoy.jd.com/common/gift/getBeanConfigs?reqSource=h5&invokeKey=NRp8OPxZMFXmGkaE&validate=${validate}`,
|
||
headers: {
|
||
'Host': 'jdjoy.jd.com',
|
||
'accept': '*/*',
|
||
'content-type': 'application/json',
|
||
'origin': 'https://h5.m.jd.com',
|
||
"User-Agent": $.isNode() ? (process.env.JD_USER_AGENT ? process.env.JD_USER_AGENT : (require('./USER_AGENTS').USER_AGENT)) : ($.getdata('JDUA') ? $.getdata('JDUA') : "jdapp;iPhone;9.4.4;14.3;network/4g;Mozilla/5.0 (iPhone; CPU iPhone OS 14_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148;supportJDSHWK/1"),
|
||
'referer': 'https://jdjoy.jd.com/',
|
||
'accept-language': 'zh-cn',
|
||
'cookie': cookie
|
||
}
|
||
}, (err, resp, data) => {
|
||
try {
|
||
data = $.toObj(data)
|
||
resolve(data);
|
||
} catch (e) {
|
||
$.logErr(e);
|
||
} finally {
|
||
resolve();
|
||
}
|
||
})
|
||
})
|
||
}
|
||
|
||
function exchange(beanId) {
|
||
console.log('开始兑换')
|
||
return new Promise(async resolve => {
|
||
while (1) {
|
||
if (new Date().getSeconds() < 30) {
|
||
break
|
||
} else {
|
||
await $.wait(100)
|
||
}
|
||
}
|
||
$.post({
|
||
url: `https://jdjoy.jd.com/common/gift/new/exchange?reqSource=h5&invokeKey=NRp8OPxZMFXmGkaE&validate=${validate}`,
|
||
headers: {
|
||
"Host": "jdjoy.jd.com",
|
||
"Accept-Language": "zh-cn",
|
||
"Content-Type": "application/json",
|
||
"Origin": "https://jdjoy.jd.com",
|
||
"User-Agent": $.isNode() ? (process.env.JD_USER_AGENT ? process.env.JD_USER_AGENT : (require('./USER_AGENTS').USER_AGENT)) : ($.getdata('JDUA') ? $.getdata('JDUA') : "jdapp;iPhone;9.4.4;14.3;network/4g;Mozilla/5.0 (iPhone; CPU iPhone OS 14_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148;supportJDSHWK/1"),
|
||
"Referer": "https://jdjoy.jd.com/pet/index",
|
||
"Cookie": cookie
|
||
},
|
||
body: JSON.stringify({ "buyParam": { "orderSource": 'pet', "saleInfoId": beanId }, "deviceInfo": {} })
|
||
}, (err, resp, data) => {
|
||
try {
|
||
data = $.toObj(data)
|
||
console.log(data)
|
||
resolve(data);
|
||
} catch (e) {
|
||
$.logErr(e);
|
||
} finally {
|
||
resolve();
|
||
}
|
||
})
|
||
})
|
||
}
|
||
|
||
function requireConfig() {
|
||
return new Promise(resolve => {
|
||
notify = $.isNode() ? require('./sendNotify') : '';
|
||
const jdCookieNode = $.isNode() ? require('./jdCookie.js') : '';
|
||
if ($.isNode()) {
|
||
Object.keys(jdCookieNode).forEach((item) => {
|
||
if (jdCookieNode[item]) {
|
||
cookiesArr.push(jdCookieNode[item])
|
||
}
|
||
})
|
||
if (process.env.JD_DEBUG && process.env.JD_DEBUG === 'false') console.log = () => {
|
||
};
|
||
} else {
|
||
cookiesArr = [$.getdata('CookieJD'), $.getdata('CookieJD2'), ...jsonParse($.getdata('CookiesJD') || "[]").map(item => item.cookie)].filter(item => !!item);
|
||
}
|
||
console.log(`共${cookiesArr.length}个京东账号\n`)
|
||
resolve()
|
||
})
|
||
}
|
||
|
||
function TotalBean() {
|
||
return new Promise(async resolve => {
|
||
const options = {
|
||
"url": `https://wq.jd.com/user/info/QueryJDUserInfo?sceneval=2`,
|
||
"headers": {
|
||
"Accept": "application/json,text/plain, */*",
|
||
"Content-Type": "application/x-www-form-urlencoded",
|
||
"Accept-Encoding": "gzip, deflate, br",
|
||
"Accept-Language": "zh-cn",
|
||
"Connection": "keep-alive",
|
||
"Cookie": cookie,
|
||
"Referer": "https://wqs.jd.com/my/jingdou/my.shtml?sceneval=2",
|
||
"User-Agent": $.isNode() ? (process.env.JD_USER_AGENT ? process.env.JD_USER_AGENT : (require('./USER_AGENTS').USER_AGENT)) : ($.getdata('JDUA') ? $.getdata('JDUA') : "jdapp;iPhone;9.4.4;14.3;network/4g;Mozilla/5.0 (iPhone; CPU iPhone OS 14_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148;supportJDSHWK/1")
|
||
}
|
||
}
|
||
$.post(options, (err, resp, data) => {
|
||
try {
|
||
if (err) {
|
||
console.log(`${JSON.stringify(err)}`)
|
||
console.log(`${$.name} API请求失败,请检查网路重试`)
|
||
} else {
|
||
if (data) {
|
||
data = JSON.parse(data);
|
||
if (data['retcode'] === 13) {
|
||
$.isLogin = false; //cookie过期
|
||
return
|
||
}
|
||
if (data['retcode'] === 0) {
|
||
$.nickName = (data['base'] && data['base'].nickname) || $.UserName;
|
||
} else {
|
||
$.nickName = $.UserName
|
||
}
|
||
} else {
|
||
console.log(`京东服务器返回空数据`)
|
||
}
|
||
}
|
||
} catch (e) {
|
||
$.logErr(e, resp)
|
||
} finally {
|
||
resolve();
|
||
}
|
||
})
|
||
})
|
||
}
|
||
|
||
function jsonParse(str) {
|
||
if (typeof str == "string") {
|
||
try {
|
||
return JSON.parse(str);
|
||
} catch (e) {
|
||
console.log(e);
|
||
$.msg($.name, '', '请勿随意在BoxJs输入框修改内容\n建议通过脚本去获取cookie')
|
||
return [];
|
||
}
|
||
}
|
||
}
|
||
|
||
function Env(t, e) { "undefined" != typeof process && JSON.stringify(process.env).indexOf("GITHUB") > -1 && process.exit(0); class s { constructor(t) { this.env = t } send(t, e = "GET") { t = "string" == typeof t ? { url: t } : t; let s = this.get; return "POST" === e && (s = this.post), new Promise((e, i) => { s.call(this, t, (t, s, r) => { t ? i(t) : e(s) }) }) } get(t) { return this.send.call(this.env, t) } post(t) { return this.send.call(this.env, t, "POST") } } return new class { constructor(t, e) { this.name = t, this.http = new s(this), this.data = null, this.dataFile = "box.dat", this.logs = [], this.isMute = !1, this.isNeedRewrite = !1, this.logSeparator = "\n", this.startTime = (new Date).getTime(), Object.assign(this, e), this.log("", `🔔${this.name}, 开始!`) } 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 } toObj(t, e = null) { try { return JSON.parse(t) } catch { return e } } toStr(t, e = null) { try { return JSON.stringify(t) } catch { return e } } getjson(t, e) { let s = e; const i = this.getdata(t); if (i) try { s = JSON.parse(this.getdata(t)) } catch { } return s } setjson(t, e) { try { return this.setdata(JSON.stringify(t), e) } catch { return !1 } } getScript(t) { return new Promise(e => { this.get({ url: t }, (t, s, i) => e(i)) }) } runScript(t, e) { return new Promise(s => { let i = this.getdata("@chavy_boxjs_userCfgs.httpapi"); i = i ? i.replace(/\n/g, "").trim() : i; let r = this.getdata("@chavy_boxjs_userCfgs.httpapi_timeout"); r = r ? 1 * r : 20, r = e && e.timeout ? e.timeout : r; const [o, h] = i.split("@"), n = { url: `http://${h}/v1/scripting/evaluate`, body: { script_text: t, mock_type: "cron", timeout: r }, headers: { "X-Key": o, Accept: "*/*" } }; this.post(n, (t, e, i) => s(i)) }).catch(t => this.logErr(t)) } 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), e = this.path.resolve(process.cwd(), this.dataFile), s = this.fs.existsSync(t), i = !s && this.fs.existsSync(e); if (!s && !i) return {}; { const i = s ? t : e; 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), e = this.path.resolve(process.cwd(), this.dataFile), s = this.fs.existsSync(t), i = !s && this.fs.existsSync(e), r = JSON.stringify(this.data); s ? this.fs.writeFileSync(t, r) : i ? this.fs.writeFileSync(e, r) : this.fs.writeFileSync(t, r) } } lodash_get(t, e, s) { const i = e.replace(/\[(\d+)\]/g, ".$1").split("."); let r = t; for (const t of i) if (r = Object(r)[t], void 0 === r) return s; return r } lodash_set(t, e, s) { return Object(t) !== t ? t : (Array.isArray(e) || (e = e.toString().match(/[^.[\]]+/g) || []), e.slice(0, -1).reduce((t, s, i) => Object(t[s]) === t[s] ? t[s] : t[s] = Math.abs(e[i + 1]) >> 0 == +e[i + 1] ? [] : {}, t)[e[e.length - 1]] = s, t) } 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 } 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, t && (t.headers = t.headers ? t.headers : {}, void 0 === t.headers.Cookie && void 0 === t.cookieJar && (t.cookieJar = this.ckjar)) } get(t, e = (() => { })) { t.headers && (delete t.headers["Content-Type"], delete t.headers["Content-Length"]), this.isSurge() || this.isLoon() ? (this.isSurge() && this.isNeedRewrite && (t.headers = t.headers || {}, Object.assign(t.headers, { "X-Surge-Skip-Scripting": !1 })), $httpClient.get(t, (t, s, i) => { !t && s && (s.body = i, s.statusCode = s.status), e(t, s, i) })) : this.isQuanX() ? (this.isNeedRewrite && (t.opts = t.opts || {}, Object.assign(t.opts, { hints: !1 })), $task.fetch(t).then(t => { const { statusCode: s, statusCode: i, headers: r, body: o } = t; e(null, { status: s, statusCode: i, headers: r, body: o }, o) }, t => e(t))) : this.isNode() && (this.initGotEnv(t), this.got(t).on("redirect", (t, e) => { try { if (t.headers["set-cookie"]) { const s = t.headers["set-cookie"].map(this.cktough.Cookie.parse).toString(); s && this.ckjar.setCookieSync(s, null), e.cookieJar = this.ckjar } } catch (t) { this.logErr(t) } }).then(t => { const { statusCode: s, statusCode: i, headers: r, body: o } = t; e(null, { status: s, statusCode: i, headers: r, body: o }, o) }, t => { const { message: s, response: i } = t; e(s, i, i && i.body) })) } post(t, e = (() => { })) { if (t.body && t.headers && !t.headers["Content-Type"] && (t.headers["Content-Type"] = "application/x-www-form-urlencoded"), t.headers && delete t.headers["Content-Length"], this.isSurge() || this.isLoon()) this.isSurge() && this.isNeedRewrite && (t.headers = t.headers || {}, Object.assign(t.headers, { "X-Surge-Skip-Scripting": !1 })), $httpClient.post(t, (t, s, i) => { !t && s && (s.body = i, s.statusCode = s.status), e(t, s, i) }); else if (this.isQuanX()) t.method = "POST", this.isNeedRewrite && (t.opts = t.opts || {}, Object.assign(t.opts, { hints: !1 })), $task.fetch(t).then(t => { const { statusCode: s, statusCode: i, headers: r, body: o } = t; e(null, { status: s, statusCode: i, headers: r, body: o }, o) }, t => e(t)); else if (this.isNode()) { this.initGotEnv(t); const { url: s, ...i } = t; this.got.post(s, i).then(t => { const { statusCode: s, statusCode: i, headers: r, body: o } = t; e(null, { status: s, statusCode: i, headers: r, body: o }, o) }, t => { const { message: s, response: i } = t; e(s, i, i && i.body) }) } } time(t, e = null) { const s = e ? new Date(e) : new Date; let i = { "M+": s.getMonth() + 1, "d+": s.getDate(), "H+": s.getHours(), "m+": s.getMinutes(), "s+": s.getSeconds(), "q+": Math.floor((s.getMonth() + 3) / 3), S: s.getMilliseconds() }; /(y+)/.test(t) && (t = t.replace(RegExp.$1, (s.getFullYear() + "").substr(4 - RegExp.$1.length))); for (let e in i) new RegExp("(" + e + ")").test(t) && (t = t.replace(RegExp.$1, 1 == RegExp.$1.length ? i[e] : ("00" + i[e]).substr(("" + i[e]).length))); return t } 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 } } } }; if (this.isMute || (this.isSurge() || this.isLoon() ? $notification.post(e, s, i, o(r)) : this.isQuanX() && $notify(e, s, i, o(r))), !this.isMuteLog) { let t = ["", "==============📣系统通知📣=============="]; t.push(e), s && t.push(s), i && t.push(i), console.log(t.join("\n")), this.logs = this.logs.concat(t) } } log(...t) { t.length > 0 && (this.logs = [...this.logs, ...t]), console.log(t.join(this.logSeparator)) } logErr(t, e) { const s = !this.isSurge() && !this.isQuanX() && !this.isLoon(); s ? this.log("", `❗️${this.name}, 错误!`, t.stack) : this.log("", `❗️${this.name}, 错误!`, t) } wait(t) { return new Promise(e => setTimeout(e, t)) } done(t = {}) { const e = (new Date).getTime(), s = (e - this.startTime) / 1e3; this.log("", `🔔${this.name}, 结束! 🕛 ${s} 秒`), this.log(), (this.isSurge() || this.isQuanX() || this.isLoon()) && $done(t) } }(t, e) } |