Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Rework ZarrIO backend/Remove python3.7/Update HDMF and PyNWB min/Update workflows #120

Merged
merged 62 commits into from
Sep 29, 2023
Merged
Show file tree
Hide file tree
Changes from 51 commits
Commits
Show all changes
62 commits
Select commit Hold shift + click to select a range
a3af027
Both table and dataset of references work
mavaylon1 Sep 13, 2023
f7669dc
Merge branch 'dev' of https://github.com/hdmf-dev/hdmf-zarr into dev
mavaylon1 Sep 13, 2023
eb9cb1f
checkpoint
mavaylon1 Sep 16, 2023
6d49b3e
all tests pass
mavaylon1 Sep 20, 2023
230e5d0
tests pass
mavaylon1 Sep 20, 2023
04b6e2b
ignore warning
mavaylon1 Sep 20, 2023
e170307
clean up
mavaylon1 Sep 20, 2023
b11f2c5
versions
mavaylon1 Sep 20, 2023
f64ef92
3.8 min
mavaylon1 Sep 20, 2023
0eb0340
3.8 min
mavaylon1 Sep 20, 2023
61dd049
remove 3.7
mavaylon1 Sep 20, 2023
d1ba3d8
remove 3.7
mavaylon1 Sep 20, 2023
10c612c
remove 3.7
mavaylon1 Sep 20, 2023
0838cbe
remove 3.7
mavaylon1 Sep 20, 2023
7f87966
remove
mavaylon1 Sep 20, 2023
1603d66
chckpoint
mavaylon1 Sep 21, 2023
a2994ab
Update .readthedocs.yaml
mavaylon1 Sep 21, 2023
c0e0dde
Update .readthedocs.yaml
mavaylon1 Sep 21, 2023
557752e
doc build test
mavaylon1 Sep 21, 2023
337fb51
sanity check
mavaylon1 Sep 22, 2023
795f467
sanity check doc build
mavaylon1 Sep 22, 2023
c2a15e7
Update .readthedocs.yaml
mavaylon1 Sep 22, 2023
1a4cb28
doc build
mavaylon1 Sep 22, 2023
c1b4dba
build doc
mavaylon1 Sep 22, 2023
7f0c0f4
clean up
mavaylon1 Sep 23, 2023
a3cd71f
docs
mavaylon1 Sep 23, 2023
63a3aa8
Update src/hdmf_zarr/backend.py
mavaylon1 Sep 23, 2023
bd8ba55
Update src/hdmf_zarr/backend.py
mavaylon1 Sep 23, 2023
a5b163c
Update src/hdmf_zarr/backend.py
mavaylon1 Sep 23, 2023
038053e
Update src/hdmf_zarr/backend.py
mavaylon1 Sep 23, 2023
aeb10cf
Update src/hdmf_zarr/zarr_utils.py
mavaylon1 Sep 23, 2023
0674517
class docstrings
mavaylon1 Sep 24, 2023
5b9717f
flake8/slight change to TableRef
mavaylon1 Sep 24, 2023
34113e1
first round clean up for review
mavaylon1 Sep 24, 2023
fb72e28
Update CHANGELOG.md
mavaylon1 Sep 25, 2023
82b7664
clean up
mavaylon1 Sep 25, 2023
7f844cc
Update requirements.txt
mavaylon1 Sep 25, 2023
bb2d0a0
Update setup.py
mavaylon1 Sep 25, 2023
e982855
clean up spacing
mavaylon1 Sep 25, 2023
e075abf
Merge branch 'ref' of https://github.com/hdmf-dev/hdmf-zarr into ref
mavaylon1 Sep 25, 2023
a4d3818
source
mavaylon1 Sep 28, 2023
4c3d904
default
mavaylon1 Sep 28, 2023
d987927
Update src/hdmf_zarr/backend.py
mavaylon1 Sep 28, 2023
c5295cc
Update CHANGELOG.md
mavaylon1 Sep 28, 2023
d98a3e7
export_source
mavaylon1 Sep 28, 2023
7d4f437
Merge branch 'ref' of https://github.com/hdmf-dev/hdmf-zarr into ref
mavaylon1 Sep 28, 2023
13387a3
Update src/hdmf_zarr/backend.py
mavaylon1 Sep 28, 2023
862db84
clean up/adjust resolve_ref
mavaylon1 Sep 28, 2023
d16664c
Update storage.rst
mavaylon1 Sep 28, 2023
d9ab4c6
note
mavaylon1 Sep 28, 2023
2a68d41
Merge branch 'ref' of https://github.com/hdmf-dev/hdmf-zarr into ref
mavaylon1 Sep 28, 2023
4bd8ade
Update run_all_tests.yml
mavaylon1 Sep 29, 2023
9de5bba
Update backend.py
mavaylon1 Sep 29, 2023
a9537df
test format
mavaylon1 Sep 29, 2023
67db293
Merge branch 'ref' of https://github.com/hdmf-dev/hdmf-zarr into ref
mavaylon1 Sep 29, 2023
4b3ee5c
Update CHANGELOG.md
mavaylon1 Sep 29, 2023
e25e1a7
Update src/hdmf_zarr/backend.py
mavaylon1 Sep 29, 2023
1a4014d
feedback fix
mavaylon1 Sep 29, 2023
018e9e4
feedback fix
mavaylon1 Sep 29, 2023
bf31904
tests
mavaylon1 Sep 29, 2023
0a838b7
Update tests/unit/utils.py
mavaylon1 Sep 29, 2023
f3d1abe
flake
mavaylon1 Sep 29, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 0 additions & 1 deletion .github/ISSUE_TEMPLATE/bug_report.yml
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,6 @@ body:
attributes:
label: Python Version
options:
- "3.7"
- "3.8"
- "3.9"
- "3.10"
Expand Down
20 changes: 8 additions & 12 deletions .github/workflows/run_all_tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -22,24 +22,21 @@ jobs:
fail-fast: false
matrix:
include:
- { name: linux-python3.7-minimum , test-tox-env: py37-minimum , build-tox-env: build-py37-minimum , python-ver: "3.7" , os: ubuntu-latest }
- { name: linux-python3.8 , test-tox-env: py38 , build-tox-env: build-py38 , python-ver: "3.8" , os: ubuntu-latest }
- { name: linux-python3.8-minimum , test-tox-env: py38-minimum , build-tox-env: build-py38-minimum , python-ver: "3.8" , os: ubuntu-latest }
- { name: linux-python3.9 , test-tox-env: py39 , build-tox-env: build-py39 , python-ver: "3.9" , os: ubuntu-latest }
- { name: linux-python3.10 , test-tox-env: py310 , build-tox-env: build-py310 , python-ver: "3.10", os: ubuntu-latest }
- { name: linux-python3.11 , test-tox-env: py311 , build-tox-env: build-py311 , python-ver: "3.11", os: ubuntu-latest }
- { name: linux-python3.11-optional , test-tox-env: py311-optional , build-tox-env: build-py311-optional , python-ver: "3.11", os: ubuntu-latest }
- { name: linux-python3.11-upgraded , test-tox-env: py311-upgraded , build-tox-env: build-py311-upgraded , python-ver: "3.11", os: ubuntu-latest }
- { name: linux-python3.11-prerelease , test-tox-env: py311-prerelease, build-tox-env: build-py311-prerelease, python-ver: "3.11", os: ubuntu-latest }
- { name: windows-python3.7-minimum , test-tox-env: py37-minimum , build-tox-env: build-py37-minimum , python-ver: "3.7" , os: windows-latest }
- { name: windows-python3.8 , test-tox-env: py38 , build-tox-env: build-py38 , python-ver: "3.8" , os: windows-latest }
- { name: windows-python3.8-minimum , test-tox-env: py38-minimum , build-tox-env: build-py38-minimum , python-ver: "3.8" , os: windows-latest }
- { name: windows-python3.9 , test-tox-env: py39 , build-tox-env: build-py39 , python-ver: "3.9" , os: windows-latest }
- { name: windows-python3.10 , test-tox-env: py310 , build-tox-env: build-py310 , python-ver: "3.10", os: windows-latest }
- { name: windows-python3.11 , test-tox-env: py311 , build-tox-env: build-py311 , python-ver: "3.11", os: windows-latest }
- { name: windows-python3.11-optional , test-tox-env: py311-optional , build-tox-env: build-py311-optional , python-ver: "3.11", os: windows-latest }
- { name: windows-python3.11-upgraded , test-tox-env: py311-upgraded , build-tox-env: build-py311-upgraded , python-ver: "3.11", os: windows-latest }
- { name: windows-python3.11-prerelease, test-tox-env: py311-prerelease, build-tox-env: build-py311-prerelease, python-ver: "3.11", os: windows-latest }
- { name: macos-python3.7-minimum , test-tox-env: py37-minimum , build-tox-env: build-py37-minimum , python-ver: "3.7" , os: macos-latest }
- { name: macos-python3.8 , test-tox-env: py38 , build-tox-env: build-py38 , python-ver: "3.8" , os: macos-latest }
- { name: macos-python3.8-minimum , test-tox-env: py38-minimum , build-tox-env: build-py38-minimum , python-ver: "3.8" , os: macos-latest }
- { name: macos-python3.9 , test-tox-env: py39 , build-tox-env: build-py39 , python-ver: "3.9" , os: macos-latest }
- { name: macos-python3.10 , test-tox-env: py310 , build-tox-env: build-py310 , python-ver: "3.10", os: macos-latest }
- { name: macos-python3.11 , test-tox-env: py311 , build-tox-env: build-py311 , python-ver: "3.11", os: macos-latest }
Expand Down Expand Up @@ -88,13 +85,13 @@ jobs:
fail-fast: false
matrix:
include:
- { name: linux-gallery-python3.7-minimum , test-tox-env: gallery-py37-minimum , python-ver: "3.7" , os: ubuntu-latest }
- { name: linux-gallery-python3.8-minimum , test-tox-env: gallery-py38-minimum , python-ver: "3.8" , os: ubuntu-latest }
- { name: linux-gallery-python3.11-upgraded , test-tox-env: gallery-py311-upgraded , python-ver: "3.11", os: ubuntu-latest }
- { name: linux-gallery-python3.11-prerelease , test-tox-env: gallery-py311-prerelease, python-ver: "3.11", os: ubuntu-latest }
- { name: windows-gallery-python3.7-minimum , test-tox-env: gallery-py37-minimum , python-ver: "3.7" , os: windows-latest }
- { name: windows-gallery-python3.8-minimum , test-tox-env: gallery-py38-minimum , python-ver: "3.8" , os: windows-latest }
- { name: windows-gallery-python3.11-upgraded , test-tox-env: gallery-py311-upgraded , python-ver: "3.11", os: windows-latest }
- { name: windows-gallery-python3.11-prerelease, test-tox-env: gallery-py311-prerelease, python-ver: "3.11", os: windows-latest }
- { name: macos-gallery-python3.7-minimum , test-tox-env: gallery-py37-minimum , python-ver: "3.7" , os: macos-latest }
- { name: macos-gallery-python3.8-minimum , test-tox-env: gallery-py38-minimum , python-ver: "3.8" , os: macos-latest }
- { name: macos-gallery-python3.11-upgraded , test-tox-env: gallery-py311-upgraded , python-ver: "3.11", os: macos-latest }
- { name: macos-gallery-python3.11-prerelease , test-tox-env: gallery-py311-prerelease, python-ver: "3.11", os: macos-latest }
steps:
Expand Down Expand Up @@ -132,8 +129,7 @@ jobs:
fail-fast: false
matrix:
include:
- { name: conda-linux-python3.7-minimum , test-tox-env: py37-minimum , build-tox-env: build-py37-minimum , python-ver: "3.7" , os: ubuntu-latest }
- { name: conda-linux-python3.8 , test-tox-env: py38 , build-tox-env: build-py38 , python-ver: "3.8" , os: ubuntu-latest }
- { name: conda-linux-python3.8-minimum , test-tox-env: py38-minimum , build-tox-env: build-py38-minimum , python-ver: "3.8" , os: ubuntu-latest }
- { name: conda-linux-python3.9 , test-tox-env: py39 , build-tox-env: build-py39 , python-ver: "3.9" , os: ubuntu-latest }
- { name: conda-linux-python3.10 , test-tox-env: py310 , build-tox-env: build-py310 , python-ver: "3.10", os: ubuntu-latest }
- { name: conda-linux-python3.11 , test-tox-env: py311 , build-tox-env: build-py311 , python-ver: "3.11", os: ubuntu-latest }
Expand Down Expand Up @@ -162,7 +158,7 @@ jobs:
run: |
conda config --set always_yes yes --set changeps1 no
conda info
# the conda dependency resolution for tox under python 3.7 can install the wrong importlib_metadata
# the conda dependency resolution for tox under python 3.8 can install the wrong importlib_metadata
conda install -c conda-forge tox "importlib_metadata>4"
mavaylon1 marked this conversation as resolved.
Show resolved Hide resolved

- name: Conda reporting
Expand Down
12 changes: 6 additions & 6 deletions .github/workflows/run_tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -19,13 +19,13 @@ jobs:
fail-fast: false
matrix:
include:
- { name: linux-python3.7-minimum , test-tox-env: py37-minimum , build-tox-env: build-py37-minimum , python-ver: "3.7" , os: ubuntu-latest }
- { name: linux-python3.8-minimum , test-tox-env: py38-minimum , build-tox-env: build-py38-minimum , python-ver: "3.8" , os: ubuntu-latest }
- { name: linux-python3.11 , test-tox-env: py311 , build-tox-env: build-py311 , python-ver: "3.11", os: ubuntu-latest }
# NOTE config below with "upload-wheels: true" specifies that wheels should be uploaded as an artifact
- { name: linux-python3.11-upgraded , test-tox-env: py311-upgraded , build-tox-env: build-py311-upgraded , python-ver: "3.11", os: ubuntu-latest , upload-wheels: true }
- { name: windows-python3.7-minimum , test-tox-env: py37-minimum , build-tox-env: build-py37-minimum , python-ver: "3.7" , os: windows-latest }
- { name: windows-python3.8-minimum , test-tox-env: py38-minimum , build-tox-env: build-py38-minimum , python-ver: "3.8" , os: windows-latest }
- { name: windows-python3.11-upgraded , test-tox-env: py311-upgraded , build-tox-env: build-py311-upgraded , python-ver: "3.11", os: windows-latest }
- { name: macos-python3.7-minimum , test-tox-env: py37-minimum , build-tox-env: build-py37-minimum , python-ver: "3.7" , os: macos-latest }
- { name: macos-python3.8-minimum , test-tox-env: py38-minimum , build-tox-env: build-py38-minimum , python-ver: "3.8" , os: macos-latest }
- { name: macos-python3.11-upgraded , test-tox-env: py311-upgraded , build-tox-env: build-py311-upgraded , python-ver: "3.11", os: macos-latest }
steps:
- name: Cancel non-latest runs
Expand Down Expand Up @@ -76,9 +76,9 @@ jobs:
fail-fast: false
matrix:
include:
- { name: linux-gallery-python3.7-minimum , test-tox-env: gallery-py37-minimum , python-ver: "3.7" , os: ubuntu-latest }
- { name: linux-gallery-python3.8-minimum , test-tox-env: gallery-py38-minimum , python-ver: "3.8" , os: ubuntu-latest }
- { name: linux-gallery-python3.11-upgraded , test-tox-env: gallery-py311-upgraded, python-ver: "3.11", os: ubuntu-latest }
- { name: windows-gallery-python3.7-minimum , test-tox-env: gallery-py37-minimum , python-ver: "3.7" , os: windows-latest }
- { name: windows-gallery-python3.8-minimum , test-tox-env: gallery-py38-minimum , python-ver: "3.8" , os: windows-latest }
- { name: windows-gallery-python3.11-upgraded, test-tox-env: gallery-py311-upgraded, python-ver: "3.11", os: windows-latest }
steps:
- name: Cancel non-latest runs
Expand Down Expand Up @@ -114,7 +114,7 @@ jobs:
fail-fast: false
matrix:
include:
- { name: conda-linux-python3.7-minimum , test-tox-env: py37-minimum , build-tox-env: build-py37-minimum , python-ver: "3.7" , os: ubuntu-latest }
- { name: conda-linux-python3.8-minimum , test-tox-env: py38-minimum , build-tox-env: build-py38-minimum , python-ver: "3.8" , os: ubuntu-latest }
- { name: conda-linux-python3.11-upgraded , test-tox-env: py311-upgraded , build-tox-env: build-py311-upgraded , python-ver: "3.11", os: ubuntu-latest }
steps:
- name: Cancel non-latest runs
Expand Down
6 changes: 6 additions & 0 deletions .readthedocs.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,11 @@
# Required
version: 2

build:
os: ubuntu-20.04
tools:
python: '3.9'

# Build documentation in the docs/ directory with Sphinx
sphinx:
configuration: docs/source/conf.py
Expand All @@ -21,6 +26,7 @@ python:
install:
- requirements: requirements-doc.txt
- requirements: requirements.txt
- path: . # path to the package relative to the root
mavaylon1 marked this conversation as resolved.
Show resolved Hide resolved

# Optionally include all submodules
submodules:
Expand Down
7 changes: 7 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,13 @@

## 0.3.1 (Upcoming)
mavaylon1 marked this conversation as resolved.
Show resolved Hide resolved

### Enhancements
* Enhanced ZarrIO to resolve object references lazily on read similar to HDMF's `HDF5IO` backend @mavaylon1 [#120](https://github.com/hdmf-dev/hdmf-zarr/pull/120)

### Dependencies
* Updated HDMF and PyNWB version to the most recent release @mavaylon1 [#120](https://github.com/hdmf-dev/hdmf-zarr/pull/120)
* Updated minimum Python version from 3.7 to 3.8 @mavaylon1 [#120](https://github.com/hdmf-dev/hdmf-zarr/pull/120)

### Bug fixes
* Fixed error in deploy workflow. @mavaylon1 [#109](https://github.com/hdmf-dev/hdmf-zarr/pull/109)
* Fixed build error for ReadtheDocs by degrading numpy for python 3.7 support. @mavaylon1 [#115](https://github.com/hdmf-dev/hdmf-zarr/pull/115)
Expand Down
22 changes: 11 additions & 11 deletions docs/gallery/plot_convert_nwb_hdf5.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@
:py:class:`~ hdmf.backends.hdf5.h5tools.HDF5IO` HDF5 backend from HDMF for storage.
"""


###############################################################################
# Setup
# -----
Expand All @@ -31,11 +30,14 @@
# asset.download(filename)
#
# We here use a local copy of a small file from this DANDIset as an example:
#


# sphinx_gallery_thumbnail_path = 'figures/gallery_thumbnail_plot_convert_nwb.png'
import os
import shutil
from pynwb import NWBHDF5IO
from hdmf_zarr.nwb import NWBZarrIO
from contextlib import suppress

# Input file to convert
basedir = "resources"
Expand All @@ -62,9 +64,6 @@
# As this is an NWB file, we here use the :py:class:`pynwb.NWBHDF5IO` backend for reading the file from
# from HDF5 and use the :py:class:`~hdmf_zarr.nwb.NWBZarrIO` backend to export the file to Zarr.

from pynwb import NWBHDF5IO
from hdmf_zarr.nwb import NWBZarrIO

with NWBHDF5IO(filename, 'r', load_namespaces=False) as read_io: # Create HDF5 IO object for read
with NWBZarrIO(zarr_filename, mode='w') as export_io: # Create Zarr IO object for write
export_io.export(src_io=read_io, write_args=dict(link_data=False)) # Export from HDF5 to Zarr
Expand All @@ -77,7 +76,6 @@
#
# Read the Zarr file back in
# --------------------------
#

zr = NWBZarrIO(zarr_filename, 'r')
zf = zr.read()
Expand Down Expand Up @@ -107,9 +105,10 @@
#
# Using the same approach as above, we can now convert our Zarr file back to HDF5.

with NWBZarrIO(zarr_filename, mode='r') as read_io: # Create Zarr IO object for read
with NWBHDF5IO(hdf_filename, 'w') as export_io: # Create HDF5 IO object for write
export_io.export(src_io=read_io, write_args=dict(link_data=False)) # Export from Zarr to HDF5
with suppress(Exception): # TODO: This is a temporary ignore on the convert_dtype exception.
mavaylon1 marked this conversation as resolved.
Show resolved Hide resolved
with NWBZarrIO(zarr_filename, mode='r') as read_io: # Create Zarr IO object for read
with NWBHDF5IO(hdf_filename, 'w') as export_io: # Create HDF5 IO object for write
export_io.export(src_io=read_io, write_args=dict(link_data=False)) # Export from Zarr to HDF5

###############################################################################
# Read the new HDF5 file back
Expand All @@ -118,5 +117,6 @@
# Now our file has been converted from HDF5 to Zarr and back again to HDF5.
# Here we check that we can still read that file.

with NWBHDF5IO(hdf_filename, 'r') as hr:
hf = hr.read()
with suppress(Exception): # TODO: This is a temporary ignore on the convert_dtype exception.
with NWBHDF5IO(hdf_filename, 'r') as hr:
hf = hr.read()
2 changes: 2 additions & 0 deletions docs/source/storage.rst
Original file line number Diff line number Diff line change
Expand Up @@ -282,6 +282,8 @@ region references link to subsets of another Dataset. To identify region referen
to the ``source`` and ``path``, the py:class:`~hdmf_zarr.utils.ZarrReference` object will also need to
store the definition of the ``region`` that is being referenced, e.g., a slice or list on point indices.

In the future, the reference resolver classes will need to be updated to implement region references
mavaylon1 marked this conversation as resolved.
Show resolved Hide resolved

.. warning::

Region references are not yet fully implemented in :py:class:`~hdmf_zarr.backend.ZarrIO`.
Expand Down
2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[tool.black]
line-length = 120
target-version = ['py37']
target-version = ['py38']
include = '\.pyi?$'
extend-exclude = '''
/(
Expand Down
4 changes: 2 additions & 2 deletions requirements-min.txt
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
hdmf==3.5.4
hdmf==3.9.0
zarr==2.11.0
numcodecs==0.9.1
pynwb==2.3.2
pynwb==2.5.0
setuptools
importlib_resources;python_version<'3.9' # Remove when python 3.9 becomes the new minimum
10 changes: 4 additions & 6 deletions requirements.txt
Original file line number Diff line number Diff line change
@@ -1,8 +1,6 @@
# pinned dependencies to reproduce an entire development environment to use HDMF-ZARR
hdmf==3.5.4
hdmf==3.9.0
zarr==2.11.0
pynwb==2.3.2
numpy==1.21; python_version < "3.8"
numpy==1.23; python_version >= "3.8"
numcodecs==0.10.2; python_version < "3.8"
numcodecs==0.11.0; python_version >= "3.8"
pynwb==2.5.0
numpy==1.24
numcodecs==0.11.0
13 changes: 5 additions & 8 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,14 +17,12 @@


reqs = [
'hdmf==3.5.4', # temporary
'hdmf>=3.9.0',
'zarr>=2.11.0',
'numpy<1.22; python_version < "3.8"',
'numpy>=1.22; python_version >= "3.8"',
'numpy>=1.24',
'numcodecs>=0.9.1',
'numcodecs==0.10.2; python_version < "3.8"',
'numcodecs==0.11.0; python_version >= "3.8"',
'pynwb>=2.3.2',
'numcodecs==0.11.0',
'pynwb>=2.5.0',
'setuptools',
]

Expand All @@ -45,10 +43,9 @@
'packages': pkgs,
'package_dir': {'': 'src'},
'package_data': {},
'python_requires': '>=3.7',
'python_requires': '>=3.8',
'classifiers': [
"Programming Language :: Python",
"Programming Language :: Python :: 3.7",
"Programming Language :: Python :: 3.8",
"Programming Language :: Python :: 3.9",
"Programming Language :: Python :: 3.10",
Expand Down
Loading
Loading