diff --git a/disco/bot/bot.py b/disco/bot/bot.py index 7693f50..8ad0fca 100644 --- a/disco/bot/bot.py +++ b/disco/bot/bot.py @@ -355,10 +355,10 @@ class Bot(LoggingClass): if event.message.author.id == self.client.state.me.id: return - if self.config.commands_allow_edit: - self.last_message_cache[event.message.channel_id] = (event.message, False) + result = self.handle_message(event.message) - self.handle_message(event.message) + if self.config.commands_allow_edit: + self.last_message_cache[event.message.channel_id] = (event.message, result) def on_message_update(self, event): if self.config.commands_allow_edit: diff --git a/disco/voice/client.py b/disco/voice/client.py index dd50cbb..d0d3e09 100644 --- a/disco/voice/client.py +++ b/disco/voice/client.py @@ -3,6 +3,8 @@ import socket import struct import time +import nacl.secret + from holster.enum import Enum from holster.emitter import Emitter @@ -40,21 +42,32 @@ class UDPVoiceClient(LoggingClass): # Connection information self.ip = None self.port = None - self.sequence = 0 - self.timestamp = 0 self.run_task = None self.connected = False def send_frame(self, frame, sequence=None, timestamp=None): - data = bytearray(12) - data[0] = 0x80 - data[1] = 0x78 - struct.pack_into('>H', data, 2, sequence or self.sequence) - struct.pack_into('>I', data, 4, timestamp or self.timestamp) - struct.pack_into('>i', data, 8, self.vc.ssrc) - self.send(data + ''.join(frame)) - self.sequence += 1 + # Convert the frame to a bytearray + frame = bytearray(frame) + + # First, pack the header (TODO: reuse bytearray?) + header = bytearray(24) + header[0] = 0x80 + header[1] = 0x78 + struct.pack_into('>H', header, 2, sequence or self.vc.sequence) + struct.pack_into('>I', header, 4, timestamp or self.vc.timestamp) + struct.pack_into('>i', header, 8, self.vc.ssrc) + + # Now encrypt the payload with the nonce as a header + raw = self.vc.secret_box.encrypt(bytes(frame), bytes(header)).ciphertext + + # Send the header (sans nonce padding) plus the payload + self.send(header[:12] + raw) + + # Increment our sequence counter + self.vc.sequence += 1 + if self.vc.sequence >= 65535: + self.vc.sequence = 0 def run(self): while True: @@ -101,24 +114,34 @@ class VoiceClient(LoggingClass): def __init__(self, channel, encoder=None): super(VoiceClient, self).__init__() - assert channel.is_voice, 'Cannot spawn a VoiceClient for a non-voice channel' + if not channel.is_voice: + raise ValueError('Cannot spawn a VoiceClient for a non-voice channel') + self.channel = channel self.client = self.channel.client self.encoder = encoder or JSONEncoder + # Bind to some WS packets self.packets = Emitter(gevent.spawn) self.packets.on(VoiceOPCode.READY, self.on_voice_ready) self.packets.on(VoiceOPCode.SESSION_DESCRIPTION, self.on_voice_sdp) - # State + # State + state change emitter self.state = VoiceState.DISCONNECTED self.state_emitter = Emitter(gevent.spawn) + + # Connection metadata self.token = None self.endpoint = None self.ssrc = None self.port = None + self.secret_box = None self.udp = None + # Voice data state + self.sequence = 0 + self.timestamp = 0 + self.update_listener = None # Websocket connection @@ -166,11 +189,14 @@ class VoiceClient(LoggingClass): 'data': { 'port': port, 'address': ip, - 'mode': 'plain' + 'mode': 'xsalsa20_poly1305' } }) - def on_voice_sdp(self, _): + def on_voice_sdp(self, sdp): + # Create a secret box for encryption/decryption + self.secret_box = nacl.secret.SecretBox(bytes(bytearray(sdp['secret_key']))) + # Toggle speaking state so clients learn of our SSRC self.set_speaking(True) self.set_speaking(False) @@ -204,7 +230,7 @@ class VoiceClient(LoggingClass): self.log.exception('Failed to parse voice gateway message: ') def on_error(self, err): - # TODO + # TODO: raise an exception here self.log.warning('Voice websocket error: {}'.format(err)) def on_open(self): diff --git a/disco/voice/opus.py b/disco/voice/opus.py index 68b8d97..0313415 100644 --- a/disco/voice/opus.py +++ b/disco/voice/opus.py @@ -137,9 +137,9 @@ class OpusEncoder(BaseOpus): return result def __del__(self): - if self.inst: - self.opus_encoder_destroy(self.inst) - self.inst = None + if self._inst: + self.opus_encoder_destroy(self._inst) + self._inst = None def encode(self, pcm, frame_size): max_data_bytes = len(pcm) @@ -159,24 +159,24 @@ class OpusDecoder(BaseOpus): class BufferedOpusEncoder(OpusEncoder): - def __init__(self, f, *args, **kwargs): - self.data = f + def __init__(self, source, *args, **kwargs): + self.source = source self.frames = Queue(kwargs.pop('queue_size', 4096)) super(BufferedOpusEncoder, self).__init__(*args, **kwargs) gevent.spawn(self._encoder_loop) def _encoder_loop(self): - while self.data: - raw = self.data.read(self.frame_size) + while self.source: + raw = self.source.read(self.frame_size) if len(raw) < self.frame_size: break self.frames.put(self.encode(raw, self.samples_per_frame)) gevent.idle() - self.data = None + self.source = None def have_frame(self): - return self.data or not self.frames.empty() + return self.source or not self.frames.empty() def next_frame(self): return self.frames.get() @@ -185,10 +185,10 @@ class BufferedOpusEncoder(OpusEncoder): class GIPCBufferedOpusEncoder(OpusEncoder): FIN = 1 - def __init__(self, f, *args, **kwargs): + def __init__(self, source, *args, **kwargs): import gipc - self.data = f + self.source = source self.parent_pipe, self.child_pipe = gipc.pipe(duplex=True) self.frames = Queue(kwargs.pop('queue_size', 4096)) super(GIPCBufferedOpusEncoder, self).__init__(*args, **kwargs) @@ -209,7 +209,7 @@ class GIPCBufferedOpusEncoder(OpusEncoder): def _writer(self): while self.data: - raw = self.data.read(self.frame_size) + raw = self.source.read(self.frame_size) if len(raw) < self.frame_size: break @@ -238,36 +238,48 @@ class GIPCBufferedOpusEncoder(OpusEncoder): class DCADOpusEncoder(OpusEncoder): - def __init__(self, pipe, *args, **kwargs): - command = kwargs.pop('command', 'dcad') + def __init__(self, source, *args, **kwargs): + self.source = source + self.command = kwargs.pop('command', 'dcad') super(DCADOpusEncoder, self).__init__(*args, **kwargs) - self.proc = subprocess.Popen([ - command, - # '--channels', str(self.channels), - # '--rate', str(self.sampling_rate), - # '--size', str(self.frame_length), - '--bitrate', '128', - '--fec', - '--packet-loss-percent', '30', - '--input', 'pipe:0', - '--output', 'pipe:1', - ], stdin=pipe, stdout=subprocess.PIPE) + self._proc = None self.header_size = struct.calcsize('0]/bestaudio/best'}) info = ydl.extract_info(url, download=False) + entries = [info] if 'entries' not in info else info['entries'] - if 'entries' in info: - info = info['entries'][0] - - playable = create_ffmpeg_playable(info['url'], *args, **kwargs) - playable.info = info - return playable + for entry in entries: + playable = create_ffmpeg_playable(entry['url'], *args, **kwargs) + playable.info = entry + yield playable class OpusPlayable(object): @@ -199,7 +202,11 @@ class Player(object): if not item.have_frame(): return - self.client.send_frame(item.next_frame()) + frame = item.next_frame() + if frame is None: + return + + self.client.send_frame(frame) self.client.timestamp += item.samples_per_frame next_time = start + 0.02 * loops diff --git a/examples/music.py b/examples/music.py index 6035515..9901f6d 100644 --- a/examples/music.py +++ b/examples/music.py @@ -1,11 +1,11 @@ from disco.bot import Plugin from disco.bot.command import CommandError -from disco.voice.player import Player, create_youtube_dl_playable +from disco.voice.player import Player, create_youtube_dl_playables from disco.voice.client import VoiceException def download(url): - return create_youtube_dl_playable(url) + return create_youtube_dl_playables(url) class MusicPlugin(Plugin): @@ -43,7 +43,8 @@ class MusicPlugin(Plugin): @Plugin.command('play', '') def on_play(self, event, url): - self.get_player(event.guild.id).queue.put(download(url)) + item = list(create_youtube_dl_playables(url))[0] + self.get_player(event.guild.id).queue.put(item) @Plugin.command('pause') def on_pause(self, event): diff --git a/requirements.txt b/requirements.txt index 88b7b59..ed9a25c 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,6 +1,7 @@ -gevent==1.1.2 +gevent==1.2.1 holster==1.0.11 inflection==0.3.1 -requests==2.11.1 +requests==2.13.0 six==1.10.0 websocket-client==0.40.0 +pynacl==1.1.2