Script/Bilibili-DailyBonus/ExchangePoints.js

240 lines
10 KiB
JavaScript

/*
哔哩哔哩漫画, 积分商城自动抢购脚本
脚本兼容: Surge, QuantumultX, Loon
*************************
【 抢购脚本注意事项 】:
*************************
该脚本需要使用签到脚本获取Cookie后方可使用.
默认兑换积分商城中的"积分兑换", 兑换数量为用户积分可兑换的最大值 (可于BoxJs内修改)
默认执行时间为中午12:00:10、12:00:20、12:00:30
BoxJs订阅地址: https://raw.githubusercontent.com/NobyDa/Script/master/NobyDa_BoxJs.json
*************************
【 Surge & Loon 脚本配置 】:
*************************
[Script]
cron "10,20,30 0 12 * * *" script-path=https://raw.githubusercontent.com/NobyDa/Script/master/Bilibili-DailyBonus/ExchangePoints.js, wake-system=1, timeout=60
*************************
【 QX 1.0.10+ 脚本配置 】 :
*************************
[task_local]
10,20,30 0 12 * * * https://raw.githubusercontent.com/NobyDa/Script/master/Bilibili-DailyBonus/ExchangePoints.js, tag=哔哩哔哩漫画抢券, enabled=true
*/
// 新建一个实例对象, 把兼容函数定义到$中, 以便统一调用
let $ = new nobyda();
// 读取兑换商品名, 默认兑换积分商城中的"积分兑换"; 该接口为BoxJs预留, 以便修改
let productName = $.read('BM_ProductName') || '积分兑换';
// 读取兑换数量, 默认兑换最大值; 该接口为BoxJs预留, 以便修改
let productNum = $.read('BM_ProductNum');
// 读取循环抢购次数, 默认100次; 该接口为BoxJs预留, 以便修改
let exchangeNum = $.read('BM_ExchangeNum') || '100';
// 读取哔哩哔哩漫画签到脚本所使用的Cookie
let cookie = $.read('CookieBM');
// 预留的空对象, 便于函数之间读取数据
let user = {};
(async function() { // 立即运行的匿名异步函数
// 使用await关键字声明, 表示以同步方式执行异步函数, 可以简单理解为顺序执行
await Promise.all([ //该方法用于将多个实例包装成一个新的实例, 可以简单理解为同时调用函数, 以进一步提高执行速度
GetUserPoint(), //查询积分函数
ListProduct() //查询商品函数
]);
await ExchangeProduct(); //上面的查询都完成后, 则执行抢购
$.done(); //抢购完成后调用Surge、QX内部特有的函数, 用于退出脚本执行
})();
function GetUserPoint() {
const pointUrl = { //查询积分接口
url: 'https://manga.bilibili.com/twirp/pointshop.v1.Pointshop/GetUserPoint',
headers: { //请求头
'Cookie': cookie //用户鉴权Cookie
}
}
return new Promise((resolve) => { //主函数返回Promise实例对象, 以便后续调用时可以实现顺序执行异步函数
$.post(pointUrl, (error, resp, data) => { //使用post请求查询, 再使用回调函数处理返回的结果
try { //使用try方法捕获可能出现的代码异常
if (error) {
throw new Error(error); //如果请求失败, 例如无法联网, 则抛出一个异常
} else {
const body = JSON.parse(data); //解析响应体json并转化为对象
if (body.code == 0 && body.data) { //如果响应体为预期格式
user.point = parseInt(body.data.point); //把查询的积分赋值到全局变量user中
console.log(`\n当前积分: ${body.data.point}`); //打印日志
} else { //否则抛出一个异常
throw new Error(body.msg || data);
}
}
} catch (e) { //接住try代码块中抛出的异常, 并打印日志
console.log(`\n查询积分: 失败\n出现错误: ${e.message}`);
} finally { //finally语句在try和catch之后无论有无异常都会执行
resolve(); //异步操作成功时调用, 将Promise对象的状态标记为"成功", 表示已完成查询积分
}
})
})
}
function ListProduct() {
const listUrl = { //查询商品接口
url: 'https://manga.bilibili.com/twirp/pointshop.v1.Pointshop/ListProduct',
headers: {}
}
return new Promise((resolve) => { //主函数返回Promise实例对象, 以便后续调用时可以实现顺序执行异步函数
$.post(listUrl, (error, resp, data) => { //使用post请求查询, 再使用回调函数处理返回的结果
try { //使用try方法捕获可能出现的代码异常
if (error) {
throw new Error(error); //如果请求失败, 例如无法联网, 则抛出一个异常
} else {
const body = JSON.parse(data); //解析响应体json并转化为对象
if (body.code == 0 && body.data.length >= 1) { //如果接口正常返回商品信息
// 按全局变量所填写的商品名进行过滤, 并把商品信息赋值到全局变量user中
user.list = body.data.filter(t => t.title == productName).pop();
if (!user.list) {
throw new Error('请检查商品名'); //如果填错商品名则抛出一个异常
} else { //否则打印日志
console.log(`\n查询商品: ${productName}\n商品库存: ${user.list.remain_amount}`)
}
} else { //否则抛出一个异常
throw new Error('无商品列表');
}
}
} catch (e) { //接住try代码块中抛出的异常并打印日志
console.log(`\n查询商品: ${productName}\n出现错误: ${e.message}`);
} finally { //finally语句在try和catch之后无论有无异常都会执行
resolve(); //异步操作成功时调用, 将Promise对象的状态标记为"成功", 表示已完成查询商品
}
})
})
}
function ExchangeProduct() {
return new Promise(async (resolve) => { //主函数返回Promise实例对象, 以便后续调用时可以实现顺序执行异步函数, 该实例函数带有async关键字, 表示里面有异步操作, 例如可使用await得到异步结果
if (user.list && user.list.remain_amount && user.point >= 100) { //如果商品有库存并且用户积分大于100则进行抢购
//兑换商品数量(用户积分 除与 商品单价得到兑换数量), 并转成整数; 默认兑换最大数量
const num = parseInt(productNum || (user.point / user.list.real_cost));
const exchangeUrl = {
url: 'https://manga.bilibili.com/twirp/pointshop.v1.Pointshop/Exchange', //兑换商品接口
headers: { //请求头
'Content-Type': 'application/json', //声明请求体数据格式
'Cookie': cookie //用户鉴权Cookie
},
body: JSON.stringify({ //请求体转成字符串类型
product_id: user.list.id, //兑换的商品id
product_num: num, //兑换的商品数量
point: num * user.list.real_cost //消耗的积分总数 (兑换数量乘单价得到积分总数)
})
};
for (let i = 0; i < parseInt(exchangeNum); i++) { //根据全局变量定义的次数, 暴力循环抢购
// 循环内调用另一个抢购函数, 并传入请求、第几次循环、兑换数量等参数,
// 使用await关键字声明, 表示需要等待每一次的执行结果
const run = await startExchange(exchangeUrl, i, num);
if (run) {
break; //如果函数返回布尔值true, 则跳出循环, 脚本结束
}
}
} else { //商品无库存或用户积分小于100等情况, 则不执行抢购, 脚本结束
console.log(`\n抢购终止: 不具备兑换条件`); //打印日志
}
resolve(); //将主函数的Promise对象状态标记为"成功", 表示已完成抢购任务
})
}
function startExchange(url, item, amount) {
return new Promise((resolve) => { //主函数返回Promise实例对象, 以便后续调用时可以实现顺序执行异步函数
$.post(url, (error, resp, data) => { //使用post请求查询, 再使用回调函数处理返回的结果
try { //使用try方法捕获可能出现的代码异常
if (error) {
throw new Error(error); //如果请求失败, 例如无法联网, 则抛出一个异常
} else {
const body = JSON.parse(data); //解析响应体json并转化为对象
if (body.code == 0) { //如果抢购成功, 则输出日志和通知
console.log(`\n抢购成功: 第${item+1}\n抢购数量: ${amount}\n消耗积分: ${amount * user.list.real_cost}`);
$.notify('哔哩哔哩漫画抢券', '', `"${productName}"抢购成功, 数量: ${amount}, 消耗积分: ${amount * user.list.real_cost}`);
resolve(true); //将Promise对象的状态标记为"成功", 然后返回一个布尔值true用于跳出循环
} else {
throw new Error(body.msg || '未知'); //抢购失败则抛出异常
}
}
} catch (e) { //接住try代码块中抛出的异常并打印日志
console.log(`\n抢购失败: 第${item+1}\n失败原因: ${e.message}`);
resolve(); //将Promise对象的状态标记为"成功", 但不返回任何值, 表示继续循环抢购
}
})
})
}
function nobyda() {
const isSurge = typeof $httpClient != "undefined";
const isQuanX = typeof $task != "undefined";
const isNode = typeof require == "function";
const node = (() => {
if (isNode) {
const request = require('request');
return {
request
}
} else {
return null;
}
})()
const adapterStatus = (response) => {
if (response) {
if (response.status) {
response["statusCode"] = response.status
} else if (response.statusCode) {
response["status"] = response.statusCode
}
}
return response
}
this.read = (key) => {
if (isQuanX) return $prefs.valueForKey(key)
if (isSurge) return $persistentStore.read(key)
}
this.notify = (title, subtitle, message) => {
if (isQuanX) $notify(title, subtitle, message)
if (isSurge) $notification.post(title, subtitle, message)
if (isNode) console.log(`${title}\n${subtitle}\n${message}`)
}
this.post = (options, callback) => {
options.headers['User-Agent'] = 'User-Agent: Mozilla/5.0 (iPhone; CPU iPhone OS 13_6_1 like Mac OS X) AppleWebKit/609.3.5.0.2 (KHTML, like Gecko) Mobile/17G80 BiliApp/822 mobi_app/ios_comic channel/AppStore BiliComic/822'
if (isQuanX) {
if (typeof options == "string") options = {
url: options
}
options["method"] = "POST"
$task.fetch(options).then(response => {
callback(null, adapterStatus(response), response.body)
}, reason => callback(reason.error, null, null))
}
if (isSurge) {
options.headers['X-Surge-Skip-Scripting'] = false
$httpClient.post(options, (error, response, body) => {
callback(error, adapterStatus(response), body)
})
}
if (isNode) {
node.request.post(options, (error, response, body) => {
callback(error, adapterStatus(response), body)
})
}
}
this.done = () => {
if (isQuanX || isSurge) {
$done()
}
}
};