synologyHelper.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  1. const scriptName = "Download Station";
  2. const synoUrlKey = "syno_https_url";
  3. const synoAccountKey = "syno_account";
  4. const synoPasswdKey = "syno_passwd";
  5. const synoSidKey = "syno_sid";
  6. let subDirs = [];
  7. let magicJS = MagicJS(scriptName, "INFO");
  8. function SynoAuth(synoUrl, account, passwd) {
  9. return new Promise((resolve, reject) => {
  10. let options = {
  11. url: `${synoUrl}/webapi/entry.cgi?api=SYNO.API.Auth&version=6&method=login&account=${account}&passwd=${passwd}`,
  12. headers: {
  13. "User-Agent": "Mozilla/5.0",
  14. },
  15. };
  16. magicJS.get(options, (err, resp, data) => {
  17. if (err) {
  18. magicJS.logError(`登录DownloadStation失败,请求异常:${err}`);
  19. reject("❌登录DownloadStation失败,请求异常,请查阅日志!");
  20. } else {
  21. try {
  22. magicJS.logDebug(`接口响应数据:${data}`);
  23. let obj = typeof data === "string" ? JSON.parse(data) : data;
  24. if (obj.success === true) {
  25. magicJS.logDebug(`登录获取的sid:${obj.data.sid}`);
  26. resolve(obj.data.sid);
  27. } else {
  28. magicJS.logWarning(`登录DownloadStation失败,接口响应:${data}`);
  29. reject(data);
  30. }
  31. } catch (err) {
  32. magicJS.logError(`登录DownloadStation失败,执行异常:${err},接口响应:${data}`);
  33. reject(data);
  34. }
  35. }
  36. });
  37. });
  38. }
  39. function AddTask(synoUrl, sid, uri, destination) {
  40. return new Promise((resolve) => {
  41. let options = {
  42. url: `${synoUrl}/webapi/DownloadStation/task.cgi`,
  43. body: `_sid=${sid}&api=SYNO.DownloadStation.Task&version=1&method=create&destination=${destination}&uri=${uri}`,
  44. };
  45. magicJS.post(options, (err, resp, data) => {
  46. if (err) {
  47. magicJS.logError(`添加下载任务失败,请求异常:${err}`);
  48. resolve(false);
  49. } else {
  50. try {
  51. let obj = typeof data === "string" ? JSON.parse(data) : data;
  52. if (obj.success === true) {
  53. resolve(true);
  54. } else {
  55. magicJS.logWarning(`添加下载任务失败,接口响应:${data}`);
  56. resolve(false);
  57. }
  58. } catch (err) {
  59. magicJS.logError(`添加下载任务失败,执行异常:${err},接口响应:${data}`);
  60. resolve(false);
  61. }
  62. }
  63. });
  64. });
  65. }
  66. /**
  67. * 获取DownloadStation下载位置
  68. * @param {*} synoUrl
  69. * @param {*} sid
  70. * @returns
  71. */
  72. function GetLocation(synoUrl, sid) {
  73. return new Promise((resolve) => {
  74. let options = {
  75. url: `${synoUrl}/webapi/DownloadStation/entry.cgi`,
  76. body: `_sid=${sid}&api=SYNO.DownloadStation2.Settings.Location&version=1&method=get`,
  77. };
  78. magicJS.post(options, (err, resp, data) => {
  79. if (err) {
  80. magicJS.logError(`添加下载任务失败,请求异常:${err}`);
  81. resolve(false);
  82. } else {
  83. try {
  84. let obj = typeof data === "string" ? JSON.parse(data) : data;
  85. if (obj.success === true) {
  86. resolve(obj.data.default_destination);
  87. } else {
  88. magicJS.logWarning(`添加下载任务失败,接口响应:${data}`);
  89. resolve("");
  90. }
  91. } catch (err) {
  92. magicJS.logError(`添加下载任务失败,执行异常:${err},接口响应:${data}`);
  93. resolve("");
  94. }
  95. }
  96. });
  97. });
  98. }
  99. /**
  100. * 创建下载目录
  101. * @param {*} synoUrl
  102. * @param {*} sid
  103. * @param {*} folderPath
  104. * @param {*} folderName
  105. * @returns
  106. */
  107. function CreateFolder(synoUrl, sid, folderPath, folderName) {
  108. magicJS.logInfo(`创建下载目录,路径: ${folderPath}/${folderName}`);
  109. return new Promise((resolve, reject) => {
  110. let options = {
  111. url: encodeURI(`${synoUrl}/webapi/entry.cgi?_sid=${sid}&api=SYNO.FileStation.CreateFolder&force_parent=true&version=2&method=create&folder_path=["${folderPath}"]&name=["${folderName}"]`),
  112. };
  113. magicJS.get(options, (err, resp, data) => {
  114. if (err) {
  115. magicJS.logError(`创建文件目录失败,请求异常:${err}`);
  116. reject(false);
  117. } else {
  118. try {
  119. let obj = typeof data === "string" ? JSON.parse(data) : data;
  120. if (obj.success === true) {
  121. resolve(true);
  122. } else if (obj.error.code === "119") {
  123. magicJS.logWarning(`SID not found,接口响应:${data}`);
  124. magicJS.write(synoSidKey, "");
  125. reject(data);
  126. } else {
  127. magicJS.logWarning(`创建文件目录失败,接口响应:${data}`);
  128. reject(data);
  129. }
  130. } catch (err) {
  131. magicJS.logError(`创建文件目录失败,执行异常:${err},接口响应:${data}`);
  132. reject(data);
  133. }
  134. }
  135. });
  136. });
  137. }
  138. async function CreateTasks(media, synoSid, synoUrl, synoAccount, synoPasswd, appName, subDirs) {
  139. let decodeMedia = [];
  140. let synoDestination = "";
  141. for (let i = 0; i < media.length; i++) {
  142. decodeMedia.push(encodeURIComponent(media[i]));
  143. }
  144. // 获取sid
  145. if (!!!synoSid) {
  146. await SynoAuth(synoUrl, synoAccount, synoPasswd)
  147. .then((value) => {
  148. synoSid = value;
  149. magicJS.write(synoSidKey, synoSid);
  150. })
  151. .catch((err) => magicJS.notify(`登录失败,异常信息:${err}`));
  152. }
  153. // 获取DownloadStation默认目录
  154. await GetLocation(synoUrl, synoSid).then((value) => {
  155. synoDestination = value;
  156. magicJS.logInfo(`当前下载目录:${synoDestination}`);
  157. });
  158. // 创建多层级目录
  159. let folderPath = `/${synoDestination}/${appName}`;
  160. for (let i = 0; i < subDirs.length; i++) {
  161. await magicJS
  162. .retry(CreateFolder, 1, 100)(synoUrl, synoSid, folderPath, subDirs[i])
  163. .then((value) => {
  164. if (value === true) {
  165. let msg = "创建下载目录成功";
  166. magicJS.logInfo(msg);
  167. }
  168. })
  169. .catch((err) => {
  170. let errMsg = `在群晖上创建目录失败,异常信息:${err}`;
  171. magicJS.logError(errMsg);
  172. magicJS.notify(errMsg);
  173. });
  174. folderPath = `${folderPath}/${subDirs[i]}`;
  175. }
  176. // 添加下载任务
  177. let uri = decodeMedia.join(",");
  178. // 必须删除路径 /donwloads 前面的第一个 /,否则群晖接口会返回目录不存在
  179. let downloadDir = folderPath.slice(1);
  180. let result = await AddTask(synoUrl, synoSid, uri, downloadDir);
  181. return result;
  182. }
  183. (async () => {
  184. // 群晖 Download Station 配置检查
  185. let synoUrl = magicJS.read(synoUrlKey);
  186. let synoAccount = magicJS.read(synoAccountKey);
  187. let synoPasswd = magicJS.read(synoPasswdKey);
  188. let synoSid = magicJS.read(synoSidKey);
  189. let media = {};
  190. let appName = "";
  191. if (!synoUrl || !synoAccount || !synoPasswd) {
  192. magicJS.logWarning("请先在BoxJS中配置DownloadStation");
  193. magicJS.notify("请先在BoxJS中配置DownloadStation");
  194. return;
  195. }
  196. if (magicJS.isResponse) {
  197. // Twitter 收藏下载
  198. if (/^https?:\/\/api\.twitter\.com\/[0-9.]*\/favorites\/create.json/.test(magicJS.request.url) === true) {
  199. appName = "twitter";
  200. let body = JSON.parse(magicJS.response.body);
  201. media = downloadTwitterMedia(body);
  202. // 网易云课堂 课程下载
  203. } else if (/^https?:\/\/ke\.study\.youdao\.com\/course\/app\/detail.json/.test(magicJS.request.url) === true) {
  204. appName = "163study";
  205. let body = JSON.parse(magicJS.response.body);
  206. media = download163StudyMedia(body);
  207. subDirs = [replaceName(body.data.courseTitle)];
  208. }
  209. // 提交给群晖DownloadStation离线下载
  210. if (media) {
  211. let mediaLength = media.length;
  212. let success = 0;
  213. let failure = 0;
  214. for (let i = 0; i < mediaLength; i++) {
  215. try {
  216. let result = await CreateTasks(media[i].url, synoSid, synoUrl, synoAccount, synoPasswd, appName, subDirs.concat(media[i].title));
  217. if (result === true) {
  218. success += 1;
  219. magicJS.notify(`${scriptName}`, `添加成功 ${success}/${failure}/${success + failure}/${mediaLength}`, `${media[i].url.join("\n")}`);
  220. } else if (media[i]) {
  221. failure += 1;
  222. magicJS.notify(scriptName, `添加失败 ${success}/${failure}/${success + failure}/${mediaLength}`, `${media[i].url.join("\n")}`);
  223. } else {
  224. failure += 1;
  225. magicJS.notify(scriptName, `添加失败 ${success}/${failure}/${success + failure}/${mediaLength}`);
  226. }
  227. } catch (err) {
  228. let errMsg = `添加任务失败,异常信息:${err}`;
  229. magicJS.logError(errMsg);
  230. magicJS.notify(errMsg);
  231. }
  232. // 休息一会
  233. magicJS.sleep(0.1);
  234. }
  235. if (success + failure > 1){
  236. magicJS.notify(scriptName, "任务添加完毕", `计划添加任务${mediaLength}个,成功${success}个,失败${failure}个,实际执行${success + failure}个。`);
  237. }
  238. }
  239. } else {
  240. await SynoAuth(synoUrl, synoAccount, synoPasswd)
  241. .then((value) => {
  242. let msg = `登录成功,获取Sid:${value}`;
  243. magicJS.notify(msg);
  244. magicJS.logInfo(msg);
  245. synoSid = value;
  246. magicJS.write(synoSidKey, synoSid);
  247. })
  248. .catch((err) => {
  249. magicJS.notify(`登录失败,异常信息:${err}`);
  250. });
  251. }
  252. magicJS.done();
  253. })();
  254. function replaceName(name) {
  255. return name
  256. .replace(" ", "")
  257. .replace("/", "_")
  258. .replace("\\", "_")
  259. .replace("(", "")
  260. .replace("(", "")
  261. .replace("【", "")
  262. .replace(")", "_")
  263. .replace(")", "_")
  264. .replace("】", "_")
  265. .replace(":", "_")
  266. .replace("*", "_")
  267. .replace("?", "_")
  268. .replace('"', "_")
  269. .replace("<", "_")
  270. .replace(">", "_")
  271. .replace("|", "_")
  272. .replace("·", "_");
  273. }
  274. /**
  275. * 推特收藏图片和视频下载
  276. *
  277. * @param {*} obj
  278. * @return {*}
  279. */
  280. function downloadTwitterMedia(obj) {
  281. try {
  282. let media = { title: "", url: [] };
  283. // 获取媒体url
  284. if (obj.extended_entities && obj.extended_entities.media) {
  285. obj.extended_entities.media.forEach((element) => {
  286. // 使用推文作者名称作为子目录名
  287. userName = obj.user.screen_name.replace(/[^a-zA-Z0-9]+/, "");
  288. media["title"] = userName;
  289. magicJS.logDebug(`当前推文的用户:${userName}`);
  290. if (element.type == "photo") {
  291. media['url'].push(element.media_url);
  292. } else if (element.type == "video") {
  293. let maxBitrate = 0;
  294. let videoUrl = "";
  295. element.video_info.variants.forEach((video) => {
  296. if (video.bitrate && video.bitrate > maxBitrate) {
  297. maxBitrate = video.bitrate;
  298. videoUrl = video.url;
  299. }
  300. });
  301. media['url'].push(videoUrl);
  302. }
  303. });
  304. }
  305. return [media];
  306. } catch (err) {
  307. magicJS.logError(`添加下载任务失败,异常信息:${err}`);
  308. magicJS.notify("添加下载任务失败,请查阅日志");
  309. }
  310. return media;
  311. }
  312. /**
  313. * 网易云课堂课程下载
  314. *
  315. * @param {*} body
  316. * @return {*}
  317. */
  318. function download163StudyMedia(body) {
  319. try {
  320. if (body.data && body.data.schedule) {
  321. let downlodaFiles = [];
  322. body.data.schedule.forEach((element) => {
  323. element.list.forEach((item) => {
  324. if (item.hasOwnProperty("list")) {
  325. item.list.forEach((media) => {
  326. if (media.hasOwnProperty("videoDuration")) {
  327. // 替换某些不能当文件夹或文件名的字符
  328. let title = replaceName(media.downloadTitle);
  329. downlodaFiles.push({
  330. title: title,
  331. url: [media.video.downloadUrl],
  332. });
  333. }
  334. });
  335. }
  336. });
  337. });
  338. magicJS.logInfo(JSON.stringify(downlodaFiles));
  339. magicJS.notify(`共发现${downlodaFiles.length}个课程视频,添加任务期间可能造成网易云课堂阻塞,请耐心等待。`);
  340. return downlodaFiles;
  341. } else {
  342. magicJS.logError("网易云课堂接口返回数据错误。");
  343. magicJS.notify("获取课程视频失败,网易云课堂接口返回数据错误。");
  344. }
  345. } catch (err) {
  346. magicJS.logError(`下载网易云课堂视频失败,异常信息:${err}`);
  347. magicJS.notifyDebug("下载网易云课堂视频失败,请查阅日志。");
  348. }
  349. }
  350. // prettier-ignore
  351. 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]<this.logLevels[level.toUpperCase()]||console.log(`[${level}] [${this.scriptName}]\n${msg}\n`)}logDebug(msg){this.log(msg,"DEBUG")}logInfo(msg){this.log(msg,"INFO")}logNotify(msg){this.log(msg,"NOTIFY")}logWarning(msg){this.log(msg,"WARNING")}logError(msg){this.log(msg,"ERROR")}logRetry(msg){this.log(msg,"RETRY")}adapterHttpOptions(options,method){let _options="object"==typeof options?Object.assign({},options):{url:options,headers:{}};_options.hasOwnProperty("header")&&!_options.hasOwnProperty("headers")&&(_options.headers=_options.header,delete _options.header),_options.headers&&"object"==typeof _options.headers&&_options.headers["User-Agent"]||(_options.headers&&"object"==typeof _options.headers||(_options.headers={}),this.isNode?_options.headers["User-Agent"]=this.pcUserAgent:_options.headers["User-Agent"]=this.iOSUserAgent);let skipScripting=!1;if(("object"==typeof _options.opts&&(!0===_options.opts.hints||!0===_options.opts["Skip-Scripting"])||"object"==typeof _options.headers&&!0===_options.headers["X-Surge-Skip-Scripting"])&&(skipScripting=!0),skipScripting||(this.isSurge?_options.headers["X-Surge-Skip-Scripting"]=!1:this.isLoon?_options.headers["X-Requested-With"]="XMLHttpRequest":this.isQuanX&&("object"!=typeof _options.opts&&(_options.opts={}),_options.opts.hints=!1)),this.isSurge&&!skipScripting||delete _options.headers["X-Surge-Skip-Scripting"],!this.isQuanX&&_options.hasOwnProperty("opts")&&delete _options.opts,this.isQuanX&&_options.hasOwnProperty("opts")&&delete _options.opts["Skip-Scripting"],"GET"===method&&!this.isNode&&_options.body){let qs=Object.keys(_options.body).map(key=>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)}