mirror of
				https://github.com/ilri/dspace-statistics-api.git
				synced 2025-10-22 09:21:23 +02:00 
			
		
		
		
	Compare commits
	
		
			240 Commits
		
	
	
		
			v1.3.1
			...
			5031e31609
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 5031e31609 | |||
| 62c7779b73 | |||
|  | 3fb3e081c4 | ||
|  | 8f8739b95e | ||
| a0dc1af45f | |||
| f26b55bc5f | |||
| 7b8ff7c02f | |||
| 6313177b4b | |||
| fb5df25ef5 | |||
| dab96ce77a | |||
| ff7641d455 | |||
| bfe12da212 | |||
| 68cb3905d1 | |||
| 41d831d29f | |||
| ce0a2d9213 | |||
| 8c1a8bde54 | |||
| ffe6030a9e | |||
| e90cfe9e69 | |||
| 719d86dd30 | |||
|  | d3982df636 | ||
| 656661de8f | |||
| 1e251a259a | |||
| eca0d28849 | |||
| 718bfab897 | |||
| 8196d28e88 | |||
| f3421e595c | |||
| 6cf8ca0245 | |||
| 14c6e5f8dc | |||
| 4c32aeb915 | |||
| 34a1a08893 | |||
| c47bb2aba7 | |||
| a7fd70bf10 | |||
| 45dfe7851f | |||
|  | c1cd0a0351 | ||
| 1912363899 | |||
| cd3c024a77 | |||
|  | e96c79bf2c | ||
|  | d6330c7bd4 | ||
| 8c7a5c4047 | |||
| a31c592fab | |||
|  | c7b179f1b5 | ||
|  | 77c166c024 | ||
| 7680b0f440 | |||
| e70a7a9675 | |||
| 24f90df13e | |||
| 780f2c1723 | |||
| 53b58d4116 | |||
|  | 19a6d2cea6 | ||
| 6c2bcda16f | |||
|  | e4d9545b02 | ||
| 1f507d3074 | |||
| 82771d7b0c | |||
| 5ff3323f88 | |||
| c7a871c2f1 | |||
|  | b948283d40 | ||
| 124a05dcaf | |||
| a2daf96fec | |||
| 8634d53fa6 | |||
|  | e2bfcef573 | ||
|  | d64c4b8cbc | ||
|  | 3d91366412 | ||
|  | c3a4e2260b | ||
|  | 10519997ac | ||
|  | 4d7e9e9401 | ||
| fe9f98bcc0 | |||
| 70f0d66c6e | |||
| 913596c61d | |||
| 7cd762a5a2 | |||
|  | 3811be18ef | ||
| a52818271c | |||
|  | b643f60dd7 | ||
| 7cec9a9545 | |||
|  | a9302506b6 | ||
| b980602a03 | |||
| a4b4843036 | |||
| 7e334f6de8 | |||
| 770f676fb5 | |||
| 6d5e3c350d | |||
| 531136183b | |||
| 1a3d0350a5 | |||
| 25c4f05f16 | |||
| 9fba8d1b81 | |||
| 568ced0f20 | |||
| 9cd93c9034 | |||
| 83a2625987 | |||
| f591ed7162 | |||
| bb0f267941 | |||
| 0720605b6a | |||
| bcb97d025c | |||
| 0ff8490275 | |||
| 0a8ac60ade | |||
| 37527c21be | |||
| eb660f8085 | |||
| e7d780f511 | |||
| c3b9a541b7 | |||
| 1a1a14a25f | |||
| c09fc789e8 | |||
| 134a4f1595 | |||
| 12ebd1aed5 | |||
| e5f3201b65 | |||
| c1ce4fe233 | |||
| b2eb1878a5 | |||
| a0213c1c97 | |||
| cd03ca2b36 | |||
| c48e6a79c7 | |||
| a2e1695ecc | |||
| b683bf211c | |||
| 3ab48743d6 | |||
| 88173eaae9 | |||
| f557d33f36 | |||
| ffc4ff4a5c | |||
| 7551b34632 | |||
| 5e71ec10eb | |||
| f80d360cf9 | |||
| e70b59ecfe | |||
| 4d0828b6c0 | |||
| dabc4c0259 | |||
| 4fd8af07c3 | |||
| 4c5326a176 | |||
| 3b1ccafab4 | |||
| 58b5ae82d3 | |||
| 562aaeef7d | |||
| 5cdba6acb1 | |||
| dd0937179c | |||
| f0c6c004db | |||
| 6843f0a8ac | |||
| f5fcfcc05a | |||
| e8ac74b6d1 | |||
| 14fc14daee | |||
| 871aae537a | |||
| 2fada6c6ff | |||
| ef0991e352 | |||
| 4502d6053c | |||
| a524068cf6 | |||
| 964d5dff06 | |||
| a9252d1771 | |||
| a63687d516 | |||
| 73dc3a292e | |||
| 1e742bad41 | |||
| 164008981e | |||
| dd1769b954 | |||
| b009820fb4 | |||
| 9830295978 | |||
| c93a4d7455 | |||
| 2f8e4f8a0a | |||
| 0650c5985e | |||
| d814f1c4f0 | |||
| 00f30591c4 | |||
| acfe87b91a | |||
| bc6d84dda2 | |||
| 889fb2f74a | |||
| c42cd7a818 | |||
| f8bba59d66 | |||
| b8cb752a29 | |||
| 09496aa2b5 | |||
| ff5dc7506d | |||
| 80a11ead97 | |||
| a282c95933 | |||
| fd7cc36306 | |||
| a20ff09570 | |||
| fdc0e73088 | |||
| b15afc9f39 | |||
| 2bc18ef719 | |||
| 49751b53f0 | |||
| d1c177e146 | |||
| 33dc210452 | |||
| 282d5f644a | |||
| 05e0e8bdca | |||
| 2567bb8604 | |||
| 4af3c656a3 | |||
| 4f8cd1097b | |||
| a02211fd60 | |||
| fc814593c7 | |||
| 7de1084f60 | |||
| 6b78e82fe9 | |||
| 4004515967 | |||
| d1229c2387 | |||
| be83514de1 | |||
| 70b2ba83ba | |||
| 893039bc6a | |||
| a4628dde4e | |||
| 68418ea053 | |||
| 6bbee7919e | |||
| 8f0061ce29 | |||
| 4b1398c67f | |||
| a9d2a6d9be | |||
| a35ecf2394 | |||
| 3e271c7852 | |||
| d7ba14c590 | |||
| ab82e90773 | |||
| 8a1244d2d0 | |||
| 04f0756c7f | |||
| 830e4415f5 | |||
| 47b4eb3df7 | |||
| 3339bf8d9c | |||
| fba6f1ead1 | |||
| 20c8ba0cf8 | |||
| b486f51dd7 | |||
| 787eec20ea | |||
| 9e6fcf279b | |||
| 4dbf734a4b | |||
| a0d0a47150 | |||
| 01e9756cf2 | |||
| b2b4eb2939 | |||
| 4bbbaa4af3 | |||
| 7e4d5f4b13 | |||
| 428172854d | |||
| 2707cb37d5 | |||
| 2407aeec70 | |||
| f3a0e3a671 | |||
| 4590fc8708 | |||
| 8b924cf450 | |||
| ea24c73a6a | |||
| cd98d33615 | |||
| 9d112266ca | |||
| 2b067050ff | |||
| dc683f2d1c | |||
| f60f529bd7 | |||
| 7db8458201 | |||
| 707f878b94 | |||
| 930250352a | |||
| e27f30ba4d | |||
| 28d1917038 | |||
| fc6a9c2ad1 | |||
| 3125e96a16 | |||
| 66143ff00f | |||
| 2d15f12be9 | |||
| 9218039e61 | |||
| 88a8db6c78 | |||
| 3995eba0a7 | |||
| 810508d038 | |||
| ecafab57cb | |||
| 9c9431b58c | |||
| 2d6520fc97 | |||
| 79a393d33f | |||
| 149f6c418f | |||
| ca1582a8b6 | |||
| 1904c243a4 | |||
| 0baa07f70a | |||
| 59214ffcb6 | 
							
								
								
									
										21
									
								
								.build.yml
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								.build.yml
									
									
									
									
									
								
							| @@ -1,21 +0,0 @@ | |||||||
| image: archlinux |  | ||||||
| packages: |  | ||||||
|   - python-poetry |  | ||||||
|   - postgresql |  | ||||||
| sources: |  | ||||||
|   - https://git.sr.ht/~alanorth/dspace-statistics-api |  | ||||||
| tasks: |  | ||||||
|   - setup: | |  | ||||||
|       id |  | ||||||
|       psql --version |  | ||||||
|       sudo su - postgres -c "initdb --locale en_US.UTF-8 -E UTF8 -D '/var/lib/postgres/data'" |  | ||||||
|       sudo systemctl start postgresql |  | ||||||
|       createuser -U postgres dspacestatistics |  | ||||||
|       psql -U postgres -c "ALTER USER dspacestatistics WITH PASSWORD 'dspacestatistics'" |  | ||||||
|       createdb -U postgres -O dspacestatistics --encoding=UNICODE dspacestatistics |  | ||||||
|       cd dspace-statistics-api |  | ||||||
|       psql -U postgres -d dspacestatistics < tests/dspacestatistics.sql |  | ||||||
|       poetry install --no-root |  | ||||||
|   - test: | |  | ||||||
|       cd dspace-statistics-api |  | ||||||
|       poetry run pytest |  | ||||||
							
								
								
									
										56
									
								
								.github/workflows/python-app.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								.github/workflows/python-app.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | |||||||
|  | # This workflow will install Python dependencies, run tests and lint with a single version of Python | ||||||
|  | # For more information see: https://help.github.com/actions/language-and-framework-guides/using-python-with-github-actions | ||||||
|  |  | ||||||
|  | name: Build and Test | ||||||
|  |  | ||||||
|  | on: ['push', 'pull_request'] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  |  | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     services: | ||||||
|  |       database: | ||||||
|  |         image: postgres:17-alpine | ||||||
|  |         env: | ||||||
|  |           # password for postgres user in the Docker container | ||||||
|  |           POSTGRES_PASSWORD: postgres | ||||||
|  |           # default database to create | ||||||
|  |           POSTGRES_DB: dspacestatistics | ||||||
|  |         options: >- | ||||||
|  |           --health-cmd pg_isready | ||||||
|  |           --health-interval 10s | ||||||
|  |           --health-timeout 5s | ||||||
|  |           --health-retries 5 | ||||||
|  |         ports: | ||||||
|  |           - 5432:5432 | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v5 | ||||||
|  |     - name: Install uv | ||||||
|  |       uses: astral-sh/setup-uv@v6 | ||||||
|  |       with: | ||||||
|  |         version: 'latest' | ||||||
|  |     - run: uv sync | ||||||
|  |     - name: Lint with flake8 | ||||||
|  |       run: | | ||||||
|  |         # stop the build if there are Python syntax errors or undefined names | ||||||
|  |         uv run flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics | ||||||
|  |         # exit-zero treats all errors as warnings. The GitHub editor is 127 chars wide | ||||||
|  |         uv run flake8 . --count --exit-zero --max-complexity=10 --max-line-length=127 --statistics | ||||||
|  |     - name: Set up PostgreSQL | ||||||
|  |       run: | | ||||||
|  |         pg_isready -U postgres -d dspacestatistics | ||||||
|  |         createuser -U postgres dspacestatistics | ||||||
|  |         psql -U postgres -c "ALTER USER dspacestatistics WITH PASSWORD 'dspacestatistics'" | ||||||
|  |         psql -U postgres -d dspacestatistics < tests/dspacestatistics.sql | ||||||
|  |       env: | ||||||
|  |         PGHOST: localhost | ||||||
|  |         PGPASSWORD: postgres | ||||||
|  |     - name: Test with pytest | ||||||
|  |       run: | | ||||||
|  |         uv run pytest | ||||||
|  |       env: | ||||||
|  |         PGHOST: localhost | ||||||
|  |         PGPASSWORD: dspacestatistics | ||||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,2 +1,3 @@ | |||||||
| __pycache__ | __pycache__ | ||||||
| venv | venv | ||||||
|  | *.egg-info | ||||||
|   | |||||||
| @@ -1,4 +0,0 @@ | |||||||
| flake8: |  | ||||||
|     enabled: true |  | ||||||
|     config_file: .flake8 |  | ||||||
|     fail_on_violations: true |  | ||||||
							
								
								
									
										24
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,24 +0,0 @@ | |||||||
| dist: bionic |  | ||||||
| language: python |  | ||||||
| python: |  | ||||||
|   - "3.6" |  | ||||||
|   - "3.7" |  | ||||||
|   - "3.8" |  | ||||||
|   - "3.8-dev"  # 3.8 development branch |  | ||||||
| jobs: |  | ||||||
|   allow_failures: |  | ||||||
|     - python: "3.8-dev" |  | ||||||
| addons: |  | ||||||
|   postgresql: "10" |  | ||||||
| before_script: |  | ||||||
|   - psql --version |  | ||||||
|   - createuser -U postgres dspacestatistics |  | ||||||
|   - psql -U postgres -c "ALTER USER dspacestatistics WITH PASSWORD 'dspacestatistics'" |  | ||||||
|   - createdb -U postgres -O dspacestatistics --encoding=UNICODE dspacestatistics |  | ||||||
|   - psql -U postgres -d dspacestatistics < tests/dspacestatistics.sql |  | ||||||
| install: |  | ||||||
|   - "pip install -r requirements.txt" |  | ||||||
|   - "pip install -r requirements-dev.txt" |  | ||||||
| script: pytest |  | ||||||
|  |  | ||||||
| # vim: ts=2 sw=2 et |  | ||||||
							
								
								
									
										78
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										78
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -4,6 +4,80 @@ All notable changes to this project will be documented in this file. | |||||||
| The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | ||||||
| and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | ||||||
|  |  | ||||||
|  | ### Unreleased | ||||||
|  |  | ||||||
|  | ## 1.4.5 - 2025-04-02 | ||||||
|  | ### Changed | ||||||
|  | - Use uv instead of poetry to manage the project and dependencies | ||||||
|  | - Update recommended Python version to 3.9+ | ||||||
|  |  | ||||||
|  | ### Updated | ||||||
|  | - Falcon 4.0.x | ||||||
|  | - psycopg2 to psycopg (version 3.2.6) | ||||||
|  |  | ||||||
|  | ## 1.4.4 - 2024-09-11 | ||||||
|  | ### Changed | ||||||
|  | - Update recommended Python version to 3.8+ | ||||||
|  | - Use PostgreSQL 15 in CI | ||||||
|  | - Use Python 3.12 in CI | ||||||
|  |  | ||||||
|  | ### Updated | ||||||
|  | - Falcon 3.1.3, a minor change for us, but good to be using a current upstream | ||||||
|  | version | ||||||
|  |  | ||||||
|  | ### Removed | ||||||
|  | - Drone CI | ||||||
|  |  | ||||||
|  | ## 1.4.3 - 2022-03-26 | ||||||
|  | ### Updated | ||||||
|  | - Update dependencies with `poetry update` | ||||||
|  | - Falcon 3.1.0, a minor change for us, but good to be using a current upstream | ||||||
|  | version | ||||||
|  |  | ||||||
|  | ## 1.4.2 - 2021-04-14 | ||||||
|  | ### Updated | ||||||
|  | - Update dependencies with `poetry update` | ||||||
|  | - Falcon 3.0.0, a minor change for us, but good to be using a current upstream | ||||||
|  | version | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - Bug in several of the "valid page" tests | ||||||
|  |  | ||||||
|  | ### Added | ||||||
|  | - GitHub Actions workflow to build and test the API | ||||||
|  |  | ||||||
|  | ## [1.4.1] - 2021-01-14 | ||||||
|  | ### Changed | ||||||
|  | - Limit Solr query to UUIDs to avoid errors with unmigrated legacy stats (https://github.com/ilri/dspace-statistics-api/issues/12) | ||||||
|  |  | ||||||
|  | ### Updated | ||||||
|  | - Dev dependencies | ||||||
|  |  | ||||||
|  | ## [1.4.0] - 2020-12-27 | ||||||
|  | ### Added | ||||||
|  | - indexer.py now indexes views and downloads for communities and collections | ||||||
|  | - API endpoints for /communities, /community/id, /collections, and /collections/id | ||||||
|  | - Swagger UI interface on /swagger | ||||||
|  | - /status page which lists the API version | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - Add ORDER BY to /items resource to make sure results are returned | ||||||
|  | deterministically | ||||||
|  | - Use `fl` parameter in indexer to return only the field we are faceting by | ||||||
|  | - Minor refactoring of imports for PEP8 style | ||||||
|  | - More correct calculation of `totalPages` parameter in REST API response | ||||||
|  |  | ||||||
|  | ## [1.3.2] - 2020-11-18 | ||||||
|  | ### Fixed | ||||||
|  | - Minor issue with limit parameter (> 0) | ||||||
|  | - Minor issue with limit parameter (<= 100) | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - Minor refactor in Solr bot filtering | ||||||
|  |  | ||||||
|  | ### Updated | ||||||
|  | - Run poetry update | ||||||
|  |  | ||||||
| ## [1.3.1] - 2020-10-06 | ## [1.3.1] - 2020-10-06 | ||||||
| ### Changed | ### Changed | ||||||
| - Fix issue with requirements.txt caused by poetry's export | - Fix issue with requirements.txt caused by poetry's export | ||||||
| @@ -47,7 +121,7 @@ and gunicorn 20.0.4 | |||||||
| - Minor syntax issues highlighted by flake8 | - Minor syntax issues highlighted by flake8 | ||||||
|  |  | ||||||
| ## [1.1.0] - 2019-05-05 | ## [1.1.0] - 2019-05-05 | ||||||
| ## Updated | ### Updated | ||||||
| - Falcon 2.0.0 (@alanorth) | - Falcon 2.0.0 (@alanorth) | ||||||
|  |  | ||||||
| ## [1.0.0] - 2019-04-15 | ## [1.0.0] - 2019-04-15 | ||||||
| @@ -65,7 +139,7 @@ and gunicorn 20.0.4 | |||||||
| ## [0.9.0] - 2019-01-22 | ## [0.9.0] - 2019-01-22 | ||||||
| ### Updated | ### Updated | ||||||
| - pytest version 4.0.0 | - pytest version 4.0.0 | ||||||
| - Fix indexing of sharded statistics cores ([#10)) | - Fix indexing of sharded statistics cores (#10) | ||||||
| - Handle case of missing views/downloads gracefully | - Handle case of missing views/downloads gracefully | ||||||
|  |  | ||||||
| ## [0.8.1] - 2018-11-14 | ## [0.8.1] - 2018-11-14 | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,10 +1,17 @@ | |||||||
| # DSpace Statistics API [](https://travis-ci.org/ilri/dspace-statistics-api) [](https://builds.sr.ht/~alanorth/dspace-statistics-api?) | <h1 align="center">DSpace Statistics API</h1> | ||||||
| DSpace stores item view and download events in a Solr "statistics" core. This information is available for use in the various DSpace user interfaces, but is not exposed externally via any APIs. The DSpace 4/5/6 [REST API](https://wiki.lyrasis.org/display/DSDOC5x/REST+API), for example, only exposes information about communities, collections, item metadata, and bitstreams. |  | ||||||
|  | <p align="center"> | ||||||
|  | <a href="https://github.com/ilri/dspace-statistics-api/actions"><img alt="Build and Test" src="https://github.com/ilri/dspace-statistics-api/actions/workflows/python-app.yml/badge.svg"></a> | ||||||
|  | <a href="https://github.com/psf/black"><img alt="Code style: black" src="https://img.shields.io/badge/code%20style-black-000000.svg"></a> | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | DSpace stores item view and download events in a Solr "statistics" core. This information is available for use in the various DSpace user interfaces, but is not exposed externally via any APIs. The DSpace 4/5/6 [REST API](https://wiki.lyrasis.org/display/DSDOC5x/REST+API), for example, only exposes _metadata_ about communities, collections, items, and bitstreams. As of DSpace 7, the [REST API](https://wiki.lyrasis.org/display/DSDOC7x/REST+API) exposes _some_ statistics about DSpace objects, but does not provide a way to get them for all items, and lacks some flexibility. | ||||||
|  |  | ||||||
| - If your DSpace is version 4 or 5, use [dspace-statistics-api v1.1.1](https://github.com/ilri/dspace-statistics-api/releases/tag/v1.1.1) | - If your DSpace is version 4 or 5, use [dspace-statistics-api v1.1.1](https://github.com/ilri/dspace-statistics-api/releases/tag/v1.1.1) | ||||||
| - If your DSpace is version 6+, use [dspace-statistics-api v1.2.0 or greater](https://github.com/ilri/dspace-statistics-api/releases/tag/v1.2.0) | - If your DSpace is version 6+, use [dspace-statistics-api v1.2.0 or greater](https://github.com/ilri/dspace-statistics-api/releases/tag/v1.2.0) | ||||||
|  |   - Please make sure your statistics have been migrated from integers to UUIDs with the [solr-upgrade-statistics-6x](https://wiki.lyrasis.org/display/DSDOC6x/SOLR+Statistics+Maintenance) command | ||||||
|  |  | ||||||
| This project contains an indexer and a [Falcon-based](https://falcon.readthedocs.io/) web application to make the statistics available via a simple REST API. You can read more about the Solr queries used to gather the item view and download statistics on the [DSpace wiki](https://wiki.lyrasis.org/display/DSPACE/Solr). | This project contains an indexer and a [Falcon-based](https://falcon.readthedocs.io/) web application to make the item, community, and collection statistics available via a simple REST API. You can read more about the Solr queries used to gather the item view and download statistics on the [DSpace wiki](https://wiki.lyrasis.org/display/DSPACE/Solr). | ||||||
|  |  | ||||||
| If you use the DSpace Statistics API please cite: | If you use the DSpace Statistics API please cite: | ||||||
|  |  | ||||||
| @@ -12,9 +19,9 @@ If you use the DSpace Statistics API please cite: | |||||||
|  |  | ||||||
| ## Requirements | ## Requirements | ||||||
|  |  | ||||||
| - Python 3.6+ | - Python 3.9+ | ||||||
| - PostgreSQL version 9.5+ (due to [`UPSERT` support](https://wiki.postgresql.org/wiki/UPSERT)) | - PostgreSQL version 9.5+ (due to [`UPSERT` support](https://wiki.postgresql.org/wiki/UPSERT)) | ||||||
| - DSpace with [Solr usage statistics enabled](https://wiki.lyrasis.org/display/DSDOC5x/SOLR+Statistics) (tested with 5.x) | - DSpace with [Solr usage statistics enabled](https://wiki.lyrasis.org/display/DSDOC5x/SOLR+Statistics) (tested with 5.8+, 6.3, and 7.6) | ||||||
|  |  | ||||||
| ## Installation | ## Installation | ||||||
| Create a Python virtual environment and install the dependencies: | Create a Python virtual environment and install the dependencies: | ||||||
| @@ -27,7 +34,7 @@ Create a Python virtual environment and install the dependencies: | |||||||
|  |  | ||||||
| Set up the environment variables for Solr and PostgreSQL: | Set up the environment variables for Solr and PostgreSQL: | ||||||
|  |  | ||||||
|     $ export SOLR_SERVER=http://localhost:8080/solr |     $ export SOLR_SERVER=http://localhost:8983/solr | ||||||
|     $ export DATABASE_NAME=dspacestatistics |     $ export DATABASE_NAME=dspacestatistics | ||||||
|     $ export DATABASE_USER=dspacestatistics |     $ export DATABASE_USER=dspacestatistics | ||||||
|     $ export DATABASE_PASS=dspacestatistics |     $ export DATABASE_PASS=dspacestatistics | ||||||
| @@ -81,14 +88,20 @@ The API exposes the following endpoints: | |||||||
|  |  | ||||||
|   - GET `/` — return a basic API documentation page. |   - GET `/` — return a basic API documentation page. | ||||||
|   - GET `/items` — return views and downloads for all items that Solr knows about¹. Accepts `limit` and `page` query parameters for pagination of results (`limit` must be an integer between 1 and 100, and `page` must be an integer greater than or equal to 0). |   - GET `/items` — return views and downloads for all items that Solr knows about¹. Accepts `limit` and `page` query parameters for pagination of results (`limit` must be an integer between 1 and 100, and `page` must be an integer greater than or equal to 0). | ||||||
|   - POST `/items` — return views and downloads for an arbitrary list of items. Accepts `limit`, `page`, `dateFrom`, and `dateTo` parameters². |   - POST `/items` — return views and downloads for an arbitrary list of items with an optional date range. Accepts `limit`, `page`, `dateFrom`, and `dateTo` parameters². | ||||||
|   - GET `/item/id` — return views and downloads for a single item (`id` must be a UUID). Returns HTTP 404 if an item id is not found. |   - GET `/item/id` — return views and downloads for a single item (`id` must be a UUID). Returns HTTP 404 if an item id is not found. | ||||||
|  |   - GET `/communities` — return views and downloads for all communities that Solr knows about¹. Accepts `limit` and `page` query parameters for pagination of results (`limit` must be an integer between 1 and 100, and `page` must be an integer greater than or equal to 0). | ||||||
|  |   - POST `/communities` — return views and downloads for an arbitrary list of communities with an optional date range. Accepts `limit`, `page`, `dateFrom`, and `dateTo` parameters². | ||||||
|  |   - GET `/community/id` — return views and downloads for a single community (`id` must be a UUID). Returns HTTP 404 if a community id is not found. | ||||||
|  |   - GET `/collections` — return views and downloads for all collections that Solr knows about¹. Accepts `limit` and `page` query parameters for pagination of results (`limit` must be an integer between 1 and 100, and `page` must be an integer greater than or equal to 0). | ||||||
|  |   - POST `/collections` — return views and downloads for an arbitrary list of collections with an optional date range. Accepts `limit`, `page`, `dateFrom`, and `dateTo` parameters². | ||||||
|  |   - GET `/collection/id` — return views and downloads for a single collection (`id` must be a UUID). Returns HTTP 404 if an collection id is not found. | ||||||
|  |  | ||||||
| The item id is the *internal* UUID for an item. You can get these from the standard DSpace REST API. | The id is the *internal* UUID for an item, community, or collection. You can get these from the standard DSpace REST API. | ||||||
|  |  | ||||||
| ¹ We are querying the Solr statistics core, which technically only knows about items that have either views or downloads. If an item is not present here you can assume it has zero views and zero downloads, but not necessarily that it does not exist in the repository. | ¹ We are querying the Solr statistics core, which technically only knows about items, communities, or collections that have either views or downloads. If an item, community, or collection is not present here you can assume it has zero views and zero downloads, but not necessarily that it does not exist in the repository. | ||||||
|  |  | ||||||
| ² POST requests to `/items` should be in JSON format with the following parameters: | ² POST requests to `/items`, `/communities`, and `/collections` should be in JSON format with the following parameters (substitute the "items" list for communities or collections accordingly): | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| { | { | ||||||
| @@ -109,11 +122,10 @@ The item id is the *internal* UUID for an item. You can get these from the stand | |||||||
|  |  | ||||||
| - Better logging | - Better logging | ||||||
| - Version API (or at least include a /version endpoint?) | - Version API (or at least include a /version endpoint?) | ||||||
|  |   - Probably use /status with a version in the response | ||||||
| - Use JSON in PostgreSQL | - Use JSON in PostgreSQL | ||||||
| - Add top items endpoint, perhaps `/top/items` or `/items/top`? | - Add top items endpoint, perhaps `/top/items` or `/items/top`? | ||||||
|   - Actually we could add `/items?limit=10&sort=views` |   - Actually we could add `/items?limit=10&sort=views` | ||||||
| - Make community and collection stats available |  | ||||||
| - Check IDs in database to see if they are deleted... |  | ||||||
|  |  | ||||||
| ## License | ## License | ||||||
| This work is licensed under the [GPLv3](https://www.gnu.org/licenses/gpl-3.0.en.html). | This work is licensed under the [GPLv3](https://www.gnu.org/licenses/gpl-3.0.en.html). | ||||||
|   | |||||||
| @@ -1,159 +0,0 @@ | |||||||
| import falcon |  | ||||||
|  |  | ||||||
| from .database import DatabaseManager |  | ||||||
| from .items import get_downloads, get_views |  | ||||||
| from .util import validate_items_post_parameters |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class RootResource: |  | ||||||
|     def on_get(self, req, resp): |  | ||||||
|         resp.status = falcon.HTTP_200 |  | ||||||
|         resp.content_type = "text/html" |  | ||||||
|         with open("dspace_statistics_api/docs/index.html", "r") as f: |  | ||||||
|             resp.body = f.read() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AllItemsResource: |  | ||||||
|     def on_get(self, req, resp): |  | ||||||
|         """Handles GET requests""" |  | ||||||
|         # Return HTTPBadRequest if id parameter is not present and valid |  | ||||||
|         limit = req.get_param_as_int("limit", min_value=0, max_value=100) or 100 |  | ||||||
|         page = req.get_param_as_int("page", min_value=0) or 0 |  | ||||||
|         offset = limit * page |  | ||||||
|  |  | ||||||
|         with DatabaseManager() as db: |  | ||||||
|             db.set_session(readonly=True) |  | ||||||
|  |  | ||||||
|             with db.cursor() as cursor: |  | ||||||
|                 # get total number of items so we can estimate the pages |  | ||||||
|                 cursor.execute("SELECT COUNT(id) FROM items") |  | ||||||
|                 pages = round(cursor.fetchone()[0] / limit) |  | ||||||
|  |  | ||||||
|                 # get statistics and use limit and offset to page through results |  | ||||||
|                 cursor.execute( |  | ||||||
|                     "SELECT id, views, downloads FROM items LIMIT %s OFFSET %s", |  | ||||||
|                     [limit, offset], |  | ||||||
|                 ) |  | ||||||
|  |  | ||||||
|                 # create a list to hold dicts of item stats |  | ||||||
|                 statistics = list() |  | ||||||
|  |  | ||||||
|                 # iterate over results and build statistics object |  | ||||||
|                 for item in cursor: |  | ||||||
|                     statistics.append( |  | ||||||
|                         { |  | ||||||
|                             "id": str(item["id"]), |  | ||||||
|                             "views": item["views"], |  | ||||||
|                             "downloads": item["downloads"], |  | ||||||
|                         } |  | ||||||
|                     ) |  | ||||||
|  |  | ||||||
|         message = { |  | ||||||
|             "currentPage": page, |  | ||||||
|             "totalPages": pages, |  | ||||||
|             "limit": limit, |  | ||||||
|             "statistics": statistics, |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         resp.media = message |  | ||||||
|  |  | ||||||
|     @falcon.before(validate_items_post_parameters) |  | ||||||
|     def on_post(self, req, resp): |  | ||||||
|         """Handles POST requests""" |  | ||||||
|  |  | ||||||
|         # Build the Solr date string, ie: [* TO *] |  | ||||||
|         if req.context.dateFrom and req.context.dateTo: |  | ||||||
|             solr_date_string = f"[{req.context.dateFrom} TO {req.context.dateTo}]" |  | ||||||
|         elif not req.context.dateFrom and req.context.dateTo: |  | ||||||
|             solr_date_string = f"[* TO {req.context.dateTo}]" |  | ||||||
|         elif req.context.dateFrom and not req.context.dateTo: |  | ||||||
|             solr_date_string = f"[{req.context.dateFrom} TO *]" |  | ||||||
|         else: |  | ||||||
|             solr_date_string = "[* TO *]" |  | ||||||
|  |  | ||||||
|         # Helper variables to make working with pages/items/results easier and |  | ||||||
|         # to make the code easier to understand |  | ||||||
|         number_of_items: int = len(req.context.items) |  | ||||||
|         pages: int = int(number_of_items / req.context.limit) |  | ||||||
|         first_item: int = req.context.page * req.context.limit |  | ||||||
|         last_item: int = first_item + req.context.limit |  | ||||||
|         # Get a subset of the POSTed items based on our limit. Note that Python |  | ||||||
|         # list slicing and indexing are both zero based, but the first and last |  | ||||||
|         # items in a slice can be confusing. See this ASCII diagram: |  | ||||||
|         # |  | ||||||
|         #                 +---+---+---+---+---+---+ |  | ||||||
|         #                 | P | y | t | h | o | n | |  | ||||||
|         #                 +---+---+---+---+---+---+ |  | ||||||
|         # Slice position: 0   1   2   3   4   5   6 |  | ||||||
|         # Index position:   0   1   2   3   4   5 |  | ||||||
|         # |  | ||||||
|         # So if we have a list items with 240 items: |  | ||||||
|         # |  | ||||||
|         #   1st set: items[0:100] would give items at indexes 0 to 99 |  | ||||||
|         #   2nd set: items[100:200] would give items at indexes 100 to 199 |  | ||||||
|         #   3rd set: items[200:300] would give items at indexes 200 to 239 |  | ||||||
|         items_subset: list = req.context.items[first_item:last_item] |  | ||||||
|  |  | ||||||
|         views: dict = get_views(solr_date_string, items_subset) |  | ||||||
|         downloads: dict = get_downloads(solr_date_string, items_subset) |  | ||||||
|  |  | ||||||
|         # create a list to hold dicts of item stats |  | ||||||
|         statistics = list() |  | ||||||
|  |  | ||||||
|         # iterate over views dict to extract views and use the item id as an |  | ||||||
|         # index to the downloads dict to extract downloads. |  | ||||||
|         for k, v in views.items(): |  | ||||||
|             statistics.append({"id": k, "views": v, "downloads": downloads[k]}) |  | ||||||
|  |  | ||||||
|         message = { |  | ||||||
|             "currentPage": req.context.page, |  | ||||||
|             "totalPages": pages, |  | ||||||
|             "limit": req.context.limit, |  | ||||||
|             "statistics": statistics, |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         resp.status = falcon.HTTP_200 |  | ||||||
|         resp.media = message |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ItemResource: |  | ||||||
|     def on_get(self, req, resp, item_id): |  | ||||||
|         """Handles GET requests""" |  | ||||||
|  |  | ||||||
|         import psycopg2.extras |  | ||||||
|  |  | ||||||
|         # Adapt Python’s uuid.UUID type to PostgreSQL’s uuid |  | ||||||
|         # See: https://www.psycopg.org/docs/extras.html |  | ||||||
|         psycopg2.extras.register_uuid() |  | ||||||
|  |  | ||||||
|         with DatabaseManager() as db: |  | ||||||
|             db.set_session(readonly=True) |  | ||||||
|  |  | ||||||
|             with db.cursor() as cursor: |  | ||||||
|                 cursor = db.cursor() |  | ||||||
|                 cursor.execute( |  | ||||||
|                     "SELECT views, downloads FROM items WHERE id=%s", [str(item_id)] |  | ||||||
|                 ) |  | ||||||
|                 if cursor.rowcount == 0: |  | ||||||
|                     raise falcon.HTTPNotFound( |  | ||||||
|                         title="Item not found", |  | ||||||
|                         description=f'The item with id "{str(item_id)}" was not found.', |  | ||||||
|                     ) |  | ||||||
|                 else: |  | ||||||
|                     results = cursor.fetchone() |  | ||||||
|  |  | ||||||
|                     statistics = { |  | ||||||
|                         "id": str(item_id), |  | ||||||
|                         "views": results["views"], |  | ||||||
|                         "downloads": results["downloads"], |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     resp.media = statistics |  | ||||||
|  |  | ||||||
|  |  | ||||||
| api = application = falcon.API() |  | ||||||
| api.add_route("/", RootResource()) |  | ||||||
| api.add_route("/items", AllItemsResource()) |  | ||||||
| api.add_route("/item/{item_id:uuid}", ItemResource()) |  | ||||||
|  |  | ||||||
| # vim: set sw=4 ts=4 expandtab: |  | ||||||
| @@ -1,37 +0,0 @@ | |||||||
| <!DOCTYPE html> |  | ||||||
| <html lang="en-US"> |  | ||||||
|     <head> |  | ||||||
|         <meta charset="UTF-8"> |  | ||||||
|         <title>DSpace Statistics API</title> |  | ||||||
|     </head> |  | ||||||
|     <body> |  | ||||||
|         <h1>DSpace Statistics API v1.3.1</h1> |  | ||||||
|         <p>This site is running the <a href="https://github.com/ilri/dspace-statistics-api" title="DSpace Statistics API project">DSpace Statistics API</a>. The following endpoints are available:</p> |  | ||||||
|         <ul> |  | ||||||
|             <li>GET <code>/</code> — return a basic API documentation page.</li> |  | ||||||
|             <li>GET <code>/items</code> — return views and downloads for all items that Solr knows about¹. Accepts <code>limit</code> and <code>page</code> query parameters for pagination of results (<code>limit</code> must be an integer between 1 and 100, and <code>page</code> must be an integer greater than or equal to 0).</li> |  | ||||||
|             <li>POST <code>/items</code> — return views and downloads for an arbitrary list of items. Accepts <code>limit</code>, <code>page</code>, <code>dateFrom</code>, and <code>dateTo</code> parameters².</li> |  | ||||||
|             <li>GET <code>/item/id</code> — return views and downloads for a single item (<code>id</code> must be a UUID). Returns HTTP 404 if an item id is not found.</li> |  | ||||||
|         </ul> |  | ||||||
|  |  | ||||||
|         <p>The item id is the <em>internal</em> uuid for an item. You can get these from the standard DSpace REST API.</p> |  | ||||||
|  |  | ||||||
|         <hr/> |  | ||||||
|  |  | ||||||
|         <p>¹ We are querying the Solr statistics core, which technically only knows about items that have either views or downloads. If an item is not present here you can assume it has zero views and zero downloads, but not necessarily that it does not exist in the repository.</p> |  | ||||||
|         <p>² POST requests to <code>/items</code> should be in JSON format with the following parameters: |  | ||||||
| 			<pre><code>{ |  | ||||||
| 	"limit": 100, // optional, integer between 0 and 100, default 100 |  | ||||||
| 	"page": 0, // optional, integer greater than 0, default 0 |  | ||||||
| 	"dateFrom": "2020-01-01T00:00:00Z", // optional, default * |  | ||||||
| 	"dateTo": "2020-09-09T00:00:00Z", // optional, default * |  | ||||||
| 	"items": [ |  | ||||||
| 		"f44cf173-2344-4eb2-8f00-ee55df32c76f", |  | ||||||
| 		"2324aa41-e9de-4a2b-bc36-16241464683e", |  | ||||||
| 		"8542f9da-9ce1-4614-abf4-f2e3fdb4b305", |  | ||||||
| 		"0fe573e7-042a-4240-a4d9-753b61233908" |  | ||||||
| 	] |  | ||||||
| }</code></pre> |  | ||||||
|         </p> |  | ||||||
|     </body> |  | ||||||
| </html> |  | ||||||
| @@ -1,105 +0,0 @@ | |||||||
| import requests |  | ||||||
|  |  | ||||||
| from .config import SOLR_SERVER |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_views(solr_date_string: str, items: list): |  | ||||||
|     """ |  | ||||||
|     Get view statistics for a list of items from Solr. |  | ||||||
|  |  | ||||||
|     :parameter solr_date_string (str): Solr date string, for example "[* TO *]" |  | ||||||
|     :parameter items (list): a list of item IDs |  | ||||||
|     :returns: A dict of item IDs and views |  | ||||||
|     """ |  | ||||||
|     from .util import get_statistics_shards |  | ||||||
|     shards = get_statistics_shards() |  | ||||||
|  |  | ||||||
|     # Join the UUIDs with "OR" and escape the hyphens for Solr |  | ||||||
|     solr_items_string: str = " OR ".join(items).replace("-", r"\-") |  | ||||||
|  |  | ||||||
|     solr_query_params = { |  | ||||||
|         "q": f"id:({solr_items_string})", |  | ||||||
|         "fq": f"type:2 AND isBot:false AND statistics_type:view AND time:{solr_date_string}", |  | ||||||
|         "facet": "true", |  | ||||||
|         "facet.field": "id", |  | ||||||
|         "facet.mincount": 1, |  | ||||||
|         "shards": shards, |  | ||||||
|         "rows": 0, |  | ||||||
|         "wt": "json", |  | ||||||
|         "json.nl": "map",  # return facets as a dict instead of a flat list |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     solr_url = SOLR_SERVER + "/statistics/select" |  | ||||||
|     res = requests.get(solr_url, params=solr_query_params) |  | ||||||
|  |  | ||||||
|     # Create an empty dict to store views |  | ||||||
|     data = {} |  | ||||||
|  |  | ||||||
|     # Solr returns facets as a dict of dicts (see the json.nl parameter) |  | ||||||
|     views = res.json()["facet_counts"]["facet_fields"] |  | ||||||
|     # iterate over the 'id' dict and get the item ids and views |  | ||||||
|     for item_id, item_views in views["id"].items(): |  | ||||||
|         data[item_id] = item_views |  | ||||||
|  |  | ||||||
|     # Check if any items have missing stats so we can set them to 0 |  | ||||||
|     if len(data) < len(items): |  | ||||||
|         # List comprehension to get a list of item ids (keys) in the data |  | ||||||
|         data_ids = [k for k, v in data.items()] |  | ||||||
|         for item_id in items: |  | ||||||
|             if item_id not in data_ids: |  | ||||||
|                 data[item_id] = 0 |  | ||||||
|                 continue |  | ||||||
|  |  | ||||||
|     return data |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_downloads(solr_date_string: str, items: list): |  | ||||||
|     """ |  | ||||||
|     Get download statistics for a list of items from Solr. |  | ||||||
|  |  | ||||||
|     :parameter solr_date_string (str): Solr date string, for example "[* TO *]" |  | ||||||
|     :parameter items (list): a list of item IDs |  | ||||||
|     :returns: A dict of item IDs and downloads |  | ||||||
|     """ |  | ||||||
|     from .util import get_statistics_shards |  | ||||||
|     shards = get_statistics_shards() |  | ||||||
|  |  | ||||||
|     # Join the UUIDs with "OR" and escape the hyphens for Solr |  | ||||||
|     solr_items_string: str = " OR ".join(items).replace("-", r"\-") |  | ||||||
|  |  | ||||||
|     solr_query_params = { |  | ||||||
|         "q": f"owningItem:({solr_items_string})", |  | ||||||
|         "fq": f"type:0 AND isBot:false AND statistics_type:view AND bundleName:ORIGINAL AND time:{solr_date_string}", |  | ||||||
|         "facet": "true", |  | ||||||
|         "facet.field": "owningItem", |  | ||||||
|         "facet.mincount": 1, |  | ||||||
|         "shards": shards, |  | ||||||
|         "rows": 0, |  | ||||||
|         "wt": "json", |  | ||||||
|         "json.nl": "map",  # return facets as a dict instead of a flat list |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     solr_url = SOLR_SERVER + "/statistics/select" |  | ||||||
|     res = requests.get(solr_url, params=solr_query_params) |  | ||||||
|  |  | ||||||
|     # Create an empty dict to store downloads |  | ||||||
|     data = {} |  | ||||||
|  |  | ||||||
|     # Solr returns facets as a dict of dicts (see the json.nl parameter) |  | ||||||
|     downloads = res.json()["facet_counts"]["facet_fields"] |  | ||||||
|     # Iterate over the 'owningItem' dict and get the item ids and downloads |  | ||||||
|     for item_id, item_downloads in downloads["owningItem"].items(): |  | ||||||
|         data[item_id] = item_downloads |  | ||||||
|  |  | ||||||
|     # Check if any items have missing stats so we can set them to 0 |  | ||||||
|     if len(data) < len(items): |  | ||||||
|         # List comprehension to get a list of item ids (keys) in the data |  | ||||||
|         data_ids = [k for k, v in data.items()] |  | ||||||
|         for item_id in items: |  | ||||||
|             if item_id not in data_ids: |  | ||||||
|                 data[item_id] = 0 |  | ||||||
|                 continue |  | ||||||
|  |  | ||||||
|     return data |  | ||||||
|  |  | ||||||
| # vim: set sw=4 ts=4 expandtab: |  | ||||||
							
								
								
									
										868
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										868
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							| @@ -1,868 +0,0 @@ | |||||||
| [[package]] |  | ||||||
| name = "appdirs" |  | ||||||
| version = "1.4.4" |  | ||||||
| description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "appnope" |  | ||||||
| version = "0.1.0" |  | ||||||
| description = "Disable App Nap on OS X 10.9" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\" and sys_platform == \"darwin\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "atomicwrites" |  | ||||||
| version = "1.4.0" |  | ||||||
| description = "Atomic file writes." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
| marker = "sys_platform == \"win32\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "attrs" |  | ||||||
| version = "20.2.0" |  | ||||||
| description = "Classes Without Boilerplate" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| dev = ["coverage (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "sphinx", "sphinx-rtd-theme", "pre-commit"] |  | ||||||
| docs = ["sphinx", "sphinx-rtd-theme", "zope.interface"] |  | ||||||
| tests = ["coverage (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface"] |  | ||||||
| tests_no_zope = ["coverage (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six"] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "backcall" |  | ||||||
| version = "0.2.0" |  | ||||||
| description = "Specifications for callback functions passed in to an API" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "black" |  | ||||||
| version = "20.8b1" |  | ||||||
| description = "The uncompromising code formatter." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.6" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| colorama = ["colorama (>=0.4.3)"] |  | ||||||
| d = ["aiohttp (>=3.3.2)", "aiohttp-cors"] |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| appdirs = "*" |  | ||||||
| click = ">=7.1.2" |  | ||||||
| mypy-extensions = ">=0.4.3" |  | ||||||
| pathspec = ">=0.6,<1" |  | ||||||
| regex = ">=2020.1.8" |  | ||||||
| toml = ">=0.10.1" |  | ||||||
| typed-ast = ">=1.4.0" |  | ||||||
| typing-extensions = ">=3.7.4" |  | ||||||
|  |  | ||||||
| [package.dependencies.dataclasses] |  | ||||||
| version = ">=0.6" |  | ||||||
| python = "<3.7" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "certifi" |  | ||||||
| version = "2020.6.20" |  | ||||||
| description = "Python package for providing Mozilla's CA Bundle." |  | ||||||
| category = "main" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "chardet" |  | ||||||
| version = "3.0.4" |  | ||||||
| description = "Universal encoding detector for Python 2 and 3" |  | ||||||
| category = "main" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "click" |  | ||||||
| version = "7.1.2" |  | ||||||
| description = "Composable command line interface toolkit" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "colorama" |  | ||||||
| version = "0.4.3" |  | ||||||
| description = "Cross-platform colored terminal text." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\" and sys_platform == \"win32\" or sys_platform == \"win32\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "dataclasses" |  | ||||||
| version = "0.6" |  | ||||||
| description = "A backport of the dataclasses module for Python 3.6" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
| marker = "python_version < \"3.7\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "decorator" |  | ||||||
| version = "4.4.2" |  | ||||||
| description = "Decorators for Humans" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.6, !=3.0.*, !=3.1.*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "falcon" |  | ||||||
| version = "2.0.0" |  | ||||||
| description = "An unladen web framework for building APIs and app backends." |  | ||||||
| category = "main" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "flake8" |  | ||||||
| version = "3.8.4" |  | ||||||
| description = "the modular source code checker: pep8 pyflakes and co" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7" |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| mccabe = ">=0.6.0,<0.7.0" |  | ||||||
| pycodestyle = ">=2.6.0a1,<2.7.0" |  | ||||||
| pyflakes = ">=2.2.0,<2.3.0" |  | ||||||
|  |  | ||||||
| [package.dependencies.importlib-metadata] |  | ||||||
| version = "*" |  | ||||||
| python = "<3.8" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "gunicorn" |  | ||||||
| version = "20.0.4" |  | ||||||
| description = "WSGI HTTP Server for UNIX" |  | ||||||
| category = "main" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.4" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| eventlet = ["eventlet (>=0.9.7)"] |  | ||||||
| gevent = ["gevent (>=0.13)"] |  | ||||||
| setproctitle = ["setproctitle"] |  | ||||||
| tornado = ["tornado (>=0.2)"] |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| setuptools = ">=3.0" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "idna" |  | ||||||
| version = "2.10" |  | ||||||
| description = "Internationalized Domain Names in Applications (IDNA)" |  | ||||||
| category = "main" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "importlib-metadata" |  | ||||||
| version = "2.0.0" |  | ||||||
| description = "Read metadata from Python packages" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" |  | ||||||
| marker = "python_version < \"3.8\"" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| docs = ["sphinx", "rst.linker"] |  | ||||||
| testing = ["packaging", "pep517", "importlib-resources (>=1.3)"] |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| zipp = ">=0.5" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "iniconfig" |  | ||||||
| version = "1.0.1" |  | ||||||
| description = "iniconfig: brain-dead simple config-ini parsing" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "ipython" |  | ||||||
| version = "7.18.1" |  | ||||||
| description = "IPython: Productive Interactive Computing" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.7" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| all = ["Sphinx (>=1.3)", "ipykernel", "ipyparallel", "ipywidgets", "nbconvert", "nbformat", "nose (>=0.10.1)", "notebook", "numpy (>=1.14)", "pygments", "qtconsole", "requests", "testpath"] |  | ||||||
| doc = ["Sphinx (>=1.3)"] |  | ||||||
| kernel = ["ipykernel"] |  | ||||||
| nbconvert = ["nbconvert"] |  | ||||||
| nbformat = ["nbformat"] |  | ||||||
| notebook = ["notebook", "ipywidgets"] |  | ||||||
| parallel = ["ipyparallel"] |  | ||||||
| qtconsole = ["qtconsole"] |  | ||||||
| test = ["nose (>=0.10.1)", "requests", "testpath", "pygments", "nbformat", "ipykernel", "numpy (>=1.14)"] |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| appnope = "*" |  | ||||||
| backcall = "*" |  | ||||||
| colorama = "*" |  | ||||||
| decorator = "*" |  | ||||||
| jedi = ">=0.10" |  | ||||||
| pexpect = ">4.3" |  | ||||||
| pickleshare = "*" |  | ||||||
| prompt-toolkit = ">=2.0.0,<3.0.0 || >3.0.0,<3.0.1 || >3.0.1,<3.1.0" |  | ||||||
| pygments = "*" |  | ||||||
| setuptools = ">=18.5" |  | ||||||
| traitlets = ">=4.2" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "ipython-genutils" |  | ||||||
| version = "0.2.0" |  | ||||||
| description = "Vestigial utilities from IPython" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "isort" |  | ||||||
| version = "5.5.4" |  | ||||||
| description = "A Python utility / library to sort Python imports." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.6,<4.0" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| pipfile_deprecated_finder = ["pipreqs", "requirementslib"] |  | ||||||
| requirements_deprecated_finder = ["pipreqs", "pip-api"] |  | ||||||
| colors = ["colorama (>=0.4.3,<0.5.0)"] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "jedi" |  | ||||||
| version = "0.17.2" |  | ||||||
| description = "An autocompletion tool for Python that can be used for text editors." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| qa = ["flake8 (3.7.9)"] |  | ||||||
| testing = ["Django (<3.1)", "colorama", "docopt", "pytest (>=3.9.0,<5.0.0)"] |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| parso = ">=0.7.0,<0.8.0" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "mccabe" |  | ||||||
| version = "0.6.1" |  | ||||||
| description = "McCabe checker, plugin for flake8" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "mypy-extensions" |  | ||||||
| version = "0.4.3" |  | ||||||
| description = "Experimental type system extensions for programs checked with the mypy typechecker." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "packaging" |  | ||||||
| version = "20.4" |  | ||||||
| description = "Core utilities for Python packages" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| pyparsing = ">=2.0.2" |  | ||||||
| six = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "parso" |  | ||||||
| version = "0.7.1" |  | ||||||
| description = "A Python Parser" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| testing = ["docopt", "pytest (>=3.0.7)"] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pathspec" |  | ||||||
| version = "0.8.0" |  | ||||||
| description = "Utility library for gitignore style pattern matching of file paths." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pexpect" |  | ||||||
| version = "4.8.0" |  | ||||||
| description = "Pexpect allows easy control of interactive console applications." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\" and sys_platform != \"win32\"" |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| ptyprocess = ">=0.5" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pickleshare" |  | ||||||
| version = "0.7.5" |  | ||||||
| description = "Tiny 'shelve'-like database with concurrency support" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pluggy" |  | ||||||
| version = "0.13.1" |  | ||||||
| description = "plugin and hook calling mechanisms for python" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| dev = ["pre-commit", "tox"] |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| [package.dependencies.importlib-metadata] |  | ||||||
| version = ">=0.12" |  | ||||||
| python = "<3.8" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "prompt-toolkit" |  | ||||||
| version = "3.0.7" |  | ||||||
| description = "Library for building powerful interactive command lines in Python" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.6.1" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| wcwidth = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "psycopg2-binary" |  | ||||||
| version = "2.8.6" |  | ||||||
| description = "psycopg2 - Python-PostgreSQL Database Adapter" |  | ||||||
| category = "main" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "ptyprocess" |  | ||||||
| version = "0.6.0" |  | ||||||
| description = "Run a subprocess in a pseudo terminal" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\" and sys_platform != \"win32\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "py" |  | ||||||
| version = "1.9.0" |  | ||||||
| description = "library with cross-python path, ini-parsing, io, code, log facilities" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pycodestyle" |  | ||||||
| version = "2.6.0" |  | ||||||
| description = "Python style guide checker" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pyflakes" |  | ||||||
| version = "2.2.0" |  | ||||||
| description = "passive checker of Python programs" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pygments" |  | ||||||
| version = "2.7.1" |  | ||||||
| description = "Pygments is a syntax highlighting package written in Python." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.5" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pyparsing" |  | ||||||
| version = "2.4.7" |  | ||||||
| description = "Python parsing module" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pytest" |  | ||||||
| version = "6.1.1" |  | ||||||
| description = "pytest: simple powerful testing with Python" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.5" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| checkqa_mypy = ["mypy (0.780)"] |  | ||||||
| testing = ["argcomplete", "hypothesis (>=3.56)", "mock", "nose", "requests", "xmlschema"] |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| atomicwrites = ">=1.0" |  | ||||||
| attrs = ">=17.4.0" |  | ||||||
| colorama = "*" |  | ||||||
| iniconfig = "*" |  | ||||||
| packaging = "*" |  | ||||||
| pluggy = ">=0.12,<1.0" |  | ||||||
| py = ">=1.8.2" |  | ||||||
| toml = "*" |  | ||||||
|  |  | ||||||
| [package.dependencies.importlib-metadata] |  | ||||||
| version = ">=0.12" |  | ||||||
| python = "<3.8" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "pytest-clarity" |  | ||||||
| version = "0.3.0a0" |  | ||||||
| description = "A plugin providing an alternative, colourful diff output for failing assertions." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| pytest = ">=3.5.0" |  | ||||||
| termcolor = "1.1.0" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "regex" |  | ||||||
| version = "2020.9.27" |  | ||||||
| description = "Alternative regular expression module, to replace re." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "requests" |  | ||||||
| version = "2.24.0" |  | ||||||
| description = "Python HTTP for Humans." |  | ||||||
| category = "main" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| security = ["pyOpenSSL (>=0.14)", "cryptography (>=1.3.4)"] |  | ||||||
| socks = ["PySocks (>=1.5.6,<1.5.7 || >1.5.7)", "win-inet-pton"] |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| certifi = ">=2017.4.17" |  | ||||||
| chardet = ">=3.0.2,<4" |  | ||||||
| idna = ">=2.5,<3" |  | ||||||
| urllib3 = ">=1.21.1,<1.25.0 || >1.25.0,<1.25.1 || >1.25.1,<1.26" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "six" |  | ||||||
| version = "1.15.0" |  | ||||||
| description = "Python 2 and 3 compatibility utilities" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "termcolor" |  | ||||||
| version = "1.1.0" |  | ||||||
| description = "ANSII Color formatting for output in terminal." |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "toml" |  | ||||||
| version = "0.10.1" |  | ||||||
| description = "Python Library for Tom's Obvious, Minimal Language" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "traitlets" |  | ||||||
| version = "5.0.4" |  | ||||||
| description = "Traitlets Python configuration system" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.7" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| test = ["pytest"] |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| ipython-genutils = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "typed-ast" |  | ||||||
| version = "1.4.1" |  | ||||||
| description = "a fork of Python 2 and 3 ast modules with type comment support" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "typing-extensions" |  | ||||||
| version = "3.7.4.3" |  | ||||||
| description = "Backported and Experimental Type Hints for Python 3.5+" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "urllib3" |  | ||||||
| version = "1.25.10" |  | ||||||
| description = "HTTP library with thread-safe connection pooling, file post, and more." |  | ||||||
| category = "main" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, <4" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| brotli = ["brotlipy (>=0.6.0)"] |  | ||||||
| secure = ["certifi", "cryptography (>=1.3.4)", "idna (>=2.0.0)", "pyOpenSSL (>=0.14)", "ipaddress"] |  | ||||||
| socks = ["PySocks (>=1.5.6,<1.5.7 || >1.5.7,<2.0)"] |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "wcwidth" |  | ||||||
| version = "0.2.5" |  | ||||||
| description = "Measures the displayed width of unicode strings in a terminal" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = "*" |  | ||||||
| marker = "python_version >= \"3.7\" and python_version < \"4.0\"" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| name = "zipp" |  | ||||||
| version = "3.3.0" |  | ||||||
| description = "Backport of pathlib-compatible object wrapper for zip files" |  | ||||||
| category = "dev" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.6" |  | ||||||
| marker = "python_version < \"3.8\"" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| docs = ["sphinx", "jaraco.packaging (>=3.2)", "rst.linker (>=1.9)"] |  | ||||||
| testing = ["pytest (>=3.5,<3.7.3 || >3.7.3)", "pytest-checkdocs (>=1.2.3)", "pytest-flake8", "pytest-cov", "jaraco.test (>=3.2.0)", "jaraco.itertools", "func-timeout", "pytest-black (>=0.3.7)", "pytest-mypy"] |  | ||||||
|  |  | ||||||
| [metadata] |  | ||||||
| lock-version = "1.0" |  | ||||||
| python-versions = "^3.6" |  | ||||||
| content-hash = "12341bae9a3c60ddc21a0334682520dd52ca54e35d88370cdfce81c10c125e7e" |  | ||||||
|  |  | ||||||
| [metadata.files] |  | ||||||
| appdirs = [ |  | ||||||
|     {file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"}, |  | ||||||
|     {file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"}, |  | ||||||
| ] |  | ||||||
| appnope = [ |  | ||||||
|     {file = "appnope-0.1.0-py2.py3-none-any.whl", hash = "sha256:5b26757dc6f79a3b7dc9fab95359328d5747fcb2409d331ea66d0272b90ab2a0"}, |  | ||||||
|     {file = "appnope-0.1.0.tar.gz", hash = "sha256:8b995ffe925347a2138d7ac0fe77155e4311a0ea6d6da4f5128fe4b3cbe5ed71"}, |  | ||||||
| ] |  | ||||||
| atomicwrites = [ |  | ||||||
|     {file = "atomicwrites-1.4.0-py2.py3-none-any.whl", hash = "sha256:6d1784dea7c0c8d4a5172b6c620f40b6e4cbfdf96d783691f2e1302a7b88e197"}, |  | ||||||
|     {file = "atomicwrites-1.4.0.tar.gz", hash = "sha256:ae70396ad1a434f9c7046fd2dd196fc04b12f9e91ffb859164193be8b6168a7a"}, |  | ||||||
| ] |  | ||||||
| attrs = [ |  | ||||||
|     {file = "attrs-20.2.0-py2.py3-none-any.whl", hash = "sha256:fce7fc47dfc976152e82d53ff92fa0407700c21acd20886a13777a0d20e655dc"}, |  | ||||||
|     {file = "attrs-20.2.0.tar.gz", hash = "sha256:26b54ddbbb9ee1d34d5d3668dd37d6cf74990ab23c828c2888dccdceee395594"}, |  | ||||||
| ] |  | ||||||
| backcall = [ |  | ||||||
|     {file = "backcall-0.2.0-py2.py3-none-any.whl", hash = "sha256:fbbce6a29f263178a1f7915c1940bde0ec2b2a967566fe1c65c1dfb7422bd255"}, |  | ||||||
|     {file = "backcall-0.2.0.tar.gz", hash = "sha256:5cbdbf27be5e7cfadb448baf0aa95508f91f2bbc6c6437cd9cd06e2a4c215e1e"}, |  | ||||||
| ] |  | ||||||
| black = [ |  | ||||||
|     {file = "black-20.8b1.tar.gz", hash = "sha256:1c02557aa099101b9d21496f8a914e9ed2222ef70336404eeeac8edba836fbea"}, |  | ||||||
| ] |  | ||||||
| certifi = [ |  | ||||||
|     {file = "certifi-2020.6.20-py2.py3-none-any.whl", hash = "sha256:8fc0819f1f30ba15bdb34cceffb9ef04d99f420f68eb75d901e9560b8749fc41"}, |  | ||||||
|     {file = "certifi-2020.6.20.tar.gz", hash = "sha256:5930595817496dd21bb8dc35dad090f1c2cd0adfaf21204bf6732ca5d8ee34d3"}, |  | ||||||
| ] |  | ||||||
| chardet = [ |  | ||||||
|     {file = "chardet-3.0.4-py2.py3-none-any.whl", hash = "sha256:fc323ffcaeaed0e0a02bf4d117757b98aed530d9ed4531e3e15460124c106691"}, |  | ||||||
|     {file = "chardet-3.0.4.tar.gz", hash = "sha256:84ab92ed1c4d4f16916e05906b6b75a6c0fb5db821cc65e70cbd64a3e2a5eaae"}, |  | ||||||
| ] |  | ||||||
| click = [ |  | ||||||
|     {file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"}, |  | ||||||
|     {file = "click-7.1.2.tar.gz", hash = "sha256:d2b5255c7c6349bc1bd1e59e08cd12acbbd63ce649f2588755783aa94dfb6b1a"}, |  | ||||||
| ] |  | ||||||
| colorama = [ |  | ||||||
|     {file = "colorama-0.4.3-py2.py3-none-any.whl", hash = "sha256:7d73d2a99753107a36ac6b455ee49046802e59d9d076ef8e47b61499fa29afff"}, |  | ||||||
|     {file = "colorama-0.4.3.tar.gz", hash = "sha256:e96da0d330793e2cb9485e9ddfd918d456036c7149416295932478192f4436a1"}, |  | ||||||
| ] |  | ||||||
| dataclasses = [ |  | ||||||
|     {file = "dataclasses-0.6-py3-none-any.whl", hash = "sha256:454a69d788c7fda44efd71e259be79577822f5e3f53f029a22d08004e951dc9f"}, |  | ||||||
|     {file = "dataclasses-0.6.tar.gz", hash = "sha256:6988bd2b895eef432d562370bb707d540f32f7360ab13da45340101bc2307d84"}, |  | ||||||
| ] |  | ||||||
| decorator = [ |  | ||||||
|     {file = "decorator-4.4.2-py2.py3-none-any.whl", hash = "sha256:41fa54c2a0cc4ba648be4fd43cff00aedf5b9465c9bf18d64325bc225f08f760"}, |  | ||||||
|     {file = "decorator-4.4.2.tar.gz", hash = "sha256:e3a62f0520172440ca0dcc823749319382e377f37f140a0b99ef45fecb84bfe7"}, |  | ||||||
| ] |  | ||||||
| falcon = [ |  | ||||||
|     {file = "falcon-2.0.0-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:733033ec80c896e30a43ab3e776856096836787197a44eb21022320a61311983"}, |  | ||||||
|     {file = "falcon-2.0.0-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:f93351459f110b4c1ee28556aef9a791832df6f910bea7b3f616109d534df06b"}, |  | ||||||
|     {file = "falcon-2.0.0-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:e9efa0791b5d9f9dd9689015ea6bce0a27fcd5ecbcd30e6d940bffa4f7f03389"}, |  | ||||||
|     {file = "falcon-2.0.0-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:59d1e8c993b9a37ea06df9d72cf907a46cc8063b30717cdac2f34d1658b6f936"}, |  | ||||||
|     {file = "falcon-2.0.0-cp34-cp34m-manylinux1_i686.whl", hash = "sha256:a5ebb22a04c9cc65081938ee7651b4e3b4d2a28522ea8ec04c7bdd2b3e9e8cd8"}, |  | ||||||
|     {file = "falcon-2.0.0-cp34-cp34m-manylinux1_x86_64.whl", hash = "sha256:95bf6ce986c1119aef12c9b348f4dee9c6dcc58391bdd0bc2b0bf353c2b15986"}, |  | ||||||
|     {file = "falcon-2.0.0-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:aa184895d1ad4573fbfaaf803563d02f019ebdf4790e41cc568a330607eae439"}, |  | ||||||
|     {file = "falcon-2.0.0-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:74cf1d18207381c665b9e6292d65100ce146d958707793174b03869dc6e614f4"}, |  | ||||||
|     {file = "falcon-2.0.0-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:24adcd2b29a8ffa9d552dc79638cd21736a3fb04eda7d102c6cebafdaadb88ad"}, |  | ||||||
|     {file = "falcon-2.0.0-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:18157af2a4fc3feedf2b5dcc6196f448639acf01c68bc33d4d5a04c3ef87f494"}, |  | ||||||
|     {file = "falcon-2.0.0-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:e3782b7b92fefd46a6ad1fd8fe63fe6c6f1b7740a95ca56957f48d1aee34b357"}, |  | ||||||
|     {file = "falcon-2.0.0-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:9712975adcf8c6e12876239085ad757b8fdeba223d46d23daef82b47658f83a9"}, |  | ||||||
|     {file = "falcon-2.0.0-py2.py3-none-any.whl", hash = "sha256:54f2cb4b687035b2a03206dbfc538055cc48b59a953187b0458aa1b574d47b53"}, |  | ||||||
|     {file = "falcon-2.0.0.tar.gz", hash = "sha256:eea593cf466b9c126ce667f6d30503624ef24459f118c75594a69353b6c3d5fc"}, |  | ||||||
| ] |  | ||||||
| flake8 = [ |  | ||||||
|     {file = "flake8-3.8.4-py2.py3-none-any.whl", hash = "sha256:749dbbd6bfd0cf1318af27bf97a14e28e5ff548ef8e5b1566ccfb25a11e7c839"}, |  | ||||||
|     {file = "flake8-3.8.4.tar.gz", hash = "sha256:aadae8761ec651813c24be05c6f7b4680857ef6afaae4651a4eccaef97ce6c3b"}, |  | ||||||
| ] |  | ||||||
| gunicorn = [ |  | ||||||
|     {file = "gunicorn-20.0.4-py2.py3-none-any.whl", hash = "sha256:cd4a810dd51bf497552cf3f863b575dabd73d6ad6a91075b65936b151cbf4f9c"}, |  | ||||||
|     {file = "gunicorn-20.0.4.tar.gz", hash = "sha256:1904bb2b8a43658807108d59c3f3d56c2b6121a701161de0ddf9ad140073c626"}, |  | ||||||
| ] |  | ||||||
| idna = [ |  | ||||||
|     {file = "idna-2.10-py2.py3-none-any.whl", hash = "sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"}, |  | ||||||
|     {file = "idna-2.10.tar.gz", hash = "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6"}, |  | ||||||
| ] |  | ||||||
| importlib-metadata = [ |  | ||||||
|     {file = "importlib_metadata-2.0.0-py2.py3-none-any.whl", hash = "sha256:cefa1a2f919b866c5beb7c9f7b0ebb4061f30a8a9bf16d609b000e2dfaceb9c3"}, |  | ||||||
|     {file = "importlib_metadata-2.0.0.tar.gz", hash = "sha256:77a540690e24b0305878c37ffd421785a6f7e53c8b5720d211b211de8d0e95da"}, |  | ||||||
| ] |  | ||||||
| iniconfig = [ |  | ||||||
|     {file = "iniconfig-1.0.1-py3-none-any.whl", hash = "sha256:80cf40c597eb564e86346103f609d74efce0f6b4d4f30ec8ce9e2c26411ba437"}, |  | ||||||
|     {file = "iniconfig-1.0.1.tar.gz", hash = "sha256:e5f92f89355a67de0595932a6c6c02ab4afddc6fcdc0bfc5becd0d60884d3f69"}, |  | ||||||
| ] |  | ||||||
| ipython = [ |  | ||||||
|     {file = "ipython-7.18.1-py3-none-any.whl", hash = "sha256:2e22c1f74477b5106a6fb301c342ab8c64bb75d702e350f05a649e8cb40a0fb8"}, |  | ||||||
|     {file = "ipython-7.18.1.tar.gz", hash = "sha256:a331e78086001931de9424940699691ad49dfb457cea31f5471eae7b78222d5e"}, |  | ||||||
| ] |  | ||||||
| ipython-genutils = [ |  | ||||||
|     {file = "ipython_genutils-0.2.0-py2.py3-none-any.whl", hash = "sha256:72dd37233799e619666c9f639a9da83c34013a73e8bbc79a7a6348d93c61fab8"}, |  | ||||||
|     {file = "ipython_genutils-0.2.0.tar.gz", hash = "sha256:eb2e116e75ecef9d4d228fdc66af54269afa26ab4463042e33785b887c628ba8"}, |  | ||||||
| ] |  | ||||||
| isort = [ |  | ||||||
|     {file = "isort-5.5.4-py3-none-any.whl", hash = "sha256:36f0c6659b9000597e92618d05b72d4181104cf59472b1c6a039e3783f930c95"}, |  | ||||||
|     {file = "isort-5.5.4.tar.gz", hash = "sha256:ba040c24d20aa302f78f4747df549573ae1eaf8e1084269199154da9c483f07f"}, |  | ||||||
| ] |  | ||||||
| jedi = [ |  | ||||||
|     {file = "jedi-0.17.2-py2.py3-none-any.whl", hash = "sha256:98cc583fa0f2f8304968199b01b6b4b94f469a1f4a74c1560506ca2a211378b5"}, |  | ||||||
|     {file = "jedi-0.17.2.tar.gz", hash = "sha256:86ed7d9b750603e4ba582ea8edc678657fb4007894a12bcf6f4bb97892f31d20"}, |  | ||||||
| ] |  | ||||||
| mccabe = [ |  | ||||||
|     {file = "mccabe-0.6.1-py2.py3-none-any.whl", hash = "sha256:ab8a6258860da4b6677da4bd2fe5dc2c659cff31b3ee4f7f5d64e79735b80d42"}, |  | ||||||
|     {file = "mccabe-0.6.1.tar.gz", hash = "sha256:dd8d182285a0fe56bace7f45b5e7d1a6ebcbf524e8f3bd87eb0f125271b8831f"}, |  | ||||||
| ] |  | ||||||
| mypy-extensions = [ |  | ||||||
|     {file = "mypy_extensions-0.4.3-py2.py3-none-any.whl", hash = "sha256:090fedd75945a69ae91ce1303b5824f428daf5a028d2f6ab8a299250a846f15d"}, |  | ||||||
|     {file = "mypy_extensions-0.4.3.tar.gz", hash = "sha256:2d82818f5bb3e369420cb3c4060a7970edba416647068eb4c5343488a6c604a8"}, |  | ||||||
| ] |  | ||||||
| packaging = [ |  | ||||||
|     {file = "packaging-20.4-py2.py3-none-any.whl", hash = "sha256:998416ba6962ae7fbd6596850b80e17859a5753ba17c32284f67bfff33784181"}, |  | ||||||
|     {file = "packaging-20.4.tar.gz", hash = "sha256:4357f74f47b9c12db93624a82154e9b120fa8293699949152b22065d556079f8"}, |  | ||||||
| ] |  | ||||||
| parso = [ |  | ||||||
|     {file = "parso-0.7.1-py2.py3-none-any.whl", hash = "sha256:97218d9159b2520ff45eb78028ba8b50d2bc61dcc062a9682666f2dc4bd331ea"}, |  | ||||||
|     {file = "parso-0.7.1.tar.gz", hash = "sha256:caba44724b994a8a5e086460bb212abc5a8bc46951bf4a9a1210745953622eb9"}, |  | ||||||
| ] |  | ||||||
| pathspec = [ |  | ||||||
|     {file = "pathspec-0.8.0-py2.py3-none-any.whl", hash = "sha256:7d91249d21749788d07a2d0f94147accd8f845507400749ea19c1ec9054a12b0"}, |  | ||||||
|     {file = "pathspec-0.8.0.tar.gz", hash = "sha256:da45173eb3a6f2a5a487efba21f050af2b41948be6ab52b6a1e3ff22bb8b7061"}, |  | ||||||
| ] |  | ||||||
| pexpect = [ |  | ||||||
|     {file = "pexpect-4.8.0-py2.py3-none-any.whl", hash = "sha256:0b48a55dcb3c05f3329815901ea4fc1537514d6ba867a152b581d69ae3710937"}, |  | ||||||
|     {file = "pexpect-4.8.0.tar.gz", hash = "sha256:fc65a43959d153d0114afe13997d439c22823a27cefceb5ff35c2178c6784c0c"}, |  | ||||||
| ] |  | ||||||
| pickleshare = [ |  | ||||||
|     {file = "pickleshare-0.7.5-py2.py3-none-any.whl", hash = "sha256:9649af414d74d4df115d5d718f82acb59c9d418196b7b4290ed47a12ce62df56"}, |  | ||||||
|     {file = "pickleshare-0.7.5.tar.gz", hash = "sha256:87683d47965c1da65cdacaf31c8441d12b8044cdec9aca500cd78fc2c683afca"}, |  | ||||||
| ] |  | ||||||
| pluggy = [ |  | ||||||
|     {file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"}, |  | ||||||
|     {file = "pluggy-0.13.1.tar.gz", hash = "sha256:15b2acde666561e1298d71b523007ed7364de07029219b604cf808bfa1c765b0"}, |  | ||||||
| ] |  | ||||||
| prompt-toolkit = [ |  | ||||||
|     {file = "prompt_toolkit-3.0.7-py3-none-any.whl", hash = "sha256:83074ee28ad4ba6af190593d4d4c607ff525272a504eb159199b6dd9f950c950"}, |  | ||||||
|     {file = "prompt_toolkit-3.0.7.tar.gz", hash = "sha256:822f4605f28f7d2ba6b0b09a31e25e140871e96364d1d377667b547bb3bf4489"}, |  | ||||||
| ] |  | ||||||
| psycopg2-binary = [ |  | ||||||
|     {file = "psycopg2-binary-2.8.6.tar.gz", hash = "sha256:11b9c0ebce097180129e422379b824ae21c8f2a6596b159c7659e2e5a00e1aa0"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp27-cp27m-macosx_10_6_intel.macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:d14b140a4439d816e3b1229a4a525df917d6ea22a0771a2a78332273fd9528a4"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:1fabed9ea2acc4efe4671b92c669a213db744d2af8a9fc5d69a8e9bc14b7a9db"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:f5ab93a2cb2d8338b1674be43b442a7f544a0971da062a5da774ed40587f18f5"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp27-cp27m-win32.whl", hash = "sha256:b4afc542c0ac0db720cf516dd20c0846f71c248d2b3d21013aa0d4ef9c71ca25"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp27-cp27m-win_amd64.whl", hash = "sha256:e74a55f6bad0e7d3968399deb50f61f4db1926acf4a6d83beaaa7df986f48b1c"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:0deac2af1a587ae12836aa07970f5cb91964f05a7c6cdb69d8425ff4c15d4e2c"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:ad20d2eb875aaa1ea6d0f2916949f5c08a19c74d05b16ce6ebf6d24f2c9f75d1"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp34-cp34m-win32.whl", hash = "sha256:950bc22bb56ee6ff142a2cb9ee980b571dd0912b0334aa3fe0fe3788d860bea2"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp34-cp34m-win_amd64.whl", hash = "sha256:b8a3715b3c4e604bcc94c90a825cd7f5635417453b253499664f784fc4da0152"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp35-cp35m-macosx_10_6_intel.macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:d1b4ab59e02d9008efe10ceabd0b31e79519da6fb67f7d8e8977118832d0f449"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:ac0c682111fbf404525dfc0f18a8b5f11be52657d4f96e9fcb75daf4f3984859"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:7d92a09b788cbb1aec325af5fcba9fed7203897bbd9269d5691bb1e3bce29550"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp35-cp35m-win32.whl", hash = "sha256:aaa4213c862f0ef00022751161df35804127b78adf4a2755b9f991a507e425fd"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp35-cp35m-win_amd64.whl", hash = "sha256:c2507d796fca339c8fb03216364cca68d87e037c1f774977c8fc377627d01c71"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp36-cp36m-macosx_10_6_intel.macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:ee69dad2c7155756ad114c02db06002f4cded41132cc51378e57aad79cc8e4f4"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:e82aba2188b9ba309fd8e271702bd0d0fc9148ae3150532bbb474f4590039ffb"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:d5227b229005a696cc67676e24c214740efd90b148de5733419ac9aaba3773da"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp36-cp36m-win32.whl", hash = "sha256:a0eb43a07386c3f1f1ebb4dc7aafb13f67188eab896e7397aa1ee95a9c884eb2"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp36-cp36m-win_amd64.whl", hash = "sha256:e1f57aa70d3f7cc6947fd88636a481638263ba04a742b4a37dd25c373e41491a"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp37-cp37m-macosx_10_6_intel.macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:833709a5c66ca52f1d21d41865a637223b368c0ee76ea54ca5bad6f2526c7679"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:ba28584e6bca48c59eecbf7efb1576ca214b47f05194646b081717fa628dfddf"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:6a32f3a4cb2f6e1a0b15215f448e8ce2da192fd4ff35084d80d5e39da683e79b"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp37-cp37m-win32.whl", hash = "sha256:0e4dc3d5996760104746e6cfcdb519d9d2cd27c738296525d5867ea695774e67"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp37-cp37m-win_amd64.whl", hash = "sha256:cec7e622ebc545dbb4564e483dd20e4e404da17ae07e06f3e780b2dacd5cee66"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp38-cp38-macosx_10_9_x86_64.macosx_10_9_intel.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:ba381aec3a5dc29634f20692349d73f2d21f17653bda1decf0b52b11d694541f"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp38-cp38-manylinux1_i686.whl", hash = "sha256:a0c50db33c32594305b0ef9abc0cb7db13de7621d2cadf8392a1d9b3c437ef77"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:2dac98e85565d5688e8ab7bdea5446674a83a3945a8f416ad0110018d1501b94"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp38-cp38-win32.whl", hash = "sha256:bd1be66dde2b82f80afb9459fc618216753f67109b859a361cf7def5c7968729"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp38-cp38-win_amd64.whl", hash = "sha256:8cd0fb36c7412996859cb4606a35969dd01f4ea34d9812a141cd920c3b18be77"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp39-cp39-manylinux1_i686.whl", hash = "sha256:42ec1035841b389e8cc3692277a0bd81cdfe0b65d575a2c8862cec7a80e62e52"}, |  | ||||||
|     {file = "psycopg2_binary-2.8.6-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:7312e931b90fe14f925729cde58022f5d034241918a5c4f9797cac62f6b3a9dd"}, |  | ||||||
| ] |  | ||||||
| ptyprocess = [ |  | ||||||
|     {file = "ptyprocess-0.6.0-py2.py3-none-any.whl", hash = "sha256:d7cc528d76e76342423ca640335bd3633420dc1366f258cb31d05e865ef5ca1f"}, |  | ||||||
|     {file = "ptyprocess-0.6.0.tar.gz", hash = "sha256:923f299cc5ad920c68f2bc0bc98b75b9f838b93b599941a6b63ddbc2476394c0"}, |  | ||||||
| ] |  | ||||||
| py = [ |  | ||||||
|     {file = "py-1.9.0-py2.py3-none-any.whl", hash = "sha256:366389d1db726cd2fcfc79732e75410e5fe4d31db13692115529d34069a043c2"}, |  | ||||||
|     {file = "py-1.9.0.tar.gz", hash = "sha256:9ca6883ce56b4e8da7e79ac18787889fa5206c79dcc67fb065376cd2fe03f342"}, |  | ||||||
| ] |  | ||||||
| pycodestyle = [ |  | ||||||
|     {file = "pycodestyle-2.6.0-py2.py3-none-any.whl", hash = "sha256:2295e7b2f6b5bd100585ebcb1f616591b652db8a741695b3d8f5d28bdc934367"}, |  | ||||||
|     {file = "pycodestyle-2.6.0.tar.gz", hash = "sha256:c58a7d2815e0e8d7972bf1803331fb0152f867bd89adf8a01dfd55085434192e"}, |  | ||||||
| ] |  | ||||||
| pyflakes = [ |  | ||||||
|     {file = "pyflakes-2.2.0-py2.py3-none-any.whl", hash = "sha256:0d94e0e05a19e57a99444b6ddcf9a6eb2e5c68d3ca1e98e90707af8152c90a92"}, |  | ||||||
|     {file = "pyflakes-2.2.0.tar.gz", hash = "sha256:35b2d75ee967ea93b55750aa9edbbf72813e06a66ba54438df2cfac9e3c27fc8"}, |  | ||||||
| ] |  | ||||||
| pygments = [ |  | ||||||
|     {file = "Pygments-2.7.1-py3-none-any.whl", hash = "sha256:307543fe65c0947b126e83dd5a61bd8acbd84abec11f43caebaf5534cbc17998"}, |  | ||||||
|     {file = "Pygments-2.7.1.tar.gz", hash = "sha256:926c3f319eda178d1bd90851e4317e6d8cdb5e292a3386aac9bd75eca29cf9c7"}, |  | ||||||
| ] |  | ||||||
| pyparsing = [ |  | ||||||
|     {file = "pyparsing-2.4.7-py2.py3-none-any.whl", hash = "sha256:ef9d7589ef3c200abe66653d3f1ab1033c3c419ae9b9bdb1240a85b024efc88b"}, |  | ||||||
|     {file = "pyparsing-2.4.7.tar.gz", hash = "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1"}, |  | ||||||
| ] |  | ||||||
| pytest = [ |  | ||||||
|     {file = "pytest-6.1.1-py3-none-any.whl", hash = "sha256:7a8190790c17d79a11f847fba0b004ee9a8122582ebff4729a082c109e81a4c9"}, |  | ||||||
|     {file = "pytest-6.1.1.tar.gz", hash = "sha256:8f593023c1a0f916110285b6efd7f99db07d59546e3d8c36fc60e2ab05d3be92"}, |  | ||||||
| ] |  | ||||||
| pytest-clarity = [ |  | ||||||
|     {file = "pytest-clarity-0.3.0a0.tar.gz", hash = "sha256:5cc99e3d9b7969dfe17e5f6072d45a917c59d363b679686d3c958a1ded2e4dcf"}, |  | ||||||
| ] |  | ||||||
| regex = [ |  | ||||||
|     {file = "regex-2020.9.27-cp27-cp27m-win32.whl", hash = "sha256:d23a18037313714fb3bb5a94434d3151ee4300bae631894b1ac08111abeaa4a3"}, |  | ||||||
|     {file = "regex-2020.9.27-cp27-cp27m-win_amd64.whl", hash = "sha256:84e9407db1b2eb368b7ecc283121b5e592c9aaedbe8c78b1a2f1102eb2e21d19"}, |  | ||||||
|     {file = "regex-2020.9.27-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:5f18875ac23d9aa2f060838e8b79093e8bb2313dbaaa9f54c6d8e52a5df097be"}, |  | ||||||
|     {file = "regex-2020.9.27-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:ae91972f8ac958039920ef6e8769277c084971a142ce2b660691793ae44aae6b"}, |  | ||||||
|     {file = "regex-2020.9.27-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:9a02d0ae31d35e1ec12a4ea4d4cca990800f66a917d0fb997b20fbc13f5321fc"}, |  | ||||||
|     {file = "regex-2020.9.27-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:ebbe29186a3d9b0c591e71b7393f1ae08c83cb2d8e517d2a822b8f7ec99dfd8b"}, |  | ||||||
|     {file = "regex-2020.9.27-cp36-cp36m-win32.whl", hash = "sha256:4707f3695b34335afdfb09be3802c87fa0bc27030471dbc082f815f23688bc63"}, |  | ||||||
|     {file = "regex-2020.9.27-cp36-cp36m-win_amd64.whl", hash = "sha256:9bc13e0d20b97ffb07821aa3e113f9998e84994fe4d159ffa3d3a9d1b805043b"}, |  | ||||||
|     {file = "regex-2020.9.27-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:f1b3afc574a3db3b25c89161059d857bd4909a1269b0b3cb3c904677c8c4a3f7"}, |  | ||||||
|     {file = "regex-2020.9.27-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:5533a959a1748a5c042a6da71fe9267a908e21eded7a4f373efd23a2cbdb0ecc"}, |  | ||||||
|     {file = "regex-2020.9.27-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:1fe0a41437bbd06063aa184c34804efa886bcc128222e9916310c92cd54c3b4c"}, |  | ||||||
|     {file = "regex-2020.9.27-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:c570f6fa14b9c4c8a4924aaad354652366577b4f98213cf76305067144f7b100"}, |  | ||||||
|     {file = "regex-2020.9.27-cp37-cp37m-win32.whl", hash = "sha256:eda4771e0ace7f67f58bc5b560e27fb20f32a148cbc993b0c3835970935c2707"}, |  | ||||||
|     {file = "regex-2020.9.27-cp37-cp37m-win_amd64.whl", hash = "sha256:60b0e9e6dc45683e569ec37c55ac20c582973841927a85f2d8a7d20ee80216ab"}, |  | ||||||
|     {file = "regex-2020.9.27-cp38-cp38-manylinux1_i686.whl", hash = "sha256:088afc8c63e7bd187a3c70a94b9e50ab3f17e1d3f52a32750b5b77dbe99ef5ef"}, |  | ||||||
|     {file = "regex-2020.9.27-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:eaf548d117b6737df379fdd53bdde4f08870e66d7ea653e230477f071f861121"}, |  | ||||||
|     {file = "regex-2020.9.27-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:41bb65f54bba392643557e617316d0d899ed5b4946dccee1cb6696152b29844b"}, |  | ||||||
|     {file = "regex-2020.9.27-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:8d69cef61fa50c8133382e61fd97439de1ae623fe943578e477e76a9d9471637"}, |  | ||||||
|     {file = "regex-2020.9.27-cp38-cp38-win32.whl", hash = "sha256:f2388013e68e750eaa16ccbea62d4130180c26abb1d8e5d584b9baf69672b30f"}, |  | ||||||
|     {file = "regex-2020.9.27-cp38-cp38-win_amd64.whl", hash = "sha256:4318d56bccfe7d43e5addb272406ade7a2274da4b70eb15922a071c58ab0108c"}, |  | ||||||
|     {file = "regex-2020.9.27.tar.gz", hash = "sha256:a6f32aea4260dfe0e55dc9733ea162ea38f0ea86aa7d0f77b15beac5bf7b369d"}, |  | ||||||
| ] |  | ||||||
| requests = [ |  | ||||||
|     {file = "requests-2.24.0-py2.py3-none-any.whl", hash = "sha256:fe75cc94a9443b9246fc7049224f75604b113c36acb93f87b80ed42c44cbb898"}, |  | ||||||
|     {file = "requests-2.24.0.tar.gz", hash = "sha256:b3559a131db72c33ee969480840fff4bb6dd111de7dd27c8ee1f820f4f00231b"}, |  | ||||||
| ] |  | ||||||
| six = [ |  | ||||||
|     {file = "six-1.15.0-py2.py3-none-any.whl", hash = "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"}, |  | ||||||
|     {file = "six-1.15.0.tar.gz", hash = "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259"}, |  | ||||||
| ] |  | ||||||
| termcolor = [ |  | ||||||
|     {file = "termcolor-1.1.0.tar.gz", hash = "sha256:1d6d69ce66211143803fbc56652b41d73b4a400a2891d7bf7a1cdf4c02de613b"}, |  | ||||||
| ] |  | ||||||
| toml = [ |  | ||||||
|     {file = "toml-0.10.1-py2.py3-none-any.whl", hash = "sha256:bda89d5935c2eac546d648028b9901107a595863cb36bae0c73ac804a9b4ce88"}, |  | ||||||
|     {file = "toml-0.10.1.tar.gz", hash = "sha256:926b612be1e5ce0634a2ca03470f95169cf16f939018233a670519cb4ac58b0f"}, |  | ||||||
| ] |  | ||||||
| traitlets = [ |  | ||||||
|     {file = "traitlets-5.0.4-py3-none-any.whl", hash = "sha256:9664ec0c526e48e7b47b7d14cd6b252efa03e0129011de0a9c1d70315d4309c3"}, |  | ||||||
|     {file = "traitlets-5.0.4.tar.gz", hash = "sha256:86c9351f94f95de9db8a04ad8e892da299a088a64fd283f9f6f18770ae5eae1b"}, |  | ||||||
| ] |  | ||||||
| typed-ast = [ |  | ||||||
|     {file = "typed_ast-1.4.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:73d785a950fc82dd2a25897d525d003f6378d1cb23ab305578394694202a58c3"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:aaee9905aee35ba5905cfb3c62f3e83b3bec7b39413f0a7f19be4e547ea01ebb"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp35-cp35m-win32.whl", hash = "sha256:0c2c07682d61a629b68433afb159376e24e5b2fd4641d35424e462169c0a7919"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp35-cp35m-win_amd64.whl", hash = "sha256:4083861b0aa07990b619bd7ddc365eb7fa4b817e99cf5f8d9cf21a42780f6e01"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:269151951236b0f9a6f04015a9004084a5ab0d5f19b57de779f908621e7d8b75"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:24995c843eb0ad11a4527b026b4dde3da70e1f2d8806c99b7b4a7cf491612652"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:fe460b922ec15dd205595c9b5b99e2f056fd98ae8f9f56b888e7a17dc2b757e7"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp36-cp36m-win32.whl", hash = "sha256:4e3e5da80ccbebfff202a67bf900d081906c358ccc3d5e3c8aea42fdfdfd51c1"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp36-cp36m-win_amd64.whl", hash = "sha256:249862707802d40f7f29f6e1aad8d84b5aa9e44552d2cc17384b209f091276aa"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:8ce678dbaf790dbdb3eba24056d5364fb45944f33553dd5869b7580cdbb83614"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:c9e348e02e4d2b4a8b2eedb48210430658df6951fa484e59de33ff773fbd4b41"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:bcd3b13b56ea479b3650b82cabd6b5343a625b0ced5429e4ccad28a8973f301b"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp37-cp37m-win32.whl", hash = "sha256:d5d33e9e7af3b34a40dc05f498939f0ebf187f07c385fd58d591c533ad8562fe"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp37-cp37m-win_amd64.whl", hash = "sha256:0666aa36131496aed8f7be0410ff974562ab7eeac11ef351def9ea6fa28f6355"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp38-cp38-macosx_10_15_x86_64.whl", hash = "sha256:d205b1b46085271b4e15f670058ce182bd1199e56b317bf2ec004b6a44f911f6"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:6daac9731f172c2a22ade6ed0c00197ee7cc1221aa84cfdf9c31defeb059a907"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:498b0f36cc7054c1fead3d7fc59d2150f4d5c6c56ba7fb150c013fbc683a8d2d"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp38-cp38-win32.whl", hash = "sha256:715ff2f2df46121071622063fc7543d9b1fd19ebfc4f5c8895af64a77a8c852c"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:fc0fea399acb12edbf8a628ba8d2312f583bdbdb3335635db062fa98cf71fca4"}, |  | ||||||
|     {file = "typed_ast-1.4.1-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:d43943ef777f9a1c42bf4e552ba23ac77a6351de620aa9acf64ad54933ad4d34"}, |  | ||||||
|     {file = "typed_ast-1.4.1.tar.gz", hash = "sha256:8c8aaad94455178e3187ab22c8b01a3837f8ee50e09cf31f1ba129eb293ec30b"}, |  | ||||||
| ] |  | ||||||
| typing-extensions = [ |  | ||||||
|     {file = "typing_extensions-3.7.4.3-py2-none-any.whl", hash = "sha256:dafc7639cde7f1b6e1acc0f457842a83e722ccca8eef5270af2d74792619a89f"}, |  | ||||||
|     {file = "typing_extensions-3.7.4.3-py3-none-any.whl", hash = "sha256:7cb407020f00f7bfc3cb3e7881628838e69d8f3fcab2f64742a5e76b2f841918"}, |  | ||||||
|     {file = "typing_extensions-3.7.4.3.tar.gz", hash = "sha256:99d4073b617d30288f569d3f13d2bd7548c3a7e4c8de87db09a9d29bb3a4a60c"}, |  | ||||||
| ] |  | ||||||
| urllib3 = [ |  | ||||||
|     {file = "urllib3-1.25.10-py2.py3-none-any.whl", hash = "sha256:e7983572181f5e1522d9c98453462384ee92a0be7fac5f1413a1e35c56cc0461"}, |  | ||||||
|     {file = "urllib3-1.25.10.tar.gz", hash = "sha256:91056c15fa70756691db97756772bb1eb9678fa585d9184f24534b100dc60f4a"}, |  | ||||||
| ] |  | ||||||
| wcwidth = [ |  | ||||||
|     {file = "wcwidth-0.2.5-py2.py3-none-any.whl", hash = "sha256:beb4802a9cebb9144e99086eff703a642a13d6a0052920003a230f3294bbe784"}, |  | ||||||
|     {file = "wcwidth-0.2.5.tar.gz", hash = "sha256:c4d647b99872929fdb7bdcaa4fbe7f01413ed3d98077df798530e5b04f116c83"}, |  | ||||||
| ] |  | ||||||
| zipp = [ |  | ||||||
|     {file = "zipp-3.3.0-py3-none-any.whl", hash = "sha256:eed8ec0b8d1416b2ca33516a37a08892442f3954dee131e92cfd92d8fe3e7066"}, |  | ||||||
|     {file = "zipp-3.3.0.tar.gz", hash = "sha256:64ad89efee774d1897a58607895d80789c59778ea02185dd846ac38394a8642b"}, |  | ||||||
| ] |  | ||||||
| @@ -1,25 +1,42 @@ | |||||||
| [tool.poetry] | [project] | ||||||
| name = "dspace-statistics-api" | name = "dspace-statistics-api" | ||||||
| version = "1.3.1" | version = "1.4.6-dev" | ||||||
| description = "A simple REST API to expose Solr view and download statistics for items in a DSpace repository." | description = "A simple REST API to expose Solr view and download statistics for items, communities, and collections in a DSpace repository." | ||||||
| authors = ["Alan Orth <aorth@mjanja.ch>"] | authors = [ | ||||||
|  |     { name = "Alan Orth", email = "git@mjanja.mozmail.com" } | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | requires-python = ">= 3.9" | ||||||
|  |  | ||||||
| license = "GPL-3.0-only" | license = "GPL-3.0-only" | ||||||
|  | license-files = ["LICENSE.txt"] | ||||||
|  | dependencies = [ | ||||||
|  |     "gunicorn==23.0.*", | ||||||
|  |     "falcon==4.0.*", | ||||||
|  |     "psycopg==3.2.*", | ||||||
|  |     "requests==2.32.*", | ||||||
|  |     "falcon-swagger-ui", | ||||||
|  | ] | ||||||
|  |  | ||||||
| [tool.poetry.dependencies] | [project.urls] | ||||||
| python = "^3.6" | repository = "https://github.com/ilri/dspace-statistics-api" | ||||||
| gunicorn = "^20.0.4" | homepage = "https://github.com/ilri/dspace-statistics-api" | ||||||
| falcon = "^2.0.0" |  | ||||||
| psycopg2-binary = "^2.8.6" |  | ||||||
| requests = "^2.24.0" |  | ||||||
|  |  | ||||||
| [tool.poetry.dev-dependencies] | [tool.uv.sources] | ||||||
| ipython = { version = "^7.18.1", python = "^3.7" } | falcon-swagger-ui = { git = "https://github.com/alanorth/falcon-swagger-ui", rev = "falcon3-update-swagger-ui" } | ||||||
| flake8 = "^3.8.4" |  | ||||||
| pytest = "^6.1.1" |  | ||||||
| isort = "^5.5.4" |  | ||||||
| black = "^20.8b1" |  | ||||||
| pytest-clarity = "^0.3.0-alpha.0" |  | ||||||
|  |  | ||||||
|  | # See: https://docs.astral.sh/uv/concepts/build-backend/#using-the-uv-build-backend | ||||||
| [build-system] | [build-system] | ||||||
| requires = ["poetry>=0.12"] | requires = ["uv_build>=0.7.19,<0.8.0"] | ||||||
| build-backend = "poetry.masonry.api" | build-backend = "uv_build" | ||||||
|  |  | ||||||
|  | [dependency-groups] | ||||||
|  | dev = [ | ||||||
|  |     "flake8==7.1.*", | ||||||
|  |     "isort==5.13.*", | ||||||
|  |     "pytest==8.3.*", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [tool.isort] | ||||||
|  | profile = "black" | ||||||
|  | line_length=88 | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| [pytest] | [pytest] | ||||||
| addopts= -rsxX -s -v --strict | addopts= -rsxX -s -v --strict-markers | ||||||
| filterwarnings = | filterwarnings = | ||||||
|     error::UserWarning |     error::UserWarning | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								renovate.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								renovate.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | { | ||||||
|  |   "$schema": "https://docs.renovatebot.com/renovate-schema.json", | ||||||
|  |   "extends": [ | ||||||
|  |     "config:base" | ||||||
|  |   ], | ||||||
|  |   "pip_requirements": { | ||||||
|  |       "enabled": false | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,51 +1,56 @@ | |||||||
| appdirs==1.4.4 | # This file was autogenerated by uv via the following command: | ||||||
| appnope==0.1.0; python_version >= "3.7" and python_version < "4.0" and sys_platform == "darwin" | #    uv export --no-hashes | ||||||
| atomicwrites==1.4.0; sys_platform == "win32" | -e . | ||||||
| attrs==20.2.0 | certifi==2025.6.15 | ||||||
| backcall==0.2.0; python_version >= "3.7" and python_version < "4.0" |     # via requests | ||||||
| black==20.8b1 | charset-normalizer==3.4.2 | ||||||
| certifi==2020.6.20 |     # via requests | ||||||
| chardet==3.0.4 | colorama==0.4.6 ; sys_platform == 'win32' | ||||||
| click==7.1.2 |     # via pytest | ||||||
| colorama==0.4.3; python_version >= "3.7" and python_version < "4.0" and sys_platform == "win32" or sys_platform == "win32" | exceptiongroup==1.3.0 ; python_full_version < '3.11' | ||||||
| dataclasses==0.6; python_version < "3.7" |     # via pytest | ||||||
| decorator==4.4.2; python_version >= "3.7" and python_version < "4.0" | falcon==4.0.2 | ||||||
| falcon==2.0.0 |     # via | ||||||
| flake8==3.8.4 |     #   dspace-statistics-api | ||||||
| gunicorn==20.0.4 |     #   falcon-swagger-ui | ||||||
| idna==2.10 | falcon-swagger-ui @ git+https://github.com/alanorth/falcon-swagger-ui@c019c270b479c03d9276e20fd95488495b0943f6 | ||||||
| importlib-metadata==2.0.0; python_version < "3.8" |     # via dspace-statistics-api | ||||||
| iniconfig==1.0.1 | flake8==7.1.2 | ||||||
| ipython==7.18.1; python_version >= "3.7" and python_version < "4.0" | gunicorn==23.0.0 | ||||||
| ipython-genutils==0.2.0; python_version >= "3.7" and python_version < "4.0" |     # via dspace-statistics-api | ||||||
| isort==5.5.4 | idna==3.10 | ||||||
| jedi==0.17.2; python_version >= "3.7" and python_version < "4.0" |     # via requests | ||||||
| mccabe==0.6.1 | iniconfig==2.1.0 | ||||||
| mypy-extensions==0.4.3 |     # via pytest | ||||||
| packaging==20.4 | isort==5.13.2 | ||||||
| parso==0.7.1; python_version >= "3.7" and python_version < "4.0" | jinja2==3.1.6 | ||||||
| pathspec==0.8.0 |     # via falcon-swagger-ui | ||||||
| pexpect==4.8.0; python_version >= "3.7" and python_version < "4.0" and sys_platform != "win32" | markupsafe==3.0.2 | ||||||
| pickleshare==0.7.5; python_version >= "3.7" and python_version < "4.0" |     # via jinja2 | ||||||
| pluggy==0.13.1 | mccabe==0.7.0 | ||||||
| prompt-toolkit==3.0.7; python_version >= "3.7" and python_version < "4.0" |     # via flake8 | ||||||
| psycopg2-binary==2.8.6 | packaging==25.0 | ||||||
| ptyprocess==0.6.0; python_version >= "3.7" and python_version < "4.0" and sys_platform != "win32" |     # via | ||||||
| py==1.9.0 |     #   gunicorn | ||||||
| pycodestyle==2.6.0 |     #   pytest | ||||||
| pyflakes==2.2.0 | pluggy==1.6.0 | ||||||
| pygments==2.7.1; python_version >= "3.7" and python_version < "4.0" |     # via pytest | ||||||
| pyparsing==2.4.7 | psycopg==3.2.9 | ||||||
| pytest==6.1.1 |     # via dspace-statistics-api | ||||||
| pytest-clarity==0.3.0a0 | pycodestyle==2.12.1 | ||||||
| regex==2020.9.27 |     # via flake8 | ||||||
| requests==2.24.0 | pyflakes==3.2.0 | ||||||
| six==1.15.0 |     # via flake8 | ||||||
| termcolor==1.1.0 | pytest==8.3.5 | ||||||
| toml==0.10.1 | requests==2.32.4 | ||||||
| traitlets==5.0.4; python_version >= "3.7" and python_version < "4.0" |     # via dspace-statistics-api | ||||||
| typed-ast==1.4.1 | tomli==2.2.1 ; python_full_version < '3.11' | ||||||
| typing-extensions==3.7.4.3 |     # via pytest | ||||||
| urllib3==1.25.10 | typing-extensions==4.14.0 ; python_full_version < '3.13' | ||||||
| wcwidth==0.2.5; python_version >= "3.7" and python_version < "4.0" |     # via | ||||||
| zipp==3.3.0; python_version < "3.8" |     #   exceptiongroup | ||||||
|  |     #   psycopg | ||||||
|  | tzdata==2025.2 ; sys_platform == 'win32' | ||||||
|  |     # via psycopg | ||||||
|  | urllib3==2.5.0 | ||||||
|  |     # via requests | ||||||
|   | |||||||
| @@ -1,8 +1,33 @@ | |||||||
| certifi==2020.6.20 | # This file was autogenerated by uv via the following command: | ||||||
| chardet==3.0.4 | #    uv export --no-dev --no-hashes | ||||||
| falcon==2.0.0 | -e . | ||||||
| gunicorn==20.0.4 | certifi==2025.6.15 | ||||||
| idna==2.10 |     # via requests | ||||||
| psycopg2-binary==2.8.6 | charset-normalizer==3.4.2 | ||||||
| requests==2.24.0 |     # via requests | ||||||
| urllib3==1.25.10 | falcon==4.0.2 | ||||||
|  |     # via | ||||||
|  |     #   dspace-statistics-api | ||||||
|  |     #   falcon-swagger-ui | ||||||
|  | falcon-swagger-ui @ git+https://github.com/alanorth/falcon-swagger-ui@c019c270b479c03d9276e20fd95488495b0943f6 | ||||||
|  |     # via dspace-statistics-api | ||||||
|  | gunicorn==23.0.0 | ||||||
|  |     # via dspace-statistics-api | ||||||
|  | idna==3.10 | ||||||
|  |     # via requests | ||||||
|  | jinja2==3.1.6 | ||||||
|  |     # via falcon-swagger-ui | ||||||
|  | markupsafe==3.0.2 | ||||||
|  |     # via jinja2 | ||||||
|  | packaging==25.0 | ||||||
|  |     # via gunicorn | ||||||
|  | psycopg==3.2.9 | ||||||
|  |     # via dspace-statistics-api | ||||||
|  | requests==2.32.4 | ||||||
|  |     # via dspace-statistics-api | ||||||
|  | typing-extensions==4.14.0 ; python_full_version < '3.13' | ||||||
|  |     # via psycopg | ||||||
|  | tzdata==2025.2 ; sys_platform == 'win32' | ||||||
|  |     # via psycopg | ||||||
|  | urllib3==2.5.0 | ||||||
|  |     # via requests | ||||||
|   | |||||||
| @@ -1,6 +0,0 @@ | |||||||
| [isort] |  | ||||||
| multi_line_output=3 |  | ||||||
| include_trailing_comma=True |  | ||||||
| force_grid_wrap=0 |  | ||||||
| use_parentheses=True |  | ||||||
| line_length=88 |  | ||||||
							
								
								
									
										254
									
								
								src/dspace_statistics_api/app.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										254
									
								
								src/dspace_statistics_api/app.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,254 @@ | |||||||
|  | # SPDX-License-Identifier: GPL-3.0-only | ||||||
|  |  | ||||||
|  | import json | ||||||
|  | import math | ||||||
|  |  | ||||||
|  | import falcon | ||||||
|  | from falcon_swagger_ui import register_swaggerui_app | ||||||
|  |  | ||||||
|  | from .config import DSPACE_STATISTICS_API_URL, VERSION | ||||||
|  | from .database import DatabaseManager | ||||||
|  | from .stats import get_downloads, get_views | ||||||
|  | from .util import set_statistics_scope, validate_post_parameters | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RootResource: | ||||||
|  |     def on_get(self, req, resp): | ||||||
|  |         resp.status = falcon.HTTP_200 | ||||||
|  |         resp.content_type = "text/html" | ||||||
|  |         docs_html = ( | ||||||
|  |             "<!DOCTYPE html>" | ||||||
|  |             '<html lang="en-US">' | ||||||
|  |             "    <head>" | ||||||
|  |             '        <meta charset="UTF-8">' | ||||||
|  |             "        <title>DSpace Statistics API</title>" | ||||||
|  |             "    </head>" | ||||||
|  |             "    <body>" | ||||||
|  |             f"        <h1>DSpace Statistics API {VERSION}</h1>" | ||||||
|  |             f"        <p>This site is running the <a href=\"https://github.com/ilri/dspace-statistics-api\" title=\"DSpace Statistics API project\">DSpace Statistics API</a>. For more information see the project's README.md or the interactive <a href=\"{DSPACE_STATISTICS_API_URL + '/swagger'}\">Swagger UI</a> built into this API.</p>" | ||||||
|  |             "    </body>" | ||||||
|  |             "</html" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         resp.text = docs_html | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class StatusResource: | ||||||
|  |     def on_get(self, req, resp): | ||||||
|  |         message = {"version": VERSION} | ||||||
|  |  | ||||||
|  |         resp.status = falcon.HTTP_200 | ||||||
|  |         resp.media = message | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class OpenAPIJSONResource: | ||||||
|  |     def on_get(self, req, resp): | ||||||
|  |         resp.status = falcon.HTTP_200 | ||||||
|  |         resp.content_type = "text/html" | ||||||
|  |         with open("dspace_statistics_api/docs/openapi.json", "r") as f: | ||||||
|  |             # Load the openapi.json schema | ||||||
|  |             data = json.load(f) | ||||||
|  |  | ||||||
|  |             # Swagger assumes your API is at the root of the current host unless | ||||||
|  |             # you configure a "servers" block in the schema. The problem is that | ||||||
|  |             # I want this to work in both development and production, so we need | ||||||
|  |             # to make this configurable. | ||||||
|  |             # | ||||||
|  |             # If the DSPACE_STATISTICS_API_URL is configured then we will add a | ||||||
|  |             # server entry to the openapi.json schema before sending it. | ||||||
|  |             if DSPACE_STATISTICS_API_URL != "": | ||||||
|  |                 data["servers"] = [{"url": DSPACE_STATISTICS_API_URL}] | ||||||
|  |  | ||||||
|  |             # Set the version in the schema so Swagger UI can display it | ||||||
|  |             data["info"]["version"] = VERSION | ||||||
|  |  | ||||||
|  |             resp.text = json.dumps(data) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AllStatisticsResource: | ||||||
|  |     @falcon.before(set_statistics_scope) | ||||||
|  |     def on_get(self, req, resp): | ||||||
|  |         """Handles GET requests""" | ||||||
|  |         # Return HTTPBadRequest if id parameter is not present and valid | ||||||
|  |         limit = req.get_param_as_int("limit", min_value=1, max_value=100) or 100 | ||||||
|  |         page = req.get_param_as_int("page", min_value=0) or 0 | ||||||
|  |         offset = limit * page | ||||||
|  |  | ||||||
|  |         with DatabaseManager() as db: | ||||||
|  |             db.set_read_only(True) | ||||||
|  |  | ||||||
|  |             with db.cursor() as cursor: | ||||||
|  |                 # get total number of communities/collections/items so we can estimate the pages | ||||||
|  |                 cursor.execute(f"SELECT COUNT(id) FROM {req.context.statistics_scope}") | ||||||
|  |                 pages = math.ceil(cursor.fetchone()['count'] / limit) | ||||||
|  |  | ||||||
|  |                 # get statistics and use limit and offset to page through results | ||||||
|  |                 cursor.execute( | ||||||
|  |                     f"SELECT id, views, downloads FROM {req.context.statistics_scope} ORDER BY id LIMIT %s OFFSET %s", | ||||||
|  |                     [limit, offset], | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |                 # create a list to hold dicts of stats | ||||||
|  |                 statistics = [] | ||||||
|  |  | ||||||
|  |                 # iterate over results and build statistics object | ||||||
|  |                 for result in cursor: | ||||||
|  |                     statistics.append( | ||||||
|  |                         { | ||||||
|  |                             "id": str(result["id"]), | ||||||
|  |                             "views": result["views"], | ||||||
|  |                             "downloads": result["downloads"], | ||||||
|  |                         } | ||||||
|  |                     ) | ||||||
|  |  | ||||||
|  |         message = { | ||||||
|  |             "currentPage": page, | ||||||
|  |             "totalPages": pages, | ||||||
|  |             "limit": limit, | ||||||
|  |             "statistics": statistics, | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         resp.media = message | ||||||
|  |  | ||||||
|  |     @falcon.before(set_statistics_scope) | ||||||
|  |     @falcon.before(validate_post_parameters) | ||||||
|  |     def on_post(self, req, resp): | ||||||
|  |         """Handles POST requests. | ||||||
|  |  | ||||||
|  |         Uses two `before` hooks to set the statistics "scope" and validate the | ||||||
|  |         POST parameters. The "scope" is the type of statistics we want, which | ||||||
|  |         will be items, communities, or collections, depending on the request. | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         # Build the Solr date string, ie: [* TO *] | ||||||
|  |         if req.context.dateFrom and req.context.dateTo: | ||||||
|  |             solr_date_string = f"[{req.context.dateFrom} TO {req.context.dateTo}]" | ||||||
|  |         elif not req.context.dateFrom and req.context.dateTo: | ||||||
|  |             solr_date_string = f"[* TO {req.context.dateTo}]" | ||||||
|  |         elif req.context.dateFrom and not req.context.dateTo: | ||||||
|  |             solr_date_string = f"[{req.context.dateFrom} TO *]" | ||||||
|  |         else: | ||||||
|  |             solr_date_string = "[* TO *]" | ||||||
|  |  | ||||||
|  |         # Helper variables to make working with pages/items/results easier and | ||||||
|  |         # to make the code easier to understand | ||||||
|  |         number_of_elements: int = len(req.context.elements) | ||||||
|  |         pages: int = math.ceil(number_of_elements / req.context.limit) | ||||||
|  |         first_element: int = req.context.page * req.context.limit | ||||||
|  |         last_element: int = first_element + req.context.limit | ||||||
|  |         # Get a subset of the POSTed items based on our limit. Note that Python | ||||||
|  |         # list slicing and indexing are both zero based, but the first and last | ||||||
|  |         # items in a slice can be confusing. See this ASCII diagram: | ||||||
|  |         # | ||||||
|  |         #                 +---+---+---+---+---+---+ | ||||||
|  |         #                 | P | y | t | h | o | n | | ||||||
|  |         #                 +---+---+---+---+---+---+ | ||||||
|  |         # Slice position: 0   1   2   3   4   5   6 | ||||||
|  |         # Index position:   0   1   2   3   4   5 | ||||||
|  |         # | ||||||
|  |         # So if we have a list of items with 240 items: | ||||||
|  |         # | ||||||
|  |         #   1st set: items[0:100] would give items at indexes 0 to 99 | ||||||
|  |         #   2nd set: items[100:200] would give items at indexes 100 to 199 | ||||||
|  |         #   3rd set: items[200:300] would give items at indexes 200 to 239 | ||||||
|  |         elements_subset: list = req.context.elements[first_element:last_element] | ||||||
|  |  | ||||||
|  |         views: dict = get_views( | ||||||
|  |             solr_date_string, elements_subset, req.context.views_facet_field | ||||||
|  |         ) | ||||||
|  |         downloads: dict = get_downloads( | ||||||
|  |             solr_date_string, elements_subset, req.context.downloads_facet_field | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         # create a list to hold dicts of stats | ||||||
|  |         statistics = [] | ||||||
|  |  | ||||||
|  |         # iterate over views dict to extract views and use the element id as an | ||||||
|  |         # index to the downloads dict to extract downloads. | ||||||
|  |         for k, v in views.items(): | ||||||
|  |             statistics.append({"id": k, "views": v, "downloads": downloads[k]}) | ||||||
|  |  | ||||||
|  |         message = { | ||||||
|  |             "currentPage": req.context.page, | ||||||
|  |             "totalPages": pages, | ||||||
|  |             "limit": req.context.limit, | ||||||
|  |             "statistics": statistics, | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         resp.status = falcon.HTTP_200 | ||||||
|  |         resp.media = message | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SingleStatisticsResource: | ||||||
|  |     @falcon.before(set_statistics_scope) | ||||||
|  |     def on_get(self, req, resp, id_): | ||||||
|  |         """Handles GET requests""" | ||||||
|  |  | ||||||
|  |         with DatabaseManager() as db: | ||||||
|  |             db.set_read_only(True) | ||||||
|  |  | ||||||
|  |             with db.cursor() as cursor: | ||||||
|  |                 cursor = db.cursor() | ||||||
|  |                 cursor.execute( | ||||||
|  |                     f"SELECT views, downloads FROM {req.context.database} WHERE id=%s", | ||||||
|  |                     [str(id_)], | ||||||
|  |                 ) | ||||||
|  |                 if cursor.rowcount == 0: | ||||||
|  |                     raise falcon.HTTPNotFound( | ||||||
|  |                         title=f"{req.context.statistics_scope} not found", | ||||||
|  |                         description=f'The {req.context.statistics_scope} with id "{str(id_)}" was not found.', | ||||||
|  |                     ) | ||||||
|  |                 else: | ||||||
|  |                     results = cursor.fetchone() | ||||||
|  |  | ||||||
|  |                     statistics = { | ||||||
|  |                         "id": str(id_), | ||||||
|  |                         "views": results["views"], | ||||||
|  |                         "downloads": results["downloads"], | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     resp.media = statistics | ||||||
|  |  | ||||||
|  |  | ||||||
|  | app = application = falcon.App() | ||||||
|  | app.add_route("/", RootResource()) | ||||||
|  | app.add_route("/status", StatusResource()) | ||||||
|  |  | ||||||
|  | # Item routes | ||||||
|  | app.add_route("/items", AllStatisticsResource()) | ||||||
|  | app.add_route("/item/{id_:uuid}", SingleStatisticsResource()) | ||||||
|  |  | ||||||
|  | # Community routes | ||||||
|  | app.add_route("/communities", AllStatisticsResource()) | ||||||
|  | app.add_route("/community/{id_:uuid}", SingleStatisticsResource()) | ||||||
|  |  | ||||||
|  | # Collection routes | ||||||
|  | app.add_route("/collections", AllStatisticsResource()) | ||||||
|  | app.add_route("/collection/{id_:uuid}", SingleStatisticsResource()) | ||||||
|  |  | ||||||
|  | # Route to the Swagger UI Openapp schema | ||||||
|  | app.add_route("/docs/openapi.json", OpenAPIJSONResource()) | ||||||
|  |  | ||||||
|  | # Path to host the Swagger UI. Keep in mind that Falcon will add a route for | ||||||
|  | # this automatically when we register Swagger and the path will be relative | ||||||
|  | # to the Falcon app like all other routes, not the absolute root. | ||||||
|  | SWAGGERUI_PATH = "/swagger" | ||||||
|  |  | ||||||
|  | # The *absolute* path to the OpenJSON schema. This must be absolute because | ||||||
|  | # it will be requested by the client and must resolve absolutely. Note: the | ||||||
|  | # name of this variable is misleading because it is actually the schema URL | ||||||
|  | # but we pass it into the register_swaggerui_app() function as the app_url | ||||||
|  | # parameter. | ||||||
|  | SWAGGERUI_API_URL = f"{DSPACE_STATISTICS_API_URL}/docs/openapi.json" | ||||||
|  |  | ||||||
|  | register_swaggerui_app( | ||||||
|  |     app, | ||||||
|  |     SWAGGERUI_PATH, | ||||||
|  |     SWAGGERUI_API_URL, | ||||||
|  |     config={ | ||||||
|  |         "supportedSubmitMethods": ["get", "post"], | ||||||
|  |     }, | ||||||
|  |     uri_prefix=DSPACE_STATISTICS_API_URL, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | # vim: set sw=4 ts=4 expandtab: | ||||||
| @@ -1,3 +1,5 @@ | |||||||
|  | # SPDX-License-Identifier: GPL-3.0-only | ||||||
|  | 
 | ||||||
| import os | import os | ||||||
| 
 | 
 | ||||||
| # Check if Solr connection information was provided in the environment | # Check if Solr connection information was provided in the environment | ||||||
| @@ -9,4 +11,13 @@ DATABASE_PASS = os.environ.get("DATABASE_PASS", "dspacestatistics") | |||||||
| DATABASE_HOST = os.environ.get("DATABASE_HOST", "localhost") | DATABASE_HOST = os.environ.get("DATABASE_HOST", "localhost") | ||||||
| DATABASE_PORT = os.environ.get("DATABASE_PORT", "5432") | DATABASE_PORT = os.environ.get("DATABASE_PORT", "5432") | ||||||
| 
 | 
 | ||||||
|  | # URL to DSpace Statistics API, which will be used as a prefix to API calls in | ||||||
|  | # the Swagger UI. An empty string will allow this to work out of the box in a | ||||||
|  | # local development environment, but for production it should be set to a value | ||||||
|  | # like "/rest/statistics", assuming that the statistics API is deployed next to | ||||||
|  | # the vanilla DSpace REST API. | ||||||
|  | DSPACE_STATISTICS_API_URL = os.environ.get("DSPACE_STATISTICS_API_URL", "") | ||||||
|  | 
 | ||||||
|  | VERSION = "1.4.6-dev" | ||||||
|  | 
 | ||||||
| # vim: set sw=4 ts=4 expandtab: | # vim: set sw=4 ts=4 expandtab: | ||||||
| @@ -1,6 +1,7 @@ | |||||||
|  | # SPDX-License-Identifier: GPL-3.0-only | ||||||
|  | 
 | ||||||
| import falcon | import falcon | ||||||
| import psycopg2 | import psycopg | ||||||
| import psycopg2.extras |  | ||||||
| 
 | 
 | ||||||
| from .config import ( | from .config import ( | ||||||
|     DATABASE_HOST, |     DATABASE_HOST, | ||||||
| @@ -19,10 +20,10 @@ class DatabaseManager: | |||||||
| 
 | 
 | ||||||
|     def __enter__(self): |     def __enter__(self): | ||||||
|         try: |         try: | ||||||
|             self._connection = psycopg2.connect( |             self._connection = psycopg.connect( | ||||||
|                 self._connection_uri, cursor_factory=psycopg2.extras.DictCursor |                 self._connection_uri, row_factory=psycopg.rows.dict_row | ||||||
|             ) |             ) | ||||||
|         except psycopg2.OperationalError: |         except psycopg.OperationalError: | ||||||
|             title = "500 Internal Server Error" |             title = "500 Internal Server Error" | ||||||
|             description = "Could not connect to database" |             description = "Could not connect to database" | ||||||
|             raise falcon.HTTPInternalServerError(title, description) |             raise falcon.HTTPInternalServerError(title, description) | ||||||
							
								
								
									
										616
									
								
								src/dspace_statistics_api/docs/openapi.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										616
									
								
								src/dspace_statistics_api/docs/openapi.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,616 @@ | |||||||
|  | { | ||||||
|  |   "openapi": "3.0.3", | ||||||
|  |   "info": { | ||||||
|  |     "version": "1.4.6-dev", | ||||||
|  |     "title": "DSpace Statistics API", | ||||||
|  |     "description": "A [Falcon-based](https://falcon.readthedocs.io/) web application to make DSpace's item, community, and collection statistics available via a simple REST API. This Swagger interface is powered by [falcon-swagger-ui](https://github.com/rdidyk/falcon-swagger-ui).", | ||||||
|  |     "license": { | ||||||
|  |       "name": "GPLv3.0", | ||||||
|  |       "url": "https://www.gnu.org/licenses/gpl-3.0.en.html" | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   "paths": { | ||||||
|  |     "/item/{item_uuid}": { | ||||||
|  |       "get": { | ||||||
|  |         "summary": "Statistics for a specific item", | ||||||
|  |         "operationId": "getItem", | ||||||
|  |         "tags": [ | ||||||
|  |           "item" | ||||||
|  |         ], | ||||||
|  |         "parameters": [ | ||||||
|  |           { | ||||||
|  |             "name": "item_uuid", | ||||||
|  |             "in": "path", | ||||||
|  |             "required": true, | ||||||
|  |             "description": "The UUID of the item to retrieve", | ||||||
|  |             "schema": { | ||||||
|  |               "type": "string", | ||||||
|  |               "format": "uuid", | ||||||
|  |               "example": "9596aeff-0b90-47d3-9fec-02d578920507" | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "Expected response to a valid request", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "$ref": "#/components/schemas/SingleElementResponse" | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "404": { | ||||||
|  |             "description": "Item not found" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "/items": { | ||||||
|  |       "get": { | ||||||
|  |         "summary": "Get statistics for all items", | ||||||
|  |         "operationId": "getItems", | ||||||
|  |         "tags": [ | ||||||
|  |           "items" | ||||||
|  |         ], | ||||||
|  |         "parameters": [ | ||||||
|  |           { | ||||||
|  |             "name": "limit", | ||||||
|  |             "in": "query", | ||||||
|  |             "description": "How many items to return at once (optional)", | ||||||
|  |             "required": false, | ||||||
|  |             "schema": { | ||||||
|  |               "type": "integer", | ||||||
|  |               "format": "int32", | ||||||
|  |               "minimum": 1, | ||||||
|  |               "maximum": 100, | ||||||
|  |               "default": 100, | ||||||
|  |               "example": 100 | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "name": "page", | ||||||
|  |             "in": "query", | ||||||
|  |             "description": "Page of results to start on (optional)", | ||||||
|  |             "required": false, | ||||||
|  |             "schema": { | ||||||
|  |               "type": "integer", | ||||||
|  |               "format": "int32", | ||||||
|  |               "minimum": 0, | ||||||
|  |               "default": 0, | ||||||
|  |               "example": 0 | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "A paged array of items", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "$ref": "#/components/schemas/SingleElementResponse" | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "400": { | ||||||
|  |             "description": "Bad request" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "post": { | ||||||
|  |         "summary": "Get statistics for a list of items with an optional date range", | ||||||
|  |         "operationId": "postItems", | ||||||
|  |         "tags": [ | ||||||
|  |           "items" | ||||||
|  |         ], | ||||||
|  |         "requestBody": { | ||||||
|  |           "content": { | ||||||
|  |             "application/json": { | ||||||
|  |               "schema": { | ||||||
|  |                 "type": "object", | ||||||
|  |                 "properties": { | ||||||
|  |                   "limit": { | ||||||
|  |                     "type": "integer", | ||||||
|  |                     "format": "int32", | ||||||
|  |                     "minimum": 1, | ||||||
|  |                     "maximum": 100, | ||||||
|  |                     "default": 100 | ||||||
|  |                   }, | ||||||
|  |                   "page": { | ||||||
|  |                     "type": "integer", | ||||||
|  |                     "format": "int32", | ||||||
|  |                     "minimum": 0, | ||||||
|  |                     "default": 0 | ||||||
|  |                   }, | ||||||
|  |                   "dateFrom": { | ||||||
|  |                     "type": "string", | ||||||
|  |                     "format": "date" | ||||||
|  |                   }, | ||||||
|  |                   "dateTo": { | ||||||
|  |                     "type": "string", | ||||||
|  |                     "format": "date" | ||||||
|  |                   }, | ||||||
|  |                   "items": { | ||||||
|  |                     "type": "array", | ||||||
|  |                     "items": { | ||||||
|  |                       "type": "string", | ||||||
|  |                       "format": "uuid" | ||||||
|  |                     } | ||||||
|  |                   } | ||||||
|  |                 }, | ||||||
|  |                 "example": { | ||||||
|  |                   "limit": 100, | ||||||
|  |                   "page": 0, | ||||||
|  |                   "dateFrom": "2020-01-01T00:00:00Z", | ||||||
|  |                   "dateTo": "2020-12-31T00:00:00Z", | ||||||
|  |                   "items": [ | ||||||
|  |                     "f44cf173-2344-4eb2-8f00-ee55df32c76f", | ||||||
|  |                     "2324aa41-e9de-4a2b-bc36-16241464683e", | ||||||
|  |                     "8542f9da-9ce1-4614-abf4-f2e3fdb4b305", | ||||||
|  |                     "0fe573e7-042a-4240-a4d9-753b61233908" | ||||||
|  |                   ] | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         }, | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "Expected response to a valid request", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "type": "object", | ||||||
|  |                   "properties": { | ||||||
|  |                     "currentPage": { | ||||||
|  |                       "type": "integer", | ||||||
|  |                       "format": "int32" | ||||||
|  |                     }, | ||||||
|  |                     "limit": { | ||||||
|  |                       "type": "integer", | ||||||
|  |                       "format": "int32" | ||||||
|  |                     }, | ||||||
|  |                     "totalPages": { | ||||||
|  |                       "type": "integer", | ||||||
|  |                       "format": "int32" | ||||||
|  |                     }, | ||||||
|  |                     "statistics": { | ||||||
|  |                       "$ref": "#/components/schemas/ListOfElements" | ||||||
|  |                     } | ||||||
|  |                   } | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "400": { | ||||||
|  |             "description": "Bad request" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "/community/{community_uuid}": { | ||||||
|  |       "get": { | ||||||
|  |         "summary": "Statistics for a specific community", | ||||||
|  |         "operationId": "getCommunity", | ||||||
|  |         "tags": [ | ||||||
|  |           "community" | ||||||
|  |         ], | ||||||
|  |         "parameters": [ | ||||||
|  |           { | ||||||
|  |             "name": "community_uuid", | ||||||
|  |             "in": "path", | ||||||
|  |             "required": true, | ||||||
|  |             "description": "The UUID of the community to retrieve", | ||||||
|  |             "schema": { | ||||||
|  |               "type": "string", | ||||||
|  |               "format": "uuid", | ||||||
|  |               "example": "bde7139c-d321-46bb-aef6-ae70799e5edb" | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "Expected response to a valid request", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "$ref": "#/components/schemas/SingleElementResponse" | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "404": { | ||||||
|  |             "description": "Community not found" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "/communities": { | ||||||
|  |       "get": { | ||||||
|  |         "summary": "Get statistics for all communities", | ||||||
|  |         "operationId": "getCommunities", | ||||||
|  |         "tags": [ | ||||||
|  |           "communities" | ||||||
|  |         ], | ||||||
|  |         "parameters": [ | ||||||
|  |           { | ||||||
|  |             "name": "limit", | ||||||
|  |             "in": "query", | ||||||
|  |             "description": "How many communities to return at once (optional)", | ||||||
|  |             "required": false, | ||||||
|  |             "schema": { | ||||||
|  |               "type": "integer", | ||||||
|  |               "format": "int32", | ||||||
|  |               "minimum": 1, | ||||||
|  |               "maximum": 100, | ||||||
|  |               "default": 100, | ||||||
|  |               "example": 100 | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "name": "page", | ||||||
|  |             "in": "query", | ||||||
|  |             "description": "Zero-based page of results to start on (optional)", | ||||||
|  |             "required": false, | ||||||
|  |             "schema": { | ||||||
|  |               "type": "integer", | ||||||
|  |               "format": "int32", | ||||||
|  |               "minimum": 0, | ||||||
|  |               "default": 0, | ||||||
|  |               "example": 0 | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "A paged array of communities", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "$ref": "#/components/schemas/SingleElementResponse" | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "400": { | ||||||
|  |             "description": "Bad request" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "post": { | ||||||
|  |         "summary": "Get statistics for a list of communities with an optional date range", | ||||||
|  |         "operationId": "postCommunities", | ||||||
|  |         "tags": [ | ||||||
|  |           "communities" | ||||||
|  |         ], | ||||||
|  |         "requestBody": { | ||||||
|  |           "content": { | ||||||
|  |             "application/json": { | ||||||
|  |               "schema": { | ||||||
|  |                 "type": "object", | ||||||
|  |                 "properties": { | ||||||
|  |                   "limit": { | ||||||
|  |                     "type": "integer", | ||||||
|  |                     "format": "int32", | ||||||
|  |                     "minimum": 1, | ||||||
|  |                     "maximum": 100, | ||||||
|  |                     "default": 100 | ||||||
|  |                   }, | ||||||
|  |                   "page": { | ||||||
|  |                     "type": "integer", | ||||||
|  |                     "format": "int32", | ||||||
|  |                     "minimum": 0, | ||||||
|  |                     "default": 0 | ||||||
|  |                   }, | ||||||
|  |                   "dateFrom": { | ||||||
|  |                     "type": "string", | ||||||
|  |                     "format": "date" | ||||||
|  |                   }, | ||||||
|  |                   "dateTo": { | ||||||
|  |                     "type": "string", | ||||||
|  |                     "format": "date" | ||||||
|  |                   }, | ||||||
|  |                   "communities": { | ||||||
|  |                     "type": "array", | ||||||
|  |                     "items": { | ||||||
|  |                       "type": "string", | ||||||
|  |                       "format": "uuid" | ||||||
|  |                     } | ||||||
|  |                   } | ||||||
|  |                 }, | ||||||
|  |                 "example": { | ||||||
|  |                   "limit": 100, | ||||||
|  |                   "page": 0, | ||||||
|  |                   "dateFrom": "2020-01-01T00:00:00Z", | ||||||
|  |                   "dateTo": "2020-12-31T00:00:00Z", | ||||||
|  |                   "communities": [ | ||||||
|  |                     "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |                     "8a8aeed1-077e-4360-bdf8-a5f3020193b1", | ||||||
|  |                     "47d0498a-203c-407d-afb8-1d44bf29badc", | ||||||
|  |                     "d3fe99a9-e27d-4035-9339-084c93228c82" | ||||||
|  |                   ] | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         }, | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "Expected response to a valid request", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "type": "object", | ||||||
|  |                   "properties": { | ||||||
|  |                     "currentPage": { | ||||||
|  |                       "type": "integer", | ||||||
|  |                       "format": "int32" | ||||||
|  |                     }, | ||||||
|  |                     "limit": { | ||||||
|  |                       "type": "integer", | ||||||
|  |                       "format": "int32" | ||||||
|  |                     }, | ||||||
|  |                     "totalPages": { | ||||||
|  |                       "type": "integer", | ||||||
|  |                       "format": "int32" | ||||||
|  |                     }, | ||||||
|  |                     "statistics": { | ||||||
|  |                       "$ref": "#/components/schemas/ListOfElements" | ||||||
|  |                     } | ||||||
|  |                   } | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "400": { | ||||||
|  |             "description": "Bad request" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "/collection/{collection_uuid}": { | ||||||
|  |       "get": { | ||||||
|  |         "summary": "Statistics for a specific collection", | ||||||
|  |         "operationId": "getCollection", | ||||||
|  |         "tags": [ | ||||||
|  |           "collection" | ||||||
|  |         ], | ||||||
|  |         "parameters": [ | ||||||
|  |           { | ||||||
|  |             "name": "collection_uuid", | ||||||
|  |             "in": "path", | ||||||
|  |             "required": true, | ||||||
|  |             "description": "The UUID of the collection to retrieve", | ||||||
|  |             "schema": { | ||||||
|  |               "type": "string", | ||||||
|  |               "format": "uuid", | ||||||
|  |               "example": "49dc95d8-bf2f-4e68-b30f-41ea266c37ae" | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "Expected response to a valid request", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "$ref": "#/components/schemas/SingleElementResponse" | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "404": { | ||||||
|  |             "description": "Collection not found" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "/collections": { | ||||||
|  |       "get": { | ||||||
|  |         "summary": "Get statistics for all collections", | ||||||
|  |         "operationId": "getCollections", | ||||||
|  |         "tags": [ | ||||||
|  |           "collections" | ||||||
|  |         ], | ||||||
|  |         "parameters": [ | ||||||
|  |           { | ||||||
|  |             "name": "limit", | ||||||
|  |             "in": "query", | ||||||
|  |             "description": "How many collections to return at once (optional)", | ||||||
|  |             "required": false, | ||||||
|  |             "schema": { | ||||||
|  |               "type": "integer", | ||||||
|  |               "format": "int32", | ||||||
|  |               "minimum": 1, | ||||||
|  |               "maximum": 100, | ||||||
|  |               "default": 100, | ||||||
|  |               "example": 100 | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "name": "page", | ||||||
|  |             "in": "query", | ||||||
|  |             "description": "Zero-based page of results to start on (optional)", | ||||||
|  |             "required": false, | ||||||
|  |             "schema": { | ||||||
|  |               "type": "integer", | ||||||
|  |               "format": "int32", | ||||||
|  |               "minimum": 0, | ||||||
|  |               "default": 0, | ||||||
|  |               "example": 0 | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "A paged array of collections", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "$ref": "#/components/schemas/SingleElementResponse" | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "400": { | ||||||
|  |             "description": "Bad request" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "post": { | ||||||
|  |         "summary": "Get statistics for a list of collections with an optional date range", | ||||||
|  |         "operationId": "postCollections", | ||||||
|  |         "tags": [ | ||||||
|  |           "collections" | ||||||
|  |         ], | ||||||
|  |         "requestBody": { | ||||||
|  |           "content": { | ||||||
|  |             "application/json": { | ||||||
|  |               "schema": { | ||||||
|  |                 "type": "object", | ||||||
|  |                 "properties": { | ||||||
|  |                   "limit": { | ||||||
|  |                     "type": "integer", | ||||||
|  |                     "format": "int32", | ||||||
|  |                     "minimum": 1, | ||||||
|  |                     "maximum": 100, | ||||||
|  |                     "default": 100 | ||||||
|  |                   }, | ||||||
|  |                   "page": { | ||||||
|  |                     "type": "integer", | ||||||
|  |                     "format": "int32", | ||||||
|  |                     "minimum": 0, | ||||||
|  |                     "default": 0 | ||||||
|  |                   }, | ||||||
|  |                   "dateFrom": { | ||||||
|  |                     "type": "string", | ||||||
|  |                     "format": "date" | ||||||
|  |                   }, | ||||||
|  |                   "dateTo": { | ||||||
|  |                     "type": "string", | ||||||
|  |                     "format": "date" | ||||||
|  |                   }, | ||||||
|  |                   "collections": { | ||||||
|  |                     "type": "array", | ||||||
|  |                     "items": { | ||||||
|  |                       "type": "string", | ||||||
|  |                       "format": "uuid" | ||||||
|  |                     } | ||||||
|  |                   } | ||||||
|  |                 }, | ||||||
|  |                 "example": { | ||||||
|  |                   "limit": 100, | ||||||
|  |                   "page": 0, | ||||||
|  |                   "dateFrom": "2020-01-01T00:00:00Z", | ||||||
|  |                   "dateTo": "2020-12-31T00:00:00Z", | ||||||
|  |                   "collections": [ | ||||||
|  |                     "5eeef6cf-b91b-42d0-9549-ea61bc8a758f", | ||||||
|  |                     "6aac3269-b4a9-4924-a24d-9e6ee2b410d2", | ||||||
|  |                     "551698dd-cd2b-4327-948e-54b5eb6deda5", | ||||||
|  |                     "39358713-bbaf-4149-a453-e2b18c09fd5d" | ||||||
|  |                   ] | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         }, | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "Expected response to a valid request", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "type": "object", | ||||||
|  |                   "properties": { | ||||||
|  |                     "currentPage": { | ||||||
|  |                       "type": "integer", | ||||||
|  |                       "format": "int32" | ||||||
|  |                     }, | ||||||
|  |                     "limit": { | ||||||
|  |                       "type": "integer", | ||||||
|  |                       "format": "int32" | ||||||
|  |                     }, | ||||||
|  |                     "totalPages": { | ||||||
|  |                       "type": "integer", | ||||||
|  |                       "format": "int32" | ||||||
|  |                     }, | ||||||
|  |                     "statistics": { | ||||||
|  |                       "$ref": "#/components/schemas/ListOfElements" | ||||||
|  |                     } | ||||||
|  |                   } | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "400": { | ||||||
|  |             "description": "Bad request" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "/status": { | ||||||
|  |       "get": { | ||||||
|  |         "summary": "Get API status", | ||||||
|  |         "operationId": "getStatus", | ||||||
|  |         "tags": [ | ||||||
|  |           "status" | ||||||
|  |         ], | ||||||
|  |         "responses": { | ||||||
|  |           "200": { | ||||||
|  |             "description": "OK", | ||||||
|  |             "content": { | ||||||
|  |               "application/json": { | ||||||
|  |                 "schema": { | ||||||
|  |                   "type": "object", | ||||||
|  |                   "properties": { | ||||||
|  |                     "version": { | ||||||
|  |                       "type": "string", | ||||||
|  |                       "example": "1.4.0-dev" | ||||||
|  |                     } | ||||||
|  |                   } | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           "405": { | ||||||
|  |             "description": "Method Not Allowed" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   "components": { | ||||||
|  |     "schemas": { | ||||||
|  |       "SingleElementResponse": { | ||||||
|  |         "type": "object", | ||||||
|  |         "required": [ | ||||||
|  |           "id", | ||||||
|  |           "views", | ||||||
|  |           "downloads" | ||||||
|  |         ], | ||||||
|  |         "properties": { | ||||||
|  |           "id": { | ||||||
|  |             "type": "string", | ||||||
|  |             "format": "uuid" | ||||||
|  |           }, | ||||||
|  |           "views": { | ||||||
|  |             "type": "integer", | ||||||
|  |             "example": 450 | ||||||
|  |           }, | ||||||
|  |           "downloads": { | ||||||
|  |             "type": "integer", | ||||||
|  |             "example": 1337 | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "ListOfElements": { | ||||||
|  |         "type": "array", | ||||||
|  |         "items": { | ||||||
|  |           "$ref": "#/components/schemas/SingleElementResponse" | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,34 +1,20 @@ | |||||||
|  | # SPDX-License-Identifier: GPL-3.0-only | ||||||
| # | # | ||||||
| # indexer.py | # indexer.py | ||||||
| # | # | ||||||
| # Copyright 2018 Alan Orth. | # Connects to a DSpace Solr statistics core and ingests views and downloads for | ||||||
| # | # communities, collections, and items into a PostgreSQL database. | ||||||
| # This program is free software: you can redistribute it and/or modify |  | ||||||
| # it under the terms of the GNU General Public License as published by |  | ||||||
| # the Free Software Foundation, either version 3 of the License, or |  | ||||||
| # (at your option) any later version. |  | ||||||
| # |  | ||||||
| # This program is distributed in the hope that it will be useful, |  | ||||||
| # but WITHOUT ANY WARRANTY; without even the implied warranty of |  | ||||||
| # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the |  | ||||||
| # GNU General Public License for more details. |  | ||||||
| # |  | ||||||
| # You should have received a copy of the GNU General Public License |  | ||||||
| # along with this program.  If not, see <http://www.gnu.org/licenses/>. |  | ||||||
| # |  | ||||||
| # --- |  | ||||||
| # |  | ||||||
| # Connects to a DSpace Solr statistics core and ingests item views and downloads |  | ||||||
| # into a PostgreSQL database for use by other applications (like an API). |  | ||||||
| # | # | ||||||
| # This script is written for Python 3.6+ and requires several modules that you | # This script is written for Python 3.6+ and requires several modules that you | ||||||
| # can install with pip (I recommend using a Python virtual environment): | # can install with pip (I recommend using a Python virtual environment): | ||||||
| # | # | ||||||
| #   $ pip install psycopg2-binary | #   $ pip install psycopg | ||||||
| # | # | ||||||
| # See: https://wiki.duraspace.org/display/DSPACE/Solr | # See: https://wiki.duraspace.org/display/DSPACE/Solr | ||||||
| 
 | 
 | ||||||
| import psycopg2.extras | import math | ||||||
|  | 
 | ||||||
|  | import psycopg | ||||||
| import requests | import requests | ||||||
| 
 | 
 | ||||||
| from .config import SOLR_SERVER | from .config import SOLR_SERVER | ||||||
| @@ -36,7 +22,7 @@ from .database import DatabaseManager | |||||||
| from .util import get_statistics_shards | from .util import get_statistics_shards | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def index_views(): | def index_views(indexType: str, facetField: str): | ||||||
|     # get total number of distinct facets for items with a minimum of 1 view, |     # get total number of distinct facets for items with a minimum of 1 view, | ||||||
|     # otherwise Solr returns all kinds of weird ids that are actually not in |     # otherwise Solr returns all kinds of weird ids that are actually not in | ||||||
|     # the database. Also, stats are expensive, but we need stats.calcdistinct |     # the database. Also, stats are expensive, but we need stats.calcdistinct | ||||||
| @@ -45,15 +31,16 @@ def index_views(): | |||||||
|     # |     # | ||||||
|     # see: https://lucene.apache.org/solr/guide/6_6/the-stats-component.html |     # see: https://lucene.apache.org/solr/guide/6_6/the-stats-component.html | ||||||
|     solr_query_params = { |     solr_query_params = { | ||||||
|         "q": "type:2", |         "q": f"type:2 AND {facetField}:/.{{36}}/", | ||||||
|         "fq": "isBot:false AND statistics_type:view", |         "fq": "-isBot:true AND statistics_type:view", | ||||||
|  |         "fl": facetField, | ||||||
|         "facet": "true", |         "facet": "true", | ||||||
|         "facet.field": "id", |         "facet.field": facetField, | ||||||
|         "facet.mincount": 1, |         "facet.mincount": 1, | ||||||
|         "facet.limit": 1, |         "facet.limit": 1, | ||||||
|         "facet.offset": 0, |         "facet.offset": 0, | ||||||
|         "stats": "true", |         "stats": "true", | ||||||
|         "stats.field": "id", |         "stats.field": facetField, | ||||||
|         "stats.calcdistinct": "true", |         "stats.calcdistinct": "true", | ||||||
|         "shards": shards, |         "shards": shards, | ||||||
|         "rows": 0, |         "rows": 0, | ||||||
| @@ -66,17 +53,17 @@ def index_views(): | |||||||
| 
 | 
 | ||||||
|     try: |     try: | ||||||
|         # get total number of distinct facets (countDistinct) |         # get total number of distinct facets (countDistinct) | ||||||
|         results_totalNumFacets = res.json()["stats"]["stats_fields"]["id"][ |         results_totalNumFacets = res.json()["stats"]["stats_fields"][facetField][ | ||||||
|             "countDistinct" |             "countDistinct" | ||||||
|         ] |         ] | ||||||
|     except TypeError: |     except TypeError: | ||||||
|         print("No item views to index, exiting.") |         print(f"{indexType}: no views, exiting.") | ||||||
| 
 | 
 | ||||||
|         exit(0) |         exit(0) | ||||||
| 
 | 
 | ||||||
|     # divide results into "pages" (cast to int to effectively round down) |     # divide results into "pages" and round up to next integer | ||||||
|     results_per_page = 100 |     results_per_page = 100 | ||||||
|     results_num_pages = int(results_totalNumFacets / results_per_page) |     results_num_pages = math.ceil(results_totalNumFacets / results_per_page) | ||||||
|     results_current_page = 0 |     results_current_page = 0 | ||||||
| 
 | 
 | ||||||
|     with DatabaseManager() as db: |     with DatabaseManager() as db: | ||||||
| @@ -87,14 +74,15 @@ def index_views(): | |||||||
|             while results_current_page <= results_num_pages: |             while results_current_page <= results_num_pages: | ||||||
|                 # "pages" are zero based, but one based is more human readable |                 # "pages" are zero based, but one based is more human readable | ||||||
|                 print( |                 print( | ||||||
|                     f"Indexing item views (page {results_current_page + 1} of {results_num_pages + 1})" |                     f"{indexType}: indexing views (page {results_current_page + 1} of {results_num_pages + 1})" | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|                 solr_query_params = { |                 solr_query_params = { | ||||||
|                     "q": "type:2", |                     "q": f"type:2 AND {facetField}:/.{{36}}/", | ||||||
|                     "fq": "isBot:false AND statistics_type:view", |                     "fq": "-isBot:true AND statistics_type:view", | ||||||
|  |                     "fl": facetField, | ||||||
|                     "facet": "true", |                     "facet": "true", | ||||||
|                     "facet.field": "id", |                     "facet.field": facetField, | ||||||
|                     "facet.mincount": 1, |                     "facet.mincount": 1, | ||||||
|                     "facet.limit": results_per_page, |                     "facet.limit": results_per_page, | ||||||
|                     "facet.offset": results_current_page * results_per_page, |                     "facet.offset": results_current_page * results_per_page, | ||||||
| @@ -108,13 +96,13 @@ def index_views(): | |||||||
| 
 | 
 | ||||||
|                 # Solr returns facets as a dict of dicts (see json.nl parameter) |                 # Solr returns facets as a dict of dicts (see json.nl parameter) | ||||||
|                 views = res.json()["facet_counts"]["facet_fields"] |                 views = res.json()["facet_counts"]["facet_fields"] | ||||||
|                 # iterate over the 'id' dict and get the item ids and views |                 # iterate over the facetField dict and get the ids and views | ||||||
|                 for item_id, item_views in views["id"].items(): |                 for id_, views in views[facetField].items(): | ||||||
|                     data.append((item_id, item_views)) |                     data.append((id_, views)) | ||||||
| 
 | 
 | ||||||
|                 # do a batch insert of values from the current "page" of results |                 # do a batch insert of values from the current "page" of results | ||||||
|                 sql = "INSERT INTO items(id, views) VALUES %s ON CONFLICT(id) DO UPDATE SET views=excluded.views" |                 sql = f"INSERT INTO {indexType}(id, views) VALUES (%s, %s) ON CONFLICT(id) DO UPDATE SET views=excluded.views" | ||||||
|                 psycopg2.extras.execute_values(cursor, sql, data, template="(%s, %s)") |                 cursor.executemany(sql, data) | ||||||
|                 db.commit() |                 db.commit() | ||||||
| 
 | 
 | ||||||
|                 # clear all items from the list so we can populate it with the next batch |                 # clear all items from the list so we can populate it with the next batch | ||||||
| @@ -123,18 +111,19 @@ def index_views(): | |||||||
|                 results_current_page += 1 |                 results_current_page += 1 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def index_downloads(): | def index_downloads(indexType: str, facetField: str): | ||||||
|     # get the total number of distinct facets for items with at least 1 download |     # get the total number of distinct facets for items with at least 1 download | ||||||
|     solr_query_params = { |     solr_query_params = { | ||||||
|         "q": "type:0", |         "q": f"type:0 AND {facetField}:/.{{36}}/", | ||||||
|         "fq": "isBot:false AND statistics_type:view AND bundleName:ORIGINAL", |         "fq": "-isBot:true AND statistics_type:view AND bundleName:ORIGINAL", | ||||||
|  |         "fl": facetField, | ||||||
|         "facet": "true", |         "facet": "true", | ||||||
|         "facet.field": "owningItem", |         "facet.field": facetField, | ||||||
|         "facet.mincount": 1, |         "facet.mincount": 1, | ||||||
|         "facet.limit": 1, |         "facet.limit": 1, | ||||||
|         "facet.offset": 0, |         "facet.offset": 0, | ||||||
|         "stats": "true", |         "stats": "true", | ||||||
|         "stats.field": "owningItem", |         "stats.field": facetField, | ||||||
|         "stats.calcdistinct": "true", |         "stats.calcdistinct": "true", | ||||||
|         "shards": shards, |         "shards": shards, | ||||||
|         "rows": 0, |         "rows": 0, | ||||||
| @@ -147,17 +136,16 @@ def index_downloads(): | |||||||
| 
 | 
 | ||||||
|     try: |     try: | ||||||
|         # get total number of distinct facets (countDistinct) |         # get total number of distinct facets (countDistinct) | ||||||
|         results_totalNumFacets = res.json()["stats"]["stats_fields"]["owningItem"][ |         results_totalNumFacets = res.json()["stats"]["stats_fields"][facetField][ | ||||||
|             "countDistinct" |             "countDistinct" | ||||||
|         ] |         ] | ||||||
|     except TypeError: |     except TypeError: | ||||||
|         print("No item downloads to index, exiting.") |         print(f"{indexType}: no downloads, exiting.") | ||||||
| 
 | 
 | ||||||
|         exit(0) |         exit(0) | ||||||
| 
 | 
 | ||||||
|     # divide results into "pages" (cast to int to effectively round down) |  | ||||||
|     results_per_page = 100 |     results_per_page = 100 | ||||||
|     results_num_pages = int(results_totalNumFacets / results_per_page) |     results_num_pages = math.ceil(results_totalNumFacets / results_per_page) | ||||||
|     results_current_page = 0 |     results_current_page = 0 | ||||||
| 
 | 
 | ||||||
|     with DatabaseManager() as db: |     with DatabaseManager() as db: | ||||||
| @@ -168,14 +156,15 @@ def index_downloads(): | |||||||
|             while results_current_page <= results_num_pages: |             while results_current_page <= results_num_pages: | ||||||
|                 # "pages" are zero based, but one based is more human readable |                 # "pages" are zero based, but one based is more human readable | ||||||
|                 print( |                 print( | ||||||
|                     f"Indexing item downloads (page {results_current_page + 1} of {results_num_pages + 1})" |                     f"{indexType}: indexing downloads (page {results_current_page + 1} of {results_num_pages + 1})" | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|                 solr_query_params = { |                 solr_query_params = { | ||||||
|                     "q": "type:0", |                     "q": f"type:0 AND {facetField}:/.{{36}}/", | ||||||
|                     "fq": "isBot:false AND statistics_type:view AND bundleName:ORIGINAL", |                     "fq": "-isBot:true AND statistics_type:view AND bundleName:ORIGINAL", | ||||||
|  |                     "fl": facetField, | ||||||
|                     "facet": "true", |                     "facet": "true", | ||||||
|                     "facet.field": "owningItem", |                     "facet.field": facetField, | ||||||
|                     "facet.mincount": 1, |                     "facet.mincount": 1, | ||||||
|                     "facet.limit": results_per_page, |                     "facet.limit": results_per_page, | ||||||
|                     "facet.offset": results_current_page * results_per_page, |                     "facet.offset": results_current_page * results_per_page, | ||||||
| @@ -189,13 +178,13 @@ def index_downloads(): | |||||||
| 
 | 
 | ||||||
|                 # Solr returns facets as a dict of dicts (see json.nl parameter) |                 # Solr returns facets as a dict of dicts (see json.nl parameter) | ||||||
|                 downloads = res.json()["facet_counts"]["facet_fields"] |                 downloads = res.json()["facet_counts"]["facet_fields"] | ||||||
|                 # iterate over the 'owningItem' dict and get the item ids and downloads |                 # iterate over the facetField dict and get the item ids and downloads | ||||||
|                 for item_id, item_downloads in downloads["owningItem"].items(): |                 for id_, downloads in downloads[facetField].items(): | ||||||
|                     data.append((item_id, item_downloads)) |                     data.append((id_, downloads)) | ||||||
| 
 | 
 | ||||||
|                 # do a batch insert of values from the current "page" of results |                 # do a batch insert of values from the current "page" of results | ||||||
|                 sql = "INSERT INTO items(id, downloads) VALUES %s ON CONFLICT(id) DO UPDATE SET downloads=excluded.downloads" |                 sql = f"INSERT INTO {indexType}(id, downloads) VALUES (%s, %s) ON CONFLICT(id) DO UPDATE SET downloads=excluded.downloads" | ||||||
|                 psycopg2.extras.execute_values(cursor, sql, data, template="(%s, %s)") |                 cursor.executemany(sql, data) | ||||||
|                 db.commit() |                 db.commit() | ||||||
| 
 | 
 | ||||||
|                 # clear all items from the list so we can populate it with the next batch |                 # clear all items from the list so we can populate it with the next batch | ||||||
| @@ -211,13 +200,32 @@ with DatabaseManager() as db: | |||||||
|             """CREATE TABLE IF NOT EXISTS items |             """CREATE TABLE IF NOT EXISTS items | ||||||
|                   (id UUID PRIMARY KEY, views INT DEFAULT 0, downloads INT DEFAULT 0)""" |                   (id UUID PRIMARY KEY, views INT DEFAULT 0, downloads INT DEFAULT 0)""" | ||||||
|         ) |         ) | ||||||
|  |         # create table to store community views and downloads | ||||||
|  |         cursor.execute( | ||||||
|  |             """CREATE TABLE IF NOT EXISTS communities | ||||||
|  |                   (id UUID PRIMARY KEY, views INT DEFAULT 0, downloads INT DEFAULT 0)""" | ||||||
|  |         ) | ||||||
|  |         # create table to store collection views and downloads | ||||||
|  |         cursor.execute( | ||||||
|  |             """CREATE TABLE IF NOT EXISTS collections | ||||||
|  |                   (id UUID PRIMARY KEY, views INT DEFAULT 0, downloads INT DEFAULT 0)""" | ||||||
|  |         ) | ||||||
| 
 | 
 | ||||||
|     # commit the table creation before closing the database connection |     # commit the table creation before closing the database connection | ||||||
|     db.commit() |     db.commit() | ||||||
| 
 | 
 | ||||||
| shards = get_statistics_shards() | shards = get_statistics_shards() | ||||||
| 
 | 
 | ||||||
| index_views() | # Index views and downloads for items, communities, and collections. Here the | ||||||
| index_downloads() | # first parameter is the type of indexing to perform, and the second parameter | ||||||
|  | # is the field to facet by in Solr's statistics to get this information. | ||||||
|  | 
 | ||||||
|  | index_views("items", "id") | ||||||
|  | index_views("communities", "owningComm") | ||||||
|  | index_views("collections", "owningColl") | ||||||
|  | 
 | ||||||
|  | index_downloads("items", "owningItem") | ||||||
|  | index_downloads("communities", "owningComm") | ||||||
|  | index_downloads("collections", "owningColl") | ||||||
| 
 | 
 | ||||||
| # vim: set sw=4 ts=4 expandtab: | # vim: set sw=4 ts=4 expandtab: | ||||||
							
								
								
									
										126
									
								
								src/dspace_statistics_api/stats.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										126
									
								
								src/dspace_statistics_api/stats.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,126 @@ | |||||||
|  | # SPDX-License-Identifier: GPL-3.0-only | ||||||
|  |  | ||||||
|  | import requests | ||||||
|  |  | ||||||
|  | from .config import SOLR_SERVER | ||||||
|  | from .util import get_statistics_shards | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_views(solr_date_string: str, elements: list, facetField: str): | ||||||
|  |     """ | ||||||
|  |     Get view statistics for a list of elements from Solr. Depending on the req- | ||||||
|  |     uest this could be items, communities, or collections. | ||||||
|  |  | ||||||
|  |     :parameter solr_date_string (str): Solr date string, for example "[* TO *]" | ||||||
|  |     :parameter elements (list): a list of IDs | ||||||
|  |     :parameter facetField (str): Solr field to facet by, for example "id" | ||||||
|  |     :returns: A dict of IDs and views | ||||||
|  |     """ | ||||||
|  |     shards = get_statistics_shards() | ||||||
|  |  | ||||||
|  |     # Join the UUIDs with "OR" and escape the hyphens for Solr | ||||||
|  |     solr_elements_string: str = " OR ".join(elements).replace("-", r"\-") | ||||||
|  |  | ||||||
|  |     solr_query_params = { | ||||||
|  |         "q": f"{facetField}:({solr_elements_string})", | ||||||
|  |         "fq": f"type:2 AND -isBot:true AND statistics_type:view AND time:{solr_date_string}", | ||||||
|  |         "fl": facetField, | ||||||
|  |         "facet": "true", | ||||||
|  |         "facet.field": facetField, | ||||||
|  |         "facet.mincount": 1, | ||||||
|  |         "shards": shards, | ||||||
|  |         "rows": 0, | ||||||
|  |         "wt": "json", | ||||||
|  |         "json.nl": "map",  # return facets as a dict instead of a flat list | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     solr_url = SOLR_SERVER + "/statistics/select" | ||||||
|  |     res = requests.get(solr_url, params=solr_query_params) | ||||||
|  |  | ||||||
|  |     # Create an empty dict to store views | ||||||
|  |     data = {} | ||||||
|  |  | ||||||
|  |     # Solr returns facets as a dict of dicts (see the json.nl parameter) | ||||||
|  |     views = res.json()["facet_counts"]["facet_fields"] | ||||||
|  |     # iterate over the facetField dict and ids and views | ||||||
|  |     for id_, views in views[facetField].items(): | ||||||
|  |         # For items we can rely on Solr returning facets for the *only* the ids | ||||||
|  |         # in our query, but for communities and collections, the owningComm and | ||||||
|  |         # owningColl fields are multi-value so Solr will return facets with the | ||||||
|  |         # values in our query as well as *any others* that happen to be present | ||||||
|  |         # in the field (which looks like Solr returning unrelated results until | ||||||
|  |         # you realize that the field is multi-value and this is correct). | ||||||
|  |         # | ||||||
|  |         # To work around this I make sure that each id in the returned dict are | ||||||
|  |         # present in the elements list POSTed by the user. | ||||||
|  |         if id_ in elements: | ||||||
|  |             data[id_] = views | ||||||
|  |  | ||||||
|  |     # Check if any ids have missing stats so we can set them to 0 | ||||||
|  |     if len(data) < len(elements): | ||||||
|  |         # List comprehension to get a list of ids (keys) in the data | ||||||
|  |         data_ids = [k for k, v in data.items()] | ||||||
|  |         for element_id in elements: | ||||||
|  |             if element_id not in data_ids: | ||||||
|  |                 data[element_id] = 0 | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |     return data | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_downloads(solr_date_string: str, elements: list, facetField: str): | ||||||
|  |     """ | ||||||
|  |     Get download statistics for a list of items from Solr. Depending on the req- | ||||||
|  |     uest this could be items, communities, or collections. | ||||||
|  |  | ||||||
|  |     :parameter solr_date_string (str): Solr date string, for example "[* TO *]" | ||||||
|  |     :parameter elements (list): a list of IDs | ||||||
|  |     :parameter facetField (str): Solr field to facet by, for example "id" | ||||||
|  |     :returns: A dict of IDs and downloads | ||||||
|  |     """ | ||||||
|  |     shards = get_statistics_shards() | ||||||
|  |  | ||||||
|  |     # Join the UUIDs with "OR" and escape the hyphens for Solr | ||||||
|  |     solr_elements_string: str = " OR ".join(elements).replace("-", r"\-") | ||||||
|  |  | ||||||
|  |     solr_query_params = { | ||||||
|  |         "q": f"{facetField}:({solr_elements_string})", | ||||||
|  |         "fq": f"type:0 AND -isBot:true AND statistics_type:view AND bundleName:ORIGINAL AND time:{solr_date_string}", | ||||||
|  |         "fl": facetField, | ||||||
|  |         "facet": "true", | ||||||
|  |         "facet.field": facetField, | ||||||
|  |         "facet.mincount": 1, | ||||||
|  |         "shards": shards, | ||||||
|  |         "rows": 0, | ||||||
|  |         "wt": "json", | ||||||
|  |         "json.nl": "map",  # return facets as a dict instead of a flat list | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     solr_url = SOLR_SERVER + "/statistics/select" | ||||||
|  |     res = requests.get(solr_url, params=solr_query_params) | ||||||
|  |  | ||||||
|  |     # Create an empty dict to store downloads | ||||||
|  |     data = {} | ||||||
|  |  | ||||||
|  |     # Solr returns facets as a dict of dicts (see the json.nl parameter) | ||||||
|  |     downloads = res.json()["facet_counts"]["facet_fields"] | ||||||
|  |     # Iterate over the facetField dict and get the ids and downloads | ||||||
|  |     for id_, downloads in downloads[facetField].items(): | ||||||
|  |         # Make sure that each id in the returned dict are present in the | ||||||
|  |         # elements list POSTed by the user. | ||||||
|  |         if id_ in elements: | ||||||
|  |             data[id_] = downloads | ||||||
|  |  | ||||||
|  |     # Check if any elements have missing stats so we can set them to 0 | ||||||
|  |     if len(data) < len(elements): | ||||||
|  |         # List comprehension to get a list of ids (keys) in the data | ||||||
|  |         data_ids = [k for k, v in data.items()] | ||||||
|  |         for element_id in elements: | ||||||
|  |             if element_id not in data_ids: | ||||||
|  |                 data[element_id] = 0 | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |     return data | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # vim: set sw=4 ts=4 expandtab: | ||||||
| @@ -1,4 +1,13 @@ | |||||||
|  | # SPDX-License-Identifier: GPL-3.0-only | ||||||
|  | 
 | ||||||
|  | import datetime | ||||||
|  | import json | ||||||
|  | import re | ||||||
|  | 
 | ||||||
| import falcon | import falcon | ||||||
|  | import requests | ||||||
|  | 
 | ||||||
|  | from .config import SOLR_SERVER | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def get_statistics_shards(): | def get_statistics_shards(): | ||||||
| @@ -8,11 +17,6 @@ def get_statistics_shards(): | |||||||
|     Returns: |     Returns: | ||||||
|         str:A list of Solr statistics shards separated by commas. |         str:A list of Solr statistics shards separated by commas. | ||||||
|     """ |     """ | ||||||
|     import re |  | ||||||
| 
 |  | ||||||
|     import requests |  | ||||||
| 
 |  | ||||||
|     from .config import SOLR_SERVER |  | ||||||
| 
 | 
 | ||||||
|     # Initialize an empty list for statistics core years |     # Initialize an empty list for statistics core years | ||||||
|     statistics_core_years = [] |     statistics_core_years = [] | ||||||
| @@ -58,8 +62,6 @@ def get_statistics_shards(): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def is_valid_date(date): | def is_valid_date(date): | ||||||
|     import datetime |  | ||||||
| 
 |  | ||||||
|     try: |     try: | ||||||
|         # Solr date format is: 2020-01-01T00:00:00Z |         # Solr date format is: 2020-01-01T00:00:00Z | ||||||
|         # See: https://docs.python.org/3/library/datetime.html#strftime-strptime-behavior |         # See: https://docs.python.org/3/library/datetime.html#strftime-strptime-behavior | ||||||
| @@ -73,12 +75,12 @@ def is_valid_date(date): | |||||||
|         ) |         ) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def validate_items_post_parameters(req, resp, resource, params): | def validate_post_parameters(req, resp, resource, params): | ||||||
|     """Check the POSTed request parameters for the `/items` endpoint. |     """Check the POSTed request parameters for the `/items`, `/communities` and | ||||||
|  |     `/collections` endpoints. | ||||||
| 
 | 
 | ||||||
|     Meant to be used as a `before` hook. |     Meant to be used as a `before` hook. | ||||||
|     """ |     """ | ||||||
|     import json |  | ||||||
| 
 | 
 | ||||||
|     # Only attempt to read the POSTed request if its length is not 0 (or |     # Only attempt to read the POSTed request if its length is not 0 (or | ||||||
|     # rather, in the Python sense, if length is not a False-y value). |     # rather, in the Python sense, if length is not a False-y value). | ||||||
| @@ -103,12 +105,12 @@ def validate_items_post_parameters(req, resp, resource, params): | |||||||
| 
 | 
 | ||||||
|     # Parse the limit parameter from the POST request body |     # Parse the limit parameter from the POST request body | ||||||
|     if "limit" in doc: |     if "limit" in doc: | ||||||
|         if isinstance(doc["limit"], int) and 0 < doc["limit"] < 100: |         if isinstance(doc["limit"], int) and 0 < doc["limit"] <= 100: | ||||||
|             req.context.limit = doc["limit"] |             req.context.limit = doc["limit"] | ||||||
|         else: |         else: | ||||||
|             raise falcon.HTTPBadRequest( |             raise falcon.HTTPBadRequest( | ||||||
|                 title="Invalid parameter", |                 title="Invalid parameter", | ||||||
|                 description='The "limit" parameter is invalid. The value must be an integer between 0 and 100.', |                 description='The "limit" parameter is invalid. The value must be an integer between 1 and 100.', | ||||||
|             ) |             ) | ||||||
|     else: |     else: | ||||||
|         req.context.limit = 100 |         req.context.limit = 100 | ||||||
| @@ -125,14 +127,67 @@ def validate_items_post_parameters(req, resp, resource, params): | |||||||
|     else: |     else: | ||||||
|         req.context.page = 0 |         req.context.page = 0 | ||||||
| 
 | 
 | ||||||
|     # Parse the list of items from the POST request body |     # Parse the list of elements from the POST request body | ||||||
|     if "items" in doc: |     if req.context.statistics_scope in doc: | ||||||
|         if isinstance(doc["items"], list) and len(doc["items"]) > 0: |         if ( | ||||||
|             req.context.items = doc["items"] |             isinstance(doc[req.context.statistics_scope], list) | ||||||
|  |             and len(doc[req.context.statistics_scope]) > 0 | ||||||
|  |         ): | ||||||
|  |             req.context.elements = doc[req.context.statistics_scope] | ||||||
|         else: |         else: | ||||||
|             raise falcon.HTTPBadRequest( |             raise falcon.HTTPBadRequest( | ||||||
|                 title="Invalid parameter", |                 title="Invalid parameter", | ||||||
|                 description='The "items" parameter is invalid. The value must be a comma-separated list of item UUIDs.', |                 description=f'The "{req.context.statistics_scope}" parameter is invalid. The value must be a comma-separated list of UUIDs.', | ||||||
|             ) |             ) | ||||||
|     else: |     else: | ||||||
|         req.context.items = list() |         req.context.elements = [] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def set_statistics_scope(req, resp, resource, params): | ||||||
|  |     """Set the statistics scope (item, collection, or community) of the request | ||||||
|  |     as well as the appropriate database (for GET requests) and Solr facet fields | ||||||
|  |     (for POST requests). | ||||||
|  | 
 | ||||||
|  |     Meant to be used as a `before` hook. | ||||||
|  |     """ | ||||||
|  | 
 | ||||||
|  |     # Extract the scope from the request path. This is *guaranteed* to be one | ||||||
|  |     # of the following values because we only send requests matching these few | ||||||
|  |     # patterns to routes using this set_statistics_scope hook. | ||||||
|  |     # | ||||||
|  |     # Note: this regex is ordered so that "items" and "collections" match before | ||||||
|  |     # "item" and "collection". | ||||||
|  |     req.context.statistics_scope = re.findall( | ||||||
|  |         r"^/(communities|community|collections|collection|items|item)", req.path | ||||||
|  |     )[0] | ||||||
|  | 
 | ||||||
|  |     # Set the correct database based on the statistics_scope. The database is | ||||||
|  |     # used for all GET requests where statistics are returned directly from the | ||||||
|  |     # database. In this case we can return early. | ||||||
|  |     if req.method == "GET": | ||||||
|  |         if re.findall(r"^(item|items)$", req.context.statistics_scope): | ||||||
|  |             req.context.database = "items" | ||||||
|  |         elif re.findall(r"^(community|communities)$", req.context.statistics_scope): | ||||||
|  |             req.context.database = "communities" | ||||||
|  |         elif re.findall(r"^(collection|collections)$", req.context.statistics_scope): | ||||||
|  |             req.context.database = "collections" | ||||||
|  | 
 | ||||||
|  |         # GET requests only need the scope and the database so we can return now | ||||||
|  |         return | ||||||
|  | 
 | ||||||
|  |     # If the current request is for a plural items, communities, or collections | ||||||
|  |     # that includes a list of element ids POSTed with the request body then we | ||||||
|  |     # need to set the Solr facet field so we can get the live results. | ||||||
|  |     if req.method == "POST": | ||||||
|  |         if req.context.statistics_scope == "items": | ||||||
|  |             req.context.views_facet_field = "id" | ||||||
|  |             req.context.downloads_facet_field = "owningItem" | ||||||
|  |         elif req.context.statistics_scope == "communities": | ||||||
|  |             req.context.views_facet_field = "owningComm" | ||||||
|  |             req.context.downloads_facet_field = "owningComm" | ||||||
|  |         elif req.context.statistics_scope == "collections": | ||||||
|  |             req.context.views_facet_field = "owningColl" | ||||||
|  |             req.context.downloads_facet_field = "owningColl" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # vim: set sw=4 ts=4 expandtab: | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										379
									
								
								tests/test_api_collections.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										379
									
								
								tests/test_api_collections.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,379 @@ | |||||||
|  | # SPDX-License-Identifier: GPL-3.0-only | ||||||
|  |  | ||||||
|  | import json | ||||||
|  | from unittest.mock import patch | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  | from falcon import testing | ||||||
|  |  | ||||||
|  | from dspace_statistics_api.app import app | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture | ||||||
|  | def client(): | ||||||
|  |     return testing.TestClient(app) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_collection(client): | ||||||
|  |     """Test requesting a single collection.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/collection/8ea4b611-1f59-4d4e-b78d-a9921a72cfe7") | ||||||
|  |     response_doc = json.loads(response.text) | ||||||
|  |  | ||||||
|  |     assert isinstance(response_doc["downloads"], int) | ||||||
|  |     assert isinstance(response_doc["id"], str) | ||||||
|  |     assert isinstance(response_doc["views"], int) | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_missing_collection(client): | ||||||
|  |     """Test requesting a single non-existing collection.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/collection/508abe0a-689f-402e-885d-2f6b02e7a39c") | ||||||
|  |  | ||||||
|  |     assert response.status_code == 404 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_collections(client): | ||||||
|  |     """Test requesting 100 collections.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/collections", query_string="limit=100") | ||||||
|  |     response_doc = json.loads(response.text) | ||||||
|  |  | ||||||
|  |     assert isinstance(response_doc["currentPage"], int) | ||||||
|  |     assert isinstance(response_doc["totalPages"], int) | ||||||
|  |     assert isinstance(response_doc["statistics"], list) | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_collections_invalid_limit(client): | ||||||
|  |     """Test requesting 100 collections with an invalid limit parameter.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/collections", query_string="limit=101") | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_collections_invalid_page(client): | ||||||
|  |     """Test requesting 100 collections with an invalid page parameter.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/collections", query_string="page=-1") | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_post_collections_valid_dateFrom(client): | ||||||
|  |     """Test POSTing a request to /collections with a valid dateFrom parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateFrom": "2020-01-01T00:00:00Z", | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_collections_valid_dateFrom_mocked(client): | ||||||
|  |     """Mock test POSTing a request to /collections with a valid dateFrom parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateFrom": "2020-01-01T00:00:00Z", | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     get_views_return_value = { | ||||||
|  |         "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7": 21, | ||||||
|  |         "260548c8-fda4-4dc8-a979-03495753cdd5": 0, | ||||||
|  |     } | ||||||
|  |     get_downloads_return_value = { | ||||||
|  |         "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7": 575, | ||||||
|  |         "260548c8-fda4-4dc8-a979-03495753cdd5": 899, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     with patch( | ||||||
|  |         "dspace_statistics_api.app.get_views", return_value=get_views_return_value | ||||||
|  |     ): | ||||||
|  |         with patch( | ||||||
|  |             "dspace_statistics_api.app.get_downloads", | ||||||
|  |             return_value=get_downloads_return_value, | ||||||
|  |         ): | ||||||
|  |             response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_collections_invalid_dateFrom(client): | ||||||
|  |     """Test POSTing a request to /collections with an invalid dateFrom parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateFrom": "2020-01-01T00:00:00", | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_post_collections_valid_dateTo(client): | ||||||
|  |     """Test POSTing a request to /collections with a valid dateTo parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateTo": "2020-01-01T00:00:00Z", | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_collections_valid_dateTo_mocked(client): | ||||||
|  |     """Mock test POSTing a request to /collections with a valid dateTo parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateTo": "2020-01-01T00:00:00Z", | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     get_views_return_value = { | ||||||
|  |         "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7": 21, | ||||||
|  |         "260548c8-fda4-4dc8-a979-03495753cdd5": 0, | ||||||
|  |     } | ||||||
|  |     get_downloads_return_value = { | ||||||
|  |         "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7": 575, | ||||||
|  |         "260548c8-fda4-4dc8-a979-03495753cdd5": 899, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     with patch( | ||||||
|  |         "dspace_statistics_api.app.get_views", return_value=get_views_return_value | ||||||
|  |     ): | ||||||
|  |         with patch( | ||||||
|  |             "dspace_statistics_api.app.get_downloads", | ||||||
|  |             return_value=get_downloads_return_value, | ||||||
|  |         ): | ||||||
|  |             response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_collections_invalid_dateTo(client): | ||||||
|  |     """Test POSTing a request to /collections with an invalid dateTo parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateFrom": "2020-01-01T00:00:00", | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_post_collections_valid_limit(client): | ||||||
|  |     """Test POSTing a request to /collections with a valid limit parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "limit": 1, | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 1 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 1 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_collections_valid_limit_mocked(client): | ||||||
|  |     """Mock test POSTing a request to /collections with a valid limit parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "limit": 1, | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     get_views_return_value = {"8ea4b611-1f59-4d4e-b78d-a9921a72cfe7": 21} | ||||||
|  |     get_downloads_return_value = {"8ea4b611-1f59-4d4e-b78d-a9921a72cfe7": 575} | ||||||
|  |  | ||||||
|  |     with patch( | ||||||
|  |         "dspace_statistics_api.app.get_views", return_value=get_views_return_value | ||||||
|  |     ): | ||||||
|  |         with patch( | ||||||
|  |             "dspace_statistics_api.app.get_downloads", | ||||||
|  |             return_value=get_downloads_return_value, | ||||||
|  |         ): | ||||||
|  |             response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 1 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 1 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_collections_invalid_limit(client): | ||||||
|  |     """Test POSTing a request to /collections with an invalid limit parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "limit": -1, | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_post_collections_valid_page(client): | ||||||
|  |     """Test POSTing a request to /collections with a valid page parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "page": 0, | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert response.json["totalPages"] == 1 | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_collections_valid_page_mocked(client): | ||||||
|  |     """Mock test POSTing a request to /collections with a valid page parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "page": 0, | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     get_views_return_value = { | ||||||
|  |         "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7": 21, | ||||||
|  |         "260548c8-fda4-4dc8-a979-03495753cdd5": 0, | ||||||
|  |     } | ||||||
|  |     get_downloads_return_value = { | ||||||
|  |         "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7": 575, | ||||||
|  |         "260548c8-fda4-4dc8-a979-03495753cdd5": 899, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     with patch( | ||||||
|  |         "dspace_statistics_api.app.get_views", return_value=get_views_return_value | ||||||
|  |     ): | ||||||
|  |         with patch( | ||||||
|  |             "dspace_statistics_api.app.get_downloads", | ||||||
|  |             return_value=get_downloads_return_value, | ||||||
|  |         ): | ||||||
|  |             response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_collections_invalid_page(client): | ||||||
|  |     """Test POSTing a request to /collections with an invalid page parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "page": -1, | ||||||
|  |         "collections": [ | ||||||
|  |             "8ea4b611-1f59-4d4e-b78d-a9921a72cfe7", | ||||||
|  |             "260548c8-fda4-4dc8-a979-03495753cdd5", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/collections", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # vim: set sw=4 ts=4 expandtab: | ||||||
							
								
								
									
										379
									
								
								tests/test_api_communities.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										379
									
								
								tests/test_api_communities.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,379 @@ | |||||||
|  | # SPDX-License-Identifier: GPL-3.0-only | ||||||
|  |  | ||||||
|  | import json | ||||||
|  | from unittest.mock import patch | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  | from falcon import testing | ||||||
|  |  | ||||||
|  | from dspace_statistics_api.app import app | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture | ||||||
|  | def client(): | ||||||
|  |     return testing.TestClient(app) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_community(client): | ||||||
|  |     """Test requesting a single community.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/community/bde7139c-d321-46bb-aef6-ae70799e5edb") | ||||||
|  |     response_doc = json.loads(response.text) | ||||||
|  |  | ||||||
|  |     assert isinstance(response_doc["downloads"], int) | ||||||
|  |     assert isinstance(response_doc["id"], str) | ||||||
|  |     assert isinstance(response_doc["views"], int) | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_missing_community(client): | ||||||
|  |     """Test requesting a single non-existing community.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/item/dec6bfc6-efeb-4f74-8436-79fa80bb5c21") | ||||||
|  |  | ||||||
|  |     assert response.status_code == 404 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_communities(client): | ||||||
|  |     """Test requesting 100 communities.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/communities", query_string="limit=100") | ||||||
|  |     response_doc = json.loads(response.text) | ||||||
|  |  | ||||||
|  |     assert isinstance(response_doc["currentPage"], int) | ||||||
|  |     assert isinstance(response_doc["totalPages"], int) | ||||||
|  |     assert isinstance(response_doc["statistics"], list) | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_communities_invalid_limit(client): | ||||||
|  |     """Test requesting 100 communities with an invalid limit parameter.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/communities", query_string="limit=101") | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_communities_invalid_page(client): | ||||||
|  |     """Test requesting 100 communities with an invalid page parameter.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/communities", query_string="page=-1") | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_post_communities_valid_dateFrom(client): | ||||||
|  |     """Test POSTing a request to /communities with a valid dateFrom parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateFrom": "2020-01-01T00:00:00Z", | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_communities_valid_dateFrom_mocked(client): | ||||||
|  |     """Mock test POSTing a request to /communities with a valid dateFrom parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateFrom": "2020-01-01T00:00:00Z", | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     get_views_return_value = { | ||||||
|  |         "bde7139c-d321-46bb-aef6-ae70799e5edb": 309, | ||||||
|  |         "2a920a61-b08a-4642-8e5d-2639c6702b1f": 0, | ||||||
|  |     } | ||||||
|  |     get_downloads_return_value = { | ||||||
|  |         "bde7139c-d321-46bb-aef6-ae70799e5edb": 400, | ||||||
|  |         "2a920a61-b08a-4642-8e5d-2639c6702b1f": 290, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     with patch( | ||||||
|  |         "dspace_statistics_api.app.get_views", return_value=get_views_return_value | ||||||
|  |     ): | ||||||
|  |         with patch( | ||||||
|  |             "dspace_statistics_api.app.get_downloads", | ||||||
|  |             return_value=get_downloads_return_value, | ||||||
|  |         ): | ||||||
|  |             response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_communities_invalid_dateFrom(client): | ||||||
|  |     """Test POSTing a request to /communities with an invalid dateFrom parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateFrom": "2020-01-01T00:00:00", | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_post_communities_valid_dateTo(client): | ||||||
|  |     """Test POSTing a request to /communities with a valid dateTo parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateTo": "2020-01-01T00:00:00Z", | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_communities_valid_dateTo_mocked(client): | ||||||
|  |     """Mock test POSTing a request to /communities with a valid dateTo parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateTo": "2020-01-01T00:00:00Z", | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     get_views_return_value = { | ||||||
|  |         "bde7139c-d321-46bb-aef6-ae70799e5edb": 21, | ||||||
|  |         "2a920a61-b08a-4642-8e5d-2639c6702b1f": 0, | ||||||
|  |     } | ||||||
|  |     get_downloads_return_value = { | ||||||
|  |         "bde7139c-d321-46bb-aef6-ae70799e5edb": 575, | ||||||
|  |         "2a920a61-b08a-4642-8e5d-2639c6702b1f": 899, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     with patch( | ||||||
|  |         "dspace_statistics_api.app.get_views", return_value=get_views_return_value | ||||||
|  |     ): | ||||||
|  |         with patch( | ||||||
|  |             "dspace_statistics_api.app.get_downloads", | ||||||
|  |             return_value=get_downloads_return_value, | ||||||
|  |         ): | ||||||
|  |             response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_communities_invalid_dateTo(client): | ||||||
|  |     """Test POSTing a request to /communities with an invalid dateTo parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "dateFrom": "2020-01-01T00:00:00", | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_post_communities_valid_limit(client): | ||||||
|  |     """Test POSTing a request to /communities with a valid limit parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "limit": 1, | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 1 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 1 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_communities_valid_limit_mocked(client): | ||||||
|  |     """Mock test POSTing a request to /communities with a valid limit parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "limit": 1, | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     get_views_return_value = {"bde7139c-d321-46bb-aef6-ae70799e5edb": 200} | ||||||
|  |     get_downloads_return_value = {"bde7139c-d321-46bb-aef6-ae70799e5edb": 309} | ||||||
|  |  | ||||||
|  |     with patch( | ||||||
|  |         "dspace_statistics_api.app.get_views", return_value=get_views_return_value | ||||||
|  |     ): | ||||||
|  |         with patch( | ||||||
|  |             "dspace_statistics_api.app.get_downloads", | ||||||
|  |             return_value=get_downloads_return_value, | ||||||
|  |         ): | ||||||
|  |             response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 1 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 1 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_communities_invalid_limit(client): | ||||||
|  |     """Test POSTing a request to /communities with an invalid limit parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "limit": -1, | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_post_communities_valid_page(client): | ||||||
|  |     """Test POSTing a request to /communities with a valid page parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "page": 0, | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert response.json["totalPages"] == 1 | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_communities_valid_page_mocked(client): | ||||||
|  |     """Mock test POSTing a request to communities with a valid page parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "page": 0, | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     get_views_return_value = { | ||||||
|  |         "bde7139c-d321-46bb-aef6-ae70799e5edb": 21, | ||||||
|  |         "2a920a61-b08a-4642-8e5d-2639c6702b1f": 0, | ||||||
|  |     } | ||||||
|  |     get_downloads_return_value = { | ||||||
|  |         "bde7139c-d321-46bb-aef6-ae70799e5edb": 575, | ||||||
|  |         "2a920a61-b08a-4642-8e5d-2639c6702b1f": 899, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     with patch( | ||||||
|  |         "dspace_statistics_api.app.get_views", return_value=get_views_return_value | ||||||
|  |     ): | ||||||
|  |         with patch( | ||||||
|  |             "dspace_statistics_api.app.get_downloads", | ||||||
|  |             return_value=get_downloads_return_value, | ||||||
|  |         ): | ||||||
|  |             response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |     assert response.json["limit"] == 100 | ||||||
|  |     assert response.json["currentPage"] == 0 | ||||||
|  |     assert isinstance(response.json["totalPages"], int) | ||||||
|  |     assert len(response.json["statistics"]) == 2 | ||||||
|  |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["views"], int) | ||||||
|  |     assert isinstance(response.json["statistics"][1]["downloads"], int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_post_communities_invalid_page(client): | ||||||
|  |     """Test POSTing a request to /communities with an invalid page parameter in the request body.""" | ||||||
|  |  | ||||||
|  |     request_body = { | ||||||
|  |         "page": -1, | ||||||
|  |         "communities": [ | ||||||
|  |             "bde7139c-d321-46bb-aef6-ae70799e5edb", | ||||||
|  |             "2a920a61-b08a-4642-8e5d-2639c6702b1f", | ||||||
|  |         ], | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     response = client.simulate_post("/communities", json=request_body) | ||||||
|  |  | ||||||
|  |     assert response.status_code == 400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # vim: set sw=4 ts=4 expandtab: | ||||||
							
								
								
									
										50
									
								
								tests/test_api_docs.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								tests/test_api_docs.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | |||||||
|  | # SPDX-License-Identifier: GPL-3.0-only | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  | from falcon import testing | ||||||
|  |  | ||||||
|  | from dspace_statistics_api.app import app | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture | ||||||
|  | def client(): | ||||||
|  |     return testing.TestClient(app) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_docs(client): | ||||||
|  |     """Test requesting the documentation at the root.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/") | ||||||
|  |  | ||||||
|  |     assert isinstance(response.content, bytes) | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_openapi_json(client): | ||||||
|  |     """Test requesting the OpenAPI JSON schema.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/docs/openapi.json") | ||||||
|  |  | ||||||
|  |     assert isinstance(response.content, bytes) | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_swagger_ui(client): | ||||||
|  |     """Test requesting the Swagger UI.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/swagger") | ||||||
|  |  | ||||||
|  |     assert isinstance(response.content, bytes) | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_get_status(client): | ||||||
|  |     """Test requesting the status page.""" | ||||||
|  |  | ||||||
|  |     response = client.simulate_get("/status") | ||||||
|  |  | ||||||
|  |     assert isinstance(response.content, bytes) | ||||||
|  |     assert response.status_code == 200 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # vim: set sw=4 ts=4 expandtab: | ||||||
| @@ -1,29 +1,23 @@ | |||||||
| from falcon import testing | # SPDX-License-Identifier: GPL-3.0-only | ||||||
|  | 
 | ||||||
| import json | import json | ||||||
| import pytest |  | ||||||
| from unittest.mock import patch | from unittest.mock import patch | ||||||
| 
 | 
 | ||||||
| from dspace_statistics_api.app import api | import pytest | ||||||
|  | from falcon import testing | ||||||
|  | 
 | ||||||
|  | from dspace_statistics_api.app import app | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @pytest.fixture | @pytest.fixture | ||||||
| def client(): | def client(): | ||||||
|     return testing.TestClient(api) |     return testing.TestClient(app) | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def test_get_docs(client): |  | ||||||
|     """Test requesting the documentation at the root.""" |  | ||||||
| 
 |  | ||||||
|     response = client.simulate_get("/") |  | ||||||
| 
 |  | ||||||
|     assert isinstance(response.content, bytes) |  | ||||||
|     assert response.status_code == 200 |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def test_get_item(client): | def test_get_item(client): | ||||||
|     """Test requesting a single item.""" |     """Test requesting a single item.""" | ||||||
| 
 | 
 | ||||||
|     response = client.simulate_get("/item/c3910974-c3a5-4053-9dce-104aa7bb1621") |     response = client.simulate_get("/item/fd8a46d5-1480-4e69-b187-cd3db96d8e4d") | ||||||
|     response_doc = json.loads(response.text) |     response_doc = json.loads(response.text) | ||||||
| 
 | 
 | ||||||
|     assert isinstance(response_doc["downloads"], int) |     assert isinstance(response_doc["downloads"], int) | ||||||
| @@ -70,13 +64,13 @@ def test_get_items_invalid_page(client): | |||||||
| 
 | 
 | ||||||
| @pytest.mark.xfail | @pytest.mark.xfail | ||||||
| def test_post_items_valid_dateFrom(client): | def test_post_items_valid_dateFrom(client): | ||||||
|     """Test POSTing a request with a valid dateFrom parameter in the request body.""" |     """Test POSTing a request to /items with a valid dateFrom parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "dateFrom": "2020-01-01T00:00:00Z", |         "dateFrom": "2020-01-01T00:00:00Z", | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -94,23 +88,23 @@ def test_post_items_valid_dateFrom(client): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def test_post_items_valid_dateFrom_mocked(client): | def test_post_items_valid_dateFrom_mocked(client): | ||||||
|     """Mock test POSTing a request with a valid dateFrom parameter in the request body.""" |     """Mock test POSTing a request to /items with a valid dateFrom parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "dateFrom": "2020-01-01T00:00:00Z", |         "dateFrom": "2020-01-01T00:00:00Z", | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     get_views_return_value = { |     get_views_return_value = { | ||||||
|         "c3910974-c3a5-4053-9dce-104aa7bb1620": 21, |         "fd8a46d5-1480-4e69-b187-cd3db96d8e4d": 21, | ||||||
|         "887cc5f8-b5e7-4a2f-9053-49c91ab81313": 0, |         "e53a2eab-1e31-448d-907b-3656ca4e86c1": 0, | ||||||
|     } |     } | ||||||
|     get_downloads_return_value = { |     get_downloads_return_value = { | ||||||
|         "c3910974-c3a5-4053-9dce-104aa7bb1620": 575, |         "fd8a46d5-1480-4e69-b187-cd3db96d8e4d": 575, | ||||||
|         "887cc5f8-b5e7-4a2f-9053-49c91ab81313": 899, |         "e53a2eab-1e31-448d-907b-3656ca4e86c1": 899, | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     with patch( |     with patch( | ||||||
| @@ -134,13 +128,13 @@ def test_post_items_valid_dateFrom_mocked(client): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def test_post_items_invalid_dateFrom(client): | def test_post_items_invalid_dateFrom(client): | ||||||
|     """Test POSTing a request with an invalid dateFrom parameter in the request body.""" |     """Test POSTing a request to /items with an invalid dateFrom parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "dateFrom": "2020-01-01T00:00:00", |         "dateFrom": "2020-01-01T00:00:00", | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -151,13 +145,13 @@ def test_post_items_invalid_dateFrom(client): | |||||||
| 
 | 
 | ||||||
| @pytest.mark.xfail | @pytest.mark.xfail | ||||||
| def test_post_items_valid_dateTo(client): | def test_post_items_valid_dateTo(client): | ||||||
|     """Test POSTing a request with a valid dateTo parameter in the request body.""" |     """Test POSTing a request to /items with a valid dateTo parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "dateTo": "2020-01-01T00:00:00Z", |         "dateTo": "2020-01-01T00:00:00Z", | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -175,23 +169,23 @@ def test_post_items_valid_dateTo(client): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def test_post_items_valid_dateTo_mocked(client): | def test_post_items_valid_dateTo_mocked(client): | ||||||
|     """Mock test POSTing a request with a valid dateTo parameter in the request body.""" |     """Mock test POSTing a request to /items with a valid dateTo parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "dateTo": "2020-01-01T00:00:00Z", |         "dateTo": "2020-01-01T00:00:00Z", | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     get_views_return_value = { |     get_views_return_value = { | ||||||
|         "c3910974-c3a5-4053-9dce-104aa7bb1620": 21, |         "fd8a46d5-1480-4e69-b187-cd3db96d8e4d": 21, | ||||||
|         "887cc5f8-b5e7-4a2f-9053-49c91ab81313": 0, |         "e53a2eab-1e31-448d-907b-3656ca4e86c1": 0, | ||||||
|     } |     } | ||||||
|     get_downloads_return_value = { |     get_downloads_return_value = { | ||||||
|         "c3910974-c3a5-4053-9dce-104aa7bb1620": 575, |         "fd8a46d5-1480-4e69-b187-cd3db96d8e4d": 575, | ||||||
|         "887cc5f8-b5e7-4a2f-9053-49c91ab81313": 899, |         "e53a2eab-1e31-448d-907b-3656ca4e86c1": 899, | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     with patch( |     with patch( | ||||||
| @@ -215,13 +209,13 @@ def test_post_items_valid_dateTo_mocked(client): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def test_post_items_invalid_dateTo(client): | def test_post_items_invalid_dateTo(client): | ||||||
|     """Test POSTing a request with an invalid dateTo parameter in the request body.""" |     """Test POSTing a request to /items with an invalid dateTo parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "dateFrom": "2020-01-01T00:00:00", |         "dateFrom": "2020-01-01T00:00:00", | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -232,13 +226,13 @@ def test_post_items_invalid_dateTo(client): | |||||||
| 
 | 
 | ||||||
| @pytest.mark.xfail | @pytest.mark.xfail | ||||||
| def test_post_items_valid_limit(client): | def test_post_items_valid_limit(client): | ||||||
|     """Test POSTing a request with a valid limit parameter in the request body.""" |     """Test POSTing a request to /items with a valid limit parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "limit": 1, |         "limit": 1, | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -254,18 +248,18 @@ def test_post_items_valid_limit(client): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def test_post_items_valid_limit_mocked(client): | def test_post_items_valid_limit_mocked(client): | ||||||
|     """Mock test POSTing a request with a valid limit parameter in the request body.""" |     """Mock test POSTing a request to /items with a valid limit parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "limit": 1, |         "limit": 1, | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     get_views_return_value = {"c3910974-c3a5-4053-9dce-104aa7bb1620": 21} |     get_views_return_value = {"fd8a46d5-1480-4e69-b187-cd3db96d8e4d": 21} | ||||||
|     get_downloads_return_value = {"c3910974-c3a5-4053-9dce-104aa7bb1620": 575} |     get_downloads_return_value = {"fd8a46d5-1480-4e69-b187-cd3db96d8e4d": 575} | ||||||
| 
 | 
 | ||||||
|     with patch( |     with patch( | ||||||
|         "dspace_statistics_api.app.get_views", return_value=get_views_return_value |         "dspace_statistics_api.app.get_views", return_value=get_views_return_value | ||||||
| @@ -286,13 +280,13 @@ def test_post_items_valid_limit_mocked(client): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def test_post_items_invalid_limit(client): | def test_post_items_invalid_limit(client): | ||||||
|     """Test POSTing a request with an invalid limit parameter in the request body.""" |     """Test POSTing a request to /items with an invalid limit parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "limit": -1, |         "limit": -1, | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -303,13 +297,13 @@ def test_post_items_invalid_limit(client): | |||||||
| 
 | 
 | ||||||
| @pytest.mark.xfail | @pytest.mark.xfail | ||||||
| def test_post_items_valid_page(client): | def test_post_items_valid_page(client): | ||||||
|     """Test POSTing a request with a valid page parameter in the request body.""" |     """Test POSTing a request to /items with a valid page parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "page": 0, |         "page": 0, | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -318,7 +312,7 @@ def test_post_items_valid_page(client): | |||||||
|     assert response.status_code == 200 |     assert response.status_code == 200 | ||||||
|     assert response.json["limit"] == 100 |     assert response.json["limit"] == 100 | ||||||
|     assert response.json["currentPage"] == 0 |     assert response.json["currentPage"] == 0 | ||||||
|     assert response.json["totalPages"] == 0 |     assert response.json["totalPages"] == 1 | ||||||
|     assert len(response.json["statistics"]) == 2 |     assert len(response.json["statistics"]) == 2 | ||||||
|     assert isinstance(response.json["statistics"][0]["views"], int) |     assert isinstance(response.json["statistics"][0]["views"], int) | ||||||
|     assert isinstance(response.json["statistics"][0]["downloads"], int) |     assert isinstance(response.json["statistics"][0]["downloads"], int) | ||||||
| @@ -327,23 +321,23 @@ def test_post_items_valid_page(client): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def test_post_items_valid_page_mocked(client): | def test_post_items_valid_page_mocked(client): | ||||||
|     """Mock test POSTing a request with a valid page parameter in the request body.""" |     """Mock test POSTing a request to /items with a valid page parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "page": 0, |         "page": 0, | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     get_views_return_value = { |     get_views_return_value = { | ||||||
|         "c3910974-c3a5-4053-9dce-104aa7bb1620": 21, |         "fd8a46d5-1480-4e69-b187-cd3db96d8e4d": 21, | ||||||
|         "887cc5f8-b5e7-4a2f-9053-49c91ab81313": 0, |         "e53a2eab-1e31-448d-907b-3656ca4e86c1": 0, | ||||||
|     } |     } | ||||||
|     get_downloads_return_value = { |     get_downloads_return_value = { | ||||||
|         "c3910974-c3a5-4053-9dce-104aa7bb1620": 575, |         "fd8a46d5-1480-4e69-b187-cd3db96d8e4d": 575, | ||||||
|         "887cc5f8-b5e7-4a2f-9053-49c91ab81313": 899, |         "e53a2eab-1e31-448d-907b-3656ca4e86c1": 899, | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     with patch( |     with patch( | ||||||
| @@ -367,16 +361,19 @@ def test_post_items_valid_page_mocked(client): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def test_post_items_invalid_page(client): | def test_post_items_invalid_page(client): | ||||||
|     """Test POSTing a request with an invalid page parameter in the request body.""" |     """Test POSTing a request to /items with an invalid page parameter in the request body.""" | ||||||
| 
 | 
 | ||||||
|     request_body = { |     request_body = { | ||||||
|         "page": -1, |         "page": -1, | ||||||
|         "items": [ |         "items": [ | ||||||
|             "c3910974-c3a5-4053-9dce-104aa7bb1620", |             "fd8a46d5-1480-4e69-b187-cd3db96d8e4d", | ||||||
|             "887cc5f8-b5e7-4a2f-9053-49c91ab81313", |             "e53a2eab-1e31-448d-907b-3656ca4e86c1", | ||||||
|         ], |         ], | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     response = client.simulate_post("/items", json=request_body) |     response = client.simulate_post("/items", json=request_body) | ||||||
| 
 | 
 | ||||||
|     assert response.status_code == 400 |     assert response.status_code == 400 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # vim: set sw=4 ts=4 expandtab: | ||||||
							
								
								
									
										489
									
								
								uv.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										489
									
								
								uv.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,489 @@ | |||||||
|  | version = 1 | ||||||
|  | revision = 2 | ||||||
|  | requires-python = ">=3.9" | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "certifi" | ||||||
|  | version = "2025.6.15" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/73/f7/f14b46d4bcd21092d7d3ccef689615220d8a08fb25e564b65d20738e672e/certifi-2025.6.15.tar.gz", hash = "sha256:d747aa5a8b9bbbb1bb8c22bb13e22bd1f18e9796defa16bab421f7f7a317323b", size = 158753, upload-time = "2025-06-15T02:45:51.329Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/84/ae/320161bd181fc06471eed047ecce67b693fd7515b16d495d8932db763426/certifi-2025.6.15-py3-none-any.whl", hash = "sha256:2e0c7ce7cb5d8f8634ca55d2ba7e6ec2689a2fd6537d8dec1296a477a4910057", size = 157650, upload-time = "2025-06-15T02:45:49.977Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "charset-normalizer" | ||||||
|  | version = "3.4.2" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/e4/33/89c2ced2b67d1c2a61c19c6751aa8902d46ce3dacb23600a283619f5a12d/charset_normalizer-3.4.2.tar.gz", hash = "sha256:5baececa9ecba31eff645232d59845c07aa030f0c81ee70184a90d35099a0e63", size = 126367, upload-time = "2025-05-02T08:34:42.01Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/95/28/9901804da60055b406e1a1c5ba7aac1276fb77f1dde635aabfc7fd84b8ab/charset_normalizer-3.4.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:7c48ed483eb946e6c04ccbe02c6b4d1d48e51944b6db70f697e089c193404941", size = 201818, upload-time = "2025-05-02T08:31:46.725Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/d9/9b/892a8c8af9110935e5adcbb06d9c6fe741b6bb02608c6513983048ba1a18/charset_normalizer-3.4.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b2d318c11350e10662026ad0eb71bb51c7812fc8590825304ae0bdd4ac283acd", size = 144649, upload-time = "2025-05-02T08:31:48.889Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/7b/a5/4179abd063ff6414223575e008593861d62abfc22455b5d1a44995b7c101/charset_normalizer-3.4.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9cbfacf36cb0ec2897ce0ebc5d08ca44213af24265bd56eca54bee7923c48fd6", size = 155045, upload-time = "2025-05-02T08:31:50.757Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/3b/95/bc08c7dfeddd26b4be8c8287b9bb055716f31077c8b0ea1cd09553794665/charset_normalizer-3.4.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:18dd2e350387c87dabe711b86f83c9c78af772c748904d372ade190b5c7c9d4d", size = 147356, upload-time = "2025-05-02T08:31:52.634Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a8/2d/7a5b635aa65284bf3eab7653e8b4151ab420ecbae918d3e359d1947b4d61/charset_normalizer-3.4.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8075c35cd58273fee266c58c0c9b670947c19df5fb98e7b66710e04ad4e9ff86", size = 149471, upload-time = "2025-05-02T08:31:56.207Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/ae/38/51fc6ac74251fd331a8cfdb7ec57beba8c23fd5493f1050f71c87ef77ed0/charset_normalizer-3.4.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5bf4545e3b962767e5c06fe1738f951f77d27967cb2caa64c28be7c4563e162c", size = 151317, upload-time = "2025-05-02T08:31:57.613Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/b7/17/edee1e32215ee6e9e46c3e482645b46575a44a2d72c7dfd49e49f60ce6bf/charset_normalizer-3.4.2-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:7a6ab32f7210554a96cd9e33abe3ddd86732beeafc7a28e9955cdf22ffadbab0", size = 146368, upload-time = "2025-05-02T08:31:59.468Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/26/2c/ea3e66f2b5f21fd00b2825c94cafb8c326ea6240cd80a91eb09e4a285830/charset_normalizer-3.4.2-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:b33de11b92e9f75a2b545d6e9b6f37e398d86c3e9e9653c4864eb7e89c5773ef", size = 154491, upload-time = "2025-05-02T08:32:01.219Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/52/47/7be7fa972422ad062e909fd62460d45c3ef4c141805b7078dbab15904ff7/charset_normalizer-3.4.2-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:8755483f3c00d6c9a77f490c17e6ab0c8729e39e6390328e42521ef175380ae6", size = 157695, upload-time = "2025-05-02T08:32:03.045Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2f/42/9f02c194da282b2b340f28e5fb60762de1151387a36842a92b533685c61e/charset_normalizer-3.4.2-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:68a328e5f55ec37c57f19ebb1fdc56a248db2e3e9ad769919a58672958e8f366", size = 154849, upload-time = "2025-05-02T08:32:04.651Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/67/44/89cacd6628f31fb0b63201a618049be4be2a7435a31b55b5eb1c3674547a/charset_normalizer-3.4.2-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:21b2899062867b0e1fde9b724f8aecb1af14f2778d69aacd1a5a1853a597a5db", size = 150091, upload-time = "2025-05-02T08:32:06.719Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/1f/79/4b8da9f712bc079c0f16b6d67b099b0b8d808c2292c937f267d816ec5ecc/charset_normalizer-3.4.2-cp310-cp310-win32.whl", hash = "sha256:e8082b26888e2f8b36a042a58307d5b917ef2b1cacab921ad3323ef91901c71a", size = 98445, upload-time = "2025-05-02T08:32:08.66Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/7d/d7/96970afb4fb66497a40761cdf7bd4f6fca0fc7bafde3a84f836c1f57a926/charset_normalizer-3.4.2-cp310-cp310-win_amd64.whl", hash = "sha256:f69a27e45c43520f5487f27627059b64aaf160415589230992cec34c5e18a509", size = 105782, upload-time = "2025-05-02T08:32:10.46Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/05/85/4c40d00dcc6284a1c1ad5de5e0996b06f39d8232f1031cd23c2f5c07ee86/charset_normalizer-3.4.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:be1e352acbe3c78727a16a455126d9ff83ea2dfdcbc83148d2982305a04714c2", size = 198794, upload-time = "2025-05-02T08:32:11.945Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/41/d9/7a6c0b9db952598e97e93cbdfcb91bacd89b9b88c7c983250a77c008703c/charset_normalizer-3.4.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:aa88ca0b1932e93f2d961bf3addbb2db902198dca337d88c89e1559e066e7645", size = 142846, upload-time = "2025-05-02T08:32:13.946Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/66/82/a37989cda2ace7e37f36c1a8ed16c58cf48965a79c2142713244bf945c89/charset_normalizer-3.4.2-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d524ba3f1581b35c03cb42beebab4a13e6cdad7b36246bd22541fa585a56cccd", size = 153350, upload-time = "2025-05-02T08:32:15.873Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/df/68/a576b31b694d07b53807269d05ec3f6f1093e9545e8607121995ba7a8313/charset_normalizer-3.4.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:28a1005facc94196e1fb3e82a3d442a9d9110b8434fc1ded7a24a2983c9888d8", size = 145657, upload-time = "2025-05-02T08:32:17.283Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/92/9b/ad67f03d74554bed3aefd56fe836e1623a50780f7c998d00ca128924a499/charset_normalizer-3.4.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fdb20a30fe1175ecabed17cbf7812f7b804b8a315a25f24678bcdf120a90077f", size = 147260, upload-time = "2025-05-02T08:32:18.807Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a6/e6/8aebae25e328160b20e31a7e9929b1578bbdc7f42e66f46595a432f8539e/charset_normalizer-3.4.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0f5d9ed7f254402c9e7d35d2f5972c9bbea9040e99cd2861bd77dc68263277c7", size = 149164, upload-time = "2025-05-02T08:32:20.333Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/8b/f2/b3c2f07dbcc248805f10e67a0262c93308cfa149a4cd3d1fe01f593e5fd2/charset_normalizer-3.4.2-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:efd387a49825780ff861998cd959767800d54f8308936b21025326de4b5a42b9", size = 144571, upload-time = "2025-05-02T08:32:21.86Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/60/5b/c3f3a94bc345bc211622ea59b4bed9ae63c00920e2e8f11824aa5708e8b7/charset_normalizer-3.4.2-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:f0aa37f3c979cf2546b73e8222bbfa3dc07a641585340179d768068e3455e544", size = 151952, upload-time = "2025-05-02T08:32:23.434Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/e2/4d/ff460c8b474122334c2fa394a3f99a04cf11c646da895f81402ae54f5c42/charset_normalizer-3.4.2-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:e70e990b2137b29dc5564715de1e12701815dacc1d056308e2b17e9095372a82", size = 155959, upload-time = "2025-05-02T08:32:24.993Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a2/2b/b964c6a2fda88611a1fe3d4c400d39c66a42d6c169c924818c848f922415/charset_normalizer-3.4.2-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:0c8c57f84ccfc871a48a47321cfa49ae1df56cd1d965a09abe84066f6853b9c0", size = 153030, upload-time = "2025-05-02T08:32:26.435Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/59/2e/d3b9811db26a5ebf444bc0fa4f4be5aa6d76fc6e1c0fd537b16c14e849b6/charset_normalizer-3.4.2-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:6b66f92b17849b85cad91259efc341dce9c1af48e2173bf38a85c6329f1033e5", size = 148015, upload-time = "2025-05-02T08:32:28.376Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/90/07/c5fd7c11eafd561bb51220d600a788f1c8d77c5eef37ee49454cc5c35575/charset_normalizer-3.4.2-cp311-cp311-win32.whl", hash = "sha256:daac4765328a919a805fa5e2720f3e94767abd632ae410a9062dff5412bae65a", size = 98106, upload-time = "2025-05-02T08:32:30.281Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a8/05/5e33dbef7e2f773d672b6d79f10ec633d4a71cd96db6673625838a4fd532/charset_normalizer-3.4.2-cp311-cp311-win_amd64.whl", hash = "sha256:e53efc7c7cee4c1e70661e2e112ca46a575f90ed9ae3fef200f2a25e954f4b28", size = 105402, upload-time = "2025-05-02T08:32:32.191Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/d7/a4/37f4d6035c89cac7930395a35cc0f1b872e652eaafb76a6075943754f095/charset_normalizer-3.4.2-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:0c29de6a1a95f24b9a1aa7aefd27d2487263f00dfd55a77719b530788f75cff7", size = 199936, upload-time = "2025-05-02T08:32:33.712Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/ee/8a/1a5e33b73e0d9287274f899d967907cd0bf9c343e651755d9307e0dbf2b3/charset_normalizer-3.4.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cddf7bd982eaa998934a91f69d182aec997c6c468898efe6679af88283b498d3", size = 143790, upload-time = "2025-05-02T08:32:35.768Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/66/52/59521f1d8e6ab1482164fa21409c5ef44da3e9f653c13ba71becdd98dec3/charset_normalizer-3.4.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fcbe676a55d7445b22c10967bceaaf0ee69407fbe0ece4d032b6eb8d4565982a", size = 153924, upload-time = "2025-05-02T08:32:37.284Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/86/2d/fb55fdf41964ec782febbf33cb64be480a6b8f16ded2dbe8db27a405c09f/charset_normalizer-3.4.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d41c4d287cfc69060fa91cae9683eacffad989f1a10811995fa309df656ec214", size = 146626, upload-time = "2025-05-02T08:32:38.803Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/8c/73/6ede2ec59bce19b3edf4209d70004253ec5f4e319f9a2e3f2f15601ed5f7/charset_normalizer-3.4.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4e594135de17ab3866138f496755f302b72157d115086d100c3f19370839dd3a", size = 148567, upload-time = "2025-05-02T08:32:40.251Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/09/14/957d03c6dc343c04904530b6bef4e5efae5ec7d7990a7cbb868e4595ee30/charset_normalizer-3.4.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:cf713fe9a71ef6fd5adf7a79670135081cd4431c2943864757f0fa3a65b1fafd", size = 150957, upload-time = "2025-05-02T08:32:41.705Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/0d/c8/8174d0e5c10ccebdcb1b53cc959591c4c722a3ad92461a273e86b9f5a302/charset_normalizer-3.4.2-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:a370b3e078e418187da8c3674eddb9d983ec09445c99a3a263c2011993522981", size = 145408, upload-time = "2025-05-02T08:32:43.709Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/58/aa/8904b84bc8084ac19dc52feb4f5952c6df03ffb460a887b42615ee1382e8/charset_normalizer-3.4.2-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:a955b438e62efdf7e0b7b52a64dc5c3396e2634baa62471768a64bc2adb73d5c", size = 153399, upload-time = "2025-05-02T08:32:46.197Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c2/26/89ee1f0e264d201cb65cf054aca6038c03b1a0c6b4ae998070392a3ce605/charset_normalizer-3.4.2-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:7222ffd5e4de8e57e03ce2cef95a4c43c98fcb72ad86909abdfc2c17d227fc1b", size = 156815, upload-time = "2025-05-02T08:32:48.105Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/fd/07/68e95b4b345bad3dbbd3a8681737b4338ff2c9df29856a6d6d23ac4c73cb/charset_normalizer-3.4.2-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:bee093bf902e1d8fc0ac143c88902c3dfc8941f7ea1d6a8dd2bcb786d33db03d", size = 154537, upload-time = "2025-05-02T08:32:49.719Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/77/1a/5eefc0ce04affb98af07bc05f3bac9094513c0e23b0562d64af46a06aae4/charset_normalizer-3.4.2-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:dedb8adb91d11846ee08bec4c8236c8549ac721c245678282dcb06b221aab59f", size = 149565, upload-time = "2025-05-02T08:32:51.404Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/37/a0/2410e5e6032a174c95e0806b1a6585eb21e12f445ebe239fac441995226a/charset_normalizer-3.4.2-cp312-cp312-win32.whl", hash = "sha256:db4c7bf0e07fc3b7d89ac2a5880a6a8062056801b83ff56d8464b70f65482b6c", size = 98357, upload-time = "2025-05-02T08:32:53.079Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/6c/4f/c02d5c493967af3eda9c771ad4d2bbc8df6f99ddbeb37ceea6e8716a32bc/charset_normalizer-3.4.2-cp312-cp312-win_amd64.whl", hash = "sha256:5a9979887252a82fefd3d3ed2a8e3b937a7a809f65dcb1e068b090e165bbe99e", size = 105776, upload-time = "2025-05-02T08:32:54.573Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/ea/12/a93df3366ed32db1d907d7593a94f1fe6293903e3e92967bebd6950ed12c/charset_normalizer-3.4.2-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:926ca93accd5d36ccdabd803392ddc3e03e6d4cd1cf17deff3b989ab8e9dbcf0", size = 199622, upload-time = "2025-05-02T08:32:56.363Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/04/93/bf204e6f344c39d9937d3c13c8cd5bbfc266472e51fc8c07cb7f64fcd2de/charset_normalizer-3.4.2-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:eba9904b0f38a143592d9fc0e19e2df0fa2e41c3c3745554761c5f6447eedabf", size = 143435, upload-time = "2025-05-02T08:32:58.551Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/22/2a/ea8a2095b0bafa6c5b5a55ffdc2f924455233ee7b91c69b7edfcc9e02284/charset_normalizer-3.4.2-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3fddb7e2c84ac87ac3a947cb4e66d143ca5863ef48e4a5ecb83bd48619e4634e", size = 153653, upload-time = "2025-05-02T08:33:00.342Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/b6/57/1b090ff183d13cef485dfbe272e2fe57622a76694061353c59da52c9a659/charset_normalizer-3.4.2-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:98f862da73774290f251b9df8d11161b6cf25b599a66baf087c1ffe340e9bfd1", size = 146231, upload-time = "2025-05-02T08:33:02.081Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/e2/28/ffc026b26f441fc67bd21ab7f03b313ab3fe46714a14b516f931abe1a2d8/charset_normalizer-3.4.2-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c9379d65defcab82d07b2a9dfbfc2e95bc8fe0ebb1b176a3190230a3ef0e07c", size = 148243, upload-time = "2025-05-02T08:33:04.063Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c0/0f/9abe9bd191629c33e69e47c6ef45ef99773320e9ad8e9cb08b8ab4a8d4cb/charset_normalizer-3.4.2-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e635b87f01ebc977342e2697d05b56632f5f879a4f15955dfe8cef2448b51691", size = 150442, upload-time = "2025-05-02T08:33:06.418Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/67/7c/a123bbcedca91d5916c056407f89a7f5e8fdfce12ba825d7d6b9954a1a3c/charset_normalizer-3.4.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:1c95a1e2902a8b722868587c0e1184ad5c55631de5afc0eb96bc4b0d738092c0", size = 145147, upload-time = "2025-05-02T08:33:08.183Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/ec/fe/1ac556fa4899d967b83e9893788e86b6af4d83e4726511eaaad035e36595/charset_normalizer-3.4.2-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:ef8de666d6179b009dce7bcb2ad4c4a779f113f12caf8dc77f0162c29d20490b", size = 153057, upload-time = "2025-05-02T08:33:09.986Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2b/ff/acfc0b0a70b19e3e54febdd5301a98b72fa07635e56f24f60502e954c461/charset_normalizer-3.4.2-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:32fc0341d72e0f73f80acb0a2c94216bd704f4f0bce10aedea38f30502b271ff", size = 156454, upload-time = "2025-05-02T08:33:11.814Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/92/08/95b458ce9c740d0645feb0e96cea1f5ec946ea9c580a94adfe0b617f3573/charset_normalizer-3.4.2-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:289200a18fa698949d2b39c671c2cc7a24d44096784e76614899a7ccf2574b7b", size = 154174, upload-time = "2025-05-02T08:33:13.707Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/78/be/8392efc43487ac051eee6c36d5fbd63032d78f7728cb37aebcc98191f1ff/charset_normalizer-3.4.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:4a476b06fbcf359ad25d34a057b7219281286ae2477cc5ff5e3f70a246971148", size = 149166, upload-time = "2025-05-02T08:33:15.458Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/44/96/392abd49b094d30b91d9fbda6a69519e95802250b777841cf3bda8fe136c/charset_normalizer-3.4.2-cp313-cp313-win32.whl", hash = "sha256:aaeeb6a479c7667fbe1099af9617c83aaca22182d6cf8c53966491a0f1b7ffb7", size = 98064, upload-time = "2025-05-02T08:33:17.06Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/e9/b0/0200da600134e001d91851ddc797809e2fe0ea72de90e09bec5a2fbdaccb/charset_normalizer-3.4.2-cp313-cp313-win_amd64.whl", hash = "sha256:aa6af9e7d59f9c12b33ae4e9450619cf2488e2bbe9b44030905877f0b2324980", size = 105641, upload-time = "2025-05-02T08:33:18.753Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/28/f8/dfb01ff6cc9af38552c69c9027501ff5a5117c4cc18dcd27cb5259fa1888/charset_normalizer-3.4.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:005fa3432484527f9732ebd315da8da8001593e2cf46a3d817669f062c3d9ed4", size = 201671, upload-time = "2025-05-02T08:34:12.696Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/32/fb/74e26ee556a9dbfe3bd264289b67be1e6d616329403036f6507bb9f3f29c/charset_normalizer-3.4.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e92fca20c46e9f5e1bb485887d074918b13543b1c2a1185e69bb8d17ab6236a7", size = 144744, upload-time = "2025-05-02T08:34:14.665Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/ad/06/8499ee5aa7addc6f6d72e068691826ff093329fe59891e83b092ae4c851c/charset_normalizer-3.4.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:50bf98d5e563b83cc29471fa114366e6806bc06bc7a25fd59641e41445327836", size = 154993, upload-time = "2025-05-02T08:34:17.134Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/f1/a2/5e4c187680728219254ef107a6949c60ee0e9a916a5dadb148c7ae82459c/charset_normalizer-3.4.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:721c76e84fe669be19c5791da68232ca2e05ba5185575086e384352e2c309597", size = 147382, upload-time = "2025-05-02T08:34:19.081Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/4c/fe/56aca740dda674f0cc1ba1418c4d84534be51f639b5f98f538b332dc9a95/charset_normalizer-3.4.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:82d8fd25b7f4675d0c47cf95b594d4e7b158aca33b76aa63d07186e13c0e0ab7", size = 149536, upload-time = "2025-05-02T08:34:21.073Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/53/13/db2e7779f892386b589173dd689c1b1e304621c5792046edd8a978cbf9e0/charset_normalizer-3.4.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b3daeac64d5b371dea99714f08ffc2c208522ec6b06fbc7866a450dd446f5c0f", size = 151349, upload-time = "2025-05-02T08:34:23.193Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/69/35/e52ab9a276186f729bce7a0638585d2982f50402046e4b0faa5d2c3ef2da/charset_normalizer-3.4.2-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:dccab8d5fa1ef9bfba0590ecf4d46df048d18ffe3eec01eeb73a42e0d9e7a8ba", size = 146365, upload-time = "2025-05-02T08:34:25.187Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a6/d8/af7333f732fc2e7635867d56cb7c349c28c7094910c72267586947561b4b/charset_normalizer-3.4.2-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:aaf27faa992bfee0264dc1f03f4c75e9fcdda66a519db6b957a3f826e285cf12", size = 154499, upload-time = "2025-05-02T08:34:27.359Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/7a/3d/a5b2e48acef264d71e036ff30bcc49e51bde80219bb628ba3e00cf59baac/charset_normalizer-3.4.2-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:eb30abc20df9ab0814b5a2524f23d75dcf83cde762c161917a2b4b7b55b1e518", size = 157735, upload-time = "2025-05-02T08:34:29.798Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/85/d8/23e2c112532a29f3eef374375a8684a4f3b8e784f62b01da931186f43494/charset_normalizer-3.4.2-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:c72fbbe68c6f32f251bdc08b8611c7b3060612236e960ef848e0a517ddbe76c5", size = 154786, upload-time = "2025-05-02T08:34:31.858Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c7/57/93e0169f08ecc20fe82d12254a200dfaceddc1c12a4077bf454ecc597e33/charset_normalizer-3.4.2-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:982bb1e8b4ffda883b3d0a521e23abcd6fd17418f6d2c4118d257a10199c0ce3", size = 150203, upload-time = "2025-05-02T08:34:33.88Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2c/9d/9bf2b005138e7e060d7ebdec7503d0ef3240141587651f4b445bdf7286c2/charset_normalizer-3.4.2-cp39-cp39-win32.whl", hash = "sha256:43e0933a0eff183ee85833f341ec567c0980dae57c464d8a508e1b2ceb336471", size = 98436, upload-time = "2025-05-02T08:34:35.907Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/6d/24/5849d46cf4311bbf21b424c443b09b459f5b436b1558c04e45dbb7cc478b/charset_normalizer-3.4.2-cp39-cp39-win_amd64.whl", hash = "sha256:d11b54acf878eef558599658b0ffca78138c8c3655cf4f3a4a673c437e67732e", size = 105772, upload-time = "2025-05-02T08:34:37.935Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/20/94/c5790835a017658cbfabd07f3bfb549140c3ac458cfc196323996b10095a/charset_normalizer-3.4.2-py3-none-any.whl", hash = "sha256:7f56930ab0abd1c45cd15be65cc741c28b1c9a34876ce8c17a2fa107810c0af0", size = 52626, upload-time = "2025-05-02T08:34:40.053Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "colorama" | ||||||
|  | version = "0.4.6" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697, upload-time = "2022-10-25T02:36:22.414Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "dspace-statistics-api" | ||||||
|  | version = "1.4.6.dev0" | ||||||
|  | source = { editable = "." } | ||||||
|  | dependencies = [ | ||||||
|  |     { name = "falcon" }, | ||||||
|  |     { name = "falcon-swagger-ui" }, | ||||||
|  |     { name = "gunicorn" }, | ||||||
|  |     { name = "psycopg" }, | ||||||
|  |     { name = "requests" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [package.dev-dependencies] | ||||||
|  | dev = [ | ||||||
|  |     { name = "flake8" }, | ||||||
|  |     { name = "isort" }, | ||||||
|  |     { name = "pytest" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [package.metadata] | ||||||
|  | requires-dist = [ | ||||||
|  |     { name = "falcon", specifier = "==4.0.*" }, | ||||||
|  |     { name = "falcon-swagger-ui", git = "https://github.com/alanorth/falcon-swagger-ui?rev=falcon3-update-swagger-ui" }, | ||||||
|  |     { name = "gunicorn", specifier = "==23.0.*" }, | ||||||
|  |     { name = "psycopg", specifier = "==3.2.*" }, | ||||||
|  |     { name = "requests", specifier = "==2.32.*" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [package.metadata.requires-dev] | ||||||
|  | dev = [ | ||||||
|  |     { name = "flake8", specifier = "==7.1.*" }, | ||||||
|  |     { name = "isort", specifier = "==5.13.*" }, | ||||||
|  |     { name = "pytest", specifier = "==8.3.*" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "exceptiongroup" | ||||||
|  | version = "1.3.0" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | dependencies = [ | ||||||
|  |     { name = "typing-extensions", marker = "python_full_version < '3.13'" }, | ||||||
|  | ] | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/0b/9f/a65090624ecf468cdca03533906e7c69ed7588582240cfe7cc9e770b50eb/exceptiongroup-1.3.0.tar.gz", hash = "sha256:b241f5885f560bc56a59ee63ca4c6a8bfa46ae4ad651af316d4e81817bb9fd88", size = 29749, upload-time = "2025-05-10T17:42:51.123Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/36/f4/c6e662dade71f56cd2f3735141b265c3c79293c109549c1e6933b0651ffc/exceptiongroup-1.3.0-py3-none-any.whl", hash = "sha256:4d111e6e0c13d0644cad6ddaa7ed0261a0b36971f6d23e7ec9b4b9097da78a10", size = 16674, upload-time = "2025-05-10T17:42:49.33Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "falcon" | ||||||
|  | version = "4.0.2" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/37/4f/d317952294dee1982cd930c8ee2b8b7fbf04140473882801061b3346c713/falcon-4.0.2.tar.gz", hash = "sha256:58f4b9c9da4c9b1e2c9f396ad7ef897701b3c7c7c87227f0bd1aee40c7fbc525", size = 630121, upload-time = "2024-11-06T19:21:20.751Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/b0/75/e33013aedec976d13f2104ab2e054b5e3863b518c9b28239d2837b521d7f/falcon-4.0.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:8eab0212e77017385d48be2dfe9f5b32305fc9e4066cd298e4bb39e666e114c8", size = 2315204, upload-time = "2024-11-06T19:49:42.423Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/90/5f/4a3ccb6d8bdb4cfcc38aea9cd5e5c49aea400305f581c839be206c3a93e2/falcon-4.0.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:942129dd3bfb56342ac368f05ff4f9be53e98883b4227089fce2fd616ebc6ef3", size = 2194755, upload-time = "2024-11-06T19:49:44.818Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/28/99/447d6f8618b3f8b882c7e74eafabb59f6e9112acbc3255dddd353fc75505/falcon-4.0.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:60e7b6e5ee44bb2411a7f47bb64e0b225f11cca6ddf91e5130d456242095f0d7", size = 10390701, upload-time = "2024-11-06T19:49:47.514Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/cb/27/a93dc68be1e70809cfa6d227424790ff502cc1f4272200bb91ebe92fafb1/falcon-4.0.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:330f1623e579575a9e3d90c2a15aebe100b2afa1e18a4bee2ddaa9a570e97902", size = 10986193, upload-time = "2024-11-06T19:49:49.885Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/cb/22/aae29170b0947a5170844bf74b671b5e0e6dc218e2cc6262d2378ec44d15/falcon-4.0.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9d7cfac5cfca69373d1f65211d75767ed4f2d53b46554307427ec00a6f7f87c1", size = 10511133, upload-time = "2024-11-06T19:49:52.944Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2f/73/0862e66b5b9c5295f065ee7c83571fd106dd84d63cf0479038986ddf2881/falcon-4.0.2-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:da3d942dd34f7a5213987bd053c3b52b6eb75fcfd342dc4fea9241f79a6529b3", size = 10201951, upload-time = "2024-11-06T19:49:56.08Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/78/4f/044454ad96a542f2c446a07b6ebc8da0bef8e7e689e32aeb2bf40594a712/falcon-4.0.2-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:5169e064bbe5dece52e088e3e8b17cae429f1e04c7aef8c31ae350303b19c620", size = 10633849, upload-time = "2024-11-06T19:49:58.532Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2f/73/bd689c2790c42b6287df1a43928ae3af6cd2541ac1d64d6fa2fc960dad5b/falcon-4.0.2-cp310-cp310-win_amd64.whl", hash = "sha256:0d62e565b9e71b52b59e03130b2b71345a6873f5299aad6a141caf4a58661b41", size = 2115914, upload-time = "2024-11-06T19:50:01Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/01/e3/dfeff966d60f2308f765736044e0a62f046d2420baf50fa4872b06338fd5/falcon-4.0.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:cb6ee1aee9ff6a656762cf5fcd2e6c5dced410ca990016be2bc193e6b74ae9da", size = 2321291, upload-time = "2024-11-06T19:50:03.281Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/73/02/8a1a68ddf9b6f9d4a7d0d63f0a485109318f08c0181a3d9f4b05dceab355/falcon-4.0.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:7f1a16d8bdc8ef9cf2832a6ca6d43b156b613fb1587cd08cc928c7b8a118ea0a", size = 2196225, upload-time = "2024-11-06T19:50:05.558Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/9f/64/6fa45987bd0fc78d991be5ea0e30f0812eb26713c2e99ce07e35c959346b/falcon-4.0.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:aee81fc4702eef5bccb640b93187fdf36ca2606fca511982069dbc60be2d1c93", size = 11603829, upload-time = "2024-11-06T19:50:07.477Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/84/ef/bc1d47ee32e2a211cffca346bd935009d2b37189c2119df95c31a9af6231/falcon-4.0.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9c1dbcec63d9118c3dfac1f810305128c4fffe26f4f99a7b4e379dec95fc3bfc", size = 12146705, upload-time = "2024-11-06T19:50:09.995Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/e9/cd/ef07fd256c2a29d3c3f1cc22e0ce59724450eb13ef5ca553e63b6abd19f7/falcon-4.0.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c2892ab1232d3a7cc9890b1b539c471fe04c54f826704f9d05efe5632f18efa1", size = 11693278, upload-time = "2024-11-06T19:50:12.393Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/7e/2b/e03066e7be01f1b09c87ecd9c48e14a0ddafd9fc8fddd05db27b5ad4e3d9/falcon-4.0.2-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:af68482b408bde53a77b36e45317767dfc5b6fce1525f5b25d65f57f35d33fca", size = 11379786, upload-time = "2024-11-06T19:50:14.964Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2d/15/f218b581df1447f743b16812c84ab8f6f7d51fb3c1950129744f6fd653bc/falcon-4.0.2-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:53d84de83abe1a2094b319a4f018ab6c5773d9c2c841b528662aa151ab9df35c", size = 11770361, upload-time = "2024-11-06T19:50:18.185Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/11/3e/855c3051cb8aad61a921959e5e62416d759761241c1da8394103c3d1d6af/falcon-4.0.2-cp311-cp311-win_amd64.whl", hash = "sha256:1d06bbbccdb58522b2a6bb2e79074844b0db0da1fff407725858a02515e15bbd", size = 2124712, upload-time = "2024-11-06T19:50:20.543Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/67/db/0b78b7ee3fe7e370ed430b7deabfa524b57a5b9eb32622ce1f1bb7aacf0d/falcon-4.0.2-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:23b0419a9a025745734022aaa2e65447595e539ba27352b3f59d86b288f614db", size = 2294078, upload-time = "2024-11-06T19:50:22.03Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/63/01/77b5b0214bc4ca717b6c6cbe8c3adaba653a7312c9c51a9b390f66efbce0/falcon-4.0.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:524d7b75f7368fe82e94ed16370db5a27bb4b2d066470cba53f02304264447e8", size = 2186007, upload-time = "2024-11-06T19:50:23.876Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/1d/0c/44abd34e38b88f15c5a7030f48ec079669218af3162de2bd1925e13a46a5/falcon-4.0.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2c6b1d7451d5dee4be9b67a75e2a4a0b024dccffedd4e7c7a09513733b5a11db", size = 11644341, upload-time = "2024-11-06T19:50:26.429Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/5e/f3/0260f70dd080d23372e2ff0e330ca37897ab5e1b4890df281558bda8e34a/falcon-4.0.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:59bb4a29626c5e610c62620a1395755e8c7b5509385b80d3637fbc8a604d29a3", size = 12218712, upload-time = "2024-11-06T19:50:29.335Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/8d/f8/704b73fc76cf283504aaacc2f466a08fd5d440cddd8d50b6d5c288f0293b/falcon-4.0.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f26c9ed2912ee48e2e1e7eca3e7e85ab664ff07bd321097a26e4ad6168059424", size = 11842131, upload-time = "2024-11-06T19:50:32.317Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/f2/9a/53f9ee7ee8758972d92bb3dfb2225a9c382fbd12f684616cab9126420602/falcon-4.0.2-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:0a12bbf3482b7ef1db0c6727c2ad8be5c3ac777d892e56a170e0b4b93651c915", size = 11346087, upload-time = "2024-11-06T19:50:34.743Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/af/ac/8098957dd5b97ed16788104b7acb33c64689f7ab04e0c6b07d6561182950/falcon-4.0.2-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:a601de7816138f17bf168262e0bceb128fdd1ea2f29ddae035585b5da9223a21", size = 11992383, upload-time = "2024-11-06T19:50:37.394Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/7a/23/f8a74294b5b0cb5b9e3eb44ccea310a5d480ef95e938704827db0dd97f99/falcon-4.0.2-cp312-cp312-win_amd64.whl", hash = "sha256:eec3feda4a9cd773203401e3cf425728a13bf5055b22243b1452e9ad963634f5", size = 2077409, upload-time = "2024-11-06T19:50:40.396Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/1f/34/71ef64406ac7f83c5726a37c9fcae0578bc9d650de09c32148aa6c58502f/falcon-4.0.2-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:110b172afe337fbae802f1402c89a5dfe6392f3b8ce4f2ecdfd5cee48f68b805", size = 2257265, upload-time = "2024-11-06T19:50:41.967Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/b7/13/528d074e8a75a9236c9f060685e4cb813fbca774269afc89d31e821d8560/falcon-4.0.2-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:b3a5db14cf2ef05f8f9630468c03939b86dc16115a5250a1870dac3dca1e04ba", size = 2151133, upload-time = "2024-11-06T19:50:44.614Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2f/70/8c7bf8bf941238a87debce72fcdc7b2301d6599271a392c8216ea2f5d91e/falcon-4.0.2-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:14e0b4d41ce29c2b5c5b18021320e9e0977ba47ade46b67face52ee1325e2ea4", size = 11438997, upload-time = "2024-11-06T19:50:46.539Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/fa/43/71e358d36ec4559737d63312d746fb5f8b0e64f1fe273cd6991e567a9225/falcon-4.0.2-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:56af3b8838da2e19ae56b4e1bac168669ba257d6941f94933dc4f814fe721c08", size = 12015197, upload-time = "2024-11-06T19:50:49.557Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/7a/70/c10acaa3486748f77d9b0e79aaa19d3023b760bb9b93389ac1883a52e366/falcon-4.0.2-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ec939d26dd77f57f08f3e13fb14b4e609c0baf073dc3f0c368f0e4cc10439528", size = 11653687, upload-time = "2024-11-06T19:50:52.091Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/b7/3b/dfdd9bd9f6114a49a55298b12048f1b65d0813b82c28676b956c4444f707/falcon-4.0.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:9bfd751dd898505e17152d7ecfcdc457c9d85bceed7e651a9915183bd4afc86b", size = 11165291, upload-time = "2024-11-06T19:50:54.459Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/01/61/eb3d1d2076df85d5a7c2cd823ba5dbe0a928053a3102effb9006b2851377/falcon-4.0.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:4b85f9c6f50a7465303290cb305404ea5c1ddeff6702179c1a8879c4693b0e5e", size = 11831049, upload-time = "2024-11-06T19:50:58.192Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/bf/c7/268cddb1f84ebe5b402acdf116083658f3fb0dd38a75571e0ee703cef212/falcon-4.0.2-cp313-cp313-win_amd64.whl", hash = "sha256:a410e4023999a74ccf615fafa646b112044b987ef5901c8e5c5b79b163f2b3ba", size = 2052994, upload-time = "2024-11-06T19:51:01.138Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/d0/00/7be6347247812e6553be50d83b0951e569d597b9c3a71e4c0de5b00789b7/falcon-4.0.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:f9709fd9181f58d492463b951cc42fb33b230e8f261128bc8252a37a4553f318", size = 2323679, upload-time = "2024-11-06T19:51:08.688Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c4/71/65266fc8433e396f42a2b045e7b7069390c4314e8b66e66c4e092166226f/falcon-4.0.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:427c20ceb367039b856506d7baeef17c7f0c40b8fcbf1147c0e76f33a574a7cf", size = 2203699, upload-time = "2024-11-06T19:51:10.212Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/80/30/82ff9d61d6baa5f908ccbbb775585cd1ac24dd47fcc2d2e6c5b9b0f44ce4/falcon-4.0.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1fb50cebc3cae6720ccf4a05fccb233ea6a88e803828a07c063d6dce10a74e0e", size = 10418688, upload-time = "2024-11-06T19:51:11.904Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/bf/e5/660262ee87a90aab812d2d10ab42e6e0bffe853890b2e14268863c4dd659/falcon-4.0.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:628c450e14af811f13db6334265d7ff8a7b8a25ece1bde35d09a367a72046533", size = 11033845, upload-time = "2024-11-06T19:51:15.135Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/8f/b2/cf4c9567ad571e3304743e6f65c42576929cc16d494cb234f972fa70a150/falcon-4.0.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e04b30a7f89e5413e00c5cd1ea62bf7948323eb0220f8a5bbf705abae266a384", size = 10547880, upload-time = "2024-11-06T19:51:17.883Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/e2/b8/503d3574be76a09b64dd48214e012fd1b911a158400bc04b1ee0d4caec0f/falcon-4.0.2-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:9095a36b8eeb80207322393b3bc88edaacd0426c2907e8427617618421bde9cc", size = 10218888, upload-time = "2024-11-06T19:51:20.683Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2c/88/d96a3e9d93aee74280a82be844c2eaa603283c5548b3293165deb2d55b4e/falcon-4.0.2-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:0adc6c2887f9d7ed55fe38edef055cc85c26762e392d80dca8765184c180b921", size = 10663138, upload-time = "2024-11-06T19:51:23.079Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/38/97/4021fce87e3feb67839405ca8d2560d989da141692214c6f1b297af23443/falcon-4.0.2-cp39-cp39-win_amd64.whl", hash = "sha256:7bffb4cadcbf7c5994695d421ef5305ad8315cfbefe971713046967614f0ffa4", size = 2121203, upload-time = "2024-11-06T19:51:25.453Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/20/e2/ef821224a9ca9d4bb81d6e7ba60c6fbf3eae2e0dc10d806e6ff21b6dfdc5/falcon-4.0.2-py3-none-any.whl", hash = "sha256:077b2abf001940c6128c9b5872ae8147fe13f6ca333f928d8045d7601a5e847e", size = 318356, upload-time = "2024-11-06T19:21:18.29Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "falcon-swagger-ui" | ||||||
|  | version = "1.2.1" | ||||||
|  | source = { git = "https://github.com/alanorth/falcon-swagger-ui?rev=falcon3-update-swagger-ui#c019c270b479c03d9276e20fd95488495b0943f6" } | ||||||
|  | dependencies = [ | ||||||
|  |     { name = "falcon" }, | ||||||
|  |     { name = "jinja2" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "flake8" | ||||||
|  | version = "7.1.2" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | dependencies = [ | ||||||
|  |     { name = "mccabe" }, | ||||||
|  |     { name = "pycodestyle" }, | ||||||
|  |     { name = "pyflakes" }, | ||||||
|  | ] | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/58/16/3f2a0bb700ad65ac9663262905a025917c020a3f92f014d2ba8964b4602c/flake8-7.1.2.tar.gz", hash = "sha256:c586ffd0b41540951ae41af572e6790dbd49fc12b3aa2541685d253d9bd504bd", size = 48119, upload-time = "2025-02-16T18:45:44.296Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/35/f8/08d37b2cd89da306e3520bd27f8a85692122b42b56c0c2c3784ff09c022f/flake8-7.1.2-py2.py3-none-any.whl", hash = "sha256:1cbc62e65536f65e6d754dfe6f1bada7f5cf392d6f5db3c2b85892466c3e7c1a", size = 57745, upload-time = "2025-02-16T18:45:42.351Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "gunicorn" | ||||||
|  | version = "23.0.0" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | dependencies = [ | ||||||
|  |     { name = "packaging" }, | ||||||
|  | ] | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/34/72/9614c465dc206155d93eff0ca20d42e1e35afc533971379482de953521a4/gunicorn-23.0.0.tar.gz", hash = "sha256:f014447a0101dc57e294f6c18ca6b40227a4c90e9bdb586042628030cba004ec", size = 375031, upload-time = "2024-08-10T20:25:27.378Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/cb/7d/6dac2a6e1eba33ee43f318edbed4ff29151a49b5d37f080aad1e6469bca4/gunicorn-23.0.0-py3-none-any.whl", hash = "sha256:ec400d38950de4dfd418cff8328b2c8faed0edb0d517d3394e457c317908ca4d", size = 85029, upload-time = "2024-08-10T20:25:24.996Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "idna" | ||||||
|  | version = "3.10" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/f1/70/7703c29685631f5a7590aa73f1f1d3fa9a380e654b86af429e0934a32f7d/idna-3.10.tar.gz", hash = "sha256:12f65c9b470abda6dc35cf8e63cc574b1c52b11df2c86030af0ac09b01b13ea9", size = 190490, upload-time = "2024-09-15T18:07:39.745Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/76/c6/c88e154df9c4e1a2a66ccf0005a88dfb2650c1dffb6f5ce603dfbd452ce3/idna-3.10-py3-none-any.whl", hash = "sha256:946d195a0d259cbba61165e88e65941f16e9b36ea6ddb97f00452bae8b1287d3", size = 70442, upload-time = "2024-09-15T18:07:37.964Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "iniconfig" | ||||||
|  | version = "2.1.0" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/f2/97/ebf4da567aa6827c909642694d71c9fcf53e5b504f2d96afea02718862f3/iniconfig-2.1.0.tar.gz", hash = "sha256:3abbd2e30b36733fee78f9c7f7308f2d0050e88f0087fd25c2645f63c773e1c7", size = 4793, upload-time = "2025-03-19T20:09:59.721Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2c/e1/e6716421ea10d38022b952c159d5161ca1193197fb744506875fbb87ea7b/iniconfig-2.1.0-py3-none-any.whl", hash = "sha256:9deba5723312380e77435581c6bf4935c94cbfab9b1ed33ef8d238ea168eb760", size = 6050, upload-time = "2025-03-19T20:10:01.071Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "isort" | ||||||
|  | version = "5.13.2" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/87/f9/c1eb8635a24e87ade2efce21e3ce8cd6b8630bb685ddc9cdaca1349b2eb5/isort-5.13.2.tar.gz", hash = "sha256:48fdfcb9face5d58a4f6dde2e72a1fb8dcaf8ab26f95ab49fab84c2ddefb0109", size = 175303, upload-time = "2023-12-13T20:37:26.124Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/d1/b3/8def84f539e7d2289a02f0524b944b15d7c75dab7628bedf1c4f0992029c/isort-5.13.2-py3-none-any.whl", hash = "sha256:8ca5e72a8d85860d5a3fa69b8745237f2939afe12dbf656afbcb47fe72d947a6", size = 92310, upload-time = "2023-12-13T20:37:23.244Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "jinja2" | ||||||
|  | version = "3.1.6" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | dependencies = [ | ||||||
|  |     { name = "markupsafe" }, | ||||||
|  | ] | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/df/bf/f7da0350254c0ed7c72f3e33cef02e048281fec7ecec5f032d4aac52226b/jinja2-3.1.6.tar.gz", hash = "sha256:0137fb05990d35f1275a587e9aee6d56da821fc83491a0fb838183be43f66d6d", size = 245115, upload-time = "2025-03-05T20:05:02.478Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/62/a1/3d680cbfd5f4b8f15abc1d571870c5fc3e594bb582bc3b64ea099db13e56/jinja2-3.1.6-py3-none-any.whl", hash = "sha256:85ece4451f492d0c13c5dd7c13a64681a86afae63a5f347908daf103ce6d2f67", size = 134899, upload-time = "2025-03-05T20:05:00.369Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "markupsafe" | ||||||
|  | version = "3.0.2" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/b2/97/5d42485e71dfc078108a86d6de8fa46db44a1a9295e89c5d6d4a06e23a62/markupsafe-3.0.2.tar.gz", hash = "sha256:ee55d3edf80167e48ea11a923c7386f4669df67d7994554387f84e7d8b0a2bf0", size = 20537, upload-time = "2024-10-18T15:21:54.129Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/04/90/d08277ce111dd22f77149fd1a5d4653eeb3b3eaacbdfcbae5afb2600eebd/MarkupSafe-3.0.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:7e94c425039cde14257288fd61dcfb01963e658efbc0ff54f5306b06054700f8", size = 14357, upload-time = "2024-10-18T15:20:51.44Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/04/e1/6e2194baeae0bca1fae6629dc0cbbb968d4d941469cbab11a3872edff374/MarkupSafe-3.0.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9e2d922824181480953426608b81967de705c3cef4d1af983af849d7bd619158", size = 12393, upload-time = "2024-10-18T15:20:52.426Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/1d/69/35fa85a8ece0a437493dc61ce0bb6d459dcba482c34197e3efc829aa357f/MarkupSafe-3.0.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:38a9ef736c01fccdd6600705b09dc574584b89bea478200c5fbf112a6b0d5579", size = 21732, upload-time = "2024-10-18T15:20:53.578Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/22/35/137da042dfb4720b638d2937c38a9c2df83fe32d20e8c8f3185dbfef05f7/MarkupSafe-3.0.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bbcb445fa71794da8f178f0f6d66789a28d7319071af7a496d4d507ed566270d", size = 20866, upload-time = "2024-10-18T15:20:55.06Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/29/28/6d029a903727a1b62edb51863232152fd335d602def598dade38996887f0/MarkupSafe-3.0.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:57cb5a3cf367aeb1d316576250f65edec5bb3be939e9247ae594b4bcbc317dfb", size = 20964, upload-time = "2024-10-18T15:20:55.906Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/cc/cd/07438f95f83e8bc028279909d9c9bd39e24149b0d60053a97b2bc4f8aa51/MarkupSafe-3.0.2-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:3809ede931876f5b2ec92eef964286840ed3540dadf803dd570c3b7e13141a3b", size = 21977, upload-time = "2024-10-18T15:20:57.189Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/29/01/84b57395b4cc062f9c4c55ce0df7d3108ca32397299d9df00fedd9117d3d/MarkupSafe-3.0.2-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:e07c3764494e3776c602c1e78e298937c3315ccc9043ead7e685b7f2b8d47b3c", size = 21366, upload-time = "2024-10-18T15:20:58.235Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/bd/6e/61ebf08d8940553afff20d1fb1ba7294b6f8d279df9fd0c0db911b4bbcfd/MarkupSafe-3.0.2-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:b424c77b206d63d500bcb69fa55ed8d0e6a3774056bdc4839fc9298a7edca171", size = 21091, upload-time = "2024-10-18T15:20:59.235Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/11/23/ffbf53694e8c94ebd1e7e491de185124277964344733c45481f32ede2499/MarkupSafe-3.0.2-cp310-cp310-win32.whl", hash = "sha256:fcabf5ff6eea076f859677f5f0b6b5c1a51e70a376b0579e0eadef8db48c6b50", size = 15065, upload-time = "2024-10-18T15:21:00.307Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/44/06/e7175d06dd6e9172d4a69a72592cb3f7a996a9c396eee29082826449bbc3/MarkupSafe-3.0.2-cp310-cp310-win_amd64.whl", hash = "sha256:6af100e168aa82a50e186c82875a5893c5597a0c1ccdb0d8b40240b1f28b969a", size = 15514, upload-time = "2024-10-18T15:21:01.122Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/6b/28/bbf83e3f76936960b850435576dd5e67034e200469571be53f69174a2dfd/MarkupSafe-3.0.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:9025b4018f3a1314059769c7bf15441064b2207cb3f065e6ea1e7359cb46db9d", size = 14353, upload-time = "2024-10-18T15:21:02.187Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/6c/30/316d194b093cde57d448a4c3209f22e3046c5bb2fb0820b118292b334be7/MarkupSafe-3.0.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:93335ca3812df2f366e80509ae119189886b0f3c2b81325d39efdb84a1e2ae93", size = 12392, upload-time = "2024-10-18T15:21:02.941Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/f2/96/9cdafba8445d3a53cae530aaf83c38ec64c4d5427d975c974084af5bc5d2/MarkupSafe-3.0.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2cb8438c3cbb25e220c2ab33bb226559e7afb3baec11c4f218ffa7308603c832", size = 23984, upload-time = "2024-10-18T15:21:03.953Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/f1/a4/aefb044a2cd8d7334c8a47d3fb2c9f328ac48cb349468cc31c20b539305f/MarkupSafe-3.0.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a123e330ef0853c6e822384873bef7507557d8e4a082961e1defa947aa59ba84", size = 23120, upload-time = "2024-10-18T15:21:06.495Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/8d/21/5e4851379f88f3fad1de30361db501300d4f07bcad047d3cb0449fc51f8c/MarkupSafe-3.0.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1e084f686b92e5b83186b07e8a17fc09e38fff551f3602b249881fec658d3eca", size = 23032, upload-time = "2024-10-18T15:21:07.295Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/00/7b/e92c64e079b2d0d7ddf69899c98842f3f9a60a1ae72657c89ce2655c999d/MarkupSafe-3.0.2-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:d8213e09c917a951de9d09ecee036d5c7d36cb6cb7dbaece4c71a60d79fb9798", size = 24057, upload-time = "2024-10-18T15:21:08.073Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/f9/ac/46f960ca323037caa0a10662ef97d0a4728e890334fc156b9f9e52bcc4ca/MarkupSafe-3.0.2-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:5b02fb34468b6aaa40dfc198d813a641e3a63b98c2b05a16b9f80b7ec314185e", size = 23359, upload-time = "2024-10-18T15:21:09.318Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/69/84/83439e16197337b8b14b6a5b9c2105fff81d42c2a7c5b58ac7b62ee2c3b1/MarkupSafe-3.0.2-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:0bff5e0ae4ef2e1ae4fdf2dfd5b76c75e5c2fa4132d05fc1b0dabcd20c7e28c4", size = 23306, upload-time = "2024-10-18T15:21:10.185Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/9a/34/a15aa69f01e2181ed8d2b685c0d2f6655d5cca2c4db0ddea775e631918cd/MarkupSafe-3.0.2-cp311-cp311-win32.whl", hash = "sha256:6c89876f41da747c8d3677a2b540fb32ef5715f97b66eeb0c6b66f5e3ef6f59d", size = 15094, upload-time = "2024-10-18T15:21:11.005Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/da/b8/3a3bd761922d416f3dc5d00bfbed11f66b1ab89a0c2b6e887240a30b0f6b/MarkupSafe-3.0.2-cp311-cp311-win_amd64.whl", hash = "sha256:70a87b411535ccad5ef2f1df5136506a10775d267e197e4cf531ced10537bd6b", size = 15521, upload-time = "2024-10-18T15:21:12.911Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/22/09/d1f21434c97fc42f09d290cbb6350d44eb12f09cc62c9476effdb33a18aa/MarkupSafe-3.0.2-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:9778bd8ab0a994ebf6f84c2b949e65736d5575320a17ae8984a77fab08db94cf", size = 14274, upload-time = "2024-10-18T15:21:13.777Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/6b/b0/18f76bba336fa5aecf79d45dcd6c806c280ec44538b3c13671d49099fdd0/MarkupSafe-3.0.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:846ade7b71e3536c4e56b386c2a47adf5741d2d8b94ec9dc3e92e5e1ee1e2225", size = 12348, upload-time = "2024-10-18T15:21:14.822Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/e0/25/dd5c0f6ac1311e9b40f4af06c78efde0f3b5cbf02502f8ef9501294c425b/MarkupSafe-3.0.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1c99d261bd2d5f6b59325c92c73df481e05e57f19837bdca8413b9eac4bd8028", size = 24149, upload-time = "2024-10-18T15:21:15.642Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/f3/f0/89e7aadfb3749d0f52234a0c8c7867877876e0a20b60e2188e9850794c17/MarkupSafe-3.0.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e17c96c14e19278594aa4841ec148115f9c7615a47382ecb6b82bd8fea3ab0c8", size = 23118, upload-time = "2024-10-18T15:21:17.133Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/d5/da/f2eeb64c723f5e3777bc081da884b414671982008c47dcc1873d81f625b6/MarkupSafe-3.0.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:88416bd1e65dcea10bc7569faacb2c20ce071dd1f87539ca2ab364bf6231393c", size = 22993, upload-time = "2024-10-18T15:21:18.064Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/da/0e/1f32af846df486dce7c227fe0f2398dc7e2e51d4a370508281f3c1c5cddc/MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:2181e67807fc2fa785d0592dc2d6206c019b9502410671cc905d132a92866557", size = 24178, upload-time = "2024-10-18T15:21:18.859Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c4/f6/bb3ca0532de8086cbff5f06d137064c8410d10779c4c127e0e47d17c0b71/MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:52305740fe773d09cffb16f8ed0427942901f00adedac82ec8b67752f58a1b22", size = 23319, upload-time = "2024-10-18T15:21:19.671Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a2/82/8be4c96ffee03c5b4a034e60a31294daf481e12c7c43ab8e34a1453ee48b/MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:ad10d3ded218f1039f11a75f8091880239651b52e9bb592ca27de44eed242a48", size = 23352, upload-time = "2024-10-18T15:21:20.971Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/51/ae/97827349d3fcffee7e184bdf7f41cd6b88d9919c80f0263ba7acd1bbcb18/MarkupSafe-3.0.2-cp312-cp312-win32.whl", hash = "sha256:0f4ca02bea9a23221c0182836703cbf8930c5e9454bacce27e767509fa286a30", size = 15097, upload-time = "2024-10-18T15:21:22.646Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c1/80/a61f99dc3a936413c3ee4e1eecac96c0da5ed07ad56fd975f1a9da5bc630/MarkupSafe-3.0.2-cp312-cp312-win_amd64.whl", hash = "sha256:8e06879fc22a25ca47312fbe7c8264eb0b662f6db27cb2d3bbbc74b1df4b9b87", size = 15601, upload-time = "2024-10-18T15:21:23.499Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/83/0e/67eb10a7ecc77a0c2bbe2b0235765b98d164d81600746914bebada795e97/MarkupSafe-3.0.2-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:ba9527cdd4c926ed0760bc301f6728ef34d841f405abf9d4f959c478421e4efd", size = 14274, upload-time = "2024-10-18T15:21:24.577Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2b/6d/9409f3684d3335375d04e5f05744dfe7e9f120062c9857df4ab490a1031a/MarkupSafe-3.0.2-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:f8b3d067f2e40fe93e1ccdd6b2e1d16c43140e76f02fb1319a05cf2b79d99430", size = 12352, upload-time = "2024-10-18T15:21:25.382Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/d2/f5/6eadfcd3885ea85fe2a7c128315cc1bb7241e1987443d78c8fe712d03091/MarkupSafe-3.0.2-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:569511d3b58c8791ab4c2e1285575265991e6d8f8700c7be0e88f86cb0672094", size = 24122, upload-time = "2024-10-18T15:21:26.199Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/0c/91/96cf928db8236f1bfab6ce15ad070dfdd02ed88261c2afafd4b43575e9e9/MarkupSafe-3.0.2-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:15ab75ef81add55874e7ab7055e9c397312385bd9ced94920f2802310c930396", size = 23085, upload-time = "2024-10-18T15:21:27.029Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c2/cf/c9d56af24d56ea04daae7ac0940232d31d5a8354f2b457c6d856b2057d69/MarkupSafe-3.0.2-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f3818cb119498c0678015754eba762e0d61e5b52d34c8b13d770f0719f7b1d79", size = 22978, upload-time = "2024-10-18T15:21:27.846Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/2a/9f/8619835cd6a711d6272d62abb78c033bda638fdc54c4e7f4272cf1c0962b/MarkupSafe-3.0.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:cdb82a876c47801bb54a690c5ae105a46b392ac6099881cdfb9f6e95e4014c6a", size = 24208, upload-time = "2024-10-18T15:21:28.744Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/f9/bf/176950a1792b2cd2102b8ffeb5133e1ed984547b75db47c25a67d3359f77/MarkupSafe-3.0.2-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:cabc348d87e913db6ab4aa100f01b08f481097838bdddf7c7a84b7575b7309ca", size = 23357, upload-time = "2024-10-18T15:21:29.545Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/ce/4f/9a02c1d335caabe5c4efb90e1b6e8ee944aa245c1aaaab8e8a618987d816/MarkupSafe-3.0.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:444dcda765c8a838eaae23112db52f1efaf750daddb2d9ca300bcae1039adc5c", size = 23344, upload-time = "2024-10-18T15:21:30.366Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/ee/55/c271b57db36f748f0e04a759ace9f8f759ccf22b4960c270c78a394f58be/MarkupSafe-3.0.2-cp313-cp313-win32.whl", hash = "sha256:bcf3e58998965654fdaff38e58584d8937aa3096ab5354d493c77d1fdd66d7a1", size = 15101, upload-time = "2024-10-18T15:21:31.207Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/29/88/07df22d2dd4df40aba9f3e402e6dc1b8ee86297dddbad4872bd5e7b0094f/MarkupSafe-3.0.2-cp313-cp313-win_amd64.whl", hash = "sha256:e6a2a455bd412959b57a172ce6328d2dd1f01cb2135efda2e4576e8a23fa3b0f", size = 15603, upload-time = "2024-10-18T15:21:32.032Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/62/6a/8b89d24db2d32d433dffcd6a8779159da109842434f1dd2f6e71f32f738c/MarkupSafe-3.0.2-cp313-cp313t-macosx_10_13_universal2.whl", hash = "sha256:b5a6b3ada725cea8a5e634536b1b01c30bcdcd7f9c6fff4151548d5bf6b3a36c", size = 14510, upload-time = "2024-10-18T15:21:33.625Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/7a/06/a10f955f70a2e5a9bf78d11a161029d278eeacbd35ef806c3fd17b13060d/MarkupSafe-3.0.2-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:a904af0a6162c73e3edcb969eeeb53a63ceeb5d8cf642fade7d39e7963a22ddb", size = 12486, upload-time = "2024-10-18T15:21:34.611Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/34/cf/65d4a571869a1a9078198ca28f39fba5fbb910f952f9dbc5220afff9f5e6/MarkupSafe-3.0.2-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4aa4e5faecf353ed117801a068ebab7b7e09ffb6e1d5e412dc852e0da018126c", size = 25480, upload-time = "2024-10-18T15:21:35.398Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/0c/e3/90e9651924c430b885468b56b3d597cabf6d72be4b24a0acd1fa0e12af67/MarkupSafe-3.0.2-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c0ef13eaeee5b615fb07c9a7dadb38eac06a0608b41570d8ade51c56539e509d", size = 23914, upload-time = "2024-10-18T15:21:36.231Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/66/8c/6c7cf61f95d63bb866db39085150df1f2a5bd3335298f14a66b48e92659c/MarkupSafe-3.0.2-cp313-cp313t-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d16a81a06776313e817c951135cf7340a3e91e8c1ff2fac444cfd75fffa04afe", size = 23796, upload-time = "2024-10-18T15:21:37.073Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/bb/35/cbe9238ec3f47ac9a7c8b3df7a808e7cb50fe149dc7039f5f454b3fba218/MarkupSafe-3.0.2-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:6381026f158fdb7c72a168278597a5e3a5222e83ea18f543112b2662a9b699c5", size = 25473, upload-time = "2024-10-18T15:21:37.932Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/e6/32/7621a4382488aa283cc05e8984a9c219abad3bca087be9ec77e89939ded9/MarkupSafe-3.0.2-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:3d79d162e7be8f996986c064d1c7c817f6df3a77fe3d6859f6f9e7be4b8c213a", size = 24114, upload-time = "2024-10-18T15:21:39.799Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/0d/80/0985960e4b89922cb5a0bac0ed39c5b96cbc1a536a99f30e8c220a996ed9/MarkupSafe-3.0.2-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:131a3c7689c85f5ad20f9f6fb1b866f402c445b220c19fe4308c0b147ccd2ad9", size = 24098, upload-time = "2024-10-18T15:21:40.813Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/82/78/fedb03c7d5380df2427038ec8d973587e90561b2d90cd472ce9254cf348b/MarkupSafe-3.0.2-cp313-cp313t-win32.whl", hash = "sha256:ba8062ed2cf21c07a9e295d5b8a2a5ce678b913b45fdf68c32d95d6c1291e0b6", size = 15208, upload-time = "2024-10-18T15:21:41.814Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/4f/65/6079a46068dfceaeabb5dcad6d674f5f5c61a6fa5673746f42a9f4c233b3/MarkupSafe-3.0.2-cp313-cp313t-win_amd64.whl", hash = "sha256:e444a31f8db13eb18ada366ab3cf45fd4b31e4db1236a4448f68778c1d1a5a2f", size = 15739, upload-time = "2024-10-18T15:21:42.784Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a7/ea/9b1530c3fdeeca613faeb0fb5cbcf2389d816072fab72a71b45749ef6062/MarkupSafe-3.0.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:eaa0a10b7f72326f1372a713e73c3f739b524b3af41feb43e4921cb529f5929a", size = 14344, upload-time = "2024-10-18T15:21:43.721Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/4b/c2/fbdbfe48848e7112ab05e627e718e854d20192b674952d9042ebd8c9e5de/MarkupSafe-3.0.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:48032821bbdf20f5799ff537c7ac3d1fba0ba032cfc06194faffa8cda8b560ff", size = 12389, upload-time = "2024-10-18T15:21:44.666Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/f0/25/7a7c6e4dbd4f867d95d94ca15449e91e52856f6ed1905d58ef1de5e211d0/MarkupSafe-3.0.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1a9d3f5f0901fdec14d8d2f66ef7d035f2157240a433441719ac9a3fba440b13", size = 21607, upload-time = "2024-10-18T15:21:45.452Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/53/8f/f339c98a178f3c1e545622206b40986a4c3307fe39f70ccd3d9df9a9e425/MarkupSafe-3.0.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:88b49a3b9ff31e19998750c38e030fc7bb937398b1f78cfa599aaef92d693144", size = 20728, upload-time = "2024-10-18T15:21:46.295Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/1a/03/8496a1a78308456dbd50b23a385c69b41f2e9661c67ea1329849a598a8f9/MarkupSafe-3.0.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:cfad01eed2c2e0c01fd0ecd2ef42c492f7f93902e39a42fc9ee1692961443a29", size = 20826, upload-time = "2024-10-18T15:21:47.134Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/e6/cf/0a490a4bd363048c3022f2f475c8c05582179bb179defcee4766fb3dcc18/MarkupSafe-3.0.2-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:1225beacc926f536dc82e45f8a4d68502949dc67eea90eab715dea3a21c1b5f0", size = 21843, upload-time = "2024-10-18T15:21:48.334Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/19/a3/34187a78613920dfd3cdf68ef6ce5e99c4f3417f035694074beb8848cd77/MarkupSafe-3.0.2-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:3169b1eefae027567d1ce6ee7cae382c57fe26e82775f460f0b2778beaad66c0", size = 21219, upload-time = "2024-10-18T15:21:49.587Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/17/d8/5811082f85bb88410ad7e452263af048d685669bbbfb7b595e8689152498/MarkupSafe-3.0.2-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:eb7972a85c54febfb25b5c4b4f3af4dcc731994c7da0d8a0b4a6eb0640e1d178", size = 20946, upload-time = "2024-10-18T15:21:50.441Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/7c/31/bd635fb5989440d9365c5e3c47556cfea121c7803f5034ac843e8f37c2f2/MarkupSafe-3.0.2-cp39-cp39-win32.whl", hash = "sha256:8c4e8c3ce11e1f92f6536ff07154f9d49677ebaaafc32db9db4620bc11ed480f", size = 15063, upload-time = "2024-10-18T15:21:51.385Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/b3/73/085399401383ce949f727afec55ec3abd76648d04b9f22e1c0e99cb4bec3/MarkupSafe-3.0.2-cp39-cp39-win_amd64.whl", hash = "sha256:6e296a513ca3d94054c2c881cc913116e90fd030ad1c656b3869762b754f5f8a", size = 15506, upload-time = "2024-10-18T15:21:52.974Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "mccabe" | ||||||
|  | version = "0.7.0" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/e7/ff/0ffefdcac38932a54d2b5eed4e0ba8a408f215002cd178ad1df0f2806ff8/mccabe-0.7.0.tar.gz", hash = "sha256:348e0240c33b60bbdf4e523192ef919f28cb2c3d7d5c7794f74009290f236325", size = 9658, upload-time = "2022-01-24T01:14:51.113Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/27/1a/1f68f9ba0c207934b35b86a8ca3aad8395a3d6dd7921c0686e23853ff5a9/mccabe-0.7.0-py2.py3-none-any.whl", hash = "sha256:6c2d30ab6be0e4a46919781807b4f0d834ebdd6c6e3dca0bda5a15f863427b6e", size = 7350, upload-time = "2022-01-24T01:14:49.62Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "packaging" | ||||||
|  | version = "25.0" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/a1/d4/1fc4078c65507b51b96ca8f8c3ba19e6a61c8253c72794544580a7b6c24d/packaging-25.0.tar.gz", hash = "sha256:d443872c98d677bf60f6a1f2f8c1cb748e8fe762d2bf9d3148b5599295b0fc4f", size = 165727, upload-time = "2025-04-19T11:48:59.673Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/20/12/38679034af332785aac8774540895e234f4d07f7545804097de4b666afd8/packaging-25.0-py3-none-any.whl", hash = "sha256:29572ef2b1f17581046b3a2227d5c611fb25ec70ca1ba8554b24b0e69331a484", size = 66469, upload-time = "2025-04-19T11:48:57.875Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "pluggy" | ||||||
|  | version = "1.6.0" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/f9/e2/3e91f31a7d2b083fe6ef3fa267035b518369d9511ffab804f839851d2779/pluggy-1.6.0.tar.gz", hash = "sha256:7dcc130b76258d33b90f61b658791dede3486c3e6bfb003ee5c9bfb396dd22f3", size = 69412, upload-time = "2025-05-15T12:30:07.975Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/54/20/4d324d65cc6d9205fabedc306948156824eb9f0ee1633355a8f7ec5c66bf/pluggy-1.6.0-py3-none-any.whl", hash = "sha256:e920276dd6813095e9377c0bc5566d94c932c33b27a3e3945d8389c374dd4746", size = 20538, upload-time = "2025-05-15T12:30:06.134Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "psycopg" | ||||||
|  | version = "3.2.9" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | dependencies = [ | ||||||
|  |     { name = "typing-extensions", marker = "python_full_version < '3.13'" }, | ||||||
|  |     { name = "tzdata", marker = "sys_platform == 'win32'" }, | ||||||
|  | ] | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/27/4a/93a6ab570a8d1a4ad171a1f4256e205ce48d828781312c0bbaff36380ecb/psycopg-3.2.9.tar.gz", hash = "sha256:2fbb46fcd17bc81f993f28c47f1ebea38d66ae97cc2dbc3cad73b37cefbff700", size = 158122, upload-time = "2025-05-13T16:11:15.533Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/44/b0/a73c195a56eb6b92e937a5ca58521a5c3346fb233345adc80fd3e2f542e2/psycopg-3.2.9-py3-none-any.whl", hash = "sha256:01a8dadccdaac2123c916208c96e06631641c0566b22005493f09663c7a8d3b6", size = 202705, upload-time = "2025-05-13T16:06:26.584Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "pycodestyle" | ||||||
|  | version = "2.12.1" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/43/aa/210b2c9aedd8c1cbeea31a50e42050ad56187754b34eb214c46709445801/pycodestyle-2.12.1.tar.gz", hash = "sha256:6838eae08bbce4f6accd5d5572075c63626a15ee3e6f842df996bf62f6d73521", size = 39232, upload-time = "2024-08-04T20:26:54.576Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/3a/d8/a211b3f85e99a0daa2ddec96c949cac6824bd305b040571b82a03dd62636/pycodestyle-2.12.1-py2.py3-none-any.whl", hash = "sha256:46f0fb92069a7c28ab7bb558f05bfc0110dac69a0cd23c61ea0040283a9d78b3", size = 31284, upload-time = "2024-08-04T20:26:53.173Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "pyflakes" | ||||||
|  | version = "3.2.0" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/57/f9/669d8c9c86613c9d568757c7f5824bd3197d7b1c6c27553bc5618a27cce2/pyflakes-3.2.0.tar.gz", hash = "sha256:1c61603ff154621fb2a9172037d84dca3500def8c8b630657d1701f026f8af3f", size = 63788, upload-time = "2024-01-05T00:28:47.703Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/d4/d7/f1b7db88d8e4417c5d47adad627a93547f44bdc9028372dbd2313f34a855/pyflakes-3.2.0-py2.py3-none-any.whl", hash = "sha256:84b5be138a2dfbb40689ca07e2152deb896a65c3a3e24c251c5c62489568074a", size = 62725, upload-time = "2024-01-05T00:28:45.903Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "pytest" | ||||||
|  | version = "8.3.5" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | dependencies = [ | ||||||
|  |     { name = "colorama", marker = "sys_platform == 'win32'" }, | ||||||
|  |     { name = "exceptiongroup", marker = "python_full_version < '3.11'" }, | ||||||
|  |     { name = "iniconfig" }, | ||||||
|  |     { name = "packaging" }, | ||||||
|  |     { name = "pluggy" }, | ||||||
|  |     { name = "tomli", marker = "python_full_version < '3.11'" }, | ||||||
|  | ] | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/ae/3c/c9d525a414d506893f0cd8a8d0de7706446213181570cdbd766691164e40/pytest-8.3.5.tar.gz", hash = "sha256:f4efe70cc14e511565ac476b57c279e12a855b11f48f212af1080ef2263d3845", size = 1450891, upload-time = "2025-03-02T12:54:54.503Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/30/3d/64ad57c803f1fa1e963a7946b6e0fea4a70df53c1a7fed304586539c2bac/pytest-8.3.5-py3-none-any.whl", hash = "sha256:c69214aa47deac29fad6c2a4f590b9c4a9fdb16a403176fe154b79c0b4d4d820", size = 343634, upload-time = "2025-03-02T12:54:52.069Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "requests" | ||||||
|  | version = "2.32.4" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | dependencies = [ | ||||||
|  |     { name = "certifi" }, | ||||||
|  |     { name = "charset-normalizer" }, | ||||||
|  |     { name = "idna" }, | ||||||
|  |     { name = "urllib3" }, | ||||||
|  | ] | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/e1/0a/929373653770d8a0d7ea76c37de6e41f11eb07559b103b1c02cafb3f7cf8/requests-2.32.4.tar.gz", hash = "sha256:27d0316682c8a29834d3264820024b62a36942083d52caf2f14c0591336d3422", size = 135258, upload-time = "2025-06-09T16:43:07.34Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/7c/e4/56027c4a6b4ae70ca9de302488c5ca95ad4a39e190093d6c1a8ace08341b/requests-2.32.4-py3-none-any.whl", hash = "sha256:27babd3cda2a6d50b30443204ee89830707d396671944c998b5975b031ac2b2c", size = 64847, upload-time = "2025-06-09T16:43:05.728Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "tomli" | ||||||
|  | version = "2.2.1" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/18/87/302344fed471e44a87289cf4967697d07e532f2421fdaf868a303cbae4ff/tomli-2.2.1.tar.gz", hash = "sha256:cd45e1dc79c835ce60f7404ec8119f2eb06d38b1deba146f07ced3bbc44505ff", size = 17175, upload-time = "2024-11-27T22:38:36.873Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/43/ca/75707e6efa2b37c77dadb324ae7d9571cb424e61ea73fad7c56c2d14527f/tomli-2.2.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:678e4fa69e4575eb77d103de3df8a895e1591b48e740211bd1067378c69e8249", size = 131077, upload-time = "2024-11-27T22:37:54.956Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c7/16/51ae563a8615d472fdbffc43a3f3d46588c264ac4f024f63f01283becfbb/tomli-2.2.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:023aa114dd824ade0100497eb2318602af309e5a55595f76b626d6d9f3b7b0a6", size = 123429, upload-time = "2024-11-27T22:37:56.698Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/f1/dd/4f6cd1e7b160041db83c694abc78e100473c15d54620083dbd5aae7b990e/tomli-2.2.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ece47d672db52ac607a3d9599a9d48dcb2f2f735c6c2d1f34130085bb12b112a", size = 226067, upload-time = "2024-11-27T22:37:57.63Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a9/6b/c54ede5dc70d648cc6361eaf429304b02f2871a345bbdd51e993d6cdf550/tomli-2.2.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6972ca9c9cc9f0acaa56a8ca1ff51e7af152a9f87fb64623e31d5c83700080ee", size = 236030, upload-time = "2024-11-27T22:37:59.344Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/1f/47/999514fa49cfaf7a92c805a86c3c43f4215621855d151b61c602abb38091/tomli-2.2.1-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c954d2250168d28797dd4e3ac5cf812a406cd5a92674ee4c8f123c889786aa8e", size = 240898, upload-time = "2024-11-27T22:38:00.429Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/73/41/0a01279a7ae09ee1573b423318e7934674ce06eb33f50936655071d81a24/tomli-2.2.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:8dd28b3e155b80f4d54beb40a441d366adcfe740969820caf156c019fb5c7ec4", size = 229894, upload-time = "2024-11-27T22:38:02.094Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/55/18/5d8bc5b0a0362311ce4d18830a5d28943667599a60d20118074ea1b01bb7/tomli-2.2.1-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:e59e304978767a54663af13c07b3d1af22ddee3bb2fb0618ca1593e4f593a106", size = 245319, upload-time = "2024-11-27T22:38:03.206Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/92/a3/7ade0576d17f3cdf5ff44d61390d4b3febb8a9fc2b480c75c47ea048c646/tomli-2.2.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:33580bccab0338d00994d7f16f4c4ec25b776af3ffaac1ed74e0b3fc95e885a8", size = 238273, upload-time = "2024-11-27T22:38:04.217Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/72/6f/fa64ef058ac1446a1e51110c375339b3ec6be245af9d14c87c4a6412dd32/tomli-2.2.1-cp311-cp311-win32.whl", hash = "sha256:465af0e0875402f1d226519c9904f37254b3045fc5084697cefb9bdde1ff99ff", size = 98310, upload-time = "2024-11-27T22:38:05.908Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/6a/1c/4a2dcde4a51b81be3530565e92eda625d94dafb46dbeb15069df4caffc34/tomli-2.2.1-cp311-cp311-win_amd64.whl", hash = "sha256:2d0f2fdd22b02c6d81637a3c95f8cd77f995846af7414c5c4b8d0545afa1bc4b", size = 108309, upload-time = "2024-11-27T22:38:06.812Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/52/e1/f8af4c2fcde17500422858155aeb0d7e93477a0d59a98e56cbfe75070fd0/tomli-2.2.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:4a8f6e44de52d5e6c657c9fe83b562f5f4256d8ebbfe4ff922c495620a7f6cea", size = 132762, upload-time = "2024-11-27T22:38:07.731Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/03/b8/152c68bb84fc00396b83e7bbddd5ec0bd3dd409db4195e2a9b3e398ad2e3/tomli-2.2.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:8d57ca8095a641b8237d5b079147646153d22552f1c637fd3ba7f4b0b29167a8", size = 123453, upload-time = "2024-11-27T22:38:09.384Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c8/d6/fc9267af9166f79ac528ff7e8c55c8181ded34eb4b0e93daa767b8841573/tomli-2.2.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4e340144ad7ae1533cb897d406382b4b6fede8890a03738ff1683af800d54192", size = 233486, upload-time = "2024-11-27T22:38:10.329Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/5c/51/51c3f2884d7bab89af25f678447ea7d297b53b5a3b5730a7cb2ef6069f07/tomli-2.2.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:db2b95f9de79181805df90bedc5a5ab4c165e6ec3fe99f970d0e302f384ad222", size = 242349, upload-time = "2024-11-27T22:38:11.443Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/ab/df/bfa89627d13a5cc22402e441e8a931ef2108403db390ff3345c05253935e/tomli-2.2.1-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:40741994320b232529c802f8bc86da4e1aa9f413db394617b9a256ae0f9a7f77", size = 252159, upload-time = "2024-11-27T22:38:13.099Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/9e/6e/fa2b916dced65763a5168c6ccb91066f7639bdc88b48adda990db10c8c0b/tomli-2.2.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:400e720fe168c0f8521520190686ef8ef033fb19fc493da09779e592861b78c6", size = 237243, upload-time = "2024-11-27T22:38:14.766Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/b4/04/885d3b1f650e1153cbb93a6a9782c58a972b94ea4483ae4ac5cedd5e4a09/tomli-2.2.1-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:02abe224de6ae62c19f090f68da4e27b10af2b93213d36cf44e6e1c5abd19fdd", size = 259645, upload-time = "2024-11-27T22:38:15.843Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/9c/de/6b432d66e986e501586da298e28ebeefd3edc2c780f3ad73d22566034239/tomli-2.2.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:b82ebccc8c8a36f2094e969560a1b836758481f3dc360ce9a3277c65f374285e", size = 244584, upload-time = "2024-11-27T22:38:17.645Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/1c/9a/47c0449b98e6e7d1be6cbac02f93dd79003234ddc4aaab6ba07a9a7482e2/tomli-2.2.1-cp312-cp312-win32.whl", hash = "sha256:889f80ef92701b9dbb224e49ec87c645ce5df3fa2cc548664eb8a25e03127a98", size = 98875, upload-time = "2024-11-27T22:38:19.159Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/ef/60/9b9638f081c6f1261e2688bd487625cd1e660d0a85bd469e91d8db969734/tomli-2.2.1-cp312-cp312-win_amd64.whl", hash = "sha256:7fc04e92e1d624a4a63c76474610238576942d6b8950a2d7f908a340494e67e4", size = 109418, upload-time = "2024-11-27T22:38:20.064Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/04/90/2ee5f2e0362cb8a0b6499dc44f4d7d48f8fff06d28ba46e6f1eaa61a1388/tomli-2.2.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:f4039b9cbc3048b2416cc57ab3bda989a6fcf9b36cf8937f01a6e731b64f80d7", size = 132708, upload-time = "2024-11-27T22:38:21.659Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c0/ec/46b4108816de6b385141f082ba99e315501ccd0a2ea23db4a100dd3990ea/tomli-2.2.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:286f0ca2ffeeb5b9bd4fcc8d6c330534323ec51b2f52da063b11c502da16f30c", size = 123582, upload-time = "2024-11-27T22:38:22.693Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a0/bd/b470466d0137b37b68d24556c38a0cc819e8febe392d5b199dcd7f578365/tomli-2.2.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a92ef1a44547e894e2a17d24e7557a5e85a9e1d0048b0b5e7541f76c5032cb13", size = 232543, upload-time = "2024-11-27T22:38:24.367Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/d9/e5/82e80ff3b751373f7cead2815bcbe2d51c895b3c990686741a8e56ec42ab/tomli-2.2.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9316dc65bed1684c9a98ee68759ceaed29d229e985297003e494aa825ebb0281", size = 241691, upload-time = "2024-11-27T22:38:26.081Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/05/7e/2a110bc2713557d6a1bfb06af23dd01e7dde52b6ee7dadc589868f9abfac/tomli-2.2.1-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e85e99945e688e32d5a35c1ff38ed0b3f41f43fad8df0bdf79f72b2ba7bc5272", size = 251170, upload-time = "2024-11-27T22:38:27.921Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/64/7b/22d713946efe00e0adbcdfd6d1aa119ae03fd0b60ebed51ebb3fa9f5a2e5/tomli-2.2.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:ac065718db92ca818f8d6141b5f66369833d4a80a9d74435a268c52bdfa73140", size = 236530, upload-time = "2024-11-27T22:38:29.591Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/38/31/3a76f67da4b0cf37b742ca76beaf819dca0ebef26d78fc794a576e08accf/tomli-2.2.1-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:d920f33822747519673ee656a4b6ac33e382eca9d331c87770faa3eef562aeb2", size = 258666, upload-time = "2024-11-27T22:38:30.639Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/07/10/5af1293da642aded87e8a988753945d0cf7e00a9452d3911dd3bb354c9e2/tomli-2.2.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:a198f10c4d1b1375d7687bc25294306e551bf1abfa4eace6650070a5c1ae2744", size = 243954, upload-time = "2024-11-27T22:38:31.702Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/5b/b9/1ed31d167be802da0fc95020d04cd27b7d7065cc6fbefdd2f9186f60d7bd/tomli-2.2.1-cp313-cp313-win32.whl", hash = "sha256:d3f5614314d758649ab2ab3a62d4f2004c825922f9e370b29416484086b264ec", size = 98724, upload-time = "2024-11-27T22:38:32.837Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/c7/32/b0963458706accd9afcfeb867c0f9175a741bf7b19cd424230714d722198/tomli-2.2.1-cp313-cp313-win_amd64.whl", hash = "sha256:a38aa0308e754b0e3c67e344754dff64999ff9b513e691d0e786265c93583c69", size = 109383, upload-time = "2024-11-27T22:38:34.455Z" }, | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/6e/c2/61d3e0f47e2b74ef40a68b9e6ad5984f6241a942f7cd3bbfbdbd03861ea9/tomli-2.2.1-py3-none-any.whl", hash = "sha256:cb55c73c5f4408779d0cf3eef9f762b9c9f147a77de7b258bef0a5628adc85cc", size = 14257, upload-time = "2024-11-27T22:38:35.385Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "typing-extensions" | ||||||
|  | version = "4.14.0" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/d1/bc/51647cd02527e87d05cb083ccc402f93e441606ff1f01739a62c8ad09ba5/typing_extensions-4.14.0.tar.gz", hash = "sha256:8676b788e32f02ab42d9e7c61324048ae4c6d844a399eebace3d4979d75ceef4", size = 107423, upload-time = "2025-06-02T14:52:11.399Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/69/e0/552843e0d356fbb5256d21449fa957fa4eff3bbc135a74a691ee70c7c5da/typing_extensions-4.14.0-py3-none-any.whl", hash = "sha256:a1514509136dd0b477638fc68d6a91497af5076466ad0fa6c338e44e359944af", size = 43839, upload-time = "2025-06-02T14:52:10.026Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "tzdata" | ||||||
|  | version = "2025.2" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/95/32/1a225d6164441be760d75c2c42e2780dc0873fe382da3e98a2e1e48361e5/tzdata-2025.2.tar.gz", hash = "sha256:b60a638fcc0daffadf82fe0f57e53d06bdec2f36c4df66280ae79bce6bd6f2b9", size = 196380, upload-time = "2025-03-23T13:54:43.652Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/5c/23/c7abc0ca0a1526a0774eca151daeb8de62ec457e77262b66b359c3c7679e/tzdata-2025.2-py2.py3-none-any.whl", hash = "sha256:1a403fada01ff9221ca8044d701868fa132215d84beb92242d9acd2147f667a8", size = 347839, upload-time = "2025-03-23T13:54:41.845Z" }, | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [[package]] | ||||||
|  | name = "urllib3" | ||||||
|  | version = "2.5.0" | ||||||
|  | source = { registry = "https://pypi.org/simple" } | ||||||
|  | sdist = { url = "https://files.pythonhosted.org/packages/15/22/9ee70a2574a4f4599c47dd506532914ce044817c7752a79b6a51286319bc/urllib3-2.5.0.tar.gz", hash = "sha256:3fc47733c7e419d4bc3f6b3dc2b4f890bb743906a30d56ba4a5bfa4bbff92760", size = 393185, upload-time = "2025-06-18T14:07:41.644Z" } | ||||||
|  | wheels = [ | ||||||
|  |     { url = "https://files.pythonhosted.org/packages/a7/c2/fe1e52489ae3122415c51f387e221dd0773709bad6c6cdaa599e8a2c5185/urllib3-2.5.0-py3-none-any.whl", hash = "sha256:e6b01673c0fa6a13e374b50871808eb3bf7046c4b125b216f6bf1cc604cff0dc", size = 129795, upload-time = "2025-06-18T14:07:40.39Z" }, | ||||||
|  | ] | ||||||
		Reference in New Issue
	
	Block a user