mirror of
https://github.com/letian1650/N3RD.git
synced 2025-01-25 04:05:17 +08:00
502 lines
19 KiB
JavaScript
502 lines
19 KiB
JavaScript
/**
|
||
* 支持本地包直播链接
|
||
* 传参 ?type=url¶ms=../json/live2cms.json
|
||
live2cms.json
|
||
支持m3u类直播,支持线路归并。支持筛选切换显示模式
|
||
[
|
||
{
|
||
"name": "GitHub",
|
||
"url": "https://cors.isteed.cc/https://raw.githubusercontent.com/ssili126/tv/main/itvlist.txt"
|
||
},
|
||
{
|
||
"name": "CNTV",
|
||
"url": "./live_cntv.txt"
|
||
}
|
||
]
|
||
*/
|
||
|
||
/**
|
||
* m3u直播格式转一般直播格式
|
||
* @param m3u
|
||
* @returns {string}
|
||
*/
|
||
function convertM3uToNormal(m3u) {
|
||
try {
|
||
const lines = m3u.split('\n');
|
||
let result = '';
|
||
let TV = '';
|
||
// let flag='#genre#';
|
||
let flag = '#m3u#';
|
||
let currentGroupTitle = '';
|
||
lines.forEach((line) => {
|
||
if (line.startsWith('#EXTINF:')) {
|
||
line = line.replace(/'/g, '"');
|
||
let groupTitle = '未知频道';
|
||
let tvg_name = '';
|
||
let tvg_logo = '';
|
||
try {
|
||
groupTitle = line.match(/group-title="(.*?)"/)[1].trim();
|
||
} catch (e) {
|
||
}
|
||
try {
|
||
tvg_name = line.match(/tvg-name="(.*?)"/)[1].trim();
|
||
} catch (e) {
|
||
}
|
||
try {
|
||
tvg_logo = line.match(/tvg-logo="(.*?)"/)[1].trim();
|
||
} catch (e) {
|
||
}
|
||
TV = line.split(',').slice(-1)[0].trim();
|
||
if (currentGroupTitle !== groupTitle) {
|
||
currentGroupTitle = groupTitle;
|
||
let ret_list = [currentGroupTitle, flag];
|
||
// if(tvg_name){
|
||
// ret_list.push(tvg_name);
|
||
// }
|
||
// if(tvg_logo){
|
||
// ret_list.push(tvg_logo);
|
||
// }
|
||
result += `\n${ret_list.join(",")}\n`;
|
||
}
|
||
} else if (line.startsWith('http')) {
|
||
const splitLine = line.split(',');
|
||
result += `${TV}\,${splitLine[0]}\n`;
|
||
}
|
||
});
|
||
// result = result.trim();
|
||
result = mergeChannels(result);
|
||
// log(result);
|
||
return result
|
||
} catch (e) {
|
||
log(`m3u直播转普通直播发生错误:${e.message}`);
|
||
return m3u
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 线路归类/小棉袄算法
|
||
* @param arr 数组
|
||
* @param parse 解析式
|
||
* @returns {[[*]]}
|
||
*/
|
||
function splitArray(arr, parse) {
|
||
parse = parse && typeof (parse) == 'function' ? parse : '';
|
||
let result = [[arr[0]]];
|
||
for (let i = 1; i < arr.length; i++) {
|
||
let index = -1;
|
||
for (let j = 0; j < result.length; j++) {
|
||
if (parse && result[j].map(parse).includes(parse(arr[i]))) {
|
||
index = j;
|
||
} else if ((!parse) && result[j].includes(arr[i])) {
|
||
index = j;
|
||
}
|
||
}
|
||
if (index >= result.length - 1) {
|
||
result.push([]);
|
||
result[result.length - 1].push(arr[i]);
|
||
} else {
|
||
result[index + 1].push(arr[i]);
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* 搜索结果生成分组字典
|
||
* @param arr
|
||
* @param parse x=>x.split(',')[0]
|
||
* @returns {{}}
|
||
*/
|
||
function gen_group_dict(arr, parse) {
|
||
let dict = {};
|
||
arr.forEach((it) => {
|
||
let k = it.split(',')[0];
|
||
if (parse && typeof (parse) === 'function') {
|
||
k = parse(k);
|
||
}
|
||
if (!dict[k]) {
|
||
dict[k] = [it];
|
||
} else {
|
||
dict[k].push(it);
|
||
}
|
||
});
|
||
return dict
|
||
}
|
||
|
||
/**
|
||
* txt格式直播自动合并频道链接
|
||
* @param text
|
||
* @returns {string}
|
||
*/
|
||
function mergeChannels(text) {
|
||
const lines = text.split('\n');
|
||
const channelMap = new Map();
|
||
let currentChannel = ''; // 当前处理的频道
|
||
|
||
lines.forEach(line => {
|
||
// 使用正则表达式匹配频道行,假设频道行包含",#"即可识别为频道行
|
||
if (/,#/.test(line)) {
|
||
// 如果是频道名称,作为键值存储,初始化为空数组
|
||
currentChannel = line;
|
||
if (!channelMap.has(line)) {
|
||
channelMap.set(line, []);
|
||
}
|
||
} else if (line) { // 忽略空行
|
||
// 将当前行(链接)添加到当前频道数组中
|
||
if (currentChannel) {
|
||
channelMap.get(currentChannel).push(line);
|
||
}
|
||
}
|
||
});
|
||
|
||
// 构建结果字符串
|
||
let result = '';
|
||
channelMap.forEach((value, key) => {
|
||
result += key + '\n' + value.join('\n') + '\n\n';
|
||
});
|
||
|
||
return result.trim(); // 移除尾部的多余换行符
|
||
}
|
||
|
||
globalThis.mergeChannels = mergeChannels;
|
||
globalThis.convertM3uToNormal = convertM3uToNormal;
|
||
globalThis.splitArray = splitArray;
|
||
globalThis.gen_group_dict = gen_group_dict;
|
||
globalThis.getRandomItem = function (items) {//从列表随机取出一个元素
|
||
return items[Math.random() * items.length | 0];
|
||
}
|
||
globalThis.__ext = {data_dict: {}};
|
||
var rule = {
|
||
title: '直播转点播[合]',
|
||
author: '道长',
|
||
version: '20240628 beta7',
|
||
update_info: `
|
||
20240628 beta6:
|
||
1.增加范冰冰v6源
|
||
2.修复带图标的m3u源识别
|
||
3.修复m3u8链接带参数转义问题
|
||
4.合并重复的频道名称下的链接
|
||
5.支持相对图片链接
|
||
20240627 beta1:
|
||
1.将原drpy项目的live2cms.js转换成hipy传参源。
|
||
【特别说明】支持m3u和txt的直播
|
||
`,
|
||
host: '',
|
||
homeUrl: '',
|
||
searchUrl: '#wd=**&pg=#TruePage##page=fypage',
|
||
url: 'fyclass#pg=fypage&t=fyfilter',
|
||
filter_url: '{{fl.show}}',
|
||
headers: {'User-Agent': 'MOBILE_UA'},
|
||
timeout: 5000, // class_name: '电影&电视剧&综艺&动漫',
|
||
limit: 20,
|
||
search_limit: 5, // 搜索限制取前5个,可以注释掉,就不限制搜索
|
||
searchable: 1,//是否启用全局搜索,
|
||
quickSearch: 0,//是否启用快速搜索,
|
||
filterable: 1,//是否启用分类筛选,
|
||
play_parse: true,
|
||
// params: 'http://127.0.0.1:5707/files/json/live2cms.json',
|
||
// 下面自定义一些源的配置
|
||
// def_pic: 'https://avatars.githubusercontent.com/u/97389433?s=120&v=4', //默认列表图片
|
||
def_pic: 'https://cors.isteed.cc/https://raw.githubusercontent.com/n3rddd/N3RD/master/JN/N3RD/W/POSTER1.png', //默认列表图片
|
||
showMode: 'groups',// groups按组分类显示 all全部一条线路展示
|
||
groupDict: {},// 搜索分组字典
|
||
tips: '', //二级提示信息
|
||
预处理: $js.toString(() => {
|
||
// 初始化保存的数据
|
||
rule.showMode = getItem('showMode', 'groups');
|
||
rule.groupDict = JSON.parse(getItem('groupDict', '{}'));
|
||
rule.tips = `雷蒙直播js-当前版本${rule.version}`;
|
||
|
||
if (typeof (batchFetch) === 'function') {
|
||
// 支持批量请求直接放飞自我。搜索限制最大线程数量16
|
||
rule.search_limit = 16;
|
||
log('当前程序支持批量请求[batchFetch],搜索限制已设置为16');
|
||
}
|
||
let _url = rule.params;
|
||
if (_url && typeof (_url) === 'string' && /^(http|file)/.test(_url)) {
|
||
let html = request(_url);
|
||
let json = JSON.parse(html);
|
||
|
||
let _classes = [];
|
||
rule.filter = {};
|
||
rule.filter_def = {};
|
||
json.forEach(it => {
|
||
if (it.url && !/^(http|file)/.test(it.url)) {
|
||
it.url = urljoin(_url, it.url);
|
||
}
|
||
if (it.img && !/^(http|file)/.test(it.img)) {
|
||
it.img = urljoin(_url, it.img);
|
||
}
|
||
let _obj = {
|
||
type_name: it.name,
|
||
type_id: it.url,
|
||
img: it.img,
|
||
};
|
||
_classes.push(_obj);
|
||
let json1 = [{'n': '多线路分组', 'v': 'groups'}, {'n': '单线路', 'v': 'all'}];
|
||
try {
|
||
rule.filter[_obj.type_id] = [
|
||
{'key': 'show', 'name': '播放展示', 'value': json1}
|
||
];
|
||
if (json1.length > 0) {
|
||
rule.filter_def[it.url] = {"show": json1[0].v};
|
||
}
|
||
} catch (e) {
|
||
rule.filter[it.url] = json1
|
||
}
|
||
});
|
||
__ext.data = json;
|
||
rule.classes = _classes;
|
||
}
|
||
}),
|
||
class_parse: $js.toString(() => {
|
||
input = rule.classes;
|
||
}),
|
||
推荐: $js.toString(() => {
|
||
let update_info = [{
|
||
vod_name: '雷蒙影视',
|
||
vod_id: 'update_info',
|
||
vod_remarks: `版本:${rule.version}`,
|
||
vod_pic: 'https://cors.isteed.cc/https://raw.githubusercontent.com/n3rddd/N3RD/master/JN/N3RD/W/POSTER1.png'
|
||
}];
|
||
VODS = [];
|
||
if (rule.classes) {
|
||
let randomClass = getRandomItem(rule.classes);
|
||
let _get_url = randomClass.type_id;
|
||
// let current_vod = rule.classes.find(item => item.type_id === _get_url);
|
||
// let _pic = current_vod ? current_vod.img : '';
|
||
let _pic = randomClass.img;
|
||
let html;
|
||
if (__ext.data_dict[_get_url]) {
|
||
html = __ext.data_dict[_get_url];
|
||
} else {
|
||
html = request(_get_url);
|
||
if (/#EXTM3U/.test(html)) {
|
||
html = convertM3uToNormal(html);
|
||
} else {
|
||
html = mergeChannels(html);
|
||
}
|
||
__ext.data_dict[_get_url] = html;
|
||
}
|
||
let arr = html.match(/.*?[,,]#[\s\S].*?#/g); // 可能存在中文逗号
|
||
try {
|
||
arr.forEach(it => {
|
||
let vname = it.split(/[,,]/)[0];
|
||
let vtab = it.match(/#(.*?)#/)[0];
|
||
VODS.push({
|
||
vod_name: vname,
|
||
vod_id: _get_url + '$' + vname,
|
||
vod_pic: _pic || rule.def_pic,
|
||
vod_remarks: vtab,
|
||
});
|
||
});
|
||
} catch (e) {
|
||
log(`直播转点播获取首页推荐发送错误:${e.message}`);
|
||
}
|
||
}
|
||
VODS = update_info.concat(VODS);
|
||
}),
|
||
一级: $js.toString(() => {
|
||
VODS = [];
|
||
// 一级限制页数不允许翻页
|
||
if (rule.classes && MY_PAGE <= 1) {
|
||
if (MY_FL.show) {
|
||
rule.showMode = MY_FL.show;
|
||
setItem('showMode', rule.showMode);
|
||
}
|
||
let _get_url = input.split('#')[0];
|
||
let current_vod = rule.classes.find(item => item.type_id === MY_CATE);
|
||
let _pic = current_vod ? current_vod.img : '';
|
||
let html;
|
||
if (__ext.data_dict[_get_url]) {
|
||
html = __ext.data_dict[_get_url];
|
||
} else {
|
||
html = request(_get_url);
|
||
if (/#EXTM3U/.test(html)) {
|
||
html = convertM3uToNormal(html);
|
||
} else {
|
||
html = mergeChannels(html);
|
||
}
|
||
__ext.data_dict[_get_url] = html;
|
||
}
|
||
let arr = html.match(/.*?[,,]#[\s\S].*?#/g); // 可能存在中文逗号
|
||
try {
|
||
arr.forEach(it => {
|
||
let vname = it.split(/[,,]/)[0];
|
||
let vtab = it.match(/#(.*?)#/)[0];
|
||
VODS.push({
|
||
// vod_name:it.split(',')[0],
|
||
vod_name: vname,
|
||
vod_id: _get_url + '$' + vname,
|
||
vod_pic: _pic || rule.def_pic,
|
||
vod_remarks: vtab,
|
||
});
|
||
});
|
||
} catch (e) {
|
||
log(`直播转点播获取一级分类页发生错误:${e.message}`);
|
||
}
|
||
}
|
||
}),
|
||
二级: $js.toString(() => {
|
||
VOD = {};
|
||
if (orId === 'update_info') {
|
||
VOD = {
|
||
vod_content: rule.update_info.trim(),
|
||
vod_name: '雷蒙影视',
|
||
type_name: '雷蒙影视',
|
||
vod_pic: 'https://raw.githubusercontent.com/n3rddd/N3RD/master/JN/N3RD/W/POSTER1.png',
|
||
vod_remarks: `版本:${rule.version}`,
|
||
vod_play_from: '雷蒙影视',
|
||
// vod_play_url: '嗅探播放$https://resource-cdn.tuxiaobei.com/video/10/8f/108fc9d1ac3f69d29a738cdc097c9018.mp4',
|
||
vod_play_url: '雷蒙影视主题曲$https://cors.isteed.cc/https://raw.githubusercontent.com/n3rddd/N3RD/master/JN/N3RD/W/CTVThemeSong2.mp4',
|
||
};
|
||
} else {
|
||
if (rule.classes) {
|
||
let _get_url = orId.split('$')[0];
|
||
let _tab = orId.split('$')[1];
|
||
if (orId.includes('#search#')) {
|
||
let vod_name = _tab.replace('#search#', '');
|
||
let vod_play_from = '来自搜索';
|
||
vod_play_from += `:${_get_url}`;
|
||
let vod_play_url = rule.groupDict[_get_url].map(x => x.replace(',', '$')).join('#');
|
||
log(orId);
|
||
VOD = {
|
||
vod_name: '搜索:' + vod_name,
|
||
type_name: "直播列表",
|
||
vod_pic: rule.def_pic,
|
||
// vod_content: orId,
|
||
vod_content: orId.replace(getHome(orId), 'http://***'),
|
||
vod_play_from: vod_play_from,
|
||
vod_play_url: vod_play_url,
|
||
vod_director: rule.tips,
|
||
vod_remarks: rule.tips,
|
||
}
|
||
} else {
|
||
let current_vod = rule.classes.find(item => item.type_id === _get_url);
|
||
let _pic = current_vod ? current_vod.img : '';
|
||
let html;
|
||
if (__ext.data_dict[_get_url]) {
|
||
html = __ext.data_dict[_get_url];
|
||
} else {
|
||
html = request(_get_url);
|
||
if (/#EXTM3U/.test(html)) {
|
||
html = convertM3uToNormal(html);
|
||
} else {
|
||
html = mergeChannels(html);
|
||
}
|
||
__ext.data_dict[_get_url] = html;
|
||
}
|
||
let a = new RegExp(`.*?${_tab.replace('(','\\(').replace(')','\\)')}[,,]#[\\s\\S].*?#`);
|
||
let b = html.match(a)[0];
|
||
let c = html.split(b)[1];
|
||
if (c.match(/.*?[,,]#[\s\S].*?#/)) {
|
||
let d = c.match(/.*?[,,]#[\s\S].*?#/)[0];
|
||
c = c.split(d)[0];
|
||
}
|
||
let arr = c.trim().split('\n');
|
||
let _list = [];
|
||
arr.forEach((it) => {
|
||
if (it.trim()) {
|
||
let t = it.trim().split(',')[0];
|
||
let u = it.trim().split(',')[1];
|
||
_list.push(t + '$' + u);
|
||
}
|
||
});
|
||
|
||
let vod_name = __ext.data.find(x => x.url === _get_url).name;
|
||
let vod_play_url;
|
||
let vod_play_from;
|
||
|
||
if (rule.showMode === 'groups') {
|
||
let groups = splitArray(_list, x => x.split('$')[0]);
|
||
let tabs = [];
|
||
for (let i = 0; i < groups.length; i++) {
|
||
if (i === 0) {
|
||
tabs.push(vod_name + ' | ' + '雷蒙直播1');
|
||
} else {
|
||
tabs.push(`雷蒙直播${i + 1}`);
|
||
}
|
||
}
|
||
vod_play_url = groups.map(it => it.join('#')).join('$$$');
|
||
vod_play_from = tabs.join('$$$');
|
||
} else {
|
||
vod_play_url = _list.join('#');
|
||
vod_play_from = vod_name;
|
||
}
|
||
log(orId);
|
||
VOD = {
|
||
vod_id: orId,
|
||
vod_name: vod_name + '|' + _tab,
|
||
type_name: "直播列表",
|
||
vod_pic: _pic || rule.def_pic,
|
||
// vod_content: orId,
|
||
vod_content: orId.replace(getHome(orId), 'http://***'),
|
||
vod_play_from: vod_play_from,
|
||
vod_play_url: vod_play_url,
|
||
vod_director: rule.tips,
|
||
vod_remarks: rule.tips,
|
||
};
|
||
|
||
}
|
||
}
|
||
}
|
||
}),
|
||
搜索: $js.toString(() => {
|
||
VODS = [];
|
||
if (rule.classes && MY_PAGE <= 1) {
|
||
let _get_url = __ext.data[0].url;
|
||
let current_vod = rule.classes.find(item => item.type_id === _get_url);
|
||
let _pic = current_vod ? current_vod.img : '';
|
||
let html;
|
||
if (__ext.data_dict[_get_url]) {
|
||
html = __ext.data_dict[_get_url];
|
||
} else {
|
||
html = request(_get_url);
|
||
if (/#EXTM3U/.test(html)) {
|
||
html = convertM3uToNormal(html);
|
||
} else {
|
||
html = mergeChannels(html);
|
||
}
|
||
__ext.data_dict[_get_url] = html;
|
||
}
|
||
let str = '';
|
||
Object.keys(__ext.data_dict).forEach(() => {
|
||
str += __ext.data_dict[_get_url];
|
||
});
|
||
let links = str.split('\n').filter(it => it.trim() && it.includes(',') && it.split(',')[1].trim().startsWith('http'));
|
||
links = links.map(it => it.trim());
|
||
let plays = Array.from(new Set(links));
|
||
log('搜索关键词:' + KEY);
|
||
log('过滤前:' + plays.length);
|
||
// plays = plays.filter(it => it.includes(KEY));
|
||
plays = plays.filter(it => new RegExp(KEY, 'i').test(it));
|
||
log('过滤后:' + plays.length);
|
||
log(plays);
|
||
let new_group = gen_group_dict(plays);
|
||
rule.groupDict = Object.assign(rule.groupDict, new_group);
|
||
// 搜索分组结果存至本地方便二级调用
|
||
setItem('groupDict', JSON.stringify(rule.groupDict));
|
||
// 返回的还是搜索的new_group
|
||
Object.keys(new_group).forEach((it) => {
|
||
VODS.push({
|
||
'vod_name': it,
|
||
'vod_id': it + '$' + KEY + '#search#',
|
||
'vod_pic': _pic || rule.def_pic,
|
||
});
|
||
});
|
||
}
|
||
}),
|
||
lazy: $js.toString(() => {
|
||
if (/\.(m3u8|mp4)/.test(input)) {
|
||
if (input.includes('?') && typeof (playObj) == 'object' && playObj.url) {
|
||
input = playObj.url;
|
||
}
|
||
input = {parse: 0, url: input}
|
||
} else if (/yangshipin|1905\.com/.test(input)) {
|
||
input = {parse: 1, jx: 0, url: input, js: '', header: {'User-Agent': PC_UA}, parse_extra: '&is_pc=1'};
|
||
} else {
|
||
input
|
||
}
|
||
}),
|
||
}
|