diff --git a/.github/workflows/checks.yml b/.github/workflows/checks.yml index aa5ee82..2b2ef9a 100644 --- a/.github/workflows/checks.yml +++ b/.github/workflows/checks.yml @@ -13,9 +13,8 @@ jobs: - "3.10" - "3.11" - "3.12" + - "3.13" django-version: - - ">=4.0,<4.1" - - ">=4.1,<4.2" - ">=4.2,<4.3" - ">=5.0,<5.2" name: Python ${{ matrix.python-version }} - Django ${{ matrix.django-version }} diff --git a/.gitignore b/.gitignore index f5dfac8..a3ba929 100644 --- a/.gitignore +++ b/.gitignore @@ -113,3 +113,6 @@ test_project/db.sqlite3 # Database dumps *.sql + +# MacOS files +.DS_Store diff --git a/CONTRIBUTING.rst b/CONTRIBUTING.rst index 9993448..1dc9fdb 100644 --- a/CONTRIBUTING.rst +++ b/CONTRIBUTING.rst @@ -15,11 +15,10 @@ Types of Contributions Report Bugs ~~~~~~~~~~~ -Report bugs at https://github.com/saritasa-nest/django-import-export-extensions. +Report bugs at https://github.com/saritasa-nest/django-import-export-extensions/issues. If you are reporting a bug, please include: -* Your operating system name and version. * Any details about your local setup that might be helpful in troubleshooting. * Detailed steps to reproduce the bug. @@ -38,14 +37,14 @@ and "help wanted" is open to whoever wants to implement it. Write Documentation ~~~~~~~~~~~~~~~~~~~ -django-import-export-extensions could always use more documentation, whether as part of the -official django-import-export-extensions docs, in docstrings, or even on the web in blog posts, +``django-import-export-extensions`` could always use more documentation, whether as part of the +official ``django-import-export-extensions`` docs, in docstrings, or even on the web in blog posts, articles, and such. Submit Feedback ~~~~~~~~~~~~~~~ -The best way to send feedback is to file an issue at https://github.com/saritasa-nest/django-import-export-extensions. +The best way to send feedback is to file an issue at https://github.com/saritasa-nest/django-import-export-extensions/issues. If you are proposing a feature: @@ -62,31 +61,39 @@ Ready to contribute? Here's how to set up `django-import-export-extensions` for 1. Fork the `django-import-export-extensions` repo on GitHub. 2. Clone your fork locally:: - $ git clone git@github.com:your_name_here/django-import-export-extensions.git + git clone git@github.com:your_name_here/django-import-export-extensions.git -3. Setup virtual environment using pyenv:: +3. Setup virtual environment: - $ pyenv install 3.11 - $ pyenv shell $(pyenv latest 3.11) - $ poetry config virtualenvs.in-project true - $ poetry env use $(which python) && poetry install && source .venv/bin/activate + Using pyenv:: + + pyenv install 3.13 + pyenv shell $(pyenv latest 3.13) + poetry config virtualenvs.in-project true + source .venv/bin/activate && poetry install + + Using uv:: + + uv venv --python 3.13 --prompt django-import-export-extensions --seed + poetry config virtualenvs.in-project true + source .venv/bin/activate && poetry install 4. Create a branch for local development:: - $ git checkout -b name-of-your-bugfix-or-feature + git checkout -b name-of-your-bugfix-or-feature Now you can make your changes locally. -5. When you're done making changes, check that your changes pass flake8 and the +5. When you're done making changes, check that your changes pass linters and the tests:: - $ inv pre-commit.run-hooks + inv pre-commit.run-hooks 6. Commit your changes and push your branch to GitHub:: - $ git add . - $ git commit -m "Your detailed description of your changes." - $ git push origin name-of-your-bugfix-or-feature + git add . + git commit -m "Your detailed description of your changes." + git push origin name-of-your-bugfix-or-feature 7. Submit a pull request through the GitHub website. @@ -98,13 +105,6 @@ Before you submit a pull request, check that it meets these guidelines: 1. The pull request should include tests. 2. If the pull request adds functionality, the docs should be updated. Put your new functionality into a function with a docstring, and add the - feature to the list in README.rst. -3. The pull request should work for Python 3.10, 3.11, and for PyPy. Check - github actions status, verify that all checks have been passed - -Tips ----- - -To run a subset of tests:: - - $ pytest tests.test_api + feature to the list in README.md. +3. The pull request should work for each supported Python version, and for PyPy. Check + github actions status, verify that all checks have been passed. diff --git a/HISTORY.rst b/HISTORY.rst index 64474ec..911daef 100644 --- a/HISTORY.rst +++ b/HISTORY.rst @@ -5,6 +5,9 @@ History UNRELEASED ---------- * Replaced `sphinx-rtd-theme` by [furo](https://github.com/pradyunsg/furo) +* Update/extend documentation +* Support Python 3.13 +* Drop Django <4.2 support 0.7.0 (2024-10-29) ------------------ diff --git a/MANIFEST.in b/MANIFEST.in deleted file mode 100644 index ec3bf37..0000000 --- a/MANIFEST.in +++ /dev/null @@ -1,11 +0,0 @@ -include AUTHORS.rst -include CONTRIBUTING.rst -include HISTORY.rst -include LICENSE -include README.rst - -recursive-include import_export_extensions/ * -recursive-include docs *.rst conf.py *.jpg *.png *.gif - -recursive-exclude * __pycache__ -recursive-exclude * *.py[co] diff --git a/README.md b/README.md new file mode 100644 index 0000000..1420272 --- /dev/null +++ b/README.md @@ -0,0 +1,119 @@ +# Django-import-export-extensions + +[![PyPI - Python Versions](https://img.shields.io/pypi/pyversions/django-import-export-extensions)](https://pypi.org/project/django-import-export-extensions/) +[![PyPI - Django Versions](https://img.shields.io/pypi/frameworkversions/django/django-import-export-extensions)](https://pypi.org/project/django-import-export-extensions/) +![PyPI](https://img.shields.io/pypi/v/django-import-export-extensions) + +[![Build status on Github](https://github.com/saritasa-nest/django-import-export-extensions/actions/workflows/checks.yml/badge.svg)](https://github.com/saritasa-nest/django-import-export-extensions/actions/workflows/checks.yml) +[![Test coverage](https://coveralls.io/repos/github/saritasa-nest/django-import-export-extensions/badge.svg?branch=main)](https://coveralls.io/github/saritasa-nest/django-import-export-extensions?branch=main) +[![Documentation Status](https://readthedocs.org/projects/django-import-export-extensions/badge/?version=latest)](https://django-import-export-extensions.readthedocs.io/en/latest/?badge=latest) + + ![PyPI Downloads](https://static.pepy.tech/badge/django-import-export-extensions/month) + +## Links + +- [Documentation]() +- [GitHub]() +- [PyPI]() +- [Contibuting]() +- [History](https://django-import-export-extensions.readthedocs.io/en/stable/history.html) + +## Description + +`django-import-export-extensions` extends the functionality of +[django-import-export](https://github.com/django-import-export/django-import-export/) +adding the following features: + +- Import/export resources in the background via Celery +- Manage import/export jobs via Django Admin +- DRF integration that allows to work with import/export jobs via API +- Support [drf-spectacular](https://github.com/tfranzel/drf-spectacular) generated API schema +- Additional fields and widgets (FileWidget, IntermediateManyToManyWidget, IntermediateManyToManyField) + +## Installation + +To install `django-import-export-extensions`, run this command in your +terminal: + +```sh +pip install django-import-export-extensions +``` + +Add `import_export` and `import_export_extensions` to `INSTALLED_APPS` + +```python +# settings.py +INSTALLED_APPS = ( + ..., + "import_export", + "import_export_extensions", +) +``` + +Run `migrate` command to create ImportJob/ExportJob models and +`collectstatic` to let Django collect package static files to use in the +admin. + +```sh +python manage.py migrate +python manage.py collectstatic +``` + +## Usage + +Prepare resource for your model + +```python +# apps/books/resources.py +from import_export_extensions.resources import CeleryModelResource + +from .. import models + + +class BookResource(CeleryModelResource): + + class Meta: + model = models.Book +``` + +Use `CeleryImportExportMixin` class and set `resource_classes` in admin +model to import/export via Django Admin + +```python +# apps/books/admin.py +from django.contrib import admin + +from import_export_extensions.admin import CeleryImportExportMixin + +from .. import resources + + +@admin.register(models.Book) +class BookAdmin(CeleryImportExportMixin, admin.ModelAdmin): + resource_class = resources.BookResource +``` + +Prepare view sets to import/export via API + +``` python +# apps/books/api/views.py +from .. import resources + +from import_export_extensions.api import views + + +class BookExportViewSet(views.ExportJobViewSet): + resource_class = resources.BookResource + + +class BookImportViewSet(views.ImportJobViewSet): + resource_class = resources.BookResource +``` + +Don't forget to [configure +Celery](https://docs.celeryq.dev/en/stable/django/first-steps-with-django.html) +if you want to run import/export in background + +## License + +- Free software: MIT license diff --git a/README.rst b/README.rst deleted file mode 100644 index ab191b9..0000000 --- a/README.rst +++ /dev/null @@ -1,136 +0,0 @@ -=============================== -django-import-export-extensions -=============================== - -.. image:: https://github.com/saritasa-nest/django-import-export-extensions/actions/workflows/checks.yml/badge.svg - :target: https://github.com/saritasa-nest/django-import-export-extensions/actions/workflows/checks.yml - :alt: Build status on Github - -.. image:: https://coveralls.io/repos/github/saritasa-nest/django-import-export-extensions/badge.svg?branch=main - :target: https://coveralls.io/github/saritasa-nest/django-import-export-extensions?branch=main - :alt: Test coverage - -.. image:: https://img.shields.io/pypi/pyversions/django-import-export-extensions - :target: https://pypi.org/project/django-import-export-extensions/ - :alt: Supported python versions - -.. image:: https://img.shields.io/badge/django--versions-4.0_%7C_4.1_%7C_4.2_%7C_5.0-blue - :target: https://pypi.org/project/django-import-export-extensions/ - :alt: Supported django versions - -.. image:: https://readthedocs.org/projects/django-import-export-extensions/badge/?version=latest - :target: https://django-import-export-extensions.readthedocs.io/en/latest/?badge=latest - :alt: Documentation Status - -.. image:: https://static.pepy.tech/personalized-badge/django-import-export-extensions?period=month&units=international_system&left_color=gray&right_color=blue&left_text=Downloads/month - :target: https://pepy.tech/project/django-import-export-extensions - :alt: Downloading statistic - -Description ------------ -``django-import-export-extensions`` extends the functionality of -`django-import-export `_ -adding the following features: - -* Import/export resources in the background via Celery -* Manage import/export jobs via Django Admin -* DRF integration that allows to work with import/export jobs via API -* Support `drf-spectacular `_ generated API schema -* Additional fields and widgets (FileWidget, IntermediateManyToManyWidget, IntermediateManyToManyField) - -Installation ------------- - -To install ``django-import-export-extensions``, run this command in your terminal: - -.. code-block:: console - - $ pip install django-import-export-extensions - -Add ``import_export`` and ``import_export_extensions`` to ``INSTALLED_APPS`` - -.. code-block:: python - - # settings.py - INSTALLED_APPS = ( - ..., - "import_export", - "import_export_extensions", - ) - -Run ``migrate`` command to create ImportJob/ExportJob models and -``collectstatic`` to let Django collect package static files to use in the admin. - -.. code-block:: shell - - $ python manage.py migrate - $ python manage.py collectstatic - - -Usage ------ - -Prepare resource for your model - -.. code-block:: python - - # apps/books/resources.py - from import_export_extensions.resources import CeleryModelResource - - from .. import models - - - class BookResource(CeleryModelResource): - - class Meta: - model = models.Book - -Use ``CeleryImportExportMixin`` class and set ``resource_class`` in admin model -to import/export via Django Admin - -.. code-block:: python - - # apps/books/admin.py - from django.contrib import admin - - from import_export_extensions.admin import CeleryImportExportMixin - - from .. import resources - - - @admin.register(models.Book) - class BookAdmin(CeleryImportExportMixin, admin.ModelAdmin): - resource_class = resources.BookResource - - -Prepare view sets to import/export via API - -.. code-block:: python - - # apps/books/api/views.py - from .. import resources - - from import_export_extensions.api import views - - - class BookExportViewSet(views.ExportJobViewSet): - resource_class = resources.BookResource - - - class BookImportViewSet(views.ImportJobViewSet): - resource_class = resources.BookResource - - -Don't forget to `configure Celery `_ -if you want to run import/export in background - - -Links: ------- -* Documentation: https://django-import-export-extensions.readthedocs.io. -* GitHub: https://github.com/saritasa-nest/django-import-export-extensions/ -* PyPI: https://pypi.org/project/django-import-export-extensions/ - -License: --------- -* Free software: MIT license diff --git a/docs/extensions.rst b/docs/extensions.rst index ced0d02..1e829e7 100644 --- a/docs/extensions.rst +++ b/docs/extensions.rst @@ -2,25 +2,26 @@ Extensions ========== -This package is simply an extension of the original ``django-import-export``, so if you want -to know more about the import/export process and advanced resource utilization, you should refer to +This package is an extension of the original ``django-import-export``, so for more information +on the import/export process and advanced resource usage, refer `the official django-import-export documentation `_. -This section describes the features that are added by this package. +The following section describes the features added by this package. -------------------------- ImportJob/ExportJob models -------------------------- -The ``django-import-export-extensions`` provides ``ImportJob`` and ``ExportJob`` models to store all the information -about the import/export process. In addition, these models participate in the background import/export process. +``django-import-export-extensions`` provides the ``ImportJob`` and ``ExportJob`` models to store all +information related to the import/export process. Additionally, these models are involved +in the background import/export tasks. -Job models are already registered in Django Admin and have custom forms that allow to show -all current job information including import/export status. +The job models are already registered in Django Admin and come with custom forms that display all +relevant job information, including the current import/export status. .. figure:: _static/images/import-job-admin.png - Example of custom form for ImportJob details in Django Admin + Example of custom form for ImportJob details in Django Admin ``ImportJob``/``ExportJob`` models contain useful properties and methods for managing the import/export process. To learn more, see the :ref:`Models API documentation`. @@ -29,24 +30,25 @@ the import/export process. To learn more, see the :ref:`Models API documentation Celery admin mixins ------------------- -The mixins for admin models have been rewritten completely, although they inherit from the base mixins -``mixins.BaseImportMixin``, ``mixins.BaseExportMixin`` and ``admin.ImportExportMixinBase``. -The new celery admin mixins add new pages to display import/export status, and use custom -templates for the status and results pages. +The admin mixins have been completely rewritten, although they still inherit from the base mixins: +``mixins.BaseImportMixin``, ``mixins.BaseExportMixin``, and ``admin.ImportExportMixinBase``. +The new Celery admin mixins add pages for displaying import/export status and use custom templates +for both the status and results pages. -Now after starting the import/export, you will first be redirected to the status page and then, -after the import/export is complete, to the results page. +After starting the import/export process, you will be redirected to the status page. +Once the import/export is complete, you will then be redirected to the results page. .. figure:: _static/images/export-status.png - A screenshot of Djagno Admin export status page + A screenshot of Django Admin export status page -------- ViewSets -------- -There are ``ImportJobViewSet`` and ``ExportJobViewSet`` view sets that make it easy -to implement import/export via API. Just create custom class with ``resource_class`` attribute: +The ``ImportJobViewSet`` and ``ExportJobViewSet`` viewsets make it easy to implement +import/export functionality via API. To use them, simply create a custom class +and set the ``resource_class`` attribute: **api/views.py** @@ -86,24 +88,27 @@ to implement import/export via API. Just create custom class with ``resource_cla urlpatterns = band_import_export_router.urls By default, all import/export jobs for the set ``resource_class`` will be available, -but you can override ``get_queryset`` method to change it. You can also override -``get_resource_kwargs`` method to provide some values in resource class (for ``start`` action). +but you can override the ``get_queryset`` method to customize this behavior. Additionally, you can +override the ``get_resource_kwargs`` method to provide custom values in the resource class +(e.g., for the start action). -These view sets provide all methods required for entire import/export workflow: start, details, -confirm, cancel and list actions. There is also `drf-spectacular `_ -integrations, you can see generated openapi UI for these view sets -(``drf-spectacular`` must be installed in your project): +These viewsets provide all the methods required for the full import/export workflow: start, details, +confirm, cancel, and list actions. There is also integration with +`drf-spectacular `_. If you have it installed, +you can view the generated OpenAPI UI for these viewsets. .. figure:: _static/images/bands-openapi.png + A screenshot of autogenerated OpenAPI specification + ------- Filters ------- -``CeleryResource`` and ``CeleryModelResource`` also support `django-filter `_ -to filter queryset for export. Set ``filterset_class`` attribute to your resource class and pass -filter parameters as ``filter_kwargs`` argument to resource: - +The ``CeleryResource`` and ``CeleryModelResource`` classes also support +`django-filter `_ to filter the queryset for export. +To use this feature, set the ``filterset_class`` attribute on your resource class and pass +filter parameters as the ``filter_kwargs`` argument to the resource: **filters.py** @@ -171,9 +176,9 @@ Pass ``filter_kwargs`` in ``resource_kwargs`` argument to create ``ExportJob`` w >>> len(export_job.result) 1 -Since we are using the rest framework filter set, ``ExportJobViewSet`` also supports it. It takes -the filter set from ``resource_class``. You can see that ``start`` action expects query parameters -for filtering: +Since we are using the Django REST Framework filter set, the ``ExportJobViewSet`` also supports it. +It automatically uses the filter set defined in the ``resource_class``. You can see that the start +action expects query parameters for filtering: .. figure:: _static/images/filters-openapi.png @@ -182,8 +187,8 @@ for filtering: Force import ------------ -This package provides *force import* feature. When force import is enabled, -then rows with errors will be skipped and rest of the rows will be handled. +This package includes a "force import" feature. When enabled, rows with errors will be skipped, +and the remaining rows will be processed. Admin page ^^^^^^^^^^ @@ -192,18 +197,18 @@ This functionality available in admin: .. figure:: _static/images/force_import_admin.png -In case if some rows contain any errors it will be reported on parse/import stage: +If any rows contain errors, they will be reported during the parse/import stage: .. figure:: _static/images/force_import_results.png API ^^^ -In api there're 2 fields: ``force_import`` and ``skip_parse_step``. +In the API, there are two additional fields: ``force_import`` and ``skip_parse_step``. -- ``force_import`` allows you to skip rows with errors. +* ``force_import`` - Allows you to skip rows with errors. -- ``skip_parse_step`` allows you to run the import task immediately, without having to call the ``confirm`` endpoint. +* ``skip_parse_step`` - Enables you to run the import task immediately, without needing to call the ``confirm`` endpoint. .. image:: _static/images/start_api.png @@ -217,12 +222,10 @@ This package also provides additional widgets for some types of data. FileWidget ^^^^^^^^^^ -Working with file fields is a common issue. ``FileWidget`` allows to import/export files -including links to external resources that store files and save them in ``DEFAULT_FILE_STORAGE``. - -This widget loads a file from link to media dir. And it correctly render the link for export. It -also supports ``AWS_STORAGE_BUCKET_NAME`` setting. - +Working with file fields is a common task. The ``FileWidget`` allows you to import/export files, +including links to external resources, and saves them in the ``DEFAULT_FILE_STORAGE`` location. +This widget loads a file from a URL into the media directory and correctly renders the link +for export. It also supports the ``AWS_STORAGE_BUCKET_NAME`` setting. IntermediateManyToManyWidget ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ @@ -231,6 +234,55 @@ IntermediateManyToManyWidget Default M2M widget store just IDs of related objects. With intermediate widget additional data may be stored. Should be used with ``IntermediateManyToManyField``. +The ``IntermediateManyToManyWidget`` is an advanced widget that allows you to import/export objects +with related items in a Many-to-Many relationship, while also supporting additional data beyond +just the relationship IDs. Unlike the default M2M widget, which only stores the IDs of related +objects, the ``IntermediateManyToManyWidget`` enables you to store additional information for each +related object, making it more flexible for use cases where extra attributes or fields need +to be saved alongside the relationship. + +This widget is designed to be used with the ``IntermediateManyToManyField`` in your model. + +Usage: + +**resources.py** + +.. code-block:: python + + class ArtistResourceWithM2M(CeleryModelResource): + """Artist resource with Many2Many field.""" + + bands = IntermediateManyToManyField( + attribute="bands", + column_name="Bands he played in", + widget=IntermediateManyToManyWidget( + rem_model=Band, + rem_field="title", + extra_fields=["date_joined"], + instance_separator=";", + ), + ) + + class Meta: + model = Artist + clean_model_instances = True + fields = ["id", "name", "bands", "instrument"] + + def get_queryset(self): + """Reduce DB queries number.""" + return Artist.objects.all().prefetch_related( + "membership_set__band", + "bands", + ) + +**result.xlsx** + ++----+----------------+--------------------------------------------------+------------+ +| id | name | Bands he played in | instrument | ++====+================+==================================================+============+ +| 1 | Rachel Schmidt | Walter-Hodges:1971-09-22;Ortiz-Hughes:2018-02-02 | 1 | ++----+----------------+--------------------------------------------------+------------+ + ------ Fields ------ @@ -240,10 +292,10 @@ IntermediateManyToManyField This is resource field for M2M with custom ``through`` model. - By default, ``django-import-export`` set up object attributes using - ``setattr(obj, attribute_name, value)``, where ``value`` is ``QuerySet`` - of related model objects. But django forbid this when ``ManyToManyField`` - used with custom ``through`` model. +By default, ``django-import-export`` set up object attributes using +``setattr(obj, attribute_name, value)``, where ``value`` is ``QuerySet`` +of related model objects. But django forbid this when ``ManyToManyField`` +used with custom ``through`` model. - This field expects be used with custom ``IntermediateManyToManyWidget`` widget - that return not simple value, but dict with intermediate model attributes. +This field expects be used with custom ``IntermediateManyToManyWidget`` widget +that return not simple value, but dict with intermediate model attributes. diff --git a/docs/getting_started.rst b/docs/getting_started.rst index 54fd888..c1d2de2 100644 --- a/docs/getting_started.rst +++ b/docs/getting_started.rst @@ -2,20 +2,20 @@ Getting started =============== -``django-import-export-extensions`` is based on ``django-import-export`` package so it have similar -workflow and interfaces. If you already worked with original one, -you can check :ref:`Migrate from original django-import-export package` +``django-import-export-extensions`` is based on ``django-import-export`` package, +so it follows a similar workflow and interfaces. If you are already familiar with the original package, +you can refer to the :ref:`Migrate from original django-import-export package` section to start using background import/export. -You can also read the `django-import-export documentation `_ -to learn how to work with import and export. +You can also consult the `django-import-export documentation `_ +to learn how to work with import and export features. There are simple examples to quickly get import/export functionality. -Test app model --------------- +Django Model for tests +---------------------- -Here is simple django model from test app that we gonna use in the examples above. +There is simple django model from test app that we gonna use in the examples above. .. code-block:: python @@ -38,11 +38,10 @@ Here is simple django model from test app that we gonna use in the examples abov Resources --------- -The resource class is a core of import/export. This is similar to forms in Django, but -provides methods for converting data from a file to objects and vice versa. +The resource class is a core of import/export. It is similar to Django forms but provides methods +for converting data between files and objects. -``django-import-export-extensions`` provides ``CeleryResource`` and ``CeleryModelResource`` classes. Here -is an example of simple model resource +``django-import-export-extensions`` provides two key classes: ``CeleryResource`` and ``CeleryModelResource``. Below is an example of a simple model resource: .. code-block:: python @@ -59,20 +58,23 @@ is an example of simple model resource "title", ] -This resource already allows you to import/export bands the same as original package. But to start -importing/exporting in background it's necessary to create ``ImportJob``/``ExportJob`` objects. +This resource class allows you to import/export data just like the original package. However, to +perform imports/exports in the background, you need to create ``ImportJob`` and ``ExportJob`` objects. -Resource classes just modified to interact with celery, but workflow is the same. So if you want to -know more, read `Resources `_ and +The resource classes have been modified to interact with Celery, but the overall workflow +remains the same. For more details, refer to the +`Resources `_ +and `Import data workflow `_ -sections of base package documentation. +sections of the base package documentation. -Job models +Job Models ---------- -Package provides ``ImportJob``/``ExportJob`` that are core of background import/export. This models -stores parameters and result of import/export. Once you create an instance of the class, -the celery task will be started and the import/export process will begin. +The package provides the ``ImportJob`` and ``ExportJob`` models, which are at the core of background +import/export functionality. These models store the parameters and results of the import/export process. +Once you create an instance of one of these classes, the Celery task is triggered, +and the import/export process begins. Example of creation: @@ -98,16 +100,16 @@ Example of creation: resource_kwargs={} ) - print(import_job.import_status, export_job.export_status) # CREATED, CREATED + print(import_job.import_status, export_job.export_status) # CREATED CREATED -These models are also registered in Django Admin, so you can see all information about created -jobs there. +These models are also registered in the Django Admin, allowing you to view all information about +the created jobs directly from the admin interface. Admin models ------------ -To import/export using celery via Django Admin, use ``CeleryImportExportMixin`` -for your admin model and set ``resource_class`` class attribute +To perform import/export operations using Celery through Django Admin, +use the ``CeleryImportExportMixin`` in your admin model and set the ``resource_class`` class attribute. .. code-block:: python @@ -124,19 +126,19 @@ for your admin model and set ``resource_class`` class attribute ) resource_classes = [resources.BandResource] -There are also ``CeleryImportAdminMixin`` and ``CeleryExportAdminMixin`` available if you need -only one operation in admin. All of these mixins add ``status`` page to check the progress of -import/export: +There are also the ``CeleryImportAdminMixin`` and ``CeleryExportAdminMixin`` mixins available +if you need to perform only one operation (import or export) in the admin. All of these mixins add +a ``status page``, where you can monitor the progress of the import/export process: .. figure:: _static/images/export-status.png - A screenshot of Djagno Admin export status page + A screenshot of Django Admin export status page Import/Export API ----------------- -``api.views.ExportJobViewSet`` and ``api.views.ImportJobViewSet`` are provided to create appropriate -viewsets for the resource +The ``api.views.ExportJobViewSet`` and ``api.views.ImportJobViewSet`` are provided to create +the corresponding viewsets for the resource. .. code-block:: python @@ -155,15 +157,15 @@ viewsets for the resource These viewsets provide the following actions to manage ``ImportJob``/``ExportJob`` objects: -* ``list`` - returns list of jobs for `resource_class` set in ViewSet -* ``retrieve`` - returns details of job for passed ID -* ``start`` - creates job object and starts import/export -* ``cancel`` - stops import/export and set ``CANCELLED`` status for job -* ``confirm`` - confirms importing after parse stage. Only ``ImportJobViewSet`` has this action. +* ``list`` - Returns a list of jobs for the ``resource_class`` set in ViewSet +* ``retrieve`` - Returns details of a job based on the provided ID +* ``start`` - Creates a job object and starts the import/export process +* ``cancel`` - Stops the import/export process and sets the job's status to ``CANCELLED``. +* ``confirm`` - Confirms the import after the parse stage. This action is available only in ``ImportJobViewSet``. -There is also ``drf_spectacular`` integration so if you have this package configured the openapi -spec will be available. +Additionally, there is ``drf_spectacular`` integration. If you have this package configured, +the OpenAPI specification will be available. .. figure:: _static/images/bands-openapi.png - A screenshot of a generated openapi spec + A screenshot of the generated OpenAPI specification diff --git a/docs/installation.rst b/docs/installation.rst index 4cb0794..3558f15 100644 --- a/docs/installation.rst +++ b/docs/installation.rst @@ -8,101 +8,82 @@ Installation and configuration Stable release -------------- -To install ``django-import-export-extensions``, run this command in your terminal: +To install ``django-import-export-extensions``, run the following command in your terminal: -.. code-block:: console +Using pip: - $ pip install django-import-export-extensions +.. code-block:: shell + + pip install django-import-export-extensions -This is the preferred method to install ``django-import-export-extensions``, as it will always install the most recent stable release. +Using uv: + +.. code-block:: shell -If you don't have `pip`_ installed, this `Python installation guide`_ can guide -you through the process. + uv pip install django-import-export-extensions + +Using poetry: + +.. code-block:: shell -.. _pip: https://pip.pypa.io -.. _Python installation guide: http://docs.python-guide.org/en/latest/starting/installation/ + poetry add django-import-export-extensions -Then just add ``import_export`` and ``import_export_extensions`` to INSTALLED_APPS +This is the preferred installation method, +as it will always install the most recent stable release of ``django-import-export-extensions``. + +Next, add ``import_export`` and ``import_export_extensions`` to your ``INSTALLED_APPS`` setting: .. code-block:: python # settings.py - INSTALLED_APPS = ( + INSTALLED_APPS = [ ... "import_export", "import_export_extensions", - ) + ] + +Finally, run the ``migrate`` and ``collectstatic`` commands: -And then run `migrate` command to create ImportJob/ExportJob models and -`collectstatic` to let Django collect package static files to use in the admin. +* ``migrate``: Creates the ImportJob and ExportJob models. +* ``collectstatic``: Allows Django to collect static files for use in the admin interface. .. code-block:: shell - $ python manage.py migrate - $ python manage.py collectstatic + python manage.py migrate + python manage.py collectstatic Celery ------ -You need to `set up Celery `_ -to use background import/export. Just plug in Celery and you don't need additional -settings. - - -From sources ------------- - -The sources for django-import-export-extensions can be downloaded from the `Github repo`_. - -You can either clone the public repository: - -.. code-block:: console - - $ git clone https://github.com/saritasa-nest/django-import-export-extensions - -Or download the `tarball`_: - -.. code-block:: console - - $ curl -OJL https://github.com/saritasa-nest/django-import-export-extensions/tarball/master - -Once you have a copy of the source, you can install it with: - -.. code-block:: console - - $ make install - - -.. _Github repo: https://github.com/saritasa-nest/django-import-export-extensions -.. _tarball: https://github.com/saritasa-nest/django-import-export-extensions/tarball/master +To use background import/export, you need to +`set up Celery `_. +Once Celery is set up, no additional configuration is required. Settings ------------- -You can configure the following in your settings file: +You can configure the following settings in your Django settings file: ``IMPORT_EXPORT_MAX_DATASET_ROWS`` -~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -Sets the maximum number of lines in the file to import in order to avoid memory -overflow. The default value is 10,000. If there are more lines in the file, -a ``ValueError`` exception will be raised on import. +Defines the maximum number of rows allowed in a file for import, helping to avoid memory overflow. +The default value is 10,000. If the file exceeds this limit, a ``ValueError`` exception +will be raised during the import process. ``MIME_TYPES_MAP`` ~~~~~~~~~~~~~~~~~~ Mapping file extensions to mime types to import files. -Default is `mimetypes.types_map `_. - +By default, it uses the `mimetypes.types_map `_ +from Python's mimetypes module. Settings from django-import-export ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +Additionally, the package supports settings from the original django-import-export package. +For full details on these settings, refer to the `official documentation `_. -There are also available `settings from original django-import-export -`_ -package. - -Only ``IMPORT_EXPORT_TMP_STORAGE_CLASS`` setting does not affect anything, because the storage -is not used in ``CeleryImportAdminMixin`` implementation. +**Note**: The only setting that does not affect functionality in this package is ``IMPORT_EXPORT_TMP_STORAGE_CLASS``, +as the storage is not used in the implementation of ``CeleryImportAdminMixin``. diff --git a/docs/migrate_from_original_import_export.rst b/docs/migrate_from_original_import_export.rst index 84cbfb5..f9c664c 100644 --- a/docs/migrate_from_original_import_export.rst +++ b/docs/migrate_from_original_import_export.rst @@ -2,16 +2,16 @@ Migrate from original `django-import-export` package ==================================================== -If you are already using ``django-import-export`` and want to use ``django-import-export-extensions``, -you can easily switch to it and get the benefits of background import/exporting. -First of all, install the package according to :ref:`the instruction`. -And now, all you need is to change base classes of resources and admin models. +If you're already using ``django-import-export`` and want to take advantage of +``django-import-export-extensions`` for background import/export, the transition is simple. First, +install the package following the provided :ref:`the instruction`. +Then, all you need to do is update the base classes for your resource and admin models. Migrate resources ----------------- -To use resources that provides import/export via Celery just change base resource classes from -original package to ``CeleryResource`` / ``CeleryModelResource`` from ``django-import-export-extensions``: +To enable import/export via Celery, simply replace the base resource classes from the original package +with ``CeleryResource`` or ``CeleryModelResource`` from ``django-import-export-extensions``: .. code-block:: diff :emphasize-lines: 2,6,13 @@ -65,13 +65,16 @@ Then you also need to change admin mixins to use celery import/export via Django If you only need import (or export) functionality, you can use ``CeleryImportAdminMixin`` (``CeleryExportAdminMixin``) instead of ``CeleryImportExportMixin``. +If you only need import (or export) functionality, you can use the ``CeleryImportAdminMixin`` +(or ``CeleryExportAdminMixin``) instead of the ``CeleryImportExportMixin``. + Migrate custom import/export ---------------------------- -Background import/export is implemented based on ``ImportJob``/``ExportJob`` models. So simple -``resource.export()`` won't trigger a celery task, it works exactly the same as the original -``Resource.export()`` method. To start background import/export, you need to create objects of -import/export job: +Background import/export is implemented using the ``ImportJob`` and ``ExportJob`` models. +As a result, calling the simple ``resource.export()`` will not trigger a Celery task — it behaves +exactly like the original ``Resource.export()`` method. To initiate background import/export, +you need to create instances of the import/export job: .. code-block:: python :linenos: @@ -88,8 +91,9 @@ import/export job: >>> export_job.export_status 'CREATED' -Using the ``export_status`` (``import_status``) property of the model, you can check the current status of the job. -There is also a ``progress`` property that returns information about the total number and number of completed rows. +You can check the current status of the job using the ``export_status`` (or ``import_status``) +property of the model. Additionally, the ``progress`` property provides information about the total +number of rows and the number of rows that have been completed. .. code-block:: python :linenos: diff --git a/poetry.lock b/poetry.lock index 3f20754..ea8b716 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry 1.8.4 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.8.3 and should not be changed by hand. [[package]] name = "alabaster" @@ -62,13 +62,13 @@ test = ["astroid (>=1,<2)", "astroid (>=2,<4)", "pytest"] [[package]] name = "async-timeout" -version = "5.0.0" +version = "5.0.1" description = "Timeout context manager for asyncio programs" optional = false python-versions = ">=3.8" files = [ - {file = "async_timeout-5.0.0-py3-none-any.whl", hash = "sha256:904719a4bd6e0520047d0ddae220aabee67b877f7ca17bf8cea20f67f6247ae0"}, - {file = "async_timeout-5.0.0.tar.gz", hash = "sha256:49675ec889daacfe65ff66d2dde7dd1447a6f4b2f23721022e4ba121f8772a85"}, + {file = "async_timeout-5.0.1-py3-none-any.whl", hash = "sha256:39e3809566ff85354557ec2398b55e096c8364bacac9405a7a1fa429e77fe76c"}, + {file = "async_timeout-5.0.1.tar.gz", hash = "sha256:d9321a7a3d5a6a5e187e824d2fa0793ce379a202935782d555d6e9d2735677d3"}, ] [[package]] @@ -518,13 +518,13 @@ files = [ [[package]] name = "django" -version = "5.1.2" +version = "5.1.3" description = "A high-level Python web framework that encourages rapid development and clean, pragmatic design." optional = false python-versions = ">=3.10" files = [ - {file = "Django-5.1.2-py3-none-any.whl", hash = "sha256:f11aa87ad8d5617171e3f77e1d5d16f004b79a2cf5d2e1d2b97a6a1f8e9ba5ed"}, - {file = "Django-5.1.2.tar.gz", hash = "sha256:bd7376f90c99f96b643722eee676498706c9fd7dc759f55ebfaf2c08ebcdf4f0"}, + {file = "Django-5.1.3-py3-none-any.whl", hash = "sha256:8b38a9a12da3ae00cb0ba72da985ec4b14de6345046b1e174b1fd7254398f818"}, + {file = "Django-5.1.3.tar.gz", hash = "sha256:c0fa0e619c39325a169208caef234f90baa925227032ad3f44842ba14d75234a"}, ] [package.dependencies] @@ -1685,114 +1685,101 @@ jupyter = ["ipywidgets (>=7.5.1,<9)"] [[package]] name = "rpds-py" -version = "0.20.1" +version = "0.21.0" description = "Python bindings to Rust's persistent data structures (rpds)" optional = false -python-versions = ">=3.8" +python-versions = ">=3.9" files = [ - {file = "rpds_py-0.20.1-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:a649dfd735fff086e8a9d0503a9f0c7d01b7912a333c7ae77e1515c08c146dad"}, - {file = "rpds_py-0.20.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:f16bc1334853e91ddaaa1217045dd7be166170beec337576818461268a3de67f"}, - {file = "rpds_py-0.20.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:14511a539afee6f9ab492b543060c7491c99924314977a55c98bfa2ee29ce78c"}, - {file = "rpds_py-0.20.1-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:3ccb8ac2d3c71cda472b75af42818981bdacf48d2e21c36331b50b4f16930163"}, - {file = "rpds_py-0.20.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c142b88039b92e7e0cb2552e8967077e3179b22359e945574f5e2764c3953dcf"}, - {file = "rpds_py-0.20.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f19169781dddae7478a32301b499b2858bc52fc45a112955e798ee307e294977"}, - {file = "rpds_py-0.20.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:13c56de6518e14b9bf6edde23c4c39dac5b48dcf04160ea7bce8fca8397cdf86"}, - {file = "rpds_py-0.20.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:925d176a549f4832c6f69fa6026071294ab5910e82a0fe6c6228fce17b0706bd"}, - {file = "rpds_py-0.20.1-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:78f0b6877bfce7a3d1ff150391354a410c55d3cdce386f862926a4958ad5ab7e"}, - {file = "rpds_py-0.20.1-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:3dd645e2b0dcb0fd05bf58e2e54c13875847687d0b71941ad2e757e5d89d4356"}, - {file = "rpds_py-0.20.1-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:4f676e21db2f8c72ff0936f895271e7a700aa1f8d31b40e4e43442ba94973899"}, - {file = "rpds_py-0.20.1-cp310-none-win32.whl", hash = "sha256:648386ddd1e19b4a6abab69139b002bc49ebf065b596119f8f37c38e9ecee8ff"}, - {file = "rpds_py-0.20.1-cp310-none-win_amd64.whl", hash = "sha256:d9ecb51120de61e4604650666d1f2b68444d46ae18fd492245a08f53ad2b7711"}, - {file = "rpds_py-0.20.1-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:762703bdd2b30983c1d9e62b4c88664df4a8a4d5ec0e9253b0231171f18f6d75"}, - {file = "rpds_py-0.20.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:0b581f47257a9fce535c4567782a8976002d6b8afa2c39ff616edf87cbeff712"}, - {file = "rpds_py-0.20.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:842c19a6ce894493563c3bd00d81d5100e8e57d70209e84d5491940fdb8b9e3a"}, - {file = "rpds_py-0.20.1-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:42cbde7789f5c0bcd6816cb29808e36c01b960fb5d29f11e052215aa85497c93"}, - {file = "rpds_py-0.20.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:6c8e9340ce5a52f95fa7d3b552b35c7e8f3874d74a03a8a69279fd5fca5dc751"}, - {file = "rpds_py-0.20.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8ba6f89cac95c0900d932c9efb7f0fb6ca47f6687feec41abcb1bd5e2bd45535"}, - {file = "rpds_py-0.20.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4a916087371afd9648e1962e67403c53f9c49ca47b9680adbeef79da3a7811b0"}, - {file = "rpds_py-0.20.1-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:200a23239781f46149e6a415f1e870c5ef1e712939fe8fa63035cd053ac2638e"}, - {file = "rpds_py-0.20.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:58b1d5dd591973d426cbb2da5e27ba0339209832b2f3315928c9790e13f159e8"}, - {file = "rpds_py-0.20.1-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:6b73c67850ca7cae0f6c56f71e356d7e9fa25958d3e18a64927c2d930859b8e4"}, - {file = "rpds_py-0.20.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:d8761c3c891cc51e90bc9926d6d2f59b27beaf86c74622c8979380a29cc23ac3"}, - {file = "rpds_py-0.20.1-cp311-none-win32.whl", hash = "sha256:cd945871335a639275eee904caef90041568ce3b42f402c6959b460d25ae8732"}, - {file = "rpds_py-0.20.1-cp311-none-win_amd64.whl", hash = "sha256:7e21b7031e17c6b0e445f42ccc77f79a97e2687023c5746bfb7a9e45e0921b84"}, - {file = "rpds_py-0.20.1-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:36785be22066966a27348444b40389f8444671630063edfb1a2eb04318721e17"}, - {file = "rpds_py-0.20.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:142c0a5124d9bd0e2976089484af5c74f47bd3298f2ed651ef54ea728d2ea42c"}, - {file = "rpds_py-0.20.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dbddc10776ca7ebf2a299c41a4dde8ea0d8e3547bfd731cb87af2e8f5bf8962d"}, - {file = "rpds_py-0.20.1-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:15a842bb369e00295392e7ce192de9dcbf136954614124a667f9f9f17d6a216f"}, - {file = "rpds_py-0.20.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:be5ef2f1fc586a7372bfc355986226484e06d1dc4f9402539872c8bb99e34b01"}, - {file = "rpds_py-0.20.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:dbcf360c9e3399b056a238523146ea77eeb2a596ce263b8814c900263e46031a"}, - {file = "rpds_py-0.20.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ecd27a66740ffd621d20b9a2f2b5ee4129a56e27bfb9458a3bcc2e45794c96cb"}, - {file = "rpds_py-0.20.1-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:d0b937b2a1988f184a3e9e577adaa8aede21ec0b38320d6009e02bd026db04fa"}, - {file = "rpds_py-0.20.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:6889469bfdc1eddf489729b471303739bf04555bb151fe8875931f8564309afc"}, - {file = "rpds_py-0.20.1-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:19b73643c802f4eaf13d97f7855d0fb527fbc92ab7013c4ad0e13a6ae0ed23bd"}, - {file = "rpds_py-0.20.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:3c6afcf2338e7f374e8edc765c79fbcb4061d02b15dd5f8f314a4af2bdc7feb5"}, - {file = "rpds_py-0.20.1-cp312-none-win32.whl", hash = "sha256:dc73505153798c6f74854aba69cc75953888cf9866465196889c7cdd351e720c"}, - {file = "rpds_py-0.20.1-cp312-none-win_amd64.whl", hash = "sha256:8bbe951244a838a51289ee53a6bae3a07f26d4e179b96fc7ddd3301caf0518eb"}, - {file = "rpds_py-0.20.1-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:6ca91093a4a8da4afae7fe6a222c3b53ee4eef433ebfee4d54978a103435159e"}, - {file = "rpds_py-0.20.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:b9c2fe36d1f758b28121bef29ed1dee9b7a2453e997528e7d1ac99b94892527c"}, - {file = "rpds_py-0.20.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f009c69bc8c53db5dfab72ac760895dc1f2bc1b62ab7408b253c8d1ec52459fc"}, - {file = "rpds_py-0.20.1-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:6740a3e8d43a32629bb9b009017ea5b9e713b7210ba48ac8d4cb6d99d86c8ee8"}, - {file = "rpds_py-0.20.1-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:32b922e13d4c0080d03e7b62991ad7f5007d9cd74e239c4b16bc85ae8b70252d"}, - {file = "rpds_py-0.20.1-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fe00a9057d100e69b4ae4a094203a708d65b0f345ed546fdef86498bf5390982"}, - {file = "rpds_py-0.20.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:49fe9b04b6fa685bd39237d45fad89ba19e9163a1ccaa16611a812e682913496"}, - {file = "rpds_py-0.20.1-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:aa7ac11e294304e615b43f8c441fee5d40094275ed7311f3420d805fde9b07b4"}, - {file = "rpds_py-0.20.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:6aa97af1558a9bef4025f8f5d8c60d712e0a3b13a2fe875511defc6ee77a1ab7"}, - {file = "rpds_py-0.20.1-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:483b29f6f7ffa6af845107d4efe2e3fa8fb2693de8657bc1849f674296ff6a5a"}, - {file = "rpds_py-0.20.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:37fe0f12aebb6a0e3e17bb4cd356b1286d2d18d2e93b2d39fe647138458b4bcb"}, - {file = "rpds_py-0.20.1-cp313-none-win32.whl", hash = "sha256:a624cc00ef2158e04188df5e3016385b9353638139a06fb77057b3498f794782"}, - {file = "rpds_py-0.20.1-cp313-none-win_amd64.whl", hash = "sha256:b71b8666eeea69d6363248822078c075bac6ed135faa9216aa85f295ff009b1e"}, - {file = "rpds_py-0.20.1-cp38-cp38-macosx_10_12_x86_64.whl", hash = "sha256:5b48e790e0355865197ad0aca8cde3d8ede347831e1959e158369eb3493d2191"}, - {file = "rpds_py-0.20.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:3e310838a5801795207c66c73ea903deda321e6146d6f282e85fa7e3e4854804"}, - {file = "rpds_py-0.20.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2249280b870e6a42c0d972339e9cc22ee98730a99cd7f2f727549af80dd5a963"}, - {file = "rpds_py-0.20.1-cp38-cp38-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:e79059d67bea28b53d255c1437b25391653263f0e69cd7dec170d778fdbca95e"}, - {file = "rpds_py-0.20.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2b431c777c9653e569986ecf69ff4a5dba281cded16043d348bf9ba505486f36"}, - {file = "rpds_py-0.20.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:da584ff96ec95e97925174eb8237e32f626e7a1a97888cdd27ee2f1f24dd0ad8"}, - {file = "rpds_py-0.20.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:02a0629ec053fc013808a85178524e3cb63a61dbc35b22499870194a63578fb9"}, - {file = "rpds_py-0.20.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:fbf15aff64a163db29a91ed0868af181d6f68ec1a3a7d5afcfe4501252840bad"}, - {file = "rpds_py-0.20.1-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:07924c1b938798797d60c6308fa8ad3b3f0201802f82e4a2c41bb3fafb44cc28"}, - {file = "rpds_py-0.20.1-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:4a5a844f68776a7715ecb30843b453f07ac89bad393431efbf7accca3ef599c1"}, - {file = "rpds_py-0.20.1-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:518d2ca43c358929bf08f9079b617f1c2ca6e8848f83c1225c88caeac46e6cbc"}, - {file = "rpds_py-0.20.1-cp38-none-win32.whl", hash = "sha256:3aea7eed3e55119635a74bbeb80b35e776bafccb70d97e8ff838816c124539f1"}, - {file = "rpds_py-0.20.1-cp38-none-win_amd64.whl", hash = "sha256:7dca7081e9a0c3b6490a145593f6fe3173a94197f2cb9891183ef75e9d64c425"}, - {file = "rpds_py-0.20.1-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:b41b6321805c472f66990c2849e152aff7bc359eb92f781e3f606609eac877ad"}, - {file = "rpds_py-0.20.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:0a90c373ea2975519b58dece25853dbcb9779b05cc46b4819cb1917e3b3215b6"}, - {file = "rpds_py-0.20.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:16d4477bcb9fbbd7b5b0e4a5d9b493e42026c0bf1f06f723a9353f5153e75d30"}, - {file = "rpds_py-0.20.1-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:84b8382a90539910b53a6307f7c35697bc7e6ffb25d9c1d4e998a13e842a5e83"}, - {file = "rpds_py-0.20.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4888e117dd41b9d34194d9e31631af70d3d526efc363085e3089ab1a62c32ed1"}, - {file = "rpds_py-0.20.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5265505b3d61a0f56618c9b941dc54dc334dc6e660f1592d112cd103d914a6db"}, - {file = "rpds_py-0.20.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e75ba609dba23f2c95b776efb9dd3f0b78a76a151e96f96cc5b6b1b0004de66f"}, - {file = "rpds_py-0.20.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:1791ff70bc975b098fe6ecf04356a10e9e2bd7dc21fa7351c1742fdeb9b4966f"}, - {file = "rpds_py-0.20.1-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:d126b52e4a473d40232ec2052a8b232270ed1f8c9571aaf33f73a14cc298c24f"}, - {file = "rpds_py-0.20.1-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:c14937af98c4cc362a1d4374806204dd51b1e12dded1ae30645c298e5a5c4cb1"}, - {file = "rpds_py-0.20.1-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:3d089d0b88996df627693639d123c8158cff41c0651f646cd8fd292c7da90eaf"}, - {file = "rpds_py-0.20.1-cp39-none-win32.whl", hash = "sha256:653647b8838cf83b2e7e6a0364f49af96deec64d2a6578324db58380cff82aca"}, - {file = "rpds_py-0.20.1-cp39-none-win_amd64.whl", hash = "sha256:fa41a64ac5b08b292906e248549ab48b69c5428f3987b09689ab2441f267d04d"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:7a07ced2b22f0cf0b55a6a510078174c31b6d8544f3bc00c2bcee52b3d613f74"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:68cb0a499f2c4a088fd2f521453e22ed3527154136a855c62e148b7883b99f9a"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fa3060d885657abc549b2a0f8e1b79699290e5d83845141717c6c90c2df38311"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:95f3b65d2392e1c5cec27cff08fdc0080270d5a1a4b2ea1d51d5f4a2620ff08d"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2cc3712a4b0b76a1d45a9302dd2f53ff339614b1c29603a911318f2357b04dd2"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5d4eea0761e37485c9b81400437adb11c40e13ef513375bbd6973e34100aeb06"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7f5179583d7a6cdb981151dd349786cbc318bab54963a192692d945dd3f6435d"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2fbb0ffc754490aff6dabbf28064be47f0f9ca0b9755976f945214965b3ace7e"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:a94e52537a0e0a85429eda9e49f272ada715506d3b2431f64b8a3e34eb5f3e75"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-musllinux_1_2_i686.whl", hash = "sha256:92b68b79c0da2a980b1c4197e56ac3dd0c8a149b4603747c4378914a68706979"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:93da1d3db08a827eda74356f9f58884adb254e59b6664f64cc04cdff2cc19b0d"}, - {file = "rpds_py-0.20.1-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:754bbed1a4ca48479e9d4182a561d001bbf81543876cdded6f695ec3d465846b"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:ca449520e7484534a2a44faf629362cae62b660601432d04c482283c47eaebab"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:9c4cb04a16b0f199a8c9bf807269b2f63b7b5b11425e4a6bd44bd6961d28282c"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bb63804105143c7e24cee7db89e37cb3f3941f8e80c4379a0b355c52a52b6780"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:55cd1fa4ecfa6d9f14fbd97ac24803e6f73e897c738f771a9fe038f2f11ff07c"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:0f8f741b6292c86059ed175d80eefa80997125b7c478fb8769fd9ac8943a16c0"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0fc212779bf8411667234b3cdd34d53de6c2b8b8b958e1e12cb473a5f367c338"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0ad56edabcdb428c2e33bbf24f255fe2b43253b7d13a2cdbf05de955217313e6"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:0a3a1e9ee9728b2c1734f65d6a1d376c6f2f6fdcc13bb007a08cc4b1ff576dc5"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:e13de156137b7095442b288e72f33503a469aa1980ed856b43c353ac86390519"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-musllinux_1_2_i686.whl", hash = "sha256:07f59760ef99f31422c49038964b31c4dfcfeb5d2384ebfc71058a7c9adae2d2"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:59240685e7da61fb78f65a9f07f8108e36a83317c53f7b276b4175dc44151684"}, - {file = "rpds_py-0.20.1-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:83cba698cfb3c2c5a7c3c6bac12fe6c6a51aae69513726be6411076185a8b24a"}, - {file = "rpds_py-0.20.1.tar.gz", hash = "sha256:e1791c4aabd117653530dccd24108fa03cc6baf21f58b950d0a73c3b3b29a350"}, + {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"}, ] [[package]] @@ -2196,13 +2183,13 @@ files = [ [[package]] name = "werkzeug" -version = "3.1.1" +version = "3.1.2" description = "The comprehensive WSGI web application library." optional = false python-versions = ">=3.9" files = [ - {file = "werkzeug-3.1.1-py3-none-any.whl", hash = "sha256:a71124d1ef06008baafa3d266c02f56e1836a5984afd6dd6c9230669d60d9fb5"}, - {file = "werkzeug-3.1.1.tar.gz", hash = "sha256:8cd39dfbdfc1e051965f156163e2974e52c210f130810e9ad36858f0fd3edad4"}, + {file = "werkzeug-3.1.2-py3-none-any.whl", hash = "sha256:4f7d1a5de312c810a8a2c6f0b47e9f6a7cffb7c8322def35e4d4d9841ff85597"}, + {file = "werkzeug-3.1.2.tar.gz", hash = "sha256:f471a4cd167233077e9d2a8190c3471c5bc520c636a9e3c1e9300c33bced03bc"}, ] [package.dependencies] @@ -2241,4 +2228,4 @@ files = [ [metadata] lock-version = "2.0" python-versions = "^3.10" -content-hash = "52e1d720ee281498a7707514e1e27a479ad8a8582744385c3380cfb6d911ece7" +content-hash = "e13604513d0d3837a6f58576b4c4bcd5b6346d6c9c987ad9caf807a1a6fa5d97" diff --git a/pyproject.toml b/pyproject.toml index 70e0014..4fb8239 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -10,7 +10,7 @@ maintainers = [ "Stanislav Khlud ", "Egor Toryshak