diff --git a/.coveragerc b/.coveragerc
new file mode 100644
index 00000000..1b2271b9
--- /dev/null
+++ b/.coveragerc
@@ -0,0 +1,8 @@
+[run]
+branch = true
+parallel = true
+source =
+ telethon
+
+[report]
+precision = 2
diff --git a/.github/workflows/python.yml b/.github/workflows/python.yml
new file mode 100644
index 00000000..067067a6
--- /dev/null
+++ b/.github/workflows/python.yml
@@ -0,0 +1,28 @@
+name: Python Library
+
+on: [push]
+
+jobs:
+ build:
+
+ runs-on: ubuntu-latest
+ strategy:
+ matrix:
+ python-version: ["3.5", "3.6", "3.7", "3.8"]
+ steps:
+ - uses: actions/checkout@v1
+ - name: Set up Python ${{ matrix.python-version }}
+ uses: actions/setup-python@v1
+ with:
+ python-version: ${{ matrix.python-version }}
+ - name: Set up env
+ run: |
+ python -m pip install --upgrade pip
+ pip install tox
+ - name: Lint with flake8
+ run: |
+ tox -e flake
+ - name: Test with pytest
+ run: |
+ # use "py", which is the default python version
+ tox -e py
diff --git a/dev-requirements.txt b/dev-requirements.txt
new file mode 100644
index 00000000..e01a8206
--- /dev/null
+++ b/dev-requirements.txt
@@ -0,0 +1,3 @@
+pytest
+pytest-cov
+pytest-asyncio
diff --git a/optional-requirements.txt b/optional-requirements.txt
index aeaf3994..00cd3324 100644
--- a/optional-requirements.txt
+++ b/optional-requirements.txt
@@ -1,4 +1,4 @@
cryptg
pysocks
-hachoir3
+hachoir
pillow
diff --git a/readthedocs/developing/testing.rst b/readthedocs/developing/testing.rst
new file mode 100644
index 00000000..badb7dc6
--- /dev/null
+++ b/readthedocs/developing/testing.rst
@@ -0,0 +1,87 @@
+=====
+Tests
+=====
+
+Telethon uses `Pytest `__, for testing, `Tox
+`__ for environment setup, and
+`pytest-asyncio `__ and `pytest-cov
+`__ for asyncio and
+`coverage `__ integration.
+
+While reading the full documentation for these is probably a good idea, there
+is a lot to read, so a brief summary of these tools is provided below for
+convienience.
+
+Brief Introduction to Pytest
+============================
+
+`Pytest `__ is a tool for discovering and running python
+tests, as well as allowing modular reuse of test setup code using fixtures.
+
+Most Pytest tests will look something like this::
+
+ from module import my_thing, my_other_thing
+
+ def test_my_thing(fixture):
+ assert my_thing(fixture) == 42
+
+ @pytest.mark.asyncio
+ async def test_my_thing(event_loop):
+ assert await my_other_thing(loop=event_loop) == 42
+
+Note here:
+
+ 1. The test imports one specific function. The role of unit tests is to test
+ that the implementation of some unit, like a function or class, works.
+ It's role is not so much to test that components interact well with each
+ other. I/O, such as connecting to remote servers, should be avoided. This
+ helps with quickly identifying the source of an error, finding silent
+ breakage, and makes it easier to cover all possible code paths.
+
+ System or integration tests can also be useful, but are currently out of
+ scope of Telethon's automated testing.
+
+ 2. A function ``test_my_thing`` is declared. Pytest searches for files
+ starting with ``test_``, classes starting with ``Test`` and executes any
+ functions or methods starting with ``test_`` it finds.
+
+ 3. The function is declared with a parameter ``fixture``. Fixtures are used to
+ request things required to run the test, such as temporary directories,
+ free TCP ports, Connections, etc. Fixtures are declared by simply adding
+ the fixture name as parameter. A full list of available fixtures can be
+ found with the ``pytest --fixtures`` command.
+
+ 4. The test uses a simple ``assert`` to test some condition is valid. Pytest
+ uses some magic to ensure that the errors from this are readable and easy
+ to debug.
+
+ 5. The ``pytest.mark.asyncio`` fixture is provided by ``pytest-asyncio``. It
+ starts a loop and executes a test function as coroutine. This should be
+ used for testing asyncio code. It also declares the ``event_loop``
+ fixture, which will request an ``asyncio`` event loop.
+
+Brief Introduction to Tox
+=========================
+
+`Tox `__ is a tool for automated setup
+of virtual environments for testing. While the tests can be run directly by
+just running ``pytest``, this only tests one specific python version in your
+existing environment, which will not catch e.g. undeclared dependencies, or
+version incompatabilities.
+
+Tox environments are declared in the ``tox.ini`` file. The default
+environments, declared at the top, can be simply run with ``tox``. The option
+``tox -e py36,flake`` can be used to request specific environments to be run.
+
+Brief Introduction to Pytest-cov
+================================
+
+Coverage is a useful metric for testing. It measures the lines of code and
+branches that are exercised by the tests. The higher the coverage, the more
+likely it is that any coding errors will be caught by the tests.
+
+A brief coverage report can be generated with the ``--cov`` option to ``tox``,
+which will be passed on to ``pytest``. Additionally, the very useful HTML
+report can be generated with ``--cov --cov-report=html``, which contains a
+browsable copy of the source code, annotated with coverage information for each
+line.
diff --git a/readthedocs/index.rst b/readthedocs/index.rst
index 9f28deda..0776e3f2 100644
--- a/readthedocs/index.rst
+++ b/readthedocs/index.rst
@@ -93,6 +93,7 @@ You can also use the menu on the left to quickly skip over sections.
developing/test-servers.rst
developing/project-structure.rst
developing/coding-style.rst
+ developing/testing.rst
developing/understanding-the-type-language.rst
developing/tips-for-porting-the-project.rst
developing/telegram-api-in-other-languages.rst
diff --git a/telethon/client/telegrambaseclient.py b/telethon/client/telegrambaseclient.py
index b5c9aec0..162835ea 100644
--- a/telethon/client/telegrambaseclient.py
+++ b/telethon/client/telegrambaseclient.py
@@ -662,7 +662,7 @@ class TelegramBaseClient(abc.ABC):
self._exported_sessions[cdn_redirect.dc_id] = session
self._log[__name__].info('Creating new CDN client')
- client = TelegramBareClient(
+ client = TelegramBaseClient(
session, self.api_id, self.api_hash,
proxy=self._sender.connection.conn.proxy,
timeout=self._sender.connection.get_timeout()
diff --git a/tests/__init__.py b/tests/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/tests/telethon/__init__.py b/tests/telethon/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/tests/telethon/crypto/test_rsa.py b/tests/telethon/crypto/test_rsa.py
new file mode 100644
index 00000000..a1a949d2
--- /dev/null
+++ b/tests/telethon/crypto/test_rsa.py
@@ -0,0 +1,37 @@
+"""
+tests for telethon.crypto.rsa
+"""
+import pytest
+
+from telethon.crypto import rsa
+
+@pytest.fixture
+def server_key_fp():
+ """factory to return a key, old if so chosen"""
+ def _server_key_fp(old: bool):
+ for fp, data in rsa._server_keys.items():
+ _, old_key = data
+ if old_key == old:
+ return fp
+
+ return _server_key_fp
+
+def test_encryption_inv_key():
+ """test for #1324"""
+ assert rsa.encrypt("invalid", b"testdata") is None
+
+def test_encryption_old_key(server_key_fp):
+ """test for #1324"""
+ assert rsa.encrypt(server_key_fp(old=True), b"testdata") is None
+
+def test_encryption_allowed_old_key(server_key_fp):
+ data = rsa.encrypt(server_key_fp(old=True), b"testdata", use_old=True)
+ # we can't verify the data is actually valid because we don't have
+ # the decryption keys
+ assert data is not None and len(data) == 256
+
+def test_encryption_current_key(server_key_fp):
+ data = rsa.encrypt(server_key_fp(old=False), b"testdata")
+ # we can't verify the data is actually valid because we don't have
+ # the decryption keys
+ assert data is not None and len(data) == 256
diff --git a/tests/telethon/test_helpers.py b/tests/telethon/test_helpers.py
new file mode 100644
index 00000000..689db8af
--- /dev/null
+++ b/tests/telethon/test_helpers.py
@@ -0,0 +1,59 @@
+"""
+tests for telethon.helpers
+"""
+
+from base64 import b64decode
+
+import pytest
+
+from telethon import helpers
+
+
+def test_strip_text():
+ assert helpers.strip_text(" text ", []) == "text"
+ # I can't interpret the rest of the code well enough yet
+
+
+class TestSyncifyAsyncContext:
+ class NoopContextManager:
+ def __init__(self, loop):
+ self.count = 0
+ self.loop = loop
+
+ async def __aenter__(self):
+ self.count += 1
+ return self
+
+ async def __aexit__(self, exc_type, *args):
+ assert exc_type is None
+ self.count -= 1
+
+ __enter__ = helpers._sync_enter
+ __exit__ = helpers._sync_exit
+
+ def test_sync_acontext(self, event_loop):
+ contm = self.NoopContextManager(event_loop)
+ assert contm.count == 0
+
+ with contm:
+ assert contm.count == 1
+
+ assert contm.count == 0
+
+ @pytest.mark.asyncio
+ async def test_async_acontext(self, event_loop):
+ contm = self.NoopContextManager(event_loop)
+ assert contm.count == 0
+
+ async with contm:
+ assert contm.count == 1
+
+ assert contm.count == 0
+
+
+def test_generate_key_data_from_nonce():
+ gkdfn = helpers.generate_key_data_from_nonce
+
+ key_expect = b64decode(b'NFwRFB8Knw/kAmvPWjtrQauWysHClVfQh0UOAaABqZA=')
+ nonce_expect = b64decode(b'1AgjhU9eDvJRjFik73bjR2zZEATzL/jLu9yodYfWEgA=')
+ assert gkdfn(123456789, 1234567) == (key_expect, nonce_expect)
diff --git a/tox.ini b/tox.ini
new file mode 100644
index 00000000..53f33505
--- /dev/null
+++ b/tox.ini
@@ -0,0 +1,24 @@
+[tox]
+envlist = py35,py36,py37,py38
+
+[testenv]
+deps =
+ -rrequirements.txt
+ -roptional-requirements.txt
+ -rdev-requirements.txt
+commands =
+ # NOTE: you can run any command line tool here - not just tests
+ pytest {posargs}
+
+# run with tox -e flake
+[testenv:flake]
+deps =
+ -rrequirements.txt
+ -roptional-requirements.txt
+ -rdev-requirements.txt
+ flake8
+commands =
+ # stop the build if there are Python syntax errors or undefined names
+ flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics
+ # exit-zero treats all errors as warnings. The GitHub editor is 127 chars wide
+ flake8 . --count --exit-zero --exclude telethon/tl/,telethon/errors/rpcerrorlist.py --max-complexity=10 --max-line-length=127 --statistics