@ -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,8 +575,10 @@ 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 ) :
_log . info ( ' Disconnecting from voice normally, close code %d . ' , exc . code )
await self . disconnect ( )
# 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
if exc . code == 4014 :
@ -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 )