From d8959116d820cd5e16c913ba0318b4c7fc8d2971 Mon Sep 17 00:00:00 2001 From: Nat Noordanus Date: Sat, 26 Oct 2024 21:56:31 +0300 Subject: [PATCH] Add poethepoet.script package with rm function for use in script tasks (#250) Also: - add support for delegating dry-run behavior to script tasks - set POE_VERBOSITY env var --- docs/env_vars.rst | 2 +- docs/tasks/task_types/script.rst | 17 + poethepoet/context.py | 11 +- poethepoet/env/manager.py | 3 + poethepoet/helpers/python.py | 91 ++++- poethepoet/scripts/__init__.py | 3 + poethepoet/scripts/_rm.py | 61 +++ poethepoet/task/base.py | 9 +- poethepoet/task/expr.py | 1 - poethepoet/task/script.py | 45 ++- poetry.lock | 363 +++++++++--------- pyproject.toml | 172 +++++---- tests/conftest.py | 7 + tests/fixtures/example_project/pyproject.toml | 5 + tests/fixtures/scripts_project/poetry.lock | 7 + tests/fixtures/scripts_project/pyproject.toml | 3 + tests/test_cli.py | 49 ++- tests/test_poe_config.py | 32 +- tests/test_script_tasks.py | 20 +- tests/test_scripts.py | 171 +++++++++ 20 files changed, 759 insertions(+), 313 deletions(-) create mode 100644 poethepoet/scripts/__init__.py create mode 100644 poethepoet/scripts/_rm.py create mode 100644 tests/fixtures/scripts_project/poetry.lock create mode 100644 tests/test_scripts.py diff --git a/docs/env_vars.rst b/docs/env_vars.rst index 345aadbef..3c67c5f82 100644 --- a/docs/env_vars.rst +++ b/docs/env_vars.rst @@ -10,7 +10,7 @@ The following environment variables are used by Poe the Poet internally, and can - ``POE_PWD``: the current working directory of the poe process (unless overridden programmatically). - ``POE_CONF_DIR``: the path to the parent directory of the config file that defines the running task or the :ref:`cwd option` set when including that config. - ``POE_ACTIVE``: identifies the active PoeExecutor, so that Poe the Poet can tell when it is running recursively. - +- ``POE_VERBOSITY``: reflects the current verbosity level. Normally 0 is the default, 1 means more verbose and -1 means less. Special variables ----------------- diff --git a/docs/tasks/task_types/script.rst b/docs/tasks/task_types/script.rst index 26b24d44f..d2b0cf846 100644 --- a/docs/tasks/task_types/script.rst +++ b/docs/tasks/task_types/script.rst @@ -68,3 +68,20 @@ whereas the same behaviour can can be reliably achieved like so: [tool.poe.tasks.build] script = "os:makedirs('build/assets', exist_ok=True)" + + +Poe scripts library +------------------- + +Poe the Poet includes the ``poethepoet.scripts`` package including the following functions as convenient cross-platform implementations of common task capabilities. +These functions can be referenced from script tasks if ``poethepoet`` is available in the project virtual environment. + +.. autofunction:: poethepoet.scripts.rm + + +Delegating dry-run behavior to a script +--------------------------------------- + +Normally if the ``--dry-run`` global option is passed to the CLI then poe will go through the motions of running the given task, including logging to stdout, without actually running the task. + +However it is possible to configure poe to delegate respecting this dry run flag to an invoked script task, by passing it the ``_dry_run`` variable. When this variable is passed as an argument to the python function called within a script task then poe will always call the task, and delegate responsibility to the script for making sure that no side effects occur when run in dry-run mode. diff --git a/poethepoet/context.py b/poethepoet/context.py index 7b65c3032..901a2ad32 100644 --- a/poethepoet/context.py +++ b/poethepoet/context.py @@ -90,9 +90,18 @@ def get_executor( invocation: Tuple[str, ...], env: "EnvVarsManager", working_dir: Path, + *, executor_config: Optional[Mapping[str, str]] = None, capture_stdout: Union[str, bool] = False, + delegate_dry_run: bool = False, ) -> "PoeExecutor": + """ + Get an Executor object for use with this invocation. + + if delegate_dry_run is set then the task will always be executed and be + entrusted to not have any side effects when the dry-run flag is set. + """ + from .executor import PoeExecutor if not executor_config: @@ -108,5 +117,5 @@ def get_executor( env=env, working_dir=working_dir, capture_stdout=capture_stdout, - dry=self.dry, + dry=False if delegate_dry_run else self.dry, ) diff --git a/poethepoet/env/manager.py b/poethepoet/env/manager.py index d5658bb0a..566678086 100644 --- a/poethepoet/env/manager.py +++ b/poethepoet/env/manager.py @@ -47,6 +47,9 @@ def __init__( # TODO: check if we still need all these args! self._vars["POE_CWD"] = self.cwd self._vars["POE_PWD"] = self.cwd + if self._ui: + self._vars["POE_VERBOSITY"] = str(self._ui.verbosity) + self._git_repo = GitRepo(config.project_dir) def __getitem__(self, key): diff --git a/poethepoet/helpers/python.py b/poethepoet/helpers/python.py index 7fd28613e..ab78dae5c 100644 --- a/poethepoet/helpers/python.py +++ b/poethepoet/helpers/python.py @@ -6,7 +6,18 @@ import ast import re import sys -from typing import Any, Collection, Container, Dict, Iterator, List, Optional, Tuple +from typing import ( + Any, + Collection, + Container, + Dict, + Iterator, + List, + NamedTuple, + Optional, + Tuple, + cast, +) from ..exceptions import ExpressionParseError @@ -67,21 +78,77 @@ Substitution = Tuple[Tuple[int, int], str] +class FunctionCall(NamedTuple): + """ + Model for a python expression consisting of a function call + """ + + expression: str + function_ref: str + referenced_args: Tuple[str, ...] = tuple() + referenced_globals: Tuple[str, ...] = tuple() + + @classmethod + def parse( + cls, + source: str, + arguments: Container[str], + *, + args_prefix: str = "__args.", + allowed_vars: Container[str] = tuple(), + ) -> "FunctionCall": + root_node = cast(ast.Call, parse_and_validate(source, True, "script")) + name_nodes = _validate_nodes_and_get_names(root_node, source) + + substitutions: List[Substitution] = [] + referenced_args: List[str] = [] + referenced_globals: List[str] = [] + for node in name_nodes: + if node.id in arguments: + substitutions.append( + (_get_name_node_abs_range(source, node), args_prefix + node.id) + ) + referenced_args.append(node.id) + elif node.id in _ALLOWED_BUILTINS or node.id in allowed_vars: + referenced_globals.append(node.id) + else: + raise ExpressionParseError( + "Invalid variable reference in script: " + + _get_name_source_segment(source, node) + ) + + # Prefix references to arguments with args_prefix + expression = _apply_substitutions(source, substitutions) + + ref_parts = [] + func_node = root_node.func + while isinstance(func_node, ast.Attribute): + ref_parts.append(func_node.attr) + func_node = func_node.value + assert isinstance(func_node, ast.Name) + function_ref = ".".join((func_node.id, *reversed(ref_parts))) + + return cls( + expression=_clean_linebreaks(expression), + function_ref=function_ref, + referenced_args=tuple(referenced_args), + referenced_globals=tuple(referenced_globals), + ) + + def resolve_expression( source: str, arguments: Container[str], *, - call_only: bool = True, args_prefix: str = "__args.", allowed_vars: Container[str] = tuple(), -): +) -> str: """ Validate function call and substitute references to arguments with their namespaced counterparts (e.g. `my_arg` => `__args.my_arg`). """ - task_type = "script" if call_only else "expr" - root_node = parse_and_validate(source, call_only, task_type) + root_node = parse_and_validate(source, False, "expr") name_nodes = _validate_nodes_and_get_names(root_node, source) substitutions: List[Substitution] = [] @@ -92,12 +159,12 @@ def resolve_expression( ) elif node.id not in _ALLOWED_BUILTINS and node.id not in allowed_vars: raise ExpressionParseError( - f"Invalid variable reference in {task_type}: " + "Invalid variable reference in expr: " + _get_name_source_segment(source, node) ) # Prefix references to arguments with args_prefix - return _apply_substitutions(source, substitutions) + return _clean_linebreaks(_apply_substitutions(source, substitutions)) def parse_and_validate( @@ -246,7 +313,7 @@ def _validate_nodes_and_get_names( ) -def _apply_substitutions(content: str, subs: List[Substitution]): +def _apply_substitutions(content: str, subs: List[Substitution]) -> str: """ Returns a copy of content with all of the substitutions applied. Uses a single pass for efficiency. @@ -319,3 +386,11 @@ def _get_name_source_segment(source: str, node: ast.Name): partial_result = partial_result[:-1] return partial_result + + +def _clean_linebreaks(expression: str): + """ + Strip out any new lines because they can be problematic on windows + """ + expression = re.sub(r"((\r\n|\r|\n) | (\r\n|\r|\n))", " ", expression) + return re.sub(r"(\r\n|\r|\n)", " ", expression) diff --git a/poethepoet/scripts/__init__.py b/poethepoet/scripts/__init__.py new file mode 100644 index 000000000..8463f9ebf --- /dev/null +++ b/poethepoet/scripts/__init__.py @@ -0,0 +1,3 @@ +from ._rm import rm + +__all__ = ["rm"] diff --git a/poethepoet/scripts/_rm.py b/poethepoet/scripts/_rm.py new file mode 100644 index 000000000..41f9d9509 --- /dev/null +++ b/poethepoet/scripts/_rm.py @@ -0,0 +1,61 @@ +# ruff: noqa: E501 +import shutil +from pathlib import Path +from typing import Union + + +def rm( + *patterns: str, + cwd: str = ".", + verbosity: Union[int, str] = 0, + dry_run: bool = False, +): + """ + This function is intended for use in a script task to delete files and directories + matching any of the given patterns, as a platform agnostic alternative to + ``rm -rf [patterns]`` + + Example usage: + + .. code-block:: toml + + [tool.poe.tasks.clean] + script = "poethepoet.scripts:rm('.mypy_cache', '.pytest_cache', './**/__pycache__')" + + :param *patterns: + One or more paths to delete. + `Glob patterns `_ are supported. + :param cwd: + The directory relative to which patterns are evaluated. Defaults to ``.``. + :param verbosity: + An integer for setting the function's verbosity. This can be set to + ``environ.get('POE_VERBOSITY')`` to match the verbosity of the poe invocation. + :param dry_run: + If true then nothing will be deleted, but output to stdout will be unaffected. + This can be set to ``_dry_run`` to make poe delegate dry_run control to the + function. + """ + verbosity = int(verbosity) + + for pattern in patterns: + matches = list(Path(cwd).glob(pattern)) + if verbosity > 0 and not matches: + print(f"No files or directories to delete matching {pattern!r}") + elif verbosity >= 0 and len(matches) > 1: + print(f"Deleting paths matching {pattern!r}") + + for match in matches: + _delete_path(match, verbosity, dry_run) + + +def _delete_path(path: Path, verbosity: int, dry_run: bool): + if path.is_dir(): + if verbosity > 0: + print(f"Deleting directory '{path}'") + if not dry_run: + shutil.rmtree(path) + else: + if verbosity > 0: + print(f"Deleting file '{path}'") + if not dry_run: + path.unlink() diff --git a/poethepoet/task/base.py b/poethepoet/task/base.py index e35505c9b..1557f8677 100644 --- a/poethepoet/task/base.py +++ b/poethepoet/task/base.py @@ -485,13 +485,20 @@ def _handle_run( """ raise NotImplementedError - def _get_executor(self, context: "RunContext", env: "EnvVarsManager"): + def _get_executor( + self, + context: "RunContext", + env: "EnvVarsManager", + *, + delegate_dry_run: bool = False, + ): return context.get_executor( self.invocation, env, working_dir=self.get_working_dir(env), executor_config=self.spec.options.get("executor"), capture_stdout=self.capture_stdout, + delegate_dry_run=delegate_dry_run, ) def get_working_dir( diff --git a/poethepoet/task/expr.py b/poethepoet/task/expr.py index d65ed1754..969a142c3 100644 --- a/poethepoet/task/expr.py +++ b/poethepoet/task/expr.py @@ -124,7 +124,6 @@ def parse_content( expression = resolve_expression( source=expression, arguments=set(args or tuple()), - call_only=False, allowed_vars={"sys", "__env", *imports}, ) # Strip out any new lines because they can be problematic on windows diff --git a/poethepoet/task/script.py b/poethepoet/task/script.py index 8ed37445c..4009e3adb 100644 --- a/poethepoet/task/script.py +++ b/poethepoet/task/script.py @@ -1,4 +1,3 @@ -import re import shlex from typing import TYPE_CHECKING, Any, Dict, Optional, Tuple @@ -9,6 +8,7 @@ from ..config import PoeConfig from ..context import RunContext from ..env.manager import EnvVarsManager + from ..helpers.python import FunctionCall from .base import TaskSpecFactory @@ -68,7 +68,7 @@ def _handle_run( # TODO: do something about extra_args, error? target_module, function_call = self.parse_content(named_arg_values) - function_ref = function_call[: function_call.index("(")] + function_ref = function_call.function_ref argv = [ self.name, @@ -78,16 +78,20 @@ def _handle_run( # TODO: check whether the project really does use src layout, and don't do # sys.path.append('src') if it doesn't + has_dry_run_ref = "_dry_run" in function_call.referenced_globals + dry_run = self.ctx.ui["dry_run"] + script = [ "import asyncio,os,sys;", "from inspect import iscoroutinefunction as _c;", "from os import environ;", "from importlib import import_module as _i;", + f"_dry_run = {'True' if dry_run else 'False'};" if has_dry_run_ref else "", f"sys.argv = {argv!r}; sys.path.append('src');", f"{format_class(named_arg_values)}", f"_m = _i('{target_module}');", - f"_r = asyncio.run(_m.{function_call}) if _c(_m.{function_ref})", - f" else _m.{function_call};", + f"_r = asyncio.run(_m.{function_call.expression}) if _c(_m.{function_ref})", + f" else _m.{function_call.expression};", ] if self.spec.options.get("print_result"): @@ -99,11 +103,13 @@ def _handle_run( cmd = ("python", "-c", "".join(script)) self._print_action(shlex.join(argv), context.dry) - return self._get_executor(context, env).execute( - cmd, use_exec=self.spec.options.get("use_exec", False) - ) + return self._get_executor( + context, env, delegate_dry_run=has_dry_run_ref + ).execute(cmd, use_exec=self.spec.options.get("use_exec", False)) - def parse_content(self, args: Optional[Dict[str, Any]]) -> Tuple[str, str]: + def parse_content( + self, args: Optional[Dict[str, Any]] + ) -> Tuple[str, "FunctionCall"]: """ Returns the module to load, and the function call to execute. @@ -111,7 +117,7 @@ def parse_content(self, args: Optional[Dict[str, Any]]) -> Tuple[str, str]: references variables that are not in scope. """ - from ..helpers.python import resolve_expression + from ..helpers.python import FunctionCall try: target_module, target_ref = self.spec.content.strip().split(":", 1) @@ -122,17 +128,14 @@ def parse_content(self, args: Optional[Dict[str, Any]]) -> Tuple[str, str]: if target_ref.isidentifier(): if args: - return target_module, f"{target_ref}(**({args}))" - return target_module, f"{target_ref}()" - - function_call = resolve_expression( - target_ref, - set(args or tuple()), - call_only=True, - allowed_vars={"sys", "os", "environ"}, - ) - # Strip out any new lines because they can be problematic on windows - function_call = re.sub(r"((\r\n|\r|\n) | (\r\n|\r|\n))", " ", function_call) - function_call = re.sub(r"(\r\n|\r|\n)", " ", function_call) + function_call = FunctionCall(f"{target_ref}(**({args}))", target_ref) + else: + function_call = FunctionCall(f"{target_ref}()", target_ref) + else: + function_call = FunctionCall.parse( + source=target_ref, + arguments=set(args or tuple()), + allowed_vars={"sys", "os", "environ", "_dry_run"}, + ) return target_module, function_call diff --git a/poetry.lock b/poetry.lock index 7d7fcfe7c..f21e8ccfd 100644 --- a/poetry.lock +++ b/poetry.lock @@ -308,101 +308,116 @@ pycparser = "*" [[package]] name = "charset-normalizer" -version = "3.3.2" +version = "3.4.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.3.2.tar.gz", hash = "sha256:f30c3cb33b24454a82faecaf01b19c18562b1e89558fb6c56de4d9118a032fd5"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:25baf083bf6f6b341f4121c2f3c548875ee6f5339300e08be3f2b2ba1721cdd3"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:06435b539f889b1f6f4ac1758871aae42dc3a8c0e24ac9e60c2384973ad73027"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9063e24fdb1e498ab71cb7419e24622516c4a04476b17a2dab57e8baa30d6e03"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6897af51655e3691ff853668779c7bad41579facacf5fd7253b0133308cf000d"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1d3193f4a680c64b4b6a9115943538edb896edc190f0b222e73761716519268e"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cd70574b12bb8a4d2aaa0094515df2463cb429d8536cfb6c7ce983246983e5a6"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8465322196c8b4d7ab6d1e049e4c5cb460d0394da4a27d23cc242fbf0034b6b5"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a9a8e9031d613fd2009c182b69c7b2c1ef8239a0efb1df3f7c8da66d5dd3d537"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:beb58fe5cdb101e3a055192ac291b7a21e3b7ef4f67fa1d74e331a7f2124341c"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:e06ed3eb3218bc64786f7db41917d4e686cc4856944f53d5bdf83a6884432e12"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:2e81c7b9c8979ce92ed306c249d46894776a909505d8f5a4ba55b14206e3222f"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:572c3763a264ba47b3cf708a44ce965d98555f618ca42c926a9c1616d8f34269"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:fd1abc0d89e30cc4e02e4064dc67fcc51bd941eb395c502aac3ec19fab46b519"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-win32.whl", hash = "sha256:3d47fa203a7bd9c5b6cee4736ee84ca03b8ef23193c0d1ca99b5089f72645c73"}, - {file = "charset_normalizer-3.3.2-cp310-cp310-win_amd64.whl", hash = "sha256:10955842570876604d404661fbccbc9c7e684caf432c09c715ec38fbae45ae09"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:802fe99cca7457642125a8a88a084cef28ff0cf9407060f7b93dca5aa25480db"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:573f6eac48f4769d667c4442081b1794f52919e7edada77495aaed9236d13a96"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:549a3a73da901d5bc3ce8d24e0600d1fa85524c10287f6004fbab87672bf3e1e"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f27273b60488abe721a075bcca6d7f3964f9f6f067c8c4c605743023d7d3944f"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1ceae2f17a9c33cb48e3263960dc5fc8005351ee19db217e9b1bb15d28c02574"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:65f6f63034100ead094b8744b3b97965785388f308a64cf8d7c34f2f2e5be0c4"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:753f10e867343b4511128c6ed8c82f7bec3bd026875576dfd88483c5c73b2fd8"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4a78b2b446bd7c934f5dcedc588903fb2f5eec172f3d29e52a9096a43722adfc"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e537484df0d8f426ce2afb2d0f8e1c3d0b114b83f8850e5f2fbea0e797bd82ae"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:eb6904c354526e758fda7167b33005998fb68c46fbc10e013ca97f21ca5c8887"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:deb6be0ac38ece9ba87dea880e438f25ca3eddfac8b002a2ec3d9183a454e8ae"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:4ab2fe47fae9e0f9dee8c04187ce5d09f48eabe611be8259444906793ab7cbce"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:80402cd6ee291dcb72644d6eac93785fe2c8b9cb30893c1af5b8fdd753b9d40f"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-win32.whl", hash = "sha256:7cd13a2e3ddeed6913a65e66e94b51d80a041145a026c27e6bb76c31a853c6ab"}, - {file = "charset_normalizer-3.3.2-cp311-cp311-win_amd64.whl", hash = "sha256:663946639d296df6a2bb2aa51b60a2454ca1cb29835324c640dafb5ff2131a77"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:0b2b64d2bb6d3fb9112bafa732def486049e63de9618b5843bcdd081d8144cd8"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:ddbb2551d7e0102e7252db79ba445cdab71b26640817ab1e3e3648dad515003b"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:55086ee1064215781fff39a1af09518bc9255b50d6333f2e4c74ca09fac6a8f6"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8f4a014bc36d3c57402e2977dada34f9c12300af536839dc38c0beab8878f38a"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a10af20b82360ab00827f916a6058451b723b4e65030c5a18577c8b2de5b3389"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8d756e44e94489e49571086ef83b2bb8ce311e730092d2c34ca8f7d925cb20aa"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:90d558489962fd4918143277a773316e56c72da56ec7aa3dc3dbbe20fdfed15b"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6ac7ffc7ad6d040517be39eb591cac5ff87416c2537df6ba3cba3bae290c0fed"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:7ed9e526742851e8d5cc9e6cf41427dfc6068d4f5a3bb03659444b4cabf6bc26"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:8bdb58ff7ba23002a4c5808d608e4e6c687175724f54a5dade5fa8c67b604e4d"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:6b3251890fff30ee142c44144871185dbe13b11bab478a88887a639655be1068"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:b4a23f61ce87adf89be746c8a8974fe1c823c891d8f86eb218bb957c924bb143"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:efcb3f6676480691518c177e3b465bcddf57cea040302f9f4e6e191af91174d4"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-win32.whl", hash = "sha256:d965bba47ddeec8cd560687584e88cf699fd28f192ceb452d1d7ee807c5597b7"}, - {file = "charset_normalizer-3.3.2-cp312-cp312-win_amd64.whl", hash = "sha256:96b02a3dc4381e5494fad39be677abcb5e6634bf7b4fa83a6dd3112607547001"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:95f2a5796329323b8f0512e09dbb7a1860c46a39da62ecb2324f116fa8fdc85c"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c002b4ffc0be611f0d9da932eb0f704fe2602a9a949d1f738e4c34c75b0863d5"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a981a536974bbc7a512cf44ed14938cf01030a99e9b3a06dd59578882f06f985"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3287761bc4ee9e33561a7e058c72ac0938c4f57fe49a09eae428fd88aafe7bb6"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:42cb296636fcc8b0644486d15c12376cb9fa75443e00fb25de0b8602e64c1714"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0a55554a2fa0d408816b3b5cedf0045f4b8e1a6065aec45849de2d6f3f8e9786"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:c083af607d2515612056a31f0a8d9e0fcb5876b7bfc0abad3ecd275bc4ebc2d5"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:87d1351268731db79e0f8e745d92493ee2841c974128ef629dc518b937d9194c"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:bd8f7df7d12c2db9fab40bdd87a7c09b1530128315d047a086fa3ae3435cb3a8"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:c180f51afb394e165eafe4ac2936a14bee3eb10debc9d9e4db8958fe36afe711"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:8c622a5fe39a48f78944a87d4fb8a53ee07344641b0562c540d840748571b811"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-win32.whl", hash = "sha256:db364eca23f876da6f9e16c9da0df51aa4f104a972735574842618b8c6d999d4"}, - {file = "charset_normalizer-3.3.2-cp37-cp37m-win_amd64.whl", hash = "sha256:86216b5cee4b06df986d214f664305142d9c76df9b6512be2738aa72a2048f99"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:6463effa3186ea09411d50efc7d85360b38d5f09b870c48e4600f63af490e56a"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6c4caeef8fa63d06bd437cd4bdcf3ffefe6738fb1b25951440d80dc7df8c03ac"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:37e55c8e51c236f95b033f6fb391d7d7970ba5fe7ff453dad675e88cf303377a"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fb69256e180cb6c8a894fee62b3afebae785babc1ee98b81cdf68bbca1987f33"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ae5f4161f18c61806f411a13b0310bea87f987c7d2ecdbdaad0e94eb2e404238"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b2b0a0c0517616b6869869f8c581d4eb2dd83a4d79e0ebcb7d373ef9956aeb0a"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:45485e01ff4d3630ec0d9617310448a8702f70e9c01906b0d0118bdf9d124cf2"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:eb00ed941194665c332bf8e078baf037d6c35d7c4f3102ea2d4f16ca94a26dc8"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:2127566c664442652f024c837091890cb1942c30937add288223dc895793f898"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:a50aebfa173e157099939b17f18600f72f84eed3049e743b68ad15bd69b6bf99"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:4d0d1650369165a14e14e1e47b372cfcb31d6ab44e6e33cb2d4e57265290044d"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:923c0c831b7cfcb071580d3f46c4baf50f174be571576556269530f4bbd79d04"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:06a81e93cd441c56a9b65d8e1d043daeb97a3d0856d177d5c90ba85acb3db087"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-win32.whl", hash = "sha256:6ef1d82a3af9d3eecdba2321dc1b3c238245d890843e040e41e470ffa64c3e25"}, - {file = "charset_normalizer-3.3.2-cp38-cp38-win_amd64.whl", hash = "sha256:eb8821e09e916165e160797a6c17edda0679379a4be5c716c260e836e122f54b"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:c235ebd9baae02f1b77bcea61bce332cb4331dc3617d254df3323aa01ab47bd4"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5b4c145409bef602a690e7cfad0a15a55c13320ff7a3ad7ca59c13bb8ba4d45d"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:68d1f8a9e9e37c1223b656399be5d6b448dea850bed7d0f87a8311f1ff3dabb0"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:22afcb9f253dac0696b5a4be4a1c0f8762f8239e21b99680099abd9b2b1b2269"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e27ad930a842b4c5eb8ac0016b0a54f5aebbe679340c26101df33424142c143c"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1f79682fbe303db92bc2b1136016a38a42e835d932bab5b3b1bfcfbf0640e519"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b261ccdec7821281dade748d088bb6e9b69e6d15b30652b74cbbac25e280b796"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:122c7fa62b130ed55f8f285bfd56d5f4b4a5b503609d181f9ad85e55c89f4185"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:d0eccceffcb53201b5bfebb52600a5fb483a20b61da9dbc885f8b103cbe7598c"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:9f96df6923e21816da7e0ad3fd47dd8f94b2a5ce594e00677c0013018b813458"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:7f04c839ed0b6b98b1a7501a002144b76c18fb1c1850c8b98d458ac269e26ed2"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:34d1c8da1e78d2e001f363791c98a272bb734000fcef47a491c1e3b0505657a8"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ff8fa367d09b717b2a17a052544193ad76cd49979c805768879cb63d9ca50561"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-win32.whl", hash = "sha256:aed38f6e4fb3f5d6bf81bfa990a07806be9d83cf7bacef998ab1a9bd660a581f"}, - {file = "charset_normalizer-3.3.2-cp39-cp39-win_amd64.whl", hash = "sha256:b01b88d45a6fcb69667cd6d2f7a9aeb4bf53760d7fc536bf679ec94fe9f3ff3d"}, - {file = "charset_normalizer-3.3.2-py3-none-any.whl", hash = "sha256:3e4d1f6587322d2788836a99c69062fbb091331ec940e02d12d179c1d53e25fc"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:4f9fc98dad6c2eaa32fc3af1417d95b5e3d08aff968df0cd320066def971f9a6"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:0de7b687289d3c1b3e8660d0741874abe7888100efe14bd0f9fd7141bcbda92b"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:5ed2e36c3e9b4f21dd9422f6893dec0abf2cca553af509b10cd630f878d3eb99"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:40d3ff7fc90b98c637bda91c89d51264a3dcf210cade3a2c6f838c7268d7a4ca"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1110e22af8ca26b90bd6364fe4c763329b0ebf1ee213ba32b68c73de5752323d"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:86f4e8cca779080f66ff4f191a685ced73d2f72d50216f7112185dc02b90b9b7"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7f683ddc7eedd742e2889d2bfb96d69573fde1d92fcb811979cdb7165bb9c7d3"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:27623ba66c183eca01bf9ff833875b459cad267aeeb044477fedac35e19ba907"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:f606a1881d2663630ea5b8ce2efe2111740df4b687bd78b34a8131baa007f79b"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:0b309d1747110feb25d7ed6b01afdec269c647d382c857ef4663bbe6ad95a912"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:136815f06a3ae311fae551c3df1f998a1ebd01ddd424aa5603a4336997629e95"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:14215b71a762336254351b00ec720a8e85cada43b987da5a042e4ce3e82bd68e"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:79983512b108e4a164b9c8d34de3992f76d48cadc9554c9e60b43f308988aabe"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-win32.whl", hash = "sha256:c94057af19bc953643a33581844649a7fdab902624d2eb739738a30e2b3e60fc"}, + {file = "charset_normalizer-3.4.0-cp310-cp310-win_amd64.whl", hash = "sha256:55f56e2ebd4e3bc50442fbc0888c9d8c94e4e06a933804e2af3e89e2f9c1c749"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:0d99dd8ff461990f12d6e42c7347fd9ab2532fb70e9621ba520f9e8637161d7c"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:c57516e58fd17d03ebe67e181a4e4e2ccab1168f8c2976c6a334d4f819fe5944"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:6dba5d19c4dfab08e58d5b36304b3f92f3bd5d42c1a3fa37b5ba5cdf6dfcbcee"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bf4475b82be41b07cc5e5ff94810e6a01f276e37c2d55571e3fe175e467a1a1c"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ce031db0408e487fd2775d745ce30a7cd2923667cf3b69d48d219f1d8f5ddeb6"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8ff4e7cdfdb1ab5698e675ca622e72d58a6fa2a8aa58195de0c0061288e6e3ea"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3710a9751938947e6327ea9f3ea6332a09bf0ba0c09cae9cb1f250bd1f1549bc"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:82357d85de703176b5587dbe6ade8ff67f9f69a41c0733cf2425378b49954de5"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:47334db71978b23ebcf3c0f9f5ee98b8d65992b65c9c4f2d34c2eaf5bcaf0594"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:8ce7fd6767a1cc5a92a639b391891bf1c268b03ec7e021c7d6d902285259685c"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:f1a2f519ae173b5b6a2c9d5fa3116ce16e48b3462c8b96dfdded11055e3d6365"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:63bc5c4ae26e4bc6be6469943b8253c0fd4e4186c43ad46e713ea61a0ba49129"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:bcb4f8ea87d03bc51ad04add8ceaf9b0f085ac045ab4d74e73bbc2dc033f0236"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-win32.whl", hash = "sha256:9ae4ef0b3f6b41bad6366fb0ea4fc1d7ed051528e113a60fa2a65a9abb5b1d99"}, + {file = "charset_normalizer-3.4.0-cp311-cp311-win_amd64.whl", hash = "sha256:cee4373f4d3ad28f1ab6290684d8e2ebdb9e7a1b74fdc39e4c211995f77bec27"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:0713f3adb9d03d49d365b70b84775d0a0d18e4ab08d12bc46baa6132ba78aaf6"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:de7376c29d95d6719048c194a9cf1a1b0393fbe8488a22008610b0361d834ecf"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:4a51b48f42d9358460b78725283f04bddaf44a9358197b889657deba38f329db"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b295729485b06c1a0683af02a9e42d2caa9db04a373dc38a6a58cdd1e8abddf1"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ee803480535c44e7f5ad00788526da7d85525cfefaf8acf8ab9a310000be4b03"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3d59d125ffbd6d552765510e3f31ed75ebac2c7470c7274195b9161a32350284"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8cda06946eac330cbe6598f77bb54e690b4ca93f593dee1568ad22b04f347c15"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:07afec21bbbbf8a5cc3651aa96b980afe2526e7f048fdfb7f1014d84acc8b6d8"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:6b40e8d38afe634559e398cc32b1472f376a4099c75fe6299ae607e404c033b2"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:b8dcd239c743aa2f9c22ce674a145e0a25cb1566c495928440a181ca1ccf6719"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:84450ba661fb96e9fd67629b93d2941c871ca86fc38d835d19d4225ff946a631"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:44aeb140295a2f0659e113b31cfe92c9061622cadbc9e2a2f7b8ef6b1e29ef4b"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:1db4e7fefefd0f548d73e2e2e041f9df5c59e178b4c72fbac4cc6f535cfb1565"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-win32.whl", hash = "sha256:5726cf76c982532c1863fb64d8c6dd0e4c90b6ece9feb06c9f202417a31f7dd7"}, + {file = "charset_normalizer-3.4.0-cp312-cp312-win_amd64.whl", hash = "sha256:b197e7094f232959f8f20541ead1d9862ac5ebea1d58e9849c1bf979255dfac9"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:dd4eda173a9fcccb5f2e2bd2a9f423d180194b1bf17cf59e3269899235b2a114"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:e9e3c4c9e1ed40ea53acf11e2a386383c3304212c965773704e4603d589343ed"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:92a7e36b000bf022ef3dbb9c46bfe2d52c047d5e3f3343f43204263c5addc250"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:54b6a92d009cbe2fb11054ba694bc9e284dad30a26757b1e372a1fdddaf21920"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1ffd9493de4c922f2a38c2bf62b831dcec90ac673ed1ca182fe11b4d8e9f2a64"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:35c404d74c2926d0287fbd63ed5d27eb911eb9e4a3bb2c6d294f3cfd4a9e0c23"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4796efc4faf6b53a18e3d46343535caed491776a22af773f366534056c4e1fbc"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e7fdd52961feb4c96507aa649550ec2a0d527c086d284749b2f582f2d40a2e0d"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:92db3c28b5b2a273346bebb24857fda45601aef6ae1c011c0a997106581e8a88"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:ab973df98fc99ab39080bfb0eb3a925181454d7c3ac8a1e695fddfae696d9e90"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:4b67fdab07fdd3c10bb21edab3cbfe8cf5696f453afce75d815d9d7223fbe88b"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:aa41e526a5d4a9dfcfbab0716c7e8a1b215abd3f3df5a45cf18a12721d31cb5d"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:ffc519621dce0c767e96b9c53f09c5d215578e10b02c285809f76509a3931482"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-win32.whl", hash = "sha256:f19c1585933c82098c2a520f8ec1227f20e339e33aca8fa6f956f6691b784e67"}, + {file = "charset_normalizer-3.4.0-cp313-cp313-win_amd64.whl", hash = "sha256:707b82d19e65c9bd28b81dde95249b07bf9f5b90ebe1ef17d9b57473f8a64b7b"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:dbe03226baf438ac4fda9e2d0715022fd579cb641c4cf639fa40d53b2fe6f3e2"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dd9a8bd8900e65504a305bf8ae6fa9fbc66de94178c420791d0293702fce2df7"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b8831399554b92b72af5932cdbbd4ddc55c55f631bb13ff8fe4e6536a06c5c51"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a14969b8691f7998e74663b77b4c36c0337cb1df552da83d5c9004a93afdb574"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dcaf7c1524c0542ee2fc82cc8ec337f7a9f7edee2532421ab200d2b920fc97cf"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:425c5f215d0eecee9a56cdb703203dda90423247421bf0d67125add85d0c4455"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-musllinux_1_2_aarch64.whl", hash = "sha256:d5b054862739d276e09928de37c79ddeec42a6e1bfc55863be96a36ba22926f6"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-musllinux_1_2_i686.whl", hash = "sha256:f3e73a4255342d4eb26ef6df01e3962e73aa29baa3124a8e824c5d3364a65748"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-musllinux_1_2_ppc64le.whl", hash = "sha256:2f6c34da58ea9c1a9515621f4d9ac379871a8f21168ba1b5e09d74250de5ad62"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-musllinux_1_2_s390x.whl", hash = "sha256:f09cb5a7bbe1ecae6e87901a2eb23e0256bb524a79ccc53eb0b7629fbe7677c4"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-musllinux_1_2_x86_64.whl", hash = "sha256:0099d79bdfcf5c1f0c2c72f91516702ebf8b0b8ddd8905f97a8aecf49712c621"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-win32.whl", hash = "sha256:9c98230f5042f4945f957d006edccc2af1e03ed5e37ce7c373f00a5a4daa6149"}, + {file = "charset_normalizer-3.4.0-cp37-cp37m-win_amd64.whl", hash = "sha256:62f60aebecfc7f4b82e3f639a7d1433a20ec32824db2199a11ad4f5e146ef5ee"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:af73657b7a68211996527dbfeffbb0864e043d270580c5aef06dc4b659a4b578"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:cab5d0b79d987c67f3b9e9c53f54a61360422a5a0bc075f43cab5621d530c3b6"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:9289fd5dddcf57bab41d044f1756550f9e7cf0c8e373b8cdf0ce8773dc4bd417"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6b493a043635eb376e50eedf7818f2f322eabbaa974e948bd8bdd29eb7ef2a51"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9fa2566ca27d67c86569e8c85297aaf413ffab85a8960500f12ea34ff98e4c41"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a8e538f46104c815be19c975572d74afb53f29650ea2025bbfaef359d2de2f7f"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6fd30dc99682dc2c603c2b315bded2799019cea829f8bf57dc6b61efde6611c8"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2006769bd1640bdf4d5641c69a3d63b71b81445473cac5ded39740a226fa88ab"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:dc15e99b2d8a656f8e666854404f1ba54765871104e50c8e9813af8a7db07f12"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:ab2e5bef076f5a235c3774b4f4028a680432cded7cad37bba0fd90d64b187d19"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:4ec9dd88a5b71abfc74e9df5ebe7921c35cbb3b641181a531ca65cdb5e8e4dea"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:43193c5cda5d612f247172016c4bb71251c784d7a4d9314677186a838ad34858"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:aa693779a8b50cd97570e5a0f343538a8dbd3e496fa5dcb87e29406ad0299654"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-win32.whl", hash = "sha256:7706f5850360ac01d80c89bcef1640683cc12ed87f42579dab6c5d3ed6888613"}, + {file = "charset_normalizer-3.4.0-cp38-cp38-win_amd64.whl", hash = "sha256:c3e446d253bd88f6377260d07c895816ebf33ffffd56c1c792b13bff9c3e1ade"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:980b4f289d1d90ca5efcf07958d3eb38ed9c0b7676bf2831a54d4f66f9c27dfa"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:f28f891ccd15c514a0981f3b9db9aa23d62fe1a99997512b0491d2ed323d229a"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:a8aacce6e2e1edcb6ac625fb0f8c3a9570ccc7bfba1f63419b3769ccf6a00ed0"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bd7af3717683bea4c87acd8c0d3d5b44d56120b26fd3f8a692bdd2d5260c620a"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5ff2ed8194587faf56555927b3aa10e6fb69d931e33953943bc4f837dfee2242"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e91f541a85298cf35433bf66f3fab2a4a2cff05c127eeca4af174f6d497f0d4b"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:309a7de0a0ff3040acaebb35ec45d18db4b28232f21998851cfa709eeff49d62"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:285e96d9d53422efc0d7a17c60e59f37fbf3dfa942073f666db4ac71e8d726d0"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:5d447056e2ca60382d460a604b6302d8db69476fd2015c81e7c35417cfabe4cd"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:20587d20f557fe189b7947d8e7ec5afa110ccf72a3128d61a2a387c3313f46be"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:130272c698667a982a5d0e626851ceff662565379baf0ff2cc58067b81d4f11d"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:ab22fbd9765e6954bc0bcff24c25ff71dcbfdb185fcdaca49e81bac68fe724d3"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:7782afc9b6b42200f7362858f9e73b1f8316afb276d316336c0ec3bd73312742"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-win32.whl", hash = "sha256:2de62e8801ddfff069cd5c504ce3bc9672b23266597d4e4f50eda28846c322f2"}, + {file = "charset_normalizer-3.4.0-cp39-cp39-win_amd64.whl", hash = "sha256:95c3c157765b031331dd4db3c775e58deaee050a3042fcad72cbc4189d7c8dca"}, + {file = "charset_normalizer-3.4.0-py3-none-any.whl", hash = "sha256:fe9f97feb71aa9896b81973a7bbada8c49501dc73e58a10fcef6663af95e5079"}, + {file = "charset_normalizer-3.4.0.tar.gz", hash = "sha256:223217c3d4f82c3ac5e29032b3f1c2eb0fb591b72161f86d93f5719079dae93e"}, ] [[package]] @@ -548,38 +563,38 @@ toml = ["tomli"] [[package]] name = "cryptography" -version = "43.0.1" +version = "43.0.3" description = "cryptography is a package which provides cryptographic recipes and primitives to Python developers." optional = true python-versions = ">=3.7" files = [ - {file = "cryptography-43.0.1-cp37-abi3-macosx_10_9_universal2.whl", hash = "sha256:8385d98f6a3bf8bb2d65a73e17ed87a3ba84f6991c155691c51112075f9ffc5d"}, - {file = "cryptography-43.0.1-cp37-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:27e613d7077ac613e399270253259d9d53872aaf657471473ebfc9a52935c062"}, - {file = "cryptography-43.0.1-cp37-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:68aaecc4178e90719e95298515979814bda0cbada1256a4485414860bd7ab962"}, - {file = "cryptography-43.0.1-cp37-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:de41fd81a41e53267cb020bb3a7212861da53a7d39f863585d13ea11049cf277"}, - {file = "cryptography-43.0.1-cp37-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:f98bf604c82c416bc829e490c700ca1553eafdf2912a91e23a79d97d9801372a"}, - {file = "cryptography-43.0.1-cp37-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:61ec41068b7b74268fa86e3e9e12b9f0c21fcf65434571dbb13d954bceb08042"}, - {file = "cryptography-43.0.1-cp37-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:014f58110f53237ace6a408b5beb6c427b64e084eb451ef25a28308270086494"}, - {file = "cryptography-43.0.1-cp37-abi3-win32.whl", hash = "sha256:2bd51274dcd59f09dd952afb696bf9c61a7a49dfc764c04dd33ef7a6b502a1e2"}, - {file = "cryptography-43.0.1-cp37-abi3-win_amd64.whl", hash = "sha256:666ae11966643886c2987b3b721899d250855718d6d9ce41b521252a17985f4d"}, - {file = "cryptography-43.0.1-cp39-abi3-macosx_10_9_universal2.whl", hash = "sha256:ac119bb76b9faa00f48128b7f5679e1d8d437365c5d26f1c2c3f0da4ce1b553d"}, - {file = "cryptography-43.0.1-cp39-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1bbcce1a551e262dfbafb6e6252f1ae36a248e615ca44ba302df077a846a8806"}, - {file = "cryptography-43.0.1-cp39-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:58d4e9129985185a06d849aa6df265bdd5a74ca6e1b736a77959b498e0505b85"}, - {file = "cryptography-43.0.1-cp39-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:d03a475165f3134f773d1388aeb19c2d25ba88b6a9733c5c590b9ff7bbfa2e0c"}, - {file = "cryptography-43.0.1-cp39-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:511f4273808ab590912a93ddb4e3914dfd8a388fed883361b02dea3791f292e1"}, - {file = "cryptography-43.0.1-cp39-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:80eda8b3e173f0f247f711eef62be51b599b5d425c429b5d4ca6a05e9e856baa"}, - {file = "cryptography-43.0.1-cp39-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:38926c50cff6f533f8a2dae3d7f19541432610d114a70808f0926d5aaa7121e4"}, - {file = "cryptography-43.0.1-cp39-abi3-win32.whl", hash = "sha256:a575913fb06e05e6b4b814d7f7468c2c660e8bb16d8d5a1faf9b33ccc569dd47"}, - {file = "cryptography-43.0.1-cp39-abi3-win_amd64.whl", hash = "sha256:d75601ad10b059ec832e78823b348bfa1a59f6b8d545db3a24fd44362a1564cb"}, - {file = "cryptography-43.0.1-pp310-pypy310_pp73-macosx_10_9_x86_64.whl", hash = "sha256:ea25acb556320250756e53f9e20a4177515f012c9eaea17eb7587a8c4d8ae034"}, - {file = "cryptography-43.0.1-pp310-pypy310_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:c1332724be35d23a854994ff0b66530119500b6053d0bd3363265f7e5e77288d"}, - {file = "cryptography-43.0.1-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:fba1007b3ef89946dbbb515aeeb41e30203b004f0b4b00e5e16078b518563289"}, - {file = "cryptography-43.0.1-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:5b43d1ea6b378b54a1dc99dd8a2b5be47658fe9a7ce0a58ff0b55f4b43ef2b84"}, - {file = "cryptography-43.0.1-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:88cce104c36870d70c49c7c8fd22885875d950d9ee6ab54df2745f83ba0dc365"}, - {file = "cryptography-43.0.1-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:9d3cdb25fa98afdd3d0892d132b8d7139e2c087da1712041f6b762e4f807cc96"}, - {file = "cryptography-43.0.1-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:e710bf40870f4db63c3d7d929aa9e09e4e7ee219e703f949ec4073b4294f6172"}, - {file = "cryptography-43.0.1-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:7c05650fe8023c5ed0d46793d4b7d7e6cd9c04e68eabe5b0aeea836e37bdcec2"}, - {file = "cryptography-43.0.1.tar.gz", hash = "sha256:203e92a75716d8cfb491dc47c79e17d0d9207ccffcbcb35f598fbe463ae3444d"}, + {file = "cryptography-43.0.3-cp37-abi3-macosx_10_9_universal2.whl", hash = "sha256:bf7a1932ac4176486eab36a19ed4c0492da5d97123f1406cf15e41b05e787d2e"}, + {file = "cryptography-43.0.3-cp37-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:63efa177ff54aec6e1c0aefaa1a241232dcd37413835a9b674b6e3f0ae2bfd3e"}, + {file = "cryptography-43.0.3-cp37-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7e1ce50266f4f70bf41a2c6dc4358afadae90e2a1e5342d3c08883df1675374f"}, + {file = "cryptography-43.0.3-cp37-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:443c4a81bb10daed9a8f334365fe52542771f25aedaf889fd323a853ce7377d6"}, + {file = "cryptography-43.0.3-cp37-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:74f57f24754fe349223792466a709f8e0c093205ff0dca557af51072ff47ab18"}, + {file = "cryptography-43.0.3-cp37-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:9762ea51a8fc2a88b70cf2995e5675b38d93bf36bd67d91721c309df184f49bd"}, + {file = "cryptography-43.0.3-cp37-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:81ef806b1fef6b06dcebad789f988d3b37ccaee225695cf3e07648eee0fc6b73"}, + {file = "cryptography-43.0.3-cp37-abi3-win32.whl", hash = "sha256:cbeb489927bd7af4aa98d4b261af9a5bc025bd87f0e3547e11584be9e9427be2"}, + {file = "cryptography-43.0.3-cp37-abi3-win_amd64.whl", hash = "sha256:f46304d6f0c6ab8e52770addfa2fc41e6629495548862279641972b6215451cd"}, + {file = "cryptography-43.0.3-cp39-abi3-macosx_10_9_universal2.whl", hash = "sha256:8ac43ae87929a5982f5948ceda07001ee5e83227fd69cf55b109144938d96984"}, + {file = "cryptography-43.0.3-cp39-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:846da004a5804145a5f441b8530b4bf35afbf7da70f82409f151695b127213d5"}, + {file = "cryptography-43.0.3-cp39-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0f996e7268af62598f2fc1204afa98a3b5712313a55c4c9d434aef49cadc91d4"}, + {file = "cryptography-43.0.3-cp39-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:f7b178f11ed3664fd0e995a47ed2b5ff0a12d893e41dd0494f406d1cf555cab7"}, + {file = "cryptography-43.0.3-cp39-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:c2e6fc39c4ab499049df3bdf567f768a723a5e8464816e8f009f121a5a9f4405"}, + {file = "cryptography-43.0.3-cp39-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:e1be4655c7ef6e1bbe6b5d0403526601323420bcf414598955968c9ef3eb7d16"}, + {file = "cryptography-43.0.3-cp39-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:df6b6c6d742395dd77a23ea3728ab62f98379eff8fb61be2744d4679ab678f73"}, + {file = "cryptography-43.0.3-cp39-abi3-win32.whl", hash = "sha256:d56e96520b1020449bbace2b78b603442e7e378a9b3bd68de65c782db1507995"}, + {file = "cryptography-43.0.3-cp39-abi3-win_amd64.whl", hash = "sha256:0c580952eef9bf68c4747774cde7ec1d85a6e61de97281f2dba83c7d2c806362"}, + {file = "cryptography-43.0.3-pp310-pypy310_pp73-macosx_10_9_x86_64.whl", hash = "sha256:d03b5621a135bffecad2c73e9f4deb1a0f977b9a8ffe6f8e002bf6c9d07b918c"}, + {file = "cryptography-43.0.3-pp310-pypy310_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:a2a431ee15799d6db9fe80c82b055bae5a752bef645bba795e8e52687c69efe3"}, + {file = "cryptography-43.0.3-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:281c945d0e28c92ca5e5930664c1cefd85efe80e5c0d2bc58dd63383fda29f83"}, + {file = "cryptography-43.0.3-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:f18c716be16bc1fea8e95def49edf46b82fccaa88587a45f8dc0ff6ab5d8e0a7"}, + {file = "cryptography-43.0.3-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:4a02ded6cd4f0a5562a8887df8b3bd14e822a90f97ac5e544c162899bc467664"}, + {file = "cryptography-43.0.3-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:53a583b6637ab4c4e3591a15bc9db855b8d9dee9a669b550f311480acab6eb08"}, + {file = "cryptography-43.0.3-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:1ec0bcf7e17c0c5669d881b1cd38c4972fade441b27bda1051665faaa89bdcaa"}, + {file = "cryptography-43.0.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:2ce6fae5bdad59577b44e4dfed356944fbf1d925269114c28be377692643b4ff"}, + {file = "cryptography-43.0.3.tar.gz", hash = "sha256:315b9001266a492a6ff443b61238f956b214dbec9910a081ba5b6646a055a805"}, ] [package.dependencies] @@ -592,7 +607,7 @@ nox = ["nox"] pep8test = ["check-sdist", "click", "mypy", "ruff"] sdist = ["build"] ssh = ["bcrypt (>=3.1.5)"] -test = ["certifi", "cryptography-vectors (==43.0.1)", "pretend", "pytest (>=6.2.0)", "pytest-benchmark", "pytest-cov", "pytest-xdist"] +test = ["certifi", "cryptography-vectors (==43.0.3)", "pretend", "pytest (>=6.2.0)", "pytest-benchmark", "pytest-cov", "pytest-xdist"] test-randomorder = ["pytest-randomly"] [[package]] @@ -657,13 +672,13 @@ files = [ [[package]] name = "distlib" -version = "0.3.8" +version = "0.3.9" description = "Distribution utilities" optional = false python-versions = "*" files = [ - {file = "distlib-0.3.8-py2.py3-none-any.whl", hash = "sha256:034db59a0b96f8ca18035f36290806a9a6e6bd9d1ff91e45a7f172eb17e51784"}, - {file = "distlib-0.3.8.tar.gz", hash = "sha256:1530ea13e350031b6312d8580ddb6b27a104275a31106523b8f123787f494f64"}, + {file = "distlib-0.3.9-py2.py3-none-any.whl", hash = "sha256:47f8c22fd27c27e25a65601af709b38e4f0a45ea4fc2e710f65755fa8caaaf87"}, + {file = "distlib-0.3.9.tar.gz", hash = "sha256:a60f20dea646b8a33f3e7772f74dc0b2d0772d2837ee1342a00645c81edf9403"}, ] [[package]] @@ -1180,38 +1195,43 @@ files = [ [[package]] name = "mypy" -version = "1.11.2" +version = "1.13.0" description = "Optional static typing for Python" optional = false python-versions = ">=3.8" files = [ - {file = "mypy-1.11.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d42a6dd818ffce7be66cce644f1dff482f1d97c53ca70908dff0b9ddc120b77a"}, - {file = "mypy-1.11.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:801780c56d1cdb896eacd5619a83e427ce436d86a3bdf9112527f24a66618fef"}, - {file = "mypy-1.11.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:41ea707d036a5307ac674ea172875f40c9d55c5394f888b168033177fce47383"}, - {file = "mypy-1.11.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:6e658bd2d20565ea86da7d91331b0eed6d2eee22dc031579e6297f3e12c758c8"}, - {file = "mypy-1.11.2-cp310-cp310-win_amd64.whl", hash = "sha256:478db5f5036817fe45adb7332d927daa62417159d49783041338921dcf646fc7"}, - {file = "mypy-1.11.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:75746e06d5fa1e91bfd5432448d00d34593b52e7e91a187d981d08d1f33d4385"}, - {file = "mypy-1.11.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a976775ab2256aadc6add633d44f100a2517d2388906ec4f13231fafbb0eccca"}, - {file = "mypy-1.11.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:cd953f221ac1379050a8a646585a29574488974f79d8082cedef62744f0a0104"}, - {file = "mypy-1.11.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:57555a7715c0a34421013144a33d280e73c08df70f3a18a552938587ce9274f4"}, - {file = "mypy-1.11.2-cp311-cp311-win_amd64.whl", hash = "sha256:36383a4fcbad95f2657642a07ba22ff797de26277158f1cc7bd234821468b1b6"}, - {file = "mypy-1.11.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:e8960dbbbf36906c5c0b7f4fbf2f0c7ffb20f4898e6a879fcf56a41a08b0d318"}, - {file = "mypy-1.11.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:06d26c277962f3fb50e13044674aa10553981ae514288cb7d0a738f495550b36"}, - {file = "mypy-1.11.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:6e7184632d89d677973a14d00ae4d03214c8bc301ceefcdaf5c474866814c987"}, - {file = "mypy-1.11.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:3a66169b92452f72117e2da3a576087025449018afc2d8e9bfe5ffab865709ca"}, - {file = "mypy-1.11.2-cp312-cp312-win_amd64.whl", hash = "sha256:969ea3ef09617aff826885a22ece0ddef69d95852cdad2f60c8bb06bf1f71f70"}, - {file = "mypy-1.11.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:37c7fa6121c1cdfcaac97ce3d3b5588e847aa79b580c1e922bb5d5d2902df19b"}, - {file = "mypy-1.11.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:4a8a53bc3ffbd161b5b2a4fff2f0f1e23a33b0168f1c0778ec70e1a3d66deb86"}, - {file = "mypy-1.11.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2ff93107f01968ed834f4256bc1fc4475e2fecf6c661260066a985b52741ddce"}, - {file = "mypy-1.11.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:edb91dded4df17eae4537668b23f0ff6baf3707683734b6a818d5b9d0c0c31a1"}, - {file = "mypy-1.11.2-cp38-cp38-win_amd64.whl", hash = "sha256:ee23de8530d99b6db0573c4ef4bd8f39a2a6f9b60655bf7a1357e585a3486f2b"}, - {file = "mypy-1.11.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:801ca29f43d5acce85f8e999b1e431fb479cb02d0e11deb7d2abb56bdaf24fd6"}, - {file = "mypy-1.11.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:af8d155170fcf87a2afb55b35dc1a0ac21df4431e7d96717621962e4b9192e70"}, - {file = "mypy-1.11.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:f7821776e5c4286b6a13138cc935e2e9b6fde05e081bdebf5cdb2bb97c9df81d"}, - {file = "mypy-1.11.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:539c570477a96a4e6fb718b8d5c3e0c0eba1f485df13f86d2970c91f0673148d"}, - {file = "mypy-1.11.2-cp39-cp39-win_amd64.whl", hash = "sha256:3f14cd3d386ac4d05c5a39a51b84387403dadbd936e17cb35882134d4f8f0d24"}, - {file = "mypy-1.11.2-py3-none-any.whl", hash = "sha256:b499bc07dbdcd3de92b0a8b29fdf592c111276f6a12fe29c30f6c417dd546d12"}, - {file = "mypy-1.11.2.tar.gz", hash = "sha256:7f9993ad3e0ffdc95c2a14b66dee63729f021968bff8ad911867579c65d13a79"}, + {file = "mypy-1.13.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:6607e0f1dd1fb7f0aca14d936d13fd19eba5e17e1cd2a14f808fa5f8f6d8f60a"}, + {file = "mypy-1.13.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:8a21be69bd26fa81b1f80a61ee7ab05b076c674d9b18fb56239d72e21d9f4c80"}, + {file = "mypy-1.13.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:7b2353a44d2179846a096e25691d54d59904559f4232519d420d64da6828a3a7"}, + {file = "mypy-1.13.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:0730d1c6a2739d4511dc4253f8274cdd140c55c32dfb0a4cf8b7a43f40abfa6f"}, + {file = "mypy-1.13.0-cp310-cp310-win_amd64.whl", hash = "sha256:c5fc54dbb712ff5e5a0fca797e6e0aa25726c7e72c6a5850cfd2adbc1eb0a372"}, + {file = "mypy-1.13.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:581665e6f3a8a9078f28d5502f4c334c0c8d802ef55ea0e7276a6e409bc0d82d"}, + {file = "mypy-1.13.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3ddb5b9bf82e05cc9a627e84707b528e5c7caaa1c55c69e175abb15a761cec2d"}, + {file = "mypy-1.13.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:20c7ee0bc0d5a9595c46f38beb04201f2620065a93755704e141fcac9f59db2b"}, + {file = "mypy-1.13.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:3790ded76f0b34bc9c8ba4def8f919dd6a46db0f5a6610fb994fe8efdd447f73"}, + {file = "mypy-1.13.0-cp311-cp311-win_amd64.whl", hash = "sha256:51f869f4b6b538229c1d1bcc1dd7d119817206e2bc54e8e374b3dfa202defcca"}, + {file = "mypy-1.13.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:5c7051a3461ae84dfb5dd15eff5094640c61c5f22257c8b766794e6dd85e72d5"}, + {file = "mypy-1.13.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:39bb21c69a5d6342f4ce526e4584bc5c197fd20a60d14a8624d8743fffb9472e"}, + {file = "mypy-1.13.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:164f28cb9d6367439031f4c81e84d3ccaa1e19232d9d05d37cb0bd880d3f93c2"}, + {file = "mypy-1.13.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:a4c1bfcdbce96ff5d96fc9b08e3831acb30dc44ab02671eca5953eadad07d6d0"}, + {file = "mypy-1.13.0-cp312-cp312-win_amd64.whl", hash = "sha256:a0affb3a79a256b4183ba09811e3577c5163ed06685e4d4b46429a271ba174d2"}, + {file = "mypy-1.13.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:a7b44178c9760ce1a43f544e595d35ed61ac2c3de306599fa59b38a6048e1aa7"}, + {file = "mypy-1.13.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:5d5092efb8516d08440e36626f0153b5006d4088c1d663d88bf79625af3d1d62"}, + {file = "mypy-1.13.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:de2904956dac40ced10931ac967ae63c5089bd498542194b436eb097a9f77bc8"}, + {file = "mypy-1.13.0-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:7bfd8836970d33c2105562650656b6846149374dc8ed77d98424b40b09340ba7"}, + {file = "mypy-1.13.0-cp313-cp313-win_amd64.whl", hash = "sha256:9f73dba9ec77acb86457a8fc04b5239822df0c14a082564737833d2963677dbc"}, + {file = "mypy-1.13.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:100fac22ce82925f676a734af0db922ecfea991e1d7ec0ceb1e115ebe501301a"}, + {file = "mypy-1.13.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:7bcb0bb7f42a978bb323a7c88f1081d1b5dee77ca86f4100735a6f541299d8fb"}, + {file = "mypy-1.13.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:bde31fc887c213e223bbfc34328070996061b0833b0a4cfec53745ed61f3519b"}, + {file = "mypy-1.13.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:07de989f89786f62b937851295ed62e51774722e5444a27cecca993fc3f9cd74"}, + {file = "mypy-1.13.0-cp38-cp38-win_amd64.whl", hash = "sha256:4bde84334fbe19bad704b3f5b78c4abd35ff1026f8ba72b29de70dda0916beb6"}, + {file = "mypy-1.13.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:0246bcb1b5de7f08f2826451abd947bf656945209b140d16ed317f65a17dc7dc"}, + {file = "mypy-1.13.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:7f5b7deae912cf8b77e990b9280f170381fdfbddf61b4ef80927edd813163732"}, + {file = "mypy-1.13.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:7029881ec6ffb8bc233a4fa364736789582c738217b133f1b55967115288a2bc"}, + {file = "mypy-1.13.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:3e38b980e5681f28f033f3be86b099a247b13c491f14bb8b1e1e134d23bb599d"}, + {file = "mypy-1.13.0-cp39-cp39-win_amd64.whl", hash = "sha256:a6789be98a2017c912ae6ccb77ea553bbaf13d27605d2ca20a76dfbced631b24"}, + {file = "mypy-1.13.0-py3-none-any.whl", hash = "sha256:9c250883f9fd81d212e0952c92dbfcc96fc237f4b7c92f56ac81fd48460b3e5a"}, + {file = "mypy-1.13.0.tar.gz", hash = "sha256:0291a61b6fbf3e6673e3405cfcc0e7650bebc7939659fdca2702958038bd835e"}, ] [package.dependencies] @@ -1221,6 +1241,7 @@ typing-extensions = ">=4.6.0" [package.extras] dmypy = ["psutil (>=4.0)"] +faster-cache = ["orjson"] install-types = ["pip"] mypyc = ["setuptools (>=50)"] reports = ["lxml"] @@ -1285,13 +1306,13 @@ ptyprocess = ">=0.5" [[package]] name = "pkginfo" -version = "1.11.1" +version = "1.11.2" description = "Query metadata from sdists / bdists / installed packages." optional = true python-versions = ">=3.8" files = [ - {file = "pkginfo-1.11.1-py3-none-any.whl", hash = "sha256:bfa76a714fdfc18a045fcd684dbfc3816b603d9d075febef17cb6582bea29573"}, - {file = "pkginfo-1.11.1.tar.gz", hash = "sha256:2e0dca1cf4c8e39644eed32408ea9966ee15e0d324c62ba899a393b3c6b467aa"}, + {file = "pkginfo-1.11.2-py3-none-any.whl", hash = "sha256:9ec518eefccd159de7ed45386a6bb4c6ca5fa2cb3bd9b71154fae44f6f1b36a3"}, + {file = "pkginfo-1.11.2.tar.gz", hash = "sha256:c6bc916b8298d159e31f2c216e35ee5b86da7da18874f879798d0a1983537c86"}, ] [package.extras] @@ -1736,19 +1757,19 @@ requests = ">=2.0.1,<3.0.0" [[package]] name = "rich" -version = "13.8.1" +version = "13.9.3" description = "Render rich text, tables, progress bars, syntax highlighting, markdown and more to the terminal" optional = false -python-versions = ">=3.7.0" +python-versions = ">=3.8.0" files = [ - {file = "rich-13.8.1-py3-none-any.whl", hash = "sha256:1760a3c0848469b97b558fc61c85233e3dafb69c7a071b4d60c38099d3cd4c06"}, - {file = "rich-13.8.1.tar.gz", hash = "sha256:8260cda28e3db6bf04d2d1ef4dbc03ba80a824c88b0e7668a0f23126a424844a"}, + {file = "rich-13.9.3-py3-none-any.whl", hash = "sha256:9836f5096eb2172c9e77df411c1b009bace4193d6a481d534fea75ebba758283"}, + {file = "rich-13.9.3.tar.gz", hash = "sha256:bc1e01b899537598cf02579d2b9f4a415104d3fc439313a7a2c165d76557a08e"}, ] [package.dependencies] 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\""} +typing-extensions = {version = ">=4.0.0,<5.0", markers = "python_version < \"3.11\""} [package.extras] jupyter = ["ipywidgets (>=7.5.1,<9)"] @@ -1845,13 +1866,13 @@ jeepney = ">=0.6" [[package]] name = "setuptools" -version = "75.1.0" +version = "75.2.0" description = "Easily download, build, install, upgrade, and uninstall Python packages" optional = true python-versions = ">=3.8" files = [ - {file = "setuptools-75.1.0-py3-none-any.whl", hash = "sha256:35ab7fd3bcd95e6b7fd704e4a1539513edad446c097797f2985e0e4b960772f2"}, - {file = "setuptools-75.1.0.tar.gz", hash = "sha256:d59a21b17a275fb872a9c3dae73963160ae079f1049ed956880cd7c09b120538"}, + {file = "setuptools-75.2.0-py3-none-any.whl", hash = "sha256:a7fcb66f68b4d9e8e66b42f9876150a3371558f98fa32222ffaa5bced76406f8"}, + {file = "setuptools-75.2.0.tar.gz", hash = "sha256:753bb6ebf1f465a1912e19ed1d41f403a79173a9acf66a42e7e6aec45c3c16ec"}, ] [package.extras] @@ -2082,13 +2103,13 @@ sphinx = ">=4.0" [[package]] name = "tomli" -version = "2.0.1" +version = "2.0.2" description = "A lil' TOML parser" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "tomli-2.0.1-py3-none-any.whl", hash = "sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc"}, - {file = "tomli-2.0.1.tar.gz", hash = "sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f"}, + {file = "tomli-2.0.2-py3-none-any.whl", hash = "sha256:2ebe24485c53d303f690b0ec092806a085f07af5a5aa1464f3931eec36caaa38"}, + {file = "tomli-2.0.2.tar.gz", hash = "sha256:d46d457a85337051c36524bc5349dd91b1877838e2979ac5ced3e710ed8a60ed"}, ] [[package]] @@ -2180,13 +2201,13 @@ zstd = ["zstandard (>=0.18.0)"] [[package]] name = "virtualenv" -version = "20.26.5" +version = "20.27.0" description = "Virtual Python Environment builder" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "virtualenv-20.26.5-py3-none-any.whl", hash = "sha256:4f3ac17b81fba3ce3bd6f4ead2749a72da5929c01774948e243db9ba41df4ff6"}, - {file = "virtualenv-20.26.5.tar.gz", hash = "sha256:ce489cac131aa58f4b25e321d6d186171f78e6cb13fafbf32a840cee67733ff4"}, + {file = "virtualenv-20.27.0-py3-none-any.whl", hash = "sha256:44a72c29cceb0ee08f300b314848c86e57bf8d1f13107a5e671fb9274138d655"}, + {file = "virtualenv-20.27.0.tar.gz", hash = "sha256:2ca56a68ed615b8fe4326d11a0dca5dfbe8fd68510fb6c6349163bed3c15f2b2"}, ] [package.dependencies] @@ -2245,4 +2266,4 @@ poetry-plugin = ["poetry"] [metadata] lock-version = "2.0" python-versions = ">=3.8" -content-hash = "4c22f1084b52e05a9d55d5c124eea7b585bddebedb967b950d565ace62ca500d" +content-hash = "a7a68cf37ccb0a11c4fe98d4e174ddd45ac1b9f85b1206568d37f2ab60aa2d84" diff --git a/pyproject.toml b/pyproject.toml index e79240852..47e6c890d 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -23,7 +23,7 @@ black = "^23.3.0" mypy = "^1.1.1" pytest = "^7.1.2" pytest-cov = "^3.0.0" -rstcheck = { version = "^6.1.2", python = "<4" } +rstcheck = { version = "^6.2.4", python = "<4" } ruff = "^0.0.291" types-pyyaml = "^6.0.12.20240808" @@ -32,7 +32,7 @@ virtualenv = "^20.14.1" poe_test_helpers = { path = "./tests/fixtures/packages/poe_test_helpers" } [tool.poetry.group.dev.dependencies] -bpython = "^0.24" +bpython = "^0.24" [tool.poetry.group.docs.dependencies] furo = "^2023.3.27" @@ -52,91 +52,93 @@ poethepoet = "poethepoet.plugin:PoetryPlugin" [tool.poe.tasks] - _clean_docs.script = "shutil:rmtree('docs/_build', ignore_errors=1)" - [tool.poe.tasks.format] - help = "Run all formatting tools on the code base" - sequence = ["format-ruff", "format-black"] - - [tool.poe.tasks.format-ruff] - help = "Run ruff fixer on code base" - cmd = "ruff check . --fix-only" - - [tool.poe.tasks.format-black] - help = "Run black on the code base" - cmd = "black ." - - [tool.poe.tasks.docs] - help = "Build the docs using Sphinx" - cmd = "sphinx-build docs docs/_build" - deps = ["_clean_docs"] - - [tool.poe.tasks.docs-serve] - help = "Serves the docs locally with livereload" - script = "docs:serve" - cwd = "./docs" - deps = ["docs"] - - [tool.poe.tasks.docs-check] - help = "Validate rst syntax in the docs" - sequence = [ - { cmd = "rstcheck -r docs --ignore-roles bash,toml,sh,python --ignore-substitutions V" }, - "docs -b linkcheck" - ] - - [tool.poe.tasks.clean] - help = "Remove generated files" - cmd = """ - # multiline commands including comments work too! - rm -rf .coverage - .mypy_cache - .pytest_cache - ./**/__pycache__ - dist - htmlcov - ./docs/_build - ./tests/fixtures/simple_project/venv - ./tests/fixtures/venv_project/myvenv - ./tests/fixtures/poetry_plugin_project/**/.venv - ./tests/temp - """ - - [tool.poe.tasks.test] - help = "Run unit and feature tests" - cmd = "pytest" - - [tool.poe.tasks.test-quick] - help = "Run unit and feature tests, excluding slow ones" - cmd = "pytest -m 'not slow'" - - [tool.poe.tasks.types] - help = "Run the type checker" - cmd = "mypy poethepoet --ignore-missing-imports" - - [tool.poe.tasks.lint] - help = "Run linting tools on the code base" - cmd = "ruff check ." - - [tool.poe.tasks.style] - help = "Validate black code style" - cmd = "black . --check --diff" - - [tool.poe.tasks.check] - help = "Run all checks on the code base" - sequence = ["docs-check", "style", "types", "lint", "test"] - - [tool.poe.tasks.install-poetry-plugin] - help = "Install or update this project as a plugin in poetry" - sequence = [ - { cmd = "poetry self remove poethepoet"}, - { cmd = "poetry self add \"${POE_ROOT}[poetry_plugin]\""} - ] - ignore_fail = true - - [tool.poe.tasks.poe] - help = "Execute poe from this repo (useful for testing)" - script = "poethepoet:main" +[tool.poe.tasks.format] +help = "Run all formatting tools on the code base" +sequence = ["format-ruff", "format-black"] + +[tool.poe.tasks.format-ruff] +help = "Run ruff fixer on code base" +cmd = "ruff check . --fix-only" + +[tool.poe.tasks.format-black] +help = "Run black on the code base" +cmd = "black ." + +[tool.poe.tasks.docs] +help = "Build the docs using Sphinx" +cmd = "sphinx-build docs docs/_build" +deps = ["_clean_docs"] + +[tool.poe.tasks.docs-serve] +help = "Serves the docs locally with livereload" +script = "docs:serve" +cwd = "./docs" +deps = ["docs"] + +[tool.poe.tasks.docs-check] +help = "Validate rst syntax in the docs" +sequence = [ + { cmd = "rstcheck -r docs --ignore-roles bash,toml,sh,python --ignore-substitutions V --ignore-directives autofunction" }, + "docs -b linkcheck" +] + +[tool.poe.tasks.clean] +help = "Remove generated files" +script = """ + poethepoet.scripts:rm( + ".coverage", + ".mypy_cache", + ".pytest_cache", + "./**/__pycache__", + "dist", + "htmlcov", + "./docs/_build", + "./tests/fixtures/**/.venv", + "./tests/fixtures/simple_project/venv", + "./tests/fixtures/venv_project/myvenv", + "./tests/temp", + verbosity=environ.get('POE_VERBOSITY'), + dry_run=_dry_run + ) +""" + +[tool.poe.tasks.test] +help = "Run unit and feature tests" +cmd = "pytest" + +[tool.poe.tasks.test-quick] +help = "Run unit and feature tests, excluding slow ones" +cmd = "pytest -m 'not slow'" + +[tool.poe.tasks.types] +help = "Run the type checker" +cmd = "mypy poethepoet --ignore-missing-imports" + +[tool.poe.tasks.lint] +help = "Run linting tools on the code base" +cmd = "ruff check ." + +[tool.poe.tasks.style] +help = "Validate black code style" +cmd = "black . --check --diff" + +[tool.poe.tasks.check] +help = "Run all checks on the code base" +sequence = ["docs-check", "style", "types", "lint", "test"] + +[tool.poe.tasks.install-poetry-plugin] +help = "Install or update this project as a plugin in poetry" +sequence = [ + { cmd = "poetry self remove poethepoet"}, + { cmd = "poetry self add \"${POE_ROOT}[poetry_plugin]\""} +] +ignore_fail = true + +[tool.poe.tasks.poe] +help = "Execute poe from this repo (useful for testing)" +script = "poethepoet:main" [tool.rstcheck] diff --git a/tests/conftest.py b/tests/conftest.py index 27e2756b3..d5753a379 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -51,6 +51,7 @@ def projects(): re.match(r"^([_\w]+)_project", path.name).groups()[0]: path.resolve() for path in base_path.glob("*_project") } + projects.update( { f"{project_key}/" @@ -102,6 +103,12 @@ def __str__(self): ")" ) + def assert_no_err(self): + # Only stderr output allowed is from poetry creating its venv + assert all( + line.startswith("Creating virtualenv ") for line in self.stderr.splitlines() + ) + @pytest.fixture() def run_poe_subproc(projects, temp_file, tmp_path, is_windows): diff --git a/tests/fixtures/example_project/pyproject.toml b/tests/fixtures/example_project/pyproject.toml index cb2f054de..53c4eda85 100644 --- a/tests/fixtures/example_project/pyproject.toml +++ b/tests/fixtures/example_project/pyproject.toml @@ -1,6 +1,7 @@ [tool.poetry] name = "dummy-project" version = "0.1.0" + description = "dummy-project" authors = ["Nat Noordanus "] @@ -8,6 +9,10 @@ packages = [ { include = "dummy_package" } ] +[tool.poetry.dependencies] +poe_test_helpers = { path = "../packages/poe_test_helpers" } + + [tool.poe] env.PLANET = "EARTH" env.DEST = "MARS" diff --git a/tests/fixtures/scripts_project/poetry.lock b/tests/fixtures/scripts_project/poetry.lock new file mode 100644 index 000000000..3aa31f8ac --- /dev/null +++ b/tests/fixtures/scripts_project/poetry.lock @@ -0,0 +1,7 @@ +# This file is automatically @generated by Poetry 1.8.2 and should not be changed by hand. +package = [] + +[metadata] +lock-version = "2.0" +python-versions = "*" +content-hash = "115cf985d932e9bf5f540555bbdd75decbb62cac81e399375fc19f6277f8c1d8" diff --git a/tests/fixtures/scripts_project/pyproject.toml b/tests/fixtures/scripts_project/pyproject.toml index 936f42539..68f3d9406 100644 --- a/tests/fixtures/scripts_project/pyproject.toml +++ b/tests/fixtures/scripts_project/pyproject.toml @@ -175,7 +175,10 @@ Default arg documentation multiline """ +[tool.poe.tasks.check-global-options] +script = "pkg:echo_script(dry=_dry_run, verbosity=environ['POE_VERBOSITY'])" [build-system] requires = ["poetry>=0.12"] build-backend = "poetry.masonry.api" + diff --git a/tests/test_cli.py b/tests/test_cli.py index 603486347..33ced40df 100644 --- a/tests/test_cli.py +++ b/tests/test_cli.py @@ -2,6 +2,10 @@ from poethepoet import __version__ +# Setting POETRY_VIRTUALENVS_CREATE stops poetry from creating the virtualenv and +# spamming about it in stderr +poetry_vars = {"POETRY_VIRTUALENVS_CREATE": "false"} + def test_call_no_args(run_poe): result = run_poe() @@ -94,13 +98,56 @@ def test_version_option(run_poe): assert result.stderr == "" -def test_dry_run(run_poe_subproc): +def test_dry_run_cmd(run_poe_subproc): result = run_poe_subproc("-d", "show_env") assert result.capture == "Poe => poe_test_env\n" assert result.stdout == "" assert result.stderr == "" +def test_dry_run_script(run_poe_subproc): + result = run_poe_subproc("-d", "multiple-lines-help", project="scripts") + assert result.capture == "Poe => multiple-lines-help\n" + assert result.stdout == "" + assert result.stderr == "" + + +def test_pass_dry_run_and_verbosity_to_script(run_poe_subproc): + result = run_poe_subproc("check-global-options", project="scripts") + assert result.capture == "Poe => check-global-options\n" + assert result.stdout == ("args ()\n" "kwargs {'dry': False, 'verbosity': '0'}\n") + assert result.stderr == "" + + result = run_poe_subproc("-d", "check-global-options", project="scripts") + assert result.capture == "Poe => check-global-options\n" + assert result.stdout == ("args ()\n" "kwargs {'dry': True, 'verbosity': '0'}\n") + assert result.stderr == "" + + result = run_poe_subproc("-d", "-v", "check-global-options", project="scripts") + assert result.capture == "Poe => check-global-options\n" + assert result.stdout == ("args ()\n" "kwargs {'dry': True, 'verbosity': '1'}\n") + assert result.stderr == "" + + result = run_poe_subproc("-q", "check-global-options", project="scripts") + assert result.capture == "" + assert result.stdout == ("args ()\n" "kwargs {'dry': False, 'verbosity': '-1'}\n") + assert result.stderr == "" + + +def test_poe_env_vars_are_set(run_poe_subproc): + result = run_poe_subproc("show_env", env=poetry_vars) + assert result.capture == "Poe => poe_test_env\n" + for env_var in ( + "POE_VERBOSITY=0", + "POE_CONF_DIR=", + "POE_ACTIVE=poetry", + "POE_CWD=", + "POE_ROOT=", + "POE_PWD=", + ): + assert env_var in result.stdout + + def test_documentation_of_task_named_args(run_poe): result = run_poe(project="scripts") assert result.capture.startswith( diff --git a/tests/test_poe_config.py b/tests/test_poe_config.py index 5323aa641..c0c9c1f3f 100644 --- a/tests/test_poe_config.py +++ b/tests/test_poe_config.py @@ -1,6 +1,6 @@ # Setting POETRY_VIRTUALENVS_CREATE stops poetry from creating the virtualenv and -# spamming about it in stdout -no_venv = {"POETRY_VIRTUALENVS_CREATE": "false"} +# spamming about it in stderr +poetry_vars = {"POETRY_VIRTUALENVS_CREATE": "false"} def test_setting_default_task_type(run_poe_subproc, projects, esc_prefix): @@ -10,31 +10,33 @@ def test_setting_default_task_type(run_poe_subproc, projects, esc_prefix): "nat,", r"welcome to " + esc_prefix + "${POE_ROOT}", project="scripts", - env=no_venv, + env=poetry_vars, ) assert ( result.capture == f"Poe => echo-args nat, 'welcome to {projects['scripts']}'\n" ) assert result.stdout == f"hello nat, welcome to {projects['scripts']}\n" - assert result.stderr == "" + result.assert_no_err() def test_setting_default_array_item_task_type(run_poe_subproc): - result = run_poe_subproc("composite_task", project="scripts", env=no_venv) + result = run_poe_subproc( + "composite_task", project="scripts", env={"POETRY_VIRTUALENVS_CREATE": "false"} + ) assert ( result.capture == "Poe => poe_test_echo Hello\nPoe => poe_test_echo 'World!'\n" ) assert result.stdout == "Hello\nWorld!\n" - assert result.stderr == "" + result.assert_no_err() def test_setting_global_env_vars(run_poe_subproc): - result = run_poe_subproc("travel", env=no_venv) + result = run_poe_subproc("travel", env=poetry_vars) assert ( result.capture == "Poe => poe_test_echo 'from EARTH to'\nPoe => 'travel[1]'\n" ) assert result.stdout == "from EARTH to\nMARS\n" - assert result.stderr == "" + result.assert_no_err() def test_setting_default_verbosity(run_poe_subproc, low_verbosity_project_path): @@ -44,7 +46,7 @@ def test_setting_default_verbosity(run_poe_subproc, low_verbosity_project_path): ) assert result.capture == "" assert result.stdout == "Hello there!\n" - assert result.stderr == "" + result.assert_no_err() def test_override_default_verbosity(run_poe_subproc, low_verbosity_project_path): @@ -56,7 +58,7 @@ def test_override_default_verbosity(run_poe_subproc, low_verbosity_project_path) ) assert result.capture == "Poe => poe_test_echo Hello 'there!'\n" assert result.stdout == "Hello there!\n" - assert result.stderr == "" + result.assert_no_err() def test_partially_decrease_verbosity(run_poe_subproc, high_verbosity_project_path): @@ -67,15 +69,11 @@ def test_partially_decrease_verbosity(run_poe_subproc, high_verbosity_project_pa ) assert result.capture == "Poe => poe_test_echo Hello 'there!'\n" assert result.stdout == "Hello there!\n" - assert result.stderr == "" + result.assert_no_err() def test_decrease_verbosity(run_poe_subproc): - result = run_poe_subproc( - "-q", - "part1", - env=no_venv, - ) + result = run_poe_subproc("-q", "part1", env=poetry_vars) assert result.capture == "" - assert result.stderr == "" assert result.stdout == "Hello\n" + result.assert_no_err() diff --git a/tests/test_script_tasks.py b/tests/test_script_tasks.py index 08195e97b..065f61c35 100644 --- a/tests/test_script_tasks.py +++ b/tests/test_script_tasks.py @@ -1,4 +1,5 @@ import difflib +import sys no_venv = {"POETRY_VIRTUALENVS_CREATE": "false"} @@ -339,9 +340,9 @@ def test_script_with_multi_value_args(run_poe_subproc): # Not enough values for option: 0 result = run_poe_subproc( "multiple-value-args", - "--widgets", "--engines", "v2", + "--widgets", project="scripts", env=no_venv, ) @@ -356,20 +357,27 @@ def test_script_with_multi_value_args(run_poe_subproc): result = run_poe_subproc( "multiple-value-args", "bloop", # without the first arg, dong gets read an positional + "--engines", + "v2", "--widgets", "ding", "dang", "dong", - "--engines", - "v2", project="scripts", env=no_venv, ) assert result.capture == "" assert result.stdout == "" - assert ( - "poe multiple-value-args: error: unrecognized arguments: dong" in result.stderr - ) + if sys.version_info > (3, 12, 6): + assert ( + "poe multiple-value-args: error: argument second: invalid int value: 'dong'" + in result.stderr + ) + else: + assert ( + "poe multiple-value-args: error: unrecognized arguments: dong" + in result.stderr + ) # wrong type for multiple values result = run_poe_subproc( diff --git a/tests/test_scripts.py b/tests/test_scripts.py new file mode 100644 index 000000000..00cd8b616 --- /dev/null +++ b/tests/test_scripts.py @@ -0,0 +1,171 @@ +import os +from pathlib import Path +from shutil import rmtree + +import pytest + +from poethepoet.scripts import rm + +_test_file_tree = { + "pkg": { + "__pycache__": {"foo.pyc": "XXX"}, + "foo.py": "XXX", + "bar": { + "baz.py": "XXX", + "__pycache__": {"baz.pyc": "XXX"}, + }, + }, + ".mypy_cache": "XXX", + ".pytest_cache": "XXX", + "readme.md": "XXX", + "garbage.md": "XXX", + "__pycache__": {}, +} + + +@pytest.fixture() +def test_file_tree_dir(poe_project_path): + path = poe_project_path / "tests" / "temp" / "rm_test" + path.mkdir(parents=True, exist_ok=True) + yield path + rmtree(path) + + +@pytest.fixture() +def test_file_tree_nodes(test_file_tree_dir): + def _iter_dir(work_dir: Path, items: dict): + for node_name, content in items.items(): + node_path = work_dir / node_name + if isinstance(content, dict): + yield (node_path, None) + yield from _iter_dir(node_path, content) + else: + yield (node_path, "content") + + return tuple(_iter_dir(test_file_tree_dir, _test_file_tree)) + + +@pytest.fixture() +def test_dir_structure(test_file_tree_dir, test_file_tree_nodes): + """ + Stage a temporary directory structure full of files so we can delete some of them + """ + + for path, content in test_file_tree_nodes: + if content: + path.write_text(content) + else: + path.mkdir(parents=True, exist_ok=True) + + cwd = Path.cwd() + os.chdir(test_file_tree_dir) + + try: + yield test_file_tree_dir + finally: + os.chdir(cwd) + + +def test_rm_dry_mode(test_dir_structure, capsys, test_file_tree_nodes): + rm(".mypy_cache", ".pytest_cache", "./**/__pycache__", verbosity=0, dry_run=True) + + captured = capsys.readouterr() + assert captured.out == ("Deleting paths matching './**/__pycache__'\n") + assert captured.err == "" + + for path, content in test_file_tree_nodes: + assert path.exists() + + +def test_rm_deletion(test_dir_structure, capsys, test_file_tree_nodes): + rm(".mypy_cache", ".pytest_cache", "./**/__pycache__") + + captured = capsys.readouterr() + assert captured.out == ("Deleting paths matching './**/__pycache__'\n") + assert captured.err == "" + + for path, content in test_file_tree_nodes: + if any( + name in str(path) + for name in ("__pycache__", ".mypy_cache", ".pytest_cache") + ): + assert not path.exists() + else: + assert path.exists() + + +def test_rm_dry_mode_quiet(test_dir_structure, capsys, test_file_tree_nodes): + rm(".mypy_cache", ".pytest_cache", "./**/__pycache__", verbosity=-1, dry_run=True) + + captured = capsys.readouterr() + assert captured.out == "" + assert captured.err == "" + + for path, content in test_file_tree_nodes: + assert path.exists() + + +def test_rm_dry_mode_verbose( + test_dir_structure, capsys, test_file_tree_nodes, is_windows +): + rm(".mypy_cache", ".pytest_cache", "./**/__pycache__", verbosity=1, dry_run=True) + + captured = capsys.readouterr() + if is_windows: + assert captured.out == ( + "Deleting file '.mypy_cache'\n" + "Deleting file '.pytest_cache'\n" + "Deleting paths matching './**/__pycache__'\n" + "Deleting directory '__pycache__'\n" + "Deleting directory 'pkg\\__pycache__'\n" + "Deleting directory 'pkg\\bar\\__pycache__'\n" + ) + else: + assert captured.out == ( + "Deleting file '.mypy_cache'\n" + "Deleting file '.pytest_cache'\n" + "Deleting paths matching './**/__pycache__'\n" + "Deleting directory '__pycache__'\n" + "Deleting directory 'pkg/__pycache__'\n" + "Deleting directory 'pkg/bar/__pycache__'\n" + ) + assert captured.err == "" + + for path, content in test_file_tree_nodes: + assert path.exists() + + +def test_rm_no_patterns_verbose(test_dir_structure, capsys, test_file_tree_nodes): + rm(verbosity=1, dry_run=True) + + captured = capsys.readouterr() + assert captured.out == "" + assert captured.err == "" + + for path, content in test_file_tree_nodes: + assert path.exists() + + +def test_rm_innert_patterns(test_dir_structure, capsys, test_file_tree_nodes): + rm("nee", "shrubbery", dry_run=True) + + captured = capsys.readouterr() + assert captured.out == "" + assert captured.err == "" + + for path, content in test_file_tree_nodes: + assert path.exists() + + +def test_rm_innert_patterns_verbose(test_dir_structure, capsys, test_file_tree_nodes): + rm("nee", "shrubbery", verbosity=1, dry_run=True) + + captured = capsys.readouterr() + assert captured.out == ( + "No files or directories to delete matching 'nee'\n" + "No files or directories to delete matching 'shrubbery'\n" + ) + assert captured.err == "" + + for path, content in test_file_tree_nodes: + assert path.exists()