diff --git a/script/synology/163study.lnplugin b/script/synology/163study.lnplugin new file mode 100644 index 00000000000..65781ff4100 --- /dev/null +++ b/script/synology/163study.lnplugin @@ -0,0 +1,8 @@ +# 网易云课堂_离线下载课程内容 + +[Script] +http-response ^https?:\/\/ke\.study\.youdao\.com\/course\/app\/detail.json requires-body=1,script-path=https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/synology/downloadstation.js,tag=网易云课堂_离线下载课程内容 + + +[MITM] +hostname = ke.study.youdao.com \ No newline at end of file diff --git a/script/synology/163study.qxrewrite b/script/synology/163study.qxrewrite new file mode 100644 index 00000000000..6abb9bc9b74 --- /dev/null +++ b/script/synology/163study.qxrewrite @@ -0,0 +1,4 @@ +# 网易云课堂_离线下载课程内容 +^https?:\/\/ke\.study\.youdao\.com\/course\/app\/detail.json url script-response-body https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/synology/downloadstation.js + +hostname = ke.study.youdao.com diff --git a/script/synology/163study.sgmodule b/script/synology/163study.sgmodule new file mode 100644 index 00000000000..519d86f8463 --- /dev/null +++ b/script/synology/163study.sgmodule @@ -0,0 +1,8 @@ +#!name=网易云课堂 +#!desc=网易云课堂离线下载课程视频 + +[Script] +网易云课堂_课程离线下载 = type=http-response,requires-body=1,max-size=0,timeout=300,pattern=^https?:\/\/ke\.study\.youdao\.com\/course\/app\/detail.json,script-path=https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/synology/downloadstation.js + +[MITM] +hostname = %APPEND% ke.study.youdao.com \ No newline at end of file diff --git a/script/synology/README.md b/script/synology/README.md index 44ff666368d..bd1fb24b042 100644 --- a/script/synology/README.md +++ b/script/synology/README.md @@ -6,7 +6,7 @@ 一个小玩具,实现将互联网某些资源添加到群晖的Download Stations下载,目前暂时支持推特第三方客户端的图片视频下载。 -这其实是个架子,提供一种思路:通过脚本获取资源后添加到群晖的Download Station,欢迎有兴趣的大佬一起完善,添加其他功能。~~计划添加自动下载京东电子发票,看什么时候有时间再实现。~~ +这其实是个架子,提供一种思路:通过脚本获取资源后添加到群晖的Download Station,欢迎有兴趣的大佬一起完善,添加其他功能。 在使用脚本前,需要有一些前提条件: @@ -41,6 +41,49 @@ **所以新建一个专用的账户给脚本使用就非常重要了。** +## 网易云课堂课程下载 + +### 操作方式 + +在App中,点击“我的学习”,在“全部课程”中,选择需要下载的课程,点击“进入学习”,就会开始提交下载任务给群晖Download Station。 + +### 注意事项 + +1. 如果课程非常多的情况,可能因为加载超时,导致网易云课堂程序没有响应。如果Surge或其他客户端正常弹出通知,说明执行正常,尽量保持手机不要锁屏,直到弹出通知显示全部课程下载完成。 +2. 不需要下载时务必关闭此模块/插件/复写,否则每次点击“进入学习”都会下载一次课程。 +3. 下载过程中不要修改Download Station的同时下载数。 + +### 没有群晖 + +如果没有群晖,脚本会将获取到的课程视频链接写入到日志中,可以从日志中获取下载链接后,使用其他下载工具下载课程。 + +### 配置说明 + +#### Surge + +使用模块 + +```ini +https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/synology/163study.sgmodule +``` + +#### Loon + +使用插件 + +```ini +https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/synology/163study.lnplugin +``` + +#### Quantumult X + +配置文件 + +```ini +[rewrite_remote] +https://raw.githubusercontent.com/blackmatrix7/ios_rule_script/master/script/synology/163study.qxrewrite, tag=网易云课堂_离线下载课程内容, enabled=true +``` + ## Twitter资源下载 ### Twitter第三方客户端 diff --git a/script/synology/downloadstation.js b/script/synology/downloadstation.js index cc5db629236..cc9c3ba415d 100644 --- a/script/synology/downloadstation.js +++ b/script/synology/downloadstation.js @@ -3,6 +3,7 @@ const synoUrlKey = "syno_https_url"; const synoAccountKey = "syno_account"; const synoPasswdKey = "syno_passwd"; const synoSidKey = "syno_sid"; +let subDirs = []; let magicJS = MagicJS(scriptName, "INFO"); function SynoAuth(synoUrl, account, passwd) { @@ -41,7 +42,7 @@ function AddTask(synoUrl, sid, uri, destination) { return new Promise((resolve) => { let options = { url: `${synoUrl}/webapi/DownloadStation/task.cgi`, - body: `_sid=${sid}&api=SYNO.DownloadStation.Task&version=1&method=create&uri=${uri}&destination=${destination}`, + body: `_sid=${sid}&api=SYNO.DownloadStation.Task&version=1&method=create&destination=${destination}&uri=${uri}`, }; magicJS.post(options, (err, resp, data) => { if (err) { @@ -108,7 +109,7 @@ function GetLocation(synoUrl, sid) { * @returns */ function CreateFolder(synoUrl, sid, folderPath, folderName) { - magicJS.logDebug(`创建下载目录,路径: ${folderPath}/${folderName}`); + magicJS.logInfo(`创建下载目录,路径: ${folderPath}/${folderName}`); return new Promise((resolve, reject) => { let options = { url: encodeURI(`${synoUrl}/webapi/entry.cgi?_sid=${sid}&api=SYNO.FileStation.CreateFolder&force_parent=true&version=2&method=create&folder_path=["${folderPath}"]&name=["${folderName}"]`), @@ -139,13 +140,13 @@ function CreateFolder(synoUrl, sid, folderPath, folderName) { }); } -async function CreateTasks(mediaList, synoSid, synoUrl, synoAccount, synoPasswd, appName, userName) { - let decodeMediaList = []; +async function CreateTasks(media, synoSid, synoUrl, synoAccount, synoPasswd, appName, subDirs) { + let decodeMedia = []; let synoDestination = ""; - for (let i = 0; i < mediaList.length; i++) { - let decodeMedia = escape(mediaList[i]); - decodeMediaList.push(decodeMedia); + for (let i = 0; i < media.length; i++) { + decodeMedia.push(encodeURIComponent(media[i])); } + // 获取sid if (!!!synoSid) { await SynoAuth(synoUrl, synoAccount, synoPasswd) @@ -155,34 +156,38 @@ async function CreateTasks(mediaList, synoSid, synoUrl, synoAccount, synoPasswd, }) .catch((err) => magicJS.notify(`登录失败,异常信息:${err}`)); } + // 获取DownloadStation默认目录 await GetLocation(synoUrl, synoSid).then((value) => { synoDestination = value; - magicJS.logDebug(`当前下载目录:${synoDestination}`); + magicJS.logInfo(`当前下载目录:${synoDestination}`); }); - // 根据用户名创建目录 - await magicJS - .retry(CreateFolder, 1, 100)(synoUrl, synoSid, `/${synoDestination}/${appName}`, userName) - .then((value) => { - if (value === true) { - magicJS.logDebug("创建下载目录成功"); - } - }) - .catch((err) => { - magicJS.logError(`在群晖上创建目录失败,异常信息:${err}`); - }); - // 添加下载任务 - let uri = decodeMediaList.join(","); - let downloadDir = `${synoDestination}/${appName}/${userName}`; - let result = await AddTask(synoUrl, synoSid, uri, downloadDir); - if (result === true) { - magicJS.notify(`${scriptName}`, `添加成功 ${downloadDir}`, `${mediaList.join("\n")}`); - } else if (mediaList) { - magicJS.notify(scriptName, `添加失败 ${downloadDir}`, `${mediaList.join("\n")}`); - } else { - magicJS.notify(scriptName, `添加失败 ${downloadDir}`); + + // 创建多层级目录 + let folderPath = `/${synoDestination}/${appName}`; + for (let i = 0; i < subDirs.length; i++) { + await magicJS + .retry(CreateFolder, 1, 100)(synoUrl, synoSid, folderPath, subDirs[i]) + .then((value) => { + if (value === true) { + let msg = "创建下载目录成功"; + magicJS.logInfo(msg); + } + }) + .catch((err) => { + let errMsg = `在群晖上创建目录失败,异常信息:${err}`; + magicJS.logError(errMsg); + magicJS.notify(errMsg); + }); + folderPath = `${folderPath}/${subDirs[i]}`; } - return { synoSid, synoDestination }; + + // 添加下载任务 + let uri = decodeMedia.join(","); + // 必须删除路径 /donwloads 前面的第一个 /,否则群晖接口会返回目录不存在 + let downloadDir = folderPath.slice(1); + let result = await AddTask(synoUrl, synoSid, uri, downloadDir); + return result; } (async () => { @@ -191,9 +196,8 @@ async function CreateTasks(mediaList, synoSid, synoUrl, synoAccount, synoPasswd, let synoAccount = magicJS.read(synoAccountKey); let synoPasswd = magicJS.read(synoPasswdKey); let synoSid = magicJS.read(synoSidKey); - let mediaList = []; + let media = {}; let appName = ""; - let userName = ""; if (!synoUrl || !synoAccount || !synoPasswd) { magicJS.logWarning("请先在BoxJS中配置DownloadStation"); magicJS.notify("请先在BoxJS中配置DownloadStation"); @@ -201,42 +205,49 @@ async function CreateTasks(mediaList, synoSid, synoUrl, synoAccount, synoPasswd, } if (magicJS.isResponse) { - // Twitter收藏下载 + // Twitter 收藏下载 if (/^https?:\/\/api\.twitter\.com\/[0-9.]*\/favorites\/create.json/.test(magicJS.request.url) === true) { - try { - appName = "twitter"; - // 获取媒体url - let obj = JSON.parse(magicJS.response.body); - if (obj.extended_entities && obj.extended_entities.media) { - obj.extended_entities.media.forEach((element) => { - // 使用推文作者名称作为子目录名 - userName = obj.user.screen_name.replace(/[^a-zA-Z0-9]+/, ""); - magicJS.logDebug(`当前推文的用户:${userName}`); - if (element.type == "photo") { - mediaList.push(element.media_url); - } else if (element.type == "video") { - let maxBitrate = 0; - let videoUrl = ""; - element.video_info.variants.forEach((video) => { - if (video.bitrate && video.bitrate > maxBitrate) { - maxBitrate = video.bitrate; - videoUrl = video.url; - } - }); - mediaList.push(videoUrl); - magicJS.logDebug(videoUrl); - } - }); - } - } catch (err) { - magicJS.logError(`添加下载任务失败,异常信息:${err}`); - magicJS.notify("添加下载任务失败,请查阅日志"); - } + appName = "twitter"; + let body = JSON.parse(magicJS.response.body); + media = downloadTwitterMedia(body); + + // 网易云课堂 课程下载 + } else if (/^https?:\/\/ke\.study\.youdao\.com\/course\/app\/detail.json/.test(magicJS.request.url) === true) { + appName = "163study"; + let body = JSON.parse(magicJS.response.body); + media = download163StudyMedia(body); + subDirs = [replaceName(body.data.courseTitle)]; } - // 添加下载任务至DownloadStation - if (mediaList.length > 0) { - await CreateTasks(mediaList, synoSid, synoUrl, synoAccount, synoPasswd, appName, userName); + // 提交给群晖DownloadStation离线下载 + if (media) { + let mediaLength = media.length; + let success = 0; + let failure = 0; + for (let i = 0; i < mediaLength; i++) { + try { + let result = await CreateTasks(media[i].url, synoSid, synoUrl, synoAccount, synoPasswd, appName, subDirs.concat(media[i].title)); + if (result === true) { + success += 1; + magicJS.notify(`${scriptName}`, `添加成功 ${success}/${failure}/${success + failure}/${mediaLength}`, `${media[i].url.join("\n")}`); + } else if (media[i]) { + failure += 1; + magicJS.notify(scriptName, `添加失败 ${success}/${failure}/${success + failure}/${mediaLength}`, `${media[i].url.join("\n")}`); + } else { + failure += 1; + magicJS.notify(scriptName, `添加失败 ${success}/${failure}/${success + failure}/${mediaLength}`); + } + } catch (err) { + let errMsg = `添加任务失败,异常信息:${err}`; + magicJS.logError(errMsg); + magicJS.notify(errMsg); + } + // 休息一会 + magicJS.sleep(0.1); + } + if (success + failure > 1){ + magicJS.notify(scriptName, "任务添加完毕", `计划添加任务${mediaLength}个,成功${success}个,失败${failure}个,实际执行${success + failure}个。`); + } } } else { await SynoAuth(synoUrl, synoAccount, synoPasswd) @@ -254,5 +265,104 @@ async function CreateTasks(mediaList, synoSid, synoUrl, synoAccount, synoPasswd, magicJS.done(); })(); +function replaceName(name) { + return name + .replace(" ", "") + .replace("/", "_") + .replace("\\", "_") + .replace("(", "") + .replace("(", "") + .replace("【", "") + .replace(")", "_") + .replace(")", "_") + .replace("】", "_") + .replace(":", "_") + .replace("*", "_") + .replace("?", "_") + .replace('"', "_") + .replace("<", "_") + .replace(">", "_") + .replace("|", "_") + .replace("·", "_"); +} + +/** + * 推特收藏图片和视频下载 + * + * @param {*} obj + * @return {*} + */ +function downloadTwitterMedia(obj) { + try { + let media = { title: "", url: [] }; + // 获取媒体url + if (obj.extended_entities && obj.extended_entities.media) { + obj.extended_entities.media.forEach((element) => { + // 使用推文作者名称作为子目录名 + userName = obj.user.screen_name.replace(/[^a-zA-Z0-9]+/, ""); + media["title"] = userName; + magicJS.logDebug(`当前推文的用户:${userName}`); + if (element.type == "photo") { + media['url'].push(element.media_url); + } else if (element.type == "video") { + let maxBitrate = 0; + let videoUrl = ""; + element.video_info.variants.forEach((video) => { + if (video.bitrate && video.bitrate > maxBitrate) { + maxBitrate = video.bitrate; + videoUrl = video.url; + } + }); + media['url'].push(videoUrl); + } + }); + } + return [media]; + } catch (err) { + magicJS.logError(`添加下载任务失败,异常信息:${err}`); + magicJS.notify("添加下载任务失败,请查阅日志"); + } + return media; +} + +/** + * 网易云课堂课程下载 + * + * @param {*} body + * @return {*} + */ +function download163StudyMedia(body) { + try { + if (body.data && body.data.schedule) { + let downlodaFiles = []; + body.data.schedule.forEach((element) => { + element.list.forEach((item) => { + if (item.hasOwnProperty("list")) { + item.list.forEach((media) => { + if (media.hasOwnProperty("videoDuration")) { + // 替换某些不能当文件夹或文件名的字符 + let title = replaceName(media.downloadTitle); + downlodaFiles.push({ + title: title, + url: [media.video.downloadUrl], + }); + } + }); + } + }); + }); + magicJS.logInfo(JSON.stringify(downlodaFiles)); + magicJS.notify(`共发现${downlodaFiles.length}个课程视频,添加任务期间可能造成网易云课堂阻塞,请耐心等待。`); + return downlodaFiles; + } else { + magicJS.logError("网易云课堂接口返回数据错误。"); + magicJS.notify("获取课程视频失败,网易云课堂接口返回数据错误。"); + } + } catch (err) { + magicJS.logError(`下载网易云课堂视频失败,异常信息:${err}`); + magicJS.notifyDebug("下载网易云课堂视频失败,请查阅日志。"); + } +} + // prettier-ignore -function MagicJS(scriptName="MagicJS",logLevel="INFO"){return new class{constructor(){if(this.version="2.2.3.3",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.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){this._barkUrl=url.replace(/\/+$/g,"")}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.isNode){if(this._barkUrl){let content=encodeURI(`${title}/${subTitle}\n${body}`);this.get(`${this._barkUrl}/${content}`,()=>{})}}else if(this.isJSBox){let push={title:title,body:subTitle?`${subTitle}\n${body}`:body};$push.schedule(push)}}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))}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)} \ No newline at end of file +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)}