assembler.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. import os
  2. import shutil
  3. import re
  4. import glob
  5. import subprocess
  6. import json
  7. import sys
  8. try:
  9. import pil
  10. import logger
  11. import helpers
  12. from constants import Constants
  13. except ImportError:
  14. from . import pil
  15. from . import logger
  16. from . import helpers
  17. from .constants import Constants
  18. """
  19. The content of this file was originally written by https://github.com/taengstagram
  20. The code has been edited for use in PyInstaLive.
  21. """
  22. def _get_file_index(filename):
  23. """ Extract the numbered index in filename for sorting """
  24. mobj = re.match(r'.+\-(?P<idx>[0-9]+)\.[a-z]+', filename)
  25. if mobj:
  26. return int(mobj.group('idx'))
  27. return -1
  28. def assemble(user_called=True):
  29. try:
  30. ass_json_file = pil.assemble_arg if pil.assemble_arg.endswith(".json") else pil.assemble_arg + ".json"
  31. ass_mp4_file = os.path.join(pil.dl_path, os.path.basename(ass_json_file).replace("_downloads", "").replace(".json", ".mp4"))
  32. ass_segment_dir = pil.assemble_arg.split('.')[0]
  33. broadcast_info = {}
  34. if not os.path.isdir(ass_segment_dir) or not os.listdir(ass_segment_dir):
  35. logger.error('Segment directory does not exist or is empty: %s' % ass_segment_dir)
  36. logger.separator()
  37. return
  38. if not os.path.isfile(ass_json_file):
  39. logger.warn("No matching json file found for the segment directory, trying to continue without it.")
  40. ass_stream_id = os.listdir(ass_segment_dir)[0].split('-')[0]
  41. broadcast_info['id'] = ass_stream_id
  42. broadcast_info['broadcast_status'] = "active"
  43. broadcast_info['segments'] = {}
  44. else:
  45. with open(ass_json_file) as info_file:
  46. broadcast_info = json.load(info_file)
  47. if broadcast_info.get('broadcast_status', '') == 'post_live':
  48. logger.error('Segments from replay downloads cannot be assembled.')
  49. return
  50. logger.info("Assembling video segments from folder: {}".format(ass_segment_dir))
  51. stream_id = str(broadcast_info['id'])
  52. segment_meta = broadcast_info.get('segments', {})
  53. if segment_meta:
  54. all_segments = [
  55. os.path.join(ass_segment_dir, k)
  56. for k in broadcast_info['segments'].keys()]
  57. else:
  58. all_segments = list(filter(
  59. os.path.isfile,
  60. glob.glob(os.path.join(ass_segment_dir, '%s-*.m4v' % stream_id))))
  61. all_segments = sorted(all_segments, key=lambda x: _get_file_index(x))
  62. sources = []
  63. audio_stream_format = 'assembled_source_{0}_{1}_mp4.tmp'
  64. video_stream_format = 'assembled_source_{0}_{1}_m4a.tmp'
  65. video_stream = ''
  66. audio_stream = ''
  67. for segment in all_segments:
  68. if not os.path.isfile(segment.replace('.m4v', '.m4a')):
  69. logger.warn('Audio segment not found: {0!s}'.format(segment.replace('.m4v', '.m4a')))
  70. continue
  71. if segment.endswith('-init.m4v'):
  72. logger.info('Replacing %s' % segment)
  73. segment = os.path.join(
  74. os.path.dirname(os.path.realpath(__file__)), 'repair', 'init.m4v')
  75. if segment.endswith('-0.m4v'):
  76. continue
  77. video_stream = os.path.join(
  78. ass_segment_dir, video_stream_format.format(stream_id, len(sources)))
  79. audio_stream = os.path.join(
  80. ass_segment_dir, audio_stream_format.format(stream_id, len(sources)))
  81. file_mode = 'ab'
  82. with open(video_stream, file_mode) as outfile, open(segment, 'rb') as readfile:
  83. shutil.copyfileobj(readfile, outfile)
  84. with open(audio_stream, file_mode) as outfile, open(segment.replace('.m4v', '.m4a'), 'rb') as readfile:
  85. shutil.copyfileobj(readfile, outfile)
  86. if audio_stream and video_stream:
  87. sources.append({'video': video_stream, 'audio': audio_stream})
  88. for n, source in enumerate(sources):
  89. ffmpeg_binary = os.getenv('FFMPEG_BINARY', 'ffmpeg')
  90. cmd = [
  91. ffmpeg_binary, '-loglevel', 'warning', '-y',
  92. '-i', source['audio'],
  93. '-i', source['video'],
  94. '-c:v', 'copy', '-c:a', 'copy', ass_mp4_file]
  95. fnull = open(os.devnull, 'w')
  96. exit_code = subprocess.call(cmd, stdout=fnull, stderr=subprocess.STDOUT)
  97. if exit_code != 0:
  98. logger.warn("FFmpeg exit code not '0' but '{:d}'.".format(exit_code))
  99. logger.separator()
  100. logger.info('The video file has been generated: %s' % os.path.basename(ass_mp4_file))
  101. if user_called:
  102. logger.separator()
  103. except Exception as e:
  104. logger.error("An error occurred: {:s}".format(str(e)))