assembler.py 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  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('The segment directory does not exist or does not contain any files: %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('Video segment files from replay downloads cannot be assembled.')
  49. return
  50. stream_id = str(broadcast_info['id'])
  51. segment_meta = broadcast_info.get('segments', {})
  52. if segment_meta:
  53. all_segments = [
  54. os.path.join(ass_segment_dir, k)
  55. for k in broadcast_info['segments'].keys()]
  56. else:
  57. all_segments = list(filter(
  58. os.path.isfile,
  59. glob.glob(os.path.join(ass_segment_dir, '%s-*.m4v' % stream_id))))
  60. all_segments = sorted(all_segments, key=lambda x: _get_file_index(x))
  61. sources = []
  62. audio_stream_format = 'assembled_source_{0}_{1}_mp4.tmp'
  63. video_stream_format = 'assembled_source_{0}_{1}_m4a.tmp'
  64. video_stream = ''
  65. audio_stream = ''
  66. if not all_segments:
  67. logger.error("No video segment files have been found in the specified folder.")
  68. logger.separator()
  69. return
  70. else:
  71. logger.info("Assembling video segment files from specified folder: {}".format(ass_segment_dir))
  72. for segment in all_segments:
  73. if not os.path.isfile(segment.replace('.m4v', '.m4a')):
  74. logger.warn('Audio segment not found: {0!s}'.format(segment.replace('.m4v', '.m4a')))
  75. continue
  76. if segment.endswith('-init.m4v'):
  77. logger.info('Replacing %s' % segment)
  78. segment = os.path.join(
  79. os.path.dirname(os.path.realpath(__file__)), 'repair', 'init.m4v')
  80. if segment.endswith('-0.m4v'):
  81. continue
  82. video_stream = os.path.join(
  83. ass_segment_dir, video_stream_format.format(stream_id, len(sources)))
  84. audio_stream = os.path.join(
  85. ass_segment_dir, audio_stream_format.format(stream_id, len(sources)))
  86. file_mode = 'ab'
  87. with open(video_stream, file_mode) as outfile, open(segment, 'rb') as readfile:
  88. shutil.copyfileobj(readfile, outfile)
  89. with open(audio_stream, file_mode) as outfile, open(segment.replace('.m4v', '.m4a'), 'rb') as readfile:
  90. shutil.copyfileobj(readfile, outfile)
  91. if audio_stream and video_stream:
  92. sources.append({'video': video_stream, 'audio': audio_stream})
  93. for n, source in enumerate(sources):
  94. ffmpeg_binary = os.getenv('FFMPEG_BINARY', 'ffmpeg')
  95. cmd = [
  96. ffmpeg_binary, '-loglevel', 'warning', '-y',
  97. '-i', source['audio'],
  98. '-i', source['video'],
  99. '-c:v', 'copy', '-c:a', 'copy', ass_mp4_file]
  100. #fnull = open(os.devnull, 'w')
  101. fnull = None
  102. exit_code = subprocess.call(cmd, stdout=fnull, stderr=subprocess.STDOUT)
  103. if exit_code != 0:
  104. logger.warn("FFmpeg exit code not '0' but '{:d}'.".format(exit_code))
  105. else:
  106. logger.info('The video file has been generated: %s' % os.path.basename(ass_mp4_file))
  107. if user_called:
  108. logger.separator()
  109. except Exception as e:
  110. logger.error("An error occurred: {:s}".format(str(e)))