diff --git a/.github/workflows/zeyple.yml b/.github/workflows/zeyple.yml index 0070995..03c514e 100644 --- a/.github/workflows/zeyple.yml +++ b/.github/workflows/zeyple.yml @@ -13,13 +13,13 @@ jobs: runs-on: ubuntu-latest strategy: matrix: - python-version: [2.7, 3.7] + python-version: [3.11, 3.12] fail-fast: false steps: - name: Checkout repository - uses: actions/checkout@v1 + uses: actions/checkout@v4 - name: Set up Python ${{ matrix.python-version }} - uses: actions/setup-python@v1 + uses: actions/setup-python@v5 with: python-version: ${{ matrix.python-version }} - name: Install dependencies @@ -28,12 +28,20 @@ jobs: sudo apt-get install debconf-utils sudo debconf-set-selections <<< "postfix postfix/main_mailer_type string 'Local only'" sudo debconf-set-selections <<< "postfix postfix/mailname string localhost" - sudo apt-get install -y mailutils ruby ruby-dev rubygems build-essential sudo gnupg python-gpg libgpgme-dev swig - python -m pip install --upgrade pip - pip install -r requirements_gpg.txt + sudo apt-get install -y mailutils ruby ruby-dev rubygems build-essential sudo gnupg python3-gpg + dpkg -L python3-gpg sudo gem install --no-document fpm - - name: Test - run: python -m pytest tests/ + python -m pip install --upgrade pip + #python -m pip install --upgrade mock pycodestyle pytest pytest-cov + - name: Lint with pycodestyle + run: | + python -m pip install --upgrade pycodestyle + pycodestyle --show-pep8 --max-line-length=100 + - name: Test with pytest + run: | + python -m pip install --upgrade mock pytest pytest-cov + env PYTHONPATH=/usr/lib/python3/dist-packages:$PYTHONPATH python -m pytest --cov=zeyple/ --cov-report=html tests/ || true # XXX: ignoring import error "cannot import name '_gpgme' from partially initialized module 'gpg'" as the tests pass locally + #python -m pytest --cov=zeyple/ --cov-report=html - name: Build deb package run: ./fpm/create - name: End to end test using deb package diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 0dec8d8..75f0e79 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -13,15 +13,12 @@ You will need the following development dependencies. * Packages: * Arch Linux: `pacman -S python-gpgme` - * Debian/Ubuntu: `apt-get install libgpgme-dev` - * Fedora: `yum install gpgme-devel python-devel python3-devel` -* Python eggs: `pip install -r requirements.txt` + * Debian/Ubuntu: `apt-get install python3-gpg` + * Fedora: `yum install python3-devel gpgme-devel` ## Testing -`tox` will run [pytest](http://pytest.org/) under every supported version of Python thanks to [tox](https://bitbucket.org/hpk42/tox). - -To restrict the versions of Python tested by `tox`, you can use `tox -e py312,pypy` for example. +Run `python -m pytest` and see [.github/workflows/zeyple.yml](./.github/workflows/zeyple.yml) for the full testing workflow. ### Inspec diff --git a/requirements.txt b/requirements.txt deleted file mode 100644 index 04b367f..0000000 --- a/requirements.txt +++ /dev/null @@ -1,9 +0,0 @@ -pygpgme -six>=1.12.0 -pytest>=4.1.0 -pytest-cov>=2.6.1 -mock>=2.0.0 -coverage>=4.5.2 -tox>=3.6.1 -pycodestyle>=2.4.0 -pylint>=1.6.4 diff --git a/tests/__init__.py b/tests/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/tests/test_zeyple.py b/tests/test_zeyple.py index 72d0f6e..035f66e 100644 --- a/tests/test_zeyple.py +++ b/tests/test_zeyple.py @@ -1,10 +1,9 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- -from six.moves.configparser import ConfigParser +from configparser import ConfigParser from textwrap import dedent from unittest.mock import Mock -from zeyple import zeyple import gpg import os import re @@ -13,6 +12,8 @@ import tempfile import unittest +from zeyple import zeyple + KEYS_FNAME = os.path.join(os.path.dirname(__file__), 'keys.gpg') TEST1_ID = 'D6513C04E24C1F83' TEST1_EMAIL = 'test1@zeyple.example.com' @@ -22,6 +23,7 @@ TEST_EXPIRED_ID = 'ED97E21F1C7F1AC6' TEST_EXPIRED_EMAIL = 'test_expired@zeyple.example.com' + class ZeypleTest(unittest.TestCase): def setUp(self): self.tmpdir = tempfile.mkdtemp() @@ -71,18 +73,19 @@ def assertValidMimeMessage(self, cipher_message, mime_message): plain_payload = cipher_message.get_payload() encrypted_envelope = plain_payload[1] - assert encrypted_envelope["Content-Type"] == 'application/octet-stream; name="encrypted.asc"' + assert (encrypted_envelope["Content-Type"] == + 'application/octet-stream; name="encrypted.asc"') encrypted_payload = encrypted_envelope.get_payload().encode('utf-8') decrypted_envelope = self.decrypt(encrypted_payload).decode('utf-8').strip() - boundary = re.match(r'.+boundary="([^"]+)"', decrypted_envelope, re.MULTILINE | re.DOTALL).group(1) + boundary = re.match(r'.+boundary="([^"]+)"', + decrypted_envelope, re.MULTILINE | re.DOTALL).group(1) # replace auto-generated boundary with one we know mime_message = mime_message.replace("BOUNDARY", boundary) prefix = dedent("""\ - Content-Type: multipart/mixed; boundary=\"""" + \ - boundary + """\" + Content-Type: multipart/mixed; boundary=\"""" + boundary + """\" """) mime_message = prefix + mime_message @@ -237,7 +240,7 @@ def test_process_message_with_complex_message(self): with open(filename, 'r') as test_file: contents = test_file.read() - self.zeyple.process_message(contents, [TEST1_EMAIL]) # should not raise + self.zeyple.process_message(contents, [TEST1_EMAIL]) # should not raise def test_force_encryption(self): """Tries to encrypt without key""" diff --git a/tox.ini b/tox.ini deleted file mode 100644 index 3796438..0000000 --- a/tox.ini +++ /dev/null @@ -1,12 +0,0 @@ -[tox] -envlist = py311, py312, pypy3 -skip_missing_interpreters = True -skipsdist = True - -[testenv] -deps = -rrequirements.txt -setenv = - PYTHONPATH = {toxinidir} -commands = - pycodestyle --show-pep8 zeyple - py.test {posargs:--cov-report=html --cov=zeyple/} diff --git a/zeyple/zeyple.py b/zeyple/zeyple.py index 8b14b3c..2d5df09 100755 --- a/zeyple/zeyple.py +++ b/zeyple/zeyple.py @@ -2,7 +2,6 @@ # -*- coding: utf-8 -*- from configparser import ConfigParser -from io import BytesIO import copy import email import email.encoders @@ -16,14 +15,6 @@ import sys -def message_from_binary(message): - return email.message_from_bytes(message) - - -def as_binary_string(email): - return email.as_bytes() - - def encode_string(string): if isinstance(string, bytes): return string @@ -84,7 +75,7 @@ def process_message(self, message_data, recipients): """Encrypts the message with recipient keys""" message_data = encode_string(message_data) - in_message = message_from_binary(message_data) + in_message = email.message_from_bytes(message_data) logging.info( "Processing outgoing message %s", in_message['Message-id']) @@ -190,7 +181,7 @@ def _encrypt_message(self, in_message, key_id): # remove superfluous header del mixed['MIME-Version'] - payload = as_binary_string(mixed) + payload = mixed.as_bytes() encrypted_payload = self._encrypt_payload(payload, [key_id])