|
@ -47,7 +47,7 @@ class Shard: |
|
|
self.loop = self._client.loop |
|
|
self.loop = self._client.loop |
|
|
self._current = self.loop.create_future() |
|
|
self._current = self.loop.create_future() |
|
|
self._current.set_result(None) # we just need an already done future |
|
|
self._current.set_result(None) # we just need an already done future |
|
|
self._pending = asyncio.Event(loop=self.loop) |
|
|
self._pending = asyncio.Event() |
|
|
self._pending_task = None |
|
|
self._pending_task = None |
|
|
|
|
|
|
|
|
@property |
|
|
@property |
|
@ -81,7 +81,7 @@ class Shard: |
|
|
coro = DiscordWebSocket.from_client(self._client, resume=True, shard_id=self.id, |
|
|
coro = DiscordWebSocket.from_client(self._client, resume=True, shard_id=self.id, |
|
|
session=self.ws.session_id, sequence=self.ws.sequence) |
|
|
session=self.ws.session_id, sequence=self.ws.sequence) |
|
|
self._dispatch('disconnect') |
|
|
self._dispatch('disconnect') |
|
|
self.ws = await asyncio.wait_for(coro, timeout=180.0, loop=self.loop) |
|
|
self.ws = await asyncio.wait_for(coro, timeout=180.0) |
|
|
|
|
|
|
|
|
def get_future(self): |
|
|
def get_future(self): |
|
|
if self._current.done(): |
|
|
if self._current.done(): |
|
@ -213,10 +213,10 @@ class AutoShardedClient(Client): |
|
|
async def launch_shard(self, gateway, shard_id): |
|
|
async def launch_shard(self, gateway, shard_id): |
|
|
try: |
|
|
try: |
|
|
coro = websockets.connect(gateway, loop=self.loop, klass=DiscordWebSocket, compression=None) |
|
|
coro = websockets.connect(gateway, loop=self.loop, klass=DiscordWebSocket, compression=None) |
|
|
ws = await asyncio.wait_for(coro, loop=self.loop, timeout=180.0) |
|
|
ws = await asyncio.wait_for(coro, timeout=180.0) |
|
|
except Exception: |
|
|
except Exception: |
|
|
log.info('Failed to connect for shard_id: %s. Retrying...', shard_id) |
|
|
log.info('Failed to connect for shard_id: %s. Retrying...', shard_id) |
|
|
await asyncio.sleep(5.0, loop=self.loop) |
|
|
await asyncio.sleep(5.0) |
|
|
return await self.launch_shard(gateway, shard_id) |
|
|
return await self.launch_shard(gateway, shard_id) |
|
|
|
|
|
|
|
|
ws.token = self.http.token |
|
|
ws.token = self.http.token |
|
@ -230,17 +230,17 @@ class AutoShardedClient(Client): |
|
|
|
|
|
|
|
|
try: |
|
|
try: |
|
|
# OP HELLO |
|
|
# OP HELLO |
|
|
await asyncio.wait_for(ws.poll_event(), loop=self.loop, timeout=180.0) |
|
|
await asyncio.wait_for(ws.poll_event(), timeout=180.0) |
|
|
await asyncio.wait_for(ws.identify(), loop=self.loop, timeout=180.0) |
|
|
await asyncio.wait_for(ws.identify(), timeout=180.0) |
|
|
except asyncio.TimeoutError: |
|
|
except asyncio.TimeoutError: |
|
|
log.info('Timed out when connecting for shard_id: %s. Retrying...', shard_id) |
|
|
log.info('Timed out when connecting for shard_id: %s. Retrying...', shard_id) |
|
|
await asyncio.sleep(5.0, loop=self.loop) |
|
|
await asyncio.sleep(5.0) |
|
|
return await self.launch_shard(gateway, shard_id) |
|
|
return await self.launch_shard(gateway, shard_id) |
|
|
|
|
|
|
|
|
# keep reading the shard while others connect |
|
|
# keep reading the shard while others connect |
|
|
self.shards[shard_id] = ret = Shard(ws, self) |
|
|
self.shards[shard_id] = ret = Shard(ws, self) |
|
|
ret.launch_pending_reads() |
|
|
ret.launch_pending_reads() |
|
|
await asyncio.sleep(5.0, loop=self.loop) |
|
|
await asyncio.sleep(5.0) |
|
|
|
|
|
|
|
|
async def launch_shards(self): |
|
|
async def launch_shards(self): |
|
|
if self.shard_count is None: |
|
|
if self.shard_count is None: |
|
@ -261,14 +261,14 @@ class AutoShardedClient(Client): |
|
|
shards_to_wait_for.append(shard.wait()) |
|
|
shards_to_wait_for.append(shard.wait()) |
|
|
|
|
|
|
|
|
# wait for all pending tasks to finish |
|
|
# wait for all pending tasks to finish |
|
|
await utils.sane_wait_for(shards_to_wait_for, timeout=300.0, loop=self.loop) |
|
|
await utils.sane_wait_for(shards_to_wait_for, timeout=300.0) |
|
|
|
|
|
|
|
|
async def _connect(self): |
|
|
async def _connect(self): |
|
|
await self.launch_shards() |
|
|
await self.launch_shards() |
|
|
|
|
|
|
|
|
while True: |
|
|
while True: |
|
|
pollers = [shard.get_future() for shard in self.shards.values()] |
|
|
pollers = [shard.get_future() for shard in self.shards.values()] |
|
|
done, _ = await asyncio.wait(pollers, loop=self.loop, return_when=asyncio.FIRST_COMPLETED) |
|
|
done, _ = await asyncio.wait(pollers, return_when=asyncio.FIRST_COMPLETED) |
|
|
for f in done: |
|
|
for f in done: |
|
|
# we wanna re-raise to the main Client.connect handler if applicable |
|
|
# we wanna re-raise to the main Client.connect handler if applicable |
|
|
f.result() |
|
|
f.result() |
|
@ -291,7 +291,7 @@ class AutoShardedClient(Client): |
|
|
|
|
|
|
|
|
to_close = [shard.ws.close() for shard in self.shards.values()] |
|
|
to_close = [shard.ws.close() for shard in self.shards.values()] |
|
|
if to_close: |
|
|
if to_close: |
|
|
await asyncio.wait(to_close, loop=self.loop) |
|
|
await asyncio.wait(to_close) |
|
|
|
|
|
|
|
|
await self.http.close() |
|
|
await self.http.close() |
|
|
|
|
|
|
|
|