diff --git a/.coveragerc b/.coveragerc deleted file mode 100644 index f17692c6..00000000 --- a/.coveragerc +++ /dev/null @@ -1,5 +0,0 @@ -[run] -omit = - tests/* -include = - nailgun/*.py diff --git a/.flake8 b/.flake8 deleted file mode 100644 index 8bdd461c..00000000 --- a/.flake8 +++ /dev/null @@ -1,6 +0,0 @@ -[flake8] -max-line-length = 99 - -ignore = W503,W504,E731 - -exclude = .git, __pycache__, build, dist diff --git a/.gitignore b/.gitignore index cb3ba779..b6df87bd 100644 --- a/.gitignore +++ b/.gitignore @@ -17,3 +17,4 @@ __pycache__/ # Common venv name .nailgun/ +venv* diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index ed9dbb96..9a5de8f8 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -1,27 +1,17 @@ # configuration for pre-commit git hooks repos: -- repo: https://github.com/asottile/reorder_python_imports - rev: v3.0.1 - hooks: - - id: reorder-python-imports -- repo: https://github.com/asottile/pyupgrade - rev: v2.32.0 - hooks: - - id: pyupgrade - args: [--py36-plus] - repo: https://github.com/psf/black - rev: 22.3.0 + rev: 23.3.0 hooks: - id: black -- repo: https://github.com/pycqa/flake8 - rev: 3.9.2 +- repo: https://github.com/astral-sh/ruff-pre-commit + rev: v0.0.277 hooks: - - id: flake8 + - id: ruff + args: [--fix, --exit-non-zero-on-fix] - repo: https://github.com/pre-commit/pre-commit-hooks rev: v4.2.0 hooks: - - id: trailing-whitespace - - id: end-of-file-fixer - id: check-yaml - id: debug-statements diff --git a/docs/conf.py b/docs/conf.py index 95b29a34..efe6cb9b 100644 --- a/docs/conf.py +++ b/docs/conf.py @@ -9,7 +9,6 @@ from packaging.version import Version - # Add the NailGun root directory to the system path. This allows references # such as :mod:`nailgun.client` to be processed correctly. ROOT_DIR = os.path.abspath(os.path.join(os.path.dirname(__file__), os.path.pardir)) diff --git a/docs/create_user_nailgun.py b/docs/create_user_nailgun.py index 50972ccc..a94ae953 100755 --- a/docs/create_user_nailgun.py +++ b/docs/create_user_nailgun.py @@ -3,17 +3,14 @@ from pprint import pprint from nailgun.config import ServerConfig -from nailgun.entities import Organization -from nailgun.entities import User +from nailgun.entities import Organization, User def main(): """Create an identical user account on a pair of satellites.""" server_configs = ServerConfig.get('sat1'), ServerConfig.get('sat2') for server_config in server_configs: - org = Organization(server_config).search(query={'search': 'name="Default_Organization"'})[ - 0 - ] + org = Organization(server_config).search(query={'search': 'name="Default_Organization"'})[0] # The LDAP authentication source with an ID of 1 is internal. It is # nearly guaranteed to exist and be functioning. user = User( diff --git a/docs/create_user_plain.py b/docs/create_user_plain.py index ede2efba..5df1165c 100755 --- a/docs/create_user_plain.py +++ b/docs/create_user_plain.py @@ -12,6 +12,7 @@ """ import json from pprint import pprint +import sys import requests @@ -66,11 +67,11 @@ def get_organization_id(server_config, label): response.raise_for_status() decoded = response.json() if decoded['subtotal'] != 1: - print( + pprint( f'Expected to find one organization, but instead found {decoded["subtotal"]}. ' f'Search results: {decoded["results"]}' ) - exit(1) + sys.exit(1) return decoded['results'][0]['id'] diff --git a/nailgun/__init__.py b/nailgun/__init__.py index 098cf95d..d1ac40c5 100644 --- a/nailgun/__init__.py +++ b/nailgun/__init__.py @@ -16,5 +16,4 @@ """ from logging import basicConfig - basicConfig() diff --git a/nailgun/client.py b/nailgun/client.py index dd6fcc5a..3ae64d00 100644 --- a/nailgun/client.py +++ b/nailgun/client.py @@ -15,14 +15,13 @@ http://docs.python-requests.org/en/latest/api/#main-interface """ -import logging from json import dumps +import logging from warnings import simplefilter import requests import urllib3 - logger = logging.getLogger(__name__) # The urllib3 module (which requests uses) refuses to make insecure HTTPS @@ -87,7 +86,7 @@ def _set_content_type(kwargs): def _truncate_data(data, max_len=500): """Truncate data to a max length""" - if isinstance(data, (str, bytes)): + if isinstance(data, str | bytes): if len(data) > max_len: return f"{data[:max_len - 3]}..." return data @@ -127,7 +126,7 @@ def _log_response(response): """ message = f'Received HTTP {response.status_code} response: {response.text}' - if response.status_code >= 400: # pragma: no cover + if not response.ok: logger.warning(message) else: logger.debug(message) diff --git a/nailgun/config.py b/nailgun/config.py index 00a8d882..ad261d30 100644 --- a/nailgun/config.py +++ b/nailgun/config.py @@ -8,8 +8,7 @@ """ import json -from os.path import isfile -from os.path import join +from os.path import isfile, join from threading import Lock from packaging.version import parse @@ -114,7 +113,7 @@ def __repr__(self): attrs = vars(self).copy() if "version" in attrs: attrs["version"] = str(attrs.pop("version")) - kv_pairs = ", ".join(f"{key}={repr(value)}" for key, value in attrs.items()) + kv_pairs = ", ".join(f"{key}={value!r}" for key, value in attrs.items()) return f"{self.__module__}.{type(self).__name__}({kv_pairs})" @classmethod @@ -201,9 +200,7 @@ def save(self, label='default', path=None): # Where is the file we're writing to? if path is None: - path = join( - BaseDirectory.save_config_path(self._xdg_config_dir), self._xdg_config_file - ) + path = join(BaseDirectory.save_config_path(self._xdg_config_dir), self._xdg_config_file) self._file_lock.acquire() try: diff --git a/nailgun/entities.py b/nailgun/entities.py index d39d3cca..7da2f9a5 100644 --- a/nailgun/entities.py +++ b/nailgun/entities.py @@ -20,30 +20,29 @@ workings of entity classes. """ -import hashlib -import os.path from datetime import datetime from functools import lru_cache -from http.client import ACCEPTED -from http.client import NO_CONTENT -from urllib.parse import urljoin # noqa:F0401,E0611 +import hashlib +from http.client import ACCEPTED, NO_CONTENT +import os.path +from urllib.parse import urljoin -from fauxfactory import gen_alphanumeric -from fauxfactory import gen_choice +from fauxfactory import gen_alphanumeric, gen_choice from packaging.version import Version -from nailgun import client -from nailgun import entity_fields -from nailgun.entity_mixins import _get_entity_ids -from nailgun.entity_mixins import _payload -from nailgun.entity_mixins import _poll_task -from nailgun.entity_mixins import Entity -from nailgun.entity_mixins import EntityCreateMixin -from nailgun.entity_mixins import EntityDeleteMixin -from nailgun.entity_mixins import EntityReadMixin -from nailgun.entity_mixins import EntitySearchMixin -from nailgun.entity_mixins import EntityUpdateMixin -from nailgun.entity_mixins import to_json_serializable # noqa: F401 +from nailgun import client, entity_fields +from nailgun.entity_mixins import ( + Entity, + EntityCreateMixin, + EntityDeleteMixin, + EntityReadMixin, + EntitySearchMixin, + EntityUpdateMixin, + _get_entity_ids, + _payload, + _poll_task, + to_json_serializable, # noqa: F401 +) # The size of this file is a direct reflection of the size of Satellite's API. # This file's size has already been significantly cut down through the use of @@ -195,7 +194,7 @@ def _get_version(server_config): return getattr(server_config, 'version', Version('1!0')) -@lru_cache() +@lru_cache def _feature_list(server_config, smart_proxy_id=1): """Get list of features enabled on capsule""" smart_proxy = SmartProxy(server_config, id=smart_proxy_id).read_json() @@ -1519,7 +1518,10 @@ def read(self, entity=None, attrs=None, ignore=None, params=None): if attr not in ignore: # We cannot call `self.update_json([])`, as an ID might not be # present on self. However, `attrs` is guaranteed to have an ID. - attrs[attr] = DiscoveryRule(self._server_config, id=attrs['id'],).update_json( + attrs[attr] = DiscoveryRule( + self._server_config, + id=attrs['id'], + ).update_json( [] )[attr] return super().read(entity, attrs, ignore, params) @@ -1943,7 +1945,7 @@ def read(self, entity=None, attrs=None, ignore=None, params=None): ) for entity_id in _get_entity_ids('template_inputs', attrs) ] - setattr(entity, 'template_inputs', referenced_entities) + entity.template_inputs = referenced_entities return entity @@ -2520,9 +2522,7 @@ def read(self, entity=None, attrs=None, ignore=None, params=None): if ignore is None: ignore = set() ignore.add('content_view_filter') - ignore.update( - [field_name for field_name in entity.get_fields().keys() if field_name not in attrs] - ) + ignore.update([field_name for field_name in entity.get_fields() if field_name not in attrs]) return super().read(entity, attrs, ignore, params) def create_payload(self): @@ -2890,7 +2890,7 @@ def add(self, synchronous=True, timeout=None, **kwargs): kwargs = kwargs.copy() # shadow the passed-in kwargs if 'data' not in kwargs: # data is required - kwargs['data'] = dict() + kwargs['data'] = {} if 'component_ids' not in kwargs['data']: kwargs['data']['components'] = [_payload(self.get_fields(), self.get_values())] kwargs.update(self._server_config.get_client_kwargs()) @@ -2914,7 +2914,7 @@ def remove(self, synchronous=True, timeout=None, **kwargs): kwargs = kwargs.copy() # shadow the passed-in kwargs if 'data' not in kwargs: # data is required - kwargs['data'] = dict() + kwargs['data'] = {} if 'data' in kwargs and 'component_ids' not in kwargs['data']: kwargs['data']['component_ids'] = [self.id] kwargs.update(self._server_config.get_client_kwargs()) @@ -3970,7 +3970,7 @@ def get_values(self): attrs.pop('_owner_type') return attrs - def create_missing(self): + def create_missing(self): # noqa: PLR0912, PLR0915 - TODO: Refactor this? """Create a bogus managed host. The exact set of attributes that are required varies depending on @@ -4976,7 +4976,7 @@ def read(self, entity=None, attrs=None, ignore=None, params=None): if attrs['type'] != 'virtual': ignore.add('attached_to') ignore.add('tag') - if attrs['type'] != 'bridge' and attrs['type'] != 'bond': + if attrs['type'] not in ('bridge', 'bond'): ignore.add('attached_devices') return super().read(entity, attrs, ignore, params) @@ -5754,9 +5754,7 @@ def __init__(self, server_config=None, **kwargs): self._fields = { 'match': entity_fields.StringField(required=True), 'value': entity_fields.StringField(required=True), - 'smart_class_parameter': entity_fields.OneToOneField( - SmartClassParameters, parent=True - ), + 'smart_class_parameter': entity_fields.OneToOneField(SmartClassParameters, parent=True), 'omit': entity_fields.BooleanField(), } super().__init__(server_config, **kwargs) @@ -6180,10 +6178,7 @@ def search_normalize(self, results): while Puppet Class entity returns dictionary with lists of subclasses split by main puppet class. """ - flattened_results = [] - for key in results.keys(): - for item in results[key]: - flattened_results.append(item) + flattened_results = [item for sublist in results.values() for item in sublist] return super().search_normalize(flattened_results) def path(self, which=None): @@ -6699,8 +6694,8 @@ def import_uploads( It expects either a list of uploads or upload_ids (but not both). - :param content_type: content type (‘deb’, ‘docker_manifest’, ‘file’, ‘ostree’, - ‘rpm’, ‘srpm’) + :param content_type: content type (`deb`, `docker_manifest`, `file`, `ostree`, + `rpm`, `srpm`) :param uploads: Array of uploads to be imported :param upload_ids: Array of upload ids to be imported :param synchronous: What should happen if the server returns an HTTP @@ -6845,7 +6840,7 @@ def available_repositories(self, **kwargs): """ if 'data' not in kwargs: - kwargs['data'] = dict() + kwargs['data'] = {} kwargs['data']['product_id'] = self.product.id kwargs = kwargs.copy() # shadow the passed-in kwargs kwargs.update(self._server_config.get_client_kwargs()) @@ -6869,7 +6864,7 @@ def enable(self, synchronous=True, timeout=None, **kwargs): """ if 'data' not in kwargs: - kwargs['data'] = dict() + kwargs['data'] = {} kwargs['data']['product_id'] = self.product.id kwargs = kwargs.copy() # shadow the passed-in kwargs kwargs.update(self._server_config.get_client_kwargs()) @@ -6891,7 +6886,7 @@ def disable(self, synchronous=True, timeout=None, **kwargs): """ if 'data' not in kwargs: - kwargs['data'] = dict() + kwargs['data'] = {} kwargs['data']['product_id'] = self.product.id kwargs = kwargs.copy() # shadow the passed-in kwargs kwargs.update(self._server_config.get_client_kwargs()) diff --git a/nailgun/entity_fields.py b/nailgun/entity_fields.py index e12c540b..a30ae568 100644 --- a/nailgun/entity_fields.py +++ b/nailgun/entity_fields.py @@ -20,18 +20,20 @@ """ import random -from fauxfactory import gen_alpha -from fauxfactory import gen_boolean -from fauxfactory import gen_choice -from fauxfactory import gen_date -from fauxfactory import gen_datetime -from fauxfactory import gen_email -from fauxfactory import gen_integer -from fauxfactory import gen_ipaddr -from fauxfactory import gen_mac -from fauxfactory import gen_netmask -from fauxfactory import gen_string -from fauxfactory import gen_url +from fauxfactory import ( + gen_alpha, + gen_boolean, + gen_choice, + gen_date, + gen_datetime, + gen_email, + gen_integer, + gen_ipaddr, + gen_mac, + gen_netmask, + gen_string, + gen_url, +) # The classes in this module serve a declarative role. It is OK that they don't # do much. @@ -63,9 +65,7 @@ class Field: mapped entity """ - def __init__( - self, required=False, choices=None, default=_SENTINEL, unique=False, parent=False - ): + def __init__(self, required=False, choices=None, default=_SENTINEL, unique=False, parent=False): self.unique = unique self.required = required self.parent = parent diff --git a/nailgun/entity_mixins.py b/nailgun/entity_mixins.py index f7f97be0..5d724844 100644 --- a/nailgun/entity_mixins.py +++ b/nailgun/entity_mixins.py @@ -1,23 +1,19 @@ """Defines a set of mixins that provide tools for interacting with entities.""" import _thread as thread +from collections.abc import Iterable +import contextlib +from datetime import date, datetime import http.client as http_client import json as std_json import threading import time -from collections.abc import Iterable -from datetime import date -from datetime import datetime from urllib.parse import urljoin from fauxfactory import gen_choice from inflection import pluralize -from nailgun import client -from nailgun import config -from nailgun.entity_fields import IntegerField -from nailgun.entity_fields import ListField -from nailgun.entity_fields import OneToManyField -from nailgun.entity_fields import OneToOneField +from nailgun import client, config +from nailgun.entity_fields import IntegerField, ListField, OneToManyField, OneToOneField # This module contains very extensive docstrings, so this module is easier to # understand than its size suggests. That said, it could be useful to split @@ -120,10 +116,10 @@ def raise_task_timeout(): # pragma: no cover if task_info['state'] in ('paused', 'stopped'): break time.sleep(poll_rate) - except KeyboardInterrupt: # pragma: no cover + except KeyboardInterrupt: # raise_task_timeout will raise a KeyboardInterrupt when the timeout # expires. Catch the exception and raise TaskTimedOutError - raise TaskTimedOutError( + raise TaskTimedOutError( # noqa: TRY200 - Not raising from KeyBoardInterrupt f"Timed out polling task {task_id}. Task information: {task_info}", task_id ) finally: @@ -535,7 +531,7 @@ def get_values(self): def __repr__(self): kv_pairs = ", ".join( - f"{key}={repr(value)}" + f"{key}={value!r}" for key, value in self.get_values().items() if not key.startswith("_") ) @@ -713,7 +709,6 @@ def delete(self, synchronous=True, timeout=None): and hasattr(response, 'content') and not response.content.strip() ): - # "The server successfully processed the request, but is not # returning any content. Usually used as a response to a successful # delete request." @@ -1291,20 +1286,17 @@ def search_normalize(self, results): attrs = {} for field_name, field in fields.items(): if isinstance(field, OneToOneField): - try: + with contextlib.suppress(MissingValueError): attrs[field_name] = _get_entity_id(field_name, result) - except MissingValueError: - pass + elif isinstance(field, OneToManyField): - try: + with contextlib.suppress(MissingValueError): attrs[field_name] = _get_entity_ids(field_name, result) - except MissingValueError: - pass + else: - try: + with contextlib.suppress(KeyError): attrs[field_name] = result[field_name] - except KeyError: - pass + normalized.append(attrs) return normalized @@ -1437,7 +1429,7 @@ def search_filter(entities, filters): f'Valid filters are {fields.keys()}, but received {filters.keys()} instead.' ) for field_name in filters: - if isinstance(fields[field_name], (OneToOneField, OneToManyField)): + if isinstance(fields[field_name], OneToOneField | OneToManyField): raise NotImplementedError( 'Search results cannot (yet?) be locally filtered by ' f'`OneToOneField`s and `OneToManyField`s. ' @@ -1447,9 +1439,7 @@ def search_filter(entities, filters): # The arguments are sane. Filter away! filtered = [entity.read() for entity in entities] # don't alter inputs for field_name, field_value in filters.items(): - filtered = [ - entity for entity in filtered if getattr(entity, field_name) == field_value - ] + filtered = [entity for entity in filtered if getattr(entity, field_name) == field_value] return filtered @@ -1466,7 +1456,7 @@ def to_json_serializable(obj): if isinstance(obj, dict): return {k: to_json_serializable(v) for k, v in obj.items()} - elif isinstance(obj, (list, tuple)): + elif isinstance(obj, list | tuple): return [to_json_serializable(v) for v in obj] elif isinstance(obj, datetime): return obj.strftime('%Y-%m-%d %H:%M:%S') diff --git a/pyproject.toml b/pyproject.toml index a61a7db4..6326758f 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,17 +1,122 @@ +[tool.pytest.ini_options] +testpaths = ["tests"] +addopts = ["-v", "-l", "--color=yes", "--code-highlight=yes"] + [tool.black] -line-length = 99 +line-length = 100 skip-string-normalization = true +target-version = ["py310", "py311"] include = '\.pyi?$' exclude = ''' /( \.git - | \.hg - | \.mypy_cache - | \.tox | \.venv - | _build - | buck-out | build | dist + | tests/data )/ ''' + +[tool.ruff] +target-version = "py311" +fixable = ["ALL"] + +select = [ + "B002", # Python does not support the unary prefix increment + "B007", # Loop control variable {name} not used within loop body + "B009", # Do not call getattr with a constant attribute value + "B010", # Do not call setattr with a constant attribute value + "B011", # Do not `assert False`, raise `AssertionError` instead + "B013", # Redundant tuple in exception handler + "B014", # Exception handler with duplicate exception + "B023", # Function definition does not bind loop variable {name} + "B026", # Star-arg unpacking after a keyword argument is strongly discouraged + "BLE001", # Using bare except clauses is prohibited + "C", # complexity + "C4", # flake8-comprehensions + "COM818", # Trailing comma on bare tuple prohibited + # "D", # docstrings + "E", # pycodestyle + "F", # pyflakes/autoflake + "G", # flake8-logging-format + "I", # isort + "ISC001", # Implicitly concatenated string literals on one line + "N804", # First argument of a class method should be named cls + "N805", # First argument of a method should be named self + "N815", # Variable {name} in class scope should not be mixedCase + "N999", # Invalid module name: '{name}' + "PERF", # Perflint rules + "PGH004", # Use specific rule codes when using noqa + "PLC0414", # Useless import alias. Import alias does not rename original package. + "PLC", # pylint + "PLE", # pylint + "PLR", # pylint + "PLW", # pylint + "RUF", # Ruff-specific rules + "S103", # bad-file-permissions + "S108", # hardcoded-temp-file + "S110", # try-except-pass + "S112", # try-except-continue + "S306", # suspicious-mktemp-usage + "S307", # suspicious-eval-usage + "S601", # paramiko-call + "S602", # subprocess-popen-with-shell-equals-true + "S604", # call-with-shell-equals-true + "S609", # unix-command-wildcard-injection + "SIM105", # Use contextlib.suppress({exception}) instead of try-except-pass + "SIM117", # Merge with-statements that use the same scope + "SIM118", # Use {key} in {dict} instead of {key} in {dict}.keys() + "SIM201", # Use {left} != {right} instead of not {left} == {right} + "SIM208", # Use {expr} instead of not (not {expr}) + "SIM212", # Use {a} if {a} else {b} instead of {b} if not {a} else {a} + "SIM300", # Yoda conditions. Use 'age == 42' instead of '42 == age'. + "SIM401", # Use get from dict with default instead of an if block + "T100", # Trace found: {name} used + "T20", # flake8-print + "TRY004", # Prefer TypeError exception for invalid type + "TRY200", # Use raise from to specify exception cause + "TRY302", # Remove exception handler; error is immediately re-raised + "PLR0911", # Too many return statements ({returns} > {max_returns}) + "PLR0912", # Too many branches ({branches} > {max_branches}) + "PLR0915", # Too many statements ({statements} > {max_statements}) + "PLR2004", # Magic value used in comparison, consider replacing {value} with a constant variable + "PLW2901", # Outer {outer_kind} variable {name} overwritten by inner {inner_kind} target + "UP", # pyupgrade + "W", # pycodestyle +] + +ignore = [ + "ANN", # flake8-annotations + "D203", # 1 blank line required before class docstring + "D213", # Multi-line docstring summary should start at the second line + "D406", # Section name should end with a newline + "D407", # Section name underlining + "E501", # line too long + "E731", # do not assign a lambda expression, use a def + "PGH001", # No builtin eval() allowed + "PLR0913", # Too many arguments to function call ({c_args} > {max_args}) + "PLW0603", # Using the global statement + "RUF012", # Mutable class attributes should be annotated with typing.ClassVar + "D107", # Missing docstring in __init__ +] + +[tool.ruff.per-file-ignores] +# Allow pprint for docs formatting +"docs/create_*.py" = ["T203"] + +[tool.ruff.flake8-pytest-style] +fixture-parentheses = false + +[tool.ruff.isort] +force-sort-within-sections = true +known-first-party = [ + "broker", +] +combine-as-imports = true + +[tool.ruff.mccabe] +max-complexity = 25 + +[tool.coverage.run] +omit = ["tests/*"] +include = ["nailgun/*.py"] diff --git a/requirements-dev.txt b/requirements-dev.txt index f4c6c354..40e934c7 100644 --- a/requirements-dev.txt +++ b/requirements-dev.txt @@ -3,7 +3,6 @@ mock unittest2 ; python_version < '3.4' # For `pre-commit` -flake8 pre-commit # For `make docs-html` and `make docs-clean` @@ -17,3 +16,7 @@ twine # For code coverage codecov==2.1.13 +coverage[toml] + +# For linting +ruff diff --git a/setup.py b/setup.py index ecca12e3..761efd78 100755 --- a/setup.py +++ b/setup.py @@ -7,9 +7,7 @@ * https://docs.python.org/distutils/sourcedist.html """ -from setuptools import find_packages -from setuptools import setup - +from setuptools import find_packages, setup with open('README.rst') as handle: LONG_DESCRIPTION = handle.read() diff --git a/tests/test_client.py b/tests/test_client.py index d5557afc..0de36914 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -1,10 +1,9 @@ """Unit tests for :mod:`nailgun.client`.""" import inspect -from unittest import mock -from unittest import TestCase +from unittest import TestCase, mock -import requests from fauxfactory import gen_alpha +import requests from nailgun import client diff --git a/tests/test_config.py b/tests/test_config.py index 5fa01b4e..5ed49783 100644 --- a/tests/test_config.py +++ b/tests/test_config.py @@ -2,20 +2,18 @@ import builtins import json from unittest import TestCase -from unittest.mock import call -from unittest.mock import mock_open -from unittest.mock import patch +from unittest.mock import call, mock_open, patch -from packaging.version import InvalidVersion -from packaging.version import parse - -from nailgun.config import _get_config_file_path -from nailgun.config import BaseServerConfig -from nailgun.config import ConfigFileError -from nailgun.config import ServerConfig +from packaging.version import InvalidVersion, parse +from nailgun.config import ( + BaseServerConfig, + ConfigFileError, + ServerConfig, + _get_config_file_path, +) -FILE_PATH = '/tmp/bogus.json' +FILE_PATH = '/tmp/bogus.json' # noqa: S108 CONFIGS = { 'default': {'url': 'http://example.com'}, 'Ask Aak': {'url': 'bogus value', 'auth': ['username', 'password']}, @@ -30,6 +28,12 @@ } ) +# Tests use an unused nailgun import +# ruff: noqa: F401 + +# Cannot use ast.literal_eval because ServerConfig isn't a basic type +# ruff: noqa: S307 + def _convert_bsc_attrs(bsc_attrs): """Alter a dict of attributes in the same way as ``BaseServerConfig``. @@ -195,7 +199,7 @@ def test_get(self): Assert that the ``auth`` attribute is a tuple. """ - for label in CONFIGS.keys(): + for label in CONFIGS: open_ = mock_open(read_data=json.dumps(CONFIGS)) with patch.object(builtins, 'open', open_): server_config = ServerConfig.get(label, FILE_PATH) @@ -215,7 +219,7 @@ def test_bsc_v1(self): """ target = "nailgun.config.BaseServerConfig(url='bogus')" self.assertEqual(target, repr(BaseServerConfig('bogus'))) - import nailgun # noqa + import nailgun self.assertEqual(target, repr(eval(repr(BaseServerConfig('bogus'))))) @@ -231,7 +235,7 @@ def test_bsc_v2(self): "nailgun.config.BaseServerConfig(auth='flam', url='flim')", ) self.assertIn(repr(BaseServerConfig('flim', auth='flam')), targets) - import nailgun # noqa + import nailgun self.assertIn(repr(eval(repr(BaseServerConfig('flim', auth='flam')))), targets) @@ -256,7 +260,7 @@ def test_sc_v1(self): """ target = "nailgun.config.ServerConfig(url='bogus')" self.assertEqual(target, repr(ServerConfig('bogus'))) - import nailgun # noqa + import nailgun self.assertEqual(target, repr(eval(repr(ServerConfig('bogus'))))) @@ -272,7 +276,7 @@ def test_sc_v2(self): "nailgun.config.ServerConfig(auth='flam', url='flim')", ) self.assertIn(repr(ServerConfig('flim', auth='flam')), targets) - import nailgun # noqa + import nailgun self.assertIn(repr(eval(repr(ServerConfig('flim', auth='flam')))), targets) @@ -301,7 +305,7 @@ def test_sc_v4(self): "nailgun.config.ServerConfig(verify='flub', url='flim')", ) self.assertIn(repr(ServerConfig('flim', verify='flub')), targets) - import nailgun # noqa + import nailgun self.assertIn(repr(eval(repr(ServerConfig('flim', verify='flub')))), targets) diff --git a/tests/test_entities.py b/tests/test_entities.py index f488c17d..8ca03edd 100644 --- a/tests/test_entities.py +++ b/tests/test_entities.py @@ -1,26 +1,21 @@ """Tests for :mod:`nailgun.entities`.""" +from datetime import date, datetime +from http.client import ACCEPTED, NO_CONTENT import inspect import json import os -from datetime import date -from datetime import datetime -from http.client import ACCEPTED -from http.client import NO_CONTENT -from unittest import mock -from unittest import TestCase - -from fauxfactory import gen_alpha -from fauxfactory import gen_integer -from fauxfactory import gen_string - -from nailgun import client -from nailgun import config -from nailgun import entities -from nailgun.entity_mixins import EntityCreateMixin -from nailgun.entity_mixins import EntityReadMixin -from nailgun.entity_mixins import EntitySearchMixin -from nailgun.entity_mixins import EntityUpdateMixin -from nailgun.entity_mixins import NoSuchPathError +from unittest import TestCase, mock + +from fauxfactory import gen_alpha, gen_integer, gen_string + +from nailgun import client, config, entities +from nailgun.entity_mixins import ( + EntityCreateMixin, + EntityReadMixin, + EntitySearchMixin, + EntityUpdateMixin, + NoSuchPathError, +) _BUILTIN_OPEN = 'builtins.open' # For inspection comparison, a tuple matching the expected func arg spec @@ -38,6 +33,9 @@ # The size of this file is a direct reflection of the size of module # `nailgun.entities` and the Satellite API. +# Due to the length of the with statements, nested is preferred over combined +# ruff: noqa: SIM117 + def make_entity(cls, **kwargs): """Helper function to create entity with dummy ServerConfig""" @@ -242,9 +240,8 @@ def test_required_params(self): entities.SyncPlan, entities.TemplateInput, ): - with self.subTest(): - with self.assertRaises(TypeError): - entity(self.cfg) + with self.subTest(), self.assertRaises(TypeError): + entity(self.cfg) class PathTestCase(TestCase): @@ -427,9 +424,8 @@ def test_no_such_path_error(self): (entities.VirtWhoConfig, 'deploy_script'), (entities.VirtWhoConfig, 'configs'), ): - with self.subTest((entity, which)): - with self.assertRaises(NoSuchPathError): - entity(self.cfg).path(which=which) + with self.subTest((entity, which)), self.assertRaises(NoSuchPathError): + entity(self.cfg).path(which=which) def test_arfreport(self): """Test :meth:`nailgun.entities.ArfReport.path`. @@ -1436,13 +1432,12 @@ def test_snapshot_ignore_arg(self): Assert that entity`s predefined values of ``ignore`` are always correctly passed on. """ - with mock.patch.object(EntityReadMixin, 'read') as read: - with mock.patch.object( - EntityReadMixin, - 'read_json', - return_value={'host': 3}, - ): - entities.Snapshot(self.cfg, id=2, host=3).read() + with mock.patch.object(EntityReadMixin, 'read') as read, mock.patch.object( + EntityReadMixin, + 'read_json', + return_value={'host': 3}, + ): + entities.Snapshot(self.cfg, id=2, host=3).read() # `call_args` is a two-tuple of (positional, keyword) args. self.assertEqual({'host'}, read.call_args[0][2]) @@ -1456,22 +1451,20 @@ def test_host_with_interface(self): EntityReadMixin, 'read', return_value=entities.Host(self.cfg, id=2), + ), mock.patch.object( + EntityReadMixin, + 'read_json', + return_value={ + 'interfaces': [{'id': 2}, {'id': 3}], + 'parameters': None, + 'puppet_proxy': None, + }, + ), mock.patch.object( + entities, + '_feature_list', + return_value={'Puppet'}, ): - with mock.patch.object( - EntityReadMixin, - 'read_json', - return_value={ - 'interfaces': [{'id': 2}, {'id': 3}], - 'parameters': None, - 'puppet_proxy': None, - }, - ): - with mock.patch.object( - entities, - '_feature_list', - return_value={'Puppet'}, - ): - host = entities.Host(self.cfg, id=2).read() + host = entities.Host(self.cfg, id=2).read() self.assertTrue(hasattr(host, 'interface')) self.assertTrue(isinstance(host.interface, list)) for interface in host.interface: @@ -1804,7 +1797,6 @@ def test_generic(self): ) for entity in entities_: with self.subTest(entity): - # Call update() with mock.patch.object(entity, 'update_json') as update_json: with mock.patch.object(entity, 'read') as read: @@ -2356,11 +2348,10 @@ def test_bulk_resume(self): {'search': gen_string('alpha')}, {'task_ids': self.foreman_task.id, 'search': gen_string('alpha')}, ): - with self.subTest(kwargs): - with mock.patch.object(client, 'post') as post: - self.foreman_task.bulk_resume(**kwargs) - self.assertEqual(post.call_count, 1) - self.assertEqual(post.mock_calls[2][1][0].ACCEPTED, 202) + with self.subTest(kwargs), mock.patch.object(client, 'post') as post: + self.foreman_task.bulk_resume(**kwargs) + self.assertEqual(post.call_count, 1) + self.assertEqual(post.mock_calls[2][1][0].ACCEPTED, 202) def test_bulk_cancel(self): """Call :meth:`nailgun.entities.ForemanTask.bulk_cancel`.""" @@ -2370,11 +2361,10 @@ def test_bulk_cancel(self): {'search': gen_string('alpha')}, {'task_ids': self.foreman_task.id, 'search': gen_string('alpha')}, ): - with self.subTest(kwargs): - with mock.patch.object(client, 'post') as post: - self.foreman_task.bulk_cancel(**kwargs) - self.assertEqual(post.call_count, 1) - self.assertEqual(post.mock_calls[2][1][0].ACCEPTED, 202) + with self.subTest(kwargs), mock.patch.object(client, 'post') as post: + self.foreman_task.bulk_cancel(**kwargs) + self.assertEqual(post.call_count, 1) + self.assertEqual(post.mock_calls[2][1][0].ACCEPTED, 202) class ContentUploadTestCase(TestCase): @@ -2451,15 +2441,14 @@ def test_content_upload_upload(self): with mock.patch.object( entities.ContentUpload, 'create', - ) as create: - with mock.patch.object( - entities.Repository, - 'import_uploads', - return_value={'status': 'success'}, - ) as import_uploads: - mock_open = mock.mock_open(read_data=gen_string('alpha').encode('ascii')) - with mock.patch(_BUILTIN_OPEN, mock_open, create=True): - response = self.content_upload.upload(filepath, filename) + ) as create, mock.patch.object( + entities.Repository, + 'import_uploads', + return_value={'status': 'success'}, + ) as import_uploads: + mock_open = mock.mock_open(read_data=gen_string('alpha').encode('ascii')) + with mock.patch(_BUILTIN_OPEN, mock_open, create=True): + response = self.content_upload.upload(filepath, filename) self.assertEqual(import_uploads.call_count, 1) self.assertEqual(create.call_count, 1) self.assertEqual(import_uploads.return_value, response) @@ -2477,15 +2466,14 @@ def test_content_upload_no_filename(self): with mock.patch.object( entities.ContentUpload, 'create', - ) as create: - with mock.patch.object( - entities.Repository, - 'import_uploads', - return_value={'status': 'success'}, - ) as import_uploads: - mock_open = mock.mock_open(read_data=gen_string('alpha').encode('ascii')) - with mock.patch(_BUILTIN_OPEN, mock_open, create=True): - response = self.content_upload.upload(filepath) + ) as create, mock.patch.object( + entities.Repository, + 'import_uploads', + return_value={'status': 'success'}, + ) as import_uploads: + mock_open = mock.mock_open(read_data=gen_string('alpha').encode('ascii')) + with mock.patch(_BUILTIN_OPEN, mock_open, create=True): + response = self.content_upload.upload(filepath) self.assertEqual(import_uploads.call_count, 1) self.assertEqual(create.call_count, 1) self.assertEqual(import_uploads.return_value, response) @@ -2580,9 +2568,7 @@ def test_search(self): response = self.cv.search() self.assertEqual(handlr.call_count, 1) self.assertEqual(type(response[0]), entities.ContentView) - self.assertEqual( - type(response[0].content_view_component[0]), entities.ContentViewComponent - ) + self.assertEqual(type(response[0].content_view_component[0]), entities.ContentViewComponent) class ContentViewComponentTestCase(TestCase): @@ -2805,19 +2791,19 @@ def setUp(self): self.cfg = config.ServerConfig('some url') self.job_template = entities.JobTemplate(self.cfg, id=2) self.entity = entities.TemplateInput(self.cfg, id=1, template=self.job_template) - self.data = dict( - id=1, - description=None, - fact_name=None, - input_type='user', - name='my new template input', - options=None, - puppet_class_name=None, - puppet_parameter_name=None, - required=False, - template_id=self.job_template.id, - variable_name=None, - ) + self.data = { + 'id': 1, + 'description': None, + 'fact_name': None, + 'input_type': 'user', + 'name': 'my new template input', + 'options': None, + 'puppet_class_name': None, + 'puppet_parameter_name': None, + 'required': False, + 'template_id': self.job_template.id, + 'variable_name': None, + } self.read_json_patcher = mock.patch.object(EntityReadMixin, 'read_json') self.read_json = self.read_json_patcher.start() self.read_json.return_value = self.data.copy() @@ -2829,7 +2815,7 @@ def tearDown(self): def test_read(self): entity = self.entity.read() self.read_json.assert_called_once() - self.assertEqual(self.data, {key: getattr(entity, key) for key in self.data.keys()}) + self.assertEqual(self.data, {key: getattr(entity, key) for key in self.data}) self.assertIsInstance(entity.template, entities.JobTemplate) self.assertEqual(entity.template.id, self.job_template.id) @@ -2842,22 +2828,22 @@ def setUp(self): self.entity = entities.JobTemplate(self.cfg, id=1) self.read_json_patcher = mock.patch.object(EntityReadMixin, 'read_json') self.read_json = self.read_json_patcher.start() - self.template_input_data = dict(id=1, template=1) - self.data = dict( - id=1, - audit_comment=None, - description_format=None, - effective_user=None, - job_category='Commands', - location=[], - locked=False, - name='my new job template', - organization=[], - provider_type=None, - snippet=False, - template='rm -rf /', - template_inputs=[self.template_input_data], - ) + self.template_input_data = {'id': 1, 'template': 1} + self.data = { + 'id': 1, + 'audit_comment': None, + 'description_format': None, + 'effective_user': None, + 'job_category': 'Commands', + 'location': [], + 'locked': False, + 'name': 'my new job template', + 'organization': [], + 'provider_type': None, + 'snippet': False, + 'template': 'rm -rf /', + 'template_inputs': [self.template_input_data], + } self.read_json.return_value = self.data.copy() del self.data['template_inputs'] @@ -2867,7 +2853,7 @@ def tearDown(self): def test_read(self): entity = self.entity.read() self.read_json.assert_called_once() - self.assertEqual(self.data, {key: getattr(entity, key) for key in self.data.keys()}) + self.assertEqual(self.data, {key: getattr(entity, key) for key in self.data}) self.assertEqual(len(entity.template_inputs), 1) template_input = entity.template_inputs[0] self.assertIsInstance(template_input, entities.TemplateInput) @@ -3180,13 +3166,12 @@ def test_upload_content_v1(self): """ kwargs = {'kwarg': gen_integer()} - with mock.patch.object(client, 'post') as post: - with mock.patch.object( - entities, - '_handle_response', - return_value={'status': 'success'}, - ) as handler: - response = self.repo.upload_content(**kwargs) + with mock.patch.object(client, 'post') as post, mock.patch.object( + entities, + '_handle_response', + return_value={'status': 'success'}, + ) as handler: + response = self.repo.upload_content(**kwargs) self.assertEqual(post.call_count, 1) self.assertEqual(len(post.call_args[0]), 1) self.assertEqual(post.call_args[1], kwargs) @@ -3201,14 +3186,12 @@ def test_upload_content_v2(self): """ kwargs = {'kwarg': gen_integer()} - with mock.patch.object(client, 'post') as post: - with mock.patch.object( - entities, - '_handle_response', - return_value={'status': 'failure'}, - ) as handler: - with self.assertRaises(entities.APIResponseError): - self.repo.upload_content(**kwargs) + with mock.patch.object(client, 'post') as post, mock.patch.object( + entities, + '_handle_response', + return_value={'status': 'failure'}, + ) as handler, self.assertRaises(entities.APIResponseError): + self.repo.upload_content(**kwargs) self.assertEqual(post.call_count, 1) self.assertEqual(len(post.call_args[0]), 1) self.assertEqual(post.call_args[1], kwargs) @@ -3232,13 +3215,12 @@ def test_import_uploads_uploads(self): 'checksum': gen_string('numeric'), } ] - with mock.patch.object(client, 'put') as put: - with mock.patch.object( - entities, - '_handle_response', - return_value={'status': 'success'}, - ) as handler: - response = self.repo.import_uploads(uploads=uploads, **kwargs) + with mock.patch.object(client, 'put') as put, mock.patch.object( + entities, + '_handle_response', + return_value={'status': 'success'}, + ) as handler: + response = self.repo.import_uploads(uploads=uploads, **kwargs) self.assertEqual(put.call_count, 1) self.assertEqual(len(put.call_args[0]), 2) self.assertEqual(put.call_args[1], kwargs) @@ -3256,13 +3238,12 @@ def test_import_uploads_upload_ids(self): """ kwargs = {'kwarg': gen_integer()} upload_ids = [gen_string('numeric')] - with mock.patch.object(client, 'put') as put: - with mock.patch.object( - entities, - '_handle_response', - return_value={'status': 'success'}, - ) as handler: - response = self.repo.import_uploads(upload_ids=upload_ids, **kwargs) + with mock.patch.object(client, 'put') as put, mock.patch.object( + entities, + '_handle_response', + return_value={'status': 'success'}, + ) as handler: + response = self.repo.import_uploads(upload_ids=upload_ids, **kwargs) self.assertEqual(put.call_count, 1) self.assertEqual(len(put.call_args[0]), 2) self.assertEqual(put.call_args[1], kwargs) diff --git a/tests/test_entity_fields.py b/tests/test_entity_fields.py index 5f330eea..d0496610 100644 --- a/tests/test_entity_fields.py +++ b/tests/test_entity_fields.py @@ -1,7 +1,7 @@ """Unit tests for :mod:`nailgun.entity_fields`.""" import datetime -import socket from random import randint +import socket from unittest import TestCase from urllib.parse import urlparse diff --git a/tests/test_entity_mixins.py b/tests/test_entity_mixins.py index f92d11f2..c2538b93 100644 --- a/tests/test_entity_mixins.py +++ b/tests/test_entity_mixins.py @@ -1,19 +1,18 @@ """Tests for :mod:`nailgun.entity_mixins`.""" import http.client as http_client -from unittest import mock -from unittest import TestCase +from unittest import TestCase, mock from fauxfactory import gen_integer from requests.exceptions import HTTPError -from nailgun import client -from nailgun import config -from nailgun import entity_mixins -from nailgun.entity_fields import IntegerField -from nailgun.entity_fields import ListField -from nailgun.entity_fields import OneToManyField -from nailgun.entity_fields import OneToOneField -from nailgun.entity_fields import StringField +from nailgun import client, config, entity_mixins +from nailgun.entity_fields import ( + IntegerField, + ListField, + OneToManyField, + OneToOneField, + StringField, +) # The size of this module is a direct reflection of the size of module # `nailgun.entity_mixins`. It would be good to split that module up, then split @@ -27,6 +26,15 @@ # # 1. Entity definitions. ------------------------------------------------- {{{1 +# Due to the length of the with statements, nested is preferred over combined +# ruff: noqa: SIM117 + +# Tests use a unused nailgun and tests imports +# ruff: noqa: F401 + +# Cannot use ast.literal_eval because ServerConfig isn't a basic type +# ruff: noqa: S307 + class SampleEntity(entity_mixins.Entity): """Sample entity to be used in the tests""" @@ -155,7 +163,7 @@ def setUp(self): def test_pass_in_emtpy_iterable(self): """Let the ``entity_objs_and_ids`` argument be an empty iterable.""" - for iterable in ([], tuple()): + for iterable in ([], ()): self.assertEqual( entity_mixins._make_entities_from_ids(SampleEntity, iterable, self.cfg), [], @@ -409,7 +417,6 @@ def test_eq(self): # Testing List nested objects mary.list = [alice] self.assertNotEqual(mary, mary_clone) - # noqa mary_clone.list = [alice_clone] self.assertEqual(mary, mary_clone) @@ -470,8 +477,8 @@ def test_repr_v1(self): config.ServerConfig.get() except (KeyError, config.ConfigFileError): self.cfg.save() - import nailgun # noqa - import tests # noqa + import nailgun + import tests self.assertEqual(repr(eval(repr(entity))), target) @@ -490,8 +497,8 @@ def test_repr_v2(self): config.ServerConfig.get() except (KeyError, config.ConfigFileError): self.cfg.save() - import nailgun # noqa - import tests # noqa + import nailgun + import tests self.assertEqual(repr(eval(repr(entity))), target) @@ -514,8 +521,8 @@ def test_repr_v3(self): config.ServerConfig.get() except (KeyError, config.ConfigFileError): self.cfg.save() - import nailgun # noqa - import tests # noqa + import nailgun + import tests self.assertEqual(repr(eval(repr(entity))), target) @@ -958,9 +965,8 @@ def test_delete_v1(self): entity_mixins.EntityDeleteMixin, 'delete_raw', return_value=response, - ): - with self.assertRaises(HTTPError): - self.entity.delete() + ), self.assertRaises(HTTPError): + self.entity.delete() def test_delete_v2(self): """What happens if the server returns an HTTP ACCEPTED status code?""" @@ -971,9 +977,8 @@ def test_delete_v2(self): entity_mixins.EntityDeleteMixin, 'delete_raw', return_value=response, - ) as delete_raw: - with mock.patch.object(entity_mixins, '_poll_task') as poll_task: - self.entity.delete() + ) as delete_raw, mock.patch.object(entity_mixins, '_poll_task') as poll_task: + self.entity.delete() self.assertEqual(delete_raw.call_count, 1) self.assertEqual(poll_task.call_count, 1) self.assertEqual( @@ -989,9 +994,8 @@ def test_delete_v3(self): entity_mixins.EntityDeleteMixin, 'delete_raw', return_value=response, - ): - with mock.patch.object(entity_mixins, '_poll_task') as poll_task: - self.assertEqual(self.entity.delete(), None) + ), mock.patch.object(entity_mixins, '_poll_task') as poll_task: + self.assertEqual(self.entity.delete(), None) self.assertEqual(poll_task.call_count, 0) def test_delete_v4(self): @@ -1016,9 +1020,8 @@ def test_delete_v5(self): entity_mixins.EntityDeleteMixin, 'delete_raw', return_value=response, - ): - with mock.patch.object(entity_mixins, '_poll_task') as poll_task: - self.assertEqual(self.entity.delete(), None) + ), mock.patch.object(entity_mixins, '_poll_task') as poll_task: + self.assertEqual(self.entity.delete(), None) self.assertEqual(poll_task.call_count, 0) def test_delete_v6(self): @@ -1033,9 +1036,8 @@ def test_delete_v6(self): entity_mixins.EntityDeleteMixin, 'delete_raw', return_value=response, - ): - with mock.patch.object(entity_mixins, '_poll_task') as poll_task: - self.assertEqual(self.entity.delete(), None) + ), mock.patch.object(entity_mixins, '_poll_task') as poll_task: + self.assertEqual(self.entity.delete(), None) self.assertEqual(poll_task.call_count, 0) @@ -1227,12 +1229,11 @@ def test_search_filter_v2(self): """ for filter_ in ({'one': 'foo'}, {'many': 'bar'}): - with self.subTest(filter_): - with self.assertRaises(NotImplementedError): - entity_mixins.EntitySearchMixin.search_filter( - [EntityWithSearch2(self.cfg)], - filter_, - ) + with self.subTest(filter_), self.assertRaises(NotImplementedError): + entity_mixins.EntitySearchMixin.search_filter( + [EntityWithSearch2(self.cfg)], + filter_, + ) def test_search_filter_v3(self): """Test :meth:`nailgun.entity_mixins.EntitySearchMixin.search_filter`.