from datetime import datetime, timedelta from time import sleep import os from subprocess import Popen import logging from config_manager import get_config def get_timefile_exact(time): ''' time is of type `datetime`; it is not "rounded" to match the real file; that work is done in get_timefile(time) ''' return os.path.join( get_config()['AUDIO_INPUT'], time.strftime(get_config()['AUDIO_INPUT_FORMAT']) ) def round_timefile(exact): ''' This will round the datetime, so to match the file organization structure ''' return datetime(exact.year, exact.month, exact.day, exact.hour) def get_timefile(exact): return get_timefile_exact(round_timefile(exact)) def get_files_and_intervals(start, end, rounder=round_timefile): ''' both arguments are datetime objects returns an iterator whose elements are (filename, start_cut, end_cut) Cuts are expressed in seconds ''' if end <= start: raise ValueError("end < start!") while start <= end: begin = rounder(start) start_cut = (start - begin).total_seconds() if end < begin + timedelta(seconds=3599): end_cut = (begin + timedelta(seconds=3599) - end).total_seconds() else: end_cut = 0 yield (begin, start_cut, end_cut) start = begin + timedelta(hours=1) def mp3_join(named_intervals): ''' Note that these are NOT the intervals returned by get_files_and_intervals, as they do not supply a filename, but only a datetime. What we want in input is basically the same thing, but with get_timefile() applied on the first element This function make the (quite usual) assumption that the only start_cut (if any) is at the first file, and the last one is at the last file ''' ffmpeg = get_config()['FFMPEG_PATH'] startskip = None endskip = None files = [] for (filename, start_cut, end_cut) in named_intervals: # this happens only one time, and only at the first iteration if start_cut: assert startskip is None startskip = start_cut # this happens only one time, and only at the first iteration if end_cut: assert endskip is None endskip = end_cut assert '|' not in filename files.append(filename) cmdline = [ffmpeg, '-i', 'concat:%s' % '|'.join(files), '-acodec', 'copy'] if startskip is not None: cmdline += ['-ss', str(startskip)] else: startskip = 0 if endskip is not None: cmdline += ['-t', str(len(files)*3600 - (startskip + endskip))] return cmdline def create_mp3(start, end, outfile, options={}, **kwargs): intervals = [(get_timefile(begin), start_cut, end_cut) for begin, start_cut, end_cut in get_files_and_intervals(start, end)] if os.path.exists(outfile): raise OSError("file '%s' already exists" % outfile) for path, _s, _e in intervals: if not os.path.exists(path): raise OSError("file '%s' does not exist; recording system broken?" % path) # metadata date/time formatted according to # https://wiki.xiph.org/VorbisComment#Date_and_time metadata = {} if outfile.endswith('.mp3'): metadata['TRDC'] = start.replace(microsecond=0).isoformat() metadata['RECORDINGTIME'] = metadata['TRDC'] metadata['ENCODINGTIME'] = datetime.now().replace( microsecond=0).isoformat() else: metadata['DATE'] = start.replace(microsecond=0).isoformat() metadata['ENCODER'] = 'https://github.com/boyska/techrec' if 'title' in options: metadata['TITLE'] = options['title'] if options.get('license_uri', None) is not None: metadata['RIGHTS-DATE'] = start.strftime('%Y-%m') metadata['RIGHTS-URI'] = options['license_uri'] if 'extra_tags' in options: metadata.update(options['extra_tags']) metadata_list = [] for tag, value in metadata.items(): if '=' in tag: logging.error('Received a tag with "=" inside, skipping') continue metadata_list.append('-metadata') metadata_list.append('%s=%s' % (tag, value)) p = Popen(mp3_join(intervals) + metadata_list + get_config()['FFMPEG_OPTIONS'] + [outfile]) if get_config()['FORGE_TIMEOUT'] == 0: p.wait() else: start = datetime.now() while (datetime.now() - start).total_seconds() \ < get_config()['FORGE_TIMEOUT']: p.poll() if p.returncode is None: sleep(1) else: break if p.returncode is None: os.kill(p.pid, 15) try: os.remove(outfile) except: pass raise Exception('timeout') # TODO: make a specific TimeoutError if p.returncode != 0: raise OSError("return code was %d" % p.returncode) return True def main_cmd(options): log = logging.getLogger('forge_main') outfile = os.path.abspath(os.path.join(options.cwd, options.outfile)) log.debug('will forge an mp3 into %s' % (outfile)) create_mp3(options.starttime, options.endtime, outfile)