saitosan.py 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475
  1. from .common import InfoExtractor
  2. from ..utils import ExtractorError, try_get
  3. class SaitosanIE(InfoExtractor):
  4. _WORKING = False
  5. IE_NAME = 'Saitosan'
  6. _VALID_URL = r'https?://(?:www\.)?saitosan\.net/bview.html\?id=(?P<id>[0-9]+)'
  7. _TESTS = [{
  8. 'url': 'http://www.saitosan.net/bview.html?id=10031846',
  9. 'info_dict': {
  10. 'id': '10031846',
  11. 'ext': 'mp4',
  12. 'title': '井下原 和弥',
  13. 'uploader': '井下原 和弥',
  14. 'thumbnail': 'http://111.171.196.85:8088/921f916f-7f55-4c97-b92e-5d9d0fef8f5f/thumb',
  15. 'is_live': True,
  16. },
  17. 'params': {
  18. # m3u8 download
  19. 'skip_download': True,
  20. },
  21. 'skip': 'Broadcasts are ephemeral',
  22. },
  23. {
  24. 'url': 'http://www.saitosan.net/bview.html?id=10031795',
  25. 'info_dict': {
  26. 'id': '10031795',
  27. 'ext': 'mp4',
  28. 'title': '橋本',
  29. 'uploader': '橋本',
  30. 'thumbnail': 'http://111.171.196.85:8088/1a3933e1-a01a-483b-8931-af15f37f8082/thumb',
  31. 'is_live': True,
  32. },
  33. 'params': {
  34. # m3u8 download
  35. 'skip_download': True,
  36. },
  37. 'skip': 'Broadcasts are ephemeral',
  38. }]
  39. def _real_extract(self, url):
  40. b_id = self._match_id(url)
  41. base = 'http://hankachi.saitosan-api.net:8002/socket.io/?transport=polling&EIO=3'
  42. sid = self._download_socket_json(base, b_id, note='Opening socket').get('sid')
  43. base += '&sid=' + sid
  44. self._download_webpage(base, b_id, note='Polling socket')
  45. payload = f'420["room_start_join",{{"room_id":"{b_id}"}}]'
  46. payload = f'{len(payload)}:{payload}'
  47. self._download_webpage(base, b_id, data=payload, note='Polling socket with payload')
  48. response = self._download_socket_json(base, b_id, note='Polling socket')
  49. if not response.get('ok'):
  50. err = response.get('error') or {}
  51. raise ExtractorError(
  52. '{} said: {} - {}'.format(self.IE_NAME, err.get('code', '?'), err.get('msg', 'Unknown')) if err
  53. else 'The socket reported that the broadcast could not be joined. Maybe it\'s offline or the URL is incorrect',
  54. expected=True, video_id=b_id)
  55. self._download_webpage(base, b_id, data='26:421["room_finish_join",{}]', note='Polling socket')
  56. b_data = self._download_socket_json(base, b_id, note='Getting broadcast metadata from socket')
  57. m3u8_url = b_data.get('url')
  58. self._download_webpage(base, b_id, data='1:1', note='Closing socket', fatal=False)
  59. return {
  60. 'id': b_id,
  61. 'title': b_data.get('name'),
  62. 'formats': self._extract_m3u8_formats(m3u8_url, b_id, 'mp4', live=True),
  63. 'thumbnail': m3u8_url.replace('av.m3u8', 'thumb'),
  64. 'uploader': try_get(b_data, lambda x: x['broadcast_user']['name']), # same as title
  65. 'is_live': True,
  66. }