pyhuya.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. #coding=utf-8
  2. #!/usr/bin/python
  3. import sys
  4. sys.path.append('..')
  5. from base.spider import Spider
  6. import json
  7. import re
  8. import time
  9. import html
  10. import base64
  11. import hashlib
  12. import urllib.parse
  13. class Spider(Spider):
  14. def getName(self):
  15. return "虎牙"
  16. def init(self,extend=""):
  17. pass
  18. def isVideoFormat(self,url):
  19. pass
  20. def manualVideoCheck(self):
  21. pass
  22. def homeContent(self,filter):
  23. result = {}
  24. cateManual = {
  25. "音乐":"音乐",
  26. "星秀":"星秀",
  27. "颜值":"颜值",
  28. "交友":"交友",
  29. "户外":"户外",
  30. "美食":"美食",
  31. "一起看":"一起看",
  32. "王者荣耀":"王者荣耀",
  33. "和平精英":"和平精英",
  34. "英雄联盟":"英雄联盟",
  35. "天天吃鸡":"天天吃鸡",
  36. "穿越火线":"穿越火线",
  37. "二次元":"二次元",
  38. "体育":"体育",
  39. "原神":"原神",
  40. "三国杀":"三国杀",
  41. "暗黑破坏神:不朽":"暗黑破坏神:不朽",
  42. "迷你世界":"迷你世界",
  43. "暗区突围":"暗区突围",
  44. "生死狙击2":"生死狙击2",
  45. "金铲铲之战":"金铲铲之战",
  46. "英雄联盟手游":"英雄联盟手游",
  47. "lol云顶之弈":"lol云顶之弈",
  48. "剑侠世界3":"剑侠世界3",
  49. "不良人3":"不良人3",
  50. "二次元":"二次元",
  51. "主机游戏":"主机游戏"
  52. }
  53. classes = []
  54. for k in cateManual:
  55. classes.append({
  56. 'type_name': k,
  57. 'type_id': cateManual[k]
  58. })
  59. result['class'] = classes
  60. if (filter):
  61. result['filters'] = self.config['filter']
  62. return result
  63. def homeVideoContent(self):
  64. result = {}
  65. return result
  66. def categoryContent(self,tid,pg,filter,extend):
  67. result = {}
  68. url = 'http://live.yj1211.work/api/live/getRecommendByPlatformArea?platform=huya&size=20&area={0}&page={1}'.format(tid, pg)
  69. rsp = self.fetch(url)
  70. content = rsp.text
  71. jo = json.loads(content)
  72. videos = []
  73. vodList = jo['data']
  74. for vod in vodList:
  75. aid = (vod['roomId']).strip()
  76. title = vod['roomName'].strip()
  77. img = vod['roomPic'].strip()
  78. remark = (vod['ownerName']).strip()
  79. videos.append({
  80. "vod_id": aid,
  81. "vod_name": title,
  82. "vod_pic": img,
  83. "vod_remarks": remark
  84. })
  85. result['list'] = videos
  86. result['page'] = pg
  87. result['pagecount'] = 9999
  88. result['limit'] = 90
  89. result['total'] = 999999
  90. return result
  91. def detailContent(self,array):
  92. aid = array[0]
  93. url = 'https://www.huya.com/' + aid
  94. header = {
  95. 'Content-Type': 'application/x-www-form-urlencoded',
  96. 'User-Agent': 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36'
  97. }
  98. rsp = self.fetch(url, headers=header)
  99. streamInfo = re.findall(r'stream: ([\s\S]*?)\n', rsp.text)
  100. if (len(streamInfo) > 0):
  101. liveData = json.loads(streamInfo[0])
  102. else:
  103. streamInfo = re.findall(r'"stream": "([\s\S]*?)"', rsp.text)
  104. if (len(streamInfo) > 0):
  105. liveDataBase64 = streamInfo[0]
  106. liveData = json.loads(str(base64.b64decode(liveDataBase64), 'utf-8'))
  107. streamInfoList = liveData['data'][0]['gameStreamInfoList']
  108. vod = {
  109. "vod_id": aid,
  110. "vod_name": liveData['data'][0]['gameLiveInfo']['roomName'],
  111. "vod_pic": liveData['data'][0]['gameLiveInfo']['screenshot'],
  112. "type_name": liveData['data'][0]['gameLiveInfo']['gameFullName'],
  113. "vod_year": "",
  114. "vod_area": "",
  115. "vod_remarks": "",
  116. "vod_actor": "",
  117. "vod_director": "",
  118. "vod_content": ""
  119. }
  120. playUrl = ''
  121. for streamInfo in streamInfoList:
  122. hls_url = streamInfo['sHlsUrl'] + '/' + streamInfo['sStreamName'] + '.' + streamInfo['sHlsUrlSuffix']
  123. srcAntiCode = html.unescape(streamInfo['sHlsAntiCode'])
  124. c = srcAntiCode.split('&')
  125. c = [i for i in c if i != '']
  126. n = {i.split('=')[0]: i.split('=')[1] for i in c}
  127. fm = urllib.parse.unquote(n['fm'])
  128. u = base64.b64decode(fm).decode('utf-8')
  129. hash_prefix = u.split('_')[0]
  130. ctype = n.get('ctype', '')
  131. txyp = n.get('txyp', '')
  132. fs = n.get('fs', '')
  133. t = n.get('t', '')
  134. seqid = str(int(time.time() * 1e3 + 1463993859134))
  135. wsTime = hex(int(time.time()) + 3600).replace('0x', '')
  136. hash = hashlib.md5('_'.join([hash_prefix, '1463993859134', streamInfo['sStreamName'], hashlib.md5((seqid + '|' + ctype + '|' + t).encode('utf-8')).hexdigest(), wsTime]).encode('utf-8')).hexdigest()
  137. ratio = ''
  138. purl = "{}?wsSecret={}&wsTime={}&seqid={}&ctype={}&ver=1&txyp={}&fs={}&ratio={}&u={}&t={}&sv=2107230339".format( hls_url, hash, wsTime, seqid, ctype, txyp, fs, ratio, '1463993859134', t)
  139. playUrl = playUrl + '{}${}#'.format(streamInfo['sCdnType'], purl)
  140. vod['vod_play_from'] = '虎牙直播'
  141. vod['vod_play_url'] = playUrl
  142. result = {
  143. 'list': [
  144. vod
  145. ]
  146. }
  147. return result
  148. def searchContent(self,key,quick):
  149. result = {}
  150. return result
  151. def playerContent(self,flag,id,vipFlags):
  152. result = {}
  153. url = id
  154. result["parse"] = 0
  155. result["playUrl"] = ''
  156. result["url"] = url
  157. result["header"] = {
  158. "User-Agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/100.0.4896.127 Safari/537.36"
  159. }
  160. result["contentType"] = ''
  161. return result
  162. config = {
  163. "player": {},
  164. "filter": {}
  165. }
  166. header = {}
  167. config = {
  168. "player": {},
  169. "filter": {}
  170. }
  171. header = {}
  172. def localProxy(self,param):
  173. action = {
  174. 'url':'',
  175. 'header':'',
  176. 'param':'',
  177. 'type':'string',
  178. 'after':''
  179. }
  180. return [200, "video/MP2T", action, ""]