import { Crypto, _ } from 'assets://js/lib/cat.js' let host = ''; let header = { 'User-Agent': 'okhttp/3.12.11' }; let siteKey = ''; let siteType = ''; let siteJx = ''; const urlPattern1 = /api\.php\/.*?\/vod/; const urlPattern2 = /api\.php\/.+?\.vod/; const parsePattern = /\/.+\\?.+=/; const parsePattern1 = /.*(url|v|vid|php\?id)=/; const parsePattern2 = /https?:\/\/[^\/]*/; const htmlVideoKeyMatch = [ /player=new/, /<div id="video"/, /<div id="[^"]*?player"/, /\/\/视频链接/, /HlsJsPlayer\(/, /<iframe[\s\S]*?src="[^"]+?"/, /<video[\s\S]*?src="[^"]+?"/, ]; async function init(cfg) { siteKey = cfg.skey; siteType = cfg.stype; host = cfg.ext; if (cfg.ext.hasOwnProperty('host')) { // for custom jx host = cfg.ext.host; siteJx = cfg.ext; } }; async function request(reqUrl, ua, timeout = 60000) { let res = await req(reqUrl, { method: 'get', headers: ua ? ua : {'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.114 Safari/537.36'}, timeout: timeout, }); return res.content; } async function home(filter) { try { let url = getCateUrl(host); let jsonArray = null; if (url) { const json = await request(url, getHeaders(url)); const obj = JSON.parse(json); if (obj.hasOwnProperty("list") && Array.isArray(obj.list)) { jsonArray = obj.list; } else if ( obj.hasOwnProperty("data") && obj.data.hasOwnProperty("list") && Array.isArray(obj.data.list) ) { jsonArray = obj.data.list; } else if (obj.hasOwnProperty("data") && Array.isArray(obj.data)) { jsonArray = obj.data; } } else { // 通过filter列表读分类 const filterStr = getFilterTypes(url, null); const classes = filterStr.split("\n")[0].split("+"); jsonArray = []; for (let i = 1; i < classes.length; i++) { const kv = classes[i].trim().split("="); if (kv.length < 2) continue; const newCls = { type_name: kv[0].trim(), type_id: kv[1].trim(), }; jsonArray.push(newCls); } } const result = { class: [] }; if (jsonArray != null) { for (let i = 0; i < jsonArray.length; i++) { const jObj = jsonArray[i]; const typeName = jObj.type_name; if (isBan(typeName)) continue; const typeId = jObj.type_id; const newCls = { type_id: typeId, type_name: typeName, }; const typeExtend = jObj.type_extend; if (filter) { const filterStr = getFilterTypes(url, typeExtend); const filters = filterStr.split("\n"); const filterArr = []; for (let k = (url) ? 1 : 0; k < filters.length; k++) { const l = filters[k].trim(); if (!l) continue; const oneLine = l.split("+"); let type = oneLine[0].trim(); let typeN = type; if (type.includes("筛选")) { type = type.replace(/筛选/g, ""); if (type === "class") typeN = "类型"; else if (type === "area") typeN = "地区"; else if (type === "lang") typeN = "语言"; else if (type === "year") typeN = "年份"; } const jOne = { key: type, name: typeN, value: [], }; for (let j = 1; j < oneLine.length; j++) { const kv = oneLine[j].trim(); const sp = kv.indexOf("="); if (sp === -1) { if (isBan(kv)) continue; jOne.value.push({ n: kv, v: kv }); } else { const n = kv.substring(0, sp); if (isBan(n)) continue; jOne.value.push({ n: n.trim(), v: kv.substring(sp + 1).trim(), }); } } filterArr.push(jOne); } if (!result.hasOwnProperty("filters")) { result.filters = {}; } result.filters[typeId] = filterArr; } result.class.push(newCls); } } return JSON.stringify(result); } catch (e) { } return ""; } async function homeVod() { try { const apiUrl = host; let url = getRecommendUrl(apiUrl); let isTV = false; if (!url) { url = getCateFilterUrlPrefix(apiUrl) + "movie&page=1&area=&type=&start="; isTV = true; } const json = await request(url, getHeaders(url)); const obj = JSON.parse(json); const videos = []; if (isTV) { const jsonArray = obj.data; for (let i = 0; i < jsonArray.length; i++) { const vObj = jsonArray[i]; const v = { vod_id: vObj.nextlink, vod_name: vObj.title, vod_pic: vObj.pic, vod_remarks: vObj.state, }; videos.push(v); } } else { const arrays = []; findJsonArray(obj, "vlist", arrays); if (arrays.length === 0) { findJsonArray(obj, "vod_list", arrays); } const ids = []; for (const jsonArray of arrays) { for (let i = 0; i < jsonArray.length; i++) { const vObj = jsonArray[i]; const vid = vObj.vod_id; if (ids.includes(vid)) continue; ids.push(vid); const v = { vod_id: vid, vod_name: vObj.vod_name, vod_pic: vObj.vod_pic, vod_remarks: vObj.vod_remarks, }; videos.push(v); } } } const result = { list: videos, }; return JSON.stringify(result); } catch (e) { } return ""; } async function category(tid, pg, filter, extend) { try { const apiUrl = host; let url = getCateFilterUrlPrefix(apiUrl) + tid + getCateFilterUrlSuffix(apiUrl); url = url.replace(/#PN#/g, pg); url = url.replace(/筛选class/g, extend?.class ?? ""); url = url.replace(/筛选area/g, extend?.area ?? ""); url = url.replace(/筛选lang/g, extend?.lang ?? ""); url = url.replace(/筛选year/g, extend?.year ?? ""); url = url.replace(/排序/g, extend?.排序 ?? ""); const json = await request(url, getHeaders(url)); const obj = JSON.parse(json); let totalPg = Infinity; try { if (obj.totalpage !== undefined && typeof obj.totalpage === "number") { totalPg = obj.totalpage; } else if ( obj.pagecount !== undefined && typeof obj.pagecount === "number" ) { totalPg = obj.pagecount; } else if ( obj.data !== undefined && typeof obj.data === "object" && obj.data.total !== undefined && typeof obj.data.total === "number" && obj.data.limit !== undefined && typeof obj.data.limit === "number" ) { const limit = obj.data.limit; const total = obj.data.total; totalPg = total % limit === 0 ? total / limit : Math.floor(total / limit) + 1; } } catch (e) { } const jsonArray = obj.list !== undefined ? obj.list : obj.data !== undefined && obj.data.list !== undefined ? obj.data.list : obj.data; const videos = []; if (jsonArray !== undefined) { for (let i = 0; i < jsonArray.length; i++) { const vObj = jsonArray[i]; const v = { vod_id: vObj.vod_id !== undefined ? vObj.vod_id : vObj.nextlink, vod_name: vObj.vod_name !== undefined ? vObj.vod_name : vObj.title, vod_pic: vObj.vod_pic !== undefined ? vObj.vod_pic : vObj.pic, vod_remarks: vObj.vod_remarks !== undefined ? vObj.vod_remarks : vObj.state, }; videos.push(v); } } const result = { page: pg, pagecount: totalPg, limit: 90, total: Infinity, list: videos, }; return JSON.stringify(result); } catch (e) { SpiderDebug.log(e); } return ""; } async function detail(ids) { try { const apiUrl = host; const url = getPlayUrlPrefix(apiUrl) + ids; const json = await request(url, getHeaders(url)); const obj = JSON.parse(json); const result = { list: [], }; const vod = {}; genPlayList(apiUrl, obj, json, vod, ids); result.list.push(vod); return JSON.stringify(result); } catch (e) { } return ""; } const parseUrlMap = new Map(); function genPlayList(URL, object, json, vod, vid) { const playUrls = []; const playFlags = []; if (URL.includes("lfytyl.com")) { const data = object.data; vod.vod_id = data.vod_id || vid; vod.vod_name = data.vod_name; vod.vod_pic = data.vod_pic; vod.type_name = data.vod_class || ""; vod.vod_year = data.vod_year || ""; vod.vod_area = data.vod_area || ""; vod.vod_remarks = data.vod_remarks || ""; vod.vod_actor = data.vod_actor || ""; vod.vod_director = data.vod_director || ""; vod.vod_content = data.vod_content || ""; vod.vod_play_from = data.vod_play_from; vod.vod_play_url = data.vod_play_url; return; } if (URL.includes("api.php/app")) { const data = object.data; vod.vod_id = data.vod_id || vid; vod.vod_name = data.vod_name; vod.vod_pic = data.vod_pic; vod.type_name = data.vod_class || ""; vod.vod_year = data.vod_year || ""; vod.vod_area = data.vod_area || ""; vod.vod_remarks = data.vod_remarks || ""; vod.vod_actor = data.vod_actor || ""; vod.vod_director = data.vod_director || ""; vod.vod_content = data.vod_content || ""; const vodUrlWithPlayer = data.vod_url_with_player; for (let i = 0; i < vodUrlWithPlayer.length; i++) { const from = vodUrlWithPlayer[i]; let flag = from.code.trim(); if (flag === "") flag = from.name.trim(); playFlags.push(flag); playUrls.push(from.url); let purl = from.parse_api; const parseUrls = parseUrlMap.get(flag) || []; if (purl && !parseUrls.includes(purl)) { parseUrls.push(purl); } parseUrlMap.set(flag, parseUrls); } } else if (URL.includes("xgapp")) { const data = object.data.vod_info; vod.vod_id = data.vod_id || vid; vod.vod_name = data.vod_name; vod.vod_pic = data.vod_pic; vod.type_name = data.vod_class || ""; vod.vod_year = data.vod_year || ""; vod.vod_area = data.vod_area || ""; vod.vod_remarks = data.vod_remarks || ""; vod.vod_actor = data.vod_actor || ""; vod.vod_director = data.vod_director || ""; vod.vod_content = data.vod_content || ""; const vodUrlWithPlayer = data.vod_url_with_player; for (let i = 0; i < vodUrlWithPlayer.length; i++) { const from = vodUrlWithPlayer[i]; let flag = from.code.trim(); if (flag === "") flag = from.name.trim(); playFlags.push(flag); playUrls.push(from.url); const purl = from.parse_api.trim(); const parseUrls = parseUrlMap.get(flag) || []; if (purl && !parseUrls.includes(purl)) { parseUrls.push(purl); } parseUrlMap.set(flag, parseUrls); } } else if (URL.includes(".vod")) { const data = object.data; vod.vod_id = data.vod_id || vid; vod.vod_name = data.vod_name; vod.vod_pic = data.vod_pic; vod.type_name = data.vod_class || ""; vod.vod_year = data.vod_year || ""; vod.vod_area = data.vod_area || ""; vod.vod_remarks = data.vod_remarks || ""; vod.vod_actor = data.vod_actor || ""; vod.vod_director = data.vod_director || ""; vod.vod_content = data.vod_content || ""; const vodUrlWithPlayer = data.vod_play_list; for (let i = 0; i < vodUrlWithPlayer.length; i++) { const from = vodUrlWithPlayer[i]; let flag = from.player_info.from.trim(); if (flag === "") flag = from.player_info.show.trim(); playFlags.push(flag); playUrls.push(from.url); try { const parses = []; const parse1 = from.player_info.parse.split(","); const parse2 = from.player_info.parse2.split(","); parses.push(...parse1, ...parse2); const parseUrls = parseUrlMap.get(flag) || []; for (const purl of parses) { if (purl.includes("http")) { const match = purl.match(parsePattern1); if (match) { parseUrls.push(match[0]); } } else if (purl.includes("//")) { const match = purl.match(parsePattern1); if (match) { parseUrls.push("http:" + match[0]); } } else { const urlMatch = URL.match(parsePattern2); if (urlMatch) { const match = URL.match(parsePattern1); if (match) { parseUrls.push(urlMatch[0] + match[0]); } } } if (purl.includes("..")) purl = purl.replace(/\.\./g, ".").trim(); if (purl && !parseUrls.includes(purl)) { parseUrls.push(purl); } } parseUrlMap.set(flag, parseUrls); } catch (e) { } } } else if (URLPattern1.matcher(URL).find()) { // Same implementation as the previous cases } vod.vod_play_from = playFlags.join("$$$"); vod.vod_play_url = playUrls.join("$$$"); } async function play(flag, id, vipFlags) { try { // let parseUrls = parseUrlMap.get(flag); let parseUrls = siteJx[flag]; // custom sitejx if (!parseUrls) { if (siteJx.hasOwnProperty('*')) { // all jx parseUrls = siteJx['*']; } else { parseUrls = []; } } if (parseUrls.length > 0) { const result = await getFinalVideo(flag, parseUrls, id); if (result !== null) { return JSON.stringify(result); } } if (isVideo(id)) { const result = { parse: 0, playUrl: "", url: id }; return JSON.stringify(result); } else { const result = { parse: 1, jx: "1", url: id }; return JSON.stringify(result); } } catch (e) { // Handle any error here } return ""; } async function search(key, quick) { try { const apiUrl = host; const url = getSearchUrl(apiUrl, encodeURIComponent(key)); const json = await request(url, getHeaders(url)); const obj = JSON.parse(json); let jsonArray = null; const videos = []; if (obj.list instanceof Array) { jsonArray = obj.list; } else if (obj.data instanceof Object && obj.data.list instanceof Array) { jsonArray = obj.data.list; } else if (obj.data instanceof Array) { jsonArray = obj.data; } if (jsonArray !== null) { for (const vObj of jsonArray) { if (vObj.vod_id) { const v = { vod_id: vObj.vod_id, vod_name: vObj.vod_name, vod_pic: vObj.vod_pic, vod_remarks: vObj.vod_remarks }; videos.push(v); } else { const v = { vod_id: vObj.nextlink, vod_name: vObj.title, vod_pic: vObj.pic, vod_remarks: vObj.state }; videos.push(v); } } } const result = { list: videos }; return JSON.stringify(result); } catch (error) { } return ""; } async function getFinalVideo(flag, parseUrls, url) { let htmlPlayUrl = ""; for (const parseUrl of parseUrls) { if (parseUrl === "" || parseUrl === "null") { continue; } const playUrl = parseUrl + url; const content = await request(playUrl, null, 10000); // 10秒请求,能更好过滤webjx let tryJson = null; try { tryJson = jsonParse(url, content); } catch (error) { } if (tryJson !== null && tryJson.hasOwnProperty("url") && tryJson.hasOwnProperty("header")) { tryJson.header = JSON.stringify(tryJson.header); return tryJson; } if (content.includes("<html")) { let sniffer = false; for (const p of htmlVideoKeyMatch) { if (p.test(content)) { sniffer = true; break; } } if (sniffer) { htmlPlayUrl = parseUrl; } } } if (htmlPlayUrl !== "") { // 不支持sniffer const result = { parse: 0, playUrl: "", url: url }; return JSON.stringify(result); } return null; } function jsonParse(input, json) { try { // 处理解析接口返回的报文,如果返回的报文中包含header信息,就加到返回值中 let jsonPlayData = JSON.parse(json); // 处理293的解析结果url在data字段的解析 if (jsonPlayData.hasOwnProperty("data") && typeof jsonPlayData.data === "object" && !jsonPlayData.hasOwnProperty("url")) { jsonPlayData = jsonPlayData.data; } let url = jsonPlayData.url; if (url.startsWith("//")) { url = "https:" + url; } if (!url.trim().startsWith("http")) { return null; } if (url === input) { if (isVip(url) || !isVideoFormat(url)) { return null; } } if (isBlackVodUrl(input, url)) { return null; } let headers = {}; if (jsonPlayData.hasOwnProperty("header")) { headers = jsonPlayData.header; } else if (jsonPlayData.hasOwnProperty("Header")) { headers = jsonPlayData.Header; } else if (jsonPlayData.hasOwnProperty("headers")) { headers = jsonPlayData.headers; } else if (jsonPlayData.hasOwnProperty("Headers")) { headers = jsonPlayData.Headers; } let ua = ""; if (jsonPlayData.hasOwnProperty("user-agent")) { ua = jsonPlayData["user-agent"]; } else if (jsonPlayData.hasOwnProperty("User-Agent")) { ua = jsonPlayData["User-Agent"]; } if (ua.trim().length > 0) { headers["User-Agent"] = " " + ua; } let referer = ""; if (jsonPlayData.hasOwnProperty("referer")) { referer = jsonPlayData.referer; } else if (jsonPlayData.hasOwnProperty("Referer")) { referer = jsonPlayData.Referer; } if (referer.trim().length > 0) { headers["Referer"] = " " + referer; } headers = fixJsonVodHeader(headers, input, url); const taskResult = { header: headers, url: url, parse: "0" }; return taskResult; } catch (error) { } return null; } function isVip(url) { try { let isVip = false; const host = new URL(url).hostname; const vipWebsites = ["iqiyi.com", "v.qq.com", "youku.com", "le.com", "tudou.com", "mgtv.com", "sohu.com", "acfun.cn", "bilibili.com", "baofeng.com", "pptv.com"]; for (let b = 0; b < vipWebsites.length; b++) { if (host.includes(vipWebsites[b])) { if (vipWebsites[b] === "iqiyi.com") { // 爱奇艺需要特殊处理 if (url.includes("iqiyi.com/a_") || url.includes("iqiyi.com/w_") || url.includes("iqiyi.com/v_")) { isVip = true; break; } } else { isVip = true; break; } } } return isVip; } catch (e) { } return false; } function isBlackVodUrl(input, url) { return url.includes("973973.xyz") || url.includes(".fit:"); } function fixJsonVodHeader(headers, input, url) { if (headers === null) { headers = {}; } if (input.includes("www.mgtv.com")) { headers["Referer"] = " "; headers["User-Agent"] = " Mozilla/5.0"; } else if (url.includes("titan.mgtv")) { headers["Referer"] = " "; headers["User-Agent"] = " Mozilla/5.0"; } else if (input.includes("bilibili")) { headers["Referer"] = " https://www.bilibili.com/"; headers["User-Agent"] = " " + Misc.UaWinChrome; } return headers; } const snifferMatch = /http((?!http).){26,}?\.(m3u8|mp4|flv|avi|mkv|rm|wmv|mpg)\?.*|http((?!http).){26,}\.(m3u8|mp4|flv|avi|mkv|rm|wmv|mpg)|http((?!http).){26,}\/m3u8\?pt=m3u8.*|http((?!http).)*?default\.ixigua\.com\/.*|http((?!http).)*?cdn-tos[^\?]*|http((?!http).)*?\/obj\/tos[^\?]*|http.*?\/player\/m3u8play\.php\?url=.*|http.*?\/player\/.*?[pP]lay\.php\?url=.*|http.*?\/playlist\/m3u8\/\?vid=.*|http.*?\.php\?type=m3u8&.*|http.*?\/download.aspx\?.*|http.*?\/api\/up_api.php\?.*|https.*?\.66yk\.cn.*|http((?!http).)*?netease\.com\/file\/.*/; function isVideoFormat(url) { if (snifferMatch.test(url)) { return !url.includes("cdn-tos") || !url.includes(".js"); } return false; } function isVideo(url) { if (!url.includes(".mp4") || !url.includes(".m3u8")) { return true; } return false; } function UA(url) { if (url.includes(".vod")) { return "okhttp/4.1.0"; } } function getCateUrl(URL) { if (URL.includes("api.php/app") || URL.includes("xgapp")) { return URL + "nav?token="; } else if (URL.includes(".vod")) { return URL + "/types"; } else { return ""; } } function getPlayUrlPrefix(URL) { if (URL.includes("api.php/app") || URL.includes("xgapp")) { return URL + "video_detail?id="; } else if (URL.includes(".vod")) { return URL + "/detail?vod_id="; } else { return ""; } } function getRecommendUrl(URL) { if (URL.includes("api.php/app") || URL.includes("xgapp")) { return URL + "index_video?token="; } else if (URL.includes(".vod")) { return URL + "/vodPhbAll"; } else { return ""; } } function getFilterTypes(URL, typeExtend) { let str = ""; if (typeExtend !== null) { for (let key in typeExtend) { if (key === "class" || key === "area" || key === "lang" || key === "year") { try { str += "筛选" + key + "+全部=+" + typeExtend[key].replace(/,/g, "+") + "\n"; } catch (e) { } } } } if (URL.includes(".vod")) { str += "\n" + "排序+全部=+最新=time+最热=hits+评分=score"; } else if (URL.includes("api.php/app") || URL.includes("xgapp")) { // Do nothing, leave the string as it is. } else { str = "分类+全部=+电影=movie+连续剧=tvplay+综艺=tvshow+动漫=comic+4K=movie_4k+体育=tiyu\n筛选class+全部=+喜剧+爱情+恐怖+动作+科幻+剧情+战争+警匪+犯罪+动画+奇幻+武侠+冒险+枪战+恐怖+悬疑+惊悚+经典+青春+文艺+微电影+古装+历史+运动+农村+惊悚+惊悚+伦理+情色+福利+三级+儿童+网络电影\n筛选area+全部=+大陆+香港+台湾+美国+英国+法国+日本+韩国+德国+泰国+印度+西班牙+加拿大+其他\n筛选year+全部=+2023+2022+2021+2020+2019+2018+2017+2016+2015+2014+2013+2012+2011+2010+2009+2008+2007+2006+2005+2004+2003+2002+2001+2000"; } return str; } function getCateFilterUrlSuffix(URL) { if (URL.includes("api.php/app") || URL.includes("xgapp")) { return "&class=筛选class&area=筛选area&lang=筛选lang&year=筛选year&limit=18&pg=#PN#"; } else if (URL.includes(".vod")) { return "&class=筛选class&area=筛选area&lang=筛选lang&year=筛选year&by=排序&limit=18&page=#PN#"; } else { return "&page=#PN#&area=筛选area&type=筛选class&start=筛选year"; } } function getCateFilterUrlPrefix(URL) { if (URL.includes("api.php/app") || URL.includes("xgapp")) { return URL + "video?tid="; } else if (URL.includes(".vod")) { return URL + "?type="; } else { return URL + "?ac=list&class="; } } function isBan(key) { return key === "伦理" || key === "情色" || key === "福利"; } function getSearchUrl(URL, KEY) { if (URL.includes(".vod")) { return URL + "?wd=" + KEY + "&page="; } else if (URL.includes("api.php/app") || URL.includes("xgapp")) { return URL + "search?text=" + KEY + "&pg="; } else if (urlPattern1.test(URL)) { return URL + "?ac=list&zm=" + KEY + "&page="; } return ""; } function findJsonArray(obj, match, result) { Object.keys(obj).forEach((k) => { try { const o = obj[k]; if (k === match && Array.isArray(o)) { result.push(o); } if (typeof o === "object" && o !== null) { if (Array.isArray(o)) { o.forEach((item) => { if (typeof item === "object" && item !== null) { findJsonArray(item, match, result); } }); } else { findJsonArray(o, match, result); } } } catch (e) { } }); } function jsonArr2Str(array) { const strings = []; for (let i = 0; i < array.length; i++) { try { strings.push(array[i]); } catch (e) { } } return strings.join(","); } function getHeaders(URL) { const headers = {}; headers["User-Agent"] = UA(URL); return headers; } function isJsonString(str) { try { JSON.parse(str); } catch (e) { return false; } return true; } export function __jsEvalReturn() { return { init: init, home: home, homeVod: homeVod, category: category, detail: detail, play: play, search: search, }; }