mirror of
https://github.com/yuanwangokk-1/TV-BOX.git
synced 2025-10-24 23:11:21 +00:00
155 lines
No EOL
4.7 KiB
Python
155 lines
No EOL
4.7 KiB
Python
# -*- coding: utf-8 -*-
|
||
# @Author : Doubebly
|
||
# @Time : 2025/3/23 21:55
|
||
import base64
|
||
import sys
|
||
import time
|
||
import json
|
||
import requests
|
||
import re # 新增导入re模块
|
||
sys.path.append('..')
|
||
from base.spider import Spider
|
||
|
||
|
||
class Spider(Spider):
|
||
def getName(self):
|
||
return "Litv"
|
||
|
||
def init(self, extend):
|
||
self.extend = extend
|
||
try:
|
||
self.extendDict = json.loads(extend)
|
||
except:
|
||
self.extendDict = {}
|
||
|
||
proxy = self.extendDict.get('proxy', None)
|
||
if proxy is None:
|
||
self.is_proxy = False
|
||
else:
|
||
self.proxy = proxy
|
||
self.is_proxy = True
|
||
pass
|
||
|
||
def getDependence(self):
|
||
return []
|
||
|
||
def isVideoFormat(self, url):
|
||
pass
|
||
|
||
def manualVideoCheck(self):
|
||
pass
|
||
|
||
def natural_sort_key(self, s):
|
||
"""
|
||
自然排序辅助函数
|
||
"""
|
||
return [
|
||
int(part) if part.isdigit() else part.lower()
|
||
for part in re.split(r'(\d+)', s)
|
||
]
|
||
|
||
def liveContent(self, url):
|
||
# 初始化默认M3U内容(至少包含EXTM3U声明)
|
||
a = ['#EXTM3U']
|
||
|
||
try:
|
||
base_url = "https://kzb29rda.com/prod-api/iptv/getIptvList?liveType=0&deviceType=1"
|
||
response = requests.get(base_url)
|
||
response.raise_for_status() # 自动抛出HTTP错误(如404/500)
|
||
|
||
data = response.json()
|
||
|
||
sorted_list = sorted(
|
||
data.get('list', []),
|
||
key=lambda x: self.natural_sort_key(x.get("play_source_name", ""))
|
||
)
|
||
|
||
channels = [
|
||
element
|
||
#for item in data.get('list', [])
|
||
for item in sorted_list
|
||
for element in (
|
||
f'#EXTINF:-1 tvg-id="{item["play_source_name"]}" tvg-name="{item["play_source_name"]}" '
|
||
f'tvg-logo="https://logo.doube.eu.org/{item["play_source_name"]}.png" group-title="",'
|
||
f'{item["play_source_name"]}',
|
||
item['play_source_url']
|
||
)
|
||
]
|
||
a += channels # 合并到初始化的a中
|
||
|
||
except requests.exceptions.RequestException as e:
|
||
print(f"网络请求失败: {e}")
|
||
a.append('# 错误:无法获取频道列表')
|
||
except KeyError as e:
|
||
print(f"数据解析错误,缺少字段: {e}")
|
||
a.append('# 错误:数据格式异常')
|
||
except json.JSONDecodeError:
|
||
print("响应内容不是有效的JSON")
|
||
a.append('# 错误:无效的API响应')
|
||
|
||
return '\n'.join(a)
|
||
|
||
def homeContent(self, filter):
|
||
return {}
|
||
|
||
def homeVideoContent(self):
|
||
return {}
|
||
|
||
def categoryContent(self, cid, page, filter, ext):
|
||
return {}
|
||
|
||
def detailContent(self, did):
|
||
return {}
|
||
|
||
def searchContent(self, key, quick, page='1'):
|
||
return {}
|
||
|
||
def searchContentPage(self, keywords, quick, page):
|
||
return {}
|
||
|
||
def playerContent(self, flag, pid, vipFlags):
|
||
return {}
|
||
|
||
def localProxy(self, params):
|
||
if params['type'] == "m3u8":
|
||
return self.proxyM3u8(params)
|
||
if params['type'] == "ts":
|
||
return self.get_ts(params)
|
||
return [302, "text/plain", None, {'Location': 'https://sf1-cdn-tos.huoshanstatic.com/obj/media-fe/xgplayer_doc_video/mp4/xgplayer-demo-720p.mp4'}]
|
||
def proxyM3u8(self, params):
|
||
pid = params['pid']
|
||
info = pid.split(',')
|
||
a = info[0]
|
||
b = info[1]
|
||
c = info[2]
|
||
timestamp = int(time.time() / 4 - 355017625)
|
||
t = timestamp * 4
|
||
m3u8_text = f'#EXTM3U\n#EXT-X-VERSION:3\n#EXT-X-TARGETDURATION:4\n#EXT-X-MEDIA-SEQUENCE:{timestamp}\n'
|
||
for i in range(10):
|
||
url = f'https://ntd-tgc.cdn.hinet.net/live/pool/{a}/litv-pc/{a}-avc1_6000000={b}-mp4a_134000_zho={c}-begin={t}0000000-dur=40000000-seq={timestamp}.ts'
|
||
if self.is_proxy:
|
||
url = f'http://127.0.0.1:9978/proxy?do=py&type=ts&url={self.b64encode(url)}'
|
||
|
||
m3u8_text += f'#EXTINF:4,\n{url}\n'
|
||
timestamp += 1
|
||
t += 4
|
||
return [200, "application/vnd.apple.mpegurl", m3u8_text]
|
||
|
||
def get_ts(self, params):
|
||
url = self.b64decode(params['url'])
|
||
headers = {'User-Agent': 'Mozilla/5.0'}
|
||
response = requests.get(url, headers=headers, stream=True, proxies=self.proxy)
|
||
return [206, "application/octet-stream", response.content]
|
||
|
||
def destroy(self):
|
||
return '正在Destroy'
|
||
|
||
def b64encode(self, data):
|
||
return base64.b64encode(data.encode('utf-8')).decode('utf-8')
|
||
|
||
def b64decode(self, data):
|
||
return base64.b64decode(data.encode('utf-8')).decode('utf-8')
|
||
|
||
|
||
if __name__ == '__main__':
|
||
pass |