From 1964295c83896cb95a30f9663790ceb9aeea9cbb Mon Sep 17 00:00:00 2001 From: itdependsnetworks Date: Sun, 10 Sep 2023 13:10:58 -0400 Subject: [PATCH 01/13] First pass at 2.0 migration (for nautobot) and nornir-nautobot 3.0 --- .github/workflows/ci.yml | 22 +- docs/dev/CHANGELOG.md | 61 +- docs/dev/CONTRIBUTING.md | 6 +- docs/requirements.txt | 8 +- docs/task/task.md | 6 +- .../plugins/tasks/dispatcher/__init__.py | 79 +- .../plugins/tasks/dispatcher/arista_eos.py | 12 +- .../plugins/tasks/dispatcher/cisco_asa.py | 12 +- .../plugins/tasks/dispatcher/cisco_ios.py | 12 +- .../plugins/tasks/dispatcher/cisco_ios_xr.py | 9 - .../plugins/tasks/dispatcher/cisco_nxos.py | 12 +- .../plugins/tasks/dispatcher/cisco_xr.py | 13 + .../plugins/tasks/dispatcher/default.py | 292 ++-- .../plugins/tasks/dispatcher/juniper_junos.py | 12 +- .../tasks/dispatcher/mikrotik_routeros.py | 126 +- .../tasks/dispatcher/mikrotik_routeros_api.py | 148 -- .../tasks/dispatcher/ruckus_fastiron.py | 6 +- ...s_smartzone_api.py => ruckus_smartzone.py} | 106 +- nornir_nautobot/utils/helpers.py | 15 + nornir_nautobot/utils/logger.py | 42 +- poetry.lock | 1472 ++++++++++------- pyproject.toml | 29 +- tasks.py | 225 ++- 23 files changed, 1439 insertions(+), 1286 deletions(-) delete mode 100644 nornir_nautobot/plugins/tasks/dispatcher/cisco_ios_xr.py create mode 100644 nornir_nautobot/plugins/tasks/dispatcher/cisco_xr.py delete mode 100644 nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros_api.py rename nornir_nautobot/plugins/tasks/dispatcher/{ruckus_smartzone_api.py => ruckus_smartzone.py} (66%) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 95fbd2d..9c1bd30 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -14,7 +14,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9.7" + python-version: "3.11" - name: "Install Python Packages for Linting" run: "pip install --upgrade pip wheel invoke toml black" - name: "Linting: Black" @@ -29,7 +29,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9" + python-version: "3.11" - name: "Install Python Packages for Linting" run: "pip install --upgrade pip wheel invoke toml bandit" - name: "Linting: Bandit" @@ -46,7 +46,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9" + python-version: "3.11" - name: "Install Python Packages for Linting" run: "pip install --upgrade pip wheel invoke toml pydocstyle" - name: "Linting: pydocstyle" @@ -63,7 +63,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9" + python-version: "3.11" - name: "Install Python Packages for Linting" run: "pip install --upgrade pip wheel invoke toml flake8" - name: "Linting: flake8" @@ -80,7 +80,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9" + python-version: "3.11" - name: "Install Python Packages for Linting" run: "pip install --upgrade pip wheel invoke toml yamllint" - name: "Linting: yamllint" @@ -95,7 +95,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9" + python-version: "3.11" - name: "Install Python Packages for Setup" run: "pip install invoke poetry toml" - name: "Build Container" @@ -113,7 +113,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9" + python-version: "3.11" - name: "Install Python Packages for Linting" run: "pip install invoke poetry toml" - name: "Build Container" @@ -126,7 +126,7 @@ jobs: strategy: fail-fast: true matrix: - python-version: ["3.7", "3.8", "3.9", "3.10"] + python-version: ["3.8", "3.9", "3.10", "3.11"] runs-on: "ubuntu-20.04" env: PYTHON_VER: "${{ matrix.python-version }}" @@ -136,7 +136,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9" + python-version: "3.11" - name: "Install Python Packages for Linting" run: "pip install invoke poetry toml" - name: "Build Container" @@ -155,7 +155,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9" + python-version: "3.11" - name: "Install Python Packages for Linting" run: "pip install invoke poetry toml" - name: "Set env" @@ -184,7 +184,7 @@ jobs: - name: "Set up Python" uses: "actions/setup-python@v2" with: - python-version: "3.9" + python-version: "3.11" - name: "Install Python Packages" run: "pip install poetry" - name: "Set env" diff --git a/docs/dev/CHANGELOG.md b/docs/dev/CHANGELOG.md index b1d3653..5d3dd40 100644 --- a/docs/dev/CHANGELOG.md +++ b/docs/dev/CHANGELOG.md @@ -1,26 +1,31 @@ # Changelog +Remove legacy dispatcher method +error codes + + + ## v2.6.0 -- (#96) Changes backup_file to be conditional for get_config -- (#98) Adds merge_config method +- [#96](https://github.com/nautobot/nornir-nautobot/pull/96) Changes backup_file to be conditional for get_config +- [#98](https://github.com/nautobot/nornir-nautobot/pull/98) Adds merge_config method ## v2.5.0 -- (#93) Updates Nornir-Netmiko to 1.0.0 release -- (#97) Adds Jinja Environment option to generate_config +- [#93](https://github.com/nautobot/nornir-nautobot/pull/93) Updates Nornir-Netmiko to 1.0.0 release +- [#97](https://github.com/nautobot/nornir-nautobot/pull/97) Adds Jinja Environment option to generate_config ## v2.4.0 -- (#61) Be more clear on error messages by @itdependsnetworks -- (#66) Add basic typing to the methods in logger by @KalleDK -- (#75) Documentation refactor by @susanhooks -- (#77) added ICX/Fastiron Nornir Driver by @pato23arg -- (#78) Fix RTD docs build by @cmsirbu -- (#83) Adds provision_config method by @joewesch -- (#76) Mikrotik RouterOS CLI Support by @pato23arg -- (#79) Mikrotik RouterOS API support by @pato23arg -- (#85) Ruckus Smartzone WLC and Access Point Driver by @pato23arg +- [#61](https://github.com/nautobot/nornir-nautobot/pull/61) Be more clear on error messages by @itdependsnetworks +- [#66](https://github.com/nautobot/nornir-nautobot/pull/66) Add basic typing to the methods in logger by @KalleDK +- [#75](https://github.com/nautobot/nornir-nautobot/pull/75) Documentation refactor by @susanhooks +- [#77](https://github.com/nautobot/nornir-nautobot/pull/77) added ICX/Fastiron Nornir Driver by @pato23arg +- [#78](https://github.com/nautobot/nornir-nautobot/pull/78) Fix RTD docs build by @cmsirbu +- [#83](https://github.com/nautobot/nornir-nautobot/pull/83) Adds provision_config method by @joewesch +- [#76](https://github.com/nautobot/nornir-nautobot/pull/76) Mikrotik RouterOS CLI Support by @pato23arg +- [#79](https://github.com/nautobot/nornir-nautobot/pull/79) Mikrotik RouterOS API support by @pato23arg +- [#85](https://github.com/nautobot/nornir-nautobot/pull/85) Ruckus Smartzone WLC and Access Point Driver by @pato23arg ## New Contributors * @KalleDK made their first contribution in https://github.com/nautobot/nornir-nautobot/pull/66 @@ -31,41 +36,41 @@ ## v2.3.0 -- (#67) fix pylint, tests, and drop py36 support #67 +- [#67](https://github.com/nautobot/nornir-nautobot/pull/67) fix pylint, tests, and drop py36 support #67 ## v2.2.0 -- (#41) Added jinja2 filter pass to generate config -- (#42) Added Cisco ASA mapping to default mapping +- [#41](https://github.com/nautobot/nornir-nautobot/pull/41) Added jinja2 filter pass to generate config +- [#42](https://github.com/nautobot/nornir-nautobot/pull/42) Added Cisco ASA mapping to default mapping ## v2.1.2 -- (#40) Fix nornir inventory defaults +- [#40](https://github.com/nautobot/nornir-nautobot/pull/40) Fix nornir inventory defaults ## v2.1.1 -- (#31) Migrate from Travis-CI to GitHub Actions for CI -- (#32) Fix data population when generating configurations from the dispatcher +- [#31](https://github.com/nautobot/nornir-nautobot/pull/31) Migrate from Travis-CI to GitHub Actions for CI +- [#32](https://github.com/nautobot/nornir-nautobot/pull/32) Fix data population when generating configurations from the dispatcher ## v2.1.0 -- (#26) Updates in poetry packaging for pre 1.0.0 dependencies in Nornir to allow all new up to 1.0.0 releases +- [#26](https://github.com/nautobot/nornir-nautobot/pull/26) Updates in poetry packaging for pre 1.0.0 dependencies in Nornir to allow all new up to 1.0.0 releases ## v2.0.3 -- (#24) Change import mechanism / changes deprecated function +- [#24](https://github.com/nautobot/nornir-nautobot/pull/24) Change import mechanism / changes deprecated function ## v2.0.1 -- (#20) Fixes inventory failure when platform is not defined for a device +- [#20](https://github.com/nautobot/nornir-nautobot/pull/20) Fixes inventory failure when platform is not defined for a device ## v2.0.0 -- (#18) Migrates functions to new NTC netutils library, which is removing methods previously available: - - compliance - - make_folder - - hostname_resolves - - test_tcp_port - - is_ip +- [#18](https://github.com/nautobot/nornir-nautobot/pull/18) Migrates functions to new NTC netutils library, which is removing methods previously available: + - compliance + - make_folder + - hostname_resolves + - test_tcp_port + - is_ip ## v0.1.0 - 2020-12-27 diff --git a/docs/dev/CONTRIBUTING.md b/docs/dev/CONTRIBUTING.md index efe7170..cb22ef6 100644 --- a/docs/dev/CONTRIBUTING.md +++ b/docs/dev/CONTRIBUTING.md @@ -10,7 +10,7 @@ This section describes how to install *nornir_nautobot* for development, how to ## Python Versions -This leverages Python3.6 or later. All features will be tested against 3.6 - 3.9 versions of Python. +This leverages Python3.8 or later. All features will be tested against 3.8 - 3.11 versions of Python. ## Versioning @@ -27,7 +27,7 @@ Follow these steps to set up your local development environment: ```bash # Double check your version $ python --version -Python 3.7.7 +Python 3.11.1 # Activate the Poetry environment, which will auto create the virtual environment related to the project $ poetry shell # Install project dependencies as well as development dependencies @@ -61,7 +61,7 @@ All tests should be located within the `tests\` directory with `tests\unit` for The following linting tasks are required: * [Bandit](https://bandit.readthedocs.io/en/latest/) - * Basic security tests, should be run on Python3.6 or Python3.7 + * Basic security tests, should be run on Python3.11 * [Black code style](https://github.com/psf/black) * Code formatting with version 20.8b1. There are some differences in the format between versions 19 and 20. * [Flake8](https://flake8.pycqa.org/en/latest/) diff --git a/docs/requirements.txt b/docs/requirements.txt index 53cab3c..fb4d95c 100644 --- a/docs/requirements.txt +++ b/docs/requirements.txt @@ -1,5 +1,5 @@ -mkdocs==1.3.1 -mkdocs-material==8.3.9 -mkdocstrings==0.19 -mkdocstrings-python==0.7.1 +mkdocs==1.5.2 +mkdocs-material==9.2.4 +mkdocstrings==0.22.0 +mkdocstrings-python==1.5.2 mkdocs-version-annotations==1.0.0 \ No newline at end of file diff --git a/docs/task/task.md b/docs/task/task.md index bdbb26d..af730a3 100644 --- a/docs/task/task.md +++ b/docs/task/task.md @@ -12,7 +12,7 @@ The only task plugin currently is the "dispatcher" plugin. This plugin dispatche try: driver_task = getattr(driver_class, method) except AttributeError: - logger.log_failure(obj, f"Unable to locate the method {method} for {driver}") + logger.log_error(obj, f"Unable to locate the method {method} for {driver}") raise NornirNautobotException() result = task.run(task=driver_task, *args, **kwargs) @@ -24,8 +24,8 @@ The only task plugin currently is the "dispatcher" plugin. This plugin dispatche class NautobotNornirDriver: """Default collection of Nornir Tasks based on Napalm.""" - @staticmethod - def get_config(task: Task, backup_file: str, *args, **kwargs) -> Result: + @classmethod + def get_config(cls, task: Task, backup_file: str, *args, **kwargs) -> Result: ``` ## Calling Dispatcher diff --git a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py index b52f206..72352f4 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py @@ -1,73 +1,58 @@ -"""Used to intialize the dispatcher.""" +"""Used to initialize the dispatcher.""" # pylint: disable=raise-missing-from -import importlib import logging from nornir.core.task import Result, Task from nornir_nautobot.exceptions import NornirNautobotException +from nornir_nautobot.utils.helpers import snake_to_title_case, import_string LOGGER = logging.getLogger(__name__) +PATH_ROOT = "nornir_nautobot.plugins.tasks.dispatcher.default" -_DEFAULT_DRIVERS_MAPPING = { - "default": "nornir_nautobot.plugins.tasks.dispatcher.default.NautobotNornirDriver", - "default_netmiko": "nornir_nautobot.plugins.tasks.dispatcher.default.NetmikoNautobotNornirDriver", - "cisco_asa": "nornir_nautobot.plugins.tasks.dispatcher.cisco_asa.NautobotNornirDriver", - "cisco_nxos": "nornir_nautobot.plugins.tasks.dispatcher.cisco_nxos.NautobotNornirDriver", - "cisco_ios": "nornir_nautobot.plugins.tasks.dispatcher.cisco_ios.NautobotNornirDriver", - "cisco_xr": "nornir_nautobot.plugins.tasks.dispatcher.cisco_ios_xr.NautobotNornirDriver", - "juniper_junos": "nornir_nautobot.plugins.tasks.dispatcher.juniper_junos.NautobotNornirDriver", - "arista_eos": "nornir_nautobot.plugins.tasks.dispatcher.arista_eos.NautobotNornirDriver", - "mikrotik_routeros_api": "nornir_nautobot.plugins.tasks.dispatcher.mikrotik_routeros_api.NautobotNornirDriver", - "ruckus_fastiron": "nornir_nautobot.plugins.tasks.dispatcher.ruckus_fastiron.NautobotNornirDriver", - "mikrotik_routeros": "nornir_nautobot.plugins.tasks.dispatcher.mikrotik_routeros.NautobotNornirDriver", - "ruckus_smartzone_api": "nornir_nautobot.plugins.tasks.dispatcher.ruckus_smartzone_api.NautobotNornirDriver", - "ruckus_access_point": "nornir_nautobot.plugins.tasks.dispatcher.ruckus_smartzone_api.NautobotNornirDriver", -} - -def dispatcher(task: Task, method: str, logger, obj, *args, **kwargs) -> Result: +def dispatcher( # pylint: disable=too-many-arguments,too-many-locals + task: Task, method: str, logger, obj, framework, *args, **kwargs +) -> Result: """Helper Task to retrieve a given Nornir task for a given platform. Args: - task (Nornir Task): Nornir Task object. - method (str): The string value of the method to dynamically find. + task: Nornir Task object. + method: The string value of the method to dynamically find. Returns: - Result: Nornir Task result. + Result: Nornir Task result object. """ - if kwargs.get("default_drivers_mapping"): - default_drivers_mapping = kwargs["default_drivers_mapping"] - del kwargs["default_drivers_mapping"] + if not kwargs.get("custom_dispatcher"): + custom_dispatcher = {} + logger.log_debug(f"Dispatcher process started for {task.host.name} ({task.host.platform.network_driver})") + + network_driver = task.host.platform.network_driver + network_driver_title = snake_to_title_case(network_driver) + custom_dispatcher_path = [custom_dispatcher.get(network_driver)] + framework_path = f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework}{network_driver_title}" + framework_default_path = f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework}Default" + + if custom_dispatcher.get(network_driver): + driver_class = import_string(custom_dispatcher_path) + checked_path = [custom_dispatcher_path] + elif import_string(framework_path): + driver_class = import_string(framework_path) else: - default_drivers_mapping = _DEFAULT_DRIVERS_MAPPING - - logger.log_debug(f"Executing dispatcher for {task.host.name} ({task.host.platform})") - - # Get the platform specific driver, if not available, get the default driver - driver = default_drivers_mapping.get(task.host.platform, default_drivers_mapping.get("default")) - logger.log_debug(f"Found driver {driver}") - - if not driver: - logger.log_failure( - obj, f"Unable to find the driver for {method} for platform: {task.host.platform}, preemptively failed." - ) - raise NornirNautobotException( - f"Unable to find the driver for {method} for platform: {task.host.platform}, preemptively failed." - ) - - module_name, class_name = driver.rsplit(".", 1) - driver_class = getattr(importlib.import_module(module_name), class_name) + driver_class = import_string(framework_default_path) + checked_path = [framework_path, framework_default_path] if not driver_class: - logger.log_failure(obj, f"Unable to locate the class {driver}, preemptively failed.") - raise NornirNautobotException(f"Unable to locate the class {driver}, preemptively failed.") + error_msg = f"E1001: Did not find a valid dispatcher in {checked_path}, preemptively failed." + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) try: driver_task = getattr(driver_class, method) except AttributeError: - logger.log_failure(obj, f"Unable to locate the method {method} for {driver}, preemptively failed.") - raise NornirNautobotException(f"Unable to locate the method {method} for {driver}, preemptively failed.") + error_msg = f"E1002: Unable to locate the method {method} for {driver_class}, preemptively failed." + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) result = task.run(task=driver_task, logger=logger, obj=obj, *args, **kwargs) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/arista_eos.py b/nornir_nautobot/plugins/tasks/dispatcher/arista_eos.py index bcb32e6..e70c881 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/arista_eos.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/arista_eos.py @@ -1,9 +1,13 @@ -"""network_importer driver for arista_eos.""" +"""nornir dispatcher for arista_eos.""" -from .default import NautobotNornirDriver as DefaultNautobotNornirDriver +from nornir_nautobot.plugins.tasks.dispatcher.default import NapalmDefault, NetmikoDefault -class NautobotNornirDriver(DefaultNautobotNornirDriver): - """Collection of Nornir Tasks specific to Arista EOS devices.""" +class NapalmAristaEos(NapalmDefault): + """Collection of Napalm Nornir Tasks specific to Arista EOS devices.""" + + +class NetmikoAristaEos(NetmikoDefault): + """Collection of Netmiko Nornir Tasks specific to Arista EOS devices.""" config_command = "show run" diff --git a/nornir_nautobot/plugins/tasks/dispatcher/cisco_asa.py b/nornir_nautobot/plugins/tasks/dispatcher/cisco_asa.py index 11e7958..39211fa 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/cisco_asa.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/cisco_asa.py @@ -1,9 +1,13 @@ -"""network_importer driver for cisco_asa.""" +"""nornir dispatcher for cisco_asa.""" -from .default import NetmikoNautobotNornirDriver as DefaultNautobotNornirDriver +from nornir_nautobot.plugins.tasks.dispatcher.default import NapalmDefault, NetmikoDefault -class NautobotNornirDriver(DefaultNautobotNornirDriver): - """Driver for Cisco ASA.""" +class NapalmCiscoAsa(NapalmDefault): + """Collection of Napalm Nornir Tasks specific to Cisco ASA devices.""" + + +class NetmikoCiscoAsa(NetmikoDefault): + """Collection of Netmiko Nornir Tasks specific to Cisco ASA devices.""" config_command = "show run" diff --git a/nornir_nautobot/plugins/tasks/dispatcher/cisco_ios.py b/nornir_nautobot/plugins/tasks/dispatcher/cisco_ios.py index 86e8d01..667c793 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/cisco_ios.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/cisco_ios.py @@ -1,9 +1,13 @@ -"""network_importer driver for cisco IOS.""" +"""nornir dispatcher for cisco IOS.""" -from .default import NetmikoNautobotNornirDriver as DefaultNautobotNornirDriver +from nornir_nautobot.plugins.tasks.dispatcher.default import NapalmDefault, NetmikoDefault -class NautobotNornirDriver(DefaultNautobotNornirDriver): - """Driver for Cisco IOS.""" +class NapalmCiscoIos(NapalmDefault): + """Collection of Napalm Nornir Tasks specific to Cisco IOS devices.""" + + +class NetmikoCiscoIos(NetmikoDefault): + """Collection of Netmiko Nornir Tasks specific to Cisco IOS devices.""" config_command = "show run" diff --git a/nornir_nautobot/plugins/tasks/dispatcher/cisco_ios_xr.py b/nornir_nautobot/plugins/tasks/dispatcher/cisco_ios_xr.py deleted file mode 100644 index da021d1..0000000 --- a/nornir_nautobot/plugins/tasks/dispatcher/cisco_ios_xr.py +++ /dev/null @@ -1,9 +0,0 @@ -"""network_importer driver for cisco IOS-XR.""" - -from .default import NetmikoNautobotNornirDriver as DefaultNautobotNornirDriver - - -class NautobotNornirDriver(DefaultNautobotNornirDriver): - """Driver for Cisco IOS-XR.""" - - config_command = "show run" diff --git a/nornir_nautobot/plugins/tasks/dispatcher/cisco_nxos.py b/nornir_nautobot/plugins/tasks/dispatcher/cisco_nxos.py index e8d58a1..3e32c7e 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/cisco_nxos.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/cisco_nxos.py @@ -1,9 +1,13 @@ -"""network_importer driver for cisco NXOS.""" +"""nornir dispatcher for cisco NXOS.""" -from .default import NetmikoNautobotNornirDriver as DefaultNautobotNornirDriver +from nornir_nautobot.plugins.tasks.dispatcher.default import NapalmDefault, NetmikoDefault -class NautobotNornirDriver(DefaultNautobotNornirDriver): - """Driver for Cisco NXOS.""" +class NapalmCiscoNxos(NapalmDefault): + """Collection of Napalm Nornir Tasks specific to Cisco NXOS devices.""" + + +class NetmikoCiscoNxos(NetmikoDefault): + """Collection of Netmiko Nornir Tasks specific to Cisco NXOS devices.""" config_command = "show run" diff --git a/nornir_nautobot/plugins/tasks/dispatcher/cisco_xr.py b/nornir_nautobot/plugins/tasks/dispatcher/cisco_xr.py new file mode 100644 index 0000000..f5386db --- /dev/null +++ b/nornir_nautobot/plugins/tasks/dispatcher/cisco_xr.py @@ -0,0 +1,13 @@ +"""nornir dispatcher for cisco XR.""" + +from nornir_nautobot.plugins.tasks.dispatcher.default import NapalmDefault, NetmikoDefault + + +class NapalmCiscoXr(NapalmDefault): + """Collection of Napalm Nornir Tasks specific to Cisco XR devices.""" + + +class NetmikoCiscoXr(NetmikoDefault): + """Collection of Netmiko Nornir Tasks specific to Cisco XR devices.""" + + config_command = "show run" diff --git a/nornir_nautobot/plugins/tasks/dispatcher/default.py b/nornir_nautobot/plugins/tasks/dispatcher/default.py index 50af5c0..254a9fe 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/default.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/default.py @@ -8,10 +8,7 @@ import jinja2 -try: - from netmiko.ssh_exception import NetmikoAuthenticationException, NetmikoTimeoutException -except ImportError: - from netmiko import NetmikoAuthenticationException, NetmikoTimeoutException +from netmiko import NetmikoAuthenticationException, NetmikoTimeoutException from netutils.config.clean import clean_config, sanitize_config from netutils.config.compliance import compliance @@ -31,64 +28,17 @@ _logger = logging.getLogger(__name__) -class NautobotNornirDriver: - """Default collection of Nornir Tasks based on Napalm.""" +class DispatcherMixin: + """Mixin for non-network driver related tasks.""" - config_command = "show run" + tcp_port = 22 @classmethod - def get_config( - cls, task: Task, logger, obj, backup_file: str, remove_lines: list, substitute_lines: list - ) -> Result: - """Get the latest configuration from the device. + def _get_hostname(cls, task: Task, obj) -> str: # pylint: disable=unused-argument + return task.host.hostname - Args: - task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. - obj (Device): A Nautobot Device Django ORM object instance. - backup_file (str): The file location of where the back configuration should be saved. - remove_lines (list): A list of regex lines to remove configurations. - substitute_lines (list): A list of dictionaries with to remove and replace lines. - - Returns: - Result: Nornir Result object with a dict as a result containing the running configuration - { "config: } - """ - logger.log_debug(f"Executing get_config for {task.host.name} on {task.host.platform}") - - # TODO: Find standard napalm exceptions and account for them - try: - result = task.run(task=napalm_get, getters=["config"], retrieve="running") - except NornirSubTaskError as exc: - logger.log_failure(obj, f"`get_config` method failed with an unexpected issue: `{exc.result.exception}`") - raise NornirNautobotException( - "`get_config` method failed with an unexpected issue: `{exc.result.exception}`" - ) - - if result[0].failed: - logger.log_failure( - obj, f"`get_config` nornir task failed with an unexpected issue: `{str(result.exception)}`" - ) - return result - - running_config = result[0].result.get("config", {}).get("running", None) - if remove_lines: - logger.log_debug("Removing lines from configuration based on `remove_lines` definition") - running_config = clean_config(running_config, remove_lines) - - if substitute_lines: - logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") - running_config = sanitize_config(running_config, substitute_lines) - - if backup_file: - make_folder(os.path.dirname(backup_file)) - - with open(backup_file, "w", encoding="utf8") as filehandler: - filehandler.write(running_config) - return Result(host=task.host, result={"config": running_config}) - - @staticmethod - def check_connectivity(task: Task, logger, obj) -> Result: + @classmethod + def check_connectivity(cls, task: Task, logger, obj) -> Result: """Check the connectivity to a network device. Args: @@ -99,31 +49,38 @@ def check_connectivity(task: Task, logger, obj) -> Result: Returns: Result: Nornir Result object. """ - if is_ip(task.host.hostname): - ip_addr = task.host.hostname + hostname = cls._get_hostname(task, obj) + if is_ip(hostname): + ip_addr = hostname else: - if not is_fqdn_resolvable(task.host.hostname): - logger.log_failure(obj, "There was not an IP or resolvable, preemptively failed.") - raise NornirNautobotException("There was not an IP or resolvable, preemptively failed.") - ip_addr = socket.gethostbyname(task.host.hostname) + if not is_fqdn_resolvable(hostname): + error_msg = ( + f"E1003: The hostname {hostname} did not have an IP nor was resolvable, preemptively failed." + ) + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + ip_addr = socket.gethostbyname(hostname) # TODO: Allow port to be configurable - port = 22 + port = cls.tcp_port if not tcp_ping(ip_addr, port): - logger.log_failure(obj, f"Could not connect to IP: {ip_addr} and port: {port}, preemptively failed.") - raise NornirNautobotException(f"Could not connect to IP: {ip_addr} and port: {port}, preemptively failed.") + error_msg = f"E1004: Could not connect to IP: {ip_addr} and port: {port}, preemptively failed." + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) if not task.host.username: - logger.log_failure(obj, "There was no username defined, preemptively failed.") - raise NornirNautobotException("There was no username defined, preemptively failed.") + error_msg = "E1005: There was no username defined, preemptively failed." + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) if not task.host.password: - logger.log_failure(obj, "There was no password defined, preemptively failed.") - raise NornirNautobotException("There was no password defined, preemptively failed.") + error_msg = "E1006: There was no password defined, preemptively failed." + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) return Result(host=task.host) - @staticmethod + @classmethod def compliance_config( - task: Task, logger, obj, features: str, backup_file: str, intended_file: str, platform: str + cls, task: Task, logger, obj, features: str, backup_file: str, intended_file: str, platform: str ) -> Result: """Compare two configurations against each other. @@ -140,26 +97,26 @@ def compliance_config( Result: Nornir Result object with a feature_data key of the compliance data. """ if not os.path.exists(backup_file): - logger.log_failure(obj, f"Backup file Not Found at location: `{backup_file}`, preemptively failed.") - raise NornirNautobotException(f"Backup file Not Found at location: `{backup_file}`, preemptively failed.") + error_msg = f"E1007: Backup file Not Found at location: `{backup_file}`, preemptively failed." + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) if not os.path.exists(intended_file): - logger.log_failure( - obj, f"Intended config file NOT Found at location: `{intended_file}`, preemptively failed." - ) - raise NornirNautobotException( - f"Intended config file NOT Found at location: `{intended_file}`, preemptively failed." - ) + error_msg = f"E1008: Intended config file NOT Found at location: `{intended_file}`, preemptively failed." + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) try: feature_data = compliance(features, backup_file, intended_file, platform) except Exception as error: # pylint: disable=broad-except - logger.log_failure(obj, f"UNKNOWN Failure of: {str(error)}") - raise NornirNautobotException(f"UNKNOWN Failure of: {str(error)}") + error_msg = f"E1009: UNKNOWN Failure of: {str(error)}" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) return Result(host=task.host, result={"feature_data": feature_data}) - @staticmethod + @classmethod def generate_config( + cls, task: Task, logger, obj, @@ -195,43 +152,38 @@ def generate_config( )[0].result except NornirSubTaskError as exc: if isinstance(exc.result.exception, jinja2.exceptions.UndefinedError): # pylint: disable=no-else-raise - logger.log_failure( - obj, - f"There was a jinja2.exceptions.UndefinedError error: ``{str(exc.result.exception)}``", - ) - raise NornirNautobotException( - f"There was a jinja2.exceptions.UndefinedError error: ``{str(exc.result.exception)}``" + error_msg = ( + f"E1010: There was a jinja2.exceptions.UndefinedError error: ``{str(exc.result.exception)}``" ) + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + elif isinstance(exc.result.exception, jinja2.TemplateSyntaxError): - logger.log_failure( - obj, - f"There was a jinja2.TemplateSyntaxError error: ``{str(exc.result.exception)}``", - ) - raise NornirNautobotException( - f"There was a jinja2.TemplateSyntaxError error: ``{str(exc.result.exception)}``" - ) + error_msg = (f"E1011: There was a jinja2.TemplateSyntaxError error: ``{str(exc.result.exception)}``",) + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + elif isinstance(exc.result.exception, jinja2.TemplateNotFound): - logger.log_failure( - obj, - f"There was an issue finding the template and a jinja2.TemplateNotFound error was raised: ``{str(exc.result.exception)}``", - ) - raise NornirNautobotException( - f"There was an issue finding the template and a jinja2.TemplateNotFound error was raised: ``{str(exc.result.exception)}``" - ) + error_msg = f"E1012: There was an issue finding the template and a jinja2.TemplateNotFound error was raised: ``{str(exc.result.exception)}``" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + elif isinstance(exc.result.exception, jinja2.TemplateError): - logger.log_failure(obj, f"There was an issue general Jinja error: ``{str(exc.result.exception)}``") - raise NornirNautobotException( - f"There was an issue general Jinja error: ``{str(exc.result.exception)}``" - ) - raise + error_msg = f"E1013: There was an issue general Jinja error: ``{str(exc.result.exception)}``" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + + error_msg = f"E1014: Failed with an unknown issue. `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) make_folder(os.path.dirname(output_file_location)) with open(output_file_location, "w", encoding="utf8") as filehandler: filehandler.write(filled_template) return Result(host=task.host, result={"config": filled_template}) - @staticmethod - def _remove_lines(logger, _running_config: str, remove_lines: list) -> str: + @classmethod + def _remove_lines(cls, logger, _running_config: str, remove_lines: list) -> str: """Removes lines in configuration as specified in Remove Lines list. Args: @@ -247,8 +199,8 @@ def _remove_lines(logger, _running_config: str, remove_lines: list) -> str: logger.log_debug("Removing lines from configuration based on `remove_lines` definition") return clean_config(_running_config, remove_lines) - @staticmethod - def _substitute_lines(logger, _running_config: str, substitute_lines: list) -> str: + @classmethod + def _substitute_lines(cls, logger, _running_config: str, substitute_lines: list) -> str: """Substitutes lines in configuration as specified in substitute Lines list. Args: @@ -264,8 +216,8 @@ def _substitute_lines(logger, _running_config: str, substitute_lines: list) -> s logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") return sanitize_config(_running_config, substitute_lines) - @staticmethod - def _save_file(logger, backup_file: str, _running_config: str) -> None: + @classmethod + def _save_file(cls, logger, backup_file: str, _running_config: str) -> None: """Saves Running Configuration to a specified file. Args: @@ -281,15 +233,64 @@ def _save_file(logger, backup_file: str, _running_config: str) -> None: with open(backup_file, "w", encoding="utf8") as filehandler: filehandler.write(_running_config) - def provision_config(self, *args, **kwargs): - """This method is being deprecated. Please use replace_config instead.""" - _logger.warning( - "WARNING: The method 'provision_config()' will be removed in the next major release. Please use 'replace_config()' instead." - ) - return self.replace_config(*args, **kwargs) - @staticmethod +class NapalmDefault(DispatcherMixin): + """Default collection of Nornir Tasks based on Napalm.""" + + @classmethod + def get_config( + cls, task: Task, logger, obj, backup_file: str, remove_lines: list, substitute_lines: list + ) -> Result: + """Get the latest configuration from the device. + + Args: + task (Task): Nornir Task. + logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + obj (Device): A Nautobot Device Django ORM object instance. + backup_file (str): The file location of where the back configuration should be saved. + remove_lines (list): A list of regex lines to remove configurations. + substitute_lines (list): A list of dictionaries with to remove and replace lines. + + Returns: + Result: Nornir Result object with a dict as a result containing the running configuration + { "config: } + """ + logger.log_debug(f"Executing get_config for {task.host.name} on {task.host.platform}") + + # TODO: Find standard napalm exceptions and account for them + try: + result = task.run(task=napalm_get, getters=["config"], retrieve="running") + except NornirSubTaskError as exc: + error_msg = f"E1015: `get_config` method failed with an unexpected issue: `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + + if result[0].failed: + # TODO: investigate this, is there a better way to handle? recursive function? + logger.log_error( + f"`get_config` nornir task failed with an unexpected issue: `{str(result.exception)}`", extra={"object": obj} + ) + return result + + running_config = result[0].result.get("config", {}).get("running", None) + if remove_lines: + logger.log_debug("Removing lines from configuration based on `remove_lines` definition") + running_config = clean_config(running_config, remove_lines) + + if substitute_lines: + logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") + running_config = sanitize_config(running_config, substitute_lines) + + if backup_file: + make_folder(os.path.dirname(backup_file)) + + with open(backup_file, "w", encoding="utf8") as filehandler: + filehandler.write(running_config) + return Result(host=task.host, result={"config": running_config}) + + @classmethod def replace_config( + cls, task: Task, logger, obj, @@ -311,7 +312,7 @@ def replace_config( Returns: Result: Nornir Result object with a dict as a result containing what changed and the result of the push. """ - logger.log_success(obj, "Config provision starting") + logger.log_info(obj, "Config provision starting") # Sending None to napalm_configure for revert_in will disable it, so we don't want a default value. revert_in = os.getenv("NORNIR_NAUTOBOT_REVERT_IN_SECONDS") if revert_in is not None: @@ -325,15 +326,17 @@ def replace_config( revert_in=revert_in, ) except NornirSubTaskError as exc: - logger.log_failure(obj, f"Failed with an unknown issue. `{exc.result.exception}`") - raise NornirNautobotException() + error_msg = f"E1015: Failed with an unknown issue. `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) - logger.log_success(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") - logger.log_success(obj, "Config provision ended") + logger.log_info(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") + logger.log_info(obj, "Config provision ended") return Result(host=task.host, result={"changed": push_result.changed, "result": push_result[0].result}) - @staticmethod + @classmethod def merge_config( + cls, task: Task, logger, obj, @@ -355,7 +358,7 @@ def merge_config( Returns: Result: Nornir Result object with a dict as a result containing what changed and the result of the push. """ - logger.log_success(obj, "Config merge starting") + logger.log_info(obj, "Config merge starting") # Sending None to napalm_configure for revert_in will disable it, so we don't want a default value. revert_in = os.getenv("NORNIR_NAUTOBOT_REVERT_IN_SECONDS") if revert_in is not None: @@ -369,15 +372,16 @@ def merge_config( revert_in=revert_in, ) except NornirSubTaskError as exc: - logger.log_failure(obj, f"Failed with an unknown issue. `{exc.result.exception}`") - raise NornirNautobotException() + error_msg = f"E1015: Failed with an unknown issue. `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) - logger.log_success(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") - logger.log_success(obj, "Config merge ended") + logger.log_info(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") + logger.log_info(obj, "Config merge ended") return Result(host=task.host, result={"changed": push_result.changed, "result": push_result[0].result}) -class NetmikoNautobotNornirDriver(NautobotNornirDriver): +class NetmikoDefault(DispatcherMixin): """Default collection of Nornir Tasks based on Netmiko.""" config_command = "show run" @@ -406,15 +410,18 @@ def get_config( result = task.run(task=netmiko_send_command, command_string=command) except NornirSubTaskError as exc: if isinstance(exc.result.exception, NetmikoAuthenticationException): - logger.log_failure(obj, f"Failed with an authentication issue: `{exc.result.exception}`") - raise NornirNautobotException(f"Failed with an authentication issue: `{exc.result.exception}`") + error_msg = f"E1017: Failed with an authentication issue: `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) if isinstance(exc.result.exception, NetmikoTimeoutException): - logger.log_failure(obj, f"Failed with a timeout issue. `{exc.result.exception}`") - raise NornirNautobotException(f"Failed with a timeout issue. `{exc.result.exception}`") + error_msg = f"E1018: Failed with a timeout issue. `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) - logger.log_failure(obj, f"Failed with an unknown issue. `{exc.result.exception}`") - raise NornirNautobotException(f"Failed with an unknown issue. `{exc.result.exception}`") + error_msg = f"E1016: Failed with an unknown issue. `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) if result[0].failed: return result @@ -423,8 +430,9 @@ def get_config( # Primarily seen in Cisco devices. if "ERROR: % Invalid input detected at" in running_config: - logger.log_failure(obj, "Discovered `ERROR: % Invalid input detected at` in the output") - raise NornirNautobotException("Discovered `ERROR: % Invalid input detected at` in the output") + error_msg = "E1019: Discovered `ERROR: % Invalid input detected at` in the output" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) if remove_lines: logger.log_debug("Removing lines from configuration based on `remove_lines` definition") diff --git a/nornir_nautobot/plugins/tasks/dispatcher/juniper_junos.py b/nornir_nautobot/plugins/tasks/dispatcher/juniper_junos.py index c36c845..6df5be0 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/juniper_junos.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/juniper_junos.py @@ -1,9 +1,13 @@ -"""default network_importer driver for Juniper.""" +"""nornir dispatcher for Juniper Junos.""" -from .default import NetmikoNautobotNornirDriver as DefaultNautobotNornirDriver +from nornir_nautobot.plugins.tasks.dispatcher.default import NapalmDefault, NetmikoDefault -class NautobotNornirDriver(DefaultNautobotNornirDriver): - """Collection of Nornir Tasks specific to Juniper Junos devices.""" +class NapalmJuniperJunos(NapalmDefault): + """Collection of Napalm Nornir Tasks specific to Juniper JUNOS devices.""" + + +class NetmikoJuniperJunos(NetmikoDefault): + """Collection of Netmiko Nornir Tasks specific to Juniper JUNOS devices.""" config_command = "show configuration | display set" diff --git a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py b/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py index f7343f6..d15bc2b 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py @@ -1,21 +1,114 @@ -"""network_importer driver for Mikrotik Router OS.""" +"""nornir dispatcher for Mikrotik Router OS.""" +# pylint: disable=raise-missing-from + +import os +import ssl +import json try: - from netmiko.ssh_exception import NetmikoAuthenticationException, NetmikoTimeoutException + import routeros_api # pylint: disable=E0401 except ImportError: - from netmiko import NetmikoAuthenticationException, NetmikoTimeoutException + routeros_api = None + +from netutils.config.clean import clean_config, sanitize_config + +from netmiko import NetmikoAuthenticationException, NetmikoTimeoutException from nornir.core.exceptions import NornirSubTaskError from nornir.core.task import Result, Task from nornir_netmiko.tasks import netmiko_send_command from nornir_nautobot.exceptions import NornirNautobotException -from .default import NetmikoNautobotNornirDriver as DefaultNautobotNornirDriver +from nornir_nautobot.plugins.tasks.dispatcher.default import DispatcherMixin, NetmikoDefault +from nornir_nautobot.utils.helpers import make_folder NETMIKO_DEVICE_TYPE = "mikrotik_routeros" -class NautobotNornirDriver(DefaultNautobotNornirDriver): +class ApiMikrotikRouteros(DispatcherMixin): + """Default collection of Nornir Tasks based on Napalm.""" + + tcp_port = 8729 + + config_command = [ + "/system/identity", + "/user", + "/interface", + "/ip/address", + "/system/ntp/client", + "/ip/dns", + "/snmp/community", + "/system/logging/action", + ] + + @classmethod + def get_config( # pylint: disable=R0913,R0914 + cls, task: Task, logger, obj, backup_file: str, remove_lines: list, substitute_lines: list + ) -> Result: + """Get the latest configuration from the device. + + Args: + task (Task): Nornir Task. + logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + obj (Device): A Nautobot Device Django ORM object instance. + backup_file (str): The file location of where the back configuration should be saved. + remove_lines (list): A list of regex lines to remove configurations. + substitute_lines (list): A list of dictionaries with to remove and replace lines. + + Returns: + Result: Nornir Result object with a dict as a result containing the running configuration + { "config: } + """ + logger.log_debug(f"Executing get_config for {task.host.name} on {task.host.platform}") + if not routeros_api: + error_msg = "E1020: The `routeros_api` is not installed in this environment." + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + + sslctx = ssl.create_default_context() + sslctx.set_ciphers("ADH-AES256-GCM-SHA384:ADH-AES256-SHA256:@SECLEVEL=0") + connection = routeros_api.RouterOsApiPool( + task.host.hostname, + username=task.host.username, + password=task.host.password, + use_ssl=True, + ssl_context=sslctx, + plaintext_login=True, + ) + config_data = {} + try: + api = connection.get_api() + except Exception as error: + error_msg = f"E1021: `get_config` method failed with an unexpected issue: `{error}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + for endpoint in cls.config_command: + try: + resource = api.get_resource(endpoint) + config_data[endpoint] = resource.get() + except Exception as error: + error_msg = f"E1022: `get_config` method failed with an unexpected issue: `{error}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + + connection.disconnect() + running_config = json.dumps(config_data, indent=4) + if remove_lines: + logger.log_debug("Removing lines from configuration based on `remove_lines` definition") + running_config = clean_config(running_config, remove_lines) + + if substitute_lines: + logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") + running_config = sanitize_config(running_config, substitute_lines) + + make_folder(os.path.dirname(backup_file)) + + with open(backup_file, "w", encoding="utf8") as filehandler: + filehandler.write(running_config) + return Result(host=task.host, result={"config": running_config}) + + +class NetmikoMikrotekRouteros(NetmikoDefault): """Driver for Mikrotik Router OS.""" config_command = "export terse" @@ -46,21 +139,18 @@ def get_config( # pylint: disable=R0913,R0914 result = task.run(task=netmiko_send_command, command_string=cls.version_command) except NornirSubTaskError as exc: if isinstance(exc.result.exception, NetmikoAuthenticationException): - logger.log_failure(obj, f"Failed with an authentication issue: `{exc.result.exception}`") - raise NornirNautobotException( # pylint: disable=W0707 - f"Failed with an authentication issue: `{exc.result.exception}`" - ) + error_msg = f"E1017: Failed with an authentication issue: `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) if isinstance(exc.result.exception, NetmikoTimeoutException): - logger.log_failure(obj, f"Failed with a timeout issue. `{exc.result.exception}`") - raise NornirNautobotException( # pylint: disable=W0707 - f"Failed with a timeout issue. `{exc.result.exception}`" - ) + error_msg = f"E1018: Failed with a timeout issue. `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) - logger.log_failure(obj, f"Failed with an unknown issue. `{exc.result.exception}`") - raise NornirNautobotException( # pylint: disable=W0707 - f"Failed with an unknown issue. `{exc.result.exception}`" - ) + error_msg = f"E1016: Failed with an unknown issue. `{exc.result.exception}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) if result[0].failed: return result @@ -77,7 +167,7 @@ def get_config( # pylint: disable=R0913,R0914 try: result = task.run(task=netmiko_send_command, command_string=command) except NornirSubTaskError as exc: - logger.log_failure(obj, f"Failed with an unknown issue. `{exc.result.exception}`") + logger.log_error(f"Failed with an unknown issue. `{exc.result.exception}`", extra={"object": obj}) raise NornirNautobotException( # pylint: disable=W0707 f"Failed with an unknown issue. `{exc.result.exception}`" ) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros_api.py b/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros_api.py deleted file mode 100644 index 85caa32..0000000 --- a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros_api.py +++ /dev/null @@ -1,148 +0,0 @@ -"""default network_importer API-based driver for Mikrotik RouterOS.""" - -import os -import ssl -import json -import socket - -try: - import routeros_api # pylint: disable=E0401 -except ImportError: - routeros_api = None - -from netutils.config.clean import clean_config, sanitize_config -from netutils.dns import is_fqdn_resolvable -from netutils.ip import is_ip -from netutils.ping import tcp_ping - -from nornir.core.task import Result, Task - -from nornir_nautobot.exceptions import NornirNautobotException -from nornir_nautobot.utils.helpers import make_folder - -from .default import NautobotNornirDriver as DefaultNautobotNornirDriver - - -class NautobotNornirDriver(DefaultNautobotNornirDriver): - """Default collection of Nornir Tasks based on Napalm.""" - - config_command = [ - "/system/identity", - "/user", - "/interface", - "/ip/address", - "/system/ntp/client", - "/ip/dns", - "/snmp/community", - "/system/logging/action", - ] - - @classmethod - def get_config( # pylint: disable=R0913,R0914 - cls, task: Task, logger, obj, backup_file: str, remove_lines: list, substitute_lines: list - ) -> Result: - """Get the latest configuration from the device. - - Args: - task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. - obj (Device): A Nautobot Device Django ORM object instance. - backup_file (str): The file location of where the back configuration should be saved. - remove_lines (list): A list of regex lines to remove configurations. - substitute_lines (list): A list of dictionaries with to remove and replace lines. - - Returns: - Result: Nornir Result object with a dict as a result containing the running configuration - { "config: } - """ - logger.log_debug(f"Executing get_config for {task.host.name} on {task.host.platform}") - sslctx = ssl.create_default_context() - sslctx.set_ciphers("ADH-AES256-GCM-SHA384:ADH-AES256-SHA256:@SECLEVEL=0") - try: - connection = routeros_api.RouterOsApiPool( - task.host.hostname, - username=task.host.username, - password=task.host.password, - use_ssl=True, - ssl_context=sslctx, - plaintext_login=True, - ) - except AttributeError: - raise NornirNautobotException( # pylint: disable=W0707 - "`routeros_api` module missing, check your environment" - ) - config_data = {} - try: - api = connection.get_api() - except Exception as error: - logger.log_failure(obj, f"`get_config` method failed with an unexpected issue: `{error}`") - raise NornirNautobotException( # pylint: disable=W0707 - f"`get_config` method failed with an unexpected issue: `{error}`" - ) - for endpoint in cls.config_command: - try: - resource = api.get_resource(endpoint) - config_data[endpoint] = resource.get() - except Exception as error: - logger.log_failure(obj, f"`get_config` method failed with an unexpected issue: `{error}`") - raise NornirNautobotException( # pylint: disable=W0707 - f"`get_config` method failed with an unexpected issue: `{error}`" - ) - - connection.disconnect() - running_config = json.dumps(config_data, indent=4) - if remove_lines: - logger.log_debug("Removing lines from configuration based on `remove_lines` definition") - running_config = clean_config(running_config, remove_lines) - - if substitute_lines: - logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") - running_config = sanitize_config(running_config, substitute_lines) - - make_folder(os.path.dirname(backup_file)) - - with open(backup_file, "w", encoding="utf8") as filehandler: - filehandler.write(running_config) - return Result(host=task.host, result={"config": running_config}) - - @staticmethod - def check_connectivity(task: Task, logger, obj) -> Result: - """Check the connectivity to a network device. - - Args: - task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. - obj (Device): A Nautobot Device Django ORM object instance. - - Returns: - Result: Nornir Result object. - """ - if is_ip(task.host.hostname): - ip_addr = task.host.hostname - else: - if not is_fqdn_resolvable(task.host.hostname): - logger.log_failure(obj, "There was not an IP or resolvable, preemptively failed.") - raise NornirNautobotException( - "There was not an IP or resolvable, preemptively failed." - ) # pylint: disable=W0707 - ip_addr = socket.gethostbyname(task.host.hostname) - - # TODO: Allow port to be configurable, allow ssl as well - port = 8729 - if not tcp_ping(ip_addr, port): - logger.log_failure(obj, f"Could not connect to IP: {ip_addr} and port: {port}, preemptively failed.") - raise NornirNautobotException( - f"Could not connect to IP: {ip_addr} and port: {port}, preemptively failed." - ) # pylint: disable=W0707 - if not task.host.username: - logger.log_failure(obj, "There was no username defined, preemptively failed.") - raise NornirNautobotException( - "There was no username defined, preemptively failed." - ) # pylint: disable=W0707 - if not task.host.password: - logger.log_failure(obj, "There was no password defined, preemptively failed.") - raise NornirNautobotException( - "There was no password defined, preemptively failed." - ) # pylint: disable=W0707 - - return Result(host=task.host) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py index a341c41..0e53194 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py @@ -1,9 +1,9 @@ -"""network_importer driver for Ruckus ICX/FastIron Switches.""" +"""nornir dispatcher for Ruckus ICX/FastIron Switches.""" -from .default import NetmikoNautobotNornirDriver as DefaultNautobotNornirDriver +from nornir_nautobot.plugins.tasks.dispatcher.default import NetmikoDefault -class NautobotNornirDriver(DefaultNautobotNornirDriver): +class NetmikoRuckusFastiron(NetmikoDefault): """Driver for Ruckus ICX/FastIron Switches.""" config_command = "show running-config" diff --git a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone_api.py b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py similarity index 66% rename from nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone_api.py rename to nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py index 4908916..e971951 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone_api.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py @@ -2,14 +2,10 @@ import os import json -import socket import asyncio import httpx # pylint: disable=E0401 import requests -from netutils.dns import is_fqdn_resolvable -from netutils.ip import is_ip -from netutils.ping import tcp_ping from netutils.config.clean import clean_config, sanitize_config from nornir.core.task import Result, Task @@ -17,12 +13,12 @@ from nornir_nautobot.exceptions import NornirNautobotException from nornir_nautobot.utils.helpers import make_folder -from .default import NautobotNornirDriver as DefaultNautobotNornirDriver +from nornir_nautobot.plugins.tasks.dispatcher.default import DispatcherMixin AP_PLATFORM_LIST = ["ruckus_access_point", "ruckus-access-point"] -class NautobotNornirDriver(DefaultNautobotNornirDriver): +class ApiRuckusSmartzone(DispatcherMixin): """Default collection of Nornir Tasks tailored for Ruckus Smart Zone Controllers.""" wlc_endpoints = { @@ -47,8 +43,17 @@ class NautobotNornirDriver(DefaultNautobotNornirDriver): ], } - @staticmethod - def _api_auth(session_params: tuple) -> dict: + @classmethod + def _get_hostname(cls, task: Task, obj) -> str: + hostname = ( + obj.get_computed_field("wireless_controller") + if task.host.platform in AP_PLATFORM_LIST + else task.host.hostname + ) + return hostname + + @classmethod + def _api_auth(cls, obj, logger, session_params: tuple) -> dict: controller_ip, username, password = session_params headers = {"Accept": "application/json", "Content-Type": "application/json;charset=UTF-8"} api_version = "v9_1" @@ -58,19 +63,24 @@ def _api_auth(session_params: tuple) -> dict: verify=False, headers=headers, json={"username": username, "password": password}, + timeout=30, ) if response.status_code == 200: data = response.json() service_ticket = data.get("serviceTicket") else: - raise NornirNautobotException( # pylint: disable=W0707 - f"`_api_auth` method failed with an unexpected issue: HTTP Error `{response.status_code}`" + error_msg = ( + f"E1023: `_api_auth` method failed with an unexpected issue: HTTP Error `{response.status_code}`" ) - + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) return service_ticket - @staticmethod - def _build_urls( + @classmethod + def _build_urls( # pylint: disable=too-many-arguments,too-many-locals + cls, + obj, + logger, wlc_ip4: tuple, token: str, endpoints: dict, @@ -94,26 +104,27 @@ def _build_urls( if uri in simple_endpoints: uri_list = [] response = requests.get( - url=f"{base_url}{uri}?serviceTicket={token}", verify=False, headers=headers # nosec + url=f"{base_url}{uri}?serviceTicket={token}", verify=False, headers=headers, timeout=30 # nosec ) if response.status_code == 200: item_list = response.json().get("list") else: - raise NornirNautobotException( # pylint: disable=W0707 - f"`{uri}` endpoint failed with code: HTTP Error `{response.status_code}`" - ) + error_msg = f"E1024: `{uri}` endpoint failed with code: HTTP Error `{response.status_code}`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + uri_list = [f'{uri}/{item["id"]}' for item in item_list] for uri_list_item in uri_list: url_dict[uri_list_item] = f"{base_url}{uri_list_item}?serviceTicket={token}" else: - raise NornirNautobotException( # pylint: disable=W0707 - f"`{uri}` endpoint missing in simple endpoints list, schema invalid`" - ) + error_msg = f"E1025: `{uri}` endpoint missing in simple endpoints list, schema invalid`" + logger.log_error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) return url_dict - @staticmethod - async def _async_get_data(url_dict: dict) -> dict: + @classmethod + async def _async_get_data(cls, url_dict: dict) -> dict: # login use session params headers = {"Accept": "application/json", "Content-Type": "application/json;charset=UTF-8"} @@ -156,8 +167,8 @@ def get_config( # pylint: disable=R0913,R0914 _endpoints = cls.wlc_endpoints _session_params = (_wlc_ip4, task.host.username, task.host.password) - _token = cls._api_auth(_session_params) - url_dict = cls._build_urls(_wlc_ip4, _token, _endpoints, _extras) + _token = cls._api_auth(obj, logger, _session_params) + url_dict = cls._build_urls(obj, logger, _wlc_ip4, _token, _endpoints, _extras) config_data = asyncio.run(cls._async_get_data(url_dict)) running_config = json.dumps(config_data, indent=4) @@ -174,50 +185,3 @@ def get_config( # pylint: disable=R0913,R0914 with open(backup_file, "w", encoding="utf8") as filehandler: filehandler.write(running_config) return Result(host=task.host, result={"config": running_config}) - - @staticmethod - def check_connectivity(task: Task, logger, obj) -> Result: - """Check the connectivity to a network device. - - Args: - task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. - obj (Device): A Nautobot Device Django ORM object instance. - - Returns: - Result: Nornir Result object. - """ - hostname = ( - obj.get_computed_field("wireless_controller") - if task.host.platform in AP_PLATFORM_LIST - else task.host.hostname - ) - if is_ip(hostname): - ip_addr = hostname - else: - if not is_fqdn_resolvable(hostname): - logger.log_failure(obj, "There was not an IP or resolvable, preemptively failed.") - raise NornirNautobotException( - "There was not an IP or resolvable, preemptively failed." - ) # pylint: disable=W0707 - ip_addr = socket.gethostbyname(hostname) - - # TODO: Allow port to be configurable, allow ssl as well - port = 8443 - if not tcp_ping(ip_addr, port): - logger.log_failure(obj, f"Could not connect to IP: {ip_addr} and port: {port}, preemptively failed.") - raise NornirNautobotException( - f"Could not connect to IP: {ip_addr} and port: {port}, preemptively failed." - ) # pylint: disable=W0707 - if not task.host.username: - logger.log_failure(obj, "There was no username defined, preemptively failed.") - raise NornirNautobotException( - "There was no username defined, preemptively failed." - ) # pylint: disable=W0707 - if not task.host.password: - logger.log_failure(obj, "There was no password defined, preemptively failed.") - raise NornirNautobotException( - "There was no password defined, preemptively failed." - ) # pylint: disable=W0707 - - return Result(host=task.host) diff --git a/nornir_nautobot/utils/helpers.py b/nornir_nautobot/utils/helpers.py index 7536ccf..4912758 100644 --- a/nornir_nautobot/utils/helpers.py +++ b/nornir_nautobot/utils/helpers.py @@ -3,6 +3,7 @@ import errno import os import logging +import importlib LOGGER = logging.getLogger(__name__) @@ -16,3 +17,17 @@ def make_folder(folder): except OSError as error: if error.errno != errno.EEXIST: raise + + +def snake_to_title_case(snake_string): + """Convert snake_case into TitleCase.""" + return "".join(word.capitalize() for word in snake_string.lower().split("_")) + + +def import_string(dotted_path): + """Import the python object by dotted_path string .""" + module_name, class_name = dotted_path.rsplit(".", 1) + try: + return getattr(importlib.import_module(module_name), class_name) + except (ModuleNotFoundError, AttributeError): + return None diff --git a/nornir_nautobot/utils/logger.py b/nornir_nautobot/utils/logger.py index a3544dc..83265de 100644 --- a/nornir_nautobot/utils/logger.py +++ b/nornir_nautobot/utils/logger.py @@ -14,39 +14,33 @@ def __init__(self, name: str, nautobot_job=None, debug: bool = False, job_result self.logger = logging.getLogger(name) self.debug = debug self.nautobot_job = nautobot_job or job_result - if job_result is not None: - log_message = ( - "The arg named `job_result` has been renamed to `nautobot_job`; please update to use this name." - ) - self.logger.warning(log_message) - job_result.log_warning(log_message) - - def log_debug(self, message: str): + + def log_debug(self, message: str, extra: Any=None): """Debug, does not take obj, and only logs to jobs result when in global debug mode.""" if self.nautobot_job and self.debug: - self.nautobot_job.log_debug(message) + self.nautobot_job.logging.debug(message, extra=extra) self.logger.debug(message) - def log_info(self, obj: Any, message: str): + def log_info(self, message: str, extra: Any=None): """Log to Python logger and jogs results for info messages.""" if self.nautobot_job: - self.nautobot_job.log_info(obj, message) - self.logger.info("%s | %s", str(obj), message) + self.nautobot_job.logging.info(message, extra=extra) + self.logger.info("%s | %s", str(extra), message) - def log_success(self, obj: Any, message: str): - """Log to Python logger and jogs results for success messages.""" + def log_warning(self, message: str, extra: Any=None): + """Log to Python logger and jogs results for warning messages.""" if self.nautobot_job: - self.nautobot_job.log_success(obj, message) - self.logger.info("%s | %s", str(obj), message) + self.nautobot_job.logging.warning(message, extra=extra) + self.logger.warning("%s | %s", str(extra), message) - def log_warning(self, obj: Any, message: str): - """Log to Python logger and jogs results for warning messages.""" + def log_error(self, message: str, extra: Any=None): + """Log to Python logger and jogs results for error messages.""" if self.nautobot_job: - self.nautobot_job.log_warning(obj, message) - self.logger.warning("%s | %s", str(obj), message) + self.nautobot_job.logging.error(message, extra=extra) + self.logger.error("%s | %s", str(extra), message) - def log_failure(self, obj: Any, message: str): - """Log to Python logger and jogs results for failure messages.""" + def log_critical(self, message: str, extra: Any=None): + """Log to Python logger and jogs results for critical messages.""" if self.nautobot_job: - self.nautobot_job.log_failure(obj, message) - self.logger.error("%s | %s", str(obj), message) + self.nautobot_job.logging.critical(message, extra=extra) + self.logger.critical("%s | %s", str(extra), message) \ No newline at end of file diff --git a/poetry.lock b/poetry.lock index 173262e..518461c 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,44 +1,58 @@ -# This file is automatically @generated by Poetry 1.5.1 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.6.1 and should not be changed by hand. [[package]] name = "anyio" -version = "3.7.0" +version = "4.0.0" description = "High level compatibility layer for multiple asynchronous event loop implementations" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "anyio-3.7.0-py3-none-any.whl", hash = "sha256:eddca883c4175f14df8aedce21054bfca3adb70ffe76a9f607aef9d7fa2ea7f0"}, - {file = "anyio-3.7.0.tar.gz", hash = "sha256:275d9973793619a5374e1c89a4f4ad3f4b0a5510a2b5b939444bee8f4c4d37ce"}, + {file = "anyio-4.0.0-py3-none-any.whl", hash = "sha256:cfdb2b588b9fc25ede96d8db56ed50848b0b649dca3dd1df0b11f683bb9e0b5f"}, + {file = "anyio-4.0.0.tar.gz", hash = "sha256:f7ed51751b2c2add651e5747c891b47e26d2a21be5d32d9311dfe9692f3e5d7a"}, ] [package.dependencies] -exceptiongroup = {version = "*", markers = "python_version < \"3.11\""} +exceptiongroup = {version = ">=1.0.2", markers = "python_version < \"3.11\""} idna = ">=2.8" sniffio = ">=1.1" -typing-extensions = {version = "*", markers = "python_version < \"3.8\""} [package.extras] -doc = ["Sphinx (>=6.1.0)", "packaging", "sphinx-autodoc-typehints (>=1.2.0)", "sphinx-rtd-theme", "sphinxcontrib-jquery"] -test = ["anyio[trio]", "coverage[toml] (>=4.5)", "hypothesis (>=4.0)", "mock (>=4)", "psutil (>=5.9)", "pytest (>=7.0)", "pytest-mock (>=3.6.1)", "trustme", "uvloop (>=0.17)"] -trio = ["trio (<0.22)"] +doc = ["Sphinx (>=7)", "packaging", "sphinx-autodoc-typehints (>=1.2.0)"] +test = ["anyio[trio]", "coverage[toml] (>=7)", "hypothesis (>=4.0)", "psutil (>=5.9)", "pytest (>=7.0)", "pytest-mock (>=3.6.1)", "trustme", "uvloop (>=0.17)"] +trio = ["trio (>=0.22)"] [[package]] name = "astroid" -version = "2.11.7" +version = "2.15.6" description = "An abstract syntax tree for Python with inference support." optional = false -python-versions = ">=3.6.2" +python-versions = ">=3.7.2" files = [ - {file = "astroid-2.11.7-py3-none-any.whl", hash = "sha256:86b0a340a512c65abf4368b80252754cda17c02cdbbd3f587dddf98112233e7b"}, - {file = "astroid-2.11.7.tar.gz", hash = "sha256:bb24615c77f4837c707669d16907331374ae8a964650a66999da3f5ca68dc946"}, + {file = "astroid-2.15.6-py3-none-any.whl", hash = "sha256:389656ca57b6108f939cf5d2f9a2a825a3be50ba9d589670f393236e0a03b91c"}, + {file = "astroid-2.15.6.tar.gz", hash = "sha256:903f024859b7c7687d7a7f3a3f73b17301f8e42dfd9cc9df9d4418172d3e2dbd"}, ] [package.dependencies] lazy-object-proxy = ">=1.4.0" -setuptools = ">=20.0" -typed-ast = {version = ">=1.4.0,<2.0", markers = "implementation_name == \"cpython\" and python_version < \"3.8\""} -typing-extensions = {version = ">=3.10", markers = "python_version < \"3.10\""} -wrapt = ">=1.11,<2" +typing-extensions = {version = ">=4.0.0", markers = "python_version < \"3.11\""} +wrapt = [ + {version = ">=1.11,<2", markers = "python_version < \"3.11\""}, + {version = ">=1.14,<2", markers = "python_version >= \"3.11\""}, +] + +[[package]] +name = "babel" +version = "2.12.1" +description = "Internationalization utilities" +optional = false +python-versions = ">=3.7" +files = [ + {file = "Babel-2.12.1-py3-none-any.whl", hash = "sha256:b4246fb7677d3b98f501a39d43396d3cafdc8eadb045f4a31be01863f655c610"}, + {file = "Babel-2.12.1.tar.gz", hash = "sha256:cc2d99999cd01d44420ae725a21c9e3711b3aadc7976d6147f622d8581963455"}, +] + +[package.dependencies] +pytz = {version = ">=2015.7", markers = "python_version < \"3.9\""} [[package]] name = "bandit" @@ -97,38 +111,53 @@ files = [ tests = ["pytest (>=3.2.1,!=3.3.0)"] typecheck = ["mypy"] +[[package]] +name = "beautifulsoup4" +version = "4.12.2" +description = "Screen-scraping library" +optional = false +python-versions = ">=3.6.0" +files = [ + {file = "beautifulsoup4-4.12.2-py3-none-any.whl", hash = "sha256:bd2520ca0d9d7d12694a53d44ac482d181b4ec1888909b035a3dbf40d0f57d4a"}, + {file = "beautifulsoup4-4.12.2.tar.gz", hash = "sha256:492bbc69dca35d12daac71c4db1bfff0c876c00ef4a2ffacce226d4638eb72da"}, +] + +[package.dependencies] +soupsieve = ">1.2" + +[package.extras] +html5lib = ["html5lib"] +lxml = ["lxml"] + [[package]] name = "black" -version = "23.3.0" +version = "23.7.0" description = "The uncompromising code formatter." optional = false -python-versions = ">=3.7" -files = [ - {file = "black-23.3.0-cp310-cp310-macosx_10_16_arm64.whl", hash = "sha256:0945e13506be58bf7db93ee5853243eb368ace1c08a24c65ce108986eac65915"}, - {file = "black-23.3.0-cp310-cp310-macosx_10_16_universal2.whl", hash = "sha256:67de8d0c209eb5b330cce2469503de11bca4085880d62f1628bd9972cc3366b9"}, - {file = "black-23.3.0-cp310-cp310-macosx_10_16_x86_64.whl", hash = "sha256:7c3eb7cea23904399866c55826b31c1f55bbcd3890ce22ff70466b907b6775c2"}, - {file = "black-23.3.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:32daa9783106c28815d05b724238e30718f34155653d4d6e125dc7daec8e260c"}, - {file = "black-23.3.0-cp310-cp310-win_amd64.whl", hash = "sha256:35d1381d7a22cc5b2be2f72c7dfdae4072a3336060635718cc7e1ede24221d6c"}, - {file = "black-23.3.0-cp311-cp311-macosx_10_16_arm64.whl", hash = "sha256:a8a968125d0a6a404842fa1bf0b349a568634f856aa08ffaff40ae0dfa52e7c6"}, - {file = "black-23.3.0-cp311-cp311-macosx_10_16_universal2.whl", hash = "sha256:c7ab5790333c448903c4b721b59c0d80b11fe5e9803d8703e84dcb8da56fec1b"}, - {file = "black-23.3.0-cp311-cp311-macosx_10_16_x86_64.whl", hash = "sha256:a6f6886c9869d4daae2d1715ce34a19bbc4b95006d20ed785ca00fa03cba312d"}, - {file = "black-23.3.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6f3c333ea1dd6771b2d3777482429864f8e258899f6ff05826c3a4fcc5ce3f70"}, - {file = "black-23.3.0-cp311-cp311-win_amd64.whl", hash = "sha256:11c410f71b876f961d1de77b9699ad19f939094c3a677323f43d7a29855fe326"}, - {file = "black-23.3.0-cp37-cp37m-macosx_10_16_x86_64.whl", hash = "sha256:1d06691f1eb8de91cd1b322f21e3bfc9efe0c7ca1f0e1eb1db44ea367dff656b"}, - {file = "black-23.3.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:50cb33cac881766a5cd9913e10ff75b1e8eb71babf4c7104f2e9c52da1fb7de2"}, - {file = "black-23.3.0-cp37-cp37m-win_amd64.whl", hash = "sha256:e114420bf26b90d4b9daa597351337762b63039752bdf72bf361364c1aa05925"}, - {file = "black-23.3.0-cp38-cp38-macosx_10_16_arm64.whl", hash = "sha256:48f9d345675bb7fbc3dd85821b12487e1b9a75242028adad0333ce36ed2a6d27"}, - {file = "black-23.3.0-cp38-cp38-macosx_10_16_universal2.whl", hash = "sha256:714290490c18fb0126baa0fca0a54ee795f7502b44177e1ce7624ba1c00f2331"}, - {file = "black-23.3.0-cp38-cp38-macosx_10_16_x86_64.whl", hash = "sha256:064101748afa12ad2291c2b91c960be28b817c0c7eaa35bec09cc63aa56493c5"}, - {file = "black-23.3.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:562bd3a70495facf56814293149e51aa1be9931567474993c7942ff7d3533961"}, - {file = "black-23.3.0-cp38-cp38-win_amd64.whl", hash = "sha256:e198cf27888ad6f4ff331ca1c48ffc038848ea9f031a3b40ba36aced7e22f2c8"}, - {file = "black-23.3.0-cp39-cp39-macosx_10_16_arm64.whl", hash = "sha256:3238f2aacf827d18d26db07524e44741233ae09a584273aa059066d644ca7b30"}, - {file = "black-23.3.0-cp39-cp39-macosx_10_16_universal2.whl", hash = "sha256:f0bd2f4a58d6666500542b26354978218a9babcdc972722f4bf90779524515f3"}, - {file = "black-23.3.0-cp39-cp39-macosx_10_16_x86_64.whl", hash = "sha256:92c543f6854c28a3c7f39f4d9b7694f9a6eb9d3c5e2ece488c327b6e7ea9b266"}, - {file = "black-23.3.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3a150542a204124ed00683f0db1f5cf1c2aaaa9cc3495b7a3b5976fb136090ab"}, - {file = "black-23.3.0-cp39-cp39-win_amd64.whl", hash = "sha256:6b39abdfb402002b8a7d030ccc85cf5afff64ee90fa4c5aebc531e3ad0175ddb"}, - {file = "black-23.3.0-py3-none-any.whl", hash = "sha256:ec751418022185b0c1bb7d7736e6933d40bbb14c14a0abcf9123d1b159f98dd4"}, - {file = "black-23.3.0.tar.gz", hash = "sha256:1c7b8d606e728a41ea1ccbd7264677e494e87cf630e399262ced92d4a8dac940"}, +python-versions = ">=3.8" +files = [ + {file = "black-23.7.0-cp310-cp310-macosx_10_16_arm64.whl", hash = "sha256:5c4bc552ab52f6c1c506ccae05681fab58c3f72d59ae6e6639e8885e94fe2587"}, + {file = "black-23.7.0-cp310-cp310-macosx_10_16_universal2.whl", hash = "sha256:552513d5cd5694590d7ef6f46e1767a4df9af168d449ff767b13b084c020e63f"}, + {file = "black-23.7.0-cp310-cp310-macosx_10_16_x86_64.whl", hash = "sha256:86cee259349b4448adb4ef9b204bb4467aae74a386bce85d56ba4f5dc0da27be"}, + {file = "black-23.7.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:501387a9edcb75d7ae8a4412bb8749900386eaef258f1aefab18adddea1936bc"}, + {file = "black-23.7.0-cp310-cp310-win_amd64.whl", hash = "sha256:fb074d8b213749fa1d077d630db0d5f8cc3b2ae63587ad4116e8a436e9bbe995"}, + {file = "black-23.7.0-cp311-cp311-macosx_10_16_arm64.whl", hash = "sha256:b5b0ee6d96b345a8b420100b7d71ebfdd19fab5e8301aff48ec270042cd40ac2"}, + {file = "black-23.7.0-cp311-cp311-macosx_10_16_universal2.whl", hash = "sha256:893695a76b140881531062d48476ebe4a48f5d1e9388177e175d76234ca247cd"}, + {file = "black-23.7.0-cp311-cp311-macosx_10_16_x86_64.whl", hash = "sha256:c333286dc3ddca6fdff74670b911cccedacb4ef0a60b34e491b8a67c833b343a"}, + {file = "black-23.7.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:831d8f54c3a8c8cf55f64d0422ee875eecac26f5f649fb6c1df65316b67c8926"}, + {file = "black-23.7.0-cp311-cp311-win_amd64.whl", hash = "sha256:7f3bf2dec7d541b4619b8ce526bda74a6b0bffc480a163fed32eb8b3c9aed8ad"}, + {file = "black-23.7.0-cp38-cp38-macosx_10_16_arm64.whl", hash = "sha256:f9062af71c59c004cd519e2fb8f5d25d39e46d3af011b41ab43b9c74e27e236f"}, + {file = "black-23.7.0-cp38-cp38-macosx_10_16_universal2.whl", hash = "sha256:01ede61aac8c154b55f35301fac3e730baf0c9cf8120f65a9cd61a81cfb4a0c3"}, + {file = "black-23.7.0-cp38-cp38-macosx_10_16_x86_64.whl", hash = "sha256:327a8c2550ddc573b51e2c352adb88143464bb9d92c10416feb86b0f5aee5ff6"}, + {file = "black-23.7.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6d1c6022b86f83b632d06f2b02774134def5d4d4f1dac8bef16d90cda18ba28a"}, + {file = "black-23.7.0-cp38-cp38-win_amd64.whl", hash = "sha256:27eb7a0c71604d5de083757fbdb245b1a4fae60e9596514c6ec497eb63f95320"}, + {file = "black-23.7.0-cp39-cp39-macosx_10_16_arm64.whl", hash = "sha256:8417dbd2f57b5701492cd46edcecc4f9208dc75529bcf76c514864e48da867d9"}, + {file = "black-23.7.0-cp39-cp39-macosx_10_16_universal2.whl", hash = "sha256:47e56d83aad53ca140da0af87678fb38e44fd6bc0af71eebab2d1f59b1acf1d3"}, + {file = "black-23.7.0-cp39-cp39-macosx_10_16_x86_64.whl", hash = "sha256:25cc308838fe71f7065df53aedd20327969d05671bac95b38fdf37ebe70ac087"}, + {file = "black-23.7.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:642496b675095d423f9b8448243336f8ec71c9d4d57ec17bf795b67f08132a91"}, + {file = "black-23.7.0-cp39-cp39-win_amd64.whl", hash = "sha256:ad0014efc7acf0bd745792bd0d8857413652979200ab924fbf239062adc12491"}, + {file = "black-23.7.0-py3-none-any.whl", hash = "sha256:9fd59d418c60c0348505f2ddf9609c1e1de8e7493eab96198fc89d9f865e7a96"}, + {file = "black-23.7.0.tar.gz", hash = "sha256:022a582720b0d9480ed82576c920a8c1dde97cc38ff11d8d8859b3bd6ca9eedb"}, ] [package.dependencies] @@ -138,7 +167,6 @@ packaging = ">=22.0" pathspec = ">=0.9.0" platformdirs = ">=2" tomli = {version = ">=1.1.0", markers = "python_version < \"3.11\""} -typed-ast = {version = ">=1.4.2", markers = "python_version < \"3.8\" and implementation_name == \"cpython\""} typing-extensions = {version = ">=3.10.0.0", markers = "python_version < \"3.10\""} [package.extras] @@ -147,26 +175,15 @@ d = ["aiohttp (>=3.7.4)"] jupyter = ["ipython (>=7.8.0)", "tokenize-rt (>=3.2.0)"] uvloop = ["uvloop (>=0.15.2)"] -[[package]] -name = "cached-property" -version = "1.5.2" -description = "A decorator for caching properties in classes." -optional = false -python-versions = "*" -files = [ - {file = "cached-property-1.5.2.tar.gz", hash = "sha256:9fa5755838eecbb2d234c3aa390bd80fbd3ac6b6869109bfc1b499f7bd89a130"}, - {file = "cached_property-1.5.2-py2.py3-none-any.whl", hash = "sha256:df4f613cf7ad9a588cc381aaf4a512d26265ecebd5eb9e1ba12f1319eb85a6a0"}, -] - [[package]] name = "certifi" -version = "2023.5.7" +version = "2023.7.22" description = "Python package for providing Mozilla's CA Bundle." optional = false python-versions = ">=3.6" files = [ - {file = "certifi-2023.5.7-py3-none-any.whl", hash = "sha256:c6c2e98f5c7869efca1f8916fed228dd91539f9f1b444c314c06eef02980c716"}, - {file = "certifi-2023.5.7.tar.gz", hash = "sha256:0f0d56dc5a6ad56fd4ba36484d6cc34451e1c6548c61daad8c320169f91eddc7"}, + {file = "certifi-2023.7.22-py3-none-any.whl", hash = "sha256:92d6037539857d8206b8f6ae472e8b77db8058fec5937a1ef3f54304089edbb9"}, + {file = "certifi-2023.7.22.tar.gz", hash = "sha256:539cc1d13202e33ca466e88b2807e29f4c13049d6d87031a3c110744495cb082"}, ] [[package]] @@ -247,102 +264,101 @@ pycparser = "*" [[package]] name = "charset-normalizer" -version = "3.1.0" +version = "3.2.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.1.0.tar.gz", hash = "sha256:34e0a2f9c370eb95597aae63bf85eb5e96826d81e3dcf88b8886012906f509b5"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:e0ac8959c929593fee38da1c2b64ee9778733cdf03c482c9ff1d508b6b593b2b"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d7fc3fca01da18fbabe4625d64bb612b533533ed10045a2ac3dd194bfa656b60"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:04eefcee095f58eaabe6dc3cc2262f3bcd776d2c67005880894f447b3f2cb9c1"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:20064ead0717cf9a73a6d1e779b23d149b53daf971169289ed2ed43a71e8d3b0"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1435ae15108b1cb6fffbcea2af3d468683b7afed0169ad718451f8db5d1aff6f"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c84132a54c750fda57729d1e2599bb598f5fa0344085dbde5003ba429a4798c0"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:75f2568b4189dda1c567339b48cba4ac7384accb9c2a7ed655cd86b04055c795"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:11d3bcb7be35e7b1bba2c23beedac81ee893ac9871d0ba79effc7fc01167db6c"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:891cf9b48776b5c61c700b55a598621fdb7b1e301a550365571e9624f270c203"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:5f008525e02908b20e04707a4f704cd286d94718f48bb33edddc7d7b584dddc1"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:b06f0d3bf045158d2fb8837c5785fe9ff9b8c93358be64461a1089f5da983137"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:49919f8400b5e49e961f320c735388ee686a62327e773fa5b3ce6721f7e785ce"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:22908891a380d50738e1f978667536f6c6b526a2064156203d418f4856d6e86a"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-win32.whl", hash = "sha256:12d1a39aa6b8c6f6248bb54550efcc1c38ce0d8096a146638fd4738e42284448"}, - {file = "charset_normalizer-3.1.0-cp310-cp310-win_amd64.whl", hash = "sha256:65ed923f84a6844de5fd29726b888e58c62820e0769b76565480e1fdc3d062f8"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:9a3267620866c9d17b959a84dd0bd2d45719b817245e49371ead79ed4f710d19"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6734e606355834f13445b6adc38b53c0fd45f1a56a9ba06c2058f86893ae8017"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:f8303414c7b03f794347ad062c0516cee0e15f7a612abd0ce1e25caf6ceb47df"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:aaf53a6cebad0eae578f062c7d462155eada9c172bd8c4d250b8c1d8eb7f916a"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3dc5b6a8ecfdc5748a7e429782598e4f17ef378e3e272eeb1340ea57c9109f41"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e1b25e3ad6c909f398df8921780d6a3d120d8c09466720226fc621605b6f92b1"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0ca564606d2caafb0abe6d1b5311c2649e8071eb241b2d64e75a0d0065107e62"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b82fab78e0b1329e183a65260581de4375f619167478dddab510c6c6fb04d9b6"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:bd7163182133c0c7701b25e604cf1611c0d87712e56e88e7ee5d72deab3e76b5"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:11d117e6c63e8f495412d37e7dc2e2fff09c34b2d09dbe2bee3c6229577818be"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:cf6511efa4801b9b38dc5546d7547d5b5c6ef4b081c60b23e4d941d0eba9cbeb"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:abc1185d79f47c0a7aaf7e2412a0eb2c03b724581139193d2d82b3ad8cbb00ac"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:cb7b2ab0188829593b9de646545175547a70d9a6e2b63bf2cd87a0a391599324"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-win32.whl", hash = "sha256:c36bcbc0d5174a80d6cccf43a0ecaca44e81d25be4b7f90f0ed7bcfbb5a00909"}, - {file = "charset_normalizer-3.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:cca4def576f47a09a943666b8f829606bcb17e2bc2d5911a46c8f8da45f56755"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:0c95f12b74681e9ae127728f7e5409cbbef9cd914d5896ef238cc779b8152373"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fca62a8301b605b954ad2e9c3666f9d97f63872aa4efcae5492baca2056b74ab"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ac0aa6cd53ab9a31d397f8303f92c42f534693528fafbdb997c82bae6e477ad9"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c3af8e0f07399d3176b179f2e2634c3ce9c1301379a6b8c9c9aeecd481da494f"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3a5fc78f9e3f501a1614a98f7c54d3969f3ad9bba8ba3d9b438c3bc5d047dd28"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:628c985afb2c7d27a4800bfb609e03985aaecb42f955049957814e0491d4006d"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:74db0052d985cf37fa111828d0dd230776ac99c740e1a758ad99094be4f1803d"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:1e8fcdd8f672a1c4fc8d0bd3a2b576b152d2a349782d1eb0f6b8e52e9954731d"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:04afa6387e2b282cf78ff3dbce20f0cc071c12dc8f685bd40960cc68644cfea6"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:dd5653e67b149503c68c4018bf07e42eeed6b4e956b24c00ccdf93ac79cdff84"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:d2686f91611f9e17f4548dbf050e75b079bbc2a82be565832bc8ea9047b61c8c"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-win32.whl", hash = "sha256:4155b51ae05ed47199dc5b2a4e62abccb274cee6b01da5b895099b61b1982974"}, - {file = "charset_normalizer-3.1.0-cp37-cp37m-win_amd64.whl", hash = "sha256:322102cdf1ab682ecc7d9b1c5eed4ec59657a65e1c146a0da342b78f4112db23"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:e633940f28c1e913615fd624fcdd72fdba807bf53ea6925d6a588e84e1151531"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:3a06f32c9634a8705f4ca9946d667609f52cf130d5548881401f1eb2c39b1e2c"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:7381c66e0561c5757ffe616af869b916c8b4e42b367ab29fedc98481d1e74e14"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3573d376454d956553c356df45bb824262c397c6e26ce43e8203c4c540ee0acb"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e89df2958e5159b811af9ff0f92614dabf4ff617c03a4c1c6ff53bf1c399e0e1"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:78cacd03e79d009d95635e7d6ff12c21eb89b894c354bd2b2ed0b4763373693b"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:de5695a6f1d8340b12a5d6d4484290ee74d61e467c39ff03b39e30df62cf83a0"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1c60b9c202d00052183c9be85e5eaf18a4ada0a47d188a83c8f5c5b23252f649"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:f645caaf0008bacf349875a974220f1f1da349c5dbe7c4ec93048cdc785a3326"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:ea9f9c6034ea2d93d9147818f17c2a0860d41b71c38b9ce4d55f21b6f9165a11"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:80d1543d58bd3d6c271b66abf454d437a438dff01c3e62fdbcd68f2a11310d4b"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:73dc03a6a7e30b7edc5b01b601e53e7fc924b04e1835e8e407c12c037e81adbd"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:6f5c2e7bc8a4bf7c426599765b1bd33217ec84023033672c1e9a8b35eaeaaaf8"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-win32.whl", hash = "sha256:12a2b561af122e3d94cdb97fe6fb2bb2b82cef0cdca131646fdb940a1eda04f0"}, - {file = "charset_normalizer-3.1.0-cp38-cp38-win_amd64.whl", hash = "sha256:3160a0fd9754aab7d47f95a6b63ab355388d890163eb03b2d2b87ab0a30cfa59"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:38e812a197bf8e71a59fe55b757a84c1f946d0ac114acafaafaf21667a7e169e"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6baf0baf0d5d265fa7944feb9f7451cc316bfe30e8df1a61b1bb08577c554f31"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:8f25e17ab3039b05f762b0a55ae0b3632b2e073d9c8fc88e89aca31a6198e88f"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3747443b6a904001473370d7810aa19c3a180ccd52a7157aacc264a5ac79265e"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b116502087ce8a6b7a5f1814568ccbd0e9f6cfd99948aa59b0e241dc57cf739f"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d16fd5252f883eb074ca55cb622bc0bee49b979ae4e8639fff6ca3ff44f9f854"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:21fa558996782fc226b529fdd2ed7866c2c6ec91cee82735c98a197fae39f706"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6f6c7a8a57e9405cad7485f4c9d3172ae486cfef1344b5ddd8e5239582d7355e"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:ac3775e3311661d4adace3697a52ac0bab17edd166087d493b52d4f4f553f9f0"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:10c93628d7497c81686e8e5e557aafa78f230cd9e77dd0c40032ef90c18f2230"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:6f4f4668e1831850ebcc2fd0b1cd11721947b6dc7c00bf1c6bd3c929ae14f2c7"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:0be65ccf618c1e7ac9b849c315cc2e8a8751d9cfdaa43027d4f6624bd587ab7e"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:53d0a3fa5f8af98a1e261de6a3943ca631c526635eb5817a87a59d9a57ebf48f"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-win32.whl", hash = "sha256:a04f86f41a8916fe45ac5024ec477f41f886b3c435da2d4e3d2709b22ab02af1"}, - {file = "charset_normalizer-3.1.0-cp39-cp39-win_amd64.whl", hash = "sha256:830d2948a5ec37c386d3170c483063798d7879037492540f10a475e3fd6f244b"}, - {file = "charset_normalizer-3.1.0-py3-none-any.whl", hash = "sha256:3d9098b479e78c85080c98e1e35ff40b4a31d8953102bb0fd7d1b6f8a2111a3d"}, + {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"}, ] [[package]] name = "click" -version = "8.1.3" +version = "8.1.7" description = "Composable command line interface toolkit" optional = false python-versions = ">=3.7" files = [ - {file = "click-8.1.3-py3-none-any.whl", hash = "sha256:bb4d8133cb15a609f44e8213d9b391b0809795062913b383c62be0ee95b1db48"}, - {file = "click-8.1.3.tar.gz", hash = "sha256:7682dc8afb30297001674575ea00d1814d808d6a36af415a82bd481d37ba7b8e"}, + {file = "click-8.1.7-py3-none-any.whl", hash = "sha256:ae74fb96c20a0277a1d615f1e4d73c8414f5a98db8b799a7931d1582f3390c28"}, + {file = "click-8.1.7.tar.gz", hash = "sha256:ca9853ad459e787e2192211578cc907e7594e294c7ccc834310722b41b9ca6de"}, ] [package.dependencies] colorama = {version = "*", markers = "platform_system == \"Windows\""} -importlib-metadata = {version = "*", markers = "python_version < \"3.8\""} [[package]] name = "colorama" @@ -357,30 +373,34 @@ files = [ [[package]] name = "cryptography" -version = "40.0.2" +version = "41.0.3" description = "cryptography is a package which provides cryptographic recipes and primitives to Python developers." optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" files = [ - {file = "cryptography-40.0.2-cp36-abi3-macosx_10_12_universal2.whl", hash = "sha256:8f79b5ff5ad9d3218afb1e7e20ea74da5f76943ee5edb7f76e56ec5161ec782b"}, - {file = "cryptography-40.0.2-cp36-abi3-macosx_10_12_x86_64.whl", hash = "sha256:05dc219433b14046c476f6f09d7636b92a1c3e5808b9a6536adf4932b3b2c440"}, - {file = "cryptography-40.0.2-cp36-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4df2af28d7bedc84fe45bd49bc35d710aede676e2a4cb7fc6d103a2adc8afe4d"}, - {file = "cryptography-40.0.2-cp36-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0dcca15d3a19a66e63662dc8d30f8036b07be851a8680eda92d079868f106288"}, - {file = "cryptography-40.0.2-cp36-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:a04386fb7bc85fab9cd51b6308633a3c271e3d0d3eae917eebab2fac6219b6d2"}, - {file = "cryptography-40.0.2-cp36-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:adc0d980fd2760c9e5de537c28935cc32b9353baaf28e0814df417619c6c8c3b"}, - {file = "cryptography-40.0.2-cp36-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:d5a1bd0e9e2031465761dfa920c16b0065ad77321d8a8c1f5ee331021fda65e9"}, - {file = "cryptography-40.0.2-cp36-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:a95f4802d49faa6a674242e25bfeea6fc2acd915b5e5e29ac90a32b1139cae1c"}, - {file = "cryptography-40.0.2-cp36-abi3-win32.whl", hash = "sha256:aecbb1592b0188e030cb01f82d12556cf72e218280f621deed7d806afd2113f9"}, - {file = "cryptography-40.0.2-cp36-abi3-win_amd64.whl", hash = "sha256:b12794f01d4cacfbd3177b9042198f3af1c856eedd0a98f10f141385c809a14b"}, - {file = "cryptography-40.0.2-pp38-pypy38_pp73-macosx_10_12_x86_64.whl", hash = "sha256:142bae539ef28a1c76794cca7f49729e7c54423f615cfd9b0b1fa90ebe53244b"}, - {file = "cryptography-40.0.2-pp38-pypy38_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:956ba8701b4ffe91ba59665ed170a2ebbdc6fc0e40de5f6059195d9f2b33ca0e"}, - {file = "cryptography-40.0.2-pp38-pypy38_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:4f01c9863da784558165f5d4d916093737a75203a5c5286fde60e503e4276c7a"}, - {file = "cryptography-40.0.2-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:3daf9b114213f8ba460b829a02896789751626a2a4e7a43a28ee77c04b5e4958"}, - {file = "cryptography-40.0.2-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:48f388d0d153350f378c7f7b41497a54ff1513c816bcbbcafe5b829e59b9ce5b"}, - {file = "cryptography-40.0.2-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:c0764e72b36a3dc065c155e5b22f93df465da9c39af65516fe04ed3c68c92636"}, - {file = "cryptography-40.0.2-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:cbaba590180cba88cb99a5f76f90808a624f18b169b90a4abb40c1fd8c19420e"}, - {file = "cryptography-40.0.2-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:7a38250f433cd41df7fcb763caa3ee9362777fdb4dc642b9a349721d2bf47404"}, - {file = "cryptography-40.0.2.tar.gz", hash = "sha256:c33c0d32b8594fa647d2e01dbccc303478e16fdd7cf98652d5b3ed11aa5e5c99"}, + {file = "cryptography-41.0.3-cp37-abi3-macosx_10_12_universal2.whl", hash = "sha256:652627a055cb52a84f8c448185922241dd5217443ca194d5739b44612c5e6507"}, + {file = "cryptography-41.0.3-cp37-abi3-macosx_10_12_x86_64.whl", hash = "sha256:8f09daa483aedea50d249ef98ed500569841d6498aa9c9f4b0531b9964658922"}, + {file = "cryptography-41.0.3-cp37-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4fd871184321100fb400d759ad0cddddf284c4b696568204d281c902fc7b0d81"}, + {file = "cryptography-41.0.3-cp37-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:84537453d57f55a50a5b6835622ee405816999a7113267739a1b4581f83535bd"}, + {file = "cryptography-41.0.3-cp37-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:3fb248989b6363906827284cd20cca63bb1a757e0a2864d4c1682a985e3dca47"}, + {file = "cryptography-41.0.3-cp37-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:42cb413e01a5d36da9929baa9d70ca90d90b969269e5a12d39c1e0d475010116"}, + {file = "cryptography-41.0.3-cp37-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:aeb57c421b34af8f9fe830e1955bf493a86a7996cc1338fe41b30047d16e962c"}, + {file = "cryptography-41.0.3-cp37-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:6af1c6387c531cd364b72c28daa29232162010d952ceb7e5ca8e2827526aceae"}, + {file = "cryptography-41.0.3-cp37-abi3-win32.whl", hash = "sha256:0d09fb5356f975974dbcb595ad2d178305e5050656affb7890a1583f5e02a306"}, + {file = "cryptography-41.0.3-cp37-abi3-win_amd64.whl", hash = "sha256:a983e441a00a9d57a4d7c91b3116a37ae602907a7618b882c8013b5762e80574"}, + {file = "cryptography-41.0.3-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:5259cb659aa43005eb55a0e4ff2c825ca111a0da1814202c64d28a985d33b087"}, + {file = "cryptography-41.0.3-pp310-pypy310_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:67e120e9a577c64fe1f611e53b30b3e69744e5910ff3b6e97e935aeb96005858"}, + {file = "cryptography-41.0.3-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:7efe8041897fe7a50863e51b77789b657a133c75c3b094e51b5e4b5cec7bf906"}, + {file = "cryptography-41.0.3-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:ce785cf81a7bdade534297ef9e490ddff800d956625020ab2ec2780a556c313e"}, + {file = "cryptography-41.0.3-pp38-pypy38_pp73-macosx_10_12_x86_64.whl", hash = "sha256:57a51b89f954f216a81c9d057bf1a24e2f36e764a1ca9a501a6964eb4a6800dd"}, + {file = "cryptography-41.0.3-pp38-pypy38_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:4c2f0d35703d61002a2bbdcf15548ebb701cfdd83cdc12471d2bae80878a4207"}, + {file = "cryptography-41.0.3-pp38-pypy38_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:23c2d778cf829f7d0ae180600b17e9fceea3c2ef8b31a99e3c694cbbf3a24b84"}, + {file = "cryptography-41.0.3-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:95dd7f261bb76948b52a5330ba5202b91a26fbac13ad0e9fc8a3ac04752058c7"}, + {file = "cryptography-41.0.3-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:41d7aa7cdfded09b3d73a47f429c298e80796c8e825ddfadc84c8a7f12df212d"}, + {file = "cryptography-41.0.3-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:d0d651aa754ef58d75cec6edfbd21259d93810b73f6ec246436a21b7841908de"}, + {file = "cryptography-41.0.3-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:ab8de0d091acbf778f74286f4989cf3d1528336af1b59f3e5d2ebca8b5fe49e1"}, + {file = "cryptography-41.0.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:a74fbcdb2a0d46fe00504f571a2a540532f4c188e6ccf26f1f178480117b33c4"}, + {file = "cryptography-41.0.3.tar.gz", hash = "sha256:6d192741113ef5e30d89dcb5b956ef4e1578f304708701b8b73d38e3e1461f34"}, ] [package.dependencies] @@ -389,22 +409,33 @@ cffi = ">=1.12" [package.extras] docs = ["sphinx (>=5.3.0)", "sphinx-rtd-theme (>=1.1.1)"] docstest = ["pyenchant (>=1.6.11)", "sphinxcontrib-spelling (>=4.0.1)", "twine (>=1.12.0)"] -pep8test = ["black", "check-manifest", "mypy", "ruff"] -sdist = ["setuptools-rust (>=0.11.4)"] +nox = ["nox"] +pep8test = ["black", "check-sdist", "mypy", "ruff"] +sdist = ["build"] ssh = ["bcrypt (>=3.1.5)"] -test = ["iso8601", "pretend", "pytest (>=6.2.0)", "pytest-benchmark", "pytest-cov", "pytest-shard (>=0.1.2)", "pytest-subtests", "pytest-xdist"] +test = ["pretend", "pytest (>=6.2.0)", "pytest-benchmark", "pytest-cov", "pytest-xdist"] test-randomorder = ["pytest-randomly"] -tox = ["tox"] + +[[package]] +name = "cssselect" +version = "1.2.0" +description = "cssselect parses CSS3 Selectors and translates them to XPath 1.0" +optional = false +python-versions = ">=3.7" +files = [ + {file = "cssselect-1.2.0-py2.py3-none-any.whl", hash = "sha256:da1885f0c10b60c03ed5eccbb6b68d6eff248d91976fcde348f395d54c9fd35e"}, + {file = "cssselect-1.2.0.tar.gz", hash = "sha256:666b19839cfaddb9ce9d36bfe4c969132c647b92fc9088c4e23f786b30f1b3dc"}, +] [[package]] name = "dill" -version = "0.3.6" -description = "serialize all of python" +version = "0.3.7" +description = "serialize all of Python" optional = false python-versions = ">=3.7" files = [ - {file = "dill-0.3.6-py3-none-any.whl", hash = "sha256:a07ffd2351b8c678dfc4a856a3005f8067aea51d6ba6c700796a4d9e280f39f0"}, - {file = "dill-0.3.6.tar.gz", hash = "sha256:e5db55f3687856d8fbdab002ed78544e1c4559a130302693d839dfe8f93f2373"}, + {file = "dill-0.3.7-py3-none-any.whl", hash = "sha256:76b122c08ef4ce2eedcd4d1abd8e641114bfc6c2867f49f3c41facf65bf19f5e"}, + {file = "dill-0.3.7.tar.gz", hash = "sha256:cc1c8b182eb3013e24bd475ff2e9295af86c1a38eb1aff128dac8962a9ce3c03"}, ] [package.extras] @@ -412,13 +443,13 @@ graph = ["objgraph (>=1.7.2)"] [[package]] name = "exceptiongroup" -version = "1.1.1" +version = "1.1.3" description = "Backport of PEP 654 (exception groups)" optional = false python-versions = ">=3.7" files = [ - {file = "exceptiongroup-1.1.1-py3-none-any.whl", hash = "sha256:232c37c63e4f682982c8b6459f33a8981039e5fb8756b2074364e5055c498c9e"}, - {file = "exceptiongroup-1.1.1.tar.gz", hash = "sha256:d484c3090ba2889ae2928419117447a14daf3c1231d5e30d0aae34f354f01785"}, + {file = "exceptiongroup-1.1.3-py3-none-any.whl", hash = "sha256:343280667a4585d195ca1cf9cef84a4e178c4b6cf2274caef9859782b567d5e3"}, + {file = "exceptiongroup-1.1.3.tar.gz", hash = "sha256:097acd85d473d75af5bb98e41b61ff7fe35efe6675e4f9370ec6ec5126d160e9"}, ] [package.extras] @@ -426,20 +457,19 @@ test = ["pytest (>=6)"] [[package]] name = "flake8" -version = "3.9.2" +version = "5.0.4" description = "the modular source code checker: pep8 pyflakes and co" optional = false -python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" +python-versions = ">=3.6.1" files = [ - {file = "flake8-3.9.2-py2.py3-none-any.whl", hash = "sha256:bf8fd333346d844f616e8d47905ef3a3384edae6b4e9beb0c5101e25e3110907"}, - {file = "flake8-3.9.2.tar.gz", hash = "sha256:07528381786f2a6237b061f6e96610a4167b226cb926e2aa2b6b1d78057c576b"}, + {file = "flake8-5.0.4-py2.py3-none-any.whl", hash = "sha256:7a1cf6b73744f5806ab95e526f6f0d8c01c66d7bbe349562d22dfca20610b248"}, + {file = "flake8-5.0.4.tar.gz", hash = "sha256:6fbe320aad8d6b95cec8b8e47bc933004678dc63095be98528b7bdd2a9f510db"}, ] [package.dependencies] -importlib-metadata = {version = "*", markers = "python_version < \"3.8\""} -mccabe = ">=0.6.0,<0.7.0" -pycodestyle = ">=2.7.0,<2.8.0" -pyflakes = ">=2.3.0,<2.4.0" +mccabe = ">=0.7.0,<0.8.0" +pycodestyle = ">=2.9.0,<2.10.0" +pyflakes = ">=2.5.0,<2.6.0" [[package]] name = "future" @@ -484,32 +514,30 @@ smmap = ">=3.0.1,<6" [[package]] name = "gitpython" -version = "3.1.31" +version = "3.1.35" description = "GitPython is a Python library used to interact with Git repositories" optional = false python-versions = ">=3.7" files = [ - {file = "GitPython-3.1.31-py3-none-any.whl", hash = "sha256:f04893614f6aa713a60cbbe1e6a97403ef633103cdd0ef5eb6efe0deb98dbe8d"}, - {file = "GitPython-3.1.31.tar.gz", hash = "sha256:8ce3bcf69adfdf7c7d503e78fd3b1c492af782d58893b650adb2ac8912ddd573"}, + {file = "GitPython-3.1.35-py3-none-any.whl", hash = "sha256:c19b4292d7a1d3c0f653858db273ff8a6614100d1eb1528b014ec97286193c09"}, + {file = "GitPython-3.1.35.tar.gz", hash = "sha256:9cbefbd1789a5fe9bcf621bb34d3f441f3a90c8461d377f84eda73e721d9b06b"}, ] [package.dependencies] gitdb = ">=4.0.1,<5" -typing-extensions = {version = ">=3.7.4.3", markers = "python_version < \"3.8\""} [[package]] name = "griffe" -version = "0.27.5" +version = "0.36.1" description = "Signatures for entire Python programs. Extract the structure, the frame, the skeleton of your project, to generate API documentation or find breaking changes in your API." optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "griffe-0.27.5-py3-none-any.whl", hash = "sha256:15b48fc3cebfc1c1a1a2f6e8177f6644a4a54517322e08e224fdf671454b34d7"}, - {file = "griffe-0.27.5.tar.gz", hash = "sha256:96fbc7a264bdb32b4da227bed6a16f2509e028a12d7471dbb48c2785bb01817f"}, + {file = "griffe-0.36.1-py3-none-any.whl", hash = "sha256:859b653fcde0a0af0e841a0109bac2b63a2f683132ae1ec8dae5fa81e94617a0"}, + {file = "griffe-0.36.1.tar.gz", hash = "sha256:11df63f1c85f605c73e4485de70ec13784049695d228241b0b582364a20c0536"}, ] [package.dependencies] -cached-property = {version = "*", markers = "python_version < \"3.8\""} colorama = ">=0.4" [[package]] @@ -523,18 +551,15 @@ files = [ {file = "h11-0.14.0.tar.gz", hash = "sha256:8f19fbbe99e72420ff35c00b27a34cb9937e902a8b810e2c88300c6f0a3b699d"}, ] -[package.dependencies] -typing-extensions = {version = "*", markers = "python_version < \"3.8\""} - [[package]] name = "httpcore" -version = "0.17.2" +version = "0.17.3" description = "A minimal low-level HTTP client." optional = false python-versions = ">=3.7" files = [ - {file = "httpcore-0.17.2-py3-none-any.whl", hash = "sha256:5581b9c12379c4288fe70f43c710d16060c10080617001e6b22a3b6dbcbefd36"}, - {file = "httpcore-0.17.2.tar.gz", hash = "sha256:125f8375ab60036db632f34f4b627a9ad085048eef7cb7d2616fea0f739f98af"}, + {file = "httpcore-0.17.3-py3-none-any.whl", hash = "sha256:c2789b767ddddfa2a5782e3199b2b7f6894540b17b16ec26b2c4d8e103510b87"}, + {file = "httpcore-0.17.3.tar.gz", hash = "sha256:a6f30213335e34c1ade7be6ec7c47f19f50c56db36abef1a9dfa3815b1cb3888"}, ] [package.dependencies] @@ -593,7 +618,6 @@ files = [ ] [package.dependencies] -typing-extensions = {version = ">=3.6.4", markers = "python_version < \"3.8\""} zipp = ">=0.5" [package.extras] @@ -614,28 +638,28 @@ files = [ [[package]] name = "invoke" -version = "2.1.2" +version = "2.2.0" description = "Pythonic task execution" optional = false python-versions = ">=3.6" files = [ - {file = "invoke-2.1.2-py3-none-any.whl", hash = "sha256:bfc904df1c9e9fe1a881933de661fe054b8db616ff2c4cf78e00407fe473ba5d"}, - {file = "invoke-2.1.2.tar.gz", hash = "sha256:a6cc1f06f75bacd0b1e11488fa3bf3e62f85e31f62e2c0172188613ba5b070e2"}, + {file = "invoke-2.2.0-py3-none-any.whl", hash = "sha256:6ea924cc53d4f78e3d98bc436b08069a03077e6f85ad1ddaa8a116d7dad15820"}, + {file = "invoke-2.2.0.tar.gz", hash = "sha256:ee6cbb101af1a859c7fe84f2a264c059020b0cb7fe3535f9424300ab568f6bd5"}, ] [[package]] name = "isort" -version = "5.11.5" +version = "5.12.0" description = "A Python utility / library to sort Python imports." optional = false -python-versions = ">=3.7.0" +python-versions = ">=3.8.0" files = [ - {file = "isort-5.11.5-py3-none-any.whl", hash = "sha256:ba1d72fb2595a01c7895a5128f9585a5cc4b6d395f1c8d514989b9a7eb2a8746"}, - {file = "isort-5.11.5.tar.gz", hash = "sha256:6be1f76a507cb2ecf16c7cf14a37e41609ca082330be4e3436a18ef74add55db"}, + {file = "isort-5.12.0-py3-none-any.whl", hash = "sha256:f84c2818376e66cf843d497486ea8fed8700b340f308f076c6fb1229dff318b6"}, + {file = "isort-5.12.0.tar.gz", hash = "sha256:8bef7dde241278824a6d83f44a544709b065191b95b6e50894bdc722fcba0504"}, ] [package.extras] -colors = ["colorama (>=0.4.3,<0.5.0)"] +colors = ["colorama (>=0.4.3)"] pipfile-deprecated-finder = ["pip-shims (>=0.5.2)", "pipreqs", "requirementslib"] plugins = ["setuptools"] requirements-deprecated-finder = ["pip-api", "pipreqs"] @@ -729,127 +753,142 @@ files = [ [[package]] name = "lxml" -version = "4.9.2" +version = "4.9.3" description = "Powerful and Pythonic XML processing library combining libxml2/libxslt with the ElementTree API." optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, != 3.4.*" files = [ - {file = "lxml-4.9.2-cp27-cp27m-macosx_10_15_x86_64.whl", hash = "sha256:76cf573e5a365e790396a5cc2b909812633409306c6531a6877c59061e42c4f2"}, - {file = "lxml-4.9.2-cp27-cp27m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:b1f42b6921d0e81b1bcb5e395bc091a70f41c4d4e55ba99c6da2b31626c44892"}, - {file = "lxml-4.9.2-cp27-cp27m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:9f102706d0ca011de571de32c3247c6476b55bb6bc65a20f682f000b07a4852a"}, - {file = "lxml-4.9.2-cp27-cp27m-win32.whl", hash = "sha256:8d0b4612b66ff5d62d03bcaa043bb018f74dfea51184e53f067e6fdcba4bd8de"}, - {file = "lxml-4.9.2-cp27-cp27m-win_amd64.whl", hash = "sha256:4c8f293f14abc8fd3e8e01c5bd86e6ed0b6ef71936ded5bf10fe7a5efefbaca3"}, - {file = "lxml-4.9.2-cp27-cp27mu-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2899456259589aa38bfb018c364d6ae7b53c5c22d8e27d0ec7609c2a1ff78b50"}, - {file = "lxml-4.9.2-cp27-cp27mu-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:6749649eecd6a9871cae297bffa4ee76f90b4504a2a2ab528d9ebe912b101975"}, - {file = "lxml-4.9.2-cp310-cp310-macosx_10_15_x86_64.whl", hash = "sha256:a08cff61517ee26cb56f1e949cca38caabe9ea9fbb4b1e10a805dc39844b7d5c"}, - {file = "lxml-4.9.2-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:85cabf64adec449132e55616e7ca3e1000ab449d1d0f9d7f83146ed5bdcb6d8a"}, - {file = "lxml-4.9.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:8340225bd5e7a701c0fa98284c849c9b9fc9238abf53a0ebd90900f25d39a4e4"}, - {file = "lxml-4.9.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:1ab8f1f932e8f82355e75dda5413a57612c6ea448069d4fb2e217e9a4bed13d4"}, - {file = "lxml-4.9.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:699a9af7dffaf67deeae27b2112aa06b41c370d5e7633e0ee0aea2e0b6c211f7"}, - {file = "lxml-4.9.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:b9cc34af337a97d470040f99ba4282f6e6bac88407d021688a5d585e44a23184"}, - {file = "lxml-4.9.2-cp310-cp310-win32.whl", hash = "sha256:d02a5399126a53492415d4906ab0ad0375a5456cc05c3fc0fc4ca11771745cda"}, - {file = "lxml-4.9.2-cp310-cp310-win_amd64.whl", hash = "sha256:a38486985ca49cfa574a507e7a2215c0c780fd1778bb6290c21193b7211702ab"}, - {file = "lxml-4.9.2-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:c83203addf554215463b59f6399835201999b5e48019dc17f182ed5ad87205c9"}, - {file = "lxml-4.9.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:2a87fa548561d2f4643c99cd13131acb607ddabb70682dcf1dff5f71f781a4bf"}, - {file = "lxml-4.9.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:d6b430a9938a5a5d85fc107d852262ddcd48602c120e3dbb02137c83d212b380"}, - {file = "lxml-4.9.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:3efea981d956a6f7173b4659849f55081867cf897e719f57383698af6f618a92"}, - {file = "lxml-4.9.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:df0623dcf9668ad0445e0558a21211d4e9a149ea8f5666917c8eeec515f0a6d1"}, - {file = "lxml-4.9.2-cp311-cp311-win32.whl", hash = "sha256:da248f93f0418a9e9d94b0080d7ebc407a9a5e6d0b57bb30db9b5cc28de1ad33"}, - {file = "lxml-4.9.2-cp311-cp311-win_amd64.whl", hash = "sha256:3818b8e2c4b5148567e1b09ce739006acfaa44ce3156f8cbbc11062994b8e8dd"}, - {file = "lxml-4.9.2-cp35-cp35m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:ca989b91cf3a3ba28930a9fc1e9aeafc2a395448641df1f387a2d394638943b0"}, - {file = "lxml-4.9.2-cp35-cp35m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:822068f85e12a6e292803e112ab876bc03ed1f03dddb80154c395f891ca6b31e"}, - {file = "lxml-4.9.2-cp35-cp35m-win32.whl", hash = "sha256:be7292c55101e22f2a3d4d8913944cbea71eea90792bf914add27454a13905df"}, - {file = "lxml-4.9.2-cp35-cp35m-win_amd64.whl", hash = "sha256:998c7c41910666d2976928c38ea96a70d1aa43be6fe502f21a651e17483a43c5"}, - {file = "lxml-4.9.2-cp36-cp36m-macosx_10_15_x86_64.whl", hash = "sha256:b26a29f0b7fc6f0897f043ca366142d2b609dc60756ee6e4e90b5f762c6adc53"}, - {file = "lxml-4.9.2-cp36-cp36m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:ab323679b8b3030000f2be63e22cdeea5b47ee0abd2d6a1dc0c8103ddaa56cd7"}, - {file = "lxml-4.9.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:689bb688a1db722485e4610a503e3e9210dcc20c520b45ac8f7533c837be76fe"}, - {file = "lxml-4.9.2-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:f49e52d174375a7def9915c9f06ec4e569d235ad428f70751765f48d5926678c"}, - {file = "lxml-4.9.2-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:36c3c175d34652a35475a73762b545f4527aec044910a651d2bf50de9c3352b1"}, - {file = "lxml-4.9.2-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:a35f8b7fa99f90dd2f5dc5a9fa12332642f087a7641289ca6c40d6e1a2637d8e"}, - {file = "lxml-4.9.2-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:58bfa3aa19ca4c0f28c5dde0ff56c520fbac6f0daf4fac66ed4c8d2fb7f22e74"}, - {file = "lxml-4.9.2-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:bc718cd47b765e790eecb74d044cc8d37d58562f6c314ee9484df26276d36a38"}, - {file = "lxml-4.9.2-cp36-cp36m-win32.whl", hash = "sha256:d5bf6545cd27aaa8a13033ce56354ed9e25ab0e4ac3b5392b763d8d04b08e0c5"}, - {file = "lxml-4.9.2-cp36-cp36m-win_amd64.whl", hash = "sha256:3ab9fa9d6dc2a7f29d7affdf3edebf6ece6fb28a6d80b14c3b2fb9d39b9322c3"}, - {file = "lxml-4.9.2-cp37-cp37m-macosx_10_15_x86_64.whl", hash = "sha256:05ca3f6abf5cf78fe053da9b1166e062ade3fa5d4f92b4ed688127ea7d7b1d03"}, - {file = "lxml-4.9.2-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:a5da296eb617d18e497bcf0a5c528f5d3b18dadb3619fbdadf4ed2356ef8d941"}, - {file = "lxml-4.9.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:04876580c050a8c5341d706dd464ff04fd597095cc8c023252566a8826505726"}, - {file = "lxml-4.9.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:c9ec3eaf616d67db0764b3bb983962b4f385a1f08304fd30c7283954e6a7869b"}, - {file = "lxml-4.9.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2a29ba94d065945944016b6b74e538bdb1751a1db6ffb80c9d3c2e40d6fa9894"}, - {file = "lxml-4.9.2-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:a82d05da00a58b8e4c0008edbc8a4b6ec5a4bc1e2ee0fb6ed157cf634ed7fa45"}, - {file = "lxml-4.9.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:223f4232855ade399bd409331e6ca70fb5578efef22cf4069a6090acc0f53c0e"}, - {file = "lxml-4.9.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:d17bc7c2ccf49c478c5bdd447594e82692c74222698cfc9b5daae7ae7e90743b"}, - {file = "lxml-4.9.2-cp37-cp37m-win32.whl", hash = "sha256:b64d891da92e232c36976c80ed7ebb383e3f148489796d8d31a5b6a677825efe"}, - {file = "lxml-4.9.2-cp37-cp37m-win_amd64.whl", hash = "sha256:a0a336d6d3e8b234a3aae3c674873d8f0e720b76bc1d9416866c41cd9500ffb9"}, - {file = "lxml-4.9.2-cp38-cp38-macosx_10_15_x86_64.whl", hash = "sha256:da4dd7c9c50c059aba52b3524f84d7de956f7fef88f0bafcf4ad7dde94a064e8"}, - {file = "lxml-4.9.2-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:821b7f59b99551c69c85a6039c65b75f5683bdc63270fec660f75da67469ca24"}, - {file = "lxml-4.9.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:e5168986b90a8d1f2f9dc1b841467c74221bd752537b99761a93d2d981e04889"}, - {file = "lxml-4.9.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:8e20cb5a47247e383cf4ff523205060991021233ebd6f924bca927fcf25cf86f"}, - {file = "lxml-4.9.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:13598ecfbd2e86ea7ae45ec28a2a54fb87ee9b9fdb0f6d343297d8e548392c03"}, - {file = "lxml-4.9.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:880bbbcbe2fca64e2f4d8e04db47bcdf504936fa2b33933efd945e1b429bea8c"}, - {file = "lxml-4.9.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:7d2278d59425777cfcb19735018d897ca8303abe67cc735f9f97177ceff8027f"}, - {file = "lxml-4.9.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:5344a43228767f53a9df6e5b253f8cdca7dfc7b7aeae52551958192f56d98457"}, - {file = "lxml-4.9.2-cp38-cp38-win32.whl", hash = "sha256:925073b2fe14ab9b87e73f9a5fde6ce6392da430f3004d8b72cc86f746f5163b"}, - {file = "lxml-4.9.2-cp38-cp38-win_amd64.whl", hash = "sha256:9b22c5c66f67ae00c0199f6055705bc3eb3fcb08d03d2ec4059a2b1b25ed48d7"}, - {file = "lxml-4.9.2-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:5f50a1c177e2fa3ee0667a5ab79fdc6b23086bc8b589d90b93b4bd17eb0e64d1"}, - {file = "lxml-4.9.2-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:090c6543d3696cbe15b4ac6e175e576bcc3f1ccfbba970061b7300b0c15a2140"}, - {file = "lxml-4.9.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:63da2ccc0857c311d764e7d3d90f429c252e83b52d1f8f1d1fe55be26827d1f4"}, - {file = "lxml-4.9.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:5b4545b8a40478183ac06c073e81a5ce4cf01bf1734962577cf2bb569a5b3bbf"}, - {file = "lxml-4.9.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2e430cd2824f05f2d4f687701144556646bae8f249fd60aa1e4c768ba7018947"}, - {file = "lxml-4.9.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:6804daeb7ef69e7b36f76caddb85cccd63d0c56dedb47555d2fc969e2af6a1a5"}, - {file = "lxml-4.9.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:a6e441a86553c310258aca15d1c05903aaf4965b23f3bc2d55f200804e005ee5"}, - {file = "lxml-4.9.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ca34efc80a29351897e18888c71c6aca4a359247c87e0b1c7ada14f0ab0c0fb2"}, - {file = "lxml-4.9.2-cp39-cp39-win32.whl", hash = "sha256:6b418afe5df18233fc6b6093deb82a32895b6bb0b1155c2cdb05203f583053f1"}, - {file = "lxml-4.9.2-cp39-cp39-win_amd64.whl", hash = "sha256:f1496ea22ca2c830cbcbd473de8f114a320da308438ae65abad6bab7867fe38f"}, - {file = "lxml-4.9.2-pp37-pypy37_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:b264171e3143d842ded311b7dccd46ff9ef34247129ff5bf5066123c55c2431c"}, - {file = "lxml-4.9.2-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:0dc313ef231edf866912e9d8f5a042ddab56c752619e92dfd3a2c277e6a7299a"}, - {file = "lxml-4.9.2-pp38-pypy38_pp73-macosx_10_15_x86_64.whl", hash = "sha256:16efd54337136e8cd72fb9485c368d91d77a47ee2d42b057564aae201257d419"}, - {file = "lxml-4.9.2-pp38-pypy38_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:0f2b1e0d79180f344ff9f321327b005ca043a50ece8713de61d1cb383fb8ac05"}, - {file = "lxml-4.9.2-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:7b770ed79542ed52c519119473898198761d78beb24b107acf3ad65deae61f1f"}, - {file = "lxml-4.9.2-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:efa29c2fe6b4fdd32e8ef81c1528506895eca86e1d8c4657fda04c9b3786ddf9"}, - {file = "lxml-4.9.2-pp39-pypy39_pp73-macosx_10_15_x86_64.whl", hash = "sha256:7e91ee82f4199af8c43d8158024cbdff3d931df350252288f0d4ce656df7f3b5"}, - {file = "lxml-4.9.2-pp39-pypy39_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:b23e19989c355ca854276178a0463951a653309fb8e57ce674497f2d9f208746"}, - {file = "lxml-4.9.2-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:01d36c05f4afb8f7c20fd9ed5badca32a2029b93b1750f571ccc0b142531caf7"}, - {file = "lxml-4.9.2-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:7b515674acfdcadb0eb5d00d8a709868173acece5cb0be3dd165950cbfdf5409"}, - {file = "lxml-4.9.2.tar.gz", hash = "sha256:2455cfaeb7ac70338b3257f41e21f0724f4b5b0c0e7702da67ee6c3640835b67"}, + {file = "lxml-4.9.3-cp27-cp27m-macosx_11_0_x86_64.whl", hash = "sha256:b0a545b46b526d418eb91754565ba5b63b1c0b12f9bd2f808c852d9b4b2f9b5c"}, + {file = "lxml-4.9.3-cp27-cp27m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:075b731ddd9e7f68ad24c635374211376aa05a281673ede86cbe1d1b3455279d"}, + {file = "lxml-4.9.3-cp27-cp27m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:1e224d5755dba2f4a9498e150c43792392ac9b5380aa1b845f98a1618c94eeef"}, + {file = "lxml-4.9.3-cp27-cp27m-win32.whl", hash = "sha256:2c74524e179f2ad6d2a4f7caf70e2d96639c0954c943ad601a9e146c76408ed7"}, + {file = "lxml-4.9.3-cp27-cp27m-win_amd64.whl", hash = "sha256:4f1026bc732b6a7f96369f7bfe1a4f2290fb34dce00d8644bc3036fb351a4ca1"}, + {file = "lxml-4.9.3-cp27-cp27mu-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:c0781a98ff5e6586926293e59480b64ddd46282953203c76ae15dbbbf302e8bb"}, + {file = "lxml-4.9.3-cp27-cp27mu-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:cef2502e7e8a96fe5ad686d60b49e1ab03e438bd9123987994528febd569868e"}, + {file = "lxml-4.9.3-cp310-cp310-macosx_11_0_x86_64.whl", hash = "sha256:b86164d2cff4d3aaa1f04a14685cbc072efd0b4f99ca5708b2ad1b9b5988a991"}, + {file = "lxml-4.9.3-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:42871176e7896d5d45138f6d28751053c711ed4d48d8e30b498da155af39aebd"}, + {file = "lxml-4.9.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:ae8b9c6deb1e634ba4f1930eb67ef6e6bf6a44b6eb5ad605642b2d6d5ed9ce3c"}, + {file = "lxml-4.9.3-cp310-cp310-manylinux_2_28_aarch64.whl", hash = "sha256:411007c0d88188d9f621b11d252cce90c4a2d1a49db6c068e3c16422f306eab8"}, + {file = "lxml-4.9.3-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:cd47b4a0d41d2afa3e58e5bf1f62069255aa2fd6ff5ee41604418ca925911d76"}, + {file = "lxml-4.9.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:0e2cb47860da1f7e9a5256254b74ae331687b9672dfa780eed355c4c9c3dbd23"}, + {file = "lxml-4.9.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:1247694b26342a7bf47c02e513d32225ededd18045264d40758abeb3c838a51f"}, + {file = "lxml-4.9.3-cp310-cp310-win32.whl", hash = "sha256:cdb650fc86227eba20de1a29d4b2c1bfe139dc75a0669270033cb2ea3d391b85"}, + {file = "lxml-4.9.3-cp310-cp310-win_amd64.whl", hash = "sha256:97047f0d25cd4bcae81f9ec9dc290ca3e15927c192df17331b53bebe0e3ff96d"}, + {file = "lxml-4.9.3-cp311-cp311-macosx_11_0_universal2.whl", hash = "sha256:1f447ea5429b54f9582d4b955f5f1985f278ce5cf169f72eea8afd9502973dd5"}, + {file = "lxml-4.9.3-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:57d6ba0ca2b0c462f339640d22882acc711de224d769edf29962b09f77129cbf"}, + {file = "lxml-4.9.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:9767e79108424fb6c3edf8f81e6730666a50feb01a328f4a016464a5893f835a"}, + {file = "lxml-4.9.3-cp311-cp311-manylinux_2_28_aarch64.whl", hash = "sha256:71c52db65e4b56b8ddc5bb89fb2e66c558ed9d1a74a45ceb7dcb20c191c3df2f"}, + {file = "lxml-4.9.3-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:d73d8ecf8ecf10a3bd007f2192725a34bd62898e8da27eb9d32a58084f93962b"}, + {file = "lxml-4.9.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:0a3d3487f07c1d7f150894c238299934a2a074ef590b583103a45002035be120"}, + {file = "lxml-4.9.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:9e28c51fa0ce5674be9f560c6761c1b441631901993f76700b1b30ca6c8378d6"}, + {file = "lxml-4.9.3-cp311-cp311-win32.whl", hash = "sha256:0bfd0767c5c1de2551a120673b72e5d4b628737cb05414f03c3277bf9bed3305"}, + {file = "lxml-4.9.3-cp311-cp311-win_amd64.whl", hash = "sha256:25f32acefac14ef7bd53e4218fe93b804ef6f6b92ffdb4322bb6d49d94cad2bc"}, + {file = "lxml-4.9.3-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:d3ff32724f98fbbbfa9f49d82852b159e9784d6094983d9a8b7f2ddaebb063d4"}, + {file = "lxml-4.9.3-cp312-cp312-manylinux_2_28_aarch64.whl", hash = "sha256:48d6ed886b343d11493129e019da91d4039826794a3e3027321c56d9e71505be"}, + {file = "lxml-4.9.3-cp312-cp312-manylinux_2_28_x86_64.whl", hash = "sha256:9a92d3faef50658dd2c5470af249985782bf754c4e18e15afb67d3ab06233f13"}, + {file = "lxml-4.9.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:b4e4bc18382088514ebde9328da057775055940a1f2e18f6ad2d78aa0f3ec5b9"}, + {file = "lxml-4.9.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:fc9b106a1bf918db68619fdcd6d5ad4f972fdd19c01d19bdb6bf63f3589a9ec5"}, + {file = "lxml-4.9.3-cp312-cp312-win_amd64.whl", hash = "sha256:d37017287a7adb6ab77e1c5bee9bcf9660f90ff445042b790402a654d2ad81d8"}, + {file = "lxml-4.9.3-cp35-cp35m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:56dc1f1ebccc656d1b3ed288f11e27172a01503fc016bcabdcbc0978b19352b7"}, + {file = "lxml-4.9.3-cp35-cp35m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:578695735c5a3f51569810dfebd05dd6f888147a34f0f98d4bb27e92b76e05c2"}, + {file = "lxml-4.9.3-cp35-cp35m-win32.whl", hash = "sha256:704f61ba8c1283c71b16135caf697557f5ecf3e74d9e453233e4771d68a1f42d"}, + {file = "lxml-4.9.3-cp35-cp35m-win_amd64.whl", hash = "sha256:c41bfca0bd3532d53d16fd34d20806d5c2b1ace22a2f2e4c0008570bf2c58833"}, + {file = "lxml-4.9.3-cp36-cp36m-macosx_11_0_x86_64.whl", hash = "sha256:64f479d719dc9f4c813ad9bb6b28f8390360660b73b2e4beb4cb0ae7104f1c12"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:dd708cf4ee4408cf46a48b108fb9427bfa00b9b85812a9262b5c668af2533ea5"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5c31c7462abdf8f2ac0577d9f05279727e698f97ecbb02f17939ea99ae8daa98"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:e3cd95e10c2610c360154afdc2f1480aea394f4a4f1ea0a5eacce49640c9b190"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_28_x86_64.whl", hash = "sha256:4930be26af26ac545c3dffb662521d4e6268352866956672231887d18f0eaab2"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:4aec80cde9197340bc353d2768e2a75f5f60bacda2bab72ab1dc499589b3878c"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:14e019fd83b831b2e61baed40cab76222139926b1fb5ed0e79225bc0cae14584"}, + {file = "lxml-4.9.3-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:0c0850c8b02c298d3c7006b23e98249515ac57430e16a166873fc47a5d549287"}, + {file = "lxml-4.9.3-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:aca086dc5f9ef98c512bac8efea4483eb84abbf926eaeedf7b91479feb092458"}, + {file = "lxml-4.9.3-cp36-cp36m-win32.whl", hash = "sha256:50baa9c1c47efcaef189f31e3d00d697c6d4afda5c3cde0302d063492ff9b477"}, + {file = "lxml-4.9.3-cp36-cp36m-win_amd64.whl", hash = "sha256:bef4e656f7d98aaa3486d2627e7d2df1157d7e88e7efd43a65aa5dd4714916cf"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:46f409a2d60f634fe550f7133ed30ad5321ae2e6630f13657fb9479506b00601"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:4c28a9144688aef80d6ea666c809b4b0e50010a2aca784c97f5e6bf143d9f129"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:141f1d1a9b663c679dc524af3ea1773e618907e96075262726c7612c02b149a4"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_28_x86_64.whl", hash = "sha256:53ace1c1fd5a74ef662f844a0413446c0629d151055340e9893da958a374f70d"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:17a753023436a18e27dd7769e798ce302963c236bc4114ceee5b25c18c52c693"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:7d298a1bd60c067ea75d9f684f5f3992c9d6766fadbc0bcedd39750bf344c2f4"}, + {file = "lxml-4.9.3-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:081d32421db5df44c41b7f08a334a090a545c54ba977e47fd7cc2deece78809a"}, + {file = "lxml-4.9.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:23eed6d7b1a3336ad92d8e39d4bfe09073c31bfe502f20ca5116b2a334f8ec02"}, + {file = "lxml-4.9.3-cp37-cp37m-win32.whl", hash = "sha256:1509dd12b773c02acd154582088820893109f6ca27ef7291b003d0e81666109f"}, + {file = "lxml-4.9.3-cp37-cp37m-win_amd64.whl", hash = "sha256:120fa9349a24c7043854c53cae8cec227e1f79195a7493e09e0c12e29f918e52"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:4d2d1edbca80b510443f51afd8496be95529db04a509bc8faee49c7b0fb6d2cc"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:8d7e43bd40f65f7d97ad8ef5c9b1778943d02f04febef12def25f7583d19baac"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:71d66ee82e7417828af6ecd7db817913cb0cf9d4e61aa0ac1fde0583d84358db"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_28_x86_64.whl", hash = "sha256:6fc3c450eaa0b56f815c7b62f2b7fba7266c4779adcf1cece9e6deb1de7305ce"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:65299ea57d82fb91c7f019300d24050c4ddeb7c5a190e076b5f48a2b43d19c42"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:eadfbbbfb41b44034a4c757fd5d70baccd43296fb894dba0295606a7cf3124aa"}, + {file = "lxml-4.9.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:3e9bdd30efde2b9ccfa9cb5768ba04fe71b018a25ea093379c857c9dad262c40"}, + {file = "lxml-4.9.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:fcdd00edfd0a3001e0181eab3e63bd5c74ad3e67152c84f93f13769a40e073a7"}, + {file = "lxml-4.9.3-cp38-cp38-win32.whl", hash = "sha256:57aba1bbdf450b726d58b2aea5fe47c7875f5afb2c4a23784ed78f19a0462574"}, + {file = "lxml-4.9.3-cp38-cp38-win_amd64.whl", hash = "sha256:92af161ecbdb2883c4593d5ed4815ea71b31fafd7fd05789b23100d081ecac96"}, + {file = "lxml-4.9.3-cp39-cp39-macosx_11_0_x86_64.whl", hash = "sha256:9bb6ad405121241e99a86efff22d3ef469024ce22875a7ae045896ad23ba2340"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:8ed74706b26ad100433da4b9d807eae371efaa266ffc3e9191ea436087a9d6a7"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:fbf521479bcac1e25a663df882c46a641a9bff6b56dc8b0fafaebd2f66fb231b"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_28_aarch64.whl", hash = "sha256:303bf1edce6ced16bf67a18a1cf8339d0db79577eec5d9a6d4a80f0fb10aa2da"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_28_x86_64.whl", hash = "sha256:5515edd2a6d1a5a70bfcdee23b42ec33425e405c5b351478ab7dc9347228f96e"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:690dafd0b187ed38583a648076865d8c229661ed20e48f2335d68e2cf7dc829d"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:b6420a005548ad52154c8ceab4a1290ff78d757f9e5cbc68f8c77089acd3c432"}, + {file = "lxml-4.9.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:bb3bb49c7a6ad9d981d734ef7c7193bc349ac338776a0360cc671eaee89bcf69"}, + {file = "lxml-4.9.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:d27be7405547d1f958b60837dc4c1007da90b8b23f54ba1f8b728c78fdb19d50"}, + {file = "lxml-4.9.3-cp39-cp39-win32.whl", hash = "sha256:8df133a2ea5e74eef5e8fc6f19b9e085f758768a16e9877a60aec455ed2609b2"}, + {file = "lxml-4.9.3-cp39-cp39-win_amd64.whl", hash = "sha256:4dd9a263e845a72eacb60d12401e37c616438ea2e5442885f65082c276dfb2b2"}, + {file = "lxml-4.9.3-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:6689a3d7fd13dc687e9102a27e98ef33730ac4fe37795d5036d18b4d527abd35"}, + {file = "lxml-4.9.3-pp37-pypy37_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:f6bdac493b949141b733c5345b6ba8f87a226029cbabc7e9e121a413e49441e0"}, + {file = "lxml-4.9.3-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:05186a0f1346ae12553d66df1cfce6f251589fea3ad3da4f3ef4e34b2d58c6a3"}, + {file = "lxml-4.9.3-pp37-pypy37_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:c2006f5c8d28dee289f7020f721354362fa304acbaaf9745751ac4006650254b"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-macosx_11_0_x86_64.whl", hash = "sha256:5c245b783db29c4e4fbbbfc9c5a78be496c9fea25517f90606aa1f6b2b3d5f7b"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:4fb960a632a49f2f089d522f70496640fdf1218f1243889da3822e0a9f5f3ba7"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:50670615eaf97227d5dc60de2dc99fb134a7130d310d783314e7724bf163f75d"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:9719fe17307a9e814580af1f5c6e05ca593b12fb7e44fe62450a5384dbf61b4b"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:3331bece23c9ee066e0fb3f96c61322b9e0f54d775fccefff4c38ca488de283a"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-macosx_11_0_x86_64.whl", hash = "sha256:ed667f49b11360951e201453fc3967344d0d0263aa415e1619e85ae7fd17b4e0"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:8b77946fd508cbf0fccd8e400a7f71d4ac0e1595812e66025bac475a8e811694"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:e4da8ca0c0c0aea88fd46be8e44bd49716772358d648cce45fe387f7b92374a7"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:fe4bda6bd4340caa6e5cf95e73f8fea5c4bfc55763dd42f1b50a94c1b4a2fbd4"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:f3df3db1d336b9356dd3112eae5f5c2b8b377f3bc826848567f10bfddfee77e9"}, + {file = "lxml-4.9.3.tar.gz", hash = "sha256:48628bd53a426c9eb9bc066a923acaa0878d1e86129fd5359aee99285f4eed9c"}, ] [package.extras] cssselect = ["cssselect (>=0.7)"] html5 = ["html5lib"] htmlsoup = ["BeautifulSoup4"] -source = ["Cython (>=0.29.7)"] +source = ["Cython (>=0.29.35)"] [[package]] name = "markdown" -version = "3.3.7" -description = "Python implementation of Markdown." +version = "3.4.4" +description = "Python implementation of John Gruber's Markdown." optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" files = [ - {file = "Markdown-3.3.7-py3-none-any.whl", hash = "sha256:f5da449a6e1c989a4cea2631aa8ee67caa5a2ef855d551c88f9e309f4634c621"}, - {file = "Markdown-3.3.7.tar.gz", hash = "sha256:cbb516f16218e643d8e0a95b309f77eb118cb138d39a4f27851e6a63581db874"}, + {file = "Markdown-3.4.4-py3-none-any.whl", hash = "sha256:a4c1b65c0957b4bd9e7d86ddc7b3c9868fb9670660f6f99f6d1bca8954d5a941"}, + {file = "Markdown-3.4.4.tar.gz", hash = "sha256:225c6123522495d4119a90b3a3ba31a1e87a70369e03f14799ea9c0d7183a3d6"}, ] [package.dependencies] importlib-metadata = {version = ">=4.4", markers = "python_version < \"3.10\""} [package.extras] +docs = ["mdx-gh-links (>=0.2)", "mkdocs (>=1.0)", "mkdocs-nature (>=0.4)"] testing = ["coverage", "pyyaml"] [[package]] name = "markdown-it-py" -version = "2.2.0" +version = "3.0.0" description = "Python port of markdown-it. Markdown parsing, done right!" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "markdown-it-py-2.2.0.tar.gz", hash = "sha256:7c9a5e412688bc771c67432cbfebcdd686c93ce6484913dccf06cb5a0bea35a1"}, - {file = "markdown_it_py-2.2.0-py3-none-any.whl", hash = "sha256:5a35f8d1870171d9acc47b99612dc146129b631baf04970128b568f190d0cc30"}, + {file = "markdown-it-py-3.0.0.tar.gz", hash = "sha256:e3f60a94fa066dc52ec76661e37c851cb232d92f9886b15cb560aaada2df8feb"}, + {file = "markdown_it_py-3.0.0-py3-none-any.whl", hash = "sha256:355216845c60bd96232cd8d8c40e8f9765cc86f46880e43a8fd22dc1a1a8cab1"}, ] [package.dependencies] mdurl = ">=0.1,<1.0" -typing_extensions = {version = ">=3.7.4", markers = "python_version < \"3.8\""} [package.extras] benchmarking = ["psutil", "pytest", "pytest-benchmark"] @@ -858,77 +897,103 @@ compare = ["commonmark (>=0.9,<1.0)", "markdown (>=3.4,<4.0)", "mistletoe (>=1.0 linkify = ["linkify-it-py (>=1,<3)"] plugins = ["mdit-py-plugins"] profiling = ["gprof2dot"] -rtd = ["attrs", "myst-parser", "pyyaml", "sphinx", "sphinx-copybutton", "sphinx-design", "sphinx_book_theme"] +rtd = ["jupyter_sphinx", "mdit-py-plugins", "myst-parser", "pyyaml", "sphinx", "sphinx-copybutton", "sphinx-design", "sphinx_book_theme"] testing = ["coverage", "pytest", "pytest-cov", "pytest-regressions"] +[[package]] +name = "markdown2" +version = "2.4.10" +description = "A fast and complete Python implementation of Markdown" +optional = false +python-versions = ">=3.5, <4" +files = [ + {file = "markdown2-2.4.10-py2.py3-none-any.whl", hash = "sha256:e6105800483783831f5dc54f827aa5b44eb137ecef5a70293d8ecfbb4109ecc6"}, + {file = "markdown2-2.4.10.tar.gz", hash = "sha256:cdba126d90dc3aef6f4070ac342f974d63f415678959329cc7909f96cc235d72"}, +] + +[package.extras] +all = ["pygments (>=2.7.3)", "wavedrom"] +code-syntax-highlighting = ["pygments (>=2.7.3)"] +wavedrom = ["wavedrom"] + [[package]] name = "markupsafe" -version = "2.1.2" +version = "2.1.3" description = "Safely add untrusted strings to HTML/XML markup." optional = false python-versions = ">=3.7" files = [ - {file = "MarkupSafe-2.1.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:665a36ae6f8f20a4676b53224e33d456a6f5a72657d9c83c2aa00765072f31f7"}, - {file = "MarkupSafe-2.1.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:340bea174e9761308703ae988e982005aedf427de816d1afe98147668cc03036"}, - {file = "MarkupSafe-2.1.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:22152d00bf4a9c7c83960521fc558f55a1adbc0631fbb00a9471e097b19d72e1"}, - {file = "MarkupSafe-2.1.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:28057e985dace2f478e042eaa15606c7efccb700797660629da387eb289b9323"}, - {file = "MarkupSafe-2.1.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ca244fa73f50a800cf8c3ebf7fd93149ec37f5cb9596aa8873ae2c1d23498601"}, - {file = "MarkupSafe-2.1.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:d9d971ec1e79906046aa3ca266de79eac42f1dbf3612a05dc9368125952bd1a1"}, - {file = "MarkupSafe-2.1.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:7e007132af78ea9df29495dbf7b5824cb71648d7133cf7848a2a5dd00d36f9ff"}, - {file = "MarkupSafe-2.1.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:7313ce6a199651c4ed9d7e4cfb4aa56fe923b1adf9af3b420ee14e6d9a73df65"}, - {file = "MarkupSafe-2.1.2-cp310-cp310-win32.whl", hash = "sha256:c4a549890a45f57f1ebf99c067a4ad0cb423a05544accaf2b065246827ed9603"}, - {file = "MarkupSafe-2.1.2-cp310-cp310-win_amd64.whl", hash = "sha256:835fb5e38fd89328e9c81067fd642b3593c33e1e17e2fdbf77f5676abb14a156"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:2ec4f2d48ae59bbb9d1f9d7efb9236ab81429a764dedca114f5fdabbc3788013"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:608e7073dfa9e38a85d38474c082d4281f4ce276ac0010224eaba11e929dd53a"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:65608c35bfb8a76763f37036547f7adfd09270fbdbf96608be2bead319728fcd"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f2bfb563d0211ce16b63c7cb9395d2c682a23187f54c3d79bfec33e6705473c6"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:da25303d91526aac3672ee6d49a2f3db2d9502a4a60b55519feb1a4c7714e07d"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:9cad97ab29dfc3f0249b483412c85c8ef4766d96cdf9dcf5a1e3caa3f3661cf1"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:085fd3201e7b12809f9e6e9bc1e5c96a368c8523fad5afb02afe3c051ae4afcc"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:1bea30e9bf331f3fef67e0a3877b2288593c98a21ccb2cf29b74c581a4eb3af0"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-win32.whl", hash = "sha256:7df70907e00c970c60b9ef2938d894a9381f38e6b9db73c5be35e59d92e06625"}, - {file = "MarkupSafe-2.1.2-cp311-cp311-win_amd64.whl", hash = "sha256:e55e40ff0cc8cc5c07996915ad367fa47da6b3fc091fdadca7f5403239c5fec3"}, - {file = "MarkupSafe-2.1.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:a6e40afa7f45939ca356f348c8e23048e02cb109ced1eb8420961b2f40fb373a"}, - {file = "MarkupSafe-2.1.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cf877ab4ed6e302ec1d04952ca358b381a882fbd9d1b07cccbfd61783561f98a"}, - {file = "MarkupSafe-2.1.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:63ba06c9941e46fa389d389644e2d8225e0e3e5ebcc4ff1ea8506dce646f8c8a"}, - {file = "MarkupSafe-2.1.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f1cd098434e83e656abf198f103a8207a8187c0fc110306691a2e94a78d0abb2"}, - {file = "MarkupSafe-2.1.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:55f44b440d491028addb3b88f72207d71eeebfb7b5dbf0643f7c023ae1fba619"}, - {file = "MarkupSafe-2.1.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:a6f2fcca746e8d5910e18782f976489939d54a91f9411c32051b4aab2bd7c513"}, - {file = "MarkupSafe-2.1.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:0b462104ba25f1ac006fdab8b6a01ebbfbce9ed37fd37fd4acd70c67c973e460"}, - {file = "MarkupSafe-2.1.2-cp37-cp37m-win32.whl", hash = "sha256:7668b52e102d0ed87cb082380a7e2e1e78737ddecdde129acadb0eccc5423859"}, - {file = "MarkupSafe-2.1.2-cp37-cp37m-win_amd64.whl", hash = "sha256:6d6607f98fcf17e534162f0709aaad3ab7a96032723d8ac8750ffe17ae5a0666"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:a806db027852538d2ad7555b203300173dd1b77ba116de92da9afbc3a3be3eed"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:a4abaec6ca3ad8660690236d11bfe28dfd707778e2442b45addd2f086d6ef094"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f03a532d7dee1bed20bc4884194a16160a2de9ffc6354b3878ec9682bb623c54"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4cf06cdc1dda95223e9d2d3c58d3b178aa5dacb35ee7e3bbac10e4e1faacb419"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:22731d79ed2eb25059ae3df1dfc9cb1546691cc41f4e3130fe6bfbc3ecbbecfa"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:f8ffb705ffcf5ddd0e80b65ddf7bed7ee4f5a441ea7d3419e861a12eaf41af58"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:8db032bf0ce9022a8e41a22598eefc802314e81b879ae093f36ce9ddf39ab1ba"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:2298c859cfc5463f1b64bd55cb3e602528db6fa0f3cfd568d3605c50678f8f03"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-win32.whl", hash = "sha256:50c42830a633fa0cf9e7d27664637532791bfc31c731a87b202d2d8ac40c3ea2"}, - {file = "MarkupSafe-2.1.2-cp38-cp38-win_amd64.whl", hash = "sha256:bb06feb762bade6bf3c8b844462274db0c76acc95c52abe8dbed28ae3d44a147"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:99625a92da8229df6d44335e6fcc558a5037dd0a760e11d84be2260e6f37002f"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:8bca7e26c1dd751236cfb0c6c72d4ad61d986e9a41bbf76cb445f69488b2a2bd"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:40627dcf047dadb22cd25ea7ecfe9cbf3bbbad0482ee5920b582f3809c97654f"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:40dfd3fefbef579ee058f139733ac336312663c6706d1163b82b3003fb1925c4"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:090376d812fb6ac5f171e5938e82e7f2d7adc2b629101cec0db8b267815c85e2"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:2e7821bffe00aa6bd07a23913b7f4e01328c3d5cc0b40b36c0bd81d362faeb65"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:c0a33bc9f02c2b17c3ea382f91b4db0e6cde90b63b296422a939886a7a80de1c"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:b8526c6d437855442cdd3d87eede9c425c4445ea011ca38d937db299382e6fa3"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-win32.whl", hash = "sha256:137678c63c977754abe9086a3ec011e8fd985ab90631145dfb9294ad09c102a7"}, - {file = "MarkupSafe-2.1.2-cp39-cp39-win_amd64.whl", hash = "sha256:0576fe974b40a400449768941d5d0858cc624e3249dfd1e0c33674e5c7ca7aed"}, - {file = "MarkupSafe-2.1.2.tar.gz", hash = "sha256:abcabc8c2b26036d62d4c746381a6f7cf60aafcc653198ad678306986b09450d"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:cd0f502fe016460680cd20aaa5a76d241d6f35a1c3350c474bac1273803893fa"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:e09031c87a1e51556fdcb46e5bd4f59dfb743061cf93c4d6831bf894f125eb57"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:68e78619a61ecf91e76aa3e6e8e33fc4894a2bebe93410754bd28fce0a8a4f9f"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:65c1a9bcdadc6c28eecee2c119465aebff8f7a584dd719facdd9e825ec61ab52"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:525808b8019e36eb524b8c68acdd63a37e75714eac50e988180b169d64480a00"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:962f82a3086483f5e5f64dbad880d31038b698494799b097bc59c2edf392fce6"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:aa7bd130efab1c280bed0f45501b7c8795f9fdbeb02e965371bbef3523627779"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:c9c804664ebe8f83a211cace637506669e7890fec1b4195b505c214e50dd4eb7"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-win32.whl", hash = "sha256:10bbfe99883db80bdbaff2dcf681dfc6533a614f700da1287707e8a5d78a8431"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-win_amd64.whl", hash = "sha256:1577735524cdad32f9f694208aa75e422adba74f1baee7551620e43a3141f559"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:ad9e82fb8f09ade1c3e1b996a6337afac2b8b9e365f926f5a61aacc71adc5b3c"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3c0fae6c3be832a0a0473ac912810b2877c8cb9d76ca48de1ed31e1c68386575"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b076b6226fb84157e3f7c971a47ff3a679d837cf338547532ab866c57930dbee"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bfce63a9e7834b12b87c64d6b155fdd9b3b96191b6bd334bf37db7ff1fe457f2"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:338ae27d6b8745585f87218a3f23f1512dbf52c26c28e322dbe54bcede54ccb9"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e4dd52d80b8c83fdce44e12478ad2e85c64ea965e75d66dbeafb0a3e77308fcc"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:df0be2b576a7abbf737b1575f048c23fb1d769f267ec4358296f31c2479db8f9"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:5bbe06f8eeafd38e5d0a4894ffec89378b6c6a625ff57e3028921f8ff59318ac"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-win32.whl", hash = "sha256:dd15ff04ffd7e05ffcb7fe79f1b98041b8ea30ae9234aed2a9168b5797c3effb"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-win_amd64.whl", hash = "sha256:134da1eca9ec0ae528110ccc9e48041e0828d79f24121a1a146161103c76e686"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:f698de3fd0c4e6972b92290a45bd9b1536bffe8c6759c62471efaa8acb4c37bc"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:aa57bd9cf8ae831a362185ee444e15a93ecb2e344c8e52e4d721ea3ab6ef1823"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffcc3f7c66b5f5b7931a5aa68fc9cecc51e685ef90282f4a82f0f5e9b704ad11"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:47d4f1c5f80fc62fdd7777d0d40a2e9dda0a05883ab11374334f6c4de38adffd"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1f67c7038d560d92149c060157d623c542173016c4babc0c1913cca0564b9939"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:9aad3c1755095ce347e26488214ef77e0485a3c34a50c5a5e2471dff60b9dd9c"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:14ff806850827afd6b07a5f32bd917fb7f45b046ba40c57abdb636674a8b559c"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8f9293864fe09b8149f0cc42ce56e3f0e54de883a9de90cd427f191c346eb2e1"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-win32.whl", hash = "sha256:715d3562f79d540f251b99ebd6d8baa547118974341db04f5ad06d5ea3eb8007"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-win_amd64.whl", hash = "sha256:1b8dd8c3fd14349433c79fa8abeb573a55fc0fdd769133baac1f5e07abf54aeb"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:8e254ae696c88d98da6555f5ace2279cf7cd5b3f52be2b5cf97feafe883b58d2"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cb0932dc158471523c9637e807d9bfb93e06a95cbf010f1a38b98623b929ef2b"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9402b03f1a1b4dc4c19845e5c749e3ab82d5078d16a2a4c2cd2df62d57bb0707"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ca379055a47383d02a5400cb0d110cef0a776fc644cda797db0c5696cfd7e18e"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:b7ff0f54cb4ff66dd38bebd335a38e2c22c41a8ee45aa608efc890ac3e3931bc"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:c011a4149cfbcf9f03994ec2edffcb8b1dc2d2aede7ca243746df97a5d41ce48"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:56d9f2ecac662ca1611d183feb03a3fa4406469dafe241673d521dd5ae92a155"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-win32.whl", hash = "sha256:8758846a7e80910096950b67071243da3e5a20ed2546e6392603c096778d48e0"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-win_amd64.whl", hash = "sha256:787003c0ddb00500e49a10f2844fac87aa6ce977b90b0feaaf9de23c22508b24"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:2ef12179d3a291be237280175b542c07a36e7f60718296278d8593d21ca937d4"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:2c1b19b3aaacc6e57b7e25710ff571c24d6c3613a45e905b1fde04d691b98ee0"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8afafd99945ead6e075b973fefa56379c5b5c53fd8937dad92c662da5d8fd5ee"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8c41976a29d078bb235fea9b2ecd3da465df42a562910f9022f1a03107bd02be"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d080e0a5eb2529460b30190fcfcc4199bd7f827663f858a226a81bc27beaa97e"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:69c0f17e9f5a7afdf2cc9fb2d1ce6aabdb3bafb7f38017c0b77862bcec2bbad8"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:504b320cd4b7eff6f968eddf81127112db685e81f7e36e75f9f84f0df46041c3"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:42de32b22b6b804f42c5d98be4f7e5e977ecdd9ee9b660fda1a3edf03b11792d"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-win32.whl", hash = "sha256:ceb01949af7121f9fc39f7d27f91be8546f3fb112c608bc4029aef0bab86a2a5"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-win_amd64.whl", hash = "sha256:1b40069d487e7edb2676d3fbdb2b0829ffa2cd63a2ec26c4938b2d34391b4ecc"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:8023faf4e01efadfa183e863fefde0046de576c6f14659e8782065bcece22198"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6b2b56950d93e41f33b4223ead100ea0fe11f8e6ee5f641eb753ce4b77a7042b"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9dcdfd0eaf283af041973bff14a2e143b8bd64e069f4c383416ecd79a81aab58"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:05fb21170423db021895e1ea1e1f3ab3adb85d1c2333cbc2310f2a26bc77272e"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:282c2cb35b5b673bbcadb33a585408104df04f14b2d9b01d4c345a3b92861c2c"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:ab4a0df41e7c16a1392727727e7998a467472d0ad65f3ad5e6e765015df08636"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:7ef3cb2ebbf91e330e3bb937efada0edd9003683db6b57bb108c4001f37a02ea"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:0a4e4a1aff6c7ac4cd55792abf96c915634c2b97e3cc1c7129578aa68ebd754e"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-win32.whl", hash = "sha256:fec21693218efe39aa7f8599346e90c705afa52c5b31ae019b2e57e8f6542bb2"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-win_amd64.whl", hash = "sha256:3fd4abcb888d15a94f32b75d8fd18ee162ca0c064f35b11134be77050296d6ba"}, + {file = "MarkupSafe-2.1.3.tar.gz", hash = "sha256:af598ed32d6ae86f1b747b82783958b1a4ab8f617b06fe68795c7f026abbdcad"}, ] [[package]] name = "mccabe" -version = "0.6.1" +version = "0.7.0" description = "McCabe checker, plugin for flake8" optional = false -python-versions = "*" +python-versions = ">=3.6" files = [ - {file = "mccabe-0.6.1-py2.py3-none-any.whl", hash = "sha256:ab8a6258860da4b6677da4bd2fe5dc2c659cff31b3ee4f7f5d64e79735b80d42"}, - {file = "mccabe-0.6.1.tar.gz", hash = "sha256:dd8d182285a0fe56bace7f45b5e7d1a6ebcbf524e8f3bd87eb0f125271b8831f"}, + {file = "mccabe-0.7.0-py2.py3-none-any.whl", hash = "sha256:6c2d30ab6be0e4a46919781807b4f0d834ebdd6c6e3dca0bda5a15f863427b6e"}, + {file = "mccabe-0.7.0.tar.gz", hash = "sha256:348e0240c33b60bbdf4e523192ef919f28cb2c3d7d5c7794f74009290f236325"}, ] [[package]] @@ -955,39 +1020,44 @@ files = [ [[package]] name = "mkdocs" -version = "1.3.1" +version = "1.5.2" description = "Project documentation with Markdown." optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" files = [ - {file = "mkdocs-1.3.1-py3-none-any.whl", hash = "sha256:fda92466393127d2da830bc6edc3a625a14b436316d1caf347690648e774c4f0"}, - {file = "mkdocs-1.3.1.tar.gz", hash = "sha256:a41a2ff25ce3bbacc953f9844ba07d106233cd76c88bac1f59cb1564ac0d87ed"}, + {file = "mkdocs-1.5.2-py3-none-any.whl", hash = "sha256:60a62538519c2e96fe8426654a67ee177350451616118a41596ae7c876bb7eac"}, + {file = "mkdocs-1.5.2.tar.gz", hash = "sha256:70d0da09c26cff288852471be03c23f0f521fc15cf16ac89c7a3bfb9ae8d24f9"}, ] [package.dependencies] -click = ">=3.3" +click = ">=7.0" +colorama = {version = ">=0.4", markers = "platform_system == \"Windows\""} ghp-import = ">=1.0" -importlib-metadata = ">=4.3" -Jinja2 = ">=2.10.2" -Markdown = ">=3.2.1,<3.4" +importlib-metadata = {version = ">=4.3", markers = "python_version < \"3.10\""} +jinja2 = ">=2.11.1" +markdown = ">=3.2.1" +markupsafe = ">=2.0.1" mergedeep = ">=1.3.4" packaging = ">=20.5" -PyYAML = ">=3.10" +pathspec = ">=0.11.1" +platformdirs = ">=2.2.0" +pyyaml = ">=5.1" pyyaml-env-tag = ">=0.1" watchdog = ">=2.0" [package.extras] i18n = ["babel (>=2.9.0)"] +min-versions = ["babel (==2.9.0)", "click (==7.0)", "colorama (==0.4)", "ghp-import (==1.0)", "importlib-metadata (==4.3)", "jinja2 (==2.11.1)", "markdown (==3.2.1)", "markupsafe (==2.0.1)", "mergedeep (==1.3.4)", "packaging (==20.5)", "pathspec (==0.11.1)", "platformdirs (==2.2.0)", "pyyaml (==5.1)", "pyyaml-env-tag (==0.1)", "typing-extensions (==3.10)", "watchdog (==2.0)"] [[package]] name = "mkdocs-autorefs" -version = "0.4.1" +version = "0.5.0" description = "Automatically link across pages in MkDocs." optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "mkdocs-autorefs-0.4.1.tar.gz", hash = "sha256:70748a7bd025f9ecd6d6feeba8ba63f8e891a1af55f48e366d6d6e78493aba84"}, - {file = "mkdocs_autorefs-0.4.1-py3-none-any.whl", hash = "sha256:a2248a9501b29dc0cc8ba4c09f4f47ff121945f6ce33d760f145d6f89d313f5b"}, + {file = "mkdocs_autorefs-0.5.0-py3-none-any.whl", hash = "sha256:7930fcb8ac1249f10e683967aeaddc0af49d90702af111a5e390e8b20b3d97ff"}, + {file = "mkdocs_autorefs-0.5.0.tar.gz", hash = "sha256:9a5054a94c08d28855cfab967ada10ed5be76e2bfad642302a610b252c3274c0"}, ] [package.dependencies] @@ -996,22 +1066,29 @@ mkdocs = ">=1.1" [[package]] name = "mkdocs-material" -version = "8.3.9" +version = "9.2.4" description = "Documentation that simply works" optional = false python-versions = ">=3.7" files = [ - {file = "mkdocs-material-8.3.9.tar.gz", hash = "sha256:dc82b667d2a83f0de581b46a6d0949732ab77e7638b87ea35b770b33bc02e75a"}, - {file = "mkdocs_material-8.3.9-py2.py3-none-any.whl", hash = "sha256:263f2721f3abe533b61f7c8bed435a0462620912742c919821ac2d698b4bfe67"}, + {file = "mkdocs_material-9.2.4-py3-none-any.whl", hash = "sha256:2df876367625ff5e0f7112bc19a57521ed21ce9a2b85656baf9bb7f5dc3cb987"}, + {file = "mkdocs_material-9.2.4.tar.gz", hash = "sha256:25008187b89fc376cb4ed2312b1fea4121bf2bd956442f38afdc6b4dcc21c57d"}, ] [package.dependencies] -jinja2 = ">=3.0.2" +babel = ">=2.10.3" +colorama = ">=0.4" +jinja2 = ">=3.0" +lxml = ">=4.6" markdown = ">=3.2" -mkdocs = ">=1.3.0" -mkdocs-material-extensions = ">=1.0.3" -pygments = ">=2.12" -pymdown-extensions = ">=9.4" +mkdocs = ">=1.5.2" +mkdocs-material-extensions = ">=1.1" +paginate = ">=0.5.6" +pygments = ">=2.14" +pymdown-extensions = ">=9.9.1" +readtime = ">=2.0" +regex = ">=2022.4.24" +requests = ">=2.26" [[package]] name = "mkdocs-material-extensions" @@ -1037,22 +1114,24 @@ files = [ [[package]] name = "mkdocstrings" -version = "0.19.0" +version = "0.22.0" description = "Automatic documentation from sources, for MkDocs." optional = false python-versions = ">=3.7" files = [ - {file = "mkdocstrings-0.19.0-py3-none-any.whl", hash = "sha256:3217d510d385c961f69385a670b2677e68e07b5fea4a504d86bf54c006c87c7d"}, - {file = "mkdocstrings-0.19.0.tar.gz", hash = "sha256:efa34a67bad11229d532d89f6836a8a215937548623b64f3698a1df62e01cc3e"}, + {file = "mkdocstrings-0.22.0-py3-none-any.whl", hash = "sha256:2d4095d461554ff6a778fdabdca3c00c468c2f1459d469f7a7f622a2b23212ba"}, + {file = "mkdocstrings-0.22.0.tar.gz", hash = "sha256:82a33b94150ebb3d4b5c73bab4598c3e21468c79ec072eff6931c8f3bfc38256"}, ] [package.dependencies] +importlib-metadata = {version = ">=4.6", markers = "python_version < \"3.10\""} Jinja2 = ">=2.11.1" Markdown = ">=3.3" MarkupSafe = ">=1.1" mkdocs = ">=1.2" mkdocs-autorefs = ">=0.3.1" pymdown-extensions = ">=6.3" +typing-extensions = {version = ">=4.1", markers = "python_version < \"3.10\""} [package.extras] crystal = ["mkdocstrings-crystal (>=0.3.4)"] @@ -1061,18 +1140,18 @@ python-legacy = ["mkdocstrings-python-legacy (>=0.2.1)"] [[package]] name = "mkdocstrings-python" -version = "0.7.1" +version = "1.5.2" description = "A Python handler for mkdocstrings." optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "mkdocstrings-python-0.7.1.tar.gz", hash = "sha256:c334b382dca202dfa37071c182418a6df5818356a95d54362a2b24822ca3af71"}, - {file = "mkdocstrings_python-0.7.1-py3-none-any.whl", hash = "sha256:a22060bfa374697678e9af4e62b020d990dad2711c98f7a9fac5c0345bef93c7"}, + {file = "mkdocstrings_python-1.5.2-py3-none-any.whl", hash = "sha256:ed37ca6d216986e2ac3530c19c3e7be381d1e3d09ea414e4ff467d6fd2cbd9c1"}, + {file = "mkdocstrings_python-1.5.2.tar.gz", hash = "sha256:81eb4a93bc454a253daf247d1a11397c435d641c64fa165324c17c06170b1dfb"}, ] [package.dependencies] -griffe = ">=0.11.1" -mkdocstrings = ">=0.19" +griffe = ">=0.35" +mkdocstrings = ">=0.20" [[package]] name = "mypy-extensions" @@ -1086,13 +1165,13 @@ files = [ [[package]] name = "napalm" -version = "4.0.0" +version = "4.1.0" description = "Network Automation and Programmability Abstraction Layer with Multivendor support" optional = false python-versions = "*" files = [ - {file = "napalm-4.0.0-py2.py3-none-any.whl", hash = "sha256:e4289f6966974b485c1f3de3e8f4a11ac2ed2825e975bbd5afc006331b1e4c36"}, - {file = "napalm-4.0.0.tar.gz", hash = "sha256:40e1bd297ac4102c14c0d427c51d61c3a12d5d5bec163750733941ad82a464ee"}, + {file = "napalm-4.1.0-py2.py3-none-any.whl", hash = "sha256:14a5b7759a0247a26fff2c444b1cfc150a08224de8addf4076c384845285bf5b"}, + {file = "napalm-4.1.0.tar.gz", hash = "sha256:3b3e18efd556861c056ba509eb46f5ffc9e3e6c42db399fa76b6ea9af272c17a"}, ] [package.dependencies] @@ -1103,7 +1182,7 @@ junos-eznc = ">=2.6.3" lxml = ">=4.3.0" ncclient = "*" netaddr = "*" -netmiko = ">=4.0.0" +netmiko = ">=4.1.0" netutils = ">=1.0.0" paramiko = ">=2.6.0" pyeapi = ">=0.8.2" @@ -1111,7 +1190,7 @@ pyYAML = "*" requests = ">=2.7.0" scp = "*" setuptools = ">=38.4.0" -textfsm = "<=1.1.2" +textfsm = "*" ttp = "*" ttp-templates = "*" typing-extensions = ">=4.3.0" @@ -1166,17 +1245,17 @@ textfsm = "1.1.2" [[package]] name = "netutils" -version = "1.4.1" +version = "1.6.0" description = "Common helper functions useful in network automation." optional = false -python-versions = ">=3.7,<4.0" +python-versions = ">=3.8,<4.0" files = [ - {file = "netutils-1.4.1-py3-none-any.whl", hash = "sha256:41002e42b205149fbe6739b7fdbc778ed843e87fabba9691d3d06a35f7876fd4"}, - {file = "netutils-1.4.1.tar.gz", hash = "sha256:4f7501478d810bcd3c64edfe064fa8962a1572636f4fceee2538fc9d3616fbe2"}, + {file = "netutils-1.6.0-py3-none-any.whl", hash = "sha256:e755e6141d0968f1deeb61693a4023f4f5fe1f0dde25d94ac1008f8191d8d237"}, + {file = "netutils-1.6.0.tar.gz", hash = "sha256:bd2fa691e172fe9d5c9e6fc5e2593316eb7fd2c36450454894ed13b274763d70"}, ] [package.extras] -optionals = ["napalm (>=4.0.0,<5.0.0)"] +optionals = ["jsonschema (>=4.17.3,<5.0.0)", "napalm (>=4.0.0,<5.0.0)"] [[package]] name = "nornir" @@ -1260,13 +1339,13 @@ nornir = ">=3,<4" [[package]] name = "ntc-templates" -version = "3.3.0" +version = "3.5.0" description = "TextFSM Templates for Network Devices, and Python wrapper for TextFSM's CliTable." optional = false python-versions = ">=3.7,<4.0" files = [ - {file = "ntc_templates-3.3.0-py3-none-any.whl", hash = "sha256:b0941824212ae50668cf9b882e848512d09487128a491909eab9afc30f808a85"}, - {file = "ntc_templates-3.3.0.tar.gz", hash = "sha256:a74014431715c2029a2d0f065bca447312d55171cea191db1189689ea076b82d"}, + {file = "ntc_templates-3.5.0-py3-none-any.whl", hash = "sha256:86d75c077eb1ceb97f4f8c69c9e3c7a32b08210ceb8228e5fa4e87e080746fd4"}, + {file = "ntc_templates-3.5.0.tar.gz", hash = "sha256:ee0dab4440dab1b3286549f8c08695b30037c1f36f55763c5a39005525f722c7"}, ] [package.dependencies] @@ -1283,15 +1362,25 @@ files = [ {file = "packaging-23.1.tar.gz", hash = "sha256:a392980d2b6cffa644431898be54b0045151319d1e7ec34f0cfed48767dd334f"}, ] +[[package]] +name = "paginate" +version = "0.5.6" +description = "Divides large result sets into pages for easier browsing" +optional = false +python-versions = "*" +files = [ + {file = "paginate-0.5.6.tar.gz", hash = "sha256:5e6007b6a9398177a7e1648d04fdd9f8c9766a1a945bceac82f1929e8c78af2d"}, +] + [[package]] name = "paramiko" -version = "3.1.0" +version = "3.3.1" description = "SSH2 protocol library" optional = false python-versions = ">=3.6" files = [ - {file = "paramiko-3.1.0-py3-none-any.whl", hash = "sha256:f0caa660e797d9cd10db6fc6ae81e2c9b2767af75c3180fcd0e46158cd368d7f"}, - {file = "paramiko-3.1.0.tar.gz", hash = "sha256:6950faca6819acd3219d4ae694a23c7a87ee38d084f70c1724b0c0dbb8b75769"}, + {file = "paramiko-3.3.1-py3-none-any.whl", hash = "sha256:b7bc5340a43de4287bbe22fe6de728aa2c22468b2a849615498dd944c2f275eb"}, + {file = "paramiko-3.3.1.tar.gz", hash = "sha256:6a3777a961ac86dbef375c5f5b8d50014a1a96d0fd7f054a43bc880134b0ff77"}, ] [package.dependencies] @@ -1306,13 +1395,13 @@ invoke = ["invoke (>=2.0)"] [[package]] name = "pathspec" -version = "0.11.1" +version = "0.11.2" description = "Utility library for gitignore style pattern matching of file paths." optional = false python-versions = ">=3.7" files = [ - {file = "pathspec-0.11.1-py3-none-any.whl", hash = "sha256:d8af70af76652554bd134c22b3e8a1cc46ed7d91edcdd721ef1a0c51a84a5293"}, - {file = "pathspec-0.11.1.tar.gz", hash = "sha256:2798de800fa92780e33acca925945e9a19a133b715067cf165b8866c15a31687"}, + {file = "pathspec-0.11.2-py3-none-any.whl", hash = "sha256:1d6ed233af05e679efb96b1851550ea95bbb64b7c490b0f5aa52996c11e92a20"}, + {file = "pathspec-0.11.2.tar.gz", hash = "sha256:e0d8d0ac2f12da61956eb2306b69f9469b42f4deb0f3cb6ed47b9cce9996ced3"}, ] [[package]] @@ -1328,49 +1417,43 @@ files = [ [[package]] name = "platformdirs" -version = "3.5.1" +version = "3.10.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.5.1-py3-none-any.whl", hash = "sha256:e2378146f1964972c03c085bb5662ae80b2b8c06226c54b2ff4aa9483e8a13a5"}, - {file = "platformdirs-3.5.1.tar.gz", hash = "sha256:412dae91f52a6f84830f39a8078cecd0e866cb72294a5c66808e74d5e88d251f"}, + {file = "platformdirs-3.10.0-py3-none-any.whl", hash = "sha256:d7c24979f292f916dc9cbf8648319032f551ea8c49a4c9bf2fb556a02070ec1d"}, + {file = "platformdirs-3.10.0.tar.gz", hash = "sha256:b45696dab2d7cc691a3226759c0d3b00c47c8b6e293d96f6436f733303f77f6d"}, ] -[package.dependencies] -typing-extensions = {version = ">=4.5", markers = "python_version < \"3.8\""} - [package.extras] -docs = ["furo (>=2023.3.27)", "proselint (>=0.13)", "sphinx (>=6.2.1)", "sphinx-autodoc-typehints (>=1.23,!=1.23.4)"] -test = ["appdirs (==1.4.4)", "covdefaults (>=2.3)", "pytest (>=7.3.1)", "pytest-cov (>=4)", "pytest-mock (>=3.10)"] +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 = "pluggy" -version = "1.0.0" +version = "1.3.0" description = "plugin and hook calling mechanisms for python" optional = false -python-versions = ">=3.6" +python-versions = ">=3.8" files = [ - {file = "pluggy-1.0.0-py2.py3-none-any.whl", hash = "sha256:74134bbf457f031a36d68416e1509f34bd5ccc019f0bcc952c7b909d06b37bd3"}, - {file = "pluggy-1.0.0.tar.gz", hash = "sha256:4224373bacce55f955a878bf9cfa763c1e360858e330072059e10bad68531159"}, + {file = "pluggy-1.3.0-py3-none-any.whl", hash = "sha256:d89c696a773f8bd377d18e5ecda92b7a3793cbe66c87060a6fb58c7b6e1061f7"}, + {file = "pluggy-1.3.0.tar.gz", hash = "sha256:cf61ae8f126ac6f7c451172cf30e3e43d3ca77615509771b3a984a0730651e12"}, ] -[package.dependencies] -importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""} - [package.extras] dev = ["pre-commit", "tox"] testing = ["pytest", "pytest-benchmark"] [[package]] name = "pycodestyle" -version = "2.7.0" +version = "2.9.1" description = "Python style guide checker" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +python-versions = ">=3.6" files = [ - {file = "pycodestyle-2.7.0-py2.py3-none-any.whl", hash = "sha256:514f76d918fcc0b55c6680472f0a37970994e07bbb80725808c17089be302068"}, - {file = "pycodestyle-2.7.0.tar.gz", hash = "sha256:c389c1d06bf7904078ca03399a4816f974a1d590090fecea0c63ec26ebaf1cef"}, + {file = "pycodestyle-2.9.1-py2.py3-none-any.whl", hash = "sha256:d1735fc58b418fd7c5f658d28d943854f8a849b01a5d0a1e6f3f3fdd0166804b"}, + {file = "pycodestyle-2.9.1.tar.gz", hash = "sha256:2c9607871d58c76354b697b42f5d57e1ada7d261c261efac224b664affdc5785"}, ] [[package]] @@ -1386,47 +1469,47 @@ files = [ [[package]] name = "pydantic" -version = "1.10.7" +version = "1.10.12" description = "Data validation and settings management using python type hints" optional = false python-versions = ">=3.7" files = [ - {file = "pydantic-1.10.7-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:e79e999e539872e903767c417c897e729e015872040e56b96e67968c3b918b2d"}, - {file = "pydantic-1.10.7-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:01aea3a42c13f2602b7ecbbea484a98169fb568ebd9e247593ea05f01b884b2e"}, - {file = "pydantic-1.10.7-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:516f1ed9bc2406a0467dd777afc636c7091d71f214d5e413d64fef45174cfc7a"}, - {file = "pydantic-1.10.7-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ae150a63564929c675d7f2303008d88426a0add46efd76c3fc797cd71cb1b46f"}, - {file = "pydantic-1.10.7-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:ecbbc51391248116c0a055899e6c3e7ffbb11fb5e2a4cd6f2d0b93272118a209"}, - {file = "pydantic-1.10.7-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:f4a2b50e2b03d5776e7f21af73e2070e1b5c0d0df255a827e7c632962f8315af"}, - {file = "pydantic-1.10.7-cp310-cp310-win_amd64.whl", hash = "sha256:a7cd2251439988b413cb0a985c4ed82b6c6aac382dbaff53ae03c4b23a70e80a"}, - {file = "pydantic-1.10.7-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:68792151e174a4aa9e9fc1b4e653e65a354a2fa0fed169f7b3d09902ad2cb6f1"}, - {file = "pydantic-1.10.7-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:dfe2507b8ef209da71b6fb5f4e597b50c5a34b78d7e857c4f8f3115effaef5fe"}, - {file = "pydantic-1.10.7-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:10a86d8c8db68086f1e30a530f7d5f83eb0685e632e411dbbcf2d5c0150e8dcd"}, - {file = "pydantic-1.10.7-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d75ae19d2a3dbb146b6f324031c24f8a3f52ff5d6a9f22f0683694b3afcb16fb"}, - {file = "pydantic-1.10.7-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:464855a7ff7f2cc2cf537ecc421291b9132aa9c79aef44e917ad711b4a93163b"}, - {file = "pydantic-1.10.7-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:193924c563fae6ddcb71d3f06fa153866423ac1b793a47936656e806b64e24ca"}, - {file = "pydantic-1.10.7-cp311-cp311-win_amd64.whl", hash = "sha256:b4a849d10f211389502059c33332e91327bc154acc1845f375a99eca3afa802d"}, - {file = "pydantic-1.10.7-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:cc1dde4e50a5fc1336ee0581c1612215bc64ed6d28d2c7c6f25d2fe3e7c3e918"}, - {file = "pydantic-1.10.7-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e0cfe895a504c060e5d36b287ee696e2fdad02d89e0d895f83037245218a87fe"}, - {file = "pydantic-1.10.7-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:670bb4683ad1e48b0ecb06f0cfe2178dcf74ff27921cdf1606e527d2617a81ee"}, - {file = "pydantic-1.10.7-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:950ce33857841f9a337ce07ddf46bc84e1c4946d2a3bba18f8280297157a3fd1"}, - {file = "pydantic-1.10.7-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:c15582f9055fbc1bfe50266a19771bbbef33dd28c45e78afbe1996fd70966c2a"}, - {file = "pydantic-1.10.7-cp37-cp37m-win_amd64.whl", hash = "sha256:82dffb306dd20bd5268fd6379bc4bfe75242a9c2b79fec58e1041fbbdb1f7914"}, - {file = "pydantic-1.10.7-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:8c7f51861d73e8b9ddcb9916ae7ac39fb52761d9ea0df41128e81e2ba42886cd"}, - {file = "pydantic-1.10.7-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:6434b49c0b03a51021ade5c4daa7d70c98f7a79e95b551201fff682fc1661245"}, - {file = "pydantic-1.10.7-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:64d34ab766fa056df49013bb6e79921a0265204c071984e75a09cbceacbbdd5d"}, - {file = "pydantic-1.10.7-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:701daea9ffe9d26f97b52f1d157e0d4121644f0fcf80b443248434958fd03dc3"}, - {file = "pydantic-1.10.7-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:cf135c46099ff3f919d2150a948ce94b9ce545598ef2c6c7bf55dca98a304b52"}, - {file = "pydantic-1.10.7-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:b0f85904f73161817b80781cc150f8b906d521fa11e3cdabae19a581c3606209"}, - {file = "pydantic-1.10.7-cp38-cp38-win_amd64.whl", hash = "sha256:9f6f0fd68d73257ad6685419478c5aece46432f4bdd8d32c7345f1986496171e"}, - {file = "pydantic-1.10.7-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:c230c0d8a322276d6e7b88c3f7ce885f9ed16e0910354510e0bae84d54991143"}, - {file = "pydantic-1.10.7-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:976cae77ba6a49d80f461fd8bba183ff7ba79f44aa5cfa82f1346b5626542f8e"}, - {file = "pydantic-1.10.7-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7d45fc99d64af9aaf7e308054a0067fdcd87ffe974f2442312372dfa66e1001d"}, - {file = "pydantic-1.10.7-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d2a5ebb48958754d386195fe9e9c5106f11275867051bf017a8059410e9abf1f"}, - {file = "pydantic-1.10.7-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:abfb7d4a7cd5cc4e1d1887c43503a7c5dd608eadf8bc615413fc498d3e4645cd"}, - {file = "pydantic-1.10.7-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:80b1fab4deb08a8292d15e43a6edccdffa5377a36a4597bb545b93e79c5ff0a5"}, - {file = "pydantic-1.10.7-cp39-cp39-win_amd64.whl", hash = "sha256:d71e69699498b020ea198468e2480a2f1e7433e32a3a99760058c6520e2bea7e"}, - {file = "pydantic-1.10.7-py3-none-any.whl", hash = "sha256:0cd181f1d0b1d00e2b705f1bf1ac7799a2d938cce3376b8007df62b29be3c2c6"}, - {file = "pydantic-1.10.7.tar.gz", hash = "sha256:cfc83c0678b6ba51b0532bea66860617c4cd4251ecf76e9846fa5a9f3454e97e"}, + {file = "pydantic-1.10.12-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:a1fcb59f2f355ec350073af41d927bf83a63b50e640f4dbaa01053a28b7a7718"}, + {file = "pydantic-1.10.12-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:b7ccf02d7eb340b216ec33e53a3a629856afe1c6e0ef91d84a4e6f2fb2ca70fe"}, + {file = "pydantic-1.10.12-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8fb2aa3ab3728d950bcc885a2e9eff6c8fc40bc0b7bb434e555c215491bcf48b"}, + {file = "pydantic-1.10.12-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:771735dc43cf8383959dc9b90aa281f0b6092321ca98677c5fb6125a6f56d58d"}, + {file = "pydantic-1.10.12-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:ca48477862372ac3770969b9d75f1bf66131d386dba79506c46d75e6b48c1e09"}, + {file = "pydantic-1.10.12-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:a5e7add47a5b5a40c49b3036d464e3c7802f8ae0d1e66035ea16aa5b7a3923ed"}, + {file = "pydantic-1.10.12-cp310-cp310-win_amd64.whl", hash = "sha256:e4129b528c6baa99a429f97ce733fff478ec955513630e61b49804b6cf9b224a"}, + {file = "pydantic-1.10.12-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b0d191db0f92dfcb1dec210ca244fdae5cbe918c6050b342d619c09d31eea0cc"}, + {file = "pydantic-1.10.12-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:795e34e6cc065f8f498c89b894a3c6da294a936ee71e644e4bd44de048af1405"}, + {file = "pydantic-1.10.12-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:69328e15cfda2c392da4e713443c7dbffa1505bc9d566e71e55abe14c97ddc62"}, + {file = "pydantic-1.10.12-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2031de0967c279df0d8a1c72b4ffc411ecd06bac607a212892757db7462fc494"}, + {file = "pydantic-1.10.12-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:ba5b2e6fe6ca2b7e013398bc7d7b170e21cce322d266ffcd57cca313e54fb246"}, + {file = "pydantic-1.10.12-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:2a7bac939fa326db1ab741c9d7f44c565a1d1e80908b3797f7f81a4f86bc8d33"}, + {file = "pydantic-1.10.12-cp311-cp311-win_amd64.whl", hash = "sha256:87afda5539d5140cb8ba9e8b8c8865cb5b1463924d38490d73d3ccfd80896b3f"}, + {file = "pydantic-1.10.12-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:549a8e3d81df0a85226963611950b12d2d334f214436a19537b2efed61b7639a"}, + {file = "pydantic-1.10.12-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:598da88dfa127b666852bef6d0d796573a8cf5009ffd62104094a4fe39599565"}, + {file = "pydantic-1.10.12-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ba5c4a8552bff16c61882db58544116d021d0b31ee7c66958d14cf386a5b5350"}, + {file = "pydantic-1.10.12-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:c79e6a11a07da7374f46970410b41d5e266f7f38f6a17a9c4823db80dadf4303"}, + {file = "pydantic-1.10.12-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:ab26038b8375581dc832a63c948f261ae0aa21f1d34c1293469f135fa92972a5"}, + {file = "pydantic-1.10.12-cp37-cp37m-win_amd64.whl", hash = "sha256:e0a16d274b588767602b7646fa05af2782576a6cf1022f4ba74cbb4db66f6ca8"}, + {file = "pydantic-1.10.12-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6a9dfa722316f4acf4460afdf5d41d5246a80e249c7ff475c43a3a1e9d75cf62"}, + {file = "pydantic-1.10.12-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:a73f489aebd0c2121ed974054cb2759af8a9f747de120acd2c3394cf84176ccb"}, + {file = "pydantic-1.10.12-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6b30bcb8cbfccfcf02acb8f1a261143fab622831d9c0989707e0e659f77a18e0"}, + {file = "pydantic-1.10.12-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2fcfb5296d7877af406ba1547dfde9943b1256d8928732267e2653c26938cd9c"}, + {file = "pydantic-1.10.12-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:2f9a6fab5f82ada41d56b0602606a5506aab165ca54e52bc4545028382ef1c5d"}, + {file = "pydantic-1.10.12-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:dea7adcc33d5d105896401a1f37d56b47d443a2b2605ff8a969a0ed5543f7e33"}, + {file = "pydantic-1.10.12-cp38-cp38-win_amd64.whl", hash = "sha256:1eb2085c13bce1612da8537b2d90f549c8cbb05c67e8f22854e201bde5d98a47"}, + {file = "pydantic-1.10.12-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:ef6c96b2baa2100ec91a4b428f80d8f28a3c9e53568219b6c298c1125572ebc6"}, + {file = "pydantic-1.10.12-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:6c076be61cd0177a8433c0adcb03475baf4ee91edf5a4e550161ad57fc90f523"}, + {file = "pydantic-1.10.12-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2d5a58feb9a39f481eda4d5ca220aa8b9d4f21a41274760b9bc66bfd72595b86"}, + {file = "pydantic-1.10.12-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e5f805d2d5d0a41633651a73fa4ecdd0b3d7a49de4ec3fadf062fe16501ddbf1"}, + {file = "pydantic-1.10.12-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:1289c180abd4bd4555bb927c42ee42abc3aee02b0fb2d1223fb7c6e5bef87dbe"}, + {file = "pydantic-1.10.12-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:5d1197e462e0364906cbc19681605cb7c036f2475c899b6f296104ad42b9f5fb"}, + {file = "pydantic-1.10.12-cp39-cp39-win_amd64.whl", hash = "sha256:fdbdd1d630195689f325c9ef1a12900524dceb503b00a987663ff4f58669b93d"}, + {file = "pydantic-1.10.12-py3-none-any.whl", hash = "sha256:b749a43aa51e32839c9d71dc67eb1e4221bb04af1033a32e3923d46f9effa942"}, + {file = "pydantic-1.10.12.tar.gz", hash = "sha256:0fe8a415cea8f340e7a9af9c54fc71a649b43e8ca3cc732986116b3cb135d303"}, ] [package.dependencies] @@ -1449,7 +1532,6 @@ files = [ ] [package.dependencies] -importlib-metadata = {version = ">=2.0.0,<5.0.0", markers = "python_version < \"3.8\""} snowballstemmer = ">=2.2.0" [package.extras] @@ -1457,12 +1539,12 @@ toml = ["tomli (>=1.2.3)"] [[package]] name = "pyeapi" -version = "0.8.4" +version = "1.0.2" description = "Python Client for eAPI" optional = false python-versions = "*" files = [ - {file = "pyeapi-0.8.4.tar.gz", hash = "sha256:c33ad1eadd8ebac75f63488df9412081ce0b024c9e1da12a37196a5c60427c54"}, + {file = "pyeapi-1.0.2.tar.gz", hash = "sha256:563a80bb19451df7dd7b6e9e38489dee67ebeaf2f54de296e8ae0b26cd68a297"}, ] [package.dependencies] @@ -1474,24 +1556,24 @@ test = ["coverage", "mock"] [[package]] name = "pyflakes" -version = "2.3.1" +version = "2.5.0" description = "passive checker of Python programs" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +python-versions = ">=3.6" files = [ - {file = "pyflakes-2.3.1-py2.py3-none-any.whl", hash = "sha256:7893783d01b8a89811dd72d7dfd4d84ff098e5eed95cfa8905b22bbffe52efc3"}, - {file = "pyflakes-2.3.1.tar.gz", hash = "sha256:f5bc8ecabc05bb9d291eb5203d6810b49040f6ff446a756326104746cc00c1db"}, + {file = "pyflakes-2.5.0-py2.py3-none-any.whl", hash = "sha256:4579f67d887f804e67edb544428f264b7b24f435b263c4614f384135cea553d2"}, + {file = "pyflakes-2.5.0.tar.gz", hash = "sha256:491feb020dca48ccc562a8c0cbe8df07ee13078df59813b83959cbdada312ea3"}, ] [[package]] name = "pygments" -version = "2.15.1" +version = "2.16.1" description = "Pygments is a syntax highlighting package written in Python." optional = false python-versions = ">=3.7" files = [ - {file = "Pygments-2.15.1-py3-none-any.whl", hash = "sha256:db2db3deb4b4179f399a09054b023b6a586b76499d36965813c71aa8ed7b5fd1"}, - {file = "Pygments-2.15.1.tar.gz", hash = "sha256:8ace4d3c1dd481894b2005f560ead0f9f19ee64fe983366be1a21e171d12775c"}, + {file = "Pygments-2.16.1-py3-none-any.whl", hash = "sha256:13fc09fa63bc8d8671a6d247e1eb303c4b343eaee81d861f3404db2935653692"}, + {file = "Pygments-2.16.1.tar.gz", hash = "sha256:1daff0494820c69bc8941e407aa20f577374ee88364ee10a98fdbe0aece96e29"}, ] [package.extras] @@ -1499,43 +1581,51 @@ plugins = ["importlib-metadata"] [[package]] name = "pylint" -version = "2.13.9" +version = "2.17.5" description = "python code static checker" optional = false -python-versions = ">=3.6.2" +python-versions = ">=3.7.2" files = [ - {file = "pylint-2.13.9-py3-none-any.whl", hash = "sha256:705c620d388035bdd9ff8b44c5bcdd235bfb49d276d488dd2c8ff1736aa42526"}, - {file = "pylint-2.13.9.tar.gz", hash = "sha256:095567c96e19e6f57b5b907e67d265ff535e588fe26b12b5ebe1fc5645b2c731"}, + {file = "pylint-2.17.5-py3-none-any.whl", hash = "sha256:73995fb8216d3bed149c8d51bba25b2c52a8251a2c8ac846ec668ce38fab5413"}, + {file = "pylint-2.17.5.tar.gz", hash = "sha256:f7b601cbc06fef7e62a754e2b41294c2aa31f1cb659624b9a85bcba29eaf8252"}, ] [package.dependencies] -astroid = ">=2.11.5,<=2.12.0-dev0" -colorama = {version = "*", markers = "sys_platform == \"win32\""} -dill = ">=0.2" +astroid = ">=2.15.6,<=2.17.0-dev0" +colorama = {version = ">=0.4.5", markers = "sys_platform == \"win32\""} +dill = [ + {version = ">=0.2", markers = "python_version < \"3.11\""}, + {version = ">=0.3.6", markers = "python_version >= \"3.11\""}, +] isort = ">=4.2.5,<6" mccabe = ">=0.6,<0.8" platformdirs = ">=2.2.0" tomli = {version = ">=1.1.0", markers = "python_version < \"3.11\""} +tomlkit = ">=0.10.1" typing-extensions = {version = ">=3.10.0", markers = "python_version < \"3.10\""} [package.extras] -testutil = ["gitpython (>3)"] +spelling = ["pyenchant (>=3.2,<4.0)"] +testutils = ["gitpython (>3)"] [[package]] name = "pymdown-extensions" -version = "10.0.1" +version = "10.3" description = "Extension pack for Python Markdown." optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "pymdown_extensions-10.0.1-py3-none-any.whl", hash = "sha256:ae66d84013c5d027ce055693e09a4628b67e9dec5bce05727e45b0918e36f274"}, - {file = "pymdown_extensions-10.0.1.tar.gz", hash = "sha256:b44e1093a43b8a975eae17b03c3a77aad4681b3b56fce60ce746dbef1944c8cb"}, + {file = "pymdown_extensions-10.3-py3-none-any.whl", hash = "sha256:77a82c621c58a83efc49a389159181d570e370fff9f810d3a4766a75fc678b66"}, + {file = "pymdown_extensions-10.3.tar.gz", hash = "sha256:94a0d8a03246712b64698af223848fd80aaf1ae4c4be29c8c61939b0467b5722"}, ] [package.dependencies] markdown = ">=3.2" pyyaml = "*" +[package.extras] +extra = ["pygments (>=2.12)"] + [[package]] name = "pynacl" version = "1.5.0" @@ -1564,32 +1654,51 @@ tests = ["hypothesis (>=3.27.0)", "pytest (>=3.2.1,!=3.3.0)"] [[package]] name = "pynautobot" -version = "1.4.0" +version = "1.5.0" description = "Nautobot API client library" optional = false python-versions = ">=3.7,<4.0" files = [ - {file = "pynautobot-1.4.0-py3-none-any.whl", hash = "sha256:6bc053b095728ed0af40d097a7513c3e16c51ec63aad46f691f50b3f6c82bdfe"}, - {file = "pynautobot-1.4.0.tar.gz", hash = "sha256:87c93976248f99f2adc0e22d7a39e7f0aac3460451607078bfee93742742c9d4"}, + {file = "pynautobot-1.5.0-py3-none-any.whl", hash = "sha256:aa5bdf18148d82715b26e1a7abf0796bb28da05fece3d206b6f42749d2f466b1"}, + {file = "pynautobot-1.5.0.tar.gz", hash = "sha256:50ac1e12f377ce2f1d156056e9ec3333c8a74bf6269e145889606da92b8752b4"}, ] [package.dependencies] -requests = ">=2.20.0,<3.0.0" +requests = ">=2.30.0,<3.0.0" +urllib3 = ">=1.21.1,<1.27" [[package]] name = "pyparsing" -version = "3.0.9" +version = "3.1.1" description = "pyparsing module - Classes and methods to define and execute parsing grammars" optional = false python-versions = ">=3.6.8" files = [ - {file = "pyparsing-3.0.9-py3-none-any.whl", hash = "sha256:5026bae9a10eeaefb61dab2f09052b9f4307d44aee4eda64b309723d8d206bbc"}, - {file = "pyparsing-3.0.9.tar.gz", hash = "sha256:2b020ecf7d21b687f219b71ecad3631f644a47f01403fa1d1036b0c6416d70fb"}, + {file = "pyparsing-3.1.1-py3-none-any.whl", hash = "sha256:32c7c0b711493c72ff18a981d24f28aaf9c1fb7ed5e9667c9e84e3db623bdbfb"}, + {file = "pyparsing-3.1.1.tar.gz", hash = "sha256:ede28a1a32462f5a9705e07aea48001a08f7cf81a021585011deba701581a0db"}, ] [package.extras] diagrams = ["jinja2", "railroad-diagrams"] +[[package]] +name = "pyquery" +version = "2.0.0" +description = "A jquery-like library for python" +optional = false +python-versions = "*" +files = [ + {file = "pyquery-2.0.0-py3-none-any.whl", hash = "sha256:8dfc9b4b7c5f877d619bbae74b1898d5743f6ca248cfd5d72b504dd614da312f"}, + {file = "pyquery-2.0.0.tar.gz", hash = "sha256:963e8d4e90262ff6d8dec072ea97285dc374a2f69cad7776f4082abcf6a1d8ae"}, +] + +[package.dependencies] +cssselect = ">=1.2.0" +lxml = ">=2.1" + +[package.extras] +test = ["pytest", "pytest-cov", "requests", "webob", "webtest"] + [[package]] name = "pyserial" version = "3.5" @@ -1606,26 +1715,25 @@ cp2110 = ["hidapi"] [[package]] name = "pytest" -version = "7.3.1" +version = "7.4.2" description = "pytest: simple powerful testing with Python" optional = false python-versions = ">=3.7" files = [ - {file = "pytest-7.3.1-py3-none-any.whl", hash = "sha256:3799fa815351fea3a5e96ac7e503a96fa51cc9942c3753cda7651b93c1cfa362"}, - {file = "pytest-7.3.1.tar.gz", hash = "sha256:434afafd78b1d78ed0addf160ad2b77a30d35d4bdf8af234fe621919d9ed15e3"}, + {file = "pytest-7.4.2-py3-none-any.whl", hash = "sha256:1d881c6124e08ff0a1bb75ba3ec0bfd8b5354a01c194ddd5a0a870a48d99b002"}, + {file = "pytest-7.4.2.tar.gz", hash = "sha256:a766259cfab564a2ad52cb1aae1b881a75c3eb7e34ca3779697c23ed47c47069"}, ] [package.dependencies] colorama = {version = "*", markers = "sys_platform == \"win32\""} exceptiongroup = {version = ">=1.0.0rc8", markers = "python_version < \"3.11\""} -importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""} iniconfig = "*" packaging = "*" pluggy = ">=0.12,<2.0" tomli = {version = ">=1.0.0", markers = "python_version < \"3.11\""} [package.extras] -testing = ["argcomplete", "attrs (>=19.2.0)", "hypothesis (>=3.56)", "mock", "nose", "pygments (>=2.7.2)", "requests", "xmlschema"] +testing = ["argcomplete", "attrs (>=19.2.0)", "hypothesis (>=3.56)", "mock", "nose", "pygments (>=2.7.2)", "requests", "setuptools", "xmlschema"] [[package]] name = "python-dateutil" @@ -1643,65 +1751,86 @@ six = ">=1.5" [[package]] name = "python-dotenv" -version = "0.21.1" +version = "1.0.0" description = "Read key-value pairs from a .env file and set them as environment variables" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "python-dotenv-0.21.1.tar.gz", hash = "sha256:1c93de8f636cde3ce377292818d0e440b6e45a82f215c3744979151fa8151c49"}, - {file = "python_dotenv-0.21.1-py3-none-any.whl", hash = "sha256:41e12e0318bebc859fcc4d97d4db8d20ad21721a6aa5047dd59f090391cb549a"}, + {file = "python-dotenv-1.0.0.tar.gz", hash = "sha256:a8df96034aae6d2d50a4ebe8216326c61c3eb64836776504fcca410e5937a3ba"}, + {file = "python_dotenv-1.0.0-py3-none-any.whl", hash = "sha256:f5971a9226b701070a4bf2c38c89e5a3f0d64de8debda981d1db98583009122a"}, ] [package.extras] cli = ["click (>=5.0)"] +[[package]] +name = "pytz" +version = "2023.3.post1" +description = "World timezone definitions, modern and historical" +optional = false +python-versions = "*" +files = [ + {file = "pytz-2023.3.post1-py2.py3-none-any.whl", hash = "sha256:ce42d816b81b68506614c11e8937d3aa9e41007ceb50bfdcb0749b921bf646c7"}, + {file = "pytz-2023.3.post1.tar.gz", hash = "sha256:7b4fddbeb94a1eba4b557da24f19fdf9db575192544270a9101d8509f9f43d7b"}, +] + [[package]] name = "pyyaml" -version = "6.0" +version = "6.0.1" description = "YAML parser and emitter for Python" optional = false python-versions = ">=3.6" files = [ - {file = "PyYAML-6.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d4db7c7aef085872ef65a8fd7d6d09a14ae91f691dec3e87ee5ee0539d516f53"}, - {file = "PyYAML-6.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9df7ed3b3d2e0ecfe09e14741b857df43adb5a3ddadc919a2d94fbdf78fea53c"}, - {file = "PyYAML-6.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:77f396e6ef4c73fdc33a9157446466f1cff553d979bd00ecb64385760c6babdc"}, - {file = "PyYAML-6.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a80a78046a72361de73f8f395f1f1e49f956c6be882eed58505a15f3e430962b"}, - {file = "PyYAML-6.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f84fbc98b019fef2ee9a1cb3ce93e3187a6df0b2538a651bfb890254ba9f90b5"}, - {file = "PyYAML-6.0-cp310-cp310-win32.whl", hash = "sha256:2cd5df3de48857ed0544b34e2d40e9fac445930039f3cfe4bcc592a1f836d513"}, - {file = "PyYAML-6.0-cp310-cp310-win_amd64.whl", hash = "sha256:daf496c58a8c52083df09b80c860005194014c3698698d1a57cbcfa182142a3a"}, - {file = "PyYAML-6.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:d4b0ba9512519522b118090257be113b9468d804b19d63c71dbcf4a48fa32358"}, - {file = "PyYAML-6.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:81957921f441d50af23654aa6c5e5eaf9b06aba7f0a19c18a538dc7ef291c5a1"}, - {file = "PyYAML-6.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:afa17f5bc4d1b10afd4466fd3a44dc0e245382deca5b3c353d8b757f9e3ecb8d"}, - {file = "PyYAML-6.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:dbad0e9d368bb989f4515da330b88a057617d16b6a8245084f1b05400f24609f"}, - {file = "PyYAML-6.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:432557aa2c09802be39460360ddffd48156e30721f5e8d917f01d31694216782"}, - {file = "PyYAML-6.0-cp311-cp311-win32.whl", hash = "sha256:bfaef573a63ba8923503d27530362590ff4f576c626d86a9fed95822a8255fd7"}, - {file = "PyYAML-6.0-cp311-cp311-win_amd64.whl", hash = "sha256:01b45c0191e6d66c470b6cf1b9531a771a83c1c4208272ead47a3ae4f2f603bf"}, - {file = "PyYAML-6.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:897b80890765f037df3403d22bab41627ca8811ae55e9a722fd0392850ec4d86"}, - {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:50602afada6d6cbfad699b0c7bb50d5ccffa7e46a3d738092afddc1f9758427f"}, - {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:48c346915c114f5fdb3ead70312bd042a953a8ce5c7106d5bfb1a5254e47da92"}, - {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:98c4d36e99714e55cfbaaee6dd5badbc9a1ec339ebfc3b1f52e293aee6bb71a4"}, - {file = "PyYAML-6.0-cp36-cp36m-win32.whl", hash = "sha256:0283c35a6a9fbf047493e3a0ce8d79ef5030852c51e9d911a27badfde0605293"}, - {file = "PyYAML-6.0-cp36-cp36m-win_amd64.whl", hash = "sha256:07751360502caac1c067a8132d150cf3d61339af5691fe9e87803040dbc5db57"}, - {file = "PyYAML-6.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:819b3830a1543db06c4d4b865e70ded25be52a2e0631ccd2f6a47a2822f2fd7c"}, - {file = "PyYAML-6.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:473f9edb243cb1935ab5a084eb238d842fb8f404ed2193a915d1784b5a6b5fc0"}, - {file = "PyYAML-6.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0ce82d761c532fe4ec3f87fc45688bdd3a4c1dc5e0b4a19814b9009a29baefd4"}, - {file = "PyYAML-6.0-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:231710d57adfd809ef5d34183b8ed1eeae3f76459c18fb4a0b373ad56bedcdd9"}, - {file = "PyYAML-6.0-cp37-cp37m-win32.whl", hash = "sha256:c5687b8d43cf58545ade1fe3e055f70eac7a5a1a0bf42824308d868289a95737"}, - {file = "PyYAML-6.0-cp37-cp37m-win_amd64.whl", hash = "sha256:d15a181d1ecd0d4270dc32edb46f7cb7733c7c508857278d3d378d14d606db2d"}, - {file = "PyYAML-6.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:0b4624f379dab24d3725ffde76559cff63d9ec94e1736b556dacdfebe5ab6d4b"}, - {file = "PyYAML-6.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:213c60cd50106436cc818accf5baa1aba61c0189ff610f64f4a3e8c6726218ba"}, - {file = "PyYAML-6.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9fa600030013c4de8165339db93d182b9431076eb98eb40ee068700c9c813e34"}, - {file = "PyYAML-6.0-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:277a0ef2981ca40581a47093e9e2d13b3f1fbbeffae064c1d21bfceba2030287"}, - {file = "PyYAML-6.0-cp38-cp38-win32.whl", hash = "sha256:d4eccecf9adf6fbcc6861a38015c2a64f38b9d94838ac1810a9023a0609e1b78"}, - {file = "PyYAML-6.0-cp38-cp38-win_amd64.whl", hash = "sha256:1e4747bc279b4f613a09eb64bba2ba602d8a6664c6ce6396a4d0cd413a50ce07"}, - {file = "PyYAML-6.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:055d937d65826939cb044fc8c9b08889e8c743fdc6a32b33e2390f66013e449b"}, - {file = "PyYAML-6.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:e61ceaab6f49fb8bdfaa0f92c4b57bcfbea54c09277b1b4f7ac376bfb7a7c174"}, - {file = "PyYAML-6.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d67d839ede4ed1b28a4e8909735fc992a923cdb84e618544973d7dfc71540803"}, - {file = "PyYAML-6.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cba8c411ef271aa037d7357a2bc8f9ee8b58b9965831d9e51baf703280dc73d3"}, - {file = "PyYAML-6.0-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:40527857252b61eacd1d9af500c3337ba8deb8fc298940291486c465c8b46ec0"}, - {file = "PyYAML-6.0-cp39-cp39-win32.whl", hash = "sha256:b5b9eccad747aabaaffbc6064800670f0c297e52c12754eb1d976c57e4f74dcb"}, - {file = "PyYAML-6.0-cp39-cp39-win_amd64.whl", hash = "sha256:b3d267842bf12586ba6c734f89d1f5b871df0273157918b0ccefa29deb05c21c"}, - {file = "PyYAML-6.0.tar.gz", hash = "sha256:68fb519c14306fec9720a2a5b45bc9f0c8d1b9c72adf45c37baedfcd949c35a2"}, + {file = "PyYAML-6.0.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d858aa552c999bc8a8d57426ed01e40bef403cd8ccdd0fc5f6f04a00414cac2a"}, + {file = "PyYAML-6.0.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:fd66fc5d0da6d9815ba2cebeb4205f95818ff4b79c3ebe268e75d961704af52f"}, + {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:69b023b2b4daa7548bcfbd4aa3da05b3a74b772db9e23b982788168117739938"}, + {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:81e0b275a9ecc9c0c0c07b4b90ba548307583c125f54d5b6946cfee6360c733d"}, + {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba336e390cd8e4d1739f42dfe9bb83a3cc2e80f567d8805e11b46f4a943f5515"}, + {file = "PyYAML-6.0.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:326c013efe8048858a6d312ddd31d56e468118ad4cdeda36c719bf5bb6192290"}, + {file = "PyYAML-6.0.1-cp310-cp310-win32.whl", hash = "sha256:bd4af7373a854424dabd882decdc5579653d7868b8fb26dc7d0e99f823aa5924"}, + {file = "PyYAML-6.0.1-cp310-cp310-win_amd64.whl", hash = "sha256:fd1592b3fdf65fff2ad0004b5e363300ef59ced41c2e6b3a99d4089fa8c5435d"}, + {file = "PyYAML-6.0.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6965a7bc3cf88e5a1c3bd2e0b5c22f8d677dc88a455344035f03399034eb3007"}, + {file = "PyYAML-6.0.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:f003ed9ad21d6a4713f0a9b5a7a0a79e08dd0f221aff4525a2be4c346ee60aab"}, + {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:42f8152b8dbc4fe7d96729ec2b99c7097d656dc1213a3229ca5383f973a5ed6d"}, + {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:062582fca9fabdd2c8b54a3ef1c978d786e0f6b3a1510e0ac93ef59e0ddae2bc"}, + {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d2b04aac4d386b172d5b9692e2d2da8de7bfb6c387fa4f801fbf6fb2e6ba4673"}, + {file = "PyYAML-6.0.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:e7d73685e87afe9f3b36c799222440d6cf362062f78be1013661b00c5c6f678b"}, + {file = "PyYAML-6.0.1-cp311-cp311-win32.whl", hash = "sha256:1635fd110e8d85d55237ab316b5b011de701ea0f29d07611174a1b42f1444741"}, + {file = "PyYAML-6.0.1-cp311-cp311-win_amd64.whl", hash = "sha256:bf07ee2fef7014951eeb99f56f39c9bb4af143d8aa3c21b1677805985307da34"}, + {file = "PyYAML-6.0.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:855fb52b0dc35af121542a76b9a84f8d1cd886ea97c84703eaa6d88e37a2ad28"}, + {file = "PyYAML-6.0.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:40df9b996c2b73138957fe23a16a4f0ba614f4c0efce1e9406a184b6d07fa3a9"}, + {file = "PyYAML-6.0.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c22bec3fbe2524cde73d7ada88f6566758a8f7227bfbf93a408a9d86bcc12a0"}, + {file = "PyYAML-6.0.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8d4e9c88387b0f5c7d5f281e55304de64cf7f9c0021a3525bd3b1c542da3b0e4"}, + {file = "PyYAML-6.0.1-cp312-cp312-win32.whl", hash = "sha256:d483d2cdf104e7c9fa60c544d92981f12ad66a457afae824d146093b8c294c54"}, + {file = "PyYAML-6.0.1-cp312-cp312-win_amd64.whl", hash = "sha256:0d3304d8c0adc42be59c5f8a4d9e3d7379e6955ad754aa9d6ab7a398b59dd1df"}, + {file = "PyYAML-6.0.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:50550eb667afee136e9a77d6dc71ae76a44df8b3e51e41b77f6de2932bfe0f47"}, + {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1fe35611261b29bd1de0070f0b2f47cb6ff71fa6595c077e42bd0c419fa27b98"}, + {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:704219a11b772aea0d8ecd7058d0082713c3562b4e271b849ad7dc4a5c90c13c"}, + {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:afd7e57eddb1a54f0f1a974bc4391af8bcce0b444685d936840f125cf046d5bd"}, + {file = "PyYAML-6.0.1-cp36-cp36m-win32.whl", hash = "sha256:fca0e3a251908a499833aa292323f32437106001d436eca0e6e7833256674585"}, + {file = "PyYAML-6.0.1-cp36-cp36m-win_amd64.whl", hash = "sha256:f22ac1c3cac4dbc50079e965eba2c1058622631e526bd9afd45fedd49ba781fa"}, + {file = "PyYAML-6.0.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:b1275ad35a5d18c62a7220633c913e1b42d44b46ee12554e5fd39c70a243d6a3"}, + {file = "PyYAML-6.0.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:18aeb1bf9a78867dc38b259769503436b7c72f7a1f1f4c93ff9a17de54319b27"}, + {file = "PyYAML-6.0.1-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:596106435fa6ad000c2991a98fa58eeb8656ef2325d7e158344fb33864ed87e3"}, + {file = "PyYAML-6.0.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:baa90d3f661d43131ca170712d903e6295d1f7a0f595074f151c0aed377c9b9c"}, + {file = "PyYAML-6.0.1-cp37-cp37m-win32.whl", hash = "sha256:9046c58c4395dff28dd494285c82ba00b546adfc7ef001486fbf0324bc174fba"}, + {file = "PyYAML-6.0.1-cp37-cp37m-win_amd64.whl", hash = "sha256:4fb147e7a67ef577a588a0e2c17b6db51dda102c71de36f8549b6816a96e1867"}, + {file = "PyYAML-6.0.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1d4c7e777c441b20e32f52bd377e0c409713e8bb1386e1099c2415f26e479595"}, + {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a0cd17c15d3bb3fa06978b4e8958dcdc6e0174ccea823003a106c7d4d7899ac5"}, + {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:28c119d996beec18c05208a8bd78cbe4007878c6dd15091efb73a30e90539696"}, + {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7e07cbde391ba96ab58e532ff4803f79c4129397514e1413a7dc761ccd755735"}, + {file = "PyYAML-6.0.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:49a183be227561de579b4a36efbb21b3eab9651dd81b1858589f796549873dd6"}, + {file = "PyYAML-6.0.1-cp38-cp38-win32.whl", hash = "sha256:184c5108a2aca3c5b3d3bf9395d50893a7ab82a38004c8f61c258d4428e80206"}, + {file = "PyYAML-6.0.1-cp38-cp38-win_amd64.whl", hash = "sha256:1e2722cc9fbb45d9b87631ac70924c11d3a401b2d7f410cc0e3bbf249f2dca62"}, + {file = "PyYAML-6.0.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:9eb6caa9a297fc2c2fb8862bc5370d0303ddba53ba97e71f08023b6cd73d16a8"}, + {file = "PyYAML-6.0.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:c8098ddcc2a85b61647b2590f825f3db38891662cfc2fc776415143f599bb859"}, + {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5773183b6446b2c99bb77e77595dd486303b4faab2b086e7b17bc6bef28865f6"}, + {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b786eecbdf8499b9ca1d697215862083bd6d2a99965554781d0d8d1ad31e13a0"}, + {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc1bf2925a1ecd43da378f4db9e4f799775d6367bdb94671027b73b393a7c42c"}, + {file = "PyYAML-6.0.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:04ac92ad1925b2cff1db0cfebffb6ffc43457495c9b3c39d3fcae417d7125dc5"}, + {file = "PyYAML-6.0.1-cp39-cp39-win32.whl", hash = "sha256:faca3bdcf85b2fc05d06ff3fbc1f83e1391b3e724afa3feba7d13eeab355484c"}, + {file = "PyYAML-6.0.1-cp39-cp39-win_amd64.whl", hash = "sha256:510c9deebc5c0225e8c96813043e62b680ba2f9c50a08d3724c7f28a747d1486"}, + {file = "PyYAML-6.0.1.tar.gz", hash = "sha256:bfdf460b1736c775f2ba9f6a92bca30bc2095067b8a9d77876d1fad6cc3b4a43"}, ] [[package]] @@ -1718,15 +1847,127 @@ files = [ [package.dependencies] pyyaml = "*" +[[package]] +name = "readtime" +version = "3.0.0" +description = "Calculates the time some text takes the average human to read, based on Medium's read time forumula" +optional = false +python-versions = "*" +files = [ + {file = "readtime-3.0.0.tar.gz", hash = "sha256:76c5a0d773ad49858c53b42ba3a942f62fbe20cc8c6f07875797ac7dc30963a9"}, +] + +[package.dependencies] +beautifulsoup4 = ">=4.0.1" +markdown2 = ">=2.4.3" +pyquery = ">=1.2" + +[[package]] +name = "regex" +version = "2023.8.8" +description = "Alternative regular expression module, to replace re." +optional = false +python-versions = ">=3.6" +files = [ + {file = "regex-2023.8.8-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:88900f521c645f784260a8d346e12a1590f79e96403971241e64c3a265c8ecdb"}, + {file = "regex-2023.8.8-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:3611576aff55918af2697410ff0293d6071b7e00f4b09e005d614686ac4cd57c"}, + {file = "regex-2023.8.8-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b8a0ccc8f2698f120e9e5742f4b38dc944c38744d4bdfc427616f3a163dd9de5"}, + {file = "regex-2023.8.8-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c662a4cbdd6280ee56f841f14620787215a171c4e2d1744c9528bed8f5816c96"}, + {file = "regex-2023.8.8-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cf0633e4a1b667bfe0bb10b5e53fe0d5f34a6243ea2530eb342491f1adf4f739"}, + {file = "regex-2023.8.8-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:551ad543fa19e94943c5b2cebc54c73353ffff08228ee5f3376bd27b3d5b9800"}, + {file = "regex-2023.8.8-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:54de2619f5ea58474f2ac211ceea6b615af2d7e4306220d4f3fe690c91988a61"}, + {file = "regex-2023.8.8-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:5ec4b3f0aebbbe2fc0134ee30a791af522a92ad9f164858805a77442d7d18570"}, + {file = "regex-2023.8.8-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:3ae646c35cb9f820491760ac62c25b6d6b496757fda2d51be429e0e7b67ae0ab"}, + {file = "regex-2023.8.8-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:ca339088839582d01654e6f83a637a4b8194d0960477b9769d2ff2cfa0fa36d2"}, + {file = "regex-2023.8.8-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:d9b6627408021452dcd0d2cdf8da0534e19d93d070bfa8b6b4176f99711e7f90"}, + {file = "regex-2023.8.8-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:bd3366aceedf274f765a3a4bc95d6cd97b130d1dda524d8f25225d14123c01db"}, + {file = "regex-2023.8.8-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:7aed90a72fc3654fba9bc4b7f851571dcc368120432ad68b226bd593f3f6c0b7"}, + {file = "regex-2023.8.8-cp310-cp310-win32.whl", hash = "sha256:80b80b889cb767cc47f31d2b2f3dec2db8126fbcd0cff31b3925b4dc6609dcdb"}, + {file = "regex-2023.8.8-cp310-cp310-win_amd64.whl", hash = "sha256:b82edc98d107cbc7357da7a5a695901b47d6eb0420e587256ba3ad24b80b7d0b"}, + {file = "regex-2023.8.8-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:1e7d84d64c84ad97bf06f3c8cb5e48941f135ace28f450d86af6b6512f1c9a71"}, + {file = "regex-2023.8.8-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ce0f9fbe7d295f9922c0424a3637b88c6c472b75eafeaff6f910494a1fa719ef"}, + {file = "regex-2023.8.8-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:06c57e14ac723b04458df5956cfb7e2d9caa6e9d353c0b4c7d5d54fcb1325c46"}, + {file = "regex-2023.8.8-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e7a9aaa5a1267125eef22cef3b63484c3241aaec6f48949b366d26c7250e0357"}, + {file = "regex-2023.8.8-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9b7408511fca48a82a119d78a77c2f5eb1b22fe88b0d2450ed0756d194fe7a9a"}, + {file = "regex-2023.8.8-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:14dc6f2d88192a67d708341f3085df6a4f5a0c7b03dec08d763ca2cd86e9f559"}, + {file = "regex-2023.8.8-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:48c640b99213643d141550326f34f0502fedb1798adb3c9eb79650b1ecb2f177"}, + {file = "regex-2023.8.8-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:0085da0f6c6393428bf0d9c08d8b1874d805bb55e17cb1dfa5ddb7cfb11140bf"}, + {file = "regex-2023.8.8-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:964b16dcc10c79a4a2be9f1273fcc2684a9eedb3906439720598029a797b46e6"}, + {file = "regex-2023.8.8-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:7ce606c14bb195b0e5108544b540e2c5faed6843367e4ab3deb5c6aa5e681208"}, + {file = "regex-2023.8.8-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:40f029d73b10fac448c73d6eb33d57b34607f40116e9f6e9f0d32e9229b147d7"}, + {file = "regex-2023.8.8-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:3b8e6ea6be6d64104d8e9afc34c151926f8182f84e7ac290a93925c0db004bfd"}, + {file = "regex-2023.8.8-cp311-cp311-win32.whl", hash = "sha256:942f8b1f3b223638b02df7df79140646c03938d488fbfb771824f3d05fc083a8"}, + {file = "regex-2023.8.8-cp311-cp311-win_amd64.whl", hash = "sha256:51d8ea2a3a1a8fe4f67de21b8b93757005213e8ac3917567872f2865185fa7fb"}, + {file = "regex-2023.8.8-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:e951d1a8e9963ea51efd7f150450803e3b95db5939f994ad3d5edac2b6f6e2b4"}, + {file = "regex-2023.8.8-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:704f63b774218207b8ccc6c47fcef5340741e5d839d11d606f70af93ee78e4d4"}, + {file = "regex-2023.8.8-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:22283c769a7b01c8ac355d5be0715bf6929b6267619505e289f792b01304d898"}, + {file = "regex-2023.8.8-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:91129ff1bb0619bc1f4ad19485718cc623a2dc433dff95baadbf89405c7f6b57"}, + {file = "regex-2023.8.8-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:de35342190deb7b866ad6ba5cbcccb2d22c0487ee0cbb251efef0843d705f0d4"}, + {file = "regex-2023.8.8-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b993b6f524d1e274a5062488a43e3f9f8764ee9745ccd8e8193df743dbe5ee61"}, + {file = "regex-2023.8.8-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:3026cbcf11d79095a32d9a13bbc572a458727bd5b1ca332df4a79faecd45281c"}, + {file = "regex-2023.8.8-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:293352710172239bf579c90a9864d0df57340b6fd21272345222fb6371bf82b3"}, + {file = "regex-2023.8.8-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:d909b5a3fff619dc7e48b6b1bedc2f30ec43033ba7af32f936c10839e81b9217"}, + {file = "regex-2023.8.8-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:3d370ff652323c5307d9c8e4c62efd1956fb08051b0e9210212bc51168b4ff56"}, + {file = "regex-2023.8.8-cp36-cp36m-musllinux_1_1_s390x.whl", hash = "sha256:b076da1ed19dc37788f6a934c60adf97bd02c7eea461b73730513921a85d4235"}, + {file = "regex-2023.8.8-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:e9941a4ada58f6218694f382e43fdd256e97615db9da135e77359da257a7168b"}, + {file = "regex-2023.8.8-cp36-cp36m-win32.whl", hash = "sha256:a8c65c17aed7e15a0c824cdc63a6b104dfc530f6fa8cb6ac51c437af52b481c7"}, + {file = "regex-2023.8.8-cp36-cp36m-win_amd64.whl", hash = "sha256:aadf28046e77a72f30dcc1ab185639e8de7f4104b8cb5c6dfa5d8ed860e57236"}, + {file = "regex-2023.8.8-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:423adfa872b4908843ac3e7a30f957f5d5282944b81ca0a3b8a7ccbbfaa06103"}, + {file = "regex-2023.8.8-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4ae594c66f4a7e1ea67232a0846649a7c94c188d6c071ac0210c3e86a5f92109"}, + {file = "regex-2023.8.8-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e51c80c168074faa793685656c38eb7a06cbad7774c8cbc3ea05552d615393d8"}, + {file = "regex-2023.8.8-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:09b7f4c66aa9d1522b06e31a54f15581c37286237208df1345108fcf4e050c18"}, + {file = "regex-2023.8.8-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2e73e5243af12d9cd6a9d6a45a43570dbe2e5b1cdfc862f5ae2b031e44dd95a8"}, + {file = "regex-2023.8.8-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:941460db8fe3bd613db52f05259c9336f5a47ccae7d7def44cc277184030a116"}, + {file = "regex-2023.8.8-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f0ccf3e01afeb412a1a9993049cb160d0352dba635bbca7762b2dc722aa5742a"}, + {file = "regex-2023.8.8-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:2e9216e0d2cdce7dbc9be48cb3eacb962740a09b011a116fd7af8c832ab116ca"}, + {file = "regex-2023.8.8-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:5cd9cd7170459b9223c5e592ac036e0704bee765706445c353d96f2890e816c8"}, + {file = "regex-2023.8.8-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:4873ef92e03a4309b3ccd8281454801b291b689f6ad45ef8c3658b6fa761d7ac"}, + {file = "regex-2023.8.8-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:239c3c2a339d3b3ddd51c2daef10874410917cd2b998f043c13e2084cb191684"}, + {file = "regex-2023.8.8-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:1005c60ed7037be0d9dea1f9c53cc42f836188227366370867222bda4c3c6bd7"}, + {file = "regex-2023.8.8-cp37-cp37m-win32.whl", hash = "sha256:e6bd1e9b95bc5614a7a9c9c44fde9539cba1c823b43a9f7bc11266446dd568e3"}, + {file = "regex-2023.8.8-cp37-cp37m-win_amd64.whl", hash = "sha256:9a96edd79661e93327cfeac4edec72a4046e14550a1d22aa0dd2e3ca52aec921"}, + {file = "regex-2023.8.8-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:f2181c20ef18747d5f4a7ea513e09ea03bdd50884a11ce46066bb90fe4213675"}, + {file = "regex-2023.8.8-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:a2ad5add903eb7cdde2b7c64aaca405f3957ab34f16594d2b78d53b8b1a6a7d6"}, + {file = "regex-2023.8.8-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9233ac249b354c54146e392e8a451e465dd2d967fc773690811d3a8c240ac601"}, + {file = "regex-2023.8.8-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:920974009fb37b20d32afcdf0227a2e707eb83fe418713f7a8b7de038b870d0b"}, + {file = "regex-2023.8.8-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cd2b6c5dfe0929b6c23dde9624483380b170b6e34ed79054ad131b20203a1a63"}, + {file = "regex-2023.8.8-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:96979d753b1dc3b2169003e1854dc67bfc86edf93c01e84757927f810b8c3c93"}, + {file = "regex-2023.8.8-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2ae54a338191e1356253e7883d9d19f8679b6143703086245fb14d1f20196be9"}, + {file = "regex-2023.8.8-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:2162ae2eb8b079622176a81b65d486ba50b888271302190870b8cc488587d280"}, + {file = "regex-2023.8.8-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:c884d1a59e69e03b93cf0dfee8794c63d7de0ee8f7ffb76e5f75be8131b6400a"}, + {file = "regex-2023.8.8-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:cf9273e96f3ee2ac89ffcb17627a78f78e7516b08f94dc435844ae72576a276e"}, + {file = "regex-2023.8.8-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:83215147121e15d5f3a45d99abeed9cf1fe16869d5c233b08c56cdf75f43a504"}, + {file = "regex-2023.8.8-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:3f7454aa427b8ab9101f3787eb178057c5250478e39b99540cfc2b889c7d0586"}, + {file = "regex-2023.8.8-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:f0640913d2c1044d97e30d7c41728195fc37e54d190c5385eacb52115127b882"}, + {file = "regex-2023.8.8-cp38-cp38-win32.whl", hash = "sha256:0c59122ceccb905a941fb23b087b8eafc5290bf983ebcb14d2301febcbe199c7"}, + {file = "regex-2023.8.8-cp38-cp38-win_amd64.whl", hash = "sha256:c12f6f67495ea05c3d542d119d270007090bad5b843f642d418eb601ec0fa7be"}, + {file = "regex-2023.8.8-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:82cd0a69cd28f6cc3789cc6adeb1027f79526b1ab50b1f6062bbc3a0ccb2dbc3"}, + {file = "regex-2023.8.8-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:bb34d1605f96a245fc39790a117ac1bac8de84ab7691637b26ab2c5efb8f228c"}, + {file = "regex-2023.8.8-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:987b9ac04d0b38ef4f89fbc035e84a7efad9cdd5f1e29024f9289182c8d99e09"}, + {file = "regex-2023.8.8-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9dd6082f4e2aec9b6a0927202c85bc1b09dcab113f97265127c1dc20e2e32495"}, + {file = "regex-2023.8.8-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7eb95fe8222932c10d4436e7a6f7c99991e3fdd9f36c949eff16a69246dee2dc"}, + {file = "regex-2023.8.8-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7098c524ba9f20717a56a8d551d2ed491ea89cbf37e540759ed3b776a4f8d6eb"}, + {file = "regex-2023.8.8-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4b694430b3f00eb02c594ff5a16db30e054c1b9589a043fe9174584c6efa8033"}, + {file = "regex-2023.8.8-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:b2aeab3895d778155054abea5238d0eb9a72e9242bd4b43f42fd911ef9a13470"}, + {file = "regex-2023.8.8-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:988631b9d78b546e284478c2ec15c8a85960e262e247b35ca5eaf7ee22f6050a"}, + {file = "regex-2023.8.8-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:67ecd894e56a0c6108ec5ab1d8fa8418ec0cff45844a855966b875d1039a2e34"}, + {file = "regex-2023.8.8-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:14898830f0a0eb67cae2bbbc787c1a7d6e34ecc06fbd39d3af5fe29a4468e2c9"}, + {file = "regex-2023.8.8-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:f2200e00b62568cfd920127782c61bc1c546062a879cdc741cfcc6976668dfcf"}, + {file = "regex-2023.8.8-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:9691a549c19c22d26a4f3b948071e93517bdf86e41b81d8c6ac8a964bb71e5a6"}, + {file = "regex-2023.8.8-cp39-cp39-win32.whl", hash = "sha256:6ab2ed84bf0137927846b37e882745a827458689eb969028af8032b1b3dac78e"}, + {file = "regex-2023.8.8-cp39-cp39-win_amd64.whl", hash = "sha256:5543c055d8ec7801901e1193a51570643d6a6ab8751b1f7dd9af71af467538bb"}, + {file = "regex-2023.8.8.tar.gz", hash = "sha256:fcbdc5f2b0f1cd0f6a56cdb46fe41d2cce1e644e3b68832f3eeebc5fb0f7712e"}, +] + [[package]] name = "requests" -version = "2.30.0" +version = "2.31.0" description = "Python HTTP for Humans." optional = false python-versions = ">=3.7" files = [ - {file = "requests-2.30.0-py3-none-any.whl", hash = "sha256:10e94cc4f3121ee6da529d358cdaeaff2f1c409cd377dbc72b825852f2f7e294"}, - {file = "requests-2.30.0.tar.gz", hash = "sha256:239d7d4458afcb28a692cdd298d87542235f4ca8d36d03a15bfc128a6559a2f4"}, + {file = "requests-2.31.0-py3-none-any.whl", hash = "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f"}, + {file = "requests-2.31.0.tar.gz", hash = "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1"}, ] [package.dependencies] @@ -1741,13 +1982,13 @@ use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] [[package]] name = "requests-mock" -version = "1.10.0" +version = "1.11.0" description = "Mock out responses from the requests package" optional = false python-versions = "*" files = [ - {file = "requests-mock-1.10.0.tar.gz", hash = "sha256:59c9c32419a9fb1ae83ec242d98e889c45bd7d7a65d48375cc243ec08441658b"}, - {file = "requests_mock-1.10.0-py2.py3-none-any.whl", hash = "sha256:2fdbb637ad17ee15c06f33d31169e71bf9fe2bdb7bc9da26185be0dd8d842699"}, + {file = "requests-mock-1.11.0.tar.gz", hash = "sha256:ef10b572b489a5f28e09b708697208c4a3b2b89ef80a9f01584340ea357ec3c4"}, + {file = "requests_mock-1.11.0-py2.py3-none-any.whl", hash = "sha256:f7fae383f228633f6bececebdab236c478ace2284d6292c6e7e2867b9ab74d15"}, ] [package.dependencies] @@ -1756,21 +1997,21 @@ six = "*" [package.extras] fixture = ["fixtures"] -test = ["fixtures", "mock", "purl", "pytest", "requests-futures", "sphinx", "testrepository (>=0.0.18)", "testtools"] +test = ["fixtures", "mock", "purl", "pytest", "requests-futures", "sphinx", "testtools"] [[package]] name = "rich" -version = "13.3.5" +version = "13.5.2" 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.3.5-py3-none-any.whl", hash = "sha256:69cdf53799e63f38b95b9bf9c875f8c90e78dd62b2f00c13a911c7a3b9fa4704"}, - {file = "rich-13.3.5.tar.gz", hash = "sha256:2d11b9b8dd03868f09b4fffadc84a6a8cda574e40dc90821bd845720ebb8e89c"}, + {file = "rich-13.5.2-py3-none-any.whl", hash = "sha256:146a90b3b6b47cac4a73c12866a499e9817426423f57c5a66949c086191a8808"}, + {file = "rich-13.5.2.tar.gz", hash = "sha256:fb9d6c0a0f643c99eed3875b5377a184132ba9be4d61516a55273d3554d75a39"}, ] [package.dependencies] -markdown-it-py = ">=2.2.0,<3.0.0" +markdown-it-py = ">=2.2.0" pygments = ">=2.13.0,<3.0.0" typing-extensions = {version = ">=4.0.0,<5.0", markers = "python_version < \"3.9\""} @@ -1793,13 +2034,13 @@ six = "*" [[package]] name = "ruamel-yaml" -version = "0.17.26" +version = "0.17.32" description = "ruamel.yaml is a YAML parser/emitter that supports roundtrip preservation of comments, seq/map flow style, and map key order" optional = false python-versions = ">=3" files = [ - {file = "ruamel.yaml-0.17.26-py3-none-any.whl", hash = "sha256:25d0ee82a0a9a6f44683dcf8c282340def4074a4562f3a24f55695bb254c1693"}, - {file = "ruamel.yaml-0.17.26.tar.gz", hash = "sha256:baa2d0a5aad2034826c439ce61c142c07082b76f4791d54145e131206e998059"}, + {file = "ruamel.yaml-0.17.32-py3-none-any.whl", hash = "sha256:23cd2ed620231677564646b0c6a89d138b6822a0d78656df7abda5879ec4f447"}, + {file = "ruamel.yaml-0.17.32.tar.gz", hash = "sha256:ec939063761914e14542972a5cba6d33c23b0859ab6342f61cf070cfc600efc2"}, ] [package.dependencies] @@ -1871,19 +2112,19 @@ paramiko = "*" [[package]] name = "setuptools" -version = "67.7.2" +version = "68.2.0" description = "Easily download, build, install, upgrade, and uninstall Python packages" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "setuptools-67.7.2-py3-none-any.whl", hash = "sha256:23aaf86b85ca52ceb801d32703f12d77517b2556af839621c641fca11287952b"}, - {file = "setuptools-67.7.2.tar.gz", hash = "sha256:f104fa03692a2602fa0fec6c6a9e63b6c8a968de13e17c026957dd1f53d80990"}, + {file = "setuptools-68.2.0-py3-none-any.whl", hash = "sha256:af3d5949030c3f493f550876b2fd1dd5ec66689c4ee5d5344f009746f71fd5a8"}, + {file = "setuptools-68.2.0.tar.gz", hash = "sha256:00478ca80aeebeecb2f288d3206b0de568df5cd2b8fada1209843cc9a8d88a48"}, ] [package.extras] -docs = ["furo", "jaraco.packaging (>=9)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-hoverxref (<2)", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (==0.8.3)", "sphinx-reredirects", "sphinxcontrib-towncrier"] -testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8 (<5)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pip (>=19.1)", "pip-run (>=8.8)", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=1.3)", "pytest-flake8", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] -testing-integration = ["build[virtualenv]", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-hoverxref (<2)", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"] +testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pip (>=19.1)", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-ruff", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] +testing-integration = ["build[virtualenv]", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] [[package]] name = "six" @@ -1929,30 +2170,40 @@ files = [ {file = "snowballstemmer-2.2.0.tar.gz", hash = "sha256:09b16deb8547d3412ad7b590689584cd0fe25ec8db3be37788be3810cbf19cb1"}, ] +[[package]] +name = "soupsieve" +version = "2.5" +description = "A modern CSS selector implementation for Beautiful Soup." +optional = false +python-versions = ">=3.8" +files = [ + {file = "soupsieve-2.5-py3-none-any.whl", hash = "sha256:eaa337ff55a1579b6549dc679565eac1e3d000563bcb1c8ab0d0fefbc0c2cdc7"}, + {file = "soupsieve-2.5.tar.gz", hash = "sha256:5663d5a7b3bfaeee0bc4372e7fc48f9cff4940b3eec54a6451cc5299f1097690"}, +] + [[package]] name = "stevedore" -version = "3.5.2" +version = "5.1.0" description = "Manage dynamic plugins for Python applications" optional = false -python-versions = ">=3.6" +python-versions = ">=3.8" files = [ - {file = "stevedore-3.5.2-py3-none-any.whl", hash = "sha256:fa2630e3d0ad3e22d4914aff2501445815b9a4467a6edc49387c667a38faf5bf"}, - {file = "stevedore-3.5.2.tar.gz", hash = "sha256:cf99f41fc0d5a4f185ca4d3d42b03be9011b0a1ec1a4ea1a282be1b4b306dcc2"}, + {file = "stevedore-5.1.0-py3-none-any.whl", hash = "sha256:8cc040628f3cea5d7128f2e76cf486b2251a4e543c7b938f58d9a377f6694a2d"}, + {file = "stevedore-5.1.0.tar.gz", hash = "sha256:a54534acf9b89bc7ed264807013b505bf07f74dbe4bcfa37d32bd063870b087c"}, ] [package.dependencies] -importlib-metadata = {version = ">=1.7.0", markers = "python_version < \"3.8\""} pbr = ">=2.0.0,<2.1.0 || >2.1.0" [[package]] name = "tenacity" -version = "8.2.2" +version = "8.2.3" description = "Retry code until it succeeds" optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" files = [ - {file = "tenacity-8.2.2-py3-none-any.whl", hash = "sha256:2f277afb21b851637e8f52e6a613ff08734c347dc19ade928e519d7d2d8569b0"}, - {file = "tenacity-8.2.2.tar.gz", hash = "sha256:43af037822bd0029025877f3b2d97cc4d7bb0c2991000a3d59d71517c5c969e0"}, + {file = "tenacity-8.2.3-py3-none-any.whl", hash = "sha256:ce510e327a630c9e1beaf17d42e6ffacc88185044ad85cf74c0a8887c6a0f88c"}, + {file = "tenacity-8.2.3.tar.gz", hash = "sha256:5398ef0d78e63f40007c1fb4c0bff96e1911394d2fa8d194f77619c05ff6cc8a"}, ] [package.extras] @@ -1995,6 +2246,17 @@ files = [ {file = "tomli-2.0.1.tar.gz", hash = "sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f"}, ] +[[package]] +name = "tomlkit" +version = "0.12.1" +description = "Style preserving TOML library" +optional = false +python-versions = ">=3.7" +files = [ + {file = "tomlkit-0.12.1-py3-none-any.whl", hash = "sha256:712cbd236609acc6a3e2e97253dfc52d4c2082982a88f61b640ecf0817eab899"}, + {file = "tomlkit-0.12.1.tar.gz", hash = "sha256:38e1ff8edb991273ec9f6181244a6a391ac30e9f5098e7535640ea6be97a7c86"}, +] + [[package]] name = "transitions" version = "0.9.0" @@ -2015,13 +2277,13 @@ test = ["pytest"] [[package]] name = "ttp" -version = "0.9.4" +version = "0.9.5" description = "Template Text Parser" optional = false python-versions = ">=2.7,<4.0" files = [ - {file = "ttp-0.9.4-py2.py3-none-any.whl", hash = "sha256:550e26fd742703f9fd0fac09e3701766190d4551978fccdead294d49cdec2423"}, - {file = "ttp-0.9.4.tar.gz", hash = "sha256:44aaef0561f83d588563918157a502fc9f928e5bc3867026c2cc7b5a8c0dc3e1"}, + {file = "ttp-0.9.5-py2.py3-none-any.whl", hash = "sha256:2c9fcf560b3f696e9fdd3554dc8e4622cbb10cac1d4fca13a7cf608c4a7fd137"}, + {file = "ttp-0.9.5.tar.gz", hash = "sha256:234414f4d3039d2d1cde09993f89f8db1b34d447f76c6a402555cefac2e59c4e"}, ] [package.extras] @@ -2045,66 +2307,32 @@ ttp = ">=0.6.0" [package.extras] docs = ["mkdocs (==1.2.4)", "mkdocs-material (==7.2.2)", "mkdocs-material-extensions (==1.0.1)", "mkdocstrings[python] (>=0.18.0,<0.19.0)", "pygments (==2.11)", "pymdown-extensions (==9.3)"] -[[package]] -name = "typed-ast" -version = "1.5.4" -description = "a fork of Python 2 and 3 ast modules with type comment support" -optional = false -python-versions = ">=3.6" -files = [ - {file = "typed_ast-1.5.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:669dd0c4167f6f2cd9f57041e03c3c2ebf9063d0757dc89f79ba1daa2bfca9d4"}, - {file = "typed_ast-1.5.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:211260621ab1cd7324e0798d6be953d00b74e0428382991adfddb352252f1d62"}, - {file = "typed_ast-1.5.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:267e3f78697a6c00c689c03db4876dd1efdfea2f251a5ad6555e82a26847b4ac"}, - {file = "typed_ast-1.5.4-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:c542eeda69212fa10a7ada75e668876fdec5f856cd3d06829e6aa64ad17c8dfe"}, - {file = "typed_ast-1.5.4-cp310-cp310-win_amd64.whl", hash = "sha256:a9916d2bb8865f973824fb47436fa45e1ebf2efd920f2b9f99342cb7fab93f72"}, - {file = "typed_ast-1.5.4-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:79b1e0869db7c830ba6a981d58711c88b6677506e648496b1f64ac7d15633aec"}, - {file = "typed_ast-1.5.4-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a94d55d142c9265f4ea46fab70977a1944ecae359ae867397757d836ea5a3f47"}, - {file = "typed_ast-1.5.4-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:183afdf0ec5b1b211724dfef3d2cad2d767cbefac291f24d69b00546c1837fb6"}, - {file = "typed_ast-1.5.4-cp36-cp36m-win_amd64.whl", hash = "sha256:639c5f0b21776605dd6c9dbe592d5228f021404dafd377e2b7ac046b0349b1a1"}, - {file = "typed_ast-1.5.4-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:cf4afcfac006ece570e32d6fa90ab74a17245b83dfd6655a6f68568098345ff6"}, - {file = "typed_ast-1.5.4-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ed855bbe3eb3715fca349c80174cfcfd699c2f9de574d40527b8429acae23a66"}, - {file = "typed_ast-1.5.4-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:6778e1b2f81dfc7bc58e4b259363b83d2e509a65198e85d5700dfae4c6c8ff1c"}, - {file = "typed_ast-1.5.4-cp37-cp37m-win_amd64.whl", hash = "sha256:0261195c2062caf107831e92a76764c81227dae162c4f75192c0d489faf751a2"}, - {file = "typed_ast-1.5.4-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:2efae9db7a8c05ad5547d522e7dbe62c83d838d3906a3716d1478b6c1d61388d"}, - {file = "typed_ast-1.5.4-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:7d5d014b7daa8b0bf2eaef684295acae12b036d79f54178b92a2b6a56f92278f"}, - {file = "typed_ast-1.5.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:370788a63915e82fd6f212865a596a0fefcbb7d408bbbb13dea723d971ed8bdc"}, - {file = "typed_ast-1.5.4-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:4e964b4ff86550a7a7d56345c7864b18f403f5bd7380edf44a3c1fb4ee7ac6c6"}, - {file = "typed_ast-1.5.4-cp38-cp38-win_amd64.whl", hash = "sha256:683407d92dc953c8a7347119596f0b0e6c55eb98ebebd9b23437501b28dcbb8e"}, - {file = "typed_ast-1.5.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:4879da6c9b73443f97e731b617184a596ac1235fe91f98d279a7af36c796da35"}, - {file = "typed_ast-1.5.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:3e123d878ba170397916557d31c8f589951e353cc95fb7f24f6bb69adc1a8a97"}, - {file = "typed_ast-1.5.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ebd9d7f80ccf7a82ac5f88c521115cc55d84e35bf8b446fcd7836eb6b98929a3"}, - {file = "typed_ast-1.5.4-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:98f80dee3c03455e92796b58b98ff6ca0b2a6f652120c263efdba4d6c5e58f72"}, - {file = "typed_ast-1.5.4-cp39-cp39-win_amd64.whl", hash = "sha256:0fdbcf2fef0ca421a3f5912555804296f0b0960f0418c440f5d6d3abb549f3e1"}, - {file = "typed_ast-1.5.4.tar.gz", hash = "sha256:39e21ceb7388e4bb37f4c679d72707ed46c2fbf2a5609b8b8ebc4b067d977df2"}, -] - [[package]] name = "typing-extensions" -version = "4.5.0" +version = "4.7.1" description = "Backported and Experimental Type Hints for Python 3.7+" optional = false python-versions = ">=3.7" files = [ - {file = "typing_extensions-4.5.0-py3-none-any.whl", hash = "sha256:fb33085c39dd998ac16d1431ebc293a8b3eedd00fd4a32de0ff79002c19511b4"}, - {file = "typing_extensions-4.5.0.tar.gz", hash = "sha256:5cb5f4a79139d699607b3ef622a1dedafa84e115ab0024e0d9c044a9479ca7cb"}, + {file = "typing_extensions-4.7.1-py3-none-any.whl", hash = "sha256:440d5dd3af93b060174bf433bccd69b0babc3b15b1a8dca43789fd7f61514b36"}, + {file = "typing_extensions-4.7.1.tar.gz", hash = "sha256:b75ddc264f0ba5615db7ba217daeb99701ad295353c45f9e95963337ceeeffb2"}, ] [[package]] name = "urllib3" -version = "2.0.2" +version = "1.26.16" description = "HTTP library with thread-safe connection pooling, file post, and more." optional = false -python-versions = ">=3.7" +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*" files = [ - {file = "urllib3-2.0.2-py3-none-any.whl", hash = "sha256:d055c2f9d38dc53c808f6fdc8eab7360b6fdbbde02340ed25cfbcd817c62469e"}, - {file = "urllib3-2.0.2.tar.gz", hash = "sha256:61717a1095d7e155cdb737ac7bb2f4324a858a1e2e6466f6d03ff630ca68d3cc"}, + {file = "urllib3-1.26.16-py2.py3-none-any.whl", hash = "sha256:8d36afa7616d8ab714608411b4a3b13e58f463aee519024578e062e141dce20f"}, + {file = "urllib3-1.26.16.tar.gz", hash = "sha256:8f135f6502756bde6b2a9b28989df5fbe87c9970cecaa69041edcce7f0589b14"}, ] [package.extras] -brotli = ["brotli (>=1.0.9)", "brotlicffi (>=0.8.0)"] -secure = ["certifi", "cryptography (>=1.9)", "idna (>=2.0.0)", "pyopenssl (>=17.1.0)", "urllib3-secure-extra"] -socks = ["pysocks (>=1.5.6,!=1.5.7,<2.0)"] -zstd = ["zstandard (>=0.18.0)"] +brotli = ["brotli (>=1.0.9)", "brotlicffi (>=0.8.0)", "brotlipy (>=0.6.0)"] +secure = ["certifi", "cryptography (>=1.3.4)", "idna (>=2.0.0)", "ipaddress", "pyOpenSSL (>=0.14)", "urllib3-secure-extra"] +socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"] [[package]] name = "watchdog" @@ -2231,13 +2459,13 @@ files = [ [[package]] name = "yamllint" -version = "1.31.0" +version = "1.32.0" description = "A linter for YAML files." optional = false python-versions = ">=3.7" files = [ - {file = "yamllint-1.31.0-py3-none-any.whl", hash = "sha256:15f4bdb645e6a4a0a22fe5415bc38b4a934c51419b30104896d2f3f95e329185"}, - {file = "yamllint-1.31.0.tar.gz", hash = "sha256:2d83f1d12f733e162a87e06b176149d7bb9c5bae4a9e5fce1c771d7f703f7a65"}, + {file = "yamllint-1.32.0-py3-none-any.whl", hash = "sha256:d97a66e48da820829d96077d76b8dfbe6c6140f106e558dae87e81ac4e6b30b7"}, + {file = "yamllint-1.32.0.tar.gz", hash = "sha256:d01dde008c65de5b235188ab3110bebc59d18e5c65fc8a58267cd211cd9df34a"}, ] [package.dependencies] @@ -2262,23 +2490,23 @@ pyyaml = "*" [[package]] name = "zipp" -version = "3.15.0" +version = "3.16.2" description = "Backport of pathlib-compatible object wrapper for zip files" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "zipp-3.15.0-py3-none-any.whl", hash = "sha256:48904fc76a60e542af151aded95726c1a5c34ed43ab4134b597665c86d7ad556"}, - {file = "zipp-3.15.0.tar.gz", hash = "sha256:112929ad649da941c23de50f356a2b5570c954b65150642bccdd66bf194d224b"}, + {file = "zipp-3.16.2-py3-none-any.whl", hash = "sha256:679e51dd4403591b2d6838a48de3d283f3d188412a9782faadf845f298736ba0"}, + {file = "zipp-3.16.2.tar.gz", hash = "sha256:ebc15946aa78bd63458992fc81ec3b6f7b1e92d51c35e6de1c3804e73b799147"}, ] [package.extras] -docs = ["furo", "jaraco.packaging (>=9)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-lint"] -testing = ["big-O", "flake8 (<5)", "jaraco.functools", "jaraco.itertools", "more-itertools", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=1.3)", "pytest-flake8", "pytest-mypy (>=0.9.1)"] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-lint"] +testing = ["big-O", "jaraco.functools", "jaraco.itertools", "more-itertools", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-ignore-flaky", "pytest-mypy (>=0.9.1)", "pytest-ruff"] [extras] mikrotik-driver = ["routeros-api"] [metadata] lock-version = "2.0" -python-versions = "^3.7" -content-hash = "0ef67839183f7725e0ef4a5ff16666e7bed68843a59b9c1f8f80fdf33fcca3a6" +python-versions = "^3.8" +content-hash = "728be2cb46ef3e08d1286afab8846f110831620aed33c3e0384d53b41bb960a2" diff --git a/pyproject.toml b/pyproject.toml index 0382b0a..c7f56bb 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,6 +1,6 @@ [tool.poetry] name = "nornir-nautobot" -version = "2.6.0" +version = "3.0.0rc1" description = "Nornir Nautobot" authors = ["Network to Code, LLC "] readme = "README.md" @@ -10,26 +10,29 @@ classifiers = [ "Intended Audience :: Developers", "Development Status :: 5 - Production/Stable", "Programming Language :: Python :: 3", - "Programming Language :: Python :: 3.7", "Programming Language :: Python :: 3.8", "Programming Language :: Python :: 3.9", "Programming Language :: Python :: 3.10", + "Programming Language :: Python :: 3.11", ] repository = "https://github.com/nautobot/nornir-nautobot" homepage = "https://nautobot.com" -documentation = "https://nornir-nautobot.readthedocs.io" +documentation = "https://docs.nautobot.com/projects/nornir-nautobot/en/latest/" +[tool.poetry.urls] +"Changelog" = "https://docs.nautobot.com/projects/nornir-nautobot/en/latest/dev/CHANGELOG/" +"Bug Tracker" = "https://github.com/nautobot/nornir-nautobot/issues" [tool.poetry.dependencies] -python = "^3.7" +python = "^3.8" nornir = "^3.0.0" requests = "^2.25.1" nornir-utils = "^0" nornir-napalm = ">=0.4.0 <1.0.0" -nornir-jinja2 = "^0" +nornir-jinja2 = "^0.2.0" nornir-netmiko = "^1" -pynautobot = "^1.0.1" -netutils = "^1" +pynautobot = "^1.5.0" +netutils = "^1.6.0" routeros-api = {version = "^0.17.0", optional = true} httpx = "^0.24.1" @@ -47,12 +50,13 @@ toml = "*" flake8 = "*" pydantic = {version = "^1.7.2", extras = ["dotenv"]} # documentation dependencies -mkdocs = "1.3.1" -mkdocs-material = "8.3.9" -mkdocstrings = "0.19" -mkdocstrings-python = "0.7.1" +mkdocs = "1.5.2" +mkdocs-material = "9.2.4" +mkdocstrings = "0.22.0" +mkdocstrings-python = "1.5.2" mkdocs-version-annotations = "1.0.0" + [tool.poetry.extras] mikrotik_driver = ["routeros-api"] @@ -61,7 +65,7 @@ mikrotik_driver = ["routeros-api"] [tool.black] line-length = 120 -target-version = ['py37'] +target-version = ['py38', 'py39', 'py310', 'py311'] include = '\.pyi?$' exclude = ''' ( @@ -92,7 +96,6 @@ no-docstring-rgx="^(_|test_|Meta$)" # Pylint and Black disagree about how to format multi-line arrays; Black wins. disable = """, line-too-long, - bad-continuation, duplicate-code, """ diff --git a/tasks.py b/tasks.py index af5a3d3..3d6587e 100644 --- a/tasks.py +++ b/tasks.py @@ -1,7 +1,7 @@ """Tasks for use with Invoke.""" import os import sys -from distutils.util import strtobool +from distutils.util import strtobool # pylint: disable=deprecated-module from invoke import task @@ -11,12 +11,6 @@ sys.exit("Please make sure to `pip install toml` or enable the Poetry shell and run `poetry install`.") -def project_ver(): - """Find version from pyproject.toml to use for docker image tagging.""" - with open("pyproject.toml", encoding="utf-8") as file: - return toml.load(file)["tool"]["poetry"].get("version", "latest") - - def is_truthy(arg): """Convert "truthy" strings into Booleans. @@ -32,26 +26,31 @@ def is_truthy(arg): return bool(strtobool(arg)) +PYPROJECT_CONFIG = toml.load("pyproject.toml") +TOOL_CONFIG = PYPROJECT_CONFIG["tool"]["poetry"] + # Can be set to a separate Python version to be used for launching or building image -PYTHON_VER = os.getenv("PYTHON_VER", "3.7") +PYTHON_VER = os.getenv("PYTHON_VER", "3.11") # Name of the docker image/image -NAME = os.getenv("IMAGE_NAME", f"nornir-nautobot-py{PYTHON_VER}") +IMAGE_NAME = os.getenv("IMAGE_NAME", TOOL_CONFIG["name"]) # Tag for the image -IMAGE_VER = os.getenv("IMAGE_VER", project_ver()) +IMAGE_VER = os.getenv("IMAGE_VER", f"{TOOL_CONFIG['version']}-py{PYTHON_VER}") # Gather current working directory for Docker commands PWD = os.getcwd() # Local or Docker execution provide "local" to run locally without docker execution INVOKE_LOCAL = is_truthy(os.getenv("INVOKE_LOCAL", False)) # pylint: disable=W1508 +# Get project name from the toml file +PROJECT_NAME = PYPROJECT_CONFIG["tool"]["poetry"]["name"] +# Get current project version from the toml file +PROJECT_VERSION = PYPROJECT_CONFIG["tool"]["poetry"]["version"] -def run_cmd(context, exec_cmd, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): +def run_cmd(context, exec_cmd, local=INVOKE_LOCAL, port=None): """Wrapper to run the invoke task commands. Args: context ([invoke.task]): Invoke task object. exec_cmd ([str]): Command to run. - name ([str], optional): Image name to use if exec_env is `docker`. Defaults to NAME. - image_ver ([str], optional): Version of image to use if exec_env is `docker`. Defaults to IMAGE_VER. local (bool): Define as `True` to execute locally Returns: @@ -61,229 +60,215 @@ def run_cmd(context, exec_cmd, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCA print(f"LOCAL - Running command {exec_cmd}") result = context.run(exec_cmd, pty=True) else: - print(f"DOCKER - Running command: {exec_cmd} container: {name}:{image_ver}") - result = context.run(f"docker run -it -v {PWD}:/local {name}:{image_ver} sh -c '{exec_cmd}'", pty=True) + print(f"DOCKER - Running command: {exec_cmd} container: {IMAGE_NAME}:{IMAGE_VER}") + if port: + result = context.run( + f"docker run -it -p {port} -v {PWD}:/local {IMAGE_NAME}:{IMAGE_VER} sh -c '{exec_cmd}'", pty=True + ) + else: + result = context.run( + f"docker run -it -v {PWD}:/local {IMAGE_NAME}:{IMAGE_VER} sh -c '{exec_cmd}'", pty=True + ) return result @task -def build( - context, name=NAME, python_ver=PYTHON_VER, image_ver=IMAGE_VER, nocache=False, forcerm=False -): # pylint: disable=too-many-arguments - """Builds an image with the provided name and python version. +def build(context, nocache=False, forcerm=False, hide=False): # pylint: disable=too-many-arguments + """Build a Docker image. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - python_ver (str): Define the Python version docker image to build from - image_ver (str): Define image version nocache (bool): Do not use cache when building the image forcerm (bool): Always remove intermediate containers + hide (bool): Hide output of Docker image build """ - print(f"Building image {name}:{image_ver}") - command = f"docker build --tag {name}:{image_ver} --build-arg PYTHON_VER={python_ver} -f Dockerfile ." + print(f"Building image {IMAGE_NAME}:{IMAGE_VER}") + command = f"docker build --tag {IMAGE_NAME}:{IMAGE_VER} --build-arg PYTHON_VER={PYTHON_VER} -f Dockerfile ." if nocache: command += " --no-cache" if forcerm: command += " --force-rm" - result = context.run(command, hide=True) + result = context.run(command, hide=hide) if result.exited != 0: - print(f"Failed to build image {name}:{image_ver}\nError: {result.stderr}") + print(f"Failed to build image {IMAGE_NAME}:{IMAGE_VER}\nError: {result.stderr}") @task -def clean_image(context, name=NAME, image_ver=IMAGE_VER): - """Removes the specific image. +def clean(context): + """Remove the project specific image. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version """ - print(f"Attempting to forcefully remove image {name}:{image_ver}") - context.run(f"docker rmi {name}:{image_ver} --force") - print(f"Successfully removed image {name}:{image_ver}") + print(f"Attempting to forcefully remove image {IMAGE_NAME}:{IMAGE_VER}") + context.run(f"docker rmi {IMAGE_NAME}:{IMAGE_VER} --force") + print(f"Successfully removed image {IMAGE_NAME}:{IMAGE_VER}") @task -def rebuild_image(context, name=NAME, python_ver=PYTHON_VER, image_ver=IMAGE_VER): - """Cleans the image and then rebuild image without using cache. +def rebuild(context): + """Clean the Docker image and then rebuild without using cache. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - python_ver (str): Define the Python version docker image to build from - image_ver (str): Define image version """ - clean_image(context, name, image_ver) - build(context, name, python_ver, image_ver) + clean(context) + build(context) + + +@task +def coverage(context, local=INVOKE_LOCAL): + """Run the coverage report against pytest. + + Args: + context (obj): Used to run specific commands + local (bool): Define as `True` to execute locally + """ + exec_cmd = "coverage run --source=nornir_nautobot -m pytest" + run_cmd(context, exec_cmd, local) + run_cmd(context, "coverage report", local) + run_cmd(context, "coverage html", local) @task -def pytest(context, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): - """Runs pytest for the specified name and Python version. +def pytest(context, local=INVOKE_LOCAL): + """Run pytest for the specified name and Python version. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Will use the container version docker image local (bool): Define as `True` to execute locally """ - # pty is set to true to properly run the docker commands due to the invocation process of docker - # https://docs.pyinvoke.org/en/latest/api/runners.html - Search for pty for more information - # Install python module exec_cmd = "pytest -vv" - run_cmd(context, exec_cmd, name, image_ver, local) + run_cmd(context, exec_cmd, local) @task -def black(context, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): - """Runs black to check that Python files adherence to black standards. +def black(context, local=INVOKE_LOCAL): + """Run black to check that Python files adherence to black standards. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version local (bool): Define as `True` to execute locally """ - # pty is set to true to properly run the docker commands due to the invocation process of docker - # https://docs.pyinvoke.org/en/latest/api/runners.html - Search for pty for more information exec_cmd = "black --check --diff ." - run_cmd(context, exec_cmd, name, image_ver, local) + run_cmd(context, exec_cmd, local) @task -def flake8(context, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): - """Runs flake8 for the specified name and Python version. +def flake8(context, local=INVOKE_LOCAL): + """Run flake8 for the specified name and Python version. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version local (bool): Define as `True` to execute locally """ - # pty is set to true to properly run the docker commands due to the invocation process of docker - # https://docs.pyinvoke.org/en/latest/api/runners.html - Search for pty for more information exec_cmd = "flake8 ." - run_cmd(context, exec_cmd, name, image_ver, local) + run_cmd(context, exec_cmd, local) @task -def pylint(context, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): - """Runs pylint for the specified name and Python version. +def pylint(context, local=INVOKE_LOCAL): + """Run pylint for the specified name and Python version. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version local (bool): Define as `True` to execute locally """ - # pty is set to true to properly run the docker commands due to the invocation process of docker - # https://docs.pyinvoke.org/en/latest/api/runners.html - Search for pty for more information exec_cmd = 'find . -name "*.py" | xargs pylint' - run_cmd(context, exec_cmd, name, image_ver, local) + run_cmd(context, exec_cmd, local) @task -def yamllint(context, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): - """Runs yamllint to validate formatting adheres to NTC defined YAML standards. +def yamllint(context, local=INVOKE_LOCAL): + """Run yamllint to validate formatting adheres to NTC defined YAML standards. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version local (bool): Define as `True` to execute locally """ - # pty is set to true to properly run the docker commands due to the invocation process of docker - # https://docs.pyinvoke.org/en/latest/api/runners.html - Search for pty for more information exec_cmd = "yamllint ." - run_cmd(context, exec_cmd, name, image_ver, local) + run_cmd(context, exec_cmd, local) @task -def pydocstyle(context, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): - """Runs pydocstyle to validate docstring formatting adheres to NTC defined standards. +def pydocstyle(context, local=INVOKE_LOCAL): + """Run pydocstyle to validate docstring formatting adheres to NTC defined standards. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version local (bool): Define as `True` to execute locally """ - # pty is set to true to properly run the docker commands due to the invocation process of docker - # https://docs.pyinvoke.org/en/latest/api/runners.html - Search for pty for more information exec_cmd = "pydocstyle ." - run_cmd(context, exec_cmd, name, image_ver, local) + run_cmd(context, exec_cmd, local) @task -def bandit(context, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): - """Runs bandit to validate basic static code security analysis. +def bandit(context, local=INVOKE_LOCAL): + """Run bandit to validate basic static code security analysis. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version local (bool): Define as `True` to execute locally """ - # pty is set to true to properly run the docker commands due to the invocation process of docker - # https://docs.pyinvoke.org/en/latest/api/runners.html - Search for pty for more information exec_cmd = "bandit --recursive ./ --configfile .bandit.yml" - run_cmd(context, exec_cmd, name, image_ver, local) + run_cmd(context, exec_cmd, local) @task -def cli(context, name=NAME, image_ver=IMAGE_VER): - """Enters the image to perform troubleshooting or dev work. +def mypy(context, local=INVOKE_LOCAL): + """Run mypy to validate typing-hints. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version + local (bool): Define as `True` to execute locally """ - dev = f"docker run -it -v {PWD}:/local {name}:{image_ver} /bin/bash" - context.run(f"{dev}", pty=True) + exec_cmd = "mypy ./nornir_nautobot" + run_cmd(context, exec_cmd, local) @task -def lint(context, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): - """Runs all tests for the specified name and Python version. +def cli(context): + """Enter the image to perform troubleshooting or dev work. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version - local (bool): Define as `True` to execute locally """ - black(context, name, image_ver, local) - flake8(context, name, image_ver, local) - pylint(context, name, image_ver, local) - yamllint(context, name, image_ver, local) - pydocstyle(context, name, image_ver, local) - bandit(context, name, image_ver, local) - - print("All linting has passed!") + dev = f"docker run -it -v {PWD}:/local {IMAGE_NAME}:{IMAGE_VER} /bin/bash" + context.run(f"{dev}", pty=True) @task -def tests(context, name=NAME, image_ver=IMAGE_VER, local=INVOKE_LOCAL): - """Runs all tests for the specified name and Python version. +def tests(context, local=INVOKE_LOCAL): + """Run all tests for the specified name and Python version. Args: context (obj): Used to run specific commands - name (str): Used to name the docker image - image_ver (str): Define image version local (bool): Define as `True` to execute locally """ - black(context, name, image_ver, local) - flake8(context, name, image_ver, local) - pylint(context, name, image_ver, local) - yamllint(context, name, image_ver, local) - pydocstyle(context, name, image_ver, local) - bandit(context, name, image_ver, local) - pytest(context, name, image_ver, local) + black(context, local) + flake8(context, local) + pylint(context, local) + yamllint(context, local) + pydocstyle(context, local) + bandit(context, local) + # mypy(context, local) + pytest(context, local) print("All tests have passed!") + + +@task +def clean_container(context): + """Remove stopped containers that source for image `nornir_nautobot:`.""" + exec_cmd = """docker container rm $(docker container ls -a | grep -E "^\S+\s+nornir_nautobot:" | awk 'NR>1 {print $1}')""" # noqa: W605 # pylint:disable=anomalous-backslash-in-string + run_cmd(context, exec_cmd, local=True) + + +@task +def docs(context, local=INVOKE_LOCAL): + """Build and serve docs locally for development.""" + exec_cmd = "mkdocs serve -v --dev-addr=0.0.0.0:8001" + run_cmd(context, exec_cmd, local, port="8001:8001") From 6b298ccaaa375bd85e1ca4a665e494be39d67a7a Mon Sep 17 00:00:00 2001 From: Patryk Szulczewski Date: Thu, 14 Sep 2023 09:31:34 +0200 Subject: [PATCH 02/13] Update for Nautobot API 2.0 (#105) Update for Nautobot API 2.0 --- docs/inventory/inventory.md | 37 ++-- ...r_multiple_sites.py => filter_location.py} | 4 +- ...r_site.py => filter_multiple_locations.py} | 4 +- ...gate_site.py => filter_negate_location.py} | 4 +- nornir_nautobot/plugins/inventory/nautobot.py | 13 +- .../plugins/tasks/dispatcher/default.py | 5 +- nornir_nautobot/utils/logger.py | 12 +- poetry.lock | 203 +++++++++++++----- pyproject.toml | 7 +- tests/unit/conftest.py | 4 +- tests/unit/mocks/00_api_root.json | 12 ++ tests/unit/mocks/01_get_devices.json | 40 ++-- tests/unit/mocks/02_get_device1.json | 14 +- tests/unit/mocks/03_get_device2.json | 14 +- tests/unit/mocks/04_get_device3.json | 14 +- ...ed.json => 05_get_locations_filtered.json} | 28 +-- tests/unit/mocks/06_get_device_msp-rtr01.json | 14 +- tests/unit/mocks/07_get_device_msp-rtr02.json | 14 +- tests/unit/test_nautobot_inventory.py | 15 +- 19 files changed, 298 insertions(+), 160 deletions(-) rename examples/{filter_multiple_sites.py => filter_location.py} (90%) rename examples/{filter_site.py => filter_multiple_locations.py} (87%) rename examples/{filter_negate_site.py => filter_negate_location.py} (89%) create mode 100644 tests/unit/mocks/00_api_root.json rename tests/unit/mocks/{05_get_sites_filtered.json => 05_get_locations_filtered.json} (86%) diff --git a/docs/inventory/inventory.md b/docs/inventory/inventory.md index d24bf2f..a88dce4 100644 --- a/docs/inventory/inventory.md +++ b/docs/inventory/inventory.md @@ -39,19 +39,26 @@ With the Nornir Nautobot Inventory plugin you have the option of using all of th ## Filtering Examples -For each of the examples the sites correspond to the first three letters of the devices below: +For each of the examples the locations correspond to the first three letters of the devices below: ```python >>> nautobot.dcim.devices.all() [den-rtr01, den-rtr02, grb-rtr01, msp-rtr01, msp-rtr02, nyc-rtr01, nyc-rtr02] ``` -### Filtering Example: Select from single site +### Filtering Example: Select from single location + +To filter from a single location you can use the filter location to get the devices at a single location based on the **Primary Key** for the location: + +> NOTE: Location names do not guarantee uniqueness in Nautobot 2.0. +> +> See this document for more information: +> +> https://docs.nautobot.com/projects/core/en/next/development/apps/api/platform-features/uniquely-identify-objects/ -To filter from a single site you can use the filter site to get the devices at a single site based on the **slug** for the site: ```python -site = "msp" +location = "db913e3b-cbe0-4463-addc-816ba6a20100" my_nornir = InitNornir( inventory={ @@ -59,7 +66,7 @@ my_nornir = InitNornir( "options": { "nautobot_url": os.getenv("NAUTOBOT_URL"), "nautobot_token": os.getenv("NAUTBOT_TOKEN"), - "filter_parameters": {"site": site}, + "filter_parameters": {"location": location}, "ssl_verify": False, }, }, @@ -73,18 +80,18 @@ print(my_nornir.inventory.hosts.keys()) This results in: ``` -root@2e8168a1c3e7:/local# python examples/filter_site.py +root@2e8168a1c3e7:/local# python examples/filter_location.py Hosts found: 2 dict_keys(['msp-rtr01', 'msp-rtr02']) ``` -### Filter Example: Multiple Sites +### Filter Example: Multiple Locations -To search within multiple sites, pass a list of site slugs. In the example below, it is the same as the previous example with a list passed in instead of a single string. +To search within multiple locations, pass a list of location Primary Keys. In the example below, it is the same as the previous example with a list passed in instead of a single string. ```python -site = ["msp", "grb"] +location = ["db913e3b-cbe0-4463-addc-816ba6a20100", "6f09aa66-96be-4b4d-955a-9c98e488f0e6"] my_nornir = InitNornir( inventory={ @@ -92,7 +99,7 @@ my_nornir = InitNornir( "options": { "nautobot_url": os.getenv("NAUTOBOT_URL"), "nautobot_token": os.getenv("NAUTBOT_TOKEN"), - "filter_parameters": {"site": site}, + "filter_parameters": {"location": location}, "ssl_verify": False, }, }, @@ -106,17 +113,17 @@ print(my_nornir.inventory.hosts.keys()) Results in: ``` -root@2e8168a1c3e7:/local# python examples/filter_multiple_sites.py +root@2e8168a1c3e7:/local# python examples/filter_multiple_locations.py Hosts found: 3 dict_keys(['grb-rtr01', 'msp-rtr01', 'msp-rtr02']) ``` -### Filtering Example: Not at a site +### Filtering Example: Not at a location The negative filters also are supported. These are all of the filters possible. Here we will search for devices **not** at _MSP_: ```python -not_site = "msp" +not_location = "db913e3b-cbe0-4463-addc-816ba6a20100" my_nornir = InitNornir( inventory={ @@ -124,7 +131,7 @@ my_nornir = InitNornir( "options": { "nautobot_url": os.getenv("NAUTOBOT_URL"), "nautobot_token": os.getenv("NAUTBOT_TOKEN"), - "filter_parameters": {"site__n": not_site}, + "filter_parameters": {"location__n": not_location}, "ssl_verify": False, }, }, @@ -138,7 +145,7 @@ print(my_nornir.inventory.hosts.keys()) Results in: ``` -root@2e8168a1c3e7:/local# python examples/filter_negate_site.py +root@2e8168a1c3e7:/local# python examples/filter_negate_location.py Hosts found: 5 dict_keys(['den-rtr01', 'den-rtr02', 'grb-rtr01', 'nyc-rtr01', 'nyc-rtr02']) ``` diff --git a/examples/filter_multiple_sites.py b/examples/filter_location.py similarity index 90% rename from examples/filter_multiple_sites.py rename to examples/filter_location.py index b113497..3eedf76 100644 --- a/examples/filter_multiple_sites.py +++ b/examples/filter_location.py @@ -22,7 +22,7 @@ def hello_world(task: Task) -> Result: def main(): """Nornir testing.""" - site = ["msp", "grb"] + location = "db913e3b-cbe0-4463-addc-816ba6a20100" my_nornir = InitNornir( inventory={ @@ -30,7 +30,7 @@ def main(): "options": { "nautobot_url": os.getenv("NAUTOBOT_URL"), "nautobot_token": os.getenv("NAUTBOT_TOKEN"), - "filter_parameters": {"site": site}, + "filter_parameters": {"location": location}, "ssl_verify": False, }, }, diff --git a/examples/filter_site.py b/examples/filter_multiple_locations.py similarity index 87% rename from examples/filter_site.py rename to examples/filter_multiple_locations.py index d0e9c78..d230bc5 100644 --- a/examples/filter_site.py +++ b/examples/filter_multiple_locations.py @@ -22,7 +22,7 @@ def hello_world(task: Task) -> Result: def main(): """Nornir testing.""" - site = "msp" + location = ["db913e3b-cbe0-4463-addc-816ba6a20100", "6f09aa66-96be-4b4d-955a-9c98e488f0e6"] my_nornir = InitNornir( inventory={ @@ -30,7 +30,7 @@ def main(): "options": { "nautobot_url": os.getenv("NAUTOBOT_URL"), "nautobot_token": os.getenv("NAUTBOT_TOKEN"), - "filter_parameters": {"site": site}, + "filter_parameters": {"location": location}, "ssl_verify": False, }, }, diff --git a/examples/filter_negate_site.py b/examples/filter_negate_location.py similarity index 89% rename from examples/filter_negate_site.py rename to examples/filter_negate_location.py index b8b0c6c..265e425 100644 --- a/examples/filter_negate_site.py +++ b/examples/filter_negate_location.py @@ -22,7 +22,7 @@ def hello_world(task: Task) -> Result: def main(): """Nornir testing.""" - not_site = "msp" + not_location = "db913e3b-cbe0-4463-addc-816ba6a20100" my_nornir = InitNornir( inventory={ @@ -30,7 +30,7 @@ def main(): "options": { "nautobot_url": os.getenv("NAUTOBOT_URL"), "nautobot_token": os.getenv("NAUTBOT_TOKEN"), - "filter_parameters": {"site__n": not_site}, + "filter_parameters": {"location__n": not_location}, "ssl_verify": False, }, }, diff --git a/nornir_nautobot/plugins/inventory/nautobot.py b/nornir_nautobot/plugins/inventory/nautobot.py index f681c48..d2727ec 100644 --- a/nornir_nautobot/plugins/inventory/nautobot.py +++ b/nornir_nautobot/plugins/inventory/nautobot.py @@ -25,7 +25,7 @@ def _set_host(data: Dict[str, Any], name: str, groups, host, defaults: Defaults) -> Host: - host_platform = getattr(data["pynautobot_object"].platform, "slug", None) + host_platform = getattr(data["pynautobot_object"].platform, "network_driver", None) connection_option = {} for key, value in data.get("connection_options", {}).items(): connection_option[key] = ConnectionOptions( @@ -106,6 +106,7 @@ def pynautobot_obj(self) -> pynautobot.core.api.Api: """ if self._pynautobot_obj is None: self._pynautobot_obj = pynautobot.api(self.nautobot_url, token=self.nautobot_token) + self.api_session.params = {"depth": 1} self._pynautobot_obj.http_session = self.api_session return self._pynautobot_obj @@ -120,8 +121,8 @@ def devices(self) -> list: else: try: self._devices = self.pynautobot_obj.dcim.devices.filter(**self.filter_parameters) - except pynautobot.core.query.RequestError: - print("Error in the query filters. Please verify the parameters.") + except pynautobot.core.query.RequestError as err: + print(f"Error in the query filters: {err.error}. Please verify the parameters.") sys.exit(1) return self._devices @@ -151,7 +152,11 @@ def load(self) -> Inventory: # Add Primary IP address, if found. Otherwise add hostname as the device name host["hostname"] = ( - str(ipaddress.IPv4Interface(device.primary_ip.address).ip) if device["primary_ip"] else device["name"] + str(ipaddress.IPv4Interface(device.primary_ip4.address).ip) + if device["primary_ip4"] + else str(ipaddress.IPv6Interface(device.primary_ip6.address).ip) + if device["primary_ip6"] + else device["name"] ) host["name"] = device.name or str(device.id) host["groups"] = [] diff --git a/nornir_nautobot/plugins/tasks/dispatcher/default.py b/nornir_nautobot/plugins/tasks/dispatcher/default.py index 254a9fe..bdd0021 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/default.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/default.py @@ -91,7 +91,7 @@ def compliance_config( features (dict): A dictionary describing the configurations required. backup_file (str): The file location of where the back configuration should be saved. intended_file (str): The file location of where the intended configuration should be saved. - platform (str): The platform slug of the device. + platform (str): The platform network_driver of the device. Returns: Result: Nornir Result object with a feature_data key of the compliance data. @@ -268,7 +268,8 @@ def get_config( if result[0].failed: # TODO: investigate this, is there a better way to handle? recursive function? logger.log_error( - f"`get_config` nornir task failed with an unexpected issue: `{str(result.exception)}`", extra={"object": obj} + f"`get_config` nornir task failed with an unexpected issue: `{str(result.exception)}`", + extra={"object": obj}, ) return result diff --git a/nornir_nautobot/utils/logger.py b/nornir_nautobot/utils/logger.py index 83265de..bf506a7 100644 --- a/nornir_nautobot/utils/logger.py +++ b/nornir_nautobot/utils/logger.py @@ -15,32 +15,32 @@ def __init__(self, name: str, nautobot_job=None, debug: bool = False, job_result self.debug = debug self.nautobot_job = nautobot_job or job_result - def log_debug(self, message: str, extra: Any=None): + def log_debug(self, message: str, extra: Any = None): """Debug, does not take obj, and only logs to jobs result when in global debug mode.""" if self.nautobot_job and self.debug: self.nautobot_job.logging.debug(message, extra=extra) self.logger.debug(message) - def log_info(self, message: str, extra: Any=None): + def log_info(self, message: str, extra: Any = None): """Log to Python logger and jogs results for info messages.""" if self.nautobot_job: self.nautobot_job.logging.info(message, extra=extra) self.logger.info("%s | %s", str(extra), message) - def log_warning(self, message: str, extra: Any=None): + def log_warning(self, message: str, extra: Any = None): """Log to Python logger and jogs results for warning messages.""" if self.nautobot_job: self.nautobot_job.logging.warning(message, extra=extra) self.logger.warning("%s | %s", str(extra), message) - def log_error(self, message: str, extra: Any=None): + def log_error(self, message: str, extra: Any = None): """Log to Python logger and jogs results for error messages.""" if self.nautobot_job: self.nautobot_job.logging.error(message, extra=extra) self.logger.error("%s | %s", str(extra), message) - def log_critical(self, message: str, extra: Any=None): + def log_critical(self, message: str, extra: Any = None): """Log to Python logger and jogs results for critical messages.""" if self.nautobot_job: self.nautobot_job.logging.critical(message, extra=extra) - self.logger.critical("%s | %s", str(extra), message) \ No newline at end of file + self.logger.critical("%s | %s", str(extra), message) diff --git a/poetry.lock b/poetry.lock index 518461c..6c24985 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,9 +1,10 @@ -# 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.4.2 and should not be changed by hand. [[package]] name = "anyio" version = "4.0.0" description = "High level compatibility layer for multiple asynchronous event loop implementations" +category = "main" optional = false python-versions = ">=3.8" files = [ @@ -25,6 +26,7 @@ trio = ["trio (>=0.22)"] name = "astroid" version = "2.15.6" description = "An abstract syntax tree for Python with inference support." +category = "dev" optional = false python-versions = ">=3.7.2" files = [ @@ -44,6 +46,7 @@ wrapt = [ name = "babel" version = "2.12.1" description = "Internationalization utilities" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -58,6 +61,7 @@ pytz = {version = ">=2015.7", markers = "python_version < \"3.9\""} name = "bandit" version = "1.7.5" description = "Security oriented static analyser for python code." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -81,6 +85,7 @@ yaml = ["PyYAML"] name = "bcrypt" version = "4.0.1" description = "Modern password hashing for your software and your servers" +category = "main" optional = false python-versions = ">=3.6" files = [ @@ -115,6 +120,7 @@ typecheck = ["mypy"] name = "beautifulsoup4" version = "4.12.2" description = "Screen-scraping library" +category = "dev" optional = false python-versions = ">=3.6.0" files = [ @@ -131,33 +137,34 @@ lxml = ["lxml"] [[package]] name = "black" -version = "23.7.0" +version = "23.9.1" description = "The uncompromising code formatter." +category = "dev" optional = false python-versions = ">=3.8" files = [ - {file = "black-23.7.0-cp310-cp310-macosx_10_16_arm64.whl", hash = "sha256:5c4bc552ab52f6c1c506ccae05681fab58c3f72d59ae6e6639e8885e94fe2587"}, - {file = "black-23.7.0-cp310-cp310-macosx_10_16_universal2.whl", hash = "sha256:552513d5cd5694590d7ef6f46e1767a4df9af168d449ff767b13b084c020e63f"}, - {file = "black-23.7.0-cp310-cp310-macosx_10_16_x86_64.whl", hash = "sha256:86cee259349b4448adb4ef9b204bb4467aae74a386bce85d56ba4f5dc0da27be"}, - {file = "black-23.7.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:501387a9edcb75d7ae8a4412bb8749900386eaef258f1aefab18adddea1936bc"}, - {file = "black-23.7.0-cp310-cp310-win_amd64.whl", hash = "sha256:fb074d8b213749fa1d077d630db0d5f8cc3b2ae63587ad4116e8a436e9bbe995"}, - {file = "black-23.7.0-cp311-cp311-macosx_10_16_arm64.whl", hash = "sha256:b5b0ee6d96b345a8b420100b7d71ebfdd19fab5e8301aff48ec270042cd40ac2"}, - {file = "black-23.7.0-cp311-cp311-macosx_10_16_universal2.whl", hash = "sha256:893695a76b140881531062d48476ebe4a48f5d1e9388177e175d76234ca247cd"}, - {file = "black-23.7.0-cp311-cp311-macosx_10_16_x86_64.whl", hash = "sha256:c333286dc3ddca6fdff74670b911cccedacb4ef0a60b34e491b8a67c833b343a"}, - {file = "black-23.7.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:831d8f54c3a8c8cf55f64d0422ee875eecac26f5f649fb6c1df65316b67c8926"}, - {file = "black-23.7.0-cp311-cp311-win_amd64.whl", hash = "sha256:7f3bf2dec7d541b4619b8ce526bda74a6b0bffc480a163fed32eb8b3c9aed8ad"}, - {file = "black-23.7.0-cp38-cp38-macosx_10_16_arm64.whl", hash = "sha256:f9062af71c59c004cd519e2fb8f5d25d39e46d3af011b41ab43b9c74e27e236f"}, - {file = "black-23.7.0-cp38-cp38-macosx_10_16_universal2.whl", hash = "sha256:01ede61aac8c154b55f35301fac3e730baf0c9cf8120f65a9cd61a81cfb4a0c3"}, - {file = "black-23.7.0-cp38-cp38-macosx_10_16_x86_64.whl", hash = "sha256:327a8c2550ddc573b51e2c352adb88143464bb9d92c10416feb86b0f5aee5ff6"}, - {file = "black-23.7.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6d1c6022b86f83b632d06f2b02774134def5d4d4f1dac8bef16d90cda18ba28a"}, - {file = "black-23.7.0-cp38-cp38-win_amd64.whl", hash = "sha256:27eb7a0c71604d5de083757fbdb245b1a4fae60e9596514c6ec497eb63f95320"}, - {file = "black-23.7.0-cp39-cp39-macosx_10_16_arm64.whl", hash = "sha256:8417dbd2f57b5701492cd46edcecc4f9208dc75529bcf76c514864e48da867d9"}, - {file = "black-23.7.0-cp39-cp39-macosx_10_16_universal2.whl", hash = "sha256:47e56d83aad53ca140da0af87678fb38e44fd6bc0af71eebab2d1f59b1acf1d3"}, - {file = "black-23.7.0-cp39-cp39-macosx_10_16_x86_64.whl", hash = "sha256:25cc308838fe71f7065df53aedd20327969d05671bac95b38fdf37ebe70ac087"}, - {file = "black-23.7.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:642496b675095d423f9b8448243336f8ec71c9d4d57ec17bf795b67f08132a91"}, - {file = "black-23.7.0-cp39-cp39-win_amd64.whl", hash = "sha256:ad0014efc7acf0bd745792bd0d8857413652979200ab924fbf239062adc12491"}, - {file = "black-23.7.0-py3-none-any.whl", hash = "sha256:9fd59d418c60c0348505f2ddf9609c1e1de8e7493eab96198fc89d9f865e7a96"}, - {file = "black-23.7.0.tar.gz", hash = "sha256:022a582720b0d9480ed82576c920a8c1dde97cc38ff11d8d8859b3bd6ca9eedb"}, + {file = "black-23.9.1-cp310-cp310-macosx_10_16_arm64.whl", hash = "sha256:d6bc09188020c9ac2555a498949401ab35bb6bf76d4e0f8ee251694664df6301"}, + {file = "black-23.9.1-cp310-cp310-macosx_10_16_universal2.whl", hash = "sha256:13ef033794029b85dfea8032c9d3b92b42b526f1ff4bf13b2182ce4e917f5100"}, + {file = "black-23.9.1-cp310-cp310-macosx_10_16_x86_64.whl", hash = "sha256:75a2dc41b183d4872d3a500d2b9c9016e67ed95738a3624f4751a0cb4818fe71"}, + {file = "black-23.9.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:13a2e4a93bb8ca74a749b6974925c27219bb3df4d42fc45e948a5d9feb5122b7"}, + {file = "black-23.9.1-cp310-cp310-win_amd64.whl", hash = "sha256:adc3e4442eef57f99b5590b245a328aad19c99552e0bdc7f0b04db6656debd80"}, + {file = "black-23.9.1-cp311-cp311-macosx_10_16_arm64.whl", hash = "sha256:8431445bf62d2a914b541da7ab3e2b4f3bc052d2ccbf157ebad18ea126efb91f"}, + {file = "black-23.9.1-cp311-cp311-macosx_10_16_universal2.whl", hash = "sha256:8fc1ddcf83f996247505db6b715294eba56ea9372e107fd54963c7553f2b6dfe"}, + {file = "black-23.9.1-cp311-cp311-macosx_10_16_x86_64.whl", hash = "sha256:7d30ec46de88091e4316b17ae58bbbfc12b2de05e069030f6b747dfc649ad186"}, + {file = "black-23.9.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:031e8c69f3d3b09e1aa471a926a1eeb0b9071f80b17689a655f7885ac9325a6f"}, + {file = "black-23.9.1-cp311-cp311-win_amd64.whl", hash = "sha256:538efb451cd50f43aba394e9ec7ad55a37598faae3348d723b59ea8e91616300"}, + {file = "black-23.9.1-cp38-cp38-macosx_10_16_arm64.whl", hash = "sha256:638619a559280de0c2aa4d76f504891c9860bb8fa214267358f0a20f27c12948"}, + {file = "black-23.9.1-cp38-cp38-macosx_10_16_universal2.whl", hash = "sha256:a732b82747235e0542c03bf352c126052c0fbc458d8a239a94701175b17d4855"}, + {file = "black-23.9.1-cp38-cp38-macosx_10_16_x86_64.whl", hash = "sha256:cf3a4d00e4cdb6734b64bf23cd4341421e8953615cba6b3670453737a72ec204"}, + {file = "black-23.9.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cf99f3de8b3273a8317681d8194ea222f10e0133a24a7548c73ce44ea1679377"}, + {file = "black-23.9.1-cp38-cp38-win_amd64.whl", hash = "sha256:14f04c990259576acd093871e7e9b14918eb28f1866f91968ff5524293f9c573"}, + {file = "black-23.9.1-cp39-cp39-macosx_10_16_arm64.whl", hash = "sha256:c619f063c2d68f19b2d7270f4cf3192cb81c9ec5bc5ba02df91471d0b88c4c5c"}, + {file = "black-23.9.1-cp39-cp39-macosx_10_16_universal2.whl", hash = "sha256:6a3b50e4b93f43b34a9d3ef00d9b6728b4a722c997c99ab09102fd5efdb88325"}, + {file = "black-23.9.1-cp39-cp39-macosx_10_16_x86_64.whl", hash = "sha256:c46767e8df1b7beefb0899c4a95fb43058fa8500b6db144f4ff3ca38eb2f6393"}, + {file = "black-23.9.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:50254ebfa56aa46a9fdd5d651f9637485068a1adf42270148cd101cdf56e0ad9"}, + {file = "black-23.9.1-cp39-cp39-win_amd64.whl", hash = "sha256:403397c033adbc45c2bd41747da1f7fc7eaa44efbee256b53842470d4ac5a70f"}, + {file = "black-23.9.1-py3-none-any.whl", hash = "sha256:6ccd59584cc834b6d127628713e4b6b968e5f79572da66284532525a042549f9"}, + {file = "black-23.9.1.tar.gz", hash = "sha256:24b6b3ff5c6d9ea08a8888f6977eae858e1f340d7260cf56d70a49823236b62d"}, ] [package.dependencies] @@ -167,7 +174,7 @@ packaging = ">=22.0" pathspec = ">=0.9.0" platformdirs = ">=2" tomli = {version = ">=1.1.0", markers = "python_version < \"3.11\""} -typing-extensions = {version = ">=3.10.0.0", markers = "python_version < \"3.10\""} +typing-extensions = {version = ">=4.0.1", markers = "python_version < \"3.11\""} [package.extras] colorama = ["colorama (>=0.4.3)"] @@ -179,6 +186,7 @@ uvloop = ["uvloop (>=0.15.2)"] name = "certifi" version = "2023.7.22" description = "Python package for providing Mozilla's CA Bundle." +category = "main" optional = false python-versions = ">=3.6" files = [ @@ -190,6 +198,7 @@ files = [ name = "cffi" version = "1.15.1" description = "Foreign Function Interface for Python calling C code." +category = "main" optional = false python-versions = "*" files = [ @@ -266,6 +275,7 @@ pycparser = "*" name = "charset-normalizer" version = "3.2.0" description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." +category = "main" optional = false python-versions = ">=3.7.0" files = [ @@ -350,6 +360,7 @@ files = [ name = "click" version = "8.1.7" description = "Composable command line interface toolkit" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -364,6 +375,7 @@ colorama = {version = "*", markers = "platform_system == \"Windows\""} name = "colorama" version = "0.4.6" description = "Cross-platform colored terminal text." +category = "main" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7" files = [ @@ -375,6 +387,7 @@ files = [ name = "cryptography" version = "41.0.3" description = "cryptography is a package which provides cryptographic recipes and primitives to Python developers." +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -420,6 +433,7 @@ test-randomorder = ["pytest-randomly"] name = "cssselect" version = "1.2.0" description = "cssselect parses CSS3 Selectors and translates them to XPath 1.0" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -431,6 +445,7 @@ files = [ name = "dill" version = "0.3.7" description = "serialize all of Python" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -445,6 +460,7 @@ graph = ["objgraph (>=1.7.2)"] name = "exceptiongroup" version = "1.1.3" description = "Backport of PEP 654 (exception groups)" +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -459,6 +475,7 @@ test = ["pytest (>=6)"] name = "flake8" version = "5.0.4" description = "the modular source code checker: pep8 pyflakes and co" +category = "dev" optional = false python-versions = ">=3.6.1" files = [ @@ -475,6 +492,7 @@ pyflakes = ">=2.5.0,<2.6.0" name = "future" version = "0.18.3" description = "Clean single-source support for Python 3 and 2" +category = "main" optional = false python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" files = [ @@ -485,6 +503,7 @@ files = [ name = "ghp-import" version = "2.1.0" description = "Copy your docs directly to the gh-pages branch." +category = "dev" optional = false python-versions = "*" files = [ @@ -502,6 +521,7 @@ dev = ["flake8", "markdown", "twine", "wheel"] name = "gitdb" version = "4.0.10" description = "Git Object Database" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -516,6 +536,7 @@ smmap = ">=3.0.1,<6" name = "gitpython" version = "3.1.35" description = "GitPython is a Python library used to interact with Git repositories" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -528,13 +549,14 @@ gitdb = ">=4.0.1,<5" [[package]] name = "griffe" -version = "0.36.1" +version = "0.36.2" description = "Signatures for entire Python programs. Extract the structure, the frame, the skeleton of your project, to generate API documentation or find breaking changes in your API." +category = "dev" optional = false python-versions = ">=3.8" files = [ - {file = "griffe-0.36.1-py3-none-any.whl", hash = "sha256:859b653fcde0a0af0e841a0109bac2b63a2f683132ae1ec8dae5fa81e94617a0"}, - {file = "griffe-0.36.1.tar.gz", hash = "sha256:11df63f1c85f605c73e4485de70ec13784049695d228241b0b582364a20c0536"}, + {file = "griffe-0.36.2-py3-none-any.whl", hash = "sha256:ba71895a3f5f606b18dcd950e8a1f8e7332a37f90f24caeb002546593f2e0eee"}, + {file = "griffe-0.36.2.tar.gz", hash = "sha256:333ade7932bb9096781d83092602625dfbfe220e87a039d2801259a1bd41d1c2"}, ] [package.dependencies] @@ -544,6 +566,7 @@ colorama = ">=0.4" name = "h11" version = "0.14.0" description = "A pure-Python, bring-your-own-I/O implementation of HTTP/1.1" +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -555,6 +578,7 @@ files = [ name = "httpcore" version = "0.17.3" description = "A minimal low-level HTTP client." +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -566,16 +590,17 @@ files = [ anyio = ">=3.0,<5.0" certifi = "*" h11 = ">=0.13,<0.15" -sniffio = "==1.*" +sniffio = ">=1.0.0,<2.0.0" [package.extras] http2 = ["h2 (>=3,<5)"] -socks = ["socksio (==1.*)"] +socks = ["socksio (>=1.0.0,<2.0.0)"] [[package]] name = "httpx" version = "0.24.1" description = "The next generation HTTP client." +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -591,14 +616,15 @@ sniffio = "*" [package.extras] brotli = ["brotli", "brotlicffi"] -cli = ["click (==8.*)", "pygments (==2.*)", "rich (>=10,<14)"] +cli = ["click (>=8.0.0,<9.0.0)", "pygments (>=2.0.0,<3.0.0)", "rich (>=10,<14)"] http2 = ["h2 (>=3,<5)"] -socks = ["socksio (==1.*)"] +socks = ["socksio (>=1.0.0,<2.0.0)"] [[package]] name = "idna" version = "3.4" description = "Internationalized Domain Names in Applications (IDNA)" +category = "main" optional = false python-versions = ">=3.5" files = [ @@ -610,6 +636,7 @@ files = [ name = "importlib-metadata" version = "4.13.0" description = "Read metadata from Python packages" +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -629,6 +656,7 @@ testing = ["flake8 (<5)", "flufl.flake8", "importlib-resources (>=1.3)", "packag name = "iniconfig" version = "2.0.0" description = "brain-dead simple config-ini parsing" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -640,6 +668,7 @@ files = [ name = "invoke" version = "2.2.0" description = "Pythonic task execution" +category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -651,6 +680,7 @@ files = [ name = "isort" version = "5.12.0" description = "A Python utility / library to sort Python imports." +category = "dev" optional = false python-versions = ">=3.8.0" files = [ @@ -668,6 +698,7 @@ requirements-deprecated-finder = ["pip-api", "pipreqs"] name = "jinja2" version = "3.1.2" description = "A very fast and expressive template engine." +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -685,6 +716,7 @@ i18n = ["Babel (>=2.7)"] name = "junos-eznc" version = "2.6.7" description = "Junos 'EZ' automation for non-programmers" +category = "main" optional = false python-versions = ">=3.5, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" files = [ @@ -710,6 +742,7 @@ yamlordereddictloader = "*" name = "lazy-object-proxy" version = "1.9.0" description = "A fast and thorough lazy object proxy." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -755,6 +788,7 @@ files = [ name = "lxml" version = "4.9.3" description = "Powerful and Pythonic XML processing library combining libxml2/libxslt with the ElementTree API." +category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, != 3.4.*" files = [ @@ -862,6 +896,7 @@ source = ["Cython (>=0.29.35)"] name = "markdown" version = "3.4.4" description = "Python implementation of John Gruber's Markdown." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -880,6 +915,7 @@ testing = ["coverage", "pyyaml"] name = "markdown-it-py" version = "3.0.0" description = "Python port of markdown-it. Markdown parsing, done right!" +category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -904,6 +940,7 @@ testing = ["coverage", "pytest", "pytest-cov", "pytest-regressions"] name = "markdown2" version = "2.4.10" description = "A fast and complete Python implementation of Markdown" +category = "dev" optional = false python-versions = ">=3.5, <4" files = [ @@ -920,6 +957,7 @@ wavedrom = ["wavedrom"] name = "markupsafe" version = "2.1.3" description = "Safely add untrusted strings to HTML/XML markup." +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -943,16 +981,6 @@ files = [ {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:5bbe06f8eeafd38e5d0a4894ffec89378b6c6a625ff57e3028921f8ff59318ac"}, {file = "MarkupSafe-2.1.3-cp311-cp311-win32.whl", hash = "sha256:dd15ff04ffd7e05ffcb7fe79f1b98041b8ea30ae9234aed2a9168b5797c3effb"}, {file = "MarkupSafe-2.1.3-cp311-cp311-win_amd64.whl", hash = "sha256:134da1eca9ec0ae528110ccc9e48041e0828d79f24121a1a146161103c76e686"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:f698de3fd0c4e6972b92290a45bd9b1536bffe8c6759c62471efaa8acb4c37bc"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:aa57bd9cf8ae831a362185ee444e15a93ecb2e344c8e52e4d721ea3ab6ef1823"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffcc3f7c66b5f5b7931a5aa68fc9cecc51e685ef90282f4a82f0f5e9b704ad11"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:47d4f1c5f80fc62fdd7777d0d40a2e9dda0a05883ab11374334f6c4de38adffd"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1f67c7038d560d92149c060157d623c542173016c4babc0c1913cca0564b9939"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:9aad3c1755095ce347e26488214ef77e0485a3c34a50c5a5e2471dff60b9dd9c"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:14ff806850827afd6b07a5f32bd917fb7f45b046ba40c57abdb636674a8b559c"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8f9293864fe09b8149f0cc42ce56e3f0e54de883a9de90cd427f191c346eb2e1"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-win32.whl", hash = "sha256:715d3562f79d540f251b99ebd6d8baa547118974341db04f5ad06d5ea3eb8007"}, - {file = "MarkupSafe-2.1.3-cp312-cp312-win_amd64.whl", hash = "sha256:1b8dd8c3fd14349433c79fa8abeb573a55fc0fdd769133baac1f5e07abf54aeb"}, {file = "MarkupSafe-2.1.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:8e254ae696c88d98da6555f5ace2279cf7cd5b3f52be2b5cf97feafe883b58d2"}, {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cb0932dc158471523c9637e807d9bfb93e06a95cbf010f1a38b98623b929ef2b"}, {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9402b03f1a1b4dc4c19845e5c749e3ab82d5078d16a2a4c2cd2df62d57bb0707"}, @@ -989,6 +1017,7 @@ files = [ name = "mccabe" version = "0.7.0" description = "McCabe checker, plugin for flake8" +category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1000,6 +1029,7 @@ files = [ name = "mdurl" version = "0.1.2" description = "Markdown URL utilities" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1011,6 +1041,7 @@ files = [ name = "mergedeep" version = "1.3.4" description = "A deep merge function for 🐍." +category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1022,6 +1053,7 @@ files = [ name = "mkdocs" version = "1.5.2" description = "Project documentation with Markdown." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1053,6 +1085,7 @@ min-versions = ["babel (==2.9.0)", "click (==7.0)", "colorama (==0.4)", "ghp-imp name = "mkdocs-autorefs" version = "0.5.0" description = "Automatically link across pages in MkDocs." +category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1068,6 +1101,7 @@ mkdocs = ">=1.1" name = "mkdocs-material" version = "9.2.4" description = "Documentation that simply works" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1094,6 +1128,7 @@ requests = ">=2.26" name = "mkdocs-material-extensions" version = "1.1.1" description = "Extension pack for Python Markdown and MkDocs Material." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1105,6 +1140,7 @@ files = [ name = "mkdocs-version-annotations" version = "1.0.0" description = "MkDocs plugin to add custom admonitions for documenting version differences" +category = "dev" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1116,6 +1152,7 @@ files = [ name = "mkdocstrings" version = "0.22.0" description = "Automatic documentation from sources, for MkDocs." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1142,6 +1179,7 @@ python-legacy = ["mkdocstrings-python-legacy (>=0.2.1)"] name = "mkdocstrings-python" version = "1.5.2" description = "A Python handler for mkdocstrings." +category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1157,6 +1195,7 @@ mkdocstrings = ">=0.20" name = "mypy-extensions" version = "0.4.4" description = "Experimental type system extensions for programs checked with the mypy typechecker." +category = "main" optional = false python-versions = ">=2.7" files = [ @@ -1167,6 +1206,7 @@ files = [ name = "napalm" version = "4.1.0" description = "Network Automation and Programmability Abstraction Layer with Multivendor support" +category = "main" optional = false python-versions = "*" files = [ @@ -1199,6 +1239,7 @@ typing-extensions = ">=4.3.0" name = "ncclient" version = "0.6.13" description = "Python library for NETCONF clients" +category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" files = [ @@ -1215,6 +1256,7 @@ six = "*" name = "netaddr" version = "0.8.0" description = "A network address manipulation library for Python" +category = "main" optional = false python-versions = "*" files = [ @@ -1226,6 +1268,7 @@ files = [ name = "netmiko" version = "4.1.2" description = "Multi-vendor library to simplify legacy CLI connections to network devices" +category = "main" optional = false python-versions = "*" files = [ @@ -1247,6 +1290,7 @@ textfsm = "1.1.2" name = "netutils" version = "1.6.0" description = "Common helper functions useful in network automation." +category = "main" optional = false python-versions = ">=3.8,<4.0" files = [ @@ -1261,6 +1305,7 @@ optionals = ["jsonschema (>=4.17.3,<5.0.0)", "napalm (>=4.0.0,<5.0.0)"] name = "nornir" version = "3.3.0" description = "Pluggable multi-threaded framework with inventory management to help operate collections of devices" +category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1281,6 +1326,7 @@ docs = ["jupyter (>=1,<2)", "nbsphinx (>=0.8,<0.9)", "pygments (>=2,<3)", "sphin name = "nornir-jinja2" version = "0.2.0" description = "Jinja2 plugins for nornir" +category = "main" optional = false python-versions = ">=3.6,<4.0" files = [ @@ -1296,6 +1342,7 @@ nornir = ">=3,<4" name = "nornir-napalm" version = "0.4.0" description = "NAPALM's plugins for nornir" +category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1311,6 +1358,7 @@ nornir = ">=3,<4" name = "nornir-netmiko" version = "1.0.0" description = "Netmiko's plugins for Nornir" +category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1326,6 +1374,7 @@ textfsm = "1.1.2" name = "nornir-utils" version = "0.2.0" description = "Collection of plugins and functions for nornir that don't require external dependencies" +category = "main" optional = false python-versions = ">=3.6.2,<4.0.0" files = [ @@ -1341,6 +1390,7 @@ nornir = ">=3,<4" name = "ntc-templates" version = "3.5.0" description = "TextFSM Templates for Network Devices, and Python wrapper for TextFSM's CliTable." +category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1355,6 +1405,7 @@ textfsm = ">=1.1.0,<2.0.0" name = "packaging" version = "23.1" description = "Core utilities for Python packages" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1366,6 +1417,7 @@ files = [ name = "paginate" version = "0.5.6" description = "Divides large result sets into pages for easier browsing" +category = "dev" optional = false python-versions = "*" files = [ @@ -1376,6 +1428,7 @@ files = [ name = "paramiko" version = "3.3.1" description = "SSH2 protocol library" +category = "main" optional = false python-versions = ">=3.6" files = [ @@ -1397,6 +1450,7 @@ invoke = ["invoke (>=2.0)"] name = "pathspec" version = "0.11.2" description = "Utility library for gitignore style pattern matching of file paths." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1408,6 +1462,7 @@ files = [ name = "pbr" version = "5.11.1" description = "Python Build Reasonableness" +category = "dev" optional = false python-versions = ">=2.6" files = [ @@ -1419,6 +1474,7 @@ files = [ name = "platformdirs" version = "3.10.0" description = "A small Python package for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1434,6 +1490,7 @@ test = ["appdirs (==1.4.4)", "covdefaults (>=2.3)", "pytest (>=7.4)", "pytest-co name = "pluggy" version = "1.3.0" description = "plugin and hook calling mechanisms for python" +category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1449,6 +1506,7 @@ testing = ["pytest", "pytest-benchmark"] name = "pycodestyle" version = "2.9.1" description = "Python style guide checker" +category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1460,6 +1518,7 @@ files = [ name = "pycparser" version = "2.21" description = "C parser in Python" +category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" files = [ @@ -1471,6 +1530,7 @@ files = [ name = "pydantic" version = "1.10.12" description = "Data validation and settings management using python type hints" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1524,6 +1584,7 @@ email = ["email-validator (>=1.0.3)"] name = "pydocstyle" version = "6.3.0" description = "Python docstring style checker" +category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1541,6 +1602,7 @@ toml = ["tomli (>=1.2.3)"] name = "pyeapi" version = "1.0.2" description = "Python Client for eAPI" +category = "main" optional = false python-versions = "*" files = [ @@ -1558,6 +1620,7 @@ test = ["coverage", "mock"] name = "pyflakes" version = "2.5.0" description = "passive checker of Python programs" +category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1569,6 +1632,7 @@ files = [ name = "pygments" version = "2.16.1" description = "Pygments is a syntax highlighting package written in Python." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1583,6 +1647,7 @@ plugins = ["importlib-metadata"] name = "pylint" version = "2.17.5" description = "python code static checker" +category = "dev" optional = false python-versions = ">=3.7.2" files = [ @@ -1612,6 +1677,7 @@ testutils = ["gitpython (>3)"] name = "pymdown-extensions" version = "10.3" description = "Extension pack for Python Markdown." +category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1630,6 +1696,7 @@ extra = ["pygments (>=2.12)"] name = "pynacl" version = "1.5.0" description = "Python binding to the Networking and Cryptography (NaCl) library" +category = "main" optional = false python-versions = ">=3.6" files = [ @@ -1654,13 +1721,14 @@ tests = ["hypothesis (>=3.27.0)", "pytest (>=3.2.1,!=3.3.0)"] [[package]] name = "pynautobot" -version = "1.5.0" +version = "2.0.0rc2" description = "Nautobot API client library" +category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ - {file = "pynautobot-1.5.0-py3-none-any.whl", hash = "sha256:aa5bdf18148d82715b26e1a7abf0796bb28da05fece3d206b6f42749d2f466b1"}, - {file = "pynautobot-1.5.0.tar.gz", hash = "sha256:50ac1e12f377ce2f1d156056e9ec3333c8a74bf6269e145889606da92b8752b4"}, + {file = "pynautobot-2.0.0rc2-py3-none-any.whl", hash = "sha256:2f3f5ece11be8b897524428d0b352985302c0b6528b33ffab24573f148b02c3b"}, + {file = "pynautobot-2.0.0rc2.tar.gz", hash = "sha256:252b9e0a9c7bd6782991ec467d65f4dfe1d8f5118c926f9c53be8156e7b4a2be"}, ] [package.dependencies] @@ -1671,6 +1739,7 @@ urllib3 = ">=1.21.1,<1.27" name = "pyparsing" version = "3.1.1" description = "pyparsing module - Classes and methods to define and execute parsing grammars" +category = "main" optional = false python-versions = ">=3.6.8" files = [ @@ -1685,6 +1754,7 @@ diagrams = ["jinja2", "railroad-diagrams"] name = "pyquery" version = "2.0.0" description = "A jquery-like library for python" +category = "dev" optional = false python-versions = "*" files = [ @@ -1703,6 +1773,7 @@ test = ["pytest", "pytest-cov", "requests", "webob", "webtest"] name = "pyserial" version = "3.5" description = "Python Serial Port Extension" +category = "main" optional = false python-versions = "*" files = [ @@ -1717,6 +1788,7 @@ cp2110 = ["hidapi"] name = "pytest" version = "7.4.2" description = "pytest: simple powerful testing with Python" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1739,6 +1811,7 @@ testing = ["argcomplete", "attrs (>=19.2.0)", "hypothesis (>=3.56)", "mock", "no name = "python-dateutil" version = "2.8.2" description = "Extensions to the standard Python datetime module" +category = "dev" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7" files = [ @@ -1753,6 +1826,7 @@ six = ">=1.5" name = "python-dotenv" version = "1.0.0" description = "Read key-value pairs from a .env file and set them as environment variables" +category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1767,6 +1841,7 @@ cli = ["click (>=5.0)"] name = "pytz" version = "2023.3.post1" description = "World timezone definitions, modern and historical" +category = "dev" optional = false python-versions = "*" files = [ @@ -1778,6 +1853,7 @@ files = [ name = "pyyaml" version = "6.0.1" description = "YAML parser and emitter for Python" +category = "main" optional = false python-versions = ">=3.6" files = [ @@ -1837,6 +1913,7 @@ files = [ name = "pyyaml-env-tag" version = "0.1" description = "A custom YAML tag for referencing environment variables in YAML files. " +category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1851,6 +1928,7 @@ pyyaml = "*" name = "readtime" version = "3.0.0" description = "Calculates the time some text takes the average human to read, based on Medium's read time forumula" +category = "dev" optional = false python-versions = "*" files = [ @@ -1866,6 +1944,7 @@ pyquery = ">=1.2" name = "regex" version = "2023.8.8" description = "Alternative regular expression module, to replace re." +category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1963,6 +2042,7 @@ files = [ name = "requests" version = "2.31.0" description = "Python HTTP for Humans." +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -1984,6 +2064,7 @@ use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] name = "requests-mock" version = "1.11.0" description = "Mock out responses from the requests package" +category = "dev" optional = false python-versions = "*" files = [ @@ -2003,6 +2084,7 @@ test = ["fixtures", "mock", "purl", "pytest", "requests-futures", "sphinx", "tes name = "rich" version = "13.5.2" description = "Render rich text, tables, progress bars, syntax highlighting, markdown and more to the terminal" +category = "dev" optional = false python-versions = ">=3.7.0" files = [ @@ -2022,6 +2104,7 @@ jupyter = ["ipywidgets (>=7.5.1,<9)"] name = "routeros-api" version = "0.17.0" description = "Python API to RouterBoard devices produced by MikroTik." +category = "main" optional = true python-versions = "*" files = [ @@ -2036,6 +2119,7 @@ six = "*" name = "ruamel-yaml" version = "0.17.32" description = "ruamel.yaml is a YAML parser/emitter that supports roundtrip preservation of comments, seq/map flow style, and map key order" +category = "main" optional = false python-versions = ">=3" files = [ @@ -2054,6 +2138,7 @@ jinja2 = ["ruamel.yaml.jinja2 (>=0.2)"] name = "ruamel-yaml-clib" version = "0.2.7" description = "C version of reader, parser and emitter for ruamel.yaml derived from libyaml" +category = "main" optional = false python-versions = ">=3.5" files = [ @@ -2064,8 +2149,7 @@ files = [ {file = "ruamel.yaml.clib-0.2.7-cp310-cp310-win32.whl", hash = "sha256:763d65baa3b952479c4e972669f679fe490eee058d5aa85da483ebae2009d231"}, {file = "ruamel.yaml.clib-0.2.7-cp310-cp310-win_amd64.whl", hash = "sha256:d000f258cf42fec2b1bbf2863c61d7b8918d31ffee905da62dede869254d3b8a"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:045e0626baf1c52e5527bd5db361bc83180faaba2ff586e763d3d5982a876a9e"}, - {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-macosx_13_0_arm64.whl", hash = "sha256:1a6391a7cabb7641c32517539ca42cf84b87b667bad38b78d4d42dd23e957c81"}, - {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-manylinux2014_aarch64.whl", hash = "sha256:9c7617df90c1365638916b98cdd9be833d31d337dbcd722485597b43c4a215bf"}, + {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-macosx_12_6_arm64.whl", hash = "sha256:721bc4ba4525f53f6a611ec0967bdcee61b31df5a56801281027a3a6d1c2daf5"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:41d0f1fa4c6830176eef5b276af04c89320ea616655d01327d5ce65e50575c94"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-win32.whl", hash = "sha256:f6d3d39611ac2e4f62c3128a9eed45f19a6608670c5a2f4f07f24e8de3441d38"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-win_amd64.whl", hash = "sha256:da538167284de58a52109a9b89b8f6a53ff8437dd6dc26d33b57bf6699153122"}, @@ -2100,6 +2184,7 @@ files = [ name = "scp" version = "0.14.5" description = "scp module for paramiko" +category = "main" optional = false python-versions = "*" files = [ @@ -2114,6 +2199,7 @@ paramiko = "*" name = "setuptools" version = "68.2.0" description = "Easily download, build, install, upgrade, and uninstall Python packages" +category = "main" optional = false python-versions = ">=3.8" files = [ @@ -2130,6 +2216,7 @@ testing-integration = ["build[virtualenv]", "filelock (>=3.4.0)", "jaraco.envs ( name = "six" version = "1.16.0" description = "Python 2 and 3 compatibility utilities" +category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" files = [ @@ -2141,6 +2228,7 @@ files = [ name = "smmap" version = "5.0.0" description = "A pure Python implementation of a sliding window memory map manager" +category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -2152,6 +2240,7 @@ files = [ name = "sniffio" version = "1.3.0" description = "Sniff out which async library your code is running under" +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2163,6 +2252,7 @@ files = [ name = "snowballstemmer" version = "2.2.0" description = "This package provides 29 stemmers for 28 languages generated from Snowball algorithms." +category = "dev" optional = false python-versions = "*" files = [ @@ -2174,6 +2264,7 @@ files = [ name = "soupsieve" version = "2.5" description = "A modern CSS selector implementation for Beautiful Soup." +category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2185,6 +2276,7 @@ files = [ name = "stevedore" version = "5.1.0" description = "Manage dynamic plugins for Python applications" +category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2199,6 +2291,7 @@ pbr = ">=2.0.0,<2.1.0 || >2.1.0" name = "tenacity" version = "8.2.3" description = "Retry code until it succeeds" +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2213,6 +2306,7 @@ doc = ["reno", "sphinx", "tornado (>=4.5)"] name = "textfsm" version = "1.1.2" description = "Python module for parsing semi-structured text into python tables." +category = "main" optional = false python-versions = "*" files = [ @@ -2228,6 +2322,7 @@ six = "*" name = "toml" version = "0.10.2" description = "Python Library for Tom's Obvious, Minimal Language" +category = "dev" optional = false python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" files = [ @@ -2239,6 +2334,7 @@ files = [ name = "tomli" version = "2.0.1" description = "A lil' TOML parser" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2250,6 +2346,7 @@ files = [ name = "tomlkit" version = "0.12.1" description = "Style preserving TOML library" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2261,6 +2358,7 @@ files = [ name = "transitions" version = "0.9.0" description = "A lightweight, object-oriented Python state machine implementation with many extensions." +category = "main" optional = false python-versions = "*" files = [ @@ -2279,6 +2377,7 @@ test = ["pytest"] name = "ttp" version = "0.9.5" description = "Template Text Parser" +category = "main" optional = false python-versions = ">=2.7,<4.0" files = [ @@ -2294,6 +2393,7 @@ full = ["cerberus (>=1.3.0,<1.4.0)", "deepdiff (>=5.8.0,<5.9.0)", "jinja2 (>=3.0 name = "ttp-templates" version = "0.3.5" description = "Template Text Parser Templates collections" +category = "main" optional = false python-versions = ">=3.6,<4.0" files = [ @@ -2311,6 +2411,7 @@ docs = ["mkdocs (==1.2.4)", "mkdocs-material (==7.2.2)", "mkdocs-material-extens name = "typing-extensions" version = "4.7.1" description = "Backported and Experimental Type Hints for Python 3.7+" +category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2322,6 +2423,7 @@ files = [ name = "urllib3" version = "1.26.16" description = "HTTP library with thread-safe connection pooling, file post, and more." +category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*" files = [ @@ -2338,6 +2440,7 @@ socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"] name = "watchdog" version = "3.0.0" description = "Filesystem events monitoring" +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2377,6 +2480,7 @@ watchmedo = ["PyYAML (>=3.10)"] name = "wrapt" version = "1.15.0" description = "Module for decorators, wrappers and monkey patching." +category = "dev" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" files = [ @@ -2461,6 +2565,7 @@ files = [ name = "yamllint" version = "1.32.0" description = "A linter for YAML files." +category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2479,6 +2584,7 @@ dev = ["doc8", "flake8", "flake8-import-order", "rstcheck[sphinx]", "sphinx"] name = "yamlordereddictloader" version = "0.4.0" description = "YAML loader and dump for PyYAML allowing to keep keys order." +category = "main" optional = false python-versions = "*" files = [ @@ -2492,6 +2598,7 @@ pyyaml = "*" name = "zipp" version = "3.16.2" description = "Backport of pathlib-compatible object wrapper for zip files" +category = "main" optional = false python-versions = ">=3.8" files = [ @@ -2509,4 +2616,4 @@ mikrotik-driver = ["routeros-api"] [metadata] lock-version = "2.0" python-versions = "^3.8" -content-hash = "728be2cb46ef3e08d1286afab8846f110831620aed33c3e0384d53b41bb960a2" +content-hash = "8dcc6f02b6e6bd9da7ae90da89bca3fbb378d3d14c5817c58bc7aa9c7d708bd0" diff --git a/pyproject.toml b/pyproject.toml index c7f56bb..c5040b5 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,6 +1,6 @@ [tool.poetry] name = "nornir-nautobot" -version = "3.0.0rc1" +version = "3.0.0" description = "Nornir Nautobot" authors = ["Network to Code, LLC "] readme = "README.md" @@ -17,7 +17,7 @@ classifiers = [ ] repository = "https://github.com/nautobot/nornir-nautobot" homepage = "https://nautobot.com" -documentation = "https://docs.nautobot.com/projects/nornir-nautobot/en/latest/" +documentation = "https://nornir-nautobot.readthedocs.io" [tool.poetry.urls] "Changelog" = "https://docs.nautobot.com/projects/nornir-nautobot/en/latest/dev/CHANGELOG/" @@ -31,7 +31,7 @@ nornir-utils = "^0" nornir-napalm = ">=0.4.0 <1.0.0" nornir-jinja2 = "^0.2.0" nornir-netmiko = "^1" -pynautobot = "^1.5.0" +pynautobot = "v2.0.0-rc.2" netutils = "^1.6.0" routeros-api = {version = "^0.17.0", optional = true} httpx = "^0.24.1" @@ -56,7 +56,6 @@ mkdocstrings = "0.22.0" mkdocstrings-python = "1.5.2" mkdocs-version-annotations = "1.0.0" - [tool.poetry.extras] mikrotik_driver = ["routeros-api"] diff --git a/tests/unit/conftest.py b/tests/unit/conftest.py index bb9f98b..f842886 100644 --- a/tests/unit/conftest.py +++ b/tests/unit/conftest.py @@ -1,13 +1,15 @@ """Used to setup fixtures to be used through tests.""" import pytest +import pynautobot from nornir_nautobot.plugins.inventory.nautobot import NautobotInventory @pytest.fixture() -def nornir_nautobot_class(): +def nornir_nautobot_class(monkeypatch): """Provide True to make tests pass. Returns: (bool): Returns True """ + monkeypatch.setattr(pynautobot.api, "version", "2.0") return NautobotInventory(nautobot_url="http://mock.example.com", nautobot_token="0123456789abcdef01234567890") diff --git a/tests/unit/mocks/00_api_root.json b/tests/unit/mocks/00_api_root.json new file mode 100644 index 0000000..5b728d6 --- /dev/null +++ b/tests/unit/mocks/00_api_root.json @@ -0,0 +1,12 @@ +{ + "circuits": "http://localhost:8000/api/circuits/", + "dcim": "http://localhost:8000/api/dcim/", + "extras": "http://localhost:8000/api/extras/", + "graphql": "http://localhost:8000/api/graphql/", + "ipam": "http://localhost:8000/api/ipam/", + "plugins": "http://localhost:8000/api/plugins/", + "status": "http://localhost:8000/api/status/", + "tenancy": "http://localhost:8000/api/tenancy/", + "users": "http://localhost:8000/api/users/", + "virtualization": "http://localhost:8000/api/virtualization/" +} diff --git a/tests/unit/mocks/01_get_devices.json b/tests/unit/mocks/01_get_devices.json index 0afc2c9..800a1b8 100644 --- a/tests/unit/mocks/01_get_devices.json +++ b/tests/unit/mocks/01_get_devices.json @@ -7,7 +7,7 @@ "id": 5, "url": "http://nautobot-demo.com/api/dcim/devices/5/", "name": "den-dist01", - "display_name": "den-dist01", + "display": "den-dist01", "device_type": { "id": 3, "url": "http://nautobot-demo.com/api/dcim/device-types/3/", @@ -19,9 +19,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 2, "url": "http://nautobot-demo.com/api/dcim/device-roles/2/", "name": "Network", @@ -31,9 +31,9 @@ "platform": null, "serial": "9ZYX8XZUMP0AF69YGO5Z5", "asset_tag": null, - "site": { + "location": { "id": 4, - "url": "http://nautobot-demo.com/api/dcim/sites/4/", + "url": "http://nautobot-demo.com/api/dcim/locations/4/", "name": "DEN", "slug": "den" }, @@ -63,7 +63,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [ { "id": 2, @@ -87,7 +87,7 @@ "id": 6, "url": "http://nautobot-demo.com/api/dcim/devices/6/", "name": "den-dist02", - "display_name": "den-dist02", + "display": "den-dist02", "device_type": { "id": 3, "url": "http://nautobot-demo.com/api/dcim/device-types/3/", @@ -99,9 +99,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 2, "url": "http://nautobot-demo.com/api/dcim/device-roles/2/", "name": "Network", @@ -112,13 +112,13 @@ "id": 2, "url": "http://nautobot-demo.com/api/dcim/platforms/2/", "name": "Cisco IOS", - "slug": "ios" + "network_driver": "ios" }, "serial": "94L2AKSPBXNMW5445NAVB", "asset_tag": null, - "site": { + "location": { "id": 4, - "url": "http://nautobot-demo.com/api/dcim/sites/4/", + "url": "http://nautobot-demo.com/api/dcim/locations/4/", "name": "DEN", "slug": "den" }, @@ -148,7 +148,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [ { "id": 2, @@ -169,7 +169,7 @@ "id": 4, "url": "http://nautobot-demo.com/api/dcim/devices/4/", "name": "den-wan01", - "display_name": "den-wan01", + "display": "den-wan01", "device_type": { "id": 3, "url": "http://nautobot-demo.com/api/dcim/device-types/3/", @@ -181,9 +181,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 3, "url": "http://nautobot-demo.com/api/dcim/device-roles/3/", "name": "Router", @@ -194,13 +194,13 @@ "id": 2, "url": "http://nautobot-demo.com/api/dcim/platforms/2/", "name": "Cisco IOS", - "slug": "ios" + "network_driver": "ios" }, "serial": "90Q1VEN47MPBMU2718KJ1", "asset_tag": null, - "site": { + "location": { "id": 4, - "url": "http://nautobot-demo.com/api/dcim/sites/4/", + "url": "http://nautobot-demo.com/api/dcim/locations/4/", "name": "DEN", "slug": "den" }, @@ -230,7 +230,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [ { "id": 2, diff --git a/tests/unit/mocks/02_get_device1.json b/tests/unit/mocks/02_get_device1.json index 9fe2efb..3f721c9 100644 --- a/tests/unit/mocks/02_get_device1.json +++ b/tests/unit/mocks/02_get_device1.json @@ -7,7 +7,7 @@ "id": 5, "url": "http://nautobot-demo.com/api/dcim/devices/5/", "name": "den-dist01", - "display_name": "den-dist01", + "display": "den-dist01", "device_type": { "id": 3, "url": "http://nautobot-demo.com/api/dcim/device-types/3/", @@ -19,9 +19,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 2, "url": "http://nautobot-demo.com/api/dcim/device-roles/2/", "name": "Network", @@ -32,13 +32,13 @@ "id": 2, "url": "http://nautobot-demo.com/api/dcim/platforms/2/", "name": "Cisco IOS", - "slug": "ios" + "network_driver": "ios" }, "serial": "9ZYX8XZUMP0AF69YGO5Z5", "asset_tag": null, - "site": { + "location": { "id": 4, - "url": "http://nautobot-demo.com/api/dcim/sites/4/", + "url": "http://nautobot-demo.com/api/dcim/locations/4/", "name": "DEN", "slug": "den" }, @@ -68,7 +68,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [ { "id": 2, diff --git a/tests/unit/mocks/03_get_device2.json b/tests/unit/mocks/03_get_device2.json index ddf19a0..401db5b 100644 --- a/tests/unit/mocks/03_get_device2.json +++ b/tests/unit/mocks/03_get_device2.json @@ -7,7 +7,7 @@ "id": 6, "url": "http://nautobot-demo.com/api/dcim/devices/6/", "name": "den-dist02", - "display_name": "den-dist02", + "display": "den-dist02", "device_type": { "id": 3, "url": "http://nautobot-demo.com/api/dcim/device-types/3/", @@ -19,9 +19,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 2, "url": "http://nautobot-demo.com/api/dcim/device-roles/2/", "name": "Network", @@ -32,13 +32,13 @@ "id": 2, "url": "http://nautobot-demo.com/api/dcim/platforms/2/", "name": "Cisco IOS", - "slug": "ios" + "network_driver": "ios" }, "serial": "94L2AKSPBXNMW5445NAVB", "asset_tag": null, - "site": { + "location": { "id": 4, - "url": "http://nautobot-demo.com/api/dcim/sites/4/", + "url": "http://nautobot-demo.com/api/dcim/locations/4/", "name": "DEN", "slug": "den" }, @@ -68,7 +68,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [ { "id": 2, diff --git a/tests/unit/mocks/04_get_device3.json b/tests/unit/mocks/04_get_device3.json index f3934b9..2e42072 100644 --- a/tests/unit/mocks/04_get_device3.json +++ b/tests/unit/mocks/04_get_device3.json @@ -7,7 +7,7 @@ "id": 4, "url": "http://nautobot-demo.com/api/dcim/devices/4/", "name": "den-wan01", - "display_name": "den-wan01", + "display": "den-wan01", "device_type": { "id": 3, "url": "http://nautobot-demo.com/api/dcim/device-types/3/", @@ -19,9 +19,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 3, "url": "http://nautobot-demo.com/api/dcim/device-roles/3/", "name": "Router", @@ -32,13 +32,13 @@ "id": 2, "url": "http://nautobot-demo.com/api/dcim/platforms/2/", "name": "Cisco IOS", - "slug": "ios" + "network_driver": "ios" }, "serial": "90Q1VEN47MPBMU2718KJ1", "asset_tag": null, - "site": { + "location": { "id": 4, - "url": "http://nautobot-demo.com/api/dcim/sites/4/", + "url": "http://nautobot-demo.com/api/dcim/locations/4/", "name": "DEN", "slug": "den" }, @@ -68,7 +68,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [ { "id": 2, diff --git a/tests/unit/mocks/05_get_sites_filtered.json b/tests/unit/mocks/05_get_locations_filtered.json similarity index 86% rename from tests/unit/mocks/05_get_sites_filtered.json rename to tests/unit/mocks/05_get_locations_filtered.json index 3dece99..a7b7727 100644 --- a/tests/unit/mocks/05_get_sites_filtered.json +++ b/tests/unit/mocks/05_get_locations_filtered.json @@ -7,7 +7,7 @@ "id": 2, "url": "http://dockerrunner:8000/api/dcim/devices/2/", "name": "msp-rtr01", - "display_name": "msp-rtr01", + "display": "msp-rtr01", "device_type": { "id": 3, "url": "http://dockerrunner:8000/api/dcim/device-types/3/", @@ -19,9 +19,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 3, "url": "http://dockerrunner:8000/api/dcim/device-roles/3/", "name": "Router", @@ -32,13 +32,13 @@ "id": 2, "url": "http://dockerrunner:8000/api/dcim/platforms/2/", "name": "Cisco IOS", - "slug": "ios" + "network_driver": "ios" }, "serial": "", "asset_tag": null, - "site": { + "location": { "id": 1, - "url": "http://dockerrunner:8000/api/dcim/sites/1/", + "url": "http://dockerrunner:8000/api/dcim/locations/1/", "name": "MSP", "slug": "msp" }, @@ -58,7 +58,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [], "custom_fields": {}, "config_context": {}, @@ -69,7 +69,7 @@ "id": 3, "url": "http://dockerrunner:8000/api/dcim/devices/3/", "name": "msp-rtr02", - "display_name": "msp-rtr02", + "display": "msp-rtr02", "device_type": { "id": 3, "url": "http://dockerrunner:8000/api/dcim/device-types/3/", @@ -81,9 +81,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 3, "url": "http://dockerrunner:8000/api/dcim/device-roles/3/", "name": "Router", @@ -94,13 +94,13 @@ "id": 2, "url": "http://dockerrunner:8000/api/dcim/platforms/2/", "name": "Cisco IOS", - "slug": "ios" + "network_driver": "ios" }, "serial": "", "asset_tag": null, - "site": { + "location": { "id": 1, - "url": "http://dockerrunner:8000/api/dcim/sites/1/", + "url": "http://dockerrunner:8000/api/dcim/locations/1/", "name": "MSP", "slug": "msp" }, @@ -120,7 +120,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [], "custom_fields": {}, "config_context": {}, diff --git a/tests/unit/mocks/06_get_device_msp-rtr01.json b/tests/unit/mocks/06_get_device_msp-rtr01.json index 59a127a..8f1f69f 100644 --- a/tests/unit/mocks/06_get_device_msp-rtr01.json +++ b/tests/unit/mocks/06_get_device_msp-rtr01.json @@ -7,7 +7,7 @@ "id": 2, "url": "http://dockerrunner:8000/api/dcim/devices/2/", "name": "msp-rtr01", - "display_name": "msp-rtr01", + "display": "msp-rtr01", "device_type": { "id": 3, "url": "http://dockerrunner:8000/api/dcim/device-types/3/", @@ -19,9 +19,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 3, "url": "http://dockerrunner:8000/api/dcim/device-roles/3/", "name": "Router", @@ -32,13 +32,13 @@ "id": 2, "url": "http://dockerrunner:8000/api/dcim/platforms/2/", "name": "Cisco IOS", - "slug": "ios" + "network_driver": "ios" }, "serial": "", "asset_tag": null, - "site": { + "location": { "id": 1, - "url": "http://dockerrunner:8000/api/dcim/sites/1/", + "url": "http://dockerrunner:8000/api/dcim/locations/1/", "name": "MSP", "slug": "msp" }, @@ -58,7 +58,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [], "custom_fields": {}, "config_context": {}, diff --git a/tests/unit/mocks/07_get_device_msp-rtr02.json b/tests/unit/mocks/07_get_device_msp-rtr02.json index b18f25a..78b690e 100644 --- a/tests/unit/mocks/07_get_device_msp-rtr02.json +++ b/tests/unit/mocks/07_get_device_msp-rtr02.json @@ -7,7 +7,7 @@ "id": 3, "url": "http://dockerrunner:8000/api/dcim/devices/3/", "name": "msp-rtr02", - "display_name": "msp-rtr02", + "display": "msp-rtr02", "device_type": { "id": 3, "url": "http://dockerrunner:8000/api/dcim/device-types/3/", @@ -19,9 +19,9 @@ }, "model": "IOSv", "slug": "iosv", - "display_name": "Cisco IOSv" + "display": "Cisco IOSv" }, - "device_role": { + "role": { "id": 3, "url": "http://dockerrunner:8000/api/dcim/device-roles/3/", "name": "Router", @@ -32,13 +32,13 @@ "id": 2, "url": "http://dockerrunner:8000/api/dcim/platforms/2/", "name": "Cisco IOS", - "slug": "ios" + "network_driver": "ios" }, "serial": "", "asset_tag": null, - "site": { + "location": { "id": 1, - "url": "http://dockerrunner:8000/api/dcim/sites/1/", + "url": "http://dockerrunner:8000/api/dcim/locations/1/", "name": "MSP", "slug": "msp" }, @@ -58,7 +58,7 @@ "vc_position": null, "vc_priority": null, "comments": "", - "local_context_data": null, + "local_config_context_data": null, "tags": [], "custom_fields": {}, "config_context": {}, diff --git a/tests/unit/test_nautobot_inventory.py b/tests/unit/test_nautobot_inventory.py index 7e07257..2e8f6ff 100644 --- a/tests/unit/test_nautobot_inventory.py +++ b/tests/unit/test_nautobot_inventory.py @@ -18,9 +18,14 @@ # GLOBALS HERE = path.abspath(path.dirname(__file__)) API_CALLS = [ + { + "fixture_path": f"{HERE}/mocks/00_api_root.json", + "url": "http://mock.example.com/api/", + "method": "get", + }, { "fixture_path": f"{HERE}/mocks/01_get_devices.json", - "url": "http://mock.example.com/api/dcim/devices/", + "url": "http://mock.example.com/api/dcim/devices/?depth=1", "method": "get", }, { @@ -39,8 +44,8 @@ "method": "get", }, { - "fixture_path": f"{HERE}/mocks/05_get_sites_filtered.json", - "url": "http://mock.example.com/api/dcim/devices/?site=msp", + "fixture_path": f"{HERE}/mocks/05_get_locations_filtered.json", + "url": "http://mock.example.com/api/dcim/devices/?depth=1&location=msp", "method": "get", }, { @@ -131,7 +136,7 @@ def test_filter_devices(): test_class = NautobotInventory( nautobot_url="http://mock.example.com", nautobot_token="0123456789abcdef01234567890", - filter_parameters={"site": "msp"}, + filter_parameters={"location": "msp"}, ) pynautobot_obj = pynautobot.api(url="http://mock.example.com", token="0123456789abcdef01234567890") expected_devices = [] @@ -158,7 +163,7 @@ def mock_nornir_task(task: Task): "options": { "nautobot_url": "http://mock.example.com", "nautobot_token": "0123456789abcdef01234567890", - "filter_parameters": {"site": "msp"}, + "filter_parameters": {"location": "msp"}, }, }, ) From 2680b339ab3b680a20553f461fa067ffff46e47e Mon Sep 17 00:00:00 2001 From: itdependsnetworks Date: Sat, 16 Sep 2023 20:16:56 -0400 Subject: [PATCH 03/13] Add tcp port configuration option --- docs/task/task.md | 57 +++++++++++++++++-- .../plugins/tasks/dispatcher/default.py | 13 ++++- 2 files changed, 63 insertions(+), 7 deletions(-) diff --git a/docs/task/task.md b/docs/task/task.md index af730a3..d082f89 100644 --- a/docs/task/task.md +++ b/docs/task/task.md @@ -8,12 +8,32 @@ The only task plugin currently is the "dispatcher" plugin. This plugin dispatche ## Dispatcher Sender +- If exists check `custom_dispatcher`, for network_driver (fail if not found) +- Check for framework & driver `f".dispatcher.{framework}{network_driver.titlecase()}"` +- Check for default, e.g. `f".dispatcher.{framework}default"` + +!!! info + Where `framework` is a library like `netmiko` or `napalm` and `network_driver` is the platform like `cisco_ios` or `arista_eos`. + ```python - try: - driver_task = getattr(driver_class, method) - except AttributeError: - logger.log_error(obj, f"Unable to locate the method {method} for {driver}") - raise NornirNautobotException() + if not kwargs.get("custom_dispatcher"): + custom_dispatcher = {} + logger.log_debug(f"Dispatcher process started for {task.host.name} ({task.host.platform.network_driver})") + + network_driver = task.host.platform.network_driver + network_driver_title = snake_to_title_case(network_driver) + custom_dispatcher_path = [custom_dispatcher.get(network_driver)] + framework_path = f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework}{network_driver_title}" + framework_default_path = f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework}Default" + + if custom_dispatcher.get(network_driver): + driver_class = import_string(custom_dispatcher_path) + checked_path = [custom_dispatcher_path] + elif import_string(framework_path): + driver_class = import_string(framework_path) + else: + driver_class = import_string(framework_default_path) + checked_path = [framework_path, framework_default_path] result = task.run(task=driver_task, *args, **kwargs) ``` @@ -46,3 +66,30 @@ task.run( The dispatcher expects the two primary objects, the `obj` and `logger` objects. The `obj` object should be a Device model instance. The logger should be `NornirLogger` instance which is imported from `nornir_nautobot.utils.logger`. This logging object optionally takes in a Nautobot Job object named nautobot_job. This is for use within the Nautobot platform Jobs. Each task will raise a `NornirNautobotException` for known issues. Using a custom processor, the user can predict when it was an well known error. + + +## Check Connectivity Configuration + +The check connectivity receiver will send attempt to tcp ping the port based on the following order or precedence. + +- Prefer `task.data["custom_field_data"]["tcp_port"]` if is a valid integer +- Prefer `task.data["config_context_data"]["tcp_port"]` if is a valid integer +- Prefer cls.tcp_port, which by default is defined in `DispatcherMixin` as 22 + +In this code you can see how it is set. + +```python +class DispatcherMixin: + + tcp_port = 22 + + @classmethod + def _get_tcp_port(cls, task, obj) -> str: + custom_field = task.data.get("custom_field_data", {}).get("tcp_port") + if isinstance(custom_field, int): + return custom_field + config_context = task.data.get("config_context_data", {}).get("tcp_port") + if isinstance(config_context, int): + return config_context + return cls.tcp_port +``` \ No newline at end of file diff --git a/nornir_nautobot/plugins/tasks/dispatcher/default.py b/nornir_nautobot/plugins/tasks/dispatcher/default.py index bdd0021..3273062 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/default.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/default.py @@ -37,6 +37,16 @@ class DispatcherMixin: def _get_hostname(cls, task: Task, obj) -> str: # pylint: disable=unused-argument return task.host.hostname + @classmethod + def _get_tcp_port(cls, task: Task) -> str: # pylint: disable=unused-argument + custom_field = task.data.get("custom_field_data", {}).get("tcp_port") + if isinstance(custom_field, int): + return custom_field + config_context = task.data.get("config_context_data", {}).get("tcp_port") + if isinstance(config_context, int): + return config_context + return cls.tcp_port + @classmethod def check_connectivity(cls, task: Task, logger, obj) -> Result: """Check the connectivity to a network device. @@ -61,8 +71,7 @@ def check_connectivity(cls, task: Task, logger, obj) -> Result: raise NornirNautobotException(error_msg) ip_addr = socket.gethostbyname(hostname) - # TODO: Allow port to be configurable - port = cls.tcp_port + port = cls._get_tcp_port(task) if not tcp_ping(ip_addr, port): error_msg = f"E1004: Could not connect to IP: {ip_addr} and port: {port}, preemptively failed." logger.log_error(error_msg, extra={"object": obj}) From aee48c7d383e4bf4d311fc11c25b02fe47ae6c82 Mon Sep 17 00:00:00 2001 From: pszulczewski Date: Mon, 18 Sep 2023 17:02:14 +0200 Subject: [PATCH 04/13] Update examples and docs. Replace PK with name and add a note. --- docs/inventory/inventory.md | 9 +++++---- examples/filter_location.py | 2 +- examples/filter_multiple_locations.py | 2 +- examples/filter_negate_location.py | 2 +- 4 files changed, 8 insertions(+), 7 deletions(-) diff --git a/docs/inventory/inventory.md b/docs/inventory/inventory.md index a88dce4..503c0fe 100644 --- a/docs/inventory/inventory.md +++ b/docs/inventory/inventory.md @@ -55,10 +55,11 @@ To filter from a single location you can use the filter location to get the devi > See this document for more information: > > https://docs.nautobot.com/projects/core/en/next/development/apps/api/platform-features/uniquely-identify-objects/ - +> +> *TLDR*: To query for a unique location, use the `id` object or a combination of fields (attributes) that are specific to the unique location. ```python -location = "db913e3b-cbe0-4463-addc-816ba6a20100" +location = "msp" my_nornir = InitNornir( inventory={ @@ -91,7 +92,7 @@ dict_keys(['msp-rtr01', 'msp-rtr02']) To search within multiple locations, pass a list of location Primary Keys. In the example below, it is the same as the previous example with a list passed in instead of a single string. ```python -location = ["db913e3b-cbe0-4463-addc-816ba6a20100", "6f09aa66-96be-4b4d-955a-9c98e488f0e6"] +location = ["msp", "grb"] my_nornir = InitNornir( inventory={ @@ -123,7 +124,7 @@ dict_keys(['grb-rtr01', 'msp-rtr01', 'msp-rtr02']) The negative filters also are supported. These are all of the filters possible. Here we will search for devices **not** at _MSP_: ```python -not_location = "db913e3b-cbe0-4463-addc-816ba6a20100" +not_location = "msp" my_nornir = InitNornir( inventory={ diff --git a/examples/filter_location.py b/examples/filter_location.py index 3eedf76..8a4c399 100644 --- a/examples/filter_location.py +++ b/examples/filter_location.py @@ -22,7 +22,7 @@ def hello_world(task: Task) -> Result: def main(): """Nornir testing.""" - location = "db913e3b-cbe0-4463-addc-816ba6a20100" + location = "msp" my_nornir = InitNornir( inventory={ diff --git a/examples/filter_multiple_locations.py b/examples/filter_multiple_locations.py index d230bc5..2228bf7 100644 --- a/examples/filter_multiple_locations.py +++ b/examples/filter_multiple_locations.py @@ -22,7 +22,7 @@ def hello_world(task: Task) -> Result: def main(): """Nornir testing.""" - location = ["db913e3b-cbe0-4463-addc-816ba6a20100", "6f09aa66-96be-4b4d-955a-9c98e488f0e6"] + location = ["msp", "grb"] my_nornir = InitNornir( inventory={ diff --git a/examples/filter_negate_location.py b/examples/filter_negate_location.py index 265e425..8704e15 100644 --- a/examples/filter_negate_location.py +++ b/examples/filter_negate_location.py @@ -22,7 +22,7 @@ def hello_world(task: Task) -> Result: def main(): """Nornir testing.""" - not_location = "db913e3b-cbe0-4463-addc-816ba6a20100" + not_location = "msp" my_nornir = InitNornir( inventory={ From cba65462552a125021f9e590153498b2ac518055 Mon Sep 17 00:00:00 2001 From: itdependsnetworks Date: Thu, 21 Sep 2023 00:40:04 -0400 Subject: [PATCH 05/13] Update logger --- docs/task/task.md | 4 +- .../plugins/tasks/dispatcher/__init__.py | 6 +- .../plugins/tasks/dispatcher/default.py | 90 +++++++++---------- .../tasks/dispatcher/mikrotik_routeros.py | 30 +++---- .../tasks/dispatcher/ruckus_smartzone.py | 14 +-- nornir_nautobot/utils/logger.py | 46 ---------- 6 files changed, 73 insertions(+), 117 deletions(-) delete mode 100644 nornir_nautobot/utils/logger.py diff --git a/docs/task/task.md b/docs/task/task.md index d082f89..e5627a0 100644 --- a/docs/task/task.md +++ b/docs/task/task.md @@ -18,7 +18,7 @@ The only task plugin currently is the "dispatcher" plugin. This plugin dispatche ```python if not kwargs.get("custom_dispatcher"): custom_dispatcher = {} - logger.log_debug(f"Dispatcher process started for {task.host.name} ({task.host.platform.network_driver})") + logger.debug(f"Dispatcher process started for {task.host.name} ({task.host.platform.network_driver})") network_driver = task.host.platform.network_driver network_driver_title = snake_to_title_case(network_driver) @@ -63,6 +63,8 @@ task.run( ) ``` +TODO: 2.0 update the logger info + The dispatcher expects the two primary objects, the `obj` and `logger` objects. The `obj` object should be a Device model instance. The logger should be `NornirLogger` instance which is imported from `nornir_nautobot.utils.logger`. This logging object optionally takes in a Nautobot Job object named nautobot_job. This is for use within the Nautobot platform Jobs. Each task will raise a `NornirNautobotException` for known issues. Using a custom processor, the user can predict when it was an well known error. diff --git a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py index 72352f4..1ac7cd0 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py @@ -25,7 +25,7 @@ def dispatcher( # pylint: disable=too-many-arguments,too-many-locals """ if not kwargs.get("custom_dispatcher"): custom_dispatcher = {} - logger.log_debug(f"Dispatcher process started for {task.host.name} ({task.host.platform.network_driver})") + logger.debug(f"Dispatcher process started for {task.host.name} ({task.host.platform.network_driver})") network_driver = task.host.platform.network_driver network_driver_title = snake_to_title_case(network_driver) @@ -44,14 +44,14 @@ def dispatcher( # pylint: disable=too-many-arguments,too-many-locals if not driver_class: error_msg = f"E1001: Did not find a valid dispatcher in {checked_path}, preemptively failed." - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) try: driver_task = getattr(driver_class, method) except AttributeError: error_msg = f"E1002: Unable to locate the method {method} for {driver_class}, preemptively failed." - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) result = task.run(task=driver_task, logger=logger, obj=obj, *args, **kwargs) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/default.py b/nornir_nautobot/plugins/tasks/dispatcher/default.py index 3273062..c48a49f 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/default.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/default.py @@ -53,7 +53,7 @@ def check_connectivity(cls, task: Task, logger, obj) -> Result: Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. Returns: @@ -67,22 +67,22 @@ def check_connectivity(cls, task: Task, logger, obj) -> Result: error_msg = ( f"E1003: The hostname {hostname} did not have an IP nor was resolvable, preemptively failed." ) - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) ip_addr = socket.gethostbyname(hostname) port = cls._get_tcp_port(task) if not tcp_ping(ip_addr, port): error_msg = f"E1004: Could not connect to IP: {ip_addr} and port: {port}, preemptively failed." - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if not task.host.username: error_msg = "E1005: There was no username defined, preemptively failed." - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if not task.host.password: error_msg = "E1006: There was no password defined, preemptively failed." - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) return Result(host=task.host) @@ -95,7 +95,7 @@ def compliance_config( Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. features (dict): A dictionary describing the configurations required. backup_file (str): The file location of where the back configuration should be saved. @@ -107,19 +107,19 @@ def compliance_config( """ if not os.path.exists(backup_file): error_msg = f"E1007: Backup file Not Found at location: `{backup_file}`, preemptively failed." - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if not os.path.exists(intended_file): error_msg = f"E1008: Intended config file NOT Found at location: `{intended_file}`, preemptively failed." - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) try: feature_data = compliance(features, backup_file, intended_file, platform) except Exception as error: # pylint: disable=broad-except error_msg = f"E1009: UNKNOWN Failure of: {str(error)}" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) return Result(host=task.host, result={"feature_data": feature_data}) @@ -139,7 +139,7 @@ def generate_config( Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. jinja_template (str): The file location of the actual Jinja template. jinja_root_path (str): The file folder where the file will be saved to. @@ -164,26 +164,26 @@ def generate_config( error_msg = ( f"E1010: There was a jinja2.exceptions.UndefinedError error: ``{str(exc.result.exception)}``" ) - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) elif isinstance(exc.result.exception, jinja2.TemplateSyntaxError): error_msg = (f"E1011: There was a jinja2.TemplateSyntaxError error: ``{str(exc.result.exception)}``",) - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) elif isinstance(exc.result.exception, jinja2.TemplateNotFound): error_msg = f"E1012: There was an issue finding the template and a jinja2.TemplateNotFound error was raised: ``{str(exc.result.exception)}``" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) elif isinstance(exc.result.exception, jinja2.TemplateError): error_msg = f"E1013: There was an issue general Jinja error: ``{str(exc.result.exception)}``" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) error_msg = f"E1014: Failed with an unknown issue. `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) make_folder(os.path.dirname(output_file_location)) @@ -196,7 +196,7 @@ def _remove_lines(cls, logger, _running_config: str, remove_lines: list) -> str: """Removes lines in configuration as specified in Remove Lines list. Args: - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. _running_config (str): a device running configuration. remove_lines (list): A list of regex lines to remove configurations. @@ -205,7 +205,7 @@ def _remove_lines(cls, logger, _running_config: str, remove_lines: list) -> str: """ if not remove_lines: return _running_config - logger.log_debug("Removing lines from configuration based on `remove_lines` definition") + logger.debug("Removing lines from configuration based on `remove_lines` definition") return clean_config(_running_config, remove_lines) @classmethod @@ -213,7 +213,7 @@ def _substitute_lines(cls, logger, _running_config: str, substitute_lines: list) """Substitutes lines in configuration as specified in substitute Lines list. Args: - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. _running_config (str): a device running configuration. substitute_lines (list): A list of dictionaries with to remove and replace lines. @@ -222,7 +222,7 @@ def _substitute_lines(cls, logger, _running_config: str, substitute_lines: list) """ if not substitute_lines: return _running_config - logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") + logger.debug("Substitute lines from configuration based on `substitute_lines` definition") return sanitize_config(_running_config, substitute_lines) @classmethod @@ -230,7 +230,7 @@ def _save_file(cls, logger, backup_file: str, _running_config: str) -> None: """Saves Running Configuration to a specified file. Args: - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. _running_config (str): a device running configuration. backup_file (str): String representing backup file path. @@ -238,7 +238,7 @@ def _save_file(cls, logger, backup_file: str, _running_config: str) -> None: Result: Running Config is saved into backup file path. """ make_folder(os.path.dirname(backup_file)) - logger.log_debug(f"Saving Configuration to file: {backup_file}") + logger.debug(f"Saving Configuration to file: {backup_file}") with open(backup_file, "w", encoding="utf8") as filehandler: filehandler.write(_running_config) @@ -254,7 +254,7 @@ def get_config( Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. backup_file (str): The file location of where the back configuration should be saved. remove_lines (list): A list of regex lines to remove configurations. @@ -264,19 +264,19 @@ def get_config( Result: Nornir Result object with a dict as a result containing the running configuration { "config: } """ - logger.log_debug(f"Executing get_config for {task.host.name} on {task.host.platform}") + logger.debug(f"Executing get_config for {task.host.name} on {task.host.platform}") # TODO: Find standard napalm exceptions and account for them try: result = task.run(task=napalm_get, getters=["config"], retrieve="running") except NornirSubTaskError as exc: error_msg = f"E1015: `get_config` method failed with an unexpected issue: `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if result[0].failed: # TODO: investigate this, is there a better way to handle? recursive function? - logger.log_error( + logger.error( f"`get_config` nornir task failed with an unexpected issue: `{str(result.exception)}`", extra={"object": obj}, ) @@ -284,11 +284,11 @@ def get_config( running_config = result[0].result.get("config", {}).get("running", None) if remove_lines: - logger.log_debug("Removing lines from configuration based on `remove_lines` definition") + logger.debug("Removing lines from configuration based on `remove_lines` definition") running_config = clean_config(running_config, remove_lines) if substitute_lines: - logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") + logger.debug("Substitute lines from configuration based on `substitute_lines` definition") running_config = sanitize_config(running_config, substitute_lines) if backup_file: @@ -310,7 +310,7 @@ def replace_config( Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job_results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. config (str): The candidate config. @@ -322,7 +322,7 @@ def replace_config( Returns: Result: Nornir Result object with a dict as a result containing what changed and the result of the push. """ - logger.log_info(obj, "Config provision starting") + logger.info(obj, "Config provision starting") # Sending None to napalm_configure for revert_in will disable it, so we don't want a default value. revert_in = os.getenv("NORNIR_NAUTOBOT_REVERT_IN_SECONDS") if revert_in is not None: @@ -337,11 +337,11 @@ def replace_config( ) except NornirSubTaskError as exc: error_msg = f"E1015: Failed with an unknown issue. `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) - logger.log_info(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") - logger.log_info(obj, "Config provision ended") + logger.info(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") + logger.info(obj, "Config provision ended") return Result(host=task.host, result={"changed": push_result.changed, "result": push_result[0].result}) @classmethod @@ -356,7 +356,7 @@ def merge_config( Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job_results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. config (str): The config set. @@ -368,7 +368,7 @@ def merge_config( Returns: Result: Nornir Result object with a dict as a result containing what changed and the result of the push. """ - logger.log_info(obj, "Config merge starting") + logger.info(obj, "Config merge starting") # Sending None to napalm_configure for revert_in will disable it, so we don't want a default value. revert_in = os.getenv("NORNIR_NAUTOBOT_REVERT_IN_SECONDS") if revert_in is not None: @@ -383,11 +383,11 @@ def merge_config( ) except NornirSubTaskError as exc: error_msg = f"E1015: Failed with an unknown issue. `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) - logger.log_info(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") - logger.log_info(obj, "Config merge ended") + logger.info(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") + logger.info(obj, "Config merge ended") return Result(host=task.host, result={"changed": push_result.changed, "result": push_result[0].result}) @@ -404,7 +404,7 @@ def get_config( Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. remove_lines (list): A list of regex lines to remove configurations. substitute_lines (list): A list of dictionaries with to remove and replace lines. @@ -413,7 +413,7 @@ def get_config( Result: Nornir Result object with a dict as a result containing the running configuration { "config: } """ - logger.log_debug(f"Executing get_config for {task.host.name} on {task.host.platform}") + logger.debug(f"Executing get_config for {task.host.name} on {task.host.platform}") command = cls.config_command try: @@ -421,16 +421,16 @@ def get_config( except NornirSubTaskError as exc: if isinstance(exc.result.exception, NetmikoAuthenticationException): error_msg = f"E1017: Failed with an authentication issue: `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if isinstance(exc.result.exception, NetmikoTimeoutException): error_msg = f"E1018: Failed with a timeout issue. `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) error_msg = f"E1016: Failed with an unknown issue. `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if result[0].failed: @@ -441,14 +441,14 @@ def get_config( # Primarily seen in Cisco devices. if "ERROR: % Invalid input detected at" in running_config: error_msg = "E1019: Discovered `ERROR: % Invalid input detected at` in the output" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if remove_lines: - logger.log_debug("Removing lines from configuration based on `remove_lines` definition") + logger.debug("Removing lines from configuration based on `remove_lines` definition") running_config = clean_config(running_config, remove_lines) if substitute_lines: - logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") + logger.debug("Substitute lines from configuration based on `substitute_lines` definition") running_config = sanitize_config(running_config, substitute_lines) if backup_file: diff --git a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py b/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py index d15bc2b..9a31bd4 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py @@ -49,7 +49,7 @@ def get_config( # pylint: disable=R0913,R0914 Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. backup_file (str): The file location of where the back configuration should be saved. remove_lines (list): A list of regex lines to remove configurations. @@ -59,10 +59,10 @@ def get_config( # pylint: disable=R0913,R0914 Result: Nornir Result object with a dict as a result containing the running configuration { "config: } """ - logger.log_debug(f"Executing get_config for {task.host.name} on {task.host.platform}") + logger.debug(f"Executing get_config for {task.host.name} on {task.host.platform}") if not routeros_api: error_msg = "E1020: The `routeros_api` is not installed in this environment." - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) sslctx = ssl.create_default_context() @@ -80,7 +80,7 @@ def get_config( # pylint: disable=R0913,R0914 api = connection.get_api() except Exception as error: error_msg = f"E1021: `get_config` method failed with an unexpected issue: `{error}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) for endpoint in cls.config_command: try: @@ -88,17 +88,17 @@ def get_config( # pylint: disable=R0913,R0914 config_data[endpoint] = resource.get() except Exception as error: error_msg = f"E1022: `get_config` method failed with an unexpected issue: `{error}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) connection.disconnect() running_config = json.dumps(config_data, indent=4) if remove_lines: - logger.log_debug("Removing lines from configuration based on `remove_lines` definition") + logger.debug("Removing lines from configuration based on `remove_lines` definition") running_config = clean_config(running_config, remove_lines) if substitute_lines: - logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") + logger.debug("Substitute lines from configuration based on `substitute_lines` definition") running_config = sanitize_config(running_config, substitute_lines) make_folder(os.path.dirname(backup_file)) @@ -124,7 +124,7 @@ def get_config( # pylint: disable=R0913,R0914 Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. remove_lines (list): A list of regex lines to remove configurations. substitute_lines (list): A list of dictionaries with to remove and replace lines. @@ -134,22 +134,22 @@ def get_config( # pylint: disable=R0913,R0914 { "config: } """ task.host.platform = NETMIKO_DEVICE_TYPE - logger.log_debug(f"Analyzing Software Version for {task.host.name} on {task.host.platform}") + logger.debug(f"Analyzing Software Version for {task.host.name} on {task.host.platform}") try: result = task.run(task=netmiko_send_command, command_string=cls.version_command) except NornirSubTaskError as exc: if isinstance(exc.result.exception, NetmikoAuthenticationException): error_msg = f"E1017: Failed with an authentication issue: `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if isinstance(exc.result.exception, NetmikoTimeoutException): error_msg = f"E1018: Failed with a timeout issue. `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) error_msg = f"E1016: Failed with an unknown issue. `{exc.result.exception}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if result[0].failed: @@ -161,13 +161,13 @@ def get_config( # pylint: disable=R0913,R0914 if major_version > "6": command += " show-sensitive" - logger.log_debug(f"Found Mikrotik Router OS version {major_version}") - logger.log_debug(f"Executing get_config for {task.host.name} on {task.host.platform}") + logger.debug(f"Found Mikrotik Router OS version {major_version}") + logger.debug(f"Executing get_config for {task.host.name} on {task.host.platform}") try: result = task.run(task=netmiko_send_command, command_string=command) except NornirSubTaskError as exc: - logger.log_error(f"Failed with an unknown issue. `{exc.result.exception}`", extra={"object": obj}) + logger.error(f"Failed with an unknown issue. `{exc.result.exception}`", extra={"object": obj}) raise NornirNautobotException( # pylint: disable=W0707 f"Failed with an unknown issue. `{exc.result.exception}`" ) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py index e971951..6a2ecfe 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py @@ -72,7 +72,7 @@ def _api_auth(cls, obj, logger, session_params: tuple) -> dict: error_msg = ( f"E1023: `_api_auth` method failed with an unexpected issue: HTTP Error `{response.status_code}`" ) - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) return service_ticket @@ -110,7 +110,7 @@ def _build_urls( # pylint: disable=too-many-arguments,too-many-locals item_list = response.json().get("list") else: error_msg = f"E1024: `{uri}` endpoint failed with code: HTTP Error `{response.status_code}`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) uri_list = [f'{uri}/{item["id"]}' for item in item_list] @@ -118,7 +118,7 @@ def _build_urls( # pylint: disable=too-many-arguments,too-many-locals url_dict[uri_list_item] = f"{base_url}{uri_list_item}?serviceTicket={token}" else: error_msg = f"E1025: `{uri}` endpoint missing in simple endpoints list, schema invalid`" - logger.log_error(error_msg, extra={"object": obj}) + logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) return url_dict @@ -145,7 +145,7 @@ def get_config( # pylint: disable=R0913,R0914 Args: task (Task): Nornir Task. - logger (NornirLogger): Custom NornirLogger object to reflect job results (via Nautobot Jobs) and Python logger. + logger (logging.Logger): Logger that may be a Nautobot Jobs or Python logger. obj (Device): A Nautobot Device Django ORM object instance. backup_file (str): The file location of where the back configuration should be saved. remove_lines (list): A list of regex lines to remove configurations. @@ -155,7 +155,7 @@ def get_config( # pylint: disable=R0913,R0914 Result: Nornir Result object with a dict as a result containing the running configuration { "config: } """ - logger.log_debug(f"Executing get_config for {task.host.name} on {task.host.platform}") + logger.debug(f"Executing get_config for {task.host.name} on {task.host.platform}") _extras = {} if task.host.platform in AP_PLATFORM_LIST: _wlc_ip4 = obj.get_computed_field("wireless_controller") @@ -173,11 +173,11 @@ def get_config( # pylint: disable=R0913,R0914 running_config = json.dumps(config_data, indent=4) if remove_lines: - logger.log_debug("Removing lines from configuration based on `remove_lines` definition") + logger.debug("Removing lines from configuration based on `remove_lines` definition") running_config = clean_config(running_config, remove_lines) if substitute_lines: - logger.log_debug("Substitute lines from configuration based on `substitute_lines` definition") + logger.debug("Substitute lines from configuration based on `substitute_lines` definition") running_config = sanitize_config(running_config, substitute_lines) make_folder(os.path.dirname(backup_file)) diff --git a/nornir_nautobot/utils/logger.py b/nornir_nautobot/utils/logger.py deleted file mode 100644 index bf506a7..0000000 --- a/nornir_nautobot/utils/logger.py +++ /dev/null @@ -1,46 +0,0 @@ -"""Provide logging facility for Nautobot/Python usage.""" - -import logging - -from typing import Any - - -class NornirLogger: - """Similar to a mixin, to utilize Python logging and Jobs Result obj.""" - - def __init__(self, name: str, nautobot_job=None, debug: bool = False, job_result=None): - """Initialize the object.""" - self.job_result = job_result or nautobot_job - self.logger = logging.getLogger(name) - self.debug = debug - self.nautobot_job = nautobot_job or job_result - - def log_debug(self, message: str, extra: Any = None): - """Debug, does not take obj, and only logs to jobs result when in global debug mode.""" - if self.nautobot_job and self.debug: - self.nautobot_job.logging.debug(message, extra=extra) - self.logger.debug(message) - - def log_info(self, message: str, extra: Any = None): - """Log to Python logger and jogs results for info messages.""" - if self.nautobot_job: - self.nautobot_job.logging.info(message, extra=extra) - self.logger.info("%s | %s", str(extra), message) - - def log_warning(self, message: str, extra: Any = None): - """Log to Python logger and jogs results for warning messages.""" - if self.nautobot_job: - self.nautobot_job.logging.warning(message, extra=extra) - self.logger.warning("%s | %s", str(extra), message) - - def log_error(self, message: str, extra: Any = None): - """Log to Python logger and jogs results for error messages.""" - if self.nautobot_job: - self.nautobot_job.logging.error(message, extra=extra) - self.logger.error("%s | %s", str(extra), message) - - def log_critical(self, message: str, extra: Any = None): - """Log to Python logger and jogs results for critical messages.""" - if self.nautobot_job: - self.nautobot_job.logging.critical(message, extra=extra) - self.logger.critical("%s | %s", str(extra), message) From c3f3cd35848d85784a7fd96b864cb6f14b5b84bf Mon Sep 17 00:00:00 2001 From: itdependsnetworks Date: Wed, 27 Sep 2023 02:10:07 -0400 Subject: [PATCH 06/13] Fix dispatcher and a few missed log conversions --- nornir_nautobot/plugins/tasks/dispatcher/__init__.py | 8 ++++---- nornir_nautobot/plugins/tasks/dispatcher/default.py | 12 ++++++------ 2 files changed, 10 insertions(+), 10 deletions(-) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py index 1ac7cd0..7dafa46 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py @@ -25,13 +25,13 @@ def dispatcher( # pylint: disable=too-many-arguments,too-many-locals """ if not kwargs.get("custom_dispatcher"): custom_dispatcher = {} - logger.debug(f"Dispatcher process started for {task.host.name} ({task.host.platform.network_driver})") + logger.debug(f"Dispatcher process started for {task.host.name} ({task.host.platform})") - network_driver = task.host.platform.network_driver + network_driver = task.host.platform network_driver_title = snake_to_title_case(network_driver) custom_dispatcher_path = [custom_dispatcher.get(network_driver)] - framework_path = f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework}{network_driver_title}" - framework_default_path = f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework}Default" + framework_path = f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework.title()}{network_driver_title}" + framework_default_path = f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework.title()}Default" if custom_dispatcher.get(network_driver): driver_class = import_string(custom_dispatcher_path) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/default.py b/nornir_nautobot/plugins/tasks/dispatcher/default.py index c48a49f..1aa44a7 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/default.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/default.py @@ -322,7 +322,7 @@ def replace_config( Returns: Result: Nornir Result object with a dict as a result containing what changed and the result of the push. """ - logger.info(obj, "Config provision starting") + logger.info("Config provision starting", extra={"object": obj}) # Sending None to napalm_configure for revert_in will disable it, so we don't want a default value. revert_in = os.getenv("NORNIR_NAUTOBOT_REVERT_IN_SECONDS") if revert_in is not None: @@ -340,8 +340,8 @@ def replace_config( logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) - logger.info(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") - logger.info(obj, "Config provision ended") + logger.info(f"result: {push_result[0].result}, changed: {push_result.changed}", extra={"object": obj}) + logger.info("Config provision ended", extra={"object": obj}) return Result(host=task.host, result={"changed": push_result.changed, "result": push_result[0].result}) @classmethod @@ -368,7 +368,7 @@ def merge_config( Returns: Result: Nornir Result object with a dict as a result containing what changed and the result of the push. """ - logger.info(obj, "Config merge starting") + logger.info("Config merge starting", extra={"object": obj}) # Sending None to napalm_configure for revert_in will disable it, so we don't want a default value. revert_in = os.getenv("NORNIR_NAUTOBOT_REVERT_IN_SECONDS") if revert_in is not None: @@ -386,8 +386,8 @@ def merge_config( logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) - logger.info(obj, f"result: {push_result[0].result}, changed: {push_result.changed}") - logger.info(obj, "Config merge ended") + logger.info(f"result: {push_result[0].result}, changed: {push_result.changed}", extra={"object": obj}) + logger.info("Config merge ended", extra={"object": obj}) return Result(host=task.host, result={"changed": push_result.changed, "result": push_result[0].result}) From 3910ca52f32f5c42057f930dad556d5f037eca4f Mon Sep 17 00:00:00 2001 From: Glenn Matthews Date: Fri, 29 Sep 2023 16:51:50 -0400 Subject: [PATCH 07/13] Unpin pynautobot dependency and fix black (#124) --- .../plugins/tasks/dispatcher/__init__.py | 4 +- poetry.lock | 140 +----------------- pyproject.toml | 2 +- 3 files changed, 10 insertions(+), 136 deletions(-) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py index 7dafa46..f4aabb7 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py @@ -30,7 +30,9 @@ def dispatcher( # pylint: disable=too-many-arguments,too-many-locals network_driver = task.host.platform network_driver_title = snake_to_title_case(network_driver) custom_dispatcher_path = [custom_dispatcher.get(network_driver)] - framework_path = f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework.title()}{network_driver_title}" + framework_path = ( + f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework.title()}{network_driver_title}" + ) framework_default_path = f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework.title()}Default" if custom_dispatcher.get(network_driver): diff --git a/poetry.lock b/poetry.lock index 6c24985..fa5f669 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,10 +1,9 @@ -# This file is automatically @generated by Poetry 1.4.2 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.6.1 and should not be changed by hand. [[package]] name = "anyio" version = "4.0.0" description = "High level compatibility layer for multiple asynchronous event loop implementations" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -26,7 +25,6 @@ trio = ["trio (>=0.22)"] name = "astroid" version = "2.15.6" description = "An abstract syntax tree for Python with inference support." -category = "dev" optional = false python-versions = ">=3.7.2" files = [ @@ -46,7 +44,6 @@ wrapt = [ name = "babel" version = "2.12.1" description = "Internationalization utilities" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -61,7 +58,6 @@ pytz = {version = ">=2015.7", markers = "python_version < \"3.9\""} name = "bandit" version = "1.7.5" description = "Security oriented static analyser for python code." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -85,7 +81,6 @@ yaml = ["PyYAML"] name = "bcrypt" version = "4.0.1" description = "Modern password hashing for your software and your servers" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -120,7 +115,6 @@ typecheck = ["mypy"] name = "beautifulsoup4" version = "4.12.2" description = "Screen-scraping library" -category = "dev" optional = false python-versions = ">=3.6.0" files = [ @@ -139,7 +133,6 @@ lxml = ["lxml"] name = "black" version = "23.9.1" description = "The uncompromising code formatter." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -186,7 +179,6 @@ uvloop = ["uvloop (>=0.15.2)"] name = "certifi" version = "2023.7.22" description = "Python package for providing Mozilla's CA Bundle." -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -198,7 +190,6 @@ files = [ name = "cffi" version = "1.15.1" description = "Foreign Function Interface for Python calling C code." -category = "main" optional = false python-versions = "*" files = [ @@ -275,7 +266,6 @@ pycparser = "*" name = "charset-normalizer" version = "3.2.0" description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." -category = "main" optional = false python-versions = ">=3.7.0" files = [ @@ -360,7 +350,6 @@ files = [ name = "click" version = "8.1.7" description = "Composable command line interface toolkit" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -375,7 +364,6 @@ colorama = {version = "*", markers = "platform_system == \"Windows\""} name = "colorama" version = "0.4.6" description = "Cross-platform colored terminal text." -category = "main" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7" files = [ @@ -387,7 +375,6 @@ files = [ name = "cryptography" version = "41.0.3" description = "cryptography is a package which provides cryptographic recipes and primitives to Python developers." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -433,7 +420,6 @@ test-randomorder = ["pytest-randomly"] name = "cssselect" version = "1.2.0" description = "cssselect parses CSS3 Selectors and translates them to XPath 1.0" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -445,7 +431,6 @@ files = [ name = "dill" version = "0.3.7" description = "serialize all of Python" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -460,7 +445,6 @@ graph = ["objgraph (>=1.7.2)"] name = "exceptiongroup" version = "1.1.3" description = "Backport of PEP 654 (exception groups)" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -475,7 +459,6 @@ test = ["pytest (>=6)"] name = "flake8" version = "5.0.4" description = "the modular source code checker: pep8 pyflakes and co" -category = "dev" optional = false python-versions = ">=3.6.1" files = [ @@ -492,7 +475,6 @@ pyflakes = ">=2.5.0,<2.6.0" name = "future" version = "0.18.3" description = "Clean single-source support for Python 3 and 2" -category = "main" optional = false python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" files = [ @@ -503,7 +485,6 @@ files = [ name = "ghp-import" version = "2.1.0" description = "Copy your docs directly to the gh-pages branch." -category = "dev" optional = false python-versions = "*" files = [ @@ -521,7 +502,6 @@ dev = ["flake8", "markdown", "twine", "wheel"] name = "gitdb" version = "4.0.10" description = "Git Object Database" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -536,7 +516,6 @@ smmap = ">=3.0.1,<6" name = "gitpython" version = "3.1.35" description = "GitPython is a Python library used to interact with Git repositories" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -551,7 +530,6 @@ gitdb = ">=4.0.1,<5" name = "griffe" version = "0.36.2" description = "Signatures for entire Python programs. Extract the structure, the frame, the skeleton of your project, to generate API documentation or find breaking changes in your API." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -566,7 +544,6 @@ colorama = ">=0.4" name = "h11" version = "0.14.0" description = "A pure-Python, bring-your-own-I/O implementation of HTTP/1.1" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -578,7 +555,6 @@ files = [ name = "httpcore" version = "0.17.3" description = "A minimal low-level HTTP client." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -590,17 +566,16 @@ files = [ anyio = ">=3.0,<5.0" certifi = "*" h11 = ">=0.13,<0.15" -sniffio = ">=1.0.0,<2.0.0" +sniffio = "==1.*" [package.extras] http2 = ["h2 (>=3,<5)"] -socks = ["socksio (>=1.0.0,<2.0.0)"] +socks = ["socksio (==1.*)"] [[package]] name = "httpx" version = "0.24.1" description = "The next generation HTTP client." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -616,15 +591,14 @@ sniffio = "*" [package.extras] brotli = ["brotli", "brotlicffi"] -cli = ["click (>=8.0.0,<9.0.0)", "pygments (>=2.0.0,<3.0.0)", "rich (>=10,<14)"] +cli = ["click (==8.*)", "pygments (==2.*)", "rich (>=10,<14)"] http2 = ["h2 (>=3,<5)"] -socks = ["socksio (>=1.0.0,<2.0.0)"] +socks = ["socksio (==1.*)"] [[package]] name = "idna" version = "3.4" description = "Internationalized Domain Names in Applications (IDNA)" -category = "main" optional = false python-versions = ">=3.5" files = [ @@ -636,7 +610,6 @@ files = [ name = "importlib-metadata" version = "4.13.0" description = "Read metadata from Python packages" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -656,7 +629,6 @@ testing = ["flake8 (<5)", "flufl.flake8", "importlib-resources (>=1.3)", "packag name = "iniconfig" version = "2.0.0" description = "brain-dead simple config-ini parsing" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -668,7 +640,6 @@ files = [ name = "invoke" version = "2.2.0" description = "Pythonic task execution" -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -680,7 +651,6 @@ files = [ name = "isort" version = "5.12.0" description = "A Python utility / library to sort Python imports." -category = "dev" optional = false python-versions = ">=3.8.0" files = [ @@ -698,7 +668,6 @@ requirements-deprecated-finder = ["pip-api", "pipreqs"] name = "jinja2" version = "3.1.2" description = "A very fast and expressive template engine." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -716,7 +685,6 @@ i18n = ["Babel (>=2.7)"] name = "junos-eznc" version = "2.6.7" description = "Junos 'EZ' automation for non-programmers" -category = "main" optional = false python-versions = ">=3.5, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" files = [ @@ -742,7 +710,6 @@ yamlordereddictloader = "*" name = "lazy-object-proxy" version = "1.9.0" description = "A fast and thorough lazy object proxy." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -788,7 +755,6 @@ files = [ name = "lxml" version = "4.9.3" description = "Powerful and Pythonic XML processing library combining libxml2/libxslt with the ElementTree API." -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, != 3.4.*" files = [ @@ -896,7 +862,6 @@ source = ["Cython (>=0.29.35)"] name = "markdown" version = "3.4.4" description = "Python implementation of John Gruber's Markdown." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -915,7 +880,6 @@ testing = ["coverage", "pyyaml"] name = "markdown-it-py" version = "3.0.0" description = "Python port of markdown-it. Markdown parsing, done right!" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -940,7 +904,6 @@ testing = ["coverage", "pytest", "pytest-cov", "pytest-regressions"] name = "markdown2" version = "2.4.10" description = "A fast and complete Python implementation of Markdown" -category = "dev" optional = false python-versions = ">=3.5, <4" files = [ @@ -957,7 +920,6 @@ wavedrom = ["wavedrom"] name = "markupsafe" version = "2.1.3" description = "Safely add untrusted strings to HTML/XML markup." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -1017,7 +979,6 @@ files = [ name = "mccabe" version = "0.7.0" description = "McCabe checker, plugin for flake8" -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1029,7 +990,6 @@ files = [ name = "mdurl" version = "0.1.2" description = "Markdown URL utilities" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1041,7 +1001,6 @@ files = [ name = "mergedeep" version = "1.3.4" description = "A deep merge function for 🐍." -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1053,7 +1012,6 @@ files = [ name = "mkdocs" version = "1.5.2" description = "Project documentation with Markdown." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1085,7 +1043,6 @@ min-versions = ["babel (==2.9.0)", "click (==7.0)", "colorama (==0.4)", "ghp-imp name = "mkdocs-autorefs" version = "0.5.0" description = "Automatically link across pages in MkDocs." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1101,7 +1058,6 @@ mkdocs = ">=1.1" name = "mkdocs-material" version = "9.2.4" description = "Documentation that simply works" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1128,7 +1084,6 @@ requests = ">=2.26" name = "mkdocs-material-extensions" version = "1.1.1" description = "Extension pack for Python Markdown and MkDocs Material." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1140,7 +1095,6 @@ files = [ name = "mkdocs-version-annotations" version = "1.0.0" description = "MkDocs plugin to add custom admonitions for documenting version differences" -category = "dev" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1152,7 +1106,6 @@ files = [ name = "mkdocstrings" version = "0.22.0" description = "Automatic documentation from sources, for MkDocs." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1179,7 +1132,6 @@ python-legacy = ["mkdocstrings-python-legacy (>=0.2.1)"] name = "mkdocstrings-python" version = "1.5.2" description = "A Python handler for mkdocstrings." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1195,7 +1147,6 @@ mkdocstrings = ">=0.20" name = "mypy-extensions" version = "0.4.4" description = "Experimental type system extensions for programs checked with the mypy typechecker." -category = "main" optional = false python-versions = ">=2.7" files = [ @@ -1206,7 +1157,6 @@ files = [ name = "napalm" version = "4.1.0" description = "Network Automation and Programmability Abstraction Layer with Multivendor support" -category = "main" optional = false python-versions = "*" files = [ @@ -1239,7 +1189,6 @@ typing-extensions = ">=4.3.0" name = "ncclient" version = "0.6.13" description = "Python library for NETCONF clients" -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" files = [ @@ -1256,7 +1205,6 @@ six = "*" name = "netaddr" version = "0.8.0" description = "A network address manipulation library for Python" -category = "main" optional = false python-versions = "*" files = [ @@ -1268,7 +1216,6 @@ files = [ name = "netmiko" version = "4.1.2" description = "Multi-vendor library to simplify legacy CLI connections to network devices" -category = "main" optional = false python-versions = "*" files = [ @@ -1290,7 +1237,6 @@ textfsm = "1.1.2" name = "netutils" version = "1.6.0" description = "Common helper functions useful in network automation." -category = "main" optional = false python-versions = ">=3.8,<4.0" files = [ @@ -1305,7 +1251,6 @@ optionals = ["jsonschema (>=4.17.3,<5.0.0)", "napalm (>=4.0.0,<5.0.0)"] name = "nornir" version = "3.3.0" description = "Pluggable multi-threaded framework with inventory management to help operate collections of devices" -category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1326,7 +1271,6 @@ docs = ["jupyter (>=1,<2)", "nbsphinx (>=0.8,<0.9)", "pygments (>=2,<3)", "sphin name = "nornir-jinja2" version = "0.2.0" description = "Jinja2 plugins for nornir" -category = "main" optional = false python-versions = ">=3.6,<4.0" files = [ @@ -1342,7 +1286,6 @@ nornir = ">=3,<4" name = "nornir-napalm" version = "0.4.0" description = "NAPALM's plugins for nornir" -category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1358,7 +1301,6 @@ nornir = ">=3,<4" name = "nornir-netmiko" version = "1.0.0" description = "Netmiko's plugins for Nornir" -category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1374,7 +1316,6 @@ textfsm = "1.1.2" name = "nornir-utils" version = "0.2.0" description = "Collection of plugins and functions for nornir that don't require external dependencies" -category = "main" optional = false python-versions = ">=3.6.2,<4.0.0" files = [ @@ -1390,7 +1331,6 @@ nornir = ">=3,<4" name = "ntc-templates" version = "3.5.0" description = "TextFSM Templates for Network Devices, and Python wrapper for TextFSM's CliTable." -category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1405,7 +1345,6 @@ textfsm = ">=1.1.0,<2.0.0" name = "packaging" version = "23.1" description = "Core utilities for Python packages" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1417,7 +1356,6 @@ files = [ name = "paginate" version = "0.5.6" description = "Divides large result sets into pages for easier browsing" -category = "dev" optional = false python-versions = "*" files = [ @@ -1428,7 +1366,6 @@ files = [ name = "paramiko" version = "3.3.1" description = "SSH2 protocol library" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -1450,7 +1387,6 @@ invoke = ["invoke (>=2.0)"] name = "pathspec" version = "0.11.2" description = "Utility library for gitignore style pattern matching of file paths." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1462,7 +1398,6 @@ files = [ name = "pbr" version = "5.11.1" description = "Python Build Reasonableness" -category = "dev" optional = false python-versions = ">=2.6" files = [ @@ -1474,7 +1409,6 @@ files = [ name = "platformdirs" version = "3.10.0" description = "A small Python package for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1490,7 +1424,6 @@ test = ["appdirs (==1.4.4)", "covdefaults (>=2.3)", "pytest (>=7.4)", "pytest-co name = "pluggy" version = "1.3.0" description = "plugin and hook calling mechanisms for python" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1506,7 +1439,6 @@ testing = ["pytest", "pytest-benchmark"] name = "pycodestyle" version = "2.9.1" description = "Python style guide checker" -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1518,7 +1450,6 @@ files = [ name = "pycparser" version = "2.21" description = "C parser in Python" -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" files = [ @@ -1530,7 +1461,6 @@ files = [ name = "pydantic" version = "1.10.12" description = "Data validation and settings management using python type hints" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1584,7 +1514,6 @@ email = ["email-validator (>=1.0.3)"] name = "pydocstyle" version = "6.3.0" description = "Python docstring style checker" -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1602,7 +1531,6 @@ toml = ["tomli (>=1.2.3)"] name = "pyeapi" version = "1.0.2" description = "Python Client for eAPI" -category = "main" optional = false python-versions = "*" files = [ @@ -1620,7 +1548,6 @@ test = ["coverage", "mock"] name = "pyflakes" version = "2.5.0" description = "passive checker of Python programs" -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1632,7 +1559,6 @@ files = [ name = "pygments" version = "2.16.1" description = "Pygments is a syntax highlighting package written in Python." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1647,7 +1573,6 @@ plugins = ["importlib-metadata"] name = "pylint" version = "2.17.5" description = "python code static checker" -category = "dev" optional = false python-versions = ">=3.7.2" files = [ @@ -1677,7 +1602,6 @@ testutils = ["gitpython (>3)"] name = "pymdown-extensions" version = "10.3" description = "Extension pack for Python Markdown." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1696,7 +1620,6 @@ extra = ["pygments (>=2.12)"] name = "pynacl" version = "1.5.0" description = "Python binding to the Networking and Cryptography (NaCl) library" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -1723,7 +1646,6 @@ tests = ["hypothesis (>=3.27.0)", "pytest (>=3.2.1,!=3.3.0)"] name = "pynautobot" version = "2.0.0rc2" description = "Nautobot API client library" -category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1739,7 +1661,6 @@ urllib3 = ">=1.21.1,<1.27" name = "pyparsing" version = "3.1.1" description = "pyparsing module - Classes and methods to define and execute parsing grammars" -category = "main" optional = false python-versions = ">=3.6.8" files = [ @@ -1754,7 +1675,6 @@ diagrams = ["jinja2", "railroad-diagrams"] name = "pyquery" version = "2.0.0" description = "A jquery-like library for python" -category = "dev" optional = false python-versions = "*" files = [ @@ -1773,7 +1693,6 @@ test = ["pytest", "pytest-cov", "requests", "webob", "webtest"] name = "pyserial" version = "3.5" description = "Python Serial Port Extension" -category = "main" optional = false python-versions = "*" files = [ @@ -1788,7 +1707,6 @@ cp2110 = ["hidapi"] name = "pytest" version = "7.4.2" description = "pytest: simple powerful testing with Python" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1811,7 +1729,6 @@ testing = ["argcomplete", "attrs (>=19.2.0)", "hypothesis (>=3.56)", "mock", "no name = "python-dateutil" version = "2.8.2" description = "Extensions to the standard Python datetime module" -category = "dev" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7" files = [ @@ -1826,7 +1743,6 @@ six = ">=1.5" name = "python-dotenv" version = "1.0.0" description = "Read key-value pairs from a .env file and set them as environment variables" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1841,7 +1757,6 @@ cli = ["click (>=5.0)"] name = "pytz" version = "2023.3.post1" description = "World timezone definitions, modern and historical" -category = "dev" optional = false python-versions = "*" files = [ @@ -1853,7 +1768,6 @@ files = [ name = "pyyaml" version = "6.0.1" description = "YAML parser and emitter for Python" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -1862,7 +1776,6 @@ files = [ {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:69b023b2b4daa7548bcfbd4aa3da05b3a74b772db9e23b982788168117739938"}, {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:81e0b275a9ecc9c0c0c07b4b90ba548307583c125f54d5b6946cfee6360c733d"}, {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba336e390cd8e4d1739f42dfe9bb83a3cc2e80f567d8805e11b46f4a943f5515"}, - {file = "PyYAML-6.0.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:326c013efe8048858a6d312ddd31d56e468118ad4cdeda36c719bf5bb6192290"}, {file = "PyYAML-6.0.1-cp310-cp310-win32.whl", hash = "sha256:bd4af7373a854424dabd882decdc5579653d7868b8fb26dc7d0e99f823aa5924"}, {file = "PyYAML-6.0.1-cp310-cp310-win_amd64.whl", hash = "sha256:fd1592b3fdf65fff2ad0004b5e363300ef59ced41c2e6b3a99d4089fa8c5435d"}, {file = "PyYAML-6.0.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6965a7bc3cf88e5a1c3bd2e0b5c22f8d677dc88a455344035f03399034eb3007"}, @@ -1870,15 +1783,8 @@ files = [ {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:42f8152b8dbc4fe7d96729ec2b99c7097d656dc1213a3229ca5383f973a5ed6d"}, {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:062582fca9fabdd2c8b54a3ef1c978d786e0f6b3a1510e0ac93ef59e0ddae2bc"}, {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d2b04aac4d386b172d5b9692e2d2da8de7bfb6c387fa4f801fbf6fb2e6ba4673"}, - {file = "PyYAML-6.0.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:e7d73685e87afe9f3b36c799222440d6cf362062f78be1013661b00c5c6f678b"}, {file = "PyYAML-6.0.1-cp311-cp311-win32.whl", hash = "sha256:1635fd110e8d85d55237ab316b5b011de701ea0f29d07611174a1b42f1444741"}, {file = "PyYAML-6.0.1-cp311-cp311-win_amd64.whl", hash = "sha256:bf07ee2fef7014951eeb99f56f39c9bb4af143d8aa3c21b1677805985307da34"}, - {file = "PyYAML-6.0.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:855fb52b0dc35af121542a76b9a84f8d1cd886ea97c84703eaa6d88e37a2ad28"}, - {file = "PyYAML-6.0.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:40df9b996c2b73138957fe23a16a4f0ba614f4c0efce1e9406a184b6d07fa3a9"}, - {file = "PyYAML-6.0.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c22bec3fbe2524cde73d7ada88f6566758a8f7227bfbf93a408a9d86bcc12a0"}, - {file = "PyYAML-6.0.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8d4e9c88387b0f5c7d5f281e55304de64cf7f9c0021a3525bd3b1c542da3b0e4"}, - {file = "PyYAML-6.0.1-cp312-cp312-win32.whl", hash = "sha256:d483d2cdf104e7c9fa60c544d92981f12ad66a457afae824d146093b8c294c54"}, - {file = "PyYAML-6.0.1-cp312-cp312-win_amd64.whl", hash = "sha256:0d3304d8c0adc42be59c5f8a4d9e3d7379e6955ad754aa9d6ab7a398b59dd1df"}, {file = "PyYAML-6.0.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:50550eb667afee136e9a77d6dc71ae76a44df8b3e51e41b77f6de2932bfe0f47"}, {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1fe35611261b29bd1de0070f0b2f47cb6ff71fa6595c077e42bd0c419fa27b98"}, {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:704219a11b772aea0d8ecd7058d0082713c3562b4e271b849ad7dc4a5c90c13c"}, @@ -1895,7 +1801,6 @@ files = [ {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a0cd17c15d3bb3fa06978b4e8958dcdc6e0174ccea823003a106c7d4d7899ac5"}, {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:28c119d996beec18c05208a8bd78cbe4007878c6dd15091efb73a30e90539696"}, {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7e07cbde391ba96ab58e532ff4803f79c4129397514e1413a7dc761ccd755735"}, - {file = "PyYAML-6.0.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:49a183be227561de579b4a36efbb21b3eab9651dd81b1858589f796549873dd6"}, {file = "PyYAML-6.0.1-cp38-cp38-win32.whl", hash = "sha256:184c5108a2aca3c5b3d3bf9395d50893a7ab82a38004c8f61c258d4428e80206"}, {file = "PyYAML-6.0.1-cp38-cp38-win_amd64.whl", hash = "sha256:1e2722cc9fbb45d9b87631ac70924c11d3a401b2d7f410cc0e3bbf249f2dca62"}, {file = "PyYAML-6.0.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:9eb6caa9a297fc2c2fb8862bc5370d0303ddba53ba97e71f08023b6cd73d16a8"}, @@ -1903,7 +1808,6 @@ files = [ {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5773183b6446b2c99bb77e77595dd486303b4faab2b086e7b17bc6bef28865f6"}, {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b786eecbdf8499b9ca1d697215862083bd6d2a99965554781d0d8d1ad31e13a0"}, {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc1bf2925a1ecd43da378f4db9e4f799775d6367bdb94671027b73b393a7c42c"}, - {file = "PyYAML-6.0.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:04ac92ad1925b2cff1db0cfebffb6ffc43457495c9b3c39d3fcae417d7125dc5"}, {file = "PyYAML-6.0.1-cp39-cp39-win32.whl", hash = "sha256:faca3bdcf85b2fc05d06ff3fbc1f83e1391b3e724afa3feba7d13eeab355484c"}, {file = "PyYAML-6.0.1-cp39-cp39-win_amd64.whl", hash = "sha256:510c9deebc5c0225e8c96813043e62b680ba2f9c50a08d3724c7f28a747d1486"}, {file = "PyYAML-6.0.1.tar.gz", hash = "sha256:bfdf460b1736c775f2ba9f6a92bca30bc2095067b8a9d77876d1fad6cc3b4a43"}, @@ -1913,7 +1817,6 @@ files = [ name = "pyyaml-env-tag" version = "0.1" description = "A custom YAML tag for referencing environment variables in YAML files. " -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -1928,7 +1831,6 @@ pyyaml = "*" name = "readtime" version = "3.0.0" description = "Calculates the time some text takes the average human to read, based on Medium's read time forumula" -category = "dev" optional = false python-versions = "*" files = [ @@ -1944,7 +1846,6 @@ pyquery = ">=1.2" name = "regex" version = "2023.8.8" description = "Alternative regular expression module, to replace re." -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -2042,7 +1943,6 @@ files = [ name = "requests" version = "2.31.0" description = "Python HTTP for Humans." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2064,7 +1964,6 @@ use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] name = "requests-mock" version = "1.11.0" description = "Mock out responses from the requests package" -category = "dev" optional = false python-versions = "*" files = [ @@ -2084,7 +1983,6 @@ test = ["fixtures", "mock", "purl", "pytest", "requests-futures", "sphinx", "tes name = "rich" version = "13.5.2" description = "Render rich text, tables, progress bars, syntax highlighting, markdown and more to the terminal" -category = "dev" optional = false python-versions = ">=3.7.0" files = [ @@ -2104,7 +2002,6 @@ jupyter = ["ipywidgets (>=7.5.1,<9)"] name = "routeros-api" version = "0.17.0" description = "Python API to RouterBoard devices produced by MikroTik." -category = "main" optional = true python-versions = "*" files = [ @@ -2119,7 +2016,6 @@ six = "*" name = "ruamel-yaml" version = "0.17.32" description = "ruamel.yaml is a YAML parser/emitter that supports roundtrip preservation of comments, seq/map flow style, and map key order" -category = "main" optional = false python-versions = ">=3" files = [ @@ -2138,7 +2034,6 @@ jinja2 = ["ruamel.yaml.jinja2 (>=0.2)"] name = "ruamel-yaml-clib" version = "0.2.7" description = "C version of reader, parser and emitter for ruamel.yaml derived from libyaml" -category = "main" optional = false python-versions = ">=3.5" files = [ @@ -2184,7 +2079,6 @@ files = [ name = "scp" version = "0.14.5" description = "scp module for paramiko" -category = "main" optional = false python-versions = "*" files = [ @@ -2199,7 +2093,6 @@ paramiko = "*" name = "setuptools" version = "68.2.0" description = "Easily download, build, install, upgrade, and uninstall Python packages" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -2216,7 +2109,6 @@ testing-integration = ["build[virtualenv]", "filelock (>=3.4.0)", "jaraco.envs ( name = "six" version = "1.16.0" description = "Python 2 and 3 compatibility utilities" -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" files = [ @@ -2228,7 +2120,6 @@ files = [ name = "smmap" version = "5.0.0" description = "A pure Python implementation of a sliding window memory map manager" -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -2240,7 +2131,6 @@ files = [ name = "sniffio" version = "1.3.0" description = "Sniff out which async library your code is running under" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2252,7 +2142,6 @@ files = [ name = "snowballstemmer" version = "2.2.0" description = "This package provides 29 stemmers for 28 languages generated from Snowball algorithms." -category = "dev" optional = false python-versions = "*" files = [ @@ -2264,7 +2153,6 @@ files = [ name = "soupsieve" version = "2.5" description = "A modern CSS selector implementation for Beautiful Soup." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2276,7 +2164,6 @@ files = [ name = "stevedore" version = "5.1.0" description = "Manage dynamic plugins for Python applications" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2291,7 +2178,6 @@ pbr = ">=2.0.0,<2.1.0 || >2.1.0" name = "tenacity" version = "8.2.3" description = "Retry code until it succeeds" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2306,7 +2192,6 @@ doc = ["reno", "sphinx", "tornado (>=4.5)"] name = "textfsm" version = "1.1.2" description = "Python module for parsing semi-structured text into python tables." -category = "main" optional = false python-versions = "*" files = [ @@ -2322,7 +2207,6 @@ six = "*" name = "toml" version = "0.10.2" description = "Python Library for Tom's Obvious, Minimal Language" -category = "dev" optional = false python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" files = [ @@ -2334,7 +2218,6 @@ files = [ name = "tomli" version = "2.0.1" description = "A lil' TOML parser" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2346,7 +2229,6 @@ files = [ name = "tomlkit" version = "0.12.1" description = "Style preserving TOML library" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2358,7 +2240,6 @@ files = [ name = "transitions" version = "0.9.0" description = "A lightweight, object-oriented Python state machine implementation with many extensions." -category = "main" optional = false python-versions = "*" files = [ @@ -2377,7 +2258,6 @@ test = ["pytest"] name = "ttp" version = "0.9.5" description = "Template Text Parser" -category = "main" optional = false python-versions = ">=2.7,<4.0" files = [ @@ -2393,7 +2273,6 @@ full = ["cerberus (>=1.3.0,<1.4.0)", "deepdiff (>=5.8.0,<5.9.0)", "jinja2 (>=3.0 name = "ttp-templates" version = "0.3.5" description = "Template Text Parser Templates collections" -category = "main" optional = false python-versions = ">=3.6,<4.0" files = [ @@ -2411,7 +2290,6 @@ docs = ["mkdocs (==1.2.4)", "mkdocs-material (==7.2.2)", "mkdocs-material-extens name = "typing-extensions" version = "4.7.1" description = "Backported and Experimental Type Hints for Python 3.7+" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2423,7 +2301,6 @@ files = [ name = "urllib3" version = "1.26.16" description = "HTTP library with thread-safe connection pooling, file post, and more." -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*" files = [ @@ -2440,7 +2317,6 @@ socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"] name = "watchdog" version = "3.0.0" description = "Filesystem events monitoring" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2480,7 +2356,6 @@ watchmedo = ["PyYAML (>=3.10)"] name = "wrapt" version = "1.15.0" description = "Module for decorators, wrappers and monkey patching." -category = "dev" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" files = [ @@ -2565,7 +2440,6 @@ files = [ name = "yamllint" version = "1.32.0" description = "A linter for YAML files." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2584,7 +2458,6 @@ dev = ["doc8", "flake8", "flake8-import-order", "rstcheck[sphinx]", "sphinx"] name = "yamlordereddictloader" version = "0.4.0" description = "YAML loader and dump for PyYAML allowing to keep keys order." -category = "main" optional = false python-versions = "*" files = [ @@ -2598,7 +2471,6 @@ pyyaml = "*" name = "zipp" version = "3.16.2" description = "Backport of pathlib-compatible object wrapper for zip files" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -2616,4 +2488,4 @@ mikrotik-driver = ["routeros-api"] [metadata] lock-version = "2.0" python-versions = "^3.8" -content-hash = "8dcc6f02b6e6bd9da7ae90da89bca3fbb378d3d14c5817c58bc7aa9c7d708bd0" +content-hash = "84f52fdf8d8c3aade58b2f52afb2407f601ff1132af64468e39e96c43e98936c" diff --git a/pyproject.toml b/pyproject.toml index c5040b5..38ccdb6 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -31,7 +31,7 @@ nornir-utils = "^0" nornir-napalm = ">=0.4.0 <1.0.0" nornir-jinja2 = "^0.2.0" nornir-netmiko = "^1" -pynautobot = "v2.0.0-rc.2" +pynautobot = ">=2.0.0rc2" netutils = "^1.6.0" routeros-api = {version = "^0.17.0", optional = true} httpx = "^0.24.1" From 0ba7b7d6172ad990a5318e08a300b0e9c0ecbca8 Mon Sep 17 00:00:00 2001 From: Jeff Kala <48843785+jeffkala@users.noreply.github.com> Date: Fri, 29 Sep 2023 14:58:00 -0600 Subject: [PATCH 08/13] fix tcp ping port (#125) --- .../plugins/tasks/dispatcher/default.py | 72 +++++++++++++------ 1 file changed, 52 insertions(+), 20 deletions(-) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/default.py b/nornir_nautobot/plugins/tasks/dispatcher/default.py index 1aa44a7..7108481 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/default.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/default.py @@ -7,9 +7,7 @@ from typing import Optional import jinja2 - from netmiko import NetmikoAuthenticationException, NetmikoTimeoutException - from netutils.config.clean import clean_config, sanitize_config from netutils.config.compliance import compliance from netutils.dns import is_fqdn_resolvable @@ -18,12 +16,10 @@ from nornir.core.exceptions import NornirSubTaskError from nornir.core.task import Result, Task from nornir_jinja2.plugins.tasks import template_file -from nornir_napalm.plugins.tasks import napalm_get, napalm_configure -from nornir_netmiko.tasks import netmiko_send_command - +from nornir_napalm.plugins.tasks import napalm_configure, napalm_get from nornir_nautobot.exceptions import NornirNautobotException from nornir_nautobot.utils.helpers import make_folder - +from nornir_netmiko.tasks import netmiko_send_command _logger = logging.getLogger(__name__) @@ -34,15 +30,15 @@ class DispatcherMixin: tcp_port = 22 @classmethod - def _get_hostname(cls, task: Task, obj) -> str: # pylint: disable=unused-argument + def _get_hostname(cls, task: Task) -> str: return task.host.hostname @classmethod - def _get_tcp_port(cls, task: Task) -> str: # pylint: disable=unused-argument - custom_field = task.data.get("custom_field_data", {}).get("tcp_port") + def _get_tcp_port(cls, obj) -> str: + custom_field = obj.cf.get("tcp_port") if isinstance(custom_field, int): return custom_field - config_context = task.data.get("config_context_data", {}).get("tcp_port") + config_context = obj.get_config_context().get("tcp_port") if isinstance(config_context, int): return config_context return cls.tcp_port @@ -71,9 +67,14 @@ def check_connectivity(cls, task: Task, logger, obj) -> Result: raise NornirNautobotException(error_msg) ip_addr = socket.gethostbyname(hostname) - port = cls._get_tcp_port(task) - if not tcp_ping(ip_addr, port): - error_msg = f"E1004: Could not connect to IP: {ip_addr} and port: {port}, preemptively failed." + port = cls._get_tcp_port(task, obj) + # TODO: Remove after fixing tcp_ping in netutils + try: + _tcp_ping = tcp_ping(ip_addr, port) + except socket.error: + _tcp_ping = False + if not _tcp_ping: + error_msg = f"E1004: Could not connect to IP: `{ip_addr}` and port: `{port}`, preemptively failed." logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if not task.host.username: @@ -89,7 +90,14 @@ def check_connectivity(cls, task: Task, logger, obj) -> Result: @classmethod def compliance_config( - cls, task: Task, logger, obj, features: str, backup_file: str, intended_file: str, platform: str + cls, + task: Task, + logger, + obj, + features: str, + backup_file: str, + intended_file: str, + platform: str, ) -> Result: """Compare two configurations against each other. @@ -248,7 +256,13 @@ class NapalmDefault(DispatcherMixin): @classmethod def get_config( - cls, task: Task, logger, obj, backup_file: str, remove_lines: list, substitute_lines: list + cls, + task: Task, + logger, + obj, + backup_file: str, + remove_lines: list, + substitute_lines: list, ) -> Result: """Get the latest configuration from the device. @@ -340,9 +354,15 @@ def replace_config( logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) - logger.info(f"result: {push_result[0].result}, changed: {push_result.changed}", extra={"object": obj}) + logger.info( + f"result: {push_result[0].result}, changed: {push_result.changed}", + extra={"object": obj}, + ) logger.info("Config provision ended", extra={"object": obj}) - return Result(host=task.host, result={"changed": push_result.changed, "result": push_result[0].result}) + return Result( + host=task.host, + result={"changed": push_result.changed, "result": push_result[0].result}, + ) @classmethod def merge_config( @@ -386,9 +406,15 @@ def merge_config( logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) - logger.info(f"result: {push_result[0].result}, changed: {push_result.changed}", extra={"object": obj}) + logger.info( + f"result: {push_result[0].result}, changed: {push_result.changed}", + extra={"object": obj}, + ) logger.info("Config merge ended", extra={"object": obj}) - return Result(host=task.host, result={"changed": push_result.changed, "result": push_result[0].result}) + return Result( + host=task.host, + result={"changed": push_result.changed, "result": push_result[0].result}, + ) class NetmikoDefault(DispatcherMixin): @@ -398,7 +424,13 @@ class NetmikoDefault(DispatcherMixin): @classmethod def get_config( - cls, task: Task, logger, obj, backup_file: str, remove_lines: list, substitute_lines: list + cls, + task: Task, + logger, + obj, + backup_file: str, + remove_lines: list, + substitute_lines: list, ) -> Result: """Get the latest configuration from the device using Netmiko. From 176965568aa80b3acca7181134c1d3d922c3c5fc Mon Sep 17 00:00:00 2001 From: Josh VanDeraa Date: Fri, 29 Sep 2023 16:23:44 -0500 Subject: [PATCH 09/13] Blacken code. --- nornir_nautobot/plugins/tasks/dispatcher/default.py | 1 - 1 file changed, 1 deletion(-) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/default.py b/nornir_nautobot/plugins/tasks/dispatcher/default.py index 1e8cdae..7108481 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/default.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/default.py @@ -364,7 +364,6 @@ def replace_config( result={"changed": push_result.changed, "result": push_result[0].result}, ) - @classmethod def merge_config( cls, From 0c2613cb93b72b8c1f78e5801d1972cddb47f641 Mon Sep 17 00:00:00 2001 From: itdependsnetworks Date: Sat, 30 Sep 2023 14:02:27 -0400 Subject: [PATCH 10/13] Update changelog, add error codes to docs, fix misspell in class name, update loggers --- docs/dev/CHANGELOG.md | 15 +- docs/task/task.md | 51 +- docs/task/troubleshooting/E1001.md | 19 + docs/task/troubleshooting/E1002.md | 19 + docs/task/troubleshooting/E1003.md | 19 + docs/task/troubleshooting/E1004.md | 19 + docs/task/troubleshooting/E1005.md | 19 + docs/task/troubleshooting/E1006.md | 19 + docs/task/troubleshooting/E1007.md | 19 + docs/task/troubleshooting/E1008.md | 19 + docs/task/troubleshooting/E1009.md | 19 + docs/task/troubleshooting/E1010.md | 19 + docs/task/troubleshooting/E1011.md | 19 + docs/task/troubleshooting/E1012.md | 19 + docs/task/troubleshooting/E1013.md | 19 + docs/task/troubleshooting/E1014.md | 19 + docs/task/troubleshooting/E1015.md | 19 + docs/task/troubleshooting/E1016.md | 19 + docs/task/troubleshooting/E1017.md | 19 + docs/task/troubleshooting/E1018.md | 19 + docs/task/troubleshooting/E1019.md | 19 + docs/task/troubleshooting/E1020.md | 19 + docs/task/troubleshooting/E1021.md | 19 + docs/task/troubleshooting/E1022.md | 19 + docs/task/troubleshooting/E1023.md | 19 + docs/task/troubleshooting/E1024.md | 19 + docs/task/troubleshooting/E1025.md | 19 + docs/task/troubleshooting/E1026.md | 19 + docs/task/troubleshooting/E1027.md | 19 + docs/task/troubleshooting/index.md | 5 + mkdocs.yml | 34 +- .../plugins/tasks/dispatcher/default.py | 15 +- .../tasks/dispatcher/mikrotik_routeros.py | 30 +- .../tasks/dispatcher/ruckus_fastiron.py | 45 +- .../tasks/dispatcher/ruckus_smartzone.py | 2 +- poetry.lock | 549 +++++++++--------- 36 files changed, 924 insertions(+), 335 deletions(-) create mode 100644 docs/task/troubleshooting/E1001.md create mode 100644 docs/task/troubleshooting/E1002.md create mode 100644 docs/task/troubleshooting/E1003.md create mode 100644 docs/task/troubleshooting/E1004.md create mode 100644 docs/task/troubleshooting/E1005.md create mode 100644 docs/task/troubleshooting/E1006.md create mode 100644 docs/task/troubleshooting/E1007.md create mode 100644 docs/task/troubleshooting/E1008.md create mode 100644 docs/task/troubleshooting/E1009.md create mode 100644 docs/task/troubleshooting/E1010.md create mode 100644 docs/task/troubleshooting/E1011.md create mode 100644 docs/task/troubleshooting/E1012.md create mode 100644 docs/task/troubleshooting/E1013.md create mode 100644 docs/task/troubleshooting/E1014.md create mode 100644 docs/task/troubleshooting/E1015.md create mode 100644 docs/task/troubleshooting/E1016.md create mode 100644 docs/task/troubleshooting/E1017.md create mode 100644 docs/task/troubleshooting/E1018.md create mode 100644 docs/task/troubleshooting/E1019.md create mode 100644 docs/task/troubleshooting/E1020.md create mode 100644 docs/task/troubleshooting/E1021.md create mode 100644 docs/task/troubleshooting/E1022.md create mode 100644 docs/task/troubleshooting/E1023.md create mode 100644 docs/task/troubleshooting/E1024.md create mode 100644 docs/task/troubleshooting/E1025.md create mode 100644 docs/task/troubleshooting/E1026.md create mode 100644 docs/task/troubleshooting/E1027.md create mode 100644 docs/task/troubleshooting/index.md diff --git a/docs/dev/CHANGELOG.md b/docs/dev/CHANGELOG.md index b91fe09..1161b5d 100644 --- a/docs/dev/CHANGELOG.md +++ b/docs/dev/CHANGELOG.md @@ -2,14 +2,21 @@ ## 3.0.0 -- Remove legacy dispatcher method -- error codes + +- [#107](https://github.com/nautobot/nornir-nautobot/pull/107) Changed the dispatcher method function signature - Breaking change +- [#107](https://github.com/nautobot/nornir-nautobot/pull/107) Restructured file layout to accommodate new dispatcher - Breaking change +- [#107](https://github.com/nautobot/nornir-nautobot/pull/107) Removed NornirLogger - Breaking Change +- [#107](https://github.com/nautobot/nornir-nautobot/pull/107) Updated to Nautobot loggin standard based on Python standard library logger +- [#107](https://github.com/nautobot/nornir-nautobot/pull/107) Migrated filters to Nautobot 2.0 standards, e.g. location +- [#107](https://github.com/nautobot/nornir-nautobot/pull/107) Added error codes e.g. E1001, to better facilitate troubleshooting +- [#107](https://github.com/nautobot/nornir-nautobot/pull/107) Added ability to control what `tcp_port` number to use in the `check_connectivity` method +- [#107](https://github.com/nautobot/nornir-nautobot/pull/107) Updated to latest NTC development standards and updated dependencies ## v2.6.1 -- (#108) Add enable threading option -- (#109) Patch merge_config methods return values +- [#108](https://github.com/nautobot/nornir-nautobot/pull/108) Add enable threading option +- [#109](https://github.com/nautobot/nornir-nautobot/pull/109) Patch merge_config methods return values ## v2.6.0 diff --git a/docs/task/task.md b/docs/task/task.md index e5627a0..91f9e88 100644 --- a/docs/task/task.md +++ b/docs/task/task.md @@ -1,30 +1,36 @@ ---- -hide: - - navigation ---- # Task Plugins The only task plugin currently is the "dispatcher" plugin. This plugin dispatches to the more specific OS specific functions. To demonstrate the primary components of the code: ## Dispatcher Sender -- If exists check `custom_dispatcher`, for network_driver (fail if not found) -- Check for framework & driver `f".dispatcher.{framework}{network_driver.titlecase()}"` -- Check for default, e.g. `f".dispatcher.{framework}default"` +- If exists check `custom_dispatcher`, for network_driver, if a custom_dispatcher is used but not found, fail immediately +- Check for framework & driver `f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework.title()}{network_driver_title}"` +- Check for default, e.g. `f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework.title()}Default"` !!! info Where `framework` is a library like `netmiko` or `napalm` and `network_driver` is the platform like `cisco_ios` or `arista_eos`. +This may seem like a lot, but it essentially can be broken down to: + +- If there is a custom_dispatcher, **only** use that +- Check for the `framework` and `network_driver` +- Check for the `framework`'s default + +For completeness here is the referenced code as of SEP-2023. + ```python if not kwargs.get("custom_dispatcher"): custom_dispatcher = {} - logger.debug(f"Dispatcher process started for {task.host.name} ({task.host.platform.network_driver})") + logger.debug(f"Dispatcher process started for {task.host.name} ({task.host.platform})") - network_driver = task.host.platform.network_driver + network_driver = task.host.platform network_driver_title = snake_to_title_case(network_driver) custom_dispatcher_path = [custom_dispatcher.get(network_driver)] - framework_path = f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework}{network_driver_title}" - framework_default_path = f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework}Default" + framework_path = ( + f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework.title()}{network_driver_title}" + ) + framework_default_path = f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework.title()}Default" if custom_dispatcher.get(network_driver): driver_class = import_string(custom_dispatcher_path) @@ -53,19 +59,18 @@ class NautobotNornirDriver: ```python task.run( task=dispatcher, - name="SAVE BACKUP CONFIGURATION TO FILE", - method="get_config", obj=obj, logger=logger, + method="get_config", + framework="netmiko", + name="SAVE BACKUP CONFIGURATION TO FILE", backup_file=backup_file, - remove_lines=global_settings, - substitute_lines=substitute_lines, + remove_lines=remove_regex_dict.get(obj.platform.network_driver, []), + substitute_lines=replace_regex_dict.get(obj.platform.network_driver, []), ) ``` -TODO: 2.0 update the logger info - -The dispatcher expects the two primary objects, the `obj` and `logger` objects. The `obj` object should be a Device model instance. The logger should be `NornirLogger` instance which is imported from `nornir_nautobot.utils.logger`. This logging object optionally takes in a Nautobot Job object named nautobot_job. This is for use within the Nautobot platform Jobs. +The dispatcher expects the two primary objects, the `obj` and `logger` objects. The `obj` object should be a Device model instance. The logger must conform to the standard Python logger, in that it should take is `message` as the first arg and allow a dictionary called `extra`. Each task will raise a `NornirNautobotException` for known issues. Using a custom processor, the user can predict when it was an well known error. @@ -74,8 +79,8 @@ Each task will raise a `NornirNautobotException` for known issues. Using a custo The check connectivity receiver will send attempt to tcp ping the port based on the following order or precedence. -- Prefer `task.data["custom_field_data"]["tcp_port"]` if is a valid integer -- Prefer `task.data["config_context_data"]["tcp_port"]` if is a valid integer +- Prefer `obj.cf["tcp_port"]` if is a valid integer +- Prefer `obj.get_config_context()["tcp_port"]` if is a valid integer - Prefer cls.tcp_port, which by default is defined in `DispatcherMixin` as 22 In this code you can see how it is set. @@ -86,11 +91,11 @@ class DispatcherMixin: tcp_port = 22 @classmethod - def _get_tcp_port(cls, task, obj) -> str: - custom_field = task.data.get("custom_field_data", {}).get("tcp_port") + def _get_tcp_port(cls, obj) -> str: + custom_field = obj.cf.get("tcp_port") if isinstance(custom_field, int): return custom_field - config_context = task.data.get("config_context_data", {}).get("tcp_port") + config_context = obj.get_config_context().get("tcp_port") if isinstance(config_context, int): return config_context return cls.tcp_port diff --git a/docs/task/troubleshooting/E1001.md b/docs/task/troubleshooting/E1001.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1001.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1002.md b/docs/task/troubleshooting/E1002.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1002.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1003.md b/docs/task/troubleshooting/E1003.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1003.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1004.md b/docs/task/troubleshooting/E1004.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1004.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1005.md b/docs/task/troubleshooting/E1005.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1005.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1006.md b/docs/task/troubleshooting/E1006.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1006.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1007.md b/docs/task/troubleshooting/E1007.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1007.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1008.md b/docs/task/troubleshooting/E1008.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1008.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1009.md b/docs/task/troubleshooting/E1009.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1009.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1010.md b/docs/task/troubleshooting/E1010.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1010.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1011.md b/docs/task/troubleshooting/E1011.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1011.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1012.md b/docs/task/troubleshooting/E1012.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1012.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1013.md b/docs/task/troubleshooting/E1013.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1013.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1014.md b/docs/task/troubleshooting/E1014.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1014.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1015.md b/docs/task/troubleshooting/E1015.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1015.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1016.md b/docs/task/troubleshooting/E1016.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1016.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1017.md b/docs/task/troubleshooting/E1017.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1017.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1018.md b/docs/task/troubleshooting/E1018.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1018.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1019.md b/docs/task/troubleshooting/E1019.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1019.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1020.md b/docs/task/troubleshooting/E1020.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1020.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1021.md b/docs/task/troubleshooting/E1021.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1021.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1022.md b/docs/task/troubleshooting/E1022.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1022.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1023.md b/docs/task/troubleshooting/E1023.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1023.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1024.md b/docs/task/troubleshooting/E1024.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1024.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1025.md b/docs/task/troubleshooting/E1025.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1025.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1026.md b/docs/task/troubleshooting/E1026.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1026.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/E1027.md b/docs/task/troubleshooting/E1027.md new file mode 100644 index 0000000..c66a010 --- /dev/null +++ b/docs/task/troubleshooting/E1027.md @@ -0,0 +1,19 @@ +# E10XX Details + +## Message emitted: + +`E10XX: Details coming soon` + +## Description: + +Description that is coming soon. + +## Troubleshooting: + +Troubleshooting that is coming soon. + +## Recommendation: + +Recommendation that is coming soon. + + diff --git a/docs/task/troubleshooting/index.md b/docs/task/troubleshooting/index.md new file mode 100644 index 0000000..4adcabd --- /dev/null +++ b/docs/task/troubleshooting/index.md @@ -0,0 +1,5 @@ +# Troubleshooting Overview + +In an effort to help with troubleshooting, each expected error, will now emit an error ID, in the format of `E1XXX`, such as `E1006: There was no password defined, preemptively failed.`. The idea will be to define the error, the error message and some recommended troubleshooting steps or even potentially some fixes. + +This is an ongoing effort, but the foundation has been built. \ No newline at end of file diff --git a/mkdocs.yml b/mkdocs.yml index 2cdb5be..a94ea97 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -1,5 +1,5 @@ --- -dev_addr: "127.0.0.1:8002" +dev_addr: "127.0.0.1:8001" site_name: "Nornir Nautobot Documentation" edit_uri: "edit/develop/docs" site_url: "https://docs.nautobot.com/projects/nornir-nautobot/en/latest/" @@ -65,7 +65,37 @@ nav: - Inventory Overview: "inventory/inventory.md" - Common Issues: "inventory/common_issues.md" - Processor Plugin: "processor/processor.md" - - Task Plugin: "task/task.md" + - Task Plugin: + - Task Overview: "task/task.md" + - Troubleshooting: + - "task/troubleshooting/index.md" + - E1001: "task/troubleshooting/E1001.md" + - E1002: "task/troubleshooting/E1002.md" + - E1003: "task/troubleshooting/E1003.md" + - E1004: "task/troubleshooting/E1004.md" + - E1005: "task/troubleshooting/E1005.md" + - E1006: "task/troubleshooting/E1006.md" + - E1007: "task/troubleshooting/E1007.md" + - E1008: "task/troubleshooting/E1008.md" + - E1009: "task/troubleshooting/E1009.md" + - E1010: "task/troubleshooting/E1010.md" + - E1011: "task/troubleshooting/E1011.md" + - E1012: "task/troubleshooting/E1012.md" + - E1013: "task/troubleshooting/E1013.md" + - E1014: "task/troubleshooting/E1014.md" + - E1015: "task/troubleshooting/E1015.md" + - E1016: "task/troubleshooting/E1016.md" + - E1017: "task/troubleshooting/E1017.md" + - E1018: "task/troubleshooting/E1018.md" + - E1019: "task/troubleshooting/E1019.md" + - E1020: "task/troubleshooting/E1020.md" + - E1021: "task/troubleshooting/E1021.md" + - E1022: "task/troubleshooting/E1022.md" + - E1023: "task/troubleshooting/E1023.md" + - E1024: "task/troubleshooting/E1024.md" + - E1025: "task/troubleshooting/E1025.md" + - E1026: "task/troubleshooting/E1026.md" + - E1027: "task/troubleshooting/E1027.md" - Developer Guide: - Contributing: "dev/CONTRIBUTING.md" - Changelog: "dev/CHANGELOG.md" diff --git a/nornir_nautobot/plugins/tasks/dispatcher/default.py b/nornir_nautobot/plugins/tasks/dispatcher/default.py index 7108481..1329bb0 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/default.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/default.py @@ -7,19 +7,24 @@ from typing import Optional import jinja2 -from netmiko import NetmikoAuthenticationException, NetmikoTimeoutException + from netutils.config.clean import clean_config, sanitize_config from netutils.config.compliance import compliance from netutils.dns import is_fqdn_resolvable from netutils.ip import is_ip from netutils.ping import tcp_ping + +from netmiko import NetmikoAuthenticationException, NetmikoTimeoutException + from nornir.core.exceptions import NornirSubTaskError from nornir.core.task import Result, Task + from nornir_jinja2.plugins.tasks import template_file from nornir_napalm.plugins.tasks import napalm_configure, napalm_get +from nornir_netmiko.tasks import netmiko_send_command + from nornir_nautobot.exceptions import NornirNautobotException from nornir_nautobot.utils.helpers import make_folder -from nornir_netmiko.tasks import netmiko_send_command _logger = logging.getLogger(__name__) @@ -30,7 +35,7 @@ class DispatcherMixin: tcp_port = 22 @classmethod - def _get_hostname(cls, task: Task) -> str: + def _get_hostname(cls, task: Task, obj=None) -> str: # pylint: disable=unused-argument return task.host.hostname @classmethod @@ -55,7 +60,7 @@ def check_connectivity(cls, task: Task, logger, obj) -> Result: Returns: Result: Nornir Result object. """ - hostname = cls._get_hostname(task, obj) + hostname = cls._get_hostname(task) if is_ip(hostname): ip_addr = hostname else: @@ -67,7 +72,7 @@ def check_connectivity(cls, task: Task, logger, obj) -> Result: raise NornirNautobotException(error_msg) ip_addr = socket.gethostbyname(hostname) - port = cls._get_tcp_port(task, obj) + port = cls._get_tcp_port(obj) # TODO: Remove after fixing tcp_ping in netutils try: _tcp_ping = tcp_ping(ip_addr, port) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py b/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py index 079d566..d58e81e 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py @@ -108,7 +108,7 @@ def get_config( # pylint: disable=R0913,R0914 return Result(host=task.host, result={"config": running_config}) -class NetmikoMikrotekRouteros(NetmikoDefault): +class NetmikoMikrotikRouteros(NetmikoDefault): """Driver for Mikrotik Router OS.""" config_command = "export terse" @@ -167,10 +167,9 @@ def get_config( # pylint: disable=R0913,R0914 try: result = task.run(task=netmiko_send_command, command_string=command) except NornirSubTaskError as exc: - logger.error(f"Failed with an unknown issue. `{exc.result.exception}`", extra={"object": obj}) - raise NornirNautobotException( # pylint: disable=W0707 - f"Failed with an unknown issue. `{exc.result.exception}`" - ) + error_msg = f"Failed with an unknown issue. `{exc.result.exception}`" + logger.error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) if result[0].failed: return result @@ -207,7 +206,7 @@ def merge_config( Result: Nornir Result object with a dict as a result containing what changed and the result of the push. """ NETMIKO_FAIL_MSG = ["bad", "failed", "failure"] # pylint: disable=C0103 - logger.log_success(obj, "Config merge starting") + logger.info("Config merge starting", extra={"object": obj}) try: config_list = config.splitlines() @@ -216,17 +215,18 @@ def merge_config( config_commands=config_list, ) except NornirSubTaskError as exc: - logger.log_failure(obj, f"Failed with error: `{exc.result.exception}`") - raise NornirNautobotException() from exc + error_msg = f"E1015 Failed with error: `{exc.result.exception}`" + logger.error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) from exc if any(msg in push_result[0].result.lower() for msg in NETMIKO_FAIL_MSG): - logger.log_warning(obj, "Config merged with errors, please check full info log below.") - logger.log_failure(obj, f"result: {push_result[0].result}") - push_result[0].failed = True - else: - logger.log_success(obj, "Config merged successfully.") - logger.log_info(obj, f"result: {push_result[0].result}") - push_result[0].failed = False + logger.warning("Config merged with errors, please check full info log below.", extra={"object": obj}) + error_msg = f"E1026: result: {push_result[0].result}" + logger.error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + logger.info("Config merged successfully.", extra={"object": obj}) + logger.info(f"result: {push_result[0].result}", extra={"object": obj}) + push_result[0].failed = False push_result[0].changed = True return Result( diff --git a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py index 9bec133..414c135 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py @@ -13,12 +13,7 @@ class NetmikoRuckusFastiron(NetmikoDefault): config_command = "show running-config" @staticmethod - def merge_config( - task: Task, - logger, - obj, - config: str, - ) -> Result: + def merge_config(task: Task, logger, obj, config: str) -> Result: """Send configuration to merge on the device. Args: @@ -36,7 +31,7 @@ def merge_config( Result: Nornir Result object with a dict as a result containing what changed and the result of the push. """ NETMIKO_FAIL_MSG = ["invalid", "fail"] # pylint: disable=C0103 - logger.log_success(obj, "Config merge starting") + logger.info("Config merge starting", extra={"object": obj}) try: config_list = config.splitlines() @@ -45,24 +40,28 @@ def merge_config( config_commands=config_list, ) except NornirSubTaskError as exc: - logger.log_failure(obj, f"Failed with error: `{exc.result.exception}`") - raise NornirNautobotException() from exc + error_msg = f"Failed with error: `{exc.result.exception}`" + logger.error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) from exc if any(msg in push_result[0].result.lower() for msg in NETMIKO_FAIL_MSG): - logger.log_warning(obj, "Config merged with errors, please check full info log below.") - logger.log_failure(obj, f"result: {push_result[0].result}") - push_result[0].failed = True - else: - logger.log_success(obj, "Config merged successfully.") - logger.log_info(obj, f"result: {push_result[0].result}") - push_result[0].failed = False - try: - save_result = task.run( - task=netmiko_save_config, - ) - logger.log_info(obj, f"config saved: {save_result[0].result}") - except NornirSubTaskError as exc: - logger.log_failure(obj, f"config merged, but failed to save: {exc.result.exception}") + logger.warning("Config merged with errors, please check full info log below.", extra={"object": obj}) + error_msg = f"E1026: result: {push_result[0].result}" + logger.error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) + + logger.info("Config merged successfully.", extra={"object": obj}) + logger.info(f"result: {push_result[0].result}", extra={"object": obj}) + push_result[0].failed = False + try: + save_result = task.run( + task=netmiko_save_config, + ) + logger.info(f"config saved: {save_result[0].result}", extra={"object": obj}) + except NornirSubTaskError as exc: + error_msg = f"E1027: config merged, but failed to save: {exc.result.exception}" + logger.error(error_msg, extra={"object": obj}) + raise NornirNautobotException(error_msg) from exc push_result[0].changed = True return Result( diff --git a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py index 6a2ecfe..fb2cfae 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py @@ -44,7 +44,7 @@ class ApiRuckusSmartzone(DispatcherMixin): } @classmethod - def _get_hostname(cls, task: Task, obj) -> str: + def _get_hostname(cls, task: Task, obj=None) -> str: hostname = ( obj.get_computed_field("wireless_controller") if task.host.platform in AP_PLATFORM_LIST diff --git a/poetry.lock b/poetry.lock index fa5f669..977b0fa 100644 --- a/poetry.lock +++ b/poetry.lock @@ -23,13 +23,13 @@ trio = ["trio (>=0.22)"] [[package]] name = "astroid" -version = "2.15.6" +version = "2.15.8" description = "An abstract syntax tree for Python with inference support." optional = false python-versions = ">=3.7.2" files = [ - {file = "astroid-2.15.6-py3-none-any.whl", hash = "sha256:389656ca57b6108f939cf5d2f9a2a825a3be50ba9d589670f393236e0a03b91c"}, - {file = "astroid-2.15.6.tar.gz", hash = "sha256:903f024859b7c7687d7a7f3a3f73b17301f8e42dfd9cc9df9d4418172d3e2dbd"}, + {file = "astroid-2.15.8-py3-none-any.whl", hash = "sha256:1aa149fc5c6589e3d0ece885b4491acd80af4f087baafa3fb5203b113e68cd3c"}, + {file = "astroid-2.15.8.tar.gz", hash = "sha256:6c107453dffee9055899705de3c9ead36e74119cee151e5a9aaf7f0b0e020a6a"}, ] [package.dependencies] @@ -188,75 +188,63 @@ files = [ [[package]] name = "cffi" -version = "1.15.1" +version = "1.16.0" description = "Foreign Function Interface for Python calling C code." optional = false -python-versions = "*" +python-versions = ">=3.8" files = [ - {file = "cffi-1.15.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:a66d3508133af6e8548451b25058d5812812ec3798c886bf38ed24a98216fab2"}, - {file = "cffi-1.15.1-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:470c103ae716238bbe698d67ad020e1db9d9dba34fa5a899b5e21577e6d52ed2"}, - {file = "cffi-1.15.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:9ad5db27f9cabae298d151c85cf2bad1d359a1b9c686a275df03385758e2f914"}, - {file = "cffi-1.15.1-cp27-cp27m-win32.whl", hash = "sha256:b3bbeb01c2b273cca1e1e0c5df57f12dce9a4dd331b4fa1635b8bec26350bde3"}, - {file = "cffi-1.15.1-cp27-cp27m-win_amd64.whl", hash = "sha256:e00b098126fd45523dd056d2efba6c5a63b71ffe9f2bbe1a4fe1716e1d0c331e"}, - {file = "cffi-1.15.1-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:d61f4695e6c866a23a21acab0509af1cdfd2c013cf256bbf5b6b5e2695827162"}, - {file = "cffi-1.15.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:ed9cb427ba5504c1dc15ede7d516b84757c3e3d7868ccc85121d9310d27eed0b"}, - {file = "cffi-1.15.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:39d39875251ca8f612b6f33e6b1195af86d1b3e60086068be9cc053aa4376e21"}, - {file = "cffi-1.15.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:285d29981935eb726a4399badae8f0ffdff4f5050eaa6d0cfc3f64b857b77185"}, - {file = "cffi-1.15.1-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3eb6971dcff08619f8d91607cfc726518b6fa2a9eba42856be181c6d0d9515fd"}, - {file = "cffi-1.15.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:21157295583fe8943475029ed5abdcf71eb3911894724e360acff1d61c1d54bc"}, - {file = "cffi-1.15.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5635bd9cb9731e6d4a1132a498dd34f764034a8ce60cef4f5319c0541159392f"}, - {file = "cffi-1.15.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2012c72d854c2d03e45d06ae57f40d78e5770d252f195b93f581acf3ba44496e"}, - {file = "cffi-1.15.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd86c085fae2efd48ac91dd7ccffcfc0571387fe1193d33b6394db7ef31fe2a4"}, - {file = "cffi-1.15.1-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:fa6693661a4c91757f4412306191b6dc88c1703f780c8234035eac011922bc01"}, - {file = "cffi-1.15.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:59c0b02d0a6c384d453fece7566d1c7e6b7bae4fc5874ef2ef46d56776d61c9e"}, - {file = "cffi-1.15.1-cp310-cp310-win32.whl", hash = "sha256:cba9d6b9a7d64d4bd46167096fc9d2f835e25d7e4c121fb2ddfc6528fb0413b2"}, - {file = "cffi-1.15.1-cp310-cp310-win_amd64.whl", hash = "sha256:ce4bcc037df4fc5e3d184794f27bdaab018943698f4ca31630bc7f84a7b69c6d"}, - {file = "cffi-1.15.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3d08afd128ddaa624a48cf2b859afef385b720bb4b43df214f85616922e6a5ac"}, - {file = "cffi-1.15.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3799aecf2e17cf585d977b780ce79ff0dc9b78d799fc694221ce814c2c19db83"}, - {file = "cffi-1.15.1-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a591fe9e525846e4d154205572a029f653ada1a78b93697f3b5a8f1f2bc055b9"}, - {file = "cffi-1.15.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3548db281cd7d2561c9ad9984681c95f7b0e38881201e157833a2342c30d5e8c"}, - {file = "cffi-1.15.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:91fc98adde3d7881af9b59ed0294046f3806221863722ba7d8d120c575314325"}, - {file = "cffi-1.15.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:94411f22c3985acaec6f83c6df553f2dbe17b698cc7f8ae751ff2237d96b9e3c"}, - {file = "cffi-1.15.1-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:03425bdae262c76aad70202debd780501fabeaca237cdfddc008987c0e0f59ef"}, - {file = "cffi-1.15.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:cc4d65aeeaa04136a12677d3dd0b1c0c94dc43abac5860ab33cceb42b801c1e8"}, - {file = "cffi-1.15.1-cp311-cp311-win32.whl", hash = "sha256:a0f100c8912c114ff53e1202d0078b425bee3649ae34d7b070e9697f93c5d52d"}, - {file = "cffi-1.15.1-cp311-cp311-win_amd64.whl", hash = "sha256:04ed324bda3cda42b9b695d51bb7d54b680b9719cfab04227cdd1e04e5de3104"}, - {file = "cffi-1.15.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:50a74364d85fd319352182ef59c5c790484a336f6db772c1a9231f1c3ed0cbd7"}, - {file = "cffi-1.15.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e263d77ee3dd201c3a142934a086a4450861778baaeeb45db4591ef65550b0a6"}, - {file = "cffi-1.15.1-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:cec7d9412a9102bdc577382c3929b337320c4c4c4849f2c5cdd14d7368c5562d"}, - {file = "cffi-1.15.1-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4289fc34b2f5316fbb762d75362931e351941fa95fa18789191b33fc4cf9504a"}, - {file = "cffi-1.15.1-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:173379135477dc8cac4bc58f45db08ab45d228b3363adb7af79436135d028405"}, - {file = "cffi-1.15.1-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:6975a3fac6bc83c4a65c9f9fcab9e47019a11d3d2cf7f3c0d03431bf145a941e"}, - {file = "cffi-1.15.1-cp36-cp36m-win32.whl", hash = "sha256:2470043b93ff09bf8fb1d46d1cb756ce6132c54826661a32d4e4d132e1977adf"}, - {file = "cffi-1.15.1-cp36-cp36m-win_amd64.whl", hash = "sha256:30d78fbc8ebf9c92c9b7823ee18eb92f2e6ef79b45ac84db507f52fbe3ec4497"}, - {file = "cffi-1.15.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:198caafb44239b60e252492445da556afafc7d1e3ab7a1fb3f0584ef6d742375"}, - {file = "cffi-1.15.1-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5ef34d190326c3b1f822a5b7a45f6c4535e2f47ed06fec77d3d799c450b2651e"}, - {file = "cffi-1.15.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8102eaf27e1e448db915d08afa8b41d6c7ca7a04b7d73af6514df10a3e74bd82"}, - {file = "cffi-1.15.1-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5df2768244d19ab7f60546d0c7c63ce1581f7af8b5de3eb3004b9b6fc8a9f84b"}, - {file = "cffi-1.15.1-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a8c4917bd7ad33e8eb21e9a5bbba979b49d9a97acb3a803092cbc1133e20343c"}, - {file = "cffi-1.15.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0e2642fe3142e4cc4af0799748233ad6da94c62a8bec3a6648bf8ee68b1c7426"}, - {file = "cffi-1.15.1-cp37-cp37m-win32.whl", hash = "sha256:e229a521186c75c8ad9490854fd8bbdd9a0c9aa3a524326b55be83b54d4e0ad9"}, - {file = "cffi-1.15.1-cp37-cp37m-win_amd64.whl", hash = "sha256:a0b71b1b8fbf2b96e41c4d990244165e2c9be83d54962a9a1d118fd8657d2045"}, - {file = "cffi-1.15.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:320dab6e7cb2eacdf0e658569d2575c4dad258c0fcc794f46215e1e39f90f2c3"}, - {file = "cffi-1.15.1-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1e74c6b51a9ed6589199c787bf5f9875612ca4a8a0785fb2d4a84429badaf22a"}, - {file = "cffi-1.15.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a5c84c68147988265e60416b57fc83425a78058853509c1b0629c180094904a5"}, - {file = "cffi-1.15.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3b926aa83d1edb5aa5b427b4053dc420ec295a08e40911296b9eb1b6170f6cca"}, - {file = "cffi-1.15.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:87c450779d0914f2861b8526e035c5e6da0a3199d8f1add1a665e1cbc6fc6d02"}, - {file = "cffi-1.15.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4f2c9f67e9821cad2e5f480bc8d83b8742896f1242dba247911072d4fa94c192"}, - {file = "cffi-1.15.1-cp38-cp38-win32.whl", hash = "sha256:8b7ee99e510d7b66cdb6c593f21c043c248537a32e0bedf02e01e9553a172314"}, - {file = "cffi-1.15.1-cp38-cp38-win_amd64.whl", hash = "sha256:00a9ed42e88df81ffae7a8ab6d9356b371399b91dbdf0c3cb1e84c03a13aceb5"}, - {file = "cffi-1.15.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:54a2db7b78338edd780e7ef7f9f6c442500fb0d41a5a4ea24fff1c929d5af585"}, - {file = "cffi-1.15.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:fcd131dd944808b5bdb38e6f5b53013c5aa4f334c5cad0c72742f6eba4b73db0"}, - {file = "cffi-1.15.1-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7473e861101c9e72452f9bf8acb984947aa1661a7704553a9f6e4baa5ba64415"}, - {file = "cffi-1.15.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6c9a799e985904922a4d207a94eae35c78ebae90e128f0c4e521ce339396be9d"}, - {file = "cffi-1.15.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3bcde07039e586f91b45c88f8583ea7cf7a0770df3a1649627bf598332cb6984"}, - {file = "cffi-1.15.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:33ab79603146aace82c2427da5ca6e58f2b3f2fb5da893ceac0c42218a40be35"}, - {file = "cffi-1.15.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5d598b938678ebf3c67377cdd45e09d431369c3b1a5b331058c338e201f12b27"}, - {file = "cffi-1.15.1-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:db0fbb9c62743ce59a9ff687eb5f4afbe77e5e8403d6697f7446e5f609976f76"}, - {file = "cffi-1.15.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:98d85c6a2bef81588d9227dde12db8a7f47f639f4a17c9ae08e773aa9c697bf3"}, - {file = "cffi-1.15.1-cp39-cp39-win32.whl", hash = "sha256:40f4774f5a9d4f5e344f31a32b5096977b5d48560c5592e2f3d2c4374bd543ee"}, - {file = "cffi-1.15.1-cp39-cp39-win_amd64.whl", hash = "sha256:70df4e3b545a17496c9b3f41f5115e69a4f2e77e94e1d2a8e1070bc0c38c8a3c"}, - {file = "cffi-1.15.1.tar.gz", hash = "sha256:d400bfb9a37b1351253cb402671cea7e89bdecc294e8016a707f6d1d8ac934f9"}, + {file = "cffi-1.16.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:6b3d6606d369fc1da4fd8c357d026317fbb9c9b75d36dc16e90e84c26854b088"}, + {file = "cffi-1.16.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:ac0f5edd2360eea2f1daa9e26a41db02dd4b0451b48f7c318e217ee092a213e9"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7e61e3e4fa664a8588aa25c883eab612a188c725755afff6289454d6362b9673"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a72e8961a86d19bdb45851d8f1f08b041ea37d2bd8d4fd19903bc3083d80c896"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5b50bf3f55561dac5438f8e70bfcdfd74543fd60df5fa5f62d94e5867deca684"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7651c50c8c5ef7bdb41108b7b8c5a83013bfaa8a935590c5d74627c047a583c7"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e4108df7fe9b707191e55f33efbcb2d81928e10cea45527879a4749cbe472614"}, + {file = "cffi-1.16.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:32c68ef735dbe5857c810328cb2481e24722a59a2003018885514d4c09af9743"}, + {file = "cffi-1.16.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:673739cb539f8cdaa07d92d02efa93c9ccf87e345b9a0b556e3ecc666718468d"}, + {file = "cffi-1.16.0-cp310-cp310-win32.whl", hash = "sha256:9f90389693731ff1f659e55c7d1640e2ec43ff725cc61b04b2f9c6d8d017df6a"}, + {file = "cffi-1.16.0-cp310-cp310-win_amd64.whl", hash = "sha256:e6024675e67af929088fda399b2094574609396b1decb609c55fa58b028a32a1"}, + {file = "cffi-1.16.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b84834d0cf97e7d27dd5b7f3aca7b6e9263c56308ab9dc8aae9784abb774d404"}, + {file = "cffi-1.16.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:1b8ebc27c014c59692bb2664c7d13ce7a6e9a629be20e54e7271fa696ff2b417"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ee07e47c12890ef248766a6e55bd38ebfb2bb8edd4142d56db91b21ea68b7627"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d8a9d3ebe49f084ad71f9269834ceccbf398253c9fac910c4fd7053ff1386936"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e70f54f1796669ef691ca07d046cd81a29cb4deb1e5f942003f401c0c4a2695d"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5bf44d66cdf9e893637896c7faa22298baebcd18d1ddb6d2626a6e39793a1d56"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7b78010e7b97fef4bee1e896df8a4bbb6712b7f05b7ef630f9d1da00f6444d2e"}, + {file = "cffi-1.16.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:c6a164aa47843fb1b01e941d385aab7215563bb8816d80ff3a363a9f8448a8dc"}, + {file = "cffi-1.16.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:e09f3ff613345df5e8c3667da1d918f9149bd623cd9070c983c013792a9a62eb"}, + {file = "cffi-1.16.0-cp311-cp311-win32.whl", hash = "sha256:2c56b361916f390cd758a57f2e16233eb4f64bcbeee88a4881ea90fca14dc6ab"}, + {file = "cffi-1.16.0-cp311-cp311-win_amd64.whl", hash = "sha256:db8e577c19c0fda0beb7e0d4e09e0ba74b1e4c092e0e40bfa12fe05b6f6d75ba"}, + {file = "cffi-1.16.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:fa3a0128b152627161ce47201262d3140edb5a5c3da88d73a1b790a959126956"}, + {file = "cffi-1.16.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:68e7c44931cc171c54ccb702482e9fc723192e88d25a0e133edd7aff8fcd1f6e"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:abd808f9c129ba2beda4cfc53bde801e5bcf9d6e0f22f095e45327c038bfe68e"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:88e2b3c14bdb32e440be531ade29d3c50a1a59cd4e51b1dd8b0865c54ea5d2e2"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fcc8eb6d5902bb1cf6dc4f187ee3ea80a1eba0a89aba40a5cb20a5087d961357"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b7be2d771cdba2942e13215c4e340bfd76398e9227ad10402a8767ab1865d2e6"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e715596e683d2ce000574bae5d07bd522c781a822866c20495e52520564f0969"}, + {file = "cffi-1.16.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:2d92b25dbf6cae33f65005baf472d2c245c050b1ce709cc4588cdcdd5495b520"}, + {file = "cffi-1.16.0-cp312-cp312-win32.whl", hash = "sha256:b2ca4e77f9f47c55c194982e10f058db063937845bb2b7a86c84a6cfe0aefa8b"}, + {file = "cffi-1.16.0-cp312-cp312-win_amd64.whl", hash = "sha256:68678abf380b42ce21a5f2abde8efee05c114c2fdb2e9eef2efdb0257fba1235"}, + {file = "cffi-1.16.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:0c9ef6ff37e974b73c25eecc13952c55bceed9112be2d9d938ded8e856138bcc"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a09582f178759ee8128d9270cd1344154fd473bb77d94ce0aeb2a93ebf0feaf0"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e760191dd42581e023a68b758769e2da259b5d52e3103c6060ddc02c9edb8d7b"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:80876338e19c951fdfed6198e70bc88f1c9758b94578d5a7c4c91a87af3cf31c"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a6a14b17d7e17fa0d207ac08642c8820f84f25ce17a442fd15e27ea18d67c59b"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6602bc8dc6f3a9e02b6c22c4fc1e47aa50f8f8e6d3f78a5e16ac33ef5fefa324"}, + {file = "cffi-1.16.0-cp38-cp38-win32.whl", hash = "sha256:131fd094d1065b19540c3d72594260f118b231090295d8c34e19a7bbcf2e860a"}, + {file = "cffi-1.16.0-cp38-cp38-win_amd64.whl", hash = "sha256:31d13b0f99e0836b7ff893d37af07366ebc90b678b6664c955b54561fc36ef36"}, + {file = "cffi-1.16.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:582215a0e9adbe0e379761260553ba11c58943e4bbe9c36430c4ca6ac74b15ed"}, + {file = "cffi-1.16.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:b29ebffcf550f9da55bec9e02ad430c992a87e5f512cd63388abb76f1036d8d2"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:dc9b18bf40cc75f66f40a7379f6a9513244fe33c0e8aa72e2d56b0196a7ef872"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9cb4a35b3642fc5c005a6755a5d17c6c8b6bcb6981baf81cea8bfbc8903e8ba8"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b86851a328eedc692acf81fb05444bdf1891747c25af7529e39ddafaf68a4f3f"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c0f31130ebc2d37cdd8e44605fb5fa7ad59049298b3f745c74fa74c62fbfcfc4"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8f8e709127c6c77446a8c0a8c8bf3c8ee706a06cd44b1e827c3e6a2ee6b8c098"}, + {file = "cffi-1.16.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:748dcd1e3d3d7cd5443ef03ce8685043294ad6bd7c02a38d1bd367cfd968e000"}, + {file = "cffi-1.16.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:8895613bcc094d4a1b2dbe179d88d7fb4a15cee43c052e8885783fac397d91fe"}, + {file = "cffi-1.16.0-cp39-cp39-win32.whl", hash = "sha256:ed86a35631f7bfbb28e108dd96773b9d5a6ce4811cf6ea468bb6a359b256b1e4"}, + {file = "cffi-1.16.0-cp39-cp39-win_amd64.whl", hash = "sha256:3686dffb02459559c74dd3d81748269ffb0eb027c39a6fc99502de37d501faa8"}, + {file = "cffi-1.16.0.tar.gz", hash = "sha256:bcb3ef43e58665bbda2fb198698fcae6776483e0c4a631aa5647806c25e02cc0"}, ] [package.dependencies] @@ -264,86 +252,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]] @@ -373,34 +376,34 @@ files = [ [[package]] name = "cryptography" -version = "41.0.3" +version = "41.0.4" description = "cryptography is a package which provides cryptographic recipes and primitives to Python developers." optional = false python-versions = ">=3.7" files = [ - {file = "cryptography-41.0.3-cp37-abi3-macosx_10_12_universal2.whl", hash = "sha256:652627a055cb52a84f8c448185922241dd5217443ca194d5739b44612c5e6507"}, - {file = "cryptography-41.0.3-cp37-abi3-macosx_10_12_x86_64.whl", hash = "sha256:8f09daa483aedea50d249ef98ed500569841d6498aa9c9f4b0531b9964658922"}, - {file = "cryptography-41.0.3-cp37-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4fd871184321100fb400d759ad0cddddf284c4b696568204d281c902fc7b0d81"}, - {file = "cryptography-41.0.3-cp37-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:84537453d57f55a50a5b6835622ee405816999a7113267739a1b4581f83535bd"}, - {file = "cryptography-41.0.3-cp37-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:3fb248989b6363906827284cd20cca63bb1a757e0a2864d4c1682a985e3dca47"}, - {file = "cryptography-41.0.3-cp37-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:42cb413e01a5d36da9929baa9d70ca90d90b969269e5a12d39c1e0d475010116"}, - {file = "cryptography-41.0.3-cp37-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:aeb57c421b34af8f9fe830e1955bf493a86a7996cc1338fe41b30047d16e962c"}, - {file = "cryptography-41.0.3-cp37-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:6af1c6387c531cd364b72c28daa29232162010d952ceb7e5ca8e2827526aceae"}, - {file = "cryptography-41.0.3-cp37-abi3-win32.whl", hash = "sha256:0d09fb5356f975974dbcb595ad2d178305e5050656affb7890a1583f5e02a306"}, - {file = "cryptography-41.0.3-cp37-abi3-win_amd64.whl", hash = "sha256:a983e441a00a9d57a4d7c91b3116a37ae602907a7618b882c8013b5762e80574"}, - {file = "cryptography-41.0.3-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:5259cb659aa43005eb55a0e4ff2c825ca111a0da1814202c64d28a985d33b087"}, - {file = "cryptography-41.0.3-pp310-pypy310_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:67e120e9a577c64fe1f611e53b30b3e69744e5910ff3b6e97e935aeb96005858"}, - {file = "cryptography-41.0.3-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:7efe8041897fe7a50863e51b77789b657a133c75c3b094e51b5e4b5cec7bf906"}, - {file = "cryptography-41.0.3-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:ce785cf81a7bdade534297ef9e490ddff800d956625020ab2ec2780a556c313e"}, - {file = "cryptography-41.0.3-pp38-pypy38_pp73-macosx_10_12_x86_64.whl", hash = "sha256:57a51b89f954f216a81c9d057bf1a24e2f36e764a1ca9a501a6964eb4a6800dd"}, - {file = "cryptography-41.0.3-pp38-pypy38_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:4c2f0d35703d61002a2bbdcf15548ebb701cfdd83cdc12471d2bae80878a4207"}, - {file = "cryptography-41.0.3-pp38-pypy38_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:23c2d778cf829f7d0ae180600b17e9fceea3c2ef8b31a99e3c694cbbf3a24b84"}, - {file = "cryptography-41.0.3-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:95dd7f261bb76948b52a5330ba5202b91a26fbac13ad0e9fc8a3ac04752058c7"}, - {file = "cryptography-41.0.3-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:41d7aa7cdfded09b3d73a47f429c298e80796c8e825ddfadc84c8a7f12df212d"}, - {file = "cryptography-41.0.3-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:d0d651aa754ef58d75cec6edfbd21259d93810b73f6ec246436a21b7841908de"}, - {file = "cryptography-41.0.3-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:ab8de0d091acbf778f74286f4989cf3d1528336af1b59f3e5d2ebca8b5fe49e1"}, - {file = "cryptography-41.0.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:a74fbcdb2a0d46fe00504f571a2a540532f4c188e6ccf26f1f178480117b33c4"}, - {file = "cryptography-41.0.3.tar.gz", hash = "sha256:6d192741113ef5e30d89dcb5b956ef4e1578f304708701b8b73d38e3e1461f34"}, + {file = "cryptography-41.0.4-cp37-abi3-macosx_10_12_universal2.whl", hash = "sha256:80907d3faa55dc5434a16579952ac6da800935cd98d14dbd62f6f042c7f5e839"}, + {file = "cryptography-41.0.4-cp37-abi3-macosx_10_12_x86_64.whl", hash = "sha256:35c00f637cd0b9d5b6c6bd11b6c3359194a8eba9c46d4e875a3660e3b400005f"}, + {file = "cryptography-41.0.4-cp37-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cecfefa17042941f94ab54f769c8ce0fe14beff2694e9ac684176a2535bf9714"}, + {file = "cryptography-41.0.4-cp37-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e40211b4923ba5a6dc9769eab704bdb3fbb58d56c5b336d30996c24fcf12aadb"}, + {file = "cryptography-41.0.4-cp37-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:23a25c09dfd0d9f28da2352503b23e086f8e78096b9fd585d1d14eca01613e13"}, + {file = "cryptography-41.0.4-cp37-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:2ed09183922d66c4ec5fdaa59b4d14e105c084dd0febd27452de8f6f74704143"}, + {file = "cryptography-41.0.4-cp37-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:5a0f09cefded00e648a127048119f77bc2b2ec61e736660b5789e638f43cc397"}, + {file = "cryptography-41.0.4-cp37-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:9eeb77214afae972a00dee47382d2591abe77bdae166bda672fb1e24702a3860"}, + {file = "cryptography-41.0.4-cp37-abi3-win32.whl", hash = "sha256:3b224890962a2d7b57cf5eeb16ccaafba6083f7b811829f00476309bce2fe0fd"}, + {file = "cryptography-41.0.4-cp37-abi3-win_amd64.whl", hash = "sha256:c880eba5175f4307129784eca96f4e70b88e57aa3f680aeba3bab0e980b0f37d"}, + {file = "cryptography-41.0.4-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:004b6ccc95943f6a9ad3142cfabcc769d7ee38a3f60fb0dddbfb431f818c3a67"}, + {file = "cryptography-41.0.4-pp310-pypy310_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:86defa8d248c3fa029da68ce61fe735432b047e32179883bdb1e79ed9bb8195e"}, + {file = "cryptography-41.0.4-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:37480760ae08065437e6573d14be973112c9e6dcaf5f11d00147ee74f37a3829"}, + {file = "cryptography-41.0.4-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:b5f4dfe950ff0479f1f00eda09c18798d4f49b98f4e2006d644b3301682ebdca"}, + {file = "cryptography-41.0.4-pp38-pypy38_pp73-macosx_10_12_x86_64.whl", hash = "sha256:7e53db173370dea832190870e975a1e09c86a879b613948f09eb49324218c14d"}, + {file = "cryptography-41.0.4-pp38-pypy38_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:5b72205a360f3b6176485a333256b9bcd48700fc755fef51c8e7e67c4b63e3ac"}, + {file = "cryptography-41.0.4-pp38-pypy38_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:93530900d14c37a46ce3d6c9e6fd35dbe5f5601bf6b3a5c325c7bffc030344d9"}, + {file = "cryptography-41.0.4-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:efc8ad4e6fc4f1752ebfb58aefece8b4e3c4cae940b0994d43649bdfce8d0d4f"}, + {file = "cryptography-41.0.4-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:c3391bd8e6de35f6f1140e50aaeb3e2b3d6a9012536ca23ab0d9c35ec18c8a91"}, + {file = "cryptography-41.0.4-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:0d9409894f495d465fe6fda92cb70e8323e9648af912d5b9141d616df40a87b8"}, + {file = "cryptography-41.0.4-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:8ac4f9ead4bbd0bc8ab2d318f97d85147167a488be0e08814a37eb2f439d5cf6"}, + {file = "cryptography-41.0.4-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:047c4603aeb4bbd8db2756e38f5b8bd7e94318c047cfe4efeb5d715e08b49311"}, + {file = "cryptography-41.0.4.tar.gz", hash = "sha256:7febc3094125fc126a7f6fb1f420d0da639f3f32cb15c8ff0dc3997c4549f51a"}, ] [package.dependencies] @@ -514,27 +517,30 @@ smmap = ">=3.0.1,<6" [[package]] name = "gitpython" -version = "3.1.35" +version = "3.1.37" description = "GitPython is a Python library used to interact with Git repositories" optional = false python-versions = ">=3.7" files = [ - {file = "GitPython-3.1.35-py3-none-any.whl", hash = "sha256:c19b4292d7a1d3c0f653858db273ff8a6614100d1eb1528b014ec97286193c09"}, - {file = "GitPython-3.1.35.tar.gz", hash = "sha256:9cbefbd1789a5fe9bcf621bb34d3f441f3a90c8461d377f84eda73e721d9b06b"}, + {file = "GitPython-3.1.37-py3-none-any.whl", hash = "sha256:5f4c4187de49616d710a77e98ddf17b4782060a1788df441846bddefbb89ab33"}, + {file = "GitPython-3.1.37.tar.gz", hash = "sha256:f9b9ddc0761c125d5780eab2d64be4873fc6817c2899cbcb34b02344bdc7bc54"}, ] [package.dependencies] gitdb = ">=4.0.1,<5" +[package.extras] +test = ["black", "coverage[toml]", "ddt (>=1.1.1,!=1.4.3)", "mypy", "pre-commit", "pytest", "pytest-cov", "pytest-sugar"] + [[package]] name = "griffe" -version = "0.36.2" +version = "0.36.4" description = "Signatures for entire Python programs. Extract the structure, the frame, the skeleton of your project, to generate API documentation or find breaking changes in your API." optional = false python-versions = ">=3.8" files = [ - {file = "griffe-0.36.2-py3-none-any.whl", hash = "sha256:ba71895a3f5f606b18dcd950e8a1f8e7332a37f90f24caeb002546593f2e0eee"}, - {file = "griffe-0.36.2.tar.gz", hash = "sha256:333ade7932bb9096781d83092602625dfbfe220e87a039d2801259a1bd41d1c2"}, + {file = "griffe-0.36.4-py3-none-any.whl", hash = "sha256:4e37a723891fa774fafdd67240571801a1d90d0236562c178707e5c37fb3ebe2"}, + {file = "griffe-0.36.4.tar.gz", hash = "sha256:7b5968f5cc6446637ed0d3ded9de07d6a928f10ccb24116b1dd843635bf1994a"}, ] [package.dependencies] @@ -943,6 +949,16 @@ files = [ {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:5bbe06f8eeafd38e5d0a4894ffec89378b6c6a625ff57e3028921f8ff59318ac"}, {file = "MarkupSafe-2.1.3-cp311-cp311-win32.whl", hash = "sha256:dd15ff04ffd7e05ffcb7fe79f1b98041b8ea30ae9234aed2a9168b5797c3effb"}, {file = "MarkupSafe-2.1.3-cp311-cp311-win_amd64.whl", hash = "sha256:134da1eca9ec0ae528110ccc9e48041e0828d79f24121a1a146161103c76e686"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:f698de3fd0c4e6972b92290a45bd9b1536bffe8c6759c62471efaa8acb4c37bc"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:aa57bd9cf8ae831a362185ee444e15a93ecb2e344c8e52e4d721ea3ab6ef1823"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffcc3f7c66b5f5b7931a5aa68fc9cecc51e685ef90282f4a82f0f5e9b704ad11"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:47d4f1c5f80fc62fdd7777d0d40a2e9dda0a05883ab11374334f6c4de38adffd"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1f67c7038d560d92149c060157d623c542173016c4babc0c1913cca0564b9939"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:9aad3c1755095ce347e26488214ef77e0485a3c34a50c5a5e2471dff60b9dd9c"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:14ff806850827afd6b07a5f32bd917fb7f45b046ba40c57abdb636674a8b559c"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8f9293864fe09b8149f0cc42ce56e3f0e54de883a9de90cd427f191c346eb2e1"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-win32.whl", hash = "sha256:715d3562f79d540f251b99ebd6d8baa547118974341db04f5ad06d5ea3eb8007"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-win_amd64.whl", hash = "sha256:1b8dd8c3fd14349433c79fa8abeb573a55fc0fdd769133baac1f5e07abf54aeb"}, {file = "MarkupSafe-2.1.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:8e254ae696c88d98da6555f5ace2279cf7cd5b3f52be2b5cf97feafe883b58d2"}, {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cb0932dc158471523c9637e807d9bfb93e06a95cbf010f1a38b98623b929ef2b"}, {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9402b03f1a1b4dc4c19845e5c749e3ab82d5078d16a2a4c2cd2df62d57bb0707"}, @@ -1082,13 +1098,13 @@ requests = ">=2.26" [[package]] name = "mkdocs-material-extensions" -version = "1.1.1" +version = "1.2" description = "Extension pack for Python Markdown and MkDocs Material." optional = false python-versions = ">=3.7" files = [ - {file = "mkdocs_material_extensions-1.1.1-py3-none-any.whl", hash = "sha256:e41d9f38e4798b6617ad98ca8f7f1157b1e4385ac1459ca1e4ea219b556df945"}, - {file = "mkdocs_material_extensions-1.1.1.tar.gz", hash = "sha256:9c003da71e2cc2493d910237448c672e00cefc800d3d6ae93d2fc69979e3bd93"}, + {file = "mkdocs_material_extensions-1.2-py3-none-any.whl", hash = "sha256:c767bd6d6305f6420a50f0b541b0c9966d52068839af97029be14443849fb8a1"}, + {file = "mkdocs_material_extensions-1.2.tar.gz", hash = "sha256:27e2d1ed2d031426a6e10d5ea06989d67e90bb02acd588bc5673106b5ee5eedf"}, ] [[package]] @@ -1145,12 +1161,13 @@ mkdocstrings = ">=0.20" [[package]] name = "mypy-extensions" -version = "0.4.4" -description = "Experimental type system extensions for programs checked with the mypy typechecker." +version = "1.0.0" +description = "Type system extensions for programs checked with the mypy type checker." optional = false -python-versions = ">=2.7" +python-versions = ">=3.5" files = [ - {file = "mypy_extensions-0.4.4.tar.gz", hash = "sha256:c8b707883a96efe9b4bb3aaf0dcc07e7e217d7d8368eec4db4049ee9e142f4fd"}, + {file = "mypy_extensions-1.0.0-py3-none-any.whl", hash = "sha256:4392f6c0eb8a5668a69e23d168ffa70f0be9ccfd32b5cc2d26a34ae5b844552d"}, + {file = "mypy_extensions-1.0.0.tar.gz", hash = "sha256:75dbf8955dc00442a438fc4d0666508a9a97b6bd41aa2f0ffe9d2f2725af0782"}, ] [[package]] @@ -1203,13 +1220,13 @@ six = "*" [[package]] name = "netaddr" -version = "0.8.0" +version = "0.9.0" description = "A network address manipulation library for Python" optional = false python-versions = "*" files = [ - {file = "netaddr-0.8.0-py2.py3-none-any.whl", hash = "sha256:9666d0232c32d2656e5e5f8d735f58fd6c7457ce52fc21c98d45f2af78f990ac"}, - {file = "netaddr-0.8.0.tar.gz", hash = "sha256:d6cc57c7a07b1d9d2e917aa8b36ae8ce61c35ba3fcd1b83ca31c5a0ee2b5a243"}, + {file = "netaddr-0.9.0-py3-none-any.whl", hash = "sha256:5148b1055679d2a1ec070c521b7db82137887fabd6d7e37f5199b44f775c3bb1"}, + {file = "netaddr-0.9.0.tar.gz", hash = "sha256:7b46fa9b1a2d71fd5de9e4a3784ef339700a53a08c8040f08baf5f1194da0128"}, ] [[package]] @@ -1249,23 +1266,19 @@ optionals = ["jsonschema (>=4.17.3,<5.0.0)", "napalm (>=4.0.0,<5.0.0)"] [[package]] name = "nornir" -version = "3.3.0" +version = "3.4.1" description = "Pluggable multi-threaded framework with inventory management to help operate collections of devices" optional = false -python-versions = ">=3.7,<4.0" +python-versions = ">=3.8,<4.0" files = [ - {file = "nornir-3.3.0-py3-none-any.whl", hash = "sha256:4590d96edb5044e6a9e6f84e15625d32932177a10654040f99e145d73b352479"}, - {file = "nornir-3.3.0.tar.gz", hash = "sha256:1c6fd283bcdff9972358b126703c0990e9076dff1dfdc211e3077d45ada937d5"}, + {file = "nornir-3.4.1-py3-none-any.whl", hash = "sha256:db079cb95e3baf855530f4f40cb6ee93f93e1bf3cb74ac08180546adb1b987b8"}, + {file = "nornir-3.4.1.tar.gz", hash = "sha256:82a90a3478a3890bef8ad51b256fa966e6e4ca326cbe20a230918ef907cf68c3"}, ] [package.dependencies] importlib-metadata = {version = ">=4,<5", markers = "python_version < \"3.10\""} -mypy_extensions = ">=0.4.1,<0.5.0" +mypy_extensions = ">=1.0.0,<2.0.0" "ruamel.yaml" = ">=0.17" -typing_extensions = ">=4.1,<5.0" - -[package.extras] -docs = ["jupyter (>=1,<2)", "nbsphinx (>=0.8,<0.9)", "pygments (>=2,<3)", "sphinx (>=4,<5)", "sphinx-issues (>=3.0,<4.0)", "sphinx_rtd_theme (>=1.0,<2.0)", "sphinxcontrib-napoleon (>=0.7,<0.8)"] [[package]] name = "nornir-jinja2" @@ -1459,47 +1472,47 @@ files = [ [[package]] name = "pydantic" -version = "1.10.12" +version = "1.10.13" description = "Data validation and settings management using python type hints" optional = false python-versions = ">=3.7" files = [ - {file = "pydantic-1.10.12-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:a1fcb59f2f355ec350073af41d927bf83a63b50e640f4dbaa01053a28b7a7718"}, - {file = "pydantic-1.10.12-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:b7ccf02d7eb340b216ec33e53a3a629856afe1c6e0ef91d84a4e6f2fb2ca70fe"}, - {file = "pydantic-1.10.12-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8fb2aa3ab3728d950bcc885a2e9eff6c8fc40bc0b7bb434e555c215491bcf48b"}, - {file = "pydantic-1.10.12-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:771735dc43cf8383959dc9b90aa281f0b6092321ca98677c5fb6125a6f56d58d"}, - {file = "pydantic-1.10.12-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:ca48477862372ac3770969b9d75f1bf66131d386dba79506c46d75e6b48c1e09"}, - {file = "pydantic-1.10.12-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:a5e7add47a5b5a40c49b3036d464e3c7802f8ae0d1e66035ea16aa5b7a3923ed"}, - {file = "pydantic-1.10.12-cp310-cp310-win_amd64.whl", hash = "sha256:e4129b528c6baa99a429f97ce733fff478ec955513630e61b49804b6cf9b224a"}, - {file = "pydantic-1.10.12-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b0d191db0f92dfcb1dec210ca244fdae5cbe918c6050b342d619c09d31eea0cc"}, - {file = "pydantic-1.10.12-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:795e34e6cc065f8f498c89b894a3c6da294a936ee71e644e4bd44de048af1405"}, - {file = "pydantic-1.10.12-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:69328e15cfda2c392da4e713443c7dbffa1505bc9d566e71e55abe14c97ddc62"}, - {file = "pydantic-1.10.12-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2031de0967c279df0d8a1c72b4ffc411ecd06bac607a212892757db7462fc494"}, - {file = "pydantic-1.10.12-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:ba5b2e6fe6ca2b7e013398bc7d7b170e21cce322d266ffcd57cca313e54fb246"}, - {file = "pydantic-1.10.12-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:2a7bac939fa326db1ab741c9d7f44c565a1d1e80908b3797f7f81a4f86bc8d33"}, - {file = "pydantic-1.10.12-cp311-cp311-win_amd64.whl", hash = "sha256:87afda5539d5140cb8ba9e8b8c8865cb5b1463924d38490d73d3ccfd80896b3f"}, - {file = "pydantic-1.10.12-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:549a8e3d81df0a85226963611950b12d2d334f214436a19537b2efed61b7639a"}, - {file = "pydantic-1.10.12-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:598da88dfa127b666852bef6d0d796573a8cf5009ffd62104094a4fe39599565"}, - {file = "pydantic-1.10.12-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ba5c4a8552bff16c61882db58544116d021d0b31ee7c66958d14cf386a5b5350"}, - {file = "pydantic-1.10.12-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:c79e6a11a07da7374f46970410b41d5e266f7f38f6a17a9c4823db80dadf4303"}, - {file = "pydantic-1.10.12-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:ab26038b8375581dc832a63c948f261ae0aa21f1d34c1293469f135fa92972a5"}, - {file = "pydantic-1.10.12-cp37-cp37m-win_amd64.whl", hash = "sha256:e0a16d274b588767602b7646fa05af2782576a6cf1022f4ba74cbb4db66f6ca8"}, - {file = "pydantic-1.10.12-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6a9dfa722316f4acf4460afdf5d41d5246a80e249c7ff475c43a3a1e9d75cf62"}, - {file = "pydantic-1.10.12-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:a73f489aebd0c2121ed974054cb2759af8a9f747de120acd2c3394cf84176ccb"}, - {file = "pydantic-1.10.12-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6b30bcb8cbfccfcf02acb8f1a261143fab622831d9c0989707e0e659f77a18e0"}, - {file = "pydantic-1.10.12-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2fcfb5296d7877af406ba1547dfde9943b1256d8928732267e2653c26938cd9c"}, - {file = "pydantic-1.10.12-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:2f9a6fab5f82ada41d56b0602606a5506aab165ca54e52bc4545028382ef1c5d"}, - {file = "pydantic-1.10.12-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:dea7adcc33d5d105896401a1f37d56b47d443a2b2605ff8a969a0ed5543f7e33"}, - {file = "pydantic-1.10.12-cp38-cp38-win_amd64.whl", hash = "sha256:1eb2085c13bce1612da8537b2d90f549c8cbb05c67e8f22854e201bde5d98a47"}, - {file = "pydantic-1.10.12-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:ef6c96b2baa2100ec91a4b428f80d8f28a3c9e53568219b6c298c1125572ebc6"}, - {file = "pydantic-1.10.12-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:6c076be61cd0177a8433c0adcb03475baf4ee91edf5a4e550161ad57fc90f523"}, - {file = "pydantic-1.10.12-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2d5a58feb9a39f481eda4d5ca220aa8b9d4f21a41274760b9bc66bfd72595b86"}, - {file = "pydantic-1.10.12-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e5f805d2d5d0a41633651a73fa4ecdd0b3d7a49de4ec3fadf062fe16501ddbf1"}, - {file = "pydantic-1.10.12-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:1289c180abd4bd4555bb927c42ee42abc3aee02b0fb2d1223fb7c6e5bef87dbe"}, - {file = "pydantic-1.10.12-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:5d1197e462e0364906cbc19681605cb7c036f2475c899b6f296104ad42b9f5fb"}, - {file = "pydantic-1.10.12-cp39-cp39-win_amd64.whl", hash = "sha256:fdbdd1d630195689f325c9ef1a12900524dceb503b00a987663ff4f58669b93d"}, - {file = "pydantic-1.10.12-py3-none-any.whl", hash = "sha256:b749a43aa51e32839c9d71dc67eb1e4221bb04af1033a32e3923d46f9effa942"}, - {file = "pydantic-1.10.12.tar.gz", hash = "sha256:0fe8a415cea8f340e7a9af9c54fc71a649b43e8ca3cc732986116b3cb135d303"}, + {file = "pydantic-1.10.13-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:efff03cc7a4f29d9009d1c96ceb1e7a70a65cfe86e89d34e4a5f2ab1e5693737"}, + {file = "pydantic-1.10.13-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:3ecea2b9d80e5333303eeb77e180b90e95eea8f765d08c3d278cd56b00345d01"}, + {file = "pydantic-1.10.13-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1740068fd8e2ef6eb27a20e5651df000978edce6da6803c2bef0bc74540f9548"}, + {file = "pydantic-1.10.13-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:84bafe2e60b5e78bc64a2941b4c071a4b7404c5c907f5f5a99b0139781e69ed8"}, + {file = "pydantic-1.10.13-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:bc0898c12f8e9c97f6cd44c0ed70d55749eaf783716896960b4ecce2edfd2d69"}, + {file = "pydantic-1.10.13-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:654db58ae399fe6434e55325a2c3e959836bd17a6f6a0b6ca8107ea0571d2e17"}, + {file = "pydantic-1.10.13-cp310-cp310-win_amd64.whl", hash = "sha256:75ac15385a3534d887a99c713aa3da88a30fbd6204a5cd0dc4dab3d770b9bd2f"}, + {file = "pydantic-1.10.13-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:c553f6a156deb868ba38a23cf0df886c63492e9257f60a79c0fd8e7173537653"}, + {file = "pydantic-1.10.13-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:5e08865bc6464df8c7d61439ef4439829e3ab62ab1669cddea8dd00cd74b9ffe"}, + {file = "pydantic-1.10.13-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e31647d85a2013d926ce60b84f9dd5300d44535a9941fe825dc349ae1f760df9"}, + {file = "pydantic-1.10.13-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:210ce042e8f6f7c01168b2d84d4c9eb2b009fe7bf572c2266e235edf14bacd80"}, + {file = "pydantic-1.10.13-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:8ae5dd6b721459bfa30805f4c25880e0dd78fc5b5879f9f7a692196ddcb5a580"}, + {file = "pydantic-1.10.13-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:f8e81fc5fb17dae698f52bdd1c4f18b6ca674d7068242b2aff075f588301bbb0"}, + {file = "pydantic-1.10.13-cp311-cp311-win_amd64.whl", hash = "sha256:61d9dce220447fb74f45e73d7ff3b530e25db30192ad8d425166d43c5deb6df0"}, + {file = "pydantic-1.10.13-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:4b03e42ec20286f052490423682016fd80fda830d8e4119f8ab13ec7464c0132"}, + {file = "pydantic-1.10.13-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f59ef915cac80275245824e9d771ee939133be38215555e9dc90c6cb148aaeb5"}, + {file = "pydantic-1.10.13-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5a1f9f747851338933942db7af7b6ee8268568ef2ed86c4185c6ef4402e80ba8"}, + {file = "pydantic-1.10.13-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:97cce3ae7341f7620a0ba5ef6cf043975cd9d2b81f3aa5f4ea37928269bc1b87"}, + {file = "pydantic-1.10.13-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:854223752ba81e3abf663d685f105c64150873cc6f5d0c01d3e3220bcff7d36f"}, + {file = "pydantic-1.10.13-cp37-cp37m-win_amd64.whl", hash = "sha256:b97c1fac8c49be29486df85968682b0afa77e1b809aff74b83081cc115e52f33"}, + {file = "pydantic-1.10.13-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:c958d053453a1c4b1c2062b05cd42d9d5c8eb67537b8d5a7e3c3032943ecd261"}, + {file = "pydantic-1.10.13-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:4c5370a7edaac06daee3af1c8b1192e305bc102abcbf2a92374b5bc793818599"}, + {file = "pydantic-1.10.13-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7d6f6e7305244bddb4414ba7094ce910560c907bdfa3501e9db1a7fd7eaea127"}, + {file = "pydantic-1.10.13-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d3a3c792a58e1622667a2837512099eac62490cdfd63bd407993aaf200a4cf1f"}, + {file = "pydantic-1.10.13-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:c636925f38b8db208e09d344c7aa4f29a86bb9947495dd6b6d376ad10334fb78"}, + {file = "pydantic-1.10.13-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:678bcf5591b63cc917100dc50ab6caebe597ac67e8c9ccb75e698f66038ea953"}, + {file = "pydantic-1.10.13-cp38-cp38-win_amd64.whl", hash = "sha256:6cf25c1a65c27923a17b3da28a0bdb99f62ee04230c931d83e888012851f4e7f"}, + {file = "pydantic-1.10.13-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:8ef467901d7a41fa0ca6db9ae3ec0021e3f657ce2c208e98cd511f3161c762c6"}, + {file = "pydantic-1.10.13-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:968ac42970f57b8344ee08837b62f6ee6f53c33f603547a55571c954a4225691"}, + {file = "pydantic-1.10.13-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9849f031cf8a2f0a928fe885e5a04b08006d6d41876b8bbd2fc68a18f9f2e3fd"}, + {file = "pydantic-1.10.13-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:56e3ff861c3b9c6857579de282ce8baabf443f42ffba355bf070770ed63e11e1"}, + {file = "pydantic-1.10.13-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:9f00790179497767aae6bcdc36355792c79e7bbb20b145ff449700eb076c5f96"}, + {file = "pydantic-1.10.13-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:75b297827b59bc229cac1a23a2f7a4ac0031068e5be0ce385be1462e7e17a35d"}, + {file = "pydantic-1.10.13-cp39-cp39-win_amd64.whl", hash = "sha256:e70ca129d2053fb8b728ee7d1af8e553a928d7e301a311094b8a0501adc8763d"}, + {file = "pydantic-1.10.13-py3-none-any.whl", hash = "sha256:b87326822e71bd5f313e7d3bfdc77ac3247035ac10b0c0618bd99dcf95b1e687"}, + {file = "pydantic-1.10.13.tar.gz", hash = "sha256:32c8b48dcd3b2ac4e78b0ba4af3a2c2eb6048cb75202f0ea7b34feb740efc340"}, ] [package.dependencies] @@ -1571,17 +1584,17 @@ plugins = ["importlib-metadata"] [[package]] name = "pylint" -version = "2.17.5" +version = "2.17.6" description = "python code static checker" optional = false python-versions = ">=3.7.2" files = [ - {file = "pylint-2.17.5-py3-none-any.whl", hash = "sha256:73995fb8216d3bed149c8d51bba25b2c52a8251a2c8ac846ec668ce38fab5413"}, - {file = "pylint-2.17.5.tar.gz", hash = "sha256:f7b601cbc06fef7e62a754e2b41294c2aa31f1cb659624b9a85bcba29eaf8252"}, + {file = "pylint-2.17.6-py3-none-any.whl", hash = "sha256:18a1412e873caf8ffb56b760ce1b5643675af23e6173a247b502406b24c716af"}, + {file = "pylint-2.17.6.tar.gz", hash = "sha256:be928cce5c76bf9acdc65ad01447a1e0b1a7bccffc609fb7fc40f2513045bd05"}, ] [package.dependencies] -astroid = ">=2.15.6,<=2.17.0-dev0" +astroid = ">=2.15.7,<=2.17.0-dev0" colorama = {version = ">=0.4.5", markers = "sys_platform == \"win32\""} dill = [ {version = ">=0.2", markers = "python_version < \"3.11\""}, @@ -1776,6 +1789,7 @@ files = [ {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:69b023b2b4daa7548bcfbd4aa3da05b3a74b772db9e23b982788168117739938"}, {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:81e0b275a9ecc9c0c0c07b4b90ba548307583c125f54d5b6946cfee6360c733d"}, {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba336e390cd8e4d1739f42dfe9bb83a3cc2e80f567d8805e11b46f4a943f5515"}, + {file = "PyYAML-6.0.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:326c013efe8048858a6d312ddd31d56e468118ad4cdeda36c719bf5bb6192290"}, {file = "PyYAML-6.0.1-cp310-cp310-win32.whl", hash = "sha256:bd4af7373a854424dabd882decdc5579653d7868b8fb26dc7d0e99f823aa5924"}, {file = "PyYAML-6.0.1-cp310-cp310-win_amd64.whl", hash = "sha256:fd1592b3fdf65fff2ad0004b5e363300ef59ced41c2e6b3a99d4089fa8c5435d"}, {file = "PyYAML-6.0.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6965a7bc3cf88e5a1c3bd2e0b5c22f8d677dc88a455344035f03399034eb3007"}, @@ -1783,8 +1797,15 @@ files = [ {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:42f8152b8dbc4fe7d96729ec2b99c7097d656dc1213a3229ca5383f973a5ed6d"}, {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:062582fca9fabdd2c8b54a3ef1c978d786e0f6b3a1510e0ac93ef59e0ddae2bc"}, {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d2b04aac4d386b172d5b9692e2d2da8de7bfb6c387fa4f801fbf6fb2e6ba4673"}, + {file = "PyYAML-6.0.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:e7d73685e87afe9f3b36c799222440d6cf362062f78be1013661b00c5c6f678b"}, {file = "PyYAML-6.0.1-cp311-cp311-win32.whl", hash = "sha256:1635fd110e8d85d55237ab316b5b011de701ea0f29d07611174a1b42f1444741"}, {file = "PyYAML-6.0.1-cp311-cp311-win_amd64.whl", hash = "sha256:bf07ee2fef7014951eeb99f56f39c9bb4af143d8aa3c21b1677805985307da34"}, + {file = "PyYAML-6.0.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:855fb52b0dc35af121542a76b9a84f8d1cd886ea97c84703eaa6d88e37a2ad28"}, + {file = "PyYAML-6.0.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:40df9b996c2b73138957fe23a16a4f0ba614f4c0efce1e9406a184b6d07fa3a9"}, + {file = "PyYAML-6.0.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c22bec3fbe2524cde73d7ada88f6566758a8f7227bfbf93a408a9d86bcc12a0"}, + {file = "PyYAML-6.0.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8d4e9c88387b0f5c7d5f281e55304de64cf7f9c0021a3525bd3b1c542da3b0e4"}, + {file = "PyYAML-6.0.1-cp312-cp312-win32.whl", hash = "sha256:d483d2cdf104e7c9fa60c544d92981f12ad66a457afae824d146093b8c294c54"}, + {file = "PyYAML-6.0.1-cp312-cp312-win_amd64.whl", hash = "sha256:0d3304d8c0adc42be59c5f8a4d9e3d7379e6955ad754aa9d6ab7a398b59dd1df"}, {file = "PyYAML-6.0.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:50550eb667afee136e9a77d6dc71ae76a44df8b3e51e41b77f6de2932bfe0f47"}, {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1fe35611261b29bd1de0070f0b2f47cb6ff71fa6595c077e42bd0c419fa27b98"}, {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:704219a11b772aea0d8ecd7058d0082713c3562b4e271b849ad7dc4a5c90c13c"}, @@ -1801,6 +1822,7 @@ files = [ {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a0cd17c15d3bb3fa06978b4e8958dcdc6e0174ccea823003a106c7d4d7899ac5"}, {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:28c119d996beec18c05208a8bd78cbe4007878c6dd15091efb73a30e90539696"}, {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7e07cbde391ba96ab58e532ff4803f79c4129397514e1413a7dc761ccd755735"}, + {file = "PyYAML-6.0.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:49a183be227561de579b4a36efbb21b3eab9651dd81b1858589f796549873dd6"}, {file = "PyYAML-6.0.1-cp38-cp38-win32.whl", hash = "sha256:184c5108a2aca3c5b3d3bf9395d50893a7ab82a38004c8f61c258d4428e80206"}, {file = "PyYAML-6.0.1-cp38-cp38-win_amd64.whl", hash = "sha256:1e2722cc9fbb45d9b87631ac70924c11d3a401b2d7f410cc0e3bbf249f2dca62"}, {file = "PyYAML-6.0.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:9eb6caa9a297fc2c2fb8862bc5370d0303ddba53ba97e71f08023b6cd73d16a8"}, @@ -1808,6 +1830,7 @@ files = [ {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5773183b6446b2c99bb77e77595dd486303b4faab2b086e7b17bc6bef28865f6"}, {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b786eecbdf8499b9ca1d697215862083bd6d2a99965554781d0d8d1ad31e13a0"}, {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc1bf2925a1ecd43da378f4db9e4f799775d6367bdb94671027b73b393a7c42c"}, + {file = "PyYAML-6.0.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:04ac92ad1925b2cff1db0cfebffb6ffc43457495c9b3c39d3fcae417d7125dc5"}, {file = "PyYAML-6.0.1-cp39-cp39-win32.whl", hash = "sha256:faca3bdcf85b2fc05d06ff3fbc1f83e1391b3e724afa3feba7d13eeab355484c"}, {file = "PyYAML-6.0.1-cp39-cp39-win_amd64.whl", hash = "sha256:510c9deebc5c0225e8c96813043e62b680ba2f9c50a08d3724c7f28a747d1486"}, {file = "PyYAML-6.0.1.tar.gz", hash = "sha256:bfdf460b1736c775f2ba9f6a92bca30bc2095067b8a9d77876d1fad6cc3b4a43"}, @@ -1981,13 +2004,13 @@ test = ["fixtures", "mock", "purl", "pytest", "requests-futures", "sphinx", "tes [[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] @@ -2014,13 +2037,13 @@ six = "*" [[package]] name = "ruamel-yaml" -version = "0.17.32" +version = "0.17.33" description = "ruamel.yaml is a YAML parser/emitter that supports roundtrip preservation of comments, seq/map flow style, and map key order" optional = false python-versions = ">=3" files = [ - {file = "ruamel.yaml-0.17.32-py3-none-any.whl", hash = "sha256:23cd2ed620231677564646b0c6a89d138b6822a0d78656df7abda5879ec4f447"}, - {file = "ruamel.yaml-0.17.32.tar.gz", hash = "sha256:ec939063761914e14542972a5cba6d33c23b0859ab6342f61cf070cfc600efc2"}, + {file = "ruamel.yaml-0.17.33-py3-none-any.whl", hash = "sha256:2080c7a02b8a30fb3c06727cdf3e254a64055eedf3aa2d17c2b669639c04971b"}, + {file = "ruamel.yaml-0.17.33.tar.gz", hash = "sha256:5c56aa0bff2afceaa93bffbfc78b450b7dc1e01d5edb80b3a570695286ae62b1"}, ] [package.dependencies] @@ -2044,7 +2067,8 @@ files = [ {file = "ruamel.yaml.clib-0.2.7-cp310-cp310-win32.whl", hash = "sha256:763d65baa3b952479c4e972669f679fe490eee058d5aa85da483ebae2009d231"}, {file = "ruamel.yaml.clib-0.2.7-cp310-cp310-win_amd64.whl", hash = "sha256:d000f258cf42fec2b1bbf2863c61d7b8918d31ffee905da62dede869254d3b8a"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:045e0626baf1c52e5527bd5db361bc83180faaba2ff586e763d3d5982a876a9e"}, - {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-macosx_12_6_arm64.whl", hash = "sha256:721bc4ba4525f53f6a611ec0967bdcee61b31df5a56801281027a3a6d1c2daf5"}, + {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-macosx_13_0_arm64.whl", hash = "sha256:1a6391a7cabb7641c32517539ca42cf84b87b667bad38b78d4d42dd23e957c81"}, + {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-manylinux2014_aarch64.whl", hash = "sha256:9c7617df90c1365638916b98cdd9be833d31d337dbcd722485597b43c4a215bf"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:41d0f1fa4c6830176eef5b276af04c89320ea616655d01327d5ce65e50575c94"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-win32.whl", hash = "sha256:f6d3d39611ac2e4f62c3128a9eed45f19a6608670c5a2f4f07f24e8de3441d38"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-win_amd64.whl", hash = "sha256:da538167284de58a52109a9b89b8f6a53ff8437dd6dc26d33b57bf6699153122"}, @@ -2091,19 +2115,19 @@ paramiko = "*" [[package]] name = "setuptools" -version = "68.2.0" +version = "68.2.2" description = "Easily download, build, install, upgrade, and uninstall Python packages" optional = false python-versions = ">=3.8" files = [ - {file = "setuptools-68.2.0-py3-none-any.whl", hash = "sha256:af3d5949030c3f493f550876b2fd1dd5ec66689c4ee5d5344f009746f71fd5a8"}, - {file = "setuptools-68.2.0.tar.gz", hash = "sha256:00478ca80aeebeecb2f288d3206b0de568df5cd2b8fada1209843cc9a8d88a48"}, + {file = "setuptools-68.2.2-py3-none-any.whl", hash = "sha256:b454a35605876da60632df1a60f736524eb73cc47bbc9f3f1ef1b644de74fd2a"}, + {file = "setuptools-68.2.2.tar.gz", hash = "sha256:4ac1475276d2f1c48684874089fefcd83bd7162ddaafb81fac866ba0db282a87"}, ] [package.extras] docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-hoverxref (<2)", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"] testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pip (>=19.1)", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-ruff", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] -testing-integration = ["build[virtualenv]", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] +testing-integration = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging (>=23.1)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] [[package]] name = "six" @@ -2118,13 +2142,13 @@ files = [ [[package]] name = "smmap" -version = "5.0.0" +version = "5.0.1" description = "A pure Python implementation of a sliding window memory map manager" optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" files = [ - {file = "smmap-5.0.0-py3-none-any.whl", hash = "sha256:2aba19d6a040e78d8b09de5c57e96207b09ed71d8e55ce0959eeee6c8e190d94"}, - {file = "smmap-5.0.0.tar.gz", hash = "sha256:c840e62059cd3be204b0c9c9f74be2c09d5648eddd4580d9314c3ecde0b30936"}, + {file = "smmap-5.0.1-py3-none-any.whl", hash = "sha256:e6d8668fa5f93e706934a62d7b4db19c8d9eb8cf2adbb75ef1b675aa332b69da"}, + {file = "smmap-5.0.1.tar.gz", hash = "sha256:dceeb6c0028fdb6734471eb07c0cd2aae706ccaecab45965ee83f11c8d3b1f62"}, ] [[package]] @@ -2288,13 +2312,13 @@ docs = ["mkdocs (==1.2.4)", "mkdocs-material (==7.2.2)", "mkdocs-material-extens [[package]] name = "typing-extensions" -version = "4.7.1" -description = "Backported and Experimental Type Hints for Python 3.7+" +version = "4.8.0" +description = "Backported and Experimental Type Hints for Python 3.8+" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "typing_extensions-4.7.1-py3-none-any.whl", hash = "sha256:440d5dd3af93b060174bf433bccd69b0babc3b15b1a8dca43789fd7f61514b36"}, - {file = "typing_extensions-4.7.1.tar.gz", hash = "sha256:b75ddc264f0ba5615db7ba217daeb99701ad295353c45f9e95963337ceeeffb2"}, + {file = "typing_extensions-4.8.0-py3-none-any.whl", hash = "sha256:8f92fc8806f9a6b641eaa5318da32b44d401efaac0f6678c9bc448ba3605faa0"}, + {file = "typing_extensions-4.8.0.tar.gz", hash = "sha256:df8e4339e9cb77357558cbdbceca33c303714cf861d1eef15e1070055ae8b7ef"}, ] [[package]] @@ -2456,12 +2480,13 @@ dev = ["doc8", "flake8", "flake8-import-order", "rstcheck[sphinx]", "sphinx"] [[package]] name = "yamlordereddictloader" -version = "0.4.0" -description = "YAML loader and dump for PyYAML allowing to keep keys order." +version = "0.4.2" +description = "YAML loader and dumper for PyYAML allowing to keep keys order." optional = false python-versions = "*" files = [ - {file = "yamlordereddictloader-0.4.0.tar.gz", hash = "sha256:7f30f0b99ea3f877f7cb340c570921fa9d639b7f69cba18be051e27f8de2080e"}, + {file = "yamlordereddictloader-0.4.2-py3-none-any.whl", hash = "sha256:dc048adb67026786cd24119bd71241f35bc8b0fd37d24b415c37bbc8049f9cd7"}, + {file = "yamlordereddictloader-0.4.2.tar.gz", hash = "sha256:36af2f6210fcff5da4fc4c12e1d815f973dceb41044e795e1f06115d634bca13"}, ] [package.dependencies] @@ -2469,17 +2494,17 @@ pyyaml = "*" [[package]] name = "zipp" -version = "3.16.2" +version = "3.17.0" description = "Backport of pathlib-compatible object wrapper for zip files" optional = false python-versions = ">=3.8" files = [ - {file = "zipp-3.16.2-py3-none-any.whl", hash = "sha256:679e51dd4403591b2d6838a48de3d283f3d188412a9782faadf845f298736ba0"}, - {file = "zipp-3.16.2.tar.gz", hash = "sha256:ebc15946aa78bd63458992fc81ec3b6f7b1e92d51c35e6de1c3804e73b799147"}, + {file = "zipp-3.17.0-py3-none-any.whl", hash = "sha256:0e923e726174922dce09c53c59ad483ff7bbb8e572e00c7f7c46b88556409f31"}, + {file = "zipp-3.17.0.tar.gz", hash = "sha256:84e64a1c28cf7e91ed2078bb8cc8c259cb19b76942096c8d7b84947690cabaf0"}, ] [package.extras] -docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-lint"] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-lint"] testing = ["big-O", "jaraco.functools", "jaraco.itertools", "more-itertools", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-ignore-flaky", "pytest-mypy (>=0.9.1)", "pytest-ruff"] [extras] From d46af3328d28a384ec37a5b92548620df3f44ac2 Mon Sep 17 00:00:00 2001 From: itdependsnetworks Date: Sat, 30 Sep 2023 14:17:39 -0400 Subject: [PATCH 11/13] Add migration docs --- docs/dev/migrating_to_3.md | 17 +++++++++++++++++ docs/task/task.md | 2 +- mkdocs.yml | 1 + 3 files changed, 19 insertions(+), 1 deletion(-) create mode 100644 docs/dev/migrating_to_3.md diff --git a/docs/dev/migrating_to_3.md b/docs/dev/migrating_to_3.md new file mode 100644 index 0000000..2d73216 --- /dev/null +++ b/docs/dev/migrating_to_3.md @@ -0,0 +1,17 @@ +# Migration Guide to 3.0 + +This is intended to document the changes and considerations for migration to Nornir-Nautobot 3.0. **Please Note** this is nornir-nautobot's version number, not Nautobot's version number. This major upgrade did coincide with the release of Nautobot 2.0, but the version trains are on their own path. + +## Inventory Update + +The inventory has been updated to reflect the changes in Nautobot, specifically that `Region` and `Site` have been collapsed into Location. Please see the [inventory docs](../inventory/inventory.md) for the latest documentation. + +## Dispatcher Update + +The dispatcher function signature has substantially changed. Overall, this will allow for easier configurations to end users and specifically users of Nautobot's Golden Config plugin. Please see the relevant [dispatcher docs](../task/task.md#dispatcher-sender) for the latest documentation. + +The dispatcher now prefers senders in the following order, in which `framework` examples would be `netmiko` or `napalm` and `network_driver` examples would be `cisco_ios` or `arista_eos`. + +- If there is a custom_dispatcher, **only** use that +- Check for the `framework` and `network_driver` +- Check for the `framework`'s default \ No newline at end of file diff --git a/docs/task/task.md b/docs/task/task.md index 91f9e88..f3a2298 100644 --- a/docs/task/task.md +++ b/docs/task/task.md @@ -17,7 +17,7 @@ This may seem like a lot, but it essentially can be broken down to: - Check for the `framework` and `network_driver` - Check for the `framework`'s default -For completeness here is the referenced code as of SEP-2023. +For completeness here is the referenced code as of September 2023. ```python if not kwargs.get("custom_dispatcher"): diff --git a/mkdocs.yml b/mkdocs.yml index a94ea97..899f084 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -99,4 +99,5 @@ nav: - Developer Guide: - Contributing: "dev/CONTRIBUTING.md" - Changelog: "dev/CHANGELOG.md" + - Migration Guide 2.x -> 3.0: "dev/migrating_to_3.md" - Nautobot Docs Home ↗︎: "https://docs.nautobot.com/" From ae73395c5bd37e6aad750fa32c6088f5805d5b7e Mon Sep 17 00:00:00 2001 From: itdependsnetworks Date: Sat, 30 Sep 2023 16:33:35 -0400 Subject: [PATCH 12/13] Update migration guide name to be more standard --- docs/dev/{migrating_to_3.md => migrating_to_v3.md} | 2 +- mkdocs.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) rename docs/dev/{migrating_to_3.md => migrating_to_v3.md} (97%) diff --git a/docs/dev/migrating_to_3.md b/docs/dev/migrating_to_v3.md similarity index 97% rename from docs/dev/migrating_to_3.md rename to docs/dev/migrating_to_v3.md index 2d73216..2bf4f7d 100644 --- a/docs/dev/migrating_to_3.md +++ b/docs/dev/migrating_to_v3.md @@ -1,4 +1,4 @@ -# Migration Guide to 3.0 +# Migrating to v3 This is intended to document the changes and considerations for migration to Nornir-Nautobot 3.0. **Please Note** this is nornir-nautobot's version number, not Nautobot's version number. This major upgrade did coincide with the release of Nautobot 2.0, but the version trains are on their own path. diff --git a/mkdocs.yml b/mkdocs.yml index 899f084..3b8b5dd 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -99,5 +99,5 @@ nav: - Developer Guide: - Contributing: "dev/CONTRIBUTING.md" - Changelog: "dev/CHANGELOG.md" - - Migration Guide 2.x -> 3.0: "dev/migrating_to_3.md" + - Migrating to v3: "dev/migrating_to_v3.md" - Nautobot Docs Home ↗︎: "https://docs.nautobot.com/" From 458a4517ca2fc6fccf92934779c08a06d48b6a90 Mon Sep 17 00:00:00 2001 From: itdependsnetworks Date: Sun, 1 Oct 2023 16:35:28 -0400 Subject: [PATCH 13/13] Fix custom dispatcher, make error message more visible, add error code links to other repos --- docs/task/task.md | 17 +++++---- docs/task/troubleshooting/index.md | 10 ++++- .../plugins/tasks/dispatcher/__init__.py | 19 ++++++---- .../plugins/tasks/dispatcher/default.py | 38 +++++++++---------- .../tasks/dispatcher/mikrotik_routeros.py | 16 ++++---- .../tasks/dispatcher/ruckus_fastiron.py | 4 +- .../tasks/dispatcher/ruckus_smartzone.py | 6 +-- 7 files changed, 62 insertions(+), 48 deletions(-) diff --git a/docs/task/task.md b/docs/task/task.md index f3a2298..d8f0d99 100644 --- a/docs/task/task.md +++ b/docs/task/task.md @@ -17,26 +17,29 @@ This may seem like a lot, but it essentially can be broken down to: - Check for the `framework` and `network_driver` - Check for the `framework`'s default -For completeness here is the referenced code as of September 2023. +For completeness here is the referenced code as of October 2023. ```python - if not kwargs.get("custom_dispatcher"): - custom_dispatcher = {} + custom_dispatcher = "" + if kwargs.get("custom_dispatcher"): + custom_dispatcher = kwargs["custom_dispatcher"] + del kwargs["custom_dispatcher"] + logger.debug(f"Dispatcher process started for {task.host.name} ({task.host.platform})") network_driver = task.host.platform network_driver_title = snake_to_title_case(network_driver) - custom_dispatcher_path = [custom_dispatcher.get(network_driver)] framework_path = ( f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework.title()}{network_driver_title}" ) framework_default_path = f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework.title()}Default" - if custom_dispatcher.get(network_driver): - driver_class = import_string(custom_dispatcher_path) - checked_path = [custom_dispatcher_path] + if custom_dispatcher: + driver_class = import_string(custom_dispatcher) + checked_path = [custom_dispatcher] elif import_string(framework_path): driver_class = import_string(framework_path) + checked_path = [framework_path] else: driver_class = import_string(framework_default_path) checked_path = [framework_path, framework_default_path] diff --git a/docs/task/troubleshooting/index.md b/docs/task/troubleshooting/index.md index 4adcabd..1f99469 100644 --- a/docs/task/troubleshooting/index.md +++ b/docs/task/troubleshooting/index.md @@ -2,4 +2,12 @@ In an effort to help with troubleshooting, each expected error, will now emit an error ID, in the format of `E1XXX`, such as `E1006: There was no password defined, preemptively failed.`. The idea will be to define the error, the error message and some recommended troubleshooting steps or even potentially some fixes. -This is an ongoing effort, but the foundation has been built. \ No newline at end of file +This is an ongoing effort, but the foundation has been built. + +Within the Nautobot ecosystem, you may see various errors, they are distributed between 3 libraries as followed. + +| Error Range | Plugin Docs | +| ----------- | ----------- | +| E1001-E1999 | [Nornir Nautobot](https://docs.nautobot.com/projects/nornir-nautobot/en/latest/task/troubleshooting/) | +| E2001-E2999 | [Nautobot Plugin Nornir](https://docs.nautobot.com/projects/plugin-nornir/en/latest/admin/troubleshooting/) | +| E3001-E3999 | [Nautobot Golden Config](https://docs.nautobot.com/projects/golden-config/en/latest/admin/troubleshooting/) | \ No newline at end of file diff --git a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py index f4aabb7..090cd74 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/__init__.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/__init__.py @@ -23,36 +23,39 @@ def dispatcher( # pylint: disable=too-many-arguments,too-many-locals Returns: Result: Nornir Task result object. """ - if not kwargs.get("custom_dispatcher"): - custom_dispatcher = {} + custom_dispatcher = "" + if kwargs.get("custom_dispatcher"): + custom_dispatcher = kwargs["custom_dispatcher"] + del kwargs["custom_dispatcher"] + logger.debug(f"Dispatcher process started for {task.host.name} ({task.host.platform})") network_driver = task.host.platform network_driver_title = snake_to_title_case(network_driver) - custom_dispatcher_path = [custom_dispatcher.get(network_driver)] framework_path = ( f"nornir_nautobot.plugins.tasks.dispatcher.{network_driver}.{framework.title()}{network_driver_title}" ) framework_default_path = f"nornir_nautobot.plugins.tasks.dispatcher.default.{framework.title()}Default" - if custom_dispatcher.get(network_driver): - driver_class = import_string(custom_dispatcher_path) - checked_path = [custom_dispatcher_path] + if custom_dispatcher: + driver_class = import_string(custom_dispatcher) + checked_path = [custom_dispatcher] elif import_string(framework_path): driver_class = import_string(framework_path) + checked_path = [framework_path] else: driver_class = import_string(framework_default_path) checked_path = [framework_path, framework_default_path] if not driver_class: - error_msg = f"E1001: Did not find a valid dispatcher in {checked_path}, preemptively failed." + error_msg = f"`E1001:` Did not find a valid dispatcher in {checked_path}, preemptively failed." logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) try: driver_task = getattr(driver_class, method) except AttributeError: - error_msg = f"E1002: Unable to locate the method {method} for {driver_class}, preemptively failed." + error_msg = f"`E1002:` Unable to locate the method {method} for {driver_class}, preemptively failed." logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/default.py b/nornir_nautobot/plugins/tasks/dispatcher/default.py index 1329bb0..e6b32d3 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/default.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/default.py @@ -66,7 +66,7 @@ def check_connectivity(cls, task: Task, logger, obj) -> Result: else: if not is_fqdn_resolvable(hostname): error_msg = ( - f"E1003: The hostname {hostname} did not have an IP nor was resolvable, preemptively failed." + f"`E1003:` The hostname {hostname} did not have an IP nor was resolvable, preemptively failed." ) logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -79,15 +79,15 @@ def check_connectivity(cls, task: Task, logger, obj) -> Result: except socket.error: _tcp_ping = False if not _tcp_ping: - error_msg = f"E1004: Could not connect to IP: `{ip_addr}` and port: `{port}`, preemptively failed." + error_msg = f"`E1004:` Could not connect to IP: `{ip_addr}` and port: `{port}`, preemptively failed." logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if not task.host.username: - error_msg = "E1005: There was no username defined, preemptively failed." + error_msg = "`E1005:` There was no username defined, preemptively failed." logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if not task.host.password: - error_msg = "E1006: There was no password defined, preemptively failed." + error_msg = "`E1006:` There was no password defined, preemptively failed." logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -119,19 +119,19 @@ def compliance_config( Result: Nornir Result object with a feature_data key of the compliance data. """ if not os.path.exists(backup_file): - error_msg = f"E1007: Backup file Not Found at location: `{backup_file}`, preemptively failed." + error_msg = f"`E1007:` Backup file Not Found at location: `{backup_file}`, preemptively failed." logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if not os.path.exists(intended_file): - error_msg = f"E1008: Intended config file NOT Found at location: `{intended_file}`, preemptively failed." + error_msg = f"`E1008:` Intended config file NOT Found at location: `{intended_file}`, preemptively failed." logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) try: feature_data = compliance(features, backup_file, intended_file, platform) except Exception as error: # pylint: disable=broad-except - error_msg = f"E1009: UNKNOWN Failure of: {str(error)}" + error_msg = f"`E1009:` UNKNOWN Failure of: {str(error)}" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) return Result(host=task.host, result={"feature_data": feature_data}) @@ -175,27 +175,27 @@ def generate_config( except NornirSubTaskError as exc: if isinstance(exc.result.exception, jinja2.exceptions.UndefinedError): # pylint: disable=no-else-raise error_msg = ( - f"E1010: There was a jinja2.exceptions.UndefinedError error: ``{str(exc.result.exception)}``" + f"`E1010:` There was a jinja2.exceptions.UndefinedError error: ``{str(exc.result.exception)}``" ) logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) elif isinstance(exc.result.exception, jinja2.TemplateSyntaxError): - error_msg = (f"E1011: There was a jinja2.TemplateSyntaxError error: ``{str(exc.result.exception)}``",) + error_msg = (f"`E1011:` There was a jinja2.TemplateSyntaxError error: ``{str(exc.result.exception)}``",) logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) elif isinstance(exc.result.exception, jinja2.TemplateNotFound): - error_msg = f"E1012: There was an issue finding the template and a jinja2.TemplateNotFound error was raised: ``{str(exc.result.exception)}``" + error_msg = f"`E1012:` There was an issue finding the template and a jinja2.TemplateNotFound error was raised: ``{str(exc.result.exception)}``" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) elif isinstance(exc.result.exception, jinja2.TemplateError): - error_msg = f"E1013: There was an issue general Jinja error: ``{str(exc.result.exception)}``" + error_msg = f"`E1013:` There was an issue general Jinja error: ``{str(exc.result.exception)}``" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) - error_msg = f"E1014: Failed with an unknown issue. `{exc.result.exception}`" + error_msg = f"`E1014:` Failed with an unknown issue. `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -289,7 +289,7 @@ def get_config( try: result = task.run(task=napalm_get, getters=["config"], retrieve="running") except NornirSubTaskError as exc: - error_msg = f"E1015: `get_config` method failed with an unexpected issue: `{exc.result.exception}`" + error_msg = f"`E1015:` `get_config` method failed with an unexpected issue: `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -355,7 +355,7 @@ def replace_config( revert_in=revert_in, ) except NornirSubTaskError as exc: - error_msg = f"E1015: Failed with an unknown issue. `{exc.result.exception}`" + error_msg = f"`E1015:` Failed with an unknown issue. `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -407,7 +407,7 @@ def merge_config( revert_in=revert_in, ) except NornirSubTaskError as exc: - error_msg = f"E1015: Failed with an unknown issue. `{exc.result.exception}`" + error_msg = f"`E1015:` Failed with an unknown issue. `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -457,16 +457,16 @@ def get_config( result = task.run(task=netmiko_send_command, command_string=command) except NornirSubTaskError as exc: if isinstance(exc.result.exception, NetmikoAuthenticationException): - error_msg = f"E1017: Failed with an authentication issue: `{exc.result.exception}`" + error_msg = f"`E1017:` Failed with an authentication issue: `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if isinstance(exc.result.exception, NetmikoTimeoutException): - error_msg = f"E1018: Failed with a timeout issue. `{exc.result.exception}`" + error_msg = f"`E1018:` Failed with a timeout issue. `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) - error_msg = f"E1016: Failed with an unknown issue. `{exc.result.exception}`" + error_msg = f"`E1016:` Failed with an unknown issue. `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -477,7 +477,7 @@ def get_config( # Primarily seen in Cisco devices. if "ERROR: % Invalid input detected at" in running_config: - error_msg = "E1019: Discovered `ERROR: % Invalid input detected at` in the output" + error_msg = "`E1019:` Discovered `ERROR: % Invalid input detected at` in the output" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py b/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py index d58e81e..2336729 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/mikrotik_routeros.py @@ -61,7 +61,7 @@ def get_config( # pylint: disable=R0913,R0914 """ logger.debug(f"Executing get_config for {task.host.name} on {task.host.platform}") if not routeros_api: - error_msg = "E1020: The `routeros_api` is not installed in this environment." + error_msg = "`E1020:` The `routeros_api` is not installed in this environment." logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -79,7 +79,7 @@ def get_config( # pylint: disable=R0913,R0914 try: api = connection.get_api() except Exception as error: - error_msg = f"E1021: `get_config` method failed with an unexpected issue: `{error}`" + error_msg = f"`E1021:` The `get_config` method failed with an unexpected issue: `{error}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) for endpoint in cls.config_command: @@ -87,7 +87,7 @@ def get_config( # pylint: disable=R0913,R0914 resource = api.get_resource(endpoint) config_data[endpoint] = resource.get() except Exception as error: - error_msg = f"E1022: `get_config` method failed with an unexpected issue: `{error}`" + error_msg = f"`E1022:` The `get_config` method failed with an unexpected issue: `{error}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -139,16 +139,16 @@ def get_config( # pylint: disable=R0913,R0914 result = task.run(task=netmiko_send_command, command_string=cls.version_command) except NornirSubTaskError as exc: if isinstance(exc.result.exception, NetmikoAuthenticationException): - error_msg = f"E1017: Failed with an authentication issue: `{exc.result.exception}`" + error_msg = f"`E1017:` Failed with an authentication issue: `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) if isinstance(exc.result.exception, NetmikoTimeoutException): - error_msg = f"E1018: Failed with a timeout issue. `{exc.result.exception}`" + error_msg = f"`E1018:` Failed with a timeout issue. `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) - error_msg = f"E1016: Failed with an unknown issue. `{exc.result.exception}`" + error_msg = f"`E1016:` Failed with an unknown issue. `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -215,13 +215,13 @@ def merge_config( config_commands=config_list, ) except NornirSubTaskError as exc: - error_msg = f"E1015 Failed with error: `{exc.result.exception}`" + error_msg = f"`E1015 `Failed with error: `{exc.result.exception}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) from exc if any(msg in push_result[0].result.lower() for msg in NETMIKO_FAIL_MSG): logger.warning("Config merged with errors, please check full info log below.", extra={"object": obj}) - error_msg = f"E1026: result: {push_result[0].result}" + error_msg = f"`E1026:` result: {push_result[0].result}" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) logger.info("Config merged successfully.", extra={"object": obj}) diff --git a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py index 414c135..428d413 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_fastiron.py @@ -46,7 +46,7 @@ def merge_config(task: Task, logger, obj, config: str) -> Result: if any(msg in push_result[0].result.lower() for msg in NETMIKO_FAIL_MSG): logger.warning("Config merged with errors, please check full info log below.", extra={"object": obj}) - error_msg = f"E1026: result: {push_result[0].result}" + error_msg = f"`E1026:` result: {push_result[0].result}" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -59,7 +59,7 @@ def merge_config(task: Task, logger, obj, config: str) -> Result: ) logger.info(f"config saved: {save_result[0].result}", extra={"object": obj}) except NornirSubTaskError as exc: - error_msg = f"E1027: config merged, but failed to save: {exc.result.exception}" + error_msg = f"`E1027:` config merged, but failed to save: {exc.result.exception}" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) from exc push_result[0].changed = True diff --git a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py index fb2cfae..bfc3b0f 100644 --- a/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py +++ b/nornir_nautobot/plugins/tasks/dispatcher/ruckus_smartzone.py @@ -70,7 +70,7 @@ def _api_auth(cls, obj, logger, session_params: tuple) -> dict: service_ticket = data.get("serviceTicket") else: error_msg = ( - f"E1023: `_api_auth` method failed with an unexpected issue: HTTP Error `{response.status_code}`" + f"`E1023:` The `_api_auth` method failed with an unexpected issue: HTTP Error `{response.status_code}`" ) logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -109,7 +109,7 @@ def _build_urls( # pylint: disable=too-many-arguments,too-many-locals if response.status_code == 200: item_list = response.json().get("list") else: - error_msg = f"E1024: `{uri}` endpoint failed with code: HTTP Error `{response.status_code}`" + error_msg = f"`E1024:` The `{uri}` endpoint failed with code: HTTP Error `{response.status_code}`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg) @@ -117,7 +117,7 @@ def _build_urls( # pylint: disable=too-many-arguments,too-many-locals for uri_list_item in uri_list: url_dict[uri_list_item] = f"{base_url}{uri_list_item}?serviceTicket={token}" else: - error_msg = f"E1025: `{uri}` endpoint missing in simple endpoints list, schema invalid`" + error_msg = f"`E1025:` The `{uri}` endpoint missing in simple endpoints list, schema invalid`" logger.error(error_msg, extra={"object": obj}) raise NornirNautobotException(error_msg)