diff --git a/script/applestore/README.md b/script/applestore/README.md index eebb30147..ca72179d6 100644 --- a/script/applestore/README.md +++ b/script/applestore/README.md @@ -1,10 +1,10 @@ -# Apple Store iPhone 库存监控 +# Apple Store 库存监控 ## 前言 -这是一个可以让你感受绝望的脚本,感受那种明知有货,但还是抢不到的那种绝望。 +脚本用来监控线下AppleStore指定商品库存,目前验证过iPhone13、AppleWatch7系列。 长话短说,需要做一些准备: @@ -26,6 +26,16 @@ https://www.apple.com.cn/shop/buy-iphone/iphone-13-pro/MLTE3CH/A 在上面的链接中选择需要的型号、颜色、容量,然后把地址中类似MLDH3CH/A的文本复制下来,就是需要监控的型号 +AppleWatch + +https://www.apple.com.cn/shop/buy-watch/apple-watch/45mm-cellular-graphite-stainless-steel-pride-edition-braided-solo-loop-size-5 + +AppleWatch因为有选项,所以会稍微麻烦,需要在浏览器的“开发人员工具”中,找到“网络”,在“筛选器”中输入`https://www.apple.com.cn/shop/fulfillment-messages`,找到网络请求,将`parts.0=xxxxx`和`option.0=xxxxxxxxxxx`中xxxxx部分复制出来保存。 + +将两个拼接在一起,如 `Z0YQ#MKMR3CH/A,MJXA3FE/A`,就是需要监控的型号。 + +AppleWatch目前没办法自动获取标题,所以可以自定义标题,如`Z0YQ#MKMR3CH/A,MJXA3FE/A#AppleWatch 7 石墨色不锈钢`,把这部分输入到BoxJs的监控型号中。 + ### 确认地区 省市示例:吉林 长春 朝阳区 @@ -62,8 +72,6 @@ https://www.apple.com.cn/shop/buy-iphone/iphone-13-pro/MLTE3CH/A 表示的是每家店铺的具体情况,小太阳表示有货,禁止符号表示售罄,紧接着是城市和店铺名称。 -~~最后的箭头,↑ 代表上次查询售罄,本次查询有货。↓ 代表上次查询有货,本次查询售罄。○ 表示上次查询售罄,本次查询也售罄。●表示上次查询有货,本次查询也有货。~~这是去年监控iPhone12用的,监控iPhone13后简化了通知,箭头无意义。 - 点击通知可以跳转到Apple Store APP,如果你有安装的话。 ## 配置说明 @@ -90,18 +98,6 @@ https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/app ```ini [task_local] -0/5 * 6-23 * * * https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/applestore/iphone.js, tag=AppleStore_iPhone库存监控, enabled=true +0/5 * 6-23 * * * https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/applestore/iphone.js, tag=AppleStore_商品库存监控, enabled=true ``` -## 暂停维护 - -没错,刚刚更新就暂停维护了,如果没有严重的Bug不会再修改。 - -因为: - -1. 依照往年惯例,iPhone稳定供货后,苹果会关闭库存查询接口,脚本失效 -2. 每年库存查询接口都会有很大的变化,每年都需要重写 -3. ~~绝对不是因为我已经首发买到iPhone13 Pro Max~~ - -明年 iPhone14/13S 再见! - diff --git a/script/applestore/applestore.js b/script/applestore/applestore.js new file mode 100644 index 000000000..b704a3bbf --- /dev/null +++ b/script/applestore/applestore.js @@ -0,0 +1,155 @@ +const SCRIPT_NAME = "AppleStore"; +const APPLESTORE_MODEL_KEY = "goods_model"; +const APPLESTORE_REGION_KEY = "applestore_region"; +const APPLESTORE_STOCK_KEY = "goods_stock"; + +let magicJS = MagicJS(SCRIPT_NAME); +magicJS.barkUrl = magicJS.read("applestore_bark_url") || magicJS.read("magicjs_bark_url"); + +function getGoodsStock(parts, location, option = "") { + return new Promise((resolve) => { + let url = ""; + if (option != "") { + url = encodeURI(`https://www.apple.com.cn/shop/fulfillment-messages?pl=true&mt=compact&parts.0=${parts}&location=${location}&option.0=${option}&_=${new Date().getTime()}`); + } else { + url = encodeURI(`https://www.apple.com.cn/shop/fulfillment-messages?pl=true&mt=compact&parts.0=${parts}&location=${location}&_=${new Date().getTime()}`); + } + magicJS.get(url, (err, resp, data) => { + let obj = JSON.parse(data); + let stores = obj["body"]["content"]["pickupMessage"]["stores"]; + if (stores) { + resolve(stores); + } else { + magicJS.notify("查询库存失败,请检查配置是否正确。"); + resolve([]); + } + }); + }); +} + +async function watchStock(goods_models, applestore_region) { + let stock = magicJS.read(APPLESTORE_STOCK_KEY); + stock = !!stock ? stock : {}; + let len = goods_models.length; + + for (let i = 0; i < len; i++) { + let partsConfig = goods_models[i].split("#"); + let parts = partsConfig[0]; + let option = partsConfig.length >= 2 ? partsConfig[1] : ""; + let name = partsConfig.length == 3 ? partsConfig[2] : ""; + let subObj = { watch: 0, pickup: 0, soldout: 0, changed: 0 }; + let availability = await getGoodsStock(parts, applestore_region, option); + + if (availability && availability.length > 0) { + // 获取AppleStore取货信息 + for (let store of availability) { + let storeNumber = store["storeNumber"]; + if (!stock.hasOwnProperty(parts)) { + stock[parts] = { title: store["partsAvailability"][parts]["storePickupProductTitle"], stores: {} }; + } + if (!stock[parts]["stores"][storeNumber]) { + stock[parts]["stores"][storeNumber] = { notify: false, pickup: false, msg: "等待查询", city: store["city"], name: store["storeName"] }; + } + if (stock[parts]["stores"][storeNumber]["msg"] != store["partsAvailability"][parts]["pickupSearchQuote"]) { + // 更新库存情况 + stock[parts]["stores"][storeNumber]["msg"] = store["partsAvailability"][parts]["pickupSearchQuote"]; + stock[parts]["stores"][storeNumber]["pickup"] = store["partsAvailability"][parts]["pickupDisplay"] != "unavailable"; + // 库存变化推送通知 + stock[parts]["stores"][storeNumber]["notify"] = true; + } else { + // 库存未变化不推送 + stock[parts]["stores"][storeNumber]["notify"] = false; + } + } + + let now = new Date(); + if (!stock[parts]["title"] && !name) { + name = "未命名商品"; + } + let logStr = `${name}\n`; + let title = `${name} - ${now.getHours()}:${now.getMinutes()}:${now.getSeconds()}`; + let watchResult = "全部售罄"; + let stockInContent = ""; // 有库存的型号与店铺 + let soldOutContent = ""; // 售罄的型号与店铺 + let unchangContent = ""; // 没有变化的型号与店铺 + let content = ""; + + // 整理通知内容 + for (let storeStock of Object.values(stock[parts]["stores"])) { + subObj["watch"] += 1; + // 有货 + if (storeStock["pickup"]) { + subObj["pickup"] += 1; + if (watchResult == "全部售罄") watchResult = `${storeStock["city"]} ${storeStock["name"]}`; + if (storeStock["notify"] === true) { + subObj["changed"] += 1; + if (!!stockInContent) stockInContent += "\n"; + stockInContent += `🔆 ${storeStock["name"]} - ${storeStock["msg"]}↑`; + } else { + if (!!unchangContent) unchangContent += "\n"; + unchangContent += `🔆 ${storeStock["name"]} - ${storeStock["msg"]}● `; + } + logStr += `${storeStock["name"]} - ${storeStock["msg"]}\n`; + } + + // 售罄 + else { + subObj["soldout"] += 1; + if (storeStock["notify"] === true) { + subObj["changed"] += 1; + if (!!soldOutContent) soldOutContent += "\n"; + soldOutContent += `🚫 ${storeStock["name"]} - ${storeStock["msg"]}↓`; + } else { + if (!!unchangContent) unchangContent += "\n"; + unchangContent += `🚫 ${storeStock["name"]} - ${storeStock["msg"]}○`; + } + logStr += `${storeStock["name"]} - ${storeStock["msg"]}\n`; + } + } + if (!!stockInContent) { + content = stockInContent; + } + // 配置为无货通知且存在无货情况时 + if (magicJS.read("applestore_settings_notify_soldout") == true) { + content = !!stockInContent ? stockInContent + `\n${soldOutContent}\n${unchangContent}` : !!soldOutContent ? `${soldOutContent}\n${unchangContent}` : unchangContent; + } + if (!!content) { + let subTitle = `监控: ${subObj.watch} 售罄: ${subObj.soldout} 有货: ${subObj.pickup} ${watchResult}`; + magicJS.notify(title, subTitle, content, "applestore://"); + } + magicJS.logInfo(logStr); + } + } + + // 存储本次库存检查结果 + magicJS.write(APPLESTORE_STOCK_KEY, stock); +} + +(async () => { + let goods_model = magicJS.read(APPLESTORE_MODEL_KEY).trim(); + let applestore_region = magicJS.read(APPLESTORE_REGION_KEY).trim(); + + if (!goods_model || !applestore_region) { + let msg = "请先在BoxJS中配置心仪的商品型号及购买地区"; + magicJS.logWarning(msg); + magicJS.notify(msg); + return; + } + + let goods_models = goods_model.split(";"); + + // 监控库存 + await watchStock(goods_models, applestore_region); + + if (magicJS.isNode) { + while (6 <= new Date().getHours() <= 23) { + await watchStock(goods_models, applestore_region); + await magicJS.sleep(3000); + } + } + + magicJS.done(); +})(); + +// prettier-ignore +function MagicJS(scriptName="MagicJS",logLevel="INFO"){return new class{constructor(){if(this._startTime=Date.now(),this.version="2.2.3.6",this.scriptName=scriptName,this.logLevels={DEBUG:5,INFO:4,NOTIFY:3,WARNING:2,ERROR:1,CRITICAL:0,NONE:-1},this.isLoon="undefined"!=typeof $loon,this.isQuanX="undefined"!=typeof $task,this.isJSBox="undefined"!=typeof $drive,this.isNode="undefined"!=typeof module&&!this.isJSBox,this.isSurge="undefined"!=typeof $httpClient&&!this.isLoon,this.node={request:void 0,fs:void 0,data:{}},this.iOSUserAgent="Mozilla/5.0 (iPhone; CPU iPhone OS 13_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/13.0.5 Mobile/15E148 Safari/604.1",this.pcUserAgent="Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/84.0.4147.125 Safari/537.36 Edg/84.0.522.59",this._logLevel="INFO",this.logLevel=logLevel,this._barkUrl="",this._barkKey="",this.isNode){this.node.fs=require("fs"),this.node.request=require("request");try{this.node.fs.accessSync("./magic.json",this.node.fs.constants.R_OK|this.node.fs.constants.W_OK)}catch(err){this.node.fs.writeFileSync("./magic.json","{}",{encoding:"utf8"})}this.node.data=require("./magic.json")}else this.isJSBox&&($file.exists("drive://MagicJS")||$file.mkdir("drive://MagicJS"),$file.exists("drive://MagicJS/magic.json")||$file.write({data:$data({string:"{}"}),path:"drive://MagicJS/magic.json"}))}set barkUrl(url){try{let _url=url.replace(/\/+$/g,"");this._barkUrl=`${/^https?:\/\/([^/]*)/.exec(_url)[0]}/push`,this._barkKey=/\/([^\/]+)\/?$/.exec(_url)[1]}catch(err){this.logDebug("Bark config error.")}}set logLevel(level){let magic_loglevel=this.read("magicjs_loglevel");this._logLevel=magic_loglevel||level.toUpperCase()}get logLevel(){return this._logLevel}get isRequest(){return"undefined"!=typeof $request&&"undefined"==typeof $response}get isResponse(){return"undefined"!=typeof $response}get isDebug(){return"DEBUG"===this.logLevel}get request(){return"undefined"!=typeof $request?$request:void 0}get response(){return"undefined"!=typeof $response?($response.hasOwnProperty("status")&&($response.statusCode=$response.status),$response.hasOwnProperty("statusCode")&&($response.status=$response.statusCode),$response):void 0}get platform(){return this.isSurge?"Surge":this.isQuanX?"Quantumult X":this.isLoon?"Loon":this.isJSBox?"JSBox":this.isNode?"Node.js":"Unknown"}read(key,session=""){let val="";this.isSurge||this.isLoon?val=$persistentStore.read(key):this.isQuanX?val=$prefs.valueForKey(key):this.isNode?val=this.node.data:this.isJSBox&&(val=$file.read("drive://MagicJS/magic.json").string);try{this.isNode&&(val=val[key]),this.isJSBox&&(val=JSON.parse(val)[key]),session&&("string"==typeof val&&(val=JSON.parse(val)),val=val&&"object"==typeof val?val[session]:null)}catch(err){this.logError(err),val=session?{}:null,this.del(key)}void 0===val&&(val=null);try{val&&"string"==typeof val&&(val=JSON.parse(val))}catch(err){}return this.logDebug(`READ DATA [${key}]${session?`[${session}]`:""}(${typeof val})\n${JSON.stringify(val)}`),val}write(key,val,session=""){let data=session?{}:"";if(session&&(this.isSurge||this.isLoon)?data=$persistentStore.read(key):session&&this.isQuanX?data=$prefs.valueForKey(key):this.isNode?data=this.node.data:this.isJSBox&&(data=JSON.parse($file.read("drive://MagicJS/magic.json").string)),session){try{"string"==typeof data&&(data=JSON.parse(data)),data="object"==typeof data&&data?data:{}}catch(err){this.logError(err),this.del(key),data={}}this.isJSBox||this.isNode?(data[key]&&"object"==typeof data[key]||(data[key]={}),data[key].hasOwnProperty(session)||(data[key][session]=null),void 0===val?delete data[key][session]:data[key][session]=val):void 0===val?delete data[session]:data[session]=val}else this.isNode||this.isJSBox?void 0===val?delete data[key]:data[key]=val:data=void 0===val?null:val;"object"==typeof data&&(data=JSON.stringify(data)),this.isSurge||this.isLoon?$persistentStore.write(data,key):this.isQuanX?$prefs.setValueForKey(data,key):this.isNode?this.node.fs.writeFileSync("./magic.json",data):this.isJSBox&&$file.write({data:$data({string:data}),path:"drive://MagicJS/magic.json"}),this.logDebug(`WRITE DATA [${key}]${session?`[${session}]`:""}(${typeof val})\n${JSON.stringify(val)}`)}del(key,session=""){this.logDebug(`DELETE KEY [${key}]${session?`[${session}]`:""}`),this.write(key,null,session)}notify(title=this.scriptName,subTitle="",body="",opts=""){let convertOptions;if(opts=(_opts=>{let newOpts={};if("string"==typeof _opts)this.isLoon?newOpts={openUrl:_opts}:this.isQuanX?newOpts={"open-url":_opts}:this.isSurge&&(newOpts={url:_opts});else if("object"==typeof _opts)if(this.isLoon)newOpts.openUrl=_opts["open-url"]?_opts["open-url"]:"",newOpts.mediaUrl=_opts["media-url"]?_opts["media-url"]:"";else if(this.isQuanX)newOpts=_opts["open-url"]||_opts["media-url"]?_opts:{};else if(this.isSurge){let openUrl=_opts["open-url"]||_opts.openUrl;newOpts=openUrl?{url:openUrl}:{}}return newOpts})(opts),1==arguments.length&&(title=this.scriptName,subTitle="",body=arguments[0]),this.logNotify(`title:${title}\nsubTitle:${subTitle}\nbody:${body}\noptions:${"object"==typeof opts?JSON.stringify(opts):opts}`),this.isSurge)$notification.post(title,subTitle,body,opts);else if(this.isLoon)opts?$notification.post(title,subTitle,body,opts):$notification.post(title,subTitle,body);else if(this.isQuanX)$notify(title,subTitle,body,opts);else if(this.isJSBox){let push={title:title,body:subTitle?`${subTitle}\n${body}`:body};$push.schedule(push)}this._barkUrl&&this._barkKey&&this.notifyBark(title,subTitle,body)}notifyDebug(title=this.scriptName,subTitle="",body="",opts=""){"DEBUG"===this.logLevel&&(1==arguments.length&&(title=this.scriptName,subTitle="",body=arguments[0]),this.notify(title,subTitle,body,opts))}notifyBark(title=this.scriptName,subTitle="",body="",opts=""){let options={url:this._barkUrl,headers:{"Content-Type":"application/json; charset=utf-8"},body:{title:title,body:subTitle?`${subTitle}\n${body}`:body,device_key:this._barkKey}};this.post(options,err=>{})}log(msg,level="INFO"){this.logLevels[this._logLevel]void 0===_options.body?"":`${encodeURIComponent(key)}=${encodeURIComponent(_options.body[key])}`).join("&");_options.url.indexOf("?")<0&&(_options.url+="?"),_options.url.lastIndexOf("&")+1!=_options.url.length&&_options.url.lastIndexOf("?")+1!=_options.url.length&&(_options.url+="&"),_options.url+=qs,delete _options.body}return this.isQuanX?(_options.hasOwnProperty("body")&&"string"!=typeof _options.body&&(_options.body=JSON.stringify(_options.body)),_options.method=method):this.isNode?(delete _options.headers["Accept-Encoding"],"object"==typeof _options.body&&("GET"===method?(_options.qs=_options.body,delete _options.body):"POST"===method&&(_options.json=!0,_options.body=_options.body))):this.isJSBox&&(_options.header=_options.headers,delete _options.headers),_options}adapterHttpResponse(resp){let _resp={body:resp.body,headers:resp.headers,json:()=>JSON.parse(_resp.body)};return resp.hasOwnProperty("statusCode")&&resp.statusCode&&(_resp.status=resp.statusCode),_resp}get(options,callback){let _options=this.adapterHttpOptions(options,"GET");this.logDebug(`HTTP GET: ${JSON.stringify(_options)}`),this.isSurge||this.isLoon?$httpClient.get(_options,callback):this.isQuanX?$task.fetch(_options).then(resp=>{resp.status=resp.statusCode,callback(null,resp,resp.body)},reason=>callback(reason.error,null,null)):this.isNode?this.node.request.get(_options,(err,resp,data)=>{resp=this.adapterHttpResponse(resp),callback(err,resp,data)}):this.isJSBox&&(_options.handler=resp=>{let err=resp.error?JSON.stringify(resp.error):void 0,data="object"==typeof resp.data?JSON.stringify(resp.data):resp.data;callback(err,resp.response,data)},$http.get(_options))}getPromise(options){return new Promise((resolve,reject)=>{magicJS.get(options,(err,resp)=>{err?reject(err):resolve(resp)})})}post(options,callback){let _options=this.adapterHttpOptions(options,"POST");if(this.logDebug(`HTTP POST: ${JSON.stringify(_options)}`),this.isSurge||this.isLoon)$httpClient.post(_options,callback);else if(this.isQuanX)$task.fetch(_options).then(resp=>{resp.status=resp.statusCode,callback(null,resp,resp.body)},reason=>{callback(reason.error,null,null)});else if(this.isNode){let resp=this.node.request.post(_options,callback);resp.status=resp.statusCode,delete resp.statusCode}else this.isJSBox&&(_options.handler=resp=>{let err=resp.error?JSON.stringify(resp.error):void 0,data="object"==typeof resp.data?JSON.stringify(resp.data):resp.data;callback(err,resp.response,data)},$http.post(_options,{}))}done(value={}){this._endTime=Date.now();let span=(this._endTime-this._startTime)/1e3;magicJS.logDebug(`SCRIPT COMPLETED: ${span}S.`),"undefined"!=typeof $done&&$done(value)}isToday(day){if(null==day)return!1;{let today=new Date;return"string"==typeof day&&(day=new Date(day)),today.getFullYear()==day.getFullYear()&&today.getMonth()==day.getMonth()&&today.getDay()==day.getDay()}}isNumber(val){return"NaN"!==parseFloat(val).toString()}attempt(promise,defaultValue=null){return promise.then(args=>[null,args]).catch(ex=>(this.logError(ex),[ex,defaultValue]))}retry(fn,retries=5,interval=0,callback=null){return(...args)=>new Promise((resolve,reject)=>{function _retry(...args){Promise.resolve().then(()=>fn.apply(this,args)).then(result=>{"function"==typeof callback?Promise.resolve().then(()=>callback(result)).then(()=>{resolve(result)}).catch(ex=>{retries>=1?interval>0?setTimeout(()=>_retry.apply(this,args),interval):_retry.apply(this,args):reject(ex),retries--}):resolve(result)}).catch(ex=>{this.logRetry(ex),retries>=1&&interval>0?setTimeout(()=>_retry.apply(this,args),interval):retries>=1?_retry.apply(this,args):reject(ex),retries--})}_retry.apply(this,args)})}formatTime(time,fmt="yyyy-MM-dd hh:mm:ss"){var o={"M+":time.getMonth()+1,"d+":time.getDate(),"h+":time.getHours(),"m+":time.getMinutes(),"s+":time.getSeconds(),"q+":Math.floor((time.getMonth()+3)/3),S:time.getMilliseconds()};/(y+)/.test(fmt)&&(fmt=fmt.replace(RegExp.$1,(time.getFullYear()+"").substr(4-RegExp.$1.length)));for(let k in o)new RegExp("("+k+")").test(fmt)&&(fmt=fmt.replace(RegExp.$1,1==RegExp.$1.length?o[k]:("00"+o[k]).substr((""+o[k]).length)));return fmt}now(){return this.formatTime(new Date,"yyyy-MM-dd hh:mm:ss")}today(){return this.formatTime(new Date,"yyyy-MM-dd")}sleep(time){return new Promise(resolve=>setTimeout(resolve,time))}}(scriptName)} diff --git a/script/applestore/applestore.lnscript b/script/applestore/applestore.lnscript new file mode 100644 index 000000000..1e8b80280 --- /dev/null +++ b/script/applestore/applestore.lnscript @@ -0,0 +1 @@ +cron "0/5 * 6-23 * * *" script-path=https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/applestore/applestore.js,timeout=5,tag=AppleStore_查询商品库存 \ No newline at end of file diff --git a/script/applestore/applestore.sgmodule b/script/applestore/applestore.sgmodule new file mode 100644 index 000000000..746d04dfb --- /dev/null +++ b/script/applestore/applestore.sgmodule @@ -0,0 +1,6 @@ +#!name=AppleStore +#!desc=定时查询AppleStore商品库存 +#!system=ios + +[Script] +AppleStore_查询商品库存 = type=cron,cronexp=0/5 * 6-23 * * *,script-path=https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/applestore/applestore.js \ No newline at end of file diff --git a/script/applestore/iphone.js b/script/applestore/iphone.js deleted file mode 100644 index d59008b86..000000000 --- a/script/applestore/iphone.js +++ /dev/null @@ -1,145 +0,0 @@ -const SCRIPT_NAME = "AppleStore"; -const IPHONE_MODEL_KEY = "iphone_model"; -const APPLE_STORE_REGION_KEY = "apple_store_region"; -const IPHONE_STOCK_KEY = "iphone_stock"; -const NOTIFY_SOLD_OUT = false; - -let magicJS = MagicJS(SCRIPT_NAME); -magicJS.barkUrl = magicJS.read("applestore_bark_url") || magicJS.read("magicjs_bark_url"); - -function getiPhoneProStock(parts, location) { - return new Promise((resolve) => { - url = encodeURI(`https://www.apple.com.cn/shop/fulfillment-messages?pl=true&mt=compact&parts.0=${parts}&location=${location}`); - magicJS.get(url, (err, resp, data) => { - let obj = JSON.parse(data); - let stores = obj["body"]["content"]["pickupMessage"]["stores"]; - if (stores) { - resolve(stores); - } else { - magicJS.notify("查询库存失败,请检查配置是否正确。"); - resolve([]); - } - }); - }); -} - -async function watchStock(iphone_models, apple_store_region) { - let stock = magicJS.read(IPHONE_STOCK_KEY); - stock = !!stock ? stock : {}; - let len = iphone_models.length; - - for (let i = 0; i < len; i++) { - let parts = iphone_models[i]; - let subObj = { watch: 0, pickup: 0, soldout: 0, changed: 0 }; - let availability = await getiPhoneProStock(parts, apple_store_region); - - if (availability && availability.length > 0) { - // 获取AppleStore取货信息 - for (let store of availability) { - let storeNumber = store["storeNumber"]; - if (!stock.hasOwnProperty(parts)) { - stock[parts] = { title: store["partsAvailability"][parts]["storePickupProductTitle"], stores: {} }; - } - if (!stock[parts]["stores"][storeNumber]) { - stock[parts]["stores"][storeNumber] = { notify: false, pickup: false, msg: "等待查询", city: store["city"], name: store["storeName"] }; - } - if (stock[parts]["stores"][storeNumber]["msg"] != store["partsAvailability"][parts]["pickupSearchQuote"]) { - // 更新库存情况 - stock[parts]["stores"][storeNumber]["msg"] = store["partsAvailability"][parts]["pickupSearchQuote"]; - stock[parts]["stores"][storeNumber]["pickup"] = store["partsAvailability"][parts]["pickupDisplay"] != "unavailable"; - // 库存变化推送通知 - stock[parts]["stores"][storeNumber]["notify"] = true; - } else { - // 库存未变化不推送 - stock[parts]["stores"][storeNumber]["notify"] = false; - } - } - - let now = new Date(); - let logStr = `${stock[parts]["title"]}\n`; - let title = `${stock[parts]["title"]} - ${now.getHours()}:${now.getMinutes()}:${now.getSeconds()}`; - let watchResult = "全部售罄"; - let stockInContent = ""; // 有库存的型号与店铺 - let soldOutContent = ""; // 售罄的型号与店铺 - let unchangContent = ""; // 没有变化的型号与店铺 - let content = ""; - - // 整理通知内容 - for (let storeStock of Object.values(stock[parts]["stores"])) { - subObj["watch"] += 1; - // 有货 - if (storeStock["pickup"]) { - subObj["pickup"] += 1; - if (watchResult == "全部售罄") watchResult = `${storeStock["city"]} ${storeStock["name"]}`; - if (storeStock["notify"] === true) { - subObj["changed"] += 1; - if (!!stockInContent) stockInContent += "\n"; - stockInContent += `🔆 ${storeStock["name"]} - ${storeStock["msg"]}↑`; - } else { - if (!!unchangContent) unchangContent += "\n"; - unchangContent += `🔆 ${storeStock["name"]} - ${storeStock["msg"]}● `; - } - logStr += `${storeStock["name"]} - ${storeStock["msg"]}\n`; - } - - // 售罄 - else { - subObj["soldout"] += 1; - if (storeStock["notify"] === true) { - subObj["changed"] += 1; - if (!!soldOutContent) soldOutContent += "\n"; - soldOutContent += `🚫 ${storeStock["name"]} - ${storeStock["msg"]}↓`; - } else { - if (!!unchangContent) unchangContent += "\n"; - unchangContent += `🚫 ${storeStock["name"]} - ${storeStock["msg"]}○`; - } - logStr += `${storeStock["name"]} - ${storeStock["msg"]}\n`; - } - } - if (!!stockInContent) { - content = stockInContent; - } - // 配置为无货通知且存在无货情况时 - if (NOTIFY_SOLD_OUT) { - content = !!stockInContent ? stockInContent + `\n${soldOutContent}\n${unchangContent}` : `${soldOutContent}\n${unchangContent}`; - } - if (!!content) { - let subTitle = `监控: ${subObj.watch} 售罄: ${subObj.soldout} 有货: ${subObj.pickup} ${watchResult}`; - magicJS.notify(title, subTitle, content, "applestore://"); - } - magicJS.logInfo(logStr); - } - } - - // 存储本次库存检查结果 - magicJS.write(IPHONE_STOCK_KEY, stock); -} - -(async () => { - let iphone_model = magicJS.read(IPHONE_MODEL_KEY).trim(); - let apple_store_region = magicJS.read(APPLE_STORE_REGION_KEY).trim(); - - if (!iphone_model || !apple_store_region) { - let msg = "请先在BoxJS中配置心仪的iPhne型号及购买地区"; - magicJS.logWarning(msg); - magicJS.notify(msg); - return; - } - - let iphone_models = iphone_model.split(";"); - - // 监控库存 - await watchStock(iphone_models, apple_store_region); - - if (magicJS.isNode) { - while (6 <= new Date().getHours() <= 23) { - await watchStock(iphone_models, apple_store_region); - await magicJS.sleep(3000); - } - } - - magicJS.done(); -})(); - -// prettier-ignore -function MagicJS(scriptName="MagicJS",logLevel="INFO"){return new class{constructor(){if(this.version="2.2.3.5",this.scriptName=scriptName,this.logLevels={DEBUG:5,INFO:4,NOTIFY:3,WARNING:2,ERROR:1,CRITICAL:0,NONE:-1},this.isLoon="undefined"!=typeof $loon,this.isQuanX="undefined"!=typeof $task,this.isJSBox="undefined"!=typeof $drive,this.isNode="undefined"!=typeof module&&!this.isJSBox,this.isSurge="undefined"!=typeof $httpClient&&!this.isLoon,this.node={request:void 0,fs:void 0,data:{}},this.iOSUserAgent="Mozilla/5.0 (iPhone; CPU iPhone OS 13_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/13.0.5 Mobile/15E148 Safari/604.1",this.pcUserAgent="Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/84.0.4147.125 Safari/537.36 Edg/84.0.522.59",this.logLevel=logLevel,this._barkUrl="",this._barkKey="",this.isNode){this.node.fs=require("fs"),this.node.request=require("request");try{this.node.fs.accessSync("./magic.json",this.node.fs.constants.R_OK|this.node.fs.constants.W_OK)}catch(err){this.node.fs.writeFileSync("./magic.json","{}",{encoding:"utf8"})}this.node.data=require("./magic.json")}else this.isJSBox&&($file.exists("drive://MagicJS")||$file.mkdir("drive://MagicJS"),$file.exists("drive://MagicJS/magic.json")||$file.write({data:$data({string:"{}"}),path:"drive://MagicJS/magic.json"}))}set barkUrl(url){try{let _url=url.replace(/\/+$/g,"");this._barkUrl=`${/^https?:\/\/([^/]*)/.exec(_url)[0]}/push`,this._barkKey=/\/([^\/]+)\/?$/.exec(_url)[1]}catch(err){this.logDebug("读取Bark推送链接失败。")}}set logLevel(level){this._logLevel="string"==typeof level?level.toUpperCase():"DEBUG"}get logLevel(){return this._logLevel}get isRequest(){return"undefined"!=typeof $request&&"undefined"==typeof $response}get isResponse(){return"undefined"!=typeof $response}get request(){return"undefined"!=typeof $request?$request:void 0}get response(){return"undefined"!=typeof $response?($response.hasOwnProperty("status")&&($response.statusCode=$response.status),$response.hasOwnProperty("statusCode")&&($response.status=$response.statusCode),$response):void 0}get platform(){return this.isSurge?"Surge":this.isQuanX?"Quantumult X":this.isLoon?"Loon":this.isJSBox?"JSBox":this.isNode?"Node.js":"Unknown"}read(key,session=""){let val="";this.isSurge||this.isLoon?val=$persistentStore.read(key):this.isQuanX?val=$prefs.valueForKey(key):this.isNode?val=this.node.data:this.isJSBox&&(val=$file.read("drive://MagicJS/magic.json").string);try{this.isNode&&(val=val[key]),this.isJSBox&&(val=JSON.parse(val)[key]),session&&("string"==typeof val&&(val=JSON.parse(val)),val=val&&"object"==typeof val?val[session]:null)}catch(err){this.logError(err),val=session?{}:null,this.del(key)}void 0===val&&(val=null);try{val&&"string"==typeof val&&(val=JSON.parse(val))}catch(err){}return this.logDebug(`READ DATA [${key}]${session?`[${session}]`:""}(${typeof val})\n${JSON.stringify(val)}`),val}write(key,val,session=""){let data=session?{}:"";if(session&&(this.isSurge||this.isLoon)?data=$persistentStore.read(key):session&&this.isQuanX?data=$prefs.valueForKey(key):this.isNode?data=this.node.data:this.isJSBox&&(data=JSON.parse($file.read("drive://MagicJS/magic.json").string)),session){try{"string"==typeof data&&(data=JSON.parse(data)),data="object"==typeof data&&data?data:{}}catch(err){this.logError(err),this.del(key),data={}}this.isJSBox||this.isNode?(data[key]&&"object"==typeof data[key]||(data[key]={}),data[key].hasOwnProperty(session)||(data[key][session]=null),void 0===val?delete data[key][session]:data[key][session]=val):void 0===val?delete data[session]:data[session]=val}else this.isNode||this.isJSBox?void 0===val?delete data[key]:data[key]=val:data=void 0===val?null:val;"object"==typeof data&&(data=JSON.stringify(data)),this.isSurge||this.isLoon?$persistentStore.write(data,key):this.isQuanX?$prefs.setValueForKey(data,key):this.isNode?this.node.fs.writeFileSync("./magic.json",data):this.isJSBox&&$file.write({data:$data({string:data}),path:"drive://MagicJS/magic.json"}),this.logDebug(`WRITE DATA [${key}]${session?`[${session}]`:""}(${typeof val})\n${JSON.stringify(val)}`)}del(key,session=""){this.logDebug(`DELETE KEY [${key}]${session?`[${session}]`:""}`),this.write(key,null,session)}notify(title=this.scriptName,subTitle="",body="",opts=""){let convertOptions;if(opts=(_opts=>{let newOpts={};if("string"==typeof _opts)this.isLoon?newOpts={openUrl:_opts}:this.isQuanX?newOpts={"open-url":_opts}:this.isSurge&&(newOpts={url:_opts});else if("object"==typeof _opts)if(this.isLoon)newOpts.openUrl=_opts["open-url"]?_opts["open-url"]:"",newOpts.mediaUrl=_opts["media-url"]?_opts["media-url"]:"";else if(this.isQuanX)newOpts=_opts["open-url"]||_opts["media-url"]?_opts:{};else if(this.isSurge){let openUrl=_opts["open-url"]||_opts.openUrl;newOpts=openUrl?{url:openUrl}:{}}return newOpts})(opts),1==arguments.length&&(title=this.scriptName,subTitle="",body=arguments[0]),this.logNotify(`title:${title}\nsubTitle:${subTitle}\nbody:${body}\noptions:${"object"==typeof opts?JSON.stringify(opts):opts}`),this.isSurge)$notification.post(title,subTitle,body,opts);else if(this.isLoon)opts?$notification.post(title,subTitle,body,opts):$notification.post(title,subTitle,body);else if(this.isQuanX)$notify(title,subTitle,body,opts);else if(this.isJSBox){let push={title:title,body:subTitle?`${subTitle}\n${body}`:body};$push.schedule(push)}this._barkUrl&&this._barkKey&&this.notifyBark(title,subTitle,body)}notifyDebug(title=this.scriptName,subTitle="",body="",opts=""){"DEBUG"===this.logLevel&&(1==arguments.length&&(title=this.scriptName,subTitle="",body=arguments[0]),this.notify(title,subTitle,body,opts))}notifyBark(title=this.scriptName,subTitle="",body="",opts=""){let options={url:this._barkUrl,headers:{"Content-Type":"application/json; charset=utf-8"},body:{title:title,body:subTitle?`${subTitle}\n${body}`:body,device_key:this._barkKey}};this.post(options,err=>{})}log(msg,level="INFO"){this.logLevels[this._logLevel]void 0===_options.body?"":`${encodeURIComponent(key)}=${encodeURIComponent(_options.body[key])}`).join("&");_options.url.indexOf("?")<0&&(_options.url+="?"),_options.url.lastIndexOf("&")+1!=_options.url.length&&_options.url.lastIndexOf("?")+1!=_options.url.length&&(_options.url+="&"),_options.url+=qs,delete _options.body}return this.isQuanX?(_options.hasOwnProperty("body")&&"string"!=typeof _options.body&&(_options.body=JSON.stringify(_options.body)),_options.method=method):this.isNode?(delete _options.headers["Accept-Encoding"],"object"==typeof _options.body&&("GET"===method?(_options.qs=_options.body,delete _options.body):"POST"===method&&(_options.json=!0,_options.body=_options.body))):this.isJSBox&&(_options.header=_options.headers,delete _options.headers),_options}adapterHttpResponse(resp){let _resp={body:resp.body,headers:resp.headers,json:()=>JSON.parse(_resp.body)};return resp.hasOwnProperty("statusCode")&&resp.statusCode&&(_resp.status=resp.statusCode),_resp}get(options,callback){let _options=this.adapterHttpOptions(options,"GET");this.logDebug(`HTTP GET: ${JSON.stringify(_options)}`),this.isSurge||this.isLoon?$httpClient.get(_options,callback):this.isQuanX?$task.fetch(_options).then(resp=>{resp.status=resp.statusCode,callback(null,resp,resp.body)},reason=>callback(reason.error,null,null)):this.isNode?this.node.request.get(_options,(err,resp,data)=>{resp=this.adapterHttpResponse(resp),callback(err,resp,data)}):this.isJSBox&&(_options.handler=resp=>{let err=resp.error?JSON.stringify(resp.error):void 0,data="object"==typeof resp.data?JSON.stringify(resp.data):resp.data;callback(err,resp.response,data)},$http.get(_options))}getPromise(options){return new Promise((resolve,reject)=>{magicJS.get(options,(err,resp)=>{err?reject(err):resolve(resp)})})}post(options,callback){let _options=this.adapterHttpOptions(options,"POST");if(this.logDebug(`HTTP POST: ${JSON.stringify(_options)}`),this.isSurge||this.isLoon)$httpClient.post(_options,callback);else if(this.isQuanX)$task.fetch(_options).then(resp=>{resp.status=resp.statusCode,callback(null,resp,resp.body)},reason=>{callback(reason.error,null,null)});else if(this.isNode){let resp=this.node.request.post(_options,callback);resp.status=resp.statusCode,delete resp.statusCode}else this.isJSBox&&(_options.handler=resp=>{let err=resp.error?JSON.stringify(resp.error):void 0,data="object"==typeof resp.data?JSON.stringify(resp.data):resp.data;callback(err,resp.response,data)},$http.post(_options,{}))}get http(){return{get:this.getPromise,post:this.post}}done(value={}){"undefined"!=typeof $done&&$done(value)}isToday(day){if(null==day)return!1;{let today=new Date;return"string"==typeof day&&(day=new Date(day)),today.getFullYear()==day.getFullYear()&&today.getMonth()==day.getMonth()&&today.getDay()==day.getDay()}}isNumber(val){return"NaN"!==parseFloat(val).toString()}attempt(promise,defaultValue=null){return promise.then(args=>[null,args]).catch(ex=>(this.logError(ex),[ex,defaultValue]))}retry(fn,retries=5,interval=0,callback=null){return(...args)=>new Promise((resolve,reject)=>{function _retry(...args){Promise.resolve().then(()=>fn.apply(this,args)).then(result=>{"function"==typeof callback?Promise.resolve().then(()=>callback(result)).then(()=>{resolve(result)}).catch(ex=>{retries>=1?interval>0?setTimeout(()=>_retry.apply(this,args),interval):_retry.apply(this,args):reject(ex),retries--}):resolve(result)}).catch(ex=>{this.logRetry(ex),retries>=1&&interval>0?setTimeout(()=>_retry.apply(this,args),interval):retries>=1?_retry.apply(this,args):reject(ex),retries--})}_retry.apply(this,args)})}formatTime(time,fmt="yyyy-MM-dd hh:mm:ss"){var o={"M+":time.getMonth()+1,"d+":time.getDate(),"h+":time.getHours(),"m+":time.getMinutes(),"s+":time.getSeconds(),"q+":Math.floor((time.getMonth()+3)/3),S:time.getMilliseconds()};/(y+)/.test(fmt)&&(fmt=fmt.replace(RegExp.$1,(time.getFullYear()+"").substr(4-RegExp.$1.length)));for(let k in o)new RegExp("("+k+")").test(fmt)&&(fmt=fmt.replace(RegExp.$1,1==RegExp.$1.length?o[k]:("00"+o[k]).substr((""+o[k]).length)));return fmt}now(){return this.formatTime(new Date,"yyyy-MM-dd hh:mm:ss")}today(){return this.formatTime(new Date,"yyyy-MM-dd")}sleep(time){return new Promise(resolve=>setTimeout(resolve,time))}}(scriptName)} diff --git a/script/applestore/iphone.lnplugin b/script/applestore/iphone.lnplugin deleted file mode 100644 index e28114fa7..000000000 --- a/script/applestore/iphone.lnplugin +++ /dev/null @@ -1,6 +0,0 @@ -#!name=Apple Store -#!desc=定时获取Apple Store iPhone库存 -#!system=ios - -[Script] -Apple_获取iPhone库存 = type=cron,cronexp=0/5 * 6-23 * * *,script-path=https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/applestore/iphone.js diff --git a/script/applestore/iphone.qxrewrite b/script/applestore/iphone.qxrewrite deleted file mode 100644 index e28114fa7..000000000 --- a/script/applestore/iphone.qxrewrite +++ /dev/null @@ -1,6 +0,0 @@ -#!name=Apple Store -#!desc=定时获取Apple Store iPhone库存 -#!system=ios - -[Script] -Apple_获取iPhone库存 = type=cron,cronexp=0/5 * 6-23 * * *,script-path=https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/applestore/iphone.js diff --git a/script/applestore/iphone.sgmodule b/script/applestore/iphone.sgmodule deleted file mode 100644 index 515a3c745..000000000 --- a/script/applestore/iphone.sgmodule +++ /dev/null @@ -1,6 +0,0 @@ -#!name=AppleStore -#!desc=定时获取AppleStore iPhone库存 -#!system=ios - -[Script] -Apple_获取iPhone库存 = type=cron,cronexp=0/5 * 6-23 * * *,script-path=https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/applestore/iphone.js \ No newline at end of file diff --git a/script/gallery.json b/script/gallery.json index 92c99dfe7..e1bc93e1e 100644 --- a/script/gallery.json +++ b/script/gallery.json @@ -55,6 +55,9 @@ { "config":"30 0 * * * https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/meituan/maicai_checkin.js, tag=美团买菜, img-url=https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/meituan/maicai.png, enabled=true", "addons":"https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/meituan/maicai_checkin.qxrewrite,tag=美团买菜_获取Cookie" + }, + { + "config":"0/5 * 6-23 * * * https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/applestore/applestore.js, tag=库存查询, img-url=https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/applestore/icon/applestore.png, enabled=true" } ], "description":"Script Gallery"