mirror of https://github.com/FongMi/CatVodOpen.git
commit
6b6a8aad24
@ -0,0 +1,508 @@ |
||||
import { Crypto, _ } from './lib/cat.js'; |
||||
|
||||
let siteKey = ''; |
||||
let siteType = 0; |
||||
|
||||
let cookie = ''; |
||||
let login = ''; |
||||
let vip = ''; |
||||
let extendObj = {}; |
||||
let vod_audio_id = { |
||||
30280: 192000, |
||||
30232: 132000, |
||||
30216: 64000, |
||||
}; |
||||
let vod_codec = { |
||||
// 13: 'AV1',
|
||||
12: 'HEVC', |
||||
7: 'AVC', |
||||
}; |
||||
|
||||
const UA = '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'; |
||||
|
||||
async function request(reqUrl, ua) { |
||||
let res = await req(reqUrl, { |
||||
method: 'get', |
||||
headers: ua ? ua : { 'User-Agent': UA }, |
||||
timeout: 60000, |
||||
}); |
||||
return res.content; |
||||
} |
||||
|
||||
function getHeaders() { |
||||
const headers = { |
||||
'User-Agent': UA, |
||||
}; |
||||
if (!_.isEmpty(cookie)) { |
||||
headers.cookie = cookie; |
||||
} |
||||
return headers; |
||||
} |
||||
|
||||
async function getCookie() { |
||||
let result = await req('https://www.bilibili.com', { |
||||
method: 'get', |
||||
headers: { 'User-Agent': UA }, |
||||
timeout: 60000, |
||||
}); |
||||
const setCookieHeaders = result.headers['set-cookie']; |
||||
cookie = setCookieHeaders.map((kk) => kk.split(';')[0] + ';').join(''); |
||||
} |
||||
|
||||
async function init(cfg) { |
||||
siteKey = cfg.skey; |
||||
siteType = cfg.stype; |
||||
let extend = cfg.ext; |
||||
if (cfg.ext.hasOwnProperty('categories')) extend = cfg.ext.categories; |
||||
if (cfg.ext.hasOwnProperty('cookie')) cookie = cfg.ext.cookie; |
||||
if (_.isEmpty(cookie)) await getCookie(); |
||||
let result = JSON.parse(await request('https://api.bilibili.com/x/web-interface/nav', getHeaders())); |
||||
login = result.data.isLogin; |
||||
vip = result.data.hasOwnProperty('vipStatus'); |
||||
const ext = extend.split('#'); |
||||
const jsonData = [ |
||||
{ |
||||
key: 'order', |
||||
name: '排序', |
||||
value: [ |
||||
{ n: '综合排序', v: '0' }, |
||||
{ n: '最多点击', v: 'click' }, |
||||
{ n: '最新发布', v: 'pubdate' }, |
||||
{ n: '最多弹幕', v: 'dm' }, |
||||
{ n: '最多收藏', v: 'stow' }, |
||||
], |
||||
}, |
||||
{ |
||||
key: 'duration', |
||||
name: '时长', |
||||
value: [ |
||||
{ n: '全部时长', v: '0' }, |
||||
{ n: '60分钟以上', v: '4' }, |
||||
{ n: '30~60分钟', v: '3' }, |
||||
{ n: '10~30分钟', v: '2' }, |
||||
{ n: '10分钟以下', v: '1' }, |
||||
], |
||||
}, |
||||
]; |
||||
const newarr = []; |
||||
const d = {}; |
||||
for (const kk of ext) { |
||||
const c = { |
||||
type_name: kk, |
||||
type_id: kk, |
||||
land: 1, |
||||
ratio: 1.78, |
||||
}; |
||||
newarr.push(c); |
||||
d[kk] = jsonData; |
||||
} |
||||
extendObj = { |
||||
classes: newarr, |
||||
filter: d, |
||||
}; |
||||
} |
||||
|
||||
function home(filter) { |
||||
try { |
||||
const jSONObject = { |
||||
class: extendObj.classes, |
||||
}; |
||||
if (filter) { |
||||
jSONObject.filters = extendObj.filter; |
||||
} |
||||
return JSON.stringify(jSONObject); |
||||
} catch (e) { |
||||
return ''; |
||||
} |
||||
} |
||||
|
||||
async function homeVod() { |
||||
try { |
||||
const list = []; |
||||
const type_id = extendObj.classes[0].type_id; |
||||
const url = `https://api.bilibili.com/x/web-interface/search/type?search_type=video&keyword=${type_id}&duration=4`; |
||||
|
||||
const response = await request(url, getHeaders()); |
||||
const responseData = JSON.parse(response); |
||||
const vods = responseData.data.result; |
||||
|
||||
for (const item of vods) { |
||||
const vod = {}; |
||||
let imageUrl = item.pic; |
||||
if (imageUrl.startsWith('//')) { |
||||
imageUrl = 'https:' + imageUrl; |
||||
} |
||||
|
||||
vod.vod_id = item.bvid; |
||||
vod.vod_name = removeTags(item.title); |
||||
vod.vod_pic = imageUrl; |
||||
vod.vod_remarks = item.duration.split(':')[0] + '分钟'; |
||||
list.push(vod); |
||||
} |
||||
|
||||
const result = { list: list }; |
||||
return JSON.stringify(result); |
||||
} catch (e) {} |
||||
} |
||||
|
||||
async function category(tid, page, filter, ext) { |
||||
if (page < 1) page = 1; |
||||
try { |
||||
if (Object.keys(ext).length > 0 && ext.hasOwnProperty('tid') && ext['tid'].length > 0) { |
||||
tid = ext['tid']; |
||||
} |
||||
let url = `https://api.bilibili.com/x/web-interface/search/type?search_type=video&keyword=${encodeURIComponent(tid)}`; |
||||
|
||||
if (Object.keys(ext).length > 0) { |
||||
for (const k in ext) { |
||||
if (k == 'tid') { |
||||
continue; |
||||
} |
||||
url += `&${encodeURIComponent(k)}=${encodeURIComponent(ext[k])}`; |
||||
} |
||||
} |
||||
|
||||
url += `&page=${encodeURIComponent(page)}`; |
||||
|
||||
const response = await request(url, getHeaders()); |
||||
|
||||
const resp = JSON.parse(response); |
||||
const data = resp.data; |
||||
|
||||
const videos = []; |
||||
const items = data.result; |
||||
|
||||
for (const item of items) { |
||||
const video = {}; |
||||
let pic = item.pic; |
||||
if (pic.startsWith('//')) { |
||||
pic = 'https:' + pic; |
||||
} |
||||
|
||||
video.vod_id = item.bvid; |
||||
video.vod_name = removeTags(item.title); |
||||
video.vod_pic = pic; |
||||
video.vod_remarks = item.duration.split(':')[0] + '分钟'; |
||||
videos.push(video); |
||||
} |
||||
|
||||
const result = { |
||||
page: page, |
||||
pagecount: data.numPages, |
||||
limit: data.pagesize, |
||||
total: data.numResults, |
||||
list: videos, |
||||
}; |
||||
|
||||
return JSON.stringify(result); |
||||
} catch (e) {} |
||||
return null; |
||||
} |
||||
|
||||
async function detail(ids) { |
||||
try { |
||||
const bvid = ids; |
||||
const bvid2aidUrl = `https://api.bilibili.com/x/web-interface/archive/stat?bvid=${bvid}`; |
||||
|
||||
const bvid2aidResp = JSON.parse(await request(bvid2aidUrl, getHeaders())); |
||||
|
||||
const aid = bvid2aidResp.data.aid + ''; |
||||
const detailUrl = `https://api.bilibili.com/x/web-interface/view?aid=${aid}`; |
||||
const detailData = JSON.parse(await request(detailUrl, getHeaders())).data; |
||||
|
||||
const video = { |
||||
vod_id: bvid, |
||||
vod_name: detailData.title, |
||||
vod_pic: detailData.pic, |
||||
type_name: detailData.tname, |
||||
vod_year: '', |
||||
vod_area: '', |
||||
vod_remarks: `${Math.floor(detailData.duration / 60)}分钟`, |
||||
vod_actor: '', |
||||
vod_director: '', |
||||
vod_content: detailData.desc, |
||||
}; |
||||
|
||||
const playurldata = 'https://api.bilibili.com/x/player/playurl?avid=' + aid + '&cid=' + detailData.cid + '&qn=127&fnval=4048&fourk=1'; |
||||
const playurldatas = JSON.parse(await request(playurldata, getHeaders())); |
||||
|
||||
const playurldatalist = playurldatas.data; |
||||
const a = playurldatalist.accept_description; |
||||
const accept_quality = playurldatalist.accept_quality; |
||||
const Aq = []; |
||||
const pFrom = []; |
||||
|
||||
for (let i = 0; i < accept_quality.length; i++) { |
||||
if (!login) { |
||||
if (accept_quality[i] > 32) continue; |
||||
} else if (!vip && login) { |
||||
if (accept_quality[i] > 80) continue; |
||||
} else { |
||||
if (accept_quality[i] > 32) continue; |
||||
} |
||||
pFrom.push(a[i]); |
||||
Aq.push(accept_quality[i]); |
||||
} |
||||
|
||||
const jSONArray = detailData.pages; |
||||
const playList = []; |
||||
for (let j = 0; j < jSONArray.length; j++) { |
||||
const jSONObject6 = jSONArray[j]; |
||||
const j2 = jSONObject6.cid; |
||||
const playUrl = j + '$' + aid + '+' + j2 + '+' + Aq.join(':') + '+' + pFrom.join(':'); |
||||
playList.push(playUrl); |
||||
} |
||||
|
||||
video.vod_play_from = 'external$$$dash$$$mp4'; |
||||
video.vod_play_url = playList.join('#'); |
||||
video.vod_play_url = [video.vod_play_url, video.vod_play_url, video.vod_play_url].join('$$$'); |
||||
|
||||
const list = [video]; |
||||
const result = { list }; |
||||
return JSON.stringify(result); |
||||
} catch (e) {} |
||||
return null; |
||||
} |
||||
|
||||
async function play(flag, id, flags) { |
||||
try { |
||||
const playHeaders = { Referer: 'https://www.bilibili.com', 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/115.0.0.0 Safari/537.36' }; |
||||
const ids = id.split('+'); |
||||
const aid = ids[0]; |
||||
const cid = ids[1]; |
||||
const qualityIds = ids[2].split(':'); |
||||
const qualityName = ids[3].split(':'); |
||||
if (flag == 'dash') { |
||||
// dash mpd 代理
|
||||
const js2Base = await js2Proxy(true, siteType, siteKey, 'dash/', {}); |
||||
let urls = []; |
||||
for (let i = 0; i < qualityIds.length; i++) { |
||||
urls.push(qualityName[i], js2Base + base64Encode(aid + '+' + cid + '+' + qualityIds[i])); |
||||
} |
||||
return JSON.stringify({ |
||||
parse: 0, |
||||
url: urls, |
||||
header: playHeaders, |
||||
}); |
||||
} else if (flag == 'mp4') { |
||||
// 直链
|
||||
let urls = []; |
||||
for (let i = 0; i < qualityIds.length; i++) { |
||||
const url = `https://api.bilibili.com/x/player/playurl?avid=${aid}&cid=${cid}&qn=${qualityIds[i]}&fourk=1`; |
||||
const resp = JSON.parse(await request(url, getHeaders())); |
||||
const data = resp.data; |
||||
if (data.quality != qualityIds[i]) continue; |
||||
let durl = data.durl[0].url; |
||||
urls.push(qualityName[i], durl); |
||||
} |
||||
|
||||
return JSON.stringify({ |
||||
parse: 0, |
||||
url: urls, |
||||
header: playHeaders, |
||||
}); |
||||
} else { |
||||
// 音频外挂
|
||||
let urls = []; |
||||
let audios = []; |
||||
for (let i = 0; i < qualityIds.length; i++) { |
||||
const url = `https://api.bilibili.com/x/player/playurl?avid=${aid}&cid=${cid}&qn=${qualityIds[i]}&fnval=4048&fourk=1`; |
||||
let resp = JSON.parse(await request(url, getHeaders())); |
||||
const dash = resp.data.dash; |
||||
const video = dash.video; |
||||
const audio = dash.audio; |
||||
for (let j = 0; j < video.length; j++) { |
||||
const dashjson = video[j]; |
||||
if (dashjson.id == qualityIds[i]) { |
||||
for (const key in vod_codec) { |
||||
if (dashjson.codecid == key) { |
||||
urls.push(qualityName[i] + ' ' + vod_codec[key], dashjson.baseUrl); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
if (audios.length == 0) { |
||||
for (let j = 0; j < audio.length; j++) { |
||||
const dashjson = audio[j]; |
||||
for (const key in vod_audio_id) { |
||||
if (dashjson.id == key) { |
||||
audios.push({ |
||||
title: _.floor(parseInt(vod_audio_id[key]) / 1024) + 'Kbps', |
||||
bit: vod_audio_id[key], |
||||
url: dashjson.baseUrl, |
||||
}); |
||||
} |
||||
} |
||||
} |
||||
audios = _.sortBy(audios, 'bit'); |
||||
} |
||||
} |
||||
|
||||
return JSON.stringify({ |
||||
parse: 0, |
||||
url: urls, |
||||
extra: { |
||||
audio: audios, |
||||
}, |
||||
header: playHeaders, |
||||
}); |
||||
} |
||||
} catch (e) {} |
||||
return null; |
||||
} |
||||
|
||||
async function search(key, quick, pg) { |
||||
let page = pg || 1; |
||||
if (page == 0) page = 1; |
||||
try { |
||||
const ext = { |
||||
duration: '4', |
||||
}; |
||||
let resp = JSON.parse(await category(key, page, true, ext)); |
||||
const catVideos = resp.list; |
||||
const pageCount = resp.pagecount; |
||||
const videos = []; |
||||
for (let i = 0; i < catVideos.length; ++i) { |
||||
videos.push(catVideos[i]); |
||||
} |
||||
const result = { |
||||
page: page, |
||||
pagecount: pageCount, |
||||
land: 1, |
||||
ratio: 1.78, |
||||
list: videos, |
||||
}; |
||||
return JSON.stringify(result); |
||||
} catch (e) {} |
||||
return null; |
||||
} |
||||
|
||||
async function proxy(segments, headers) { |
||||
let what = segments[0]; |
||||
let url = base64Decode(segments[1]); |
||||
if (what == 'dash') { |
||||
const ids = url.split('+'); |
||||
const aid = ids[0]; |
||||
const cid = ids[1]; |
||||
const str5 = ids[2]; |
||||
const urls = `https://api.bilibili.com/x/player/playurl?avid=${aid}&cid=${cid}&qn=${str5}&fnval=4048&fourk=1`; |
||||
let videoList = ''; |
||||
let audioList = ''; |
||||
|
||||
let resp = JSON.parse(await request(urls, getHeaders())); |
||||
const dash = resp.data.dash; |
||||
const video = dash.video; |
||||
const audio = dash.audio; |
||||
|
||||
for (let i = 0; i < video.length; i++) { |
||||
// if (i > 0) continue; // 只取一个
|
||||
const dashjson = video[i]; |
||||
if (dashjson.id == str5) { |
||||
videoList += getDashMedia(dashjson); |
||||
} |
||||
} |
||||
|
||||
for (let i = 0; i < audio.length; i++) { |
||||
// if (i > 0) continue;
|
||||
const ajson = audio[i]; |
||||
for (const key in vod_audio_id) { |
||||
if (ajson.id == key) { |
||||
audioList += getDashMedia(ajson); |
||||
} |
||||
} |
||||
} |
||||
|
||||
let mpd = getDash(resp, videoList, audioList); |
||||
return JSON.stringify({ |
||||
code: 200, |
||||
content: mpd, |
||||
headers: { |
||||
'Content-Type': 'application/dash+xml', |
||||
}, |
||||
}); |
||||
} |
||||
return JSON.stringify({ |
||||
code: 500, |
||||
content: '', |
||||
}); |
||||
} |
||||
|
||||
function getDashMedia(dash) { |
||||
try { |
||||
let qnid = dash.id; |
||||
const codecid = dash.codecid; |
||||
const media_codecs = dash.codecs; |
||||
const media_bandwidth = dash.bandwidth; |
||||
const media_startWithSAP = dash.startWithSap; |
||||
const media_mimeType = dash.mimeType; |
||||
const media_BaseURL = dash.baseUrl.replace(/&/g, '&'); |
||||
const media_SegmentBase_indexRange = dash.SegmentBase.indexRange; |
||||
const media_SegmentBase_Initialization = dash.SegmentBase.Initialization; |
||||
const mediaType = media_mimeType.split('/')[0]; |
||||
let media_type_params = ''; |
||||
|
||||
if (mediaType == 'video') { |
||||
const media_frameRate = dash.frameRate; |
||||
const media_sar = dash.sar; |
||||
const media_width = dash.width; |
||||
const media_height = dash.height; |
||||
media_type_params = `height='${media_height}' width='${media_width}' frameRate='${media_frameRate}' sar='${media_sar}'`; |
||||
} else if (mediaType == 'audio') { |
||||
for (const key in vod_audio_id) { |
||||
if (qnid == key) { |
||||
const audioSamplingRate = vod_audio_id[key]; |
||||
media_type_params = `numChannels='2' sampleRate='${audioSamplingRate}'`; |
||||
} |
||||
} |
||||
} |
||||
qnid += '_' + codecid; |
||||
|
||||
return `<AdaptationSet lang="chi">
|
||||
<ContentComponent contentType="${mediaType}"/> |
||||
<Representation id="${qnid}" bandwidth="${media_bandwidth}" codecs="${media_codecs}" mimeType="${media_mimeType}" ${media_type_params} startWithSAP="${media_startWithSAP}"> |
||||
<BaseURL>${media_BaseURL}</BaseURL> |
||||
<SegmentBase indexRange="${media_SegmentBase_indexRange}"> |
||||
<Initialization range="${media_SegmentBase_Initialization}"/> |
||||
</SegmentBase> |
||||
</Representation> |
||||
</AdaptationSet>`; |
||||
} catch (e) { |
||||
// Handle exceptions here
|
||||
} |
||||
} |
||||
|
||||
function getDash(ja, videoList, audioList) { |
||||
const duration = ja.data.dash.duration; |
||||
const minBufferTime = ja.data.dash.minBufferTime; |
||||
return `<MPD xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="urn:mpeg:dash:schema:mpd:2011" xsi:schemaLocation="urn:mpeg:dash:schema:mpd:2011 DASH-MPD.xsd" type="static" mediaPresentationDuration="PT${duration}S" minBufferTime="PT${minBufferTime}S" profiles="urn:mpeg:dash:profile:isoff-on-demand:2011">
|
||||
<Period duration="PT${duration}S" start="PT0S"> |
||||
${videoList} |
||||
${audioList} |
||||
</Period> |
||||
</MPD>`; |
||||
} |
||||
|
||||
function base64Encode(text) { |
||||
return Crypto.enc.Base64.stringify(Crypto.enc.Utf8.parse(text)); |
||||
} |
||||
|
||||
function base64Decode(text) { |
||||
return Crypto.enc.Utf8.stringify(Crypto.enc.Base64.parse(text)); |
||||
} |
||||
|
||||
function removeTags(input) { |
||||
return input.replace(/<[^>]*>/g, ''); |
||||
} |
||||
|
||||
export function __jsEvalReturn() { |
||||
return { |
||||
init: init, |
||||
home: home, |
||||
homeVod: homeVod, |
||||
category: category, |
||||
detail: detail, |
||||
play: play, |
||||
proxy: proxy, |
||||
search: search, |
||||
}; |
||||
} |
||||
Loading…
Reference in new issue