/*
 * 2022-07-20 修复获取试用列表风控问题;  
 * 2022-08-06 修复申请试用时风控,感谢TG大佬的公益接口;
 * By https://github.com/6dylan6/jdpro/
 * 基于X1a0He版本修改
 * @Address: https://github.com/X1a0He/jd_scripts_fixed/blob/main/jd_try_xh.js

如需运行请自行添加环境变量:JD_TRY="true" 即可运行
脚本是否耗时只看args_xh.maxLength的大小(申请数量),默认50个,申请100个差不多15分钟
上一作者说每天申请上限300个(自测,没有申请过上限),关注店铺上限500个
关注店铺满了就无法继续申请,可用批量取关店铺取消关注

部分环境变量说明,详细请参考58行往下:
export JD_TRY="true"是否允许,默认false
export JD_TRY_PASSZC="false" #不过滤种草官类试用,默认true过滤
export JD_TRY_MAXLENGTH="50" #商品数组的最大长度,默认50个
export JD_TRY_PRICE="XX"#商品原价格,大于XX才申请,默认20
export JD_TRY_APPLYINTERVAL="6000" #商品试用之间和获取商品之间的间隔ms
export JD_TRY_APPLYNUMFILTER="10000" #过滤大于设定值的已申请人数
export JD_TRY_MINSUPPLYNUM="1" #最小提供数量
export JD_TRY_SENDNUM="10" #每隔多少账号发送一次通知,默认为4
export JD_TRY_UNIFIED="false" 默认采用不同试用组

定时自定义,能用多久随缘了!!!
 */
const $ = new Env('京东试用')
const URL = 'https://api.m.jd.com/client.action'
let trialActivityIdList = []
let trialActivityTitleList = []
let notifyMsg = ''
let size = 1;
$.isPush = true;
$.isLimit = false;
$.isForbidden = false;
$.wrong = false;
$.giveupNum = 0;
$.successNum = 0;
$.completeNum = 0;
$.getNum = 0;
$.try = true;
$.sentNum = 0;
$.cookiesArr = []
//默认的过滤关键词
$.innerKeyWords =
    [
        "幼儿园", "教程", "英语", "辅导", "培训",
        "孩子", "小学", "成人用品", "套套", "情趣",
        "自慰", "阳具", "飞机杯", "男士用品", "女士用品",
        "内衣", "高潮", "避孕", "乳腺", "肛塞", "肛门",
        "宝宝", "芭比", "娃娃", "男用",
        "女用", "神油", "足力健", "老年", "老人",
        "宠物", "饲料", "丝袜", "黑丝", "磨脚",
        "脚皮", "除臭", "性感", "内裤", "跳蛋",
        "安全套", "龟头", "阴道", "阴部", "手机卡", "电话卡", "流量卡",
        "习题","试卷",
    ]
//下面很重要,遇到问题请把下面注释看一遍再来问
let args_xh = {
    /*
     * 控制是否输出当前环境变量设置,默认为false
     * 环境变量名称:XH_TRY_ENV
     */
    env: process.env.XH_TRY_ENV === 'true' || false,
    /*
     * 跳过某个指定账号,默认为全部账号清空
     * 填写规则:例如当前Cookie1为pt_key=key; pt_pin=pin1;则环境变量填写pin1即可,此时pin1的购物车将不会被清空
     * 若有更多,则按照pin1@pin2@pin3进行填写
     * 环境变量名称:XH_TRY_EXCEPT
     */
    except: process.env.XH_TRY_EXCEPT && process.env.XH_TRY_EXCEPT.split('@') || [],
    //以上环境变量新增于2022.01.30
    /*
     * 每个Tab页要便遍历的申请页数,由于京东试用又改了,获取不到每一个Tab页的总页数了(显示null),所以特定增加一个环境变了以控制申请页数
     * 例如设置 JD_TRY_PRICE 为 30,假如现在正在遍历tab1,那tab1就会被遍历到30页,到31页就会跳到tab2,或下一个预设的tab页继续遍历到30页
     * 默认为20
     */
    totalPages: process.env.JD_TRY_TOTALPAGES * 1 || 20,
    /*
     * 由于每个账号每次获取的试用产品都不一样,所以为了保证每个账号都能试用到不同的商品,之前的脚本都不支持采用统一试用组的
     * 以下环境变量是用于指定是否采用统一试用组的
     * 例如当 JD_TRY_UNIFIED 为 true时,有3个账号,第一个账号跑脚本的时候,试用组是空的
     * 而当第一个账号跑完试用组后,第二个,第三个账号所采用的试用组默认采用的第一个账号的试用组
     * 优点:减少除第一个账号外的所有账号遍历,以减少每个账号的遍历时间
     * 缺点:A账号能申请的东西,B账号不一定有
     * 提示:想每个账号独立不同的试用产品的,请设置为false,想减少脚本运行时间的,请设置为true
     * 默认为false
     */
    unified: process.env.JD_TRY_UNIFIED === 'true' || false,
    //以上环境变量新增于2022.01.25
    /*
     * 商品原价,低于这个价格都不会试用,意思是
     * A商品原价49元,试用价1元,如果下面设置为50,那么A商品不会被加入到待提交的试用组
     * B商品原价99元,试用价0元,如果下面设置为50,那么B商品将会被加入到待提交的试用组
     * C商品原价99元,试用价1元,如果下面设置为50,那么C商品将会被加入到待提交的试用组
     * 默认为0
     * */
    jdPrice: process.env.JD_TRY_PRICE * 1 || 20,
    /*
     * 下面有一个function是可以获取tabId列表,名为try_tabList
     * 可设置环境变量:JD_TRY_TABID,用@进行分隔
     * tabId不定期会变,获取不到商品,自行获取并修改tabId
     * */
    tabId: process.env.JD_TRY_TABID && process.env.JD_TRY_TABID.split('@').map(Number) || [200, 201, 202, 203, 204, 205, 206, 207, 208, 209, 210, 211, 212],
    /*
     * 试用商品标题过滤,黑名单,当标题存在关键词时,则不加入试用组
     * 当白名单和黑名单共存时,黑名单会自动失效,优先匹配白名单,匹配完白名单后不会再匹配黑名单,望周知
     * 例如A商品的名称为『旺仔牛奶48瓶特价』,设置了匹配白名单,白名单关键词为『牛奶』,但黑名单关键词存在『旺仔』
     * 这时,A商品还是会被添加到待提交试用组,白名单优先于黑名单
     * 已内置对应的 成人类 幼儿类 宠物 老年人类关键词,请勿重复添加
     * 可设置环境变量:JD_TRY_TITLEFILTERS,关键词与关键词之间用@分隔
     * */
    titleFilters: process.env.JD_TRY_TITLEFILTERS && process.env.JD_TRY_TITLEFILTERS.split('@') || [],
    /*
     * 试用价格(中了要花多少钱),高于这个价格都不会试用,小于等于才会试用,意思就是
     * A商品原价49元,现在试用价1元,如果下面设置为10,那A商品将会被添加到待提交试用组,因为1 < 10
     * B商品原价49元,现在试用价2元,如果下面设置为1,那B商品将不会被添加到待提交试用组,因为2 > 1
     * C商品原价49元,现在试用价1元,如果下面设置为1,那C商品也会被添加到带提交试用组,因为1 = 1
     * 可设置环境变量:JD_TRY_TRIALPRICE,默认为0
     * */
    trialPrice: process.env.JD_TRY_TRIALPRICE * 1 || 0,
    /*
     * 最小提供数量,例如试用商品只提供2份试用资格,当前设置为1,则会进行申请
     * 若只提供5分试用资格,当前设置为10,则不会申请
     * 可设置环境变量:JD_TRY_MINSUPPLYNUM
     * */
    minSupplyNum: process.env.JD_TRY_MINSUPPLYNUM * 1 || 1,
    /*
     * 过滤大于设定值的已申请人数,例如下面设置的10000,A商品已经有10001人申请了,则A商品不会进行申请,会被跳过
     * 可设置环境变量:JD_TRY_APPLYNUMFILTER
     * */
    applyNumFilter: process.env.JD_TRY_APPLYNUMFILTER * 1 || 10000,
    /*
     * 商品试用之间和获取商品之间的间隔, 单位:毫秒(1秒=1000毫秒)
     * 可设置环境变量:JD_TRY_APPLYINTERVAL
     * 默认为6000-9000随机
     * */
    applyInterval: process.env.JD_TRY_APPLYINTERVAL * 1 || 6000,
    /*
     * 商品数组的最大长度,通俗来说就是即将申请的商品队列长度
     * 例如设置为20,当第一次获取后获得12件,过滤后剩下5件,将会进行第二次获取,过滤后加上第一次剩余件数
     * 例如是18件,将会进行第三次获取,直到过滤完毕后为20件才会停止,不建议设置太大
     * 可设置环境变量:JD_TRY_MAXLENGTH
     * */
    maxLength: process.env.JD_TRY_MAXLENGTH * 1 || 50,
    /*
     * 过滤种草官类试用,某些试用商品是专属官专属,考虑到部分账号不是种草官账号
     * 例如A商品是种草官专属试用商品,下面设置为true,而你又不是种草官账号,那A商品将不会被添加到待提交试用组
     * 例如B商品是种草官专属试用商品,下面设置为false,而你是种草官账号,那A商品将会被添加到待提交试用组
     * 例如B商品是种草官专属试用商品,下面设置为true,即使你是种草官账号,A商品也不会被添加到待提交试用组
     * 可设置环境变量:JD_TRY_PASSZC,默认为true
     * */
    passZhongCao: process.env.JD_TRY_PASSZC === 'false' || true,
    /*
     * 是否打印输出到日志,考虑到如果试用组长度过大,例如100以上,如果每个商品检测都打印一遍,日志长度会非常长
     * 打印的优点:清晰知道每个商品为什么会被过滤,哪个商品被添加到了待提交试用组
     * 打印的缺点:会使日志变得很长
     *
     * 不打印的优点:简短日志长度
     * 不打印的缺点:无法清晰知道每个商品为什么会被过滤,哪个商品被添加到了待提交试用组
     * 可设置环境变量:JD_TRY_PLOG,默认为true
     * */
    printLog: process.env.JD_TRY_PLOG === 'false' || true,
    /*
     * 白名单,是否打开,如果下面为true,那么黑名单会自动失效
     * 白名单和黑名单无法共存,白名单永远优先于黑名单
     * 可通过环境变量控制:JD_TRY_WHITELIST,默认为false
     * */
    whiteList: process.env.JD_TRY_WHITELIST === 'true' || false,
    /*
     * 白名单关键词,当标题存在关键词时,加入到试用组
     * 例如A商品的名字为『旺仔牛奶48瓶特价』,白名单其中一个关键词是『牛奶』,那么A将会直接被添加到待提交试用组,不再进行另外判断
     * 就算设置了黑名单也不会判断,希望这种写得那么清楚的脑瘫问题就别提issues了
     * 可通过环境变量控制:JD_TRY_WHITELIST,用@分隔
     * */
    whiteListKeywords: process.env.JD_TRY_WHITELISTKEYWORDS && process.env.JD_TRY_WHITELISTKEYWORDS.split('@') || [],
    /*
     * 每多少个账号发送一次通知,默认为4
     * 可通过环境变量控制 JD_TRY_SENDNUM
     * */
    sendNum: process.env.JD_TRY_SENDNUM * 1 || 4,
}
//上面很重要,遇到问题请把上面注释看一遍再来问
!(async() => {
    await $.wait(500)
    // 如果你要运行京东试用这个脚本,麻烦你把环境变量 JD_TRY 设置为 true
    if (process.env.JD_TRY && process.env.JD_TRY === 'true') {
        $.log('\n遇到问题请先看脚本内注释;解决不了在联系我https://t.me/dylan_jdpro\n');
        await requireConfig()
        if (!$.cookiesArr[0]) {
            $.msg($.name, '【提示】请先获取京东账号一cookie\n直接使用NobyDa的京东签到获取', 'https://bean.m.jd.com/', {
                "open-url": "https://bean.m.jd.com/"
            })
            return
        }
        args_xh.tabId = args_xh.tabId.sort(() => 0.5 - Math.random())
        for (let i = 0; i < $.cookiesArr.length; i++) {
            if ($.cookiesArr[i]) {
                $.cookie = $.cookiesArr[i];
                $.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`);
                $.except = false;
                if(args_xh.except.includes($.UserName)){
                    console.log(`跳过账号:${$.nickName || $.UserName}`)
                    $.except = true;
                    continue
                }
                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"
                    });
                    await $.notify.sendNotify(`${$.name}cookie已失效 - ${$.UserName}`, `京东账号${$.index} ${$.UserName}\n请重新登录获取cookie`);
                    continue
                }
                $.totalTry = 0
                $.totalSuccess = 0
                $.nowTabIdIndex = 0;
                $.nowPage = 1;
                $.nowItem = 1;
                $.retrynum = 0
                $.jda='__jda='+_jda('1xxxxxxxx.164xxxxxxxxxxxxxxxxxxx.164xxxxxxx.165xxxxxx.165xxxxxx.1xx')
                if (!args_xh.unified) {
                    trialActivityIdList = []
                    trialActivityTitleList = []
                }
                $.isLimit = false;
                // 获取tabList的,不知道有哪些的把这里的注释解开跑一遍就行了
                 //await try_tabList();
                // return;
                $.isForbidden = false
                $.wrong = false
                size = 1

                while(trialActivityIdList.length < args_xh.maxLength && $.retrynum < 3){
                    if($.nowTabIdIndex === args_xh.tabId.length){
                        console.log(`tabId组已遍历完毕,不在获取商品\n`);
                        break;
                    } else {
                        await try_feedsList(args_xh.tabId[$.nowTabIdIndex], $.nowPage)  //获取对应tabId的试用页面
                    }
                    if(trialActivityIdList.length < args_xh.maxLength){
                        console.log(`间隔等待中,请等待3秒 \n`)
                        await $.wait(3000);
                    }
                }
                if ($.isForbidden === false && $.isLimit === false) {
                    console.log(`稍后将执行试用申请,请等待 2 秒\n`)
                    await $.wait(2000);
                    for(let i = 0; i < trialActivityIdList.length && $.isLimit === false; i++){
                        if($.isLimit){
                            console.log("试用上限")
                            break
                        }
                        if($.isForbidden){console.log('403了,跳出');break}
                        await try_apply(trialActivityTitleList[i], trialActivityIdList[i])
                        //console.log(`间隔等待中,请等待 ${args_xh.applyInterval} ms\n`)
                        const waitTime = generateRandomInteger(args_xh.applyInterval, 9000);
                        console.log(`随机等待${waitTime}ms后继续`);
                        await $.wait(waitTime);
                    }
                    console.log("试用申请执行完毕...")
                    // await try_MyTrials(1, 1)    //申请中的商品
                    $.giveupNum = 0;
                    $.successNum = 0;
                    $.getNum = 0;
                    $.completeNum = 0;
                    await try_MyTrials(1, 2)    //申请成功的商品
                    // await try_MyTrials(1, 3)    //申请失败的商品
                    await showMsg()
                }
            }
            if($.isNode()){
                if($.index % args_xh.sendNum === 0){
                    $.sentNum++;
                    console.log(`正在进行第 ${$.sentNum} 次发送通知,发送数量:${args_xh.sendNum}`)
                    await $.notify.sendNotify(`${$.name}`, `${notifyMsg}`)
                    notifyMsg = "";
                }
            }
        }
        if($.isNode() && $.except === false){
            if(($.cookiesArr.length - ($.sentNum * args_xh.sendNum)) < args_xh.sendNum && notifyMsg.length != 0) {
                console.log(`正在进行最后一次发送通知,发送数量:${($.cookiesArr.length - ($.sentNum * args_xh.sendNum))}`)
                await $.notify.sendNotify(`${$.name}`, `${notifyMsg}`)
                notifyMsg = "";
            }
        }
    } else {
        console.log(`\n您未设置变量export JD_TRY="true"运行【京东试用】脚本, 结束运行!\n`)
    }
})().catch((e) => {
    console.error(`❗️ ${$.name} 运行错误!\n${e}`)
}).finally(() => $.done())

function requireConfig() {
    return new Promise(resolve => {
        $.notify = $.isNode() ? require('./sendNotify') : { sendNotify: async () => { } }
        //获取 Cookies
        $.cookiesArr = []
        if ($.isNode()) {
            //Node.js用户请在jdCookie.js处填写京东ck;
            const jdCookieNode = require('./jdCookie.js');
            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 {
            //IOS等用户直接用NobyDa的jd $.cookie
            $.cookiesArr = [$.getdata('CookieJD'), $.getdata('CookieJD2'), ...jsonParse($.getdata('CookiesJD') || "[]").map(item => item.cookie)].filter(item => !!item);
        }
        for(let keyWord of $.innerKeyWords) args_xh.titleFilters.push(keyWord)
        console.log(`共${$.cookiesArr.length}个京东账号\n`)
        if(args_xh.env){
            console.log('=========环境变量配置如下=========')
            console.log(`env: ${typeof args_xh.env}, ${args_xh.env}`)
            console.log(`except: ${typeof args_xh.except}, ${args_xh.except}`)
            console.log(`totalPages: ${typeof args_xh.totalPages}, ${args_xh.totalPages}`)
            console.log(`unified: ${typeof args_xh.unified}, ${args_xh.unified}`)
            console.log(`jdPrice: ${typeof args_xh.jdPrice}, ${args_xh.jdPrice}`)
            console.log(`tabId: ${typeof args_xh.tabId}, ${args_xh.tabId}`)
            console.log(`titleFilters: ${typeof args_xh.titleFilters}, ${args_xh.titleFilters}`)
            console.log(`trialPrice: ${typeof args_xh.trialPrice}, ${args_xh.trialPrice}`)
            console.log(`minSupplyNum: ${typeof args_xh.minSupplyNum}, ${args_xh.minSupplyNum}`)
            console.log(`applyNumFilter: ${typeof args_xh.applyNumFilter}, ${args_xh.applyNumFilter}`)
            console.log(`applyInterval: ${typeof args_xh.applyInterval}, ${args_xh.applyInterval}`)
            console.log(`maxLength: ${typeof args_xh.maxLength}, ${args_xh.maxLength}`)
            console.log(`passZhongCao: ${typeof args_xh.passZhongCao}, ${args_xh.passZhongCao}`)
            console.log(`printLog: ${typeof args_xh.printLog}, ${args_xh.printLog}`)
            console.log(`whiteList: ${typeof args_xh.whiteList}, ${args_xh.whiteList}`)
            console.log(`whiteListKeywords: ${typeof args_xh.whiteListKeywords}, ${args_xh.whiteListKeywords}`)
            console.log('===============================')
        }
        resolve()
    })
}

//获取tabList的,如果不知道tabList有哪些,跑一遍这个function就行了
function try_tabList() {
    return new Promise((resolve, reject) => {
        console.log(`获取tabList中...`)
        const body = JSON.stringify({
            "version": 2,
            "previewTime": ""
        });
        let option = taskurl_xh('newtry', 'try_tabList', body)
        $.post(option, (err, resp, data) => {
            try{
                if(err){
                    if(JSON.stringify(err) === `\"Response code 403 (Forbidden)\"`){
                        $.isForbidden = true
                        console.log('账号被京东服务器风控,不再请求该帐号')
                    } else {
                        console.log(JSON.stringify(err))
                        console.log(`${$.name} API请求失败,请检查网路重试`)
                    }
                } else {
                    data = JSON.parse(data)
                    if (data.success) {
                        for (let tabId of data.data.tabList) console.log(`${tabId.tabName} - ${tabId.tabId}`)
                    } else {
                        console.log("获取失败", data)
                    }
                }
            } catch (e) {
                reject(`⚠️ ${arguments.callee.name.toString()} API返回结果解析出错\n${e}\n${JSON.stringify(data)}`)
            } finally {
                resolve()
            }
        })
    })
}

//获取商品列表并且过滤 By X1a0He
function try_feedsList(tabId, page) {
    return new Promise((resolve, reject) => {
        const body = JSON.stringify({
            "tabId": `${tabId}`,
            "page": page,
            "version": 2,
            "source": "default",
            "client": "app",
            //"previewTime": ""
        });
        let option = taskurl_xh('newtry', 'try_feedsList', body)
        $.post(option, (err, resp, data) => {
            try{
                if(err){
                    if(JSON.stringify(err) === `\"Response code 403 (Forbidden)\"`){ 
                        console.log(`请求失败,第 ${$.retrynum+1} 次重试`)
                        $.retrynum++
                        if($.retrynum === 3) {$.isForbidden = true;$.log('多次尝试失败,换个时间再试!')}
                    } else {
                        console.log(JSON.stringify(err))
                        console.log(`${$.name} API请求失败,请检查网路重试`)
                    }
                } else {
                    data = JSON.parse(data)
                    let tempKeyword = ``;
                    if (data.data) {
                        $.nowPage === args_xh.totalPages ? $.nowPage = 1 : $.nowPage++;
                        console.log(`第 ${size++} 次获取试用商品成功,tabId:${args_xh.tabId[$.nowTabIdIndex]} 的 第 ${page}/${args_xh.totalPages} 页`)
                        console.log(`获取到商品 ${data.data.feedList.length} 条`)
                        for (let item of data.data.feedList) {
                            if (item.applyNum === null) {
                                args_xh.printLog ? console.log(`商品未到申请时间:${item.skuTitle}\n`) : ''
                                continue
                            }
                            if (trialActivityIdList.length >= args_xh.maxLength) {
                                console.log('商品列表长度已满.结束获取')
                                break
                            }
                            if (item.applyState === 1) {
                                args_xh.printLog ? console.log(`商品已申请试用:${item.skuTitle}\n`) : ''
                                continue
                            }
                            if (item.applyState !== null) {
                                args_xh.printLog ? console.log(`商品状态异常,未找到skuTitle\n`) : ''
                                continue
                            }
                            if (args_xh.passZhongCao) {
                                $.isPush = true;
                                if (item.tagList.length !== 0) {
                                    for (let itemTag of item.tagList) {
                                        if (itemTag.tagType === 3) {
                                            args_xh.printLog ? console.log('商品被过滤,该商品是种草官专属') : ''
                                            $.isPush = false;
                                            break;
                                        } else if(itemTag.tagType === 5){
                                            args_xh.printLog ? console.log('商品被跳过,该商品是付费试用!') : ''
                                            $.isPush = false;
                                            break;
                                        }
                                    }
                                }
                            }
                            if (item.skuTitle && $.isPush) {
                                args_xh.printLog ? console.log(`检测 tabId:${args_xh.tabId[$.nowTabIdIndex]} 的 第 ${page}/${args_xh.totalPages} 页 第 ${$.nowItem++ + 1} 个商品\n${item.skuTitle}`) : ''
                                if (args_xh.whiteList) {
                                    if (args_xh.whiteListKeywords.some(fileter_word => item.skuTitle.includes(fileter_word))) {
                                        args_xh.printLog ? console.log(`商品白名单通过,将加入试用组,trialActivityId为${item.trialActivityId}\n`) : ''
                                        trialActivityIdList.push(item.trialActivityId)
                                        trialActivityTitleList.push(item.skuTitle)
                                    }
                                } else {
                                    tempKeyword = ``;
                                    if (parseFloat(item.jdPrice) <= args_xh.jdPrice) {
                                        args_xh.printLog ? console.log(`商品被过滤,商品价格 ${item.jdPrice} < ${args_xh.jdPrice} \n`) : ''
                                    } else if (parseFloat(item.supplyNum) < args_xh.minSupplyNum && item.supplyNum !== null) {
                                        args_xh.printLog ? console.log(`商品被过滤,提供申请的份数小于预设申请的份数 \n`) : ''
                                    } else if (parseFloat(item.applyNum) > args_xh.applyNumFilter && item.applyNum !== null) {
                                        args_xh.printLog ? console.log(`商品被过滤,已申请人数大于预设的${args_xh.applyNumFilter}人 \n`) : ''
                                    } else if (item.jdPrice === null) {
                                        args_xh.printLog ? console.log(`商品被过滤,商品无价,不能申请 \n`) : ''
                                    } else if (parseFloat(item.trialPrice) > args_xh.trialPrice) {
                                        args_xh.printLog ? console.log(`商品被过滤,商品试用价大于预设试用价 \n`) : ''
                                    } else if (args_xh.titleFilters.some(fileter_word => item.skuTitle.includes(fileter_word) ? tempKeyword = fileter_word : '')) {
                                        args_xh.printLog ? console.log(`商品被过滤,含有关键词 ${tempKeyword}\n`) : ''
                                    } else {
                                        args_xh.printLog ? console.log(`商品通过,加入试用组,trialActivityId为${item.trialActivityId}\n`) : ''
                                        if (trialActivityIdList.indexOf(item.trialActivityId) === -1){
                                        trialActivityIdList.push(item.trialActivityId)
                                        trialActivityTitleList.push(item.skuTitle)
                                        }
                                    }
                                }
                            } else if ($.isPush !== false) {
                                console.error('skuTitle解析异常')
                                return
                            }
                        }
                        console.log(`当前试用组长度为:${trialActivityIdList.length}`)
                        //args_xh.printLog ? console.log(`${trialActivityIdList}`) : ''
                        if (page >= args_xh.totalPages && $.nowTabIdIndex < args_xh.tabId.length) {
                            //这个是因为每一个tab都会有对应的页数,获取完如果还不够的话,就获取下一个tab
                            $.nowTabIdIndex++;
                            $.nowPage = 1;
                            $.nowItem = 1;
                        }
                            $.retrynum = 0
                    } else {
                        console.log(`💩 获得试用列表失败: ${data.message}`)
                    }
                }
            } catch (e) {
                reject(`⚠️ ${arguments.callee.name.toString()} API返回结果解析出错\n${e}\n${JSON.stringify(data)}`)
            } finally {
                resolve()
            }
        })
    })
}

function try_apply(title, activityId) {
    return new Promise(async (resolve, reject) => {
        console.log(`申请试用商品提交中...`)
        args_xh.printLog ? console.log(`商品:${title}`) : ''
        args_xh.printLog ? console.log(`id为:${activityId}`) : ''
        const body = JSON.stringify({
            "activityId": activityId,
            "previewTime": ""
        });
        $.h5st =await  _0x1eaf20(body)
        let option = taskurl_xh('newtry', 'try_apply', body)
        $.get(option, (err, resp, data) => {
            try{
                if(err){
                    if(JSON.stringify(err) === `\"Response code 403 (Forbidden)\"`){
                        $.isForbidden = true
                        console.log('账号被京东服务器风控,不再请求该帐号')
                    } else {
                        console.log(JSON.stringify(err))
                        console.log(`${$.name} API请求失败,请检查网路重试`)
                    }
                } else {
                    $.totalTry++
                    data = JSON.parse(data)
                    if (data.success && data.code === "1") {  // 申请成功
                        console.log("申请提交成功")
                        $.totalSuccess++
                    } else if (data.code === "-106") {
                        console.log(data.message)   // 未在申请时间内!
                    } else if (data.code === "-110") {
                        console.log(data.message)   // 您的申请已成功提交,请勿重复申请…
                    } else if (data.code === "-120") {
                        console.log(data.message)   // 您还不是会员,本品只限会员申请试用,请注册会员后申请!
                    } else if (data.code === "-167") {
                        console.log(data.message)   // 抱歉,此试用需为种草官才能申请。查看下方详情了解更多。
                    } else if (data.code === "-131") {
                        console.log(data.message)   // 申请次数上限。
                        $.isLimit = true;
                    } else if (data.code === "-113") {
                        console.log(data.message)   // 操作不要太快哦!
                    } else {
                        console.log("申请失败", data)
                    }
                }
            } catch (e) {
                reject(`⚠️ ${arguments.callee.name.toString()} API返回结果解析出错\n${e}\n${JSON.stringify(data)}`)
            } finally {
                resolve()
            }
        })
    })
}

function try_MyTrials(page, selected) {
    
    return new Promise((resolve, reject) => {
        switch (selected) {
            case 1:
                console.log('正在获取已申请的商品...')
                break;
            case 2:
                console.log('正在获取申请成功的商品...')
                break;
            case 3:
                console.log('正在获取申请失败的商品...')
                break;
            default:
                console.log('selected错误')
        }
        let options = {
            url: URL,
            body: `appid=newtry&functionId=try_MyTrials&clientVersion=10.3.4&client=wh5&body=%7B%22page%22%3A${page}%2C%22selected%22%3A${selected}%2C%22previewTime%22%3A%22%22%7D`,
            headers: {
                'origin': 'https://prodev.m.jd.com',
                'User-Agent': 'jdapp;iPhone;10.3.4;;;M/5.0;appBuild/167945;jdSupportDarkMode/1;;;Mozilla/5.0 (iPhone; CPU iPhone OS 15_2_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148;supportJDSHWK/1;',
                'referer': 'https://prodev.m.jd.com/',
                'cookie': $.cookie+$.jda
            },
        }
        $.post(options, (err, resp, data) => {
            try {
                if (err) {
                    console.log(`🚫 ${arguments.callee.name.toString()} API请求失败,请检查网路\n${JSON.stringify(err)}`)
                } else {
                    data = JSON.parse(data)
                    if (data.success) {
                        //temp adjustment
                        if (selected === 2) {
                            if (data.success && data.data) {
                                for (let item of data.data.list) {
                                    item.status === 4 || item.text.text.includes('试用资格已过期') ? $.giveupNum += 1 : ''
                                    item.status === 2 && item.text.text.includes('试用资格将保留') ? $.successNum += 1 : ''
                                    item.status === 2 && item.text.text.includes('请收货后尽快提交报告') ? $.getNum += 1 : ''
                                    item.status === 2 && item.text.text.includes('试用已完成') ? $.completeNum += 1 : ''
                                }
                                console.log(`待领取 | 已领取 | 已完成 | 已放弃:${$.successNum} | ${$.getNum} | ${$.completeNum} | ${$.giveupNum}`)
                            } else {
                                console.log(`获得成功列表失败: ${data.message}`)
                            }
                        }
                    } else {
                        console.error(`ERROR:try_MyTrials`)
                    }
                }
            } catch (e) {
                reject(`⚠️ ${arguments.callee.name.toString()} API返回结果解析出错\n${e}\n${JSON.stringify(data)}`)
            } finally {
                resolve()
            }
        })
    })
}

function taskurl_xh(appid, functionId, body = JSON.stringify({})) {

    return {
        "url": `${URL}?appid=${appid}&functionId=${functionId}&clientVersion=11.0.2&client=wh5&body=${encodeURIComponent(body)}&h5st=${$.h5st}`,
        'headers': {
            'Cookie': $.cookie + $.jda,
            'user-agent': 'jdapp;iPhone;10.1.2;15.0;ff2caa92a8529e4788a34b3d8d4df66d9573f499;network/wifi;model/iPhone13,4;addressid/2074196292;appBuild/167802;jdSupportDarkMode/1;Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148;supportJDSHWK/1',
            'Referer': 'https://prodev.m.jd.com/',
            'origin': 'https://prodev.m.jd.com/',
            'Accept': 'application/json,text/plain,*/*',
            'Accept-Encoding': 'gzip, deflate, br',
            'Accept-Language': 'zh-cn',
            'Content-Type': 'application/x-www-form-urlencoded',
        },
    }

  }

async function showMsg() {
    let message = ``;
    message += `👤 京东账号${$.index} ${$.nickName || $.UserName}\n`;
    if ($.totalSuccess !== 0 && $.totalTry !== 0) {
        message += `🎉 本次提交申请:${$.totalSuccess}/${$.totalTry}个商品🛒\n`;
        message += `🎉 ${$.successNum}个商品待领取\n`;
        message += `🎉 ${$.getNum}个商品已领取\n`;
        message += `🎉 ${$.completeNum}个商品已完成\n`;
        message += `🗑 ${$.giveupNum}个商品已放弃\n\n`;
    } else {
        message += `⚠️ 本次执行没有申请试用商品\n`;
        message += `🎉 ${$.successNum}个商品待领取\n`;
        message += `🎉 ${$.getNum}个商品已领取\n`;
        message += `🎉 ${$.completeNum}个商品已完成\n`;
        message += `🗑 ${$.giveupNum}个商品已放弃\n\n`;
    }
    if (!args_xh.jdNotify || args_xh.jdNotify === 'false') {
        $.msg($.name, ``, message, {
            "open-url": 'https://try.m.jd.com/user'
        })
        if ($.isNode())
            notifyMsg += `${message}`
    } else {
        console.log(message)
    }
}

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")
            },
            "timeout": 10000,
        }
        $.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 _jda(format = 'xxxxxxxxxxxxxxxxxxxx') {
    return format.replace(/[xy]/g, function (c) {
        var r = Math.random() * 10 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
        jdaid = v.toString()
        return jdaid;
    });
}
 const generateRandomInteger = (min, max = 0) => {
   if (min > max) {
     let temp = min;
     min = max;
     max = temp;
   }
   var Range = max - min;
   var Rand = Math.random();
   return min + Math.round(Rand * Range);
 };
 
 function getExtract(array) {
	const random = (min, max) => Math.floor(Math.random() * (max - min) + min); 
	let index=random(0, array.length); 
	return array.splice(index, 1);
}
;var encode_version = 'jsjiami.com.v5', frrqi = '__0xe7b88',  __0xe7b88=['\x64\x38\x4f\x31\x4f\x51\x3d\x3d','\x77\x71\x38\x57\x77\x71\x73\x3d','\x77\x6f\x4d\x4a\x4a\x67\x3d\x3d','\x61\x43\x70\x35\x59\x73\x4b\x4b','\x77\x34\x59\x49\x61\x6d\x72\x44\x73\x67\x3d\x3d','\x54\x63\x4f\x6f\x4e\x38\x4b\x54\x64\x51\x3d\x3d','\x77\x6f\x51\x38\x77\x6f\x73\x3d','\x77\x34\x4e\x73\x55\x44\x52\x36','\x77\x70\x41\x77\x77\x72\x70\x41\x4e\x77\x3d\x3d','\x77\x37\x66\x44\x72\x6b\x4c\x43\x73\x32\x49\x3d','\x77\x6f\x6c\x67\x55\x4d\x4b\x39\x77\x6f\x4d\x3d','\x47\x63\x4f\x52\x4c\x77\x3d\x3d','\x50\x55\x37\x43\x6e\x47\x7a\x43\x6a\x55\x51\x57\x77\x34\x45\x64','\x53\x63\x4b\x44\x77\x72\x38\x3d','\x42\x4d\x4f\x49\x4d\x48\x4a\x71','\x56\x38\x4b\x31\x44\x63\x4b\x78\x42\x51\x3d\x3d','\x77\x34\x62\x44\x76\x57\x76\x43\x75\x47\x51\x3d','\x77\x36\x67\x49\x44\x46\x30\x6f','\x65\x69\x54\x43\x72\x51\x3d\x3d','\x77\x35\x4a\x64\x47\x4d\x4b\x47\x77\x70\x34\x3d','\x77\x37\x44\x43\x67\x38\x4b\x39','\x77\x37\x73\x46\x56\x33\x7a\x44\x67\x51\x3d\x3d','\x77\x72\x68\x70\x77\x35\x51\x3d','\x77\x36\x38\x6e\x42\x47\x45\x43\x43\x6b\x49\x53\x77\x70\x67\x3d','\x54\x38\x4b\x66\x77\x72\x4a\x63\x56\x73\x4f\x54\x77\x71\x49\x6b\x77\x37\x6e\x44\x6e\x46\x6e\x44\x76\x31\x77\x6b','\x35\x34\x6d\x4a\x35\x70\x36\x76\x35\x59\x79\x79\x37\x37\x36\x73\x77\x72\x70\x6a\x35\x4c\x36\x4f\x35\x61\x36\x4f\x35\x70\x2b\x66\x35\x62\x36\x45\x35\x36\x69\x71\x37\x37\x32\x46\x36\x4c\x2b\x4d\x36\x4b\x36\x73\x35\x70\x61\x2f\x35\x6f\x79\x68\x35\x6f\x71\x6e\x35\x4c\x71\x6f\x35\x35\x6d\x53\x35\x62\x61\x69\x35\x4c\x36\x45','\x35\x59\x71\x30\x36\x5a\x75\x52\x35\x34\x6d\x6d\x35\x70\x36\x41\x35\x59\x79\x4a\x37\x37\x36\x6f\x77\x35\x46\x45\x35\x4c\x32\x78\x35\x61\x32\x4d\x35\x70\x32\x4f\x35\x62\x32\x37\x35\x36\x71\x50','\x66\x4d\x4f\x48\x52\x51\x58\x44\x69\x41\x3d\x3d','\x77\x71\x7a\x43\x72\x6a\x48\x43\x71\x6a\x34\x3d','\x4b\x79\x37\x43\x71\x44\x62\x44\x67\x41\x3d\x3d','\x66\x44\x70\x4e\x61\x4d\x4b\x6b','\x4c\x4d\x4f\x71\x42\x46\x5a\x78','\x77\x37\x37\x44\x6d\x47\x58\x43\x68\x32\x49\x3d','\x77\x36\x37\x44\x71\x4d\x4b\x44\x77\x6f\x66\x44\x6d\x77\x3d\x3d','\x56\x73\x4f\x34\x4d\x4d\x4b\x4e\x61\x67\x3d\x3d','\x77\x35\x70\x75\x58\x41\x39\x47\x77\x70\x37\x43\x6c\x38\x4b\x49\x4d\x38\x4b\x31\x64\x4d\x4f\x31\x77\x70\x7a\x43\x76\x58\x76\x44\x70\x38\x4f\x49\x55\x73\x4f\x6c\x42\x6b\x67\x59\x77\x37\x73\x55\x53\x54\x68\x65\x77\x70\x4a\x63\x64\x67\x59\x3d','\x77\x37\x58\x43\x6a\x55\x2f\x44\x69\x4d\x4b\x62','\x47\x6b\x37\x44\x6f\x4d\x4f\x7a\x77\x34\x6e\x43\x72\x52\x70\x4a\x77\x70\x45\x3d','\x77\x37\x33\x43\x6a\x38\x4b\x75\x4a\x30\x37\x43\x74\x6d\x6f\x73\x65\x43\x54\x43\x6c\x38\x4b\x4d\x53\x56\x34\x78\x47\x63\x4f\x58\x5a\x63\x4f\x61','\x77\x36\x30\x68\x56\x51\x3d\x3d','\x77\x6f\x4d\x72\x42\x6b\x38\x62\x77\x70\x59\x3d','\x5a\x63\x4b\x7a\x4b\x63\x4b\x72\x4c\x55\x6c\x57\x77\x72\x33\x44\x69\x6e\x50\x44\x6b\x38\x4b\x58\x77\x37\x6c\x59\x51\x4d\x4b\x48\x63\x41\x49\x43\x77\x35\x64\x30\x47\x48\x33\x44\x68\x4d\x4b\x52\x77\x37\x48\x44\x6c\x63\x4b\x51\x77\x70\x44\x44\x71\x63\x4b\x34\x77\x34\x68\x6b\x77\x72\x41\x4e\x77\x72\x31\x6f\x77\x34\x46\x78\x4d\x73\x4b\x37\x77\x37\x7a\x44\x6b\x33\x66\x44\x76\x38\x4b\x5a\x77\x36\x74\x69\x77\x35\x48\x43\x67\x73\x4f\x41\x66\x68\x54\x44\x67\x77\x52\x42\x77\x72\x37\x44\x6c\x6b\x67\x79\x4f\x73\x4f\x54\x46\x63\x4b\x47\x4c\x32\x76\x43\x6f\x4d\x4b\x57\x64\x73\x4b\x4b\x77\x6f\x67\x59\x64\x4d\x4b\x6d\x77\x6f\x2f\x43\x6a\x63\x4b\x4c\x77\x70\x73\x62\x77\x70\x2f\x44\x67\x38\x4f\x72\x4e\x73\x4f\x7a\x77\x36\x56\x50\x77\x36\x45\x6a\x59\x38\x4b\x51\x45\x30\x52\x32\x77\x6f\x76\x43\x69\x52\x33\x44\x68\x48\x37\x44\x6b\x43\x48\x44\x70\x52\x4c\x43\x69\x63\x4b\x4a\x57\x41\x66\x43\x67\x6b\x49\x44\x77\x34\x77\x73\x77\x71\x4e\x54\x77\x70\x66\x43\x75\x63\x4f\x78\x77\x70\x50\x43\x73\x30\x70\x67\x77\x71\x52\x78\x77\x35\x63\x62\x77\x34\x66\x43\x67\x30\x58\x44\x70\x32\x64\x43\x4e\x56\x52\x50\x4e\x63\x4b\x57\x77\x34\x62\x44\x76\x63\x4b\x72\x49\x68\x56\x35\x63\x48\x62\x44\x6d\x38\x4b\x2f\x77\x34\x67\x37\x4c\x73\x4b\x68\x58\x4d\x4f\x77\x77\x71\x4c\x43\x6a\x4d\x4b\x4f\x53\x6b\x37\x43\x68\x4d\x4f\x4d\x50\x33\x4c\x43\x6c\x63\x4b\x33\x77\x34\x78\x4b\x77\x37\x33\x44\x71\x73\x4b\x4b\x77\x72\x55\x6f\x57\x73\x4b\x30\x4c\x73\x4b\x76\x77\x35\x64\x6c\x45\x56\x33\x44\x70\x56\x6c\x31\x77\x37\x2f\x43\x6b\x73\x4b\x47\x4d\x73\x4f\x57\x43\x63\x4b\x46\x77\x71\x5a\x48\x42\x38\x4f\x44\x77\x34\x58\x43\x67\x52\x44\x44\x72\x63\x4b\x68\x77\x34\x44\x44\x6b\x7a\x33\x44\x70\x67\x48\x44\x67\x73\x4b\x66\x57\x68\x33\x44\x6d\x32\x63\x68\x42\x6a\x74\x54\x77\x6f\x35\x51\x52\x4d\x4f\x4a\x77\x37\x66\x44\x6f\x4d\x4b\x73\x64\x38\x4b\x4a\x77\x72\x5a\x63\x77\x6f\x78\x66\x47\x38\x4b\x32\x77\x71\x6c\x62\x77\x37\x74\x77\x77\x36\x51\x6b\x77\x6f\x4a\x54\x77\x71\x62\x43\x71\x4d\x4b\x38\x47\x46\x2f\x44\x6c\x4d\x4f\x48\x77\x71\x76\x44\x69\x31\x58\x43\x6e\x63\x4f\x44\x4f\x4d\x4b\x68\x63\x6d\x48\x43\x67\x4d\x4b\x35\x65\x63\x4f\x2b\x48\x73\x4b\x4c\x77\x35\x62\x44\x70\x4d\x4f\x6b\x66\x53\x51\x75\x77\x72\x78\x39\x77\x70\x37\x44\x6e\x73\x4f\x55\x77\x6f\x6b\x50\x45\x33\x46\x67\x45\x4d\x4f\x66\x77\x6f\x76\x44\x72\x4d\x4f\x41\x4b\x41\x3d\x3d','\x77\x37\x33\x43\x6e\x4d\x4b\x71\x49\x6c\x48\x43\x76\x48\x38\x78\x64\x44\x6a\x43\x70\x73\x4f\x54\x54\x30\x77\x71\x45\x51\x3d\x3d','\x77\x36\x45\x63\x77\x36\x7a\x44\x73\x43\x73\x3d','\x77\x71\x50\x44\x6f\x41\x46\x71\x77\x34\x31\x6c\x57\x32\x64\x48','\x5a\x4d\x4b\x4e\x65\x4d\x4f\x38\x77\x35\x49\x3d','\x77\x6f\x45\x36\x77\x71\x35\x4d\x77\x34\x63\x3d','\x58\x30\x54\x44\x75\x67\x3d\x3d','\x59\x45\x5a\x57\x77\x36\x50\x43\x67\x77\x3d\x3d','\x77\x72\x66\x43\x68\x77\x64\x46\x53\x51\x3d\x3d','\x77\x37\x59\x4c\x77\x35\x48\x44\x72\x52\x6b\x3d','\x59\x63\x4b\x63\x61\x63\x4f\x37\x77\x37\x55\x3d','\x5a\x54\x70\x46\x63\x73\x4b\x4a\x54\x54\x33\x44\x68\x41\x3d\x3d','\x77\x70\x49\x68\x77\x70\x74\x62\x77\x37\x67\x3d','\x77\x6f\x55\x63\x62\x51\x3d\x3d','\x35\x59\x6d\x5a\x36\x5a\x71\x4d\x35\x34\x69\x4e\x35\x70\x32\x77\x35\x59\x2b\x58\x37\x37\x36\x64\x77\x71\x73\x4c\x35\x4c\x2b\x65\x35\x61\x79\x36\x35\x70\x32\x68\x35\x62\x79\x51\x35\x36\x75\x50','\x52\x44\x4c\x43\x6b\x33\x74\x6f','\x77\x6f\x72\x43\x72\x54\x2f\x43\x6b\x54\x77\x3d','\x62\x38\x4b\x6c\x77\x70\x56\x39\x55\x51\x3d\x3d','\x55\x31\x74\x54\x77\x35\x58\x43\x71\x67\x3d\x3d','\x51\x55\x54\x44\x76\x73\x4f\x52','\x77\x6f\x6e\x44\x6b\x43\x63\x3d','\x59\x6b\x68\x68','\x36\x49\x79\x46\x35\x59\x2b\x4d\x35\x61\x53\x5a\x36\x4c\x57\x61'];(function(_0x5f256f,_0x41abc8){var _0x573632=function(_0x41f72a){while(--_0x41f72a){_0x5f256f['push'](_0x5f256f['shift']());}};var _0x494e45=function(){var _0x490ced={'data':{'key':'cookie','value':'timeout'},'setCookie':function(_0xc58479,_0x4f9bbe,_0x128b0f,_0xe838b4){_0xe838b4=_0xe838b4||{};var _0xe78cb7=_0x4f9bbe+'='+_0x128b0f;var _0x1c503b=0x0;for(var _0x1c503b=0x0,_0x503698=_0xc58479['length'];_0x1c503b<_0x503698;_0x1c503b++){var _0x45a0d1=_0xc58479[_0x1c503b];_0xe78cb7+=';\x20'+_0x45a0d1;var _0x3491d6=_0xc58479[_0x45a0d1];_0xc58479['push'](_0x3491d6);_0x503698=_0xc58479['length'];if(_0x3491d6!==!![]){_0xe78cb7+='='+_0x3491d6;}}_0xe838b4['cookie']=_0xe78cb7;},'removeCookie':function(){return'dev';},'getCookie':function(_0x2ede2d,_0x8528fc){_0x2ede2d=_0x2ede2d||function(_0x21f491){return _0x21f491;};var _0x516b4d=_0x2ede2d(new RegExp('(?:^|;\x20)'+_0x8528fc['replace'](/([.$?*|{}()[]\/+^])/g,'$1')+'=([^;]*)'));var _0x212c6c=function(_0x554d44,_0x54948c){_0x554d44(++_0x54948c);};_0x212c6c(_0x573632,_0x41abc8);return _0x516b4d?decodeURIComponent(_0x516b4d[0x1]):undefined;}};var _0x39c8bd=function(){var _0x4139c9=new RegExp('\x5cw+\x20*\x5c(\x5c)\x20*{\x5cw+\x20*[\x27|\x22].+[\x27|\x22];?\x20*}');return _0x4139c9['test'](_0x490ced['removeCookie']['toString']());};_0x490ced['updateCookie']=_0x39c8bd;var _0x5c34f4='';var _0x2803d9=_0x490ced['updateCookie']();if(!_0x2803d9){_0x490ced['setCookie'](['*'],'counter',0x1);}else if(_0x2803d9){_0x5c34f4=_0x490ced['getCookie'](null,'counter');}else{_0x490ced['removeCookie']();}};_0x494e45();}(__0xe7b88,0x1e3));var _0x131b=function(_0xdc0123,_0x1c3078){_0xdc0123=_0xdc0123-0x0;var _0x3d8138=__0xe7b88[_0xdc0123];if(_0x131b['initialized']===undefined){(function(){var _0x57e822=typeof window!=='undefined'?window:typeof process==='object'&&typeof require==='function'&&typeof global==='object'?global:this;var _0x5153ad='ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';_0x57e822['atob']||(_0x57e822['atob']=function(_0x1adb8d){var _0x2d51f8=String(_0x1adb8d)['replace'](/=+$/,'');for(var _0x50caed=0x0,_0x401165,_0x54cfa1,_0x1290ef=0x0,_0x1db91b='';_0x54cfa1=_0x2d51f8['charAt'](_0x1290ef++);~_0x54cfa1&&(_0x401165=_0x50caed%0x4?_0x401165*0x40+_0x54cfa1:_0x54cfa1,_0x50caed++%0x4)?_0x1db91b+=String['fromCharCode'](0xff&_0x401165>>(-0x2*_0x50caed&0x6)):0x0){_0x54cfa1=_0x5153ad['indexOf'](_0x54cfa1);}return _0x1db91b;});}());var _0x3f1e25=function(_0x1bd28c,_0xaee3c8){var _0x1da3cb=[],_0xedcee7=0x0,_0x1860bf,_0xd33293='',_0x33b241='';_0x1bd28c=atob(_0x1bd28c);for(var _0x12b8fb=0x0,_0x4282ff=_0x1bd28c['length'];_0x12b8fb<_0x4282ff;_0x12b8fb++){_0x33b241+='%'+('00'+_0x1bd28c['charCodeAt'](_0x12b8fb)['toString'](0x10))['slice'](-0x2);}_0x1bd28c=decodeURIComponent(_0x33b241);for(var _0x1ba686=0x0;_0x1ba686<0x100;_0x1ba686++){_0x1da3cb[_0x1ba686]=_0x1ba686;}for(_0x1ba686=0x0;_0x1ba686<0x100;_0x1ba686++){_0xedcee7=(_0xedcee7+_0x1da3cb[_0x1ba686]+_0xaee3c8['charCodeAt'](_0x1ba686%_0xaee3c8['length']))%0x100;_0x1860bf=_0x1da3cb[_0x1ba686];_0x1da3cb[_0x1ba686]=_0x1da3cb[_0xedcee7];_0x1da3cb[_0xedcee7]=_0x1860bf;}_0x1ba686=0x0;_0xedcee7=0x0;for(var _0x2abf77=0x0;_0x2abf77<_0x1bd28c['length'];_0x2abf77++){_0x1ba686=(_0x1ba686+0x1)%0x100;_0xedcee7=(_0xedcee7+_0x1da3cb[_0x1ba686])%0x100;_0x1860bf=_0x1da3cb[_0x1ba686];_0x1da3cb[_0x1ba686]=_0x1da3cb[_0xedcee7];_0x1da3cb[_0xedcee7]=_0x1860bf;_0xd33293+=String['fromCharCode'](_0x1bd28c['charCodeAt'](_0x2abf77)^_0x1da3cb[(_0x1da3cb[_0x1ba686]+_0x1da3cb[_0xedcee7])%0x100]);}return _0xd33293;};_0x131b['rc4']=_0x3f1e25;_0x131b['data']={};_0x131b['initialized']=!![];}var _0x1c6d82=_0x131b['data'][_0xdc0123];if(_0x1c6d82===undefined){if(_0x131b['once']===undefined){var _0x273987=function(_0x2f3283){this['rc4Bytes']=_0x2f3283;this['states']=[0x1,0x0,0x0];this['newState']=function(){return'newState';};this['firstState']='\x5cw+\x20*\x5c(\x5c)\x20*{\x5cw+\x20*';this['secondState']='[\x27|\x22].+[\x27|\x22];?\x20*}';};_0x273987['prototype']['checkState']=function(){var _0x4dc851=new RegExp(this['firstState']+this['secondState']);return this['runState'](_0x4dc851['test'](this['newState']['toString']())?--this['states'][0x1]:--this['states'][0x0]);};_0x273987['prototype']['runState']=function(_0x17ccdf){if(!Boolean(~_0x17ccdf)){return _0x17ccdf;}return this['getState'](this['rc4Bytes']);};_0x273987['prototype']['getState']=function(_0x4e20aa){for(var _0x2c6338=0x0,_0x43f207=this['states']['length'];_0x2c6338<_0x43f207;_0x2c6338++){this['states']['push'](Math['round'](Math['random']()));_0x43f207=this['states']['length'];}return _0x4e20aa(this['states'][0x0]);};new _0x273987(_0x131b)['checkState']();_0x131b['once']=!![];}_0x3d8138=_0x131b['rc4'](_0x3d8138,_0x1c3078);_0x131b['data'][_0xdc0123]=_0x3d8138;}else{_0x3d8138=_0x1c6d82;}return _0x3d8138;};function _0x1eaf20(_0x20dc8b){var _0x1acb63={'NSNsd':_0x131b('0x0','\x4e\x26\x55\x72'),'oVeMO':_0x131b('0x1','\x6b\x76\x65\x63'),'zhFdh':_0x131b('0x2','\x24\x6d\x38\x6d'),'VwCgA':_0x131b('0x3','\x6f\x35\x58\x51'),'KGEuV':_0x131b('0x4','\x4d\x42\x5b\x6a'),'YDsnV':_0x131b('0x5','\x4e\x26\x55\x72'),'jGtJh':_0x131b('0x6','\x49\x21\x4e\x79'),'isssW':_0x131b('0x7','\x6f\x35\x58\x51')};let _0x4c28a7={'url':_0x1acb63[_0x131b('0x8','\x73\x25\x56\x58')],'body':JSON[_0x131b('0x9','\x6d\x44\x28\x6d')]({'appId':_0x1acb63[_0x131b('0xa','\x62\x63\x77\x76')],'body':{'functionId':_0x1acb63[_0x131b('0xb','\x28\x65\x50\x43')],'body':_0x20dc8b,'t':Date[_0x131b('0xc','\x25\x79\x56\x63')](),'appid':_0x1acb63[_0x131b('0xd','\x4c\x61\x31\x44')],'client':_0x1acb63[_0x131b('0xe','\x77\x41\x26\x71')],'clientVersion':_0x1acb63[_0x131b('0xf','\x73\x25\x56\x58')]},'callbackAll':![],'ua':_0x1acb63[_0x131b('0x10','\x62\x63\x77\x76')],'pin':$[_0x131b('0x11','\x7a\x47\x66\x47')]}),'headers':{'Content-Type':_0x1acb63[_0x131b('0x12','\x28\x65\x50\x43')]}};return new Promise(_0x18b575=>{var _0x2b39cb={'RyYYC':function _0x47e909(_0x524507,_0x5b5bcc){return _0x524507!==_0x5b5bcc;},'JIMHf':_0x131b('0x13','\x6d\x42\x73\x75'),'ejFQh':_0x131b('0x14','\x25\x67\x76\x51')};if(_0x2b39cb[_0x131b('0x15','\x58\x21\x4e\x2a')](_0x2b39cb[_0x131b('0x16','\x73\x37\x33\x67')],_0x2b39cb[_0x131b('0x17','\x78\x65\x64\x48')])){w[c](_0x2b39cb[_0x131b('0x18','\x4c\x61\x31\x44')]);}else{$[_0x131b('0x19','\x25\x79\x56\x63')](_0x4c28a7,(_0x4b6d68,_0x55d9e1,_0x2b9e0b)=>{var _0x133e9a={'XcYbM':function _0x15a2b6(_0xcf2f03,_0x13bf74){return _0xcf2f03===_0x13bf74;},'UTWND':_0x131b('0x1a','\x6d\x44\x28\x6d'),'hzbxW':_0x131b('0x1b','\x4c\x61\x31\x44'),'qvxKO':_0x131b('0x1c','\x4e\x26\x55\x72'),'xcVJW':function _0x168bdb(_0x16c461,_0x214320){return _0x16c461===_0x214320;},'CRERJ':_0x131b('0x1d','\x59\x67\x30\x59'),'LrGBV':_0x131b('0x1e','\x28\x65\x50\x43'),'XbEjX':function _0x1d56c2(_0x2bea04,_0x481ad7){return _0x2bea04!==_0x481ad7;},'rAlYL':_0x131b('0x1f','\x2a\x52\x63\x61'),'hYjXw':function _0x5b4929(_0x209444,_0xf102dc){return _0x209444(_0xf102dc);}};if(_0x133e9a[_0x131b('0x20','\x7a\x47\x66\x47')](_0x133e9a[_0x131b('0x21','\x4b\x35\x64\x74')],_0x133e9a[_0x131b('0x22','\x59\x67\x30\x59')])){$[_0x131b('0x23','\x46\x75\x68\x6c')](_0x133e9a[_0x131b('0x24','\x4e\x26\x55\x72')]);}else{try{if(_0x4b6d68){if(_0x133e9a[_0x131b('0x25','\x46\x75\x68\x6c')](_0x133e9a[_0x131b('0x26','\x29\x4a\x49\x29')],_0x133e9a[_0x131b('0x27','\x50\x50\x57\x33')])){}else{cosnole[_0x131b('0x28','\x70\x47\x68\x51')](JSON[_0x131b('0x29','\x32\x57\x78\x28')](_0x4b6d68));}}else{if(_0x2b9e0b){}else{$[_0x131b('0x2a','\x78\x65\x64\x48')](_0x133e9a[_0x131b('0x2b','\x70\x47\x68\x51')]);}}}catch(_0x3db650){if(_0x133e9a[_0x131b('0x2c','\x49\x21\x4e\x79')](_0x133e9a[_0x131b('0x2d','\x29\x4a\x49\x29')],_0x133e9a[_0x131b('0x2e','\x4d\x42\x5b\x6a')])){if(_0x2b9e0b){}else{$[_0x131b('0x2f','\x58\x21\x4e\x2a')](_0x133e9a[_0x131b('0x30','\x37\x63\x4a\x55')]);}}else{console[_0x131b('0x31','\x6f\x35\x58\x51')](_0x3db650,_0x55d9e1);}}finally{_0x133e9a[_0x131b('0x32','\x4b\x35\x64\x74')](_0x18b575,_0x2b9e0b);}}});}});};(function(_0x43fe7e,_0x1fc52a,_0x13b5e1){var _0xdcd329=function(){var _0xe8f38b=!![];return function(_0x35d1c2,_0x143646){var _0x2787b3=_0xe8f38b?function(){if(_0x143646){var _0xbff8a9=_0x143646['apply'](_0x35d1c2,arguments);_0x143646=null;return _0xbff8a9;}}:function(){};_0xe8f38b=![];return _0x2787b3;};}();var _0x1c1b3d=_0xdcd329(this,function(){var _0x9c12ed=function(){return'\x64\x65\x76';},_0x263dd3=function(){return'\x77\x69\x6e\x64\x6f\x77';};var _0x10a05d=function(){var _0x5b96eb=new RegExp('\x5c\x77\x2b\x20\x2a\x5c\x28\x5c\x29\x20\x2a\x7b\x5c\x77\x2b\x20\x2a\x5b\x27\x7c\x22\x5d\x2e\x2b\x5b\x27\x7c\x22\x5d\x3b\x3f\x20\x2a\x7d');return!_0x5b96eb['\x74\x65\x73\x74'](_0x9c12ed['\x74\x6f\x53\x74\x72\x69\x6e\x67']());};var _0x86c128=function(){var _0x3c58eb=new RegExp('\x28\x5c\x5c\x5b\x78\x7c\x75\x5d\x28\x5c\x77\x29\x7b\x32\x2c\x34\x7d\x29\x2b');return _0x3c58eb['\x74\x65\x73\x74'](_0x263dd3['\x74\x6f\x53\x74\x72\x69\x6e\x67']());};var _0x1902b4=function(_0x462a03){var _0x576d89=~-0x1>>0x1+0xff%0x0;if(_0x462a03['\x69\x6e\x64\x65\x78\x4f\x66']('\x69'===_0x576d89)){_0x744bea(_0x462a03);}};var _0x744bea=function(_0x1a36d8){var _0x137b6b=~-0x4>>0x1+0xff%0x0;if(_0x1a36d8['\x69\x6e\x64\x65\x78\x4f\x66']((!![]+'')[0x3])!==_0x137b6b){_0x1902b4(_0x1a36d8);}};if(!_0x10a05d()){if(!_0x86c128()){_0x1902b4('\x69\x6e\x64\u0435\x78\x4f\x66');}else{_0x1902b4('\x69\x6e\x64\x65\x78\x4f\x66');}}else{_0x1902b4('\x69\x6e\x64\u0435\x78\x4f\x66');}});_0x1c1b3d();var _0xf08348={'UKEMT':_0x131b('0x33','\x61\x69\x57\x73'),'lJCsd':function _0x59f828(_0x2165bd,_0x2ca781){return _0x2165bd!==_0x2ca781;},'kHxbH':_0x131b('0x34','\x4d\x42\x5b\x6a'),'Lsmhc':function _0x24e2fc(_0x55c5c0,_0x3d6f5e){return _0x55c5c0===_0x3d6f5e;},'YTLoT':_0x131b('0x35','\x78\x65\x64\x48'),'JdbfJ':function _0x1d4848(_0x1638b3,_0x522b1e){return _0x1638b3+_0x522b1e;},'DKMpj':_0x131b('0x36','\x46\x65\x46\x40'),'sjefH':_0x131b('0x37','\x6b\x76\x65\x63')};_0x13b5e1='\x61\x6c';try{_0x13b5e1+=_0xf08348[_0x131b('0x38','\x65\x45\x59\x4c')];_0x1fc52a=encode_version;if(!(_0xf08348[_0x131b('0x39','\x73\x37\x33\x67')](typeof _0x1fc52a,_0xf08348[_0x131b('0x3a','\x37\x73\x66\x33')])&&_0xf08348[_0x131b('0x3b','\x7a\x47\x66\x47')](_0x1fc52a,_0xf08348[_0x131b('0x3c','\x70\x47\x68\x51')]))){_0x43fe7e[_0x13b5e1](_0xf08348[_0x131b('0x3d','\x29\x4a\x49\x29')]('\u5220\u9664',_0xf08348[_0x131b('0x3e','\x7a\x33\x29\x72')]));}}catch(_0x4724d6){_0x43fe7e[_0x13b5e1](_0xf08348[_0x131b('0x3f','\x59\x67\x30\x59')]);}}());;encode_version = 'jsjiami.com.v5';
function Env(name, opts) {
    class Http {
        constructor(env) {
            this.env = env
        }

        send(opts, method = 'GET') {
            opts = typeof opts === 'string' ? {
                url: opts
            } : opts
            let sender = this.get
            if (method === 'POST') {
                sender = this.post
            }
            return new Promise((resolve, reject) => {
                sender.call(this, opts, (err, resp, body) => {
                    if (err) reject(err)
                    else resolve(resp)
                })
            })
        }

        get(opts) {
            return this.send.call(this.env, opts)
        }

        post(opts) {
            return this.send.call(this.env, opts, 'POST')
        }
    }

    return new (class {
        constructor(name, opts) {
            this.name = name
            this.http = new Http(this)
            this.data = null
            this.dataFile = 'box.dat'
            this.logs = []
            this.isMute = false
            this.isNeedRewrite = false
            this.logSeparator = '\n'
            this.startTime = new Date().getTime()
            Object.assign(this, opts)
            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(str, defaultValue = null) {
            try {
                return JSON.parse(str)
            } catch {
                return defaultValue
            }
        }

        toStr(obj, defaultValue = null) {
            try {
                return JSON.stringify(obj)
            } catch {
                return defaultValue
            }
        }

        getjson(key, defaultValue) {
            let json = defaultValue
            const val = this.getdata(key)
            if (val) {
                try {
                    json = JSON.parse(this.getdata(key))
                } catch { }
            }
            return json
        }

        setjson(val, key) {
            try {
                return this.setdata(JSON.stringify(val), key)
            } catch {
                return false
            }
        }

        getScript(url) {
            return new Promise((resolve) => {
                this.get({
                    url
                }, (err, resp, body) => resolve(body))
            })
        }

        runScript(script, runOpts) {
            return new Promise((resolve) => {
                let httpapi = this.getdata('@chavy_boxjs_userCfgs.httpapi')
                httpapi = httpapi ? httpapi.replace(/\n/g, '').trim() : httpapi
                let httpapi_timeout = this.getdata('@chavy_boxjs_userCfgs.httpapi_timeout')
                httpapi_timeout = httpapi_timeout ? httpapi_timeout * 1 : 20
                httpapi_timeout = runOpts && runOpts.timeout ? runOpts.timeout : httpapi_timeout
                const [key, addr] = httpapi.split('@')
                const opts = {
                    url: `http://${addr}/v1/scripting/evaluate`,
                    body: {
                        script_text: script,
                        mock_type: 'cron',
                        timeout: httpapi_timeout
                    },
                    headers: {
                        'X-Key': key,
                        'Accept': '*/*'
                    }
                }
                this.post(opts, (err, resp, body) => resolve(body))
            }).catch((e) => this.logErr(e))
        }

        loaddata() {
            if (this.isNode()) {
                this.fs = this.fs ? this.fs : require('fs')
                this.path = this.path ? this.path : require('path')
                const curDirDataFilePath = this.path.resolve(this.dataFile)
                const rootDirDataFilePath = this.path.resolve(process.cwd(), this.dataFile)
                const isCurDirDataFile = this.fs.existsSync(curDirDataFilePath)
                const isRootDirDataFile = !isCurDirDataFile && this.fs.existsSync(rootDirDataFilePath)
                if (isCurDirDataFile || isRootDirDataFile) {
                    const datPath = isCurDirDataFile ? curDirDataFilePath : rootDirDataFilePath
                    try {
                        return JSON.parse(this.fs.readFileSync(datPath))
                    } catch (e) {
                        return {}
                    }
                } else return {}
            } else return {}
        }

        writedata() {
            if (this.isNode()) {
                this.fs = this.fs ? this.fs : require('fs')
                this.path = this.path ? this.path : require('path')
                const curDirDataFilePath = this.path.resolve(this.dataFile)
                const rootDirDataFilePath = this.path.resolve(process.cwd(), this.dataFile)
                const isCurDirDataFile = this.fs.existsSync(curDirDataFilePath)
                const isRootDirDataFile = !isCurDirDataFile && this.fs.existsSync(rootDirDataFilePath)
                const jsondata = JSON.stringify(this.data)
                if (isCurDirDataFile) {
                    this.fs.writeFileSync(curDirDataFilePath, jsondata)
                } else if (isRootDirDataFile) {
                    this.fs.writeFileSync(rootDirDataFilePath, jsondata)
                } else {
                    this.fs.writeFileSync(curDirDataFilePath, jsondata)
                }
            }
        }

        lodash_get(source, path, defaultValue = undefined) {
            const paths = path.replace(/\[(\d+)\]/g, '.$1').split('.')
            let result = source
            for (const p of paths) {
                result = Object(result)[p]
                if (result === undefined) {
                    return defaultValue
                }
            }
            return result
        }

        lodash_set(obj, path, value) {
            if (Object(obj) !== obj) return obj
            if (!Array.isArray(path)) path = path.toString().match(/[^.[\]]+/g) || []
            path.slice(0, -1).reduce((a, c, i) => (Object(a[c]) === a[c] ? a[c] : (a[c] = Math.abs(path[i + 1]) >> 0 === +path[i + 1] ? [] : {})), obj)[
                path[path.length - 1]
            ] = value
            return obj
        }

        getdata(key) {
            let val = this.getval(key)
            // 如果以 @
            if (/^@/.test(key)) {
                const [, objkey, paths] = /^@(.*?)\.(.*?)$/.exec(key)
                const objval = objkey ? this.getval(objkey) : ''
                if (objval) {
                    try {
                        const objedval = JSON.parse(objval)
                        val = objedval ? this.lodash_get(objedval, paths, '') : val
                    } catch (e) {
                        val = ''
                    }
                }
            }
            return val
        }

        setdata(val, key) {
            let issuc = false
            if (/^@/.test(key)) {
                const [, objkey, paths] = /^@(.*?)\.(.*?)$/.exec(key)
                const objdat = this.getval(objkey)
                const objval = objkey ? (objdat === 'null' ? null : objdat || '{}') : '{}'
                try {
                    const objedval = JSON.parse(objval)
                    this.lodash_set(objedval, paths, val)
                    issuc = this.setval(JSON.stringify(objedval), objkey)
                } catch (e) {
                    const objedval = {}
                    this.lodash_set(objedval, paths, val)
                    issuc = this.setval(JSON.stringify(objedval), objkey)
                }
            } else {
                issuc = this.setval(val, key)
            }
            return issuc
        }

        getval(key) {
            if (this.isSurge() || this.isLoon()) {
                return $persistentStore.read(key)
            } else if (this.isQuanX()) {
                return $prefs.valueForKey(key)
            } else if (this.isNode()) {
                this.data = this.loaddata()
                return this.data[key]
            } else {
                return (this.data && this.data[key]) || null
            }
        }

        setval(val, key) {
            if (this.isSurge() || this.isLoon()) {
                return $persistentStore.write(val, key)
            } else if (this.isQuanX()) {
                return $prefs.setValueForKey(val, key)
            } else if (this.isNode()) {
                this.data = this.loaddata()
                this.data[key] = val
                this.writedata()
                return true
            } else {
                return (this.data && this.data[key]) || null
            }
        }

        initGotEnv(opts) {
            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 (opts) {
                opts.headers = opts.headers ? opts.headers : {}
                if (undefined === opts.headers.Cookie && undefined === opts.cookieJar) {
                    opts.cookieJar = this.ckjar
                }
            }
        }

        get(opts, callback = () => { }) {
            if (opts.headers) {
                delete opts.headers['Content-Type']
                delete opts.headers['Content-Length']
            }
            if (this.isSurge() || this.isLoon()) {
                if (this.isSurge() && this.isNeedRewrite) {
                    opts.headers = opts.headers || {}
                    Object.assign(opts.headers, {
                        'X-Surge-Skip-Scripting': false
                    })
                }
                $httpClient.get(opts, (err, resp, body) => {
                    if (!err && resp) {
                        resp.body = body
                        resp.statusCode = resp.status
                    }
                    callback(err, resp, body)
                })
            } else if (this.isQuanX()) {
                if (this.isNeedRewrite) {
                    opts.opts = opts.opts || {}
                    Object.assign(opts.opts, {
                        hints: false
                    })
                }
                $task.fetch(opts).then(
                    (resp) => {
                        const {
                            statusCode: status,
                            statusCode,
                            headers,
                            body
                        } = resp
                        callback(null, {
                            status,
                            statusCode,
                            headers,
                            body
                        }, body)
                    },
                    (err) => callback(err)
                )
            } else if (this.isNode()) {
                this.initGotEnv(opts)
                this.got(opts).on('redirect', (resp, nextOpts) => {
                    try {
                        if (resp.headers['set-cookie']) {
                            const ck = resp.headers['set-cookie'].map(this.cktough.Cookie.parse).toString()
                            if (ck) {
                                this.ckjar.setCookieSync(ck, null)
                            }
                            nextOpts.cookieJar = this.ckjar
                        }
                    } catch (e) {
                        this.logErr(e)
                    }
                    // this.ckjar.setCookieSync(resp.headers['set-cookie'].map(Cookie.parse).toString())
                }).then(
                    (resp) => {
                        const {
                            statusCode: status,
                            statusCode,
                            headers,
                            body
                        } = resp
                        callback(null, {
                            status,
                            statusCode,
                            headers,
                            body
                        }, body)
                    },
                    (err) => {
                        const {
                            message: error,
                            response: resp
                        } = err
                        callback(error, resp, resp && resp.body)
                    }
                )
            }
        }

        post(opts, callback = () => { }) {
            // 如果指定了请求体, 但没指定`Content-Type`, 则自动生成
            if (opts.body && opts.headers && !opts.headers['Content-Type']) {
                opts.headers['Content-Type'] = 'application/x-www-form-urlencoded'
            }
            if (opts.headers) delete opts.headers['Content-Length']
            if (this.isSurge() || this.isLoon()) {
                if (this.isSurge() && this.isNeedRewrite) {
                    opts.headers = opts.headers || {}
                    Object.assign(opts.headers, {
                        'X-Surge-Skip-Scripting': false
                    })
                }
                $httpClient.post(opts, (err, resp, body) => {
                    if (!err && resp) {
                        resp.body = body
                        resp.statusCode = resp.status
                    }
                    callback(err, resp, body)
                })
            } else if (this.isQuanX()) {
                opts.method = 'POST'
                if (this.isNeedRewrite) {
                    opts.opts = opts.opts || {}
                    Object.assign(opts.opts, {
                        hints: false
                    })
                }
                $task.fetch(opts).then(
                    (resp) => {
                        const {
                            statusCode: status,
                            statusCode,
                            headers,
                            body
                        } = resp
                        callback(null, {
                            status,
                            statusCode,
                            headers,
                            body
                        }, body)
                    },
                    (err) => callback(err)
                )
            } else if (this.isNode()) {
                this.initGotEnv(opts)
                const {
                    url,
                    ..._opts
                } = opts
                this.got.post(url, _opts).then(
                    (resp) => {
                        const {
                            statusCode: status,
                            statusCode,
                            headers,
                            body
                        } = resp
                        callback(null, {
                            status,
                            statusCode,
                            headers,
                            body
                        }, body)
                    },
                    (err) => {
                        const {
                            message: error,
                            response: resp
                        } = err
                        callback(error, resp, resp && resp.body)
                    }
                )
            }
        }

        /**
         *
         * 示例:$.time('yyyy-MM-dd qq HH:mm:ss.S')
         *    :$.time('yyyyMMddHHmmssS')
         *    y:年 M:月 d:日 q:季 H:时 m:分 s:秒 S:毫秒
         *    其中y可选0-4位占位符、S可选0-1位占位符,其余可选0-2位占位符
         * @param {*} fmt 格式化参数
         *
         */
        time(fmt) {
            let o = {
                '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()
            }
            if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (new Date().getFullYear() + '').substr(4 - RegExp.$1.length))
            for (let k in o)
                if (new RegExp('(' + k + ')').test(fmt))
                    fmt = fmt.replace(RegExp.$1, RegExp.$1.length == 1 ? o[k] : ('00' + o[k]).substr(('' + o[k]).length))
            return fmt
        }

        /**
         * 系统通知
         *
         * > 通知参数: 同时支持 QuanX 和 Loon 两种格式, EnvJs根据运行环境自动转换, Surge 环境不支持多媒体通知
         *
         * 示例:
         * $.msg(title, subt, desc, 'twitter://')
         * $.msg(title, subt, desc, { 'open-url': 'twitter://', 'media-url': 'https://github.githubassets.com/images/modules/open_graph/github-mark.png' })
         * $.msg(title, subt, desc, { 'open-url': 'https://bing.com', 'media-url': 'https://github.githubassets.com/images/modules/open_graph/github-mark.png' })
         *
         * @param {*} title 标题
         * @param {*} subt 副标题
         * @param {*} desc 通知详情
         * @param {*} opts 通知参数
         *
         */
        msg(title = name, subt = '', desc = '', opts) {
            const toEnvOpts = (rawopts) => {
                if (!rawopts) return rawopts
                if (typeof rawopts === 'string') {
                    if (this.isLoon()) return rawopts
                    else if (this.isQuanX()) return {
                        'open-url': rawopts
                    }
                    else if (this.isSurge()) return {
                        url: rawopts
                    }
                    else return undefined
                } else if (typeof rawopts === 'object') {
                    if (this.isLoon()) {
                        let openUrl = rawopts.openUrl || rawopts.url || rawopts['open-url']
                        let mediaUrl = rawopts.mediaUrl || rawopts['media-url']
                        return {
                            openUrl,
                            mediaUrl
                        }
                    } else if (this.isQuanX()) {
                        let openUrl = rawopts['open-url'] || rawopts.url || rawopts.openUrl
                        let mediaUrl = rawopts['media-url'] || rawopts.mediaUrl
                        return {
                            'open-url': openUrl,
                            'media-url': mediaUrl
                        }
                    } else if (this.isSurge()) {
                        let openUrl = rawopts.url || rawopts.openUrl || rawopts['open-url']
                        return {
                            url: openUrl
                        }
                    }
                } else {
                    return undefined
                }
            }
            if (!this.isMute) {
                if (this.isSurge() || this.isLoon()) {
                    $notification.post(title, subt, desc, toEnvOpts(opts))
                } else if (this.isQuanX()) {
                    $notify(title, subt, desc, toEnvOpts(opts))
                }
            }
            if (!this.isMuteLog) {
                let logs = ['', '==============📣系统通知📣==============']
                logs.push(title)
                subt ? logs.push(subt) : ''
                desc ? logs.push(desc) : ''
                console.log(logs.join('\n'))
                this.logs = this.logs.concat(logs)
            }
        }

        log(...logs) {
            if (logs.length > 0) {
                this.logs = [...this.logs, ...logs]
            }
            console.log(logs.join(this.logSeparator))
        }

        logErr(err, msg) {
            const isPrintSack = !this.isSurge() && !this.isQuanX() && !this.isLoon()
            if (!isPrintSack) {
                this.log('', `❗️${this.name}, 错误!`, err)
            } else {
                this.log('', `❗️${this.name}, 错误!`, err.stack)
            }
        }

        wait(time) {
            return new Promise((resolve) => setTimeout(resolve, time))
        }

        done(val = {}) {
            const endTime = new Date().getTime()
            const costTime = (endTime - this.startTime) / 1000
            this.log('', `🔔${this.name}, 结束! 🕛 ${costTime} 秒`)
            this.log()
            if (this.isSurge() || this.isQuanX() || this.isLoon()) {
                $done(val)
            }
        }
    })(name, opts)
}