From dddbd799ecf9dc76408ce66cd2cf43fa85094445 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Mon, 30 Dec 2024 16:02:57 -0500 Subject: [PATCH 01/19] add checking for netcdf or hdf type and update high level file handling --- ncompare/console.py | 6 +-- ncompare/core.py | 128 +++++++++++++++++++++++++++++--------------- poetry.lock | 60 ++++++++++++++++++++- pyproject.toml | 2 + 4 files changed, 149 insertions(+), 47 deletions(-) diff --git a/ncompare/console.py b/ncompare/console.py index 8aad37a..f1fc811 100755 --- a/ncompare/console.py +++ b/ncompare/console.py @@ -47,10 +47,10 @@ def _cli(args: Optional[Sequence[str]]) -> argparse.Namespace: if None, then argparse will use `sys.argv[1:]` """ parser = argparse.ArgumentParser( - description="Compare the variables contained within two different NetCDF datasets" + description="Compare the variables contained within two different netCDF datasets" ) - parser.add_argument("nc_a", help="First NetCDF file") - parser.add_argument("nc_b", help="Second NetCDF file") + parser.add_argument("file_a", help="First (netCDF or HDF) file") + parser.add_argument("file_b", help="Second (netCDF or HDF) file") parser.add_argument( "--only-diffs", action="store_true", diff --git a/ncompare/core.py b/ncompare/core.py index 9d6ca0b..8f73526 100644 --- a/ncompare/core.py +++ b/ncompare/core.py @@ -27,14 +27,16 @@ # pylint: disable=no-member # pylint: disable=fixme -"""Compare the structure of two NetCDF files.""" +"""Compare the structure of two netCDF or HDF files.""" import warnings from collections import namedtuple from collections.abc import Iterable, Iterator +from dataclasses import dataclass from pathlib import Path -from typing import Optional, TypedDict, Union +from typing import Literal, Optional, TypedDict, Union +import h5py import netCDF4 import xarray as xr from colorama import Fore @@ -53,6 +55,24 @@ defaults=("", None, "", None), ) +valid_file_type_ids = Literal["netcdf", "hdf5"] + + +@dataclass +class FileToCompare: + path: Union[Path, str] + type: valid_file_type_ids = "netcdf" + + def __post_init__(self): + # We'll validate the inputs here. + if not isinstance(self.path, (str, Path)): + raise ValueError(f"'path' must be a str or Path, was {type(self.path)}") + if self.type not in ("netcdf", "hdf5"): + raise ValueError("'type' must be either 'netcdf' or 'hdf5'") + + def __str__(self): + return f"path: {self.path} is considered a {self.type} file" + class SummaryDifferencesDict(TypedDict): shared: int @@ -63,8 +83,8 @@ class SummaryDifferencesDict(TypedDict): def compare( - nc_a: Union[str, Path], - nc_b: Union[str, Path], + file_a: Union[str, Path], + file_b: Union[str, Path], only_diffs: bool = False, no_color: bool = False, show_chunks: bool = False, @@ -74,14 +94,14 @@ def compare( file_xlsx: Union[str, Path] = "", column_widths: Optional[tuple[Union[int, str], Union[int, str], Union[int, str]]] = None, ) -> int: - """Compare the variables contained within two different netCDF datasets. + """Compare the variables contained within two netCDF or HDF files. Parameters ---------- - nc_a - filepath to the first netCDF - nc_b - filepath to the second netCDF + file_a + filepath to the first netCDF or HDF + file_b + filepath to the second netCDF or HDF only_diffs Whether to show only the variables/attributes that are different between the two files no_color @@ -105,8 +125,8 @@ def compare( total number of differences found (across variables, groups, and attributes) """ # Check the validity of paths. - nc_a = ensure_valid_path_exists(nc_a) - nc_b = ensure_valid_path_exists(nc_b) + file_a = ensure_valid_path_exists(file_a) + file_b = ensure_valid_path_exists(file_b) if file_text: file_text = ensure_valid_path_with_suffix(file_text, ".txt") if file_csv: @@ -122,16 +142,12 @@ def compare( text_file=file_text, column_widths=column_widths, ) as out: - out.print(f"File A: {nc_a}") - out.print(f"File B: {nc_b}") + out.print(f"File A: {file_a}") + out.print(f"File B: {file_b}") # Start the comparison process. total_diff_count = run_through_comparisons( - out, - nc_a, - nc_b, - show_chunks=show_chunks, - show_attributes=show_attributes, + out, file_a, file_b, show_chunks=show_chunks, show_attributes=show_attributes ) # Write to CSV and Excel files. @@ -147,20 +163,20 @@ def compare( def run_through_comparisons( out: Outputter, - nc_a: Union[str, Path], - nc_b: Union[str, Path], + file_a: Path, + file_b: Path, show_chunks: bool, show_attributes: bool, ) -> int: - """Execute a series of comparisons between two netCDF files. + """Execute a series of comparisons between two netCDF or HDF files. Parameters ---------- out instance of Outputter - nc_a + file_a path to the first netCDF file - nc_b + file_b path to the second netCDF file show_chunks whether to include data chunk sizes in the displayed comparison of variables @@ -172,27 +188,51 @@ def run_through_comparisons( int total number of differences found (across variables, groups, and attributes) """ + # Get types of files + FileA = validate_file_type(file_a) + FileB = validate_file_type(file_b) + if FileA.type != FileB.type: + # I'm not sure if there is ever a use-case where we'd want to compare a netCDF with an HDF file? + # This assumption, of both files being the same type, affects the rest of the comparison logic. + raise TypeError("Both files must be of the same type (either both netCDF or both HDF).") + # Show the dimensions of each file and evaluate differences. out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Dimensions:", add_to_history=True) - list_a = _get_dims(nc_a) - list_b = _get_dims(nc_b) + list_a = _get_dims(FileA) + list_b = _get_dims(FileB) _, _, _ = out.lists_diff(list_a, list_b) # Show the groups in each NetCDF file and evaluate differences. out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Groups:", add_to_history=True) - list_a = _get_groups(nc_a) - list_b = _get_groups(nc_b) + list_a = _get_root_groups(FileA) + list_b = _get_root_groups(FileB) _, _, _ = out.lists_diff(list_a, list_b) out.print(Fore.LIGHTBLUE_EX + "\nAll variables:", add_to_history=True) total_diff_count = compare_two_nc_files( - out, nc_a, nc_b, show_chunks=show_chunks, show_attributes=show_attributes + out, FileA, FileB, show_chunks=show_chunks, show_attributes=show_attributes ) return total_diff_count -def walk_common_groups_tree( +def validate_file_type(file_path: Path) -> FileToCompare: + """Validate a file type and return a FileToCompare instance.""" + if file_path.suffix in (".h5", ".hdf", ".hdf5"): + file_type: valid_file_type_ids = "hdf5" + elif file_path.suffix in (".nc", ".nc4", ".nc3"): + file_type = "netcdf" + else: + raise TypeError( + f"{file_path.suffix} is not a valid file type. " + f"Expected a netcdf ('.nc', '.nc4', '.nc3') or " + f"hdf5 ('.h5', '.hdf', '.hdf5')." + ) + + return FileToCompare(path=file_path, type=file_type) + + +def _walk_common_groups_tree( top_a_name: str, top_a: Union[netCDF4.Dataset, netCDF4.Group], top_b_name: str, @@ -238,7 +278,7 @@ def walk_common_groups_tree( top_a.groups if top_a is not None else "", top_b.groups if top_b is not None else "", ): - yield from walk_common_groups_tree( + yield from _walk_common_groups_tree( top_a_name + "/" + subgroup_a_name if subgroup_a_name else "", ( top_a[subgroup_a_name] @@ -256,8 +296,8 @@ def walk_common_groups_tree( def compare_two_nc_files( out: Outputter, - nc_one: Union[str, Path], - nc_two: Union[str, Path], + file_one: FileToCompare, + file_two: FileToCompare, show_chunks: bool = False, show_attributes: bool = False, ) -> int: @@ -268,9 +308,9 @@ def compare_two_nc_files( ---------- out instance of Outputter - nc_one + file_one path to the first dataset - nc_two + file_two path to the second dataset show_chunks whether to include chunks alongside variables @@ -304,7 +344,7 @@ def compare_two_nc_files( "both": 0, "difference_types": set(), } - with netCDF4.Dataset(nc_one) as nc_a, netCDF4.Dataset(nc_two) as nc_b: + with netCDF4.Dataset(file_one.path) as nc_a, netCDF4.Dataset(file_two.path) as nc_b: out.side_by_side( "All Variables", " ", " ", dash_line=False, force_display_even_if_same=True ) @@ -325,7 +365,7 @@ def compare_two_nc_files( ) group_counter += 1 - for group_pair in walk_common_groups_tree("", nc_a, "", nc_b): + for group_pair in _walk_common_groups_tree("", nc_a, "", nc_b): if group_pair.group_a_name == "": num_group_diffs["right"] += 1 elif group_pair.group_b_name == "": @@ -622,20 +662,24 @@ def _get_attribute_value_as_str(varprops: VarProperties, attribute_key: str) -> return "" -def _get_groups(nc_filepath: Union[str, Path]) -> list: +def _get_root_groups(file: FileToCompare) -> list: """Get a list of groups from a netCDF.""" - with netCDF4.Dataset(nc_filepath) as dataset: - groups_list = list(dataset.groups.keys()) + if file.type == "netcdf": + with netCDF4.Dataset(file.path) as dataset: + groups_list = list(dataset.groups.keys()) + elif file.type == "hdf5": + with h5py.File(file.path) as dataset: + groups_list = list(dataset.keys()) return groups_list -def _get_dims(nc_filepath: Union[str, Path]) -> list: - """Get a list of dimensions from a netCDF.""" +def _get_dims(file: FileToCompare) -> list: + """Get a list of dimensions from a netCDF or HDF5.""" def __get_dim_list(decode_times=True): with warnings.catch_warnings(): warnings.simplefilter("ignore") - with xr.open_dataset(nc_filepath, decode_times=decode_times) as dataset: + with xr.open_dataset(file.path, decode_times=decode_times) as dataset: return list(dataset.sizes.items()) try: diff --git a/poetry.lock b/poetry.lock index cfb84e1..e972d4a 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry 1.8.3 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.8.4 and should not be changed by hand. [[package]] name = "appnope" @@ -601,6 +601,62 @@ python-dateutil = ">=2.8.1" [package.extras] dev = ["flake8", "markdown", "twine", "wheel"] +[[package]] +name = "h5netcdf" +version = "1.4.1" +description = "netCDF4 via h5py" +optional = false +python-versions = ">=3.9" +files = [ + {file = "h5netcdf-1.4.1-py3-none-any.whl", hash = "sha256:dd86c78ae69b92b16aa8a3c1ff3a14e7622571b5788dcf6d8b68569035bf71ce"}, + {file = "h5netcdf-1.4.1.tar.gz", hash = "sha256:7c8401ab807ff37c9798edc90d99467595892e6c541a5d5abeb8f53aab5335fe"}, +] + +[package.dependencies] +h5py = "*" +packaging = "*" + +[package.extras] +test = ["netCDF4", "pytest"] + +[[package]] +name = "h5py" +version = "3.12.1" +description = "Read and write HDF5 files from Python" +optional = false +python-versions = ">=3.9" +files = [ + {file = "h5py-3.12.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:2f0f1a382cbf494679c07b4371f90c70391dedb027d517ac94fa2c05299dacda"}, + {file = "h5py-3.12.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:cb65f619dfbdd15e662423e8d257780f9a66677eae5b4b3fc9dca70b5fd2d2a3"}, + {file = "h5py-3.12.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3b15d8dbd912c97541312c0e07438864d27dbca857c5ad634de68110c6beb1c2"}, + {file = "h5py-3.12.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:59685fe40d8c1fbbee088c88cd4da415a2f8bee5c270337dc5a1c4aa634e3307"}, + {file = "h5py-3.12.1-cp310-cp310-win_amd64.whl", hash = "sha256:577d618d6b6dea3da07d13cc903ef9634cde5596b13e832476dd861aaf651f3e"}, + {file = "h5py-3.12.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:ccd9006d92232727d23f784795191bfd02294a4f2ba68708825cb1da39511a93"}, + {file = "h5py-3.12.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ad8a76557880aed5234cfe7279805f4ab5ce16b17954606cca90d578d3e713ef"}, + {file = "h5py-3.12.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1473348139b885393125126258ae2d70753ef7e9cec8e7848434f385ae72069e"}, + {file = "h5py-3.12.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:018a4597f35092ae3fb28ee851fdc756d2b88c96336b8480e124ce1ac6fb9166"}, + {file = "h5py-3.12.1-cp311-cp311-win_amd64.whl", hash = "sha256:3fdf95092d60e8130ba6ae0ef7a9bd4ade8edbe3569c13ebbaf39baefffc5ba4"}, + {file = "h5py-3.12.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:06a903a4e4e9e3ebbc8b548959c3c2552ca2d70dac14fcfa650d9261c66939ed"}, + {file = "h5py-3.12.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:7b3b8f3b48717e46c6a790e3128d39c61ab595ae0a7237f06dfad6a3b51d5351"}, + {file = "h5py-3.12.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:050a4f2c9126054515169c49cb900949814987f0c7ae74c341b0c9f9b5056834"}, + {file = "h5py-3.12.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5c4b41d1019322a5afc5082864dfd6359f8935ecd37c11ac0029be78c5d112c9"}, + {file = "h5py-3.12.1-cp312-cp312-win_amd64.whl", hash = "sha256:e4d51919110a030913201422fb07987db4338eba5ec8c5a15d6fab8e03d443fc"}, + {file = "h5py-3.12.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:513171e90ed92236fc2ca363ce7a2fc6f2827375efcbb0cc7fbdd7fe11fecafc"}, + {file = "h5py-3.12.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:59400f88343b79655a242068a9c900001a34b63e3afb040bd7cdf717e440f653"}, + {file = "h5py-3.12.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d3e465aee0ec353949f0f46bf6c6f9790a2006af896cee7c178a8c3e5090aa32"}, + {file = "h5py-3.12.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba51c0c5e029bb5420a343586ff79d56e7455d496d18a30309616fdbeed1068f"}, + {file = "h5py-3.12.1-cp313-cp313-win_amd64.whl", hash = "sha256:52ab036c6c97055b85b2a242cb540ff9590bacfda0c03dd0cf0661b311f522f8"}, + {file = "h5py-3.12.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:d2b8dd64f127d8b324f5d2cd1c0fd6f68af69084e9e47d27efeb9e28e685af3e"}, + {file = "h5py-3.12.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4532c7e97fbef3d029735db8b6f5bf01222d9ece41e309b20d63cfaae2fb5c4d"}, + {file = "h5py-3.12.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6fdf6d7936fa824acfa27305fe2d9f39968e539d831c5bae0e0d83ed521ad1ac"}, + {file = "h5py-3.12.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:84342bffd1f82d4f036433e7039e241a243531a1d3acd7341b35ae58cdab05bf"}, + {file = "h5py-3.12.1-cp39-cp39-win_amd64.whl", hash = "sha256:62be1fc0ef195891949b2c627ec06bc8e837ff62d5b911b6e42e38e0f20a897d"}, + {file = "h5py-3.12.1.tar.gz", hash = "sha256:326d70b53d31baa61f00b8aa5f95c2fcb9621a3ee8365d770c551a13dbbcbfdf"}, +] + +[package.dependencies] +numpy = ">=1.19.3" + [[package]] name = "idna" version = "3.10" @@ -2618,4 +2674,4 @@ type = ["pytest-mypy"] [metadata] lock-version = "2.0" python-versions = "^3.9" -content-hash = "96eeae493d6ddb99129f57dd7f124315c3312da2d07bdd1dfa37f912b804145c" +content-hash = "6ccef9d5072f2cce6aff3bc6efddeeda1dda91f6b655bb655c277815f0e53253" diff --git a/pyproject.toml b/pyproject.toml index 30af360..e72f47e 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -29,6 +29,8 @@ netCDF4 = ">=1.6.4" xarray = ">=2023.9" colorama = ">=0.4.6" openpyxl = ">=3.1.2" +h5py = "^3.12.1" +h5netcdf = "^1.4.1" [tool.poetry.group.dev.dependencies] pytest = ">=7.4.2,<9.0.0" From d4934af7a755edd08538228d0ab6012e73747d23 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 09:26:13 -0500 Subject: [PATCH 02/19] add ICESat-2 test files (track with git lfs) and associated test --- .gitattributes | 2 ++ .../ATL06_20230816161508_08782002_006_02.h5 | 3 +++ .../ATL06_20230816234629_08822013_006_01.h5 | 3 +++ tests/test_core.py | 17 +++++++++++++++++ 4 files changed, 25 insertions(+) create mode 100644 .gitattributes create mode 100644 tests/data/icesat-2-ATL06/ATL06_20230816161508_08782002_006_02.h5 create mode 100644 tests/data/icesat-2-ATL06/ATL06_20230816234629_08822013_006_01.h5 diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..03bbf03 --- /dev/null +++ b/.gitattributes @@ -0,0 +1,2 @@ +tests/data/icesat-2-ATL06/ATL06_20230816161508_08782002_006_02.h5 filter=lfs diff=lfs merge=lfs -text +tests/data/icesat-2-ATL06/ATL06_20230816234629_08822013_006_01.h5 filter=lfs diff=lfs merge=lfs -text diff --git a/tests/data/icesat-2-ATL06/ATL06_20230816161508_08782002_006_02.h5 b/tests/data/icesat-2-ATL06/ATL06_20230816161508_08782002_006_02.h5 new file mode 100644 index 0000000..469647e --- /dev/null +++ b/tests/data/icesat-2-ATL06/ATL06_20230816161508_08782002_006_02.h5 @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:64184f7b614bc12a0107d4521f56ad566e958f4f657a3bda50a8cae0acb39e59 +size 3494063 diff --git a/tests/data/icesat-2-ATL06/ATL06_20230816234629_08822013_006_01.h5 b/tests/data/icesat-2-ATL06/ATL06_20230816234629_08822013_006_01.h5 new file mode 100644 index 0000000..41032ed --- /dev/null +++ b/tests/data/icesat-2-ATL06/ATL06_20230816234629_08822013_006_01.h5 @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:d0ae713f6f3d686600eafd66b36ea020ffb926198849db23a067414bd14c4280 +size 1577886 diff --git a/tests/test_core.py b/tests/test_core.py index b8595d2..1f59ef6 100644 --- a/tests/test_core.py +++ b/tests/test_core.py @@ -38,6 +38,8 @@ compare, ) +from . import data_for_tests_dir + def compare_ab(a, b): with does_not_raise(): @@ -83,3 +85,18 @@ def test_var_properties(ds_3dims_3vars_4coords_1group): assert result.shape == "(3,)" assert result.chunking == "contiguous" assert result.attributes == {} + + +def test_icesat(temp_data_dir): + # Compare the `ncompare` output when testing ICESat + out_path = temp_data_dir / "output_file_icesat-2-atl06.txt" + + num_differences = compare( + data_for_tests_dir / "icesat-2-ATL06" / "ATL06_20230816161508_08782002_006_02.h5", + data_for_tests_dir / "icesat-2-ATL06" / "ATL06_20230816234629_08822013_006_01.h5", + show_chunks=True, + show_attributes=True, + file_text=str(out_path), + ) + + assert num_differences == 4982 From 062b64f33d4a7b24d0280ffaa91126e5a6747e61 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 09:26:55 -0500 Subject: [PATCH 03/19] update tests with new hdf changes --- tests/test_cli.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tests/test_cli.py b/tests/test_cli.py index d314ae8..09bea7a 100644 --- a/tests/test_cli.py +++ b/tests/test_cli.py @@ -41,8 +41,8 @@ def test_console_help(): def test_arg_parser(): parsed = _cli(["first_netcdf.nc", "second_netcdf.nc"]) - assert getattr(parsed, "nc_a") == "first_netcdf.nc" - assert getattr(parsed, "nc_b") == "second_netcdf.nc" + assert getattr(parsed, "file_a") == "first_netcdf.nc" + assert getattr(parsed, "file_b") == "second_netcdf.nc" assert getattr(parsed, "show_attributes") is False assert getattr(parsed, "show_chunks") is False assert getattr(parsed, "only_diffs") is False From 9927b8350d40dd2339c74d28bc9c0d403d477f96 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 09:27:17 -0500 Subject: [PATCH 04/19] formatting --- README.md | 9 ++------- 1 file changed, 2 insertions(+), 7 deletions(-) diff --git a/README.md b/README.md index 3233110..3a670e6 100644 --- a/README.md +++ b/README.md @@ -75,13 +75,8 @@ ncompare S001G01.nc S001G01_SUBSET.nc --file-text subset_comparison.txt ```python from ncompare import compare -total_number_of_differences = compare( - "", - "", - only_diffs=True, - show_attributes=True, - show_chunks=True, -) +total_number_of_differences = compare("", "", only_diffs=True, + show_chunks=True, show_attributes=True) ``` From e06a99979594e2d60ce0f3d4820a9a85aa3aa1f1 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 09:28:25 -0500 Subject: [PATCH 05/19] formatting --- docs/example/ncompare-example-usage.ipynb | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/example/ncompare-example-usage.ipynb b/docs/example/ncompare-example-usage.ipynb index 689cf5b..e5e4954 100644 --- a/docs/example/ncompare-example-usage.ipynb +++ b/docs/example/ncompare-example-usage.ipynb @@ -40,7 +40,7 @@ "id": "6a145933-e57b-4e33-bed1-95b13800878d", "metadata": {}, "source": [ - "***✍️ Syntax Note:*** Commands preceeded by an exclamation point \"!\" \n", + "***✍️ Syntax Note:*** Commands are preceded by an exclamation point \"!\"\n", "(which is needed to [run shell commands in a Jupyter notebook](https://stackoverflow.com/a/48529220)) can be run from a terminal. \n", "In a shell/terminal, the exclamation point should not be used." ] @@ -785,8 +785,8 @@ " file_names[0],\n", " file_names[2],\n", " only_diffs=True,\n", - " show_attributes=True,\n", " show_chunks=True,\n", + " show_attributes=True,\n", " column_widths=[33, 30, 30],\n", ")" ] From 023dca57b783a979de9119d5eaea232e84c78375 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 09:41:03 -0500 Subject: [PATCH 06/19] add notes about hdf5 --- README.md | 16 ++++++++++++---- 1 file changed, 12 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 3a670e6..f5a6b17 100644 --- a/README.md +++ b/README.md @@ -32,8 +32,13 @@ _____ DOI badge -Compare the structure of two NetCDF files at the command line or via Python. -`ncompare` generates a view of the matching and non-matching groups and variables between two NetCDF datasets. +Compare the structure of two [netCDF](https://www.unidata.ucar.edu/software/netcdf) files +at the command line or via Python. `ncompare` generates a view of the matching and +non-matching groups and variables between two netCDF datasets. + +Allthough tailored for netCDF files, `ncompare` +also works with some [HDF5](https://www.hdfgroup.org/solutions/hdf5/) files +(see [notes and known limitations](#notes-and-known-limitations)). ## Installing @@ -134,7 +139,7 @@ poetry run ncompare ## Why ncompare? The `cdo` (climate data operators) tool -[does not support NetCDF4 groups](https://code.mpimet.mpg.de/boards/2/topics/12073). +[does not support netCDF4 groups](https://code.mpimet.mpg.de/boards/2/topics/12073). Moreover, `nco` operators' `ncdiff` function computes value differences, but --- as far as the developers of this tool are aware --- `nco` does not have a simple function to show structural differences between NetCDF4 datasets. @@ -145,8 +150,11 @@ and can generate report files formatted for other applications. However, note th `h5diff` provides comparison of some otherwise "hidden" hdf5 properties, such as _Netcdf4Dimid or _Netcdf4Coordinates, which are not currently assessed by `ncompare`. -## Known limitations +## Notes and known limitations +- `ncompare` works successfully with select HDF5 files, + although it has not been tested extensively; therefore, + it would not be surprising to find additional limitations with other HDF files. - `ncompare` uses `xarray` to access the root-level dimensions. In some cases, `xarray` will miss dimensions whose names do not also exist as variable names in the dataset (also known as non-coordinate dimensions). From 20ef547cef996601810b6df1431a22134d0e41cd Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 09:49:42 -0500 Subject: [PATCH 07/19] explicitly set xarray backend based on file type --- ncompare/core.py | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/ncompare/core.py b/ncompare/core.py index 8f73526..bb5579d 100644 --- a/ncompare/core.py +++ b/ncompare/core.py @@ -198,8 +198,8 @@ def run_through_comparisons( # Show the dimensions of each file and evaluate differences. out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Dimensions:", add_to_history=True) - list_a = _get_dims(FileA) - list_b = _get_dims(FileB) + list_a = _get_root_dims(FileA) + list_b = _get_root_dims(FileB) _, _, _ = out.lists_diff(list_a, list_b) # Show the groups in each NetCDF file and evaluate differences. @@ -673,13 +673,20 @@ def _get_root_groups(file: FileToCompare) -> list: return groups_list -def _get_dims(file: FileToCompare) -> list: +def _get_root_dims(file: FileToCompare) -> list: """Get a list of dimensions from a netCDF or HDF5.""" def __get_dim_list(decode_times=True): with warnings.catch_warnings(): warnings.simplefilter("ignore") - with xr.open_dataset(file.path, decode_times=decode_times) as dataset: + if file.type == "netcdf": + xarray_engine = "netcdf4" + elif file.type == "hdf5": + xarray_engine = "h5netcdf" + + with xr.open_dataset( + file.path, decode_times=decode_times, engine=xarray_engine + ) as dataset: return list(dataset.sizes.items()) try: From 756fc2e4ce8f4724ed0eef5055e77d5d0db5f263 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 10:06:20 -0500 Subject: [PATCH 08/19] include lfs in workflow checkout for tests to pass --- .github/workflows/reusable_run_tests.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/.github/workflows/reusable_run_tests.yml b/.github/workflows/reusable_run_tests.yml index e970e9b..96037d0 100644 --- a/.github/workflows/reusable_run_tests.yml +++ b/.github/workflows/reusable_run_tests.yml @@ -22,6 +22,8 @@ jobs: name: Python ${{ matrix.python-version }} tests steps: - uses: actions/checkout@v4 + with: + lfs: 'true' - name: Set up Python uses: actions/setup-python@v5 From d01793e66d9a5c62cc9e64a4d048161438f2d3c0 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 10:30:13 -0500 Subject: [PATCH 09/19] refactor: types and getters to modules and remove redundant core function --- ncompare/core.py | 437 ++++++++++++----------------------------- ncompare/core_types.py | 40 ++++ ncompare/getters.py | 144 ++++++++++++++ tests/test_core.py | 4 +- 4 files changed, 308 insertions(+), 317 deletions(-) create mode 100644 ncompare/core_types.py create mode 100644 ncompare/getters.py diff --git a/ncompare/core.py b/ncompare/core.py index bb5579d..40b56f6 100644 --- a/ncompare/core.py +++ b/ncompare/core.py @@ -29,62 +29,35 @@ """Compare the structure of two netCDF or HDF files.""" -import warnings -from collections import namedtuple -from collections.abc import Iterable, Iterator -from dataclasses import dataclass +from collections.abc import Iterator from pathlib import Path -from typing import Literal, Optional, TypedDict, Union +from typing import Optional, Union -import h5py import netCDF4 -import xarray as xr from colorama import Fore +from ncompare.core_types import ( + FileToCompare, + GroupPair, + SummaryDifferencesDict, + VarProperties, + valid_file_type_ids, +) +from ncompare.getters import ( + _get_and_check_variable_attributes, + _get_and_check_variable_scale_factor, + _get_root_dims, + _get_root_groups, + _get_var_properties, +) from ncompare.printing import Outputter, SummaryDifferenceKeys from ncompare.sequence_operations import common_elements, count_diffs from ncompare.utils import ensure_valid_path_exists, ensure_valid_path_with_suffix -VarProperties = namedtuple( - "VarProperties", "varname, variable, dtype, dimensions, shape, chunking, attributes" -) - -GroupPair = namedtuple( - "GroupPair", - "group_a_name group_a group_b_name group_b", - defaults=("", None, "", None), -) - -valid_file_type_ids = Literal["netcdf", "hdf5"] - - -@dataclass -class FileToCompare: - path: Union[Path, str] - type: valid_file_type_ids = "netcdf" - - def __post_init__(self): - # We'll validate the inputs here. - if not isinstance(self.path, (str, Path)): - raise ValueError(f"'path' must be a str or Path, was {type(self.path)}") - if self.type not in ("netcdf", "hdf5"): - raise ValueError("'type' must be either 'netcdf' or 'hdf5'") - - def __str__(self): - return f"path: {self.path} is considered a {self.type} file" - - -class SummaryDifferencesDict(TypedDict): - shared: int - left: int - right: int - both: int - difference_types: set - def compare( - file_a: Union[str, Path], - file_b: Union[str, Path], + path_a: Union[str, Path], + path_b: Union[str, Path], only_diffs: bool = False, no_color: bool = False, show_chunks: bool = False, @@ -98,9 +71,9 @@ def compare( Parameters ---------- - file_a + path_a filepath to the first netCDF or HDF - file_b + path_b filepath to the second netCDF or HDF only_diffs Whether to show only the variables/attributes that are different between the two files @@ -125,8 +98,8 @@ def compare( total number of differences found (across variables, groups, and attributes) """ # Check the validity of paths. - file_a = ensure_valid_path_exists(file_a) - file_b = ensure_valid_path_exists(file_b) + path_a = ensure_valid_path_exists(path_a) + path_b = ensure_valid_path_exists(path_b) if file_text: file_text = ensure_valid_path_with_suffix(file_text, ".txt") if file_csv: @@ -134,6 +107,14 @@ def compare( if file_xlsx: file_xlsx = ensure_valid_path_with_suffix(file_xlsx, ".xlsx") + # Check the validity of file types + file_a = _validate_file_type(path_a) + file_b = _validate_file_type(path_b) + if file_a.type != file_b.type: + # I'm not sure if there is a use-case where we'd want to compare a netCDF with an HDF file? + # This assumption of files being the same type, affects the rest of the comparison logic. + raise TypeError("Both files must be of the same type (either both netCDF or both HDF).") + # The Outputter object is initialized to handle stdout and optional writing to a text file. with Outputter( keep_print_history=True, @@ -142,11 +123,11 @@ def compare( text_file=file_text, column_widths=column_widths, ) as out: - out.print(f"File A: {file_a}") - out.print(f"File B: {file_b}") + out.print(f"File A: {file_a.path}") + out.print(f"File B: {file_b.path}") # Start the comparison process. - total_diff_count = run_through_comparisons( + total_diff_count = _run_through_comparisons( out, file_a, file_b, show_chunks=show_chunks, show_attributes=show_attributes ) @@ -161,10 +142,10 @@ def compare( return total_diff_count -def run_through_comparisons( +def _run_through_comparisons( out: Outputter, - file_a: Path, - file_b: Path, + file_a: FileToCompare, + file_b: FileToCompare, show_chunks: bool, show_attributes: bool, ) -> int: @@ -175,9 +156,9 @@ def run_through_comparisons( out instance of Outputter file_a - path to the first netCDF file + path and type (netCDF or HDF5) of the first file file_b - path to the second netCDF file + path and type (netCDF or HDF5) of the second file show_chunks whether to include data chunk sizes in the displayed comparison of variables show_attributes @@ -188,140 +169,20 @@ def run_through_comparisons( int total number of differences found (across variables, groups, and attributes) """ - # Get types of files - FileA = validate_file_type(file_a) - FileB = validate_file_type(file_b) - if FileA.type != FileB.type: - # I'm not sure if there is ever a use-case where we'd want to compare a netCDF with an HDF file? - # This assumption, of both files being the same type, affects the rest of the comparison logic. - raise TypeError("Both files must be of the same type (either both netCDF or both HDF).") - # Show the dimensions of each file and evaluate differences. out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Dimensions:", add_to_history=True) - list_a = _get_root_dims(FileA) - list_b = _get_root_dims(FileB) + list_a = _get_root_dims(file_a) + list_b = _get_root_dims(file_b) _, _, _ = out.lists_diff(list_a, list_b) # Show the groups in each NetCDF file and evaluate differences. out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Groups:", add_to_history=True) - list_a = _get_root_groups(FileA) - list_b = _get_root_groups(FileB) + list_a = _get_root_groups(file_a) + list_b = _get_root_groups(file_b) _, _, _ = out.lists_diff(list_a, list_b) + # Run through all the rest of the groups and variables, tallying differences along the way. out.print(Fore.LIGHTBLUE_EX + "\nAll variables:", add_to_history=True) - total_diff_count = compare_two_nc_files( - out, FileA, FileB, show_chunks=show_chunks, show_attributes=show_attributes - ) - - return total_diff_count - - -def validate_file_type(file_path: Path) -> FileToCompare: - """Validate a file type and return a FileToCompare instance.""" - if file_path.suffix in (".h5", ".hdf", ".hdf5"): - file_type: valid_file_type_ids = "hdf5" - elif file_path.suffix in (".nc", ".nc4", ".nc3"): - file_type = "netcdf" - else: - raise TypeError( - f"{file_path.suffix} is not a valid file type. " - f"Expected a netcdf ('.nc', '.nc4', '.nc3') or " - f"hdf5 ('.h5', '.hdf', '.hdf5')." - ) - - return FileToCompare(path=file_path, type=file_type) - - -def _walk_common_groups_tree( - top_a_name: str, - top_a: Union[netCDF4.Dataset, netCDF4.Group], - top_b_name: str, - top_b: Union[netCDF4.Dataset, netCDF4.Group], -) -> Iterator[GroupPair]: - """Yield names and groups from a netCDF4's group tree. - - Parameters - ---------- - top_a_name - name of the first group or dataset - top_a - the first group or dataset - top_b_name - name of the second group or dataset - top_b - the second group or dataset - - Yields - ------ - tuple - group A name : str - group A object : netCDF4.Group or None - group B name : str - group B object : netCDF4.Group or None - """ - for _, group_a_name, group_b_name in common_elements( - top_a.groups if top_a is not None else "", - top_b.groups if top_b is not None else "", - ): - yield GroupPair( - group_a_name=top_a_name + "/" + group_a_name if group_a_name else "", - group_a=( - top_a[group_a_name] if (group_a_name and (group_a_name in top_a.groups)) else None - ), - group_b_name=top_b_name + "/" + group_b_name if group_b_name else "", - group_b=( - top_b[group_b_name] if (group_b_name and (group_b_name in top_b.groups)) else None - ), - ) - - for _, subgroup_a_name, subgroup_b_name in common_elements( - top_a.groups if top_a is not None else "", - top_b.groups if top_b is not None else "", - ): - yield from _walk_common_groups_tree( - top_a_name + "/" + subgroup_a_name if subgroup_a_name else "", - ( - top_a[subgroup_a_name] - if (subgroup_a_name and (subgroup_a_name in top_a.groups)) - else None - ), - top_a_name + "/" + subgroup_b_name if subgroup_b_name else "", - ( - top_b[subgroup_b_name] - if (subgroup_b_name and (subgroup_b_name in top_b.groups)) - else None - ), - ) - - -def compare_two_nc_files( - out: Outputter, - file_one: FileToCompare, - file_two: FileToCompare, - show_chunks: bool = False, - show_attributes: bool = False, -) -> int: - """Go through all groups and all variables, and show them side by side, - highlighting whether they align and where they don't. - - Parameters - ---------- - out - instance of Outputter - file_one - path to the first dataset - file_two - path to the second dataset - show_chunks - whether to include chunks alongside variables - show_attributes - whether to include variable attributes - - Returns - ------- - int - total number of differences found (across variables, groups, and attributes) - """ out.side_by_side(" ", "File A", "File B", force_display_even_if_same=True) num_group_diffs: SummaryDifferencesDict = { "shared": 0, @@ -344,7 +205,7 @@ def compare_two_nc_files( "both": 0, "difference_types": set(), } - with netCDF4.Dataset(file_one.path) as nc_a, netCDF4.Dataset(file_two.path) as nc_b: + with netCDF4.Dataset(file_a.path) as nc_a, netCDF4.Dataset(file_b.path) as nc_b: out.side_by_side( "All Variables", " ", " ", dash_line=False, force_display_even_if_same=True ) @@ -387,9 +248,9 @@ def compare_two_nc_files( ) group_counter += 1 + # Print summary counts of similarities and differences at the end of the comparison report. out.side_by_side("-", "-", "-", dash_line=True, force_display_even_if_same=True) out.side_by_side("SUMMARY", "-", "-", dash_line=True, force_display_even_if_same=True) - _print_summary_count_comparison_side_by_side(out, "variable", num_var_diffs) _print_summary_count_comparison_side_by_side(out, "group", num_group_diffs) _print_summary_count_comparison_side_by_side(out, "attribute", num_attribute_diffs) @@ -402,7 +263,7 @@ def compare_two_nc_files( add_to_history=True, ) - # Return the total number of differences; thus, zero means no differences were found. + # Return the total number of differences; zero indicates no differences were found. total_diff_count = sum( [x["left"] + x["right"] for x in [num_var_diffs, num_group_diffs, num_attribute_diffs]] ) @@ -410,6 +271,68 @@ def compare_two_nc_files( return total_diff_count +def _walk_common_groups_tree( + top_a_name: str, + top_a: Union[netCDF4.Dataset, netCDF4.Group], + top_b_name: str, + top_b: Union[netCDF4.Dataset, netCDF4.Group], +) -> Iterator[GroupPair]: + """Yield names and groups from a netCDF4 or HDF's group tree. + + Parameters + ---------- + top_a_name + name of the first group or dataset + top_a + the first group or dataset + top_b_name + name of the second group or dataset + top_b + the second group or dataset + + Yields + ------ + tuple + group A name : str + group A object : netCDF4.Group or None + group B name : str + group B object : netCDF4.Group or None + """ + for _, group_a_name, group_b_name in common_elements( + top_a.groups if top_a is not None else "", + top_b.groups if top_b is not None else "", + ): + yield GroupPair( + group_a_name=top_a_name + "/" + group_a_name if group_a_name else "", + group_a=( + top_a[group_a_name] if (group_a_name and (group_a_name in top_a.groups)) else None + ), + group_b_name=top_b_name + "/" + group_b_name if group_b_name else "", + group_b=( + top_b[group_b_name] if (group_b_name and (group_b_name in top_b.groups)) else None + ), + ) + + for _, subgroup_a_name, subgroup_b_name in common_elements( + top_a.groups if top_a is not None else "", + top_b.groups if top_b is not None else "", + ): + yield from _walk_common_groups_tree( + top_a_name + "/" + subgroup_a_name if subgroup_a_name else "", + ( + top_a[subgroup_a_name] + if (subgroup_a_name and (subgroup_a_name in top_a.groups)) + else None + ), + top_a_name + "/" + subgroup_b_name if subgroup_b_name else "", + ( + top_b[subgroup_b_name] + if (subgroup_b_name and (subgroup_b_name in top_b.groups)) + else None + ), + ) + + def _print_summary_count_comparison_side_by_side( out: Outputter, item_type: str, @@ -486,8 +409,8 @@ def _print_group_details_side_by_side( # Get and print the properties of each variable _print_var_properties_side_by_side( out, - _var_properties(group_a, variable_pair[1]), - _var_properties(group_b, variable_pair[2]), + _get_var_properties(group_a, variable_pair[1]), + _get_var_properties(group_b, variable_pair[2]), num_attribute_diffs, show_chunks=show_chunks, show_attributes=show_attributes, @@ -568,133 +491,17 @@ def _var_attribute_side_by_side(attribute_name, attribute_a, attribute_b): _var_attribute_side_by_side(attribute_key, attr_a, attr_b) -def _get_and_check_variable_scale_factor( - v_a: VarProperties, v_b: VarProperties -) -> Union[None, tuple[str, str]]: - """Get a string representation of the scale factor for two variables.""" - if getattr(v_a.variable, "scale_factor", None): - sf_a = v_a.variable.scale_factor - else: - sf_a = " " - if getattr(v_b.variable, "scale_factor", None): - sf_b = v_b.variable.scale_factor - else: - sf_b = " " - if (sf_a != " ") or (sf_b != " "): - return str(sf_a), str(sf_b) - else: - return None - - -def _get_and_check_variable_attributes( - v_a: VarProperties, v_b: VarProperties -) -> Iterator[tuple[str, str, str, str]]: - """Go through and yield each attribute pair for two variables.""" - # Get the name of attributes if they exist - attrs_a_names = [] - if v_a.attributes: - attrs_a_names = v_a.attributes.keys() - attrs_b_names = [] - if v_b.attributes: - attrs_b_names = v_b.attributes.keys() - # Iterate and print each attribute - for _, attr_a_key, attr_b_key in common_elements(attrs_a_names, attrs_b_names): - attr_a = _get_attribute_value_as_str(v_a, attr_a_key) - attr_b = _get_attribute_value_as_str(v_b, attr_b_key) - yield attr_a_key, attr_a, attr_b_key, attr_b - - -def _var_properties(group: Union[netCDF4.Dataset, netCDF4.Group], varname: str) -> VarProperties: - """Get the properties of a variable. - - Parameters - ---------- - group - a dataset or group of variables - varname - the name of the variable - - Returns - ------- - VarProperties - """ - if varname: - the_variable = group.variables[varname] - v_dtype = str(the_variable.dtype) - v_dimensions = str(the_variable.dimensions) - v_shape = str(the_variable.shape).strip() - v_chunking = str(the_variable.chunking()).strip() - - v_attributes = {} - for name in the_variable.ncattrs(): - try: - v_attributes[name] = the_variable.getncattr(name) - except KeyError as key_err: - # Added this check because of "unsupported datatype" error that prevented - # fully running comparisons on S5P_OFFL_L1B_IR_UVN collections. - v_attributes[name] = f"netCDF error: {str(key_err)}" +def _validate_file_type(file_path: Path) -> FileToCompare: + """Validate a file type and return a FileToCompare instance.""" + if file_path.suffix in (".h5", ".hdf", ".hdf5"): + file_type: valid_file_type_ids = "hdf5" + elif file_path.suffix in (".nc", ".nc4", ".nc3"): + file_type = "netcdf" else: - the_variable = None - v_dtype = "" - v_dimensions = "" - v_shape = "" - v_chunking = "" - v_attributes = None - - return VarProperties( - varname, the_variable, v_dtype, v_dimensions, v_shape, v_chunking, v_attributes - ) - + raise TypeError( + f"{file_path.suffix} is not a valid file type. " + f"Expected a netcdf ('.nc', '.nc4', '.nc3') or " + f"hdf5 ('.h5', '.hdf', '.hdf5')." + ) -def _get_attribute_value_as_str(varprops: VarProperties, attribute_key: str) -> str: - """Get a string representation of the attribute value.""" - if attribute_key and (attribute_key in varprops.attributes): - attr = varprops.attributes[attribute_key] - if isinstance(attr, Iterable) and not isinstance(attr, (str, float)): - # TODO: by truncating a list (or other iterable) here, - # we are preventing any subsequent difference checker from detecting - # differences past the 5th element in the iterable. - # So, we need to figure out a way to still check for other differences past the 5th element. - return "[" + ", ".join([str(x) for x in attr[:5]]) + ", ..." + "]" # type:ignore[index] - - return str(attr) - - return "" - - -def _get_root_groups(file: FileToCompare) -> list: - """Get a list of groups from a netCDF.""" - if file.type == "netcdf": - with netCDF4.Dataset(file.path) as dataset: - groups_list = list(dataset.groups.keys()) - elif file.type == "hdf5": - with h5py.File(file.path) as dataset: - groups_list = list(dataset.keys()) - return groups_list - - -def _get_root_dims(file: FileToCompare) -> list: - """Get a list of dimensions from a netCDF or HDF5.""" - - def __get_dim_list(decode_times=True): - with warnings.catch_warnings(): - warnings.simplefilter("ignore") - if file.type == "netcdf": - xarray_engine = "netcdf4" - elif file.type == "hdf5": - xarray_engine = "h5netcdf" - - with xr.open_dataset( - file.path, decode_times=decode_times, engine=xarray_engine - ) as dataset: - return list(dataset.sizes.items()) - - try: - dims_list = __get_dim_list() - except ValueError as err: - if "decode_times" in str(err): # then try again without decoding the times - dims_list = __get_dim_list(decode_times=False) - else: - raise err from None # "from None" prevents additional trace (see https://stackoverflow.com/a/18188660) - - return dims_list + return FileToCompare(path=file_path, type=file_type) diff --git a/ncompare/core_types.py b/ncompare/core_types.py new file mode 100644 index 0000000..22ad5b6 --- /dev/null +++ b/ncompare/core_types.py @@ -0,0 +1,40 @@ +from collections import namedtuple +from dataclasses import dataclass +from pathlib import Path +from typing import Literal, TypedDict, Union + +VarProperties = namedtuple( + "VarProperties", "varname, variable, dtype, dimensions, shape, chunking, attributes" +) + +GroupPair = namedtuple( + "GroupPair", + "group_a_name group_a group_b_name group_b", + defaults=("", None, "", None), +) + +valid_file_type_ids = Literal["netcdf", "hdf5"] + + +@dataclass +class FileToCompare: + path: Union[Path, str] + type: valid_file_type_ids = "netcdf" + + def __post_init__(self): + # We'll validate the inputs here. + if not isinstance(self.path, (str, Path)): + raise ValueError(f"'path' must be a str or Path, was {type(self.path)}") + if self.type not in ("netcdf", "hdf5"): + raise ValueError("'type' must be either 'netcdf' or 'hdf5'") + + def __str__(self): + return f"path: {self.path} is considered a {self.type} file" + + +class SummaryDifferencesDict(TypedDict): + shared: int + left: int + right: int + both: int + difference_types: set diff --git a/ncompare/getters.py b/ncompare/getters.py new file mode 100644 index 0000000..791c45b --- /dev/null +++ b/ncompare/getters.py @@ -0,0 +1,144 @@ +import warnings +from collections.abc import Iterable, Iterator +from typing import Union + +import h5py +import netCDF4 +import xarray as xr + +from ncompare.core_types import FileToCompare, VarProperties +from ncompare.sequence_operations import common_elements + + +def _get_and_check_variable_scale_factor( + v_a: VarProperties, v_b: VarProperties +) -> Union[None, tuple[str, str]]: + """Get a string representation of the scale factor for two variables.""" + if getattr(v_a.variable, "scale_factor", None): + sf_a = v_a.variable.scale_factor + else: + sf_a = " " + if getattr(v_b.variable, "scale_factor", None): + sf_b = v_b.variable.scale_factor + else: + sf_b = " " + if (sf_a != " ") or (sf_b != " "): + return str(sf_a), str(sf_b) + else: + return None + + +def _get_and_check_variable_attributes( + v_a: VarProperties, v_b: VarProperties +) -> Iterator[tuple[str, str, str, str]]: + """Go through and yield each attribute pair for two variables.""" + # Get the name of attributes if they exist + attrs_a_names = [] + if v_a.attributes: + attrs_a_names = v_a.attributes.keys() + attrs_b_names = [] + if v_b.attributes: + attrs_b_names = v_b.attributes.keys() + # Iterate and print each attribute + for _, attr_a_key, attr_b_key in common_elements(attrs_a_names, attrs_b_names): + attr_a = _get_attribute_value_as_str(v_a, attr_a_key) + attr_b = _get_attribute_value_as_str(v_b, attr_b_key) + yield attr_a_key, attr_a, attr_b_key, attr_b + + +def _get_var_properties( + group: Union[netCDF4.Dataset, netCDF4.Group], varname: str +) -> VarProperties: + """Get the properties of a variable. + + Parameters + ---------- + group + a dataset or group of variables + varname + the name of the variable + + Returns + ------- + VarProperties + """ + if varname: + the_variable = group.variables[varname] + v_dtype = str(the_variable.dtype) + v_dimensions = str(the_variable.dimensions) + v_shape = str(the_variable.shape).strip() + v_chunking = str(the_variable.chunking()).strip() + + v_attributes = {} + for name in the_variable.ncattrs(): + try: + v_attributes[name] = the_variable.getncattr(name) + except KeyError as key_err: + # Added this check because of "unsupported datatype" error that prevented + # fully running comparisons on S5P_OFFL_L1B_IR_UVN collections. + v_attributes[name] = f"netCDF error: {str(key_err)}" + else: + the_variable = None + v_dtype = "" + v_dimensions = "" + v_shape = "" + v_chunking = "" + v_attributes = None + + return VarProperties( + varname, the_variable, v_dtype, v_dimensions, v_shape, v_chunking, v_attributes + ) + + +def _get_attribute_value_as_str(varprops: VarProperties, attribute_key: str) -> str: + """Get a string representation of the attribute value.""" + if attribute_key and (attribute_key in varprops.attributes): + attr = varprops.attributes[attribute_key] + if isinstance(attr, Iterable) and not isinstance(attr, (str, float)): + # TODO: by truncating a list (or other iterable) here, + # we are preventing any subsequent difference checker from detecting + # differences past the 5th element in the iterable. + # So, we need to figure out a way to still check for other differences past the 5th element. + return "[" + ", ".join([str(x) for x in attr[:5]]) + ", ..." + "]" # type:ignore[index] + + return str(attr) + + return "" + + +def _get_root_groups(file: FileToCompare) -> list: + """Get a list of groups from a netCDF.""" + if file.type == "netcdf": + with netCDF4.Dataset(file.path) as dataset: + groups_list = list(dataset.groups.keys()) + elif file.type == "hdf5": + with h5py.File(file.path) as dataset: + groups_list = list(dataset.keys()) + return groups_list + + +def _get_root_dims(file: FileToCompare) -> list: + """Get a list of dimensions from a netCDF or HDF5.""" + + def __get_dim_list(decode_times=True): + with warnings.catch_warnings(): + warnings.simplefilter("ignore") + if file.type == "netcdf": + xarray_engine = "netcdf4" + elif file.type == "hdf5": + xarray_engine = "h5netcdf" + + with xr.open_dataset( + file.path, decode_times=decode_times, engine=xarray_engine + ) as dataset: + return list(dataset.sizes.items()) + + try: + dims_list = __get_dim_list() + except ValueError as err: + if "decode_times" in str(err): # then try again without decoding the times + dims_list = __get_dim_list(decode_times=False) + else: + raise err from None # "from None" prevents additional trace (see https://stackoverflow.com/a/18188660) + + return dims_list diff --git a/tests/test_core.py b/tests/test_core.py index 1f59ef6..407de8e 100644 --- a/tests/test_core.py +++ b/tests/test_core.py @@ -34,9 +34,9 @@ import netCDF4 as nc from ncompare.core import ( - _var_properties, compare, ) +from ncompare.getters import _get_var_properties from . import data_for_tests_dir @@ -79,7 +79,7 @@ def test_zero_for_comparison_with_no_differences(ds_3dims_3vars_4coords_1subgrou def test_var_properties(ds_3dims_3vars_4coords_1group): with nc.Dataset(ds_3dims_3vars_4coords_1group) as ds: - result = _var_properties(ds.groups["Group1"], varname="step") + result = _get_var_properties(ds.groups["Group1"], varname="step") assert result.varname == "step" assert result.dtype == "float32" assert result.shape == "(3,)" From d60d33875e27b0fcea77a4da9eba60cde59993ca Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 12:22:32 -0500 Subject: [PATCH 10/19] simplify logic for scale factor getattr --- ncompare/getters.py | 11 +++-------- 1 file changed, 3 insertions(+), 8 deletions(-) diff --git a/ncompare/getters.py b/ncompare/getters.py index 791c45b..6f1b083 100644 --- a/ncompare/getters.py +++ b/ncompare/getters.py @@ -14,14 +14,9 @@ def _get_and_check_variable_scale_factor( v_a: VarProperties, v_b: VarProperties ) -> Union[None, tuple[str, str]]: """Get a string representation of the scale factor for two variables.""" - if getattr(v_a.variable, "scale_factor", None): - sf_a = v_a.variable.scale_factor - else: - sf_a = " " - if getattr(v_b.variable, "scale_factor", None): - sf_b = v_b.variable.scale_factor - else: - sf_b = " " + sf_a = getattr(v_a.variable, "scale_factor", " ") + sf_b = getattr(v_b.variable, "scale_factor", " ") + if (sf_a != " ") or (sf_b != " "): return str(sf_a), str(sf_b) else: From 520baf244226a60e6558d80220e501d5defdbed4 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 12:26:33 -0500 Subject: [PATCH 11/19] adjust prefix to function names in new module --- ncompare/core.py | 30 +++++++++++++++--------------- ncompare/getters.py | 18 ++++++++---------- 2 files changed, 23 insertions(+), 25 deletions(-) diff --git a/ncompare/core.py b/ncompare/core.py index 40b56f6..624dd40 100644 --- a/ncompare/core.py +++ b/ncompare/core.py @@ -44,11 +44,11 @@ valid_file_type_ids, ) from ncompare.getters import ( - _get_and_check_variable_attributes, - _get_and_check_variable_scale_factor, - _get_root_dims, - _get_root_groups, - _get_var_properties, + get_and_check_variable_attributes, + get_and_check_variable_scale_factor, + get_root_dims, + get_root_groups, + get_var_properties, ) from ncompare.printing import Outputter, SummaryDifferenceKeys from ncompare.sequence_operations import common_elements, count_diffs @@ -171,14 +171,14 @@ def _run_through_comparisons( """ # Show the dimensions of each file and evaluate differences. out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Dimensions:", add_to_history=True) - list_a = _get_root_dims(file_a) - list_b = _get_root_dims(file_b) + list_a = get_root_dims(file_a) + list_b = get_root_dims(file_b) _, _, _ = out.lists_diff(list_a, list_b) # Show the groups in each NetCDF file and evaluate differences. out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Groups:", add_to_history=True) - list_a = _get_root_groups(file_a) - list_b = _get_root_groups(file_b) + list_a = get_root_groups(file_a) + list_b = get_root_groups(file_b) _, _, _ = out.lists_diff(list_a, list_b) # Run through all the rest of the groups and variables, tallying differences along the way. @@ -409,8 +409,8 @@ def _print_group_details_side_by_side( # Get and print the properties of each variable _print_var_properties_side_by_side( out, - _get_var_properties(group_a, variable_pair[1]), - _get_var_properties(group_b, variable_pair[2]), + get_var_properties(group_a, variable_pair[1]), + get_var_properties(group_b, variable_pair[2]), num_attribute_diffs, show_chunks=show_chunks, show_attributes=show_attributes, @@ -436,12 +436,12 @@ def _print_var_properties_side_by_side( if show_chunks: pairs_to_check_and_show.append((v_a.chunking, v_b.chunking)) if show_attributes: - for attr_a_key, attr_a, attr_b_key, attr_b in _get_and_check_variable_attributes(v_a, v_b): + for attr_a_key, attr_a, attr_b_key, attr_b in get_and_check_variable_attributes(v_a, v_b): # Check whether attr_a_key is empty, # because it might be if the variable doesn't exist in File A. pairs_to_check_and_show.append((attr_a, attr_b)) # Scale Factor - scale_factor_pair = _get_and_check_variable_scale_factor(v_a, v_b) + scale_factor_pair = get_and_check_variable_scale_factor(v_a, v_b) if scale_factor_pair: pairs_to_check_and_show.append((scale_factor_pair[0], scale_factor_pair[1])) @@ -479,12 +479,12 @@ def _var_attribute_side_by_side(attribute_name, attribute_a, attribute_b): if show_chunks: _var_attribute_side_by_side("chunksize", v_a.chunking, v_b.chunking) # Scale Factor - scale_factor_pair = _get_and_check_variable_scale_factor(v_a, v_b) + scale_factor_pair = get_and_check_variable_scale_factor(v_a, v_b) if scale_factor_pair: _var_attribute_side_by_side("scale_factor", scale_factor_pair[0], scale_factor_pair[1]) # Other attributes if show_attributes: - for attr_a_key, attr_a, attr_b_key, attr_b in _get_and_check_variable_attributes(v_a, v_b): + for attr_a_key, attr_a, attr_b_key, attr_b in get_and_check_variable_attributes(v_a, v_b): # Check whether attr_a_key is empty, # because it might be if the variable doesn't exist in File A. attribute_key = attr_a_key if attr_a_key else attr_b_key diff --git a/ncompare/getters.py b/ncompare/getters.py index 6f1b083..a109b00 100644 --- a/ncompare/getters.py +++ b/ncompare/getters.py @@ -10,7 +10,7 @@ from ncompare.sequence_operations import common_elements -def _get_and_check_variable_scale_factor( +def get_and_check_variable_scale_factor( v_a: VarProperties, v_b: VarProperties ) -> Union[None, tuple[str, str]]: """Get a string representation of the scale factor for two variables.""" @@ -23,7 +23,7 @@ def _get_and_check_variable_scale_factor( return None -def _get_and_check_variable_attributes( +def get_and_check_variable_attributes( v_a: VarProperties, v_b: VarProperties ) -> Iterator[tuple[str, str, str, str]]: """Go through and yield each attribute pair for two variables.""" @@ -36,14 +36,12 @@ def _get_and_check_variable_attributes( attrs_b_names = v_b.attributes.keys() # Iterate and print each attribute for _, attr_a_key, attr_b_key in common_elements(attrs_a_names, attrs_b_names): - attr_a = _get_attribute_value_as_str(v_a, attr_a_key) - attr_b = _get_attribute_value_as_str(v_b, attr_b_key) + attr_a = get_attribute_value_as_str(v_a, attr_a_key) + attr_b = get_attribute_value_as_str(v_b, attr_b_key) yield attr_a_key, attr_a, attr_b_key, attr_b -def _get_var_properties( - group: Union[netCDF4.Dataset, netCDF4.Group], varname: str -) -> VarProperties: +def get_var_properties(group: Union[netCDF4.Dataset, netCDF4.Group], varname: str) -> VarProperties: """Get the properties of a variable. Parameters @@ -85,7 +83,7 @@ def _get_var_properties( ) -def _get_attribute_value_as_str(varprops: VarProperties, attribute_key: str) -> str: +def get_attribute_value_as_str(varprops: VarProperties, attribute_key: str) -> str: """Get a string representation of the attribute value.""" if attribute_key and (attribute_key in varprops.attributes): attr = varprops.attributes[attribute_key] @@ -101,7 +99,7 @@ def _get_attribute_value_as_str(varprops: VarProperties, attribute_key: str) -> return "" -def _get_root_groups(file: FileToCompare) -> list: +def get_root_groups(file: FileToCompare) -> list: """Get a list of groups from a netCDF.""" if file.type == "netcdf": with netCDF4.Dataset(file.path) as dataset: @@ -112,7 +110,7 @@ def _get_root_groups(file: FileToCompare) -> list: return groups_list -def _get_root_dims(file: FileToCompare) -> list: +def get_root_dims(file: FileToCompare) -> list: """Get a list of dimensions from a netCDF or HDF5.""" def __get_dim_list(decode_times=True): From b9d7de6a382c11d906db8b30d5e5c7dad7ce6c75 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Tue, 31 Dec 2024 16:51:45 -0500 Subject: [PATCH 12/19] update tests --- tests/conftest.py | 2 ++ tests/test_core.py | 17 +---------------- tests/test_getters.py | 21 +++++++++++++++++++++ 3 files changed, 24 insertions(+), 16 deletions(-) create mode 100644 tests/test_getters.py diff --git a/tests/conftest.py b/tests/conftest.py index 6a11927..4a763a4 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -147,6 +147,8 @@ def ds_3dims_3vars_4coords_1group(temp_data_dir): # grp1.createVariable("step", "f4", ("step",), fill_value=False) grp1["step"][:] = [-0.9, -1.8, -2.7] + grp1["step"].scale_factor = 0.5 + grp1["step"].add_offset = 5 # grp1.createVariable("w", "u1", ("x", "step"), fill_value=False) diff --git a/tests/test_core.py b/tests/test_core.py index 407de8e..2b2c95f 100644 --- a/tests/test_core.py +++ b/tests/test_core.py @@ -31,12 +31,7 @@ from contextlib import nullcontext as does_not_raise -import netCDF4 as nc - -from ncompare.core import ( - compare, -) -from ncompare.getters import _get_var_properties +from ncompare.core import compare from . import data_for_tests_dir @@ -77,16 +72,6 @@ def test_zero_for_comparison_with_no_differences(ds_3dims_3vars_4coords_1subgrou assert compare(ds_3dims_3vars_4coords_1subgroup, ds_3dims_3vars_4coords_1subgroup) == 0 -def test_var_properties(ds_3dims_3vars_4coords_1group): - with nc.Dataset(ds_3dims_3vars_4coords_1group) as ds: - result = _get_var_properties(ds.groups["Group1"], varname="step") - assert result.varname == "step" - assert result.dtype == "float32" - assert result.shape == "(3,)" - assert result.chunking == "contiguous" - assert result.attributes == {} - - def test_icesat(temp_data_dir): # Compare the `ncompare` output when testing ICESat out_path = temp_data_dir / "output_file_icesat-2-atl06.txt" diff --git a/tests/test_getters.py b/tests/test_getters.py new file mode 100644 index 0000000..d5e8dd2 --- /dev/null +++ b/tests/test_getters.py @@ -0,0 +1,21 @@ +import netCDF4 as nc + +from ncompare.getters import get_and_check_variable_scale_factor, get_var_properties + + +def test_var_properties(ds_3dims_3vars_4coords_1group): + with nc.Dataset(ds_3dims_3vars_4coords_1group) as ds: + result = get_var_properties(ds.groups["Group1"], varname="step") + assert result.varname == "step" + assert result.dtype == "float32" + assert result.shape == "(3,)" + assert result.chunking == "contiguous" + assert result.attributes == {"add_offset": 5, "scale_factor": 0.5} + + +def test_get_scale_factor(ds_3dims_3vars_4coords_1group): + with nc.Dataset(ds_3dims_3vars_4coords_1group) as ds: + step_varProps = get_var_properties(ds.groups["Group1"], varname="step") + + result = get_and_check_variable_scale_factor(step_varProps, step_varProps) + assert result == ("0.5", "0.5") From a849328dbe4f2dfa80ef895618fbffefa6c8a132 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Thu, 2 Jan 2025 09:46:30 -0500 Subject: [PATCH 13/19] poetry lock --- poetry.lock | 768 +++++++++++++++++++++++++++------------------------- 1 file changed, 399 insertions(+), 369 deletions(-) diff --git a/poetry.lock b/poetry.lock index 7117133..371e7d8 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry 1.8.5 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.8.4 and should not be changed by hand. [[package]] name = "appnope" @@ -13,37 +13,34 @@ files = [ [[package]] name = "asttokens" -version = "2.4.1" +version = "3.0.0" description = "Annotate AST trees with source code positions" optional = false -python-versions = "*" +python-versions = ">=3.8" files = [ - {file = "asttokens-2.4.1-py2.py3-none-any.whl", hash = "sha256:051ed49c3dcae8913ea7cd08e46a606dba30b79993209636c4875bc1d637bc24"}, - {file = "asttokens-2.4.1.tar.gz", hash = "sha256:b03869718ba9a6eb027e134bfdf69f38a236d681c83c160d510768af11254ba0"}, + {file = "asttokens-3.0.0-py3-none-any.whl", hash = "sha256:e3078351a059199dd5138cb1c706e6430c05eff2ff136af5eb4790f9d28932e2"}, + {file = "asttokens-3.0.0.tar.gz", hash = "sha256:0dcd8baa8d62b0c1d118b399b2ddba3c4aff271d0d7a9e0d4c1681c79035bbc7"}, ] -[package.dependencies] -six = ">=1.12.0" - [package.extras] -astroid = ["astroid (>=1,<2)", "astroid (>=2,<4)"] -test = ["astroid (>=1,<2)", "astroid (>=2,<4)", "pytest"] +astroid = ["astroid (>=2,<4)"] +test = ["astroid (>=2,<4)", "pytest", "pytest-cov", "pytest-xdist"] [[package]] name = "attrs" -version = "24.2.0" +version = "24.3.0" description = "Classes Without Boilerplate" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "attrs-24.2.0-py3-none-any.whl", hash = "sha256:81921eb96de3191c8258c199618104dd27ac608d9366f5e35d011eae1867ede2"}, - {file = "attrs-24.2.0.tar.gz", hash = "sha256:5cfb1b9148b5b086569baec03f20d7b6bf3bcacc9a42bebf87ffaaca362f6346"}, + {file = "attrs-24.3.0-py3-none-any.whl", hash = "sha256:ac96cd038792094f438ad1f6ff80837353805ac950cd2aa0e0625ef19850c308"}, + {file = "attrs-24.3.0.tar.gz", hash = "sha256:8f5c07333d543103541ba7be0e2ce16eeee8130cb0b3f9238ab904ce1e85baff"}, ] [package.extras] benchmark = ["cloudpickle", "hypothesis", "mypy (>=1.11.1)", "pympler", "pytest (>=4.3.0)", "pytest-codspeed", "pytest-mypy-plugins", "pytest-xdist[psutil]"] cov = ["cloudpickle", "coverage[toml] (>=5.3)", "hypothesis", "mypy (>=1.11.1)", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "pytest-xdist[psutil]"] -dev = ["cloudpickle", "hypothesis", "mypy (>=1.11.1)", "pre-commit", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "pytest-xdist[psutil]"] +dev = ["cloudpickle", "hypothesis", "mypy (>=1.11.1)", "pre-commit-uv", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "pytest-xdist[psutil]"] docs = ["cogapp", "furo", "myst-parser", "sphinx", "sphinx-notfound-page", "sphinxcontrib-towncrier", "towncrier (<24.7)"] tests = ["cloudpickle", "hypothesis", "mypy (>=1.11.1)", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "pytest-xdist[psutil]"] tests-mypy = ["mypy (>=1.11.1)", "pytest-mypy-plugins"] @@ -95,6 +92,7 @@ files = [ ] [package.dependencies] +tinycss2 = {version = ">=1.1.0,<1.5", optional = true, markers = "extra == \"css\""} webencodings = "*" [package.extras] @@ -102,13 +100,13 @@ css = ["tinycss2 (>=1.1.0,<1.5)"] [[package]] name = "certifi" -version = "2024.8.30" +version = "2024.12.14" description = "Python package for providing Mozilla's CA Bundle." optional = false python-versions = ">=3.6" files = [ - {file = "certifi-2024.8.30-py3-none-any.whl", hash = "sha256:922820b53db7a7257ffbda3f597266d435245903d80737e34f8a45ff3e3230d8"}, - {file = "certifi-2024.8.30.tar.gz", hash = "sha256:bec941d2aa8195e248a60b31ff9f0558284cf01a52591ceda73ea9afffd69fd9"}, + {file = "certifi-2024.12.14-py3-none-any.whl", hash = "sha256:1275f7a45be9464efc1173084eaa30f866fe2e47d389406136d332ed4967ec56"}, + {file = "certifi-2024.12.14.tar.gz", hash = "sha256:b650d30f370c2b724812bee08008be0c4163b163ddaec3f2546c1caf65f191db"}, ] [[package]] @@ -243,127 +241,114 @@ numpy = [ [[package]] name = "charset-normalizer" -version = "3.4.0" +version = "3.4.1" description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." optional = false -python-versions = ">=3.7.0" +python-versions = ">=3.7" files = [ - {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"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:91b36a978b5ae0ee86c394f5a54d6ef44db1de0815eb43de826d41d21e4af3de"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7461baadb4dc00fd9e0acbe254e3d7d2112e7f92ced2adc96e54ef6501c5f176"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e218488cd232553829be0664c2292d3af2eeeb94b32bea483cf79ac6a694e037"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:80ed5e856eb7f30115aaf94e4a08114ccc8813e6ed1b5efa74f9f82e8509858f"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b010a7a4fd316c3c484d482922d13044979e78d1861f0e0650423144c616a46a"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4532bff1b8421fd0a320463030c7520f56a79c9024a4e88f01c537316019005a"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:d973f03c0cb71c5ed99037b870f2be986c3c05e63622c017ea9816881d2dd247"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:3a3bd0dcd373514dcec91c411ddb9632c0d7d92aed7093b8c3bbb6d69ca74408"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:d9c3cdf5390dcd29aa8056d13e8e99526cda0305acc038b96b30352aff5ff2bb"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:2bdfe3ac2e1bbe5b59a1a63721eb3b95fc9b6817ae4a46debbb4e11f6232428d"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:eab677309cdb30d047996b36d34caeda1dc91149e4fdca0b1a039b3f79d9a807"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-win32.whl", hash = "sha256:c0429126cf75e16c4f0ad00ee0eae4242dc652290f940152ca8c75c3a4b6ee8f"}, + {file = "charset_normalizer-3.4.1-cp310-cp310-win_amd64.whl", hash = "sha256:9f0b8b1c6d84c8034a44893aba5e767bf9c7a211e313a9605d9c617d7083829f"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:8bfa33f4f2672964266e940dd22a195989ba31669bd84629f05fab3ef4e2d125"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:28bf57629c75e810b6ae989f03c0828d64d6b26a5e205535585f96093e405ed1"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f08ff5e948271dc7e18a35641d2f11a4cd8dfd5634f55228b691e62b37125eb3"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:234ac59ea147c59ee4da87a0c0f098e9c8d169f4dc2a159ef720f1a61bbe27cd"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fd4ec41f914fa74ad1b8304bbc634b3de73d2a0889bd32076342a573e0779e00"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:eea6ee1db730b3483adf394ea72f808b6e18cf3cb6454b4d86e04fa8c4327a12"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:c96836c97b1238e9c9e3fe90844c947d5afbf4f4c92762679acfe19927d81d77"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:4d86f7aff21ee58f26dcf5ae81a9addbd914115cdebcbb2217e4f0ed8982e146"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:09b5e6733cbd160dcc09589227187e242a30a49ca5cefa5a7edd3f9d19ed53fd"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:5777ee0881f9499ed0f71cc82cf873d9a0ca8af166dfa0af8ec4e675b7df48e6"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:237bdbe6159cff53b4f24f397d43c6336c6b0b42affbe857970cefbb620911c8"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-win32.whl", hash = "sha256:8417cb1f36cc0bc7eaba8ccb0e04d55f0ee52df06df3ad55259b9a323555fc8b"}, + {file = "charset_normalizer-3.4.1-cp311-cp311-win_amd64.whl", hash = "sha256:d7f50a1f8c450f3925cb367d011448c39239bb3eb4117c36a6d354794de4ce76"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:73d94b58ec7fecbc7366247d3b0b10a21681004153238750bb67bd9012414545"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dad3e487649f498dd991eeb901125411559b22e8d7ab25d3aeb1af367df5efd7"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c30197aa96e8eed02200a83fba2657b4c3acd0f0aa4bdc9f6c1af8e8962e0757"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2369eea1ee4a7610a860d88f268eb39b95cb588acd7235e02fd5a5601773d4fa"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc2722592d8998c870fa4e290c2eec2c1569b87fe58618e67d38b4665dfa680d"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ffc9202a29ab3920fa812879e95a9e78b2465fd10be7fcbd042899695d75e616"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:804a4d582ba6e5b747c625bf1255e6b1507465494a40a2130978bda7b932c90b"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:0f55e69f030f7163dffe9fd0752b32f070566451afe180f99dbeeb81f511ad8d"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:c4c3e6da02df6fa1410a7680bd3f63d4f710232d3139089536310d027950696a"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:5df196eb874dae23dcfb968c83d4f8fdccb333330fe1fc278ac5ceeb101003a9"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:e358e64305fe12299a08e08978f51fc21fac060dcfcddd95453eabe5b93ed0e1"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-win32.whl", hash = "sha256:9b23ca7ef998bc739bf6ffc077c2116917eabcc901f88da1b9856b210ef63f35"}, + {file = "charset_normalizer-3.4.1-cp312-cp312-win_amd64.whl", hash = "sha256:6ff8a4a60c227ad87030d76e99cd1698345d4491638dfa6673027c48b3cd395f"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:aabfa34badd18f1da5ec1bc2715cadc8dca465868a4e73a0173466b688f29dda"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:22e14b5d70560b8dd51ec22863f370d1e595ac3d024cb8ad7d308b4cd95f8313"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8436c508b408b82d87dc5f62496973a1805cd46727c34440b0d29d8a2f50a6c9"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2d074908e1aecee37a7635990b2c6d504cd4766c7bc9fc86d63f9c09af3fa11b"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:955f8851919303c92343d2f66165294848d57e9bba6cf6e3625485a70a038d11"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:44ecbf16649486d4aebafeaa7ec4c9fed8b88101f4dd612dcaf65d5e815f837f"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:0924e81d3d5e70f8126529951dac65c1010cdf117bb75eb02dd12339b57749dd"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:2967f74ad52c3b98de4c3b32e1a44e32975e008a9cd2a8cc8966d6a5218c5cb2"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:c75cb2a3e389853835e84a2d8fb2b81a10645b503eca9bcb98df6b5a43eb8886"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:09b26ae6b1abf0d27570633b2b078a2a20419c99d66fb2823173d73f188ce601"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:fa88b843d6e211393a37219e6a1c1df99d35e8fd90446f1118f4216e307e48cd"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-win32.whl", hash = "sha256:eb8178fe3dba6450a3e024e95ac49ed3400e506fd4e9e5c32d30adda88cbd407"}, + {file = "charset_normalizer-3.4.1-cp313-cp313-win_amd64.whl", hash = "sha256:b1ac5992a838106edb89654e0aebfc24f5848ae2547d22c2c3f66454daa11971"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f30bf9fd9be89ecb2360c7d94a711f00c09b976258846efe40db3d05828e8089"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:97f68b8d6831127e4787ad15e6757232e14e12060bec17091b85eb1486b91d8d"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7974a0b5ecd505609e3b19742b60cee7aa2aa2fb3151bc917e6e2646d7667dcf"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fc54db6c8593ef7d4b2a331b58653356cf04f67c960f584edb7c3d8c97e8f39e"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:311f30128d7d333eebd7896965bfcfbd0065f1716ec92bd5638d7748eb6f936a"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-musllinux_1_2_aarch64.whl", hash = "sha256:7d053096f67cd1241601111b698f5cad775f97ab25d81567d3f59219b5f1adbd"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-musllinux_1_2_i686.whl", hash = "sha256:807f52c1f798eef6cf26beb819eeb8819b1622ddfeef9d0977a8502d4db6d534"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-musllinux_1_2_ppc64le.whl", hash = "sha256:dccbe65bd2f7f7ec22c4ff99ed56faa1e9f785482b9bbd7c717e26fd723a1d1e"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-musllinux_1_2_s390x.whl", hash = "sha256:2fb9bd477fdea8684f78791a6de97a953c51831ee2981f8e4f583ff3b9d9687e"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-musllinux_1_2_x86_64.whl", hash = "sha256:01732659ba9b5b873fc117534143e4feefecf3b2078b0a6a2e925271bb6f4cfa"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-win32.whl", hash = "sha256:7a4f97a081603d2050bfaffdefa5b02a9ec823f8348a572e39032caa8404a487"}, + {file = "charset_normalizer-3.4.1-cp37-cp37m-win_amd64.whl", hash = "sha256:7b1bef6280950ee6c177b326508f86cad7ad4dff12454483b51d8b7d673a2c5d"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:ecddf25bee22fe4fe3737a399d0d177d72bc22be6913acfab364b40bce1ba83c"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8c60ca7339acd497a55b0ea5d506b2a2612afb2826560416f6894e8b5770d4a9"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b7b2d86dd06bfc2ade3312a83a5c364c7ec2e3498f8734282c6c3d4b07b346b8"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:dd78cfcda14a1ef52584dbb008f7ac81c1328c0f58184bf9a84c49c605002da6"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6e27f48bcd0957c6d4cb9d6fa6b61d192d0b13d5ef563e5f2ae35feafc0d179c"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:01ad647cdd609225c5350561d084b42ddf732f4eeefe6e678765636791e78b9a"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:619a609aa74ae43d90ed2e89bdd784765de0a25ca761b93e196d938b8fd1dbbd"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:89149166622f4db9b4b6a449256291dc87a99ee53151c74cbd82a53c8c2f6ccd"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:7709f51f5f7c853f0fb938bcd3bc59cdfdc5203635ffd18bf354f6967ea0f824"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:345b0426edd4e18138d6528aed636de7a9ed169b4aaf9d61a8c19e39d26838ca"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:0907f11d019260cdc3f94fbdb23ff9125f6b5d1039b76003b5b0ac9d6a6c9d5b"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-win32.whl", hash = "sha256:ea0d8d539afa5eb2728aa1932a988a9a7af94f18582ffae4bc10b3fbdad0626e"}, + {file = "charset_normalizer-3.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:329ce159e82018d646c7ac45b01a430369d526569ec08516081727a20e9e4af4"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:b97e690a2118911e39b4042088092771b4ae3fc3aa86518f84b8cf6888dbdb41"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:78baa6d91634dfb69ec52a463534bc0df05dbd546209b79a3880a34487f4b84f"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1a2bc9f351a75ef49d664206d51f8e5ede9da246602dc2d2726837620ea034b2"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:75832c08354f595c760a804588b9357d34ec00ba1c940c15e31e96d902093770"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0af291f4fe114be0280cdd29d533696a77b5b49cfde5467176ecab32353395c4"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0167ddc8ab6508fe81860a57dd472b2ef4060e8d378f0cc555707126830f2537"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:2a75d49014d118e4198bcee5ee0a6f25856b29b12dbf7cd012791f8a6cc5c496"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:363e2f92b0f0174b2f8238240a1a30142e3db7b957a5dd5689b0e75fb717cc78"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:ab36c8eb7e454e34e60eb55ca5d241a5d18b2c6244f6827a30e451c42410b5f7"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:4c0907b1928a36d5a998d72d64d8eaa7244989f7aaaf947500d3a800c83a3fd6"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:04432ad9479fa40ec0f387795ddad4437a2b50417c69fa275e212933519ff294"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-win32.whl", hash = "sha256:3bed14e9c89dcb10e8f3a29f9ccac4955aebe93c71ae803af79265c9ca5644c5"}, + {file = "charset_normalizer-3.4.1-cp39-cp39-win_amd64.whl", hash = "sha256:49402233c892a461407c512a19435d1ce275543138294f7ef013f0b63d5d3765"}, + {file = "charset_normalizer-3.4.1-py3-none-any.whl", hash = "sha256:d98b1668f06378c6dbefec3b92299716b931cd4e6061f3c875a71ced1780ab85"}, + {file = "charset_normalizer-3.4.1.tar.gz", hash = "sha256:44251f18cd68a75b56585dd00dae26183e102cd5e0f9f1466e6df5da2ed64ea3"}, ] [[package]] name = "click" -version = "8.1.7" +version = "8.1.8" description = "Composable command line interface toolkit" optional = false python-versions = ">=3.7" files = [ - {file = "click-8.1.7-py3-none-any.whl", hash = "sha256:ae74fb96c20a0277a1d615f1e4d73c8414f5a98db8b799a7931d1582f3390c28"}, - {file = "click-8.1.7.tar.gz", hash = "sha256:ca9853ad459e787e2192211578cc907e7594e294c7ccc834310722b41b9ca6de"}, + {file = "click-8.1.8-py3-none-any.whl", hash = "sha256:63c132bbbed01578a06712a2d1f497bb62d9c1c0d329b7903a866228027263b2"}, + {file = "click-8.1.8.tar.gz", hash = "sha256:ed53c9d8990d83c2a27deae68e4ee337473f6330c040a31d4225c9574d16096a"}, ] [package.dependencies] @@ -399,73 +384,73 @@ test = ["pytest"] [[package]] name = "coverage" -version = "7.6.5" +version = "7.6.10" description = "Code coverage measurement for Python" optional = false python-versions = ">=3.9" files = [ - {file = "coverage-7.6.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d5fc459f1b62aa328b5c6943b4fa060fa63e7749e41c974929c503dc01d0527b"}, - {file = "coverage-7.6.5-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:197fc6b5e6271c4f822486cabbd91f32e73f784076b69c91179c5a9fec2d1442"}, - {file = "coverage-7.6.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1a7cab0762dfbf0b0cd6eb22f7bceade31bda0f0647f9420cbb45571de4493a3"}, - {file = "coverage-7.6.5-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ee4559597f53455d70b9935e25c21fd05aebbb8d540af04097f7cf6dc7562754"}, - {file = "coverage-7.6.5-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:16e68b894ee1a170da94b7da381527f277ec00c67f6141e79aa1ce8eebbb5561"}, - {file = "coverage-7.6.5-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:fe4ea637711f1f1895895578972e3d0ed5efb6ef970ba0e2e26d9fad1e3c820e"}, - {file = "coverage-7.6.5-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:1d5f036235a747cd30be433ef7ba6dab5ac41d8dc69d54094d5438c34fe8d565"}, - {file = "coverage-7.6.5-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:7a6ab7b88b1a614bc1db015e68048eb29b0c30ffa01be3d7d04da1f320db0f01"}, - {file = "coverage-7.6.5-cp310-cp310-win32.whl", hash = "sha256:ad712a72cd734fb4265041005011bbf61f8d6cba74e12c91f14a9cda63a80a64"}, - {file = "coverage-7.6.5-cp310-cp310-win_amd64.whl", hash = "sha256:61e03bb66c087b74aea6c28d10a49f72eca98b95438a8db1ae6dfcdd060f9039"}, - {file = "coverage-7.6.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:dffec9f67f4eb8bc9c5df720833f1f1ca36b73d86e6f95b422ca5210e264cc26"}, - {file = "coverage-7.6.5-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:2fde790ac0024af19fc5327fd50890dad0c31b653f6d2ed91ab2810c046bfe22"}, - {file = "coverage-7.6.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3250186381ec8e9b71234fb92ef77da87d81cbf20df3364f8f5ebf7180ec030d"}, - {file = "coverage-7.6.5-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2ecfa205ce1fab6d8e94fe011eec04f6035a6069f70c331efd7cd1cd2d33d897"}, - {file = "coverage-7.6.5-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:15af7bfbc37de33e7df3f740cc735057606c63bbe44aee8b07339a3e7bb8ecf6"}, - {file = "coverage-7.6.5-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:caf4d6af23af0e0df4e40e9985f6063d7f5434f225ee4d4ed7001f1428302403"}, - {file = "coverage-7.6.5-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:5dcf2da597fe616a41c59e29fd8d390ac2149aeed421172eef14470c7e9dcd06"}, - {file = "coverage-7.6.5-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:ebc76107d896a53116e5ef21998f321b630b574a65b78b01176ca64e8978b43e"}, - {file = "coverage-7.6.5-cp311-cp311-win32.whl", hash = "sha256:0e9e4cd48dca252d99bb97b14f13b5940813937cc7ec568418c1a195dec9cbcc"}, - {file = "coverage-7.6.5-cp311-cp311-win_amd64.whl", hash = "sha256:a6eb14739a20c5a46073c8ad066ada17d91d14599ed98d724614db46fbae867b"}, - {file = "coverage-7.6.5-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:9ae01c434cb0d445008257bb42dcd38112190e5bfc3a4480fde49572b16bc2ae"}, - {file = "coverage-7.6.5-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:c72ef3be899f389c9f0934a9d06a28fa097ade096760102c732583c04cc31d75"}, - {file = "coverage-7.6.5-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d2fc574b4fb082a0141d4df00079c4877d46cb98e8ec979cbd9a92426f5abd8a"}, - {file = "coverage-7.6.5-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1bc0eba158ad9d1883efb4f1bf08f88a999e091daf30454fd5f136322e700c72"}, - {file = "coverage-7.6.5-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a360b282c0acbf3541cc67e8d8a2a65589ea6cfa10c7e8a48e318bf28ca90f94"}, - {file = "coverage-7.6.5-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:b22f96d3f2425942a649d786f57ae431425c9a970afae784cd865c1ffee34bad"}, - {file = "coverage-7.6.5-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:70eca9c6bf742feaf3ee453c1aaa932c2ab88ca420f411d90aa43ae831127b22"}, - {file = "coverage-7.6.5-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:2c4bafec5da3498d498a4ca3136f5a01fded487c6a54f18aea0bcd673feedf1b"}, - {file = "coverage-7.6.5-cp312-cp312-win32.whl", hash = "sha256:edecf498cabb335e8a683eb672558355bb9536d4397c54f1e135d9b8910512a3"}, - {file = "coverage-7.6.5-cp312-cp312-win_amd64.whl", hash = "sha256:e7c40ae56761d3c08f916019b2f8579a147f93be8e12f0f2bf4edc4ea9e1c0ab"}, - {file = "coverage-7.6.5-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:49ea4a739dc14856d7c5f935da90db123b77a850cfddcfacb490a28de8f87257"}, - {file = "coverage-7.6.5-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:e0c51339a28aa43d0f2b1211e57ceeeeed5e09f4deb6fc543d939de68069e81e"}, - {file = "coverage-7.6.5-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:040c3d5cf4db24e7cb890bf4b547a25bd3a3516c58c9f2a22f822199ee2ad8ed"}, - {file = "coverage-7.6.5-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f0b7e67f9d3b156ab93fce71485fadd043ab04b45d5d88623c6d94f7d16ced5b"}, - {file = "coverage-7.6.5-cp313-cp313-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e078bfb114025c55fdbaa802f4c13e20e6ce4e10a96918d7234656b41f69e649"}, - {file = "coverage-7.6.5-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:559cdb21aca30810e648ac08270535c1d2e17226ebbdf90860a060d3680cb05f"}, - {file = "coverage-7.6.5-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:23e2dd956277061f24d9eda7539113a9c35a9409a9935647a34ced79b8aacb75"}, - {file = "coverage-7.6.5-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:3e7c4ccb41dc9830b2ca8592e401045a81740f627c7c0348bdc3b7373ce52f8e"}, - {file = "coverage-7.6.5-cp313-cp313-win32.whl", hash = "sha256:9d3565bb7deaa12d634426f113e6b106028c535667ba7756af65f00464981ba5"}, - {file = "coverage-7.6.5-cp313-cp313-win_amd64.whl", hash = "sha256:5039410420d9ddcd5b8566d3afbb28b89d70c4481dbb283ea543263cbefa2b67"}, - {file = "coverage-7.6.5-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:77b640aa78d4d9f620fb2e1b2a41b0d196120c188d0a7f678761d668d6251fcc"}, - {file = "coverage-7.6.5-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:bb3799f6279df37e369027128926de4c159e6399000316ebd7a69e55b84dc97f"}, - {file = "coverage-7.6.5-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:55aba7ab64e8af37a18064f23f399dff10041fa3aaf201528f12004968638b9f"}, - {file = "coverage-7.6.5-cp313-cp313t-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6065a988d724dd3328cb21e97378bef0549b2f8b7ac0a3376785d9f7f05dc736"}, - {file = "coverage-7.6.5-cp313-cp313t-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5f092d222e4286cdd1ab9707da36944c11ba6294d8c9b18534057f03e6866367"}, - {file = "coverage-7.6.5-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:1dc99aece5f899955eece053a798e279f7fe7059dd5e2a95af82878cfe4a44e1"}, - {file = "coverage-7.6.5-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:1b14515f83ffa7a6787e725d804c6b11dd317a6bd0373d8519a61e4a587fe534"}, - {file = "coverage-7.6.5-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:9fa6d90130165346935541f3762933dae07e237ff7d6d780fae556039f08a470"}, - {file = "coverage-7.6.5-cp313-cp313t-win32.whl", hash = "sha256:1be9ec4c49becb35955b9d69c27e6385aedd40d233f1cf065e8430c59924b30e"}, - {file = "coverage-7.6.5-cp313-cp313t-win_amd64.whl", hash = "sha256:7ff4fd7679df56e36fc838ef227e95e3aa1b0ca0548daede7f8ae6e54479c115"}, - {file = "coverage-7.6.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:23abf0846290aa57d629c4f4181d0d56cbaa45d3999e60cb0df1d2bab7bc6bfe"}, - {file = "coverage-7.6.5-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:b4903685e8059e170182ac4681ee72d2dfbb92692225023c1e325a9d85c1be31"}, - {file = "coverage-7.6.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4ad9621fd9773b1461f8942da4130fbb16ee0a877eb58bc57532ea41cce20d3e"}, - {file = "coverage-7.6.5-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7324358a77f37ffd8ba94d3c8326eb316c972ec72264f36fc3be04cff8542465"}, - {file = "coverage-7.6.5-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bf182001229411cd6a90d180973b345bd6fe255dbbac362100e6a625dfb107f5"}, - {file = "coverage-7.6.5-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:4601dacd88556c94c9fb5063b9354b1fe971af9a5b25b2575faefd12bf8170a5"}, - {file = "coverage-7.6.5-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:e5aa3d62285ef1b16f655e1ae298c6fa919209637d317934e382e9b99c28c118"}, - {file = "coverage-7.6.5-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:8cb5601620c3d98d2c98847272acc2406333d43c9d7d49386d879bd451677429"}, - {file = "coverage-7.6.5-cp39-cp39-win32.whl", hash = "sha256:c32428f6285344caedd945236f31c46645bb10faae8702d1409bb49df218e55a"}, - {file = "coverage-7.6.5-cp39-cp39-win_amd64.whl", hash = "sha256:809e868eee27d056bc72590c69940c119775d218681b1a8ef9ba0ef8d7693e53"}, - {file = "coverage-7.6.5-pp39.pp310-none-any.whl", hash = "sha256:49145276f39f940b18a539e1e4a378e06c64a127922450ffd2fb82b9fe1ad3d9"}, - {file = "coverage-7.6.5.tar.gz", hash = "sha256:6069188329fbe0a63876719099076261ce7a1adeea95bf236cff4353a8451b0d"}, + {file = "coverage-7.6.10-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:5c912978f7fbf47ef99cec50c4401340436d200d41d714c7a4766f377c5b7b78"}, + {file = "coverage-7.6.10-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a01ec4af7dfeb96ff0078ad9a48810bb0cc8abcb0115180c6013a6b26237626c"}, + {file = "coverage-7.6.10-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a3b204c11e2b2d883946fe1d97f89403aa1811df28ce0447439178cc7463448a"}, + {file = "coverage-7.6.10-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:32ee6d8491fcfc82652a37109f69dee9a830e9379166cb73c16d8dc5c2915165"}, + {file = "coverage-7.6.10-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:675cefc4c06e3b4c876b85bfb7c59c5e2218167bbd4da5075cbe3b5790a28988"}, + {file = "coverage-7.6.10-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:f4f620668dbc6f5e909a0946a877310fb3d57aea8198bde792aae369ee1c23b5"}, + {file = "coverage-7.6.10-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:4eea95ef275de7abaef630c9b2c002ffbc01918b726a39f5a4353916ec72d2f3"}, + {file = "coverage-7.6.10-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:e2f0280519e42b0a17550072861e0bc8a80a0870de260f9796157d3fca2733c5"}, + {file = "coverage-7.6.10-cp310-cp310-win32.whl", hash = "sha256:bc67deb76bc3717f22e765ab3e07ee9c7a5e26b9019ca19a3b063d9f4b874244"}, + {file = "coverage-7.6.10-cp310-cp310-win_amd64.whl", hash = "sha256:0f460286cb94036455e703c66988851d970fdfd8acc2a1122ab7f4f904e4029e"}, + {file = "coverage-7.6.10-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:ea3c8f04b3e4af80e17bab607c386a830ffc2fb88a5484e1df756478cf70d1d3"}, + {file = "coverage-7.6.10-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:507a20fc863cae1d5720797761b42d2d87a04b3e5aeb682ef3b7332e90598f43"}, + {file = "coverage-7.6.10-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d37a84878285b903c0fe21ac8794c6dab58150e9359f1aaebbeddd6412d53132"}, + {file = "coverage-7.6.10-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a534738b47b0de1995f85f582d983d94031dffb48ab86c95bdf88dc62212142f"}, + {file = "coverage-7.6.10-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0d7a2bf79378d8fb8afaa994f91bfd8215134f8631d27eba3e0e2c13546ce994"}, + {file = "coverage-7.6.10-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:6713ba4b4ebc330f3def51df1d5d38fad60b66720948112f114968feb52d3f99"}, + {file = "coverage-7.6.10-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:ab32947f481f7e8c763fa2c92fd9f44eeb143e7610c4ca9ecd6a36adab4081bd"}, + {file = "coverage-7.6.10-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:7bbd8c8f1b115b892e34ba66a097b915d3871db7ce0e6b9901f462ff3a975377"}, + {file = "coverage-7.6.10-cp311-cp311-win32.whl", hash = "sha256:299e91b274c5c9cdb64cbdf1b3e4a8fe538a7a86acdd08fae52301b28ba297f8"}, + {file = "coverage-7.6.10-cp311-cp311-win_amd64.whl", hash = "sha256:489a01f94aa581dbd961f306e37d75d4ba16104bbfa2b0edb21d29b73be83609"}, + {file = "coverage-7.6.10-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:27c6e64726b307782fa5cbe531e7647aee385a29b2107cd87ba7c0105a5d3853"}, + {file = "coverage-7.6.10-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:c56e097019e72c373bae32d946ecf9858fda841e48d82df7e81c63ac25554078"}, + {file = "coverage-7.6.10-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c7827a5bc7bdb197b9e066cdf650b2887597ad124dd99777332776f7b7c7d0d0"}, + {file = "coverage-7.6.10-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:204a8238afe787323a8b47d8be4df89772d5c1e4651b9ffa808552bdf20e1d50"}, + {file = "coverage-7.6.10-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e67926f51821b8e9deb6426ff3164870976fe414d033ad90ea75e7ed0c2e5022"}, + {file = "coverage-7.6.10-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:e78b270eadb5702938c3dbe9367f878249b5ef9a2fcc5360ac7bff694310d17b"}, + {file = "coverage-7.6.10-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:714f942b9c15c3a7a5fe6876ce30af831c2ad4ce902410b7466b662358c852c0"}, + {file = "coverage-7.6.10-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:abb02e2f5a3187b2ac4cd46b8ced85a0858230b577ccb2c62c81482ca7d18852"}, + {file = "coverage-7.6.10-cp312-cp312-win32.whl", hash = "sha256:55b201b97286cf61f5e76063f9e2a1d8d2972fc2fcfd2c1272530172fd28c359"}, + {file = "coverage-7.6.10-cp312-cp312-win_amd64.whl", hash = "sha256:e4ae5ac5e0d1e4edfc9b4b57b4cbecd5bc266a6915c500f358817a8496739247"}, + {file = "coverage-7.6.10-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:05fca8ba6a87aabdd2d30d0b6c838b50510b56cdcfc604d40760dae7153b73d9"}, + {file = "coverage-7.6.10-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:9e80eba8801c386f72e0712a0453431259c45c3249f0009aff537a517b52942b"}, + {file = "coverage-7.6.10-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a372c89c939d57abe09e08c0578c1d212e7a678135d53aa16eec4430adc5e690"}, + {file = "coverage-7.6.10-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ec22b5e7fe7a0fa8509181c4aac1db48f3dd4d3a566131b313d1efc102892c18"}, + {file = "coverage-7.6.10-cp313-cp313-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:26bcf5c4df41cad1b19c84af71c22cbc9ea9a547fc973f1f2cc9a290002c8b3c"}, + {file = "coverage-7.6.10-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:4e4630c26b6084c9b3cb53b15bd488f30ceb50b73c35c5ad7871b869cb7365fd"}, + {file = "coverage-7.6.10-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:2396e8116db77789f819d2bc8a7e200232b7a282c66e0ae2d2cd84581a89757e"}, + {file = "coverage-7.6.10-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:79109c70cc0882e4d2d002fe69a24aa504dec0cc17169b3c7f41a1d341a73694"}, + {file = "coverage-7.6.10-cp313-cp313-win32.whl", hash = "sha256:9e1747bab246d6ff2c4f28b4d186b205adced9f7bd9dc362051cc37c4a0c7bd6"}, + {file = "coverage-7.6.10-cp313-cp313-win_amd64.whl", hash = "sha256:254f1a3b1eef5f7ed23ef265eaa89c65c8c5b6b257327c149db1ca9d4a35f25e"}, + {file = "coverage-7.6.10-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:2ccf240eb719789cedbb9fd1338055de2761088202a9a0b73032857e53f612fe"}, + {file = "coverage-7.6.10-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:0c807ca74d5a5e64427c8805de15b9ca140bba13572d6d74e262f46f50b13273"}, + {file = "coverage-7.6.10-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2bcfa46d7709b5a7ffe089075799b902020b62e7ee56ebaed2f4bdac04c508d8"}, + {file = "coverage-7.6.10-cp313-cp313t-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4e0de1e902669dccbf80b0415fb6b43d27edca2fbd48c74da378923b05316098"}, + {file = "coverage-7.6.10-cp313-cp313t-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3f7b444c42bbc533aaae6b5a2166fd1a797cdb5eb58ee51a92bee1eb94a1e1cb"}, + {file = "coverage-7.6.10-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:b330368cb99ef72fcd2dc3ed260adf67b31499584dc8a20225e85bfe6f6cfed0"}, + {file = "coverage-7.6.10-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:9a7cfb50515f87f7ed30bc882f68812fd98bc2852957df69f3003d22a2aa0abf"}, + {file = "coverage-7.6.10-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:6f93531882a5f68c28090f901b1d135de61b56331bba82028489bc51bdd818d2"}, + {file = "coverage-7.6.10-cp313-cp313t-win32.whl", hash = "sha256:89d76815a26197c858f53c7f6a656686ec392b25991f9e409bcef020cd532312"}, + {file = "coverage-7.6.10-cp313-cp313t-win_amd64.whl", hash = "sha256:54a5f0f43950a36312155dae55c505a76cd7f2b12d26abeebbe7a0b36dbc868d"}, + {file = "coverage-7.6.10-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:656c82b8a0ead8bba147de9a89bda95064874c91a3ed43a00e687f23cc19d53a"}, + {file = "coverage-7.6.10-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ccc2b70a7ed475c68ceb548bf69cec1e27305c1c2606a5eb7c3afff56a1b3b27"}, + {file = "coverage-7.6.10-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a5e37dc41d57ceba70956fa2fc5b63c26dba863c946ace9705f8eca99daecdc4"}, + {file = "coverage-7.6.10-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0aa9692b4fdd83a4647eeb7db46410ea1322b5ed94cd1715ef09d1d5922ba87f"}, + {file = "coverage-7.6.10-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:aa744da1820678b475e4ba3dfd994c321c5b13381d1041fe9c608620e6676e25"}, + {file = "coverage-7.6.10-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:c0b1818063dc9e9d838c09e3a473c1422f517889436dd980f5d721899e66f315"}, + {file = "coverage-7.6.10-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:59af35558ba08b758aec4d56182b222976330ef8d2feacbb93964f576a7e7a90"}, + {file = "coverage-7.6.10-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:7ed2f37cfce1ce101e6dffdfd1c99e729dd2ffc291d02d3e2d0af8b53d13840d"}, + {file = "coverage-7.6.10-cp39-cp39-win32.whl", hash = "sha256:4bcc276261505d82f0ad426870c3b12cb177752834a633e737ec5ee79bbdff18"}, + {file = "coverage-7.6.10-cp39-cp39-win_amd64.whl", hash = "sha256:457574f4599d2b00f7f637a0700a6422243b3565509457b2dbd3f50703e11f59"}, + {file = "coverage-7.6.10-pp39.pp310-none-any.whl", hash = "sha256:fd34e7b3405f0cc7ab03d54a334c17a9e802897580d964bd8c2001f4b9fd488f"}, + {file = "coverage-7.6.10.tar.gz", hash = "sha256:7fb105327c8f8f0682e29843e2ff96af9dcbe5bab8eeb4b398c6a33a16d80a23"}, ] [package.dependencies] @@ -476,37 +461,37 @@ toml = ["tomli"] [[package]] name = "debugpy" -version = "1.8.8" +version = "1.8.11" description = "An implementation of the Debug Adapter Protocol for Python" optional = false python-versions = ">=3.8" files = [ - {file = "debugpy-1.8.8-cp310-cp310-macosx_14_0_x86_64.whl", hash = "sha256:e59b1607c51b71545cb3496876544f7186a7a27c00b436a62f285603cc68d1c6"}, - {file = "debugpy-1.8.8-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a6531d952b565b7cb2fbd1ef5df3d333cf160b44f37547a4e7cf73666aca5d8d"}, - {file = "debugpy-1.8.8-cp310-cp310-win32.whl", hash = "sha256:b01f4a5e5c5fb1d34f4ccba99a20ed01eabc45a4684f4948b5db17a319dfb23f"}, - {file = "debugpy-1.8.8-cp310-cp310-win_amd64.whl", hash = "sha256:535f4fb1c024ddca5913bb0eb17880c8f24ba28aa2c225059db145ee557035e9"}, - {file = "debugpy-1.8.8-cp311-cp311-macosx_14_0_universal2.whl", hash = "sha256:c399023146e40ae373753a58d1be0a98bf6397fadc737b97ad612886b53df318"}, - {file = "debugpy-1.8.8-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:09cc7b162586ea2171eea055985da2702b0723f6f907a423c9b2da5996ad67ba"}, - {file = "debugpy-1.8.8-cp311-cp311-win32.whl", hash = "sha256:eea8821d998ebeb02f0625dd0d76839ddde8cbf8152ebbe289dd7acf2cdc6b98"}, - {file = "debugpy-1.8.8-cp311-cp311-win_amd64.whl", hash = "sha256:d4483836da2a533f4b1454dffc9f668096ac0433de855f0c22cdce8c9f7e10c4"}, - {file = "debugpy-1.8.8-cp312-cp312-macosx_14_0_universal2.whl", hash = "sha256:0cc94186340be87b9ac5a707184ec8f36547fb66636d1029ff4f1cc020e53996"}, - {file = "debugpy-1.8.8-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:64674e95916e53c2e9540a056e5f489e0ad4872645399d778f7c598eacb7b7f9"}, - {file = "debugpy-1.8.8-cp312-cp312-win32.whl", hash = "sha256:5c6e885dbf12015aed73770f29dec7023cb310d0dc2ba8bfbeb5c8e43f80edc9"}, - {file = "debugpy-1.8.8-cp312-cp312-win_amd64.whl", hash = "sha256:19ffbd84e757a6ca0113574d1bf5a2298b3947320a3e9d7d8dc3377f02d9f864"}, - {file = "debugpy-1.8.8-cp313-cp313-macosx_14_0_universal2.whl", hash = "sha256:705cd123a773d184860ed8dae99becd879dfec361098edbefb5fc0d3683eb804"}, - {file = "debugpy-1.8.8-cp313-cp313-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:890fd16803f50aa9cb1a9b9b25b5ec321656dd6b78157c74283de241993d086f"}, - {file = "debugpy-1.8.8-cp313-cp313-win32.whl", hash = "sha256:90244598214bbe704aa47556ec591d2f9869ff9e042e301a2859c57106649add"}, - {file = "debugpy-1.8.8-cp313-cp313-win_amd64.whl", hash = "sha256:4b93e4832fd4a759a0c465c967214ed0c8a6e8914bced63a28ddb0dd8c5f078b"}, - {file = "debugpy-1.8.8-cp38-cp38-macosx_14_0_x86_64.whl", hash = "sha256:143ef07940aeb8e7316de48f5ed9447644da5203726fca378f3a6952a50a9eae"}, - {file = "debugpy-1.8.8-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f95651bdcbfd3b27a408869a53fbefcc2bcae13b694daee5f1365b1b83a00113"}, - {file = "debugpy-1.8.8-cp38-cp38-win32.whl", hash = "sha256:26b461123a030e82602a750fb24d7801776aa81cd78404e54ab60e8b5fecdad5"}, - {file = "debugpy-1.8.8-cp38-cp38-win_amd64.whl", hash = "sha256:f3cbf1833e644a3100eadb6120f25be8a532035e8245584c4f7532937edc652a"}, - {file = "debugpy-1.8.8-cp39-cp39-macosx_14_0_x86_64.whl", hash = "sha256:53709d4ec586b525724819dc6af1a7703502f7e06f34ded7157f7b1f963bb854"}, - {file = "debugpy-1.8.8-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3a9c013077a3a0000e83d97cf9cc9328d2b0bbb31f56b0e99ea3662d29d7a6a2"}, - {file = "debugpy-1.8.8-cp39-cp39-win32.whl", hash = "sha256:ffe94dd5e9a6739a75f0b85316dc185560db3e97afa6b215628d1b6a17561cb2"}, - {file = "debugpy-1.8.8-cp39-cp39-win_amd64.whl", hash = "sha256:5c0e5a38c7f9b481bf31277d2f74d2109292179081f11108e668195ef926c0f9"}, - {file = "debugpy-1.8.8-py2.py3-none-any.whl", hash = "sha256:ec684553aba5b4066d4de510859922419febc710df7bba04fe9e7ef3de15d34f"}, - {file = "debugpy-1.8.8.zip", hash = "sha256:e6355385db85cbd666be703a96ab7351bc9e6c61d694893206f8001e22aee091"}, + {file = "debugpy-1.8.11-cp310-cp310-macosx_14_0_x86_64.whl", hash = "sha256:2b26fefc4e31ff85593d68b9022e35e8925714a10ab4858fb1b577a8a48cb8cd"}, + {file = "debugpy-1.8.11-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:61bc8b3b265e6949855300e84dc93d02d7a3a637f2aec6d382afd4ceb9120c9f"}, + {file = "debugpy-1.8.11-cp310-cp310-win32.whl", hash = "sha256:c928bbf47f65288574b78518449edaa46c82572d340e2750889bbf8cd92f3737"}, + {file = "debugpy-1.8.11-cp310-cp310-win_amd64.whl", hash = "sha256:8da1db4ca4f22583e834dcabdc7832e56fe16275253ee53ba66627b86e304da1"}, + {file = "debugpy-1.8.11-cp311-cp311-macosx_14_0_universal2.whl", hash = "sha256:85de8474ad53ad546ff1c7c7c89230db215b9b8a02754d41cb5a76f70d0be296"}, + {file = "debugpy-1.8.11-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8ffc382e4afa4aee367bf413f55ed17bd91b191dcaf979890af239dda435f2a1"}, + {file = "debugpy-1.8.11-cp311-cp311-win32.whl", hash = "sha256:40499a9979c55f72f4eb2fc38695419546b62594f8af194b879d2a18439c97a9"}, + {file = "debugpy-1.8.11-cp311-cp311-win_amd64.whl", hash = "sha256:987bce16e86efa86f747d5151c54e91b3c1e36acc03ce1ddb50f9d09d16ded0e"}, + {file = "debugpy-1.8.11-cp312-cp312-macosx_14_0_universal2.whl", hash = "sha256:84e511a7545d11683d32cdb8f809ef63fc17ea2a00455cc62d0a4dbb4ed1c308"}, + {file = "debugpy-1.8.11-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ce291a5aca4985d82875d6779f61375e959208cdf09fcec40001e65fb0a54768"}, + {file = "debugpy-1.8.11-cp312-cp312-win32.whl", hash = "sha256:28e45b3f827d3bf2592f3cf7ae63282e859f3259db44ed2b129093ca0ac7940b"}, + {file = "debugpy-1.8.11-cp312-cp312-win_amd64.whl", hash = "sha256:44b1b8e6253bceada11f714acf4309ffb98bfa9ac55e4fce14f9e5d4484287a1"}, + {file = "debugpy-1.8.11-cp313-cp313-macosx_14_0_universal2.whl", hash = "sha256:8988f7163e4381b0da7696f37eec7aca19deb02e500245df68a7159739bbd0d3"}, + {file = "debugpy-1.8.11-cp313-cp313-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c1f6a173d1140e557347419767d2b14ac1c9cd847e0b4c5444c7f3144697e4e"}, + {file = "debugpy-1.8.11-cp313-cp313-win32.whl", hash = "sha256:bb3b15e25891f38da3ca0740271e63ab9db61f41d4d8541745cfc1824252cb28"}, + {file = "debugpy-1.8.11-cp313-cp313-win_amd64.whl", hash = "sha256:d8768edcbeb34da9e11bcb8b5c2e0958d25218df7a6e56adf415ef262cd7b6d1"}, + {file = "debugpy-1.8.11-cp38-cp38-macosx_14_0_x86_64.whl", hash = "sha256:ad7efe588c8f5cf940f40c3de0cd683cc5b76819446abaa50dc0829a30c094db"}, + {file = "debugpy-1.8.11-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:189058d03a40103a57144752652b3ab08ff02b7595d0ce1f651b9acc3a3a35a0"}, + {file = "debugpy-1.8.11-cp38-cp38-win32.whl", hash = "sha256:32db46ba45849daed7ccf3f2e26f7a386867b077f39b2a974bb5c4c2c3b0a280"}, + {file = "debugpy-1.8.11-cp38-cp38-win_amd64.whl", hash = "sha256:116bf8342062246ca749013df4f6ea106f23bc159305843491f64672a55af2e5"}, + {file = "debugpy-1.8.11-cp39-cp39-macosx_14_0_x86_64.whl", hash = "sha256:654130ca6ad5de73d978057eaf9e582244ff72d4574b3e106fb8d3d2a0d32458"}, + {file = "debugpy-1.8.11-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:23dc34c5e03b0212fa3c49a874df2b8b1b8fda95160bd79c01eb3ab51ea8d851"}, + {file = "debugpy-1.8.11-cp39-cp39-win32.whl", hash = "sha256:52d8a3166c9f2815bfae05f386114b0b2d274456980d41f320299a8d9a5615a7"}, + {file = "debugpy-1.8.11-cp39-cp39-win_amd64.whl", hash = "sha256:52c3cf9ecda273a19cc092961ee34eb9ba8687d67ba34cc7b79a521c1c64c4c0"}, + {file = "debugpy-1.8.11-py2.py3-none-any.whl", hash = "sha256:0e22f846f4211383e6a416d04b4c13ed174d24cc5d43f5fd52e7821d0ebc8920"}, + {file = "debugpy-1.8.11.tar.gz", hash = "sha256:6ad2688b69235c43b020e04fecccdf6a96c8943ca9c2fb340b8adc103c655e57"}, ] [[package]] @@ -572,13 +557,13 @@ tests = ["asttokens (>=2.1.0)", "coverage", "coverage-enable-subprocess", "ipyth [[package]] name = "fastjsonschema" -version = "2.20.0" +version = "2.21.1" description = "Fastest Python implementation of JSON schema" optional = false python-versions = "*" files = [ - {file = "fastjsonschema-2.20.0-py3-none-any.whl", hash = "sha256:5875f0b0fa7a0043a91e93a9b8f793bcbbba9691e7fd83dca95c28ba26d21f0a"}, - {file = "fastjsonschema-2.20.0.tar.gz", hash = "sha256:3d48fc5300ee96f5d116f10fe6f28d938e6008f59a6a025c2649475b87f76a23"}, + {file = "fastjsonschema-2.21.1-py3-none-any.whl", hash = "sha256:c9e5b7e908310918cf494a434eeb31384dd84a98b57a30bcb1f535015b554667"}, + {file = "fastjsonschema-2.21.1.tar.gz", hash = "sha256:794d4f0a58f848961ba16af7b9c85a3e88cd360df008c59aac6fc5ae9323b5d4"}, ] [package.extras] @@ -796,13 +781,13 @@ testing = ["Django", "attrs", "colorama", "docopt", "pytest (<9.0.0)"] [[package]] name = "jinja2" -version = "3.1.4" +version = "3.1.5" description = "A very fast and expressive template engine." optional = false python-versions = ">=3.7" files = [ - {file = "jinja2-3.1.4-py3-none-any.whl", hash = "sha256:bc5dd2abb727a5319567b7a813e6a2e7318c39f4f487cfe6c89c6f9c7d25197d"}, - {file = "jinja2-3.1.4.tar.gz", hash = "sha256:4a3aee7acbbe7303aede8e9648d13b8bf88a429282aa6122a993f0ac800cb369"}, + {file = "jinja2-3.1.5-py3-none-any.whl", hash = "sha256:aba0f4dc9ed8013c424088f68a5c226f7d6097ed89b246d7749c2ec4175c6adb"}, + {file = "jinja2-3.1.5.tar.gz", hash = "sha256:8fefff8dc3034e27bb80d67c671eb8a9bc424c0ef4c0826edbff304cceff43bb"}, ] [package.dependencies] @@ -902,13 +887,13 @@ files = [ [[package]] name = "jupytext" -version = "1.16.4" +version = "1.16.6" description = "Jupyter notebooks as Markdown documents, Julia, Python or R scripts" optional = false python-versions = ">=3.8" files = [ - {file = "jupytext-1.16.4-py3-none-any.whl", hash = "sha256:76989d2690e65667ea6fb411d8056abe7cd0437c07bd774660b83d62acf9490a"}, - {file = "jupytext-1.16.4.tar.gz", hash = "sha256:28e33f46f2ce7a41fb9d677a4a2c95327285579b64ca104437c4b9eb1e4174e9"}, + {file = "jupytext-1.16.6-py3-none-any.whl", hash = "sha256:900132031f73fee15a1c9ebd862e05eb5f51e1ad6ab3a2c6fdd97ce2f9c913b4"}, + {file = "jupytext-1.16.6.tar.gz", hash = "sha256:dbd03f9263c34b737003f388fc069e9030834fb7136879c4c32c32473557baa0"}, ] [package.dependencies] @@ -920,11 +905,11 @@ pyyaml = "*" tomli = {version = "*", markers = "python_version < \"3.11\""} [package.extras] -dev = ["autopep8", "black", "flake8", "gitpython", "ipykernel", "isort", "jupyter-fs (>=1.0)", "jupyter-server (!=2.11)", "nbconvert", "pre-commit", "pytest", "pytest-cov (>=2.6.1)", "pytest-randomly", "pytest-xdist", "sphinx-gallery (<0.8)"] +dev = ["autopep8", "black", "flake8", "gitpython", "ipykernel", "isort", "jupyter-fs (>=1.0)", "jupyter-server (!=2.11)", "nbconvert", "pre-commit", "pytest", "pytest-cov (>=2.6.1)", "pytest-randomly", "pytest-xdist", "sphinx (<8)", "sphinx-gallery (<0.8)"] docs = ["myst-parser", "sphinx", "sphinx-copybutton", "sphinx-rtd-theme"] test = ["pytest", "pytest-randomly", "pytest-xdist"] test-cov = ["ipykernel", "jupyter-server (!=2.11)", "nbconvert", "pytest", "pytest-cov (>=2.6.1)", "pytest-randomly", "pytest-xdist"] -test-external = ["autopep8", "black", "flake8", "gitpython", "ipykernel", "isort", "jupyter-fs (>=1.0)", "jupyter-server (!=2.11)", "nbconvert", "pre-commit", "pytest", "pytest-randomly", "pytest-xdist", "sphinx-gallery (<0.8)"] +test-external = ["autopep8", "black", "flake8", "gitpython", "ipykernel", "isort", "jupyter-fs (>=1.0)", "jupyter-server (!=2.11)", "nbconvert", "pre-commit", "pytest", "pytest-randomly", "pytest-xdist", "sphinx (<8)", "sphinx-gallery (<0.8)"] test-functional = ["pytest", "pytest-randomly", "pytest-xdist"] test-integration = ["ipykernel", "jupyter-server (!=2.11)", "nbconvert", "pytest", "pytest-randomly", "pytest-xdist"] test-ui = ["calysto-bash"] @@ -1129,15 +1114,18 @@ files = [ [[package]] name = "mistune" -version = "3.0.2" +version = "3.1.0" description = "A sane and fast Markdown parser with useful plugins and renderers" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "mistune-3.0.2-py3-none-any.whl", hash = "sha256:71481854c30fdbc938963d3605b72501f5c10a9320ecd412c121c163a1c7d205"}, - {file = "mistune-3.0.2.tar.gz", hash = "sha256:fc7f93ded930c92394ef2cb6f04a8aabab4117a91449e72dcc8dfa646a508be8"}, + {file = "mistune-3.1.0-py3-none-any.whl", hash = "sha256:b05198cf6d671b3deba6c87ec6cf0d4eb7b72c524636eddb6dbf13823b52cee1"}, + {file = "mistune-3.1.0.tar.gz", hash = "sha256:dbcac2f78292b9dc066cd03b7a3a26b62d85f8159f2ea5fd28e55df79908d667"}, ] +[package.dependencies] +typing-extensions = {version = "*", markers = "python_version < \"3.11\""} + [[package]] name = "mkdocs" version = "1.6.1" @@ -1361,13 +1349,13 @@ files = [ [[package]] name = "nbclient" -version = "0.10.0" +version = "0.10.2" description = "A client library for executing notebooks. Formerly nbconvert's ExecutePreprocessor." optional = false -python-versions = ">=3.8.0" +python-versions = ">=3.9.0" files = [ - {file = "nbclient-0.10.0-py3-none-any.whl", hash = "sha256:f13e3529332a1f1f81d82a53210322476a168bb7090a0289c795fe9cc11c9d3f"}, - {file = "nbclient-0.10.0.tar.gz", hash = "sha256:4b3f1b7dba531e498449c4db4f53da339c91d449dc11e9af3a43b4eb5c5abb09"}, + {file = "nbclient-0.10.2-py3-none-any.whl", hash = "sha256:4ffee11e788b4a27fabeb7955547e4318a5298f34342a4bfd01f2e1faaeadc3d"}, + {file = "nbclient-0.10.2.tar.gz", hash = "sha256:90b7fc6b810630db87a6d0c2250b1f0ab4cf4d3c27a299b0cde78a4ed3fd9193"}, ] [package.dependencies] @@ -1378,23 +1366,23 @@ traitlets = ">=5.4" [package.extras] dev = ["pre-commit"] -docs = ["autodoc-traits", "mock", "moto", "myst-parser", "nbclient[test]", "sphinx (>=1.7)", "sphinx-book-theme", "sphinxcontrib-spelling"] -test = ["flaky", "ipykernel (>=6.19.3)", "ipython", "ipywidgets", "nbconvert (>=7.0.0)", "pytest (>=7.0,<8)", "pytest-asyncio", "pytest-cov (>=4.0)", "testpath", "xmltodict"] +docs = ["autodoc-traits", "flaky", "ipykernel (>=6.19.3)", "ipython", "ipywidgets", "mock", "moto", "myst-parser", "nbconvert (>=7.1.0)", "pytest (>=7.0,<8)", "pytest-asyncio", "pytest-cov (>=4.0)", "sphinx (>=1.7)", "sphinx-book-theme", "sphinxcontrib-spelling", "testpath", "xmltodict"] +test = ["flaky", "ipykernel (>=6.19.3)", "ipython", "ipywidgets", "nbconvert (>=7.1.0)", "pytest (>=7.0,<8)", "pytest-asyncio", "pytest-cov (>=4.0)", "testpath", "xmltodict"] [[package]] name = "nbconvert" -version = "7.16.4" +version = "7.16.5" description = "Converting Jupyter Notebooks (.ipynb files) to other formats. Output formats include asciidoc, html, latex, markdown, pdf, py, rst, script. nbconvert can be used both as a Python library (`import nbconvert`) or as a command line tool (invoked as `jupyter nbconvert ...`)." optional = false python-versions = ">=3.8" files = [ - {file = "nbconvert-7.16.4-py3-none-any.whl", hash = "sha256:05873c620fe520b6322bf8a5ad562692343fe3452abda5765c7a34b7d1aa3eb3"}, - {file = "nbconvert-7.16.4.tar.gz", hash = "sha256:86ca91ba266b0a448dc96fa6c5b9d98affabde2867b363258703536807f9f7f4"}, + {file = "nbconvert-7.16.5-py3-none-any.whl", hash = "sha256:e12eac052d6fd03040af4166c563d76e7aeead2e9aadf5356db552a1784bd547"}, + {file = "nbconvert-7.16.5.tar.gz", hash = "sha256:c83467bb5777fdfaac5ebbb8e864f300b277f68692ecc04d6dab72f2d8442344"}, ] [package.dependencies] beautifulsoup4 = "*" -bleach = "!=5.0.0" +bleach = {version = "!=5.0.0", extras = ["css"]} defusedxml = "*" importlib-metadata = {version = ">=3.6", markers = "python_version < \"3.10\""} jinja2 = ">=3.0" @@ -1407,7 +1395,6 @@ nbformat = ">=5.7" packaging = "*" pandocfilters = ">=1.4.1" pygments = ">=2.4.1" -tinycss2 = "*" traitlets = ">=5.1" [package.extras] @@ -1776,32 +1763,32 @@ wcwidth = "*" [[package]] name = "psutil" -version = "6.1.0" +version = "6.1.1" description = "Cross-platform lib for process and system monitoring in Python." optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7" files = [ - {file = "psutil-6.1.0-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:ff34df86226c0227c52f38b919213157588a678d049688eded74c76c8ba4a5d0"}, - {file = "psutil-6.1.0-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:c0e0c00aa18ca2d3b2b991643b799a15fc8f0563d2ebb6040f64ce8dc027b942"}, - {file = "psutil-6.1.0-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:000d1d1ebd634b4efb383f4034437384e44a6d455260aaee2eca1e9c1b55f047"}, - {file = "psutil-6.1.0-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:5cd2bcdc75b452ba2e10f0e8ecc0b57b827dd5d7aaffbc6821b2a9a242823a76"}, - {file = "psutil-6.1.0-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:045f00a43c737f960d273a83973b2511430d61f283a44c96bf13a6e829ba8fdc"}, - {file = "psutil-6.1.0-cp27-none-win32.whl", hash = "sha256:9118f27452b70bb1d9ab3198c1f626c2499384935aaf55388211ad982611407e"}, - {file = "psutil-6.1.0-cp27-none-win_amd64.whl", hash = "sha256:a8506f6119cff7015678e2bce904a4da21025cc70ad283a53b099e7620061d85"}, - {file = "psutil-6.1.0-cp36-abi3-macosx_10_9_x86_64.whl", hash = "sha256:6e2dcd475ce8b80522e51d923d10c7871e45f20918e027ab682f94f1c6351688"}, - {file = "psutil-6.1.0-cp36-abi3-macosx_11_0_arm64.whl", hash = "sha256:0895b8414afafc526712c498bd9de2b063deaac4021a3b3c34566283464aff8e"}, - {file = "psutil-6.1.0-cp36-abi3-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9dcbfce5d89f1d1f2546a2090f4fcf87c7f669d1d90aacb7d7582addece9fb38"}, - {file = "psutil-6.1.0-cp36-abi3-manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:498c6979f9c6637ebc3a73b3f87f9eb1ec24e1ce53a7c5173b8508981614a90b"}, - {file = "psutil-6.1.0-cp36-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d905186d647b16755a800e7263d43df08b790d709d575105d419f8b6ef65423a"}, - {file = "psutil-6.1.0-cp36-cp36m-win32.whl", hash = "sha256:6d3fbbc8d23fcdcb500d2c9f94e07b1342df8ed71b948a2649b5cb060a7c94ca"}, - {file = "psutil-6.1.0-cp36-cp36m-win_amd64.whl", hash = "sha256:1209036fbd0421afde505a4879dee3b2fd7b1e14fee81c0069807adcbbcca747"}, - {file = "psutil-6.1.0-cp37-abi3-win32.whl", hash = "sha256:1ad45a1f5d0b608253b11508f80940985d1d0c8f6111b5cb637533a0e6ddc13e"}, - {file = "psutil-6.1.0-cp37-abi3-win_amd64.whl", hash = "sha256:a8fb3752b491d246034fa4d279ff076501588ce8cbcdbb62c32fd7a377d996be"}, - {file = "psutil-6.1.0.tar.gz", hash = "sha256:353815f59a7f64cdaca1c0307ee13558a0512f6db064e92fe833784f08539c7a"}, + {file = "psutil-6.1.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:9ccc4316f24409159897799b83004cb1e24f9819b0dcf9c0b68bdcb6cefee6a8"}, + {file = "psutil-6.1.1-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:ca9609c77ea3b8481ab005da74ed894035936223422dc591d6772b147421f777"}, + {file = "psutil-6.1.1-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:8df0178ba8a9e5bc84fed9cfa61d54601b371fbec5c8eebad27575f1e105c0d4"}, + {file = "psutil-6.1.1-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:1924e659d6c19c647e763e78670a05dbb7feaf44a0e9c94bf9e14dfc6ba50468"}, + {file = "psutil-6.1.1-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:018aeae2af92d943fdf1da6b58665124897cfc94faa2ca92098838f83e1b1bca"}, + {file = "psutil-6.1.1-cp27-none-win32.whl", hash = "sha256:6d4281f5bbca041e2292be3380ec56a9413b790579b8e593b1784499d0005dac"}, + {file = "psutil-6.1.1-cp27-none-win_amd64.whl", hash = "sha256:c777eb75bb33c47377c9af68f30e9f11bc78e0f07fbf907be4a5d70b2fe5f030"}, + {file = "psutil-6.1.1-cp36-abi3-macosx_10_9_x86_64.whl", hash = "sha256:fc0ed7fe2231a444fc219b9c42d0376e0a9a1a72f16c5cfa0f68d19f1a0663e8"}, + {file = "psutil-6.1.1-cp36-abi3-macosx_11_0_arm64.whl", hash = "sha256:0bdd4eab935276290ad3cb718e9809412895ca6b5b334f5a9111ee6d9aff9377"}, + {file = "psutil-6.1.1-cp36-abi3-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b6e06c20c05fe95a3d7302d74e7097756d4ba1247975ad6905441ae1b5b66003"}, + {file = "psutil-6.1.1-cp36-abi3-manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:97f7cb9921fbec4904f522d972f0c0e1f4fabbdd4e0287813b21215074a0f160"}, + {file = "psutil-6.1.1-cp36-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:33431e84fee02bc84ea36d9e2c4a6d395d479c9dd9bba2376c1f6ee8f3a4e0b3"}, + {file = "psutil-6.1.1-cp36-cp36m-win32.whl", hash = "sha256:384636b1a64b47814437d1173be1427a7c83681b17a450bfc309a1953e329603"}, + {file = "psutil-6.1.1-cp36-cp36m-win_amd64.whl", hash = "sha256:8be07491f6ebe1a693f17d4f11e69d0dc1811fa082736500f649f79df7735303"}, + {file = "psutil-6.1.1-cp37-abi3-win32.whl", hash = "sha256:eaa912e0b11848c4d9279a93d7e2783df352b082f40111e078388701fd479e53"}, + {file = "psutil-6.1.1-cp37-abi3-win_amd64.whl", hash = "sha256:f35cfccb065fff93529d2afb4a2e89e363fe63ca1e4a5da22b603a85833c2649"}, + {file = "psutil-6.1.1.tar.gz", hash = "sha256:cf8496728c18f2d0b45198f06895be52f36611711746b7f30c464b422b50e2f5"}, ] [package.extras] -dev = ["black", "check-manifest", "coverage", "packaging", "pylint", "pyperf", "pypinfo", "pytest-cov", "requests", "rstcheck", "ruff", "sphinx", "sphinx_rtd_theme", "toml-sort", "twine", "virtualenv", "wheel"] +dev = ["abi3audit", "black", "check-manifest", "coverage", "packaging", "pylint", "pyperf", "pypinfo", "pytest-cov", "requests", "rstcheck", "ruff", "sphinx", "sphinx_rtd_theme", "toml-sort", "twine", "virtualenv", "vulture", "wheel"] test = ["pytest", "pytest-xdist", "setuptools"] [[package]] @@ -1856,13 +1843,13 @@ windows-terminal = ["colorama (>=0.4.6)"] [[package]] name = "pymdown-extensions" -version = "10.12" +version = "10.13" description = "Extension pack for Python Markdown." optional = false python-versions = ">=3.8" files = [ - {file = "pymdown_extensions-10.12-py3-none-any.whl", hash = "sha256:49f81412242d3527b8b4967b990df395c89563043bc51a3d2d7d500e52123b77"}, - {file = "pymdown_extensions-10.12.tar.gz", hash = "sha256:b0ee1e0b2bef1071a47891ab17003bfe5bf824a398e13f49f8ed653b699369a7"}, + {file = "pymdown_extensions-10.13-py3-none-any.whl", hash = "sha256:80bc33d715eec68e683e04298946d47d78c7739e79d808203df278ee8ef89428"}, + {file = "pymdown_extensions-10.13.tar.gz", hash = "sha256:e0b351494dc0d8d14a1f52b39b1499a00ef1566b4ba23dc74f1eba75c736f5dd"}, ] [package.dependencies] @@ -2302,101 +2289,114 @@ use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] [[package]] name = "rpds-py" -version = "0.21.0" +version = "0.22.3" description = "Python bindings to Rust's persistent data structures (rpds)" optional = false python-versions = ">=3.9" files = [ - {file = "rpds_py-0.21.0-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:a017f813f24b9df929674d0332a374d40d7f0162b326562daae8066b502d0590"}, - {file = "rpds_py-0.21.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:20cc1ed0bcc86d8e1a7e968cce15be45178fd16e2ff656a243145e0b439bd250"}, - {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ad116dda078d0bc4886cb7840e19811562acdc7a8e296ea6ec37e70326c1b41c"}, - {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:808f1ac7cf3b44f81c9475475ceb221f982ef548e44e024ad5f9e7060649540e"}, - {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:de552f4a1916e520f2703ec474d2b4d3f86d41f353e7680b597512ffe7eac5d0"}, - {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:efec946f331349dfc4ae9d0e034c263ddde19414fe5128580f512619abed05f1"}, - {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b80b4690bbff51a034bfde9c9f6bf9357f0a8c61f548942b80f7b66356508bf5"}, - {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:085ed25baac88953d4283e5b5bd094b155075bb40d07c29c4f073e10623f9f2e"}, - {file = "rpds_py-0.21.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:daa8efac2a1273eed2354397a51216ae1e198ecbce9036fba4e7610b308b6153"}, - {file = "rpds_py-0.21.0-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:95a5bad1ac8a5c77b4e658671642e4af3707f095d2b78a1fdd08af0dfb647624"}, - {file = "rpds_py-0.21.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:3e53861b29a13d5b70116ea4230b5f0f3547b2c222c5daa090eb7c9c82d7f664"}, - {file = "rpds_py-0.21.0-cp310-none-win32.whl", hash = "sha256:ea3a6ac4d74820c98fcc9da4a57847ad2cc36475a8bd9683f32ab6d47a2bd682"}, - {file = "rpds_py-0.21.0-cp310-none-win_amd64.whl", hash = "sha256:b8f107395f2f1d151181880b69a2869c69e87ec079c49c0016ab96860b6acbe5"}, - {file = "rpds_py-0.21.0-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:5555db3e618a77034954b9dc547eae94166391a98eb867905ec8fcbce1308d95"}, - {file = "rpds_py-0.21.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:97ef67d9bbc3e15584c2f3c74bcf064af36336c10d2e21a2131e123ce0f924c9"}, - {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4ab2c2a26d2f69cdf833174f4d9d86118edc781ad9a8fa13970b527bf8236027"}, - {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:4e8921a259f54bfbc755c5bbd60c82bb2339ae0324163f32868f63f0ebb873d9"}, - {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8a7ff941004d74d55a47f916afc38494bd1cfd4b53c482b77c03147c91ac0ac3"}, - {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5145282a7cd2ac16ea0dc46b82167754d5e103a05614b724457cffe614f25bd8"}, - {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:de609a6f1b682f70bb7163da745ee815d8f230d97276db049ab447767466a09d"}, - {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:40c91c6e34cf016fa8e6b59d75e3dbe354830777fcfd74c58b279dceb7975b75"}, - {file = "rpds_py-0.21.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:d2132377f9deef0c4db89e65e8bb28644ff75a18df5293e132a8d67748397b9f"}, - {file = "rpds_py-0.21.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:0a9e0759e7be10109645a9fddaaad0619d58c9bf30a3f248a2ea57a7c417173a"}, - {file = "rpds_py-0.21.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:9e20da3957bdf7824afdd4b6eeb29510e83e026473e04952dca565170cd1ecc8"}, - {file = "rpds_py-0.21.0-cp311-none-win32.whl", hash = "sha256:f71009b0d5e94c0e86533c0b27ed7cacc1239cb51c178fd239c3cfefefb0400a"}, - {file = "rpds_py-0.21.0-cp311-none-win_amd64.whl", hash = "sha256:e168afe6bf6ab7ab46c8c375606298784ecbe3ba31c0980b7dcbb9631dcba97e"}, - {file = "rpds_py-0.21.0-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:30b912c965b2aa76ba5168fd610087bad7fcde47f0a8367ee8f1876086ee6d1d"}, - {file = "rpds_py-0.21.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:ca9989d5d9b1b300bc18e1801c67b9f6d2c66b8fd9621b36072ed1df2c977f72"}, - {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6f54e7106f0001244a5f4cf810ba8d3f9c542e2730821b16e969d6887b664266"}, - {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:fed5dfefdf384d6fe975cc026886aece4f292feaf69d0eeb716cfd3c5a4dd8be"}, - {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:590ef88db231c9c1eece44dcfefd7515d8bf0d986d64d0caf06a81998a9e8cab"}, - {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f983e4c2f603c95dde63df633eec42955508eefd8d0f0e6d236d31a044c882d7"}, - {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b229ce052ddf1a01c67d68166c19cb004fb3612424921b81c46e7ea7ccf7c3bf"}, - {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:ebf64e281a06c904a7636781d2e973d1f0926a5b8b480ac658dc0f556e7779f4"}, - {file = "rpds_py-0.21.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:998a8080c4495e4f72132f3d66ff91f5997d799e86cec6ee05342f8f3cda7dca"}, - {file = "rpds_py-0.21.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:98486337f7b4f3c324ab402e83453e25bb844f44418c066623db88e4c56b7c7b"}, - {file = "rpds_py-0.21.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:a78d8b634c9df7f8d175451cfeac3810a702ccb85f98ec95797fa98b942cea11"}, - {file = "rpds_py-0.21.0-cp312-none-win32.whl", hash = "sha256:a58ce66847711c4aa2ecfcfaff04cb0327f907fead8945ffc47d9407f41ff952"}, - {file = "rpds_py-0.21.0-cp312-none-win_amd64.whl", hash = "sha256:e860f065cc4ea6f256d6f411aba4b1251255366e48e972f8a347cf88077b24fd"}, - {file = "rpds_py-0.21.0-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:ee4eafd77cc98d355a0d02f263efc0d3ae3ce4a7c24740010a8b4012bbb24937"}, - {file = "rpds_py-0.21.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:688c93b77e468d72579351a84b95f976bd7b3e84aa6686be6497045ba84be560"}, - {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c38dbf31c57032667dd5a2f0568ccde66e868e8f78d5a0d27dcc56d70f3fcd3b"}, - {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2d6129137f43f7fa02d41542ffff4871d4aefa724a5fe38e2c31a4e0fd343fb0"}, - {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:520ed8b99b0bf86a176271f6fe23024323862ac674b1ce5b02a72bfeff3fff44"}, - {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:aaeb25ccfb9b9014a10eaf70904ebf3f79faaa8e60e99e19eef9f478651b9b74"}, - {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:af04ac89c738e0f0f1b913918024c3eab6e3ace989518ea838807177d38a2e94"}, - {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:b9b76e2afd585803c53c5b29e992ecd183f68285b62fe2668383a18e74abe7a3"}, - {file = "rpds_py-0.21.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:5afb5efde74c54724e1a01118c6e5c15e54e642c42a1ba588ab1f03544ac8c7a"}, - {file = "rpds_py-0.21.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:52c041802a6efa625ea18027a0723676a778869481d16803481ef6cc02ea8cb3"}, - {file = "rpds_py-0.21.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:ee1e4fc267b437bb89990b2f2abf6c25765b89b72dd4a11e21934df449e0c976"}, - {file = "rpds_py-0.21.0-cp313-none-win32.whl", hash = "sha256:0c025820b78817db6a76413fff6866790786c38f95ea3f3d3c93dbb73b632202"}, - {file = "rpds_py-0.21.0-cp313-none-win_amd64.whl", hash = "sha256:320c808df533695326610a1b6a0a6e98f033e49de55d7dc36a13c8a30cfa756e"}, - {file = "rpds_py-0.21.0-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:2c51d99c30091f72a3c5d126fad26236c3f75716b8b5e5cf8effb18889ced928"}, - {file = "rpds_py-0.21.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:cbd7504a10b0955ea287114f003b7ad62330c9e65ba012c6223dba646f6ffd05"}, - {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6dcc4949be728ede49e6244eabd04064336012b37f5c2200e8ec8eb2988b209c"}, - {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f414da5c51bf350e4b7960644617c130140423882305f7574b6cf65a3081cecb"}, - {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9afe42102b40007f588666bc7de82451e10c6788f6f70984629db193849dced1"}, - {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3b929c2bb6e29ab31f12a1117c39f7e6d6450419ab7464a4ea9b0b417174f044"}, - {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8404b3717da03cbf773a1d275d01fec84ea007754ed380f63dfc24fb76ce4592"}, - {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:e12bb09678f38b7597b8346983d2323a6482dcd59e423d9448108c1be37cac9d"}, - {file = "rpds_py-0.21.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:58a0e345be4b18e6b8501d3b0aa540dad90caeed814c515e5206bb2ec26736fd"}, - {file = "rpds_py-0.21.0-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:c3761f62fcfccf0864cc4665b6e7c3f0c626f0380b41b8bd1ce322103fa3ef87"}, - {file = "rpds_py-0.21.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:c2b2f71c6ad6c2e4fc9ed9401080badd1469fa9889657ec3abea42a3d6b2e1ed"}, - {file = "rpds_py-0.21.0-cp39-none-win32.whl", hash = "sha256:b21747f79f360e790525e6f6438c7569ddbfb1b3197b9e65043f25c3c9b489d8"}, - {file = "rpds_py-0.21.0-cp39-none-win_amd64.whl", hash = "sha256:0626238a43152918f9e72ede9a3b6ccc9e299adc8ade0d67c5e142d564c9a83d"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:6b4ef7725386dc0762857097f6b7266a6cdd62bfd209664da6712cb26acef035"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:6bc0e697d4d79ab1aacbf20ee5f0df80359ecf55db33ff41481cf3e24f206919"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:da52d62a96e61c1c444f3998c434e8b263c384f6d68aca8274d2e08d1906325c"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:98e4fe5db40db87ce1c65031463a760ec7906ab230ad2249b4572c2fc3ef1f9f"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:30bdc973f10d28e0337f71d202ff29345320f8bc49a31c90e6c257e1ccef4333"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:faa5e8496c530f9c71f2b4e1c49758b06e5f4055e17144906245c99fa6d45356"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:32eb88c30b6a4f0605508023b7141d043a79b14acb3b969aa0b4f99b25bc7d4a"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:a89a8ce9e4e75aeb7fa5d8ad0f3fecdee813802592f4f46a15754dcb2fd6b061"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:241e6c125568493f553c3d0fdbb38c74babf54b45cef86439d4cd97ff8feb34d"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-musllinux_1_2_i686.whl", hash = "sha256:3b766a9f57663396e4f34f5140b3595b233a7b146e94777b97a8413a1da1be18"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:af4a644bf890f56e41e74be7d34e9511e4954894d544ec6b8efe1e21a1a8da6c"}, - {file = "rpds_py-0.21.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:3e30a69a706e8ea20444b98a49f386c17b26f860aa9245329bab0851ed100677"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:031819f906bb146561af051c7cef4ba2003d28cff07efacef59da973ff7969ba"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:b876f2bc27ab5954e2fd88890c071bd0ed18b9c50f6ec3de3c50a5ece612f7a6"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dc5695c321e518d9f03b7ea6abb5ea3af4567766f9852ad1560f501b17588c7b"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:b4de1da871b5c0fd5537b26a6fc6814c3cc05cabe0c941db6e9044ffbb12f04a"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:878f6fea96621fda5303a2867887686d7a198d9e0f8a40be100a63f5d60c88c9"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a8eeec67590e94189f434c6d11c426892e396ae59e4801d17a93ac96b8c02a6c"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1ff2eba7f6c0cb523d7e9cff0903f2fe1feff8f0b2ceb6bd71c0e20a4dcee271"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:a429b99337062877d7875e4ff1a51fe788424d522bd64a8c0a20ef3021fdb6ed"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:d167e4dbbdac48bd58893c7e446684ad5d425b407f9336e04ab52e8b9194e2ed"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-musllinux_1_2_i686.whl", hash = "sha256:4eb2de8a147ffe0626bfdc275fc6563aa7bf4b6db59cf0d44f0ccd6ca625a24e"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:e78868e98f34f34a88e23ee9ccaeeec460e4eaf6db16d51d7a9b883e5e785a5e"}, - {file = "rpds_py-0.21.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:4991ca61656e3160cdaca4851151fd3f4a92e9eba5c7a530ab030d6aee96ec89"}, - {file = "rpds_py-0.21.0.tar.gz", hash = "sha256:ed6378c9d66d0de903763e7706383d60c33829581f0adff47b6535f1802fa6db"}, + {file = "rpds_py-0.22.3-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:6c7b99ca52c2c1752b544e310101b98a659b720b21db00e65edca34483259967"}, + {file = "rpds_py-0.22.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:be2eb3f2495ba669d2a985f9b426c1797b7d48d6963899276d22f23e33d47e37"}, + {file = "rpds_py-0.22.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:70eb60b3ae9245ddea20f8a4190bd79c705a22f8028aaf8bbdebe4716c3fab24"}, + {file = "rpds_py-0.22.3-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:4041711832360a9b75cfb11b25a6a97c8fb49c07b8bd43d0d02b45d0b499a4ff"}, + {file = "rpds_py-0.22.3-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:64607d4cbf1b7e3c3c8a14948b99345eda0e161b852e122c6bb71aab6d1d798c"}, + {file = "rpds_py-0.22.3-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:81e69b0a0e2537f26d73b4e43ad7bc8c8efb39621639b4434b76a3de50c6966e"}, + {file = "rpds_py-0.22.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc27863442d388870c1809a87507727b799c8460573cfbb6dc0eeaef5a11b5ec"}, + {file = "rpds_py-0.22.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:e79dd39f1e8c3504be0607e5fc6e86bb60fe3584bec8b782578c3b0fde8d932c"}, + {file = "rpds_py-0.22.3-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:e0fa2d4ec53dc51cf7d3bb22e0aa0143966119f42a0c3e4998293a3dd2856b09"}, + {file = "rpds_py-0.22.3-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:fda7cb070f442bf80b642cd56483b5548e43d366fe3f39b98e67cce780cded00"}, + {file = "rpds_py-0.22.3-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:cff63a0272fcd259dcc3be1657b07c929c466b067ceb1c20060e8d10af56f5bf"}, + {file = "rpds_py-0.22.3-cp310-cp310-win32.whl", hash = "sha256:9bd7228827ec7bb817089e2eb301d907c0d9827a9e558f22f762bb690b131652"}, + {file = "rpds_py-0.22.3-cp310-cp310-win_amd64.whl", hash = "sha256:9beeb01d8c190d7581a4d59522cd3d4b6887040dcfc744af99aa59fef3e041a8"}, + {file = "rpds_py-0.22.3-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:d20cfb4e099748ea39e6f7b16c91ab057989712d31761d3300d43134e26e165f"}, + {file = "rpds_py-0.22.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:68049202f67380ff9aa52f12e92b1c30115f32e6895cd7198fa2a7961621fc5a"}, + {file = "rpds_py-0.22.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fb4f868f712b2dd4bcc538b0a0c1f63a2b1d584c925e69a224d759e7070a12d5"}, + {file = "rpds_py-0.22.3-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:bc51abd01f08117283c5ebf64844a35144a0843ff7b2983e0648e4d3d9f10dbb"}, + {file = "rpds_py-0.22.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:0f3cec041684de9a4684b1572fe28c7267410e02450f4561700ca5a3bc6695a2"}, + {file = "rpds_py-0.22.3-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7ef9d9da710be50ff6809fed8f1963fecdfecc8b86656cadfca3bc24289414b0"}, + {file = "rpds_py-0.22.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:59f4a79c19232a5774aee369a0c296712ad0e77f24e62cad53160312b1c1eaa1"}, + {file = "rpds_py-0.22.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:1a60bce91f81ddaac922a40bbb571a12c1070cb20ebd6d49c48e0b101d87300d"}, + {file = "rpds_py-0.22.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:e89391e6d60251560f0a8f4bd32137b077a80d9b7dbe6d5cab1cd80d2746f648"}, + {file = "rpds_py-0.22.3-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:e3fb866d9932a3d7d0c82da76d816996d1667c44891bd861a0f97ba27e84fc74"}, + {file = "rpds_py-0.22.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:1352ae4f7c717ae8cba93421a63373e582d19d55d2ee2cbb184344c82d2ae55a"}, + {file = "rpds_py-0.22.3-cp311-cp311-win32.whl", hash = "sha256:b0b4136a252cadfa1adb705bb81524eee47d9f6aab4f2ee4fa1e9d3cd4581f64"}, + {file = "rpds_py-0.22.3-cp311-cp311-win_amd64.whl", hash = "sha256:8bd7c8cfc0b8247c8799080fbff54e0b9619e17cdfeb0478ba7295d43f635d7c"}, + {file = "rpds_py-0.22.3-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:27e98004595899949bd7a7b34e91fa7c44d7a97c40fcaf1d874168bb652ec67e"}, + {file = "rpds_py-0.22.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:1978d0021e943aae58b9b0b196fb4895a25cc53d3956b8e35e0b7682eefb6d56"}, + {file = "rpds_py-0.22.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:655ca44a831ecb238d124e0402d98f6212ac527a0ba6c55ca26f616604e60a45"}, + {file = "rpds_py-0.22.3-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:feea821ee2a9273771bae61194004ee2fc33f8ec7db08117ef9147d4bbcbca8e"}, + {file = "rpds_py-0.22.3-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:22bebe05a9ffc70ebfa127efbc429bc26ec9e9b4ee4d15a740033efda515cf3d"}, + {file = "rpds_py-0.22.3-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3af6e48651c4e0d2d166dc1b033b7042ea3f871504b6805ba5f4fe31581d8d38"}, + {file = "rpds_py-0.22.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e67ba3c290821343c192f7eae1d8fd5999ca2dc99994114643e2f2d3e6138b15"}, + {file = "rpds_py-0.22.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:02fbb9c288ae08bcb34fb41d516d5eeb0455ac35b5512d03181d755d80810059"}, + {file = "rpds_py-0.22.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:f56a6b404f74ab372da986d240e2e002769a7d7102cc73eb238a4f72eec5284e"}, + {file = "rpds_py-0.22.3-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:0a0461200769ab3b9ab7e513f6013b7a97fdeee41c29b9db343f3c5a8e2b9e61"}, + {file = "rpds_py-0.22.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:8633e471c6207a039eff6aa116e35f69f3156b3989ea3e2d755f7bc41754a4a7"}, + {file = "rpds_py-0.22.3-cp312-cp312-win32.whl", hash = "sha256:593eba61ba0c3baae5bc9be2f5232430453fb4432048de28399ca7376de9c627"}, + {file = "rpds_py-0.22.3-cp312-cp312-win_amd64.whl", hash = "sha256:d115bffdd417c6d806ea9069237a4ae02f513b778e3789a359bc5856e0404cc4"}, + {file = "rpds_py-0.22.3-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:ea7433ce7e4bfc3a85654aeb6747babe3f66eaf9a1d0c1e7a4435bbdf27fea84"}, + {file = "rpds_py-0.22.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:6dd9412824c4ce1aca56c47b0991e65bebb7ac3f4edccfd3f156150c96a7bf25"}, + {file = "rpds_py-0.22.3-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:20070c65396f7373f5df4005862fa162db5d25d56150bddd0b3e8214e8ef45b4"}, + {file = "rpds_py-0.22.3-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:0b09865a9abc0ddff4e50b5ef65467cd94176bf1e0004184eb915cbc10fc05c5"}, + {file = "rpds_py-0.22.3-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3453e8d41fe5f17d1f8e9c383a7473cd46a63661628ec58e07777c2fff7196dc"}, + {file = "rpds_py-0.22.3-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f5d36399a1b96e1a5fdc91e0522544580dbebeb1f77f27b2b0ab25559e103b8b"}, + {file = "rpds_py-0.22.3-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:009de23c9c9ee54bf11303a966edf4d9087cd43a6003672e6aa7def643d06518"}, + {file = "rpds_py-0.22.3-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:1aef18820ef3e4587ebe8b3bc9ba6e55892a6d7b93bac6d29d9f631a3b4befbd"}, + {file = "rpds_py-0.22.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:f60bd8423be1d9d833f230fdbccf8f57af322d96bcad6599e5a771b151398eb2"}, + {file = "rpds_py-0.22.3-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:62d9cfcf4948683a18a9aff0ab7e1474d407b7bab2ca03116109f8464698ab16"}, + {file = "rpds_py-0.22.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:9253fc214112405f0afa7db88739294295f0e08466987f1d70e29930262b4c8f"}, + {file = "rpds_py-0.22.3-cp313-cp313-win32.whl", hash = "sha256:fb0ba113b4983beac1a2eb16faffd76cb41e176bf58c4afe3e14b9c681f702de"}, + {file = "rpds_py-0.22.3-cp313-cp313-win_amd64.whl", hash = "sha256:c58e2339def52ef6b71b8f36d13c3688ea23fa093353f3a4fee2556e62086ec9"}, + {file = "rpds_py-0.22.3-cp313-cp313t-macosx_10_12_x86_64.whl", hash = "sha256:f82a116a1d03628a8ace4859556fb39fd1424c933341a08ea3ed6de1edb0283b"}, + {file = "rpds_py-0.22.3-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:3dfcbc95bd7992b16f3f7ba05af8a64ca694331bd24f9157b49dadeeb287493b"}, + {file = "rpds_py-0.22.3-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:59259dc58e57b10e7e18ce02c311804c10c5a793e6568f8af4dead03264584d1"}, + {file = "rpds_py-0.22.3-cp313-cp313t-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:5725dd9cc02068996d4438d397e255dcb1df776b7ceea3b9cb972bdb11260a83"}, + {file = "rpds_py-0.22.3-cp313-cp313t-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:99b37292234e61325e7a5bb9689e55e48c3f5f603af88b1642666277a81f1fbd"}, + {file = "rpds_py-0.22.3-cp313-cp313t-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:27b1d3b3915a99208fee9ab092b8184c420f2905b7d7feb4aeb5e4a9c509b8a1"}, + {file = "rpds_py-0.22.3-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f612463ac081803f243ff13cccc648578e2279295048f2a8d5eb430af2bae6e3"}, + {file = "rpds_py-0.22.3-cp313-cp313t-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:f73d3fef726b3243a811121de45193c0ca75f6407fe66f3f4e183c983573e130"}, + {file = "rpds_py-0.22.3-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:3f21f0495edea7fdbaaa87e633a8689cd285f8f4af5c869f27bc8074638ad69c"}, + {file = "rpds_py-0.22.3-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:1e9663daaf7a63ceccbbb8e3808fe90415b0757e2abddbfc2e06c857bf8c5e2b"}, + {file = "rpds_py-0.22.3-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:a76e42402542b1fae59798fab64432b2d015ab9d0c8c47ba7addddbaf7952333"}, + {file = "rpds_py-0.22.3-cp313-cp313t-win32.whl", hash = "sha256:69803198097467ee7282750acb507fba35ca22cc3b85f16cf45fb01cb9097730"}, + {file = "rpds_py-0.22.3-cp313-cp313t-win_amd64.whl", hash = "sha256:f5cf2a0c2bdadf3791b5c205d55a37a54025c6e18a71c71f82bb536cf9a454bf"}, + {file = "rpds_py-0.22.3-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:378753b4a4de2a7b34063d6f95ae81bfa7b15f2c1a04a9518e8644e81807ebea"}, + {file = "rpds_py-0.22.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:3445e07bf2e8ecfeef6ef67ac83de670358abf2996916039b16a218e3d95e97e"}, + {file = "rpds_py-0.22.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7b2513ba235829860b13faa931f3b6846548021846ac808455301c23a101689d"}, + {file = "rpds_py-0.22.3-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:eaf16ae9ae519a0e237a0f528fd9f0197b9bb70f40263ee57ae53c2b8d48aeb3"}, + {file = "rpds_py-0.22.3-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:583f6a1993ca3369e0f80ba99d796d8e6b1a3a2a442dd4e1a79e652116413091"}, + {file = "rpds_py-0.22.3-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4617e1915a539a0d9a9567795023de41a87106522ff83fbfaf1f6baf8e85437e"}, + {file = "rpds_py-0.22.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0c150c7a61ed4a4f4955a96626574e9baf1adf772c2fb61ef6a5027e52803543"}, + {file = "rpds_py-0.22.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2fa4331c200c2521512595253f5bb70858b90f750d39b8cbfd67465f8d1b596d"}, + {file = "rpds_py-0.22.3-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:214b7a953d73b5e87f0ebece4a32a5bd83c60a3ecc9d4ec8f1dca968a2d91e99"}, + {file = "rpds_py-0.22.3-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:f47ad3d5f3258bd7058d2d506852217865afefe6153a36eb4b6928758041d831"}, + {file = "rpds_py-0.22.3-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:f276b245347e6e36526cbd4a266a417796fc531ddf391e43574cf6466c492520"}, + {file = "rpds_py-0.22.3-cp39-cp39-win32.whl", hash = "sha256:bbb232860e3d03d544bc03ac57855cd82ddf19c7a07651a7c0fdb95e9efea8b9"}, + {file = "rpds_py-0.22.3-cp39-cp39-win_amd64.whl", hash = "sha256:cfbc454a2880389dbb9b5b398e50d439e2e58669160f27b60e5eca11f68ae17c"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:d48424e39c2611ee1b84ad0f44fb3b2b53d473e65de061e3f460fc0be5f1939d"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:24e8abb5878e250f2eb0d7859a8e561846f98910326d06c0d51381fed59357bd"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4b232061ca880db21fa14defe219840ad9b74b6158adb52ddf0e87bead9e8493"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:ac0a03221cdb5058ce0167ecc92a8c89e8d0decdc9e99a2ec23380793c4dcb96"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:eb0c341fa71df5a4595f9501df4ac5abfb5a09580081dffbd1ddd4654e6e9123"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:bf9db5488121b596dbfc6718c76092fda77b703c1f7533a226a5a9f65248f8ad"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0b8db6b5b2d4491ad5b6bdc2bc7c017eec108acbf4e6785f42a9eb0ba234f4c9"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:b3d504047aba448d70cf6fa22e06cb09f7cbd761939fdd47604f5e007675c24e"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:e61b02c3f7a1e0b75e20c3978f7135fd13cb6cf551bf4a6d29b999a88830a338"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-musllinux_1_2_i686.whl", hash = "sha256:e35ba67d65d49080e8e5a1dd40101fccdd9798adb9b050ff670b7d74fa41c566"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:26fd7cac7dd51011a245f29a2cc6489c4608b5a8ce8d75661bb4a1066c52dfbe"}, + {file = "rpds_py-0.22.3-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:177c7c0fce2855833819c98e43c262007f42ce86651ffbb84f37883308cb0e7d"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:bb47271f60660803ad11f4c61b42242b8c1312a31c98c578f79ef9387bbde21c"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:70fb28128acbfd264eda9bf47015537ba3fe86e40d046eb2963d75024be4d055"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:44d61b4b7d0c2c9ac019c314e52d7cbda0ae31078aabd0f22e583af3e0d79723"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:5f0e260eaf54380380ac3808aa4ebe2d8ca28b9087cf411649f96bad6900c728"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b25bc607423935079e05619d7de556c91fb6adeae9d5f80868dde3468657994b"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fb6116dfb8d1925cbdb52595560584db42a7f664617a1f7d7f6e32f138cdf37d"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a63cbdd98acef6570c62b92a1e43266f9e8b21e699c363c0fef13bd530799c11"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2b8f60e1b739a74bab7e01fcbe3dddd4657ec685caa04681df9d562ef15b625f"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:2e8b55d8517a2fda8d95cb45d62a5a8bbf9dd0ad39c5b25c8833efea07b880ca"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-musllinux_1_2_i686.whl", hash = "sha256:2de29005e11637e7a2361fa151f780ff8eb2543a0da1413bb951e9f14b699ef3"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:666ecce376999bf619756a24ce15bb14c5bfaf04bf00abc7e663ce17c3f34fe7"}, + {file = "rpds_py-0.22.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:5246b14ca64a8675e0a7161f7af68fe3e910e6b90542b4bfb5439ba752191df6"}, + {file = "rpds_py-0.22.3.tar.gz", hash = "sha256:e32fee8ab45d3c2db6da19a5323bc3362237c8b653c70194414b892fd06a080d"}, ] [[package]] @@ -2428,13 +2428,13 @@ files = [ [[package]] name = "six" -version = "1.16.0" +version = "1.17.0" description = "Python 2 and 3 compatibility utilities" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7" files = [ - {file = "six-1.16.0-py2.py3-none-any.whl", hash = "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254"}, - {file = "six-1.16.0.tar.gz", hash = "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926"}, + {file = "six-1.17.0-py2.py3-none-any.whl", hash = "sha256:4721f391ed90541fddacab5acf947aa0d3dc7d27b2e1e8eda2be8970586c3274"}, + {file = "six-1.17.0.tar.gz", hash = "sha256:ff70335d468e7eb6ec65b95b99d3a2836546063f63acc5171de367e834932a81"}, ] [[package]] @@ -2487,13 +2487,43 @@ test = ["pytest", "ruff"] [[package]] name = "tomli" -version = "2.1.0" +version = "2.2.1" description = "A lil' TOML parser" optional = false python-versions = ">=3.8" files = [ - {file = "tomli-2.1.0-py3-none-any.whl", hash = "sha256:a5c57c3d1c56f5ccdf89f6523458f60ef716e210fc47c4cfb188c5ba473e0391"}, - {file = "tomli-2.1.0.tar.gz", hash = "sha256:3f646cae2aec94e17d04973e4249548320197cfabdf130015d023de4b74d8ab8"}, + {file = "tomli-2.2.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:678e4fa69e4575eb77d103de3df8a895e1591b48e740211bd1067378c69e8249"}, + {file = "tomli-2.2.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:023aa114dd824ade0100497eb2318602af309e5a55595f76b626d6d9f3b7b0a6"}, + {file = "tomli-2.2.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ece47d672db52ac607a3d9599a9d48dcb2f2f735c6c2d1f34130085bb12b112a"}, + {file = "tomli-2.2.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6972ca9c9cc9f0acaa56a8ca1ff51e7af152a9f87fb64623e31d5c83700080ee"}, + {file = "tomli-2.2.1-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c954d2250168d28797dd4e3ac5cf812a406cd5a92674ee4c8f123c889786aa8e"}, + {file = "tomli-2.2.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:8dd28b3e155b80f4d54beb40a441d366adcfe740969820caf156c019fb5c7ec4"}, + {file = "tomli-2.2.1-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:e59e304978767a54663af13c07b3d1af22ddee3bb2fb0618ca1593e4f593a106"}, + {file = "tomli-2.2.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:33580bccab0338d00994d7f16f4c4ec25b776af3ffaac1ed74e0b3fc95e885a8"}, + {file = "tomli-2.2.1-cp311-cp311-win32.whl", hash = "sha256:465af0e0875402f1d226519c9904f37254b3045fc5084697cefb9bdde1ff99ff"}, + {file = "tomli-2.2.1-cp311-cp311-win_amd64.whl", hash = "sha256:2d0f2fdd22b02c6d81637a3c95f8cd77f995846af7414c5c4b8d0545afa1bc4b"}, + {file = "tomli-2.2.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:4a8f6e44de52d5e6c657c9fe83b562f5f4256d8ebbfe4ff922c495620a7f6cea"}, + {file = "tomli-2.2.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:8d57ca8095a641b8237d5b079147646153d22552f1c637fd3ba7f4b0b29167a8"}, + {file = "tomli-2.2.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4e340144ad7ae1533cb897d406382b4b6fede8890a03738ff1683af800d54192"}, + {file = "tomli-2.2.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:db2b95f9de79181805df90bedc5a5ab4c165e6ec3fe99f970d0e302f384ad222"}, + {file = "tomli-2.2.1-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:40741994320b232529c802f8bc86da4e1aa9f413db394617b9a256ae0f9a7f77"}, + {file = "tomli-2.2.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:400e720fe168c0f8521520190686ef8ef033fb19fc493da09779e592861b78c6"}, + {file = "tomli-2.2.1-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:02abe224de6ae62c19f090f68da4e27b10af2b93213d36cf44e6e1c5abd19fdd"}, + {file = "tomli-2.2.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:b82ebccc8c8a36f2094e969560a1b836758481f3dc360ce9a3277c65f374285e"}, + {file = "tomli-2.2.1-cp312-cp312-win32.whl", hash = "sha256:889f80ef92701b9dbb224e49ec87c645ce5df3fa2cc548664eb8a25e03127a98"}, + {file = "tomli-2.2.1-cp312-cp312-win_amd64.whl", hash = "sha256:7fc04e92e1d624a4a63c76474610238576942d6b8950a2d7f908a340494e67e4"}, + {file = "tomli-2.2.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:f4039b9cbc3048b2416cc57ab3bda989a6fcf9b36cf8937f01a6e731b64f80d7"}, + {file = "tomli-2.2.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:286f0ca2ffeeb5b9bd4fcc8d6c330534323ec51b2f52da063b11c502da16f30c"}, + {file = "tomli-2.2.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a92ef1a44547e894e2a17d24e7557a5e85a9e1d0048b0b5e7541f76c5032cb13"}, + {file = "tomli-2.2.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9316dc65bed1684c9a98ee68759ceaed29d229e985297003e494aa825ebb0281"}, + {file = "tomli-2.2.1-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e85e99945e688e32d5a35c1ff38ed0b3f41f43fad8df0bdf79f72b2ba7bc5272"}, + {file = "tomli-2.2.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:ac065718db92ca818f8d6141b5f66369833d4a80a9d74435a268c52bdfa73140"}, + {file = "tomli-2.2.1-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:d920f33822747519673ee656a4b6ac33e382eca9d331c87770faa3eef562aeb2"}, + {file = "tomli-2.2.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:a198f10c4d1b1375d7687bc25294306e551bf1abfa4eace6650070a5c1ae2744"}, + {file = "tomli-2.2.1-cp313-cp313-win32.whl", hash = "sha256:d3f5614314d758649ab2ab3a62d4f2004c825922f9e370b29416484086b264ec"}, + {file = "tomli-2.2.1-cp313-cp313-win_amd64.whl", hash = "sha256:a38aa0308e754b0e3c67e344754dff64999ff9b513e691d0e786265c93583c69"}, + {file = "tomli-2.2.1-py3-none-any.whl", hash = "sha256:cb55c73c5f4408779d0cf3eef9f762b9c9f147a77de7b258bef0a5628adc85cc"}, + {file = "tomli-2.2.1.tar.gz", hash = "sha256:cd45e1dc79c835ce60f7404ec8119f2eb06d38b1deba146f07ced3bbc44505ff"}, ] [[package]] @@ -2555,13 +2585,13 @@ files = [ [[package]] name = "urllib3" -version = "2.2.3" +version = "2.3.0" description = "HTTP library with thread-safe connection pooling, file post, and more." optional = false -python-versions = ">=3.8" +python-versions = ">=3.9" files = [ - {file = "urllib3-2.2.3-py3-none-any.whl", hash = "sha256:ca899ca043dcb1bafa3e262d73aa25c465bfb49e0bd9dd5d59f1d0acba2f8fac"}, - {file = "urllib3-2.2.3.tar.gz", hash = "sha256:e7d814a81dad81e6caf2ec9fdedb284ecc9c73076b62654547cc64ccdcae26e9"}, + {file = "urllib3-2.3.0-py3-none-any.whl", hash = "sha256:1cee9ad369867bfdbbb48b7dd50374c0967a0bb7710050facf0dd6911440e3df"}, + {file = "urllib3-2.3.0.tar.gz", hash = "sha256:f8c5449b3cf0861679ce7e0503c7b44b5ec981bec0d1d3795a07f1ba96f0204d"}, ] [package.extras] @@ -2680,4 +2710,4 @@ type = ["pytest-mypy"] [metadata] lock-version = "2.0" python-versions = "^3.9" -content-hash = "563eafd03cf4755d613d36ed8fb0feb6cfd15a2fd893aa240a171899c1c74a28" +content-hash = "a193dfef3cbd636299b9675ba69077c9b2b760d960550d72ca16dfab3993aa0d" From 9304bd017412c2bb21cab36acc607009c456626b Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Thu, 2 Jan 2025 16:49:27 -0500 Subject: [PATCH 14/19] major refactor and usage of h5py to extract groups, variables, etc --- ncompare/Comparison.py | 481 ++++++++++++++++++ ncompare/console.py | 4 +- ncompare/core.py | 406 +-------------- ncompare/getters.py | 73 ++- ...utils.py => path_and_string_operations.py} | 18 + ncompare/printing.py | 5 +- ncompare/sequence_operations.py | 7 +- ncompare/{core_types.py => utility_types.py} | 27 +- tests/test_cli.py | 4 +- tests/test_getters.py | 37 +- ....py => test_path_and_string_operations.py} | 2 +- 11 files changed, 587 insertions(+), 477 deletions(-) create mode 100644 ncompare/Comparison.py rename ncompare/{utils.py => path_and_string_operations.py} (79%) rename ncompare/{core_types.py => utility_types.py} (83%) rename tests/{test_utils.py => test_path_and_string_operations.py} (96%) diff --git a/ncompare/Comparison.py b/ncompare/Comparison.py new file mode 100644 index 0000000..aed32a9 --- /dev/null +++ b/ncompare/Comparison.py @@ -0,0 +1,481 @@ +from collections.abc import Iterator +from typing import Union + +import h5py +import netCDF4 +import numpy as np +from colorama import Fore + +from ncompare.getters import ( + get_and_check_variable_attributes, + get_and_check_variable_scale_factor, + get_root_dims, + get_root_groups, + get_subgroups, + get_variables, +) +from ncompare.printing import Outputter +from ncompare.sequence_operations import common_elements, count_diffs +from ncompare.utility_types import ( + FileToCompare, + GroupPair, + SummaryDifferenceKeys, + SummaryDifferencesDict, + VarProperties, +) + + +class Comparison: + def __init__( + self, + file1: FileToCompare, + file2: FileToCompare, + out: Outputter, + show_chunks: bool, + show_attributes: bool, + ): + assert file1.type == file2.type + self.file1 = file1 + self.file2 = file2 + self.file_types = file1.type + self.out: Outputter = out + self.show_chunks: bool = show_chunks + self.show_attributes: bool = show_attributes + + blank_difference_dict: SummaryDifferencesDict = { + "shared": 0, + "left": 0, + "right": 0, + "both": 0, + "difference_types": set(), + } + self.num_group_diffs: SummaryDifferencesDict = blank_difference_dict.copy() + self.num_var_diffs: SummaryDifferencesDict = blank_difference_dict.copy() + self.num_attribute_diffs: SummaryDifferencesDict = blank_difference_dict.copy() + + self.open_file1 = None + self.open_file2 = None + + def run_through_comparisons(self) -> int: + """Execute a series of comparisons between two netCDF or HDF files. + + Returns + ------- + int + total number of differences found (across variables, groups, and attributes) + """ + self._print_root_dimensions() + self._print_root_groups() + + # Run through all the rest of the groups and variables, tallying differences along the way. + self.out.print(Fore.LIGHTBLUE_EX + "\nAll variables:", add_to_history=True) + self.out.side_by_side(" ", "File A", "File B", force_display_even_if_same=True) + + self._traverse_hierarchy() + self._print_summary() + + # Return the total number of differences; zero indicates no differences were found. + total_diff_count = sum( + [ + x["left"] + x["right"] + for x in [self.num_var_diffs, self.num_group_diffs, self.num_attribute_diffs] + ] + ) + + return total_diff_count + + def _traverse_hierarchy(self): + self.out.side_by_side( + "All Variables", " ", " ", dash_line=False, force_display_even_if_same=True + ) + self.out.side_by_side("-", "-", "-", dash_line=True, force_display_even_if_same=True) + + # Determine how the files will be opened. + file_opener = netCDF4.Dataset if self.file_types == "netcdf" else h5py.File + + # Open and go through files. + with ( + file_opener(self.file1.path, mode="r") as ds_a, + file_opener(self.file2.path, mode="r") as ds_b, + ): + self.open_file1 = ds_a + self.open_file2 = ds_b + + # Start with the Root Group, printing all the variables from it. + group_counter = 0 + self._print_group_details_side_by_side( + ds_a, + "/", + ds_b, + "/", + group_counter, + ) + group_counter += 1 + + for group_pair in self._dataset_pair_iterator( + "", + ds_a, + get_subgroups(ds_a, self.file_types), + "", + ds_b, + get_subgroups(ds_b, self.file_types), + ): + if group_pair.group_a_name == "": + self.num_group_diffs["right"] += 1 + elif group_pair.group_b_name == "": + self.num_group_diffs["left"] += 1 + else: + self.num_group_diffs["shared"] += 1 + + self._print_group_details_side_by_side( + group_pair.group_a, + group_pair.group_a_name, + group_pair.group_b, + group_pair.group_b_name, + group_counter, + ) + group_counter += 1 + + def _print_group_details_side_by_side( + self, + group_a: Union[netCDF4.Dataset, netCDF4.Group, h5py.Group], + group_a_name: str, + group_b: Union[netCDF4.Dataset, netCDF4.Group, h5py.Group], + group_b_name: str, + group_counter: int, + ) -> None: + """Align and display group details side by side.""" + self.out.side_by_side( + " ", " ", " ", dash_line=False, highlight_diff=False, force_display_even_if_same=True + ) + self.out.side_by_side( + f"GROUP #{group_counter:02}", + group_a_name.strip(), + group_b_name.strip(), + dash_line=True, + highlight_diff=False, + force_display_even_if_same=True, + ) + + # Count the number of variables in this group as long as this group exists. + vars_a_sorted: Union[list, str] = "" + vars_b_sorted: Union[list, str] = "" + if group_a: + vars_a_sorted = get_variables(group_a, self.file_types) + if group_b: + vars_b_sorted = get_variables(group_b, self.file_types) + self.out.side_by_side( + "num variables in group:", + len(vars_a_sorted), + len(vars_b_sorted), + highlight_diff=True, + force_display_even_if_same=True, + ) + self.out.side_by_side("-", "-", "-", dash_line=True, force_display_even_if_same=True) + + # Count differences between the lists of variables in this group. + left, right, shared = count_diffs(vars_a_sorted, vars_b_sorted) + self.num_var_diffs["left"] += left + self.num_var_diffs["right"] += right + self.num_var_diffs["shared"] += shared + + # Go through each variable in the current group. + for variable_pair in common_elements(vars_a_sorted, vars_b_sorted): + # Get and print the properties of each variable + self._print_var_properties_side_by_side( + self._create_var_properties( + group_a, variable_pair[1], original_dataset=self.open_file1 + ), + self._create_var_properties( + group_b, variable_pair[2], original_dataset=self.open_file1 + ), + ) + + def _print_var_properties_side_by_side( + self, + v_a: VarProperties, + v_b: VarProperties, + ) -> None: + """Align and display variable properties side by side.""" + # Gather all variable property pairs first, before printing, + # so we can decide whether to highlight the variable header. + pairs_to_check_and_show = [ + (v_a.dtype, v_b.dtype), + (v_a.dimensions, v_b.dimensions), + (v_a.shape, v_b.shape), + ] + if self.show_chunks: + pairs_to_check_and_show.append((v_a.chunking, v_b.chunking)) + if self.show_attributes: + for attr_a_key, attr_a, attr_b_key, attr_b in get_and_check_variable_attributes( + v_a, v_b + ): + # Check whether attr_a_key is empty, + # because it might be if the variable doesn't exist in File A. + pairs_to_check_and_show.append((attr_a, attr_b)) + # Scale Factor + scale_factor_pair = get_and_check_variable_scale_factor(v_a, v_b) + if scale_factor_pair: + pairs_to_check_and_show.append((scale_factor_pair[0], scale_factor_pair[1])) + + there_is_a_difference = False + for pair in pairs_to_check_and_show: + if pair[0] != pair[1]: + there_is_a_difference = True + break + + # If all attributes are the same, and keep-only-diffs is set -> DON'T print + # If all attributes are the same, and keep-only-diffs is NOT set -> print + # If some attributes are different -> print no matter else + if there_is_a_difference or (not self.out.keep_only_diffs): + self.out.side_by_side( + "-----VARIABLE-----:", + v_a.varname[:47], + v_b.varname[:47], + highlight_diff=False, + force_display_even_if_same=True, + ) + + # Go through each attribute, show differences, and add differences to running tally. + def _var_attribute_side_by_side(attribute_name, attribute_a, attribute_b): + diff_condition: SummaryDifferenceKeys = self.out.side_by_side( + f"{attribute_name}:", attribute_a, attribute_b, highlight_diff=True + ) + self.num_attribute_diffs[diff_condition] += 1 + if diff_condition in ("left", "right", "both"): + self.num_attribute_diffs["difference_types"].add(attribute_name) + + _var_attribute_side_by_side("dtype", v_a.dtype, v_b.dtype) + _var_attribute_side_by_side("dimensions", v_a.dimensions, v_b.dimensions) + _var_attribute_side_by_side("shape", v_a.shape, v_b.shape) + # Chunking + if self.show_chunks: + _var_attribute_side_by_side("chunksize", v_a.chunking, v_b.chunking) + # Scale Factor + scale_factor_pair = get_and_check_variable_scale_factor(v_a, v_b) + if scale_factor_pair: + _var_attribute_side_by_side("scale_factor", scale_factor_pair[0], scale_factor_pair[1]) + # Other attributes + if self.show_attributes: + for attr_a_key, attr_a, attr_b_key, attr_b in get_and_check_variable_attributes( + v_a, v_b + ): + # Check whether attr_a_key is empty, + # because it might be if the variable doesn't exist in File A. + attribute_key = attr_a_key if attr_a_key else attr_b_key + _var_attribute_side_by_side(attribute_key, attr_a, attr_b) + + def _print_root_dimensions(self): + # Show the dimensions of each file and evaluate differences. + self.out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Dimensions:", add_to_history=True) + list_a = get_root_dims(self.file1) + list_b = get_root_dims(self.file2) + _, _, _ = self.out.lists_diff(list_a, list_b) + + def _print_root_groups(self): + # Show the groups in each NetCDF file and evaluate differences. + self.out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Groups:", add_to_history=True) + list_a = get_root_groups(self.file1) + list_b = get_root_groups(self.file2) + _, _, _ = self.out.lists_diff(list_a, list_b) + + def _print_summary(self): + """Print summary counts of similarities and differences.""" + self.out.side_by_side("-", "-", "-", dash_line=True, force_display_even_if_same=True) + self.out.side_by_side("SUMMARY", "-", "-", dash_line=True, force_display_even_if_same=True) + + self.__print_summary_count_comparison_side_by_side("variable", self.num_var_diffs) + self.__print_summary_count_comparison_side_by_side("group", self.num_group_diffs) + self.__print_summary_count_comparison_side_by_side("attribute", self.num_attribute_diffs) + + if self.num_attribute_diffs["difference_types"]: + self.out.print( + Fore.LIGHTBLUE_EX + "\nDifferences were found in these attributes:", + add_to_history=True, + ) + self.out.print( + Fore.LIGHTBLUE_EX + f"\n{sorted(self.num_attribute_diffs['difference_types'])}", + add_to_history=True, + ) + + def __print_summary_count_comparison_side_by_side( + self, + item_type: str, + diff_dictionary: SummaryDifferencesDict, + ) -> None: + # Tally up instances where there were non-empty entries on both left and right sides. + diff_dictionary["left"] += diff_dictionary["both"] + diff_dictionary["right"] += diff_dictionary["both"] + + self.out.side_by_side( + f"Total # of shared {item_type}s:", + str(diff_dictionary["shared"]), + str(diff_dictionary["shared"]), + force_display_even_if_same=True, + ) + + self.out.side_by_side( + f"Total # of non-shared {item_type}s:", + str(diff_dictionary["left"]), + str(diff_dictionary["right"]), + force_display_even_if_same=True, + ) + + def _dataset_pair_iterator( + self, + node_a_name: str, + node_a: Union[netCDF4.Dataset, netCDF4.Group, h5py.Dataset, h5py.Group], + node_a_subgroups: list, + node_b_name: str, + node_b: Union[netCDF4.Dataset, netCDF4.Group, h5py.Dataset, h5py.Group], + node_b_subgroups: list, + ) -> Iterator[GroupPair]: + """Yield names and groups, as pairs, from two netCDF or HDF hierarchies. + + Parameters + ---------- + node_a_name + name of the first group or dataset + node_a + the first group or dataset + node_b_name + name of the second group or dataset + node_b + the second group or dataset + + Yields + ------ + tuple + group A name : str + group A object : netCDF4.Group or None + group B name : str + group B object : netCDF4.Group or None + """ + # get a sorted list of subgroups from both node_a and node_b + for _, group_a_name, group_b_name in common_elements( + node_a_subgroups if node_a is not None else "", + node_b_subgroups if node_b is not None else "", + ): + yield GroupPair( + group_a_name=node_a_name + "/" + group_a_name if group_a_name else "", + group_a=( + node_a[group_a_name] + if (group_a_name and (group_a_name in node_a_subgroups)) + else None + ), + group_b_name=node_b_name + "/" + group_b_name if group_b_name else "", + group_b=( + node_b[group_b_name] + if (group_b_name and (group_b_name in node_b_subgroups)) + else None + ), + ) + + for _, subgroup_a_name, subgroup_b_name in common_elements( + node_a_subgroups if node_a is not None else "", + node_b_subgroups if node_b is not None else "", + ): + subnode_a_name = node_a_name + "/" + subgroup_a_name if subgroup_a_name else "" + subnode_a = ( + node_a[subgroup_a_name] + if (subgroup_a_name and (subgroup_a_name in node_a_subgroups)) + else None + ) + subnode_a_subgroups = get_subgroups(subnode_a, file_type=self.file_types) + + subnode_b_name = node_a_name + "/" + subgroup_b_name if subgroup_b_name else "" + subnode_b = ( + node_b[subgroup_b_name] + if (subgroup_b_name and (subgroup_b_name in node_b_subgroups)) + else None + ) + subnode_b_subgroups = get_subgroups(subnode_b, file_type=self.file_types) + + yield from self._dataset_pair_iterator( + subnode_a_name, + subnode_a, + subnode_a_subgroups, + subnode_b_name, + subnode_b, + subnode_b_subgroups, + ) + + def _create_var_properties( + self, group: Union[netCDF4.Dataset, netCDF4.Group], varname: str, original_dataset + ) -> VarProperties: + """Get the properties of a variable. + + Parameters + ---------- + group + a dataset or group of variables + varname + the name of the variable + + Returns + ------- + VarProperties + """ + if varname: + if self.file_types == "netcdf": + the_variable = group.variables[varname] + elif self.file_types == "hdf5": + the_variable = group[varname] + + v_dtype = str(the_variable.dtype) + + if self.file_types == "netcdf": + v_dimensions = str(the_variable.dimensions) + elif self.file_types == "hdf5": + v_dimensions = str([dim.label for dim in the_variable.dims]) + + v_shape = str(the_variable.shape).strip() + + if self.file_types == "netcdf": + v_chunking = str(the_variable.chunking()).strip() + elif self.file_types == "hdf5": + v_chunking = str(the_variable.chunks) + + def __name_from_h5_ref(ref): + return original_dataset[ref].name + + v_attributes = {} + if self.file_types == "netcdf": + for name in the_variable.ncattrs(): + try: + retrieved_value = the_variable.getncattr(name) + except KeyError as key_err: + # Added this check because of "unsupported datatype" error that prevented + # fully running comparisons on S5P_OFFL_L1B_IR_UVN collections. + retrieved_value = f"netCDF error: {str(key_err)}" + + v_attributes[name] = retrieved_value + elif self.file_types == "hdf5": + for name in the_variable.attrs.keys(): + attribute_value = the_variable.attrs[name] + if isinstance(attribute_value, np.ndarray): + if attribute_value.dtype == h5py.ref_dtype: + retrieved_value = __name_from_h5_ref(attribute_value[0][0]) + else: + try: + retrieved_value = str( + [__name_from_h5_ref(a[0]) for a in attribute_value] + ) + except IndexError: + retrieved_value = str(attribute_value) + + else: + retrieved_value = str(attribute_value) + + v_attributes[name] = retrieved_value + else: + the_variable = None + v_dtype = "" + v_dimensions = "" + v_shape = "" + v_chunking = "" + v_attributes = None + + return VarProperties( + varname, the_variable, v_dtype, v_dimensions, v_shape, v_chunking, v_attributes + ) diff --git a/ncompare/console.py b/ncompare/console.py index f1fc811..764099a 100755 --- a/ncompare/console.py +++ b/ncompare/console.py @@ -49,8 +49,8 @@ def _cli(args: Optional[Sequence[str]]) -> argparse.Namespace: parser = argparse.ArgumentParser( description="Compare the variables contained within two different netCDF datasets" ) - parser.add_argument("file_a", help="First (netCDF or HDF) file") - parser.add_argument("file_b", help="Second (netCDF or HDF) file") + parser.add_argument("path_a", help="First (netCDF or HDF) file") + parser.add_argument("path_b", help="Second (netCDF or HDF) file") parser.add_argument( "--only-diffs", action="store_true", diff --git a/ncompare/core.py b/ncompare/core.py index 624dd40..c404188 100644 --- a/ncompare/core.py +++ b/ncompare/core.py @@ -29,30 +29,16 @@ """Compare the structure of two netCDF or HDF files.""" -from collections.abc import Iterator from pathlib import Path from typing import Optional, Union -import netCDF4 -from colorama import Fore - -from ncompare.core_types import ( - FileToCompare, - GroupPair, - SummaryDifferencesDict, - VarProperties, - valid_file_type_ids, -) -from ncompare.getters import ( - get_and_check_variable_attributes, - get_and_check_variable_scale_factor, - get_root_dims, - get_root_groups, - get_var_properties, +from ncompare.Comparison import Comparison +from ncompare.path_and_string_operations import ( + ensure_valid_path_exists, + ensure_valid_path_with_suffix, + validate_file_type, ) -from ncompare.printing import Outputter, SummaryDifferenceKeys -from ncompare.sequence_operations import common_elements, count_diffs -from ncompare.utils import ensure_valid_path_exists, ensure_valid_path_with_suffix +from ncompare.printing import Outputter def compare( @@ -108,8 +94,8 @@ def compare( file_xlsx = ensure_valid_path_with_suffix(file_xlsx, ".xlsx") # Check the validity of file types - file_a = _validate_file_type(path_a) - file_b = _validate_file_type(path_b) + file_a = validate_file_type(path_a) + file_b = validate_file_type(path_b) if file_a.type != file_b.type: # I'm not sure if there is a use-case where we'd want to compare a netCDF with an HDF file? # This assumption of files being the same type, affects the rest of the comparison logic. @@ -127,381 +113,17 @@ def compare( out.print(f"File B: {file_b.path}") # Start the comparison process. - total_diff_count = _run_through_comparisons( - out, file_a, file_b, show_chunks=show_chunks, show_attributes=show_attributes + comparison = Comparison( + file_a, file_b, out, show_chunks=show_chunks, show_attributes=show_attributes ) + total_diff_count = comparison.run_through_comparisons() # Write to CSV and Excel files. if file_csv: - out.write_history_to_csv(filename=file_csv) + comparison.out.write_history_to_csv(filename=file_csv) if file_xlsx: - out.write_history_to_excel(filename=file_xlsx) + comparison.out.write_history_to_excel(filename=file_xlsx) - out.print("\nDone.", colors=False) + comparison.out.print("\nDone.", colors=False) return total_diff_count - - -def _run_through_comparisons( - out: Outputter, - file_a: FileToCompare, - file_b: FileToCompare, - show_chunks: bool, - show_attributes: bool, -) -> int: - """Execute a series of comparisons between two netCDF or HDF files. - - Parameters - ---------- - out - instance of Outputter - file_a - path and type (netCDF or HDF5) of the first file - file_b - path and type (netCDF or HDF5) of the second file - show_chunks - whether to include data chunk sizes in the displayed comparison of variables - show_attributes - whether to include variable attributes in the displayed comparison of variables - - Returns - ------- - int - total number of differences found (across variables, groups, and attributes) - """ - # Show the dimensions of each file and evaluate differences. - out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Dimensions:", add_to_history=True) - list_a = get_root_dims(file_a) - list_b = get_root_dims(file_b) - _, _, _ = out.lists_diff(list_a, list_b) - - # Show the groups in each NetCDF file and evaluate differences. - out.print(Fore.LIGHTBLUE_EX + "\nRoot-level Groups:", add_to_history=True) - list_a = get_root_groups(file_a) - list_b = get_root_groups(file_b) - _, _, _ = out.lists_diff(list_a, list_b) - - # Run through all the rest of the groups and variables, tallying differences along the way. - out.print(Fore.LIGHTBLUE_EX + "\nAll variables:", add_to_history=True) - out.side_by_side(" ", "File A", "File B", force_display_even_if_same=True) - num_group_diffs: SummaryDifferencesDict = { - "shared": 0, - "left": 0, - "right": 0, - "both": 0, - "difference_types": set(), - } - num_var_diffs: SummaryDifferencesDict = { - "shared": 0, - "left": 0, - "right": 0, - "both": 0, - "difference_types": set(), - } - num_attribute_diffs: SummaryDifferencesDict = { - "shared": 0, - "left": 0, - "right": 0, - "both": 0, - "difference_types": set(), - } - with netCDF4.Dataset(file_a.path) as nc_a, netCDF4.Dataset(file_b.path) as nc_b: - out.side_by_side( - "All Variables", " ", " ", dash_line=False, force_display_even_if_same=True - ) - out.side_by_side("-", "-", "-", dash_line=True, force_display_even_if_same=True) - - group_counter = 0 - _print_group_details_side_by_side( - out, - nc_a, - "/", - nc_b, - "/", - group_counter, - num_var_diffs, - num_attribute_diffs, - show_attributes, - show_chunks, - ) - group_counter += 1 - - for group_pair in _walk_common_groups_tree("", nc_a, "", nc_b): - if group_pair.group_a_name == "": - num_group_diffs["right"] += 1 - elif group_pair.group_b_name == "": - num_group_diffs["left"] += 1 - else: - num_group_diffs["shared"] += 1 - - _print_group_details_side_by_side( - out, - group_pair.group_a, - group_pair.group_a_name, - group_pair.group_b, - group_pair.group_b_name, - group_counter, - num_var_diffs, - num_attribute_diffs, - show_attributes, - show_chunks, - ) - group_counter += 1 - - # Print summary counts of similarities and differences at the end of the comparison report. - out.side_by_side("-", "-", "-", dash_line=True, force_display_even_if_same=True) - out.side_by_side("SUMMARY", "-", "-", dash_line=True, force_display_even_if_same=True) - _print_summary_count_comparison_side_by_side(out, "variable", num_var_diffs) - _print_summary_count_comparison_side_by_side(out, "group", num_group_diffs) - _print_summary_count_comparison_side_by_side(out, "attribute", num_attribute_diffs) - if num_attribute_diffs["difference_types"]: - out.print( - Fore.LIGHTBLUE_EX + "\nDifferences were found in these attributes:", add_to_history=True - ) - out.print( - Fore.LIGHTBLUE_EX + f"\n{sorted(num_attribute_diffs['difference_types'])}", - add_to_history=True, - ) - - # Return the total number of differences; zero indicates no differences were found. - total_diff_count = sum( - [x["left"] + x["right"] for x in [num_var_diffs, num_group_diffs, num_attribute_diffs]] - ) - - return total_diff_count - - -def _walk_common_groups_tree( - top_a_name: str, - top_a: Union[netCDF4.Dataset, netCDF4.Group], - top_b_name: str, - top_b: Union[netCDF4.Dataset, netCDF4.Group], -) -> Iterator[GroupPair]: - """Yield names and groups from a netCDF4 or HDF's group tree. - - Parameters - ---------- - top_a_name - name of the first group or dataset - top_a - the first group or dataset - top_b_name - name of the second group or dataset - top_b - the second group or dataset - - Yields - ------ - tuple - group A name : str - group A object : netCDF4.Group or None - group B name : str - group B object : netCDF4.Group or None - """ - for _, group_a_name, group_b_name in common_elements( - top_a.groups if top_a is not None else "", - top_b.groups if top_b is not None else "", - ): - yield GroupPair( - group_a_name=top_a_name + "/" + group_a_name if group_a_name else "", - group_a=( - top_a[group_a_name] if (group_a_name and (group_a_name in top_a.groups)) else None - ), - group_b_name=top_b_name + "/" + group_b_name if group_b_name else "", - group_b=( - top_b[group_b_name] if (group_b_name and (group_b_name in top_b.groups)) else None - ), - ) - - for _, subgroup_a_name, subgroup_b_name in common_elements( - top_a.groups if top_a is not None else "", - top_b.groups if top_b is not None else "", - ): - yield from _walk_common_groups_tree( - top_a_name + "/" + subgroup_a_name if subgroup_a_name else "", - ( - top_a[subgroup_a_name] - if (subgroup_a_name and (subgroup_a_name in top_a.groups)) - else None - ), - top_a_name + "/" + subgroup_b_name if subgroup_b_name else "", - ( - top_b[subgroup_b_name] - if (subgroup_b_name and (subgroup_b_name in top_b.groups)) - else None - ), - ) - - -def _print_summary_count_comparison_side_by_side( - out: Outputter, - item_type: str, - diff_dictionary: SummaryDifferencesDict, -) -> None: - # Tally up instances where there were non-empty entries on both left and right sides. - diff_dictionary["left"] += diff_dictionary["both"] - diff_dictionary["right"] += diff_dictionary["both"] - - out.side_by_side( - f"Total # of shared {item_type}s:", - str(diff_dictionary["shared"]), - str(diff_dictionary["shared"]), - force_display_even_if_same=True, - ) - - out.side_by_side( - f"Total # of non-shared {item_type}s:", - str(diff_dictionary["left"]), - str(diff_dictionary["right"]), - force_display_even_if_same=True, - ) - - -def _print_group_details_side_by_side( - out: Outputter, - group_a: Union[netCDF4.Dataset, netCDF4.Group], - group_a_name: str, - group_b: Union[netCDF4.Dataset, netCDF4.Group], - group_b_name: str, - group_counter: int, - num_var_diffs: SummaryDifferencesDict, - num_attribute_diffs: SummaryDifferencesDict, - show_attributes: bool, - show_chunks: bool, -) -> None: - """Align and display group details side by side.""" - out.side_by_side( - " ", " ", " ", dash_line=False, highlight_diff=False, force_display_even_if_same=True - ) - out.side_by_side( - f"GROUP #{group_counter:02}", - group_a_name.strip(), - group_b_name.strip(), - dash_line=True, - highlight_diff=False, - force_display_even_if_same=True, - ) - - # Count the number of variables in this group as long as this group exists. - vars_a_sorted: Union[list, str] = "" - vars_b_sorted: Union[list, str] = "" - if group_a: - vars_a_sorted = sorted(group_a.variables) - if group_b: - vars_b_sorted = sorted(group_b.variables) - out.side_by_side( - "num variables in group:", - len(vars_a_sorted), - len(vars_b_sorted), - highlight_diff=True, - force_display_even_if_same=True, - ) - out.side_by_side("-", "-", "-", dash_line=True, force_display_even_if_same=True) - - # Count differences between the lists of variables in this group. - left, right, shared = count_diffs(vars_a_sorted, vars_b_sorted) - num_var_diffs["left"] += left - num_var_diffs["right"] += right - num_var_diffs["shared"] += shared - - # Go through each variable in the current group. - for variable_pair in common_elements(vars_a_sorted, vars_b_sorted): - # Get and print the properties of each variable - _print_var_properties_side_by_side( - out, - get_var_properties(group_a, variable_pair[1]), - get_var_properties(group_b, variable_pair[2]), - num_attribute_diffs, - show_chunks=show_chunks, - show_attributes=show_attributes, - ) - - -def _print_var_properties_side_by_side( - out: Outputter, - v_a: VarProperties, - v_b: VarProperties, - num_attribute_diffs: SummaryDifferencesDict, - show_chunks: bool = False, - show_attributes: bool = False, -) -> None: - """Align and display variable properties side by side.""" - # Gather all variable property pairs first, before printing, - # so we can decide whether to highlight the variable header. - pairs_to_check_and_show = [ - (v_a.dtype, v_b.dtype), - (v_a.dimensions, v_b.dimensions), - (v_a.shape, v_b.shape), - ] - if show_chunks: - pairs_to_check_and_show.append((v_a.chunking, v_b.chunking)) - if show_attributes: - for attr_a_key, attr_a, attr_b_key, attr_b in get_and_check_variable_attributes(v_a, v_b): - # Check whether attr_a_key is empty, - # because it might be if the variable doesn't exist in File A. - pairs_to_check_and_show.append((attr_a, attr_b)) - # Scale Factor - scale_factor_pair = get_and_check_variable_scale_factor(v_a, v_b) - if scale_factor_pair: - pairs_to_check_and_show.append((scale_factor_pair[0], scale_factor_pair[1])) - - there_is_a_difference = False - for pair in pairs_to_check_and_show: - if pair[0] != pair[1]: - there_is_a_difference = True - break - - # If all attributes are the same, and keep-only-diffs is set -> DON'T print - # If all attributes are the same, and keep-only-diffs is NOT set -> print - # If some attributes are different -> print no matter else - if there_is_a_difference or (not out.keep_only_diffs): - out.side_by_side( - "-----VARIABLE-----:", - v_a.varname[:47], - v_b.varname[:47], - highlight_diff=False, - force_display_even_if_same=True, - ) - - # Go through each attribute, show differences, and add differences to running tally. - def _var_attribute_side_by_side(attribute_name, attribute_a, attribute_b): - diff_condition: SummaryDifferenceKeys = out.side_by_side( - f"{attribute_name}:", attribute_a, attribute_b, highlight_diff=True - ) - num_attribute_diffs[diff_condition] += 1 - if diff_condition in ("left", "right", "both"): - num_attribute_diffs["difference_types"].add(attribute_name) - - _var_attribute_side_by_side("dtype", v_a.dtype, v_b.dtype) - _var_attribute_side_by_side("dimensions", v_a.dimensions, v_b.dimensions) - _var_attribute_side_by_side("shape", v_a.shape, v_b.shape) - # Chunking - if show_chunks: - _var_attribute_side_by_side("chunksize", v_a.chunking, v_b.chunking) - # Scale Factor - scale_factor_pair = get_and_check_variable_scale_factor(v_a, v_b) - if scale_factor_pair: - _var_attribute_side_by_side("scale_factor", scale_factor_pair[0], scale_factor_pair[1]) - # Other attributes - if show_attributes: - for attr_a_key, attr_a, attr_b_key, attr_b in get_and_check_variable_attributes(v_a, v_b): - # Check whether attr_a_key is empty, - # because it might be if the variable doesn't exist in File A. - attribute_key = attr_a_key if attr_a_key else attr_b_key - _var_attribute_side_by_side(attribute_key, attr_a, attr_b) - - -def _validate_file_type(file_path: Path) -> FileToCompare: - """Validate a file type and return a FileToCompare instance.""" - if file_path.suffix in (".h5", ".hdf", ".hdf5"): - file_type: valid_file_type_ids = "hdf5" - elif file_path.suffix in (".nc", ".nc4", ".nc3"): - file_type = "netcdf" - else: - raise TypeError( - f"{file_path.suffix} is not a valid file type. " - f"Expected a netcdf ('.nc', '.nc4', '.nc3') or " - f"hdf5 ('.h5', '.hdf', '.hdf5')." - ) - - return FileToCompare(path=file_path, type=file_type) diff --git a/ncompare/getters.py b/ncompare/getters.py index a109b00..090deff 100644 --- a/ncompare/getters.py +++ b/ncompare/getters.py @@ -6,8 +6,8 @@ import netCDF4 import xarray as xr -from ncompare.core_types import FileToCompare, VarProperties from ncompare.sequence_operations import common_elements +from ncompare.utility_types import FileToCompare, VarProperties def get_and_check_variable_scale_factor( @@ -41,48 +41,6 @@ def get_and_check_variable_attributes( yield attr_a_key, attr_a, attr_b_key, attr_b -def get_var_properties(group: Union[netCDF4.Dataset, netCDF4.Group], varname: str) -> VarProperties: - """Get the properties of a variable. - - Parameters - ---------- - group - a dataset or group of variables - varname - the name of the variable - - Returns - ------- - VarProperties - """ - if varname: - the_variable = group.variables[varname] - v_dtype = str(the_variable.dtype) - v_dimensions = str(the_variable.dimensions) - v_shape = str(the_variable.shape).strip() - v_chunking = str(the_variable.chunking()).strip() - - v_attributes = {} - for name in the_variable.ncattrs(): - try: - v_attributes[name] = the_variable.getncattr(name) - except KeyError as key_err: - # Added this check because of "unsupported datatype" error that prevented - # fully running comparisons on S5P_OFFL_L1B_IR_UVN collections. - v_attributes[name] = f"netCDF error: {str(key_err)}" - else: - the_variable = None - v_dtype = "" - v_dimensions = "" - v_shape = "" - v_chunking = "" - v_attributes = None - - return VarProperties( - varname, the_variable, v_dtype, v_dimensions, v_shape, v_chunking, v_attributes - ) - - def get_attribute_value_as_str(varprops: VarProperties, attribute_key: str) -> str: """Get a string representation of the attribute value.""" if attribute_key and (attribute_key in varprops.attributes): @@ -110,6 +68,35 @@ def get_root_groups(file: FileToCompare) -> list: return groups_list +def get_subgroups(node: Union[netCDF4.Dataset, netCDF4.Group, h5py.Group], file_type: str) -> list: + """Get a list of subgroups from a netCDF or HDF5 group. + + Parameters + ---------- + node + file_type + + Returns + ------- + list + subgroups under the node + """ + if file_type == "hdf5": + return [key for key in node.keys() if isinstance(node[key], h5py.Group)] + else: # should be "netcdf" + return list(node.groups) + + +def get_variables(node: Union[netCDF4.Dataset, netCDF4.Group, h5py.Group], file_type: str) -> list: + """Get a sorted list of variables from a netCDF or HDF5 group.""" + if file_type == "hdf5": + return [key for key in node.keys() if isinstance(node[key], h5py.Dataset)] + elif file_type == "netcdf": + return sorted(node.variables) + else: + raise RuntimeError(f"Unsupported file type: {file_type}") + + def get_root_dims(file: FileToCompare) -> list: """Get a list of dimensions from a netCDF or HDF5.""" diff --git a/ncompare/utils.py b/ncompare/path_and_string_operations.py similarity index 79% rename from ncompare/utils.py rename to ncompare/path_and_string_operations.py index 7ad79db..cb53dce 100644 --- a/ncompare/utils.py +++ b/ncompare/path_and_string_operations.py @@ -28,6 +28,8 @@ from pathlib import Path from typing import Union +from ncompare.utility_types import FileToCompare, valid_file_type_ids + def ensure_valid_path_exists(should_be_path: Union[str, Path]) -> Path: """Coerce input to a pathlib.Path and check that the resulting filepath exists.""" @@ -49,3 +51,19 @@ def coerce_to_str(some_object: Union[str, int, tuple]) -> str: if isinstance(some_object, (str, int, tuple)): return str(some_object) raise TypeError(f"Unable to coerce value to str. Unexpected type <{type(some_object)}>.") + + +def validate_file_type(file_path: Path) -> FileToCompare: + """Validate a file type and return a FileToCompare instance.""" + if file_path.suffix in (".h5", ".hdf", ".hdf5"): + file_type: valid_file_type_ids = "hdf5" + elif file_path.suffix in (".nc", ".nc4", ".nc3"): + file_type = "netcdf" + else: + raise TypeError( + f"{file_path.suffix} is not a valid file type. " + f"Expected a netcdf ('.nc', '.nc4', '.nc3') or " + f"hdf5 ('.h5', '.hdf', '.hdf5')." + ) + + return FileToCompare(path=file_path, type=file_type) diff --git a/ncompare/printing.py b/ncompare/printing.py index 5224b30..b031d87 100644 --- a/ncompare/printing.py +++ b/ncompare/printing.py @@ -31,7 +31,7 @@ import warnings from collections.abc import Iterable, Iterator from pathlib import Path -from typing import Literal, Optional, TextIO, Union +from typing import Optional, TextIO, Union import colorama import openpyxl @@ -40,8 +40,7 @@ from openpyxl.styles import Font from ncompare.sequence_operations import common_elements, count_diffs - -SummaryDifferenceKeys = Literal["shared", "left", "right", "both"] +from ncompare.utility_types import SummaryDifferenceKeys # Set up regex remover of ANSI color escape sequences # From diff --git a/ncompare/sequence_operations.py b/ncompare/sequence_operations.py index c673f90..9a9c998 100644 --- a/ncompare/sequence_operations.py +++ b/ncompare/sequence_operations.py @@ -28,17 +28,18 @@ from collections.abc import Generator, Iterable from typing import Union -from ncompare.utils import coerce_to_str +from ncompare.path_and_string_operations import coerce_to_str def common_elements( sequence_a: Iterable, sequence_b: Iterable ) -> Generator[tuple[int, str, str], None, None]: - """Loop over combined items of two iterables, and yield aligned item pairs. + """Yield all items from two iterables, sorted and as aligned pairs. Note ---- - When there isn't a matching item, an empty string is used instead. + When there isn't a matching item for one of the iterables, + an empty string is used instead. Yields ------ diff --git a/ncompare/core_types.py b/ncompare/utility_types.py similarity index 83% rename from ncompare/core_types.py rename to ncompare/utility_types.py index 22ad5b6..7c9f218 100644 --- a/ncompare/core_types.py +++ b/ncompare/utility_types.py @@ -3,21 +3,13 @@ from pathlib import Path from typing import Literal, TypedDict, Union -VarProperties = namedtuple( - "VarProperties", "varname, variable, dtype, dimensions, shape, chunking, attributes" -) - -GroupPair = namedtuple( - "GroupPair", - "group_a_name group_a group_b_name group_b", - defaults=("", None, "", None), -) - valid_file_type_ids = Literal["netcdf", "hdf5"] @dataclass class FileToCompare: + """Represents an input file to compare against, and its file type.""" + path: Union[Path, str] type: valid_file_type_ids = "netcdf" @@ -33,8 +25,23 @@ def __str__(self): class SummaryDifferencesDict(TypedDict): + """Represents the number and type of differences between two files.""" + shared: int left: int right: int both: int difference_types: set + + +SummaryDifferenceKeys = Literal["shared", "left", "right", "both"] + +VarProperties = namedtuple( + "VarProperties", "varname, variable, dtype, dimensions, shape, chunking, attributes" +) + +GroupPair = namedtuple( + "GroupPair", + "group_a_name group_a group_b_name group_b", + defaults=("", None, "", None), +) diff --git a/tests/test_cli.py b/tests/test_cli.py index 09bea7a..9c81308 100644 --- a/tests/test_cli.py +++ b/tests/test_cli.py @@ -41,8 +41,8 @@ def test_console_help(): def test_arg_parser(): parsed = _cli(["first_netcdf.nc", "second_netcdf.nc"]) - assert getattr(parsed, "file_a") == "first_netcdf.nc" - assert getattr(parsed, "file_b") == "second_netcdf.nc" + assert getattr(parsed, "path_a") == "first_netcdf.nc" + assert getattr(parsed, "path_b") == "second_netcdf.nc" assert getattr(parsed, "show_attributes") is False assert getattr(parsed, "show_chunks") is False assert getattr(parsed, "only_diffs") is False diff --git a/tests/test_getters.py b/tests/test_getters.py index d5e8dd2..0c2e1ae 100644 --- a/tests/test_getters.py +++ b/tests/test_getters.py @@ -1,21 +1,16 @@ -import netCDF4 as nc - -from ncompare.getters import get_and_check_variable_scale_factor, get_var_properties - - -def test_var_properties(ds_3dims_3vars_4coords_1group): - with nc.Dataset(ds_3dims_3vars_4coords_1group) as ds: - result = get_var_properties(ds.groups["Group1"], varname="step") - assert result.varname == "step" - assert result.dtype == "float32" - assert result.shape == "(3,)" - assert result.chunking == "contiguous" - assert result.attributes == {"add_offset": 5, "scale_factor": 0.5} - - -def test_get_scale_factor(ds_3dims_3vars_4coords_1group): - with nc.Dataset(ds_3dims_3vars_4coords_1group) as ds: - step_varProps = get_var_properties(ds.groups["Group1"], varname="step") - - result = get_and_check_variable_scale_factor(step_varProps, step_varProps) - assert result == ("0.5", "0.5") +# def test_var_properties(ds_3dims_3vars_4coords_1group): +# with nc.Dataset(ds_3dims_3vars_4coords_1group) as ds: +# result = get_var_properties(ds.groups["Group1"], varname="step", file_type="netcdf") +# assert result.varname == "step" +# assert result.dtype == "float32" +# assert result.shape == "(3,)" +# assert result.chunking == "contiguous" +# assert result.attributes == {"add_offset": 5, "scale_factor": 0.5} +# +# +# def test_get_scale_factor(ds_3dims_3vars_4coords_1group): +# with nc.Dataset(ds_3dims_3vars_4coords_1group) as ds: +# step_varProps = get_var_properties(ds.groups["Group1"], varname="step", file_type="netcdf") +# +# result = get_and_check_variable_scale_factor(step_varProps, step_varProps) +# assert result == ("0.5", "0.5") diff --git a/tests/test_utils.py b/tests/test_path_and_string_operations.py similarity index 96% rename from tests/test_utils.py rename to tests/test_path_and_string_operations.py index b87f535..2955d2d 100644 --- a/tests/test_utils.py +++ b/tests/test_path_and_string_operations.py @@ -27,7 +27,7 @@ import pytest -from ncompare.utils import coerce_to_str, ensure_valid_path_exists +from ncompare.path_and_string_operations import coerce_to_str, ensure_valid_path_exists def test_make_valid_path_with_simple_invalid_str_path(): From d6710a64f90d41da50b8fa8236f82e5a922519af Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Thu, 2 Jan 2025 16:56:45 -0500 Subject: [PATCH 15/19] fix None type check and assert number --- ncompare/getters.py | 10 +++++++--- tests/test_core.py | 2 +- 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/ncompare/getters.py b/ncompare/getters.py index 090deff..58309e2 100644 --- a/ncompare/getters.py +++ b/ncompare/getters.py @@ -78,13 +78,17 @@ def get_subgroups(node: Union[netCDF4.Dataset, netCDF4.Group, h5py.Group], file_ Returns ------- - list + list or None subgroups under the node """ - if file_type == "hdf5": + if node is None: + return [] + elif file_type == "hdf5": return [key for key in node.keys() if isinstance(node[key], h5py.Group)] - else: # should be "netcdf" + elif file_type == "netcdf": # should be "netcdf" return list(node.groups) + else: + return [] def get_variables(node: Union[netCDF4.Dataset, netCDF4.Group, h5py.Group], file_type: str) -> list: diff --git a/tests/test_core.py b/tests/test_core.py index 2b2c95f..da2d0a3 100644 --- a/tests/test_core.py +++ b/tests/test_core.py @@ -84,4 +84,4 @@ def test_icesat(temp_data_dir): file_text=str(out_path), ) - assert num_differences == 4982 + assert num_differences == 5280 From d3c5d7f738f7445f53ab40198d63867990667a9b Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Fri, 3 Jan 2025 11:04:39 -0500 Subject: [PATCH 16/19] simplify logic and tests --- ncompare/getters.py | 8 ++------ ncompare/utility_types.py | 5 +---- tests/test_path_and_string_operations.py | 11 ++++++++++- tests/test_utility_types.py | 13 +++++++++++++ 4 files changed, 26 insertions(+), 11 deletions(-) create mode 100644 tests/test_utility_types.py diff --git a/ncompare/getters.py b/ncompare/getters.py index 58309e2..9996759 100644 --- a/ncompare/getters.py +++ b/ncompare/getters.py @@ -85,20 +85,16 @@ def get_subgroups(node: Union[netCDF4.Dataset, netCDF4.Group, h5py.Group], file_ return [] elif file_type == "hdf5": return [key for key in node.keys() if isinstance(node[key], h5py.Group)] - elif file_type == "netcdf": # should be "netcdf" + else: # should be "netcdf" return list(node.groups) - else: - return [] def get_variables(node: Union[netCDF4.Dataset, netCDF4.Group, h5py.Group], file_type: str) -> list: """Get a sorted list of variables from a netCDF or HDF5 group.""" if file_type == "hdf5": return [key for key in node.keys() if isinstance(node[key], h5py.Dataset)] - elif file_type == "netcdf": + else: # should be "netcdf" return sorted(node.variables) - else: - raise RuntimeError(f"Unsupported file type: {file_type}") def get_root_dims(file: FileToCompare) -> list: diff --git a/ncompare/utility_types.py b/ncompare/utility_types.py index 7c9f218..7ac94b0 100644 --- a/ncompare/utility_types.py +++ b/ncompare/utility_types.py @@ -16,13 +16,10 @@ class FileToCompare: def __post_init__(self): # We'll validate the inputs here. if not isinstance(self.path, (str, Path)): - raise ValueError(f"'path' must be a str or Path, was {type(self.path)}") + raise TypeError(f"'path' must be a str or Path, was {type(self.path)}") if self.type not in ("netcdf", "hdf5"): raise ValueError("'type' must be either 'netcdf' or 'hdf5'") - def __str__(self): - return f"path: {self.path} is considered a {self.type} file" - class SummaryDifferencesDict(TypedDict): """Represents the number and type of differences between two files.""" diff --git a/tests/test_path_and_string_operations.py b/tests/test_path_and_string_operations.py index 2955d2d..4b476e7 100644 --- a/tests/test_path_and_string_operations.py +++ b/tests/test_path_and_string_operations.py @@ -27,7 +27,11 @@ import pytest -from ncompare.path_and_string_operations import coerce_to_str, ensure_valid_path_exists +from ncompare.path_and_string_operations import ( + coerce_to_str, + ensure_valid_path_exists, + validate_file_type, +) def test_make_valid_path_with_simple_invalid_str_path(): @@ -50,6 +54,11 @@ def test_make_valid_path_from_Path_in_repo(): assert isinstance(returnval, Path) +def test_validate_file_type(): + with pytest.raises(TypeError): + validate_file_type(Path(__file__)) + + def test_error_from_wrong_path_type(): with pytest.raises(TypeError): ensure_valid_path_exists((0, 1)) diff --git a/tests/test_utility_types.py b/tests/test_utility_types.py new file mode 100644 index 0000000..a5fe1d4 --- /dev/null +++ b/tests/test_utility_types.py @@ -0,0 +1,13 @@ +from pathlib import Path + +import pytest + +from ncompare.utility_types import FileToCompare + + +def test_FileToCompare(): + with pytest.raises(TypeError): + assert FileToCompare(path=123, type="netcdf") + + with pytest.raises(ValueError): + assert FileToCompare(path=Path(__file__), type="beebop_type") From 1a9ff8d173dfbac422e60cbf5e398ffb3515f29b Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Fri, 3 Jan 2025 15:31:25 -0500 Subject: [PATCH 17/19] add test for file type mismatch --- tests/test_core.py | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/tests/test_core.py b/tests/test_core.py index da2d0a3..ca8e30c 100644 --- a/tests/test_core.py +++ b/tests/test_core.py @@ -31,6 +31,8 @@ from contextlib import nullcontext as does_not_raise +import pytest + from ncompare.core import compare from . import data_for_tests_dir @@ -85,3 +87,11 @@ def test_icesat(temp_data_dir): ) assert num_differences == 5280 + + +def test_error_on_different_file_types(temp_data_dir): + file1 = data_for_tests_dir / "icesat-2-ATL06" / "ATL06_20230816161508_08782002_006_02.h5" + file2 = data_for_tests_dir / "test_a.nc" + + with pytest.raises(TypeError): + compare(file1, file2) From dc9415dbe716ecec0a97bb9ca60aa34b19e6de24 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Mon, 6 Jan 2025 09:33:48 -0500 Subject: [PATCH 18/19] include .he5 in hdf5 file extension check and use lower() --- ncompare/path_and_string_operations.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/ncompare/path_and_string_operations.py b/ncompare/path_and_string_operations.py index cb53dce..fbc3e05 100644 --- a/ncompare/path_and_string_operations.py +++ b/ncompare/path_and_string_operations.py @@ -55,15 +55,15 @@ def coerce_to_str(some_object: Union[str, int, tuple]) -> str: def validate_file_type(file_path: Path) -> FileToCompare: """Validate a file type and return a FileToCompare instance.""" - if file_path.suffix in (".h5", ".hdf", ".hdf5"): + if file_path.suffix.lower() in (".h5", ".hdf5", ".he5"): file_type: valid_file_type_ids = "hdf5" - elif file_path.suffix in (".nc", ".nc4", ".nc3"): + elif file_path.suffix.lower() in (".nc", ".nc4", ".nc3"): file_type = "netcdf" else: raise TypeError( f"{file_path.suffix} is not a valid file type. " f"Expected a netcdf ('.nc', '.nc4', '.nc3') or " - f"hdf5 ('.h5', '.hdf', '.hdf5')." + f"hdf5 ('.h5', '.hdf5', '.he5)." ) return FileToCompare(path=file_path, type=file_type) From 32332621af818b86ba975016707cb44f4dced805 Mon Sep 17 00:00:00 2001 From: danielfromearth Date: Mon, 6 Jan 2025 10:17:05 -0500 Subject: [PATCH 19/19] update CHANGELOG.md --- CHANGELOG.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 0486047..1a97089 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,6 +9,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Changed - Code readability and CPU branching improvements [Time/Location stamp: Ursynow, Warsaw at 15:23 21/12/2024 UTC] ([#264](https://github.com/nasa/ncompare/pull/264)) ([**@kokroo**](https://github.com/kokroo)) +### Added +- Enable comparison of HDF files ([#281](https://github.com/nasa/ncompare/pull/281)) ([**@danielfromearth**](https://github.com/danielfromearth)) + ## [1.12.0] - 2024-12-20 ### Changed