drpy.js 48 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504
  1. import './util.js';
  2. import cheerio from './cheerio.min.js';
  3. import './crypto-js.js'
  4. // import muban from 'https://gitcode.net/qq_32394351/dr_py/-/raw/master/js/模板.js';
  5. // const key = 'drpy_zbk';
  6. // eval(req('http://192.168.1.124:5705/libs/es6py.js').content);
  7. function init_test() {
  8. console.log("init_test_start");
  9. console.log(RKEY);
  10. console.log(JSON.stringify(rule));
  11. // clearItem(RULE_CK);
  12. // console.log(JSON.stringify(rule));
  13. // let aa = base64Encode('编码测试一下')
  14. // log(aa);
  15. // let bb = base64Decode(aa);
  16. // log('bb:' + bb);
  17. // console.log(request('https://www.baidu.com',{withHeaders:true}));
  18. // console.log(request('https://www.baidu.com/favicon.ico', { toBase64: true }));
  19. // require('http://192.168.10.99:5705/txt/pluto/drT.js');
  20. // console.log(typeof(drT));
  21. // console.log(drT.renderText('{{fl.cate}},hi, {{fl}}哈哈.{{fl}}',{sort: 1,cate:'movie'},'fl'));
  22. console.log("init_test_end");
  23. }
  24. let rule = {};
  25. /** 已知问题记录
  26. * 1.影魔的jinjia2引擎不支持 {{fl}}对象直接渲染
  27. * Array.prototype.append = Array.prototype.push; 这种js执行后有毛病,for in 循环列表会把属性给打印出来
  28. * 2.import es6py.js但是里面的函数没有被装载进来.比如drpy规则报错setResult2 is undefiend
  29. * 3.无法重复导入cheerio(怎么解决drpy和parseTag里都需要导入cheerio的问题) 无法在副文件导入cheerio (现在是全部放在drpy一个文件里了,凑合解决?)
  30. * 4.有个错误不知道哪儿来的 executeScript: com.quickjs.JSObject$Undefined cannot be cast to java.lang.String 在 点击选集播放打印init_test_end后面打印
  31. * 5.需要实现 stringify 函数,比起JSON.stringify函数,它会原封不动保留中文不会编码unicode
  32. * todo: jsp:{pdfa,pdfh,pd},json:{pdfa,pdfh,pd},jq:{pdfa,pdfh,pd}
  33. * * 电脑看日志调试
  34. adb tcpip 5555
  35. adb connect 192.168.10.192
  36. adb devices -l
  37. adb logcat -c
  38. adb logcat | grep -i QuickJS
  39. * **/
  40. /*** 以下是内置变量和解析方法 **/
  41. const MOBILE_UA = 'Mozilla/5.0 (Linux; Android 11; M2007J3SC Build/RKQ1.200826.002; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/77.0.3865.120 MQQBrowser/6.2 TBS/045714 Mobile Safari/537.36';
  42. const PC_UA = 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36';
  43. const UA = 'Mozilla/5.0';
  44. const UC_UA = 'Mozilla/5.0 (Linux; U; Android 9; zh-CN; MI 9 Build/PKQ1.181121.001) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/57.0.2987.108 UCBrowser/12.5.5.1035 Mobile Safari/537.36';
  45. const IOS_UA = 'Mozilla/5.0 (iPhone; CPU iPhone OS 13_2_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/13.0.3 Mobile/15E148 Safari/604.1';
  46. const RULE_CK = 'cookie'; // 源cookie的key值
  47. // const KEY = typeof(key)!=='undefined'&&key?key:'drpy_' + (rule.title || rule.host); // 源的唯一标识
  48. const CATE_EXCLUDE = '首页|留言|APP|下载|资讯|新闻|动态';
  49. const TAB_EXCLUDE = '猜你|喜欢|APP|下载|剧情|热播';
  50. const OCR_RETRY = 3;//ocr验证重试次数
  51. // const OCR_API = 'http://dm.mudery.com:10000';//ocr在线识别接口
  52. // const OCR_API = 'http://192.168.3.239:5705/parse/ocr';//ocr在线识别接口
  53. const OCR_API = 'http://cms.nokia.press/parse/ocr';//ocr在线识别接口
  54. if (typeof (MY_URL) === 'undefined') {
  55. var MY_URL; // 全局注入变量,pd函数需要
  56. }
  57. var RKEY; // 源的唯一标识
  58. var fetch;
  59. var print;
  60. var log;
  61. var rule_fetch_params;
  62. var fetch_params; // 每个位置单独的
  63. var oheaders;
  64. var _pdfh;
  65. var _pdfa;
  66. var _pd;
  67. // const DOM_CHECK_ATTR = ['url', 'src', 'href', 'data-original', 'data-src'];
  68. const DOM_CHECK_ATTR = /(url|src|href|data-original|data-src)$/;
  69. const SELECT_REGEX = /:eq|:lt|:gt|#/g;
  70. const SELECT_REGEX_A = /:eq|:lt|:gt/g;
  71. function setResult(d) {
  72. if (!Array.isArray(d)) {
  73. return []
  74. }
  75. VODS = [];
  76. // print(d);
  77. d.forEach(function (it) {
  78. let obj = {
  79. vod_id: it.url || '',
  80. vod_name: it.title || '',
  81. vod_remarks: it.desc || '',
  82. vod_content: it.content || '',
  83. vod_pic: it.pic_url || it.img || '',
  84. };
  85. let keys = Object.keys(it);
  86. if (keys.includes('tname')) {
  87. obj.type_name = it.tname || '';
  88. }
  89. if (keys.includes('tid')) {
  90. obj.type_id = it.tid || '';
  91. }
  92. if (keys.includes('year')) {
  93. obj.vod_year = it.year || '';
  94. }
  95. if (keys.includes('actor')) {
  96. obj.vod_actor = it.actor || '';
  97. }
  98. if (keys.includes('director')) {
  99. obj.vod_director = it.director || '';
  100. }
  101. if (keys.includes('area')) {
  102. obj.vod_area = it.area || '';
  103. }
  104. VODS.push(obj);
  105. });
  106. return VODS
  107. }
  108. function setResult2(res) {
  109. VODS = res.list || [];
  110. return VODS
  111. }
  112. function setHomeResult(res) {
  113. if (!res || typeof (res) !== 'object') {
  114. return []
  115. }
  116. return setResult(res.list);
  117. }
  118. // 猫了个咪
  119. function rc(js) {
  120. if (js === 'maomi_aes.js') {
  121. var a = CryptoJS.enc.Utf8.parse("625222f9149e961d");
  122. var t = CryptoJS.enc.Utf8.parse("5efdtf6060e2o330");
  123. return {
  124. De: function (word) {
  125. word = CryptoJS.enc.Hex.parse(word)
  126. return CryptoJS.AES.decrypt(CryptoJS.enc.Base64.stringify(word), a, {
  127. iv: t,
  128. mode: CryptoJS.mode.CBC,
  129. padding: CryptoJS.pad.Pkcs7
  130. }).toString(CryptoJS.enc.Utf8)
  131. },
  132. En: function (word) {
  133. // print(a);
  134. // print(word);
  135. var Encrypted = CryptoJS.AES.encrypt(word, a, {
  136. iv: t,
  137. mode: CryptoJS.mode.CBC,
  138. padding: CryptoJS.pad.Pkcs7
  139. });
  140. return Encrypted.ciphertext.toString();
  141. }
  142. };
  143. }
  144. return {};
  145. }
  146. // 千万不要用for in 推荐 forEach (for in 会打乱顺序)
  147. //猫函数
  148. function maoss(jxurl, ref, key) {
  149. eval(getCryptoJS());
  150. try {
  151. var getVideoInfo = function (text) {
  152. return CryptoJS.AES.decrypt(text, key, { iv: iv, padding: CryptoJS.pad.Pkcs7 }).toString(CryptoJS.enc.Utf8);
  153. };
  154. var token_key = key == undefined ? 'dvyYRQlnPRCMdQSe' : key;
  155. if (ref) {
  156. var html = request(jxurl, {
  157. headers: {
  158. 'Referer': ref
  159. }
  160. });
  161. } else {
  162. var html = request(jxurl);
  163. }
  164. // print(html);
  165. if (html.indexOf('&btwaf=') != -1) {
  166. html = request(jxurl + '&btwaf' + html.match(/&btwaf(.*?)"/)[1], {
  167. headers: {
  168. 'Referer': ref
  169. }
  170. })
  171. }
  172. var token_iv = html.split('_token = "')[1].split('"')[0];
  173. var key = CryptoJS.enc.Utf8.parse(token_key);
  174. var iv = CryptoJS.enc.Utf8.parse(token_iv);
  175. // log("iv:"+iv);
  176. // log(html);
  177. // print(key);
  178. // print(iv);
  179. eval(html.match(/var config = {[\s\S]*?}/)[0] + '');
  180. // config.url = config.url.replace(/,/g,'');
  181. // print(config.url);
  182. if (!config.url.startsWith('http')) {
  183. //config.url = decodeURIComponent(AES(config.url, key, iv));
  184. config.url = CryptoJS.AES.decrypt(config.url, key, {
  185. iv: iv,
  186. padding: CryptoJS.pad.Pkcs7
  187. }).toString(CryptoJS.enc.Utf8)
  188. }
  189. return config.url;
  190. } catch (e) {
  191. return '';
  192. }
  193. }
  194. let VODS = [];// 一级或者搜索需要的数据列表
  195. let VOD = {};// 二级的单个数据
  196. /**
  197. * 重写pd方法-增加自动urljoin(没法重写,改个名继续骗)
  198. * @param html
  199. * @param parse
  200. * @param uri
  201. * @returns {*}
  202. */
  203. function pD(html, parse, uri) {
  204. let ret = pdfh(html, parse);
  205. if (typeof (uri) === 'undefined' || !uri) {
  206. uri = '';
  207. }
  208. if (DOM_CHECK_ATTR.test(parse)) {
  209. if (/http/.test(ret)) {
  210. ret = ret.substr(ret.indexOf('http'));
  211. } else {
  212. ret = urljoin(MY_URL, ret)
  213. }
  214. }
  215. // MY_URL = getItem('MY_URL',MY_URL);
  216. // console.log(`规则${RKEY}打印MY_URL:${MY_URL},uri:${uri}`);
  217. return ret
  218. }
  219. const parseTags = {
  220. jsp: {
  221. pdfh: pdfh,
  222. pdfa: pdfa,
  223. pd: pD,
  224. },
  225. json: {
  226. pdfh(html, parse) {
  227. if (!parse || !parse.trim()) {
  228. return '';
  229. }
  230. if (typeof (html) === 'string') {
  231. html = JSON.parse(html);
  232. }
  233. parse = parse.trim();
  234. if (!parse.startsWith('$.')) {
  235. parse = '$.' + parse;
  236. }
  237. parse = parse.split('||');
  238. for (let ps of parse) {
  239. let ret = cheerio.jp(ps, html);
  240. if (Array.isArray(ret)) {
  241. ret = ret[0] || '';
  242. } else {
  243. ret = ret || ''
  244. }
  245. if (ret && typeof (ret) !== 'string') {
  246. ret = ret.toString();
  247. }
  248. if (ret) {
  249. return ret
  250. }
  251. }
  252. return '';
  253. },
  254. pdfa(html, parse) {
  255. if (!parse || !parse.trim()) {
  256. return '';
  257. }
  258. if (typeof (html) === 'string') {
  259. html = JSON.parse(html);
  260. }
  261. parse = parse.trim()
  262. if (!parse.startsWith('$.')) {
  263. parse = '$.' + parse;
  264. }
  265. let ret = cheerio.jp(parse, html);
  266. if (Array.isArray(ret) && Array.isArray(ret[0]) && ret.length === 1) {
  267. return ret[0] || []
  268. }
  269. return ret || []
  270. },
  271. pd(html, parse) {
  272. let ret = parseTags.json.pdfh(html, parse);
  273. if (ret) {
  274. return urljoin(MY_URL, ret);
  275. }
  276. return ret
  277. },
  278. },
  279. jq: {
  280. pdfh(html, parse, base_url) {
  281. if (!parse || !parse.trim()) {
  282. return ''
  283. }
  284. let option = undefined;
  285. if (parse.indexOf('&&') > -1) {
  286. let sp = parse.split('&&');
  287. option = sp[sp.length - 1];
  288. sp.splice(sp.length - 1);
  289. if (sp.length > 1) {
  290. for (let i in sp) {
  291. if (!SELECT_REGEX.test(sp[i])) {
  292. sp[i] = sp[i] + ':eq(0)';
  293. }
  294. }
  295. } else {
  296. if (!SELECT_REGEX.test(sp[0])) {
  297. sp[0] = sp[0] + ':eq(0)';
  298. }
  299. }
  300. parse = sp.join(' ');
  301. }
  302. let result = '';
  303. const $ = cheerio.load(html);
  304. let ret = $(parse);
  305. if (option) {
  306. if (option === 'Text') {
  307. result = $(ret).text();
  308. }
  309. else if (option === 'Html') {
  310. result = $(ret).html();
  311. }
  312. else {
  313. result = $(ret).attr(option);
  314. }
  315. if (result && base_url && DOM_CHECK_ATTR.test(option)) {
  316. if (/http/.test(result)) {
  317. result = result.substr(result.indexOf('http'));
  318. } else {
  319. result = urljoin(base_url, result)
  320. }
  321. }
  322. } else {
  323. result = $(ret).toString();
  324. }
  325. return result;
  326. },
  327. pdfa(html, parse) {
  328. if (!parse || !parse.trim()) {
  329. return [];
  330. }
  331. if (parse.indexOf('&&') > -1) {
  332. let sp = parse.split('&&');
  333. for (let i in sp) {
  334. if (!SELECT_REGEX_A.test(sp[i]) && i < sp.length - 1) {
  335. sp[i] = sp[i] + ':eq(0)';
  336. }
  337. }
  338. parse = sp.join(' ');
  339. }
  340. const $ = cheerio.load(html);
  341. let ret = $(parse);
  342. let result = [];
  343. if (ret) {
  344. ret.each(function (idx, ele) {
  345. result.push($(ele).toString());
  346. });
  347. }
  348. return result;
  349. },
  350. pd(html, parse, uri) {
  351. return parseTags.jq.pdfh(html, parse, MY_URL);
  352. },
  353. },
  354. getParse(p0) {//非js开头的情况自动获取解析标签
  355. if (p0.startsWith('jsp:')) {
  356. return this.jsp
  357. } else if (p0.startsWith('json:')) {
  358. return this.json
  359. } else if (p0.startsWith('jq:')) {
  360. return this.jq
  361. } else {
  362. return this.jq
  363. }
  364. }
  365. };
  366. const stringify = JSON.stringify;
  367. const jsp = parseTags.jsp;
  368. /*** 后台需要实现的java方法并注入到js中 ***/
  369. /**
  370. * 读取本地文件->应用程序目录
  371. * @param filePath
  372. * @returns {string}
  373. */
  374. function readFile(filePath) {
  375. filePath = filePath || './uri.min.js';
  376. var fd = os.open(filePath);
  377. var buffer = new ArrayBuffer(1024);
  378. var len = os.read(fd, buffer, 0, 1024);
  379. console.log(len);
  380. let text = String.fromCharCode.apply(null, new Uint8Array(buffer));
  381. console.log(text);
  382. return text
  383. }
  384. /**
  385. * 处理返回的json数据
  386. * @param html
  387. * @returns {*}
  388. */
  389. function dealJson(html) {
  390. try {
  391. return html.match(/[\w|\W|\s|\S]*?(\{[\w|\W|\s|\S]*\})/).group[1];
  392. } catch (e) {
  393. }
  394. try {
  395. html = JSON.parse(html);
  396. } catch (e) { }
  397. // console.log(typeof(html));
  398. return html;
  399. }
  400. /**
  401. * 验证码识别逻辑,需要java实现(js没有bytes类型,无法调用后端的传递图片二进制获取验证码文本的接口)
  402. * @type {{api: string, classification: (function(*=): string)}}
  403. */
  404. var OcrApi = {
  405. api: OCR_API,
  406. classification: function (img) { // img是byte类型,这里不方便搞啊
  407. let code = '';
  408. try {
  409. let html = request(this.api, { data: { img: img }, headers: { 'User-Agent': PC_UA }, 'method': 'POST' });
  410. html = JSON.parse(html);
  411. code = html.url || '';
  412. } catch (e) { }
  413. return code
  414. }
  415. };
  416. /**
  417. * 验证码识别,暂未实现
  418. * @param url 验证码图片链接
  419. * @returns {string} 验证成功后的cookie
  420. */
  421. function verifyCode(url) {
  422. let cnt = 0;
  423. let host = getHome(url);
  424. let cookie = '';
  425. while (cnt < OCR_RETRY) {
  426. try {
  427. // let obj = {headers:headers,timeout:timeout};
  428. let yzm_url = `${host}/index.php/verify/index.html`;
  429. console.log(`验证码链接:${yzm_url}`);
  430. let hhtml = request(yzm_url, { withHeaders: true, toBase64: true });
  431. let json = JSON.parse(hhtml);
  432. if (!cookie) {
  433. cookie = json['set-cookie'] ? json['set-cookie'].split(';')[0] : '';
  434. }
  435. // console.log(hhtml);
  436. console.log('cookie:' + cookie);
  437. let img = json.body;
  438. // console.log(img);
  439. let code = OcrApi.classification(img);
  440. console.log(`第${cnt + 1}次验证码识别结果:${code}`);
  441. let submit_url = `${host}/index.php/ajax/verify_check?type=search&verify=${code}`;
  442. console.log(submit_url);
  443. let html = request(submit_url, { headers: { Cookie: cookie, 'User-Agent': MOBILE_UA }, 'method': 'POST' });
  444. // console.log(html);
  445. html = JSON.parse(html);
  446. if (html.msg === 'ok') {
  447. console.log(`第${cnt + 1}次验证码提交成功`);
  448. return cookie // 需要返回cookie
  449. } else if (html.msg !== 'ok' && cnt + 1 >= OCR_RETRY) {
  450. cookie = ''; // 需要清空返回cookie
  451. }
  452. } catch (e) {
  453. console.log(`第${cnt + 1}次验证码提交失败:${e.message}`);
  454. if (cnt + 1 >= OCR_RETRY) {
  455. cookie = '';
  456. }
  457. }
  458. cnt += 1
  459. }
  460. return cookie
  461. }
  462. /**
  463. * 存在数据库配置表里, key字段对应值value,没有就新增,有就更新,调用此方法会清除key对应的内存缓存
  464. * @param k 键
  465. * @param v 值
  466. */
  467. function setItem(k, v) {
  468. local.set(RKEY, k, v);
  469. console.log(`规则${RKEY}设置${k} => ${v}`)
  470. }
  471. /**
  472. * 获取数据库配置表对应的key字段的value,没有这个key就返回value默认传参.需要有缓存,第一次获取后会存在内存里
  473. * @param k 键
  474. * @param v 值
  475. * @returns {*}
  476. */
  477. function getItem(k, v) {
  478. return local.get(RKEY, k) || v;
  479. }
  480. /**
  481. * 删除数据库key对应的一条数据,并清除此key对应的内存缓存
  482. * @param k
  483. */
  484. function clearItem(k) {
  485. local.delete(RKEY, k);
  486. }
  487. /*** js自封装的方法 ***/
  488. /**
  489. * 获取链接的host(带http协议的完整链接)
  490. * @param url 任意一个正常完整的Url,自动提取根
  491. * @returns {string}
  492. */
  493. function getHome(url) {
  494. if (!url) {
  495. return ''
  496. }
  497. let tmp = url.split('//');
  498. url = tmp[0] + '//' + tmp[1].split('/')[0];
  499. try {
  500. url = decodeURIComponent(url);
  501. } catch (e) { }
  502. return url
  503. }
  504. /**
  505. * get参数编译链接,类似python params字典自动拼接
  506. * @param url 访问链接
  507. * @param obj 参数字典
  508. * @returns {*}
  509. */
  510. function buildUrl(url, obj) {
  511. obj = obj || {};
  512. if (url.indexOf('?') < 0) {
  513. url += '?'
  514. }
  515. let param_list = [];
  516. let keys = Object.keys(obj);
  517. keys.forEach(it => {
  518. param_list.push(it + '=' + obj[it])
  519. });
  520. let prs = param_list.join('&');
  521. if (keys.length > 0 && !url.endsWith('?')) {
  522. url += '&'
  523. }
  524. url += prs;
  525. return url
  526. }
  527. /**
  528. * 远程依赖执行函数
  529. * @param url 远程js地址
  530. */
  531. function require(url) {
  532. eval(request(url));
  533. }
  534. /**
  535. * 海阔网页请求函数完整封装
  536. * @param url 请求链接
  537. * @param obj 请求对象 {headers:{},method:'',timeout:5000,body:'',withHeaders:false}
  538. * @returns {string|string|DocumentFragment|*}
  539. */
  540. function request(url, obj) {
  541. if (typeof (obj) === 'undefined' || !obj || obj === {}) {
  542. if (!fetch_params || !fetch_params.headers) {
  543. let headers = {
  544. 'User-Agent': MOBILE_UA,
  545. };
  546. if (rule.headers) {
  547. Object.assign(headers, rule.headers);
  548. }
  549. fetch_params.headers = headers;
  550. }
  551. if (!fetch_params.headers.Referer) {
  552. fetch_params.headers.Referer = getHome(url)
  553. }
  554. obj = fetch_params;
  555. } else {
  556. let headers = obj.headers || {};
  557. let keys = Object.keys(headers).map(it => it.toLowerCase());
  558. if (!keys.includes('user-agent')) {
  559. headers['User-Agent'] = MOBILE_UA;
  560. } if (!keys.includes('referer')) {
  561. headers['Referer'] = getHome(url);
  562. }
  563. obj.headers = headers;
  564. }
  565. console.log(JSON.stringify(obj.headers));
  566. if (typeof (obj.headers.body) != 'undefined' && obj.headers.body && typeof (obj.headers.body) === 'string') {
  567. let data = {};
  568. obj.headers.body.split('&').forEach(it => {
  569. data[it.split('=')[0]] = it.split('=')[1]
  570. });
  571. obj.data = data;
  572. delete obj.headers.body
  573. }
  574. if (!url) {
  575. return obj.withHeaders ? '{}' : ''
  576. }
  577. if (obj.toBase64) { // 返回base64,用于请求图片
  578. obj.buffer = 2;
  579. delete obj.toBase64
  580. }
  581. console.log('request:' + url);
  582. let res = req(url, obj);
  583. let html = res.content || '';
  584. // console.log(html);
  585. if (obj.withHeaders) {
  586. let htmlWithHeaders = res.headers;
  587. htmlWithHeaders.body = html;
  588. return JSON.stringify(htmlWithHeaders);
  589. } else {
  590. return html
  591. }
  592. }
  593. fetch = request;
  594. print = function (data) {
  595. data = data || '';
  596. if (typeof (data) !== 'string') {
  597. try {
  598. data = JSON.stringify(data);
  599. } catch (e) {
  600. console.log('print:' + e.message)
  601. }
  602. }
  603. console.log(data);
  604. }
  605. log = console.log;
  606. /**
  607. * 检查宝塔验证并自动跳过获取正确源码
  608. * @param html 之前获取的html
  609. * @param url 之前的来源url
  610. * @param obj 来源obj
  611. * @returns {string|DocumentFragment|*}
  612. */
  613. function checkHtml(html, url, obj) {
  614. if (/\?btwaf=/.test(html)) {
  615. let btwaf = html.match(/btwaf(.*?)"/)[1];
  616. url = url.split('#')[0] + '?btwaf' + btwaf;
  617. html = request(url, obj);
  618. }
  619. return html
  620. }
  621. /**
  622. * 带一次宝塔验证的源码获取
  623. * @param url 请求链接
  624. * @param obj 请求参数
  625. * @returns {string|DocumentFragment}
  626. */
  627. function getCode(url, obj) {
  628. let html = request(url, obj);
  629. html = checkHtml(html, url, obj);
  630. return html
  631. }
  632. /**
  633. * 源rule专用的请求方法,自动注入cookie
  634. * @param url 请求链接
  635. * @returns {string|DocumentFragment}
  636. */
  637. function getHtml(url) {
  638. let obj = {};
  639. if (rule.headers) {
  640. obj.headers = rule.headers;
  641. }
  642. let cookie = getItem(RULE_CK, '');
  643. if (cookie) {
  644. if (obj.headers && !Object.keys(obj.headers).map(it => it.toLowerCase()).includes('cookie')) {
  645. obj.headers['Cookie'] = cookie;
  646. } else if (!obj.headers) {
  647. obj.headers = { Cookie: cookie };
  648. }
  649. }
  650. let html = getCode(url, obj);
  651. return html
  652. }
  653. /**
  654. * 首页分类解析,筛选暂未实现
  655. * @param homeObj 首页传参对象
  656. * @returns {string}
  657. */
  658. function homeParse(homeObj) {
  659. fetch_params = JSON.parse(JSON.stringify(rule_fetch_params));
  660. let classes = [];
  661. if (homeObj.class_name && homeObj.class_url) {
  662. let names = homeObj.class_name.split('&');
  663. let urls = homeObj.class_url.split('&');
  664. let cnt = Math.min(names.length, urls.length);
  665. for (let i = 0; i < cnt; i++) {
  666. classes.push({
  667. 'type_id': urls[i],
  668. 'type_name': names[i]
  669. });
  670. }
  671. }
  672. if (homeObj.class_parse) {
  673. let p = homeObj.class_parse.split(';');
  674. if (p.length >= 4) {
  675. try {
  676. let html = getHtml(homeObj.MY_URL);
  677. if (html) {
  678. let list = pdfa(html, p[0]);
  679. if (list && list.length > 0) {
  680. list.forEach((it, idex) => {
  681. try {
  682. let name = pdfh(it, p[1]);
  683. if (homeObj.cate_exclude && (new RegExp(homeObj.cate_exclude).test(name))) {
  684. return;
  685. }
  686. let url = pdfh(it, p[2]);
  687. if (p[3]) {
  688. let exp = new RegExp(p[3]);
  689. url = url.match(exp)[1];
  690. }
  691. classes.push({
  692. 'type_id': url,
  693. 'type_name': name
  694. });
  695. } catch (e) {
  696. console.log(`分类列表定位第${idex}个元素正常报错:${e.message}`);
  697. }
  698. });
  699. }
  700. }
  701. } catch (e) {
  702. console.log(e.message);
  703. }
  704. }
  705. }
  706. // 排除分类
  707. classes = classes.filter(it => !homeObj.cate_exclude || !(new RegExp(homeObj.cate_exclude).test(it.type_name)));
  708. let resp = {
  709. 'class': classes
  710. };
  711. if (homeObj.filter) {
  712. resp.filters = homeObj.filter;
  713. }
  714. console.log(JSON.stringify(resp));
  715. return JSON.stringify(resp);
  716. }
  717. /**
  718. * 首页推荐列表解析
  719. * @param homeVodObj
  720. * @returns {string}
  721. */
  722. function homeVodParse(homeVodObj) {
  723. fetch_params = JSON.parse(JSON.stringify(rule_fetch_params));
  724. let d = [];
  725. MY_URL = homeVodObj.homeUrl;
  726. // setItem('MY_URL',MY_URL);
  727. console.log(MY_URL);
  728. let p = homeVodObj.推荐;
  729. if (!p || typeof (p) !== 'string') {
  730. return '{}'
  731. }
  732. p = p.trim();
  733. if (p.startsWith('js:')) {
  734. const TYPE = 'home';
  735. var input = MY_URL;
  736. const HOST = rule.host;
  737. eval(p.replace('js:', ''));
  738. d = VODS;
  739. } else {
  740. p = p.split(';');
  741. if (!homeVodObj.double && p.length < 5) {
  742. return '{}'
  743. } else if (homeVodObj.double && p.length < 6) {
  744. return '{}'
  745. }
  746. let _ps = parseTags.getParse(p[0]);
  747. _pdfa = _ps.pdfa;
  748. _pdfh = _ps.pdfh;
  749. _pd = _ps.pd;
  750. let is_json = p[0].startsWith('json:');
  751. p[0] = p[0].replace(/^(jsp:|json:|jq:)/, '');
  752. // print(p[0]);
  753. let html = getHtml(MY_URL);
  754. if (is_json) {
  755. html = dealJson(html);
  756. }
  757. try {
  758. console.log('double:' + homeVodObj.double);
  759. if (homeVodObj.double) {
  760. let items = _pdfa(html, p[0]);
  761. // console.log(items.length);
  762. for (let item of items) {
  763. // console.log(p[1]);
  764. let items2 = _pdfa(item, p[1]);
  765. // console.log(items2.length);
  766. for (let item2 of items2) {
  767. try {
  768. let title = _pdfh(item2, p[2]);
  769. let img = '';
  770. try {
  771. img = _pd(item2, p[3])
  772. } catch (e) {
  773. }
  774. let desc = _pdfh(item2, p[4]);
  775. let links = [];
  776. for (let p5 of p[5].split('+')) {
  777. let link = !homeVodObj.detailUrl ? _pd(item2, p5, MY_URL) : _pdfh(item2, p5);
  778. links.push(link);
  779. }
  780. let vod = {
  781. vod_name: title,
  782. vod_pic: img,
  783. vod_remarks: desc,
  784. vod_id: links.join('$')
  785. };
  786. // print(vod);
  787. d.push(vod);
  788. } catch (e) {
  789. console.log('首页列表处理发生错误:' + e.message);
  790. }
  791. }
  792. }
  793. } else {
  794. let items = _pdfa(html, p[0]);
  795. for (let item of items) {
  796. try {
  797. let title = _pdfh(item, p[1]);
  798. let img = '';
  799. try {
  800. img = _pd(item, p[2], MY_URL);
  801. } catch (e) {
  802. }
  803. let desc = _pdfh(item, p[3]);
  804. let links = [];
  805. for (let p5 of p[4].split('+')) {
  806. let link = !homeVodObj.detailUrl ? _pd(item, p5, MY_URL) : _pdfh(item, p5);
  807. links.push(link);
  808. }
  809. let vod = {
  810. vod_name: title,
  811. vod_pic: img,
  812. vod_remarks: desc,
  813. vod_id: links.join('$')
  814. };
  815. d.push(vod);
  816. } catch (e) {
  817. }
  818. }
  819. }
  820. } catch (e) {
  821. }
  822. }
  823. // console.log(JSON.stringify(d));
  824. return JSON.stringify({
  825. list: d
  826. })
  827. }
  828. /**
  829. * 一级分类页数据解析
  830. * @param cateObj
  831. * @returns {string}
  832. */
  833. function categoryParse(cateObj) {
  834. fetch_params = JSON.parse(JSON.stringify(rule_fetch_params));
  835. let p = cateObj.一级;
  836. if (!p || typeof (p) !== 'string') {
  837. return '{}'
  838. }
  839. let d = [];
  840. // let url = cateObj.url.replaceAll('fyclass', cateObj.tid).replaceAll('fypage', cateObj.pg);
  841. let url = cateObj.url.replaceAll('fyclass', cateObj.tid);
  842. if (rule.filter_url) {
  843. if (!/fyfilter/.test(url)) {
  844. if (!url.endsWith('&') && !rule.filter_url.startsWith('&')) {
  845. url += '&'
  846. }
  847. url += rule.filter_url;
  848. } else {
  849. url = url.replace('fyfilter', rule.filter_url);
  850. }
  851. // console.log('filter:'+cateObj.filter);
  852. let fl = cateObj.filter ? cateObj.extend : {};
  853. let new_url;
  854. new_url = cheerio.jinja2(url, { fl: fl });
  855. // if (/object Object/.test(new_url)) {
  856. // new_url = drT.renderText(url, fl);
  857. // }
  858. url = new_url;
  859. }
  860. if (/fypage/.test(url)) {
  861. if (url.includes('(') && url.includes(')')) {
  862. let url_rep = url.match(/.*?\((.*)\)/)[1];
  863. // console.log(url_rep);
  864. let cnt_page = url_rep.replaceAll('fypage', cateObj.pg);
  865. // console.log(cnt_page);
  866. let cnt_pg = eval(cnt_page);
  867. // console.log(cnt_pg);
  868. url = url.replaceAll(url_rep, cnt_pg).replaceAll('(', '').replaceAll(')', '');
  869. } else {
  870. url = url.replaceAll('fypage', cateObj.pg);
  871. }
  872. }
  873. if (cateObj.pg === 1 && url.includes('[') && url.includes(']')) {
  874. url = url.split('[')[1].split(']')[0];
  875. }
  876. MY_URL = url;
  877. // setItem('MY_URL',MY_URL);
  878. console.log(MY_URL);
  879. p = p.trim();
  880. const MY_CATE = cateObj.tid;
  881. if (p.startsWith('js:')) {
  882. var MY_FL = cateObj.extend;
  883. const TYPE = 'cate';
  884. var input = MY_URL;
  885. const MY_PAGE = cateObj.pg;
  886. var desc = '';
  887. eval(p.trim().replace('js:', ''));
  888. d = VODS;
  889. } else {
  890. p = p.split(';');
  891. if (p.length < 5) {
  892. return '{}'
  893. }
  894. let _ps = parseTags.getParse(p[0]);
  895. _pdfa = _ps.pdfa;
  896. _pdfh = _ps.pdfh;
  897. _pd = _ps.pd;
  898. let is_json = p[0].startsWith('json:');
  899. p[0] = p[0].replace(/^(jsp:|json:|jq:)/, '');
  900. try {
  901. let html = getHtml(MY_URL);
  902. if (html) {
  903. if (is_json) {
  904. html = dealJson(html);
  905. }
  906. let list = _pdfa(html, p[0]);
  907. list.forEach(it => {
  908. let links = p[4].split('+').map(p4 => {
  909. return !rule.detailUrl ? _pd(it, p4, MY_URL) : _pdfh(it, p4);
  910. });
  911. let link = links.join('$');
  912. let vod_id = rule.detailUrl ? MY_CATE + '$' + link : link;
  913. d.push({
  914. 'vod_id': vod_id,
  915. 'vod_name': _pdfh(it, p[1]).replace(/\n|\t/g, '').trim(),
  916. 'vod_pic': _pd(it, p[2], MY_URL),
  917. 'vod_remarks': _pdfh(it, p[3]).replace(/\n|\t/g, '').trim(),
  918. });
  919. });
  920. }
  921. } catch (e) {
  922. console.log(e.message);
  923. }
  924. }
  925. // print(d);
  926. return d.length < 1 ? '{}' : JSON.stringify({
  927. 'page': parseInt(cateObj.pg),
  928. 'pagecount': 999,
  929. 'limit': 20,
  930. 'total': 999,
  931. 'list': d,
  932. });
  933. }
  934. /**
  935. * 搜索列表数据解析
  936. * @param searchObj
  937. * @returns {string}
  938. */
  939. function searchParse(searchObj) {
  940. fetch_params = JSON.parse(JSON.stringify(rule_fetch_params));
  941. let d = [];
  942. if (!searchObj.searchUrl) {
  943. return '{}'
  944. }
  945. let p = searchObj.搜索 === '*' && rule.一级 ? rule.一级 : searchObj.搜索;
  946. if (!p || typeof (p) !== 'string') {
  947. return '{}'
  948. }
  949. p = p.trim();
  950. let url = searchObj.searchUrl.replaceAll('**', searchObj.wd).replaceAll('fypage', searchObj.pg);
  951. MY_URL = url;
  952. console.log(MY_URL);
  953. // setItem('MY_URL',MY_URL);
  954. if (p.startsWith('js:')) {
  955. const TYPE = 'search';
  956. const MY_PAGE = searchObj.pg;
  957. const KEY = searchObj.wd;
  958. var input = MY_URL;
  959. var detailUrl = rule.detailUrl || '';
  960. eval(p.trim().replace('js:', ''));
  961. d = VODS;
  962. } else {
  963. p = p.split(';');
  964. if (p.length < 5) {
  965. return '{}'
  966. }
  967. let _ps = parseTags.getParse(p[0]);
  968. _pdfa = _ps.pdfa;
  969. _pdfh = _ps.pdfh;
  970. _pd = _ps.pd;
  971. let is_json = p[0].startsWith('json:');
  972. p[0] = p[0].replace(/^(jsp:|json:|jq:)/, '');
  973. try {
  974. let html = getHtml(MY_URL);
  975. if (html) {
  976. if (/系统安全验证|输入验证码/.test(html)) {
  977. let cookie = verifyCode(MY_URL);
  978. if (cookie) {
  979. console.log(`本次成功过验证,cookie:${cookie}`);
  980. setItem(RULE_CK, cookie);
  981. } else {
  982. console.log(`本次自动过搜索验证失败,cookie:${cookie}`);
  983. }
  984. // obj.headers['Cookie'] = cookie;
  985. html = getHtml(MY_URL);
  986. }
  987. if (!html.includes(searchObj.wd)) {
  988. console.log('搜索结果源码未包含关键字,疑似搜索失败,正为您打印结果源码');
  989. console.log(html);
  990. }
  991. if (is_json) {
  992. html = dealJson(html);
  993. }
  994. let list = _pdfa(html, p[0]);
  995. list.forEach(it => {
  996. let links = p[4].split('+').map(p4 => {
  997. return !rule.detailUrl ? _pd(it, p4, MY_URL) : _pdfh(it, p4)
  998. });
  999. let link = links.join('$');
  1000. let ob = {
  1001. 'vod_id': link,
  1002. 'vod_name': _pdfh(it, p[1]).replace(/\n|\t/g, '').trim(),
  1003. 'vod_pic': _pd(it, p[2], MY_URL),
  1004. 'vod_remarks': _pdfh(it, p[3]).replace(/\n|\t/g, '').trim(),
  1005. };
  1006. if (p.length > 5 && p[5]) {
  1007. ob.vod_content = _pdfh(it, p[5]);
  1008. }
  1009. d.push(ob);
  1010. });
  1011. }
  1012. } catch (e) {
  1013. return '{}'
  1014. }
  1015. }
  1016. return JSON.stringify({
  1017. 'page': parseInt(searchObj.pg),
  1018. 'pagecount': 10,
  1019. 'limit': 20,
  1020. 'total': 100,
  1021. 'list': d,
  1022. });
  1023. }
  1024. /**
  1025. * 二级详情页数据解析
  1026. * @param detailObj
  1027. * @returns {string}
  1028. */
  1029. function detailParse(detailObj) {
  1030. fetch_params = JSON.parse(JSON.stringify(rule_fetch_params));
  1031. let vod = {
  1032. vod_id: detailObj.orId, //"id",
  1033. vod_name: "片名",
  1034. vod_pic: "",
  1035. type_name: "剧情",
  1036. vod_year: "年份",
  1037. vod_area: "地区",
  1038. vod_remarks: "更新信息",
  1039. vod_actor: "主演",
  1040. vod_director: "导演",
  1041. vod_content: "简介"
  1042. };
  1043. let p = detailObj.二级;
  1044. let url = detailObj.url;
  1045. let detailUrl = detailObj.detailUrl;
  1046. let fyclass = detailObj.fyclass;
  1047. let tab_exclude = detailObj.tab_exclude;
  1048. let html = detailObj.html || '';
  1049. MY_URL = url;
  1050. console.log(MY_URL);
  1051. // setItem('MY_URL',MY_URL);
  1052. if (p === '*') {
  1053. vod.vod_play_from = '道长在线';
  1054. vod.vod_remarks = detailUrl;
  1055. vod.vod_actor = '没有二级,只有一级链接直接嗅探播放';
  1056. vod.vod_content = MY_URL;
  1057. vod.vod_play_url = '嗅探播放$' + MY_URL;
  1058. } else if (typeof (p) === 'string' && p.trim().startsWith('js:')) {
  1059. const TYPE = 'detail';
  1060. var input = MY_URL;
  1061. eval(p.trim().replace('js:', ''));
  1062. vod = VOD;
  1063. console.log(JSON.stringify(vod));
  1064. } else if (p && typeof (p) === 'object') {
  1065. if (!html) {
  1066. html = getHtml(MY_URL);
  1067. }
  1068. let _ps;
  1069. if (p.is_json) {
  1070. _ps = parseTags.json;
  1071. html = dealJson(html);
  1072. } else if (p.is_jsp) {
  1073. _ps = parseTags.jsp;
  1074. } else if (p.is_jq) {
  1075. _ps = parseTags.jq;
  1076. } else {
  1077. _ps = parseTags.jq;
  1078. }
  1079. _pdfa = _ps.pdfa;
  1080. _pdfh = _ps.pdfh;
  1081. _pd = _ps.pd;
  1082. if (p.title) {
  1083. let p1 = p.title.split(';');
  1084. vod.vod_name = _pdfh(html, p1[0]).replace(/\n|\t/g, '').trim();
  1085. let type_name = p1.length > 1 ? _pdfh(html, p1[1]).replace(/\n|\t/g, '').replace(/ /g, '').trim() : '';
  1086. vod.type_name = type_name || vod.type_name;
  1087. }
  1088. if (p.desc) {
  1089. try {
  1090. let p1 = p.desc.split(';');
  1091. vod.vod_remarks = _pdfh(html, p1[0]).replace(/\n|\t/g, '').trim();
  1092. vod.vod_year = p1.length > 1 ? _pdfh(html, p1[1]).replace(/\n|\t/g, '').trim() : '';
  1093. vod.vod_area = p1.length > 2 ? _pdfh(html, p1[2]).replace(/\n|\t/g, '').trim() : '';
  1094. // vod.vod_actor = p1.length > 3 ? _pdfh(html, p1[3]).replaceAll('\n', ' ').trim():'';
  1095. vod.vod_actor = p1.length > 3 ? _pdfh(html, p1[3]).replace(/\n|\t/g, '').trim() : '';
  1096. vod.vod_director = p1.length > 4 ? _pdfh(html, p1[4]).replace(/\n|\t/g, '').trim() : '';
  1097. }
  1098. catch (e) {
  1099. }
  1100. }
  1101. if (p.content) {
  1102. try {
  1103. let p1 = p.content.split(';');
  1104. vod.vod_content = _pdfh(html, p1[0]).replace(/\n|\t/g, '').trim();
  1105. }
  1106. catch (e) { }
  1107. }
  1108. if (p.img) {
  1109. try {
  1110. let p1 = p.img.split(';');
  1111. vod.vod_pic = _pd(html, p1[0], MY_URL);
  1112. }
  1113. catch (e) { }
  1114. }
  1115. let vod_play_from = '$$$';
  1116. let playFrom = [];
  1117. if (p.重定向 && p.重定向.startsWith('js:')) {
  1118. html = eval(p.重定向.replace('js:', ''));
  1119. }
  1120. // console.log(2);
  1121. if (p.tabs) {
  1122. let p_tab = p.tabs.split(';')[0];
  1123. console.log(p_tab);
  1124. let vHeader = _pdfa(html, p_tab);
  1125. console.log(vHeader.length);
  1126. for (let v of vHeader) {
  1127. let v_title = _pdfh(v, 'body&&Text');
  1128. console.log(v_title);
  1129. if (tab_exclude && (new RegExp(tab_exclude)).test(v_title)) {
  1130. continue;
  1131. }
  1132. playFrom.push(v_title);
  1133. }
  1134. console.log(JSON.stringify(playFrom));
  1135. } else {
  1136. playFrom = ['道长在线']
  1137. }
  1138. vod.vod_play_from = playFrom.join(vod_play_from);
  1139. // console.log(3);
  1140. let vod_play_url = '$$$';
  1141. let vod_tab_list = [];
  1142. if (p.lists) {
  1143. for (let i = 0; i < playFrom.length; i++) {
  1144. let tab_name = playFrom[i];
  1145. let tab_ext = p.tabs.split(';').length > 1 ? p.tabs.split(';')[1] : '';
  1146. let p1 = p.lists.replaceAll('#idv', tab_name).replaceAll('#id', i);
  1147. tab_ext = tab_ext.replaceAll('#idv', tab_name).replaceAll('#id', i);
  1148. console.log(p1);
  1149. // console.log(html);
  1150. let vodList = [];
  1151. try {
  1152. vodList = _pdfa(html, p1);
  1153. console.log('len(vodList):' + vodList.length);
  1154. } catch (e) {
  1155. // console.log(e.message);
  1156. }
  1157. let new_vod_list = [];
  1158. let tabName = tab_ext ? _pdfh(html, tab_ext) : tab_name;
  1159. console.log(tabName);
  1160. // console.log('cheerio解析Text');
  1161. vodList.forEach(it => {
  1162. // 请注意,这里要固定pdfh解析body&&Text,不需要下划线,没写错
  1163. // new_vod_list.push(pdfh(it,'body&&Text')+'$'+_pd(it,'a&&href',MY_URL));
  1164. new_vod_list.push(cheerio.load(it).text() + '$' + _pd(it, 'a&&href', MY_URL));
  1165. });
  1166. let vlist = new_vod_list.join('#');
  1167. vod_tab_list.push(vlist);
  1168. }
  1169. }
  1170. vod.vod_play_url = vod_tab_list.join(vod_play_url);
  1171. }
  1172. // print(vod);
  1173. return JSON.stringify({
  1174. list: [vod]
  1175. })
  1176. }
  1177. /**
  1178. * 选集播放点击事件解析
  1179. * @param playObj
  1180. * @returns {string}
  1181. */
  1182. function playParse(playObj) {
  1183. fetch_params = JSON.parse(JSON.stringify(rule_fetch_params));
  1184. MY_URL = playObj.url;
  1185. if (!/http/.test(MY_URL)) {
  1186. try {
  1187. MY_URL = base64Decode(MY_URL);
  1188. } catch (e) { }
  1189. }
  1190. MY_URL = decodeURIComponent(MY_URL);
  1191. var input = MY_URL;//注入给免嗅js
  1192. let common_play = {
  1193. parse: 1,
  1194. url: input
  1195. };
  1196. let lazy_play;
  1197. if (!rule.play_parse || !rule.lazy) {
  1198. lazy_play = common_play;
  1199. } else if (rule.play_parse && rule.lazy && typeof (rule.lazy) === 'string') {
  1200. try {
  1201. let lazy_code = rule.lazy.replace('js:', '').trim();
  1202. print('开始执行js免嗅=>' + lazy_code);
  1203. eval(lazy_code);
  1204. lazy_play = typeof (input) === 'object' ? input : {
  1205. parse: 1,
  1206. jx: 1,
  1207. url: input
  1208. };
  1209. } catch (e) {
  1210. print('js免嗅错误:' + e.message);
  1211. lazy_play = common_play;
  1212. }
  1213. } else {
  1214. lazy_play = common_play;
  1215. }
  1216. console.log(JSON.stringify(lazy_play));
  1217. return JSON.stringify(lazy_play);
  1218. }
  1219. /**
  1220. * js源预处理特定返回对象中的函数
  1221. * @param ext
  1222. */
  1223. function init(ext) {
  1224. console.log('init');
  1225. try {
  1226. // make shared jsContext happy
  1227. if (typeof (globalThis.mubanJs) === 'undefined') {
  1228. let mubanJs = request('https://gitcode.net/qq_32394351/dr_py/-/raw/master/js/模板.js', { 'User-Agent': MOBILE_UA });
  1229. mubanJs = mubanJs.replace('export default', '(function() {return muban;}()) // export default');
  1230. globalThis.mubanJs = mubanJs;
  1231. }
  1232. let muban = eval(globalThis.mubanJs);
  1233. if (typeof ext == 'object') {
  1234. rule = ext;
  1235. if (rule.template) {
  1236. rule = Object.assign(muban[rule.template], rule);
  1237. }
  1238. } else if (typeof ext == 'string') {
  1239. if (ext.startsWith('http')) {
  1240. let js = request(ext, { 'method': 'GET' });
  1241. if (js) {
  1242. eval(js.replace('var rule', 'rule'));
  1243. }
  1244. }
  1245. } else {
  1246. eval(ext.replace('var rule', 'rule'));
  1247. }
  1248. /** 处理一下 rule规则关键字段没传递的情况 **/
  1249. let rule_cate_excludes = (rule.cate_exclude || '').split('|').filter(it => it.trim());
  1250. let rule_tab_excludes = (rule.tab_exclude || '').split('|').filter(it => it.trim());
  1251. rule_cate_excludes = rule_cate_excludes.concat(CATE_EXCLUDE.split('|').filter(it => it.trim()));
  1252. rule_tab_excludes = rule_tab_excludes.concat(TAB_EXCLUDE.split('|').filter(it => it.trim()));
  1253. rule.cate_exclude = rule_cate_excludes.join('|');
  1254. rule.tab_exclude = rule_tab_excludes.join('|');
  1255. rule.host = (rule.host || '').rstrip('/');
  1256. rule.url = rule.url || '';
  1257. rule.double = rule.double || false;
  1258. rule.homeUrl = rule.homeUrl || '';
  1259. rule.detailUrl = rule.detailUrl || '';
  1260. rule.searchUrl = rule.searchUrl || '';
  1261. rule.homeUrl = rule.host && rule.homeUrl ? urljoin(rule.host, rule.homeUrl) : (rule.homeUrl || rule.host);
  1262. rule.detailUrl = rule.host && rule.detailUrl ? urljoin(rule.host, rule.detailUrl) : rule.detailUrl;
  1263. if (rule.url.includes('[') && rule.url.includes(']')) {
  1264. let u1 = rule.url.split('[')[0]
  1265. let u2 = rule.url.split('[')[1].split(']')[0]
  1266. rule.url = rule.host && rule.url ? urljoin(rule.host, u1) + '[' + urljoin(rule.host, u2) + ']' : rule.url;
  1267. } else {
  1268. rule.url = rule.host && rule.url ? urljoin(rule.host, rule.url) : rule.url;
  1269. }
  1270. rule.searchUrl = rule.host && rule.searchUrl ? urljoin(rule.host, rule.searchUrl) : rule.searchUrl;
  1271. rule.timeout = rule.timeout || 5000;
  1272. rule.encoding = rule.编码 || rule.encoding || 'utf-8';
  1273. if (rule.headers && typeof (rule.headers) === 'object') {
  1274. try {
  1275. let header_keys = Object.keys(rule.headers);
  1276. for (let k of header_keys) {
  1277. if (k.toLowerCase() === 'user-agent') {
  1278. let v = rule.headers[k];
  1279. console.log(v);
  1280. if (['MOBILE_UA', 'PC_UA', 'UC_UA', 'IOS_UA', 'UA'].includes(v)) {
  1281. rule.headers[k] = eval(v);
  1282. }
  1283. }
  1284. }
  1285. } catch (e) {
  1286. console.log('处理headers发生错误:' + e.message);
  1287. }
  1288. }
  1289. // print(rule.headers);
  1290. rule_fetch_params = { 'headers': rule.headers || false, 'timeout': rule.timeout, 'encoding': rule.encoding };
  1291. oheaders = rule.headers || {};
  1292. RKEY = typeof (key) !== 'undefined' && key ? key : 'drpy_' + (rule.title || rule.host);
  1293. init_test();
  1294. } catch (e) {
  1295. console.log('init_test发生错误:' + e.message);
  1296. }
  1297. }
  1298. /**
  1299. * js源获取首页分类和筛选特定返回对象中的函数
  1300. * @param filter 筛选条件字典对象
  1301. * @returns {string}
  1302. */
  1303. function home(filter) {
  1304. console.log("home");
  1305. let homeObj = {
  1306. filter: rule.filter || false,
  1307. MY_URL: rule.homeUrl,
  1308. class_name: rule.class_name || '',
  1309. class_url: rule.class_url || '',
  1310. class_parse: rule.class_parse || '',
  1311. cate_exclude: rule.cate_exclude,
  1312. };
  1313. return homeParse(homeObj);
  1314. }
  1315. /**
  1316. * js源获取首页推荐数据列表特定返回对象中的函数
  1317. * @param params
  1318. * @returns {string}
  1319. */
  1320. function homeVod(params) {
  1321. let homeVodObj = {
  1322. 推荐: rule.推荐,
  1323. double: rule.double,
  1324. homeUrl: rule.homeUrl,
  1325. detailUrl: rule.detailUrl
  1326. };
  1327. return homeVodParse(homeVodObj)
  1328. // return "{}";
  1329. }
  1330. /**
  1331. * js源获取分类页一级数据列表特定返回对象中的函数
  1332. * @param tid 分类id
  1333. * @param pg 页数
  1334. * @param filter 当前选中的筛选条件
  1335. * @param extend 扩展
  1336. * @returns {string}
  1337. */
  1338. function category(tid, pg, filter, extend) {
  1339. let cateObj = {
  1340. url: rule.url,
  1341. 一级: rule.一级,
  1342. tid: tid,
  1343. pg: parseInt(pg),
  1344. filter: filter,
  1345. extend: extend
  1346. };
  1347. // console.log(JSON.stringify(extend));
  1348. return categoryParse(cateObj)
  1349. }
  1350. /**
  1351. * js源获取二级详情页数据特定返回对象中的函数
  1352. * @param vod_url 一级列表中的vod_id或者是带分类的自拼接 vod_id 如 fyclass$vod_id
  1353. * @returns {string}
  1354. */
  1355. function detail(vod_url) {
  1356. let orId = vod_url;
  1357. let fyclass = '';
  1358. if (vod_url.indexOf('$') > -1) {
  1359. let tmp = vod_url.split('$');
  1360. fyclass = tmp[0];
  1361. vod_url = tmp[1];
  1362. }
  1363. let detailUrl = vod_url;
  1364. let url;
  1365. if (!detailUrl.startsWith('http') && !detailUrl.includes('/')) {
  1366. url = rule.detailUrl.replaceAll('fyid', detailUrl).replaceAll('fyclass', fyclass);
  1367. } else if (detailUrl.includes('/')) {
  1368. url = urljoin(rule.homeUrl, detailUrl);
  1369. } else {
  1370. url = detailUrl
  1371. }
  1372. let detailObj = {
  1373. orId: orId,
  1374. url: url,
  1375. 二级: rule.二级,
  1376. detailUrl: detailUrl,
  1377. fyclass: fyclass,
  1378. tab_exclude: rule.tab_exclude,
  1379. }
  1380. return detailParse(detailObj)
  1381. }
  1382. /**
  1383. * js源选集按钮播放点击事件特定返回对象中的函数
  1384. * @param flag 线路名
  1385. * @param id 播放按钮的链接
  1386. * @param flags 全局配置的flags是否需要解析的标识列表
  1387. * @returns {string}
  1388. */
  1389. function play(flag, id, flags) {
  1390. let playObj = {
  1391. url: id,
  1392. flag: flag,
  1393. flags: flags
  1394. }
  1395. return playParse(playObj);
  1396. }
  1397. /**
  1398. * js源搜索返回的数据列表特定返回对象中的函数
  1399. * @param wd 搜索关键字
  1400. * @param quick 是否来自快速搜索
  1401. * @returns {string}
  1402. */
  1403. function search(wd, quick) {
  1404. let searchObj = {
  1405. searchUrl: rule.searchUrl,
  1406. 搜索: rule.搜索,
  1407. wd: wd,
  1408. //pg: pg,
  1409. pg: 1,
  1410. quick: quick,
  1411. };
  1412. // console.log(JSON.stringify(searchObj));
  1413. return searchParse(searchObj)
  1414. }
  1415. function DRPY() {//导出函数
  1416. return {
  1417. init: init,
  1418. home: home,
  1419. homeVod: homeVod,
  1420. category: category,
  1421. detail: detail,
  1422. play: play,
  1423. search: search,
  1424. }
  1425. }
  1426. let pd = pD;
  1427. // 导出函数对象
  1428. __JS_SPIDER__ = {
  1429. init: init,
  1430. home: home,
  1431. homeVod: homeVod,
  1432. category: category,
  1433. detail: detail,
  1434. play: play,
  1435. search: search,
  1436. DRPY: DRPY
  1437. }