From 73f707690237f1180d92d653d78bbbf7fea0c0cb Mon Sep 17 00:00:00 2001 From: Andrew Teixeira Date: Thu, 12 Oct 2023 14:16:16 -0400 Subject: [PATCH] Migrate Python linting to Ruff * Change exceptions Pending and Revoked to PendingError and RevokedError (linting) * Refactor hard-coded numbers (linting) * Bump minor version for exception name changes --- .bumpversion.cfg | 2 +- .github/workflows/checks.yaml | 7 +- .github/workflows/deploy.yaml | 2 +- .github/workflows/test_deploy.yaml | 2 +- .gitignore | 1 + .pre-commit-config.yaml | 11 +- README.md | 2 +- cert_manager/__init__.py | 9 +- cert_manager/__version__.py | 3 +- cert_manager/_certificates.py | 16 +- cert_manager/_endpoint.py | 3 +- cert_manager/_helpers.py | 34 ++-- cert_manager/acme.py | 8 +- cert_manager/admin.py | 20 +-- cert_manager/client.py | 1 - cert_manager/domain.py | 19 +-- cert_manager/organization.py | 1 - cert_manager/person.py | 11 +- cert_manager/report.py | 7 +- cert_manager/smime.py | 29 ++-- cert_manager/ssl.py | 1 - poetry.lock | 249 +++++++++++++---------------- pylama.ini | 11 -- pyproject.toml | 21 ++- tests/__init__.py | 1 - tests/lib/__init__.py | 1 - tests/lib/testbase.py | 5 +- tests/test_acme.py | 177 ++++++-------------- tests/test_admin.py | 108 ++++--------- tests/test_certificates.py | 72 ++++----- tests/test_client.py | 65 ++++---- tests/test_domain.py | 174 ++++++-------------- tests/test_endpoint.py | 3 +- tests/test_helpers.py | 22 +-- tests/test_organization.py | 29 ++-- tests/test_person.py | 31 ++-- tests/test_report.py | 46 ++---- tests/test_smime.py | 68 ++++---- tests/test_ssl.py | 20 ++- 39 files changed, 507 insertions(+), 785 deletions(-) delete mode 100644 pylama.ini diff --git a/.bumpversion.cfg b/.bumpversion.cfg index 1aa7ca5..68b55bd 100644 --- a/.bumpversion.cfg +++ b/.bumpversion.cfg @@ -1,6 +1,6 @@ [bumpversion] commit = True -current_version = 2.3.1 +current_version = 2.4.0 tag = True tag_name = {new_version} diff --git a/.github/workflows/checks.yaml b/.github/workflows/checks.yaml index 8b6466b..6121ae4 100644 --- a/.github/workflows/checks.yaml +++ b/.github/workflows/checks.yaml @@ -11,8 +11,11 @@ name: 'checks' jobs: linting: - uses: broadinstitute/shared-workflows/.github/workflows/poetry-linting.yml@v1.4.0 + uses: broadinstitute/shared-workflows/.github/workflows/poetry-linting.yml@v1.4.1 + with: + use_pylama: false + use_ruff: true unit-tests: - uses: broadinstitute/shared-workflows/.github/workflows/poetry-unit-tests.yml@v1.4.0 + uses: broadinstitute/shared-workflows/.github/workflows/poetry-unit-tests.yml@v1.4.1 with: python_package_name: cert_manager diff --git a/.github/workflows/deploy.yaml b/.github/workflows/deploy.yaml index 8344bb5..9c736e3 100644 --- a/.github/workflows/deploy.yaml +++ b/.github/workflows/deploy.yaml @@ -8,6 +8,6 @@ name: 'deploy' jobs: pypi_deploy: - uses: broadinstitute/shared-workflows/.github/workflows/poetry-deploy.yml@v1.4.0 + uses: broadinstitute/shared-workflows/.github/workflows/poetry-deploy.yml@v1.4.1 secrets: pypi_token: ${{ secrets.PYPI_TOKEN }} diff --git a/.github/workflows/test_deploy.yaml b/.github/workflows/test_deploy.yaml index b7e319e..29b980b 100644 --- a/.github/workflows/test_deploy.yaml +++ b/.github/workflows/test_deploy.yaml @@ -11,6 +11,6 @@ name: 'test_deploy' jobs: pypi_test_deploy: - uses: broadinstitute/shared-workflows/.github/workflows/poetry-test-deploy.yml@v1.4.0 + uses: broadinstitute/shared-workflows/.github/workflows/poetry-test-deploy.yml@v1.4.1 secrets: pypi_test_token: ${{ secrets.PYPI_TEST_TOKEN }} diff --git a/.gitignore b/.gitignore index 000b97b..bbebc60 100644 --- a/.gitignore +++ b/.gitignore @@ -17,3 +17,4 @@ __pycache__/ # VSCode settings .vscode +.ruff_cache diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 8345bb7..69087b9 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -1,7 +1,7 @@ --- repos: - repo: https://github.com/pre-commit/pre-commit-hooks - rev: v4.4.0 + rev: v4.5.0 hooks: - id: check-added-large-files - id: check-ast @@ -18,7 +18,6 @@ repos: - id: detect-private-key - id: end-of-file-fixer exclude: '.bumpversion.cfg' - - id: fix-encoding-pragma - id: mixed-line-ending - id: name-tests-test args: @@ -27,10 +26,12 @@ repos: args: - -b main - id: trailing-whitespace - - repo: https://github.com/broadinstitute/mirrors-pylama.git - rev: v1.0.2 + - repo: https://github.com/astral-sh/ruff-pre-commit + # Ruff version. + rev: v0.1.0 hooks: - - id: pylama + - id: ruff + args: [--fix, --exit-non-zero-on-fix] - repo: https://github.com/adrienverge/yamllint.git rev: v1.32.0 hooks: diff --git a/README.md b/README.md index 2c506c3..e625aa0 100644 --- a/README.md +++ b/README.md @@ -94,7 +94,7 @@ while(True): cert_pem = ssl.collect(cert_id=result["sslId"], cert_format="x509CO") print(cert_pem) break - except Pending: + except PendingError: print("Certificate is still pending...sleeping for 60s") sleep(60) continue diff --git a/cert_manager/__init__.py b/cert_manager/__init__.py index 3fb3f0d..38d3037 100644 --- a/cert_manager/__init__.py +++ b/cert_manager/__init__.py @@ -1,15 +1,16 @@ -# -*- coding: utf-8 -*- """Initialize the cert_manager module.""" +from ._helpers import PendingError from .acme import ACMEAccount from .admin import Admin from .client import Client from .domain import Domain -from .report import Report -from ._helpers import Pending from .organization import Organization from .person import Person +from .report import Report from .smime import SMIME from .ssl import SSL -__all__ = ["ACMEAccount", "Admin", "Client", "Domain", "Organization", "Pending", "Person", "Report", "SMIME", "SSL"] +__all__ = [ + "ACMEAccount", "Admin", "Client", "Domain", "Organization", "PendingError", "Person", "Report", "SMIME", "SSL" +] diff --git a/cert_manager/__version__.py b/cert_manager/__version__.py index b127e36..cf1febe 100644 --- a/cert_manager/__version__.py +++ b/cert_manager/__version__.py @@ -1,7 +1,6 @@ -# -*- coding: utf-8 -*- """Keep track of the version for the cert_manager module.""" __title__ = "cert_manager" __description__ = "Python interface to the Sectigo Certificate Manager REST API" __url__ = "https://github.com/broadinstitute/python-cert_manager" -__version__ = "2.3.1" +__version__ = "2.4.0" diff --git a/cert_manager/_certificates.py b/cert_manager/_certificates.py index e1c5bd6..fc94bb3 100644 --- a/cert_manager/_certificates.py +++ b/cert_manager/_certificates.py @@ -1,11 +1,11 @@ -# -*- coding: utf-8 -*- """Define the cert_manager._certificate.Certificates base class.""" import logging + from requests.exceptions import HTTPError -from ._helpers import CustomFieldsError, Pending from ._endpoint import Endpoint +from ._helpers import CustomFieldsError, PendingError LOGGER = logging.getLogger(__name__) @@ -41,6 +41,7 @@ def __init__(self, client, endpoint, api_version="v1"): # Set to None initially. Will be filled in by methods later. self.__cert_types = None self.__custom_fields = None + self.__reason_maxlen = 512 @property def types(self): @@ -81,8 +82,7 @@ def custom_fields(self): return self.__custom_fields def _validate_custom_fields(self, custom_fields): - """Check the structure and contents of a list of dicts representing custom fields - Raise exceptions if validation fails + """Check the structure and contents of a list of custom fields dicts. Raise exceptions if validation fails. :raises Exception: if any of the validation steps fail """ @@ -112,7 +112,7 @@ def _validate_custom_fields(self, custom_fields): def collect(self, cert_id, cert_format): """Retrieve an existing certificate from the API. - This method will raise a Pending exception if the certificate is still in a pending state. + This method will raise a PendingError exception if the certificate is still in a pending state. :param int cert_id: The certificate ID :param str cert_format: The format in which to retreive the certificate. Allowed values: *self.valid_formats* @@ -126,7 +126,7 @@ def collect(self, cert_id, cert_format): try: result = self._client.get(url) except HTTPError as exc: - raise Pending(f"certificate {cert_id} still in 'pending' state") from exc + raise PendingError(f"certificate {cert_id} still in 'pending' state") from exc # The certificate is ready for collection return result.content.decode(result.encoding) @@ -229,8 +229,8 @@ def revoke(self, cert_id, reason=""): url = self._url(f"/revoke/{cert_id}") # Sectigo has a 512 character limit on the "reason" message, so catch that here. - if (not reason) or (len(reason) > 511): - raise ValueError("Sectigo limit: reason must be > 0 character and < 512 characters") + if not reason or len(reason) >= self.__reason_maxlen: + raise ValueError(f"Sectigo limit: reason must be > 0 character and < {self.__reason_maxlen} characters") data = {"reason": reason} diff --git a/cert_manager/_endpoint.py b/cert_manager/_endpoint.py index c8934b6..bba0224 100644 --- a/cert_manager/_endpoint.py +++ b/cert_manager/_endpoint.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager._endpoint.Endpoint base class.""" import logging @@ -19,6 +18,8 @@ def __init__(self, client, endpoint, api_version="v1"): self._client = client self._api_version = api_version self._api_url = self.create_api_url(client.base_url, endpoint, self._api_version) + self._expected_code = 201 + self._capture_err_code = 400 @property def api_version(self): diff --git a/cert_manager/_helpers.py b/cert_manager/_helpers.py index 579f3d5..cc059d0 100644 --- a/cert_manager/_helpers.py +++ b/cert_manager/_helpers.py @@ -1,9 +1,8 @@ -# -*- coding: utf-8 -*- """Define helper functions used by classes in this module.""" -from functools import wraps import logging import re +from functools import wraps from requests.exceptions import HTTPError @@ -36,23 +35,18 @@ def log_traffic(*args, **kwargs): # Check if the URL or headers exist in the parameters # Note: *self* will be the first argument, so actual arguments start after that. - url = kwargs.get("url", "") - if not url: - if len(args) > 1: - url = args[1] - headers = kwargs.get("headers", "") - if not headers: - if len(args) > 2: - headers = args[2] - data = kwargs.get("data", "") - if not data: - if len(args) > 3: - data = args[3] + params = ["url", "headers", "data"] + values = {} + for index, param in enumerate(params, start=1): + values[param] = kwargs.get(param, "") + if not values[param]: + if len(args) > index: + values[param] = args[index] # Print out before messages with URL and header data - traffic_logger.debug(f"Performing a {func_name} on url: {url}") - traffic_logger.debug(f"Extra request headers: {headers}") - traffic_logger.debug(f"Data: {data}") + traffic_logger.debug(f"Performing a {func_name} on url: {values['url']}") + traffic_logger.debug(f"Extra request headers: {values['headers']}") + traffic_logger.debug(f"Data: {values['data']}") # Run the wrapped function try: @@ -149,13 +143,13 @@ def decorator(*args, **kwargs): return decorator -class Pending(Exception): +class PendingError(Exception): """Serve as a generic Exception indicating a certificate is in a pending state.""" CODE = -183 -class Revoked(Exception): - """Serve as a generic Exception indicating a certificate has been revoked""" +class RevokedError(Exception): + """Serve as a generic Exception indicating a certificate has been revoked.""" CODE = -192 diff --git a/cert_manager/acme.py b/cert_manager/acme.py index d67c820..bfadb8c 100644 --- a/cert_manager/acme.py +++ b/cert_manager/acme.py @@ -1,8 +1,7 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.acme.ACMEAccount class.""" -import re import logging +import re from ._endpoint import Endpoint from ._helpers import paginate @@ -11,7 +10,7 @@ class ACMEAccountCreationResponseError(Exception): - """An error (other than HTTPError) occurred while processing ACME Account Creation API response""" + """An error (other than HTTPError) occurred while processing ACME Account Creation API response.""" class ACMEAccount(Endpoint): @@ -36,7 +35,6 @@ def __init__(self, client, api_version="v1"): """ super().__init__(client=client, endpoint="/acme", api_version=api_version) self._api_url = self._url("/account") - self.__acme_accounts = None def all(self, org_id, force=False): @@ -111,7 +109,7 @@ def create(self, name, acme_server, org_id, ev_details=None): result = self._client.post(self._api_url, data=data) # for status >= 400, HTTPError is raised - if result.status_code != 201: + if result.status_code != self._expected_code: raise ACMEAccountCreationResponseError(f"Unexpected HTTP status {result.status_code}") try: loc = result.headers["Location"] diff --git a/cert_manager/admin.py b/cert_manager/admin.py index a9abbfa..4ef2a24 100644 --- a/cert_manager/admin.py +++ b/cert_manager/admin.py @@ -1,8 +1,8 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.admin.Admin class.""" -import re import logging +import re + from requests.exceptions import HTTPError from ._endpoint import Endpoint @@ -11,7 +11,7 @@ class AdminCreationResponseError(Exception): - """An error (other than HTTPError) occurred while processing Admin Creation API response""" + """An error (other than HTTPError) occurred while processing Admin Creation API response.""" class Admin(Endpoint): @@ -44,9 +44,8 @@ def all(self, force=False): return self.__admins - def create(self, login, email, forename, surname, # pylint: disable=too-many-arguments - password, credentials, **kwargs): - """Create a new administrator + def create(self, login, email, forename, surname, password, credentials, **kwargs): # noqa: PLR0913 + """Create a new administrator. :param str login: Login name of admin to create :param str email: Email of admin to create @@ -71,6 +70,7 @@ def create(self, login, email, forename, surname, # pylint: disable=too-many-ar "password": password, "credentials": credentials, } + for key, value in kwargs.items(): data[key] = value @@ -78,13 +78,13 @@ def create(self, login, email, forename, surname, # pylint: disable=too-many-ar result = self._client.post(self._api_url, data=data) except HTTPError as exc: status_code = exc.response.status_code - if status_code == 400: + if status_code == self._capture_err_code: err_response = exc.response.json() raise ValueError(err_response["description"]) from exc raise exc # for status >= 400, HTTPError is raised - if result.status_code != 201: + if result.status_code != self._expected_code: raise AdminCreationResponseError(f"Unexpected HTTP status {result.status_code}") try: loc = result.headers["Location"] @@ -115,7 +115,7 @@ def get(self, admin_id): return result.json() def get_idps(self): - """Return a list of IDPs + """Return a list of IDPs. :return list: A list of dictionaries representing the IDPs """ @@ -153,7 +153,7 @@ def update(self, admin_id, **kwargs): result = self._client.put(url, data=data) except HTTPError as exc: status_code = exc.response.status_code - if status_code == 400: + if status_code == self._capture_err_code: err_response = exc.response.json() raise ValueError(err_response["description"]) from exc raise exc diff --git a/cert_manager/client.py b/cert_manager/client.py index c385aeb..bf4634d 100644 --- a/cert_manager/client.py +++ b/cert_manager/client.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.client.Client class.""" import logging diff --git a/cert_manager/domain.py b/cert_manager/domain.py index d4f029c..512d3fa 100644 --- a/cert_manager/domain.py +++ b/cert_manager/domain.py @@ -1,8 +1,8 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.domain.Domain class.""" -import re import logging +import re + from requests.exceptions import HTTPError from ._endpoint import Endpoint @@ -11,7 +11,7 @@ class DomainCreationResponseError(Exception): - """An error (other than HTTPError) occurred while processing Domain Creation API response""" + """An error (other than HTTPError) occurred while processing Domain Creation API response.""" class Domain(Endpoint): @@ -56,7 +56,8 @@ def find(self, **kwargs): def count(self, **kwargs): """Return a count of domains matching the given parameters from Sectigo. - If no parameters are given, the count will be of all domains + + If no parameters are given, the count will be of all domains. :return dict: Count of domains matching the given parameters """ @@ -66,7 +67,7 @@ def count(self, **kwargs): return result.json() def create(self, name, org_id, cert_types, **kwargs): - """Create a domain + """Create a domain. :param str name: Name of domain to create :param int org_id: Organization Id to delegate the newly created domain to @@ -91,13 +92,13 @@ def create(self, name, org_id, cert_types, **kwargs): result = self._client.post(self._api_url, data=data) except HTTPError as exc: status_code = exc.response.status_code - if status_code == 400: + if status_code == self._capture_err_code: err_response = exc.response.json() raise ValueError(err_response["description"]) from exc raise exc # for status >= 400, HTTPError is raised - if result.status_code != 201: + if result.status_code != self._expected_code: raise DomainCreationResponseError(f"Unexpected HTTP status {result.status_code}") try: loc = result.headers["Location"] @@ -201,7 +202,7 @@ def remove_delegation(self, domain_id, org_id, cert_types): return result.ok def approve_delegation(self, domain_id, org_id): - """Approve a requested delegation + """Approve a requested delegation. :param int domain_id: The ID of the domain to approve :param int org_id: The ID of the organization requesting the delegation @@ -217,7 +218,7 @@ def approve_delegation(self, domain_id, org_id): return result.ok def reject_delegation(self, domain_id, org_id): - """Reject a requested delegation + """Reject a requested delegation. :param int domain_id: The ID of the domain to approve :param int org_id: The ID of the organization requesting the delegation diff --git a/cert_manager/organization.py b/cert_manager/organization.py index a43fcf8..5ccc7b0 100644 --- a/cert_manager/organization.py +++ b/cert_manager/organization.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.organization.Organization class.""" import logging diff --git a/cert_manager/person.py b/cert_manager/person.py index 6238ded..5fb06d0 100644 --- a/cert_manager/person.py +++ b/cert_manager/person.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.person.Person class.""" import logging @@ -23,7 +22,7 @@ def __init__(self, client, api_version="v1"): @paginate def list(self, **kwargs): - """Return a list of people in sectigo + """Return a list of people in sectigo. The 'size' and 'position' parameters passed as arguments to this function will be used by the pagination wrapper to page through results. All other filtering parameters can be @@ -51,7 +50,7 @@ def find(self, email): return result.json() def get(self, person_id): - """Returns the details of a person + """Returns the details of a person. :param int person_id: The person's ID :return dict: A dictionary of the person's details @@ -61,7 +60,7 @@ def get(self, person_id): return result.json() def create(self, **kwargs) -> dict: - """Create a person + """Create a person. :param string first_name: The person's first name :param string middleName: The person's middle name @@ -102,7 +101,7 @@ def create(self, **kwargs) -> dict: return {"personId": created_id} def update(self, **kwargs) -> None: - """Update a person + """Update a person. :param string person_id: The person's id :param string first_name: The person's first name @@ -139,7 +138,7 @@ def update(self, **kwargs) -> None: self._client.put(self._url(str(person_id)), data=data) def delete(self, **kwargs): - """Delete a person + """Delete a person. :param string person_id: The person's id """ diff --git a/cert_manager/report.py b/cert_manager/report.py index d8f4c4d..8253442 100644 --- a/cert_manager/report.py +++ b/cert_manager/report.py @@ -1,7 +1,7 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.report.Report class.""" import logging + from requests.exceptions import HTTPError from ._endpoint import Endpoint @@ -46,7 +46,7 @@ def get(self, report_name, **kwargs): result = self._client.post(url, data=data) except HTTPError as exc: status_code = exc.response.status_code - if status_code == 400: + if status_code == self._capture_err_code: err_response = exc.response.json() raise ValueError(err_response["description"]) from exc raise exc @@ -67,7 +67,6 @@ def get_ssl_certs(self, **kwargs): return dict: The report data """ - report_url = "ssl-certificates" result = self.get(report_url, **kwargs) @@ -87,7 +86,6 @@ def get_client_certs(self, **kwargs): return dict: The report data """ - report_url = "client-certificates" return self.get(report_url, **kwargs) @@ -105,7 +103,6 @@ def get_device_certs(self, **kwargs): return dict: The report data """ - report_url = "device-certificates" return self.get(report_url, **kwargs) diff --git a/cert_manager/smime.py b/cert_manager/smime.py index 8267491..5770fd6 100644 --- a/cert_manager/smime.py +++ b/cert_manager/smime.py @@ -1,10 +1,10 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.certificates.smime.SMIME class.""" import logging + from requests.exceptions import HTTPError + from ._certificates import Certificates -from ._helpers import Pending, Revoked -from ._helpers import paginate, version_hack +from ._helpers import PendingError, RevokedError, paginate, version_hack LOGGER = logging.getLogger(__name__) @@ -19,6 +19,7 @@ def __init__(self, client, api_version="v1"): :param string api_version: The API version to use; the default is "v1" """ super().__init__(client=client, endpoint="/smime", api_version=api_version) + self.__reason_maxlen = 512 @paginate def list(self, **kwargs): @@ -38,7 +39,7 @@ def list(self, **kwargs): @version_hack(service="smime", version="v2") def list_by_email(self, **kwargs): - """Return a list of all client certificates for a person with given email + """Return a list of all client certificates for a person with given email. :param str email: Person email :return iter: An iterator object is returned to cycle through the certificates @@ -118,7 +119,7 @@ def enroll(self, **kwargs): def collect(self, cert_id): """Retrieve an existing client certificate from the API. - This method will raise a Pending exception if the certificate is still in a pending state. + This method will raise a PendingError exception if the certificate is still in a pending state. :param int cert_id: The Certificate ID given on enroll success :return str: the string representing the certificate in the requested format @@ -131,10 +132,10 @@ def collect(self, cert_id): result = self._client.get(url) except HTTPError as exc: err_code = exc.response.json().get("code") - if err_code == Revoked.CODE: - raise Revoked(f"certificate {cert_id} in 'revoked' state") from exc - if err_code == Pending.CODE: - raise Pending(f"certificate {cert_id} still in 'pending' state") from exc + if err_code == RevokedError.CODE: + raise RevokedError(f"certificate {cert_id} in 'revoked' state") from exc + if err_code == PendingError.CODE: + raise PendingError(f"certificate {cert_id} still in 'pending' state") from exc raise exc # The certificate is ready for collection @@ -193,14 +194,14 @@ def revoke(self, cert_id, reason=""): raise ValueError("Argument 'cert_id' can't be None") # Sectigo has a 512 character limit on the "reason" message, so catch that here. - if (not reason) or (len(reason) > 511): - raise ValueError("Sectigo limit: reason must be > 0 character and < 512 characters") + if not reason or len(reason) >= self.__reason_maxlen: + raise ValueError(f"Sectigo limit: reason must be > 0 character and < {self.__reason_maxlen} characters") data = {"reason": reason} self._client.post(url, data=data) def revoke_by_email(self, email, reason=""): - """Revoke all client certificate related to an email + """Revoke all client certificate related to an email. :param str email: The person email address :param str reason: The Reason for revocation. @@ -212,8 +213,8 @@ def revoke_by_email(self, email, reason=""): raise ValueError("Argument 'email' can't be empty or None") # Sectigo has a 512 character limit on the "reason" message, so catch that here. - if (not reason) or (len(reason) > 511): - raise ValueError("Sectigo limit: reason must be > 0 character and < 512 characters") + if not reason or len(reason) >= self.__reason_maxlen: + raise ValueError(f"Sectigo limit: reason must be > 0 character and < {self.__reason_maxlen} characters") data = {"email": email, "reason": reason} self._client.post(url, data=data) diff --git a/cert_manager/ssl.py b/cert_manager/ssl.py index 06c5439..4769548 100644 --- a/cert_manager/ssl.py +++ b/cert_manager/ssl.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.certificates.ssl.SSL class.""" import logging diff --git a/poetry.lock b/poetry.lock index 88c0f4f..5d70716 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry 1.6.1 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.5.1 and should not be changed by hand. [[package]] name = "astroid" @@ -136,86 +136,101 @@ pycparser = "*" [[package]] name = "charset-normalizer" -version = "3.2.0" +version = "3.3.0" description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." optional = false python-versions = ">=3.7.0" files = [ - {file = "charset-normalizer-3.2.0.tar.gz", hash = "sha256:3bb3d25a8e6c0aedd251753a79ae98a093c7e7b471faa3aa9a93a81431987ace"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:0b87549028f680ca955556e3bd57013ab47474c3124dc069faa0b6545b6c9710"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:7c70087bfee18a42b4040bb9ec1ca15a08242cf5867c58726530bdf3945672ed"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a103b3a7069b62f5d4890ae1b8f0597618f628b286b03d4bc9195230b154bfa9"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:94aea8eff76ee6d1cdacb07dd2123a68283cb5569e0250feab1240058f53b623"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:db901e2ac34c931d73054d9797383d0f8009991e723dab15109740a63e7f902a"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b0dac0ff919ba34d4df1b6131f59ce95b08b9065233446be7e459f95554c0dc8"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:193cbc708ea3aca45e7221ae58f0fd63f933753a9bfb498a3b474878f12caaad"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:09393e1b2a9461950b1c9a45d5fd251dc7c6f228acab64da1c9c0165d9c7765c"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:baacc6aee0b2ef6f3d308e197b5d7a81c0e70b06beae1f1fcacffdbd124fe0e3"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:bf420121d4c8dce6b889f0e8e4ec0ca34b7f40186203f06a946fa0276ba54029"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:c04a46716adde8d927adb9457bbe39cf473e1e2c2f5d0a16ceb837e5d841ad4f"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:aaf63899c94de41fe3cf934601b0f7ccb6b428c6e4eeb80da72c58eab077b19a"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:d62e51710986674142526ab9f78663ca2b0726066ae26b78b22e0f5e571238dd"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-win32.whl", hash = "sha256:04e57ab9fbf9607b77f7d057974694b4f6b142da9ed4a199859d9d4d5c63fe96"}, - {file = "charset_normalizer-3.2.0-cp310-cp310-win_amd64.whl", hash = "sha256:48021783bdf96e3d6de03a6e39a1171ed5bd7e8bb93fc84cc649d11490f87cea"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:4957669ef390f0e6719db3613ab3a7631e68424604a7b448f079bee145da6e09"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:46fb8c61d794b78ec7134a715a3e564aafc8f6b5e338417cb19fe9f57a5a9bf2"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:f779d3ad205f108d14e99bb3859aa7dd8e9c68874617c72354d7ecaec2a054ac"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f25c229a6ba38a35ae6e25ca1264621cc25d4d38dca2942a7fce0b67a4efe918"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2efb1bd13885392adfda4614c33d3b68dee4921fd0ac1d3988f8cbb7d589e72a"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1f30b48dd7fa1474554b0b0f3fdfdd4c13b5c737a3c6284d3cdc424ec0ffff3a"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:246de67b99b6851627d945db38147d1b209a899311b1305dd84916f2b88526c6"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9bd9b3b31adcb054116447ea22caa61a285d92e94d710aa5ec97992ff5eb7cf3"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:8c2f5e83493748286002f9369f3e6607c565a6a90425a3a1fef5ae32a36d749d"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:3170c9399da12c9dc66366e9d14da8bf7147e1e9d9ea566067bbce7bb74bd9c2"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:7a4826ad2bd6b07ca615c74ab91f32f6c96d08f6fcc3902ceeedaec8cdc3bcd6"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:3b1613dd5aee995ec6d4c69f00378bbd07614702a315a2cf6c1d21461fe17c23"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:9e608aafdb55eb9f255034709e20d5a83b6d60c054df0802fa9c9883d0a937aa"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-win32.whl", hash = "sha256:f2a1d0fd4242bd8643ce6f98927cf9c04540af6efa92323e9d3124f57727bfc1"}, - {file = "charset_normalizer-3.2.0-cp311-cp311-win_amd64.whl", hash = "sha256:681eb3d7e02e3c3655d1b16059fbfb605ac464c834a0c629048a30fad2b27489"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:c57921cda3a80d0f2b8aec7e25c8aa14479ea92b5b51b6876d975d925a2ea346"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:41b25eaa7d15909cf3ac4c96088c1f266a9a93ec44f87f1d13d4a0e86c81b982"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f058f6963fd82eb143c692cecdc89e075fa0828db2e5b291070485390b2f1c9c"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a7647ebdfb9682b7bb97e2a5e7cb6ae735b1c25008a70b906aecca294ee96cf4"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:eef9df1eefada2c09a5e7a40991b9fc6ac6ef20b1372abd48d2794a316dc0449"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e03b8895a6990c9ab2cdcd0f2fe44088ca1c65ae592b8f795c3294af00a461c3"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:ee4006268ed33370957f55bf2e6f4d263eaf4dc3cfc473d1d90baff6ed36ce4a"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:c4983bf937209c57240cff65906b18bb35e64ae872da6a0db937d7b4af845dd7"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:3bb7fda7260735efe66d5107fb7e6af6a7c04c7fce9b2514e04b7a74b06bf5dd"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:72814c01533f51d68702802d74f77ea026b5ec52793c791e2da806a3844a46c3"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:70c610f6cbe4b9fce272c407dd9d07e33e6bf7b4aa1b7ffb6f6ded8e634e3592"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-win32.whl", hash = "sha256:a401b4598e5d3f4a9a811f3daf42ee2291790c7f9d74b18d75d6e21dda98a1a1"}, - {file = "charset_normalizer-3.2.0-cp37-cp37m-win_amd64.whl", hash = "sha256:c0b21078a4b56965e2b12f247467b234734491897e99c1d51cee628da9786959"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:95eb302ff792e12aba9a8b8f8474ab229a83c103d74a750ec0bd1c1eea32e669"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1a100c6d595a7f316f1b6f01d20815d916e75ff98c27a01ae817439ea7726329"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:6339d047dab2780cc6220f46306628e04d9750f02f983ddb37439ca47ced7149"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e4b749b9cc6ee664a3300bb3a273c1ca8068c46be705b6c31cf5d276f8628a94"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a38856a971c602f98472050165cea2cdc97709240373041b69030be15047691f"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f87f746ee241d30d6ed93969de31e5ffd09a2961a051e60ae6bddde9ec3583aa"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:89f1b185a01fe560bc8ae5f619e924407efca2191b56ce749ec84982fc59a32a"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e1c8a2f4c69e08e89632defbfabec2feb8a8d99edc9f89ce33c4b9e36ab63037"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:2f4ac36d8e2b4cc1aa71df3dd84ff8efbe3bfb97ac41242fbcfc053c67434f46"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:a386ebe437176aab38c041de1260cd3ea459c6ce5263594399880bbc398225b2"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:ccd16eb18a849fd8dcb23e23380e2f0a354e8daa0c984b8a732d9cfaba3a776d"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:e6a5bf2cba5ae1bb80b154ed68a3cfa2fa00fde979a7f50d6598d3e17d9ac20c"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:45de3f87179c1823e6d9e32156fb14c1927fcc9aba21433f088fdfb555b77c10"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-win32.whl", hash = "sha256:1000fba1057b92a65daec275aec30586c3de2401ccdcd41f8a5c1e2c87078706"}, - {file = "charset_normalizer-3.2.0-cp38-cp38-win_amd64.whl", hash = "sha256:8b2c760cfc7042b27ebdb4a43a4453bd829a5742503599144d54a032c5dc7e9e"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:855eafa5d5a2034b4621c74925d89c5efef61418570e5ef9b37717d9c796419c"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:203f0c8871d5a7987be20c72442488a0b8cfd0f43b7973771640fc593f56321f"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:e857a2232ba53ae940d3456f7533ce6ca98b81917d47adc3c7fd55dad8fab858"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5e86d77b090dbddbe78867a0275cb4df08ea195e660f1f7f13435a4649e954e5"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c4fb39a81950ec280984b3a44f5bd12819953dc5fa3a7e6fa7a80db5ee853952"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2dee8e57f052ef5353cf608e0b4c871aee320dd1b87d351c28764fc0ca55f9f4"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8700f06d0ce6f128de3ccdbc1acaea1ee264d2caa9ca05daaf492fde7c2a7200"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1920d4ff15ce893210c1f0c0e9d19bfbecb7983c76b33f046c13a8ffbd570252"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:c1c76a1743432b4b60ab3358c937a3fe1341c828ae6194108a94c69028247f22"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:f7560358a6811e52e9c4d142d497f1a6e10103d3a6881f18d04dbce3729c0e2c"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:c8063cf17b19661471ecbdb3df1c84f24ad2e389e326ccaf89e3fb2484d8dd7e"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:cd6dbe0238f7743d0efe563ab46294f54f9bc8f4b9bcf57c3c666cc5bc9d1299"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:1249cbbf3d3b04902ff081ffbb33ce3377fa6e4c7356f759f3cd076cc138d020"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-win32.whl", hash = "sha256:6c409c0deba34f147f77efaa67b8e4bb83d2f11c8806405f76397ae5b8c0d1c9"}, - {file = "charset_normalizer-3.2.0-cp39-cp39-win_amd64.whl", hash = "sha256:7095f6fbfaa55defb6b733cfeb14efaae7a29f0b59d8cf213be4e7ca0b857b80"}, - {file = "charset_normalizer-3.2.0-py3-none-any.whl", hash = "sha256:8e098148dd37b4ce3baca71fb394c81dc5d9c7728c95df695d2dca218edf40e6"}, + {file = "charset-normalizer-3.3.0.tar.gz", hash = "sha256:63563193aec44bce707e0c5ca64ff69fa72ed7cf34ce6e11d5127555756fd2f6"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:effe5406c9bd748a871dbcaf3ac69167c38d72db8c9baf3ff954c344f31c4cbe"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:4162918ef3098851fcd8a628bf9b6a98d10c380725df9e04caf5ca6dd48c847a"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:0570d21da019941634a531444364f2482e8db0b3425fcd5ac0c36565a64142c8"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5707a746c6083a3a74b46b3a631d78d129edab06195a92a8ece755aac25a3f3d"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:278c296c6f96fa686d74eb449ea1697f3c03dc28b75f873b65b5201806346a69"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a4b71f4d1765639372a3b32d2638197f5cd5221b19531f9245fcc9ee62d38f56"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f5969baeaea61c97efa706b9b107dcba02784b1601c74ac84f2a532ea079403e"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a3f93dab657839dfa61025056606600a11d0b696d79386f974e459a3fbc568ec"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:db756e48f9c5c607b5e33dd36b1d5872d0422e960145b08ab0ec7fd420e9d649"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:232ac332403e37e4a03d209a3f92ed9071f7d3dbda70e2a5e9cff1c4ba9f0678"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:e5c1502d4ace69a179305abb3f0bb6141cbe4714bc9b31d427329a95acfc8bdd"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:2502dd2a736c879c0f0d3e2161e74d9907231e25d35794584b1ca5284e43f596"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:23e8565ab7ff33218530bc817922fae827420f143479b753104ab801145b1d5b"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-win32.whl", hash = "sha256:1872d01ac8c618a8da634e232f24793883d6e456a66593135aeafe3784b0848d"}, + {file = "charset_normalizer-3.3.0-cp310-cp310-win_amd64.whl", hash = "sha256:557b21a44ceac6c6b9773bc65aa1b4cc3e248a5ad2f5b914b91579a32e22204d"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:d7eff0f27edc5afa9e405f7165f85a6d782d308f3b6b9d96016c010597958e63"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6a685067d05e46641d5d1623d7c7fdf15a357546cbb2f71b0ebde91b175ffc3e"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:0d3d5b7db9ed8a2b11a774db2bbea7ba1884430a205dbd54a32d61d7c2a190fa"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2935ffc78db9645cb2086c2f8f4cfd23d9b73cc0dc80334bc30aac6f03f68f8c"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9fe359b2e3a7729010060fbca442ca225280c16e923b37db0e955ac2a2b72a05"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:380c4bde80bce25c6e4f77b19386f5ec9db230df9f2f2ac1e5ad7af2caa70459"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f0d1e3732768fecb052d90d62b220af62ead5748ac51ef61e7b32c266cac9293"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1b2919306936ac6efb3aed1fbf81039f7087ddadb3160882a57ee2ff74fd2382"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:f8888e31e3a85943743f8fc15e71536bda1c81d5aa36d014a3c0c44481d7db6e"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:82eb849f085624f6a607538ee7b83a6d8126df6d2f7d3b319cb837b289123078"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:7b8b8bf1189b3ba9b8de5c8db4d541b406611a71a955bbbd7385bbc45fcb786c"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:5adf257bd58c1b8632046bbe43ee38c04e1038e9d37de9c57a94d6bd6ce5da34"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:c350354efb159b8767a6244c166f66e67506e06c8924ed74669b2c70bc8735b1"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-win32.whl", hash = "sha256:02af06682e3590ab952599fbadac535ede5d60d78848e555aa58d0c0abbde786"}, + {file = "charset_normalizer-3.3.0-cp311-cp311-win_amd64.whl", hash = "sha256:86d1f65ac145e2c9ed71d8ffb1905e9bba3a91ae29ba55b4c46ae6fc31d7c0d4"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:3b447982ad46348c02cb90d230b75ac34e9886273df3a93eec0539308a6296d7"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:abf0d9f45ea5fb95051c8bfe43cb40cda383772f7e5023a83cc481ca2604d74e"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:b09719a17a2301178fac4470d54b1680b18a5048b481cb8890e1ef820cb80455"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b3d9b48ee6e3967b7901c052b670c7dda6deb812c309439adaffdec55c6d7b78"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:edfe077ab09442d4ef3c52cb1f9dab89bff02f4524afc0acf2d46be17dc479f5"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3debd1150027933210c2fc321527c2299118aa929c2f5a0a80ab6953e3bd1908"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:86f63face3a527284f7bb8a9d4f78988e3c06823f7bea2bd6f0e0e9298ca0403"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:24817cb02cbef7cd499f7c9a2735286b4782bd47a5b3516a0e84c50eab44b98e"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:c71f16da1ed8949774ef79f4a0260d28b83b3a50c6576f8f4f0288d109777989"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:9cf3126b85822c4e53aa28c7ec9869b924d6fcfb76e77a45c44b83d91afd74f9"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:b3b2316b25644b23b54a6f6401074cebcecd1244c0b8e80111c9a3f1c8e83d65"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:03680bb39035fbcffe828eae9c3f8afc0428c91d38e7d61aa992ef7a59fb120e"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:4cc152c5dd831641e995764f9f0b6589519f6f5123258ccaca8c6d34572fefa8"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-win32.whl", hash = "sha256:b8f3307af845803fb0b060ab76cf6dd3a13adc15b6b451f54281d25911eb92df"}, + {file = "charset_normalizer-3.3.0-cp312-cp312-win_amd64.whl", hash = "sha256:8eaf82f0eccd1505cf39a45a6bd0a8cf1c70dcfc30dba338207a969d91b965c0"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:dc45229747b67ffc441b3de2f3ae5e62877a282ea828a5bdb67883c4ee4a8810"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2f4a0033ce9a76e391542c182f0d48d084855b5fcba5010f707c8e8c34663d77"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ada214c6fa40f8d800e575de6b91a40d0548139e5dc457d2ebb61470abf50186"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b1121de0e9d6e6ca08289583d7491e7fcb18a439305b34a30b20d8215922d43c"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1063da2c85b95f2d1a430f1c33b55c9c17ffaf5e612e10aeaad641c55a9e2b9d"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:70f1d09c0d7748b73290b29219e854b3207aea922f839437870d8cc2168e31cc"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:250c9eb0f4600361dd80d46112213dff2286231d92d3e52af1e5a6083d10cad9"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:750b446b2ffce1739e8578576092179160f6d26bd5e23eb1789c4d64d5af7dc7"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:fc52b79d83a3fe3a360902d3f5d79073a993597d48114c29485e9431092905d8"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:588245972aca710b5b68802c8cad9edaa98589b1b42ad2b53accd6910dad3545"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:e39c7eb31e3f5b1f88caff88bcff1b7f8334975b46f6ac6e9fc725d829bc35d4"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-win32.whl", hash = "sha256:abecce40dfebbfa6abf8e324e1860092eeca6f7375c8c4e655a8afb61af58f2c"}, + {file = "charset_normalizer-3.3.0-cp37-cp37m-win_amd64.whl", hash = "sha256:24a91a981f185721542a0b7c92e9054b7ab4fea0508a795846bc5b0abf8118d4"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:67b8cc9574bb518ec76dc8e705d4c39ae78bb96237cb533edac149352c1f39fe"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:ac71b2977fb90c35d41c9453116e283fac47bb9096ad917b8819ca8b943abecd"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:3ae38d325b512f63f8da31f826e6cb6c367336f95e418137286ba362925c877e"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:542da1178c1c6af8873e143910e2269add130a299c9106eef2594e15dae5e482"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:30a85aed0b864ac88309b7d94be09f6046c834ef60762a8833b660139cfbad13"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:aae32c93e0f64469f74ccc730a7cb21c7610af3a775157e50bbd38f816536b38"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:15b26ddf78d57f1d143bdf32e820fd8935d36abe8a25eb9ec0b5a71c82eb3895"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7f5d10bae5d78e4551b7be7a9b29643a95aded9d0f602aa2ba584f0388e7a557"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:249c6470a2b60935bafd1d1d13cd613f8cd8388d53461c67397ee6a0f5dce741"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:c5a74c359b2d47d26cdbbc7845e9662d6b08a1e915eb015d044729e92e7050b7"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:b5bcf60a228acae568e9911f410f9d9e0d43197d030ae5799e20dca8df588287"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:187d18082694a29005ba2944c882344b6748d5be69e3a89bf3cc9d878e548d5a"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:81bf654678e575403736b85ba3a7867e31c2c30a69bc57fe88e3ace52fb17b89"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-win32.whl", hash = "sha256:85a32721ddde63c9df9ebb0d2045b9691d9750cb139c161c80e500d210f5e26e"}, + {file = "charset_normalizer-3.3.0-cp38-cp38-win_amd64.whl", hash = "sha256:468d2a840567b13a590e67dd276c570f8de00ed767ecc611994c301d0f8c014f"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:e0fc42822278451bc13a2e8626cf2218ba570f27856b536e00cfa53099724828"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:09c77f964f351a7369cc343911e0df63e762e42bac24cd7d18525961c81754f4"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:12ebea541c44fdc88ccb794a13fe861cc5e35d64ed689513a5c03d05b53b7c82"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:805dfea4ca10411a5296bcc75638017215a93ffb584c9e344731eef0dcfb026a"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:96c2b49eb6a72c0e4991d62406e365d87067ca14c1a729a870d22354e6f68115"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:aaf7b34c5bc56b38c931a54f7952f1ff0ae77a2e82496583b247f7c969eb1479"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:619d1c96099be5823db34fe89e2582b336b5b074a7f47f819d6b3a57ff7bdb86"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a0ac5e7015a5920cfce654c06618ec40c33e12801711da6b4258af59a8eff00a"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:93aa7eef6ee71c629b51ef873991d6911b906d7312c6e8e99790c0f33c576f89"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:7966951325782121e67c81299a031f4c115615e68046f79b85856b86ebffc4cd"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:02673e456dc5ab13659f85196c534dc596d4ef260e4d86e856c3b2773ce09843"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:c2af80fb58f0f24b3f3adcb9148e6203fa67dd3f61c4af146ecad033024dde43"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:153e7b6e724761741e0974fc4dcd406d35ba70b92bfe3fedcb497226c93b9da7"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-win32.whl", hash = "sha256:d47ecf253780c90ee181d4d871cd655a789da937454045b17b5798da9393901a"}, + {file = "charset_normalizer-3.3.0-cp39-cp39-win_amd64.whl", hash = "sha256:d97d85fa63f315a8bdaba2af9a6a686e0eceab77b3089af45133252618e70884"}, + {file = "charset_normalizer-3.3.0-py3-none-any.whl", hash = "sha256:e46cd37076971c1040fc8c41273a8b3e2c624ce4f2be3f5dfcb7a430c1d3acc2"}, ] [[package]] @@ -793,13 +808,13 @@ testing = ["pytest", "pytest-cov"] [[package]] name = "platformdirs" -version = "3.10.0" +version = "3.11.0" description = "A small Python package for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." optional = false python-versions = ">=3.7" files = [ - {file = "platformdirs-3.10.0-py3-none-any.whl", hash = "sha256:d7c24979f292f916dc9cbf8648319032f551ea8c49a4c9bf2fb556a02070ec1d"}, - {file = "platformdirs-3.10.0.tar.gz", hash = "sha256:b45696dab2d7cc691a3226759c0d3b00c47c8b6e293d96f6436f733303f77f6d"}, + {file = "platformdirs-3.11.0-py3-none-any.whl", hash = "sha256:e9d171d00af68be50e9202731309c4e658fd8bc76f55c11c7dd760d023bda68e"}, + {file = "platformdirs-3.11.0.tar.gz", hash = "sha256:cf8ee52a3afdb965072dcc652433e0c7e3e40cf5ea1477cd4b3b1d2eb75495b3"}, ] [package.dependencies] @@ -809,17 +824,6 @@ typing-extensions = {version = ">=4.7.1", markers = "python_version < \"3.8\""} docs = ["furo (>=2023.7.26)", "proselint (>=0.13)", "sphinx (>=7.1.1)", "sphinx-autodoc-typehints (>=1.24)"] test = ["appdirs (==1.4.4)", "covdefaults (>=2.3)", "pytest (>=7.4)", "pytest-cov (>=4.1)", "pytest-mock (>=3.11.1)"] -[[package]] -name = "pycodestyle" -version = "2.10.0" -description = "Python style guide checker" -optional = false -python-versions = ">=3.6" -files = [ - {file = "pycodestyle-2.10.0-py2.py3-none-any.whl", hash = "sha256:8a4eaf0d0495c7395bdab3589ac2db602797d76207242c17d470186815706610"}, - {file = "pycodestyle-2.10.0.tar.gz", hash = "sha256:347187bdb476329d98f695c213d7295a846d1152ff4fe9bacb8a9590b8ee7053"}, -] - [[package]] name = "pycparser" version = "2.21" @@ -849,17 +853,6 @@ snowballstemmer = ">=2.2.0" [package.extras] toml = ["tomli (>=1.2.3)"] -[[package]] -name = "pyflakes" -version = "2.4.0" -description = "passive checker of Python programs" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" -files = [ - {file = "pyflakes-2.4.0-py2.py3-none-any.whl", hash = "sha256:3bb3a3f256f4b7968c9c788781e4ff07dce46bdf12339dcda61053375426ee2e"}, - {file = "pyflakes-2.4.0.tar.gz", hash = "sha256:05a85c2872edf37a4ed30b0cce2f6093e1d0581f8c19d7393122da7e25b2b24c"}, -] - [[package]] name = "pygments" version = "2.16.1" @@ -874,32 +867,6 @@ files = [ [package.extras] plugins = ["importlib-metadata"] -[[package]] -name = "pylama" -version = "8.3.8" -description = "Code audit tool for python" -optional = false -python-versions = ">=3.7" -files = [ - {file = "pylama-8.3.8-py3-none-any.whl", hash = "sha256:aff89423f7de118713f638c7f937fa83a5873e3bdf06d413661d9cb8dc5f3a7b"}, - {file = "pylama-8.3.8.tar.gz", hash = "sha256:2dd852fe9312ea6012466cf17ff179668fc3d2716856fcfaaee8ce7876d83620"}, -] - -[package.dependencies] -mccabe = ">=0.6.1" -pycodestyle = ">=2.8.0" -pydocstyle = ">=6.1.1" -pyflakes = ">=2.4.0" - -[package.extras] -all = ["eradicate", "mypy", "pylint", "radon", "vulture"] -eradicate = ["eradicate"] -mypy = ["mypy"] -pylint = ["pylint"] -radon = ["radon"] -tests = ["eradicate (>=2.0.0)", "mypy", "pylama-quotes", "pylint (>=2.11.1)", "pytest", "pytest-mypy", "radon (>=5.1.0)", "types-setuptools", "vulture"] -vulture = ["vulture"] - [[package]] name = "pylint" version = "2.13.9" @@ -1085,13 +1052,13 @@ idna2008 = ["idna"] [[package]] name = "rich" -version = "13.5.2" +version = "13.6.0" description = "Render rich text, tables, progress bars, syntax highlighting, markdown and more to the terminal" optional = false python-versions = ">=3.7.0" files = [ - {file = "rich-13.5.2-py3-none-any.whl", hash = "sha256:146a90b3b6b47cac4a73c12866a499e9817426423f57c5a66949c086191a8808"}, - {file = "rich-13.5.2.tar.gz", hash = "sha256:fb9d6c0a0f643c99eed3875b5377a184132ba9be4d61516a55273d3554d75a39"}, + {file = "rich-13.6.0-py3-none-any.whl", hash = "sha256:2b38e2fe9ca72c9a00170a1a2d20c63c790d0e10ef1fe35eba76e1e7b1d7d245"}, + {file = "rich-13.6.0.tar.gz", hash = "sha256:5c14d22737e6d5084ef4771b62d5d4363165b403455a30a1c8ca39dc7b644bef"}, ] [package.dependencies] @@ -1270,13 +1237,13 @@ files = [ [[package]] name = "types-pyyaml" -version = "6.0.12.11" +version = "6.0.12.12" description = "Typing stubs for PyYAML" optional = false python-versions = "*" files = [ - {file = "types-PyYAML-6.0.12.11.tar.gz", hash = "sha256:7d340b19ca28cddfdba438ee638cd4084bde213e501a3978738543e27094775b"}, - {file = "types_PyYAML-6.0.12.11-py3-none-any.whl", hash = "sha256:a461508f3096d1d5810ec5ab95d7eeecb651f3a15b71959999988942063bf01d"}, + {file = "types-PyYAML-6.0.12.12.tar.gz", hash = "sha256:334373d392fde0fdf95af5c3f1661885fa10c52167b14593eb856289e1855062"}, + {file = "types_PyYAML-6.0.12.12-py3-none-any.whl", hash = "sha256:c05bc6c158facb0676674b7f11fe3960db4f389718e19e62bd2b84d6205cfd24"}, ] [[package]] @@ -1292,24 +1259,24 @@ files = [ [[package]] name = "unidecode" -version = "1.3.6" +version = "1.3.7" description = "ASCII transliterations of Unicode text" optional = false python-versions = ">=3.5" files = [ - {file = "Unidecode-1.3.6-py3-none-any.whl", hash = "sha256:547d7c479e4f377b430dd91ac1275d593308dce0fc464fb2ab7d41f82ec653be"}, - {file = "Unidecode-1.3.6.tar.gz", hash = "sha256:fed09cf0be8cf415b391642c2a5addfc72194407caee4f98719e40ec2a72b830"}, + {file = "Unidecode-1.3.7-py3-none-any.whl", hash = "sha256:663a537f506834ed836af26a81b210d90cbde044c47bfbdc0fbbc9f94c86a6e4"}, + {file = "Unidecode-1.3.7.tar.gz", hash = "sha256:3c90b4662aa0de0cb591884b934ead8d2225f1800d8da675a7750cbc3bd94610"}, ] [[package]] name = "urllib3" -version = "2.0.6" +version = "2.0.7" description = "HTTP library with thread-safe connection pooling, file post, and more." optional = false python-versions = ">=3.7" files = [ - {file = "urllib3-2.0.6-py3-none-any.whl", hash = "sha256:7a7c7003b000adf9e7ca2a377c9688bbc54ed41b985789ed576570342a375cd2"}, - {file = "urllib3-2.0.6.tar.gz", hash = "sha256:b19e1a85d206b56d7df1d5e683df4a7725252a964e3993648dd0fb5a1c157564"}, + {file = "urllib3-2.0.7-py3-none-any.whl", hash = "sha256:fdb6d215c776278489906c2f8916e6e7d4f5a9b602ccbcfdf7f016fc8da0596e"}, + {file = "urllib3-2.0.7.tar.gz", hash = "sha256:c97dfde1f7bd43a71c8d2a58e369e9b2bf692d1334ea9f9cae55add7d0dd0f84"}, ] [package.extras] @@ -1463,4 +1430,4 @@ testing = ["big-O", "flake8 (<5)", "jaraco.functools", "jaraco.itertools", "more [metadata] lock-version = "2.0" python-versions = "^3.7" -content-hash = "82b1e647fadeaab8eb363ed5d9cacf516d3439b402cb2915bc6ce75a786ac4ca" +content-hash = "df6b63e8c53471f8ecd665fb400706f8dff3838a2e13f0d2f2dc873a8d68de33" diff --git a/pylama.ini b/pylama.ini deleted file mode 100644 index 8a33d65..0000000 --- a/pylama.ini +++ /dev/null @@ -1,11 +0,0 @@ -[pylama] -format = pylint -ignore = E0401,W0221,W0222,R0914 -linters = pylint,pycodestyle -skip = */.tox/*,*/.env/* - -[pylama:pycodestyle] -max_line_length = 120 - -[pylama:pylint] -max_line_length = 120 diff --git a/pyproject.toml b/pyproject.toml index 5e4a8dd..7d3e709 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,6 +1,6 @@ [tool.poetry] name = "cert_manager" -version = "2.3.1" +version = "2.4.0" description = "Python interface to the Sectigo Certificate Manager REST API" license = "BSD-3-Clause" authors = ["Andrew Teixeira "] @@ -21,9 +21,6 @@ fixtures = "*" green = ">=2.12.0" mock = ">=2.0.0" pydocstyle = "<7" -# 2.5.0 breaks pylama 8.3.8 -pyflakes = "< 2.5.0" -pylama = "*" pylint = "*" pyyaml = "==6.0.1" responses = "*" @@ -31,3 +28,19 @@ testtools = "*" twine = "*" wheel = "*" yamllint = "*" + +[tool.ruff] +fixable = ["ALL"] +line-length = 120 +select = [ + "D", # pydocstyle + "E", # pycodestyle + "F", # pyflakes + "I", # isort + "N", # pep8-naming + "PL", # Pylint + "UP", # pyupgrade +] + +[tool.ruff.pydocstyle] +convention = "google" diff --git a/tests/__init__.py b/tests/__init__.py index 07191ec..fa8679d 100644 --- a/tests/__init__.py +++ b/tests/__init__.py @@ -1,2 +1 @@ -# -*- coding: utf-8 -*- """Initialize the test module.""" diff --git a/tests/lib/__init__.py b/tests/lib/__init__.py index baf3c7e..39c30df 100644 --- a/tests/lib/__init__.py +++ b/tests/lib/__init__.py @@ -1,2 +1 @@ -# -*- coding: utf-8 -*- """Initialize the testing library.""" diff --git a/tests/lib/testbase.py b/tests/lib/testbase.py index d9972c1..699b008 100644 --- a/tests/lib/testbase.py +++ b/tests/lib/testbase.py @@ -1,12 +1,11 @@ -# -*- coding: utf-8 -*- """Define some basic classes and functions for use in unit tests.""" import sys import fixtures -from cert_manager.client import Client from cert_manager import __version__ +from cert_manager.client import Client # pylint:disable=too-few-public-methods @@ -15,7 +14,7 @@ class ClientFixture(fixtures.Fixture): """Build a fixture for a default cert_manager.client.Client object.""" - def _setUp(self): # pylint: disable=invalid-name + def _setUp(self): # noqa: N802 """Setup the Client object and the values used to build the object.""" # Setup default testing values self.base_url = "https://certs.example.com/api" diff --git a/tests/test_acme.py b/tests/test_acme.py index 3eb233b..6a7d7ee 100644 --- a/tests/test_acme.py +++ b/tests/test_acme.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.acme.ACMEAccount unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=protected-access @@ -6,11 +5,10 @@ from functools import wraps +import responses from requests.exceptions import HTTPError from testtools import TestCase -import responses - from cert_manager.acme import ACMEAccount, ACMEAccountCreationResponseError from .lib.testbase import ClientFixture @@ -21,27 +19,29 @@ class TestACMEAccount(TestCase): # pylint: disable=too-few-public-methods @property def api_url(self): - """Return the base ACME Account URL for the default API version""" + """Return the base ACME Account URL for the default API version.""" return self.get_api_url() def get_api_url(self, api_version="v1"): - """Return the base ACME Account URL for a particular API version""" + """Return the base ACME Account URL for a particular API version.""" return f"{self.cfixt.base_url}/acme/{api_version}/account" def get_acme_account_url(self, acme_id, **kwargs): - """Return the ACME Account URL for the specified acme_id""" + """Return the ACME Account URL for the specified acme_id.""" return f"{self.get_api_url(**kwargs)}/{acme_id}" def get_valid_response_entry(self, acme_id): - """Return the first entry in valid_response with a matching acme_id""" + """Return the first entry in valid_response with a matching acme_id.""" for entry in self.valid_response: if entry["id"] == acme_id: return entry raise KeyError(f"id {acme_id} not found in valid_response") def get_acme_account_data(self, acme_id, domains=None): - """Return a matching entry from valid_response as ACME account data, - i.e. with SCM domains (empty by default)""" + """Return a matching entry from valid_response as ACME account data. + + i.e. with SCM domains (empty by default). + """ valid_response = self.get_valid_response_entry(acme_id).copy() valid_response.setdefault("domains", domains or []) return valid_response @@ -126,9 +126,10 @@ def setUp(self): # pylint: disable=invalid-name self.error_response = {"description": "acme error"} def match_url_with_qs(self, url, extra_params=None, api_url=None): - """Check that a URL containing a query string matches - the base self.api_url and the query params contain self.base_params - and extra_params + """Check that a URL matches completely. + + Check that a URL containing a query string matches the base self.api_url and the + query params contain self.base_params and extra_params. :param string url: The URL to check :param dict extra_params: The params the query string must contain, in addition to self.base_params @@ -148,7 +149,7 @@ class TestInit(TestACMEAccount): """Test the class initializer.""" def test_need_client(self): - """The class should raise an exception without a client parameter.""" + """Raise an exception if called without a client parameter.""" self.assertRaises(TypeError, ACMEAccount) @@ -157,8 +158,7 @@ class TestAll(TestACMEAccount): @responses.activate def test_init_param(self): - """The URL should change if api_version is passed as a parameter to - class initialization.""" + """Change the URL if api_version is passed as a parameter to class initialization.""" # Set a new version version = "v3" api_url = self.get_api_url(api_version=version) @@ -180,7 +180,7 @@ class initialization.""" @responses.activate def test_bad_http(self): - """The function should raise an HTTPError exception if acme accounts cannot be retrieved from the API.""" + """Raise an HTTPError exception if acme accounts cannot be retrieved from the API.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.error_response, status=404, match_querystring=False) @@ -194,7 +194,7 @@ def test_bad_http(self): @responses.activate def test_cached(self): - """The function should return all the data, but should not query the API twice.""" + """Return all the data, but should not query the API twice.""" # Setup the mocked response, refrain from matching the query string responses.add(responses.GET, self.api_url, json=self.valid_response, status=200, match_querystring=False) @@ -213,7 +213,7 @@ def test_cached(self): @responses.activate def test_forced(self): - """The function should return all the data, but should query the API twice.""" + """Return all the data, but should query the API twice.""" # Setup the mocked response, refrain from matching the query string responses.add(responses.GET, self.api_url, json=self.valid_response, status=200, match_querystring=False) @@ -232,7 +232,7 @@ def test_forced(self): self.assertEqual(data, self.valid_response) def test_need_org_id(self): - """The function should raise an exception without an org_id parameter.""" + """Raise an exception without an org_id parameter.""" acme = ACMEAccount(client=self.client) self.assertRaises(TypeError, acme.all) @@ -247,7 +247,7 @@ def _test_find_test_factory(params=None): @responses.activate def generic_test(self): - """Generic test for .find request parameters/response fields""" + """Generic test for .find request parameters/response fields.""" api_params[params_to_api["org_id"]] = str(self.org_id) valid_response = [ entry for entry in self.valid_response @@ -277,46 +277,38 @@ class TestFind(TestACMEAccount): """Test the .find method.""" test_name = _test_find_test_factory({"name": "api_account1"}) - test_name.__doc__ = """The function should return all the data about the - matched acme account name(s).""" + test_name.__doc__ = """Return all the data about the matched acme account name(s).""" test_acme_server = _test_find_test_factory( {"acme_server": "https://acme.sectigo.com/v2/OV"}) - test_acme_server.__doc__ = """The function should return all the data about - the matched acme account server(s).""" + test_acme_server.__doc__ = """Return all the data about the matched acme account server(s).""" test_cert_validation_type = _test_find_test_factory( {"cert_validation_type": "EV"}) - test_cert_validation_type.__doc__ = """The function should return all the - data about the matched acme account validation type(s).""" + test_cert_validation_type.__doc__ = """Return all the data about the matched acme account validation type(s).""" test_cert_status = _test_find_test_factory({"status": "Valid"}) - test_cert_status.__doc__ = """The function should return all the data about - the matched acme account status(es).""" + test_cert_status.__doc__ = """Return all the data about the matched acme account status(es).""" test_ne_server_and_cert_validation_type = _test_find_test_factory({ "acme_server": "https://acme.sectigo.com/v2/OV", "cert_validation_type": "EV" }) - test_ne_server_and_cert_validation_type.__doc__ = """The function should - return an empty list if the acme account server and vaildation type do not - match.""" + test_ne_server_and_cert_validation_type.__doc__ = """Return an empty list if the acme account server and vaildation + type do not match.""" test_ne_name = _test_find_test_factory({"name": "no such account"}) - test_ne_name.__doc__ = """The function should return an empty list if the - acme account name does not match.""" + test_ne_name.__doc__ = """Return an empty list if the acme account name does not match.""" test_ne_server = _test_find_test_factory( {"acme_server": "https://acme.sectigo.com/v2/XYZ"}) - test_ne_server.__doc__ = """The function should return an empty list if the - acme account server does not match.""" + test_ne_server.__doc__ = """Return an empty list if the acme account server does not match.""" test_no_params = _test_find_test_factory() - test_no_params.__doc__ = """The function should return the entire list of - acme accounts if no parameters are passed.""" + test_no_params.__doc__ = """Return the entire list of acme accounts if no parameters are passed.""" def test_need_org_id(self): - """The function should raise an exception without an org_id parameter.""" + """Raise an exception without an org_id parameter.""" acme = ACMEAccount(client=self.client) # We need to wrap this all crazy because it now returns an iterator result = acme.find() # pylint:disable=no-value-for-parameter @@ -327,13 +319,13 @@ class TestGet(TestACMEAccount): """Test the .get method.""" def test_need_acme_id(self): - """The function should raise an exception without an acme_id parameter.""" + """Raise an exception without an acme_id parameter.""" acme = ACMEAccount(client=self.client) self.assertRaises(TypeError, acme.get) @responses.activate def test_acme_id(self): - """The function should return all the data about the specified ACME ID.""" + """Return all the data about the specified ACME ID.""" acme_id = 1234 api_url = self.get_acme_account_url(acme_id) valid_response = self.get_acme_account_data(acme_id) @@ -350,7 +342,7 @@ def test_acme_id(self): @responses.activate def test_ne_acme_id(self): - """The function should raise an HTTPError exception if the specified ACME ID does not exist.""" + """Raise an HTTPError exception if the specified ACME ID does not exist.""" acme_id = 2345 api_url = self.get_acme_account_url(acme_id) @@ -363,6 +355,7 @@ def test_ne_acme_id(self): def _test_create_test_factory(acme_id=1234, header="location", **kwargs): + """Act as a wrapper to inject headers and parameters.""" params = ["name", "acmeServer", "organizationId", "evDetails"] def wrapper(func): @@ -392,11 +385,7 @@ class TestCreate(TestACMEAccount): """Test the .create method.""" def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" acme = ACMEAccount(client=self.client) # missing name, acme_server, org_id self.assertRaises(TypeError, acme.create) @@ -408,8 +397,7 @@ def test_need_params(self): @responses.activate @_test_create_test_factory() def test_create_success(self, acme_id, response_headers, args, request_params): - """The function should return the created ACME ID.""" - + """Return the created ACME ID.""" # Setup the mocked response responses.add(responses.POST, self.api_url, headers=response_headers, match=[responses.json_params_matcher(request_params)], @@ -423,11 +411,7 @@ def test_create_success(self, acme_id, response_headers, args, request_params): @responses.activate @_test_create_test_factory() def test_create_failure_http_error(self, _, __, args, request_params): - """ - The function should return an error code and description if the ACME - Account creation failed. - """ - + """Raise an exception if the ACME Account creation fails with an http error.""" # Setup the mocked response responses.add(responses.POST, self.api_url, json=self.error_response, match=[responses.json_params_matcher(request_params)], @@ -441,12 +425,7 @@ def test_create_failure_http_error(self, _, __, args, request_params): @_test_create_test_factory() def test_create_failure_http_status_unexpected(self, _, __, args, request_params): - """ - The function should return an error code and description if the ACME - Account creation failed with ACMEAccountCreationResponseError - (unexpected HTTP status code). - """ - + """Raise an exception if the ACME Account creation fails with unexpected http code.""" # Setup the mocked response responses.add(responses.POST, self.api_url, json=self.error_response, match=[responses.json_params_matcher(request_params)], @@ -461,12 +440,7 @@ def test_create_failure_http_status_unexpected(self, _, __, args, @_test_create_test_factory(header="NotYourHeader") def test_create_failure_missing_location_header(self, _, response_headers, args, request_params): - """ - The function should return an error code and description if the ACME - Account creation failed with ACMEAccountCreationResponseError - (no Location header in response). - """ - + """Raise an exception if the ACME Account creation fails due to no Location header in response.""" # Setup the mocked response responses.add(responses.POST, self.api_url, json=self.error_response, headers=response_headers, @@ -482,12 +456,7 @@ def test_create_failure_missing_location_header(self, _, response_headers, @_test_create_test_factory(location="not_an_ACME_account_URL") def test_create_failure_acme_id_not_found(self, _, response_headers, args, request_params): - """ - The function should return an error code and description if the ACME - Account creation failed with ACMEAccountCreationResponseError - (ACME ID not found in response). - """ - + """Raise an exception if the ACME Account creation fails because ACME ID not found in response.""" # Setup the mocked response responses.add(responses.POST, self.api_url, json=self.error_response, headers=response_headers, @@ -518,11 +487,7 @@ class TestUpdate(TestACMEAccount): """Test the .update method.""" def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" acme = ACMEAccount(client=self.client) # missing acme_id, name self.assertRaises(TypeError, acme.create) @@ -532,8 +497,7 @@ def test_need_params(self): @responses.activate @_test_update_delete_test_factory def test_update_success(self, acme_id, new_name): - """The function should return True if the update succeeded.""" - + """Return True if the update succeeded.""" api_url = self.get_acme_account_url(acme_id) # Setup the mocked response @@ -547,10 +511,7 @@ def test_update_success(self, acme_id, new_name): @responses.activate @_test_update_delete_test_factory def test_update_failure_http_error(self, acme_id, new_name): - """ - The function should raise an HTTPError exception if the update failed. - """ - + """Raise an HTTPError exception if the update failed.""" api_url = self.get_acme_account_url(acme_id) # Setup the mocked response @@ -565,11 +526,7 @@ class TestDelete(TestACMEAccount): """Test the .delete method.""" def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" acme = ACMEAccount(client=self.client) # missing acme_id self.assertRaises(TypeError, acme.delete) @@ -577,8 +534,7 @@ def test_need_params(self): @responses.activate @_test_update_delete_test_factory def test_delete_success(self, acme_id): - """The function should return True if the deletion succeeded.""" - + """Return True if the deletion succeeded.""" api_url = self.get_acme_account_url(acme_id) # Setup the mocked response @@ -592,11 +548,7 @@ def test_delete_success(self, acme_id): @responses.activate @_test_update_delete_test_factory def test_delete_failure_http_error(self, acme_id): - """ - The function should raise an HTTPError exception if the deletion - failed. - """ - + """Raise an HTTPError exception if the deletion failed.""" api_url = self.get_acme_account_url(acme_id) # Setup the mocked response @@ -608,6 +560,7 @@ def test_delete_failure_http_error(self, acme_id): def _test_add_remove_domains_test_factory(func): + """Act as a wrapper to inject domains.""" acme_id = 1234 @wraps(func) @@ -633,11 +586,7 @@ class TestAddDomains(TestACMEAccount): """Test the .add_domains method.""" def test_need_params(self): - """ - The function should raise an exception when called without required - parameters or domains argument is not a list - """ - + """Raise an exception when called without required parameters or domains argument is not a list.""" acme = ACMEAccount(client=self.client) # missing acme_id, domains self.assertRaises(TypeError, acme.add_domains) @@ -649,11 +598,7 @@ def test_need_params(self): @responses.activate @_test_add_remove_domains_test_factory def test_add_domains_success(self, acme_id, api_url, req_domains, resp): - """ - The function should return a dictionary containing a list of domains - not added. - """ - + """Return a dictionary containing a list of domains not added.""" # Setup the mocked response responses.add(responses.POST, api_url, json=resp, status=200) @@ -666,11 +611,7 @@ def test_add_domains_success(self, acme_id, api_url, req_domains, resp): @_test_add_remove_domains_test_factory def test_add_domains_failure_http_error(self, acme_id, api_url, req_domains, _): - """ - The function should raise an HTTPError exception if the domain addition - failed. - """ - + """Raise an HTTPError exception if the domain addition failed.""" # Setup the mocked response responses.add(responses.POST, api_url, status=400) @@ -683,11 +624,7 @@ class TestRemoveDomains(TestACMEAccount): """Test the .remove_domains method.""" def test_need_params(self): - """ - The function should raise an exception when called without required - parameters or domains argument is not a list - """ - + """Raise an exception when called without required parameters or domains argument is not a list.""" acme = ACMEAccount(client=self.client) # missing acme_id, domains self.assertRaises(TypeError, acme.remove_domains) @@ -699,11 +636,7 @@ def test_need_params(self): @responses.activate @_test_add_remove_domains_test_factory def test_remove_domains_success(self, acme_id, api_url, req_domains, resp): - """ - The function should return a dictionary containing a list of domains - not removed. - """ - + """Return a dictionary containing a list of domains not removed.""" # Setup the mocked response responses.add(responses.DELETE, api_url, json=resp, status=200) @@ -716,11 +649,7 @@ def test_remove_domains_success(self, acme_id, api_url, req_domains, resp): @_test_add_remove_domains_test_factory def test_remove_domains_failure_http_error(self, acme_id, api_url, req_domains, _): - """ - The function should raise an HTTPError exception if the domain removal - failed. - """ - + """Raise an HTTPError exception if the domain removal failed.""" # Setup the mocked response responses.add(responses.DELETE, api_url, status=400) diff --git a/tests/test_admin.py b/tests/test_admin.py index a17d881..ae36791 100644 --- a/tests/test_admin.py +++ b/tests/test_admin.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.admin.Admin unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=protected-access @@ -6,11 +5,10 @@ import json +import responses from requests.exceptions import HTTPError from testtools import TestCase -import responses - from cert_manager.admin import Admin, AdminCreationResponseError from .lib.testbase import ClientFixture @@ -79,7 +77,7 @@ def test_defaults(self): @responses.activate def test_param(self): - """The URL should change if api_version is passed as a parameter.""" + """Change the URL if api_version is passed as a parameter.""" # Set a new version version = "v3" api_url = f"{self.cfixt.base_url}/admin/{version}" @@ -96,12 +94,12 @@ def test_param(self): self.assertEqual(admin._Admin__admins, self.valid_response) def test_need_client(self): - """The class should raise an exception without a client parameter.""" + """Raise an exception if called without a client parameter.""" self.assertRaises(TypeError, Admin) @responses.activate def test_bad_http(self): - """The function should raise an HTTPError exception if admin accounts cannot be retrieved from the API.""" + """Raise an HTTPError exception if admin accounts cannot be retrieved from the API.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.error_response, status=400) @@ -117,7 +115,7 @@ class TestAll(TestAdmin): @responses.activate def test_cached(self): - """The function should return all the data, but should not query the API twice.""" + """Return all the data, but should not query the API twice.""" # Setup the mocked response, refrain from matching the query string responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -134,7 +132,7 @@ def test_cached(self): @responses.activate def test_forced(self): - """The function should return all the data, but should query the API twice.""" + """Return all the data, but should query the API twice.""" # Setup the mocked response, refrain from matching the query string responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -156,8 +154,7 @@ class TestGet(TestAdmin): @responses.activate def test_need_admin_id(self): - """The function should raise an exception without an admin_id parameter.""" - + """Raise an exception without an admin_id parameter.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) admin = Admin(client=self.client) @@ -165,8 +162,7 @@ def test_need_admin_id(self): @responses.activate def test_admin_id(self): - """The function should return data about the specified Admin ID.""" - + """Return data about the specified Admin ID.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -185,8 +181,7 @@ def test_admin_id(self): @responses.activate def test_ne_admin_id(self): - """The function should raise an HTTPError exception if the specified Admin ID does not exist.""" - + """Raise an HTTPError exception if the specified Admin ID does not exist.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -205,8 +200,7 @@ class TestGetIdps(TestAdmin): @responses.activate def test_get(self): - """The function should return all IDPs.""" - + """Return all IDPs.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -229,8 +223,7 @@ def test_get(self): @responses.activate def test_get_http_failure(self): - """The function should raise an HTTPError exception if IDPs cannot be retrieved from the API.""" - + """Raise an HTTPError exception if IDPs cannot be retrieved from the API.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -248,11 +241,7 @@ class TestCreate(TestAdmin): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -263,11 +252,7 @@ def test_need_params(self): @responses.activate def test_create_success(self): - """ - The function should return the created admin ID, - as well as add all parameters to the request body - """ - + """Return the created admin ID, as well as add all parameters to the request body.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) # Setup the mocked response @@ -291,11 +276,10 @@ def test_create_success(self): @responses.activate def test_create_success_optional_params(self): - """ - The function should return the created admin ID when additional params are specified, - as well add the non-required parameters to the request body - """ + """Return the created admin ID when additional params are specified. + Also, add the non-required parameters to the request body. + """ # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) # Setup the mocked response @@ -321,11 +305,7 @@ def test_create_success_optional_params(self): @responses.activate def test_create_failure_http_error(self): - """ - The function should return an error code and description if the Admin - creation failed. - """ - + """Return an error code and description if the Admin creation failed.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) # Setup the mocked response @@ -346,12 +326,7 @@ def test_create_failure_http_error(self): @responses.activate def test_create_failure_http_status_unexpected(self): - """ - The function should return an error code and description if the Admin - creation failed with AdminCreationResponseError - (unexpected HTTP status code). - """ - + """Raises an exception if the Admin creation fails with unexpected http code.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) # Setup the mocked response @@ -372,12 +347,7 @@ def test_create_failure_http_status_unexpected(self): @responses.activate def test_create_failure_missing_location_header(self): - """ - The function should return an error code and description if the Admin - creation failed with AdminCreationResponseError - (no Location header in response). - """ - + """Raises an exception if the Admin creation fails due to no Location header in response.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) # Setup the mocked response @@ -397,12 +367,7 @@ def test_create_failure_missing_location_header(self): @responses.activate def test_create_failure_admin_id_not_found(self): - """ - The function should return an error code and description if the Admin - creation failed with AdminCreationResponseError - (Admin ID not found in response). - """ - + """Raises an exception if the Admin creation fails because Admin ID not found in response.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) # Setup the mocked response @@ -426,11 +391,7 @@ class TestDelete(TestAdmin): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -440,8 +401,7 @@ def test_need_params(self): @responses.activate def test_delete_success(self): - """The function should return True if the deletion succeeded.""" - + """Return True if the deletion succeeded.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -458,11 +418,7 @@ def test_delete_success(self): @responses.activate def test_delete_failure_http_error(self): - """ - The function should raise an HTTPError exception if the deletion - failed. - """ - + """Raise an HTTPError exception if the deletion failed.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -482,11 +438,7 @@ class TestUpdate(TestAdmin): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -496,8 +448,7 @@ def test_need_params(self): @responses.activate def test_update_success(self): - """The function should return True if the update succeeded.""" - + """Return True if the update succeeded.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -514,8 +465,7 @@ def test_update_success(self): @responses.activate def test_update_body_success(self): - """Additional **kwargs should be added to request body""" - + """Additional **kwargs should be added to request body.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -538,11 +488,7 @@ def test_update_body_success(self): @responses.activate def test_update_failure_http_error(self): - """ - The function should return an error code and description if the Admin - creation failed. - """ - + """Return an error code and description if the Admin creation failed.""" # Setup the mocked response when class is initialized responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) diff --git a/tests/test_certificates.py b/tests/test_certificates.py index 31fede3..ab9f48c 100644 --- a/tests/test_certificates.py +++ b/tests/test_certificates.py @@ -1,16 +1,15 @@ -# -*- coding: utf-8 -*- """Define the cert_manager._certificates.Certificates unit tests.""" # responses is too tricky for pylint, so ignore the false-positive errors generated. # pylint: disable=no-member import json -from requests.exceptions import HTTPError import responses +from requests.exceptions import HTTPError from testtools import TestCase from cert_manager._certificates import Certificates -from cert_manager._helpers import Pending +from cert_manager._helpers import PendingError from .lib.testbase import ClientFixture @@ -38,7 +37,7 @@ def setUp(self): # pylint: disable=invalid-name @staticmethod def fake_csr(): - """Build a fake certificate signing request to use with tests""" + """Build a fake certificate signing request to use with tests.""" data = "-----BEGIN CERTIFICATE REQUEST-----\n" for row in range(1, 18): char = chr(row + 64) @@ -49,11 +48,12 @@ def fake_csr(): @staticmethod def fake_cert(): - """Build a fake certificate to use with tests""" + """Build a fake certificate to use with tests.""" + numrows = 27 data = "-----BEGIN CERTIFICATE-----\n" for row in range(1, 33): char = chr(row + 64) - if row > 26: + if row >= numrows: char = chr(row + 70) data += f"{char * 64}\n" data += "-----END CERTIFICATE-----\n" @@ -129,7 +129,7 @@ def setUp(self): @responses.activate def test_success(self): - """It should return data correctly if a 200-level status code is returned with data.""" + """Return data correctly if a 200-level status code is returned with data.""" # Setup the mocked response responses.add(responses.GET, self.test_url, json=self.types_data, status=200) @@ -159,7 +159,7 @@ def test_caching(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response responses.add(responses.GET, self.test_url, json={"description": "some error"}, status=404) @@ -186,7 +186,7 @@ def setUp(self): @responses.activate def test_success(self): - """It should return data correctly if a 200-level status code is returned with data.""" + """Return data correctly if a 200-level status code is returned with data.""" # Setup the mocked response responses.add(responses.GET, self.test_url, json=self.cf_data, status=200) @@ -200,7 +200,7 @@ def test_success(self): @responses.activate def test_empty(self): - """It should return an empty list if no custom fields were found.""" + """Return an empty list if no custom fields were found.""" # Setup the mocked response responses.add(responses.GET, self.test_url, json=[], status=200) @@ -214,7 +214,7 @@ def test_empty(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response responses.add(responses.GET, self.test_url, json={"description": "some error"}, status=404) @@ -241,7 +241,7 @@ def setUp(self): @responses.activate def test_success(self): - """It should return a certificate if a 200-level status code is returned with data.""" + """Return a certificate if a 200-level status code is returned with data.""" # Setup the mocked response responses.add(responses.GET, self.test_url, body=self.test_cert, status=200) @@ -255,12 +255,12 @@ def test_success(self): @responses.activate def test_pending(self): - """It should raise a Pending exception if an error status code is returned.""" + """Raise a PendingError exception if an error status code is returned.""" # Setup the mocked response responses.add(responses.GET, self.test_url, body="", status=404) # Call the function, expecting an exception - self.assertRaises(Pending, self.certobj.collect, self.test_id, self.test_type) + self.assertRaises(PendingError, self.certobj.collect, self.test_id, self.test_type) # Verify all the query information self.assertEqual(len(responses.calls), 1) @@ -268,27 +268,27 @@ def test_pending(self): @responses.activate def test_no_params(self): - """It should raise an Exception if no parameters are used.""" + """Raise an Exception if no parameters are used.""" # Call the function, expecting an exception self.assertRaises(Exception, self.certobj.collect) def test_no_type(self): - """It should raise an Exception if no cert_format is provided.""" + """Raise an Exception if no cert_format is provided.""" # Call the function, expecting an exception self.assertRaises(Exception, self.certobj.collect, self.test_id) def test_no_id(self): - """It should raise an Exception if no cert_id is provided.""" + """Raise an Exception if no cert_id is provided.""" # Call the function, expecting an exception self.assertRaises(Exception, self.certobj.collect, cert_format=self.test_type) def test_blank_params(self): - """It should raise an Exception if both parameters are blank.""" + """Raise an Exception if both parameters are blank.""" # Call the function, expecting an exception self.assertRaises(Exception, self.certobj.collect, cert_id=None, cert_format=None) def test_bad_type(self): - """It should raise an Exception if the cert_format is not recognized.""" + """Raise an Exception if the cert_format is not recognized.""" # Call the function, expecting an exception self.assertRaises(Exception, self.certobj.collect, cert_id=self.test_id, cert_format="x509OC") @@ -331,7 +331,7 @@ def setUp(self): @staticmethod def fake_csr(): - """Build a fake certificate signing request to use with tests""" + """Build a fake certificate signing request to use with tests.""" data = "-----BEGIN CERTIFICATE REQUEST-----\n" for row in range(1, 18): char = chr(row + 64) @@ -342,7 +342,7 @@ def fake_csr(): @responses.activate def test_success(self): - """It should return JSON if a 200-level status code is returned with data.""" + """Return JSON if a 200-level status code is returned with data.""" # Setup the mocked responses # We need to mock the /types and /customFields URLs as well # since Certificates.types and Certificate.custom_fields are called from enroll @@ -373,7 +373,7 @@ def test_success(self): @responses.activate def test_san_list(self): - """It should correctly handle a list of SANs.""" + """Handle a list of SANs correctly.""" # Setup the mocked responses # We need to mock the /types and /customFields URLs as well # since Certificates.types and Certificate.custom_fields are called from enroll @@ -407,7 +407,7 @@ def test_san_list(self): @responses.activate def test_bad_cert_name(self): - """It should raise an Exception if the cert_type_name was not found.""" + """Raise an Exception if the cert_type_name was not found.""" # Setup the mocked responses # We need to mock the /types and /customFields URLs as well # since Certificates.types and Certificate.custom_fields are called from enroll @@ -427,7 +427,7 @@ def test_bad_cert_name(self): @responses.activate def test_bad_term(self): - """It should raise an Exception if the term was not valid.""" + """Raise an Exception if the term was not valid.""" # Setup the mocked responses # since Certificates.types and Certificate.custom_fields are called from enroll responses.add(responses.GET, self.test_types_url, json=self.types_data, status=200) @@ -446,7 +446,7 @@ def test_bad_term(self): @responses.activate def test_mandatory_custom_fields_success(self): - """It should return a 200-level status code if a mandatory custom field is included.""" + """Return a 200-level status code if a mandatory custom field is included.""" # Setup the mocked responses # We need to mock the /types and /customFields URLs as well # since Certificates.types and Certificate.custom_fields are called from enroll @@ -479,7 +479,7 @@ def test_mandatory_custom_fields_success(self): @responses.activate def test_mandatory_custom_fields_missing(self): - """It should raise an Exception if mandatory custom fields are missing """ + """Raise an Exception if mandatory custom fields are missing.""" # Setup the mocked responses test_cf_missing_mandatory_field = [{"name": "testName2", "value": "testValue"}] @@ -503,7 +503,7 @@ def test_mandatory_custom_fields_missing(self): @responses.activate def test_custom_fields_duplicate_keys(self): - """It should raise an Exception if mandatory custom fields are missing """ + """Raise an Exception if mandatory custom fields are missing.""" # Setup the mocked responses test_cf_duplicate_fields = [ {"name": "testName", "value": "testValue"}, @@ -530,7 +530,7 @@ def test_custom_fields_duplicate_keys(self): @responses.activate def test_custom_fields_invalid(self): - """It should raise an Exception if elements of the custom_fields list are anything other than dicts """ + """Raise an Exception if elements of the custom_fields list are anything other than dicts.""" # Setup the mocked responses test_cf_invalid = ["I'm not a dict, I'm a string!"] @@ -554,7 +554,7 @@ def test_custom_fields_invalid(self): @responses.activate def test_custom_fields_keys_missing(self): - """It should raise an Exception if a dict in the custom fields list is missing keys """ + """Raise an Exception if a dict in the custom fields list is missing keys.""" # Setup the mocked responses test_cf_missing_keys = [{"name": "testName", "missingValue": True}] @@ -577,7 +577,7 @@ def test_custom_fields_keys_missing(self): @responses.activate def test_custom_fields_key_invalid(self): - """It should raise an Exception if a supplied custom field name doesn't exist """ + """Raise an Exception if a supplied custom field name doesn't exist.""" # Setup the mocked responses test_cf_invalid_name = [{"name": "someOtherName", "value": "testValue"}] @@ -611,7 +611,7 @@ def setUp(self): @responses.activate def test_success(self): - """It should return an empty dict if a 204 No Content response is returned.""" + """Return an empty dict if a 204 No Content response is returned.""" # Setup the mocked responses responses.add(responses.POST, self.test_url, body='', status=204) @@ -628,13 +628,13 @@ def test_success(self): @responses.activate def test_no_reason(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Call the function, expecting an exception self.assertRaises(Exception, self.certobj.revoke, self.test_id) @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response responses.add(responses.POST, self.test_url, json={}, status=404) @@ -666,7 +666,7 @@ def setUp(self): @responses.activate def test_success(self): - """It should return an empty dict if a 204 No Content response is returned.""" + """Return an empty dict if a 204 No Content response is returned.""" # Setup the mocked responses responses.add(responses.POST, self.test_url, body='', status=200) @@ -687,7 +687,7 @@ def test_success(self): @responses.activate def test_san_string(self): - """It should correctly handle a list of SANs.""" + """Handle a list of SANs correctly.""" # Setup the mocked responses responses.add(responses.POST, self.test_url, body='', status=200) @@ -710,7 +710,7 @@ def test_san_string(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked responses responses.add(responses.POST, self.test_url, json={}, status=404) diff --git a/tests/test_client.py b/tests/test_client.py index 60111c4..e255e2d 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.client.Client unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=protected-access @@ -9,12 +8,11 @@ # import sys +from unittest import mock -import mock -from testtools import TestCase - -from requests.exceptions import HTTPError import responses +from requests.exceptions import HTTPError +from testtools import TestCase from cert_manager.client import Client @@ -34,8 +32,7 @@ def setUp(self): # pylint: disable=invalid-name self.client = self.cfixt.client def tearDown(self): # pylint: disable=invalid-name - """Test tear down method""" - + """Test tear down method.""" super().tearDown() mock.patch.stopall() @@ -55,7 +52,7 @@ class TestInit(TestClient): """Test the class initializer.""" def test_defaults(self): - """Parameters should be set correctly inside the class using defaults.""" + """Set parameters correctly inside the class using defaults.""" client = Client(login_uri=self.cfixt.login_uri, username=self.cfixt.username, password=self.cfixt.password) # Use the hackity object mangling when dealing with double-underscore values in an object @@ -76,7 +73,7 @@ def test_defaults(self): self.assertEqual(self.cfixt.password, client._Client__session.headers["password"]) def test_params(self): - """Parameters should be set correctly inside the class using all parameters.""" + """Set parameters correctly inside the class using all parameters.""" client = Client( base_url=self.cfixt.base_url, login_uri=self.cfixt.login_uri, username=self.cfixt.username, password=self.cfixt.password, cert_auth=True, user_crt_file=self.cfixt.user_crt_file, @@ -101,7 +98,7 @@ def test_params(self): self.assertFalse("password" in client._Client__session.headers) def test_no_pass_with_certs(self): - """Parameters should be set correctly inside the class certificate auth without a password.""" + """Set parameters correctly inside the class certificate auth without a password.""" client = Client( base_url=self.cfixt.base_url, login_uri=self.cfixt.login_uri, username=self.cfixt.username, cert_auth=True, user_crt_file=self.cfixt.user_crt_file, user_key_file=self.cfixt.user_key_file, @@ -125,7 +122,7 @@ def test_no_pass_with_certs(self): self.assertFalse("password" in client._Client__session.headers) def test_versioning(self): - """The user-agent header should change if the version number changes.""" + """Change the user-agent header if the version number changes.""" test_version = "10.9.8" ver_info = list(map(str, sys.version_info)) pyver = ".".join(ver_info[:3]) @@ -141,36 +138,36 @@ def test_versioning(self): self.assertEqual(client._Client__session.headers["User-Agent"], user_agent) def test_need_crt(self): - """Class should raise an exception without a cert file if cert_auth=True.""" + """Raise an exception without a cert file if cert_auth=True.""" self.assertRaises(KeyError, Client, base_url=self.cfixt.base_url, login_uri=self.cfixt.login_uri, username=self.cfixt.username, cert_auth=True, user_key_file=self.cfixt.user_key_file) def test_need_key(self): - """Class should raise an exception without a key file if cert_auth=True.""" + """Raise an exception without a key file if cert_auth=True.""" self.assertRaises(KeyError, Client, base_url=self.cfixt.base_url, login_uri=self.cfixt.login_uri, username=self.cfixt.username, cert_auth=True, user_crt_file=self.cfixt.user_crt_file) def test_need_login_uri(self): - """Class should raise an exception without a login_uri.""" + """Raise an exception without a login_uri.""" self.assertRaises(KeyError, Client, username=self.cfixt.username, password=self.cfixt.password) def test_need_username(self): - """Class should raise an exception without a username.""" + """Raise an exception without a username.""" self.assertRaises(KeyError, Client, login_uri=self.cfixt.login_uri, password=self.cfixt.password) def test_need_password(self): - """Class should raise an exception without a password.""" + """Raise an exception without a password.""" self.assertRaises(KeyError, Client, login_uri=self.cfixt.login_uri, username=self.cfixt.username) def test_need_public_key(self): - """Class should raise an exception without a public key if cert_auth is enabled.""" + """Raise an exception without a public key if cert_auth is enabled.""" self.assertRaises( KeyError, Client, login_uri=self.cfixt.login_uri, username=self.cfixt.username, password=self.cfixt.password, cert_auth=True, user_key_file=self.cfixt.user_key_file, ) def test_need_private_key(self): - """Class should raise an exception without a private key if cert_auth is enabled.""" + """Raise an exception without a private key if cert_auth is enabled.""" self.assertRaises( KeyError, Client, login_uri=self.cfixt.login_uri, username=self.cfixt.username, password=self.cfixt.password, cert_auth=True, user_crt_file=self.cfixt.user_crt_file, @@ -237,7 +234,7 @@ def test_replace(self): self.assertEqual(self.client._Client__session.headers[head], headdata) def test_not_dictionary(self): - """It should raise an exception when not passed a dictionary.""" + """Raise an exception when not passed a dictionary.""" headers = ["User-Agent", "test123"] self.assertRaises(ValueError, self.client.add_headers, headers) @@ -246,7 +243,7 @@ class TestRemoveHeaders(TestClient): """Test the remove_headers method.""" def test_remove(self): - """The headers should be removed correctly if passed a list.""" + """Removed headers correctly if passed a list.""" headers = ["Accept", "customerUri"] self.client.remove_headers(headers) @@ -262,7 +259,7 @@ def test_remove(self): self.assertEqual(self.client._Client__session.headers[head], headdata) def test_dictionary(self): - """The headers should be removed correctly if passed a dictionary.""" + """Removed headers correctly if passed a dictionary.""" headers = {"Accept": "test123"} self.client.remove_headers(headers) @@ -291,7 +288,7 @@ def setUp(self): # pylint: disable=invalid-name @responses.activate def test_success(self): - """It should return data correctly if a 200-level status code is returned with data.""" + """Return data correctly if a 200-level status code is returned with data.""" # Setup the mocked response json_data = {"some": "data"} responses.add(responses.GET, self.test_url, json=json_data, status=200) @@ -306,7 +303,7 @@ def test_success(self): @responses.activate def test_headers(self): - """It should add passed headers.""" + """Add passed headers.""" # Setup the mocked response json_data = {"some": "data"} responses.add(responses.GET, self.test_url, json=json_data, status=200) @@ -326,7 +323,7 @@ def test_headers(self): @responses.activate def test_params(self): - """It should add passed parameters.""" + """Add passed parameters.""" # Setup the mocked response json_data = {"some": "data"} responses.add(responses.GET, self.test_url, json=json_data, status=200, @@ -350,7 +347,7 @@ def test_params(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response json_data = {"description": "some error"} responses.add(responses.GET, self.test_url, json=json_data, status=404) @@ -376,7 +373,7 @@ def setUp(self): # pylint: disable=invalid-name @responses.activate def test_success(self): - """It should return data correctly if a 200-level status code is returned with data.""" + """Return data correctly if a 200-level status code is returned with data.""" # Setup the mocked response input_data = {"input": "data"} output_data = {"output": "data"} @@ -392,7 +389,7 @@ def test_success(self): @responses.activate def test_headers(self): - """It should add passed headers.""" + """Add passed headers.""" # Setup the mocked response input_data = {"input": "data"} output_data = {"output": "data"} @@ -413,7 +410,7 @@ def test_headers(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response input_data = {"input": "data"} output_data = {"output": "data"} @@ -440,7 +437,7 @@ def setUp(self): # pylint: disable=invalid-name @responses.activate def test_success(self): - """It should return data correctly if a 200-level status code is returned with data.""" + """Return data correctly if a 200-level status code is returned with data.""" # Setup the mocked response input_data = {"input": "data"} output_data = {"output": "data"} @@ -456,7 +453,7 @@ def test_success(self): @responses.activate def test_headers(self): - """It should add passed headers.""" + """Add passed headers.""" # Setup the mocked response input_data = {"input": "data"} output_data = {"output": "data"} @@ -477,7 +474,7 @@ def test_headers(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response input_data = {"input": "data"} output_data = {"output": "data"} @@ -504,7 +501,7 @@ def setUp(self): # pylint: disable=invalid-name @responses.activate def test_success(self): - """It should complete correctly if a 200-level status code is returned.""" + """Return successful if a 200-level status code is returned.""" # Setup the mocked response input_data = {"input": "data"} responses.add(responses.DELETE, self.test_url, status=204) @@ -518,7 +515,7 @@ def test_success(self): @responses.activate def test_headers(self): - """It should add passed headers.""" + """Add passed headers.""" # Setup the mocked response input_data = {"input": "data"} responses.add(responses.DELETE, self.test_url, status=204) @@ -537,7 +534,7 @@ def test_headers(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if a non-200 status code is returned.""" + """Raise an HTTPError exception if a non-200 status code is returned.""" # Setup the mocked response input_data = {"input": "data"} responses.add(responses.DELETE, self.test_url, status=404) diff --git a/tests/test_domain.py b/tests/test_domain.py index 4c0ab34..1746339 100644 --- a/tests/test_domain.py +++ b/tests/test_domain.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.domain.Domain unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=protected-access @@ -6,11 +5,10 @@ import json +import responses from requests.exceptions import HTTPError from testtools import TestCase -import responses - from cert_manager.domain import Domain, DomainCreationResponseError from .lib.testbase import ClientFixture @@ -50,7 +48,7 @@ class TestInit(TestDomain): @responses.activate def test_param(self): - """The URL should change if api_version is passed as a parameter.""" + """Change the URL if api_version is passed as a parameter.""" # Set a new version version = "v3" api_url = f"{self.cfixt.base_url}/domain/{version}" @@ -68,7 +66,7 @@ def test_param(self): self.assertEqual(data, self.valid_response) def test_need_client(self): - """The class should raise an exception without a client parameter.""" + """Raise an exception when called without a client parameter.""" self.assertRaises(TypeError, Domain) @@ -77,7 +75,7 @@ class TestAll(TestDomain): @responses.activate def test_cached(self): - """The function should return all the data, but should not query the API twice.""" + """Return all the data, but it should not query the API twice.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -95,7 +93,7 @@ def test_cached(self): @responses.activate def test_forced(self): - """The function should return all the data, but should query the API twice.""" + """Return all the data, but it should query the API twice.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -114,7 +112,7 @@ def test_forced(self): @responses.activate def test_bad_http(self): - """The function should raise an HTTPError exception if domains cannot be retrieved from the API.""" + """Raises an exception if domains cannot be retrieved from the API.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.error_response, status=400) @@ -131,7 +129,7 @@ class TestFind(TestDomain): @responses.activate def test_no_params(self): - """Without parameters, the method will return all domains""" + """Return all domains when called without parameters.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -142,7 +140,7 @@ def test_no_params(self): @responses.activate def test_params(self): - """Parameters will be passed to API""" + """Parameters will be passed to API.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response[0], status=200) @@ -157,7 +155,7 @@ def test_params(self): @responses.activate def test_bad_http(self): - """The function should raise an HTTPError exception if domains cannot be retrieved from the API.""" + """Raises an exception if domains cannot be retrieved from the API.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.error_response, status=400) @@ -174,7 +172,7 @@ class TestCount(TestDomain): @responses.activate def test_no_params(self): - """Without parameters, the method will count all domains""" + """Return the count of all domains when called without parameters.""" # Setup the mocked response count = {"count": len(self.valid_response)} api_url = f"{self.api_url}/count" @@ -188,7 +186,7 @@ def test_no_params(self): @responses.activate def test_params(self): - """Parameters will be passed to API""" + """Parameters will be passed to API.""" # Setup the mocked response count = {"count": len(self.valid_response[0])} api_url = f"{self.api_url}/count" @@ -203,7 +201,7 @@ def test_params(self): @responses.activate def test_bad_http(self): - """The function should raise an HTTPError exception if counts cannot be retrieved from the API.""" + """Raise an HTTPError exception if counts cannot be retrieved from the API.""" # Setup the mocked response api_url = f"{self.api_url}/count" responses.add(responses.GET, api_url, json=self.error_response, status=400) @@ -221,15 +219,13 @@ class TestGet(TestDomain): @responses.activate def test_need_domain_id(self): - """The function should raise an exception without an domain_id parameter.""" - + """Raise an exception without an domain_id parameter.""" domain = Domain(client=self.client) self.assertRaises(TypeError, domain.get) @responses.activate def test_domain_id(self): - """The function should return data about the specified Domain ID.""" - + """Return data about the specified Domain ID.""" domain_id = 1234 api_url = f"{self.api_url}/{str(domain_id)}" @@ -245,8 +241,7 @@ def test_domain_id(self): @responses.activate def test_ne_domain_id(self): - """The function should raise an HTTPError exception if the specified Domain ID does not exist.""" - + """Raise an HTTPError exception if the specified Domain ID does not exist.""" domain_id = 2345 api_url = f"{self.api_url}/{str(domain_id)}" @@ -262,11 +257,7 @@ class TestCreate(TestDomain): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" domain = Domain(client=self.client) # Not going to check every permutation of missing parameters, # but verify that something is required @@ -274,11 +265,7 @@ def test_need_params(self): @responses.activate def test_create_success(self): - """ - The function should return the created domain ID, - as well as add all parameters to the request body - """ - + """Return the created domain ID, as well as add all parameters to the request body.""" # Setup the mocked response domain_id = 1234 org_id = 4321 @@ -298,11 +285,10 @@ def test_create_success(self): @responses.activate def test_create_success_optional_params(self): - """ - The function should return the created domain ID when additional params are specified, - as well add the non-required parameters to the request body - """ + """Return the created domain ID when additional params are specified. + Also, add the non-required parameters to the request body. + """ # Setup the mocked response domain_id = 1234 location = f"{self.api_url}/{str(domain_id)}" @@ -321,11 +307,7 @@ def test_create_success_optional_params(self): @responses.activate def test_create_failure_http_error(self): - """ - The function should return an error code and description if the Domain - creation failed. - """ - + """Return an error code and description if the Domain creation failed.""" # Setup the mocked response responses.add(responses.POST, self.api_url, json=self.error_response, status=400) @@ -341,12 +323,7 @@ def test_create_failure_http_error(self): @responses.activate def test_create_failure_http_status_unexpected(self): - """ - The function should return an error code and description if the Domain - creation failed with DomainCreationResponseError - (unexpected HTTP status code). - """ - + """Raise an exception if the Domain creation fails with unexpected http code.""" # Setup the mocked response responses.add(responses.POST, self.api_url, json=self.error_response, status=200) @@ -362,12 +339,7 @@ def test_create_failure_http_status_unexpected(self): @responses.activate def test_create_failure_missing_location_header(self): - """ - The function should return an error code and description if the Domain - creation failed with DomainCreationResponseError - (no Location header in response). - """ - + """Raise an exception if the Domain creation fails due to no Location header in response.""" # Setup the mocked response responses.add(responses.POST, self.api_url, status=201) @@ -382,12 +354,7 @@ def test_create_failure_missing_location_header(self): @responses.activate def test_create_failure_domain_id_not_found(self): - """ - The function should return an error code and description if the Domain - creation failed with DomainCreationResponseError - (Domain ID not found in response). - """ - + """Raise an exception if the Domain creation fails because Domain ID not found in response.""" # Setup the mocked response responses.add(responses.POST, self.api_url, headers={"Location": "not a url"}, status=201) @@ -406,19 +373,14 @@ class TestDelete(TestDomain): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" domain = Domain(client=self.client) # missing domain_id self.assertRaises(TypeError, domain.delete) @responses.activate def test_delete_success(self): - """The function should return True if the deletion succeeded.""" - + """Return True if the deletion succeeded.""" domain_id = 1234 api_url = f"{self.api_url}/{str(domain_id)}" @@ -432,11 +394,7 @@ def test_delete_success(self): @responses.activate def test_delete_failure_http_error(self): - """ - The function should raise an HTTPError exception if the deletion - failed. - """ - + """Raise an HTTPError exception if the deletion failed.""" domain_id = 1234 api_url = f"{self.api_url}/{str(domain_id)}" @@ -453,19 +411,14 @@ class TestActivate(TestDomain): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" domain = Domain(client=self.client) # missing domain_id self.assertRaises(TypeError, domain.activate) @responses.activate def test_activate_success(self): - """The function should return True if the activation succeeded.""" - + """Return True if the activation succeeded.""" domain_id = 1234 api_url = f"{self.api_url}/{str(domain_id)}/activate" @@ -479,11 +432,7 @@ def test_activate_success(self): @responses.activate def test_activate_failure_http_error(self): - """ - The function should raise an HTTPError exception if the deletion - failed. - """ - + """Raise an HTTPError exception if the deletion failed.""" domain_id = 1234 api_url = f"{self.api_url}/{str(domain_id)}/activate" @@ -500,19 +449,14 @@ class TestSuspend(TestDomain): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" domain = Domain(client=self.client) # missing domain_id self.assertRaises(TypeError, domain.suspend) @responses.activate def test_suspend_success(self): - """The function should return True if the suspension succeeded.""" - + """Return True if the suspension succeeded.""" domain_id = 1234 api_url = f"{self.api_url}/{str(domain_id)}/suspend" @@ -526,11 +470,7 @@ def test_suspend_success(self): @responses.activate def test_suspend_failure_http_error(self): - """ - The function should raise an HTTPError exception if the suspension - failed. - """ - + """Raise an HTTPError exception if the suspension failed.""" domain_id = 1234 api_url = f"{self.api_url}/{str(domain_id)}/suspend" @@ -547,19 +487,14 @@ class TestDelegate(TestDomain): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" domain = Domain(client=self.client) # missing domain_id self.assertRaises(TypeError, domain.delegate) @responses.activate def test_delegate_success(self): - """The function should return True if the delegation succeeded.""" - + """Return True if the delegation succeeded.""" domain_id = 1234 org_id = 4321 types = ["SSL"] @@ -580,8 +515,7 @@ def test_delegate_success(self): @responses.activate def test_delegate_failure_http_error(self): - """The function should raise an HTTPError exception if the delegation failed.""" - + """Raise an HTTPError exception if the delegation failed.""" domain_id = 1234 org_id = 4321 types = ["SSL"] @@ -600,19 +534,14 @@ class TestRemoveDelegation(TestDomain): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" domain = Domain(client=self.client) # missing domain_id self.assertRaises(TypeError, domain.remove_delegation) @responses.activate def test_remove_delegation_success(self): - """The function should return True if the delegation removal succeeded.""" - + """Return True if the delegation removal succeeded.""" domain_id = 1234 org_id = 4321 types = ["SSL"] @@ -633,8 +562,7 @@ def test_remove_delegation_success(self): @responses.activate def test_remove_delegation_failure_http_error(self): - """The function should raise an HTTPError exception if the delegation removal failed.""" - + """Raise an HTTPError exception if the delegation removal failed.""" domain_id = 1234 org_id = 4321 types = ["SSL"] @@ -653,19 +581,14 @@ class TestApproveDelegation(TestDomain): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" domain = Domain(client=self.client) # missing domain_id self.assertRaises(TypeError, domain.approve_delegation) @responses.activate def test_approve_delegation_success(self): - """The function should return True if the approval succeeded.""" - + """Return True if the approval succeeded.""" domain_id = 1234 org_id = 4321 api_url = f"{self.api_url}/{str(domain_id)}/delegation/approve" @@ -684,8 +607,7 @@ def test_approve_delegation_success(self): @responses.activate def test_approval_failure_http_error(self): - """The function should raise an HTTPError exception if the approval failed.""" - + """Raise an HTTPError exception if the approval failed.""" domain_id = 1234 org_id = 4321 api_url = f"{self.api_url}/{str(domain_id)}/delegation/approve" @@ -703,19 +625,14 @@ class TestRejectDelegation(TestDomain): @responses.activate def test_need_params(self): - """ - The function should raise an exception when called without required - parameters. - """ - + """Raise an exception when called without required parameters.""" domain = Domain(client=self.client) # missing domain_id self.assertRaises(TypeError, domain.reject_delegation) @responses.activate def test_reject_delegation_success(self): - """The function should return True if the rejection succeeded.""" - + """Return True if the rejection succeeded.""" domain_id = 1234 org_id = 4321 api_url = f"{self.api_url}/{str(domain_id)}/delegation/reject" @@ -734,8 +651,7 @@ def test_reject_delegation_success(self): @responses.activate def test_reject_failure_http_error(self): - """The function should raise an HTTPError exception if the rejection failed.""" - + """Raise an HTTPError exception if the rejection failed.""" domain_id = 1234 org_id = 4321 api_url = f"{self.api_url}/{str(domain_id)}/delegation/reject" diff --git a/tests/test_endpoint.py b/tests/test_endpoint.py index e37c05e..6c43347 100644 --- a/tests/test_endpoint.py +++ b/tests/test_endpoint.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager._endpoint.Endpoint unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=protected-access @@ -127,7 +126,7 @@ def test_many_slashes(self): self.assertEqual(end._url(suffix), url) def test_multiple_args(self): - """Return the full API URL when multiple arguments are passed""" + """Return the full API URL when multiple arguments are passed.""" end = Endpoint(client=self.client, endpoint=self.ep_path) suffix1 = "help" diff --git a/tests/test_helpers.py b/tests/test_helpers.py index 2ee0605..df868dd 100644 --- a/tests/test_helpers.py +++ b/tests/test_helpers.py @@ -1,18 +1,15 @@ -# -*- coding: utf-8 -*- """Define the cert_manager._helpers.traffic_log wrapper function unit tests.""" # Because pylint can't figure out how requests does some magic things # pylint: disable=no-member import json import logging -import sys import types - -import mock -from testtools import TestCase +from unittest import mock import requests import responses +from testtools import TestCase from cert_manager._helpers import paginate, traffic_log @@ -20,7 +17,7 @@ class TestPaginate(TestCase): """Tests for the cert_manager._helpers.paginate wrapper function.""" - def setUp(self): # pylint: disable=invalid-name + def setUp(self): # noqa: N802 """Initialize the class.""" # Call the inherited setUp method super().setUp() @@ -63,7 +60,7 @@ def fake_paging(self, *args, **kwargs): # pylint: disable=unused-argument return [] def test_correct(self): - """The inner function should be called with the correct parameters.""" + """Call the inner function with the correct parameters.""" data = [] # Call the test function @@ -77,7 +74,7 @@ def test_correct(self): self.assertEqual(data, self.test_data) def test_paging(self): - """The inner function should be called with the correct parameters the correct number of times.""" + """Call the inner function with the correct parameters the correct number of times.""" data = [] # Call the test function @@ -113,10 +110,7 @@ def setUp(self): # pylint: disable=invalid-name # Reset the mock on every run since it lives at the root of the class self.mock_logger.reset_mock() - # This is special because of the difference in strings between Python 2 and 3 self.res_headers = "{'Content-Type': 'application/json'}" - if sys.version_info[0] < 3: - self.res_headers = "{u'Content-Type': u'application/json'}" @responses.activate @traffic_log(traffic_logger=mock_logger) @@ -196,7 +190,7 @@ def test_correct_positional(self): self.mock_logger.debug.assert_any_call(f"Text result: {json.dumps(self.test_json_resp)}") def test_inner_exception(self): - """An exception should be raised by the wrapper if an exception is raised by the wrapped function.""" + """Wrapper raises an exception if an exception is raised by the wrapped function.""" err_msg = "this is an error" self.exc = Exception(err_msg) @@ -204,7 +198,7 @@ def test_inner_exception(self): self.assertRaisesRegex(Exception, err_msg, self.wrapped_function, url=self.test_url) def test_inner_http_error(self): - """An exception should be raised by wrapper if an HTTPError is raised by the wrapped function.""" + """Wrapper raises an exception if an HTTPError is raised by the wrapped function.""" err_msg = "404 Client Error: Not Found for url: http://example.com/api" # Make sure the proper exception is raised @@ -214,7 +208,7 @@ def test_inner_http_error(self): self.mock_logger.debug.assert_any_call(f"Text result: {json.dumps(self.test_json_resp)}") def test_bad_param_exception(self): - """An exception should be raised by wrapper if no logging instance is provided.""" + """Wrapper raises an exception if no logging instance is provided.""" err_msg = "traffic_log: No logging.Logger instance provided" # Make sure the proper exception is raised diff --git a/tests/test_organization.py b/tests/test_organization.py index 591a4af..d6dbc71 100644 --- a/tests/test_organization.py +++ b/tests/test_organization.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.organization.Organization unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=protected-access @@ -8,8 +7,8 @@ # https://stackoverflow.com/questions/9323749/python-check-if-one-dictionary-is-a-subset-of-another-larger-dictionary # -from requests.exceptions import HTTPError import responses +from requests.exceptions import HTTPError from testtools import TestCase from cert_manager.organization import Organization @@ -62,7 +61,7 @@ def test_defaults(self): @responses.activate def test_param(self): - """The URL should change if api_version is passed as a parameter.""" + """Change the URL if api_version is passed as a parameter.""" # Set a new version version = "v3" api_url = f"{self.cfixt.base_url}/organization/{version}" @@ -79,12 +78,12 @@ def test_param(self): self.assertEqual(org._Organization__orgs, self.valid_response) def test_need_client(self): - """The class should raise an exception without a client parameter.""" + """Raise an exception if called without a client parameter.""" self.assertRaises(TypeError, Organization) @responses.activate def test_bad_http(self): - """The class should raise an HTTPError exception if organizations cannot be retrieved from the API.""" + """Raise an HTTPError exception if organizations cannot be retrieved from the API.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.error_response, status=404) @@ -100,7 +99,7 @@ class TestAll(TestOrganization): @responses.activate def test_cached(self): - """The function should return all the data, but should not query the API twice.""" + """Return all the data, but should not query the API twice.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -115,7 +114,7 @@ def test_cached(self): @responses.activate def test_forced(self): - """The function should return all the data, but should query the API twice.""" + """Return all the data, but should query the API twice.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -135,7 +134,7 @@ class TestFind(TestOrganization): @responses.activate def test_org(self): - """The function should return all the data about the specified organization name.""" + """Return all the data about the specified organization name.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -150,7 +149,7 @@ def test_org(self): @responses.activate def test_dept(self): - """The function should return all the data about the specified department name.""" + """Return all the data about the specified department name.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -165,7 +164,7 @@ def test_dept(self): @responses.activate def test_org_and_dept(self): - """The function should return department data for department under a specific org.""" + """Return department data for department under a specific org.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -180,7 +179,7 @@ def test_org_and_dept(self): @responses.activate def test_ne_org_and_dept(self): - """The function should return an empty list if the org name doesn't exist but the department does.""" + """Return an empty list if the org name doesn't exist but the department does.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -195,7 +194,7 @@ def test_ne_org_and_dept(self): @responses.activate def test_ne_org(self): - """The function should return an empty list if the organization name doesn't exist.""" + """Return an empty list if the organization name doesn't exist.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -210,7 +209,7 @@ def test_ne_org(self): @responses.activate def test_ne_dept(self): - """The function should return an empty list if the department name doesn't exist.""" + """Return an empty list if the department name doesn't exist.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -225,7 +224,7 @@ def test_ne_dept(self): @responses.activate def test_no_params(self): - """The function should return the entire list of orgs if no parameters are passed.""" + """Return the entire list of orgs if no parameters are passed.""" # Setup the mocked response responses.add(responses.GET, self.api_url, json=self.valid_response, status=200) @@ -240,7 +239,7 @@ def test_no_params(self): @responses.activate def test_no_org_depts(self): - """The function should not error out if the departments field doesn't appear in any orgs.""" + """Do not return an error if the departments field doesn't appear in any orgs.""" org_data = [ {"id": 1234, "name": "Some Organization", "certTypes": []}, {"id": 4321, "name": "Another Organization", "certTypes": []}, diff --git a/tests/test_person.py b/tests/test_person.py index a572843..5355a8f 100644 --- a/tests/test_person.py +++ b/tests/test_person.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.person.Person unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=no-member @@ -6,9 +5,9 @@ import json from urllib.parse import quote, unquote -from testtools import TestCase -from requests.exceptions import HTTPError import responses +from requests.exceptions import HTTPError +from testtools import TestCase from cert_manager.person import Person @@ -43,11 +42,11 @@ def test_defaults(self): class TestList(TestPerson): - """Test the list method""" + """Test the list method.""" @responses.activate def test_list(self): - """The function should return all persons""" + """Return all persons.""" # Setup the mocked response test_url = f"{self.api_url}?size=200&position=0" test_result = [ @@ -73,7 +72,7 @@ class TestFind(TestPerson): @responses.activate def test_email(self): - """The function should return the Person ID corresponding to the email passed.""" + """Return the Person ID corresponding to the email passed.""" # Setup the mocked response test_email = "test@example.com" quoted_email = quote(test_email) @@ -92,7 +91,7 @@ def test_email(self): @responses.activate def test_not_found(self): - """The function should raise an HTTPError exception if a person by that email is not found.""" + """Raise an HTTPError exception if a person by that email is not found.""" # Setup the mocked response test_email = "test@example.com" quoted_email = quote(test_email) @@ -110,11 +109,11 @@ def test_not_found(self): class TestGet(TestPerson): - """Test the get details method""" + """Test the get details method.""" @responses.activate def test_id(self): - """The function should return the details of the Person""" + """Return the details of the Person.""" # Setup the mocked response test_id = 1234 test_url = f"{self.api_url}/{test_id}" @@ -131,7 +130,7 @@ def test_id(self): @responses.activate def test_not_found(self): - """The function should raise an HTTPError exception if a person with an ID is not found.""" + """Raise an HTTPError exception if a person with an ID is not found.""" # Setup the mocked response test_id = 1234 test_url = f"{self.api_url}/{test_id}" @@ -147,7 +146,7 @@ def test_not_found(self): class TestCreate(TestPerson): - """Test the create method""" + """Test the create method.""" def setUp(self): """Initialize the class.""" @@ -163,7 +162,7 @@ def setUp(self): @responses.activate def test_create_success(self): - """The function should return a JSON with the `personID` of the person created if a 201 response is given""" + """Return a JSON with the `personID` of the person created if a 201 response is given.""" responses.add( responses.POST, self.test_url, @@ -187,11 +186,11 @@ def test_create_success(self): class TestUpdate(TestPerson): - """Test the update method""" + """Test the update method.""" @responses.activate def test_update_success(self): - """The function should return nothing when updating a person""" + """Return nothing when updating a person.""" test_id = 1234 test_url = f"{self.api_url}/{test_id}" responses.add(responses.PUT, test_url, status=200) @@ -208,11 +207,11 @@ def test_update_success(self): class TestDelete(TestPerson): - """Test the delete method""" + """Test the delete method.""" @responses.activate def test_delete_success(self): - """The function should return nothing when deleting a person""" + """Return nothing when deleting a person.""" test_id = 1234 test_url = f"{self.api_url}/{test_id}" responses.add(responses.DELETE, test_url, status=200) diff --git a/tests/test_report.py b/tests/test_report.py index a61032c..5b617d9 100644 --- a/tests/test_report.py +++ b/tests/test_report.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.report.Report unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=protected-access @@ -6,18 +5,18 @@ import json -from testtools import TestCase - import responses +from testtools import TestCase from cert_manager.report import Report + from .lib.testbase import ClientFixture # pylint: disable=too-few-public-methods # pylint: disable=too-many-instance-attributes class TestReport(TestCase): - """Serve as a Base class for all tests of the Domain class.""" + """Serve as a Base class for all tests of the Report class.""" def setUp(self): # pylint: disable=invalid-name """Initialize the class.""" @@ -169,7 +168,7 @@ class TestInit(TestReport): @responses.activate def test_param(self): - """The URL should change if api_version is passed as a parameter.""" + """Change the URL if api_version is passed as a parameter.""" # Set a new version version = "v3" api_url = f"{self.cfixt.base_url}/report/{version}/activity" @@ -186,7 +185,7 @@ def test_param(self): self.assertEqual(data, self.valid_activity_report_response) def test_need_client(self): - """The class should raise an exception without a client parameter.""" + """Raise an exception if called without a client parameter.""" self.assertRaises(TypeError, Report) @@ -195,17 +194,13 @@ class TestGet(TestReport): @responses.activate def test_need_report_name(self): - """The function should raise an exception without an report_name parameter.""" - + """Raise an exception without an report_name parameter.""" report = Report(client=self.client) self.assertRaises(TypeError, report.get) @responses.activate def test_bad_http(self): - """ - The function should raise an HTTPError exception, - if domains cannot be retrieved from the API. - """ + """Raise an HTTPError exception, if reports cannot be retrieved from the API.""" # Setup the mocked response api_url = f"{self.api_url}/ssl-cert" @@ -223,8 +218,7 @@ def test_bad_http(self): @responses.activate def test_report_parm_ssl_cert(self): - """The function should return data for the chosen report.""" - + """Return data for the chosen report.""" api_url = f"{self.api_url}/ssl-certificates" # Setup the mocked response @@ -240,8 +234,7 @@ def test_report_parm_ssl_cert(self): @responses.activate def test_report_parm_ssl_cert_filter(self): - """The function should return data for chosen report with organization filter.""" - + """Return data for chosen report with organization filter.""" api_url = f"{self.api_url}/ssl-certificates" organization_array = ['51', '52'] filter_data = {"organizationIds": organization_array} @@ -270,8 +263,7 @@ class TestGetSSLCert(TestReport): @responses.activate def test_report_ssl_cert(self): - """The function should return data for the SSL Cert report.""" - + """Return data for the SSL Cert report.""" api_url = f"{self.api_url}/ssl-certificates" # Setup the mocked response @@ -286,8 +278,7 @@ def test_report_ssl_cert(self): @responses.activate def test_report_ssl_cert_filter(self): - """The function should return data for the SSL Cert report.""" - + """Return data for the SSL Cert report.""" api_url = f"{self.api_url}/ssl-certificates" organization_array = ['51', '52'] filter_data = {"organizationIds": organization_array} @@ -315,8 +306,7 @@ class TestGetActivity(TestReport): @responses.activate def test_report_activity(self): - """The function should return data for the Activity report.""" - + """Return data for the Activity report.""" api_url = f"{self.api_url}/activity" # Setup the mocked response @@ -331,8 +321,7 @@ def test_report_activity(self): @responses.activate def test_report_activity_filter(self): - """The function should return data for the Activity report with search filter.""" - + """Return data for the Activity report with search filter.""" api_url = f"{self.api_url}/activity" filter_data = {"from": "2022-03-07T00:00:00.000Z", "to": "2022-03-16T00:00:00.000Z"} @@ -359,8 +348,7 @@ class TestGetDomains(TestReport): @responses.activate def test_report_domains(self): - """The function should return data for the Domains report.""" - + """Return data for the Domains report.""" api_url = f"{self.api_url}/domains" # Setup the mocked response @@ -379,8 +367,7 @@ class TestGetClientCert(TestReport): @responses.activate def test_report_client_certs_filter(self): - """The function should return data for the Client Certs report with search filter.""" - + """Return data for the Client Certs report with search filter.""" api_url = f"{self.api_url}/client-certificates" filter_data = { "from": "2022-03-07T00:00:00.000Z", @@ -412,8 +399,7 @@ class TestGetDeviceCert(TestReport): @responses.activate def test_report_device_certs_filter(self): - """The function should return data for the Device Certs report with search filter.""" - + """Return data for the Device Certs report with search filter.""" api_url = f"{self.api_url}/device-certificates" filter_data = {"certificateStatus": 2} diff --git a/tests/test_smime.py b/tests/test_smime.py index 7df44ac..a06b5e0 100644 --- a/tests/test_smime.py +++ b/tests/test_smime.py @@ -1,16 +1,16 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.ssl.SMIME unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=protected-access # pylint: disable=invalid-name import json + import responses from requests import HTTPError from testtools import TestCase +from cert_manager._helpers import PendingError, RevokedError from cert_manager.smime import SMIME -from cert_manager._helpers import Pending, Revoked from .lib.testbase import ClientFixture from .test_certificates import TestCertificates @@ -65,7 +65,7 @@ class TestList(TestSMIME): @responses.activate def test_defaults(self): - """The function should return all client certificate records""" + """Return all client certificate records.""" # Setup the mocked response test_url = f"{self.api_url}?size=200&position=0" test_result = [ @@ -105,13 +105,13 @@ def setUp(self): @responses.activate def test_defaults(self): - """The function should raise an exception when no email is passed""" + """Raise an exception when no email is passed.""" smime = SMIME(client=self.client) self.assertRaises(Exception, smime.list_by_email) @responses.activate def test_success(self): - """The function should return the list of certificates of an email""" + """Return the list of certificates of an email.""" # Setup the mocked response responses.add(responses.GET, self.test_url, json=self.test_result, status=200) @@ -128,7 +128,7 @@ def test_success(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response responses.add(responses.GET, self.test_url, json={}, status=404) @@ -184,13 +184,13 @@ def setUp(self): self.test_result = json.dumps({"orderNumber": 123456, "backendCertId": "123456"}) def test_defaults(self): - """The function should raise an exception when no params are passed""" + """Raise an exception when no params are passed.""" smime = SMIME(client=self.client) self.assertRaises(Exception, smime.enroll) @responses.activate def test_success(self): - """It should return a JSON if a 200-level status code is returned with data.""" + """Return a JSON if a 200-level status code is returned with data.""" # Setup the mocked response # We need to mock the /types and /customFields URLs as well # since SMIME.types and SMIME.custom_fields are called from enroll @@ -215,7 +215,7 @@ def test_success(self): @responses.activate def test_bad_cert_type_name(self): - """It should raise an Exception if the cert_type_name was not found.""" + """Raise an Exception if the cert_type_name was not found.""" # Setup the mocked responses # We need to mock the /types and /customFields URLs as well # since Certificates.types and Certificate.custom_fields are called from enroll @@ -238,7 +238,7 @@ def test_bad_cert_type_name(self): @responses.activate def test_bad_term(self): - """It should raise an Exception if the term was not valid.""" + """Raise an Exception if the term was not valid.""" # Setup the mocked responses # since Certificates.types and Certificate.custom_fields are called from enroll responses.add(responses.GET, self.test_types_url, json=self.types_data, status=200) @@ -259,7 +259,7 @@ def test_bad_term(self): @responses.activate def test_mandatory_custom_fields_success(self): - """It should return a 200-level status code if a mandatory custom field is included.""" + """Return a 200-level status code if a mandatory custom field is included.""" # Setup the mocked responses # We need to mock the /types and /customFields URLs as well # since Certificates.types and Certificate.custom_fields are called from enroll @@ -297,13 +297,13 @@ def setUp(self): self.test_cert = TestCertificates.fake_cert() def test_defaults(self): - """The function should raise an exception when no certificate id is passed""" + """Raise an exception when no certificate id is passed.""" smime = SMIME(client=self.client) self.assertRaises(Exception, smime.collect) @responses.activate def test_success(self): - """It should return a certificate if a 200-level status code is returned with data.""" + """Return a certificate if a 200-level status code is returned with data.""" # Setup the mocked response responses.add(responses.GET, self.test_url, body=self.test_cert, status=200) @@ -317,20 +317,20 @@ def test_success(self): self.assertEqual(responses.calls[0].request.url, self.test_url) def test_no_cert_id(self): - """The function should raise a ValueError exception if no cert_id is passed""" + """Raise a ValueError exception if no cert_id is passed.""" smime = SMIME(client=self.client) self.assertRaises(ValueError, smime.collect, None) @responses.activate def test_pending(self): - """It should raise a Pending exception if an Http error with the pending code in the body is returned.""" + """Raise a PendingError exception if an Http error with the pending code in the body is returned.""" # Setup the mocked response - body = json.dumps({"code": Pending.CODE, "description": "Certificate is not collectable."}) + body = json.dumps({"code": PendingError.CODE, "description": "Certificate is not collectable."}) responses.add(responses.GET, self.test_url, body=body, status=400) # Call the function, expecting an exception smime = SMIME(client=self.client) - self.assertRaises(Pending, smime.collect, self.test_id) + self.assertRaises(PendingError, smime.collect, self.test_id) # Verify all the query information self.assertEqual(len(responses.calls), 1) @@ -338,14 +338,14 @@ def test_pending(self): @responses.activate def test_revoked(self): - """It should raise a Revoked exception if an Http error with the pending code in the body is returned.""" + """Raise a RevokedError exception if an Http error with the pending code in the body is returned.""" # Setup the mocked response - body = json.dumps({"code": Revoked.CODE, "description": "The Certificate has been revoked!"}) + body = json.dumps({"code": RevokedError.CODE, "description": "The Certificate has been revoked!"}) responses.add(responses.GET, self.test_url, body=body, status=400) # Call the function, expecting an exception smime = SMIME(client=self.client) - self.assertRaises(Revoked, smime.collect, self.test_id) + self.assertRaises(RevokedError, smime.collect, self.test_id) # Verify all the query information self.assertEqual(len(responses.calls), 1) @@ -353,7 +353,7 @@ def test_revoked(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response responses.add(responses.GET, self.test_url, json={}, status=400) @@ -383,7 +383,7 @@ def setUp(self): self.test_serial_data = {'orderNumber': 987654321, 'backendCertId': '987654321'} def test_defaults(self): - """Should raise an exception when no params are passed""" + """Should raise an exception when no params are passed.""" smime = SMIME(client=self.client) self.assertRaises(Exception, smime.renew) @@ -441,13 +441,13 @@ def setUp(self): self.test_csr = TestCertificates.fake_csr() def test_defaults(self): - """The function should raise an exception when no params are passed""" + """Raise an exception when no params are passed.""" smime = SMIME(client=self.client) self.assertRaises(Exception, smime.replace) @responses.activate def test_success(self): - """It should return nothing if a 200-level status code is returned""" + """Return nothing if a 200-level status code is returned.""" # Setup the mocked response responses.add(responses.POST, self.test_url, body='', status=204) @@ -473,13 +473,13 @@ def setUp(self): self.test_url = f"{self.api_url}/revoke/order/{self.test_cert_id}" def test_defaults(self): - """The function should raise an exception when no params are passed""" + """Raise an exception when no params are passed.""" smime = SMIME(client=self.client) self.assertRaises(Exception, smime.revoke) @responses.activate def test_success(self): - """It should return nothing if a 200-level status code is returned""" + """Return nothing if a 200-level status code is returned.""" # Setup the mocked response responses.add(responses.POST, self.test_url, body='', status=204) @@ -494,19 +494,19 @@ def test_success(self): self.assertEqual(responses.calls[0].request.url, self.test_url) def test_no_cert_id(self): - """The function should raise a ValueError exception if no cert_id is passed""" + """Raise a ValueError exception if no cert_id is passed.""" smime = SMIME(client=self.client) self.assertRaises(ValueError, smime.revoke, None) def test_no_reason(self): - """It should raise a ValueError exception if the reason is left empty.""" + """Raise a ValueError exception if the reason is left empty.""" # Call the function, expecting an exception smime = SMIME(client=self.client) self.assertRaises(ValueError, smime.revoke, self.test_cert_id) @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response responses.add(responses.POST, self.test_url, json={}, status=404) @@ -530,13 +530,13 @@ def setUp(self): self.test_url = f"{self.api_url}/revoke" def test_defaults(self): - """The function should raise an exception when no params are passed""" + """Raise an exception when no params are passed.""" smime = SMIME(client=self.client) self.assertRaises(Exception, smime.revoke_by_email) @responses.activate def test_success(self): - """It should return nothing if a 200-level status code is returned""" + """Return nothing if a 200-level status code is returned.""" # Setup the mocked response responses.add(responses.POST, self.test_url, body='', status=204) @@ -551,19 +551,19 @@ def test_success(self): self.assertEqual(responses.calls[0].request.url, self.test_url) def test_no_email(self): - """The function should raise a ValueError exception if no email is passed""" + """Raise a ValueError exception if no email is passed.""" smime = SMIME(client=self.client) self.assertRaises(ValueError, smime.revoke_by_email, "") def test_no_reason(self): - """It should raise a ValueError exception if the reason is left empty""" + """Raise a ValueError exception if the reason is left empty.""" # Call the function, expecting an exception smime = SMIME(client=self.client) self.assertRaises(ValueError, smime.revoke_by_email, self.test_email) @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response responses.add(responses.POST, self.test_url, json={}, status=404) diff --git a/tests/test_ssl.py b/tests/test_ssl.py index 7c1e8c2..e750408 100644 --- a/tests/test_ssl.py +++ b/tests/test_ssl.py @@ -1,13 +1,11 @@ -# -*- coding: utf-8 -*- """Define the cert_manager.ssl.SSL unit tests.""" # Don't warn about things that happen as that is part of unit testing # pylint: disable=protected-access # pylint: disable=invalid-name -from testtools import TestCase - -from requests.exceptions import HTTPError import responses +from requests.exceptions import HTTPError +from testtools import TestCase from cert_manager.ssl import SSL @@ -62,14 +60,14 @@ class TestGet(TestSSL): """Test the class get method.""" def test_defaults(self): - """The function should raise an exception when no cert_id is passed.""" + """Raise an exception when no cert_id is passed.""" # Setup the mocked response ssl = SSL(client=self.client) self.assertRaises(TypeError, ssl.get) @responses.activate def test_cert(self): - """The function should return the record for the cert_id passed.""" + """Return the record for the cert_id passed.""" # Setup the mocked response cert_id = "1234567" test_url = f"{self.api_url}/{cert_id}" @@ -87,7 +85,7 @@ def test_cert(self): @responses.activate def test_no_cert(self): - """The function should raise an exception when the cert_id doesn't exist.""" + """Raise an exception when the cert_id doesn't exist.""" # Setup the mocked response cert_id = "1234567" test_url = f"{self.api_url}/{cert_id}" @@ -104,7 +102,7 @@ class TestList(TestSSL): @responses.activate def test_defaults(self): - """The function should return all certificate records.""" + """Return all certificate records.""" # Setup the mocked response test_url = f"{self.api_url}?size=200&position=0" test_result = [ @@ -140,7 +138,7 @@ def setUp(self): @responses.activate def test_success(self): - """It should return JSON if a 200-level status code is returned with data.""" + """Return JSON if a 200-level status code is returned with data.""" # Setup the mocked responses responses.add(responses.POST, self.test_url, status=204) @@ -154,7 +152,7 @@ def test_success(self): @responses.activate def test_failure(self): - """It should raise an HTTPError exception if an error status code is returned.""" + """Raise an HTTPError exception if an error status code is returned.""" # Setup the mocked response responses.add(responses.POST, self.test_url, json={}, status=404) @@ -171,7 +169,7 @@ class TestCount(TestSSL): @responses.activate def test_defaults(self): - """The function should return all certificate records.""" + """Return all certificate records.""" # Setup the mocked response test_url = f"{self.api_url}" count = 10