From e8c2b2f3a429fdea0ef30bba20ad492746ce3c1d Mon Sep 17 00:00:00 2001 From: Howard Wu Date: Sat, 6 Jan 2024 11:37:39 +0800 Subject: [PATCH] =?UTF-8?q?fix(balh):=20=E4=BF=AE=E5=A4=8D=E6=9F=90?= =?UTF-8?q?=E4=BA=9B=E5=A4=A7=E9=99=86=E7=95=AA=E5=89=A7=E4=B8=8D=E8=83=BD?= =?UTF-8?q?=E6=92=AD=E6=94=BE=E7=9A=84=E9=97=AE=E9=A2=98=20(#1302)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * fix(balh): 修复某些大陆番剧不能播放的问题 增加一个新魔法拼凑番剧信息,现在只要能打开作品详情页的大概率都能播放 * wip: 整理代码 --------- Co-authored-by: ipcjs --- .../unblock-area-limit/src/api/bilibili.ts | 305 ++++++++++++++---- .../src/feature/bili/area_limit_for_vue.ts | 46 ++- 2 files changed, 270 insertions(+), 81 deletions(-) diff --git a/packages/unblock-area-limit/src/api/bilibili.ts b/packages/unblock-area-limit/src/api/bilibili.ts index 774bbbb5..03a8528d 100644 --- a/packages/unblock-area-limit/src/api/bilibili.ts +++ b/packages/unblock-area-limit/src/api/bilibili.ts @@ -1,4 +1,4 @@ -import { Async } from "../util/async" +import { Async, Promise } from "../util/async" import { generateMobiPlayUrlParams } from "./biliplus" import { Converters } from "../util/converters" @@ -30,7 +30,7 @@ interface SeasonInfo { export interface AppSeasonInfo { result?: { - actor: { + actor?: { info: string title: string } @@ -50,7 +50,7 @@ export interface AppSeasonInfo { text: string } cover: string - detail: string + detail?: string dialog?: { code: number config: any @@ -65,7 +65,7 @@ export interface AppSeasonInfo { } type: string } - dynamic_subtitle: string + dynamic_subtitle?: string earphone_conf?: { sp_phones: any[] } @@ -75,12 +75,12 @@ export interface AppSeasonInfo { name: string text: string } - link: string + link?: string media_badge_info?: any media_id?: number mode: number - modules: any[] - new_ep: { + modules?: any[] + new_ep?: { desc?: string id: number is_new?: number @@ -111,16 +111,16 @@ export interface AppSeasonInfo { publish: { is_finish: number is_started: number - pub_time: string - pub_time_show: string + pub_time?: string + pub_time_show?: string release_date_show: string time_length_show: string - unknow_pub_date: number + unknow_pub_date?: number update_info_desc?: string - weekday: number + weekday?: number } record?: string - refine_cover: string + refine_cover?: string reserve?: { episodes: any[] tip: string @@ -130,53 +130,53 @@ export interface AppSeasonInfo { allow_bp_rank: number allow_download?: number allow_review: number - area_limit: number - ban_area_show: number + area_limit?: number + ban_area_show?: number can_watch: number copyright: string forbid_pre?: number freya_white?: number is_cover_show?: number - is_preview: number + is_preview?: number only_vip_download?: number resource?: string watch_platform?: number } season_id: number season_title?: string - series: { + series?: { display_type: number series_id: number series_title: string } - share_copy: string - share_url: string - short_link: string + share_copy?: string + share_url?: string + short_link?: string show_season_type?: number - square_cover: string + square_cover?: string staff: { info: string title: string } stat: { - coins: number + coins?: number danmakus: number favorite?: number favorites: number - followers: string - likes: number - play: string - reply: number - share: number + followers?: string + likes?: number + play?: string + reply?: number + share?: number views: number vt?: number } - status: number + status?: number jp_title?: string | undefined styles: [{ name: string }] - subtitle: string + subtitle?: string test_switch?: { channel_entrance_exp_action: number enable_ep_vt: boolean @@ -195,7 +195,7 @@ export interface AppSeasonInfo { was_pugv_style_optimize: boolean } title: string - total: number + total?: number type: number type_desc?: string type_name: string @@ -220,7 +220,7 @@ export interface AppSeasonInfo { vip_frozen: number } season_status?: number | undefined - is_paster_ads: number + is_paster_ads?: number user_thumbup?: { url_image_ani: string url_image_ani_cut: string @@ -241,7 +241,7 @@ export interface AppSeasonInfo { } code: number data: { - actor: { + actor?: { info: string title: string } @@ -256,7 +256,7 @@ export interface AppSeasonInfo { text: string } cover: string - detail: string + detail?: string dialog?: { code: number config: any @@ -271,7 +271,7 @@ export interface AppSeasonInfo { } type: string } - dynamic_subtitle: string + dynamic_subtitle?: string earphone_conf?: { sp_phones: any[] } enable_vt?: boolean evaluate: string @@ -279,12 +279,12 @@ export interface AppSeasonInfo { name: string text: string } - link: string + link?: string media_badge_info?: any media_id?: number mode: number - modules: any[] - new_ep: { + modules?: any[] + new_ep?: { desc?: string id: number is_new?: number @@ -311,16 +311,16 @@ export interface AppSeasonInfo { publish: { is_finish: number is_started: number - pub_time: string - pub_time_show: string + pub_time?: string + pub_time_show?: string release_date_show: string time_length_show: string - unknow_pub_date: number + unknow_pub_date?: number update_info_desc?: string - weekday: number + weekday?: number } record?: string - refine_cover: string + refine_cover?: string reserve?: { episodes: any[] tip: string @@ -330,53 +330,53 @@ export interface AppSeasonInfo { allow_bp_rank: number allow_download?: number allow_review: number - area_limit: number - ban_area_show: number + area_limit?: number + ban_area_show?: number can_watch: number copyright: string forbid_pre?: number freya_white?: number is_cover_show?: number - is_preview: number + is_preview?: number only_vip_download?: number resource?: string watch_platform?: number } season_id: number season_title?: string - series: { + series?: { display_type: number series_id: number series_title: string } - share_copy: string - share_url: string - short_link: string + share_copy?: string + share_url?: string + short_link?: string show_season_type?: number - square_cover: string + square_cover?: string staff: { info: string title: string } stat: { - coins: number + coins?: number danmakus: number favorite?: number favorites: number - followers: string - likes: number - play: string - reply: number - share: number + followers?: string + likes?: number + play?: string + reply?: number + share?: number views: number vt?: number } - status: number + status?: number jp_title?: string styles: [{ name: string }] - subtitle: string + subtitle?: string test_switch?: { channel_entrance_exp_action: number enable_ep_vt: boolean @@ -395,7 +395,7 @@ export interface AppSeasonInfo { was_pugv_style_optimize: boolean } title: string - total: number + total?: number type: number type_desc?: string type_name: string @@ -420,7 +420,7 @@ export interface AppSeasonInfo { vip_frozen: number } season_status?: number - is_paster_ads: number + is_paster_ads?: number user_thumbup?: { url_image_ani: string url_image_ani_cut: string @@ -596,17 +596,177 @@ interface SeasonInfoOnThailand { } } +interface MediaInfo { + activity: { + head_bg_url: string + id: number + title: string + }, + actors: string + alias: string + areas: [{ id: number, name: string }], + copyright: { + is_finish: number + is_started: number + }, + cover: string + refine_cover?: string + enable_vt: boolean + episode_index: { + id: number + index: string + index_show: string + is_new: number + play_index_show: string + }, + evaluate: string + icon_font: { + name: string + text: string + }, + long_review: { + next: number + normal: number + total: number + count: number + list: [{ + review_id: number + stat: { + likes: number + }, + author: { + uname: string + level: number + mid: number + avatar: string + label: { + bg_color: string + bg_style: number + }, + vip: { + vipType: number + nickname_color: string + vipStatus: number + } + }, + mid: number + push_time_str: string + title: string + mtime: number + content: string + url: string + article_id: number + score: number + is_origin: number + media_id: number + ctime: number + progress: string + }] + }, + media_id: number + mode: number + origin_name: string + publish: { + is_finish: number + is_started: number + pub_date: string + pub_date_show: string + release_date_show: string + time_length_show: string + }, + rating: { + count: number + score: number + }, + rights: { + allow_bp: number + allow_bp_rank: number + allow_review: number + can_watch: number + copyright: string + area_limit?: number + ban_area_show?: number + is_preview?: number + } + season_id: number + season_status: number + status?: number + seasons: [{ + is_new: number + media_id: number + season_id: number + season_title: string + title: string + type: number + }] + share_copy?: string + share_url?: string + short_link?: string + short_review: { + next: number + total: number + list: [{ + review_id: number + score: number + stat: { + likes: number + }, + author: { + uname: string + level: number + mid: number + avatar: string + label: { + bg_color: string + bg_style: number + }, + vip: { + vipType: number + nickname_color: string + vipStatus: number + } + }, + media_id: number + mid: number + ctime: number + progress: string + push_time_str: string + mtime: number + content: string + }] + }, + show_season_type: number + staff: any + stat: { + danmakus: number + favorites: number + series_follow: number + views: number + vt: number + }, + styles: [{ + id: number + name: string + }], + time_length: number + title: string + type: number + type_name: string + param: { + season_id: number + season_type: number + show_season_type: number + } +} + export class BiliBiliApi { private server: string constructor(server: string = '//api.bilibili.com') { this.server = server } - getSeasonInfoByEpId(ep_id: string | number) { - return Async.ajax(`${this.server}/pgc/view/web/season?ep_id=${ep_id}`) - } - getSeasonInfo(season_id: string | number) { - return Async.ajax(`${this.server}/pgc/view/web/season?season_id=${season_id}`) + getSeasonInfoByEpSsId(ep_id: string | number | undefined, season_id: string | number | undefined) { + return Async.ajax(`${this.server}/pgc/view/web/season?` + (ep_id ? `ep_id=${ep_id}` : `season_id=${season_id}`)) } getSeasonInfoById(season_id: string, ep_id: string) { let paramDict = { @@ -627,9 +787,22 @@ export class BiliBiliApi { getEpisodeInfoByEpId(ep_id: string) { return Async.ajax('//api.bilibili.com/pgc/season/episode/web/info?' + `ep_id=${ep_id}`) } - getSeasonInfoByEpSsIdOnThailand(ep_id: string, season_id: string) { - const params = '?' + (ep_id != '' ? `ep_id=${ep_id}` : `season_id=${season_id}`) + `&mobi_app=bstar_a&s_locale=zh_SG` + getSeasonInfoByEpSsIdOnThailand(ep_id: string | undefined, season_id: string | undefined) { + const params = '?' + (ep_id ? `ep_id=${ep_id}` : `season_id=${season_id}`) + `&mobi_app=bstar_a&s_locale=zh_SG` const newParams = generateMobiPlayUrlParams(params, 'th') return Async.ajax(`${this.server}/intl/gateway/v2/ogv/view/app/season?` + newParams) } + + async getMediaInfoBySeasonId(season_id: string): Promise { + return Async.ajax(`//www.bilibili.com/bangumi/media/md${season_id}`) + .then(resp => { + const matchResult = (resp as string).match(/window\.__INITIAL_STATE__=(.*);\(function\(\)/) + if (matchResult) { + const initialState = JSON.parse(matchResult[1]) + return initialState.mediaInfo as MediaInfo + } + return Promise.reject(new Error('__INITIAL_STATE__ is not found.')) + }) + + } } diff --git a/packages/unblock-area-limit/src/feature/bili/area_limit_for_vue.ts b/packages/unblock-area-limit/src/feature/bili/area_limit_for_vue.ts index a6263db0..343c6967 100644 --- a/packages/unblock-area-limit/src/feature/bili/area_limit_for_vue.ts +++ b/packages/unblock-area-limit/src/feature/bili/area_limit_for_vue.ts @@ -173,7 +173,27 @@ function fixBangumiPlayPage() { // 如果该接口失效,自动尝试后面的方法 try { let result = await bilibiliApi.getSeasonInfoById(season_id, ep_id) - if (balh_config.server_custom_th && (result.code == -404)) { + if (result.code == -404) { + if (season_id) { + try { + let mediaInfo = await bilibiliApi.getMediaInfoBySeasonId(season_id) + if (mediaInfo.season_id) { + mediaInfo.refine_cover = decodeURI(mediaInfo.cover) + mediaInfo.share_copy = mediaInfo.title + mediaInfo.share_url = `https://www.bilibili.com/bangumi/play/ss${mediaInfo.season_id}` + mediaInfo.short_link = `https://b23.tv/ss${mediaInfo.season_id}` + mediaInfo.status = mediaInfo.season_status + mediaInfo.rights.area_limit = 0 + mediaInfo.rights.ban_area_show = 0 + mediaInfo.rights.is_preview = 0 + mediaInfo.staff = { info: mediaInfo.staff } + result = { code: 0, data: mediaInfo, message: "success" } + } + } catch (error) { + } + } + } + if (result.code != 0 && balh_config.server_custom_th) { result = await fixThailandSeason(ep_id, season_id) appOnly = true } @@ -182,15 +202,14 @@ function fixBangumiPlayPage() { } if (ep_id != '') season_id = result.data.season_id.toString() result.result = result.data - result.result.modules.forEach((module: { data: { [x: string]: any }; id: any }, mid: number) => { + result.result.modules?.forEach((module: { data: { [x: string]: any }; id: any }, mid: number) => { if (module.data) { let sid = module.id ? module.id : mid + 1 module.data['id'] = sid } }) let seasons: any[] = [] - - result.result.modules.forEach((module: { data: { seasons?: any[], episodes?: any[] } }) => { + result.result.modules?.forEach((module: { data: { seasons?: any[], episodes?: any[] } }) => { if (module.data.seasons) { module.data.seasons.forEach(season => { seasons.push(season) @@ -234,9 +253,6 @@ function fixBangumiPlayPage() { }) } const ep = ep_id != '' ? result.result.episodes.find(ep => ep.ep_id === +ep_id) : result.result.episodes[0] - if (!ep) { - throw `通过bangumi接口未找到${ep_id}对应的视频信息` - } const eps = JSON.stringify(result.result.episodes.map((item, index) => { // 返回的数据是有序的,不需要另外排序 if (/^\d+(\.\d+)?$/.exec(item.title)) { @@ -256,17 +272,17 @@ function fixBangumiPlayPage() { return item })) let titleForma - if (ep.index_title) { + if (ep?.index_title) { titleForma = ep.index_title } else { - titleForma = "第" + ep.index + "话" + titleForma = "第" + ep?.index + "话" } templateArgs = { - id: ep.ep_id, - aid: ep.aid, - cid: ep.cid, - bvid: ep.bvid, - title: ep.index, + id: ep?.ep_id, + aid: ep?.aid, + cid: ep?.cid, + bvid: ep?.bvid, + title: ep?.index, titleFormat: Strings.escapeSpecialChars(titleForma), htmlTitle: result.result.title, mediaInfoId: result.result.media_id, @@ -283,7 +299,7 @@ function fixBangumiPlayPage() { if (balh_config.server_bilibili_api_proxy && !templateArgs) { try { - const result = await bilibiliApi.getSeasonInfoByEpId(ep_id) + const result = await bilibiliApi.getSeasonInfoByEpSsId(ep_id, season_id) if (result.code) { throw result }