Handle IncompleteReadError and InvalidChecksumError

This commit is contained in:
Lonami Exo 2018-10-03 14:46:10 +02:00
parent 3fd7c33127
commit e319fa3aa9
2 changed files with 38 additions and 14 deletions

View File

@ -1,5 +1,9 @@
import abc
import asyncio
import logging
__log__ = logging.getLogger(__name__)
class Connection(abc.ABC):
@ -74,36 +78,44 @@ class Connection(abc.ABC):
"""
return self._send_queue.put(data)
def recv(self):
async def recv(self):
"""
Receives a packet of data through this connection mode.
This method returns a coroutine.
"""
return self._recv_queue.get()
ok, result = await self._recv_queue.get()
if ok:
return result
else:
raise result from None
# TODO Get/put to the queue with cancellation
async def _send_loop(self):
"""
This loop is constantly popping items off the queue to send them.
"""
try:
while not self._disconnected.is_set():
self._send(await self._send_queue.get())
await self._writer.drain()
except asyncio.CancelledError:
pass
except Exception:
logging.exception('Unhandled exception in the sending loop')
self.disconnect()
# TODO Handle IncompleteReadError and InvalidChecksumError
async def _recv_loop(self):
"""
This loop is constantly putting items on the queue as they're read.
"""
while not self._disconnected.is_set():
try:
while not self._disconnected.is_set():
data = await self._recv()
except asyncio.IncompleteReadError:
if not self._disconnected.is_set():
raise
else:
await self._recv_queue.put(data)
await self._recv_queue.put((True, data))
except Exception as e:
await self._recv_queue.put((False, e))
self.disconnect()
@abc.abstractmethod
def _send(self, data):

View File

@ -9,7 +9,7 @@ from .requeststate import RequestState
from .. import utils
from ..errors import (
BadMessageError, BrokenAuthKeyError, SecurityError, TypeNotFoundError,
rpc_message_to_error
InvalidChecksumError, rpc_message_to_error
)
from ..extensions import BinaryReader
from ..helpers import _ReadyQueue
@ -376,6 +376,11 @@ class MTProtoSender:
__log__.warning('Security error while unpacking a '
'received message: %s', e)
continue
except InvalidChecksumError as e:
__log__.warning(
'Invalid checksum on the read packet (was %s expected %s)',
e.checksum, e.valid_checksum
)
except asyncio.CancelledError:
return
except (BrokenAuthKeyError, BufferError):
@ -383,6 +388,13 @@ class MTProtoSender:
self._connection._state.auth_key = None
self._start_reconnect()
return
except asyncio.IncompleteReadError:
# TODO Handle packets that are too big and trigger this
# If it's not a packet that triggered this, just reconnect
__log__.info('Telegram closed the connection')
self._pending_state.clear()
self._start_reconnect()
return
except Exception:
__log__.exception('Unhandled error while receiving data')
self._start_reconnect()