mirror of
				https://github.com/Rapptz/discord.py.git
				synced 2025-10-25 02:23:04 +00:00 
			
		
		
		
	Fix voice disconnect+connect race condition
Fixes a race condition when disconnecting and immediately connecting again. Also fixes disconnect() being called twice. Let me be clear, I DO NOT LIKE THIS SOLUTION. I think it's dumb but I don't see any other reasonable alternative. There isn't a way to transfer state to a new connection state object and I can't think of a nice way to do it either. That said, waiting an arbitrary amount of time for an arbitrary websocket event doesn't seem like the right solution either, but it's the best I can do at this point.
This commit is contained in:
		| @@ -212,6 +212,7 @@ class VoiceConnectionState: | ||||
|         self._expecting_disconnect: bool = False | ||||
|         self._connected = threading.Event() | ||||
|         self._state_event = asyncio.Event() | ||||
|         self._disconnected = asyncio.Event() | ||||
|         self._runner: Optional[asyncio.Task] = None | ||||
|         self._connector: Optional[asyncio.Task] = None | ||||
|         self._socket_reader = SocketReader(self) | ||||
| @@ -254,8 +255,10 @@ class VoiceConnectionState: | ||||
|         channel_id = data['channel_id'] | ||||
|  | ||||
|         if channel_id is None: | ||||
|             self._disconnected.set() | ||||
|  | ||||
|             # If we know we're going to get a voice_state_update where we have no channel due to | ||||
|             # being in the reconnect flow, we ignore it.  Otherwise, it probably wasn't from us. | ||||
|             # being in the reconnect or disconnect flow, we ignore it.  Otherwise, it probably wasn't from us. | ||||
|             if self._expecting_disconnect: | ||||
|                 self._expecting_disconnect = False | ||||
|             else: | ||||
| @@ -419,9 +422,9 @@ class VoiceConnectionState: | ||||
|             return | ||||
|  | ||||
|         try: | ||||
|             await self._voice_disconnect() | ||||
|             if self.ws: | ||||
|                 await self.ws.close() | ||||
|             await self._voice_disconnect() | ||||
|         except Exception: | ||||
|             _log.debug('Ignoring exception disconnecting from voice', exc_info=True) | ||||
|         finally: | ||||
| @@ -436,11 +439,25 @@ class VoiceConnectionState: | ||||
|  | ||||
|             if cleanup: | ||||
|                 self._socket_reader.stop() | ||||
|                 self.voice_client.cleanup() | ||||
|  | ||||
|             if self.socket: | ||||
|                 self.socket.close() | ||||
|  | ||||
|             # Skip this part if disconnect was called from the poll loop task | ||||
|             if self._runner and asyncio.current_task() != self._runner: | ||||
|                 # Wait for the voice_state_update event confirming the bot left the voice channel. | ||||
|                 # This prevents a race condition caused by disconnecting and immediately connecting again. | ||||
|                 # The new VoiceConnectionState object receives the voice_state_update event containing channel=None while still | ||||
|                 # connecting leaving it in a bad state.  Since there's no nice way to transfer state to the new one, we have to do this. | ||||
|                 try: | ||||
|                     async with atimeout(self.timeout): | ||||
|                         await self._disconnected.wait() | ||||
|                 except TimeoutError: | ||||
|                     _log.debug('Timed out waiting for disconnect confirmation event') | ||||
|  | ||||
|             if cleanup: | ||||
|                 self.voice_client.cleanup() | ||||
|  | ||||
|     async def soft_disconnect(self, *, with_state: ConnectionFlowState = ConnectionFlowState.got_both_voice_updates) -> None: | ||||
|         _log.debug('Soft disconnecting from voice') | ||||
|         # Stop the websocket reader because closing the websocket will trigger an unwanted reconnect | ||||
| @@ -524,6 +541,7 @@ class VoiceConnectionState: | ||||
|         self.state = ConnectionFlowState.disconnected | ||||
|         await self.voice_client.channel.guild.change_voice_state(channel=None) | ||||
|         self._expecting_disconnect = True | ||||
|         self._disconnected.clear() | ||||
|  | ||||
|     async def _connect_websocket(self, resume: bool) -> DiscordVoiceWebSocket: | ||||
|         ws = await DiscordVoiceWebSocket.from_connection_state(self, resume=resume, hook=self.hook) | ||||
| @@ -557,6 +575,8 @@ class VoiceConnectionState: | ||||
|                     # 4014 - we were externally disconnected (voice channel deleted, we were moved, etc) | ||||
|                     # 4015 - voice server has crashed | ||||
|                     if exc.code in (1000, 4015): | ||||
|                         # Don't call disconnect a second time if the websocket closed from a disconnect call | ||||
|                         if not self._expecting_disconnect: | ||||
|                             _log.info('Disconnecting from voice normally, close code %d.', exc.code) | ||||
|                             await self.disconnect() | ||||
|                         break | ||||
| @@ -602,6 +622,8 @@ class VoiceConnectionState: | ||||
|             ) | ||||
|         except asyncio.TimeoutError: | ||||
|             return False | ||||
|  | ||||
|         previous_ws = self.ws | ||||
|         try: | ||||
|             self.ws = await self._connect_websocket(False) | ||||
|             await self._handshake_websocket() | ||||
| @@ -609,6 +631,8 @@ class VoiceConnectionState: | ||||
|             return False | ||||
|         else: | ||||
|             return True | ||||
|         finally: | ||||
|             await previous_ws.close() | ||||
|  | ||||
|     async def _move_to(self, channel: abc.Snowflake) -> None: | ||||
|         await self.voice_client.channel.guild.change_voice_state(channel=channel) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user