mirror of
				https://github.com/qist/tvbox.git
				synced 2025-10-26 01:32:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			412 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			412 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #coding=utf-8
 | |
| #!/usr/bin/python
 | |
| import sys
 | |
| sys.path.append('..')
 | |
| from base.spider import Spider
 | |
| import base64
 | |
| import math
 | |
| import json
 | |
| import requests
 | |
| import urllib
 | |
| from urllib import request, parse
 | |
| import urllib.request
 | |
| import re
 | |
| 
 | |
| class Spider(Spider):
 | |
| 	def getName(self):
 | |
| 		return "西瓜视频(个人中心)"
 | |
| 	def init(self,extend=""):
 | |
| 		self.userid=self.get_userid()
 | |
| 	def isVideoFormat(self,url):
 | |
| 		pass
 | |
| 	def manualVideoCheck(self):
 | |
| 		pass
 | |
| 	def homeContent(self,filter):
 | |
| 		result = {}
 | |
| 		cateManual = {
 | |
| 			"电视剧":"dianshiju",
 | |
| 			"电影":"dianying",
 | |
| 			"动漫":"dongman",
 | |
| 			"纪录片":"jilupian",
 | |
| 			"少儿":"shaoer",
 | |
| 			"综艺":"zongyi",
 | |
| 			"关注":"follow"
 | |
| 
 | |
| 		}
 | |
| 		if self.userid=='':
 | |
| 			del cateManual['关注']
 | |
| 		classes = []
 | |
| 		for k in cateManual:
 | |
| 			classes.append({
 | |
| 				'type_name': k,
 | |
| 				'type_id': cateManual[k]
 | |
| 			})
 | |
| 
 | |
| 		result['class'] = classes
 | |
| 		if (filter):
 | |
| 			result['filters'] = self.config['filter']
 | |
| 		return result
 | |
| 	def homeVideoContent(self):
 | |
| 		result = {
 | |
| 			'list': []
 | |
| 		}
 | |
| 		return result
 | |
| 	userid=''
 | |
| 	def categoryContent(self,tid,pg,filter,extend):
 | |
| 		result = {}
 | |
| 		idTxt='电视剧'
 | |
| 		maximum=17
 | |
| 		url = 'https://www.ixigua.com/api/cinema/filterv2/albums'
 | |
| 		if tid=='dianying':
 | |
| 			idTxt='电影'
 | |
| 		elif tid=='zongyi':
 | |
| 			idTxt='综艺'
 | |
| 		elif tid=='dianshiju':
 | |
| 			idTxt='电视剧'
 | |
| 		elif tid=='dongman':
 | |
| 			idTxt='动漫'
 | |
| 		elif tid=='jilupian':
 | |
| 			idTxt='纪录片'
 | |
| 		elif tid=='shaoer':
 | |
| 			idTxt='少儿'	
 | |
| 		elif tid=='follow':
 | |
| 			offset=0 if int(pg)<2 else 20*int(pg)
 | |
| 			url='https://www.ixigua.com/api/userv2/follow/list?authorId={0}&sortType=desc&sortType=desc&cursor={1}'.format(self.userid,offset)
 | |
| 			maximum=12
 | |
| 		videos=[]
 | |
| 		if tid!='follow':
 | |
| 			offset=0 if int(pg)<2 else 18*int(pg)
 | |
| 			self.header['Referer']='https://www.ixigua.com/cinema/filter/'.format(tid)
 | |
| 			data=r'{"pinyin":"'+tid+'","filters":{"type":"'+idTxt+'","area":"全部地区","tag":"全部类型","sort":"综合排序","paid":"全部资费"},"offset":'+str(offset)+',"limit":18}'
 | |
| 			req = request.Request(url=url, data=bytes(data, encoding='utf8'),headers=self.header, method='POST')
 | |
| 			response = request.urlopen(req)
 | |
| 			urlTxt=response.read().decode('utf-8')
 | |
| 			videos= self.get_list_videoGroup_json(jsonTxt=urlTxt)
 | |
| 		else:
 | |
| 			rsp=self.fetch(url,headers=self.header)
 | |
| 			urlTxt=rsp.text
 | |
| 			videos= self.get_list_videoGroup_follow_json(jsonTxt=urlTxt)
 | |
| 		numvL = len(videos)
 | |
| 		result['list'] = videos
 | |
| 		result['page'] = pg
 | |
| 		result['pagecount'] = pg if int(numvL)<maximum else int(pg)+1
 | |
| 		result['limit'] = numvL
 | |
| 		result['total'] = numvL
 | |
| 		return result
 | |
| 	def get_userid(self):
 | |
| 		Url='https://www.ixigua.com/'
 | |
| 		rsp=self.fetch(Url,headers=self.header)
 | |
| 		htmlTxt = rsp.text
 | |
| 		userid= self.get_RegexGetText(Text=htmlTxt,RegexText=r'"identity":{"id":"(\d+?)",',Index=1)
 | |
| 		return userid
 | |
| 	def detailContent(self,array):
 | |
| 		result = {}
 | |
| 		aid = array[0].split('###')
 | |
| 		key = aid[1]
 | |
| 		title = aid[0]
 | |
| 		act=aid[2]
 | |
| 		logo = aid[3]
 | |
| 		Url='https://www.ixigua.com/api/albumv2/details?albumId={0}'.format(key)
 | |
| 		if len(aid)==5:
 | |
| 			Url='https://www.ixigua.com/api/videov2/author/new_video_list?to_user_id={0}'.format(key)
 | |
| 		rsp = self.fetch(Url,headers=self.header)
 | |
| 		htmlTxt = rsp.text
 | |
| 		typeName=''
 | |
| 		area=''
 | |
| 		dir=''
 | |
| 		cont=''
 | |
| 		vip='true'
 | |
| 		videoList=[]
 | |
| 		if len(aid)==5:
 | |
| 			jRoot = json.loads(htmlTxt)
 | |
| 			if jRoot['code']!=200:
 | |
| 				return result
 | |
| 			jo = jRoot['data']
 | |
| 			jsonList=jo['videoList']
 | |
| 			for value in jsonList:
 | |
| 					id="{0}${1}_false".format(value['title'],value.get('group_id'))
 | |
| 					videoList.append(id)
 | |
| 			dir=title
 | |
| 		elif htmlTxt.find('playlist')>2:
 | |
| 			jRoot = json.loads(htmlTxt)
 | |
| 			if jRoot['code']!=200:
 | |
| 				return result
 | |
| 			jo = jRoot['data']
 | |
| 			jsonList=jo['playlist']
 | |
| 			if jsonList is not None:
 | |
| 				for value in jsonList:
 | |
| 					id="{0}${1}?id={2}_{3}".format(value['title'],value['albumId'],value['episodeId'],vip)
 | |
| 					videoList.append(id)
 | |
| 			playFrom=[v for v in jo['albumInfo']['tagList']]
 | |
| 			typeName='/'.join(playFrom)
 | |
| 			playFrom=[v for v in jo['albumInfo']['areaList']]
 | |
| 			area='/'.join(playFrom)
 | |
| 			playFrom=[v['name'] for v in jo['albumInfo']['directorList']]
 | |
| 			dir='/'.join(playFrom)
 | |
| 			cont=jo['albumInfo']['intro']
 | |
| 		if len(videoList)<1:
 | |
| 			return result
 | |
| 		vod = {
 | |
| 			"vod_id":array[0],
 | |
| 			"vod_name":title,
 | |
| 			"vod_pic":logo,
 | |
| 			"type_name":typeName,
 | |
| 			"vod_year":'',
 | |
| 			"vod_area":area,
 | |
| 			"vod_remarks":"",
 | |
| 			"vod_actor":'',
 | |
| 			"vod_director":dir,
 | |
| 			"vod_content":cont
 | |
| 		}
 | |
| 		vod['vod_play_from'] = '西瓜'
 | |
| 		vod['vod_play_url'] = "#".join(videoList)
 | |
| 		result = {
 | |
| 			'list':[
 | |
| 				vod
 | |
| 			]
 | |
| 		}
 | |
| 		return result
 | |
| 
 | |
| 	def verifyCode(self):
 | |
| 		pass
 | |
| 
 | |
| 	def searchContent(self,key,quick):
 | |
| 		Url='https://www.ixigua.com/api/searchv2/lvideo/{0}/0'.format(urllib.parse.quote(key))
 | |
| 		rsp = self.fetch(Url,headers=self.header)
 | |
| 		htmlTxt = rsp.text
 | |
| 		videos=self.get_list(html=htmlTxt)
 | |
| 		'''
 | |
| 		Url='https://www.ixigua.com/api/searchv2/user/{0}/10'.format(urllib.parse.quote(key))
 | |
| 		rsp = self.fetch(Url,headers=self.header)
 | |
| 		htmlTxt1 = rsp.text
 | |
| 		videos=self.get_list_user(html=htmlTxt1)
 | |
| 		'''
 | |
| 		result = {
 | |
| 				'list': videos
 | |
| 			}
 | |
| 		return result
 | |
| 
 | |
| 	def playerContent(self,flag,id,vipFlags):
 | |
| 		result={}
 | |
| 		UrlId=id.split('_')
 | |
| 		Url='https://www.ixigua.com/{0}'.format(UrlId[0])
 | |
| 		headers = {
 | |
| 			'User-Agent':'Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/69.0.3947.100 Mobile Safari/537.36'
 | |
| 		}
 | |
| 		jx=1 if UrlId[1]=='true' else 0
 | |
| 		result["parse"] = 1#0=直接播放,1=解析
 | |
| 		result["playUrl"] = ''
 | |
| 		result["url"] = Url
 | |
| 		result['jx'] = jx#VIP解析
 | |
| 		result["header"] =headers
 | |
| 		return result
 | |
| 	def get_RegexGetText(self,Text,RegexText,Index):
 | |
| 		returnTxt=""
 | |
| 		Regex=re.search(RegexText, Text, re.M|re.I)
 | |
| 		if Regex is None:
 | |
| 			returnTxt=""
 | |
| 		else:
 | |
| 			returnTxt=Regex.group(Index)
 | |
| 		return returnTxt	
 | |
| 	def get_RegexGetTextLine(self,Text,RegexText,Index):
 | |
| 		returnTxt=[]
 | |
| 		pattern = re.compile(RegexText)
 | |
| 		ListRe=pattern.findall(Text)
 | |
| 		if len(ListRe)<1:
 | |
| 			return returnTxt
 | |
| 		for value in ListRe:
 | |
| 			returnTxt.append(value)	
 | |
| 		return returnTxt
 | |
| 	def get_playlist(self,Text,headStr,endStr):
 | |
| 		circuit=""
 | |
| 		origin=Text.find(headStr)
 | |
| 		if origin>8:
 | |
| 			end=Text.find(endStr,origin)
 | |
| 			circuit=Text[origin:end]
 | |
| 		return circuit
 | |
| 	def removeHtml(self,txt):
 | |
| 		soup = re.compile(r'<[^>]+>',re.S)
 | |
| 		txt =soup.sub('', txt)
 | |
| 		return txt.replace(" "," ")
 | |
| 	def get_webReadFile(self,urlStr):
 | |
| 		headers = {
 | |
| 			'Referer':urlStr,
 | |
| 			'User-Agent': 'Mozilla/5.0 (Windows NT 5.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/49.0.2623.112 Safari/537.36',
 | |
| 			'Host': 'www.ikmjw.com'
 | |
| 		}
 | |
| 		req = urllib.request.Request(url=urlStr, headers=headers)
 | |
| 		html = urllib.request.urlopen(req).read().decode('utf-8')
 | |
| 		return html
 | |
| 	def get_list(self,html):
 | |
| 		result={}
 | |
| 		jRoot = json.loads(html)
 | |
| 		if jRoot['code']!=0:
 | |
| 			return result
 | |
| 		jo = jRoot['data']
 | |
| 		if len(jo)<1:
 | |
| 			return result
 | |
| 		vodList = jo['data']
 | |
| 		if len(vodList)<1:
 | |
| 			return result
 | |
| 		videos=[]
 | |
| 		img='_'
 | |
| 		artist='_'
 | |
| 		for vod in vodList:
 | |
| 			if vod['type']!='lvideo':
 | |
| 				continue
 | |
| 			data=vod['data']
 | |
| 			display=data['display']
 | |
| 			title =display['name']
 | |
| 			if len(title)==0:
 | |
| 				continue
 | |
| 			episode_link=display.get('episode_link')
 | |
| 			asc_link=episode_link.get('asc_link')
 | |
| 			KeyName='album_id'
 | |
| 			if asc_link is None:
 | |
| 				asc_link=episode_link.get('desc_link')
 | |
| 				KeyName='album_id'
 | |
| 			id=asc_link[0].get('album_id')
 | |
| 			sslocal=asc_link[0]['scheme_url']
 | |
| 			if sslocal!=None and sslocal.find('sslocal:')<0:
 | |
| 				continue
 | |
| 			video_cover_info=display.get('video_cover_info')
 | |
| 			img=video_cover_info['url'] if video_cover_info is not None else '_'
 | |
| 			try:
 | |
| 				artist=display['actor']
 | |
| 			except Exception:
 | |
| 				artist=''
 | |
| 			remarks=display['rating']
 | |
| 			vod_id="{0}###{1}###{2}###{3}".format(title,id,artist,img)
 | |
| 			videos.append({
 | |
| 				"vod_id":vod_id,
 | |
| 				"vod_name":title,
 | |
| 				"vod_pic":img,
 | |
| 				"vod_remarks":remarks
 | |
| 			})
 | |
| 		return videos
 | |
| 	def get_list_user(self,html):
 | |
| 		result={}
 | |
| 		jRoot = json.loads(html)
 | |
| 		if jRoot['code']!=0:
 | |
| 			return result
 | |
| 		jo = jRoot['data']
 | |
| 		if len(jo)<1:
 | |
| 			return result
 | |
| 		vodList = jo['data']
 | |
| 		if len(vodList)<1:
 | |
| 			return result
 | |
| 		videos=[]
 | |
| 		img='_'
 | |
| 		artist='_'
 | |
| 		for vod in vodList:
 | |
| 			if vod['type']!='user':
 | |
| 				continue
 | |
| 			data=vod['data']
 | |
| 			img=data['avatar']
 | |
| 			title =data['name']
 | |
| 			log_pb=json.loads(data.get('log_pb'))
 | |
| 			user_auth_info=json.loads(data.get('user_auth_info'))
 | |
| 			remarks=user_auth_info['auth_info']
 | |
| 			id=log_pb.get('search_result_id')
 | |
| 			vod_id="{0}###{1}###{2}###{3}###{4}".format(title,id,artist,img,'user')
 | |
| 			videos.append({
 | |
| 				"vod_id":vod_id,
 | |
| 				"vod_name":title,
 | |
| 				"vod_pic":img,
 | |
| 				"vod_remarks":remarks
 | |
| 			})
 | |
| 		return videos
 | |
| 	def get_list_videoGroup_json(self,jsonTxt):
 | |
| 		result={}
 | |
| 		jRoot = json.loads(jsonTxt)
 | |
| 		if jRoot['code']!=200:
 | |
| 			return result
 | |
| 		jo = jRoot['data']
 | |
| 		vodList = jo['albumList']
 | |
| 		if len(vodList)<1:
 | |
| 			return result
 | |
| 		videos=[]
 | |
| 		img='_'
 | |
| 		artist='_'
 | |
| 		for vod in vodList:
 | |
| 			url =vod['albumId']
 | |
| 			title =vod['title']
 | |
| 			imgList =vod.get('coverList') 
 | |
| 			if len(imgList)>0:
 | |
| 				img=imgList[0]['url']
 | |
| 			remarks=vod['subTitle']
 | |
| 			artistList=vod.get('actorList') 
 | |
| 			if artistList is not None:
 | |
| 				artistList=artistList if len(artistList)<5 else artistList[0:4]
 | |
| 				artist='/'.join(artistList)
 | |
| 			if len(title)==0:
 | |
| 				continue
 | |
| 			#标题###地址###演员###封面
 | |
| 			vod_id="{0}###{1}###{2}###{3}".format(title,url,artist,img)
 | |
| 			videos.append({
 | |
| 				"vod_id":vod_id,
 | |
| 				"vod_name":title,
 | |
| 				"vod_pic":img,
 | |
| 				"vod_remarks":remarks
 | |
| 			})
 | |
| 		return videos
 | |
| 	def get_list_videoGroup_follow_json(self,jsonTxt):
 | |
| 		videos=[]
 | |
| 		jRoot = json.loads(jsonTxt)
 | |
| 		if jRoot['code']!=0:
 | |
| 			return videos
 | |
| 		jo = jRoot['data']
 | |
| 		vodList=jo['data']
 | |
| 		if len(vodList)<1:
 | |
| 			return videos
 | |
| 		img='_'
 | |
| 		artist=''
 | |
| 		for vod in vodList:
 | |
| 			url =vod.get('user_id') 
 | |
| 			title =vod['name']
 | |
| 			img =vod.get('avatar_url') 
 | |
| 			remarks=vod['description']
 | |
| 			artistList=vod.get('actorList') 
 | |
| 			artist=title
 | |
| 			if len(title)==0:
 | |
| 				continue
 | |
| 			#标题###地址###演员###封面
 | |
| 			vod_id="{0}###{1}###{2}###{3}###{4}".format(title,url,artist,img,'user')
 | |
| 			videos.append({
 | |
| 				"vod_id":vod_id,
 | |
| 				"vod_name":title,
 | |
| 				"vod_pic":img,
 | |
| 				"vod_remarks":remarks
 | |
| 			})
 | |
| 		return videos
 | |
| 	def get_lineList(self,Txt,mark,after):
 | |
| 		circuit=[]
 | |
| 		origin=Txt.find(mark)
 | |
| 		while origin>8:
 | |
| 			end=Txt.find(after,origin)
 | |
| 			circuit.append(Txt[origin:end])
 | |
| 			origin=Txt.find(mark,end)
 | |
| 		return circuit
 | |
| 	def get_EpisodesList(self,jsonList):
 | |
| 		vodItems=[]
 | |
| 		for value in jsonList:
 | |
| 			vodItems.append(value['title']+"$"+'https://www.ixigua.com/{0}?logTag=55abe18cfb733871bb04'.format(value['episodeId']))
 | |
| 		return vodItems
 | |
| 	config = {
 | |
| 		"player": {},
 | |
| 		"filter": {}
 | |
| 	}
 | |
| 	header = {
 | |
| 		"Cookie":"s_v_web_id=verify_lev3h43l_rrTPrFDG_ztWQ_4ugg_8WBA_yGVYsXlVyoBh; passport_csrf_token=80e0efe90bc8bd6681a896dd90cd08cc; passport_csrf_token_default=80e0efe90bc8bd6681a896dd90cd08cc; __ac_nonce=0643361890096533c765; __ac_signature=_02B4Z6wo00f01JPVVrAAAIDBcisHPfWA66CT91IAAEDK9840CE-PheNOCgA4VtrFG0-K.KkLmAR5KsI-Xx-6dBXxz.ABWU2OpEd22kF7biwGaVmGR7an4S1heLEU9xpv0ObRSHFHDslR7uL8fb; support_webp=true; support_avif=false; MONITOR_WEB_ID=45c3b6ab-7ad4-4805-b971-5962d1d6909a; ttwid=1%7CCueNR-HU9tGVF30WaiFCjXDxh0FUXoXsZr-cIb9Dogg%7C1681089268%7Cf0eeaa2016a602a277055494954f083e9f7fa8121c5dd1162db9195932fa167b; odin_tt=386a42a5740f9859d4670373fc8c70cf320ea5b227aed04bcc53fde26c233e8c952a8b0ba5f80fd3d46f9663fa595d8c; sid_guard=842b56710f55021912487890e7d5bef3%7C1681089337%7C3024001%7CMon%2C+15-May-2023+01%3A15%3A38+GMT; uid_tt=b7eb5b73cd72bed6d3c2e1e1e8aaa9f3; uid_tt_ss=b7eb5b73cd72bed6d3c2e1e1e8aaa9f3; sid_tt=842b56710f55021912487890e7d5bef3; sessionid=842b56710f55021912487890e7d5bef3; sessionid_ss=842b56710f55021912487890e7d5bef3; sid_ucp_v1=1.0.0-KGQ2YzBlMDFiMzIyMjY0YTIwMDg2MjZmZGQzMTE5MmFlYTYzY2EwMTMKFQjL2cnx9AIQucbNoQYYGCAMOAhABRoCaGwiIDg0MmI1NjcxMGY1NTAyMTkxMjQ4Nzg5MGU3ZDViZWYz; ssid_ucp_v1=1.0.0-KGQ2YzBlMDFiMzIyMjY0YTIwMDg2MjZmZGQzMTE5MmFlYTYzY2EwMTMKFQjL2cnx9AIQucbNoQYYGCAMOAhABRoCaGwiIDg0MmI1NjcxMGY1NTAyMTkxMjQ4Nzg5MGU3ZDViZWYz; csrf_session_id=c58adecac1d20d91d8c61e72ce0c6fdb; ixigua-a-s=3; msToken=St3ptsHkwBjPgGvRWhZfVdhHO_K16vTizxxna17draCvt4ekz6DQXO6c2Ctrp6gOnn9_Abm83-a9URQp5rKb-JIPs4mQPn2fpKlQRY6jLDBWUdLYTePIAACU6cY2fk40; tt_scid=nN3WPItw72gnB5PDiGcHveirXDZ8oDl5n.ihKN583mmZhoe.uLhFpy3JOL8wrsQO0ed1",
 | |
| 		"Referer": 'https://www.ixigua.com/cinema/filter/dianshiju/',
 | |
| 		'User-Agent':'User-Agent: Mozilla%2F5.0+(Windows+NT+10.0%3B+WOW64)+AppleWebKit%2F537.36+(KHTML%2C+like+Gecko)+Chrome%2F63.0.3239.132+Safari%2F537.36',
 | |
| 		'Host': 'www.ixigua.com',
 | |
| 		'Accept': 'application/json, text/plain, */*',
 | |
| 		'x-secsdk-csrf-token': '0001000000017b593ba6251b18bd7bce2753042917bb36e534867b9606317584c00b0ae836c61754314b7365128e',
 | |
| 		'tt-anti-token': 'oDr7A3PDDFq4pWzk-707faf92a9e3040f5c6ed4284d53b05b7091221852c0e9d32bca9fcfe5035225',
 | |
| 		'content-type': 'application/json'	
 | |
| 	}
 | |
| 
 | |
| 	def localProxy(self,param):
 | |
| 		return [200, "video/MP2T", action, ""]
 |