diff --git a/disco/voice/opus.py b/disco/voice/opus.py index b40f6b5..047b05a 100644 --- a/disco/voice/opus.py +++ b/disco/voice/opus.py @@ -86,17 +86,12 @@ class OpusEncoder(BaseOpus): 'opus_encoder_destroy': ([EncoderStructPtr], None), } - def __init__(self, sampling, channels, application=Application.AUDIO, library_path=None): + def __init__(self, sampling_rate, channels, application=Application.AUDIO, library_path=None): super(OpusEncoder, self).__init__(library_path) - self.sampling_rate = sampling + self.sampling_rate = sampling_rate self.channels = channels self.application = application - self.frame_length = 20 - self.sample_size = 2 * self.channels - self.samples_per_frame = int(self.sampling_rate / 1000 * self.frame_length) - self.frame_size = self.samples_per_frame * self.sample_size - self._inst = None @property diff --git a/disco/voice/playable.py b/disco/voice/playable.py index 263d838..0f438cf 100644 --- a/disco/voice/playable.py +++ b/disco/voice/playable.py @@ -1,3 +1,5 @@ +import abc +import six import gevent import struct import subprocess @@ -17,65 +19,111 @@ except: OPUS_HEADER_SIZE = struct.calcsize(' OpusPlayable +# FFMpegInput.youtube_dl('youtube.com/yolo').pipe(DCADOpusEncoder) => OpusPlayable +# FFMpegInput.youtube_dl('youtube.com/yolo').pipe(OpusEncoder).pipe(DuplexStream, open('cache_file.opus', 'w')) => OpusPlayable + + +class AbstractOpus(object): + def __init__(self, sampling_rate=48000, frame_length=20, channels=2): + self.sampling_rate = sampling_rate + self.frame_length = frame_length + self.channels = 2 + self.sample_size = 2 * self.channels + self.samples_per_frame = int(self.sampling_rate / 1000 * self.frame_length) + self.frame_size = self.samples_per_frame * self.sample_size + + +@six.add_metaclass(abc.ABCMeta) class BasePlayable(object): - pass + @abc.abstractmethod + def next_frame(self): + raise NotImplementedError + + +@six.add_metaclass(abc.ABCMeta) +class BaseInput(object): + @abc.abstractmethod + def read(self, size): + raise NotImplementedError + @abc.abstractmethod + def fileobj(self): + raise NotImplementedError + def pipe(self, other, *args, **kwargs): + return other(self, *args, **kwargs) -class FFmpegPlayable(BasePlayable): - def __init__(self, source='-', command='avconv', sampling_rate=48000, channels=2, **kwargs): + +class OpusFilePlayable(BasePlayable): + """ + An input which reads opus data from a file or file-like object. + """ + def __init__(self, fobj): + self.fobj = fobj + self.done = False + + def next_frame(self): + if self.done: + return None + + header = self.fobj.read(OPUS_HEADER_SIZE) + if len(header) < OPUS_HEADER_SIZE: + self.done = True + return None + + data_size = struct.unpack('0]/bestaudio/best'}) info = ydl.extract_info(url, download=False) - entries = [info] if 'entries' not in info else info['entries'] - for entry in entries: - playable = _cls.create(entry['url'], *args, **kwargs) - playable.info = entry - yield playable + if 'entries' in info: + info = info['entries'][0] - @property - def stdout(self): - return self.proc.stdout + result = cls(source=info['url'], *args, **kwargs) + result.info = info + return result def read(self, sz): if self.streaming: - return self.proc.stdout.read(sz) - else: - if not self._buffer: - data, _ = self.proc.communicate() - self._buffer = StringIO(data) - return self._buffer.read(sz) - - def pipe(self, other, streaming=True): - if issubclass(other, OpusEncoder): - self._child = other(self, self.sampling_rate, self.channels, **self.kwargs) - else: - raise TypeError('Invalid pipe target') + raise TypeError('Cannot read from a streaming FFmpegInput') - @property - def samples_per_frame(self): - return self._child.samples_per_frame + # First read blocks until the subprocess finishes + if not self._buffer: + data, _ = self.proc.communicate() + self._buffer = StringIO(data) + + # Subsequent reads can just do dis thang + return self._buffer.read(sz) + + def fileobj(self): + if self.streaming: + return self.proc.stdout + else: + return self @property def proc(self): @@ -92,50 +140,12 @@ class FFmpegPlayable(BasePlayable): self._proc = subprocess.Popen(args, stdin=None, stdout=subprocess.PIPE) return self._proc - def have_frame(self): - return self._child and self._child.have_frame() - - def next_frame(self): - return self._child.next_frame() - - -class OpusFilePlayable(BasePlayable): - """ - A Playable which supports reading from an on-disk opus-format file. This is - useful in combination with other playables and the OpusFileOutputDuplex. - """ - - def __init__(self, obj, sampling_rate=48000, frame_length=20, channels=2): - super(OpusFilePlayable, self).__init__() - self.obj = obj - self.sampling_rate = sampling_rate - self.frame_length = frame_length - self.channels = channels - self.samples_per_frame = int(self.sampling_rate / 1000 * self.frame_length) - - def have_frame(self): - return self.obj - - def next_frame(self): - header = self.obj.read(OPUS_HEADER_SIZE) - if len(header) < OPUS_HEADER_SIZE: - self.obj = None - return - - size = struct.unpack(' 0: + self._proc.stdin.write(data) + if data < 2048: + break + + if source == subprocess.PIPE: + gevent.spawn(writer) return self._proc - def have_frame(self): - return bool(self.proc) - def next_frame(self): + if self._done: + return None + header = self.proc.stdout.read(OPUS_HEADER_SIZE) if len(header) < OPUS_HEADER_SIZE: - self._proc = None + self._done = True return size = struct.unpack('') def on_play(self, event, url): - item = list(create_youtube_dl_playables(url))[0] + item = FFmpegInput.youtube_dl(url).pipe(DCADOpusEncoderPlayable) self.get_player(event.guild.id).queue.put(item) @Plugin.command('pause')